From 2b2539727d7c611a6016ccc28762eec83777b913 Mon Sep 17 00:00:00 2001 From: Liangliang Nan Date: Wed, 8 Jan 2025 13:30:37 +0100 Subject: [PATCH] Squashed commit of the following: commit 96cdc73608397b0d413f88e31762bb8febd25ef9 Author: Liangliang Nan Date: Wed Jan 8 13:19:35 2025 +0100 Update cmake-build.yml commit bb6237ad6e042bbe5fdc9778b728133da6925a62 Author: Liangliang Nan Date: Wed Jan 8 13:01:41 2025 +0100 speedup github actions workflow commit 977c001bc017ec25db77decf59ae91bb0287cd63 Author: Liangliang Nan Date: Wed Jan 8 11:36:33 2025 +0100 remove glpk and lpsolve commit 3de6f5e8d76b3950e7abb4871b357ddcdcc1ec01 Author: Liangliang Nan Date: Tue Jan 7 23:30:28 2025 +0100 test commit 4a2fc5c838a0ed76ce70ab29d90c6e812e7f1405 Author: Liangliang Nan Date: Tue Jan 7 21:56:51 2025 +0100 update github actions workflow commit 0721b15ec0bf98ec82cd6d1819dd0367fa6cd03f Author: Liangliang Nan Date: Tue Jan 7 21:32:41 2025 +0100 Update test-build-gcc.yml commit 40d70d7587c7acafe74171ae01b4c707d40a15b6 Author: Liangliang Nan Date: Tue Jan 7 21:23:54 2025 +0100 Update test-build-gcc.yml commit 759e6e2009310272a8d59e5b9ab9e29dfe1510d5 Author: Liangliang Nan Date: Tue Jan 7 21:21:25 2025 +0100 Update test-build-gcc.yml commit dd5801ed61b36e47079ff8b34c1ec4e84fae6ce5 Author: Liangliang Nan Date: Tue Jan 7 21:19:10 2025 +0100 Update test-build-gcc.yml commit 5aa0b1e19e5862147ba2fbc03f1ac27c9115cd2d Author: Liangliang Nan Date: Tue Jan 7 21:09:01 2025 +0100 Update test-build-gcc.yml commit 3fa84a0efcded9deb975ef97cfe2f015b45d9aef Author: Liangliang Nan Date: Tue Jan 7 21:06:24 2025 +0100 Update test-build-gcc.yml commit 12ae530007db770f71e4ba728c6544f90020cf75 Author: Liangliang Nan Date: Tue Jan 7 20:11:56 2025 +0100 Update test-build-gcc.yml commit 37d5de013c6e3d669749aa54aab9f26a02b5439b Author: Liangliang Nan Date: Tue Jan 7 19:41:16 2025 +0100 Update test-build-gcc.yml commit 94090ba5325d5eb10c567414929fd47da39d3d5c Author: Liangliang Nan Date: Tue Jan 7 19:25:38 2025 +0100 Update test-build-gcc.yml commit 546b121abb570a2365facf3b067325b6f026fef0 Author: Liangliang Nan Date: Tue Jan 7 18:27:01 2025 +0100 github actions workflow for three platforms --- .github/workflows/cmake-build.yml | 84 + .github/workflows/test-build-gcc.yml | 43 - ReadMe.md | 6 +- code/3rd_QGLViewer/QGLViewer/config.h | 3 + code/3rd_glpk/CMakeLists.txt | 234 - code/3rd_glpk/COPYING | 674 - code/3rd_glpk/NEWS | 2008 --- code/3rd_glpk/README | 39 - code/3rd_glpk/amd/COPYING | 502 - code/3rd_glpk/amd/README | 58 - code/3rd_glpk/amd/amd.h | 67 - code/3rd_glpk/amd/amd_1.c | 181 - code/3rd_glpk/amd/amd_2.c | 1842 --- code/3rd_glpk/amd/amd_aat.c | 185 - code/3rd_glpk/amd/amd_control.c | 64 - code/3rd_glpk/amd/amd_defaults.c | 38 - code/3rd_glpk/amd/amd_dump.c | 180 - code/3rd_glpk/amd/amd_info.c | 120 - code/3rd_glpk/amd/amd_internal.h | 117 - code/3rd_glpk/amd/amd_order.c | 200 - code/3rd_glpk/amd/amd_post_tree.c | 121 - code/3rd_glpk/amd/amd_postorder.c | 207 - code/3rd_glpk/amd/amd_preprocess.c | 119 - code/3rd_glpk/amd/amd_valid.c | 93 - code/3rd_glpk/api/advbas.c | 155 - code/3rd_glpk/api/asnhall.c | 163 - code/3rd_glpk/api/asnlp.c | 104 - code/3rd_glpk/api/asnokalg.c | 154 - code/3rd_glpk/api/ckasn.c | 78 - code/3rd_glpk/api/ckcnf.c | 82 - code/3rd_glpk/api/cplex.c | 1283 -- code/3rd_glpk/api/cpp.c | 185 - code/3rd_glpk/api/cpxbas.c | 269 - code/3rd_glpk/api/graph.c | 504 - code/3rd_glpk/api/gridgen.c | 769 -- code/3rd_glpk/api/intfeas1.c | 267 - code/3rd_glpk/api/maxffalg.c | 130 - code/3rd_glpk/api/maxflp.c | 114 - code/3rd_glpk/api/mcflp.c | 114 - code/3rd_glpk/api/mcfokalg.c | 221 - code/3rd_glpk/api/mcfrelax.c | 251 - code/3rd_glpk/api/minisat1.c | 161 - code/3rd_glpk/api/mpl.c | 269 - code/3rd_glpk/api/mps.c | 1452 --- code/3rd_glpk/api/netgen.c | 1020 -- code/3rd_glpk/api/npp.c | 143 - code/3rd_glpk/api/pript.c | 186 - code/3rd_glpk/api/prmip.c | 155 - code/3rd_glpk/api/prob.h | 286 - code/3rd_glpk/api/prob1.c | 1588 --- code/3rd_glpk/api/prob2.c | 491 - code/3rd_glpk/api/prob3.c | 166 - code/3rd_glpk/api/prob4.c | 156 - code/3rd_glpk/api/prob5.c | 168 - code/3rd_glpk/api/prrngs.c | 302 - code/3rd_glpk/api/prsol.c | 202 - code/3rd_glpk/api/rdasn.c | 164 - code/3rd_glpk/api/rdcc.c | 162 - code/3rd_glpk/api/rdcnf.c | 136 - code/3rd_glpk/api/rdipt.c | 185 - code/3rd_glpk/api/rdmaxf.c | 163 - code/3rd_glpk/api/rdmcf.c | 186 - code/3rd_glpk/api/rdmip.c | 172 - code/3rd_glpk/api/rdprob.c | 377 - code/3rd_glpk/api/rdsol.c | 225 - code/3rd_glpk/api/rmfgen.c | 368 - code/3rd_glpk/api/strong.c | 110 - code/3rd_glpk/api/topsort.c | 123 - code/3rd_glpk/api/wcliqex.c | 122 - code/3rd_glpk/api/weak.c | 150 - code/3rd_glpk/api/wrasn.c | 107 - code/3rd_glpk/api/wrcc.c | 102 - code/3rd_glpk/api/wrcnf.c | 87 - code/3rd_glpk/api/wript.c | 124 - code/3rd_glpk/api/wrmaxf.c | 104 - code/3rd_glpk/api/wrmcf.c | 122 - code/3rd_glpk/api/wrmip.c | 122 - code/3rd_glpk/api/wrprob.c | 166 - code/3rd_glpk/api/wrsol.c | 174 - code/3rd_glpk/bflib/btf.c | 569 - code/3rd_glpk/bflib/btf.h | 207 - code/3rd_glpk/bflib/btfint.c | 407 - code/3rd_glpk/bflib/btfint.h | 73 - code/3rd_glpk/bflib/fhv.c | 586 - code/3rd_glpk/bflib/fhv.h | 114 - code/3rd_glpk/bflib/fhvint.c | 168 - code/3rd_glpk/bflib/fhvint.h | 78 - code/3rd_glpk/bflib/ifu.c | 392 - code/3rd_glpk/bflib/ifu.h | 99 - code/3rd_glpk/bflib/luf.c | 713 -- code/3rd_glpk/bflib/luf.h | 227 - code/3rd_glpk/bflib/lufint.c | 182 - code/3rd_glpk/bflib/lufint.h | 73 - code/3rd_glpk/bflib/scf.c | 523 - code/3rd_glpk/bflib/scf.h | 211 - code/3rd_glpk/bflib/scfint.c | 255 - code/3rd_glpk/bflib/scfint.h | 89 - code/3rd_glpk/bflib/sgf.c | 1443 --- code/3rd_glpk/bflib/sgf.h | 203 - code/3rd_glpk/bflib/sva.c | 572 - code/3rd_glpk/bflib/sva.h | 161 - code/3rd_glpk/colamd/COPYING | 502 - code/3rd_glpk/colamd/README | 98 - code/3rd_glpk/colamd/colamd.c | 3622 ------ code/3rd_glpk/colamd/colamd.h | 69 - code/3rd_glpk/config.h | 19 - code/3rd_glpk/doc/glpk.pdf | Bin 493688 -> 0 bytes code/3rd_glpk/doc/graphs.pdf | Bin 210602 -> 0 bytes code/3rd_glpk/draft/bfd.c | 544 - code/3rd_glpk/draft/bfd.h | 107 - code/3rd_glpk/draft/bfx.c | 89 - code/3rd_glpk/draft/bfx.h | 67 - code/3rd_glpk/draft/draft.h | 22 - code/3rd_glpk/draft/glpapi06.c | 860 -- code/3rd_glpk/draft/glpapi07.c | 499 - code/3rd_glpk/draft/glpapi08.c | 388 - code/3rd_glpk/draft/glpapi09.c | 798 -- code/3rd_glpk/draft/glpapi10.c | 305 - code/3rd_glpk/draft/glpapi12.c | 2185 ---- code/3rd_glpk/draft/glpapi13.c | 710 -- code/3rd_glpk/draft/glphbm.c | 533 - code/3rd_glpk/draft/glphbm.h | 127 - code/3rd_glpk/draft/glpios01.c | 1685 --- code/3rd_glpk/draft/glpios02.c | 826 -- code/3rd_glpk/draft/glpios03.c | 1513 --- code/3rd_glpk/draft/glpios07.c | 551 - code/3rd_glpk/draft/glpios09.c | 664 - code/3rd_glpk/draft/glpios11.c | 435 - code/3rd_glpk/draft/glpios12.c | 177 - code/3rd_glpk/draft/glpipm.c | 1144 -- code/3rd_glpk/draft/glpipm.h | 36 - code/3rd_glpk/draft/glpmat.c | 924 -- code/3rd_glpk/draft/glpmat.h | 198 - code/3rd_glpk/draft/glprgr.c | 173 - code/3rd_glpk/draft/glprgr.h | 34 - code/3rd_glpk/draft/glpscl.c | 478 - code/3rd_glpk/draft/glpspm.c | 847 -- code/3rd_glpk/draft/glpspm.h | 165 - code/3rd_glpk/draft/glpssx.h | 437 - code/3rd_glpk/draft/glpssx01.c | 839 -- code/3rd_glpk/draft/glpssx02.c | 523 - code/3rd_glpk/draft/ios.h | 547 - code/3rd_glpk/draft/lux.c | 1030 -- code/3rd_glpk/draft/lux.h | 220 - code/3rd_glpk/env/alloc.c | 252 - code/3rd_glpk/env/dlsup.c | 167 - code/3rd_glpk/env/env.c | 316 - code/3rd_glpk/env/env.h | 274 - code/3rd_glpk/env/error.c | 200 - code/3rd_glpk/env/stdc.c | 98 - code/3rd_glpk/env/stdc.h | 73 - code/3rd_glpk/env/stdout.c | 262 - code/3rd_glpk/env/stream.c | 517 - code/3rd_glpk/env/time.c | 150 - code/3rd_glpk/env/tls.c | 128 - code/3rd_glpk/glpk.h | 1175 -- code/3rd_glpk/intopt/cfg.c | 409 - code/3rd_glpk/intopt/cfg.h | 138 - code/3rd_glpk/intopt/cfg1.c | 703 -- code/3rd_glpk/intopt/cfg2.c | 91 - code/3rd_glpk/intopt/clqcut.c | 134 - code/3rd_glpk/intopt/covgen.c | 885 -- code/3rd_glpk/intopt/fpump.c | 360 - code/3rd_glpk/intopt/gmicut.c | 284 - code/3rd_glpk/intopt/gmigen.c | 142 - code/3rd_glpk/intopt/mirgen.c | 1529 --- code/3rd_glpk/intopt/spv.c | 303 - code/3rd_glpk/intopt/spv.h | 83 - code/3rd_glpk/minisat/LICENSE | 20 - code/3rd_glpk/minisat/README | 22 - code/3rd_glpk/minisat/minisat.c | 1315 -- code/3rd_glpk/minisat/minisat.h | 230 - code/3rd_glpk/misc/avl.c | 405 - code/3rd_glpk/misc/avl.h | 73 - code/3rd_glpk/misc/bignum.c | 286 - code/3rd_glpk/misc/bignum.h | 37 - code/3rd_glpk/misc/dimacs.c | 147 - code/3rd_glpk/misc/dimacs.h | 81 - code/3rd_glpk/misc/dmp.c | 243 - code/3rd_glpk/misc/dmp.h | 63 - code/3rd_glpk/misc/ffalg.c | 221 - code/3rd_glpk/misc/ffalg.h | 34 - code/3rd_glpk/misc/fp2rat.c | 164 - code/3rd_glpk/misc/fvs.c | 137 - code/3rd_glpk/misc/fvs.h | 76 - code/3rd_glpk/misc/gcd.c | 102 - code/3rd_glpk/misc/jd.c | 152 - code/3rd_glpk/misc/jd.h | 32 - code/3rd_glpk/misc/keller.c | 235 - code/3rd_glpk/misc/keller.h | 34 - code/3rd_glpk/misc/ks.c | 466 - code/3rd_glpk/misc/ks.h | 44 - code/3rd_glpk/misc/mc13d.c | 314 - code/3rd_glpk/misc/mc13d.h | 34 - code/3rd_glpk/misc/mc21a.c | 301 - code/3rd_glpk/misc/mc21a.h | 34 - code/3rd_glpk/misc/misc.h | 61 - code/3rd_glpk/misc/mt1.c | 1110 -- code/3rd_glpk/misc/mt1.f | 277 - code/3rd_glpk/misc/mt1.h | 34 - code/3rd_glpk/misc/mygmp.c | 1162 -- code/3rd_glpk/misc/mygmp.h | 254 - code/3rd_glpk/misc/okalg.c | 382 - code/3rd_glpk/misc/okalg.h | 35 - code/3rd_glpk/misc/qmd.c | 584 - code/3rd_glpk/misc/qmd.h | 58 - code/3rd_glpk/misc/relax4.c | 2850 ----- code/3rd_glpk/misc/relax4.h | 102 - code/3rd_glpk/misc/rng.c | 227 - code/3rd_glpk/misc/rng.h | 67 - code/3rd_glpk/misc/rng1.c | 73 - code/3rd_glpk/misc/round2n.c | 64 - code/3rd_glpk/misc/str2int.c | 92 - code/3rd_glpk/misc/str2num.c | 110 - code/3rd_glpk/misc/strspx.c | 60 - code/3rd_glpk/misc/strtrim.c | 62 - code/3rd_glpk/misc/triang.c | 311 - code/3rd_glpk/misc/triang.h | 34 - code/3rd_glpk/misc/wclique.c | 242 - code/3rd_glpk/misc/wclique.h | 33 - code/3rd_glpk/misc/wclique1.c | 317 - code/3rd_glpk/misc/wclique1.h | 34 - code/3rd_glpk/mpl/mpl.h | 2598 ---- code/3rd_glpk/mpl/mpl1.c | 4718 ------- code/3rd_glpk/mpl/mpl2.c | 1202 -- code/3rd_glpk/mpl/mpl3.c | 6100 --------- code/3rd_glpk/mpl/mpl4.c | 1426 --- code/3rd_glpk/mpl/mpl5.c | 566 - code/3rd_glpk/mpl/mpl6.c | 1039 -- code/3rd_glpk/mpl/mplsql.c | 1659 --- code/3rd_glpk/mpl/mplsql.h | 63 - code/3rd_glpk/npp/npp.h | 645 - code/3rd_glpk/npp/npp1.c | 937 -- code/3rd_glpk/npp/npp2.c | 1433 --- code/3rd_glpk/npp/npp3.c | 2861 ----- code/3rd_glpk/npp/npp4.c | 1414 --- code/3rd_glpk/npp/npp5.c | 809 -- code/3rd_glpk/npp/npp6.c | 1500 --- code/3rd_glpk/proxy/main.c | 87 - code/3rd_glpk/proxy/proxy.c | 1073 -- code/3rd_glpk/proxy/proxy.h | 36 - code/3rd_glpk/proxy/proxy1.c | 88 - code/3rd_glpk/simplex/simplex.h | 39 - code/3rd_glpk/simplex/spxat.c | 265 - code/3rd_glpk/simplex/spxat.h | 80 - code/3rd_glpk/simplex/spxchuzc.c | 381 - code/3rd_glpk/simplex/spxchuzc.h | 85 - code/3rd_glpk/simplex/spxchuzr.c | 594 - code/3rd_glpk/simplex/spxchuzr.h | 77 - code/3rd_glpk/simplex/spxlp.c | 819 -- code/3rd_glpk/simplex/spxlp.h | 234 - code/3rd_glpk/simplex/spxnt.c | 303 - code/3rd_glpk/simplex/spxnt.h | 96 - code/3rd_glpk/simplex/spxprim.c | 1860 --- code/3rd_glpk/simplex/spxprob.c | 679 - code/3rd_glpk/simplex/spxprob.h | 64 - code/3rd_glpk/simplex/spychuzc.c | 567 - code/3rd_glpk/simplex/spychuzc.h | 85 - code/3rd_glpk/simplex/spychuzr.c | 483 - code/3rd_glpk/simplex/spychuzr.h | 97 - code/3rd_glpk/simplex/spydual.c | 2101 ---- code/3rd_glpk/zlib/README | 45 - code/3rd_glpk/zlib/adler32.c | 169 - code/3rd_glpk/zlib/compress.c | 80 - code/3rd_glpk/zlib/crc32.c | 442 - code/3rd_glpk/zlib/crc32.h | 441 - code/3rd_glpk/zlib/deflate.c | 1834 --- code/3rd_glpk/zlib/deflate.h | 342 - code/3rd_glpk/zlib/gzclose.c | 25 - code/3rd_glpk/zlib/gzguts.h | 74 - code/3rd_glpk/zlib/gzlib.c | 537 - code/3rd_glpk/zlib/gzread.c | 653 - code/3rd_glpk/zlib/gzwrite.c | 531 - code/3rd_glpk/zlib/inffast.c | 340 - code/3rd_glpk/zlib/inffast.h | 11 - code/3rd_glpk/zlib/inffixed.h | 94 - code/3rd_glpk/zlib/inflate.c | 1480 --- code/3rd_glpk/zlib/inflate.h | 122 - code/3rd_glpk/zlib/inftrees.c | 330 - code/3rd_glpk/zlib/inftrees.h | 62 - code/3rd_glpk/zlib/trees.c | 1244 -- code/3rd_glpk/zlib/trees.h | 128 - code/3rd_glpk/zlib/uncompr.c | 59 - code/3rd_glpk/zlib/zconf.h | 168 - code/3rd_glpk/zlib/zio.c | 92 - code/3rd_glpk/zlib/zio.h | 37 - code/3rd_glpk/zlib/zlib.h | 1613 --- code/3rd_glpk/zlib/zutil.c | 318 - code/3rd_glpk/zlib/zutil.h | 93 - code/3rd_lpsolve/CMakeLists.txt | 100 - code/3rd_lpsolve/README.txt | 527 - .../bfp/bfp_LUSOL/LUSOL/LUSOL_LGPL.txt | 504 - code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.c | 808 -- code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.h | 357 - code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol1.c | 3725 ------ code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol2.c | 204 - .../3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6a.c | 867 -- .../bfp/bfp_LUSOL/LUSOL/lusol6l0.c | 163 - .../3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6u.c | 176 - .../3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol7a.c | 704 -- .../3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol8a.c | 279 - code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.c | 27 - code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.h | 15 - code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.c | 735 -- code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.h | 63 - code/3rd_lpsolve/bfp/lp_BFP.h | 82 - code/3rd_lpsolve/bfp/lp_BFP1.c | 206 - code/3rd_lpsolve/bfp/lp_BFP2.c | 177 - code/3rd_lpsolve/colamd/colamd.c | 3469 ------ code/3rd_lpsolve/colamd/colamd.h | 286 - code/3rd_lpsolve/fortify.c | 2352 ---- code/3rd_lpsolve/fortify.h | 293 - code/3rd_lpsolve/ini.c | 76 - code/3rd_lpsolve/ini.h | 17 - code/3rd_lpsolve/lp_Hash.c | 238 - code/3rd_lpsolve/lp_Hash.h | 43 - code/3rd_lpsolve/lp_MDO.c | 241 - code/3rd_lpsolve/lp_MDO.h | 18 - code/3rd_lpsolve/lp_MPS.c | 1854 --- code/3rd_lpsolve/lp_MPS.h | 33 - code/3rd_lpsolve/lp_SOS.c | 1575 --- code/3rd_lpsolve/lp_SOS.h | 108 - code/3rd_lpsolve/lp_bit.h | 22 - code/3rd_lpsolve/lp_crash.c | 863 -- code/3rd_lpsolve/lp_crash.h | 27 - code/3rd_lpsolve/lp_explicit.h | 1062 -- code/3rd_lpsolve/lp_lib.c | 10323 ---------------- code/3rd_lpsolve/lp_lib.h | 2303 ---- code/3rd_lpsolve/lp_matrix.c | 3820 ------ code/3rd_lpsolve/lp_matrix.h | 257 - code/3rd_lpsolve/lp_mipbb.c | 1441 --- code/3rd_lpsolve/lp_mipbb.h | 64 - code/3rd_lpsolve/lp_params.c | 692 -- code/3rd_lpsolve/lp_presolve.c | 5897 --------- code/3rd_lpsolve/lp_presolve.h | 126 - code/3rd_lpsolve/lp_price.c | 2107 ---- code/3rd_lpsolve/lp_price.h | 99 - code/3rd_lpsolve/lp_pricePSE.c | 536 - code/3rd_lpsolve/lp_pricePSE.h | 28 - code/3rd_lpsolve/lp_report.c | 790 -- code/3rd_lpsolve/lp_report.h | 42 - code/3rd_lpsolve/lp_rlp.c | 2484 ---- code/3rd_lpsolve/lp_rlp.h | 2453 ---- code/3rd_lpsolve/lp_scale.c | 1075 -- code/3rd_lpsolve/lp_scale.h | 31 - code/3rd_lpsolve/lp_simplex.c | 2206 ---- code/3rd_lpsolve/lp_simplex.h | 34 - code/3rd_lpsolve/lp_types.h | 330 - code/3rd_lpsolve/lp_utils.c | 1058 -- code/3rd_lpsolve/lp_utils.h | 146 - code/3rd_lpsolve/lp_wlp.c | 362 - code/3rd_lpsolve/lp_wlp.h | 21 - code/3rd_lpsolve/lpkit.h | 32 - code/3rd_lpsolve/shared/commonlib.c | 992 -- code/3rd_lpsolve/shared/commonlib.h | 334 - code/3rd_lpsolve/shared/mmio.c | 496 - code/3rd_lpsolve/shared/mmio.h | 133 - code/3rd_lpsolve/shared/myblas.c | 841 -- code/3rd_lpsolve/shared/myblas.h | 132 - code/3rd_lpsolve/yacc_read.c | 1300 -- code/3rd_lpsolve/yacc_read.h | 57 - code/CMakeLists.txt | 4 - code/Example/main.cpp | 6 + code/PolyFit/main_window.cpp | 15 +- code/math/CMakeLists.txt | 13 +- code/math/linear_program_solver.cpp | 8 +- code/math/linear_program_solver.h | 8 +- code/renderer/opengl_info.cpp | 3 + code/renderer/surface_render.cpp | 3 + 369 files changed, 128 insertions(+), 193266 deletions(-) create mode 100644 .github/workflows/cmake-build.yml delete mode 100644 .github/workflows/test-build-gcc.yml delete mode 100644 code/3rd_glpk/CMakeLists.txt delete mode 100644 code/3rd_glpk/COPYING delete mode 100644 code/3rd_glpk/NEWS delete mode 100644 code/3rd_glpk/README delete mode 100644 code/3rd_glpk/amd/COPYING delete mode 100644 code/3rd_glpk/amd/README delete mode 100644 code/3rd_glpk/amd/amd.h delete mode 100644 code/3rd_glpk/amd/amd_1.c delete mode 100644 code/3rd_glpk/amd/amd_2.c delete mode 100644 code/3rd_glpk/amd/amd_aat.c delete mode 100644 code/3rd_glpk/amd/amd_control.c delete mode 100644 code/3rd_glpk/amd/amd_defaults.c delete mode 100644 code/3rd_glpk/amd/amd_dump.c delete mode 100644 code/3rd_glpk/amd/amd_info.c delete mode 100644 code/3rd_glpk/amd/amd_internal.h delete mode 100644 code/3rd_glpk/amd/amd_order.c delete mode 100644 code/3rd_glpk/amd/amd_post_tree.c delete mode 100644 code/3rd_glpk/amd/amd_postorder.c delete mode 100644 code/3rd_glpk/amd/amd_preprocess.c delete mode 100644 code/3rd_glpk/amd/amd_valid.c delete mode 100644 code/3rd_glpk/api/advbas.c delete mode 100644 code/3rd_glpk/api/asnhall.c delete mode 100644 code/3rd_glpk/api/asnlp.c delete mode 100644 code/3rd_glpk/api/asnokalg.c delete mode 100644 code/3rd_glpk/api/ckasn.c delete mode 100644 code/3rd_glpk/api/ckcnf.c delete mode 100644 code/3rd_glpk/api/cplex.c delete mode 100644 code/3rd_glpk/api/cpp.c delete mode 100644 code/3rd_glpk/api/cpxbas.c delete mode 100644 code/3rd_glpk/api/graph.c delete mode 100644 code/3rd_glpk/api/gridgen.c delete mode 100644 code/3rd_glpk/api/intfeas1.c delete mode 100644 code/3rd_glpk/api/maxffalg.c delete mode 100644 code/3rd_glpk/api/maxflp.c delete mode 100644 code/3rd_glpk/api/mcflp.c delete mode 100644 code/3rd_glpk/api/mcfokalg.c delete mode 100644 code/3rd_glpk/api/mcfrelax.c delete mode 100644 code/3rd_glpk/api/minisat1.c delete mode 100644 code/3rd_glpk/api/mpl.c delete mode 100644 code/3rd_glpk/api/mps.c delete mode 100644 code/3rd_glpk/api/netgen.c delete mode 100644 code/3rd_glpk/api/npp.c delete mode 100644 code/3rd_glpk/api/pript.c delete mode 100644 code/3rd_glpk/api/prmip.c delete mode 100644 code/3rd_glpk/api/prob.h delete mode 100644 code/3rd_glpk/api/prob1.c delete mode 100644 code/3rd_glpk/api/prob2.c delete mode 100644 code/3rd_glpk/api/prob3.c delete mode 100644 code/3rd_glpk/api/prob4.c delete mode 100644 code/3rd_glpk/api/prob5.c delete mode 100644 code/3rd_glpk/api/prrngs.c delete mode 100644 code/3rd_glpk/api/prsol.c delete mode 100644 code/3rd_glpk/api/rdasn.c delete mode 100644 code/3rd_glpk/api/rdcc.c delete mode 100644 code/3rd_glpk/api/rdcnf.c delete mode 100644 code/3rd_glpk/api/rdipt.c delete mode 100644 code/3rd_glpk/api/rdmaxf.c delete mode 100644 code/3rd_glpk/api/rdmcf.c delete mode 100644 code/3rd_glpk/api/rdmip.c delete mode 100644 code/3rd_glpk/api/rdprob.c delete mode 100644 code/3rd_glpk/api/rdsol.c delete mode 100644 code/3rd_glpk/api/rmfgen.c delete mode 100644 code/3rd_glpk/api/strong.c delete mode 100644 code/3rd_glpk/api/topsort.c delete mode 100644 code/3rd_glpk/api/wcliqex.c delete mode 100644 code/3rd_glpk/api/weak.c delete mode 100644 code/3rd_glpk/api/wrasn.c delete mode 100644 code/3rd_glpk/api/wrcc.c delete mode 100644 code/3rd_glpk/api/wrcnf.c delete mode 100644 code/3rd_glpk/api/wript.c delete mode 100644 code/3rd_glpk/api/wrmaxf.c delete mode 100644 code/3rd_glpk/api/wrmcf.c delete mode 100644 code/3rd_glpk/api/wrmip.c delete mode 100644 code/3rd_glpk/api/wrprob.c delete mode 100644 code/3rd_glpk/api/wrsol.c delete mode 100644 code/3rd_glpk/bflib/btf.c delete mode 100644 code/3rd_glpk/bflib/btf.h delete mode 100644 code/3rd_glpk/bflib/btfint.c delete mode 100644 code/3rd_glpk/bflib/btfint.h delete mode 100644 code/3rd_glpk/bflib/fhv.c delete mode 100644 code/3rd_glpk/bflib/fhv.h delete mode 100644 code/3rd_glpk/bflib/fhvint.c delete mode 100644 code/3rd_glpk/bflib/fhvint.h delete mode 100644 code/3rd_glpk/bflib/ifu.c delete mode 100644 code/3rd_glpk/bflib/ifu.h delete mode 100644 code/3rd_glpk/bflib/luf.c delete mode 100644 code/3rd_glpk/bflib/luf.h delete mode 100644 code/3rd_glpk/bflib/lufint.c delete mode 100644 code/3rd_glpk/bflib/lufint.h delete mode 100644 code/3rd_glpk/bflib/scf.c delete mode 100644 code/3rd_glpk/bflib/scf.h delete mode 100644 code/3rd_glpk/bflib/scfint.c delete mode 100644 code/3rd_glpk/bflib/scfint.h delete mode 100644 code/3rd_glpk/bflib/sgf.c delete mode 100644 code/3rd_glpk/bflib/sgf.h delete mode 100644 code/3rd_glpk/bflib/sva.c delete mode 100644 code/3rd_glpk/bflib/sva.h delete mode 100644 code/3rd_glpk/colamd/COPYING delete mode 100644 code/3rd_glpk/colamd/README delete mode 100644 code/3rd_glpk/colamd/colamd.c delete mode 100644 code/3rd_glpk/colamd/colamd.h delete mode 100644 code/3rd_glpk/config.h delete mode 100644 code/3rd_glpk/doc/glpk.pdf delete mode 100644 code/3rd_glpk/doc/graphs.pdf delete mode 100644 code/3rd_glpk/draft/bfd.c delete mode 100644 code/3rd_glpk/draft/bfd.h delete mode 100644 code/3rd_glpk/draft/bfx.c delete mode 100644 code/3rd_glpk/draft/bfx.h delete mode 100644 code/3rd_glpk/draft/draft.h delete mode 100644 code/3rd_glpk/draft/glpapi06.c delete mode 100644 code/3rd_glpk/draft/glpapi07.c delete mode 100644 code/3rd_glpk/draft/glpapi08.c delete mode 100644 code/3rd_glpk/draft/glpapi09.c delete mode 100644 code/3rd_glpk/draft/glpapi10.c delete mode 100644 code/3rd_glpk/draft/glpapi12.c delete mode 100644 code/3rd_glpk/draft/glpapi13.c delete mode 100644 code/3rd_glpk/draft/glphbm.c delete mode 100644 code/3rd_glpk/draft/glphbm.h delete mode 100644 code/3rd_glpk/draft/glpios01.c delete mode 100644 code/3rd_glpk/draft/glpios02.c delete mode 100644 code/3rd_glpk/draft/glpios03.c delete mode 100644 code/3rd_glpk/draft/glpios07.c delete mode 100644 code/3rd_glpk/draft/glpios09.c delete mode 100644 code/3rd_glpk/draft/glpios11.c delete mode 100644 code/3rd_glpk/draft/glpios12.c delete mode 100644 code/3rd_glpk/draft/glpipm.c delete mode 100644 code/3rd_glpk/draft/glpipm.h delete mode 100644 code/3rd_glpk/draft/glpmat.c delete mode 100644 code/3rd_glpk/draft/glpmat.h delete mode 100644 code/3rd_glpk/draft/glprgr.c delete mode 100644 code/3rd_glpk/draft/glprgr.h delete mode 100644 code/3rd_glpk/draft/glpscl.c delete mode 100644 code/3rd_glpk/draft/glpspm.c delete mode 100644 code/3rd_glpk/draft/glpspm.h delete mode 100644 code/3rd_glpk/draft/glpssx.h delete mode 100644 code/3rd_glpk/draft/glpssx01.c delete mode 100644 code/3rd_glpk/draft/glpssx02.c delete mode 100644 code/3rd_glpk/draft/ios.h delete mode 100644 code/3rd_glpk/draft/lux.c delete mode 100644 code/3rd_glpk/draft/lux.h delete mode 100644 code/3rd_glpk/env/alloc.c delete mode 100644 code/3rd_glpk/env/dlsup.c delete mode 100644 code/3rd_glpk/env/env.c delete mode 100644 code/3rd_glpk/env/env.h delete mode 100644 code/3rd_glpk/env/error.c delete mode 100644 code/3rd_glpk/env/stdc.c delete mode 100644 code/3rd_glpk/env/stdc.h delete mode 100644 code/3rd_glpk/env/stdout.c delete mode 100644 code/3rd_glpk/env/stream.c delete mode 100644 code/3rd_glpk/env/time.c delete mode 100644 code/3rd_glpk/env/tls.c delete mode 100644 code/3rd_glpk/glpk.h delete mode 100644 code/3rd_glpk/intopt/cfg.c delete mode 100644 code/3rd_glpk/intopt/cfg.h delete mode 100644 code/3rd_glpk/intopt/cfg1.c delete mode 100644 code/3rd_glpk/intopt/cfg2.c delete mode 100644 code/3rd_glpk/intopt/clqcut.c delete mode 100644 code/3rd_glpk/intopt/covgen.c delete mode 100644 code/3rd_glpk/intopt/fpump.c delete mode 100644 code/3rd_glpk/intopt/gmicut.c delete mode 100644 code/3rd_glpk/intopt/gmigen.c delete mode 100644 code/3rd_glpk/intopt/mirgen.c delete mode 100644 code/3rd_glpk/intopt/spv.c delete mode 100644 code/3rd_glpk/intopt/spv.h delete mode 100644 code/3rd_glpk/minisat/LICENSE delete mode 100644 code/3rd_glpk/minisat/README delete mode 100644 code/3rd_glpk/minisat/minisat.c delete mode 100644 code/3rd_glpk/minisat/minisat.h delete mode 100644 code/3rd_glpk/misc/avl.c delete mode 100644 code/3rd_glpk/misc/avl.h delete mode 100644 code/3rd_glpk/misc/bignum.c delete mode 100644 code/3rd_glpk/misc/bignum.h delete mode 100644 code/3rd_glpk/misc/dimacs.c delete mode 100644 code/3rd_glpk/misc/dimacs.h delete mode 100644 code/3rd_glpk/misc/dmp.c delete mode 100644 code/3rd_glpk/misc/dmp.h delete mode 100644 code/3rd_glpk/misc/ffalg.c delete mode 100644 code/3rd_glpk/misc/ffalg.h delete mode 100644 code/3rd_glpk/misc/fp2rat.c delete mode 100644 code/3rd_glpk/misc/fvs.c delete mode 100644 code/3rd_glpk/misc/fvs.h delete mode 100644 code/3rd_glpk/misc/gcd.c delete mode 100644 code/3rd_glpk/misc/jd.c delete mode 100644 code/3rd_glpk/misc/jd.h delete mode 100644 code/3rd_glpk/misc/keller.c delete mode 100644 code/3rd_glpk/misc/keller.h delete mode 100644 code/3rd_glpk/misc/ks.c delete mode 100644 code/3rd_glpk/misc/ks.h delete mode 100644 code/3rd_glpk/misc/mc13d.c delete mode 100644 code/3rd_glpk/misc/mc13d.h delete mode 100644 code/3rd_glpk/misc/mc21a.c delete mode 100644 code/3rd_glpk/misc/mc21a.h delete mode 100644 code/3rd_glpk/misc/misc.h delete mode 100644 code/3rd_glpk/misc/mt1.c delete mode 100644 code/3rd_glpk/misc/mt1.f delete mode 100644 code/3rd_glpk/misc/mt1.h delete mode 100644 code/3rd_glpk/misc/mygmp.c delete mode 100644 code/3rd_glpk/misc/mygmp.h delete mode 100644 code/3rd_glpk/misc/okalg.c delete mode 100644 code/3rd_glpk/misc/okalg.h delete mode 100644 code/3rd_glpk/misc/qmd.c delete mode 100644 code/3rd_glpk/misc/qmd.h delete mode 100644 code/3rd_glpk/misc/relax4.c delete mode 100644 code/3rd_glpk/misc/relax4.h delete mode 100644 code/3rd_glpk/misc/rng.c delete mode 100644 code/3rd_glpk/misc/rng.h delete mode 100644 code/3rd_glpk/misc/rng1.c delete mode 100644 code/3rd_glpk/misc/round2n.c delete mode 100644 code/3rd_glpk/misc/str2int.c delete mode 100644 code/3rd_glpk/misc/str2num.c delete mode 100644 code/3rd_glpk/misc/strspx.c delete mode 100644 code/3rd_glpk/misc/strtrim.c delete mode 100644 code/3rd_glpk/misc/triang.c delete mode 100644 code/3rd_glpk/misc/triang.h delete mode 100644 code/3rd_glpk/misc/wclique.c delete mode 100644 code/3rd_glpk/misc/wclique.h delete mode 100644 code/3rd_glpk/misc/wclique1.c delete mode 100644 code/3rd_glpk/misc/wclique1.h delete mode 100644 code/3rd_glpk/mpl/mpl.h delete mode 100644 code/3rd_glpk/mpl/mpl1.c delete mode 100644 code/3rd_glpk/mpl/mpl2.c delete mode 100644 code/3rd_glpk/mpl/mpl3.c delete mode 100644 code/3rd_glpk/mpl/mpl4.c delete mode 100644 code/3rd_glpk/mpl/mpl5.c delete mode 100644 code/3rd_glpk/mpl/mpl6.c delete mode 100644 code/3rd_glpk/mpl/mplsql.c delete mode 100644 code/3rd_glpk/mpl/mplsql.h delete mode 100644 code/3rd_glpk/npp/npp.h delete mode 100644 code/3rd_glpk/npp/npp1.c delete mode 100644 code/3rd_glpk/npp/npp2.c delete mode 100644 code/3rd_glpk/npp/npp3.c delete mode 100644 code/3rd_glpk/npp/npp4.c delete mode 100644 code/3rd_glpk/npp/npp5.c delete mode 100644 code/3rd_glpk/npp/npp6.c delete mode 100644 code/3rd_glpk/proxy/main.c delete mode 100644 code/3rd_glpk/proxy/proxy.c delete mode 100644 code/3rd_glpk/proxy/proxy.h delete mode 100644 code/3rd_glpk/proxy/proxy1.c delete mode 100644 code/3rd_glpk/simplex/simplex.h delete mode 100644 code/3rd_glpk/simplex/spxat.c delete mode 100644 code/3rd_glpk/simplex/spxat.h delete mode 100644 code/3rd_glpk/simplex/spxchuzc.c delete mode 100644 code/3rd_glpk/simplex/spxchuzc.h delete mode 100644 code/3rd_glpk/simplex/spxchuzr.c delete mode 100644 code/3rd_glpk/simplex/spxchuzr.h delete mode 100644 code/3rd_glpk/simplex/spxlp.c delete mode 100644 code/3rd_glpk/simplex/spxlp.h delete mode 100644 code/3rd_glpk/simplex/spxnt.c delete mode 100644 code/3rd_glpk/simplex/spxnt.h delete mode 100644 code/3rd_glpk/simplex/spxprim.c delete mode 100644 code/3rd_glpk/simplex/spxprob.c delete mode 100644 code/3rd_glpk/simplex/spxprob.h delete mode 100644 code/3rd_glpk/simplex/spychuzc.c delete mode 100644 code/3rd_glpk/simplex/spychuzc.h delete mode 100644 code/3rd_glpk/simplex/spychuzr.c delete mode 100644 code/3rd_glpk/simplex/spychuzr.h delete mode 100644 code/3rd_glpk/simplex/spydual.c delete mode 100644 code/3rd_glpk/zlib/README delete mode 100644 code/3rd_glpk/zlib/adler32.c delete mode 100644 code/3rd_glpk/zlib/compress.c delete mode 100644 code/3rd_glpk/zlib/crc32.c delete mode 100644 code/3rd_glpk/zlib/crc32.h delete mode 100644 code/3rd_glpk/zlib/deflate.c delete mode 100644 code/3rd_glpk/zlib/deflate.h delete mode 100644 code/3rd_glpk/zlib/gzclose.c delete mode 100644 code/3rd_glpk/zlib/gzguts.h delete mode 100644 code/3rd_glpk/zlib/gzlib.c delete mode 100644 code/3rd_glpk/zlib/gzread.c delete mode 100644 code/3rd_glpk/zlib/gzwrite.c delete mode 100644 code/3rd_glpk/zlib/inffast.c delete mode 100644 code/3rd_glpk/zlib/inffast.h delete mode 100644 code/3rd_glpk/zlib/inffixed.h delete mode 100644 code/3rd_glpk/zlib/inflate.c delete mode 100644 code/3rd_glpk/zlib/inflate.h delete mode 100644 code/3rd_glpk/zlib/inftrees.c delete mode 100644 code/3rd_glpk/zlib/inftrees.h delete mode 100644 code/3rd_glpk/zlib/trees.c delete mode 100644 code/3rd_glpk/zlib/trees.h delete mode 100644 code/3rd_glpk/zlib/uncompr.c delete mode 100644 code/3rd_glpk/zlib/zconf.h delete mode 100644 code/3rd_glpk/zlib/zio.c delete mode 100644 code/3rd_glpk/zlib/zio.h delete mode 100644 code/3rd_glpk/zlib/zlib.h delete mode 100644 code/3rd_glpk/zlib/zutil.c delete mode 100644 code/3rd_glpk/zlib/zutil.h delete mode 100644 code/3rd_lpsolve/CMakeLists.txt delete mode 100644 code/3rd_lpsolve/README.txt delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/LUSOL_LGPL.txt delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.h delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol1.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol2.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6a.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6l0.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6u.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol7a.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol8a.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.h delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.c delete mode 100644 code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.h delete mode 100644 code/3rd_lpsolve/bfp/lp_BFP.h delete mode 100644 code/3rd_lpsolve/bfp/lp_BFP1.c delete mode 100644 code/3rd_lpsolve/bfp/lp_BFP2.c delete mode 100644 code/3rd_lpsolve/colamd/colamd.c delete mode 100644 code/3rd_lpsolve/colamd/colamd.h delete mode 100644 code/3rd_lpsolve/fortify.c delete mode 100644 code/3rd_lpsolve/fortify.h delete mode 100644 code/3rd_lpsolve/ini.c delete mode 100644 code/3rd_lpsolve/ini.h delete mode 100644 code/3rd_lpsolve/lp_Hash.c delete mode 100644 code/3rd_lpsolve/lp_Hash.h delete mode 100644 code/3rd_lpsolve/lp_MDO.c delete mode 100644 code/3rd_lpsolve/lp_MDO.h delete mode 100644 code/3rd_lpsolve/lp_MPS.c delete mode 100644 code/3rd_lpsolve/lp_MPS.h delete mode 100644 code/3rd_lpsolve/lp_SOS.c delete mode 100644 code/3rd_lpsolve/lp_SOS.h delete mode 100644 code/3rd_lpsolve/lp_bit.h delete mode 100644 code/3rd_lpsolve/lp_crash.c delete mode 100644 code/3rd_lpsolve/lp_crash.h delete mode 100644 code/3rd_lpsolve/lp_explicit.h delete mode 100644 code/3rd_lpsolve/lp_lib.c delete mode 100644 code/3rd_lpsolve/lp_lib.h delete mode 100644 code/3rd_lpsolve/lp_matrix.c delete mode 100644 code/3rd_lpsolve/lp_matrix.h delete mode 100644 code/3rd_lpsolve/lp_mipbb.c delete mode 100644 code/3rd_lpsolve/lp_mipbb.h delete mode 100644 code/3rd_lpsolve/lp_params.c delete mode 100644 code/3rd_lpsolve/lp_presolve.c delete mode 100644 code/3rd_lpsolve/lp_presolve.h delete mode 100644 code/3rd_lpsolve/lp_price.c delete mode 100644 code/3rd_lpsolve/lp_price.h delete mode 100644 code/3rd_lpsolve/lp_pricePSE.c delete mode 100644 code/3rd_lpsolve/lp_pricePSE.h delete mode 100644 code/3rd_lpsolve/lp_report.c delete mode 100644 code/3rd_lpsolve/lp_report.h delete mode 100644 code/3rd_lpsolve/lp_rlp.c delete mode 100644 code/3rd_lpsolve/lp_rlp.h delete mode 100644 code/3rd_lpsolve/lp_scale.c delete mode 100644 code/3rd_lpsolve/lp_scale.h delete mode 100644 code/3rd_lpsolve/lp_simplex.c delete mode 100644 code/3rd_lpsolve/lp_simplex.h delete mode 100644 code/3rd_lpsolve/lp_types.h delete mode 100644 code/3rd_lpsolve/lp_utils.c delete mode 100644 code/3rd_lpsolve/lp_utils.h delete mode 100644 code/3rd_lpsolve/lp_wlp.c delete mode 100644 code/3rd_lpsolve/lp_wlp.h delete mode 100644 code/3rd_lpsolve/lpkit.h delete mode 100644 code/3rd_lpsolve/shared/commonlib.c delete mode 100644 code/3rd_lpsolve/shared/commonlib.h delete mode 100644 code/3rd_lpsolve/shared/mmio.c delete mode 100644 code/3rd_lpsolve/shared/mmio.h delete mode 100644 code/3rd_lpsolve/shared/myblas.c delete mode 100644 code/3rd_lpsolve/shared/myblas.h delete mode 100644 code/3rd_lpsolve/yacc_read.c delete mode 100644 code/3rd_lpsolve/yacc_read.h diff --git a/.github/workflows/cmake-build.yml b/.github/workflows/cmake-build.yml new file mode 100644 index 00000000..b8a53c54 --- /dev/null +++ b/.github/workflows/cmake-build.yml @@ -0,0 +1,84 @@ +############################################################################## +# GitHub Actions Workflow to test building PolyFit on Windows, Ubuntu, and macOS. +# +# Copyright (C) 2022 Liangliang Nan +# +# Licensed under GNU LGPL.3, see LICENCE file +############################################################################## +name: Test Build PolyFit + +on: [push, pull_request] + +jobs: + build: + name: "Build on ${{ matrix.platform }} - ${{ matrix.build_type }}" + strategy: + fail-fast: false + matrix: + platform: [windows-latest, ubuntu-latest, macos-latest] + build_type: [Debug, Release] + runs-on: ${{ matrix.platform }} + + steps: + # Checkout the code + - uses: actions/checkout@v3 + + # Install dependencies for each platform + - name: Install Dependencies (Ubuntu) + if: runner.os == 'Linux' + run: | + sudo apt-get update || true + sudo apt-get install -y cmake build-essential libgl1-mesa-dev \ + mesa-common-dev libxrandr-dev libxinerama-dev libxcursor-dev libxi-dev \ + libcgal-dev qt6-base-dev libglu1-mesa-dev freeglut3-dev + + - name: Install Dependencies (macOS) + if: runner.os == 'macOS' + run: | + brew update || true + brew install cmake cgal qt + + # Set up Conda on Windows + - name: Set up Conda (Windows) + if: runner.os == 'Windows' + uses: conda-incubator/setup-miniconda@v3 + with: + architecture: x64 + activate-environment: PolyFitDev + auto-activate-base: false + channels: conda-forge,defaults + + # Install Conda packages on Windows + - name: Install Dependencies via Conda (Windows) + if: runner.os == 'Windows' + shell: bash -l {0} + run: | + conda install -y \ + ninja \ + conda-forge::qt6-main \ + conda-forge::cgal || exit 1 + + # Configure the project + - name: Configure (Linux & macOS) + if: runner.os != 'Windows' + run: | + cmake -S . -B build/${{ matrix.build_type }} -G "Unix Makefiles" + + - name: Configure (Windows) + if: runner.os == 'Windows' + shell: bash -l {0} + run: | + cmake -S . -B build/${{ matrix.build_type }} \ + -G "Visual Studio 17 2022" \ + -A x64 \ + -DCMAKE_PREFIX_PATH="$CONDA_PREFIX/Library" + + # Build the project + - name: Build (Linux & macOS) + if: runner.os != 'Windows' + run: cmake --build build/${{ matrix.build_type }} + + - name: Build (Windows) + if: runner.os == 'Windows' + shell: bash -l {0} + run: cmake --build build/${{ matrix.build_type }} --config ${{ matrix.build_type }} \ No newline at end of file diff --git a/.github/workflows/test-build-gcc.yml b/.github/workflows/test-build-gcc.yml deleted file mode 100644 index 946ffec6..00000000 --- a/.github/workflows/test-build-gcc.yml +++ /dev/null @@ -1,43 +0,0 @@ -############################################################################## -# GitHub Actions Workflow to test building PolyFit with GCC -# -# Copyright (C) 2022 Liangliang Nan -# -# Licensed under GNU LGPL.3, see LICENCE file -############################################################################## -name: test-build-gcc - -on: [push, pull_request] - -jobs: - build: - name: ${{ matrix.compilers }} - strategy: - fail-fast: false - matrix: - compilers: [g++-7, g++-8, g++-9, g++-10, g++-11] - runs-on: ubuntu-20.04 - steps: - - uses: actions/checkout@v1 - - name: Install GCC - run: | - sudo apt-get update || true - if [ "${{ matrix.compilers }}" == "g++-11" ]; then - sudo add-apt-repository ppa:ubuntu-toolchain-r/test -y - sudo apt-get update - fi - sudo apt-get install ${{ matrix.compilers }} - - name: Install Dependencies - run: | - sudo apt-get install build-essential libglu1-mesa-dev mesa-common-dev libxrandr-dev libxinerama-dev libxcursor-dev libxi-dev libboost-all-dev libcgal-dev qt5-default - - name: Build Project - run: | - rm -rf Release - mkdir Release - cd Release - if [ "${{ matrix.compilers }}" == "g++-11" ]; then - cmake -DCMAKE_CXX_COMPILER=/usr/bin/g++-11 -DCMAKE_BUILD_TYPE=Release .. - else - cmake -DCMAKE_CXX_COMPILER=${{ matrix.compilers }} -DCMAKE_BUILD_TYPE=Release .. - fi - make \ No newline at end of file diff --git a/ReadMe.md b/ReadMe.md index 0073f195..12ef733c 100644 --- a/ReadMe.md +++ b/ReadMe.md @@ -96,14 +96,14 @@ Please note, PolyFit assumes that the model is closed and all necessary planes a --- ### About the solvers -Four solvers, namely Gurobi, SCIP, GLPK, and lp_solve, are provided (with source code) in PolyFit. +Two solvers, namely Gurobi and SCIP (with source code), are provided in PolyFit. The Gurobi solver is more efficient and reliable and should always be your first choice. To use Gurobi, you need to install it and also obtain a license (free for academic use) from [here](https://www.gurobi.com/downloads/end-user-license-agreement-academic/). You may also need to modify the path(s) to Gurobi in [FindGUROBI.cmake](./code/cmake/FindGUROBI.cmake), for CMake to find Gurobi. In case you want a fast but open-source solver, please try SCIP, which is slower than Gurobi but acceptable. -The GLPK and lp_solve solvers only manage to solve small problems. They are too slow (and thus may not guarantee to succeed). -For example the data "Fig1", Gurobi takes only 0.02 seconds, while lp_solve 15 minutes. +The GLPK and lp_solve solvers (only available in previous PolyFit distributions) can only manage to solve small problems. +They are too slow (and thus may not guarantee to succeed). For example the data "Fig1", Gurobi takes only 0.02 seconds, while lp_solve 15 minutes. **Note for Linux users:** You may have to build the Gurobi library (`libgurobi_c++.a`) because the prebuilt one in the original package might NOT be compatible with your compiler. To do so, go to `src/build` and run `make`. Then replace the original `libgurobi_c++.a` (in the `lib` directory) with your generated file. diff --git a/code/3rd_QGLViewer/QGLViewer/config.h b/code/3rd_QGLViewer/QGLViewer/config.h index 25f0e95e..0f238a85 100644 --- a/code/3rd_QGLViewer/QGLViewer/config.h +++ b/code/3rd_QGLViewer/QGLViewer/config.h @@ -53,6 +53,9 @@ Error : libQGLViewer requires a minimum Qt version of 5.4 Error #ifdef Q_OS_MAC # include #else +#ifdef WIN32 +#include +#endif # include #endif diff --git a/code/3rd_glpk/CMakeLists.txt b/code/3rd_glpk/CMakeLists.txt deleted file mode 100644 index 9ee8e630..00000000 --- a/code/3rd_glpk/CMakeLists.txt +++ /dev/null @@ -1,234 +0,0 @@ -get_filename_component(PROJECT_NAME ${CMAKE_CURRENT_SOURCE_DIR} NAME) -project(${PROJECT_NAME}) - - -set(glpk_HEADERS - glpk.h - ) - -set(glpk_SOURCES - amd/amd_1.c - amd/amd_2.c - amd/amd_aat.c - amd/amd_control.c - amd/amd_defaults.c - amd/amd_dump.c - amd/amd_info.c - amd/amd_order.c - amd/amd_post_tree.c - amd/amd_postorder.c - amd/amd_preprocess.c - amd/amd_valid.c - api/advbas.c - api/asnhall.c - api/asnlp.c - api/asnokalg.c - api/ckasn.c - api/ckcnf.c - api/cplex.c - api/cpp.c - api/cpxbas.c - api/graph.c - api/gridgen.c - api/intfeas1.c - api/maxffalg.c - api/maxflp.c - api/mcflp.c - api/mcfokalg.c - api/mcfrelax.c - api/minisat1.c - api/mpl.c - api/mps.c - api/netgen.c - api/npp.c - api/pript.c - api/prmip.c - api/prob1.c - api/prob2.c - api/prob3.c - api/prob4.c - api/prob5.c - api/prrngs.c - api/prsol.c - api/rdasn.c - api/rdcc.c - api/rdcnf.c - api/rdipt.c - api/rdmaxf.c - api/rdmcf.c - api/rdmip.c - api/rdprob.c - api/rdsol.c - api/rmfgen.c - api/strong.c - api/topsort.c - api/wcliqex.c - api/weak.c - api/wrasn.c - api/wrcc.c - api/wrcnf.c - api/wript.c - api/wrmaxf.c - api/wrmcf.c - api/wrmip.c - api/wrprob.c - api/wrsol.c - bflib/btf.c - bflib/btfint.c - bflib/fhv.c - bflib/fhvint.c - bflib/ifu.c - bflib/luf.c - bflib/lufint.c - bflib/scf.c - bflib/scfint.c - bflib/sgf.c - bflib/sva.c - colamd/colamd.c - draft/bfd.c - draft/bfx.c - draft/glpapi06.c - draft/glpapi07.c - draft/glpapi08.c - draft/glpapi09.c - draft/glpapi10.c - draft/glpapi12.c - draft/glpapi13.c - draft/glphbm.c - draft/glpios01.c - draft/glpios02.c - draft/glpios03.c - draft/glpios07.c - draft/glpios09.c - draft/glpios11.c - draft/glpios12.c - draft/glpipm.c - draft/glpmat.c - draft/glprgr.c - draft/glpscl.c - draft/glpspm.c - draft/glpssx01.c - draft/glpssx02.c - draft/lux.c - env/alloc.c - env/dlsup.c - env/env.c - env/error.c - env/stdc.c - env/stdout.c - env/stream.c - env/time.c - env/tls.c - intopt/cfg.c - intopt/cfg1.c - intopt/cfg2.c - intopt/clqcut.c - intopt/covgen.c - intopt/fpump.c - intopt/gmicut.c - intopt/gmigen.c - intopt/mirgen.c - intopt/spv.c - minisat/minisat.c - misc/avl.c - misc/bignum.c - misc/dimacs.c - misc/dmp.c - misc/ffalg.c - misc/fp2rat.c - misc/fvs.c - misc/gcd.c - misc/jd.c - misc/keller.c - misc/ks.c - misc/mc13d.c - misc/mc21a.c - misc/mt1.c - misc/mygmp.c - misc/okalg.c - misc/qmd.c - misc/relax4.c - misc/rng.c - misc/rng1.c - misc/round2n.c - misc/str2int.c - misc/str2num.c - misc/strspx.c - misc/strtrim.c - misc/triang.c - misc/wclique.c - misc/wclique1.c - mpl/mpl1.c - mpl/mpl2.c - mpl/mpl3.c - mpl/mpl4.c - mpl/mpl5.c - mpl/mpl6.c - mpl/mplsql.c - npp/npp1.c - npp/npp2.c - npp/npp3.c - npp/npp4.c - npp/npp5.c - npp/npp6.c - proxy/main.c - proxy/proxy.c - proxy/proxy1.c - simplex/spxat.c - simplex/spxchuzc.c - simplex/spxchuzr.c - simplex/spxlp.c - simplex/spxnt.c - simplex/spxprim.c - simplex/spxprob.c - simplex/spychuzc.c - simplex/spychuzr.c - simplex/spydual.c - zlib/adler32.c - zlib/compress.c - zlib/crc32.c - zlib/deflate.c - zlib/gzclose.c - zlib/gzlib.c - zlib/gzread.c - zlib/gzwrite.c - zlib/inffast.c - zlib/inflate.c - zlib/inftrees.c - zlib/trees.c - zlib/uncompr.c - zlib/zio.c - zlib/zutil.c - ) - - -add_library(${PROJECT_NAME} STATIC ${glpk_SOURCES} ${glpk_HEADERS}) -set_target_properties(${PROJECT_NAME} PROPERTIES - FOLDER "3rd_party") - -set_target_properties(${PROJECT_NAME} PROPERTIES POSITION_INDEPENDENT_CODE ON) - -target_include_directories(${PROJECT_NAME} PRIVATE - ${POLYFIT_glpk_DIR} - ${POLYFIT_glpk_DIR}/amd - ${POLYFIT_glpk_DIR}/api - ${POLYFIT_glpk_DIR}/draft - ${POLYFIT_glpk_DIR}/bflib - ${POLYFIT_glpk_DIR}/cglib - ${POLYFIT_glpk_DIR}/colamd - ${POLYFIT_glpk_DIR}/env - ${POLYFIT_glpk_DIR}/minisat - ${POLYFIT_glpk_DIR}/misc - ${POLYFIT_glpk_DIR}/mpl - ${POLYFIT_glpk_DIR}/npp - ${POLYFIT_glpk_DIR}/proxy - ${POLYFIT_glpk_DIR}/simplex - ${POLYFIT_glpk_DIR}/zlib - ) - -if (MSVC) - target_compile_definitions(${PROJECT_NAME} PRIVATE - _CRT_SECURE_NO_WARNINGS - _CRT_SECURE_NO_DEPRECATE - ) -endif() diff --git a/code/3rd_glpk/COPYING b/code/3rd_glpk/COPYING deleted file mode 100644 index 94a9ed02..00000000 --- a/code/3rd_glpk/COPYING +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see . - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - Copyright (C) - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -. diff --git a/code/3rd_glpk/NEWS b/code/3rd_glpk/NEWS deleted file mode 100644 index bc9bc789..00000000 --- a/code/3rd_glpk/NEWS +++ /dev/null @@ -1,2008 +0,0 @@ -GLPK 4.64 (release date: Dec 02, 2017) - - The dual simplex solver routine was changed to perform more - aggressive perturbation to prevent dual degeneracy and avoid - stalling even if the current dual basic solution is strongly - feasible (mainly if the objective is zero). Thanks to David - Monniaux for bug report - and example model. - - The exact simplex solver routine was changed to perform - terminal output according to the verbosity level (specified by - the control parameter smcp.msg_lev). Thanks to Jeroen Demeyer - for bug report. - - A minor bug (related to MS Windows version) was fixed. Thanks - to Heinrich Schuchardt for bug report. - - An example model (Graceful Tree Labeling Problem) in MathProg - was added. Thanks to Mike Appleby for - contribution. - - Three example models (Power plant LP scheduler, Neumann CA - grid emulator generator) in MathProg and one in Cplex LP format - were added. Thanks to Peter Naszvadi for - contribution. - -GLPK 4.63 (release date: Jul 25, 2017) - - A "smart" LP perturbation was implemented in the primal and - dual simplex solvers. Now LP is perturbed only if it is - necessary, and by default perturbation is not activated. - The sum of primal infeasibilites that appears in the terminal - output of the primal simplex solver (as "inf = ...") now - corresponds to the original bounds of variables. This allows to - see how much perturbed solution violates the original bounds. - - The long-step technique was implemented for phase I of the - primal simplex solver. This feature can be enabled by - specifying --flip option for glpsol or by specifying - glp_smcp.r_test = GLP_RT_FLIP on api level. For many LP - instances the long-step technique allows reducing the number - of simplex iterations to 30-70%. Please note that unlike the - dual simplex, where this technique can be used on both phase I - and II, for the primal simplex it can be used only on phase I, - where the sum of primal infeasibilities (which is a convex - piecewise linear function) is minimized. - - An internal objective scaling was included in both primal and - dual simplex solvers. For many LP/MIP instances this feature - improves numerical stability (for the dual solver) and prevents - cycling near the optimum (for the primal solver). - - The Posix version of glp_time (glpk/src/env/time.c) was changed - to resolve time_t issue on msys2. Thanks to Rob Schroeder - for bug report. - - Three new example models in MathProg were added: - life_goe.mod (Conway's Game of Life garden of eden checker); - tiling.mod (Rectifiable polyomino tilings generator); - toto.mod (Covering code generator). - Thanks to Peter Naszvadi for contribution. - -GLPK 4.62 (release date: Jun 14, 2017) - - The bound perturbation technique was included in the primal - simplex solver to improve numerical stability and avoid cycling. - Currently this feature is enabled by default. - - A range bug was fixed in the MPS reading routine. Thanks to - Chris Matrakidis for bug report and patch. - - Changes were made to provide 64-bit portability of the Minisat - solver. Thanks to Chris Matrakidis for - patch. - - Calls to non-thread-safe functions gmtime, strerror, and strtok - were replaced by calls to corresponding thread-safe equivalents - (gmtime_r, strerror_r, and strtok_r for GNU/Linux). - -GLPK 4.61 (release date: Jan 22, 2017) - - An option was added to build a re-entrant version of the - package suitable for running in a multi-threaded environment. - This option is enabled by default on configuring the package - if the compiler used supports the thread local storage class - specifier (e.g. _Thread_local or __thread). Thanks to - David Monniaux for suggestion and - Chris Matrakidis for configure.ac patch. - - A re-entrant version of the package allows running multiple - independent instances of glpk in different threads of a multi- - threaded application. Note that glpk is not thread-safe by - design, so if the application calls glpk routines from multiple - threads, no thread may access glpk program objects (glp_prob, - glp_tree, etc.) created by other threads; besides, each thread - before termination should call the API routine glp_free_env to - prevent memory leak. - - A DLL support was added. Thanks to Heinrich Schuchardt - for contribution. - - Some changes were made to allow compiling the package using - stdcall calling convention (this is needed only on compiling - GLPK with MSVC to run under MS Windows). - -GLPK 4.60 (release date: Apr 01, 2016) - - Some improvements were made in the primal and dual simplex - solvers to make the solution process more numerically stable. - - An experimental long-step ratio test feature was added to the - dual simplex. On API level this feature is available thru the - GLP_RT_FLIP option. For glpsol it is available thru the options - --flip (for MIP) or --flip and --dual (for LP). This feature is - not documented yet. - - Additional check was added to reject wrong solutions sometimes - reported by the PROXY heuristic. - - A bug (memory leak) was fixed in the FPUMP heuristic routine. - Thanks to for bug report. - - The header sql.h was renamed to avoid conflicts with standard - ODBC headers. Thanks to Noli Sicad for bug - report. - -GLPK 4.59 (release date: Mar 11, 2016) - - This is a maintainer release. - - Some bugs were fixed and some improvements were made in the MIP - solver. Thanks to Chris Matrakidis for bug - reports and patches. - - The data file format used by API routines glp_read_graph and - glp_write_graph was changed. For more details please see the - document "GLPK: Graph and Network Routines" included in the - distribution. - - Translation of the document "Modeling Language GNU MathProg" - to Brazilian Portuguese (pt-BR) was included (in LaTeX and pdf - formats). Thanks to Joao Flavio de Freitas Almeida - for contribution. - -GLPK 4.58 (release date: Feb 18, 2016) - - The solution file format used by API routines glp_read_sol, - glp_write_sol, glp_read_ipt, glp_write_ipt, glp_read_mip, and - glp_write_mip was changed. For more details please see the GLPK - reference manual included in the distribution. - - The tan function (trigonometric tangent) was added to - GNU MathProg modeling language. Thanks to Chris Matrakidis - for contribution. - - A new version of the document "Modeling Language GNU MathProg" - in Spanish was included (in LaTeX and pdf formats). Thanks to - Pablo Yapura for contribution. - - A check to determine if libtool needs '-no-undefined' flag to - build shared libraries on some platforms was added. - Thanks to Marco Atzeri and Heinrich - Schuchardt for suggestion. - - A script to regenerate the configure script and the Makefiles - was added. Thanks to Heinrich Schuchardt . - -GLPK 4.57 (release date: Nov 08, 2015) - - A new, more efficient implementation of the dual simplex method - was included in the package. This new implementation replaces - the old one, which was removed. - - Option sr_heur was added to struct glp_iocp to enable/disable - the simple rounding heuristic used by the MIP solver. Thanks to - Chris Matrakidis for suggestion. - - New API routine glp_at_error was added and documented. Thanks - to Jeroen Demeyer for suggestion. - - Some minor typos were corrected in the GLPK documentation. - Thanks to Anton Voropaev for typo - report. - - An example application program TSPSOL was added. It uses the - GLPK MIP optimizer to solve the Symmetric Traveling Salesman - Problem and illustrates "lazy" constraints generation. For more - details please see glpk/examples/tsp/README. - -GLPK 4.56 (release date: Oct 01, 2015) - - A new, more efficient and more robust implementation of the - primal simplex method was included in the package. This new - implementation replaces the old one, which was removed. - - A bug was fixed in a basis factorization routine. (The bug - appeared if the basis matrix was structurally singular having - duplicate row and/or column singletons.) Thanks to Martin Jacob - for bug report. - - Scripts to build GLPK with Microsoft Visual Studio 2015 were - added. Thanks to Xypron for contribution - and testing. - -GLPK 4.55 (release date: Aug 22, 2014) - - Some internal (non-API) routines to estimate the condition of - the basis matrix were added. These routines are mainly intended - to be used by the simplex-based solvers. - - Two open modes "a" and "ab" were added to GLPK I/O routines. - Thanks to Pedro P. Wong for bug report. - - Minor bug was fixed in the solver glpsol (command-line options - --btf, --cbg, and --cgr didn't work properly). - - A serious bug was fixed in a basis factorization routine used - on the dense phase. (The bug might appear only if the number of - rows exceeded sqrt(2**31) ~= 46,340 and caused access violation - exception because of integer overflow.) Thanks to Mark Meketon - for bug report. - - Two API routines glp_alloc and glp_realloc were documented. - Thanks to Brian Gladman for suggestion. - - Translation of the document "Modeling Language GNU MathProg" - to Spanish was included (in LaTeX and pdf formats). Thanks to - Pablo Yapura for contribution. - -GLPK 4.54 (release date: Mar 28, 2014) - - Block-triangular LU-factorization was implemented to be used - on computing an initial factorization of the basis matrix. - - A new version of the Schur-complement-based factorization - module was included in the package. Now it can be used along - with plain as well as with block-triangular LU-factorization. - - Currently the following flags can be used to specify the type - of the basis matrix factorization (glp_bfcp.type): - - GLP_BF_LUF + GLP_BF_FT LUF, Forrest-Tomlin update (default) - GLP_BF_LUF + GLP_BF_BG LUF, Schur complement, Bartels-Golub - update - GLP_BF_LUF + GLP_BF_GR LUF, Schur complement, Givens rotation - update - GLP_BF_BTF + GLP_BF_BG BTF, Schur complement, Bartels-Golub - update - GLP_BF_BTF + GLP_BF_GR BTF, Schur complement, Givens rotation - update - - In case of GLP_BF_FT the update is applied to matrix U, while - in cases of GLP_BF_BG and GLP_BF_GR the update is applied to - the Schur complement. - - Corresponding new options --luf and --btf were added to glpsol. - - For more details please see a new edition of the GLPK reference - manual included in the distribution. - - A minor bug (in reporting the mip solution status) was fixed. - Thanks to Remy Roy for bug report. - - A call to "iodbc-config --cflags" was added in configure.ac - to correctly detect iodbc flags. Thanks to Sebastien Villemot - for patch. - -GLPK 4.53 (release date: Feb 13, 2014) - - The API routine glp_read_mps was changed to remove free rows. - - A bug was fixed in the API routine glp_read_lp. Thanks to - Gabriel Hackebeil for bug report. - - The zlib compression library used by some GLPK routines and - included in the package was downgraded from 1.2.7 to 1.2.5 (as - in GLPK 4.50) because of addressability bugs on some 64-bit - platforms. Thanks to Carlo Baldassi - for bug report. - - A bug was fixed in a routine that reads gzipped files. Thanks - to Achim Gaedke for bug report. - - Two API routines glp_get_it_cnt and glp_set_it_cnt were added. - Thanks to Joey Rios for suggestion. - - All obsolete GLPK API routines (prefixed with lpx) were removed - from the package. - - A set of routines that simulate the old GLPK API (as defined - in 4.48) were added; see examples/oldapi/api/lpx.c. Thanks to - Jan Engelhardt for suggestion. - - A namespace bug was fixed in the SQL table drive module. Thanks - to Dennis Schridde for bug report. - -GLPK 4.52.1 (release date: Jul 28, 2013) - - This is a bug-fix release. - - A version information bug in Makefile.am was fixed. Thanks to - Sebastien Villemot for bug report. - -GLPK 4.52 (release date: Jul 18, 2013) - - The clique cut generator was essentially reimplemented, and now - it is able to process very large and/or dense conflict graphs. - - A simple rounding heuristic was added to the MIP optimizer. - - Some bugs were fixed in the proximity search heuristic routine. - Thanks to Giorgio Sartor <0gioker0@gmail.com>. - - New command-line option '--proxy [nnn]' was added to glpsol to - enable using the proximity search heuristic. - - A bug (incorrect processing of LI column indicator) was fixed - in the mps format reading routine. Thanks to Charles Brixko for - bug report. - -GLPK 4.51 (release date: Jun 19, 2013) - - Singleton and dense phases were implemented on computing - LU-factorization with Gaussian elimination. The singleton phase - is a feature that allows processing row and column singletons - on initial elimination steps more efficiently. The dense phase - is a feature used on final elimination steps when the active - submatrix becomes relatively dense. It significantly reduces - the time needed, especially if the active submatrix fits in CPU - cache, and improves numerical accuracy due to full pivoting. - - The API routine glp_adv_basis that constructs advanced initial - LP basis was replaced by an improved version, which (unlike the - old version) takes into account numerical values of constraint - coefficients. - - The proximity search heuristic for MIP was included in the GLPK - integer optimizer glp_intopt. On API level the heuristic can be - enabled by setting the parameter ps_heur in glp_iocp to GLP_ON. - This feature is also available in the solver glpsol through - command-line option '--proxy'. Many thanks to Giorgio Sartor - <0gioker0@gmail.com> for contribution. - - A bug was fixed that caused numerical instability in the FPUMP - heuristic. - -GLPK 4.50 (release date: May 24, 2013) - - A new version of LU-factorization routines were added. - Currently this version provides the same functionality as the - old one, however, the new version allows further improving. - - Old routines for FHV-factorization used to update the basis - factorization were replaced by a new version conforming to the - new version of LU-factorization. - - Some clarifications about using the name index routines were - added. Thanks to Xypron for suggestion. - - Some typos were corrected in the MathProg language reference. - Thanks to Jeffrey Kantor for report. - - A serious bug (out-of-range indexing error) was *tentatively* - fixed in the routine glp_relax4. Unfortunatly, this bug is - inherited from the original Fortran version of the RELAX-IV - code (for details please see ChangeLog), and since the code is - very intricate, the bug is still under investigation. Thanks to - Sylvain Fournier for bug report. - -GLPK 4.49 (release date: Apr 16, 2013) - - The new API routine glp_mincost_relax4, which is a driver to - relaxation method of Bertsekas and Tseng (RELAX-IV), was added - to the package. RELAX-IV is a code for solving minimum cost - flow problems. On large instances it is 100-1000 times faster - than the standard primal simplex method. Prof. Bertsekas, the - author of the original RELAX-IV Fortran code, kindly permitted - to include a C translation of his code in GLPK under GPLv3. - - A bug (wrong dual feasibility test) was fixed in API routine - glp_warm_up. Thanks to David T. Price - for bug report. - - Obsolete API routine lpx_check_kkt was replaced by new routine - glp_check_kkt. - - IMPORTANT: All old API routines whose names begin with 'lpx_' - were removed from API level and NO MORE AVAILABLE. - -GLPK 4.48 (release date: Jan 28, 2013) - - This is a maintainer release. - - Some minor changes in API (glpk.h) were made. For details - please see ChangeLog. - - Some bugs/typos were fixed. Thanks to - Raniere Gaia Costa da Silva, - Heinrich Schuchardt , and - Robbie Morrison for reports. - -GLPK 4.47 (release date: Sep 09, 2011) - - The new API routine glp_intfeas1 was added to the package. - This routine is a tentative implementation of the integer (0-1) - feasibility solver based on the CNF-SAT solver (which currently - is MiniSat). It may be used in the same way as glp_intopt to - find either any integer feasible solution or a solution, for - which the objective function is not worse than the specified - value. Detailed description of this routine can be found in the - document "CNF Satisfiability Problem", which is included in the - distribution (see doc/cnfsat.pdf). - - The following two options were added to glpsol: - - --minisat translate 0-1 feasibility problem to CNF-SAT - problem and solve it with glp_intfeas1/MiniSat - (if the problem instance is already in CNF-SAT - format, no translation is performed) - - --objbnd bound add inequality obj <= bound (minimization) or - obj >= bound (maximization) to 0-1 feasibility - problem (this option assumes --minisat) - - The paint-by-numbers puzzle model (pbn.mod) included in the - distribution is a nice example of the 0-1 feasibility problem, - which can be efficiently solved with glp_intfeas1/MiniSat. This - model along with a brief instruction (pbn.pdf) and benchmark - examples from encoded in GNU MathProg (*.dat) can - be found in subdirectory examples/pbn/. - - The glpsol lp/mip solver was modified to bypass postprocessing - of MathProg models if the solution reported is neither optimal - nor feasible. - - A minor bug in examples/Makefile.am was fixed to correctly - build glpk in a separate directory. Thanks to Marco Atzeri - for bug report and patch. - -GLPK 4.46 (release date: Aug 09, 2011) - - The following new API routines were added: - - glp_read_cnfsat read CNF-SAT problem data in DIMACS format - glp_check_cnfsat check for CNF-SAT problem instance - glp_write_cnfsat write CNF-SAT problem data in DIMACS format - glp_minisat1 solve CNF-SAT problem instance with MiniSat - - The routine glp_minisat1 is a driver to MiniSat, a CNF-SAT - solver developed by Niklas Een and Niklas Sorensson, Chalmers - University of Technology, Sweden. This routine is similar to - the routine glp_intopt, however, it is intended to solve a 0-1 - programming problem instance, which is the MIP translation of - a CNF-SAT problem instance. - - Detailed description of these new API routines can be found in - the document "CNF Satisfiability Problem", which is included in - the distribution (see files doc/cnfsat.tex and doc/cnfsat.pdf). - - The following new glpsol command-line options were added: - - --cnf filename read CNF-SAT problem instance in DIMACS - format from filename and translate it to MIP - --wcnf filename write CNF-SAT problem instance in DIMACS - format to filename - --minisat solve CNF-SAT problem instance with MiniSat - solver - - The zlib compression library (version 1.2.5) was ANSIfied, - modified according to GLPK requirements and included in the - distribution as an external software module. Thus, now this - feature is platform independent. - - Some bugs were fixed in the SQL table driver. Thanks to Xypron - . - -GLPK 4.45 (release date: Dec 05, 2010) - - This is a bug-fix release. - - Several bugs/typos were fixed. Thanks to - Xypron , - Robbie Morrison , and - Ali Baharev for reports. - - Some glpk documents was re-formatted and merged into a single - document. Now the glpk documentation consists of the following - three main documents (all included in the distribution): - - GLPK: Reference Manual - - GLPK: Graph and Network Routines - - Modeling Language GNU MathProg: Language Reference - -GLPK 4.44 (release date: Jun 03, 2010) - - The following suffixes for variables and constraints were - implemented in the MathProg language: - - .lb (lower bound), - .ub (upper bound), - .status (status in the solution), - .val (primal value), and - .dual (dual value). - - Thanks to Xypron for draft implementation - and testing. - - Now the MathProg language allows comment records (marked by - '#' in the very first position) in CSV data files read with the - table statements. Note that the comment records may appear only - in the beginning of a CSV data file. - - The API routine glp_cpp to solve the Critical Path Problem was - added and documented. - -GLPK 4.43 (release date: Feb 20, 2010) - - This is a maintainer release. - - `configure.ac' was changed to allow building the package under - Mac OS and Darwin with ODBC support. - Thanks to Xypron for suggestions and Noli - Sicad for testing. - - The SQL table driver was improved to process NULL data. Thanks - to Xypron . - - Some bugs were fixed in the LP/MIP preprocessor. - -GLPK 4.42 (release date: Jan 13, 2010) - - The following new API routines were added: - - glp_check_dup check for duplicate elements in sparse - matrix - glp_sort_matrix sort elements of the constraint matrix - glp_read_prob read problem data in GLPK format - glp_write_prob write problem data in GLPK format - glp_analyze_bound analyze active bound of non-basic variable - glp_analyze_coef analyze objective coefficient at basic - variable - glp_print_ranges print sensitivity analysis report (this - routine replaces lpx_print_sens_bnds and - makes it deprecated) - - For description of these new routines and the GLPK LP/MIP - format see a new edition of the reference manual included in - the distribution. (Chapter "Graph and network API routines" was - carried out from the main reference manual and included in the - distribution as a separate document.) - - The following new command-line options were added to the stand- - alone solver glpsol: - --glp filename read problem data in GLPK format - --wglp filename write problem data in GLPK format - --ranges filename print sensitivity analysis report (this - option replaces --bounds) - - Now all GLPK routines performing file I/O support special - filenames "/dev/stdin", "/dev/stdout", and "/dev/stderr", which - can be specified in the same way as regular filenames. This - feature is plaform-independent. - -GLPK 4.41 (release date: Dec 21, 2009) - - The following new API routies were added: - - glp_transform_row transform explicitly specified row - glp_transform_col transform explicitly specified column - glp_prim_rtest perform primal ratio test - glp_dual_rtest perform dual ratio test - - For description of these new routines see a new edition of the - reference manual included in the distribution. - - The following API routines are deprecated: lpx_transform_row, - lpx_transform_col, lpx_prim_ratio_test, lpx_dual_ratio_test. - - Some improvements were made in the MIP solver (glp_intopt). - - The SQL table driver used to read/write data in MathProg models - was changed to allow multiple arguments separated by semicolon - in SQL statements. Thanks to Xypron . - - Two new options were added to the glpsol stand-alone solver: - --seed value (to initialize the pseudo-random number generator - used in MathProg models with specified value), and - --ini filename (to use a basis previously saved with -w option - as an initial basis on solving similar LP's). - - Two new MathProg example models were included. Thanks to - Nigel Galloway and Noli Sicad - for contribution. - - Scripts to build GLPK with Microsoft Visual Studio 2010 for - both 32-bit and 64-bit Windows were included. Thanks to Xypron - for contribution and testing. - -GLPK 4.40 (release date: Nov 03, 2009) - - The following new API routines were added: - - glp_del_vertices remove vertices from graph - glp_del_arc remove arc from graph - glp_wclique_exact find maximum weight clique with the exact - algorithm developed by Prof. P. Ostergard - glp_read_ccdata read graph in DIMACS clique/coloring - format - glp_write_ccdata write graph in DIMACS clique/coloring - format - - For description of these new routines see a new edition of the - reference manual included in the distribution. - - The hybrid pseudocost branching heuristic was included in the - MIP solver. It is available on API level (iocp.br_tech should - be set to GLP_BR_PCH) and in the stand-alone solver glpsol - (via the command-line option --pcost). This heuristic may be - useful on solving hard MIP instances. - - The branching heuristic by Driebeck and Tomlin (used in the - MIP solver by default) was changed to switch to branching on - most fractional variable if an lower bound of degradation of - the objective is close to zero for all branching candidates. - - A bug was fixed in the LP preprocessor (routine npp_empty_col). - Thanks to Stefan Vigerske for the - bug report. - - A bug was fixed and some improvements were made in the FPUMP - heuristic module. Thanks to Xypron . - - A bug was fixed in the API routine glp_warm_up (dual - feasibility test was incorrect in maximization case). Thanks to - Uday Venkatadri for the bug report. - -GLPK 4.39 (release date: Jul 26, 2009) - - The following new API routines were added: - - glp_warm_up "warm up" LP basis - glp_set_vertex_name assign (change) vertex name - glp_create_v_index create vertex name index - glp_find_vertex find vertex by its name - glp_delete_v_index delete vertex name index - glp_read_asnprob read assignment problem data in DIMACS - format - glp_write_asnprob write assignment problem data in DIMACS - format - glp_check_asnprob check correctness of assignment problem - data - glp_asnprob_lp convert assignment problem to LP - glp_asnprob_okalg solve assignment problem with the - out-of-kilter algorithm - glp_asnprob_hall find bipartite matching of maxumum - cardinality with Hall's algorithm - - Also were added some API routines to read plain data files. - - The API routines glp_read_lp and glp_write_lp to read/write - files in CPLEX LP format were re-implemented. Now glp_write_lp - correctly writes double-bounded (ranged) rows by introducing - slack variables rather than by duplicating the rows. - - For description of these new routines see a new edition of the - reference manual included in the distribution. - - The 'xfree(NULL)' bug was fixed in the AMD routines. Thanks to - Niels Klitgord for bug report. - - The message "Crashing..." was changed to "Constructing initial - basis..." due to suggestion by Thomas Kahle . - - Some typos were corrected in glpsol output messages. Thanks to - Xypron for patch. - -GLPK 4.38 (release date: May 02, 2009) - - API routines glp_read_mps and glp_write_mps were improved. - - Some improvements were made in the dual simplex routines. - - Two external software modules AMD and COLAMD were included in - the distribution (for more details please see src/amd/README - and src/colamd/README). Now they are used in the interior-point - solver to reorder the matrix prior to Cholesky factorization. - - API routine glp_ipt_status may return two new statuses due to - changes in the routine glp_interior. For details please see the - reference manual included in the distribution. - - A minor bug was fixed in the graph/network routines. Thanks to - Nelson H. F. Beebe for bug report. - -GLPK 4.37 (release date: Mar 29, 2009) - - The 0-1 Feasibility Pump heuristic was included in the GLPK - integer optimizer glp_intopt. On API level the heuristic can be - enabled by setting the parameter fp_heur in glp_iocp to GLP_ON. - This feature is also available in the solver glpsol through - command-line option '--fpump'. For more details please see the - reference manual included in the distribution. - - The following new API routines were added: - - glp_print_sol write basic solution in printable format - glp_print_ipt write interior-point solution in printable - format - glp_print_mip write MIP solution in printable format - glp_read_graph read (di)graph from plain text file - glp_write_graph write (di)graph to plain text file - glp_weak_comp find all weakly connected components - glp_strong_comp find all strongly connected components - - The following API routines are deprecated: lpx_print_sol, - lpx_print_ips, lpx_print_mip, lpx_print_prob (the latter is - equivalent to glp_write_lp). - - A bug was fixed in the interior-point solver (glp_interior) to - correctly compute dual solution components when the problem is - scaled. - - The files configure.ac and Makefile.am were changed: - (a) to allow using autoreconf/autoheader; - (b) to allow building the package in a directory other than its - source directory. - Thanks to Marco Atzeri for bug report. - - An example model in the GNU MathProg language was added. - Thanks to Larry D'Agostino for - contribution. - -GLPK 4.36 (release date: Feb 06, 2009) - - The following new API routines were added to the package: - - glp_mincost_okalg find minimum-cost flow with out-of-kilter - algorithm - glp_maxflow_ffalg find maximal flow with Ford-Fulkerson - algorithm - - For detailed description of these new routines and related data - structures see chapter "Graph and Network API Routines" in a new - edition of the reference manual included in the distribution. - - The following two new command-line options were added to the - solver glpsol: - - --mincost read min-cost flow data in DIMACS format - --maxflow read maximum flow data in DIMACS format - - Duplicate symbols in the header glpk.h were removed to allow - using swig. - Thanks to Kelly Westbrooks and - Nigel Galloway for suggestion. - - A minor defect was fixed in the routine glp_write_lp. - Thanks to Sebastien Briais for bug report. - - A minor bug was fixed in the SQL module. - Thanks to Xypron for patch. - - Some new example models in the GNU MathProg modeling language - were added. Thanks to Sebastian Nowozin and - Nigel Galloway for contribution. - -GLPK 4.35 (release date: Jan 09, 2009) - - The following new API routines were added to the package: - - glp_create_graph create graph - glp_set_graph_name assign (change) graph name - glp_add_vertices add new vertices to graph - glp_add_arc add new arc to graph - glp_erase_graph erase graph content - glp_delete_graph delete graph - glp_read_mincost read minimum cost flow problem data in - DIMACS format - glp_write_mincost write minimum cost flow problem data in - DIMACS format - glp_mincost_lp convert minimum cost flow problem to LP - glp_netgen Klingman's network problem generator - glp_gridgen grid-like network problem generator - glp_read_maxflow read maximum flow problem data in DIMACS - format - glp_write_maxflow write maximum flow problem data in DIMACS - format - glp_maxflow_lp convert maximum flow problem to LP - glp_rmfgen Goldfarb's maximum flow problem generator - - For detailed description of these new routines and related data - structures see chapter "Graph and Network API Routines" in a new - edition of the reference manual included in the distribution. - - A minor change were made in the internal routine xputc. Thanks - to Luiz Bettoni for suggestion. - - A minor bug was fixed in the internal routine mpl_fn_time2str. - Thanks to Stefan Vigerske for bug report. - -GLPK 4.34 (release date: Dec 04, 2008) - - The GNU MathProg modeling language was supplemented with three - new built-in functions: - - gmtime obtaining current calendar time - str2time converting character string to calendar time - time2str converting calendar time to character string - - (Thanks to Xypron .) - - For detailed description of these functions see Appendix A in - the document "Modeling Language GNU MathProg", a new edition of - which was included in the distribution. - - A bug was fixed in the MIP solver. Thanks to Nigel Galloway - for bug report. - - A new makefile was added to build the GLPK DLL with Microsoft - Visual Studio Express 2008 for 64-bit Windows. Thanks to Xypron - for contribution and testing. - -GLPK 4.33 (release date: Oct 30, 2008) - - The following new API routines were added to the package: - glp_copy_prob copy problem object content - glp_exact solve LP in exact arithmetic - (makes lpx_exact deprecated) - glp_get_unbnd_ray determine variable causing unboundedness - (makes lpx_get_ray_info deprecated) - glp_interior solve LP with interior-point method - (makes lpx_interior deprecated) - - The following new API routines for processing models written in - the GNU Mathprog language were added to the package: - glp_mpl_alloc_wksp allocate the translator workspace - glp_mpl_read_model read and translate model section - glp_mpl_read_data read and translate data section - glp_mpl_generate generate the model - glp_mpl_build_prob build LP/MIP instance from the model - glp_mpl_postsolve postsolve the model - glp_mpl_free_wksp deallocate the translator workspace - (These routines make lpx_read_model deprecated.) - - For description of all these new API routines see the reference - manual included in the distribution. - - A crude implementation of CPLEX-like interface to GLPK API was - added to the package. Currently it allows using GLPK as a core - LP solver for Concorde, a well known computer code for solving - the symmetric TSP. For details see examples/cplex/README. - - Some bugs were fixed in the SQL table driver. Thanks to Xypron - . - -GLPK 4.32 (release date: Oct 03, 2008) - - The following new features were included in the MIP solver - (the API routine glp_intopt): - - * MIP presolver - * mixed cover cut generator - * clique cut generator - * Euclidean reduction of the objective value - - Due to changes the routine glp_intopt may additionally return - GLP_ENOPFS, GLP_ENODFS, and GLP_EMIPGAP. - - The API routines lpx_integer are lpx_intopt are deprecated, - since they are completely superseded by glp_intopt. - - The following new branch-and-cut API routines were added: - glp_ios_row_attr determine additional row attributes - glp_ios_pool_size determine current size of the cut pool - glp_ios_add_row add constraint to the cut pool - glp_ios_del_row delete constraint from the cut pool - glp_ios_clear_pool delete all constraints from the cut pool - - For description of these new routines see the reference manual - included in the distribution. - - The stand-alone solver glpsol was changed to allow multiple - data files. - - A new edition of the supplement "Using Data Tables in the GNU - MathProg Modeling Language" was included. - - As usual, some bugs were fixed (in the MathProg translator). - Thanks to Xypron . - -GLPK 4.31 (release date: Sep 02, 2008) - - The core LP solver based on the dual simplex method was - re-implemented and now it provides both phases I and II. - - The following new API routines were added: - glp_scale_prob automatic scaling of problem data - glp_std_basis construct standard initial LP basis - glp_adv_basis construct advanced initial LP basis - glp_cpx_basis construct Bixby's initial LP basis - - For description of these new routines see the reference manual - included in the distribution. - - The following API routines are deprecated: - lpx_scale_prob, lpx_std_basis, lpx_adv_basis, lpx_cpx_basis. - - Necessary changes were made in memory allocation routines to - resolve portability issues for 64-bit platforms. - - New version of the routine lpx_write_pb to write problem data - in OPB (pseudo boolean format) was added to the package. Thanks - to Oscar Gustafsson for the contribution. - - Two new makefiles were added to build the package for 32- and - 64-bit Windows with Microsoft Visual Studio Express 2008. - Thanks to Heinrich Schuchardt (aka - Xypron) for the contribution and testing. - - Two new makefiles were added to build the package with Digital - Mars C/C++ 8.50 and Open Watcom C/C++ 1.6 (for 32-bit Windows). - -GLPK 4.30 (release date: Aug 13, 2008) - - The core LP solver based on the primal simplex method was - re-implemented to allow its further improvements. Currently the - new version provides the same features as the old one, however, - it is a bit faster and more numerically stable. - - Some changes were made in the MathProg translator to allow <, - <=, >=, and > on comparing symbolic values. Thanks to Heinrich - Schuchardt for patches. - - Internal routine set_d_eps in the exact LP solver was changed - to prevent approximation errors in case of integral data. - Thanks to Markus Pilz for bug report. - -GLPK 4.29 (release date: Jul 06, 2008) - - The configure script was changed to disable all optional - features by default. For details please see file INSTALL. - - The following new API routines were added: - glp_erase_prob erase problem object content - glp_read_mps read problem data in MPS format - glp_write_mps write problem data in MPS format - glp_read_lp read problem data in CPLEX LP format - glp_write_lp write problem data in CPLEX LP format - - For description of these new routines see the reference manual - included in the distribution. - - The following API routines are deprecated: - lpx_read_mps, lpx_read_freemps, lpx_write_mps, - lpx_write_freemps, lpx_read_cpxlp, and lpx_write_cpxlp. - - Two bugs were fixed. Thanks to - Anne-Laurence Putz and - Xypron for bug report. - -GLPK 4.28 (release date: Mar 25, 2008) - - The iODBC and MySQL table drivers, which allows transmitting - data between MathProg model objects and relational databases, - were re-implemented to replace a static linking by a dynamic - linking to corresponding shared libraries. - Many thanks to Heinrich Schuchardt - for the contribution, Rafael Laboissiere - for useful advices concerning the shared library support under - GNU/Linux, and Vijay Patil for testing - this feature under Windows XP. - - A new optional feature was added to the package. This feature - is based on the zlib data compression library and allows GLPK - API routines and the stand-alone solver to read and write - compressed data files performing compression/decompression "on - the fly" (compressed data files are recognized by suffix `.gz' - in the file name). It may be useful in case of large MPS files - to save the disk space (up to ten times). - - The `configure' script was re-implemented. Now it supports the - following specific options: - - --with-gmp Enable using the GNU MP bignum library - --without-gmp Disable using the GNU MP bignum library - --with-zlib Enable using the zlib data compression - library - --without-zlib Disable using the zlib data compression - library - --enable-dl Enable shared library support (auto check) - --enable-dl=ltdl Enable shared library support (GNU) - --enable-dl=dlfcn Enable shared library support (POSIX) - --disable-dl Disable shared library support - --enable-odbc Enable using ODBC table driver - --disable-odbc Disable using ODBC table driver - --enable-mysql Enable using MySQL table driver - --disable-mysql Disable using MySQL table driver - - For more details please see file INSTALL. - -GLPK 4.27 (release date: Mar 02, 2008) - - Three new table drivers were added to the MathProg translator: - - xBASE built-in table driver, which allows reading and writing - data in .dbf format (only C and N fields are supported); - - MySQL table driver, which provides connection to a MySQL - database; - - iODBC table driver, which provides connection to a database - through ODBC. - - The MySQL and iODBC table drivers were contributed to GLPK by - Heinrich Schuchardt . - - The table driver is a program module which allows transmitting - data between MathProg model objects and external data tables. - - For detailed description of the table statement and table - drivers see the document "Using Data Tables in the GNU MathProg - Modeling Language" (file doc/tables.txt) included in the - distribution. Some examples which demonstrate using MySQL and - iODBC table drivers can be found in subdirectory examples/sql. - -GLPK 4.26 (release date: Feb 17, 2008) - - The table statement was implemented in the GNU MathProg - modeling language. This new feature allows reading data from - external tables into model objects such as sets and parameters - as well as writing results of computations to external tables. - - A table is a (unordered) set of records, where each record - consists of the same number of fields, and each field is - provided with a unique symbolic name called the field name. - - Currently the GLPK package has the only built-in table driver, - which supports tables in the CSV (comma-separated values) file - format. This format is very simple and supported by almost all - spreadsheets and database management systems. - - Detailed description of the table statement and CSV format can - be found in file doc/tables.txt, included in the distribution. - -GLPK 4.25 (release date: Dec 19, 2007) - - A tentative implementation of Gomory's mixed integer cuts was - included in the branch-and-cut solver. To enable generating - Gomory's cuts the control parameter gmi_cuts passed to the - routine glp_intopt should be set to GLP_ON. This feature is - also available in the solver glpsol through command-line option - '--gomory'. For more details please see the reference manual - included in the distribution. - -GLPK 4.24 (release date: Nov 21, 2007) - - A tentative implementation of MIR (mixed integer rounding) cuts - was included in the MIP solver. To enable generating MIR cuts - the control parameter mir_cuts passed to the routine glp_intopt - should be set to GLP_ON. This feature is also available in the - stand-alone solver glpsol via command-line option '--mir'. For - more details please see the reference manual included in the - distribution. - - The implementation is mainly based on the following two papers: - - 1. H. Marchand and L. A. Wolsey. Aggregation and mixed integer - rounding to solve MIPs. CORE discussion paper 9839, CORE, - Universite catholique de Louvain, June 1998. - - 2. G. Andreello, A. Caprara, and M. Fischetti. Embedding cuts - in a Branch&Cut framework. Preliminary draft, October 2003. - - MIR cuts can be generated on any level of the search tree that - makes the GLPK MIP solver to be a real branch-and-cut solver. - - A bug was fixed in the routine lpx_write_cpxlp. If a variable - x has upper bound and no lower bound, it should appear in the - bounds section as "-inf <= x <= u", not as "x <= u". Thanks to - Enric Rodriguez for the bug report. - -GLPK 4.23 (release date: Oct 28, 2007) - - The following new API routines were added: - - glp_read_sol read basic solution from text file - glp_write_sol write basic solution to text file - glp_read_ipt read interior-point solution from text file - glp_write_ipt write interior-point solution to text file - glp_read_mip read MIP solution from text file - glp_write_mip write MIP solution to text file - - For description of these routines and corresponding file - formats see Chapter "API Routines", Section "Utility routines" - in the reference manual included in the distribution. - - Advanced API routine glp_free_env was added. It may be used by - the application program to free all resources allocated by GLPK - routines. - - The following three new command-line options were added to the - solver glpsol: - - --mipgap tol set relative MIP gap tolerance - -r filename read solution from filename - -w filename write solution to filename - -GLPK 4.22 (release date: Sep 19, 2007) - - This is a maintainer release. - - A bug was fixed in the MIP preprocessor (ios_preprocess_node). - Thanks to Roberto Bagnara (Department of - Mathematics, University of Parma, Italy) for the bug report. - - A bug was fixed in the MIP preprocessor (col_implied_bounds), - due to which constraint coefficients with small magnitude could - lead to wrong implied bounds of structural variables. - - A similar bug was fixed in the routine reduce_bounds. - - A bug was fixed in the routines glp_set_mat_row and - glp_set_mat_col. (The bug appeared due to incorrect removing - zero elements from the row/column lists.) - - A bug was fixed in the API routines lpx_read_mps and - lpx_read_freemps, due to which bounds of type LI specified in - BOUNDS section were incorrectly processed. - - A call to standard function vsprintf was replaced by a call to - vsnprintf for security reasons. Many thanks to Peter T. Breuer - and Rafael Laboissiere . - -GLPK 4.21 (release date: Aug 28, 2007) - - Additional reasons for calling the callback routine used in the - MIP solver (glp_intopt) were introduced. Currently the following - reasons are supported: - - * request for subproblem selection - * request for preprocessing - * request for row generation - * request for heuristic solution - * request for cut generation - * request for branching - * better integer solution found - - A basic preprocessing component used to improve subproblem - formulations by tightening bounds of variables was included in - the MIP solver. Depending on the control parameter pp_tech - passed to the routine glp_intopt the preprocessing can be - performed either on the root level or on all levels (default) - or can be disabled. - - Backtracking heuristic used by default in the MIP solver was - changed to the "best local bound". - - For more details see Chapter "Advanced API routines", Section - "Branch-and-bound interface routines" in a new edition of the - reference manual included in the distribution. - -GLPK 4.20 (release date: Jul 26, 2007) - - API routine lpx_integer was replaced by API routine glp_intopt, - which provides equivalent functionality and additionally allows - the application to control the solution process by means of the - user-written callback routine, which is called by the solver at - various points of the branch-and-bound algorithm. Besides, the - new MIP solver allows generating "lazy" constraints and cutting - planes on all levels of the branch-and-bound tree, not only on - the root level. The routine lpx_integer is also still available - for the backward compatibility. - - The following new advanced API routines, which may be called - from the B&B callback routine, were included in the package: - - glp_ios_reason determine reason for calling callback - routine - glp_ios_get_prob access the problem object - glp_ios_tree_size determine size of the branch-and-bound tree - glp_ios_curr_node determine current active subproblem - glp_ios_next_node determine next active subproblem - glp_ios_prev_node determine previous active subproblem - glp_ios_up_node determine parent subproblem - glp_ios_node_level determine subproblem level - glp_ios_node_bound determine subproblem local bound - glp_ios_mip_gap compute relative MIP gap - glp_ios_heur_sol provide solution found by heuristic - glp_ios_terminate terminate the solution process - - For description of these routines see Chapter "Advanced API - routines", Section "Branch-and-bound interface routines" in a - new edition of the reference manual, which was included in the - distribution. - - Old version of the integer optimization suite (IOS) as well as - TSP solver tspsol based on it are no longer supported and were - removed from the package. - - A minor error in the MIP presolver was fixed; thanks to Graham - Rockwell for the bug report. - -GLPK 4.19 (release date: Jul 05, 2007) - - The principal change is upgrading to GPLv3. - - A serious bug in the routine glp_del_cols was fixed; thanks to - Cedric[FR] for the bug report. The bug - appeared because on deleting non-basic columns the basis header - remained valid, however, contained invalid (old) column ordinal - numbers. - - A new advanced API routine glp_mem_limit was added. - - The case GLP_EBOUND was added to the routine lpx_simplex. - Thanks to Cameron Kellough for the - bug report. - - An API routine lpx_write_pb to write the problem instance in - OPB (pseudo boolean) format format was added. Thanks to Oscar - Gustafsson for the contribution. - - Two new options --wpb and --wnpb were added to glpsol to write - the problem instance in OPB format. - -GLPK 4.18 (release date: Jun 25, 2007) - - The following new API routines were added: - - glp_set_rii set (change) row scale factor - glp_set_sjj set (change) column scale factor - glp_get_rii retrieve row scale factor - glp_get_sjj retrieve column scale factor - glp_simplex solve LP problem with the simplex method - (this routine replaces lpx_simplex, which is - also available for backward compatibility) - glp_init_smcp initialize simplex method control params - glp_bf_exists check if the basis factorization exists - glp_factorize compute the basis factorization - glp_bf_updated check if the basis factorization has been - updated - glp_get_bfcp retrieve basis factorization control params - glp_set_bfcp change basis factorization control params - glp_get_bhead retrieve the basis header information - glp_get_row_bind retrieve row index in the basis header - glp_get_col_bind retrieve column index in the basis header - glp_ftran perform forward transformation - glp_btran perform backward transformation - - For description of all these routines see a new edition of the - reference manual included in the distribution. - - Type names ulong_t and uldiv_t were changed to glp_ulong and - glp_uldiv to avoid conflicts with standard type names on some - platforms. Thanks to Boris Wirtz - for the bug report. - - Some new examples in the MathProg language were added. Thanks - to Sebastian Nowozin . - -GLPK 4.17 (release date: May 26, 2007) - - API routines glp_set_mat_row, glp_set_mat_col, and glp_load_mat - were modified to allow zero constraint coefficients (which are - not stored in the constraint matrix). Note that constraint - coefficients with duplicate row/column indices are not allowed. - - Another form of LP basis factorization was implemented in the - package. It is based on LU-factorization of an initial basis - and Schur complement to reflect changes in the basis. Currently - the implementation is incomplete and provides only updating the - factorization on replacing a column of the basis matrix. On API - level the user can set the control parameter LPX_K_BFTYPE to - choose between the folloiwng forms of LP basis factorization to - be used in the simplex method routines: - 1) LU + Forrest-Tomlin update; - 2) LU + Schur complement + Bartels-Golub update; - 3) LU + Schur complement + Givens rotation update. - The GLPK implementation is similar to LUSOL/LUMOD developed by - Michael A. Saunders. - - The user can choose the form of LP basis factorzation used by - the simplex method routines by specifying the folloiwng options - of glpsol: --luf, --cbg, --cgr. - -GLPK 4.16 (release date: May 05, 2007) - - A number of basic GLPK API routines, which now are in the - stable stable, were renamed to be prefixed with 'glp_'. Note - that all these routines are available via their old names - prefixed with 'lpx_' that keeps the downward compatibility with - older versions of the package. - - Three new GLPK API routines were added to the package: - glp_version, glp_term_hook, and glp_mem_usage; for more details - see a new edition of the GLPK reference manual included in the - distribution. The routine glp_version reports the actual version - of the GLPK library and also can be used (along with the header - glpk.h) in Autotools specification files to check if the GLPK - library has been installed. - - The header glpk.h was changed to conform to C++ environment. - -GLPK 4.15 (release date: Feb 18, 2007) - - Autotools specification files (configure.ac, Makefile.am) were - changed to use GNU Libtool. This allows building the static as - well as shared GLPK library. - -GLPK 4.14 (release date: Feb 05, 2007) - - Now GLPK conforms to ILP32, LLP64, and LP64 programming models - (the latter seems to be the ultimate choice regarding 64-bit - architectures). Note that GLPK itself is a 32-bit application, - and the conformity only means that the package works correctly - on all these arenae. Nevertheless, on 64-bit platforms it is - possible to use more than 4GB of memory, if necessary. - -GLPK 4.13 (release date: Nov 13, 2006) - - A tentative implementation of the "exact" simplex method based - on bignum (rational) arithmetic was included in the package. - - On API level this new feature is available through the routine - lpx_exact, which is similar to the routine lpx_simplex. - - In the solver glpsol this feature is available through two new - command-line options: --exact and --xcheck. If the '--exact' - option is specified, glpsol solves LP instance using the exact - simplex method; in case of MIP it is used to obtain optimal - solution of LP relaxation. If the --xcheck option is specified, - LP instance (or LP relaxation) is solved using the standard - (floating-point) simplex method, however, then glpsol calls the - exact simplex routine to make sure that the final LP basis is - exactly optimal, and if it is not, to perform some additional - simplex iterations in exact arithmetic. - -GLPK 4.12 (release date: Nov 08, 2006) - - A tentative implementation of some simplex method routines - based on exact (bignum) arithmetic was included in the package. - Currently these routines provide computing LU-factorization of - the basis matrix and computing components of basic solution. - - These routines were used to implement a routine, which checks - primal and dual feasibility of basic solution exactly, i.e. in - rational numbers, without round-off errors. In glpsol this - feature is available through the command-line option --xcheck. - - GLPK has its own low-level routines implementing operations on - integer and rational numbers that makes it independent on other - software packages. However, to attain a much better performance - it is highly recommended to install (before configuring GLPK) - the GNU Multiple Precision Arithmetic Library (GMP). Using GMP - makes computations 100-200 times faster. - -GLPK 4.11 (release date: Jul 25, 2006) - - Three new built-in functions in the modeling language were - implemented: card (cardinality of set), length (length of - character string), and substr (substring of character string). - Another improvement concerns the printf statement which now - allows redirecting its output to a specified file. These new - features are illustrated in example models crypto.mod and - graph.mod included in the distribution. For more details see - the document "Modeling Language GNU MathProg". - - Four batch files (along with corresponding makefiles) were - included in the distribution to simplify building GLPK under - MS Windows; see them in subdirectory 'w32'. - -GLPK 4.10 (release date: May 11, 2006) - - Cutting planes of two new classes were implemented: mixed cover - cuts and clique cuts. On API level this feature can be enabled - by setting control parameter LPX_K_USECUTS passed to the routine - lpx_intopt. In glpsol this feature is available through the - command-line options --cover and --clique. For more details see - the reference manual. - - Now the routines lpx_read_mps and lpx_read_freemps support LI - bound type. It is similar to LO, however, indicates the column - as of integer kind. - -GLPK 4.9 (release date: Jan 17, 2006) - - An advanced MIP solver was implemented. It includes: - - - basic presolving technique (removing free, singleton and - redundant rows, improving bounds of columns, removing fixed - columns, reducing constraint coefficents); - - - generating cutting planes to improve LP relaxation (currently - only Gomory's mixed integer cuts are implemented); - - - using the branch-and-bound method to solve resultant MIP; - - - recovering solution of the original MIP. - - The solver is available on API level via the routine lpx_intopt - (see the reference manual). It is similar to the routine - lpx_integer, however, does not require initial solution of LP - relaxation. - - The solver is also available in the command-line utility glpsol - via two options: --intopt (only presolving) and --cuts (assumes - --intopt plus generating cuts). - - Note that efficiency of the MIP solver strongly depends on the - internal structure of the problem to be solved. For some hard - instances it is very efficient, but for other instances it may - be significantly worse than the standard branch-and-bound. - - For some comparative benchmarks see doc/bench1.txt. - - Well, what else... - - Three built-in functions were added to MathProg: sin, cos, and - atan (the latter allows one or two arguments). - - Some bugs were fixed. - - Several new examples in MathProg were included: color.mod - (graph coloring problem), tsp.mod (traveling salesman problem), - and pbn.mod (paint-by-numbers puzzle). - -GLPK 4.8 (release date: Jan 12, 2005) - - Core simplex method and interior-point method routines were - re-implemented and now they use a new, "storage-by-rows" sparse - matrix format (unlike previous versions where linked lists were - used to represent sparse matrices). For details see ChangeLog. - - Also a minor bug was fixed in API routine lpx_read_cpxlp. - -GLPK 4.7 (release date: Aug 23, 2004) - - Now GLPK supports free MPS format. Two new API routines - lpx_read_freemps (to read problem data in free MPS format) and - lpx_write_freemps (to write problem data in free MPS format) - were added. This feature is also available in the solver glpsol - via new command-line options --freemps and --wfreemps. For more - details see the GLPK reference manual. - - API routines lpx_read_cpxlp and lpx_write_cpxlp for reading and - writing problem data in CPLEX LP format were re-implemented to - allow long symbolic names (up to 255 characters). - - The following three modules were temporarily removed from the - GLPK distribution due to licensing problems: DELI (an interface - module to Delphi), GLPKMEX (an interface module to Matlab), and - JNI (an interface module to Java). - -GLPK 4.6 (release date: Aug 04, 2004) - - Three new statements were implemented in the GNU MathProg - language: solve, printf, and for. Their detailed description can - be found in the GLPK documentation included in the distribution. - (See also a sample model, examples/queens.mod, which illustrates - using these new statements.) - - Two new API routines were added to the package: lpx_read_prob - and lpx_write_prob. They allow reading/writing problem data in - GNU LP low-level text format. - - Three new command-line options were implemented in the LP/MIP - solver glpsol: --glp (to read problem data in GNU LP format), - --wglp (to write problem data in GNU LP format), and --name (to - change problem name). Now glpsol also supports processing models - where the new statements (see above) are used. - - A new version of GLPKMEX, a Matlab MEX interface to GLPK, was - included. For more details see contrib/glpkmex/ChangeLog. - -GLPK 4.5 (release date: Jul 19, 2004) - - The branch-and-bound solver was completely re-implemented. - - Some modifications were made in memory allocation routines that - allows using the package on 64-bit platforms. - - For more details see ChangeLog. - -GLPK 4.4 (release date: Jan 17, 2004) - - All API routines were re-implemented using new data structures. - The new implementation provides the same specifications and - functionality of API routines as the old one, however, it has - some important advantages, in particular: - * linked lists are used everywhere that allows creating and - modifying the problem object as efficiently as possible - * all data stored in the problem object are non-scaled (even if - the internal scaling is used) that prevents distortion of the - original problem data - * solution components obtained by the solver remain available - even if the problem object has been modified - * no solver-specific data are used in the new data structures - that allows attaching any external lp/mip solver using GLPK - API as an uniform interface - Note that some API routines became obsolete being replaced by - new, more convenient routines. These obsolete routines are kept - for backward compatibility, however, they will be removed in - the future. For more details please see ChangeLog and the GLPK - Reference Manual. - - New edition of the GLPK Reference Manual was included in the - distribution. - - GLPKMEX, a Matlab MEX interface to GLPK package, contributed by - Nicolo Giorgetti was included in the - distribution. - - GLPK FAQ contributed by Harley Mackenzie was - included in the distribution. - -GLPK 4.3 (release date: Dec 12, 2003) - - The bug, due to which the standard math library is not linked - on building the package on some platforms, was fixed. - - The following new built-in functions were added to the MathProg - language: round, trunc, Irand224, Uniform01, Uniform, Normal01, - Normal. For details see the language description. - - The MathProg syntax was changed to allow writing 'subj to' that - means 'subject to'. - - The new api routine lpx_get_ray_info was added. It is intended - to determine which (non-basic) variable causes unboundness. For - details see the reference manual. - - The module glpmps.c was changed to avoid compilation errors on - building the package on Mac OS X. - - Several typos was fixed and some new material was added to the - GLPK documentation. - -GLPK 4.2 (release date: Nov 14, 2003) - - A preliminary implementation of the Integer Optimization Suite - (IOS) was included in the package. The Branch-and-Cut Framework - being completely superseded by IOS was removed from the package. - - New API routine lpx_print_sens_bnds intended for bounds - sensitivity analysis was contributed to GLPK by Brady Hunsaker - . This function is also available in - the solver glpsol (via command-line option --bounds). - - An improved version of GLPK JNI (Java Native Interface) was - contributed by Chris Rosebrugh . - - GLPK DELI (Delphi Interface) was contributed by Ivo van Baren - . - - Several makefiles were added to allow compiling GLPK on some - non-GNU 32-bit platforms: - * Windows single-threaded static library, Visual C++ 6.0 - * Windows multi-threaded dynamic library, Visual C++ 6.0 - * Windows single-threaded static library, Borland C++ 5.2 - * DOS single-threaded static library, Digital Mars C++ 7.50 - - And, of course, some bugs were fixed. - - For more details see ChangeLog. - -GLPK 4.1 (release date: Aug 23, 2003) - - Some improvements were made in the lp/mip solver routines and - several bugs were fixed in the model translator. - - For more details see ChangeLog. - -GLPK 4.0 (release date: May 06, 2003) - - Now GLPK supports the GNU MathProg modeling language, which is - a subset of the AMPL modeling language. - - The document "GLPK: Modeling Language GNU MathProg" included in - the distribution is a complete description of GNU MathProg. (See - the files lang.latex, lang.dvi, and lang.ps in the subdirectory - 'doc'. See also some examples in the subdirectory 'sample'.) - - New version of the solver glpsol, which supports models written - in GNU MathProg, was implemented. (Brief instructions how to use - glpsol can be found in the GNU MathProg documentation.) - - The GLPK/L modeling language is no more supported. The reason is - that GNU MathProg being much more powerful completely supersedes - all features of GLPK/L. - -GLPK 3.3 (release date: Mar 25, 2003) - - LP PRESOLVER - ------------ - - Now the routine lpx_simplex (which is a driver to the simplex - method for solving LP) is provided with the built-in LP - presolver, which is a program that transforms the original LP - problem to an equivalent LP problem, which may be easier for - solving with the simplex method than the original one. Once the - transformed LP has been solver, the presolver transforms its - basic solution back to a corresponding basic solution of the - original problem. For details about this feature please see the - GLPK reference manual. - - Currently the LP presolver implements the following features: - * removing empty rows; - * removing empty columns; - * removing free rows; - * removing fixed columns; - * removing row singletons, which have the form of equations; - * removing row singletons, which have the form of inequalities; - * removing column singletons, which are implied slack variables; - * fixing and removing column singletons, which are implied free - variables; - * removing forcing rows that involves fixing and removing the - corresponding columns; - * checking for primal and dual infeasibilities. - - The LP presolver is also used by default in the stand-alone - program glpsol. In order *not* to use it, the option --nopresol - should be specified in the command-line. - - CHANGES IN GLPK/L - ----------------- - - The syntax and semantics of the GLPK/L modeling language was - changed to allow declaration of "interval" sets. This means that - now the user can declare a set, for example, as: - - set task = [8:11]; - - that is exactly equivalent to the following declaration: - - set task = (task_8, task_9, task_10, task_11); - - For details see the language description. - - JAVA INTERFACE - -------------- - - Now GLPK includes the package GLPK JNI (Java Native Interface) - that implements Java binding for GLPK. It allows Java programs - to utilize GLPK in solving LP and MIP problems. For details see - a brief user's guide in the subdirectory contrib/java-binding. - This package was developed and programmed by Yuri Victorovich - , who contributed it to GLPK. - -GLPK 3.2.4 (release date: Feb 18, 2003) - - This is a bug-fix release. For details see ChangeLog. - -GLPK 3.2.3 (release date: Nov 11, 2002) - - A new implementation of the api routine lpx_integer which now - is based on the b&b driver (which is based on the implicit - enumeration suite) was included in the package. This new - implementation has exactly the same functionality as the old - version, so all changes are transparent to the api user. - - Four new api routines were included in the package: - lpx_check_kkt checks Karush-Kuhn-Tucker optmality conditions; - lpx_read_bas reads predifined basis in MPS format; - lpx_write_bas writes current basis in MPS format; - lpx_write_lpt writes problem data in CPLEX LP format. - - Also other minor improvements were made (for details see the - file 'ChangeLog'). - -GLPK 3.2.2 (release date: Oct 14, 2002) - - The api routine lpx_read_lpt was included in the package. It - is similar to the routine lpx_read_mps and intended to read - LP/MIP data prepared in CPLEX LP format. Description of this - format is given in the GLPK reference manual, a new edition of - which was also included in the distribution (see the files - 'refman.latex', 'refman.dvi', 'refman.ps' in the subdirectory - 'doc'). In order to use data files in CPLEX LP format with the - solver glpsol the option '--lpt' should be specified in the - command line. - - Several bugs were fixed and some minor improvements were made - (for details see the file 'ChangeLog'). - -GLPK 3.2.1 (release date: Aug 12, 2002) - - Now GLPK includes a preliminary implementation of the - branch-and-cut framework, which is a set of data structures and - routines intended for developing branch-and-cut methods for - solving mixed-integer and combinatorial optimization problems. - - Detailed decsription of the branch-and-cut framework is given in - the document "GLPK: A Preliminary Implementation of the - Branch-And-Cut Framework" included in the distribution (see the - file 'brcut.txt' in the subdirectory 'doc'). - - In order to illustrate how the GLPK branch-and-cut framework - can be used for solving a particular optimization problem there - is an example included in the package. This is a stand-alone - program, TSPSOL, which is intended for solving to optimality the - symmetric Traveling Salesman Problem (TSP), a classical problem - of the combinatorial optimization (see the file 'tspsol.c' in - the subdirectory 'sample'). - -GLPK 3.2 (release date: Jul 15, 2002) - - New edition of the document "GLPK: Reference Manual" was - included (see the files 'refman.latex', 'refman.dvi', and - 'refman.ps' in the subdirectory 'doc'). - - New edition of the document "GLPK: Modeling Language GLPK/L" was - included (see the files 'lang.latex', 'lang.dvi', and 'lang.ps' - in the subdirectory 'doc'). - - The following new API routines were added to the package: - - lpx_transform_row (transform explicitly specified row); - lpx_transform_col (transform explicitly specified column); - lpx_prim_ratio_test (perform primal ratio test); - lpx_dual_ratio_test (perform dual ratio test); - lpx_interior (solve LP problem using interior point method); - lpx_get_ips_stat (query status of interior point solution); - lpx_get_ips_row (obtain row interior point solution); - lpx_get_ips_col (obtain column interior point solution); - lpx_get_ips_obj (obtain interior point value of obj.func.); - lpx_read_lpm (read LP/MIP model written in GLPK/L); - lpx_write_mps (write problem data using MPS format); - lpx_print_ips (print interior point solution). - - Detailed description of all these new API routines are given in - the new edition of the reference manual. - - New version of the stand-alone solver glpsol (which is based on - the new API) was implemented. - - So long as the new API (introduced in glpk 3.0) now provides - all the functions, which were provided by the old API, the old - API routines were removed from the package at all. - -GLPK 3.1 (release date: May 27, 2002) - - A preliminary implementation of new API routines was completed - and included in the package. - - These new API routines provide much more flexible interaction - between the application program, LP/MIP problem instances, and - solver routines. Based on completely changed data structures - they are, however, similar to the API routines and provide the - same functionality. Please note that three routines, namely, - solving LPs using interior point method, reading model written - in the GLPK/L modeling language, and writing problem data in - the MPS format, are not implemented in the new API, however, - these routines are planned to be implemented in the next version - of the package. - - A description of the new API routines is given in the document - "GLPK Reference Manual", a draft edition of which is included - in the package (see the files 'refman.latex', 'refman.dvi', and - 'refman.ps' in the subdirectory 'doc'). - - Although the old API routines are kept in the package, they are - no longer supported and will be removed in the future. - -GLPK 3.0.8 (release date: May 13, 2002) - - A preliminary implementation of new API routines was included - in the package. These new API routines are intended to provide - much more flexible interaction between the application program, - LP/MIP problem and solver routines. See the document "New GLPK - API Routines" (the file 'newapi.txt' in the subdirectory 'doc') - also included in the package. - - The api routines glp_simplex2, glp_call_ipm1, glp_call_bbm1 were - renamed, respectively, to glp_simplex, glp_interior, glp_integer - in order to reflect changes in implementation. The api routines - glp_call_rsm1, glp_simplex1, glp_pivot_in, glp_pivout_out were - removed from the package since they are completely supreseded by - the new API routines (however, these routines still can be found - in the subdirectory 'oldsrc'). Please consult a new edition of - the document "GLPK User's Guide" about all these changes in the - existing api routines. - - The document "GLPK Library Reference" was removed from the - package (into the subdirectory 'oldsrc') since it describes the - obsolete library routines, most of which are no longer used. - -GLPK 3.0.7 (release date: Apr 22, 2002) - - A new, more efficient implementation of the primal/dual simplex - method was included in the package. Due to some improvements the - simplex-based solver allows solving many LP problems faster and - provides more reliable results. Note that the new implementation - is currently incomplete and available only via the api routine - glp_simplex2. - - All the changes are transparent on API level. - -GLPK 3.0.6 (release date: Mar 28, 2002) - - New version of LU-factorization and basis maintenance routines - (based on Forrest-Tomlin updating technique) was implemented. - Since these new routines functionally supersede some routines - (which implement other forms of the basis matrix) and make them - obsolete, the latter were removed from the package (they still - can be found in the subdirectory 'oldsrc'). - - All the changes are transparent on API level. - -GLPK 3.0.5 (release date: Jan 29, 2002) - - New edition of the document "GLPK User's Guide" was included in - the distribution. Now it describes all additional API routines, - which were recently added to the package. - - Structure of the package was re-organized in order to make its - maintenance easier (all small files in the subdurectory 'source' - were merged in bigger units). These changes are transparent for - the user. - -GLPK 3.0.4 (release date: Dec 10, 2001) - - A new, more efficient implementation of the two-phase primal - simplex method was included in the package. Due to some new - features (an advanced initial basis, projected steepest edge, - recursive updating values and reduced costs) the new LP solver - is faster and numerically more stable than the old one. - - The new LP solver is available as API routine glp_simplex2 and - has the same purpose as API routine glp_call_rsm1. For detailed - specification see the file 'newapi.txt' in the directory 'doc'. - - Now the new LP solver is also used by default to solve an - initial LP problem in the branch-and-bound routine glp_call_bbm1 - instead the routine rsm1_driver. Note that the branch-and-bound - procedure itself is still based on rsm1_driver. - - The new LP solver is also used as default solver in GLPSOL for - solving LP and MIP problems. In order to choose the old solver - the option '--old-sim' can be specified in the command line. - -GLPK 3.0.3 (release date: Oct 03, 2001) - - Some minor changes were made in the simplex method routines in - order to improve numerical stability of the method. - -GLPK 3.0.2 (release date: Sep 24, 2001) - - A new implementation of the basis maintaining routines was - included in the package. These routines, which are based on so - called FHV-factorization (a variety of LU-factorization) of the - basis matrix and Gustavson's data structures, allows performing - the main operations faster at the expense of some worsening - numerical accuracy. - - AFI (Advanced Form of the Inverse), which is the form of the - basis matrix based on FHV-factorization, is available via the - parameter form = 3 (on API level) or via the option --afi (in - GLPSOL solver). - -GLPK 3.0.1 (release date: Aug 01, 2001) - - Old GLPK API routines have been removed from the package. - - New GLPK API routines were added: - - - scaling routines; - - - a routine for writing problem data in MPS format; - - - a comprehensive driver to the simplex method; - - - basis maintaining routines. - - A description of the new API routines is given in the document - "Additional GLPK API Routines". This document is included into - the distribution in plain text format (see the file 'newapi.txt' - in the subdirectory 'doc'). - - Now the distribution includes a non-trivial example of using - GLPK as a base LP solver for Concorde, a well known program that - solves Traveling Salesman Problem (TSP). For further details see - comments in the file 'sample/lpglpk30.c'. - -GLPK 3.0 (release date: Jul 19, 2001) - - Now GLPK is provided with new API, which being more flexible - can be used in more complex algorithmic schemes. - - New edition of the document "GLPK User's Guide" is included in - the distribution. Now it completely corresponds to the new GLPK - API routines. - - Old API routines are not removed yet from the package, however - they became obsolete and therefore should not be used. Since now - the header glpk.h corresponds to new API, in order to compile - existing programs that use old GLPK API routines the statement - - #define GLP_OLD_API - - should be inserted before the statement - - #include "glpk.h" - -GLPK 2.4.1 (release date: Jun 14, 2001) - - The document "Modeling language GLPK/L" is included into the - distribution in texinfo format. - - New edition of the document "GLPK User's Guide" is included in - the distribution. Now it describes all additional API routines - which were recently added to the package. - -GLPK 2.4 (release date: May 10, 2001) - - Now GLPK includes an implementation of a preliminary version - of the GLPK/L modeling language. This language is intended for - writing mathematcal programming models. The name GLPK/L is - derived from GNU Linear Programming Kit Language. - - A brief description of the GLPK/L language is given in the - document "GLPK/L Modeling Language: A Brief Description". This - document is included into the distribution in plain text format - (see the file 'language.txt' in the subdirectory 'doc'). - - The language processor (which is a program that analyzes model - description written in GLPK/L and translates it to internal data - structures) is available as the GLPK API routine. - - The stand-alone solver GLPSOL now is able: a) to process model - descriptions written in the GLPK/L language; b) to solve pure LP - problems using the interior point method (therefore the program - GLPIPM was removed from the package). - -GLPK 2.3 (release date: Apr 09, 2001) - - New edition of the document "GLPK User's Guide" is included in - the distribution. Now it describes all additional API routines - which were recently added to the package. - - The MIP solver was fully re-programmed in order to improve its - robustness and performance. In particular, a basis recovering - procedure was implemented (this procedure allows switching to - the primal simplex method in case when the dual simplex method - fails). - -GLPK 2.2 (release date: Mar 15, 2001) - - Now GLPK includes a tentative implementation of the - branch-and-bound procedure based on the dual simplex method for - mixed integer linear programming (MIP). - - Complete description of this new feature of the package is given - in the preliminary document "Mixed Integer Linear Programming - Using GLPK Version 2.2 (Supplement to GLPK User's Guide)". This - document is included into the distribution in plain text format - (see the file 'mip.txt' in the subdirectory 'doc'). - - The MIP solver (glp_integer) can be used as GLPK API routine in - the same way as the pure LP solver (glp_simplex). - - The stand-alone program 'glpsol' is now able to solve LP as well - as MIP problems. - - Note that the current version of GLPK MIP solver is based on - easiest heuristics for branching and backtrackng. Therefore the - solver is fit mainly for MIP problems which are not very hard - and have few integer variables. - -GLPK 2.1 (release date: Feb 19, 2001) - - The document "GLPK Implementation of the Revised Simplex Method" - is included into the distribution. This document describes most - of routines related to the revised simplex method. - -GLPK 2.0 (release date: Jan 25, 2001) - - Now GLPK includes a tentative implementation of the primal-dual - interior point method for large-scale linear programming. - - The interior point solver can be used as GLPK API routine in the - same manner as the simplex method solver (glp_simplex): - - ret = glp_interior(); - - Note that currently the interior point solver implemented in - GLPK doesn't include many important features, in particular: - - * it can't process dense columns; therefore if the problem has - dense columns, the solving will be extremely inefficient; - - * it has no special features against numerical unstability; - some problems may cause premature termination of the solving - when the matrix A*D*A' becomes ill-conditioned; - - * it computes only values of primal (auxiliary and structural) - variables and doesn't compute values of dual variables (i.e. - reduced costs) which are just set to zero; - - * it doesn't identify optimal basis corresponding to the found - interior point solution; all variables in the found solution - are just marked as basic variables. - - GLPK also includes a stand-alone program 'glpipm' which is a - demo based on the interior point method. It may be used in the - same way as the program 'glpsol' that is based on the simplex - method. diff --git a/code/3rd_glpk/README b/code/3rd_glpk/README deleted file mode 100644 index 53693e94..00000000 --- a/code/3rd_glpk/README +++ /dev/null @@ -1,39 +0,0 @@ - Olga K. gewidmet - -GLPK (GNU Linear Programming Kit) Version 4.64 - -Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -2009, 2010, 2011, 2013, 2014, 2015, 2016, 2017 Andrew Makhorin, -Department for Applied Informatics, Moscow Aviation Institute, Moscow, -Russia. All rights reserved. E-mail: . - -GLPK is part of the GNU Project released under the aegis of GNU. - -GLPK is free software: you can redistribute it and/or modify it -under the terms of the GNU General Public License as published by the -Free Software Foundation, either version 3 of the License, or (at your -option) any later version. - -See the file COPYING for the GNU General Public License. - -See the file INSTALL for compilation and installation instructions. - -The GLPK package is a set of routines written in ANSI C and organized -in the form of a callable library. This package is intended for solving -large-scale linear programming (LP), mixed integer linear programming -(MIP), and other related problems. - -The GLPK package includes the following main components: - -* primal simplex method; -* dual simplex method; -* exact simplex method based on rational arithmetic; -* primal-dual interior-point method; -* branch-and-cut method; -* application program interface (API); -* GNU MathProg modeling language (a subset of AMPL); -* GLPSOL (stand-alone LP/MIP solver). - -See GLPK webpage . - -Please report bugs to . diff --git a/code/3rd_glpk/amd/COPYING b/code/3rd_glpk/amd/COPYING deleted file mode 100644 index 84bba36d..00000000 --- a/code/3rd_glpk/amd/COPYING +++ /dev/null @@ -1,502 +0,0 @@ - GNU LESSER GENERAL PUBLIC LICENSE - Version 2.1, February 1999 - - Copyright (C) 1991, 1999 Free Software Foundation, Inc. - 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - -[This is the first released version of the Lesser GPL. It also counts - as the successor of the GNU Library Public License, version 2, hence - the version number 2.1.] - - Preamble - - The licenses for most software are designed to take away your -freedom to share and change it. By contrast, the GNU General Public -Licenses are intended to guarantee your freedom to share and change -free software--to make sure the software is free for all its users. - - This license, the Lesser General Public License, applies to some -specially designated software packages--typically libraries--of the -Free Software Foundation and other authors who decide to use it. You -can use it too, but we suggest you first think carefully about whether -this license or the ordinary General Public License is the better -strategy to use in any particular case, based on the explanations below. - - When we speak of free software, we are referring to freedom of use, -not price. Our General Public Licenses are designed to make sure that -you have the freedom to distribute copies of free software (and charge -for this service if you wish); that you receive source code or can get -it if you want it; that you can change the software and use pieces of -it in new free programs; and that you are informed that you can do -these things. - - To protect your rights, we need to make restrictions that forbid -distributors to deny you these rights or to ask you to surrender these -rights. These restrictions translate to certain responsibilities for -you if you distribute copies of the library or if you modify it. - - For example, if you distribute copies of the library, whether gratis -or for a fee, you must give the recipients all the rights that we gave -you. You must make sure that they, too, receive or can get the source -code. If you link other code with the library, you must provide -complete object files to the recipients, so that they can relink them -with the library after making changes to the library and recompiling -it. And you must show them these terms so they know their rights. - - We protect your rights with a two-step method: (1) we copyright the -library, and (2) we offer you this license, which gives you legal -permission to copy, distribute and/or modify the library. - - To protect each distributor, we want to make it very clear that -there is no warranty for the free library. Also, if the library is -modified by someone else and passed on, the recipients should know -that what they have is not the original version, so that the original -author's reputation will not be affected by problems that might be -introduced by others. - - Finally, software patents pose a constant threat to the existence of -any free program. We wish to make sure that a company cannot -effectively restrict the users of a free program by obtaining a -restrictive license from a patent holder. Therefore, we insist that -any patent license obtained for a version of the library must be -consistent with the full freedom of use specified in this license. - - Most GNU software, including some libraries, is covered by the -ordinary GNU General Public License. This license, the GNU Lesser -General Public License, applies to certain designated libraries, and -is quite different from the ordinary General Public License. We use -this license for certain libraries in order to permit linking those -libraries into non-free programs. - - When a program is linked with a library, whether statically or using -a shared library, the combination of the two is legally speaking a -combined work, a derivative of the original library. The ordinary -General Public License therefore permits such linking only if the -entire combination fits its criteria of freedom. The Lesser General -Public License permits more lax criteria for linking other code with -the library. - - We call this license the "Lesser" General Public License because it -does Less to protect the user's freedom than the ordinary General -Public License. It also provides other free software developers Less -of an advantage over competing non-free programs. These disadvantages -are the reason we use the ordinary General Public License for many -libraries. However, the Lesser license provides advantages in certain -special circumstances. - - For example, on rare occasions, there may be a special need to -encourage the widest possible use of a certain library, so that it becomes -a de-facto standard. To achieve this, non-free programs must be -allowed to use the library. A more frequent case is that a free -library does the same job as widely used non-free libraries. In this -case, there is little to gain by limiting the free library to free -software only, so we use the Lesser General Public License. - - In other cases, permission to use a particular library in non-free -programs enables a greater number of people to use a large body of -free software. For example, permission to use the GNU C Library in -non-free programs enables many more people to use the whole GNU -operating system, as well as its variant, the GNU/Linux operating -system. - - Although the Lesser General Public License is Less protective of the -users' freedom, it does ensure that the user of a program that is -linked with the Library has the freedom and the wherewithal to run -that program using a modified version of the Library. - - The precise terms and conditions for copying, distribution and -modification follow. Pay close attention to the difference between a -"work based on the library" and a "work that uses the library". The -former contains code derived from the library, whereas the latter must -be combined with the library in order to run. - - GNU LESSER GENERAL PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. This License Agreement applies to any software library or other -program which contains a notice placed by the copyright holder or -other authorized party saying it may be distributed under the terms of -this Lesser General Public License (also called "this License"). -Each licensee is addressed as "you". - - A "library" means a collection of software functions and/or data -prepared so as to be conveniently linked with application programs -(which use some of those functions and data) to form executables. - - The "Library", below, refers to any such software library or work -which has been distributed under these terms. A "work based on the -Library" means either the Library or any derivative work under -copyright law: that is to say, a work containing the Library or a -portion of it, either verbatim or with modifications and/or translated -straightforwardly into another language. (Hereinafter, translation is -included without limitation in the term "modification".) - - "Source code" for a work means the preferred form of the work for -making modifications to it. For a library, complete source code means -all the source code for all modules it contains, plus any associated -interface definition files, plus the scripts used to control compilation -and installation of the library. - - Activities other than copying, distribution and modification are not -covered by this License; they are outside its scope. The act of -running a program using the Library is not restricted, and output from -such a program is covered only if its contents constitute a work based -on the Library (independent of the use of the Library in a tool for -writing it). Whether that is true depends on what the Library does -and what the program that uses the Library does. - - 1. You may copy and distribute verbatim copies of the Library's -complete source code as you receive it, in any medium, provided that -you conspicuously and appropriately publish on each copy an -appropriate copyright notice and disclaimer of warranty; keep intact -all the notices that refer to this License and to the absence of any -warranty; and distribute a copy of this License along with the -Library. - - You may charge a fee for the physical act of transferring a copy, -and you may at your option offer warranty protection in exchange for a -fee. - - 2. You may modify your copy or copies of the Library or any portion -of it, thus forming a work based on the Library, and copy and -distribute such modifications or work under the terms of Section 1 -above, provided that you also meet all of these conditions: - - a) The modified work must itself be a software library. - - b) You must cause the files modified to carry prominent notices - stating that you changed the files and the date of any change. - - c) You must cause the whole of the work to be licensed at no - charge to all third parties under the terms of this License. - - d) If a facility in the modified Library refers to a function or a - table of data to be supplied by an application program that uses - the facility, other than as an argument passed when the facility - is invoked, then you must make a good faith effort to ensure that, - in the event an application does not supply such function or - table, the facility still operates, and performs whatever part of - its purpose remains meaningful. - - (For example, a function in a library to compute square roots has - a purpose that is entirely well-defined independent of the - application. Therefore, Subsection 2d requires that any - application-supplied function or table used by this function must - be optional: if the application does not supply it, the square - root function must still compute square roots.) - -These requirements apply to the modified work as a whole. If -identifiable sections of that work are not derived from the Library, -and can be reasonably considered independent and separate works in -themselves, then this License, and its terms, do not apply to those -sections when you distribute them as separate works. But when you -distribute the same sections as part of a whole which is a work based -on the Library, the distribution of the whole must be on the terms of -this License, whose permissions for other licensees extend to the -entire whole, and thus to each and every part regardless of who wrote -it. - -Thus, it is not the intent of this section to claim rights or contest -your rights to work written entirely by you; rather, the intent is to -exercise the right to control the distribution of derivative or -collective works based on the Library. - -In addition, mere aggregation of another work not based on the Library -with the Library (or with a work based on the Library) on a volume of -a storage or distribution medium does not bring the other work under -the scope of this License. - - 3. You may opt to apply the terms of the ordinary GNU General Public -License instead of this License to a given copy of the Library. To do -this, you must alter all the notices that refer to this License, so -that they refer to the ordinary GNU General Public License, version 2, -instead of to this License. (If a newer version than version 2 of the -ordinary GNU General Public License has appeared, then you can specify -that version instead if you wish.) Do not make any other change in -these notices. - - Once this change is made in a given copy, it is irreversible for -that copy, so the ordinary GNU General Public License applies to all -subsequent copies and derivative works made from that copy. - - This option is useful when you wish to copy part of the code of -the Library into a program that is not a library. - - 4. You may copy and distribute the Library (or a portion or -derivative of it, under Section 2) in object code or executable form -under the terms of Sections 1 and 2 above provided that you accompany -it with the complete corresponding machine-readable source code, which -must be distributed under the terms of Sections 1 and 2 above on a -medium customarily used for software interchange. - - If distribution of object code is made by offering access to copy -from a designated place, then offering equivalent access to copy the -source code from the same place satisfies the requirement to -distribute the source code, even though third parties are not -compelled to copy the source along with the object code. - - 5. A program that contains no derivative of any portion of the -Library, but is designed to work with the Library by being compiled or -linked with it, is called a "work that uses the Library". Such a -work, in isolation, is not a derivative work of the Library, and -therefore falls outside the scope of this License. - - However, linking a "work that uses the Library" with the Library -creates an executable that is a derivative of the Library (because it -contains portions of the Library), rather than a "work that uses the -library". The executable is therefore covered by this License. -Section 6 states terms for distribution of such executables. - - When a "work that uses the Library" uses material from a header file -that is part of the Library, the object code for the work may be a -derivative work of the Library even though the source code is not. -Whether this is true is especially significant if the work can be -linked without the Library, or if the work is itself a library. The -threshold for this to be true is not precisely defined by law. - - If such an object file uses only numerical parameters, data -structure layouts and accessors, and small macros and small inline -functions (ten lines or less in length), then the use of the object -file is unrestricted, regardless of whether it is legally a derivative -work. (Executables containing this object code plus portions of the -Library will still fall under Section 6.) - - Otherwise, if the work is a derivative of the Library, you may -distribute the object code for the work under the terms of Section 6. -Any executables containing that work also fall under Section 6, -whether or not they are linked directly with the Library itself. - - 6. As an exception to the Sections above, you may also combine or -link a "work that uses the Library" with the Library to produce a -work containing portions of the Library, and distribute that work -under terms of your choice, provided that the terms permit -modification of the work for the customer's own use and reverse -engineering for debugging such modifications. - - You must give prominent notice with each copy of the work that the -Library is used in it and that the Library and its use are covered by -this License. You must supply a copy of this License. If the work -during execution displays copyright notices, you must include the -copyright notice for the Library among them, as well as a reference -directing the user to the copy of this License. Also, you must do one -of these things: - - a) Accompany the work with the complete corresponding - machine-readable source code for the Library including whatever - changes were used in the work (which must be distributed under - Sections 1 and 2 above); and, if the work is an executable linked - with the Library, with the complete machine-readable "work that - uses the Library", as object code and/or source code, so that the - user can modify the Library and then relink to produce a modified - executable containing the modified Library. (It is understood - that the user who changes the contents of definitions files in the - Library will not necessarily be able to recompile the application - to use the modified definitions.) - - b) Use a suitable shared library mechanism for linking with the - Library. A suitable mechanism is one that (1) uses at run time a - copy of the library already present on the user's computer system, - rather than copying library functions into the executable, and (2) - will operate properly with a modified version of the library, if - the user installs one, as long as the modified version is - interface-compatible with the version that the work was made with. - - c) Accompany the work with a written offer, valid for at - least three years, to give the same user the materials - specified in Subsection 6a, above, for a charge no more - than the cost of performing this distribution. - - d) If distribution of the work is made by offering access to copy - from a designated place, offer equivalent access to copy the above - specified materials from the same place. - - e) Verify that the user has already received a copy of these - materials or that you have already sent this user a copy. - - For an executable, the required form of the "work that uses the -Library" must include any data and utility programs needed for -reproducing the executable from it. However, as a special exception, -the materials to be distributed need not include anything that is -normally distributed (in either source or binary form) with the major -components (compiler, kernel, and so on) of the operating system on -which the executable runs, unless that component itself accompanies -the executable. - - It may happen that this requirement contradicts the license -restrictions of other proprietary libraries that do not normally -accompany the operating system. Such a contradiction means you cannot -use both them and the Library together in an executable that you -distribute. - - 7. You may place library facilities that are a work based on the -Library side-by-side in a single library together with other library -facilities not covered by this License, and distribute such a combined -library, provided that the separate distribution of the work based on -the Library and of the other library facilities is otherwise -permitted, and provided that you do these two things: - - a) Accompany the combined library with a copy of the same work - based on the Library, uncombined with any other library - facilities. This must be distributed under the terms of the - Sections above. - - b) Give prominent notice with the combined library of the fact - that part of it is a work based on the Library, and explaining - where to find the accompanying uncombined form of the same work. - - 8. You may not copy, modify, sublicense, link with, or distribute -the Library except as expressly provided under this License. Any -attempt otherwise to copy, modify, sublicense, link with, or -distribute the Library is void, and will automatically terminate your -rights under this License. However, parties who have received copies, -or rights, from you under this License will not have their licenses -terminated so long as such parties remain in full compliance. - - 9. You are not required to accept this License, since you have not -signed it. However, nothing else grants you permission to modify or -distribute the Library or its derivative works. These actions are -prohibited by law if you do not accept this License. Therefore, by -modifying or distributing the Library (or any work based on the -Library), you indicate your acceptance of this License to do so, and -all its terms and conditions for copying, distributing or modifying -the Library or works based on it. - - 10. Each time you redistribute the Library (or any work based on the -Library), the recipient automatically receives a license from the -original licensor to copy, distribute, link with or modify the Library -subject to these terms and conditions. You may not impose any further -restrictions on the recipients' exercise of the rights granted herein. -You are not responsible for enforcing compliance by third parties with -this License. - - 11. If, as a consequence of a court judgment or allegation of patent -infringement or for any other reason (not limited to patent issues), -conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot -distribute so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you -may not distribute the Library at all. For example, if a patent -license would not permit royalty-free redistribution of the Library by -all those who receive copies directly or indirectly through you, then -the only way you could satisfy both it and this License would be to -refrain entirely from distribution of the Library. - -If any portion of this section is held invalid or unenforceable under any -particular circumstance, the balance of the section is intended to apply, -and the section as a whole is intended to apply in other circumstances. - -It is not the purpose of this section to induce you to infringe any -patents or other property right claims or to contest validity of any -such claims; this section has the sole purpose of protecting the -integrity of the free software distribution system which is -implemented by public license practices. Many people have made -generous contributions to the wide range of software distributed -through that system in reliance on consistent application of that -system; it is up to the author/donor to decide if he or she is willing -to distribute software through any other system and a licensee cannot -impose that choice. - -This section is intended to make thoroughly clear what is believed to -be a consequence of the rest of this License. - - 12. If the distribution and/or use of the Library is restricted in -certain countries either by patents or by copyrighted interfaces, the -original copyright holder who places the Library under this License may add -an explicit geographical distribution limitation excluding those countries, -so that distribution is permitted only in or among countries not thus -excluded. In such case, this License incorporates the limitation as if -written in the body of this License. - - 13. The Free Software Foundation may publish revised and/or new -versions of the Lesser General Public License from time to time. -Such new versions will be similar in spirit to the present version, -but may differ in detail to address new problems or concerns. - -Each version is given a distinguishing version number. If the Library -specifies a version number of this License which applies to it and -"any later version", you have the option of following the terms and -conditions either of that version or of any later version published by -the Free Software Foundation. If the Library does not specify a -license version number, you may choose any version ever published by -the Free Software Foundation. - - 14. If you wish to incorporate parts of the Library into other free -programs whose distribution conditions are incompatible with these, -write to the author to ask for permission. For software which is -copyrighted by the Free Software Foundation, write to the Free -Software Foundation; we sometimes make exceptions for this. Our -decision will be guided by the two goals of preserving the free status -of all derivatives of our free software and of promoting the sharing -and reuse of software generally. - - NO WARRANTY - - 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO -WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. -EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR -OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY -KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE -IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE -LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME -THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN -WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY -AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU -FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR -CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE -LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING -RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A -FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF -SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH -DAMAGES. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Libraries - - If you develop a new library, and you want it to be of the greatest -possible use to the public, we recommend making it free software that -everyone can redistribute and change. You can do so by permitting -redistribution under these terms (or, alternatively, under the terms of the -ordinary General Public License). - - To apply these terms, attach the following notices to the library. It is -safest to attach them to the start of each source file to most effectively -convey the exclusion of warranty; and each file should have at least the -"copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This library is free software; you can redistribute it and/or - modify it under the terms of the GNU Lesser General Public - License as published by the Free Software Foundation; either - version 2.1 of the License, or (at your option) any later version. - - This library is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - Lesser General Public License for more details. - - You should have received a copy of the GNU Lesser General Public - License along with this library; if not, write to the Free Software - Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA - -Also add information on how to contact you by electronic and paper mail. - -You should also get your employer (if you work as a programmer) or your -school, if any, to sign a "copyright disclaimer" for the library, if -necessary. Here is a sample; alter the names: - - Yoyodyne, Inc., hereby disclaims all copyright interest in the - library `Frob' (a library for tweaking knobs) written by James Random Hacker. - - , 1 April 1990 - Ty Coon, President of Vice - -That's all there is to it! diff --git a/code/3rd_glpk/amd/README b/code/3rd_glpk/amd/README deleted file mode 100644 index de950eb4..00000000 --- a/code/3rd_glpk/amd/README +++ /dev/null @@ -1,58 +0,0 @@ -NOTE: Files in this subdirectory are NOT part of the GLPK package, but - are used with GLPK. - - The original code was modified according to GLPK requirements by - Andrew Makhorin . -************************************************************************ -AMD Version 2.2, Copyright (C) 2007 by Timothy A. Davis, -Patrick R. Amestoy, and Iain S. Duff. All Rights Reserved. - -Description: - - AMD is a set of routines for pre-ordering sparse matrices prior to - Cholesky or LU factorization, using the approximate minimum degree - ordering algorithm. Written in ANSI/ISO C with a MATLAB interface, - and in Fortran 77. - -Authors: - - Timothy A. Davis (davis at cise.ufl.edu), University of Florida. - Patrick R. Amestoy, ENSEEIHT, Toulouse, France. - Iain S. Duff, Rutherford Appleton Laboratory, UK. - -AMD License: - - Your use or distribution of AMD or any modified version of AMD - implies that you agree to this License. - - This library is free software; you can redistribute it and/or - modify it under the terms of the GNU Lesser General Public License - as published by the Free Software Foundation; either version 2.1 of - the License, or (at your option) any later version. - - This library is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - Lesser General Public License for more details. - - You should have received a copy of the GNU Lesser General Public - License along with this library; if not, write to the Free Software - Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 - USA. - - Permission is hereby granted to use or copy this program under the - terms of the GNU LGPL, provided that the Copyright, this License, - and the Availability of the original version is retained on all - copies. User documentation of any code that uses this code or any - modified version of this code must cite the Copyright, this License, - the Availability note, and "Used by permission." Permission to - modify the code and to distribute modified code is granted, provided - the Copyright, this License, and the Availability note are retained, - and a notice that the code was modified is included. - - AMD is available under alternate licences; contact T. Davis for - details. - -Availability: - - http://www.cise.ufl.edu/research/sparse/amd diff --git a/code/3rd_glpk/amd/amd.h b/code/3rd_glpk/amd/amd.h deleted file mode 100644 index be662d95..00000000 --- a/code/3rd_glpk/amd/amd.h +++ /dev/null @@ -1,67 +0,0 @@ -/* amd.h */ - -/* Written by Andrew Makhorin . */ - -#ifndef GLPAMD_H -#define GLPAMD_H - -#define AMD_DATE "May 31, 2007" -#define AMD_VERSION_CODE(main, sub) ((main) * 1000 + (sub)) -#define AMD_MAIN_VERSION 2 -#define AMD_SUB_VERSION 2 -#define AMD_SUBSUB_VERSION 0 -#define AMD_VERSION AMD_VERSION_CODE(AMD_MAIN_VERSION, AMD_SUB_VERSION) - -#define AMD_CONTROL 5 -#define AMD_INFO 20 - -#define AMD_DENSE 0 -#define AMD_AGGRESSIVE 1 - -#define AMD_DEFAULT_DENSE 10.0 -#define AMD_DEFAULT_AGGRESSIVE 1 - -#define AMD_STATUS 0 -#define AMD_N 1 -#define AMD_NZ 2 -#define AMD_SYMMETRY 3 -#define AMD_NZDIAG 4 -#define AMD_NZ_A_PLUS_AT 5 -#define AMD_NDENSE 6 -#define AMD_MEMORY 7 -#define AMD_NCMPA 8 -#define AMD_LNZ 9 -#define AMD_NDIV 10 -#define AMD_NMULTSUBS_LDL 11 -#define AMD_NMULTSUBS_LU 12 -#define AMD_DMAX 13 - -#define AMD_OK 0 -#define AMD_OUT_OF_MEMORY (-1) -#define AMD_INVALID (-2) -#define AMD_OK_BUT_JUMBLED 1 - -#define amd_order _glp_amd_order -int amd_order(int n, const int Ap[], const int Ai[], int P[], - double Control[], double Info[]); - -#define amd_2 _glp_amd_2 -void amd_2(int n, int Pe[], int Iw[], int Len[], int iwlen, int pfree, - int Nv[], int Next[], int Last[], int Head[], int Elen[], - int Degree[], int W[], double Control[], double Info[]); - -#define amd_valid _glp_amd_valid -int amd_valid(int n_row, int n_col, const int Ap[], const int Ai[]); - -#define amd_defaults _glp_amd_defaults -void amd_defaults(double Control[]); - -#define amd_control _glp_amd_control -void amd_control(double Control[]); - -#define amd_info _glp_amd_info -void amd_info(double Info[]); - -#endif - -/* eof */ diff --git a/code/3rd_glpk/amd/amd_1.c b/code/3rd_glpk/amd/amd_1.c deleted file mode 100644 index 4f9b07d7..00000000 --- a/code/3rd_glpk/amd/amd_1.c +++ /dev/null @@ -1,181 +0,0 @@ -/* ========================================================================= */ -/* === AMD_1 =============================================================== */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* AMD_1: Construct A+A' for a sparse matrix A and perform the AMD ordering. - * - * The n-by-n sparse matrix A can be unsymmetric. It is stored in MATLAB-style - * compressed-column form, with sorted row indices in each column, and no - * duplicate entries. Diagonal entries may be present, but they are ignored. - * Row indices of column j of A are stored in Ai [Ap [j] ... Ap [j+1]-1]. - * Ap [0] must be zero, and nz = Ap [n] is the number of entries in A. The - * size of the matrix, n, must be greater than or equal to zero. - * - * This routine must be preceded by a call to AMD_aat, which computes the - * number of entries in each row/column in A+A', excluding the diagonal. - * Len [j], on input, is the number of entries in row/column j of A+A'. This - * routine constructs the matrix A+A' and then calls AMD_2. No error checking - * is performed (this was done in AMD_valid). - */ - -#include "amd_internal.h" - -GLOBAL void AMD_1 -( - Int n, /* n > 0 */ - const Int Ap [ ], /* input of size n+1, not modified */ - const Int Ai [ ], /* input of size nz = Ap [n], not modified */ - Int P [ ], /* size n output permutation */ - Int Pinv [ ], /* size n output inverse permutation */ - Int Len [ ], /* size n input, undefined on output */ - Int slen, /* slen >= sum (Len [0..n-1]) + 7n, - * ideally slen = 1.2 * sum (Len) + 8n */ - Int S [ ], /* size slen workspace */ - double Control [ ], /* input array of size AMD_CONTROL */ - double Info [ ] /* output array of size AMD_INFO */ -) -{ - Int i, j, k, p, pfree, iwlen, pj, p1, p2, pj2, *Iw, *Pe, *Nv, *Head, - *Elen, *Degree, *s, *W, *Sp, *Tp ; - - /* --------------------------------------------------------------------- */ - /* construct the matrix for AMD_2 */ - /* --------------------------------------------------------------------- */ - - ASSERT (n > 0) ; - - iwlen = slen - 6*n ; - s = S ; - Pe = s ; s += n ; - Nv = s ; s += n ; - Head = s ; s += n ; - Elen = s ; s += n ; - Degree = s ; s += n ; - W = s ; s += n ; - Iw = s ; s += iwlen ; - - ASSERT (AMD_valid (n, n, Ap, Ai) == AMD_OK) ; - - /* construct the pointers for A+A' */ - Sp = Nv ; /* use Nv and W as workspace for Sp and Tp [ */ - Tp = W ; - pfree = 0 ; - for (j = 0 ; j < n ; j++) - { - Pe [j] = pfree ; - Sp [j] = pfree ; - pfree += Len [j] ; - } - - /* Note that this restriction on iwlen is slightly more restrictive than - * what is strictly required in AMD_2. AMD_2 can operate with no elbow - * room at all, but it will be very slow. For better performance, at - * least size-n elbow room is enforced. */ - ASSERT (iwlen >= pfree + n) ; - -#ifndef NDEBUG - for (p = 0 ; p < iwlen ; p++) Iw [p] = EMPTY ; -#endif - - for (k = 0 ; k < n ; k++) - { - AMD_DEBUG1 (("Construct row/column k= "ID" of A+A'\n", k)) ; - p1 = Ap [k] ; - p2 = Ap [k+1] ; - - /* construct A+A' */ - for (p = p1 ; p < p2 ; ) - { - /* scan the upper triangular part of A */ - j = Ai [p] ; - ASSERT (j >= 0 && j < n) ; - if (j < k) - { - /* entry A (j,k) in the strictly upper triangular part */ - ASSERT (Sp [j] < (j == n-1 ? pfree : Pe [j+1])) ; - ASSERT (Sp [k] < (k == n-1 ? pfree : Pe [k+1])) ; - Iw [Sp [j]++] = k ; - Iw [Sp [k]++] = j ; - p++ ; - } - else if (j == k) - { - /* skip the diagonal */ - p++ ; - break ; - } - else /* j > k */ - { - /* first entry below the diagonal */ - break ; - } - /* scan lower triangular part of A, in column j until reaching - * row k. Start where last scan left off. */ - ASSERT (Ap [j] <= Tp [j] && Tp [j] <= Ap [j+1]) ; - pj2 = Ap [j+1] ; - for (pj = Tp [j] ; pj < pj2 ; ) - { - i = Ai [pj] ; - ASSERT (i >= 0 && i < n) ; - if (i < k) - { - /* A (i,j) is only in the lower part, not in upper */ - ASSERT (Sp [i] < (i == n-1 ? pfree : Pe [i+1])) ; - ASSERT (Sp [j] < (j == n-1 ? pfree : Pe [j+1])) ; - Iw [Sp [i]++] = j ; - Iw [Sp [j]++] = i ; - pj++ ; - } - else if (i == k) - { - /* entry A (k,j) in lower part and A (j,k) in upper */ - pj++ ; - break ; - } - else /* i > k */ - { - /* consider this entry later, when k advances to i */ - break ; - } - } - Tp [j] = pj ; - } - Tp [k] = p ; - } - - /* clean up, for remaining mismatched entries */ - for (j = 0 ; j < n ; j++) - { - for (pj = Tp [j] ; pj < Ap [j+1] ; pj++) - { - i = Ai [pj] ; - ASSERT (i >= 0 && i < n) ; - /* A (i,j) is only in the lower part, not in upper */ - ASSERT (Sp [i] < (i == n-1 ? pfree : Pe [i+1])) ; - ASSERT (Sp [j] < (j == n-1 ? pfree : Pe [j+1])) ; - Iw [Sp [i]++] = j ; - Iw [Sp [j]++] = i ; - } - } - -#ifndef NDEBUG - for (j = 0 ; j < n-1 ; j++) ASSERT (Sp [j] == Pe [j+1]) ; - ASSERT (Sp [n-1] == pfree) ; -#endif - - /* Tp and Sp no longer needed ] */ - - /* --------------------------------------------------------------------- */ - /* order the matrix */ - /* --------------------------------------------------------------------- */ - - AMD_2 (n, Pe, Iw, Len, iwlen, pfree, - Nv, Pinv, P, Head, Elen, Degree, W, Control, Info) ; -} diff --git a/code/3rd_glpk/amd/amd_2.c b/code/3rd_glpk/amd/amd_2.c deleted file mode 100644 index 36ae828a..00000000 --- a/code/3rd_glpk/amd/amd_2.c +++ /dev/null @@ -1,1842 +0,0 @@ -/* ========================================================================= */ -/* === AMD_2 =============================================================== */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* AMD_2: performs the AMD ordering on a symmetric sparse matrix A, followed - * by a postordering (via depth-first search) of the assembly tree using the - * AMD_postorder routine. - */ - -#include "amd_internal.h" - -/* ========================================================================= */ -/* === clear_flag ========================================================== */ -/* ========================================================================= */ - -static Int clear_flag (Int wflg, Int wbig, Int W [ ], Int n) -{ - Int x ; - if (wflg < 2 || wflg >= wbig) - { - for (x = 0 ; x < n ; x++) - { - if (W [x] != 0) W [x] = 1 ; - } - wflg = 2 ; - } - /* at this point, W [0..n-1] < wflg holds */ - return (wflg) ; -} - - -/* ========================================================================= */ -/* === AMD_2 =============================================================== */ -/* ========================================================================= */ - -GLOBAL void AMD_2 -( - Int n, /* A is n-by-n, where n > 0 */ - Int Pe [ ], /* Pe [0..n-1]: index in Iw of row i on input */ - Int Iw [ ], /* workspace of size iwlen. Iw [0..pfree-1] - * holds the matrix on input */ - Int Len [ ], /* Len [0..n-1]: length for row/column i on input */ - Int iwlen, /* length of Iw. iwlen >= pfree + n */ - Int pfree, /* Iw [pfree ... iwlen-1] is empty on input */ - - /* 7 size-n workspaces, not defined on input: */ - Int Nv [ ], /* the size of each supernode on output */ - Int Next [ ], /* the output inverse permutation */ - Int Last [ ], /* the output permutation */ - Int Head [ ], - Int Elen [ ], /* the size columns of L for each supernode */ - Int Degree [ ], - Int W [ ], - - /* control parameters and output statistics */ - double Control [ ], /* array of size AMD_CONTROL */ - double Info [ ] /* array of size AMD_INFO */ -) -{ - -/* - * Given a representation of the nonzero pattern of a symmetric matrix, A, - * (excluding the diagonal) perform an approximate minimum (UMFPACK/MA38-style) - * degree ordering to compute a pivot order such that the introduction of - * nonzeros (fill-in) in the Cholesky factors A = LL' is kept low. At each - * step, the pivot selected is the one with the minimum UMFAPACK/MA38-style - * upper-bound on the external degree. This routine can optionally perform - * aggresive absorption (as done by MC47B in the Harwell Subroutine - * Library). - * - * The approximate degree algorithm implemented here is the symmetric analog of - * the degree update algorithm in MA38 and UMFPACK (the Unsymmetric-pattern - * MultiFrontal PACKage, both by Davis and Duff). The routine is based on the - * MA27 minimum degree ordering algorithm by Iain Duff and John Reid. - * - * This routine is a translation of the original AMDBAR and MC47B routines, - * in Fortran, with the following modifications: - * - * (1) dense rows/columns are removed prior to ordering the matrix, and placed - * last in the output order. The presence of a dense row/column can - * increase the ordering time by up to O(n^2), unless they are removed - * prior to ordering. - * - * (2) the minimum degree ordering is followed by a postordering (depth-first - * search) of the assembly tree. Note that mass elimination (discussed - * below) combined with the approximate degree update can lead to the mass - * elimination of nodes with lower exact degree than the current pivot - * element. No additional fill-in is caused in the representation of the - * Schur complement. The mass-eliminated nodes merge with the current - * pivot element. They are ordered prior to the current pivot element. - * Because they can have lower exact degree than the current element, the - * merger of two or more of these nodes in the current pivot element can - * lead to a single element that is not a "fundamental supernode". The - * diagonal block can have zeros in it. Thus, the assembly tree used here - * is not guaranteed to be the precise supernodal elemination tree (with - * "funadmental" supernodes), and the postordering performed by this - * routine is not guaranteed to be a precise postordering of the - * elimination tree. - * - * (3) input parameters are added, to control aggressive absorption and the - * detection of "dense" rows/columns of A. - * - * (4) additional statistical information is returned, such as the number of - * nonzeros in L, and the flop counts for subsequent LDL' and LU - * factorizations. These are slight upper bounds, because of the mass - * elimination issue discussed above. - * - * (5) additional routines are added to interface this routine to MATLAB - * to provide a simple C-callable user-interface, to check inputs for - * errors, compute the symmetry of the pattern of A and the number of - * nonzeros in each row/column of A+A', to compute the pattern of A+A', - * to perform the assembly tree postordering, and to provide debugging - * ouput. Many of these functions are also provided by the Fortran - * Harwell Subroutine Library routine MC47A. - * - * (6) both int and UF_long versions are provided. In the descriptions below - * and integer is and int or UF_long depending on which version is - * being used. - - ********************************************************************** - ***** CAUTION: ARGUMENTS ARE NOT CHECKED FOR ERRORS ON INPUT. ****** - ********************************************************************** - ** If you want error checking, a more versatile input format, and a ** - ** simpler user interface, use amd_order or amd_l_order instead. ** - ** This routine is not meant to be user-callable. ** - ********************************************************************** - - * ---------------------------------------------------------------------------- - * References: - * ---------------------------------------------------------------------------- - * - * [1] Timothy A. Davis and Iain Duff, "An unsymmetric-pattern multifrontal - * method for sparse LU factorization", SIAM J. Matrix Analysis and - * Applications, vol. 18, no. 1, pp. 140-158. Discusses UMFPACK / MA38, - * which first introduced the approximate minimum degree used by this - * routine. - * - * [2] Patrick Amestoy, Timothy A. Davis, and Iain S. Duff, "An approximate - * minimum degree ordering algorithm," SIAM J. Matrix Analysis and - * Applications, vol. 17, no. 4, pp. 886-905, 1996. Discusses AMDBAR and - * MC47B, which are the Fortran versions of this routine. - * - * [3] Alan George and Joseph Liu, "The evolution of the minimum degree - * ordering algorithm," SIAM Review, vol. 31, no. 1, pp. 1-19, 1989. - * We list below the features mentioned in that paper that this code - * includes: - * - * mass elimination: - * Yes. MA27 relied on supervariable detection for mass elimination. - * - * indistinguishable nodes: - * Yes (we call these "supervariables"). This was also in the MA27 - * code - although we modified the method of detecting them (the - * previous hash was the true degree, which we no longer keep track - * of). A supervariable is a set of rows with identical nonzero - * pattern. All variables in a supervariable are eliminated together. - * Each supervariable has as its numerical name that of one of its - * variables (its principal variable). - * - * quotient graph representation: - * Yes. We use the term "element" for the cliques formed during - * elimination. This was also in the MA27 code. The algorithm can - * operate in place, but it will work more efficiently if given some - * "elbow room." - * - * element absorption: - * Yes. This was also in the MA27 code. - * - * external degree: - * Yes. The MA27 code was based on the true degree. - * - * incomplete degree update and multiple elimination: - * No. This was not in MA27, either. Our method of degree update - * within MC47B is element-based, not variable-based. It is thus - * not well-suited for use with incomplete degree update or multiple - * elimination. - * - * Authors, and Copyright (C) 2004 by: - * Timothy A. Davis, Patrick Amestoy, Iain S. Duff, John K. Reid. - * - * Acknowledgements: This work (and the UMFPACK package) was supported by the - * National Science Foundation (ASC-9111263, DMS-9223088, and CCR-0203270). - * The UMFPACK/MA38 approximate degree update algorithm, the unsymmetric analog - * which forms the basis of AMD, was developed while Tim Davis was supported by - * CERFACS (Toulouse, France) in a post-doctoral position. This C version, and - * the etree postorder, were written while Tim Davis was on sabbatical at - * Stanford University and Lawrence Berkeley National Laboratory. - - * ---------------------------------------------------------------------------- - * INPUT ARGUMENTS (unaltered): - * ---------------------------------------------------------------------------- - - * n: The matrix order. Restriction: n >= 1. - * - * iwlen: The size of the Iw array. On input, the matrix is stored in - * Iw [0..pfree-1]. However, Iw [0..iwlen-1] should be slightly larger - * than what is required to hold the matrix, at least iwlen >= pfree + n. - * Otherwise, excessive compressions will take place. The recommended - * value of iwlen is 1.2 * pfree + n, which is the value used in the - * user-callable interface to this routine (amd_order.c). The algorithm - * will not run at all if iwlen < pfree. Restriction: iwlen >= pfree + n. - * Note that this is slightly more restrictive than the actual minimum - * (iwlen >= pfree), but AMD_2 will be very slow with no elbow room. - * Thus, this routine enforces a bare minimum elbow room of size n. - * - * pfree: On input the tail end of the array, Iw [pfree..iwlen-1], is empty, - * and the matrix is stored in Iw [0..pfree-1]. During execution, - * additional data is placed in Iw, and pfree is modified so that - * Iw [pfree..iwlen-1] is always the unused part of Iw. - * - * Control: A double array of size AMD_CONTROL containing input parameters - * that affect how the ordering is computed. If NULL, then default - * settings are used. - * - * Control [AMD_DENSE] is used to determine whether or not a given input - * row is "dense". A row is "dense" if the number of entries in the row - * exceeds Control [AMD_DENSE] times sqrt (n), except that rows with 16 or - * fewer entries are never considered "dense". To turn off the detection - * of dense rows, set Control [AMD_DENSE] to a negative number, or to a - * number larger than sqrt (n). The default value of Control [AMD_DENSE] - * is AMD_DEFAULT_DENSE, which is defined in amd.h as 10. - * - * Control [AMD_AGGRESSIVE] is used to determine whether or not aggressive - * absorption is to be performed. If nonzero, then aggressive absorption - * is performed (this is the default). - - * ---------------------------------------------------------------------------- - * INPUT/OUPUT ARGUMENTS: - * ---------------------------------------------------------------------------- - * - * Pe: An integer array of size n. On input, Pe [i] is the index in Iw of - * the start of row i. Pe [i] is ignored if row i has no off-diagonal - * entries. Thus Pe [i] must be in the range 0 to pfree-1 for non-empty - * rows. - * - * During execution, it is used for both supervariables and elements: - * - * Principal supervariable i: index into Iw of the description of - * supervariable i. A supervariable represents one or more rows of - * the matrix with identical nonzero pattern. In this case, - * Pe [i] >= 0. - * - * Non-principal supervariable i: if i has been absorbed into another - * supervariable j, then Pe [i] = FLIP (j), where FLIP (j) is defined - * as (-(j)-2). Row j has the same pattern as row i. Note that j - * might later be absorbed into another supervariable j2, in which - * case Pe [i] is still FLIP (j), and Pe [j] = FLIP (j2) which is - * < EMPTY, where EMPTY is defined as (-1) in amd_internal.h. - * - * Unabsorbed element e: the index into Iw of the description of element - * e, if e has not yet been absorbed by a subsequent element. Element - * e is created when the supervariable of the same name is selected as - * the pivot. In this case, Pe [i] >= 0. - * - * Absorbed element e: if element e is absorbed into element e2, then - * Pe [e] = FLIP (e2). This occurs when the pattern of e (which we - * refer to as Le) is found to be a subset of the pattern of e2 (that - * is, Le2). In this case, Pe [i] < EMPTY. If element e is "null" - * (it has no nonzeros outside its pivot block), then Pe [e] = EMPTY, - * and e is the root of an assembly subtree (or the whole tree if - * there is just one such root). - * - * Dense variable i: if i is "dense", then Pe [i] = EMPTY. - * - * On output, Pe holds the assembly tree/forest, which implicitly - * represents a pivot order with identical fill-in as the actual order - * (via a depth-first search of the tree), as follows. If Nv [i] > 0, - * then i represents a node in the assembly tree, and the parent of i is - * Pe [i], or EMPTY if i is a root. If Nv [i] = 0, then (i, Pe [i]) - * represents an edge in a subtree, the root of which is a node in the - * assembly tree. Note that i refers to a row/column in the original - * matrix, not the permuted matrix. - * - * Info: A double array of size AMD_INFO. If present, (that is, not NULL), - * then statistics about the ordering are returned in the Info array. - * See amd.h for a description. - - * ---------------------------------------------------------------------------- - * INPUT/MODIFIED (undefined on output): - * ---------------------------------------------------------------------------- - * - * Len: An integer array of size n. On input, Len [i] holds the number of - * entries in row i of the matrix, excluding the diagonal. The contents - * of Len are undefined on output. - * - * Iw: An integer array of size iwlen. On input, Iw [0..pfree-1] holds the - * description of each row i in the matrix. The matrix must be symmetric, - * and both upper and lower triangular parts must be present. The - * diagonal must not be present. Row i is held as follows: - * - * Len [i]: the length of the row i data structure in the Iw array. - * Iw [Pe [i] ... Pe [i] + Len [i] - 1]: - * the list of column indices for nonzeros in row i (simple - * supervariables), excluding the diagonal. All supervariables - * start with one row/column each (supervariable i is just row i). - * If Len [i] is zero on input, then Pe [i] is ignored on input. - * - * Note that the rows need not be in any particular order, and there - * may be empty space between the rows. - * - * During execution, the supervariable i experiences fill-in. This is - * represented by placing in i a list of the elements that cause fill-in - * in supervariable i: - * - * Len [i]: the length of supervariable i in the Iw array. - * Iw [Pe [i] ... Pe [i] + Elen [i] - 1]: - * the list of elements that contain i. This list is kept short - * by removing absorbed elements. - * Iw [Pe [i] + Elen [i] ... Pe [i] + Len [i] - 1]: - * the list of supervariables in i. This list is kept short by - * removing nonprincipal variables, and any entry j that is also - * contained in at least one of the elements (j in Le) in the list - * for i (e in row i). - * - * When supervariable i is selected as pivot, we create an element e of - * the same name (e=i): - * - * Len [e]: the length of element e in the Iw array. - * Iw [Pe [e] ... Pe [e] + Len [e] - 1]: - * the list of supervariables in element e. - * - * An element represents the fill-in that occurs when supervariable i is - * selected as pivot (which represents the selection of row i and all - * non-principal variables whose principal variable is i). We use the - * term Le to denote the set of all supervariables in element e. Absorbed - * supervariables and elements are pruned from these lists when - * computationally convenient. - * - * CAUTION: THE INPUT MATRIX IS OVERWRITTEN DURING COMPUTATION. - * The contents of Iw are undefined on output. - - * ---------------------------------------------------------------------------- - * OUTPUT (need not be set on input): - * ---------------------------------------------------------------------------- - * - * Nv: An integer array of size n. During execution, ABS (Nv [i]) is equal to - * the number of rows that are represented by the principal supervariable - * i. If i is a nonprincipal or dense variable, then Nv [i] = 0. - * Initially, Nv [i] = 1 for all i. Nv [i] < 0 signifies that i is a - * principal variable in the pattern Lme of the current pivot element me. - * After element me is constructed, Nv [i] is set back to a positive - * value. - * - * On output, Nv [i] holds the number of pivots represented by super - * row/column i of the original matrix, or Nv [i] = 0 for non-principal - * rows/columns. Note that i refers to a row/column in the original - * matrix, not the permuted matrix. - * - * Elen: An integer array of size n. See the description of Iw above. At the - * start of execution, Elen [i] is set to zero for all rows i. During - * execution, Elen [i] is the number of elements in the list for - * supervariable i. When e becomes an element, Elen [e] = FLIP (esize) is - * set, where esize is the size of the element (the number of pivots, plus - * the number of nonpivotal entries). Thus Elen [e] < EMPTY. - * Elen (i) = EMPTY set when variable i becomes nonprincipal. - * - * For variables, Elen (i) >= EMPTY holds until just before the - * postordering and permutation vectors are computed. For elements, - * Elen [e] < EMPTY holds. - * - * On output, Elen [i] is the degree of the row/column in the Cholesky - * factorization of the permuted matrix, corresponding to the original row - * i, if i is a super row/column. It is equal to EMPTY if i is - * non-principal. Note that i refers to a row/column in the original - * matrix, not the permuted matrix. - * - * Note that the contents of Elen on output differ from the Fortran - * version (Elen holds the inverse permutation in the Fortran version, - * which is instead returned in the Next array in this C version, - * described below). - * - * Last: In a degree list, Last [i] is the supervariable preceding i, or EMPTY - * if i is the head of the list. In a hash bucket, Last [i] is the hash - * key for i. - * - * Last [Head [hash]] is also used as the head of a hash bucket if - * Head [hash] contains a degree list (see the description of Head, - * below). - * - * On output, Last [0..n-1] holds the permutation. That is, if - * i = Last [k], then row i is the kth pivot row (where k ranges from 0 to - * n-1). Row Last [k] of A is the kth row in the permuted matrix, PAP'. - * - * Next: Next [i] is the supervariable following i in a link list, or EMPTY if - * i is the last in the list. Used for two kinds of lists: degree lists - * and hash buckets (a supervariable can be in only one kind of list at a - * time). - * - * On output Next [0..n-1] holds the inverse permutation. That is, if - * k = Next [i], then row i is the kth pivot row. Row i of A appears as - * the (Next[i])-th row in the permuted matrix, PAP'. - * - * Note that the contents of Next on output differ from the Fortran - * version (Next is undefined on output in the Fortran version). - - * ---------------------------------------------------------------------------- - * LOCAL WORKSPACE (not input or output - used only during execution): - * ---------------------------------------------------------------------------- - * - * Degree: An integer array of size n. If i is a supervariable, then - * Degree [i] holds the current approximation of the external degree of - * row i (an upper bound). The external degree is the number of nonzeros - * in row i, minus ABS (Nv [i]), the diagonal part. The bound is equal to - * the exact external degree if Elen [i] is less than or equal to two. - * - * We also use the term "external degree" for elements e to refer to - * |Le \ Lme|. If e is an element, then Degree [e] is |Le|, which is the - * degree of the off-diagonal part of the element e (not including the - * diagonal part). - * - * Head: An integer array of size n. Head is used for degree lists. - * Head [deg] is the first supervariable in a degree list. All - * supervariables i in a degree list Head [deg] have the same approximate - * degree, namely, deg = Degree [i]. If the list Head [deg] is empty then - * Head [deg] = EMPTY. - * - * During supervariable detection Head [hash] also serves as a pointer to - * a hash bucket. If Head [hash] >= 0, there is a degree list of degree - * hash. The hash bucket head pointer is Last [Head [hash]]. If - * Head [hash] = EMPTY, then the degree list and hash bucket are both - * empty. If Head [hash] < EMPTY, then the degree list is empty, and - * FLIP (Head [hash]) is the head of the hash bucket. After supervariable - * detection is complete, all hash buckets are empty, and the - * (Last [Head [hash]] = EMPTY) condition is restored for the non-empty - * degree lists. - * - * W: An integer array of size n. The flag array W determines the status of - * elements and variables, and the external degree of elements. - * - * for elements: - * if W [e] = 0, then the element e is absorbed. - * if W [e] >= wflg, then W [e] - wflg is the size of the set - * |Le \ Lme|, in terms of nonzeros (the sum of ABS (Nv [i]) for - * each principal variable i that is both in the pattern of - * element e and NOT in the pattern of the current pivot element, - * me). - * if wflg > W [e] > 0, then e is not absorbed and has not yet been - * seen in the scan of the element lists in the computation of - * |Le\Lme| in Scan 1 below. - * - * for variables: - * during supervariable detection, if W [j] != wflg then j is - * not in the pattern of variable i. - * - * The W array is initialized by setting W [i] = 1 for all i, and by - * setting wflg = 2. It is reinitialized if wflg becomes too large (to - * ensure that wflg+n does not cause integer overflow). - - * ---------------------------------------------------------------------------- - * LOCAL INTEGERS: - * ---------------------------------------------------------------------------- - */ - - Int deg, degme, dext, lemax, e, elenme, eln, i, ilast, inext, j, - jlast, jnext, k, knt1, knt2, knt3, lenj, ln, me, mindeg, nel, nleft, - nvi, nvj, nvpiv, slenme, wbig, we, wflg, wnvi, ok, ndense, ncmpa, - dense, aggressive ; - - unsigned Int hash ; /* unsigned, so that hash % n is well defined.*/ - -/* - * deg: the degree of a variable or element - * degme: size, |Lme|, of the current element, me (= Degree [me]) - * dext: external degree, |Le \ Lme|, of some element e - * lemax: largest |Le| seen so far (called dmax in Fortran version) - * e: an element - * elenme: the length, Elen [me], of element list of pivotal variable - * eln: the length, Elen [...], of an element list - * hash: the computed value of the hash function - * i: a supervariable - * ilast: the entry in a link list preceding i - * inext: the entry in a link list following i - * j: a supervariable - * jlast: the entry in a link list preceding j - * jnext: the entry in a link list, or path, following j - * k: the pivot order of an element or variable - * knt1: loop counter used during element construction - * knt2: loop counter used during element construction - * knt3: loop counter used during compression - * lenj: Len [j] - * ln: length of a supervariable list - * me: current supervariable being eliminated, and the current - * element created by eliminating that supervariable - * mindeg: current minimum degree - * nel: number of pivots selected so far - * nleft: n - nel, the number of nonpivotal rows/columns remaining - * nvi: the number of variables in a supervariable i (= Nv [i]) - * nvj: the number of variables in a supervariable j (= Nv [j]) - * nvpiv: number of pivots in current element - * slenme: number of variables in variable list of pivotal variable - * wbig: = INT_MAX - n for the int version, UF_long_max - n for the - * UF_long version. wflg is not allowed to be >= wbig. - * we: W [e] - * wflg: used for flagging the W array. See description of Iw. - * wnvi: wflg - Nv [i] - * x: either a supervariable or an element - * - * ok: true if supervariable j can be absorbed into i - * ndense: number of "dense" rows/columns - * dense: rows/columns with initial degree > dense are considered "dense" - * aggressive: true if aggressive absorption is being performed - * ncmpa: number of garbage collections - - * ---------------------------------------------------------------------------- - * LOCAL DOUBLES, used for statistical output only (except for alpha): - * ---------------------------------------------------------------------------- - */ - - double f, r, ndiv, s, nms_lu, nms_ldl, dmax, alpha, lnz, lnzme ; - -/* - * f: nvpiv - * r: degme + nvpiv - * ndiv: number of divisions for LU or LDL' factorizations - * s: number of multiply-subtract pairs for LU factorization, for the - * current element me - * nms_lu number of multiply-subtract pairs for LU factorization - * nms_ldl number of multiply-subtract pairs for LDL' factorization - * dmax: the largest number of entries in any column of L, including the - * diagonal - * alpha: "dense" degree ratio - * lnz: the number of nonzeros in L (excluding the diagonal) - * lnzme: the number of nonzeros in L (excl. the diagonal) for the - * current element me - - * ---------------------------------------------------------------------------- - * LOCAL "POINTERS" (indices into the Iw array) - * ---------------------------------------------------------------------------- -*/ - - Int p, p1, p2, p3, p4, pdst, pend, pj, pme, pme1, pme2, pn, psrc ; - -/* - * Any parameter (Pe [...] or pfree) or local variable starting with "p" (for - * Pointer) is an index into Iw, and all indices into Iw use variables starting - * with "p." The only exception to this rule is the iwlen input argument. - * - * p: pointer into lots of things - * p1: Pe [i] for some variable i (start of element list) - * p2: Pe [i] + Elen [i] - 1 for some variable i - * p3: index of first supervariable in clean list - * p4: - * pdst: destination pointer, for compression - * pend: end of memory to compress - * pj: pointer into an element or variable - * pme: pointer into the current element (pme1...pme2) - * pme1: the current element, me, is stored in Iw [pme1...pme2] - * pme2: the end of the current element - * pn: pointer into a "clean" variable, also used to compress - * psrc: source pointer, for compression -*/ - -/* ========================================================================= */ -/* INITIALIZATIONS */ -/* ========================================================================= */ - - /* Note that this restriction on iwlen is slightly more restrictive than - * what is actually required in AMD_2. AMD_2 can operate with no elbow - * room at all, but it will be slow. For better performance, at least - * size-n elbow room is enforced. */ - ASSERT (iwlen >= pfree + n) ; - ASSERT (n > 0) ; - - /* initialize output statistics */ - lnz = 0 ; - ndiv = 0 ; - nms_lu = 0 ; - nms_ldl = 0 ; - dmax = 1 ; - me = EMPTY ; - - mindeg = 0 ; - ncmpa = 0 ; - nel = 0 ; - lemax = 0 ; - - /* get control parameters */ - if (Control != (double *) NULL) - { - alpha = Control [AMD_DENSE] ; - aggressive = (Control [AMD_AGGRESSIVE] != 0) ; - } - else - { - alpha = AMD_DEFAULT_DENSE ; - aggressive = AMD_DEFAULT_AGGRESSIVE ; - } - /* Note: if alpha is NaN, this is undefined: */ - if (alpha < 0) - { - /* only remove completely dense rows/columns */ - dense = n-2 ; - } - else - { - dense = alpha * sqrt ((double) n) ; - } - dense = MAX (16, dense) ; - dense = MIN (n, dense) ; - AMD_DEBUG1 (("\n\nAMD (debug), alpha %g, aggr. "ID"\n", - alpha, aggressive)) ; - - for (i = 0 ; i < n ; i++) - { - Last [i] = EMPTY ; - Head [i] = EMPTY ; - Next [i] = EMPTY ; - /* if separate Hhead array is used for hash buckets: * - Hhead [i] = EMPTY ; - */ - Nv [i] = 1 ; - W [i] = 1 ; - Elen [i] = 0 ; - Degree [i] = Len [i] ; - } - -#ifndef NDEBUG - AMD_DEBUG1 (("\n======Nel "ID" initial\n", nel)) ; - AMD_dump (n, Pe, Iw, Len, iwlen, pfree, Nv, Next, Last, - Head, Elen, Degree, W, -1) ; -#endif - - /* initialize wflg */ - wbig = Int_MAX - n ; - wflg = clear_flag (0, wbig, W, n) ; - - /* --------------------------------------------------------------------- */ - /* initialize degree lists and eliminate dense and empty rows */ - /* --------------------------------------------------------------------- */ - - ndense = 0 ; - - for (i = 0 ; i < n ; i++) - { - deg = Degree [i] ; - ASSERT (deg >= 0 && deg < n) ; - if (deg == 0) - { - - /* ------------------------------------------------------------- - * we have a variable that can be eliminated at once because - * there is no off-diagonal non-zero in its row. Note that - * Nv [i] = 1 for an empty variable i. It is treated just - * the same as an eliminated element i. - * ------------------------------------------------------------- */ - - Elen [i] = FLIP (1) ; - nel++ ; - Pe [i] = EMPTY ; - W [i] = 0 ; - - } - else if (deg > dense) - { - - /* ------------------------------------------------------------- - * Dense variables are not treated as elements, but as unordered, - * non-principal variables that have no parent. They do not take - * part in the postorder, since Nv [i] = 0. Note that the Fortran - * version does not have this option. - * ------------------------------------------------------------- */ - - AMD_DEBUG1 (("Dense node "ID" degree "ID"\n", i, deg)) ; - ndense++ ; - Nv [i] = 0 ; /* do not postorder this node */ - Elen [i] = EMPTY ; - nel++ ; - Pe [i] = EMPTY ; - - } - else - { - - /* ------------------------------------------------------------- - * place i in the degree list corresponding to its degree - * ------------------------------------------------------------- */ - - inext = Head [deg] ; - ASSERT (inext >= EMPTY && inext < n) ; - if (inext != EMPTY) Last [inext] = i ; - Next [i] = inext ; - Head [deg] = i ; - - } - } - -/* ========================================================================= */ -/* WHILE (selecting pivots) DO */ -/* ========================================================================= */ - - while (nel < n) - { - -#ifndef NDEBUG - AMD_DEBUG1 (("\n======Nel "ID"\n", nel)) ; - if (AMD_debug >= 2) - { - AMD_dump (n, Pe, Iw, Len, iwlen, pfree, Nv, Next, - Last, Head, Elen, Degree, W, nel) ; - } -#endif - -/* ========================================================================= */ -/* GET PIVOT OF MINIMUM DEGREE */ -/* ========================================================================= */ - - /* ----------------------------------------------------------------- */ - /* find next supervariable for elimination */ - /* ----------------------------------------------------------------- */ - - ASSERT (mindeg >= 0 && mindeg < n) ; - for (deg = mindeg ; deg < n ; deg++) - { - me = Head [deg] ; - if (me != EMPTY) break ; - } - mindeg = deg ; - ASSERT (me >= 0 && me < n) ; - AMD_DEBUG1 (("=================me: "ID"\n", me)) ; - - /* ----------------------------------------------------------------- */ - /* remove chosen variable from link list */ - /* ----------------------------------------------------------------- */ - - inext = Next [me] ; - ASSERT (inext >= EMPTY && inext < n) ; - if (inext != EMPTY) Last [inext] = EMPTY ; - Head [deg] = inext ; - - /* ----------------------------------------------------------------- */ - /* me represents the elimination of pivots nel to nel+Nv[me]-1. */ - /* place me itself as the first in this set. */ - /* ----------------------------------------------------------------- */ - - elenme = Elen [me] ; - nvpiv = Nv [me] ; - ASSERT (nvpiv > 0) ; - nel += nvpiv ; - -/* ========================================================================= */ -/* CONSTRUCT NEW ELEMENT */ -/* ========================================================================= */ - - /* ----------------------------------------------------------------- - * At this point, me is the pivotal supervariable. It will be - * converted into the current element. Scan list of the pivotal - * supervariable, me, setting tree pointers and constructing new list - * of supervariables for the new element, me. p is a pointer to the - * current position in the old list. - * ----------------------------------------------------------------- */ - - /* flag the variable "me" as being in Lme by negating Nv [me] */ - Nv [me] = -nvpiv ; - degme = 0 ; - ASSERT (Pe [me] >= 0 && Pe [me] < iwlen) ; - - if (elenme == 0) - { - - /* ------------------------------------------------------------- */ - /* construct the new element in place */ - /* ------------------------------------------------------------- */ - - pme1 = Pe [me] ; - pme2 = pme1 - 1 ; - - for (p = pme1 ; p <= pme1 + Len [me] - 1 ; p++) - { - i = Iw [p] ; - ASSERT (i >= 0 && i < n && Nv [i] >= 0) ; - nvi = Nv [i] ; - if (nvi > 0) - { - - /* ----------------------------------------------------- */ - /* i is a principal variable not yet placed in Lme. */ - /* store i in new list */ - /* ----------------------------------------------------- */ - - /* flag i as being in Lme by negating Nv [i] */ - degme += nvi ; - Nv [i] = -nvi ; - Iw [++pme2] = i ; - - /* ----------------------------------------------------- */ - /* remove variable i from degree list. */ - /* ----------------------------------------------------- */ - - ilast = Last [i] ; - inext = Next [i] ; - ASSERT (ilast >= EMPTY && ilast < n) ; - ASSERT (inext >= EMPTY && inext < n) ; - if (inext != EMPTY) Last [inext] = ilast ; - if (ilast != EMPTY) - { - Next [ilast] = inext ; - } - else - { - /* i is at the head of the degree list */ - ASSERT (Degree [i] >= 0 && Degree [i] < n) ; - Head [Degree [i]] = inext ; - } - } - } - } - else - { - - /* ------------------------------------------------------------- */ - /* construct the new element in empty space, Iw [pfree ...] */ - /* ------------------------------------------------------------- */ - - p = Pe [me] ; - pme1 = pfree ; - slenme = Len [me] - elenme ; - - for (knt1 = 1 ; knt1 <= elenme + 1 ; knt1++) - { - - if (knt1 > elenme) - { - /* search the supervariables in me. */ - e = me ; - pj = p ; - ln = slenme ; - AMD_DEBUG2 (("Search sv: "ID" "ID" "ID"\n", me,pj,ln)) ; - } - else - { - /* search the elements in me. */ - e = Iw [p++] ; - ASSERT (e >= 0 && e < n) ; - pj = Pe [e] ; - ln = Len [e] ; - AMD_DEBUG2 (("Search element e "ID" in me "ID"\n", e,me)) ; - ASSERT (Elen [e] < EMPTY && W [e] > 0 && pj >= 0) ; - } - ASSERT (ln >= 0 && (ln == 0 || (pj >= 0 && pj < iwlen))) ; - - /* --------------------------------------------------------- - * search for different supervariables and add them to the - * new list, compressing when necessary. this loop is - * executed once for each element in the list and once for - * all the supervariables in the list. - * --------------------------------------------------------- */ - - for (knt2 = 1 ; knt2 <= ln ; knt2++) - { - i = Iw [pj++] ; - ASSERT (i >= 0 && i < n && (i == me || Elen [i] >= EMPTY)); - nvi = Nv [i] ; - AMD_DEBUG2 ((": "ID" "ID" "ID" "ID"\n", - i, Elen [i], Nv [i], wflg)) ; - - if (nvi > 0) - { - - /* ------------------------------------------------- */ - /* compress Iw, if necessary */ - /* ------------------------------------------------- */ - - if (pfree >= iwlen) - { - - AMD_DEBUG1 (("GARBAGE COLLECTION\n")) ; - - /* prepare for compressing Iw by adjusting pointers - * and lengths so that the lists being searched in - * the inner and outer loops contain only the - * remaining entries. */ - - Pe [me] = p ; - Len [me] -= knt1 ; - /* check if nothing left of supervariable me */ - if (Len [me] == 0) Pe [me] = EMPTY ; - Pe [e] = pj ; - Len [e] = ln - knt2 ; - /* nothing left of element e */ - if (Len [e] == 0) Pe [e] = EMPTY ; - - ncmpa++ ; /* one more garbage collection */ - - /* store first entry of each object in Pe */ - /* FLIP the first entry in each object */ - for (j = 0 ; j < n ; j++) - { - pn = Pe [j] ; - if (pn >= 0) - { - ASSERT (pn >= 0 && pn < iwlen) ; - Pe [j] = Iw [pn] ; - Iw [pn] = FLIP (j) ; - } - } - - /* psrc/pdst point to source/destination */ - psrc = 0 ; - pdst = 0 ; - pend = pme1 - 1 ; - - while (psrc <= pend) - { - /* search for next FLIP'd entry */ - j = FLIP (Iw [psrc++]) ; - if (j >= 0) - { - AMD_DEBUG2 (("Got object j: "ID"\n", j)) ; - Iw [pdst] = Pe [j] ; - Pe [j] = pdst++ ; - lenj = Len [j] ; - /* copy from source to destination */ - for (knt3 = 0 ; knt3 <= lenj - 2 ; knt3++) - { - Iw [pdst++] = Iw [psrc++] ; - } - } - } - - /* move the new partially-constructed element */ - p1 = pdst ; - for (psrc = pme1 ; psrc <= pfree-1 ; psrc++) - { - Iw [pdst++] = Iw [psrc] ; - } - pme1 = p1 ; - pfree = pdst ; - pj = Pe [e] ; - p = Pe [me] ; - - } - - /* ------------------------------------------------- */ - /* i is a principal variable not yet placed in Lme */ - /* store i in new list */ - /* ------------------------------------------------- */ - - /* flag i as being in Lme by negating Nv [i] */ - degme += nvi ; - Nv [i] = -nvi ; - Iw [pfree++] = i ; - AMD_DEBUG2 ((" s: "ID" nv "ID"\n", i, Nv [i])); - - /* ------------------------------------------------- */ - /* remove variable i from degree link list */ - /* ------------------------------------------------- */ - - ilast = Last [i] ; - inext = Next [i] ; - ASSERT (ilast >= EMPTY && ilast < n) ; - ASSERT (inext >= EMPTY && inext < n) ; - if (inext != EMPTY) Last [inext] = ilast ; - if (ilast != EMPTY) - { - Next [ilast] = inext ; - } - else - { - /* i is at the head of the degree list */ - ASSERT (Degree [i] >= 0 && Degree [i] < n) ; - Head [Degree [i]] = inext ; - } - } - } - - if (e != me) - { - /* set tree pointer and flag to indicate element e is - * absorbed into new element me (the parent of e is me) */ - AMD_DEBUG1 ((" Element "ID" => "ID"\n", e, me)) ; - Pe [e] = FLIP (me) ; - W [e] = 0 ; - } - } - - pme2 = pfree - 1 ; - } - - /* ----------------------------------------------------------------- */ - /* me has now been converted into an element in Iw [pme1..pme2] */ - /* ----------------------------------------------------------------- */ - - /* degme holds the external degree of new element */ - Degree [me] = degme ; - Pe [me] = pme1 ; - Len [me] = pme2 - pme1 + 1 ; - ASSERT (Pe [me] >= 0 && Pe [me] < iwlen) ; - - Elen [me] = FLIP (nvpiv + degme) ; - /* FLIP (Elen (me)) is now the degree of pivot (including - * diagonal part). */ - -#ifndef NDEBUG - AMD_DEBUG2 (("New element structure: length= "ID"\n", pme2-pme1+1)) ; - for (pme = pme1 ; pme <= pme2 ; pme++) AMD_DEBUG3 ((" "ID"", Iw[pme])); - AMD_DEBUG3 (("\n")) ; -#endif - - /* ----------------------------------------------------------------- */ - /* make sure that wflg is not too large. */ - /* ----------------------------------------------------------------- */ - - /* With the current value of wflg, wflg+n must not cause integer - * overflow */ - - wflg = clear_flag (wflg, wbig, W, n) ; - -/* ========================================================================= */ -/* COMPUTE (W [e] - wflg) = |Le\Lme| FOR ALL ELEMENTS */ -/* ========================================================================= */ - - /* ----------------------------------------------------------------- - * Scan 1: compute the external degrees of previous elements with - * respect to the current element. That is: - * (W [e] - wflg) = |Le \ Lme| - * for each element e that appears in any supervariable in Lme. The - * notation Le refers to the pattern (list of supervariables) of a - * previous element e, where e is not yet absorbed, stored in - * Iw [Pe [e] + 1 ... Pe [e] + Len [e]]. The notation Lme - * refers to the pattern of the current element (stored in - * Iw [pme1..pme2]). If aggressive absorption is enabled, and - * (W [e] - wflg) becomes zero, then the element e will be absorbed - * in Scan 2. - * ----------------------------------------------------------------- */ - - AMD_DEBUG2 (("me: ")) ; - for (pme = pme1 ; pme <= pme2 ; pme++) - { - i = Iw [pme] ; - ASSERT (i >= 0 && i < n) ; - eln = Elen [i] ; - AMD_DEBUG3 ((""ID" Elen "ID": \n", i, eln)) ; - if (eln > 0) - { - /* note that Nv [i] has been negated to denote i in Lme: */ - nvi = -Nv [i] ; - ASSERT (nvi > 0 && Pe [i] >= 0 && Pe [i] < iwlen) ; - wnvi = wflg - nvi ; - for (p = Pe [i] ; p <= Pe [i] + eln - 1 ; p++) - { - e = Iw [p] ; - ASSERT (e >= 0 && e < n) ; - we = W [e] ; - AMD_DEBUG4 ((" e "ID" we "ID" ", e, we)) ; - if (we >= wflg) - { - /* unabsorbed element e has been seen in this loop */ - AMD_DEBUG4 ((" unabsorbed, first time seen")) ; - we -= nvi ; - } - else if (we != 0) - { - /* e is an unabsorbed element */ - /* this is the first we have seen e in all of Scan 1 */ - AMD_DEBUG4 ((" unabsorbed")) ; - we = Degree [e] + wnvi ; - } - AMD_DEBUG4 (("\n")) ; - W [e] = we ; - } - } - } - AMD_DEBUG2 (("\n")) ; - -/* ========================================================================= */ -/* DEGREE UPDATE AND ELEMENT ABSORPTION */ -/* ========================================================================= */ - - /* ----------------------------------------------------------------- - * Scan 2: for each i in Lme, sum up the degree of Lme (which is - * degme), plus the sum of the external degrees of each Le for the - * elements e appearing within i, plus the supervariables in i. - * Place i in hash list. - * ----------------------------------------------------------------- */ - - for (pme = pme1 ; pme <= pme2 ; pme++) - { - i = Iw [pme] ; - ASSERT (i >= 0 && i < n && Nv [i] < 0 && Elen [i] >= 0) ; - AMD_DEBUG2 (("Updating: i "ID" "ID" "ID"\n", i, Elen[i], Len [i])); - p1 = Pe [i] ; - p2 = p1 + Elen [i] - 1 ; - pn = p1 ; - hash = 0 ; - deg = 0 ; - ASSERT (p1 >= 0 && p1 < iwlen && p2 >= -1 && p2 < iwlen) ; - - /* ------------------------------------------------------------- */ - /* scan the element list associated with supervariable i */ - /* ------------------------------------------------------------- */ - - /* UMFPACK/MA38-style approximate degree: */ - if (aggressive) - { - for (p = p1 ; p <= p2 ; p++) - { - e = Iw [p] ; - ASSERT (e >= 0 && e < n) ; - we = W [e] ; - if (we != 0) - { - /* e is an unabsorbed element */ - /* dext = | Le \ Lme | */ - dext = we - wflg ; - if (dext > 0) - { - deg += dext ; - Iw [pn++] = e ; - hash += e ; - AMD_DEBUG4 ((" e: "ID" hash = "ID"\n",e,hash)) ; - } - else - { - /* external degree of e is zero, absorb e into me*/ - AMD_DEBUG1 ((" Element "ID" =>"ID" (aggressive)\n", - e, me)) ; - ASSERT (dext == 0) ; - Pe [e] = FLIP (me) ; - W [e] = 0 ; - } - } - } - } - else - { - for (p = p1 ; p <= p2 ; p++) - { - e = Iw [p] ; - ASSERT (e >= 0 && e < n) ; - we = W [e] ; - if (we != 0) - { - /* e is an unabsorbed element */ - dext = we - wflg ; - ASSERT (dext >= 0) ; - deg += dext ; - Iw [pn++] = e ; - hash += e ; - AMD_DEBUG4 ((" e: "ID" hash = "ID"\n",e,hash)) ; - } - } - } - - /* count the number of elements in i (including me): */ - Elen [i] = pn - p1 + 1 ; - - /* ------------------------------------------------------------- */ - /* scan the supervariables in the list associated with i */ - /* ------------------------------------------------------------- */ - - /* The bulk of the AMD run time is typically spent in this loop, - * particularly if the matrix has many dense rows that are not - * removed prior to ordering. */ - p3 = pn ; - p4 = p1 + Len [i] ; - for (p = p2 + 1 ; p < p4 ; p++) - { - j = Iw [p] ; - ASSERT (j >= 0 && j < n) ; - nvj = Nv [j] ; - if (nvj > 0) - { - /* j is unabsorbed, and not in Lme. */ - /* add to degree and add to new list */ - deg += nvj ; - Iw [pn++] = j ; - hash += j ; - AMD_DEBUG4 ((" s: "ID" hash "ID" Nv[j]= "ID"\n", - j, hash, nvj)) ; - } - } - - /* ------------------------------------------------------------- */ - /* update the degree and check for mass elimination */ - /* ------------------------------------------------------------- */ - - /* with aggressive absorption, deg==0 is identical to the - * Elen [i] == 1 && p3 == pn test, below. */ - ASSERT (IMPLIES (aggressive, (deg==0) == (Elen[i]==1 && p3==pn))) ; - - if (Elen [i] == 1 && p3 == pn) - { - - /* --------------------------------------------------------- */ - /* mass elimination */ - /* --------------------------------------------------------- */ - - /* There is nothing left of this node except for an edge to - * the current pivot element. Elen [i] is 1, and there are - * no variables adjacent to node i. Absorb i into the - * current pivot element, me. Note that if there are two or - * more mass eliminations, fillin due to mass elimination is - * possible within the nvpiv-by-nvpiv pivot block. It is this - * step that causes AMD's analysis to be an upper bound. - * - * The reason is that the selected pivot has a lower - * approximate degree than the true degree of the two mass - * eliminated nodes. There is no edge between the two mass - * eliminated nodes. They are merged with the current pivot - * anyway. - * - * No fillin occurs in the Schur complement, in any case, - * and this effect does not decrease the quality of the - * ordering itself, just the quality of the nonzero and - * flop count analysis. It also means that the post-ordering - * is not an exact elimination tree post-ordering. */ - - AMD_DEBUG1 ((" MASS i "ID" => parent e "ID"\n", i, me)) ; - Pe [i] = FLIP (me) ; - nvi = -Nv [i] ; - degme -= nvi ; - nvpiv += nvi ; - nel += nvi ; - Nv [i] = 0 ; - Elen [i] = EMPTY ; - - } - else - { - - /* --------------------------------------------------------- */ - /* update the upper-bound degree of i */ - /* --------------------------------------------------------- */ - - /* the following degree does not yet include the size - * of the current element, which is added later: */ - - Degree [i] = MIN (Degree [i], deg) ; - - /* --------------------------------------------------------- */ - /* add me to the list for i */ - /* --------------------------------------------------------- */ - - /* move first supervariable to end of list */ - Iw [pn] = Iw [p3] ; - /* move first element to end of element part of list */ - Iw [p3] = Iw [p1] ; - /* add new element, me, to front of list. */ - Iw [p1] = me ; - /* store the new length of the list in Len [i] */ - Len [i] = pn - p1 + 1 ; - - /* --------------------------------------------------------- */ - /* place in hash bucket. Save hash key of i in Last [i]. */ - /* --------------------------------------------------------- */ - - /* NOTE: this can fail if hash is negative, because the ANSI C - * standard does not define a % b when a and/or b are negative. - * That's why hash is defined as an unsigned Int, to avoid this - * problem. */ - hash = hash % n ; - ASSERT (((Int) hash) >= 0 && ((Int) hash) < n) ; - - /* if the Hhead array is not used: */ - j = Head [hash] ; - if (j <= EMPTY) - { - /* degree list is empty, hash head is FLIP (j) */ - Next [i] = FLIP (j) ; - Head [hash] = FLIP (i) ; - } - else - { - /* degree list is not empty, use Last [Head [hash]] as - * hash head. */ - Next [i] = Last [j] ; - Last [j] = i ; - } - - /* if a separate Hhead array is used: * - Next [i] = Hhead [hash] ; - Hhead [hash] = i ; - */ - - Last [i] = hash ; - } - } - - Degree [me] = degme ; - - /* ----------------------------------------------------------------- */ - /* Clear the counter array, W [...], by incrementing wflg. */ - /* ----------------------------------------------------------------- */ - - /* make sure that wflg+n does not cause integer overflow */ - lemax = MAX (lemax, degme) ; - wflg += lemax ; - wflg = clear_flag (wflg, wbig, W, n) ; - /* at this point, W [0..n-1] < wflg holds */ - -/* ========================================================================= */ -/* SUPERVARIABLE DETECTION */ -/* ========================================================================= */ - - AMD_DEBUG1 (("Detecting supervariables:\n")) ; - for (pme = pme1 ; pme <= pme2 ; pme++) - { - i = Iw [pme] ; - ASSERT (i >= 0 && i < n) ; - AMD_DEBUG2 (("Consider i "ID" nv "ID"\n", i, Nv [i])) ; - if (Nv [i] < 0) - { - /* i is a principal variable in Lme */ - - /* --------------------------------------------------------- - * examine all hash buckets with 2 or more variables. We do - * this by examing all unique hash keys for supervariables in - * the pattern Lme of the current element, me - * --------------------------------------------------------- */ - - /* let i = head of hash bucket, and empty the hash bucket */ - ASSERT (Last [i] >= 0 && Last [i] < n) ; - hash = Last [i] ; - - /* if Hhead array is not used: */ - j = Head [hash] ; - if (j == EMPTY) - { - /* hash bucket and degree list are both empty */ - i = EMPTY ; - } - else if (j < EMPTY) - { - /* degree list is empty */ - i = FLIP (j) ; - Head [hash] = EMPTY ; - } - else - { - /* degree list is not empty, restore Last [j] of head j */ - i = Last [j] ; - Last [j] = EMPTY ; - } - - /* if separate Hhead array is used: * - i = Hhead [hash] ; - Hhead [hash] = EMPTY ; - */ - - ASSERT (i >= EMPTY && i < n) ; - AMD_DEBUG2 (("----i "ID" hash "ID"\n", i, hash)) ; - - while (i != EMPTY && Next [i] != EMPTY) - { - - /* ----------------------------------------------------- - * this bucket has one or more variables following i. - * scan all of them to see if i can absorb any entries - * that follow i in hash bucket. Scatter i into w. - * ----------------------------------------------------- */ - - ln = Len [i] ; - eln = Elen [i] ; - ASSERT (ln >= 0 && eln >= 0) ; - ASSERT (Pe [i] >= 0 && Pe [i] < iwlen) ; - /* do not flag the first element in the list (me) */ - for (p = Pe [i] + 1 ; p <= Pe [i] + ln - 1 ; p++) - { - ASSERT (Iw [p] >= 0 && Iw [p] < n) ; - W [Iw [p]] = wflg ; - } - - /* ----------------------------------------------------- */ - /* scan every other entry j following i in bucket */ - /* ----------------------------------------------------- */ - - jlast = i ; - j = Next [i] ; - ASSERT (j >= EMPTY && j < n) ; - - while (j != EMPTY) - { - /* ------------------------------------------------- */ - /* check if j and i have identical nonzero pattern */ - /* ------------------------------------------------- */ - - AMD_DEBUG3 (("compare i "ID" and j "ID"\n", i,j)) ; - - /* check if i and j have the same Len and Elen */ - ASSERT (Len [j] >= 0 && Elen [j] >= 0) ; - ASSERT (Pe [j] >= 0 && Pe [j] < iwlen) ; - ok = (Len [j] == ln) && (Elen [j] == eln) ; - /* skip the first element in the list (me) */ - for (p = Pe [j] + 1 ; ok && p <= Pe [j] + ln - 1 ; p++) - { - ASSERT (Iw [p] >= 0 && Iw [p] < n) ; - if (W [Iw [p]] != wflg) ok = 0 ; - } - if (ok) - { - /* --------------------------------------------- */ - /* found it! j can be absorbed into i */ - /* --------------------------------------------- */ - - AMD_DEBUG1 (("found it! j "ID" => i "ID"\n", j,i)); - Pe [j] = FLIP (i) ; - /* both Nv [i] and Nv [j] are negated since they */ - /* are in Lme, and the absolute values of each */ - /* are the number of variables in i and j: */ - Nv [i] += Nv [j] ; - Nv [j] = 0 ; - Elen [j] = EMPTY ; - /* delete j from hash bucket */ - ASSERT (j != Next [j]) ; - j = Next [j] ; - Next [jlast] = j ; - - } - else - { - /* j cannot be absorbed into i */ - jlast = j ; - ASSERT (j != Next [j]) ; - j = Next [j] ; - } - ASSERT (j >= EMPTY && j < n) ; - } - - /* ----------------------------------------------------- - * no more variables can be absorbed into i - * go to next i in bucket and clear flag array - * ----------------------------------------------------- */ - - wflg++ ; - i = Next [i] ; - ASSERT (i >= EMPTY && i < n) ; - - } - } - } - AMD_DEBUG2 (("detect done\n")) ; - -/* ========================================================================= */ -/* RESTORE DEGREE LISTS AND REMOVE NONPRINCIPAL SUPERVARIABLES FROM ELEMENT */ -/* ========================================================================= */ - - p = pme1 ; - nleft = n - nel ; - for (pme = pme1 ; pme <= pme2 ; pme++) - { - i = Iw [pme] ; - ASSERT (i >= 0 && i < n) ; - nvi = -Nv [i] ; - AMD_DEBUG3 (("Restore i "ID" "ID"\n", i, nvi)) ; - if (nvi > 0) - { - /* i is a principal variable in Lme */ - /* restore Nv [i] to signify that i is principal */ - Nv [i] = nvi ; - - /* --------------------------------------------------------- */ - /* compute the external degree (add size of current element) */ - /* --------------------------------------------------------- */ - - deg = Degree [i] + degme - nvi ; - deg = MIN (deg, nleft - nvi) ; - ASSERT (IMPLIES (aggressive, deg > 0) && deg >= 0 && deg < n) ; - - /* --------------------------------------------------------- */ - /* place the supervariable at the head of the degree list */ - /* --------------------------------------------------------- */ - - inext = Head [deg] ; - ASSERT (inext >= EMPTY && inext < n) ; - if (inext != EMPTY) Last [inext] = i ; - Next [i] = inext ; - Last [i] = EMPTY ; - Head [deg] = i ; - - /* --------------------------------------------------------- */ - /* save the new degree, and find the minimum degree */ - /* --------------------------------------------------------- */ - - mindeg = MIN (mindeg, deg) ; - Degree [i] = deg ; - - /* --------------------------------------------------------- */ - /* place the supervariable in the element pattern */ - /* --------------------------------------------------------- */ - - Iw [p++] = i ; - - } - } - AMD_DEBUG2 (("restore done\n")) ; - -/* ========================================================================= */ -/* FINALIZE THE NEW ELEMENT */ -/* ========================================================================= */ - - AMD_DEBUG2 (("ME = "ID" DONE\n", me)) ; - Nv [me] = nvpiv ; - /* save the length of the list for the new element me */ - Len [me] = p - pme1 ; - if (Len [me] == 0) - { - /* there is nothing left of the current pivot element */ - /* it is a root of the assembly tree */ - Pe [me] = EMPTY ; - W [me] = 0 ; - } - if (elenme != 0) - { - /* element was not constructed in place: deallocate part of */ - /* it since newly nonprincipal variables may have been removed */ - pfree = p ; - } - - /* The new element has nvpiv pivots and the size of the contribution - * block for a multifrontal method is degme-by-degme, not including - * the "dense" rows/columns. If the "dense" rows/columns are included, - * the frontal matrix is no larger than - * (degme+ndense)-by-(degme+ndense). - */ - - if (Info != (double *) NULL) - { - f = nvpiv ; - r = degme + ndense ; - dmax = MAX (dmax, f + r) ; - - /* number of nonzeros in L (excluding the diagonal) */ - lnzme = f*r + (f-1)*f/2 ; - lnz += lnzme ; - - /* number of divide operations for LDL' and for LU */ - ndiv += lnzme ; - - /* number of multiply-subtract pairs for LU */ - s = f*r*r + r*(f-1)*f + (f-1)*f*(2*f-1)/6 ; - nms_lu += s ; - - /* number of multiply-subtract pairs for LDL' */ - nms_ldl += (s + lnzme)/2 ; - } - -#ifndef NDEBUG - AMD_DEBUG2 (("finalize done nel "ID" n "ID"\n ::::\n", nel, n)) ; - for (pme = Pe [me] ; pme <= Pe [me] + Len [me] - 1 ; pme++) - { - AMD_DEBUG3 ((" "ID"", Iw [pme])) ; - } - AMD_DEBUG3 (("\n")) ; -#endif - - } - -/* ========================================================================= */ -/* DONE SELECTING PIVOTS */ -/* ========================================================================= */ - - if (Info != (double *) NULL) - { - - /* count the work to factorize the ndense-by-ndense submatrix */ - f = ndense ; - dmax = MAX (dmax, (double) ndense) ; - - /* number of nonzeros in L (excluding the diagonal) */ - lnzme = (f-1)*f/2 ; - lnz += lnzme ; - - /* number of divide operations for LDL' and for LU */ - ndiv += lnzme ; - - /* number of multiply-subtract pairs for LU */ - s = (f-1)*f*(2*f-1)/6 ; - nms_lu += s ; - - /* number of multiply-subtract pairs for LDL' */ - nms_ldl += (s + lnzme)/2 ; - - /* number of nz's in L (excl. diagonal) */ - Info [AMD_LNZ] = lnz ; - - /* number of divide ops for LU and LDL' */ - Info [AMD_NDIV] = ndiv ; - - /* number of multiply-subtract pairs for LDL' */ - Info [AMD_NMULTSUBS_LDL] = nms_ldl ; - - /* number of multiply-subtract pairs for LU */ - Info [AMD_NMULTSUBS_LU] = nms_lu ; - - /* number of "dense" rows/columns */ - Info [AMD_NDENSE] = ndense ; - - /* largest front is dmax-by-dmax */ - Info [AMD_DMAX] = dmax ; - - /* number of garbage collections in AMD */ - Info [AMD_NCMPA] = ncmpa ; - - /* successful ordering */ - Info [AMD_STATUS] = AMD_OK ; - } - -/* ========================================================================= */ -/* POST-ORDERING */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- - * Variables at this point: - * - * Pe: holds the elimination tree. The parent of j is FLIP (Pe [j]), - * or EMPTY if j is a root. The tree holds both elements and - * non-principal (unordered) variables absorbed into them. - * Dense variables are non-principal and unordered. - * - * Elen: holds the size of each element, including the diagonal part. - * FLIP (Elen [e]) > 0 if e is an element. For unordered - * variables i, Elen [i] is EMPTY. - * - * Nv: Nv [e] > 0 is the number of pivots represented by the element e. - * For unordered variables i, Nv [i] is zero. - * - * Contents no longer needed: - * W, Iw, Len, Degree, Head, Next, Last. - * - * The matrix itself has been destroyed. - * - * n: the size of the matrix. - * No other scalars needed (pfree, iwlen, etc.) - * ------------------------------------------------------------------------- */ - - /* restore Pe */ - for (i = 0 ; i < n ; i++) - { - Pe [i] = FLIP (Pe [i]) ; - } - - /* restore Elen, for output information, and for postordering */ - for (i = 0 ; i < n ; i++) - { - Elen [i] = FLIP (Elen [i]) ; - } - -/* Now the parent of j is Pe [j], or EMPTY if j is a root. Elen [e] > 0 - * is the size of element e. Elen [i] is EMPTY for unordered variable i. */ - -#ifndef NDEBUG - AMD_DEBUG2 (("\nTree:\n")) ; - for (i = 0 ; i < n ; i++) - { - AMD_DEBUG2 ((" "ID" parent: "ID" ", i, Pe [i])) ; - ASSERT (Pe [i] >= EMPTY && Pe [i] < n) ; - if (Nv [i] > 0) - { - /* this is an element */ - e = i ; - AMD_DEBUG2 ((" element, size is "ID"\n", Elen [i])) ; - ASSERT (Elen [e] > 0) ; - } - AMD_DEBUG2 (("\n")) ; - } - AMD_DEBUG2 (("\nelements:\n")) ; - for (e = 0 ; e < n ; e++) - { - if (Nv [e] > 0) - { - AMD_DEBUG3 (("Element e= "ID" size "ID" nv "ID" \n", e, - Elen [e], Nv [e])) ; - } - } - AMD_DEBUG2 (("\nvariables:\n")) ; - for (i = 0 ; i < n ; i++) - { - Int cnt ; - if (Nv [i] == 0) - { - AMD_DEBUG3 (("i unordered: "ID"\n", i)) ; - j = Pe [i] ; - cnt = 0 ; - AMD_DEBUG3 ((" j: "ID"\n", j)) ; - if (j == EMPTY) - { - AMD_DEBUG3 ((" i is a dense variable\n")) ; - } - else - { - ASSERT (j >= 0 && j < n) ; - while (Nv [j] == 0) - { - AMD_DEBUG3 ((" j : "ID"\n", j)) ; - j = Pe [j] ; - AMD_DEBUG3 ((" j:: "ID"\n", j)) ; - cnt++ ; - if (cnt > n) break ; - } - e = j ; - AMD_DEBUG3 ((" got to e: "ID"\n", e)) ; - } - } - } -#endif - -/* ========================================================================= */ -/* compress the paths of the variables */ -/* ========================================================================= */ - - for (i = 0 ; i < n ; i++) - { - if (Nv [i] == 0) - { - - /* ------------------------------------------------------------- - * i is an un-ordered row. Traverse the tree from i until - * reaching an element, e. The element, e, was the principal - * supervariable of i and all nodes in the path from i to when e - * was selected as pivot. - * ------------------------------------------------------------- */ - - AMD_DEBUG1 (("Path compression, i unordered: "ID"\n", i)) ; - j = Pe [i] ; - ASSERT (j >= EMPTY && j < n) ; - AMD_DEBUG3 ((" j: "ID"\n", j)) ; - if (j == EMPTY) - { - /* Skip a dense variable. It has no parent. */ - AMD_DEBUG3 ((" i is a dense variable\n")) ; - continue ; - } - - /* while (j is a variable) */ - while (Nv [j] == 0) - { - AMD_DEBUG3 ((" j : "ID"\n", j)) ; - j = Pe [j] ; - AMD_DEBUG3 ((" j:: "ID"\n", j)) ; - ASSERT (j >= 0 && j < n) ; - } - /* got to an element e */ - e = j ; - AMD_DEBUG3 (("got to e: "ID"\n", e)) ; - - /* ------------------------------------------------------------- - * traverse the path again from i to e, and compress the path - * (all nodes point to e). Path compression allows this code to - * compute in O(n) time. - * ------------------------------------------------------------- */ - - j = i ; - /* while (j is a variable) */ - while (Nv [j] == 0) - { - jnext = Pe [j] ; - AMD_DEBUG3 (("j "ID" jnext "ID"\n", j, jnext)) ; - Pe [j] = e ; - j = jnext ; - ASSERT (j >= 0 && j < n) ; - } - } - } - -/* ========================================================================= */ -/* postorder the assembly tree */ -/* ========================================================================= */ - - AMD_postorder (n, Pe, Nv, Elen, - W, /* output order */ - Head, Next, Last) ; /* workspace */ - -/* ========================================================================= */ -/* compute output permutation and inverse permutation */ -/* ========================================================================= */ - - /* W [e] = k means that element e is the kth element in the new - * order. e is in the range 0 to n-1, and k is in the range 0 to - * the number of elements. Use Head for inverse order. */ - - for (k = 0 ; k < n ; k++) - { - Head [k] = EMPTY ; - Next [k] = EMPTY ; - } - for (e = 0 ; e < n ; e++) - { - k = W [e] ; - ASSERT ((k == EMPTY) == (Nv [e] == 0)) ; - if (k != EMPTY) - { - ASSERT (k >= 0 && k < n) ; - Head [k] = e ; - } - } - - /* construct output inverse permutation in Next, - * and permutation in Last */ - nel = 0 ; - for (k = 0 ; k < n ; k++) - { - e = Head [k] ; - if (e == EMPTY) break ; - ASSERT (e >= 0 && e < n && Nv [e] > 0) ; - Next [e] = nel ; - nel += Nv [e] ; - } - ASSERT (nel == n - ndense) ; - - /* order non-principal variables (dense, & those merged into supervar's) */ - for (i = 0 ; i < n ; i++) - { - if (Nv [i] == 0) - { - e = Pe [i] ; - ASSERT (e >= EMPTY && e < n) ; - if (e != EMPTY) - { - /* This is an unordered variable that was merged - * into element e via supernode detection or mass - * elimination of i when e became the pivot element. - * Place i in order just before e. */ - ASSERT (Next [i] == EMPTY && Nv [e] > 0) ; - Next [i] = Next [e] ; - Next [e]++ ; - } - else - { - /* This is a dense unordered variable, with no parent. - * Place it last in the output order. */ - Next [i] = nel++ ; - } - } - } - ASSERT (nel == n) ; - - AMD_DEBUG2 (("\n\nPerm:\n")) ; - for (i = 0 ; i < n ; i++) - { - k = Next [i] ; - ASSERT (k >= 0 && k < n) ; - Last [k] = i ; - AMD_DEBUG2 ((" perm ["ID"] = "ID"\n", k, i)) ; - } -} diff --git a/code/3rd_glpk/amd/amd_aat.c b/code/3rd_glpk/amd/amd_aat.c deleted file mode 100644 index 63bf55f5..00000000 --- a/code/3rd_glpk/amd/amd_aat.c +++ /dev/null @@ -1,185 +0,0 @@ -/* ========================================================================= */ -/* === AMD_aat ============================================================= */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* AMD_aat: compute the symmetry of the pattern of A, and count the number of - * nonzeros each column of A+A' (excluding the diagonal). Assumes the input - * matrix has no errors, with sorted columns and no duplicates - * (AMD_valid (n, n, Ap, Ai) must be AMD_OK, but this condition is not - * checked). - */ - -#include "amd_internal.h" - -GLOBAL size_t AMD_aat /* returns nz in A+A' */ -( - Int n, - const Int Ap [ ], - const Int Ai [ ], - Int Len [ ], /* Len [j]: length of column j of A+A', excl diagonal*/ - Int Tp [ ], /* workspace of size n */ - double Info [ ] -) -{ - Int p1, p2, p, i, j, pj, pj2, k, nzdiag, nzboth, nz ; - double sym ; - size_t nzaat ; - -#ifndef NDEBUG - AMD_debug_init ("AMD AAT") ; - for (k = 0 ; k < n ; k++) Tp [k] = EMPTY ; - ASSERT (AMD_valid (n, n, Ap, Ai) == AMD_OK) ; -#endif - - if (Info != (double *) NULL) - { - /* clear the Info array, if it exists */ - for (i = 0 ; i < AMD_INFO ; i++) - { - Info [i] = EMPTY ; - } - Info [AMD_STATUS] = AMD_OK ; - } - - for (k = 0 ; k < n ; k++) - { - Len [k] = 0 ; - } - - nzdiag = 0 ; - nzboth = 0 ; - nz = Ap [n] ; - - for (k = 0 ; k < n ; k++) - { - p1 = Ap [k] ; - p2 = Ap [k+1] ; - AMD_DEBUG2 (("\nAAT Column: "ID" p1: "ID" p2: "ID"\n", k, p1, p2)) ; - - /* construct A+A' */ - for (p = p1 ; p < p2 ; ) - { - /* scan the upper triangular part of A */ - j = Ai [p] ; - if (j < k) - { - /* entry A (j,k) is in the strictly upper triangular part, - * add both A (j,k) and A (k,j) to the matrix A+A' */ - Len [j]++ ; - Len [k]++ ; - AMD_DEBUG3 ((" upper ("ID","ID") ("ID","ID")\n", j,k, k,j)); - p++ ; - } - else if (j == k) - { - /* skip the diagonal */ - p++ ; - nzdiag++ ; - break ; - } - else /* j > k */ - { - /* first entry below the diagonal */ - break ; - } - /* scan lower triangular part of A, in column j until reaching - * row k. Start where last scan left off. */ - ASSERT (Tp [j] != EMPTY) ; - ASSERT (Ap [j] <= Tp [j] && Tp [j] <= Ap [j+1]) ; - pj2 = Ap [j+1] ; - for (pj = Tp [j] ; pj < pj2 ; ) - { - i = Ai [pj] ; - if (i < k) - { - /* A (i,j) is only in the lower part, not in upper. - * add both A (i,j) and A (j,i) to the matrix A+A' */ - Len [i]++ ; - Len [j]++ ; - AMD_DEBUG3 ((" lower ("ID","ID") ("ID","ID")\n", - i,j, j,i)) ; - pj++ ; - } - else if (i == k) - { - /* entry A (k,j) in lower part and A (j,k) in upper */ - pj++ ; - nzboth++ ; - break ; - } - else /* i > k */ - { - /* consider this entry later, when k advances to i */ - break ; - } - } - Tp [j] = pj ; - } - /* Tp [k] points to the entry just below the diagonal in column k */ - Tp [k] = p ; - } - - /* clean up, for remaining mismatched entries */ - for (j = 0 ; j < n ; j++) - { - for (pj = Tp [j] ; pj < Ap [j+1] ; pj++) - { - i = Ai [pj] ; - /* A (i,j) is only in the lower part, not in upper. - * add both A (i,j) and A (j,i) to the matrix A+A' */ - Len [i]++ ; - Len [j]++ ; - AMD_DEBUG3 ((" lower cleanup ("ID","ID") ("ID","ID")\n", - i,j, j,i)) ; - } - } - - /* --------------------------------------------------------------------- */ - /* compute the symmetry of the nonzero pattern of A */ - /* --------------------------------------------------------------------- */ - - /* Given a matrix A, the symmetry of A is: - * B = tril (spones (A), -1) + triu (spones (A), 1) ; - * sym = nnz (B & B') / nnz (B) ; - * or 1 if nnz (B) is zero. - */ - - if (nz == nzdiag) - { - sym = 1 ; - } - else - { - sym = (2 * (double) nzboth) / ((double) (nz - nzdiag)) ; - } - - nzaat = 0 ; - for (k = 0 ; k < n ; k++) - { - nzaat += Len [k] ; - } - - AMD_DEBUG1 (("AMD nz in A+A', excluding diagonal (nzaat) = %g\n", - (double) nzaat)) ; - AMD_DEBUG1 ((" nzboth: "ID" nz: "ID" nzdiag: "ID" symmetry: %g\n", - nzboth, nz, nzdiag, sym)) ; - - if (Info != (double *) NULL) - { - Info [AMD_STATUS] = AMD_OK ; - Info [AMD_N] = n ; - Info [AMD_NZ] = nz ; - Info [AMD_SYMMETRY] = sym ; /* symmetry of pattern of A */ - Info [AMD_NZDIAG] = nzdiag ; /* nonzeros on diagonal of A */ - Info [AMD_NZ_A_PLUS_AT] = nzaat ; /* nonzeros in A+A' */ - } - - return (nzaat) ; -} diff --git a/code/3rd_glpk/amd/amd_control.c b/code/3rd_glpk/amd/amd_control.c deleted file mode 100644 index f4d4f0df..00000000 --- a/code/3rd_glpk/amd/amd_control.c +++ /dev/null @@ -1,64 +0,0 @@ -/* ========================================================================= */ -/* === AMD_control ========================================================= */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* User-callable. Prints the control parameters for AMD. See amd.h - * for details. If the Control array is not present, the defaults are - * printed instead. - */ - -#include "amd_internal.h" - -GLOBAL void AMD_control -( - double Control [ ] -) -{ - double alpha ; - Int aggressive ; - - if (Control != (double *) NULL) - { - alpha = Control [AMD_DENSE] ; - aggressive = Control [AMD_AGGRESSIVE] != 0 ; - } - else - { - alpha = AMD_DEFAULT_DENSE ; - aggressive = AMD_DEFAULT_AGGRESSIVE ; - } - - PRINTF (("\nAMD version %d.%d.%d, %s: approximate minimum degree ordering\n" - " dense row parameter: %g\n", AMD_MAIN_VERSION, AMD_SUB_VERSION, - AMD_SUBSUB_VERSION, AMD_DATE, alpha)) ; - - if (alpha < 0) - { - PRINTF ((" no rows treated as dense\n")) ; - } - else - { - PRINTF (( - " (rows with more than max (%g * sqrt (n), 16) entries are\n" - " considered \"dense\", and placed last in output permutation)\n", - alpha)) ; - } - - if (aggressive) - { - PRINTF ((" aggressive absorption: yes\n")) ; - } - else - { - PRINTF ((" aggressive absorption: no\n")) ; - } - - PRINTF ((" size of AMD integer: %d\n\n", sizeof (Int))) ; -} diff --git a/code/3rd_glpk/amd/amd_defaults.c b/code/3rd_glpk/amd/amd_defaults.c deleted file mode 100644 index 820e8942..00000000 --- a/code/3rd_glpk/amd/amd_defaults.c +++ /dev/null @@ -1,38 +0,0 @@ -/* ========================================================================= */ -/* === AMD_defaults ======================================================== */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* User-callable. Sets default control parameters for AMD. See amd.h - * for details. - */ - -#include "amd_internal.h" - -/* ========================================================================= */ -/* === AMD defaults ======================================================== */ -/* ========================================================================= */ - -GLOBAL void AMD_defaults -( - double Control [ ] -) -{ - Int i ; - - if (Control != (double *) NULL) - { - for (i = 0 ; i < AMD_CONTROL ; i++) - { - Control [i] = 0 ; - } - Control [AMD_DENSE] = AMD_DEFAULT_DENSE ; - Control [AMD_AGGRESSIVE] = AMD_DEFAULT_AGGRESSIVE ; - } -} diff --git a/code/3rd_glpk/amd/amd_dump.c b/code/3rd_glpk/amd/amd_dump.c deleted file mode 100644 index 39bbe1d8..00000000 --- a/code/3rd_glpk/amd/amd_dump.c +++ /dev/null @@ -1,180 +0,0 @@ -/* ========================================================================= */ -/* === AMD_dump ============================================================ */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* Debugging routines for AMD. Not used if NDEBUG is not defined at compile- - * time (the default). See comments in amd_internal.h on how to enable - * debugging. Not user-callable. - */ - -#include "amd_internal.h" - -#ifndef NDEBUG - -/* This global variable is present only when debugging */ -GLOBAL Int AMD_debug = -999 ; /* default is no debug printing */ - -/* ========================================================================= */ -/* === AMD_debug_init ====================================================== */ -/* ========================================================================= */ - -/* Sets the debug print level, by reading the file debug.amd (if it exists) */ - -GLOBAL void AMD_debug_init ( char *s ) -{ - FILE *f ; - f = fopen ("debug.amd", "r") ; - if (f == (FILE *) NULL) - { - AMD_debug = -999 ; - } - else - { - fscanf (f, ID, &AMD_debug) ; - fclose (f) ; - } - if (AMD_debug >= 0) - { - printf ("%s: AMD_debug_init, D= "ID"\n", s, AMD_debug) ; - } -} - -/* ========================================================================= */ -/* === AMD_dump ============================================================ */ -/* ========================================================================= */ - -/* Dump AMD's data structure, except for the hash buckets. This routine - * cannot be called when the hash buckets are non-empty. - */ - -GLOBAL void AMD_dump ( - Int n, /* A is n-by-n */ - Int Pe [ ], /* pe [0..n-1]: index in iw of start of row i */ - Int Iw [ ], /* workspace of size iwlen, iwlen [0..pfree-1] - * holds the matrix on input */ - Int Len [ ], /* len [0..n-1]: length for row i */ - Int iwlen, /* length of iw */ - Int pfree, /* iw [pfree ... iwlen-1] is empty on input */ - Int Nv [ ], /* nv [0..n-1] */ - Int Next [ ], /* next [0..n-1] */ - Int Last [ ], /* last [0..n-1] */ - Int Head [ ], /* head [0..n-1] */ - Int Elen [ ], /* size n */ - Int Degree [ ], /* size n */ - Int W [ ], /* size n */ - Int nel -) -{ - Int i, pe, elen, nv, len, e, p, k, j, deg, w, cnt, ilast ; - - if (AMD_debug < 0) return ; - ASSERT (pfree <= iwlen) ; - AMD_DEBUG3 (("\nAMD dump, pfree: "ID"\n", pfree)) ; - for (i = 0 ; i < n ; i++) - { - pe = Pe [i] ; - elen = Elen [i] ; - nv = Nv [i] ; - len = Len [i] ; - w = W [i] ; - - if (elen >= EMPTY) - { - if (nv == 0) - { - AMD_DEBUG3 (("\nI "ID": nonprincipal: ", i)) ; - ASSERT (elen == EMPTY) ; - if (pe == EMPTY) - { - AMD_DEBUG3 ((" dense node\n")) ; - ASSERT (w == 1) ; - } - else - { - ASSERT (pe < EMPTY) ; - AMD_DEBUG3 ((" i "ID" -> parent "ID"\n", i, FLIP (Pe[i]))); - } - } - else - { - AMD_DEBUG3 (("\nI "ID": active principal supervariable:\n",i)); - AMD_DEBUG3 ((" nv(i): "ID" Flag: %d\n", nv, (nv < 0))) ; - ASSERT (elen >= 0) ; - ASSERT (nv > 0 && pe >= 0) ; - p = pe ; - AMD_DEBUG3 ((" e/s: ")) ; - if (elen == 0) AMD_DEBUG3 ((" : ")) ; - ASSERT (pe + len <= pfree) ; - for (k = 0 ; k < len ; k++) - { - j = Iw [p] ; - AMD_DEBUG3 ((" "ID"", j)) ; - ASSERT (j >= 0 && j < n) ; - if (k == elen-1) AMD_DEBUG3 ((" : ")) ; - p++ ; - } - AMD_DEBUG3 (("\n")) ; - } - } - else - { - e = i ; - if (w == 0) - { - AMD_DEBUG3 (("\nE "ID": absorbed element: w "ID"\n", e, w)) ; - ASSERT (nv > 0 && pe < 0) ; - AMD_DEBUG3 ((" e "ID" -> parent "ID"\n", e, FLIP (Pe [e]))) ; - } - else - { - AMD_DEBUG3 (("\nE "ID": unabsorbed element: w "ID"\n", e, w)) ; - ASSERT (nv > 0 && pe >= 0) ; - p = pe ; - AMD_DEBUG3 ((" : ")) ; - ASSERT (pe + len <= pfree) ; - for (k = 0 ; k < len ; k++) - { - j = Iw [p] ; - AMD_DEBUG3 ((" "ID"", j)) ; - ASSERT (j >= 0 && j < n) ; - p++ ; - } - AMD_DEBUG3 (("\n")) ; - } - } - } - - /* this routine cannot be called when the hash buckets are non-empty */ - AMD_DEBUG3 (("\nDegree lists:\n")) ; - if (nel >= 0) - { - cnt = 0 ; - for (deg = 0 ; deg < n ; deg++) - { - if (Head [deg] == EMPTY) continue ; - ilast = EMPTY ; - AMD_DEBUG3 ((ID": \n", deg)) ; - for (i = Head [deg] ; i != EMPTY ; i = Next [i]) - { - AMD_DEBUG3 ((" "ID" : next "ID" last "ID" deg "ID"\n", - i, Next [i], Last [i], Degree [i])) ; - ASSERT (i >= 0 && i < n && ilast == Last [i] && - deg == Degree [i]) ; - cnt += Nv [i] ; - ilast = i ; - } - AMD_DEBUG3 (("\n")) ; - } - ASSERT (cnt == n - nel) ; - } - -} - -#endif diff --git a/code/3rd_glpk/amd/amd_info.c b/code/3rd_glpk/amd/amd_info.c deleted file mode 100644 index e7b806a9..00000000 --- a/code/3rd_glpk/amd/amd_info.c +++ /dev/null @@ -1,120 +0,0 @@ -/* ========================================================================= */ -/* === AMD_info ============================================================ */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* User-callable. Prints the output statistics for AMD. See amd.h - * for details. If the Info array is not present, nothing is printed. - */ - -#include "amd_internal.h" - -#define PRI(format,x) { if (x >= 0) { PRINTF ((format, x)) ; }} - -GLOBAL void AMD_info -( - double Info [ ] -) -{ - double n, ndiv, nmultsubs_ldl, nmultsubs_lu, lnz, lnzd ; - - PRINTF (("\nAMD version %d.%d.%d, %s, results:\n", - AMD_MAIN_VERSION, AMD_SUB_VERSION, AMD_SUBSUB_VERSION, AMD_DATE)) ; - - if (!Info) - { - return ; - } - - n = Info [AMD_N] ; - ndiv = Info [AMD_NDIV] ; - nmultsubs_ldl = Info [AMD_NMULTSUBS_LDL] ; - nmultsubs_lu = Info [AMD_NMULTSUBS_LU] ; - lnz = Info [AMD_LNZ] ; - lnzd = (n >= 0 && lnz >= 0) ? (n + lnz) : (-1) ; - - /* AMD return status */ - PRINTF ((" status: ")) ; - if (Info [AMD_STATUS] == AMD_OK) - { - PRINTF (("OK\n")) ; - } - else if (Info [AMD_STATUS] == AMD_OUT_OF_MEMORY) - { - PRINTF (("out of memory\n")) ; - } - else if (Info [AMD_STATUS] == AMD_INVALID) - { - PRINTF (("invalid matrix\n")) ; - } - else if (Info [AMD_STATUS] == AMD_OK_BUT_JUMBLED) - { - PRINTF (("OK, but jumbled\n")) ; - } - else - { - PRINTF (("unknown\n")) ; - } - - /* statistics about the input matrix */ - PRI (" n, dimension of A: %.20g\n", n); - PRI (" nz, number of nonzeros in A: %.20g\n", - Info [AMD_NZ]) ; - PRI (" symmetry of A: %.4f\n", - Info [AMD_SYMMETRY]) ; - PRI (" number of nonzeros on diagonal: %.20g\n", - Info [AMD_NZDIAG]) ; - PRI (" nonzeros in pattern of A+A' (excl. diagonal): %.20g\n", - Info [AMD_NZ_A_PLUS_AT]) ; - PRI (" # dense rows/columns of A+A': %.20g\n", - Info [AMD_NDENSE]) ; - - /* statistics about AMD's behavior */ - PRI (" memory used, in bytes: %.20g\n", - Info [AMD_MEMORY]) ; - PRI (" # of memory compactions: %.20g\n", - Info [AMD_NCMPA]) ; - - /* statistics about the ordering quality */ - PRINTF (("\n" - " The following approximate statistics are for a subsequent\n" - " factorization of A(P,P) + A(P,P)'. They are slight upper\n" - " bounds if there are no dense rows/columns in A+A', and become\n" - " looser if dense rows/columns exist.\n\n")) ; - - PRI (" nonzeros in L (excluding diagonal): %.20g\n", - lnz) ; - PRI (" nonzeros in L (including diagonal): %.20g\n", - lnzd) ; - PRI (" # divide operations for LDL' or LU: %.20g\n", - ndiv) ; - PRI (" # multiply-subtract operations for LDL': %.20g\n", - nmultsubs_ldl) ; - PRI (" # multiply-subtract operations for LU: %.20g\n", - nmultsubs_lu) ; - PRI (" max nz. in any column of L (incl. diagonal): %.20g\n", - Info [AMD_DMAX]) ; - - /* total flop counts for various factorizations */ - - if (n >= 0 && ndiv >= 0 && nmultsubs_ldl >= 0 && nmultsubs_lu >= 0) - { - PRINTF (("\n" - " chol flop count for real A, sqrt counted as 1 flop: %.20g\n" - " LDL' flop count for real A: %.20g\n" - " LDL' flop count for complex A: %.20g\n" - " LU flop count for real A (with no pivoting): %.20g\n" - " LU flop count for complex A (with no pivoting): %.20g\n\n", - n + ndiv + 2*nmultsubs_ldl, - ndiv + 2*nmultsubs_ldl, - 9*ndiv + 8*nmultsubs_ldl, - ndiv + 2*nmultsubs_lu, - 9*ndiv + 8*nmultsubs_lu)) ; - } -} diff --git a/code/3rd_glpk/amd/amd_internal.h b/code/3rd_glpk/amd/amd_internal.h deleted file mode 100644 index d51ac5fb..00000000 --- a/code/3rd_glpk/amd/amd_internal.h +++ /dev/null @@ -1,117 +0,0 @@ -/* amd_internal.h */ - -/* Written by Andrew Makhorin . */ - -#ifndef AMD_INTERNAL_H -#define AMD_INTERNAL_H - -/* AMD will be exceedingly slow when running in debug mode. */ -#ifndef NDEBUG -#define NDEBUG -#endif - -#include "amd.h" -#define _GLPSTD_STDIO -#include "env.h" - -#define Int int -#define ID "%d" -#define Int_MAX INT_MAX - -#if 0 /* 15/II-2012 */ -/* now this macro is defined in glpenv.h; besides, the definiton below - depends on implementation, because size_t is an unsigned type */ -#define SIZE_T_MAX ((size_t)(-1)) -#endif - -#define EMPTY (-1) -#define FLIP(i) (-(i)-2) -#define UNFLIP(i) ((i < EMPTY) ? FLIP (i) : (i)) - -#define MAX(a,b) (((a) > (b)) ? (a) : (b)) -#define MIN(a,b) (((a) < (b)) ? (a) : (b)) - -#define IMPLIES(p, q) (!(p) || (q)) - -#define GLOBAL - -#define AMD_order amd_order -#define AMD_defaults amd_defaults -#define AMD_control amd_control -#define AMD_info amd_info -#define AMD_1 amd_1 -#define AMD_2 amd_2 -#define AMD_valid amd_valid -#define AMD_aat amd_aat -#define AMD_postorder amd_postorder -#define AMD_post_tree amd_post_tree -#define AMD_dump amd_dump -#define AMD_debug amd_debug -#define AMD_debug_init amd_debug_init -#define AMD_preprocess amd_preprocess - -#define amd_malloc xmalloc -#if 0 /* 24/V-2009 */ -#define amd_free xfree -#else -#define amd_free(ptr) { if ((ptr) != NULL) xfree(ptr); } -#endif -#define amd_printf xprintf - -#define PRINTF(params) { amd_printf params; } - -#ifndef NDEBUG -#define ASSERT(expr) xassert(expr) -#define AMD_DEBUG0(params) { PRINTF(params); } -#define AMD_DEBUG1(params) { if (AMD_debug >= 1) PRINTF(params); } -#define AMD_DEBUG2(params) { if (AMD_debug >= 2) PRINTF(params); } -#define AMD_DEBUG3(params) { if (AMD_debug >= 3) PRINTF(params); } -#define AMD_DEBUG4(params) { if (AMD_debug >= 4) PRINTF(params); } -#else -#define ASSERT(expression) -#define AMD_DEBUG0(params) -#define AMD_DEBUG1(params) -#define AMD_DEBUG2(params) -#define AMD_DEBUG3(params) -#define AMD_DEBUG4(params) -#endif - -#define amd_aat _glp_amd_aat -size_t AMD_aat(Int n, const Int Ap[], const Int Ai[], Int Len[], - Int Tp[], double Info[]); - -#define amd_1 _glp_amd_1 -void AMD_1(Int n, const Int Ap[], const Int Ai[], Int P[], Int Pinv[], - Int Len[], Int slen, Int S[], double Control[], double Info[]); - -#define amd_postorder _glp_amd_postorder -void AMD_postorder(Int nn, Int Parent[], Int Npiv[], Int Fsize[], - Int Order[], Int Child[], Int Sibling[], Int Stack[]); - -#define amd_post_tree _glp_amd_post_tree -#ifndef NDEBUG -Int AMD_post_tree(Int root, Int k, Int Child[], const Int Sibling[], - Int Order[], Int Stack[], Int nn); -#else -Int AMD_post_tree(Int root, Int k, Int Child[], const Int Sibling[], - Int Order[], Int Stack[]); -#endif - -#define amd_preprocess _glp_amd_preprocess -void AMD_preprocess(Int n, const Int Ap[], const Int Ai[], Int Rp[], - Int Ri[], Int W[], Int Flag[]); - -#define amd_debug _glp_amd_debug -extern Int AMD_debug; - -#define amd_debug_init _glp_amd_debug_init -void AMD_debug_init(char *s); - -#define amd_dump _glp_amd_dump -void AMD_dump(Int n, Int Pe[], Int Iw[], Int Len[], Int iwlen, - Int pfree, Int Nv[], Int Next[], Int Last[], Int Head[], - Int Elen[], Int Degree[], Int W[], Int nel); - -#endif - -/* eof */ diff --git a/code/3rd_glpk/amd/amd_order.c b/code/3rd_glpk/amd/amd_order.c deleted file mode 100644 index 332d5663..00000000 --- a/code/3rd_glpk/amd/amd_order.c +++ /dev/null @@ -1,200 +0,0 @@ -/* ========================================================================= */ -/* === AMD_order =========================================================== */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* User-callable AMD minimum degree ordering routine. See amd.h for - * documentation. - */ - -#include "amd_internal.h" - -/* ========================================================================= */ -/* === AMD_order =========================================================== */ -/* ========================================================================= */ - -GLOBAL Int AMD_order -( - Int n, - const Int Ap [ ], - const Int Ai [ ], - Int P [ ], - double Control [ ], - double Info [ ] -) -{ - Int *Len, *S, nz, i, *Pinv, info, status, *Rp, *Ri, *Cp, *Ci, ok ; - size_t nzaat, slen ; - double mem = 0 ; - -#ifndef NDEBUG - AMD_debug_init ("amd") ; -#endif - - /* clear the Info array, if it exists */ - info = Info != (double *) NULL ; - if (info) - { - for (i = 0 ; i < AMD_INFO ; i++) - { - Info [i] = EMPTY ; - } - Info [AMD_N] = n ; - Info [AMD_STATUS] = AMD_OK ; - } - - /* make sure inputs exist and n is >= 0 */ - if (Ai == (Int *) NULL || Ap == (Int *) NULL || P == (Int *) NULL || n < 0) - { - if (info) Info [AMD_STATUS] = AMD_INVALID ; - return (AMD_INVALID) ; /* arguments are invalid */ - } - - if (n == 0) - { - return (AMD_OK) ; /* n is 0 so there's nothing to do */ - } - - nz = Ap [n] ; - if (info) - { - Info [AMD_NZ] = nz ; - } - if (nz < 0) - { - if (info) Info [AMD_STATUS] = AMD_INVALID ; - return (AMD_INVALID) ; - } - - /* check if n or nz will cause size_t overflow */ - if (((size_t) n) >= SIZE_T_MAX / sizeof (Int) - || ((size_t) nz) >= SIZE_T_MAX / sizeof (Int)) - { - if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ; - return (AMD_OUT_OF_MEMORY) ; /* problem too large */ - } - - /* check the input matrix: AMD_OK, AMD_INVALID, or AMD_OK_BUT_JUMBLED */ - status = AMD_valid (n, n, Ap, Ai) ; - - if (status == AMD_INVALID) - { - if (info) Info [AMD_STATUS] = AMD_INVALID ; - return (AMD_INVALID) ; /* matrix is invalid */ - } - - /* allocate two size-n integer workspaces */ - Len = amd_malloc (n * sizeof (Int)) ; - Pinv = amd_malloc (n * sizeof (Int)) ; - mem += n ; - mem += n ; - if (!Len || !Pinv) - { - /* :: out of memory :: */ - amd_free (Len) ; - amd_free (Pinv) ; - if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ; - return (AMD_OUT_OF_MEMORY) ; - } - - if (status == AMD_OK_BUT_JUMBLED) - { - /* sort the input matrix and remove duplicate entries */ - AMD_DEBUG1 (("Matrix is jumbled\n")) ; - Rp = amd_malloc ((n+1) * sizeof (Int)) ; - Ri = amd_malloc (MAX (nz,1) * sizeof (Int)) ; - mem += (n+1) ; - mem += MAX (nz,1) ; - if (!Rp || !Ri) - { - /* :: out of memory :: */ - amd_free (Rp) ; - amd_free (Ri) ; - amd_free (Len) ; - amd_free (Pinv) ; - if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ; - return (AMD_OUT_OF_MEMORY) ; - } - /* use Len and Pinv as workspace to create R = A' */ - AMD_preprocess (n, Ap, Ai, Rp, Ri, Len, Pinv) ; - Cp = Rp ; - Ci = Ri ; - } - else - { - /* order the input matrix as-is. No need to compute R = A' first */ - Rp = NULL ; - Ri = NULL ; - Cp = (Int *) Ap ; - Ci = (Int *) Ai ; - } - - /* --------------------------------------------------------------------- */ - /* determine the symmetry and count off-diagonal nonzeros in A+A' */ - /* --------------------------------------------------------------------- */ - - nzaat = AMD_aat (n, Cp, Ci, Len, P, Info) ; - AMD_DEBUG1 (("nzaat: %g\n", (double) nzaat)) ; - ASSERT ((MAX (nz-n, 0) <= nzaat) && (nzaat <= 2 * (size_t) nz)) ; - - /* --------------------------------------------------------------------- */ - /* allocate workspace for matrix, elbow room, and 6 size-n vectors */ - /* --------------------------------------------------------------------- */ - - S = NULL ; - slen = nzaat ; /* space for matrix */ - ok = ((slen + nzaat/5) >= slen) ; /* check for size_t overflow */ - slen += nzaat/5 ; /* add elbow room */ - for (i = 0 ; ok && i < 7 ; i++) - { - ok = ((slen + n) > slen) ; /* check for size_t overflow */ - slen += n ; /* size-n elbow room, 6 size-n work */ - } - mem += slen ; - ok = ok && (slen < SIZE_T_MAX / sizeof (Int)) ; /* check for overflow */ - ok = ok && (slen < Int_MAX) ; /* S[i] for Int i must be OK */ - if (ok) - { - S = amd_malloc (slen * sizeof (Int)) ; - } - AMD_DEBUG1 (("slen %g\n", (double) slen)) ; - if (!S) - { - /* :: out of memory :: (or problem too large) */ - amd_free (Rp) ; - amd_free (Ri) ; - amd_free (Len) ; - amd_free (Pinv) ; - if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ; - return (AMD_OUT_OF_MEMORY) ; - } - if (info) - { - /* memory usage, in bytes. */ - Info [AMD_MEMORY] = mem * sizeof (Int) ; - } - - /* --------------------------------------------------------------------- */ - /* order the matrix */ - /* --------------------------------------------------------------------- */ - - AMD_1 (n, Cp, Ci, P, Pinv, Len, slen, S, Control, Info) ; - - /* --------------------------------------------------------------------- */ - /* free the workspace */ - /* --------------------------------------------------------------------- */ - - amd_free (Rp) ; - amd_free (Ri) ; - amd_free (Len) ; - amd_free (Pinv) ; - amd_free (S) ; - if (info) Info [AMD_STATUS] = status ; - return (status) ; /* successful ordering */ -} diff --git a/code/3rd_glpk/amd/amd_post_tree.c b/code/3rd_glpk/amd/amd_post_tree.c deleted file mode 100644 index bff0e263..00000000 --- a/code/3rd_glpk/amd/amd_post_tree.c +++ /dev/null @@ -1,121 +0,0 @@ -/* ========================================================================= */ -/* === AMD_post_tree ======================================================= */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* Post-ordering of a supernodal elimination tree. */ - -#include "amd_internal.h" - -GLOBAL Int AMD_post_tree -( - Int root, /* root of the tree */ - Int k, /* start numbering at k */ - Int Child [ ], /* input argument of size nn, undefined on - * output. Child [i] is the head of a link - * list of all nodes that are children of node - * i in the tree. */ - const Int Sibling [ ], /* input argument of size nn, not modified. - * If f is a node in the link list of the - * children of node i, then Sibling [f] is the - * next child of node i. - */ - Int Order [ ], /* output order, of size nn. Order [i] = k - * if node i is the kth node of the reordered - * tree. */ - Int Stack [ ] /* workspace of size nn */ -#ifndef NDEBUG - , Int nn /* nodes are in the range 0..nn-1. */ -#endif -) -{ - Int f, head, h, i ; - -#if 0 - /* --------------------------------------------------------------------- */ - /* recursive version (Stack [ ] is not used): */ - /* --------------------------------------------------------------------- */ - - /* this is simple, but can caouse stack overflow if nn is large */ - i = root ; - for (f = Child [i] ; f != EMPTY ; f = Sibling [f]) - { - k = AMD_post_tree (f, k, Child, Sibling, Order, Stack, nn) ; - } - Order [i] = k++ ; - return (k) ; -#endif - - /* --------------------------------------------------------------------- */ - /* non-recursive version, using an explicit stack */ - /* --------------------------------------------------------------------- */ - - /* push root on the stack */ - head = 0 ; - Stack [0] = root ; - - while (head >= 0) - { - /* get head of stack */ - ASSERT (head < nn) ; - i = Stack [head] ; - AMD_DEBUG1 (("head of stack "ID" \n", i)) ; - ASSERT (i >= 0 && i < nn) ; - - if (Child [i] != EMPTY) - { - /* the children of i are not yet ordered */ - /* push each child onto the stack in reverse order */ - /* so that small ones at the head of the list get popped first */ - /* and the biggest one at the end of the list gets popped last */ - for (f = Child [i] ; f != EMPTY ; f = Sibling [f]) - { - head++ ; - ASSERT (head < nn) ; - ASSERT (f >= 0 && f < nn) ; - } - h = head ; - ASSERT (head < nn) ; - for (f = Child [i] ; f != EMPTY ; f = Sibling [f]) - { - ASSERT (h > 0) ; - Stack [h--] = f ; - AMD_DEBUG1 (("push "ID" on stack\n", f)) ; - ASSERT (f >= 0 && f < nn) ; - } - ASSERT (Stack [h] == i) ; - - /* delete child list so that i gets ordered next time we see it */ - Child [i] = EMPTY ; - } - else - { - /* the children of i (if there were any) are already ordered */ - /* remove i from the stack and order it. Front i is kth front */ - head-- ; - AMD_DEBUG1 (("pop "ID" order "ID"\n", i, k)) ; - Order [i] = k++ ; - ASSERT (k <= nn) ; - } - -#ifndef NDEBUG - AMD_DEBUG1 (("\nStack:")) ; - for (h = head ; h >= 0 ; h--) - { - Int j = Stack [h] ; - AMD_DEBUG1 ((" "ID, j)) ; - ASSERT (j >= 0 && j < nn) ; - } - AMD_DEBUG1 (("\n\n")) ; - ASSERT (head < nn) ; -#endif - - } - return (k) ; -} diff --git a/code/3rd_glpk/amd/amd_postorder.c b/code/3rd_glpk/amd/amd_postorder.c deleted file mode 100644 index a3ece915..00000000 --- a/code/3rd_glpk/amd/amd_postorder.c +++ /dev/null @@ -1,207 +0,0 @@ -/* ========================================================================= */ -/* === AMD_postorder ======================================================= */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* Perform a postordering (via depth-first search) of an assembly tree. */ - -#include "amd_internal.h" - -GLOBAL void AMD_postorder -( - /* inputs, not modified on output: */ - Int nn, /* nodes are in the range 0..nn-1 */ - Int Parent [ ], /* Parent [j] is the parent of j, or EMPTY if root */ - Int Nv [ ], /* Nv [j] > 0 number of pivots represented by node j, - * or zero if j is not a node. */ - Int Fsize [ ], /* Fsize [j]: size of node j */ - - /* output, not defined on input: */ - Int Order [ ], /* output post-order */ - - /* workspaces of size nn: */ - Int Child [ ], - Int Sibling [ ], - Int Stack [ ] -) -{ - Int i, j, k, parent, frsize, f, fprev, maxfrsize, bigfprev, bigf, fnext ; - - for (j = 0 ; j < nn ; j++) - { - Child [j] = EMPTY ; - Sibling [j] = EMPTY ; - } - - /* --------------------------------------------------------------------- */ - /* place the children in link lists - bigger elements tend to be last */ - /* --------------------------------------------------------------------- */ - - for (j = nn-1 ; j >= 0 ; j--) - { - if (Nv [j] > 0) - { - /* this is an element */ - parent = Parent [j] ; - if (parent != EMPTY) - { - /* place the element in link list of the children its parent */ - /* bigger elements will tend to be at the end of the list */ - Sibling [j] = Child [parent] ; - Child [parent] = j ; - } - } - } - -#ifndef NDEBUG - { - Int nels, ff, nchild ; - AMD_DEBUG1 (("\n\n================================ AMD_postorder:\n")); - nels = 0 ; - for (j = 0 ; j < nn ; j++) - { - if (Nv [j] > 0) - { - AMD_DEBUG1 (( ""ID" : nels "ID" npiv "ID" size "ID - " parent "ID" maxfr "ID"\n", j, nels, - Nv [j], Fsize [j], Parent [j], Fsize [j])) ; - /* this is an element */ - /* dump the link list of children */ - nchild = 0 ; - AMD_DEBUG1 ((" Children: ")) ; - for (ff = Child [j] ; ff != EMPTY ; ff = Sibling [ff]) - { - AMD_DEBUG1 ((ID" ", ff)) ; - ASSERT (Parent [ff] == j) ; - nchild++ ; - ASSERT (nchild < nn) ; - } - AMD_DEBUG1 (("\n")) ; - parent = Parent [j] ; - if (parent != EMPTY) - { - ASSERT (Nv [parent] > 0) ; - } - nels++ ; - } - } - } - AMD_DEBUG1 (("\n\nGo through the children of each node, and put\n" - "the biggest child last in each list:\n")) ; -#endif - - /* --------------------------------------------------------------------- */ - /* place the largest child last in the list of children for each node */ - /* --------------------------------------------------------------------- */ - - for (i = 0 ; i < nn ; i++) - { - if (Nv [i] > 0 && Child [i] != EMPTY) - { - -#ifndef NDEBUG - Int nchild ; - AMD_DEBUG1 (("Before partial sort, element "ID"\n", i)) ; - nchild = 0 ; - for (f = Child [i] ; f != EMPTY ; f = Sibling [f]) - { - ASSERT (f >= 0 && f < nn) ; - AMD_DEBUG1 ((" f: "ID" size: "ID"\n", f, Fsize [f])) ; - nchild++ ; - ASSERT (nchild <= nn) ; - } -#endif - - /* find the biggest element in the child list */ - fprev = EMPTY ; - maxfrsize = EMPTY ; - bigfprev = EMPTY ; - bigf = EMPTY ; - for (f = Child [i] ; f != EMPTY ; f = Sibling [f]) - { - ASSERT (f >= 0 && f < nn) ; - frsize = Fsize [f] ; - if (frsize >= maxfrsize) - { - /* this is the biggest seen so far */ - maxfrsize = frsize ; - bigfprev = fprev ; - bigf = f ; - } - fprev = f ; - } - ASSERT (bigf != EMPTY) ; - - fnext = Sibling [bigf] ; - - AMD_DEBUG1 (("bigf "ID" maxfrsize "ID" bigfprev "ID" fnext "ID - " fprev " ID"\n", bigf, maxfrsize, bigfprev, fnext, fprev)) ; - - if (fnext != EMPTY) - { - /* if fnext is EMPTY then bigf is already at the end of list */ - - if (bigfprev == EMPTY) - { - /* delete bigf from the element of the list */ - Child [i] = fnext ; - } - else - { - /* delete bigf from the middle of the list */ - Sibling [bigfprev] = fnext ; - } - - /* put bigf at the end of the list */ - Sibling [bigf] = EMPTY ; - ASSERT (Child [i] != EMPTY) ; - ASSERT (fprev != bigf) ; - ASSERT (fprev != EMPTY) ; - Sibling [fprev] = bigf ; - } - -#ifndef NDEBUG - AMD_DEBUG1 (("After partial sort, element "ID"\n", i)) ; - for (f = Child [i] ; f != EMPTY ; f = Sibling [f]) - { - ASSERT (f >= 0 && f < nn) ; - AMD_DEBUG1 ((" "ID" "ID"\n", f, Fsize [f])) ; - ASSERT (Nv [f] > 0) ; - nchild-- ; - } - ASSERT (nchild == 0) ; -#endif - - } - } - - /* --------------------------------------------------------------------- */ - /* postorder the assembly tree */ - /* --------------------------------------------------------------------- */ - - for (i = 0 ; i < nn ; i++) - { - Order [i] = EMPTY ; - } - - k = 0 ; - - for (i = 0 ; i < nn ; i++) - { - if (Parent [i] == EMPTY && Nv [i] > 0) - { - AMD_DEBUG1 (("Root of assembly tree "ID"\n", i)) ; - k = AMD_post_tree (i, k, Child, Sibling, Order, Stack -#ifndef NDEBUG - , nn -#endif - ) ; - } - } -} diff --git a/code/3rd_glpk/amd/amd_preprocess.c b/code/3rd_glpk/amd/amd_preprocess.c deleted file mode 100644 index fc223fb5..00000000 --- a/code/3rd_glpk/amd/amd_preprocess.c +++ /dev/null @@ -1,119 +0,0 @@ -/* ========================================================================= */ -/* === AMD_preprocess ====================================================== */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* Sorts, removes duplicate entries, and transposes from the nonzero pattern of - * a column-form matrix A, to obtain the matrix R. The input matrix can have - * duplicate entries and/or unsorted columns (AMD_valid (n,Ap,Ai) must not be - * AMD_INVALID). - * - * This input condition is NOT checked. This routine is not user-callable. - */ - -#include "amd_internal.h" - -/* ========================================================================= */ -/* === AMD_preprocess ====================================================== */ -/* ========================================================================= */ - -/* AMD_preprocess does not check its input for errors or allocate workspace. - * On input, the condition (AMD_valid (n,n,Ap,Ai) != AMD_INVALID) must hold. - */ - -GLOBAL void AMD_preprocess -( - Int n, /* input matrix: A is n-by-n */ - const Int Ap [ ], /* size n+1 */ - const Int Ai [ ], /* size nz = Ap [n] */ - - /* output matrix R: */ - Int Rp [ ], /* size n+1 */ - Int Ri [ ], /* size nz (or less, if duplicates present) */ - - Int W [ ], /* workspace of size n */ - Int Flag [ ] /* workspace of size n */ -) -{ - - /* --------------------------------------------------------------------- */ - /* local variables */ - /* --------------------------------------------------------------------- */ - - Int i, j, p, p2 ; - - ASSERT (AMD_valid (n, n, Ap, Ai) != AMD_INVALID) ; - - /* --------------------------------------------------------------------- */ - /* count the entries in each row of A (excluding duplicates) */ - /* --------------------------------------------------------------------- */ - - for (i = 0 ; i < n ; i++) - { - W [i] = 0 ; /* # of nonzeros in row i (excl duplicates) */ - Flag [i] = EMPTY ; /* Flag [i] = j if i appears in column j */ - } - for (j = 0 ; j < n ; j++) - { - p2 = Ap [j+1] ; - for (p = Ap [j] ; p < p2 ; p++) - { - i = Ai [p] ; - if (Flag [i] != j) - { - /* row index i has not yet appeared in column j */ - W [i]++ ; /* one more entry in row i */ - Flag [i] = j ; /* flag row index i as appearing in col j*/ - } - } - } - - /* --------------------------------------------------------------------- */ - /* compute the row pointers for R */ - /* --------------------------------------------------------------------- */ - - Rp [0] = 0 ; - for (i = 0 ; i < n ; i++) - { - Rp [i+1] = Rp [i] + W [i] ; - } - for (i = 0 ; i < n ; i++) - { - W [i] = Rp [i] ; - Flag [i] = EMPTY ; - } - - /* --------------------------------------------------------------------- */ - /* construct the row form matrix R */ - /* --------------------------------------------------------------------- */ - - /* R = row form of pattern of A */ - for (j = 0 ; j < n ; j++) - { - p2 = Ap [j+1] ; - for (p = Ap [j] ; p < p2 ; p++) - { - i = Ai [p] ; - if (Flag [i] != j) - { - /* row index i has not yet appeared in column j */ - Ri [W [i]++] = j ; /* put col j in row i */ - Flag [i] = j ; /* flag row index i as appearing in col j*/ - } - } - } - -#ifndef NDEBUG - ASSERT (AMD_valid (n, n, Rp, Ri) == AMD_OK) ; - for (j = 0 ; j < n ; j++) - { - ASSERT (W [j] == Rp [j+1]) ; - } -#endif -} diff --git a/code/3rd_glpk/amd/amd_valid.c b/code/3rd_glpk/amd/amd_valid.c deleted file mode 100644 index e9e2e5ab..00000000 --- a/code/3rd_glpk/amd/amd_valid.c +++ /dev/null @@ -1,93 +0,0 @@ -/* ========================================================================= */ -/* === AMD_valid =========================================================== */ -/* ========================================================================= */ - -/* ------------------------------------------------------------------------- */ -/* AMD, Copyright (c) Timothy A. Davis, */ -/* Patrick R. Amestoy, and Iain S. Duff. See ../README.txt for License. */ -/* email: davis at cise.ufl.edu CISE Department, Univ. of Florida. */ -/* web: http://www.cise.ufl.edu/research/sparse/amd */ -/* ------------------------------------------------------------------------- */ - -/* Check if a column-form matrix is valid or not. The matrix A is - * n_row-by-n_col. The row indices of entries in column j are in - * Ai [Ap [j] ... Ap [j+1]-1]. Required conditions are: - * - * n_row >= 0 - * n_col >= 0 - * nz = Ap [n_col] >= 0 number of entries in the matrix - * Ap [0] == 0 - * Ap [j] <= Ap [j+1] for all j in the range 0 to n_col. - * Ai [0 ... nz-1] must be in the range 0 to n_row-1. - * - * If any of the above conditions hold, AMD_INVALID is returned. If the - * following condition holds, AMD_OK_BUT_JUMBLED is returned (a warning, - * not an error): - * - * row indices in Ai [Ap [j] ... Ap [j+1]-1] are not sorted in ascending - * order, and/or duplicate entries exist. - * - * Otherwise, AMD_OK is returned. - * - * In v1.2 and earlier, this function returned TRUE if the matrix was valid - * (now returns AMD_OK), or FALSE otherwise (now returns AMD_INVALID or - * AMD_OK_BUT_JUMBLED). - */ - -#include "amd_internal.h" - -GLOBAL Int AMD_valid -( - /* inputs, not modified on output: */ - Int n_row, /* A is n_row-by-n_col */ - Int n_col, - const Int Ap [ ], /* column pointers of A, of size n_col+1 */ - const Int Ai [ ] /* row indices of A, of size nz = Ap [n_col] */ -) -{ - Int nz, j, p1, p2, ilast, i, p, result = AMD_OK ; - - if (n_row < 0 || n_col < 0 || Ap == NULL || Ai == NULL) - { - return (AMD_INVALID) ; - } - nz = Ap [n_col] ; - if (Ap [0] != 0 || nz < 0) - { - /* column pointers must start at Ap [0] = 0, and Ap [n] must be >= 0 */ - AMD_DEBUG0 (("column 0 pointer bad or nz < 0\n")) ; - return (AMD_INVALID) ; - } - for (j = 0 ; j < n_col ; j++) - { - p1 = Ap [j] ; - p2 = Ap [j+1] ; - AMD_DEBUG2 (("\nColumn: "ID" p1: "ID" p2: "ID"\n", j, p1, p2)) ; - if (p1 > p2) - { - /* column pointers must be ascending */ - AMD_DEBUG0 (("column "ID" pointer bad\n", j)) ; - return (AMD_INVALID) ; - } - ilast = EMPTY ; - for (p = p1 ; p < p2 ; p++) - { - i = Ai [p] ; - AMD_DEBUG3 (("row: "ID"\n", i)) ; - if (i < 0 || i >= n_row) - { - /* row index out of range */ - AMD_DEBUG0 (("index out of range, col "ID" row "ID"\n", j, i)); - return (AMD_INVALID) ; - } - if (i <= ilast) - { - /* row index unsorted, or duplicate entry present */ - AMD_DEBUG1 (("index unsorted/dupl col "ID" row "ID"\n", j, i)); - result = AMD_OK_BUT_JUMBLED ; - } - ilast = i ; - } - } - return (result) ; -} diff --git a/code/3rd_glpk/api/advbas.c b/code/3rd_glpk/api/advbas.c deleted file mode 100644 index 23067624..00000000 --- a/code/3rd_glpk/api/advbas.c +++ /dev/null @@ -1,155 +0,0 @@ -/* advbas.c (construct advanced initial LP basis) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" -#include "triang.h" - -/*********************************************************************** -* NAME -* -* glp_adv_basis - construct advanced initial LP basis -* -* SYNOPSIS -* -* void glp_adv_basis(glp_prob *P, int flags); -* -* DESCRIPTION -* -* The routine glp_adv_basis constructs an advanced initial LP basis -* for the specified problem object. -* -* The parameter flag is reserved for use in the future and should be -* specified as zero. -* -* NOTE -* -* The routine glp_adv_basis should be called after the constraint -* matrix has been scaled (if scaling is used). */ - -static int mat(void *info, int k, int ind[], double val[]) -{ glp_prob *P = info; - int m = P->m; - int n = P->n; - GLPROW **row = P->row; - GLPCOL **col = P->col; - GLPAIJ *aij; - int i, j, len; - if (k > 0) - { /* retrieve scaled row of constraint matrix */ - i = +k; - xassert(1 <= i && i <= m); - len = 0; - if (row[i]->type == GLP_FX) - { for (aij = row[i]->ptr; aij != NULL; aij = aij->r_next) - { j = aij->col->j; - if (col[j]->type != GLP_FX) - { len++; - ind[len] = j; - val[len] = aij->row->rii * aij->val * aij->col->sjj; - } - } - } - } - else - { /* retrieve scaled column of constraint matrix */ - j = -k; - xassert(1 <= j && j <= n); - len = 0; - if (col[j]->type != GLP_FX) - { for (aij = col[j]->ptr; aij != NULL; aij = aij->c_next) - { i = aij->row->i; - if (row[i]->type == GLP_FX) - { len++; - ind[len] = i; - val[len] = aij->row->rii * aij->val * aij->col->sjj; - } - } - } - } - return len; -} - -void glp_adv_basis(glp_prob *P, int flags) -{ int i, j, k, m, n, min_mn, size, *rn, *cn; - char *flag; - if (flags != 0) - xerror("glp_adv_basis: flags = %d; invalid flags\n", flags); - m = P->m; /* number of rows */ - n = P->n; /* number of columns */ - if (m == 0 || n == 0) - { /* trivial case */ - glp_std_basis(P); - goto done; - } - xprintf("Constructing initial basis...\n"); - /* allocate working arrays */ - min_mn = (m < n ? m : n); - rn = talloc(1+min_mn, int); - cn = talloc(1+min_mn, int); - flag = talloc(1+m, char); - /* make the basis empty */ - for (i = 1; i <= m; i++) - { flag[i] = 0; - glp_set_row_stat(P, i, GLP_NS); - } - for (j = 1; j <= n; j++) - glp_set_col_stat(P, j, GLP_NS); - /* find maximal triangular part of the constraint matrix; - to prevent including non-fixed rows and fixed columns in the - triangular part, such rows and columns are temporarily made - empty by the routine mat */ -#if 1 /* FIXME: tolerance */ - size = triang(m, n, mat, P, 0.001, rn, cn); -#endif - xassert(0 <= size && size <= min_mn); - /* include in the basis non-fixed structural variables, whose - columns constitute the triangular part */ - for (k = 1; k <= size; k++) - { i = rn[k]; - xassert(1 <= i && i <= m); - flag[i] = 1; - j = cn[k]; - xassert(1 <= j && j <= n); - glp_set_col_stat(P, j, GLP_BS); - } - /* include in the basis appropriate auxiliary variables, whose - unity columns preserve triangular form of the basis matrix */ - for (i = 1; i <= m; i++) - { if (flag[i] == 0) - { glp_set_row_stat(P, i, GLP_BS); - if (P->row[i]->type != GLP_FX) - size++; - } - } - /* size of triangular part = (number of rows) - (number of basic - fixed auxiliary variables) */ - xprintf("Size of triangular part is %d\n", size); - /* deallocate working arrays */ - tfree(rn); - tfree(cn); - tfree(flag); -done: return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/asnhall.c b/code/3rd_glpk/api/asnhall.c deleted file mode 100644 index d7112a10..00000000 --- a/code/3rd_glpk/api/asnhall.c +++ /dev/null @@ -1,163 +0,0 @@ -/* asnhall.c (find bipartite matching of maximum cardinality) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "mc21a.h" - -/*********************************************************************** -* NAME -* -* glp_asnprob_hall - find bipartite matching of maximum cardinality -* -* SYNOPSIS -* -* int glp_asnprob_hall(glp_graph *G, int v_set, int a_x); -* -* DESCRIPTION -* -* The routine glp_asnprob_hall finds a matching of maximal cardinality -* in the specified bipartite graph G. It uses a version of the Fortran -* routine MC21A developed by I.S.Duff [1], which implements Hall's -* algorithm [2]. -* -* RETURNS -* -* The routine glp_asnprob_hall returns the cardinality of the matching -* found. However, if the specified graph is incorrect (as detected by -* the routine glp_check_asnprob), the routine returns negative value. -* -* REFERENCES -* -* 1. I.S.Duff, Algorithm 575: Permutations for zero-free diagonal, ACM -* Trans. on Math. Softw. 7 (1981), 387-390. -* -* 2. M.Hall, "An Algorithm for distinct representatives," Amer. Math. -* Monthly 63 (1956), 716-717. */ - -int glp_asnprob_hall(glp_graph *G, int v_set, int a_x) -{ glp_vertex *v; - glp_arc *a; - int card, i, k, loc, n, n1, n2, xij; - int *num, *icn, *ip, *lenr, *iperm, *pr, *arp, *cv, *out; - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_asnprob_hall: v_set = %d; invalid offset\n", - v_set); - if (a_x >= 0 && a_x > G->a_size - (int)sizeof(int)) - xerror("glp_asnprob_hall: a_x = %d; invalid offset\n", a_x); - if (glp_check_asnprob(G, v_set)) - return -1; - /* determine the number of vertices in sets R and S and renumber - vertices in S which correspond to columns of the matrix; skip - all isolated vertices */ - num = xcalloc(1+G->nv, sizeof(int)); - n1 = n2 = 0; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (v->in == NULL && v->out != NULL) - n1++, num[i] = 0; /* vertex in R */ - else if (v->in != NULL && v->out == NULL) - n2++, num[i] = n2; /* vertex in S */ - else - { xassert(v->in == NULL && v->out == NULL); - num[i] = -1; /* isolated vertex */ - } - } - /* the matrix must be square, thus, if it has more columns than - rows, extra rows will be just empty, and vice versa */ - n = (n1 >= n2 ? n1 : n2); - /* allocate working arrays */ - icn = xcalloc(1+G->na, sizeof(int)); - ip = xcalloc(1+n, sizeof(int)); - lenr = xcalloc(1+n, sizeof(int)); - iperm = xcalloc(1+n, sizeof(int)); - pr = xcalloc(1+n, sizeof(int)); - arp = xcalloc(1+n, sizeof(int)); - cv = xcalloc(1+n, sizeof(int)); - out = xcalloc(1+n, sizeof(int)); - /* build the adjacency matrix of the bipartite graph in row-wise - format (rows are vertices in R, columns are vertices in S) */ - k = 0, loc = 1; - for (i = 1; i <= G->nv; i++) - { if (num[i] != 0) continue; - /* vertex i in R */ - ip[++k] = loc; - v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { xassert(num[a->head->i] != 0); - icn[loc++] = num[a->head->i]; - } - lenr[k] = loc - ip[k]; - } - xassert(loc-1 == G->na); - /* make all extra rows empty (all extra columns are empty due to - the row-wise format used) */ - for (k++; k <= n; k++) - ip[k] = loc, lenr[k] = 0; - /* find a row permutation that maximizes the number of non-zeros - on the main diagonal */ - card = mc21a(n, icn, ip, lenr, iperm, pr, arp, cv, out); -#if 1 /* 18/II-2010 */ - /* FIXED: if card = n, arp remains clobbered on exit */ - for (i = 1; i <= n; i++) - arp[i] = 0; - for (i = 1; i <= card; i++) - { k = iperm[i]; - xassert(1 <= k && k <= n); - xassert(arp[k] == 0); - arp[k] = i; - } -#endif - /* store solution, if necessary */ - if (a_x < 0) goto skip; - k = 0; - for (i = 1; i <= G->nv; i++) - { if (num[i] != 0) continue; - /* vertex i in R */ - k++; - v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { /* arp[k] is the number of matched column or zero */ - if (arp[k] == num[a->head->i]) - { xassert(arp[k] != 0); - xij = 1; - } - else - xij = 0; - memcpy((char *)a->data + a_x, &xij, sizeof(int)); - } - } -skip: /* free working arrays */ - xfree(num); - xfree(icn); - xfree(ip); - xfree(lenr); - xfree(iperm); - xfree(pr); - xfree(arp); - xfree(cv); - xfree(out); - return card; -} - -/* eof */ diff --git a/code/3rd_glpk/api/asnlp.c b/code/3rd_glpk/api/asnlp.c deleted file mode 100644 index cfa925d0..00000000 --- a/code/3rd_glpk/api/asnlp.c +++ /dev/null @@ -1,104 +0,0 @@ -/* asnlp.c (convert assignment problem to LP) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_asnprob_lp - convert assignment problem to LP -* -* SYNOPSIS -* -* int glp_asnprob_lp(glp_prob *P, int form, glp_graph *G, int names, -* int v_set, int a_cost); -* -* DESCRIPTION -* -* The routine glp_asnprob_lp builds an LP problem, which corresponds -* to the assignment problem on the specified graph G. -* -* RETURNS -* -* If the LP problem has been successfully built, the routine returns -* zero, otherwise, non-zero. */ - -int glp_asnprob_lp(glp_prob *P, int form, glp_graph *G, int names, - int v_set, int a_cost) -{ glp_vertex *v; - glp_arc *a; - int i, j, ret, ind[1+2]; - double cost, val[1+2]; - if (!(form == GLP_ASN_MIN || form == GLP_ASN_MAX || - form == GLP_ASN_MMP)) - xerror("glp_asnprob_lp: form = %d; invalid parameter\n", - form); - if (!(names == GLP_ON || names == GLP_OFF)) - xerror("glp_asnprob_lp: names = %d; invalid parameter\n", - names); - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_asnprob_lp: v_set = %d; invalid offset\n", - v_set); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_asnprob_lp: a_cost = %d; invalid offset\n", - a_cost); - ret = glp_check_asnprob(G, v_set); - if (ret != 0) goto done; - glp_erase_prob(P); - if (names) glp_set_prob_name(P, G->name); - glp_set_obj_dir(P, form == GLP_ASN_MIN ? GLP_MIN : GLP_MAX); - if (G->nv > 0) glp_add_rows(P, G->nv); - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (names) glp_set_row_name(P, i, v->name); - glp_set_row_bnds(P, i, form == GLP_ASN_MMP ? GLP_UP : GLP_FX, - 1.0, 1.0); - } - if (G->na > 0) glp_add_cols(P, G->na); - for (i = 1, j = 0; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { j++; - if (names) - { char name[50+1]; - sprintf(name, "x[%d,%d]", a->tail->i, a->head->i); - xassert(strlen(name) < sizeof(name)); - glp_set_col_name(P, j, name); - } - ind[1] = a->tail->i, val[1] = +1.0; - ind[2] = a->head->i, val[2] = +1.0; - glp_set_mat_col(P, j, 2, ind, val); - glp_set_col_bnds(P, j, GLP_DB, 0.0, 1.0); - if (a_cost >= 0) - memcpy(&cost, (char *)a->data + a_cost, sizeof(double)); - else - cost = 1.0; - glp_set_obj_coef(P, j, cost); - } - } - xassert(j == G->na); -done: return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/asnokalg.c b/code/3rd_glpk/api/asnokalg.c deleted file mode 100644 index d55dbac7..00000000 --- a/code/3rd_glpk/api/asnokalg.c +++ /dev/null @@ -1,154 +0,0 @@ -/* asnokalg.c (solve assignment problem with out-of-kilter alg.) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "okalg.h" - -int glp_asnprob_okalg(int form, glp_graph *G, int v_set, int a_cost, - double *sol, int a_x) -{ /* solve assignment problem with out-of-kilter algorithm */ - glp_vertex *v; - glp_arc *a; - int nv, na, i, k, *tail, *head, *low, *cap, *cost, *x, *pi, ret; - double temp; - if (!(form == GLP_ASN_MIN || form == GLP_ASN_MAX || - form == GLP_ASN_MMP)) - xerror("glp_asnprob_okalg: form = %d; invalid parameter\n", - form); - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_asnprob_okalg: v_set = %d; invalid offset\n", - v_set); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_asnprob_okalg: a_cost = %d; invalid offset\n", - a_cost); - if (a_x >= 0 && a_x > G->a_size - (int)sizeof(int)) - xerror("glp_asnprob_okalg: a_x = %d; invalid offset\n", a_x); - if (glp_check_asnprob(G, v_set)) - return GLP_EDATA; - /* nv is the total number of nodes in the resulting network */ - nv = G->nv + 1; - /* na is the total number of arcs in the resulting network */ - na = G->na + G->nv; - /* allocate working arrays */ - tail = xcalloc(1+na, sizeof(int)); - head = xcalloc(1+na, sizeof(int)); - low = xcalloc(1+na, sizeof(int)); - cap = xcalloc(1+na, sizeof(int)); - cost = xcalloc(1+na, sizeof(int)); - x = xcalloc(1+na, sizeof(int)); - pi = xcalloc(1+nv, sizeof(int)); - /* construct the resulting network */ - k = 0; - /* (original arcs) */ - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { k++; - tail[k] = a->tail->i; - head[k] = a->head->i; - low[k] = 0; - cap[k] = 1; - if (a_cost >= 0) - memcpy(&temp, (char *)a->data + a_cost, sizeof(double)); - else - temp = 1.0; - if (!(fabs(temp) <= (double)INT_MAX && temp == floor(temp))) - { ret = GLP_EDATA; - goto done; - } - cost[k] = (int)temp; - if (form != GLP_ASN_MIN) cost[k] = - cost[k]; - } - } - /* (artificial arcs) */ - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - k++; - if (v->out == NULL) - tail[k] = i, head[k] = nv; - else if (v->in == NULL) - tail[k] = nv, head[k] = i; - else - xassert(v != v); - low[k] = (form == GLP_ASN_MMP ? 0 : 1); - cap[k] = 1; - cost[k] = 0; - } - xassert(k == na); - /* find minimal-cost circulation in the resulting network */ - ret = okalg(nv, na, tail, head, low, cap, cost, x, pi); - switch (ret) - { case 0: - /* optimal circulation found */ - ret = 0; - break; - case 1: - /* no feasible circulation exists */ - ret = GLP_ENOPFS; - break; - case 2: - /* integer overflow occured */ - ret = GLP_ERANGE; - goto done; - case 3: - /* optimality test failed (logic error) */ - ret = GLP_EFAIL; - goto done; - default: - xassert(ret != ret); - } - /* store solution components */ - /* (objective function = the total cost) */ - if (sol != NULL) - { temp = 0.0; - for (k = 1; k <= na; k++) - temp += (double)cost[k] * (double)x[k]; - if (form != GLP_ASN_MIN) temp = - temp; - *sol = temp; - } - /* (arc flows) */ - if (a_x >= 0) - { k = 0; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { k++; - if (ret == 0) - xassert(x[k] == 0 || x[k] == 1); - memcpy((char *)a->data + a_x, &x[k], sizeof(int)); - } - } - } -done: /* free working arrays */ - xfree(tail); - xfree(head); - xfree(low); - xfree(cap); - xfree(cost); - xfree(x); - xfree(pi); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/ckasn.c b/code/3rd_glpk/api/ckasn.c deleted file mode 100644 index 56221a8a..00000000 --- a/code/3rd_glpk/api/ckasn.c +++ /dev/null @@ -1,78 +0,0 @@ -/* ckasn.c (check correctness of assignment problem data) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_check_asnprob - check correctness of assignment problem data -* -* SYNOPSIS -* -* int glp_check_asnprob(glp_graph *G, int v_set); -* -* RETURNS -* -* If the specified assignment problem data are correct, the routine -* glp_check_asnprob returns zero, otherwise, non-zero. */ - -int glp_check_asnprob(glp_graph *G, int v_set) -{ glp_vertex *v; - int i, k, ret = 0; - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_check_asnprob: v_set = %d; invalid offset\n", - v_set); - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (v_set >= 0) - { memcpy(&k, (char *)v->data + v_set, sizeof(int)); - if (k == 0) - { if (v->in != NULL) - { ret = 1; - break; - } - } - else if (k == 1) - { if (v->out != NULL) - { ret = 2; - break; - } - } - else - { ret = 3; - break; - } - } - else - { if (v->in != NULL && v->out != NULL) - { ret = 4; - break; - } - } - } - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/ckcnf.c b/code/3rd_glpk/api/ckcnf.c deleted file mode 100644 index 0ee47ed9..00000000 --- a/code/3rd_glpk/api/ckcnf.c +++ /dev/null @@ -1,82 +0,0 @@ -/* ckcnf.c (check for CNF-SAT problem instance) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -int glp_check_cnfsat(glp_prob *P) -{ /* check for CNF-SAT problem instance */ - int m = P->m; - int n = P->n; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int i, j, neg; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_check_cnfsat: P = %p; invalid problem object\n", - P); -#endif - /* check columns */ - for (j = 1; j <= n; j++) - { col = P->col[j]; - /* the variable should be binary */ - if (!(col->kind == GLP_IV && col->type == GLP_DB && - col->lb == 0.0 && col->ub == 1.0)) - return 1; - } - /* objective function should be zero */ - if (P->c0 != 0.0) - return 2; - for (j = 1; j <= n; j++) - { col = P->col[j]; - if (col->coef != 0.0) - return 3; - } - /* check rows */ - for (i = 1; i <= m; i++) - { row = P->row[i]; - /* the row should be of ">=" type */ - if (row->type != GLP_LO) - return 4; - /* check constraint coefficients */ - neg = 0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { /* the constraint coefficient should be +1 or -1 */ - if (aij->val == +1.0) - ; - else if (aij->val == -1.0) - neg++; - else - return 5; - } - /* the right-hand side should be (1 - neg), where neg is the - number of negative constraint coefficients in the row */ - if (row->lb != (double)(1 - neg)) - return 6; - } - /* congratulations; this is CNF-SAT */ - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/api/cplex.c b/code/3rd_glpk/api/cplex.c deleted file mode 100644 index 8403a646..00000000 --- a/code/3rd_glpk/api/cplex.c +++ /dev/null @@ -1,1283 +0,0 @@ -/* cplex.c (CPLEX LP format routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "misc.h" -#include "prob.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_init_cpxcp - initialize CPLEX LP format control parameters -* -* SYNOPSIS -* -* void glp_init_cpxcp(glp_cpxcp *parm): -* -* The routine glp_init_cpxcp initializes control parameters used by -* the CPLEX LP input/output routines glp_read_lp and glp_write_lp with -* default values. -* -* Default values of the control parameters are stored in the glp_cpxcp -* structure, which the parameter parm points to. */ - -void glp_init_cpxcp(glp_cpxcp *parm) -{ xassert(parm != NULL); - return; -} - -static void check_parm(const char *func, const glp_cpxcp *parm) -{ /* check control parameters */ - xassert(func != NULL); - xassert(parm != NULL); - return; -} - -/*********************************************************************** -* NAME -* -* glp_read_lp - read problem data in CPLEX LP format -* -* SYNOPSIS -* -* int glp_read_lp(glp_prob *P, const glp_cpxcp *parm, const char -* *fname); -* -* DESCRIPTION -* -* The routine glp_read_lp reads problem data in CPLEX LP format from -* a text file. -* -* The parameter parm is a pointer to the structure glp_cpxcp, which -* specifies control parameters used by the routine. If parm is NULL, -* the routine uses default settings. -* -* The character string fname specifies a name of the text file to be -* read. -* -* Note that before reading data the current content of the problem -* object is completely erased with the routine glp_erase_prob. -* -* RETURNS -* -* If the operation was successful, the routine glp_read_lp returns -* zero. Otherwise, it prints an error message and returns non-zero. */ - -struct csa -{ /* common storage area */ - glp_prob *P; - /* LP/MIP problem object */ - const glp_cpxcp *parm; - /* pointer to control parameters */ - const char *fname; - /* name of input CPLEX LP file */ - glp_file *fp; - /* stream assigned to input CPLEX LP file */ - jmp_buf jump; - /* label for go to in case of error */ - int count; - /* line count */ - int c; - /* current character or EOF */ - int token; - /* current token: */ -#define T_EOF 0x00 /* end of file */ -#define T_MINIMIZE 0x01 /* keyword 'minimize' */ -#define T_MAXIMIZE 0x02 /* keyword 'maximize' */ -#define T_SUBJECT_TO 0x03 /* keyword 'subject to' */ -#define T_BOUNDS 0x04 /* keyword 'bounds' */ -#define T_GENERAL 0x05 /* keyword 'general' */ -#define T_INTEGER 0x06 /* keyword 'integer' */ -#define T_BINARY 0x07 /* keyword 'binary' */ -#define T_END 0x08 /* keyword 'end' */ -#define T_NAME 0x09 /* symbolic name */ -#define T_NUMBER 0x0A /* numeric constant */ -#define T_PLUS 0x0B /* delimiter '+' */ -#define T_MINUS 0x0C /* delimiter '-' */ -#define T_COLON 0x0D /* delimiter ':' */ -#define T_LE 0x0E /* delimiter '<=' */ -#define T_GE 0x0F /* delimiter '>=' */ -#define T_EQ 0x10 /* delimiter '=' */ - char image[255+1]; - /* image of current token */ - int imlen; - /* length of token image */ - double value; - /* value of numeric constant */ - int n_max; - /* length of the following five arrays (enlarged automatically, - if necessary) */ - int *ind; /* int ind[1+n_max]; */ - double *val; /* double val[1+n_max]; */ - char *flag; /* char flag[1+n_max]; */ - /* working arrays used to construct linear forms */ - double *lb; /* double lb[1+n_max]; */ - double *ub; /* double ub[1+n_max]; */ - /* lower and upper bounds of variables (columns) */ -#if 1 /* 27/VII-2013 */ - int lb_warn, ub_warn; - /* warning 'lower/upper bound redefined' already issued */ -#endif -}; - -#define CHAR_SET "!\"#$%&()/,.;?@_`'{}|~" -/* characters that may appear in symbolic names */ - -static void error(struct csa *csa, const char *fmt, ...) -{ /* print error message and terminate processing */ - va_list arg; - xprintf("%s:%d: ", csa->fname, csa->count); - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - longjmp(csa->jump, 1); - /* no return */ -} - -static void warning(struct csa *csa, const char *fmt, ...) -{ /* print warning message and continue processing */ - va_list arg; - xprintf("%s:%d: warning: ", csa->fname, csa->count); - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - return; -} - -static void read_char(struct csa *csa) -{ /* read next character from input file */ - int c; - xassert(csa->c != EOF); - if (csa->c == '\n') csa->count++; - c = glp_getc(csa->fp); - if (c < 0) - { if (glp_ioerr(csa->fp)) - error(csa, "read error - %s\n", get_err_msg()); - else if (csa->c == '\n') - { csa->count--; - c = EOF; - } - else - { warning(csa, "missing final end of line\n"); - c = '\n'; - } - } - else if (c == '\n') - ; - else if (isspace(c)) - c = ' '; - else if (iscntrl(c)) - error(csa, "invalid control character 0x%02X\n", c); - csa->c = c; - return; -} - -static void add_char(struct csa *csa) -{ /* append current character to current token */ - if (csa->imlen == sizeof(csa->image)-1) - error(csa, "token '%.15s...' too long\n", csa->image); - csa->image[csa->imlen++] = (char)csa->c; - csa->image[csa->imlen] = '\0'; - read_char(csa); - return; -} - -static int the_same(char *s1, char *s2) -{ /* compare two character strings ignoring case sensitivity */ - for (; *s1 != '\0'; s1++, s2++) - { if (tolower((unsigned char)*s1) != tolower((unsigned char)*s2)) - return 0; - } - return 1; -} - -static void scan_token(struct csa *csa) -{ /* scan next token */ - int flag; - csa->token = -1; - csa->image[0] = '\0'; - csa->imlen = 0; - csa->value = 0.0; -loop: flag = 0; - /* skip non-significant characters */ - while (csa->c == ' ') read_char(csa); - /* recognize and scan current token */ - if (csa->c == EOF) - csa->token = T_EOF; - else if (csa->c == '\n') - { read_char(csa); - /* if the next character is letter, it may begin a keyword */ - if (isalpha(csa->c)) - { flag = 1; - goto name; - } - goto loop; - } - else if (csa->c == '\\') - { /* comment; ignore everything until end-of-line */ - while (csa->c != '\n') read_char(csa); - goto loop; - } - else if (isalpha(csa->c) || csa->c != '.' && strchr(CHAR_SET, - csa->c) != NULL) -name: { /* symbolic name */ - csa->token = T_NAME; - while (isalnum(csa->c) || strchr(CHAR_SET, csa->c) != NULL) - add_char(csa); - if (flag) - { /* check for keyword */ - if (the_same(csa->image, "minimize")) - csa->token = T_MINIMIZE; - else if (the_same(csa->image, "minimum")) - csa->token = T_MINIMIZE; - else if (the_same(csa->image, "min")) - csa->token = T_MINIMIZE; - else if (the_same(csa->image, "maximize")) - csa->token = T_MAXIMIZE; - else if (the_same(csa->image, "maximum")) - csa->token = T_MAXIMIZE; - else if (the_same(csa->image, "max")) - csa->token = T_MAXIMIZE; - else if (the_same(csa->image, "subject")) - { if (csa->c == ' ') - { read_char(csa); - if (tolower(csa->c) == 't') - { csa->token = T_SUBJECT_TO; - csa->image[csa->imlen++] = ' '; - csa->image[csa->imlen] = '\0'; - add_char(csa); - if (tolower(csa->c) != 'o') - error(csa, "keyword 'subject to' incomplete\n"); - add_char(csa); - if (isalpha(csa->c)) - error(csa, "keyword '%s%c...' not recognized\n", - csa->image, csa->c); - } - } - } - else if (the_same(csa->image, "such")) - { if (csa->c == ' ') - { read_char(csa); - if (tolower(csa->c) == 't') - { csa->token = T_SUBJECT_TO; - csa->image[csa->imlen++] = ' '; - csa->image[csa->imlen] = '\0'; - add_char(csa); - if (tolower(csa->c) != 'h') -err: error(csa, "keyword 'such that' incomplete\n"); - add_char(csa); - if (tolower(csa->c) != 'a') goto err; - add_char(csa); - if (tolower(csa->c) != 't') goto err; - add_char(csa); - if (isalpha(csa->c)) - error(csa, "keyword '%s%c...' not recognized\n", - csa->image, csa->c); - } - } - } - else if (the_same(csa->image, "st")) - csa->token = T_SUBJECT_TO; - else if (the_same(csa->image, "s.t.")) - csa->token = T_SUBJECT_TO; - else if (the_same(csa->image, "st.")) - csa->token = T_SUBJECT_TO; - else if (the_same(csa->image, "bounds")) - csa->token = T_BOUNDS; - else if (the_same(csa->image, "bound")) - csa->token = T_BOUNDS; - else if (the_same(csa->image, "general")) - csa->token = T_GENERAL; - else if (the_same(csa->image, "generals")) - csa->token = T_GENERAL; - else if (the_same(csa->image, "gen")) - csa->token = T_GENERAL; - else if (the_same(csa->image, "integer")) - csa->token = T_INTEGER; - else if (the_same(csa->image, "integers")) - csa->token = T_INTEGER; - else if (the_same(csa->image, "int")) - csa->token = T_INTEGER; - else if (the_same(csa->image, "binary")) - csa->token = T_BINARY; - else if (the_same(csa->image, "binaries")) - csa->token = T_BINARY; - else if (the_same(csa->image, "bin")) - csa->token = T_BINARY; - else if (the_same(csa->image, "end")) - csa->token = T_END; - } - } - else if (isdigit(csa->c) || csa->c == '.') - { /* numeric constant */ - csa->token = T_NUMBER; - /* scan integer part */ - while (isdigit(csa->c)) add_char(csa); - /* scan optional fractional part (it is mandatory, if there is - no integer part) */ - if (csa->c == '.') - { add_char(csa); - if (csa->imlen == 1 && !isdigit(csa->c)) - error(csa, "invalid use of decimal point\n"); - while (isdigit(csa->c)) add_char(csa); - } - /* scan optional decimal exponent */ - if (csa->c == 'e' || csa->c == 'E') - { add_char(csa); - if (csa->c == '+' || csa->c == '-') add_char(csa); - if (!isdigit(csa->c)) - error(csa, "numeric constant '%s' incomplete\n", - csa->image); - while (isdigit(csa->c)) add_char(csa); - } - /* convert the numeric constant to floating-point */ - if (str2num(csa->image, &csa->value)) - error(csa, "numeric constant '%s' out of range\n", - csa->image); - } - else if (csa->c == '+') - csa->token = T_PLUS, add_char(csa); - else if (csa->c == '-') - csa->token = T_MINUS, add_char(csa); - else if (csa->c == ':') - csa->token = T_COLON, add_char(csa); - else if (csa->c == '<') - { csa->token = T_LE, add_char(csa); - if (csa->c == '=') add_char(csa); - } - else if (csa->c == '>') - { csa->token = T_GE, add_char(csa); - if (csa->c == '=') add_char(csa); - } - else if (csa->c == '=') - { csa->token = T_EQ, add_char(csa); - if (csa->c == '<') - csa->token = T_LE, add_char(csa); - else if (csa->c == '>') - csa->token = T_GE, add_char(csa); - } - else - error(csa, "character '%c' not recognized\n", csa->c); - /* skip non-significant characters */ - while (csa->c == ' ') read_char(csa); - return; -} - -static int find_col(struct csa *csa, char *name) -{ /* find column by its symbolic name */ - int j; - j = glp_find_col(csa->P, name); - if (j == 0) - { /* not found; create new column */ - j = glp_add_cols(csa->P, 1); - glp_set_col_name(csa->P, j, name); - /* enlarge working arrays, if necessary */ - if (csa->n_max < j) - { int n_max = csa->n_max; - int *ind = csa->ind; - double *val = csa->val; - char *flag = csa->flag; - double *lb = csa->lb; - double *ub = csa->ub; - csa->n_max += csa->n_max; - csa->ind = xcalloc(1+csa->n_max, sizeof(int)); - memcpy(&csa->ind[1], &ind[1], n_max * sizeof(int)); - xfree(ind); - csa->val = xcalloc(1+csa->n_max, sizeof(double)); - memcpy(&csa->val[1], &val[1], n_max * sizeof(double)); - xfree(val); - csa->flag = xcalloc(1+csa->n_max, sizeof(char)); - memset(&csa->flag[1], 0, csa->n_max * sizeof(char)); - memcpy(&csa->flag[1], &flag[1], n_max * sizeof(char)); - xfree(flag); - csa->lb = xcalloc(1+csa->n_max, sizeof(double)); - memcpy(&csa->lb[1], &lb[1], n_max * sizeof(double)); - xfree(lb); - csa->ub = xcalloc(1+csa->n_max, sizeof(double)); - memcpy(&csa->ub[1], &ub[1], n_max * sizeof(double)); - xfree(ub); - } - csa->lb[j] = +DBL_MAX, csa->ub[j] = -DBL_MAX; - } - return j; -} - -/*********************************************************************** -* parse_linear_form - parse linear form -* -* This routine parses the linear form using the following syntax: -* -* ::= -* ::= -* ::= | -* ::= | + | - | -* + | - -* -* The routine returns the number of terms in the linear form. */ - -static int parse_linear_form(struct csa *csa) -{ int j, k, len = 0, newlen; - double s, coef; -loop: /* parse an optional sign */ - if (csa->token == T_PLUS) - s = +1.0, scan_token(csa); - else if (csa->token == T_MINUS) - s = -1.0, scan_token(csa); - else - s = +1.0; - /* parse an optional coefficient */ - if (csa->token == T_NUMBER) - coef = csa->value, scan_token(csa); - else - coef = 1.0; - /* parse a variable name */ - if (csa->token != T_NAME) - error(csa, "missing variable name\n"); - /* find the corresponding column */ - j = find_col(csa, csa->image); - /* check if the variable is already used in the linear form */ - if (csa->flag[j]) - error(csa, "multiple use of variable '%s' not allowed\n", - csa->image); - /* add new term to the linear form */ - len++, csa->ind[len] = j, csa->val[len] = s * coef; - /* and mark that the variable is used in the linear form */ - csa->flag[j] = 1; - scan_token(csa); - /* if the next token is a sign, there is another term */ - if (csa->token == T_PLUS || csa->token == T_MINUS) goto loop; - /* clear marks of the variables used in the linear form */ - for (k = 1; k <= len; k++) csa->flag[csa->ind[k]] = 0; - /* remove zero coefficients */ - newlen = 0; - for (k = 1; k <= len; k++) - { if (csa->val[k] != 0.0) - { newlen++; - csa->ind[newlen] = csa->ind[k]; - csa->val[newlen] = csa->val[k]; - } - } - return newlen; -} - -/*********************************************************************** -* parse_objective - parse objective function -* -* This routine parses definition of the objective function using the -* following syntax: -* -* ::= minimize | minimum | min | maximize | maximum | max -* ::= | : -* ::= */ - -static void parse_objective(struct csa *csa) -{ /* parse objective sense */ - int k, len; - /* parse the keyword 'minimize' or 'maximize' */ - if (csa->token == T_MINIMIZE) - glp_set_obj_dir(csa->P, GLP_MIN); - else if (csa->token == T_MAXIMIZE) - glp_set_obj_dir(csa->P, GLP_MAX); - else - xassert(csa != csa); - scan_token(csa); - /* parse objective name */ - if (csa->token == T_NAME && csa->c == ':') - { /* objective name is followed by a colon */ - glp_set_obj_name(csa->P, csa->image); - scan_token(csa); - xassert(csa->token == T_COLON); - scan_token(csa); - } - else - { /* objective name is not specified; use default */ - glp_set_obj_name(csa->P, "obj"); - } - /* parse linear form */ - len = parse_linear_form(csa); - for (k = 1; k <= len; k++) - glp_set_obj_coef(csa->P, csa->ind[k], csa->val[k]); - return; -} - -/*********************************************************************** -* parse_constraints - parse constraints section -* -* This routine parses the constraints section using the following -* syntax: -* -* ::= | : -* ::= < | <= | =< | > | >= | => | = -* ::= | + | -* - -* ::= -* -* ::= subject to | such that | st | s.t. | st. -* ::= | -* */ - -static void parse_constraints(struct csa *csa) -{ int i, len, type; - double s; - /* parse the keyword 'subject to' */ - xassert(csa->token == T_SUBJECT_TO); - scan_token(csa); -loop: /* create new row (constraint) */ - i = glp_add_rows(csa->P, 1); - /* parse row name */ - if (csa->token == T_NAME && csa->c == ':') - { /* row name is followed by a colon */ - if (glp_find_row(csa->P, csa->image) != 0) - error(csa, "constraint '%s' multiply defined\n", - csa->image); - glp_set_row_name(csa->P, i, csa->image); - scan_token(csa); - xassert(csa->token == T_COLON); - scan_token(csa); - } - else - { /* row name is not specified; use default */ - char name[50]; - sprintf(name, "r.%d", csa->count); - glp_set_row_name(csa->P, i, name); - } - /* parse linear form */ - len = parse_linear_form(csa); - glp_set_mat_row(csa->P, i, len, csa->ind, csa->val); - /* parse constraint sense */ - if (csa->token == T_LE) - type = GLP_UP, scan_token(csa); - else if (csa->token == T_GE) - type = GLP_LO, scan_token(csa); - else if (csa->token == T_EQ) - type = GLP_FX, scan_token(csa); - else - error(csa, "missing constraint sense\n"); - /* parse right-hand side */ - if (csa->token == T_PLUS) - s = +1.0, scan_token(csa); - else if (csa->token == T_MINUS) - s = -1.0, scan_token(csa); - else - s = +1.0; - if (csa->token != T_NUMBER) - error(csa, "missing right-hand side\n"); - glp_set_row_bnds(csa->P, i, type, s * csa->value, s * csa->value); - /* the rest of the current line must be empty */ - if (!(csa->c == '\n' || csa->c == EOF)) - error(csa, "invalid symbol(s) beyond right-hand side\n"); - scan_token(csa); - /* if the next token is a sign, numeric constant, or a symbolic - name, here is another constraint */ - if (csa->token == T_PLUS || csa->token == T_MINUS || - csa->token == T_NUMBER || csa->token == T_NAME) goto loop; - return; -} - -static void set_lower_bound(struct csa *csa, int j, double lb) -{ /* set lower bound of j-th variable */ - if (csa->lb[j] != +DBL_MAX && !csa->lb_warn) - { warning(csa, "lower bound of variable '%s' redefined\n", - glp_get_col_name(csa->P, j)); - csa->lb_warn = 1; - } - csa->lb[j] = lb; - return; -} - -static void set_upper_bound(struct csa *csa, int j, double ub) -{ /* set upper bound of j-th variable */ - if (csa->ub[j] != -DBL_MAX && !csa->ub_warn) - { warning(csa, "upper bound of variable '%s' redefined\n", - glp_get_col_name(csa->P, j)); - csa->ub_warn = 1; - } - csa->ub[j] = ub; - return; -} - -/*********************************************************************** -* parse_bounds - parse bounds section -* -* This routine parses the bounds section using the following syntax: -* -* ::= -* ::= infinity | inf -* ::= | + | -* - | + | - -* ::= < | <= | =< -* ::= > | >= | => -* ::= | -* | | -* | = | free -* ::= bounds | bound -* ::= | -* */ - -static void parse_bounds(struct csa *csa) -{ int j, lb_flag; - double lb, s; - /* parse the keyword 'bounds' */ - xassert(csa->token == T_BOUNDS); - scan_token(csa); -loop: /* bound definition can start with a sign, numeric constant, or - a symbolic name */ - if (!(csa->token == T_PLUS || csa->token == T_MINUS || - csa->token == T_NUMBER || csa->token == T_NAME)) goto done; - /* parse bound definition */ - if (csa->token == T_PLUS || csa->token == T_MINUS) - { /* parse signed lower bound */ - lb_flag = 1; - s = (csa->token == T_PLUS ? +1.0 : -1.0); - scan_token(csa); - if (csa->token == T_NUMBER) - lb = s * csa->value, scan_token(csa); - else if (the_same(csa->image, "infinity") || - the_same(csa->image, "inf")) - { if (s > 0.0) - error(csa, "invalid use of '+inf' as lower bound\n"); - lb = -DBL_MAX, scan_token(csa); - } - else - error(csa, "missing lower bound\n"); - } - else if (csa->token == T_NUMBER) - { /* parse unsigned lower bound */ - lb_flag = 1; - lb = csa->value, scan_token(csa); - } - else - { /* lower bound is not specified */ - lb_flag = 0; - } - /* parse the token that should follow the lower bound */ - if (lb_flag) - { if (csa->token != T_LE) - error(csa, "missing '<', '<=', or '=<' after lower bound\n") - ; - scan_token(csa); - } - /* parse variable name */ - if (csa->token != T_NAME) - error(csa, "missing variable name\n"); - j = find_col(csa, csa->image); - /* set lower bound */ - if (lb_flag) set_lower_bound(csa, j, lb); - scan_token(csa); - /* parse the context that follows the variable name */ - if (csa->token == T_LE) - { /* parse upper bound */ - scan_token(csa); - if (csa->token == T_PLUS || csa->token == T_MINUS) - { /* parse signed upper bound */ - s = (csa->token == T_PLUS ? +1.0 : -1.0); - scan_token(csa); - if (csa->token == T_NUMBER) - { set_upper_bound(csa, j, s * csa->value); - scan_token(csa); - } - else if (the_same(csa->image, "infinity") || - the_same(csa->image, "inf")) - { if (s < 0.0) - error(csa, "invalid use of '-inf' as upper bound\n"); - set_upper_bound(csa, j, +DBL_MAX); - scan_token(csa); - } - else - error(csa, "missing upper bound\n"); - } - else if (csa->token == T_NUMBER) - { /* parse unsigned upper bound */ - set_upper_bound(csa, j, csa->value); - scan_token(csa); - } - else - error(csa, "missing upper bound\n"); - } - else if (csa->token == T_GE) - { /* parse lower bound */ - if (lb_flag) - { /* the context '... <= x >= ...' is invalid */ - error(csa, "invalid bound definition\n"); - } - scan_token(csa); - if (csa->token == T_PLUS || csa->token == T_MINUS) - { /* parse signed lower bound */ - s = (csa->token == T_PLUS ? +1.0 : -1.0); - scan_token(csa); - if (csa->token == T_NUMBER) - { set_lower_bound(csa, j, s * csa->value); - scan_token(csa); - } - else if (the_same(csa->image, "infinity") || - the_same(csa->image, "inf") == 0) - { if (s > 0.0) - error(csa, "invalid use of '+inf' as lower bound\n"); - set_lower_bound(csa, j, -DBL_MAX); - scan_token(csa); - } - else - error(csa, "missing lower bound\n"); - } - else if (csa->token == T_NUMBER) - { /* parse unsigned lower bound */ - set_lower_bound(csa, j, csa->value); - scan_token(csa); - } - else - error(csa, "missing lower bound\n"); - } - else if (csa->token == T_EQ) - { /* parse fixed value */ - if (lb_flag) - { /* the context '... <= x = ...' is invalid */ - error(csa, "invalid bound definition\n"); - } - scan_token(csa); - if (csa->token == T_PLUS || csa->token == T_MINUS) - { /* parse signed fixed value */ - s = (csa->token == T_PLUS ? +1.0 : -1.0); - scan_token(csa); - if (csa->token == T_NUMBER) - { set_lower_bound(csa, j, s * csa->value); - set_upper_bound(csa, j, s * csa->value); - scan_token(csa); - } - else - error(csa, "missing fixed value\n"); - } - else if (csa->token == T_NUMBER) - { /* parse unsigned fixed value */ - set_lower_bound(csa, j, csa->value); - set_upper_bound(csa, j, csa->value); - scan_token(csa); - } - else - error(csa, "missing fixed value\n"); - } - else if (the_same(csa->image, "free")) - { /* parse the keyword 'free' */ - if (lb_flag) - { /* the context '... <= x free ...' is invalid */ - error(csa, "invalid bound definition\n"); - } - set_lower_bound(csa, j, -DBL_MAX); - set_upper_bound(csa, j, +DBL_MAX); - scan_token(csa); - } - else if (!lb_flag) - { /* neither lower nor upper bounds are specified */ - error(csa, "invalid bound definition\n"); - } - goto loop; -done: return; -} - -/*********************************************************************** -* parse_integer - parse general, integer, or binary section -* -* ::= -* ::= general | generals | gen -* ::= integer | integers | int -* ::= binary | binaries | bin -*
::= -* ::=
| -* */ - -static void parse_integer(struct csa *csa) -{ int j, binary; - /* parse the keyword 'general', 'integer', or 'binary' */ - if (csa->token == T_GENERAL) - binary = 0, scan_token(csa); - else if (csa->token == T_INTEGER) - binary = 0, scan_token(csa); - else if (csa->token == T_BINARY) - binary = 1, scan_token(csa); - else - xassert(csa != csa); - /* parse list of variables (may be empty) */ - while (csa->token == T_NAME) - { /* find the corresponding column */ - j = find_col(csa, csa->image); - /* change kind of the variable */ - glp_set_col_kind(csa->P, j, GLP_IV); - /* set bounds for the binary variable */ - if (binary) -#if 0 /* 07/VIII-2013 */ - { set_lower_bound(csa, j, 0.0); - set_upper_bound(csa, j, 1.0); - } -#else - { set_lower_bound(csa, j, - csa->lb[j] == +DBL_MAX ? 0.0 : csa->lb[j]); - set_upper_bound(csa, j, - csa->ub[j] == -DBL_MAX ? 1.0 : csa->ub[j]); - } -#endif - scan_token(csa); - } - return; -} - -int glp_read_lp(glp_prob *P, const glp_cpxcp *parm, const char *fname) -{ /* read problem data in CPLEX LP format */ - glp_cpxcp _parm; - struct csa _csa, *csa = &_csa; - int ret; - xprintf("Reading problem data from '%s'...\n", fname); - if (parm == NULL) - glp_init_cpxcp(&_parm), parm = &_parm; - /* check control parameters */ - check_parm("glp_read_lp", parm); - /* initialize common storage area */ - csa->P = P; - csa->parm = parm; - csa->fname = fname; - csa->fp = NULL; - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->count = 0; - csa->c = '\n'; - csa->token = T_EOF; - csa->image[0] = '\0'; - csa->imlen = 0; - csa->value = 0.0; - csa->n_max = 100; - csa->ind = xcalloc(1+csa->n_max, sizeof(int)); - csa->val = xcalloc(1+csa->n_max, sizeof(double)); - csa->flag = xcalloc(1+csa->n_max, sizeof(char)); - memset(&csa->flag[1], 0, csa->n_max * sizeof(char)); - csa->lb = xcalloc(1+csa->n_max, sizeof(double)); - csa->ub = xcalloc(1+csa->n_max, sizeof(double)); -#if 1 /* 27/VII-2013 */ - csa->lb_warn = csa->ub_warn = 0; -#endif - /* erase problem object */ - glp_erase_prob(P); - glp_create_index(P); - /* open input CPLEX LP file */ - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* scan very first token */ - scan_token(csa); - /* parse definition of the objective function */ - if (!(csa->token == T_MINIMIZE || csa->token == T_MAXIMIZE)) - error(csa, "'minimize' or 'maximize' keyword missing\n"); - parse_objective(csa); - /* parse constraints section */ - if (csa->token != T_SUBJECT_TO) - error(csa, "constraints section missing\n"); - parse_constraints(csa); - /* parse optional bounds section */ - if (csa->token == T_BOUNDS) parse_bounds(csa); - /* parse optional general, integer, and binary sections */ - while (csa->token == T_GENERAL || - csa->token == T_INTEGER || - csa->token == T_BINARY) parse_integer(csa); - /* check for the keyword 'end' */ - if (csa->token == T_END) - scan_token(csa); - else if (csa->token == T_EOF) - warning(csa, "keyword 'end' missing\n"); - else - error(csa, "symbol '%s' in wrong position\n", csa->image); - /* nothing must follow the keyword 'end' (except comments) */ - if (csa->token != T_EOF) - error(csa, "extra symbol(s) detected beyond 'end'\n"); - /* set bounds of variables */ - { int j, type; - double lb, ub; - for (j = 1; j <= P->n; j++) - { lb = csa->lb[j]; - ub = csa->ub[j]; - if (lb == +DBL_MAX) lb = 0.0; /* default lb */ - if (ub == -DBL_MAX) ub = +DBL_MAX; /* default ub */ - if (lb == -DBL_MAX && ub == +DBL_MAX) - type = GLP_FR; - else if (ub == +DBL_MAX) - type = GLP_LO; - else if (lb == -DBL_MAX) - type = GLP_UP; - else if (lb != ub) - type = GLP_DB; - else - type = GLP_FX; - glp_set_col_bnds(csa->P, j, type, lb, ub); - } - } - /* print some statistics */ - xprintf("%d row%s, %d column%s, %d non-zero%s\n", - P->m, P->m == 1 ? "" : "s", P->n, P->n == 1 ? "" : "s", - P->nnz, P->nnz == 1 ? "" : "s"); - if (glp_get_num_int(P) > 0) - { int ni = glp_get_num_int(P); - int nb = glp_get_num_bin(P); - if (ni == 1) - { if (nb == 0) - xprintf("One variable is integer\n"); - else - xprintf("One variable is binary\n"); - } - else - { xprintf("%d integer variables, ", ni); - if (nb == 0) - xprintf("none"); - else if (nb == 1) - xprintf("one"); - else if (nb == ni) - xprintf("all"); - else - xprintf("%d", nb); - xprintf(" of which %s binary\n", nb == 1 ? "is" : "are"); - } - } - xprintf("%d lines were read\n", csa->count); - /* problem data has been successfully read */ - glp_delete_index(P); - glp_sort_matrix(P); - ret = 0; -done: if (csa->fp != NULL) glp_close(csa->fp); - xfree(csa->ind); - xfree(csa->val); - xfree(csa->flag); - xfree(csa->lb); - xfree(csa->ub); - if (ret != 0) glp_erase_prob(P); - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_write_lp - write problem data in CPLEX LP format -* -* SYNOPSIS -* -* int glp_write_lp(glp_prob *P, const glp_cpxcp *parm, const char -* *fname); -* -* DESCRIPTION -* -* The routine glp_write_lp writes problem data in CPLEX LP format to -* a text file. -* -* The parameter parm is a pointer to the structure glp_cpxcp, which -* specifies control parameters used by the routine. If parm is NULL, -* the routine uses default settings. -* -* The character string fname specifies a name of the text file to be -* written. -* -* RETURNS -* -* If the operation was successful, the routine glp_write_lp returns -* zero. Otherwise, it prints an error message and returns non-zero. */ - -#define csa csa1 - -struct csa -{ /* common storage area */ - glp_prob *P; - /* pointer to problem object */ - const glp_cpxcp *parm; - /* pointer to control parameters */ -}; - -static int check_name(char *name) -{ /* check if specified name is valid for CPLEX LP format */ - if (*name == '.') return 1; - if (isdigit((unsigned char)*name)) return 1; - for (; *name; name++) - { if (!isalnum((unsigned char)*name) && - strchr(CHAR_SET, (unsigned char)*name) == NULL) return 1; - } - return 0; /* name is ok */ -} - -static void adjust_name(char *name) -{ /* attempt to adjust specified name to make it valid for CPLEX LP - format */ - for (; *name; name++) - { if (*name == ' ') - *name = '_'; - else if (*name == '-') - *name = '~'; - else if (*name == '[') - *name = '('; - else if (*name == ']') - *name = ')'; - } - return; -} - -static char *row_name(struct csa *csa, int i, char rname[255+1]) -{ /* construct symbolic name of i-th row (constraint) */ - const char *name; - if (i == 0) - name = glp_get_obj_name(csa->P); - else - name = glp_get_row_name(csa->P, i); - if (name == NULL) goto fake; - strcpy(rname, name); - adjust_name(rname); - if (check_name(rname)) goto fake; - return rname; -fake: if (i == 0) - strcpy(rname, "obj"); - else - sprintf(rname, "r_%d", i); - return rname; -} - -static char *col_name(struct csa *csa, int j, char cname[255+1]) -{ /* construct symbolic name of j-th column (variable) */ - const char *name; - name = glp_get_col_name(csa->P, j); - if (name == NULL) goto fake; - strcpy(cname, name); - adjust_name(cname); - if (check_name(cname)) goto fake; - return cname; -#if 0 /* 18/I-2018 */ -fake: sprintf(cname, "x_%d", j); -#else -fake: /* construct fake name depending on column's attributes */ - { GLPCOL *col = csa->P->col[j]; - if (col->type == GLP_FX) - { /* fixed column */ - sprintf(cname, "s_%d", j); - } - else if (col->kind == GLP_CV) - { /* continuous variable */ - sprintf(cname, "x_%d", j); - } - else if (!(col->lb == 0 && col->ub == 1)) - { /* general (non-binary) integer variable */ - sprintf(cname, "y_%d", j); - } - else - { /* binary variable */ - sprintf(cname, "z_%d", j); - } - } -#endif - return cname; -} - -int glp_write_lp(glp_prob *P, const glp_cpxcp *parm, const char *fname) -{ /* write problem data in CPLEX LP format */ - glp_cpxcp _parm; - struct csa _csa, *csa = &_csa; - glp_file *fp; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int i, j, len, flag, count, ret; - char line[1000+1], term[500+1], name[255+1]; - xprintf("Writing problem data to '%s'...\n", fname); - if (parm == NULL) - glp_init_cpxcp(&_parm), parm = &_parm; - /* check control parameters */ - check_parm("glp_write_lp", parm); - /* initialize common storage area */ - csa->P = P; - csa->parm = parm; - /* create output CPLEX LP file */ - fp = glp_open(fname, "w"), count = 0; - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* write problem name */ - xfprintf(fp, "\\* Problem: %s *\\\n", - P->name == NULL ? "Unknown" : P->name), count++; - xfprintf(fp, "\n"), count++; - /* the problem should contain at least one row and one column */ - if (!(P->m > 0 && P->n > 0)) - { xprintf("Warning: problem has no rows/columns\n"); - xfprintf(fp, "\\* WARNING: PROBLEM HAS NO ROWS/COLUMNS *\\\n"), - count++; - xfprintf(fp, "\n"), count++; - goto skip; - } - /* write the objective function definition */ - if (P->dir == GLP_MIN) - xfprintf(fp, "Minimize\n"), count++; - else if (P->dir == GLP_MAX) - xfprintf(fp, "Maximize\n"), count++; - else - xassert(P != P); - row_name(csa, 0, name); - sprintf(line, " %s:", name); - len = 0; - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->coef != 0.0 || col->ptr == NULL) - { len++; - col_name(csa, j, name); - if (col->coef == 0.0) - sprintf(term, " + 0 %s", name); /* empty column */ - else if (col->coef == +1.0) - sprintf(term, " + %s", name); - else if (col->coef == -1.0) - sprintf(term, " - %s", name); - else if (col->coef > 0.0) - sprintf(term, " + %.*g %s", DBL_DIG, +col->coef, name); - else - sprintf(term, " - %.*g %s", DBL_DIG, -col->coef, name); - if (strlen(line) + strlen(term) > 72) - xfprintf(fp, "%s\n", line), line[0] = '\0', count++; - strcat(line, term); - } - } - if (len == 0) - { /* empty objective */ - sprintf(term, " 0 %s", col_name(csa, 1, name)); - strcat(line, term); - } - xfprintf(fp, "%s\n", line), count++; - if (P->c0 != 0.0) - xfprintf(fp, "\\* constant term = %.*g *\\\n", DBL_DIG, P->c0), - count++; - xfprintf(fp, "\n"), count++; - /* write the constraints section */ - xfprintf(fp, "Subject To\n"), count++; - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->type == GLP_FR) continue; /* skip free row */ - row_name(csa, i, name); - sprintf(line, " %s:", name); - /* linear form */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col_name(csa, aij->col->j, name); - if (aij->val == +1.0) - sprintf(term, " + %s", name); - else if (aij->val == -1.0) - sprintf(term, " - %s", name); - else if (aij->val > 0.0) - sprintf(term, " + %.*g %s", DBL_DIG, +aij->val, name); - else - sprintf(term, " - %.*g %s", DBL_DIG, -aij->val, name); - if (strlen(line) + strlen(term) > 72) - xfprintf(fp, "%s\n", line), line[0] = '\0', count++; - strcat(line, term); - } - if (row->type == GLP_DB) - { /* double-bounded (ranged) constraint */ - sprintf(term, " - ~r_%d", i); - if (strlen(line) + strlen(term) > 72) - xfprintf(fp, "%s\n", line), line[0] = '\0', count++; - strcat(line, term); - } - else if (row->ptr == NULL) - { /* empty constraint */ - sprintf(term, " 0 %s", col_name(csa, 1, name)); - strcat(line, term); - } - /* right hand-side */ - if (row->type == GLP_LO) - sprintf(term, " >= %.*g", DBL_DIG, row->lb); - else if (row->type == GLP_UP) - sprintf(term, " <= %.*g", DBL_DIG, row->ub); - else if (row->type == GLP_DB || row->type == GLP_FX) - sprintf(term, " = %.*g", DBL_DIG, row->lb); - else - xassert(row != row); - if (strlen(line) + strlen(term) > 72) - xfprintf(fp, "%s\n", line), line[0] = '\0', count++; - strcat(line, term); - xfprintf(fp, "%s\n", line), count++; - } - xfprintf(fp, "\n"), count++; - /* write the bounds section */ - flag = 0; - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->type != GLP_DB) continue; - if (!flag) - xfprintf(fp, "Bounds\n"), flag = 1, count++; - xfprintf(fp, " 0 <= ~r_%d <= %.*g\n", - i, DBL_DIG, row->ub - row->lb), count++; - } - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->type == GLP_LO && col->lb == 0.0) continue; - if (!flag) - xfprintf(fp, "Bounds\n"), flag = 1, count++; - col_name(csa, j, name); - if (col->type == GLP_FR) - xfprintf(fp, " %s free\n", name), count++; - else if (col->type == GLP_LO) - xfprintf(fp, " %s >= %.*g\n", - name, DBL_DIG, col->lb), count++; - else if (col->type == GLP_UP) - xfprintf(fp, " -Inf <= %s <= %.*g\n", - name, DBL_DIG, col->ub), count++; - else if (col->type == GLP_DB) - xfprintf(fp, " %.*g <= %s <= %.*g\n", - DBL_DIG, col->lb, name, DBL_DIG, col->ub), count++; - else if (col->type == GLP_FX) - xfprintf(fp, " %s = %.*g\n", - name, DBL_DIG, col->lb), count++; - else - xassert(col != col); - } - if (flag) xfprintf(fp, "\n"), count++; - /* write the integer section */ - flag = 0; - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->kind == GLP_CV) continue; - xassert(col->kind == GLP_IV); - if (!flag) - xfprintf(fp, "Generals\n"), flag = 1, count++; - xfprintf(fp, " %s\n", col_name(csa, j, name)), count++; - } - if (flag) xfprintf(fp, "\n"), count++; -skip: /* write the end keyword */ - xfprintf(fp, "End\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* problem data has been successfully written */ - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/cpp.c b/code/3rd_glpk/api/cpp.c deleted file mode 100644 index ac3d63ef..00000000 --- a/code/3rd_glpk/api/cpp.c +++ /dev/null @@ -1,185 +0,0 @@ -/* cpp.c (solve critical path problem) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_cpp - solve critical path problem -* -* SYNOPSIS -* -* double glp_cpp(glp_graph *G, int v_t, int v_es, int v_ls); -* -* DESCRIPTION -* -* The routine glp_cpp solves the critical path problem represented in -* the form of the project network. -* -* The parameter G is a pointer to the graph object, which specifies -* the project network. This graph must be acyclic. Multiple arcs are -* allowed being considered as single arcs. -* -* The parameter v_t specifies an offset of the field of type double -* in the vertex data block, which contains time t[i] >= 0 needed to -* perform corresponding job j. If v_t < 0, it is assumed that t[i] = 1 -* for all jobs. -* -* The parameter v_es specifies an offset of the field of type double -* in the vertex data block, to which the routine stores earliest start -* time for corresponding job. If v_es < 0, this time is not stored. -* -* The parameter v_ls specifies an offset of the field of type double -* in the vertex data block, to which the routine stores latest start -* time for corresponding job. If v_ls < 0, this time is not stored. -* -* RETURNS -* -* The routine glp_cpp returns the minimal project duration, that is, -* minimal time needed to perform all jobs in the project. */ - -static void sorting(glp_graph *G, int list[]); - -double glp_cpp(glp_graph *G, int v_t, int v_es, int v_ls) -{ glp_vertex *v; - glp_arc *a; - int i, j, k, nv, *list; - double temp, total, *t, *es, *ls; - if (v_t >= 0 && v_t > G->v_size - (int)sizeof(double)) - xerror("glp_cpp: v_t = %d; invalid offset\n", v_t); - if (v_es >= 0 && v_es > G->v_size - (int)sizeof(double)) - xerror("glp_cpp: v_es = %d; invalid offset\n", v_es); - if (v_ls >= 0 && v_ls > G->v_size - (int)sizeof(double)) - xerror("glp_cpp: v_ls = %d; invalid offset\n", v_ls); - nv = G->nv; - if (nv == 0) - { total = 0.0; - goto done; - } - /* allocate working arrays */ - t = xcalloc(1+nv, sizeof(double)); - es = xcalloc(1+nv, sizeof(double)); - ls = xcalloc(1+nv, sizeof(double)); - list = xcalloc(1+nv, sizeof(int)); - /* retrieve job times */ - for (i = 1; i <= nv; i++) - { v = G->v[i]; - if (v_t >= 0) - { memcpy(&t[i], (char *)v->data + v_t, sizeof(double)); - if (t[i] < 0.0) - xerror("glp_cpp: t[%d] = %g; invalid time\n", i, t[i]); - } - else - t[i] = 1.0; - } - /* perform topological sorting to determine the list of nodes - (jobs) such that if list[k] = i and list[kk] = j and there - exists arc (i->j), then k < kk */ - sorting(G, list); - /* FORWARD PASS */ - /* determine earliest start times */ - for (k = 1; k <= nv; k++) - { j = list[k]; - es[j] = 0.0; - for (a = G->v[j]->in; a != NULL; a = a->h_next) - { i = a->tail->i; - /* there exists arc (i->j) in the project network */ - temp = es[i] + t[i]; - if (es[j] < temp) es[j] = temp; - } - } - /* determine the minimal project duration */ - total = 0.0; - for (i = 1; i <= nv; i++) - { temp = es[i] + t[i]; - if (total < temp) total = temp; - } - /* BACKWARD PASS */ - /* determine latest start times */ - for (k = nv; k >= 1; k--) - { i = list[k]; - ls[i] = total - t[i]; - for (a = G->v[i]->out; a != NULL; a = a->t_next) - { j = a->head->i; - /* there exists arc (i->j) in the project network */ - temp = ls[j] - t[i]; - if (ls[i] > temp) ls[i] = temp; - } - /* avoid possible round-off errors */ - if (ls[i] < es[i]) ls[i] = es[i]; - } - /* store results, if necessary */ - if (v_es >= 0) - { for (i = 1; i <= nv; i++) - { v = G->v[i]; - memcpy((char *)v->data + v_es, &es[i], sizeof(double)); - } - } - if (v_ls >= 0) - { for (i = 1; i <= nv; i++) - { v = G->v[i]; - memcpy((char *)v->data + v_ls, &ls[i], sizeof(double)); - } - } - /* free working arrays */ - xfree(t); - xfree(es); - xfree(ls); - xfree(list); -done: return total; -} - -static void sorting(glp_graph *G, int list[]) -{ /* perform topological sorting to determine the list of nodes - (jobs) such that if list[k] = i and list[kk] = j and there - exists arc (i->j), then k < kk */ - int i, k, nv, v_size, *num; - void **save; - nv = G->nv; - v_size = G->v_size; - save = xcalloc(1+nv, sizeof(void *)); - num = xcalloc(1+nv, sizeof(int)); - G->v_size = sizeof(int); - for (i = 1; i <= nv; i++) - { save[i] = G->v[i]->data; - G->v[i]->data = &num[i]; - list[i] = 0; - } - if (glp_top_sort(G, 0) != 0) - xerror("glp_cpp: project network is not acyclic\n"); - G->v_size = v_size; - for (i = 1; i <= nv; i++) - { G->v[i]->data = save[i]; - k = num[i]; - xassert(1 <= k && k <= nv); - xassert(list[k] == 0); - list[k] = i; - } - xfree(save); - xfree(num); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/cpxbas.c b/code/3rd_glpk/api/cpxbas.c deleted file mode 100644 index e1c656a7..00000000 --- a/code/3rd_glpk/api/cpxbas.c +++ /dev/null @@ -1,269 +0,0 @@ -/* cpxbas.c (construct Bixby's initial LP basis) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -struct var -{ /* structural variable */ - int j; - /* ordinal number */ - double q; - /* penalty value */ -}; - -static int CDECL fcmp(const void *ptr1, const void *ptr2) -{ /* this routine is passed to the qsort() function */ - struct var *col1 = (void *)ptr1, *col2 = (void *)ptr2; - if (col1->q < col2->q) return -1; - if (col1->q > col2->q) return +1; - return 0; -} - -static int get_column(glp_prob *lp, int j, int ind[], double val[]) -{ /* Bixby's algorithm assumes that the constraint matrix is scaled - such that the maximum absolute value in every non-zero row and - column is 1 */ - int k, len; - double big; - len = glp_get_mat_col(lp, j, ind, val); - big = 0.0; - for (k = 1; k <= len; k++) - if (big < fabs(val[k])) big = fabs(val[k]); - if (big == 0.0) big = 1.0; - for (k = 1; k <= len; k++) val[k] /= big; - return len; -} - -static void cpx_basis(glp_prob *lp) -{ /* main routine */ - struct var *C, *C2, *C3, *C4; - int m, n, i, j, jk, k, l, ll, t, n2, n3, n4, type, len, *I, *r, - *ind; - double alpha, gamma, cmax, temp, *v, *val; - xprintf("Constructing initial basis...\n"); - /* determine the number of rows and columns */ - m = glp_get_num_rows(lp); - n = glp_get_num_cols(lp); - /* allocate working arrays */ - C = xcalloc(1+n, sizeof(struct var)); - I = xcalloc(1+m, sizeof(int)); - r = xcalloc(1+m, sizeof(int)); - v = xcalloc(1+m, sizeof(double)); - ind = xcalloc(1+m, sizeof(int)); - val = xcalloc(1+m, sizeof(double)); - /* make all auxiliary variables non-basic */ - for (i = 1; i <= m; i++) - { if (glp_get_row_type(lp, i) != GLP_DB) - glp_set_row_stat(lp, i, GLP_NS); - else if (fabs(glp_get_row_lb(lp, i)) <= - fabs(glp_get_row_ub(lp, i))) - glp_set_row_stat(lp, i, GLP_NL); - else - glp_set_row_stat(lp, i, GLP_NU); - } - /* make all structural variables non-basic */ - for (j = 1; j <= n; j++) - { if (glp_get_col_type(lp, j) != GLP_DB) - glp_set_col_stat(lp, j, GLP_NS); - else if (fabs(glp_get_col_lb(lp, j)) <= - fabs(glp_get_col_ub(lp, j))) - glp_set_col_stat(lp, j, GLP_NL); - else - glp_set_col_stat(lp, j, GLP_NU); - } - /* C2 is a set of free structural variables */ - n2 = 0, C2 = C + 0; - for (j = 1; j <= n; j++) - { type = glp_get_col_type(lp, j); - if (type == GLP_FR) - { n2++; - C2[n2].j = j; - C2[n2].q = 0.0; - } - } - /* C3 is a set of structural variables having excatly one (lower - or upper) bound */ - n3 = 0, C3 = C2 + n2; - for (j = 1; j <= n; j++) - { type = glp_get_col_type(lp, j); - if (type == GLP_LO) - { n3++; - C3[n3].j = j; - C3[n3].q = + glp_get_col_lb(lp, j); - } - else if (type == GLP_UP) - { n3++; - C3[n3].j = j; - C3[n3].q = - glp_get_col_ub(lp, j); - } - } - /* C4 is a set of structural variables having both (lower and - upper) bounds */ - n4 = 0, C4 = C3 + n3; - for (j = 1; j <= n; j++) - { type = glp_get_col_type(lp, j); - if (type == GLP_DB) - { n4++; - C4[n4].j = j; - C4[n4].q = glp_get_col_lb(lp, j) - glp_get_col_ub(lp, j); - } - } - /* compute gamma = max{|c[j]|: 1 <= j <= n} */ - gamma = 0.0; - for (j = 1; j <= n; j++) - { temp = fabs(glp_get_obj_coef(lp, j)); - if (gamma < temp) gamma = temp; - } - /* compute cmax */ - cmax = (gamma == 0.0 ? 1.0 : 1000.0 * gamma); - /* compute final penalty for all structural variables within sets - C2, C3, and C4 */ - switch (glp_get_obj_dir(lp)) - { case GLP_MIN: temp = +1.0; break; - case GLP_MAX: temp = -1.0; break; - default: xassert(lp != lp); - } - for (k = 1; k <= n2+n3+n4; k++) - { j = C[k].j; - C[k].q += (temp * glp_get_obj_coef(lp, j)) / cmax; - } - /* sort structural variables within C2, C3, and C4 in ascending - order of penalty value */ - qsort(C2+1, n2, sizeof(struct var), fcmp); - for (k = 1; k < n2; k++) xassert(C2[k].q <= C2[k+1].q); - qsort(C3+1, n3, sizeof(struct var), fcmp); - for (k = 1; k < n3; k++) xassert(C3[k].q <= C3[k+1].q); - qsort(C4+1, n4, sizeof(struct var), fcmp); - for (k = 1; k < n4; k++) xassert(C4[k].q <= C4[k+1].q); - /*** STEP 1 ***/ - for (i = 1; i <= m; i++) - { type = glp_get_row_type(lp, i); - if (type != GLP_FX) - { /* row i is either free or inequality constraint */ - glp_set_row_stat(lp, i, GLP_BS); - I[i] = 1; - r[i] = 1; - } - else - { /* row i is equality constraint */ - I[i] = 0; - r[i] = 0; - } - v[i] = +DBL_MAX; - } - /*** STEP 2 ***/ - for (k = 1; k <= n2+n3+n4; k++) - { jk = C[k].j; - len = get_column(lp, jk, ind, val); - /* let alpha = max{|A[l,jk]|: r[l] = 0} and let l' be such - that alpha = |A[l',jk]| */ - alpha = 0.0, ll = 0; - for (t = 1; t <= len; t++) - { l = ind[t]; - if (r[l] == 0 && alpha < fabs(val[t])) - alpha = fabs(val[t]), ll = l; - } - if (alpha >= 0.99) - { /* B := B union {jk} */ - glp_set_col_stat(lp, jk, GLP_BS); - I[ll] = 1; - v[ll] = alpha; - /* r[l] := r[l] + 1 for all l such that |A[l,jk]| != 0 */ - for (t = 1; t <= len; t++) - { l = ind[t]; - if (val[t] != 0.0) r[l]++; - } - /* continue to the next k */ - continue; - } - /* if |A[l,jk]| > 0.01 * v[l] for some l, continue to the - next k */ - for (t = 1; t <= len; t++) - { l = ind[t]; - if (fabs(val[t]) > 0.01 * v[l]) break; - } - if (t <= len) continue; - /* otherwise, let alpha = max{|A[l,jk]|: I[l] = 0} and let l' - be such that alpha = |A[l',jk]| */ - alpha = 0.0, ll = 0; - for (t = 1; t <= len; t++) - { l = ind[t]; - if (I[l] == 0 && alpha < fabs(val[t])) - alpha = fabs(val[t]), ll = l; - } - /* if alpha = 0, continue to the next k */ - if (alpha == 0.0) continue; - /* B := B union {jk} */ - glp_set_col_stat(lp, jk, GLP_BS); - I[ll] = 1; - v[ll] = alpha; - /* r[l] := r[l] + 1 for all l such that |A[l,jk]| != 0 */ - for (t = 1; t <= len; t++) - { l = ind[t]; - if (val[t] != 0.0) r[l]++; - } - } - /*** STEP 3 ***/ - /* add an artificial variable (auxiliary variable for equality - constraint) to cover each remaining uncovered row */ - for (i = 1; i <= m; i++) - if (I[i] == 0) glp_set_row_stat(lp, i, GLP_BS); - /* free working arrays */ - xfree(C); - xfree(I); - xfree(r); - xfree(v); - xfree(ind); - xfree(val); - return; -} - -/*********************************************************************** -* NAME -* -* glp_cpx_basis - construct Bixby's initial LP basis -* -* SYNOPSIS -* -* void glp_cpx_basis(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_cpx_basis constructs an advanced initial basis for -* the specified problem object. -* -* The routine is based on Bixby's algorithm described in the paper: -* -* Robert E. Bixby. Implementing the Simplex Method: The Initial Basis. -* ORSA Journal on Computing, Vol. 4, No. 3, 1992, pp. 267-84. */ - -void glp_cpx_basis(glp_prob *lp) -{ if (lp->m == 0 || lp->n == 0) - glp_std_basis(lp); - else - cpx_basis(lp); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/graph.c b/code/3rd_glpk/api/graph.c deleted file mode 100644 index 82994c84..00000000 --- a/code/3rd_glpk/api/graph.c +++ /dev/null @@ -1,504 +0,0 @@ -/* graph.c (basic graph routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "avl.h" -#include "dmp.h" -#include "env.h" -#include "glpk.h" - -/* CAUTION: DO NOT CHANGE THE LIMITS BELOW */ - -#define NV_MAX 100000000 /* = 100*10^6 */ -/* maximal number of vertices in the graph */ - -#define NA_MAX 500000000 /* = 500*10^6 */ -/* maximal number of arcs in the graph */ - -/*********************************************************************** -* NAME -* -* glp_create_graph - create graph -* -* SYNOPSIS -* -* glp_graph *glp_create_graph(int v_size, int a_size); -* -* DESCRIPTION -* -* The routine creates a new graph, which initially is empty, i.e. has -* no vertices and arcs. -* -* The parameter v_size specifies the size of data associated with each -* vertex of the graph (0 to 256 bytes). -* -* The parameter a_size specifies the size of data associated with each -* arc of the graph (0 to 256 bytes). -* -* RETURNS -* -* The routine returns a pointer to the graph created. */ - -static void create_graph(glp_graph *G, int v_size, int a_size) -{ G->pool = dmp_create_pool(); - G->name = NULL; - G->nv_max = 50; - G->nv = G->na = 0; - G->v = xcalloc(1+G->nv_max, sizeof(glp_vertex *)); - G->index = NULL; - G->v_size = v_size; - G->a_size = a_size; - return; -} - -glp_graph *glp_create_graph(int v_size, int a_size) -{ glp_graph *G; - if (!(0 <= v_size && v_size <= 256)) - xerror("glp_create_graph: v_size = %d; invalid size of vertex " - "data\n", v_size); - if (!(0 <= a_size && a_size <= 256)) - xerror("glp_create_graph: a_size = %d; invalid size of arc dat" - "a\n", a_size); - G = xmalloc(sizeof(glp_graph)); - create_graph(G, v_size, a_size); - return G; -} - -/*********************************************************************** -* NAME -* -* glp_set_graph_name - assign (change) graph name -* -* SYNOPSIS -* -* void glp_set_graph_name(glp_graph *G, const char *name); -* -* DESCRIPTION -* -* The routine glp_set_graph_name assigns a symbolic name specified by -* the character string name (1 to 255 chars) to the graph. -* -* If the parameter name is NULL or an empty string, the routine erases -* the existing symbolic name of the graph. */ - -void glp_set_graph_name(glp_graph *G, const char *name) -{ if (G->name != NULL) - { dmp_free_atom(G->pool, G->name, strlen(G->name)+1); - G->name = NULL; - } - if (!(name == NULL || name[0] == '\0')) - { int j; - for (j = 0; name[j] != '\0'; j++) - { if (j == 256) - xerror("glp_set_graph_name: graph name too long\n"); - if (iscntrl((unsigned char)name[j])) - xerror("glp_set_graph_name: graph name contains invalid " - "character(s)\n"); - } - G->name = dmp_get_atom(G->pool, strlen(name)+1); - strcpy(G->name, name); - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_add_vertices - add new vertices to graph -* -* SYNOPSIS -* -* int glp_add_vertices(glp_graph *G, int nadd); -* -* DESCRIPTION -* -* The routine glp_add_vertices adds nadd vertices to the specified -* graph. New vertices are always added to the end of the vertex list, -* so ordinal numbers of existing vertices remain unchanged. -* -* Being added each new vertex is isolated (has no incident arcs). -* -* RETURNS -* -* The routine glp_add_vertices returns an ordinal number of the first -* new vertex added to the graph. */ - -int glp_add_vertices(glp_graph *G, int nadd) -{ int i, nv_new; - if (nadd < 1) - xerror("glp_add_vertices: nadd = %d; invalid number of vertice" - "s\n", nadd); - if (nadd > NV_MAX - G->nv) - xerror("glp_add_vertices: nadd = %d; too many vertices\n", - nadd); - /* determine new number of vertices */ - nv_new = G->nv + nadd; - /* increase the room, if necessary */ - if (G->nv_max < nv_new) - { glp_vertex **save = G->v; - while (G->nv_max < nv_new) - { G->nv_max += G->nv_max; - xassert(G->nv_max > 0); - } - G->v = xcalloc(1+G->nv_max, sizeof(glp_vertex *)); - memcpy(&G->v[1], &save[1], G->nv * sizeof(glp_vertex *)); - xfree(save); - } - /* add new vertices to the end of the vertex list */ - for (i = G->nv+1; i <= nv_new; i++) - { glp_vertex *v; - G->v[i] = v = dmp_get_atom(G->pool, sizeof(glp_vertex)); - v->i = i; - v->name = NULL; - v->entry = NULL; - if (G->v_size == 0) - v->data = NULL; - else - { v->data = dmp_get_atom(G->pool, G->v_size); - memset(v->data, 0, G->v_size); - } - v->temp = NULL; - v->in = v->out = NULL; - } - /* set new number of vertices */ - G->nv = nv_new; - /* return the ordinal number of the first vertex added */ - return nv_new - nadd + 1; -} - -/**********************************************************************/ - -void glp_set_vertex_name(glp_graph *G, int i, const char *name) -{ /* assign (change) vertex name */ - glp_vertex *v; - if (!(1 <= i && i <= G->nv)) - xerror("glp_set_vertex_name: i = %d; vertex number out of rang" - "e\n", i); - v = G->v[i]; - if (v->name != NULL) - { if (v->entry != NULL) - { xassert(G->index != NULL); - avl_delete_node(G->index, v->entry); - v->entry = NULL; - } - dmp_free_atom(G->pool, v->name, strlen(v->name)+1); - v->name = NULL; - } - if (!(name == NULL || name[0] == '\0')) - { int k; - for (k = 0; name[k] != '\0'; k++) - { if (k == 256) - xerror("glp_set_vertex_name: i = %d; vertex name too lon" - "g\n", i); - if (iscntrl((unsigned char)name[k])) - xerror("glp_set_vertex_name: i = %d; vertex name contain" - "s invalid character(s)\n", i); - } - v->name = dmp_get_atom(G->pool, strlen(name)+1); - strcpy(v->name, name); - if (G->index != NULL) - { xassert(v->entry == NULL); - v->entry = avl_insert_node(G->index, v->name); - avl_set_node_link(v->entry, v); - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_add_arc - add new arc to graph -* -* SYNOPSIS -* -* glp_arc *glp_add_arc(glp_graph *G, int i, int j); -* -* DESCRIPTION -* -* The routine glp_add_arc adds a new arc to the specified graph. -* -* The parameters i and j specify the ordinal numbers of, resp., tail -* and head vertices of the arc. Note that self-loops and multiple arcs -* are allowed. -* -* RETURNS -* -* The routine glp_add_arc returns a pointer to the arc added. */ - -glp_arc *glp_add_arc(glp_graph *G, int i, int j) -{ glp_arc *a; - if (!(1 <= i && i <= G->nv)) - xerror("glp_add_arc: i = %d; tail vertex number out of range\n" - , i); - if (!(1 <= j && j <= G->nv)) - xerror("glp_add_arc: j = %d; head vertex number out of range\n" - , j); - if (G->na == NA_MAX) - xerror("glp_add_arc: too many arcs\n"); - a = dmp_get_atom(G->pool, sizeof(glp_arc)); - a->tail = G->v[i]; - a->head = G->v[j]; - if (G->a_size == 0) - a->data = NULL; - else - { a->data = dmp_get_atom(G->pool, G->a_size); - memset(a->data, 0, G->a_size); - } - a->temp = NULL; - a->t_prev = NULL; - a->t_next = G->v[i]->out; - if (a->t_next != NULL) a->t_next->t_prev = a; - a->h_prev = NULL; - a->h_next = G->v[j]->in; - if (a->h_next != NULL) a->h_next->h_prev = a; - G->v[i]->out = G->v[j]->in = a; - G->na++; - return a; -} - -/*********************************************************************** -* NAME -* -* glp_del_vertices - delete vertices from graph -* -* SYNOPSIS -* -* void glp_del_vertices(glp_graph *G, int ndel, const int num[]); -* -* DESCRIPTION -* -* The routine glp_del_vertices deletes vertices along with all -* incident arcs from the specified graph. Ordinal numbers of vertices -* to be deleted should be placed in locations num[1], ..., num[ndel], -* ndel > 0. -* -* Note that deleting vertices involves changing ordinal numbers of -* other vertices remaining in the graph. New ordinal numbers of the -* remaining vertices are assigned under the assumption that the -* original order of vertices is not changed. */ - -void glp_del_vertices(glp_graph *G, int ndel, const int num[]) -{ glp_vertex *v; - int i, k, nv_new; - /* scan the list of vertices to be deleted */ - if (!(1 <= ndel && ndel <= G->nv)) - xerror("glp_del_vertices: ndel = %d; invalid number of vertice" - "s\n", ndel); - for (k = 1; k <= ndel; k++) - { /* take the number of vertex to be deleted */ - i = num[k]; - /* obtain pointer to i-th vertex */ - if (!(1 <= i && i <= G->nv)) - xerror("glp_del_vertices: num[%d] = %d; vertex number out o" - "f range\n", k, i); - v = G->v[i]; - /* check that the vertex is not marked yet */ - if (v->i == 0) - xerror("glp_del_vertices: num[%d] = %d; duplicate vertex nu" - "mbers not allowed\n", k, i); - /* erase symbolic name assigned to the vertex */ - glp_set_vertex_name(G, i, NULL); - xassert(v->name == NULL); - xassert(v->entry == NULL); - /* free vertex data, if allocated */ - if (v->data != NULL) - dmp_free_atom(G->pool, v->data, G->v_size); - /* delete all incoming arcs */ - while (v->in != NULL) - glp_del_arc(G, v->in); - /* delete all outgoing arcs */ - while (v->out != NULL) - glp_del_arc(G, v->out); - /* mark the vertex to be deleted */ - v->i = 0; - } - /* delete all marked vertices from the vertex list */ - nv_new = 0; - for (i = 1; i <= G->nv; i++) - { /* obtain pointer to i-th vertex */ - v = G->v[i]; - /* check if the vertex is marked */ - if (v->i == 0) - { /* it is marked, delete it */ - dmp_free_atom(G->pool, v, sizeof(glp_vertex)); - } - else - { /* it is not marked, keep it */ - v->i = ++nv_new; - G->v[v->i] = v; - } - } - /* set new number of vertices in the graph */ - G->nv = nv_new; - return; -} - -/*********************************************************************** -* NAME -* -* glp_del_arc - delete arc from graph -* -* SYNOPSIS -* -* void glp_del_arc(glp_graph *G, glp_arc *a); -* -* DESCRIPTION -* -* The routine glp_del_arc deletes an arc from the specified graph. -* The arc to be deleted must exist. */ - -void glp_del_arc(glp_graph *G, glp_arc *a) -{ /* some sanity checks */ - xassert(G->na > 0); - xassert(1 <= a->tail->i && a->tail->i <= G->nv); - xassert(a->tail == G->v[a->tail->i]); - xassert(1 <= a->head->i && a->head->i <= G->nv); - xassert(a->head == G->v[a->head->i]); - /* remove the arc from the list of incoming arcs */ - if (a->h_prev == NULL) - a->head->in = a->h_next; - else - a->h_prev->h_next = a->h_next; - if (a->h_next == NULL) - ; - else - a->h_next->h_prev = a->h_prev; - /* remove the arc from the list of outgoing arcs */ - if (a->t_prev == NULL) - a->tail->out = a->t_next; - else - a->t_prev->t_next = a->t_next; - if (a->t_next == NULL) - ; - else - a->t_next->t_prev = a->t_prev; - /* free arc data, if allocated */ - if (a->data != NULL) - dmp_free_atom(G->pool, a->data, G->a_size); - /* delete the arc from the graph */ - dmp_free_atom(G->pool, a, sizeof(glp_arc)); - G->na--; - return; -} - -/*********************************************************************** -* NAME -* -* glp_erase_graph - erase graph content -* -* SYNOPSIS -* -* void glp_erase_graph(glp_graph *G, int v_size, int a_size); -* -* DESCRIPTION -* -* The routine glp_erase_graph erases the content of the specified -* graph. The effect of this operation is the same as if the graph -* would be deleted with the routine glp_delete_graph and then created -* anew with the routine glp_create_graph, with exception that the -* handle (pointer) to the graph remains valid. */ - -static void delete_graph(glp_graph *G) -{ dmp_delete_pool(G->pool); - xfree(G->v); - if (G->index != NULL) avl_delete_tree(G->index); - return; -} - -void glp_erase_graph(glp_graph *G, int v_size, int a_size) -{ if (!(0 <= v_size && v_size <= 256)) - xerror("glp_erase_graph: v_size = %d; invalid size of vertex d" - "ata\n", v_size); - if (!(0 <= a_size && a_size <= 256)) - xerror("glp_erase_graph: a_size = %d; invalid size of arc data" - "\n", a_size); - delete_graph(G); - create_graph(G, v_size, a_size); - return; -} - -/*********************************************************************** -* NAME -* -* glp_delete_graph - delete graph -* -* SYNOPSIS -* -* void glp_delete_graph(glp_graph *G); -* -* DESCRIPTION -* -* The routine glp_delete_graph deletes the specified graph and frees -* all the memory allocated to this program object. */ - -void glp_delete_graph(glp_graph *G) -{ delete_graph(G); - xfree(G); - return; -} - -/**********************************************************************/ - -void glp_create_v_index(glp_graph *G) -{ /* create vertex name index */ - glp_vertex *v; - int i; - if (G->index == NULL) - { G->index = avl_create_tree(avl_strcmp, NULL); - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - xassert(v->entry == NULL); - if (v->name != NULL) - { v->entry = avl_insert_node(G->index, v->name); - avl_set_node_link(v->entry, v); - } - } - } - return; -} - -int glp_find_vertex(glp_graph *G, const char *name) -{ /* find vertex by its name */ - AVLNODE *node; - int i = 0; - if (G->index == NULL) - xerror("glp_find_vertex: vertex name index does not exist\n"); - if (!(name == NULL || name[0] == '\0' || strlen(name) > 255)) - { node = avl_find_node(G->index, name); - if (node != NULL) - i = ((glp_vertex *)avl_get_node_link(node))->i; - } - return i; -} - -void glp_delete_v_index(glp_graph *G) -{ /* delete vertex name index */ - int i; - if (G->index != NULL) - { avl_delete_tree(G->index), G->index = NULL; - for (i = 1; i <= G->nv; i++) G->v[i]->entry = NULL; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/gridgen.c b/code/3rd_glpk/api/gridgen.c deleted file mode 100644 index 8cd3517f..00000000 --- a/code/3rd_glpk/api/gridgen.c +++ /dev/null @@ -1,769 +0,0 @@ -/* gridgen.c (grid-like network problem generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* This code is a modified version of the program GRIDGEN, a grid-like -* network problem generator developed by Yusin Lee and Jim Orlin. -* The original code is publically available on the DIMACS ftp site at: -* . -* -* All changes concern only the program interface, so this modified -* version produces exactly the same instances as the original version. -* -* Changes were made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_gridgen - grid-like network problem generator -* -* SYNOPSIS -* -* int glp_gridgen(glp_graph *G, int v_rhs, int a_cap, int a_cost, -* const int parm[1+14]); -* -* DESCRIPTION -* -* The routine glp_gridgen is a grid-like network problem generator -* developed by Yusin Lee and Jim Orlin. -* -* The parameter G specifies the graph object, to which the generated -* problem data have to be stored. Note that on entry the graph object -* is erased with the routine glp_erase_graph. -* -* The parameter v_rhs specifies an offset of the field of type double -* in the vertex data block, to which the routine stores the supply or -* demand value. If v_rhs < 0, the value is not stored. -* -* The parameter a_cap specifies an offset of the field of type double -* in the arc data block, to which the routine stores the arc capacity. -* If a_cap < 0, the capacity is not stored. -* -* The parameter a_cost specifies an offset of the field of type double -* in the arc data block, to which the routine stores the per-unit cost -* if the arc flow. If a_cost < 0, the cost is not stored. -* -* The array parm contains description of the network to be generated: -* -* parm[0] not used -* parm[1] two-ways arcs indicator: -* 1 - if links in both direction should be generated -* 0 - otherwise -* parm[2] random number seed (a positive integer) -* parm[3] number of nodes (the number of nodes generated might be -* slightly different to make the network a grid) -* parm[4] grid width -* parm[5] number of sources -* parm[6] number of sinks -* parm[7] average degree -* parm[8] total flow -* parm[9] distribution of arc costs: -* 1 - uniform -* 2 - exponential -* parm[10] lower bound for arc cost (uniform) -* 100 * lambda (exponential) -* parm[11] upper bound for arc cost (uniform) -* not used (exponential) -* parm[12] distribution of arc capacities: -* 1 - uniform -* 2 - exponential -* parm[13] lower bound for arc capacity (uniform) -* 100 * lambda (exponential) -* parm[14] upper bound for arc capacity (uniform) -* not used (exponential) -* -* RETURNS -* -* If the instance was successfully generated, the routine glp_gridgen -* returns zero; otherwise, if specified parameters are inconsistent, -* the routine returns a non-zero error code. -* -* COMMENTS -* -* This network generator generates a grid-like network plus a super -* node. In additional to the arcs connecting the nodes in the grid, -* there is an arc from each supply node to the super node and from the -* super node to each demand node to guarantee feasiblity. These arcs -* have very high costs and very big capacities. -* -* The idea of this network generator is as follows: First, a grid of -* n1 * n2 is generated. For example, 5 * 3. The nodes are numbered as -* 1 to 15, and the supernode is numbered as n1*n2+1. Then arcs between -* adjacent nodes are generated. For these arcs, the user is allowed to -* specify either to generate two-way arcs or one-way arcs. If two-way -* arcs are to be generated, two arcs, one in each direction, will be -* generated between each adjacent node pairs. Otherwise, only one arc -* will be generated. If this is the case, the arcs will be generated -* in alterntive directions as shown below. -* -* 1 ---> 2 ---> 3 ---> 4 ---> 5 -* | ^ | ^ | -* | | | | | -* V | V | V -* 6 <--- 7 <--- 8 <--- 9 <--- 10 -* | ^ | ^ | -* | | | | | -* V | V | V -* 11 --->12 --->13 --->14 ---> 15 -* -* Then the arcs between the super node and the source/sink nodes are -* added as mentioned before. If the number of arcs still doesn't reach -* the requirement, additional arcs will be added by uniformly picking -* random node pairs. There is no checking to prevent multiple arcs -* between any pair of nodes. However, there will be no self-arcs (arcs -* that poins back to its tail node) in the network. -* -* The source and sink nodes are selected uniformly in the network, and -* the imbalances of each source/sink node are also assigned by uniform -* distribution. */ - -struct stat_para -{ /* structure for statistical distributions */ - int distribution; - /* the distribution: */ -#define UNIFORM 1 /* uniform distribution */ -#define EXPONENTIAL 2 /* exponential distribution */ - double parameter[5]; - /* the parameters of the distribution */ -}; - -struct arcs -{ int from; - /* the FROM node of that arc */ - int to; - /* the TO node of that arc */ - int cost; - /* original cost of that arc */ - int u; - /* capacity of the arc */ -}; - -struct imbalance -{ int node; - /* Node ID */ - int supply; - /* Supply of that node */ -}; - -struct csa -{ /* common storage area */ - glp_graph *G; - int v_rhs, a_cap, a_cost; - int seed; - /* random number seed */ - int seed_original; - /* the original seed from input */ - int two_way; - /* 0: generate arcs in both direction for the basic grid, except - for the arcs to/from the super node. 1: o/w */ - int n_node; - /* total number of nodes in the network, numbered 1 to n_node, - including the super node, which is the last one */ - int n_arc; - /* total number of arcs in the network, counting EVERY arc. */ - int n_grid_arc; - /* number of arcs in the basic grid, including the arcs to/from - the super node */ - int n_source, n_sink; - /* number of source and sink nodes */ - int avg_degree; - /* average degree, arcs to and from the super node are counted */ - int t_supply; - /* total supply in the network */ - int n1, n2; - /* the two edges of the network grid. n1 >= n2 */ - struct imbalance *source_list, *sink_list; - /* head of the array of source/sink nodes */ - struct stat_para arc_costs; - /* the distribution of arc costs */ - struct stat_para capacities; - /* distribution of the capacities of the arcs */ - struct arcs *arc_list; - /* head of the arc list array. Arcs in this array are in the - order of grid_arcs, arcs to/from super node, and other arcs */ -}; - -#define G (csa->G) -#define v_rhs (csa->v_rhs) -#define a_cap (csa->a_cap) -#define a_cost (csa->a_cost) -#define seed (csa->seed) -#define seed_original (csa->seed_original) -#define two_way (csa->two_way) -#define n_node (csa->n_node) -#define n_arc (csa->n_arc) -#define n_grid_arc (csa->n_grid_arc) -#define n_source (csa->n_source) -#define n_sink (csa->n_sink) -#define avg_degree (csa->avg_degree) -#define t_supply (csa->t_supply) -#define n1 (csa->n1) -#define n2 (csa->n2) -#define source_list (csa->source_list) -#define sink_list (csa->sink_list) -#define arc_costs (csa->arc_costs) -#define capacities (csa->capacities) -#define arc_list (csa->arc_list) - -static void assign_capacities(struct csa *csa); -static void assign_costs(struct csa *csa); -static void assign_imbalance(struct csa *csa); -static int exponential(struct csa *csa, double lambda[1]); -static struct arcs *gen_additional_arcs(struct csa *csa, struct arcs - *arc_ptr); -static struct arcs *gen_basic_grid(struct csa *csa, struct arcs - *arc_ptr); -static void gen_more_arcs(struct csa *csa, struct arcs *arc_ptr); -static void generate(struct csa *csa); -static void output(struct csa *csa); -static double randy(struct csa *csa); -static void select_source_sinks(struct csa *csa); -static int uniform(struct csa *csa, double a[2]); - -int glp_gridgen(glp_graph *G_, int _v_rhs, int _a_cap, int _a_cost, - const int parm[1+14]) -{ struct csa _csa, *csa = &_csa; - int n, ret; - G = G_; - v_rhs = _v_rhs; - a_cap = _a_cap; - a_cost = _a_cost; - if (G != NULL) - { if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_gridgen: v_rhs = %d; invalid offset\n", v_rhs); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_gridgen: a_cap = %d; invalid offset\n", a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_gridgen: a_cost = %d; invalid offset\n", a_cost) - ; - } - /* Check the parameters for consistency. */ - if (!(parm[1] == 0 || parm[1] == 1)) - { ret = 1; - goto done; - } - if (parm[2] < 1) - { ret = 2; - goto done; - } - if (!(10 <= parm[3] && parm[3] <= 40000)) - { ret = 3; - goto done; - } - if (!(1 <= parm[4] && parm[4] <= 40000)) - { ret = 4; - goto done; - } - if (!(parm[5] >= 0 && parm[6] >= 0 && parm[5] + parm[6] <= - parm[3])) - { ret = 5; - goto done; - } - if (!(1 <= parm[7] && parm[7] <= parm[3])) - { ret = 6; - goto done; - } - if (parm[8] < 0) - { ret = 7; - goto done; - } - if (!(parm[9] == 1 || parm[9] == 2)) - { ret = 8; - goto done; - } - if (parm[9] == 1 && parm[10] > parm[11] || - parm[9] == 2 && parm[10] < 1) - { ret = 9; - goto done; - } - if (!(parm[12] == 1 || parm[12] == 2)) - { ret = 10; - goto done; - } - if (parm[12] == 1 && !(0 <= parm[13] && parm[13] <= parm[14]) || - parm[12] == 2 && parm[13] < 1) - { ret = 11; - goto done; - } - /* Initialize the graph object. */ - if (G != NULL) - { glp_erase_graph(G, G->v_size, G->a_size); - glp_set_graph_name(G, "GRIDGEN"); - } - /* Copy the generator parameters. */ - two_way = parm[1]; - seed_original = seed = parm[2]; - n_node = parm[3]; - n = parm[4]; - n_source = parm[5]; - n_sink = parm[6]; - avg_degree = parm[7]; - t_supply = parm[8]; - arc_costs.distribution = parm[9]; - if (parm[9] == 1) - { arc_costs.parameter[0] = parm[10]; - arc_costs.parameter[1] = parm[11]; - } - else - { arc_costs.parameter[0] = (double)parm[10] / 100.0; - arc_costs.parameter[1] = 0.0; - } - capacities.distribution = parm[12]; - if (parm[12] == 1) - { capacities.parameter[0] = parm[13]; - capacities.parameter[1] = parm[14]; - } - else - { capacities.parameter[0] = (double)parm[13] / 100.0; - capacities.parameter[1] = 0.0; - } - /* Calculate the edge lengths of the grid according to the - input. */ - if (n * n >= n_node) - { n1 = n; - n2 = (int)((double)n_node / (double)n + 0.5); - } - else - { n2 = n; - n1 = (int)((double)n_node / (double)n + 0.5); - } - /* Recalculate the total number of nodes and plus 1 for the super - node. */ - n_node = n1 * n2 + 1; - n_arc = n_node * avg_degree; - n_grid_arc = (two_way + 1) * ((n1 - 1) * n2 + (n2 - 1) * n1) + - n_source + n_sink; - if (n_grid_arc > n_arc) n_arc = n_grid_arc; - arc_list = xcalloc(n_arc, sizeof(struct arcs)); - source_list = xcalloc(n_source, sizeof(struct imbalance)); - sink_list = xcalloc(n_sink, sizeof(struct imbalance)); - /* Generate a random network. */ - generate(csa); - /* Output the network. */ - output(csa); - /* Free all allocated memory. */ - xfree(arc_list); - xfree(source_list); - xfree(sink_list); - /* The instance has been successfully generated. */ - ret = 0; -done: return ret; -} - -#undef random - -static void assign_capacities(struct csa *csa) -{ /* Assign a capacity to each arc. */ - struct arcs *arc_ptr = arc_list; - int (*random)(struct csa *csa, double *); - int i; - /* Determine the random number generator to use. */ - switch (arc_costs.distribution) - { case UNIFORM: - random = uniform; - break; - case EXPONENTIAL: - random = exponential; - break; - default: - xassert(csa != csa); - } - /* Assign capacities to grid arcs. */ - for (i = n_source + n_sink; i < n_grid_arc; i++, arc_ptr++) - arc_ptr->u = random(csa, capacities.parameter); - i = i - n_source - n_sink; - /* Assign capacities to arcs to/from supernode. */ - for (; i < n_grid_arc; i++, arc_ptr++) - arc_ptr->u = t_supply; - /* Assign capacities to all other arcs. */ - for (; i < n_arc; i++, arc_ptr++) - arc_ptr->u = random(csa, capacities.parameter); - return; -} - -static void assign_costs(struct csa *csa) -{ /* Assign a cost to each arc. */ - struct arcs *arc_ptr = arc_list; - int (*random)(struct csa *csa, double *); - int i; - /* A high cost assigned to arcs to/from the supernode. */ - int high_cost; - /* The maximum cost assigned to arcs in the base grid. */ - int max_cost = 0; - /* Determine the random number generator to use. */ - switch (arc_costs.distribution) - { case UNIFORM: - random = uniform; - break; - case EXPONENTIAL: - random = exponential; - break; - default: - xassert(csa != csa); - } - /* Assign costs to arcs in the base grid. */ - for (i = n_source + n_sink; i < n_grid_arc; i++, arc_ptr++) - { arc_ptr->cost = random(csa, arc_costs.parameter); - if (max_cost < arc_ptr->cost) max_cost = arc_ptr->cost; - } - i = i - n_source - n_sink; - /* Assign costs to arcs to/from the super node. */ - high_cost = max_cost * 2; - for (; i < n_grid_arc; i++, arc_ptr++) - arc_ptr->cost = high_cost; - /* Assign costs to all other arcs. */ - for (; i < n_arc; i++, arc_ptr++) - arc_ptr->cost = random(csa, arc_costs.parameter); - return; -} - -static void assign_imbalance(struct csa *csa) -{ /* Assign an imbalance to each node. */ - int total, i; - double avg; - struct imbalance *ptr; - /* assign the supply nodes */ - avg = 2.0 * t_supply / n_source; - do - { for (i = 1, total = t_supply, ptr = source_list + 1; - i < n_source; i++, ptr++) - { ptr->supply = (int)(randy(csa) * avg + 0.5); - total -= ptr->supply; - } - source_list->supply = total; - } - /* redo all if the assignment "overshooted" */ - while (total <= 0); - /* assign the demand nodes */ - avg = -2.0 * t_supply / n_sink; - do - { for (i = 1, total = t_supply, ptr = sink_list + 1; - i < n_sink; i++, ptr++) - { ptr->supply = (int)(randy(csa) * avg - 0.5); - total += ptr->supply; - } - sink_list->supply = - total; - } - while (total <= 0); - return; -} - -static int exponential(struct csa *csa, double lambda[1]) -{ /* Returns an "exponentially distributed" integer with parameter - lambda. */ - return ((int)(- lambda[0] * log((double)randy(csa)) + 0.5)); -} - -static struct arcs *gen_additional_arcs(struct csa *csa, struct arcs - *arc_ptr) -{ /* Generate an arc from each source to the supernode and from - supernode to each sink. */ - int i; - for (i = 0; i < n_source; i++, arc_ptr++) - { arc_ptr->from = source_list[i].node; - arc_ptr->to = n_node; - } - for (i = 0; i < n_sink; i++, arc_ptr++) - { arc_ptr->to = sink_list[i].node; - arc_ptr->from = n_node; - } - return arc_ptr; -} - -static struct arcs *gen_basic_grid(struct csa *csa, struct arcs - *arc_ptr) -{ /* Generate the basic grid. */ - int direction = 1, i, j, k; - if (two_way) - { /* Generate an arc in each direction. */ - for (i = 1; i < n_node; i += n1) - { for (j = i, k = j + n1 - 1; j < k; j++) - { arc_ptr->from = j; - arc_ptr->to = j + 1; - arc_ptr++; - arc_ptr->from = j + 1; - arc_ptr->to = j; - arc_ptr++; - } - } - for (i = 1; i <= n1; i++) - { for (j = i + n1; j < n_node; j += n1) - { arc_ptr->from = j; - arc_ptr->to = j - n1; - arc_ptr++; - arc_ptr->from = j - n1; - arc_ptr->to = j; - arc_ptr++; - } - } - } - else - { /* Generate one arc in each direction. */ - for (i = 1; i < n_node; i += n1) - { if (direction == 1) - j = i; - else - j = i + 1; - for (k = j + n1 - 1; j < k; j++) - { arc_ptr->from = j; - arc_ptr->to = j + direction; - arc_ptr++; - } - direction = - direction; - } - for (i = 1; i <= n1; i++) - { j = i + n1; - if (direction == 1) - { for (; j < n_node; j += n1) - { arc_ptr->from = j - n1; - arc_ptr->to = j; - arc_ptr++; - } - } - else - { for (; j < n_node; j += n1) - { arc_ptr->from = j - n1; - arc_ptr->to = j; - arc_ptr++; - } - } - direction = - direction; - } - } - return arc_ptr; -} - -static void gen_more_arcs(struct csa *csa, struct arcs *arc_ptr) -{ /* Generate random arcs to meet the specified density. */ - int i; - double ab[2]; - ab[0] = 0.9; - ab[1] = n_node - 0.99; /* upper limit is n_node-1 because the - supernode cannot be selected */ - for (i = n_grid_arc; i < n_arc; i++, arc_ptr++) - { arc_ptr->from = uniform(csa, ab); - arc_ptr->to = uniform(csa, ab); - if (arc_ptr->from == arc_ptr->to) - { arc_ptr--; - i--; - } - } - return; -} - -static void generate(struct csa *csa) -{ /* Generate a random network. */ - struct arcs *arc_ptr = arc_list; - arc_ptr = gen_basic_grid(csa, arc_ptr); - select_source_sinks(csa); - arc_ptr = gen_additional_arcs(csa, arc_ptr); - gen_more_arcs(csa, arc_ptr); - assign_costs(csa); - assign_capacities(csa); - assign_imbalance(csa); - return; -} - -static void output(struct csa *csa) -{ /* Output the network in DIMACS format. */ - struct arcs *arc_ptr; - struct imbalance *imb_ptr; - int i; - if (G != NULL) goto skip; - /* Output "c", "p" records. */ - xprintf("c generated by GRIDGEN\n"); - xprintf("c seed %d\n", seed_original); - xprintf("c nodes %d\n", n_node); - xprintf("c grid size %d X %d\n", n1, n2); - xprintf("c sources %d sinks %d\n", n_source, n_sink); - xprintf("c avg. degree %d\n", avg_degree); - xprintf("c supply %d\n", t_supply); - switch (arc_costs.distribution) - { case UNIFORM: - xprintf("c arc costs: UNIFORM distr. min %d max %d\n", - (int)arc_costs.parameter[0], - (int)arc_costs.parameter[1]); - break; - case EXPONENTIAL: - xprintf("c arc costs: EXPONENTIAL distr. lambda %d\n", - (int)arc_costs.parameter[0]); - break; - default: - xassert(csa != csa); - } - switch (capacities.distribution) - { case UNIFORM: - xprintf("c arc caps : UNIFORM distr. min %d max %d\n", - (int)capacities.parameter[0], - (int)capacities.parameter[1]); - break; - case EXPONENTIAL: - xprintf("c arc caps : EXPONENTIAL distr. %d lambda %d\n", - (int)capacities.parameter[0]); - break; - default: - xassert(csa != csa); - } -skip: if (G == NULL) - xprintf("p min %d %d\n", n_node, n_arc); - else - { glp_add_vertices(G, n_node); - if (v_rhs >= 0) - { double zero = 0.0; - for (i = 1; i <= n_node; i++) - { glp_vertex *v = G->v[i]; - memcpy((char *)v->data + v_rhs, &zero, sizeof(double)); - } - } - } - /* Output "n node supply". */ - for (i = 0, imb_ptr = source_list; i < n_source; i++, imb_ptr++) - { if (G == NULL) - xprintf("n %d %d\n", imb_ptr->node, imb_ptr->supply); - else - { if (v_rhs >= 0) - { double temp = (double)imb_ptr->supply; - glp_vertex *v = G->v[imb_ptr->node]; - memcpy((char *)v->data + v_rhs, &temp, sizeof(double)); - } - } - } - for (i = 0, imb_ptr = sink_list; i < n_sink; i++, imb_ptr++) - { if (G == NULL) - xprintf("n %d %d\n", imb_ptr->node, imb_ptr->supply); - else - { if (v_rhs >= 0) - { double temp = (double)imb_ptr->supply; - glp_vertex *v = G->v[imb_ptr->node]; - memcpy((char *)v->data + v_rhs, &temp, sizeof(double)); - } - } - } - /* Output "a from to lowcap=0 hicap cost". */ - for (i = 0, arc_ptr = arc_list; i < n_arc; i++, arc_ptr++) - { if (G == NULL) - xprintf("a %d %d 0 %d %d\n", arc_ptr->from, arc_ptr->to, - arc_ptr->u, arc_ptr->cost); - else - { glp_arc *a = glp_add_arc(G, arc_ptr->from, arc_ptr->to); - if (a_cap >= 0) - { double temp = (double)arc_ptr->u; - memcpy((char *)a->data + a_cap, &temp, sizeof(double)); - } - if (a_cost >= 0) - { double temp = (double)arc_ptr->cost; - memcpy((char *)a->data + a_cost, &temp, sizeof(double)); - } - } - } - return; -} - -static double randy(struct csa *csa) -{ /* Returns a random number between 0.0 and 1.0. - See Ward Cheney & David Kincaid, "Numerical Mathematics and - Computing," 2Ed, pp. 335. */ - seed = 16807 * seed % 2147483647; - if (seed < 0) seed = - seed; - return seed * 4.6566128752459e-10; -} - -static void select_source_sinks(struct csa *csa) -{ /* Randomly select the source nodes and sink nodes. */ - int i, *int_ptr; - int *temp_list; /* a temporary list of nodes */ - struct imbalance *ptr; - double ab[2]; /* parameter for random number generator */ - ab[0] = 0.9; - ab[1] = n_node - 0.99; /* upper limit is n_node-1 because the - supernode cannot be selected */ - temp_list = xcalloc(n_node, sizeof(int)); - for (i = 0, int_ptr = temp_list; i < n_node; i++, int_ptr++) - *int_ptr = 0; - /* Select the source nodes. */ - for (i = 0, ptr = source_list; i < n_source; i++, ptr++) - { ptr->node = uniform(csa, ab); - if (temp_list[ptr->node] == 1) /* check for duplicates */ - { ptr--; - i--; - } - else - temp_list[ptr->node] = 1; - } - /* Select the sink nodes. */ - for (i = 0, ptr = sink_list; i < n_sink; i++, ptr++) - { ptr->node = uniform(csa, ab); - if (temp_list[ptr->node] == 1) - { ptr--; - i--; - } - else - temp_list[ptr->node] = 1; - } - xfree(temp_list); - return; -} - -int uniform(struct csa *csa, double a[2]) -{ /* Generates an integer uniformly selected from [a[0],a[1]]. */ - return (int)((a[1] - a[0]) * randy(csa) + a[0] + 0.5); -} - -/**********************************************************************/ - -#if 0 -int main(void) -{ int parm[1+14]; - double temp; - scanf("%d", &parm[1]); - scanf("%d", &parm[2]); - scanf("%d", &parm[3]); - scanf("%d", &parm[4]); - scanf("%d", &parm[5]); - scanf("%d", &parm[6]); - scanf("%d", &parm[7]); - scanf("%d", &parm[8]); - scanf("%d", &parm[9]); - if (parm[9] == 1) - { scanf("%d", &parm[10]); - scanf("%d", &parm[11]); - } - else - { scanf("%le", &temp); - parm[10] = (int)(100.0 * temp + .5); - parm[11] = 0; - } - scanf("%d", &parm[12]); - if (parm[12] == 1) - { scanf("%d", &parm[13]); - scanf("%d", &parm[14]); - } - else - { scanf("%le", &temp); - parm[13] = (int)(100.0 * temp + .5); - parm[14] = 0; - } - glp_gridgen(NULL, 0, 0, 0, parm); - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/api/intfeas1.c b/code/3rd_glpk/api/intfeas1.c deleted file mode 100644 index 43064351..00000000 --- a/code/3rd_glpk/api/intfeas1.c +++ /dev/null @@ -1,267 +0,0 @@ -/* intfeas1.c (solve integer feasibility problem) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2011-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -int glp_intfeas1(glp_prob *P, int use_bound, int obj_bound) -{ /* solve integer feasibility problem */ - NPP *npp = NULL; - glp_prob *mip = NULL; - int *obj_ind = NULL; - double *obj_val = NULL; - int obj_row = 0; - int i, j, k, obj_len, temp, ret; -#if 0 /* 04/IV-2016 */ - /* check the problem object */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_intfeas1: P = %p; invalid problem object\n", - P); -#endif - if (P->tree != NULL) - xerror("glp_intfeas1: operation not allowed\n"); - /* integer solution is currently undefined */ - P->mip_stat = GLP_UNDEF; - P->mip_obj = 0.0; - /* check columns (variables) */ - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; -#if 0 /* binarization is not yet implemented */ - if (!(col->kind == GLP_IV || col->type == GLP_FX)) - { xprintf("glp_intfeas1: column %d: non-integer non-fixed var" - "iable not allowed\n", j); -#else - if (!((col->kind == GLP_IV && col->lb == 0.0 && col->ub == 1.0) - || col->type == GLP_FX)) - { xprintf("glp_intfeas1: column %d: non-binary non-fixed vari" - "able not allowed\n", j); -#endif - ret = GLP_EDATA; - goto done; - } - temp = (int)col->lb; - if ((double)temp != col->lb) - { if (col->type == GLP_FX) - xprintf("glp_intfeas1: column %d: fixed value %g is non-" - "integer or out of range\n", j, col->lb); - else - xprintf("glp_intfeas1: column %d: lower bound %g is non-" - "integer or out of range\n", j, col->lb); - ret = GLP_EDATA; - goto done; - } - temp = (int)col->ub; - if ((double)temp != col->ub) - { xprintf("glp_intfeas1: column %d: upper bound %g is non-int" - "eger or out of range\n", j, col->ub); - ret = GLP_EDATA; - goto done; - } - if (col->type == GLP_DB && col->lb > col->ub) - { xprintf("glp_intfeas1: column %d: lower bound %g is greater" - " than upper bound %g\n", j, col->lb, col->ub); - ret = GLP_EBOUND; - goto done; - } - } - /* check rows (constraints) */ - for (i = 1; i <= P->m; i++) - { GLPROW *row = P->row[i]; - GLPAIJ *aij; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { temp = (int)aij->val; - if ((double)temp != aij->val) - { xprintf("glp_intfeas1: row = %d, column %d: constraint c" - "oefficient %g is non-integer or out of range\n", - i, aij->col->j, aij->val); - ret = GLP_EDATA; - goto done; - } - } - temp = (int)row->lb; - if ((double)temp != row->lb) - { if (row->type == GLP_FX) - xprintf("glp_intfeas1: row = %d: fixed value %g is non-i" - "nteger or out of range\n", i, row->lb); - else - xprintf("glp_intfeas1: row = %d: lower bound %g is non-i" - "nteger or out of range\n", i, row->lb); - ret = GLP_EDATA; - goto done; - } - temp = (int)row->ub; - if ((double)temp != row->ub) - { xprintf("glp_intfeas1: row = %d: upper bound %g is non-inte" - "ger or out of range\n", i, row->ub); - ret = GLP_EDATA; - goto done; - } - if (row->type == GLP_DB && row->lb > row->ub) - { xprintf("glp_intfeas1: row %d: lower bound %g is greater th" - "an upper bound %g\n", i, row->lb, row->ub); - ret = GLP_EBOUND; - goto done; - } - } - /* check the objective function */ -#if 1 /* 08/I-2017 by cmatraki & mao */ - if (!use_bound) - { /* skip check if no obj. bound is specified */ - goto skip; - } -#endif - temp = (int)P->c0; - if ((double)temp != P->c0) - { xprintf("glp_intfeas1: objective constant term %g is non-integ" - "er or out of range\n", P->c0); - ret = GLP_EDATA; - goto done; - } - for (j = 1; j <= P->n; j++) - { temp = (int)P->col[j]->coef; - if ((double)temp != P->col[j]->coef) - { xprintf("glp_intfeas1: column %d: objective coefficient is " - "non-integer or out of range\n", j, P->col[j]->coef); - ret = GLP_EDATA; - goto done; - } - } -#if 1 /* 08/I-2017 by cmatraki & mao */ -skip: ; -#endif - /* save the objective function and set it to zero */ - obj_ind = xcalloc(1+P->n, sizeof(int)); - obj_val = xcalloc(1+P->n, sizeof(double)); - obj_len = 0; - obj_ind[0] = 0; - obj_val[0] = P->c0; - P->c0 = 0.0; - for (j = 1; j <= P->n; j++) - { if (P->col[j]->coef != 0.0) - { obj_len++; - obj_ind[obj_len] = j; - obj_val[obj_len] = P->col[j]->coef; - P->col[j]->coef = 0.0; - } - } - /* add inequality to bound the objective function, if required */ - if (!use_bound) - xprintf("Will search for ANY feasible solution\n"); - else - { xprintf("Will search only for solution not worse than %d\n", - obj_bound); - obj_row = glp_add_rows(P, 1); - glp_set_mat_row(P, obj_row, obj_len, obj_ind, obj_val); - if (P->dir == GLP_MIN) - glp_set_row_bnds(P, obj_row, - GLP_UP, 0.0, (double)obj_bound - obj_val[0]); - else if (P->dir == GLP_MAX) - glp_set_row_bnds(P, obj_row, - GLP_LO, (double)obj_bound - obj_val[0], 0.0); - else - xassert(P != P); - } - /* create preprocessor workspace */ - xprintf("Translating to CNF-SAT...\n"); - xprintf("Original problem has %d row%s, %d column%s, and %d non-z" - "ero%s\n", P->m, P->m == 1 ? "" : "s", P->n, P->n == 1 ? "" : - "s", P->nnz, P->nnz == 1 ? "" : "s"); - npp = npp_create_wksp(); - /* load the original problem into the preprocessor workspace */ - npp_load_prob(npp, P, GLP_OFF, GLP_MIP, GLP_OFF); - /* perform translation to SAT-CNF problem instance */ - ret = npp_sat_encode_prob(npp); - if (ret == 0) - ; - else if (ret == GLP_ENOPFS) - xprintf("PROBLEM HAS NO INTEGER FEASIBLE SOLUTION\n"); - else if (ret == GLP_ERANGE) - xprintf("glp_intfeas1: translation to SAT-CNF failed because o" - "f integer overflow\n"); - else - xassert(ret != ret); - if (ret != 0) - goto done; - /* build SAT-CNF problem instance and try to solve it */ - mip = glp_create_prob(); - npp_build_prob(npp, mip); - ret = glp_minisat1(mip); - /* only integer feasible solution can be postprocessed */ - if (!(mip->mip_stat == GLP_OPT || mip->mip_stat == GLP_FEAS)) - { P->mip_stat = mip->mip_stat; - goto done; - } - /* postprocess the solution found */ - npp_postprocess(npp, mip); - /* the transformed problem is no longer needed */ - glp_delete_prob(mip), mip = NULL; - /* store solution to the original problem object */ - npp_unload_sol(npp, P); - /* change the solution status to 'integer feasible' */ - P->mip_stat = GLP_FEAS; - /* check integer feasibility */ - for (i = 1; i <= P->m; i++) - { GLPROW *row; - GLPAIJ *aij; - double sum; - row = P->row[i]; - sum = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - sum += aij->val * aij->col->mipx; - xassert(sum == row->mipx); - if (row->type == GLP_LO || row->type == GLP_DB || - row->type == GLP_FX) - xassert(sum >= row->lb); - if (row->type == GLP_UP || row->type == GLP_DB || - row->type == GLP_FX) - xassert(sum <= row->ub); - } - /* compute value of the original objective function */ - P->mip_obj = obj_val[0]; - for (k = 1; k <= obj_len; k++) - P->mip_obj += obj_val[k] * P->col[obj_ind[k]]->mipx; - xprintf("Objective value = %17.9e\n", P->mip_obj); -done: /* delete the transformed problem, if it exists */ - if (mip != NULL) - glp_delete_prob(mip); - /* delete the preprocessor workspace, if it exists */ - if (npp != NULL) - npp_delete_wksp(npp); - /* remove inequality used to bound the objective function */ - if (obj_row > 0) - { int ind[1+1]; - ind[1] = obj_row; - glp_del_rows(P, 1, ind); - } - /* restore the original objective function */ - if (obj_ind != NULL) - { P->c0 = obj_val[0]; - for (k = 1; k <= obj_len; k++) - P->col[obj_ind[k]]->coef = obj_val[k]; - xfree(obj_ind); - xfree(obj_val); - } - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/maxffalg.c b/code/3rd_glpk/api/maxffalg.c deleted file mode 100644 index 0f3f9b04..00000000 --- a/code/3rd_glpk/api/maxffalg.c +++ /dev/null @@ -1,130 +0,0 @@ -/* maxffalg.c (find maximal flow with Ford-Fulkerson algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ffalg.h" -#include "glpk.h" - -int glp_maxflow_ffalg(glp_graph *G, int s, int t, int a_cap, - double *sol, int a_x, int v_cut) -{ /* find maximal flow with Ford-Fulkerson algorithm */ - glp_vertex *v; - glp_arc *a; - int nv, na, i, k, flag, *tail, *head, *cap, *x, ret; - char *cut; - double temp; - if (!(1 <= s && s <= G->nv)) - xerror("glp_maxflow_ffalg: s = %d; source node number out of r" - "ange\n", s); - if (!(1 <= t && t <= G->nv)) - xerror("glp_maxflow_ffalg: t = %d: sink node number out of ran" - "ge\n", t); - if (s == t) - xerror("glp_maxflow_ffalg: s = t = %d; source and sink nodes m" - "ust be distinct\n", s); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_maxflow_ffalg: a_cap = %d; invalid offset\n", - a_cap); - if (v_cut >= 0 && v_cut > G->v_size - (int)sizeof(int)) - xerror("glp_maxflow_ffalg: v_cut = %d; invalid offset\n", - v_cut); - /* allocate working arrays */ - nv = G->nv; - na = G->na; - tail = xcalloc(1+na, sizeof(int)); - head = xcalloc(1+na, sizeof(int)); - cap = xcalloc(1+na, sizeof(int)); - x = xcalloc(1+na, sizeof(int)); - if (v_cut < 0) - cut = NULL; - else - cut = xcalloc(1+nv, sizeof(char)); - /* copy the flow network */ - k = 0; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { k++; - tail[k] = a->tail->i; - head[k] = a->head->i; - if (tail[k] == head[k]) - { ret = GLP_EDATA; - goto done; - } - if (a_cap >= 0) - memcpy(&temp, (char *)a->data + a_cap, sizeof(double)); - else - temp = 1.0; - if (!(0.0 <= temp && temp <= (double)INT_MAX && - temp == floor(temp))) - { ret = GLP_EDATA; - goto done; - } - cap[k] = (int)temp; - } - } - xassert(k == na); - /* find maximal flow in the flow network */ - ffalg(nv, na, tail, head, s, t, cap, x, cut); - ret = 0; - /* store solution components */ - /* (objective function = total flow through the network) */ - if (sol != NULL) - { temp = 0.0; - for (k = 1; k <= na; k++) - { if (tail[k] == s) - temp += (double)x[k]; - else if (head[k] == s) - temp -= (double)x[k]; - } - *sol = temp; - } - /* (arc flows) */ - if (a_x >= 0) - { k = 0; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { temp = (double)x[++k]; - memcpy((char *)a->data + a_x, &temp, sizeof(double)); - } - } - } - /* (node flags) */ - if (v_cut >= 0) - { for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - flag = cut[i]; - memcpy((char *)v->data + v_cut, &flag, sizeof(int)); - } - } -done: /* free working arrays */ - xfree(tail); - xfree(head); - xfree(cap); - xfree(x); - if (cut != NULL) xfree(cut); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/maxflp.c b/code/3rd_glpk/api/maxflp.c deleted file mode 100644 index 1135b78c..00000000 --- a/code/3rd_glpk/api/maxflp.c +++ /dev/null @@ -1,114 +0,0 @@ -/* maxflp.c (convert maximum flow problem to LP) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_maxflow_lp - convert maximum flow problem to LP -* -* SYNOPSIS -* -* void glp_maxflow_lp(glp_prob *lp, glp_graph *G, int names, int s, -* int t, int a_cap); -* -* DESCRIPTION -* -* The routine glp_maxflow_lp builds an LP problem, which corresponds -* to the maximum flow problem on the specified network G. */ - -void glp_maxflow_lp(glp_prob *lp, glp_graph *G, int names, int s, - int t, int a_cap) -{ glp_vertex *v; - glp_arc *a; - int i, j, type, ind[1+2]; - double cap, val[1+2]; - if (!(names == GLP_ON || names == GLP_OFF)) - xerror("glp_maxflow_lp: names = %d; invalid parameter\n", - names); - if (!(1 <= s && s <= G->nv)) - xerror("glp_maxflow_lp: s = %d; source node number out of rang" - "e\n", s); - if (!(1 <= t && t <= G->nv)) - xerror("glp_maxflow_lp: t = %d: sink node number out of range " - "\n", t); - if (s == t) - xerror("glp_maxflow_lp: s = t = %d; source and sink nodes must" - " be distinct\n", s); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_maxflow_lp: a_cap = %d; invalid offset\n", a_cap); - glp_erase_prob(lp); - if (names) glp_set_prob_name(lp, G->name); - glp_set_obj_dir(lp, GLP_MAX); - glp_add_rows(lp, G->nv); - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (names) glp_set_row_name(lp, i, v->name); - if (i == s) - type = GLP_LO; - else if (i == t) - type = GLP_UP; - else - type = GLP_FX; - glp_set_row_bnds(lp, i, type, 0.0, 0.0); - } - if (G->na > 0) glp_add_cols(lp, G->na); - for (i = 1, j = 0; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { j++; - if (names) - { char name[50+1]; - sprintf(name, "x[%d,%d]", a->tail->i, a->head->i); - xassert(strlen(name) < sizeof(name)); - glp_set_col_name(lp, j, name); - } - if (a->tail->i != a->head->i) - { ind[1] = a->tail->i, val[1] = +1.0; - ind[2] = a->head->i, val[2] = -1.0; - glp_set_mat_col(lp, j, 2, ind, val); - } - if (a_cap >= 0) - memcpy(&cap, (char *)a->data + a_cap, sizeof(double)); - else - cap = 1.0; - if (cap == DBL_MAX) - type = GLP_LO; - else if (cap != 0.0) - type = GLP_DB; - else - type = GLP_FX; - glp_set_col_bnds(lp, j, type, 0.0, cap); - if (a->tail->i == s) - glp_set_obj_coef(lp, j, +1.0); - else if (a->head->i == s) - glp_set_obj_coef(lp, j, -1.0); - } - } - xassert(j == G->na); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/mcflp.c b/code/3rd_glpk/api/mcflp.c deleted file mode 100644 index 5cd24060..00000000 --- a/code/3rd_glpk/api/mcflp.c +++ /dev/null @@ -1,114 +0,0 @@ -/* mcflp.c (convert minimum cost flow problem to LP) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_mincost_lp - convert minimum cost flow problem to LP -* -* SYNOPSIS -* -* void glp_mincost_lp(glp_prob *lp, glp_graph *G, int names, -* int v_rhs, int a_low, int a_cap, int a_cost); -* -* DESCRIPTION -* -* The routine glp_mincost_lp builds an LP problem, which corresponds -* to the minimum cost flow problem on the specified network G. */ - -void glp_mincost_lp(glp_prob *lp, glp_graph *G, int names, int v_rhs, - int a_low, int a_cap, int a_cost) -{ glp_vertex *v; - glp_arc *a; - int i, j, type, ind[1+2]; - double rhs, low, cap, cost, val[1+2]; - if (!(names == GLP_ON || names == GLP_OFF)) - xerror("glp_mincost_lp: names = %d; invalid parameter\n", - names); - if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_mincost_lp: v_rhs = %d; invalid offset\n", v_rhs); - if (a_low >= 0 && a_low > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_lp: a_low = %d; invalid offset\n", a_low); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_lp: a_cap = %d; invalid offset\n", a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_lp: a_cost = %d; invalid offset\n", a_cost) - ; - glp_erase_prob(lp); - if (names) glp_set_prob_name(lp, G->name); - if (G->nv > 0) glp_add_rows(lp, G->nv); - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (names) glp_set_row_name(lp, i, v->name); - if (v_rhs >= 0) - memcpy(&rhs, (char *)v->data + v_rhs, sizeof(double)); - else - rhs = 0.0; - glp_set_row_bnds(lp, i, GLP_FX, rhs, rhs); - } - if (G->na > 0) glp_add_cols(lp, G->na); - for (i = 1, j = 0; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { j++; - if (names) - { char name[50+1]; - sprintf(name, "x[%d,%d]", a->tail->i, a->head->i); - xassert(strlen(name) < sizeof(name)); - glp_set_col_name(lp, j, name); - } - if (a->tail->i != a->head->i) - { ind[1] = a->tail->i, val[1] = +1.0; - ind[2] = a->head->i, val[2] = -1.0; - glp_set_mat_col(lp, j, 2, ind, val); - } - if (a_low >= 0) - memcpy(&low, (char *)a->data + a_low, sizeof(double)); - else - low = 0.0; - if (a_cap >= 0) - memcpy(&cap, (char *)a->data + a_cap, sizeof(double)); - else - cap = 1.0; - if (cap == DBL_MAX) - type = GLP_LO; - else if (low != cap) - type = GLP_DB; - else - type = GLP_FX; - glp_set_col_bnds(lp, j, type, low, cap); - if (a_cost >= 0) - memcpy(&cost, (char *)a->data + a_cost, sizeof(double)); - else - cost = 0.0; - glp_set_obj_coef(lp, j, cost); - } - } - xassert(j == G->na); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/mcfokalg.c b/code/3rd_glpk/api/mcfokalg.c deleted file mode 100644 index 786dc71b..00000000 --- a/code/3rd_glpk/api/mcfokalg.c +++ /dev/null @@ -1,221 +0,0 @@ -/* mcfokalg.c (find minimum-cost flow with out-of-kilter algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "okalg.h" - -int glp_mincost_okalg(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, double *sol, int a_x, int v_pi) -{ /* find minimum-cost flow with out-of-kilter algorithm */ - glp_vertex *v; - glp_arc *a; - int nv, na, i, k, s, t, *tail, *head, *low, *cap, *cost, *x, *pi, - ret; - double sum, temp; - if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_mincost_okalg: v_rhs = %d; invalid offset\n", - v_rhs); - if (a_low >= 0 && a_low > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_okalg: a_low = %d; invalid offset\n", - a_low); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_okalg: a_cap = %d; invalid offset\n", - a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_okalg: a_cost = %d; invalid offset\n", - a_cost); - if (a_x >= 0 && a_x > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_okalg: a_x = %d; invalid offset\n", a_x); - if (v_pi >= 0 && v_pi > G->v_size - (int)sizeof(double)) - xerror("glp_mincost_okalg: v_pi = %d; invalid offset\n", v_pi); - /* s is artificial source node */ - s = G->nv + 1; - /* t is artificial sink node */ - t = s + 1; - /* nv is the total number of nodes in the resulting network */ - nv = t; - /* na is the total number of arcs in the resulting network */ - na = G->na + 1; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (v_rhs >= 0) - memcpy(&temp, (char *)v->data + v_rhs, sizeof(double)); - else - temp = 0.0; - if (temp != 0.0) na++; - } - /* allocate working arrays */ - tail = xcalloc(1+na, sizeof(int)); - head = xcalloc(1+na, sizeof(int)); - low = xcalloc(1+na, sizeof(int)); - cap = xcalloc(1+na, sizeof(int)); - cost = xcalloc(1+na, sizeof(int)); - x = xcalloc(1+na, sizeof(int)); - pi = xcalloc(1+nv, sizeof(int)); - /* construct the resulting network */ - k = 0; - /* (original arcs) */ - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { k++; - tail[k] = a->tail->i; - head[k] = a->head->i; - if (tail[k] == head[k]) - { ret = GLP_EDATA; - goto done; - } - if (a_low >= 0) - memcpy(&temp, (char *)a->data + a_low, sizeof(double)); - else - temp = 0.0; - if (!(0.0 <= temp && temp <= (double)INT_MAX && - temp == floor(temp))) - { ret = GLP_EDATA; - goto done; - } - low[k] = (int)temp; - if (a_cap >= 0) - memcpy(&temp, (char *)a->data + a_cap, sizeof(double)); - else - temp = 1.0; - if (!((double)low[k] <= temp && temp <= (double)INT_MAX && - temp == floor(temp))) - { ret = GLP_EDATA; - goto done; - } - cap[k] = (int)temp; - if (a_cost >= 0) - memcpy(&temp, (char *)a->data + a_cost, sizeof(double)); - else - temp = 0.0; - if (!(fabs(temp) <= (double)INT_MAX && temp == floor(temp))) - { ret = GLP_EDATA; - goto done; - } - cost[k] = (int)temp; - } - } - /* (artificial arcs) */ - sum = 0.0; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (v_rhs >= 0) - memcpy(&temp, (char *)v->data + v_rhs, sizeof(double)); - else - temp = 0.0; - if (!(fabs(temp) <= (double)INT_MAX && temp == floor(temp))) - { ret = GLP_EDATA; - goto done; - } - if (temp > 0.0) - { /* artificial arc from s to original source i */ - k++; - tail[k] = s; - head[k] = i; - low[k] = cap[k] = (int)(+temp); /* supply */ - cost[k] = 0; - sum += (double)temp; - } - else if (temp < 0.0) - { /* artificial arc from original sink i to t */ - k++; - tail[k] = i; - head[k] = t; - low[k] = cap[k] = (int)(-temp); /* demand */ - cost[k] = 0; - } - } - /* (feedback arc from t to s) */ - k++; - xassert(k == na); - tail[k] = t; - head[k] = s; - if (sum > (double)INT_MAX) - { ret = GLP_EDATA; - goto done; - } - low[k] = cap[k] = (int)sum; /* total supply/demand */ - cost[k] = 0; - /* find minimal-cost circulation in the resulting network */ - ret = okalg(nv, na, tail, head, low, cap, cost, x, pi); - switch (ret) - { case 0: - /* optimal circulation found */ - ret = 0; - break; - case 1: - /* no feasible circulation exists */ - ret = GLP_ENOPFS; - break; - case 2: - /* integer overflow occured */ - ret = GLP_ERANGE; - goto done; - case 3: - /* optimality test failed (logic error) */ - ret = GLP_EFAIL; - goto done; - default: - xassert(ret != ret); - } - /* store solution components */ - /* (objective function = the total cost) */ - if (sol != NULL) - { temp = 0.0; - for (k = 1; k <= na; k++) - temp += (double)cost[k] * (double)x[k]; - *sol = temp; - } - /* (arc flows) */ - if (a_x >= 0) - { k = 0; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { temp = (double)x[++k]; - memcpy((char *)a->data + a_x, &temp, sizeof(double)); - } - } - } - /* (node potentials = Lagrange multipliers) */ - if (v_pi >= 0) - { for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - temp = - (double)pi[i]; - memcpy((char *)v->data + v_pi, &temp, sizeof(double)); - } - } -done: /* free working arrays */ - xfree(tail); - xfree(head); - xfree(low); - xfree(cap); - xfree(cost); - xfree(x); - xfree(pi); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/mcfrelax.c b/code/3rd_glpk/api/mcfrelax.c deleted file mode 100644 index 9b34949a..00000000 --- a/code/3rd_glpk/api/mcfrelax.c +++ /dev/null @@ -1,251 +0,0 @@ -/* mcfrelax.c (find minimum-cost flow with RELAX-IV) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "relax4.h" - -static int overflow(int u, int v) -{ /* check for integer overflow on computing u + v */ - if (u > 0 && v > 0 && u + v < 0) return 1; - if (u < 0 && v < 0 && u + v > 0) return 1; - return 0; -} - -int glp_mincost_relax4(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, int crash, double *sol, int a_x, int a_rc) -{ /* find minimum-cost flow with Bertsekas-Tseng relaxation method - (RELAX-IV) */ - glp_vertex *v; - glp_arc *a; - struct relax4_csa csa; - int i, k, large, n, na, ret; - double cap, cost, low, rc, rhs, sum, x; - if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_mincost_relax4: v_rhs = %d; invalid offset\n", - v_rhs); - if (a_low >= 0 && a_low > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_relax4: a_low = %d; invalid offset\n", - a_low); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_relax4: a_cap = %d; invalid offset\n", - a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_relax4: a_cost = %d; invalid offset\n", - a_cost); - if (a_x >= 0 && a_x > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_relax4: a_x = %d; invalid offset\n", - a_x); - if (a_rc >= 0 && a_rc > G->a_size - (int)sizeof(double)) - xerror("glp_mincost_relax4: a_rc = %d; invalid offset\n", - a_rc); - csa.n = n = G->nv; /* number of nodes */ - csa.na = na = G->na; /* number of arcs */ - csa.large = large = INT_MAX / 4; - csa.repeat = 0; - csa.crash = crash; - /* allocate working arrays */ - csa.startn = xcalloc(1+na, sizeof(int)); - csa.endn = xcalloc(1+na, sizeof(int)); - csa.fou = xcalloc(1+n, sizeof(int)); - csa.nxtou = xcalloc(1+na, sizeof(int)); - csa.fin = xcalloc(1+n, sizeof(int)); - csa.nxtin = xcalloc(1+na, sizeof(int)); - csa.rc = xcalloc(1+na, sizeof(int)); - csa.u = xcalloc(1+na, sizeof(int)); - csa.dfct = xcalloc(1+n, sizeof(int)); - csa.x = xcalloc(1+na, sizeof(int)); - csa.label = xcalloc(1+n, sizeof(int)); - csa.prdcsr = xcalloc(1+n, sizeof(int)); - csa.save = xcalloc(1+na, sizeof(int)); - csa.tfstou = xcalloc(1+n, sizeof(int)); - csa.tnxtou = xcalloc(1+na, sizeof(int)); - csa.tfstin = xcalloc(1+n, sizeof(int)); - csa.tnxtin = xcalloc(1+na, sizeof(int)); - csa.nxtqueue = xcalloc(1+n, sizeof(int)); - csa.scan = xcalloc(1+n, sizeof(char)); - csa.mark = xcalloc(1+n, sizeof(char)); - if (crash) - { csa.extend_arc = xcalloc(1+n, sizeof(int)); - csa.sb_level = xcalloc(1+n, sizeof(int)); - csa.sb_arc = xcalloc(1+n, sizeof(int)); - } - else - { csa.extend_arc = NULL; - csa.sb_level = NULL; - csa.sb_arc = NULL; - } - /* scan nodes */ - for (i = 1; i <= n; i++) - { v = G->v[i]; - /* get supply at i-th node */ - if (v_rhs >= 0) - memcpy(&rhs, (char *)v->data + v_rhs, sizeof(double)); - else - rhs = 0.0; - if (!(fabs(rhs) <= (double)large && rhs == floor(rhs))) - { ret = GLP_EDATA; - goto done; - } - /* set demand at i-th node */ - csa.dfct[i] = -(int)rhs; - } - /* scan arcs */ - k = 0; - for (i = 1; i <= n; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { k++; - /* set endpoints of k-th arc */ - if (a->tail->i == a->head->i) - { /* self-loops not allowed */ - ret = GLP_EDATA; - goto done; - } - csa.startn[k] = a->tail->i; - csa.endn[k] = a->head->i; - /* set per-unit cost for k-th arc flow */ - if (a_cost >= 0) - memcpy(&cost, (char *)a->data + a_cost, sizeof(double)); - else - cost = 0.0; - if (!(fabs(cost) <= (double)large && cost == floor(cost))) - { ret = GLP_EDATA; - goto done; - } - csa.rc[k] = (int)cost; - /* get lower bound for k-th arc flow */ - if (a_low >= 0) - memcpy(&low, (char *)a->data + a_low, sizeof(double)); - else - low = 0.0; - if (!(0.0 <= low && low <= (double)large && - low == floor(low))) - { ret = GLP_EDATA; - goto done; - } - /* get upper bound for k-th arc flow */ - if (a_cap >= 0) - memcpy(&cap, (char *)a->data + a_cap, sizeof(double)); - else - cap = 1.0; - if (!(low <= cap && cap <= (double)large && - cap == floor(cap))) - { ret = GLP_EDATA; - goto done; - } - /* substitute x = x' + low, where 0 <= x' <= cap - low */ - csa.u[k] = (int)(cap - low); - /* correct demands at endpoints of k-th arc */ - if (overflow(csa.dfct[a->tail->i], +low)) - { ret = GLP_ERANGE; - goto done; - } -#if 0 /* 29/IX-2017 */ - csa.dfct[a->tail->i] += low; -#else - csa.dfct[a->tail->i] += (int)low; -#endif - if (overflow(csa.dfct[a->head->i], -low)) - { ret = GLP_ERANGE; - goto done; - } -#if 0 /* 29/IX-2017 */ - csa.dfct[a->head->i] -= low; -#else - csa.dfct[a->head->i] -= (int)low; -#endif - } - } - /* construct linked list for network topology */ - relax4_inidat(&csa); - /* find minimum-cost flow */ - ret = relax4(&csa); - if (ret != 0) - { /* problem is found to be infeasible */ - xassert(1 <= ret && ret <= 8); - ret = GLP_ENOPFS; - goto done; - } - /* store solution */ - sum = 0.0; - k = 0; - for (i = 1; i <= n; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { k++; - /* get lower bound for k-th arc flow */ - if (a_low >= 0) - memcpy(&low, (char *)a->data + a_low, sizeof(double)); - else - low = 0.0; - /* store original flow x = x' + low thru k-th arc */ - x = (double)csa.x[k] + low; - if (a_x >= 0) - memcpy((char *)a->data + a_x, &x, sizeof(double)); - /* store reduced cost for k-th arc flow */ - rc = (double)csa.rc[k]; - if (a_rc >= 0) - memcpy((char *)a->data + a_rc, &rc, sizeof(double)); - /* get per-unit cost for k-th arc flow */ - if (a_cost >= 0) - memcpy(&cost, (char *)a->data + a_cost, sizeof(double)); - else - cost = 0.0; - /* compute the total cost */ - sum += cost * x; - } - } - /* store the total cost */ - if (sol != NULL) - *sol = sum; -done: /* free working arrays */ - xfree(csa.startn); - xfree(csa.endn); - xfree(csa.fou); - xfree(csa.nxtou); - xfree(csa.fin); - xfree(csa.nxtin); - xfree(csa.rc); - xfree(csa.u); - xfree(csa.dfct); - xfree(csa.x); - xfree(csa.label); - xfree(csa.prdcsr); - xfree(csa.save); - xfree(csa.tfstou); - xfree(csa.tnxtou); - xfree(csa.tfstin); - xfree(csa.tnxtin); - xfree(csa.nxtqueue); - xfree(csa.scan); - xfree(csa.mark); - if (crash) - { xfree(csa.extend_arc); - xfree(csa.sb_level); - xfree(csa.sb_arc); - } - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/minisat1.c b/code/3rd_glpk/api/minisat1.c deleted file mode 100644 index a669c487..00000000 --- a/code/3rd_glpk/api/minisat1.c +++ /dev/null @@ -1,161 +0,0 @@ -/* minisat1.c (driver to MiniSat solver) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2011-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "minisat.h" -#include "prob.h" - -int glp_minisat1(glp_prob *P) -{ /* solve CNF-SAT problem with MiniSat solver */ - solver *s; - GLPAIJ *aij; - int i, j, len, ret, *ind; - double sum; -#if 0 /* 04/IV-2016 */ - /* check problem object */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_minisat1: P = %p; invalid problem object\n", - P); -#endif - if (P->tree != NULL) - xerror("glp_minisat1: operation not allowed\n"); - /* integer solution is currently undefined */ - P->mip_stat = GLP_UNDEF; - P->mip_obj = 0.0; - /* check that problem object encodes CNF-SAT instance */ - if (glp_check_cnfsat(P) != 0) - { xprintf("glp_minisat1: problem object does not encode CNF-SAT " - "instance\n"); - ret = GLP_EDATA; - goto done; - } -#if 0 /* 08/I-2017 by cmatraki */ -#if 1 /* 07/XI-2015 */ - if (sizeof(void *) != sizeof(int)) - { xprintf("glp_minisat1: sorry, MiniSat solver is not supported " - "on 64-bit platforms\n"); - ret = GLP_EFAIL; - goto done; - } -#endif -#else - if (sizeof(void *) != sizeof(size_t)) - { xprintf("glp_minisat1: sorry, MiniSat solver is not supported " - "on this platform\n"); - ret = GLP_EFAIL; - goto done; - } -#endif - /* solve CNF-SAT problem */ - xprintf("Solving CNF-SAT problem...\n"); - xprintf("Instance has %d variable%s, %d clause%s, and %d literal%" - "s\n", P->n, P->n == 1 ? "" : "s", P->m, P->m == 1 ? "" : "s", - P->nnz, P->nnz == 1 ? "" : "s"); - /* if CNF-SAT has no clauses, it is satisfiable */ - if (P->m == 0) - { P->mip_stat = GLP_OPT; - for (j = 1; j <= P->n; j++) - P->col[j]->mipx = 0.0; - goto fini; - } - /* if CNF-SAT has an empty clause, it is unsatisfiable */ - for (i = 1; i <= P->m; i++) - { if (P->row[i]->ptr == NULL) - { P->mip_stat = GLP_NOFEAS; - goto fini; - } - } - /* prepare input data for the solver */ - s = solver_new(); - solver_setnvars(s, P->n); - ind = xcalloc(1+P->n, sizeof(int)); - for (i = 1; i <= P->m; i++) - { len = 0; - for (aij = P->row[i]->ptr; aij != NULL; aij = aij->r_next) - { ind[++len] = toLit(aij->col->j-1); - if (aij->val < 0.0) - ind[len] = lit_neg(ind[len]); - } - xassert(len > 0); -#if 0 /* 08/I-2017 by cmatraki */ - xassert(solver_addclause(s, &ind[1], &ind[1+len])); -#else - if (!solver_addclause(s, &ind[1], &ind[1+len])) - { /* found trivial conflict */ - xfree(ind); - solver_delete(s); - P->mip_stat = GLP_NOFEAS; - goto fini; - } -#endif - } - xfree(ind); - /* call the solver */ - s->verbosity = 1; - if (solver_solve(s, 0, 0)) - { /* instance is reported as satisfiable */ - P->mip_stat = GLP_OPT; - /* copy solution to the problem object */ - xassert(s->model.size == P->n); - for (j = 1; j <= P->n; j++) - { P->col[j]->mipx = - s->model.ptr[j-1] == l_True ? 1.0 : 0.0; - } - /* compute row values */ - for (i = 1; i <= P->m; i++) - { sum = 0; - for (aij = P->row[i]->ptr; aij != NULL; aij = aij->r_next) - sum += aij->val * aij->col->mipx; - P->row[i]->mipx = sum; - } - /* check integer feasibility */ - for (i = 1; i <= P->m; i++) - { if (P->row[i]->mipx < P->row[i]->lb) - { /* solution is wrong */ - P->mip_stat = GLP_UNDEF; - break; - } - } - } - else - { /* instance is reported as unsatisfiable */ - P->mip_stat = GLP_NOFEAS; - } - solver_delete(s); -fini: /* report the instance status */ - if (P->mip_stat == GLP_OPT) - { xprintf("SATISFIABLE\n"); - ret = 0; - } - else if (P->mip_stat == GLP_NOFEAS) - { xprintf("UNSATISFIABLE\n"); - ret = 0; - } - else - { xprintf("glp_minisat1: solver failed\n"); - ret = GLP_EFAIL; - } -done: return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/mpl.c b/code/3rd_glpk/api/mpl.c deleted file mode 100644 index cfa6f75b..00000000 --- a/code/3rd_glpk/api/mpl.c +++ /dev/null @@ -1,269 +0,0 @@ -/* mpl.c (processing model in GNU MathProg language) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mpl.h" -#include "prob.h" - -glp_tran *glp_mpl_alloc_wksp(void) -{ /* allocate the MathProg translator workspace */ - glp_tran *tran; - tran = mpl_initialize(); - return tran; -} - -void glp_mpl_init_rand(glp_tran *tran, int seed) -{ /* initialize pseudo-random number generator */ - if (tran->phase != 0) - xerror("glp_mpl_init_rand: invalid call sequence\n"); - rng_init_rand(tran->rand, seed); - return; -} - -int glp_mpl_read_model(glp_tran *tran, const char *fname, int skip) -{ /* read and translate model section */ - int ret; - if (tran->phase != 0) - xerror("glp_mpl_read_model: invalid call sequence\n"); - ret = mpl_read_model(tran, (char *)fname, skip); - if (ret == 1 || ret == 2) - ret = 0; - else if (ret == 4) - ret = 1; - else - xassert(ret != ret); - return ret; -} - -int glp_mpl_read_data(glp_tran *tran, const char *fname) -{ /* read and translate data section */ - int ret; - if (!(tran->phase == 1 || tran->phase == 2)) - xerror("glp_mpl_read_data: invalid call sequence\n"); - ret = mpl_read_data(tran, (char *)fname); - if (ret == 2) - ret = 0; - else if (ret == 4) - ret = 1; - else - xassert(ret != ret); - return ret; -} - -int glp_mpl_generate(glp_tran *tran, const char *fname) -{ /* generate the model */ - int ret; - if (!(tran->phase == 1 || tran->phase == 2)) - xerror("glp_mpl_generate: invalid call sequence\n"); - ret = mpl_generate(tran, (char *)fname); - if (ret == 3) - ret = 0; - else if (ret == 4) - ret = 1; - return ret; -} - -void glp_mpl_build_prob(glp_tran *tran, glp_prob *prob) -{ /* build LP/MIP problem instance from the model */ - int m, n, i, j, t, kind, type, len, *ind; - double lb, ub, *val; - if (tran->phase != 3) - xerror("glp_mpl_build_prob: invalid call sequence\n"); - /* erase the problem object */ - glp_erase_prob(prob); - /* set problem name */ - glp_set_prob_name(prob, mpl_get_prob_name(tran)); - /* build rows (constraints) */ - m = mpl_get_num_rows(tran); - if (m > 0) - glp_add_rows(prob, m); - for (i = 1; i <= m; i++) - { /* set row name */ - glp_set_row_name(prob, i, mpl_get_row_name(tran, i)); - /* set row bounds */ - type = mpl_get_row_bnds(tran, i, &lb, &ub); - switch (type) - { case MPL_FR: type = GLP_FR; break; - case MPL_LO: type = GLP_LO; break; - case MPL_UP: type = GLP_UP; break; - case MPL_DB: type = GLP_DB; break; - case MPL_FX: type = GLP_FX; break; - default: xassert(type != type); - } - if (type == GLP_DB && fabs(lb - ub) < 1e-9 * (1.0 + fabs(lb))) - { type = GLP_FX; - if (fabs(lb) <= fabs(ub)) ub = lb; else lb = ub; - } - glp_set_row_bnds(prob, i, type, lb, ub); - /* warn about non-zero constant term */ - if (mpl_get_row_c0(tran, i) != 0.0) - xprintf("glp_mpl_build_prob: row %s; constant term %.12g ig" - "nored\n", - mpl_get_row_name(tran, i), mpl_get_row_c0(tran, i)); - } - /* build columns (variables) */ - n = mpl_get_num_cols(tran); - if (n > 0) - glp_add_cols(prob, n); - for (j = 1; j <= n; j++) - { /* set column name */ - glp_set_col_name(prob, j, mpl_get_col_name(tran, j)); - /* set column kind */ - kind = mpl_get_col_kind(tran, j); - switch (kind) - { case MPL_NUM: - break; - case MPL_INT: - case MPL_BIN: - glp_set_col_kind(prob, j, GLP_IV); - break; - default: - xassert(kind != kind); - } - /* set column bounds */ - type = mpl_get_col_bnds(tran, j, &lb, &ub); - switch (type) - { case MPL_FR: type = GLP_FR; break; - case MPL_LO: type = GLP_LO; break; - case MPL_UP: type = GLP_UP; break; - case MPL_DB: type = GLP_DB; break; - case MPL_FX: type = GLP_FX; break; - default: xassert(type != type); - } - if (kind == MPL_BIN) - { if (type == GLP_FR || type == GLP_UP || lb < 0.0) lb = 0.0; - if (type == GLP_FR || type == GLP_LO || ub > 1.0) ub = 1.0; - type = GLP_DB; - } - if (type == GLP_DB && fabs(lb - ub) < 1e-9 * (1.0 + fabs(lb))) - { type = GLP_FX; - if (fabs(lb) <= fabs(ub)) ub = lb; else lb = ub; - } - glp_set_col_bnds(prob, j, type, lb, ub); - } - /* load the constraint matrix */ - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - for (i = 1; i <= m; i++) - { len = mpl_get_mat_row(tran, i, ind, val); - glp_set_mat_row(prob, i, len, ind, val); - } - /* build objective function (the first objective is used) */ - for (i = 1; i <= m; i++) - { kind = mpl_get_row_kind(tran, i); - if (kind == MPL_MIN || kind == MPL_MAX) - { /* set objective name */ - glp_set_obj_name(prob, mpl_get_row_name(tran, i)); - /* set optimization direction */ - glp_set_obj_dir(prob, kind == MPL_MIN ? GLP_MIN : GLP_MAX); - /* set constant term */ - glp_set_obj_coef(prob, 0, mpl_get_row_c0(tran, i)); - /* set objective coefficients */ - len = mpl_get_mat_row(tran, i, ind, val); - for (t = 1; t <= len; t++) - glp_set_obj_coef(prob, ind[t], val[t]); - break; - } - } - /* free working arrays */ - xfree(ind); - xfree(val); - return; -} - -int glp_mpl_postsolve(glp_tran *tran, glp_prob *prob, int sol) -{ /* postsolve the model */ - int i, j, m, n, stat, ret; - double prim, dual; - if (!(tran->phase == 3 && !tran->flag_p)) - xerror("glp_mpl_postsolve: invalid call sequence\n"); - if (!(sol == GLP_SOL || sol == GLP_IPT || sol == GLP_MIP)) - xerror("glp_mpl_postsolve: sol = %d; invalid parameter\n", - sol); - m = mpl_get_num_rows(tran); - n = mpl_get_num_cols(tran); - if (!(m == glp_get_num_rows(prob) && - n == glp_get_num_cols(prob))) - xerror("glp_mpl_postsolve: wrong problem object\n"); - if (!mpl_has_solve_stmt(tran)) - { ret = 0; - goto done; - } - for (i = 1; i <= m; i++) - { if (sol == GLP_SOL) - { stat = glp_get_row_stat(prob, i); - prim = glp_get_row_prim(prob, i); - dual = glp_get_row_dual(prob, i); - } - else if (sol == GLP_IPT) - { stat = 0; - prim = glp_ipt_row_prim(prob, i); - dual = glp_ipt_row_dual(prob, i); - } - else if (sol == GLP_MIP) - { stat = 0; - prim = glp_mip_row_val(prob, i); - dual = 0.0; - } - else - xassert(sol != sol); - if (fabs(prim) < 1e-9) prim = 0.0; - if (fabs(dual) < 1e-9) dual = 0.0; - mpl_put_row_soln(tran, i, stat, prim, dual); - } - for (j = 1; j <= n; j++) - { if (sol == GLP_SOL) - { stat = glp_get_col_stat(prob, j); - prim = glp_get_col_prim(prob, j); - dual = glp_get_col_dual(prob, j); - } - else if (sol == GLP_IPT) - { stat = 0; - prim = glp_ipt_col_prim(prob, j); - dual = glp_ipt_col_dual(prob, j); - } - else if (sol == GLP_MIP) - { stat = 0; - prim = glp_mip_col_val(prob, j); - dual = 0.0; - } - else - xassert(sol != sol); - if (fabs(prim) < 1e-9) prim = 0.0; - if (fabs(dual) < 1e-9) dual = 0.0; - mpl_put_col_soln(tran, j, stat, prim, dual); - } - ret = mpl_postsolve(tran); - if (ret == 3) - ret = 0; - else if (ret == 4) - ret = 1; -done: return ret; -} - -void glp_mpl_free_wksp(glp_tran *tran) -{ /* free the MathProg translator workspace */ - mpl_terminate(tran); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/mps.c b/code/3rd_glpk/api/mps.c deleted file mode 100644 index 3bdc6db1..00000000 --- a/code/3rd_glpk/api/mps.c +++ /dev/null @@ -1,1452 +0,0 @@ -/* mps.c (MPS format routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "misc.h" -#include "prob.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_init_mpscp - initialize MPS format control parameters -* -* SYNOPSIS -* -* void glp_init_mpscp(glp_mpscp *parm); -* -* DESCRIPTION -* -* The routine glp_init_mpscp initializes control parameters, which are -* used by the MPS input/output routines glp_read_mps and glp_write_mps, -* with default values. -* -* Default values of the control parameters are stored in the glp_mpscp -* structure, which the parameter parm points to. */ - -void glp_init_mpscp(glp_mpscp *parm) -{ parm->blank = '\0'; - parm->obj_name = NULL; - parm->tol_mps = 1e-12; - return; -} - -static void check_parm(const char *func, const glp_mpscp *parm) -{ /* check control parameters */ - if (!(0x00 <= parm->blank && parm->blank <= 0xFF) || - !(parm->blank == '\0' || isprint(parm->blank))) - xerror("%s: blank = 0x%02X; invalid parameter\n", - func, parm->blank); - if (!(parm->obj_name == NULL || strlen(parm->obj_name) <= 255)) - xerror("%s: obj_name = \"%.12s...\"; parameter too long\n", - func, parm->obj_name); - if (!(0.0 <= parm->tol_mps && parm->tol_mps < 1.0)) - xerror("%s: tol_mps = %g; invalid parameter\n", - func, parm->tol_mps); - return; -} - -/*********************************************************************** -* NAME -* -* glp_read_mps - read problem data in MPS format -* -* SYNOPSIS -* -* int glp_read_mps(glp_prob *P, int fmt, const glp_mpscp *parm, -* const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_mps reads problem data in MPS format from a -* text file. -* -* The parameter fmt specifies the version of MPS format: -* -* GLP_MPS_DECK - fixed (ancient) MPS format; -* GLP_MPS_FILE - free (modern) MPS format. -* -* The parameter parm is a pointer to the structure glp_mpscp, which -* specifies control parameters used by the routine. If parm is NULL, -* the routine uses default settings. -* -* The character string fname specifies a name of the text file to be -* read. -* -* Note that before reading data the current content of the problem -* object is completely erased with the routine glp_erase_prob. -* -* RETURNS -* -* If the operation was successful, the routine glp_read_mps returns -* zero. Otherwise, it prints an error message and returns non-zero. */ - -struct csa -{ /* common storage area */ - glp_prob *P; - /* pointer to problem object */ - int deck; - /* MPS format (0 - free, 1 - fixed) */ - const glp_mpscp *parm; - /* pointer to control parameters */ - const char *fname; - /* name of input MPS file */ - glp_file *fp; - /* stream assigned to input MPS file */ - jmp_buf jump; - /* label for go to in case of error */ - int recno; - /* current record (card) number */ - int recpos; - /* current record (card) position */ - int c; - /* current character */ - int fldno; - /* current field number */ - char field[255+1]; - /* current field content */ - int w80; - /* warning 'record must not be longer than 80 chars' issued */ - int wef; - /* warning 'extra fields detected beyond field 6' issued */ - int obj_row; - /* objective row number */ - void *work1, *work2, *work3; - /* working arrays */ -}; - -static void error(struct csa *csa, const char *fmt, ...) -{ /* print error message and terminate processing */ - va_list arg; - xprintf("%s:%d: ", csa->fname, csa->recno); - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - longjmp(csa->jump, 1); - /* no return */ -} - -static void warning(struct csa *csa, const char *fmt, ...) -{ /* print warning message and continue processing */ - va_list arg; - xprintf("%s:%d: warning: ", csa->fname, csa->recno); - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - return; -} - -static void read_char(struct csa *csa) -{ /* read next character */ - int c; - if (csa->c == '\n') - csa->recno++, csa->recpos = 0; - csa->recpos++; -read: c = glp_getc(csa->fp); - if (c < 0) - { if (glp_ioerr(csa->fp)) - error(csa, "read error - %s\n", get_err_msg()); - else if (csa->c == '\n') - error(csa, "unexpected end of file\n"); - else - { warning(csa, "missing final end of line\n"); - c = '\n'; - } - } - else if (c == '\n') - ; - else if (csa->c == '\r') - { c = '\r'; - goto badc; - } - else if (csa->deck && c == '\r') - { csa->c = '\r'; - goto read; - } - else if (c == ' ') - ; - else if (isspace(c)) - { if (csa->deck) -badc: error(csa, "in fixed MPS format white-space character 0x%02" - "X is not allowed\n", c); - c = ' '; - } - else if (iscntrl(c)) - error(csa, "invalid control character 0x%02X\n", c); - if (csa->deck && csa->recpos == 81 && c != '\n' && csa->w80 < 1) - { warning(csa, "in fixed MPS format record must not be longer th" - "an 80 characters\n"); - csa->w80++; - } - csa->c = c; - return; -} - -static int indicator(struct csa *csa, int name) -{ /* skip comment records and read possible indicator record */ - int ret; - /* reset current field number */ - csa->fldno = 0; -loop: /* read the very first character of the next record */ - xassert(csa->c == '\n'); - read_char(csa); - if (csa->c == ' ' || csa->c == '\n') - { /* data record */ - ret = 0; - } - else if (csa->c == '*') - { /* comment record */ - while (csa->c != '\n') - read_char(csa); - goto loop; - } - else - { /* indicator record */ - int len = 0; - while (csa->c != ' ' && csa->c != '\n' && len < 12) - { csa->field[len++] = (char)csa->c; - read_char(csa); - } - csa->field[len] = '\0'; - if (!(strcmp(csa->field, "NAME") == 0 || - strcmp(csa->field, "ROWS") == 0 || - strcmp(csa->field, "COLUMNS") == 0 || - strcmp(csa->field, "RHS") == 0 || - strcmp(csa->field, "RANGES") == 0 || - strcmp(csa->field, "BOUNDS") == 0 || - strcmp(csa->field, "ENDATA") == 0)) - error(csa, "invalid indicator record\n"); - if (!name) - { while (csa->c != '\n') - read_char(csa); - } - ret = 1; - } - return ret; -} - -static void read_field(struct csa *csa) -{ /* read next field of the current data record */ - csa->fldno++; - if (csa->deck) - { /* fixed MPS format */ - int beg, end, pos; - /* determine predefined field positions */ - if (csa->fldno == 1) - beg = 2, end = 3; - else if (csa->fldno == 2) - beg = 5, end = 12; - else if (csa->fldno == 3) - beg = 15, end = 22; - else if (csa->fldno == 4) - beg = 25, end = 36; - else if (csa->fldno == 5) - beg = 40, end = 47; - else if (csa->fldno == 6) - beg = 50, end = 61; - else - xassert(csa != csa); - /* skip blanks preceding the current field */ - if (csa->c != '\n') - { pos = csa->recpos; - while (csa->recpos < beg) - { if (csa->c == ' ') - ; - else if (csa->c == '\n') - break; - else - error(csa, "in fixed MPS format positions %d-%d must " - "be blank\n", pos, beg-1); - read_char(csa); - } - } - /* skip possible comment beginning in the field 3 or 5 */ - if ((csa->fldno == 3 || csa->fldno == 5) && csa->c == '$') - { while (csa->c != '\n') - read_char(csa); - } - /* read the current field */ - for (pos = beg; pos <= end; pos++) - { if (csa->c == '\n') break; - csa->field[pos-beg] = (char)csa->c; - read_char(csa); - } - csa->field[pos-beg] = '\0'; - strtrim(csa->field); - /* skip blanks following the last field */ - if (csa->fldno == 6 && csa->c != '\n') - { while (csa->recpos <= 72) - { if (csa->c == ' ') - ; - else if (csa->c == '\n') - break; - else - error(csa, "in fixed MPS format positions 62-72 must " - "be blank\n"); - read_char(csa); - } - while (csa->c != '\n') - read_char(csa); - } - } - else - { /* free MPS format */ - int len; - /* skip blanks preceding the current field */ - while (csa->c == ' ') - read_char(csa); - /* skip possible comment */ - if (csa->c == '$') - { while (csa->c != '\n') - read_char(csa); - } - /* read the current field */ - len = 0; - while (!(csa->c == ' ' || csa->c == '\n')) - { if (len == 255) - error(csa, "length of field %d exceeds 255 characters\n", - csa->fldno++); - csa->field[len++] = (char)csa->c; - read_char(csa); - } - csa->field[len] = '\0'; - /* skip anything following the last field (any extra fields - are considered to be comments) */ - if (csa->fldno == 6) - { while (csa->c == ' ') - read_char(csa); - if (csa->c != '$' && csa->c != '\n' && csa->wef < 1) - { warning(csa, "some extra field(s) detected beyond field " - "6; field(s) ignored\n"); - csa->wef++; - } - while (csa->c != '\n') - read_char(csa); - } - } - return; -} - -static void patch_name(struct csa *csa, char *name) -{ /* process embedded blanks in symbolic name */ - int blank = csa->parm->blank; - if (blank == '\0') - { /* remove emedded blanks */ - strspx(name); - } - else - { /* replace embedded blanks by specified character */ - for (; *name != '\0'; name++) - if (*name == ' ') *name = (char)blank; - } - return; -} - -static double read_number(struct csa *csa) -{ /* read next field and convert it to floating-point number */ - double x; - char *s; - /* read next field */ - read_field(csa); - xassert(csa->fldno == 4 || csa->fldno == 6); - if (csa->field[0] == '\0') - error(csa, "missing numeric value in field %d\n", csa->fldno); - /* skip initial spaces of the field */ - for (s = csa->field; *s == ' '; s++); - /* perform conversion */ - if (str2num(s, &x) != 0) - error(csa, "cannot convert '%s' to floating-point number\n", - s); - return x; -} - -static void skip_field(struct csa *csa) -{ /* read and skip next field (assumed to be blank) */ - read_field(csa); - if (csa->field[0] != '\0') - error(csa, "field %d must be blank\n", csa->fldno); - return; -} - -static void read_name(struct csa *csa) -{ /* read NAME indicator record */ - if (!(indicator(csa, 1) && strcmp(csa->field, "NAME") == 0)) - error(csa, "missing NAME indicator record\n"); - /* this indicator record looks like a data record; simulate that - fields 1 and 2 were read */ - csa->fldno = 2; - /* field 3: model name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - warning(csa, "missing model name in field 3\n"); - else - glp_set_prob_name(csa->P, csa->field); - /* skip anything following field 3 */ - while (csa->c != '\n') - read_char(csa); - return; -} - -static void read_rows(struct csa *csa) -{ /* read ROWS section */ - int i, type; -loop: if (indicator(csa, 0)) goto done; - /* field 1: row type */ - read_field(csa), strspx(csa->field); - if (strcmp(csa->field, "N") == 0) - type = GLP_FR; - else if (strcmp(csa->field, "G") == 0) - type = GLP_LO; - else if (strcmp(csa->field, "L") == 0) - type = GLP_UP; - else if (strcmp(csa->field, "E") == 0) - type = GLP_FX; - else if (csa->field[0] == '\0') - error(csa, "missing row type in field 1\n"); - else - error(csa, "invalid row type in field 1\n"); - /* field 2: row name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - error(csa, "missing row name in field 2\n"); - if (glp_find_row(csa->P, csa->field) != 0) - error(csa, "row '%s' multiply specified\n", csa->field); - i = glp_add_rows(csa->P, 1); - glp_set_row_name(csa->P, i, csa->field); - glp_set_row_bnds(csa->P, i, type, 0.0, 0.0); - /* fields 3, 4, 5, and 6 must be blank */ - skip_field(csa); - skip_field(csa); - skip_field(csa); - skip_field(csa); - goto loop; -done: return; -} - -static void read_columns(struct csa *csa) -{ /* read COLUMNS section */ - int i, j, f, len, kind = GLP_CV, *ind; - double aij, *val; - char name[255+1], *flag; - /* allocate working arrays */ - csa->work1 = ind = xcalloc(1+csa->P->m, sizeof(int)); - csa->work2 = val = xcalloc(1+csa->P->m, sizeof(double)); - csa->work3 = flag = xcalloc(1+csa->P->m, sizeof(char)); - memset(&flag[1], 0, csa->P->m); - /* no current column exists */ - j = 0, len = 0; -loop: if (indicator(csa, 0)) goto done; - /* field 1 must be blank */ - if (csa->deck) - { read_field(csa); - if (csa->field[0] != '\0') - error(csa, "field 1 must be blank\n"); - } - else - csa->fldno++; - /* field 2: column or kind name */ - read_field(csa), patch_name(csa, csa->field); - strcpy(name, csa->field); - /* field 3: row name or keyword 'MARKER' */ - read_field(csa), patch_name(csa, csa->field); - if (strcmp(csa->field, "'MARKER'") == 0) - { /* process kind data record */ - /* field 4 must be blank */ - if (csa->deck) - { read_field(csa); - if (csa->field[0] != '\0') - error(csa, "field 4 must be blank\n"); - } - else - csa->fldno++; - /* field 5: keyword 'INTORG' or 'INTEND' */ - read_field(csa), patch_name(csa, csa->field); - if (strcmp(csa->field, "'INTORG'") == 0) - kind = GLP_IV; - else if (strcmp(csa->field, "'INTEND'") == 0) - kind = GLP_CV; - else if (csa->field[0] == '\0') - error(csa, "missing keyword in field 5\n"); - else - error(csa, "invalid keyword in field 5\n"); - /* field 6 must be blank */ - skip_field(csa); - goto loop; - } - /* process column name specified in field 2 */ - if (name[0] == '\0') - { /* the same column as in previous data record */ - if (j == 0) - error(csa, "missing column name in field 2\n"); - } - else if (j != 0 && strcmp(name, csa->P->col[j]->name) == 0) - { /* the same column as in previous data record */ - xassert(j != 0); - } - else - { /* store the current column */ - if (j != 0) - { glp_set_mat_col(csa->P, j, len, ind, val); - while (len > 0) flag[ind[len--]] = 0; - } - /* create new column */ - if (glp_find_col(csa->P, name) != 0) - error(csa, "column '%s' multiply specified\n", name); - j = glp_add_cols(csa->P, 1); - glp_set_col_name(csa->P, j, name); - glp_set_col_kind(csa->P, j, kind); - if (kind == GLP_CV) - glp_set_col_bnds(csa->P, j, GLP_LO, 0.0, 0.0); - else if (kind == GLP_IV) - glp_set_col_bnds(csa->P, j, GLP_DB, 0.0, 1.0); - else - xassert(kind != kind); - } - /* process fields 3-4 and 5-6 */ - for (f = 3; f <= 5; f += 2) - { /* field 3 or 5: row name */ - if (f == 3) - { if (csa->field[0] == '\0') - error(csa, "missing row name in field 3\n"); - } - else - { read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - { /* if field 5 is blank, field 6 also must be blank */ - skip_field(csa); - continue; - } - } - i = glp_find_row(csa->P, csa->field); - if (i == 0) - error(csa, "row '%s' not found\n", csa->field); - if (flag[i]) - error(csa, "duplicate coefficient in row '%s'\n", - csa->field); - /* field 4 or 6: coefficient value */ - aij = read_number(csa); - if (fabs(aij) < csa->parm->tol_mps) aij = 0.0; - len++, ind[len] = i, val[len] = aij, flag[i] = 1; - } - goto loop; -done: /* store the last column */ - if (j != 0) - glp_set_mat_col(csa->P, j, len, ind, val); - /* free working arrays */ - xfree(ind); - xfree(val); - xfree(flag); - csa->work1 = csa->work2 = csa->work3 = NULL; - return; -} - -static void read_rhs(struct csa *csa) -{ /* read RHS section */ - int i, f, v, type; - double rhs; - char name[255+1], *flag; - /* allocate working array */ - csa->work3 = flag = xcalloc(1+csa->P->m, sizeof(char)); - memset(&flag[1], 0, csa->P->m); - /* no current RHS vector exists */ - v = 0; -loop: if (indicator(csa, 0)) goto done; - /* field 1 must be blank */ - if (csa->deck) - { read_field(csa); - if (csa->field[0] != '\0') - error(csa, "field 1 must be blank\n"); - } - else - csa->fldno++; - /* field 2: RHS vector name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - { /* the same RHS vector as in previous data record */ - if (v == 0) - { warning(csa, "missing RHS vector name in field 2\n"); - goto blnk; - } - } - else if (v != 0 && strcmp(csa->field, name) == 0) - { /* the same RHS vector as in previous data record */ - xassert(v != 0); - } - else -blnk: { /* new RHS vector */ - if (v != 0) - error(csa, "multiple RHS vectors not supported\n"); - v++; - strcpy(name, csa->field); - } - /* process fields 3-4 and 5-6 */ - for (f = 3; f <= 5; f += 2) - { /* field 3 or 5: row name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - { if (f == 3) - error(csa, "missing row name in field 3\n"); - else - { /* if field 5 is blank, field 6 also must be blank */ - skip_field(csa); - continue; - } - } - i = glp_find_row(csa->P, csa->field); - if (i == 0) - error(csa, "row '%s' not found\n", csa->field); - if (flag[i]) - error(csa, "duplicate right-hand side for row '%s'\n", - csa->field); - /* field 4 or 6: right-hand side value */ - rhs = read_number(csa); - if (fabs(rhs) < csa->parm->tol_mps) rhs = 0.0; - type = csa->P->row[i]->type; - if (type == GLP_FR) - { if (i == csa->obj_row) - glp_set_obj_coef(csa->P, 0, rhs); - else if (rhs != 0.0) - warning(csa, "non-zero right-hand side for free row '%s'" - " ignored\n", csa->P->row[i]->name); - } - else - glp_set_row_bnds(csa->P, i, type, rhs, rhs); - flag[i] = 1; - } - goto loop; -done: /* free working array */ - xfree(flag); - csa->work3 = NULL; - return; -} - -static void read_ranges(struct csa *csa) -{ /* read RANGES section */ - int i, f, v, type; - double rhs, rng; - char name[255+1], *flag; - /* allocate working array */ - csa->work3 = flag = xcalloc(1+csa->P->m, sizeof(char)); - memset(&flag[1], 0, csa->P->m); - /* no current RANGES vector exists */ - v = 0; -loop: if (indicator(csa, 0)) goto done; - /* field 1 must be blank */ - if (csa->deck) - { read_field(csa); - if (csa->field[0] != '\0') - error(csa, "field 1 must be blank\n"); - } - else - csa->fldno++; - /* field 2: RANGES vector name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - { /* the same RANGES vector as in previous data record */ - if (v == 0) - { warning(csa, "missing RANGES vector name in field 2\n"); - goto blnk; - } - } - else if (v != 0 && strcmp(csa->field, name) == 0) - { /* the same RANGES vector as in previous data record */ - xassert(v != 0); - } - else -blnk: { /* new RANGES vector */ - if (v != 0) - error(csa, "multiple RANGES vectors not supported\n"); - v++; - strcpy(name, csa->field); - } - /* process fields 3-4 and 5-6 */ - for (f = 3; f <= 5; f += 2) - { /* field 3 or 5: row name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - { if (f == 3) - error(csa, "missing row name in field 3\n"); - else - { /* if field 5 is blank, field 6 also must be blank */ - skip_field(csa); - continue; - } - } - i = glp_find_row(csa->P, csa->field); - if (i == 0) - error(csa, "row '%s' not found\n", csa->field); - if (flag[i]) - error(csa, "duplicate range for row '%s'\n", csa->field); - /* field 4 or 6: range value */ - rng = read_number(csa); - if (fabs(rng) < csa->parm->tol_mps) rng = 0.0; - type = csa->P->row[i]->type; - if (type == GLP_FR) - warning(csa, "range for free row '%s' ignored\n", - csa->P->row[i]->name); - else if (type == GLP_LO) - { rhs = csa->P->row[i]->lb; -#if 0 /* 26/V-2017 by cmatraki */ - glp_set_row_bnds(csa->P, i, rhs == 0.0 ? GLP_FX : GLP_DB, -#else - glp_set_row_bnds(csa->P, i, rng == 0.0 ? GLP_FX : GLP_DB, -#endif - rhs, rhs + fabs(rng)); - } - else if (type == GLP_UP) - { rhs = csa->P->row[i]->ub; -#if 0 /* 26/V-2017 by cmatraki */ - glp_set_row_bnds(csa->P, i, rhs == 0.0 ? GLP_FX : GLP_DB, -#else - glp_set_row_bnds(csa->P, i, rng == 0.0 ? GLP_FX : GLP_DB, -#endif - rhs - fabs(rng), rhs); - } - else if (type == GLP_FX) - { rhs = csa->P->row[i]->lb; - if (rng > 0.0) - glp_set_row_bnds(csa->P, i, GLP_DB, rhs, rhs + rng); - else if (rng < 0.0) - glp_set_row_bnds(csa->P, i, GLP_DB, rhs + rng, rhs); - } - else - xassert(type != type); - flag[i] = 1; - } - goto loop; -done: /* free working array */ - xfree(flag); - csa->work3 = NULL; - return; -} - -static void read_bounds(struct csa *csa) -{ /* read BOUNDS section */ - GLPCOL *col; - int j, v, mask, data; - double bnd, lb, ub; - char type[2+1], name[255+1], *flag; - /* allocate working array */ - csa->work3 = flag = xcalloc(1+csa->P->n, sizeof(char)); - memset(&flag[1], 0, csa->P->n); - /* no current BOUNDS vector exists */ - v = 0; -loop: if (indicator(csa, 0)) goto done; - /* field 1: bound type */ - read_field(csa); - if (strcmp(csa->field, "LO") == 0) - mask = 0x01, data = 1; - else if (strcmp(csa->field, "UP") == 0) - mask = 0x10, data = 1; - else if (strcmp(csa->field, "FX") == 0) - mask = 0x11, data = 1; - else if (strcmp(csa->field, "FR") == 0) - mask = 0x11, data = 0; - else if (strcmp(csa->field, "MI") == 0) - mask = 0x01, data = 0; - else if (strcmp(csa->field, "PL") == 0) - mask = 0x10, data = 0; - else if (strcmp(csa->field, "LI") == 0) - mask = 0x01, data = 1; - else if (strcmp(csa->field, "UI") == 0) - mask = 0x10, data = 1; - else if (strcmp(csa->field, "BV") == 0) - mask = 0x11, data = 0; - else if (csa->field[0] == '\0') - error(csa, "missing bound type in field 1\n"); - else - error(csa, "invalid bound type in field 1\n"); - strcpy(type, csa->field); - /* field 2: BOUNDS vector name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - { /* the same BOUNDS vector as in previous data record */ - if (v == 0) - { warning(csa, "missing BOUNDS vector name in field 2\n"); - goto blnk; - } - } - else if (v != 0 && strcmp(csa->field, name) == 0) - { /* the same BOUNDS vector as in previous data record */ - xassert(v != 0); - } - else -blnk: { /* new BOUNDS vector */ - if (v != 0) - error(csa, "multiple BOUNDS vectors not supported\n"); - v++; - strcpy(name, csa->field); - } - /* field 3: column name */ - read_field(csa), patch_name(csa, csa->field); - if (csa->field[0] == '\0') - error(csa, "missing column name in field 3\n"); - j = glp_find_col(csa->P, csa->field); - if (j == 0) - error(csa, "column '%s' not found\n", csa->field); - if ((flag[j] & mask) == 0x01) - error(csa, "duplicate lower bound for column '%s'\n", - csa->field); - if ((flag[j] & mask) == 0x10) - error(csa, "duplicate upper bound for column '%s'\n", - csa->field); - xassert((flag[j] & mask) == 0x00); - /* field 4: bound value */ - if (data) - { bnd = read_number(csa); - if (fabs(bnd) < csa->parm->tol_mps) bnd = 0.0; - } - else - read_field(csa), bnd = 0.0; - /* get current column bounds */ - col = csa->P->col[j]; - if (col->type == GLP_FR) - lb = -DBL_MAX, ub = +DBL_MAX; - else if (col->type == GLP_LO) - lb = col->lb, ub = +DBL_MAX; - else if (col->type == GLP_UP) - lb = -DBL_MAX, ub = col->ub; - else if (col->type == GLP_DB) - lb = col->lb, ub = col->ub; - else if (col->type == GLP_FX) - lb = ub = col->lb; - else - xassert(col != col); - /* change column bounds */ - if (strcmp(type, "LO") == 0) - lb = bnd; - else if (strcmp(type, "UP") == 0) - ub = bnd; - else if (strcmp(type, "FX") == 0) - lb = ub = bnd; - else if (strcmp(type, "FR") == 0) - lb = -DBL_MAX, ub = +DBL_MAX; - else if (strcmp(type, "MI") == 0) - lb = -DBL_MAX; - else if (strcmp(type, "PL") == 0) - ub = +DBL_MAX; - else if (strcmp(type, "LI") == 0) - { glp_set_col_kind(csa->P, j, GLP_IV); - lb = ceil(bnd); -#if 1 /* 16/VII-2013 */ - /* if column upper bound has not been explicitly specified, - take it as +inf */ - if (!(flag[j] & 0x10)) - ub = +DBL_MAX; -#endif - } - else if (strcmp(type, "UI") == 0) - { glp_set_col_kind(csa->P, j, GLP_IV); - ub = floor(bnd); - } - else if (strcmp(type, "BV") == 0) - { glp_set_col_kind(csa->P, j, GLP_IV); - lb = 0.0, ub = 1.0; - } - else - xassert(type != type); - /* set new column bounds */ - if (lb == -DBL_MAX && ub == +DBL_MAX) - glp_set_col_bnds(csa->P, j, GLP_FR, lb, ub); - else if (ub == +DBL_MAX) - glp_set_col_bnds(csa->P, j, GLP_LO, lb, ub); - else if (lb == -DBL_MAX) - glp_set_col_bnds(csa->P, j, GLP_UP, lb, ub); - else if (lb != ub) - glp_set_col_bnds(csa->P, j, GLP_DB, lb, ub); - else - glp_set_col_bnds(csa->P, j, GLP_FX, lb, ub); - flag[j] |= (char)mask; - /* fields 5 and 6 must be blank */ - skip_field(csa); - skip_field(csa); - goto loop; -done: /* free working array */ - xfree(flag); - csa->work3 = NULL; - return; -} - -int glp_read_mps(glp_prob *P, int fmt, const glp_mpscp *parm, - const char *fname) -{ /* read problem data in MPS format */ - glp_mpscp _parm; - struct csa _csa, *csa = &_csa; - int ret; - xprintf("Reading problem data from '%s'...\n", fname); - if (!(fmt == GLP_MPS_DECK || fmt == GLP_MPS_FILE)) - xerror("glp_read_mps: fmt = %d; invalid parameter\n", fmt); - if (parm == NULL) - glp_init_mpscp(&_parm), parm = &_parm; - /* check control parameters */ - check_parm("glp_read_mps", parm); - /* initialize common storage area */ - csa->P = P; - csa->deck = (fmt == GLP_MPS_DECK); - csa->parm = parm; - csa->fname = fname; - csa->fp = NULL; - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->recno = csa->recpos = 0; - csa->c = '\n'; - csa->fldno = 0; - csa->field[0] = '\0'; - csa->w80 = csa->wef = 0; - csa->obj_row = 0; - csa->work1 = csa->work2 = csa->work3 = NULL; - /* erase problem object */ - glp_erase_prob(P); - glp_create_index(P); - /* open input MPS file */ - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* read NAME indicator record */ - read_name(csa); - if (P->name != NULL) - xprintf("Problem: %s\n", P->name); - /* read ROWS section */ - if (!(indicator(csa, 0) && strcmp(csa->field, "ROWS") == 0)) - error(csa, "missing ROWS indicator record\n"); - read_rows(csa); - /* determine objective row */ - if (parm->obj_name == NULL || parm->obj_name[0] == '\0') - { /* use the first row of N type */ - int i; - for (i = 1; i <= P->m; i++) - { if (P->row[i]->type == GLP_FR) - { csa->obj_row = i; - break; - } - } - if (csa->obj_row == 0) - warning(csa, "unable to determine objective row\n"); - } - else - { /* use a row with specified name */ - int i; - for (i = 1; i <= P->m; i++) - { xassert(P->row[i]->name != NULL); - if (strcmp(parm->obj_name, P->row[i]->name) == 0) - { csa->obj_row = i; - break; - } - } - if (csa->obj_row == 0) - error(csa, "objective row '%s' not found\n", - parm->obj_name); - } - if (csa->obj_row != 0) - { glp_set_obj_name(P, P->row[csa->obj_row]->name); - xprintf("Objective: %s\n", P->obj); - } - /* read COLUMNS section */ - if (strcmp(csa->field, "COLUMNS") != 0) - error(csa, "missing COLUMNS indicator record\n"); - read_columns(csa); - /* set objective coefficients */ - if (csa->obj_row != 0) - { GLPAIJ *aij; - for (aij = P->row[csa->obj_row]->ptr; aij != NULL; aij = - aij->r_next) glp_set_obj_coef(P, aij->col->j, aij->val); - } - /* read optional RHS section */ - if (strcmp(csa->field, "RHS") == 0) - read_rhs(csa); - /* read optional RANGES section */ - if (strcmp(csa->field, "RANGES") == 0) - read_ranges(csa); - /* read optional BOUNDS section */ - if (strcmp(csa->field, "BOUNDS") == 0) - read_bounds(csa); - /* read ENDATA indicator record */ - if (strcmp(csa->field, "ENDATA") != 0) - error(csa, "invalid use of %s indicator record\n", - csa->field); - /* print some statistics */ - xprintf("%d row%s, %d column%s, %d non-zero%s\n", - P->m, P->m == 1 ? "" : "s", P->n, P->n == 1 ? "" : "s", - P->nnz, P->nnz == 1 ? "" : "s"); - if (glp_get_num_int(P) > 0) - { int ni = glp_get_num_int(P); - int nb = glp_get_num_bin(P); - if (ni == 1) - { if (nb == 0) - xprintf("One variable is integer\n"); - else - xprintf("One variable is binary\n"); - } - else - { xprintf("%d integer variables, ", ni); - if (nb == 0) - xprintf("none"); - else if (nb == 1) - xprintf("one"); - else if (nb == ni) - xprintf("all"); - else - xprintf("%d", nb); - xprintf(" of which %s binary\n", nb == 1 ? "is" : "are"); - } - } - xprintf("%d records were read\n", csa->recno); -#if 1 /* 31/III-2016 */ - /* free (unbounded) row(s) in MPS file are intended to specify - * objective function(s), so all such rows can be removed */ -#if 1 /* 08/VIII-2013 */ - /* remove free rows */ - { int i, nrs, *num; - num = talloc(1+P->m, int); - nrs = 0; - for (i = 1; i <= P->m; i++) - { if (P->row[i]->type == GLP_FR) - num[++nrs] = i; - } - if (nrs > 0) - { glp_del_rows(P, nrs, num); - if (nrs == 1) - xprintf("One free row was removed\n"); - else - xprintf("%d free rows were removed\n", nrs); - } - tfree(num); - } -#endif -#else - /* if objective function row is free, remove it */ - if (csa->obj_row != 0 && P->row[csa->obj_row]->type == GLP_FR) - { int num[1+1]; - num[1] = csa->obj_row; - glp_del_rows(P, 1, num); - xprintf("Free objective row was removed\n"); - } -#endif - /* problem data has been successfully read */ - glp_delete_index(P); - glp_sort_matrix(P); - ret = 0; -done: if (csa->fp != NULL) glp_close(csa->fp); - if (csa->work1 != NULL) xfree(csa->work1); - if (csa->work2 != NULL) xfree(csa->work2); - if (csa->work3 != NULL) xfree(csa->work3); - if (ret != 0) glp_erase_prob(P); - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_write_mps - write problem data in MPS format -* -* SYNOPSIS -* -* int glp_write_mps(glp_prob *P, int fmt, const glp_mpscp *parm, -* const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_mps writes problem data in MPS format to a -* text file. -* -* The parameter fmt specifies the version of MPS format: -* -* GLP_MPS_DECK - fixed (ancient) MPS format; -* GLP_MPS_FILE - free (modern) MPS format. -* -* The parameter parm is a pointer to the structure glp_mpscp, which -* specifies control parameters used by the routine. If parm is NULL, -* the routine uses default settings. -* -* The character string fname specifies a name of the text file to be -* written. -* -* RETURNS -* -* If the operation was successful, the routine glp_read_mps returns -* zero. Otherwise, it prints an error message and returns non-zero. */ - -#define csa csa1 - -struct csa -{ /* common storage area */ - glp_prob *P; - /* pointer to problem object */ - int deck; - /* MPS format (0 - free, 1 - fixed) */ - const glp_mpscp *parm; - /* pointer to control parameters */ - char field[255+1]; - /* field buffer */ -}; - -static char *mps_name(struct csa *csa) -{ /* make problem name */ - char *f; - if (csa->P->name == NULL) - csa->field[0] = '\0'; - else if (csa->deck) - { strncpy(csa->field, csa->P->name, 8); - csa->field[8] = '\0'; - } - else - strcpy(csa->field, csa->P->name); - for (f = csa->field; *f != '\0'; f++) - if (*f == ' ') *f = '_'; - return csa->field; -} - -static char *row_name(struct csa *csa, int i) -{ /* make i-th row name */ - char *f; - xassert(0 <= i && i <= csa->P->m); - if (i == 0 || csa->P->row[i]->name == NULL || - csa->deck && strlen(csa->P->row[i]->name) > 8) - sprintf(csa->field, "R%07d", i); - else - { strcpy(csa->field, csa->P->row[i]->name); - for (f = csa->field; *f != '\0'; f++) - if (*f == ' ') *f = '_'; - } - return csa->field; -} - -static char *col_name(struct csa *csa, int j) -{ /* make j-th column name */ - char *f; - xassert(1 <= j && j <= csa->P->n); - if (csa->P->col[j]->name == NULL || - csa->deck && strlen(csa->P->col[j]->name) > 8) - sprintf(csa->field, "C%07d", j); - else - { strcpy(csa->field, csa->P->col[j]->name); - for (f = csa->field; *f != '\0'; f++) - if (*f == ' ') *f = '_'; - } - return csa->field; -} - -static char *mps_numb(struct csa *csa, double val) -{ /* format floating-point number */ - int dig; - char *exp; - for (dig = 12; dig >= 6; dig--) - { if (val != 0.0 && fabs(val) < 0.002) - sprintf(csa->field, "%.*E", dig-1, val); - else - sprintf(csa->field, "%.*G", dig, val); - exp = strchr(csa->field, 'E'); - if (exp != NULL) - sprintf(exp+1, "%d", atoi(exp+1)); - if (strlen(csa->field) <= 12) break; - } - xassert(strlen(csa->field) <= 12); - return csa->field; -} - -int glp_write_mps(glp_prob *P, int fmt, const glp_mpscp *parm, - const char *fname) -{ /* write problem data in MPS format */ - glp_mpscp _parm; - struct csa _csa, *csa = &_csa; - glp_file *fp; - int out_obj, one_col = 0, empty = 0; - int i, j, recno, marker, count, gap, ret; - xprintf("Writing problem data to '%s'...\n", fname); - if (!(fmt == GLP_MPS_DECK || fmt == GLP_MPS_FILE)) - xerror("glp_write_mps: fmt = %d; invalid parameter\n", fmt); - if (parm == NULL) - glp_init_mpscp(&_parm), parm = &_parm; - /* check control parameters */ - check_parm("glp_write_mps", parm); - /* initialize common storage area */ - csa->P = P; - csa->deck = (fmt == GLP_MPS_DECK); - csa->parm = parm; - /* create output MPS file */ - fp = glp_open(fname, "w"), recno = 0; - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* write comment records */ - xfprintf(fp, "* %-*s%s\n", P->name == NULL ? 1 : 12, "Problem:", - P->name == NULL ? "" : P->name), recno++; - xfprintf(fp, "* %-12s%s\n", "Class:", glp_get_num_int(P) == 0 ? - "LP" : "MIP"), recno++; - xfprintf(fp, "* %-12s%d\n", "Rows:", P->m), recno++; - if (glp_get_num_int(P) == 0) - xfprintf(fp, "* %-12s%d\n", "Columns:", P->n), recno++; - else - xfprintf(fp, "* %-12s%d (%d integer, %d binary)\n", - "Columns:", P->n, glp_get_num_int(P), glp_get_num_bin(P)), - recno++; - xfprintf(fp, "* %-12s%d\n", "Non-zeros:", P->nnz), recno++; - xfprintf(fp, "* %-12s%s\n", "Format:", csa->deck ? "Fixed MPS" : - "Free MPS"), recno++; - xfprintf(fp, "*\n", recno++); - /* write NAME indicator record */ - xfprintf(fp, "NAME%*s%s\n", - P->name == NULL ? 0 : csa->deck ? 10 : 1, "", mps_name(csa)), - recno++; -#if 1 - /* determine whether to write the objective row */ - out_obj = 1; - for (i = 1; i <= P->m; i++) - { if (P->row[i]->type == GLP_FR) - { out_obj = 0; - break; - } - } -#endif - /* write ROWS section */ - xfprintf(fp, "ROWS\n"), recno++; - for (i = (out_obj ? 0 : 1); i <= P->m; i++) - { int type; - type = (i == 0 ? GLP_FR : P->row[i]->type); - if (type == GLP_FR) - type = 'N'; - else if (type == GLP_LO) - type = 'G'; - else if (type == GLP_UP) - type = 'L'; - else if (type == GLP_DB || type == GLP_FX) - type = 'E'; - else - xassert(type != type); - xfprintf(fp, " %c%*s%s\n", type, csa->deck ? 2 : 1, "", - row_name(csa, i)), recno++; - } - /* write COLUMNS section */ - xfprintf(fp, "COLUMNS\n"), recno++; - marker = 0; - for (j = 1; j <= P->n; j++) - { GLPAIJ cj, *aij; - int kind; - kind = P->col[j]->kind; - if (kind == GLP_CV) - { if (marker % 2 == 1) - { /* close current integer block */ - marker++; - xfprintf(fp, "%*sM%07d%*s'MARKER'%*s'INTEND'\n", - csa->deck ? 4 : 1, "", marker, csa->deck ? 2 : 1, "", - csa->deck ? 17 : 1, ""), recno++; - } - } - else if (kind == GLP_IV) - { if (marker % 2 == 0) - { /* open new integer block */ - marker++; - xfprintf(fp, "%*sM%07d%*s'MARKER'%*s'INTORG'\n", - csa->deck ? 4 : 1, "", marker, csa->deck ? 2 : 1, "", - csa->deck ? 17 : 1, ""), recno++; - } - } - else - xassert(kind != kind); - if (out_obj && P->col[j]->coef != 0.0) - { /* make fake objective coefficient */ - aij = &cj; - aij->row = NULL; - aij->val = P->col[j]->coef; - aij->c_next = P->col[j]->ptr; - } - else - aij = P->col[j]->ptr; -#if 1 /* FIXME */ - if (aij == NULL) - { /* empty column */ - empty++; - xfprintf(fp, "%*s%-*s", csa->deck ? 4 : 1, "", - csa->deck ? 8 : 1, col_name(csa, j)); - /* we need a row */ - xassert(P->m > 0); - xfprintf(fp, "%*s%-*s", - csa->deck ? 2 : 1, "", csa->deck ? 8 : 1, - row_name(csa, 1)); - xfprintf(fp, "%*s0%*s$ empty column\n", - csa->deck ? 13 : 1, "", csa->deck ? 3 : 1, ""), recno++; - } -#endif - count = 0; - for (aij = aij; aij != NULL; aij = aij->c_next) - { if (one_col || count % 2 == 0) - xfprintf(fp, "%*s%-*s", csa->deck ? 4 : 1, "", - csa->deck ? 8 : 1, col_name(csa, j)); - gap = (one_col || count % 2 == 0 ? 2 : 3); - xfprintf(fp, "%*s%-*s", - csa->deck ? gap : 1, "", csa->deck ? 8 : 1, - row_name(csa, aij->row == NULL ? 0 : aij->row->i)); - xfprintf(fp, "%*s%*s", - csa->deck ? 2 : 1, "", csa->deck ? 12 : 1, - mps_numb(csa, aij->val)), count++; - if (one_col || count % 2 == 0) - xfprintf(fp, "\n"), recno++; - } - if (!(one_col || count % 2 == 0)) - xfprintf(fp, "\n"), recno++; - } - if (marker % 2 == 1) - { /* close last integer block */ - marker++; - xfprintf(fp, "%*sM%07d%*s'MARKER'%*s'INTEND'\n", - csa->deck ? 4 : 1, "", marker, csa->deck ? 2 : 1, "", - csa->deck ? 17 : 1, ""), recno++; - } -#if 1 - if (empty > 0) - xprintf("Warning: problem has %d empty column(s)\n", empty); -#endif - /* write RHS section */ - xfprintf(fp, "RHS\n"), recno++; - count = 0; - for (i = (out_obj ? 0 : 1); i <= P->m; i++) - { int type; - double rhs; - if (i == 0) - rhs = P->c0; - else - { type = P->row[i]->type; - if (type == GLP_FR) - rhs = 0.0; - else if (type == GLP_LO) - rhs = P->row[i]->lb; - else if (type == GLP_UP) - rhs = P->row[i]->ub; - else if (type == GLP_DB || type == GLP_FX) - rhs = P->row[i]->lb; - else - xassert(type != type); - } - if (rhs != 0.0) - { if (one_col || count % 2 == 0) - xfprintf(fp, "%*s%-*s", csa->deck ? 4 : 1, "", - csa->deck ? 8 : 1, "RHS1"); - gap = (one_col || count % 2 == 0 ? 2 : 3); - xfprintf(fp, "%*s%-*s", - csa->deck ? gap : 1, "", csa->deck ? 8 : 1, - row_name(csa, i)); - xfprintf(fp, "%*s%*s", - csa->deck ? 2 : 1, "", csa->deck ? 12 : 1, - mps_numb(csa, rhs)), count++; - if (one_col || count % 2 == 0) - xfprintf(fp, "\n"), recno++; - } - } - if (!(one_col || count % 2 == 0)) - xfprintf(fp, "\n"), recno++; - /* write RANGES section */ - for (i = P->m; i >= 1; i--) - if (P->row[i]->type == GLP_DB) break; - if (i == 0) goto bnds; - xfprintf(fp, "RANGES\n"), recno++; - count = 0; - for (i = 1; i <= P->m; i++) - { if (P->row[i]->type == GLP_DB) - { if (one_col || count % 2 == 0) - xfprintf(fp, "%*s%-*s", csa->deck ? 4 : 1, "", - csa->deck ? 8 : 1, "RNG1"); - gap = (one_col || count % 2 == 0 ? 2 : 3); - xfprintf(fp, "%*s%-*s", - csa->deck ? gap : 1, "", csa->deck ? 8 : 1, - row_name(csa, i)); - xfprintf(fp, "%*s%*s", - csa->deck ? 2 : 1, "", csa->deck ? 12 : 1, - mps_numb(csa, P->row[i]->ub - P->row[i]->lb)), count++; - if (one_col || count % 2 == 0) - xfprintf(fp, "\n"), recno++; - } - } - if (!(one_col || count % 2 == 0)) - xfprintf(fp, "\n"), recno++; -bnds: /* write BOUNDS section */ - for (j = P->n; j >= 1; j--) - if (!(P->col[j]->kind == GLP_CV && - P->col[j]->type == GLP_LO && P->col[j]->lb == 0.0)) - break; - if (j == 0) goto endt; - xfprintf(fp, "BOUNDS\n"), recno++; - for (j = 1; j <= P->n; j++) - { int type, data[2]; - double bnd[2]; - char *spec[2]; - spec[0] = spec[1] = NULL; - type = P->col[j]->type; - if (type == GLP_FR) - spec[0] = "FR", data[0] = 0; - else if (type == GLP_LO) - { if (P->col[j]->lb != 0.0) - spec[0] = "LO", data[0] = 1, bnd[0] = P->col[j]->lb; - if (P->col[j]->kind == GLP_IV) - spec[1] = "PL", data[1] = 0; - } - else if (type == GLP_UP) - { spec[0] = "MI", data[0] = 0; - spec[1] = "UP", data[1] = 1, bnd[1] = P->col[j]->ub; - } - else if (type == GLP_DB) - { if (P->col[j]->lb != 0.0) - spec[0] = "LO", data[0] = 1, bnd[0] = P->col[j]->lb; - spec[1] = "UP", data[1] = 1, bnd[1] = P->col[j]->ub; - } - else if (type == GLP_FX) - spec[0] = "FX", data[0] = 1, bnd[0] = P->col[j]->lb; - else - xassert(type != type); - for (i = 0; i <= 1; i++) - { if (spec[i] != NULL) - { xfprintf(fp, " %s %-*s%*s%-*s", spec[i], - csa->deck ? 8 : 1, "BND1", csa->deck ? 2 : 1, "", - csa->deck ? 8 : 1, col_name(csa, j)); - if (data[i]) - xfprintf(fp, "%*s%*s", csa->deck ? 2 : 1, "", - csa->deck ? 12 : 1, mps_numb(csa, bnd[i])); - xfprintf(fp, "\n"), recno++; - } - } - } -endt: /* write ENDATA indicator record */ - xfprintf(fp, "ENDATA\n"), recno++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* problem data has been successfully written */ - xprintf("%d records were written\n", recno); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/netgen.c b/code/3rd_glpk/api/netgen.c deleted file mode 100644 index 519fd609..00000000 --- a/code/3rd_glpk/api/netgen.c +++ /dev/null @@ -1,1020 +0,0 @@ -/* netgen.c (Klingman's network problem generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* This code is the result of translation of the Fortran program NETGEN -* developed by Dr. Darwin Klingman, which is publically available from -* NETLIB at . -* -* The translation was made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_netgen - Klingman's network problem generator -* -* SYNOPSIS -* -* int glp_netgen(glp_graph *G, int v_rhs, int a_cap, int a_cost, -* const int parm[1+15]); -* -* DESCRIPTION -* -* The routine glp_netgen is a network problem generator developed by -* Dr. Darwin Klingman. It can create capacitated and uncapacitated -* minimum cost flow (or transshipment), transportation, and assignment -* problems. -* -* The parameter G specifies the graph object, to which the generated -* problem data have to be stored. Note that on entry the graph object -* is erased with the routine glp_erase_graph. -* -* The parameter v_rhs specifies an offset of the field of type double -* in the vertex data block, to which the routine stores the supply or -* demand value. If v_rhs < 0, the value is not stored. -* -* The parameter a_cap specifies an offset of the field of type double -* in the arc data block, to which the routine stores the arc capacity. -* If a_cap < 0, the capacity is not stored. -* -* The parameter a_cost specifies an offset of the field of type double -* in the arc data block, to which the routine stores the per-unit cost -* if the arc flow. If a_cost < 0, the cost is not stored. -* -* The array parm contains description of the network to be generated: -* -* parm[0] not used -* parm[1] (iseed) 8-digit positive random number seed -* parm[2] (nprob) 8-digit problem id number -* parm[3] (nodes) total number of nodes -* parm[4] (nsorc) total number of source nodes (including -* transshipment nodes) -* parm[5] (nsink) total number of sink nodes (including -* transshipment nodes) -* parm[6] (iarcs) number of arcs -* parm[7] (mincst) minimum cost for arcs -* parm[8] (maxcst) maximum cost for arcs -* parm[9] (itsup) total supply -* parm[10] (ntsorc) number of transshipment source nodes -* parm[11] (ntsink) number of transshipment sink nodes -* parm[12] (iphic) percentage of skeleton arcs to be given -* the maximum cost -* parm[13] (ipcap) percentage of arcs to be capacitated -* parm[14] (mincap) minimum upper bound for capacitated arcs -* parm[15] (maxcap) maximum upper bound for capacitated arcs -* -* The routine generates a transportation problem if: -* -* nsorc + nsink = nodes, ntsorc = 0, and ntsink = 0. -* -* The routine generates an assignment problem if the requirements for -* a transportation problem are met and: -* -* nsorc = nsink and itsup = nsorc. -* -* RETURNS -* -* If the instance was successfully generated, the routine glp_netgen -* returns zero; otherwise, if specified parameters are inconsistent, -* the routine returns a non-zero error code. -* -* REFERENCES -* -* D.Klingman, A.Napier, and J.Stutz. NETGEN: A program for generating -* large scale capacitated assignment, transportation, and minimum cost -* flow networks. Management Science 20 (1974), 814-20. */ - -struct csa -{ /* common storage area */ - glp_graph *G; - int v_rhs, a_cap, a_cost; - int nodes, iarcs, mincst, maxcst, itsup, nsorc, nsink, nonsor, - nfsink, narcs, nsort, nftsor, ipcap, mincap, maxcap, ktl, - nodlft, *ipred, *ihead, *itail, *iflag, *isup, *lsinks, mult, - modul, i15, i16, jran; -}; - -#define G (csa->G) -#define v_rhs (csa->v_rhs) -#define a_cap (csa->a_cap) -#define a_cost (csa->a_cost) -#define nodes (csa->nodes) -#define iarcs (csa->iarcs) -#define mincst (csa->mincst) -#define maxcst (csa->maxcst) -#define itsup (csa->itsup) -#define nsorc (csa->nsorc) -#define nsink (csa->nsink) -#define nonsor (csa->nonsor) -#define nfsink (csa->nfsink) -#define narcs (csa->narcs) -#define nsort (csa->nsort) -#define nftsor (csa->nftsor) -#define ipcap (csa->ipcap) -#define mincap (csa->mincap) -#define maxcap (csa->maxcap) -#define ktl (csa->ktl) -#define nodlft (csa->nodlft) -#if 0 -/* spent a day to find out this bug */ -#define ist (csa->ist) -#else -#define ist (ipred[0]) -#endif -#define ipred (csa->ipred) -#define ihead (csa->ihead) -#define itail (csa->itail) -#define iflag (csa->iflag) -#define isup (csa->isup) -#define lsinks (csa->lsinks) -#define mult (csa->mult) -#define modul (csa->modul) -#define i15 (csa->i15) -#define i16 (csa->i16) -#define jran (csa->jran) - -static void cresup(struct csa *csa); -static void chain(struct csa *csa, int lpick, int lsorc); -static void chnarc(struct csa *csa, int lsorc); -static void sort(struct csa *csa); -static void pickj(struct csa *csa, int it); -static void assign(struct csa *csa); -static void setran(struct csa *csa, int iseed); -static int iran(struct csa *csa, int ilow, int ihigh); - -int glp_netgen(glp_graph *G_, int _v_rhs, int _a_cap, int _a_cost, - const int parm[1+15]) -{ struct csa _csa, *csa = &_csa; - int iseed, nprob, ntsorc, ntsink, iphic, i, nskel, nltr, ltsink, - ntrans, npsink, nftr, npsorc, ntravl, ntrrem, lsorc, lpick, - nsksr, nsrchn, j, item, l, ks, k, ksp, li, n, ii, it, ih, icap, - jcap, icost, jcost, ret; - G = G_; - v_rhs = _v_rhs; - a_cap = _a_cap; - a_cost = _a_cost; - if (G != NULL) - { if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_netgen: v_rhs = %d; invalid offset\n", v_rhs); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_netgen: a_cap = %d; invalid offset\n", a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_netgen: a_cost = %d; invalid offset\n", a_cost); - } - /* Input the user's random number seed and fix it if - non-positive. */ - iseed = parm[1]; - nprob = parm[2]; - if (iseed <= 0) iseed = 13502460; - setran(csa, iseed); - /* Input the user's problem characteristics. */ - nodes = parm[3]; - nsorc = parm[4]; - nsink = parm[5]; - iarcs = parm[6]; - mincst = parm[7]; - maxcst = parm[8]; - itsup = parm[9]; - ntsorc = parm[10]; - ntsink = parm[11]; - iphic = parm[12]; - ipcap = parm[13]; - mincap = parm[14]; - maxcap = parm[15]; - /* Check the size of the problem. */ - if (!(10 <= nodes && nodes <= 100000)) - { ret = 1; - goto done; - } - /* Check user supplied parameters for consistency. */ - if (!(nsorc >= 0 && nsink >= 0 && nsorc + nsink <= nodes)) - { ret = 2; - goto done; - } - if (iarcs < 0) - { ret = 3; - goto done; - } - if (mincst > maxcst) - { ret = 4; - goto done; - } - if (itsup < 0) - { ret = 5; - goto done; - } - if (!(0 <= ntsorc && ntsorc <= nsorc)) - { ret = 6; - goto done; - } - if (!(0 <= ntsink && ntsink <= nsink)) - { ret = 7; - goto done; - } - if (!(0 <= iphic && iphic <= 100)) - { ret = 8; - goto done; - } - if (!(0 <= ipcap && ipcap <= 100)) - { ret = 9; - goto done; - } - if (mincap > maxcap) - { ret = 10; - goto done; - } - /* Initailize the graph object. */ - if (G != NULL) - { glp_erase_graph(G, G->v_size, G->a_size); - glp_add_vertices(G, nodes); - if (v_rhs >= 0) - { double zero = 0.0; - for (i = 1; i <= nodes; i++) - { glp_vertex *v = G->v[i]; - memcpy((char *)v->data + v_rhs, &zero, sizeof(double)); - } - } - } - /* Allocate working arrays. */ - ipred = xcalloc(1+nodes, sizeof(int)); - ihead = xcalloc(1+nodes, sizeof(int)); - itail = xcalloc(1+nodes, sizeof(int)); - iflag = xcalloc(1+nodes, sizeof(int)); - isup = xcalloc(1+nodes, sizeof(int)); - lsinks = xcalloc(1+nodes, sizeof(int)); - /* Print the problem documentation records. */ - if (G == NULL) - { xprintf("BEGIN\n"); - xprintf("NETGEN PROBLEM%8d%10s%10d NODES AND%10d ARCS\n", - nprob, "", nodes, iarcs); - xprintf("USER:%11d%11d%11d%11d%11d%11d\nDATA:%11d%11d%11d%11d%" - "11d%11d\n", iseed, nsorc, nsink, mincst, - maxcst, itsup, ntsorc, ntsink, iphic, ipcap, - mincap, maxcap); - } - else - glp_set_graph_name(G, "NETGEN"); - /* Set various constants used in the program. */ - narcs = 0; - nskel = 0; - nltr = nodes - nsink; - ltsink = nltr + ntsink; - ntrans = nltr - nsorc; - nfsink = nltr + 1; - nonsor = nodes - nsorc + ntsorc; - npsink = nsink - ntsink; - nodlft = nodes - nsink + ntsink; - nftr = nsorc + 1; - nftsor = nsorc - ntsorc + 1; - npsorc = nsorc - ntsorc; - /* Randomly distribute the supply among the source nodes. */ - if (npsorc + npsink == nodes && npsorc == npsink && - itsup == nsorc) - { assign(csa); - nskel = nsorc; - goto L390; - } - cresup(csa); - /* Print the supply records. */ - if (G == NULL) - { xprintf("SUPPLY\n"); - for (i = 1; i <= nsorc; i++) - xprintf("%6s%6d%18s%10d\n", "", i, "", isup[i]); - xprintf("ARCS\n"); - } - else - { if (v_rhs >= 0) - { for (i = 1; i <= nsorc; i++) - { double temp = (double)isup[i]; - glp_vertex *v = G->v[i]; - memcpy((char *)v->data + v_rhs, &temp, sizeof(double)); - } - } - } - /* Make the sources point to themselves in ipred array. */ - for (i = 1; i <= nsorc; i++) - ipred[i] = i; - if (ntrans == 0) goto L170; - /* Chain the transshipment nodes together in the ipred array. */ - ist = nftr; - ipred[nltr] = 0; - for (i = nftr; i < nltr; i++) - ipred[i] = i+1; - /* Form even length chains for 60 percent of the transshipments.*/ - ntravl = 6 * ntrans / 10; - ntrrem = ntrans - ntravl; -L140: lsorc = 1; - while (ntravl != 0) - { lpick = iran(csa, 1, ntravl + ntrrem); - ntravl--; - chain(csa, lpick, lsorc); - if (lsorc == nsorc) goto L140; - lsorc++; - } - /* Add the remaining transshipments to the chains. */ - while (ntrrem != 0) - { - lpick = iran(csa, 1, ntrrem); - ntrrem--; - lsorc = iran(csa, 1, nsorc); - chain(csa, lpick, lsorc); - } -L170: /* Set all demands equal to zero. */ - for (i = nfsink; i <= nodes; i++) - ipred[i] = 0; - /* The following loop takes one chain at a time (through the use - of logic contained in the loop and calls to other routines) and - creates the remaining network arcs. */ - for (lsorc = 1; lsorc <= nsorc; lsorc++) - { chnarc(csa, lsorc); - for (i = nfsink; i <= nodes; i++) - iflag[i] = 0; - /* Choose the number of sinks to be hooked up to the current - chain. */ - if (ntrans != 0) - nsksr = (nsort * 2 * nsink) / ntrans; - else - nsksr = nsink / nsorc + 1; - if (nsksr < 2) nsksr = 2; - if (nsksr > nsink) nsksr = nsink; - nsrchn = nsort; - /* Randomly pick nsksr sinks and put their names in lsinks. */ - ktl = nsink; - for (j = 1; j <= nsksr; j++) - { item = iran(csa, 1, ktl); - ktl--; - for (l = nfsink; l <= nodes; l++) - { if (iflag[l] != 1) - { item--; - if (item == 0) goto L230; - } - } - break; -L230: lsinks[j] = l; - iflag[l] = 1; - } - /* If last source chain, add all sinks with zero demand to - lsinks list. */ - if (lsorc == nsorc) - { for (j = nfsink; j <= nodes; j++) - { if (ipred[j] == 0 && iflag[j] != 1) - { nsksr++; - lsinks[nsksr] = j; - iflag[j] = 1; - } - } - } - /* Create demands for group of sinks in lsinks. */ - ks = isup[lsorc] / nsksr; - k = ipred[lsorc]; - for (i = 1; i <= nsksr; i++) - { nsort++; - ksp = iran(csa, 1, ks); - j = iran(csa, 1, nsksr); - itail[nsort] = k; - li = lsinks[i]; - ihead[nsort] = li; - ipred[li] += ksp; - li = lsinks[j]; - ipred[li] += ks - ksp; - n = iran(csa, 1, nsrchn); - k = lsorc; - for (ii = 1; ii <= n; ii++) - k = ipred[k]; - } - li = lsinks[1]; - ipred[li] += isup[lsorc] - ks * nsksr; - nskel += nsort; - /* Sort the arcs in the chain from source lsorc using itail as - sort key. */ - sort(csa); - /* Print this part of skeleton and create the arcs for these - nodes. */ - i = 1; - itail[nsort+1] = 0; -L300: for (j = nftsor; j <= nodes; j++) - iflag[j] = 0; - ktl = nonsor - 1; - it = itail[i]; - iflag[it] = 1; -L320: ih = ihead[i]; - iflag[ih] = 1; - narcs++; - ktl--; - /* Determine if this skeleton arc should be capacitated. */ - icap = itsup; - jcap = iran(csa, 1, 100); - if (jcap <= ipcap) - { icap = isup[lsorc]; - if (mincap > icap) icap = mincap; - } - /* Determine if this skeleton arc should have the maximum - cost. */ - icost = maxcst; - jcost = iran(csa, 1, 100); - if (jcost > iphic) - icost = iran(csa, mincst, maxcst); - if (G == NULL) - xprintf("%6s%6d%6d%2s%10d%10d\n", "", it, ih, "", icost, - icap); - else - { glp_arc *a = glp_add_arc(G, it, ih); - if (a_cap >= 0) - { double temp = (double)icap; - memcpy((char *)a->data + a_cap, &temp, sizeof(double)); - } - if (a_cost >= 0) - { double temp = (double)icost; - memcpy((char *)a->data + a_cost, &temp, sizeof(double)); - } - } - i++; - if (itail[i] == it) goto L320; - pickj(csa, it); - if (i <= nsort) goto L300; - } - /* Create arcs from the transshipment sinks. */ - if (ntsink != 0) - { for (i = nfsink; i <= ltsink; i++) - { for (j = nftsor; j <= nodes; j++) - iflag[j] = 0; - ktl = nonsor - 1; - iflag[i] = 1; - pickj(csa, i); - } - } -L390: /* Print the demand records and end record. */ - if (G == NULL) - { xprintf("DEMAND\n"); - for (i = nfsink; i <= nodes; i++) - xprintf("%6s%6d%18s%10d\n", "", i, "", ipred[i]); - xprintf("END\n"); - } - else - { if (v_rhs >= 0) - { for (i = nfsink; i <= nodes; i++) - { double temp = - (double)ipred[i]; - glp_vertex *v = G->v[i]; - memcpy((char *)v->data + v_rhs, &temp, sizeof(double)); - } - } - } - /* Free working arrays. */ - xfree(ipred); - xfree(ihead); - xfree(itail); - xfree(iflag); - xfree(isup); - xfree(lsinks); - /* The instance has been successfully generated. */ - ret = 0; -done: return ret; -} - -/*********************************************************************** -* The routine cresup randomly distributes the total supply among the -* source nodes. */ - -static void cresup(struct csa *csa) -{ int i, j, ks, ksp; - xassert(itsup > nsorc); - ks = itsup / nsorc; - for (i = 1; i <= nsorc; i++) - isup[i] = 0; - for (i = 1; i <= nsorc; i++) - { ksp = iran(csa, 1, ks); - j = iran(csa, 1, nsorc); - isup[i] += ksp; - isup[j] += ks - ksp; - } - j = iran(csa, 1, nsorc); - isup[j] += itsup - ks * nsorc; - return; -} - -/*********************************************************************** -* The routine chain adds node lpick to the end of the chain with source -* node lsorc. */ - -static void chain(struct csa *csa, int lpick, int lsorc) -{ int i, j, k, l, m; - k = 0; - m = ist; - for (i = 1; i <= lpick; i++) - { l = k; - k = m; - m = ipred[k]; - } - ipred[l] = m; - j = ipred[lsorc]; - ipred[k] = j; - ipred[lsorc] = k; - return; -} - -/*********************************************************************** -* The routine chnarc puts the arcs in the chain from source lsorc into -* the ihead and itail arrays for sorting. */ - -static void chnarc(struct csa *csa, int lsorc) -{ int ito, ifrom; - nsort = 0; - ito = ipred[lsorc]; -L10: if (ito == lsorc) return; - nsort++; - ifrom = ipred[ito]; - ihead[nsort] = ito; - itail[nsort] = ifrom; - ito = ifrom; - goto L10; -} - -/*********************************************************************** -* The routine sort sorts the nsort arcs in the ihead and itail arrays. -* ihead is used as the sort key (i.e. forward star sort order). */ - -static void sort(struct csa *csa) -{ int i, j, k, l, m, n, it; - n = nsort; - m = n; -L10: m /= 2; - if (m == 0) return; - k = n - m; - j = 1; -L20: i = j; -L30: l = i + m; - if (itail[i] <= itail[l]) goto L40; - it = itail[i]; - itail[i] = itail[l]; - itail[l] = it; - it = ihead[i]; - ihead[i] = ihead[l]; - ihead[l] = it; - i -= m; - if (i >= 1) goto L30; -L40: j++; - if (j <= k) goto L20; - goto L10; -} - -/*********************************************************************** -* The routine pickj creates a random number of arcs out of node 'it'. -* Various parameters are dynamically adjusted in an attempt to ensure -* that the generated network has the correct number of arcs. */ - -static void pickj(struct csa *csa, int it) -{ int j, k, l, nn, nupbnd, icap, jcap, icost; - if ((nodlft - 1) * 2 > iarcs - narcs - 1) - { nodlft--; - return; - } - if ((iarcs - narcs + nonsor - ktl - 1) / nodlft - nonsor + 1 >= 0) - k = nonsor; - else - { nupbnd = (iarcs - narcs - nodlft) / nodlft * 2; -L40: k = iran(csa, 1, nupbnd); - if (nodlft == 1) k = iarcs - narcs; - if ((nodlft - 1) * (nonsor - 1) < iarcs - narcs - k) goto L40; - } - nodlft--; - for (j = 1; j <= k; j++) - { nn = iran(csa, 1, ktl); - ktl--; - for (l = nftsor; l <= nodes; l++) - { if (iflag[l] != 1) - { nn--; - if (nn == 0) goto L70; - } - } - return; -L70: iflag[l] = 1; - icap = itsup; - jcap = iran(csa, 1, 100); - if (jcap <= ipcap) - icap = iran(csa, mincap, maxcap); - icost = iran(csa, mincst, maxcst); - if (G == NULL) - xprintf("%6s%6d%6d%2s%10d%10d\n", "", it, l, "", icost, - icap); - else - { glp_arc *a = glp_add_arc(G, it, l); - if (a_cap >= 0) - { double temp = (double)icap; - memcpy((char *)a->data + a_cap, &temp, sizeof(double)); - } - if (a_cost >= 0) - { double temp = (double)icost; - memcpy((char *)a->data + a_cost, &temp, sizeof(double)); - } - } - narcs++; - } - return; -} - -/*********************************************************************** -* The routine assign generate assignment problems. It defines the unit -* supplies, builds a skeleton, then calls pickj to create the arcs. */ - -static void assign(struct csa *csa) -{ int i, it, nn, l, ll, icost; - if (G == NULL) - xprintf("SUPPLY\n"); - for (i = 1; i <= nsorc; i++) - { isup[i] = 1; - iflag[i] = 0; - if (G == NULL) - xprintf("%6s%6d%18s%10d\n", "", i, "", isup[i]); - else - { if (v_rhs >= 0) - { double temp = (double)isup[i]; - glp_vertex *v = G->v[i]; - memcpy((char *)v->data + v_rhs, &temp, sizeof(double)); - } - } - } - if (G == NULL) - xprintf("ARCS\n"); - for (i = nfsink; i <= nodes; i++) - ipred[i] = 1; - for (it = 1; it <= nsorc; it++) - { for (i = nfsink; i <= nodes; i++) - iflag[i] = 0; - ktl = nsink - 1; - nn = iran(csa, 1, nsink - it + 1); - for (l = 1; l <= nsorc; l++) - { if (iflag[l] != 1) - { nn--; - if (nn == 0) break; - } - } - narcs++; - ll = nsorc + l; - icost = iran(csa, mincst, maxcst); - if (G == NULL) - xprintf("%6s%6d%6d%2s%10d%10d\n", "", it, ll, "", icost, - isup[1]); - else - { glp_arc *a = glp_add_arc(G, it, ll); - if (a_cap >= 0) - { double temp = (double)isup[1]; - memcpy((char *)a->data + a_cap, &temp, sizeof(double)); - } - if (a_cost >= 0) - { double temp = (double)icost; - memcpy((char *)a->data + a_cost, &temp, sizeof(double)); - } - } - iflag[l] = 1; - iflag[ll] = 1; - pickj(csa, it); - } - return; -} - -/*********************************************************************** -* Portable congruential (uniform) random number generator: -* -* next_value = ((7**5) * previous_value) modulo ((2**31)-1) -* -* This generator consists of three routines: -* -* (1) setran - initializes constants and seed -* (2) iran - generates an integer random number -* (3) rran - generates a real random number -* -* The generator requires a machine with at least 32 bits of precision. -* The seed (iseed) must be in the range [1,(2**31)-1]. */ - -static void setran(struct csa *csa, int iseed) -{ xassert(iseed >= 1); - mult = 16807; - modul = 2147483647; - i15 = 1 << 15; - i16 = 1 << 16; - jran = iseed; - return; -} - -/*********************************************************************** -* The routine iran generates an integer random number between ilow and -* ihigh. If ilow > ihigh then iran returns ihigh. */ - -static int iran(struct csa *csa, int ilow, int ihigh) -{ int ixhi, ixlo, ixalo, leftlo, ixahi, ifulhi, irtlo, iover, - irthi, j; - ixhi = jran / i16; - ixlo = jran - ixhi * i16; - ixalo = ixlo * mult; - leftlo = ixalo / i16; - ixahi = ixhi * mult; - ifulhi = ixahi + leftlo; - irtlo = ixalo - leftlo * i16; - iover = ifulhi / i15; - irthi = ifulhi - iover * i15; - jran = ((irtlo - modul) + irthi * i16) + iover; - if (jran < 0) jran += modul; - j = ihigh - ilow + 1; - if (j > 0) - return jran % j + ilow; - else - return ihigh; -} - -/*********************************************************************** -* NAME -* -* glp_netgen_prob - Klingman's standard network problem instance -* -* SYNOPSIS -* -* void glp_netgen_prob(int nprob, int parm[1+15]); -* -* DESCRIPTION -* -* The routine glp_netgen_prob provides the set of parameters for -* Klingman's network problem generator (see the routine glp_netgen), -* which describe a standard network problem instance. -* -* The parameter nprob (101 <= nprob <= 150) specifies the problem -* instance number. -* -* The array parm contains description of the network, provided by the -* routine. (For detailed description of these parameters see comments -* to the routine glp_netgen.) -* -* PROBLEM CHARACTERISTICS -* -* The table below shows characteristics of Klingman's standard network -* problem instances. -* -* Problem Nodes Arcs Optimum -* ------- ----- ----- ---------- -* 101 5000 25336 6191726 -* 102 5000 25387 72337144 -* 103 5000 25355 218947553 -* 104 5000 25344 -19100371 -* 105 5000 25332 31192578 -* 106 5000 12870 4314276 -* 107 5000 37832 7393769 -* 108 5000 50309 8405738 -* 109 5000 75299 9190300 -* 110 5000 12825 8975048 -* 111 5000 37828 4747532 -* 112 5000 50325 4012671 -* 113 5000 75318 2979725 -* 114 5000 26514 5821181 -* 115 5000 25962 6353310 -* 116 5000 25304 5915426 -* 117 5000 12816 4420560 -* 118 5000 37797 7045842 -* 119 5000 50301 7724179 -* 120 5000 75330 8455200 -* 121 5000 25000 66366360 -* 122 5000 25000 30997529 -* 123 5000 25000 23388777 -* 124 5000 25000 17803443 -* 125 5000 25000 14119622 -* 126 5000 12500 18802218 -* 127 5000 37500 27674647 -* 128 5000 50000 30906194 -* 129 5000 75000 40905209 -* 130 5000 12500 38939608 -* 131 5000 37500 16752978 -* 132 5000 50000 13302951 -* 133 5000 75000 9830268 -* 134 1000 25000 3804874 -* 135 2500 25000 11729616 -* 136 7500 25000 33318101 -* 137 10000 25000 46426030 -* 138 5000 25000 60710879 -* 139 5000 25000 32729682 -* 140 5000 25000 27183831 -* 141 5000 25000 19963286 -* 142 5000 25000 20243457 -* 143 5000 25000 18586777 -* 144 5000 25000 2504591 -* 145 5000 25000 215956138 -* 146 5000 25000 2253113811 -* 147 5000 25000 -427908373 -* 148 5000 25000 -92965318 -* 149 5000 25000 86051224 -* 150 5000 25000 619314919 */ - -static const int data[50][1+15] = -{ { 0, 13502460, 101, 5000, 2500, 2500, 25000, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 4281922, 102, 5000, 2500, 2500, 25000, - 1, 100, 2500000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 44820113, 103, 5000, 2500, 2500, 25000, - 1, 100, 6250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 13450451, 104, 5000, 2500, 2500, 25000, - -100, -1, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 14719436, 105, 5000, 2500, 2500, 25000, - 101, 200, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 17365786, 106, 5000, 2500, 2500, 12500, - 1, 100, 125000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 19540113, 107, 5000, 2500, 2500, 37500, - 1, 100, 375000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 19560313, 108, 5000, 2500, 2500, 50000, - 1, 100, 500000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 2403509, 109, 5000, 2500, 2500, 75000, - 1, 100, 750000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 92480414, 110, 5000, 2500, 2500, 12500, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 4230140, 111, 5000, 2500, 2500, 37500, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 10032490, 112, 5000, 2500, 2500, 50000, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 17307474, 113, 5000, 2500, 2500, 75000, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 4925114, 114, 5000, 500, 4500, 25000, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 19842704, 115, 5000, 1500, 3500, 25000, - 1, 100, 250000, 0, 0, 0, 100, 1, 1000 - }, - { 0, 88392060, 116, 5000, 2500, 2500, 25000, - 1, 100, 250000, 0, 0, 0, 0, 1, 1000 - }, - { 0, 12904407, 117, 5000, 2500, 2500, 12500, - 1, 100, 125000, 0, 0, 0, 0, 1, 1000 - }, - { 0, 11811811, 118, 5000, 2500, 2500, 37500, - 1, 100, 375000, 0, 0, 0, 0, 1, 1000 - }, - { 0, 90023593, 119, 5000, 2500, 2500, 50000, - 1, 100, 500000, 0, 0, 0, 0, 1, 1000 - }, - { 0, 93028922, 120, 5000, 2500, 2500, 75000, - 1, 100, 750000, 0, 0, 0, 0, 1, 1000 - }, - { 0, 72707401, 121, 5000, 50, 50, 25000, - 1, 100, 250000, 50, 50, 0, 100, 1, 1000 - }, - { 0, 93040771, 122, 5000, 250, 250, 25000, - 1, 100, 250000, 250, 250, 0, 100, 1, 1000 - }, - { 0, 70220611, 123, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 52774811, 124, 5000, 1000, 1000, 25000, - 1, 100, 250000, 1000, 1000, 0, 100, 1, 1000 - }, - { 0, 22492311, 125, 5000, 1500, 1500, 25000, - 1, 100, 250000, 1500, 1500, 0, 100, 1, 1000 - }, - { 0, 35269337, 126, 5000, 500, 500, 12500, - 1, 100, 125000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 30140502, 127, 5000, 500, 500, 37500, - 1, 100, 375000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 49205455, 128, 5000, 500, 500, 50000, - 1, 100, 500000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 42958341, 129, 5000, 500, 500, 75000, - 1, 100, 750000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 25440925, 130, 5000, 500, 500, 12500, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 75294924, 131, 5000, 500, 500, 37500, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 4463965, 132, 5000, 500, 500, 50000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 13390427, 133, 5000, 500, 500, 75000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 95250971, 134, 1000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 54830522, 135, 2500, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 520593, 136, 7500, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 52900925, 137, 10000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 22603395, 138, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 50 - }, - { 0, 55253099, 139, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 250 - }, - { 0, 75357001, 140, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 500 - }, - { 0, 10072459, 141, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 2500 - }, - { 0, 55728492, 142, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 100, 1, 5000 - }, - { 0, 593043, 143, 5000, 500, 500, 25000, - 1, 100, 250000, 500, 500, 0, 0, 1, 1000 - }, - { 0, 94236572, 144, 5000, 500, 500, 25000, - 1, 10, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 94882955, 145, 5000, 500, 500, 25000, - 1, 1000, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 48489922, 146, 5000, 500, 500, 25000, - 1, 10000, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 75578374, 147, 5000, 500, 500, 25000, - -100, -1, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 44821152, 148, 5000, 500, 500, 25000, - -50, 49, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 45224103, 149, 5000, 500, 500, 25000, - 101, 200, 250000, 500, 500, 0, 100, 1, 1000 - }, - { 0, 63491741, 150, 5000, 500, 500, 25000, - 1001, 1100, 250000, 500, 500, 0, 100, 1, 1000 - }, -}; - -void glp_netgen_prob(int nprob, int parm[1+15]) -{ int k; - if (!(101 <= nprob && nprob <= 150)) - xerror("glp_netgen_prob: nprob = %d; invalid problem instance " - "number\n", nprob); - for (k = 1; k <= 15; k++) - parm[k] = data[nprob-101][k]; - return; -} - -/**********************************************************************/ - -#if 0 -static int scan(char card[80+1], int pos, int len) -{ char buf[10+1]; - memcpy(buf, &card[pos-1], len); - buf[len] = '\0'; - return atoi(buf); -} - -int main(void) -{ int parm[1+15]; - char card[80+1]; - xassert(fgets(card, sizeof(card), stdin) == card); - parm[1] = scan(card, 1, 8); - parm[2] = scan(card, 9, 8); - xassert(fgets(card, sizeof(card), stdin) == card); - parm[3] = scan(card, 1, 5); - parm[4] = scan(card, 6, 5); - parm[5] = scan(card, 11, 5); - parm[6] = scan(card, 16, 5); - parm[7] = scan(card, 21, 5); - parm[8] = scan(card, 26, 5); - parm[9] = scan(card, 31, 10); - parm[10] = scan(card, 41, 5); - parm[11] = scan(card, 46, 5); - parm[12] = scan(card, 51, 5); - parm[13] = scan(card, 56, 5); - parm[14] = scan(card, 61, 10); - parm[15] = scan(card, 71, 10); - glp_netgen(NULL, 0, 0, 0, parm); - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/api/npp.c b/code/3rd_glpk/api/npp.c deleted file mode 100644 index a7ae07c1..00000000 --- a/code/3rd_glpk/api/npp.c +++ /dev/null @@ -1,143 +0,0 @@ -/* npp.c (LP/MIP preprocessing) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -glp_prep *glp_npp_alloc_wksp(void) -{ /* allocate the preprocessor workspace */ - glp_prep *prep; - prep = npp_create_wksp(); - return prep; -} - -void glp_npp_load_prob(glp_prep *prep, glp_prob *P, int sol, int names) -{ /* load original problem instance */ - if (prep->sol != 0) - xerror("glp_npp_load_prob: invalid call sequence (original ins" - "tance already loaded)\n"); - if (!(sol == GLP_SOL || sol == GLP_IPT || sol == GLP_MIP)) - xerror("glp_npp_load_prob: sol = %d; invalid parameter\n", - sol); - if (!(names == GLP_ON || names == GLP_OFF)) - xerror("glp_npp_load_prob: names = %d; invalid parameter\n", - names); - npp_load_prob(prep, P, names, sol, GLP_OFF); - return; -} - -int glp_npp_preprocess1(glp_prep *prep, int hard) -{ /* perform basic LP/MIP preprocessing */ - if (prep->sol == 0) - xerror("glp_npp_preprocess1: invalid call sequence (original i" - "nstance not loaded yet)\n"); - if (prep->pool == NULL) - xerror("glp_npp_preprocess1: invalid call sequence (preprocess" - "ing already finished)\n"); - if (!(hard == GLP_ON || hard == GLP_OFF)) - xerror("glp_npp_preprocess1: hard = %d; invalid parameter\n", - hard); - return npp_process_prob(prep, hard); -} - -void glp_npp_build_prob(glp_prep *prep, glp_prob *Q) -{ /* build resultant problem instance */ - if (prep->sol == 0) - xerror("glp_npp_build_prob: invalid call sequence (original in" - "stance not loaded yet)\n"); - if (prep->pool == NULL) - xerror("glp_npp_build_prob: invalid call sequence (resultant i" - "nstance already built)\n"); - npp_build_prob(prep, Q); - return; -} - -void glp_npp_postprocess(glp_prep *prep, glp_prob *Q) -{ /* postprocess solution to resultant problem */ - if (prep->pool != NULL) - xerror("glp_npp_postprocess: invalid call sequence (resultant " - "instance not built yet)\n"); - if (!(prep->m == Q->m && prep->n == Q->n && prep->nnz == Q->nnz)) - xerror("glp_npp_postprocess: resultant instance mismatch\n"); - switch (prep->sol) - { case GLP_SOL: - if (glp_get_status(Q) != GLP_OPT) - xerror("glp_npp_postprocess: unable to recover non-optim" - "al basic solution\n"); - break; - case GLP_IPT: - if (glp_ipt_status(Q) != GLP_OPT) - xerror("glp_npp_postprocess: unable to recover non-optim" - "al interior-point solution\n"); - break; - case GLP_MIP: - if (!(glp_mip_status(Q) == GLP_OPT || glp_mip_status(Q) == - GLP_FEAS)) - xerror("glp_npp_postprocess: unable to recover integer n" - "on-feasible solution\n"); - break; - default: - xassert(prep != prep); - } - npp_postprocess(prep, Q); - return; -} - -void glp_npp_obtain_sol(glp_prep *prep, glp_prob *P) -{ /* obtain solution to original problem */ - if (prep->pool != NULL) - xerror("glp_npp_obtain_sol: invalid call sequence (resultant i" - "nstance not built yet)\n"); - switch (prep->sol) - { case GLP_SOL: - if (prep->p_stat == 0 || prep->d_stat == 0) - xerror("glp_npp_obtain_sol: invalid call sequence (basic" - " solution not provided yet)\n"); - break; - case GLP_IPT: - if (prep->t_stat == 0) - xerror("glp_npp_obtain_sol: invalid call sequence (inter" - "ior-point solution not provided yet)\n"); - break; - case GLP_MIP: - if (prep->i_stat == 0) - xerror("glp_npp_obtain_sol: invalid call sequence (MIP s" - "olution not provided yet)\n"); - break; - default: - xassert(prep != prep); - } - if (!(prep->orig_dir == P->dir && prep->orig_m == P->m && - prep->orig_n == P->n && prep->orig_nnz == P->nnz)) - xerror("glp_npp_obtain_sol: original instance mismatch\n"); - npp_unload_sol(prep, P); - return; -} - -void glp_npp_free_wksp(glp_prep *prep) -{ /* free the preprocessor workspace */ - npp_delete_wksp(prep); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/pript.c b/code/3rd_glpk/api/pript.c deleted file mode 100644 index f123089d..00000000 --- a/code/3rd_glpk/api/pript.c +++ /dev/null @@ -1,186 +0,0 @@ -/* pript.c (write interior-point solution in printable format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -#define xfprintf glp_format - -int glp_print_ipt(glp_prob *P, const char *fname) -{ /* write interior-point solution in printable format */ - glp_file *fp; - GLPROW *row; - GLPCOL *col; - int i, j, t, ae_ind, re_ind, ret; - double ae_max, re_max; - xprintf("Writing interior-point solution to '%s'...\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "%-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name); - xfprintf(fp, "%-12s%d\n", "Rows:", P->m); - xfprintf(fp, "%-12s%d\n", "Columns:", P->n); - xfprintf(fp, "%-12s%d\n", "Non-zeros:", P->nnz); - t = glp_ipt_status(P); - xfprintf(fp, "%-12s%s\n", "Status:", - t == GLP_OPT ? "OPTIMAL" : - t == GLP_UNDEF ? "UNDEFINED" : - t == GLP_INFEAS ? "INFEASIBLE (INTERMEDIATE)" : - t == GLP_NOFEAS ? "INFEASIBLE (FINAL)" : "???"); - xfprintf(fp, "%-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->ipt_obj, - P->dir == GLP_MIN ? "MINimum" : - P->dir == GLP_MAX ? "MAXimum" : "???"); - xfprintf(fp, "\n"); - xfprintf(fp, " No. Row name Activity Lower bound " - " Upper bound Marginal\n"); - xfprintf(fp, "------ ------------ ------------- ------------- " - "------------- -------------\n"); - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - xfprintf(fp, "%6d ", i); - if (row->name == NULL || strlen(row->name) <= 12) - xfprintf(fp, "%-12s ", row->name == NULL ? "" : row->name); - else - xfprintf(fp, "%s\n%20s", row->name, ""); - xfprintf(fp, "%3s", ""); - xfprintf(fp, "%13.6g ", - fabs(row->pval) <= 1e-9 ? 0.0 : row->pval); - if (row->type == GLP_LO || row->type == GLP_DB || - row->type == GLP_FX) - xfprintf(fp, "%13.6g ", row->lb); - else - xfprintf(fp, "%13s ", ""); - if (row->type == GLP_UP || row->type == GLP_DB) - xfprintf(fp, "%13.6g ", row->ub); - else - xfprintf(fp, "%13s ", row->type == GLP_FX ? "=" : ""); - if (fabs(row->dval) <= 1e-9) - xfprintf(fp, "%13s", "< eps"); - else - xfprintf(fp, "%13.6g ", row->dval); - xfprintf(fp, "\n"); - } - xfprintf(fp, "\n"); - xfprintf(fp, " No. Column name Activity Lower bound " - " Upper bound Marginal\n"); - xfprintf(fp, "------ ------------ ------------- ------------- " - "------------- -------------\n"); - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - xfprintf(fp, "%6d ", j); - if (col->name == NULL || strlen(col->name) <= 12) - xfprintf(fp, "%-12s ", col->name == NULL ? "" : col->name); - else - xfprintf(fp, "%s\n%20s", col->name, ""); - xfprintf(fp, "%3s", ""); - xfprintf(fp, "%13.6g ", - fabs(col->pval) <= 1e-9 ? 0.0 : col->pval); - if (col->type == GLP_LO || col->type == GLP_DB || - col->type == GLP_FX) - xfprintf(fp, "%13.6g ", col->lb); - else - xfprintf(fp, "%13s ", ""); - if (col->type == GLP_UP || col->type == GLP_DB) - xfprintf(fp, "%13.6g ", col->ub); - else - xfprintf(fp, "%13s ", col->type == GLP_FX ? "=" : ""); - if (fabs(col->dval) <= 1e-9) - xfprintf(fp, "%13s", "< eps"); - else - xfprintf(fp, "%13.6g ", col->dval); - xfprintf(fp, "\n"); - } - xfprintf(fp, "\n"); - xfprintf(fp, "Karush-Kuhn-Tucker optimality conditions:\n"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_IPT, GLP_KKT_PE, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.PE: max.abs.err = %.2e on row %d\n", - ae_max, ae_ind); - xfprintf(fp, " max.rel.err = %.2e on row %d\n", - re_max, re_ind); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "PRIMAL SOLUTION IS WRONG"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_IPT, GLP_KKT_PB, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.PB: max.abs.err = %.2e on %s %d\n", - ae_max, ae_ind <= P->m ? "row" : "column", - ae_ind <= P->m ? ae_ind : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on %s %d\n", - re_max, re_ind <= P->m ? "row" : "column", - re_ind <= P->m ? re_ind : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "PRIMAL SOLUTION IS INFEASIBL" - "E"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_IPT, GLP_KKT_DE, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.DE: max.abs.err = %.2e on column %d\n", - ae_max, ae_ind == 0 ? 0 : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on column %d\n", - re_max, re_ind == 0 ? 0 : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "DUAL SOLUTION IS WRONG"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_IPT, GLP_KKT_DB, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.DB: max.abs.err = %.2e on %s %d\n", - ae_max, ae_ind <= P->m ? "row" : "column", - ae_ind <= P->m ? ae_ind : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on %s %d\n", - re_max, re_ind <= P->m ? "row" : "column", - re_ind <= P->m ? re_ind : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "DUAL SOLUTION IS INFEASIBLE") - ; - xfprintf(fp, "\n"); - xfprintf(fp, "End of output\n"); -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prmip.c b/code/3rd_glpk/api/prmip.c deleted file mode 100644 index 885ed82a..00000000 --- a/code/3rd_glpk/api/prmip.c +++ /dev/null @@ -1,155 +0,0 @@ -/* prmip.c (write MIP solution in printable format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -#define xfprintf glp_format - -int glp_print_mip(glp_prob *P, const char *fname) -{ /* write MIP solution in printable format */ - glp_file *fp; - GLPROW *row; - GLPCOL *col; - int i, j, t, ae_ind, re_ind, ret; - double ae_max, re_max; - xprintf("Writing MIP solution to '%s'...\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "%-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name); - xfprintf(fp, "%-12s%d\n", "Rows:", P->m); - xfprintf(fp, "%-12s%d (%d integer, %d binary)\n", "Columns:", - P->n, glp_get_num_int(P), glp_get_num_bin(P)); - xfprintf(fp, "%-12s%d\n", "Non-zeros:", P->nnz); - t = glp_mip_status(P); - xfprintf(fp, "%-12s%s\n", "Status:", - t == GLP_OPT ? "INTEGER OPTIMAL" : - t == GLP_FEAS ? "INTEGER NON-OPTIMAL" : - t == GLP_NOFEAS ? "INTEGER EMPTY" : - t == GLP_UNDEF ? "INTEGER UNDEFINED" : "???"); - xfprintf(fp, "%-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->mip_obj, - P->dir == GLP_MIN ? "MINimum" : - P->dir == GLP_MAX ? "MAXimum" : "???"); - xfprintf(fp, "\n"); - xfprintf(fp, " No. Row name Activity Lower bound " - " Upper bound\n"); - xfprintf(fp, "------ ------------ ------------- ------------- " - "-------------\n"); - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - xfprintf(fp, "%6d ", i); - if (row->name == NULL || strlen(row->name) <= 12) - xfprintf(fp, "%-12s ", row->name == NULL ? "" : row->name); - else - xfprintf(fp, "%s\n%20s", row->name, ""); - xfprintf(fp, "%3s", ""); - xfprintf(fp, "%13.6g ", - fabs(row->mipx) <= 1e-9 ? 0.0 : row->mipx); - if (row->type == GLP_LO || row->type == GLP_DB || - row->type == GLP_FX) - xfprintf(fp, "%13.6g ", row->lb); - else - xfprintf(fp, "%13s ", ""); - if (row->type == GLP_UP || row->type == GLP_DB) - xfprintf(fp, "%13.6g ", row->ub); - else - xfprintf(fp, "%13s ", row->type == GLP_FX ? "=" : ""); - xfprintf(fp, "\n"); - } - xfprintf(fp, "\n"); - xfprintf(fp, " No. Column name Activity Lower bound " - " Upper bound\n"); - xfprintf(fp, "------ ------------ ------------- ------------- " - "-------------\n"); - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - xfprintf(fp, "%6d ", j); - if (col->name == NULL || strlen(col->name) <= 12) - xfprintf(fp, "%-12s ", col->name == NULL ? "" : col->name); - else - xfprintf(fp, "%s\n%20s", col->name, ""); - xfprintf(fp, "%s ", - col->kind == GLP_CV ? " " : - col->kind == GLP_IV ? "*" : "?"); - xfprintf(fp, "%13.6g ", - fabs(col->mipx) <= 1e-9 ? 0.0 : col->mipx); - if (col->type == GLP_LO || col->type == GLP_DB || - col->type == GLP_FX) - xfprintf(fp, "%13.6g ", col->lb); - else - xfprintf(fp, "%13s ", ""); - if (col->type == GLP_UP || col->type == GLP_DB) - xfprintf(fp, "%13.6g ", col->ub); - else - xfprintf(fp, "%13s ", col->type == GLP_FX ? "=" : ""); - xfprintf(fp, "\n"); - } - xfprintf(fp, "\n"); - xfprintf(fp, "Integer feasibility conditions:\n"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_MIP, GLP_KKT_PE, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.PE: max.abs.err = %.2e on row %d\n", - ae_max, ae_ind); - xfprintf(fp, " max.rel.err = %.2e on row %d\n", - re_max, re_ind); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "SOLUTION IS WRONG"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_MIP, GLP_KKT_PB, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.PB: max.abs.err = %.2e on %s %d\n", - ae_max, ae_ind <= P->m ? "row" : "column", - ae_ind <= P->m ? ae_ind : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on %s %d\n", - re_max, re_ind <= P->m ? "row" : "column", - re_ind <= P->m ? re_ind : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "SOLUTION IS INFEASIBLE"); - xfprintf(fp, "\n"); - xfprintf(fp, "End of output\n"); -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prob.h b/code/3rd_glpk/api/prob.h deleted file mode 100644 index cc9389b5..00000000 --- a/code/3rd_glpk/api/prob.h +++ /dev/null @@ -1,286 +0,0 @@ -/* prob.h (LP/MIP problem object) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef PROB_H -#define PROB_H - -#include "avl.h" -#include "bfd.h" -#include "dmp.h" -#if 1 /* 28/III-2016 */ -#define GLP_UNDOC 1 -#endif -#include "glpk.h" - -typedef struct GLPROW GLPROW; -typedef struct GLPCOL GLPCOL; -typedef struct GLPAIJ GLPAIJ; - -#if 0 /* 04/IV-2016 */ -#define GLP_PROB_MAGIC 0xD7D9D6C2 -#endif - -struct glp_prob -{ /* LP/MIP problem object */ -#if 0 /* 04/IV-2016 */ - unsigned magic; - /* magic value used for debugging */ -#endif - DMP *pool; - /* memory pool to store problem object components */ - glp_tree *tree; - /* pointer to the search tree; set by the MIP solver when this - object is used in the tree as a core MIP object */ -#if 0 /* 08/III-2014 */ - void *parms; - /* reserved for backward compatibility */ -#endif - /*--------------------------------------------------------------*/ - /* LP/MIP data */ - char *name; - /* problem name (1 to 255 chars); NULL means no name is assigned - to the problem */ - char *obj; - /* objective function name (1 to 255 chars); NULL means no name - is assigned to the objective function */ - int dir; - /* optimization direction flag (objective "sense"): - GLP_MIN - minimization - GLP_MAX - maximization */ - double c0; - /* constant term of the objective function ("shift") */ - int m_max; - /* length of the array of rows (enlarged automatically) */ - int n_max; - /* length of the array of columns (enlarged automatically) */ - int m; - /* number of rows, 0 <= m <= m_max */ - int n; - /* number of columns, 0 <= n <= n_max */ - int nnz; - /* number of non-zero constraint coefficients, nnz >= 0 */ - GLPROW **row; /* GLPROW *row[1+m_max]; */ - /* row[i], 1 <= i <= m, is a pointer to i-th row */ - GLPCOL **col; /* GLPCOL *col[1+n_max]; */ - /* col[j], 1 <= j <= n, is a pointer to j-th column */ - AVL *r_tree; - /* row index to find rows by their names; NULL means this index - does not exist */ - AVL *c_tree; - /* column index to find columns by their names; NULL means this - index does not exist */ - /*--------------------------------------------------------------*/ - /* basis factorization (LP) */ - int valid; - /* the factorization is valid only if this flag is set */ - int *head; /* int head[1+m_max]; */ - /* basis header (valid only if the factorization is valid); - head[i] = k is the ordinal number of auxiliary (1 <= k <= m) - or structural (m+1 <= k <= m+n) variable which corresponds to - i-th basic variable xB[i], 1 <= i <= m */ -#if 0 /* 08/III-2014 */ - glp_bfcp *bfcp; - /* basis factorization control parameters; may be NULL */ -#endif - BFD *bfd; /* BFD bfd[1:m,1:m]; */ - /* basis factorization driver; may be NULL */ - /*--------------------------------------------------------------*/ - /* basic solution (LP) */ - int pbs_stat; - /* primal basic solution status: - GLP_UNDEF - primal solution is undefined - GLP_FEAS - primal solution is feasible - GLP_INFEAS - primal solution is infeasible - GLP_NOFEAS - no primal feasible solution exists */ - int dbs_stat; - /* dual basic solution status: - GLP_UNDEF - dual solution is undefined - GLP_FEAS - dual solution is feasible - GLP_INFEAS - dual solution is infeasible - GLP_NOFEAS - no dual feasible solution exists */ - double obj_val; - /* objective function value */ - int it_cnt; - /* simplex method iteration count; increases by one on performing - one simplex iteration */ - int some; - /* ordinal number of some auxiliary or structural variable having - certain property, 0 <= some <= m+n */ - /*--------------------------------------------------------------*/ - /* interior-point solution (LP) */ - int ipt_stat; - /* interior-point solution status: - GLP_UNDEF - interior solution is undefined - GLP_OPT - interior solution is optimal - GLP_INFEAS - interior solution is infeasible - GLP_NOFEAS - no feasible solution exists */ - double ipt_obj; - /* objective function value */ - /*--------------------------------------------------------------*/ - /* integer solution (MIP) */ - int mip_stat; - /* integer solution status: - GLP_UNDEF - integer solution is undefined - GLP_OPT - integer solution is optimal - GLP_FEAS - integer solution is feasible - GLP_NOFEAS - no integer solution exists */ - double mip_obj; - /* objective function value */ -}; - -struct GLPROW -{ /* LP/MIP row (auxiliary variable) */ - int i; - /* ordinal number (1 to m) assigned to this row */ - char *name; - /* row name (1 to 255 chars); NULL means no name is assigned to - this row */ - AVLNODE *node; - /* pointer to corresponding node in the row index; NULL means - that either the row index does not exist or this row has no - name assigned */ -#if 1 /* 20/IX-2008 */ - int level; - unsigned char origin; - unsigned char klass; -#endif - int type; - /* type of the auxiliary variable: - GLP_FR - free variable - GLP_LO - variable with lower bound - GLP_UP - variable with upper bound - GLP_DB - double-bounded variable - GLP_FX - fixed variable */ - double lb; /* non-scaled */ - /* lower bound; if the row has no lower bound, lb is zero */ - double ub; /* non-scaled */ - /* upper bound; if the row has no upper bound, ub is zero */ - /* if the row type is GLP_FX, ub is equal to lb */ - GLPAIJ *ptr; /* non-scaled */ - /* pointer to doubly linked list of constraint coefficients which - are placed in this row */ - double rii; - /* diagonal element r[i,i] of scaling matrix R for this row; - if the scaling is not used, r[i,i] is 1 */ - int stat; - /* status of the auxiliary variable: - GLP_BS - basic variable - GLP_NL - non-basic variable on lower bound - GLP_NU - non-basic variable on upper bound - GLP_NF - non-basic free variable - GLP_NS - non-basic fixed variable */ - int bind; - /* if the auxiliary variable is basic, head[bind] refers to this - row, otherwise, bind is 0; this attribute is valid only if the - basis factorization is valid */ - double prim; /* non-scaled */ - /* primal value of the auxiliary variable in basic solution */ - double dual; /* non-scaled */ - /* dual value of the auxiliary variable in basic solution */ - double pval; /* non-scaled */ - /* primal value of the auxiliary variable in interior solution */ - double dval; /* non-scaled */ - /* dual value of the auxiliary variable in interior solution */ - double mipx; /* non-scaled */ - /* primal value of the auxiliary variable in integer solution */ -}; - -struct GLPCOL -{ /* LP/MIP column (structural variable) */ - int j; - /* ordinal number (1 to n) assigned to this column */ - char *name; - /* column name (1 to 255 chars); NULL means no name is assigned - to this column */ - AVLNODE *node; - /* pointer to corresponding node in the column index; NULL means - that either the column index does not exist or the column has - no name assigned */ - int kind; - /* kind of the structural variable: - GLP_CV - continuous variable - GLP_IV - integer or binary variable */ - int type; - /* type of the structural variable: - GLP_FR - free variable - GLP_LO - variable with lower bound - GLP_UP - variable with upper bound - GLP_DB - double-bounded variable - GLP_FX - fixed variable */ - double lb; /* non-scaled */ - /* lower bound; if the column has no lower bound, lb is zero */ - double ub; /* non-scaled */ - /* upper bound; if the column has no upper bound, ub is zero */ - /* if the column type is GLP_FX, ub is equal to lb */ - double coef; /* non-scaled */ - /* objective coefficient at the structural variable */ - GLPAIJ *ptr; /* non-scaled */ - /* pointer to doubly linked list of constraint coefficients which - are placed in this column */ - double sjj; - /* diagonal element s[j,j] of scaling matrix S for this column; - if the scaling is not used, s[j,j] is 1 */ - int stat; - /* status of the structural variable: - GLP_BS - basic variable - GLP_NL - non-basic variable on lower bound - GLP_NU - non-basic variable on upper bound - GLP_NF - non-basic free variable - GLP_NS - non-basic fixed variable */ - int bind; - /* if the structural variable is basic, head[bind] refers to - this column; otherwise, bind is 0; this attribute is valid only - if the basis factorization is valid */ - double prim; /* non-scaled */ - /* primal value of the structural variable in basic solution */ - double dual; /* non-scaled */ - /* dual value of the structural variable in basic solution */ - double pval; /* non-scaled */ - /* primal value of the structural variable in interior solution */ - double dval; /* non-scaled */ - /* dual value of the structural variable in interior solution */ - double mipx; /* non-scaled */ - /* primal value of the structural variable in integer solution */ -}; - -struct GLPAIJ -{ /* constraint coefficient a[i,j] */ - GLPROW *row; - /* pointer to row, where this coefficient is placed */ - GLPCOL *col; - /* pointer to column, where this coefficient is placed */ - double val; - /* numeric (non-zero) value of this coefficient */ - GLPAIJ *r_prev; - /* pointer to previous coefficient in the same row */ - GLPAIJ *r_next; - /* pointer to next coefficient in the same row */ - GLPAIJ *c_prev; - /* pointer to previous coefficient in the same column */ - GLPAIJ *c_next; - /* pointer to next coefficient in the same column */ -}; - -#endif - -/* eof */ diff --git a/code/3rd_glpk/api/prob1.c b/code/3rd_glpk/api/prob1.c deleted file mode 100644 index 6afad442..00000000 --- a/code/3rd_glpk/api/prob1.c +++ /dev/null @@ -1,1588 +0,0 @@ -/* prob1.c (problem creating and modifying routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/* CAUTION: DO NOT CHANGE THE LIMITS BELOW */ - -#define M_MAX 100000000 /* = 100*10^6 */ -/* maximal number of rows in the problem object */ - -#define N_MAX 100000000 /* = 100*10^6 */ -/* maximal number of columns in the problem object */ - -#define NNZ_MAX 500000000 /* = 500*10^6 */ -/* maximal number of constraint coefficients in the problem object */ - -/*********************************************************************** -* NAME -* -* glp_create_prob - create problem object -* -* SYNOPSIS -* -* glp_prob *glp_create_prob(void); -* -* DESCRIPTION -* -* The routine glp_create_prob creates a new problem object, which is -* initially "empty", i.e. has no rows and columns. -* -* RETURNS -* -* The routine returns a pointer to the object created, which should be -* used in any subsequent operations on this object. */ - -static void create_prob(glp_prob *lp) -#if 0 /* 04/IV-2016 */ -{ lp->magic = GLP_PROB_MAGIC; -#else -{ -#endif - lp->pool = dmp_create_pool(); -#if 0 /* 08/III-2014 */ -#if 0 /* 17/XI-2009 */ - lp->cps = xmalloc(sizeof(struct LPXCPS)); - lpx_reset_parms(lp); -#else - lp->parms = NULL; -#endif -#endif - lp->tree = NULL; -#if 0 - lp->lwa = 0; - lp->cwa = NULL; -#endif - /* LP/MIP data */ - lp->name = NULL; - lp->obj = NULL; - lp->dir = GLP_MIN; - lp->c0 = 0.0; - lp->m_max = 100; - lp->n_max = 200; - lp->m = lp->n = 0; - lp->nnz = 0; - lp->row = xcalloc(1+lp->m_max, sizeof(GLPROW *)); - lp->col = xcalloc(1+lp->n_max, sizeof(GLPCOL *)); - lp->r_tree = lp->c_tree = NULL; - /* basis factorization */ - lp->valid = 0; - lp->head = xcalloc(1+lp->m_max, sizeof(int)); -#if 0 /* 08/III-2014 */ - lp->bfcp = NULL; -#endif - lp->bfd = NULL; - /* basic solution (LP) */ - lp->pbs_stat = lp->dbs_stat = GLP_UNDEF; - lp->obj_val = 0.0; - lp->it_cnt = 0; - lp->some = 0; - /* interior-point solution (LP) */ - lp->ipt_stat = GLP_UNDEF; - lp->ipt_obj = 0.0; - /* integer solution (MIP) */ - lp->mip_stat = GLP_UNDEF; - lp->mip_obj = 0.0; - return; -} - -glp_prob *glp_create_prob(void) -{ glp_prob *lp; - lp = xmalloc(sizeof(glp_prob)); - create_prob(lp); - return lp; -} - -/*********************************************************************** -* NAME -* -* glp_set_prob_name - assign (change) problem name -* -* SYNOPSIS -* -* void glp_set_prob_name(glp_prob *lp, const char *name); -* -* DESCRIPTION -* -* The routine glp_set_prob_name assigns a given symbolic name (1 up to -* 255 characters) to the specified problem object. -* -* If the parameter name is NULL or empty string, the routine erases an -* existing symbolic name of the problem object. */ - -void glp_set_prob_name(glp_prob *lp, const char *name) -{ glp_tree *tree = lp->tree; - if (tree != NULL && tree->reason != 0) - xerror("glp_set_prob_name: operation not allowed\n"); - if (lp->name != NULL) - { dmp_free_atom(lp->pool, lp->name, strlen(lp->name)+1); - lp->name = NULL; - } - if (!(name == NULL || name[0] == '\0')) - { int k; - for (k = 0; name[k] != '\0'; k++) - { if (k == 256) - xerror("glp_set_prob_name: problem name too long\n"); - if (iscntrl((unsigned char)name[k])) - xerror("glp_set_prob_name: problem name contains invalid" - " character(s)\n"); - } - lp->name = dmp_get_atom(lp->pool, strlen(name)+1); - strcpy(lp->name, name); - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_obj_name - assign (change) objective function name -* -* SYNOPSIS -* -* void glp_set_obj_name(glp_prob *lp, const char *name); -* -* DESCRIPTION -* -* The routine glp_set_obj_name assigns a given symbolic name (1 up to -* 255 characters) to the objective function of the specified problem -* object. -* -* If the parameter name is NULL or empty string, the routine erases an -* existing name of the objective function. */ - -void glp_set_obj_name(glp_prob *lp, const char *name) -{ glp_tree *tree = lp->tree; - if (tree != NULL && tree->reason != 0) - xerror("glp_set_obj_name: operation not allowed\n"); - if (lp->obj != NULL) - { dmp_free_atom(lp->pool, lp->obj, strlen(lp->obj)+1); - lp->obj = NULL; - } - if (!(name == NULL || name[0] == '\0')) - { int k; - for (k = 0; name[k] != '\0'; k++) - { if (k == 256) - xerror("glp_set_obj_name: objective name too long\n"); - if (iscntrl((unsigned char)name[k])) - xerror("glp_set_obj_name: objective name contains invali" - "d character(s)\n"); - } - lp->obj = dmp_get_atom(lp->pool, strlen(name)+1); - strcpy(lp->obj, name); - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_obj_dir - set (change) optimization direction flag -* -* SYNOPSIS -* -* void glp_set_obj_dir(glp_prob *lp, int dir); -* -* DESCRIPTION -* -* The routine glp_set_obj_dir sets (changes) optimization direction -* flag (i.e. "sense" of the objective function) as specified by the -* parameter dir: -* -* GLP_MIN - minimization; -* GLP_MAX - maximization. */ - -void glp_set_obj_dir(glp_prob *lp, int dir) -{ glp_tree *tree = lp->tree; - if (tree != NULL && tree->reason != 0) - xerror("glp_set_obj_dir: operation not allowed\n"); - if (!(dir == GLP_MIN || dir == GLP_MAX)) - xerror("glp_set_obj_dir: dir = %d; invalid direction flag\n", - dir); - lp->dir = dir; - return; -} - -/*********************************************************************** -* NAME -* -* glp_add_rows - add new rows to problem object -* -* SYNOPSIS -* -* int glp_add_rows(glp_prob *lp, int nrs); -* -* DESCRIPTION -* -* The routine glp_add_rows adds nrs rows (constraints) to the specified -* problem object. New rows are always added to the end of the row list, -* so the ordinal numbers of existing rows remain unchanged. -* -* Being added each new row is initially free (unbounded) and has empty -* list of the constraint coefficients. -* -* RETURNS -* -* The routine glp_add_rows returns the ordinal number of the first new -* row added to the problem object. */ - -int glp_add_rows(glp_prob *lp, int nrs) -{ glp_tree *tree = lp->tree; - GLPROW *row; - int m_new, i; - /* determine new number of rows */ - if (nrs < 1) - xerror("glp_add_rows: nrs = %d; invalid number of rows\n", - nrs); - if (nrs > M_MAX - lp->m) - xerror("glp_add_rows: nrs = %d; too many rows\n", nrs); - m_new = lp->m + nrs; - /* increase the room, if necessary */ - if (lp->m_max < m_new) - { GLPROW **save = lp->row; - while (lp->m_max < m_new) - { lp->m_max += lp->m_max; - xassert(lp->m_max > 0); - } - lp->row = xcalloc(1+lp->m_max, sizeof(GLPROW *)); - memcpy(&lp->row[1], &save[1], lp->m * sizeof(GLPROW *)); - xfree(save); - /* do not forget about the basis header */ - xfree(lp->head); - lp->head = xcalloc(1+lp->m_max, sizeof(int)); - } - /* add new rows to the end of the row list */ - for (i = lp->m+1; i <= m_new; i++) - { /* create row descriptor */ - lp->row[i] = row = dmp_get_atom(lp->pool, sizeof(GLPROW)); - row->i = i; - row->name = NULL; - row->node = NULL; -#if 1 /* 20/IX-2008 */ - row->level = 0; - row->origin = 0; - row->klass = 0; - if (tree != NULL) - { switch (tree->reason) - { case 0: - break; - case GLP_IROWGEN: - xassert(tree->curr != NULL); - row->level = tree->curr->level; - row->origin = GLP_RF_LAZY; - break; - case GLP_ICUTGEN: - xassert(tree->curr != NULL); - row->level = tree->curr->level; - row->origin = GLP_RF_CUT; - break; - default: - xassert(tree != tree); - } - } -#endif - row->type = GLP_FR; - row->lb = row->ub = 0.0; - row->ptr = NULL; - row->rii = 1.0; - row->stat = GLP_BS; -#if 0 - row->bind = -1; -#else - row->bind = 0; -#endif - row->prim = row->dual = 0.0; - row->pval = row->dval = 0.0; - row->mipx = 0.0; - } - /* set new number of rows */ - lp->m = m_new; - /* invalidate the basis factorization */ - lp->valid = 0; -#if 1 - if (tree != NULL && tree->reason != 0) tree->reopt = 1; -#endif - /* return the ordinal number of the first row added */ - return m_new - nrs + 1; -} - -/*********************************************************************** -* NAME -* -* glp_add_cols - add new columns to problem object -* -* SYNOPSIS -* -* int glp_add_cols(glp_prob *lp, int ncs); -* -* DESCRIPTION -* -* The routine glp_add_cols adds ncs columns (structural variables) to -* the specified problem object. New columns are always added to the end -* of the column list, so the ordinal numbers of existing columns remain -* unchanged. -* -* Being added each new column is initially fixed at zero and has empty -* list of the constraint coefficients. -* -* RETURNS -* -* The routine glp_add_cols returns the ordinal number of the first new -* column added to the problem object. */ - -int glp_add_cols(glp_prob *lp, int ncs) -{ glp_tree *tree = lp->tree; - GLPCOL *col; - int n_new, j; - if (tree != NULL && tree->reason != 0) - xerror("glp_add_cols: operation not allowed\n"); - /* determine new number of columns */ - if (ncs < 1) - xerror("glp_add_cols: ncs = %d; invalid number of columns\n", - ncs); - if (ncs > N_MAX - lp->n) - xerror("glp_add_cols: ncs = %d; too many columns\n", ncs); - n_new = lp->n + ncs; - /* increase the room, if necessary */ - if (lp->n_max < n_new) - { GLPCOL **save = lp->col; - while (lp->n_max < n_new) - { lp->n_max += lp->n_max; - xassert(lp->n_max > 0); - } - lp->col = xcalloc(1+lp->n_max, sizeof(GLPCOL *)); - memcpy(&lp->col[1], &save[1], lp->n * sizeof(GLPCOL *)); - xfree(save); - } - /* add new columns to the end of the column list */ - for (j = lp->n+1; j <= n_new; j++) - { /* create column descriptor */ - lp->col[j] = col = dmp_get_atom(lp->pool, sizeof(GLPCOL)); - col->j = j; - col->name = NULL; - col->node = NULL; - col->kind = GLP_CV; - col->type = GLP_FX; - col->lb = col->ub = 0.0; - col->coef = 0.0; - col->ptr = NULL; - col->sjj = 1.0; - col->stat = GLP_NS; -#if 0 - col->bind = -1; -#else - col->bind = 0; /* the basis may remain valid */ -#endif - col->prim = col->dual = 0.0; - col->pval = col->dval = 0.0; - col->mipx = 0.0; - } - /* set new number of columns */ - lp->n = n_new; - /* return the ordinal number of the first column added */ - return n_new - ncs + 1; -} - -/*********************************************************************** -* NAME -* -* glp_set_row_name - assign (change) row name -* -* SYNOPSIS -* -* void glp_set_row_name(glp_prob *lp, int i, const char *name); -* -* DESCRIPTION -* -* The routine glp_set_row_name assigns a given symbolic name (1 up to -* 255 characters) to i-th row (auxiliary variable) of the specified -* problem object. -* -* If the parameter name is NULL or empty string, the routine erases an -* existing name of i-th row. */ - -void glp_set_row_name(glp_prob *lp, int i, const char *name) -{ glp_tree *tree = lp->tree; - GLPROW *row; - if (!(1 <= i && i <= lp->m)) - xerror("glp_set_row_name: i = %d; row number out of range\n", - i); - row = lp->row[i]; - if (tree != NULL && tree->reason != 0) - { xassert(tree->curr != NULL); - xassert(row->level == tree->curr->level); - } - if (row->name != NULL) - { if (row->node != NULL) - { xassert(lp->r_tree != NULL); - avl_delete_node(lp->r_tree, row->node); - row->node = NULL; - } - dmp_free_atom(lp->pool, row->name, strlen(row->name)+1); - row->name = NULL; - } - if (!(name == NULL || name[0] == '\0')) - { int k; - for (k = 0; name[k] != '\0'; k++) - { if (k == 256) - xerror("glp_set_row_name: i = %d; row name too long\n", - i); - if (iscntrl((unsigned char)name[k])) - xerror("glp_set_row_name: i = %d: row name contains inva" - "lid character(s)\n", i); - } - row->name = dmp_get_atom(lp->pool, strlen(name)+1); - strcpy(row->name, name); - if (lp->r_tree != NULL) - { xassert(row->node == NULL); - row->node = avl_insert_node(lp->r_tree, row->name); - avl_set_node_link(row->node, row); - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_col_name - assign (change) column name -* -* SYNOPSIS -* -* void glp_set_col_name(glp_prob *lp, int j, const char *name); -* -* DESCRIPTION -* -* The routine glp_set_col_name assigns a given symbolic name (1 up to -* 255 characters) to j-th column (structural variable) of the specified -* problem object. -* -* If the parameter name is NULL or empty string, the routine erases an -* existing name of j-th column. */ - -void glp_set_col_name(glp_prob *lp, int j, const char *name) -{ glp_tree *tree = lp->tree; - GLPCOL *col; - if (tree != NULL && tree->reason != 0) - xerror("glp_set_col_name: operation not allowed\n"); - if (!(1 <= j && j <= lp->n)) - xerror("glp_set_col_name: j = %d; column number out of range\n" - , j); - col = lp->col[j]; - if (col->name != NULL) - { if (col->node != NULL) - { xassert(lp->c_tree != NULL); - avl_delete_node(lp->c_tree, col->node); - col->node = NULL; - } - dmp_free_atom(lp->pool, col->name, strlen(col->name)+1); - col->name = NULL; - } - if (!(name == NULL || name[0] == '\0')) - { int k; - for (k = 0; name[k] != '\0'; k++) - { if (k == 256) - xerror("glp_set_col_name: j = %d; column name too long\n" - , j); - if (iscntrl((unsigned char)name[k])) - xerror("glp_set_col_name: j = %d: column name contains i" - "nvalid character(s)\n", j); - } - col->name = dmp_get_atom(lp->pool, strlen(name)+1); - strcpy(col->name, name); - if (lp->c_tree != NULL && col->name != NULL) - { xassert(col->node == NULL); - col->node = avl_insert_node(lp->c_tree, col->name); - avl_set_node_link(col->node, col); - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_row_bnds - set (change) row bounds -* -* SYNOPSIS -* -* void glp_set_row_bnds(glp_prob *lp, int i, int type, double lb, -* double ub); -* -* DESCRIPTION -* -* The routine glp_set_row_bnds sets (changes) the type and bounds of -* i-th row (auxiliary variable) of the specified problem object. -* -* Parameters type, lb, and ub specify the type, lower bound, and upper -* bound, respectively, as follows: -* -* Type Bounds Comments -* ------------------------------------------------------ -* GLP_FR -inf < x < +inf Free variable -* GLP_LO lb <= x < +inf Variable with lower bound -* GLP_UP -inf < x <= ub Variable with upper bound -* GLP_DB lb <= x <= ub Double-bounded variable -* GLP_FX x = lb Fixed variable -* -* where x is the auxiliary variable associated with i-th row. -* -* If the row has no lower bound, the parameter lb is ignored. If the -* row has no upper bound, the parameter ub is ignored. If the row is -* an equality constraint (i.e. the corresponding auxiliary variable is -* of fixed type), only the parameter lb is used while the parameter ub -* is ignored. */ - -void glp_set_row_bnds(glp_prob *lp, int i, int type, double lb, - double ub) -{ GLPROW *row; - if (!(1 <= i && i <= lp->m)) - xerror("glp_set_row_bnds: i = %d; row number out of range\n", - i); - row = lp->row[i]; - row->type = type; - switch (type) - { case GLP_FR: - row->lb = row->ub = 0.0; - if (row->stat != GLP_BS) row->stat = GLP_NF; - break; - case GLP_LO: - row->lb = lb, row->ub = 0.0; - if (row->stat != GLP_BS) row->stat = GLP_NL; - break; - case GLP_UP: - row->lb = 0.0, row->ub = ub; - if (row->stat != GLP_BS) row->stat = GLP_NU; - break; - case GLP_DB: - row->lb = lb, row->ub = ub; - if (!(row->stat == GLP_BS || - row->stat == GLP_NL || row->stat == GLP_NU)) - row->stat = (fabs(lb) <= fabs(ub) ? GLP_NL : GLP_NU); - break; - case GLP_FX: - row->lb = row->ub = lb; - if (row->stat != GLP_BS) row->stat = GLP_NS; - break; - default: - xerror("glp_set_row_bnds: i = %d; type = %d; invalid row ty" - "pe\n", i, type); - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_col_bnds - set (change) column bounds -* -* SYNOPSIS -* -* void glp_set_col_bnds(glp_prob *lp, int j, int type, double lb, -* double ub); -* -* DESCRIPTION -* -* The routine glp_set_col_bnds sets (changes) the type and bounds of -* j-th column (structural variable) of the specified problem object. -* -* Parameters type, lb, and ub specify the type, lower bound, and upper -* bound, respectively, as follows: -* -* Type Bounds Comments -* ------------------------------------------------------ -* GLP_FR -inf < x < +inf Free variable -* GLP_LO lb <= x < +inf Variable with lower bound -* GLP_UP -inf < x <= ub Variable with upper bound -* GLP_DB lb <= x <= ub Double-bounded variable -* GLP_FX x = lb Fixed variable -* -* where x is the structural variable associated with j-th column. -* -* If the column has no lower bound, the parameter lb is ignored. If the -* column has no upper bound, the parameter ub is ignored. If the column -* is of fixed type, only the parameter lb is used while the parameter -* ub is ignored. */ - -void glp_set_col_bnds(glp_prob *lp, int j, int type, double lb, - double ub) -{ GLPCOL *col; - if (!(1 <= j && j <= lp->n)) - xerror("glp_set_col_bnds: j = %d; column number out of range\n" - , j); - col = lp->col[j]; - col->type = type; - switch (type) - { case GLP_FR: - col->lb = col->ub = 0.0; - if (col->stat != GLP_BS) col->stat = GLP_NF; - break; - case GLP_LO: - col->lb = lb, col->ub = 0.0; - if (col->stat != GLP_BS) col->stat = GLP_NL; - break; - case GLP_UP: - col->lb = 0.0, col->ub = ub; - if (col->stat != GLP_BS) col->stat = GLP_NU; - break; - case GLP_DB: - col->lb = lb, col->ub = ub; - if (!(col->stat == GLP_BS || - col->stat == GLP_NL || col->stat == GLP_NU)) - col->stat = (fabs(lb) <= fabs(ub) ? GLP_NL : GLP_NU); - break; - case GLP_FX: - col->lb = col->ub = lb; - if (col->stat != GLP_BS) col->stat = GLP_NS; - break; - default: - xerror("glp_set_col_bnds: j = %d; type = %d; invalid column" - " type\n", j, type); - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_obj_coef - set (change) obj. coefficient or constant term -* -* SYNOPSIS -* -* void glp_set_obj_coef(glp_prob *lp, int j, double coef); -* -* DESCRIPTION -* -* The routine glp_set_obj_coef sets (changes) objective coefficient at -* j-th column (structural variable) of the specified problem object. -* -* If the parameter j is 0, the routine sets (changes) the constant term -* ("shift") of the objective function. */ - -void glp_set_obj_coef(glp_prob *lp, int j, double coef) -{ glp_tree *tree = lp->tree; - if (tree != NULL && tree->reason != 0) - xerror("glp_set_obj_coef: operation not allowed\n"); - if (!(0 <= j && j <= lp->n)) - xerror("glp_set_obj_coef: j = %d; column number out of range\n" - , j); - if (j == 0) - lp->c0 = coef; - else - lp->col[j]->coef = coef; - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_mat_row - set (replace) row of the constraint matrix -* -* SYNOPSIS -* -* void glp_set_mat_row(glp_prob *lp, int i, int len, const int ind[], -* const double val[]); -* -* DESCRIPTION -* -* The routine glp_set_mat_row stores (replaces) the contents of i-th -* row of the constraint matrix of the specified problem object. -* -* Column indices and numeric values of new row elements must be placed -* in locations ind[1], ..., ind[len] and val[1], ..., val[len], where -* 0 <= len <= n is the new length of i-th row, n is the current number -* of columns in the problem object. Elements with identical column -* indices are not allowed. Zero elements are allowed, but they are not -* stored in the constraint matrix. -* -* If the parameter len is zero, the parameters ind and/or val can be -* specified as NULL. */ - -void glp_set_mat_row(glp_prob *lp, int i, int len, const int ind[], - const double val[]) -{ glp_tree *tree = lp->tree; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij, *next; - int j, k; - /* obtain pointer to i-th row */ - if (!(1 <= i && i <= lp->m)) - xerror("glp_set_mat_row: i = %d; row number out of range\n", - i); - row = lp->row[i]; - if (tree != NULL && tree->reason != 0) - { xassert(tree->curr != NULL); - xassert(row->level == tree->curr->level); - } - /* remove all existing elements from i-th row */ - while (row->ptr != NULL) - { /* take next element in the row */ - aij = row->ptr; - /* remove the element from the row list */ - row->ptr = aij->r_next; - /* obtain pointer to corresponding column */ - col = aij->col; - /* remove the element from the column list */ - if (aij->c_prev == NULL) - col->ptr = aij->c_next; - else - aij->c_prev->c_next = aij->c_next; - if (aij->c_next == NULL) - ; - else - aij->c_next->c_prev = aij->c_prev; - /* return the element to the memory pool */ - dmp_free_atom(lp->pool, aij, sizeof(GLPAIJ)), lp->nnz--; - /* if the corresponding column is basic, invalidate the basis - factorization */ - if (col->stat == GLP_BS) lp->valid = 0; - } - /* store new contents of i-th row */ - if (!(0 <= len && len <= lp->n)) - xerror("glp_set_mat_row: i = %d; len = %d; invalid row length " - "\n", i, len); - if (len > NNZ_MAX - lp->nnz) - xerror("glp_set_mat_row: i = %d; len = %d; too many constraint" - " coefficients\n", i, len); - for (k = 1; k <= len; k++) - { /* take number j of corresponding column */ - j = ind[k]; - /* obtain pointer to j-th column */ - if (!(1 <= j && j <= lp->n)) - xerror("glp_set_mat_row: i = %d; ind[%d] = %d; column index" - " out of range\n", i, k, j); - col = lp->col[j]; - /* if there is element with the same column index, it can only - be found in the beginning of j-th column list */ - if (col->ptr != NULL && col->ptr->row->i == i) - xerror("glp_set_mat_row: i = %d; ind[%d] = %d; duplicate co" - "lumn indices not allowed\n", i, k, j); - /* create new element */ - aij = dmp_get_atom(lp->pool, sizeof(GLPAIJ)), lp->nnz++; - aij->row = row; - aij->col = col; - aij->val = val[k]; - /* add the new element to the beginning of i-th row and j-th - column lists */ - aij->r_prev = NULL; - aij->r_next = row->ptr; - aij->c_prev = NULL; - aij->c_next = col->ptr; - if (aij->r_next != NULL) aij->r_next->r_prev = aij; - if (aij->c_next != NULL) aij->c_next->c_prev = aij; - row->ptr = col->ptr = aij; - /* if the corresponding column is basic, invalidate the basis - factorization */ - if (col->stat == GLP_BS && aij->val != 0.0) lp->valid = 0; - } - /* remove zero elements from i-th row */ - for (aij = row->ptr; aij != NULL; aij = next) - { next = aij->r_next; - if (aij->val == 0.0) - { /* remove the element from the row list */ - if (aij->r_prev == NULL) - row->ptr = next; - else - aij->r_prev->r_next = next; - if (next == NULL) - ; - else - next->r_prev = aij->r_prev; - /* remove the element from the column list */ - xassert(aij->c_prev == NULL); - aij->col->ptr = aij->c_next; - if (aij->c_next != NULL) aij->c_next->c_prev = NULL; - /* return the element to the memory pool */ - dmp_free_atom(lp->pool, aij, sizeof(GLPAIJ)), lp->nnz--; - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_mat_col - set (replace) column of the constraint matrix -* -* SYNOPSIS -* -* void glp_set_mat_col(glp_prob *lp, int j, int len, const int ind[], -* const double val[]); -* -* DESCRIPTION -* -* The routine glp_set_mat_col stores (replaces) the contents of j-th -* column of the constraint matrix of the specified problem object. -* -* Row indices and numeric values of new column elements must be placed -* in locations ind[1], ..., ind[len] and val[1], ..., val[len], where -* 0 <= len <= m is the new length of j-th column, m is the current -* number of rows in the problem object. Elements with identical column -* indices are not allowed. Zero elements are allowed, but they are not -* stored in the constraint matrix. -* -* If the parameter len is zero, the parameters ind and/or val can be -* specified as NULL. */ - -void glp_set_mat_col(glp_prob *lp, int j, int len, const int ind[], - const double val[]) -{ glp_tree *tree = lp->tree; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij, *next; - int i, k; - if (tree != NULL && tree->reason != 0) - xerror("glp_set_mat_col: operation not allowed\n"); - /* obtain pointer to j-th column */ - if (!(1 <= j && j <= lp->n)) - xerror("glp_set_mat_col: j = %d; column number out of range\n", - j); - col = lp->col[j]; - /* remove all existing elements from j-th column */ - while (col->ptr != NULL) - { /* take next element in the column */ - aij = col->ptr; - /* remove the element from the column list */ - col->ptr = aij->c_next; - /* obtain pointer to corresponding row */ - row = aij->row; - /* remove the element from the row list */ - if (aij->r_prev == NULL) - row->ptr = aij->r_next; - else - aij->r_prev->r_next = aij->r_next; - if (aij->r_next == NULL) - ; - else - aij->r_next->r_prev = aij->r_prev; - /* return the element to the memory pool */ - dmp_free_atom(lp->pool, aij, sizeof(GLPAIJ)), lp->nnz--; - } - /* store new contents of j-th column */ - if (!(0 <= len && len <= lp->m)) - xerror("glp_set_mat_col: j = %d; len = %d; invalid column leng" - "th\n", j, len); - if (len > NNZ_MAX - lp->nnz) - xerror("glp_set_mat_col: j = %d; len = %d; too many constraint" - " coefficients\n", j, len); - for (k = 1; k <= len; k++) - { /* take number i of corresponding row */ - i = ind[k]; - /* obtain pointer to i-th row */ - if (!(1 <= i && i <= lp->m)) - xerror("glp_set_mat_col: j = %d; ind[%d] = %d; row index ou" - "t of range\n", j, k, i); - row = lp->row[i]; - /* if there is element with the same row index, it can only be - found in the beginning of i-th row list */ - if (row->ptr != NULL && row->ptr->col->j == j) - xerror("glp_set_mat_col: j = %d; ind[%d] = %d; duplicate ro" - "w indices not allowed\n", j, k, i); - /* create new element */ - aij = dmp_get_atom(lp->pool, sizeof(GLPAIJ)), lp->nnz++; - aij->row = row; - aij->col = col; - aij->val = val[k]; - /* add the new element to the beginning of i-th row and j-th - column lists */ - aij->r_prev = NULL; - aij->r_next = row->ptr; - aij->c_prev = NULL; - aij->c_next = col->ptr; - if (aij->r_next != NULL) aij->r_next->r_prev = aij; - if (aij->c_next != NULL) aij->c_next->c_prev = aij; - row->ptr = col->ptr = aij; - } - /* remove zero elements from j-th column */ - for (aij = col->ptr; aij != NULL; aij = next) - { next = aij->c_next; - if (aij->val == 0.0) - { /* remove the element from the row list */ - xassert(aij->r_prev == NULL); - aij->row->ptr = aij->r_next; - if (aij->r_next != NULL) aij->r_next->r_prev = NULL; - /* remove the element from the column list */ - if (aij->c_prev == NULL) - col->ptr = next; - else - aij->c_prev->c_next = next; - if (next == NULL) - ; - else - next->c_prev = aij->c_prev; - /* return the element to the memory pool */ - dmp_free_atom(lp->pool, aij, sizeof(GLPAIJ)), lp->nnz--; - } - } - /* if j-th column is basic, invalidate the basis factorization */ - if (col->stat == GLP_BS) lp->valid = 0; - return; -} - -/*********************************************************************** -* NAME -* -* glp_load_matrix - load (replace) the whole constraint matrix -* -* SYNOPSIS -* -* void glp_load_matrix(glp_prob *lp, int ne, const int ia[], -* const int ja[], const double ar[]); -* -* DESCRIPTION -* -* The routine glp_load_matrix loads the constraint matrix passed in -* the arrays ia, ja, and ar into the specified problem object. Before -* loading the current contents of the constraint matrix is destroyed. -* -* Constraint coefficients (elements of the constraint matrix) must be -* specified as triplets (ia[k], ja[k], ar[k]) for k = 1, ..., ne, -* where ia[k] is the row index, ja[k] is the column index, ar[k] is a -* numeric value of corresponding constraint coefficient. The parameter -* ne specifies the total number of (non-zero) elements in the matrix -* to be loaded. Coefficients with identical indices are not allowed. -* Zero coefficients are allowed, however, they are not stored in the -* constraint matrix. -* -* If the parameter ne is zero, the parameters ia, ja, and ar can be -* specified as NULL. */ - -void glp_load_matrix(glp_prob *lp, int ne, const int ia[], - const int ja[], const double ar[]) -{ glp_tree *tree = lp->tree; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij, *next; - int i, j, k; - if (tree != NULL && tree->reason != 0) - xerror("glp_load_matrix: operation not allowed\n"); - /* clear the constraint matrix */ - for (i = 1; i <= lp->m; i++) - { row = lp->row[i]; - while (row->ptr != NULL) - { aij = row->ptr; - row->ptr = aij->r_next; - dmp_free_atom(lp->pool, aij, sizeof(GLPAIJ)), lp->nnz--; - } - } - xassert(lp->nnz == 0); - for (j = 1; j <= lp->n; j++) lp->col[j]->ptr = NULL; - /* load the new contents of the constraint matrix and build its - row lists */ - if (ne < 0) - xerror("glp_load_matrix: ne = %d; invalid number of constraint" - " coefficients\n", ne); - if (ne > NNZ_MAX) - xerror("glp_load_matrix: ne = %d; too many constraint coeffici" - "ents\n", ne); - for (k = 1; k <= ne; k++) - { /* take indices of new element */ - i = ia[k], j = ja[k]; - /* obtain pointer to i-th row */ - if (!(1 <= i && i <= lp->m)) - xerror("glp_load_matrix: ia[%d] = %d; row index out of rang" - "e\n", k, i); - row = lp->row[i]; - /* obtain pointer to j-th column */ - if (!(1 <= j && j <= lp->n)) - xerror("glp_load_matrix: ja[%d] = %d; column index out of r" - "ange\n", k, j); - col = lp->col[j]; - /* create new element */ - aij = dmp_get_atom(lp->pool, sizeof(GLPAIJ)), lp->nnz++; - aij->row = row; - aij->col = col; - aij->val = ar[k]; - /* add the new element to the beginning of i-th row list */ - aij->r_prev = NULL; - aij->r_next = row->ptr; - if (aij->r_next != NULL) aij->r_next->r_prev = aij; - row->ptr = aij; - } - xassert(lp->nnz == ne); - /* build column lists of the constraint matrix and check elements - with identical indices */ - for (i = 1; i <= lp->m; i++) - { for (aij = lp->row[i]->ptr; aij != NULL; aij = aij->r_next) - { /* obtain pointer to corresponding column */ - col = aij->col; - /* if there is element with identical indices, it can only - be found in the beginning of j-th column list */ - if (col->ptr != NULL && col->ptr->row->i == i) - { for (k = 1; k <= ne; k++) - if (ia[k] == i && ja[k] == col->j) break; - xerror("glp_load_mat: ia[%d] = %d; ja[%d] = %d; duplicat" - "e indices not allowed\n", k, i, k, col->j); - } - /* add the element to the beginning of j-th column list */ - aij->c_prev = NULL; - aij->c_next = col->ptr; - if (aij->c_next != NULL) aij->c_next->c_prev = aij; - col->ptr = aij; - } - } - /* remove zero elements from the constraint matrix */ - for (i = 1; i <= lp->m; i++) - { row = lp->row[i]; - for (aij = row->ptr; aij != NULL; aij = next) - { next = aij->r_next; - if (aij->val == 0.0) - { /* remove the element from the row list */ - if (aij->r_prev == NULL) - row->ptr = next; - else - aij->r_prev->r_next = next; - if (next == NULL) - ; - else - next->r_prev = aij->r_prev; - /* remove the element from the column list */ - if (aij->c_prev == NULL) - aij->col->ptr = aij->c_next; - else - aij->c_prev->c_next = aij->c_next; - if (aij->c_next == NULL) - ; - else - aij->c_next->c_prev = aij->c_prev; - /* return the element to the memory pool */ - dmp_free_atom(lp->pool, aij, sizeof(GLPAIJ)), lp->nnz--; - } - } - } - /* invalidate the basis factorization */ - lp->valid = 0; - return; -} - -/*********************************************************************** -* NAME -* -* glp_check_dup - check for duplicate elements in sparse matrix -* -* SYNOPSIS -* -* int glp_check_dup(int m, int n, int ne, const int ia[], -* const int ja[]); -* -* DESCRIPTION -* -* The routine glp_check_dup checks for duplicate elements (that is, -* elements with identical indices) in a sparse matrix specified in the -* coordinate format. -* -* The parameters m and n specifies, respectively, the number of rows -* and columns in the matrix, m >= 0, n >= 0. -* -* The parameter ne specifies the number of (structurally) non-zero -* elements in the matrix, ne >= 0. -* -* Elements of the matrix are specified as doublets (ia[k],ja[k]) for -* k = 1,...,ne, where ia[k] is a row index, ja[k] is a column index. -* -* The routine glp_check_dup can be used prior to a call to the routine -* glp_load_matrix to check that the constraint matrix to be loaded has -* no duplicate elements. -* -* RETURNS -* -* The routine glp_check_dup returns one of the following values: -* -* 0 - the matrix has no duplicate elements; -* -* -k - indices ia[k] or/and ja[k] are out of range; -* -* +k - element (ia[k],ja[k]) is duplicate. */ - -int glp_check_dup(int m, int n, int ne, const int ia[], const int ja[]) -{ int i, j, k, *ptr, *next, ret; - char *flag; - if (m < 0) - xerror("glp_check_dup: m = %d; invalid parameter\n"); - if (n < 0) - xerror("glp_check_dup: n = %d; invalid parameter\n"); - if (ne < 0) - xerror("glp_check_dup: ne = %d; invalid parameter\n"); - if (ne > 0 && ia == NULL) - xerror("glp_check_dup: ia = %p; invalid parameter\n", ia); - if (ne > 0 && ja == NULL) - xerror("glp_check_dup: ja = %p; invalid parameter\n", ja); - for (k = 1; k <= ne; k++) - { i = ia[k], j = ja[k]; - if (!(1 <= i && i <= m && 1 <= j && j <= n)) - { ret = -k; - goto done; - } - } - if (m == 0 || n == 0) - { ret = 0; - goto done; - } - /* allocate working arrays */ - ptr = xcalloc(1+m, sizeof(int)); - next = xcalloc(1+ne, sizeof(int)); - flag = xcalloc(1+n, sizeof(char)); - /* build row lists */ - for (i = 1; i <= m; i++) - ptr[i] = 0; - for (k = 1; k <= ne; k++) - { i = ia[k]; - next[k] = ptr[i]; - ptr[i] = k; - } - /* clear column flags */ - for (j = 1; j <= n; j++) - flag[j] = 0; - /* check for duplicate elements */ - for (i = 1; i <= m; i++) - { for (k = ptr[i]; k != 0; k = next[k]) - { j = ja[k]; - if (flag[j]) - { /* find first element (i,j) */ - for (k = 1; k <= ne; k++) - if (ia[k] == i && ja[k] == j) break; - xassert(k <= ne); - /* find next (duplicate) element (i,j) */ - for (k++; k <= ne; k++) - if (ia[k] == i && ja[k] == j) break; - xassert(k <= ne); - ret = +k; - goto skip; - } - flag[j] = 1; - } - /* clear column flags */ - for (k = ptr[i]; k != 0; k = next[k]) - flag[ja[k]] = 0; - } - /* no duplicate element found */ - ret = 0; -skip: /* free working arrays */ - xfree(ptr); - xfree(next); - xfree(flag); -done: return ret; -} - -/*********************************************************************** -* NAME -* -* glp_sort_matrix - sort elements of the constraint matrix -* -* SYNOPSIS -* -* void glp_sort_matrix(glp_prob *P); -* -* DESCRIPTION -* -* The routine glp_sort_matrix sorts elements of the constraint matrix -* rebuilding its row and column linked lists. On exit from the routine -* the constraint matrix is not changed, however, elements in the row -* linked lists become ordered by ascending column indices, and the -* elements in the column linked lists become ordered by ascending row -* indices. */ - -void glp_sort_matrix(glp_prob *P) -{ GLPAIJ *aij; - int i, j; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_sort_matrix: P = %p; invalid problem object\n", - P); -#endif - /* rebuild row linked lists */ - for (i = P->m; i >= 1; i--) - P->row[i]->ptr = NULL; - for (j = P->n; j >= 1; j--) - { for (aij = P->col[j]->ptr; aij != NULL; aij = aij->c_next) - { i = aij->row->i; - aij->r_prev = NULL; - aij->r_next = P->row[i]->ptr; - if (aij->r_next != NULL) aij->r_next->r_prev = aij; - P->row[i]->ptr = aij; - } - } - /* rebuild column linked lists */ - for (j = P->n; j >= 1; j--) - P->col[j]->ptr = NULL; - for (i = P->m; i >= 1; i--) - { for (aij = P->row[i]->ptr; aij != NULL; aij = aij->r_next) - { j = aij->col->j; - aij->c_prev = NULL; - aij->c_next = P->col[j]->ptr; - if (aij->c_next != NULL) aij->c_next->c_prev = aij; - P->col[j]->ptr = aij; - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_del_rows - delete rows from problem object -* -* SYNOPSIS -* -* void glp_del_rows(glp_prob *lp, int nrs, const int num[]); -* -* DESCRIPTION -* -* The routine glp_del_rows deletes rows from the specified problem -* object. Ordinal numbers of rows to be deleted should be placed in -* locations num[1], ..., num[nrs], where nrs > 0. -* -* Note that deleting rows involves changing ordinal numbers of other -* rows remaining in the problem object. New ordinal numbers of the -* remaining rows are assigned under the assumption that the original -* order of rows is not changed. */ - -void glp_del_rows(glp_prob *lp, int nrs, const int num[]) -{ glp_tree *tree = lp->tree; - GLPROW *row; - int i, k, m_new; - /* mark rows to be deleted */ - if (!(1 <= nrs && nrs <= lp->m)) - xerror("glp_del_rows: nrs = %d; invalid number of rows\n", - nrs); - for (k = 1; k <= nrs; k++) - { /* take the number of row to be deleted */ - i = num[k]; - /* obtain pointer to i-th row */ - if (!(1 <= i && i <= lp->m)) - xerror("glp_del_rows: num[%d] = %d; row number out of range" - "\n", k, i); - row = lp->row[i]; - if (tree != NULL && tree->reason != 0) - { if (!(tree->reason == GLP_IROWGEN || - tree->reason == GLP_ICUTGEN)) - xerror("glp_del_rows: operation not allowed\n"); - xassert(tree->curr != NULL); - if (row->level != tree->curr->level) - xerror("glp_del_rows: num[%d] = %d; invalid attempt to d" - "elete row created not in current subproblem\n", k,i); - if (row->stat != GLP_BS) - xerror("glp_del_rows: num[%d] = %d; invalid attempt to d" - "elete active row (constraint)\n", k, i); - tree->reinv = 1; - } - /* check that the row is not marked yet */ - if (row->i == 0) - xerror("glp_del_rows: num[%d] = %d; duplicate row numbers n" - "ot allowed\n", k, i); - /* erase symbolic name assigned to the row */ - glp_set_row_name(lp, i, NULL); - xassert(row->node == NULL); - /* erase corresponding row of the constraint matrix */ - glp_set_mat_row(lp, i, 0, NULL, NULL); - xassert(row->ptr == NULL); - /* mark the row to be deleted */ - row->i = 0; - } - /* delete all marked rows from the row list */ - m_new = 0; - for (i = 1; i <= lp->m; i++) - { /* obtain pointer to i-th row */ - row = lp->row[i]; - /* check if the row is marked */ - if (row->i == 0) - { /* it is marked, delete it */ - dmp_free_atom(lp->pool, row, sizeof(GLPROW)); - } - else - { /* it is not marked; keep it */ - row->i = ++m_new; - lp->row[row->i] = row; - } - } - /* set new number of rows */ - lp->m = m_new; - /* invalidate the basis factorization */ - lp->valid = 0; - return; -} - -/*********************************************************************** -* NAME -* -* glp_del_cols - delete columns from problem object -* -* SYNOPSIS -* -* void glp_del_cols(glp_prob *lp, int ncs, const int num[]); -* -* DESCRIPTION -* -* The routine glp_del_cols deletes columns from the specified problem -* object. Ordinal numbers of columns to be deleted should be placed in -* locations num[1], ..., num[ncs], where ncs > 0. -* -* Note that deleting columns involves changing ordinal numbers of -* other columns remaining in the problem object. New ordinal numbers -* of the remaining columns are assigned under the assumption that the -* original order of columns is not changed. */ - -void glp_del_cols(glp_prob *lp, int ncs, const int num[]) -{ glp_tree *tree = lp->tree; - GLPCOL *col; - int j, k, n_new; - if (tree != NULL && tree->reason != 0) - xerror("glp_del_cols: operation not allowed\n"); - /* mark columns to be deleted */ - if (!(1 <= ncs && ncs <= lp->n)) - xerror("glp_del_cols: ncs = %d; invalid number of columns\n", - ncs); - for (k = 1; k <= ncs; k++) - { /* take the number of column to be deleted */ - j = num[k]; - /* obtain pointer to j-th column */ - if (!(1 <= j && j <= lp->n)) - xerror("glp_del_cols: num[%d] = %d; column number out of ra" - "nge", k, j); - col = lp->col[j]; - /* check that the column is not marked yet */ - if (col->j == 0) - xerror("glp_del_cols: num[%d] = %d; duplicate column number" - "s not allowed\n", k, j); - /* erase symbolic name assigned to the column */ - glp_set_col_name(lp, j, NULL); - xassert(col->node == NULL); - /* erase corresponding column of the constraint matrix */ - glp_set_mat_col(lp, j, 0, NULL, NULL); - xassert(col->ptr == NULL); - /* mark the column to be deleted */ - col->j = 0; - /* if it is basic, invalidate the basis factorization */ - if (col->stat == GLP_BS) lp->valid = 0; - } - /* delete all marked columns from the column list */ - n_new = 0; - for (j = 1; j <= lp->n; j++) - { /* obtain pointer to j-th column */ - col = lp->col[j]; - /* check if the column is marked */ - if (col->j == 0) - { /* it is marked; delete it */ - dmp_free_atom(lp->pool, col, sizeof(GLPCOL)); - } - else - { /* it is not marked; keep it */ - col->j = ++n_new; - lp->col[col->j] = col; - } - } - /* set new number of columns */ - lp->n = n_new; - /* if the basis header is still valid, adjust it */ - if (lp->valid) - { int m = lp->m; - int *head = lp->head; - for (j = 1; j <= n_new; j++) - { k = lp->col[j]->bind; - if (k != 0) - { xassert(1 <= k && k <= m); - head[k] = m + j; - } - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_copy_prob - copy problem object content -* -* SYNOPSIS -* -* void glp_copy_prob(glp_prob *dest, glp_prob *prob, int names); -* -* DESCRIPTION -* -* The routine glp_copy_prob copies the content of the problem object -* prob to the problem object dest. -* -* The parameter names is a flag. If it is non-zero, the routine also -* copies all symbolic names; otherwise, if it is zero, symbolic names -* are not copied. */ - -void glp_copy_prob(glp_prob *dest, glp_prob *prob, int names) -{ glp_tree *tree = dest->tree; - glp_bfcp bfcp; - int i, j, len, *ind; - double *val; - if (tree != NULL && tree->reason != 0) - xerror("glp_copy_prob: operation not allowed\n"); - if (dest == prob) - xerror("glp_copy_prob: copying problem object to itself not al" - "lowed\n"); - if (!(names == GLP_ON || names == GLP_OFF)) - xerror("glp_copy_prob: names = %d; invalid parameter\n", - names); - glp_erase_prob(dest); - if (names && prob->name != NULL) - glp_set_prob_name(dest, prob->name); - if (names && prob->obj != NULL) - glp_set_obj_name(dest, prob->obj); - dest->dir = prob->dir; - dest->c0 = prob->c0; - if (prob->m > 0) - glp_add_rows(dest, prob->m); - if (prob->n > 0) - glp_add_cols(dest, prob->n); - glp_get_bfcp(prob, &bfcp); - glp_set_bfcp(dest, &bfcp); - dest->pbs_stat = prob->pbs_stat; - dest->dbs_stat = prob->dbs_stat; - dest->obj_val = prob->obj_val; - dest->some = prob->some; - dest->ipt_stat = prob->ipt_stat; - dest->ipt_obj = prob->ipt_obj; - dest->mip_stat = prob->mip_stat; - dest->mip_obj = prob->mip_obj; - for (i = 1; i <= prob->m; i++) - { GLPROW *to = dest->row[i]; - GLPROW *from = prob->row[i]; - if (names && from->name != NULL) - glp_set_row_name(dest, i, from->name); - to->type = from->type; - to->lb = from->lb; - to->ub = from->ub; - to->rii = from->rii; - to->stat = from->stat; - to->prim = from->prim; - to->dual = from->dual; - to->pval = from->pval; - to->dval = from->dval; - to->mipx = from->mipx; - } - ind = xcalloc(1+prob->m, sizeof(int)); - val = xcalloc(1+prob->m, sizeof(double)); - for (j = 1; j <= prob->n; j++) - { GLPCOL *to = dest->col[j]; - GLPCOL *from = prob->col[j]; - if (names && from->name != NULL) - glp_set_col_name(dest, j, from->name); - to->kind = from->kind; - to->type = from->type; - to->lb = from->lb; - to->ub = from->ub; - to->coef = from->coef; - len = glp_get_mat_col(prob, j, ind, val); - glp_set_mat_col(dest, j, len, ind, val); - to->sjj = from->sjj; - to->stat = from->stat; - to->prim = from->prim; - to->dual = from->dual; - to->pval = from->pval; - to->dval = from->dval; - to->mipx = from->mipx; - } - xfree(ind); - xfree(val); - return; -} - -/*********************************************************************** -* NAME -* -* glp_erase_prob - erase problem object content -* -* SYNOPSIS -* -* void glp_erase_prob(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_erase_prob erases the content of the specified -* problem object. The effect of this operation is the same as if the -* problem object would be deleted with the routine glp_delete_prob and -* then created anew with the routine glp_create_prob, with exception -* that the handle (pointer) to the problem object remains valid. */ - -static void delete_prob(glp_prob *lp); - -void glp_erase_prob(glp_prob *lp) -{ glp_tree *tree = lp->tree; - if (tree != NULL && tree->reason != 0) - xerror("glp_erase_prob: operation not allowed\n"); - delete_prob(lp); - create_prob(lp); - return; -} - -/*********************************************************************** -* NAME -* -* glp_delete_prob - delete problem object -* -* SYNOPSIS -* -* void glp_delete_prob(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_delete_prob deletes the specified problem object and -* frees all the memory allocated to it. */ - -static void delete_prob(glp_prob *lp) -#if 0 /* 04/IV-2016 */ -{ lp->magic = 0x3F3F3F3F; -#else -{ -#endif - dmp_delete_pool(lp->pool); -#if 0 /* 08/III-2014 */ -#if 0 /* 17/XI-2009 */ - xfree(lp->cps); -#else - if (lp->parms != NULL) xfree(lp->parms); -#endif -#endif - xassert(lp->tree == NULL); -#if 0 - if (lp->cwa != NULL) xfree(lp->cwa); -#endif - xfree(lp->row); - xfree(lp->col); - if (lp->r_tree != NULL) avl_delete_tree(lp->r_tree); - if (lp->c_tree != NULL) avl_delete_tree(lp->c_tree); - xfree(lp->head); -#if 0 /* 08/III-2014 */ - if (lp->bfcp != NULL) xfree(lp->bfcp); -#endif - if (lp->bfd != NULL) bfd_delete_it(lp->bfd); - return; -} - -void glp_delete_prob(glp_prob *lp) -{ glp_tree *tree = lp->tree; - if (tree != NULL && tree->reason != 0) - xerror("glp_delete_prob: operation not allowed\n"); - delete_prob(lp); - xfree(lp); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prob2.c b/code/3rd_glpk/api/prob2.c deleted file mode 100644 index d352db12..00000000 --- a/code/3rd_glpk/api/prob2.c +++ /dev/null @@ -1,491 +0,0 @@ -/* prob2.c (problem retrieving routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_get_prob_name - retrieve problem name -* -* SYNOPSIS -* -* const char *glp_get_prob_name(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_prob_name returns a pointer to an internal -* buffer, which contains symbolic name of the problem. However, if the -* problem has no assigned name, the routine returns NULL. */ - -const char *glp_get_prob_name(glp_prob *lp) -{ char *name; - name = lp->name; - return name; -} - -/*********************************************************************** -* NAME -* -* glp_get_obj_name - retrieve objective function name -* -* SYNOPSIS -* -* const char *glp_get_obj_name(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_obj_name returns a pointer to an internal -* buffer, which contains a symbolic name of the objective function. -* However, if the objective function has no assigned name, the routine -* returns NULL. */ - -const char *glp_get_obj_name(glp_prob *lp) -{ char *name; - name = lp->obj; - return name; -} - -/*********************************************************************** -* NAME -* -* glp_get_obj_dir - retrieve optimization direction flag -* -* SYNOPSIS -* -* int glp_get_obj_dir(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_obj_dir returns the optimization direction flag -* (i.e. "sense" of the objective function): -* -* GLP_MIN - minimization; -* GLP_MAX - maximization. */ - -int glp_get_obj_dir(glp_prob *lp) -{ int dir = lp->dir; - return dir; -} - -/*********************************************************************** -* NAME -* -* glp_get_num_rows - retrieve number of rows -* -* SYNOPSIS -* -* int glp_get_num_rows(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_num_rows returns the current number of rows in -* the specified problem object. */ - -int glp_get_num_rows(glp_prob *lp) -{ int m = lp->m; - return m; -} - -/*********************************************************************** -* NAME -* -* glp_get_num_cols - retrieve number of columns -* -* SYNOPSIS -* -* int glp_get_num_cols(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_num_cols returns the current number of columns -* in the specified problem object. */ - -int glp_get_num_cols(glp_prob *lp) -{ int n = lp->n; - return n; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_name - retrieve row name -* -* SYNOPSIS -* -* const char *glp_get_row_name(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_name returns a pointer to an internal -* buffer, which contains symbolic name of i-th row. However, if i-th -* row has no assigned name, the routine returns NULL. */ - -const char *glp_get_row_name(glp_prob *lp, int i) -{ char *name; - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_name: i = %d; row number out of range\n", - i); - name = lp->row[i]->name; - return name; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_name - retrieve column name -* -* SYNOPSIS -* -* const char *glp_get_col_name(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_name returns a pointer to an internal -* buffer, which contains symbolic name of j-th column. However, if j-th -* column has no assigned name, the routine returns NULL. */ - -const char *glp_get_col_name(glp_prob *lp, int j) -{ char *name; - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_name: j = %d; column number out of range\n" - , j); - name = lp->col[j]->name; - return name; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_type - retrieve row type -* -* SYNOPSIS -* -* int glp_get_row_type(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_type returns the type of i-th row, i.e. the -* type of corresponding auxiliary variable, as follows: -* -* GLP_FR - free (unbounded) variable; -* GLP_LO - variable with lower bound; -* GLP_UP - variable with upper bound; -* GLP_DB - double-bounded variable; -* GLP_FX - fixed variable. */ - -int glp_get_row_type(glp_prob *lp, int i) -{ if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_type: i = %d; row number out of range\n", - i); - return lp->row[i]->type; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_lb - retrieve row lower bound -* -* SYNOPSIS -* -* double glp_get_row_lb(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_lb returns the lower bound of i-th row, i.e. -* the lower bound of corresponding auxiliary variable. However, if the -* row has no lower bound, the routine returns -DBL_MAX. */ - -double glp_get_row_lb(glp_prob *lp, int i) -{ double lb; - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_lb: i = %d; row number out of range\n", i); - switch (lp->row[i]->type) - { case GLP_FR: - case GLP_UP: - lb = -DBL_MAX; break; - case GLP_LO: - case GLP_DB: - case GLP_FX: - lb = lp->row[i]->lb; break; - default: - xassert(lp != lp); - } - return lb; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_ub - retrieve row upper bound -* -* SYNOPSIS -* -* double glp_get_row_ub(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_ub returns the upper bound of i-th row, i.e. -* the upper bound of corresponding auxiliary variable. However, if the -* row has no upper bound, the routine returns +DBL_MAX. */ - -double glp_get_row_ub(glp_prob *lp, int i) -{ double ub; - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_ub: i = %d; row number out of range\n", i); - switch (lp->row[i]->type) - { case GLP_FR: - case GLP_LO: - ub = +DBL_MAX; break; - case GLP_UP: - case GLP_DB: - case GLP_FX: - ub = lp->row[i]->ub; break; - default: - xassert(lp != lp); - } - return ub; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_type - retrieve column type -* -* SYNOPSIS -* -* int glp_get_col_type(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_type returns the type of j-th column, i.e. -* the type of corresponding structural variable, as follows: -* -* GLP_FR - free (unbounded) variable; -* GLP_LO - variable with lower bound; -* GLP_UP - variable with upper bound; -* GLP_DB - double-bounded variable; -* GLP_FX - fixed variable. */ - -int glp_get_col_type(glp_prob *lp, int j) -{ if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_type: j = %d; column number out of range\n" - , j); - return lp->col[j]->type; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_lb - retrieve column lower bound -* -* SYNOPSIS -* -* double glp_get_col_lb(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_lb returns the lower bound of j-th column, -* i.e. the lower bound of corresponding structural variable. However, -* if the column has no lower bound, the routine returns -DBL_MAX. */ - -double glp_get_col_lb(glp_prob *lp, int j) -{ double lb; - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_lb: j = %d; column number out of range\n", - j); - switch (lp->col[j]->type) - { case GLP_FR: - case GLP_UP: - lb = -DBL_MAX; break; - case GLP_LO: - case GLP_DB: - case GLP_FX: - lb = lp->col[j]->lb; break; - default: - xassert(lp != lp); - } - return lb; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_ub - retrieve column upper bound -* -* SYNOPSIS -* -* double glp_get_col_ub(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_ub returns the upper bound of j-th column, -* i.e. the upper bound of corresponding structural variable. However, -* if the column has no upper bound, the routine returns +DBL_MAX. */ - -double glp_get_col_ub(glp_prob *lp, int j) -{ double ub; - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_ub: j = %d; column number out of range\n", - j); - switch (lp->col[j]->type) - { case GLP_FR: - case GLP_LO: - ub = +DBL_MAX; break; - case GLP_UP: - case GLP_DB: - case GLP_FX: - ub = lp->col[j]->ub; break; - default: - xassert(lp != lp); - } - return ub; -} - -/*********************************************************************** -* NAME -* -* glp_get_obj_coef - retrieve obj. coefficient or constant term -* -* SYNOPSIS -* -* double glp_get_obj_coef(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_obj_coef returns the objective coefficient at -* j-th structural variable (column) of the specified problem object. -* -* If the parameter j is zero, the routine returns the constant term -* ("shift") of the objective function. */ - -double glp_get_obj_coef(glp_prob *lp, int j) -{ if (!(0 <= j && j <= lp->n)) - xerror("glp_get_obj_coef: j = %d; column number out of range\n" - , j); - return j == 0 ? lp->c0 : lp->col[j]->coef; -} - -/*********************************************************************** -* NAME -* -* glp_get_num_nz - retrieve number of constraint coefficients -* -* SYNOPSIS -* -* int glp_get_num_nz(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_num_nz returns the number of (non-zero) elements -* in the constraint matrix of the specified problem object. */ - -int glp_get_num_nz(glp_prob *lp) -{ int nnz = lp->nnz; - return nnz; -} - -/*********************************************************************** -* NAME -* -* glp_get_mat_row - retrieve row of the constraint matrix -* -* SYNOPSIS -* -* int glp_get_mat_row(glp_prob *lp, int i, int ind[], double val[]); -* -* DESCRIPTION -* -* The routine glp_get_mat_row scans (non-zero) elements of i-th row -* of the constraint matrix of the specified problem object and stores -* their column indices and numeric values to locations ind[1], ..., -* ind[len] and val[1], ..., val[len], respectively, where 0 <= len <= n -* is the number of elements in i-th row, n is the number of columns. -* -* The parameter ind and/or val can be specified as NULL, in which case -* corresponding information is not stored. -* -* RETURNS -* -* The routine glp_get_mat_row returns the length len, i.e. the number -* of (non-zero) elements in i-th row. */ - -int glp_get_mat_row(glp_prob *lp, int i, int ind[], double val[]) -{ GLPAIJ *aij; - int len; - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_mat_row: i = %d; row number out of range\n", - i); - len = 0; - for (aij = lp->row[i]->ptr; aij != NULL; aij = aij->r_next) - { len++; - if (ind != NULL) ind[len] = aij->col->j; - if (val != NULL) val[len] = aij->val; - } - xassert(len <= lp->n); - return len; -} - -/*********************************************************************** -* NAME -* -* glp_get_mat_col - retrieve column of the constraint matrix -* -* SYNOPSIS -* -* int glp_get_mat_col(glp_prob *lp, int j, int ind[], double val[]); -* -* DESCRIPTION -* -* The routine glp_get_mat_col scans (non-zero) elements of j-th column -* of the constraint matrix of the specified problem object and stores -* their row indices and numeric values to locations ind[1], ..., -* ind[len] and val[1], ..., val[len], respectively, where 0 <= len <= m -* is the number of elements in j-th column, m is the number of rows. -* -* The parameter ind or/and val can be specified as NULL, in which case -* corresponding information is not stored. -* -* RETURNS -* -* The routine glp_get_mat_col returns the length len, i.e. the number -* of (non-zero) elements in j-th column. */ - -int glp_get_mat_col(glp_prob *lp, int j, int ind[], double val[]) -{ GLPAIJ *aij; - int len; - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_mat_col: j = %d; column number out of range\n", - j); - len = 0; - for (aij = lp->col[j]->ptr; aij != NULL; aij = aij->c_next) - { len++; - if (ind != NULL) ind[len] = aij->row->i; - if (val != NULL) val[len] = aij->val; - } - xassert(len <= lp->m); - return len; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prob3.c b/code/3rd_glpk/api/prob3.c deleted file mode 100644 index d7edbd33..00000000 --- a/code/3rd_glpk/api/prob3.c +++ /dev/null @@ -1,166 +0,0 @@ -/* prob3.c (problem row/column searching routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_create_index - create the name index -* -* SYNOPSIS -* -* void glp_create_index(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_create_index creates the name index for the -* specified problem object. The name index is an auxiliary data -* structure, which is intended to quickly (i.e. for logarithmic time) -* find rows and columns by their names. -* -* This routine can be called at any time. If the name index already -* exists, the routine does nothing. */ - -void glp_create_index(glp_prob *lp) -{ GLPROW *row; - GLPCOL *col; - int i, j; - /* create row name index */ - if (lp->r_tree == NULL) - { lp->r_tree = avl_create_tree(avl_strcmp, NULL); - for (i = 1; i <= lp->m; i++) - { row = lp->row[i]; - xassert(row->node == NULL); - if (row->name != NULL) - { row->node = avl_insert_node(lp->r_tree, row->name); - avl_set_node_link(row->node, row); - } - } - } - /* create column name index */ - if (lp->c_tree == NULL) - { lp->c_tree = avl_create_tree(avl_strcmp, NULL); - for (j = 1; j <= lp->n; j++) - { col = lp->col[j]; - xassert(col->node == NULL); - if (col->name != NULL) - { col->node = avl_insert_node(lp->c_tree, col->name); - avl_set_node_link(col->node, col); - } - } - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_find_row - find row by its name -* -* SYNOPSIS -* -* int glp_find_row(glp_prob *lp, const char *name); -* -* RETURNS -* -* The routine glp_find_row returns the ordinal number of a row, -* which is assigned (by the routine glp_set_row_name) the specified -* symbolic name. If no such row exists, the routine returns 0. */ - -int glp_find_row(glp_prob *lp, const char *name) -{ AVLNODE *node; - int i = 0; - if (lp->r_tree == NULL) - xerror("glp_find_row: row name index does not exist\n"); - if (!(name == NULL || name[0] == '\0' || strlen(name) > 255)) - { node = avl_find_node(lp->r_tree, name); - if (node != NULL) - i = ((GLPROW *)avl_get_node_link(node))->i; - } - return i; -} - -/*********************************************************************** -* NAME -* -* glp_find_col - find column by its name -* -* SYNOPSIS -* -* int glp_find_col(glp_prob *lp, const char *name); -* -* RETURNS -* -* The routine glp_find_col returns the ordinal number of a column, -* which is assigned (by the routine glp_set_col_name) the specified -* symbolic name. If no such column exists, the routine returns 0. */ - -int glp_find_col(glp_prob *lp, const char *name) -{ AVLNODE *node; - int j = 0; - if (lp->c_tree == NULL) - xerror("glp_find_col: column name index does not exist\n"); - if (!(name == NULL || name[0] == '\0' || strlen(name) > 255)) - { node = avl_find_node(lp->c_tree, name); - if (node != NULL) - j = ((GLPCOL *)avl_get_node_link(node))->j; - } - return j; -} - -/*********************************************************************** -* NAME -* -* glp_delete_index - delete the name index -* -* SYNOPSIS -* -* void glp_delete_index(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_delete_index deletes the name index previously -* created by the routine glp_create_index and frees the memory -* allocated to this auxiliary data structure. -* -* This routine can be called at any time. If the name index does not -* exist, the routine does nothing. */ - -void glp_delete_index(glp_prob *lp) -{ int i, j; - /* delete row name index */ - if (lp->r_tree != NULL) - { for (i = 1; i <= lp->m; i++) lp->row[i]->node = NULL; - avl_delete_tree(lp->r_tree), lp->r_tree = NULL; - } - /* delete column name index */ - if (lp->c_tree != NULL) - { for (j = 1; j <= lp->n; j++) lp->col[j]->node = NULL; - avl_delete_tree(lp->c_tree), lp->c_tree = NULL; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prob4.c b/code/3rd_glpk/api/prob4.c deleted file mode 100644 index 8c2b5ae5..00000000 --- a/code/3rd_glpk/api/prob4.c +++ /dev/null @@ -1,156 +0,0 @@ -/* prob4.c (problem scaling routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_set_rii - set (change) row scale factor -* -* SYNOPSIS -* -* void glp_set_rii(glp_prob *lp, int i, double rii); -* -* DESCRIPTION -* -* The routine glp_set_rii sets (changes) the scale factor r[i,i] for -* i-th row of the specified problem object. */ - -void glp_set_rii(glp_prob *lp, int i, double rii) -{ if (!(1 <= i && i <= lp->m)) - xerror("glp_set_rii: i = %d; row number out of range\n", i); - if (rii <= 0.0) - xerror("glp_set_rii: i = %d; rii = %g; invalid scale factor\n", - i, rii); - if (lp->valid && lp->row[i]->rii != rii) - { GLPAIJ *aij; - for (aij = lp->row[i]->ptr; aij != NULL; aij = aij->r_next) - { if (aij->col->stat == GLP_BS) - { /* invalidate the basis factorization */ - lp->valid = 0; - break; - } - } - } - lp->row[i]->rii = rii; - return; -} - -/*********************************************************************** -* NAME -* -* glp_set sjj - set (change) column scale factor -* -* SYNOPSIS -* -* void glp_set_sjj(glp_prob *lp, int j, double sjj); -* -* DESCRIPTION -* -* The routine glp_set_sjj sets (changes) the scale factor s[j,j] for -* j-th column of the specified problem object. */ - -void glp_set_sjj(glp_prob *lp, int j, double sjj) -{ if (!(1 <= j && j <= lp->n)) - xerror("glp_set_sjj: j = %d; column number out of range\n", j); - if (sjj <= 0.0) - xerror("glp_set_sjj: j = %d; sjj = %g; invalid scale factor\n", - j, sjj); - if (lp->valid && lp->col[j]->sjj != sjj && lp->col[j]->stat == - GLP_BS) - { /* invalidate the basis factorization */ - lp->valid = 0; - } - lp->col[j]->sjj = sjj; - return; -} - -/*********************************************************************** -* NAME -* -* glp_get_rii - retrieve row scale factor -* -* SYNOPSIS -* -* double glp_get_rii(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_rii returns current scale factor r[i,i] for i-th -* row of the specified problem object. */ - -double glp_get_rii(glp_prob *lp, int i) -{ if (!(1 <= i && i <= lp->m)) - xerror("glp_get_rii: i = %d; row number out of range\n", i); - return lp->row[i]->rii; -} - -/*********************************************************************** -* NAME -* -* glp_get_sjj - retrieve column scale factor -* -* SYNOPSIS -* -* double glp_get_sjj(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_sjj returns current scale factor s[j,j] for j-th -* column of the specified problem object. */ - -double glp_get_sjj(glp_prob *lp, int j) -{ if (!(1 <= j && j <= lp->n)) - xerror("glp_get_sjj: j = %d; column number out of range\n", j); - return lp->col[j]->sjj; -} - -/*********************************************************************** -* NAME -* -* glp_unscale_prob - unscale problem data -* -* SYNOPSIS -* -* void glp_unscale_prob(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_unscale_prob performs unscaling of problem data for -* the specified problem object. -* -* "Unscaling" means replacing the current scaling matrices R and S by -* unity matrices that cancels the scaling effect. */ - -void glp_unscale_prob(glp_prob *lp) -{ int m = glp_get_num_rows(lp); - int n = glp_get_num_cols(lp); - int i, j; - for (i = 1; i <= m; i++) glp_set_rii(lp, i, 1.0); - for (j = 1; j <= n; j++) glp_set_sjj(lp, j, 1.0); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prob5.c b/code/3rd_glpk/api/prob5.c deleted file mode 100644 index 1c1d3160..00000000 --- a/code/3rd_glpk/api/prob5.c +++ /dev/null @@ -1,168 +0,0 @@ -/* prob5.c (LP problem basis constructing routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_set_row_stat - set (change) row status -* -* SYNOPSIS -* -* void glp_set_row_stat(glp_prob *lp, int i, int stat); -* -* DESCRIPTION -* -* The routine glp_set_row_stat sets (changes) status of the auxiliary -* variable associated with i-th row. -* -* The new status of the auxiliary variable should be specified by the -* parameter stat as follows: -* -* GLP_BS - basic variable; -* GLP_NL - non-basic variable; -* GLP_NU - non-basic variable on its upper bound; if the variable is -* not double-bounded, this means the same as GLP_NL (only in -* case of this routine); -* GLP_NF - the same as GLP_NL (only in case of this routine); -* GLP_NS - the same as GLP_NL (only in case of this routine). */ - -void glp_set_row_stat(glp_prob *lp, int i, int stat) -{ GLPROW *row; - if (!(1 <= i && i <= lp->m)) - xerror("glp_set_row_stat: i = %d; row number out of range\n", - i); - if (!(stat == GLP_BS || stat == GLP_NL || stat == GLP_NU || - stat == GLP_NF || stat == GLP_NS)) - xerror("glp_set_row_stat: i = %d; stat = %d; invalid status\n", - i, stat); - row = lp->row[i]; - if (stat != GLP_BS) - { switch (row->type) - { case GLP_FR: stat = GLP_NF; break; - case GLP_LO: stat = GLP_NL; break; - case GLP_UP: stat = GLP_NU; break; - case GLP_DB: if (stat != GLP_NU) stat = GLP_NL; break; - case GLP_FX: stat = GLP_NS; break; - default: xassert(row != row); - } - } - if (row->stat == GLP_BS && stat != GLP_BS || - row->stat != GLP_BS && stat == GLP_BS) - { /* invalidate the basis factorization */ - lp->valid = 0; - } - row->stat = stat; - return; -} - -/*********************************************************************** -* NAME -* -* glp_set_col_stat - set (change) column status -* -* SYNOPSIS -* -* void glp_set_col_stat(glp_prob *lp, int j, int stat); -* -* DESCRIPTION -* -* The routine glp_set_col_stat sets (changes) status of the structural -* variable associated with j-th column. -* -* The new status of the structural variable should be specified by the -* parameter stat as follows: -* -* GLP_BS - basic variable; -* GLP_NL - non-basic variable; -* GLP_NU - non-basic variable on its upper bound; if the variable is -* not double-bounded, this means the same as GLP_NL (only in -* case of this routine); -* GLP_NF - the same as GLP_NL (only in case of this routine); -* GLP_NS - the same as GLP_NL (only in case of this routine). */ - -void glp_set_col_stat(glp_prob *lp, int j, int stat) -{ GLPCOL *col; - if (!(1 <= j && j <= lp->n)) - xerror("glp_set_col_stat: j = %d; column number out of range\n" - , j); - if (!(stat == GLP_BS || stat == GLP_NL || stat == GLP_NU || - stat == GLP_NF || stat == GLP_NS)) - xerror("glp_set_col_stat: j = %d; stat = %d; invalid status\n", - j, stat); - col = lp->col[j]; - if (stat != GLP_BS) - { switch (col->type) - { case GLP_FR: stat = GLP_NF; break; - case GLP_LO: stat = GLP_NL; break; - case GLP_UP: stat = GLP_NU; break; - case GLP_DB: if (stat != GLP_NU) stat = GLP_NL; break; - case GLP_FX: stat = GLP_NS; break; - default: xassert(col != col); - } - } - if (col->stat == GLP_BS && stat != GLP_BS || - col->stat != GLP_BS && stat == GLP_BS) - { /* invalidate the basis factorization */ - lp->valid = 0; - } - col->stat = stat; - return; -} - -/*********************************************************************** -* NAME -* -* glp_std_basis - construct standard initial LP basis -* -* SYNOPSIS -* -* void glp_std_basis(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_std_basis builds the "standard" (trivial) initial -* basis for the specified problem object. -* -* In the "standard" basis all auxiliary variables are basic, and all -* structural variables are non-basic. */ - -void glp_std_basis(glp_prob *lp) -{ int i, j; - /* make all auxiliary variables basic */ - for (i = 1; i <= lp->m; i++) - glp_set_row_stat(lp, i, GLP_BS); - /* make all structural variables non-basic */ - for (j = 1; j <= lp->n; j++) - { GLPCOL *col = lp->col[j]; - if (col->type == GLP_DB && fabs(col->lb) > fabs(col->ub)) - glp_set_col_stat(lp, j, GLP_NU); - else - glp_set_col_stat(lp, j, GLP_NL); - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prrngs.c b/code/3rd_glpk/api/prrngs.c deleted file mode 100644 index 41a141ff..00000000 --- a/code/3rd_glpk/api/prrngs.c +++ /dev/null @@ -1,302 +0,0 @@ -/* prrngs.c (print sensitivity analysis report) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -#define xfprintf glp_format - -static char *format(char buf[13+1], double x) -{ /* format floating-point number in MPS/360-like style */ - if (x == -DBL_MAX) - strcpy(buf, " -Inf"); - else if (x == +DBL_MAX) - strcpy(buf, " +Inf"); - else if (fabs(x) <= 999999.99998) - { sprintf(buf, "%13.5f", x); -#if 1 - if (strcmp(buf, " 0.00000") == 0 || - strcmp(buf, " -0.00000") == 0) - strcpy(buf, " . "); - else if (memcmp(buf, " 0.", 8) == 0) - memcpy(buf, " .", 8); - else if (memcmp(buf, " -0.", 8) == 0) - memcpy(buf, " -.", 8); -#endif - } - else - sprintf(buf, "%13.6g", x); - return buf; -} - -int glp_print_ranges(glp_prob *P, int len, const int list[], - int flags, const char *fname) -{ /* print sensitivity analysis report */ - glp_file *fp = NULL; - GLPROW *row; - GLPCOL *col; - int m, n, pass, k, t, numb, type, stat, var1, var2, count, page, - ret; - double lb, ub, slack, coef, prim, dual, value1, value2, coef1, - coef2, obj1, obj2; - const char *name, *limit; - char buf[13+1]; - /* sanity checks */ -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_print_ranges: P = %p; invalid problem object\n", - P); -#endif - m = P->m, n = P->n; - if (len < 0) - xerror("glp_print_ranges: len = %d; invalid list length\n", - len); - if (len > 0) - { if (list == NULL) - xerror("glp_print_ranges: list = %p: invalid parameter\n", - list); - for (t = 1; t <= len; t++) - { k = list[t]; - if (!(1 <= k && k <= m+n)) - xerror("glp_print_ranges: list[%d] = %d; row/column numb" - "er out of range\n", t, k); - } - } - if (flags != 0) - xerror("glp_print_ranges: flags = %d; invalid parameter\n", - flags); - if (fname == NULL) - xerror("glp_print_ranges: fname = %p; invalid parameter\n", - fname); - if (glp_get_status(P) != GLP_OPT) - { xprintf("glp_print_ranges: optimal basic solution required\n"); - ret = 1; - goto done; - } - if (!glp_bf_exists(P)) - { xprintf("glp_print_ranges: basis factorization required\n"); - ret = 2; - goto done; - } - /* start reporting */ - xprintf("Write sensitivity analysis report to '%s'...\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 3; - goto done; - } - page = count = 0; - for (pass = 1; pass <= 2; pass++) - for (t = 1; t <= (len == 0 ? m+n : len); t++) - { if (t == 1) count = 0; - k = (len == 0 ? t : list[t]); - if (pass == 1 && k > m || pass == 2 && k <= m) - continue; - if (count == 0) - { xfprintf(fp, "GLPK %-4s - SENSITIVITY ANALYSIS REPORT%73sPa" - "ge%4d\n", glp_version(), "", ++page); - xfprintf(fp, "\n"); - xfprintf(fp, "%-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name); - xfprintf(fp, "%-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->obj_val, - P->dir == GLP_MIN ? "MINimum" : - P->dir == GLP_MAX ? "MAXimum" : "???"); - xfprintf(fp, "\n"); - xfprintf(fp, "%6s %-12s %2s %13s %13s %13s %13s %13s %13s " - "%s\n", "No.", pass == 1 ? "Row name" : "Column name", - "St", "Activity", pass == 1 ? "Slack" : "Obj coef", - "Lower bound", "Activity", "Obj coef", "Obj value at", - "Limiting"); - xfprintf(fp, "%6s %-12s %2s %13s %13s %13s %13s %13s %13s " - "%s\n", "", "", "", "", "Marginal", "Upper bound", - "range", "range", "break point", "variable"); - xfprintf(fp, "------ ------------ -- ------------- --------" - "----- ------------- ------------- ------------- ------" - "------- ------------\n"); - } - if (pass == 1) - { numb = k; - xassert(1 <= numb && numb <= m); - row = P->row[numb]; - name = row->name; - type = row->type; - lb = glp_get_row_lb(P, numb); - ub = glp_get_row_ub(P, numb); - coef = 0.0; - stat = row->stat; - prim = row->prim; - if (type == GLP_FR) - slack = - prim; - else if (type == GLP_LO) - slack = lb - prim; - else if (type == GLP_UP || type == GLP_DB || type == GLP_FX) - slack = ub - prim; - dual = row->dual; - } - else - { numb = k - m; - xassert(1 <= numb && numb <= n); - col = P->col[numb]; - name = col->name; - lb = glp_get_col_lb(P, numb); - ub = glp_get_col_ub(P, numb); - coef = col->coef; - stat = col->stat; - prim = col->prim; - slack = 0.0; - dual = col->dual; - } - if (stat != GLP_BS) - { glp_analyze_bound(P, k, &value1, &var1, &value2, &var2); - if (stat == GLP_NF) - coef1 = coef2 = coef; - else if (stat == GLP_NS) - coef1 = -DBL_MAX, coef2 = +DBL_MAX; - else if (stat == GLP_NL && P->dir == GLP_MIN || - stat == GLP_NU && P->dir == GLP_MAX) - coef1 = coef - dual, coef2 = +DBL_MAX; - else - coef1 = -DBL_MAX, coef2 = coef - dual; - if (value1 == -DBL_MAX) - { if (dual < -1e-9) - obj1 = +DBL_MAX; - else if (dual > +1e-9) - obj1 = -DBL_MAX; - else - obj1 = P->obj_val; - } - else - obj1 = P->obj_val + dual * (value1 - prim); - if (value2 == +DBL_MAX) - { if (dual < -1e-9) - obj2 = -DBL_MAX; - else if (dual > +1e-9) - obj2 = +DBL_MAX; - else - obj2 = P->obj_val; - } - else - obj2 = P->obj_val + dual * (value2 - prim); - } - else - { glp_analyze_coef(P, k, &coef1, &var1, &value1, &coef2, - &var2, &value2); - if (coef1 == -DBL_MAX) - { if (prim < -1e-9) - obj1 = +DBL_MAX; - else if (prim > +1e-9) - obj1 = -DBL_MAX; - else - obj1 = P->obj_val; - } - else - obj1 = P->obj_val + (coef1 - coef) * prim; - if (coef2 == +DBL_MAX) - { if (prim < -1e-9) - obj2 = -DBL_MAX; - else if (prim > +1e-9) - obj2 = +DBL_MAX; - else - obj2 = P->obj_val; - } - else - obj2 = P->obj_val + (coef2 - coef) * prim; - } - /*** first line ***/ - /* row/column number */ - xfprintf(fp, "%6d", numb); - /* row/column name */ - xfprintf(fp, " %-12.12s", name == NULL ? "" : name); - if (name != NULL && strlen(name) > 12) - xfprintf(fp, "%s\n%6s %12s", name+12, "", ""); - /* row/column status */ - xfprintf(fp, " %2s", - stat == GLP_BS ? "BS" : stat == GLP_NL ? "NL" : - stat == GLP_NU ? "NU" : stat == GLP_NF ? "NF" : - stat == GLP_NS ? "NS" : "??"); - /* row/column activity */ - xfprintf(fp, " %s", format(buf, prim)); - /* row slack, column objective coefficient */ - xfprintf(fp, " %s", format(buf, k <= m ? slack : coef)); - /* row/column lower bound */ - xfprintf(fp, " %s", format(buf, lb)); - /* row/column activity range */ - xfprintf(fp, " %s", format(buf, value1)); - /* row/column objective coefficient range */ - xfprintf(fp, " %s", format(buf, coef1)); - /* objective value at break point */ - xfprintf(fp, " %s", format(buf, obj1)); - /* limiting variable name */ - if (var1 != 0) - { if (var1 <= m) - limit = glp_get_row_name(P, var1); - else - limit = glp_get_col_name(P, var1 - m); - if (limit != NULL) - xfprintf(fp, " %s", limit); - } - xfprintf(fp, "\n"); - /*** second line ***/ - xfprintf(fp, "%6s %-12s %2s %13s", "", "", "", ""); - /* row/column reduced cost */ - xfprintf(fp, " %s", format(buf, dual)); - /* row/column upper bound */ - xfprintf(fp, " %s", format(buf, ub)); - /* row/column activity range */ - xfprintf(fp, " %s", format(buf, value2)); - /* row/column objective coefficient range */ - xfprintf(fp, " %s", format(buf, coef2)); - /* objective value at break point */ - xfprintf(fp, " %s", format(buf, obj2)); - /* limiting variable name */ - if (var2 != 0) - { if (var2 <= m) - limit = glp_get_row_name(P, var2); - else - limit = glp_get_col_name(P, var2 - m); - if (limit != NULL) - xfprintf(fp, " %s", limit); - } - xfprintf(fp, "\n"); - xfprintf(fp, "\n"); - /* print 10 items per page */ - count = (count + 1) % 10; - } - xfprintf(fp, "End of report\n"); -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 4; - goto done; - } - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/prsol.c b/code/3rd_glpk/api/prsol.c deleted file mode 100644 index d785dc2e..00000000 --- a/code/3rd_glpk/api/prsol.c +++ /dev/null @@ -1,202 +0,0 @@ -/* prsol.c (write basic solution in printable format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -#define xfprintf glp_format - -int glp_print_sol(glp_prob *P, const char *fname) -{ /* write basic solution in printable format */ - glp_file *fp; - GLPROW *row; - GLPCOL *col; - int i, j, t, ae_ind, re_ind, ret; - double ae_max, re_max; - xprintf("Writing basic solution to '%s'...\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "%-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name); - xfprintf(fp, "%-12s%d\n", "Rows:", P->m); - xfprintf(fp, "%-12s%d\n", "Columns:", P->n); - xfprintf(fp, "%-12s%d\n", "Non-zeros:", P->nnz); - t = glp_get_status(P); - xfprintf(fp, "%-12s%s\n", "Status:", - t == GLP_OPT ? "OPTIMAL" : - t == GLP_FEAS ? "FEASIBLE" : - t == GLP_INFEAS ? "INFEASIBLE (INTERMEDIATE)" : - t == GLP_NOFEAS ? "INFEASIBLE (FINAL)" : - t == GLP_UNBND ? "UNBOUNDED" : - t == GLP_UNDEF ? "UNDEFINED" : "???"); - xfprintf(fp, "%-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->obj_val, - P->dir == GLP_MIN ? "MINimum" : - P->dir == GLP_MAX ? "MAXimum" : "???"); - xfprintf(fp, "\n"); - xfprintf(fp, " No. Row name St Activity Lower bound " - " Upper bound Marginal\n"); - xfprintf(fp, "------ ------------ -- ------------- ------------- " - "------------- -------------\n"); - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - xfprintf(fp, "%6d ", i); - if (row->name == NULL || strlen(row->name) <= 12) - xfprintf(fp, "%-12s ", row->name == NULL ? "" : row->name); - else - xfprintf(fp, "%s\n%20s", row->name, ""); - xfprintf(fp, "%s ", - row->stat == GLP_BS ? "B " : - row->stat == GLP_NL ? "NL" : - row->stat == GLP_NU ? "NU" : - row->stat == GLP_NF ? "NF" : - row->stat == GLP_NS ? "NS" : "??"); - xfprintf(fp, "%13.6g ", - fabs(row->prim) <= 1e-9 ? 0.0 : row->prim); - if (row->type == GLP_LO || row->type == GLP_DB || - row->type == GLP_FX) - xfprintf(fp, "%13.6g ", row->lb); - else - xfprintf(fp, "%13s ", ""); - if (row->type == GLP_UP || row->type == GLP_DB) - xfprintf(fp, "%13.6g ", row->ub); - else - xfprintf(fp, "%13s ", row->type == GLP_FX ? "=" : ""); - if (row->stat != GLP_BS) - { if (fabs(row->dual) <= 1e-9) - xfprintf(fp, "%13s", "< eps"); - else - xfprintf(fp, "%13.6g ", row->dual); - } - xfprintf(fp, "\n"); - } - xfprintf(fp, "\n"); - xfprintf(fp, " No. Column name St Activity Lower bound " - " Upper bound Marginal\n"); - xfprintf(fp, "------ ------------ -- ------------- ------------- " - "------------- -------------\n"); - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - xfprintf(fp, "%6d ", j); - if (col->name == NULL || strlen(col->name) <= 12) - xfprintf(fp, "%-12s ", col->name == NULL ? "" : col->name); - else - xfprintf(fp, "%s\n%20s", col->name, ""); - xfprintf(fp, "%s ", - col->stat == GLP_BS ? "B " : - col->stat == GLP_NL ? "NL" : - col->stat == GLP_NU ? "NU" : - col->stat == GLP_NF ? "NF" : - col->stat == GLP_NS ? "NS" : "??"); - xfprintf(fp, "%13.6g ", - fabs(col->prim) <= 1e-9 ? 0.0 : col->prim); - if (col->type == GLP_LO || col->type == GLP_DB || - col->type == GLP_FX) - xfprintf(fp, "%13.6g ", col->lb); - else - xfprintf(fp, "%13s ", ""); - if (col->type == GLP_UP || col->type == GLP_DB) - xfprintf(fp, "%13.6g ", col->ub); - else - xfprintf(fp, "%13s ", col->type == GLP_FX ? "=" : ""); - if (col->stat != GLP_BS) - { if (fabs(col->dual) <= 1e-9) - xfprintf(fp, "%13s", "< eps"); - else - xfprintf(fp, "%13.6g ", col->dual); - } - xfprintf(fp, "\n"); - } - xfprintf(fp, "\n"); - xfprintf(fp, "Karush-Kuhn-Tucker optimality conditions:\n"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_SOL, GLP_KKT_PE, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.PE: max.abs.err = %.2e on row %d\n", - ae_max, ae_ind); - xfprintf(fp, " max.rel.err = %.2e on row %d\n", - re_max, re_ind); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "PRIMAL SOLUTION IS WRONG"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_SOL, GLP_KKT_PB, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.PB: max.abs.err = %.2e on %s %d\n", - ae_max, ae_ind <= P->m ? "row" : "column", - ae_ind <= P->m ? ae_ind : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on %s %d\n", - re_max, re_ind <= P->m ? "row" : "column", - re_ind <= P->m ? re_ind : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "PRIMAL SOLUTION IS INFEASIBL" - "E"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_SOL, GLP_KKT_DE, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.DE: max.abs.err = %.2e on column %d\n", - ae_max, ae_ind == 0 ? 0 : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on column %d\n", - re_max, re_ind == 0 ? 0 : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "DUAL SOLUTION IS WRONG"); - xfprintf(fp, "\n"); - glp_check_kkt(P, GLP_SOL, GLP_KKT_DB, &ae_max, &ae_ind, &re_max, - &re_ind); - xfprintf(fp, "KKT.DB: max.abs.err = %.2e on %s %d\n", - ae_max, ae_ind <= P->m ? "row" : "column", - ae_ind <= P->m ? ae_ind : ae_ind - P->m); - xfprintf(fp, " max.rel.err = %.2e on %s %d\n", - re_max, re_ind <= P->m ? "row" : "column", - re_ind <= P->m ? re_ind : re_ind - P->m); - xfprintf(fp, "%8s%s\n", "", - re_max <= 1e-9 ? "High quality" : - re_max <= 1e-6 ? "Medium quality" : - re_max <= 1e-3 ? "Low quality" : "DUAL SOLUTION IS INFEASIBLE") - ; - xfprintf(fp, "\n"); - xfprintf(fp, "End of output\n"); -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdasn.c b/code/3rd_glpk/api/rdasn.c deleted file mode 100644 index 05dcb9fc..00000000 --- a/code/3rd_glpk/api/rdasn.c +++ /dev/null @@ -1,164 +0,0 @@ -/* rdasn.c (read assignment problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "glpk.h" -#include "misc.h" - -#define error dmx_error -#define warning dmx_warning -#define read_char dmx_read_char -#define read_designator dmx_read_designator -#define read_field dmx_read_field -#define end_of_line dmx_end_of_line -#define check_int dmx_check_int - -/*********************************************************************** -* NAME -* -* glp_read_asnprob - read assignment problem data in DIMACS format -* -* SYNOPSIS -* -* int glp_read_asnprob(glp_graph *G, int v_set, int a_cost, -* const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_asnprob reads assignment problem data in DIMACS -* format from a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_asnprob(glp_graph *G, int v_set, int a_cost, const char - *fname) -{ DMX _csa, *csa = &_csa; - glp_vertex *v; - glp_arc *a; - int nv, na, n1, i, j, k, ret = 0; - double cost; - char *flag = NULL; - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_read_asnprob: v_set = %d; invalid offset\n", - v_set); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_read_asnprob: a_cost = %d; invalid offset\n", - a_cost); - glp_erase_graph(G, G->v_size, G->a_size); - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->fname = fname; - csa->fp = NULL; - csa->count = 0; - csa->c = '\n'; - csa->field[0] = '\0'; - csa->empty = csa->nonint = 0; - xprintf("Reading assignment problem data from '%s'...\n", fname); - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - longjmp(csa->jump, 1); - } - /* read problem line */ - read_designator(csa); - if (strcmp(csa->field, "p") != 0) - error(csa, "problem line missing or invalid"); - read_field(csa); - if (strcmp(csa->field, "asn") != 0) - error(csa, "wrong problem designator; 'asn' expected"); - read_field(csa); - if (!(str2int(csa->field, &nv) == 0 && nv >= 0)) - error(csa, "number of nodes missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &na) == 0 && na >= 0)) - error(csa, "number of arcs missing or invalid"); - if (nv > 0) glp_add_vertices(G, nv); - end_of_line(csa); - /* read node descriptor lines */ - flag = xcalloc(1+nv, sizeof(char)); - memset(&flag[1], 0, nv * sizeof(char)); - n1 = 0; - for (;;) - { read_designator(csa); - if (strcmp(csa->field, "n") != 0) break; - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "node number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "node number %d out of range", i); - if (flag[i]) - error(csa, "duplicate descriptor of node %d", i); - flag[i] = 1, n1++; - end_of_line(csa); - } - xprintf( - "Assignment problem has %d + %d = %d node%s and %d arc%s\n", - n1, nv - n1, nv, nv == 1 ? "" : "s", na, na == 1 ? "" : "s"); - if (v_set >= 0) - { for (i = 1; i <= nv; i++) - { v = G->v[i]; - k = (flag[i] ? 0 : 1); - memcpy((char *)v->data + v_set, &k, sizeof(int)); - } - } - /* read arc descriptor lines */ - for (k = 1; k <= na; k++) - { if (k > 1) read_designator(csa); - if (strcmp(csa->field, "a") != 0) - error(csa, "wrong line designator; 'a' expected"); - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "starting node number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "starting node number %d out of range", i); - if (!flag[i]) - error(csa, "node %d cannot be a starting node", i); - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "ending node number missing or invalid"); - if (!(1 <= j && j <= nv)) - error(csa, "ending node number %d out of range", j); - if (flag[j]) - error(csa, "node %d cannot be an ending node", j); - read_field(csa); - if (str2num(csa->field, &cost) != 0) - error(csa, "arc cost missing or invalid"); - check_int(csa, cost); - a = glp_add_arc(G, i, j); - if (a_cost >= 0) - memcpy((char *)a->data + a_cost, &cost, sizeof(double)); - end_of_line(csa); - } - xprintf("%d lines were read\n", csa->count); -done: if (ret) glp_erase_graph(G, G->v_size, G->a_size); - if (csa->fp != NULL) glp_close(csa->fp); - if (flag != NULL) xfree(flag); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdcc.c b/code/3rd_glpk/api/rdcc.c deleted file mode 100644 index c63d60d8..00000000 --- a/code/3rd_glpk/api/rdcc.c +++ /dev/null @@ -1,162 +0,0 @@ -/* rdcc.c (read graph in DIMACS clique/coloring format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "glpk.h" -#include "misc.h" - -#define error dmx_error -#define warning dmx_warning -#define read_char dmx_read_char -#define read_designator dmx_read_designator -#define read_field dmx_read_field -#define end_of_line dmx_end_of_line -#define check_int dmx_check_int - -/*********************************************************************** -* NAME -* -* glp_read_ccdata - read graph in DIMACS clique/coloring format -* -* SYNOPSIS -* -* int glp_read_ccdata(glp_graph *G, int v_wgt, const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_ccdata reads an (undirected) graph in DIMACS -* clique/coloring format from a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_ccdata(glp_graph *G, int v_wgt, const char *fname) -{ DMX _csa, *csa = &_csa; - glp_vertex *v; - int i, j, k, nv, ne, ret = 0; - double w; - char *flag = NULL; - if (v_wgt >= 0 && v_wgt > G->v_size - (int)sizeof(double)) - xerror("glp_read_ccdata: v_wgt = %d; invalid offset\n", - v_wgt); - glp_erase_graph(G, G->v_size, G->a_size); - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->fname = fname; - csa->fp = NULL; - csa->count = 0; - csa->c = '\n'; - csa->field[0] = '\0'; - csa->empty = csa->nonint = 0; - xprintf("Reading graph from '%s'...\n", fname); - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - longjmp(csa->jump, 1); - } - /* read problem line */ - read_designator(csa); - if (strcmp(csa->field, "p") != 0) - error(csa, "problem line missing or invalid"); - read_field(csa); - if (strcmp(csa->field, "edge") != 0) - error(csa, "wrong problem designator; 'edge' expected"); - read_field(csa); - if (!(str2int(csa->field, &nv) == 0 && nv >= 0)) - error(csa, "number of vertices missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &ne) == 0 && ne >= 0)) - error(csa, "number of edges missing or invalid"); - xprintf("Graph has %d vert%s and %d edge%s\n", - nv, nv == 1 ? "ex" : "ices", ne, ne == 1 ? "" : "s"); - if (nv > 0) glp_add_vertices(G, nv); - end_of_line(csa); - /* read node descriptor lines */ - flag = xcalloc(1+nv, sizeof(char)); - memset(&flag[1], 0, nv * sizeof(char)); - if (v_wgt >= 0) - { w = 1.0; - for (i = 1; i <= nv; i++) - { v = G->v[i]; - memcpy((char *)v->data + v_wgt, &w, sizeof(double)); - } - } - for (;;) - { read_designator(csa); - if (strcmp(csa->field, "n") != 0) break; - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "vertex number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "vertex number %d out of range", i); - if (flag[i]) - error(csa, "duplicate descriptor of vertex %d", i); - read_field(csa); - if (str2num(csa->field, &w) != 0) - error(csa, "vertex weight missing or invalid"); - check_int(csa, w); - if (v_wgt >= 0) - { v = G->v[i]; - memcpy((char *)v->data + v_wgt, &w, sizeof(double)); - } - flag[i] = 1; - end_of_line(csa); - } - xfree(flag), flag = NULL; - /* read edge descriptor lines */ - for (k = 1; k <= ne; k++) - { if (k > 1) read_designator(csa); - if (strcmp(csa->field, "e") != 0) - error(csa, "wrong line designator; 'e' expected"); - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "first vertex number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "first vertex number %d out of range", i); - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "second vertex number missing or invalid"); - if (!(1 <= j && j <= nv)) - error(csa, "second vertex number %d out of range", j); - glp_add_arc(G, i, j); - end_of_line(csa); - } - xprintf("%d lines were read\n", csa->count); -done: if (ret) glp_erase_graph(G, G->v_size, G->a_size); - if (csa->fp != NULL) glp_close(csa->fp); - if (flag != NULL) xfree(flag); - return ret; -} - -/**********************************************************************/ - -int glp_read_graph(glp_graph *G, const char *fname) -{ return - glp_read_ccdata(G, -1, fname); -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdcnf.c b/code/3rd_glpk/api/rdcnf.c deleted file mode 100644 index acab50fe..00000000 --- a/code/3rd_glpk/api/rdcnf.c +++ /dev/null @@ -1,136 +0,0 @@ -/* rdcnf.c (read CNF-SAT problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "misc.h" -#include "prob.h" - -#define xfprintf glp_format -#define error dmx_error -#define warning dmx_warning -#define read_char dmx_read_char -#define read_designator dmx_read_designator -#define read_field dmx_read_field -#define end_of_line dmx_end_of_line -#define check_int dmx_check_int - -int glp_read_cnfsat(glp_prob *P, const char *fname) -{ /* read CNF-SAT problem data in DIMACS format */ - DMX _csa, *csa = &_csa; - int m, n, i, j, len, neg, rhs, ret = 0, *ind = NULL; - double *val = NULL; - char *map = NULL; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_read_cnfsat: P = %p; invalid problem object\n", - P); -#endif - if (fname == NULL) - xerror("glp_read_cnfsat: fname = %p; invalid parameter\n", - fname); - glp_erase_prob(P); - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->fname = fname; - csa->fp = NULL; - csa->count = 0; - csa->c = '\n'; - csa->field[0] = '\0'; - csa->empty = csa->nonint = 0; - xprintf("Reading CNF-SAT problem data from '%s'...\n", fname); - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - longjmp(csa->jump, 1); - } - /* read problem line */ - read_designator(csa); - if (strcmp(csa->field, "p") != 0) - error(csa, "problem line missing or invalid"); - read_field(csa); - if (strcmp(csa->field, "cnf") != 0) - error(csa, "wrong problem designator; 'cnf' expected\n"); - read_field(csa); - if (!(str2int(csa->field, &n) == 0 && n >= 0)) - error(csa, "number of variables missing or invalid\n"); - read_field(csa); - if (!(str2int(csa->field, &m) == 0 && m >= 0)) - error(csa, "number of clauses missing or invalid\n"); - xprintf("Instance has %d variable%s and %d clause%s\n", - n, n == 1 ? "" : "s", m, m == 1 ? "" : "s"); - end_of_line(csa); - if (m > 0) - glp_add_rows(P, m); - if (n > 0) - { glp_add_cols(P, n); - for (j = 1; j <= n; j++) - glp_set_col_kind(P, j, GLP_BV); - } - /* allocate working arrays */ - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - map = xcalloc(1+n, sizeof(char)); - for (j = 1; j <= n; j++) map[j] = 0; - /* read clauses */ - for (i = 1; i <= m; i++) - { /* read i-th clause */ - len = 0, rhs = 1; - for (;;) - { /* skip white-space characters */ - while (csa->c == ' ' || csa->c == '\n') - read_char(csa); - /* read term */ - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "variable number missing or invalid\n"); - if (j > 0) - neg = 0; - else if (j < 0) - neg = 1, j = -j, rhs--; - else - break; - if (!(1 <= j && j <= n)) - error(csa, "variable number out of range\n"); - if (map[j]) - error(csa, "duplicate variable number\n"); - len++, ind[len] = j, val[len] = (neg ? -1.0 : +1.0); - map[j] = 1; - } - glp_set_row_bnds(P, i, GLP_LO, (double)rhs, 0.0); - glp_set_mat_row(P, i, len, ind, val); - while (len > 0) map[ind[len--]] = 0; - } - xprintf("%d lines were read\n", csa->count); - /* problem data has been successfully read */ - glp_sort_matrix(P); -done: if (csa->fp != NULL) glp_close(csa->fp); - if (ind != NULL) xfree(ind); - if (val != NULL) xfree(val); - if (map != NULL) xfree(map); - if (ret) glp_erase_prob(P); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdipt.c b/code/3rd_glpk/api/rdipt.c deleted file mode 100644 index aaf8e9d4..00000000 --- a/code/3rd_glpk/api/rdipt.c +++ /dev/null @@ -1,185 +0,0 @@ -/* rdipt.c (read interior-point solution in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "env.h" -#include "misc.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_read_ipt - read interior-point solution in GLPK format -* -* SYNOPSIS -* -* int glp_read_ipt(glp_prob *P, const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_ipt reads interior-point solution from a text -* file in GLPK format. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_ipt(glp_prob *P, const char *fname) -{ DMX dmx_, *dmx = &dmx_; - int i, j, k, m, n, sst, ret = 1; - char *stat = NULL; - double obj, *prim = NULL, *dual = NULL; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_read_ipt: P = %p; invalid problem object\n", P); -#endif - if (fname == NULL) - xerror("glp_read_ipt: fname = %d; invalid parameter\n", fname); - if (setjmp(dmx->jump)) - goto done; - dmx->fname = fname; - dmx->fp = NULL; - dmx->count = 0; - dmx->c = '\n'; - dmx->field[0] = '\0'; - dmx->empty = dmx->nonint = 0; - xprintf("Reading interior-point solution from '%s'...\n", fname); - dmx->fp = glp_open(fname, "r"); - if (dmx->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* read solution line */ - dmx_read_designator(dmx); - if (strcmp(dmx->field, "s") != 0) - dmx_error(dmx, "solution line missing or invalid"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "ipt") != 0) - dmx_error(dmx, "wrong solution designator; 'ipt' expected"); - dmx_read_field(dmx); - if (!(str2int(dmx->field, &m) == 0 && m >= 0)) - dmx_error(dmx, "number of rows missing or invalid"); - if (m != P->m) - dmx_error(dmx, "number of rows mismatch"); - dmx_read_field(dmx); - if (!(str2int(dmx->field, &n) == 0 && n >= 0)) - dmx_error(dmx, "number of columns missing or invalid"); - if (n != P->n) - dmx_error(dmx, "number of columns mismatch"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "o") == 0) - sst = GLP_OPT; - else if (strcmp(dmx->field, "i") == 0) - sst = GLP_INFEAS; - else if (strcmp(dmx->field, "n") == 0) - sst = GLP_NOFEAS; - else if (strcmp(dmx->field, "u") == 0) - sst = GLP_UNDEF; - else - dmx_error(dmx, "solution status missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &obj) != 0) - dmx_error(dmx, "objective value missing or invalid"); - dmx_end_of_line(dmx); - /* allocate working arrays */ - stat = xalloc(1+m+n, sizeof(stat[0])); - for (k = 1; k <= m+n; k++) - stat[k] = '?'; - prim = xalloc(1+m+n, sizeof(prim[0])); - dual = xalloc(1+m+n, sizeof(dual[0])); - /* read solution descriptor lines */ - for (;;) - { dmx_read_designator(dmx); - if (strcmp(dmx->field, "i") == 0) - { /* row solution descriptor */ - dmx_read_field(dmx); - if (str2int(dmx->field, &i) != 0) - dmx_error(dmx, "row number missing or invalid"); - if (!(1 <= i && i <= m)) - dmx_error(dmx, "row number out of range"); - if (stat[i] != '?') - dmx_error(dmx, "duplicate row solution descriptor"); - stat[i] = GLP_BS; - dmx_read_field(dmx); - if (str2num(dmx->field, &prim[i]) != 0) - dmx_error(dmx, "row primal value missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &dual[i]) != 0) - dmx_error(dmx, "row dual value missing or invalid"); - dmx_end_of_line(dmx); - } - else if (strcmp(dmx->field, "j") == 0) - { /* column solution descriptor */ - dmx_read_field(dmx); - if (str2int(dmx->field, &j) != 0) - dmx_error(dmx, "column number missing or invalid"); - if (!(1 <= j && j <= n)) - dmx_error(dmx, "column number out of range"); - if (stat[m+j] != '?') - dmx_error(dmx, "duplicate column solution descriptor"); - stat[m+j] = GLP_BS; - dmx_read_field(dmx); - if (str2num(dmx->field, &prim[m+j]) != 0) - dmx_error(dmx, "column primal value missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &dual[m+j]) != 0) - dmx_error(dmx, "column dual value missing or invalid"); - dmx_end_of_line(dmx); - } - else if (strcmp(dmx->field, "e") == 0) - break; - else - dmx_error(dmx, "line designator missing or invalid"); - dmx_end_of_line(dmx); - } - /* store solution components into problem object */ - for (k = 1; k <= m+n; k++) - { if (stat[k] == '?') - dmx_error(dmx, "incomplete interior-point solution"); - } - P->ipt_stat = sst; - P->ipt_obj = obj; - for (i = 1; i <= m; i++) - { P->row[i]->pval = prim[i]; - P->row[i]->dval = dual[i]; - } - for (j = 1; j <= n; j++) - { P->col[j]->pval = prim[m+j]; - P->col[j]->dval = dual[m+j]; - } - /* interior-point solution has been successfully read */ - xprintf("%d lines were read\n", dmx->count); - ret = 0; -done: if (dmx->fp != NULL) - glp_close(dmx->fp); - if (stat != NULL) - xfree(stat); - if (prim != NULL) - xfree(prim); - if (dual != NULL) - xfree(dual); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdmaxf.c b/code/3rd_glpk/api/rdmaxf.c deleted file mode 100644 index a45405c9..00000000 --- a/code/3rd_glpk/api/rdmaxf.c +++ /dev/null @@ -1,163 +0,0 @@ -/* rdmaxf.c (read maximum flow problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "glpk.h" -#include "misc.h" - -#define error dmx_error -#define warning dmx_warning -#define read_char dmx_read_char -#define read_designator dmx_read_designator -#define read_field dmx_read_field -#define end_of_line dmx_end_of_line -#define check_int dmx_check_int - -/*********************************************************************** -* NAME -* -* glp_read_maxflow - read maximum flow problem data in DIMACS format -* -* SYNOPSIS -* -* int glp_read_maxflow(glp_graph *G, int *s, int *t, int a_cap, -* const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_maxflow reads maximum flow problem data in -* DIMACS format from a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_maxflow(glp_graph *G, int *_s, int *_t, int a_cap, - const char *fname) -{ DMX _csa, *csa = &_csa; - glp_arc *a; - int i, j, k, s, t, nv, na, ret = 0; - double cap; - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_read_maxflow: a_cap = %d; invalid offset\n", - a_cap); - glp_erase_graph(G, G->v_size, G->a_size); - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->fname = fname; - csa->fp = NULL; - csa->count = 0; - csa->c = '\n'; - csa->field[0] = '\0'; - csa->empty = csa->nonint = 0; - xprintf("Reading maximum flow problem data from '%s'...\n", - fname); - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - longjmp(csa->jump, 1); - } - /* read problem line */ - read_designator(csa); - if (strcmp(csa->field, "p") != 0) - error(csa, "problem line missing or invalid"); - read_field(csa); - if (strcmp(csa->field, "max") != 0) - error(csa, "wrong problem designator; 'max' expected"); - read_field(csa); - if (!(str2int(csa->field, &nv) == 0 && nv >= 2)) - error(csa, "number of nodes missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &na) == 0 && na >= 0)) - error(csa, "number of arcs missing or invalid"); - xprintf("Flow network has %d node%s and %d arc%s\n", - nv, nv == 1 ? "" : "s", na, na == 1 ? "" : "s"); - if (nv > 0) glp_add_vertices(G, nv); - end_of_line(csa); - /* read node descriptor lines */ - s = t = 0; - for (;;) - { read_designator(csa); - if (strcmp(csa->field, "n") != 0) break; - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "node number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "node number %d out of range", i); - read_field(csa); - if (strcmp(csa->field, "s") == 0) - { if (s > 0) - error(csa, "only one source node allowed"); - s = i; - } - else if (strcmp(csa->field, "t") == 0) - { if (t > 0) - error(csa, "only one sink node allowed"); - t = i; - } - else - error(csa, "wrong node designator; 's' or 't' expected"); - if (s > 0 && s == t) - error(csa, "source and sink nodes must be distinct"); - end_of_line(csa); - } - if (s == 0) - error(csa, "source node descriptor missing\n"); - if (t == 0) - error(csa, "sink node descriptor missing\n"); - if (_s != NULL) *_s = s; - if (_t != NULL) *_t = t; - /* read arc descriptor lines */ - for (k = 1; k <= na; k++) - { if (k > 1) read_designator(csa); - if (strcmp(csa->field, "a") != 0) - error(csa, "wrong line designator; 'a' expected"); - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "starting node number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "starting node number %d out of range", i); - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "ending node number missing or invalid"); - if (!(1 <= j && j <= nv)) - error(csa, "ending node number %d out of range", j); - read_field(csa); - if (!(str2num(csa->field, &cap) == 0 && cap >= 0.0)) - error(csa, "arc capacity missing or invalid"); - check_int(csa, cap); - a = glp_add_arc(G, i, j); - if (a_cap >= 0) - memcpy((char *)a->data + a_cap, &cap, sizeof(double)); - end_of_line(csa); - } - xprintf("%d lines were read\n", csa->count); -done: if (ret) glp_erase_graph(G, G->v_size, G->a_size); - if (csa->fp != NULL) glp_close(csa->fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdmcf.c b/code/3rd_glpk/api/rdmcf.c deleted file mode 100644 index bab1ec79..00000000 --- a/code/3rd_glpk/api/rdmcf.c +++ /dev/null @@ -1,186 +0,0 @@ -/* rdmcf.c (read min-cost flow problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "glpk.h" -#include "misc.h" - -#define error dmx_error -#define warning dmx_warning -#define read_char dmx_read_char -#define read_designator dmx_read_designator -#define read_field dmx_read_field -#define end_of_line dmx_end_of_line -#define check_int dmx_check_int - -/*********************************************************************** -* NAME -* -* glp_read_mincost - read min-cost flow problem data in DIMACS format -* -* SYNOPSIS -* -* int glp_read_mincost(glp_graph *G, int v_rhs, int a_low, int a_cap, -* int a_cost, const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_mincost reads minimum cost flow problem data in -* DIMACS format from a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_mincost(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, const char *fname) -{ DMX _csa, *csa = &_csa; - glp_vertex *v; - glp_arc *a; - int i, j, k, nv, na, ret = 0; - double rhs, low, cap, cost; - char *flag = NULL; - if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_read_mincost: v_rhs = %d; invalid offset\n", - v_rhs); - if (a_low >= 0 && a_low > G->a_size - (int)sizeof(double)) - xerror("glp_read_mincost: a_low = %d; invalid offset\n", - a_low); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_read_mincost: a_cap = %d; invalid offset\n", - a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_read_mincost: a_cost = %d; invalid offset\n", - a_cost); - glp_erase_graph(G, G->v_size, G->a_size); - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->fname = fname; - csa->fp = NULL; - csa->count = 0; - csa->c = '\n'; - csa->field[0] = '\0'; - csa->empty = csa->nonint = 0; - xprintf("Reading min-cost flow problem data from '%s'...\n", - fname); - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - longjmp(csa->jump, 1); - } - /* read problem line */ - read_designator(csa); - if (strcmp(csa->field, "p") != 0) - error(csa, "problem line missing or invalid"); - read_field(csa); - if (strcmp(csa->field, "min") != 0) - error(csa, "wrong problem designator; 'min' expected"); - read_field(csa); - if (!(str2int(csa->field, &nv) == 0 && nv >= 0)) - error(csa, "number of nodes missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &na) == 0 && na >= 0)) - error(csa, "number of arcs missing or invalid"); - xprintf("Flow network has %d node%s and %d arc%s\n", - nv, nv == 1 ? "" : "s", na, na == 1 ? "" : "s"); - if (nv > 0) glp_add_vertices(G, nv); - end_of_line(csa); - /* read node descriptor lines */ - flag = xcalloc(1+nv, sizeof(char)); - memset(&flag[1], 0, nv * sizeof(char)); - if (v_rhs >= 0) - { rhs = 0.0; - for (i = 1; i <= nv; i++) - { v = G->v[i]; - memcpy((char *)v->data + v_rhs, &rhs, sizeof(double)); - } - } - for (;;) - { read_designator(csa); - if (strcmp(csa->field, "n") != 0) break; - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "node number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "node number %d out of range", i); - if (flag[i]) - error(csa, "duplicate descriptor of node %d", i); - read_field(csa); - if (str2num(csa->field, &rhs) != 0) - error(csa, "node supply/demand missing or invalid"); - check_int(csa, rhs); - if (v_rhs >= 0) - { v = G->v[i]; - memcpy((char *)v->data + v_rhs, &rhs, sizeof(double)); - } - flag[i] = 1; - end_of_line(csa); - } - xfree(flag), flag = NULL; - /* read arc descriptor lines */ - for (k = 1; k <= na; k++) - { if (k > 1) read_designator(csa); - if (strcmp(csa->field, "a") != 0) - error(csa, "wrong line designator; 'a' expected"); - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "starting node number missing or invalid"); - if (!(1 <= i && i <= nv)) - error(csa, "starting node number %d out of range", i); - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "ending node number missing or invalid"); - if (!(1 <= j && j <= nv)) - error(csa, "ending node number %d out of range", j); - read_field(csa); - if (!(str2num(csa->field, &low) == 0 && low >= 0.0)) - error(csa, "lower bound of arc flow missing or invalid"); - check_int(csa, low); - read_field(csa); - if (!(str2num(csa->field, &cap) == 0 && cap >= low)) - error(csa, "upper bound of arc flow missing or invalid"); - check_int(csa, cap); - read_field(csa); - if (str2num(csa->field, &cost) != 0) - error(csa, "per-unit cost of arc flow missing or invalid"); - check_int(csa, cost); - a = glp_add_arc(G, i, j); - if (a_low >= 0) - memcpy((char *)a->data + a_low, &low, sizeof(double)); - if (a_cap >= 0) - memcpy((char *)a->data + a_cap, &cap, sizeof(double)); - if (a_cost >= 0) - memcpy((char *)a->data + a_cost, &cost, sizeof(double)); - end_of_line(csa); - } - xprintf("%d lines were read\n", csa->count); -done: if (ret) glp_erase_graph(G, G->v_size, G->a_size); - if (csa->fp != NULL) glp_close(csa->fp); - if (flag != NULL) xfree(flag); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdmip.c b/code/3rd_glpk/api/rdmip.c deleted file mode 100644 index 7aec26b3..00000000 --- a/code/3rd_glpk/api/rdmip.c +++ /dev/null @@ -1,172 +0,0 @@ -/* rdmip.c (read MIP solution in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "env.h" -#include "misc.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_read_mip - read MIP solution in GLPK format -* -* SYNOPSIS -* -* int glp_read_mip(glp_prob *P, const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_mip reads MIP solution from a text file in GLPK -* format. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_mip(glp_prob *P, const char *fname) -{ DMX dmx_, *dmx = &dmx_; - int i, j, k, m, n, sst, ret = 1; - char *stat = NULL; - double obj, *prim = NULL; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_read_mip: P = %p; invalid problem object\n", P); -#endif - if (fname == NULL) - xerror("glp_read_mip: fname = %d; invalid parameter\n", fname); - if (setjmp(dmx->jump)) - goto done; - dmx->fname = fname; - dmx->fp = NULL; - dmx->count = 0; - dmx->c = '\n'; - dmx->field[0] = '\0'; - dmx->empty = dmx->nonint = 0; - xprintf("Reading MIP solution from '%s'...\n", fname); - dmx->fp = glp_open(fname, "r"); - if (dmx->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* read solution line */ - dmx_read_designator(dmx); - if (strcmp(dmx->field, "s") != 0) - dmx_error(dmx, "solution line missing or invalid"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "mip") != 0) - dmx_error(dmx, "wrong solution designator; 'mip' expected"); - dmx_read_field(dmx); - if (!(str2int(dmx->field, &m) == 0 && m >= 0)) - dmx_error(dmx, "number of rows missing or invalid"); - if (m != P->m) - dmx_error(dmx, "number of rows mismatch"); - dmx_read_field(dmx); - if (!(str2int(dmx->field, &n) == 0 && n >= 0)) - dmx_error(dmx, "number of columns missing or invalid"); - if (n != P->n) - dmx_error(dmx, "number of columns mismatch"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "o") == 0) - sst = GLP_OPT; - else if (strcmp(dmx->field, "f") == 0) - sst = GLP_FEAS; - else if (strcmp(dmx->field, "n") == 0) - sst = GLP_NOFEAS; - else if (strcmp(dmx->field, "u") == 0) - sst = GLP_UNDEF; - else - dmx_error(dmx, "solution status missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &obj) != 0) - dmx_error(dmx, "objective value missing or invalid"); - dmx_end_of_line(dmx); - /* allocate working arrays */ - stat = xalloc(1+m+n, sizeof(stat[0])); - for (k = 1; k <= m+n; k++) - stat[k] = '?'; - prim = xalloc(1+m+n, sizeof(prim[0])); - /* read solution descriptor lines */ - for (;;) - { dmx_read_designator(dmx); - if (strcmp(dmx->field, "i") == 0) - { /* row solution descriptor */ - dmx_read_field(dmx); - if (str2int(dmx->field, &i) != 0) - dmx_error(dmx, "row number missing or invalid"); - if (!(1 <= i && i <= m)) - dmx_error(dmx, "row number out of range"); - if (stat[i] != '?') - dmx_error(dmx, "duplicate row solution descriptor"); - stat[i] = GLP_BS; - dmx_read_field(dmx); - if (str2num(dmx->field, &prim[i]) != 0) - dmx_error(dmx, "row value missing or invalid"); - dmx_end_of_line(dmx); - } - else if (strcmp(dmx->field, "j") == 0) - { /* column solution descriptor */ - dmx_read_field(dmx); - if (str2int(dmx->field, &j) != 0) - dmx_error(dmx, "column number missing or invalid"); - if (!(1 <= j && j <= n)) - dmx_error(dmx, "column number out of range"); - if (stat[m+j] != '?') - dmx_error(dmx, "duplicate column solution descriptor"); - stat[m+j] = GLP_BS; - dmx_read_field(dmx); - if (str2num(dmx->field, &prim[m+j]) != 0) - dmx_error(dmx, "column value missing or invalid"); - dmx_end_of_line(dmx); - } - else if (strcmp(dmx->field, "e") == 0) - break; - else - dmx_error(dmx, "line designator missing or invalid"); - dmx_end_of_line(dmx); - } - /* store solution components into problem object */ - for (k = 1; k <= m+n; k++) - { if (stat[k] == '?') - dmx_error(dmx, "incomplete MIP solution"); - } - P->mip_stat = sst; - P->mip_obj = obj; - for (i = 1; i <= m; i++) - P->row[i]->mipx = prim[i]; - for (j = 1; j <= n; j++) - P->col[j]->mipx = prim[m+j]; - /* MIP solution has been successfully read */ - xprintf("%d lines were read\n", dmx->count); - ret = 0; -done: if (dmx->fp != NULL) - glp_close(dmx->fp); - if (stat != NULL) - xfree(stat); - if (prim != NULL) - xfree(prim); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdprob.c b/code/3rd_glpk/api/rdprob.c deleted file mode 100644 index 1ad544a5..00000000 --- a/code/3rd_glpk/api/rdprob.c +++ /dev/null @@ -1,377 +0,0 @@ -/* rdprob.c (read problem data in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "misc.h" -#include "prob.h" - -#define xfprintf glp_format -#define error dmx_error -#define warning dmx_warning -#define read_char dmx_read_char -#define read_designator dmx_read_designator -#define read_field dmx_read_field -#define end_of_line dmx_end_of_line -#define check_int dmx_check_int - -/*********************************************************************** -* NAME -* -* glp_read_prob - read problem data in GLPK format -* -* SYNOPSIS -* -* int glp_read_prob(glp_prob *P, int flags, const char *fname); -* -* The routine glp_read_prob reads problem data in GLPK LP/MIP format -* from a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_prob(glp_prob *P, int flags, const char *fname) -{ DMX _csa, *csa = &_csa; - int mip, m, n, nnz, ne, i, j, k, type, kind, ret, *ln = NULL, - *ia = NULL, *ja = NULL; - double lb, ub, temp, *ar = NULL; - char *rf = NULL, *cf = NULL; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_read_prob: P = %p; invalid problem object\n", - P); -#endif - if (flags != 0) - xerror("glp_read_prob: flags = %d; invalid parameter\n", - flags); - if (fname == NULL) - xerror("glp_read_prob: fname = %d; invalid parameter\n", - fname); - glp_erase_prob(P); - if (setjmp(csa->jump)) - { ret = 1; - goto done; - } - csa->fname = fname; - csa->fp = NULL; - csa->count = 0; - csa->c = '\n'; - csa->field[0] = '\0'; - csa->empty = csa->nonint = 0; - xprintf("Reading problem data from '%s'...\n", fname); - csa->fp = glp_open(fname, "r"); - if (csa->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - longjmp(csa->jump, 1); - } - /* read problem line */ - read_designator(csa); - if (strcmp(csa->field, "p") != 0) - error(csa, "problem line missing or invalid"); - read_field(csa); - if (strcmp(csa->field, "lp") == 0) - mip = 0; - else if (strcmp(csa->field, "mip") == 0) - mip = 1; - else - error(csa, "wrong problem designator; 'lp' or 'mip' expected"); - read_field(csa); - if (strcmp(csa->field, "min") == 0) - glp_set_obj_dir(P, GLP_MIN); - else if (strcmp(csa->field, "max") == 0) - glp_set_obj_dir(P, GLP_MAX); - else - error(csa, "objective sense missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &m) == 0 && m >= 0)) - error(csa, "number of rows missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &n) == 0 && n >= 0)) - error(csa, "number of columns missing or invalid"); - read_field(csa); - if (!(str2int(csa->field, &nnz) == 0 && nnz >= 0)) - error(csa, "number of constraint coefficients missing or inval" - "id"); - if (m > 0) - { glp_add_rows(P, m); - for (i = 1; i <= m; i++) - glp_set_row_bnds(P, i, GLP_FX, 0.0, 0.0); - } - if (n > 0) - { glp_add_cols(P, n); - for (j = 1; j <= n; j++) - { if (!mip) - glp_set_col_bnds(P, j, GLP_LO, 0.0, 0.0); - else - glp_set_col_kind(P, j, GLP_BV); - } - } - end_of_line(csa); - /* allocate working arrays */ - rf = xcalloc(1+m, sizeof(char)); - memset(rf, 0, 1+m); - cf = xcalloc(1+n, sizeof(char)); - memset(cf, 0, 1+n); - ln = xcalloc(1+nnz, sizeof(int)); - ia = xcalloc(1+nnz, sizeof(int)); - ja = xcalloc(1+nnz, sizeof(int)); - ar = xcalloc(1+nnz, sizeof(double)); - /* read descriptor lines */ - ne = 0; - for (;;) - { read_designator(csa); - if (strcmp(csa->field, "i") == 0) - { /* row descriptor */ - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "row number missing or invalid"); - if (!(1 <= i && i <= m)) - error(csa, "row number out of range"); - read_field(csa); - if (strcmp(csa->field, "f") == 0) - type = GLP_FR; - else if (strcmp(csa->field, "l") == 0) - type = GLP_LO; - else if (strcmp(csa->field, "u") == 0) - type = GLP_UP; - else if (strcmp(csa->field, "d") == 0) - type = GLP_DB; - else if (strcmp(csa->field, "s") == 0) - type = GLP_FX; - else - error(csa, "row type missing or invalid"); - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { read_field(csa); - if (str2num(csa->field, &lb) != 0) - error(csa, "row lower bound/fixed value missing or in" - "valid"); - } - else - lb = 0.0; - if (type == GLP_UP || type == GLP_DB) - { read_field(csa); - if (str2num(csa->field, &ub) != 0) - error(csa, "row upper bound missing or invalid"); - } - else - ub = 0.0; - if (rf[i] & 0x01) - error(csa, "duplicate row descriptor"); - glp_set_row_bnds(P, i, type, lb, ub), rf[i] |= 0x01; - } - else if (strcmp(csa->field, "j") == 0) - { /* column descriptor */ - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "column number missing or invalid"); - if (!(1 <= j && j <= n)) - error(csa, "column number out of range"); - if (!mip) - kind = GLP_CV; - else - { read_field(csa); - if (strcmp(csa->field, "c") == 0) - kind = GLP_CV; - else if (strcmp(csa->field, "i") == 0) - kind = GLP_IV; - else if (strcmp(csa->field, "b") == 0) - { kind = GLP_IV; - type = GLP_DB, lb = 0.0, ub = 1.0; - goto skip; - } - else - error(csa, "column kind missing or invalid"); - } - read_field(csa); - if (strcmp(csa->field, "f") == 0) - type = GLP_FR; - else if (strcmp(csa->field, "l") == 0) - type = GLP_LO; - else if (strcmp(csa->field, "u") == 0) - type = GLP_UP; - else if (strcmp(csa->field, "d") == 0) - type = GLP_DB; - else if (strcmp(csa->field, "s") == 0) - type = GLP_FX; - else - error(csa, "column type missing or invalid"); - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { read_field(csa); - if (str2num(csa->field, &lb) != 0) - error(csa, "column lower bound/fixed value missing or" - " invalid"); - } - else - lb = 0.0; - if (type == GLP_UP || type == GLP_DB) - { read_field(csa); - if (str2num(csa->field, &ub) != 0) - error(csa, "column upper bound missing or invalid"); - } - else - ub = 0.0; -skip: if (cf[j] & 0x01) - error(csa, "duplicate column descriptor"); - glp_set_col_kind(P, j, kind); - glp_set_col_bnds(P, j, type, lb, ub), cf[j] |= 0x01; - } - else if (strcmp(csa->field, "a") == 0) - { /* coefficient descriptor */ - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "row number missing or invalid"); - if (!(0 <= i && i <= m)) - error(csa, "row number out of range"); - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "column number missing or invalid"); - if (!((i == 0 ? 0 : 1) <= j && j <= n)) - error(csa, "column number out of range"); - read_field(csa); - if (i == 0) - { if (str2num(csa->field, &temp) != 0) - error(csa, "objective %s missing or invalid", - j == 0 ? "constant term" : "coefficient"); - if (cf[j] & 0x10) - error(csa, "duplicate objective %s", - j == 0 ? "constant term" : "coefficient"); - glp_set_obj_coef(P, j, temp), cf[j] |= 0x10; - } - else - { if (str2num(csa->field, &temp) != 0) - error(csa, "constraint coefficient missing or invalid" - ); - if (ne == nnz) - error(csa, "too many constraint coefficient descripto" - "rs"); - ln[++ne] = csa->count; - ia[ne] = i, ja[ne] = j, ar[ne] = temp; - } - } - else if (strcmp(csa->field, "n") == 0) - { /* symbolic name descriptor */ - read_field(csa); - if (strcmp(csa->field, "p") == 0) - { /* problem name */ - read_field(csa); - if (P->name != NULL) - error(csa, "duplicate problem name"); - glp_set_prob_name(P, csa->field); - } - else if (strcmp(csa->field, "z") == 0) - { /* objective name */ - read_field(csa); - if (P->obj != NULL) - error(csa, "duplicate objective name"); - glp_set_obj_name(P, csa->field); - } - else if (strcmp(csa->field, "i") == 0) - { /* row name */ - read_field(csa); - if (str2int(csa->field, &i) != 0) - error(csa, "row number missing or invalid"); - if (!(1 <= i && i <= m)) - error(csa, "row number out of range"); - read_field(csa); - if (P->row[i]->name != NULL) - error(csa, "duplicate row name"); - glp_set_row_name(P, i, csa->field); - } - else if (strcmp(csa->field, "j") == 0) - { /* column name */ - read_field(csa); - if (str2int(csa->field, &j) != 0) - error(csa, "column number missing or invalid"); - if (!(1 <= j && j <= n)) - error(csa, "column number out of range"); - read_field(csa); - if (P->col[j]->name != NULL) - error(csa, "duplicate column name"); - glp_set_col_name(P, j, csa->field); - } - else - error(csa, "object designator missing or invalid"); - } - else if (strcmp(csa->field, "e") == 0) - break; - else - error(csa, "line designator missing or invalid"); - end_of_line(csa); - } - if (ne < nnz) - error(csa, "too few constraint coefficient descriptors"); - xassert(ne == nnz); - k = glp_check_dup(m, n, ne, ia, ja); - xassert(0 <= k && k <= nnz); - if (k > 0) - { csa->count = ln[k]; - error(csa, "duplicate constraint coefficient"); - } - glp_load_matrix(P, ne, ia, ja, ar); - /* print some statistics */ - if (P->name != NULL) - xprintf("Problem: %s\n", P->name); - if (P->obj != NULL) - xprintf("Objective: %s\n", P->obj); - xprintf("%d row%s, %d column%s, %d non-zero%s\n", - m, m == 1 ? "" : "s", n, n == 1 ? "" : "s", nnz, nnz == 1 ? - "" : "s"); - if (glp_get_num_int(P) > 0) - { int ni = glp_get_num_int(P); - int nb = glp_get_num_bin(P); - if (ni == 1) - { if (nb == 0) - xprintf("One variable is integer\n"); - else - xprintf("One variable is binary\n"); - } - else - { xprintf("%d integer variables, ", ni); - if (nb == 0) - xprintf("none"); - else if (nb == 1) - xprintf("one"); - else if (nb == ni) - xprintf("all"); - else - xprintf("%d", nb); - xprintf(" of which %s binary\n", nb == 1 ? "is" : "are"); - } - } - xprintf("%d lines were read\n", csa->count); - /* problem data has been successfully read */ - glp_sort_matrix(P); - ret = 0; -done: if (csa->fp != NULL) glp_close(csa->fp); - if (rf != NULL) xfree(rf); - if (cf != NULL) xfree(cf); - if (ln != NULL) xfree(ln); - if (ia != NULL) xfree(ia); - if (ja != NULL) xfree(ja); - if (ar != NULL) xfree(ar); - if (ret) glp_erase_prob(P); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rdsol.c b/code/3rd_glpk/api/rdsol.c deleted file mode 100644 index d85a2562..00000000 --- a/code/3rd_glpk/api/rdsol.c +++ /dev/null @@ -1,225 +0,0 @@ -/* rdsol.c (read basic solution in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" -#include "env.h" -#include "misc.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_read_sol - read basic solution in GLPK format -* -* SYNOPSIS -* -* int glp_read_sol(glp_prob *P, const char *fname); -* -* DESCRIPTION -* -* The routine glp_read_sol reads basic solution from a text file in -* GLPK format. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_read_sol(glp_prob *P, const char *fname) -{ DMX dmx_, *dmx = &dmx_; - int i, j, k, m, n, pst, dst, ret = 1; - char *stat = NULL; - double obj, *prim = NULL, *dual = NULL; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_read_sol: P = %p; invalid problem object\n", P); -#endif - if (fname == NULL) - xerror("glp_read_sol: fname = %d; invalid parameter\n", fname); - if (setjmp(dmx->jump)) - goto done; - dmx->fname = fname; - dmx->fp = NULL; - dmx->count = 0; - dmx->c = '\n'; - dmx->field[0] = '\0'; - dmx->empty = dmx->nonint = 0; - xprintf("Reading basic solution from '%s'...\n", fname); - dmx->fp = glp_open(fname, "r"); - if (dmx->fp == NULL) - { xprintf("Unable to open '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* read solution line */ - dmx_read_designator(dmx); - if (strcmp(dmx->field, "s") != 0) - dmx_error(dmx, "solution line missing or invalid"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "bas") != 0) - dmx_error(dmx, "wrong solution designator; 'bas' expected"); - dmx_read_field(dmx); - if (!(str2int(dmx->field, &m) == 0 && m >= 0)) - dmx_error(dmx, "number of rows missing or invalid"); - if (m != P->m) - dmx_error(dmx, "number of rows mismatch"); - dmx_read_field(dmx); - if (!(str2int(dmx->field, &n) == 0 && n >= 0)) - dmx_error(dmx, "number of columns missing or invalid"); - if (n != P->n) - dmx_error(dmx, "number of columns mismatch"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "u") == 0) - pst = GLP_UNDEF; - else if (strcmp(dmx->field, "f") == 0) - pst = GLP_FEAS; - else if (strcmp(dmx->field, "i") == 0) - pst = GLP_INFEAS; - else if (strcmp(dmx->field, "n") == 0) - pst = GLP_NOFEAS; - else - dmx_error(dmx, "primal solution status missing or invalid"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "u") == 0) - dst = GLP_UNDEF; - else if (strcmp(dmx->field, "f") == 0) - dst = GLP_FEAS; - else if (strcmp(dmx->field, "i") == 0) - dst = GLP_INFEAS; - else if (strcmp(dmx->field, "n") == 0) - dst = GLP_NOFEAS; - else - dmx_error(dmx, "dual solution status missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &obj) != 0) - dmx_error(dmx, "objective value missing or invalid"); - dmx_end_of_line(dmx); - /* allocate working arrays */ - stat = xalloc(1+m+n, sizeof(stat[0])); - for (k = 1; k <= m+n; k++) - stat[k] = '?'; - prim = xalloc(1+m+n, sizeof(prim[0])); - dual = xalloc(1+m+n, sizeof(dual[0])); - /* read solution descriptor lines */ - for (;;) - { dmx_read_designator(dmx); - if (strcmp(dmx->field, "i") == 0) - { /* row solution descriptor */ - dmx_read_field(dmx); - if (str2int(dmx->field, &i) != 0) - dmx_error(dmx, "row number missing or invalid"); - if (!(1 <= i && i <= m)) - dmx_error(dmx, "row number out of range"); - if (stat[i] != '?') - dmx_error(dmx, "duplicate row solution descriptor"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "b") == 0) - stat[i] = GLP_BS; - else if (strcmp(dmx->field, "l") == 0) - stat[i] = GLP_NL; - else if (strcmp(dmx->field, "u") == 0) - stat[i] = GLP_NU; - else if (strcmp(dmx->field, "f") == 0) - stat[i] = GLP_NF; - else if (strcmp(dmx->field, "s") == 0) - stat[i] = GLP_NS; - else - dmx_error(dmx, "row status missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &prim[i]) != 0) - dmx_error(dmx, "row primal value missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &dual[i]) != 0) - dmx_error(dmx, "row dual value missing or invalid"); - dmx_end_of_line(dmx); - } - else if (strcmp(dmx->field, "j") == 0) - { /* column solution descriptor */ - dmx_read_field(dmx); - if (str2int(dmx->field, &j) != 0) - dmx_error(dmx, "column number missing or invalid"); - if (!(1 <= j && j <= n)) - dmx_error(dmx, "column number out of range"); - if (stat[m+j] != '?') - dmx_error(dmx, "duplicate column solution descriptor"); - dmx_read_field(dmx); - if (strcmp(dmx->field, "b") == 0) - stat[m+j] = GLP_BS; - else if (strcmp(dmx->field, "l") == 0) - stat[m+j] = GLP_NL; - else if (strcmp(dmx->field, "u") == 0) - stat[m+j] = GLP_NU; - else if (strcmp(dmx->field, "f") == 0) - stat[m+j] = GLP_NF; - else if (strcmp(dmx->field, "s") == 0) - stat[m+j] = GLP_NS; - else - dmx_error(dmx, "column status missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &prim[m+j]) != 0) - dmx_error(dmx, "column primal value missing or invalid"); - dmx_read_field(dmx); - if (str2num(dmx->field, &dual[m+j]) != 0) - dmx_error(dmx, "column dual value missing or invalid"); - dmx_end_of_line(dmx); - } - else if (strcmp(dmx->field, "e") == 0) - break; - else - dmx_error(dmx, "line designator missing or invalid"); - dmx_end_of_line(dmx); - } - /* store solution components into problem object */ - for (k = 1; k <= m+n; k++) - { if (stat[k] == '?') - dmx_error(dmx, "incomplete basic solution"); - } - P->pbs_stat = pst; - P->dbs_stat = dst; - P->obj_val = obj; - P->it_cnt = 0; - P->some = 0; - for (i = 1; i <= m; i++) - { glp_set_row_stat(P, i, stat[i]); - P->row[i]->prim = prim[i]; - P->row[i]->dual = dual[i]; - } - for (j = 1; j <= n; j++) - { glp_set_col_stat(P, j, stat[m+j]); - P->col[j]->prim = prim[m+j]; - P->col[j]->dual = dual[m+j]; - } - /* basic solution has been successfully read */ - xprintf("%d lines were read\n", dmx->count); - ret = 0; -done: if (dmx->fp != NULL) - glp_close(dmx->fp); - if (stat != NULL) - xfree(stat); - if (prim != NULL) - xfree(prim); - if (dual != NULL) - xfree(dual); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/rmfgen.c b/code/3rd_glpk/api/rmfgen.c deleted file mode 100644 index a1ba27bb..00000000 --- a/code/3rd_glpk/api/rmfgen.c +++ /dev/null @@ -1,368 +0,0 @@ -/* rmfgen.c (Goldfarb's maximum flow problem generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* This code is a modified version of the program RMFGEN, a maxflow -* problem generator developed by D.Goldfarb and M.Grigoriadis, and -* originally implemented by Tamas Badics . -* The original code is publically available on the DIMACS ftp site at: -* . -* -* All changes concern only the program interface, so this modified -* version produces exactly the same instances as the original version. -* -* Changes were made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "rng.h" - -/*********************************************************************** -* NAME -* -* glp_rmfgen - Goldfarb's maximum flow problem generator -* -* SYNOPSIS -* -* int glp_rmfgen(glp_graph *G, int *s, int *t, int a_cap, -* const int parm[1+5]); -* -* DESCRIPTION -* -* The routine glp_rmfgen is a maximum flow problem generator developed -* by D.Goldfarb and M.Grigoriadis. -* -* The parameter G specifies the graph object, to which the generated -* problem data have to be stored. Note that on entry the graph object -* is erased with the routine glp_erase_graph. -* -* The pointer s specifies a location, to which the routine stores the -* source node number. If s is NULL, the node number is not stored. -* -* The pointer t specifies a location, to which the routine stores the -* sink node number. If t is NULL, the node number is not stored. -* -* The parameter a_cap specifies an offset of the field of type double -* in the arc data block, to which the routine stores the arc capacity. -* If a_cap < 0, the capacity is not stored. -* -* The array parm contains description of the network to be generated: -* -* parm[0] not used -* parm[1] (seed) random number seed (a positive integer) -* parm[2] (a) frame size -* parm[3] (b) depth -* parm[4] (c1) minimal arc capacity -* parm[5] (c2) maximal arc capacity -* -* RETURNS -* -* If the instance was successfully generated, the routine glp_netgen -* returns zero; otherwise, if specified parameters are inconsistent, -* the routine returns a non-zero error code. -* -* COMMENTS -* -* The generated network is as follows. It has b pieces of frames of -* size a * a. (So alltogether the number of vertices is a * a * b) -* -* In each frame all the vertices are connected with their neighbours -* (forth and back). In addition the vertices of a frame are connected -* one to one with the vertices of next frame using a random permutation -* of those vertices. -* -* The source is the lower left vertex of the first frame, the sink is -* the upper right vertex of the b'th frame. -* -* t -* +-------+ -* | .| -* | . | -* / | / | -* +-------+/ -+ b -* | | |/. -* a | -v- |/ -* | | |/ -* +-------+ 1 -* s a -* -* The capacities are randomly chosen integers from the range of [c1,c2] -* in the case of interconnecting edges, and c2 * a * a for the in-frame -* edges. -* -* REFERENCES -* -* D.Goldfarb and M.D.Grigoriadis, "A computational comparison of the -* Dinic and network simplex methods for maximum flow." Annals of Op. -* Res. 13 (1988), pp. 83-123. -* -* U.Derigs and W.Meier, "Implementing Goldberg's max-flow algorithm: -* A computational investigation." Zeitschrift fuer Operations Research -* 33 (1989), pp. 383-403. */ - -typedef struct VERTEX -{ struct EDGE **edgelist; - /* Pointer to the list of pointers to the adjacent edges. - (No matter that to or from edges) */ - struct EDGE **current; - /* Pointer to the current edge */ - int degree; - /* Number of adjacent edges (both direction) */ - int index; -} vertex; - -typedef struct EDGE -{ int from; - int to; - int cap; - /* Capacity */ -} edge; - -typedef struct NETWORK -{ struct NETWORK *next, *prev; - int vertnum; - int edgenum; - vertex *verts; - /* Vertex array[1..vertnum] */ - edge *edges; - /* Edge array[1..edgenum] */ - int source; - /* Pointer to the source */ - int sink; - /* Pointer to the sink */ -} network; - -struct csa -{ /* common storage area */ - glp_graph *G; - int *s, *t, a_cap; - RNG *rand; - network *N; - int *Parr; - int A, AA, C2AA, Ec; -}; - -#define G (csa->G) -#define s (csa->s) -#define t (csa->t) -#define a_cap (csa->a_cap) -#define N (csa->N) -#define Parr (csa->Parr) -#define A (csa->A) -#define AA (csa->AA) -#define C2AA (csa->C2AA) -#define Ec (csa->Ec) - -#undef random -#define random(A) (int)(rng_unif_01(csa->rand) * (double)(A)) -#define RANDOM(A, B) (int)(random((B) - (A) + 1) + (A)) -#define sgn(A) (((A) > 0) ? 1 : ((A) == 0) ? 0 : -1) - -static void make_edge(struct csa *csa, int from, int to, int c1, int c2) -{ Ec++; - N->edges[Ec].from = from; - N->edges[Ec].to = to; - N->edges[Ec].cap = RANDOM(c1, c2); - return; -} - -static void permute(struct csa *csa) -{ int i, j, tmp; - for (i = 1; i < AA; i++) - { j = RANDOM(i, AA); - tmp = Parr[i]; - Parr[i] = Parr[j]; - Parr[j] = tmp; - } - return; -} - -static void connect(struct csa *csa, int offset, int cv, int x1, int y1) -{ int cv1; - cv1 = offset + (x1 - 1) * A + y1; - Ec++; - N->edges[Ec].from = cv; - N->edges[Ec].to = cv1; - N->edges[Ec].cap = C2AA; - return; -} - -static network *gen_rmf(struct csa *csa, int a, int b, int c1, int c2) -{ /* generates a network with a*a*b nodes and 6a*a*b-4ab-2a*a edges - random_frame network: - Derigs & Meier, Methods & Models of OR (1989), 33:383-403 */ - int x, y, z, offset, cv; - A = a; - AA = a * a; - C2AA = c2 * AA; - Ec = 0; - N = (network *)xmalloc(sizeof(network)); - N->vertnum = AA * b; - N->edgenum = 5 * AA * b - 4 * A * b - AA; - N->edges = (edge *)xcalloc(N->edgenum + 1, sizeof(edge)); - N->source = 1; - N->sink = N->vertnum; - Parr = (int *)xcalloc(AA + 1, sizeof(int)); - for (x = 1; x <= AA; x++) - Parr[x] = x; - for (z = 1; z <= b; z++) - { offset = AA * (z - 1); - if (z != b) - permute(csa); - for (x = 1; x <= A; x++) - { for (y = 1; y <= A; y++) - { cv = offset + (x - 1) * A + y; - if (z != b) - make_edge(csa, cv, offset + AA + Parr[cv - offset], - c1, c2); /* the intermediate edges */ - if (y < A) - connect(csa, offset, cv, x, y + 1); - if (y > 1) - connect(csa, offset, cv, x, y - 1); - if (x < A) - connect(csa, offset, cv, x + 1, y); - if (x > 1) - connect(csa, offset, cv, x - 1, y); - } - } - } - xfree(Parr); - return N; -} - -static void print_max_format(struct csa *csa, network *n, char *comm[], - int dim) -{ /* prints a network heading with dim lines of comments (no \n - needs at the ends) */ - int i, vnum, e_num; - edge *e; - vnum = n->vertnum; - e_num = n->edgenum; - if (G == NULL) - { for (i = 0; i < dim; i++) - xprintf("c %s\n", comm[i]); - xprintf("p max %7d %10d\n", vnum, e_num); - xprintf("n %7d s\n", n->source); - xprintf("n %7d t\n", n->sink); - } - else - { glp_add_vertices(G, vnum); - if (s != NULL) *s = n->source; - if (t != NULL) *t = n->sink; - } - for (i = 1; i <= e_num; i++) - { e = &n->edges[i]; - if (G == NULL) - xprintf("a %7d %7d %10d\n", e->from, e->to, (int)e->cap); - else - { glp_arc *a = glp_add_arc(G, e->from, e->to); - if (a_cap >= 0) - { double temp = (double)e->cap; - memcpy((char *)a->data + a_cap, &temp, sizeof(double)); - } - } - } - return; -} - -static void gen_free_net(network *n) -{ xfree(n->edges); - xfree(n); - return; -} - -int glp_rmfgen(glp_graph *G_, int *_s, int *_t, int _a_cap, - const int parm[1+5]) -{ struct csa _csa, *csa = &_csa; - network *n; - char comm[10][80], *com1[10]; - int seed, a, b, c1, c2, ret; - G = G_; - s = _s; - t = _t; - a_cap = _a_cap; - if (G != NULL) - { if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_rmfgen: a_cap = %d; invalid offset\n", a_cap); - } - seed = parm[1]; - a = parm[2]; - b = parm[3]; - c1 = parm[4]; - c2 = parm[5]; - if (!(seed > 0 && 1 <= a && a <= 1000 && 1 <= b && b <= 1000 && - 0 <= c1 && c1 <= c2 && c2 <= 1000)) - { ret = 1; - goto done; - } - if (G != NULL) - { glp_erase_graph(G, G->v_size, G->a_size); - glp_set_graph_name(G, "RMFGEN"); - } - csa->rand = rng_create_rand(); - rng_init_rand(csa->rand, seed); - n = gen_rmf(csa, a, b, c1, c2); - sprintf(comm[0], "This file was generated by genrmf."); - sprintf(comm[1], "The parameters are: a: %d b: %d c1: %d c2: %d", - a, b, c1, c2); - com1[0] = comm[0]; - com1[1] = comm[1]; - print_max_format(csa, n, com1, 2); - gen_free_net(n); - rng_delete_rand(csa->rand); - ret = 0; -done: return ret; -} - -/**********************************************************************/ - -#if 0 -int main(int argc, char *argv[]) -{ int seed, a, b, c1, c2, i, parm[1+5]; - seed = 123; - a = b = c1 = c2 = -1; - for (i = 1; i < argc; i++) - { if (strcmp(argv[i], "-seed") == 0) - seed = atoi(argv[++i]); - else if (strcmp(argv[i], "-a") == 0) - a = atoi(argv[++i]); - else if (strcmp(argv[i], "-b") == 0) - b = atoi(argv[++i]); - else if (strcmp(argv[i], "-c1") == 0) - c1 = atoi(argv[++i]); - else if (strcmp(argv[i], "-c2") == 0) - c2 = atoi(argv[++i]); - } - if (a < 0 || b < 0 || c1 < 0 || c2 < 0) - { xprintf("Usage:\n"); - xprintf("genrmf [-seed seed] -a frame_size -b depth\n"); - xprintf(" -c1 cap_range1 -c2 cap_range2\n"); - } - else - { parm[1] = seed; - parm[2] = a; - parm[3] = b; - parm[4] = c1; - parm[5] = c2; - glp_rmfgen(NULL, NULL, NULL, 0, parm); - } - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/api/strong.c b/code/3rd_glpk/api/strong.c deleted file mode 100644 index 9ddcacfb..00000000 --- a/code/3rd_glpk/api/strong.c +++ /dev/null @@ -1,110 +0,0 @@ -/* strong.c (find all strongly connected components of graph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "mc13d.h" - -/*********************************************************************** -* NAME -* -* glp_strong_comp - find all strongly connected components of graph -* -* SYNOPSIS -* -* int glp_strong_comp(glp_graph *G, int v_num); -* -* DESCRIPTION -* -* The routine glp_strong_comp finds all strongly connected components -* of the specified graph. -* -* The parameter v_num specifies an offset of the field of type int -* in the vertex data block, to which the routine stores the number of -* a strongly connected component containing that vertex. If v_num < 0, -* no component numbers are stored. -* -* The components are numbered in arbitrary order from 1 to nc, where -* nc is the total number of components found, 0 <= nc <= |V|. However, -* the component numbering has the property that for every arc (i->j) -* in the graph the condition num(i) >= num(j) holds. -* -* RETURNS -* -* The routine returns nc, the total number of components found. */ - -int glp_strong_comp(glp_graph *G, int v_num) -{ glp_vertex *v; - glp_arc *a; - int i, k, last, n, na, nc, *icn, *ip, *lenr, *ior, *ib, *lowl, - *numb, *prev; - if (v_num >= 0 && v_num > G->v_size - (int)sizeof(int)) - xerror("glp_strong_comp: v_num = %d; invalid offset\n", - v_num); - n = G->nv; - if (n == 0) - { nc = 0; - goto done; - } - na = G->na; - icn = xcalloc(1+na, sizeof(int)); - ip = xcalloc(1+n, sizeof(int)); - lenr = xcalloc(1+n, sizeof(int)); - ior = xcalloc(1+n, sizeof(int)); - ib = xcalloc(1+n, sizeof(int)); - lowl = xcalloc(1+n, sizeof(int)); - numb = xcalloc(1+n, sizeof(int)); - prev = xcalloc(1+n, sizeof(int)); - k = 1; - for (i = 1; i <= n; i++) - { v = G->v[i]; - ip[i] = k; - for (a = v->out; a != NULL; a = a->t_next) - icn[k++] = a->head->i; - lenr[i] = k - ip[i]; - } - xassert(na == k-1); - nc = mc13d(n, icn, ip, lenr, ior, ib, lowl, numb, prev); - if (v_num >= 0) - { xassert(ib[1] == 1); - for (k = 1; k <= nc; k++) - { last = (k < nc ? ib[k+1] : n+1); - xassert(ib[k] < last); - for (i = ib[k]; i < last; i++) - { v = G->v[ior[i]]; - memcpy((char *)v->data + v_num, &k, sizeof(int)); - } - } - } - xfree(icn); - xfree(ip); - xfree(lenr); - xfree(ior); - xfree(ib); - xfree(lowl); - xfree(numb); - xfree(prev); -done: return nc; -} - -/* eof */ diff --git a/code/3rd_glpk/api/topsort.c b/code/3rd_glpk/api/topsort.c deleted file mode 100644 index 971937f2..00000000 --- a/code/3rd_glpk/api/topsort.c +++ /dev/null @@ -1,123 +0,0 @@ -/* topsort.c (topological sorting of acyclic digraph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_top_sort - topological sorting of acyclic digraph -* -* SYNOPSIS -* -* int glp_top_sort(glp_graph *G, int v_num); -* -* DESCRIPTION -* -* The routine glp_top_sort performs topological sorting of vertices of -* the specified acyclic digraph. -* -* The parameter v_num specifies an offset of the field of type int in -* the vertex data block, to which the routine stores the vertex number -* assigned. If v_num < 0, vertex numbers are not stored. -* -* The vertices are numbered from 1 to n, where n is the total number -* of vertices in the graph. The vertex numbering has the property that -* for every arc (i->j) in the graph the condition num(i) < num(j) -* holds. Special case num(i) = 0 means that vertex i is not assigned a -* number, because the graph is *not* acyclic. -* -* RETURNS -* -* If the graph is acyclic and therefore all the vertices have been -* assigned numbers, the routine glp_top_sort returns zero. Otherwise, -* if the graph is not acyclic, the routine returns the number of -* vertices which have not been numbered, i.e. for which num(i) = 0. */ - -static int top_sort(glp_graph *G, int num[]) -{ glp_arc *a; - int i, j, cnt, top, *stack, *indeg; - /* allocate working arrays */ - indeg = xcalloc(1+G->nv, sizeof(int)); - stack = xcalloc(1+G->nv, sizeof(int)); - /* determine initial indegree of each vertex; push into the stack - the vertices having zero indegree */ - top = 0; - for (i = 1; i <= G->nv; i++) - { num[i] = indeg[i] = 0; - for (a = G->v[i]->in; a != NULL; a = a->h_next) - indeg[i]++; - if (indeg[i] == 0) - stack[++top] = i; - } - /* assign numbers to vertices in the sorted order */ - cnt = 0; - while (top > 0) - { /* pull vertex i from the stack */ - i = stack[top--]; - /* it has zero indegree in the current graph */ - xassert(indeg[i] == 0); - /* so assign it a next number */ - xassert(num[i] == 0); - num[i] = ++cnt; - /* remove vertex i from the current graph, update indegree of - its adjacent vertices, and push into the stack new vertices - whose indegree becomes zero */ - for (a = G->v[i]->out; a != NULL; a = a->t_next) - { j = a->head->i; - /* there exists arc (i->j) in the graph */ - xassert(indeg[j] > 0); - indeg[j]--; - if (indeg[j] == 0) - stack[++top] = j; - } - } - /* free working arrays */ - xfree(indeg); - xfree(stack); - return G->nv - cnt; -} - -int glp_top_sort(glp_graph *G, int v_num) -{ glp_vertex *v; - int i, cnt, *num; - if (v_num >= 0 && v_num > G->v_size - (int)sizeof(int)) - xerror("glp_top_sort: v_num = %d; invalid offset\n", v_num); - if (G->nv == 0) - { cnt = 0; - goto done; - } - num = xcalloc(1+G->nv, sizeof(int)); - cnt = top_sort(G, num); - if (v_num >= 0) - { for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - memcpy((char *)v->data + v_num, &num[i], sizeof(int)); - } - } - xfree(num); -done: return cnt; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wcliqex.c b/code/3rd_glpk/api/wcliqex.c deleted file mode 100644 index 53c2d521..00000000 --- a/code/3rd_glpk/api/wcliqex.c +++ /dev/null @@ -1,122 +0,0 @@ -/* wcliqex.c (find maximum weight clique with exact algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" -#include "wclique.h" - -static void set_edge(int nv, unsigned char a[], int i, int j) -{ int k; - xassert(1 <= j && j < i && i <= nv); - k = ((i - 1) * (i - 2)) / 2 + (j - 1); - a[k / CHAR_BIT] |= - (unsigned char)(1 << ((CHAR_BIT - 1) - k % CHAR_BIT)); - return; -} - -int glp_wclique_exact(glp_graph *G, int v_wgt, double *sol, int v_set) -{ /* find maximum weight clique with exact algorithm */ - glp_arc *e; - int i, j, k, len, x, *w, *ind, ret = 0; - unsigned char *a; - double s, t; - if (v_wgt >= 0 && v_wgt > G->v_size - (int)sizeof(double)) - xerror("glp_wclique_exact: v_wgt = %d; invalid parameter\n", - v_wgt); - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_wclique_exact: v_set = %d; invalid parameter\n", - v_set); - if (G->nv == 0) - { /* empty graph has only empty clique */ - if (sol != NULL) *sol = 0.0; - return 0; - } - /* allocate working arrays */ - w = xcalloc(1+G->nv, sizeof(int)); - ind = xcalloc(1+G->nv, sizeof(int)); - len = G->nv; /* # vertices */ - len = len * (len - 1) / 2; /* # entries in lower triangle */ - len = (len + (CHAR_BIT - 1)) / CHAR_BIT; /* # bytes needed */ - a = xcalloc(len, sizeof(char)); - memset(a, 0, len * sizeof(char)); - /* determine vertex weights */ - s = 0.0; - for (i = 1; i <= G->nv; i++) - { if (v_wgt >= 0) - { memcpy(&t, (char *)G->v[i]->data + v_wgt, sizeof(double)); - if (!(0.0 <= t && t <= (double)INT_MAX && t == floor(t))) - { ret = GLP_EDATA; - goto done; - } - w[i] = (int)t; - } - else - w[i] = 1; - s += (double)w[i]; - } - if (s > (double)INT_MAX) - { ret = GLP_EDATA; - goto done; - } - /* build the adjacency matrix */ - for (i = 1; i <= G->nv; i++) - { for (e = G->v[i]->in; e != NULL; e = e->h_next) - { j = e->tail->i; - /* there exists edge (j,i) in the graph */ - if (i > j) set_edge(G->nv, a, i, j); - } - for (e = G->v[i]->out; e != NULL; e = e->t_next) - { j = e->head->i; - /* there exists edge (i,j) in the graph */ - if (i > j) set_edge(G->nv, a, i, j); - } - } - /* find maximum weight clique in the graph */ - len = wclique(G->nv, w, a, ind); - /* compute the clique weight */ - s = 0.0; - for (k = 1; k <= len; k++) - { i = ind[k]; - xassert(1 <= i && i <= G->nv); - s += (double)w[i]; - } - if (sol != NULL) *sol = s; - /* mark vertices included in the clique */ - if (v_set >= 0) - { x = 0; - for (i = 1; i <= G->nv; i++) - memcpy((char *)G->v[i]->data + v_set, &x, sizeof(int)); - x = 1; - for (k = 1; k <= len; k++) - { i = ind[k]; - memcpy((char *)G->v[i]->data + v_set, &x, sizeof(int)); - } - } -done: /* free working arrays */ - xfree(w); - xfree(ind); - xfree(a); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/weak.c b/code/3rd_glpk/api/weak.c deleted file mode 100644 index 027c09c1..00000000 --- a/code/3rd_glpk/api/weak.c +++ /dev/null @@ -1,150 +0,0 @@ -/* weak.c (find all weakly connected components of graph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -/*********************************************************************** -* NAME -* -* glp_weak_comp - find all weakly connected components of graph -* -* SYNOPSIS -* -* int glp_weak_comp(glp_graph *G, int v_num); -* -* DESCRIPTION -* -* The routine glp_weak_comp finds all weakly connected components of -* the specified graph. -* -* The parameter v_num specifies an offset of the field of type int -* in the vertex data block, to which the routine stores the number of -* a (weakly) connected component containing that vertex. If v_num < 0, -* no component numbers are stored. -* -* The components are numbered in arbitrary order from 1 to nc, where -* nc is the total number of components found, 0 <= nc <= |V|. -* -* RETURNS -* -* The routine returns nc, the total number of components found. */ - -int glp_weak_comp(glp_graph *G, int v_num) -{ glp_vertex *v; - glp_arc *a; - int f, i, j, nc, nv, pos1, pos2, *prev, *next, *list; - if (v_num >= 0 && v_num > G->v_size - (int)sizeof(int)) - xerror("glp_weak_comp: v_num = %d; invalid offset\n", v_num); - nv = G->nv; - if (nv == 0) - { nc = 0; - goto done; - } - /* allocate working arrays */ - prev = xcalloc(1+nv, sizeof(int)); - next = xcalloc(1+nv, sizeof(int)); - list = xcalloc(1+nv, sizeof(int)); - /* if vertex i is unlabelled, prev[i] is the index of previous - unlabelled vertex, and next[i] is the index of next unlabelled - vertex; if vertex i is labelled, then prev[i] < 0, and next[i] - is the connected component number */ - /* initially all vertices are unlabelled */ - f = 1; - for (i = 1; i <= nv; i++) - prev[i] = i - 1, next[i] = i + 1; - next[nv] = 0; - /* main loop (until all vertices have been labelled) */ - nc = 0; - while (f != 0) - { /* take an unlabelled vertex */ - i = f; - /* and remove it from the list of unlabelled vertices */ - f = next[i]; - if (f != 0) prev[f] = 0; - /* label the vertex; it begins a new component */ - prev[i] = -1, next[i] = ++nc; - /* breadth first search */ - list[1] = i, pos1 = pos2 = 1; - while (pos1 <= pos2) - { /* dequeue vertex i */ - i = list[pos1++]; - /* consider all arcs incoming to vertex i */ - for (a = G->v[i]->in; a != NULL; a = a->h_next) - { /* vertex j is adjacent to vertex i */ - j = a->tail->i; - if (prev[j] >= 0) - { /* vertex j is unlabelled */ - /* remove it from the list of unlabelled vertices */ - if (prev[j] == 0) - f = next[j]; - else - next[prev[j]] = next[j]; - if (next[j] == 0) - ; - else - prev[next[j]] = prev[j]; - /* label the vertex */ - prev[j] = -1, next[j] = nc; - /* and enqueue it for further consideration */ - list[++pos2] = j; - } - } - /* consider all arcs outgoing from vertex i */ - for (a = G->v[i]->out; a != NULL; a = a->t_next) - { /* vertex j is adjacent to vertex i */ - j = a->head->i; - if (prev[j] >= 0) - { /* vertex j is unlabelled */ - /* remove it from the list of unlabelled vertices */ - if (prev[j] == 0) - f = next[j]; - else - next[prev[j]] = next[j]; - if (next[j] == 0) - ; - else - prev[next[j]] = prev[j]; - /* label the vertex */ - prev[j] = -1, next[j] = nc; - /* and enqueue it for further consideration */ - list[++pos2] = j; - } - } - } - } - /* store component numbers */ - if (v_num >= 0) - { for (i = 1; i <= nv; i++) - { v = G->v[i]; - memcpy((char *)v->data + v_num, &next[i], sizeof(int)); - } - } - /* free working arrays */ - xfree(prev); - xfree(next); - xfree(list); -done: return nc; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrasn.c b/code/3rd_glpk/api/wrasn.c deleted file mode 100644 index 81433da8..00000000 --- a/code/3rd_glpk/api/wrasn.c +++ /dev/null @@ -1,107 +0,0 @@ -/* wrasn.c (write assignment problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_write_asnprob - write assignment problem data in DIMACS format -* -* SYNOPSIS -* -* int glp_write_asnprob(glp_graph *G, int v_set, int a_cost, -* const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_asnprob writes assignment problem data in -* DIMACS format to a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_asnprob(glp_graph *G, int v_set, int a_cost, const char - *fname) -{ glp_file *fp; - glp_vertex *v; - glp_arc *a; - int i, k, count = 0, ret; - double cost; - if (v_set >= 0 && v_set > G->v_size - (int)sizeof(int)) - xerror("glp_write_asnprob: v_set = %d; invalid offset\n", - v_set); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_write_asnprob: a_cost = %d; invalid offset\n", - a_cost); - xprintf("Writing assignment problem data to '%s'...\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "c %s\n", - G->name == NULL ? "unknown" : G->name), count++; - xfprintf(fp, "p asn %d %d\n", G->nv, G->na), count++; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - if (v_set >= 0) - memcpy(&k, (char *)v->data + v_set, sizeof(int)); - else - k = (v->out != NULL ? 0 : 1); - if (k == 0) - xfprintf(fp, "n %d\n", i), count++; - } - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { if (a_cost >= 0) - memcpy(&cost, (char *)a->data + a_cost, sizeof(double)); - else - cost = 1.0; - xfprintf(fp, "a %d %d %.*g\n", - a->tail->i, a->head->i, DBL_DIG, cost), count++; - } - } - xfprintf(fp, "c eof\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrcc.c b/code/3rd_glpk/api/wrcc.c deleted file mode 100644 index 2069c8ac..00000000 --- a/code/3rd_glpk/api/wrcc.c +++ /dev/null @@ -1,102 +0,0 @@ -/* wrcc.c (write graph in DIMACS clique/coloring format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_write_ccdata - write graph in DIMACS clique/coloring format -* -* SYNOPSIS -* -* int glp_write_ccdata(glp_graph *G, int v_wgt, const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_ccdata writes the specified graph in DIMACS -* clique/coloring format to a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_ccdata(glp_graph *G, int v_wgt, const char *fname) -{ glp_file *fp; - glp_vertex *v; - glp_arc *e; - int i, count = 0, ret; - double w; - if (v_wgt >= 0 && v_wgt > G->v_size - (int)sizeof(double)) - xerror("glp_write_ccdata: v_wgt = %d; invalid offset\n", - v_wgt); - xprintf("Writing graph to '%s'\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "c %s\n", - G->name == NULL ? "unknown" : G->name), count++; - xfprintf(fp, "p edge %d %d\n", G->nv, G->na), count++; - if (v_wgt >= 0) - { for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - memcpy(&w, (char *)v->data + v_wgt, sizeof(double)); - if (w != 1.0) - xfprintf(fp, "n %d %.*g\n", i, DBL_DIG, w), count++; - } - } - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (e = v->out; e != NULL; e = e->t_next) - xfprintf(fp, "e %d %d\n", e->tail->i, e->head->i), count++; - } - xfprintf(fp, "c eof\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/**********************************************************************/ - -int glp_write_graph(glp_graph *G, const char *fname) -{ return - glp_write_ccdata(G, -1, fname); -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrcnf.c b/code/3rd_glpk/api/wrcnf.c deleted file mode 100644 index c7974386..00000000 --- a/code/3rd_glpk/api/wrcnf.c +++ /dev/null @@ -1,87 +0,0 @@ -/* wrcnf.c (write CNF-SAT problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -#define xfprintf glp_format - -int glp_write_cnfsat(glp_prob *P, const char *fname) -{ /* write CNF-SAT problem data in DIMACS format */ - glp_file *fp = NULL; - GLPAIJ *aij; - int i, j, len, count = 0, ret; - char s[50]; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_write_cnfsat: P = %p; invalid problem object\n", - P); -#endif - if (glp_check_cnfsat(P) != 0) - { xprintf("glp_write_cnfsat: problem object does not encode CNF-" - "SAT instance\n"); - ret = 1; - goto done; - } - xprintf("Writing CNF-SAT problem data to '%s'...\n", fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "c %s\n", - P->name == NULL ? "unknown" : P->name), count++; - xfprintf(fp, "p cnf %d %d\n", P->n, P->m), count++; - for (i = 1; i <= P->m; i++) - { len = 0; - for (aij = P->row[i]->ptr; aij != NULL; aij = aij->r_next) - { j = aij->col->j; - if (aij->val < 0.0) j = -j; - sprintf(s, "%d", j); - if (len > 0 && len + 1 + strlen(s) > 72) - xfprintf(fp, "\n"), count++, len = 0; - xfprintf(fp, "%s%s", len == 0 ? "" : " ", s); - if (len > 0) len++; - len += strlen(s); - } - if (len > 0 && len + 1 + 1 > 72) - xfprintf(fp, "\n"), count++, len = 0; - xfprintf(fp, "%s0\n", len == 0 ? "" : " "), count++; - } - xfprintf(fp, "c eof\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wript.c b/code/3rd_glpk/api/wript.c deleted file mode 100644 index f2ca802c..00000000 --- a/code/3rd_glpk/api/wript.c +++ /dev/null @@ -1,124 +0,0 @@ -/* wript.c (write interior-point solution in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_write_ipt - write interior-point solution in GLPK format -* -* SYNOPSIS -* -* int glp_write_ipt(glp_prob *P, const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_ipt writes interior-point solution to a text -* file in GLPK format. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_ipt(glp_prob *P, const char *fname) -{ glp_file *fp; - GLPROW *row; - GLPCOL *col; - int i, j, count, ret = 1; - char *s; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_write_ipt: P = %p; invalid problem object\n", P); -#endif - if (fname == NULL) - xerror("glp_write_ipt: fname = %d; invalid parameter\n", fname) - ; - xprintf("Writing interior-point solution to '%s'...\n", fname); - fp = glp_open(fname, "w"), count = 0; - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* write comment lines */ - glp_format(fp, "c %-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name), count++; - glp_format(fp, "c %-12s%d\n", "Rows:", P->m), count++; - glp_format(fp, "c %-12s%d\n", "Columns:", P->n), count++; - glp_format(fp, "c %-12s%d\n", "Non-zeros:", P->nnz), count++; - switch (P->ipt_stat) - { case GLP_OPT: s = "OPTIMAL"; break; - case GLP_INFEAS: s = "INFEASIBLE (INTERMEDIATE)"; break; - case GLP_NOFEAS: s = "INFEASIBLE (FINAL)"; break; - case GLP_UNDEF: s = "UNDEFINED"; break; - default: s = "???"; break; - } - glp_format(fp, "c %-12s%s\n", "Status:", s), count++; - switch (P->dir) - { case GLP_MIN: s = "MINimum"; break; - case GLP_MAX: s = "MAXimum"; break; - default: s = "???"; break; - } - glp_format(fp, "c %-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->ipt_obj, s), count++; - glp_format(fp, "c\n"), count++; - /* write solution line */ - glp_format(fp, "s ipt %d %d ", P->m, P->n), count++; - switch (P->ipt_stat) - { case GLP_OPT: glp_format(fp, "o"); break; - case GLP_INFEAS: glp_format(fp, "i"); break; - case GLP_NOFEAS: glp_format(fp, "n"); break; - case GLP_UNDEF: glp_format(fp, "u"); break; - default: glp_format(fp, "?"); break; - } - glp_format(fp, " %.*g\n", DBL_DIG, P->ipt_obj); - /* write row solution descriptor lines */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - glp_format(fp, "i %d %.*g %.*g\n", i, DBL_DIG, row->pval, - DBL_DIG, row->dval), count++; - } - /* write column solution descriptor lines */ - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - glp_format(fp, "j %d %.*g %.*g\n", j, DBL_DIG, col->pval, - DBL_DIG, col->dval), count++; - } - /* write end line */ - glp_format(fp, "e o f\n"), count++; - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* interior-point solution has been successfully written */ - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) - glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrmaxf.c b/code/3rd_glpk/api/wrmaxf.c deleted file mode 100644 index d3101ca8..00000000 --- a/code/3rd_glpk/api/wrmaxf.c +++ /dev/null @@ -1,104 +0,0 @@ -/* wrmaxf.c (write maximum flow problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_write_maxflow - write maximum flow problem data in DIMACS format -* -* SYNOPSIS -* -* int glp_write_maxflow(glp_graph *G, int s, int t, int a_cap, -* const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_maxflow writes maximum flow problem data in -* DIMACS format to a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_maxflow(glp_graph *G, int s, int t, int a_cap, - const char *fname) -{ glp_file *fp; - glp_vertex *v; - glp_arc *a; - int i, count = 0, ret; - double cap; - if (!(1 <= s && s <= G->nv)) - xerror("glp_write_maxflow: s = %d; source node number out of r" - "ange\n", s); - if (!(1 <= t && t <= G->nv)) - xerror("glp_write_maxflow: t = %d: sink node number out of ran" - "ge\n", t); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_write_mincost: a_cap = %d; invalid offset\n", - a_cap); - xprintf("Writing maximum flow problem data to '%s'...\n", - fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "c %s\n", - G->name == NULL ? "unknown" : G->name), count++; - xfprintf(fp, "p max %d %d\n", G->nv, G->na), count++; - xfprintf(fp, "n %d s\n", s), count++; - xfprintf(fp, "n %d t\n", t), count++; - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { if (a_cap >= 0) - memcpy(&cap, (char *)a->data + a_cap, sizeof(double)); - else - cap = 1.0; - xfprintf(fp, "a %d %d %.*g\n", - a->tail->i, a->head->i, DBL_DIG, cap), count++; - } - } - xfprintf(fp, "c eof\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrmcf.c b/code/3rd_glpk/api/wrmcf.c deleted file mode 100644 index 0da37f42..00000000 --- a/code/3rd_glpk/api/wrmcf.c +++ /dev/null @@ -1,122 +0,0 @@ -/* wrmcf.c (write min-cost flow problem data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpk.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_write_mincost - write min-cost flow probl. data in DIMACS format -* -* SYNOPSIS -* -* int glp_write_mincost(glp_graph *G, int v_rhs, int a_low, int a_cap, -* int a_cost, const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_mincost writes minimum cost flow problem data -* in DIMACS format to a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_mincost(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, const char *fname) -{ glp_file *fp; - glp_vertex *v; - glp_arc *a; - int i, count = 0, ret; - double rhs, low, cap, cost; - if (v_rhs >= 0 && v_rhs > G->v_size - (int)sizeof(double)) - xerror("glp_write_mincost: v_rhs = %d; invalid offset\n", - v_rhs); - if (a_low >= 0 && a_low > G->a_size - (int)sizeof(double)) - xerror("glp_write_mincost: a_low = %d; invalid offset\n", - a_low); - if (a_cap >= 0 && a_cap > G->a_size - (int)sizeof(double)) - xerror("glp_write_mincost: a_cap = %d; invalid offset\n", - a_cap); - if (a_cost >= 0 && a_cost > G->a_size - (int)sizeof(double)) - xerror("glp_write_mincost: a_cost = %d; invalid offset\n", - a_cost); - xprintf("Writing min-cost flow problem data to '%s'...\n", - fname); - fp = glp_open(fname, "w"); - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xfprintf(fp, "c %s\n", - G->name == NULL ? "unknown" : G->name), count++; - xfprintf(fp, "p min %d %d\n", G->nv, G->na), count++; - if (v_rhs >= 0) - { for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - memcpy(&rhs, (char *)v->data + v_rhs, sizeof(double)); - if (rhs != 0.0) - xfprintf(fp, "n %d %.*g\n", i, DBL_DIG, rhs), count++; - } - } - for (i = 1; i <= G->nv; i++) - { v = G->v[i]; - for (a = v->out; a != NULL; a = a->t_next) - { if (a_low >= 0) - memcpy(&low, (char *)a->data + a_low, sizeof(double)); - else - low = 0.0; - if (a_cap >= 0) - memcpy(&cap, (char *)a->data + a_cap, sizeof(double)); - else - cap = 1.0; - if (a_cost >= 0) - memcpy(&cost, (char *)a->data + a_cost, sizeof(double)); - else - cost = 0.0; - xfprintf(fp, "a %d %d %.*g %.*g %.*g\n", - a->tail->i, a->head->i, DBL_DIG, low, DBL_DIG, cap, - DBL_DIG, cost), count++; - } - } - xfprintf(fp, "c eof\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrmip.c b/code/3rd_glpk/api/wrmip.c deleted file mode 100644 index 407a5fec..00000000 --- a/code/3rd_glpk/api/wrmip.c +++ /dev/null @@ -1,122 +0,0 @@ -/* wrmip.c (write MIP solution in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_write_mip - write MIP solution in GLPK format -* -* SYNOPSIS -* -* int glp_write_mip(glp_prob *P, const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_mip writes MIP solution to a text file in GLPK -* format. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_mip(glp_prob *P, const char *fname) -{ glp_file *fp; - GLPROW *row; - GLPCOL *col; - int i, j, count, ret = 1; - char *s; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_write_mip: P = %p; invalid problem object\n", P); -#endif - if (fname == NULL) - xerror("glp_write_mip: fname = %d; invalid parameter\n", fname) - ; - xprintf("Writing MIP solution to '%s'...\n", fname); - fp = glp_open(fname, "w"), count = 0; - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* write comment lines */ - glp_format(fp, "c %-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name), count++; - glp_format(fp, "c %-12s%d\n", "Rows:", P->m), count++; - glp_format(fp, "c %-12s%d\n", "Columns:", P->n), count++; - glp_format(fp, "c %-12s%d\n", "Non-zeros:", P->nnz), count++; - switch (P->mip_stat) - { case GLP_OPT: s = "INTEGER OPTIMAL"; break; - case GLP_FEAS: s = "INTEGER NON-OPTIMAL"; break; - case GLP_NOFEAS: s = "INTEGER EMPTY"; break; - case GLP_UNDEF: s = "INTEGER UNDEFINED"; break; - default: s = "???"; break; - } - glp_format(fp, "c %-12s%s\n", "Status:", s), count++; - switch (P->dir) - { case GLP_MIN: s = "MINimum"; break; - case GLP_MAX: s = "MAXimum"; break; - default: s = "???"; break; - } - glp_format(fp, "c %-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->mip_obj, s), count++; - glp_format(fp, "c\n"), count++; - /* write solution line */ - glp_format(fp, "s mip %d %d ", P->m, P->n), count++; - switch (P->mip_stat) - { case GLP_OPT: glp_format(fp, "o"); break; - case GLP_FEAS: glp_format(fp, "f"); break; - case GLP_NOFEAS: glp_format(fp, "n"); break; - case GLP_UNDEF: glp_format(fp, "u"); break; - default: glp_format(fp, "?"); break; - } - glp_format(fp, " %.*g\n", DBL_DIG, P->mip_obj); - /* write row solution descriptor lines */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - glp_format(fp, "i %d %.*g\n", i, DBL_DIG, row->mipx), count++; - } - /* write column solution descriptor lines */ - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - glp_format(fp, "j %d %.*g\n", j, DBL_DIG, col->mipx), count++; - } - /* write end line */ - glp_format(fp, "e o f\n"), count++; - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* MIP solution has been successfully written */ - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) - glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrprob.c b/code/3rd_glpk/api/wrprob.c deleted file mode 100644 index 99983d35..00000000 --- a/code/3rd_glpk/api/wrprob.c +++ /dev/null @@ -1,166 +0,0 @@ -/* wrprob.c (write problem data in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -#define xfprintf glp_format - -/*********************************************************************** -* NAME -* -* glp_write_prob - write problem data in GLPK format -* -* SYNOPSIS -* -* int glp_write_prob(glp_prob *P, int flags, const char *fname); -* -* The routine glp_write_prob writes problem data in GLPK LP/MIP format -* to a text file. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_prob(glp_prob *P, int flags, const char *fname) -{ glp_file *fp; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int mip, i, j, count, ret; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_write_prob: P = %p; invalid problem object\n", - P); -#endif - if (flags != 0) - xerror("glp_write_prob: flags = %d; invalid parameter\n", - flags); - if (fname == NULL) - xerror("glp_write_prob: fname = %d; invalid parameter\n", - fname); - xprintf("Writing problem data to '%s'...\n", fname); - fp = glp_open(fname, "w"), count = 0; - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - /* write problem line */ - mip = (glp_get_num_int(P) > 0); - xfprintf(fp, "p %s %s %d %d %d\n", !mip ? "lp" : "mip", - P->dir == GLP_MIN ? "min" : P->dir == GLP_MAX ? "max" : "???", - P->m, P->n, P->nnz), count++; - if (P->name != NULL) - xfprintf(fp, "n p %s\n", P->name), count++; - if (P->obj != NULL) - xfprintf(fp, "n z %s\n", P->obj), count++; - /* write row descriptors */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->type == GLP_FX && row->lb == 0.0) - goto skip1; - xfprintf(fp, "i %d ", i), count++; - if (row->type == GLP_FR) - xfprintf(fp, "f\n"); - else if (row->type == GLP_LO) - xfprintf(fp, "l %.*g\n", DBL_DIG, row->lb); - else if (row->type == GLP_UP) - xfprintf(fp, "u %.*g\n", DBL_DIG, row->ub); - else if (row->type == GLP_DB) - xfprintf(fp, "d %.*g %.*g\n", DBL_DIG, row->lb, DBL_DIG, - row->ub); - else if (row->type == GLP_FX) - xfprintf(fp, "s %.*g\n", DBL_DIG, row->lb); - else - xassert(row != row); -skip1: if (row->name != NULL) - xfprintf(fp, "n i %d %s\n", i, row->name), count++; - } - /* write column descriptors */ - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (!mip && col->type == GLP_LO && col->lb == 0.0) - goto skip2; - if (mip && col->kind == GLP_IV && col->type == GLP_DB && - col->lb == 0.0 && col->ub == 1.0) - goto skip2; - xfprintf(fp, "j %d ", j), count++; - if (mip) - { if (col->kind == GLP_CV) - xfprintf(fp, "c "); - else if (col->kind == GLP_IV) - xfprintf(fp, "i "); - else - xassert(col != col); - } - if (col->type == GLP_FR) - xfprintf(fp, "f\n"); - else if (col->type == GLP_LO) - xfprintf(fp, "l %.*g\n", DBL_DIG, col->lb); - else if (col->type == GLP_UP) - xfprintf(fp, "u %.*g\n", DBL_DIG, col->ub); - else if (col->type == GLP_DB) - xfprintf(fp, "d %.*g %.*g\n", DBL_DIG, col->lb, DBL_DIG, - col->ub); - else if (col->type == GLP_FX) - xfprintf(fp, "s %.*g\n", DBL_DIG, col->lb); - else - xassert(col != col); -skip2: if (col->name != NULL) - xfprintf(fp, "n j %d %s\n", j, col->name), count++; - } - /* write objective coefficient descriptors */ - if (P->c0 != 0.0) - xfprintf(fp, "a 0 0 %.*g\n", DBL_DIG, P->c0), count++; - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->coef != 0.0) - xfprintf(fp, "a 0 %d %.*g\n", j, DBL_DIG, col->coef), - count++; - } - /* write constraint coefficient descriptors */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - xfprintf(fp, "a %d %d %.*g\n", i, aij->col->j, DBL_DIG, - aij->val), count++; - } - /* write end line */ - xfprintf(fp, "e o f\n"), count++; -#if 0 /* FIXME */ - xfflush(fp); -#endif - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - ret = 1; - goto done; - } - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/api/wrsol.c b/code/3rd_glpk/api/wrsol.c deleted file mode 100644 index 66c69233..00000000 --- a/code/3rd_glpk/api/wrsol.c +++ /dev/null @@ -1,174 +0,0 @@ -/* wrsol.c (write basic solution in GLPK format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2010-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_write_sol - write basic solution in GLPK format -* -* SYNOPSIS -* -* int glp_write_sol(glp_prob *P, const char *fname); -* -* DESCRIPTION -* -* The routine glp_write_sol writes basic solution to a text file in -* GLPK format. -* -* RETURNS -* -* If the operation was successful, the routine returns zero. Otherwise -* it prints an error message and returns non-zero. */ - -int glp_write_sol(glp_prob *P, const char *fname) -{ glp_file *fp; - GLPROW *row; - GLPCOL *col; - int i, j, count, ret = 1; - char *s; -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_write_sol: P = %p; invalid problem object\n", P); -#endif - if (fname == NULL) - xerror("glp_write_sol: fname = %d; invalid parameter\n", fname) - ; - xprintf("Writing basic solution to '%s'...\n", fname); - fp = glp_open(fname, "w"), count = 0; - if (fp == NULL) - { xprintf("Unable to create '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* write comment lines */ - glp_format(fp, "c %-12s%s\n", "Problem:", - P->name == NULL ? "" : P->name), count++; - glp_format(fp, "c %-12s%d\n", "Rows:", P->m), count++; - glp_format(fp, "c %-12s%d\n", "Columns:", P->n), count++; - glp_format(fp, "c %-12s%d\n", "Non-zeros:", P->nnz), count++; - switch (glp_get_status(P)) - { case GLP_OPT: s = "OPTIMAL"; break; - case GLP_FEAS: s = "FEASIBLE"; break; - case GLP_INFEAS: s = "INFEASIBLE (INTERMEDIATE)"; break; - case GLP_NOFEAS: s = "INFEASIBLE (FINAL)"; break; - case GLP_UNBND: s = "UNBOUNDED"; break; - case GLP_UNDEF: s = "UNDEFINED"; break; - default: s = "???"; break; - } - glp_format(fp, "c %-12s%s\n", "Status:", s), count++; - switch (P->dir) - { case GLP_MIN: s = "MINimum"; break; - case GLP_MAX: s = "MAXimum"; break; - default: s = "???"; break; - } - glp_format(fp, "c %-12s%s%s%.10g (%s)\n", "Objective:", - P->obj == NULL ? "" : P->obj, - P->obj == NULL ? "" : " = ", P->obj_val, s), count++; - glp_format(fp, "c\n"), count++; - /* write solution line */ - glp_format(fp, "s bas %d %d ", P->m, P->n), count++; - switch (P->pbs_stat) - { case GLP_UNDEF: glp_format(fp, "u"); break; - case GLP_FEAS: glp_format(fp, "f"); break; - case GLP_INFEAS: glp_format(fp, "i"); break; - case GLP_NOFEAS: glp_format(fp, "n"); break; - default: glp_format(fp, "?"); break; - } - glp_format(fp, " "); - switch (P->dbs_stat) - { case GLP_UNDEF: glp_format(fp, "u"); break; - case GLP_FEAS: glp_format(fp, "f"); break; - case GLP_INFEAS: glp_format(fp, "i"); break; - case GLP_NOFEAS: glp_format(fp, "n"); break; - default: glp_format(fp, "?"); break; - } - glp_format(fp, " %.*g\n", DBL_DIG, P->obj_val); - /* write row solution descriptor lines */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - glp_format(fp, "i %d ", i), count++; - switch (row->stat) - { case GLP_BS: - glp_format(fp, "b"); - break; - case GLP_NL: - glp_format(fp, "l"); - break; - case GLP_NU: - glp_format(fp, "u"); - break; - case GLP_NF: - glp_format(fp, "f"); - break; - case GLP_NS: - glp_format(fp, "s"); - break; - default: - xassert(row != row); - } - glp_format(fp, " %.*g %.*g\n", DBL_DIG, row->prim, DBL_DIG, - row->dual); - } - /* write column solution descriptor lines */ - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - glp_format(fp, "j %d ", j), count++; - switch (col->stat) - { case GLP_BS: - glp_format(fp, "b"); - break; - case GLP_NL: - glp_format(fp, "l"); - break; - case GLP_NU: - glp_format(fp, "u"); - break; - case GLP_NF: - glp_format(fp, "f"); - break; - case GLP_NS: - glp_format(fp, "s"); - break; - default: - xassert(col != col); - } - glp_format(fp, " %.*g %.*g\n", DBL_DIG, col->prim, DBL_DIG, - col->dual); - } - /* write end line */ - glp_format(fp, "e o f\n"), count++; - if (glp_ioerr(fp)) - { xprintf("Write error on '%s' - %s\n", fname, get_err_msg()); - goto done; - } - /* basic solution has been successfully written */ - xprintf("%d lines were written\n", count); - ret = 0; -done: if (fp != NULL) - glp_close(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/btf.c b/code/3rd_glpk/bflib/btf.c deleted file mode 100644 index 993c9ca1..00000000 --- a/code/3rd_glpk/bflib/btf.c +++ /dev/null @@ -1,569 +0,0 @@ -/* btf.c (sparse block triangular LU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "btf.h" -#include "env.h" -#include "luf.h" -#include "mc13d.h" -#include "mc21a.h" - -/*********************************************************************** -* btf_store_a_cols - store pattern of matrix A in column-wise format -* -* This routine stores the pattern (that is, only indices of non-zero -* elements) of the original matrix A in column-wise format. -* -* On exit the routine returns the number of non-zeros in matrix A. */ - -int btf_store_a_cols(BTF *btf, int (*col)(void *info, int j, int ind[], - double val[]), void *info, int ind[], double val[]) -{ int n = btf->n; - SVA *sva = btf->sva; - int *sv_ind = sva->ind; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - int j, len, ptr, nnz; - nnz = 0; - for (j = 1; j <= n; j++) - { /* get j-th column */ - len = col(info, j, ind, val); - xassert(0 <= len && len <= n); - /* reserve locations for j-th column */ - if (len > 0) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - } - sva_reserve_cap(sva, ac_ref+(j-1), len); - } - /* store pattern of j-th column */ - ptr = ac_ptr[j]; - memcpy(&sv_ind[ptr], &ind[1], len * sizeof(int)); - ac_len[j] = len; - nnz += len; - } - return nnz; -} - -/*********************************************************************** -* btf_make_blocks - permutations to block triangular form -* -* This routine analyzes the pattern of the original matrix A and -* determines permutation matrices P and Q such that A = P * A~* Q, -* where A~ is an upper block triangular matrix. -* -* On exit the routine returns symbolic rank of matrix A. */ - -int btf_make_blocks(BTF *btf) -{ int n = btf->n; - SVA *sva = btf->sva; - int *sv_ind = sva->ind; - int *pp_ind = btf->pp_ind; - int *pp_inv = btf->pp_inv; - int *qq_ind = btf->qq_ind; - int *qq_inv = btf->qq_inv; - int *beg = btf->beg; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - int i, j, rank, *iperm, *pr, *arp, *cv, *out, *ip, *lenr, *lowl, - *numb, *prev; - /* determine column permutation matrix M such that matrix A * M - * has zero-free diagonal */ - iperm = qq_inv; /* matrix M */ - pr = btf->p1_ind; /* working array */ - arp = btf->p1_inv; /* working array */ - cv = btf->q1_ind; /* working array */ - out = btf->q1_inv; /* working array */ - rank = mc21a(n, sv_ind, ac_ptr, ac_len, iperm, pr, arp, cv, out); - xassert(0 <= rank && rank <= n); - if (rank < n) - { /* A is structurally singular (rank is its symbolic rank) */ - goto done; - } - /* build pattern of matrix A * M */ - ip = pp_ind; /* working array */ - lenr = qq_ind; /* working array */ - for (j = 1; j <= n; j++) - { ip[j] = ac_ptr[iperm[j]]; - lenr[j] = ac_len[iperm[j]]; - } - /* determine symmetric permutation matrix S such that matrix - * S * (A * M) * S' = A~ is upper block triangular */ - lowl = btf->p1_ind; /* working array */ - numb = btf->p1_inv; /* working array */ - prev = btf->q1_ind; /* working array */ - btf->num = - mc13d(n, sv_ind, ip, lenr, pp_inv, beg, lowl, numb, prev); - xassert(beg[1] == 1); - beg[btf->num+1] = n+1; - /* A * M = S' * A~ * S ==> A = S' * A~ * (S * M') */ - /* determine permutation matrix P = S' */ - for (j = 1; j <= n; j++) - pp_ind[pp_inv[j]] = j; - /* determine permutation matrix Q = S * M' = P' * M' */ - for (i = 1; i <= n; i++) - qq_ind[i] = iperm[pp_inv[i]]; - for (i = 1; i <= n; i++) - qq_inv[qq_ind[i]] = i; -done: return rank; -} - -/*********************************************************************** -* btf_check_blocks - check structure of matrix A~ -* -* This routine checks that structure of upper block triangular matrix -* A~ is correct. -* -* NOTE: For testing/debugging only. */ - -void btf_check_blocks(BTF *btf) -{ int n = btf->n; - SVA *sva = btf->sva; - int *sv_ind = sva->ind; - int *pp_ind = btf->pp_ind; - int *pp_inv = btf->pp_inv; - int *qq_ind = btf->qq_ind; - int *qq_inv = btf->qq_inv; - int num = btf->num; - int *beg = btf->beg; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - int i, ii, j, jj, k, size, ptr, end, diag; - xassert(n > 0); - /* check permutation matrices P and Q */ - for (k = 1; k <= n; k++) - { xassert(1 <= pp_ind[k] && pp_ind[k] <= n); - xassert(pp_inv[pp_ind[k]] == k); - xassert(1 <= qq_ind[k] && qq_ind[k] <= n); - xassert(qq_inv[qq_ind[k]] == k); - } - /* check that matrix A~ is upper block triangular with non-zero - * diagonal */ - xassert(1 <= num && num <= n); - xassert(beg[1] == 1); - xassert(beg[num+1] == n+1); - /* walk thru blocks of A~ */ - for (k = 1; k <= num; k++) - { /* determine size of k-th block */ - size = beg[k+1] - beg[k]; - xassert(size >= 1); - /* walk thru columns of k-th block */ - for (jj = beg[k]; jj < beg[k+1]; jj++) - { diag = 0; - /* jj-th column of A~ = j-th column of A */ - j = qq_ind[jj]; - /* walk thru elements of j-th column of A */ - ptr = ac_ptr[j]; - end = ptr + ac_len[j]; - for (; ptr < end; ptr++) - { /* determine row index of a[i,j] */ - i = sv_ind[ptr]; - /* i-th row of A = ii-th row of A~ */ - ii = pp_ind[i]; - /* a~[ii,jj] should not be below k-th block */ - xassert(ii < beg[k+1]); - if (ii == jj) - { /* non-zero diagonal element of A~ encountered */ - diag = 1; - } - } - xassert(diag); - } - } - return; -} - -/*********************************************************************** -* btf_build_a_rows - build matrix A in row-wise format -* -* This routine builds the row-wise representation of matrix A in the -* right part of SVA using its column-wise representation. -* -* The working array len should have at least 1+n elements (len[0] is -* not used). */ - -void btf_build_a_rows(BTF *btf, int len[/*1+n*/]) -{ int n = btf->n; - SVA *sva = btf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int ar_ref = btf->ar_ref; - int *ar_ptr = &sva->ptr[ar_ref-1]; - int *ar_len = &sva->len[ar_ref-1]; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - int i, j, end, nnz, ptr, ptr1; - /* calculate the number of non-zeros in each row of matrix A and - * the total number of non-zeros */ - nnz = 0; - for (i = 1; i <= n; i++) - len[i] = 0; - for (j = 1; j <= n; j++) - { nnz += ac_len[j]; - for (end = (ptr = ac_ptr[j]) + ac_len[j]; ptr < end; ptr++) - len[sv_ind[ptr]]++; - } - /* we need at least nnz free locations in SVA */ - if (sva->r_ptr - sva->m_ptr < nnz) - { sva_more_space(sva, nnz); - sv_ind = sva->ind; - sv_val = sva->val; - } - /* reserve locations for rows of matrix A */ - for (i = 1; i <= n; i++) - { if (len[i] > 0) - sva_reserve_cap(sva, ar_ref-1+i, len[i]); - ar_len[i] = len[i]; - } - /* walk thru columns of matrix A and build its rows */ - for (j = 1; j <= n; j++) - { for (end = (ptr = ac_ptr[j]) + ac_len[j]; ptr < end; ptr++) - { i = sv_ind[ptr]; - sv_ind[ptr1 = ar_ptr[i] + (--len[i])] = j; - sv_val[ptr1] = sv_val[ptr]; - } - } - return; -} - -/*********************************************************************** -* btf_a_solve - solve system A * x = b -* -* This routine solves the system A * x = b, where A is the original -* matrix. -* -* On entry the array b should contain elements of the right-hand size -* vector b in locations b[1], ..., b[n], where n is the order of the -* matrix A. On exit the array x will contain elements of the solution -* vector in locations x[1], ..., x[n]. Note that the array b will be -* clobbered on exit. -* -* The routine also uses locations [1], ..., [max_size] of two working -* arrays w1 and w2, where max_size is the maximal size of diagonal -* blocks in BT-factorization (max_size <= n). */ - -void btf_a_solve(BTF *btf, double b[/*1+n*/], double x[/*1+n*/], - double w1[/*1+n*/], double w2[/*1+n*/]) -{ SVA *sva = btf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int *pp_inv = btf->pp_inv; - int *qq_ind = btf->qq_ind; - int num = btf->num; - int *beg = btf->beg; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - double *bb = w1; - double *xx = w2; - LUF luf; - int i, j, jj, k, beg_k, flag; - double t; - for (k = num; k >= 1; k--) - { /* determine order of diagonal block A~[k,k] */ - luf.n = beg[k+1] - (beg_k = beg[k]); - if (luf.n == 1) - { /* trivial case */ - /* solve system A~[k,k] * X[k] = B[k] */ - t = x[qq_ind[beg_k]] = - b[pp_inv[beg_k]] / btf->vr_piv[beg_k]; - /* substitute X[k] into other equations */ - if (t != 0.0) - { int ptr = ac_ptr[qq_ind[beg_k]]; - int end = ptr + ac_len[qq_ind[beg_k]]; - for (; ptr < end; ptr++) - b[sv_ind[ptr]] -= sv_val[ptr] * t; - } - } - else - { /* general case */ - /* construct B[k] */ - flag = 0; - for (i = 1; i <= luf.n; i++) - { if ((bb[i] = b[pp_inv[i + (beg_k-1)]]) != 0.0) - flag = 1; - } - /* solve system A~[k,k] * X[k] = B[k] */ - if (!flag) - { /* B[k] = 0, so X[k] = 0 */ - for (j = 1; j <= luf.n; j++) - x[qq_ind[j + (beg_k-1)]] = 0.0; - continue; - } - luf.sva = sva; - luf.fr_ref = btf->fr_ref + (beg_k-1); - luf.fc_ref = btf->fc_ref + (beg_k-1); - luf.vr_ref = btf->vr_ref + (beg_k-1); - luf.vr_piv = btf->vr_piv + (beg_k-1); - luf.vc_ref = btf->vc_ref + (beg_k-1); - luf.pp_ind = btf->p1_ind + (beg_k-1); - luf.pp_inv = btf->p1_inv + (beg_k-1); - luf.qq_ind = btf->q1_ind + (beg_k-1); - luf.qq_inv = btf->q1_inv + (beg_k-1); - luf_f_solve(&luf, bb); - luf_v_solve(&luf, bb, xx); - /* store X[k] and substitute it into other equations */ - for (j = 1; j <= luf.n; j++) - { jj = j + (beg_k-1); - t = x[qq_ind[jj]] = xx[j]; - if (t != 0.0) - { int ptr = ac_ptr[qq_ind[jj]]; - int end = ptr + ac_len[qq_ind[jj]]; - for (; ptr < end; ptr++) - b[sv_ind[ptr]] -= sv_val[ptr] * t; - } - } - } - } - return; -} - -/*********************************************************************** -* btf_at_solve - solve system A'* x = b -* -* This routine solves the system A'* x = b, where A' is a matrix -* transposed to the original matrix A. -* -* On entry the array b should contain elements of the right-hand size -* vector b in locations b[1], ..., b[n], where n is the order of the -* matrix A. On exit the array x will contain elements of the solution -* vector in locations x[1], ..., x[n]. Note that the array b will be -* clobbered on exit. -* -* The routine also uses locations [1], ..., [max_size] of two working -* arrays w1 and w2, where max_size is the maximal size of diagonal -* blocks in BT-factorization (max_size <= n). */ - -void btf_at_solve(BTF *btf, double b[/*1+n*/], double x[/*1+n*/], - double w1[/*1+n*/], double w2[/*1+n*/]) -{ SVA *sva = btf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int *pp_inv = btf->pp_inv; - int *qq_ind = btf->qq_ind; - int num = btf->num; - int *beg = btf->beg; - int ar_ref = btf->ar_ref; - int *ar_ptr = &sva->ptr[ar_ref-1]; - int *ar_len = &sva->len[ar_ref-1]; - double *bb = w1; - double *xx = w2; - LUF luf; - int i, j, jj, k, beg_k, flag; - double t; - for (k = 1; k <= num; k++) - { /* determine order of diagonal block A~[k,k] */ - luf.n = beg[k+1] - (beg_k = beg[k]); - if (luf.n == 1) - { /* trivial case */ - /* solve system A~'[k,k] * X[k] = B[k] */ - t = x[pp_inv[beg_k]] = - b[qq_ind[beg_k]] / btf->vr_piv[beg_k]; - /* substitute X[k] into other equations */ - if (t != 0.0) - { int ptr = ar_ptr[pp_inv[beg_k]]; - int end = ptr + ar_len[pp_inv[beg_k]]; - for (; ptr < end; ptr++) - b[sv_ind[ptr]] -= sv_val[ptr] * t; - } - } - else - { /* general case */ - /* construct B[k] */ - flag = 0; - for (i = 1; i <= luf.n; i++) - { if ((bb[i] = b[qq_ind[i + (beg_k-1)]]) != 0.0) - flag = 1; - } - /* solve system A~'[k,k] * X[k] = B[k] */ - if (!flag) - { /* B[k] = 0, so X[k] = 0 */ - for (j = 1; j <= luf.n; j++) - x[pp_inv[j + (beg_k-1)]] = 0.0; - continue; - } - luf.sva = sva; - luf.fr_ref = btf->fr_ref + (beg_k-1); - luf.fc_ref = btf->fc_ref + (beg_k-1); - luf.vr_ref = btf->vr_ref + (beg_k-1); - luf.vr_piv = btf->vr_piv + (beg_k-1); - luf.vc_ref = btf->vc_ref + (beg_k-1); - luf.pp_ind = btf->p1_ind + (beg_k-1); - luf.pp_inv = btf->p1_inv + (beg_k-1); - luf.qq_ind = btf->q1_ind + (beg_k-1); - luf.qq_inv = btf->q1_inv + (beg_k-1); - luf_vt_solve(&luf, bb, xx); - luf_ft_solve(&luf, xx); - /* store X[k] and substitute it into other equations */ - for (j = 1; j <= luf.n; j++) - { jj = j + (beg_k-1); - t = x[pp_inv[jj]] = xx[j]; - if (t != 0.0) - { int ptr = ar_ptr[pp_inv[jj]]; - int end = ptr + ar_len[pp_inv[jj]]; - for (; ptr < end; ptr++) - b[sv_ind[ptr]] -= sv_val[ptr] * t; - } - } - } - } - return; -} - -/*********************************************************************** -* btf_at_solve1 - solve system A'* y = e' to cause growth in y -* -* This routine is a special version of btf_at_solve. It solves the -* system A'* y = e' = e + delta e, where A' is a matrix transposed to -* the original matrix A, e is the specified right-hand side vector, -* and delta e is a vector of +1 and -1 chosen to cause growth in the -* solution vector y. -* -* On entry the array e should contain elements of the right-hand size -* vector e in locations e[1], ..., e[n], where n is the order of the -* matrix A. On exit the array y will contain elements of the solution -* vector in locations y[1], ..., y[n]. Note that the array e will be -* clobbered on exit. -* -* The routine also uses locations [1], ..., [max_size] of two working -* arrays w1 and w2, where max_size is the maximal size of diagonal -* blocks in BT-factorization (max_size <= n). */ - -void btf_at_solve1(BTF *btf, double e[/*1+n*/], double y[/*1+n*/], - double w1[/*1+n*/], double w2[/*1+n*/]) -{ SVA *sva = btf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int *pp_inv = btf->pp_inv; - int *qq_ind = btf->qq_ind; - int num = btf->num; - int *beg = btf->beg; - int ar_ref = btf->ar_ref; - int *ar_ptr = &sva->ptr[ar_ref-1]; - int *ar_len = &sva->len[ar_ref-1]; - double *ee = w1; - double *yy = w2; - LUF luf; - int i, j, jj, k, beg_k, ptr, end; - double e_k, y_k; - for (k = 1; k <= num; k++) - { /* determine order of diagonal block A~[k,k] */ - luf.n = beg[k+1] - (beg_k = beg[k]); - if (luf.n == 1) - { /* trivial case */ - /* determine E'[k] = E[k] + delta E[k] */ - e_k = e[qq_ind[beg_k]]; - e_k = (e_k >= 0.0 ? e_k + 1.0 : e_k - 1.0); - /* solve system A~'[k,k] * Y[k] = E[k] */ - y_k = y[pp_inv[beg_k]] = e_k / btf->vr_piv[beg_k]; - /* substitute Y[k] into other equations */ - ptr = ar_ptr[pp_inv[beg_k]]; - end = ptr + ar_len[pp_inv[beg_k]]; - for (; ptr < end; ptr++) - e[sv_ind[ptr]] -= sv_val[ptr] * y_k; - } - else - { /* general case */ - /* construct E[k] */ - for (i = 1; i <= luf.n; i++) - ee[i] = e[qq_ind[i + (beg_k-1)]]; - /* solve system A~'[k,k] * Y[k] = E[k] + delta E[k] */ - luf.sva = sva; - luf.fr_ref = btf->fr_ref + (beg_k-1); - luf.fc_ref = btf->fc_ref + (beg_k-1); - luf.vr_ref = btf->vr_ref + (beg_k-1); - luf.vr_piv = btf->vr_piv + (beg_k-1); - luf.vc_ref = btf->vc_ref + (beg_k-1); - luf.pp_ind = btf->p1_ind + (beg_k-1); - luf.pp_inv = btf->p1_inv + (beg_k-1); - luf.qq_ind = btf->q1_ind + (beg_k-1); - luf.qq_inv = btf->q1_inv + (beg_k-1); - luf_vt_solve1(&luf, ee, yy); - luf_ft_solve(&luf, yy); - /* store Y[k] and substitute it into other equations */ - for (j = 1; j <= luf.n; j++) - { jj = j + (beg_k-1); - y_k = y[pp_inv[jj]] = yy[j]; - ptr = ar_ptr[pp_inv[jj]]; - end = ptr + ar_len[pp_inv[jj]]; - for (; ptr < end; ptr++) - e[sv_ind[ptr]] -= sv_val[ptr] * y_k; - } - } - } - return; -} - -/*********************************************************************** -* btf_estimate_norm - estimate 1-norm of inv(A) -* -* This routine estimates 1-norm of inv(A) by one step of inverse -* iteration for the small singular vector as described in [1]. This -* involves solving two systems of equations: -* -* A'* y = e, -* -* A * z = y, -* -* where A' is a matrix transposed to A, and e is a vector of +1 and -1 -* chosen to cause growth in y. Then -* -* estimate 1-norm of inv(A) = (1-norm of z) / (1-norm of y) -* -* REFERENCES -* -* 1. G.E.Forsythe, M.A.Malcolm, C.B.Moler. Computer Methods for -* Mathematical Computations. Prentice-Hall, Englewood Cliffs, N.J., -* pp. 30-62 (subroutines DECOMP and SOLVE). */ - -double btf_estimate_norm(BTF *btf, double w1[/*1+n*/], double - w2[/*1+n*/], double w3[/*1+n*/], double w4[/*1+n*/]) -{ int n = btf->n; - double *e = w1; - double *y = w2; - double *z = w1; - int i; - double y_norm, z_norm; - /* compute y = inv(A') * e to cause growth in y */ - for (i = 1; i <= n; i++) - e[i] = 0.0; - btf_at_solve1(btf, e, y, w3, w4); - /* compute 1-norm of y = sum |y[i]| */ - y_norm = 0.0; - for (i = 1; i <= n; i++) - y_norm += (y[i] >= 0.0 ? +y[i] : -y[i]); - /* compute z = inv(A) * y */ - btf_a_solve(btf, y, z, w3, w4); - /* compute 1-norm of z = sum |z[i]| */ - z_norm = 0.0; - for (i = 1; i <= n; i++) - z_norm += (z[i] >= 0.0 ? +z[i] : -z[i]); - /* estimate 1-norm of inv(A) = (1-norm of z) / (1-norm of y) */ - return z_norm / y_norm; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/btf.h b/code/3rd_glpk/bflib/btf.h deleted file mode 100644 index 3f1b5926..00000000 --- a/code/3rd_glpk/bflib/btf.h +++ /dev/null @@ -1,207 +0,0 @@ -/* btf.h (sparse block triangular LU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef BTF_H -#define BTF_H - -#include "sva.h" - -/*********************************************************************** -* The structure BTF describes BT-factorization, which is sparse block -* triangular LU-factorization. -* -* The BT-factorization has the following format: -* -* A = P * A~ * Q, (1) -* -* where A is a given (unsymmetric) square matrix, A~ is an upper block -* triangular matrix (see below), P and Q are permutation matrices. All -* the matrices have the same order n. -* -* The matrix A~, which is a permuted version of the original matrix A, -* has the following structure: -* -* A~[1,1] A~[1,2] ... A~[1,num-1] A~[1,num] -* -* A~[2,2] ... A~[2,num-1] A~[2,num] -* -* . . . . . . . . . (2) -* -* A~[num-1,num-1] A~[num-1,num] -* -* A~[num,num] -* -* where A~[i,j] is a submatrix called a "block," num is the number of -* blocks. Each diagonal block A~[k,k] is a non-singular square matrix, -* and each subdiagonal block A~[i,j], i > j, is a zero submatrix, thus -* A~ is an upper block triangular matrix. -* -* Permutation matrices P and Q are stored in ordinary arrays in both -* row- and column-like formats. -* -* The original matrix A is stored in both row- and column-wise sparse -* formats in the associated sparse vector area (SVA). Should note that -* elements of all diagonal blocks A~[k,k] in matrix A are set to zero -* (i.e. removed), so only elements of non-diagonal blocks are stored. -* -* Each diagonal block A~[k,k], 1 <= k <= num, is stored in the form of -* LU-factorization (see the module LUF). */ - -typedef struct BTF BTF; - -struct BTF -{ /* sparse block triangular LU-factorization */ - int n; - /* order of matrices A, A~, P, Q */ - SVA *sva; - /* associated sparse vector area used to store rows and columns - * of matrix A as well as sparse vectors for LU-factorizations of - * all diagonal blocks A~[k,k] */ - /*--------------------------------------------------------------*/ - /* matrix P */ - int *pp_ind; /* int pp_ind[1+n]; */ - /* pp_ind[i] = j means that P[i,j] = 1 */ - int *pp_inv; /* int pp_inv[1+n]; */ - /* pp_inv[j] = i means that P[i,j] = 1 */ - /* if i-th row of matrix A is i'-th row of matrix A~, then - * pp_ind[i] = i' and pp_inv[i'] = i */ - /*--------------------------------------------------------------*/ - /* matrix Q */ - int *qq_ind; /* int qq_ind[1+n]; */ - /* qq_ind[i] = j means that Q[i,j] = 1 */ - int *qq_inv; /* int qq_inv[1+n]; */ - /* qq_inv[j] = i means that Q[i,j] = 1 */ - /* if j-th column of matrix A is j'-th column of matrix A~, then - * qq_ind[j'] = j and qq_inv[j] = j' */ - /*--------------------------------------------------------------*/ - /* block triangular structure of matrix A~ */ - int num; - /* number of diagonal blocks, 1 <= num <= n */ - int *beg; /* int beg[1+num+1]; */ - /* beg[0] is not used; - * beg[k], 1 <= k <= num, is index of first row/column of k-th - * block of matrix A~; - * beg[num+1] is always n+1; - * note that order (size) of k-th diagonal block can be computed - * as beg[k+1] - beg[k] */ - /*--------------------------------------------------------------*/ - /* original matrix A in row-wise format */ - /* NOTE: elements of all diagonal blocks A~[k,k] are removed */ - int ar_ref; - /* reference number of sparse vector in SVA, which is the first - * row of matrix A */ -#if 0 + 0 - int *ar_ptr = &sva->ptr[ar_ref-1]; - /* ar_ptr[0] is not used; - * ar_ptr[i], 1 <= i <= n, is pointer to i-th row in SVA */ - int *ar_len = &sva->ptr[ar_ref-1]; - /* ar_len[0] is not used; - * ar_len[i], 1 <= i <= n, is length of i-th row */ -#endif - /*--------------------------------------------------------------*/ - /* original matrix A in column-wise format */ - /* NOTE: elements of all diagonal blocks A~[k,k] are removed */ - int ac_ref; - /* reference number of sparse vector in SVA, which is the first - * column of matrix A */ -#if 0 + 0 - int *ac_ptr = &sva->ptr[ac_ref-1]; - /* ac_ptr[0] is not used; - * ac_ptr[j], 1 <= j <= n, is pointer to j-th column in SVA */ - int *ac_len = &sva->ptr[ac_ref-1]; - /* ac_len[0] is not used; - * ac_len[j], 1 <= j <= n, is length of j-th column */ -#endif - /*--------------------------------------------------------------*/ - /* LU-factorizations of diagonal blocks A~[k,k] */ - /* to decrease overhead expenses similar arrays for all LUFs are - * packed into a single array; for example, elements fr_ptr[1], - * ..., fr_ptr[n1], where n1 = beg[2] - beg[1], are related to - * LUF for first diagonal block A~[1,1], elements fr_ptr[n1+1], - * ..., fr_ptr[n1+n2], where n2 = beg[3] - beg[2], are related to - * LUF for second diagonal block A~[2,2], etc.; in other words, - * elements related to LUF for k-th diagonal block A~[k,k] have - * indices beg[k], beg[k]+1, ..., beg[k+1]-1 */ - /* for details about LUF see description of the LUF module */ - int fr_ref; - /* reference number of sparse vector in SVA, which is the first - row of matrix F for first diagonal block A~[1,1] */ - int fc_ref; - /* reference number of sparse vector in SVA, which is the first - column of matrix F for first diagonal block A~[1,1] */ - int vr_ref; - /* reference number of sparse vector in SVA, which is the first - row of matrix V for first diagonal block A~[1,1] */ - double *vr_piv; /* double vr_piv[1+n]; */ - /* vr_piv[0] is not used; - vr_piv[1,...,n] are pivot elements for all diagonal blocks */ - int vc_ref; - /* reference number of sparse vector in SVA, which is the first - column of matrix V for first diagonal block A~[1,1] */ - int *p1_ind; /* int p1_ind[1+n]; */ - int *p1_inv; /* int p1_inv[1+n]; */ - int *q1_ind; /* int q1_ind[1+n]; */ - int *q1_inv; /* int q1_inv[1+n]; */ - /* permutation matrices P and Q for all diagonal blocks */ -}; - -#define btf_store_a_cols _glp_btf_store_a_cols -int btf_store_a_cols(BTF *btf, int (*col)(void *info, int j, int ind[], - double val[]), void *info, int ind[], double val[]); -/* store pattern of matrix A in column-wise format */ - -#define btf_make_blocks _glp_btf_make_blocks -int btf_make_blocks(BTF *btf); -/* permutations to block triangular form */ - -#define btf_check_blocks _glp_btf_check_blocks -void btf_check_blocks(BTF *btf); -/* check structure of matrix A~ */ - -#define btf_build_a_rows _glp_btf_build_a_rows -void btf_build_a_rows(BTF *btf, int len[/*1+n*/]); -/* build matrix A in row-wise format */ - -#define btf_a_solve _glp_btf_a_solve -void btf_a_solve(BTF *btf, double b[/*1+n*/], double x[/*1+n*/], - double w1[/*1+n*/], double w2[/*1+n*/]); -/* solve system A * x = b */ - -#define btf_at_solve _glp_btf_at_solve -void btf_at_solve(BTF *btf, double b[/*1+n*/], double x[/*1+n*/], - double w1[/*1+n*/], double w2[/*1+n*/]); -/* solve system A'* x = b */ - -#define btf_at_solve1 _glp_btf_at_solve1 -void btf_at_solve1(BTF *btf, double e[/*1+n*/], double y[/*1+n*/], - double w1[/*1+n*/], double w2[/*1+n*/]); -/* solve system A'* y = e' to cause growth in y */ - -#define btf_estimate_norm _glp_btf_estimate_norm -double btf_estimate_norm(BTF *btf, double w1[/*1+n*/], double - w2[/*1+n*/], double w3[/*1+n*/], double w4[/*1+n*/]); -/* estimate 1-norm of inv(A) */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/btfint.c b/code/3rd_glpk/bflib/btfint.c deleted file mode 100644 index 378d3a81..00000000 --- a/code/3rd_glpk/bflib/btfint.c +++ /dev/null @@ -1,407 +0,0 @@ -/* btfint.c (interface to BT-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "btfint.h" - -BTFINT *btfint_create(void) -{ /* create interface to BT-factorization */ - BTFINT *fi; - fi = talloc(1, BTFINT); - fi->n_max = 0; - fi->valid = 0; - fi->sva = NULL; - fi->btf = NULL; - fi->sgf = NULL; - fi->sva_n_max = fi->sva_size = 0; - fi->delta_n0 = fi->delta_n = 0; - fi->sgf_piv_tol = 0.10; - fi->sgf_piv_lim = 4; - fi->sgf_suhl = 1; - fi->sgf_eps_tol = DBL_EPSILON; - return fi; -} - -static void factorize_triv(BTFINT *fi, int k, int (*col)(void *info, - int j, int ind[], double val[]), void *info) -{ /* compute LU-factorization of diagonal block A~[k,k] and store - * corresponding columns of matrix A except elements of A~[k,k] - * (trivial case when the block has unity size) */ - SVA *sva = fi->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - BTF *btf = fi->btf; - int *pp_inv = btf->pp_inv; - int *qq_ind = btf->qq_ind; - int *beg = btf->beg; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - SGF *sgf = fi->sgf; - int *ind = (int *)sgf->vr_max; /* working array */ - double *val = sgf->work; /* working array */ - int i, j, t, len, ptr, beg_k; - /* diagonal block A~[k,k] has the only element in matrix A~, - * which is a~[beg[k],beg[k]] = a[i,j] */ - beg_k = beg[k]; - i = pp_inv[beg_k]; - j = qq_ind[beg_k]; - /* get j-th column of A */ - len = col(info, j, ind, val); - /* find element a[i,j] = a~[beg[k],beg[k]] in j-th column */ - for (t = 1; t <= len; t++) - { if (ind[t] == i) - break; - } - xassert(t <= len); - /* compute LU-factorization of diagonal block A~[k,k], where - * F = (1), V = (a[i,j]), P = Q = (1) (see the module LUF) */ -#if 1 /* FIXME */ - xassert(val[t] != 0.0); -#endif - btf->vr_piv[beg_k] = val[t]; - btf->p1_ind[beg_k] = btf->p1_inv[beg_k] = 1; - btf->q1_ind[beg_k] = btf->q1_inv[beg_k] = 1; - /* remove element a[i,j] = a~[beg[k],beg[k]] from j-th column */ - memmove(&ind[t], &ind[t+1], (len-t) * sizeof(int)); - memmove(&val[t], &val[t+1], (len-t) * sizeof(double)); - len--; - /* and store resulting j-th column of A into BTF */ - if (len > 0) - { /* reserve locations for j-th column of A */ - if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, ac_ref+(j-1), len); - /* store j-th column of A (except elements of A~[k,k]) */ - ptr = ac_ptr[j]; - memcpy(&sv_ind[ptr], &ind[1], len * sizeof(int)); - memcpy(&sv_val[ptr], &val[1], len * sizeof(double)); - ac_len[j] = len; - } - return; -} - -static int factorize_block(BTFINT *fi, int k, int (*col)(void *info, - int j, int ind[], double val[]), void *info) -{ /* compute LU-factorization of diagonal block A~[k,k] and store - * corresponding columns of matrix A except elements of A~[k,k] - * (general case) */ - SVA *sva = fi->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - BTF *btf = fi->btf; - int *pp_ind = btf->pp_ind; - int *qq_ind = btf->qq_ind; - int *beg = btf->beg; - int ac_ref = btf->ac_ref; - int *ac_ptr = &sva->ptr[ac_ref-1]; - int *ac_len = &sva->len[ac_ref-1]; - SGF *sgf = fi->sgf; - int *ind = (int *)sgf->vr_max; /* working array */ - double *val = sgf->work; /* working array */ - LUF luf; - int *vc_ptr, *vc_len, *vc_cap; - int i, ii, j, jj, t, len, cnt, ptr, beg_k; - /* construct fake LUF for LU-factorization of A~[k,k] */ - sgf->luf = &luf; - luf.n = beg[k+1] - (beg_k = beg[k]); - luf.sva = sva; - luf.fr_ref = btf->fr_ref + (beg_k-1); - luf.fc_ref = btf->fc_ref + (beg_k-1); - luf.vr_ref = btf->vr_ref + (beg_k-1); - luf.vr_piv = btf->vr_piv + (beg_k-1); - luf.vc_ref = btf->vc_ref + (beg_k-1); - luf.pp_ind = btf->p1_ind + (beg_k-1); - luf.pp_inv = btf->p1_inv + (beg_k-1); - luf.qq_ind = btf->q1_ind + (beg_k-1); - luf.qq_inv = btf->q1_inv + (beg_k-1); - /* process columns of k-th block of matrix A~ */ - vc_ptr = &sva->ptr[luf.vc_ref-1]; - vc_len = &sva->len[luf.vc_ref-1]; - vc_cap = &sva->cap[luf.vc_ref-1]; - for (jj = 1; jj <= luf.n; jj++) - { /* jj-th column of A~ = j-th column of A */ - j = qq_ind[jj + (beg_k-1)]; - /* get j-th column of A */ - len = col(info, j, ind, val); - /* move elements of diagonal block A~[k,k] to the beginning of - * the column list */ - cnt = 0; - for (t = 1; t <= len; t++) - { /* i = row index of element a[i,j] */ - i = ind[t]; - /* i-th row of A = ii-th row of A~ */ - ii = pp_ind[i]; - if (ii >= beg_k) - { /* a~[ii,jj] = a[i,j] is in diagonal block A~[k,k] */ - double temp; - cnt++; - ind[t] = ind[cnt]; - ind[cnt] = ii - (beg_k-1); /* local index */ - temp = val[t], val[t] = val[cnt], val[cnt] = temp; - } - } - /* first cnt elements in the column list give jj-th column of - * diagonal block A~[k,k], which is initial matrix V in LUF */ - /* enlarge capacity of jj-th column of V = A~[k,k] */ - if (vc_cap[jj] < cnt) - { if (sva->r_ptr - sva->m_ptr < cnt) - { sva_more_space(sva, cnt); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, luf.vc_ref+(jj-1), cnt, 0); - } - /* store jj-th column of V = A~[k,k] */ - ptr = vc_ptr[jj]; - memcpy(&sv_ind[ptr], &ind[1], cnt * sizeof(int)); - memcpy(&sv_val[ptr], &val[1], cnt * sizeof(double)); - vc_len[jj] = cnt; - /* other (len-cnt) elements in the column list are stored in - * j-th column of the original matrix A */ - len -= cnt; - if (len > 0) - { /* reserve locations for j-th column of A */ - if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, ac_ref-1+j, len); - /* store j-th column of A (except elements of A~[k,k]) */ - ptr = ac_ptr[j]; - memcpy(&sv_ind[ptr], &ind[cnt+1], len * sizeof(int)); - memcpy(&sv_val[ptr], &val[cnt+1], len * sizeof(double)); - ac_len[j] = len; - } - } - /* compute LU-factorization of diagonal block A~[k,k]; may note - * that A~[k,k] is irreducible (strongly connected), so singleton - * phase will have no effect */ - k = sgf_factorize(sgf, 0 /* disable singleton phase */); - /* now left (dynamic) part of SVA should be empty (wichtig!) */ - xassert(sva->m_ptr == 1); - return k; -} - -int btfint_factorize(BTFINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info) -{ /* compute BT-factorization of specified matrix A */ - SVA *sva; - BTF *btf; - SGF *sgf; - int k, rank; - xassert(n > 0); - fi->valid = 0; - /* create sparse vector area (SVA), if necessary */ - sva = fi->sva; - if (sva == NULL) - { int sva_n_max = fi->sva_n_max; - int sva_size = fi->sva_size; - if (sva_n_max == 0) - sva_n_max = 6 * n; - if (sva_size == 0) - sva_size = 10 * n; - sva = fi->sva = sva_create_area(sva_n_max, sva_size); - } - /* allocate/reallocate underlying objects, if necessary */ - if (fi->n_max < n) - { int n_max = fi->n_max; - if (n_max == 0) - n_max = fi->n_max = n + fi->delta_n0; - else - n_max = fi->n_max = n + fi->delta_n; - xassert(n_max >= n); - /* allocate/reallocate block triangular factorization (BTF) */ - btf = fi->btf; - if (btf == NULL) - { btf = fi->btf = talloc(1, BTF); - memset(btf, 0, sizeof(BTF)); - btf->sva = sva; - } - else - { tfree(btf->pp_ind); - tfree(btf->pp_inv); - tfree(btf->qq_ind); - tfree(btf->qq_inv); - tfree(btf->beg); - tfree(btf->vr_piv); - tfree(btf->p1_ind); - tfree(btf->p1_inv); - tfree(btf->q1_ind); - tfree(btf->q1_inv); - } - btf->pp_ind = talloc(1+n_max, int); - btf->pp_inv = talloc(1+n_max, int); - btf->qq_ind = talloc(1+n_max, int); - btf->qq_inv = talloc(1+n_max, int); - btf->beg = talloc(1+n_max+1, int); - btf->vr_piv = talloc(1+n_max, double); - btf->p1_ind = talloc(1+n_max, int); - btf->p1_inv = talloc(1+n_max, int); - btf->q1_ind = talloc(1+n_max, int); - btf->q1_inv = talloc(1+n_max, int); - /* allocate/reallocate factorizer workspace (SGF) */ - /* (note that for SGF we could use the size of largest block - * rather than n_max) */ - sgf = fi->sgf; - sgf = fi->sgf; - if (sgf == NULL) - { sgf = fi->sgf = talloc(1, SGF); - memset(sgf, 0, sizeof(SGF)); - } - else - { tfree(sgf->rs_head); - tfree(sgf->rs_prev); - tfree(sgf->rs_next); - tfree(sgf->cs_head); - tfree(sgf->cs_prev); - tfree(sgf->cs_next); - tfree(sgf->vr_max); - tfree(sgf->flag); - tfree(sgf->work); - } - sgf->rs_head = talloc(1+n_max, int); - sgf->rs_prev = talloc(1+n_max, int); - sgf->rs_next = talloc(1+n_max, int); - sgf->cs_head = talloc(1+n_max, int); - sgf->cs_prev = talloc(1+n_max, int); - sgf->cs_next = talloc(1+n_max, int); - sgf->vr_max = talloc(1+n_max, double); - sgf->flag = talloc(1+n_max, char); - sgf->work = talloc(1+n_max, double); - } - btf = fi->btf; - btf->n = n; - sgf = fi->sgf; -#if 1 /* FIXME */ - /* initialize SVA */ - sva->n = 0; - sva->m_ptr = 1; - sva->r_ptr = sva->size + 1; - sva->head = sva->tail = 0; -#endif - /* store pattern of original matrix A in column-wise format */ - btf->ac_ref = sva_alloc_vecs(btf->sva, btf->n); - btf_store_a_cols(btf, col, info, btf->pp_ind, btf->vr_piv); -#ifdef GLP_DEBUG - sva_check_area(sva); -#endif - /* analyze pattern of original matrix A and determine permutation - * matrices P and Q such that A = P * A~* Q, where A~ is an upper - * block triangular matrix */ - rank = btf_make_blocks(btf); - if (rank != n) - { /* original matrix A is structurally singular */ - return 1; - } -#ifdef GLP_DEBUG - btf_check_blocks(btf); -#endif -#if 1 /* FIXME */ - /* initialize SVA */ - sva->n = 0; - sva->m_ptr = 1; - sva->r_ptr = sva->size + 1; - sva->head = sva->tail = 0; -#endif - /* allocate sparse vectors in SVA */ - btf->ar_ref = sva_alloc_vecs(btf->sva, btf->n); - btf->ac_ref = sva_alloc_vecs(btf->sva, btf->n); - btf->fr_ref = sva_alloc_vecs(btf->sva, btf->n); - btf->fc_ref = sva_alloc_vecs(btf->sva, btf->n); - btf->vr_ref = sva_alloc_vecs(btf->sva, btf->n); - btf->vc_ref = sva_alloc_vecs(btf->sva, btf->n); - /* setup factorizer control parameters */ - sgf->updat = 0; /* wichtig! */ - sgf->piv_tol = fi->sgf_piv_tol; - sgf->piv_lim = fi->sgf_piv_lim; - sgf->suhl = fi->sgf_suhl; - sgf->eps_tol = fi->sgf_eps_tol; - /* compute LU-factorizations of diagonal blocks A~[k,k] and also - * store corresponding columns of matrix A except elements of all - * blocks A~[k,k] */ - for (k = 1; k <= btf->num; k++) - { if (btf->beg[k+1] - btf->beg[k] == 1) - { /* trivial case (A~[k,k] has unity order) */ - factorize_triv(fi, k, col, info); - } - else - { /* general case */ - if (factorize_block(fi, k, col, info) != 0) - return 2; /* factorization of A~[k,k] failed */ - } - } -#ifdef GLP_DEBUG - sva_check_area(sva); -#endif - /* build row-wise representation of matrix A */ - btf_build_a_rows(fi->btf, fi->sgf->rs_head); -#ifdef GLP_DEBUG - sva_check_area(sva); -#endif - /* BT-factorization has been successfully computed */ - fi->valid = 1; - return 0; -} - -void btfint_delete(BTFINT *fi) -{ /* delete interface to BT-factorization */ - SVA *sva = fi->sva; - BTF *btf = fi->btf; - SGF *sgf = fi->sgf; - if (sva != NULL) - sva_delete_area(sva); - if (btf != NULL) - { tfree(btf->pp_ind); - tfree(btf->pp_inv); - tfree(btf->qq_ind); - tfree(btf->qq_inv); - tfree(btf->beg); - tfree(btf->vr_piv); - tfree(btf->p1_ind); - tfree(btf->p1_inv); - tfree(btf->q1_ind); - tfree(btf->q1_inv); - tfree(btf); - } - if (sgf != NULL) - { tfree(sgf->rs_head); - tfree(sgf->rs_prev); - tfree(sgf->rs_next); - tfree(sgf->cs_head); - tfree(sgf->cs_prev); - tfree(sgf->cs_next); - tfree(sgf->vr_max); - tfree(sgf->flag); - tfree(sgf->work); - tfree(sgf); - } - tfree(fi); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/btfint.h b/code/3rd_glpk/bflib/btfint.h deleted file mode 100644 index 8d0e70e2..00000000 --- a/code/3rd_glpk/bflib/btfint.h +++ /dev/null @@ -1,73 +0,0 @@ -/* btfint.h (interface to BT-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef BTFINT_H -#define BTFINT_H - -#include "btf.h" -#include "sgf.h" - -typedef struct BTFINT BTFINT; - -struct BTFINT -{ /* interface to BT-factorization */ - int n_max; - /* maximal value of n (increased automatically) */ - int valid; - /* factorization is valid only if this flag is set */ - SVA *sva; - /* sparse vector area (SVA) */ - BTF *btf; - /* sparse block triangular LU-factorization */ - SGF *sgf; - /* sparse Gaussian factorizer workspace */ - /*--------------------------------------------------------------*/ - /* control parameters */ - int sva_n_max, sva_size; - /* parameters passed to sva_create_area */ - int delta_n0, delta_n; - /* if n_max = 0, set n_max = n + delta_n0 - * if n_max < n, set n_max = n + delta_n */ - double sgf_piv_tol; - int sgf_piv_lim; - int sgf_suhl; - double sgf_eps_tol; - /* factorizer control parameters */ -}; - -#define btfint_create _glp_btfint_create -BTFINT *btfint_create(void); -/* create interface to BT-factorization */ - -#define btfint_factorize _glp_btfint_factorize -int btfint_factorize(BTFINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info); -/* compute BT-factorization of specified matrix A */ - -#define btfint_delete _glp_btfint_delete -void btfint_delete(BTFINT *fi); -/* delete interface to BT-factorization */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/fhv.c b/code/3rd_glpk/bflib/fhv.c deleted file mode 100644 index e4bdf855..00000000 --- a/code/3rd_glpk/bflib/fhv.c +++ /dev/null @@ -1,586 +0,0 @@ -/* fhv.c (sparse updatable FHV-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "fhv.h" - -/*********************************************************************** -* fhv_ft_update - update FHV-factorization (Forrest-Tomlin) -* -* This routine updates FHV-factorization of the original matrix A -* after replacing its j-th column by a new one. The routine is based -* on the method proposed by Forrest and Tomlin [1]. -* -* The parameter q specifies the number of column of A, which has been -* replaced, 1 <= q <= n, where n is the order of A. -* -* Row indices and numerical values of non-zero elements of the new -* j-th column of A should be placed in locations aq_ind[1], ..., -* aq_ind[aq_len] and aq_val[1], ..., aq_val[aq_len], respectively, -* where aq_len is the number of non-zeros. Neither zero nor duplicate -* elements are allowed. -* -* The working arrays ind, val, and work should have at least 1+n -* elements (0-th elements are not used). -* -* RETURNS -* -* 0 The factorization has been successfully updated. -* -* 1 New matrix U = P'* V * Q' is upper triangular with zero diagonal -* element u[s,s]. (Elimination was not performed.) -* -* 2 New matrix U = P'* V * Q' is upper triangular, and its diagonal -* element u[s,s] or u[t,t] is too small in magnitude. (Elimination -* was not performed.) -* -* 3 The same as 2, but after performing elimination. -* -* 4 The factorization has not been updated, because maximal number of -* updates has been reached. -* -* 5 Accuracy test failed for the updated factorization. -* -* BACKGROUND -* -* The routine is based on the updating method proposed by Forrest and -* Tomlin [1]. -* -* Let q-th column of the original matrix A have been replaced by new -* column A[q]. Then, to keep the equality A = F * H * V, q-th column -* of matrix V should be replaced by column V[q] = inv(F * H) * A[q]. -* From the standpoint of matrix U = P'* V * Q' such replacement is -* equivalent to replacement of s-th column of matrix U, where s is -* determined from q by permutation matrix Q. Thus, matrix U loses its -* upper triangular form and becomes the following: -* -* 1 s t n -* 1 x x * x x x x x x -* . x * x x x x x x -* s . . * x x x x x x -* . . * x x x x x x -* . . * . x x x x x -* . . * . . x x x x -* t . . * . . . x x x -* . . . . . . . x x -* n . . . . . . . . x -* -* where t is largest row index of a non-zero element in s-th column. -* -* The routine makes matrix U upper triangular as follows. First, it -* moves rows and columns s+1, ..., t by one position to the left and -* upwards, resp., and moves s-th row and s-th column to position t. -* Due to such symmetric permutations matrix U becomes the following -* (note that all diagonal elements remain on the diagonal, and element -* u[s,s] becomes u[t,t]): -* -* 1 s t n -* 1 x x x x x x * x x -* . x x x x x * x x -* s . . x x x x * x x -* . . . x x x * x x -* . . . . x x * x x -* . . . . . x * x x -* t . . x x x x * x x -* . . . . . . . x x -* n . . . . . . . . x -* -* Then the routine performs gaussian elimination to eliminate -* subdiagonal elements u[t,s], ..., u[t,t-1] using diagonal elements -* u[s,s], ..., u[t-1,t-1] as pivots. During the elimination process -* the routine permutes neither rows nor columns, so only t-th row is -* changed. Should note that actually all operations are performed on -* matrix V = P * U * Q, since matrix U is not stored. -* -* To keep the equality A = F * H * V, the routine appends new row-like -* factor H[k] to matrix H, and every time it applies elementary -* gaussian transformation to eliminate u[t,j'] = v[p,j] using pivot -* u[j',j'] = v[i,j], it also adds new element f[p,j] = v[p,j] / v[i,j] -* (gaussian multiplier) to factor H[k], which initially is a unity -* matrix. At the end of elimination process the row-like factor H[k] -* may look as follows: -* -* 1 n 1 s t n -* 1 1 . . . . . . . . 1 1 . . . . . . . . -* . 1 . . . . . . . . 1 . . . . . . . -* . . 1 . . . . . . s . . 1 . . . . . . -* p . x x 1 . x . x . . . . 1 . . . . . -* . . . . 1 . . . . . . . . 1 . . . . -* . . . . . 1 . . . . . . . . 1 . . . -* . . . . . . 1 . . t . . x x x x 1 . . -* . . . . . . . 1 . . . . . . . . 1 . -* n . . . . . . . . 1 n . . . . . . . . 1 -* -* H[k] inv(P) * H[k] * P -* -* If, however, s = t, no elimination is needed, in which case no new -* row-like factor is created. -* -* REFERENCES -* -* 1. J.J.H.Forrest and J.A.Tomlin, "Updated triangular factors of the -* basis to maintain sparsity in the product form simplex method," -* Math. Prog. 2 (1972), pp. 263-78. */ - -int fhv_ft_update(FHV *fhv, int q, int aq_len, const int aq_ind[], - const double aq_val[], int ind[/*1+n*/], double val[/*1+n*/], - double work[/*1+n*/]) -{ LUF *luf = fhv->luf; - int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int *vr_cap = &sva->cap[vr_ref-1]; - double *vr_piv = luf->vr_piv; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *vc_cap = &sva->cap[vc_ref-1]; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int *qq_inv = luf->qq_inv; - int *hh_ind = fhv->hh_ind; - int hh_ref = fhv->hh_ref; - int *hh_ptr = &sva->ptr[hh_ref-1]; - int *hh_len = &sva->len[hh_ref-1]; -#if 1 /* FIXME */ - const double eps_tol = DBL_EPSILON; - const double vpq_tol = 1e-5; - const double err_tol = 1e-10; -#endif - int end, i, i_end, i_ptr, j, j_end, j_ptr, k, len, nnz, p, p_end, - p_ptr, ptr, q_end, q_ptr, s, t; - double f, vpq, temp; - /*--------------------------------------------------------------*/ - /* replace current q-th column of matrix V by new one */ - /*--------------------------------------------------------------*/ - xassert(1 <= q && q <= n); - /* convert new q-th column of matrix A to dense format */ - for (i = 1; i <= n; i++) - val[i] = 0.0; - xassert(0 <= aq_len && aq_len <= n); - for (k = 1; k <= aq_len; k++) - { i = aq_ind[k]; - xassert(1 <= i && i <= n); - xassert(val[i] == 0.0); - xassert(aq_val[k] != 0.0); - val[i] = aq_val[k]; - } - /* compute new q-th column of matrix V: - * new V[q] = inv(F * H) * (new A[q]) */ - luf->pp_ind = fhv->p0_ind; - luf->pp_inv = fhv->p0_inv; - luf_f_solve(luf, val); - luf->pp_ind = pp_ind; - luf->pp_inv = pp_inv; - fhv_h_solve(fhv, val); - /* q-th column of V = s-th column of U */ - s = qq_inv[q]; - /* determine row number of element v[p,q] that corresponds to - * diagonal element u[s,s] */ - p = pp_inv[s]; - /* convert new q-th column of V to sparse format; - * element v[p,q] = u[s,s] is not included in the element list - * and stored separately */ - vpq = 0.0; - len = 0; - for (i = 1; i <= n; i++) - { temp = val[i]; -#if 1 /* FIXME */ - if (-eps_tol < temp && temp < +eps_tol) -#endif - /* nop */; - else if (i == p) - vpq = temp; - else - { ind[++len] = i; - val[len] = temp; - } - } - /* clear q-th column of matrix V */ - for (q_end = (q_ptr = vc_ptr[q]) + vc_len[q]; - q_ptr < q_end; q_ptr++) - { /* get row index of v[i,q] */ - i = sv_ind[q_ptr]; - /* find and remove v[i,q] from i-th row */ - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - sv_ind[i_ptr] != q; i_ptr++) - /* nop */; - xassert(i_ptr < i_end); - sv_ind[i_ptr] = sv_ind[i_end-1]; - sv_val[i_ptr] = sv_val[i_end-1]; - vr_len[i]--; - } - /* now q-th column of matrix V is empty */ - vc_len[q] = 0; - /* put new q-th column of V (except element v[p,q] = u[s,s]) in - * column-wise format */ - if (len > 0) - { if (vc_cap[q] < len) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vc_ref-1+q, len, 0); - } - ptr = vc_ptr[q]; - memcpy(&sv_ind[ptr], &ind[1], len * sizeof(int)); - memcpy(&sv_val[ptr], &val[1], len * sizeof(double)); - vc_len[q] = len; - } - /* put new q-th column of V (except element v[p,q] = u[s,s]) in - * row-wise format, and determine largest row number t such that - * u[s,t] != 0 */ - t = (vpq == 0.0 ? 0 : s); - for (k = 1; k <= len; k++) - { /* get row index of v[i,q] */ - i = ind[k]; - /* put v[i,q] to i-th row */ - if (vr_cap[i] == vr_len[i]) - { /* reserve extra locations in i-th row to reduce further - * relocations of that row */ -#if 1 /* FIXME */ - int need = vr_len[i] + 5; -#endif - if (sva->r_ptr - sva->m_ptr < need) - { sva_more_space(sva, need); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vr_ref-1+i, need, 0); - } - sv_ind[ptr = vr_ptr[i] + (vr_len[i]++)] = q; - sv_val[ptr] = val[k]; - /* v[i,q] is non-zero; increase t */ - if (t < pp_ind[i]) - t = pp_ind[i]; - } - /*--------------------------------------------------------------*/ - /* check if matrix U is already upper triangular */ - /*--------------------------------------------------------------*/ - /* check if there is a spike in s-th column of matrix U, which - * is q-th column of matrix V */ - if (s >= t) - { /* no spike; matrix U is already upper triangular */ - /* store its diagonal element u[s,s] = v[p,q] */ - vr_piv[p] = vpq; - if (s > t) - { /* matrix U is structurally singular, because its diagonal - * element u[s,s] = v[p,q] is exact zero */ - xassert(vpq == 0.0); - return 1; - } -#if 1 /* FIXME */ - else if (-vpq_tol < vpq && vpq < +vpq_tol) -#endif - { /* matrix U is not well conditioned, because its diagonal - * element u[s,s] = v[p,q] is too small in magnitude */ - return 2; - } - else - { /* normal case */ - return 0; - } - } - /*--------------------------------------------------------------*/ - /* perform implicit symmetric permutations of rows and columns */ - /* of matrix U */ - /*--------------------------------------------------------------*/ - /* currently v[p,q] = u[s,s] */ - xassert(p == pp_inv[s] && q == qq_ind[s]); - for (k = s; k < t; k++) - { pp_ind[pp_inv[k] = pp_inv[k+1]] = k; - qq_inv[qq_ind[k] = qq_ind[k+1]] = k; - } - /* now v[p,q] = u[t,t] */ - pp_ind[pp_inv[t] = p] = qq_inv[qq_ind[t] = q] = t; - /*--------------------------------------------------------------*/ - /* check if matrix U is already upper triangular */ - /*--------------------------------------------------------------*/ - /* check if there is a spike in t-th row of matrix U, which is - * p-th row of matrix V */ - for (p_end = (p_ptr = vr_ptr[p]) + vr_len[p]; - p_ptr < p_end; p_ptr++) - { if (qq_inv[sv_ind[p_ptr]] < t) - break; /* spike detected */ - } - if (p_ptr == p_end) - { /* no spike; matrix U is already upper triangular */ - /* store its diagonal element u[t,t] = v[p,q] */ - vr_piv[p] = vpq; -#if 1 /* FIXME */ - if (-vpq_tol < vpq && vpq < +vpq_tol) -#endif - { /* matrix U is not well conditioned, because its diagonal - * element u[t,t] = v[p,q] is too small in magnitude */ - return 2; - } - else - { /* normal case */ - return 0; - } - } - /*--------------------------------------------------------------*/ - /* copy p-th row of matrix V, which is t-th row of matrix U, to */ - /* working array */ - /*--------------------------------------------------------------*/ - /* copy p-th row of matrix V, including element v[p,q] = u[t,t], - * to the working array in dense format and remove these elements - * from matrix V; since no pivoting is used, only this row will - * change during elimination */ - for (j = 1; j <= n; j++) - work[j] = 0.0; - work[q] = vpq; - for (p_end = (p_ptr = vr_ptr[p]) + vr_len[p]; - p_ptr < p_end; p_ptr++) - { /* get column index of v[p,j] and store this element to the - * working array */ - work[j = sv_ind[p_ptr]] = sv_val[p_ptr]; - /* find and remove v[p,j] from j-th column */ - for (j_end = (j_ptr = vc_ptr[j]) + vc_len[j]; - sv_ind[j_ptr] != p; j_ptr++) - /* nop */; - xassert(j_ptr < j_end); - sv_ind[j_ptr] = sv_ind[j_end-1]; - sv_val[j_ptr] = sv_val[j_end-1]; - vc_len[j]--; - } - /* now p-th row of matrix V is temporarily empty */ - vr_len[p] = 0; - /*--------------------------------------------------------------*/ - /* perform gaussian elimination */ - /*--------------------------------------------------------------*/ - /* transform p-th row of matrix V stored in working array, which - * is t-th row of matrix U, to eliminate subdiagonal elements - * u[t,s], ..., u[t,t-1]; corresponding gaussian multipliers will - * form non-trivial row of new row-like factor */ - nnz = 0; /* number of non-zero gaussian multipliers */ - for (k = s; k < t; k++) - { /* diagonal element u[k,k] = v[i,j] is used as pivot */ - i = pp_inv[k], j = qq_ind[k]; - /* take subdiagonal element u[t,k] = v[p,j] */ - temp = work[j]; -#if 1 /* FIXME */ - if (-eps_tol < temp && temp < +eps_tol) - continue; -#endif - /* compute and save gaussian multiplier: - * f := u[t,k] / u[k,k] = v[p,j] / v[i,j] */ - ind[++nnz] = i; - val[nnz] = f = work[j] / vr_piv[i]; - /* gaussian transformation to eliminate u[t,k] = v[p,j]: - * (p-th row of V) := (p-th row of V) - f * (i-th row of V) */ - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - i_ptr < i_end; i_ptr++) - work[sv_ind[i_ptr]] -= f * sv_val[i_ptr]; - } - /* now matrix U is again upper triangular */ -#if 1 /* FIXME */ - if (-vpq_tol < work[q] && work[q] < +vpq_tol) -#endif - { /* however, its new diagonal element u[t,t] = v[p,q] is too - * small in magnitude */ - return 3; - } - /*--------------------------------------------------------------*/ - /* create new row-like factor H[k] and add to eta file H */ - /*--------------------------------------------------------------*/ - /* (nnz = 0 means that all subdiagonal elements were too small - * in magnitude) */ - if (nnz > 0) - { if (fhv->nfs == fhv->nfs_max) - { /* maximal number of row-like factors has been reached */ - return 4; - } - k = ++(fhv->nfs); - hh_ind[k] = p; - /* store non-trivial row of H[k] in right (dynamic) part of - * SVA (diagonal unity element is not stored) */ - if (sva->r_ptr - sva->m_ptr < nnz) - { sva_more_space(sva, nnz); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, fhv->hh_ref-1+k, nnz); - ptr = hh_ptr[k]; - memcpy(&sv_ind[ptr], &ind[1], nnz * sizeof(int)); - memcpy(&sv_val[ptr], &val[1], nnz * sizeof(double)); - hh_len[k] = nnz; - } - /*--------------------------------------------------------------*/ - /* copy transformed p-th row of matrix V, which is t-th row of */ - /* matrix U, from working array back to matrix V */ - /*--------------------------------------------------------------*/ - /* copy elements of transformed p-th row of matrix V, which are - * non-diagonal elements u[t,t+1], ..., u[t,n] of matrix U, from - * working array to corresponding columns of matrix V (note that - * diagonal element u[t,t] = v[p,q] not copied); also transform - * p-th row of matrix V to sparse format */ - len = 0; - for (k = t+1; k <= n; k++) - { /* j-th column of V = k-th column of U */ - j = qq_ind[k]; - /* take non-diagonal element v[p,j] = u[t,k] */ - temp = work[j]; -#if 1 /* FIXME */ - if (-eps_tol < temp && temp < +eps_tol) - continue; -#endif - /* add v[p,j] to j-th column of matrix V */ - if (vc_cap[j] == vc_len[j]) - { /* reserve extra locations in j-th column to reduce further - * relocations of that column */ -#if 1 /* FIXME */ - int need = vc_len[j] + 5; -#endif - if (sva->r_ptr - sva->m_ptr < need) - { sva_more_space(sva, need); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vc_ref-1+j, need, 0); - } - sv_ind[ptr = vc_ptr[j] + (vc_len[j]++)] = p; - sv_val[ptr] = temp; - /* store element v[p,j] = u[t,k] to working sparse vector */ - ind[++len] = j; - val[len] = temp; - } - /* copy elements from working sparse vector to p-th row of matrix - * V (this row is currently empty) */ - if (vr_cap[p] < len) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vr_ref-1+p, len, 0); - } - ptr = vr_ptr[p]; - memcpy(&sv_ind[ptr], &ind[1], len * sizeof(int)); - memcpy(&sv_val[ptr], &val[1], len * sizeof(double)); - vr_len[p] = len; - /* store new diagonal element u[t,t] = v[p,q] */ - vr_piv[p] = work[q]; - /*--------------------------------------------------------------*/ - /* perform accuracy test (only if new H[k] was added) */ - /*--------------------------------------------------------------*/ - if (nnz > 0) - { /* copy p-th (non-trivial) row of row-like factor H[k] (except - * unity diagonal element) to working array in dense format */ - for (j = 1; j <= n; j++) - work[j] = 0.0; - k = fhv->nfs; - for (end = (ptr = hh_ptr[k]) + hh_len[k]; ptr < end; ptr++) - work[sv_ind[ptr]] = sv_val[ptr]; - /* compute inner product of p-th (non-trivial) row of matrix - * H[k] and q-th column of matrix V */ - temp = vr_piv[p]; /* 1 * v[p,q] */ - ptr = vc_ptr[q]; - end = ptr + vc_len[q]; - for (; ptr < end; ptr++) - temp += work[sv_ind[ptr]] * sv_val[ptr]; - /* inner product should be equal to element v[p,q] *before* - * matrix V was transformed */ - /* compute relative error */ - temp = fabs(vpq - temp) / (1.0 + fabs(vpq)); -#if 1 /* FIXME */ - if (temp > err_tol) -#endif - { /* relative error is too large */ - return 5; - } - } - /* factorization has been successfully updated */ - return 0; -} - -/*********************************************************************** -* fhv_h_solve - solve system H * x = b -* -* This routine solves the system H * x = b, where the matrix H is the -* middle factor of the sparse updatable FHV-factorization. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix H. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void fhv_h_solve(FHV *fhv, double x[/*1+n*/]) -{ SVA *sva = fhv->luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int nfs = fhv->nfs; - int *hh_ind = fhv->hh_ind; - int hh_ref = fhv->hh_ref; - int *hh_ptr = &sva->ptr[hh_ref-1]; - int *hh_len = &sva->len[hh_ref-1]; - int i, k, end, ptr; - double x_i; - for (k = 1; k <= nfs; k++) - { x_i = x[i = hh_ind[k]]; - for (end = (ptr = hh_ptr[k]) + hh_len[k]; ptr < end; ptr++) - x_i -= sv_val[ptr] * x[sv_ind[ptr]]; - x[i] = x_i; - } - return; -} - -/*********************************************************************** -* fhv_ht_solve - solve system H' * x = b -* -* This routine solves the system H' * x = b, where H' is a matrix -* transposed to the matrix H, which is the middle factor of the sparse -* updatable FHV-factorization. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix H. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void fhv_ht_solve(FHV *fhv, double x[/*1+n*/]) -{ SVA *sva = fhv->luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int nfs = fhv->nfs; - int *hh_ind = fhv->hh_ind; - int hh_ref = fhv->hh_ref; - int *hh_ptr = &sva->ptr[hh_ref-1]; - int *hh_len = &sva->len[hh_ref-1]; - int k, end, ptr; - double x_j; - for (k = nfs; k >= 1; k--) - { if ((x_j = x[hh_ind[k]]) == 0.0) - continue; - for (end = (ptr = hh_ptr[k]) + hh_len[k]; ptr < end; ptr++) - x[sv_ind[ptr]] -= sv_val[ptr] * x_j; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/fhv.h b/code/3rd_glpk/bflib/fhv.h deleted file mode 100644 index df39ca5c..00000000 --- a/code/3rd_glpk/bflib/fhv.h +++ /dev/null @@ -1,114 +0,0 @@ -/* fhv.h (sparse updatable FHV-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef FHV_H -#define FHV_H - -#include "luf.h" - -/*********************************************************************** -* The structure FHV describes sparse updatable FHV-factorization. -* -* The FHV-factorization has the following format: -* -* A = F * H * V, (1) -* -* F = P0 * L * P0', (2) -* -* H = H[1] * H[2] * ... * H[nfs], (3) -* -* V = P * U * Q, (4) -* -* where: A is a given (unsymmetric) square matrix; F, H, V are matrix -* factors actually computed; L is a lower triangular matrix with unity -* diagonal; U is an upper tringular matrix; H[k], k = 1, 2, ..., nfs, -* is a row-like factor, which differs from unity matrix only in one -* row called a non-trivial row; P0, P, Q are permutation matrices; and -* P0' is a matrix transposed to P0. -* -* Matrices F, V, P, Q are stored in the underlying LUF object. -* -* Non-trivial rows of factors H[k] are stored as sparse vectors in the -* right (static) part of the sparse vector area (SVA). Note that unity -* diagonal elements of non-trivial rows are not stored. -* -* Matrix P0 is stored in the same way as matrix P. -* -* Matrices L and U are completely defined by matrices F, V, P, and Q, -* and therefore not stored explicitly. */ - -typedef struct FHV FHV; - -struct FHV -{ /* FHV-factorization */ - LUF *luf; - /* LU-factorization (contains matrices F, V, P, Q) */ - /*--------------------------------------------------------------*/ - /* matrix H in the form of eta file */ - int nfs_max; - /* maximal number of row-like factors (this limits the number of - * updates of the factorization) */ - int nfs; - /* current number of row-like factors, 0 <= nfs <= nfs_max */ - int *hh_ind; /* int hh_ind[1+nfs_max]; */ - /* hh_ind[0] is not used; - * hh_ind[k], 1 <= k <= nfs, is number of non-trivial row of - * factor H[k] */ - int hh_ref; - /* reference number of sparse vector in SVA, which is non-trivial - * row of factor H[1] */ -#if 0 + 0 - int *hh_ptr = &sva->ptr[hh_ref-1]; - /* hh_ptr[0] is not used; - * hh_ptr[k], 1 <= k <= nfs, is pointer to non-trivial row of - * factor H[k] */ - int *hh_len = &sva->len[hh_ref-1]; - /* hh_len[0] is not used; - * hh_len[k], 1 <= k <= nfs, is number of non-zero elements in - * non-trivial row of factor H[k] */ -#endif - /*--------------------------------------------------------------*/ - /* matrix P0 */ - int *p0_ind; /* int p0_ind[1+n]; */ - /* p0_ind[i] = j means that P0[i,j] = 1 */ - int *p0_inv; /* int p0_inv[1+n]; */ - /* p0_inv[j] = i means that P0[i,j] = 1 */ -}; - -#define fhv_ft_update _glp_fhv_ft_update -int fhv_ft_update(FHV *fhv, int q, int aq_len, const int aq_ind[], - const double aq_val[], int ind[/*1+n*/], double val[/*1+n*/], - double work[/*1+n*/]); -/* update FHV-factorization (Forrest-Tomlin) */ - -#define fhv_h_solve _glp_fhv_h_solve -void fhv_h_solve(FHV *fhv, double x[/*1+n*/]); -/* solve system H * x = b */ - -#define fhv_ht_solve _glp_fhv_ht_solve -void fhv_ht_solve(FHV *fhv, double x[/*1+n*/]); -/* solve system H' * x = b */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/fhvint.c b/code/3rd_glpk/bflib/fhvint.c deleted file mode 100644 index a21b71c6..00000000 --- a/code/3rd_glpk/bflib/fhvint.c +++ /dev/null @@ -1,168 +0,0 @@ -/* fhvint.c (interface to FHV-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "fhvint.h" - -FHVINT *fhvint_create(void) -{ /* create interface to FHV-factorization */ - FHVINT *fi; - fi = talloc(1, FHVINT); - memset(fi, 0, sizeof(FHVINT)); - fi->lufi = lufint_create(); - return fi; -} - -int fhvint_factorize(FHVINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info) -{ /* compute FHV-factorization of specified matrix A */ - int nfs_max, old_n_max, n_max, k, ret; - xassert(n > 0); - fi->valid = 0; - /* get required value of nfs_max */ - nfs_max = fi->nfs_max; - if (nfs_max == 0) - nfs_max = 100; - xassert(nfs_max > 0); - /* compute factorization of specified matrix A */ - old_n_max = fi->lufi->n_max; - fi->lufi->sva_n_max = 4 * n + nfs_max; - fi->lufi->sgf_updat = 1; - ret = lufint_factorize(fi->lufi, n, col, info); - n_max = fi->lufi->n_max; - /* allocate/reallocate arrays, if necessary */ - if (fi->fhv.nfs_max != nfs_max) - { if (fi->fhv.hh_ind != NULL) - tfree(fi->fhv.hh_ind); - fi->fhv.hh_ind = talloc(1+nfs_max, int); - } - if (old_n_max < n_max) - { if (fi->fhv.p0_ind != NULL) - tfree(fi->fhv.p0_ind); - if (fi->fhv.p0_inv != NULL) - tfree(fi->fhv.p0_inv); - fi->fhv.p0_ind = talloc(1+n_max, int); - fi->fhv.p0_inv = talloc(1+n_max, int); - } - /* initialize FHV-factorization */ - fi->fhv.luf = fi->lufi->luf; - fi->fhv.nfs_max = nfs_max; - /* H := I */ - fi->fhv.nfs = 0; - fi->fhv.hh_ref = sva_alloc_vecs(fi->lufi->sva, nfs_max); - /* P0 := P */ - for (k = 1; k <= n; k++) - { fi->fhv.p0_ind[k] = fi->fhv.luf->pp_ind[k]; - fi->fhv.p0_inv[k] = fi->fhv.luf->pp_inv[k]; - } - /* set validation flag */ - if (ret == 0) - fi->valid = 1; - return ret; -} - -int fhvint_update(FHVINT *fi, int j, int len, const int ind[], - const double val[]) -{ /* update FHV-factorization after replacing j-th column of A */ - SGF *sgf = fi->lufi->sgf; - int *ind1 = sgf->rs_next; - double *val1 = sgf->vr_max; - double *work = sgf->work; - int ret; - xassert(fi->valid); - ret = fhv_ft_update(&fi->fhv, j, len, ind, val, ind1, val1, work); - if (ret != 0) - fi->valid = 0; - return ret; -} - -void fhvint_ftran(FHVINT *fi, double x[]) -{ /* solve system A * x = b */ - FHV *fhv = &fi->fhv; - LUF *luf = fhv->luf; - int n = luf->n; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - SGF *sgf = fi->lufi->sgf; - double *work = sgf->work; - xassert(fi->valid); - /* A = F * H * V */ - /* x = inv(A) * b = inv(V) * inv(H) * inv(F) * b */ - luf->pp_ind = fhv->p0_ind; - luf->pp_inv = fhv->p0_inv; - luf_f_solve(luf, x); - luf->pp_ind = pp_ind; - luf->pp_inv = pp_inv; - fhv_h_solve(fhv, x); - luf_v_solve(luf, x, work); - memcpy(&x[1], &work[1], n * sizeof(double)); - return; -} - -void fhvint_btran(FHVINT *fi, double x[]) -{ /* solve system A'* x = b */ - FHV *fhv = &fi->fhv; - LUF *luf = fhv->luf; - int n = luf->n; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - SGF *sgf = fi->lufi->sgf; - double *work = sgf->work; - xassert(fi->valid); - /* A' = (F * H * V)' = V'* H'* F' */ - /* x = inv(A') * b = inv(F') * inv(H') * inv(V') * b */ - luf_vt_solve(luf, x, work); - fhv_ht_solve(fhv, work); - luf->pp_ind = fhv->p0_ind; - luf->pp_inv = fhv->p0_inv; - luf_ft_solve(luf, work); - luf->pp_ind = pp_ind; - luf->pp_inv = pp_inv; - memcpy(&x[1], &work[1], n * sizeof(double)); - return; -} - -double fhvint_estimate(FHVINT *fi) -{ /* estimate 1-norm of inv(A) */ - double norm; - xassert(fi->valid); - xassert(fi->fhv.nfs == 0); - norm = luf_estimate_norm(fi->fhv.luf, fi->lufi->sgf->vr_max, - fi->lufi->sgf->work); - return norm; -} - -void fhvint_delete(FHVINT *fi) -{ /* delete interface to FHV-factorization */ - lufint_delete(fi->lufi); - if (fi->fhv.hh_ind != NULL) - tfree(fi->fhv.hh_ind); - if (fi->fhv.p0_ind != NULL) - tfree(fi->fhv.p0_ind); - if (fi->fhv.p0_inv != NULL) - tfree(fi->fhv.p0_inv); - tfree(fi); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/fhvint.h b/code/3rd_glpk/bflib/fhvint.h deleted file mode 100644 index 000829c6..00000000 --- a/code/3rd_glpk/bflib/fhvint.h +++ /dev/null @@ -1,78 +0,0 @@ -/* fhvint.h (interface to FHV-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef FHVINT_H -#define FHVINT_H - -#include "fhv.h" -#include "lufint.h" - -typedef struct FHVINT FHVINT; - -struct FHVINT -{ /* interface to FHV-factorization */ - int valid; - /* factorization is valid only if this flag is set */ - FHV fhv; - /* FHV-factorization */ - LUFINT *lufi; - /* interface to underlying LU-factorization */ - /*--------------------------------------------------------------*/ - /* control parameters */ - int nfs_max; - /* required maximal number of row-like factors */ -}; - -#define fhvint_create _glp_fhvint_create -FHVINT *fhvint_create(void); -/* create interface to FHV-factorization */ - -#define fhvint_factorize _glp_fhvint_factorize -int fhvint_factorize(FHVINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info); -/* compute FHV-factorization of specified matrix A */ - -#define fhvint_update _glp_fhvint_update -int fhvint_update(FHVINT *fi, int j, int len, const int ind[], - const double val[]); -/* update FHV-factorization after replacing j-th column of A */ - -#define fhvint_ftran _glp_fhvint_ftran -void fhvint_ftran(FHVINT *fi, double x[]); -/* solve system A * x = b */ - -#define fhvint_btran _glp_fhvint_btran -void fhvint_btran(FHVINT *fi, double x[]); -/* solve system A'* x = b */ - -#define fhvint_estimate _glp_fhvint_estimate -double fhvint_estimate(FHVINT *fi); -/* estimate 1-norm of inv(A) */ - -#define fhvint_delete _glp_fhvint_delete -void fhvint_delete(FHVINT *fi); -/* delete interface to FHV-factorization */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/ifu.c b/code/3rd_glpk/bflib/ifu.c deleted file mode 100644 index aa47fb09..00000000 --- a/code/3rd_glpk/bflib/ifu.c +++ /dev/null @@ -1,392 +0,0 @@ -/* ifu.c (dense updatable IFU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ifu.h" - -/*********************************************************************** -* ifu_expand - expand IFU-factorization -* -* This routine expands the IFU-factorization of the matrix A according -* to the following expansion of A: -* -* ( A c ) -* new A = ( ) -* ( r' d ) -* -* where c[1,...,n] is a new column, r[1,...,n] is a new row, and d is -* a new diagonal element. -* -* From the main equality F * A = U it follows that: -* -* ( F 0 ) ( A c ) ( FA Fc ) ( U Fc ) -* ( ) ( ) = ( ) = ( ), -* ( 0 1 ) ( r' d ) ( r' d ) ( r' d ) -* -* thus, -* -* ( F 0 ) ( U Fc ) -* new F = ( ), new U = ( ). -* ( 0 1 ) ( r' d ) -* -* Note that the resulting matrix U loses its upper triangular form due -* to row spike r', which should be eliminated. */ - -void ifu_expand(IFU *ifu, double c[/*1+n*/], double r[/*1+n*/], - double d) -{ /* non-optimized version */ - int n_max = ifu->n_max; - int n = ifu->n; - double *f_ = ifu->f; - double *u_ = ifu->u; - int i, j; - double t; -# define f(i,j) f_[(i)*n_max+(j)] -# define u(i,j) u_[(i)*n_max+(j)] - xassert(0 <= n && n < n_max); - /* adjust indexing */ - c++, r++; - /* set new zero column of matrix F */ - for (i = 0; i < n; i++) - f(i,n) = 0.0; - /* set new zero row of matrix F */ - for (j = 0; j < n; j++) - f(n,j) = 0.0; - /* set new unity diagonal element of matrix F */ - f(n,n) = 1.0; - /* set new column of matrix U to vector (old F) * c */ - for (i = 0; i < n; i++) - { /* u[i,n] := (i-th row of old F) * c */ - t = 0.0; - for (j = 0; j < n; j++) - t += f(i,j) * c[j]; - u(i,n) = t; - } - /* set new row of matrix U to vector r */ - for (j = 0; j < n; j++) - u(n,j) = r[j]; - /* set new diagonal element of matrix U to scalar d */ - u(n,n) = d; - /* increase factorization order */ - ifu->n++; -# undef f -# undef u - return; -} - -/*********************************************************************** -* ifu_bg_update - update IFU-factorization (Bartels-Golub) -* -* This routine updates IFU-factorization of the matrix A according to -* its expansion (see comments to the routine ifu_expand). The routine -* is based on the method proposed by Bartels and Golub [1]. -* -* RETURNS -* -* 0 The factorization has been successfully updated. -* -* 1 On some elimination step diagional element u[k,k] to be used as -* pivot is too small in magnitude. -* -* 2 Diagonal element u[n,n] is too small in magnitude (at the end of -* update). -* -* REFERENCES -* -* 1. R.H.Bartels, G.H.Golub, "The Simplex Method of Linear Programming -* Using LU-decomposition", Comm. ACM, 12, pp. 266-68, 1969. */ - -int ifu_bg_update(IFU *ifu, double c[/*1+n*/], double r[/*1+n*/], - double d) -{ /* non-optimized version */ - int n_max = ifu->n_max; - int n = ifu->n; - double *f_ = ifu->f; - double *u_ = ifu->u; -#if 1 /* FIXME */ - double tol = 1e-5; -#endif - int j, k; - double t; -# define f(i,j) f_[(i)*n_max+(j)] -# define u(i,j) u_[(i)*n_max+(j)] - /* expand factorization */ - ifu_expand(ifu, c, r, d); - /* NOTE: n keeps its old value */ - /* eliminate spike (non-zero subdiagonal elements) in last row of - * matrix U */ - for (k = 0; k < n; k++) - { /* if |u[k,k]| < |u[n,k]|, interchange k-th and n-th rows to - * provide |u[k,k]| >= |u[n,k]| for numeric stability */ - if (fabs(u(k,k)) < fabs(u(n,k))) - { /* interchange k-th and n-th rows of matrix U */ - for (j = k; j <= n; j++) - t = u(k,j), u(k,j) = u(n,j), u(n,j) = t; - /* interchange k-th and n-th rows of matrix F to keep the - * main equality F * A = U */ - for (j = 0; j <= n; j++) - t = f(k,j), f(k,j) = f(n,j), f(n,j) = t; - } - /* now |u[k,k]| >= |u[n,k]| */ - /* check if diagonal element u[k,k] can be used as pivot */ - if (fabs(u(k,k)) < tol) - { /* u[k,k] is too small in magnitude */ - return 1; - } - /* if u[n,k] = 0, elimination is not needed */ - if (u(n,k) == 0.0) - continue; - /* compute gaussian multiplier t = u[n,k] / u[k,k] */ - t = u(n,k) / u(k,k); - /* apply gaussian transformation to eliminate u[n,k] */ - /* (n-th row of U) := (n-th row of U) - t * (k-th row of U) */ - for (j = k+1; j <= n; j++) - u(n,j) -= t * u(k,j); - /* apply the same transformation to matrix F to keep the main - * equality F * A = U */ - for (j = 0; j <= n; j++) - f(n,j) -= t * f(k,j); - } - /* now matrix U is upper triangular */ - if (fabs(u(n,n)) < tol) - { /* u[n,n] is too small in magnitude */ - return 2; - } -# undef f -# undef u - return 0; -} - -/*********************************************************************** -* The routine givens computes the parameters of Givens plane rotation -* c = cos(teta) and s = sin(teta) such that: -* -* ( c -s ) ( a ) ( r ) -* ( ) ( ) = ( ) , -* ( s c ) ( b ) ( 0 ) -* -* where a and b are given scalars. -* -* REFERENCES -* -* G.H.Golub, C.F.Van Loan, "Matrix Computations", 2nd ed. */ - -static void givens(double a, double b, double *c, double *s) -{ /* non-optimized version */ - double t; - if (b == 0.0) - (*c) = 1.0, (*s) = 0.0; - else if (fabs(a) <= fabs(b)) - t = - a / b, (*s) = 1.0 / sqrt(1.0 + t * t), (*c) = (*s) * t; - else - t = - b / a, (*c) = 1.0 / sqrt(1.0 + t * t), (*s) = (*c) * t; - return; -} - -/*********************************************************************** -* ifu_gr_update - update IFU-factorization (Givens rotations) -* -* This routine updates IFU-factorization of the matrix A according to -* its expansion (see comments to the routine ifu_expand). The routine -* is based on Givens plane rotations [1]. -* -* RETURNS -* -* 0 The factorization has been successfully updated. -* -* 1 On some elimination step both elements u[k,k] and u[n,k] are too -* small in magnitude. -* -* 2 Diagonal element u[n,n] is too small in magnitude (at the end of -* update). -* -* REFERENCES -* -* 1. G.H.Golub, C.F.Van Loan, "Matrix Computations", 2nd ed. */ - -int ifu_gr_update(IFU *ifu, double c[/*1+n*/], double r[/*1+n*/], - double d) -{ /* non-optimized version */ - int n_max = ifu->n_max; - int n = ifu->n; - double *f_ = ifu->f; - double *u_ = ifu->u; -#if 1 /* FIXME */ - double tol = 1e-5; -#endif - int j, k; - double cs, sn; -# define f(i,j) f_[(i)*n_max+(j)] -# define u(i,j) u_[(i)*n_max+(j)] - /* expand factorization */ - ifu_expand(ifu, c, r, d); - /* NOTE: n keeps its old value */ - /* eliminate spike (non-zero subdiagonal elements) in last row of - * matrix U */ - for (k = 0; k < n; k++) - { /* check if elements u[k,k] and u[n,k] are eligible */ - if (fabs(u(k,k)) < tol && fabs(u(n,k)) < tol) - { /* both u[k,k] and u[n,k] are too small in magnitude */ - return 1; - } - /* if u[n,k] = 0, elimination is not needed */ - if (u(n,k) == 0.0) - continue; - /* compute parameters of Givens plane rotation */ - givens(u(k,k), u(n,k), &cs, &sn); - /* apply Givens rotation to k-th and n-th rows of matrix U to - * eliminate u[n,k] */ - for (j = k; j <= n; j++) - { double ukj = u(k,j), unj = u(n,j); - u(k,j) = cs * ukj - sn * unj; - u(n,j) = sn * ukj + cs * unj; - } - /* apply the same transformation to matrix F to keep the main - * equality F * A = U */ - for (j = 0; j <= n; j++) - { double fkj = f(k,j), fnj = f(n,j); - f(k,j) = cs * fkj - sn * fnj; - f(n,j) = sn * fkj + cs * fnj; - } - } - /* now matrix U is upper triangular */ - if (fabs(u(n,n)) < tol) - { /* u[n,n] is too small in magnitude */ - return 2; - } -# undef f -# undef u - return 0; -} - -/*********************************************************************** -* ifu_a_solve - solve system A * x = b -* -* This routine solves the system A * x = b, where the matrix A is -* specified by its IFU-factorization. -* -* Using the main equality F * A = U we have: -* -* A * x = b => F * A * x = F * b => U * x = F * b => -* -* x = inv(U) * F * b. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix A. On exit this array will contain elements of the solution -* vector x in the same locations. -* -* The working array w should have at least 1+n elements (0-th element -* is not used). */ - -void ifu_a_solve(IFU *ifu, double x[/*1+n*/], double w[/*1+n*/]) -{ /* non-optimized version */ - int n_max = ifu->n_max; - int n = ifu->n; - double *f_ = ifu->f; - double *u_ = ifu->u; - int i, j; - double t; -# define f(i,j) f_[(i)*n_max+(j)] -# define u(i,j) u_[(i)*n_max+(j)] - xassert(0 <= n && n <= n_max); - /* adjust indexing */ - x++, w++; - /* y := F * b */ - memcpy(w, x, n * sizeof(double)); - for (i = 0; i < n; i++) - { /* y[i] := (i-th row of F) * b */ - t = 0.0; - for (j = 0; j < n; j++) - t += f(i,j) * w[j]; - x[i] = t; - } - /* x := inv(U) * y */ - for (i = n-1; i >= 0; i--) - { t = x[i]; - for (j = i+1; j < n; j++) - t -= u(i,j) * x[j]; - x[i] = t / u(i,i); - } -# undef f -# undef u - return; -} - -/*********************************************************************** -* ifu_at_solve - solve system A'* x = b -* -* This routine solves the system A'* x = b, where A' is a matrix -* transposed to the matrix A, specified by its IFU-factorization. -* -* Using the main equality F * A = U, from which it follows that -* A'* F' = U', we have: -* -* A'* x = b => A'* F'* inv(F') * x = b => -* -* U'* inv(F') * x = b => inv(F') * x = inv(U') * b => -* -* x = F' * inv(U') * b. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix A. On exit this array will contain elements of the solution -* vector x in the same locations. -* -* The working array w should have at least 1+n elements (0-th element -* is not used). */ - -void ifu_at_solve(IFU *ifu, double x[/*1+n*/], double w[/*1+n*/]) -{ /* non-optimized version */ - int n_max = ifu->n_max; - int n = ifu->n; - double *f_ = ifu->f; - double *u_ = ifu->u; - int i, j; - double t; -# define f(i,j) f_[(i)*n_max+(j)] -# define u(i,j) u_[(i)*n_max+(j)] - xassert(0 <= n && n <= n_max); - /* adjust indexing */ - x++, w++; - /* y := inv(U') * b */ - for (i = 0; i < n; i++) - { t = (x[i] /= u(i,i)); - for (j = i+1; j < n; j++) - x[j] -= u(i,j) * t; - } - /* x := F'* y */ - for (j = 0; j < n; j++) - { /* x[j] := (j-th column of F) * y */ - t = 0.0; - for (i = 0; i < n; i++) - t += f(i,j) * x[i]; - w[j] = t; - } - memcpy(x, w, n * sizeof(double)); -# undef f -# undef u - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/ifu.h b/code/3rd_glpk/bflib/ifu.h deleted file mode 100644 index 1c67a801..00000000 --- a/code/3rd_glpk/bflib/ifu.h +++ /dev/null @@ -1,99 +0,0 @@ -/* ifu.h (dense updatable IFU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef IFU_H -#define IFU_H - -/*********************************************************************** -* The structure IFU describes dense updatable IFU-factorization. -* -* The IFU-factorization has the following format: -* -* A = inv(F) * U, (1) -* -* where A is a given (unsymmetric) nxn square matrix, F is a square -* matrix, U is an upper triangular matrix. Obviously, the equality (1) -* is equivalent to the following equality: -* -* F * A = U. (2) -* -* It is assumed that matrix A is small and dense, so matrices F and U -* are stored by rows in dense format as follows: -* -* 1 n n_max 1 n n_max -* 1 * * * * * * x x x x 1 * * * * * * x x x x -* * * * * * * x x x x ? * * * * * x x x x -* * * * * * * x x x x ? ? * * * * x x x x -* * * * * * * x x x x ? ? ? * * * x x x x -* * * * * * * x x x x ? ? ? ? * * x x x x -* n * * * * * * x x x x n ? ? ? ? ? * x x x x -* x x x x x x x x x x x x x x x x x x x x -* x x x x x x x x x x x x x x x x x x x x -* x x x x x x x x x x x x x x x x x x x x -* n_max x x x x x x x x x x n_max x x x x x x x x x x -* -* matrix F matrix U -* -* where '*' are matrix elements, '?' are unused locations, 'x' are -* reserved locations. */ - -typedef struct IFU IFU; - -struct IFU -{ /* IFU-factorization */ - int n_max; - /* maximal order of matrices A, F, U; n_max >= 1 */ - int n; - /* current order of matrices A, F, U; 0 <= n <= n_max */ - double *f; /* double f[n_max*n_max]; */ - /* matrix F stored by rows */ - double *u; /* double u[n_max*n_max]; */ - /* matrix U stored by rows */ -}; - -#define ifu_expand _glp_ifu_expand -void ifu_expand(IFU *ifu, double c[/*1+n*/], double r[/*1+n*/], - double d); -/* expand IFU-factorization */ - -#define ifu_bg_update _glp_ifu_bg_update -int ifu_bg_update(IFU *ifu, double c[/*1+n*/], double r[/*1+n*/], - double d); -/* update IFU-factorization (Bartels-Golub) */ - -#define ifu_gr_update _glp_ifu_gr_update -int ifu_gr_update(IFU *ifu, double c[/*1+n*/], double r[/*1+n*/], - double d); -/* update IFU-factorization (Givens rotations) */ - -#define ifu_a_solve _glp_ifu_a_solve -void ifu_a_solve(IFU *ifu, double x[/*1+n*/], double w[/*1+n*/]); -/* solve system A * x = b */ - -#define ifu_at_solve _glp_ifu_at_solve -void ifu_at_solve(IFU *ifu, double x[/*1+n*/], double w[/*1+n*/]); -/* solve system A'* x = b */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/luf.c b/code/3rd_glpk/bflib/luf.c deleted file mode 100644 index 2797407d..00000000 --- a/code/3rd_glpk/bflib/luf.c +++ /dev/null @@ -1,713 +0,0 @@ -/* luf.c (sparse LU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "luf.h" - -/*********************************************************************** -* luf_store_v_cols - store matrix V = A in column-wise format -* -* This routine stores matrix V = A in column-wise format, where A is -* the original matrix to be factorized. -* -* On exit the routine returns the number of non-zeros in matrix V. */ - -int luf_store_v_cols(LUF *luf, int (*col)(void *info, int j, int ind[], - double val[]), void *info, int ind[], double val[]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *vc_cap = &sva->cap[vc_ref-1]; - int j, len, ptr, nnz; - nnz = 0; - for (j = 1; j <= n; j++) - { /* get j-th column */ - len = col(info, j, ind, val); - xassert(0 <= len && len <= n); - /* enlarge j-th column capacity */ - if (vc_cap[j] < len) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vc_ref-1+j, len, 0); - } - /* store j-th column */ - ptr = vc_ptr[j]; - memcpy(&sv_ind[ptr], &ind[1], len * sizeof(int)); - memcpy(&sv_val[ptr], &val[1], len * sizeof(double)); - vc_len[j] = len; - nnz += len; - } - return nnz; -} - -/*********************************************************************** -* luf_check_all - check LU-factorization before k-th elimination step -* -* This routine checks that before performing k-th elimination step, -* 1 <= k <= n+1, all components of the LU-factorization are correct. -* -* In case of k = n+1, i.e. after last elimination step, it is assumed -* that rows of F and columns of V are *not* built yet. -* -* NOTE: For testing/debugging only. */ - -void luf_check_all(LUF *luf, int k) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fr_ref = luf->fr_ref; - int *fr_len = &sva->len[fr_ref-1]; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int *qq_inv = luf->qq_inv; - int i, ii, i_ptr, i_end, j, jj, j_ptr, j_end; - xassert(n > 0); - xassert(1 <= k && k <= n+1); - /* check permutation matrix P */ - for (i = 1; i <= n; i++) - { ii = pp_ind[i]; - xassert(1 <= ii && ii <= n); - xassert(pp_inv[ii] == i); - } - /* check permutation matrix Q */ - for (j = 1; j <= n; j++) - { jj = qq_inv[j]; - xassert(1 <= jj && jj <= n); - xassert(qq_ind[jj] == j); - } - /* check row-wise representation of matrix F */ - for (i = 1; i <= n; i++) - xassert(fr_len[i] == 0); - /* check column-wise representation of matrix F */ - for (j = 1; j <= n; j++) - { /* j-th column of F = jj-th column of L */ - jj = pp_ind[j]; - if (jj < k) - { j_ptr = fc_ptr[j]; - j_end = j_ptr + fc_len[j]; - for (; j_ptr < j_end; j_ptr++) - { i = sv_ind[j_ptr]; - xassert(1 <= i && i <= n); - ii = pp_ind[i]; /* f[i,j] = l[ii,jj] */ - xassert(ii > jj); - xassert(sv_val[j_ptr] != 0.0); - } - } - else /* jj >= k */ - xassert(fc_len[j] == 0); - } - /* check row-wise representation of matrix V */ - for (i = 1; i <= n; i++) - { /* i-th row of V = ii-th row of U */ - ii = pp_ind[i]; - i_ptr = vr_ptr[i]; - i_end = i_ptr + vr_len[i]; - for (; i_ptr < i_end; i_ptr++) - { j = sv_ind[i_ptr]; - xassert(1 <= j && j <= n); - jj = qq_inv[j]; /* v[i,j] = u[ii,jj] */ - if (ii < k) - xassert(jj > ii); - else /* ii >= k */ - { xassert(jj >= k); - /* find v[i,j] in j-th column */ - j_ptr = vc_ptr[j]; - j_end = j_ptr + vc_len[j]; - for (; sv_ind[j_ptr] != i; j_ptr++) - /* nop */; - xassert(j_ptr < j_end); - } - xassert(sv_val[i_ptr] != 0.0); - } - } - /* check column-wise representation of matrix V */ - for (j = 1; j <= n; j++) - { /* j-th column of V = jj-th column of U */ - jj = qq_inv[j]; - if (jj < k) - xassert(vc_len[j] == 0); - else /* jj >= k */ - { j_ptr = vc_ptr[j]; - j_end = j_ptr + vc_len[j]; - for (; j_ptr < j_end; j_ptr++) - { i = sv_ind[j_ptr]; - ii = pp_ind[i]; /* v[i,j] = u[ii,jj] */ - xassert(ii >= k); - /* find v[i,j] in i-th row */ - i_ptr = vr_ptr[i]; - i_end = i_ptr + vr_len[i]; - for (; sv_ind[i_ptr] != j; i_ptr++) - /* nop */; - xassert(i_ptr < i_end); - } - } - } - return; -} - -/*********************************************************************** -* luf_build_v_rows - build matrix V in row-wise format -* -* This routine builds the row-wise representation of matrix V in the -* left part of SVA using its column-wise representation. -* -* NOTE: On entry to the routine all rows of matrix V should have zero -* capacity. -* -* The working array len should have at least 1+n elements (len[0] is -* not used). */ - -void luf_build_v_rows(LUF *luf, int len[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int i, j, end, nnz, ptr, ptr1; - /* calculate the number of non-zeros in each row of matrix V and - * the total number of non-zeros */ - nnz = 0; - for (i = 1; i <= n; i++) - len[i] = 0; - for (j = 1; j <= n; j++) - { nnz += vc_len[j]; - for (end = (ptr = vc_ptr[j]) + vc_len[j]; ptr < end; ptr++) - len[sv_ind[ptr]]++; - } - /* we need at least nnz free locations in SVA */ - if (sva->r_ptr - sva->m_ptr < nnz) - { sva_more_space(sva, nnz); - sv_ind = sva->ind; - sv_val = sva->val; - } - /* reserve locations for rows of matrix V */ - for (i = 1; i <= n; i++) - { if (len[i] > 0) - sva_enlarge_cap(sva, vr_ref-1+i, len[i], 0); - vr_len[i] = len[i]; - } - /* walk thru column of matrix V and build its rows */ - for (j = 1; j <= n; j++) - { for (end = (ptr = vc_ptr[j]) + vc_len[j]; ptr < end; ptr++) - { i = sv_ind[ptr]; - sv_ind[ptr1 = vr_ptr[i] + (--len[i])] = j; - sv_val[ptr1] = sv_val[ptr]; - } - } - return; -} - -/*********************************************************************** -* luf_build_f_rows - build matrix F in row-wise format -* -* This routine builds the row-wise representation of matrix F in the -* right part of SVA using its column-wise representation. -* -* NOTE: On entry to the routine all rows of matrix F should have zero -* capacity. -* -* The working array len should have at least 1+n elements (len[0] is -* not used). */ - -void luf_build_f_rows(LUF *luf, int len[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fr_ref = luf->fr_ref; - int *fr_ptr = &sva->ptr[fr_ref-1]; - int *fr_len = &sva->len[fr_ref-1]; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int i, j, end, nnz, ptr, ptr1; - /* calculate the number of non-zeros in each row of matrix F and - * the total number of non-zeros (except diagonal elements) */ - nnz = 0; - for (i = 1; i <= n; i++) - len[i] = 0; - for (j = 1; j <= n; j++) - { nnz += fc_len[j]; - for (end = (ptr = fc_ptr[j]) + fc_len[j]; ptr < end; ptr++) - len[sv_ind[ptr]]++; - } - /* we need at least nnz free locations in SVA */ - if (sva->r_ptr - sva->m_ptr < nnz) - { sva_more_space(sva, nnz); - sv_ind = sva->ind; - sv_val = sva->val; - } - /* reserve locations for rows of matrix F */ - for (i = 1; i <= n; i++) - { if (len[i] > 0) - sva_reserve_cap(sva, fr_ref-1+i, len[i]); - fr_len[i] = len[i]; - } - /* walk through columns of matrix F and build its rows */ - for (j = 1; j <= n; j++) - { for (end = (ptr = fc_ptr[j]) + fc_len[j]; ptr < end; ptr++) - { i = sv_ind[ptr]; - sv_ind[ptr1 = fr_ptr[i] + (--len[i])] = j; - sv_val[ptr1] = sv_val[ptr]; - } - } - return; -} - -/*********************************************************************** -* luf_build_v_cols - build matrix V in column-wise format -* -* This routine builds the column-wise representation of matrix V in -* the left (if the flag updat is set) or right (if the flag updat is -* clear) part of SVA using its row-wise representation. -* -* NOTE: On entry to the routine all columns of matrix V should have -* zero capacity. -* -* The working array len should have at least 1+n elements (len[0] is -* not used). */ - -void luf_build_v_cols(LUF *luf, int updat, int len[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int i, j, end, nnz, ptr, ptr1; - /* calculate the number of non-zeros in each column of matrix V - * and the total number of non-zeros (except pivot elements) */ - nnz = 0; - for (j = 1; j <= n; j++) - len[j] = 0; - for (i = 1; i <= n; i++) - { nnz += vr_len[i]; - for (end = (ptr = vr_ptr[i]) + vr_len[i]; ptr < end; ptr++) - len[sv_ind[ptr]]++; - } - /* we need at least nnz free locations in SVA */ - if (sva->r_ptr - sva->m_ptr < nnz) - { sva_more_space(sva, nnz); - sv_ind = sva->ind; - sv_val = sva->val; - } - /* reserve locations for columns of matrix V */ - for (j = 1; j <= n; j++) - { if (len[j] > 0) - { if (updat) - sva_enlarge_cap(sva, vc_ref-1+j, len[j], 0); - else - sva_reserve_cap(sva, vc_ref-1+j, len[j]); - } - vc_len[j] = len[j]; - } - /* walk through rows of matrix V and build its columns */ - for (i = 1; i <= n; i++) - { for (end = (ptr = vr_ptr[i]) + vr_len[i]; ptr < end; ptr++) - { j = sv_ind[ptr]; - sv_ind[ptr1 = vc_ptr[j] + (--len[j])] = i; - sv_val[ptr1] = sv_val[ptr]; - } - } - return; -} - -/*********************************************************************** -* luf_check_f_rc - check rows and columns of matrix F -* -* This routine checks that the row- and column-wise representations -* of matrix F are identical. -* -* NOTE: For testing/debugging only. */ - -void luf_check_f_rc(LUF *luf) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fr_ref = luf->fr_ref; - int *fr_ptr = &sva->ptr[fr_ref-1]; - int *fr_len = &sva->len[fr_ref-1]; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int i, i_end, i_ptr, j, j_end, j_ptr; - /* walk thru rows of matrix F */ - for (i = 1; i <= n; i++) - { for (i_end = (i_ptr = fr_ptr[i]) + fr_len[i]; - i_ptr < i_end; i_ptr++) - { j = sv_ind[i_ptr]; - /* find element f[i,j] in j-th column of matrix F */ - for (j_end = (j_ptr = fc_ptr[j]) + fc_len[j]; - sv_ind[j_ptr] != i; j_ptr++) - /* nop */; - xassert(j_ptr < j_end); - xassert(sv_val[i_ptr] == sv_val[j_ptr]); - /* mark element f[i,j] */ - sv_ind[j_ptr] = -i; - } - } - /* walk thru column of matix F and check that all elements has - been marked */ - for (j = 1; j <= n; j++) - { for (j_end = (j_ptr = fc_ptr[j]) + fc_len[j]; - j_ptr < j_end; j_ptr++) - { xassert((i = sv_ind[j_ptr]) < 0); - /* unmark element f[i,j] */ - sv_ind[j_ptr] = -i; - } - } - return; -} - -/*********************************************************************** -* luf_check_v_rc - check rows and columns of matrix V -* -* This routine checks that the row- and column-wise representations -* of matrix V are identical. -* -* NOTE: For testing/debugging only. */ - -void luf_check_v_rc(LUF *luf) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int i, i_end, i_ptr, j, j_end, j_ptr; - /* walk thru rows of matrix V */ - for (i = 1; i <= n; i++) - { for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - i_ptr < i_end; i_ptr++) - { j = sv_ind[i_ptr]; - /* find element v[i,j] in j-th column of matrix V */ - for (j_end = (j_ptr = vc_ptr[j]) + vc_len[j]; - sv_ind[j_ptr] != i; j_ptr++) - /* nop */; - xassert(j_ptr < j_end); - xassert(sv_val[i_ptr] == sv_val[j_ptr]); - /* mark element v[i,j] */ - sv_ind[j_ptr] = -i; - } - } - /* walk thru column of matix V and check that all elements has - been marked */ - for (j = 1; j <= n; j++) - { for (j_end = (j_ptr = vc_ptr[j]) + vc_len[j]; - j_ptr < j_end; j_ptr++) - { xassert((i = sv_ind[j_ptr]) < 0); - /* unmark element v[i,j] */ - sv_ind[j_ptr] = -i; - } - } - return; -} - -/*********************************************************************** -* luf_f_solve - solve system F * x = b -* -* This routine solves the system F * x = b, where the matrix F is the -* left factor of the sparse LU-factorization. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix F. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void luf_f_solve(LUF *luf, double x[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int *pp_inv = luf->pp_inv; - int j, k, ptr, end; - double x_j; - for (k = 1; k <= n; k++) - { /* k-th column of L = j-th column of F */ - j = pp_inv[k]; - /* x[j] is already computed */ - /* walk thru j-th column of matrix F and substitute x[j] into - * other equations */ - if ((x_j = x[j]) != 0.0) - { for (end = (ptr = fc_ptr[j]) + fc_len[j]; ptr < end; ptr++) - x[sv_ind[ptr]] -= sv_val[ptr] * x_j; - } - } - return; -} - -/*********************************************************************** -* luf_ft_solve - solve system F' * x = b -* -* This routine solves the system F' * x = b, where F' is a matrix -* transposed to the matrix F, which is the left factor of the sparse -* LU-factorization. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix F. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void luf_ft_solve(LUF *luf, double x[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fr_ref = luf->fr_ref; - int *fr_ptr = &sva->ptr[fr_ref-1]; - int *fr_len = &sva->len[fr_ref-1]; - int *pp_inv = luf->pp_inv; - int i, k, ptr, end; - double x_i; - for (k = n; k >= 1; k--) - { /* k-th column of L' = i-th row of F */ - i = pp_inv[k]; - /* x[i] is already computed */ - /* walk thru i-th row of matrix F and substitute x[i] into - * other equations */ - if ((x_i = x[i]) != 0.0) - { for (end = (ptr = fr_ptr[i]) + fr_len[i]; ptr < end; ptr++) - x[sv_ind[ptr]] -= sv_val[ptr] * x_i; - } - } - return; -} - -/*********************************************************************** -* luf_v_solve - solve system V * x = b -* -* This routine solves the system V * x = b, where the matrix V is the -* right factor of the sparse LU-factorization. -* -* On entry the array b should contain elements of the right-hand side -* vector b in locations b[1], ..., b[n], where n is the order of the -* matrix V. On exit the array x will contain elements of the solution -* vector x in locations x[1], ..., x[n]. Note that the array b will be -* clobbered on exit. */ - -void luf_v_solve(LUF *luf, double b[/*1+n*/], double x[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - double *vr_piv = luf->vr_piv; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int i, j, k, ptr, end; - double x_j; - for (k = n; k >= 1; k--) - { /* k-th row of U = i-th row of V */ - /* k-th column of U = j-th column of V */ - i = pp_inv[k]; - j = qq_ind[k]; - /* compute x[j] = b[i] / u[k,k], where u[k,k] = v[i,j]; - * walk through j-th column of matrix V and substitute x[j] - * into other equations */ - if ((x_j = x[j] = b[i] / vr_piv[i]) != 0.0) - { for (end = (ptr = vc_ptr[j]) + vc_len[j]; ptr < end; ptr++) - b[sv_ind[ptr]] -= sv_val[ptr] * x_j; - } - } - return; -} - -/*********************************************************************** -* luf_vt_solve - solve system V' * x = b -* -* This routine solves the system V' * x = b, where V' is a matrix -* transposed to the matrix V, which is the right factor of the sparse -* LU-factorization. -* -* On entry the array b should contain elements of the right-hand side -* vector b in locations b[1], ..., b[n], where n is the order of the -* matrix V. On exit the array x will contain elements of the solution -* vector x in locations x[1], ..., x[n]. Note that the array b will be -* clobbered on exit. */ - -void luf_vt_solve(LUF *luf, double b[/*1+n*/], double x[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - double *vr_piv = luf->vr_piv; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int i, j, k, ptr, end; - double x_i; - for (k = 1; k <= n; k++) - { /* k-th row of U' = j-th column of V */ - /* k-th column of U' = i-th row of V */ - i = pp_inv[k]; - j = qq_ind[k]; - /* compute x[i] = b[j] / u'[k,k], where u'[k,k] = v[i,j]; - * walk through i-th row of matrix V and substitute x[i] into - * other equations */ - if ((x_i = x[i] = b[j] / vr_piv[i]) != 0.0) - { for (end = (ptr = vr_ptr[i]) + vr_len[i]; ptr < end; ptr++) - b[sv_ind[ptr]] -= sv_val[ptr] * x_i; - } - } - return; -} - -/*********************************************************************** -* luf_vt_solve1 - solve system V' * y = e' to cause growth in y -* -* This routine is a special version of luf_vt_solve. It solves the -* system V'* y = e' = e + delta e, where V' is a matrix transposed to -* the matrix V, e is the specified right-hand side vector, and delta e -* is a vector of +1 and -1 chosen to cause growth in the solution -* vector y. -* -* On entry the array e should contain elements of the right-hand side -* vector e in locations e[1], ..., e[n], where n is the order of the -* matrix V. On exit the array y will contain elements of the solution -* vector y in locations y[1], ..., y[n]. Note that the array e will be -* clobbered on exit. */ - -void luf_vt_solve1(LUF *luf, double e[/*1+n*/], double y[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - double *vr_piv = luf->vr_piv; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int i, j, k, ptr, end; - double e_j, y_i; - for (k = 1; k <= n; k++) - { /* k-th row of U' = j-th column of V */ - /* k-th column of U' = i-th row of V */ - i = pp_inv[k]; - j = qq_ind[k]; - /* determine e'[j] = e[j] + delta e[j] */ - e_j = (e[j] >= 0.0 ? e[j] + 1.0 : e[j] - 1.0); - /* compute y[i] = e'[j] / u'[k,k], where u'[k,k] = v[i,j] */ - y_i = y[i] = e_j / vr_piv[i]; - /* walk through i-th row of matrix V and substitute y[i] into - * other equations */ - for (end = (ptr = vr_ptr[i]) + vr_len[i]; ptr < end; ptr++) - e[sv_ind[ptr]] -= sv_val[ptr] * y_i; - } - return; -} - -/*********************************************************************** -* luf_estimate_norm - estimate 1-norm of inv(A) -* -* This routine estimates 1-norm of inv(A) by one step of inverse -* iteration for the small singular vector as described in [1]. This -* involves solving two systems of equations: -* -* A'* y = e, -* -* A * z = y, -* -* where A' is a matrix transposed to A, and e is a vector of +1 and -1 -* chosen to cause growth in y. Then -* -* estimate 1-norm of inv(A) = (1-norm of z) / (1-norm of y) -* -* REFERENCES -* -* 1. G.E.Forsythe, M.A.Malcolm, C.B.Moler. Computer Methods for -* Mathematical Computations. Prentice-Hall, Englewood Cliffs, N.J., -* pp. 30-62 (subroutines DECOMP and SOLVE). */ - -double luf_estimate_norm(LUF *luf, double w1[/*1+n*/], double - w2[/*1+n*/]) -{ int n = luf->n; - double *e = w1; - double *y = w2; - double *z = w1; - int i; - double y_norm, z_norm; - /* y = inv(A') * e = inv(F') * inv(V') * e */ - /* compute y' = inv(V') * e to cause growth in y' */ - for (i = 1; i <= n; i++) - e[i] = 0.0; - luf_vt_solve1(luf, e, y); - /* compute y = inv(F') * y' */ - luf_ft_solve(luf, y); - /* compute 1-norm of y = sum |y[i]| */ - y_norm = 0.0; - for (i = 1; i <= n; i++) - y_norm += (y[i] >= 0.0 ? +y[i] : -y[i]); - /* z = inv(A) * y = inv(V) * inv(F) * y */ - /* compute z' = inv(F) * y */ - luf_f_solve(luf, y); - /* compute z = inv(V) * z' */ - luf_v_solve(luf, y, z); - /* compute 1-norm of z = sum |z[i]| */ - z_norm = 0.0; - for (i = 1; i <= n; i++) - z_norm += (z[i] >= 0.0 ? +z[i] : -z[i]); - /* estimate 1-norm of inv(A) = (1-norm of z) / (1-norm of y) */ - return z_norm / y_norm; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/luf.h b/code/3rd_glpk/bflib/luf.h deleted file mode 100644 index 5634a753..00000000 --- a/code/3rd_glpk/bflib/luf.h +++ /dev/null @@ -1,227 +0,0 @@ -/* luf.h (sparse LU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef LUF_H -#define LUF_H - -#include "sva.h" - -/*********************************************************************** -* The structure LUF describes sparse LU-factorization. -* -* The LU-factorization has the following format: -* -* A = F * V = P * L * U * Q, (1) -* -* F = P * L * P', (2) -* -* V = P * U * Q, (3) -* -* where A is a given (unsymmetric) square matrix, F and V are matrix -* factors actually computed, L is a lower triangular matrix with unity -* diagonal, U is an upper triangular matrix, P and Q are permutation -* matrices, P' is a matrix transposed to P. All the matrices have the -* same order n. -* -* Matrices F and V are stored in both row- and column-wise sparse -* formats in the associated sparse vector area (SVA). Unity diagonal -* elements of matrix F are not stored. Pivot elements of matrix V -* (which correspond to diagonal elements of matrix U) are stored in -* a separate ordinary array. -* -* Permutation matrices P and Q are stored in ordinary arrays in both -* row- and column-like formats. -* -* Matrices L and U are completely defined by matrices F, V, P, and Q, -* and therefore not stored explicitly. */ - -typedef struct LUF LUF; - -struct LUF -{ /* sparse LU-factorization */ - int n; - /* order of matrices A, F, V, P, Q */ - SVA *sva; - /* associated sparse vector area (SVA) used to store rows and - * columns of matrices F and V; note that different objects may - * share the same SVA */ - /*--------------------------------------------------------------*/ - /* matrix F in row-wise format */ - /* during the factorization process this object is not used */ - int fr_ref; - /* reference number of sparse vector in SVA, which is the first - * row of matrix F */ -#if 0 + 0 - int *fr_ptr = &sva->ptr[fr_ref-1]; - /* fr_ptr[0] is not used; - * fr_ptr[i], 1 <= i <= n, is pointer to i-th row in SVA */ - int *fr_len = &sva->len[fr_ref-1]; - /* fr_len[0] is not used; - * fr_len[i], 1 <= i <= n, is length of i-th row */ -#endif - /*--------------------------------------------------------------*/ - /* matrix F in column-wise format */ - /* during the factorization process this object is constructed - * by columns */ - int fc_ref; - /* reference number of sparse vector in SVA, which is the first - * column of matrix F */ -#if 0 + 0 - int *fc_ptr = &sva->ptr[fc_ref-1]; - /* fc_ptr[0] is not used; - * fc_ptr[j], 1 <= j <= n, is pointer to j-th column in SVA */ - int *fc_len = &sva->len[fc_ref-1]; - /* fc_len[0] is not used; - * fc_len[j], 1 <= j <= n, is length of j-th column */ -#endif - /*--------------------------------------------------------------*/ - /* matrix V in row-wise format */ - int vr_ref; - /* reference number of sparse vector in SVA, which is the first - * row of matrix V */ -#if 0 + 0 - int *vr_ptr = &sva->ptr[vr_ref-1]; - /* vr_ptr[0] is not used; - * vr_ptr[i], 1 <= i <= n, is pointer to i-th row in SVA */ - int *vr_len = &sva->len[vr_ref-1]; - /* vr_len[0] is not used; - * vr_len[i], 1 <= i <= n, is length of i-th row */ - int *vr_cap = &sva->cap[vr_ref-1]; - /* vr_cap[0] is not used; - * vr_cap[i], 1 <= i <= n, is capacity of i-th row */ -#endif - double *vr_piv; /* double vr_piv[1+n]; */ - /* vr_piv[0] is not used; - * vr_piv[i], 1 <= i <= n, is pivot element of i-th row */ - /*--------------------------------------------------------------*/ - /* matrix V in column-wise format */ - /* during the factorization process this object contains only the - * patterns (row indices) of columns of the active submatrix */ - int vc_ref; - /* reference number of sparse vector in SVA, which is the first - * column of matrix V */ -#if 0 + 0 - int *vc_ptr = &sva->ptr[vc_ref-1]; - /* vc_ptr[0] is not used; - * vc_ptr[j], 1 <= j <= n, is pointer to j-th column in SVA */ - int *vc_len = &sva->len[vc_ref-1]; - /* vc_len[0] is not used; - * vc_len[j], 1 <= j <= n, is length of j-th column */ - int *vc_cap = &sva->cap[vc_ref-1]; - /* vc_cap[0] is not used; - * vc_cap[j], 1 <= j <= n, is capacity of j-th column */ -#endif - /*--------------------------------------------------------------*/ - /* matrix P */ - int *pp_ind; /* int pp_ind[1+n]; */ - /* pp_ind[i] = j means that P[i,j] = 1 */ - int *pp_inv; /* int pp_inv[1+n]; */ - /* pp_inv[j] = i means that P[i,j] = 1 */ - /* if i-th row or column of matrix F is i'-th row or column of - * matrix L, or if i-th row of matrix V is i'-th row of matrix U, - * then pp_ind[i] = i' and pp_inv[i'] = i */ - /*--------------------------------------------------------------*/ - /* matrix Q */ - int *qq_ind; /* int qq_ind[1+n]; */ - /* qq_ind[i] = j means that Q[i,j] = 1 */ - int *qq_inv; /* int qq_inv[1+n]; */ - /* qq_inv[j] = i means that Q[i,j] = 1 */ - /* if j-th column of matrix V is j'-th column of matrix U, then - * qq_ind[j'] = j and qq_inv[j] = j' */ -}; - -#define luf_swap_u_rows(i1, i2) \ - do \ - { int j1, j2; \ - j1 = pp_inv[i1], j2 = pp_inv[i2]; \ - pp_ind[j1] = i2, pp_inv[i2] = j1; \ - pp_ind[j2] = i1, pp_inv[i1] = j2; \ - } while (0) -/* swap rows i1 and i2 of matrix U = P'* V * Q' */ - -#define luf_swap_u_cols(j1, j2) \ - do \ - { int i1, i2; \ - i1 = qq_ind[j1], i2 = qq_ind[j2]; \ - qq_ind[j1] = i2, qq_inv[i2] = j1; \ - qq_ind[j2] = i1, qq_inv[i1] = j2; \ - } while (0) -/* swap columns j1 and j2 of matrix U = P'* V * Q' */ - -#define luf_store_v_cols _glp_luf_store_v_cols -int luf_store_v_cols(LUF *luf, int (*col)(void *info, int j, int ind[], - double val[]), void *info, int ind[], double val[]); -/* store matrix V = A in column-wise format */ - -#define luf_check_all _glp_luf_check_all -void luf_check_all(LUF *luf, int k); -/* check LU-factorization before k-th elimination step */ - -#define luf_build_v_rows _glp_luf_build_v_rows -void luf_build_v_rows(LUF *luf, int len[/*1+n*/]); -/* build matrix V in row-wise format */ - -#define luf_build_f_rows _glp_luf_build_f_rows -void luf_build_f_rows(LUF *luf, int len[/*1+n*/]); -/* build matrix F in row-wise format */ - -#define luf_build_v_cols _glp_luf_build_v_cols -void luf_build_v_cols(LUF *luf, int updat, int len[/*1+n*/]); -/* build matrix V in column-wise format */ - -#define luf_check_f_rc _glp_luf_check_f_rc -void luf_check_f_rc(LUF *luf); -/* check rows and columns of matrix F */ - -#define luf_check_v_rc _glp_luf_check_v_rc -void luf_check_v_rc(LUF *luf); -/* check rows and columns of matrix V */ - -#define luf_f_solve _glp_luf_f_solve -void luf_f_solve(LUF *luf, double x[/*1+n*/]); -/* solve system F * x = b */ - -#define luf_ft_solve _glp_luf_ft_solve -void luf_ft_solve(LUF *luf, double x[/*1+n*/]); -/* solve system F' * x = b */ - -#define luf_v_solve _glp_luf_v_solve -void luf_v_solve(LUF *luf, double b[/*1+n*/], double x[/*1+n*/]); -/* solve system V * x = b */ - -#define luf_vt_solve _glp_luf_vt_solve -void luf_vt_solve(LUF *luf, double b[/*1+n*/], double x[/*1+n*/]); -/* solve system V' * x = b */ - -#define luf_vt_solve1 _glp_luf_vt_solve1 -void luf_vt_solve1(LUF *luf, double e[/*1+n*/], double y[/*1+n*/]); -/* solve system V' * y = e' to cause growth in y */ - -#define luf_estimate_norm _glp_luf_estimate_norm -double luf_estimate_norm(LUF *luf, double w1[/*1+n*/], double - w2[/*1+n*/]); -/* estimate 1-norm of inv(A) */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/lufint.c b/code/3rd_glpk/bflib/lufint.c deleted file mode 100644 index 7cd00924..00000000 --- a/code/3rd_glpk/bflib/lufint.c +++ /dev/null @@ -1,182 +0,0 @@ -/* lufint.c (interface to LU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "lufint.h" - -LUFINT *lufint_create(void) -{ /* create interface to LU-factorization */ - LUFINT *fi; - fi = talloc(1, LUFINT); - fi->n_max = 0; - fi->valid = 0; - fi->sva = NULL; - fi->luf = NULL; - fi->sgf = NULL; - fi->sva_n_max = fi->sva_size = 0; - fi->delta_n0 = fi->delta_n = 0; - fi->sgf_updat = 0; - fi->sgf_piv_tol = 0.10; - fi->sgf_piv_lim = 4; - fi->sgf_suhl = 1; - fi->sgf_eps_tol = DBL_EPSILON; - return fi; -} - -int lufint_factorize(LUFINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info) -{ /* compute LU-factorization of specified matrix A */ - SVA *sva; - LUF *luf; - SGF *sgf; - int k; - xassert(n > 0); - fi->valid = 0; - /* create sparse vector area (SVA), if necessary */ - sva = fi->sva; - if (sva == NULL) - { int sva_n_max = fi->sva_n_max; - int sva_size = fi->sva_size; - if (sva_n_max == 0) - sva_n_max = 4 * n; - if (sva_size == 0) - sva_size = 10 * n; - sva = fi->sva = sva_create_area(sva_n_max, sva_size); - } - /* allocate/reallocate underlying objects, if necessary */ - if (fi->n_max < n) - { int n_max = fi->n_max; - if (n_max == 0) - n_max = fi->n_max = n + fi->delta_n0; - else - n_max = fi->n_max = n + fi->delta_n; - xassert(n_max >= n); - /* allocate/reallocate LU-factorization (LUF) */ - luf = fi->luf; - if (luf == NULL) - { luf = fi->luf = talloc(1, LUF); - memset(luf, 0, sizeof(LUF)); - luf->sva = sva; - } - else - { tfree(luf->vr_piv); - tfree(luf->pp_ind); - tfree(luf->pp_inv); - tfree(luf->qq_ind); - tfree(luf->qq_inv); - } - luf->vr_piv = talloc(1+n_max, double); - luf->pp_ind = talloc(1+n_max, int); - luf->pp_inv = talloc(1+n_max, int); - luf->qq_ind = talloc(1+n_max, int); - luf->qq_inv = talloc(1+n_max, int); - /* allocate/reallocate factorizer workspace (SGF) */ - sgf = fi->sgf; - if (sgf == NULL) - { sgf = fi->sgf = talloc(1, SGF); - memset(sgf, 0, sizeof(SGF)); - sgf->luf = luf; - } - else - { tfree(sgf->rs_head); - tfree(sgf->rs_prev); - tfree(sgf->rs_next); - tfree(sgf->cs_head); - tfree(sgf->cs_prev); - tfree(sgf->cs_next); - tfree(sgf->vr_max); - tfree(sgf->flag); - tfree(sgf->work); - } - sgf->rs_head = talloc(1+n_max, int); - sgf->rs_prev = talloc(1+n_max, int); - sgf->rs_next = talloc(1+n_max, int); - sgf->cs_head = talloc(1+n_max, int); - sgf->cs_prev = talloc(1+n_max, int); - sgf->cs_next = talloc(1+n_max, int); - sgf->vr_max = talloc(1+n_max, double); - sgf->flag = talloc(1+n_max, char); - sgf->work = talloc(1+n_max, double); - } - luf = fi->luf; - sgf = fi->sgf; -#if 1 /* FIXME */ - /* initialize SVA */ - sva->n = 0; - sva->m_ptr = 1; - sva->r_ptr = sva->size + 1; - sva->head = sva->tail = 0; -#endif - /* allocate sparse vectors in SVA */ - luf->n = n; - luf->fr_ref = sva_alloc_vecs(sva, n); - luf->fc_ref = sva_alloc_vecs(sva, n); - luf->vr_ref = sva_alloc_vecs(sva, n); - luf->vc_ref = sva_alloc_vecs(sva, n); - /* store matrix V = A in column-wise format */ - luf_store_v_cols(luf, col, info, sgf->rs_prev, sgf->work); - /* setup factorizer control parameters */ - sgf->updat = fi->sgf_updat; - sgf->piv_tol = fi->sgf_piv_tol; - sgf->piv_lim = fi->sgf_piv_lim; - sgf->suhl = fi->sgf_suhl; - sgf->eps_tol = fi->sgf_eps_tol; - /* compute LU-factorization of specified matrix A */ - k = sgf_factorize(sgf, 1); - if (k == 0) - fi->valid = 1; - return k; -} - -void lufint_delete(LUFINT *fi) -{ /* delete interface to LU-factorization */ - SVA *sva = fi->sva; - LUF *luf = fi->luf; - SGF *sgf = fi->sgf; - if (sva != NULL) - sva_delete_area(sva); - if (luf != NULL) - { tfree(luf->vr_piv); - tfree(luf->pp_ind); - tfree(luf->pp_inv); - tfree(luf->qq_ind); - tfree(luf->qq_inv); - tfree(luf); - } - if (sgf != NULL) - { tfree(sgf->rs_head); - tfree(sgf->rs_prev); - tfree(sgf->rs_next); - tfree(sgf->cs_head); - tfree(sgf->cs_prev); - tfree(sgf->cs_next); - tfree(sgf->vr_max); - tfree(sgf->flag); - tfree(sgf->work); - tfree(sgf); - } - tfree(fi); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/lufint.h b/code/3rd_glpk/bflib/lufint.h deleted file mode 100644 index b3ad5b64..00000000 --- a/code/3rd_glpk/bflib/lufint.h +++ /dev/null @@ -1,73 +0,0 @@ -/* lufint.h (interface to LU-factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef LUFINT_H -#define LUFINT_H - -#include "sgf.h" - -typedef struct LUFINT LUFINT; - -struct LUFINT -{ /* interface to LU-factorization */ - int n_max; - /* maximal value of n (increased automatically) */ - int valid; - /* factorization is valid only if this flag is set */ - SVA *sva; - /* sparse vector area (SVA) */ - LUF *luf; - /* sparse LU-factorization */ - SGF *sgf; - /* sparse Gaussian factorizer workspace */ - /*--------------------------------------------------------------*/ - /* control parameters */ - int sva_n_max, sva_size; - /* parameters passed to sva_create_area */ - int delta_n0, delta_n; - /* if n_max = 0, set n_max = n + delta_n0 - * if n_max < n, set n_max = n + delta_n */ - int sgf_updat; - double sgf_piv_tol; - int sgf_piv_lim; - int sgf_suhl; - double sgf_eps_tol; - /* factorizer control parameters */ -}; - -#define lufint_create _glp_lufint_create -LUFINT *lufint_create(void); -/* create interface to LU-factorization */ - -#define lufint_factorize _glp_lufint_factorize -int lufint_factorize(LUFINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info); -/* compute LU-factorization of specified matrix A */ - -#define lufint_delete _glp_lufint_delete -void lufint_delete(LUFINT *fi); -/* delete interface to LU-factorization */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/scf.c b/code/3rd_glpk/bflib/scf.c deleted file mode 100644 index 556b1911..00000000 --- a/code/3rd_glpk/bflib/scf.c +++ /dev/null @@ -1,523 +0,0 @@ -/* scf.c (sparse updatable Schur-complement-based factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "scf.h" - -/*********************************************************************** -* scf_r0_solve - solve system R0 * x = b or R0'* x = b -* -* This routine solves the system R0 * x = b (if tr is zero) or the -* system R0'* x = b (if tr is non-zero), where R0 is the left factor -* of the initial matrix A0 = R0 * S0. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n0], where n0 is the order of the -* matrix R0. On exit the array x will contain elements of the solution -* vector in the same locations. */ - -void scf_r0_solve(SCF *scf, int tr, double x[/*1+n0*/]) -{ switch (scf->type) - { case 1: - /* A0 = F0 * V0, so R0 = F0 */ - if (!tr) - luf_f_solve(scf->a0.luf, x); - else - luf_ft_solve(scf->a0.luf, x); - break; - case 2: - /* A0 = I * A0, so R0 = I */ - break; - default: - xassert(scf != scf); - } - return; -} - -/*********************************************************************** -* scf_s0_solve - solve system S0 * x = b or S0'* x = b -* -* This routine solves the system S0 * x = b (if tr is zero) or the -* system S0'* x = b (if tr is non-zero), where S0 is the right factor -* of the initial matrix A0 = R0 * S0. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n0], where n0 is the order of the -* matrix S0. On exit the array x will contain elements of the solution -* vector in the same locations. -* -* The routine uses locations [1], ..., [n0] of three working arrays -* w1, w2, and w3. (In case of type = 1 arrays w2 and w3 are not used -* and can be specified as NULL.) */ - -void scf_s0_solve(SCF *scf, int tr, double x[/*1+n0*/], - double w1[/*1+n0*/], double w2[/*1+n0*/], double w3[/*1+n0*/]) -{ int n0 = scf->n0; - switch (scf->type) - { case 1: - /* A0 = F0 * V0, so S0 = V0 */ - if (!tr) - luf_v_solve(scf->a0.luf, x, w1); - else - luf_vt_solve(scf->a0.luf, x, w1); - break; - case 2: - /* A0 = I * A0, so S0 = A0 */ - if (!tr) - btf_a_solve(scf->a0.btf, x, w1, w2, w3); - else - btf_at_solve(scf->a0.btf, x, w1, w2, w3); - break; - default: - xassert(scf != scf); - } - memcpy(&x[1], &w1[1], n0 * sizeof(double)); - return; -} - -/*********************************************************************** -* scf_r_prod - compute product y := y + alpha * R * x -* -* This routine computes the product y := y + alpha * R * x, where -* x is a n0-vector, alpha is a scalar, y is a nn-vector. -* -* Since matrix R is available by rows, the product components are -* computed as inner products: -* -* y[i] = y[i] + alpha * (i-th row of R) * x -* -* for i = 1, 2, ..., nn. */ - -void scf_r_prod(SCF *scf, double y[/*1+nn*/], double a, const double - x[/*1+n0*/]) -{ int nn = scf->nn; - SVA *sva = scf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int rr_ref = scf->rr_ref; - int *rr_ptr = &sva->ptr[rr_ref-1]; - int *rr_len = &sva->len[rr_ref-1]; - int i, ptr, end; - double t; - for (i = 1; i <= nn; i++) - { /* t := (i-th row of R) * x */ - t = 0.0; - for (end = (ptr = rr_ptr[i]) + rr_len[i]; ptr < end; ptr++) - t += sv_val[ptr] * x[sv_ind[ptr]]; - /* y[i] := y[i] + alpha * t */ - y[i] += a * t; - } - return; -} - -/*********************************************************************** -* scf_rt_prod - compute product y := y + alpha * R'* x -* -* This routine computes the product y := y + alpha * R'* x, where -* R' is a matrix transposed to R, x is a nn-vector, alpha is a scalar, -* y is a n0-vector. -* -* Since matrix R is available by rows, the product is computed as a -* linear combination: -* -* y := y + alpha * (R'[1] * x[1] + ... + R'[nn] * x[nn]), -* -* where R'[i] is i-th row of R. */ - -void scf_rt_prod(SCF *scf, double y[/*1+n0*/], double a, const double - x[/*1+nn*/]) -{ int nn = scf->nn; - SVA *sva = scf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int rr_ref = scf->rr_ref; - int *rr_ptr = &sva->ptr[rr_ref-1]; - int *rr_len = &sva->len[rr_ref-1]; - int i, ptr, end; - double t; - for (i = 1; i <= nn; i++) - { if (x[i] == 0.0) - continue; - /* y := y + alpha * R'[i] * x[i] */ - t = a * x[i]; - for (end = (ptr = rr_ptr[i]) + rr_len[i]; ptr < end; ptr++) - y[sv_ind[ptr]] += sv_val[ptr] * t; - } - return; -} - -/*********************************************************************** -* scf_s_prod - compute product y := y + alpha * S * x -* -* This routine computes the product y := y + alpha * S * x, where -* x is a nn-vector, alpha is a scalar, y is a n0 vector. -* -* Since matrix S is available by columns, the product is computed as -* a linear combination: -* -* y := y + alpha * (S[1] * x[1] + ... + S[nn] * x[nn]), -* -* where S[j] is j-th column of S. */ - -void scf_s_prod(SCF *scf, double y[/*1+n0*/], double a, const double - x[/*1+nn*/]) -{ int nn = scf->nn; - SVA *sva = scf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int ss_ref = scf->ss_ref; - int *ss_ptr = &sva->ptr[ss_ref-1]; - int *ss_len = &sva->len[ss_ref-1]; - int j, ptr, end; - double t; - for (j = 1; j <= nn; j++) - { if (x[j] == 0.0) - continue; - /* y := y + alpha * S[j] * x[j] */ - t = a * x[j]; - for (end = (ptr = ss_ptr[j]) + ss_len[j]; ptr < end; ptr++) - y[sv_ind[ptr]] += sv_val[ptr] * t; - } - return; -} - -/*********************************************************************** -* scf_st_prod - compute product y := y + alpha * S'* x -* -* This routine computes the product y := y + alpha * S'* x, where -* S' is a matrix transposed to S, x is a n0-vector, alpha is a scalar, -* y is a nn-vector. -* -* Since matrix S is available by columns, the product components are -* computed as inner products: -* -* y[j] := y[j] + alpha * (j-th column of S) * x -* -* for j = 1, 2, ..., nn. */ - -void scf_st_prod(SCF *scf, double y[/*1+nn*/], double a, const double - x[/*1+n0*/]) -{ int nn = scf->nn; - SVA *sva = scf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int ss_ref = scf->ss_ref; - int *ss_ptr = &sva->ptr[ss_ref-1]; - int *ss_len = &sva->len[ss_ref-1]; - int j, ptr, end; - double t; - for (j = 1; j <= nn; j++) - { /* t := (j-th column of S) * x */ - t = 0.0; - for (end = (ptr = ss_ptr[j]) + ss_len[j]; ptr < end; ptr++) - t += sv_val[ptr] * x[sv_ind[ptr]]; - /* y[j] := y[j] + alpha * t */ - y[j] += a * t; - } - return; -} - -/*********************************************************************** -* scf_a_solve - solve system A * x = b -* -* This routine solves the system A * x = b, where A is the current -* matrix. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix A. On exit the array x will contain elements of the solution -* vector in the same locations. -* -* For details see the program documentation. */ - -void scf_a_solve(SCF *scf, double x[/*1+n*/], - double w[/*1+n0+nn*/], double work1[/*1+max(n0,nn)*/], - double work2[/*1+n*/], double work3[/*1+n*/]) -{ int n = scf->n; - int n0 = scf->n0; - int nn = scf->nn; - int *pp_ind = scf->pp_ind; - int *qq_inv = scf->qq_inv; - int i, ii; - /* (u1, u2) := inv(P) * (b, 0) */ - for (ii = 1; ii <= n0+nn; ii++) - { i = pp_ind[ii]; -#if 1 /* FIXME: currently P = I */ - xassert(i == ii); -#endif - w[ii] = (i <= n ? x[i] : 0.0); - } - /* v1 := inv(R0) * u1 */ - scf_r0_solve(scf, 0, &w[0]); - /* v2 := u2 - R * v1 */ - scf_r_prod(scf, &w[n0], -1.0, &w[0]); - /* w2 := inv(C) * v2 */ - ifu_a_solve(&scf->ifu, &w[n0], work1); - /* w1 := inv(S0) * (v1 - S * w2) */ - scf_s_prod(scf, &w[0], -1.0, &w[n0]); - scf_s0_solve(scf, 0, &w[0], work1, work2, work3); - /* (x, x~) := inv(Q) * (w1, w2); x~ is not needed */ - for (i = 1; i <= n; i++) - x[i] = w[qq_inv[i]]; - return; -} - -/*********************************************************************** -* scf_at_solve - solve system A'* x = b -* -* This routine solves the system A'* x = b, where A' is a matrix -* transposed to the current matrix A. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix A. On exit the array x will contain elements of the solution -* vector in the same locations. -* -* For details see the program documentation. */ - -void scf_at_solve(SCF *scf, double x[/*1+n*/], - double w[/*1+n0+nn*/], double work1[/*1+max(n0,nn)*/], - double work2[/*1+n*/], double work3[/*1+n*/]) -{ int n = scf->n; - int n0 = scf->n0; - int nn = scf->nn; - int *pp_inv = scf->pp_inv; - int *qq_ind = scf->qq_ind; - int i, ii; - /* (u1, u2) := Q * (b, 0) */ - for (ii = 1; ii <= n0+nn; ii++) - { i = qq_ind[ii]; - w[ii] = (i <= n ? x[i] : 0.0); - } - /* v1 := inv(S0') * u1 */ - scf_s0_solve(scf, 1, &w[0], work1, work2, work3); - /* v2 := inv(C') * (u2 - S'* v1) */ - scf_st_prod(scf, &w[n0], -1.0, &w[0]); - ifu_at_solve(&scf->ifu, &w[n0], work1); - /* w2 := v2 */ - /* nop */ - /* w1 := inv(R0') * (v1 - R'* w2) */ - scf_rt_prod(scf, &w[0], -1.0, &w[n0]); - scf_r0_solve(scf, 1, &w[0]); - /* compute (x, x~) := P * (w1, w2); x~ is not needed */ - for (i = 1; i <= n; i++) - { -#if 1 /* FIXME: currently P = I */ - xassert(pp_inv[i] == i); -#endif - x[i] = w[pp_inv[i]]; - } - return; -} - -/*********************************************************************** -* scf_add_r_row - add new row to matrix R -* -* This routine adds new (nn+1)-th row to matrix R, whose elements are -* specified in locations w[1,...,n0]. */ - -void scf_add_r_row(SCF *scf, const double w[/*1+n0*/]) -{ int n0 = scf->n0; - int nn = scf->nn; - SVA *sva = scf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int rr_ref = scf->rr_ref; - int *rr_ptr = &sva->ptr[rr_ref-1]; - int *rr_len = &sva->len[rr_ref-1]; - int j, len, ptr; - xassert(0 <= nn && nn < scf->nn_max); - /* determine length of new row */ - len = 0; - for (j = 1; j <= n0; j++) - { if (w[j] != 0.0) - len++; - } - /* reserve locations for new row in static part of SVA */ - if (len > 0) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, rr_ref + nn, len); - } - /* store new row in sparse format */ - ptr = rr_ptr[nn+1]; - for (j = 1; j <= n0; j++) - { if (w[j] != 0.0) - { sv_ind[ptr] = j; - sv_val[ptr] = w[j]; - ptr++; - } - } - xassert(ptr - rr_ptr[nn+1] == len); - rr_len[nn+1] = len; -#ifdef GLP_DEBUG - sva_check_area(sva); -#endif - return; -} - -/*********************************************************************** -* scf_add_s_col - add new column to matrix S -* -* This routine adds new (nn+1)-th column to matrix S, whose elements -* are specified in locations v[1,...,n0]. */ - -void scf_add_s_col(SCF *scf, const double v[/*1+n0*/]) -{ int n0 = scf->n0; - int nn = scf->nn; - SVA *sva = scf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int ss_ref = scf->ss_ref; - int *ss_ptr = &sva->ptr[ss_ref-1]; - int *ss_len = &sva->len[ss_ref-1]; - int i, len, ptr; - xassert(0 <= nn && nn < scf->nn_max); - /* determine length of new column */ - len = 0; - for (i = 1; i <= n0; i++) - { if (v[i] != 0.0) - len++; - } - /* reserve locations for new column in static part of SVA */ - if (len > 0) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, ss_ref + nn, len); - } - /* store new column in sparse format */ - ptr = ss_ptr[nn+1]; - for (i = 1; i <= n0; i++) - { if (v[i] != 0.0) - { sv_ind[ptr] = i; - sv_val[ptr] = v[i]; - ptr++; - } - } - xassert(ptr - ss_ptr[nn+1] == len); - ss_len[nn+1] = len; -#ifdef GLP_DEBUG - sva_check_area(sva); -#endif - return; -} - -/*********************************************************************** -* scf_update_aug - update factorization of augmented matrix -* -* Given factorization of the current augmented matrix: -* -* ( A0 A1 ) ( R0 ) ( S0 S ) -* ( ) = ( ) ( ), -* ( A2 A3 ) ( R I ) ( C ) -* -* this routine computes factorization of the new augmented matrix: -* -* ( A0 | A1 b ) -* ( ---+------ ) ( A0 A1^ ) ( R0 ) ( S0 S^ ) -* ( A2 | A3 f ) = ( ) = ( ) ( ), -* ( | ) ( A2^ A3^ ) ( R^ I ) ( C^ ) -* ( d' | g' h ) -* -* where b and d are specified n0-vectors, f and g are specified -* nn-vectors, and h is a specified scalar. (Note that corresponding -* arrays are clobbered on exit.) -* -* The parameter upd specifies how to update factorization of the Schur -* complement C: -* -* 1 Bartels-Golub updating. -* -* 2 Givens rotations updating. -* -* The working arrays w1, w2, and w3 are used in the same way as in the -* routine scf_s0_solve. -* -* RETURNS -* -* 0 Factorization has been successfully updated. -* -* 1 Updating limit has been reached. -* -* 2 Updating IFU-factorization of matrix C failed. -* -* For details see the program documentation. */ - -int scf_update_aug(SCF *scf, double b[/*1+n0*/], double d[/*1+n0*/], - double f[/*1+nn*/], double g[/*1+nn*/], double h, int upd, - double w1[/*1+n0*/], double w2[/*1+n0*/], double w3[/*1+n0*/]) -{ int n0 = scf->n0; - int k, ret; - double *v, *w, *x, *y, z; - if (scf->nn == scf->nn_max) - { /* updating limit has been reached */ - return 1; - } - /* v := inv(R0) * b */ - scf_r0_solve(scf, 0, (v = b)); - /* w := inv(S0') * d */ - scf_s0_solve(scf, 1, (w = d), w1, w2, w3); - /* x := f - R * v */ - scf_r_prod(scf, (x = f), -1.0, v); - /* y := g - S'* w */ - scf_st_prod(scf, (y = g), -1.0, w); - /* z := h - v'* w */ - z = h; - for (k = 1; k <= n0; k++) - z -= v[k] * w[k]; - /* new R := R with row w added */ - scf_add_r_row(scf, w); - /* new S := S with column v added */ - scf_add_s_col(scf, v); - /* update IFU-factorization of C */ - switch (upd) - { case 1: - ret = ifu_bg_update(&scf->ifu, x, y, z); - break; - case 2: - ret = ifu_gr_update(&scf->ifu, x, y, z); - break; - default: - xassert(upd != upd); - } - if (ret != 0) - { /* updating IFU-factorization failed */ - return 2; - } - /* increase number of additional rows and columns */ - scf->nn++; - /* expand P and Q */ - k = n0 + scf->nn; - scf->pp_ind[k] = scf->pp_inv[k] = k; - scf->qq_ind[k] = scf->qq_inv[k] = k; - /* factorization has been successfully updated */ - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/scf.h b/code/3rd_glpk/bflib/scf.h deleted file mode 100644 index 69d8cfc2..00000000 --- a/code/3rd_glpk/bflib/scf.h +++ /dev/null @@ -1,211 +0,0 @@ -/* scf.h (sparse updatable Schur-complement-based factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SCF_H -#define SCF_H - -#include "btf.h" -#include "ifu.h" -#include "luf.h" - -/*********************************************************************** -* The structure SCF describes sparse updatable factorization based on -* Schur complement. -* -* The SCF-factorization has the following format: -* -* ( A A1~ ) ( A0 A1 ) ( R0 ) ( S0 S ) -* ( ) = P ( ) Q = P ( ) ( ) Q, (1) -* ( A2~ A3~ ) ( A2 A3 ) ( R I ) ( C ) -* -* where: -* -* A is current (unsymmetric) square matrix (not stored); -* -* A1~, A2~, A3~ are some additional matrices (not stored); -* -* A0 is initial (unsymmetric) square matrix (not stored); -* -* A1, A2, A3 are some additional matrices (not stored); -* -* R0 and S0 are matrices that define factorization of the initial -* matrix A0 = R0 * S0 (stored in an invertable form); -* -* R is a matrix defined from R * S0 = A2, so R = A2 * inv(S0) (stored -* in row-wise sparse format); -* -* S is a matrix defined from R0 * S = A1, so S = inv(R0) * A1 (stored -* in column-wise sparse format); -* -* C is Schur complement (to matrix A0) defined from R * S + C = A3, -* so C = A3 - R * S = A3 - A2 * inv(A0) * A1 (stored in an invertable -* form). -* -* P, Q are permutation matrices (stored in both row- and column-like -* formats). */ - -typedef struct SCF SCF; - -struct SCF -{ /* Schur-complement-based factorization */ - int n; - /* order of current matrix A */ - /*--------------------------------------------------------------*/ - /* initial matrix A0 = R0 * S0 of order n0 in invertable form */ - int n0; - /* order of matrix A0 */ - int type; - /* type of factorization used: - * 1 - LU-factorization (R0 = F0, S0 = V0) - * 2 - BT-factorization (R0 = I, S0 = A0) */ - union - { LUF *luf; /* type = 1 */ - BTF *btf; /* type = 2 */ - } a0; - /* factorization of matrix A0 */ - /*--------------------------------------------------------------*/ - /* augmented matrix (A0, A1; A2, A3) of order n0+nn */ - int nn_max; - /* maximal number of additional rows and columns in the augmented - * matrix (this limits the number of updates) */ - int nn; - /* current number of additional rows and columns in the augmented - * matrix, 0 <= nn <= nn_max */ - SVA *sva; - /* associated sparse vector area (SVA) used to store rows of - * matrix R and columns of matrix S */ - /*--------------------------------------------------------------*/ - /* nn*n0-matrix R in row-wise format */ - int rr_ref; - /* reference number of sparse vector in SVA, which is the first - * row of matrix R */ -#if 0 + 0 - int *rr_ptr = &sva->ptr[rr_ref-1]; - /* rr_ptr[0] is not used; - * rr_ptr[i], 1 <= i <= nn, is pointer to i-th row in SVA; - * rr_ptr[nn+1,...,nn_max] are reserved locations */ - int *rr_len = &sva->len[rr_ref-1]; - /* rr_len[0] is not used; - * rr_len[i], 1 <= i <= nn, is length of i-th row; - * rr_len[nn+1,...,nn_max] are reserved locations */ -#endif - /*--------------------------------------------------------------*/ - /* n0*nn-matrix S in column-wise format */ - int ss_ref; - /* reference number of sparse vector in SVA, which is the first - * column of matrix S */ -#if 0 + 0 - int *ss_ptr = &sva->ptr[ss_ref-1]; - /* ss_ptr[0] is not used; - * ss_ptr[j], 1 <= j <= nn, is pointer to j-th column in SVA; - * ss_ptr[nn+1,...,nn_max] are reserved locations */ - int *ss_len = &sva->len[ss_ref-1]; - /* ss_len[0] is not used; - * ss_len[j], 1 <= j <= nn, is length of j-th column; - * ss_len[nn+1,...,nn_max] are reserved locations */ -#endif - /*--------------------------------------------------------------*/ - /* Schur complement C of order nn in invertable form */ - IFU ifu; - /* IFU-factorization of matrix C */ - /*--------------------------------------------------------------*/ - /* permutation matrix P of order n0+nn */ - int *pp_ind; /* int pp_ind[1+n0+nn_max]; */ - /* pp_ind[i] = j means that P[i,j] = 1 */ - int *pp_inv; /* int pp_inv[1+n0+nn_max]; */ - /* pp_inv[j] = i means that P[i,j] = 1 */ - /*--------------------------------------------------------------*/ - /* permutation matrix Q of order n0+nn */ - int *qq_ind; /* int qq_ind[1+n0+nn_max]; */ - /* qq_ind[i] = j means that Q[i,j] = 1 */ - int *qq_inv; /* int qq_inv[1+n0+nn_max]; */ - /* qq_inv[j] = i means that Q[i,j] = 1 */ -}; - -#define scf_swap_q_cols(j1, j2) \ - do \ - { int i1, i2; \ - i1 = qq_inv[j1], i2 = qq_inv[j2]; \ - qq_ind[i1] = j2, qq_inv[j2] = i1; \ - qq_ind[i2] = j1, qq_inv[j1] = i2; \ - } while (0) -/* swap columns j1 and j2 of permutation matrix Q */ - -#define scf_r0_solve _glp_scf_r0_solve -void scf_r0_solve(SCF *scf, int tr, double x[/*1+n0*/]); -/* solve system R0 * x = b or R0'* x = b */ - -#define scf_s0_solve _glp_scf_s0_solve -void scf_s0_solve(SCF *scf, int tr, double x[/*1+n0*/], - double w1[/*1+n0*/], double w2[/*1+n0*/], double w3[/*1+n0*/]); -/* solve system S0 * x = b or S0'* x = b */ - -#define scf_r_prod _glp_scf_r_prod -void scf_r_prod(SCF *scf, double y[/*1+nn*/], double a, const double - x[/*1+n0*/]); -/* compute product y := y + alpha * R * x */ - -#define scf_rt_prod _glp_scf_rt_prod -void scf_rt_prod(SCF *scf, double y[/*1+n0*/], double a, const double - x[/*1+nn*/]); -/* compute product y := y + alpha * R'* x */ - -#define scf_s_prod _glp_scf_s_prod -void scf_s_prod(SCF *scf, double y[/*1+n0*/], double a, const double - x[/*1+nn*/]); -/* compute product y := y + alpha * S * x */ - -#define scf_st_prod _glp_scf_st_prod -void scf_st_prod(SCF *scf, double y[/*1+nn*/], double a, const double - x[/*1+n0*/]); -/* compute product y := y + alpha * S'* x */ - -#define scf_a_solve _glp_scf_a_solve -void scf_a_solve(SCF *scf, double x[/*1+n*/], - double w[/*1+n0+nn*/], double work1[/*1+max(n0,nn)*/], - double work2[/*1+n*/], double work3[/*1+n*/]); -/* solve system A * x = b */ - -#define scf_at_solve _glp_scf_at_solve -void scf_at_solve(SCF *scf, double x[/*1+n*/], - double w[/*1+n0+nn*/], double work1[/*1+max(n0,nn)*/], - double work2[/*1+n*/], double work3[/*1+n*/]); -/* solve system A'* x = b */ - -#define scf_add_r_row _glp_scf_add_r_row -void scf_add_r_row(SCF *scf, const double w[/*1+n0*/]); -/* add new row to matrix R */ - -#define scf_add_s_col _glp_scf_add_s_col -void scf_add_s_col(SCF *scf, const double v[/*1+n0*/]); -/* add new column to matrix S */ - -#define scf_update_aug _glp_scf_update_aug -int scf_update_aug(SCF *scf, double b[/*1+n0*/], double d[/*1+n0*/], - double f[/*1+nn*/], double g[/*1+nn*/], double h, int upd, - double w1[/*1+n0*/], double w2[/*1+n0*/], double w3[/*1+n0*/]); -/* update factorization of augmented matrix */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/scfint.c b/code/3rd_glpk/bflib/scfint.c deleted file mode 100644 index 06aa8f7d..00000000 --- a/code/3rd_glpk/bflib/scfint.c +++ /dev/null @@ -1,255 +0,0 @@ -/* scfint.c (interface to Schur-complement-based factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "scfint.h" - -SCFINT *scfint_create(int type) -{ /* create interface to SC-factorization */ - SCFINT *fi; - fi = talloc(1, SCFINT); - memset(fi, 0, sizeof(SCFINT)); - switch ((fi->scf.type = type)) - { case 1: - fi->u.lufi = lufint_create(); - break; - case 2: - fi->u.btfi = btfint_create(); - break; - default: - xassert(type != type); - } - return fi; -} - -int scfint_factorize(SCFINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info) -{ /* compute SC-factorization of specified matrix A */ - int nn_max, old_n0_max, n0_max, k, ret; - xassert(n > 0); - fi->valid = 0; - /* get required value of nn_max */ - nn_max = fi->nn_max; - if (nn_max == 0) - nn_max = 100; - xassert(nn_max > 0); - /* compute factorization of specified matrix A */ - switch (fi->scf.type) - { case 1: - old_n0_max = fi->u.lufi->n_max; - fi->u.lufi->sva_n_max = 4 * n + 2 * nn_max; - ret = lufint_factorize(fi->u.lufi, n, col, info); - n0_max = fi->u.lufi->n_max; - fi->scf.sva = fi->u.lufi->sva; - fi->scf.a0.luf = fi->u.lufi->luf; - break; - case 2: - old_n0_max = fi->u.btfi->n_max; - fi->u.btfi->sva_n_max = 6 * n + 2 * nn_max; - ret = btfint_factorize(fi->u.btfi, n, col, info); - n0_max = fi->u.btfi->n_max; - fi->scf.sva = fi->u.btfi->sva; - fi->scf.a0.btf = fi->u.btfi->btf; - break; - default: - xassert(fi != fi); - } - /* allocate/reallocate arrays, if necessary */ - if (old_n0_max < n0_max) - { if (fi->w1 != NULL) - tfree(fi->w1); - if (fi->w2 != NULL) - tfree(fi->w2); - if (fi->w3 != NULL) - tfree(fi->w3); - fi->w1 = talloc(1+n0_max, double); - fi->w2 = talloc(1+n0_max, double); - fi->w3 = talloc(1+n0_max, double); - } - if (fi->scf.nn_max != nn_max) - { if (fi->scf.ifu.f != NULL) - tfree(fi->scf.ifu.f); - if (fi->scf.ifu.u != NULL) - tfree(fi->scf.ifu.u); - fi->scf.ifu.f = talloc(nn_max * nn_max, double); - fi->scf.ifu.u = talloc(nn_max * nn_max, double); - } - if (old_n0_max < n0_max || fi->scf.nn_max != nn_max) - { if (fi->scf.pp_ind != NULL) - tfree(fi->scf.pp_ind); - if (fi->scf.pp_inv != NULL) - tfree(fi->scf.pp_inv); - if (fi->scf.qq_ind != NULL) - tfree(fi->scf.qq_ind); - if (fi->scf.qq_inv != NULL) - tfree(fi->scf.qq_inv); - if (fi->w4 != NULL) - tfree(fi->w4); - if (fi->w5 != NULL) - tfree(fi->w5); - fi->scf.pp_ind = talloc(1+n0_max+nn_max, int); - fi->scf.pp_inv = talloc(1+n0_max+nn_max, int); - fi->scf.qq_ind = talloc(1+n0_max+nn_max, int); - fi->scf.qq_inv = talloc(1+n0_max+nn_max, int); - fi->w4 = talloc(1+n0_max+nn_max, double); - fi->w5 = talloc(1+n0_max+nn_max, double); - } - /* initialize SC-factorization */ - fi->scf.n = n; - fi->scf.n0 = n; - fi->scf.nn_max = nn_max; - fi->scf.nn = 0; - fi->scf.rr_ref = sva_alloc_vecs(fi->scf.sva, nn_max); - fi->scf.ss_ref = sva_alloc_vecs(fi->scf.sva, nn_max); - fi->scf.ifu.n_max = nn_max; - fi->scf.ifu.n = 0; - for (k = 1; k <= n; k++) - { fi->scf.pp_ind[k] = k; - fi->scf.pp_inv[k] = k; - fi->scf.qq_ind[k] = k; - fi->scf.qq_inv[k] = k; - } - /* set validation flag */ - if (ret == 0) - fi->valid = 1; - return ret; -} - -int scfint_update(SCFINT *fi, int upd, int j, int len, const int ind[], - const double val[]) -{ /* update SC-factorization after replacing j-th column of A */ - int n = fi->scf.n; - int n0 = fi->scf.n0; - int nn = fi->scf.nn; - int *pp_ind = fi->scf.pp_ind; - int *qq_ind = fi->scf.qq_ind; - int *qq_inv = fi->scf.qq_inv; - double *bf = fi->w4; - double *dg = fi->w5; - int k, t, ret; - xassert(fi->valid); - xassert(0 <= n && n <= n0+nn); - /* (b, f) := inv(P) * (beta, 0) */ - for (k = 1; k <= n0+nn; k++) - bf[k] = 0.0; - for (t = 1; t <= len; t++) - { k = ind[t]; - xassert(1 <= k && k <= n); -#if 1 /* FIXME: currently P = I */ - xassert(pp_ind[k] == k); -#endif - xassert(bf[k] == 0.0); - xassert(val[t] != 0.0); - bf[k] = val[t]; - } - /* (d, g) := Q * (cj, 0) */ - for (k = 1; k <= n0+nn; k++) - dg[k] = 0.0; - xassert(1 <= j && j <= n); - dg[fi->scf.qq_inv[j]] = 1; - /* update factorization of augmented matrix */ - ret = scf_update_aug(&fi->scf, &bf[0], &dg[0], &bf[n0], &dg[n0], - 0.0, upd, fi->w1, fi->w2, fi->w3); - if (ret == 0) - { /* swap j-th and last columns of new matrix Q */ - scf_swap_q_cols(j, n0+nn+1); - } - else - { /* updating failed */ - fi->valid = 0; - } - return ret; -} - -void scfint_ftran(SCFINT *fi, double x[]) -{ /* solve system A * x = b */ - xassert(fi->valid); - scf_a_solve(&fi->scf, x, fi->w4, fi->w5, fi->w1, fi->w2); - return; -} - -void scfint_btran(SCFINT *fi, double x[]) -{ /* solve system A'* x = b */ - xassert(fi->valid); - scf_at_solve(&fi->scf, x, fi->w4, fi->w5, fi->w1, fi->w2); - return; -} - -double scfint_estimate(SCFINT *fi) -{ /* estimate 1-norm of inv(A) */ - double norm; - xassert(fi->valid); - xassert(fi->scf.n == fi->scf.n0); - switch (fi->scf.type) - { case 1: - norm = luf_estimate_norm(fi->scf.a0.luf, fi->w1, fi->w2); - break; - case 2: - norm = btf_estimate_norm(fi->scf.a0.btf, fi->w1, fi->w2, - fi->w3, fi->w4); - break; - default: - xassert(fi != fi); - } - return norm; -} - -void scfint_delete(SCFINT *fi) -{ /* delete interface to SC-factorization */ - switch (fi->scf.type) - { case 1: - lufint_delete(fi->u.lufi); - break; - case 2: - btfint_delete(fi->u.btfi); - break; - default: - xassert(fi != fi); - } - if (fi->scf.ifu.f != NULL) - tfree(fi->scf.ifu.f); - if (fi->scf.ifu.u != NULL) - tfree(fi->scf.ifu.u); - if (fi->scf.pp_ind != NULL) - tfree(fi->scf.pp_ind); - if (fi->scf.pp_inv != NULL) - tfree(fi->scf.pp_inv); - if (fi->scf.qq_ind != NULL) - tfree(fi->scf.qq_ind); - if (fi->scf.qq_inv != NULL) - tfree(fi->scf.qq_inv); - if (fi->w1 != NULL) - tfree(fi->w1); - if (fi->w2 != NULL) - tfree(fi->w2); - if (fi->w3 != NULL) - tfree(fi->w3); - if (fi->w4 != NULL) - tfree(fi->w4); - if (fi->w5 != NULL) - tfree(fi->w5); - tfree(fi); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/scfint.h b/code/3rd_glpk/bflib/scfint.h deleted file mode 100644 index 3e56355b..00000000 --- a/code/3rd_glpk/bflib/scfint.h +++ /dev/null @@ -1,89 +0,0 @@ -/* scfint.h (interface to Schur-complement-based factorization) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013-2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SCFINT_H -#define SCFINT_H - -#include "scf.h" -#include "lufint.h" -#include "btfint.h" - -typedef struct SCFINT SCFINT; - -struct SCFINT -{ /* interface to SC-factorization */ - int valid; - /* factorization is valid only if this flag is set */ - SCF scf; - /* Schur-complement based factorization */ - union - { LUFINT *lufi; /* scf.type = 1 */ - BTFINT *btfi; /* scf.type = 2 */ - } u; - /* interface to factorize initial matrix A0 */ - /*--------------------------------------------------------------*/ - /* working arrays */ - double *w1; /* double w1[1+n0_max]; */ - double *w2; /* double w2[1+n0_max]; */ - double *w3; /* double w3[1+n0_max]; */ - double *w4; /* double w4[1+n0_max+nn_max]; */ - double *w5; /* double w5[1+n0_max+nn_max]; */ - /*--------------------------------------------------------------*/ - /* control parameters */ - int nn_max; - /* required maximal number of updates */ -}; - -#define scfint_create _glp_scfint_create -SCFINT *scfint_create(int type); -/* create interface to SC-factorization */ - -#define scfint_factorize _glp_scfint_factorize -int scfint_factorize(SCFINT *fi, int n, int (*col)(void *info, int j, - int ind[], double val[]), void *info); -/* compute SC-factorization of specified matrix A */ - -#define scfint_update _glp_scfint_update -int scfint_update(SCFINT *fi, int upd, int j, int len, const int ind[], - const double val[]); -/* update SC-factorization after replacing j-th column of A */ - -#define scfint_ftran _glp_scfint_ftran -void scfint_ftran(SCFINT *fi, double x[]); -/* solve system A * x = b */ - -#define scfint_btran _glp_scfint_btran -void scfint_btran(SCFINT *fi, double x[]); -/* solve system A'* x = b */ - -#define scfint_estimate _glp_scfint_estimate -double scfint_estimate(SCFINT *fi); -/* estimate 1-norm of inv(A) */ - -#define scfint_delete _glp_scfint_delete -void scfint_delete(SCFINT *fi); -/* delete interface to SC-factorization */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/sgf.c b/code/3rd_glpk/bflib/sgf.c deleted file mode 100644 index 1c1f49a6..00000000 --- a/code/3rd_glpk/bflib/sgf.c +++ /dev/null @@ -1,1443 +0,0 @@ -/* sgf.c (sparse Gaussian factorizer) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "sgf.h" - -/*********************************************************************** -* sgf_reduce_nuc - initial reordering to minimize nucleus size -* -* On entry to this routine it is assumed that V = A and F = P = Q = I, -* where A is the original matrix to be factorized. It is also assumed -* that matrix V = A is stored in both row- and column-wise formats. -* -* This routine performs (implicit) non-symmetric permutations of rows -* and columns of matrix U = P'* V * Q' to reduce it to the form: -* -* 1 k1 k2 n -* 1 x x x x x x x x x x -* . x x x x x x x x x -* . . x x x x x x x x -* k1 . . . * * * * x x x -* . . . * * * * x x x -* . . . * * * * x x x -* k2 . . . * * * * x x x -* . . . . . . . x x x -* . . . . . . . . x x -* n . . . . . . . . . x -* -* where non-zeros in rows and columns k1, k1+1, ..., k2 constitute so -* called nucleus ('*'), whose size is minimized by the routine. -* -* The numbers k1 and k2 are returned by the routine on exit. Usually, -* if the nucleus exists, 1 <= k1 < k2 <= n. However, if the resultant -* matrix U is upper triangular (has no nucleus), k1 = n+1 and k2 = n. -* -* Note that the routines sgf_choose_pivot and sgf_eliminate perform -* exactly the same transformations (by processing row and columns -* singletons), so preliminary minimization of the nucleus may not be -* used. However, processing row and column singletons by the routines -* sgf_minimize_nuc and sgf_singl_phase is more efficient. */ - -#if 1 /* 21/II-2016 */ -/* Normally this routine returns zero. If the matrix is structurally -* singular, the routine returns non-zero. */ -#endif - -int sgf_reduce_nuc(LUF *luf, int *k1_, int *k2_, int cnt[/*1+n*/], - int list[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int *qq_inv = luf->qq_inv; - int i, ii, j, jj, k1, k2, ns, ptr, end; - /* initial nucleus is U = V = A */ - k1 = 1, k2 = n; - /*--------------------------------------------------------------*/ - /* process column singletons */ - /*--------------------------------------------------------------*/ - /* determine initial counts of columns of V and initialize list - * of active column singletons */ - ns = 0; /* number of active column singletons */ - for (j = 1; j <= n; j++) - { if ((cnt[j] = vc_len[j]) == 1) - list[++ns] = j; - } - /* process active column singletons */ - while (ns > 0) - { /* column singleton is in j-th column of V */ - j = list[ns--]; -#if 1 /* 21/II-2016 */ - if (cnt[j] == 0) - { /* j-th column in the current nucleus is actually empty */ - /* this happened because on a previous step in the nucleus - * there were two or more identical column singletons (that - * means structural singularity), so removing one of them - * from the nucleus made other columns empty */ - return 1; - } -#endif - /* find i-th row of V containing column singleton */ - ptr = vc_ptr[j]; - end = ptr + vc_len[j]; - for (; pp_ind[i = sv_ind[ptr]] < k1; ptr++) - /* nop */; - xassert(ptr < end); - /* permute rows and columns of U to move column singleton to - * position u[k1,k1] */ - ii = pp_ind[i]; - luf_swap_u_rows(k1, ii); - jj = qq_inv[j]; - luf_swap_u_cols(k1, jj); - /* nucleus size decreased */ - k1++; - /* walk thru i-th row of V and decrease column counts; this - * may cause new column singletons to appear */ - ptr = vr_ptr[i]; - end = ptr + vr_len[i]; - for (; ptr < end; ptr++) - { if (--(cnt[j = sv_ind[ptr]]) == 1) - list[++ns] = j; - } - } - /* nucleus begins at k1-th row/column of U */ - if (k1 > n) - { /* U is upper triangular; no nucleus exist */ - goto done; - } - /*--------------------------------------------------------------*/ - /* process row singletons */ - /*--------------------------------------------------------------*/ - /* determine initial counts of rows of V and initialize list of - * active row singletons */ - ns = 0; /* number of active row singletons */ - for (i = 1; i <= n; i++) - { if (pp_ind[i] < k1) - { /* corresponding row of U is above its k1-th row; set its - * count to zero to prevent including it in active list */ - cnt[i] = 0; - } - else if ((cnt[i] = vr_len[i]) == 1) - list[++ns] = i; - } - /* process active row singletons */ - while (ns > 0) - { /* row singleton is in i-th row of V */ - i = list[ns--]; -#if 1 /* 21/II-2016 */ - if (cnt[i] == 0) - { /* i-th row in the current nucleus is actually empty */ - /* (see comments above for similar case of empty column) */ - return 2; - } -#endif - /* find j-th column of V containing row singleton */ - ptr = vr_ptr[i]; - end = ptr + vr_len[i]; - for (; qq_inv[j = sv_ind[ptr]] > k2; ptr++) - /* nop */; - xassert(ptr < end); - /* permute rows and columns of U to move row singleton to - * position u[k2,k2] */ - ii = pp_ind[i]; - luf_swap_u_rows(k2, ii); - jj = qq_inv[j]; - luf_swap_u_cols(k2, jj); - /* nucleus size decreased */ - k2--; - /* walk thru j-th column of V and decrease row counts; this - * may cause new row singletons to appear */ - ptr = vc_ptr[j]; - end = ptr + vc_len[j]; - for (; ptr < end; ptr++) - { if (--(cnt[i = sv_ind[ptr]]) == 1) - list[++ns] = i; - } - } - /* nucleus ends at k2-th row/column of U */ - xassert(k1 < k2); -done: *k1_ = k1, *k2_ = k2; - return 0; -} - -/*********************************************************************** -* sgf_singl_phase - compute LU-factorization (singleton phase) -* -* It is assumed that on entry to the routine L = P'* F * P = F = I -* and matrix U = P'* V * Q' has the following structure (provided by -* the routine sgf_reduce_nuc): -* -* 1 k1 k2 n -* 1 a a a b b b b c c c -* . a a b b b b c c c -* . . a b b b b c c c -* k1 . . . * * * * d d d -* . . . * * * * d d d -* . . . * * * * d d d -* k2 . . . * * * * d d d -* . . . . . . . e e e -* . . . . . . . . e e -* n . . . . . . . . . e -* -* First, the routine performs (implicit) symmetric permutations of -* rows and columns of matrix U to place them in the following order: -* -* 1, 2, ..., k1-1; n, n-1, ..., k2+1; k1, k1+1, ..., k2 -* -* This changes the structure of matrix U as follows: -* -* 1 k1 k2' n -* 1 a a a c c c b b b b -* . a a c c c b b b b -* . . a c c c b b b b -* k1 . . . e . . . . . . -* . . . e e . . . . . -* . . . e e e . . . . -* k2'. . . d d d * * * * -* . . . d d d * * * * -* . . . d d d * * * * -* n . . . d d d * * * * -* -* where k2' = n - k2 + k1. -* -* Then the routine performs elementary gaussian transformations to -* eliminate subdiagonal elements in columns k1, ..., k2'-1 of U. The -* effect is the same as if the routine sgf_eliminate would be called -* for k = 1, ..., k2'-1 using diagonal elements u[k,k] as pivots. -* -* After elimination matrices L and U becomes the following: -* -* 1 k1 k2' n 1 k1 k2' n -* 1 1 . . . . . . . . . 1 a a a c c c b b b b -* . 1 . . . . . . . . . a a c c c b b b b -* . . 1 . . . . . . . . . a c c c b b b b -* k1 . . . 1 . . . . . . k1 . . . e . . . . . . -* . . . e'1 . . . . . . . . . e . . . . . -* . . . e'e'1 . . . . . . . . . e . . . . -* k2'. . . d'd'd'1 . . . k2'. . . . . . * * * * -* . . . d'd'd'. 1 . . . . . . . . * * * * -* . . . d'd'd'. . 1 . . . . . . . * * * * -* n . . . d'd'd'. . . 1 n . . . . . . * * * * -* -* matrix L matrix U -* -* where columns k1, ..., k2'-1 of L consist of subdiagonal elements -* of initial matrix U divided by pivots u[k,k]. -* -* On exit the routine returns k2', the elimination step number, from -* which computing of the factorization should be continued. Note that -* k2' = n+1 means that matrix U is already upper triangular. */ - -int sgf_singl_phase(LUF *luf, int k1, int k2, int updat, - int ind[/*1+n*/], double val[/*1+n*/]) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - double *vr_piv = luf->vr_piv; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int *qq_inv = luf->qq_inv; - int i, j, k, ptr, ptr1, end, len; - double piv; - /* (see routine sgf_reduce_nuc) */ - xassert((1 <= k1 && k1 < k2 && k2 <= n) - || (k1 == n+1 && k2 == n)); - /* perform symmetric permutations of rows/columns of U */ - for (k = k1; k <= k2; k++) - pp_ind[pp_inv[k]] = qq_inv[qq_ind[k]] = k - k2 + n; - for (k = k2+1; k <= n; k++) - pp_ind[pp_inv[k]] = qq_inv[qq_ind[k]] = n - k + k1; - for (k = 1; k <= n; k++) - pp_inv[pp_ind[k]] = qq_ind[qq_inv[k]] = k; - /* determine k2' */ - k2 = n - k2 + k1; - /* process rows and columns of V corresponding to rows and - * columns 1, ..., k1-1 of U */ - for (k = 1; k < k1; k++) - { /* k-th row of U = i-th row of V */ - i = pp_inv[k]; - /* find pivot u[k,k] = v[i,j] in i-th row of V */ - ptr = vr_ptr[i]; - end = ptr + vr_len[i]; - for (; qq_inv[sv_ind[ptr]] != k; ptr++) - /* nop */; - xassert(ptr < end); - /* store pivot */ - vr_piv[i] = sv_val[ptr]; - /* and remove it from i-th row of V */ - sv_ind[ptr] = sv_ind[end-1]; - sv_val[ptr] = sv_val[end-1]; - vr_len[i]--; - /* clear column of V corresponding to k-th column of U */ - vc_len[qq_ind[k]] = 0; - } - /* clear rows of V corresponding to rows k1, ..., k2'-1 of U */ - for (k = k1; k < k2; k++) - vr_len[pp_inv[k]] = 0; - /* process rows and columns of V corresponding to rows and - * columns k2', ..., n of U */ - for (k = k2; k <= n; k++) - { /* k-th row of U = i-th row of V */ - i = pp_inv[k]; - /* remove elements from i-th row of V that correspond to - * elements u[k,k1], ..., u[k,k2'-1] */ - ptr = ptr1 = vr_ptr[i]; - end = ptr + vr_len[i]; - for (; ptr < end; ptr++) - { if (qq_inv[sv_ind[ptr]] >= k2) - { sv_ind[ptr1] = sv_ind[ptr]; - sv_val[ptr1] = sv_val[ptr]; - ptr1++; - } - } - vr_len[i] = ptr1 - vr_ptr[i]; - /* k-th column of U = j-th column of V */ - j = qq_ind[k]; - /* remove elements from j-th column of V that correspond to - * elements u[1,k], ..., u[k1-1,k] */ - ptr = ptr1 = vc_ptr[j]; - end = ptr + vc_len[j]; - for (; ptr < end; ptr++) - { if (pp_ind[sv_ind[ptr]] >= k2) - /* element value is not needed in this case */ - sv_ind[ptr1++] = sv_ind[ptr]; - } - vc_len[j] = ptr1 - vc_ptr[j]; - } - /* process columns of V corresponding to columns k1, ..., k2'-1 - * of U, build columns of F */ - for (k = k1; k < k2; k++) - { /* k-th column of U = j-th column of V */ - j = qq_ind[k]; - /* remove elements from j-th column of V that correspond to - * pivot (diagonal) element u[k,k] and subdiagonal elements - * u[k+1,k], ..., u[n,k]; subdiagonal elements are stored for - * further addition to matrix F */ - len = 0; - piv = 0.0; - ptr = vc_ptr[j]; - end = ptr + vc_len[j]; - for (; ptr < end; ptr++) - { i = sv_ind[ptr]; /* v[i,j] */ - if (pp_ind[i] == k) - { /* store pivot v[i,j] = u[k,k] */ - piv = vr_piv[i] = sv_val[ptr]; - } - else if (pp_ind[i] > k) - { /* store subdiagonal element v[i,j] = u[i',k] */ - len++; - ind[len] = i; - val[len] = sv_val[ptr]; - } - } - /* clear j-th column of V = k-th column of U */ - vc_len[j] = 0; - /* build k-th column of L = j-th column of F */ - j = pp_inv[k]; - xassert(piv != 0.0); - if (len > 0) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, fc_ref-1+j, len); - for (ptr = fc_ptr[j], ptr1 = 1; ptr1 <= len; ptr++, ptr1++) - { sv_ind[ptr] = ind[ptr1]; - sv_val[ptr] = val[ptr1] / piv; - } - fc_len[j] = len; - } - } - /* if it is not planned to update matrix V, relocate all its - * non-active rows corresponding to rows 1, ..., k2'-1 of U to - * the right (static) part of SVA */ - if (!updat) - { for (k = 1; k < k2; k++) - { i = pp_inv[k]; - len = vr_len[i]; - if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_make_static(sva, vr_ref-1+i); - } - } - /* elimination steps 1, ..., k2'-1 have been performed */ - return k2; -} - -/*********************************************************************** -* sgf_choose_pivot - choose pivot element v[p,q] -* -* This routine chooses pivot element v[p,q], k <= p, q <= n, in the -* active submatrix of matrix V = P * U * Q, where k is the number of -* current elimination step, 1 <= k <= n. -* -* It is assumed that on entry to the routine matrix U = P'* V * Q' has -* the following partially triangularized form: -* -* 1 k n -* 1 x x x x x x x x x x -* . x x x x x x x x x -* . . x x x x x x x x -* . . . x x x x x x x -* k . . . . * * * * * * -* . . . . * * * * * * -* . . . . * * * * * * -* . . . . * * * * * * -* . . . . * * * * * * -* n . . . . * * * * * * -* -* where rows and columns k, k+1, ..., n belong to the active submatrix -* (its elements are marked by '*'). -* -* Since the matrix U is not stored, the routine works with the matrix -* V = P * U * Q. It is assumed that the row-wise representation -* corresponds to the matrix V, but the column-wise representation -* corresponds to the active submatrix of the matrix V, i.e. elements, -* which are not in the active submatrix, are not included in column -* vectors. It is also assumed that each active row of the matrix V is -* in the set R[len], where len is the number of non-zeros in the row, -* and each active column of the matrix V is in the set C[len], where -* len is the number of non-zeros in the column (in the latter case -* only elements of the active submatrix are counted; such elements are -* marked by '*' on the figure above). -* -* For the reason of numerical stability the routine applies so called -* threshold pivoting proposed by J.Reid. It is assumed that an element -* v[i,j] can be selected as a pivot candidate if it is not very small -* (in magnitude) among other elements in the same row, i.e. if it -* satisfies to the stability condition |v[i,j]| >= tol * max|v[i,*]|, -* where 0 < tol < 1 is a given tolerance. -* -* In order to keep sparsity of the matrix V the routine uses Markowitz -* strategy, trying to choose such element v[p,q], which satisfies to -* the stability condition (see above) and has smallest Markowitz cost -* (nr[p]-1) * (nc[q]-1), where nr[p] and nc[q] are, resp., numbers of -* non-zeros in p-th row and q-th column of the active submatrix. -* -* In order to reduce the search, i.e. not to walk through all elements -* of the active submatrix, the routine uses a technique proposed by -* I.Duff. This technique is based on using the sets R[len] and C[len] -* of active rows and columns. -* -* If the pivot element v[p,q] has been chosen, the routine stores its -* indices to locations *p and *q and returns zero. Otherwise, non-zero -* is returned. */ - -int sgf_choose_pivot(SGF *sgf, int *p_, int *q_) -{ LUF *luf = sgf->luf; - int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *rs_head = sgf->rs_head; - int *rs_next = sgf->rs_next; - int *cs_head = sgf->cs_head; - int *cs_prev = sgf->cs_prev; - int *cs_next = sgf->cs_next; - double *vr_max = sgf->vr_max; - double piv_tol = sgf->piv_tol; - int piv_lim = sgf->piv_lim; - int suhl = sgf->suhl; - int i, i_ptr, i_end, j, j_ptr, j_end, len, min_i, min_j, min_len, - ncand, next_j, p, q; - double best, big, cost, temp; - /* no pivot candidate has been chosen so far */ - p = q = 0, best = DBL_MAX, ncand = 0; - /* if the active submatrix contains a column having the only - * non-zero element (column singleton), choose it as the pivot */ - j = cs_head[1]; - if (j != 0) - { xassert(vc_len[j] == 1); - p = sv_ind[vc_ptr[j]], q = j; - goto done; - } - /* if the active submatrix contains a row having the only - * non-zero element (row singleton), choose it as the pivot */ - i = rs_head[1]; - if (i != 0) - { xassert(vr_len[i] == 1); - p = i, q = sv_ind[vr_ptr[i]]; - goto done; - } - /* the active submatrix contains no singletons; walk thru its - * other non-empty rows and columns */ - for (len = 2; len <= n; len++) - { /* consider active columns containing len non-zeros */ - for (j = cs_head[len]; j != 0; j = next_j) - { /* save the number of next column of the same length */ - next_j = cs_next[j]; - /* find an element in j-th column, which is placed in the - * row with minimal number of non-zeros and satisfies to - * the stability condition (such element may not exist) */ - min_i = min_j = 0, min_len = INT_MAX; - for (j_end = (j_ptr = vc_ptr[j]) + vc_len[j]; - j_ptr < j_end; j_ptr++) - { /* get row index of v[i,j] */ - i = sv_ind[j_ptr]; - /* if i-th row is not shorter, skip v[i,j] */ - if (vr_len[i] >= min_len) - continue; - /* big := max|v[i,*]| */ - if ((big = vr_max[i]) < 0.0) - { /* largest magnitude is unknown; compute it */ - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - i_ptr < i_end; i_ptr++) - { if ((temp = sv_val[i_ptr]) < 0.0) - temp = -temp; - if (big < temp) - big = temp; - } - xassert(big > 0.0); - vr_max[i] = big; - } - /* find v[i,j] in i-th row */ - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - sv_ind[i_ptr] != j; i_ptr++) - /* nop */; - xassert(i_ptr < i_end); - /* if |v[i,j]| < piv_tol * max|v[i,*]|, skip v[i,j] */ - if ((temp = sv_val[i_ptr]) < 0.0) - temp = -temp; - if (temp < piv_tol * big) - continue; - /* v[i,j] is a better candidate */ - min_i = i, min_j = j, min_len = vr_len[i]; - /* if Markowitz cost of v[i,j] is not greater than - * (len-1)**2, v[i,j] can be chosen as the pivot right - * now; this heuristic reduces the search and works well - * in many cases */ - if (min_len <= len) - { p = min_i, q = min_j; - goto done; - } - } - /* j-th column has been scanned */ - if (min_i != 0) - { /* element v[min_i,min_j] is a next pivot candidate */ - ncand++; - /* compute its Markowitz cost */ - cost = (double)(min_len - 1) * (double)(len - 1); - /* if this element is better, choose it as the pivot */ - if (cost < best) - p = min_i, q = min_j, best = cost; - /* if piv_lim candidates were considered, terminate - * the search, because it is doubtful that a much better - * candidate will be found */ - if (ncand == piv_lim) - goto done; - } - else if (suhl) - { /* j-th column has no eligible elements that satisfy to - * the stability criterion; Uwe Suhl suggests to exclude - * such column from further considerations until it - * becomes a column singleton; in hard cases this may - * significantly reduce the time needed to choose the - * pivot element */ - sgf_deactivate_col(j); - cs_prev[j] = cs_next[j] = j; - } - } - /* consider active rows containing len non-zeros */ - for (i = rs_head[len]; i != 0; i = rs_next[i]) - { /* big := max|v[i,*]| */ - if ((big = vr_max[i]) < 0.0) - { /* largest magnitude is unknown; compute it */ - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - i_ptr < i_end; i_ptr++) - { if ((temp = sv_val[i_ptr]) < 0.0) - temp = -temp; - if (big < temp) - big = temp; - } - xassert(big > 0.0); - vr_max[i] = big; - } - /* find an element in i-th row, which is placed in the - * column with minimal number of non-zeros and satisfies to - * the stability condition (such element always exists) */ - min_i = min_j = 0, min_len = INT_MAX; - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - i_ptr < i_end; i_ptr++) - { /* get column index of v[i,j] */ - j = sv_ind[i_ptr]; - /* if j-th column is not shorter, skip v[i,j] */ - if (vc_len[j] >= min_len) - continue; - /* if |v[i,j]| < piv_tol * max|v[i,*]|, skip v[i,j] */ - if ((temp = sv_val[i_ptr]) < 0.0) - temp = -temp; - if (temp < piv_tol * big) - continue; - /* v[i,j] is a better candidate */ - min_i = i, min_j = j, min_len = vc_len[j]; - /* if Markowitz cost of v[i,j] is not greater than - * (len-1)**2, v[i,j] can be chosen as the pivot right - * now; this heuristic reduces the search and works well - * in many cases */ - if (min_len <= len) - { p = min_i, q = min_j; - goto done; - } - } - /* i-th row has been scanned */ - if (min_i != 0) - { /* element v[min_i,min_j] is a next pivot candidate */ - ncand++; - /* compute its Markowitz cost */ - cost = (double)(len - 1) * (double)(min_len - 1); - /* if this element is better, choose it as the pivot */ - if (cost < best) - p = min_i, q = min_j, best = cost; - /* if piv_lim candidates were considered, terminate - * the search, because it is doubtful that a much better - * candidate will be found */ - if (ncand == piv_lim) - goto done; - } - else - { /* this can never be */ - xassert(min_i != min_i); - } - } - } -done: /* report the pivot to the factorization routine */ - *p_ = p, *q_ = q; - return (p == 0); -} - -/*********************************************************************** -* sgf_eliminate - perform gaussian elimination -* -* This routine performs elementary gaussian transformations in order -* to eliminate subdiagonal elements in k-th column of matrix -* U = P'* V * Q' using pivot element u[k,k], where k is the number of -* current elimination step, 1 <= k <= n. -* -* The parameters p and q specify, resp., row and column indices of the -* pivot element v[p,q] = u[k,k]. -* -* On entry the routine assumes that partially triangularized matrices -* L = P'* F * P and U = P'* V * Q' have the following structure: -* -* 1 k n 1 k n -* 1 1 . . . . . . . . . 1 x x x x x x x x x x -* x 1 . . . . . . . . . x x x x x x x x x -* x x 1 . . . . . . . . . x x x x x x x x -* x x x 1 . . . . . . . . . x x x x x x x -* k x x x x 1 . . . . . k . . . . * * * * * * -* x x x x _ 1 . . . . . . . . # * * * * * -* x x x x _ . 1 . . . . . . . # * * * * * -* x x x x _ . . 1 . . . . . . # * * * * * -* x x x x _ . . . 1 . . . . . # * * * * * -* n x x x x _ . . . . 1 n . . . . # * * * * * -* -* matrix L matrix U -* -* where rows and columns k, k+1, ..., n of matrix U constitute the -* active submatrix. Elements to be eliminated are marked by '#', and -* other elements of the active submatrix are marked by '*'. May note -* that each eliminated non-zero element u[i,k] of matrix U gives -* corresponding non-zero element l[i,k] of matrix L (marked by '_'). -* -* Actually all operations are performed on matrix V. It is assumed -* that the row-wise representation corresponds to matrix V, but the -* column-wise representation corresponds to the active submatrix of -* matrix V (or, more precisely, to its pattern, because only row -* indices for columns of the active submatrix are used on this stage). -* -* Let u[k,k] = v[p,q] be the pivot. In order to eliminate subdiagonal -* elements u[i',k] = v[i,q], i'= k+1, k+2, ..., n, the routine applies -* the following elementary gaussian transformations: -* -* (i-th row of V) := (i-th row of V) - f[i,p] * (p-th row of V), -* -* where f[i,p] = v[i,q] / v[p,q] is a gaussian multiplier stored to -* p-th column of matrix F to keep the main equality A = F * V -* (corresponding elements l[i',k] of matrix L are marked by '_' on the -* figure above). -* -* NOTE: On entry to the routine the working arrays flag and work -* should contain zeros. This status is retained by the routine -* on exit. */ - -int sgf_eliminate(SGF *sgf, int p, int q) -{ LUF *luf = sgf->luf; - int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int *vr_cap = &sva->cap[vr_ref-1]; - double *vr_piv = luf->vr_piv; - int vc_ref = luf->vc_ref; - int *vc_ptr = &sva->ptr[vc_ref-1]; - int *vc_len = &sva->len[vc_ref-1]; - int *vc_cap = &sva->cap[vc_ref-1]; - int *rs_head = sgf->rs_head; - int *rs_prev = sgf->rs_prev; - int *rs_next = sgf->rs_next; - int *cs_head = sgf->cs_head; - int *cs_prev = sgf->cs_prev; - int *cs_next = sgf->cs_next; - double *vr_max = sgf->vr_max; - char *flag = sgf->flag; - double *work = sgf->work; - double eps_tol = sgf->eps_tol; - int nnz_diff = 0; - int fill, i, i_ptr, i_end, j, j_ptr, j_end, ptr, len, loc, loc1; - double vpq, fip, vij; - xassert(1 <= p && p <= n); - xassert(1 <= q && q <= n); - /* remove p-th row from the active set; this row will never - * return there */ - sgf_deactivate_row(p); - /* process p-th (pivot) row */ - ptr = 0; - for (i_end = (i_ptr = vr_ptr[p]) + vr_len[p]; - i_ptr < i_end; i_ptr++) - { /* get column index of v[p,j] */ - j = sv_ind[i_ptr]; - if (j == q) - { /* save pointer to pivot v[p,q] */ - ptr = i_ptr; - } - else - { /* store v[p,j], j != q, to working array */ - flag[j] = 1; - work[j] = sv_val[i_ptr]; - } - /* remove j-th column from the active set; q-th column will - * never return there while other columns will return to the - * active set with new length */ - if (cs_next[j] == j) - { /* j-th column was marked by the pivoting routine according - * to Uwe Suhl's suggestion and is already inactive */ - xassert(cs_prev[j] == j); - } - else - sgf_deactivate_col(j); - nnz_diff -= vc_len[j]; - /* find and remove v[p,j] from j-th column */ - for (j_end = (j_ptr = vc_ptr[j]) + vc_len[j]; - sv_ind[j_ptr] != p; j_ptr++) - /* nop */; - xassert(j_ptr < j_end); - sv_ind[j_ptr] = sv_ind[j_end-1]; - vc_len[j]--; - } - /* save pivot v[p,q] and remove it from p-th row */ - xassert(ptr > 0); - vpq = vr_piv[p] = sv_val[ptr]; - sv_ind[ptr] = sv_ind[i_end-1]; - sv_val[ptr] = sv_val[i_end-1]; - vr_len[p]--; - /* if it is not planned to update matrix V, relocate p-th row to - * the right (static) part of SVA */ - if (!sgf->updat) - { len = vr_len[p]; - if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_make_static(sva, vr_ref-1+p); - } - /* copy the pattern (row indices) of q-th column of the active - * submatrix (from which v[p,q] has been just removed) to p-th - * column of matrix F (without unity diagonal element) */ - len = vc_len[q]; - if (len > 0) - { if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_reserve_cap(sva, fc_ref-1+p, len); - memcpy(&sv_ind[fc_ptr[p]], &sv_ind[vc_ptr[q]], - len * sizeof(int)); - fc_len[p] = len; - } - /* make q-th column of the active submatrix empty */ - vc_len[q] = 0; - /* transform non-pivot rows of the active submatrix */ - for (loc = fc_len[p]-1; loc >= 0; loc--) - { /* get row index of v[i,q] = row index of f[i,p] */ - i = sv_ind[fc_ptr[p] + loc]; - xassert(i != p); /* v[p,q] was removed */ - /* remove i-th row from the active set; this row will return - * there with new length */ - sgf_deactivate_row(i); - /* find v[i,q] in i-th row */ - for (i_end = (i_ptr = vr_ptr[i]) + vr_len[i]; - sv_ind[i_ptr] != q; i_ptr++) - /* nop */; - xassert(i_ptr < i_end); - /* compute gaussian multiplier f[i,p] = v[i,q] / v[p,q] */ - fip = sv_val[fc_ptr[p] + loc] = sv_val[i_ptr] / vpq; - /* remove v[i,q] from i-th row */ - sv_ind[i_ptr] = sv_ind[i_end-1]; - sv_val[i_ptr] = sv_val[i_end-1]; - vr_len[i]--; - /* perform elementary gaussian transformation: - * (i-th row) := (i-th row) - f[i,p] * (p-th row) - * note that p-th row of V, which is in the working array, - * doesn't contain pivot v[p,q], and i-th row of V doesn't - * contain v[i,q] to be eliminated */ - /* walk thru i-th row and transform existing elements */ - fill = vr_len[p]; - for (i_end = (i_ptr = ptr = vr_ptr[i]) + vr_len[i]; - i_ptr < i_end; i_ptr++) - { /* get column index and value of v[i,j] */ - j = sv_ind[i_ptr]; - vij = sv_val[i_ptr]; - if (flag[j]) - { /* v[p,j] != 0 */ - flag[j] = 0, fill--; - /* v[i,j] := v[i,j] - f[i,p] * v[p,j] */ - vij -= fip * work[j]; - if (-eps_tol < vij && vij < +eps_tol) - { /* new v[i,j] is close to zero; remove it from the - * active submatrix, i.e. replace it by exact zero */ - /* find and remove v[i,j] from j-th column */ - for (j_end = (j_ptr = vc_ptr[j]) + vc_len[j]; - sv_ind[j_ptr] != i; j_ptr++) - /* nop */; - xassert(j_ptr < j_end); - sv_ind[j_ptr] = sv_ind[j_end-1]; - vc_len[j]--; - continue; - } - } - /* keep new v[i,j] in i-th row */ - sv_ind[ptr] = j; - sv_val[ptr] = vij; - ptr++; - } - /* (new length of i-th row may decrease because of numerical - * cancellation) */ - vr_len[i] = len = ptr - vr_ptr[i]; - /* now flag[*] is the pattern of the set v[p,*] \ v[i,*], and - * fill is the number of non-zeros in this set */ - if (fill == 0) - { /* no fill-in occurs */ - /* walk thru p-th row and restore the column flags */ - for (i_end = (i_ptr = vr_ptr[p]) + vr_len[p]; - i_ptr < i_end; i_ptr++) - flag[sv_ind[i_ptr]] = 1; /* v[p,j] != 0 */ - goto skip; - } - /* up to fill new non-zero elements may appear in i-th row due - * to fill-in; reserve locations for these elements (note that - * actual length of i-th row is currently stored in len) */ - if (vr_cap[i] < len + fill) - { if (sva->r_ptr - sva->m_ptr < len + fill) - { sva_more_space(sva, len + fill); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vr_ref-1+i, len + fill, 0); - } - vr_len[i] += fill; - /* walk thru p-th row and add new elements to i-th row */ - for (loc1 = vr_len[p]-1; loc1 >= 0; loc1--) - { /* get column index of v[p,j] */ - j = sv_ind[vr_ptr[p] + loc1]; - if (!flag[j]) - { /* restore j-th column flag */ - flag[j] = 1; - /* v[i,j] was computed earlier on transforming existing - * elements of i-th row */ - continue; - } - /* v[i,j] := 0 - f[i,p] * v[p,j] */ - vij = - fip * work[j]; - if (-eps_tol < vij && vij < +eps_tol) - { /* new v[i,j] is close to zero; do not add it to the - * active submatrix, i.e. replace it by exact zero */ - continue; - } - /* add new v[i,j] to i-th row */ - sv_ind[ptr = vr_ptr[i] + (len++)] = j; - sv_val[ptr] = vij; - /* add new v[i,j] to j-th column */ - if (vc_cap[j] == vc_len[j]) - { /* we reserve extra locations in j-th column to reduce - * further relocations of that column */ -#if 1 /* FIXME */ - /* use control parameter to specify the number of extra - * locations reserved */ - int need = vc_len[j] + 10; -#endif - if (sva->r_ptr - sva->m_ptr < need) - { sva_more_space(sva, need); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_enlarge_cap(sva, vc_ref-1+j, need, 1); - } - sv_ind[vc_ptr[j] + (vc_len[j]++)] = i; - } - /* set final length of i-th row just transformed */ - xassert(len <= vr_len[i]); - vr_len[i] = len; -skip: /* return i-th row to the active set with new length */ - sgf_activate_row(i); - /* since i-th row has been changed, largest magnitude of its - * elements becomes unknown */ - vr_max[i] = -1.0; - } - /* walk thru p-th (pivot) row */ - for (i_end = (i_ptr = vr_ptr[p]) + vr_len[p]; - i_ptr < i_end; i_ptr++) - { /* get column index of v[p,j] */ - j = sv_ind[i_ptr]; - xassert(j != q); /* v[p,q] was removed */ - /* return j-th column to the active set with new length */ - if (cs_next[j] == j && vc_len[j] != 1) - { /* j-th column was marked by the pivoting routine and it is - * still not a column singleton, so leave it incative */ - xassert(cs_prev[j] == j); - } - else - sgf_activate_col(j); - nnz_diff += vc_len[j]; - /* restore zero content of the working arrays */ - flag[j] = 0; - work[j] = 0.0; - } - /* return the difference between the numbers of non-zeros in the - * active submatrix on entry and on exit, resp. */ - return nnz_diff; -} - -/*********************************************************************** -* sgf_dense_lu - compute dense LU-factorization with full pivoting -* -* This routine performs Gaussian elimination with full pivoting to -* compute dense LU-factorization of the specified matrix A of order n -* in the form: -* -* A = P * L * U * Q, (1) -* -* where L is lower triangular matrix with unit diagonal, U is upper -* triangular matrix, P and Q are permutation matrices. -* -* On entry to the routine elements of matrix A = (a[i,j]) should be -* placed in the array elements a[0], ..., a[n^2-1] in dense row-wise -* format. On exit from the routine matrix A is replaced by factors L -* and U as follows: -* -* u[1,1] u[1,2] ... u[1,n-1] u[1,n] -* l[2,1] u[2,2] ... u[2,n-1] u[2,n] -* . . . . . . . . . . . . . . -* l[n-1,1] l[n-1,2] u[n-1,n-1] u[n-1,n] -* l[n,1] l[n,2] ... l[n,n-1] u[n,n] -* -* The unit diagonal elements of L are not stored. -* -* Information on permutations of rows and columns of active submatrix -* during factorization is accumulated by the routine as follows. Every -* time the routine permutes rows i and i' or columns j and j', it also -* permutes elements r[i-1] and r[i'-1] or c[j-1] and c[j'-1], resp. -* Thus, on entry to the routine elements r[0], r[1], ..., r[n-1] and -* c[0], c[1], ..., c[n-1] should be initialized by some integers that -* identify rows and columns of the original matrix A. -* -* If the factorization has been successfully computed, the routine -* returns zero. Otherwise, if on k-th elimination step, 1 <= k <= n, -* all elements of the active submatrix are close to zero, the routine -* returns k, in which case a partial factorization is stored in the -* array a. */ - -int sgf_dense_lu(int n, double a_[], int r[], int c[], double eps) -{ /* non-optimized version */ - int i, j, k, p, q, ref; - double akk, big, temp; -# define a(i,j) a_[(i)*n+(j)] - /* initially U = A, L = P = Q = I */ - /* main elimination loop */ - for (k = 0; k < n; k++) - { /* choose pivot u[p,q], k <= p, q <= n */ - p = q = -1, big = eps; - for (i = k; i < n; i++) - { for (j = k; j < n; j++) - { /* temp = |u[i,j]| */ - if ((temp = a(i,j)) < 0.0) - temp = -temp; - if (big < temp) - p = i, q = j, big = temp; - } - } - if (p < 0) - { /* k-th elimination step failed */ - return k+1; - } - /* permute rows k and p */ - if (k != p) - { for (j = 0; j < n; j++) - temp = a(k,j), a(k,j) = a(p,j), a(p,j) = temp; - ref = r[k], r[k] = r[p], r[p] = ref; - } - /* permute columns k and q */ - if (k != q) - { for (i = 0; i < n; i++) - temp = a(i,k), a(i,k) = a(i,q), a(i,q) = temp; - ref = c[k], c[k] = c[q], c[q] = ref; - } - /* now pivot is in position u[k,k] */ - akk = a(k,k); - /* eliminate subdiagonal elements u[k+1,k], ..., u[n,k] */ - for (i = k+1; i < n; i++) - { if (a(i,k) != 0.0) - { /* gaussian multiplier l[i,k] := u[i,k] / u[k,k] */ - temp = (a(i,k) /= akk); - /* (i-th row) := (i-th row) - l[i,k] * (k-th row) */ - for (j = k+1; j < n; j++) - a(i,j) -= temp * a(k,j); - } - } - } -# undef a - return 0; -} - -/*********************************************************************** -* sgf_dense_phase - compute LU-factorization (dense phase) -* -* This routine performs dense phase of computing LU-factorization. -* -* The aim is two-fold. First, the main factorization routine switches -* to dense phase when the active submatrix is relatively dense, so -* using dense format allows significantly reduces overheads needed to -* maintain sparse data structures. And second, that is more important, -* on dense phase full pivoting is used (rather than partial pivoting) -* that allows improving numerical stability, since round-off errors -* tend to increase on last steps of the elimination process. -* -* On entry the routine assumes that elimination steps 1, 2, ..., k-1 -* have been performed, so partially transformed matrices L = P'* F * P -* and U = P'* V * Q' have the following structure: -* -* 1 k n 1 k n -* 1 1 . . . . . . . . . 1 x x x x x x x x x x -* x 1 . . . . . . . . . x x x x x x x x x -* x x 1 . . . . . . . . . x x x x x x x x -* x x x 1 . . . . . . . . . x x x x x x x -* k x x x x 1 . . . . . k . . . . * * * * * * -* x x x x . 1 . . . . . . . . * * * * * * -* x x x x . . 1 . . . . . . . * * * * * * -* x x x x . . . 1 . . . . . . * * * * * * -* x x x x . . . . 1 . . . . . * * * * * * -* n x x x x . . . . . 1 n . . . . * * * * * * -* -* matrix L matrix U -* -* where rows and columns k, k+1, ..., n of matrix U constitute the -* active submatrix A~, whose elements are marked by '*'. -* -* The routine copies the active submatrix A~ to a working array in -* dense format, compute dense factorization A~ = P~* L~* U~* Q~ using -* full pivoting, and then copies non-zero elements of factors L~ and -* U~ back to factors L and U (more precisely, to factors F and V). -* -* If the factorization has been successfully computed, the routine -* returns zero. Otherwise, if on k-th elimination step, 1 <= k <= n, -* all elements of the active submatrix are close to zero, the routine -* returns k (information on linearly dependent rows/columns in this -* case is provided by matrices P and Q). */ - -int sgf_dense_phase(LUF *luf, int k, int updat) -{ int n = luf->n; - SVA *sva = luf->sva; - int *sv_ind = sva->ind; - double *sv_val = sva->val; - int fc_ref = luf->fc_ref; - int *fc_ptr = &sva->ptr[fc_ref-1]; - int *fc_len = &sva->len[fc_ref-1]; - int *fc_cap = &sva->cap[fc_ref-1]; - int vr_ref = luf->vr_ref; - int *vr_ptr = &sva->ptr[vr_ref-1]; - int *vr_len = &sva->len[vr_ref-1]; - int *vr_cap = &sva->cap[vr_ref-1]; - double *vr_piv = luf->vr_piv; - int vc_ref = luf->vc_ref; - int *vc_len = &sva->len[vc_ref-1]; - int *pp_inv = luf->pp_inv; - int *pp_ind = luf->pp_ind; - int *qq_ind = luf->qq_ind; - int *qq_inv = luf->qq_inv; - int a_end, a_ptr, end, i, ia, ii, j, ja, jj, ka, len, na, ne, - need, ptr; - double *a_; - xassert(1 <= k && k <= n); - /* active columns of V are not longer needed; make them empty */ - for (jj = k; jj <= n; jj++) - { /* jj is number of active column of U = P'* V * Q' */ - vc_len[qq_ind[jj]] = 0; - } - /* determine order of active submatrix A~ of matrix U */ - na = n - k + 1; - xassert(1 <= na && na <= n); - /* determine number of elements in dense triangular factor (L~ or - * U~), except diagonal elements */ - ne = na * (na - 1) / 2; - /* we allocate active submatrix A~ in free (middle) part of SVA; - * to avoid defragmentation that could destroy A~ we also should - * reserve ne locations to build rows of V from rows of U~ and ne - * locations to build columns of F from columns of L~ */ - need = na * na + ne + ne; - if (sva->r_ptr - sva->m_ptr < need) - { sva_more_space(sva, need); - sv_ind = sva->ind; - sv_val = sva->val; - } - /* free (middle) part of SVA is structured as follows: - * end of left (dynamic) part - * ne free locations for new rows of V - * na free locations for active submatrix A~ - * unused locations, if any - * ne free locations for new columns of F - * beginning of right (static) part */ - a_ptr = sva->m_ptr + ne; - a_end = a_ptr + na * na; - /* copy active submatrix A~ from matrix V to working array in - * dense row-wise format */ - a_ = &sva->val[a_ptr]; -# define a(ia, ja) a_[((ia) - 1) * na + ((ja) - 1)] - for (ia = 1; ia <= na; ia++) - { /* clear ia-th row of A~ */ - for (ja = 1; ja <= na; ja++) - a(ia, ja) = 0.0; - /* ia-th row of A~ = (k-1+ia)-th row of U = i-th row of V */ - i = pp_inv[k-1+ia]; - ptr = vr_ptr[i]; - end = ptr + vr_len[i]; - for (; ptr < end; ptr++) - a(ia, qq_inv[sv_ind[ptr]]-k+1) = sv_val[ptr]; - /* i-th row of V is no longer needed; make it empty */ - vr_len[i] = 0; - } - /* compute dense factorization A~ = P~* L~* U~* Q~ */ -#if 1 /* FIXME: epsilon tolerance */ - ka = sgf_dense_lu(na, &a(1, 1), &pp_inv[k], &qq_ind[k], 1e-20); -#endif - /* rows of U with numbers pp_inv[k, k+1, ..., n] were permuted - * due to row permutations of A~; update matrix P using P~ */ - for (ii = k; ii <= n; ii++) - pp_ind[pp_inv[ii]] = ii; - /* columns of U with numbers qq_ind[k, k+1, ..., n] were permuted - * due to column permutations of A~; update matrix Q using Q~ */ - for (jj = k; jj <= n; jj++) - qq_inv[qq_ind[jj]] = jj; - /* check if dense factorization is complete */ - if (ka != 0) - { /* A~ is singular to working precision */ - /* information on linearly dependent rows/columns is provided - * by matrices P and Q */ - xassert(1 <= ka && ka <= na); - return k - 1 + ka; - } - /* build new rows of V from rows of U~ */ - for (ia = 1; ia <= na; ia++) - { /* ia-th row of U~ = (k-1+ia)-th row of U = i-th row of V */ - i = pp_inv[k-1+ia]; - xassert(vr_len[i] == 0); - /* store diagonal element u~[ia,ia] */ - vr_piv[i] = a(ia, ia); - /* determine number of non-zero non-diagonal elements in ia-th - * row of U~ */ - len = 0; - for (ja = ia+1; ja <= na; ja++) - { if (a(ia, ja) != 0.0) - len++; - } - /* reserve len locations for i-th row of matrix V in left - * (dynamic) part of SVA */ - if (vr_cap[i] < len) - { /* there should be enough room in free part of SVA */ - xassert(sva->r_ptr - sva->m_ptr >= len); - sva_enlarge_cap(sva, vr_ref-1+i, len, 0); - /* left part of SVA should not overlap matrix A~ */ - xassert(sva->m_ptr <= a_ptr); - } - /* copy non-zero non-diaginal elements of ia-th row of U~ to - * i-th row of V */ - ptr = vr_ptr[i]; - for (ja = ia+1; ja <= na; ja++) - { if (a(ia, ja) != 0.0) - { sv_ind[ptr] = qq_ind[k-1+ja]; - sv_val[ptr] = a(ia, ja); - ptr++; - } - } - xassert(ptr - vr_ptr[i] == len); - vr_len[i] = len; - } - /* build new columns of F from columns of L~ */ - for (ja = 1; ja <= na; ja++) - { /* ja-th column of L~ = (k-1+ja)-th column of L = j-th column - * of F */ - j = pp_inv[k-1+ja]; - xassert(fc_len[j] == 0); - xassert(fc_cap[j] == 0); - /* determine number of non-zero non-diagonal elements in ja-th - * column of L~ */ - len = 0; - for (ia = ja+1; ia <= na; ia++) - { if (a(ia, ja) != 0.0) - len++; - } - /* reserve len locations for j-th column of matrix F in right - * (static) part of SVA */ - /* there should be enough room in free part of SVA */ - xassert(sva->r_ptr - sva->m_ptr >= len); - if (len > 0) - sva_reserve_cap(sva, fc_ref-1+j, len); - /* right part of SVA should not overlap matrix A~ */ - xassert(a_end <= sva->r_ptr); - /* copy non-zero non-diagonal elements of ja-th column of L~ - * to j-th column of F */ - ptr = fc_ptr[j]; - for (ia = ja+1; ia <= na; ia++) - { if (a(ia, ja) != 0.0) - { sv_ind[ptr] = pp_inv[k-1+ia]; - sv_val[ptr] = a(ia, ja); - ptr++; - } - } - xassert(ptr - fc_ptr[j] == len); - fc_len[j] = len; - } - /* factors L~ and U~ are no longer needed */ -# undef a - /* if it is not planned to update matrix V, relocate all its new - * rows to the right (static) part of SVA */ - if (!updat) - { for (ia = 1; ia <= na; ia++) - { i = pp_inv[k-1+ia]; - len = vr_len[i]; - if (sva->r_ptr - sva->m_ptr < len) - { sva_more_space(sva, len); - sv_ind = sva->ind; - sv_val = sva->val; - } - sva_make_static(sva, vr_ref-1+i); - } - } - return 0; -} - -/*********************************************************************** -* sgf_factorize - compute LU-factorization (main routine) -* -* This routine computes sparse LU-factorization of specified matrix A -* using Gaussian elimination. -* -* On entry to the routine matrix V = A should be stored in column-wise -* format. -* -* If the factorization has been successfully computed, the routine -* returns zero. Otherwise, if on k-th elimination step, 1 <= k <= n, -* all elements of the active submatrix are close to zero, the routine -* returns k (information on linearly dependent rows/columns in this -* case is provided by matrices P and Q). */ - -#if 1 /* 21/II-2016 */ -/* If the matrix A is structurally singular, the routine returns -1. -* NOTE: This case can be detected only if the singl flag is set. */ -#endif - -int sgf_factorize(SGF *sgf, int singl) -{ LUF *luf = sgf->luf; - int n = luf->n; - SVA *sva = luf->sva; - int vr_ref = luf->vr_ref; - int *vr_len = &sva->len[vr_ref-1]; - double *vr_piv = luf->vr_piv; - int vc_ref = luf->vc_ref; - int *vc_len = &sva->len[vc_ref-1]; - int *pp_ind = luf->pp_ind; - int *pp_inv = luf->pp_inv; - int *qq_ind = luf->qq_ind; - int *qq_inv = luf->qq_inv; - int *rs_head = sgf->rs_head; - int *rs_prev = sgf->rs_prev; - int *rs_next = sgf->rs_next; - int *cs_head = sgf->cs_head; - int *cs_prev = sgf->cs_prev; - int *cs_next = sgf->cs_next; - double *vr_max = sgf->vr_max; - char *flag = sgf->flag; - double *work = sgf->work; - int i, j, k, k1, k2, p, q, nnz; - /* build matrix V = A in row-wise format */ - luf_build_v_rows(luf, rs_prev); - /* P := Q := I, so V = U = A, F = L = I */ - for (k = 1; k <= n; k++) - { vr_piv[k] = 0.0; - pp_ind[k] = pp_inv[k] = qq_ind[k] = qq_inv[k] = k; - } -#ifdef GLP_DEBUG - sva_check_area(sva); - luf_check_all(luf, 1); -#endif - /* perform singleton phase, if required */ - if (!singl) - { /* assume that nucleus is entire matrix U */ - k2 = 1; - } - else - { /* minimize nucleus size */ -#if 0 /* 21/II-2016 */ - sgf_reduce_nuc(luf, &k1, &k2, rs_prev, rs_next); -#else - if (sgf_reduce_nuc(luf, &k1, &k2, rs_prev, rs_next)) - return -1; -#endif -#ifdef GLP_DEBUG - xprintf("n = %d; k1 = %d; k2 = %d\n", n, k1, k2); -#endif - /* perform singleton phase */ - k2 = sgf_singl_phase(luf, k1, k2, sgf->updat, rs_prev, work); - } -#ifdef GLP_DEBUG - sva_check_area(sva); - luf_check_all(luf, k2); -#endif - /* initialize working arrays */ - rs_head[0] = cs_head[0] = 0; - for (k = 1; k <= n; k++) - { rs_head[k] = cs_head[k] = 0; - vr_max[k] = -1.0; - flag[k] = 0; - work[k] = 0.0; - } - /* build lists of active rows and columns of matrix V; determine - * number of non-zeros in initial active submatrix */ - nnz = 0; - for (k = k2; k <= n; k++) - { i = pp_inv[k]; - sgf_activate_row(i); - nnz += vr_len[i]; - j = qq_ind[k]; - sgf_activate_col(j); - } - /* main factorization loop */ - for (k = k2; k <= n; k++) - { int na; - double den; - /* calculate density of active submatrix */ - na = n - k + 1; /* order of active submatrix */ -#if 0 /* 21/VIII-2014 */ - den = (double)nnz / (double)(na * na); -#else - den = (double)nnz / ((double)(na) * (double)(na)); -#endif - /* if active submatrix is relatively dense, switch to dense - * phase */ -#if 1 /* FIXME */ - if (na >= 5 && den >= 0.71) - { -#ifdef GLP_DEBUG - xprintf("na = %d; nnz = %d; den = %g\n", na, nnz, den); -#endif - break; - } -#endif - /* choose pivot v[p,q] */ - if (sgf_choose_pivot(sgf, &p, &q) != 0) - return k; /* failure */ - /* u[i,j] = v[p,q], k <= i, j <= n */ - i = pp_ind[p]; - xassert(k <= i && i <= n); - j = qq_inv[q]; - xassert(k <= j && j <= n); - /* move u[i,j] to position u[k,k] by implicit permutations of - * rows and columns of matrix U */ - luf_swap_u_rows(k, i); - luf_swap_u_cols(k, j); - /* perform gaussian elimination */ - nnz += sgf_eliminate(sgf, p, q); - } -#if 1 /* FIXME */ - if (k <= n) - { /* continue computing factorization in dense mode */ -#ifdef GLP_DEBUG - sva_check_area(sva); - luf_check_all(luf, k); -#endif - k = sgf_dense_phase(luf, k, sgf->updat); - if (k != 0) - return k; /* failure */ - } -#endif -#ifdef GLP_DEBUG - sva_check_area(sva); - luf_check_all(luf, n+1); -#endif - /* defragment SVA; currently all columns of V are empty, so they - * will have zero capacity as required by luf_build_v_cols */ - sva_defrag_area(sva); - /* build matrix F in row-wise format */ - luf_build_f_rows(luf, rs_head); - /* build matrix V in column-wise format */ - luf_build_v_cols(luf, sgf->updat, rs_head); - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/sgf.h b/code/3rd_glpk/bflib/sgf.h deleted file mode 100644 index 4f744610..00000000 --- a/code/3rd_glpk/bflib/sgf.h +++ /dev/null @@ -1,203 +0,0 @@ -/* sgf.h (sparse Gaussian factorizer) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SGF_H -#define SGF_H - -#include "luf.h" - -typedef struct SGF SGF; - -struct SGF -{ /* sparse Gaussian factorizer workspace */ - LUF *luf; - /* LU-factorization being computed */ - /*--------------------------------------------------------------*/ - /* to efficiently choose pivot elements according to Markowitz - * strategy, the search technique proposed by Iain Duff is used; - * it is based on using two families of sets {R[0], ..., R[n]} - * and {C[0], ..., C[n]}, where R[k] and C[k], 0 <= k <= n, are, - * respectively, sets of rows and columns of the active submatrix - * of matrix V having k non-zeros (i.e. whose length is k); each - * set R[k] and C[k] is implemented as a doubly linked list */ - int *rs_head; /* int rs_head[1+n]; */ - /* rs_head[k], 0 <= k <= n, is the number of first row, which - * has k non-zeros in the active submatrix */ - int *rs_prev; /* int rs_prev[1+n]; */ - /* rs_prev[0] is not used; - * rs_prev[i], 1 <= i <= n, is the number of previous row, which - * has the same number of non-zeros as i-th row; - * rs_prev[i] < 0 means that i-th row is inactive */ - int *rs_next; /* int rs_next[1+n]; */ - /* rs_next[0] is not used; - * rs_next[i], 1 <= i <= n, is the number of next row, which has - * the same number of non-zeros as i-th row; - * rs_next[i] < 0 means that i-th row is inactive */ - int *cs_head; /* int cs_head[1+n]; */ - /* cs_head[k], 0 <= k <= n, is the number of first column, which - * has k non-zeros in the active submatrix */ - int *cs_prev; /* int cs_prev[1+n]; */ - /* cs_prev[0] is not used; - * cs_prev[j], 1 <= j <= n, is the number of previous column, - * which has the same number of non-zeros as j-th column; - * cs_prev[j] < 0 means that j-th column is inactive */ - int *cs_next; /* int cs_next[1+n]; */ - /* cs_next[0] is not used; - * cs_next[j], 1 <= j <= n, is the number of next column, which - * has the same number of non-zeros as j-th column; - * cs_next[j] < 0 means that j-th column is inactive */ - /* NOTE: cs_prev[j] = cs_next[j] = j means that j-th column was - * temporarily removed from corresponding set C[k] by the - * pivoting routine according to Uwe Suhl's heuristic */ - /*--------------------------------------------------------------*/ - /* working arrays */ - double *vr_max; /* int vr_max[1+n]; */ - /* vr_max[0] is not used; - * vr_max[i], 1 <= i <= n, is used only if i-th row of matrix V - * is active (i.e. belongs to the active submatrix), and is the - * largest magnitude of elements in that row; if vr_max[i] < 0, - * the largest magnitude is unknown yet */ - char *flag; /* char flag[1+n]; */ - /* boolean working array */ - double *work; /* double work[1+n]; */ - /* floating-point working array */ - /*--------------------------------------------------------------*/ - /* control parameters */ - int updat; - /* if this flag is set, the matrix V is assumed to be updatable; - * in this case factorized (non-active) part of V is stored in - * the left part of SVA rather than in its right part */ - double piv_tol; - /* threshold pivoting tolerance, 0 < piv_tol < 1; element v[i,j] - * of the active submatrix fits to be pivot if it satisfies to - * the stability criterion |v[i,j]| >= piv_tol * max |v[i,*]|, - * i.e. if it is not very small in the magnitude among other - * elements in the same row; decreasing this parameter gives - * better sparsity at the expense of numerical accuracy and vice - * versa */ - int piv_lim; - /* maximal allowable number of pivot candidates to be considered; - * if piv_lim pivot candidates have been considered, the pivoting - * routine terminates the search with the best candidate found */ - int suhl; - /* if this flag is set, the pivoting routine applies a heuristic - * proposed by Uwe Suhl: if a column of the active submatrix has - * no eligible pivot candidates (i.e. all its elements do not - * satisfy to the stability criterion), the routine excludes it - * from futher consideration until it becomes column singleton; - * in many cases this allows reducing the time needed to choose - * the pivot */ - double eps_tol; - /* epsilon tolerance; each element of the active submatrix, whose - * magnitude is less than eps_tol, is replaced by exact zero */ -#if 0 /* FIXME */ - double den_lim; - /* density limit; if the density of the active submatrix reaches - * this limit, the factorization routine switches from sparse to - * dense mode */ -#endif -}; - -#define sgf_activate_row(i) \ - do \ - { int len = vr_len[i]; \ - rs_prev[i] = 0; \ - rs_next[i] = rs_head[len]; \ - if (rs_next[i] != 0) \ - rs_prev[rs_next[i]] = i; \ - rs_head[len] = i; \ - } while (0) -/* include i-th row of matrix V in active set R[len] */ - -#define sgf_deactivate_row(i) \ - do \ - { if (rs_prev[i] == 0) \ - rs_head[vr_len[i]] = rs_next[i]; \ - else \ - rs_next[rs_prev[i]] = rs_next[i]; \ - if (rs_next[i] == 0) \ - ; \ - else \ - rs_prev[rs_next[i]] = rs_prev[i]; \ - rs_prev[i] = rs_next[i] = -1; \ - } while (0) -/* remove i-th row of matrix V from active set R[len] */ - -#define sgf_activate_col(j) \ - do \ - { int len = vc_len[j]; \ - cs_prev[j] = 0; \ - cs_next[j] = cs_head[len]; \ - if (cs_next[j] != 0) \ - cs_prev[cs_next[j]] = j; \ - cs_head[len] = j; \ - } while (0) -/* include j-th column of matrix V in active set C[len] */ - -#define sgf_deactivate_col(j) \ - do \ - { if (cs_prev[j] == 0) \ - cs_head[vc_len[j]] = cs_next[j]; \ - else \ - cs_next[cs_prev[j]] = cs_next[j]; \ - if (cs_next[j] == 0) \ - ; \ - else \ - cs_prev[cs_next[j]] = cs_prev[j]; \ - cs_prev[j] = cs_next[j] = -1; \ - } while (0) -/* remove j-th column of matrix V from active set C[len] */ - -#define sgf_reduce_nuc _glp_sgf_reduce_nuc -int sgf_reduce_nuc(LUF *luf, int *k1, int *k2, int cnt[/*1+n*/], - int list[/*1+n*/]); -/* initial reordering to minimize nucleus size */ - -#define sgf_singl_phase _glp_sgf_singl_phase -int sgf_singl_phase(LUF *luf, int k1, int k2, int updat, - int ind[/*1+n*/], double val[/*1+n*/]); -/* compute LU-factorization (singleton phase) */ - -#define sgf_choose_pivot _glp_sgf_choose_pivot -int sgf_choose_pivot(SGF *sgf, int *p, int *q); -/* choose pivot element v[p,q] */ - -#define sgf_eliminate _glp_sgf_eliminate -int sgf_eliminate(SGF *sgf, int p, int q); -/* perform gaussian elimination */ - -#define sgf_dense_lu _glp_sgf_dense_lu -int sgf_dense_lu(int n, double a[], int r[], int c[], double eps); -/* compute dense LU-factorization with full pivoting */ - -#define sgf_dense_phase _glp_sgf_dense_phase -int sgf_dense_phase(LUF *luf, int k, int updat); -/* compute LU-factorization (dense phase) */ - -#define sgf_factorize _glp_sgf_factorize -int sgf_factorize(SGF *sgf, int singl); -/* compute LU-factorization (main routine) */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/bflib/sva.c b/code/3rd_glpk/bflib/sva.c deleted file mode 100644 index e6a675cc..00000000 --- a/code/3rd_glpk/bflib/sva.c +++ /dev/null @@ -1,572 +0,0 @@ -/* sva.c (sparse vector area) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "sva.h" - -/*********************************************************************** -* sva_create_area - create sparse vector area (SVA) -* -* This routine creates the sparse vector area (SVA), which initially -* is empty. -* -* The parameter n_max specifies the initial number of vectors that can -* be allocated in the SVA, n_max > 0. -* -* The parameter size specifies the initial number of free locations in -* the SVA, size > 0. -* -* On exit the routine returns a pointer to the SVA created. */ - -SVA *sva_create_area(int n_max, int size) -{ SVA *sva; - xassert(0 < n_max && n_max < INT_MAX); - xassert(0 < size && size < INT_MAX); - sva = talloc(1, SVA); - sva->n_max = n_max; - sva->n = 0; - sva->ptr = talloc(1+n_max, int); - sva->len = talloc(1+n_max, int); - sva->cap = talloc(1+n_max, int); - sva->size = size; - sva->m_ptr = 1; - sva->r_ptr = size+1; - sva->head = sva->tail = 0; - sva->prev = talloc(1+n_max, int); - sva->next = talloc(1+n_max, int); - sva->ind = talloc(1+size, int); - sva->val = talloc(1+size, double); - sva->talky = 0; - return sva; -} - -/*********************************************************************** -* sva_alloc_vecs - allocate new vectors in SVA -* -* This routine allocates nnn new empty vectors, nnn > 0, in the sparse -* vector area (SVA). -* -* The new vectors are assigned reference numbers k, k+1, ..., k+nnn-1, -* where k is a reference number assigned to the very first new vector, -* which is returned by the routine on exit. */ - -int sva_alloc_vecs(SVA *sva, int nnn) -{ int n = sva->n; - int n_max = sva->n_max; - int *ptr = sva->ptr; - int *len = sva->len; - int *cap = sva->cap; - int *prev = sva->prev; - int *next = sva->next; - int k, new_n; -#if 1 - if (sva->talky) - xprintf("sva_alloc_vecs: nnn = %d\n", nnn); -#endif - xassert(nnn > 0); - /* determine new number of vectors in SVA */ - new_n = n + nnn; - xassert(new_n > n); - if (n_max < new_n) - { /* enlarge the SVA arrays */ - while (n_max < new_n) - { n_max += n_max; - xassert(n_max > 0); - } - sva->n_max = n_max; - sva->ptr = ptr = trealloc(ptr, 1+n_max, int); - sva->len = len = trealloc(len, 1+n_max, int); - sva->cap = cap = trealloc(cap, 1+n_max, int); - sva->prev = prev = trealloc(prev, 1+n_max, int); - sva->next = next = trealloc(next, 1+n_max, int); - } - /* initialize new vectors */ - sva->n = new_n; - for (k = n+1; k <= new_n; k++) - { ptr[k] = len[k] = cap[k] = 0; - prev[k] = next[k] = -1; - } -#if 1 - if (sva->talky) - xprintf("now sva->n_max = %d, sva->n = %d\n", - sva->n_max, sva->n); -#endif - /* return reference number of very first new vector */ - return n+1; -} - -/*********************************************************************** -* sva_resize_area - change size of SVA storage -* -* This routine increases or decrases the size of the SVA storage by -* reallocating it. -* -* The parameter delta specifies the number of location by which the -* current size of the SVA storage should be increased (if delta > 0) -* or decreased (if delta < 0). Note that if delta is negative, it -* should not be less than the current size of the middle part. -* -* As a result of this operation the size of the middle part of SVA is -* increased/decreased by delta locations. -* -* NOTE: This operation changes ptr[k] for all vectors stored in the -* right part of SVA. */ - -void sva_resize_area(SVA *sva, int delta) -{ int n = sva->n; - int *ptr = sva->ptr; - int size = sva->size; - int m_ptr = sva->m_ptr; - int r_ptr = sva->r_ptr; - int k, r_size; -#if 1 - if (sva->talky) - xprintf("sva_resize_area: delta = %d\n", delta); -#endif - xassert(delta != 0); - /* determine size of the right part, in locations */ - r_size = size - r_ptr + 1; - /* relocate the right part in case of negative delta */ - if (delta < 0) - { xassert(delta >= m_ptr - r_ptr); - sva->r_ptr += delta; - memmove(&sva->ind[sva->r_ptr], &sva->ind[r_ptr], - r_size * sizeof(int)); - memmove(&sva->val[sva->r_ptr], &sva->val[r_ptr], - r_size * sizeof(double)); - } - /* reallocate the storage arrays */ - xassert(delta < INT_MAX - sva->size); - sva->size += delta; - sva->ind = trealloc(sva->ind, 1+sva->size, int); - sva->val = trealloc(sva->val, 1+sva->size, double); - /* relocate the right part in case of positive delta */ - if (delta > 0) - { sva->r_ptr += delta; - memmove(&sva->ind[sva->r_ptr], &sva->ind[r_ptr], - r_size * sizeof(int)); - memmove(&sva->val[sva->r_ptr], &sva->val[r_ptr], - r_size * sizeof(double)); - } - /* update pointers to vectors stored in the right part */ - for (k = 1; k <= n; k++) - { if (ptr[k] >= r_ptr) - ptr[k] += delta; - } -#if 1 - if (sva->talky) - xprintf("now sva->size = %d\n", sva->size); -#endif - return; -} - -/*********************************************************************** -* sva_defrag_area - defragment left part of SVA -* -* This routine performs "garbage" collection to defragment the left -* part of SVA. -* -* NOTE: This operation may change ptr[k] and cap[k] for all vectors -* stored in the left part of SVA. */ - -void sva_defrag_area(SVA *sva) -{ int *ptr = sva->ptr; - int *len = sva->len; - int *cap = sva->cap; - int *prev = sva->prev; - int *next = sva->next; - int *ind = sva->ind; - double *val = sva->val; - int k, next_k, ptr_k, len_k, m_ptr, head, tail; -#if 1 - if (sva->talky) - { xprintf("sva_defrag_area:\n"); - xprintf("before defragmenting = %d %d %d\n", sva->m_ptr - 1, - sva->r_ptr - sva->m_ptr, sva->size + 1 - sva->r_ptr); - } -#endif - m_ptr = 1; - head = tail = 0; - /* walk through the linked list of vectors stored in the left - * part of SVA */ - for (k = sva->head; k != 0; k = next_k) - { /* save number of next vector in the list */ - next_k = next[k]; - /* determine length of k-th vector */ - len_k = len[k]; - if (len_k == 0) - { /* k-th vector is empty; remove it from the left part */ - ptr[k] = cap[k] = 0; - prev[k] = next[k] = -1; - } - else - { /* determine pointer to first location of k-th vector */ - ptr_k = ptr[k]; - xassert(m_ptr <= ptr_k); - /* relocate k-th vector to the beginning of the left part, - * if necessary */ - if (m_ptr < ptr_k) - { memmove(&ind[m_ptr], &ind[ptr_k], - len_k * sizeof(int)); - memmove(&val[m_ptr], &val[ptr_k], - len_k * sizeof(double)); - ptr[k] = m_ptr; - } - /* remove unused locations from k-th vector */ - cap[k] = len_k; - /* the left part of SVA has been enlarged */ - m_ptr += len_k; - /* add k-th vector to the end of the new linked list */ - prev[k] = tail; - next[k] = 0; - if (head == 0) - head = k; - else - next[tail] = k; - tail = k; - } - } - /* set new pointer to the middle part of SVA */ - xassert(m_ptr <= sva->r_ptr); - sva->m_ptr = m_ptr; - /* set new head and tail of the linked list */ - sva->head = head; - sva->tail = tail; -#if 1 - if (sva->talky) - xprintf("after defragmenting = %d %d %d\n", sva->m_ptr - 1, - sva->r_ptr - sva->m_ptr, sva->size + 1 - sva->r_ptr); -#endif - return; -} - -/*********************************************************************** -* sva_more_space - increase size of middle (free) part of SVA -* -* This routine increases the size of the middle (free) part of the -* sparse vector area (SVA). -* -* The parameter m_size specifies the minimal size, in locations, of -* the middle part to be provided. This new size should be greater than -* the current size of the middle part. -* -* First, the routine defragments the left part of SVA. Then, if the -* size of the left part has not sufficiently increased, the routine -* increases the total size of the SVA storage by reallocating it. */ - -void sva_more_space(SVA *sva, int m_size) -{ int size, delta; -#if 1 - if (sva->talky) - xprintf("sva_more_space: m_size = %d\n", m_size); -#endif - xassert(m_size > sva->r_ptr - sva->m_ptr); - /* defragment the left part */ - sva_defrag_area(sva); - /* set, heuristically, the minimal size of the middle part to be - * not less than the size of the defragmented left part */ - if (m_size < sva->m_ptr - 1) - m_size = sva->m_ptr - 1; - /* if there is still not enough room, increase the total size of - * the SVA storage */ - if (sva->r_ptr - sva->m_ptr < m_size) - { size = sva->size; /* new sva size */ - for (;;) - { delta = size - sva->size; - if (sva->r_ptr - sva->m_ptr + delta >= m_size) - break; - size += size; - xassert(size > 0); - } - sva_resize_area(sva, delta); - xassert(sva->r_ptr - sva->m_ptr >= m_size); - } - return; -} - -/*********************************************************************** -* sva_enlarge_cap - enlarge capacity of specified vector -* -* This routine enlarges the current capacity of the specified vector -* by relocating its content. -* -* The parameter k specifies the reference number of the vector whose -* capacity should be enlarged, 1 <= k <= n. This vector should either -* have zero capacity or be stored in the left (dynamic) part of SVA. -* -* The parameter new_cap specifies the new capacity of the vector, -* in locations. This new capacity should be greater than the current -* capacity of the vector. -* -* The parameter skip is a flag. If this flag is set, the routine does -* *not* copy numerical values of elements of the vector on relocating -* its content, i.e. only element indices are copied. -* -* NOTE: On entry to the routine the middle part of SVA should have at -* least new_cap free locations. */ - -void sva_enlarge_cap(SVA *sva, int k, int new_cap, int skip) -{ int *ptr = sva->ptr; - int *len = sva->len; - int *cap = sva->cap; - int *prev = sva->prev; - int *next = sva->next; - int *ind = sva->ind; - double *val = sva->val; - xassert(1 <= k && k <= sva->n); - xassert(new_cap > cap[k]); - /* there should be at least new_cap free locations */ - xassert(sva->r_ptr - sva->m_ptr >= new_cap); - /* relocate the vector */ - if (cap[k] == 0) - { /* the vector is empty */ - xassert(ptr[k] == 0); - xassert(len[k] == 0); - } - else - { /* the vector has non-zero capacity */ - xassert(ptr[k] + len[k] <= sva->m_ptr); - /* copy the current vector content to the beginning of the - * middle part */ - if (len[k] > 0) - { memcpy(&ind[sva->m_ptr], &ind[ptr[k]], - len[k] * sizeof(int)); - if (!skip) - memcpy(&val[sva->m_ptr], &val[ptr[k]], - len[k] * sizeof(double)); - } - /* remove the vector from the linked list */ - if (prev[k] == 0) - sva->head = next[k]; - else - { /* preceding vector exists; increase its capacity */ - cap[prev[k]] += cap[k]; - next[prev[k]] = next[k]; - } - if (next[k] == 0) - sva->tail = prev[k]; - else - prev[next[k]] = prev[k]; - } - /* set new pointer and capacity of the vector */ - ptr[k] = sva->m_ptr; - cap[k] = new_cap; - /* add the vector to the end of the linked list */ - prev[k] = sva->tail; - next[k] = 0; - if (sva->head == 0) - sva->head = k; - else - next[sva->tail] = k; - sva->tail = k; - /* new_cap free locations have been consumed */ - sva->m_ptr += new_cap; - xassert(sva->m_ptr <= sva->r_ptr); - return; -} - -/*********************************************************************** -* sva_reserve_cap - reserve locations for specified vector -* -* This routine reserves locations for the specified vector in the -* right (static) part of SVA. -* -* The parameter k specifies the reference number of the vector (this -* vector should have zero capacity), 1 <= k <= n. -* -* The parameter new_cap specifies a non-zero capacity of the vector, -* in locations. -* -* NOTE: On entry to the routine the middle part of SVA should have at -* least new_cap free locations. */ - -void sva_reserve_cap(SVA *sva, int k, int new_cap) -{ int *ptr = sva->ptr; - int *len = sva->len; - int *cap = sva->cap; - xassert(1 <= k && k <= sva->n); - xassert(new_cap > 0); - xassert(ptr[k] == 0 && len[k] == 0 && cap[k] == 0); - /* there should be at least new_cap free locations */ - xassert(sva->r_ptr - sva->m_ptr >= new_cap); - /* set the pointer and capacity of the vector */ - ptr[k] = sva->r_ptr - new_cap; - cap[k] = new_cap; - /* new_cap free locations have been consumed */ - sva->r_ptr -= new_cap; - return; -} - -/*********************************************************************** -* sva_make_static - relocate specified vector to right part of SVA -* -* Assuming that the specified vector is stored in the left (dynamic) -* part of SVA, this routine makes the vector static by relocating its -* content to the right (static) part of SVA. However, if the specified -* vector has zero capacity, the routine does nothing. -* -* The parameter k specifies the reference number of the vector to be -* relocated, 1 <= k <= n. -* -* NOTE: On entry to the routine the middle part of SVA should have at -* least len[k] free locations, where len[k] is the length of the -* vector to be relocated. */ - -void sva_make_static(SVA *sva, int k) -{ int *ptr = sva->ptr; - int *len = sva->len; - int *cap = sva->cap; - int *prev = sva->prev; - int *next = sva->next; - int *ind = sva->ind; - double *val = sva->val; - int ptr_k, len_k; - xassert(1 <= k && k <= sva->n); - /* if the vector has zero capacity, do nothing */ - if (cap[k] == 0) - { xassert(ptr[k] == 0); - xassert(len[k] == 0); - goto done; - } - /* there should be at least len[k] free locations */ - len_k = len[k]; - xassert(sva->r_ptr - sva->m_ptr >= len_k); - /* remove the vector from the linked list */ - if (prev[k] == 0) - sva->head = next[k]; - else - { /* preceding vector exists; increase its capacity */ - cap[prev[k]] += cap[k]; - next[prev[k]] = next[k]; - } - if (next[k] == 0) - sva->tail = prev[k]; - else - prev[next[k]] = prev[k]; - /* if the vector has zero length, make it empty */ - if (len_k == 0) - { ptr[k] = cap[k] = 0; - goto done; - } - /* copy the vector content to the beginning of the right part */ - ptr_k = sva->r_ptr - len_k; - memcpy(&ind[ptr_k], &ind[ptr[k]], len_k * sizeof(int)); - memcpy(&val[ptr_k], &val[ptr[k]], len_k * sizeof(double)); - /* set new pointer and capacity of the vector */ - ptr[k] = ptr_k; - cap[k] = len_k; - /* len[k] free locations have been consumed */ - sva->r_ptr -= len_k; -done: return; -} - -/*********************************************************************** -* sva_check_area - check sparse vector area (SVA) -* -* This routine checks the SVA data structures for correctness. -* -* NOTE: For testing/debugging only. */ - -void sva_check_area(SVA *sva) -{ int n_max = sva->n_max; - int n = sva->n; - int *ptr = sva->ptr; - int *len = sva->len; - int *cap = sva->cap; - int size = sva->size; - int m_ptr = sva->m_ptr; - int r_ptr = sva->r_ptr; - int head = sva->head; - int tail = sva->tail; - int *prev = sva->prev; - int *next = sva->next; - int k; -#if 0 /* 16/II-2004; SVA may be empty */ - xassert(1 <= n && n <= n_max); -#else - xassert(0 <= n && n <= n_max); -#endif - xassert(1 <= m_ptr && m_ptr <= r_ptr && r_ptr <= size+1); - /* all vectors included the linked list should have non-zero - * capacity and be stored in the left part */ - for (k = head; k != 0; k = next[k]) - { xassert(1 <= k && k <= n); - xassert(cap[k] > 0); - xassert(0 <= len[k] && len[k] <= cap[k]); - if (prev[k] == 0) - xassert(k == head); - else - { xassert(1 <= prev[k] && prev[k] <= n); - xassert(next[prev[k]] == k); - } - if (next[k] == 0) - { xassert(k == tail); - xassert(ptr[k] + cap[k] <= m_ptr); - } - else - { xassert(1 <= next[k] && next[k] <= n); - xassert(prev[next[k]] == k); - xassert(ptr[k] + cap[k] <= ptr[next[k]]); - } - cap[k] = -cap[k]; - } - /* all other vectors should either have zero capacity or be - * stored in the right part */ - for (k = 1; k <= n; k++) - { if (cap[k] < 0) - { /* k-th vector is stored in the left part */ - cap[k] = -cap[k]; - } - else if (cap[k] == 0) - { /* k-th vector has zero capacity */ - xassert(ptr[k] == 0); - xassert(len[k] == 0); - } - else /* cap[k] > 0 */ - { /* k-th vector is stored in the right part */ - xassert(0 <= len[k] && len[k] <= cap[k]); - xassert(r_ptr <= ptr[k] && ptr[k] + cap[k] <= size+1); - } - } - return; -} - -/*********************************************************************** -* sva_delete_area - delete sparse vector area (SVA) -* -* This routine deletes the sparse vector area (SVA) freeing all the -* memory allocated to it. */ - -void sva_delete_area(SVA *sva) -{ tfree(sva->ptr); - tfree(sva->len); - tfree(sva->cap); - tfree(sva->prev); - tfree(sva->next); - tfree(sva->ind); - tfree(sva->val); - tfree(sva); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/bflib/sva.h b/code/3rd_glpk/bflib/sva.h deleted file mode 100644 index 0eab317b..00000000 --- a/code/3rd_glpk/bflib/sva.h +++ /dev/null @@ -1,161 +0,0 @@ -/* sva.h (sparse vector area) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SVA_H -#define SVA_H - -/*********************************************************************** -* Sparse Vector Area (SVA) is a container for sparse vectors. This -* program object is used mainly on computing factorization, where the -* sparse vectors are rows and columns of sparse matrices. -* -* The SVA storage is a set of locations numbered 1, 2, ..., size, -* where size is the size of SVA, which is the total number of -* locations currently allocated. Each location is identified by its -* pointer p, 1 <= p <= size, and is the pair (ind[p], val[p]), where -* ind[p] and val[p] are, respectively, the index and value fields used -* to store the index and numeric value of a particular vector element. -* -* Each sparse vector is identified by its reference number k, -* 1 <= k <= n, where n is the total number of vectors currently stored -* in SVA, and defined by the triplet (ptr[k], len[k], cap[k]), where: -* ptr[k] is a pointer to the first location of the vector; len[k] is -* the vector length, which is the number of its non-zero elements, -* len[k] >= 0; and cap[k] is the capacity of the vector, which is the -* total number of adjacent locations allocated to that vector, -* cap[k] >= len[k]. Thus, non-zero elements of k-th vector are stored -* in locations ptr[k], ptr[k]+1, ..., ptr[k]+len[k]-1, and locations -* ptr[k]+len[k], ptr[k]+len[k]+1, ..., ptr[k]+cap[k]-1 are reserved. -* -* The SVA storage is divided into three parts as follows: -* -* Locations 1, 2, ..., m_ptr-1 constitute the left (dynamic) part of -* SVA. This part is used to store vectors, whose capacity may change. -* Note that all vectors stored in the left part are also included in -* a doubly linked list, where they are ordered by increasing their -* pointers ptr[k] (this list is needed for efficient implementation -* of the garbage collector used to defragment the left part of SVA); -* -* Locations m_ptr, m_ptr+1, ..., r_ptr-1 are free and constitute the -* middle (free) part of SVA. -* -* Locations r_ptr, r_ptr+1, ..., size constitute the right (static) -* part of SVA. This part is used to store vectors, whose capacity is -* not changed. */ - -typedef struct SVA SVA; - -struct SVA -{ /* sparse vector area */ - int n_max; - /* maximal value of n (enlarged automatically) */ - int n; - /* number of currently allocated vectors, 0 <= n <= n_max */ - int *ptr; /* int ptr[1+n_max]; */ - /* ptr[0] is not used; - * ptr[k], 1 <= i <= n, is pointer to first location of k-th - * vector in the arrays ind and val */ - int *len; /* int len[1+n_max]; */ - /* len[0] is not used; - * len[k], 1 <= k <= n, is length of k-th vector, len[k] >= 0 */ - int *cap; /* int cap[1+n_max]; */ - /* cap[0] is not used; - * cap[k], 1 <= k <= n, is capacity of k-th vector (the number - * of adjacent locations allocated to it), cap[k] >= len[k] */ - /* NOTE: if cap[k] = 0, then ptr[k] = 0 and len[k] = 0 */ - int size; - /* total number of locations in SVA */ - int m_ptr, r_ptr; - /* partitioning pointers that define the left, middle, and right - * parts of SVA (see above); 1 <= m_ptr <= r_ptr <= size+1 */ - int head; - /* number of first (leftmost) vector in the linked list */ - int tail; - /* number of last (rightmost) vector in the linked list */ - int *prev; /* int prev[1+n_max]; */ - /* prev[0] is not used; - * prev[k] is number of vector which precedes k-th vector in the - * linked list; - * prev[k] < 0 means that k-th vector is not in the list */ - int *next; /* int next[1+n_max]; */ - /* next[0] is not used; - * next[k] is number of vector which succedes k-th vector in the - * linked list; - * next[k] < 0 means that k-th vector is not in the list */ - /* NOTE: only vectors having non-zero capacity and stored in the - * left part of SVA are included in this linked list */ - int *ind; /* int ind[1+size]; */ - /* ind[0] is not used; - * ind[p], 1 <= p <= size, is index field of location p */ - double *val; /* double val[1+size]; */ - /* val[0] is not used; - * val[p], 1 <= p <= size, is value field of location p */ -#if 1 - int talky; - /* option to enable talky mode */ -#endif -}; - -#define sva_create_area _glp_sva_create_area -SVA *sva_create_area(int n_max, int size); -/* create sparse vector area (SVA) */ - -#define sva_alloc_vecs _glp_sva_alloc_vecs -int sva_alloc_vecs(SVA *sva, int nnn); -/* allocate new vectors in SVA */ - -#define sva_resize_area _glp_sva_resize_area -void sva_resize_area(SVA *sva, int delta); -/* change size of SVA storage */ - -#define sva_defrag_area _glp_sva_defrag_area -void sva_defrag_area(SVA *sva); -/* defragment left part of SVA */ - -#define sva_more_space _glp_sva_more_space -void sva_more_space(SVA *sva, int m_size); -/* increase size of middle (free) part of SVA */ - -#define sva_enlarge_cap _glp_sva_enlarge_cap -void sva_enlarge_cap(SVA *sva, int k, int new_cap, int skip); -/* enlarge capacity of specified vector */ - -#define sva_reserve_cap _glp_sva_reserve_cap -void sva_reserve_cap(SVA *sva, int k, int new_cap); -/* reserve locations for specified vector */ - -#define sva_make_static _glp_sva_make_static -void sva_make_static(SVA *sva, int k); -/* relocate specified vector to right part of SVA */ - -#define sva_check_area _glp_sva_check_area -void sva_check_area(SVA *sva); -/* check sparse vector area (SVA) */ - -#define sva_delete_area _glp_sva_delete_area -void sva_delete_area(SVA *sva); -/* delete sparse vector area (SVA) */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/colamd/COPYING b/code/3rd_glpk/colamd/COPYING deleted file mode 100644 index 84bba36d..00000000 --- a/code/3rd_glpk/colamd/COPYING +++ /dev/null @@ -1,502 +0,0 @@ - GNU LESSER GENERAL PUBLIC LICENSE - Version 2.1, February 1999 - - Copyright (C) 1991, 1999 Free Software Foundation, Inc. - 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - -[This is the first released version of the Lesser GPL. It also counts - as the successor of the GNU Library Public License, version 2, hence - the version number 2.1.] - - Preamble - - The licenses for most software are designed to take away your -freedom to share and change it. By contrast, the GNU General Public -Licenses are intended to guarantee your freedom to share and change -free software--to make sure the software is free for all its users. - - This license, the Lesser General Public License, applies to some -specially designated software packages--typically libraries--of the -Free Software Foundation and other authors who decide to use it. You -can use it too, but we suggest you first think carefully about whether -this license or the ordinary General Public License is the better -strategy to use in any particular case, based on the explanations below. - - When we speak of free software, we are referring to freedom of use, -not price. Our General Public Licenses are designed to make sure that -you have the freedom to distribute copies of free software (and charge -for this service if you wish); that you receive source code or can get -it if you want it; that you can change the software and use pieces of -it in new free programs; and that you are informed that you can do -these things. - - To protect your rights, we need to make restrictions that forbid -distributors to deny you these rights or to ask you to surrender these -rights. These restrictions translate to certain responsibilities for -you if you distribute copies of the library or if you modify it. - - For example, if you distribute copies of the library, whether gratis -or for a fee, you must give the recipients all the rights that we gave -you. You must make sure that they, too, receive or can get the source -code. If you link other code with the library, you must provide -complete object files to the recipients, so that they can relink them -with the library after making changes to the library and recompiling -it. And you must show them these terms so they know their rights. - - We protect your rights with a two-step method: (1) we copyright the -library, and (2) we offer you this license, which gives you legal -permission to copy, distribute and/or modify the library. - - To protect each distributor, we want to make it very clear that -there is no warranty for the free library. Also, if the library is -modified by someone else and passed on, the recipients should know -that what they have is not the original version, so that the original -author's reputation will not be affected by problems that might be -introduced by others. - - Finally, software patents pose a constant threat to the existence of -any free program. We wish to make sure that a company cannot -effectively restrict the users of a free program by obtaining a -restrictive license from a patent holder. Therefore, we insist that -any patent license obtained for a version of the library must be -consistent with the full freedom of use specified in this license. - - Most GNU software, including some libraries, is covered by the -ordinary GNU General Public License. This license, the GNU Lesser -General Public License, applies to certain designated libraries, and -is quite different from the ordinary General Public License. We use -this license for certain libraries in order to permit linking those -libraries into non-free programs. - - When a program is linked with a library, whether statically or using -a shared library, the combination of the two is legally speaking a -combined work, a derivative of the original library. The ordinary -General Public License therefore permits such linking only if the -entire combination fits its criteria of freedom. The Lesser General -Public License permits more lax criteria for linking other code with -the library. - - We call this license the "Lesser" General Public License because it -does Less to protect the user's freedom than the ordinary General -Public License. It also provides other free software developers Less -of an advantage over competing non-free programs. These disadvantages -are the reason we use the ordinary General Public License for many -libraries. However, the Lesser license provides advantages in certain -special circumstances. - - For example, on rare occasions, there may be a special need to -encourage the widest possible use of a certain library, so that it becomes -a de-facto standard. To achieve this, non-free programs must be -allowed to use the library. A more frequent case is that a free -library does the same job as widely used non-free libraries. In this -case, there is little to gain by limiting the free library to free -software only, so we use the Lesser General Public License. - - In other cases, permission to use a particular library in non-free -programs enables a greater number of people to use a large body of -free software. For example, permission to use the GNU C Library in -non-free programs enables many more people to use the whole GNU -operating system, as well as its variant, the GNU/Linux operating -system. - - Although the Lesser General Public License is Less protective of the -users' freedom, it does ensure that the user of a program that is -linked with the Library has the freedom and the wherewithal to run -that program using a modified version of the Library. - - The precise terms and conditions for copying, distribution and -modification follow. Pay close attention to the difference between a -"work based on the library" and a "work that uses the library". The -former contains code derived from the library, whereas the latter must -be combined with the library in order to run. - - GNU LESSER GENERAL PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. This License Agreement applies to any software library or other -program which contains a notice placed by the copyright holder or -other authorized party saying it may be distributed under the terms of -this Lesser General Public License (also called "this License"). -Each licensee is addressed as "you". - - A "library" means a collection of software functions and/or data -prepared so as to be conveniently linked with application programs -(which use some of those functions and data) to form executables. - - The "Library", below, refers to any such software library or work -which has been distributed under these terms. A "work based on the -Library" means either the Library or any derivative work under -copyright law: that is to say, a work containing the Library or a -portion of it, either verbatim or with modifications and/or translated -straightforwardly into another language. (Hereinafter, translation is -included without limitation in the term "modification".) - - "Source code" for a work means the preferred form of the work for -making modifications to it. For a library, complete source code means -all the source code for all modules it contains, plus any associated -interface definition files, plus the scripts used to control compilation -and installation of the library. - - Activities other than copying, distribution and modification are not -covered by this License; they are outside its scope. The act of -running a program using the Library is not restricted, and output from -such a program is covered only if its contents constitute a work based -on the Library (independent of the use of the Library in a tool for -writing it). Whether that is true depends on what the Library does -and what the program that uses the Library does. - - 1. You may copy and distribute verbatim copies of the Library's -complete source code as you receive it, in any medium, provided that -you conspicuously and appropriately publish on each copy an -appropriate copyright notice and disclaimer of warranty; keep intact -all the notices that refer to this License and to the absence of any -warranty; and distribute a copy of this License along with the -Library. - - You may charge a fee for the physical act of transferring a copy, -and you may at your option offer warranty protection in exchange for a -fee. - - 2. You may modify your copy or copies of the Library or any portion -of it, thus forming a work based on the Library, and copy and -distribute such modifications or work under the terms of Section 1 -above, provided that you also meet all of these conditions: - - a) The modified work must itself be a software library. - - b) You must cause the files modified to carry prominent notices - stating that you changed the files and the date of any change. - - c) You must cause the whole of the work to be licensed at no - charge to all third parties under the terms of this License. - - d) If a facility in the modified Library refers to a function or a - table of data to be supplied by an application program that uses - the facility, other than as an argument passed when the facility - is invoked, then you must make a good faith effort to ensure that, - in the event an application does not supply such function or - table, the facility still operates, and performs whatever part of - its purpose remains meaningful. - - (For example, a function in a library to compute square roots has - a purpose that is entirely well-defined independent of the - application. Therefore, Subsection 2d requires that any - application-supplied function or table used by this function must - be optional: if the application does not supply it, the square - root function must still compute square roots.) - -These requirements apply to the modified work as a whole. If -identifiable sections of that work are not derived from the Library, -and can be reasonably considered independent and separate works in -themselves, then this License, and its terms, do not apply to those -sections when you distribute them as separate works. But when you -distribute the same sections as part of a whole which is a work based -on the Library, the distribution of the whole must be on the terms of -this License, whose permissions for other licensees extend to the -entire whole, and thus to each and every part regardless of who wrote -it. - -Thus, it is not the intent of this section to claim rights or contest -your rights to work written entirely by you; rather, the intent is to -exercise the right to control the distribution of derivative or -collective works based on the Library. - -In addition, mere aggregation of another work not based on the Library -with the Library (or with a work based on the Library) on a volume of -a storage or distribution medium does not bring the other work under -the scope of this License. - - 3. You may opt to apply the terms of the ordinary GNU General Public -License instead of this License to a given copy of the Library. To do -this, you must alter all the notices that refer to this License, so -that they refer to the ordinary GNU General Public License, version 2, -instead of to this License. (If a newer version than version 2 of the -ordinary GNU General Public License has appeared, then you can specify -that version instead if you wish.) Do not make any other change in -these notices. - - Once this change is made in a given copy, it is irreversible for -that copy, so the ordinary GNU General Public License applies to all -subsequent copies and derivative works made from that copy. - - This option is useful when you wish to copy part of the code of -the Library into a program that is not a library. - - 4. You may copy and distribute the Library (or a portion or -derivative of it, under Section 2) in object code or executable form -under the terms of Sections 1 and 2 above provided that you accompany -it with the complete corresponding machine-readable source code, which -must be distributed under the terms of Sections 1 and 2 above on a -medium customarily used for software interchange. - - If distribution of object code is made by offering access to copy -from a designated place, then offering equivalent access to copy the -source code from the same place satisfies the requirement to -distribute the source code, even though third parties are not -compelled to copy the source along with the object code. - - 5. A program that contains no derivative of any portion of the -Library, but is designed to work with the Library by being compiled or -linked with it, is called a "work that uses the Library". Such a -work, in isolation, is not a derivative work of the Library, and -therefore falls outside the scope of this License. - - However, linking a "work that uses the Library" with the Library -creates an executable that is a derivative of the Library (because it -contains portions of the Library), rather than a "work that uses the -library". The executable is therefore covered by this License. -Section 6 states terms for distribution of such executables. - - When a "work that uses the Library" uses material from a header file -that is part of the Library, the object code for the work may be a -derivative work of the Library even though the source code is not. -Whether this is true is especially significant if the work can be -linked without the Library, or if the work is itself a library. The -threshold for this to be true is not precisely defined by law. - - If such an object file uses only numerical parameters, data -structure layouts and accessors, and small macros and small inline -functions (ten lines or less in length), then the use of the object -file is unrestricted, regardless of whether it is legally a derivative -work. (Executables containing this object code plus portions of the -Library will still fall under Section 6.) - - Otherwise, if the work is a derivative of the Library, you may -distribute the object code for the work under the terms of Section 6. -Any executables containing that work also fall under Section 6, -whether or not they are linked directly with the Library itself. - - 6. As an exception to the Sections above, you may also combine or -link a "work that uses the Library" with the Library to produce a -work containing portions of the Library, and distribute that work -under terms of your choice, provided that the terms permit -modification of the work for the customer's own use and reverse -engineering for debugging such modifications. - - You must give prominent notice with each copy of the work that the -Library is used in it and that the Library and its use are covered by -this License. You must supply a copy of this License. If the work -during execution displays copyright notices, you must include the -copyright notice for the Library among them, as well as a reference -directing the user to the copy of this License. Also, you must do one -of these things: - - a) Accompany the work with the complete corresponding - machine-readable source code for the Library including whatever - changes were used in the work (which must be distributed under - Sections 1 and 2 above); and, if the work is an executable linked - with the Library, with the complete machine-readable "work that - uses the Library", as object code and/or source code, so that the - user can modify the Library and then relink to produce a modified - executable containing the modified Library. (It is understood - that the user who changes the contents of definitions files in the - Library will not necessarily be able to recompile the application - to use the modified definitions.) - - b) Use a suitable shared library mechanism for linking with the - Library. A suitable mechanism is one that (1) uses at run time a - copy of the library already present on the user's computer system, - rather than copying library functions into the executable, and (2) - will operate properly with a modified version of the library, if - the user installs one, as long as the modified version is - interface-compatible with the version that the work was made with. - - c) Accompany the work with a written offer, valid for at - least three years, to give the same user the materials - specified in Subsection 6a, above, for a charge no more - than the cost of performing this distribution. - - d) If distribution of the work is made by offering access to copy - from a designated place, offer equivalent access to copy the above - specified materials from the same place. - - e) Verify that the user has already received a copy of these - materials or that you have already sent this user a copy. - - For an executable, the required form of the "work that uses the -Library" must include any data and utility programs needed for -reproducing the executable from it. However, as a special exception, -the materials to be distributed need not include anything that is -normally distributed (in either source or binary form) with the major -components (compiler, kernel, and so on) of the operating system on -which the executable runs, unless that component itself accompanies -the executable. - - It may happen that this requirement contradicts the license -restrictions of other proprietary libraries that do not normally -accompany the operating system. Such a contradiction means you cannot -use both them and the Library together in an executable that you -distribute. - - 7. You may place library facilities that are a work based on the -Library side-by-side in a single library together with other library -facilities not covered by this License, and distribute such a combined -library, provided that the separate distribution of the work based on -the Library and of the other library facilities is otherwise -permitted, and provided that you do these two things: - - a) Accompany the combined library with a copy of the same work - based on the Library, uncombined with any other library - facilities. This must be distributed under the terms of the - Sections above. - - b) Give prominent notice with the combined library of the fact - that part of it is a work based on the Library, and explaining - where to find the accompanying uncombined form of the same work. - - 8. You may not copy, modify, sublicense, link with, or distribute -the Library except as expressly provided under this License. Any -attempt otherwise to copy, modify, sublicense, link with, or -distribute the Library is void, and will automatically terminate your -rights under this License. However, parties who have received copies, -or rights, from you under this License will not have their licenses -terminated so long as such parties remain in full compliance. - - 9. You are not required to accept this License, since you have not -signed it. However, nothing else grants you permission to modify or -distribute the Library or its derivative works. These actions are -prohibited by law if you do not accept this License. Therefore, by -modifying or distributing the Library (or any work based on the -Library), you indicate your acceptance of this License to do so, and -all its terms and conditions for copying, distributing or modifying -the Library or works based on it. - - 10. Each time you redistribute the Library (or any work based on the -Library), the recipient automatically receives a license from the -original licensor to copy, distribute, link with or modify the Library -subject to these terms and conditions. You may not impose any further -restrictions on the recipients' exercise of the rights granted herein. -You are not responsible for enforcing compliance by third parties with -this License. - - 11. If, as a consequence of a court judgment or allegation of patent -infringement or for any other reason (not limited to patent issues), -conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot -distribute so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you -may not distribute the Library at all. For example, if a patent -license would not permit royalty-free redistribution of the Library by -all those who receive copies directly or indirectly through you, then -the only way you could satisfy both it and this License would be to -refrain entirely from distribution of the Library. - -If any portion of this section is held invalid or unenforceable under any -particular circumstance, the balance of the section is intended to apply, -and the section as a whole is intended to apply in other circumstances. - -It is not the purpose of this section to induce you to infringe any -patents or other property right claims or to contest validity of any -such claims; this section has the sole purpose of protecting the -integrity of the free software distribution system which is -implemented by public license practices. Many people have made -generous contributions to the wide range of software distributed -through that system in reliance on consistent application of that -system; it is up to the author/donor to decide if he or she is willing -to distribute software through any other system and a licensee cannot -impose that choice. - -This section is intended to make thoroughly clear what is believed to -be a consequence of the rest of this License. - - 12. If the distribution and/or use of the Library is restricted in -certain countries either by patents or by copyrighted interfaces, the -original copyright holder who places the Library under this License may add -an explicit geographical distribution limitation excluding those countries, -so that distribution is permitted only in or among countries not thus -excluded. In such case, this License incorporates the limitation as if -written in the body of this License. - - 13. The Free Software Foundation may publish revised and/or new -versions of the Lesser General Public License from time to time. -Such new versions will be similar in spirit to the present version, -but may differ in detail to address new problems or concerns. - -Each version is given a distinguishing version number. If the Library -specifies a version number of this License which applies to it and -"any later version", you have the option of following the terms and -conditions either of that version or of any later version published by -the Free Software Foundation. If the Library does not specify a -license version number, you may choose any version ever published by -the Free Software Foundation. - - 14. If you wish to incorporate parts of the Library into other free -programs whose distribution conditions are incompatible with these, -write to the author to ask for permission. For software which is -copyrighted by the Free Software Foundation, write to the Free -Software Foundation; we sometimes make exceptions for this. Our -decision will be guided by the two goals of preserving the free status -of all derivatives of our free software and of promoting the sharing -and reuse of software generally. - - NO WARRANTY - - 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO -WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. -EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR -OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY -KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE -IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE -LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME -THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN -WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY -AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU -FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR -CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE -LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING -RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A -FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF -SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH -DAMAGES. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Libraries - - If you develop a new library, and you want it to be of the greatest -possible use to the public, we recommend making it free software that -everyone can redistribute and change. You can do so by permitting -redistribution under these terms (or, alternatively, under the terms of the -ordinary General Public License). - - To apply these terms, attach the following notices to the library. It is -safest to attach them to the start of each source file to most effectively -convey the exclusion of warranty; and each file should have at least the -"copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This library is free software; you can redistribute it and/or - modify it under the terms of the GNU Lesser General Public - License as published by the Free Software Foundation; either - version 2.1 of the License, or (at your option) any later version. - - This library is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - Lesser General Public License for more details. - - You should have received a copy of the GNU Lesser General Public - License along with this library; if not, write to the Free Software - Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA - -Also add information on how to contact you by electronic and paper mail. - -You should also get your employer (if you work as a programmer) or your -school, if any, to sign a "copyright disclaimer" for the library, if -necessary. Here is a sample; alter the names: - - Yoyodyne, Inc., hereby disclaims all copyright interest in the - library `Frob' (a library for tweaking knobs) written by James Random Hacker. - - , 1 April 1990 - Ty Coon, President of Vice - -That's all there is to it! diff --git a/code/3rd_glpk/colamd/README b/code/3rd_glpk/colamd/README deleted file mode 100644 index a365059f..00000000 --- a/code/3rd_glpk/colamd/README +++ /dev/null @@ -1,98 +0,0 @@ -NOTE: Files in this subdirectory are NOT part of the GLPK package, but - are used with GLPK. - - The original code was modified according to GLPK requirements by - Andrew Makhorin . -************************************************************************ -COLAMD/SYMAMD Version 2.7, Copyright (C) 1998-2007, Timothy A. Davis, -All Rights Reserved. - -Description: - - colamd: an approximate minimum degree column ordering algorithm, - for LU factorization of symmetric or unsymmetric matrices, - QR factorization, least squares, interior point methods for - linear programming problems, and other related problems. - - symamd: an approximate minimum degree ordering algorithm for - Cholesky factorization of symmetric matrices. - -Purpose: - - Colamd computes a permutation Q such that the Cholesky factorization - of (AQ)'(AQ) has less fill-in and requires fewer floating point - operations than A'A. This also provides a good ordering for sparse - partial pivoting methods, P(AQ) = LU, where Q is computed prior to - numerical factorization, and P is computed during numerical - factorization via conventional partial pivoting with row - interchanges. Colamd is the column ordering method used in SuperLU, - part of the ScaLAPACK library. It is also available as built-in - function in MATLAB Version 6, available from MathWorks, Inc. - (http://www.mathworks.com). This routine can be used in place of - colmmd in MATLAB. - - Symamd computes a permutation P of a symmetric matrix A such that - the Cholesky factorization of PAP' has less fill-in and requires - fewer floating point operations than A. Symamd constructs a matrix - M such that M'M has the same nonzero pattern of A, and then orders - the columns of M using colmmd. The column ordering of M is then - returned as the row and column ordering P of A. - -Authors: - - The authors of the code itself are Stefan I. Larimore and Timothy A. - Davis (davis at cise.ufl.edu), University of Florida. The algorithm - was developed in collaboration with John Gilbert, Xerox PARC, and - Esmond Ng, Oak Ridge National Laboratory. - -Acknowledgements: - - This work was supported by the National Science Foundation, under - grants DMS-9504974 and DMS-9803599. - -License: - - This library is free software; you can redistribute it and/or - modify it under the terms of the GNU Lesser General Public License - as published by the Free Software Foundation; either version 2.1 of - the License, or (at your option) any later version. - - This library is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - Lesser General Public License for more details. - - You should have received a copy of the GNU Lesser General Public - License along with this library; if not, write to the Free Software - Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 - USA. - - Permission is hereby granted to use or copy this program under the - terms of the GNU LGPL, provided that the Copyright, this License, - and the Availability of the original version is retained on all - copies. User documentation of any code that uses this code or any - modified version of this code must cite the Copyright, this License, - the Availability note, and "Used by permission." Permission to - modify the code and to distribute modified code is granted, provided - the Copyright, this License, and the Availability note are retained, - and a notice that the code was modified is included. - - COLAMD is also available under alternate licenses, contact T. Davis - for details. - -Availability: - - The colamd/symamd library is available at: - - http://www.cise.ufl.edu/research/sparse/colamd/ - -References: - - T. A. Davis, J. R. Gilbert, S. Larimore, E. Ng, An approximate - column minimum degree ordering algorithm, ACM Transactions on - Mathematical Software, vol. 30, no. 3., pp. 353-376, 2004. - - T. A. Davis, J. R. Gilbert, S. Larimore, E. Ng, Algorithm 836: - COLAMD, an approximate column minimum degree ordering algorithm, ACM - Transactions on Mathematical Software, vol. 30, no. 3., pp. 377-380, - 2004. diff --git a/code/3rd_glpk/colamd/colamd.c b/code/3rd_glpk/colamd/colamd.c deleted file mode 100644 index 86ddd6b7..00000000 --- a/code/3rd_glpk/colamd/colamd.c +++ /dev/null @@ -1,3622 +0,0 @@ -/* ========================================================================== */ -/* === colamd/symamd - a sparse matrix column ordering algorithm ============ */ -/* ========================================================================== */ - -/* COLAMD / SYMAMD - - colamd: an approximate minimum degree column ordering algorithm, - for LU factorization of symmetric or unsymmetric matrices, - QR factorization, least squares, interior point methods for - linear programming problems, and other related problems. - - symamd: an approximate minimum degree ordering algorithm for Cholesky - factorization of symmetric matrices. - - Purpose: - - Colamd computes a permutation Q such that the Cholesky factorization of - (AQ)'(AQ) has less fill-in and requires fewer floating point operations - than A'A. This also provides a good ordering for sparse partial - pivoting methods, P(AQ) = LU, where Q is computed prior to numerical - factorization, and P is computed during numerical factorization via - conventional partial pivoting with row interchanges. Colamd is the - column ordering method used in SuperLU, part of the ScaLAPACK library. - It is also available as built-in function in MATLAB Version 6, - available from MathWorks, Inc. (http://www.mathworks.com). This - routine can be used in place of colmmd in MATLAB. - - Symamd computes a permutation P of a symmetric matrix A such that the - Cholesky factorization of PAP' has less fill-in and requires fewer - floating point operations than A. Symamd constructs a matrix M such - that M'M has the same nonzero pattern of A, and then orders the columns - of M using colmmd. The column ordering of M is then returned as the - row and column ordering P of A. - - Authors: - - The authors of the code itself are Stefan I. Larimore and Timothy A. - Davis (davis at cise.ufl.edu), University of Florida. The algorithm was - developed in collaboration with John Gilbert, Xerox PARC, and Esmond - Ng, Oak Ridge National Laboratory. - - Acknowledgements: - - This work was supported by the National Science Foundation, under - grants DMS-9504974 and DMS-9803599. - - Copyright and License: - - Copyright (c) 1998-2007, Timothy A. Davis, All Rights Reserved. - COLAMD is also available under alternate licenses, contact T. Davis - for details. - - This library is free software; you can redistribute it and/or - modify it under the terms of the GNU Lesser General Public - License as published by the Free Software Foundation; either - version 2.1 of the License, or (at your option) any later version. - - This library is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - Lesser General Public License for more details. - - You should have received a copy of the GNU Lesser General Public - License along with this library; if not, write to the Free Software - Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 - USA - - Permission is hereby granted to use or copy this program under the - terms of the GNU LGPL, provided that the Copyright, this License, - and the Availability of the original version is retained on all copies. - User documentation of any code that uses this code or any modified - version of this code must cite the Copyright, this License, the - Availability note, and "Used by permission." Permission to modify - the code and to distribute modified code is granted, provided the - Copyright, this License, and the Availability note are retained, - and a notice that the code was modified is included. - - Availability: - - The colamd/symamd library is available at - - http://www.cise.ufl.edu/research/sparse/colamd/ - - This is the http://www.cise.ufl.edu/research/sparse/colamd/colamd.c - file. It requires the colamd.h file. It is required by the colamdmex.c - and symamdmex.c files, for the MATLAB interface to colamd and symamd. - Appears as ACM Algorithm 836. - - See the ChangeLog file for changes since Version 1.0. - - References: - - T. A. Davis, J. R. Gilbert, S. Larimore, E. Ng, An approximate column - minimum degree ordering algorithm, ACM Transactions on Mathematical - Software, vol. 30, no. 3., pp. 353-376, 2004. - - T. A. Davis, J. R. Gilbert, S. Larimore, E. Ng, Algorithm 836: COLAMD, - an approximate column minimum degree ordering algorithm, ACM - Transactions on Mathematical Software, vol. 30, no. 3., pp. 377-380, - 2004. - -*/ - -/* ========================================================================== */ -/* === Description of user-callable routines ================================ */ -/* ========================================================================== */ - -/* COLAMD includes both int and UF_long versions of all its routines. The - * description below is for the int version. For UF_long, all int arguments - * become UF_long. UF_long is normally defined as long, except for WIN64. - - ---------------------------------------------------------------------------- - colamd_recommended: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - size_t colamd_recommended (int nnz, int n_row, int n_col) ; - size_t colamd_l_recommended (UF_long nnz, UF_long n_row, - UF_long n_col) ; - - Purpose: - - Returns recommended value of Alen for use by colamd. Returns 0 - if any input argument is negative. The use of this routine - is optional. Not needed for symamd, which dynamically allocates - its own memory. - - Note that in v2.4 and earlier, these routines returned int or long. - They now return a value of type size_t. - - Arguments (all input arguments): - - int nnz ; Number of nonzeros in the matrix A. This must - be the same value as p [n_col] in the call to - colamd - otherwise you will get a wrong value - of the recommended memory to use. - - int n_row ; Number of rows in the matrix A. - - int n_col ; Number of columns in the matrix A. - - ---------------------------------------------------------------------------- - colamd_set_defaults: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - colamd_set_defaults (double knobs [COLAMD_KNOBS]) ; - colamd_l_set_defaults (double knobs [COLAMD_KNOBS]) ; - - Purpose: - - Sets the default parameters. The use of this routine is optional. - - Arguments: - - double knobs [COLAMD_KNOBS] ; Output only. - - NOTE: the meaning of the dense row/col knobs has changed in v2.4 - - knobs [0] and knobs [1] control dense row and col detection: - - Colamd: rows with more than - max (16, knobs [COLAMD_DENSE_ROW] * sqrt (n_col)) - entries are removed prior to ordering. Columns with more than - max (16, knobs [COLAMD_DENSE_COL] * sqrt (MIN (n_row,n_col))) - entries are removed prior to - ordering, and placed last in the output column ordering. - - Symamd: uses only knobs [COLAMD_DENSE_ROW], which is knobs [0]. - Rows and columns with more than - max (16, knobs [COLAMD_DENSE_ROW] * sqrt (n)) - entries are removed prior to ordering, and placed last in the - output ordering. - - COLAMD_DENSE_ROW and COLAMD_DENSE_COL are defined as 0 and 1, - respectively, in colamd.h. Default values of these two knobs - are both 10. Currently, only knobs [0] and knobs [1] are - used, but future versions may use more knobs. If so, they will - be properly set to their defaults by the future version of - colamd_set_defaults, so that the code that calls colamd will - not need to change, assuming that you either use - colamd_set_defaults, or pass a (double *) NULL pointer as the - knobs array to colamd or symamd. - - knobs [2]: aggressive absorption - - knobs [COLAMD_AGGRESSIVE] controls whether or not to do - aggressive absorption during the ordering. Default is TRUE. - - - ---------------------------------------------------------------------------- - colamd: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - int colamd (int n_row, int n_col, int Alen, int *A, int *p, - double knobs [COLAMD_KNOBS], int stats [COLAMD_STATS]) ; - UF_long colamd_l (UF_long n_row, UF_long n_col, UF_long Alen, - UF_long *A, UF_long *p, double knobs [COLAMD_KNOBS], - UF_long stats [COLAMD_STATS]) ; - - Purpose: - - Computes a column ordering (Q) of A such that P(AQ)=LU or - (AQ)'AQ=LL' have less fill-in and require fewer floating point - operations than factorizing the unpermuted matrix A or A'A, - respectively. - - Returns: - - TRUE (1) if successful, FALSE (0) otherwise. - - Arguments: - - int n_row ; Input argument. - - Number of rows in the matrix A. - Restriction: n_row >= 0. - Colamd returns FALSE if n_row is negative. - - int n_col ; Input argument. - - Number of columns in the matrix A. - Restriction: n_col >= 0. - Colamd returns FALSE if n_col is negative. - - int Alen ; Input argument. - - Restriction (see note): - Alen >= 2*nnz + 6*(n_col+1) + 4*(n_row+1) + n_col - Colamd returns FALSE if these conditions are not met. - - Note: this restriction makes an modest assumption regarding - the size of the two typedef's structures in colamd.h. - We do, however, guarantee that - - Alen >= colamd_recommended (nnz, n_row, n_col) - - will be sufficient. Note: the macro version does not check - for integer overflow, and thus is not recommended. Use - the colamd_recommended routine instead. - - int A [Alen] ; Input argument, undefined on output. - - A is an integer array of size Alen. Alen must be at least as - large as the bare minimum value given above, but this is very - low, and can result in excessive run time. For best - performance, we recommend that Alen be greater than or equal to - colamd_recommended (nnz, n_row, n_col), which adds - nnz/5 to the bare minimum value given above. - - On input, the row indices of the entries in column c of the - matrix are held in A [(p [c]) ... (p [c+1]-1)]. The row indices - in a given column c need not be in ascending order, and - duplicate row indices may be be present. However, colamd will - work a little faster if both of these conditions are met - (Colamd puts the matrix into this format, if it finds that the - the conditions are not met). - - The matrix is 0-based. That is, rows are in the range 0 to - n_row-1, and columns are in the range 0 to n_col-1. Colamd - returns FALSE if any row index is out of range. - - The contents of A are modified during ordering, and are - undefined on output. - - int p [n_col+1] ; Both input and output argument. - - p is an integer array of size n_col+1. On input, it holds the - "pointers" for the column form of the matrix A. Column c of - the matrix A is held in A [(p [c]) ... (p [c+1]-1)]. The first - entry, p [0], must be zero, and p [c] <= p [c+1] must hold - for all c in the range 0 to n_col-1. The value p [n_col] is - thus the total number of entries in the pattern of the matrix A. - Colamd returns FALSE if these conditions are not met. - - On output, if colamd returns TRUE, the array p holds the column - permutation (Q, for P(AQ)=LU or (AQ)'(AQ)=LL'), where p [0] is - the first column index in the new ordering, and p [n_col-1] is - the last. That is, p [k] = j means that column j of A is the - kth pivot column, in AQ, where k is in the range 0 to n_col-1 - (p [0] = j means that column j of A is the first column in AQ). - - If colamd returns FALSE, then no permutation is returned, and - p is undefined on output. - - double knobs [COLAMD_KNOBS] ; Input argument. - - See colamd_set_defaults for a description. - - int stats [COLAMD_STATS] ; Output argument. - - Statistics on the ordering, and error status. - See colamd.h for related definitions. - Colamd returns FALSE if stats is not present. - - stats [0]: number of dense or empty rows ignored. - - stats [1]: number of dense or empty columns ignored (and - ordered last in the output permutation p) - Note that a row can become "empty" if it - contains only "dense" and/or "empty" columns, - and similarly a column can become "empty" if it - only contains "dense" and/or "empty" rows. - - stats [2]: number of garbage collections performed. - This can be excessively high if Alen is close - to the minimum required value. - - stats [3]: status code. < 0 is an error code. - > 1 is a warning or notice. - - 0 OK. Each column of the input matrix contained - row indices in increasing order, with no - duplicates. - - 1 OK, but columns of input matrix were jumbled - (unsorted columns or duplicate entries). Colamd - had to do some extra work to sort the matrix - first and remove duplicate entries, but it - still was able to return a valid permutation - (return value of colamd was TRUE). - - stats [4]: highest numbered column that - is unsorted or has duplicate - entries. - stats [5]: last seen duplicate or - unsorted row index. - stats [6]: number of duplicate or - unsorted row indices. - - -1 A is a null pointer - - -2 p is a null pointer - - -3 n_row is negative - - stats [4]: n_row - - -4 n_col is negative - - stats [4]: n_col - - -5 number of nonzeros in matrix is negative - - stats [4]: number of nonzeros, p [n_col] - - -6 p [0] is nonzero - - stats [4]: p [0] - - -7 A is too small - - stats [4]: required size - stats [5]: actual size (Alen) - - -8 a column has a negative number of entries - - stats [4]: column with < 0 entries - stats [5]: number of entries in col - - -9 a row index is out of bounds - - stats [4]: column with bad row index - stats [5]: bad row index - stats [6]: n_row, # of rows of matrx - - -10 (unused; see symamd.c) - - -999 (unused; see symamd.c) - - Future versions may return more statistics in the stats array. - - Example: - - See http://www.cise.ufl.edu/research/sparse/colamd/example.c - for a complete example. - - To order the columns of a 5-by-4 matrix with 11 nonzero entries in - the following nonzero pattern - - x 0 x 0 - x 0 x x - 0 x x 0 - 0 0 x x - x x 0 0 - - with default knobs and no output statistics, do the following: - - #include "colamd.h" - #define ALEN 100 - int A [ALEN] = {0, 1, 4, 2, 4, 0, 1, 2, 3, 1, 3} ; - int p [ ] = {0, 3, 5, 9, 11} ; - int stats [COLAMD_STATS] ; - colamd (5, 4, ALEN, A, p, (double *) NULL, stats) ; - - The permutation is returned in the array p, and A is destroyed. - - ---------------------------------------------------------------------------- - symamd: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - int symamd (int n, int *A, int *p, int *perm, - double knobs [COLAMD_KNOBS], int stats [COLAMD_STATS], - void (*allocate) (size_t, size_t), void (*release) (void *)) ; - UF_long symamd_l (UF_long n, UF_long *A, UF_long *p, UF_long *perm, - double knobs [COLAMD_KNOBS], UF_long stats [COLAMD_STATS], - void (*allocate) (size_t, size_t), void (*release) (void *)) ; - - Purpose: - - The symamd routine computes an ordering P of a symmetric sparse - matrix A such that the Cholesky factorization PAP' = LL' remains - sparse. It is based on a column ordering of a matrix M constructed - so that the nonzero pattern of M'M is the same as A. The matrix A - is assumed to be symmetric; only the strictly lower triangular part - is accessed. You must pass your selected memory allocator (usually - calloc/free or mxCalloc/mxFree) to symamd, for it to allocate - memory for the temporary matrix M. - - Returns: - - TRUE (1) if successful, FALSE (0) otherwise. - - Arguments: - - int n ; Input argument. - - Number of rows and columns in the symmetrix matrix A. - Restriction: n >= 0. - Symamd returns FALSE if n is negative. - - int A [nnz] ; Input argument. - - A is an integer array of size nnz, where nnz = p [n]. - - The row indices of the entries in column c of the matrix are - held in A [(p [c]) ... (p [c+1]-1)]. The row indices in a - given column c need not be in ascending order, and duplicate - row indices may be present. However, symamd will run faster - if the columns are in sorted order with no duplicate entries. - - The matrix is 0-based. That is, rows are in the range 0 to - n-1, and columns are in the range 0 to n-1. Symamd - returns FALSE if any row index is out of range. - - The contents of A are not modified. - - int p [n+1] ; Input argument. - - p is an integer array of size n+1. On input, it holds the - "pointers" for the column form of the matrix A. Column c of - the matrix A is held in A [(p [c]) ... (p [c+1]-1)]. The first - entry, p [0], must be zero, and p [c] <= p [c+1] must hold - for all c in the range 0 to n-1. The value p [n] is - thus the total number of entries in the pattern of the matrix A. - Symamd returns FALSE if these conditions are not met. - - The contents of p are not modified. - - int perm [n+1] ; Output argument. - - On output, if symamd returns TRUE, the array perm holds the - permutation P, where perm [0] is the first index in the new - ordering, and perm [n-1] is the last. That is, perm [k] = j - means that row and column j of A is the kth column in PAP', - where k is in the range 0 to n-1 (perm [0] = j means - that row and column j of A are the first row and column in - PAP'). The array is used as a workspace during the ordering, - which is why it must be of length n+1, not just n. - - double knobs [COLAMD_KNOBS] ; Input argument. - - See colamd_set_defaults for a description. - - int stats [COLAMD_STATS] ; Output argument. - - Statistics on the ordering, and error status. - See colamd.h for related definitions. - Symamd returns FALSE if stats is not present. - - stats [0]: number of dense or empty row and columns ignored - (and ordered last in the output permutation - perm). Note that a row/column can become - "empty" if it contains only "dense" and/or - "empty" columns/rows. - - stats [1]: (same as stats [0]) - - stats [2]: number of garbage collections performed. - - stats [3]: status code. < 0 is an error code. - > 1 is a warning or notice. - - 0 OK. Each column of the input matrix contained - row indices in increasing order, with no - duplicates. - - 1 OK, but columns of input matrix were jumbled - (unsorted columns or duplicate entries). Symamd - had to do some extra work to sort the matrix - first and remove duplicate entries, but it - still was able to return a valid permutation - (return value of symamd was TRUE). - - stats [4]: highest numbered column that - is unsorted or has duplicate - entries. - stats [5]: last seen duplicate or - unsorted row index. - stats [6]: number of duplicate or - unsorted row indices. - - -1 A is a null pointer - - -2 p is a null pointer - - -3 (unused, see colamd.c) - - -4 n is negative - - stats [4]: n - - -5 number of nonzeros in matrix is negative - - stats [4]: # of nonzeros (p [n]). - - -6 p [0] is nonzero - - stats [4]: p [0] - - -7 (unused) - - -8 a column has a negative number of entries - - stats [4]: column with < 0 entries - stats [5]: number of entries in col - - -9 a row index is out of bounds - - stats [4]: column with bad row index - stats [5]: bad row index - stats [6]: n_row, # of rows of matrx - - -10 out of memory (unable to allocate temporary - workspace for M or count arrays using the - "allocate" routine passed into symamd). - - Future versions may return more statistics in the stats array. - - void * (*allocate) (size_t, size_t) - - A pointer to a function providing memory allocation. The - allocated memory must be returned initialized to zero. For a - C application, this argument should normally be a pointer to - calloc. For a MATLAB mexFunction, the routine mxCalloc is - passed instead. - - void (*release) (size_t, size_t) - - A pointer to a function that frees memory allocated by the - memory allocation routine above. For a C application, this - argument should normally be a pointer to free. For a MATLAB - mexFunction, the routine mxFree is passed instead. - - - ---------------------------------------------------------------------------- - colamd_report: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - colamd_report (int stats [COLAMD_STATS]) ; - colamd_l_report (UF_long stats [COLAMD_STATS]) ; - - Purpose: - - Prints the error status and statistics recorded in the stats - array on the standard error output (for a standard C routine) - or on the MATLAB output (for a mexFunction). - - Arguments: - - int stats [COLAMD_STATS] ; Input only. Statistics from colamd. - - - ---------------------------------------------------------------------------- - symamd_report: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - symamd_report (int stats [COLAMD_STATS]) ; - symamd_l_report (UF_long stats [COLAMD_STATS]) ; - - Purpose: - - Prints the error status and statistics recorded in the stats - array on the standard error output (for a standard C routine) - or on the MATLAB output (for a mexFunction). - - Arguments: - - int stats [COLAMD_STATS] ; Input only. Statistics from symamd. - - -*/ - -/* ========================================================================== */ -/* === Scaffolding code definitions ======================================== */ -/* ========================================================================== */ - -/* Ensure that debugging is turned off: */ -#ifndef NDEBUG -#define NDEBUG -#endif - -/* turn on debugging by uncommenting the following line - #undef NDEBUG -*/ - -/* - Our "scaffolding code" philosophy: In our opinion, well-written library - code should keep its "debugging" code, and just normally have it turned off - by the compiler so as not to interfere with performance. This serves - several purposes: - - (1) assertions act as comments to the reader, telling you what the code - expects at that point. All assertions will always be true (unless - there really is a bug, of course). - - (2) leaving in the scaffolding code assists anyone who would like to modify - the code, or understand the algorithm (by reading the debugging output, - one can get a glimpse into what the code is doing). - - (3) (gasp!) for actually finding bugs. This code has been heavily tested - and "should" be fully functional and bug-free ... but you never know... - - The code will become outrageously slow when debugging is - enabled. To control the level of debugging output, set an environment - variable D to 0 (little), 1 (some), 2, 3, or 4 (lots). When debugging, - you should see the following message on the standard output: - - colamd: debug version, D = 1 (THIS WILL BE SLOW!) - - or a similar message for symamd. If you don't, then debugging has not - been enabled. - -*/ - -/* ========================================================================== */ -/* === Include files ======================================================== */ -/* ========================================================================== */ - -#include "colamd.h" - -#if 0 /* by mao */ -#include -#include - -#ifdef MATLAB_MEX_FILE -#include "mex.h" -#include "matrix.h" -#endif /* MATLAB_MEX_FILE */ - -#if !defined (NPRINT) || !defined (NDEBUG) -#include -#endif - -#ifndef NULL -#define NULL ((void *) 0) -#endif -#endif - -/* ========================================================================== */ -/* === int or UF_long ======================================================= */ -/* ========================================================================== */ - -#if 0 /* by mao */ -/* define UF_long */ -#include "UFconfig.h" -#endif - -#ifdef DLONG - -#define Int UF_long -#define ID UF_long_id -#define Int_MAX UF_long_max - -#define COLAMD_recommended colamd_l_recommended -#define COLAMD_set_defaults colamd_l_set_defaults -#define COLAMD_MAIN colamd_l -#define SYMAMD_MAIN symamd_l -#define COLAMD_report colamd_l_report -#define SYMAMD_report symamd_l_report - -#else - -#define Int int -#define ID "%d" -#define Int_MAX INT_MAX - -#define COLAMD_recommended colamd_recommended -#define COLAMD_set_defaults colamd_set_defaults -#define COLAMD_MAIN colamd -#define SYMAMD_MAIN symamd -#define COLAMD_report colamd_report -#define SYMAMD_report symamd_report - -#endif - -/* ========================================================================== */ -/* === Row and Column structures ============================================ */ -/* ========================================================================== */ - -/* User code that makes use of the colamd/symamd routines need not directly */ -/* reference these structures. They are used only for colamd_recommended. */ - -typedef struct Colamd_Col_struct -{ - Int start ; /* index for A of first row in this column, or DEAD */ - /* if column is dead */ - Int length ; /* number of rows in this column */ - union - { - Int thickness ; /* number of original columns represented by this */ - /* col, if the column is alive */ - Int parent ; /* parent in parent tree super-column structure, if */ - /* the column is dead */ - } shared1 ; - union - { - Int score ; /* the score used to maintain heap, if col is alive */ - Int order ; /* pivot ordering of this column, if col is dead */ - } shared2 ; - union - { - Int headhash ; /* head of a hash bucket, if col is at the head of */ - /* a degree list */ - Int hash ; /* hash value, if col is not in a degree list */ - Int prev ; /* previous column in degree list, if col is in a */ - /* degree list (but not at the head of a degree list) */ - } shared3 ; - union - { - Int degree_next ; /* next column, if col is in a degree list */ - Int hash_next ; /* next column, if col is in a hash list */ - } shared4 ; - -} Colamd_Col ; - -typedef struct Colamd_Row_struct -{ - Int start ; /* index for A of first col in this row */ - Int length ; /* number of principal columns in this row */ - union - { - Int degree ; /* number of principal & non-principal columns in row */ - Int p ; /* used as a row pointer in init_rows_cols () */ - } shared1 ; - union - { - Int mark ; /* for computing set differences and marking dead rows*/ - Int first_column ;/* first column in row (used in garbage collection) */ - } shared2 ; - -} Colamd_Row ; - -/* ========================================================================== */ -/* === Definitions ========================================================== */ -/* ========================================================================== */ - -/* Routines are either PUBLIC (user-callable) or PRIVATE (not user-callable) */ -#define PUBLIC -#define PRIVATE static - -#define DENSE_DEGREE(alpha,n) \ - ((Int) MAX (16.0, (alpha) * sqrt ((double) (n)))) - -#define MAX(a,b) (((a) > (b)) ? (a) : (b)) -#define MIN(a,b) (((a) < (b)) ? (a) : (b)) - -#define ONES_COMPLEMENT(r) (-(r)-1) - -/* -------------------------------------------------------------------------- */ -/* Change for version 2.1: define TRUE and FALSE only if not yet defined */ -/* -------------------------------------------------------------------------- */ - -#ifndef TRUE -#define TRUE (1) -#endif - -#ifndef FALSE -#define FALSE (0) -#endif - -/* -------------------------------------------------------------------------- */ - -#define EMPTY (-1) - -/* Row and column status */ -#define ALIVE (0) -#define DEAD (-1) - -/* Column status */ -#define DEAD_PRINCIPAL (-1) -#define DEAD_NON_PRINCIPAL (-2) - -/* Macros for row and column status update and checking. */ -#define ROW_IS_DEAD(r) ROW_IS_MARKED_DEAD (Row[r].shared2.mark) -#define ROW_IS_MARKED_DEAD(row_mark) (row_mark < ALIVE) -#define ROW_IS_ALIVE(r) (Row [r].shared2.mark >= ALIVE) -#define COL_IS_DEAD(c) (Col [c].start < ALIVE) -#define COL_IS_ALIVE(c) (Col [c].start >= ALIVE) -#define COL_IS_DEAD_PRINCIPAL(c) (Col [c].start == DEAD_PRINCIPAL) -#define KILL_ROW(r) { Row [r].shared2.mark = DEAD ; } -#define KILL_PRINCIPAL_COL(c) { Col [c].start = DEAD_PRINCIPAL ; } -#define KILL_NON_PRINCIPAL_COL(c) { Col [c].start = DEAD_NON_PRINCIPAL ; } - -/* ========================================================================== */ -/* === Colamd reporting mechanism =========================================== */ -/* ========================================================================== */ - -#if defined (MATLAB_MEX_FILE) || defined (MATHWORKS) -/* In MATLAB, matrices are 1-based to the user, but 0-based internally */ -#define INDEX(i) ((i)+1) -#else -/* In C, matrices are 0-based and indices are reported as such in *_report */ -#define INDEX(i) (i) -#endif - -/* All output goes through the PRINTF macro. */ -#define PRINTF(params) { if (colamd_printf != NULL) (void) colamd_printf params ; } - -/* ========================================================================== */ -/* === Prototypes of PRIVATE routines ======================================= */ -/* ========================================================================== */ - -PRIVATE Int init_rows_cols -( - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [], - Int p [], - Int stats [COLAMD_STATS] -) ; - -PRIVATE void init_scoring -( - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [], - Int head [], - double knobs [COLAMD_KNOBS], - Int *p_n_row2, - Int *p_n_col2, - Int *p_max_deg -) ; - -PRIVATE Int find_ordering -( - Int n_row, - Int n_col, - Int Alen, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [], - Int head [], - Int n_col2, - Int max_deg, - Int pfree, - Int aggressive -) ; - -PRIVATE void order_children -( - Int n_col, - Colamd_Col Col [], - Int p [] -) ; - -PRIVATE void detect_super_cols -( - -#ifndef NDEBUG - Int n_col, - Colamd_Row Row [], -#endif /* NDEBUG */ - - Colamd_Col Col [], - Int A [], - Int head [], - Int row_start, - Int row_length -) ; - -PRIVATE Int garbage_collection -( - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [], - Int *pfree -) ; - -PRIVATE Int clear_mark -( - Int tag_mark, - Int max_mark, - Int n_row, - Colamd_Row Row [] -) ; - -PRIVATE void print_report -( - char *method, - Int stats [COLAMD_STATS] -) ; - -/* ========================================================================== */ -/* === Debugging prototypes and definitions ================================= */ -/* ========================================================================== */ - -#ifndef NDEBUG - -#if 0 /* by mao */ -#include -#endif - -/* colamd_debug is the *ONLY* global variable, and is only */ -/* present when debugging */ - -PRIVATE Int colamd_debug = 0 ; /* debug print level */ - -#define DEBUG0(params) { PRINTF (params) ; } -#define DEBUG1(params) { if (colamd_debug >= 1) PRINTF (params) ; } -#define DEBUG2(params) { if (colamd_debug >= 2) PRINTF (params) ; } -#define DEBUG3(params) { if (colamd_debug >= 3) PRINTF (params) ; } -#define DEBUG4(params) { if (colamd_debug >= 4) PRINTF (params) ; } - -#if 0 /* by mao */ -#ifdef MATLAB_MEX_FILE -#define ASSERT(expression) (mxAssert ((expression), "")) -#else -#define ASSERT(expression) (assert (expression)) -#endif /* MATLAB_MEX_FILE */ -#else -#define ASSERT xassert -#endif - -PRIVATE void colamd_get_debug /* gets the debug print level from getenv */ -( - char *method -) ; - -PRIVATE void debug_deg_lists -( - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int head [], - Int min_score, - Int should, - Int max_deg -) ; - -PRIVATE void debug_mark -( - Int n_row, - Colamd_Row Row [], - Int tag_mark, - Int max_mark -) ; - -PRIVATE void debug_matrix -( - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [] -) ; - -PRIVATE void debug_structures -( - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [], - Int n_col2 -) ; - -#else /* NDEBUG */ - -/* === No debugging ========================================================= */ - -#define DEBUG0(params) ; -#define DEBUG1(params) ; -#define DEBUG2(params) ; -#define DEBUG3(params) ; -#define DEBUG4(params) ; - -#define ASSERT(expression) - -#endif /* NDEBUG */ - -/* ========================================================================== */ -/* === USER-CALLABLE ROUTINES: ============================================== */ -/* ========================================================================== */ - -/* ========================================================================== */ -/* === colamd_recommended =================================================== */ -/* ========================================================================== */ - -/* - The colamd_recommended routine returns the suggested size for Alen. This - value has been determined to provide good balance between the number of - garbage collections and the memory requirements for colamd. If any - argument is negative, or if integer overflow occurs, a 0 is returned as an - error condition. 2*nnz space is required for the row and column - indices of the matrix. COLAMD_C (n_col) + COLAMD_R (n_row) space is - required for the Col and Row arrays, respectively, which are internal to - colamd (roughly 6*n_col + 4*n_row). An additional n_col space is the - minimal amount of "elbow room", and nnz/5 more space is recommended for - run time efficiency. - - Alen is approximately 2.2*nnz + 7*n_col + 4*n_row + 10. - - This function is not needed when using symamd. -*/ - -/* add two values of type size_t, and check for integer overflow */ -static size_t t_add (size_t a, size_t b, int *ok) -{ - (*ok) = (*ok) && ((a + b) >= MAX (a,b)) ; - return ((*ok) ? (a + b) : 0) ; -} - -/* compute a*k where k is a small integer, and check for integer overflow */ -static size_t t_mult (size_t a, size_t k, int *ok) -{ - size_t i, s = 0 ; - for (i = 0 ; i < k ; i++) - { - s = t_add (s, a, ok) ; - } - return (s) ; -} - -/* size of the Col and Row structures */ -#define COLAMD_C(n_col,ok) \ - ((t_mult (t_add (n_col, 1, ok), sizeof (Colamd_Col), ok) / sizeof (Int))) - -#define COLAMD_R(n_row,ok) \ - ((t_mult (t_add (n_row, 1, ok), sizeof (Colamd_Row), ok) / sizeof (Int))) - - -PUBLIC size_t COLAMD_recommended /* returns recommended value of Alen. */ -( - /* === Parameters ======================================================= */ - - Int nnz, /* number of nonzeros in A */ - Int n_row, /* number of rows in A */ - Int n_col /* number of columns in A */ -) -{ - size_t s, c, r ; - int ok = TRUE ; - if (nnz < 0 || n_row < 0 || n_col < 0) - { - return (0) ; - } - s = t_mult (nnz, 2, &ok) ; /* 2*nnz */ - c = COLAMD_C (n_col, &ok) ; /* size of column structures */ - r = COLAMD_R (n_row, &ok) ; /* size of row structures */ - s = t_add (s, c, &ok) ; - s = t_add (s, r, &ok) ; - s = t_add (s, n_col, &ok) ; /* elbow room */ - s = t_add (s, nnz/5, &ok) ; /* elbow room */ - ok = ok && (s < Int_MAX) ; - return (ok ? s : 0) ; -} - - -/* ========================================================================== */ -/* === colamd_set_defaults ================================================== */ -/* ========================================================================== */ - -/* - The colamd_set_defaults routine sets the default values of the user- - controllable parameters for colamd and symamd: - - Colamd: rows with more than max (16, knobs [0] * sqrt (n_col)) - entries are removed prior to ordering. Columns with more than - max (16, knobs [1] * sqrt (MIN (n_row,n_col))) entries are removed - prior to ordering, and placed last in the output column ordering. - - Symamd: Rows and columns with more than max (16, knobs [0] * sqrt (n)) - entries are removed prior to ordering, and placed last in the - output ordering. - - knobs [0] dense row control - - knobs [1] dense column control - - knobs [2] if nonzero, do aggresive absorption - - knobs [3..19] unused, but future versions might use this - -*/ - -PUBLIC void COLAMD_set_defaults -( - /* === Parameters ======================================================= */ - - double knobs [COLAMD_KNOBS] /* knob array */ -) -{ - /* === Local variables ================================================== */ - - Int i ; - - if (!knobs) - { - return ; /* no knobs to initialize */ - } - for (i = 0 ; i < COLAMD_KNOBS ; i++) - { - knobs [i] = 0 ; - } - knobs [COLAMD_DENSE_ROW] = 10 ; - knobs [COLAMD_DENSE_COL] = 10 ; - knobs [COLAMD_AGGRESSIVE] = TRUE ; /* default: do aggressive absorption*/ -} - - -/* ========================================================================== */ -/* === symamd =============================================================== */ -/* ========================================================================== */ - -PUBLIC Int SYMAMD_MAIN /* return TRUE if OK, FALSE otherwise */ -( - /* === Parameters ======================================================= */ - - Int n, /* number of rows and columns of A */ - Int A [], /* row indices of A */ - Int p [], /* column pointers of A */ - Int perm [], /* output permutation, size n+1 */ - double knobs [COLAMD_KNOBS], /* parameters (uses defaults if NULL) */ - Int stats [COLAMD_STATS], /* output statistics and error codes */ - void * (*allocate) (size_t, size_t), - /* pointer to calloc (ANSI C) or */ - /* mxCalloc (for MATLAB mexFunction) */ - void (*release) (void *) - /* pointer to free (ANSI C) or */ - /* mxFree (for MATLAB mexFunction) */ -) -{ - /* === Local variables ================================================== */ - - Int *count ; /* length of each column of M, and col pointer*/ - Int *mark ; /* mark array for finding duplicate entries */ - Int *M ; /* row indices of matrix M */ - size_t Mlen ; /* length of M */ - Int n_row ; /* number of rows in M */ - Int nnz ; /* number of entries in A */ - Int i ; /* row index of A */ - Int j ; /* column index of A */ - Int k ; /* row index of M */ - Int mnz ; /* number of nonzeros in M */ - Int pp ; /* index into a column of A */ - Int last_row ; /* last row seen in the current column */ - Int length ; /* number of nonzeros in a column */ - - double cknobs [COLAMD_KNOBS] ; /* knobs for colamd */ - double default_knobs [COLAMD_KNOBS] ; /* default knobs for colamd */ - -#ifndef NDEBUG - colamd_get_debug ("symamd") ; -#endif /* NDEBUG */ - - /* === Check the input arguments ======================================== */ - - if (!stats) - { - DEBUG0 (("symamd: stats not present\n")) ; - return (FALSE) ; - } - for (i = 0 ; i < COLAMD_STATS ; i++) - { - stats [i] = 0 ; - } - stats [COLAMD_STATUS] = COLAMD_OK ; - stats [COLAMD_INFO1] = -1 ; - stats [COLAMD_INFO2] = -1 ; - - if (!A) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_A_not_present ; - DEBUG0 (("symamd: A not present\n")) ; - return (FALSE) ; - } - - if (!p) /* p is not present */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p_not_present ; - DEBUG0 (("symamd: p not present\n")) ; - return (FALSE) ; - } - - if (n < 0) /* n must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_ncol_negative ; - stats [COLAMD_INFO1] = n ; - DEBUG0 (("symamd: n negative %d\n", n)) ; - return (FALSE) ; - } - - nnz = p [n] ; - if (nnz < 0) /* nnz must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_nnz_negative ; - stats [COLAMD_INFO1] = nnz ; - DEBUG0 (("symamd: number of entries negative %d\n", nnz)) ; - return (FALSE) ; - } - - if (p [0] != 0) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p0_nonzero ; - stats [COLAMD_INFO1] = p [0] ; - DEBUG0 (("symamd: p[0] not zero %d\n", p [0])) ; - return (FALSE) ; - } - - /* === If no knobs, set default knobs =================================== */ - - if (!knobs) - { - COLAMD_set_defaults (default_knobs) ; - knobs = default_knobs ; - } - - /* === Allocate count and mark ========================================== */ - - count = (Int *) ((*allocate) (n+1, sizeof (Int))) ; - if (!count) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_out_of_memory ; - DEBUG0 (("symamd: allocate count (size %d) failed\n", n+1)) ; - return (FALSE) ; - } - - mark = (Int *) ((*allocate) (n+1, sizeof (Int))) ; - if (!mark) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_out_of_memory ; - (*release) ((void *) count) ; - DEBUG0 (("symamd: allocate mark (size %d) failed\n", n+1)) ; - return (FALSE) ; - } - - /* === Compute column counts of M, check if A is valid ================== */ - - stats [COLAMD_INFO3] = 0 ; /* number of duplicate or unsorted row indices*/ - - for (i = 0 ; i < n ; i++) - { - mark [i] = -1 ; - } - - for (j = 0 ; j < n ; j++) - { - last_row = -1 ; - - length = p [j+1] - p [j] ; - if (length < 0) - { - /* column pointers must be non-decreasing */ - stats [COLAMD_STATUS] = COLAMD_ERROR_col_length_negative ; - stats [COLAMD_INFO1] = j ; - stats [COLAMD_INFO2] = length ; - (*release) ((void *) count) ; - (*release) ((void *) mark) ; - DEBUG0 (("symamd: col %d negative length %d\n", j, length)) ; - return (FALSE) ; - } - - for (pp = p [j] ; pp < p [j+1] ; pp++) - { - i = A [pp] ; - if (i < 0 || i >= n) - { - /* row index i, in column j, is out of bounds */ - stats [COLAMD_STATUS] = COLAMD_ERROR_row_index_out_of_bounds ; - stats [COLAMD_INFO1] = j ; - stats [COLAMD_INFO2] = i ; - stats [COLAMD_INFO3] = n ; - (*release) ((void *) count) ; - (*release) ((void *) mark) ; - DEBUG0 (("symamd: row %d col %d out of bounds\n", i, j)) ; - return (FALSE) ; - } - - if (i <= last_row || mark [i] == j) - { - /* row index is unsorted or repeated (or both), thus col */ - /* is jumbled. This is a notice, not an error condition. */ - stats [COLAMD_STATUS] = COLAMD_OK_BUT_JUMBLED ; - stats [COLAMD_INFO1] = j ; - stats [COLAMD_INFO2] = i ; - (stats [COLAMD_INFO3]) ++ ; - DEBUG1 (("symamd: row %d col %d unsorted/duplicate\n", i, j)) ; - } - - if (i > j && mark [i] != j) - { - /* row k of M will contain column indices i and j */ - count [i]++ ; - count [j]++ ; - } - - /* mark the row as having been seen in this column */ - mark [i] = j ; - - last_row = i ; - } - } - - /* v2.4: removed free(mark) */ - - /* === Compute column pointers of M ===================================== */ - - /* use output permutation, perm, for column pointers of M */ - perm [0] = 0 ; - for (j = 1 ; j <= n ; j++) - { - perm [j] = perm [j-1] + count [j-1] ; - } - for (j = 0 ; j < n ; j++) - { - count [j] = perm [j] ; - } - - /* === Construct M ====================================================== */ - - mnz = perm [n] ; - n_row = mnz / 2 ; - Mlen = COLAMD_recommended (mnz, n_row, n) ; - M = (Int *) ((*allocate) (Mlen, sizeof (Int))) ; - DEBUG0 (("symamd: M is %d-by-%d with %d entries, Mlen = %g\n", - n_row, n, mnz, (double) Mlen)) ; - - if (!M) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_out_of_memory ; - (*release) ((void *) count) ; - (*release) ((void *) mark) ; - DEBUG0 (("symamd: allocate M (size %g) failed\n", (double) Mlen)) ; - return (FALSE) ; - } - - k = 0 ; - - if (stats [COLAMD_STATUS] == COLAMD_OK) - { - /* Matrix is OK */ - for (j = 0 ; j < n ; j++) - { - ASSERT (p [j+1] - p [j] >= 0) ; - for (pp = p [j] ; pp < p [j+1] ; pp++) - { - i = A [pp] ; - ASSERT (i >= 0 && i < n) ; - if (i > j) - { - /* row k of M contains column indices i and j */ - M [count [i]++] = k ; - M [count [j]++] = k ; - k++ ; - } - } - } - } - else - { - /* Matrix is jumbled. Do not add duplicates to M. Unsorted cols OK. */ - DEBUG0 (("symamd: Duplicates in A.\n")) ; - for (i = 0 ; i < n ; i++) - { - mark [i] = -1 ; - } - for (j = 0 ; j < n ; j++) - { - ASSERT (p [j+1] - p [j] >= 0) ; - for (pp = p [j] ; pp < p [j+1] ; pp++) - { - i = A [pp] ; - ASSERT (i >= 0 && i < n) ; - if (i > j && mark [i] != j) - { - /* row k of M contains column indices i and j */ - M [count [i]++] = k ; - M [count [j]++] = k ; - k++ ; - mark [i] = j ; - } - } - } - /* v2.4: free(mark) moved below */ - } - - /* count and mark no longer needed */ - (*release) ((void *) count) ; - (*release) ((void *) mark) ; /* v2.4: free (mark) moved here */ - ASSERT (k == n_row) ; - - /* === Adjust the knobs for M =========================================== */ - - for (i = 0 ; i < COLAMD_KNOBS ; i++) - { - cknobs [i] = knobs [i] ; - } - - /* there are no dense rows in M */ - cknobs [COLAMD_DENSE_ROW] = -1 ; - cknobs [COLAMD_DENSE_COL] = knobs [COLAMD_DENSE_ROW] ; - - /* === Order the columns of M =========================================== */ - - /* v2.4: colamd cannot fail here, so the error check is removed */ - (void) COLAMD_MAIN (n_row, n, (Int) Mlen, M, perm, cknobs, stats) ; - - /* Note that the output permutation is now in perm */ - - /* === get the statistics for symamd from colamd ======================== */ - - /* a dense column in colamd means a dense row and col in symamd */ - stats [COLAMD_DENSE_ROW] = stats [COLAMD_DENSE_COL] ; - - /* === Free M =========================================================== */ - - (*release) ((void *) M) ; - DEBUG0 (("symamd: done.\n")) ; - return (TRUE) ; - -} - -/* ========================================================================== */ -/* === colamd =============================================================== */ -/* ========================================================================== */ - -/* - The colamd routine computes a column ordering Q of a sparse matrix - A such that the LU factorization P(AQ) = LU remains sparse, where P is - selected via partial pivoting. The routine can also be viewed as - providing a permutation Q such that the Cholesky factorization - (AQ)'(AQ) = LL' remains sparse. -*/ - -PUBLIC Int COLAMD_MAIN /* returns TRUE if successful, FALSE otherwise*/ -( - /* === Parameters ======================================================= */ - - Int n_row, /* number of rows in A */ - Int n_col, /* number of columns in A */ - Int Alen, /* length of A */ - Int A [], /* row indices of A */ - Int p [], /* pointers to columns in A */ - double knobs [COLAMD_KNOBS],/* parameters (uses defaults if NULL) */ - Int stats [COLAMD_STATS] /* output statistics and error codes */ -) -{ - /* === Local variables ================================================== */ - - Int i ; /* loop index */ - Int nnz ; /* nonzeros in A */ - size_t Row_size ; /* size of Row [], in integers */ - size_t Col_size ; /* size of Col [], in integers */ - size_t need ; /* minimum required length of A */ - Colamd_Row *Row ; /* pointer into A of Row [0..n_row] array */ - Colamd_Col *Col ; /* pointer into A of Col [0..n_col] array */ - Int n_col2 ; /* number of non-dense, non-empty columns */ - Int n_row2 ; /* number of non-dense, non-empty rows */ - Int ngarbage ; /* number of garbage collections performed */ - Int max_deg ; /* maximum row degree */ - double default_knobs [COLAMD_KNOBS] ; /* default knobs array */ - Int aggressive ; /* do aggressive absorption */ - int ok ; - -#ifndef NDEBUG - colamd_get_debug ("colamd") ; -#endif /* NDEBUG */ - - /* === Check the input arguments ======================================== */ - - if (!stats) - { - DEBUG0 (("colamd: stats not present\n")) ; - return (FALSE) ; - } - for (i = 0 ; i < COLAMD_STATS ; i++) - { - stats [i] = 0 ; - } - stats [COLAMD_STATUS] = COLAMD_OK ; - stats [COLAMD_INFO1] = -1 ; - stats [COLAMD_INFO2] = -1 ; - - if (!A) /* A is not present */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_A_not_present ; - DEBUG0 (("colamd: A not present\n")) ; - return (FALSE) ; - } - - if (!p) /* p is not present */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p_not_present ; - DEBUG0 (("colamd: p not present\n")) ; - return (FALSE) ; - } - - if (n_row < 0) /* n_row must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_nrow_negative ; - stats [COLAMD_INFO1] = n_row ; - DEBUG0 (("colamd: nrow negative %d\n", n_row)) ; - return (FALSE) ; - } - - if (n_col < 0) /* n_col must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_ncol_negative ; - stats [COLAMD_INFO1] = n_col ; - DEBUG0 (("colamd: ncol negative %d\n", n_col)) ; - return (FALSE) ; - } - - nnz = p [n_col] ; - if (nnz < 0) /* nnz must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_nnz_negative ; - stats [COLAMD_INFO1] = nnz ; - DEBUG0 (("colamd: number of entries negative %d\n", nnz)) ; - return (FALSE) ; - } - - if (p [0] != 0) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p0_nonzero ; - stats [COLAMD_INFO1] = p [0] ; - DEBUG0 (("colamd: p[0] not zero %d\n", p [0])) ; - return (FALSE) ; - } - - /* === If no knobs, set default knobs =================================== */ - - if (!knobs) - { - COLAMD_set_defaults (default_knobs) ; - knobs = default_knobs ; - } - - aggressive = (knobs [COLAMD_AGGRESSIVE] != FALSE) ; - - /* === Allocate the Row and Col arrays from array A ===================== */ - - ok = TRUE ; - Col_size = COLAMD_C (n_col, &ok) ; /* size of Col array of structs */ - Row_size = COLAMD_R (n_row, &ok) ; /* size of Row array of structs */ - - /* need = 2*nnz + n_col + Col_size + Row_size ; */ - need = t_mult (nnz, 2, &ok) ; - need = t_add (need, n_col, &ok) ; - need = t_add (need, Col_size, &ok) ; - need = t_add (need, Row_size, &ok) ; - - if (!ok || need > (size_t) Alen || need > Int_MAX) - { - /* not enough space in array A to perform the ordering */ - stats [COLAMD_STATUS] = COLAMD_ERROR_A_too_small ; - stats [COLAMD_INFO1] = need ; - stats [COLAMD_INFO2] = Alen ; - DEBUG0 (("colamd: Need Alen >= %d, given only Alen = %d\n", need,Alen)); - return (FALSE) ; - } - - Alen -= Col_size + Row_size ; - Col = (Colamd_Col *) &A [Alen] ; - Row = (Colamd_Row *) &A [Alen + Col_size] ; - - /* === Construct the row and column data structures ===================== */ - - if (!init_rows_cols (n_row, n_col, Row, Col, A, p, stats)) - { - /* input matrix is invalid */ - DEBUG0 (("colamd: Matrix invalid\n")) ; - return (FALSE) ; - } - - /* === Initialize scores, kill dense rows/columns ======================= */ - - init_scoring (n_row, n_col, Row, Col, A, p, knobs, - &n_row2, &n_col2, &max_deg) ; - - /* === Order the supercolumns =========================================== */ - - ngarbage = find_ordering (n_row, n_col, Alen, Row, Col, A, p, - n_col2, max_deg, 2*nnz, aggressive) ; - - /* === Order the non-principal columns ================================== */ - - order_children (n_col, Col, p) ; - - /* === Return statistics in stats ======================================= */ - - stats [COLAMD_DENSE_ROW] = n_row - n_row2 ; - stats [COLAMD_DENSE_COL] = n_col - n_col2 ; - stats [COLAMD_DEFRAG_COUNT] = ngarbage ; - DEBUG0 (("colamd: done.\n")) ; - return (TRUE) ; -} - - -/* ========================================================================== */ -/* === colamd_report ======================================================== */ -/* ========================================================================== */ - -PUBLIC void COLAMD_report -( - Int stats [COLAMD_STATS] -) -{ - print_report ("colamd", stats) ; -} - - -/* ========================================================================== */ -/* === symamd_report ======================================================== */ -/* ========================================================================== */ - -PUBLIC void SYMAMD_report -( - Int stats [COLAMD_STATS] -) -{ - print_report ("symamd", stats) ; -} - - - -/* ========================================================================== */ -/* === NON-USER-CALLABLE ROUTINES: ========================================== */ -/* ========================================================================== */ - -/* There are no user-callable routines beyond this point in the file */ - - -/* ========================================================================== */ -/* === init_rows_cols ======================================================= */ -/* ========================================================================== */ - -/* - Takes the column form of the matrix in A and creates the row form of the - matrix. Also, row and column attributes are stored in the Col and Row - structs. If the columns are un-sorted or contain duplicate row indices, - this routine will also sort and remove duplicate row indices from the - column form of the matrix. Returns FALSE if the matrix is invalid, - TRUE otherwise. Not user-callable. -*/ - -PRIVATE Int init_rows_cols /* returns TRUE if OK, or FALSE otherwise */ -( - /* === Parameters ======================================================= */ - - Int n_row, /* number of rows of A */ - Int n_col, /* number of columns of A */ - Colamd_Row Row [], /* of size n_row+1 */ - Colamd_Col Col [], /* of size n_col+1 */ - Int A [], /* row indices of A, of size Alen */ - Int p [], /* pointers to columns in A, of size n_col+1 */ - Int stats [COLAMD_STATS] /* colamd statistics */ -) -{ - /* === Local variables ================================================== */ - - Int col ; /* a column index */ - Int row ; /* a row index */ - Int *cp ; /* a column pointer */ - Int *cp_end ; /* a pointer to the end of a column */ - Int *rp ; /* a row pointer */ - Int *rp_end ; /* a pointer to the end of a row */ - Int last_row ; /* previous row */ - - /* === Initialize columns, and check column pointers ==================== */ - - for (col = 0 ; col < n_col ; col++) - { - Col [col].start = p [col] ; - Col [col].length = p [col+1] - p [col] ; - - if (Col [col].length < 0) - { - /* column pointers must be non-decreasing */ - stats [COLAMD_STATUS] = COLAMD_ERROR_col_length_negative ; - stats [COLAMD_INFO1] = col ; - stats [COLAMD_INFO2] = Col [col].length ; - DEBUG0 (("colamd: col %d length %d < 0\n", col, Col [col].length)) ; - return (FALSE) ; - } - - Col [col].shared1.thickness = 1 ; - Col [col].shared2.score = 0 ; - Col [col].shared3.prev = EMPTY ; - Col [col].shared4.degree_next = EMPTY ; - } - - /* p [0..n_col] no longer needed, used as "head" in subsequent routines */ - - /* === Scan columns, compute row degrees, and check row indices ========= */ - - stats [COLAMD_INFO3] = 0 ; /* number of duplicate or unsorted row indices*/ - - for (row = 0 ; row < n_row ; row++) - { - Row [row].length = 0 ; - Row [row].shared2.mark = -1 ; - } - - for (col = 0 ; col < n_col ; col++) - { - last_row = -1 ; - - cp = &A [p [col]] ; - cp_end = &A [p [col+1]] ; - - while (cp < cp_end) - { - row = *cp++ ; - - /* make sure row indices within range */ - if (row < 0 || row >= n_row) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_row_index_out_of_bounds ; - stats [COLAMD_INFO1] = col ; - stats [COLAMD_INFO2] = row ; - stats [COLAMD_INFO3] = n_row ; - DEBUG0 (("colamd: row %d col %d out of bounds\n", row, col)) ; - return (FALSE) ; - } - - if (row <= last_row || Row [row].shared2.mark == col) - { - /* row index are unsorted or repeated (or both), thus col */ - /* is jumbled. This is a notice, not an error condition. */ - stats [COLAMD_STATUS] = COLAMD_OK_BUT_JUMBLED ; - stats [COLAMD_INFO1] = col ; - stats [COLAMD_INFO2] = row ; - (stats [COLAMD_INFO3]) ++ ; - DEBUG1 (("colamd: row %d col %d unsorted/duplicate\n",row,col)); - } - - if (Row [row].shared2.mark != col) - { - Row [row].length++ ; - } - else - { - /* this is a repeated entry in the column, */ - /* it will be removed */ - Col [col].length-- ; - } - - /* mark the row as having been seen in this column */ - Row [row].shared2.mark = col ; - - last_row = row ; - } - } - - /* === Compute row pointers ============================================= */ - - /* row form of the matrix starts directly after the column */ - /* form of matrix in A */ - Row [0].start = p [n_col] ; - Row [0].shared1.p = Row [0].start ; - Row [0].shared2.mark = -1 ; - for (row = 1 ; row < n_row ; row++) - { - Row [row].start = Row [row-1].start + Row [row-1].length ; - Row [row].shared1.p = Row [row].start ; - Row [row].shared2.mark = -1 ; - } - - /* === Create row form ================================================== */ - - if (stats [COLAMD_STATUS] == COLAMD_OK_BUT_JUMBLED) - { - /* if cols jumbled, watch for repeated row indices */ - for (col = 0 ; col < n_col ; col++) - { - cp = &A [p [col]] ; - cp_end = &A [p [col+1]] ; - while (cp < cp_end) - { - row = *cp++ ; - if (Row [row].shared2.mark != col) - { - A [(Row [row].shared1.p)++] = col ; - Row [row].shared2.mark = col ; - } - } - } - } - else - { - /* if cols not jumbled, we don't need the mark (this is faster) */ - for (col = 0 ; col < n_col ; col++) - { - cp = &A [p [col]] ; - cp_end = &A [p [col+1]] ; - while (cp < cp_end) - { - A [(Row [*cp++].shared1.p)++] = col ; - } - } - } - - /* === Clear the row marks and set row degrees ========================== */ - - for (row = 0 ; row < n_row ; row++) - { - Row [row].shared2.mark = 0 ; - Row [row].shared1.degree = Row [row].length ; - } - - /* === See if we need to re-create columns ============================== */ - - if (stats [COLAMD_STATUS] == COLAMD_OK_BUT_JUMBLED) - { - DEBUG0 (("colamd: reconstructing column form, matrix jumbled\n")) ; - -#ifndef NDEBUG - /* make sure column lengths are correct */ - for (col = 0 ; col < n_col ; col++) - { - p [col] = Col [col].length ; - } - for (row = 0 ; row < n_row ; row++) - { - rp = &A [Row [row].start] ; - rp_end = rp + Row [row].length ; - while (rp < rp_end) - { - p [*rp++]-- ; - } - } - for (col = 0 ; col < n_col ; col++) - { - ASSERT (p [col] == 0) ; - } - /* now p is all zero (different than when debugging is turned off) */ -#endif /* NDEBUG */ - - /* === Compute col pointers ========================================= */ - - /* col form of the matrix starts at A [0]. */ - /* Note, we may have a gap between the col form and the row */ - /* form if there were duplicate entries, if so, it will be */ - /* removed upon the first garbage collection */ - Col [0].start = 0 ; - p [0] = Col [0].start ; - for (col = 1 ; col < n_col ; col++) - { - /* note that the lengths here are for pruned columns, i.e. */ - /* no duplicate row indices will exist for these columns */ - Col [col].start = Col [col-1].start + Col [col-1].length ; - p [col] = Col [col].start ; - } - - /* === Re-create col form =========================================== */ - - for (row = 0 ; row < n_row ; row++) - { - rp = &A [Row [row].start] ; - rp_end = rp + Row [row].length ; - while (rp < rp_end) - { - A [(p [*rp++])++] = row ; - } - } - } - - /* === Done. Matrix is not (or no longer) jumbled ====================== */ - - return (TRUE) ; -} - - -/* ========================================================================== */ -/* === init_scoring ========================================================= */ -/* ========================================================================== */ - -/* - Kills dense or empty columns and rows, calculates an initial score for - each column, and places all columns in the degree lists. Not user-callable. -*/ - -PRIVATE void init_scoring -( - /* === Parameters ======================================================= */ - - Int n_row, /* number of rows of A */ - Int n_col, /* number of columns of A */ - Colamd_Row Row [], /* of size n_row+1 */ - Colamd_Col Col [], /* of size n_col+1 */ - Int A [], /* column form and row form of A */ - Int head [], /* of size n_col+1 */ - double knobs [COLAMD_KNOBS],/* parameters */ - Int *p_n_row2, /* number of non-dense, non-empty rows */ - Int *p_n_col2, /* number of non-dense, non-empty columns */ - Int *p_max_deg /* maximum row degree */ -) -{ - /* === Local variables ================================================== */ - - Int c ; /* a column index */ - Int r, row ; /* a row index */ - Int *cp ; /* a column pointer */ - Int deg ; /* degree of a row or column */ - Int *cp_end ; /* a pointer to the end of a column */ - Int *new_cp ; /* new column pointer */ - Int col_length ; /* length of pruned column */ - Int score ; /* current column score */ - Int n_col2 ; /* number of non-dense, non-empty columns */ - Int n_row2 ; /* number of non-dense, non-empty rows */ - Int dense_row_count ; /* remove rows with more entries than this */ - Int dense_col_count ; /* remove cols with more entries than this */ - Int min_score ; /* smallest column score */ - Int max_deg ; /* maximum row degree */ - Int next_col ; /* Used to add to degree list.*/ - -#ifndef NDEBUG - Int debug_count ; /* debug only. */ -#endif /* NDEBUG */ - - /* === Extract knobs ==================================================== */ - - /* Note: if knobs contains a NaN, this is undefined: */ - if (knobs [COLAMD_DENSE_ROW] < 0) - { - /* only remove completely dense rows */ - dense_row_count = n_col-1 ; - } - else - { - dense_row_count = DENSE_DEGREE (knobs [COLAMD_DENSE_ROW], n_col) ; - } - if (knobs [COLAMD_DENSE_COL] < 0) - { - /* only remove completely dense columns */ - dense_col_count = n_row-1 ; - } - else - { - dense_col_count = - DENSE_DEGREE (knobs [COLAMD_DENSE_COL], MIN (n_row, n_col)) ; - } - - DEBUG1 (("colamd: densecount: %d %d\n", dense_row_count, dense_col_count)) ; - max_deg = 0 ; - n_col2 = n_col ; - n_row2 = n_row ; - - /* === Kill empty columns =============================================== */ - - /* Put the empty columns at the end in their natural order, so that LU */ - /* factorization can proceed as far as possible. */ - for (c = n_col-1 ; c >= 0 ; c--) - { - deg = Col [c].length ; - if (deg == 0) - { - /* this is a empty column, kill and order it last */ - Col [c].shared2.order = --n_col2 ; - KILL_PRINCIPAL_COL (c) ; - } - } - DEBUG1 (("colamd: null columns killed: %d\n", n_col - n_col2)) ; - - /* === Kill dense columns =============================================== */ - - /* Put the dense columns at the end, in their natural order */ - for (c = n_col-1 ; c >= 0 ; c--) - { - /* skip any dead columns */ - if (COL_IS_DEAD (c)) - { - continue ; - } - deg = Col [c].length ; - if (deg > dense_col_count) - { - /* this is a dense column, kill and order it last */ - Col [c].shared2.order = --n_col2 ; - /* decrement the row degrees */ - cp = &A [Col [c].start] ; - cp_end = cp + Col [c].length ; - while (cp < cp_end) - { - Row [*cp++].shared1.degree-- ; - } - KILL_PRINCIPAL_COL (c) ; - } - } - DEBUG1 (("colamd: Dense and null columns killed: %d\n", n_col - n_col2)) ; - - /* === Kill dense and empty rows ======================================== */ - - for (r = 0 ; r < n_row ; r++) - { - deg = Row [r].shared1.degree ; - ASSERT (deg >= 0 && deg <= n_col) ; - if (deg > dense_row_count || deg == 0) - { - /* kill a dense or empty row */ - KILL_ROW (r) ; - --n_row2 ; - } - else - { - /* keep track of max degree of remaining rows */ - max_deg = MAX (max_deg, deg) ; - } - } - DEBUG1 (("colamd: Dense and null rows killed: %d\n", n_row - n_row2)) ; - - /* === Compute initial column scores ==================================== */ - - /* At this point the row degrees are accurate. They reflect the number */ - /* of "live" (non-dense) columns in each row. No empty rows exist. */ - /* Some "live" columns may contain only dead rows, however. These are */ - /* pruned in the code below. */ - - /* now find the initial matlab score for each column */ - for (c = n_col-1 ; c >= 0 ; c--) - { - /* skip dead column */ - if (COL_IS_DEAD (c)) - { - continue ; - } - score = 0 ; - cp = &A [Col [c].start] ; - new_cp = cp ; - cp_end = cp + Col [c].length ; - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - /* skip if dead */ - if (ROW_IS_DEAD (row)) - { - continue ; - } - /* compact the column */ - *new_cp++ = row ; - /* add row's external degree */ - score += Row [row].shared1.degree - 1 ; - /* guard against integer overflow */ - score = MIN (score, n_col) ; - } - /* determine pruned column length */ - col_length = (Int) (new_cp - &A [Col [c].start]) ; - if (col_length == 0) - { - /* a newly-made null column (all rows in this col are "dense" */ - /* and have already been killed) */ - DEBUG2 (("Newly null killed: %d\n", c)) ; - Col [c].shared2.order = --n_col2 ; - KILL_PRINCIPAL_COL (c) ; - } - else - { - /* set column length and set score */ - ASSERT (score >= 0) ; - ASSERT (score <= n_col) ; - Col [c].length = col_length ; - Col [c].shared2.score = score ; - } - } - DEBUG1 (("colamd: Dense, null, and newly-null columns killed: %d\n", - n_col-n_col2)) ; - - /* At this point, all empty rows and columns are dead. All live columns */ - /* are "clean" (containing no dead rows) and simplicial (no supercolumns */ - /* yet). Rows may contain dead columns, but all live rows contain at */ - /* least one live column. */ - -#ifndef NDEBUG - debug_structures (n_row, n_col, Row, Col, A, n_col2) ; -#endif /* NDEBUG */ - - /* === Initialize degree lists ========================================== */ - -#ifndef NDEBUG - debug_count = 0 ; -#endif /* NDEBUG */ - - /* clear the hash buckets */ - for (c = 0 ; c <= n_col ; c++) - { - head [c] = EMPTY ; - } - min_score = n_col ; - /* place in reverse order, so low column indices are at the front */ - /* of the lists. This is to encourage natural tie-breaking */ - for (c = n_col-1 ; c >= 0 ; c--) - { - /* only add principal columns to degree lists */ - if (COL_IS_ALIVE (c)) - { - DEBUG4 (("place %d score %d minscore %d ncol %d\n", - c, Col [c].shared2.score, min_score, n_col)) ; - - /* === Add columns score to DList =============================== */ - - score = Col [c].shared2.score ; - - ASSERT (min_score >= 0) ; - ASSERT (min_score <= n_col) ; - ASSERT (score >= 0) ; - ASSERT (score <= n_col) ; - ASSERT (head [score] >= EMPTY) ; - - /* now add this column to dList at proper score location */ - next_col = head [score] ; - Col [c].shared3.prev = EMPTY ; - Col [c].shared4.degree_next = next_col ; - - /* if there already was a column with the same score, set its */ - /* previous pointer to this new column */ - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = c ; - } - head [score] = c ; - - /* see if this score is less than current min */ - min_score = MIN (min_score, score) ; - -#ifndef NDEBUG - debug_count++ ; -#endif /* NDEBUG */ - - } - } - -#ifndef NDEBUG - DEBUG1 (("colamd: Live cols %d out of %d, non-princ: %d\n", - debug_count, n_col, n_col-debug_count)) ; - ASSERT (debug_count == n_col2) ; - debug_deg_lists (n_row, n_col, Row, Col, head, min_score, n_col2, max_deg) ; -#endif /* NDEBUG */ - - /* === Return number of remaining columns, and max row degree =========== */ - - *p_n_col2 = n_col2 ; - *p_n_row2 = n_row2 ; - *p_max_deg = max_deg ; -} - - -/* ========================================================================== */ -/* === find_ordering ======================================================== */ -/* ========================================================================== */ - -/* - Order the principal columns of the supercolumn form of the matrix - (no supercolumns on input). Uses a minimum approximate column minimum - degree ordering method. Not user-callable. -*/ - -PRIVATE Int find_ordering /* return the number of garbage collections */ -( - /* === Parameters ======================================================= */ - - Int n_row, /* number of rows of A */ - Int n_col, /* number of columns of A */ - Int Alen, /* size of A, 2*nnz + n_col or larger */ - Colamd_Row Row [], /* of size n_row+1 */ - Colamd_Col Col [], /* of size n_col+1 */ - Int A [], /* column form and row form of A */ - Int head [], /* of size n_col+1 */ - Int n_col2, /* Remaining columns to order */ - Int max_deg, /* Maximum row degree */ - Int pfree, /* index of first free slot (2*nnz on entry) */ - Int aggressive -) -{ - /* === Local variables ================================================== */ - - Int k ; /* current pivot ordering step */ - Int pivot_col ; /* current pivot column */ - Int *cp ; /* a column pointer */ - Int *rp ; /* a row pointer */ - Int pivot_row ; /* current pivot row */ - Int *new_cp ; /* modified column pointer */ - Int *new_rp ; /* modified row pointer */ - Int pivot_row_start ; /* pointer to start of pivot row */ - Int pivot_row_degree ; /* number of columns in pivot row */ - Int pivot_row_length ; /* number of supercolumns in pivot row */ - Int pivot_col_score ; /* score of pivot column */ - Int needed_memory ; /* free space needed for pivot row */ - Int *cp_end ; /* pointer to the end of a column */ - Int *rp_end ; /* pointer to the end of a row */ - Int row ; /* a row index */ - Int col ; /* a column index */ - Int max_score ; /* maximum possible score */ - Int cur_score ; /* score of current column */ - unsigned Int hash ; /* hash value for supernode detection */ - Int head_column ; /* head of hash bucket */ - Int first_col ; /* first column in hash bucket */ - Int tag_mark ; /* marker value for mark array */ - Int row_mark ; /* Row [row].shared2.mark */ - Int set_difference ; /* set difference size of row with pivot row */ - Int min_score ; /* smallest column score */ - Int col_thickness ; /* "thickness" (no. of columns in a supercol) */ - Int max_mark ; /* maximum value of tag_mark */ - Int pivot_col_thickness ; /* number of columns represented by pivot col */ - Int prev_col ; /* Used by Dlist operations. */ - Int next_col ; /* Used by Dlist operations. */ - Int ngarbage ; /* number of garbage collections performed */ - -#ifndef NDEBUG - Int debug_d ; /* debug loop counter */ - Int debug_step = 0 ; /* debug loop counter */ -#endif /* NDEBUG */ - - /* === Initialization and clear mark ==================================== */ - - max_mark = INT_MAX - n_col ; /* INT_MAX defined in */ - tag_mark = clear_mark (0, max_mark, n_row, Row) ; - min_score = 0 ; - ngarbage = 0 ; - DEBUG1 (("colamd: Ordering, n_col2=%d\n", n_col2)) ; - - /* === Order the columns ================================================ */ - - for (k = 0 ; k < n_col2 ; /* 'k' is incremented below */) - { - -#ifndef NDEBUG - if (debug_step % 100 == 0) - { - DEBUG2 (("\n... Step k: %d out of n_col2: %d\n", k, n_col2)) ; - } - else - { - DEBUG3 (("\n----------Step k: %d out of n_col2: %d\n", k, n_col2)) ; - } - debug_step++ ; - debug_deg_lists (n_row, n_col, Row, Col, head, - min_score, n_col2-k, max_deg) ; - debug_matrix (n_row, n_col, Row, Col, A) ; -#endif /* NDEBUG */ - - /* === Select pivot column, and order it ============================ */ - - /* make sure degree list isn't empty */ - ASSERT (min_score >= 0) ; - ASSERT (min_score <= n_col) ; - ASSERT (head [min_score] >= EMPTY) ; - -#ifndef NDEBUG - for (debug_d = 0 ; debug_d < min_score ; debug_d++) - { - ASSERT (head [debug_d] == EMPTY) ; - } -#endif /* NDEBUG */ - - /* get pivot column from head of minimum degree list */ - while (head [min_score] == EMPTY && min_score < n_col) - { - min_score++ ; - } - pivot_col = head [min_score] ; - ASSERT (pivot_col >= 0 && pivot_col <= n_col) ; - next_col = Col [pivot_col].shared4.degree_next ; - head [min_score] = next_col ; - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = EMPTY ; - } - - ASSERT (COL_IS_ALIVE (pivot_col)) ; - - /* remember score for defrag check */ - pivot_col_score = Col [pivot_col].shared2.score ; - - /* the pivot column is the kth column in the pivot order */ - Col [pivot_col].shared2.order = k ; - - /* increment order count by column thickness */ - pivot_col_thickness = Col [pivot_col].shared1.thickness ; - k += pivot_col_thickness ; - ASSERT (pivot_col_thickness > 0) ; - DEBUG3 (("Pivot col: %d thick %d\n", pivot_col, pivot_col_thickness)) ; - - /* === Garbage_collection, if necessary ============================= */ - - needed_memory = MIN (pivot_col_score, n_col - k) ; - if (pfree + needed_memory >= Alen) - { - pfree = garbage_collection (n_row, n_col, Row, Col, A, &A [pfree]) ; - ngarbage++ ; - /* after garbage collection we will have enough */ - ASSERT (pfree + needed_memory < Alen) ; - /* garbage collection has wiped out the Row[].shared2.mark array */ - tag_mark = clear_mark (0, max_mark, n_row, Row) ; - -#ifndef NDEBUG - debug_matrix (n_row, n_col, Row, Col, A) ; -#endif /* NDEBUG */ - } - - /* === Compute pivot row pattern ==================================== */ - - /* get starting location for this new merged row */ - pivot_row_start = pfree ; - - /* initialize new row counts to zero */ - pivot_row_degree = 0 ; - - /* tag pivot column as having been visited so it isn't included */ - /* in merged pivot row */ - Col [pivot_col].shared1.thickness = -pivot_col_thickness ; - - /* pivot row is the union of all rows in the pivot column pattern */ - cp = &A [Col [pivot_col].start] ; - cp_end = cp + Col [pivot_col].length ; - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - DEBUG4 (("Pivot col pattern %d %d\n", ROW_IS_ALIVE (row), row)) ; - /* skip if row is dead */ - if (ROW_IS_ALIVE (row)) - { - rp = &A [Row [row].start] ; - rp_end = rp + Row [row].length ; - while (rp < rp_end) - { - /* get a column */ - col = *rp++ ; - /* add the column, if alive and untagged */ - col_thickness = Col [col].shared1.thickness ; - if (col_thickness > 0 && COL_IS_ALIVE (col)) - { - /* tag column in pivot row */ - Col [col].shared1.thickness = -col_thickness ; - ASSERT (pfree < Alen) ; - /* place column in pivot row */ - A [pfree++] = col ; - pivot_row_degree += col_thickness ; - } - } - } - } - - /* clear tag on pivot column */ - Col [pivot_col].shared1.thickness = pivot_col_thickness ; - max_deg = MAX (max_deg, pivot_row_degree) ; - -#ifndef NDEBUG - DEBUG3 (("check2\n")) ; - debug_mark (n_row, Row, tag_mark, max_mark) ; -#endif /* NDEBUG */ - - /* === Kill all rows used to construct pivot row ==================== */ - - /* also kill pivot row, temporarily */ - cp = &A [Col [pivot_col].start] ; - cp_end = cp + Col [pivot_col].length ; - while (cp < cp_end) - { - /* may be killing an already dead row */ - row = *cp++ ; - DEBUG3 (("Kill row in pivot col: %d\n", row)) ; - KILL_ROW (row) ; - } - - /* === Select a row index to use as the new pivot row =============== */ - - pivot_row_length = pfree - pivot_row_start ; - if (pivot_row_length > 0) - { - /* pick the "pivot" row arbitrarily (first row in col) */ - pivot_row = A [Col [pivot_col].start] ; - DEBUG3 (("Pivotal row is %d\n", pivot_row)) ; - } - else - { - /* there is no pivot row, since it is of zero length */ - pivot_row = EMPTY ; - ASSERT (pivot_row_length == 0) ; - } - ASSERT (Col [pivot_col].length > 0 || pivot_row_length == 0) ; - - /* === Approximate degree computation =============================== */ - - /* Here begins the computation of the approximate degree. The column */ - /* score is the sum of the pivot row "length", plus the size of the */ - /* set differences of each row in the column minus the pattern of the */ - /* pivot row itself. The column ("thickness") itself is also */ - /* excluded from the column score (we thus use an approximate */ - /* external degree). */ - - /* The time taken by the following code (compute set differences, and */ - /* add them up) is proportional to the size of the data structure */ - /* being scanned - that is, the sum of the sizes of each column in */ - /* the pivot row. Thus, the amortized time to compute a column score */ - /* is proportional to the size of that column (where size, in this */ - /* context, is the column "length", or the number of row indices */ - /* in that column). The number of row indices in a column is */ - /* monotonically non-decreasing, from the length of the original */ - /* column on input to colamd. */ - - /* === Compute set differences ====================================== */ - - DEBUG3 (("** Computing set differences phase. **\n")) ; - - /* pivot row is currently dead - it will be revived later. */ - - DEBUG3 (("Pivot row: ")) ; - /* for each column in pivot row */ - rp = &A [pivot_row_start] ; - rp_end = rp + pivot_row_length ; - while (rp < rp_end) - { - col = *rp++ ; - ASSERT (COL_IS_ALIVE (col) && col != pivot_col) ; - DEBUG3 (("Col: %d\n", col)) ; - - /* clear tags used to construct pivot row pattern */ - col_thickness = -Col [col].shared1.thickness ; - ASSERT (col_thickness > 0) ; - Col [col].shared1.thickness = col_thickness ; - - /* === Remove column from degree list =========================== */ - - cur_score = Col [col].shared2.score ; - prev_col = Col [col].shared3.prev ; - next_col = Col [col].shared4.degree_next ; - ASSERT (cur_score >= 0) ; - ASSERT (cur_score <= n_col) ; - ASSERT (cur_score >= EMPTY) ; - if (prev_col == EMPTY) - { - head [cur_score] = next_col ; - } - else - { - Col [prev_col].shared4.degree_next = next_col ; - } - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = prev_col ; - } - - /* === Scan the column ========================================== */ - - cp = &A [Col [col].start] ; - cp_end = cp + Col [col].length ; - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - row_mark = Row [row].shared2.mark ; - /* skip if dead */ - if (ROW_IS_MARKED_DEAD (row_mark)) - { - continue ; - } - ASSERT (row != pivot_row) ; - set_difference = row_mark - tag_mark ; - /* check if the row has been seen yet */ - if (set_difference < 0) - { - ASSERT (Row [row].shared1.degree <= max_deg) ; - set_difference = Row [row].shared1.degree ; - } - /* subtract column thickness from this row's set difference */ - set_difference -= col_thickness ; - ASSERT (set_difference >= 0) ; - /* absorb this row if the set difference becomes zero */ - if (set_difference == 0 && aggressive) - { - DEBUG3 (("aggressive absorption. Row: %d\n", row)) ; - KILL_ROW (row) ; - } - else - { - /* save the new mark */ - Row [row].shared2.mark = set_difference + tag_mark ; - } - } - } - -#ifndef NDEBUG - debug_deg_lists (n_row, n_col, Row, Col, head, - min_score, n_col2-k-pivot_row_degree, max_deg) ; -#endif /* NDEBUG */ - - /* === Add up set differences for each column ======================= */ - - DEBUG3 (("** Adding set differences phase. **\n")) ; - - /* for each column in pivot row */ - rp = &A [pivot_row_start] ; - rp_end = rp + pivot_row_length ; - while (rp < rp_end) - { - /* get a column */ - col = *rp++ ; - ASSERT (COL_IS_ALIVE (col) && col != pivot_col) ; - hash = 0 ; - cur_score = 0 ; - cp = &A [Col [col].start] ; - /* compact the column */ - new_cp = cp ; - cp_end = cp + Col [col].length ; - - DEBUG4 (("Adding set diffs for Col: %d.\n", col)) ; - - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - ASSERT(row >= 0 && row < n_row) ; - row_mark = Row [row].shared2.mark ; - /* skip if dead */ - if (ROW_IS_MARKED_DEAD (row_mark)) - { - DEBUG4 ((" Row %d, dead\n", row)) ; - continue ; - } - DEBUG4 ((" Row %d, set diff %d\n", row, row_mark-tag_mark)); - ASSERT (row_mark >= tag_mark) ; - /* compact the column */ - *new_cp++ = row ; - /* compute hash function */ - hash += row ; - /* add set difference */ - cur_score += row_mark - tag_mark ; - /* integer overflow... */ - cur_score = MIN (cur_score, n_col) ; - } - - /* recompute the column's length */ - Col [col].length = (Int) (new_cp - &A [Col [col].start]) ; - - /* === Further mass elimination ================================= */ - - if (Col [col].length == 0) - { - DEBUG4 (("further mass elimination. Col: %d\n", col)) ; - /* nothing left but the pivot row in this column */ - KILL_PRINCIPAL_COL (col) ; - pivot_row_degree -= Col [col].shared1.thickness ; - ASSERT (pivot_row_degree >= 0) ; - /* order it */ - Col [col].shared2.order = k ; - /* increment order count by column thickness */ - k += Col [col].shared1.thickness ; - } - else - { - /* === Prepare for supercolumn detection ==================== */ - - DEBUG4 (("Preparing supercol detection for Col: %d.\n", col)) ; - - /* save score so far */ - Col [col].shared2.score = cur_score ; - - /* add column to hash table, for supercolumn detection */ - hash %= n_col + 1 ; - - DEBUG4 ((" Hash = %d, n_col = %d.\n", hash, n_col)) ; - ASSERT (((Int) hash) <= n_col) ; - - head_column = head [hash] ; - if (head_column > EMPTY) - { - /* degree list "hash" is non-empty, use prev (shared3) of */ - /* first column in degree list as head of hash bucket */ - first_col = Col [head_column].shared3.headhash ; - Col [head_column].shared3.headhash = col ; - } - else - { - /* degree list "hash" is empty, use head as hash bucket */ - first_col = - (head_column + 2) ; - head [hash] = - (col + 2) ; - } - Col [col].shared4.hash_next = first_col ; - - /* save hash function in Col [col].shared3.hash */ - Col [col].shared3.hash = (Int) hash ; - ASSERT (COL_IS_ALIVE (col)) ; - } - } - - /* The approximate external column degree is now computed. */ - - /* === Supercolumn detection ======================================== */ - - DEBUG3 (("** Supercolumn detection phase. **\n")) ; - - detect_super_cols ( - -#ifndef NDEBUG - n_col, Row, -#endif /* NDEBUG */ - - Col, A, head, pivot_row_start, pivot_row_length) ; - - /* === Kill the pivotal column ====================================== */ - - KILL_PRINCIPAL_COL (pivot_col) ; - - /* === Clear mark =================================================== */ - - tag_mark = clear_mark (tag_mark+max_deg+1, max_mark, n_row, Row) ; - -#ifndef NDEBUG - DEBUG3 (("check3\n")) ; - debug_mark (n_row, Row, tag_mark, max_mark) ; -#endif /* NDEBUG */ - - /* === Finalize the new pivot row, and column scores ================ */ - - DEBUG3 (("** Finalize scores phase. **\n")) ; - - /* for each column in pivot row */ - rp = &A [pivot_row_start] ; - /* compact the pivot row */ - new_rp = rp ; - rp_end = rp + pivot_row_length ; - while (rp < rp_end) - { - col = *rp++ ; - /* skip dead columns */ - if (COL_IS_DEAD (col)) - { - continue ; - } - *new_rp++ = col ; - /* add new pivot row to column */ - A [Col [col].start + (Col [col].length++)] = pivot_row ; - - /* retrieve score so far and add on pivot row's degree. */ - /* (we wait until here for this in case the pivot */ - /* row's degree was reduced due to mass elimination). */ - cur_score = Col [col].shared2.score + pivot_row_degree ; - - /* calculate the max possible score as the number of */ - /* external columns minus the 'k' value minus the */ - /* columns thickness */ - max_score = n_col - k - Col [col].shared1.thickness ; - - /* make the score the external degree of the union-of-rows */ - cur_score -= Col [col].shared1.thickness ; - - /* make sure score is less or equal than the max score */ - cur_score = MIN (cur_score, max_score) ; - ASSERT (cur_score >= 0) ; - - /* store updated score */ - Col [col].shared2.score = cur_score ; - - /* === Place column back in degree list ========================= */ - - ASSERT (min_score >= 0) ; - ASSERT (min_score <= n_col) ; - ASSERT (cur_score >= 0) ; - ASSERT (cur_score <= n_col) ; - ASSERT (head [cur_score] >= EMPTY) ; - next_col = head [cur_score] ; - Col [col].shared4.degree_next = next_col ; - Col [col].shared3.prev = EMPTY ; - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = col ; - } - head [cur_score] = col ; - - /* see if this score is less than current min */ - min_score = MIN (min_score, cur_score) ; - - } - -#ifndef NDEBUG - debug_deg_lists (n_row, n_col, Row, Col, head, - min_score, n_col2-k, max_deg) ; -#endif /* NDEBUG */ - - /* === Resurrect the new pivot row ================================== */ - - if (pivot_row_degree > 0) - { - /* update pivot row length to reflect any cols that were killed */ - /* during super-col detection and mass elimination */ - Row [pivot_row].start = pivot_row_start ; - Row [pivot_row].length = (Int) (new_rp - &A[pivot_row_start]) ; - ASSERT (Row [pivot_row].length > 0) ; - Row [pivot_row].shared1.degree = pivot_row_degree ; - Row [pivot_row].shared2.mark = 0 ; - /* pivot row is no longer dead */ - - DEBUG1 (("Resurrect Pivot_row %d deg: %d\n", - pivot_row, pivot_row_degree)) ; - } - } - - /* === All principal columns have now been ordered ====================== */ - - return (ngarbage) ; -} - - -/* ========================================================================== */ -/* === order_children ======================================================= */ -/* ========================================================================== */ - -/* - The find_ordering routine has ordered all of the principal columns (the - representatives of the supercolumns). The non-principal columns have not - yet been ordered. This routine orders those columns by walking up the - parent tree (a column is a child of the column which absorbed it). The - final permutation vector is then placed in p [0 ... n_col-1], with p [0] - being the first column, and p [n_col-1] being the last. It doesn't look - like it at first glance, but be assured that this routine takes time linear - in the number of columns. Although not immediately obvious, the time - taken by this routine is O (n_col), that is, linear in the number of - columns. Not user-callable. -*/ - -PRIVATE void order_children -( - /* === Parameters ======================================================= */ - - Int n_col, /* number of columns of A */ - Colamd_Col Col [], /* of size n_col+1 */ - Int p [] /* p [0 ... n_col-1] is the column permutation*/ -) -{ - /* === Local variables ================================================== */ - - Int i ; /* loop counter for all columns */ - Int c ; /* column index */ - Int parent ; /* index of column's parent */ - Int order ; /* column's order */ - - /* === Order each non-principal column ================================== */ - - for (i = 0 ; i < n_col ; i++) - { - /* find an un-ordered non-principal column */ - ASSERT (COL_IS_DEAD (i)) ; - if (!COL_IS_DEAD_PRINCIPAL (i) && Col [i].shared2.order == EMPTY) - { - parent = i ; - /* once found, find its principal parent */ - do - { - parent = Col [parent].shared1.parent ; - } while (!COL_IS_DEAD_PRINCIPAL (parent)) ; - - /* now, order all un-ordered non-principal columns along path */ - /* to this parent. collapse tree at the same time */ - c = i ; - /* get order of parent */ - order = Col [parent].shared2.order ; - - do - { - ASSERT (Col [c].shared2.order == EMPTY) ; - - /* order this column */ - Col [c].shared2.order = order++ ; - /* collaps tree */ - Col [c].shared1.parent = parent ; - - /* get immediate parent of this column */ - c = Col [c].shared1.parent ; - - /* continue until we hit an ordered column. There are */ - /* guarranteed not to be anymore unordered columns */ - /* above an ordered column */ - } while (Col [c].shared2.order == EMPTY) ; - - /* re-order the super_col parent to largest order for this group */ - Col [parent].shared2.order = order ; - } - } - - /* === Generate the permutation ========================================= */ - - for (c = 0 ; c < n_col ; c++) - { - p [Col [c].shared2.order] = c ; - } -} - - -/* ========================================================================== */ -/* === detect_super_cols ==================================================== */ -/* ========================================================================== */ - -/* - Detects supercolumns by finding matches between columns in the hash buckets. - Check amongst columns in the set A [row_start ... row_start + row_length-1]. - The columns under consideration are currently *not* in the degree lists, - and have already been placed in the hash buckets. - - The hash bucket for columns whose hash function is equal to h is stored - as follows: - - if head [h] is >= 0, then head [h] contains a degree list, so: - - head [h] is the first column in degree bucket h. - Col [head [h]].headhash gives the first column in hash bucket h. - - otherwise, the degree list is empty, and: - - -(head [h] + 2) is the first column in hash bucket h. - - For a column c in a hash bucket, Col [c].shared3.prev is NOT a "previous - column" pointer. Col [c].shared3.hash is used instead as the hash number - for that column. The value of Col [c].shared4.hash_next is the next column - in the same hash bucket. - - Assuming no, or "few" hash collisions, the time taken by this routine is - linear in the sum of the sizes (lengths) of each column whose score has - just been computed in the approximate degree computation. - Not user-callable. -*/ - -PRIVATE void detect_super_cols -( - /* === Parameters ======================================================= */ - -#ifndef NDEBUG - /* these two parameters are only needed when debugging is enabled: */ - Int n_col, /* number of columns of A */ - Colamd_Row Row [], /* of size n_row+1 */ -#endif /* NDEBUG */ - - Colamd_Col Col [], /* of size n_col+1 */ - Int A [], /* row indices of A */ - Int head [], /* head of degree lists and hash buckets */ - Int row_start, /* pointer to set of columns to check */ - Int row_length /* number of columns to check */ -) -{ - /* === Local variables ================================================== */ - - Int hash ; /* hash value for a column */ - Int *rp ; /* pointer to a row */ - Int c ; /* a column index */ - Int super_c ; /* column index of the column to absorb into */ - Int *cp1 ; /* column pointer for column super_c */ - Int *cp2 ; /* column pointer for column c */ - Int length ; /* length of column super_c */ - Int prev_c ; /* column preceding c in hash bucket */ - Int i ; /* loop counter */ - Int *rp_end ; /* pointer to the end of the row */ - Int col ; /* a column index in the row to check */ - Int head_column ; /* first column in hash bucket or degree list */ - Int first_col ; /* first column in hash bucket */ - - /* === Consider each column in the row ================================== */ - - rp = &A [row_start] ; - rp_end = rp + row_length ; - while (rp < rp_end) - { - col = *rp++ ; - if (COL_IS_DEAD (col)) - { - continue ; - } - - /* get hash number for this column */ - hash = Col [col].shared3.hash ; - ASSERT (hash <= n_col) ; - - /* === Get the first column in this hash bucket ===================== */ - - head_column = head [hash] ; - if (head_column > EMPTY) - { - first_col = Col [head_column].shared3.headhash ; - } - else - { - first_col = - (head_column + 2) ; - } - - /* === Consider each column in the hash bucket ====================== */ - - for (super_c = first_col ; super_c != EMPTY ; - super_c = Col [super_c].shared4.hash_next) - { - ASSERT (COL_IS_ALIVE (super_c)) ; - ASSERT (Col [super_c].shared3.hash == hash) ; - length = Col [super_c].length ; - - /* prev_c is the column preceding column c in the hash bucket */ - prev_c = super_c ; - - /* === Compare super_c with all columns after it ================ */ - - for (c = Col [super_c].shared4.hash_next ; - c != EMPTY ; c = Col [c].shared4.hash_next) - { - ASSERT (c != super_c) ; - ASSERT (COL_IS_ALIVE (c)) ; - ASSERT (Col [c].shared3.hash == hash) ; - - /* not identical if lengths or scores are different */ - if (Col [c].length != length || - Col [c].shared2.score != Col [super_c].shared2.score) - { - prev_c = c ; - continue ; - } - - /* compare the two columns */ - cp1 = &A [Col [super_c].start] ; - cp2 = &A [Col [c].start] ; - - for (i = 0 ; i < length ; i++) - { - /* the columns are "clean" (no dead rows) */ - ASSERT (ROW_IS_ALIVE (*cp1)) ; - ASSERT (ROW_IS_ALIVE (*cp2)) ; - /* row indices will same order for both supercols, */ - /* no gather scatter nessasary */ - if (*cp1++ != *cp2++) - { - break ; - } - } - - /* the two columns are different if the for-loop "broke" */ - if (i != length) - { - prev_c = c ; - continue ; - } - - /* === Got it! two columns are identical =================== */ - - ASSERT (Col [c].shared2.score == Col [super_c].shared2.score) ; - - Col [super_c].shared1.thickness += Col [c].shared1.thickness ; - Col [c].shared1.parent = super_c ; - KILL_NON_PRINCIPAL_COL (c) ; - /* order c later, in order_children() */ - Col [c].shared2.order = EMPTY ; - /* remove c from hash bucket */ - Col [prev_c].shared4.hash_next = Col [c].shared4.hash_next ; - } - } - - /* === Empty this hash bucket ======================================= */ - - if (head_column > EMPTY) - { - /* corresponding degree list "hash" is not empty */ - Col [head_column].shared3.headhash = EMPTY ; - } - else - { - /* corresponding degree list "hash" is empty */ - head [hash] = EMPTY ; - } - } -} - - -/* ========================================================================== */ -/* === garbage_collection =================================================== */ -/* ========================================================================== */ - -/* - Defragments and compacts columns and rows in the workspace A. Used when - all avaliable memory has been used while performing row merging. Returns - the index of the first free position in A, after garbage collection. The - time taken by this routine is linear is the size of the array A, which is - itself linear in the number of nonzeros in the input matrix. - Not user-callable. -*/ - -PRIVATE Int garbage_collection /* returns the new value of pfree */ -( - /* === Parameters ======================================================= */ - - Int n_row, /* number of rows */ - Int n_col, /* number of columns */ - Colamd_Row Row [], /* row info */ - Colamd_Col Col [], /* column info */ - Int A [], /* A [0 ... Alen-1] holds the matrix */ - Int *pfree /* &A [0] ... pfree is in use */ -) -{ - /* === Local variables ================================================== */ - - Int *psrc ; /* source pointer */ - Int *pdest ; /* destination pointer */ - Int j ; /* counter */ - Int r ; /* a row index */ - Int c ; /* a column index */ - Int length ; /* length of a row or column */ - -#ifndef NDEBUG - Int debug_rows ; - DEBUG2 (("Defrag..\n")) ; - for (psrc = &A[0] ; psrc < pfree ; psrc++) ASSERT (*psrc >= 0) ; - debug_rows = 0 ; -#endif /* NDEBUG */ - - /* === Defragment the columns =========================================== */ - - pdest = &A[0] ; - for (c = 0 ; c < n_col ; c++) - { - if (COL_IS_ALIVE (c)) - { - psrc = &A [Col [c].start] ; - - /* move and compact the column */ - ASSERT (pdest <= psrc) ; - Col [c].start = (Int) (pdest - &A [0]) ; - length = Col [c].length ; - for (j = 0 ; j < length ; j++) - { - r = *psrc++ ; - if (ROW_IS_ALIVE (r)) - { - *pdest++ = r ; - } - } - Col [c].length = (Int) (pdest - &A [Col [c].start]) ; - } - } - - /* === Prepare to defragment the rows =================================== */ - - for (r = 0 ; r < n_row ; r++) - { - if (ROW_IS_DEAD (r) || (Row [r].length == 0)) - { - /* This row is already dead, or is of zero length. Cannot compact - * a row of zero length, so kill it. NOTE: in the current version, - * there are no zero-length live rows. Kill the row (for the first - * time, or again) just to be safe. */ - KILL_ROW (r) ; - } - else - { - /* save first column index in Row [r].shared2.first_column */ - psrc = &A [Row [r].start] ; - Row [r].shared2.first_column = *psrc ; - ASSERT (ROW_IS_ALIVE (r)) ; - /* flag the start of the row with the one's complement of row */ - *psrc = ONES_COMPLEMENT (r) ; -#ifndef NDEBUG - debug_rows++ ; -#endif /* NDEBUG */ - } - } - - /* === Defragment the rows ============================================== */ - - psrc = pdest ; - while (psrc < pfree) - { - /* find a negative number ... the start of a row */ - if (*psrc++ < 0) - { - psrc-- ; - /* get the row index */ - r = ONES_COMPLEMENT (*psrc) ; - ASSERT (r >= 0 && r < n_row) ; - /* restore first column index */ - *psrc = Row [r].shared2.first_column ; - ASSERT (ROW_IS_ALIVE (r)) ; - ASSERT (Row [r].length > 0) ; - /* move and compact the row */ - ASSERT (pdest <= psrc) ; - Row [r].start = (Int) (pdest - &A [0]) ; - length = Row [r].length ; - for (j = 0 ; j < length ; j++) - { - c = *psrc++ ; - if (COL_IS_ALIVE (c)) - { - *pdest++ = c ; - } - } - Row [r].length = (Int) (pdest - &A [Row [r].start]) ; - ASSERT (Row [r].length > 0) ; -#ifndef NDEBUG - debug_rows-- ; -#endif /* NDEBUG */ - } - } - /* ensure we found all the rows */ - ASSERT (debug_rows == 0) ; - - /* === Return the new value of pfree ==================================== */ - - return ((Int) (pdest - &A [0])) ; -} - - -/* ========================================================================== */ -/* === clear_mark =========================================================== */ -/* ========================================================================== */ - -/* - Clears the Row [].shared2.mark array, and returns the new tag_mark. - Return value is the new tag_mark. Not user-callable. -*/ - -PRIVATE Int clear_mark /* return the new value for tag_mark */ -( - /* === Parameters ======================================================= */ - - Int tag_mark, /* new value of tag_mark */ - Int max_mark, /* max allowed value of tag_mark */ - - Int n_row, /* number of rows in A */ - Colamd_Row Row [] /* Row [0 ... n_row-1].shared2.mark is set to zero */ -) -{ - /* === Local variables ================================================== */ - - Int r ; - - if (tag_mark <= 0 || tag_mark >= max_mark) - { - for (r = 0 ; r < n_row ; r++) - { - if (ROW_IS_ALIVE (r)) - { - Row [r].shared2.mark = 0 ; - } - } - tag_mark = 1 ; - } - - return (tag_mark) ; -} - - -/* ========================================================================== */ -/* === print_report ========================================================= */ -/* ========================================================================== */ - -PRIVATE void print_report -( - char *method, - Int stats [COLAMD_STATS] -) -{ - - Int i1, i2, i3 ; - - PRINTF (("\n%s version %d.%d, %s: ", method, - COLAMD_MAIN_VERSION, COLAMD_SUB_VERSION, COLAMD_DATE)) ; - - if (!stats) - { - PRINTF (("No statistics available.\n")) ; - return ; - } - - i1 = stats [COLAMD_INFO1] ; - i2 = stats [COLAMD_INFO2] ; - i3 = stats [COLAMD_INFO3] ; - - if (stats [COLAMD_STATUS] >= 0) - { - PRINTF (("OK. ")) ; - } - else - { - PRINTF (("ERROR. ")) ; - } - - switch (stats [COLAMD_STATUS]) - { - - case COLAMD_OK_BUT_JUMBLED: - - PRINTF(("Matrix has unsorted or duplicate row indices.\n")) ; - - PRINTF(("%s: number of duplicate or out-of-order row indices: %d\n", - method, i3)) ; - - PRINTF(("%s: last seen duplicate or out-of-order row index: %d\n", - method, INDEX (i2))) ; - - PRINTF(("%s: last seen in column: %d", - method, INDEX (i1))) ; - - /* no break - fall through to next case instead */ - - case COLAMD_OK: - - PRINTF(("\n")) ; - - PRINTF(("%s: number of dense or empty rows ignored: %d\n", - method, stats [COLAMD_DENSE_ROW])) ; - - PRINTF(("%s: number of dense or empty columns ignored: %d\n", - method, stats [COLAMD_DENSE_COL])) ; - - PRINTF(("%s: number of garbage collections performed: %d\n", - method, stats [COLAMD_DEFRAG_COUNT])) ; - break ; - - case COLAMD_ERROR_A_not_present: - - PRINTF(("Array A (row indices of matrix) not present.\n")) ; - break ; - - case COLAMD_ERROR_p_not_present: - - PRINTF(("Array p (column pointers for matrix) not present.\n")) ; - break ; - - case COLAMD_ERROR_nrow_negative: - - PRINTF(("Invalid number of rows (%d).\n", i1)) ; - break ; - - case COLAMD_ERROR_ncol_negative: - - PRINTF(("Invalid number of columns (%d).\n", i1)) ; - break ; - - case COLAMD_ERROR_nnz_negative: - - PRINTF(("Invalid number of nonzero entries (%d).\n", i1)) ; - break ; - - case COLAMD_ERROR_p0_nonzero: - - PRINTF(("Invalid column pointer, p [0] = %d, must be zero.\n", i1)); - break ; - - case COLAMD_ERROR_A_too_small: - - PRINTF(("Array A too small.\n")) ; - PRINTF((" Need Alen >= %d, but given only Alen = %d.\n", - i1, i2)) ; - break ; - - case COLAMD_ERROR_col_length_negative: - - PRINTF - (("Column %d has a negative number of nonzero entries (%d).\n", - INDEX (i1), i2)) ; - break ; - - case COLAMD_ERROR_row_index_out_of_bounds: - - PRINTF - (("Row index (row %d) out of bounds (%d to %d) in column %d.\n", - INDEX (i2), INDEX (0), INDEX (i3-1), INDEX (i1))) ; - break ; - - case COLAMD_ERROR_out_of_memory: - - PRINTF(("Out of memory.\n")) ; - break ; - - /* v2.4: internal-error case deleted */ - } -} - - - - -/* ========================================================================== */ -/* === colamd debugging routines ============================================ */ -/* ========================================================================== */ - -/* When debugging is disabled, the remainder of this file is ignored. */ - -#ifndef NDEBUG - - -/* ========================================================================== */ -/* === debug_structures ===================================================== */ -/* ========================================================================== */ - -/* - At this point, all empty rows and columns are dead. All live columns - are "clean" (containing no dead rows) and simplicial (no supercolumns - yet). Rows may contain dead columns, but all live rows contain at - least one live column. -*/ - -PRIVATE void debug_structures -( - /* === Parameters ======================================================= */ - - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [], - Int n_col2 -) -{ - /* === Local variables ================================================== */ - - Int i ; - Int c ; - Int *cp ; - Int *cp_end ; - Int len ; - Int score ; - Int r ; - Int *rp ; - Int *rp_end ; - Int deg ; - - /* === Check A, Row, and Col ============================================ */ - - for (c = 0 ; c < n_col ; c++) - { - if (COL_IS_ALIVE (c)) - { - len = Col [c].length ; - score = Col [c].shared2.score ; - DEBUG4 (("initial live col %5d %5d %5d\n", c, len, score)) ; - ASSERT (len > 0) ; - ASSERT (score >= 0) ; - ASSERT (Col [c].shared1.thickness == 1) ; - cp = &A [Col [c].start] ; - cp_end = cp + len ; - while (cp < cp_end) - { - r = *cp++ ; - ASSERT (ROW_IS_ALIVE (r)) ; - } - } - else - { - i = Col [c].shared2.order ; - ASSERT (i >= n_col2 && i < n_col) ; - } - } - - for (r = 0 ; r < n_row ; r++) - { - if (ROW_IS_ALIVE (r)) - { - i = 0 ; - len = Row [r].length ; - deg = Row [r].shared1.degree ; - ASSERT (len > 0) ; - ASSERT (deg > 0) ; - rp = &A [Row [r].start] ; - rp_end = rp + len ; - while (rp < rp_end) - { - c = *rp++ ; - if (COL_IS_ALIVE (c)) - { - i++ ; - } - } - ASSERT (i > 0) ; - } - } -} - - -/* ========================================================================== */ -/* === debug_deg_lists ====================================================== */ -/* ========================================================================== */ - -/* - Prints the contents of the degree lists. Counts the number of columns - in the degree list and compares it to the total it should have. Also - checks the row degrees. -*/ - -PRIVATE void debug_deg_lists -( - /* === Parameters ======================================================= */ - - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int head [], - Int min_score, - Int should, - Int max_deg -) -{ - /* === Local variables ================================================== */ - - Int deg ; - Int col ; - Int have ; - Int row ; - - /* === Check the degree lists =========================================== */ - - if (n_col > 10000 && colamd_debug <= 0) - { - return ; - } - have = 0 ; - DEBUG4 (("Degree lists: %d\n", min_score)) ; - for (deg = 0 ; deg <= n_col ; deg++) - { - col = head [deg] ; - if (col == EMPTY) - { - continue ; - } - DEBUG4 (("%d:", deg)) ; - while (col != EMPTY) - { - DEBUG4 ((" %d", col)) ; - have += Col [col].shared1.thickness ; - ASSERT (COL_IS_ALIVE (col)) ; - col = Col [col].shared4.degree_next ; - } - DEBUG4 (("\n")) ; - } - DEBUG4 (("should %d have %d\n", should, have)) ; - ASSERT (should == have) ; - - /* === Check the row degrees ============================================ */ - - if (n_row > 10000 && colamd_debug <= 0) - { - return ; - } - for (row = 0 ; row < n_row ; row++) - { - if (ROW_IS_ALIVE (row)) - { - ASSERT (Row [row].shared1.degree <= max_deg) ; - } - } -} - - -/* ========================================================================== */ -/* === debug_mark =========================================================== */ -/* ========================================================================== */ - -/* - Ensures that the tag_mark is less that the maximum and also ensures that - each entry in the mark array is less than the tag mark. -*/ - -PRIVATE void debug_mark -( - /* === Parameters ======================================================= */ - - Int n_row, - Colamd_Row Row [], - Int tag_mark, - Int max_mark -) -{ - /* === Local variables ================================================== */ - - Int r ; - - /* === Check the Row marks ============================================== */ - - ASSERT (tag_mark > 0 && tag_mark <= max_mark) ; - if (n_row > 10000 && colamd_debug <= 0) - { - return ; - } - for (r = 0 ; r < n_row ; r++) - { - ASSERT (Row [r].shared2.mark < tag_mark) ; - } -} - - -/* ========================================================================== */ -/* === debug_matrix ========================================================= */ -/* ========================================================================== */ - -/* - Prints out the contents of the columns and the rows. -*/ - -PRIVATE void debug_matrix -( - /* === Parameters ======================================================= */ - - Int n_row, - Int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - Int A [] -) -{ - /* === Local variables ================================================== */ - - Int r ; - Int c ; - Int *rp ; - Int *rp_end ; - Int *cp ; - Int *cp_end ; - - /* === Dump the rows and columns of the matrix ========================== */ - - if (colamd_debug < 3) - { - return ; - } - DEBUG3 (("DUMP MATRIX:\n")) ; - for (r = 0 ; r < n_row ; r++) - { - DEBUG3 (("Row %d alive? %d\n", r, ROW_IS_ALIVE (r))) ; - if (ROW_IS_DEAD (r)) - { - continue ; - } - DEBUG3 (("start %d length %d degree %d\n", - Row [r].start, Row [r].length, Row [r].shared1.degree)) ; - rp = &A [Row [r].start] ; - rp_end = rp + Row [r].length ; - while (rp < rp_end) - { - c = *rp++ ; - DEBUG4 ((" %d col %d\n", COL_IS_ALIVE (c), c)) ; - } - } - - for (c = 0 ; c < n_col ; c++) - { - DEBUG3 (("Col %d alive? %d\n", c, COL_IS_ALIVE (c))) ; - if (COL_IS_DEAD (c)) - { - continue ; - } - DEBUG3 (("start %d length %d shared1 %d shared2 %d\n", - Col [c].start, Col [c].length, - Col [c].shared1.thickness, Col [c].shared2.score)) ; - cp = &A [Col [c].start] ; - cp_end = cp + Col [c].length ; - while (cp < cp_end) - { - r = *cp++ ; - DEBUG4 ((" %d row %d\n", ROW_IS_ALIVE (r), r)) ; - } - } -} - -PRIVATE void colamd_get_debug -( - char *method -) -{ - FILE *f ; - colamd_debug = 0 ; /* no debug printing */ - f = fopen ("debug", "r") ; - if (f == (FILE *) NULL) - { - colamd_debug = 0 ; - } - else - { - fscanf (f, "%d", &colamd_debug) ; - fclose (f) ; - } - DEBUG0 (("%s: debug version, D = %d (THIS WILL BE SLOW!)\n", - method, colamd_debug)) ; -} - -#endif /* NDEBUG */ diff --git a/code/3rd_glpk/colamd/colamd.h b/code/3rd_glpk/colamd/colamd.h deleted file mode 100644 index 511735e5..00000000 --- a/code/3rd_glpk/colamd/colamd.h +++ /dev/null @@ -1,69 +0,0 @@ -/* colamd.h */ - -/* Written by Andrew Makhorin . */ - -#ifndef COLAMD_H -#define COLAMD_H - -#define _GLPSTD_STDIO -#include "env.h" - -#define COLAMD_DATE "Nov 1, 2007" -#define COLAMD_VERSION_CODE(main, sub) ((main) * 1000 + (sub)) -#define COLAMD_MAIN_VERSION 2 -#define COLAMD_SUB_VERSION 7 -#define COLAMD_SUBSUB_VERSION 1 -#define COLAMD_VERSION \ - COLAMD_VERSION_CODE(COLAMD_MAIN_VERSION, COLAMD_SUB_VERSION) - -#define COLAMD_KNOBS 20 -#define COLAMD_STATS 20 -#define COLAMD_DENSE_ROW 0 -#define COLAMD_DENSE_COL 1 -#define COLAMD_AGGRESSIVE 2 -#define COLAMD_DEFRAG_COUNT 2 -#define COLAMD_STATUS 3 -#define COLAMD_INFO1 4 -#define COLAMD_INFO2 5 -#define COLAMD_INFO3 6 - -#define COLAMD_OK (0) -#define COLAMD_OK_BUT_JUMBLED (1) -#define COLAMD_ERROR_A_not_present (-1) -#define COLAMD_ERROR_p_not_present (-2) -#define COLAMD_ERROR_nrow_negative (-3) -#define COLAMD_ERROR_ncol_negative (-4) -#define COLAMD_ERROR_nnz_negative (-5) -#define COLAMD_ERROR_p0_nonzero (-6) -#define COLAMD_ERROR_A_too_small (-7) -#define COLAMD_ERROR_col_length_negative (-8) -#define COLAMD_ERROR_row_index_out_of_bounds (-9) -#define COLAMD_ERROR_out_of_memory (-10) -#define COLAMD_ERROR_internal_error (-999) - -#define colamd_recommended _glp_colamd_recommended -size_t colamd_recommended(int nnz, int n_row, int n_col); - -#define colamd_set_defaults _glp_colamd_set_defaults -void colamd_set_defaults(double knobs [COLAMD_KNOBS]); - -#define colamd _glp_colamd -int colamd(int n_row, int n_col, int Alen, int A[], int p[], - double knobs[COLAMD_KNOBS], int stats[COLAMD_STATS]); - -#define symamd _glp_symamd -int symamd(int n, int A[], int p[], int perm[], - double knobs[COLAMD_KNOBS], int stats[COLAMD_STATS], - void *(*allocate)(size_t, size_t), void(*release)(void *)); - -#define colamd_report _glp_colamd_report -void colamd_report(int stats[COLAMD_STATS]); - -#define symamd_report _glp_symamd_report -void symamd_report(int stats[COLAMD_STATS]); - -#define colamd_printf xprintf - -#endif - -/* eof */ diff --git a/code/3rd_glpk/config.h b/code/3rd_glpk/config.h deleted file mode 100644 index bf92cf42..00000000 --- a/code/3rd_glpk/config.h +++ /dev/null @@ -1,19 +0,0 @@ -/* GLPK configuration file (Microsoft Visual Studio Express) */ - -#if defined (_WIN32) || defined (WIN32) || defined (_WIN64) || defined (WIN64) -#define __WOE__ 1 - - -#define TLS __declspec(thread) -/* thread local storage-class specifier for reentrancy */ - -#define ODBC_DLNAME "odbc32.dll" -/* ODBC shared library name if this feature is enabled */ - -#if 0 -#define MYSQL_DLNAME "libmysql.dll" -/* MySQL shared library name if this feature is enabled */ -#endif - -#endif -/* eof */ diff --git a/code/3rd_glpk/doc/glpk.pdf b/code/3rd_glpk/doc/glpk.pdf deleted file mode 100644 index 34edcb198edaecfc432a7ae61bcc18472c642625..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 493688 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

_s+#cd^xlD(axiz>pCs&jVV*6#Y7t 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

f zHhl85C@E6JfzZLeX3hkR^fz%nD`>7+8RyMs3P7e%_t6_3;=~wB456draQFX_<8I72 zWTO84lQrS9@g8Zb+MMJ1O*wwC*>2+WdY*eljR0j)G&EIix&DTUTLap3xf+KzoX)Ny z0?5jzt!mOTQXVb{oK%A^Yy7!0Ki&vMl>iQFZfRW+U&B?oWP_b7*tiqJ+V7*$IIG!O z9KfiA%bhGRy(4peQ+_Wo{|mGvsN7pT9M;0tVhxGXEDwwTvW)T3mLB3Q+BuEwRWjo$ z2@5n=#?RJjGj?t=v1O$>KpriRdqC3~rDbFu>#MsL(+uF@v^{Zgb|sNdgbu9*+~5$;_2Pdo-|CKZt9XvYHel#VamJ@tvxT0v+J=PTBr|jb{9ap zVsI=J1cm_XZc6|SB=Uob2gxIKexjIEiVRy@Q3MG|YTZLHV@wT8dp{Kbr zXgI10!qFow&2B5zT6s?dr$?8P?qj>Ys5l%V9pr?-4+GvGKxq@g@C6-^s8J@XO$J6E zTp&57SXuzUX>8L=_r>^fIX#9h8);Mr+RuzZ#CCFJuMO8e#9{b$B|#7}UyJ=&I`+6SqaMY5vA^)sMQ9nYfbU$EAZtK`r>_%0o0!XX*xI2# z7Uw413}ogD+$E$VSRH4WZnrh4HN`Mb9kCd)#C{R*?FIGH`-$Dr$Sy9;=Rk3GRCKy5 zv|PQQWGpvyq#WNGR=NH4gy}k;I)(l-Kml7aZ<7pzbDil3c8$Jk-Vnvtv{UT2_~!@a z_tmL@~`v3H@UFVkTC956g-v+lcXLbRwprU=e*vSP+in&~e-n z<)cLKwa9PMtCNPlQ=HuVj3aM=d_c&LO?;m8IU+w=SW#xW^#n^ZNC+%;zyslsMkTsK z1>sD{&IDL7{nR?iPLo`2d~8427KU8*c&1W}y)J{J&a;X9DSD7OfXjRTpr z;`cUURs7#nWEF+c=vx~TCScIu!V`Wm4;m8gwEd9LiQENI$zvT2KavP^hb2b#iYJa} zKw#4rp%8_GA?hT#Qeza?YNElPLj}y@(iRK-0o?uD@VK0$6pUi?LMAT0;(bS8R6|ed z4?}3>Gw6@l_otiF*aXsb^0WB~44Gh;;IHR` zShzG3+Mr1@<6EYfo(h{=s08qB}L+HEM^hU>p znk+I}vT~rt6=UTFJM}+`LLS9Zhy$7>Sq@-`OlUTX!EK`@k`dHg8G!6I_!|>TG;Dgd zW>lr5N!Xkl7;qotpYNbNhHkjzXRD^)n7I|S1cDx_b4MK(v~rhfAA_Jf@ek!;8cCFs zxu6E68}Ur^>MprV^n`w;TNkx0aWU=e?3+l+CqvLoLybYz1^S015qZ zBa>~$8k%@{67@;hV2*UGj6zf&nz9P&hPD2OhifOOGu|mV2CH`c3&Tg}H+E*{v{O?C_a+tkDM zKTSP!Z2vL!46EtJ9<{*w-j?1KmafTDk0+3VoC(f@iT$%=HrTEY_Fvv#b!6ASTFqW- z&%x^|Bw4wyFm{ee4lbT-U-KsP?vmjnkyVT5AXld^#1zhr(^E3AOv8=O zj>}SkOELXKhqlXmm&H1F#&h9< zYmG13$Z{SUN(WY0hSt+!=S?ichjmFUy=LA@1%w)-$?kp}1{+k^jsAYMU+09FK%^2! z8{+BErP(GMVAgwGWo(;Nj;-CwufIC&Wox=!bb4<5Jof%WKGc}+3p)%ZIT8J;z$@=^ zMN6-Gc%WM5rA-??Xm)kc{kiU3Tx)7RRWGZCX+#48J)eEjVjoaT5tofl0v>;TK^MkS znfrOe6$iy#$1gWyd_#G?wf}BZZSCP@4%J@dnd>4f$GY)Xnr+f(;Vj)xu?&}QI8M;N z^RrNPWy969Ye+iJH~$UPdWJg%IvHf68+{+p<0Xm(-H@FYEA-yQH5ZBp3Dlc9zIgw( zvyM>5Uae!t>7F2FMLfy)Uf;j|Rsr0bBIGJIzabKX( z;`@X~`2j){r>0RrzfGK1cJM`2z}4zh!bLM#-mX!_9jU*d%RYc0*Kuv}+Bh{|;`@`f zXm%gM1cl=uQ)u%dft5Y8wj7FX5Un4aF|5%K^_a0076vjnOeoItNpk*3k);yLBTdB5 zg7|h8>O7^gCzzpku)y%`svTW9o+$GTJFW;Z$)r1;lXvjhfs)l0u1WylGd9`W~#eH%UDN>}CdWpndm|?uwZDL)x$B7T^jP zTWWE`kkfezP}vBse{fZr1&-OJZoYu;>4tag-=bZ?j!O@|omk~_7v-6&{}6C4JQKYX zK+WkANiKoh5#pKS&cWqUI%pGoaW1FgUj`}O<%psjK7+ABO;yC_8`nKI{k0t-Z#+}< zBE(99bu;VCkB@7@S}BS$m39UK(cH}FFdU_pbN>ZlZZ2KKBy~s7kV`XisMwf(L~Ch3 zmJT&)oHRrbTEWKPdv@o3$`Xn}DT>9e+ZL3jiONpfnubWP*om}+TZRNX#EklP_FREx z7&byZ7TajZ2Y`|zY#}G*fKgusKi|!x{>ZIpG*5@t+>uMu8EqEJVGI7baW?}9d*URx zuVtx@Ah=fakBXQFv^kWmT>Z+WS-WuU*iKgL728;` z?X1|gZQHhO+qP}nc22(2Isg90ZtO-i>K&sxRpXt{Q?u@QT}qR7K6?%s@gsRju^b!j zwg+@4Z%4i7SWs!(E3{_G?#3BwP%4E@KqC~2@oP{XtF4mXs?n?s&DU-}%<9D9Vu8!# z&vXCHm(09AzQ&qeOJU@n`9D@&+{3~|{3*TZL`t~|6*)<3BOxCn-FL6cQF=?QT{xK% zHAC6~tX1dowr;n!?W$!pGph53&y9#^6A2v9$`15~n*L1iaK)GEhwTmT#BX^4FyI?X zTP=H3rQHcF`et2*66~}ad&G>;kFe1Ts@>Xw@JZhSe$FN8(z9|f;rcig!%w$Xyj@G# zE@bxczu5Y=5W#aH(WGt*4Bcj+G|W%Ht7rigG+ksT4PCe_zMU}BjN-yS!0qE0!8G!; zyYN{ePo_ieGy@7%(azjR7|_Fau3h@*tFVWzfv2g;%W<;+WN+%xj}~YfC^x4Vii#*G zczIx3WRK|k_J=Op$hS?SAdeJh9M%O;C4GDdXy!(&kJH0b1$&dRFUDGr4YG+DDagm( z{@J>2Md2vJZlB=81le`!t+KY%KZrR%vpM&Nd{nWCeCL7RT=1Y z!Ff92aP#!5a)QO`u4lCoapYWAeddA&qv_~f@F9+$t98L|Idpgu*z1(v1Q~SXb9wl~ShI{zD9CvaCEtYv8*c7$Vehyf+16k2rv(6s4 zIh0&gW)ksTw)9Qo3D6Ut3yBjSO8aXJzNh2GH|*F5KE184_J^8jK1JYr3RHk zR*O{+p|P+Qyv8dO45BuIE8@H9qK{Fio*-gAdJ9%AI%05s3qlkKXuY3dI37uBb)j+1 zkr4dT=NAl>o2>uk07gmfwk&5!sy97k&D?HK;+9b#f|)PD05nW4s>#NVR8yM|O&?O! z2=Jqf_ZuFvVzK@-;D|I1E;o}TF+4!=&NAD00O%#Vg=H~Ps&=}~W$;mLBvYK6{% zI?cL$cZXzbP(ZJ^X`QOa%R@{qDse!n2@Z>fty9iJG}dF{#FK~+&YO^e=)*`T+6X|J0at(Z3YmCsKk;ggj=2;SJ7V;T9KN9A&2za+LjJG8tu~uU z-@eZ=%dLyKvqL~!Ys7cWxNd@_tWyna8XFt3Hdh+N;yn`JJffUg5p?j&(@Mj8*Of@{ z$ky@tQMEugGiAN3M{TVb1tYTaWpJ)lMYsJ?5sM^Bj4YY= z(J~A#;M0K?>VesM{u?2#Fg8r!geF;muWZY_lTERV?xIfnvMsXXpcbJ_$p z)p?b~2Orm>{qc@J<*6tIx^eN0vB%l=4-Zo08LZT>>)F*2BlGr({u+xdyte5bAJ~e` zfL2MvP`_^Ln6${BgyxyL2IrHaumA1roTF~h(Eu3(1s9)pUJTa5-3cp#oRz9(SJSj> zXz8B~2*$Sj!Vr;&Wz3q6X{fi(`spVoXOtgGO-(YAjQiM=G8!X9WI@b8J{5SQ`&Nl6 zOAu;OkBj}#$+Xf}Xv}vaCaO%HQJ6JannyJkeB?laqt8aM+NDc1Ko81(WCWV+WX2@+ zIc#uPA_^`0a_wWC+3Q%-A+kDi0)ev?cOs1n1*Ea@nbM{`d(sy^j_Y14RW?7@0+T>i zU!b-)3xx}CS1Bv68w3Akp!Y&914A%8wZY6Nby-~!#;S|@zN81sEY#%%GT5Q3}Cief+8eMAfC zQiUgllVGKQJGqk{uhPO-Vvs3<_XgRaB;FIXf*oiIg$2S>lM$gvMs$fU3U@rXr+n@GCz=5QQU9mgIJze zTW+=m$erLnwh-OE^^=)+%2VUZW4Rf0G#QIHOz%wiP+QjDb^?7|*>f@N%YR+(V~(<> z_eLt`ipJ)mRcq@H?^X;?* zy5hvK4q|RPhJ)cmGPc64@7pGMT&OKgyL1!&O~X1`T}g08)Usq3&11TPgr$kw-V|d3 znNkc)7lQgghT3*|ZWiNtR-8P3*p|;?hFe|!4wgDbtms2vo7MFGZ6?%J%Uk8EDG}U) zsPvaRnrqnQc}-2TbO|A3+)SF6x(@^jm;Cjf!nY#ya!^aDQ=d#@7^LEeBbl|;7G!H8 zK=8ON;4Y%g6Hv4fn`Oflz1w$y74%9eG{PtaF${j?h1%E=t%sSeEbf}K)KGh`8TQY~NT4#f0q_#w zT8?xPfmTnVA`0y?X7}<5Ma-q=7Rm+9O{gE;DG%27x#s0^{34^fxr79b^`A%4<>jdX z0R(U>DKX(hx%{zFGDI3AkWM>XVv@GLM4j=_M5ckT#5CtmEW)VH5fGd2_Gut~5VPS5 zDfxpX;KF;QRI`XbZKq8ZjD`88uPYT^*op-Lq6dHoJQ!tjcN2IiYcTl;aT)?k?8p^A@n*tx-cs|L+Bb*9zpav z%q#B*gh2s~M*I2`DUh=?I0i6F;Z~<6D7C|~Ahei5C7kjRMPau#^nQuDD2nF7e1mjT zD-2NJXfVNsu7g<<EqS(_ve!(u40@%-QK&dh)Oy)VsdDp1Iw>S6dqmjqjfKU%1>ekzbUOPKQ~= zR3B-)6zwkz!+U?h}d_XKCmW>q0z1Aql>RZsU+o- zDwAk<2BqPB--eo1>;61q#Di7+`nL-A-*^rT4F8zi*i>G(S{MEwSotQFVgmMAm?phB z$1%LmBvTL&Y9Ww7qCv=?@RwE|`pB?xP<@AJo)CuVY@X;2acjNtl-AU2#r^DV6#RD! zGStg8$9Uwu(N#y=!PstwkQ&Fm`^j_5bY4$t^_K&vg5)&kRG<^y+~q~}YxR_j{c(Q3 zfk8%Q{%<2iMtqozNMuo?z9(kKEe>lDqz68hQ2`AcK&;-r8gLdsMuis5h`>psIp_09 zM5be;EH{fOfuaGFH~g_Z+Mkuqm0{ru%`B{iBJKsbDDpO(9}1hiXPe^%Xd4=8^12i> z!Yw`#c33%ih|hti&qFI+)}*MPWcR(8YCB16bt*&@4!jq^<+;I#8&<5&?8tPjLtW)~ zy<%&F@HYKuylLdRXF$km#Z5 zK7f}bOnC7TYh{O8R<(}%hIae*zp6{HlxZSQZCp7+Qig=$VBn!;&6EDNN8i@zy{|ix zF=OOns#Ss-o4^W#?DASXa2GalPm{Mct|&*SDG6FCzRh1Vi;+`1GMw8Jt{YMLG%ozRY}ZLv5w*ssanPqox8beZvH(4IWAzvG}(WsE8(fPOEha4=guK{f#j?>gSv6b-mXj zVfRK25!!^{&o4J@^EJ5N0zr#)U*8z0yztn(pwn1ND66X6S^6qIf+BFAP-p4JAL1By zl#&}SQmj!~qg6Y(wUj2$qfJhQd0sBo%Hr8|l(c(P2mQpaM$_^)B#OtXMr-vp0o88P zb=SyQ%p>-F^?xkt@*z*SM+jvSRdG6~?MZIHoIwoZCHVlR5gbB7BWRw`~ zOyHQphu~>y(tJVn>0bm#9u=R8%v1X}%L}SF)rnVuQ!^lZN3!zJ9*>EnPzRBJ(PkUj zsH>rj0@3_MvuqT;^vo&}Ed9{6h!T0fypb!k~|9D>=OxFBK%b^5aKT|rGTK}Sq+xr>R&lwLR7@S)IhN24G9T3ms zuhmc$$IRjJ3dV~gQ%%aS7>ofxCBO58C%0Jf@NBr33MR=Rndb&Q9iM4o;AgiaqXUkw7O^Kd{G~#CIbxOWEJ6u8dWLLpA3697ekybrkGz6Qad%sDlnI0$fRxwIOE9#`alW>1NzDIi^ z+)TLI)BAFxqcQg*)>}yN5FIO9x!6!S8t#t?`^*hjhafRRX)2?@hQtJKIB;$=VV(p2&>#E08^DAFk-_f*|9}O-qUvbFS7(zVwoQ_ViTK;4$z0vDL4*C5nj10Z^o zFvJkM-i9;*iU~g90t@`?&x5i6Lj-e9EUqZwhESN>A}iMPH;C4^=_#=GspU8OK}#J` zl}U^j$fXD|Pn2CQ)r`kQNX1R8nI?=D=feCbhu8+{&b$pVFpM!QS}}E0nMdl->y+^f z16uY$gpHQZ{YBh%e4YrR#0C!8hE0O?cTR1ojH2{`hCfvT!ildK8Y2gJVTR)o^7DY{ zO`h;cavqoTi^1uy@WJ9OTx@3fBkaKGPnZ$>x2J^p+hMm(86bWbRz5Y=K~s8$4s>kR{=T+6?U2T(EA~K;?KV5NVUzl zxy?w}wIUS_5yTyvYn>%FwQpBWE2hI39Ycw&L-k$$VYveY&Gmjj6{eJzt z@Nj+8)oE-Xd3G+=APVhlOZkgsIon66sy&fDgXz+R3x!`$XYKuTjQI4GW;rcjh6_~q zIj7{@Sc`)-sD1p+0F`~|2yrJy(81152%;VtQ0Z2;6@95PJTD`{rX|Ssb=2yzYD*Pe z6Ji>`>KY(^%j=Jny&J4-Obtef@QJVdodZvszgO~g!9y4)Pd=@f@cv#!bYW$z!|&_Frzx$ z&#z){fLUNlp?@o^|BYVp{}Wmb{y!i0MSRC##Y+*|#e7)RN|2SxG7LHbxUb%LEtI+) zF}rH`=ga1;P@ct_I0uY?zZy}rv&a?~d^8L<2vf)o9xM`=w_MEzC~dVrT6HM|8F6Yt z?<3~WF%zLGwnnuTzc5Iy2TO6n=#6fB_NLZ1_J<9Q2L#0Q+zxq&KA}GI?@)iu7{LN- zOMmBH^=MzMp%oXhHsnTAVL_IXNw!!I@oRho6x#F4P`$7}aeR!0egzKVjlT^Y@UD#Q zvqEmHKVo$i9Hdk^r(94_-2fu+oDy-lE}n1r`txQjGsgSgWGfTAX|;+0&7v3@n45=E zh#2ICwmd7&Jk^_=nAiB6k)$9Rj^*P-WmFe~->F*&9XGCY;;(xO+xtKH8=s{-t3|qB zFU7qR{uc9W(Hd>L;>-)1guO=yd(pNDC^QNywS9@kK`KAE`M>uGt}FysRX{NsH$bGu zCJ20$7NEag)W+Dw$^PCz=SlFLQj>3=lJ7j933{v(ylyjFHqS;*u)EZYVuX9VuRa|kJ)P*dcYF?4< z8%{;h-AZ5=#Hkc$`c#?LnM3{pPc5y;4@QI^Z&b?r=wD?~t+kJDZmTbI3_j<4+Ac@^ zQ@t?C7a3WZwiFSk-!L4jl`Bj6@w7P3>fG(|U2I$)NJP%)E;TgjJ)shZ+zU2_G@V10 z(znv1+Q!bO=ASB2VaTj5+*Yv`%g~0HM$fp)YD5dC9R&)N5+J5PqCu~(uWBx?W-k6` zTz@H9Z#7ZhYyKV$F*4auZdca02t7@G>F@gCm8dck2hXADcbCtdNs>y`t*w^lH!fR& z%6D|du)JIBqkJ4rY?kAiJ|QyhngDbrbDqHirmlh z5c^JB?+Gija~8S|ffIvZpTOC@;vVgjZ`gP&3WqJ2>H*J#4Ipq;S zqPjrI_K9C-1aRTFlP*SgL|3;NCY+?kv{rQ3nU5etxm zfi(#oG7|XYL?~|YWJEm5@?W<4>%1Vz=&z^Up}=O`FSIscB7fp+p}3_i;Tf0K?8~I* zkv|w4Fb!E*iJaCn2n__F0VIfqlif_@gF%rCbE%!gqMmmoW9FdmOYcsonr*W)T+3C) z5E2TfJm8dy+#H>?JiYufe&Kb--Sj|TIsuac+f?TE^xK=F@X`^WX)Al1_jI-!XTx%EiNQ4|3V>n0 z{U8u+r|1>}<8DuDTmYG|BfMVi-Ql`VKY~7b;)BSftfY2452gbB}sxozSTFk zOC7QA+ftRTxs`2*fZ%(bLu1?chx_klDiB<$qY$UD;X^gWK3ggqVf@3Y2vLFo1_ybh zemLt3Knf3UojXkSr!AIakl*)P<|U{UcCYEgSKzdX|>(rVoiwKYnC=*WV`^gJGJ0a)V8HHVBIFwt1X z?FpN%RMopco4%cBZ{k$# zZ|xObtm$5ampMZHrtoR^e7rumJq(UuQ>SuXieK%JRZJcIIIgyIMlaKig4g8!>&Lds zAs5D=SgKZ&PZ!C0bHu814VJMX4IbRpK6tME-PSXicXRWRW9F>9{AII@V{IP(KAhLp zwl&lZs?Y1~auj#);c6XRR}H7X*r=tI*;Z|ml@(@PR4gI2;5=MwR=La`uFCi<38U(W zYuE!~GvTtEOSRV9Ydp(h*3R-e{j|w4gsi?k)Kp71%a#KjGv(t*V?S0P;sJrKxG=Fi z+9FaCNUUR`2U#(Kv{FH3J0zD&9tZg+HmWWvp4Xk(SNcKBzzP9UNpP&*glzy)XQgh9 zA!`jMtF1u5Mr-)+$57XVeVb#_xkBgEiGw;&Hf*^)$S)#uCBPxBNDO6IUld9*936RC zjs%aZUznH2omUVw1jlTq0t%wR_6H_+*%00IthJ+kZ`qeLi9B(wn;t>2jx%qjv!PdL zS{sc3xv%2E7S(J$$fAJ#&>mN_fPhH2ej>aj?}c3ZF?F10r@a0qtvbwTZ$TI=(~J=K zrVYDXBpF}bJCc!oo9EKl$hxqRPLO*8&-d#K zy^btI+I^#NMr)bSL@^f=jY}67Zt~!cCRw{zX+E5qwtMvPU-#ulsahE35z#ZS8_4w| zl%~y)^wQ2N6CIPfdCpJEVg2rhA-t5tw@Ovhy6n`%*Nmnu4j|McRZuLldv|>I9{H&y zl!{5-&X>wMCvxEX05ubD3a;83*-c4@8Bxc}<03Emw#WL}&V&Ap<9t{rx>~q%WusPz zWj3yNDJSWM2@I(QdZuww%~7LMnQjl@M&f;oe=8yXjU2$h@K4(j#%SC|)8ke3PHbig zfgeB%TeIRtoqa6M_etAq_Q+nN%VV_w_CyKHH?}C~+bt2<}!FfA6u+{~8s1 znEl&iWbHfK&b{l|_dXIN21pxRJ-4wURKbpqBIt~cY<9TgH{Shxg1d({D5WsrU44O2 z8RuTRbw7Tm$NkhXT*1zD$9JsGxL+LW$~l;o1t2Oz|~P zofb7y%~uYb*$7My_JN}ph_M_`4qIXuXJ5aYkrHjQf9~sC?X;1^b^vIU{*sy_YHoq-xAs0tVf3==w;E#WU zsBpWrn_!o|>gXJ0zn-MnudfH8b@_U9b5g-5nilqmdC5EIM8>QaC|+72QbkBpI5ezc zxv|$!U-RH$y?~Ad-?$S(bofjjz&03t;e_AEHHIw^;hoM=WdK0M>lTBa(@Wb)SHbt3(~qef?rwrLflbgvr&0WPxPtZZ6iGuGUBY4E6C8sRraPADy>-s;?qU0G0Z{>!EE#Z{ zAU0(L;YXEgf`qWbrPXJ}9{uBEtzT=RxV1#A_@7ls$icc5usr%)nW?Jg&kJ6@Q*{pD zqMj;BJ(D7TdMYBvw9Pr*;;%$+R#2VF_(BCkIq?c8bov2MV>*98b2GyL1wDa0i-(e> zizFn;YTlWUb*U59&P6|LGm>u14-)O?ehu}o!H;}ssNfg@4U(9uZ36ufo+HUJuhd9S zl?n^L6WnwMvo*wTzf*1}blj1n$pmP=RljHH2hxp3O z&MS**4O>*HB)$4+ix512PHM-}BofLi$Ie97>40Q+>kV;;#&>+r41MN%0%dED} zE`7z-5q1$shi|*}75Z%K8M=ez^H3vEfC>w4#bp&eEdLwK;^3^30%gyEEWUr)eWpsn ziehjORJs8Pkp8%Al=jF=#7+*yQkrl$!#ynk-dG$Uc9t5IR8Q0Six?o%cTJVc;dkqh z{3=1kG35&R$^GC-%l6B9o7wPX^TG}4=kGFrB1x>3oPoeNkwXiXnIKl&Fsb;8W72N1 zA$wL~%^%}l+hHtXEC-^lp(FH{GVDT;i_UIX(lR_Mt{hP5}3piW+sDJ8b)?;w)6o>v!B zlCwuUuJS>46GLoztG$e1oUXkcfT;ImFrlGEwFbbV)FPo}3$`33c(mtFIwv6sd@Gp2 z-ytnYS6 zDn7OFy`@TF>QU7M?c`@=%tjdWX47A!MloY@^pAhWmVGg!(!t}Q=0^qzMmvn4s#iTz z33SHK2x~8qPj8Aj+*A&b;yt5qi?YQHDjZ64i-m?B_w)EyFB+!-kzG+WfY3u3#aQQ< zPW%LGs2sN~Z@6DPgM+{gPf`lLkhr3AmJV^faQPOIwuyH9F`!r=3m87eIm6fZKVU^m;$#H{Nvsd`L|tj^KHu|4>v=bUJ&$%gjZm@9^&d4| ziIr`vi(o=|Jm35ds9Ztj5}eELzS*7%ywS2-ZnoFmn$jLEc9x06fUj^qf%MIaU8e85 zYO2iShGLXxiIs#$Y4hnPia+P?JkdjwSTB7C)nYwG`U&*7KtiPE0Up40;VHR6I0T{P<<4I7cOoK5pL4X^%v%P7RNwUmLPWHQj#Cd+{(Uez;oaCJgj39OH^sAVxg|cF02uF7kk!)o>~IlOvb=Q zKkwG)4g3%i#<;J_uO{3r4J*oFjEKA{hXeP_IGpvuGA3ps9t|tO>0HLl{!2y^G}<4D zb9+)V+X%a%V`kY=P5+i_jfI^FG6~1HyoMWs3a`^AsEd_GCw&>~XE$kUE@Xk@%H)7= zt4j%DgS6DlMn6tH!cgW19f?dJ_^?_PbYFsZp&)3atxm4aGu^j0Q|P&1#K6kd!|a2Y zwx4(G8JL=omEB()VsA(e6LHT<2Z!id>w`VKOqT6frGq;M9-uALeRFqup6ecKm$Gi{ z-tbL%a2U-uB`*E<73ffQ`a8rfcfdSBfV9v>ymj*uIstmDt~deu5WZrOU5+N9Apvcg z%vsCaL+PdGY4y`rq048(XlbZD3%~hyc`%!V^gUx*wtV!?WCR*5gH~Dg^vkJ1--L`X z=eUGg0_nh@)weXL%2N9J? zo3MlG!7s>;$M-a^5R8~)0Se1P`!!0%p0#QFv4(XMG4rgpl0+gUv!i*U4H9>*Lyh4= zLM&Ay&zy2NgH{=ir$6^@Z1byU)XhTOUN|wN>Cy>3`lasdj}9_TF6OvnlmLXyHz9uB z8qb-uGC@gc5fn9J$glY{LCLT&qM_d8#3SyzKdGi}PM_v*xiCy-hWVhjMH9 zT?j2{Cum{8$gNrJy-T~FuVrnLbvUevjja=O#`I2o+c}sOR_w8TP(9SKsCV+-EC(}) zyrsjpjTdaDc5vaN)LkCLzzvm)_v@iHwjdqhztx2Q6l7tbr~k)XQ={tt(#A+%Ufuk7 zhXNBJd*$*EJ}8!9TJXUQtJa%&vIdt`EKLWZc27VX(bq7+2?oJUXs}^m%tCO+Y}V5- zt)s&Mq!=H=$npI@Q%z2z&a9*3tU`Un$nvDD&NUrJ7?e64uhf&NT9xe6hHp*RTh`~B zX4Dbl!DSmPmt39?o4OWjZYrN%_tyu(SbHOeee9G8cT|Q$4u~oYS1Dx=?R9F9SzayR zxR*opF0aDGWIU8Mq{ol%u4U=iFMit<&B1A=_Rhtz;c9 z0ov1?64M<9*t7bsx?ZzMAF$FkyRKu7<|&%V$w5fwDiCfc(bo{8mLqJ~2n5ewaGV^s zeMoI6WPE#ZQ+ltwEY5sEon!jFKWE|J+^_LN02Z(!Ba~Q1*T{@UFFm#*-anE;^r6N< za4y0A4&`6UsFWQlv z;5G4~6GMDKk2O@OiVp>GPz6qksozb`pq5&Dob;S;a|t}1U+7^Mk3^f*H}fZjoC>9= z7_3KuvpVlyK+VC|Hd_}2IaRRy%>bi7F!%L~ryR0)MAq@J(h$LE1}VL$k&q`U&9L?`m=fasDltJ=)bR~r6 zI3f~YSCz@gGO|9tf46vu1y z#7f0+y}#(nz_^~ zsgGxrS^TYox$vb-j?x{?$RzMiGp)IQyx*bn`s&_&z)M|HPnE?E3?U60+yp|QmCwf0 zHVVkH5GRtu;FvgT?2`s<v8rVPH|lfK8eBhN zl3Pt?pZL1GQ3NcaKa4SP+(dOYFzDEte!U$}C2A$DkZs#n-vC&jv%~HZT-y|=Tbz4M zCY?a_yYxrZr8q+*2GlM1*#=HcL?w(4?K89nVWy1}blHM>O3_ue+=uH@CG5_cBZd7d zyN@k-GSP$=d+R<3ps0N}Y%%Y{fU-0e?9Gs2xVtcUrQpTPLc|ba&q0DtiolEl2Le3}c|n>=>b!tvd+%SHB+; zcqLkPGus@#*PsVT5d}={TQ8pk?C-wnHu_=rU4$uT+**zmmVJ& zJo)^Ka6WrvEZX|rImx4J_s`N-#5z?5gyP;jxfDI?v+bDQ3h4Sv+~r545c6o0U~ROU z-dal&rC4Jz@#O8&*|R}BFlWfgtYQ2*1!8VS2!<`gP$ZAE!4E)@p$$4Yev9fS@1}p6 zEN0UrnuCxrWSS43=C}XWH2RDJnC{f;wd)ZChXVTWy=Cbyg@#Y$24c{jGB2gx6B{wN z<*K#~R{SBvUsJkv#@G{7lWuYQR^4dHf^GrSO;4G(Ax~8odh4rEa%&=XOPaGiJT`bm z6fj$Lq%G~y7B?Oo$*Y-jZy#2pu=gi$i<=u<&Z*JPsol+J*TE_pp z_N&@*wZhI5wCZate7Nk!VKu_=1%>HY?O#_Mm|7|PfsRx@7K+r~$QEL1E*A7lSZW=5 z+2|z(y%L;%W(alg`%HdJld$PT^*h{q(M>gh)d+^Fit~R$q6i5zl6^>4-NiJ@^B2aEG?0Bwb2Ok4F9|9FOL7~tiH1q zz1q?my3y$snPAS>A)Yr7q)2?rzwATU|67WTiS-|W+D4Uit9_RL1!^HY9uhHgETs~EcR@S#zyGZTlrc5-z7gg z5%>;^{m+h@t&ZYc5puDB8&q?Z?0Fxe*k+Z_OGczvo1dfuUril)A^aOuG>BLldM5ZPFU+wF9@RiX^Th6 znWcIIYPT>mH$SoOtp)_r$?tNvW85Mp1BVc`nKf>bzeU)tOzV1kgJ$9l5S4)@{Lj_?tEz+Gs+cVb{?CX~nQ+hnpVdXwSuj12r&f^~W zOcU&&xI3V1y)U4vV~>5^p-?ATIneqqK4DAIzXA-QgxLbH6+>``i~K|aSq z@aYi;PghocMXH)tEQwm4kP_sGqvI;#hN3`kL{flfOj6fhv}p_>&5TUH;zZp96&fo2 zi~Is^>80e_3QGmk7<9^OS*kK~$-^O=KTKp?DIgX(VHe?Unqs~((<>Mr-XdjUf5lmw zG_)3wIsUHOGjESGIkPDfbNvU$#kv41#IUh%9OiBkcj4W=YcN5;GFF7GMh8vC>dN z@-(lhY&};s@_-=pYcVp&$2KayQ^qF8<7Z-6)2wVhWq&@ckUyKhDaQ``%|D)<=dc&K zl``Rr#d1z+>o9Vh0?912@^MU{ia`>xd4h+Knxw|Y0AN9xvN<(TvpfWLUSh%;{;j=! zC88Tb#ZMU3juRDdmA>-fLVkU8VhIbSC18F@kBUtk^_|}lR)=O2zh}7aMDbg;!Dw$E zO<#6DkYv!IylBp>kcWl5Nfw?>2RID4^}LPBe4`PZgoEo)t4Vhm;&71T$&AJt5QZcqJ>c60Js7)v9ag8}cNkMuEzL5j5Zpc=2C;{89y`?#bvjdb|7~ONjVK{$~d^?|JR0ag}eZwA1ajfK@kzP0$@|ik8 zu9P}Kq)&AcrLQ8R&KV$ydBUci7Dx-v&d$m~pSY5=i^3wyrf&$Rn+?-a@1`v+dEWMc zp-)>RdeXuYDXleSU`hN5C_QYMeNZSnwT3A!;}8*ygf!Z)=|4sdP9(|L0I&i^;M^0A z%uI{Bb4W+;=1Aa4LPLt;Mc}x1N#T>}$y@+(LtKdpJ&W#VuD(7ty?L-mvHool_)lN= z8R(f={?VaQRkc3&pIW_oCjXC({h)-%lPTCJ6$u8Fk^V6`986n!Q)7O<;OAmlf2YFP%Ht150Y=)CG-r{*C}0 zXHlOD4CC?KvM?;#H=j@Df_1i>hu^eiO_-#fbht$1w&;Fc9@Nh7)UG9*2q+^0y*fzyMX?4o>2T$wZDEJeoL)J0--0K} zI%CsP&qlVlm1fn9sOk~Cy=swL0iWnebhifQ+#`5!o27(h{>)X0k=D9xf|FjhF>kS& zAYe-IW^<&IZrL0EqA&2k=d0%*fo^@LM))riDNRe<>G&9U`79z4iZ)5NGE4^$Q^}Sjft1lxdn<}C*eSAd zN>*P6-e^WlqjFn0%e8KN1siX?jA;|umSr+anjg;LLejA{m$;@WOiDTJeC)}XPI=^I zcgiVgf`e%XJy!uB4TJ!p5T!q6kqhb{GfHD-y}-QZdq~CNn5D6y)X&7W zD^pmEIaRXnaI@OZExc_?%PgM`UR4CtKB$LafD{yCicm_Ct|TMOPk6X2xRr|h+Y(4X zU4+=>nFSqv>{PE}jwaG&3#e2R^4HRah!xZY%_VvfPswT=nM$;}1 zmJ(IbCq#epYS({oBG0C;MN58bxV=f8;utc}PG-7wea|%JNL)r`O~7vgamYRyw`29V zh?CzWz++X^;2dVQU3xUGtoJ4)(z1GvKp)rCTciW{Kv;f$g7chMK_XF~86?$XXDE|R z1}N~;(_ao(nz#2T*E@8bN1(8!sK0}q@{=VU+jDcu(Tbx5U>LQdx6m&d1DM!l#cwfp zApze8WCmcyqreT@MG7X2Q&&ddBgf~+l50S0fzt4`z}VopN8oF?k_B_9*`h`IJ7@FE ztpTNG6_`PWI7BS1pV@%6oRyj5RUyb$fLH^nsDc16=`9*QDgV(mi;V-)AFov0mcnydr-metA2 z*IEIUg!!X)1zK3*y_%4~xuu&My&2wQfGI`tz|xv@du$6~T*$hjc-S1KrM+JRi8}zy z_$z>K7NX4Vg7kv0K{48~j*a^`+lll!$_=iZ(=Yy2oaG_p0!rn1AHy%l82xV| z0GxJ1Dv7jl;>X{6gV)Gd-pgLPLmbIors-;zH3}tJ0ywC4_+2o?Fq#C%xW9G~M{lQc zGF}iIhF#rXAmjO#A`>9xuDb_P{w1GUOA%6RfA<1ce%{`rz)YCl`p<(lTi$(Dv=X>g z7WlO6+aBl-B9~v|6HWQe|0JAgt}JmRe#s=ufjZ(}hx6?~i~#J2-$vBG=rjXRH**2i zz_Mej31-GbE)EV4oLF#Z=7QA4x)Tw3;Uk+9UtjvP09oP&bF%7b3|^2(8r=5g$5 z-AG#Z3U;95=>vep*nt5Q0hYk^DCgt|NnA;lprt*NR^Fjh%E4*Hh!g(E#KJYdH$e7z zmOPqMWZ^h)?i^l#aom=aVGbCszolT0k%8;l1uSpwFz6P(6v+4%Uev064XeSbq9oox zO)M3KATcsmGi~1+m2mlT>W2WQ9@xu&=uyj4D$8c&sli*i*#~6cc|NaIP~XFLXhO+K z)IHg-Q|^IYZ4wO)rVI#Bsg7|>WwvRl<&r$W9y|YAL!~VI_N4wcq7e0#1xsyo8!|Bs zmP9Z3GC2j<+EtdzXa3wIamQc2VxS{GYuu^TVW5K|Z7}=yo*|Zh-aqeFhQu@)f#9Dm z@xjN1e-uspEn_GDV+U?L8waLeq5Mk*1HkYIao_TrWe|l5F7OYGogqyNyhx)A7I6tk z3}=nRPg_iyfL{b%Fh$={mduugLW;b$#GH?Uu}2VHalUb{QxDiFDYWPVZj~pt^qdbE z{#TA?ecie!#SSyz1&x57(cU;(a|NDyn1tU}kz)S`V~%o2=i*0vzX_|q5@|mc zAO+&(!`{0E(!IjK}oJK zyE?!QSIwO(#o%OogdTUhyhO4KF3KMFo%!jsbf`I{LVU)B^I-jIH>xP$bQIJ?b)~;i zs3A)aGPywS}!>{Shhm+$!#7EfSo_reKHJm}m-G97B! zmZ)rmB$3gDoF$Ld_G=X~{jE}*E53y4C~{1nhnsGzb~UI;{nx}Y?k0hK+ogVavlm0B z1j_i1i7g_o_2%qxrJ{ZwpS$SQ7na|QgT1sEvgmT6oB4aloB8x+2%*(BSMV(oq}7g? zt9OCf15G+Mnw7-3=;<;DOKLcy1iG1hvq zM_wvaPv|LJb#xXKkHMybe*|+>Ul{4lSUyAtVfkoJ~Z>g(4;ej%3(1GoKog-TXCH0$U5o zI!}fHzH^p(#bG)V5jPEy!0e7)L{yCRQC4uleNL*93l%I6hP+J>C&Qy8Ie|9AzY?NI~!9uw`yU%)>@ce zy2?1f=Z|JY0}GMLV>1Q(CCyyhUWT1^S@~J%dz0U0!6rJN8)M#M7->@W`s<9Yi*HYP zd8Rvc@QA!qEuUtEEjrqB=+8?wx6uf;-9&AnvcZRH63;5f4Cjuk=~^H{fP(P$Cw2?` zK%HQwQo$gknNYecPexfxiPqR9QlWL7+s1V5b7hIPAy|B=EOA37m%rIkt0+oX75I=z zpr*2`%vP+zpkjzO_!s9mslN9TCq7*_Lm+oH9Oj4WG30Rs8Spwe{yFZQE07bO&D@(y z1Yb92T%Z+5xG~@DVpY%a6B5Kc(mS>@Sron7fEJ zQzLD+N%t`neqff%es>c^7RTJy10si1Fjv!Y(#sYX@-RNi`rf$Wdpes^dVy>@){E9=AxmB!4MuG z@Tudmg!U6vCz6utQ@eST+iQ-=!CxMS6fu-RG*&2%^bZJcEalkn7HcWc1X)XY7Osju zFHLc|#jAx$&H%anQCLlFqUT>AvgRl$zd7Rw;Z0O}F`*}aC2$>9`}aM_q^PEj>&d2f z;rYPFHa?%!G7YHm7j0_2{5>W{NB>AKfL%@_tIG^~nC8dysWdf58>!QO@|2ZlYnYj* zcU&+=Fl)lFTXQ{Eb#`@N;9|Zg$ejCmGL$ge#j4>JZWjTjBSiye7D$I<)Nzf(1vL}Y zOtxbpiu9fhS+gW5^tgdW3HD!^5yF2iA;?Rl$!p`2eSemB`Gj*#|NjdUGyVUR9%E(w z5A<=c@^3)_J&ez;8iZFm_fmyRYHO*BRVZp_lSEyr2|3cJ3^#F#Ri_Qq)gf7iY?eM~ z2uPkdF0LZ6iK865#6T=E)c_yCdPtVc*g2?LkW-4K45?c`-eXX4h%_o+xqeDI`H8l9 zhql^$?f4&#+&HfR<&^R$49|Hz$LQr3USAd@6auZlPCTg zY^U>|F>rY~-dq+D#leUM^T>6~ZoL`>LRe*Y z&wYqFGK!xIRsDFvHhvTl(QD5Q`AnzYFk?Uv533T9G0c8_EKj8D_)G`;>Zle>NIs z*vg!77M|STgg^(C=4-$(%ji!u>DNfd^_2uo8lM+d0nyltlK3IQv7cd&02aBSJaYG` z&%y$f+i0`bB_>>jM$A|~w(u@Y_+IX2)Gh1J!9tVlQFHK6@I%cZwBa)GcJdIxMj8pW zG)P}Q_II$5z$|n|Y15#HSQG^lh#ZGBe;a3|Lt%v}^kJn@>WI$t+B|dzGxxG`n3CKz zz*AYCK&(?_OH(r`oq6f{kylJ0UqDVl1rC9}w3K^Jt|#1%BHa|KP)1FFSC9W?3y`h> zykO#o_~J$L<-JevEgi!lr?e;HeU}9C=)C;-4$CU=oyJ#Ja#@(gs^##8?D60?7a|#= zknqTexFI)Jap9E;u6KC5!DhL&Zw|Jv=! z{|^`UHI`27X4`#t??9|3v*|E=xH6oNxj$}P!D^?_!^-tv!%3@SYYU?B^{ahf@6*2t z_TFp20d& z8BsBkR)1oW?A4?wlEZbAKR0#m_btn9kgg+bwS*KztZdCMgw#|;yqHIk3TaG*Z1(?izKS5do!D@5#U~;^ zQ$Sj~7>4_*YHziAsMubb?%}~yWjmJ9*r%>*@3!kSy!B$u`RWwXy_tXuc8rx_)Ld49 zMwd!~tUX?!%UvRr2>ZDdhjQ@&O_VWNLLm=s=-8*6+kGXG56`4gTOkqGLYpSPW{%k? z7<`ueQy-*%Tbn>Uk;w@D-V7t$02bF}h_KgTT!%yG1kA`M6cjwX`R9`34^!T;sFw|a zh6P(hTFIqA;JfSCOw>uaa8dcF3oOv-vYnNBkt5$#5Tp+lbfWASnQO(RG8f>5wzCk=@U>p-lRGlXSzqXi@a+LfC^84KDWq^1@ zw|G$LeJuu|?2s!1zrSRzy`?AdDMzy)DN(5(FHy|V{DQn*H1iUfBoV5nr)=)venQDt z{rqG5;$T=&?rLFhrO^I{|FLC=U|@ExRNF_E{VBT+z`z=i0Jqj z$Lv;QFoQ0hNEdFg(U42B<0@;W&+9cPgz#_j__A72V!g2bH*?5Y+m)T@x51(>UQ&_o zTh`HJj6YreAGhA`gT`YUOVrQ5mns44Ydon zyrN(EJX^Er+-u#6emJ&CQOW)y3bN|Q=_*cJrsAb&j}(|R2mfH^auSOrusO{~!jo*y zw_D7i)NzOonHcBS(bGLOJ74yAc{DTaBfZ%=!?hWM=6a}m9}lEUxQxGw8XJ}SseA7J zj~uW>0!5kdTkm|103|1eBys9E@mfxy zTCS~x!XMX6WqO6=U~>#;CG%2o7_0R5I{LTmFj@Jka&9E6#ItFGy>BtU`_5|6n;=w> zQqF0(Zi#Xjmnn6uId-HWED+-Fg#bG3D($)+=cEWKNWS$+XkTndw=wt@hoIaCgIpS) zM4;nvG=!dN#L%$ zB3IJ6O)Ltqg#vUtsFgL?4R`)VPGzdTpxCqv8JRqnL>CN`Fl!o2`F1+aBEV3Bo)Xj# zxX)RkiwBPuDr|Yvt@OyyO4^?wm5{H16b$!xq$Z;}#o&|nMQhUgcK%!v5@pK#(r?ci z_6c!;gsoaT5fF2^De}Y&`UKn}7-Hm5!pM^Tb?%V*cnoq=Pg6t!vKmTW>kqGjV7S(H zKVyfm$L;YLJ!fgc;>b{LYcfo1Bvj6~I#9~0INidCqxG#K_1G)e8!==VCce{vS2b5H zZA1t)t262$8{sdsExy+Pmi*L%u<$s>+;;@S>+hE!wi|!ZTsQO1@h@Q%cTnEOK<#IH z+3H-o3i%!hrC@O__E~{RGdmkZWE+F3F@59fuNDTz*79CmC(`5fQ7XaRomADCXvHdw zmb8l6D8BFye(}91UgJEEU=g|ub57`2!)|QyUV1ea;3grmM189_4p7y(t1KIgEd_fG zZQ-4*6ADn2Mu!1<@|TW!|0HLv?JX%6pKO6Y9sgZDoP^XF;Te>k7-@5EoZ`jUm(QCU zsK6O}Y~ojAWCjBiP7h$(6kIJZrIZ^ypUzI z4gPrfEIgI_$;a~VX;_V>$m**se7fa@4BY$P04~*&q-d3M@Fh6L2R8~UMhz=NJ89R;|+V)h8eHdzXPwz_re_f}K&ElJ@DWHS1LGp11K+hkEEscWOiz#@xPdXn5 z1no-1ZdZq%r>fp%Ms!~3Fr@e0Q-_fJ&SO@8XQT7*RcP%<0VZ(7Nk8}$ub}Pdqq)vN zQyC(q#1s|=#CU*dqSLQ1j@|s_b@j{g{>(x-B~WC^;b-dZPu2SV*;{t+TcC{Wb>SeL zoa3Vp*H02dl-oZ9b*i@MA^jU)h$vaQa5w0Drq(ts_P@0a{6L~bk>5btsECz4Jz8Cw zU*5WwBl=%wQc$p6* z6!uONyRTO+M@s+KfFi(G2Jz8Ah;3{4UiD1Yk8Ba(EC2CvSo9OFd57rDEsNDJwlH9l zW;U9K;o2pfUN9Mc!-x9Emb#6a)!B=|Vb8O2?(+U0sqZL17_05u<~zC^fZDP&6?bW4 zIn$v;2wkE9z~`%n%is)RQ_?d3@sb(39h~TfxejZvTS7flO?#fYrYYcLg@C%FUOv1B z+rm3}ILXMp6gcQ}+$xpL&|223WWGV2L{;<}vn3Iz$VmSsA@L=*hL`XPtT9atHzYm+ zw5kVcUSDT`4dIH|hci~A&l33P)p3!MJTPg6GaLvg{4wW-n{5K64 zCJ=mz-w(j*R^MNeJy)(FfO@W1DFHO4S{v(H$LRu6@Bro zrD1q#I+!*@2z0(U7HMw>^Ep4NV~rD(vjZ-@benyQlL@zK_|mdD-gyvULqywpNscFk z3MC!kzw^SYc|@%pc>3W;G$a*^?5R)j;_PvSRmmQ(-8t&1{(;$X@CRgYn{rn@Q^1mA zQyK%m3QKwg)?}>6CgHp+$>{!$g5S<1D|8|=BpH$@ENa}w? zGqy1_ereY4-MxZm$SPInGk2o72iC0idprxzFU6FB3>*z>@Lp3O;8lkWT%3A8Zgjek0&>CMFMPpQv9^dSKOgXP7i-uI zvOKeIDt+Z;@*Q$LI%|0DE(D%c`uX)%9gfOt6yck1zMn}^%Uhp!Dx0Od(ZDP=1RgEd zDgzjlVSnJf{&uMj=5c{hDc8!?-16Sv)>{S41?<06oGP8LKTaqtU;6F*NIwMmVkdHQ z!Cnne`9liT+Fd%7ZH(;JKt7FnsfLEB3chjl5cmwG=Po#%oAs#M1;y3Bs^eW3?yC8w znN^mmWPNG~-`!u-Nec2wP1or9=zco_e!>i*d|cR9Ju}^rG2X&|jmJ;dUeez&=O%pF zb5FvD$}QKO8m%C2tAWn}R&U1mNl-;OlO}nA^IR#|4Nysx-0qmfh*`rVGQJ!}wS9dp zG>P4X6|Zcu5k4Eh=YUWmKb2Va?J!W?s0sN z#5c`r`|M^|6%`K89kEqb_h3^%RLS&-~|U8a$mOJ@f_4z zP9l8?Z4r7OpdYwmT@ji^yN>aTe6hy{51ujV!&MF_@*B(^Y67Jc?6DtJ0H4J^h^MNQ z^Y$G)4IE?8L!c`Tm71`1&xfhQb&s7opc{f7fY;Gt0tpHSBAde@Jpg&w-(5|AQszmJ z0qpC+rUnioOqUOWgAx*A>2lY{!m?vTn%+?3;jl;}%i7paZ3d3?JsUAZ?~JS`0Z^md z+8e0ji>)GmVZzeTUlE76wSX6aXdA^&8)fR22$_5e$|VIuJV}ObGQOCc3;ew-7_PX1 zEl2$W6XF*NRl*HjT8yi#qQ#O#=3$Nu7Jw(EW@Qam<(NdJ&r%uDnt{p@{tci#_ zhZF0fB9DC!C=7AuwOq2Zy)ncrjGl-9#YsyMFc2|o3g1PIs z-El5R^9+8HKVrE|pZ3^{B<*thKcs%`j|IceyW^W1B!Tl&L)jW_!S_{}hCP}RmfUX| z%_SOFA1%XEa-o!GfSsSU&(scf45Y*u#-EaYV;R2y4f=E#?{QdDet>coST$92fwFX@ znJ)j(7MeY(np(q&qz;2JUuvlB*UtJHJ#_s9gwF~P$Iq1^L|s%G9y-{S1TK1>bVB<@ z)S%@dswk0EOyxVuO)J)pwKSL{(leHrTA)yR7sSwjAo)E`oD4+hS?PFqIZS8^ENIAm zUPH4WdKwf;Ozb;v%=o%Uh;FUFbv9|*;zyoR!RNuLn~af`7m4B;WqOCYV`&#QJq;PH zgX`f`cKjZ{7+a)=+#BBSkHhI#J3CqviOLhY=!12+lM;u&EgtgQ`Y~(K`!Gl_#ckhM zyzcr!8DGI<$W;`YQGlWQJA3T#Qt6rUjybc}gjjyNV~-s-`DamjwGK;G&Tc zguIDR%!Yu;dQX7SAxI0U)$ni14@T z*wv}Yi~ENell!nT0uBDbcKOxiWk0AtsvtThtKZeoe-9jrAfM2!dcB4F7t}1!v;lPCWon4}Qy58S!pdf|o zk^o9N+Vc+9+TsB6%pm7J2zi@(>QtamWeqB>PSJ!rurdeSyCJB7X2|OJ%1)t;3|;&I z;hRl2;kBJJ%p!PpA&@ofjY3o%BWADxBLY&jTl5V|>8oPWM<22BC|u_E$@B~%`&vXv z)idPs*AAs~1?xWpX28UXsJ5Z29*?0e0jQ)dAyzpLhB)o(nr65MwJz66;0)-r#M)K@ zwg))tz@L^uAJs3{YS7W7F8ctrz)=gc$2fPi7W)!ph#JlZ6Ra>V`fwJ*tQn&plR33~ z;ePOBoV@(pKL7P*kPb)>m*kI=Ml5nXIW*G4`c%8fARqUiJ7aAcB6I1jJ1F%_J>r0r zW9sZr??hQmdJZKEWo8uQLSxR>ru_W-W_buzq}5UCsG*S4LC#D4eU*9v)u$OG&{JJv zEQod|-82Pn76-k|LQU>73td|$Fb~f|1VcDMe^2KA10E7%{rWE(C5Hbg!pg|P$oT(u z2xe<&I32b8Ux%PP3H*K$H;lxcwi@o$P^eh0hcDa;okx$TWCeJ{KbpCX&v zsGk?#*VDh$+npY-pYPeupRI3XapX`WSIDr78*5ju})Ni|Q^kuW-7nD|BnJT^B<}htK_Bsy1LjW7{dy_ z2?&LMa;lTqU{BoF-U$~azj(x(V}bNUXaRoSkN^^*-o+ik0nXR~03d+@20;M4W0Aw* zX~ewoorgl(HN3a(hIa2i)YZH*fZ}WQ)g%2#eAM44kbSvSp4)2Ug0=4{?wU)><^sDy zNTk0ALu?%R6yUh)9Lq9BH@J;Lm!PvsrYNKd=?miewqdInT3Zow*Et-&Y0OU%7B)b@X6dMeU2Os==#3 zQw}-3Dk#5+Tm;ILm5(t%p25U+`XqUkRKW?LXA)(S0KztQwJKh+Z1dFRc=l#EqBO+x zc9>W`ZYuk1k%X68Ef&Bc=(vyft4V)5*GLj`J{lL+KMqjcq;svKMkP3dCVk^0q}9tD^4QPhp4_zZ4O3R!&kRhy}st%CUjcM_!I-C0P>9RLMOk*cw^WOjsfm_ zbIJc<=T~ELq(OLHtKiC$HQ;%WjaHA+TmxF4=< zd*&AFPA(P!K{U+%ELtMVO}>x+^a00iJ+0~Fh1kxEEInZh1a;p;e?$H5u5k?t3T)w~BltE}<{Y}6+ zRxy@XElr@hF9_`BU)6hSnd7mvH#5yH7Q%w7aRII>zrJGa9}j{)m1fQErsQ)pZ6+5k z25*`byv9xy`#mQZ>oMHfiUnRi0Pje^*%vc4#Tx_7r#1Ws1sZ7i5CcbO9|z7)4-)Ri z5!L&RXc3~3#UquhXI7w&k%xFtKCXdj0te6l3>n;(CtGHc<~}F*k_VdXG=Jx=JQ8jM zNH-X}5X#0l7>u*r7u}@Ek{)J=^3~V)p9O<;04}Nw^AvYS@|oB0eSy^;mWZuOY-(A# z@2aEU_8@IHw7v0j)S{Gd(2U_Kn%cUATfwk&IPi_r5fuT-vvA>A%^r3;9E=vXGH^w7Vo>^IfImt7pw zYHbNSic8E4)}HpV2;KzoM5Ec^z5Xg0kY;$ARo>2VB+|qecyHO~2EX05?1pehW z;|2$54McX69T$d3HJCFSwgfQ>;Q$BhF_yn0$+v}_(%MCj6}c%}du3uE(>4G&l9;2&W(`u!9!c3AFjzAiZg3WBaKFzL5Rvlpvj~iDA>k zlZ>mTNWV|*lFeo`(wK#whEo9Cn5zkP)+eZORySZL+g30(>;m<~1(2O(ELgURR6gx* zFALn5?wo2TB$xahh7sGP`+lgZHpa9SZqvJ3ogjv1$YgO!ITZ(x4RUHcqW*3G_NNxc zC+L@CJIUzB0{JLBWpfS8im$zn1Q=Cp8x{V(h@Lc5AQn(6VAZ))#Axnm6?tU0<|}+i zW@JR;aWvqqO$z%kjAqKeI^=9K?X!K+>}@1q=)gFkiQVxzjW9KFV|2kG+D`@9gQav0 z{@Ss8lTSu-R^Auj_+}mRBqcdtzTya)2s1Sdl151UNOMZrWmJjx2xX~$@WP~(4)CZz z9*@GY6$OO`!wj%DmjNBGi;E2Hn?=+6q08O5AvZUyTM2^`A-n}&@sG8)jhAU#JTUnN zx7S`l8R*cjXujFmS;j%tt_*JvkNuH;tJ0IYL)mp(jd@*JovAa(Te_^wllIr$>MLrv z?=@20zJnBtft+CZFO0aLjWAjrLi1rrRs0e(c89DaPL?yhp-24+q|yN<8MV&P8A_=a7qho~xb{Yb&fc;s+%vTFgE(zG_yZ@TfBC*MmgyYaT@XN;YS-M8z}}k3MB&iAQDwQq6_rmi zv8X-+p(c2J+e~S_bJGN%>7vK6`zKA40_^0qJO+W>rI#MC`_Go9kQ(C@wlc*2Wn+x_>MC6!eY5(ZM=W7?l6wYyt9kH&oGN`HSla=zu8aqy0mNW!H;yH=6e(^FY+ae&k(8Nt4)XiDRW)MfVw+K?P?T?eXk zWjYP_Oy(b$G#slc{#_RO7!nspQ~u;cJ}h`wYLm2ds^G}nElSZLOwlbW(N6xYH*!?F zFmiN|jZ}pZK>+~^8jc8_@}cMVhIiE5`}E&7i2oCvo`IR;KYTC4X&S$M1_=JUpHx9T z%3zkNND?^LN*mfV38sfGxgEoTR*Nl28ji%2;B$Dp7;q+&olnWY!?gJExQ-!%cI-mG z0H8VOJ-Vj^h=0I#Eib;=o_stq-SqkzaUP4HU)&?2S2Cx5pS{b=VWB;?7mmy)7Dsju zqYXB0A^UC4Edp9vX^L zG03A2bS+Tek`wq>nInuR05-5xOmZhxO^J^0(39 zmzKNAQkHsEfxTLo(nb2sHBp?KI?hcyfI*vJ_wAuQQtHrKF=6#mGB;g$HrZdK#8X}j zr Ig&~p4*-ij?l8)+-H9JSm$7u&Nj=9UUxPiU$>UX~Y2;T{jth6gfU>CP)Hp}Q6C zB9R#|S=IS#kH?K2b5NI=zdIf*Wu-p+CucG(5!6O#d?3`0OQEMdo=@AYY*&ufX0=~m6ZRPka9bUWlIOyU1H-7?PWWv6$w%ub$zgVrMle$*l z1uW;2`qY){a~^^Hi^;yd5(g=x=e&4be;clyDu;^@p-qGfUMp)O&fesd&uqe|pHB z;x#yoG{c}gH}1S%#yoMPg4bB!ka1HKsQ0D=d-3%UfDbaGl9(j4MjDJ+ET}3PnHgT# zbE6&~N8HK+xP^=&vt4_8gD){Uij9^^*X2?=1ABLO4lgf!hbl0y2h%P5QYKb4uVsbm zIYQa4uOCx~zx&*cZkNz!2_TN*huxAim6Ste@`k*`n&pM!V{r${$VvYV5C>vGB5By# z2C{sFb!KRbE7?#v`-*vTp=o5j=ReAz$EBIVSC&3I&vPK zcmAoFe_!)Q+I=Pw!#e#Pzgvlo6^SC!oWbebJfL6o9-&^3 z;GnTB)3vy{Ti+3O(4dO?r`=;^iPSHQD$6ULAJRGi&P6E>syVYrzGql&@z-9@XQu8;CdS4L8Zu19-Zm0~gUr^~VJ^Fny&MNeI{oLq%Ydis(v1-AW+tCL~c0PY%bRSmqLf z77=f2vJQXNdQ=#v=2F@!_F1K%RK;0&E{qsdgv}?fQTV8!N*z)p^&Ri!*TBN6wn!B3i{I}KNf1-FZ(6jxgTVKuUw*m|0yGO4-7DcuG zKq~fTF@i=Yw8}yO4ZRNLui<SjCcL?b|nwcfq>xh?N z=&Wfn;`{}XkKOJrU;p@!q&ah82_XSNPskJEiT$th&D>RgQIH30`B{)R<|N{jI~?Tk z=`okcv(ALK1`bl$@L^$i`3`Th?_~~~U3m@=2~$<`JlP|?yruWJ zCbc_{^kfQfr>W*&-}`&-Jd;uInt@_O_AAKMT&kcmCqSjDdviRq&}cxpN9$$5eHwIk z9OP8??r+)wr^`Pwd*OrIn!lIKsRQwGB{*5YnUD}-P=s%Dyd+r!luJ;oq&cZn0|tQM z>%YeOi|3R4J`ZViWeC@fNZcqWCCuodQP(CJ~m3QYJhNU@7N*HD+btn7^9iJyTb!QHa`19<3 z`OIvZt&Yn<5MFF6CmS1;MfRU+G{~75=m+*r9(2yde9ERsWBG>CgqaL` z;_x>`HqTOL88&L;I`w!K|bMwdsd+&yv(+IY^zdRg|@4p3_w zc5Jn3DQMnHk9w{cD#a(wdVFrtox)%pL!4Q5EBJ4&FC;O`oK*nh4<~r|1U#W3N$HSv z>Xqy8`)cCW+k$nxAJ=Mrz-#^hkpE@)F#b>4B?A-3e`uGwRc7E3>F9- znyw43CGE194RO>Kc#>wL(1s2d(Dk6!i|2l~@g$oFz&37fD;?ga1x}0cuj@p(A$kWA zP=0G##ygGaY@UBSK?)(ncC0}+g_t0_)*sk@j(xIh6j{(4;@*w#KCjI9BY~L_$q)Du z3HvQ0)-mU@K>t;UFBa;M** zznT5kLdNmSz<5rz{9~uYgNL5{i-p9o)+6m$iDdEY2SC}mC>o;?Mst1c!lPi1Su_&6 z@bVdnj~-Pxgt=th4~|0(m_(Gw48!BWgaKVE*w|7-*ae}P(te8-W#Q5P1@~0f?~=X~ zZqp_gzx6Ck7V%%=Vdc#WWdW9qU#pGnp*+xTe>wPH3X`?BASAjw`Kb|4zUR%}(A_Bj zkncGeAB-$4 zFxBmG|M0HW>iq1$?K%jq^TESDi6Z!~B<3=PiWk~-e?uU;sdlbFL@4j{1MHw?knarW zyh~Clxnc!#F|khZ+$NpN<@#Tp*=g@5oXvskpUOi2&s3mq>GlR|WV>p~RAmX1#d6Kf z?=2wh1%ZrmMyUtRO+X#%XVc2-wwh|xMp^BemgXI4?cm-td1k{g%Qry7!X74J9Ruxq zHZtuI&K4r7g(zIt_pU(b?9{2?0?H>-MAZaiv@Q|2nbg!R2!7h3LlbkfEt;`I=2&+o1Fa z2f~J;MAdyl)46neI7JnN+x&eDQcrn@x02Tkg4j7wNa$SyR**&I(ZLep@QeDQauLBg zq`*F#5J!OGRAN1LbrTYy58VCRqtxvMvH;j1Sctn2L$NHDTvQ04*7L>7WcA-6rebF| zSOk*{vFUa3j=OX?OhOwt3P%@W84LeV&p^W=K$_s^ecV9@LJklE^;b1Ixc84m>wE4G z>5EOVm$K)fff#{M>*KWsY7nVuCs3V{v=bi@!VLZpq2*d4k}j@iy75KZa7XKa+w)-f zWp$@(udi2*k5;1jOz7pOTT#VOFxIGCe6)8Ipo}Kt(5|&K3?ARF3sFOLcN^f)hl4yb z!f%C}bIEmE-FA1H0KwbxpEXSDK*WPU;@QL9@ZGf1sJgNtLB@r_pn5)!bz4gd6pGm) zT5QWp4YEC-$16I#peZ+2llt-eMmzh$tsI6*@4|p_|V!V`OLd-j0m%yw|kl z$@AxS;ep-9f>tE6Sk7K|zkt=?bHxAK*#6%Xh5t`OVzlc212g{Ru^Tk?QMxe!v|Q;*K>BeCtQKxyzIIyDfpHf*E#?w968N8iS6Jle|C;~qWD&o zvVSe^a9?+4Kh4;E;T96-3D`CG9|w5cdnTU2pMG#itA*dd2hm~g{5U;8=N;o_NFcT8 z?;}%KMvf!R(LQF2Hc{)LM&m|IT;^OKSx5VEFMlZ zuuKm%$^z`vC{W6ZYHWaNYzGY`;l&ES^f6H#WiZ!96Q0<3wqL>Xcf4l)um=l(zM*D_ z;SEKw&#$9$_{~mfq>ktjXg{q=i(Jz8D_~${sWX?48@P2@@&2o8sSd1iEAgU{x093+P7Po6fHw`ru+j(XXyWU^-qNam*ZZyu z4ypu0#*<6yM-I%_t-(J5u+IRX-0-AS*3^fSm=it&+dgdp$>-j|bt>vbsWvkVHUIqK zjTq^i)z3BcFQh+lSiwN^&IojpmCU?DTKM2c;qgY?;MG?7x?7c6UkCGXdiEUa9sU3& zqKQdNbF9+JW!pvfX7e}lOZ(!u+b-4K?84A(mdf7(lb+6O9)7iT+7rXe7jgY1^}lH6 z#N{tS?%wWnAgh{N?ihDv~hCrh{{Q^TS-$ve`_mcFSF&IVkQlaS zo=tSgusOvzgbLNjf#tmGVu_67oQ4I>}R3!!646fHCu8Bu^+;}vK4v}F0JE6IC*NL z$No)!ghB)&BYFjaqXcOXO#v-KbdD08kP(Iq#nmXHe5^qlh?|?VXj%t@en(WF0FAhcEG7kspqC!rO(YuM2(R*t22o$n8Yi-d~$K`*|(5e0BTJ)|VJ$ z&N6(efW97gL^yf1tGpWwhD17T)@4c={9GK&5$qwm>`{n(<)J!mgMDAJ{Q$zZKUMs< z`Sib?Y1vu+qro*=UE){qMEOPHoROmui4anZ3(86SAI9DxNR)W#;%wWtZR@se^R{i< zwr$(CZQC|)+dch7%wpceTf9w0R7CBn;#B@K^PFF!UW9oD%S^cv3`#m)f6YdwCf!?^fO>S@XLQ;Dl;eZ>Q_J=y@bcml%#aS5OTWT3rRf~|R2#=nu-+!{Ce z78yR`n@a~%N*58eT{<7m(n3@W`a$7@zSa)q*}}iawb{&aH`N^-~@=9KzzS8(P=@WZ+Vy#eZrI;RErB-ty!O1Yq#$0 zJ6FBy6ubBmI;u(Na{PEH?5h?LTlc4-t8nu@+16_!fDBejtRW^iQXj!bO8Gl%94&ev z(xh0$1}Nh_PZpMiBpo9*MTg8UeFm#Q^A)&UpHQ zQPet{z1ua*PNqoAH0lm&EmyyPQR=a*7a%@xhTBhh8Dq&mZNY_k&nd=OldA=Cg4o|H zpnKpy=?Uvpy{j^{wPSf^KAB`ogjhA7GNt2$Mb1RQbL^LS-O&c)*d3P}HK+g9luvY* zmhpfpia^7sz3Yg7%xNXtXyLj5k~zXlW3Th7p{*kK9mvD0 z$x{I2v*g8}YO3Af@5KQpE@yH!P`hD)M!ZUB_<5WRIit2Uocj$B$sVxN6j#b8s321f zZrdOnfj#_2?7q}a*D6`0l{^b>Zf~jXZkW9Y>1C{DkE?l*nYUC*cn5_la;?{3a;C2C)h#rJW5w z+=YiDaL5F4Ms^aUn@Si1^9P`D@{Alb!11zhoXrXJsip|U`Egf1UweK6 zv@P5efAFmzHuQmR+wB}%yww5p71Ez#%YiwR4=KW0G-Fy;_vF9NI(t->Y_`8 zJBA)X;H68Z>S>eg*SE9`UAlmj01}72g%yCY!F8j~=v|Hwna*$p26wh1j}Ci{_8)Fw z0_GuoKkmfK@}cbLMy-6TuCmknH**~}JE;2GwHzZTASi|c-cB2pwQ{|%zN#;i+YKMB0valze?IYz;R>&y9ctq#qh>595kZAr%)4}3kv|H%W zlBa3{U1K_g!}7kE!r|}G=YCWw|ApRN2e;P0e<6JoBQT#1xpPzVGjP`7m&0GB&VA#y zH`E!YfR(fmThhY4MFgj@V{-7c+zs)c0n@N8{po!*F`oT-+Dm4#)hR{S^0<@kuP$iR z{x=wa)L@y~p8}w}L!9O-7farz_u(<9QK!^Fq@uX8EC_xX8h zFjY^{xtjisN3qPRLDX=I%Ba8fZ!Ka%zqi8zOBrjK#BsQk!lF*`>Nsm1;Yr1O{TKZ; zE#Y${8^1JxEk!-36?`DXUlNC@MXZkg?A?D>0jW7sIYPgBZsP(!e!L_+E*d5 zOglI~^-HW_6)fpb*FH)EZ9|{#e#KeY4eMv`%N!BQ_wWR zj7K7^aJSIwG@Iv=?e?yK7TP)9pmJIdQ_KQ+d!4x}h*W!i2SR77`|S~$atP&My619V zfZmC92le~zRZHD|O?Y|=62~egINVgX)b$hrN*LNUzipZd$Jx~vCE5RUtkHgJe4XId zcFEyE{sNgixJii?uoKp(yE#aR><#v_J@HKez(}DJ)Z9BwHH3xl`Wq00i$ZDOC081# zRy&88q=|n1eA~e$n~HqPU}+p5*Tm^o$)o3@>+emDTxqPLMp!GMn;4|BYEWvfQ(WI{ zBJUHb#u2?P6fn!W70l1}sy@jcPh?MXaPJ5q1KlS$B9XA6TpNtb)X~qcOK}ebqfP7E z{S&h*>Qe8@d>F_m3b>Qc?b?gaMa@sIAlKANs8p89U_vvW*7PR#anq^QTQwGp9}$2MEw%;2t03 zun5O7m6mLjXtrt-Gmdbl4{=0gcN*N;z`0>EiMnya5$#N1nxPd2`0wbeesQte%VQXy zw6do2#&UXD*L>l_=0ug9u`Nk03 zV0}s^EovNXn9oI`(<#QA7Dnva1W%(dIXk!e07o}Ps4EImZI6GZVsnEg?uv%1iRFW$ z`{o#Vm19}eUKDpswlA24p?XT6&)Do>1m3dbfY1WB6`d8sj6LG&g)j@M<%7vT;g5sX zz~Wu_7xU$Nyn&=#GIiTR7lDP_uEIAxY);?S`|0EIwr|CXT#O+aL)n9UIjA8&L2Clb zT09lu+IXc`wFSt{`Th1u8m0;PFW8J5@P2hMHQ;2KSx=6S8_y+SA3Jrvg6++9PYFGD zZ$h{9RxVcOKtfIbU)R9ji&gDrd-ey79vs}@=d8%dKvRNh=06N8RpFv7GITPl{;w>~ zv?CZ}NSPV1d7ceCd>lTTtnTF!HZj7$=p$7dQ_s*&8+3X7GO4I9n9}*)jvya)$uR|C z@+)fQK5dd>3gU`v%I4JgWedbv;1y1`o67ybCE1F$n1)@Z8G@1=*y|-M%d~TVmw|t6 zb;UeC^$+M^e0MpV zF$AhzX9}Trinl1DaM{5B&Bq!xXq5VI%E#LZ{7cCU~{x=K78 zy;_bs1qM5!N_(r+odHKv7L_BRQYs7R>))TQ@g`bL`*|C2fbfO=7jN#P_OUZ$B8m{c z0FdAx3|hSTOh^xB8AV};84RIB)UH_ZDCQWl$zI?H(i44Q2;=@Q59Dv=H(+gl2<%#`vR0uB#h?taC#eXQ+*7hsS-#8M8qtzvx%281qs35J{@o15xu#_YQI`va17 z(k^{LeKfpa*BTmkCE2a|Nf2mWjHz6S>!L ze_gE=lK);+T85*@WKkZBF)D?m&rP-FPFgIYL~CX!T*?sv4RRR3jG9O)1gM&{*n#cj zP+~F{+g9LPUtG*!e7@RX_24b(GlUP(%%}MtTdSo;HyiWv3~F3t%JNdhGGHyCRy9st zXYovPk5C3Smd+05PY@=g=~-M~dLfG{gQ#p%xi*rR;TraW-^t$sa$o9hO#7+XFfYNh zVtO=HTnGsdlLmSzNE;@cNL9mb5i?BgW#~B@@F(4$bNL2IK+03UH0%-TmI);m^>;DS zt1f$J2j_WjesUkB5bJ(FKW|?RPS>if0`mgmn+&}lMgE3g1^uzrW}w0+{V4(Ab$`2oVeLqv8Ivz#O{^ ze;vF%eE@N?ftNG#-%*uGm{n@_wHpbKC%5UnW2|+;EyUx@7*S>09*Y}?>>}f6mglQ5 z>6rr1>;w~cu%E4ijR%*?A6$go;GeCToBRDGt1$xR$w?+=Vw}>fv4f#QPGoYP7D#`@v+Q`08fs^X zR$zS~4qk5mL^-j4Jk(Nm0WPBE1bO|p)?kH#hQ1=oBt-p`_%`Yn7M=yVZ{u5!%n`R} zq*<}TkSuW@3J1#Yc1AD{j}+2vXM@6L41hB|!TzZaC6qV$s+kI!n9#_K7tSwQRgvyr z>oA0Ud5dL{?Ud|#kPnq(&NL|G<-7GX5$=`wgd;5?ANFR+QHRp%>%)Bw7-)1V!hU*6 zkv4YHbNDR0pUQ{G#aeFRyLQL=Yw&d=GR zmYT4eC&3l2RngPgHzd?&{W?|GgRb){*3!V%LB)U5+N9T>jt5koZJ6It-sPRqQ0P~? z-W;&7XiFru+JakCD;2cw8vlIv%+91@j`;aK8Azme-L{yGE(m}pjf##!mkX{%g4!pZ7P$tyW5T2_1)H%)# z?-&4kvs`%VA3*Y;!zBT7VLe9ZTVc#kncQR_e&iGHRQMZK_f?d)MmZt+Rr;bJ1d_6t z=+m7>d@&u>=k|r?3o;QTR+I$ty5G11ZUy1i-3pIQwCe0|v{N`D$Pk}CPPnHyneE+V zf`H3gYfuRt)+ZZMR+?q<&61wQTi>78P#Y8#Q!J^U=`t1>A;olai3iu6Cz<68^&6@K zbq02)e;Ox2#c_$hL4bdr0?mgAOK?>4{-TLhh;e|7MrGWMP$-~~EzWs7bFQX`*c{My z1D&qHt5fL>mFS*g+C`3&q9uCPe1Az2J98E+R*oMNd4%Nq(_n9}i zGPU=OGmywWtFCNg(%+hwOj*TOE0pGEQv`O`Od`?;nj?c}A%rL~Mc#y9zP3zuVEzdV z_%2_tHFn#L!B#hv0xn&}Oe;cx<}qO?WCe<=B{}c%#@C(d0pQiwt#NQS)2;N_>S6d? z>#3_)^)GB+&S|EOt*=}3>KdzY>UGG=Zhm*yO*N)9ZMW2o{Rcvm0y_SNwB!BS=HdBAP{JCo z*Yd2higu;roNRBJ0#+uL^ZVTnP)tRw9}}-zWZ9%gWcA`}c4{S7k>1t`rb_>8*}J)2 zeUp#ApnBJ4*{3>~k(8!gc5V7;;3kuMwYqp!%|R@+tNvh`y@J3Wt#wbU>g3sjbuzB5 z*5#>_GqZ+?g{%P%UWF{9mdQa}IF!n;V^L)!KfR0zxS<@Mc}_N|Kc|8TxNZBubZ(mg zxYs(Ol)54c##J`j=wRt}P$V>%AtalJZ30(o;x*YhbOslj@VeAl?AW8}X6~%$24FbO z%EQgBD4Z8r(UYv`pl)WaP$(bEAjYp;Nok>xRRotXT{FCQ|~8I z{i&Rw$xLuo?CZm4Go2CnHwaXsh1vN*(YD0ESF`L5@?bOEL5L-8cvJ8`v$g)~Uf%%l z*?u=2OKMzh&nl0OTP=2D;~%mJ#d6ZHMB(o+!qJ8$QQ!C9^$%+=!vCT}EdLu*gOTk& zbx2j}w=?{Iu=F$WX6le4aY;d2P^>&26q!R!Et)2#ZWFZ>HjklBeqS$BamkeKip4A_ zYy|v4+aC9X`O`2l7$5ElkWUR|+`?S8`OX(D(X2(masrCcy z@<;VPF+W?MSYKV?ykac0_h%^PGRZFFOt`p(QSiAT1d6b;OQ=P@me zKJTO}SxMQn_!fu96%JR{4`;2wVrg~@G1D_wSc@+)cbzZpK^5x$6NwU?^2)}tAET=! z-Z$2K>8F_V@c*I)xPP(fzC&z6Srqt_o_3#KYks~ro%ha{DLI|b{m@=NA8gMJ>#m*pfx;N<#olqXnbYta-57cfd79G)kqDESCv>500}6iLW< z5SOI-`Q>x||56~^O<2+8_taGB1)wS(Z2dTDIBN1N0@?ZP#KEVxoSApz0l8P2m$@DAsUk14uvreBi8C zq+eqJlzK6hYB)D2S_({-ich`de%EOHwI2Z?OlrQ-A9*#EV!xiYot+B~edNX7w%lHL7&M;@9ZpN1 zw!6{4)f47bLSz{ZYu0DGzqY?sKM4Aus4DbzK`s{^s}{I;RnTl{^?f6GZoTP-;L3T1 zfUI?P{jZ&k-ly62(r&=^z%w6fYG$@>{Su+wu@M}dG6hL0`h}kflO+|;h^Y)%Jl?Su z1N_pg?G!kGY0)~!(i)&|65J_5kOO`{0e18qaiE0SQmdd{Ml@u__8o-*Ri{}_P3t`$ zXCx1qC+Pm+q=Aj+Xj~Atu-Ym;YMDv)fm{)I9Sj|H}IKGdGvk=L1H3U{Syy6rJ=k zFe^gd`R++zL=rinDUFhRvy%=&4z=?|um?!rPNh2lJS$0h3c&q8Q52hvQOydLc(oRc*Lf!b1HNHOy2PTJ0}+7y|VSilew z+(4v_XT;x1&g}4-iY@&Qp}2kZl&vN=>StM~P&mZtiYAJYY*4p=Hrb-=)?(nB4V~2o zcz88;0Z}b`!3#TGw;P)^Th|vSDbnf05?C$FCq zyWP_5y345xANe0RWq9L^bI%-nbzQuTVI;kjcD?{!iiMYJT9#`{nFEuO7=BzV^#ha; zpRS*QDdYk@mb9_A_gA#~X8O*o0zcS|`SLvf-PozIk@&V%R@Hg9_EzIvFCxQDm01y| zWcn3`700U~FtgW+mZL&`ZFb`CEEH$N*}DUiTVKyLfaN0=rRUk1b-ob8J-?LT+MlHl z&QU&$AVc6ubo#8Rg`bRXaSN%j@w7$U-TyxJ-eN=WIECpBZquroD^C{AVmzJ}iZ9^) zIH$i-1gX@H%j(f>W8x8j4QA)Pu5nu!aD3+>4U90n&u$!{gta8oUFlmRTHMC2NvN?u zx;_gS4baCJ-3vXwhhtg*m}w7Cm0dL!B>;oAkj9*yfyhk)G*U#(eq@hBl#MVQ^cZvn zS30iFW$^|JPaEjKHp(s_1>eh+gpn=uvmYQjfcj-h4C6a1;7Zi4g?{a5?)4)X7UpS z?!+!8@CV+3!8K&)*eX0F(VMa*l(}hX&Glvnn@AtCr01oH3aZV1yA%a~h&9{Da%84%1ofN?l zPTX=#$iX0-3wyIFs5$6jVoF^vY;kN$-zitX78#=l6u9zPIi=U0)ZYpEm-XZp(MsFhU+7U%cC}GJewr7?q8u( zMB(+4rfdyV5y0L!E1z6S29uZ?PIp}(1=xWCx8&Sj*>T}4-nK-8pN)Cpk78vc&e{^- zVB^W?j@0~$S2FFnhEkz{xXsOCS8Uw_8%k*hO#M$iY9_qwuU2Axyh>PDJUk}nN=`Qv zB|${^V>X?Z3`T;1q90ctz$lGSV2fqg>F4-Y!B70}zDvj}_vYq#a#x8DY4AWuN6`bX z9Y&UV2`1D_+2rXHkptPiga~po5KbUp#0F6DW!*q@{)LNEZFjydlwB7?YXlGu9w*jb z0CW-VxRNH6#lJ6zZJNg%-`Eb$K?ZjR|J-WjjT+OQLa|RR&EP$^fJDx+6N}Ne+x$@_ z{*h&yr!p{`DE0!B`311wzkq|pSi!+5?{=QAqm6%OXtJqlSZ!?ay9LDt8-!UsrEH;xu3g*aN0Je!Y6*;%}>Z~0WFZ#!jqO<`r zL>sqcnbSpGv<691iWl?Vsv}OLqplYzQaKMdM`@QkMZ;I(s&TOt_PS`=^rj< zh+Y`dtZ;l1_@z;=7FVLC7U)(ck^|*8j$`oN9Bh0&k)2YNlfM|gqi_LFLUrCQ%%dZ& zfnyD%>;Q8YFe$>&vbb916obb{pz@xC5soRbmkV@lbnP9Ow~Hi#3_QbK2KgvGUhZ3$ z*6TcvuDxB}ZI(Xm>Zc1<;Nyia-)v#eK4q@QEqHeaR<72ad}D|>Y`df@`^YQ+0VLO`5S08YB}3)hybYev4>QkWcq|mp!kQeXw#W0ipi^V94rh)HeK$7VA8I! znE5>J+$#?@Is?WvkmDJ2Qty($Gl_-9kv@`h%<8E^7-prlPd7rd;~@sn$z^vG<|j;V z4pvU_L?tE0Pv`#S$x;-MVrV4<s$JIp#G*w=Q!fwbhgb!vXXif5Ts2#v3&{hf}TFM1q5y` za?vWsoBlvk@VmV^L_8cj0C>1I!u+itqkaAXKyZS6Jxw&-_SY~~Wf+;mi`)(bu{1;A-sW61G6^j86wytQ@UHa> zEJJv9_~$b&;{fx35~IA$ZmgRp7N^NWq>2~j@0~T)y8#&e@A$Y{q!riKV-}by8Muy9OVL07Lmo$c)-WST_9PhvL26|sZ z$CUq88~#tdHzpR=|KLa-Ye*&j)`mT&YSPC@vJFDW0J+DB#+o%U)<&EguL%Qrs@#PlSNYM8pb5l_~ql0N~aj3_E!RiT}UW?|>;6a9Pr=jcv7*@J-2 z-Es)u2r@$pN{p#{dV2Dwk`d>mJf)B>eEsb&0xMGA%%6x|5nH~4d zIGt7hm|^Ig_wp0?Xh3+gUg2oy3%4Fq zudz|n%9A3cz#qPTLPNP1VM|ueEwT927ZslhLY9^qm@YP-6G_4L8W47>&ou7$)oE(Q zbDNSiX_)X@tG{-imb9W=1{JW37wH&vWqu6wtw*>6^8WkVl@3{Uq-w*xj$JVKdqn7< zf}%$7{jt;35GqGppbN*{C2#JhaqQfzj>9~xiO!jnY_eYI4YZlC&cQ=r`A@xhyQVo? zQ-tC0_`<6;B^C@gv_i1r14|z^2HlJ%@`LSeb4|`Jo)ZzD*Lfgm3=X#effsoPemJZ; zp$IV4-`1Va@@myHo?ZVq{P~Ro;(a2!ienQY9b^JIvly!iraUV~!2gb)pW55nSmbTt zr#B3$LTQyklae_$3}Sz=0JTFDy^zCcPtYmsC1aCG4;F8wLnWs{i4=v(kN6;Unh=r> zl04rhz*u?#j(i2pTk=N#OC0jFjVYbwXy65%9A0{H8K~-*-~FF^tX31kk?lRNpTzUgF{b9DV~m4psZwYQ&5!&*ntVB7{+<{`K#KWjeqYS^;=6Wg6a0( z)nsnmPtZm)aE@Z6;y4g&?>c)AT8&-$L7kp5t(;!l>A8WKQz+Nu?%i-&mt5VN1Kp@1 zj1ZeNsXOgm2L{I%`N{7pdDRM5_Mek6#V8u8*qK{oCZ=3W@>H?794fj0`q01t`z81Id7?4h-&{6sKkJ`X z(roiw1DKZ*UPBDmtPfAfCAw?f?p*t9#O=%16 z&}p}0Y~wd~qRwJG;CnOhnhgu=mzp?^J6bCTxrTn;_#WE#5{HT*6fQm8=Vhc~suMi& zdLG#UTA3-l83-h(S4wjE3Wg?C&pg$~G+p>(VQ*yz#1Lc`I3+%qZT{9i?thY2OoIs0 zf*s(EGI|6A_oF=vP7pe}T?k;tyeGmE0^zzAsF6aO9TIhdk$U{sdYQ{oW{R55@>nv)Fy z?}311tnaw8((=0jf`B0cyYWNyyfCjkb>%$n-5hDY;-X!ddgZ*y!tlZ*I`xNgq&f1m z=u7vBQv_z7l^SHdPp6?EvM?`1OD59mgcy;`Edp;=3llkf@PT_6PZF|$4#X2hN-_{y zk0nsDQA#&-RQS)XkrIUsSHo1sj*Rhjcd12vdd?XE*`y#t~Yh{-k=jaO6aZ zz_Cy?9dYuH?pDe&Iih1fc`kyyG@xf?zQN~6SljSWP zm*v*Hd09@nj?3i77_Ptz-W(0#C^EHZzmfA0S~kK|fw?*no0z@%$h+U7CVqnp&1nJbZ}90qSGWqTk0SQk=@icp_CWdv$7XzZxI#?`g(H5m}!zfrcr$-l+9x>5nhk` z7o|maKRCUq+vEE+{WMJ$<*I!E>QVTN!@Ch05gaD2iCqx4Y1SV{#pt+|kqB{=G?BTtmpo$>XfeiW*w} zt)8a1=X(yS)$8v1F$7%JQkF%#1lm-jx0&`$xbvp8^SP=) zBhm}pl&FW0euBtXMS;&TvSWn^ymcD<>~>vPC9Y0KeT&;6 zyU%xRp`%5|BVrwjN>IN>1DPy7BQJB)6zjK`KXepCct{atH^M-DkkKz1B;zgwklaIk z#KBhbtkiF;#`BwZYDZ^loLFgv&)J#pso7U&+qarqHXch-4%_K{e(l_3bXYF%wLWgI z!(MEuyoClH%g*!Jd^&u7Sn8VHm_=#zzU|>^^oZl=&{St@O171_nN&BRGidn`k z0UNC88lATYu?fGwmJ zsCj^^H)W#>bcZ5sSsuPZ=&q&hoPd@&nhX;H7Y!2V>(T<^LZHZ2RZB^c4naG+cYQK^ zjHKA8zk|TjJy2aUFiZh+1qck_JbMZbe57fvFqZj1dix<8K1&R$91N~(_WohD^%P)5 z(6qp#RmL~BJsGgYvO@fOj$j+iVou=@mG3Q#M!Zpld5#=c__pr&RON%@`#tbz(BGk&~L_h9)8&ib@v$8jn zaLMsb#pxjSiRF$?viR5yo&K1dAgqwc1cB>rUR$=d(qnNNOu2=HZv(ud4sX2J?47!x znV`1WK{s!+I&ixdA!Gjr*wpu#xp$xtll#geN4J2w+UGRd)i4NxKL1 z{tIK|&#l+YC*}AUs#}SnmWF%)Q|KR4~%?#WkdoF%9I`0BA8FQL^) zc}wYZ;xljW?98-3heXA5ZK+>%#AY*gTe4m2+rt#8qpH{~A+(6+^Z@gLv-g3_k>oYY ztZc7GTD!uig(679mm*8-en&bAw}=^ zXxDj$GaPB5*eb6v8jE%U>&~4_$O!#P+JQ57!Xb5|h&i{Pou5`k3skrc>+ML!nI03go!p+mbx zIy5rxa;04_WnxJR;ae5CSXVk-57(blW(h+XO&ogn5 zl^OU@+IortvrF|M6ycy*2?DH?m%_93Bi*BNDnPjU2Ed!Ek9+PhL`!2v3mO8l4+kCS zJE%;Z%ZsgM_8dx!Jo1X=P|#~jf2~J4HDWlEoV7EDsCY0^EHzq49e*5ll>Q5@II96! z^;U(^n_etULeHc#gyl$T0`UntAqJW1Q05l!fHeDRzm)kRS>!tpBikX=gN~TM+b-<^ zn}k*?;_(Hvpd^{SJNn7HAV=p{H(;GoXP2>mBw{0m)G9-4vpWmqo^^P#mA=iyJ$YR} z>63LBrJQS;k~OvDMW%2yK2C)fYt_a_?piPh-_J)7O&+rar1FfKI))2?kLuyrSBM_p znLGG%t1QR|-He|8QjXyKY0ohTqc#Zeg3g9#({|?o!Gx$*ixv>o`iS93f+t)@TYYE3 zj(Bz2PDlVU^hJH1nD2XpZo$y-(W(w9msmW{8mMO|AwwL z)VGTq_N`1Wdk-xB`jOMKcIba{5yUH%~!eDP|6A~~(tg$P&7xx=8E_lUgGPw171#@6}Qpw7kBcFoX% zHHc?(KLRlG5B&)@Rnq7mGCt|cKe+cXbjtrKYs|!dDL!R)ajk8n{cj+PYB4D$2HVjf z_{XH6p^aimY<3GQZM!ISfk}fe6vP5`%YB(?pDCd??hv)sWb~3GSpDt4Yu2Os*?DCl zg>c%Q9k>hsyrFrCv+MAwxKI$GU{HHh{ykiZi#&&400o+#g0A4pB=ZfzR{pY0A+_gsPOHlot$|*qP;N+LO-8i`}B%pKA>Q_eo=z zj?Jfo-T0EPuSwHDbm#$^TiSLPeR;(GV67qDX3f?xMW%$uWr8}%fgT$i+u zw}BU_r3arr`sevl=C52y%;=w<=BP?oI&EgeC&dzceZ&mG_#q8Z{M5W9$Rx$-sw5-< z>SBV7E5KHq#%d8k{KbHrX;B1rD~=w>Epy6J)T=j>^5?Ktx<*YcsBto-)#vHTP_K=4 z_GLW#6ZdQ}GY7pCjSk_RF`B-*44p|-o?ZsN&wP;>)TIiAocV=&V9h=2;adE9r$LXJQk3 z!wK4x5M1rQ{46}9%nu&h^*FlNxv(RJb8XDrxdJ0Hz%v}|4uh>}*XPD2TA1ibA%D0$ zh-S}Oj1F9eR7VZ+3$3*Ib*c~8EOlZ#Zu9vIKJA!%jf$=v&_ueETd?)o;LyHQ1tIaaMK<{l})siZ~W8yOR}*%-vwA}JG0x1&p5kE`J$!(cie+HO_PU*QB8`k ztHVpFAud4&5QPHSy_FwYPt$=l5S>LNPfZ+gwQniS!S`n`bISGbnRQ?kz9qHCq7irf zvB$4M)5N;riGfztOf)g&0%p5$Ld8h(jU@oh=@!0M|P z<$gt8yAYfPH-S=VmVWoT@>r*XYEftb+(`0${=Cyyd*+nMKc)m8Pw?+V)A~_UkxkgB zuKsG_w5ZDNrc8SH`yw!6)qyc^m9LY;gL3%`B(y*~=O4a5c=bF%gK*9Z0jNdwVfm!z z2kQJb#g-tLb~ViT$~95wFyT0i`vxo?f6%;aCdpJ1fe3N)G&h_VRD-DyEh&T8V2$X= zbj?TS_5U1HB#>Hk%jI(oH%fDQq)^_(P^$Ije2cBb9pMSPT=_?}WD`c=?2fu8ho44= zeN^ z4?1;31BGHjL=(6chUGi=K$Ll~c8*H;0KDhFPFp0DSI@>!zRzj0A>Y0YC`%!{2C>cF zfpji*CV4Hlv9>32sry)=3sNkVGawtvBeG!G~pyq z;gy3p&^)XbH~ahG4k_^X2?yZQZ`!NK2s+!H3%BDc;c>ET|gO(M()sAX1Uwi)3>=T=uYF^HQY z0G~)w)h}!i1OUML>L&qnhr5)^$>WN0omrE+Uni2V>ckUGrHRDW5c4$mV@}}5v)S8i zRDxZS-;5F;08Ac*yOxr!*8}{ZpW4WO6a4>)HN(in@gH8b=(J6T->txq+Ygkk$ShTw zy1s)rS^BJ|o1#Ktn~ElMXH!YdrYT`0m4t|GZf}7ARSIPF;rl3lIyiinA5a34A2M$J zn?YpQn_N=%%%fwXw{IZi8O2GfhBg_F5>>Lr=U_>+$zC%02WNLM4+md1R3I`}_vBVm zM#_t4P1%b-w!808)7m$??C#B!E$bVrbT893yRY}wDmL$2u080C5r*DCJ1zNe5J z7B^Dt#~!`wcsq`88|baaF7KSR6CP{CE<99|*1N_&N~)6^t!=jh3n^5mF#xtIRbC7m zC5=+LFLalgFIj62Ix-iPjS!b}Db}e9cQrd{2eS?DZV)N}0ilFZs-=W!Ca)?8whbjz zc-(_;$U*${tlNXPJwG+{kZGoM-+sT(*->7#MloQpy&fJOXji-%zt%KvU zIa4ZmN1sI>KI^B>-g^#?mQZ}0SyvMVo7|Wymp(m~5x))**&4^vv-kQ28B%<>0i8#> zj&HlPRHep9F$;c(YSgt)%@DrFgiTc43Pz)v3_Fu=Go6?z2R$9*PQym$>e8s#(-}i` zHqhZTVVizu8bkFp{dx?v5(_;%{qm)z6GpV5I^hUa4a;VddbbD^G&%o)16sE)@xc>eN6Nr%0P}37(;!WNka3QIKp7i zY%{_TP#Y~E>Iz!a)GkGsCV-U|6h@k5P=%p7qf=VMK_qPiV0EsL<1O3bIIyU^I*{n$ z*ePNfy9H>2bPlvbv|{-IMRi_DB0BfWDW;lg#1vg0YZ$3g9RB7?%JXU$P=fAYl#~mD zD^fmZ3Gb$YhHnTqK_yCu@PX=b^2cFt!(aS@3FydN(d==b58a)@j2zy;oaIF1)B}8hNBO{=-q=PJVu7Z}2l&XD=I*xG^w30Qd zLFa*zR!b!D2s07|9bPo?NPE)4jIpBKU)_X*`%L8|=r1zVJw*_7^p)(;i^{|{sD*d$7{HDNYR+qQk$wr!ubZQHhO+qP}nw(aRRVkTm4 z#Qku8L+#4iwN~bnJ(CnP264yzeZ2@Y;tp#A^6#hfhTd(iU&o%yJC;)>H)u5{HeMeQ zZyznZAl8rN`52lqN|n4fDDzDI$cAoxKn> zoxf8{EtPR(^1CW&|AMmp&3DnPVPBGjzrpp3<+YKGlirhP>@!!3IkZfM12}G_F-ny2<4{(bL5bdOfXmL8y)r>7Y3kv7~J@fMsG>jnd8T zgmCQgixBV1v~?sRX7R-e?)gI_X~oV=2Eu<|Yg6+b;jSoJ?G1Tr8B1oiX9RBg_}iVT zyjZK{Z9TU~$+faa*h!@M3QCy?;g>+OWdMi}EDSp#2Yt&ZztDw$8Z}E?U?Y-*e~myG zc&D$ZiC2~tz-P0r*CFTP6tIjBLHbb7t4crvx)6kt+?=E-*NO)7tD#`R!WcN9er%*u zP-@YZtHWulB8vu+ePMYV*P{oS=5AvUW8LWR1#E!Qu^gE)R>9~(dD58(%)(+0J^4Pw zx=g>+R1R8{Le}d@hd7e*nZHrF1LmaR_&5Ej?aga8d+$plIMNHv*%gilf7}+LsEp?a~qdfF^l?)%4le9vfH1<+`Vaw!K#Kql~gjOSgn()qw6a7<#qubVE#orR%lrq z+NP=wcW)RB;uKpW%dE!SgpBAsAc`KFx;hDw7~$D;Su%JXBevb7tyxs=D<6FT^1K{> z5wv52+_v%2p84tK=xN0c1!gS>9`7mmQd~51t8=jz3>oiZIGM2q#(#q}vT3H%NIao)gQO?9lY{nFz*IG7< zc6~o3k;_PFaQ!ke2-TzTvQ7)%-{hP$=U+Yg+E<$J_Q5?vZxX?+&6X~;w9~uMN`fO;d`EuuYVv6!Wo&=T$;(H zpv28f7GB%A;tz(g%_3gG@%;c9oa86wH$tI`lSo zl#rl9cOut=GS6j0Agq|0A)xWQ+&<}?47rY{N#ZWUD6+`NtlKEhjkK)Me*~^{Eo5yr zBq1DL{Zf0OFc_CzD~HWW+ZB_u{7YcVBQtQ`7!(Z%_q{f)PUK@j_6M+{%wgfb$>;xs zR%2!P5A|$H*0RZAoiQ{pL37kp`Kl04udCP+xvera+!msb#K@5nY*iS z;CY)LYL{Pl5wxcdK->hAh+(jyWU3-tfCvTD;TXyN)=-WS)}V;k0aEdvJDCuY{t(=d zlF<-3h&-Sed*zQ9cd#b=#K@_f=YN=93pB7L3>_KSowoVKg0NfgTU^6&q!ARMJe(9B&;RBx`hS=r=IwS|QC*21xk3yFb$ zu9CDP7|#J>q_+AC)eTY68YxZ=VMyt(I|P__b1rD0rAS?>t)T%cACXK0bN zm3kfn?p8(1|p(UaGMKSOJh)#_!#%Cmzy2+IUSJP7gae+*x<%j)?=S}j7q*% zEpZ4saZiW}JNd~Q^4K7oKE0}$#7T1_jA8wv9^E%he}xsoU2n^cMsDf7FIvsYu%# zu)ufyANmUgL4{aY{&Ktd3YB_YzXihGES#?nMWdtna7lC?;NyUEAnzAtChvJ#B?ch5 zc-uu_!e(>ob=Qj4hxgLx`!lL>)#akHo$)e!K%Zc@ueH^85-U2{`|wVyLx&4Oa6zrefzRRyB(u4*l!TyV4P^r zWH*S&I7??8me{j~^KPcYMJj#H?JC}{q;E1fyy1aq{A|N{GylORNVm!sO=^?VTTBu0 z2yW2}*B!`-$-b-H4t&!hQ43Ej^^$@=ER z$FeKlLVMPc6=vP#mg9~-JBly7>Ht2ZWX?{5H*r3TOjegJv!CAQkvbe{yr}}>T15Vj zb$1?yZU_IM3CGW|wdiKYyEQi?-|}p4;mkJ+wKMA}$<=K$CCebW^T+DsZZ#9th)}+11e|=DPm9n7+^JX?GG-y zw3P`J`$&}p`?vsIrr-X9-K^n{sc&_b(R}^SO2vc|vkflszM*~8+QscgRs;doh!~zW zP^hS|y9%GS*DTZtU!kX>J$SfeTguv3`|>8@|==FgFidLHxKPdDvrMfBLKa4s5f#)7in&uEDk8+s;ZgT?#uB z4uL$)Sp3~4{(5hT>vsdX4fZuf3dKy;~DUxKd|nGzFu6esl(iQ)Jrpl*LI{Q7%xN1RA_&`ij6nnjEcU zUaACvKU@-wTuc$Xs7w*OiQ#lrhj|DI!5v<@8#N*%Ebxy`#PnRg$%a=P1Hni?iO$^3 zK%!VQ3~v{W<0sJ+RFDOqGl?1C|DEVP$iXk`pGMcP&}%Smz|Aall?Z4>Mp1dyXy-id zwGMPPAi$7&K%`TJ;#*&s7G|EVsyR0^LaKOw5jq^f_cU?h%wX+F8SwWj%VVJ%5O`LaAus(N(@Q?fjAR>^pW|WE_1DoIqQbx^(1Ev zwvRTa)~$$QYEQ#7Gtp(`T>O(`0V3iHth#*7t9B=Eb)H~Nu4?R&mmuV^ej$tG`6AV` z?e_!LAE2&P2B7~Uf~@}&dx?RO@jsaUNvdiN2ct0mR|$9&CK1(o8K81xRnJ)_ZalJ$ zBdfi?s1#wYHmOjUJmT7!gol^wcdTb-WV5Dx4h)lv)eGiZq4V3H+?S(Tg!o;cY^T}P zrQ!7=bD9LNhdWZy!F_2XgnT@9ZvAq@7fiQ&5e3fxi(9_%&g=4{`eobl?u1g#rNLbt zPaKKlLMZ)jL>V*apiE6FqC-{T(YizksHj#s?zL4J#m=0=^BzmQZ`i)^4Ud@g!8eiEKFwdV(U>C+dv>W8>}Okm zIYNj08_-^A8axfNY|{fuen0Fbs^_l!f7-#3S}H@3W;w+f{6|tKc)BNV@ANKl@mMww zfQx)=TU_xRy*QD3N0l1yop46{%p@DCEM0oM11cC?xe`Gbc;Mj{=9O6p?cll)c7y5` z&P~^KV&|`YmBxB=iHmw+EzEux!JZg^>;C2hnsSP4}f~Lr9;L6+jL)VCIPD+fQlrHPpuy`lAFqh zQ}R3U%oQvdUrb&$Nq0haHX(I`%QwWGqc)=H?Ya7tw}opOs?u5`V7f8|eKq>RSV#|9 z&V=oB_-Pr|)7xP|spEs39d6#r2@3{9lqZA0#aSKWa)^oENS@)XbB3XgQP;N1jp{jg zeO1{acjn`>7P?es0b!TXyd3QFv6?GLwi{oTnoUqd5&E!I@Yn^7Zpf zq#p{4)S4G`q$;z+G~{zxL15sYT8v06Pa{x_jdt@I4y%di^U%a%_736=>1J2HYPs&)4{PtNJVSQcuC zp?Z<3diNQ#e!52vP!Xn?DEXUMG`f4(-zYVmoL$Jwo&krDsBZ&VcfCZ;Rog}b+(2hd zv(Xw9q&t;}>p0tC;+$=gIH8NbQ2C)%(TyRj1J@$@lfoGDyO?uL7|RZ1Z#6kmh#s*4u)}1GpZ7{>!MJrwrcbeKn@(r zMt#11P`+8YuW_WfCAON&Mmp)`pHZ?55s&RHqr|_vydkZ9Q-(d%idC}s`UVe1C2r}` z{^HYF1SjEAzp}rCsXc&spt|;nGdA6!Xw(6D4mK1`a17|0eJpwNawQQ!C6_)o3TEM^ zQvf4x|0cv`ta_f9&V#MMARsGuedPkqZeVHbEc_|CR5Usq&ajrTYWL!^aFb@I{og+Zpm zxEF1)C9*INq zg#<8?cM=$=w8J$j@;hS!o4a2xXdnWy$+%l$a(F#kiRFUZQf-~wM@YH_V5y``(Ecs< zJupE$A1kp9rzx(r`zJ=o*cKBjJ*<4JK90_|)%P*-xZ2*nz_t7o&NcihEd^WStbjxP zu&rroijByNQUgWQ7q(TB)cy=KOZyBcsu68*yxBRZx<&i2MNo_mZkw5Qv%%a&cO*dCA4}wDD*HcWE;jf3O0UoV zCO0(b|4J~bWbOJE%q}xs1p`Ioe90-xcJ5!K<18D}qsILaX;rfo?bmc+YAWCqeGs4v z9=xRFYP=pwS9?pxD6fSXa8B1foU+^>Z>YCd;|Iz{PpL-c#ILsFQJ5&E+pmWP^EmH{ zHLk3)+ge#obud8bGzpfL-cLkYaibuP!S+s8Df^uJ*Akh~;%(Q^ zJ`3oYHFgt{`HjBc4{T)2;Pvz*T68LUK6j^>U&G?NyDvP?bUQi%EFfo9&oARPd7|3l z$wyC`u0Z!XGI}Z4vw;#v#Apk7*s|+I_96q8o35k!zjeZ=rR#$d;z?6w^!OAlywEz4 zA|;EYL3hMbWw<*$A6@dSwhR}kDfT7I^sbx1Ycu<2M&3S5O>J3N=91*aC$Db+gdCQ) zvb6Xuk1=B5j~$jFCU~86C-t@m;9%Qoi6B`5-??mqauD(y1=#8VJ1Tm1*`E+<0}DhW z#N?0}Bfx>C7kQ3R@IFxJ*v+z~Zt_wROs_LBdyC!WkZOkqq(Sg(Z+3q?mb1ZA4Ec=? zv-uK~1zAL}#m+sj0U)ku4&3Cop>7gPx*@gCKmYF+UVhH;sX=>){^aPvo;s_6(=_zMkjxm+Ti^NJE)}^ zx@JfNDbq5YyH>D0EdDx0er^ZvRR7yW>wiMIF|g6G{Qn)eMHs*3Nh0?jUyylx@{m!k zhxTt0czpAuK~N=jwB52Nf-$Hnh)Cgy;^oaZD?6RZa1qg2l=^`msaaWRSuLonP(FKo zygmq&uMemfU;9r-`!_aUhI<~?=S2|{Fqbu+SLM2PX+INhN1yvKtUX;nZ&x8lJ+wX` zv$lUARJr`$O_H=yjN2eTSI^8pNo6*Sq(=sEu%OMJ-9yrtaLE?aWOG#cAtpL_jN6)>pH7R8WKtRVFe3W2K_l79E(vK zD~K9f_s8&6(?}8RT+ym_jOQ>xpJOOyIcT47kS zVT2PC4nw3T(r8;rh8<(NEX2rAuVA*1kD63013^4dTMZ2dD_tj=Y|B*}SVAQ!R*S|i zyfQu~$ckOZ$3=PAZ_uQK(4}~|IP#aZ4d2WHaPt=JjXjQ;J6@uHFHQ*&f|U;6de@>2 zknb3DIUMpgze6p7<03*uGM7q{6x@$ksS^dKO$}l;b8>n}(exO}ziUXs5K=(2Cva`B zhub6&Vmoy-?H{9qQ~Uuvvk6r`;0GJ9f8Y=LEwM%@>`O}tWQl_b-1rq%BEA4bf^0St z>vu#$H)zaOw~B(kFX3iQSHIR$Gd;~~QlG*};8H8zeUG}sY2gM$eARFbUQQw8QoM$y zyquN^F%jcmT!Cl%5St|ZFVT1Oiwf9W7eMJGMFmZ^Vb?7=C{)##eT}RRAb|oZn^)hD ziQW4&5Z;gsxN)pnmj`{To9%X)C4pZhr5GUxctI^^0Mk%vsNPI!m{8Qqy_kz;Co8AR zLmgW}T1V-P$gbD$FRX8NxZqjHJ|BDpMx~gDQ5`J@?lpQ-Jy@7{YI;W#{hK0ow9@my z(MQRShOi>2-8N%r0z^P7-9I%H5I)ijvTHxL+u`j7Z0F7vSCK^*T7VxRRW91f7p})x@px-V`w) zh$OmNT+-XSW<~jrybSZ+gjPMec?#VZOy*3X?(uTZCZ7ER(QbG=`n$Q@uiMS-KXoU6 zsc&4l31u|)XTPWgX9$2!!X5-z#s7eH{sY&{bdth_0LLXbF!SgP_HD4L=bj;X2jTQQvzd!5Z8SUQTZ zGp~xHbKp(2Bh{w&s13~?AFI~JC+bT2c4x%S7I+YBTdSy^lqkeBG7(Zse|(-R(b6X? z=teaH@AvYH#i%K1WN)HwMT<^?LLVmrAY4kfRE%NTd-%Iz_$>mWN8nq$;7m~(YC4<_+AvTD z1vBJp%XiXatJ|YKw6Jn$M^7x)NRa+B?Exzs=fq)9#p<6LMFS}^#wfL#kPv~*#V+Ns zrx+SK4|s<}%UH)E<;H;l-9|cH&8V;|dwFR|T33zLI~`w9k%|1Ur#9u(x7r6AKFp4C zH$s0p*ri4H7Rhn97Da3s_Bu3Dje$0a4>;6HsDbr;-$iF?o?Fh~&?f`Zjio-*L=ESI ztPECa;;D(=@fVdZ(WzV*1954DnwW_-<5JbRQ+~BZE(eX;BqA4X_tyos0HTw(#g^sX zn)&5CE)Y^vm!x^QrXo|-TsYuRFhDjG0Rq|ES1HDufq8_&ub1ox%%XKOS;8RBwE`EY z+-G4-oUh1Q=U1?`G{Rt_P-#*iZoxs&mfb;lrxgK&6!p?GwTVP*9Sb>&f}dEMyZ7yi zh#E^7N)>`rv9Yuyi_}eT*6i7$Q->(=tg=krx=~TYiPYOwr`_#O3|Npx3=u{Qf!+L8 zMVYxqyw?_|c`-hG^(<{*&Nw(5J9eH<{CI2v=bIArAtyTbTFFV z#5;C*=i3B`jaT1HGLb}i+U0rG#*zJs-%XT#sHQHOO1aAr-!?usbPicmRTm9bT+J^0 zGuTVWOjm|3G0J1Qx7=-g7`*8yci3p3MIU)tqUC)0I*-11^u!eUJy(7XFj-qN(7H>8k5rReJ-MzpE4p;ICN0eZO#H+jHPcB9yRA9xa^!f} zAumPcrJ=keX?r7AEzgfxKKtg!>*C_V_1e9*#Cw1H{hbY_)VRMp&W;VcbN70DTwFuZ zKk|6tY=Q}_BC>1!!+65gc=H$Qz_*>fD?)2Wz;i&KQvIJd{rspIloY_tswyOWJvida zDR;DoBg?)#)?u&W!=E>LC;B&@#BoOQaQASA??(*Wg$Q{(iiO)F4N0w7e_0_+KG_nHkE;*nKD**Fp50Yl zD)$9A*8I~R!g49s5kIDu;vdI7glZ61_Q9Y&5V_&NNOJ+=R-eGJ>_WQ`MD7pZ99mw9 z!V%(*x5@G}O{$b@e(4nW<}3pDfgF%+nYa+8 zcMn*%dkbNPVQmBgXRRd&z|DF!eonnMP+wTkZ0_4V2|F;zVe^5sUmvzU>NRiv1`HmN z@Pwdxt0bK)5!al*ALwJH=4Ykn7f_jIHSHB6?s3!$qS-vJWbpz1qzx|Lt{0cRbh7=V z@*n*9(J*;0qcaIw&XidATp z6%M!4iLW7s*}EjL+CRp2LJ89J*|bvp>#WG^%@%n8Phl)B8`7QBRA6c@nwNAq z(slu~!Wm^?06p%)?J23=8 z?F0Qmm8&ZA6En)2mMvy z@K`6;@;bf%YEMR_5bx34By0Q>SIjvFp_F_2`)V$%tlW4$+8ARl9*ir!8Vfc zV3RI?bM_&t-ju@O9&fA=rI4YwiMNHWn0S{6Sr2wE~R$B-I~C!woM6n>yKJT^pN8 z^}<+T7V)-dJX#bQQ}*Yo%4Zt}i0lk13;O=D`2kM#&U*iEm*@Y9OYrLy`Hxjim+G4J z!7mf^{|NEkbQM91^KRv2cTy2l*+Qu)#dL;g(MzUKBu4PMJl&hb3-mZ+X}^Yxg!?1z zkVE#B!bs5p7OHm>tZu;{YND|yHN#IHIOq+aJQU2iU~g_CoSD$&pVXWQ{rz=g!Es9l zC*YAveZ-U2zQXKJtS{*hLZi|sKJcir0D$|qThol?1PVt>p6l$vrGJhqtukWm=~6g5 zgABye40B@Sr3d12r>cKT!A&9hn1+8pU21`i%-RvO3j&@j8dR7mKGZ@5G!0}d;7D|7 z@5wgXJSUa3rip6lx9rkKOqOW)gC+8Ej*wUF)oRv*WGAOCP=a`+*lN6_s!Z>Fau`x? zD=0xQll7`4LtU=f20^kpVbY;d$4L5c=fREQ3km>rAkFPw)~CQRR*Xd6Im#1_Qh%qk z`TzLSK%XMckvsrGpE`r`Q3#;I(IQeBl;@K%JI=URzUUnfIB(`r1m1J)3v3dP#sCz$TpU|i&9U&yY) z6wEV#JZ31Q!De#PHrB&=y1$m1p&@mtGq4P2DafIJo@yLmXloB)j@328(`kqLdOK`3 zU&j`)ELP1SM8(dc)aiy-)wzX3!Sd$w^{1{*5q7MAHFgEXA$#2yDu=U=s4WRN!2s+^ zkKpzNqu=y%l=6*+bdjKB8d-PTzPlZmJtpgJ?!M)iq;1pIunh&VU;7km+-R!Lw$3C{ z#=H^TxOG0Zx<268h2^Fq`-FP?I}(6wmU?digkF|LTI7_ZA$HL@TT~|xHyq;g$yE~Z z;WPCFWkrar#4^t14pSu%`LHBro#VF%vbT{TuV@B$4X;|gd63@hs9dx=KF979roF$Z z5VNh>1E`zo)o9*`XAJQ=E*auWsV98Hxs`EKff+}m?4lvFP1Kwb`vb# B~?<-=F) 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

q!lCQ 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

i$?Ap<~7NC6>8?))T1I=qs7FMz-Px(aL{gtF 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 z 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

obs2IorRTC$^?VoGMS>e`O(%ag8c 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 z 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 kf4da6fAySW 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

lSW;b5C8XlQLlxRqjL;X5cxfpQT` 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 Ag#Z8m diff --git a/code/3rd_glpk/draft/bfd.c b/code/3rd_glpk/draft/bfd.c deleted file mode 100644 index dece376c..00000000 --- a/code/3rd_glpk/draft/bfd.c +++ /dev/null @@ -1,544 +0,0 @@ -/* bfd.c (LP basis factorization driver) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2007, 2014 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "glpk.h" -#include "env.h" -#include "bfd.h" -#include "fhvint.h" -#include "scfint.h" -#ifdef GLP_DEBUG -#include "glpspm.h" -#endif - -struct BFD -{ /* LP basis factorization driver */ - int valid; - /* factorization is valid only if this flag is set */ - int type; - /* type of factorization used: - 0 - interface not established yet - 1 - FHV-factorization - 2 - Schur-complement-based factorization */ - union - { void *none; /* type = 0 */ - FHVINT *fhvi; /* type = 1 */ - SCFINT *scfi; /* type = 2 */ - } u; - /* interface to factorization of LP basis */ - glp_bfcp parm; - /* factorization control parameters */ -#ifdef GLP_DEBUG - SPM *B; - /* current basis (for testing/debugging only) */ -#endif - int upd_cnt; - /* factorization update count */ -#if 1 /* 21/IV-2014 */ - double b_norm; - /* 1-norm of matrix B */ - double i_norm; - /* estimated 1-norm of matrix inv(B) */ -#endif -}; - -BFD *bfd_create_it(void) -{ /* create LP basis factorization */ - BFD *bfd; -#ifdef GLP_DEBUG - xprintf("bfd_create_it: warning: debugging version used\n"); -#endif - bfd = talloc(1, BFD); - bfd->valid = 0; - bfd->type = 0; - bfd->u.none = NULL; - bfd_set_bfcp(bfd, NULL); -#ifdef GLP_DEBUG - bfd->B = NULL; -#endif - bfd->upd_cnt = 0; - return bfd; -} - -#if 0 /* 08/III-2014 */ -void bfd_set_parm(BFD *bfd, const void *parm) -{ /* change LP basis factorization control parameters */ - memcpy(&bfd->parm, parm, sizeof(glp_bfcp)); - return; -} -#endif - -void bfd_get_bfcp(BFD *bfd, void /* glp_bfcp */ *parm) -{ /* retrieve LP basis factorization control parameters */ - memcpy(parm, &bfd->parm, sizeof(glp_bfcp)); - return; -} - -void bfd_set_bfcp(BFD *bfd, const void /* glp_bfcp */ *parm) -{ /* change LP basis factorization control parameters */ - if (parm == NULL) - { /* reset to default */ - memset(&bfd->parm, 0, sizeof(glp_bfcp)); - bfd->parm.type = GLP_BF_LUF + GLP_BF_FT; - bfd->parm.piv_tol = 0.10; - bfd->parm.piv_lim = 4; - bfd->parm.suhl = 1; - bfd->parm.eps_tol = DBL_EPSILON; - bfd->parm.nfs_max = 100; - bfd->parm.nrs_max = 70; - } - else - memcpy(&bfd->parm, parm, sizeof(glp_bfcp)); - return; -} - -#if 1 /* 21/IV-2014 */ -struct bfd_info -{ BFD *bfd; - int (*col)(void *info, int j, int ind[], double val[]); - void *info; -}; - -static int bfd_col(void *info_, int j, int ind[], double val[]) -{ struct bfd_info *info = info_; - int t, len; - double sum; - len = info->col(info->info, j, ind, val); - sum = 0.0; - for (t = 1; t <= len; t++) - { if (val[t] >= 0.0) - sum += val[t]; - else - sum -= val[t]; - } - if (info->bfd->b_norm < sum) - info->bfd->b_norm = sum; - return len; -} -#endif - -int bfd_factorize(BFD *bfd, int m, /*const int bh[],*/ int (*col1) - (void *info, int j, int ind[], double val[]), void *info1) -{ /* compute LP basis factorization */ -#if 1 /* 21/IV-2014 */ - struct bfd_info info; -#endif - int type, ret; - /*xassert(bh == bh);*/ - /* invalidate current factorization */ - bfd->valid = 0; - /* determine required factorization type */ - switch (bfd->parm.type) - { case GLP_BF_LUF + GLP_BF_FT: - type = 1; - break; - case GLP_BF_LUF + GLP_BF_BG: - case GLP_BF_LUF + GLP_BF_GR: - case GLP_BF_BTF + GLP_BF_BG: - case GLP_BF_BTF + GLP_BF_GR: - type = 2; - break; - default: - xassert(bfd != bfd); - } - /* delete factorization interface, if necessary */ - switch (bfd->type) - { case 0: - break; - case 1: - if (type != 1) - { bfd->type = 0; - fhvint_delete(bfd->u.fhvi); - bfd->u.fhvi = NULL; - } - break; - case 2: - if (type != 2) - { bfd->type = 0; - scfint_delete(bfd->u.scfi); - bfd->u.scfi = NULL; - } - break; - default: - xassert(bfd != bfd); - } - /* establish factorization interface, if necessary */ - if (bfd->type == 0) - { switch (type) - { case 1: - bfd->type = 1; - xassert(bfd->u.fhvi == NULL); - bfd->u.fhvi = fhvint_create(); - break; - case 2: - bfd->type = 2; - xassert(bfd->u.scfi == NULL); - if (!(bfd->parm.type & GLP_BF_BTF)) - bfd->u.scfi = scfint_create(1); - else - bfd->u.scfi = scfint_create(2); - break; - default: - xassert(type != type); - } - } - /* try to compute factorization */ -#if 1 /* 21/IV-2014 */ - bfd->b_norm = bfd->i_norm = 0.0; - info.bfd = bfd; - info.col = col1; - info.info = info1; -#endif - switch (bfd->type) - { case 1: - bfd->u.fhvi->lufi->sgf_piv_tol = bfd->parm.piv_tol; - bfd->u.fhvi->lufi->sgf_piv_lim = bfd->parm.piv_lim; - bfd->u.fhvi->lufi->sgf_suhl = bfd->parm.suhl; - bfd->u.fhvi->lufi->sgf_eps_tol = bfd->parm.eps_tol; - bfd->u.fhvi->nfs_max = bfd->parm.nfs_max; - ret = fhvint_factorize(bfd->u.fhvi, m, bfd_col, &info); -#if 1 /* FIXME */ - if (ret == 0) - bfd->i_norm = fhvint_estimate(bfd->u.fhvi); - else - ret = BFD_ESING; -#endif - break; - case 2: - if (bfd->u.scfi->scf.type == 1) - { bfd->u.scfi->u.lufi->sgf_piv_tol = bfd->parm.piv_tol; - bfd->u.scfi->u.lufi->sgf_piv_lim = bfd->parm.piv_lim; - bfd->u.scfi->u.lufi->sgf_suhl = bfd->parm.suhl; - bfd->u.scfi->u.lufi->sgf_eps_tol = bfd->parm.eps_tol; - } - else if (bfd->u.scfi->scf.type == 2) - { bfd->u.scfi->u.btfi->sgf_piv_tol = bfd->parm.piv_tol; - bfd->u.scfi->u.btfi->sgf_piv_lim = bfd->parm.piv_lim; - bfd->u.scfi->u.btfi->sgf_suhl = bfd->parm.suhl; - bfd->u.scfi->u.btfi->sgf_eps_tol = bfd->parm.eps_tol; - } - else - xassert(bfd != bfd); - bfd->u.scfi->nn_max = bfd->parm.nrs_max; - ret = scfint_factorize(bfd->u.scfi, m, bfd_col, &info); -#if 1 /* FIXME */ - if (ret == 0) - bfd->i_norm = scfint_estimate(bfd->u.scfi); - else - ret = BFD_ESING; -#endif - break; - default: - xassert(bfd != bfd); - } -#ifdef GLP_DEBUG - /* save specified LP basis */ - if (bfd->B != NULL) - spm_delete_mat(bfd->B); - bfd->B = spm_create_mat(m, m); - { int *ind = talloc(1+m, int); - double *val = talloc(1+m, double); - int j, k, len; - for (j = 1; j <= m; j++) - { len = col(info, j, ind, val); - for (k = 1; k <= len; k++) - spm_new_elem(bfd->B, ind[k], j, val[k]); - } - tfree(ind); - tfree(val); - } -#endif - if (ret == 0) - { /* factorization has been successfully computed */ - double cond; - bfd->valid = 1; -#ifdef GLP_DEBUG - cond = bfd_condest(bfd); - if (cond > 1e9) - xprintf("bfd_factorize: warning: cond(B) = %g\n", cond); -#endif - } -#ifdef GLP_DEBUG - xprintf("bfd_factorize: m = %d; ret = %d\n", m, ret); -#endif - bfd->upd_cnt = 0; - return ret; -} - -#if 0 /* 21/IV-2014 */ -double bfd_estimate(BFD *bfd) -{ /* estimate 1-norm of inv(B) */ - double norm; - xassert(bfd->valid); - xassert(bfd->upd_cnt == 0); - switch (bfd->type) - { case 1: - norm = fhvint_estimate(bfd->u.fhvi); - break; - case 2: - norm = scfint_estimate(bfd->u.scfi); - break; - default: - xassert(bfd != bfd); - } - return norm; -} -#endif - -#if 1 /* 21/IV-2014 */ -double bfd_condest(BFD *bfd) -{ /* estimate condition of B */ - double cond; - xassert(bfd->valid); - /*xassert(bfd->upd_cnt == 0);*/ - cond = bfd->b_norm * bfd->i_norm; - if (cond < 1.0) - cond = 1.0; - return cond; -} -#endif - -void bfd_ftran(BFD *bfd, double x[]) -{ /* perform forward transformation (solve system B * x = b) */ -#ifdef GLP_DEBUG - SPM *B = bfd->B; - int m = B->m; - double *b = talloc(1+m, double); - SPME *e; - int k; - double s, relerr, maxerr; - for (k = 1; k <= m; k++) - b[k] = x[k]; -#endif - xassert(bfd->valid); - switch (bfd->type) - { case 1: - fhvint_ftran(bfd->u.fhvi, x); - break; - case 2: - scfint_ftran(bfd->u.scfi, x); - break; - default: - xassert(bfd != bfd); - } -#ifdef GLP_DEBUG - maxerr = 0.0; - for (k = 1; k <= m; k++) - { s = 0.0; - for (e = B->row[k]; e != NULL; e = e->r_next) - s += e->val * x[e->j]; - relerr = (b[k] - s) / (1.0 + fabs(b[k])); - if (maxerr < relerr) - maxerr = relerr; - } - if (maxerr > 1e-8) - xprintf("bfd_ftran: maxerr = %g; relative error too large\n", - maxerr); - tfree(b); -#endif - return; -} - -#if 1 /* 30/III-2016 */ -void bfd_ftran_s(BFD *bfd, FVS *x) -{ /* sparse version of bfd_ftran */ - /* (sparse mode is not implemented yet) */ - int n = x->n; - int *ind = x->ind; - double *vec = x->vec; - int j, nnz = 0; - bfd_ftran(bfd, vec); - for (j = n; j >= 1; j--) - { if (vec[j] != 0.0) - ind[++nnz] = j; - } - x->nnz = nnz; - return; -} -#endif - -void bfd_btran(BFD *bfd, double x[]) -{ /* perform backward transformation (solve system B'* x = b) */ -#ifdef GLP_DEBUG - SPM *B = bfd->B; - int m = B->m; - double *b = talloc(1+m, double); - SPME *e; - int k; - double s, relerr, maxerr; - for (k = 1; k <= m; k++) - b[k] = x[k]; -#endif - xassert(bfd->valid); - switch (bfd->type) - { case 1: - fhvint_btran(bfd->u.fhvi, x); - break; - case 2: - scfint_btran(bfd->u.scfi, x); - break; - default: - xassert(bfd != bfd); - } -#ifdef GLP_DEBUG - maxerr = 0.0; - for (k = 1; k <= m; k++) - { s = 0.0; - for (e = B->col[k]; e != NULL; e = e->c_next) - s += e->val * x[e->i]; - relerr = (b[k] - s) / (1.0 + fabs(b[k])); - if (maxerr < relerr) - maxerr = relerr; - } - if (maxerr > 1e-8) - xprintf("bfd_btran: maxerr = %g; relative error too large\n", - maxerr); - tfree(b); -#endif - return; -} - -#if 1 /* 30/III-2016 */ -void bfd_btran_s(BFD *bfd, FVS *x) -{ /* sparse version of bfd_btran */ - /* (sparse mode is not implemented yet) */ - int n = x->n; - int *ind = x->ind; - double *vec = x->vec; - int j, nnz = 0; - bfd_btran(bfd, vec); - for (j = n; j >= 1; j--) - { if (vec[j] != 0.0) - ind[++nnz] = j; - } - x->nnz = nnz; - return; -} -#endif - -int bfd_update(BFD *bfd, int j, int len, const int ind[], const double - val[]) -{ /* update LP basis factorization */ - int ret; - xassert(bfd->valid); - switch (bfd->type) - { case 1: - ret = fhvint_update(bfd->u.fhvi, j, len, ind, val); -#if 1 /* FIXME */ - switch (ret) - { case 0: - break; - case 1: - ret = BFD_ESING; - break; - case 2: - case 3: - ret = BFD_ECOND; - break; - case 4: - ret = BFD_ELIMIT; - break; - case 5: - ret = BFD_ECHECK; - break; - default: - xassert(ret != ret); - } -#endif - break; - case 2: - switch (bfd->parm.type & 0x0F) - { case GLP_BF_BG: - ret = scfint_update(bfd->u.scfi, 1, j, len, ind, val); - break; - case GLP_BF_GR: - ret = scfint_update(bfd->u.scfi, 2, j, len, ind, val); - break; - default: - xassert(bfd != bfd); - } -#if 1 /* FIXME */ - switch (ret) - { case 0: - break; - case 1: - ret = BFD_ELIMIT; - break; - case 2: - ret = BFD_ECOND; - break; - default: - xassert(ret != ret); - } -#endif - break; - default: - xassert(bfd != bfd); - } - if (ret != 0) - { /* updating factorization failed */ - bfd->valid = 0; - } -#ifdef GLP_DEBUG - /* save updated LP basis */ - { SPME *e; - int k; - for (e = bfd->B->col[j]; e != NULL; e = e->c_next) - e->val = 0.0; - spm_drop_zeros(bfd->B, 0.0); - for (k = 1; k <= len; k++) - spm_new_elem(bfd->B, ind[k], j, val[k]); - } -#endif - if (ret == 0) - bfd->upd_cnt++; - return ret; -} - -int bfd_get_count(BFD *bfd) -{ /* determine factorization update count */ - return bfd->upd_cnt; -} - -void bfd_delete_it(BFD *bfd) -{ /* delete LP basis factorization */ - switch (bfd->type) - { case 0: - break; - case 1: - fhvint_delete(bfd->u.fhvi); - break; - case 2: - scfint_delete(bfd->u.scfi); - break; - default: - xassert(bfd != bfd); - } -#ifdef GLP_DEBUG - if (bfd->B != NULL) - spm_delete_mat(bfd->B); -#endif - tfree(bfd); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/bfd.h b/code/3rd_glpk/draft/bfd.h deleted file mode 100644 index 0ef4c023..00000000 --- a/code/3rd_glpk/draft/bfd.h +++ /dev/null @@ -1,107 +0,0 @@ -/* bfd.h (LP basis factorization driver) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef BFD_H -#define BFD_H - -#if 1 /* 30/III-2016 */ -#include "fvs.h" -#endif - -typedef struct BFD BFD; - -/* return codes: */ -#define BFD_ESING 1 /* singular matrix */ -#define BFD_ECOND 2 /* ill-conditioned matrix */ -#define BFD_ECHECK 3 /* insufficient accuracy */ -#define BFD_ELIMIT 4 /* update limit reached */ -#if 0 /* 05/III-2014 */ -#define BFD_EROOM 5 /* SVA overflow */ -#endif - -#define bfd_create_it _glp_bfd_create_it -BFD *bfd_create_it(void); -/* create LP basis factorization */ - -#if 0 /* 08/III-2014 */ -#define bfd_set_parm _glp_bfd_set_parm -void bfd_set_parm(BFD *bfd, const void *parm); -/* change LP basis factorization control parameters */ -#endif - -#define bfd_get_bfcp _glp_bfd_get_bfcp -void bfd_get_bfcp(BFD *bfd, void /* glp_bfcp */ *parm); -/* retrieve LP basis factorization control parameters */ - -#define bfd_set_bfcp _glp_bfd_set_bfcp -void bfd_set_bfcp(BFD *bfd, const void /* glp_bfcp */ *parm); -/* change LP basis factorization control parameters */ - -#define bfd_factorize _glp_bfd_factorize -int bfd_factorize(BFD *bfd, int m, /*const int bh[],*/ int (*col) - (void *info, int j, int ind[], double val[]), void *info); -/* compute LP basis factorization */ - -#if 1 /* 21/IV-2014 */ -#define bfd_condest _glp_bfd_condest -double bfd_condest(BFD *bfd); -/* estimate condition of B */ -#endif - -#define bfd_ftran _glp_bfd_ftran -void bfd_ftran(BFD *bfd, double x[]); -/* perform forward transformation (solve system B*x = b) */ - -#if 1 /* 30/III-2016 */ -#define bfd_ftran_s _glp_bfd_ftran_s -void bfd_ftran_s(BFD *bfd, FVS *x); -/* sparse version of bfd_ftran */ -#endif - -#define bfd_btran _glp_bfd_btran -void bfd_btran(BFD *bfd, double x[]); -/* perform backward transformation (solve system B'*x = b) */ - -#if 1 /* 30/III-2016 */ -#define bfd_btran_s _glp_bfd_btran_s -void bfd_btran_s(BFD *bfd, FVS *x); -/* sparse version of bfd_btran */ -#endif - -#define bfd_update _glp_bfd_update -int bfd_update(BFD *bfd, int j, int len, const int ind[], const double - val[]); -/* update LP basis factorization */ - -#define bfd_get_count _glp_bfd_get_count -int bfd_get_count(BFD *bfd); -/* determine factorization update count */ - -#define bfd_delete_it _glp_bfd_delete_it -void bfd_delete_it(BFD *bfd); -/* delete LP basis factorization */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/bfx.c b/code/3rd_glpk/draft/bfx.c deleted file mode 100644 index 565480b6..00000000 --- a/code/3rd_glpk/draft/bfx.c +++ /dev/null @@ -1,89 +0,0 @@ -/* bfx.c (LP basis factorization driver, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "bfx.h" -#include "env.h" -#include "lux.h" - -struct BFX -{ int valid; - LUX *lux; -}; - -BFX *bfx_create_binv(void) -{ /* create factorization of the basis matrix */ - BFX *bfx; - bfx = xmalloc(sizeof(BFX)); - bfx->valid = 0; - bfx->lux = NULL; - return bfx; -} - -int bfx_factorize(BFX *binv, int m, int (*col)(void *info, int j, - int ind[], mpq_t val[]), void *info) -{ /* compute factorization of the basis matrix */ - int ret; - xassert(m > 0); - if (binv->lux != NULL && binv->lux->n != m) - { lux_delete(binv->lux); - binv->lux = NULL; - } - if (binv->lux == NULL) - binv->lux = lux_create(m); - ret = lux_decomp(binv->lux, col, info); - binv->valid = (ret == 0); - return ret; -} - -void bfx_ftran(BFX *binv, mpq_t x[], int save) -{ /* perform forward transformation (FTRAN) */ - xassert(binv->valid); - lux_solve(binv->lux, 0, x); - xassert(save == save); - return; -} - -void bfx_btran(BFX *binv, mpq_t x[]) -{ /* perform backward transformation (BTRAN) */ - xassert(binv->valid); - lux_solve(binv->lux, 1, x); - return; -} - -int bfx_update(BFX *binv, int j) -{ /* update factorization of the basis matrix */ - xassert(binv->valid); - xassert(1 <= j && j <= binv->lux->n); - return 1; -} - -void bfx_delete_binv(BFX *binv) -{ /* delete factorization of the basis matrix */ - if (binv->lux != NULL) - lux_delete(binv->lux); - xfree(binv); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/bfx.h b/code/3rd_glpk/draft/bfx.h deleted file mode 100644 index c67d5ea4..00000000 --- a/code/3rd_glpk/draft/bfx.h +++ /dev/null @@ -1,67 +0,0 @@ -/* bfx.h (LP basis factorization driver, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef BFX_H -#define BFX_H - -#include "mygmp.h" - -typedef struct BFX BFX; - -#define bfx_create_binv _glp_bfx_create_binv -BFX *bfx_create_binv(void); -/* create factorization of the basis matrix */ - -#define bfx_is_valid _glp_bfx_is_valid -int bfx_is_valid(BFX *binv); -/* check if factorization is valid */ - -#define bfx_invalidate _glp_bfx_invalidate -void bfx_invalidate(BFX *binv); -/* invalidate factorization of the basis matrix */ - -#define bfx_factorize _glp_bfx_factorize -int bfx_factorize(BFX *binv, int m, int (*col)(void *info, int j, - int ind[], mpq_t val[]), void *info); -/* compute factorization of the basis matrix */ - -#define bfx_ftran _glp_bfx_ftran -void bfx_ftran(BFX *binv, mpq_t x[], int save); -/* perform forward transformation (FTRAN) */ - -#define bfx_btran _glp_bfx_btran -void bfx_btran(BFX *binv, mpq_t x[]); -/* perform backward transformation (BTRAN) */ - -#define bfx_update _glp_bfx_update -int bfx_update(BFX *binv, int j); -/* update factorization of the basis matrix */ - -#define bfx_delete_binv _glp_bfx_delete_binv -void bfx_delete_binv(BFX *binv); -/* delete factorization of the basis matrix */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/draft.h b/code/3rd_glpk/draft/draft.h deleted file mode 100644 index cefd2124..00000000 --- a/code/3rd_glpk/draft/draft.h +++ /dev/null @@ -1,22 +0,0 @@ -/* draft.h */ - -/* (reserved for copyright notice) */ - -#ifndef DRAFT_H -#define DRAFT_H - -#if 1 /* 28/III-2016 */ -#define GLP_UNDOC 1 -#endif -#include "glpk.h" - -#if 1 /* 28/XI-2009 */ -int _glp_analyze_row(glp_prob *P, int len, const int ind[], - const double val[], int type, double rhs, double eps, int *_piv, - double *_x, double *_dx, double *_y, double *_dy, double *_dz); -/* simulate one iteration of dual simplex method */ -#endif - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi06.c b/code/3rd_glpk/draft/glpapi06.c deleted file mode 100644 index a31e3968..00000000 --- a/code/3rd_glpk/draft/glpapi06.c +++ /dev/null @@ -1,860 +0,0 @@ -/* glpapi06.c (simplex method routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" -#include "npp.h" -#if 0 /* 07/XI-2015 */ -#include "glpspx.h" -#else -#include "simplex.h" -#define spx_dual spy_dual -#endif - -/*********************************************************************** -* NAME -* -* glp_simplex - solve LP problem with the simplex method -* -* SYNOPSIS -* -* int glp_simplex(glp_prob *P, const glp_smcp *parm); -* -* DESCRIPTION -* -* The routine glp_simplex is a driver to the LP solver based on the -* simplex method. This routine retrieves problem data from the -* specified problem object, calls the solver to solve the problem -* instance, and stores results of computations back into the problem -* object. -* -* The simplex solver has a set of control parameters. Values of the -* control parameters can be passed in a structure glp_smcp, which the -* parameter parm points to. -* -* The parameter parm can be specified as NULL, in which case the LP -* solver uses default settings. -* -* RETURNS -* -* 0 The LP problem instance has been successfully solved. This code -* does not necessarily mean that the solver has found optimal -* solution. It only means that the solution process was successful. -* -* GLP_EBADB -* Unable to start the search, because the initial basis specified -* in the problem object is invalid--the number of basic (auxiliary -* and structural) variables is not the same as the number of rows in -* the problem object. -* -* GLP_ESING -* Unable to start the search, because the basis matrix correspodning -* to the initial basis is singular within the working precision. -* -* GLP_ECOND -* Unable to start the search, because the basis matrix correspodning -* to the initial basis is ill-conditioned, i.e. its condition number -* is too large. -* -* GLP_EBOUND -* Unable to start the search, because some double-bounded variables -* have incorrect bounds. -* -* GLP_EFAIL -* The search was prematurely terminated due to the solver failure. -* -* GLP_EOBJLL -* The search was prematurely terminated, because the objective -* function being maximized has reached its lower limit and continues -* decreasing (dual simplex only). -* -* GLP_EOBJUL -* The search was prematurely terminated, because the objective -* function being minimized has reached its upper limit and continues -* increasing (dual simplex only). -* -* GLP_EITLIM -* The search was prematurely terminated, because the simplex -* iteration limit has been exceeded. -* -* GLP_ETMLIM -* The search was prematurely terminated, because the time limit has -* been exceeded. -* -* GLP_ENOPFS -* The LP problem instance has no primal feasible solution (only if -* the LP presolver is used). -* -* GLP_ENODFS -* The LP problem instance has no dual feasible solution (only if the -* LP presolver is used). */ - -static void trivial_lp(glp_prob *P, const glp_smcp *parm) -{ /* solve trivial LP which has empty constraint matrix */ - GLPROW *row; - GLPCOL *col; - int i, j; - double p_infeas, d_infeas, zeta; - P->valid = 0; - P->pbs_stat = P->dbs_stat = GLP_FEAS; - P->obj_val = P->c0; - P->some = 0; - p_infeas = d_infeas = 0.0; - /* make all auxiliary variables basic */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - row->stat = GLP_BS; - row->prim = row->dual = 0.0; - /* check primal feasibility */ - if (row->type == GLP_LO || row->type == GLP_DB || - row->type == GLP_FX) - { /* row has lower bound */ - if (row->lb > + parm->tol_bnd) - { P->pbs_stat = GLP_NOFEAS; - if (P->some == 0 && parm->meth != GLP_PRIMAL) - P->some = i; - } - if (p_infeas < + row->lb) - p_infeas = + row->lb; - } - if (row->type == GLP_UP || row->type == GLP_DB || - row->type == GLP_FX) - { /* row has upper bound */ - if (row->ub < - parm->tol_bnd) - { P->pbs_stat = GLP_NOFEAS; - if (P->some == 0 && parm->meth != GLP_PRIMAL) - P->some = i; - } - if (p_infeas < - row->ub) - p_infeas = - row->ub; - } - } - /* determine scale factor for the objective row */ - zeta = 1.0; - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (zeta < fabs(col->coef)) zeta = fabs(col->coef); - } - zeta = (P->dir == GLP_MIN ? +1.0 : -1.0) / zeta; - /* make all structural variables non-basic */ - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->type == GLP_FR) - col->stat = GLP_NF, col->prim = 0.0; - else if (col->type == GLP_LO) -lo: col->stat = GLP_NL, col->prim = col->lb; - else if (col->type == GLP_UP) -up: col->stat = GLP_NU, col->prim = col->ub; - else if (col->type == GLP_DB) - { if (zeta * col->coef > 0.0) - goto lo; - else if (zeta * col->coef < 0.0) - goto up; - else if (fabs(col->lb) <= fabs(col->ub)) - goto lo; - else - goto up; - } - else if (col->type == GLP_FX) - col->stat = GLP_NS, col->prim = col->lb; - col->dual = col->coef; - P->obj_val += col->coef * col->prim; - /* check dual feasibility */ - if (col->type == GLP_FR || col->type == GLP_LO) - { /* column has no upper bound */ - if (zeta * col->dual < - parm->tol_dj) - { P->dbs_stat = GLP_NOFEAS; - if (P->some == 0 && parm->meth == GLP_PRIMAL) - P->some = P->m + j; - } - if (d_infeas < - zeta * col->dual) - d_infeas = - zeta * col->dual; - } - if (col->type == GLP_FR || col->type == GLP_UP) - { /* column has no lower bound */ - if (zeta * col->dual > + parm->tol_dj) - { P->dbs_stat = GLP_NOFEAS; - if (P->some == 0 && parm->meth == GLP_PRIMAL) - P->some = P->m + j; - } - if (d_infeas < + zeta * col->dual) - d_infeas = + zeta * col->dual; - } - } - /* simulate the simplex solver output */ - if (parm->msg_lev >= GLP_MSG_ON && parm->out_dly == 0) - { xprintf("~%6d: obj = %17.9e infeas = %10.3e\n", P->it_cnt, - P->obj_val, parm->meth == GLP_PRIMAL ? p_infeas : d_infeas); - } - if (parm->msg_lev >= GLP_MSG_ALL && parm->out_dly == 0) - { if (P->pbs_stat == GLP_FEAS && P->dbs_stat == GLP_FEAS) - xprintf("OPTIMAL SOLUTION FOUND\n"); - else if (P->pbs_stat == GLP_NOFEAS) - xprintf("PROBLEM HAS NO FEASIBLE SOLUTION\n"); - else if (parm->meth == GLP_PRIMAL) - xprintf("PROBLEM HAS UNBOUNDED SOLUTION\n"); - else - xprintf("PROBLEM HAS NO DUAL FEASIBLE SOLUTION\n"); - } - return; -} - -static int solve_lp(glp_prob *P, const glp_smcp *parm) -{ /* solve LP directly without using the preprocessor */ - int ret; - if (!glp_bf_exists(P)) - { ret = glp_factorize(P); - if (ret == 0) - ; - else if (ret == GLP_EBADB) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_simplex: initial basis is invalid\n"); - } - else if (ret == GLP_ESING) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_simplex: initial basis is singular\n"); - } - else if (ret == GLP_ECOND) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf( - "glp_simplex: initial basis is ill-conditioned\n"); - } - else - xassert(ret != ret); - if (ret != 0) goto done; - } - if (parm->meth == GLP_PRIMAL) - ret = spx_primal(P, parm); - else if (parm->meth == GLP_DUALP) - { ret = spx_dual(P, parm); - if (ret == GLP_EFAIL && P->valid) - ret = spx_primal(P, parm); - } - else if (parm->meth == GLP_DUAL) - ret = spx_dual(P, parm); - else - xassert(parm != parm); -done: return ret; -} - -static int preprocess_and_solve_lp(glp_prob *P, const glp_smcp *parm) -{ /* solve LP using the preprocessor */ - NPP *npp; - glp_prob *lp = NULL; - glp_bfcp bfcp; - int ret; - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("Preprocessing...\n"); - /* create preprocessor workspace */ - npp = npp_create_wksp(); - /* load original problem into the preprocessor workspace */ - npp_load_prob(npp, P, GLP_OFF, GLP_SOL, GLP_OFF); - /* process LP prior to applying primal/dual simplex method */ - ret = npp_simplex(npp, parm); - if (ret == 0) - ; - else if (ret == GLP_ENOPFS) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("PROBLEM HAS NO PRIMAL FEASIBLE SOLUTION\n"); - } - else if (ret == GLP_ENODFS) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("PROBLEM HAS NO DUAL FEASIBLE SOLUTION\n"); - } - else - xassert(ret != ret); - if (ret != 0) goto done; - /* build transformed LP */ - lp = glp_create_prob(); - npp_build_prob(npp, lp); - /* if the transformed LP is empty, it has empty solution, which - is optimal */ - if (lp->m == 0 && lp->n == 0) - { lp->pbs_stat = lp->dbs_stat = GLP_FEAS; - lp->obj_val = lp->c0; - if (parm->msg_lev >= GLP_MSG_ON && parm->out_dly == 0) - { xprintf("~%6d: obj = %17.9e infeas = %10.3e\n", P->it_cnt, - lp->obj_val, 0.0); - } - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("OPTIMAL SOLUTION FOUND BY LP PREPROCESSOR\n"); - goto post; - } - if (parm->msg_lev >= GLP_MSG_ALL) - { xprintf("%d row%s, %d column%s, %d non-zero%s\n", - lp->m, lp->m == 1 ? "" : "s", lp->n, lp->n == 1 ? "" : "s", - lp->nnz, lp->nnz == 1 ? "" : "s"); - } - /* inherit basis factorization control parameters */ - glp_get_bfcp(P, &bfcp); - glp_set_bfcp(lp, &bfcp); - /* scale the transformed problem */ - { ENV *env = get_env_ptr(); - int term_out = env->term_out; - if (!term_out || parm->msg_lev < GLP_MSG_ALL) - env->term_out = GLP_OFF; - else - env->term_out = GLP_ON; - glp_scale_prob(lp, GLP_SF_AUTO); - env->term_out = term_out; - } - /* build advanced initial basis */ - { ENV *env = get_env_ptr(); - int term_out = env->term_out; - if (!term_out || parm->msg_lev < GLP_MSG_ALL) - env->term_out = GLP_OFF; - else - env->term_out = GLP_ON; - glp_adv_basis(lp, 0); - env->term_out = term_out; - } - /* solve the transformed LP */ - lp->it_cnt = P->it_cnt; - ret = solve_lp(lp, parm); - P->it_cnt = lp->it_cnt; - /* only optimal solution can be postprocessed */ - if (!(ret == 0 && lp->pbs_stat == GLP_FEAS && lp->dbs_stat == - GLP_FEAS)) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_simplex: unable to recover undefined or non-op" - "timal solution\n"); - if (ret == 0) - { if (lp->pbs_stat == GLP_NOFEAS) - ret = GLP_ENOPFS; - else if (lp->dbs_stat == GLP_NOFEAS) - ret = GLP_ENODFS; - else - xassert(lp != lp); - } - goto done; - } -post: /* postprocess solution from the transformed LP */ - npp_postprocess(npp, lp); - /* the transformed LP is no longer needed */ - glp_delete_prob(lp), lp = NULL; - /* store solution to the original problem */ - npp_unload_sol(npp, P); - /* the original LP has been successfully solved */ - ret = 0; -done: /* delete the transformed LP, if it exists */ - if (lp != NULL) glp_delete_prob(lp); - /* delete preprocessor workspace */ - npp_delete_wksp(npp); - return ret; -} - -int glp_simplex(glp_prob *P, const glp_smcp *parm) -{ /* solve LP problem with the simplex method */ - glp_smcp _parm; - int i, j, ret; - /* check problem object */ -#if 0 /* 04/IV-2016 */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_simplex: P = %p; invalid problem object\n", P); -#endif - if (P->tree != NULL && P->tree->reason != 0) - xerror("glp_simplex: operation not allowed\n"); - /* check control parameters */ - if (parm == NULL) - parm = &_parm, glp_init_smcp((glp_smcp *)parm); - if (!(parm->msg_lev == GLP_MSG_OFF || - parm->msg_lev == GLP_MSG_ERR || - parm->msg_lev == GLP_MSG_ON || - parm->msg_lev == GLP_MSG_ALL || - parm->msg_lev == GLP_MSG_DBG)) - xerror("glp_simplex: msg_lev = %d; invalid parameter\n", - parm->msg_lev); - if (!(parm->meth == GLP_PRIMAL || - parm->meth == GLP_DUALP || - parm->meth == GLP_DUAL)) - xerror("glp_simplex: meth = %d; invalid parameter\n", - parm->meth); - if (!(parm->pricing == GLP_PT_STD || - parm->pricing == GLP_PT_PSE)) - xerror("glp_simplex: pricing = %d; invalid parameter\n", - parm->pricing); - if (!(parm->r_test == GLP_RT_STD || -#if 1 /* 16/III-2016 */ - parm->r_test == GLP_RT_FLIP || -#endif - parm->r_test == GLP_RT_HAR)) - xerror("glp_simplex: r_test = %d; invalid parameter\n", - parm->r_test); - if (!(0.0 < parm->tol_bnd && parm->tol_bnd < 1.0)) - xerror("glp_simplex: tol_bnd = %g; invalid parameter\n", - parm->tol_bnd); - if (!(0.0 < parm->tol_dj && parm->tol_dj < 1.0)) - xerror("glp_simplex: tol_dj = %g; invalid parameter\n", - parm->tol_dj); - if (!(0.0 < parm->tol_piv && parm->tol_piv < 1.0)) - xerror("glp_simplex: tol_piv = %g; invalid parameter\n", - parm->tol_piv); - if (parm->it_lim < 0) - xerror("glp_simplex: it_lim = %d; invalid parameter\n", - parm->it_lim); - if (parm->tm_lim < 0) - xerror("glp_simplex: tm_lim = %d; invalid parameter\n", - parm->tm_lim); -#if 0 /* 15/VII-2017 */ - if (parm->out_frq < 1) -#else - if (parm->out_frq < 0) -#endif - xerror("glp_simplex: out_frq = %d; invalid parameter\n", - parm->out_frq); - if (parm->out_dly < 0) - xerror("glp_simplex: out_dly = %d; invalid parameter\n", - parm->out_dly); - if (!(parm->presolve == GLP_ON || parm->presolve == GLP_OFF)) - xerror("glp_simplex: presolve = %d; invalid parameter\n", - parm->presolve); -#if 1 /* 11/VII-2017 */ - if (!(parm->excl == GLP_ON || parm->excl == GLP_OFF)) - xerror("glp_simplex: excl = %d; invalid parameter\n", - parm->excl); - if (!(parm->shift == GLP_ON || parm->shift == GLP_OFF)) - xerror("glp_simplex: shift = %d; invalid parameter\n", - parm->shift); - if (!(parm->aorn == GLP_USE_AT || parm->aorn == GLP_USE_NT)) - xerror("glp_simplex: aorn = %d; invalid parameter\n", - parm->aorn); -#endif - /* basic solution is currently undefined */ - P->pbs_stat = P->dbs_stat = GLP_UNDEF; - P->obj_val = 0.0; - P->some = 0; - /* check bounds of double-bounded variables */ - for (i = 1; i <= P->m; i++) - { GLPROW *row = P->row[i]; - if (row->type == GLP_DB && row->lb >= row->ub) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_simplex: row %d: lb = %g, ub = %g; incorrec" - "t bounds\n", i, row->lb, row->ub); - ret = GLP_EBOUND; - goto done; - } - } - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - if (col->type == GLP_DB && col->lb >= col->ub) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_simplex: column %d: lb = %g, ub = %g; incor" - "rect bounds\n", j, col->lb, col->ub); - ret = GLP_EBOUND; - goto done; - } - } - /* solve LP problem */ - if (parm->msg_lev >= GLP_MSG_ALL) - { xprintf("GLPK Simplex Optimizer, v%s\n", glp_version()); - xprintf("%d row%s, %d column%s, %d non-zero%s\n", - P->m, P->m == 1 ? "" : "s", P->n, P->n == 1 ? "" : "s", - P->nnz, P->nnz == 1 ? "" : "s"); - } - if (P->nnz == 0) - trivial_lp(P, parm), ret = 0; - else if (!parm->presolve) - ret = solve_lp(P, parm); - else - ret = preprocess_and_solve_lp(P, parm); -done: /* return to the application program */ - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_init_smcp - initialize simplex method control parameters -* -* SYNOPSIS -* -* void glp_init_smcp(glp_smcp *parm); -* -* DESCRIPTION -* -* The routine glp_init_smcp initializes control parameters, which are -* used by the simplex solver, with default values. -* -* Default values of the control parameters are stored in a glp_smcp -* structure, which the parameter parm points to. */ - -void glp_init_smcp(glp_smcp *parm) -{ parm->msg_lev = GLP_MSG_ALL; - parm->meth = GLP_PRIMAL; - parm->pricing = GLP_PT_PSE; - parm->r_test = GLP_RT_HAR; - parm->tol_bnd = 1e-7; - parm->tol_dj = 1e-7; -#if 0 /* 07/XI-2015 */ - parm->tol_piv = 1e-10; -#else - parm->tol_piv = 1e-9; -#endif - parm->obj_ll = -DBL_MAX; - parm->obj_ul = +DBL_MAX; - parm->it_lim = INT_MAX; - parm->tm_lim = INT_MAX; -#if 0 /* 15/VII-2017 */ - parm->out_frq = 500; -#else - parm->out_frq = 5000; /* 5 seconds */ -#endif - parm->out_dly = 0; - parm->presolve = GLP_OFF; -#if 1 /* 11/VII-2017 */ - parm->excl = GLP_ON; - parm->shift = GLP_ON; - parm->aorn = GLP_USE_NT; -#endif - return; -} - -/*********************************************************************** -* NAME -* -* glp_get_status - retrieve generic status of basic solution -* -* SYNOPSIS -* -* int glp_get_status(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_status reports the generic status of the basic -* solution for the specified problem object as follows: -* -* GLP_OPT - solution is optimal; -* GLP_FEAS - solution is feasible; -* GLP_INFEAS - solution is infeasible; -* GLP_NOFEAS - problem has no feasible solution; -* GLP_UNBND - problem has unbounded solution; -* GLP_UNDEF - solution is undefined. */ - -int glp_get_status(glp_prob *lp) -{ int status; - status = glp_get_prim_stat(lp); - switch (status) - { case GLP_FEAS: - switch (glp_get_dual_stat(lp)) - { case GLP_FEAS: - status = GLP_OPT; - break; - case GLP_NOFEAS: - status = GLP_UNBND; - break; - case GLP_UNDEF: - case GLP_INFEAS: - status = status; - break; - default: - xassert(lp != lp); - } - break; - case GLP_UNDEF: - case GLP_INFEAS: - case GLP_NOFEAS: - status = status; - break; - default: - xassert(lp != lp); - } - return status; -} - -/*********************************************************************** -* NAME -* -* glp_get_prim_stat - retrieve status of primal basic solution -* -* SYNOPSIS -* -* int glp_get_prim_stat(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_prim_stat reports the status of the primal basic -* solution for the specified problem object as follows: -* -* GLP_UNDEF - primal solution is undefined; -* GLP_FEAS - primal solution is feasible; -* GLP_INFEAS - primal solution is infeasible; -* GLP_NOFEAS - no primal feasible solution exists. */ - -int glp_get_prim_stat(glp_prob *lp) -{ int pbs_stat = lp->pbs_stat; - return pbs_stat; -} - -/*********************************************************************** -* NAME -* -* glp_get_dual_stat - retrieve status of dual basic solution -* -* SYNOPSIS -* -* int glp_get_dual_stat(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_dual_stat reports the status of the dual basic -* solution for the specified problem object as follows: -* -* GLP_UNDEF - dual solution is undefined; -* GLP_FEAS - dual solution is feasible; -* GLP_INFEAS - dual solution is infeasible; -* GLP_NOFEAS - no dual feasible solution exists. */ - -int glp_get_dual_stat(glp_prob *lp) -{ int dbs_stat = lp->dbs_stat; - return dbs_stat; -} - -/*********************************************************************** -* NAME -* -* glp_get_obj_val - retrieve objective value (basic solution) -* -* SYNOPSIS -* -* double glp_get_obj_val(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_obj_val returns value of the objective function -* for basic solution. */ - -double glp_get_obj_val(glp_prob *lp) -{ /*struct LPXCPS *cps = lp->cps;*/ - double z; - z = lp->obj_val; - /*if (cps->round && fabs(z) < 1e-9) z = 0.0;*/ - return z; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_stat - retrieve row status -* -* SYNOPSIS -* -* int glp_get_row_stat(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_stat returns current status assigned to the -* auxiliary variable associated with i-th row as follows: -* -* GLP_BS - basic variable; -* GLP_NL - non-basic variable on its lower bound; -* GLP_NU - non-basic variable on its upper bound; -* GLP_NF - non-basic free (unbounded) variable; -* GLP_NS - non-basic fixed variable. */ - -int glp_get_row_stat(glp_prob *lp, int i) -{ if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_stat: i = %d; row number out of range\n", - i); - return lp->row[i]->stat; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_prim - retrieve row primal value (basic solution) -* -* SYNOPSIS -* -* double glp_get_row_prim(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_prim returns primal value of the auxiliary -* variable associated with i-th row. */ - -double glp_get_row_prim(glp_prob *lp, int i) -{ /*struct LPXCPS *cps = lp->cps;*/ - double prim; - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_prim: i = %d; row number out of range\n", - i); - prim = lp->row[i]->prim; - /*if (cps->round && fabs(prim) < 1e-9) prim = 0.0;*/ - return prim; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_dual - retrieve row dual value (basic solution) -* -* SYNOPSIS -* -* double glp_get_row_dual(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_dual returns dual value (i.e. reduced cost) -* of the auxiliary variable associated with i-th row. */ - -double glp_get_row_dual(glp_prob *lp, int i) -{ /*struct LPXCPS *cps = lp->cps;*/ - double dual; - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_dual: i = %d; row number out of range\n", - i); - dual = lp->row[i]->dual; - /*if (cps->round && fabs(dual) < 1e-9) dual = 0.0;*/ - return dual; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_stat - retrieve column status -* -* SYNOPSIS -* -* int glp_get_col_stat(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_stat returns current status assigned to the -* structural variable associated with j-th column as follows: -* -* GLP_BS - basic variable; -* GLP_NL - non-basic variable on its lower bound; -* GLP_NU - non-basic variable on its upper bound; -* GLP_NF - non-basic free (unbounded) variable; -* GLP_NS - non-basic fixed variable. */ - -int glp_get_col_stat(glp_prob *lp, int j) -{ if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_stat: j = %d; column number out of range\n" - , j); - return lp->col[j]->stat; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_prim - retrieve column primal value (basic solution) -* -* SYNOPSIS -* -* double glp_get_col_prim(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_prim returns primal value of the structural -* variable associated with j-th column. */ - -double glp_get_col_prim(glp_prob *lp, int j) -{ /*struct LPXCPS *cps = lp->cps;*/ - double prim; - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_prim: j = %d; column number out of range\n" - , j); - prim = lp->col[j]->prim; - /*if (cps->round && fabs(prim) < 1e-9) prim = 0.0;*/ - return prim; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_dual - retrieve column dual value (basic solution) -* -* SYNOPSIS -* -* double glp_get_col_dual(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_dual returns dual value (i.e. reduced cost) -* of the structural variable associated with j-th column. */ - -double glp_get_col_dual(glp_prob *lp, int j) -{ /*struct LPXCPS *cps = lp->cps;*/ - double dual; - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_dual: j = %d; column number out of range\n" - , j); - dual = lp->col[j]->dual; - /*if (cps->round && fabs(dual) < 1e-9) dual = 0.0;*/ - return dual; -} - -/*********************************************************************** -* NAME -* -* glp_get_unbnd_ray - determine variable causing unboundedness -* -* SYNOPSIS -* -* int glp_get_unbnd_ray(glp_prob *lp); -* -* RETURNS -* -* The routine glp_get_unbnd_ray returns the number k of a variable, -* which causes primal or dual unboundedness. If 1 <= k <= m, it is -* k-th auxiliary variable, and if m+1 <= k <= m+n, it is (k-m)-th -* structural variable, where m is the number of rows, n is the number -* of columns in the problem object. If such variable is not defined, -* the routine returns 0. -* -* COMMENTS -* -* If it is not exactly known which version of the simplex solver -* detected unboundedness, i.e. whether the unboundedness is primal or -* dual, it is sufficient to check the status of the variable reported -* with the routine glp_get_row_stat or glp_get_col_stat. If the -* variable is non-basic, the unboundedness is primal, otherwise, if -* the variable is basic, the unboundedness is dual (the latter case -* means that the problem has no primal feasible dolution). */ - -int glp_get_unbnd_ray(glp_prob *lp) -{ int k; - k = lp->some; - xassert(k >= 0); - if (k > lp->m + lp->n) k = 0; - return k; -} - -#if 1 /* 08/VIII-2013 */ -int glp_get_it_cnt(glp_prob *P) -{ /* get simplex solver iteration count */ - return P->it_cnt; -} -#endif - -#if 1 /* 08/VIII-2013 */ -void glp_set_it_cnt(glp_prob *P, int it_cnt) -{ /* set simplex solver iteration count */ - P->it_cnt = it_cnt; - return; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi07.c b/code/3rd_glpk/draft/glpapi07.c deleted file mode 100644 index 9ac294bd..00000000 --- a/code/3rd_glpk/draft/glpapi07.c +++ /dev/null @@ -1,499 +0,0 @@ -/* glpapi07.c (exact simplex solver) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "draft.h" -#include "glpssx.h" -#include "misc.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_exact - solve LP problem in exact arithmetic -* -* SYNOPSIS -* -* int glp_exact(glp_prob *lp, const glp_smcp *parm); -* -* DESCRIPTION -* -* The routine glp_exact is a tentative implementation of the primal -* two-phase simplex method based on exact (rational) arithmetic. It is -* similar to the routine glp_simplex, however, for all internal -* computations it uses arithmetic of rational numbers, which is exact -* in mathematical sense, i.e. free of round-off errors unlike floating -* point arithmetic. -* -* Note that the routine glp_exact uses inly two control parameters -* passed in the structure glp_smcp, namely, it_lim and tm_lim. -* -* RETURNS -* -* 0 The LP problem instance has been successfully solved. This code -* does not necessarily mean that the solver has found optimal -* solution. It only means that the solution process was successful. -* -* GLP_EBADB -* Unable to start the search, because the initial basis specified -* in the problem object is invalid--the number of basic (auxiliary -* and structural) variables is not the same as the number of rows in -* the problem object. -* -* GLP_ESING -* Unable to start the search, because the basis matrix correspodning -* to the initial basis is exactly singular. -* -* GLP_EBOUND -* Unable to start the search, because some double-bounded variables -* have incorrect bounds. -* -* GLP_EFAIL -* The problem has no rows/columns. -* -* GLP_EITLIM -* The search was prematurely terminated, because the simplex -* iteration limit has been exceeded. -* -* GLP_ETMLIM -* The search was prematurely terminated, because the time limit has -* been exceeded. */ - -static void set_d_eps(mpq_t x, double val) -{ /* convert double val to rational x obtaining a more adequate - fraction than provided by mpq_set_d due to allowing a small - approximation error specified by a given relative tolerance; - for example, mpq_set_d would give the following - 1/3 ~= 0.333333333333333314829616256247391... -> - -> 6004799503160661/18014398509481984 - while this routine gives exactly 1/3 */ - int s, n, j; - double f, p, q, eps = 1e-9; - mpq_t temp; - xassert(-DBL_MAX <= val && val <= +DBL_MAX); -#if 1 /* 30/VII-2008 */ - if (val == floor(val)) - { /* if val is integral, do not approximate */ - mpq_set_d(x, val); - goto done; - } -#endif - if (val > 0.0) - s = +1; - else if (val < 0.0) - s = -1; - else - { mpq_set_si(x, 0, 1); - goto done; - } - f = frexp(fabs(val), &n); - /* |val| = f * 2^n, where 0.5 <= f < 1.0 */ - fp2rat(f, 0.1 * eps, &p, &q); - /* f ~= p / q, where p and q are integers */ - mpq_init(temp); - mpq_set_d(x, p); - mpq_set_d(temp, q); - mpq_div(x, x, temp); - mpq_set_si(temp, 1, 1); - for (j = 1; j <= abs(n); j++) - mpq_add(temp, temp, temp); - if (n > 0) - mpq_mul(x, x, temp); - else if (n < 0) - mpq_div(x, x, temp); - mpq_clear(temp); - if (s < 0) mpq_neg(x, x); - /* check that the desired tolerance has been attained */ - xassert(fabs(val - mpq_get_d(x)) <= eps * (1.0 + fabs(val))); -done: return; -} - -static void load_data(SSX *ssx, glp_prob *lp) -{ /* load LP problem data into simplex solver workspace */ - int m = ssx->m; - int n = ssx->n; - int nnz = ssx->A_ptr[n+1]-1; - int j, k, type, loc, len, *ind; - double lb, ub, coef, *val; - xassert(lp->m == m); - xassert(lp->n == n); - xassert(lp->nnz == nnz); - /* types and bounds of rows and columns */ - for (k = 1; k <= m+n; k++) - { if (k <= m) - { type = lp->row[k]->type; - lb = lp->row[k]->lb; - ub = lp->row[k]->ub; - } - else - { type = lp->col[k-m]->type; - lb = lp->col[k-m]->lb; - ub = lp->col[k-m]->ub; - } - switch (type) - { case GLP_FR: type = SSX_FR; break; - case GLP_LO: type = SSX_LO; break; - case GLP_UP: type = SSX_UP; break; - case GLP_DB: type = SSX_DB; break; - case GLP_FX: type = SSX_FX; break; - default: xassert(type != type); - } - ssx->type[k] = type; - set_d_eps(ssx->lb[k], lb); - set_d_eps(ssx->ub[k], ub); - } - /* optimization direction */ - switch (lp->dir) - { case GLP_MIN: ssx->dir = SSX_MIN; break; - case GLP_MAX: ssx->dir = SSX_MAX; break; - default: xassert(lp != lp); - } - /* objective coefficients */ - for (k = 0; k <= m+n; k++) - { if (k == 0) - coef = lp->c0; - else if (k <= m) - coef = 0.0; - else - coef = lp->col[k-m]->coef; - set_d_eps(ssx->coef[k], coef); - } - /* constraint coefficients */ - ind = xcalloc(1+m, sizeof(int)); - val = xcalloc(1+m, sizeof(double)); - loc = 0; - for (j = 1; j <= n; j++) - { ssx->A_ptr[j] = loc+1; - len = glp_get_mat_col(lp, j, ind, val); - for (k = 1; k <= len; k++) - { loc++; - ssx->A_ind[loc] = ind[k]; - set_d_eps(ssx->A_val[loc], val[k]); - } - } - xassert(loc == nnz); - xfree(ind); - xfree(val); - return; -} - -static int load_basis(SSX *ssx, glp_prob *lp) -{ /* load current LP basis into simplex solver workspace */ - int m = ssx->m; - int n = ssx->n; - int *type = ssx->type; - int *stat = ssx->stat; - int *Q_row = ssx->Q_row; - int *Q_col = ssx->Q_col; - int i, j, k; - xassert(lp->m == m); - xassert(lp->n == n); - /* statuses of rows and columns */ - for (k = 1; k <= m+n; k++) - { if (k <= m) - stat[k] = lp->row[k]->stat; - else - stat[k] = lp->col[k-m]->stat; - switch (stat[k]) - { case GLP_BS: - stat[k] = SSX_BS; - break; - case GLP_NL: - stat[k] = SSX_NL; - xassert(type[k] == SSX_LO || type[k] == SSX_DB); - break; - case GLP_NU: - stat[k] = SSX_NU; - xassert(type[k] == SSX_UP || type[k] == SSX_DB); - break; - case GLP_NF: - stat[k] = SSX_NF; - xassert(type[k] == SSX_FR); - break; - case GLP_NS: - stat[k] = SSX_NS; - xassert(type[k] == SSX_FX); - break; - default: - xassert(stat != stat); - } - } - /* build permutation matix Q */ - i = j = 0; - for (k = 1; k <= m+n; k++) - { if (stat[k] == SSX_BS) - { i++; - if (i > m) return 1; - Q_row[k] = i, Q_col[i] = k; - } - else - { j++; - if (j > n) return 1; - Q_row[k] = m+j, Q_col[m+j] = k; - } - } - xassert(i == m && j == n); - return 0; -} - -int glp_exact(glp_prob *lp, const glp_smcp *parm) -{ glp_smcp _parm; - SSX *ssx; - int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - int i, j, k, type, pst, dst, ret, stat; - double lb, ub, prim, dual, sum; - if (parm == NULL) - parm = &_parm, glp_init_smcp((glp_smcp *)parm); - /* check control parameters */ -#if 1 /* 25/XI-2017 */ - switch (parm->msg_lev) - { case GLP_MSG_OFF: - case GLP_MSG_ERR: - case GLP_MSG_ON: - case GLP_MSG_ALL: - case GLP_MSG_DBG: - break; - default: - xerror("glp_exact: msg_lev = %d; invalid parameter\n", - parm->msg_lev); - } -#endif - if (parm->it_lim < 0) - xerror("glp_exact: it_lim = %d; invalid parameter\n", - parm->it_lim); - if (parm->tm_lim < 0) - xerror("glp_exact: tm_lim = %d; invalid parameter\n", - parm->tm_lim); - /* the problem must have at least one row and one column */ - if (!(m > 0 && n > 0)) -#if 0 /* 25/XI-2017 */ - { xprintf("glp_exact: problem has no rows/columns\n"); -#else - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_exact: problem has no rows/columns\n"); -#endif - return GLP_EFAIL; - } -#if 1 - /* basic solution is currently undefined */ - lp->pbs_stat = lp->dbs_stat = GLP_UNDEF; - lp->obj_val = 0.0; - lp->some = 0; -#endif - /* check that all double-bounded variables have correct bounds */ - for (k = 1; k <= m+n; k++) - { if (k <= m) - { type = lp->row[k]->type; - lb = lp->row[k]->lb; - ub = lp->row[k]->ub; - } - else - { type = lp->col[k-m]->type; - lb = lp->col[k-m]->lb; - ub = lp->col[k-m]->ub; - } - if (type == GLP_DB && lb >= ub) -#if 0 /* 25/XI-2017 */ - { xprintf("glp_exact: %s %d has invalid bounds\n", - k <= m ? "row" : "column", k <= m ? k : k-m); -#else - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_exact: %s %d has invalid bounds\n", - k <= m ? "row" : "column", k <= m ? k : k-m); -#endif - return GLP_EBOUND; - } - } - /* create the simplex solver workspace */ -#if 1 /* 25/XI-2017 */ - if (parm->msg_lev >= GLP_MSG_ALL) - { -#endif - xprintf("glp_exact: %d rows, %d columns, %d non-zeros\n", - m, n, nnz); -#ifdef HAVE_GMP - xprintf("GNU MP bignum library is being used\n"); -#else - xprintf("GLPK bignum module is being used\n"); - xprintf("(Consider installing GNU MP to attain a much better perf" - "ormance.)\n"); -#endif -#if 1 /* 25/XI-2017 */ - } -#endif - ssx = ssx_create(m, n, nnz); - /* load LP problem data into the workspace */ - load_data(ssx, lp); - /* load current LP basis into the workspace */ - if (load_basis(ssx, lp)) -#if 0 /* 25/XI-2017 */ - { xprintf("glp_exact: initial LP basis is invalid\n"); -#else - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_exact: initial LP basis is invalid\n"); -#endif - ret = GLP_EBADB; - goto done; - } -#if 0 - /* inherit some control parameters from the LP object */ - ssx->it_lim = lpx_get_int_parm(lp, LPX_K_ITLIM); - ssx->it_cnt = lpx_get_int_parm(lp, LPX_K_ITCNT); - ssx->tm_lim = lpx_get_real_parm(lp, LPX_K_TMLIM); -#else -#if 1 /* 25/XI-2017 */ - ssx->msg_lev = parm->msg_lev; -#endif - ssx->it_lim = parm->it_lim; - ssx->it_cnt = lp->it_cnt; - ssx->tm_lim = (double)parm->tm_lim / 1000.0; -#endif - ssx->out_frq = 5.0; - ssx->tm_beg = xtime(); -#if 0 /* 10/VI-2013 */ - ssx->tm_lag = xlset(0); -#else - ssx->tm_lag = 0.0; -#endif - /* solve LP */ - ret = ssx_driver(ssx); -#if 0 - /* copy back some statistics to the LP object */ - lpx_set_int_parm(lp, LPX_K_ITLIM, ssx->it_lim); - lpx_set_int_parm(lp, LPX_K_ITCNT, ssx->it_cnt); - lpx_set_real_parm(lp, LPX_K_TMLIM, ssx->tm_lim); -#else - lp->it_cnt = ssx->it_cnt; -#endif - /* analyze the return code */ - switch (ret) - { case 0: - /* optimal solution found */ - ret = 0; - pst = dst = GLP_FEAS; - break; - case 1: - /* problem has no feasible solution */ - ret = 0; - pst = GLP_NOFEAS, dst = GLP_INFEAS; - break; - case 2: - /* problem has unbounded solution */ - ret = 0; - pst = GLP_FEAS, dst = GLP_NOFEAS; -#if 1 - xassert(1 <= ssx->q && ssx->q <= n); - lp->some = ssx->Q_col[m + ssx->q]; - xassert(1 <= lp->some && lp->some <= m+n); -#endif - break; - case 3: - /* iteration limit exceeded (phase I) */ - ret = GLP_EITLIM; - pst = dst = GLP_INFEAS; - break; - case 4: - /* iteration limit exceeded (phase II) */ - ret = GLP_EITLIM; - pst = GLP_FEAS, dst = GLP_INFEAS; - break; - case 5: - /* time limit exceeded (phase I) */ - ret = GLP_ETMLIM; - pst = dst = GLP_INFEAS; - break; - case 6: - /* time limit exceeded (phase II) */ - ret = GLP_ETMLIM; - pst = GLP_FEAS, dst = GLP_INFEAS; - break; - case 7: - /* initial basis matrix is singular */ - ret = GLP_ESING; - goto done; - default: - xassert(ret != ret); - } - /* store final basic solution components into LP object */ - lp->pbs_stat = pst; - lp->dbs_stat = dst; - sum = lp->c0; - for (k = 1; k <= m+n; k++) - { if (ssx->stat[k] == SSX_BS) - { i = ssx->Q_row[k]; /* x[k] = xB[i] */ - xassert(1 <= i && i <= m); - stat = GLP_BS; - prim = mpq_get_d(ssx->bbar[i]); - dual = 0.0; - } - else - { j = ssx->Q_row[k] - m; /* x[k] = xN[j] */ - xassert(1 <= j && j <= n); - switch (ssx->stat[k]) - { case SSX_NF: - stat = GLP_NF; - prim = 0.0; - break; - case SSX_NL: - stat = GLP_NL; - prim = mpq_get_d(ssx->lb[k]); - break; - case SSX_NU: - stat = GLP_NU; - prim = mpq_get_d(ssx->ub[k]); - break; - case SSX_NS: - stat = GLP_NS; - prim = mpq_get_d(ssx->lb[k]); - break; - default: - xassert(ssx != ssx); - } - dual = mpq_get_d(ssx->cbar[j]); - } - if (k <= m) - { glp_set_row_stat(lp, k, stat); - lp->row[k]->prim = prim; - lp->row[k]->dual = dual; - } - else - { glp_set_col_stat(lp, k-m, stat); - lp->col[k-m]->prim = prim; - lp->col[k-m]->dual = dual; - sum += lp->col[k-m]->coef * prim; - } - } - lp->obj_val = sum; -done: /* delete the simplex solver workspace */ - ssx_delete(ssx); -#if 1 /* 23/XI-2015 */ - xassert(gmp_pool_count() == 0); - gmp_free_mem(); -#endif - /* return to the application program */ - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi08.c b/code/3rd_glpk/draft/glpapi08.c deleted file mode 100644 index 652292cb..00000000 --- a/code/3rd_glpk/draft/glpapi08.c +++ /dev/null @@ -1,388 +0,0 @@ -/* glpapi08.c (interior-point method routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpipm.h" -#include "npp.h" - -/*********************************************************************** -* NAME -* -* glp_interior - solve LP problem with the interior-point method -* -* SYNOPSIS -* -* int glp_interior(glp_prob *P, const glp_iptcp *parm); -* -* The routine glp_interior is a driver to the LP solver based on the -* interior-point method. -* -* The interior-point solver has a set of control parameters. Values of -* the control parameters can be passed in a structure glp_iptcp, which -* the parameter parm points to. -* -* Currently this routine implements an easy variant of the primal-dual -* interior-point method based on Mehrotra's technique. -* -* This routine transforms the original LP problem to an equivalent LP -* problem in the standard formulation (all constraints are equalities, -* all variables are non-negative), calls the routine ipm_main to solve -* the transformed problem, and then transforms an obtained solution to -* the solution of the original problem. -* -* RETURNS -* -* 0 The LP problem instance has been successfully solved. This code -* does not necessarily mean that the solver has found optimal -* solution. It only means that the solution process was successful. -* -* GLP_EFAIL -* The problem has no rows/columns. -* -* GLP_ENOCVG -* Very slow convergence or divergence. -* -* GLP_EITLIM -* Iteration limit exceeded. -* -* GLP_EINSTAB -* Numerical instability on solving Newtonian system. */ - -static void transform(NPP *npp) -{ /* transform LP to the standard formulation */ - NPPROW *row, *prev_row; - NPPCOL *col, *prev_col; - for (row = npp->r_tail; row != NULL; row = prev_row) - { prev_row = row->prev; - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) - npp_free_row(npp, row); - else if (row->lb == -DBL_MAX) - npp_leq_row(npp, row); - else if (row->ub == +DBL_MAX) - npp_geq_row(npp, row); - else if (row->lb != row->ub) - { if (fabs(row->lb) < fabs(row->ub)) - npp_geq_row(npp, row); - else - npp_leq_row(npp, row); - } - } - for (col = npp->c_tail; col != NULL; col = prev_col) - { prev_col = col->prev; - if (col->lb == -DBL_MAX && col->ub == +DBL_MAX) - npp_free_col(npp, col); - else if (col->lb == -DBL_MAX) - npp_ubnd_col(npp, col); - else if (col->ub == +DBL_MAX) - { if (col->lb != 0.0) - npp_lbnd_col(npp, col); - } - else if (col->lb != col->ub) - { if (fabs(col->lb) < fabs(col->ub)) - { if (col->lb != 0.0) - npp_lbnd_col(npp, col); - } - else - npp_ubnd_col(npp, col); - npp_dbnd_col(npp, col); - } - else - npp_fixed_col(npp, col); - } - for (row = npp->r_head; row != NULL; row = row->next) - xassert(row->lb == row->ub); - for (col = npp->c_head; col != NULL; col = col->next) - xassert(col->lb == 0.0 && col->ub == +DBL_MAX); - return; -} - -int glp_interior(glp_prob *P, const glp_iptcp *parm) -{ glp_iptcp _parm; - GLPROW *row; - GLPCOL *col; - NPP *npp = NULL; - glp_prob *prob = NULL; - int i, j, ret; - /* check control parameters */ - if (parm == NULL) - glp_init_iptcp(&_parm), parm = &_parm; - if (!(parm->msg_lev == GLP_MSG_OFF || - parm->msg_lev == GLP_MSG_ERR || - parm->msg_lev == GLP_MSG_ON || - parm->msg_lev == GLP_MSG_ALL)) - xerror("glp_interior: msg_lev = %d; invalid parameter\n", - parm->msg_lev); - if (!(parm->ord_alg == GLP_ORD_NONE || - parm->ord_alg == GLP_ORD_QMD || - parm->ord_alg == GLP_ORD_AMD || - parm->ord_alg == GLP_ORD_SYMAMD)) - xerror("glp_interior: ord_alg = %d; invalid parameter\n", - parm->ord_alg); - /* interior-point solution is currently undefined */ - P->ipt_stat = GLP_UNDEF; - P->ipt_obj = 0.0; - /* check bounds of double-bounded variables */ - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->type == GLP_DB && row->lb >= row->ub) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_interior: row %d: lb = %g, ub = %g; incorre" - "ct bounds\n", i, row->lb, row->ub); - ret = GLP_EBOUND; - goto done; - } - } - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->type == GLP_DB && col->lb >= col->ub) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_interior: column %d: lb = %g, ub = %g; inco" - "rrect bounds\n", j, col->lb, col->ub); - ret = GLP_EBOUND; - goto done; - } - } - /* transform LP to the standard formulation */ - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("Original LP has %d row(s), %d column(s), and %d non-z" - "ero(s)\n", P->m, P->n, P->nnz); - npp = npp_create_wksp(); - npp_load_prob(npp, P, GLP_OFF, GLP_IPT, GLP_ON); - transform(npp); - prob = glp_create_prob(); - npp_build_prob(npp, prob); - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("Working LP has %d row(s), %d column(s), and %d non-ze" - "ro(s)\n", prob->m, prob->n, prob->nnz); -#if 1 - /* currently empty problem cannot be solved */ - if (!(prob->m > 0 && prob->n > 0)) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_interior: unable to solve empty problem\n"); - ret = GLP_EFAIL; - goto done; - } -#endif - /* scale the resultant LP */ - { ENV *env = get_env_ptr(); - int term_out = env->term_out; - env->term_out = GLP_OFF; - glp_scale_prob(prob, GLP_SF_EQ); - env->term_out = term_out; - } - /* warn about dense columns */ - if (parm->msg_lev >= GLP_MSG_ON && prob->m >= 200) - { int len, cnt = 0; - for (j = 1; j <= prob->n; j++) - { len = glp_get_mat_col(prob, j, NULL, NULL); - if ((double)len >= 0.20 * (double)prob->m) cnt++; - } - if (cnt == 1) - xprintf("WARNING: PROBLEM HAS ONE DENSE COLUMN\n"); - else if (cnt > 0) - xprintf("WARNING: PROBLEM HAS %d DENSE COLUMNS\n", cnt); - } - /* solve the transformed LP */ - ret = ipm_solve(prob, parm); - /* postprocess solution from the transformed LP */ - npp_postprocess(npp, prob); - /* and store solution to the original LP */ - npp_unload_sol(npp, P); -done: /* free working program objects */ - if (npp != NULL) npp_delete_wksp(npp); - if (prob != NULL) glp_delete_prob(prob); - /* return to the application program */ - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_init_iptcp - initialize interior-point solver control parameters -* -* SYNOPSIS -* -* void glp_init_iptcp(glp_iptcp *parm); -* -* DESCRIPTION -* -* The routine glp_init_iptcp initializes control parameters, which are -* used by the interior-point solver, with default values. -* -* Default values of the control parameters are stored in the glp_iptcp -* structure, which the parameter parm points to. */ - -void glp_init_iptcp(glp_iptcp *parm) -{ parm->msg_lev = GLP_MSG_ALL; - parm->ord_alg = GLP_ORD_AMD; - return; -} - -/*********************************************************************** -* NAME -* -* glp_ipt_status - retrieve status of interior-point solution -* -* SYNOPSIS -* -* int glp_ipt_status(glp_prob *lp); -* -* RETURNS -* -* The routine glp_ipt_status reports the status of solution found by -* the interior-point solver as follows: -* -* GLP_UNDEF - interior-point solution is undefined; -* GLP_OPT - interior-point solution is optimal; -* GLP_INFEAS - interior-point solution is infeasible; -* GLP_NOFEAS - no feasible solution exists. */ - -int glp_ipt_status(glp_prob *lp) -{ int ipt_stat = lp->ipt_stat; - return ipt_stat; -} - -/*********************************************************************** -* NAME -* -* glp_ipt_obj_val - retrieve objective value (interior point) -* -* SYNOPSIS -* -* double glp_ipt_obj_val(glp_prob *lp); -* -* RETURNS -* -* The routine glp_ipt_obj_val returns value of the objective function -* for interior-point solution. */ - -double glp_ipt_obj_val(glp_prob *lp) -{ /*struct LPXCPS *cps = lp->cps;*/ - double z; - z = lp->ipt_obj; - /*if (cps->round && fabs(z) < 1e-9) z = 0.0;*/ - return z; -} - -/*********************************************************************** -* NAME -* -* glp_ipt_row_prim - retrieve row primal value (interior point) -* -* SYNOPSIS -* -* double glp_ipt_row_prim(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_ipt_row_prim returns primal value of the auxiliary -* variable associated with i-th row. */ - -double glp_ipt_row_prim(glp_prob *lp, int i) -{ /*struct LPXCPS *cps = lp->cps;*/ - double pval; - if (!(1 <= i && i <= lp->m)) - xerror("glp_ipt_row_prim: i = %d; row number out of range\n", - i); - pval = lp->row[i]->pval; - /*if (cps->round && fabs(pval) < 1e-9) pval = 0.0;*/ - return pval; -} - -/*********************************************************************** -* NAME -* -* glp_ipt_row_dual - retrieve row dual value (interior point) -* -* SYNOPSIS -* -* double glp_ipt_row_dual(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_ipt_row_dual returns dual value (i.e. reduced cost) -* of the auxiliary variable associated with i-th row. */ - -double glp_ipt_row_dual(glp_prob *lp, int i) -{ /*struct LPXCPS *cps = lp->cps;*/ - double dval; - if (!(1 <= i && i <= lp->m)) - xerror("glp_ipt_row_dual: i = %d; row number out of range\n", - i); - dval = lp->row[i]->dval; - /*if (cps->round && fabs(dval) < 1e-9) dval = 0.0;*/ - return dval; -} - -/*********************************************************************** -* NAME -* -* glp_ipt_col_prim - retrieve column primal value (interior point) -* -* SYNOPSIS -* -* double glp_ipt_col_prim(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_ipt_col_prim returns primal value of the structural -* variable associated with j-th column. */ - -double glp_ipt_col_prim(glp_prob *lp, int j) -{ /*struct LPXCPS *cps = lp->cps;*/ - double pval; - if (!(1 <= j && j <= lp->n)) - xerror("glp_ipt_col_prim: j = %d; column number out of range\n" - , j); - pval = lp->col[j]->pval; - /*if (cps->round && fabs(pval) < 1e-9) pval = 0.0;*/ - return pval; -} - -/*********************************************************************** -* NAME -* -* glp_ipt_col_dual - retrieve column dual value (interior point) -* -* SYNOPSIS -* -* double glp_ipt_col_dual(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_ipt_col_dual returns dual value (i.e. reduced cost) -* of the structural variable associated with j-th column. */ - -double glp_ipt_col_dual(glp_prob *lp, int j) -{ /*struct LPXCPS *cps = lp->cps;*/ - double dval; - if (!(1 <= j && j <= lp->n)) - xerror("glp_ipt_col_dual: j = %d; column number out of range\n" - , j); - dval = lp->col[j]->dval; - /*if (cps->round && fabs(dval) < 1e-9) dval = 0.0;*/ - return dval; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi09.c b/code/3rd_glpk/draft/glpapi09.c deleted file mode 100644 index 0d3ab57b..00000000 --- a/code/3rd_glpk/draft/glpapi09.c +++ /dev/null @@ -1,798 +0,0 @@ -/* glpapi09.c (mixed integer programming routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "draft.h" -#include "env.h" -#include "ios.h" -#include "npp.h" - -/*********************************************************************** -* NAME -* -* glp_set_col_kind - set (change) column kind -* -* SYNOPSIS -* -* void glp_set_col_kind(glp_prob *mip, int j, int kind); -* -* DESCRIPTION -* -* The routine glp_set_col_kind sets (changes) the kind of j-th column -* (structural variable) as specified by the parameter kind: -* -* GLP_CV - continuous variable; -* GLP_IV - integer variable; -* GLP_BV - binary variable. */ - -void glp_set_col_kind(glp_prob *mip, int j, int kind) -{ GLPCOL *col; - if (!(1 <= j && j <= mip->n)) - xerror("glp_set_col_kind: j = %d; column number out of range\n" - , j); - col = mip->col[j]; - switch (kind) - { case GLP_CV: - col->kind = GLP_CV; - break; - case GLP_IV: - col->kind = GLP_IV; - break; - case GLP_BV: - col->kind = GLP_IV; - if (!(col->type == GLP_DB && col->lb == 0.0 && col->ub == - 1.0)) glp_set_col_bnds(mip, j, GLP_DB, 0.0, 1.0); - break; - default: - xerror("glp_set_col_kind: j = %d; kind = %d; invalid column" - " kind\n", j, kind); - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_kind - retrieve column kind -* -* SYNOPSIS -* -* int glp_get_col_kind(glp_prob *mip, int j); -* -* RETURNS -* -* The routine glp_get_col_kind returns the kind of j-th column, i.e. -* the kind of corresponding structural variable, as follows: -* -* GLP_CV - continuous variable; -* GLP_IV - integer variable; -* GLP_BV - binary variable */ - -int glp_get_col_kind(glp_prob *mip, int j) -{ GLPCOL *col; - int kind; - if (!(1 <= j && j <= mip->n)) - xerror("glp_get_col_kind: j = %d; column number out of range\n" - , j); - col = mip->col[j]; - kind = col->kind; - switch (kind) - { case GLP_CV: - break; - case GLP_IV: - if (col->type == GLP_DB && col->lb == 0.0 && col->ub == 1.0) - kind = GLP_BV; - break; - default: - xassert(kind != kind); - } - return kind; -} - -/*********************************************************************** -* NAME -* -* glp_get_num_int - retrieve number of integer columns -* -* SYNOPSIS -* -* int glp_get_num_int(glp_prob *mip); -* -* RETURNS -* -* The routine glp_get_num_int returns the current number of columns, -* which are marked as integer. */ - -int glp_get_num_int(glp_prob *mip) -{ GLPCOL *col; - int j, count = 0; - for (j = 1; j <= mip->n; j++) - { col = mip->col[j]; - if (col->kind == GLP_IV) count++; - } - return count; -} - -/*********************************************************************** -* NAME -* -* glp_get_num_bin - retrieve number of binary columns -* -* SYNOPSIS -* -* int glp_get_num_bin(glp_prob *mip); -* -* RETURNS -* -* The routine glp_get_num_bin returns the current number of columns, -* which are marked as binary. */ - -int glp_get_num_bin(glp_prob *mip) -{ GLPCOL *col; - int j, count = 0; - for (j = 1; j <= mip->n; j++) - { col = mip->col[j]; - if (col->kind == GLP_IV && col->type == GLP_DB && col->lb == - 0.0 && col->ub == 1.0) count++; - } - return count; -} - -/*********************************************************************** -* NAME -* -* glp_intopt - solve MIP problem with the branch-and-bound method -* -* SYNOPSIS -* -* int glp_intopt(glp_prob *P, const glp_iocp *parm); -* -* DESCRIPTION -* -* The routine glp_intopt is a driver to the MIP solver based on the -* branch-and-bound method. -* -* On entry the problem object should contain optimal solution to LP -* relaxation (which can be obtained with the routine glp_simplex). -* -* The MIP solver has a set of control parameters. Values of the control -* parameters can be passed in a structure glp_iocp, which the parameter -* parm points to. -* -* The parameter parm can be specified as NULL, in which case the MIP -* solver uses default settings. -* -* RETURNS -* -* 0 The MIP problem instance has been successfully solved. This code -* does not necessarily mean that the solver has found optimal -* solution. It only means that the solution process was successful. -* -* GLP_EBOUND -* Unable to start the search, because some double-bounded variables -* have incorrect bounds or some integer variables have non-integer -* (fractional) bounds. -* -* GLP_EROOT -* Unable to start the search, because optimal basis for initial LP -* relaxation is not provided. -* -* GLP_EFAIL -* The search was prematurely terminated due to the solver failure. -* -* GLP_EMIPGAP -* The search was prematurely terminated, because the relative mip -* gap tolerance has been reached. -* -* GLP_ETMLIM -* The search was prematurely terminated, because the time limit has -* been exceeded. -* -* GLP_ENOPFS -* The MIP problem instance has no primal feasible solution (only if -* the MIP presolver is used). -* -* GLP_ENODFS -* LP relaxation of the MIP problem instance has no dual feasible -* solution (only if the MIP presolver is used). -* -* GLP_ESTOP -* The search was prematurely terminated by application. */ - -#if 0 /* 11/VII-2013 */ -static int solve_mip(glp_prob *P, const glp_iocp *parm) -#else -static int solve_mip(glp_prob *P, const glp_iocp *parm, - glp_prob *P0 /* problem passed to glp_intopt */, - NPP *npp /* preprocessor workspace or NULL */) -#endif -{ /* solve MIP directly without using the preprocessor */ - glp_tree *T; - int ret; - /* optimal basis to LP relaxation must be provided */ - if (glp_get_status(P) != GLP_OPT) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: optimal basis to initial LP relaxation" - " not provided\n"); - ret = GLP_EROOT; - goto done; - } - /* it seems all is ok */ - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("Integer optimization begins...\n"); - /* create the branch-and-bound tree */ - T = ios_create_tree(P, parm); -#if 1 /* 11/VII-2013 */ - T->P = P0; - T->npp = npp; -#endif - /* solve the problem instance */ - ret = ios_driver(T); - /* delete the branch-and-bound tree */ - ios_delete_tree(T); - /* analyze exit code reported by the mip driver */ - if (ret == 0) - { if (P->mip_stat == GLP_FEAS) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("INTEGER OPTIMAL SOLUTION FOUND\n"); - P->mip_stat = GLP_OPT; - } - else - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("PROBLEM HAS NO INTEGER FEASIBLE SOLUTION\n"); - P->mip_stat = GLP_NOFEAS; - } - } - else if (ret == GLP_EMIPGAP) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("RELATIVE MIP GAP TOLERANCE REACHED; SEARCH TERMINA" - "TED\n"); - } - else if (ret == GLP_ETMLIM) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("TIME LIMIT EXCEEDED; SEARCH TERMINATED\n"); - } - else if (ret == GLP_EFAIL) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: cannot solve current LP relaxation\n"); - } - else if (ret == GLP_ESTOP) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("SEARCH TERMINATED BY APPLICATION\n"); - } - else - xassert(ret != ret); -done: return ret; -} - -static int preprocess_and_solve_mip(glp_prob *P, const glp_iocp *parm) -{ /* solve MIP using the preprocessor */ - ENV *env = get_env_ptr(); - int term_out = env->term_out; - NPP *npp; - glp_prob *mip = NULL; - glp_bfcp bfcp; - glp_smcp smcp; - int ret; - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("Preprocessing...\n"); - /* create preprocessor workspace */ - npp = npp_create_wksp(); - /* load original problem into the preprocessor workspace */ - npp_load_prob(npp, P, GLP_OFF, GLP_MIP, GLP_OFF); - /* process MIP prior to applying the branch-and-bound method */ - if (!term_out || parm->msg_lev < GLP_MSG_ALL) - env->term_out = GLP_OFF; - else - env->term_out = GLP_ON; - ret = npp_integer(npp, parm); - env->term_out = term_out; - if (ret == 0) - ; - else if (ret == GLP_ENOPFS) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("PROBLEM HAS NO PRIMAL FEASIBLE SOLUTION\n"); - } - else if (ret == GLP_ENODFS) - { if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("LP RELAXATION HAS NO DUAL FEASIBLE SOLUTION\n"); - } - else - xassert(ret != ret); - if (ret != 0) goto done; - /* build transformed MIP */ - mip = glp_create_prob(); - npp_build_prob(npp, mip); - /* if the transformed MIP is empty, it has empty solution, which - is optimal */ - if (mip->m == 0 && mip->n == 0) - { mip->mip_stat = GLP_OPT; - mip->mip_obj = mip->c0; - if (parm->msg_lev >= GLP_MSG_ALL) - { xprintf("Objective value = %17.9e\n", mip->mip_obj); - xprintf("INTEGER OPTIMAL SOLUTION FOUND BY MIP PREPROCESSOR" - "\n"); - } - goto post; - } - /* display some statistics */ - if (parm->msg_lev >= GLP_MSG_ALL) - { int ni = glp_get_num_int(mip); - int nb = glp_get_num_bin(mip); - char s[50]; - xprintf("%d row%s, %d column%s, %d non-zero%s\n", - mip->m, mip->m == 1 ? "" : "s", mip->n, mip->n == 1 ? "" : - "s", mip->nnz, mip->nnz == 1 ? "" : "s"); - if (nb == 0) - strcpy(s, "none of"); - else if (ni == 1 && nb == 1) - strcpy(s, ""); - else if (nb == 1) - strcpy(s, "one of"); - else if (nb == ni) - strcpy(s, "all of"); - else - sprintf(s, "%d of", nb); - xprintf("%d integer variable%s, %s which %s binary\n", - ni, ni == 1 ? "" : "s", s, nb == 1 ? "is" : "are"); - } - /* inherit basis factorization control parameters */ - glp_get_bfcp(P, &bfcp); - glp_set_bfcp(mip, &bfcp); - /* scale the transformed problem */ - if (!term_out || parm->msg_lev < GLP_MSG_ALL) - env->term_out = GLP_OFF; - else - env->term_out = GLP_ON; - glp_scale_prob(mip, - GLP_SF_GM | GLP_SF_EQ | GLP_SF_2N | GLP_SF_SKIP); - env->term_out = term_out; - /* build advanced initial basis */ - if (!term_out || parm->msg_lev < GLP_MSG_ALL) - env->term_out = GLP_OFF; - else - env->term_out = GLP_ON; - glp_adv_basis(mip, 0); - env->term_out = term_out; - /* solve initial LP relaxation */ - if (parm->msg_lev >= GLP_MSG_ALL) - xprintf("Solving LP relaxation...\n"); - glp_init_smcp(&smcp); - smcp.msg_lev = parm->msg_lev; - /* respect time limit */ - smcp.tm_lim = parm->tm_lim; - mip->it_cnt = P->it_cnt; - ret = glp_simplex(mip, &smcp); - P->it_cnt = mip->it_cnt; - if (ret == GLP_ETMLIM) - goto done; - else if (ret != 0) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: cannot solve LP relaxation\n"); - ret = GLP_EFAIL; - goto done; - } - /* check status of the basic solution */ - ret = glp_get_status(mip); - if (ret == GLP_OPT) - ret = 0; - else if (ret == GLP_NOFEAS) - ret = GLP_ENOPFS; - else if (ret == GLP_UNBND) - ret = GLP_ENODFS; - else - xassert(ret != ret); - if (ret != 0) goto done; - /* solve the transformed MIP */ - mip->it_cnt = P->it_cnt; -#if 0 /* 11/VII-2013 */ - ret = solve_mip(mip, parm); -#else - if (parm->use_sol) - { mip->mip_stat = P->mip_stat; - mip->mip_obj = P->mip_obj; - } - ret = solve_mip(mip, parm, P, npp); -#endif - P->it_cnt = mip->it_cnt; - /* only integer feasible solution can be postprocessed */ - if (!(mip->mip_stat == GLP_OPT || mip->mip_stat == GLP_FEAS)) - { P->mip_stat = mip->mip_stat; - goto done; - } - /* postprocess solution from the transformed MIP */ -post: npp_postprocess(npp, mip); - /* the transformed MIP is no longer needed */ - glp_delete_prob(mip), mip = NULL; - /* store solution to the original problem */ - npp_unload_sol(npp, P); -done: /* delete the transformed MIP, if it exists */ - if (mip != NULL) glp_delete_prob(mip); - /* delete preprocessor workspace */ - npp_delete_wksp(npp); - return ret; -} - -#ifndef HAVE_ALIEN_SOLVER /* 28/V-2010 */ -int _glp_intopt1(glp_prob *P, const glp_iocp *parm) -{ xassert(P == P); - xassert(parm == parm); - xprintf("glp_intopt: no alien solver is available\n"); - return GLP_EFAIL; -} -#endif - -int glp_intopt(glp_prob *P, const glp_iocp *parm) -{ /* solve MIP problem with the branch-and-bound method */ - glp_iocp _parm; - int i, j, ret; -#if 0 /* 04/IV-2016 */ - /* check problem object */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_intopt: P = %p; invalid problem object\n", P); -#endif - if (P->tree != NULL) - xerror("glp_intopt: operation not allowed\n"); - /* check control parameters */ - if (parm == NULL) - parm = &_parm, glp_init_iocp((glp_iocp *)parm); - if (!(parm->msg_lev == GLP_MSG_OFF || - parm->msg_lev == GLP_MSG_ERR || - parm->msg_lev == GLP_MSG_ON || - parm->msg_lev == GLP_MSG_ALL || - parm->msg_lev == GLP_MSG_DBG)) - xerror("glp_intopt: msg_lev = %d; invalid parameter\n", - parm->msg_lev); - if (!(parm->br_tech == GLP_BR_FFV || - parm->br_tech == GLP_BR_LFV || - parm->br_tech == GLP_BR_MFV || - parm->br_tech == GLP_BR_DTH || - parm->br_tech == GLP_BR_PCH)) - xerror("glp_intopt: br_tech = %d; invalid parameter\n", - parm->br_tech); - if (!(parm->bt_tech == GLP_BT_DFS || - parm->bt_tech == GLP_BT_BFS || - parm->bt_tech == GLP_BT_BLB || - parm->bt_tech == GLP_BT_BPH)) - xerror("glp_intopt: bt_tech = %d; invalid parameter\n", - parm->bt_tech); - if (!(0.0 < parm->tol_int && parm->tol_int < 1.0)) - xerror("glp_intopt: tol_int = %g; invalid parameter\n", - parm->tol_int); - if (!(0.0 < parm->tol_obj && parm->tol_obj < 1.0)) - xerror("glp_intopt: tol_obj = %g; invalid parameter\n", - parm->tol_obj); - if (parm->tm_lim < 0) - xerror("glp_intopt: tm_lim = %d; invalid parameter\n", - parm->tm_lim); - if (parm->out_frq < 0) - xerror("glp_intopt: out_frq = %d; invalid parameter\n", - parm->out_frq); - if (parm->out_dly < 0) - xerror("glp_intopt: out_dly = %d; invalid parameter\n", - parm->out_dly); - if (!(0 <= parm->cb_size && parm->cb_size <= 256)) - xerror("glp_intopt: cb_size = %d; invalid parameter\n", - parm->cb_size); - if (!(parm->pp_tech == GLP_PP_NONE || - parm->pp_tech == GLP_PP_ROOT || - parm->pp_tech == GLP_PP_ALL)) - xerror("glp_intopt: pp_tech = %d; invalid parameter\n", - parm->pp_tech); - if (parm->mip_gap < 0.0) - xerror("glp_intopt: mip_gap = %g; invalid parameter\n", - parm->mip_gap); - if (!(parm->mir_cuts == GLP_ON || parm->mir_cuts == GLP_OFF)) - xerror("glp_intopt: mir_cuts = %d; invalid parameter\n", - parm->mir_cuts); - if (!(parm->gmi_cuts == GLP_ON || parm->gmi_cuts == GLP_OFF)) - xerror("glp_intopt: gmi_cuts = %d; invalid parameter\n", - parm->gmi_cuts); - if (!(parm->cov_cuts == GLP_ON || parm->cov_cuts == GLP_OFF)) - xerror("glp_intopt: cov_cuts = %d; invalid parameter\n", - parm->cov_cuts); - if (!(parm->clq_cuts == GLP_ON || parm->clq_cuts == GLP_OFF)) - xerror("glp_intopt: clq_cuts = %d; invalid parameter\n", - parm->clq_cuts); - if (!(parm->presolve == GLP_ON || parm->presolve == GLP_OFF)) - xerror("glp_intopt: presolve = %d; invalid parameter\n", - parm->presolve); - if (!(parm->binarize == GLP_ON || parm->binarize == GLP_OFF)) - xerror("glp_intopt: binarize = %d; invalid parameter\n", - parm->binarize); - if (!(parm->fp_heur == GLP_ON || parm->fp_heur == GLP_OFF)) - xerror("glp_intopt: fp_heur = %d; invalid parameter\n", - parm->fp_heur); -#if 1 /* 28/V-2010 */ - if (!(parm->alien == GLP_ON || parm->alien == GLP_OFF)) - xerror("glp_intopt: alien = %d; invalid parameter\n", - parm->alien); -#endif -#if 0 /* 11/VII-2013 */ - /* integer solution is currently undefined */ - P->mip_stat = GLP_UNDEF; - P->mip_obj = 0.0; -#else - if (!parm->use_sol) - P->mip_stat = GLP_UNDEF; - if (P->mip_stat == GLP_NOFEAS) - P->mip_stat = GLP_UNDEF; - if (P->mip_stat == GLP_UNDEF) - P->mip_obj = 0.0; - else if (P->mip_stat == GLP_OPT) - P->mip_stat = GLP_FEAS; -#endif - /* check bounds of double-bounded variables */ - for (i = 1; i <= P->m; i++) - { GLPROW *row = P->row[i]; - if (row->type == GLP_DB && row->lb >= row->ub) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: row %d: lb = %g, ub = %g; incorrect" - " bounds\n", i, row->lb, row->ub); - ret = GLP_EBOUND; - goto done; - } - } - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - if (col->type == GLP_DB && col->lb >= col->ub) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: column %d: lb = %g, ub = %g; incorr" - "ect bounds\n", j, col->lb, col->ub); - ret = GLP_EBOUND; - goto done; - } - } - /* bounds of all integer variables must be integral */ - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - if (col->kind != GLP_IV) continue; - if (col->type == GLP_LO || col->type == GLP_DB) - { if (col->lb != floor(col->lb)) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: integer column %d has non-intege" - "r lower bound %g\n", j, col->lb); - ret = GLP_EBOUND; - goto done; - } - } - if (col->type == GLP_UP || col->type == GLP_DB) - { if (col->ub != floor(col->ub)) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: integer column %d has non-intege" - "r upper bound %g\n", j, col->ub); - ret = GLP_EBOUND; - goto done; - } - } - if (col->type == GLP_FX) - { if (col->lb != floor(col->lb)) - { if (parm->msg_lev >= GLP_MSG_ERR) - xprintf("glp_intopt: integer column %d has non-intege" - "r fixed value %g\n", j, col->lb); - ret = GLP_EBOUND; - goto done; - } - } - } - /* solve MIP problem */ - if (parm->msg_lev >= GLP_MSG_ALL) - { int ni = glp_get_num_int(P); - int nb = glp_get_num_bin(P); - char s[50]; - xprintf("GLPK Integer Optimizer, v%s\n", glp_version()); - xprintf("%d row%s, %d column%s, %d non-zero%s\n", - P->m, P->m == 1 ? "" : "s", P->n, P->n == 1 ? "" : "s", - P->nnz, P->nnz == 1 ? "" : "s"); - if (nb == 0) - strcpy(s, "none of"); - else if (ni == 1 && nb == 1) - strcpy(s, ""); - else if (nb == 1) - strcpy(s, "one of"); - else if (nb == ni) - strcpy(s, "all of"); - else - sprintf(s, "%d of", nb); - xprintf("%d integer variable%s, %s which %s binary\n", - ni, ni == 1 ? "" : "s", s, nb == 1 ? "is" : "are"); - } -#if 1 /* 28/V-2010 */ - if (parm->alien) - { /* use alien integer optimizer */ - ret = _glp_intopt1(P, parm); - goto done; - } -#endif - if (!parm->presolve) -#if 0 /* 11/VII-2013 */ - ret = solve_mip(P, parm); -#else - ret = solve_mip(P, parm, P, NULL); -#endif - else - ret = preprocess_and_solve_mip(P, parm); -#if 1 /* 12/III-2013 */ - if (ret == GLP_ENOPFS) - P->mip_stat = GLP_NOFEAS; -#endif -done: /* return to the application program */ - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_init_iocp - initialize integer optimizer control parameters -* -* SYNOPSIS -* -* void glp_init_iocp(glp_iocp *parm); -* -* DESCRIPTION -* -* The routine glp_init_iocp initializes control parameters, which are -* used by the integer optimizer, with default values. -* -* Default values of the control parameters are stored in a glp_iocp -* structure, which the parameter parm points to. */ - -void glp_init_iocp(glp_iocp *parm) -{ parm->msg_lev = GLP_MSG_ALL; - parm->br_tech = GLP_BR_DTH; - parm->bt_tech = GLP_BT_BLB; - parm->tol_int = 1e-5; - parm->tol_obj = 1e-7; - parm->tm_lim = INT_MAX; - parm->out_frq = 5000; - parm->out_dly = 10000; - parm->cb_func = NULL; - parm->cb_info = NULL; - parm->cb_size = 0; - parm->pp_tech = GLP_PP_ALL; - parm->mip_gap = 0.0; - parm->mir_cuts = GLP_OFF; - parm->gmi_cuts = GLP_OFF; - parm->cov_cuts = GLP_OFF; - parm->clq_cuts = GLP_OFF; - parm->presolve = GLP_OFF; - parm->binarize = GLP_OFF; - parm->fp_heur = GLP_OFF; - parm->ps_heur = GLP_OFF; - parm->ps_tm_lim = 60000; /* 1 minute */ - parm->sr_heur = GLP_ON; -#if 1 /* 24/X-2015; not documented--should not be used */ - parm->use_sol = GLP_OFF; - parm->save_sol = NULL; - parm->alien = GLP_OFF; -#endif -#if 0 /* 20/I-2018 */ -#if 1 /* 16/III-2016; not documented--should not be used */ - parm->flip = GLP_OFF; -#endif -#else - parm->flip = GLP_ON; -#endif - return; -} - -/*********************************************************************** -* NAME -* -* glp_mip_status - retrieve status of MIP solution -* -* SYNOPSIS -* -* int glp_mip_status(glp_prob *mip); -* -* RETURNS -* -* The routine lpx_mip_status reports the status of MIP solution found -* by the branch-and-bound solver as follows: -* -* GLP_UNDEF - MIP solution is undefined; -* GLP_OPT - MIP solution is integer optimal; -* GLP_FEAS - MIP solution is integer feasible but its optimality -* (or non-optimality) has not been proven, perhaps due to -* premature termination of the search; -* GLP_NOFEAS - problem has no integer feasible solution (proven by the -* solver). */ - -int glp_mip_status(glp_prob *mip) -{ int mip_stat = mip->mip_stat; - return mip_stat; -} - -/*********************************************************************** -* NAME -* -* glp_mip_obj_val - retrieve objective value (MIP solution) -* -* SYNOPSIS -* -* double glp_mip_obj_val(glp_prob *mip); -* -* RETURNS -* -* The routine glp_mip_obj_val returns value of the objective function -* for MIP solution. */ - -double glp_mip_obj_val(glp_prob *mip) -{ /*struct LPXCPS *cps = mip->cps;*/ - double z; - z = mip->mip_obj; - /*if (cps->round && fabs(z) < 1e-9) z = 0.0;*/ - return z; -} - -/*********************************************************************** -* NAME -* -* glp_mip_row_val - retrieve row value (MIP solution) -* -* SYNOPSIS -* -* double glp_mip_row_val(glp_prob *mip, int i); -* -* RETURNS -* -* The routine glp_mip_row_val returns value of the auxiliary variable -* associated with i-th row. */ - -double glp_mip_row_val(glp_prob *mip, int i) -{ /*struct LPXCPS *cps = mip->cps;*/ - double mipx; - if (!(1 <= i && i <= mip->m)) - xerror("glp_mip_row_val: i = %d; row number out of range\n", i) - ; - mipx = mip->row[i]->mipx; - /*if (cps->round && fabs(mipx) < 1e-9) mipx = 0.0;*/ - return mipx; -} - -/*********************************************************************** -* NAME -* -* glp_mip_col_val - retrieve column value (MIP solution) -* -* SYNOPSIS -* -* double glp_mip_col_val(glp_prob *mip, int j); -* -* RETURNS -* -* The routine glp_mip_col_val returns value of the structural variable -* associated with j-th column. */ - -double glp_mip_col_val(glp_prob *mip, int j) -{ /*struct LPXCPS *cps = mip->cps;*/ - double mipx; - if (!(1 <= j && j <= mip->n)) - xerror("glp_mip_col_val: j = %d; column number out of range\n", - j); - mipx = mip->col[j]->mipx; - /*if (cps->round && fabs(mipx) < 1e-9) mipx = 0.0;*/ - return mipx; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi10.c b/code/3rd_glpk/draft/glpapi10.c deleted file mode 100644 index 5550aa39..00000000 --- a/code/3rd_glpk/draft/glpapi10.c +++ /dev/null @@ -1,305 +0,0 @@ -/* glpapi10.c (solution checking routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -void glp_check_kkt(glp_prob *P, int sol, int cond, double *_ae_max, - int *_ae_ind, double *_re_max, int *_re_ind) -{ /* check feasibility and optimality conditions */ - int m = P->m; - int n = P->n; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int i, j, ae_ind, re_ind; - double e, sp, sn, t, ae_max, re_max; - if (!(sol == GLP_SOL || sol == GLP_IPT || sol == GLP_MIP)) - xerror("glp_check_kkt: sol = %d; invalid solution indicator\n", - sol); - if (!(cond == GLP_KKT_PE || cond == GLP_KKT_PB || - cond == GLP_KKT_DE || cond == GLP_KKT_DB || - cond == GLP_KKT_CS)) - xerror("glp_check_kkt: cond = %d; invalid condition indicator " - "\n", cond); - ae_max = re_max = 0.0; - ae_ind = re_ind = 0; - if (cond == GLP_KKT_PE) - { /* xR - A * xS = 0 */ - for (i = 1; i <= m; i++) - { row = P->row[i]; - sp = sn = 0.0; - /* t := xR[i] */ - if (sol == GLP_SOL) - t = row->prim; - else if (sol == GLP_IPT) - t = row->pval; - else if (sol == GLP_MIP) - t = row->mipx; - else - xassert(sol != sol); - if (t >= 0.0) sp += t; else sn -= t; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col = aij->col; - /* t := - a[i,j] * xS[j] */ - if (sol == GLP_SOL) - t = - aij->val * col->prim; - else if (sol == GLP_IPT) - t = - aij->val * col->pval; - else if (sol == GLP_MIP) - t = - aij->val * col->mipx; - else - xassert(sol != sol); - if (t >= 0.0) sp += t; else sn -= t; - } - /* absolute error */ - e = fabs(sp - sn); - if (ae_max < e) - ae_max = e, ae_ind = i; - /* relative error */ - e /= (1.0 + sp + sn); - if (re_max < e) - re_max = e, re_ind = i; - } - } - else if (cond == GLP_KKT_PB) - { /* lR <= xR <= uR */ - for (i = 1; i <= m; i++) - { row = P->row[i]; - /* t := xR[i] */ - if (sol == GLP_SOL) - t = row->prim; - else if (sol == GLP_IPT) - t = row->pval; - else if (sol == GLP_MIP) - t = row->mipx; - else - xassert(sol != sol); - /* check lower bound */ - if (row->type == GLP_LO || row->type == GLP_DB || - row->type == GLP_FX) - { if (t < row->lb) - { /* absolute error */ - e = row->lb - t; - if (ae_max < e) - ae_max = e, ae_ind = i; - /* relative error */ - e /= (1.0 + fabs(row->lb)); - if (re_max < e) - re_max = e, re_ind = i; - } - } - /* check upper bound */ - if (row->type == GLP_UP || row->type == GLP_DB || - row->type == GLP_FX) - { if (t > row->ub) - { /* absolute error */ - e = t - row->ub; - if (ae_max < e) - ae_max = e, ae_ind = i; - /* relative error */ - e /= (1.0 + fabs(row->ub)); - if (re_max < e) - re_max = e, re_ind = i; - } - } - } - /* lS <= xS <= uS */ - for (j = 1; j <= n; j++) - { col = P->col[j]; - /* t := xS[j] */ - if (sol == GLP_SOL) - t = col->prim; - else if (sol == GLP_IPT) - t = col->pval; - else if (sol == GLP_MIP) - t = col->mipx; - else - xassert(sol != sol); - /* check lower bound */ - if (col->type == GLP_LO || col->type == GLP_DB || - col->type == GLP_FX) - { if (t < col->lb) - { /* absolute error */ - e = col->lb - t; - if (ae_max < e) - ae_max = e, ae_ind = m+j; - /* relative error */ - e /= (1.0 + fabs(col->lb)); - if (re_max < e) - re_max = e, re_ind = m+j; - } - } - /* check upper bound */ - if (col->type == GLP_UP || col->type == GLP_DB || - col->type == GLP_FX) - { if (t > col->ub) - { /* absolute error */ - e = t - col->ub; - if (ae_max < e) - ae_max = e, ae_ind = m+j; - /* relative error */ - e /= (1.0 + fabs(col->ub)); - if (re_max < e) - re_max = e, re_ind = m+j; - } - } - } - } - else if (cond == GLP_KKT_DE) - { /* A' * (lambdaR - cR) + (lambdaS - cS) = 0 */ - for (j = 1; j <= n; j++) - { col = P->col[j]; - sp = sn = 0.0; - /* t := lambdaS[j] - cS[j] */ - if (sol == GLP_SOL) - t = col->dual - col->coef; - else if (sol == GLP_IPT) - t = col->dval - col->coef; - else - xassert(sol != sol); - if (t >= 0.0) sp += t; else sn -= t; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - { row = aij->row; - /* t := a[i,j] * (lambdaR[i] - cR[i]) */ - if (sol == GLP_SOL) - t = aij->val * row->dual; - else if (sol == GLP_IPT) - t = aij->val * row->dval; - else - xassert(sol != sol); - if (t >= 0.0) sp += t; else sn -= t; - } - /* absolute error */ - e = fabs(sp - sn); - if (ae_max < e) - ae_max = e, ae_ind = m+j; - /* relative error */ - e /= (1.0 + sp + sn); - if (re_max < e) - re_max = e, re_ind = m+j; - } - } - else if (cond == GLP_KKT_DB) - { /* check lambdaR */ - for (i = 1; i <= m; i++) - { row = P->row[i]; - /* t := lambdaR[i] */ - if (sol == GLP_SOL) - t = row->dual; - else if (sol == GLP_IPT) - t = row->dval; - else - xassert(sol != sol); - /* correct sign */ - if (P->dir == GLP_MIN) - t = + t; - else if (P->dir == GLP_MAX) - t = - t; - else - xassert(P != P); - /* check for positivity */ -#if 1 /* 08/III-2013 */ - /* the former check was correct */ - /* the bug reported by David Price is related to violation - of complementarity slackness, not to this condition */ - if (row->type == GLP_FR || row->type == GLP_LO) -#else - if (row->stat == GLP_NF || row->stat == GLP_NL) -#endif - { if (t < 0.0) - { e = - t; - if (ae_max < e) - ae_max = re_max = e, ae_ind = re_ind = i; - } - } - /* check for negativity */ -#if 1 /* 08/III-2013 */ - /* see comment above */ - if (row->type == GLP_FR || row->type == GLP_UP) -#else - if (row->stat == GLP_NF || row->stat == GLP_NU) -#endif - { if (t > 0.0) - { e = + t; - if (ae_max < e) - ae_max = re_max = e, ae_ind = re_ind = i; - } - } - } - /* check lambdaS */ - for (j = 1; j <= n; j++) - { col = P->col[j]; - /* t := lambdaS[j] */ - if (sol == GLP_SOL) - t = col->dual; - else if (sol == GLP_IPT) - t = col->dval; - else - xassert(sol != sol); - /* correct sign */ - if (P->dir == GLP_MIN) - t = + t; - else if (P->dir == GLP_MAX) - t = - t; - else - xassert(P != P); - /* check for positivity */ -#if 1 /* 08/III-2013 */ - /* see comment above */ - if (col->type == GLP_FR || col->type == GLP_LO) -#else - if (col->stat == GLP_NF || col->stat == GLP_NL) -#endif - { if (t < 0.0) - { e = - t; - if (ae_max < e) - ae_max = re_max = e, ae_ind = re_ind = m+j; - } - } - /* check for negativity */ -#if 1 /* 08/III-2013 */ - /* see comment above */ - if (col->type == GLP_FR || col->type == GLP_UP) -#else - if (col->stat == GLP_NF || col->stat == GLP_NU) -#endif - { if (t > 0.0) - { e = + t; - if (ae_max < e) - ae_max = re_max = e, ae_ind = re_ind = m+j; - } - } - } - } - else - xassert(cond != cond); - if (_ae_max != NULL) *_ae_max = ae_max; - if (_ae_ind != NULL) *_ae_ind = ae_ind; - if (_re_max != NULL) *_re_max = re_max; - if (_re_ind != NULL) *_re_ind = re_ind; - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi12.c b/code/3rd_glpk/draft/glpapi12.c deleted file mode 100644 index 020c8981..00000000 --- a/code/3rd_glpk/draft/glpapi12.c +++ /dev/null @@ -1,2185 +0,0 @@ -/* glpapi12.c (basis factorization and simplex tableau routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "draft.h" -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_bf_exists - check if the basis factorization exists -* -* SYNOPSIS -* -* int glp_bf_exists(glp_prob *lp); -* -* RETURNS -* -* If the basis factorization for the current basis associated with -* the specified problem object exists and therefore is available for -* computations, the routine glp_bf_exists returns non-zero. Otherwise -* the routine returns zero. */ - -int glp_bf_exists(glp_prob *lp) -{ int ret; - ret = (lp->m == 0 || lp->valid); - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_factorize - compute the basis factorization -* -* SYNOPSIS -* -* int glp_factorize(glp_prob *lp); -* -* DESCRIPTION -* -* The routine glp_factorize computes the basis factorization for the -* current basis associated with the specified problem object. -* -* RETURNS -* -* 0 The basis factorization has been successfully computed. -* -* GLP_EBADB -* The basis matrix is invalid, i.e. the number of basic (auxiliary -* and structural) variables differs from the number of rows in the -* problem object. -* -* GLP_ESING -* The basis matrix is singular within the working precision. -* -* GLP_ECOND -* The basis matrix is ill-conditioned. */ - -static int b_col(void *info, int j, int ind[], double val[]) -{ glp_prob *lp = info; - int m = lp->m; - GLPAIJ *aij; - int k, len; - xassert(1 <= j && j <= m); - /* determine the ordinal number of basic auxiliary or structural - variable x[k] corresponding to basic variable xB[j] */ - k = lp->head[j]; - /* build j-th column of the basic matrix, which is k-th column of - the scaled augmented matrix (I | -R*A*S) */ - if (k <= m) - { /* x[k] is auxiliary variable */ - len = 1; - ind[1] = k; - val[1] = 1.0; - } - else - { /* x[k] is structural variable */ - len = 0; - for (aij = lp->col[k-m]->ptr; aij != NULL; aij = aij->c_next) - { len++; - ind[len] = aij->row->i; - val[len] = - aij->row->rii * aij->val * aij->col->sjj; - } - } - return len; -} - -int glp_factorize(glp_prob *lp) -{ int m = lp->m; - int n = lp->n; - GLPROW **row = lp->row; - GLPCOL **col = lp->col; - int *head = lp->head; - int j, k, stat, ret; - /* invalidate the basis factorization */ - lp->valid = 0; - /* build the basis header */ - j = 0; - for (k = 1; k <= m+n; k++) - { if (k <= m) - { stat = row[k]->stat; - row[k]->bind = 0; - } - else - { stat = col[k-m]->stat; - col[k-m]->bind = 0; - } - if (stat == GLP_BS) - { j++; - if (j > m) - { /* too many basic variables */ - ret = GLP_EBADB; - goto fini; - } - head[j] = k; - if (k <= m) - row[k]->bind = j; - else - col[k-m]->bind = j; - } - } - if (j < m) - { /* too few basic variables */ - ret = GLP_EBADB; - goto fini; - } - /* try to factorize the basis matrix */ - if (m > 0) - { if (lp->bfd == NULL) - { lp->bfd = bfd_create_it(); -#if 0 /* 08/III-2014 */ - copy_bfcp(lp); -#endif - } - switch (bfd_factorize(lp->bfd, m, /*lp->head,*/ b_col, lp)) - { case 0: - /* ok */ - break; - case BFD_ESING: - /* singular matrix */ - ret = GLP_ESING; - goto fini; - case BFD_ECOND: - /* ill-conditioned matrix */ - ret = GLP_ECOND; - goto fini; - default: - xassert(lp != lp); - } - lp->valid = 1; - } - /* factorization successful */ - ret = 0; -fini: /* bring the return code to the calling program */ - return ret; -} - -/*********************************************************************** -* NAME -* -* glp_bf_updated - check if the basis factorization has been updated -* -* SYNOPSIS -* -* int glp_bf_updated(glp_prob *lp); -* -* RETURNS -* -* If the basis factorization has been just computed from scratch, the -* routine glp_bf_updated returns zero. Otherwise, if the factorization -* has been updated one or more times, the routine returns non-zero. */ - -int glp_bf_updated(glp_prob *lp) -{ int cnt; - if (!(lp->m == 0 || lp->valid)) - xerror("glp_bf_update: basis factorization does not exist\n"); -#if 0 /* 15/XI-2009 */ - cnt = (lp->m == 0 ? 0 : lp->bfd->upd_cnt); -#else - cnt = (lp->m == 0 ? 0 : bfd_get_count(lp->bfd)); -#endif - return cnt; -} - -/*********************************************************************** -* NAME -* -* glp_get_bfcp - retrieve basis factorization control parameters -* -* SYNOPSIS -* -* void glp_get_bfcp(glp_prob *lp, glp_bfcp *parm); -* -* DESCRIPTION -* -* The routine glp_get_bfcp retrieves control parameters, which are -* used on computing and updating the basis factorization associated -* with the specified problem object. -* -* Current values of control parameters are stored by the routine in -* a glp_bfcp structure, which the parameter parm points to. */ - -#if 1 /* 08/III-2014 */ -void glp_get_bfcp(glp_prob *P, glp_bfcp *parm) -{ if (P->bfd == NULL) - P->bfd = bfd_create_it(); - bfd_get_bfcp(P->bfd, parm); - return; -} -#endif - -/*********************************************************************** -* NAME -* -* glp_set_bfcp - change basis factorization control parameters -* -* SYNOPSIS -* -* void glp_set_bfcp(glp_prob *lp, const glp_bfcp *parm); -* -* DESCRIPTION -* -* The routine glp_set_bfcp changes control parameters, which are used -* by internal GLPK routines in computing and updating the basis -* factorization associated with the specified problem object. -* -* New values of the control parameters should be passed in a structure -* glp_bfcp, which the parameter parm points to. -* -* The parameter parm can be specified as NULL, in which case all -* control parameters are reset to their default values. */ - -#if 1 /* 08/III-2014 */ -void glp_set_bfcp(glp_prob *P, const glp_bfcp *parm) -{ if (P->bfd == NULL) - P->bfd = bfd_create_it(); - if (parm != NULL) - { if (!(parm->type == GLP_BF_LUF + GLP_BF_FT || - parm->type == GLP_BF_LUF + GLP_BF_BG || - parm->type == GLP_BF_LUF + GLP_BF_GR || - parm->type == GLP_BF_BTF + GLP_BF_BG || - parm->type == GLP_BF_BTF + GLP_BF_GR)) - xerror("glp_set_bfcp: type = 0x%02X; invalid parameter\n", - parm->type); - if (!(0.0 < parm->piv_tol && parm->piv_tol < 1.0)) - xerror("glp_set_bfcp: piv_tol = %g; invalid parameter\n", - parm->piv_tol); - if (parm->piv_lim < 1) - xerror("glp_set_bfcp: piv_lim = %d; invalid parameter\n", - parm->piv_lim); - if (!(parm->suhl == GLP_ON || parm->suhl == GLP_OFF)) - xerror("glp_set_bfcp: suhl = %d; invalid parameter\n", - parm->suhl); - if (!(0.0 <= parm->eps_tol && parm->eps_tol <= 1e-6)) - xerror("glp_set_bfcp: eps_tol = %g; invalid parameter\n", - parm->eps_tol); - if (!(1 <= parm->nfs_max && parm->nfs_max <= 32767)) - xerror("glp_set_bfcp: nfs_max = %d; invalid parameter\n", - parm->nfs_max); - if (!(1 <= parm->nrs_max && parm->nrs_max <= 32767)) - xerror("glp_set_bfcp: nrs_max = %d; invalid parameter\n", - parm->nrs_max); - } - bfd_set_bfcp(P->bfd, parm); - return; -} -#endif - -/*********************************************************************** -* NAME -* -* glp_get_bhead - retrieve the basis header information -* -* SYNOPSIS -* -* int glp_get_bhead(glp_prob *lp, int k); -* -* DESCRIPTION -* -* The routine glp_get_bhead returns the basis header information for -* the current basis associated with the specified problem object. -* -* RETURNS -* -* If xB[k], 1 <= k <= m, is i-th auxiliary variable (1 <= i <= m), the -* routine returns i. Otherwise, if xB[k] is j-th structural variable -* (1 <= j <= n), the routine returns m+j. Here m is the number of rows -* and n is the number of columns in the problem object. */ - -int glp_get_bhead(glp_prob *lp, int k) -{ if (!(lp->m == 0 || lp->valid)) - xerror("glp_get_bhead: basis factorization does not exist\n"); - if (!(1 <= k && k <= lp->m)) - xerror("glp_get_bhead: k = %d; index out of range\n", k); - return lp->head[k]; -} - -/*********************************************************************** -* NAME -* -* glp_get_row_bind - retrieve row index in the basis header -* -* SYNOPSIS -* -* int glp_get_row_bind(glp_prob *lp, int i); -* -* RETURNS -* -* The routine glp_get_row_bind returns the index k of basic variable -* xB[k], 1 <= k <= m, which is i-th auxiliary variable, 1 <= i <= m, -* in the current basis associated with the specified problem object, -* where m is the number of rows. However, if i-th auxiliary variable -* is non-basic, the routine returns zero. */ - -int glp_get_row_bind(glp_prob *lp, int i) -{ if (!(lp->m == 0 || lp->valid)) - xerror("glp_get_row_bind: basis factorization does not exist\n" - ); - if (!(1 <= i && i <= lp->m)) - xerror("glp_get_row_bind: i = %d; row number out of range\n", - i); - return lp->row[i]->bind; -} - -/*********************************************************************** -* NAME -* -* glp_get_col_bind - retrieve column index in the basis header -* -* SYNOPSIS -* -* int glp_get_col_bind(glp_prob *lp, int j); -* -* RETURNS -* -* The routine glp_get_col_bind returns the index k of basic variable -* xB[k], 1 <= k <= m, which is j-th structural variable, 1 <= j <= n, -* in the current basis associated with the specified problem object, -* where m is the number of rows, n is the number of columns. However, -* if j-th structural variable is non-basic, the routine returns zero.*/ - -int glp_get_col_bind(glp_prob *lp, int j) -{ if (!(lp->m == 0 || lp->valid)) - xerror("glp_get_col_bind: basis factorization does not exist\n" - ); - if (!(1 <= j && j <= lp->n)) - xerror("glp_get_col_bind: j = %d; column number out of range\n" - , j); - return lp->col[j]->bind; -} - -/*********************************************************************** -* NAME -* -* glp_ftran - perform forward transformation (solve system B*x = b) -* -* SYNOPSIS -* -* void glp_ftran(glp_prob *lp, double x[]); -* -* DESCRIPTION -* -* The routine glp_ftran performs forward transformation, i.e. solves -* the system B*x = b, where B is the basis matrix corresponding to the -* current basis for the specified problem object, x is the vector of -* unknowns to be computed, b is the vector of right-hand sides. -* -* On entry elements of the vector b should be stored in dense format -* in locations x[1], ..., x[m], where m is the number of rows. On exit -* the routine stores elements of the vector x in the same locations. -* -* SCALING/UNSCALING -* -* Let A~ = (I | -A) is the augmented constraint matrix of the original -* (unscaled) problem. In the scaled LP problem instead the matrix A the -* scaled matrix A" = R*A*S is actually used, so -* -* A~" = (I | A") = (I | R*A*S) = (R*I*inv(R) | R*A*S) = -* (1) -* = R*(I | A)*S~ = R*A~*S~, -* -* is the scaled augmented constraint matrix, where R and S are diagonal -* scaling matrices used to scale rows and columns of the matrix A, and -* -* S~ = diag(inv(R) | S) (2) -* -* is an augmented diagonal scaling matrix. -* -* By definition: -* -* A~ = (B | N), (3) -* -* where B is the basic matrix, which consists of basic columns of the -* augmented constraint matrix A~, and N is a matrix, which consists of -* non-basic columns of A~. From (1) it follows that: -* -* A~" = (B" | N") = (R*B*SB | R*N*SN), (4) -* -* where SB and SN are parts of the augmented scaling matrix S~, which -* correspond to basic and non-basic variables, respectively. Therefore -* -* B" = R*B*SB, (5) -* -* which is the scaled basis matrix. */ - -void glp_ftran(glp_prob *lp, double x[]) -{ int m = lp->m; - GLPROW **row = lp->row; - GLPCOL **col = lp->col; - int i, k; - /* B*x = b ===> (R*B*SB)*(inv(SB)*x) = R*b ===> - B"*x" = b", where b" = R*b, x = SB*x" */ - if (!(m == 0 || lp->valid)) - xerror("glp_ftran: basis factorization does not exist\n"); - /* b" := R*b */ - for (i = 1; i <= m; i++) - x[i] *= row[i]->rii; - /* x" := inv(B")*b" */ - if (m > 0) bfd_ftran(lp->bfd, x); - /* x := SB*x" */ - for (i = 1; i <= m; i++) - { k = lp->head[i]; - if (k <= m) - x[i] /= row[k]->rii; - else - x[i] *= col[k-m]->sjj; - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_btran - perform backward transformation (solve system B'*x = b) -* -* SYNOPSIS -* -* void glp_btran(glp_prob *lp, double x[]); -* -* DESCRIPTION -* -* The routine glp_btran performs backward transformation, i.e. solves -* the system B'*x = b, where B' is a matrix transposed to the basis -* matrix corresponding to the current basis for the specified problem -* problem object, x is the vector of unknowns to be computed, b is the -* vector of right-hand sides. -* -* On entry elements of the vector b should be stored in dense format -* in locations x[1], ..., x[m], where m is the number of rows. On exit -* the routine stores elements of the vector x in the same locations. -* -* SCALING/UNSCALING -* -* See comments to the routine glp_ftran. */ - -void glp_btran(glp_prob *lp, double x[]) -{ int m = lp->m; - GLPROW **row = lp->row; - GLPCOL **col = lp->col; - int i, k; - /* B'*x = b ===> (SB*B'*R)*(inv(R)*x) = SB*b ===> - (B")'*x" = b", where b" = SB*b, x = R*x" */ - if (!(m == 0 || lp->valid)) - xerror("glp_btran: basis factorization does not exist\n"); - /* b" := SB*b */ - for (i = 1; i <= m; i++) - { k = lp->head[i]; - if (k <= m) - x[i] /= row[k]->rii; - else - x[i] *= col[k-m]->sjj; - } - /* x" := inv[(B")']*b" */ - if (m > 0) bfd_btran(lp->bfd, x); - /* x := R*x" */ - for (i = 1; i <= m; i++) - x[i] *= row[i]->rii; - return; -} - -/*********************************************************************** -* NAME -* -* glp_warm_up - "warm up" LP basis -* -* SYNOPSIS -* -* int glp_warm_up(glp_prob *P); -* -* DESCRIPTION -* -* The routine glp_warm_up "warms up" the LP basis for the specified -* problem object using current statuses assigned to rows and columns -* (that is, to auxiliary and structural variables). -* -* This operation includes computing factorization of the basis matrix -* (if it does not exist), computing primal and dual components of basic -* solution, and determining the solution status. -* -* RETURNS -* -* 0 The operation has been successfully performed. -* -* GLP_EBADB -* The basis matrix is invalid, i.e. the number of basic (auxiliary -* and structural) variables differs from the number of rows in the -* problem object. -* -* GLP_ESING -* The basis matrix is singular within the working precision. -* -* GLP_ECOND -* The basis matrix is ill-conditioned. */ - -int glp_warm_up(glp_prob *P) -{ GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int i, j, type, stat, ret; - double eps, temp, *work; - /* invalidate basic solution */ - P->pbs_stat = P->dbs_stat = GLP_UNDEF; - P->obj_val = 0.0; - P->some = 0; - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - row->prim = row->dual = 0.0; - } - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - col->prim = col->dual = 0.0; - } - /* compute the basis factorization, if necessary */ - if (!glp_bf_exists(P)) - { ret = glp_factorize(P); - if (ret != 0) goto done; - } - /* allocate working array */ - work = xcalloc(1+P->m, sizeof(double)); - /* determine and store values of non-basic variables, compute - vector (- N * xN) */ - for (i = 1; i <= P->m; i++) - work[i] = 0.0; - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->stat == GLP_BS) - continue; - else if (row->stat == GLP_NL) - row->prim = row->lb; - else if (row->stat == GLP_NU) - row->prim = row->ub; - else if (row->stat == GLP_NF) - row->prim = 0.0; - else if (row->stat == GLP_NS) - row->prim = row->lb; - else - xassert(row != row); - /* N[j] is i-th column of matrix (I|-A) */ - work[i] -= row->prim; - } - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->stat == GLP_BS) - continue; - else if (col->stat == GLP_NL) - col->prim = col->lb; - else if (col->stat == GLP_NU) - col->prim = col->ub; - else if (col->stat == GLP_NF) - col->prim = 0.0; - else if (col->stat == GLP_NS) - col->prim = col->lb; - else - xassert(col != col); - /* N[j] is (m+j)-th column of matrix (I|-A) */ - if (col->prim != 0.0) - { for (aij = col->ptr; aij != NULL; aij = aij->c_next) - work[aij->row->i] += aij->val * col->prim; - } - } - /* compute vector of basic variables xB = - inv(B) * N * xN */ - glp_ftran(P, work); - /* store values of basic variables, check primal feasibility */ - P->pbs_stat = GLP_FEAS; - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->stat != GLP_BS) - continue; - row->prim = work[row->bind]; - type = row->type; - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { eps = 1e-6 + 1e-9 * fabs(row->lb); - if (row->prim < row->lb - eps) - P->pbs_stat = GLP_INFEAS; - } - if (type == GLP_UP || type == GLP_DB || type == GLP_FX) - { eps = 1e-6 + 1e-9 * fabs(row->ub); - if (row->prim > row->ub + eps) - P->pbs_stat = GLP_INFEAS; - } - } - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->stat != GLP_BS) - continue; - col->prim = work[col->bind]; - type = col->type; - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { eps = 1e-6 + 1e-9 * fabs(col->lb); - if (col->prim < col->lb - eps) - P->pbs_stat = GLP_INFEAS; - } - if (type == GLP_UP || type == GLP_DB || type == GLP_FX) - { eps = 1e-6 + 1e-9 * fabs(col->ub); - if (col->prim > col->ub + eps) - P->pbs_stat = GLP_INFEAS; - } - } - /* compute value of the objective function */ - P->obj_val = P->c0; - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - P->obj_val += col->coef * col->prim; - } - /* build vector cB of objective coefficients at basic variables */ - for (i = 1; i <= P->m; i++) - work[i] = 0.0; - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->stat == GLP_BS) - work[col->bind] = col->coef; - } - /* compute vector of simplex multipliers pi = inv(B') * cB */ - glp_btran(P, work); - /* compute and store reduced costs of non-basic variables d[j] = - c[j] - N'[j] * pi, check dual feasibility */ - P->dbs_stat = GLP_FEAS; - for (i = 1; i <= P->m; i++) - { row = P->row[i]; - if (row->stat == GLP_BS) - { row->dual = 0.0; - continue; - } - /* N[j] is i-th column of matrix (I|-A) */ - row->dual = - work[i]; -#if 0 /* 07/III-2013 */ - type = row->type; - temp = (P->dir == GLP_MIN ? + row->dual : - row->dual); - if ((type == GLP_FR || type == GLP_LO) && temp < -1e-5 || - (type == GLP_FR || type == GLP_UP) && temp > +1e-5) - P->dbs_stat = GLP_INFEAS; -#else - stat = row->stat; - temp = (P->dir == GLP_MIN ? + row->dual : - row->dual); - if ((stat == GLP_NF || stat == GLP_NL) && temp < -1e-5 || - (stat == GLP_NF || stat == GLP_NU) && temp > +1e-5) - P->dbs_stat = GLP_INFEAS; -#endif - } - for (j = 1; j <= P->n; j++) - { col = P->col[j]; - if (col->stat == GLP_BS) - { col->dual = 0.0; - continue; - } - /* N[j] is (m+j)-th column of matrix (I|-A) */ - col->dual = col->coef; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - col->dual += aij->val * work[aij->row->i]; -#if 0 /* 07/III-2013 */ - type = col->type; - temp = (P->dir == GLP_MIN ? + col->dual : - col->dual); - if ((type == GLP_FR || type == GLP_LO) && temp < -1e-5 || - (type == GLP_FR || type == GLP_UP) && temp > +1e-5) - P->dbs_stat = GLP_INFEAS; -#else - stat = col->stat; - temp = (P->dir == GLP_MIN ? + col->dual : - col->dual); - if ((stat == GLP_NF || stat == GLP_NL) && temp < -1e-5 || - (stat == GLP_NF || stat == GLP_NU) && temp > +1e-5) - P->dbs_stat = GLP_INFEAS; -#endif - } - /* free working array */ - xfree(work); - ret = 0; -done: return ret; -} - -/*********************************************************************** -* NAME -* -* glp_eval_tab_row - compute row of the simplex tableau -* -* SYNOPSIS -* -* int glp_eval_tab_row(glp_prob *lp, int k, int ind[], double val[]); -* -* DESCRIPTION -* -* The routine glp_eval_tab_row computes a row of the current simplex -* tableau for the basic variable, which is specified by the number k: -* if 1 <= k <= m, x[k] is k-th auxiliary variable; if m+1 <= k <= m+n, -* x[k] is (k-m)-th structural variable, where m is number of rows, and -* n is number of columns. The current basis must be available. -* -* The routine stores column indices and numerical values of non-zero -* elements of the computed row using sparse format to the locations -* ind[1], ..., ind[len] and val[1], ..., val[len], respectively, where -* 0 <= len <= n is number of non-zeros returned on exit. -* -* Element indices stored in the array ind have the same sense as the -* index k, i.e. indices 1 to m denote auxiliary variables and indices -* m+1 to m+n denote structural ones (all these variables are obviously -* non-basic by definition). -* -* The computed row shows how the specified basic variable x[k] = xB[i] -* depends on non-basic variables: -* -* xB[i] = alfa[i,1]*xN[1] + alfa[i,2]*xN[2] + ... + alfa[i,n]*xN[n], -* -* where alfa[i,j] are elements of the simplex table row, xN[j] are -* non-basic (auxiliary and structural) variables. -* -* RETURNS -* -* The routine returns number of non-zero elements in the simplex table -* row stored in the arrays ind and val. -* -* BACKGROUND -* -* The system of equality constraints of the LP problem is: -* -* xR = A * xS, (1) -* -* where xR is the vector of auxliary variables, xS is the vector of -* structural variables, A is the matrix of constraint coefficients. -* -* The system (1) can be written in homogenous form as follows: -* -* A~ * x = 0, (2) -* -* where A~ = (I | -A) is the augmented constraint matrix (has m rows -* and m+n columns), x = (xR | xS) is the vector of all (auxiliary and -* structural) variables. -* -* By definition for the current basis we have: -* -* A~ = (B | N), (3) -* -* where B is the basis matrix. Thus, the system (2) can be written as: -* -* B * xB + N * xN = 0. (4) -* -* From (4) it follows that: -* -* xB = A^ * xN, (5) -* -* where the matrix -* -* A^ = - inv(B) * N (6) -* -* is called the simplex table. -* -* It is understood that i-th row of the simplex table is: -* -* e * A^ = - e * inv(B) * N, (7) -* -* where e is a unity vector with e[i] = 1. -* -* To compute i-th row of the simplex table the routine first computes -* i-th row of the inverse: -* -* rho = inv(B') * e, (8) -* -* where B' is a matrix transposed to B, and then computes elements of -* i-th row of the simplex table as scalar products: -* -* alfa[i,j] = - rho * N[j] for all j, (9) -* -* where N[j] is a column of the augmented constraint matrix A~, which -* corresponds to some non-basic auxiliary or structural variable. */ - -int glp_eval_tab_row(glp_prob *lp, int k, int ind[], double val[]) -{ int m = lp->m; - int n = lp->n; - int i, t, len, lll, *iii; - double alfa, *rho, *vvv; - if (!(m == 0 || lp->valid)) - xerror("glp_eval_tab_row: basis factorization does not exist\n" - ); - if (!(1 <= k && k <= m+n)) - xerror("glp_eval_tab_row: k = %d; variable number out of range" - , k); - /* determine xB[i] which corresponds to x[k] */ - if (k <= m) - i = glp_get_row_bind(lp, k); - else - i = glp_get_col_bind(lp, k-m); - if (i == 0) - xerror("glp_eval_tab_row: k = %d; variable must be basic", k); - xassert(1 <= i && i <= m); - /* allocate working arrays */ - rho = xcalloc(1+m, sizeof(double)); - iii = xcalloc(1+m, sizeof(int)); - vvv = xcalloc(1+m, sizeof(double)); - /* compute i-th row of the inverse; see (8) */ - for (t = 1; t <= m; t++) rho[t] = 0.0; - rho[i] = 1.0; - glp_btran(lp, rho); - /* compute i-th row of the simplex table */ - len = 0; - for (k = 1; k <= m+n; k++) - { if (k <= m) - { /* x[k] is auxiliary variable, so N[k] is a unity column */ - if (glp_get_row_stat(lp, k) == GLP_BS) continue; - /* compute alfa[i,j]; see (9) */ - alfa = - rho[k]; - } - else - { /* x[k] is structural variable, so N[k] is a column of the - original constraint matrix A with negative sign */ - if (glp_get_col_stat(lp, k-m) == GLP_BS) continue; - /* compute alfa[i,j]; see (9) */ - lll = glp_get_mat_col(lp, k-m, iii, vvv); - alfa = 0.0; - for (t = 1; t <= lll; t++) alfa += rho[iii[t]] * vvv[t]; - } - /* store alfa[i,j] */ - if (alfa != 0.0) len++, ind[len] = k, val[len] = alfa; - } - xassert(len <= n); - /* free working arrays */ - xfree(rho); - xfree(iii); - xfree(vvv); - /* return to the calling program */ - return len; -} - -/*********************************************************************** -* NAME -* -* glp_eval_tab_col - compute column of the simplex tableau -* -* SYNOPSIS -* -* int glp_eval_tab_col(glp_prob *lp, int k, int ind[], double val[]); -* -* DESCRIPTION -* -* The routine glp_eval_tab_col computes a column of the current simplex -* table for the non-basic variable, which is specified by the number k: -* if 1 <= k <= m, x[k] is k-th auxiliary variable; if m+1 <= k <= m+n, -* x[k] is (k-m)-th structural variable, where m is number of rows, and -* n is number of columns. The current basis must be available. -* -* The routine stores row indices and numerical values of non-zero -* elements of the computed column using sparse format to the locations -* ind[1], ..., ind[len] and val[1], ..., val[len] respectively, where -* 0 <= len <= m is number of non-zeros returned on exit. -* -* Element indices stored in the array ind have the same sense as the -* index k, i.e. indices 1 to m denote auxiliary variables and indices -* m+1 to m+n denote structural ones (all these variables are obviously -* basic by the definition). -* -* The computed column shows how basic variables depend on the specified -* non-basic variable x[k] = xN[j]: -* -* xB[1] = ... + alfa[1,j]*xN[j] + ... -* xB[2] = ... + alfa[2,j]*xN[j] + ... -* . . . . . . -* xB[m] = ... + alfa[m,j]*xN[j] + ... -* -* where alfa[i,j] are elements of the simplex table column, xB[i] are -* basic (auxiliary and structural) variables. -* -* RETURNS -* -* The routine returns number of non-zero elements in the simplex table -* column stored in the arrays ind and val. -* -* BACKGROUND -* -* As it was explained in comments to the routine glp_eval_tab_row (see -* above) the simplex table is the following matrix: -* -* A^ = - inv(B) * N. (1) -* -* Therefore j-th column of the simplex table is: -* -* A^ * e = - inv(B) * N * e = - inv(B) * N[j], (2) -* -* where e is a unity vector with e[j] = 1, B is the basis matrix, N[j] -* is a column of the augmented constraint matrix A~, which corresponds -* to the given non-basic auxiliary or structural variable. */ - -int glp_eval_tab_col(glp_prob *lp, int k, int ind[], double val[]) -{ int m = lp->m; - int n = lp->n; - int t, len, stat; - double *col; - if (!(m == 0 || lp->valid)) - xerror("glp_eval_tab_col: basis factorization does not exist\n" - ); - if (!(1 <= k && k <= m+n)) - xerror("glp_eval_tab_col: k = %d; variable number out of range" - , k); - if (k <= m) - stat = glp_get_row_stat(lp, k); - else - stat = glp_get_col_stat(lp, k-m); - if (stat == GLP_BS) - xerror("glp_eval_tab_col: k = %d; variable must be non-basic", - k); - /* obtain column N[k] with negative sign */ - col = xcalloc(1+m, sizeof(double)); - for (t = 1; t <= m; t++) col[t] = 0.0; - if (k <= m) - { /* x[k] is auxiliary variable, so N[k] is a unity column */ - col[k] = -1.0; - } - else - { /* x[k] is structural variable, so N[k] is a column of the - original constraint matrix A with negative sign */ - len = glp_get_mat_col(lp, k-m, ind, val); - for (t = 1; t <= len; t++) col[ind[t]] = val[t]; - } - /* compute column of the simplex table, which corresponds to the - specified non-basic variable x[k] */ - glp_ftran(lp, col); - len = 0; - for (t = 1; t <= m; t++) - { if (col[t] != 0.0) - { len++; - ind[len] = glp_get_bhead(lp, t); - val[len] = col[t]; - } - } - xfree(col); - /* return to the calling program */ - return len; -} - -/*********************************************************************** -* NAME -* -* glp_transform_row - transform explicitly specified row -* -* SYNOPSIS -* -* int glp_transform_row(glp_prob *P, int len, int ind[], double val[]); -* -* DESCRIPTION -* -* The routine glp_transform_row performs the same operation as the -* routine glp_eval_tab_row with exception that the row to be -* transformed is specified explicitly as a sparse vector. -* -* The explicitly specified row may be thought as a linear form: -* -* x = a[1]*x[m+1] + a[2]*x[m+2] + ... + a[n]*x[m+n], (1) -* -* where x is an auxiliary variable for this row, a[j] are coefficients -* of the linear form, x[m+j] are structural variables. -* -* On entry column indices and numerical values of non-zero elements of -* the row should be stored in locations ind[1], ..., ind[len] and -* val[1], ..., val[len], where len is the number of non-zero elements. -* -* This routine uses the system of equality constraints and the current -* basis in order to express the auxiliary variable x in (1) through the -* current non-basic variables (as if the transformed row were added to -* the problem object and its auxiliary variable were basic), i.e. the -* resultant row has the form: -* -* x = alfa[1]*xN[1] + alfa[2]*xN[2] + ... + alfa[n]*xN[n], (2) -* -* where xN[j] are non-basic (auxiliary or structural) variables, n is -* the number of columns in the LP problem object. -* -* On exit the routine stores indices and numerical values of non-zero -* elements of the resultant row (2) in locations ind[1], ..., ind[len'] -* and val[1], ..., val[len'], where 0 <= len' <= n is the number of -* non-zero elements in the resultant row returned by the routine. Note -* that indices (numbers) of non-basic variables stored in the array ind -* correspond to original ordinal numbers of variables: indices 1 to m -* mean auxiliary variables and indices m+1 to m+n mean structural ones. -* -* RETURNS -* -* The routine returns len', which is the number of non-zero elements in -* the resultant row stored in the arrays ind and val. -* -* BACKGROUND -* -* The explicitly specified row (1) is transformed in the same way as it -* were the objective function row. -* -* From (1) it follows that: -* -* x = aB * xB + aN * xN, (3) -* -* where xB is the vector of basic variables, xN is the vector of -* non-basic variables. -* -* The simplex table, which corresponds to the current basis, is: -* -* xB = [-inv(B) * N] * xN. (4) -* -* Therefore substituting xB from (4) to (3) we have: -* -* x = aB * [-inv(B) * N] * xN + aN * xN = -* (5) -* = rho * (-N) * xN + aN * xN = alfa * xN, -* -* where: -* -* rho = inv(B') * aB, (6) -* -* and -* -* alfa = aN + rho * (-N) (7) -* -* is the resultant row computed by the routine. */ - -int glp_transform_row(glp_prob *P, int len, int ind[], double val[]) -{ int i, j, k, m, n, t, lll, *iii; - double alfa, *a, *aB, *rho, *vvv; - if (!glp_bf_exists(P)) - xerror("glp_transform_row: basis factorization does not exist " - "\n"); - m = glp_get_num_rows(P); - n = glp_get_num_cols(P); - /* unpack the row to be transformed to the array a */ - a = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) a[j] = 0.0; - if (!(0 <= len && len <= n)) - xerror("glp_transform_row: len = %d; invalid row length\n", - len); - for (t = 1; t <= len; t++) - { j = ind[t]; - if (!(1 <= j && j <= n)) - xerror("glp_transform_row: ind[%d] = %d; column index out o" - "f range\n", t, j); - if (val[t] == 0.0) - xerror("glp_transform_row: val[%d] = 0; zero coefficient no" - "t allowed\n", t); - if (a[j] != 0.0) - xerror("glp_transform_row: ind[%d] = %d; duplicate column i" - "ndices not allowed\n", t, j); - a[j] = val[t]; - } - /* construct the vector aB */ - aB = xcalloc(1+m, sizeof(double)); - for (i = 1; i <= m; i++) - { k = glp_get_bhead(P, i); - /* xB[i] is k-th original variable */ - xassert(1 <= k && k <= m+n); - aB[i] = (k <= m ? 0.0 : a[k-m]); - } - /* solve the system B'*rho = aB to compute the vector rho */ - rho = aB, glp_btran(P, rho); - /* compute coefficients at non-basic auxiliary variables */ - len = 0; - for (i = 1; i <= m; i++) - { if (glp_get_row_stat(P, i) != GLP_BS) - { alfa = - rho[i]; - if (alfa != 0.0) - { len++; - ind[len] = i; - val[len] = alfa; - } - } - } - /* compute coefficients at non-basic structural variables */ - iii = xcalloc(1+m, sizeof(int)); - vvv = xcalloc(1+m, sizeof(double)); - for (j = 1; j <= n; j++) - { if (glp_get_col_stat(P, j) != GLP_BS) - { alfa = a[j]; - lll = glp_get_mat_col(P, j, iii, vvv); - for (t = 1; t <= lll; t++) alfa += vvv[t] * rho[iii[t]]; - if (alfa != 0.0) - { len++; - ind[len] = m+j; - val[len] = alfa; - } - } - } - xassert(len <= n); - xfree(iii); - xfree(vvv); - xfree(aB); - xfree(a); - return len; -} - -/*********************************************************************** -* NAME -* -* glp_transform_col - transform explicitly specified column -* -* SYNOPSIS -* -* int glp_transform_col(glp_prob *P, int len, int ind[], double val[]); -* -* DESCRIPTION -* -* The routine glp_transform_col performs the same operation as the -* routine glp_eval_tab_col with exception that the column to be -* transformed is specified explicitly as a sparse vector. -* -* The explicitly specified column may be thought as if it were added -* to the original system of equality constraints: -* -* x[1] = a[1,1]*x[m+1] + ... + a[1,n]*x[m+n] + a[1]*x -* x[2] = a[2,1]*x[m+1] + ... + a[2,n]*x[m+n] + a[2]*x (1) -* . . . . . . . . . . . . . . . -* x[m] = a[m,1]*x[m+1] + ... + a[m,n]*x[m+n] + a[m]*x -* -* where x[i] are auxiliary variables, x[m+j] are structural variables, -* x is a structural variable for the explicitly specified column, a[i] -* are constraint coefficients for x. -* -* On entry row indices and numerical values of non-zero elements of -* the column should be stored in locations ind[1], ..., ind[len] and -* val[1], ..., val[len], where len is the number of non-zero elements. -* -* This routine uses the system of equality constraints and the current -* basis in order to express the current basic variables through the -* structural variable x in (1) (as if the transformed column were added -* to the problem object and the variable x were non-basic), i.e. the -* resultant column has the form: -* -* xB[1] = ... + alfa[1]*x -* xB[2] = ... + alfa[2]*x (2) -* . . . . . . -* xB[m] = ... + alfa[m]*x -* -* where xB are basic (auxiliary and structural) variables, m is the -* number of rows in the problem object. -* -* On exit the routine stores indices and numerical values of non-zero -* elements of the resultant column (2) in locations ind[1], ..., -* ind[len'] and val[1], ..., val[len'], where 0 <= len' <= m is the -* number of non-zero element in the resultant column returned by the -* routine. Note that indices (numbers) of basic variables stored in -* the array ind correspond to original ordinal numbers of variables: -* indices 1 to m mean auxiliary variables and indices m+1 to m+n mean -* structural ones. -* -* RETURNS -* -* The routine returns len', which is the number of non-zero elements -* in the resultant column stored in the arrays ind and val. -* -* BACKGROUND -* -* The explicitly specified column (1) is transformed in the same way -* as any other column of the constraint matrix using the formula: -* -* alfa = inv(B) * a, (3) -* -* where alfa is the resultant column computed by the routine. */ - -int glp_transform_col(glp_prob *P, int len, int ind[], double val[]) -{ int i, m, t; - double *a, *alfa; - if (!glp_bf_exists(P)) - xerror("glp_transform_col: basis factorization does not exist " - "\n"); - m = glp_get_num_rows(P); - /* unpack the column to be transformed to the array a */ - a = xcalloc(1+m, sizeof(double)); - for (i = 1; i <= m; i++) a[i] = 0.0; - if (!(0 <= len && len <= m)) - xerror("glp_transform_col: len = %d; invalid column length\n", - len); - for (t = 1; t <= len; t++) - { i = ind[t]; - if (!(1 <= i && i <= m)) - xerror("glp_transform_col: ind[%d] = %d; row index out of r" - "ange\n", t, i); - if (val[t] == 0.0) - xerror("glp_transform_col: val[%d] = 0; zero coefficient no" - "t allowed\n", t); - if (a[i] != 0.0) - xerror("glp_transform_col: ind[%d] = %d; duplicate row indi" - "ces not allowed\n", t, i); - a[i] = val[t]; - } - /* solve the system B*a = alfa to compute the vector alfa */ - alfa = a, glp_ftran(P, alfa); - /* store resultant coefficients */ - len = 0; - for (i = 1; i <= m; i++) - { if (alfa[i] != 0.0) - { len++; - ind[len] = glp_get_bhead(P, i); - val[len] = alfa[i]; - } - } - xfree(a); - return len; -} - -/*********************************************************************** -* NAME -* -* glp_prim_rtest - perform primal ratio test -* -* SYNOPSIS -* -* int glp_prim_rtest(glp_prob *P, int len, const int ind[], -* const double val[], int dir, double eps); -* -* DESCRIPTION -* -* The routine glp_prim_rtest performs the primal ratio test using an -* explicitly specified column of the simplex table. -* -* The current basic solution associated with the LP problem object -* must be primal feasible. -* -* The explicitly specified column of the simplex table shows how the -* basic variables xB depend on some non-basic variable x (which is not -* necessarily presented in the problem object): -* -* xB[1] = ... + alfa[1] * x + ... -* xB[2] = ... + alfa[2] * x + ... (*) -* . . . . . . . . -* xB[m] = ... + alfa[m] * x + ... -* -* The column (*) is specifed on entry to the routine using the sparse -* format. Ordinal numbers of basic variables xB[i] should be placed in -* locations ind[1], ..., ind[len], where ordinal number 1 to m denote -* auxiliary variables, and ordinal numbers m+1 to m+n denote structural -* variables. The corresponding non-zero coefficients alfa[i] should be -* placed in locations val[1], ..., val[len]. The arrays ind and val are -* not changed on exit. -* -* The parameter dir specifies direction in which the variable x changes -* on entering the basis: +1 means increasing, -1 means decreasing. -* -* The parameter eps is an absolute tolerance (small positive number) -* used by the routine to skip small alfa[j] of the row (*). -* -* The routine determines which basic variable (among specified in -* ind[1], ..., ind[len]) should leave the basis in order to keep primal -* feasibility. -* -* RETURNS -* -* The routine glp_prim_rtest returns the index piv in the arrays ind -* and val corresponding to the pivot element chosen, 1 <= piv <= len. -* If the adjacent basic solution is primal unbounded and therefore the -* choice cannot be made, the routine returns zero. -* -* COMMENTS -* -* If the non-basic variable x is presented in the LP problem object, -* the column (*) can be computed with the routine glp_eval_tab_col; -* otherwise it can be computed with the routine glp_transform_col. */ - -int glp_prim_rtest(glp_prob *P, int len, const int ind[], - const double val[], int dir, double eps) -{ int k, m, n, piv, t, type, stat; - double alfa, big, beta, lb, ub, temp, teta; - if (glp_get_prim_stat(P) != GLP_FEAS) - xerror("glp_prim_rtest: basic solution is not primal feasible " - "\n"); - if (!(dir == +1 || dir == -1)) - xerror("glp_prim_rtest: dir = %d; invalid parameter\n", dir); - if (!(0.0 < eps && eps < 1.0)) - xerror("glp_prim_rtest: eps = %g; invalid parameter\n", eps); - m = glp_get_num_rows(P); - n = glp_get_num_cols(P); - /* initial settings */ - piv = 0, teta = DBL_MAX, big = 0.0; - /* walk through the entries of the specified column */ - for (t = 1; t <= len; t++) - { /* get the ordinal number of basic variable */ - k = ind[t]; - if (!(1 <= k && k <= m+n)) - xerror("glp_prim_rtest: ind[%d] = %d; variable number out o" - "f range\n", t, k); - /* determine type, bounds, status and primal value of basic - variable xB[i] = x[k] in the current basic solution */ - if (k <= m) - { type = glp_get_row_type(P, k); - lb = glp_get_row_lb(P, k); - ub = glp_get_row_ub(P, k); - stat = glp_get_row_stat(P, k); - beta = glp_get_row_prim(P, k); - } - else - { type = glp_get_col_type(P, k-m); - lb = glp_get_col_lb(P, k-m); - ub = glp_get_col_ub(P, k-m); - stat = glp_get_col_stat(P, k-m); - beta = glp_get_col_prim(P, k-m); - } - if (stat != GLP_BS) - xerror("glp_prim_rtest: ind[%d] = %d; non-basic variable no" - "t allowed\n", t, k); - /* determine influence coefficient at basic variable xB[i] - in the explicitly specified column and turn to the case of - increasing the variable x in order to simplify the program - logic */ - alfa = (dir > 0 ? + val[t] : - val[t]); - /* analyze main cases */ - if (type == GLP_FR) - { /* xB[i] is free variable */ - continue; - } - else if (type == GLP_LO) -lo: { /* xB[i] has an lower bound */ - if (alfa > - eps) continue; - temp = (lb - beta) / alfa; - } - else if (type == GLP_UP) -up: { /* xB[i] has an upper bound */ - if (alfa < + eps) continue; - temp = (ub - beta) / alfa; - } - else if (type == GLP_DB) - { /* xB[i] has both lower and upper bounds */ - if (alfa < 0.0) goto lo; else goto up; - } - else if (type == GLP_FX) - { /* xB[i] is fixed variable */ - if (- eps < alfa && alfa < + eps) continue; - temp = 0.0; - } - else - xassert(type != type); - /* if the value of the variable xB[i] violates its lower or - upper bound (slightly, because the current basis is assumed - to be primal feasible), temp is negative; we can think this - happens due to round-off errors and the value is exactly on - the bound; this allows replacing temp by zero */ - if (temp < 0.0) temp = 0.0; - /* apply the minimal ratio test */ - if (teta > temp || teta == temp && big < fabs(alfa)) - piv = t, teta = temp, big = fabs(alfa); - } - /* return index of the pivot element chosen */ - return piv; -} - -/*********************************************************************** -* NAME -* -* glp_dual_rtest - perform dual ratio test -* -* SYNOPSIS -* -* int glp_dual_rtest(glp_prob *P, int len, const int ind[], -* const double val[], int dir, double eps); -* -* DESCRIPTION -* -* The routine glp_dual_rtest performs the dual ratio test using an -* explicitly specified row of the simplex table. -* -* The current basic solution associated with the LP problem object -* must be dual feasible. -* -* The explicitly specified row of the simplex table is a linear form -* that shows how some basic variable x (which is not necessarily -* presented in the problem object) depends on non-basic variables xN: -* -* x = alfa[1] * xN[1] + alfa[2] * xN[2] + ... + alfa[n] * xN[n]. (*) -* -* The row (*) is specified on entry to the routine using the sparse -* format. Ordinal numbers of non-basic variables xN[j] should be placed -* in locations ind[1], ..., ind[len], where ordinal numbers 1 to m -* denote auxiliary variables, and ordinal numbers m+1 to m+n denote -* structural variables. The corresponding non-zero coefficients alfa[j] -* should be placed in locations val[1], ..., val[len]. The arrays ind -* and val are not changed on exit. -* -* The parameter dir specifies direction in which the variable x changes -* on leaving the basis: +1 means that x goes to its lower bound, and -1 -* means that x goes to its upper bound. -* -* The parameter eps is an absolute tolerance (small positive number) -* used by the routine to skip small alfa[j] of the row (*). -* -* The routine determines which non-basic variable (among specified in -* ind[1], ..., ind[len]) should enter the basis in order to keep dual -* feasibility. -* -* RETURNS -* -* The routine glp_dual_rtest returns the index piv in the arrays ind -* and val corresponding to the pivot element chosen, 1 <= piv <= len. -* If the adjacent basic solution is dual unbounded and therefore the -* choice cannot be made, the routine returns zero. -* -* COMMENTS -* -* If the basic variable x is presented in the LP problem object, the -* row (*) can be computed with the routine glp_eval_tab_row; otherwise -* it can be computed with the routine glp_transform_row. */ - -int glp_dual_rtest(glp_prob *P, int len, const int ind[], - const double val[], int dir, double eps) -{ int k, m, n, piv, t, stat; - double alfa, big, cost, obj, temp, teta; - if (glp_get_dual_stat(P) != GLP_FEAS) - xerror("glp_dual_rtest: basic solution is not dual feasible\n") - ; - if (!(dir == +1 || dir == -1)) - xerror("glp_dual_rtest: dir = %d; invalid parameter\n", dir); - if (!(0.0 < eps && eps < 1.0)) - xerror("glp_dual_rtest: eps = %g; invalid parameter\n", eps); - m = glp_get_num_rows(P); - n = glp_get_num_cols(P); - /* take into account optimization direction */ - obj = (glp_get_obj_dir(P) == GLP_MIN ? +1.0 : -1.0); - /* initial settings */ - piv = 0, teta = DBL_MAX, big = 0.0; - /* walk through the entries of the specified row */ - for (t = 1; t <= len; t++) - { /* get ordinal number of non-basic variable */ - k = ind[t]; - if (!(1 <= k && k <= m+n)) - xerror("glp_dual_rtest: ind[%d] = %d; variable number out o" - "f range\n", t, k); - /* determine status and reduced cost of non-basic variable - x[k] = xN[j] in the current basic solution */ - if (k <= m) - { stat = glp_get_row_stat(P, k); - cost = glp_get_row_dual(P, k); - } - else - { stat = glp_get_col_stat(P, k-m); - cost = glp_get_col_dual(P, k-m); - } - if (stat == GLP_BS) - xerror("glp_dual_rtest: ind[%d] = %d; basic variable not al" - "lowed\n", t, k); - /* determine influence coefficient at non-basic variable xN[j] - in the explicitly specified row and turn to the case of - increasing the variable x in order to simplify the program - logic */ - alfa = (dir > 0 ? + val[t] : - val[t]); - /* analyze main cases */ - if (stat == GLP_NL) - { /* xN[j] is on its lower bound */ - if (alfa < + eps) continue; - temp = (obj * cost) / alfa; - } - else if (stat == GLP_NU) - { /* xN[j] is on its upper bound */ - if (alfa > - eps) continue; - temp = (obj * cost) / alfa; - } - else if (stat == GLP_NF) - { /* xN[j] is non-basic free variable */ - if (- eps < alfa && alfa < + eps) continue; - temp = 0.0; - } - else if (stat == GLP_NS) - { /* xN[j] is non-basic fixed variable */ - continue; - } - else - xassert(stat != stat); - /* if the reduced cost of the variable xN[j] violates its zero - bound (slightly, because the current basis is assumed to be - dual feasible), temp is negative; we can think this happens - due to round-off errors and the reduced cost is exact zero; - this allows replacing temp by zero */ - if (temp < 0.0) temp = 0.0; - /* apply the minimal ratio test */ - if (teta > temp || teta == temp && big < fabs(alfa)) - piv = t, teta = temp, big = fabs(alfa); - } - /* return index of the pivot element chosen */ - return piv; -} - -/*********************************************************************** -* NAME -* -* glp_analyze_row - simulate one iteration of dual simplex method -* -* SYNOPSIS -* -* int glp_analyze_row(glp_prob *P, int len, const int ind[], -* const double val[], int type, double rhs, double eps, int *piv, -* double *x, double *dx, double *y, double *dy, double *dz); -* -* DESCRIPTION -* -* Let the current basis be optimal or dual feasible, and there be -* specified a row (constraint), which is violated by the current basic -* solution. The routine glp_analyze_row simulates one iteration of the -* dual simplex method to determine some information on the adjacent -* basis (see below), where the specified row becomes active constraint -* (i.e. its auxiliary variable becomes non-basic). -* -* The current basic solution associated with the problem object passed -* to the routine must be dual feasible, and its primal components must -* be defined. -* -* The row to be analyzed must be previously transformed either with -* the routine glp_eval_tab_row (if the row is in the problem object) -* or with the routine glp_transform_row (if the row is external, i.e. -* not in the problem object). This is needed to express the row only -* through (auxiliary and structural) variables, which are non-basic in -* the current basis: -* -* y = alfa[1] * xN[1] + alfa[2] * xN[2] + ... + alfa[n] * xN[n], -* -* where y is an auxiliary variable of the row, alfa[j] is an influence -* coefficient, xN[j] is a non-basic variable. -* -* The row is passed to the routine in sparse format. Ordinal numbers -* of non-basic variables are stored in locations ind[1], ..., ind[len], -* where numbers 1 to m denote auxiliary variables while numbers m+1 to -* m+n denote structural variables. Corresponding non-zero coefficients -* alfa[j] are stored in locations val[1], ..., val[len]. The arrays -* ind and val are ot changed on exit. -* -* The parameters type and rhs specify the row type and its right-hand -* side as follows: -* -* type = GLP_LO: y = sum alfa[j] * xN[j] >= rhs -* -* type = GLP_UP: y = sum alfa[j] * xN[j] <= rhs -* -* The parameter eps is an absolute tolerance (small positive number) -* used by the routine to skip small coefficients alfa[j] on performing -* the dual ratio test. -* -* If the operation was successful, the routine stores the following -* information to corresponding location (if some parameter is NULL, -* its value is not stored): -* -* piv index in the array ind and val, 1 <= piv <= len, determining -* the non-basic variable, which would enter the adjacent basis; -* -* x value of the non-basic variable in the current basis; -* -* dx difference between values of the non-basic variable in the -* adjacent and current bases, dx = x.new - x.old; -* -* y value of the row (i.e. of its auxiliary variable) in the -* current basis; -* -* dy difference between values of the row in the adjacent and -* current bases, dy = y.new - y.old; -* -* dz difference between values of the objective function in the -* adjacent and current bases, dz = z.new - z.old. Note that in -* case of minimization dz >= 0, and in case of maximization -* dz <= 0, i.e. in the adjacent basis the objective function -* always gets worse (degrades). */ - -int _glp_analyze_row(glp_prob *P, int len, const int ind[], - const double val[], int type, double rhs, double eps, int *_piv, - double *_x, double *_dx, double *_y, double *_dy, double *_dz) -{ int t, k, dir, piv, ret = 0; - double x, dx, y, dy, dz; - if (P->pbs_stat == GLP_UNDEF) - xerror("glp_analyze_row: primal basic solution components are " - "undefined\n"); - if (P->dbs_stat != GLP_FEAS) - xerror("glp_analyze_row: basic solution is not dual feasible\n" - ); - /* compute the row value y = sum alfa[j] * xN[j] in the current - basis */ - if (!(0 <= len && len <= P->n)) - xerror("glp_analyze_row: len = %d; invalid row length\n", len); - y = 0.0; - for (t = 1; t <= len; t++) - { /* determine value of x[k] = xN[j] in the current basis */ - k = ind[t]; - if (!(1 <= k && k <= P->m+P->n)) - xerror("glp_analyze_row: ind[%d] = %d; row/column index out" - " of range\n", t, k); - if (k <= P->m) - { /* x[k] is auxiliary variable */ - if (P->row[k]->stat == GLP_BS) - xerror("glp_analyze_row: ind[%d] = %d; basic auxiliary v" - "ariable is not allowed\n", t, k); - x = P->row[k]->prim; - } - else - { /* x[k] is structural variable */ - if (P->col[k-P->m]->stat == GLP_BS) - xerror("glp_analyze_row: ind[%d] = %d; basic structural " - "variable is not allowed\n", t, k); - x = P->col[k-P->m]->prim; - } - y += val[t] * x; - } - /* check if the row is primal infeasible in the current basis, - i.e. the constraint is violated at the current point */ - if (type == GLP_LO) - { if (y >= rhs) - { /* the constraint is not violated */ - ret = 1; - goto done; - } - /* in the adjacent basis y goes to its lower bound */ - dir = +1; - } - else if (type == GLP_UP) - { if (y <= rhs) - { /* the constraint is not violated */ - ret = 1; - goto done; - } - /* in the adjacent basis y goes to its upper bound */ - dir = -1; - } - else - xerror("glp_analyze_row: type = %d; invalid parameter\n", - type); - /* compute dy = y.new - y.old */ - dy = rhs - y; - /* perform dual ratio test to determine which non-basic variable - should enter the adjacent basis to keep it dual feasible */ - piv = glp_dual_rtest(P, len, ind, val, dir, eps); - if (piv == 0) - { /* no dual feasible adjacent basis exists */ - ret = 2; - goto done; - } - /* non-basic variable x[k] = xN[j] should enter the basis */ - k = ind[piv]; - xassert(1 <= k && k <= P->m+P->n); - /* determine its value in the current basis */ - if (k <= P->m) - x = P->row[k]->prim; - else - x = P->col[k-P->m]->prim; - /* compute dx = x.new - x.old = dy / alfa[j] */ - xassert(val[piv] != 0.0); - dx = dy / val[piv]; - /* compute dz = z.new - z.old = d[j] * dx, where d[j] is reduced - cost of xN[j] in the current basis */ - if (k <= P->m) - dz = P->row[k]->dual * dx; - else - dz = P->col[k-P->m]->dual * dx; - /* store the analysis results */ - if (_piv != NULL) *_piv = piv; - if (_x != NULL) *_x = x; - if (_dx != NULL) *_dx = dx; - if (_y != NULL) *_y = y; - if (_dy != NULL) *_dy = dy; - if (_dz != NULL) *_dz = dz; -done: return ret; -} - -#if 0 -int main(void) -{ /* example program for the routine glp_analyze_row */ - glp_prob *P; - glp_smcp parm; - int i, k, len, piv, ret, ind[1+100]; - double rhs, x, dx, y, dy, dz, val[1+100]; - P = glp_create_prob(); - /* read plan.mps (see glpk/examples) */ - ret = glp_read_mps(P, GLP_MPS_DECK, NULL, "plan.mps"); - glp_assert(ret == 0); - /* and solve it to optimality */ - ret = glp_simplex(P, NULL); - glp_assert(ret == 0); - glp_assert(glp_get_status(P) == GLP_OPT); - /* the optimal objective value is 296.217 */ - /* we would like to know what happens if we would add a new row - (constraint) to plan.mps: - .01 * bin1 + .01 * bin2 + .02 * bin4 + .02 * bin5 <= 12 */ - /* first, we specify this new row */ - glp_create_index(P); - len = 0; - ind[++len] = glp_find_col(P, "BIN1"), val[len] = .01; - ind[++len] = glp_find_col(P, "BIN2"), val[len] = .01; - ind[++len] = glp_find_col(P, "BIN4"), val[len] = .02; - ind[++len] = glp_find_col(P, "BIN5"), val[len] = .02; - rhs = 12; - /* then we can compute value of the row (i.e. of its auxiliary - variable) in the current basis to see if the constraint is - violated */ - y = 0.0; - for (k = 1; k <= len; k++) - y += val[k] * glp_get_col_prim(P, ind[k]); - glp_printf("y = %g\n", y); - /* this prints y = 15.1372, so the constraint is violated, since - we require that y <= rhs = 12 */ - /* now we transform the row to express it only through non-basic - (auxiliary and artificial) variables */ - len = glp_transform_row(P, len, ind, val); - /* finally, we simulate one step of the dual simplex method to - obtain necessary information for the adjacent basis */ - ret = _glp_analyze_row(P, len, ind, val, GLP_UP, rhs, 1e-9, &piv, - &x, &dx, &y, &dy, &dz); - glp_assert(ret == 0); - glp_printf("k = %d, x = %g; dx = %g; y = %g; dy = %g; dz = %g\n", - ind[piv], x, dx, y, dy, dz); - /* this prints dz = 5.64418 and means that in the adjacent basis - the objective function would be 296.217 + 5.64418 = 301.861 */ - /* now we actually include the row into the problem object; note - that the arrays ind and val are clobbered, so we need to build - them once again */ - len = 0; - ind[++len] = glp_find_col(P, "BIN1"), val[len] = .01; - ind[++len] = glp_find_col(P, "BIN2"), val[len] = .01; - ind[++len] = glp_find_col(P, "BIN4"), val[len] = .02; - ind[++len] = glp_find_col(P, "BIN5"), val[len] = .02; - rhs = 12; - i = glp_add_rows(P, 1); - glp_set_row_bnds(P, i, GLP_UP, 0, rhs); - glp_set_mat_row(P, i, len, ind, val); - /* and perform one dual simplex iteration */ - glp_init_smcp(&parm); - parm.meth = GLP_DUAL; - parm.it_lim = 1; - glp_simplex(P, &parm); - /* the current objective value is 301.861 */ - return 0; -} -#endif - -/*********************************************************************** -* NAME -* -* glp_analyze_bound - analyze active bound of non-basic variable -* -* SYNOPSIS -* -* void glp_analyze_bound(glp_prob *P, int k, double *limit1, int *var1, -* double *limit2, int *var2); -* -* DESCRIPTION -* -* The routine glp_analyze_bound analyzes the effect of varying the -* active bound of specified non-basic variable. -* -* The non-basic variable is specified by the parameter k, where -* 1 <= k <= m means auxiliary variable of corresponding row while -* m+1 <= k <= m+n means structural variable (column). -* -* Note that the current basic solution must be optimal, and the basis -* factorization must exist. -* -* Results of the analysis have the following meaning. -* -* value1 is the minimal value of the active bound, at which the basis -* still remains primal feasible and thus optimal. -DBL_MAX means that -* the active bound has no lower limit. -* -* var1 is the ordinal number of an auxiliary (1 to m) or structural -* (m+1 to n) basic variable, which reaches its bound first and thereby -* limits further decreasing the active bound being analyzed. -* if value1 = -DBL_MAX, var1 is set to 0. -* -* value2 is the maximal value of the active bound, at which the basis -* still remains primal feasible and thus optimal. +DBL_MAX means that -* the active bound has no upper limit. -* -* var2 is the ordinal number of an auxiliary (1 to m) or structural -* (m+1 to n) basic variable, which reaches its bound first and thereby -* limits further increasing the active bound being analyzed. -* if value2 = +DBL_MAX, var2 is set to 0. */ - -void glp_analyze_bound(glp_prob *P, int k, double *value1, int *var1, - double *value2, int *var2) -{ GLPROW *row; - GLPCOL *col; - int m, n, stat, kase, p, len, piv, *ind; - double x, new_x, ll, uu, xx, delta, *val; -#if 0 /* 04/IV-2016 */ - /* sanity checks */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_analyze_bound: P = %p; invalid problem object\n", - P); -#endif - m = P->m, n = P->n; - if (!(P->pbs_stat == GLP_FEAS && P->dbs_stat == GLP_FEAS)) - xerror("glp_analyze_bound: optimal basic solution required\n"); - if (!(m == 0 || P->valid)) - xerror("glp_analyze_bound: basis factorization required\n"); - if (!(1 <= k && k <= m+n)) - xerror("glp_analyze_bound: k = %d; variable number out of rang" - "e\n", k); - /* retrieve information about the specified non-basic variable - x[k] whose active bound is to be analyzed */ - if (k <= m) - { row = P->row[k]; - stat = row->stat; - x = row->prim; - } - else - { col = P->col[k-m]; - stat = col->stat; - x = col->prim; - } - if (stat == GLP_BS) - xerror("glp_analyze_bound: k = %d; basic variable not allowed " - "\n", k); - /* allocate working arrays */ - ind = xcalloc(1+m, sizeof(int)); - val = xcalloc(1+m, sizeof(double)); - /* compute column of the simplex table corresponding to the - non-basic variable x[k] */ - len = glp_eval_tab_col(P, k, ind, val); - xassert(0 <= len && len <= m); - /* perform analysis */ - for (kase = -1; kase <= +1; kase += 2) - { /* kase < 0 means active bound of x[k] is decreasing; - kase > 0 means active bound of x[k] is increasing */ - /* use the primal ratio test to determine some basic variable - x[p] which reaches its bound first */ - piv = glp_prim_rtest(P, len, ind, val, kase, 1e-9); - if (piv == 0) - { /* nothing limits changing the active bound of x[k] */ - p = 0; - new_x = (kase < 0 ? -DBL_MAX : +DBL_MAX); - goto store; - } - /* basic variable x[p] limits changing the active bound of - x[k]; determine its value in the current basis */ - xassert(1 <= piv && piv <= len); - p = ind[piv]; - if (p <= m) - { row = P->row[p]; - ll = glp_get_row_lb(P, row->i); - uu = glp_get_row_ub(P, row->i); - stat = row->stat; - xx = row->prim; - } - else - { col = P->col[p-m]; - ll = glp_get_col_lb(P, col->j); - uu = glp_get_col_ub(P, col->j); - stat = col->stat; - xx = col->prim; - } - xassert(stat == GLP_BS); - /* determine delta x[p] = bound of x[p] - value of x[p] */ - if (kase < 0 && val[piv] > 0.0 || - kase > 0 && val[piv] < 0.0) - { /* delta x[p] < 0, so x[p] goes toward its lower bound */ - xassert(ll != -DBL_MAX); - delta = ll - xx; - } - else - { /* delta x[p] > 0, so x[p] goes toward its upper bound */ - xassert(uu != +DBL_MAX); - delta = uu - xx; - } - /* delta x[p] = alfa[p,k] * delta x[k], so new x[k] = x[k] + - delta x[k] = x[k] + delta x[p] / alfa[p,k] is the value of - x[k] in the adjacent basis */ - xassert(val[piv] != 0.0); - new_x = x + delta / val[piv]; -store: /* store analysis results */ - if (kase < 0) - { if (value1 != NULL) *value1 = new_x; - if (var1 != NULL) *var1 = p; - } - else - { if (value2 != NULL) *value2 = new_x; - if (var2 != NULL) *var2 = p; - } - } - /* free working arrays */ - xfree(ind); - xfree(val); - return; -} - -/*********************************************************************** -* NAME -* -* glp_analyze_coef - analyze objective coefficient at basic variable -* -* SYNOPSIS -* -* void glp_analyze_coef(glp_prob *P, int k, double *coef1, int *var1, -* double *value1, double *coef2, int *var2, double *value2); -* -* DESCRIPTION -* -* The routine glp_analyze_coef analyzes the effect of varying the -* objective coefficient at specified basic variable. -* -* The basic variable is specified by the parameter k, where -* 1 <= k <= m means auxiliary variable of corresponding row while -* m+1 <= k <= m+n means structural variable (column). -* -* Note that the current basic solution must be optimal, and the basis -* factorization must exist. -* -* Results of the analysis have the following meaning. -* -* coef1 is the minimal value of the objective coefficient, at which -* the basis still remains dual feasible and thus optimal. -DBL_MAX -* means that the objective coefficient has no lower limit. -* -* var1 is the ordinal number of an auxiliary (1 to m) or structural -* (m+1 to n) non-basic variable, whose reduced cost reaches its zero -* bound first and thereby limits further decreasing the objective -* coefficient being analyzed. If coef1 = -DBL_MAX, var1 is set to 0. -* -* value1 is value of the basic variable being analyzed in an adjacent -* basis, which is defined as follows. Let the objective coefficient -* reaches its minimal value (coef1) and continues decreasing. Then the -* reduced cost of the limiting non-basic variable (var1) becomes dual -* infeasible and the current basis becomes non-optimal that forces the -* limiting non-basic variable to enter the basis replacing there some -* basic variable that leaves the basis to keep primal feasibility. -* Should note that on determining the adjacent basis current bounds -* of the basic variable being analyzed are ignored as if it were free -* (unbounded) variable, so it cannot leave the basis. It may happen -* that no dual feasible adjacent basis exists, in which case value1 is -* set to -DBL_MAX or +DBL_MAX. -* -* coef2 is the maximal value of the objective coefficient, at which -* the basis still remains dual feasible and thus optimal. +DBL_MAX -* means that the objective coefficient has no upper limit. -* -* var2 is the ordinal number of an auxiliary (1 to m) or structural -* (m+1 to n) non-basic variable, whose reduced cost reaches its zero -* bound first and thereby limits further increasing the objective -* coefficient being analyzed. If coef2 = +DBL_MAX, var2 is set to 0. -* -* value2 is value of the basic variable being analyzed in an adjacent -* basis, which is defined exactly in the same way as value1 above with -* exception that now the objective coefficient is increasing. */ - -void glp_analyze_coef(glp_prob *P, int k, double *coef1, int *var1, - double *value1, double *coef2, int *var2, double *value2) -{ GLPROW *row; GLPCOL *col; - int m, n, type, stat, kase, p, q, dir, clen, cpiv, rlen, rpiv, - *cind, *rind; - double lb, ub, coef, x, lim_coef, new_x, d, delta, ll, uu, xx, - *rval, *cval; -#if 0 /* 04/IV-2016 */ - /* sanity checks */ - if (P == NULL || P->magic != GLP_PROB_MAGIC) - xerror("glp_analyze_coef: P = %p; invalid problem object\n", - P); -#endif - m = P->m, n = P->n; - if (!(P->pbs_stat == GLP_FEAS && P->dbs_stat == GLP_FEAS)) - xerror("glp_analyze_coef: optimal basic solution required\n"); - if (!(m == 0 || P->valid)) - xerror("glp_analyze_coef: basis factorization required\n"); - if (!(1 <= k && k <= m+n)) - xerror("glp_analyze_coef: k = %d; variable number out of range" - "\n", k); - /* retrieve information about the specified basic variable x[k] - whose objective coefficient c[k] is to be analyzed */ - if (k <= m) - { row = P->row[k]; - type = row->type; - lb = row->lb; - ub = row->ub; - coef = 0.0; - stat = row->stat; - x = row->prim; - } - else - { col = P->col[k-m]; - type = col->type; - lb = col->lb; - ub = col->ub; - coef = col->coef; - stat = col->stat; - x = col->prim; - } - if (stat != GLP_BS) - xerror("glp_analyze_coef: k = %d; non-basic variable not allow" - "ed\n", k); - /* allocate working arrays */ - cind = xcalloc(1+m, sizeof(int)); - cval = xcalloc(1+m, sizeof(double)); - rind = xcalloc(1+n, sizeof(int)); - rval = xcalloc(1+n, sizeof(double)); - /* compute row of the simplex table corresponding to the basic - variable x[k] */ - rlen = glp_eval_tab_row(P, k, rind, rval); - xassert(0 <= rlen && rlen <= n); - /* perform analysis */ - for (kase = -1; kase <= +1; kase += 2) - { /* kase < 0 means objective coefficient c[k] is decreasing; - kase > 0 means objective coefficient c[k] is increasing */ - /* note that decreasing c[k] is equivalent to increasing dual - variable lambda[k] and vice versa; we need to correctly set - the dir flag as required by the routine glp_dual_rtest */ - if (P->dir == GLP_MIN) - dir = - kase; - else if (P->dir == GLP_MAX) - dir = + kase; - else - xassert(P != P); - /* use the dual ratio test to determine non-basic variable - x[q] whose reduced cost d[q] reaches zero bound first */ - rpiv = glp_dual_rtest(P, rlen, rind, rval, dir, 1e-9); - if (rpiv == 0) - { /* nothing limits changing c[k] */ - lim_coef = (kase < 0 ? -DBL_MAX : +DBL_MAX); - q = 0; - /* x[k] keeps its current value */ - new_x = x; - goto store; - } - /* non-basic variable x[q] limits changing coefficient c[k]; - determine its status and reduced cost d[k] in the current - basis */ - xassert(1 <= rpiv && rpiv <= rlen); - q = rind[rpiv]; - xassert(1 <= q && q <= m+n); - if (q <= m) - { row = P->row[q]; - stat = row->stat; - d = row->dual; - } - else - { col = P->col[q-m]; - stat = col->stat; - d = col->dual; - } - /* note that delta d[q] = new d[q] - d[q] = - d[q], because - new d[q] = 0; delta d[q] = alfa[k,q] * delta c[k], so - delta c[k] = delta d[q] / alfa[k,q] = - d[q] / alfa[k,q] */ - xassert(rval[rpiv] != 0.0); - delta = - d / rval[rpiv]; - /* compute new c[k] = c[k] + delta c[k], which is the limiting - value of the objective coefficient c[k] */ - lim_coef = coef + delta; - /* let c[k] continue decreasing/increasing that makes d[q] - dual infeasible and forces x[q] to enter the basis; - to perform the primal ratio test we need to know in which - direction x[q] changes on entering the basis; we determine - that analyzing the sign of delta d[q] (see above), since - d[q] may be close to zero having wrong sign */ - /* let, for simplicity, the problem is minimization */ - if (kase < 0 && rval[rpiv] > 0.0 || - kase > 0 && rval[rpiv] < 0.0) - { /* delta d[q] < 0, so d[q] being non-negative will become - negative, so x[q] will increase */ - dir = +1; - } - else - { /* delta d[q] > 0, so d[q] being non-positive will become - positive, so x[q] will decrease */ - dir = -1; - } - /* if the problem is maximization, correct the direction */ - if (P->dir == GLP_MAX) dir = - dir; - /* check that we didn't make a silly mistake */ - if (dir > 0) - xassert(stat == GLP_NL || stat == GLP_NF); - else - xassert(stat == GLP_NU || stat == GLP_NF); - /* compute column of the simplex table corresponding to the - non-basic variable x[q] */ - clen = glp_eval_tab_col(P, q, cind, cval); - /* make x[k] temporarily free (unbounded) */ - if (k <= m) - { row = P->row[k]; - row->type = GLP_FR; - row->lb = row->ub = 0.0; - } - else - { col = P->col[k-m]; - col->type = GLP_FR; - col->lb = col->ub = 0.0; - } - /* use the primal ratio test to determine some basic variable - which leaves the basis */ - cpiv = glp_prim_rtest(P, clen, cind, cval, dir, 1e-9); - /* restore original bounds of the basic variable x[k] */ - if (k <= m) - { row = P->row[k]; - row->type = type; - row->lb = lb, row->ub = ub; - } - else - { col = P->col[k-m]; - col->type = type; - col->lb = lb, col->ub = ub; - } - if (cpiv == 0) - { /* non-basic variable x[q] can change unlimitedly */ - if (dir < 0 && rval[rpiv] > 0.0 || - dir > 0 && rval[rpiv] < 0.0) - { /* delta x[k] = alfa[k,q] * delta x[q] < 0 */ - new_x = -DBL_MAX; - } - else - { /* delta x[k] = alfa[k,q] * delta x[q] > 0 */ - new_x = +DBL_MAX; - } - goto store; - } - /* some basic variable x[p] limits changing non-basic variable - x[q] in the adjacent basis */ - xassert(1 <= cpiv && cpiv <= clen); - p = cind[cpiv]; - xassert(1 <= p && p <= m+n); - xassert(p != k); - if (p <= m) - { row = P->row[p]; - xassert(row->stat == GLP_BS); - ll = glp_get_row_lb(P, row->i); - uu = glp_get_row_ub(P, row->i); - xx = row->prim; - } - else - { col = P->col[p-m]; - xassert(col->stat == GLP_BS); - ll = glp_get_col_lb(P, col->j); - uu = glp_get_col_ub(P, col->j); - xx = col->prim; - } - /* determine delta x[p] = new x[p] - x[p] */ - if (dir < 0 && cval[cpiv] > 0.0 || - dir > 0 && cval[cpiv] < 0.0) - { /* delta x[p] < 0, so x[p] goes toward its lower bound */ - xassert(ll != -DBL_MAX); - delta = ll - xx; - } - else - { /* delta x[p] > 0, so x[p] goes toward its upper bound */ - xassert(uu != +DBL_MAX); - delta = uu - xx; - } - /* compute new x[k] = x[k] + alfa[k,q] * delta x[q], where - delta x[q] = delta x[p] / alfa[p,q] */ - xassert(cval[cpiv] != 0.0); - new_x = x + (rval[rpiv] / cval[cpiv]) * delta; -store: /* store analysis results */ - if (kase < 0) - { if (coef1 != NULL) *coef1 = lim_coef; - if (var1 != NULL) *var1 = q; - if (value1 != NULL) *value1 = new_x; - } - else - { if (coef2 != NULL) *coef2 = lim_coef; - if (var2 != NULL) *var2 = q; - if (value2 != NULL) *value2 = new_x; - } - } - /* free working arrays */ - xfree(cind); - xfree(cval); - xfree(rind); - xfree(rval); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpapi13.c b/code/3rd_glpk/draft/glpapi13.c deleted file mode 100644 index 1181b397..00000000 --- a/code/3rd_glpk/draft/glpapi13.c +++ /dev/null @@ -1,710 +0,0 @@ -/* glpapi13.c (branch-and-bound interface routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/*********************************************************************** -* NAME -* -* glp_ios_reason - determine reason for calling the callback routine -* -* SYNOPSIS -* -* glp_ios_reason(glp_tree *tree); -* -* RETURNS -* -* The routine glp_ios_reason returns a code, which indicates why the -* user-defined callback routine is being called. */ - -int glp_ios_reason(glp_tree *tree) -{ return - tree->reason; -} - -/*********************************************************************** -* NAME -* -* glp_ios_get_prob - access the problem object -* -* SYNOPSIS -* -* glp_prob *glp_ios_get_prob(glp_tree *tree); -* -* DESCRIPTION -* -* The routine glp_ios_get_prob can be called from the user-defined -* callback routine to access the problem object, which is used by the -* MIP solver. It is the original problem object passed to the routine -* glp_intopt if the MIP presolver is not used; otherwise it is an -* internal problem object built by the presolver. If the current -* subproblem exists, LP segment of the problem object corresponds to -* its LP relaxation. -* -* RETURNS -* -* The routine glp_ios_get_prob returns a pointer to the problem object -* used by the MIP solver. */ - -glp_prob *glp_ios_get_prob(glp_tree *tree) -{ return - tree->mip; -} - -/*********************************************************************** -* NAME -* -* glp_ios_tree_size - determine size of the branch-and-bound tree -* -* SYNOPSIS -* -* void glp_ios_tree_size(glp_tree *tree, int *a_cnt, int *n_cnt, -* int *t_cnt); -* -* DESCRIPTION -* -* The routine glp_ios_tree_size stores the following three counts which -* characterize the current size of the branch-and-bound tree: -* -* a_cnt is the current number of active nodes, i.e. the current size of -* the active list; -* -* n_cnt is the current number of all (active and inactive) nodes; -* -* t_cnt is the total number of nodes including those which have been -* already removed from the tree. This count is increased whenever -* a new node appears in the tree and never decreased. -* -* If some of the parameters a_cnt, n_cnt, t_cnt is a null pointer, the -* corresponding count is not stored. */ - -void glp_ios_tree_size(glp_tree *tree, int *a_cnt, int *n_cnt, - int *t_cnt) -{ if (a_cnt != NULL) *a_cnt = tree->a_cnt; - if (n_cnt != NULL) *n_cnt = tree->n_cnt; - if (t_cnt != NULL) *t_cnt = tree->t_cnt; - return; -} - -/*********************************************************************** -* NAME -* -* glp_ios_curr_node - determine current active subproblem -* -* SYNOPSIS -* -* int glp_ios_curr_node(glp_tree *tree); -* -* RETURNS -* -* The routine glp_ios_curr_node returns the reference number of the -* current active subproblem. However, if the current subproblem does -* not exist, the routine returns zero. */ - -int glp_ios_curr_node(glp_tree *tree) -{ IOSNPD *node; - /* obtain pointer to the current subproblem */ - node = tree->curr; - /* return its reference number */ - return node == NULL ? 0 : node->p; -} - -/*********************************************************************** -* NAME -* -* glp_ios_next_node - determine next active subproblem -* -* SYNOPSIS -* -* int glp_ios_next_node(glp_tree *tree, int p); -* -* RETURNS -* -* If the parameter p is zero, the routine glp_ios_next_node returns -* the reference number of the first active subproblem. However, if the -* tree is empty, zero is returned. -* -* If the parameter p is not zero, it must specify the reference number -* of some active subproblem, in which case the routine returns the -* reference number of the next active subproblem. However, if there is -* no next active subproblem in the list, zero is returned. -* -* All subproblems in the active list are ordered chronologically, i.e. -* subproblem A precedes subproblem B if A was created before B. */ - -int glp_ios_next_node(glp_tree *tree, int p) -{ IOSNPD *node; - if (p == 0) - { /* obtain pointer to the first active subproblem */ - node = tree->head; - } - else - { /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_next_node: p = %d; invalid subproblem refer" - "ence number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* the specified subproblem must be active */ - if (node->count != 0) - xerror("glp_ios_next_node: p = %d; subproblem not in the ac" - "tive list\n", p); - /* obtain pointer to the next active subproblem */ - node = node->next; - } - /* return the reference number */ - return node == NULL ? 0 : node->p; -} - -/*********************************************************************** -* NAME -* -* glp_ios_prev_node - determine previous active subproblem -* -* SYNOPSIS -* -* int glp_ios_prev_node(glp_tree *tree, int p); -* -* RETURNS -* -* If the parameter p is zero, the routine glp_ios_prev_node returns -* the reference number of the last active subproblem. However, if the -* tree is empty, zero is returned. -* -* If the parameter p is not zero, it must specify the reference number -* of some active subproblem, in which case the routine returns the -* reference number of the previous active subproblem. However, if there -* is no previous active subproblem in the list, zero is returned. -* -* All subproblems in the active list are ordered chronologically, i.e. -* subproblem A precedes subproblem B if A was created before B. */ - -int glp_ios_prev_node(glp_tree *tree, int p) -{ IOSNPD *node; - if (p == 0) - { /* obtain pointer to the last active subproblem */ - node = tree->tail; - } - else - { /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_prev_node: p = %d; invalid subproblem refer" - "ence number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* the specified subproblem must be active */ - if (node->count != 0) - xerror("glp_ios_prev_node: p = %d; subproblem not in the ac" - "tive list\n", p); - /* obtain pointer to the previous active subproblem */ - node = node->prev; - } - /* return the reference number */ - return node == NULL ? 0 : node->p; -} - -/*********************************************************************** -* NAME -* -* glp_ios_up_node - determine parent subproblem -* -* SYNOPSIS -* -* int glp_ios_up_node(glp_tree *tree, int p); -* -* RETURNS -* -* The parameter p must specify the reference number of some (active or -* inactive) subproblem, in which case the routine iet_get_up_node -* returns the reference number of its parent subproblem. However, if -* the specified subproblem is the root of the tree and, therefore, has -* no parent, the routine returns zero. */ - -int glp_ios_up_node(glp_tree *tree, int p) -{ IOSNPD *node; - /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_up_node: p = %d; invalid subproblem reference " - "number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* obtain pointer to the parent subproblem */ - node = node->up; - /* return the reference number */ - return node == NULL ? 0 : node->p; -} - -/*********************************************************************** -* NAME -* -* glp_ios_node_level - determine subproblem level -* -* SYNOPSIS -* -* int glp_ios_node_level(glp_tree *tree, int p); -* -* RETURNS -* -* The routine glp_ios_node_level returns the level of the subproblem, -* whose reference number is p, in the branch-and-bound tree. (The root -* subproblem has level 0, and the level of any other subproblem is the -* level of its parent plus one.) */ - -int glp_ios_node_level(glp_tree *tree, int p) -{ IOSNPD *node; - /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_node_level: p = %d; invalid subproblem referen" - "ce number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* return the node level */ - return node->level; -} - -/*********************************************************************** -* NAME -* -* glp_ios_node_bound - determine subproblem local bound -* -* SYNOPSIS -* -* double glp_ios_node_bound(glp_tree *tree, int p); -* -* RETURNS -* -* The routine glp_ios_node_bound returns the local bound for (active or -* inactive) subproblem, whose reference number is p. -* -* COMMENTS -* -* The local bound for subproblem p is an lower (minimization) or upper -* (maximization) bound for integer optimal solution to this subproblem -* (not to the original problem). This bound is local in the sense that -* only subproblems in the subtree rooted at node p cannot have better -* integer feasible solutions. -* -* On creating a subproblem (due to the branching step) its local bound -* is inherited from its parent and then may get only stronger (never -* weaker). For the root subproblem its local bound is initially set to -* -DBL_MAX (minimization) or +DBL_MAX (maximization) and then improved -* as the root LP relaxation has been solved. -* -* Note that the local bound is not necessarily the optimal objective -* value to corresponding LP relaxation; it may be stronger. */ - -double glp_ios_node_bound(glp_tree *tree, int p) -{ IOSNPD *node; - /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_node_bound: p = %d; invalid subproblem referen" - "ce number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* return the node local bound */ - return node->bound; -} - -/*********************************************************************** -* NAME -* -* glp_ios_best_node - find active subproblem with best local bound -* -* SYNOPSIS -* -* int glp_ios_best_node(glp_tree *tree); -* -* RETURNS -* -* The routine glp_ios_best_node returns the reference number of the -* active subproblem, whose local bound is best (i.e. smallest in case -* of minimization or largest in case of maximization). However, if the -* tree is empty, the routine returns zero. -* -* COMMENTS -* -* The best local bound is an lower (minimization) or upper -* (maximization) bound for integer optimal solution to the original -* MIP problem. */ - -int glp_ios_best_node(glp_tree *tree) -{ return - ios_best_node(tree); -} - -/*********************************************************************** -* NAME -* -* glp_ios_mip_gap - compute relative MIP gap -* -* SYNOPSIS -* -* double glp_ios_mip_gap(glp_tree *tree); -* -* DESCRIPTION -* -* The routine glp_ios_mip_gap computes the relative MIP gap with the -* following formula: -* -* gap = |best_mip - best_bnd| / (|best_mip| + DBL_EPSILON), -* -* where best_mip is the best integer feasible solution found so far, -* best_bnd is the best (global) bound. If no integer feasible solution -* has been found yet, gap is set to DBL_MAX. -* -* RETURNS -* -* The routine glp_ios_mip_gap returns the relative MIP gap. */ - -double glp_ios_mip_gap(glp_tree *tree) -{ return - ios_relative_gap(tree); -} - -/*********************************************************************** -* NAME -* -* glp_ios_node_data - access subproblem application-specific data -* -* SYNOPSIS -* -* void *glp_ios_node_data(glp_tree *tree, int p); -* -* DESCRIPTION -* -* The routine glp_ios_node_data allows the application accessing a -* memory block allocated for the subproblem (which may be active or -* inactive), whose reference number is p. -* -* The size of the block is defined by the control parameter cb_size -* passed to the routine glp_intopt. The block is initialized by binary -* zeros on creating corresponding subproblem, and its contents is kept -* until the subproblem will be removed from the tree. -* -* The application may use these memory blocks to store specific data -* for each subproblem. -* -* RETURNS -* -* The routine glp_ios_node_data returns a pointer to the memory block -* for the specified subproblem. Note that if cb_size = 0, the routine -* returns a null pointer. */ - -void *glp_ios_node_data(glp_tree *tree, int p) -{ IOSNPD *node; - /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_node_level: p = %d; invalid subproblem referen" - "ce number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* return pointer to the application-specific data */ - return node->data; -} - -/*********************************************************************** -* NAME -* -* glp_ios_row_attr - retrieve additional row attributes -* -* SYNOPSIS -* -* void glp_ios_row_attr(glp_tree *tree, int i, glp_attr *attr); -* -* DESCRIPTION -* -* The routine glp_ios_row_attr retrieves additional attributes of row -* i and stores them in the structure glp_attr. */ - -void glp_ios_row_attr(glp_tree *tree, int i, glp_attr *attr) -{ GLPROW *row; - if (!(1 <= i && i <= tree->mip->m)) - xerror("glp_ios_row_attr: i = %d; row number out of range\n", - i); - row = tree->mip->row[i]; - attr->level = row->level; - attr->origin = row->origin; - attr->klass = row->klass; - return; -} - -/**********************************************************************/ - -int glp_ios_pool_size(glp_tree *tree) -{ /* determine current size of the cut pool */ - if (tree->reason != GLP_ICUTGEN) - xerror("glp_ios_pool_size: operation not allowed\n"); - xassert(tree->local != NULL); -#ifdef NEW_LOCAL /* 02/II-2018 */ - return tree->local->m; -#else - return tree->local->size; -#endif -} - -/**********************************************************************/ - -int glp_ios_add_row(glp_tree *tree, - const char *name, int klass, int flags, int len, const int ind[], - const double val[], int type, double rhs) -{ /* add row (constraint) to the cut pool */ - int num; - if (tree->reason != GLP_ICUTGEN) - xerror("glp_ios_add_row: operation not allowed\n"); - xassert(tree->local != NULL); - num = ios_add_row(tree, tree->local, name, klass, flags, len, - ind, val, type, rhs); - return num; -} - -/**********************************************************************/ - -void glp_ios_del_row(glp_tree *tree, int i) -{ /* remove row (constraint) from the cut pool */ - if (tree->reason != GLP_ICUTGEN) - xerror("glp_ios_del_row: operation not allowed\n"); - ios_del_row(tree, tree->local, i); - return; -} - -/**********************************************************************/ - -void glp_ios_clear_pool(glp_tree *tree) -{ /* remove all rows (constraints) from the cut pool */ - if (tree->reason != GLP_ICUTGEN) - xerror("glp_ios_clear_pool: operation not allowed\n"); - ios_clear_pool(tree, tree->local); - return; -} - -/*********************************************************************** -* NAME -* -* glp_ios_can_branch - check if can branch upon specified variable -* -* SYNOPSIS -* -* int glp_ios_can_branch(glp_tree *tree, int j); -* -* RETURNS -* -* If j-th variable (column) can be used to branch upon, the routine -* glp_ios_can_branch returns non-zero, otherwise zero. */ - -int glp_ios_can_branch(glp_tree *tree, int j) -{ if (!(1 <= j && j <= tree->mip->n)) - xerror("glp_ios_can_branch: j = %d; column number out of range" - "\n", j); - return tree->non_int[j]; -} - -/*********************************************************************** -* NAME -* -* glp_ios_branch_upon - choose variable to branch upon -* -* SYNOPSIS -* -* void glp_ios_branch_upon(glp_tree *tree, int j, int sel); -* -* DESCRIPTION -* -* The routine glp_ios_branch_upon can be called from the user-defined -* callback routine in response to the reason GLP_IBRANCH to choose a -* branching variable, whose ordinal number is j. Should note that only -* variables, for which the routine glp_ios_can_branch returns non-zero, -* can be used to branch upon. -* -* The parameter sel is a flag that indicates which branch (subproblem) -* should be selected next to continue the search: -* -* GLP_DN_BRNCH - select down-branch; -* GLP_UP_BRNCH - select up-branch; -* GLP_NO_BRNCH - use general selection technique. */ - -void glp_ios_branch_upon(glp_tree *tree, int j, int sel) -{ if (!(1 <= j && j <= tree->mip->n)) - xerror("glp_ios_branch_upon: j = %d; column number out of rang" - "e\n", j); - if (!(sel == GLP_DN_BRNCH || sel == GLP_UP_BRNCH || - sel == GLP_NO_BRNCH)) - xerror("glp_ios_branch_upon: sel = %d: invalid branch selectio" - "n flag\n", sel); - if (!(tree->non_int[j])) - xerror("glp_ios_branch_upon: j = %d; variable cannot be used t" - "o branch upon\n", j); - if (tree->br_var != 0) - xerror("glp_ios_branch_upon: branching variable already chosen" - "\n"); - tree->br_var = j; - tree->br_sel = sel; - return; -} - -/*********************************************************************** -* NAME -* -* glp_ios_select_node - select subproblem to continue the search -* -* SYNOPSIS -* -* void glp_ios_select_node(glp_tree *tree, int p); -* -* DESCRIPTION -* -* The routine glp_ios_select_node can be called from the user-defined -* callback routine in response to the reason GLP_ISELECT to select an -* active subproblem, whose reference number is p. The search will be -* continued from the subproblem selected. */ - -void glp_ios_select_node(glp_tree *tree, int p) -{ IOSNPD *node; - /* obtain pointer to the specified subproblem */ - if (!(1 <= p && p <= tree->nslots)) -err: xerror("glp_ios_select_node: p = %d; invalid subproblem refere" - "nce number\n", p); - node = tree->slot[p].node; - if (node == NULL) goto err; - /* the specified subproblem must be active */ - if (node->count != 0) - xerror("glp_ios_select_node: p = %d; subproblem not in the act" - "ive list\n", p); - /* no subproblem must be selected yet */ - if (tree->next_p != 0) - xerror("glp_ios_select_node: subproblem already selected\n"); - /* select the specified subproblem to continue the search */ - tree->next_p = p; - return; -} - -/*********************************************************************** -* NAME -* -* glp_ios_heur_sol - provide solution found by heuristic -* -* SYNOPSIS -* -* int glp_ios_heur_sol(glp_tree *tree, const double x[]); -* -* DESCRIPTION -* -* The routine glp_ios_heur_sol can be called from the user-defined -* callback routine in response to the reason GLP_IHEUR to provide an -* integer feasible solution found by a primal heuristic. -* -* Primal values of *all* variables (columns) found by the heuristic -* should be placed in locations x[1], ..., x[n], where n is the number -* of columns in the original problem object. Note that the routine -* glp_ios_heur_sol *does not* check primal feasibility of the solution -* provided. -* -* Using the solution passed in the array x the routine computes value -* of the objective function. If the objective value is better than the -* best known integer feasible solution, the routine computes values of -* auxiliary variables (rows) and stores all solution components in the -* problem object. -* -* RETURNS -* -* If the provided solution is accepted, the routine glp_ios_heur_sol -* returns zero. Otherwise, if the provided solution is rejected, the -* routine returns non-zero. */ - -int glp_ios_heur_sol(glp_tree *tree, const double x[]) -{ glp_prob *mip = tree->mip; - int m = tree->orig_m; - int n = tree->n; - int i, j; - double obj; - xassert(mip->m >= m); - xassert(mip->n == n); - /* check values of integer variables and compute value of the - objective function */ - obj = mip->c0; - for (j = 1; j <= n; j++) - { GLPCOL *col = mip->col[j]; - if (col->kind == GLP_IV) - { /* provided value must be integral */ - if (x[j] != floor(x[j])) return 1; - } - obj += col->coef * x[j]; - } - /* check if the provided solution is better than the best known - integer feasible solution */ - if (mip->mip_stat == GLP_FEAS) - { switch (mip->dir) - { case GLP_MIN: - if (obj >= tree->mip->mip_obj) return 1; - break; - case GLP_MAX: - if (obj <= tree->mip->mip_obj) return 1; - break; - default: - xassert(mip != mip); - } - } - /* it is better; store it in the problem object */ - if (tree->parm->msg_lev >= GLP_MSG_ON) - xprintf("Solution found by heuristic: %.12g\n", obj); - mip->mip_stat = GLP_FEAS; - mip->mip_obj = obj; - for (j = 1; j <= n; j++) - mip->col[j]->mipx = x[j]; - for (i = 1; i <= m; i++) - { GLPROW *row = mip->row[i]; - GLPAIJ *aij; - row->mipx = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - row->mipx += aij->val * aij->col->mipx; - } -#if 1 /* 11/VII-2013 */ - ios_process_sol(tree); -#endif - return 0; -} - -/*********************************************************************** -* NAME -* -* glp_ios_terminate - terminate the solution process. -* -* SYNOPSIS -* -* void glp_ios_terminate(glp_tree *tree); -* -* DESCRIPTION -* -* The routine glp_ios_terminate sets a flag indicating that the MIP -* solver should prematurely terminate the search. */ - -void glp_ios_terminate(glp_tree *tree) -{ if (tree->parm->msg_lev >= GLP_MSG_DBG) - xprintf("The search is prematurely terminated due to applicati" - "on request\n"); - tree->stop = 1; - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glphbm.c b/code/3rd_glpk/draft/glphbm.c deleted file mode 100644 index 8b33c172..00000000 --- a/code/3rd_glpk/draft/glphbm.c +++ /dev/null @@ -1,533 +0,0 @@ -/* glphbm.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glphbm.h" -#include "misc.h" - -/*********************************************************************** -* NAME -* -* hbm_read_mat - read sparse matrix in Harwell-Boeing format -* -* SYNOPSIS -* -* #include "glphbm.h" -* HBM *hbm_read_mat(const char *fname); -* -* DESCRIPTION -* -* The routine hbm_read_mat reads a sparse matrix in the Harwell-Boeing -* format from a text file whose name is the character string fname. -* -* Detailed description of the Harwell-Boeing format recognised by this -* routine is given in the following report: -* -* I.S.Duff, R.G.Grimes, J.G.Lewis. User's Guide for the Harwell-Boeing -* Sparse Matrix Collection (Release I), TR/PA/92/86, October 1992. -* -* RETURNS -* -* If no error occured, the routine hbm_read_mat returns a pointer to -* a data structure containing the matrix. In case of error the routine -* prints an appropriate error message and returns NULL. */ - -struct dsa -{ /* working area used by routine hbm_read_mat */ - const char *fname; - /* name of input text file */ - FILE *fp; - /* stream assigned to input text file */ - int seqn; - /* card sequential number */ - char card[80+1]; - /* card image buffer */ - int fmt_p; - /* scale factor */ - int fmt_k; - /* iterator */ - int fmt_f; - /* format code */ - int fmt_w; - /* field width */ - int fmt_d; - /* number of decimal places after point */ -}; - -/*********************************************************************** -* read_card - read next data card -* -* This routine reads the next 80-column card from the input text file -* and stores its image into the character string card. If the card was -* read successfully, the routine returns zero, otherwise non-zero. */ - -#if 1 /* 11/III-2012 */ -static int read_card(struct dsa *dsa) -{ int c, len = 0; - char buf[255+1]; - dsa->seqn++; - for (;;) - { c = fgetc(dsa->fp); - if (c == EOF) - { if (ferror(dsa->fp)) - xprintf("%s:%d: read error\n", - dsa->fname, dsa->seqn); - else - xprintf("%s:%d: unexpected end-of-file\n", - dsa->fname, dsa->seqn); - return 1; - } - else if (c == '\r') - /* nop */; - else if (c == '\n') - break; - else if (iscntrl(c)) - { xprintf("%s:%d: invalid control character\n", - dsa->fname, dsa->seqn, c); - return 1; - } - else - { if (len == sizeof(buf)-1) - goto err; - buf[len++] = (char)c; - } - } - /* remove trailing spaces */ - while (len > 80 && buf[len-1] == ' ') - len--; - buf[len] = '\0'; - /* line should not be longer than 80 chars */ - if (len > 80) -err: { xerror("%s:%d: card image too long\n", - dsa->fname, dsa->seqn); - return 1; - } - /* padd by spaces to 80-column card image */ - strcpy(dsa->card, buf); - memset(&dsa->card[len], ' ', 80 - len); - dsa->card[80] = '\0'; - return 0; -} -#endif - -/*********************************************************************** -* scan_int - scan integer value from the current card -* -* This routine scans an integer value from the current card, where fld -* is the name of the field, pos is the position of the field, width is -* the width of the field, val points to a location to which the scanned -* value should be stored. If the value was scanned successfully, the -* routine returns zero, otherwise non-zero. */ - -static int scan_int(struct dsa *dsa, char *fld, int pos, int width, - int *val) -{ char str[80+1]; - xassert(1 <= width && width <= 80); - memcpy(str, dsa->card + pos, width), str[width] = '\0'; - if (str2int(strspx(str), val)) - { xprintf("%s:%d: field '%s' contains invalid value '%s'\n", - dsa->fname, dsa->seqn, fld, str); - return 1; - } - return 0; -} - -/*********************************************************************** -* parse_fmt - parse Fortran format specification -* -* This routine parses the Fortran format specification represented as -* character string which fmt points to and stores format elements into -* appropriate static locations. Should note that not all valid Fortran -* format specifications may be recognised. If the format specification -* was recognised, the routine returns zero, otherwise non-zero. */ - -static int parse_fmt(struct dsa *dsa, char *fmt) -{ int k, s, val; - char str[80+1]; - /* first character should be left parenthesis */ - if (fmt[0] != '(') -fail: { xprintf("hbm_read_mat: format '%s' not recognised\n", fmt); - return 1; - } - k = 1; - /* optional scale factor */ - dsa->fmt_p = 0; - if (isdigit((unsigned char)fmt[k])) - { s = 0; - while (isdigit((unsigned char)fmt[k])) - { if (s == 80) goto fail; - str[s++] = fmt[k++]; - } - str[s] = '\0'; - if (str2int(str, &val)) goto fail; - if (toupper((unsigned char)fmt[k]) != 'P') goto iter; - dsa->fmt_p = val, k++; - if (!(0 <= dsa->fmt_p && dsa->fmt_p <= 255)) goto fail; - /* optional comma may follow scale factor */ - if (fmt[k] == ',') k++; - } - /* optional iterator */ - dsa->fmt_k = 1; - if (isdigit((unsigned char)fmt[k])) - { s = 0; - while (isdigit((unsigned char)fmt[k])) - { if (s == 80) goto fail; - str[s++] = fmt[k++]; - } - str[s] = '\0'; - if (str2int(str, &val)) goto fail; -iter: dsa->fmt_k = val; - if (!(1 <= dsa->fmt_k && dsa->fmt_k <= 255)) goto fail; - } - /* format code */ - dsa->fmt_f = toupper((unsigned char)fmt[k++]); - if (!(dsa->fmt_f == 'D' || dsa->fmt_f == 'E' || - dsa->fmt_f == 'F' || dsa->fmt_f == 'G' || - dsa->fmt_f == 'I')) goto fail; - /* field width */ - if (!isdigit((unsigned char)fmt[k])) goto fail; - s = 0; - while (isdigit((unsigned char)fmt[k])) - { if (s == 80) goto fail; - str[s++] = fmt[k++]; - } - str[s] = '\0'; - if (str2int(str, &dsa->fmt_w)) goto fail; - if (!(1 <= dsa->fmt_w && dsa->fmt_w <= 255)) goto fail; - /* optional number of decimal places after point */ - dsa->fmt_d = 0; - if (fmt[k] == '.') - { k++; - if (!isdigit((unsigned char)fmt[k])) goto fail; - s = 0; - while (isdigit((unsigned char)fmt[k])) - { if (s == 80) goto fail; - str[s++] = fmt[k++]; - } - str[s] = '\0'; - if (str2int(str, &dsa->fmt_d)) goto fail; - if (!(0 <= dsa->fmt_d && dsa->fmt_d <= 255)) goto fail; - } - /* last character should be right parenthesis */ - if (!(fmt[k] == ')' && fmt[k+1] == '\0')) goto fail; - return 0; -} - -/*********************************************************************** -* read_int_array - read array of integer type -* -* This routine reads an integer array from the input text file, where -* name is array name, fmt is Fortran format specification that controls -* reading, n is number of array elements, val is array of integer type. -* If the array was read successful, the routine returns zero, otherwise -* non-zero. */ - -static int read_int_array(struct dsa *dsa, char *name, char *fmt, - int n, int val[]) -{ int k, pos; - char str[80+1]; - if (parse_fmt(dsa, fmt)) return 1; - if (!(dsa->fmt_f == 'I' && dsa->fmt_w <= 80 && - dsa->fmt_k * dsa->fmt_w <= 80)) - { xprintf( - "%s:%d: can't read array '%s' - invalid format '%s'\n", - dsa->fname, dsa->seqn, name, fmt); - return 1; - } - for (k = 1, pos = INT_MAX; k <= n; k++, pos++) - { if (pos >= dsa->fmt_k) - { if (read_card(dsa)) return 1; - pos = 0; - } - memcpy(str, dsa->card + dsa->fmt_w * pos, dsa->fmt_w); - str[dsa->fmt_w] = '\0'; - strspx(str); - if (str2int(str, &val[k])) - { xprintf( - "%s:%d: can't read array '%s' - invalid value '%s'\n", - dsa->fname, dsa->seqn, name, str); - return 1; - } - } - return 0; -} - -/*********************************************************************** -* read_real_array - read array of real type -* -* This routine reads a real array from the input text file, where name -* is array name, fmt is Fortran format specification that controls -* reading, n is number of array elements, val is array of real type. -* If the array was read successful, the routine returns zero, otherwise -* non-zero. */ - -static int read_real_array(struct dsa *dsa, char *name, char *fmt, - int n, double val[]) -{ int k, pos; - char str[80+1], *ptr; - if (parse_fmt(dsa, fmt)) return 1; - if (!(dsa->fmt_f != 'I' && dsa->fmt_w <= 80 && - dsa->fmt_k * dsa->fmt_w <= 80)) - { xprintf( - "%s:%d: can't read array '%s' - invalid format '%s'\n", - dsa->fname, dsa->seqn, name, fmt); - return 1; - } - for (k = 1, pos = INT_MAX; k <= n; k++, pos++) - { if (pos >= dsa->fmt_k) - { if (read_card(dsa)) return 1; - pos = 0; - } - memcpy(str, dsa->card + dsa->fmt_w * pos, dsa->fmt_w); - str[dsa->fmt_w] = '\0'; - strspx(str); - if (strchr(str, '.') == NULL && strcmp(str, "0")) - { xprintf("%s(%d): can't read array '%s' - value '%s' has no " - "decimal point\n", dsa->fname, dsa->seqn, name, str); - return 1; - } - /* sometimes lower case letters appear */ - for (ptr = str; *ptr; ptr++) - *ptr = (char)toupper((unsigned char)*ptr); - ptr = strchr(str, 'D'); - if (ptr != NULL) *ptr = 'E'; - /* value may appear with decimal exponent but without letters - E or D (for example, -123.456-012), so missing letter should - be inserted */ - ptr = strchr(str+1, '+'); - if (ptr == NULL) ptr = strchr(str+1, '-'); - if (ptr != NULL && *(ptr-1) != 'E') - { xassert(strlen(str) < 80); - memmove(ptr+1, ptr, strlen(ptr)+1); - *ptr = 'E'; - } - if (str2num(str, &val[k])) - { xprintf( - "%s:%d: can't read array '%s' - invalid value '%s'\n", - dsa->fname, dsa->seqn, name, str); - return 1; - } - } - return 0; -} - -HBM *hbm_read_mat(const char *fname) -{ struct dsa _dsa, *dsa = &_dsa; - HBM *hbm = NULL; - dsa->fname = fname; - xprintf("hbm_read_mat: reading matrix from '%s'...\n", - dsa->fname); - dsa->fp = fopen(dsa->fname, "r"); - if (dsa->fp == NULL) - { xprintf("hbm_read_mat: unable to open '%s' - %s\n", -#if 0 /* 29/I-2017 */ - dsa->fname, strerror(errno)); -#else - dsa->fname, xstrerr(errno)); -#endif - goto fail; - } - dsa->seqn = 0; - hbm = xmalloc(sizeof(HBM)); - memset(hbm, 0, sizeof(HBM)); - /* read the first heading card */ - if (read_card(dsa)) goto fail; - memcpy(hbm->title, dsa->card, 72), hbm->title[72] = '\0'; - strtrim(hbm->title); - xprintf("%s\n", hbm->title); - memcpy(hbm->key, dsa->card+72, 8), hbm->key[8] = '\0'; - strspx(hbm->key); - xprintf("key = %s\n", hbm->key); - /* read the second heading card */ - if (read_card(dsa)) goto fail; - if (scan_int(dsa, "totcrd", 0, 14, &hbm->totcrd)) goto fail; - if (scan_int(dsa, "ptrcrd", 14, 14, &hbm->ptrcrd)) goto fail; - if (scan_int(dsa, "indcrd", 28, 14, &hbm->indcrd)) goto fail; - if (scan_int(dsa, "valcrd", 42, 14, &hbm->valcrd)) goto fail; - if (scan_int(dsa, "rhscrd", 56, 14, &hbm->rhscrd)) goto fail; - xprintf("totcrd = %d; ptrcrd = %d; indcrd = %d; valcrd = %d; rhsc" - "rd = %d\n", hbm->totcrd, hbm->ptrcrd, hbm->indcrd, - hbm->valcrd, hbm->rhscrd); - /* read the third heading card */ - if (read_card(dsa)) goto fail; - memcpy(hbm->mxtype, dsa->card, 3), hbm->mxtype[3] = '\0'; - if (strchr("RCP", hbm->mxtype[0]) == NULL || - strchr("SUHZR", hbm->mxtype[1]) == NULL || - strchr("AE", hbm->mxtype[2]) == NULL) - { xprintf("%s:%d: matrix type '%s' not recognised\n", - dsa->fname, dsa->seqn, hbm->mxtype); - goto fail; - } - if (scan_int(dsa, "nrow", 14, 14, &hbm->nrow)) goto fail; - if (scan_int(dsa, "ncol", 28, 14, &hbm->ncol)) goto fail; - if (scan_int(dsa, "nnzero", 42, 14, &hbm->nnzero)) goto fail; - if (scan_int(dsa, "neltvl", 56, 14, &hbm->neltvl)) goto fail; - xprintf("mxtype = %s; nrow = %d; ncol = %d; nnzero = %d; neltvl =" - " %d\n", hbm->mxtype, hbm->nrow, hbm->ncol, hbm->nnzero, - hbm->neltvl); - /* read the fourth heading card */ - if (read_card(dsa)) goto fail; - memcpy(hbm->ptrfmt, dsa->card, 16), hbm->ptrfmt[16] = '\0'; - strspx(hbm->ptrfmt); - memcpy(hbm->indfmt, dsa->card+16, 16), hbm->indfmt[16] = '\0'; - strspx(hbm->indfmt); - memcpy(hbm->valfmt, dsa->card+32, 20), hbm->valfmt[20] = '\0'; - strspx(hbm->valfmt); - memcpy(hbm->rhsfmt, dsa->card+52, 20), hbm->rhsfmt[20] = '\0'; - strspx(hbm->rhsfmt); - xprintf("ptrfmt = %s; indfmt = %s; valfmt = %s; rhsfmt = %s\n", - hbm->ptrfmt, hbm->indfmt, hbm->valfmt, hbm->rhsfmt); - /* read the fifth heading card (optional) */ - if (hbm->rhscrd <= 0) - { strcpy(hbm->rhstyp, "???"); - hbm->nrhs = 0; - hbm->nrhsix = 0; - } - else - { if (read_card(dsa)) goto fail; - memcpy(hbm->rhstyp, dsa->card, 3), hbm->rhstyp[3] = '\0'; - if (scan_int(dsa, "nrhs", 14, 14, &hbm->nrhs)) goto fail; - if (scan_int(dsa, "nrhsix", 28, 14, &hbm->nrhsix)) goto fail; - xprintf("rhstyp = '%s'; nrhs = %d; nrhsix = %d\n", - hbm->rhstyp, hbm->nrhs, hbm->nrhsix); - } - /* read matrix structure */ - hbm->colptr = xcalloc(1+hbm->ncol+1, sizeof(int)); - if (read_int_array(dsa, "colptr", hbm->ptrfmt, hbm->ncol+1, - hbm->colptr)) goto fail; - hbm->rowind = xcalloc(1+hbm->nnzero, sizeof(int)); - if (read_int_array(dsa, "rowind", hbm->indfmt, hbm->nnzero, - hbm->rowind)) goto fail; - /* read matrix values */ - if (hbm->valcrd <= 0) goto done; - if (hbm->mxtype[2] == 'A') - { /* assembled matrix */ - hbm->values = xcalloc(1+hbm->nnzero, sizeof(double)); - if (read_real_array(dsa, "values", hbm->valfmt, hbm->nnzero, - hbm->values)) goto fail; - } - else - { /* elemental (unassembled) matrix */ - hbm->values = xcalloc(1+hbm->neltvl, sizeof(double)); - if (read_real_array(dsa, "values", hbm->valfmt, hbm->neltvl, - hbm->values)) goto fail; - } - /* read right-hand sides */ - if (hbm->nrhs <= 0) goto done; - if (hbm->rhstyp[0] == 'F') - { /* dense format */ - hbm->nrhsvl = hbm->nrow * hbm->nrhs; - hbm->rhsval = xcalloc(1+hbm->nrhsvl, sizeof(double)); - if (read_real_array(dsa, "rhsval", hbm->rhsfmt, hbm->nrhsvl, - hbm->rhsval)) goto fail; - } - else if (hbm->rhstyp[0] == 'M' && hbm->mxtype[2] == 'A') - { /* sparse format */ - /* read pointers */ - hbm->rhsptr = xcalloc(1+hbm->nrhs+1, sizeof(int)); - if (read_int_array(dsa, "rhsptr", hbm->ptrfmt, hbm->nrhs+1, - hbm->rhsptr)) goto fail; - /* read sparsity pattern */ - hbm->rhsind = xcalloc(1+hbm->nrhsix, sizeof(int)); - if (read_int_array(dsa, "rhsind", hbm->indfmt, hbm->nrhsix, - hbm->rhsind)) goto fail; - /* read values */ - hbm->rhsval = xcalloc(1+hbm->nrhsix, sizeof(double)); - if (read_real_array(dsa, "rhsval", hbm->rhsfmt, hbm->nrhsix, - hbm->rhsval)) goto fail; - } - else if (hbm->rhstyp[0] == 'M' && hbm->mxtype[2] == 'E') - { /* elemental format */ - hbm->rhsval = xcalloc(1+hbm->nrhsvl, sizeof(double)); - if (read_real_array(dsa, "rhsval", hbm->rhsfmt, hbm->nrhsvl, - hbm->rhsval)) goto fail; - } - else - { xprintf("%s:%d: right-hand side type '%c' not recognised\n", - dsa->fname, dsa->seqn, hbm->rhstyp[0]); - goto fail; - } - /* read starting guesses */ - if (hbm->rhstyp[1] == 'G') - { hbm->nguess = hbm->nrow * hbm->nrhs; - hbm->sguess = xcalloc(1+hbm->nguess, sizeof(double)); - if (read_real_array(dsa, "sguess", hbm->rhsfmt, hbm->nguess, - hbm->sguess)) goto fail; - } - /* read solution vectors */ - if (hbm->rhstyp[2] == 'X') - { hbm->nexact = hbm->nrow * hbm->nrhs; - hbm->xexact = xcalloc(1+hbm->nexact, sizeof(double)); - if (read_real_array(dsa, "xexact", hbm->rhsfmt, hbm->nexact, - hbm->xexact)) goto fail; - } -done: /* reading has been completed */ - xprintf("hbm_read_mat: %d cards were read\n", dsa->seqn); - fclose(dsa->fp); - return hbm; -fail: /* something wrong in Danish kingdom */ - if (hbm != NULL) - { if (hbm->colptr != NULL) xfree(hbm->colptr); - if (hbm->rowind != NULL) xfree(hbm->rowind); - if (hbm->rhsptr != NULL) xfree(hbm->rhsptr); - if (hbm->rhsind != NULL) xfree(hbm->rhsind); - if (hbm->values != NULL) xfree(hbm->values); - if (hbm->rhsval != NULL) xfree(hbm->rhsval); - if (hbm->sguess != NULL) xfree(hbm->sguess); - if (hbm->xexact != NULL) xfree(hbm->xexact); - xfree(hbm); - } - if (dsa->fp != NULL) fclose(dsa->fp); - return NULL; -} - -/*********************************************************************** -* NAME -* -* hbm_free_mat - free sparse matrix in Harwell-Boeing format -* -* SYNOPSIS -* -* #include "glphbm.h" -* void hbm_free_mat(HBM *hbm); -* -* DESCRIPTION -* -* The hbm_free_mat routine frees all the memory allocated to the data -* structure containing a sparse matrix in the Harwell-Boeing format. */ - -void hbm_free_mat(HBM *hbm) -{ if (hbm->colptr != NULL) xfree(hbm->colptr); - if (hbm->rowind != NULL) xfree(hbm->rowind); - if (hbm->rhsptr != NULL) xfree(hbm->rhsptr); - if (hbm->rhsind != NULL) xfree(hbm->rhsind); - if (hbm->values != NULL) xfree(hbm->values); - if (hbm->rhsval != NULL) xfree(hbm->rhsval); - if (hbm->sguess != NULL) xfree(hbm->sguess); - if (hbm->xexact != NULL) xfree(hbm->xexact); - xfree(hbm); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glphbm.h b/code/3rd_glpk/draft/glphbm.h deleted file mode 100644 index 688a78ec..00000000 --- a/code/3rd_glpk/draft/glphbm.h +++ /dev/null @@ -1,127 +0,0 @@ -/* glphbm.h (Harwell-Boeing sparse matrix format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPHBM_H -#define GLPHBM_H - -typedef struct HBM HBM; - -struct HBM -{ /* sparse matrix in Harwell-Boeing format; for details see the - report: I.S.Duff, R.G.Grimes, J.G.Lewis. User's Guide for the - Harwell-Boeing Sparse Matrix Collection (Release I), 1992 */ - char title[72+1]; - /* matrix title (informative) */ - char key[8+1]; - /* matrix key (informative) */ - char mxtype[3+1]; - /* matrix type: - R.. real matrix - C.. complex matrix - P.. pattern only (no numerical values supplied) - .S. symmetric (lower triangle + main diagonal) - .U. unsymmetric - .H. hermitian (lower triangle + main diagonal) - .Z. skew symmetric (lower triangle only) - .R. rectangular - ..A assembled - ..E elemental (unassembled) */ - char rhstyp[3+1]; - /* optional types: - F.. right-hand sides in dense format - M.. right-hand sides in same format as matrix - .G. starting vector(s) (guess) is supplied - ..X exact solution vector(s) is supplied */ - char ptrfmt[16+1]; - /* format for pointers */ - char indfmt[16+1]; - /* format for row (or variable) indices */ - char valfmt[20+1]; - /* format for numerical values of coefficient matrix */ - char rhsfmt[20+1]; - /* format for numerical values of right-hand sides */ - int totcrd; - /* total number of cards excluding header */ - int ptrcrd; - /* number of cards for ponters */ - int indcrd; - /* number of cards for row (or variable) indices */ - int valcrd; - /* number of cards for numerical values */ - int rhscrd; - /* number of lines for right-hand sides; - including starting guesses and solution vectors if present; - zero indicates no right-hand side data is present */ - int nrow; - /* number of rows (or variables) */ - int ncol; - /* number of columns (or elements) */ - int nnzero; - /* number of row (or variable) indices; - equal to number of entries for assembled matrix */ - int neltvl; - /* number of elemental matrix entries; - zero in case of assembled matrix */ - int nrhs; - /* number of right-hand sides */ - int nrhsix; - /* number of row indices; - ignored in case of unassembled matrix */ - int nrhsvl; - /* total number of entries in all right-hand sides */ - int nguess; - /* total number of entries in all starting guesses */ - int nexact; - /* total number of entries in all solution vectors */ - int *colptr; /* alias: eltptr */ - /* column pointers (in case of assembled matrix); - elemental matrix pointers (in case of unassembled matrix) */ - int *rowind; /* alias: varind */ - /* row indices (in case of assembled matrix); - variable indices (in case of unassembled matrix) */ - int *rhsptr; - /* right-hand side pointers */ - int *rhsind; - /* right-hand side indices */ - double *values; - /* matrix values */ - double *rhsval; - /* right-hand side values */ - double *sguess; - /* starting guess values */ - double *xexact; - /* solution vector values */ -}; - -#define hbm_read_mat _glp_hbm_read_mat -HBM *hbm_read_mat(const char *fname); -/* read sparse matrix in Harwell-Boeing format */ - -#define hbm_free_mat _glp_hbm_free_mat -void hbm_free_mat(HBM *hbm); -/* free sparse matrix in Harwell-Boeing format */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios01.c b/code/3rd_glpk/draft/glpios01.c deleted file mode 100644 index cb1a0dab..00000000 --- a/code/3rd_glpk/draft/glpios01.c +++ /dev/null @@ -1,1685 +0,0 @@ -/* glpios01.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" -#include "misc.h" - -static int lpx_eval_tab_row(glp_prob *lp, int k, int ind[], - double val[]) -{ /* compute row of the simplex tableau */ - return glp_eval_tab_row(lp, k, ind, val); -} - -static int lpx_dual_ratio_test(glp_prob *lp, int len, const int ind[], - const double val[], int how, double tol) -{ /* perform dual ratio test */ - int piv; - piv = glp_dual_rtest(lp, len, ind, val, how, tol); - xassert(0 <= piv && piv <= len); - return piv == 0 ? 0 : ind[piv]; -} - -/*********************************************************************** -* NAME -* -* ios_create_tree - create branch-and-bound tree -* -* SYNOPSIS -* -* #include "glpios.h" -* glp_tree *ios_create_tree(glp_prob *mip, const glp_iocp *parm); -* -* DESCRIPTION -* -* The routine ios_create_tree creates the branch-and-bound tree. -* -* Being created the tree consists of the only root subproblem whose -* reference number is 1. Note that initially the root subproblem is in -* frozen state and therefore needs to be revived. -* -* RETURNS -* -* The routine returns a pointer to the tree created. */ - -static IOSNPD *new_node(glp_tree *tree, IOSNPD *parent); - -glp_tree *ios_create_tree(glp_prob *mip, const glp_iocp *parm) -{ int m = mip->m; - int n = mip->n; - glp_tree *tree; - int i, j; - xassert(mip->tree == NULL); - mip->tree = tree = xmalloc(sizeof(glp_tree)); - tree->pool = dmp_create_pool(); - tree->n = n; - /* save original problem components */ - tree->orig_m = m; - tree->orig_type = xcalloc(1+m+n, sizeof(char)); - tree->orig_lb = xcalloc(1+m+n, sizeof(double)); - tree->orig_ub = xcalloc(1+m+n, sizeof(double)); - tree->orig_stat = xcalloc(1+m+n, sizeof(char)); - tree->orig_prim = xcalloc(1+m+n, sizeof(double)); - tree->orig_dual = xcalloc(1+m+n, sizeof(double)); - for (i = 1; i <= m; i++) - { GLPROW *row = mip->row[i]; - tree->orig_type[i] = (char)row->type; - tree->orig_lb[i] = row->lb; - tree->orig_ub[i] = row->ub; - tree->orig_stat[i] = (char)row->stat; - tree->orig_prim[i] = row->prim; - tree->orig_dual[i] = row->dual; - } - for (j = 1; j <= n; j++) - { GLPCOL *col = mip->col[j]; - tree->orig_type[m+j] = (char)col->type; - tree->orig_lb[m+j] = col->lb; - tree->orig_ub[m+j] = col->ub; - tree->orig_stat[m+j] = (char)col->stat; - tree->orig_prim[m+j] = col->prim; - tree->orig_dual[m+j] = col->dual; - } - tree->orig_obj = mip->obj_val; - /* initialize the branch-and-bound tree */ - tree->nslots = 0; - tree->avail = 0; - tree->slot = NULL; - tree->head = tree->tail = NULL; - tree->a_cnt = tree->n_cnt = tree->t_cnt = 0; - /* the root subproblem is not solved yet, so its final components - are unknown so far */ - tree->root_m = 0; - tree->root_type = NULL; - tree->root_lb = tree->root_ub = NULL; - tree->root_stat = NULL; - /* the current subproblem does not exist yet */ - tree->curr = NULL; - tree->mip = mip; - /*tree->solved = 0;*/ - tree->non_int = xcalloc(1+n, sizeof(char)); - memset(&tree->non_int[1], 0, n); - /* arrays to save parent subproblem components will be allocated - later */ - tree->pred_m = tree->pred_max = 0; - tree->pred_type = NULL; - tree->pred_lb = tree->pred_ub = NULL; - tree->pred_stat = NULL; - /* cut generators */ - tree->local = ios_create_pool(tree); - /*tree->first_attempt = 1;*/ - /*tree->max_added_cuts = 0;*/ - /*tree->min_eff = 0.0;*/ - /*tree->miss = 0;*/ - /*tree->just_selected = 0;*/ -#ifdef NEW_COVER /* 13/II-2018 */ - tree->cov_gen = NULL; -#endif - tree->mir_gen = NULL; - tree->clq_gen = NULL; - /*tree->round = 0;*/ -#if 0 - /* create the conflict graph */ - tree->n_ref = xcalloc(1+n, sizeof(int)); - memset(&tree->n_ref[1], 0, n * sizeof(int)); - tree->c_ref = xcalloc(1+n, sizeof(int)); - memset(&tree->c_ref[1], 0, n * sizeof(int)); - tree->g = scg_create_graph(0); - tree->j_ref = xcalloc(1+tree->g->n_max, sizeof(int)); -#endif - /* pseudocost branching */ - tree->pcost = NULL; - tree->iwrk = xcalloc(1+n, sizeof(int)); - tree->dwrk = xcalloc(1+n, sizeof(double)); - /* initialize control parameters */ - tree->parm = parm; - tree->tm_beg = xtime(); -#if 0 /* 10/VI-2013 */ - tree->tm_lag = xlset(0); -#else - tree->tm_lag = 0.0; -#endif - tree->sol_cnt = 0; -#if 1 /* 11/VII-2013 */ - tree->P = NULL; - tree->npp = NULL; - tree->save_sol = parm->save_sol; - tree->save_cnt = 0; -#endif - /* initialize advanced solver interface */ - tree->reason = 0; - tree->reopt = 0; - tree->reinv = 0; - tree->br_var = 0; - tree->br_sel = 0; - tree->child = 0; - tree->next_p = 0; - /*tree->btrack = NULL;*/ - tree->stop = 0; - /* create the root subproblem, which initially is identical to - the original MIP */ - new_node(tree, NULL); - return tree; -} - -/*********************************************************************** -* NAME -* -* ios_revive_node - revive specified subproblem -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_revive_node(glp_tree *tree, int p); -* -* DESCRIPTION -* -* The routine ios_revive_node revives the specified subproblem, whose -* reference number is p, and thereby makes it the current subproblem. -* Note that the specified subproblem must be active. Besides, if the -* current subproblem already exists, it must be frozen before reviving -* another subproblem. */ - -void ios_revive_node(glp_tree *tree, int p) -{ glp_prob *mip = tree->mip; - IOSNPD *node, *root; - /* obtain pointer to the specified subproblem */ - xassert(1 <= p && p <= tree->nslots); - node = tree->slot[p].node; - xassert(node != NULL); - /* the specified subproblem must be active */ - xassert(node->count == 0); - /* the current subproblem must not exist */ - xassert(tree->curr == NULL); - /* the specified subproblem becomes current */ - tree->curr = node; - /*tree->solved = 0;*/ - /* obtain pointer to the root subproblem */ - root = tree->slot[1].node; - xassert(root != NULL); - /* at this point problem object components correspond to the root - subproblem, so if the root subproblem should be revived, there - is nothing more to do */ - if (node == root) goto done; - xassert(mip->m == tree->root_m); - /* build path from the root to the current node */ - node->temp = NULL; - for (node = node; node != NULL; node = node->up) - { if (node->up == NULL) - xassert(node == root); - else - node->up->temp = node; - } - /* go down from the root to the current node and make necessary - changes to restore components of the current subproblem */ - for (node = root; node != NULL; node = node->temp) - { int m = mip->m; - int n = mip->n; - /* if the current node is reached, the problem object at this - point corresponds to its parent, so save attributes of rows - and columns for the parent subproblem */ - if (node->temp == NULL) - { int i, j; - tree->pred_m = m; - /* allocate/reallocate arrays, if necessary */ - if (tree->pred_max < m + n) - { int new_size = m + n + 100; - if (tree->pred_type != NULL) xfree(tree->pred_type); - if (tree->pred_lb != NULL) xfree(tree->pred_lb); - if (tree->pred_ub != NULL) xfree(tree->pred_ub); - if (tree->pred_stat != NULL) xfree(tree->pred_stat); - tree->pred_max = new_size; - tree->pred_type = xcalloc(1+new_size, sizeof(char)); - tree->pred_lb = xcalloc(1+new_size, sizeof(double)); - tree->pred_ub = xcalloc(1+new_size, sizeof(double)); - tree->pred_stat = xcalloc(1+new_size, sizeof(char)); - } - /* save row attributes */ - for (i = 1; i <= m; i++) - { GLPROW *row = mip->row[i]; - tree->pred_type[i] = (char)row->type; - tree->pred_lb[i] = row->lb; - tree->pred_ub[i] = row->ub; - tree->pred_stat[i] = (char)row->stat; - } - /* save column attributes */ - for (j = 1; j <= n; j++) - { GLPCOL *col = mip->col[j]; - tree->pred_type[mip->m+j] = (char)col->type; - tree->pred_lb[mip->m+j] = col->lb; - tree->pred_ub[mip->m+j] = col->ub; - tree->pred_stat[mip->m+j] = (char)col->stat; - } - } - /* change bounds of rows and columns */ - { IOSBND *b; - for (b = node->b_ptr; b != NULL; b = b->next) - { if (b->k <= m) - glp_set_row_bnds(mip, b->k, b->type, b->lb, b->ub); - else - glp_set_col_bnds(mip, b->k-m, b->type, b->lb, b->ub); - } - } - /* change statuses of rows and columns */ - { IOSTAT *s; - for (s = node->s_ptr; s != NULL; s = s->next) - { if (s->k <= m) - glp_set_row_stat(mip, s->k, s->stat); - else - glp_set_col_stat(mip, s->k-m, s->stat); - } - } - /* add new rows */ - if (node->r_ptr != NULL) - { IOSROW *r; - IOSAIJ *a; - int i, len, *ind; - double *val; - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - for (r = node->r_ptr; r != NULL; r = r->next) - { i = glp_add_rows(mip, 1); - glp_set_row_name(mip, i, r->name); -#if 1 /* 20/IX-2008 */ - xassert(mip->row[i]->level == 0); - mip->row[i]->level = node->level; - mip->row[i]->origin = r->origin; - mip->row[i]->klass = r->klass; -#endif - glp_set_row_bnds(mip, i, r->type, r->lb, r->ub); - len = 0; - for (a = r->ptr; a != NULL; a = a->next) - len++, ind[len] = a->j, val[len] = a->val; - glp_set_mat_row(mip, i, len, ind, val); - glp_set_rii(mip, i, r->rii); - glp_set_row_stat(mip, i, r->stat); - } - xfree(ind); - xfree(val); - } -#if 0 - /* add new edges to the conflict graph */ - /* add new cliques to the conflict graph */ - /* (not implemented yet) */ - xassert(node->own_nn == 0); - xassert(node->own_nc == 0); - xassert(node->e_ptr == NULL); -#endif - } - /* the specified subproblem has been revived */ - node = tree->curr; - /* delete its bound change list */ - while (node->b_ptr != NULL) - { IOSBND *b; - b = node->b_ptr; - node->b_ptr = b->next; - dmp_free_atom(tree->pool, b, sizeof(IOSBND)); - } - /* delete its status change list */ - while (node->s_ptr != NULL) - { IOSTAT *s; - s = node->s_ptr; - node->s_ptr = s->next; - dmp_free_atom(tree->pool, s, sizeof(IOSTAT)); - } -#if 1 /* 20/XI-2009 */ - /* delete its row addition list (additional rows may appear, for - example, due to branching on GUB constraints */ - while (node->r_ptr != NULL) - { IOSROW *r; - r = node->r_ptr; - node->r_ptr = r->next; - xassert(r->name == NULL); - while (r->ptr != NULL) - { IOSAIJ *a; - a = r->ptr; - r->ptr = a->next; - dmp_free_atom(tree->pool, a, sizeof(IOSAIJ)); - } - dmp_free_atom(tree->pool, r, sizeof(IOSROW)); - } -#endif -done: return; -} - -/*********************************************************************** -* NAME -* -* ios_freeze_node - freeze current subproblem -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_freeze_node(glp_tree *tree); -* -* DESCRIPTION -* -* The routine ios_freeze_node freezes the current subproblem. */ - -void ios_freeze_node(glp_tree *tree) -{ glp_prob *mip = tree->mip; - int m = mip->m; - int n = mip->n; - IOSNPD *node; - /* obtain pointer to the current subproblem */ - node = tree->curr; - xassert(node != NULL); - if (node->up == NULL) - { /* freeze the root subproblem */ - int k; - xassert(node->p == 1); - xassert(tree->root_m == 0); - xassert(tree->root_type == NULL); - xassert(tree->root_lb == NULL); - xassert(tree->root_ub == NULL); - xassert(tree->root_stat == NULL); - tree->root_m = m; - tree->root_type = xcalloc(1+m+n, sizeof(char)); - tree->root_lb = xcalloc(1+m+n, sizeof(double)); - tree->root_ub = xcalloc(1+m+n, sizeof(double)); - tree->root_stat = xcalloc(1+m+n, sizeof(char)); - for (k = 1; k <= m+n; k++) - { if (k <= m) - { GLPROW *row = mip->row[k]; - tree->root_type[k] = (char)row->type; - tree->root_lb[k] = row->lb; - tree->root_ub[k] = row->ub; - tree->root_stat[k] = (char)row->stat; - } - else - { GLPCOL *col = mip->col[k-m]; - tree->root_type[k] = (char)col->type; - tree->root_lb[k] = col->lb; - tree->root_ub[k] = col->ub; - tree->root_stat[k] = (char)col->stat; - } - } - } - else - { /* freeze non-root subproblem */ - int root_m = tree->root_m; - int pred_m = tree->pred_m; - int i, j, k; - xassert(pred_m <= m); - /* build change lists for rows and columns which exist in the - parent subproblem */ - xassert(node->b_ptr == NULL); - xassert(node->s_ptr == NULL); - for (k = 1; k <= pred_m + n; k++) - { int pred_type, pred_stat, type, stat; - double pred_lb, pred_ub, lb, ub; - /* determine attributes in the parent subproblem */ - pred_type = tree->pred_type[k]; - pred_lb = tree->pred_lb[k]; - pred_ub = tree->pred_ub[k]; - pred_stat = tree->pred_stat[k]; - /* determine attributes in the current subproblem */ - if (k <= pred_m) - { GLPROW *row = mip->row[k]; - type = row->type; - lb = row->lb; - ub = row->ub; - stat = row->stat; - } - else - { GLPCOL *col = mip->col[k - pred_m]; - type = col->type; - lb = col->lb; - ub = col->ub; - stat = col->stat; - } - /* save type and bounds of a row/column, if changed */ - if (!(pred_type == type && pred_lb == lb && pred_ub == ub)) - { IOSBND *b; - b = dmp_get_atom(tree->pool, sizeof(IOSBND)); - b->k = k; - b->type = (unsigned char)type; - b->lb = lb; - b->ub = ub; - b->next = node->b_ptr; - node->b_ptr = b; - } - /* save status of a row/column, if changed */ - if (pred_stat != stat) - { IOSTAT *s; - s = dmp_get_atom(tree->pool, sizeof(IOSTAT)); - s->k = k; - s->stat = (unsigned char)stat; - s->next = node->s_ptr; - node->s_ptr = s; - } - } - /* save new rows added to the current subproblem */ - xassert(node->r_ptr == NULL); - if (pred_m < m) - { int i, len, *ind; - double *val; - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - for (i = m; i > pred_m; i--) - { GLPROW *row = mip->row[i]; - IOSROW *r; - const char *name; - r = dmp_get_atom(tree->pool, sizeof(IOSROW)); - name = glp_get_row_name(mip, i); - if (name == NULL) - r->name = NULL; - else - { r->name = dmp_get_atom(tree->pool, strlen(name)+1); - strcpy(r->name, name); - } -#if 1 /* 20/IX-2008 */ - r->origin = row->origin; - r->klass = row->klass; -#endif - r->type = (unsigned char)row->type; - r->lb = row->lb; - r->ub = row->ub; - r->ptr = NULL; - len = glp_get_mat_row(mip, i, ind, val); - for (k = 1; k <= len; k++) - { IOSAIJ *a; - a = dmp_get_atom(tree->pool, sizeof(IOSAIJ)); - a->j = ind[k]; - a->val = val[k]; - a->next = r->ptr; - r->ptr = a; - } - r->rii = row->rii; - r->stat = (unsigned char)row->stat; - r->next = node->r_ptr; - node->r_ptr = r; - } - xfree(ind); - xfree(val); - } - /* remove all rows missing in the root subproblem */ - if (m != root_m) - { int nrs, *num; - nrs = m - root_m; - xassert(nrs > 0); - num = xcalloc(1+nrs, sizeof(int)); - for (i = 1; i <= nrs; i++) num[i] = root_m + i; - glp_del_rows(mip, nrs, num); - xfree(num); - } - m = mip->m; - /* and restore attributes of all rows and columns for the root - subproblem */ - xassert(m == root_m); - for (i = 1; i <= m; i++) - { glp_set_row_bnds(mip, i, tree->root_type[i], - tree->root_lb[i], tree->root_ub[i]); - glp_set_row_stat(mip, i, tree->root_stat[i]); - } - for (j = 1; j <= n; j++) - { glp_set_col_bnds(mip, j, tree->root_type[m+j], - tree->root_lb[m+j], tree->root_ub[m+j]); - glp_set_col_stat(mip, j, tree->root_stat[m+j]); - } -#if 1 - /* remove all edges and cliques missing in the conflict graph - for the root subproblem */ - /* (not implemented yet) */ -#endif - } - /* the current subproblem has been frozen */ - tree->curr = NULL; - return; -} - -/*********************************************************************** -* NAME -* -* ios_clone_node - clone specified subproblem -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_clone_node(glp_tree *tree, int p, int nnn, int ref[]); -* -* DESCRIPTION -* -* The routine ios_clone_node clones the specified subproblem, whose -* reference number is p, creating its nnn exact copies. Note that the -* specified subproblem must be active and must be in the frozen state -* (i.e. it must not be the current subproblem). -* -* Each clone, an exact copy of the specified subproblem, becomes a new -* active subproblem added to the end of the active list. After cloning -* the specified subproblem becomes inactive. -* -* The reference numbers of clone subproblems are stored to locations -* ref[1], ..., ref[nnn]. */ - -static int get_slot(glp_tree *tree) -{ int p; - /* if no free slots are available, increase the room */ - if (tree->avail == 0) - { int nslots = tree->nslots; - IOSLOT *save = tree->slot; - if (nslots == 0) - tree->nslots = 20; - else - { tree->nslots = nslots + nslots; - xassert(tree->nslots > nslots); - } - tree->slot = xcalloc(1+tree->nslots, sizeof(IOSLOT)); - if (save != NULL) - { memcpy(&tree->slot[1], &save[1], nslots * sizeof(IOSLOT)); - xfree(save); - } - /* push more free slots into the stack */ - for (p = tree->nslots; p > nslots; p--) - { tree->slot[p].node = NULL; - tree->slot[p].next = tree->avail; - tree->avail = p; - } - } - /* pull a free slot from the stack */ - p = tree->avail; - tree->avail = tree->slot[p].next; - xassert(tree->slot[p].node == NULL); - tree->slot[p].next = 0; - return p; -} - -static IOSNPD *new_node(glp_tree *tree, IOSNPD *parent) -{ IOSNPD *node; - int p; - /* pull a free slot for the new node */ - p = get_slot(tree); - /* create descriptor of the new subproblem */ - node = dmp_get_atom(tree->pool, sizeof(IOSNPD)); - tree->slot[p].node = node; - node->p = p; - node->up = parent; - node->level = (parent == NULL ? 0 : parent->level + 1); - node->count = 0; - node->b_ptr = NULL; - node->s_ptr = NULL; - node->r_ptr = NULL; - node->solved = 0; -#if 0 - node->own_nn = node->own_nc = 0; - node->e_ptr = NULL; -#endif -#if 1 /* 04/X-2008 */ - node->lp_obj = (parent == NULL ? (tree->mip->dir == GLP_MIN ? - -DBL_MAX : +DBL_MAX) : parent->lp_obj); -#endif - node->bound = (parent == NULL ? (tree->mip->dir == GLP_MIN ? - -DBL_MAX : +DBL_MAX) : parent->bound); - node->br_var = 0; - node->br_val = 0.0; - node->ii_cnt = 0; - node->ii_sum = 0.0; -#if 1 /* 30/XI-2009 */ - node->changed = 0; -#endif - if (tree->parm->cb_size == 0) - node->data = NULL; - else - { node->data = dmp_get_atom(tree->pool, tree->parm->cb_size); - memset(node->data, 0, tree->parm->cb_size); - } - node->temp = NULL; - node->prev = tree->tail; - node->next = NULL; - /* add the new subproblem to the end of the active list */ - if (tree->head == NULL) - tree->head = node; - else - tree->tail->next = node; - tree->tail = node; - tree->a_cnt++; - tree->n_cnt++; - tree->t_cnt++; - /* increase the number of child subproblems */ - if (parent == NULL) - xassert(p == 1); - else - parent->count++; - return node; -} - -void ios_clone_node(glp_tree *tree, int p, int nnn, int ref[]) -{ IOSNPD *node; - int k; - /* obtain pointer to the subproblem to be cloned */ - xassert(1 <= p && p <= tree->nslots); - node = tree->slot[p].node; - xassert(node != NULL); - /* the specified subproblem must be active */ - xassert(node->count == 0); - /* and must be in the frozen state */ - xassert(tree->curr != node); - /* remove the specified subproblem from the active list, because - it becomes inactive */ - if (node->prev == NULL) - tree->head = node->next; - else - node->prev->next = node->next; - if (node->next == NULL) - tree->tail = node->prev; - else - node->next->prev = node->prev; - node->prev = node->next = NULL; - tree->a_cnt--; - /* create clone subproblems */ - xassert(nnn > 0); - for (k = 1; k <= nnn; k++) - ref[k] = new_node(tree, node)->p; - return; -} - -/*********************************************************************** -* NAME -* -* ios_delete_node - delete specified subproblem -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_delete_node(glp_tree *tree, int p); -* -* DESCRIPTION -* -* The routine ios_delete_node deletes the specified subproblem, whose -* reference number is p. The subproblem must be active and must be in -* the frozen state (i.e. it must not be the current subproblem). -* -* Note that deletion is performed recursively, i.e. if a subproblem to -* be deleted is the only child of its parent, the parent subproblem is -* also deleted, etc. */ - -void ios_delete_node(glp_tree *tree, int p) -{ IOSNPD *node, *temp; - /* obtain pointer to the subproblem to be deleted */ - xassert(1 <= p && p <= tree->nslots); - node = tree->slot[p].node; - xassert(node != NULL); - /* the specified subproblem must be active */ - xassert(node->count == 0); - /* and must be in the frozen state */ - xassert(tree->curr != node); - /* remove the specified subproblem from the active list, because - it is gone from the tree */ - if (node->prev == NULL) - tree->head = node->next; - else - node->prev->next = node->next; - if (node->next == NULL) - tree->tail = node->prev; - else - node->next->prev = node->prev; - node->prev = node->next = NULL; - tree->a_cnt--; -loop: /* recursive deletion starts here */ - /* delete the bound change list */ - { IOSBND *b; - while (node->b_ptr != NULL) - { b = node->b_ptr; - node->b_ptr = b->next; - dmp_free_atom(tree->pool, b, sizeof(IOSBND)); - } - } - /* delete the status change list */ - { IOSTAT *s; - while (node->s_ptr != NULL) - { s = node->s_ptr; - node->s_ptr = s->next; - dmp_free_atom(tree->pool, s, sizeof(IOSTAT)); - } - } - /* delete the row addition list */ - while (node->r_ptr != NULL) - { IOSROW *r; - r = node->r_ptr; - if (r->name != NULL) - dmp_free_atom(tree->pool, r->name, strlen(r->name)+1); - while (r->ptr != NULL) - { IOSAIJ *a; - a = r->ptr; - r->ptr = a->next; - dmp_free_atom(tree->pool, a, sizeof(IOSAIJ)); - } - node->r_ptr = r->next; - dmp_free_atom(tree->pool, r, sizeof(IOSROW)); - } -#if 0 - /* delete the edge addition list */ - /* delete the clique addition list */ - /* (not implemented yet) */ - xassert(node->own_nn == 0); - xassert(node->own_nc == 0); - xassert(node->e_ptr == NULL); -#endif - /* free application-specific data */ - if (tree->parm->cb_size == 0) - xassert(node->data == NULL); - else - dmp_free_atom(tree->pool, node->data, tree->parm->cb_size); - /* free the corresponding node slot */ - p = node->p; - xassert(tree->slot[p].node == node); - tree->slot[p].node = NULL; - tree->slot[p].next = tree->avail; - tree->avail = p; - /* save pointer to the parent subproblem */ - temp = node->up; - /* delete the subproblem descriptor */ - dmp_free_atom(tree->pool, node, sizeof(IOSNPD)); - tree->n_cnt--; - /* take pointer to the parent subproblem */ - node = temp; - if (node != NULL) - { /* the parent subproblem exists; decrease the number of its - child subproblems */ - xassert(node->count > 0); - node->count--; - /* if now the parent subproblem has no childs, it also must be - deleted */ - if (node->count == 0) goto loop; - } - return; -} - -/*********************************************************************** -* NAME -* -* ios_delete_tree - delete branch-and-bound tree -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_delete_tree(glp_tree *tree); -* -* DESCRIPTION -* -* The routine ios_delete_tree deletes the branch-and-bound tree, which -* the parameter tree points to, and frees all the memory allocated to -* this program object. -* -* On exit components of the problem object are restored to correspond -* to the original MIP passed to the routine ios_create_tree. */ - -void ios_delete_tree(glp_tree *tree) -{ glp_prob *mip = tree->mip; - int i, j; - int m = mip->m; - int n = mip->n; - xassert(mip->tree == tree); - /* remove all additional rows */ - if (m != tree->orig_m) - { int nrs, *num; - nrs = m - tree->orig_m; - xassert(nrs > 0); - num = xcalloc(1+nrs, sizeof(int)); - for (i = 1; i <= nrs; i++) num[i] = tree->orig_m + i; - glp_del_rows(mip, nrs, num); - xfree(num); - } - m = tree->orig_m; - /* restore original attributes of rows and columns */ - xassert(m == tree->orig_m); - xassert(n == tree->n); - for (i = 1; i <= m; i++) - { glp_set_row_bnds(mip, i, tree->orig_type[i], - tree->orig_lb[i], tree->orig_ub[i]); - glp_set_row_stat(mip, i, tree->orig_stat[i]); - mip->row[i]->prim = tree->orig_prim[i]; - mip->row[i]->dual = tree->orig_dual[i]; - } - for (j = 1; j <= n; j++) - { glp_set_col_bnds(mip, j, tree->orig_type[m+j], - tree->orig_lb[m+j], tree->orig_ub[m+j]); - glp_set_col_stat(mip, j, tree->orig_stat[m+j]); - mip->col[j]->prim = tree->orig_prim[m+j]; - mip->col[j]->dual = tree->orig_dual[m+j]; - } - mip->pbs_stat = mip->dbs_stat = GLP_FEAS; - mip->obj_val = tree->orig_obj; - /* delete the branch-and-bound tree */ - xassert(tree->local != NULL); - ios_delete_pool(tree, tree->local); - dmp_delete_pool(tree->pool); - xfree(tree->orig_type); - xfree(tree->orig_lb); - xfree(tree->orig_ub); - xfree(tree->orig_stat); - xfree(tree->orig_prim); - xfree(tree->orig_dual); - xfree(tree->slot); - if (tree->root_type != NULL) xfree(tree->root_type); - if (tree->root_lb != NULL) xfree(tree->root_lb); - if (tree->root_ub != NULL) xfree(tree->root_ub); - if (tree->root_stat != NULL) xfree(tree->root_stat); - xfree(tree->non_int); -#if 0 - xfree(tree->n_ref); - xfree(tree->c_ref); - xfree(tree->j_ref); -#endif - if (tree->pcost != NULL) ios_pcost_free(tree); - xfree(tree->iwrk); - xfree(tree->dwrk); -#if 0 - scg_delete_graph(tree->g); -#endif - if (tree->pred_type != NULL) xfree(tree->pred_type); - if (tree->pred_lb != NULL) xfree(tree->pred_lb); - if (tree->pred_ub != NULL) xfree(tree->pred_ub); - if (tree->pred_stat != NULL) xfree(tree->pred_stat); -#if 0 - xassert(tree->cut_gen == NULL); -#endif - xassert(tree->mir_gen == NULL); - xassert(tree->clq_gen == NULL); - xfree(tree); - mip->tree = NULL; - return; -} - -/*********************************************************************** -* NAME -* -* ios_eval_degrad - estimate obj. degrad. for down- and up-branches -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_eval_degrad(glp_tree *tree, int j, double *dn, double *up); -* -* DESCRIPTION -* -* Given optimal basis to LP relaxation of the current subproblem the -* routine ios_eval_degrad performs the dual ratio test to compute the -* objective values in the adjacent basis for down- and up-branches, -* which are stored in locations *dn and *up, assuming that x[j] is a -* variable chosen to branch upon. */ - -void ios_eval_degrad(glp_tree *tree, int j, double *dn, double *up) -{ glp_prob *mip = tree->mip; - int m = mip->m, n = mip->n; - int len, kase, k, t, stat; - double alfa, beta, gamma, delta, dz; - int *ind = tree->iwrk; - double *val = tree->dwrk; - /* current basis must be optimal */ - xassert(glp_get_status(mip) == GLP_OPT); - /* basis factorization must exist */ - xassert(glp_bf_exists(mip)); - /* obtain (fractional) value of x[j] in optimal basic solution - to LP relaxation of the current subproblem */ - xassert(1 <= j && j <= n); - beta = mip->col[j]->prim; - /* since the value of x[j] is fractional, it is basic; compute - corresponding row of the simplex table */ - len = lpx_eval_tab_row(mip, m+j, ind, val); - /* kase < 0 means down-branch; kase > 0 means up-branch */ - for (kase = -1; kase <= +1; kase += 2) - { /* for down-branch we introduce new upper bound floor(beta) - for x[j]; similarly, for up-branch we introduce new lower - bound ceil(beta) for x[j]; in the current basis this new - upper/lower bound is violated, so in the adjacent basis - x[j] will leave the basis and go to its new upper/lower - bound; we need to know which non-basic variable x[k] should - enter the basis to keep dual feasibility */ -#if 0 /* 23/XI-2009 */ - k = lpx_dual_ratio_test(mip, len, ind, val, kase, 1e-7); -#else - k = lpx_dual_ratio_test(mip, len, ind, val, kase, 1e-9); -#endif - /* if no variable has been chosen, current basis being primal - infeasible due to the new upper/lower bound of x[j] is dual - unbounded, therefore, LP relaxation to corresponding branch - has no primal feasible solution */ - if (k == 0) - { if (mip->dir == GLP_MIN) - { if (kase < 0) - *dn = +DBL_MAX; - else - *up = +DBL_MAX; - } - else if (mip->dir == GLP_MAX) - { if (kase < 0) - *dn = -DBL_MAX; - else - *up = -DBL_MAX; - } - else - xassert(mip != mip); - continue; - } - xassert(1 <= k && k <= m+n); - /* row of the simplex table corresponding to specified basic - variable x[j] is the following: - x[j] = ... + alfa * x[k] + ... ; - we need to know influence coefficient, alfa, at non-basic - variable x[k] chosen with the dual ratio test */ - for (t = 1; t <= len; t++) - if (ind[t] == k) break; - xassert(1 <= t && t <= len); - alfa = val[t]; - /* determine status and reduced cost of variable x[k] */ - if (k <= m) - { stat = mip->row[k]->stat; - gamma = mip->row[k]->dual; - } - else - { stat = mip->col[k-m]->stat; - gamma = mip->col[k-m]->dual; - } - /* x[k] cannot be basic or fixed non-basic */ - xassert(stat == GLP_NL || stat == GLP_NU || stat == GLP_NF); - /* if the current basis is dual degenerative, some reduced - costs, which are close to zero, may have wrong sign due to - round-off errors, so correct the sign of gamma */ - if (mip->dir == GLP_MIN) - { if (stat == GLP_NL && gamma < 0.0 || - stat == GLP_NU && gamma > 0.0 || - stat == GLP_NF) gamma = 0.0; - } - else if (mip->dir == GLP_MAX) - { if (stat == GLP_NL && gamma > 0.0 || - stat == GLP_NU && gamma < 0.0 || - stat == GLP_NF) gamma = 0.0; - } - else - xassert(mip != mip); - /* determine the change of x[j] in the adjacent basis: - delta x[j] = new x[j] - old x[j] */ - delta = (kase < 0 ? floor(beta) : ceil(beta)) - beta; - /* compute the change of x[k] in the adjacent basis: - delta x[k] = new x[k] - old x[k] = delta x[j] / alfa */ - delta /= alfa; - /* compute the change of the objective in the adjacent basis: - delta z = new z - old z = gamma * delta x[k] */ - dz = gamma * delta; - if (mip->dir == GLP_MIN) - xassert(dz >= 0.0); - else if (mip->dir == GLP_MAX) - xassert(dz <= 0.0); - else - xassert(mip != mip); - /* compute the new objective value in the adjacent basis: - new z = old z + delta z */ - if (kase < 0) - *dn = mip->obj_val + dz; - else - *up = mip->obj_val + dz; - } - /*xprintf("obj = %g; dn = %g; up = %g\n", - mip->obj_val, *dn, *up);*/ - return; -} - -/*********************************************************************** -* NAME -* -* ios_round_bound - improve local bound by rounding -* -* SYNOPSIS -* -* #include "glpios.h" -* double ios_round_bound(glp_tree *tree, double bound); -* -* RETURNS -* -* For the given local bound for any integer feasible solution to the -* current subproblem the routine ios_round_bound returns an improved -* local bound for the same integer feasible solution. -* -* BACKGROUND -* -* Let the current subproblem has the following objective function: -* -* z = sum c[j] * x[j] + s >= b, (1) -* j in J -* -* where J = {j: c[j] is non-zero and integer, x[j] is integer}, s is -* the sum of terms corresponding to fixed variables, b is an initial -* local bound (minimization). -* -* From (1) it follows that: -* -* d * sum (c[j] / d) * x[j] + s >= b, (2) -* j in J -* -* or, equivalently, -* -* sum (c[j] / d) * x[j] >= (b - s) / d = h, (3) -* j in J -* -* where d = gcd(c[j]). Since the left-hand side of (3) is integer, -* h = (b - s) / d can be rounded up to the nearest integer: -* -* h' = ceil(h) = (b' - s) / d, (4) -* -* that gives an rounded, improved local bound: -* -* b' = d * h' + s. (5) -* -* In case of maximization '>=' in (1) should be replaced by '<=' that -* leads to the following formula: -* -* h' = floor(h) = (b' - s) / d, (6) -* -* which should used in the same way as (4). -* -* NOTE: If b is a valid local bound for a child of the current -* subproblem, b' is also valid for that child subproblem. */ - -double ios_round_bound(glp_tree *tree, double bound) -{ glp_prob *mip = tree->mip; - int n = mip->n; - int d, j, nn, *c = tree->iwrk; - double s, h; - /* determine c[j] and compute s */ - nn = 0, s = mip->c0, d = 0; - for (j = 1; j <= n; j++) - { GLPCOL *col = mip->col[j]; - if (col->coef == 0.0) continue; - if (col->type == GLP_FX) - { /* fixed variable */ - s += col->coef * col->prim; - } - else - { /* non-fixed variable */ - if (col->kind != GLP_IV) goto skip; - if (col->coef != floor(col->coef)) goto skip; - if (fabs(col->coef) <= (double)INT_MAX) - c[++nn] = (int)fabs(col->coef); - else - d = 1; - } - } - /* compute d = gcd(c[1],...c[nn]) */ - if (d == 0) - { if (nn == 0) goto skip; - d = gcdn(nn, c); - } - xassert(d > 0); - /* compute new local bound */ - if (mip->dir == GLP_MIN) - { if (bound != +DBL_MAX) - { h = (bound - s) / (double)d; - if (h >= floor(h) + 0.001) - { /* round up */ - h = ceil(h); - /*xprintf("d = %d; old = %g; ", d, bound);*/ - bound = (double)d * h + s; - /*xprintf("new = %g\n", bound);*/ - } - } - } - else if (mip->dir == GLP_MAX) - { if (bound != -DBL_MAX) - { h = (bound - s) / (double)d; - if (h <= ceil(h) - 0.001) - { /* round down */ - h = floor(h); - bound = (double)d * h + s; - } - } - } - else - xassert(mip != mip); -skip: return bound; -} - -/*********************************************************************** -* NAME -* -* ios_is_hopeful - check if subproblem is hopeful -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_is_hopeful(glp_tree *tree, double bound); -* -* DESCRIPTION -* -* Given the local bound of a subproblem the routine ios_is_hopeful -* checks if the subproblem can have an integer optimal solution which -* is better than the best one currently known. -* -* RETURNS -* -* If the subproblem can have a better integer optimal solution, the -* routine returns non-zero; otherwise, if the corresponding branch can -* be pruned, the routine returns zero. */ - -int ios_is_hopeful(glp_tree *tree, double bound) -{ glp_prob *mip = tree->mip; - int ret = 1; - double eps; - if (mip->mip_stat == GLP_FEAS) - { eps = tree->parm->tol_obj * (1.0 + fabs(mip->mip_obj)); - switch (mip->dir) - { case GLP_MIN: - if (bound >= mip->mip_obj - eps) ret = 0; - break; - case GLP_MAX: - if (bound <= mip->mip_obj + eps) ret = 0; - break; - default: - xassert(mip != mip); - } - } - else - { switch (mip->dir) - { case GLP_MIN: - if (bound == +DBL_MAX) ret = 0; - break; - case GLP_MAX: - if (bound == -DBL_MAX) ret = 0; - break; - default: - xassert(mip != mip); - } - } - return ret; -} - -/*********************************************************************** -* NAME -* -* ios_best_node - find active node with best local bound -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_best_node(glp_tree *tree); -* -* DESCRIPTION -* -* The routine ios_best_node finds an active node whose local bound is -* best among other active nodes. -* -* It is understood that the integer optimal solution of the original -* mip problem cannot be better than the best bound, so the best bound -* is an lower (minimization) or upper (maximization) global bound for -* the original problem. -* -* RETURNS -* -* The routine ios_best_node returns the subproblem reference number -* for the best node. However, if the tree is empty, it returns zero. */ - -int ios_best_node(glp_tree *tree) -{ IOSNPD *node, *best = NULL; - switch (tree->mip->dir) - { case GLP_MIN: - /* minimization */ - for (node = tree->head; node != NULL; node = node->next) - if (best == NULL || best->bound > node->bound) - best = node; - break; - case GLP_MAX: - /* maximization */ - for (node = tree->head; node != NULL; node = node->next) - if (best == NULL || best->bound < node->bound) - best = node; - break; - default: - xassert(tree != tree); - } - return best == NULL ? 0 : best->p; -} - -/*********************************************************************** -* NAME -* -* ios_relative_gap - compute relative mip gap -* -* SYNOPSIS -* -* #include "glpios.h" -* double ios_relative_gap(glp_tree *tree); -* -* DESCRIPTION -* -* The routine ios_relative_gap computes the relative mip gap using the -* formula: -* -* gap = |best_mip - best_bnd| / (|best_mip| + DBL_EPSILON), -* -* where best_mip is the best integer feasible solution found so far, -* best_bnd is the best (global) bound. If no integer feasible solution -* has been found yet, rel_gap is set to DBL_MAX. -* -* RETURNS -* -* The routine ios_relative_gap returns the relative mip gap. */ - -double ios_relative_gap(glp_tree *tree) -{ glp_prob *mip = tree->mip; - int p; - double best_mip, best_bnd, gap; - if (mip->mip_stat == GLP_FEAS) - { best_mip = mip->mip_obj; - p = ios_best_node(tree); - if (p == 0) - { /* the tree is empty */ - gap = 0.0; - } - else - { best_bnd = tree->slot[p].node->bound; - gap = fabs(best_mip - best_bnd) / (fabs(best_mip) + - DBL_EPSILON); - } - } - else - { /* no integer feasible solution has been found yet */ - gap = DBL_MAX; - } - return gap; -} - -/*********************************************************************** -* NAME -* -* ios_solve_node - solve LP relaxation of current subproblem -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_solve_node(glp_tree *tree); -* -* DESCRIPTION -* -* The routine ios_solve_node re-optimizes LP relaxation of the current -* subproblem using the dual simplex method. -* -* RETURNS -* -* The routine returns the code which is reported by glp_simplex. */ - -int ios_solve_node(glp_tree *tree) -{ glp_prob *mip = tree->mip; - glp_smcp parm; - int ret; - /* the current subproblem must exist */ - xassert(tree->curr != NULL); - /* set some control parameters */ - glp_init_smcp(&parm); - switch (tree->parm->msg_lev) - { case GLP_MSG_OFF: - parm.msg_lev = GLP_MSG_OFF; break; - case GLP_MSG_ERR: - parm.msg_lev = GLP_MSG_ERR; break; - case GLP_MSG_ON: - case GLP_MSG_ALL: - parm.msg_lev = GLP_MSG_ON; break; - case GLP_MSG_DBG: - parm.msg_lev = GLP_MSG_ALL; break; - default: - xassert(tree != tree); - } - parm.meth = GLP_DUALP; -#if 1 /* 16/III-2016 */ - if (tree->parm->flip) - parm.r_test = GLP_RT_FLIP; -#endif - /* respect time limit */ - if (tree->parm->tm_lim < INT_MAX) - parm.tm_lim = tree->parm->tm_lim - (glp_time() - tree->tm_beg); - if (parm.tm_lim < 0) - parm.tm_lim = 0; - if (tree->parm->msg_lev < GLP_MSG_DBG) - parm.out_dly = tree->parm->out_dly; - else - parm.out_dly = 0; - /* if the incumbent objective value is already known, use it to - prematurely terminate the dual simplex search */ - if (mip->mip_stat == GLP_FEAS) - { switch (tree->mip->dir) - { case GLP_MIN: - parm.obj_ul = mip->mip_obj; - break; - case GLP_MAX: - parm.obj_ll = mip->mip_obj; - break; - default: - xassert(mip != mip); - } - } - /* try to solve/re-optimize the LP relaxation */ - ret = glp_simplex(mip, &parm); -#if 1 /* 21/II-2016 by Chris */ - if (ret == GLP_EFAIL) - { /* retry with a new basis */ - glp_adv_basis(mip, 0); - ret = glp_simplex(mip, &parm); - } -#endif - tree->curr->solved++; -#if 0 - xprintf("ret = %d; status = %d; pbs = %d; dbs = %d; some = %d\n", - ret, glp_get_status(mip), mip->pbs_stat, mip->dbs_stat, - mip->some); - lpx_print_sol(mip, "sol"); -#endif - return ret; -} - -/**********************************************************************/ - -#ifdef NEW_LOCAL /* 02/II-2018 */ -IOSPOOL *ios_create_pool(glp_tree *tree) -{ /* create cut pool */ - IOSPOOL *pool; - pool = glp_create_prob(); - glp_add_cols(pool, tree->mip->n); - return pool; -} -#else -IOSPOOL *ios_create_pool(glp_tree *tree) -{ /* create cut pool */ - IOSPOOL *pool; -#if 0 - pool = dmp_get_atom(tree->pool, sizeof(IOSPOOL)); -#else - xassert(tree == tree); - pool = xmalloc(sizeof(IOSPOOL)); -#endif - pool->size = 0; - pool->head = pool->tail = NULL; - pool->ord = 0, pool->curr = NULL; - return pool; -} -#endif - -#ifdef NEW_LOCAL /* 02/II-2018 */ -int ios_add_row(glp_tree *tree, IOSPOOL *pool, - const char *name, int klass, int flags, int len, const int ind[], - const double val[], int type, double rhs) -{ /* add row (constraint) to the cut pool */ - int i; - i = glp_add_rows(pool, 1); - glp_set_row_name(pool, i, name); - pool->row[i]->klass = klass; - xassert(flags == 0); - glp_set_mat_row(pool, i, len, ind, val); - glp_set_row_bnds(pool, i, type, rhs, rhs); - return i; -} -#else -int ios_add_row(glp_tree *tree, IOSPOOL *pool, - const char *name, int klass, int flags, int len, const int ind[], - const double val[], int type, double rhs) -{ /* add row (constraint) to the cut pool */ - IOSCUT *cut; - IOSAIJ *aij; - int k; - xassert(pool != NULL); - cut = dmp_get_atom(tree->pool, sizeof(IOSCUT)); - if (name == NULL || name[0] == '\0') - cut->name = NULL; - else - { for (k = 0; name[k] != '\0'; k++) - { if (k == 256) - xerror("glp_ios_add_row: cut name too long\n"); - if (iscntrl((unsigned char)name[k])) - xerror("glp_ios_add_row: cut name contains invalid chara" - "cter(s)\n"); - } - cut->name = dmp_get_atom(tree->pool, strlen(name)+1); - strcpy(cut->name, name); - } - if (!(0 <= klass && klass <= 255)) - xerror("glp_ios_add_row: klass = %d; invalid cut class\n", - klass); - cut->klass = (unsigned char)klass; - if (flags != 0) - xerror("glp_ios_add_row: flags = %d; invalid cut flags\n", - flags); - cut->ptr = NULL; - if (!(0 <= len && len <= tree->n)) - xerror("glp_ios_add_row: len = %d; invalid cut length\n", - len); - for (k = 1; k <= len; k++) - { aij = dmp_get_atom(tree->pool, sizeof(IOSAIJ)); - if (!(1 <= ind[k] && ind[k] <= tree->n)) - xerror("glp_ios_add_row: ind[%d] = %d; column index out of " - "range\n", k, ind[k]); - aij->j = ind[k]; - aij->val = val[k]; - aij->next = cut->ptr; - cut->ptr = aij; - } - if (!(type == GLP_LO || type == GLP_UP || type == GLP_FX)) - xerror("glp_ios_add_row: type = %d; invalid cut type\n", - type); - cut->type = (unsigned char)type; - cut->rhs = rhs; - cut->prev = pool->tail; - cut->next = NULL; - if (cut->prev == NULL) - pool->head = cut; - else - cut->prev->next = cut; - pool->tail = cut; - pool->size++; - return pool->size; -} -#endif - -#ifdef NEW_LOCAL /* 02/II-2018 */ -IOSCUT *ios_find_row(IOSPOOL *pool, int i) -{ /* find row (constraint) in the cut pool */ - xassert(0); -} -#else -IOSCUT *ios_find_row(IOSPOOL *pool, int i) -{ /* find row (constraint) in the cut pool */ - /* (smart linear search) */ - xassert(pool != NULL); - xassert(1 <= i && i <= pool->size); - if (pool->ord == 0) - { xassert(pool->curr == NULL); - pool->ord = 1; - pool->curr = pool->head; - } - xassert(pool->curr != NULL); - if (i < pool->ord) - { if (i < pool->ord - i) - { pool->ord = 1; - pool->curr = pool->head; - while (pool->ord != i) - { pool->ord++; - xassert(pool->curr != NULL); - pool->curr = pool->curr->next; - } - } - else - { while (pool->ord != i) - { pool->ord--; - xassert(pool->curr != NULL); - pool->curr = pool->curr->prev; - } - } - } - else if (i > pool->ord) - { if (i - pool->ord < pool->size - i) - { while (pool->ord != i) - { pool->ord++; - xassert(pool->curr != NULL); - pool->curr = pool->curr->next; - } - } - else - { pool->ord = pool->size; - pool->curr = pool->tail; - while (pool->ord != i) - { pool->ord--; - xassert(pool->curr != NULL); - pool->curr = pool->curr->prev; - } - } - } - xassert(pool->ord == i); - xassert(pool->curr != NULL); - return pool->curr; -} -#endif - -#ifdef NEW_LOCAL /* 02/II-2018 */ -void ios_del_row(glp_tree *tree, IOSPOOL *pool, int i) -{ /* remove row (constraint) from the cut pool */ - xassert(0); -} -#else -void ios_del_row(glp_tree *tree, IOSPOOL *pool, int i) -{ /* remove row (constraint) from the cut pool */ - IOSCUT *cut; - IOSAIJ *aij; - xassert(pool != NULL); - if (!(1 <= i && i <= pool->size)) - xerror("glp_ios_del_row: i = %d; cut number out of range\n", - i); - cut = ios_find_row(pool, i); - xassert(pool->curr == cut); - if (cut->next != NULL) - pool->curr = cut->next; - else if (cut->prev != NULL) - pool->ord--, pool->curr = cut->prev; - else - pool->ord = 0, pool->curr = NULL; - if (cut->name != NULL) - dmp_free_atom(tree->pool, cut->name, strlen(cut->name)+1); - if (cut->prev == NULL) - { xassert(pool->head == cut); - pool->head = cut->next; - } - else - { xassert(cut->prev->next == cut); - cut->prev->next = cut->next; - } - if (cut->next == NULL) - { xassert(pool->tail == cut); - pool->tail = cut->prev; - } - else - { xassert(cut->next->prev == cut); - cut->next->prev = cut->prev; - } - while (cut->ptr != NULL) - { aij = cut->ptr; - cut->ptr = aij->next; - dmp_free_atom(tree->pool, aij, sizeof(IOSAIJ)); - } - dmp_free_atom(tree->pool, cut, sizeof(IOSCUT)); - pool->size--; - return; -} -#endif - -#ifdef NEW_LOCAL /* 02/II-2018 */ -void ios_clear_pool(glp_tree *tree, IOSPOOL *pool) -{ /* remove all rows (constraints) from the cut pool */ - if (pool->m > 0) - { int i, *num; - num = talloc(1+pool->m, int); - for (i = 1; i <= pool->m; i++) - num[i] = i; - glp_del_rows(pool, pool->m, num); - tfree(num); - } - return; -} -#else -void ios_clear_pool(glp_tree *tree, IOSPOOL *pool) -{ /* remove all rows (constraints) from the cut pool */ - xassert(pool != NULL); - while (pool->head != NULL) - { IOSCUT *cut = pool->head; - pool->head = cut->next; - if (cut->name != NULL) - dmp_free_atom(tree->pool, cut->name, strlen(cut->name)+1); - while (cut->ptr != NULL) - { IOSAIJ *aij = cut->ptr; - cut->ptr = aij->next; - dmp_free_atom(tree->pool, aij, sizeof(IOSAIJ)); - } - dmp_free_atom(tree->pool, cut, sizeof(IOSCUT)); - } - pool->size = 0; - pool->head = pool->tail = NULL; - pool->ord = 0, pool->curr = NULL; - return; -} -#endif - -#ifdef NEW_LOCAL /* 02/II-2018 */ -void ios_delete_pool(glp_tree *tree, IOSPOOL *pool) -{ /* delete cut pool */ - xassert(pool != NULL); - glp_delete_prob(pool); - return; -} -#else -void ios_delete_pool(glp_tree *tree, IOSPOOL *pool) -{ /* delete cut pool */ - xassert(pool != NULL); - ios_clear_pool(tree, pool); - xfree(pool); - return; -} -#endif - -#if 1 /* 11/VII-2013 */ -#include "npp.h" - -void ios_process_sol(glp_tree *T) -{ /* process integer feasible solution just found */ - if (T->npp != NULL) - { /* postprocess solution from transformed mip */ - npp_postprocess(T->npp, T->mip); - /* store solution to problem passed to glp_intopt */ - npp_unload_sol(T->npp, T->P); - } - xassert(T->P != NULL); - /* save solution to text file, if requested */ - if (T->save_sol != NULL) - { char *fn, *mark; - fn = talloc(strlen(T->save_sol) + 50, char); - mark = strrchr(T->save_sol, '*'); - if (mark == NULL) - strcpy(fn, T->save_sol); - else - { memcpy(fn, T->save_sol, mark - T->save_sol); - fn[mark - T->save_sol] = '\0'; - sprintf(fn + strlen(fn), "%03d", ++(T->save_cnt)); - strcat(fn, &mark[1]); - } - glp_write_mip(T->P, fn); - tfree(fn); - } - return; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios02.c b/code/3rd_glpk/draft/glpios02.c deleted file mode 100644 index a73458aa..00000000 --- a/code/3rd_glpk/draft/glpios02.c +++ /dev/null @@ -1,826 +0,0 @@ -/* glpios02.c (preprocess current subproblem) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/*********************************************************************** -* prepare_row_info - prepare row info to determine implied bounds -* -* Given a row (linear form) -* -* n -* sum a[j] * x[j] (1) -* j=1 -* -* and bounds of columns (variables) -* -* l[j] <= x[j] <= u[j] (2) -* -* this routine computes f_min, j_min, f_max, j_max needed to determine -* implied bounds. -* -* ALGORITHM -* -* Let J+ = {j : a[j] > 0} and J- = {j : a[j] < 0}. -* -* Parameters f_min and j_min are computed as follows: -* -* 1) if there is no x[k] such that k in J+ and l[k] = -inf or k in J- -* and u[k] = +inf, then -* -* f_min := sum a[j] * l[j] + sum a[j] * u[j] -* j in J+ j in J- -* (3) -* j_min := 0 -* -* 2) if there is exactly one x[k] such that k in J+ and l[k] = -inf -* or k in J- and u[k] = +inf, then -* -* f_min := sum a[j] * l[j] + sum a[j] * u[j] -* j in J+\{k} j in J-\{k} -* (4) -* j_min := k -* -* 3) if there are two or more x[k] such that k in J+ and l[k] = -inf -* or k in J- and u[k] = +inf, then -* -* f_min := -inf -* (5) -* j_min := 0 -* -* Parameters f_max and j_max are computed in a similar way as follows: -* -* 1) if there is no x[k] such that k in J+ and u[k] = +inf or k in J- -* and l[k] = -inf, then -* -* f_max := sum a[j] * u[j] + sum a[j] * l[j] -* j in J+ j in J- -* (6) -* j_max := 0 -* -* 2) if there is exactly one x[k] such that k in J+ and u[k] = +inf -* or k in J- and l[k] = -inf, then -* -* f_max := sum a[j] * u[j] + sum a[j] * l[j] -* j in J+\{k} j in J-\{k} -* (7) -* j_max := k -* -* 3) if there are two or more x[k] such that k in J+ and u[k] = +inf -* or k in J- and l[k] = -inf, then -* -* f_max := +inf -* (8) -* j_max := 0 */ - -struct f_info -{ int j_min, j_max; - double f_min, f_max; -}; - -static void prepare_row_info(int n, const double a[], const double l[], - const double u[], struct f_info *f) -{ int j, j_min, j_max; - double f_min, f_max; - xassert(n >= 0); - /* determine f_min and j_min */ - f_min = 0.0, j_min = 0; - for (j = 1; j <= n; j++) - { if (a[j] > 0.0) - { if (l[j] == -DBL_MAX) - { if (j_min == 0) - j_min = j; - else - { f_min = -DBL_MAX, j_min = 0; - break; - } - } - else - f_min += a[j] * l[j]; - } - else if (a[j] < 0.0) - { if (u[j] == +DBL_MAX) - { if (j_min == 0) - j_min = j; - else - { f_min = -DBL_MAX, j_min = 0; - break; - } - } - else - f_min += a[j] * u[j]; - } - else - xassert(a != a); - } - f->f_min = f_min, f->j_min = j_min; - /* determine f_max and j_max */ - f_max = 0.0, j_max = 0; - for (j = 1; j <= n; j++) - { if (a[j] > 0.0) - { if (u[j] == +DBL_MAX) - { if (j_max == 0) - j_max = j; - else - { f_max = +DBL_MAX, j_max = 0; - break; - } - } - else - f_max += a[j] * u[j]; - } - else if (a[j] < 0.0) - { if (l[j] == -DBL_MAX) - { if (j_max == 0) - j_max = j; - else - { f_max = +DBL_MAX, j_max = 0; - break; - } - } - else - f_max += a[j] * l[j]; - } - else - xassert(a != a); - } - f->f_max = f_max, f->j_max = j_max; - return; -} - -/*********************************************************************** -* row_implied_bounds - determine row implied bounds -* -* Given a row (linear form) -* -* n -* sum a[j] * x[j] -* j=1 -* -* and bounds of columns (variables) -* -* l[j] <= x[j] <= u[j] -* -* this routine determines implied bounds of the row. -* -* ALGORITHM -* -* Let J+ = {j : a[j] > 0} and J- = {j : a[j] < 0}. -* -* The implied lower bound of the row is computed as follows: -* -* L' := sum a[j] * l[j] + sum a[j] * u[j] (9) -* j in J+ j in J- -* -* and as it follows from (3), (4), and (5): -* -* L' := if j_min = 0 then f_min else -inf (10) -* -* The implied upper bound of the row is computed as follows: -* -* U' := sum a[j] * u[j] + sum a[j] * l[j] (11) -* j in J+ j in J- -* -* and as it follows from (6), (7), and (8): -* -* U' := if j_max = 0 then f_max else +inf (12) -* -* The implied bounds are stored in locations LL and UU. */ - -static void row_implied_bounds(const struct f_info *f, double *LL, - double *UU) -{ *LL = (f->j_min == 0 ? f->f_min : -DBL_MAX); - *UU = (f->j_max == 0 ? f->f_max : +DBL_MAX); - return; -} - -/*********************************************************************** -* col_implied_bounds - determine column implied bounds -* -* Given a row (constraint) -* -* n -* L <= sum a[j] * x[j] <= U (13) -* j=1 -* -* and bounds of columns (variables) -* -* l[j] <= x[j] <= u[j] -* -* this routine determines implied bounds of variable x[k]. -* -* It is assumed that if L != -inf, the lower bound of the row can be -* active, and if U != +inf, the upper bound of the row can be active. -* -* ALGORITHM -* -* From (13) it follows that -* -* L <= sum a[j] * x[j] + a[k] * x[k] <= U -* j!=k -* or -* -* L - sum a[j] * x[j] <= a[k] * x[k] <= U - sum a[j] * x[j] -* j!=k j!=k -* -* Thus, if the row lower bound L can be active, implied lower bound of -* term a[k] * x[k] can be determined as follows: -* -* ilb(a[k] * x[k]) = min(L - sum a[j] * x[j]) = -* j!=k -* (14) -* = L - max sum a[j] * x[j] -* j!=k -* -* where, as it follows from (6), (7), and (8) -* -* / f_max - a[k] * u[k], j_max = 0, a[k] > 0 -* | -* | f_max - a[k] * l[k], j_max = 0, a[k] < 0 -* max sum a[j] * x[j] = { -* j!=k | f_max, j_max = k -* | -* \ +inf, j_max != 0 -* -* and if the upper bound U can be active, implied upper bound of term -* a[k] * x[k] can be determined as follows: -* -* iub(a[k] * x[k]) = max(U - sum a[j] * x[j]) = -* j!=k -* (15) -* = U - min sum a[j] * x[j] -* j!=k -* -* where, as it follows from (3), (4), and (5) -* -* / f_min - a[k] * l[k], j_min = 0, a[k] > 0 -* | -* | f_min - a[k] * u[k], j_min = 0, a[k] < 0 -* min sum a[j] * x[j] = { -* j!=k | f_min, j_min = k -* | -* \ -inf, j_min != 0 -* -* Since -* -* ilb(a[k] * x[k]) <= a[k] * x[k] <= iub(a[k] * x[k]) -* -* implied lower and upper bounds of x[k] are determined as follows: -* -* l'[k] := if a[k] > 0 then ilb / a[k] else ulb / a[k] (16) -* -* u'[k] := if a[k] > 0 then ulb / a[k] else ilb / a[k] (17) -* -* The implied bounds are stored in locations ll and uu. */ - -static void col_implied_bounds(const struct f_info *f, int n, - const double a[], double L, double U, const double l[], - const double u[], int k, double *ll, double *uu) -{ double ilb, iub; - xassert(n >= 0); - xassert(1 <= k && k <= n); - /* determine implied lower bound of term a[k] * x[k] (14) */ - if (L == -DBL_MAX || f->f_max == +DBL_MAX) - ilb = -DBL_MAX; - else if (f->j_max == 0) - { if (a[k] > 0.0) - { xassert(u[k] != +DBL_MAX); - ilb = L - (f->f_max - a[k] * u[k]); - } - else if (a[k] < 0.0) - { xassert(l[k] != -DBL_MAX); - ilb = L - (f->f_max - a[k] * l[k]); - } - else - xassert(a != a); - } - else if (f->j_max == k) - ilb = L - f->f_max; - else - ilb = -DBL_MAX; - /* determine implied upper bound of term a[k] * x[k] (15) */ - if (U == +DBL_MAX || f->f_min == -DBL_MAX) - iub = +DBL_MAX; - else if (f->j_min == 0) - { if (a[k] > 0.0) - { xassert(l[k] != -DBL_MAX); - iub = U - (f->f_min - a[k] * l[k]); - } - else if (a[k] < 0.0) - { xassert(u[k] != +DBL_MAX); - iub = U - (f->f_min - a[k] * u[k]); - } - else - xassert(a != a); - } - else if (f->j_min == k) - iub = U - f->f_min; - else - iub = +DBL_MAX; - /* determine implied bounds of x[k] (16) and (17) */ -#if 1 - /* do not use a[k] if it has small magnitude to prevent wrong - implied bounds; for example, 1e-15 * x1 >= x2 + x3, where - x1 >= -10, x2, x3 >= 0, would lead to wrong conclusion that - x1 >= 0 */ - if (fabs(a[k]) < 1e-6) - *ll = -DBL_MAX, *uu = +DBL_MAX; else -#endif - if (a[k] > 0.0) - { *ll = (ilb == -DBL_MAX ? -DBL_MAX : ilb / a[k]); - *uu = (iub == +DBL_MAX ? +DBL_MAX : iub / a[k]); - } - else if (a[k] < 0.0) - { *ll = (iub == +DBL_MAX ? -DBL_MAX : iub / a[k]); - *uu = (ilb == -DBL_MAX ? +DBL_MAX : ilb / a[k]); - } - else - xassert(a != a); - return; -} - -/*********************************************************************** -* check_row_bounds - check and relax original row bounds -* -* Given a row (constraint) -* -* n -* L <= sum a[j] * x[j] <= U -* j=1 -* -* and bounds of columns (variables) -* -* l[j] <= x[j] <= u[j] -* -* this routine checks the original row bounds L and U for feasibility -* and redundancy. If the original lower bound L or/and upper bound U -* cannot be active due to bounds of variables, the routine remove them -* replacing by -inf or/and +inf, respectively. -* -* If no primal infeasibility is detected, the routine returns zero, -* otherwise non-zero. */ - -static int check_row_bounds(const struct f_info *f, double *L_, - double *U_) -{ int ret = 0; - double L = *L_, U = *U_, LL, UU; - /* determine implied bounds of the row */ - row_implied_bounds(f, &LL, &UU); - /* check if the original lower bound is infeasible */ - if (L != -DBL_MAX) - { double eps = 1e-3 * (1.0 + fabs(L)); - if (UU < L - eps) - { ret = 1; - goto done; - } - } - /* check if the original upper bound is infeasible */ - if (U != +DBL_MAX) - { double eps = 1e-3 * (1.0 + fabs(U)); - if (LL > U + eps) - { ret = 1; - goto done; - } - } - /* check if the original lower bound is redundant */ - if (L != -DBL_MAX) - { double eps = 1e-12 * (1.0 + fabs(L)); - if (LL > L - eps) - { /* it cannot be active, so remove it */ - *L_ = -DBL_MAX; - } - } - /* check if the original upper bound is redundant */ - if (U != +DBL_MAX) - { double eps = 1e-12 * (1.0 + fabs(U)); - if (UU < U + eps) - { /* it cannot be active, so remove it */ - *U_ = +DBL_MAX; - } - } -done: return ret; -} - -/*********************************************************************** -* check_col_bounds - check and tighten original column bounds -* -* Given a row (constraint) -* -* n -* L <= sum a[j] * x[j] <= U -* j=1 -* -* and bounds of columns (variables) -* -* l[j] <= x[j] <= u[j] -* -* for column (variable) x[j] this routine checks the original column -* bounds l[j] and u[j] for feasibility and redundancy. If the original -* lower bound l[j] or/and upper bound u[j] cannot be active due to -* bounds of the constraint and other variables, the routine tighten -* them replacing by corresponding implied bounds, if possible. -* -* NOTE: It is assumed that if L != -inf, the row lower bound can be -* active, and if U != +inf, the row upper bound can be active. -* -* The flag means that variable x[j] is required to be integer. -* -* New actual bounds for x[j] are stored in locations lj and uj. -* -* If no primal infeasibility is detected, the routine returns zero, -* otherwise non-zero. */ - -static int check_col_bounds(const struct f_info *f, int n, - const double a[], double L, double U, const double l[], - const double u[], int flag, int j, double *_lj, double *_uj) -{ int ret = 0; - double lj, uj, ll, uu; - xassert(n >= 0); - xassert(1 <= j && j <= n); - lj = l[j], uj = u[j]; - /* determine implied bounds of the column */ - col_implied_bounds(f, n, a, L, U, l, u, j, &ll, &uu); - /* if x[j] is integral, round its implied bounds */ - if (flag) - { if (ll != -DBL_MAX) - ll = (ll - floor(ll) < 1e-3 ? floor(ll) : ceil(ll)); - if (uu != +DBL_MAX) - uu = (ceil(uu) - uu < 1e-3 ? ceil(uu) : floor(uu)); - } - /* check if the original lower bound is infeasible */ - if (lj != -DBL_MAX) - { double eps = 1e-3 * (1.0 + fabs(lj)); - if (uu < lj - eps) - { ret = 1; - goto done; - } - } - /* check if the original upper bound is infeasible */ - if (uj != +DBL_MAX) - { double eps = 1e-3 * (1.0 + fabs(uj)); - if (ll > uj + eps) - { ret = 1; - goto done; - } - } - /* check if the original lower bound is redundant */ - if (ll != -DBL_MAX) - { double eps = 1e-3 * (1.0 + fabs(ll)); - if (lj < ll - eps) - { /* it cannot be active, so tighten it */ - lj = ll; - } - } - /* check if the original upper bound is redundant */ - if (uu != +DBL_MAX) - { double eps = 1e-3 * (1.0 + fabs(uu)); - if (uj > uu + eps) - { /* it cannot be active, so tighten it */ - uj = uu; - } - } - /* due to round-off errors it may happen that lj > uj (although - lj < uj + eps, since no primal infeasibility is detected), so - adjuct the new actual bounds to provide lj <= uj */ - if (!(lj == -DBL_MAX || uj == +DBL_MAX)) - { double t1 = fabs(lj), t2 = fabs(uj); - double eps = 1e-10 * (1.0 + (t1 <= t2 ? t1 : t2)); - if (lj > uj - eps) - { if (lj == l[j]) - uj = lj; - else if (uj == u[j]) - lj = uj; - else if (t1 <= t2) - uj = lj; - else - lj = uj; - } - } - *_lj = lj, *_uj = uj; -done: return ret; -} - -/*********************************************************************** -* check_efficiency - check if change in column bounds is efficient -* -* Given the original bounds of a column l and u and its new actual -* bounds l' and u' (possibly tighten by the routine check_col_bounds) -* this routine checks if the change in the column bounds is efficient -* enough. If so, the routine returns non-zero, otherwise zero. -* -* The flag means that the variable is required to be integer. */ - -static int check_efficiency(int flag, double l, double u, double ll, - double uu) -{ int eff = 0; - /* check efficiency for lower bound */ - if (l < ll) - { if (flag || l == -DBL_MAX) - eff++; - else - { double r; - if (u == +DBL_MAX) - r = 1.0 + fabs(l); - else - r = 1.0 + (u - l); - if (ll - l >= 0.25 * r) - eff++; - } - } - /* check efficiency for upper bound */ - if (u > uu) - { if (flag || u == +DBL_MAX) - eff++; - else - { double r; - if (l == -DBL_MAX) - r = 1.0 + fabs(u); - else - r = 1.0 + (u - l); - if (u - uu >= 0.25 * r) - eff++; - } - } - return eff; -} - -/*********************************************************************** -* basic_preprocessing - perform basic preprocessing -* -* This routine performs basic preprocessing of the specified MIP that -* includes relaxing some row bounds and tightening some column bounds. -* -* On entry the arrays L and U contains original row bounds, and the -* arrays l and u contains original column bounds: -* -* L[0] is the lower bound of the objective row; -* L[i], i = 1,...,m, is the lower bound of i-th row; -* U[0] is the upper bound of the objective row; -* U[i], i = 1,...,m, is the upper bound of i-th row; -* l[0] is not used; -* l[j], j = 1,...,n, is the lower bound of j-th column; -* u[0] is not used; -* u[j], j = 1,...,n, is the upper bound of j-th column. -* -* On exit the arrays L, U, l, and u contain new actual bounds of rows -* and column in the same locations. -* -* The parameters nrs and num specify an initial list of rows to be -* processed: -* -* nrs is the number of rows in the initial list, 0 <= nrs <= m+1; -* num[0] is not used; -* num[1,...,nrs] are row numbers (0 means the objective row). -* -* The parameter max_pass specifies the maximal number of times that -* each row can be processed, max_pass > 0. -* -* If no primal infeasibility is detected, the routine returns zero, -* otherwise non-zero. */ - -static int basic_preprocessing(glp_prob *mip, double L[], double U[], - double l[], double u[], int nrs, const int num[], int max_pass) -{ int m = mip->m; - int n = mip->n; - struct f_info f; - int i, j, k, len, size, ret = 0; - int *ind, *list, *mark, *pass; - double *val, *lb, *ub; - xassert(0 <= nrs && nrs <= m+1); - xassert(max_pass > 0); - /* allocate working arrays */ - ind = xcalloc(1+n, sizeof(int)); - list = xcalloc(1+m+1, sizeof(int)); - mark = xcalloc(1+m+1, sizeof(int)); - memset(&mark[0], 0, (m+1) * sizeof(int)); - pass = xcalloc(1+m+1, sizeof(int)); - memset(&pass[0], 0, (m+1) * sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - lb = xcalloc(1+n, sizeof(double)); - ub = xcalloc(1+n, sizeof(double)); - /* initialize the list of rows to be processed */ - size = 0; - for (k = 1; k <= nrs; k++) - { i = num[k]; - xassert(0 <= i && i <= m); - /* duplicate row numbers are not allowed */ - xassert(!mark[i]); - list[++size] = i, mark[i] = 1; - } - xassert(size == nrs); - /* process rows in the list until it becomes empty */ - while (size > 0) - { /* get a next row from the list */ - i = list[size--], mark[i] = 0; - /* increase the row processing count */ - pass[i]++; - /* if the row is free, skip it */ - if (L[i] == -DBL_MAX && U[i] == +DBL_MAX) continue; - /* obtain coefficients of the row */ - len = 0; - if (i == 0) - { for (j = 1; j <= n; j++) - { GLPCOL *col = mip->col[j]; - if (col->coef != 0.0) - len++, ind[len] = j, val[len] = col->coef; - } - } - else - { GLPROW *row = mip->row[i]; - GLPAIJ *aij; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - len++, ind[len] = aij->col->j, val[len] = aij->val; - } - /* determine lower and upper bounds of columns corresponding - to non-zero row coefficients */ - for (k = 1; k <= len; k++) - j = ind[k], lb[k] = l[j], ub[k] = u[j]; - /* prepare the row info to determine implied bounds */ - prepare_row_info(len, val, lb, ub, &f); - /* check and relax bounds of the row */ - if (check_row_bounds(&f, &L[i], &U[i])) - { /* the feasible region is empty */ - ret = 1; - goto done; - } - /* if the row became free, drop it */ - if (L[i] == -DBL_MAX && U[i] == +DBL_MAX) continue; - /* process columns having non-zero coefficients in the row */ - for (k = 1; k <= len; k++) - { GLPCOL *col; - int flag, eff; - double ll, uu; - /* take a next column in the row */ - j = ind[k], col = mip->col[j]; - flag = col->kind != GLP_CV; - /* check and tighten bounds of the column */ - if (check_col_bounds(&f, len, val, L[i], U[i], lb, ub, - flag, k, &ll, &uu)) - { /* the feasible region is empty */ - ret = 1; - goto done; - } - /* check if change in the column bounds is efficient */ - eff = check_efficiency(flag, l[j], u[j], ll, uu); - /* set new actual bounds of the column */ - l[j] = ll, u[j] = uu; - /* if the change is efficient, add all rows affected by the - corresponding column, to the list */ - if (eff > 0) - { GLPAIJ *aij; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - { int ii = aij->row->i; - /* if the row was processed maximal number of times, - skip it */ - if (pass[ii] >= max_pass) continue; - /* if the row is free, skip it */ - if (L[ii] == -DBL_MAX && U[ii] == +DBL_MAX) continue; - /* put the row into the list */ - if (mark[ii] == 0) - { xassert(size <= m); - list[++size] = ii, mark[ii] = 1; - } - } - } - } - } -done: /* free working arrays */ - xfree(ind); - xfree(list); - xfree(mark); - xfree(pass); - xfree(val); - xfree(lb); - xfree(ub); - return ret; -} - -/*********************************************************************** -* NAME -* -* ios_preprocess_node - preprocess current subproblem -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_preprocess_node(glp_tree *tree, int max_pass); -* -* DESCRIPTION -* -* The routine ios_preprocess_node performs basic preprocessing of the -* current subproblem. -* -* RETURNS -* -* If no primal infeasibility is detected, the routine returns zero, -* otherwise non-zero. */ - -int ios_preprocess_node(glp_tree *tree, int max_pass) -{ glp_prob *mip = tree->mip; - int m = mip->m; - int n = mip->n; - int i, j, nrs, *num, ret = 0; - double *L, *U, *l, *u; - /* the current subproblem must exist */ - xassert(tree->curr != NULL); - /* determine original row bounds */ - L = xcalloc(1+m, sizeof(double)); - U = xcalloc(1+m, sizeof(double)); - switch (mip->mip_stat) - { case GLP_UNDEF: - L[0] = -DBL_MAX, U[0] = +DBL_MAX; - break; - case GLP_FEAS: - switch (mip->dir) - { case GLP_MIN: - L[0] = -DBL_MAX, U[0] = mip->mip_obj - mip->c0; - break; - case GLP_MAX: - L[0] = mip->mip_obj - mip->c0, U[0] = +DBL_MAX; - break; - default: - xassert(mip != mip); - } - break; - default: - xassert(mip != mip); - } - for (i = 1; i <= m; i++) - { L[i] = glp_get_row_lb(mip, i); - U[i] = glp_get_row_ub(mip, i); - } - /* determine original column bounds */ - l = xcalloc(1+n, sizeof(double)); - u = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) - { l[j] = glp_get_col_lb(mip, j); - u[j] = glp_get_col_ub(mip, j); - } - /* build the initial list of rows to be analyzed */ - nrs = m + 1; - num = xcalloc(1+nrs, sizeof(int)); - for (i = 1; i <= nrs; i++) num[i] = i - 1; - /* perform basic preprocessing */ - if (basic_preprocessing(mip , L, U, l, u, nrs, num, max_pass)) - { ret = 1; - goto done; - } - /* set new actual (relaxed) row bounds */ - for (i = 1; i <= m; i++) - { /* consider only non-active rows to keep dual feasibility */ - if (glp_get_row_stat(mip, i) == GLP_BS) - { if (L[i] == -DBL_MAX && U[i] == +DBL_MAX) - glp_set_row_bnds(mip, i, GLP_FR, 0.0, 0.0); - else if (U[i] == +DBL_MAX) - glp_set_row_bnds(mip, i, GLP_LO, L[i], 0.0); - else if (L[i] == -DBL_MAX) - glp_set_row_bnds(mip, i, GLP_UP, 0.0, U[i]); - } - } - /* set new actual (tightened) column bounds */ - for (j = 1; j <= n; j++) - { int type; - if (l[j] == -DBL_MAX && u[j] == +DBL_MAX) - type = GLP_FR; - else if (u[j] == +DBL_MAX) - type = GLP_LO; - else if (l[j] == -DBL_MAX) - type = GLP_UP; - else if (l[j] != u[j]) - type = GLP_DB; - else - type = GLP_FX; - glp_set_col_bnds(mip, j, type, l[j], u[j]); - } -done: /* free working arrays and return */ - xfree(L); - xfree(U); - xfree(l); - xfree(u); - xfree(num); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios03.c b/code/3rd_glpk/draft/glpios03.c deleted file mode 100644 index 2f4ea684..00000000 --- a/code/3rd_glpk/draft/glpios03.c +++ /dev/null @@ -1,1513 +0,0 @@ -/* glpios03.c (branch-and-cut driver) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/*********************************************************************** -* show_progress - display current progress of the search -* -* This routine displays some information about current progress of the -* search. -* -* The information includes: -* -* the current number of iterations performed by the simplex solver; -* -* the objective value for the best known integer feasible solution, -* which is upper (minimization) or lower (maximization) global bound -* for optimal solution of the original mip problem; -* -* the best local bound for active nodes, which is lower (minimization) -* or upper (maximization) global bound for optimal solution of the -* original mip problem; -* -* the relative mip gap, in percents; -* -* the number of open (active) subproblems; -* -* the number of completely explored subproblems, i.e. whose nodes have -* been removed from the tree. */ - -static void show_progress(glp_tree *T, int bingo) -{ int p; - double temp; - char best_mip[50], best_bound[50], *rho, rel_gap[50]; - /* format the best known integer feasible solution */ - if (T->mip->mip_stat == GLP_FEAS) - sprintf(best_mip, "%17.9e", T->mip->mip_obj); - else - sprintf(best_mip, "%17s", "not found yet"); - /* determine reference number of an active subproblem whose local - bound is best */ - p = ios_best_node(T); - /* format the best bound */ - if (p == 0) - sprintf(best_bound, "%17s", "tree is empty"); - else - { temp = T->slot[p].node->bound; - if (temp == -DBL_MAX) - sprintf(best_bound, "%17s", "-inf"); - else if (temp == +DBL_MAX) - sprintf(best_bound, "%17s", "+inf"); - else - { if (fabs(temp) < 1e-9) - temp = 0; - sprintf(best_bound, "%17.9e", temp); - } - } - /* choose the relation sign between global bounds */ - if (T->mip->dir == GLP_MIN) - rho = ">="; - else if (T->mip->dir == GLP_MAX) - rho = "<="; - else - xassert(T != T); - /* format the relative mip gap */ - temp = ios_relative_gap(T); - if (temp == 0.0) - sprintf(rel_gap, " 0.0%%"); - else if (temp < 0.001) - sprintf(rel_gap, "< 0.1%%"); - else if (temp <= 9.999) - sprintf(rel_gap, "%5.1f%%", 100.0 * temp); - else - sprintf(rel_gap, "%6s", ""); - /* display progress of the search */ - xprintf("+%6d: %s %s %s %s %s (%d; %d)\n", - T->mip->it_cnt, bingo ? ">>>>>" : "mip =", best_mip, rho, - best_bound, rel_gap, T->a_cnt, T->t_cnt - T->n_cnt); - T->tm_lag = xtime(); - return; -} - -/*********************************************************************** -* is_branch_hopeful - check if specified branch is hopeful -* -* This routine checks if the specified subproblem can have an integer -* optimal solution which is better than the best known one. -* -* The check is based on comparison of the local objective bound stored -* in the subproblem descriptor and the incumbent objective value which -* is the global objective bound. -* -* If there is a chance that the specified subproblem can have a better -* integer optimal solution, the routine returns non-zero. Otherwise, if -* the corresponding branch can pruned, zero is returned. */ - -static int is_branch_hopeful(glp_tree *T, int p) -{ xassert(1 <= p && p <= T->nslots); - xassert(T->slot[p].node != NULL); - return ios_is_hopeful(T, T->slot[p].node->bound); -} - -/*********************************************************************** -* check_integrality - check integrality of basic solution -* -* This routine checks if the basic solution of LP relaxation of the -* current subproblem satisfies to integrality conditions, i.e. that all -* variables of integer kind have integral primal values. (The solution -* is assumed to be optimal.) -* -* For each variable of integer kind the routine computes the following -* quantity: -* -* ii(x[j]) = min(x[j] - floor(x[j]), ceil(x[j]) - x[j]), (1) -* -* which is a measure of the integer infeasibility (non-integrality) of -* x[j] (for example, ii(2.1) = 0.1, ii(3.7) = 0.3, ii(5.0) = 0). It is -* understood that 0 <= ii(x[j]) <= 0.5, and variable x[j] is integer -* feasible if ii(x[j]) = 0. However, due to floating-point arithmetic -* the routine checks less restrictive condition: -* -* ii(x[j]) <= tol_int, (2) -* -* where tol_int is a given tolerance (small positive number) and marks -* each variable which does not satisfy to (2) as integer infeasible by -* setting its fractionality flag. -* -* In order to characterize integer infeasibility of the basic solution -* in the whole the routine computes two parameters: ii_cnt, which is -* the number of variables with the fractionality flag set, and ii_sum, -* which is the sum of integer infeasibilities (1). */ - -static void check_integrality(glp_tree *T) -{ glp_prob *mip = T->mip; - int j, type, ii_cnt = 0; - double lb, ub, x, temp1, temp2, ii_sum = 0.0; - /* walk through the set of columns (structural variables) */ - for (j = 1; j <= mip->n; j++) - { GLPCOL *col = mip->col[j]; - T->non_int[j] = 0; - /* if the column is not integer, skip it */ - if (col->kind != GLP_IV) continue; - /* if the column is non-basic, it is integer feasible */ - if (col->stat != GLP_BS) continue; - /* obtain the type and bounds of the column */ - type = col->type, lb = col->lb, ub = col->ub; - /* obtain value of the column in optimal basic solution */ - x = col->prim; - /* if the column's primal value is close to the lower bound, - the column is integer feasible within given tolerance */ - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { temp1 = lb - T->parm->tol_int; - temp2 = lb + T->parm->tol_int; - if (temp1 <= x && x <= temp2) continue; -#if 0 - /* the lower bound must not be violated */ - xassert(x >= lb); -#else - if (x < lb) continue; -#endif - } - /* if the column's primal value is close to the upper bound, - the column is integer feasible within given tolerance */ - if (type == GLP_UP || type == GLP_DB || type == GLP_FX) - { temp1 = ub - T->parm->tol_int; - temp2 = ub + T->parm->tol_int; - if (temp1 <= x && x <= temp2) continue; -#if 0 - /* the upper bound must not be violated */ - xassert(x <= ub); -#else - if (x > ub) continue; -#endif - } - /* if the column's primal value is close to nearest integer, - the column is integer feasible within given tolerance */ - temp1 = floor(x + 0.5) - T->parm->tol_int; - temp2 = floor(x + 0.5) + T->parm->tol_int; - if (temp1 <= x && x <= temp2) continue; - /* otherwise the column is integer infeasible */ - T->non_int[j] = 1; - /* increase the number of fractional-valued columns */ - ii_cnt++; - /* compute the sum of integer infeasibilities */ - temp1 = x - floor(x); - temp2 = ceil(x) - x; - xassert(temp1 > 0.0 && temp2 > 0.0); - ii_sum += (temp1 <= temp2 ? temp1 : temp2); - } - /* store ii_cnt and ii_sum to the current problem descriptor */ - xassert(T->curr != NULL); - T->curr->ii_cnt = ii_cnt; - T->curr->ii_sum = ii_sum; - /* and also display these parameters */ - if (T->parm->msg_lev >= GLP_MSG_DBG) - { if (ii_cnt == 0) - xprintf("There are no fractional columns\n"); - else if (ii_cnt == 1) - xprintf("There is one fractional column, integer infeasibil" - "ity is %.3e\n", ii_sum); - else - xprintf("There are %d fractional columns, integer infeasibi" - "lity is %.3e\n", ii_cnt, ii_sum); - } - return; -} - -/*********************************************************************** -* record_solution - record better integer feasible solution -* -* This routine records optimal basic solution of LP relaxation of the -* current subproblem, which being integer feasible is better than the -* best known integer feasible solution. */ - -static void record_solution(glp_tree *T) -{ glp_prob *mip = T->mip; - int i, j; - mip->mip_stat = GLP_FEAS; - mip->mip_obj = mip->obj_val; - for (i = 1; i <= mip->m; i++) - { GLPROW *row = mip->row[i]; - row->mipx = row->prim; - } - for (j = 1; j <= mip->n; j++) - { GLPCOL *col = mip->col[j]; - if (col->kind == GLP_CV) - col->mipx = col->prim; - else if (col->kind == GLP_IV) - { /* value of the integer column must be integral */ - col->mipx = floor(col->prim + 0.5); - } - else - xassert(col != col); - } - T->sol_cnt++; - return; -} - -/*********************************************************************** -* fix_by_red_cost - fix non-basic integer columns by reduced costs -* -* This routine fixes some non-basic integer columns if their reduced -* costs indicate that increasing (decreasing) the column at least by -* one involves the objective value becoming worse than the incumbent -* objective value. */ - -static void fix_by_red_cost(glp_tree *T) -{ glp_prob *mip = T->mip; - int j, stat, fixed = 0; - double obj, lb, ub, dj; - /* the global bound must exist */ - xassert(T->mip->mip_stat == GLP_FEAS); - /* basic solution of LP relaxation must be optimal */ - xassert(mip->pbs_stat == GLP_FEAS && mip->dbs_stat == GLP_FEAS); - /* determine the objective function value */ - obj = mip->obj_val; - /* walk through the column list */ - for (j = 1; j <= mip->n; j++) - { GLPCOL *col = mip->col[j]; - /* if the column is not integer, skip it */ - if (col->kind != GLP_IV) continue; - /* obtain bounds of j-th column */ - lb = col->lb, ub = col->ub; - /* and determine its status and reduced cost */ - stat = col->stat, dj = col->dual; - /* analyze the reduced cost */ - switch (mip->dir) - { case GLP_MIN: - /* minimization */ - if (stat == GLP_NL) - { /* j-th column is non-basic on its lower bound */ - if (dj < 0.0) dj = 0.0; - if (obj + dj >= mip->mip_obj) - glp_set_col_bnds(mip, j, GLP_FX, lb, lb), fixed++; - } - else if (stat == GLP_NU) - { /* j-th column is non-basic on its upper bound */ - if (dj > 0.0) dj = 0.0; - if (obj - dj >= mip->mip_obj) - glp_set_col_bnds(mip, j, GLP_FX, ub, ub), fixed++; - } - break; - case GLP_MAX: - /* maximization */ - if (stat == GLP_NL) - { /* j-th column is non-basic on its lower bound */ - if (dj > 0.0) dj = 0.0; - if (obj + dj <= mip->mip_obj) - glp_set_col_bnds(mip, j, GLP_FX, lb, lb), fixed++; - } - else if (stat == GLP_NU) - { /* j-th column is non-basic on its upper bound */ - if (dj < 0.0) dj = 0.0; - if (obj - dj <= mip->mip_obj) - glp_set_col_bnds(mip, j, GLP_FX, ub, ub), fixed++; - } - break; - default: - xassert(T != T); - } - } - if (T->parm->msg_lev >= GLP_MSG_DBG) - { if (fixed == 0) - /* nothing to say */; - else if (fixed == 1) - xprintf("One column has been fixed by reduced cost\n"); - else - xprintf("%d columns have been fixed by reduced costs\n", - fixed); - } - /* fixing non-basic columns on their current bounds does not - change the basic solution */ - xassert(mip->pbs_stat == GLP_FEAS && mip->dbs_stat == GLP_FEAS); - return; -} - -/*********************************************************************** -* branch_on - perform branching on specified variable -* -* This routine performs branching on j-th column (structural variable) -* of the current subproblem. The specified column must be of integer -* kind and must have a fractional value in optimal basic solution of -* LP relaxation of the current subproblem (i.e. only columns for which -* the flag non_int[j] is set are valid candidates to branch on). -* -* Let x be j-th structural variable, and beta be its primal fractional -* value in the current basic solution. Branching on j-th variable is -* dividing the current subproblem into two new subproblems, which are -* identical to the current subproblem with the following exception: in -* the first subproblem that begins the down-branch x has a new upper -* bound x <= floor(beta), and in the second subproblem that begins the -* up-branch x has a new lower bound x >= ceil(beta). -* -* Depending on estimation of local bounds for down- and up-branches -* this routine returns the following: -* -* 0 - both branches have been created; -* 1 - one branch is hopeless and has been pruned, so now the current -* subproblem is other branch; -* 2 - both branches are hopeless and have been pruned; new subproblem -* selection is needed to continue the search. */ - -static int branch_on(glp_tree *T, int j, int next) -{ glp_prob *mip = T->mip; - IOSNPD *node; - int m = mip->m; - int n = mip->n; - int type, dn_type, up_type, dn_bad, up_bad, p, ret, clone[1+2]; - double lb, ub, beta, new_ub, new_lb, dn_lp, up_lp, dn_bnd, up_bnd; - /* determine bounds and value of x[j] in optimal solution to LP - relaxation of the current subproblem */ - xassert(1 <= j && j <= n); - type = mip->col[j]->type; - lb = mip->col[j]->lb; - ub = mip->col[j]->ub; - beta = mip->col[j]->prim; - /* determine new bounds of x[j] for down- and up-branches */ - new_ub = floor(beta); - new_lb = ceil(beta); - switch (type) - { case GLP_FR: - dn_type = GLP_UP; - up_type = GLP_LO; - break; - case GLP_LO: - xassert(lb <= new_ub); - dn_type = (lb == new_ub ? GLP_FX : GLP_DB); - xassert(lb + 1.0 <= new_lb); - up_type = GLP_LO; - break; - case GLP_UP: - xassert(new_ub <= ub - 1.0); - dn_type = GLP_UP; - xassert(new_lb <= ub); - up_type = (new_lb == ub ? GLP_FX : GLP_DB); - break; - case GLP_DB: - xassert(lb <= new_ub && new_ub <= ub - 1.0); - dn_type = (lb == new_ub ? GLP_FX : GLP_DB); - xassert(lb + 1.0 <= new_lb && new_lb <= ub); - up_type = (new_lb == ub ? GLP_FX : GLP_DB); - break; - default: - xassert(type != type); - } - /* compute local bounds to LP relaxation for both branches */ - ios_eval_degrad(T, j, &dn_lp, &up_lp); - /* and improve them by rounding */ - dn_bnd = ios_round_bound(T, dn_lp); - up_bnd = ios_round_bound(T, up_lp); - /* check local bounds for down- and up-branches */ - dn_bad = !ios_is_hopeful(T, dn_bnd); - up_bad = !ios_is_hopeful(T, up_bnd); - if (dn_bad && up_bad) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Both down- and up-branches are hopeless\n"); - ret = 2; - goto done; - } - else if (up_bad) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Up-branch is hopeless\n"); - glp_set_col_bnds(mip, j, dn_type, lb, new_ub); - T->curr->lp_obj = dn_lp; - if (mip->dir == GLP_MIN) - { if (T->curr->bound < dn_bnd) - T->curr->bound = dn_bnd; - } - else if (mip->dir == GLP_MAX) - { if (T->curr->bound > dn_bnd) - T->curr->bound = dn_bnd; - } - else - xassert(mip != mip); - ret = 1; - goto done; - } - else if (dn_bad) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Down-branch is hopeless\n"); - glp_set_col_bnds(mip, j, up_type, new_lb, ub); - T->curr->lp_obj = up_lp; - if (mip->dir == GLP_MIN) - { if (T->curr->bound < up_bnd) - T->curr->bound = up_bnd; - } - else if (mip->dir == GLP_MAX) - { if (T->curr->bound > up_bnd) - T->curr->bound = up_bnd; - } - else - xassert(mip != mip); - ret = 1; - goto done; - } - /* both down- and up-branches seem to be hopeful */ - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Branching on column %d, primal value is %.9e\n", - j, beta); - /* determine the reference number of the current subproblem */ - xassert(T->curr != NULL); - p = T->curr->p; - T->curr->br_var = j; - T->curr->br_val = beta; - /* freeze the current subproblem */ - ios_freeze_node(T); - /* create two clones of the current subproblem; the first clone - begins the down-branch, the second one begins the up-branch */ - ios_clone_node(T, p, 2, clone); - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Node %d begins down branch, node %d begins up branch " - "\n", clone[1], clone[2]); - /* set new upper bound of j-th column in the down-branch */ - node = T->slot[clone[1]].node; - xassert(node != NULL); - xassert(node->up != NULL); - xassert(node->b_ptr == NULL); - node->b_ptr = dmp_get_atom(T->pool, sizeof(IOSBND)); - node->b_ptr->k = m + j; - node->b_ptr->type = (unsigned char)dn_type; - node->b_ptr->lb = lb; - node->b_ptr->ub = new_ub; - node->b_ptr->next = NULL; - node->lp_obj = dn_lp; - if (mip->dir == GLP_MIN) - { if (node->bound < dn_bnd) - node->bound = dn_bnd; - } - else if (mip->dir == GLP_MAX) - { if (node->bound > dn_bnd) - node->bound = dn_bnd; - } - else - xassert(mip != mip); - /* set new lower bound of j-th column in the up-branch */ - node = T->slot[clone[2]].node; - xassert(node != NULL); - xassert(node->up != NULL); - xassert(node->b_ptr == NULL); - node->b_ptr = dmp_get_atom(T->pool, sizeof(IOSBND)); - node->b_ptr->k = m + j; - node->b_ptr->type = (unsigned char)up_type; - node->b_ptr->lb = new_lb; - node->b_ptr->ub = ub; - node->b_ptr->next = NULL; - node->lp_obj = up_lp; - if (mip->dir == GLP_MIN) - { if (node->bound < up_bnd) - node->bound = up_bnd; - } - else if (mip->dir == GLP_MAX) - { if (node->bound > up_bnd) - node->bound = up_bnd; - } - else - xassert(mip != mip); - /* suggest the subproblem to be solved next */ - xassert(T->child == 0); - if (next == GLP_NO_BRNCH) - T->child = 0; - else if (next == GLP_DN_BRNCH) - T->child = clone[1]; - else if (next == GLP_UP_BRNCH) - T->child = clone[2]; - else - xassert(next != next); - ret = 0; -done: return ret; -} - -/*********************************************************************** -* cleanup_the_tree - prune hopeless branches from the tree -* -* This routine walks through the active list and checks the local -* bound for every active subproblem. If the local bound indicates that -* the subproblem cannot have integer optimal solution better than the -* incumbent objective value, the routine deletes such subproblem that, -* in turn, involves pruning the corresponding branch of the tree. */ - -static void cleanup_the_tree(glp_tree *T) -{ IOSNPD *node, *next_node; - int count = 0; - /* the global bound must exist */ - xassert(T->mip->mip_stat == GLP_FEAS); - /* walk through the list of active subproblems */ - for (node = T->head; node != NULL; node = next_node) - { /* deleting some active problem node may involve deleting its - parents recursively; however, all its parents being created - *before* it are always *precede* it in the node list, so - the next problem node is never affected by such deletion */ - next_node = node->next; - /* if the branch is hopeless, prune it */ - if (!is_branch_hopeful(T, node->p)) - ios_delete_node(T, node->p), count++; - } - if (T->parm->msg_lev >= GLP_MSG_DBG) - { if (count == 1) - xprintf("One hopeless branch has been pruned\n"); - else if (count > 1) - xprintf("%d hopeless branches have been pruned\n", count); - } - return; -} - -/*********************************************************************** -* round_heur - simple rounding heuristic -* -* This routine attempts to guess an integer feasible solution by -* simple rounding values of all integer variables in basic solution to -* nearest integers. */ - -static int round_heur(glp_tree *T) -{ glp_prob *P = T->mip; - /*int m = P->m;*/ - int n = P->n; - int i, j, ret; - double *x; - /* compute rounded values of variables */ - x = talloc(1+n, double); - for (j = 1; j <= n; j++) - { GLPCOL *col = P->col[j]; - if (col->kind == GLP_IV) - { /* integer variable */ - x[j] = floor(col->prim + 0.5); - } - else if (col->type == GLP_FX) - { /* fixed variable */ - x[j] = col->prim; - } - else - { /* non-integer non-fixed variable */ - ret = 3; - goto done; - } - } - /* check that no constraints are violated */ - for (i = 1; i <= T->orig_m; i++) - { int type = T->orig_type[i]; - GLPAIJ *aij; - double sum; - if (type == GLP_FR) - continue; - /* compute value of linear form */ - sum = 0.0; - for (aij = P->row[i]->ptr; aij != NULL; aij = aij->r_next) - sum += aij->val * x[aij->col->j]; - /* check lower bound */ - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { if (sum < T->orig_lb[i] - 1e-9) - { /* lower bound is violated */ - ret = 2; - goto done; - } - } - /* check upper bound */ - if (type == GLP_UP || type == GLP_DB || type == GLP_FX) - { if (sum > T->orig_ub[i] + 1e-9) - { /* upper bound is violated */ - ret = 2; - goto done; - } - } - } - /* rounded solution is integer feasible */ - if (glp_ios_heur_sol(T, x) == 0) - { /* solution is accepted */ - ret = 0; - } - else - { /* solution is rejected */ - ret = 1; - } -done: tfree(x); - return ret; -} - -/**********************************************************************/ - -#if 1 /* 08/III-2016 */ -static void gmi_gen(glp_tree *T) -{ /* generate Gomory's mixed integer cuts */ - glp_prob *P, *pool; - P = T->mip; - pool = glp_create_prob(); - glp_add_cols(pool, P->n); - glp_gmi_gen(P, pool, 50); - if (pool->m > 0) - { int i, len, *ind; - double *val; - ind = xcalloc(1+P->n, sizeof(int)); - val = xcalloc(1+P->n, sizeof(double)); - for (i = 1; i <= pool->m; i++) - { len = glp_get_mat_row(pool, i, ind, val); - glp_ios_add_row(T, NULL, GLP_RF_GMI, 0, len, ind, val, - GLP_LO, pool->row[i]->lb); - } - xfree(ind); - xfree(val); - } - glp_delete_prob(pool); - return; -} -#endif - -#ifdef NEW_COVER /* 13/II-2018 */ -static void cov_gen(glp_tree *T) -{ /* generate cover cuts */ - glp_prob *P, *pool; - if (T->cov_gen == NULL) - return; - P = T->mip; - pool = glp_create_prob(); - glp_add_cols(pool, P->n); - glp_cov_gen1(P, T->cov_gen, pool); - if (pool->m > 0) - { int i, len, *ind; - double *val; - ind = xcalloc(1+P->n, sizeof(int)); - val = xcalloc(1+P->n, sizeof(double)); - for (i = 1; i <= pool->m; i++) - { len = glp_get_mat_row(pool, i, ind, val); - glp_ios_add_row(T, NULL, GLP_RF_COV, 0, len, ind, val, - GLP_UP, pool->row[i]->ub); - } - xfree(ind); - xfree(val); - } - glp_delete_prob(pool); - return; -} -#endif - -#if 1 /* 08/III-2016 */ -static void mir_gen(glp_tree *T) -{ /* generate mixed integer rounding cuts */ - glp_prob *P, *pool; - P = T->mip; - pool = glp_create_prob(); - glp_add_cols(pool, P->n); - glp_mir_gen(P, T->mir_gen, pool); - if (pool->m > 0) - { int i, len, *ind; - double *val; - ind = xcalloc(1+P->n, sizeof(int)); - val = xcalloc(1+P->n, sizeof(double)); - for (i = 1; i <= pool->m; i++) - { len = glp_get_mat_row(pool, i, ind, val); - glp_ios_add_row(T, NULL, GLP_RF_MIR, 0, len, ind, val, - GLP_UP, pool->row[i]->ub); - } - xfree(ind); - xfree(val); - } - glp_delete_prob(pool); - return; -} -#endif - -#if 1 /* 08/III-2016 */ -static void clq_gen(glp_tree *T, glp_cfg *G) -{ /* generate clique cut from conflict graph */ - glp_prob *P = T->mip; - int n = P->n; - int len, *ind; - double *val; - ind = talloc(1+n, int); - val = talloc(1+n, double); - len = glp_clq_cut(T->mip, G, ind, val); - if (len > 0) - glp_ios_add_row(T, NULL, GLP_RF_CLQ, 0, len, ind, val, GLP_UP, - val[0]); - tfree(ind); - tfree(val); - return; -} -#endif - -static void generate_cuts(glp_tree *T) -{ /* generate generic cuts with built-in generators */ - if (!(T->parm->mir_cuts == GLP_ON || - T->parm->gmi_cuts == GLP_ON || - T->parm->cov_cuts == GLP_ON || - T->parm->clq_cuts == GLP_ON)) goto done; -#if 1 /* 20/IX-2008 */ - { int i, max_cuts, added_cuts; - max_cuts = T->n; - if (max_cuts < 1000) max_cuts = 1000; - added_cuts = 0; - for (i = T->orig_m+1; i <= T->mip->m; i++) - { if (T->mip->row[i]->origin == GLP_RF_CUT) - added_cuts++; - } - /* xprintf("added_cuts = %d\n", added_cuts); */ - if (added_cuts >= max_cuts) goto done; - } -#endif - /* generate and add to POOL all cuts violated by x* */ - if (T->parm->gmi_cuts == GLP_ON) - { if (T->curr->changed < 7) -#if 0 /* 08/III-2016 */ - ios_gmi_gen(T); -#else - gmi_gen(T); -#endif - } - if (T->parm->mir_cuts == GLP_ON) - { xassert(T->mir_gen != NULL); -#if 0 /* 08/III-2016 */ - ios_mir_gen(T, T->mir_gen); -#else - mir_gen(T); -#endif - } - if (T->parm->cov_cuts == GLP_ON) - { /* cover cuts works well along with mir cuts */ -#ifdef NEW_COVER /* 13/II-2018 */ - cov_gen(T); -#else - ios_cov_gen(T); -#endif - } - if (T->parm->clq_cuts == GLP_ON) - { if (T->clq_gen != NULL) -#if 0 /* 29/VI-2013 */ - { if (T->curr->level == 0 && T->curr->changed < 50 || - T->curr->level > 0 && T->curr->changed < 5) -#else /* FIXME */ - { if (T->curr->level == 0 && T->curr->changed < 500 || - T->curr->level > 0 && T->curr->changed < 50) -#endif -#if 0 /* 08/III-2016 */ - ios_clq_gen(T, T->clq_gen); -#else - clq_gen(T, T->clq_gen); -#endif - } - } -done: return; -} - -/**********************************************************************/ - -static void remove_cuts(glp_tree *T) -{ /* remove inactive cuts (some valueable globally valid cut might - be saved in the global cut pool) */ - int i, cnt = 0, *num = NULL; - xassert(T->curr != NULL); - for (i = T->orig_m+1; i <= T->mip->m; i++) - { if (T->mip->row[i]->origin == GLP_RF_CUT && - T->mip->row[i]->level == T->curr->level && - T->mip->row[i]->stat == GLP_BS) - { if (num == NULL) - num = xcalloc(1+T->mip->m, sizeof(int)); - num[++cnt] = i; - } - } - if (cnt > 0) - { glp_del_rows(T->mip, cnt, num); -#if 0 - xprintf("%d inactive cut(s) removed\n", cnt); -#endif - xfree(num); - xassert(glp_factorize(T->mip) == 0); - } - return; -} - -/**********************************************************************/ - -static void display_cut_info(glp_tree *T) -{ glp_prob *mip = T->mip; - int i, gmi = 0, mir = 0, cov = 0, clq = 0, app = 0; - for (i = mip->m; i > 0; i--) - { GLPROW *row; - row = mip->row[i]; - /* if (row->level < T->curr->level) break; */ - if (row->origin == GLP_RF_CUT) - { if (row->klass == GLP_RF_GMI) - gmi++; - else if (row->klass == GLP_RF_MIR) - mir++; - else if (row->klass == GLP_RF_COV) - cov++; - else if (row->klass == GLP_RF_CLQ) - clq++; - else - app++; - } - } - xassert(T->curr != NULL); - if (gmi + mir + cov + clq + app > 0) - { xprintf("Cuts on level %d:", T->curr->level); - if (gmi > 0) xprintf(" gmi = %d;", gmi); - if (mir > 0) xprintf(" mir = %d;", mir); - if (cov > 0) xprintf(" cov = %d;", cov); - if (clq > 0) xprintf(" clq = %d;", clq); - if (app > 0) xprintf(" app = %d;", app); - xprintf("\n"); - } - return; -} - -/*********************************************************************** -* NAME -* -* ios_driver - branch-and-cut driver -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_driver(glp_tree *T); -* -* DESCRIPTION -* -* The routine ios_driver is a branch-and-cut driver. It controls the -* MIP solution process. -* -* RETURNS -* -* 0 The MIP problem instance has been successfully solved. This code -* does not necessarily mean that the solver has found optimal -* solution. It only means that the solution process was successful. -* -* GLP_EFAIL -* The search was prematurely terminated due to the solver failure. -* -* GLP_EMIPGAP -* The search was prematurely terminated, because the relative mip -* gap tolerance has been reached. -* -* GLP_ETMLIM -* The search was prematurely terminated, because the time limit has -* been exceeded. -* -* GLP_ESTOP -* The search was prematurely terminated by application. */ - -int ios_driver(glp_tree *T) -{ int p, curr_p, p_stat, d_stat, ret; -#if 1 /* carry out to glp_tree */ - int pred_p = 0; - /* if the current subproblem has been just created due to - branching, pred_p is the reference number of its parent - subproblem, otherwise pred_p is zero */ -#endif -#if 1 /* 18/VII-2013 */ - int bad_cut; - double old_obj; -#endif -#if 0 /* 10/VI-2013 */ - glp_long ttt = T->tm_beg; -#else - double ttt = T->tm_beg; -#endif -#if 1 /* 27/II-2016 by Chris */ - int root_done = 0; -#endif -#if 0 - ((glp_iocp *)T->parm)->msg_lev = GLP_MSG_DBG; -#endif -#if 1 /* 16/III-2016 */ - if (((glp_iocp *)T->parm)->flip) -#if 0 /* 20/I-2018 */ - xprintf("WARNING: LONG-STEP DUAL SIMPLEX WILL BE USED\n"); -#else - // Liangliang: disable this message - //xprintf("Long-step dual simplex will be used\n"); -#endif -#endif - /* on entry to the B&B driver it is assumed that the active list - contains the only active (i.e. root) subproblem, which is the - original MIP problem to be solved */ -loop: /* main loop starts here */ - /* at this point the current subproblem does not exist */ - xassert(T->curr == NULL); - /* if the active list is empty, the search is finished */ - if (T->head == NULL) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Active list is empty!\n"); -#if 0 /* 10/VI-2013 */ - xassert(dmp_in_use(T->pool).lo == 0); -#else - xassert(dmp_in_use(T->pool) == 0); -#endif - ret = 0; - goto done; - } - /* select some active subproblem to continue the search */ - xassert(T->next_p == 0); - /* let the application program select subproblem */ - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - T->reason = GLP_ISELECT; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - } - if (T->next_p != 0) - { /* the application program has selected something */ - ; - } - else if (T->a_cnt == 1) - { /* the only active subproblem exists, so select it */ - xassert(T->head->next == NULL); - T->next_p = T->head->p; - } - else if (T->child != 0) - { /* select one of branching childs suggested by the branching - heuristic */ - T->next_p = T->child; - } - else - { /* select active subproblem as specified by the backtracking - technique option */ - T->next_p = ios_choose_node(T); - } - /* the active subproblem just selected becomes current */ - ios_revive_node(T, T->next_p); - T->next_p = T->child = 0; - /* invalidate pred_p, if it is not the reference number of the - parent of the current subproblem */ - if (T->curr->up != NULL && T->curr->up->p != pred_p) pred_p = 0; - /* determine the reference number of the current subproblem */ - p = T->curr->p; - if (T->parm->msg_lev >= GLP_MSG_DBG) - { xprintf("-----------------------------------------------------" - "-------------------\n"); - xprintf("Processing node %d at level %d\n", p, T->curr->level); - } -#if 0 - if (p == 1) - glp_write_lp(T->mip, NULL, "root.lp"); -#endif -#if 1 /* 24/X-2015 */ - if (p == 1) - { if (T->parm->sr_heur == GLP_OFF) - { if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Simple rounding heuristic disabled\n"); - } - } -#endif - /* if it is the root subproblem, initialize cut generators */ - if (p == 1) - { if (T->parm->gmi_cuts == GLP_ON) - { if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Gomory's cuts enabled\n"); - } - if (T->parm->mir_cuts == GLP_ON) - { if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("MIR cuts enabled\n"); - xassert(T->mir_gen == NULL); -#if 0 /* 06/III-2016 */ - T->mir_gen = ios_mir_init(T); -#else - T->mir_gen = glp_mir_init(T->mip); -#endif - } - if (T->parm->cov_cuts == GLP_ON) - { if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Cover cuts enabled\n"); -#ifdef NEW_COVER /* 13/II-2018 */ - xassert(T->cov_gen == NULL); - T->cov_gen = glp_cov_init(T->mip); -#endif - } - if (T->parm->clq_cuts == GLP_ON) - { xassert(T->clq_gen == NULL); - if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Clique cuts enabled\n"); -#if 0 /* 08/III-2016 */ - T->clq_gen = ios_clq_init(T); -#else - T->clq_gen = glp_cfg_init(T->mip); -#endif - } - } -#if 1 /* 18/VII-2013 */ - bad_cut = 0; -#endif -more: /* minor loop starts here */ - /* at this point the current subproblem needs either to be solved - for the first time or re-optimized due to reformulation */ - /* display current progress of the search */ - if (T->parm->msg_lev >= GLP_MSG_DBG || - T->parm->msg_lev >= GLP_MSG_ON && - (double)(T->parm->out_frq - 1) <= - 1000.0 * xdifftime(xtime(), T->tm_lag)) - show_progress(T, 0); - if (T->parm->msg_lev >= GLP_MSG_ALL && - xdifftime(xtime(), ttt) >= 60.0) -#if 0 /* 16/II-2012 */ - { glp_long total; - glp_mem_usage(NULL, NULL, &total, NULL); - xprintf("Time used: %.1f secs. Memory used: %.1f Mb.\n", - xdifftime(xtime(), T->tm_beg), xltod(total) / 1048576.0); - ttt = xtime(); - } -#else - { size_t total; - glp_mem_usage(NULL, NULL, &total, NULL); - xprintf("Time used: %.1f secs. Memory used: %.1f Mb.\n", - xdifftime(xtime(), T->tm_beg), (double)total / 1048576.0); - ttt = xtime(); - } -#endif - /* check the mip gap */ - if (T->parm->mip_gap > 0.0 && - ios_relative_gap(T) <= T->parm->mip_gap) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Relative gap tolerance reached; search terminated " - "\n"); - ret = GLP_EMIPGAP; - goto done; - } - /* check if the time limit has been exhausted */ - if (T->parm->tm_lim < INT_MAX && - (double)(T->parm->tm_lim - 1) <= - 1000.0 * xdifftime(xtime(), T->tm_beg)) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Time limit exhausted; search terminated\n"); - ret = GLP_ETMLIM; - goto done; - } - /* let the application program preprocess the subproblem */ - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - T->reason = GLP_IPREPRO; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - } - /* perform basic preprocessing */ - if (T->parm->pp_tech == GLP_PP_NONE) - ; - else if (T->parm->pp_tech == GLP_PP_ROOT) -#if 0 /* 27/II-2016 by Chris */ - { if (T->curr->level == 0) -#else - { if (!root_done) -#endif - { if (ios_preprocess_node(T, 100)) - goto fath; - } - } - else if (T->parm->pp_tech == GLP_PP_ALL) -#if 0 /* 27/II-2016 by Chris */ - { if (ios_preprocess_node(T, T->curr->level == 0 ? 100 : 10)) -#else - { if (ios_preprocess_node(T, !root_done ? 100 : 10)) -#endif - goto fath; - } - else - xassert(T != T); - /* preprocessing may improve the global bound */ - if (!is_branch_hopeful(T, p)) - { xprintf("*** not tested yet ***\n"); - goto fath; - } - /* solve LP relaxation of the current subproblem */ - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Solving LP relaxation...\n"); - ret = ios_solve_node(T); - if (ret == GLP_ETMLIM) - goto done; - else if (!(ret == 0 || ret == GLP_EOBJLL || ret == GLP_EOBJUL)) - { if (T->parm->msg_lev >= GLP_MSG_ERR) - xprintf("ios_driver: unable to solve current LP relaxation;" - " glp_simplex returned %d\n", ret); - ret = GLP_EFAIL; - goto done; - } - /* analyze status of the basic solution to LP relaxation found */ - p_stat = T->mip->pbs_stat; - d_stat = T->mip->dbs_stat; - if (p_stat == GLP_FEAS && d_stat == GLP_FEAS) - { /* LP relaxation has optimal solution */ - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Found optimal solution to LP relaxation\n"); - } - else if (d_stat == GLP_NOFEAS) - { /* LP relaxation has no dual feasible solution */ - /* since the current subproblem cannot have a larger feasible - region than its parent, there is something wrong */ - if (T->parm->msg_lev >= GLP_MSG_ERR) - xprintf("ios_driver: current LP relaxation has no dual feas" - "ible solution\n"); - ret = GLP_EFAIL; - goto done; - } - else if (p_stat == GLP_INFEAS && d_stat == GLP_FEAS) - { /* LP relaxation has no primal solution which is better than - the incumbent objective value */ - xassert(T->mip->mip_stat == GLP_FEAS); - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("LP relaxation has no solution better than incumben" - "t objective value\n"); - /* prune the branch */ - goto fath; - } - else if (p_stat == GLP_NOFEAS) - { /* LP relaxation has no primal feasible solution */ - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("LP relaxation has no feasible solution\n"); - /* prune the branch */ - goto fath; - } - else - { /* other cases cannot appear */ - xassert(T->mip != T->mip); - } - /* at this point basic solution to LP relaxation of the current - subproblem is optimal */ - xassert(p_stat == GLP_FEAS && d_stat == GLP_FEAS); - xassert(T->curr != NULL); - T->curr->lp_obj = T->mip->obj_val; - /* thus, it defines a local bound to integer optimal solution of - the current subproblem */ - { double bound = T->mip->obj_val; - /* some local bound to the current subproblem could be already - set before, so we should only improve it */ - bound = ios_round_bound(T, bound); - if (T->mip->dir == GLP_MIN) - { if (T->curr->bound < bound) - T->curr->bound = bound; - } - else if (T->mip->dir == GLP_MAX) - { if (T->curr->bound > bound) - T->curr->bound = bound; - } - else - xassert(T->mip != T->mip); - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Local bound is %.9e\n", bound); - } - /* if the local bound indicates that integer optimal solution of - the current subproblem cannot be better than the global bound, - prune the branch */ - if (!is_branch_hopeful(T, p)) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Current branch is hopeless and can be pruned\n"); - goto fath; - } - /* let the application program generate additional rows ("lazy" - constraints) */ - xassert(T->reopt == 0); - xassert(T->reinv == 0); - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - T->reason = GLP_IROWGEN; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - if (T->reopt) - { /* some rows were added; re-optimization is needed */ - T->reopt = T->reinv = 0; - goto more; - } - if (T->reinv) - { /* no rows were added, however, some inactive rows were - removed */ - T->reinv = 0; - xassert(glp_factorize(T->mip) == 0); - } - } - /* check if the basic solution is integer feasible */ - check_integrality(T); - /* if the basic solution satisfies to all integrality conditions, - it is a new, better integer feasible solution */ - if (T->curr->ii_cnt == 0) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("New integer feasible solution found\n"); - if (T->parm->msg_lev >= GLP_MSG_ALL) - display_cut_info(T); - record_solution(T); - if (T->parm->msg_lev >= GLP_MSG_ON) - show_progress(T, 1); -#if 1 /* 11/VII-2013 */ - ios_process_sol(T); -#endif - /* make the application program happy */ - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - T->reason = GLP_IBINGO; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - } - /* since the current subproblem has been fathomed, prune its - branch */ - goto fath; - } - /* at this point basic solution to LP relaxation of the current - subproblem is optimal, but integer infeasible */ - /* try to fix some non-basic structural variables of integer kind - on their current bounds due to reduced costs */ - if (T->mip->mip_stat == GLP_FEAS) - fix_by_red_cost(T); - /* let the application program try to find some solution to the - original MIP with a primal heuristic */ - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - T->reason = GLP_IHEUR; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - /* check if the current branch became hopeless */ - if (!is_branch_hopeful(T, p)) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Current branch became hopeless and can be prune" - "d\n"); - goto fath; - } - } - /* try to find solution with the feasibility pump heuristic */ -#if 0 /* 27/II-2016 by Chris */ - if (T->parm->fp_heur) -#else - if (T->parm->fp_heur && !root_done) -#endif - { xassert(T->reason == 0); - T->reason = GLP_IHEUR; - ios_feas_pump(T); - T->reason = 0; - /* check if the current branch became hopeless */ - if (!is_branch_hopeful(T, p)) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Current branch became hopeless and can be prune" - "d\n"); - goto fath; - } - } -#if 1 /* 25/V-2013 */ - /* try to find solution with the proximity search heuristic */ -#if 0 /* 27/II-2016 by Chris */ - if (T->parm->ps_heur) -#else - if (T->parm->ps_heur && !root_done) -#endif - { xassert(T->reason == 0); - T->reason = GLP_IHEUR; - ios_proxy_heur(T); - T->reason = 0; - /* check if the current branch became hopeless */ - if (!is_branch_hopeful(T, p)) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Current branch became hopeless and can be prune" - "d\n"); - goto fath; - } - } -#endif -#if 1 /* 24/X-2015 */ - /* try to find solution with a simple rounding heuristic */ - if (T->parm->sr_heur) - { xassert(T->reason == 0); - T->reason = GLP_IHEUR; - round_heur(T); - T->reason = 0; - /* check if the current branch became hopeless */ - if (!is_branch_hopeful(T, p)) - { if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Current branch became hopeless and can be prune" - "d\n"); - goto fath; - } - } -#endif - /* it's time to generate cutting planes */ - xassert(T->local != NULL); -#ifdef NEW_LOCAL /* 02/II-2018 */ - xassert(T->local->m == 0); -#else - xassert(T->local->size == 0); -#endif - /* let the application program generate some cuts; note that it - can add cuts either to the local cut pool or directly to the - current subproblem */ - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - T->reason = GLP_ICUTGEN; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - } -#if 1 /* 18/VII-2013 */ - if (T->curr->changed > 0) - { double degrad = fabs(T->curr->lp_obj - old_obj); - if (degrad < 1e-4 * (1.0 + fabs(old_obj))) - bad_cut++; - else - bad_cut = 0; - } - old_obj = T->curr->lp_obj; -#if 0 /* 27/II-2016 by Chris */ - if (bad_cut == 0 || (T->curr->level == 0 && bad_cut <= 3)) -#else - if (bad_cut == 0 || (!root_done && bad_cut <= 3)) -#endif -#endif - /* try to generate generic cuts with built-in generators - (as suggested by Prof. Fischetti et al. the built-in cuts are - not generated at each branching node; an intense attempt of - generating new cuts is only made at the root node, and then - a moderate effort is spent after each backtracking step) */ -#if 0 /* 27/II-2016 by Chris */ - if (T->curr->level == 0 || pred_p == 0) -#else - if (!root_done || pred_p == 0) -#endif - { xassert(T->reason == 0); - T->reason = GLP_ICUTGEN; - generate_cuts(T); - T->reason = 0; - } - /* if the local cut pool is not empty, select useful cuts and add - them to the current subproblem */ -#ifdef NEW_LOCAL /* 02/II-2018 */ - if (T->local->m > 0) -#else - if (T->local->size > 0) -#endif - { xassert(T->reason == 0); - T->reason = GLP_ICUTGEN; - ios_process_cuts(T); - T->reason = 0; - } - /* clear the local cut pool */ - ios_clear_pool(T, T->local); - /* perform re-optimization, if necessary */ - if (T->reopt) - { T->reopt = 0; - T->curr->changed++; - goto more; - } - /* no cuts were generated; remove inactive cuts */ - remove_cuts(T); -#if 0 /* 27/II-2016 by Chris */ - if (T->parm->msg_lev >= GLP_MSG_ALL && T->curr->level == 0) -#else - if (T->parm->msg_lev >= GLP_MSG_ALL && !root_done) -#endif - display_cut_info(T); -#if 1 /* 27/II-2016 by Chris */ - /* the first node will not be treated as root any more */ - if (!root_done) root_done = 1; -#endif - /* update history information used on pseudocost branching */ - if (T->pcost != NULL) ios_pcost_update(T); - /* it's time to perform branching */ - xassert(T->br_var == 0); - xassert(T->br_sel == 0); - /* let the application program choose variable to branch on */ - if (T->parm->cb_func != NULL) - { xassert(T->reason == 0); - xassert(T->br_var == 0); - xassert(T->br_sel == 0); - T->reason = GLP_IBRANCH; - T->parm->cb_func(T, T->parm->cb_info); - T->reason = 0; - if (T->stop) - { ret = GLP_ESTOP; - goto done; - } - } - /* if nothing has been chosen, choose some variable as specified - by the branching technique option */ - if (T->br_var == 0) - T->br_var = ios_choose_var(T, &T->br_sel); - /* perform actual branching */ - curr_p = T->curr->p; - ret = branch_on(T, T->br_var, T->br_sel); - T->br_var = T->br_sel = 0; - if (ret == 0) - { /* both branches have been created */ - pred_p = curr_p; - goto loop; - } - else if (ret == 1) - { /* one branch is hopeless and has been pruned, so now the - current subproblem is other branch */ - /* the current subproblem should be considered as a new one, - since one bound of the branching variable was changed */ - T->curr->solved = T->curr->changed = 0; -#if 1 /* 18/VII-2013 */ - /* bad_cut = 0; */ -#endif - goto more; - } - else if (ret == 2) - { /* both branches are hopeless and have been pruned; new - subproblem selection is needed to continue the search */ - goto fath; - } - else - xassert(ret != ret); -fath: /* the current subproblem has been fathomed */ - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("Node %d fathomed\n", p); - /* freeze the current subproblem */ - ios_freeze_node(T); - /* and prune the corresponding branch of the tree */ - ios_delete_node(T, p); - /* if a new integer feasible solution has just been found, other - branches may become hopeless and therefore must be pruned */ - if (T->mip->mip_stat == GLP_FEAS) cleanup_the_tree(T); - /* new subproblem selection is needed due to backtracking */ - pred_p = 0; - goto loop; -done: /* display progress of the search on exit from the solver */ - if (T->parm->msg_lev >= GLP_MSG_ON) - show_progress(T, 0); - if (T->mir_gen != NULL) -#if 0 /* 06/III-2016 */ - ios_mir_term(T->mir_gen), T->mir_gen = NULL; -#else - glp_mir_free(T->mir_gen), T->mir_gen = NULL; -#endif -#ifdef NEW_COVER /* 13/II-2018 */ - if (T->cov_gen != NULL) - glp_cov_free(T->cov_gen), T->cov_gen = NULL; -#endif - if (T->clq_gen != NULL) -#if 0 /* 08/III-2016 */ - ios_clq_term(T->clq_gen), T->clq_gen = NULL; -#else - glp_cfg_free(T->clq_gen), T->clq_gen = NULL; -#endif - /* return to the calling program */ - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios07.c b/code/3rd_glpk/draft/glpios07.c deleted file mode 100644 index f750e571..00000000 --- a/code/3rd_glpk/draft/glpios07.c +++ /dev/null @@ -1,551 +0,0 @@ -/* glpios07.c (mixed cover cut generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/*---------------------------------------------------------------------- --- COVER INEQUALITIES --- --- Consider the set of feasible solutions to 0-1 knapsack problem: --- --- sum a[j]*x[j] <= b, (1) --- j in J --- --- x[j] is binary, (2) --- --- where, wlog, we assume that a[j] > 0 (since 0-1 variables can be --- complemented) and a[j] <= b (since a[j] > b implies x[j] = 0). --- --- A set C within J is called a cover if --- --- sum a[j] > b. (3) --- j in C --- --- For any cover C the inequality --- --- sum x[j] <= |C| - 1 (4) --- j in C --- --- is called a cover inequality and is valid for (1)-(2). --- --- MIXED COVER INEQUALITIES --- --- Consider the set of feasible solutions to mixed knapsack problem: --- --- sum a[j]*x[j] + y <= b, (5) --- j in J --- --- x[j] is binary, (6) --- --- 0 <= y <= u is continuous, (7) --- --- where again we assume that a[j] > 0. --- --- Let C within J be some set. From (1)-(4) it follows that --- --- sum a[j] > b - y (8) --- j in C --- --- implies --- --- sum x[j] <= |C| - 1. (9) --- j in C --- --- Thus, we need to modify the inequality (9) in such a way that it be --- a constraint only if the condition (8) is satisfied. --- --- Consider the following inequality: --- --- sum x[j] <= |C| - t. (10) --- j in C --- --- If 0 < t <= 1, then (10) is equivalent to (9), because all x[j] are --- binary variables. On the other hand, if t <= 0, (10) being satisfied --- for any values of x[j] is not a constraint. --- --- Let --- --- t' = sum a[j] + y - b. (11) --- j in C --- --- It is understood that the condition t' > 0 is equivalent to (8). --- Besides, from (6)-(7) it follows that t' has an implied upper bound: --- --- t'max = sum a[j] + u - b. (12) --- j in C --- --- This allows to express the parameter t having desired properties: --- --- t = t' / t'max. (13) --- --- In fact, t <= 1 by definition, and t > 0 being equivalent to t' > 0 --- is equivalent to (8). --- --- Thus, the inequality (10), where t is given by formula (13) is valid --- for (5)-(7). --- --- Note that if u = 0, then y = 0, so t = 1, and the conditions (8) and --- (10) is transformed to the conditions (3) and (4). --- --- GENERATING MIXED COVER CUTS --- --- To generate a mixed cover cut in the form (10) we need to find such --- set C which satisfies to the inequality (8) and for which, in turn, --- the inequality (10) is violated in the current point. --- --- Substituting t from (13) to (10) gives: --- --- 1 --- sum x[j] <= |C| - ----- (sum a[j] + y - b), (14) --- j in C t'max j in C --- --- and finally we have the cut inequality in the standard form: --- --- sum x[j] + alfa * y <= beta, (15) --- j in C --- --- where: --- --- alfa = 1 / t'max, (16) --- --- beta = |C| - alfa * (sum a[j] - b). (17) --- j in C */ - -#if 1 -#define MAXTRY 1000 -#else -#define MAXTRY 10000 -#endif - -static int cover2(int n, double a[], double b, double u, double x[], - double y, int cov[], double *_alfa, double *_beta) -{ /* try to generate mixed cover cut using two-element cover */ - int i, j, try = 0, ret = 0; - double eps, alfa, beta, temp, rmax = 0.001; - eps = 0.001 * (1.0 + fabs(b)); - for (i = 0+1; i <= n; i++) - for (j = i+1; j <= n; j++) - { /* C = {i, j} */ - try++; - if (try > MAXTRY) goto done; - /* check if condition (8) is satisfied */ - if (a[i] + a[j] + y > b + eps) - { /* compute parameters for inequality (15) */ - temp = a[i] + a[j] - b; - alfa = 1.0 / (temp + u); - beta = 2.0 - alfa * temp; - /* compute violation of inequality (15) */ - temp = x[i] + x[j] + alfa * y - beta; - /* choose C providing maximum violation */ - if (rmax < temp) - { rmax = temp; - cov[1] = i; - cov[2] = j; - *_alfa = alfa; - *_beta = beta; - ret = 1; - } - } - } -done: return ret; -} - -static int cover3(int n, double a[], double b, double u, double x[], - double y, int cov[], double *_alfa, double *_beta) -{ /* try to generate mixed cover cut using three-element cover */ - int i, j, k, try = 0, ret = 0; - double eps, alfa, beta, temp, rmax = 0.001; - eps = 0.001 * (1.0 + fabs(b)); - for (i = 0+1; i <= n; i++) - for (j = i+1; j <= n; j++) - for (k = j+1; k <= n; k++) - { /* C = {i, j, k} */ - try++; - if (try > MAXTRY) goto done; - /* check if condition (8) is satisfied */ - if (a[i] + a[j] + a[k] + y > b + eps) - { /* compute parameters for inequality (15) */ - temp = a[i] + a[j] + a[k] - b; - alfa = 1.0 / (temp + u); - beta = 3.0 - alfa * temp; - /* compute violation of inequality (15) */ - temp = x[i] + x[j] + x[k] + alfa * y - beta; - /* choose C providing maximum violation */ - if (rmax < temp) - { rmax = temp; - cov[1] = i; - cov[2] = j; - cov[3] = k; - *_alfa = alfa; - *_beta = beta; - ret = 1; - } - } - } -done: return ret; -} - -static int cover4(int n, double a[], double b, double u, double x[], - double y, int cov[], double *_alfa, double *_beta) -{ /* try to generate mixed cover cut using four-element cover */ - int i, j, k, l, try = 0, ret = 0; - double eps, alfa, beta, temp, rmax = 0.001; - eps = 0.001 * (1.0 + fabs(b)); - for (i = 0+1; i <= n; i++) - for (j = i+1; j <= n; j++) - for (k = j+1; k <= n; k++) - for (l = k+1; l <= n; l++) - { /* C = {i, j, k, l} */ - try++; - if (try > MAXTRY) goto done; - /* check if condition (8) is satisfied */ - if (a[i] + a[j] + a[k] + a[l] + y > b + eps) - { /* compute parameters for inequality (15) */ - temp = a[i] + a[j] + a[k] + a[l] - b; - alfa = 1.0 / (temp + u); - beta = 4.0 - alfa * temp; - /* compute violation of inequality (15) */ - temp = x[i] + x[j] + x[k] + x[l] + alfa * y - beta; - /* choose C providing maximum violation */ - if (rmax < temp) - { rmax = temp; - cov[1] = i; - cov[2] = j; - cov[3] = k; - cov[4] = l; - *_alfa = alfa; - *_beta = beta; - ret = 1; - } - } - } -done: return ret; -} - -static int cover(int n, double a[], double b, double u, double x[], - double y, int cov[], double *alfa, double *beta) -{ /* try to generate mixed cover cut; - input (see (5)): - n is the number of binary variables; - a[1:n] are coefficients at binary variables; - b is the right-hand side; - u is upper bound of continuous variable; - x[1:n] are values of binary variables at current point; - y is value of continuous variable at current point; - output (see (15), (16), (17)): - cov[1:r] are indices of binary variables included in cover C, - where r is the set cardinality returned on exit; - alfa coefficient at continuous variable; - beta is the right-hand side; */ - int j; - /* perform some sanity checks */ - xassert(n >= 2); - for (j = 1; j <= n; j++) xassert(a[j] > 0.0); -#if 1 /* ??? */ - xassert(b > -1e-5); -#else - xassert(b > 0.0); -#endif - xassert(u >= 0.0); - for (j = 1; j <= n; j++) xassert(0.0 <= x[j] && x[j] <= 1.0); - xassert(0.0 <= y && y <= u); - /* try to generate mixed cover cut */ - if (cover2(n, a, b, u, x, y, cov, alfa, beta)) return 2; - if (cover3(n, a, b, u, x, y, cov, alfa, beta)) return 3; - if (cover4(n, a, b, u, x, y, cov, alfa, beta)) return 4; - return 0; -} - -/*---------------------------------------------------------------------- --- lpx_cover_cut - generate mixed cover cut. --- --- SYNOPSIS --- --- int lpx_cover_cut(LPX *lp, int len, int ind[], double val[], --- double work[]); --- --- DESCRIPTION --- --- The routine lpx_cover_cut generates a mixed cover cut for a given --- row of the MIP problem. --- --- The given row of the MIP problem should be explicitly specified in --- the form: --- --- sum{j in J} a[j]*x[j] <= b. (1) --- --- On entry indices (ordinal numbers) of structural variables, which --- have non-zero constraint coefficients, should be placed in locations --- ind[1], ..., ind[len], and corresponding constraint coefficients --- should be placed in locations val[1], ..., val[len]. The right-hand --- side b should be stored in location val[0]. --- --- The working array work should have at least nb locations, where nb --- is the number of binary variables in (1). --- --- The routine generates a mixed cover cut in the same form as (1) and --- stores the cut coefficients and right-hand side in the same way as --- just described above. --- --- RETURNS --- --- If the cutting plane has been successfully generated, the routine --- returns 1 <= len' <= n, which is the number of non-zero coefficients --- in the inequality constraint. Otherwise, the routine returns zero. */ - -static int lpx_cover_cut(glp_prob *lp, int len, int ind[], - double val[], double work[]) -{ int cov[1+4], j, k, nb, newlen, r; - double f_min, f_max, alfa, beta, u, *x = work, y; - /* substitute and remove fixed variables */ - newlen = 0; - for (k = 1; k <= len; k++) - { j = ind[k]; - if (glp_get_col_type(lp, j) == GLP_FX) - val[0] -= val[k] * glp_get_col_lb(lp, j); - else - { newlen++; - ind[newlen] = ind[k]; - val[newlen] = val[k]; - } - } - len = newlen; - /* move binary variables to the beginning of the list so that - elements 1, 2, ..., nb correspond to binary variables, and - elements nb+1, nb+2, ..., len correspond to rest variables */ - nb = 0; - for (k = 1; k <= len; k++) - { j = ind[k]; - if (glp_get_col_kind(lp, j) == GLP_BV) - { /* binary variable */ - int ind_k; - double val_k; - nb++; - ind_k = ind[nb], val_k = val[nb]; - ind[nb] = ind[k], val[nb] = val[k]; - ind[k] = ind_k, val[k] = val_k; - } - } - /* now the specified row has the form: - sum a[j]*x[j] + sum a[j]*y[j] <= b, - where x[j] are binary variables, y[j] are rest variables */ - /* at least two binary variables are needed */ - if (nb < 2) return 0; - /* compute implied lower and upper bounds for sum a[j]*y[j] */ - f_min = f_max = 0.0; - for (k = nb+1; k <= len; k++) - { j = ind[k]; - /* both bounds must be finite */ - if (glp_get_col_type(lp, j) != GLP_DB) return 0; - if (val[k] > 0.0) - { f_min += val[k] * glp_get_col_lb(lp, j); - f_max += val[k] * glp_get_col_ub(lp, j); - } - else - { f_min += val[k] * glp_get_col_ub(lp, j); - f_max += val[k] * glp_get_col_lb(lp, j); - } - } - /* sum a[j]*x[j] + sum a[j]*y[j] <= b ===> - sum a[j]*x[j] + (sum a[j]*y[j] - f_min) <= b - f_min ===> - sum a[j]*x[j] + y <= b - f_min, - where y = sum a[j]*y[j] - f_min; - note that 0 <= y <= u, u = f_max - f_min */ - /* determine upper bound of y */ - u = f_max - f_min; - /* determine value of y at the current point */ - y = 0.0; - for (k = nb+1; k <= len; k++) - { j = ind[k]; - y += val[k] * glp_get_col_prim(lp, j); - } - y -= f_min; - if (y < 0.0) y = 0.0; - if (y > u) y = u; - /* modify the right-hand side b */ - val[0] -= f_min; - /* now the transformed row has the form: - sum a[j]*x[j] + y <= b, where 0 <= y <= u */ - /* determine values of x[j] at the current point */ - for (k = 1; k <= nb; k++) - { j = ind[k]; - x[k] = glp_get_col_prim(lp, j); - if (x[k] < 0.0) x[k] = 0.0; - if (x[k] > 1.0) x[k] = 1.0; - } - /* if a[j] < 0, replace x[j] by its complement 1 - x'[j] */ - for (k = 1; k <= nb; k++) - { if (val[k] < 0.0) - { ind[k] = - ind[k]; - val[k] = - val[k]; - val[0] += val[k]; - x[k] = 1.0 - x[k]; - } - } - /* try to generate a mixed cover cut for the transformed row */ - r = cover(nb, val, val[0], u, x, y, cov, &alfa, &beta); - if (r == 0) return 0; - xassert(2 <= r && r <= 4); - /* now the cut is in the form: - sum{j in C} x[j] + alfa * y <= beta */ - /* store the right-hand side beta */ - ind[0] = 0, val[0] = beta; - /* restore the original ordinal numbers of x[j] */ - for (j = 1; j <= r; j++) cov[j] = ind[cov[j]]; - /* store cut coefficients at binary variables complementing back - the variables having negative row coefficients */ - xassert(r <= nb); - for (k = 1; k <= r; k++) - { if (cov[k] > 0) - { ind[k] = +cov[k]; - val[k] = +1.0; - } - else - { ind[k] = -cov[k]; - val[k] = -1.0; - val[0] -= 1.0; - } - } - /* substitute y = sum a[j]*y[j] - f_min */ - for (k = nb+1; k <= len; k++) - { r++; - ind[r] = ind[k]; - val[r] = alfa * val[k]; - } - val[0] += alfa * f_min; - xassert(r <= len); - len = r; - return len; -} - -/*---------------------------------------------------------------------- --- lpx_eval_row - compute explictily specified row. --- --- SYNOPSIS --- --- double lpx_eval_row(LPX *lp, int len, int ind[], double val[]); --- --- DESCRIPTION --- --- The routine lpx_eval_row computes the primal value of an explicitly --- specified row using current values of structural variables. --- --- The explicitly specified row may be thought as a linear form: --- --- y = a[1]*x[m+1] + a[2]*x[m+2] + ... + a[n]*x[m+n], --- --- where y is an auxiliary variable for this row, a[j] are coefficients --- of the linear form, x[m+j] are structural variables. --- --- On entry column indices and numerical values of non-zero elements of --- the row should be stored in locations ind[1], ..., ind[len] and --- val[1], ..., val[len], where len is the number of non-zero elements. --- The array ind and val are not changed on exit. --- --- RETURNS --- --- The routine returns a computed value of y, the auxiliary variable of --- the specified row. */ - -static double lpx_eval_row(glp_prob *lp, int len, int ind[], - double val[]) -{ int n = glp_get_num_cols(lp); - int j, k; - double sum = 0.0; - if (len < 0) - xerror("lpx_eval_row: len = %d; invalid row length\n", len); - for (k = 1; k <= len; k++) - { j = ind[k]; - if (!(1 <= j && j <= n)) - xerror("lpx_eval_row: j = %d; column number out of range\n", - j); - sum += val[k] * glp_get_col_prim(lp, j); - } - return sum; -} - -/*********************************************************************** -* NAME -* -* ios_cov_gen - generate mixed cover cuts -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_cov_gen(glp_tree *tree); -* -* DESCRIPTION -* -* The routine ios_cov_gen generates mixed cover cuts for the current -* point and adds them to the cut pool. */ - -void ios_cov_gen(glp_tree *tree) -{ glp_prob *prob = tree->mip; - int m = glp_get_num_rows(prob); - int n = glp_get_num_cols(prob); - int i, k, type, kase, len, *ind; - double r, *val, *work; - xassert(glp_get_status(prob) == GLP_OPT); - /* allocate working arrays */ - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - work = xcalloc(1+n, sizeof(double)); - /* look through all rows */ - for (i = 1; i <= m; i++) - for (kase = 1; kase <= 2; kase++) - { type = glp_get_row_type(prob, i); - if (kase == 1) - { /* consider rows of '<=' type */ - if (!(type == GLP_UP || type == GLP_DB)) continue; - len = glp_get_mat_row(prob, i, ind, val); - val[0] = glp_get_row_ub(prob, i); - } - else - { /* consider rows of '>=' type */ - if (!(type == GLP_LO || type == GLP_DB)) continue; - len = glp_get_mat_row(prob, i, ind, val); - for (k = 1; k <= len; k++) val[k] = - val[k]; - val[0] = - glp_get_row_lb(prob, i); - } - /* generate mixed cover cut: - sum{j in J} a[j] * x[j] <= b */ - len = lpx_cover_cut(prob, len, ind, val, work); - if (len == 0) continue; - /* at the current point the cut inequality is violated, i.e. - sum{j in J} a[j] * x[j] - b > 0 */ - r = lpx_eval_row(prob, len, ind, val) - val[0]; - if (r < 1e-3) continue; - /* add the cut to the cut pool */ - glp_ios_add_row(tree, NULL, GLP_RF_COV, 0, len, ind, val, - GLP_UP, val[0]); - } - /* free working arrays */ - xfree(ind); - xfree(val); - xfree(work); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios09.c b/code/3rd_glpk/draft/glpios09.c deleted file mode 100644 index d80ed9a3..00000000 --- a/code/3rd_glpk/draft/glpios09.c +++ /dev/null @@ -1,664 +0,0 @@ -/* glpios09.c (branching heuristics) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/*********************************************************************** -* NAME -* -* ios_choose_var - select variable to branch on -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_choose_var(glp_tree *T, int *next); -* -* The routine ios_choose_var chooses a variable from the candidate -* list to branch on. Additionally the routine provides a flag stored -* in the location next to suggests which of the child subproblems -* should be solved next. -* -* RETURNS -* -* The routine ios_choose_var returns the ordinal number of the column -* choosen. */ - -static int branch_first(glp_tree *T, int *next); -static int branch_last(glp_tree *T, int *next); -static int branch_mostf(glp_tree *T, int *next); -static int branch_drtom(glp_tree *T, int *next); - -int ios_choose_var(glp_tree *T, int *next) -{ int j; - if (T->parm->br_tech == GLP_BR_FFV) - { /* branch on first fractional variable */ - j = branch_first(T, next); - } - else if (T->parm->br_tech == GLP_BR_LFV) - { /* branch on last fractional variable */ - j = branch_last(T, next); - } - else if (T->parm->br_tech == GLP_BR_MFV) - { /* branch on most fractional variable */ - j = branch_mostf(T, next); - } - else if (T->parm->br_tech == GLP_BR_DTH) - { /* branch using the heuristic by Dreebeck and Tomlin */ - j = branch_drtom(T, next); - } - else if (T->parm->br_tech == GLP_BR_PCH) - { /* hybrid pseudocost heuristic */ - j = ios_pcost_branch(T, next); - } - else - xassert(T != T); - return j; -} - -/*********************************************************************** -* branch_first - choose first branching variable -* -* This routine looks up the list of structural variables and chooses -* the first one, which is of integer kind and has fractional value in -* optimal solution to the current LP relaxation. -* -* This routine also selects the branch to be solved next where integer -* infeasibility of the chosen variable is less than in other one. */ - -static int branch_first(glp_tree *T, int *_next) -{ int j, next; - double beta; - /* choose the column to branch on */ - for (j = 1; j <= T->n; j++) - if (T->non_int[j]) break; - xassert(1 <= j && j <= T->n); - /* select the branch to be solved next */ - beta = glp_get_col_prim(T->mip, j); - if (beta - floor(beta) < ceil(beta) - beta) - next = GLP_DN_BRNCH; - else - next = GLP_UP_BRNCH; - *_next = next; - return j; -} - -/*********************************************************************** -* branch_last - choose last branching variable -* -* This routine looks up the list of structural variables and chooses -* the last one, which is of integer kind and has fractional value in -* optimal solution to the current LP relaxation. -* -* This routine also selects the branch to be solved next where integer -* infeasibility of the chosen variable is less than in other one. */ - -static int branch_last(glp_tree *T, int *_next) -{ int j, next; - double beta; - /* choose the column to branch on */ - for (j = T->n; j >= 1; j--) - if (T->non_int[j]) break; - xassert(1 <= j && j <= T->n); - /* select the branch to be solved next */ - beta = glp_get_col_prim(T->mip, j); - if (beta - floor(beta) < ceil(beta) - beta) - next = GLP_DN_BRNCH; - else - next = GLP_UP_BRNCH; - *_next = next; - return j; -} - -/*********************************************************************** -* branch_mostf - choose most fractional branching variable -* -* This routine looks up the list of structural variables and chooses -* that one, which is of integer kind and has most fractional value in -* optimal solution to the current LP relaxation. -* -* This routine also selects the branch to be solved next where integer -* infeasibility of the chosen variable is less than in other one. -* -* (Alexander Martin notices that "...most infeasible is as good as -* random...".) */ - -static int branch_mostf(glp_tree *T, int *_next) -{ int j, jj, next; - double beta, most, temp; - /* choose the column to branch on */ - jj = 0, most = DBL_MAX; - for (j = 1; j <= T->n; j++) - { if (T->non_int[j]) - { beta = glp_get_col_prim(T->mip, j); - temp = floor(beta) + 0.5; - if (most > fabs(beta - temp)) - { jj = j, most = fabs(beta - temp); - if (beta < temp) - next = GLP_DN_BRNCH; - else - next = GLP_UP_BRNCH; - } - } - } - *_next = next; - return jj; -} - -/*********************************************************************** -* branch_drtom - choose branching var using Driebeck-Tomlin heuristic -* -* This routine chooses a structural variable, which is required to be -* integral and has fractional value in optimal solution of the current -* LP relaxation, using a heuristic proposed by Driebeck and Tomlin. -* -* The routine also selects the branch to be solved next, again due to -* Driebeck and Tomlin. -* -* This routine is based on the heuristic proposed in: -* -* Driebeck N.J. An algorithm for the solution of mixed-integer -* programming problems, Management Science, 12: 576-87 (1966); -* -* and improved in: -* -* Tomlin J.A. Branch and bound methods for integer and non-convex -* programming, in J.Abadie (ed.), Integer and Nonlinear Programming, -* North-Holland, Amsterdam, pp. 437-50 (1970). -* -* Must note that this heuristic is time-expensive, because computing -* one-step degradation (see the routine below) requires one BTRAN for -* each fractional-valued structural variable. */ - -static int branch_drtom(glp_tree *T, int *_next) -{ glp_prob *mip = T->mip; - int m = mip->m; - int n = mip->n; - unsigned char *non_int = T->non_int; - int j, jj, k, t, next, kase, len, stat, *ind; - double x, dk, alfa, delta_j, delta_k, delta_z, dz_dn, dz_up, - dd_dn, dd_up, degrad, *val; - /* basic solution of LP relaxation must be optimal */ - xassert(glp_get_status(mip) == GLP_OPT); - /* allocate working arrays */ - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - /* nothing has been chosen so far */ - jj = 0, degrad = -1.0; - /* walk through the list of columns (structural variables) */ - for (j = 1; j <= n; j++) - { /* if j-th column is not marked as fractional, skip it */ - if (!non_int[j]) continue; - /* obtain (fractional) value of j-th column in basic solution - of LP relaxation */ - x = glp_get_col_prim(mip, j); - /* since the value of j-th column is fractional, the column is - basic; compute corresponding row of the simplex table */ - len = glp_eval_tab_row(mip, m+j, ind, val); - /* the following fragment computes a change in the objective - function: delta Z = new Z - old Z, where old Z is the - objective value in the current optimal basis, and new Z is - the objective value in the adjacent basis, for two cases: - 1) if new upper bound ub' = floor(x[j]) is introduced for - j-th column (down branch); - 2) if new lower bound lb' = ceil(x[j]) is introduced for - j-th column (up branch); - since in both cases the solution remaining dual feasible - becomes primal infeasible, one implicit simplex iteration - is performed to determine the change delta Z; - it is obvious that new Z, which is never better than old Z, - is a lower (minimization) or upper (maximization) bound of - the objective function for down- and up-branches. */ - for (kase = -1; kase <= +1; kase += 2) - { /* if kase < 0, the new upper bound of x[j] is introduced; - in this case x[j] should decrease in order to leave the - basis and go to its new upper bound */ - /* if kase > 0, the new lower bound of x[j] is introduced; - in this case x[j] should increase in order to leave the - basis and go to its new lower bound */ - /* apply the dual ratio test in order to determine which - auxiliary or structural variable should enter the basis - to keep dual feasibility */ - k = glp_dual_rtest(mip, len, ind, val, kase, 1e-9); - if (k != 0) k = ind[k]; - /* if no non-basic variable has been chosen, LP relaxation - of corresponding branch being primal infeasible and dual - unbounded has no primal feasible solution; in this case - the change delta Z is formally set to infinity */ - if (k == 0) - { delta_z = - (T->mip->dir == GLP_MIN ? +DBL_MAX : -DBL_MAX); - goto skip; - } - /* row of the simplex table that corresponds to non-basic - variable x[k] choosen by the dual ratio test is: - x[j] = ... + alfa * x[k] + ... - where alfa is the influence coefficient (an element of - the simplex table row) */ - /* determine the coefficient alfa */ - for (t = 1; t <= len; t++) if (ind[t] == k) break; - xassert(1 <= t && t <= len); - alfa = val[t]; - /* since in the adjacent basis the variable x[j] becomes - non-basic, knowing its value in the current basis we can - determine its change delta x[j] = new x[j] - old x[j] */ - delta_j = (kase < 0 ? floor(x) : ceil(x)) - x; - /* and knowing the coefficient alfa we can determine the - corresponding change delta x[k] = new x[k] - old x[k], - where old x[k] is a value of x[k] in the current basis, - and new x[k] is a value of x[k] in the adjacent basis */ - delta_k = delta_j / alfa; - /* Tomlin noticed that if the variable x[k] is of integer - kind, its change cannot be less (eventually) than one in - the magnitude */ - if (k > m && glp_get_col_kind(mip, k-m) != GLP_CV) - { /* x[k] is structural integer variable */ - if (fabs(delta_k - floor(delta_k + 0.5)) > 1e-3) - { if (delta_k > 0.0) - delta_k = ceil(delta_k); /* +3.14 -> +4 */ - else - delta_k = floor(delta_k); /* -3.14 -> -4 */ - } - } - /* now determine the status and reduced cost of x[k] in the - current basis */ - if (k <= m) - { stat = glp_get_row_stat(mip, k); - dk = glp_get_row_dual(mip, k); - } - else - { stat = glp_get_col_stat(mip, k-m); - dk = glp_get_col_dual(mip, k-m); - } - /* if the current basis is dual degenerate, some reduced - costs which are close to zero may have wrong sign due to - round-off errors, so correct the sign of d[k] */ - switch (T->mip->dir) - { case GLP_MIN: - if (stat == GLP_NL && dk < 0.0 || - stat == GLP_NU && dk > 0.0 || - stat == GLP_NF) dk = 0.0; - break; - case GLP_MAX: - if (stat == GLP_NL && dk > 0.0 || - stat == GLP_NU && dk < 0.0 || - stat == GLP_NF) dk = 0.0; - break; - default: - xassert(T != T); - } - /* now knowing the change of x[k] and its reduced cost d[k] - we can compute the corresponding change in the objective - function delta Z = new Z - old Z = d[k] * delta x[k]; - note that due to Tomlin's modification new Z can be even - worse than in the adjacent basis */ - delta_z = dk * delta_k; -skip: /* new Z is never better than old Z, therefore the change - delta Z is always non-negative (in case of minimization) - or non-positive (in case of maximization) */ - switch (T->mip->dir) - { case GLP_MIN: xassert(delta_z >= 0.0); break; - case GLP_MAX: xassert(delta_z <= 0.0); break; - default: xassert(T != T); - } - /* save the change in the objective fnction for down- and - up-branches, respectively */ - if (kase < 0) dz_dn = delta_z; else dz_up = delta_z; - } - /* thus, in down-branch no integer feasible solution can be - better than Z + dz_dn, and in up-branch no integer feasible - solution can be better than Z + dz_up, where Z is value of - the objective function in the current basis */ - /* following the heuristic by Driebeck and Tomlin we choose a - column (i.e. structural variable) which provides largest - degradation of the objective function in some of branches; - besides, we select the branch with smaller degradation to - be solved next and keep other branch with larger degradation - in the active list hoping to minimize the number of further - backtrackings */ - if (degrad < fabs(dz_dn) || degrad < fabs(dz_up)) - { jj = j; - if (fabs(dz_dn) < fabs(dz_up)) - { /* select down branch to be solved next */ - next = GLP_DN_BRNCH; - degrad = fabs(dz_up); - } - else - { /* select up branch to be solved next */ - next = GLP_UP_BRNCH; - degrad = fabs(dz_dn); - } - /* save the objective changes for printing */ - dd_dn = dz_dn, dd_up = dz_up; - /* if down- or up-branch has no feasible solution, we does - not need to consider other candidates (in principle, the - corresponding branch could be pruned right now) */ - if (degrad == DBL_MAX) break; - } - } - /* free working arrays */ - xfree(ind); - xfree(val); - /* something must be chosen */ - xassert(1 <= jj && jj <= n); -#if 1 /* 02/XI-2009 */ - if (degrad < 1e-6 * (1.0 + 0.001 * fabs(mip->obj_val))) - { jj = branch_mostf(T, &next); - goto done; - } -#endif - if (T->parm->msg_lev >= GLP_MSG_DBG) - { xprintf("branch_drtom: column %d chosen to branch on\n", jj); - if (fabs(dd_dn) == DBL_MAX) - xprintf("branch_drtom: down-branch is infeasible\n"); - else - xprintf("branch_drtom: down-branch bound is %.9e\n", - glp_get_obj_val(mip) + dd_dn); - if (fabs(dd_up) == DBL_MAX) - xprintf("branch_drtom: up-branch is infeasible\n"); - else - xprintf("branch_drtom: up-branch bound is %.9e\n", - glp_get_obj_val(mip) + dd_up); - } -done: *_next = next; - return jj; -} - -/**********************************************************************/ - -struct csa -{ /* common storage area */ - int *dn_cnt; /* int dn_cnt[1+n]; */ - /* dn_cnt[j] is the number of subproblems, whose LP relaxations - have been solved and which are down-branches for variable x[j]; - dn_cnt[j] = 0 means the down pseudocost is uninitialized */ - double *dn_sum; /* double dn_sum[1+n]; */ - /* dn_sum[j] is the sum of per unit degradations of the objective - over all dn_cnt[j] subproblems */ - int *up_cnt; /* int up_cnt[1+n]; */ - /* up_cnt[j] is the number of subproblems, whose LP relaxations - have been solved and which are up-branches for variable x[j]; - up_cnt[j] = 0 means the up pseudocost is uninitialized */ - double *up_sum; /* double up_sum[1+n]; */ - /* up_sum[j] is the sum of per unit degradations of the objective - over all up_cnt[j] subproblems */ -}; - -void *ios_pcost_init(glp_tree *tree) -{ /* initialize working data used on pseudocost branching */ - struct csa *csa; - int n = tree->n, j; - csa = xmalloc(sizeof(struct csa)); - csa->dn_cnt = xcalloc(1+n, sizeof(int)); - csa->dn_sum = xcalloc(1+n, sizeof(double)); - csa->up_cnt = xcalloc(1+n, sizeof(int)); - csa->up_sum = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) - { csa->dn_cnt[j] = csa->up_cnt[j] = 0; - csa->dn_sum[j] = csa->up_sum[j] = 0.0; - } - return csa; -} - -static double eval_degrad(glp_prob *P, int j, double bnd) -{ /* compute degradation of the objective on fixing x[j] at given - value with a limited number of dual simplex iterations */ - /* this routine fixes column x[j] at specified value bnd, - solves resulting LP, and returns a lower bound to degradation - of the objective, degrad >= 0 */ - glp_prob *lp; - glp_smcp parm; - int ret; - double degrad; - /* the current basis must be optimal */ - xassert(glp_get_status(P) == GLP_OPT); - /* create a copy of P */ - lp = glp_create_prob(); - glp_copy_prob(lp, P, 0); - /* fix column x[j] at specified value */ - glp_set_col_bnds(lp, j, GLP_FX, bnd, bnd); - /* try to solve resulting LP */ - glp_init_smcp(&parm); - parm.msg_lev = GLP_MSG_OFF; - parm.meth = GLP_DUAL; - parm.it_lim = 30; - parm.out_dly = 1000; - parm.meth = GLP_DUAL; - ret = glp_simplex(lp, &parm); - if (ret == 0 || ret == GLP_EITLIM) - { if (glp_get_prim_stat(lp) == GLP_NOFEAS) - { /* resulting LP has no primal feasible solution */ - degrad = DBL_MAX; - } - else if (glp_get_dual_stat(lp) == GLP_FEAS) - { /* resulting basis is optimal or at least dual feasible, - so we have the correct lower bound to degradation */ - if (P->dir == GLP_MIN) - degrad = lp->obj_val - P->obj_val; - else if (P->dir == GLP_MAX) - degrad = P->obj_val - lp->obj_val; - else - xassert(P != P); - /* degradation cannot be negative by definition */ - /* note that the lower bound to degradation may be close - to zero even if its exact value is zero due to round-off - errors on computing the objective value */ - if (degrad < 1e-6 * (1.0 + 0.001 * fabs(P->obj_val))) - degrad = 0.0; - } - else - { /* the final basis reported by the simplex solver is dual - infeasible, so we cannot determine a non-trivial lower - bound to degradation */ - degrad = 0.0; - } - } - else - { /* the simplex solver failed */ - degrad = 0.0; - } - /* delete the copy of P */ - glp_delete_prob(lp); - return degrad; -} - -void ios_pcost_update(glp_tree *tree) -{ /* update history information for pseudocost branching */ - /* this routine is called every time when LP relaxation of the - current subproblem has been solved to optimality with all lazy - and cutting plane constraints included */ - int j; - double dx, dz, psi; - struct csa *csa = tree->pcost; - xassert(csa != NULL); - xassert(tree->curr != NULL); - /* if the current subproblem is the root, skip updating */ - if (tree->curr->up == NULL) goto skip; - /* determine branching variable x[j], which was used in the - parent subproblem to create the current subproblem */ - j = tree->curr->up->br_var; - xassert(1 <= j && j <= tree->n); - /* determine the change dx[j] = new x[j] - old x[j], - where new x[j] is a value of x[j] in optimal solution to LP - relaxation of the current subproblem, old x[j] is a value of - x[j] in optimal solution to LP relaxation of the parent - subproblem */ - dx = tree->mip->col[j]->prim - tree->curr->up->br_val; - xassert(dx != 0.0); - /* determine corresponding change dz = new dz - old dz in the - objective function value */ - dz = tree->mip->obj_val - tree->curr->up->lp_obj; - /* determine per unit degradation of the objective function */ - psi = fabs(dz / dx); - /* update history information */ - if (dx < 0.0) - { /* the current subproblem is down-branch */ - csa->dn_cnt[j]++; - csa->dn_sum[j] += psi; - } - else /* dx > 0.0 */ - { /* the current subproblem is up-branch */ - csa->up_cnt[j]++; - csa->up_sum[j] += psi; - } -skip: return; -} - -void ios_pcost_free(glp_tree *tree) -{ /* free working area used on pseudocost branching */ - struct csa *csa = tree->pcost; - xassert(csa != NULL); - xfree(csa->dn_cnt); - xfree(csa->dn_sum); - xfree(csa->up_cnt); - xfree(csa->up_sum); - xfree(csa); - tree->pcost = NULL; - return; -} - -static double eval_psi(glp_tree *T, int j, int brnch) -{ /* compute estimation of pseudocost of variable x[j] for down- - or up-branch */ - struct csa *csa = T->pcost; - double beta, degrad, psi; - xassert(csa != NULL); - xassert(1 <= j && j <= T->n); - if (brnch == GLP_DN_BRNCH) - { /* down-branch */ - if (csa->dn_cnt[j] == 0) - { /* initialize down pseudocost */ - beta = T->mip->col[j]->prim; - degrad = eval_degrad(T->mip, j, floor(beta)); - if (degrad == DBL_MAX) - { psi = DBL_MAX; - goto done; - } - csa->dn_cnt[j] = 1; - csa->dn_sum[j] = degrad / (beta - floor(beta)); - } - psi = csa->dn_sum[j] / (double)csa->dn_cnt[j]; - } - else if (brnch == GLP_UP_BRNCH) - { /* up-branch */ - if (csa->up_cnt[j] == 0) - { /* initialize up pseudocost */ - beta = T->mip->col[j]->prim; - degrad = eval_degrad(T->mip, j, ceil(beta)); - if (degrad == DBL_MAX) - { psi = DBL_MAX; - goto done; - } - csa->up_cnt[j] = 1; - csa->up_sum[j] = degrad / (ceil(beta) - beta); - } - psi = csa->up_sum[j] / (double)csa->up_cnt[j]; - } - else - xassert(brnch != brnch); -done: return psi; -} - -static void progress(glp_tree *T) -{ /* display progress of pseudocost initialization */ - struct csa *csa = T->pcost; - int j, nv = 0, ni = 0; - for (j = 1; j <= T->n; j++) - { if (glp_ios_can_branch(T, j)) - { nv++; - if (csa->dn_cnt[j] > 0 && csa->up_cnt[j] > 0) ni++; - } - } - xprintf("Pseudocosts initialized for %d of %d variables\n", - ni, nv); - return; -} - -int ios_pcost_branch(glp_tree *T, int *_next) -{ /* choose branching variable with pseudocost branching */ -#if 0 /* 10/VI-2013 */ - glp_long t = xtime(); -#else - double t = xtime(); -#endif - int j, jjj, sel; - double beta, psi, d1, d2, d, dmax; - /* initialize the working arrays */ - if (T->pcost == NULL) - T->pcost = ios_pcost_init(T); - /* nothing has been chosen so far */ - jjj = 0, dmax = -1.0; - /* go through the list of branching candidates */ - for (j = 1; j <= T->n; j++) - { if (!glp_ios_can_branch(T, j)) continue; - /* determine primal value of x[j] in optimal solution to LP - relaxation of the current subproblem */ - beta = T->mip->col[j]->prim; - /* estimate pseudocost of x[j] for down-branch */ - psi = eval_psi(T, j, GLP_DN_BRNCH); - if (psi == DBL_MAX) - { /* down-branch has no primal feasible solution */ - jjj = j, sel = GLP_DN_BRNCH; - goto done; - } - /* estimate degradation of the objective for down-branch */ - d1 = psi * (beta - floor(beta)); - /* estimate pseudocost of x[j] for up-branch */ - psi = eval_psi(T, j, GLP_UP_BRNCH); - if (psi == DBL_MAX) - { /* up-branch has no primal feasible solution */ - jjj = j, sel = GLP_UP_BRNCH; - goto done; - } - /* estimate degradation of the objective for up-branch */ - d2 = psi * (ceil(beta) - beta); - /* determine d = max(d1, d2) */ - d = (d1 > d2 ? d1 : d2); - /* choose x[j] which provides maximal estimated degradation of - the objective either in down- or up-branch */ - if (dmax < d) - { dmax = d; - jjj = j; - /* continue the search from a subproblem, where degradation - is less than in other one */ - sel = (d1 <= d2 ? GLP_DN_BRNCH : GLP_UP_BRNCH); - } - /* display progress of pseudocost initialization */ - if (T->parm->msg_lev >= GLP_ON) - { if (xdifftime(xtime(), t) >= 10.0) - { progress(T); - t = xtime(); - } - } - } - if (dmax == 0.0) - { /* no degradation is indicated; choose a variable having most - fractional value */ - jjj = branch_mostf(T, &sel); - } -done: *_next = sel; - return jjj; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios11.c b/code/3rd_glpk/draft/glpios11.c deleted file mode 100644 index 09fccef6..00000000 --- a/code/3rd_glpk/draft/glpios11.c +++ /dev/null @@ -1,435 +0,0 @@ -/* glpios11.c (process cuts stored in the local cut pool) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2017, 2018 Andrew Makhorin, Department for -* Applied Informatics, Moscow Aviation Institute, Moscow, Russia. All -* rights reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "draft.h" -#include "env.h" -#include "ios.h" - -/*********************************************************************** -* NAME -* -* ios_process_cuts - process cuts stored in the local cut pool -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_process_cuts(glp_tree *T); -* -* DESCRIPTION -* -* The routine ios_process_cuts analyzes each cut currently stored in -* the local cut pool, which must be non-empty, and either adds the cut -* to the current subproblem or just discards it. All cuts are assumed -* to be locally valid. On exit the local cut pool remains unchanged. -* -* REFERENCES -* -* 1. E.Balas, S.Ceria, G.Cornuejols, "Mixed 0-1 Programming by -* Lift-and-Project in a Branch-and-Cut Framework", Management Sc., -* 42 (1996) 1229-1246. -* -* 2. G.Andreello, A.Caprara, and M.Fischetti, "Embedding Cuts in -* a Branch&Cut Framework: a Computational Study with {0,1/2}-Cuts", -* Preliminary Draft, October 28, 2003, pp.6-8. */ - -struct info -{ /* estimated cut efficiency */ - IOSCUT *cut; - /* pointer to cut in the cut pool */ - char flag; - /* if this flag is set, the cut is included into the current - subproblem */ - double eff; - /* cut efficacy (normalized residual) */ - double deg; - /* lower bound to objective degradation */ -}; - -static int CDECL fcmp(const void *arg1, const void *arg2) -{ const struct info *info1 = arg1, *info2 = arg2; - if (info1->deg == 0.0 && info2->deg == 0.0) - { if (info1->eff > info2->eff) return -1; - if (info1->eff < info2->eff) return +1; - } - else - { if (info1->deg > info2->deg) return -1; - if (info1->deg < info2->deg) return +1; - } - return 0; -} - -static double parallel(IOSCUT *a, IOSCUT *b, double work[]); - -#ifdef NEW_LOCAL /* 02/II-2018 */ -void ios_process_cuts(glp_tree *T) -{ IOSPOOL *pool; - IOSCUT *cut; - GLPAIJ *aij; - struct info *info; - int k, kk, max_cuts, len, ret, *ind; - double *val, *work, rhs; - /* the current subproblem must exist */ - xassert(T->curr != NULL); - /* the pool must exist and be non-empty */ - pool = T->local; - xassert(pool != NULL); - xassert(pool->m > 0); - /* allocate working arrays */ - info = xcalloc(1+pool->m, sizeof(struct info)); - ind = xcalloc(1+T->n, sizeof(int)); - val = xcalloc(1+T->n, sizeof(double)); - work = xcalloc(1+T->n, sizeof(double)); - for (k = 1; k <= T->n; k++) work[k] = 0.0; - /* build the list of cuts stored in the cut pool */ - for (k = 1; k <= pool->m; k++) - info[k].cut = pool->row[k], info[k].flag = 0; - /* estimate efficiency of all cuts in the cut pool */ - for (k = 1; k <= pool->m; k++) - { double temp, dy, dz; - cut = info[k].cut; - /* build the vector of cut coefficients and compute its - Euclidean norm */ - len = 0; temp = 0.0; - for (aij = cut->ptr; aij != NULL; aij = aij->r_next) - { xassert(1 <= aij->col->j && aij->col->j <= T->n); - len++, ind[len] = aij->col->j, val[len] = aij->val; - temp += aij->val * aij->val; - } - if (temp < DBL_EPSILON * DBL_EPSILON) temp = DBL_EPSILON; - /* transform the cut to express it only through non-basic - (auxiliary and structural) variables */ - len = glp_transform_row(T->mip, len, ind, val); - /* determine change in the cut value and in the objective - value for the adjacent basis by simulating one step of the - dual simplex */ - switch (cut->type) - { case GLP_LO: rhs = cut->lb; break; - case GLP_UP: rhs = cut->ub; break; - default: xassert(cut != cut); - } - ret = _glp_analyze_row(T->mip, len, ind, val, cut->type, - rhs, 1e-9, NULL, NULL, NULL, NULL, &dy, &dz); - /* determine normalized residual and lower bound to objective - degradation */ - if (ret == 0) - { info[k].eff = fabs(dy) / sqrt(temp); - /* if some reduced costs violates (slightly) their zero - bounds (i.e. have wrong signs) due to round-off errors, - dz also may have wrong sign being close to zero */ - if (T->mip->dir == GLP_MIN) - { if (dz < 0.0) dz = 0.0; - info[k].deg = + dz; - } - else /* GLP_MAX */ - { if (dz > 0.0) dz = 0.0; - info[k].deg = - dz; - } - } - else if (ret == 1) - { /* the constraint is not violated at the current point */ - info[k].eff = info[k].deg = 0.0; - } - else if (ret == 2) - { /* no dual feasible adjacent basis exists */ - info[k].eff = 1.0; - info[k].deg = DBL_MAX; - } - else - xassert(ret != ret); - /* if the degradation is too small, just ignore it */ - if (info[k].deg < 0.01) info[k].deg = 0.0; - } - /* sort the list of cuts by decreasing objective degradation and - then by decreasing efficacy */ - qsort(&info[1], pool->m, sizeof(struct info), fcmp); - /* only first (most efficient) max_cuts in the list are qualified - as candidates to be added to the current subproblem */ - max_cuts = (T->curr->level == 0 ? 90 : 10); - if (max_cuts > pool->m) max_cuts = pool->m; - /* add cuts to the current subproblem */ -#if 0 - xprintf("*** adding cuts ***\n"); -#endif - for (k = 1; k <= max_cuts; k++) - { int i, len; - /* if this cut seems to be inefficient, skip it */ - if (info[k].deg < 0.01 && info[k].eff < 0.01) continue; - /* if the angle between this cut and every other cut included - in the current subproblem is small, skip this cut */ - for (kk = 1; kk < k; kk++) - { if (info[kk].flag) - { if (parallel(info[k].cut, info[kk].cut, work) > 0.90) - break; - } - } - if (kk < k) continue; - /* add this cut to the current subproblem */ -#if 0 - xprintf("eff = %g; deg = %g\n", info[k].eff, info[k].deg); -#endif - cut = info[k].cut, info[k].flag = 1; - i = glp_add_rows(T->mip, 1); - if (cut->name != NULL) - glp_set_row_name(T->mip, i, cut->name); - xassert(T->mip->row[i]->origin == GLP_RF_CUT); - T->mip->row[i]->klass = cut->klass; - len = 0; - for (aij = cut->ptr; aij != NULL; aij = aij->r_next) - len++, ind[len] = aij->col->j, val[len] = aij->val; - glp_set_mat_row(T->mip, i, len, ind, val); - switch (cut->type) - { case GLP_LO: rhs = cut->lb; break; - case GLP_UP: rhs = cut->ub; break; - default: xassert(cut != cut); - } - glp_set_row_bnds(T->mip, i, cut->type, rhs, rhs); - } - /* free working arrays */ - xfree(info); - xfree(ind); - xfree(val); - xfree(work); - return; -} -#else -void ios_process_cuts(glp_tree *T) -{ IOSPOOL *pool; - IOSCUT *cut; - IOSAIJ *aij; - struct info *info; - int k, kk, max_cuts, len, ret, *ind; - double *val, *work; - /* the current subproblem must exist */ - xassert(T->curr != NULL); - /* the pool must exist and be non-empty */ - pool = T->local; - xassert(pool != NULL); - xassert(pool->size > 0); - /* allocate working arrays */ - info = xcalloc(1+pool->size, sizeof(struct info)); - ind = xcalloc(1+T->n, sizeof(int)); - val = xcalloc(1+T->n, sizeof(double)); - work = xcalloc(1+T->n, sizeof(double)); - for (k = 1; k <= T->n; k++) work[k] = 0.0; - /* build the list of cuts stored in the cut pool */ - for (k = 0, cut = pool->head; cut != NULL; cut = cut->next) - k++, info[k].cut = cut, info[k].flag = 0; - xassert(k == pool->size); - /* estimate efficiency of all cuts in the cut pool */ - for (k = 1; k <= pool->size; k++) - { double temp, dy, dz; - cut = info[k].cut; - /* build the vector of cut coefficients and compute its - Euclidean norm */ - len = 0; temp = 0.0; - for (aij = cut->ptr; aij != NULL; aij = aij->next) - { xassert(1 <= aij->j && aij->j <= T->n); - len++, ind[len] = aij->j, val[len] = aij->val; - temp += aij->val * aij->val; - } - if (temp < DBL_EPSILON * DBL_EPSILON) temp = DBL_EPSILON; - /* transform the cut to express it only through non-basic - (auxiliary and structural) variables */ - len = glp_transform_row(T->mip, len, ind, val); - /* determine change in the cut value and in the objective - value for the adjacent basis by simulating one step of the - dual simplex */ - ret = _glp_analyze_row(T->mip, len, ind, val, cut->type, - cut->rhs, 1e-9, NULL, NULL, NULL, NULL, &dy, &dz); - /* determine normalized residual and lower bound to objective - degradation */ - if (ret == 0) - { info[k].eff = fabs(dy) / sqrt(temp); - /* if some reduced costs violates (slightly) their zero - bounds (i.e. have wrong signs) due to round-off errors, - dz also may have wrong sign being close to zero */ - if (T->mip->dir == GLP_MIN) - { if (dz < 0.0) dz = 0.0; - info[k].deg = + dz; - } - else /* GLP_MAX */ - { if (dz > 0.0) dz = 0.0; - info[k].deg = - dz; - } - } - else if (ret == 1) - { /* the constraint is not violated at the current point */ - info[k].eff = info[k].deg = 0.0; - } - else if (ret == 2) - { /* no dual feasible adjacent basis exists */ - info[k].eff = 1.0; - info[k].deg = DBL_MAX; - } - else - xassert(ret != ret); - /* if the degradation is too small, just ignore it */ - if (info[k].deg < 0.01) info[k].deg = 0.0; - } - /* sort the list of cuts by decreasing objective degradation and - then by decreasing efficacy */ - qsort(&info[1], pool->size, sizeof(struct info), fcmp); - /* only first (most efficient) max_cuts in the list are qualified - as candidates to be added to the current subproblem */ - max_cuts = (T->curr->level == 0 ? 90 : 10); - if (max_cuts > pool->size) max_cuts = pool->size; - /* add cuts to the current subproblem */ -#if 0 - xprintf("*** adding cuts ***\n"); -#endif - for (k = 1; k <= max_cuts; k++) - { int i, len; - /* if this cut seems to be inefficient, skip it */ - if (info[k].deg < 0.01 && info[k].eff < 0.01) continue; - /* if the angle between this cut and every other cut included - in the current subproblem is small, skip this cut */ - for (kk = 1; kk < k; kk++) - { if (info[kk].flag) - { if (parallel(info[k].cut, info[kk].cut, work) > 0.90) - break; - } - } - if (kk < k) continue; - /* add this cut to the current subproblem */ -#if 0 - xprintf("eff = %g; deg = %g\n", info[k].eff, info[k].deg); -#endif - cut = info[k].cut, info[k].flag = 1; - i = glp_add_rows(T->mip, 1); - if (cut->name != NULL) - glp_set_row_name(T->mip, i, cut->name); - xassert(T->mip->row[i]->origin == GLP_RF_CUT); - T->mip->row[i]->klass = cut->klass; - len = 0; - for (aij = cut->ptr; aij != NULL; aij = aij->next) - len++, ind[len] = aij->j, val[len] = aij->val; - glp_set_mat_row(T->mip, i, len, ind, val); - xassert(cut->type == GLP_LO || cut->type == GLP_UP); - glp_set_row_bnds(T->mip, i, cut->type, cut->rhs, cut->rhs); - } - /* free working arrays */ - xfree(info); - xfree(ind); - xfree(val); - xfree(work); - return; -} -#endif - -#if 0 -/*********************************************************************** -* Given a cut a * x >= b (<= b) the routine efficacy computes the cut -* efficacy as follows: -* -* eff = d * (a * x~ - b) / ||a||, -* -* where d is -1 (in case of '>= b') or +1 (in case of '<= b'), x~ is -* the vector of values of structural variables in optimal solution to -* LP relaxation of the current subproblem, ||a|| is the Euclidean norm -* of the vector of cut coefficients. -* -* If the cut is violated at point x~, the efficacy eff is positive, -* and its value is the Euclidean distance between x~ and the cut plane -* a * x = b in the space of structural variables. -* -* Following geometrical intuition, it is quite natural to consider -* this distance as a first-order measure of the expected efficacy of -* the cut: the larger the distance the better the cut [1]. */ - -static double efficacy(glp_tree *T, IOSCUT *cut) -{ glp_prob *mip = T->mip; - IOSAIJ *aij; - double s = 0.0, t = 0.0, temp; - for (aij = cut->ptr; aij != NULL; aij = aij->next) - { xassert(1 <= aij->j && aij->j <= mip->n); - s += aij->val * mip->col[aij->j]->prim; - t += aij->val * aij->val; - } - temp = sqrt(t); - if (temp < DBL_EPSILON) temp = DBL_EPSILON; - if (cut->type == GLP_LO) - temp = (s >= cut->rhs ? 0.0 : (cut->rhs - s) / temp); - else if (cut->type == GLP_UP) - temp = (s <= cut->rhs ? 0.0 : (s - cut->rhs) / temp); - else - xassert(cut != cut); - return temp; -} -#endif - -/*********************************************************************** -* Given two cuts a1 * x >= b1 (<= b1) and a2 * x >= b2 (<= b2) the -* routine parallel computes the cosine of angle between the cut planes -* a1 * x = b1 and a2 * x = b2 (which is the acute angle between two -* normals to these planes) in the space of structural variables as -* follows: -* -* cos phi = (a1' * a2) / (||a1|| * ||a2||), -* -* where (a1' * a2) is a dot product of vectors of cut coefficients, -* ||a1|| and ||a2|| are Euclidean norms of vectors a1 and a2. -* -* Note that requirement cos phi = 0 forces the cuts to be orthogonal, -* i.e. with disjoint support, while requirement cos phi <= 0.999 means -* only avoiding duplicate (parallel) cuts [1]. */ - -#ifdef NEW_LOCAL /* 02/II-2018 */ -static double parallel(IOSCUT *a, IOSCUT *b, double work[]) -{ GLPAIJ *aij; - double s = 0.0, sa = 0.0, sb = 0.0, temp; - for (aij = a->ptr; aij != NULL; aij = aij->r_next) - { work[aij->col->j] = aij->val; - sa += aij->val * aij->val; - } - for (aij = b->ptr; aij != NULL; aij = aij->r_next) - { s += work[aij->col->j] * aij->val; - sb += aij->val * aij->val; - } - for (aij = a->ptr; aij != NULL; aij = aij->r_next) - work[aij->col->j] = 0.0; - temp = sqrt(sa) * sqrt(sb); - if (temp < DBL_EPSILON * DBL_EPSILON) temp = DBL_EPSILON; - return s / temp; -} -#else -static double parallel(IOSCUT *a, IOSCUT *b, double work[]) -{ IOSAIJ *aij; - double s = 0.0, sa = 0.0, sb = 0.0, temp; - for (aij = a->ptr; aij != NULL; aij = aij->next) - { work[aij->j] = aij->val; - sa += aij->val * aij->val; - } - for (aij = b->ptr; aij != NULL; aij = aij->next) - { s += work[aij->j] * aij->val; - sb += aij->val * aij->val; - } - for (aij = a->ptr; aij != NULL; aij = aij->next) - work[aij->j] = 0.0; - temp = sqrt(sa) * sqrt(sb); - if (temp < DBL_EPSILON * DBL_EPSILON) temp = DBL_EPSILON; - return s / temp; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpios12.c b/code/3rd_glpk/draft/glpios12.c deleted file mode 100644 index bec6fa2c..00000000 --- a/code/3rd_glpk/draft/glpios12.c +++ /dev/null @@ -1,177 +0,0 @@ -/* glpios12.c (node selection heuristics) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" - -/*********************************************************************** -* NAME -* -* ios_choose_node - select subproblem to continue the search -* -* SYNOPSIS -* -* #include "glpios.h" -* int ios_choose_node(glp_tree *T); -* -* DESCRIPTION -* -* The routine ios_choose_node selects a subproblem from the active -* list to continue the search. The choice depends on the backtracking -* technique option. -* -* RETURNS -* -* The routine ios_choose_node return the reference number of the -* subproblem selected. */ - -static int most_feas(glp_tree *T); -static int best_proj(glp_tree *T); -static int best_node(glp_tree *T); - -int ios_choose_node(glp_tree *T) -{ int p; - if (T->parm->bt_tech == GLP_BT_DFS) - { /* depth first search */ - xassert(T->tail != NULL); - p = T->tail->p; - } - else if (T->parm->bt_tech == GLP_BT_BFS) - { /* breadth first search */ - xassert(T->head != NULL); - p = T->head->p; - } - else if (T->parm->bt_tech == GLP_BT_BLB) - { /* select node with best local bound */ - p = best_node(T); - } - else if (T->parm->bt_tech == GLP_BT_BPH) - { if (T->mip->mip_stat == GLP_UNDEF) - { /* "most integer feasible" subproblem */ - p = most_feas(T); - } - else - { /* best projection heuristic */ - p = best_proj(T); - } - } - else - xassert(T != T); - return p; -} - -static int most_feas(glp_tree *T) -{ /* select subproblem whose parent has minimal sum of integer - infeasibilities */ - IOSNPD *node; - int p; - double best; - p = 0, best = DBL_MAX; - for (node = T->head; node != NULL; node = node->next) - { xassert(node->up != NULL); - if (best > node->up->ii_sum) - p = node->p, best = node->up->ii_sum; - } - return p; -} - -static int best_proj(glp_tree *T) -{ /* select subproblem using the best projection heuristic */ - IOSNPD *root, *node; - int p; - double best, deg, obj; - /* the global bound must exist */ - xassert(T->mip->mip_stat == GLP_FEAS); - /* obtain pointer to the root node, which must exist */ - root = T->slot[1].node; - xassert(root != NULL); - /* deg estimates degradation of the objective function per unit - of the sum of integer infeasibilities */ - xassert(root->ii_sum > 0.0); - deg = (T->mip->mip_obj - root->bound) / root->ii_sum; - /* nothing has been selected so far */ - p = 0, best = DBL_MAX; - /* walk through the list of active subproblems */ - for (node = T->head; node != NULL; node = node->next) - { xassert(node->up != NULL); - /* obj estimates optimal objective value if the sum of integer - infeasibilities were zero */ - obj = node->up->bound + deg * node->up->ii_sum; - if (T->mip->dir == GLP_MAX) obj = - obj; - /* select the subproblem which has the best estimated optimal - objective value */ - if (best > obj) p = node->p, best = obj; - } - return p; -} - -static int best_node(glp_tree *T) -{ /* select subproblem with best local bound */ - IOSNPD *node, *best = NULL; - double bound, eps; - switch (T->mip->dir) - { case GLP_MIN: - bound = +DBL_MAX; - for (node = T->head; node != NULL; node = node->next) - if (bound > node->bound) bound = node->bound; - xassert(bound != +DBL_MAX); - eps = 1e-10 * (1.0 + fabs(bound)); - for (node = T->head; node != NULL; node = node->next) - { if (node->bound <= bound + eps) - { xassert(node->up != NULL); - if (best == NULL || -#if 1 - best->up->ii_sum > node->up->ii_sum) best = node; -#else - best->lp_obj > node->lp_obj) best = node; -#endif - } - } - break; - case GLP_MAX: - bound = -DBL_MAX; - for (node = T->head; node != NULL; node = node->next) - if (bound < node->bound) bound = node->bound; - xassert(bound != -DBL_MAX); - eps = 1e-10 * (1.0 + fabs(bound)); - for (node = T->head; node != NULL; node = node->next) - { if (node->bound >= bound - eps) - { xassert(node->up != NULL); - if (best == NULL || -#if 1 - best->up->ii_sum > node->up->ii_sum) best = node; -#else - best->lp_obj < node->lp_obj) best = node; -#endif - } - } - break; - default: - xassert(T != T); - } - xassert(best != NULL); - return best->p; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpipm.c b/code/3rd_glpk/draft/glpipm.c deleted file mode 100644 index 2b3a8176..00000000 --- a/code/3rd_glpk/draft/glpipm.c +++ /dev/null @@ -1,1144 +0,0 @@ -/* glpipm.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpipm.h" -#include "glpmat.h" - -#define ITER_MAX 100 -/* maximal number of iterations */ - -struct csa -{ /* common storage area */ - /*--------------------------------------------------------------*/ - /* LP data */ - int m; - /* number of rows (equality constraints) */ - int n; - /* number of columns (structural variables) */ - int *A_ptr; /* int A_ptr[1+m+1]; */ - int *A_ind; /* int A_ind[A_ptr[m+1]]; */ - double *A_val; /* double A_val[A_ptr[m+1]]; */ - /* mxn-matrix A in storage-by-rows format */ - double *b; /* double b[1+m]; */ - /* m-vector b of right-hand sides */ - double *c; /* double c[1+n]; */ - /* n-vector c of objective coefficients; c[0] is constant term of - the objective function */ - /*--------------------------------------------------------------*/ - /* LP solution */ - double *x; /* double x[1+n]; */ - double *y; /* double y[1+m]; */ - double *z; /* double z[1+n]; */ - /* current point in primal-dual space; the best point on exit */ - /*--------------------------------------------------------------*/ - /* control parameters */ - const glp_iptcp *parm; - /*--------------------------------------------------------------*/ - /* working arrays and variables */ - double *D; /* double D[1+n]; */ - /* diagonal nxn-matrix D = X*inv(Z), where X = diag(x[j]) and - Z = diag(z[j]) */ - int *P; /* int P[1+m+m]; */ - /* permutation mxm-matrix P used to minimize fill-in in Cholesky - factorization */ - int *S_ptr; /* int S_ptr[1+m+1]; */ - int *S_ind; /* int S_ind[S_ptr[m+1]]; */ - double *S_val; /* double S_val[S_ptr[m+1]]; */ - double *S_diag; /* double S_diag[1+m]; */ - /* symmetric mxm-matrix S = P*A*D*A'*P' whose upper triangular - part without diagonal elements is stored in S_ptr, S_ind, and - S_val in storage-by-rows format, diagonal elements are stored - in S_diag */ - int *U_ptr; /* int U_ptr[1+m+1]; */ - int *U_ind; /* int U_ind[U_ptr[m+1]]; */ - double *U_val; /* double U_val[U_ptr[m+1]]; */ - double *U_diag; /* double U_diag[1+m]; */ - /* upper triangular mxm-matrix U defining Cholesky factorization - S = U'*U; its non-diagonal elements are stored in U_ptr, U_ind, - U_val in storage-by-rows format, diagonal elements are stored - in U_diag */ - int iter; - /* iteration number (0, 1, 2, ...); iter = 0 corresponds to the - initial point */ - double obj; - /* current value of the objective function */ - double rpi; - /* relative primal infeasibility rpi = ||A*x-b||/(1+||b||) */ - double rdi; - /* relative dual infeasibility rdi = ||A'*y+z-c||/(1+||c||) */ - double gap; - /* primal-dual gap = |c'*x-b'*y|/(1+|c'*x|) which is a relative - difference between primal and dual objective functions */ - double phi; - /* merit function phi = ||A*x-b||/max(1,||b||) + - + ||A'*y+z-c||/max(1,||c||) + - + |c'*x-b'*y|/max(1,||b||,||c||) */ - double mu; - /* duality measure mu = x'*z/n (used as barrier parameter) */ - double rmu; - /* rmu = max(||A*x-b||,||A'*y+z-c||)/mu */ - double rmu0; - /* the initial value of rmu on iteration 0 */ - double *phi_min; /* double phi_min[1+ITER_MAX]; */ - /* phi_min[k] = min(phi[k]), where phi[k] is the value of phi on - k-th iteration, 0 <= k <= iter */ - int best_iter; - /* iteration number, on which the value of phi reached its best - (minimal) value */ - double *best_x; /* double best_x[1+n]; */ - double *best_y; /* double best_y[1+m]; */ - double *best_z; /* double best_z[1+n]; */ - /* best point (in the sense of the merit function phi) which has - been reached on iteration iter_best */ - double best_obj; - /* objective value at the best point */ - double *dx_aff; /* double dx_aff[1+n]; */ - double *dy_aff; /* double dy_aff[1+m]; */ - double *dz_aff; /* double dz_aff[1+n]; */ - /* affine scaling direction */ - double alfa_aff_p, alfa_aff_d; - /* maximal primal and dual stepsizes in affine scaling direction, - on which x and z are still non-negative */ - double mu_aff; - /* duality measure mu_aff = x_aff'*z_aff/n in the boundary point - x_aff' = x+alfa_aff_p*dx_aff, z_aff' = z+alfa_aff_d*dz_aff */ - double sigma; - /* Mehrotra's heuristic parameter (0 <= sigma <= 1) */ - double *dx_cc; /* double dx_cc[1+n]; */ - double *dy_cc; /* double dy_cc[1+m]; */ - double *dz_cc; /* double dz_cc[1+n]; */ - /* centering corrector direction */ - double *dx; /* double dx[1+n]; */ - double *dy; /* double dy[1+m]; */ - double *dz; /* double dz[1+n]; */ - /* final combined direction dx = dx_aff+dx_cc, dy = dy_aff+dy_cc, - dz = dz_aff+dz_cc */ - double alfa_max_p; - double alfa_max_d; - /* maximal primal and dual stepsizes in combined direction, on - which x and z are still non-negative */ -}; - -/*********************************************************************** -* initialize - allocate and initialize common storage area -* -* This routine allocates and initializes the common storage area (CSA) -* used by interior-point method routines. */ - -static void initialize(struct csa *csa) -{ int m = csa->m; - int n = csa->n; - int i; - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Matrix A has %d non-zeros\n", csa->A_ptr[m+1]-1); - csa->D = xcalloc(1+n, sizeof(double)); - /* P := I */ - csa->P = xcalloc(1+m+m, sizeof(int)); - for (i = 1; i <= m; i++) csa->P[i] = csa->P[m+i] = i; - /* S := A*A', symbolically */ - csa->S_ptr = xcalloc(1+m+1, sizeof(int)); - csa->S_ind = adat_symbolic(m, n, csa->P, csa->A_ptr, csa->A_ind, - csa->S_ptr); - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Matrix S = A*A' has %d non-zeros (upper triangle)\n", - csa->S_ptr[m+1]-1 + m); - /* determine P using specified ordering algorithm */ - if (csa->parm->ord_alg == GLP_ORD_NONE) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Original ordering is being used\n"); - for (i = 1; i <= m; i++) - csa->P[i] = csa->P[m+i] = i; - } - else if (csa->parm->ord_alg == GLP_ORD_QMD) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Minimum degree ordering (QMD)...\n"); - min_degree(m, csa->S_ptr, csa->S_ind, csa->P); - } - else if (csa->parm->ord_alg == GLP_ORD_AMD) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Approximate minimum degree ordering (AMD)...\n"); - amd_order1(m, csa->S_ptr, csa->S_ind, csa->P); - } - else if (csa->parm->ord_alg == GLP_ORD_SYMAMD) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Approximate minimum degree ordering (SYMAMD)...\n") - ; - symamd_ord(m, csa->S_ptr, csa->S_ind, csa->P); - } - else - xassert(csa != csa); - /* S := P*A*A'*P', symbolically */ - xfree(csa->S_ind); - csa->S_ind = adat_symbolic(m, n, csa->P, csa->A_ptr, csa->A_ind, - csa->S_ptr); - csa->S_val = xcalloc(csa->S_ptr[m+1], sizeof(double)); - csa->S_diag = xcalloc(1+m, sizeof(double)); - /* compute Cholesky factorization S = U'*U, symbolically */ - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Computing Cholesky factorization S = L*L'...\n"); - csa->U_ptr = xcalloc(1+m+1, sizeof(int)); - csa->U_ind = chol_symbolic(m, csa->S_ptr, csa->S_ind, csa->U_ptr); - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Matrix L has %d non-zeros\n", csa->U_ptr[m+1]-1 + m); - csa->U_val = xcalloc(csa->U_ptr[m+1], sizeof(double)); - csa->U_diag = xcalloc(1+m, sizeof(double)); - csa->iter = 0; - csa->obj = 0.0; - csa->rpi = 0.0; - csa->rdi = 0.0; - csa->gap = 0.0; - csa->phi = 0.0; - csa->mu = 0.0; - csa->rmu = 0.0; - csa->rmu0 = 0.0; - csa->phi_min = xcalloc(1+ITER_MAX, sizeof(double)); - csa->best_iter = 0; - csa->best_x = xcalloc(1+n, sizeof(double)); - csa->best_y = xcalloc(1+m, sizeof(double)); - csa->best_z = xcalloc(1+n, sizeof(double)); - csa->best_obj = 0.0; - csa->dx_aff = xcalloc(1+n, sizeof(double)); - csa->dy_aff = xcalloc(1+m, sizeof(double)); - csa->dz_aff = xcalloc(1+n, sizeof(double)); - csa->alfa_aff_p = 0.0; - csa->alfa_aff_d = 0.0; - csa->mu_aff = 0.0; - csa->sigma = 0.0; - csa->dx_cc = xcalloc(1+n, sizeof(double)); - csa->dy_cc = xcalloc(1+m, sizeof(double)); - csa->dz_cc = xcalloc(1+n, sizeof(double)); - csa->dx = csa->dx_aff; - csa->dy = csa->dy_aff; - csa->dz = csa->dz_aff; - csa->alfa_max_p = 0.0; - csa->alfa_max_d = 0.0; - return; -} - -/*********************************************************************** -* A_by_vec - compute y = A*x -* -* This routine computes matrix-vector product y = A*x, where A is the -* constraint matrix. */ - -static void A_by_vec(struct csa *csa, double x[], double y[]) -{ /* compute y = A*x */ - int m = csa->m; - int *A_ptr = csa->A_ptr; - int *A_ind = csa->A_ind; - double *A_val = csa->A_val; - int i, t, beg, end; - double temp; - for (i = 1; i <= m; i++) - { temp = 0.0; - beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) temp += A_val[t] * x[A_ind[t]]; - y[i] = temp; - } - return; -} - -/*********************************************************************** -* AT_by_vec - compute y = A'*x -* -* This routine computes matrix-vector product y = A'*x, where A' is a -* matrix transposed to the constraint matrix A. */ - -static void AT_by_vec(struct csa *csa, double x[], double y[]) -{ /* compute y = A'*x, where A' is transposed to A */ - int m = csa->m; - int n = csa->n; - int *A_ptr = csa->A_ptr; - int *A_ind = csa->A_ind; - double *A_val = csa->A_val; - int i, j, t, beg, end; - double temp; - for (j = 1; j <= n; j++) y[j] = 0.0; - for (i = 1; i <= m; i++) - { temp = x[i]; - if (temp == 0.0) continue; - beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) y[A_ind[t]] += A_val[t] * temp; - } - return; -} - -/*********************************************************************** -* decomp_NE - numeric factorization of matrix S = P*A*D*A'*P' -* -* This routine implements numeric phase of Cholesky factorization of -* the matrix S = P*A*D*A'*P', which is a permuted matrix of the normal -* equation system. Matrix D is assumed to be already computed. */ - -static void decomp_NE(struct csa *csa) -{ adat_numeric(csa->m, csa->n, csa->P, csa->A_ptr, csa->A_ind, - csa->A_val, csa->D, csa->S_ptr, csa->S_ind, csa->S_val, - csa->S_diag); - chol_numeric(csa->m, csa->S_ptr, csa->S_ind, csa->S_val, - csa->S_diag, csa->U_ptr, csa->U_ind, csa->U_val, csa->U_diag); - return; -} - -/*********************************************************************** -* solve_NE - solve normal equation system -* -* This routine solves the normal equation system: -* -* A*D*A'*y = h. -* -* It is assumed that the matrix A*D*A' has been previously factorized -* by the routine decomp_NE. -* -* On entry the array y contains the vector of right-hand sides h. On -* exit this array contains the computed vector of unknowns y. -* -* Once the vector y has been computed the routine checks for numeric -* stability. If the residual vector: -* -* r = A*D*A'*y - h -* -* is relatively small, the routine returns zero, otherwise non-zero is -* returned. */ - -static int solve_NE(struct csa *csa, double y[]) -{ int m = csa->m; - int n = csa->n; - int *P = csa->P; - int i, j, ret = 0; - double *h, *r, *w; - /* save vector of right-hand sides h */ - h = xcalloc(1+m, sizeof(double)); - for (i = 1; i <= m; i++) h[i] = y[i]; - /* solve normal equation system (A*D*A')*y = h */ - /* since S = P*A*D*A'*P' = U'*U, then A*D*A' = P'*U'*U*P, so we - have inv(A*D*A') = P'*inv(U)*inv(U')*P */ - /* w := P*h */ - w = xcalloc(1+m, sizeof(double)); - for (i = 1; i <= m; i++) w[i] = y[P[i]]; - /* w := inv(U')*w */ - ut_solve(m, csa->U_ptr, csa->U_ind, csa->U_val, csa->U_diag, w); - /* w := inv(U)*w */ - u_solve(m, csa->U_ptr, csa->U_ind, csa->U_val, csa->U_diag, w); - /* y := P'*w */ - for (i = 1; i <= m; i++) y[i] = w[P[m+i]]; - xfree(w); - /* compute residual vector r = A*D*A'*y - h */ - r = xcalloc(1+m, sizeof(double)); - /* w := A'*y */ - w = xcalloc(1+n, sizeof(double)); - AT_by_vec(csa, y, w); - /* w := D*w */ - for (j = 1; j <= n; j++) w[j] *= csa->D[j]; - /* r := A*w */ - A_by_vec(csa, w, r); - xfree(w); - /* r := r - h */ - for (i = 1; i <= m; i++) r[i] -= h[i]; - /* check for numeric stability */ - for (i = 1; i <= m; i++) - { if (fabs(r[i]) / (1.0 + fabs(h[i])) > 1e-4) - { ret = 1; - break; - } - } - xfree(h); - xfree(r); - return ret; -} - -/*********************************************************************** -* solve_NS - solve Newtonian system -* -* This routine solves the Newtonian system: -* -* A*dx = p -* -* A'*dy + dz = q -* -* Z*dx + X*dz = r -* -* where X = diag(x[j]), Z = diag(z[j]), by reducing it to the normal -* equation system: -* -* (A*inv(Z)*X*A')*dy = A*inv(Z)*(X*q-r)+p -* -* (it is assumed that the matrix A*inv(Z)*X*A' has been factorized by -* the routine decomp_NE). -* -* Once vector dy has been computed the routine computes vectors dx and -* dz as follows: -* -* dx = inv(Z)*(X*(A'*dy-q)+r) -* -* dz = inv(X)*(r-Z*dx) -* -* The routine solve_NS returns the same code which was reported by the -* routine solve_NE (see above). */ - -static int solve_NS(struct csa *csa, double p[], double q[], double r[], - double dx[], double dy[], double dz[]) -{ int m = csa->m; - int n = csa->n; - double *x = csa->x; - double *z = csa->z; - int i, j, ret; - double *w = dx; - /* compute the vector of right-hand sides A*inv(Z)*(X*q-r)+p for - the normal equation system */ - for (j = 1; j <= n; j++) - w[j] = (x[j] * q[j] - r[j]) / z[j]; - A_by_vec(csa, w, dy); - for (i = 1; i <= m; i++) dy[i] += p[i]; - /* solve the normal equation system to compute vector dy */ - ret = solve_NE(csa, dy); - /* compute vectors dx and dz */ - AT_by_vec(csa, dy, dx); - for (j = 1; j <= n; j++) - { dx[j] = (x[j] * (dx[j] - q[j]) + r[j]) / z[j]; - dz[j] = (r[j] - z[j] * dx[j]) / x[j]; - } - return ret; -} - -/*********************************************************************** -* initial_point - choose initial point using Mehrotra's heuristic -* -* This routine chooses a starting point using a heuristic proposed in -* the paper: -* -* S. Mehrotra. On the implementation of a primal-dual interior point -* method. SIAM J. on Optim., 2(4), pp. 575-601, 1992. -* -* The starting point x in the primal space is chosen as a solution of -* the following least squares problem: -* -* minimize ||x|| -* -* subject to A*x = b -* -* which can be computed explicitly as follows: -* -* x = A'*inv(A*A')*b -* -* Similarly, the starting point (y, z) in the dual space is chosen as -* a solution of the following least squares problem: -* -* minimize ||z|| -* -* subject to A'*y + z = c -* -* which can be computed explicitly as follows: -* -* y = inv(A*A')*A*c -* -* z = c - A'*y -* -* However, some components of the vectors x and z may be non-positive -* or close to zero, so the routine uses a Mehrotra's heuristic to find -* a more appropriate starting point. */ - -static void initial_point(struct csa *csa) -{ int m = csa->m; - int n = csa->n; - double *b = csa->b; - double *c = csa->c; - double *x = csa->x; - double *y = csa->y; - double *z = csa->z; - double *D = csa->D; - int i, j; - double dp, dd, ex, ez, xz; - /* factorize A*A' */ - for (j = 1; j <= n; j++) D[j] = 1.0; - decomp_NE(csa); - /* x~ = A'*inv(A*A')*b */ - for (i = 1; i <= m; i++) y[i] = b[i]; - solve_NE(csa, y); - AT_by_vec(csa, y, x); - /* y~ = inv(A*A')*A*c */ - A_by_vec(csa, c, y); - solve_NE(csa, y); - /* z~ = c - A'*y~ */ - AT_by_vec(csa, y,z); - for (j = 1; j <= n; j++) z[j] = c[j] - z[j]; - /* use Mehrotra's heuristic in order to choose more appropriate - starting point with positive components of vectors x and z */ - dp = dd = 0.0; - for (j = 1; j <= n; j++) - { if (dp < -1.5 * x[j]) dp = -1.5 * x[j]; - if (dd < -1.5 * z[j]) dd = -1.5 * z[j]; - } - /* note that b = 0 involves x = 0, and c = 0 involves y = 0 and - z = 0, so we need to be careful */ - if (dp == 0.0) dp = 1.5; - if (dd == 0.0) dd = 1.5; - ex = ez = xz = 0.0; - for (j = 1; j <= n; j++) - { ex += (x[j] + dp); - ez += (z[j] + dd); - xz += (x[j] + dp) * (z[j] + dd); - } - dp += 0.5 * (xz / ez); - dd += 0.5 * (xz / ex); - for (j = 1; j <= n; j++) - { x[j] += dp; - z[j] += dd; - xassert(x[j] > 0.0 && z[j] > 0.0); - } - return; -} - -/*********************************************************************** -* basic_info - perform basic computations at the current point -* -* This routine computes the following quantities at the current point: -* -* 1) value of the objective function: -* -* F = c'*x + c[0] -* -* 2) relative primal infeasibility: -* -* rpi = ||A*x-b|| / (1+||b||) -* -* 3) relative dual infeasibility: -* -* rdi = ||A'*y+z-c|| / (1+||c||) -* -* 4) primal-dual gap (relative difference between the primal and the -* dual objective function values): -* -* gap = |c'*x-b'*y| / (1+|c'*x|) -* -* 5) merit function: -* -* phi = ||A*x-b|| / max(1,||b||) + ||A'*y+z-c|| / max(1,||c||) + -* -* + |c'*x-b'*y| / max(1,||b||,||c||) -* -* 6) duality measure: -* -* mu = x'*z / n -* -* 7) the ratio of infeasibility to mu: -* -* rmu = max(||A*x-b||,||A'*y+z-c||) / mu -* -* where ||*|| denotes euclidian norm, *' denotes transposition. */ - -static void basic_info(struct csa *csa) -{ int m = csa->m; - int n = csa->n; - double *b = csa->b; - double *c = csa->c; - double *x = csa->x; - double *y = csa->y; - double *z = csa->z; - int i, j; - double norm1, bnorm, norm2, cnorm, cx, by, *work, temp; - /* compute value of the objective function */ - temp = c[0]; - for (j = 1; j <= n; j++) temp += c[j] * x[j]; - csa->obj = temp; - /* norm1 = ||A*x-b|| */ - work = xcalloc(1+m, sizeof(double)); - A_by_vec(csa, x, work); - norm1 = 0.0; - for (i = 1; i <= m; i++) - norm1 += (work[i] - b[i]) * (work[i] - b[i]); - norm1 = sqrt(norm1); - xfree(work); - /* bnorm = ||b|| */ - bnorm = 0.0; - for (i = 1; i <= m; i++) bnorm += b[i] * b[i]; - bnorm = sqrt(bnorm); - /* compute relative primal infeasibility */ - csa->rpi = norm1 / (1.0 + bnorm); - /* norm2 = ||A'*y+z-c|| */ - work = xcalloc(1+n, sizeof(double)); - AT_by_vec(csa, y, work); - norm2 = 0.0; - for (j = 1; j <= n; j++) - norm2 += (work[j] + z[j] - c[j]) * (work[j] + z[j] - c[j]); - norm2 = sqrt(norm2); - xfree(work); - /* cnorm = ||c|| */ - cnorm = 0.0; - for (j = 1; j <= n; j++) cnorm += c[j] * c[j]; - cnorm = sqrt(cnorm); - /* compute relative dual infeasibility */ - csa->rdi = norm2 / (1.0 + cnorm); - /* by = b'*y */ - by = 0.0; - for (i = 1; i <= m; i++) by += b[i] * y[i]; - /* cx = c'*x */ - cx = 0.0; - for (j = 1; j <= n; j++) cx += c[j] * x[j]; - /* compute primal-dual gap */ - csa->gap = fabs(cx - by) / (1.0 + fabs(cx)); - /* compute merit function */ - csa->phi = 0.0; - csa->phi += norm1 / (bnorm > 1.0 ? bnorm : 1.0); - csa->phi += norm2 / (cnorm > 1.0 ? cnorm : 1.0); - temp = 1.0; - if (temp < bnorm) temp = bnorm; - if (temp < cnorm) temp = cnorm; - csa->phi += fabs(cx - by) / temp; - /* compute duality measure */ - temp = 0.0; - for (j = 1; j <= n; j++) temp += x[j] * z[j]; - csa->mu = temp / (double)n; - /* compute the ratio of infeasibility to mu */ - csa->rmu = (norm1 > norm2 ? norm1 : norm2) / csa->mu; - return; -} - -/*********************************************************************** -* make_step - compute next point using Mehrotra's technique -* -* This routine computes the next point using the predictor-corrector -* technique proposed in the paper: -* -* S. Mehrotra. On the implementation of a primal-dual interior point -* method. SIAM J. on Optim., 2(4), pp. 575-601, 1992. -* -* At first, the routine computes so called affine scaling (predictor) -* direction (dx_aff,dy_aff,dz_aff) which is a solution of the system: -* -* A*dx_aff = b - A*x -* -* A'*dy_aff + dz_aff = c - A'*y - z -* -* Z*dx_aff + X*dz_aff = - X*Z*e -* -* where (x,y,z) is the current point, X = diag(x[j]), Z = diag(z[j]), -* e = (1,...,1)'. -* -* Then, the routine computes the centering parameter sigma, using the -* following Mehrotra's heuristic: -* -* alfa_aff_p = inf{0 <= alfa <= 1 | x+alfa*dx_aff >= 0} -* -* alfa_aff_d = inf{0 <= alfa <= 1 | z+alfa*dz_aff >= 0} -* -* mu_aff = (x+alfa_aff_p*dx_aff)'*(z+alfa_aff_d*dz_aff)/n -* -* sigma = (mu_aff/mu)^3 -* -* where alfa_aff_p is the maximal stepsize along the affine scaling -* direction in the primal space, alfa_aff_d is the maximal stepsize -* along the same direction in the dual space. -* -* After determining sigma the routine computes so called centering -* (corrector) direction (dx_cc,dy_cc,dz_cc) which is the solution of -* the system: -* -* A*dx_cc = 0 -* -* A'*dy_cc + dz_cc = 0 -* -* Z*dx_cc + X*dz_cc = sigma*mu*e - X*Z*e -* -* Finally, the routine computes the combined direction -* -* (dx,dy,dz) = (dx_aff,dy_aff,dz_aff) + (dx_cc,dy_cc,dz_cc) -* -* and determines maximal primal and dual stepsizes along the combined -* direction: -* -* alfa_max_p = inf{0 <= alfa <= 1 | x+alfa*dx >= 0} -* -* alfa_max_d = inf{0 <= alfa <= 1 | z+alfa*dz >= 0} -* -* In order to prevent the next point to be too close to the boundary -* of the positive ortant, the routine decreases maximal stepsizes: -* -* alfa_p = gamma_p * alfa_max_p -* -* alfa_d = gamma_d * alfa_max_d -* -* where gamma_p and gamma_d are scaling factors, and computes the next -* point: -* -* x_new = x + alfa_p * dx -* -* y_new = y + alfa_d * dy -* -* z_new = z + alfa_d * dz -* -* which becomes the current point on the next iteration. */ - -static int make_step(struct csa *csa) -{ int m = csa->m; - int n = csa->n; - double *b = csa->b; - double *c = csa->c; - double *x = csa->x; - double *y = csa->y; - double *z = csa->z; - double *dx_aff = csa->dx_aff; - double *dy_aff = csa->dy_aff; - double *dz_aff = csa->dz_aff; - double *dx_cc = csa->dx_cc; - double *dy_cc = csa->dy_cc; - double *dz_cc = csa->dz_cc; - double *dx = csa->dx; - double *dy = csa->dy; - double *dz = csa->dz; - int i, j, ret = 0; - double temp, gamma_p, gamma_d, *p, *q, *r; - /* allocate working arrays */ - p = xcalloc(1+m, sizeof(double)); - q = xcalloc(1+n, sizeof(double)); - r = xcalloc(1+n, sizeof(double)); - /* p = b - A*x */ - A_by_vec(csa, x, p); - for (i = 1; i <= m; i++) p[i] = b[i] - p[i]; - /* q = c - A'*y - z */ - AT_by_vec(csa, y,q); - for (j = 1; j <= n; j++) q[j] = c[j] - q[j] - z[j]; - /* r = - X * Z * e */ - for (j = 1; j <= n; j++) r[j] = - x[j] * z[j]; - /* solve the first Newtonian system */ - if (solve_NS(csa, p, q, r, dx_aff, dy_aff, dz_aff)) - { ret = 1; - goto done; - } - /* alfa_aff_p = inf{0 <= alfa <= 1 | x + alfa*dx_aff >= 0} */ - /* alfa_aff_d = inf{0 <= alfa <= 1 | z + alfa*dz_aff >= 0} */ - csa->alfa_aff_p = csa->alfa_aff_d = 1.0; - for (j = 1; j <= n; j++) - { if (dx_aff[j] < 0.0) - { temp = - x[j] / dx_aff[j]; - if (csa->alfa_aff_p > temp) csa->alfa_aff_p = temp; - } - if (dz_aff[j] < 0.0) - { temp = - z[j] / dz_aff[j]; - if (csa->alfa_aff_d > temp) csa->alfa_aff_d = temp; - } - } - /* mu_aff = (x+alfa_aff_p*dx_aff)' * (z+alfa_aff_d*dz_aff) / n */ - temp = 0.0; - for (j = 1; j <= n; j++) - temp += (x[j] + csa->alfa_aff_p * dx_aff[j]) * - (z[j] + csa->alfa_aff_d * dz_aff[j]); - csa->mu_aff = temp / (double)n; - /* sigma = (mu_aff/mu)^3 */ - temp = csa->mu_aff / csa->mu; - csa->sigma = temp * temp * temp; - /* p = 0 */ - for (i = 1; i <= m; i++) p[i] = 0.0; - /* q = 0 */ - for (j = 1; j <= n; j++) q[j] = 0.0; - /* r = sigma * mu * e - X * Z * e */ - for (j = 1; j <= n; j++) - r[j] = csa->sigma * csa->mu - dx_aff[j] * dz_aff[j]; - /* solve the second Newtonian system with the same coefficients - but with altered right-hand sides */ - if (solve_NS(csa, p, q, r, dx_cc, dy_cc, dz_cc)) - { ret = 1; - goto done; - } - /* (dx,dy,dz) = (dx_aff,dy_aff,dz_aff) + (dx_cc,dy_cc,dz_cc) */ - for (j = 1; j <= n; j++) dx[j] = dx_aff[j] + dx_cc[j]; - for (i = 1; i <= m; i++) dy[i] = dy_aff[i] + dy_cc[i]; - for (j = 1; j <= n; j++) dz[j] = dz_aff[j] + dz_cc[j]; - /* alfa_max_p = inf{0 <= alfa <= 1 | x + alfa*dx >= 0} */ - /* alfa_max_d = inf{0 <= alfa <= 1 | z + alfa*dz >= 0} */ - csa->alfa_max_p = csa->alfa_max_d = 1.0; - for (j = 1; j <= n; j++) - { if (dx[j] < 0.0) - { temp = - x[j] / dx[j]; - if (csa->alfa_max_p > temp) csa->alfa_max_p = temp; - } - if (dz[j] < 0.0) - { temp = - z[j] / dz[j]; - if (csa->alfa_max_d > temp) csa->alfa_max_d = temp; - } - } - /* determine scale factors (not implemented yet) */ - gamma_p = 0.90; - gamma_d = 0.90; - /* compute the next point */ - for (j = 1; j <= n; j++) - { x[j] += gamma_p * csa->alfa_max_p * dx[j]; - xassert(x[j] > 0.0); - } - for (i = 1; i <= m; i++) - y[i] += gamma_d * csa->alfa_max_d * dy[i]; - for (j = 1; j <= n; j++) - { z[j] += gamma_d * csa->alfa_max_d * dz[j]; - xassert(z[j] > 0.0); - } -done: /* free working arrays */ - xfree(p); - xfree(q); - xfree(r); - return ret; -} - -/*********************************************************************** -* terminate - deallocate common storage area -* -* This routine frees all memory allocated to the common storage area -* used by interior-point method routines. */ - -static void terminate(struct csa *csa) -{ xfree(csa->D); - xfree(csa->P); - xfree(csa->S_ptr); - xfree(csa->S_ind); - xfree(csa->S_val); - xfree(csa->S_diag); - xfree(csa->U_ptr); - xfree(csa->U_ind); - xfree(csa->U_val); - xfree(csa->U_diag); - xfree(csa->phi_min); - xfree(csa->best_x); - xfree(csa->best_y); - xfree(csa->best_z); - xfree(csa->dx_aff); - xfree(csa->dy_aff); - xfree(csa->dz_aff); - xfree(csa->dx_cc); - xfree(csa->dy_cc); - xfree(csa->dz_cc); - return; -} - -/*********************************************************************** -* ipm_main - main interior-point method routine -* -* This is a main routine of the primal-dual interior-point method. -* -* The routine ipm_main returns one of the following codes: -* -* 0 - optimal solution found; -* 1 - problem has no feasible (primal or dual) solution; -* 2 - no convergence; -* 3 - iteration limit exceeded; -* 4 - numeric instability on solving Newtonian system. -* -* In case of non-zero return code the routine returns the best point, -* which has been reached during optimization. */ - -static int ipm_main(struct csa *csa) -{ int m = csa->m; - int n = csa->n; - int i, j, status; - double temp; - /* choose initial point using Mehrotra's heuristic */ - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Guessing initial point...\n"); - initial_point(csa); - /* main loop starts here */ - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Optimization begins...\n"); - for (;;) - { /* perform basic computations at the current point */ - basic_info(csa); - /* save initial value of rmu */ - if (csa->iter == 0) csa->rmu0 = csa->rmu; - /* accumulate values of min(phi[k]) and save the best point */ - xassert(csa->iter <= ITER_MAX); - if (csa->iter == 0 || csa->phi_min[csa->iter-1] > csa->phi) - { csa->phi_min[csa->iter] = csa->phi; - csa->best_iter = csa->iter; - for (j = 1; j <= n; j++) csa->best_x[j] = csa->x[j]; - for (i = 1; i <= m; i++) csa->best_y[i] = csa->y[i]; - for (j = 1; j <= n; j++) csa->best_z[j] = csa->z[j]; - csa->best_obj = csa->obj; - } - else - csa->phi_min[csa->iter] = csa->phi_min[csa->iter-1]; - /* display information at the current point */ - if (csa->parm->msg_lev >= GLP_MSG_ON) - xprintf("%3d: obj = %17.9e; rpi = %8.1e; rdi = %8.1e; gap =" - " %8.1e\n", csa->iter, csa->obj, csa->rpi, csa->rdi, - csa->gap); - /* check if the current point is optimal */ - if (csa->rpi < 1e-8 && csa->rdi < 1e-8 && csa->gap < 1e-8) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("OPTIMAL SOLUTION FOUND\n"); - status = 0; - break; - } - /* check if the problem has no feasible solution */ - temp = 1e5 * csa->phi_min[csa->iter]; - if (temp < 1e-8) temp = 1e-8; - if (csa->phi >= temp) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("PROBLEM HAS NO FEASIBLE PRIMAL/DUAL SOLUTION\n") - ; - status = 1; - break; - } - /* check for very slow convergence or divergence */ - if (((csa->rpi >= 1e-8 || csa->rdi >= 1e-8) && csa->rmu / - csa->rmu0 >= 1e6) || - (csa->iter >= 30 && csa->phi_min[csa->iter] >= 0.5 * - csa->phi_min[csa->iter - 30])) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("NO CONVERGENCE; SEARCH TERMINATED\n"); - status = 2; - break; - } - /* check for maximal number of iterations */ - if (csa->iter == ITER_MAX) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("ITERATION LIMIT EXCEEDED; SEARCH TERMINATED\n"); - status = 3; - break; - } - /* start the next iteration */ - csa->iter++; - /* factorize normal equation system */ - for (j = 1; j <= n; j++) csa->D[j] = csa->x[j] / csa->z[j]; - decomp_NE(csa); - /* compute the next point using Mehrotra's predictor-corrector - technique */ - if (make_step(csa)) - { if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("NUMERIC INSTABILITY; SEARCH TERMINATED\n"); - status = 4; - break; - } - } - /* restore the best point */ - if (status != 0) - { for (j = 1; j <= n; j++) csa->x[j] = csa->best_x[j]; - for (i = 1; i <= m; i++) csa->y[i] = csa->best_y[i]; - for (j = 1; j <= n; j++) csa->z[j] = csa->best_z[j]; - if (csa->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Best point %17.9e was reached on iteration %d\n", - csa->best_obj, csa->best_iter); - } - /* return to the calling program */ - return status; -} - -/*********************************************************************** -* NAME -* -* ipm_solve - core LP solver based on the interior-point method -* -* SYNOPSIS -* -* #include "glpipm.h" -* int ipm_solve(glp_prob *P, const glp_iptcp *parm); -* -* DESCRIPTION -* -* The routine ipm_solve is a core LP solver based on the primal-dual -* interior-point method. -* -* The routine assumes the following standard formulation of LP problem -* to be solved: -* -* minimize -* -* F = c[0] + c[1]*x[1] + c[2]*x[2] + ... + c[n]*x[n] -* -* subject to linear constraints -* -* a[1,1]*x[1] + a[1,2]*x[2] + ... + a[1,n]*x[n] = b[1] -* -* a[2,1]*x[1] + a[2,2]*x[2] + ... + a[2,n]*x[n] = b[2] -* -* . . . . . . -* -* a[m,1]*x[1] + a[m,2]*x[2] + ... + a[m,n]*x[n] = b[m] -* -* and non-negative variables -* -* x[1] >= 0, x[2] >= 0, ..., x[n] >= 0 -* -* where: -* F is the objective function; -* x[1], ..., x[n] are (structural) variables; -* c[0] is a constant term of the objective function; -* c[1], ..., c[n] are objective coefficients; -* a[1,1], ..., a[m,n] are constraint coefficients; -* b[1], ..., b[n] are right-hand sides. -* -* The solution is three vectors x, y, and z, which are stored by the -* routine in the arrays x, y, and z, respectively. These vectors -* correspond to the best primal-dual point found during optimization. -* They are approximate solution of the following system (which is the -* Karush-Kuhn-Tucker optimality conditions): -* -* A*x = b (primal feasibility condition) -* -* A'*y + z = c (dual feasibility condition) -* -* x'*z = 0 (primal-dual complementarity condition) -* -* x >= 0, z >= 0 (non-negativity condition) -* -* where: -* x[1], ..., x[n] are primal (structural) variables; -* y[1], ..., y[m] are dual variables (Lagrange multipliers) for -* equality constraints; -* z[1], ..., z[n] are dual variables (Lagrange multipliers) for -* non-negativity constraints. -* -* RETURNS -* -* 0 LP has been successfully solved. -* -* GLP_ENOCVG -* No convergence. -* -* GLP_EITLIM -* Iteration limit exceeded. -* -* GLP_EINSTAB -* Numeric instability on solving Newtonian system. -* -* In case of non-zero return code the routine returns the best point, -* which has been reached during optimization. */ - -int ipm_solve(glp_prob *P, const glp_iptcp *parm) -{ struct csa _dsa, *csa = &_dsa; - int m = P->m; - int n = P->n; - int nnz = P->nnz; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int i, j, loc, ret, *A_ind, *A_ptr; - double dir, *A_val, *b, *c, *x, *y, *z; - xassert(m > 0); - xassert(n > 0); - /* allocate working arrays */ - A_ptr = xcalloc(1+m+1, sizeof(int)); - A_ind = xcalloc(1+nnz, sizeof(int)); - A_val = xcalloc(1+nnz, sizeof(double)); - b = xcalloc(1+m, sizeof(double)); - c = xcalloc(1+n, sizeof(double)); - x = xcalloc(1+n, sizeof(double)); - y = xcalloc(1+m, sizeof(double)); - z = xcalloc(1+n, sizeof(double)); - /* prepare rows and constraint coefficients */ - loc = 1; - for (i = 1; i <= m; i++) - { row = P->row[i]; - xassert(row->type == GLP_FX); - b[i] = row->lb * row->rii; - A_ptr[i] = loc; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { A_ind[loc] = aij->col->j; - A_val[loc] = row->rii * aij->val * aij->col->sjj; - loc++; - } - } - A_ptr[m+1] = loc; - xassert(loc-1 == nnz); - /* prepare columns and objective coefficients */ - if (P->dir == GLP_MIN) - dir = +1.0; - else if (P->dir == GLP_MAX) - dir = -1.0; - else - xassert(P != P); - c[0] = dir * P->c0; - for (j = 1; j <= n; j++) - { col = P->col[j]; - xassert(col->type == GLP_LO && col->lb == 0.0); - c[j] = dir * col->coef * col->sjj; - } - /* allocate and initialize the common storage area */ - csa->m = m; - csa->n = n; - csa->A_ptr = A_ptr; - csa->A_ind = A_ind; - csa->A_val = A_val; - csa->b = b; - csa->c = c; - csa->x = x; - csa->y = y; - csa->z = z; - csa->parm = parm; - initialize(csa); - /* solve LP with the interior-point method */ - ret = ipm_main(csa); - /* deallocate the common storage area */ - terminate(csa); - /* determine solution status */ - if (ret == 0) - { /* optimal solution found */ - P->ipt_stat = GLP_OPT; - ret = 0; - } - else if (ret == 1) - { /* problem has no feasible (primal or dual) solution */ - P->ipt_stat = GLP_NOFEAS; - ret = 0; - } - else if (ret == 2) - { /* no convergence */ - P->ipt_stat = GLP_INFEAS; - ret = GLP_ENOCVG; - } - else if (ret == 3) - { /* iteration limit exceeded */ - P->ipt_stat = GLP_INFEAS; - ret = GLP_EITLIM; - } - else if (ret == 4) - { /* numeric instability on solving Newtonian system */ - P->ipt_stat = GLP_INFEAS; - ret = GLP_EINSTAB; - } - else - xassert(ret != ret); - /* store row solution components */ - for (i = 1; i <= m; i++) - { row = P->row[i]; - row->pval = row->lb; - row->dval = dir * y[i] * row->rii; - } - /* store column solution components */ - P->ipt_obj = P->c0; - for (j = 1; j <= n; j++) - { col = P->col[j]; - col->pval = x[j] * col->sjj; - col->dval = dir * z[j] / col->sjj; - P->ipt_obj += col->coef * col->pval; - } - /* free working arrays */ - xfree(A_ptr); - xfree(A_ind); - xfree(A_val); - xfree(b); - xfree(c); - xfree(x); - xfree(y); - xfree(z); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpipm.h b/code/3rd_glpk/draft/glpipm.h deleted file mode 100644 index a5f94fec..00000000 --- a/code/3rd_glpk/draft/glpipm.h +++ /dev/null @@ -1,36 +0,0 @@ -/* glpipm.h (primal-dual interior-point method) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPIPM_H -#define GLPIPM_H - -#include "prob.h" - -#define ipm_solve _glp_ipm_solve -int ipm_solve(glp_prob *P, const glp_iptcp *parm); -/* core LP solver based on the interior-point method */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpmat.c b/code/3rd_glpk/draft/glpmat.c deleted file mode 100644 index 97d1c651..00000000 --- a/code/3rd_glpk/draft/glpmat.c +++ /dev/null @@ -1,924 +0,0 @@ -/* glpmat.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpmat.h" -#include "qmd.h" -#include "amd.h" -#include "colamd.h" - -/*---------------------------------------------------------------------- --- check_fvs - check sparse vector in full-vector storage format. --- --- SYNOPSIS --- --- #include "glpmat.h" --- int check_fvs(int n, int nnz, int ind[], double vec[]); --- --- DESCRIPTION --- --- The routine check_fvs checks if a given vector of dimension n in --- full-vector storage format has correct representation. --- --- RETURNS --- --- The routine returns one of the following codes: --- --- 0 - the vector is correct; --- 1 - the number of elements (n) is negative; --- 2 - the number of non-zero elements (nnz) is negative; --- 3 - some element index is out of range; --- 4 - some element index is duplicate; --- 5 - some non-zero element is out of pattern. */ - -int check_fvs(int n, int nnz, int ind[], double vec[]) -{ int i, t, ret, *flag = NULL; - /* check the number of elements */ - if (n < 0) - { ret = 1; - goto done; - } - /* check the number of non-zero elements */ - if (nnz < 0) - { ret = 2; - goto done; - } - /* check vector indices */ - flag = xcalloc(1+n, sizeof(int)); - for (i = 1; i <= n; i++) flag[i] = 0; - for (t = 1; t <= nnz; t++) - { i = ind[t]; - if (!(1 <= i && i <= n)) - { ret = 3; - goto done; - } - if (flag[i]) - { ret = 4; - goto done; - } - flag[i] = 1; - } - /* check vector elements */ - for (i = 1; i <= n; i++) - { if (!flag[i] && vec[i] != 0.0) - { ret = 5; - goto done; - } - } - /* the vector is ok */ - ret = 0; -done: if (flag != NULL) xfree(flag); - return ret; -} - -/*---------------------------------------------------------------------- --- check_pattern - check pattern of sparse matrix. --- --- SYNOPSIS --- --- #include "glpmat.h" --- int check_pattern(int m, int n, int A_ptr[], int A_ind[]); --- --- DESCRIPTION --- --- The routine check_pattern checks the pattern of a given mxn matrix --- in storage-by-rows format. --- --- RETURNS --- --- The routine returns one of the following codes: --- --- 0 - the pattern is correct; --- 1 - the number of rows (m) is negative; --- 2 - the number of columns (n) is negative; --- 3 - A_ptr[1] is not 1; --- 4 - some column index is out of range; --- 5 - some column indices are duplicate. */ - -int check_pattern(int m, int n, int A_ptr[], int A_ind[]) -{ int i, j, ptr, ret, *flag = NULL; - /* check the number of rows */ - if (m < 0) - { ret = 1; - goto done; - } - /* check the number of columns */ - if (n < 0) - { ret = 2; - goto done; - } - /* check location A_ptr[1] */ - if (A_ptr[1] != 1) - { ret = 3; - goto done; - } - /* check row patterns */ - flag = xcalloc(1+n, sizeof(int)); - for (j = 1; j <= n; j++) flag[j] = 0; - for (i = 1; i <= m; i++) - { /* check pattern of row i */ - for (ptr = A_ptr[i]; ptr < A_ptr[i+1]; ptr++) - { j = A_ind[ptr]; - /* check column index */ - if (!(1 <= j && j <= n)) - { ret = 4; - goto done; - } - /* check for duplication */ - if (flag[j]) - { ret = 5; - goto done; - } - flag[j] = 1; - } - /* clear flags */ - for (ptr = A_ptr[i]; ptr < A_ptr[i+1]; ptr++) - { j = A_ind[ptr]; - flag[j] = 0; - } - } - /* the pattern is ok */ - ret = 0; -done: if (flag != NULL) xfree(flag); - return ret; -} - -/*---------------------------------------------------------------------- --- transpose - transpose sparse matrix. --- --- *Synopsis* --- --- #include "glpmat.h" --- void transpose(int m, int n, int A_ptr[], int A_ind[], --- double A_val[], int AT_ptr[], int AT_ind[], double AT_val[]); --- --- *Description* --- --- For a given mxn sparse matrix A the routine transpose builds a nxm --- sparse matrix A' which is a matrix transposed to A. --- --- The arrays A_ptr, A_ind, and A_val specify a given mxn matrix A to --- be transposed in storage-by-rows format. The parameter A_val can be --- NULL, in which case numeric values are not copied. The arrays A_ptr, --- A_ind, and A_val are not changed on exit. --- --- On entry the arrays AT_ptr, AT_ind, and AT_val must be allocated, --- but their content is ignored. On exit the routine stores a resultant --- nxm matrix A' in these arrays in storage-by-rows format. Note that --- if the parameter A_val is NULL, the array AT_val is not used. --- --- The routine transpose has a side effect that elements in rows of the --- resultant matrix A' follow in ascending their column indices. */ - -void transpose(int m, int n, int A_ptr[], int A_ind[], double A_val[], - int AT_ptr[], int AT_ind[], double AT_val[]) -{ int i, j, t, beg, end, pos, len; - /* determine row lengths of resultant matrix */ - for (j = 1; j <= n; j++) AT_ptr[j] = 0; - for (i = 1; i <= m; i++) - { beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) AT_ptr[A_ind[t]]++; - } - /* set up row pointers of resultant matrix */ - pos = 1; - for (j = 1; j <= n; j++) - len = AT_ptr[j], pos += len, AT_ptr[j] = pos; - AT_ptr[n+1] = pos; - /* build resultant matrix */ - for (i = m; i >= 1; i--) - { beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) - { pos = --AT_ptr[A_ind[t]]; - AT_ind[pos] = i; - if (A_val != NULL) AT_val[pos] = A_val[t]; - } - } - return; -} - -/*---------------------------------------------------------------------- --- adat_symbolic - compute S = P*A*D*A'*P' (symbolic phase). --- --- *Synopsis* --- --- #include "glpmat.h" --- int *adat_symbolic(int m, int n, int P_per[], int A_ptr[], --- int A_ind[], int S_ptr[]); --- --- *Description* --- --- The routine adat_symbolic implements the symbolic phase to compute --- symmetric matrix S = P*A*D*A'*P', where P is a permutation matrix, --- A is a given sparse matrix, D is a diagonal matrix, A' is a matrix --- transposed to A, P' is an inverse of P. --- --- The parameter m is the number of rows in A and the order of P. --- --- The parameter n is the number of columns in A and the order of D. --- --- The array P_per specifies permutation matrix P. It is not changed on --- exit. --- --- The arrays A_ptr and A_ind specify the pattern of matrix A. They are --- not changed on exit. --- --- On exit the routine stores the pattern of upper triangular part of --- matrix S without diagonal elements in the arrays S_ptr and S_ind in --- storage-by-rows format. The array S_ptr should be allocated on entry, --- however, its content is ignored. The array S_ind is allocated by the --- routine itself which returns a pointer to it. --- --- *Returns* --- --- The routine returns a pointer to the array S_ind. */ - -int *adat_symbolic(int m, int n, int P_per[], int A_ptr[], int A_ind[], - int S_ptr[]) -{ int i, j, t, ii, jj, tt, k, size, len; - int *S_ind, *AT_ptr, *AT_ind, *ind, *map, *temp; - /* build the pattern of A', which is a matrix transposed to A, to - efficiently access A in column-wise manner */ - AT_ptr = xcalloc(1+n+1, sizeof(int)); - AT_ind = xcalloc(A_ptr[m+1], sizeof(int)); - transpose(m, n, A_ptr, A_ind, NULL, AT_ptr, AT_ind, NULL); - /* allocate the array S_ind */ - size = A_ptr[m+1] - 1; - if (size < m) size = m; - S_ind = xcalloc(1+size, sizeof(int)); - /* allocate and initialize working arrays */ - ind = xcalloc(1+m, sizeof(int)); - map = xcalloc(1+m, sizeof(int)); - for (jj = 1; jj <= m; jj++) map[jj] = 0; - /* compute pattern of S; note that symbolically S = B*B', where - B = P*A, B' is matrix transposed to B */ - S_ptr[1] = 1; - for (ii = 1; ii <= m; ii++) - { /* compute pattern of ii-th row of S */ - len = 0; - i = P_per[ii]; /* i-th row of A = ii-th row of B */ - for (t = A_ptr[i]; t < A_ptr[i+1]; t++) - { k = A_ind[t]; - /* walk through k-th column of A */ - for (tt = AT_ptr[k]; tt < AT_ptr[k+1]; tt++) - { j = AT_ind[tt]; - jj = P_per[m+j]; /* j-th row of A = jj-th row of B */ - /* a[i,k] != 0 and a[j,k] != 0 ergo s[ii,jj] != 0 */ - if (ii < jj && !map[jj]) ind[++len] = jj, map[jj] = 1; - } - } - /* now (ind) is pattern of ii-th row of S */ - S_ptr[ii+1] = S_ptr[ii] + len; - /* at least (S_ptr[ii+1] - 1) locations should be available in - the array S_ind */ - if (S_ptr[ii+1] - 1 > size) - { temp = S_ind; - size += size; - S_ind = xcalloc(1+size, sizeof(int)); - memcpy(&S_ind[1], &temp[1], (S_ptr[ii] - 1) * sizeof(int)); - xfree(temp); - } - xassert(S_ptr[ii+1] - 1 <= size); - /* (ii-th row of S) := (ind) */ - memcpy(&S_ind[S_ptr[ii]], &ind[1], len * sizeof(int)); - /* clear the row pattern map */ - for (t = 1; t <= len; t++) map[ind[t]] = 0; - } - /* free working arrays */ - xfree(AT_ptr); - xfree(AT_ind); - xfree(ind); - xfree(map); - /* reallocate the array S_ind to free unused locations */ - temp = S_ind; - size = S_ptr[m+1] - 1; - S_ind = xcalloc(1+size, sizeof(int)); - memcpy(&S_ind[1], &temp[1], size * sizeof(int)); - xfree(temp); - return S_ind; -} - -/*---------------------------------------------------------------------- --- adat_numeric - compute S = P*A*D*A'*P' (numeric phase). --- --- *Synopsis* --- --- #include "glpmat.h" --- void adat_numeric(int m, int n, int P_per[], --- int A_ptr[], int A_ind[], double A_val[], double D_diag[], --- int S_ptr[], int S_ind[], double S_val[], double S_diag[]); --- --- *Description* --- --- The routine adat_numeric implements the numeric phase to compute --- symmetric matrix S = P*A*D*A'*P', where P is a permutation matrix, --- A is a given sparse matrix, D is a diagonal matrix, A' is a matrix --- transposed to A, P' is an inverse of P. --- --- The parameter m is the number of rows in A and the order of P. --- --- The parameter n is the number of columns in A and the order of D. --- --- The matrix P is specified in the array P_per, which is not changed --- on exit. --- --- The matrix A is specified in the arrays A_ptr, A_ind, and A_val in --- storage-by-rows format. These arrays are not changed on exit. --- --- Diagonal elements of the matrix D are specified in the array D_diag, --- where D_diag[0] is not used, D_diag[i] = d[i,i] for i = 1, ..., n. --- The array D_diag is not changed on exit. --- --- The pattern of the upper triangular part of the matrix S without --- diagonal elements (previously computed by the routine adat_symbolic) --- is specified in the arrays S_ptr and S_ind, which are not changed on --- exit. Numeric values of non-diagonal elements of S are stored in --- corresponding locations of the array S_val, and values of diagonal --- elements of S are stored in locations S_diag[1], ..., S_diag[n]. */ - -void adat_numeric(int m, int n, int P_per[], - int A_ptr[], int A_ind[], double A_val[], double D_diag[], - int S_ptr[], int S_ind[], double S_val[], double S_diag[]) -{ int i, j, t, ii, jj, tt, beg, end, beg1, end1, k; - double sum, *work; - work = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) work[j] = 0.0; - /* compute S = B*D*B', where B = P*A, B' is a matrix transposed - to B */ - for (ii = 1; ii <= m; ii++) - { i = P_per[ii]; /* i-th row of A = ii-th row of B */ - /* (work) := (i-th row of A) */ - beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) - work[A_ind[t]] = A_val[t]; - /* compute ii-th row of S */ - beg = S_ptr[ii], end = S_ptr[ii+1]; - for (t = beg; t < end; t++) - { jj = S_ind[t]; - j = P_per[jj]; /* j-th row of A = jj-th row of B */ - /* s[ii,jj] := sum a[i,k] * d[k,k] * a[j,k] */ - sum = 0.0; - beg1 = A_ptr[j], end1 = A_ptr[j+1]; - for (tt = beg1; tt < end1; tt++) - { k = A_ind[tt]; - sum += work[k] * D_diag[k] * A_val[tt]; - } - S_val[t] = sum; - } - /* s[ii,ii] := sum a[i,k] * d[k,k] * a[i,k] */ - sum = 0.0; - beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) - { k = A_ind[t]; - sum += A_val[t] * D_diag[k] * A_val[t]; - work[k] = 0.0; - } - S_diag[ii] = sum; - } - xfree(work); - return; -} - -/*---------------------------------------------------------------------- --- min_degree - minimum degree ordering. --- --- *Synopsis* --- --- #include "glpmat.h" --- void min_degree(int n, int A_ptr[], int A_ind[], int P_per[]); --- --- *Description* --- --- The routine min_degree uses the minimum degree ordering algorithm --- to find a permutation matrix P for a given sparse symmetric positive --- matrix A which minimizes the number of non-zeros in upper triangular --- factor U for Cholesky factorization P*A*P' = U'*U. --- --- The parameter n is the order of matrices A and P. --- --- The pattern of the given matrix A is specified on entry in the arrays --- A_ptr and A_ind in storage-by-rows format. Only the upper triangular --- part without diagonal elements (which all are assumed to be non-zero) --- should be specified as if A were upper triangular. The arrays A_ptr --- and A_ind are not changed on exit. --- --- The permutation matrix P is stored by the routine in the array P_per --- on exit. --- --- *Algorithm* --- --- The routine min_degree is based on some subroutines from the package --- SPARSPAK (see comments in the module glpqmd). */ - -void min_degree(int n, int A_ptr[], int A_ind[], int P_per[]) -{ int i, j, ne, t, pos, len; - int *xadj, *adjncy, *deg, *marker, *rchset, *nbrhd, *qsize, - *qlink, nofsub; - /* determine number of non-zeros in complete pattern */ - ne = A_ptr[n+1] - 1; - ne += ne; - /* allocate working arrays */ - xadj = xcalloc(1+n+1, sizeof(int)); - adjncy = xcalloc(1+ne, sizeof(int)); - deg = xcalloc(1+n, sizeof(int)); - marker = xcalloc(1+n, sizeof(int)); - rchset = xcalloc(1+n, sizeof(int)); - nbrhd = xcalloc(1+n, sizeof(int)); - qsize = xcalloc(1+n, sizeof(int)); - qlink = xcalloc(1+n, sizeof(int)); - /* determine row lengths in complete pattern */ - for (i = 1; i <= n; i++) xadj[i] = 0; - for (i = 1; i <= n; i++) - { for (t = A_ptr[i]; t < A_ptr[i+1]; t++) - { j = A_ind[t]; - xassert(i < j && j <= n); - xadj[i]++, xadj[j]++; - } - } - /* set up row pointers for complete pattern */ - pos = 1; - for (i = 1; i <= n; i++) - len = xadj[i], pos += len, xadj[i] = pos; - xadj[n+1] = pos; - xassert(pos - 1 == ne); - /* construct complete pattern */ - for (i = 1; i <= n; i++) - { for (t = A_ptr[i]; t < A_ptr[i+1]; t++) - { j = A_ind[t]; - adjncy[--xadj[i]] = j, adjncy[--xadj[j]] = i; - } - } - /* call the main minimimum degree ordering routine */ - genqmd(&n, xadj, adjncy, P_per, P_per + n, deg, marker, rchset, - nbrhd, qsize, qlink, &nofsub); - /* make sure that permutation matrix P is correct */ - for (i = 1; i <= n; i++) - { j = P_per[i]; - xassert(1 <= j && j <= n); - xassert(P_per[n+j] == i); - } - /* free working arrays */ - xfree(xadj); - xfree(adjncy); - xfree(deg); - xfree(marker); - xfree(rchset); - xfree(nbrhd); - xfree(qsize); - xfree(qlink); - return; -} - -/**********************************************************************/ - -void amd_order1(int n, int A_ptr[], int A_ind[], int P_per[]) -{ /* approximate minimum degree ordering (AMD) */ - int k, ret; - double Control[AMD_CONTROL], Info[AMD_INFO]; - /* get the default parameters */ - amd_defaults(Control); -#if 0 - /* and print them */ - amd_control(Control); -#endif - /* make all indices 0-based */ - for (k = 1; k < A_ptr[n+1]; k++) A_ind[k]--; - for (k = 1; k <= n+1; k++) A_ptr[k]--; - /* call the ordering routine */ - ret = amd_order(n, &A_ptr[1], &A_ind[1], &P_per[1], Control, Info) - ; -#if 0 - amd_info(Info); -#endif - xassert(ret == AMD_OK || ret == AMD_OK_BUT_JUMBLED); - /* retsore 1-based indices */ - for (k = 1; k <= n+1; k++) A_ptr[k]++; - for (k = 1; k < A_ptr[n+1]; k++) A_ind[k]++; - /* patch up permutation matrix */ - memset(&P_per[n+1], 0, n * sizeof(int)); - for (k = 1; k <= n; k++) - { P_per[k]++; - xassert(1 <= P_per[k] && P_per[k] <= n); - xassert(P_per[n+P_per[k]] == 0); - P_per[n+P_per[k]] = k; - } - return; -} - -/**********************************************************************/ - -static void *allocate(size_t n, size_t size) -{ void *ptr; - ptr = xcalloc(n, size); - memset(ptr, 0, n * size); - return ptr; -} - -static void release(void *ptr) -{ xfree(ptr); - return; -} - -void symamd_ord(int n, int A_ptr[], int A_ind[], int P_per[]) -{ /* approximate minimum degree ordering (SYMAMD) */ - int k, ok; - int stats[COLAMD_STATS]; - /* make all indices 0-based */ - for (k = 1; k < A_ptr[n+1]; k++) A_ind[k]--; - for (k = 1; k <= n+1; k++) A_ptr[k]--; - /* call the ordering routine */ - ok = symamd(n, &A_ind[1], &A_ptr[1], &P_per[1], NULL, stats, - allocate, release); -#if 0 - symamd_report(stats); -#endif - xassert(ok); - /* restore 1-based indices */ - for (k = 1; k <= n+1; k++) A_ptr[k]++; - for (k = 1; k < A_ptr[n+1]; k++) A_ind[k]++; - /* patch up permutation matrix */ - memset(&P_per[n+1], 0, n * sizeof(int)); - for (k = 1; k <= n; k++) - { P_per[k]++; - xassert(1 <= P_per[k] && P_per[k] <= n); - xassert(P_per[n+P_per[k]] == 0); - P_per[n+P_per[k]] = k; - } - return; -} - -/*---------------------------------------------------------------------- --- chol_symbolic - compute Cholesky factorization (symbolic phase). --- --- *Synopsis* --- --- #include "glpmat.h" --- int *chol_symbolic(int n, int A_ptr[], int A_ind[], int U_ptr[]); --- --- *Description* --- --- The routine chol_symbolic implements the symbolic phase of Cholesky --- factorization A = U'*U, where A is a given sparse symmetric positive --- definite matrix, U is a resultant upper triangular factor, U' is a --- matrix transposed to U. --- --- The parameter n is the order of matrices A and U. --- --- The pattern of the given matrix A is specified on entry in the arrays --- A_ptr and A_ind in storage-by-rows format. Only the upper triangular --- part without diagonal elements (which all are assumed to be non-zero) --- should be specified as if A were upper triangular. The arrays A_ptr --- and A_ind are not changed on exit. --- --- The pattern of the matrix U without diagonal elements (which all are --- assumed to be non-zero) is stored on exit from the routine in the --- arrays U_ptr and U_ind in storage-by-rows format. The array U_ptr --- should be allocated on entry, however, its content is ignored. The --- array U_ind is allocated by the routine which returns a pointer to it --- on exit. --- --- *Returns* --- --- The routine returns a pointer to the array U_ind. --- --- *Method* --- --- The routine chol_symbolic computes the pattern of the matrix U in a --- row-wise manner. No pivoting is used. --- --- It is known that to compute the pattern of row k of the matrix U we --- need to merge the pattern of row k of the matrix A and the patterns --- of each row i of U, where u[i,k] is non-zero (these rows are already --- computed and placed above row k). --- --- However, to reduce the number of rows to be merged the routine uses --- an advanced algorithm proposed in: --- --- D.J.Rose, R.E.Tarjan, and G.S.Lueker. Algorithmic aspects of vertex --- elimination on graphs. SIAM J. Comput. 5, 1976, 266-83. --- --- The authors of the cited paper show that we have the same result if --- we merge row k of the matrix A and such rows of the matrix U (among --- rows 1, ..., k-1) whose leftmost non-diagonal non-zero element is --- placed in k-th column. This feature signficantly reduces the number --- of rows to be merged, especially on the final steps, where rows of --- the matrix U become quite dense. --- --- To determine rows, which should be merged on k-th step, for a fixed --- time the routine uses linked lists of row numbers of the matrix U. --- Location head[k] contains the number of a first row, whose leftmost --- non-diagonal non-zero element is placed in column k, and location --- next[i] contains the number of a next row with the same property as --- row i. */ - -int *chol_symbolic(int n, int A_ptr[], int A_ind[], int U_ptr[]) -{ int i, j, k, t, len, size, beg, end, min_j, *U_ind, *head, *next, - *ind, *map, *temp; - /* initially we assume that on computing the pattern of U fill-in - will double the number of non-zeros in A */ - size = A_ptr[n+1] - 1; - if (size < n) size = n; - size += size; - U_ind = xcalloc(1+size, sizeof(int)); - /* allocate and initialize working arrays */ - head = xcalloc(1+n, sizeof(int)); - for (i = 1; i <= n; i++) head[i] = 0; - next = xcalloc(1+n, sizeof(int)); - ind = xcalloc(1+n, sizeof(int)); - map = xcalloc(1+n, sizeof(int)); - for (j = 1; j <= n; j++) map[j] = 0; - /* compute the pattern of matrix U */ - U_ptr[1] = 1; - for (k = 1; k <= n; k++) - { /* compute the pattern of k-th row of U, which is the union of - k-th row of A and those rows of U (among 1, ..., k-1) whose - leftmost non-diagonal non-zero is placed in k-th column */ - /* (ind) := (k-th row of A) */ - len = A_ptr[k+1] - A_ptr[k]; - memcpy(&ind[1], &A_ind[A_ptr[k]], len * sizeof(int)); - for (t = 1; t <= len; t++) - { j = ind[t]; - xassert(k < j && j <= n); - map[j] = 1; - } - /* walk through rows of U whose leftmost non-diagonal non-zero - is placed in k-th column */ - for (i = head[k]; i != 0; i = next[i]) - { /* (ind) := (ind) union (i-th row of U) */ - beg = U_ptr[i], end = U_ptr[i+1]; - for (t = beg; t < end; t++) - { j = U_ind[t]; - if (j > k && !map[j]) ind[++len] = j, map[j] = 1; - } - } - /* now (ind) is the pattern of k-th row of U */ - U_ptr[k+1] = U_ptr[k] + len; - /* at least (U_ptr[k+1] - 1) locations should be available in - the array U_ind */ - if (U_ptr[k+1] - 1 > size) - { temp = U_ind; - size += size; - U_ind = xcalloc(1+size, sizeof(int)); - memcpy(&U_ind[1], &temp[1], (U_ptr[k] - 1) * sizeof(int)); - xfree(temp); - } - xassert(U_ptr[k+1] - 1 <= size); - /* (k-th row of U) := (ind) */ - memcpy(&U_ind[U_ptr[k]], &ind[1], len * sizeof(int)); - /* determine column index of leftmost non-diagonal non-zero in - k-th row of U and clear the row pattern map */ - min_j = n + 1; - for (t = 1; t <= len; t++) - { j = ind[t], map[j] = 0; - if (min_j > j) min_j = j; - } - /* include k-th row into corresponding linked list */ - if (min_j <= n) next[k] = head[min_j], head[min_j] = k; - } - /* free working arrays */ - xfree(head); - xfree(next); - xfree(ind); - xfree(map); - /* reallocate the array U_ind to free unused locations */ - temp = U_ind; - size = U_ptr[n+1] - 1; - U_ind = xcalloc(1+size, sizeof(int)); - memcpy(&U_ind[1], &temp[1], size * sizeof(int)); - xfree(temp); - return U_ind; -} - -/*---------------------------------------------------------------------- --- chol_numeric - compute Cholesky factorization (numeric phase). --- --- *Synopsis* --- --- #include "glpmat.h" --- int chol_numeric(int n, --- int A_ptr[], int A_ind[], double A_val[], double A_diag[], --- int U_ptr[], int U_ind[], double U_val[], double U_diag[]); --- --- *Description* --- --- The routine chol_symbolic implements the numeric phase of Cholesky --- factorization A = U'*U, where A is a given sparse symmetric positive --- definite matrix, U is a resultant upper triangular factor, U' is a --- matrix transposed to U. --- --- The parameter n is the order of matrices A and U. --- --- Upper triangular part of the matrix A without diagonal elements is --- specified in the arrays A_ptr, A_ind, and A_val in storage-by-rows --- format. Diagonal elements of A are specified in the array A_diag, --- where A_diag[0] is not used, A_diag[i] = a[i,i] for i = 1, ..., n. --- The arrays A_ptr, A_ind, A_val, and A_diag are not changed on exit. --- --- The pattern of the matrix U without diagonal elements (previously --- computed with the routine chol_symbolic) is specified in the arrays --- U_ptr and U_ind, which are not changed on exit. Numeric values of --- non-diagonal elements of U are stored in corresponding locations of --- the array U_val, and values of diagonal elements of U are stored in --- locations U_diag[1], ..., U_diag[n]. --- --- *Returns* --- --- The routine returns the number of non-positive diagonal elements of --- the matrix U which have been replaced by a huge positive number (see --- the method description below). Zero return code means the matrix A --- has been successfully factorized. --- --- *Method* --- --- The routine chol_numeric computes the matrix U in a row-wise manner --- using standard gaussian elimination technique. No pivoting is used. --- --- Initially the routine sets U = A, and before k-th elimination step --- the matrix U is the following: --- --- 1 k n --- 1 x x x x x x x x x x --- . x x x x x x x x x --- . . x x x x x x x x --- . . . x x x x x x x --- k . . . . * * * * * * --- . . . . * * * * * * --- . . . . * * * * * * --- . . . . * * * * * * --- . . . . * * * * * * --- n . . . . * * * * * * --- --- where 'x' are elements of already computed rows, '*' are elements of --- the active submatrix. (Note that the lower triangular part of the --- active submatrix being symmetric is not stored and diagonal elements --- are stored separately in the array U_diag.) --- --- The matrix A is assumed to be positive definite. However, if it is --- close to semi-definite, on some elimination step a pivot u[k,k] may --- happen to be non-positive due to round-off errors. In this case the --- routine uses a technique proposed in: --- --- S.J.Wright. The Cholesky factorization in interior-point and barrier --- methods. Preprint MCS-P600-0596, Mathematics and Computer Science --- Division, Argonne National Laboratory, Argonne, Ill., May 1996. --- --- The routine just replaces non-positive u[k,k] by a huge positive --- number. This involves non-diagonal elements in k-th row of U to be --- close to zero that, in turn, involves k-th component of a solution --- vector to be close to zero. Note, however, that this technique works --- only if the system A*x = b is consistent. */ - -int chol_numeric(int n, - int A_ptr[], int A_ind[], double A_val[], double A_diag[], - int U_ptr[], int U_ind[], double U_val[], double U_diag[]) -{ int i, j, k, t, t1, beg, end, beg1, end1, count = 0; - double ukk, uki, *work; - work = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) work[j] = 0.0; - /* U := (upper triangle of A) */ - /* note that the upper traingle of A is a subset of U */ - for (i = 1; i <= n; i++) - { beg = A_ptr[i], end = A_ptr[i+1]; - for (t = beg; t < end; t++) - j = A_ind[t], work[j] = A_val[t]; - beg = U_ptr[i], end = U_ptr[i+1]; - for (t = beg; t < end; t++) - j = U_ind[t], U_val[t] = work[j], work[j] = 0.0; - U_diag[i] = A_diag[i]; - } - /* main elimination loop */ - for (k = 1; k <= n; k++) - { /* transform k-th row of U */ - ukk = U_diag[k]; - if (ukk > 0.0) - U_diag[k] = ukk = sqrt(ukk); - else - U_diag[k] = ukk = DBL_MAX, count++; - /* (work) := (transformed k-th row) */ - beg = U_ptr[k], end = U_ptr[k+1]; - for (t = beg; t < end; t++) - work[U_ind[t]] = (U_val[t] /= ukk); - /* transform other rows of U */ - for (t = beg; t < end; t++) - { i = U_ind[t]; - xassert(i > k); - /* (i-th row) := (i-th row) - u[k,i] * (k-th row) */ - uki = work[i]; - beg1 = U_ptr[i], end1 = U_ptr[i+1]; - for (t1 = beg1; t1 < end1; t1++) - U_val[t1] -= uki * work[U_ind[t1]]; - U_diag[i] -= uki * uki; - } - /* (work) := 0 */ - for (t = beg; t < end; t++) - work[U_ind[t]] = 0.0; - } - xfree(work); - return count; -} - -/*---------------------------------------------------------------------- --- u_solve - solve upper triangular system U*x = b. --- --- *Synopsis* --- --- #include "glpmat.h" --- void u_solve(int n, int U_ptr[], int U_ind[], double U_val[], --- double U_diag[], double x[]); --- --- *Description* --- --- The routine u_solve solves an linear system U*x = b, where U is an --- upper triangular matrix. --- --- The parameter n is the order of matrix U. --- --- The matrix U without diagonal elements is specified in the arrays --- U_ptr, U_ind, and U_val in storage-by-rows format. Diagonal elements --- of U are specified in the array U_diag, where U_diag[0] is not used, --- U_diag[i] = u[i,i] for i = 1, ..., n. All these four arrays are not --- changed on exit. --- --- The right-hand side vector b is specified on entry in the array x, --- where x[0] is not used, and x[i] = b[i] for i = 1, ..., n. On exit --- the routine stores computed components of the vector of unknowns x --- in the array x in the same manner. */ - -void u_solve(int n, int U_ptr[], int U_ind[], double U_val[], - double U_diag[], double x[]) -{ int i, t, beg, end; - double temp; - for (i = n; i >= 1; i--) - { temp = x[i]; - beg = U_ptr[i], end = U_ptr[i+1]; - for (t = beg; t < end; t++) - temp -= U_val[t] * x[U_ind[t]]; - xassert(U_diag[i] != 0.0); - x[i] = temp / U_diag[i]; - } - return; -} - -/*---------------------------------------------------------------------- --- ut_solve - solve lower triangular system U'*x = b. --- --- *Synopsis* --- --- #include "glpmat.h" --- void ut_solve(int n, int U_ptr[], int U_ind[], double U_val[], --- double U_diag[], double x[]); --- --- *Description* --- --- The routine ut_solve solves an linear system U'*x = b, where U is a --- matrix transposed to an upper triangular matrix. --- --- The parameter n is the order of matrix U. --- --- The matrix U without diagonal elements is specified in the arrays --- U_ptr, U_ind, and U_val in storage-by-rows format. Diagonal elements --- of U are specified in the array U_diag, where U_diag[0] is not used, --- U_diag[i] = u[i,i] for i = 1, ..., n. All these four arrays are not --- changed on exit. --- --- The right-hand side vector b is specified on entry in the array x, --- where x[0] is not used, and x[i] = b[i] for i = 1, ..., n. On exit --- the routine stores computed components of the vector of unknowns x --- in the array x in the same manner. */ - -void ut_solve(int n, int U_ptr[], int U_ind[], double U_val[], - double U_diag[], double x[]) -{ int i, t, beg, end; - double temp; - for (i = 1; i <= n; i++) - { xassert(U_diag[i] != 0.0); - temp = (x[i] /= U_diag[i]); - if (temp == 0.0) continue; - beg = U_ptr[i], end = U_ptr[i+1]; - for (t = beg; t < end; t++) - x[U_ind[t]] -= U_val[t] * temp; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpmat.h b/code/3rd_glpk/draft/glpmat.h deleted file mode 100644 index 5b058437..00000000 --- a/code/3rd_glpk/draft/glpmat.h +++ /dev/null @@ -1,198 +0,0 @@ -/* glpmat.h (linear algebra routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPMAT_H -#define GLPMAT_H - -/*********************************************************************** -* FULL-VECTOR STORAGE -* -* For a sparse vector x having n elements, ne of which are non-zero, -* the full-vector storage format uses two arrays x_ind and x_vec, which -* are set up as follows: -* -* x_ind is an integer array of length [1+ne]. Location x_ind[0] is -* not used, and locations x_ind[1], ..., x_ind[ne] contain indices of -* non-zero elements in vector x. -* -* x_vec is a floating-point array of length [1+n]. Location x_vec[0] -* is not used, and locations x_vec[1], ..., x_vec[n] contain numeric -* values of ALL elements in vector x, including its zero elements. -* -* Let, for example, the following sparse vector x be given: -* -* (0, 1, 0, 0, 2, 3, 0, 4) -* -* Then the arrays are: -* -* x_ind = { X; 2, 5, 6, 8 } -* -* x_vec = { X; 0, 1, 0, 0, 2, 3, 0, 4 } -* -* COMPRESSED-VECTOR STORAGE -* -* For a sparse vector x having n elements, ne of which are non-zero, -* the compressed-vector storage format uses two arrays x_ind and x_vec, -* which are set up as follows: -* -* x_ind is an integer array of length [1+ne]. Location x_ind[0] is -* not used, and locations x_ind[1], ..., x_ind[ne] contain indices of -* non-zero elements in vector x. -* -* x_vec is a floating-point array of length [1+ne]. Location x_vec[0] -* is not used, and locations x_vec[1], ..., x_vec[ne] contain numeric -* values of corresponding non-zero elements in vector x. -* -* Let, for example, the following sparse vector x be given: -* -* (0, 1, 0, 0, 2, 3, 0, 4) -* -* Then the arrays are: -* -* x_ind = { X; 2, 5, 6, 8 } -* -* x_vec = { X; 1, 2, 3, 4 } -* -* STORAGE-BY-ROWS -* -* For a sparse matrix A, which has m rows, n columns, and ne non-zero -* elements the storage-by-rows format uses three arrays A_ptr, A_ind, -* and A_val, which are set up as follows: -* -* A_ptr is an integer array of length [1+m+1] also called "row pointer -* array". It contains the relative starting positions of each row of A -* in the arrays A_ind and A_val, i.e. element A_ptr[i], 1 <= i <= m, -* indicates where row i begins in the arrays A_ind and A_val. If all -* elements in row i are zero, then A_ptr[i] = A_ptr[i+1]. Location -* A_ptr[0] is not used, location A_ptr[1] must contain 1, and location -* A_ptr[m+1] must contain ne+1 that indicates the position after the -* last element in the arrays A_ind and A_val. -* -* A_ind is an integer array of length [1+ne]. Location A_ind[0] is not -* used, and locations A_ind[1], ..., A_ind[ne] contain column indices -* of (non-zero) elements in matrix A. -* -* A_val is a floating-point array of length [1+ne]. Location A_val[0] -* is not used, and locations A_val[1], ..., A_val[ne] contain numeric -* values of non-zero elements in matrix A. -* -* Non-zero elements of matrix A are stored contiguously, and the rows -* of matrix A are stored consecutively from 1 to m in the arrays A_ind -* and A_val. The elements in each row of A may be stored in any order -* in A_ind and A_val. Note that elements with duplicate column indices -* are not allowed. -* -* Let, for example, the following sparse matrix A be given: -* -* | 11 . 13 . . . | -* | 21 22 . 24 . . | -* | . 32 33 . . . | -* | . . 43 44 . 46 | -* | . . . . . . | -* | 61 62 . . . 66 | -* -* Then the arrays are: -* -* A_ptr = { X; 1, 3, 6, 8, 11, 11; 14 } -* -* A_ind = { X; 1, 3; 4, 2, 1; 2, 3; 4, 3, 6; 1, 2, 6 } -* -* A_val = { X; 11, 13; 24, 22, 21; 32, 33; 44, 43, 46; 61, 62, 66 } -* -* PERMUTATION MATRICES -* -* Let P be a permutation matrix of the order n. It is represented as -* an integer array P_per of length [1+n+n] as follows: if p[i,j] = 1, -* then P_per[i] = j and P_per[n+j] = i. Location P_per[0] is not used. -* -* Let A' = P*A. If i-th row of A corresponds to i'-th row of A', then -* P_per[i'] = i and P_per[n+i] = i'. -* -* References: -* -* 1. Gustavson F.G. Some basic techniques for solving sparse systems of -* linear equations. In Rose and Willoughby (1972), pp. 41-52. -* -* 2. Basic Linear Algebra Subprograms Technical (BLAST) Forum Standard. -* University of Tennessee (2001). */ - -#define check_fvs _glp_mat_check_fvs -int check_fvs(int n, int nnz, int ind[], double vec[]); -/* check sparse vector in full-vector storage format */ - -#define check_pattern _glp_mat_check_pattern -int check_pattern(int m, int n, int A_ptr[], int A_ind[]); -/* check pattern of sparse matrix */ - -#define transpose _glp_mat_transpose -void transpose(int m, int n, int A_ptr[], int A_ind[], double A_val[], - int AT_ptr[], int AT_ind[], double AT_val[]); -/* transpose sparse matrix */ - -#define adat_symbolic _glp_mat_adat_symbolic -int *adat_symbolic(int m, int n, int P_per[], int A_ptr[], int A_ind[], - int S_ptr[]); -/* compute S = P*A*D*A'*P' (symbolic phase) */ - -#define adat_numeric _glp_mat_adat_numeric -void adat_numeric(int m, int n, int P_per[], - int A_ptr[], int A_ind[], double A_val[], double D_diag[], - int S_ptr[], int S_ind[], double S_val[], double S_diag[]); -/* compute S = P*A*D*A'*P' (numeric phase) */ - -#define min_degree _glp_mat_min_degree -void min_degree(int n, int A_ptr[], int A_ind[], int P_per[]); -/* minimum degree ordering */ - -#define amd_order1 _glp_mat_amd_order1 -void amd_order1(int n, int A_ptr[], int A_ind[], int P_per[]); -/* approximate minimum degree ordering (AMD) */ - -#define symamd_ord _glp_mat_symamd_ord -void symamd_ord(int n, int A_ptr[], int A_ind[], int P_per[]); -/* approximate minimum degree ordering (SYMAMD) */ - -#define chol_symbolic _glp_mat_chol_symbolic -int *chol_symbolic(int n, int A_ptr[], int A_ind[], int U_ptr[]); -/* compute Cholesky factorization (symbolic phase) */ - -#define chol_numeric _glp_mat_chol_numeric -int chol_numeric(int n, - int A_ptr[], int A_ind[], double A_val[], double A_diag[], - int U_ptr[], int U_ind[], double U_val[], double U_diag[]); -/* compute Cholesky factorization (numeric phase) */ - -#define u_solve _glp_mat_u_solve -void u_solve(int n, int U_ptr[], int U_ind[], double U_val[], - double U_diag[], double x[]); -/* solve upper triangular system U*x = b */ - -#define ut_solve _glp_mat_ut_solve -void ut_solve(int n, int U_ptr[], int U_ind[], double U_val[], - double U_diag[], double x[]); -/* solve lower triangular system U'*x = b */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glprgr.c b/code/3rd_glpk/draft/glprgr.c deleted file mode 100644 index fbff6b8d..00000000 --- a/code/3rd_glpk/draft/glprgr.c +++ /dev/null @@ -1,173 +0,0 @@ -/* glprgr.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#define _GLPSTD_ERRNO -#define _GLPSTD_STDIO -#include "env.h" -#include "glprgr.h" -#define xfault xerror - -/*********************************************************************** -* NAME -* -* rgr_write_bmp16 - write 16-color raster image in BMP file format -* -* SYNOPSIS -* -* #include "glprgr.h" -* int rgr_write_bmp16(const char *fname, int m, int n, const char -* map[]); -* -* DESCRIPTION -* -* The routine rgr_write_bmp16 writes 16-color raster image in -* uncompressed BMP file format (Windows bitmap) to a binary file whose -* name is specified by the character string fname. -* -* The parameters m and n specify, respectively, the number of rows and -* the numbers of columns (i.e. height and width) of the raster image. -* -* The character array map has m*n elements. Elements map[0, ..., n-1] -* correspond to the first (top) scanline, elements map[n, ..., 2*n-1] -* correspond to the second scanline, etc. -* -* Each element of the array map specifies a color of the corresponding -* pixel as 8-bit binary number XXXXIRGB, where four high-order bits (X) -* are ignored, I is high intensity bit, R is red color bit, G is green -* color bit, and B is blue color bit. Thus, all 16 possible colors are -* coded as following hexadecimal numbers: -* -* 0x00 = black 0x08 = dark gray -* 0x01 = blue 0x09 = bright blue -* 0x02 = green 0x0A = bright green -* 0x03 = cyan 0x0B = bright cyan -* 0x04 = red 0x0C = bright red -* 0x05 = magenta 0x0D = bright magenta -* 0x06 = brown 0x0E = yellow -* 0x07 = light gray 0x0F = white -* -* RETURNS -* -* If no error occured, the routine returns zero; otherwise, it prints -* an appropriate error message and returns non-zero. */ - -static void put_byte(FILE *fp, int c) -{ fputc(c, fp); - return; -} - -static void put_word(FILE *fp, int w) -{ /* big endian */ - put_byte(fp, w); - put_byte(fp, w >> 8); - return; -} - -static void put_dword(FILE *fp, int d) -{ /* big endian */ - put_word(fp, d); - put_word(fp, d >> 16); - return; -} - -int rgr_write_bmp16(const char *fname, int m, int n, const char map[]) -{ FILE *fp; - int offset, bmsize, i, j, b, ret = 0; - if (!(1 <= m && m <= 32767)) - xfault("rgr_write_bmp16: m = %d; invalid height\n", m); - if (!(1 <= n && n <= 32767)) - xfault("rgr_write_bmp16: n = %d; invalid width\n", n); - fp = fopen(fname, "wb"); - if (fp == NULL) - { xprintf("rgr_write_bmp16: unable to create '%s' - %s\n", -#if 0 /* 29/I-2017 */ - fname, strerror(errno)); -#else - fname, xstrerr(errno)); -#endif - ret = 1; - goto fini; - } - offset = 14 + 40 + 16 * 4; - bmsize = (4 * n + 31) / 32; - /* struct BMPFILEHEADER (14 bytes) */ - /* UINT bfType */ put_byte(fp, 'B'), put_byte(fp, 'M'); - /* DWORD bfSize */ put_dword(fp, offset + bmsize * 4); - /* UINT bfReserved1 */ put_word(fp, 0); - /* UNIT bfReserved2 */ put_word(fp, 0); - /* DWORD bfOffBits */ put_dword(fp, offset); - /* struct BMPINFOHEADER (40 bytes) */ - /* DWORD biSize */ put_dword(fp, 40); - /* LONG biWidth */ put_dword(fp, n); - /* LONG biHeight */ put_dword(fp, m); - /* WORD biPlanes */ put_word(fp, 1); - /* WORD biBitCount */ put_word(fp, 4); - /* DWORD biCompression */ put_dword(fp, 0 /* BI_RGB */); - /* DWORD biSizeImage */ put_dword(fp, 0); - /* LONG biXPelsPerMeter */ put_dword(fp, 2953 /* 75 dpi */); - /* LONG biYPelsPerMeter */ put_dword(fp, 2953 /* 75 dpi */); - /* DWORD biClrUsed */ put_dword(fp, 0); - /* DWORD biClrImportant */ put_dword(fp, 0); - /* struct RGBQUAD (16 * 4 = 64 bytes) */ - /* CGA-compatible colors: */ - /* 0x00 = black */ put_dword(fp, 0x000000); - /* 0x01 = blue */ put_dword(fp, 0x000080); - /* 0x02 = green */ put_dword(fp, 0x008000); - /* 0x03 = cyan */ put_dword(fp, 0x008080); - /* 0x04 = red */ put_dword(fp, 0x800000); - /* 0x05 = magenta */ put_dword(fp, 0x800080); - /* 0x06 = brown */ put_dword(fp, 0x808000); - /* 0x07 = light gray */ put_dword(fp, 0xC0C0C0); - /* 0x08 = dark gray */ put_dword(fp, 0x808080); - /* 0x09 = bright blue */ put_dword(fp, 0x0000FF); - /* 0x0A = bright green */ put_dword(fp, 0x00FF00); - /* 0x0B = bright cyan */ put_dword(fp, 0x00FFFF); - /* 0x0C = bright red */ put_dword(fp, 0xFF0000); - /* 0x0D = bright magenta */ put_dword(fp, 0xFF00FF); - /* 0x0E = yellow */ put_dword(fp, 0xFFFF00); - /* 0x0F = white */ put_dword(fp, 0xFFFFFF); - /* pixel data bits */ - b = 0; - for (i = m - 1; i >= 0; i--) - { for (j = 0; j < ((n + 7) / 8) * 8; j++) - { b <<= 4; - b |= (j < n ? map[i * n + j] & 15 : 0); - if (j & 1) put_byte(fp, b); - } - } - fflush(fp); - if (ferror(fp)) - { xprintf("rgr_write_bmp16: write error on '%s' - %s\n", -#if 0 /* 29/I-2017 */ - fname, strerror(errno)); -#else - fname, xstrerr(errno)); -#endif - ret = 1; - } -fini: if (fp != NULL) fclose(fp); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glprgr.h b/code/3rd_glpk/draft/glprgr.h deleted file mode 100644 index 71e089e9..00000000 --- a/code/3rd_glpk/draft/glprgr.h +++ /dev/null @@ -1,34 +0,0 @@ -/* glprgr.h (raster graphics) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPRGR_H -#define GLPRGR_H - -#define rgr_write_bmp16 _glp_rgr_write_bmp16 -int rgr_write_bmp16(const char *fname, int m, int n, const char map[]); -/* write 16-color raster image in BMP file format */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpscl.c b/code/3rd_glpk/draft/glpscl.c deleted file mode 100644 index de769a8b..00000000 --- a/code/3rd_glpk/draft/glpscl.c +++ /dev/null @@ -1,478 +0,0 @@ -/* glpscl.c (problem scaling routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "misc.h" -#include "prob.h" - -/*********************************************************************** -* min_row_aij - determine minimal |a[i,j]| in i-th row -* -* This routine returns minimal magnitude of (non-zero) constraint -* coefficients in i-th row of the constraint matrix. -* -* If the parameter scaled is zero, the original constraint matrix A is -* assumed. Otherwise, the scaled constraint matrix R*A*S is assumed. -* -* If i-th row of the matrix is empty, the routine returns 1. */ - -static double min_row_aij(glp_prob *lp, int i, int scaled) -{ GLPAIJ *aij; - double min_aij, temp; - xassert(1 <= i && i <= lp->m); - min_aij = 1.0; - for (aij = lp->row[i]->ptr; aij != NULL; aij = aij->r_next) - { temp = fabs(aij->val); - if (scaled) temp *= (aij->row->rii * aij->col->sjj); - if (aij->r_prev == NULL || min_aij > temp) - min_aij = temp; - } - return min_aij; -} - -/*********************************************************************** -* max_row_aij - determine maximal |a[i,j]| in i-th row -* -* This routine returns maximal magnitude of (non-zero) constraint -* coefficients in i-th row of the constraint matrix. -* -* If the parameter scaled is zero, the original constraint matrix A is -* assumed. Otherwise, the scaled constraint matrix R*A*S is assumed. -* -* If i-th row of the matrix is empty, the routine returns 1. */ - -static double max_row_aij(glp_prob *lp, int i, int scaled) -{ GLPAIJ *aij; - double max_aij, temp; - xassert(1 <= i && i <= lp->m); - max_aij = 1.0; - for (aij = lp->row[i]->ptr; aij != NULL; aij = aij->r_next) - { temp = fabs(aij->val); - if (scaled) temp *= (aij->row->rii * aij->col->sjj); - if (aij->r_prev == NULL || max_aij < temp) - max_aij = temp; - } - return max_aij; -} - -/*********************************************************************** -* min_col_aij - determine minimal |a[i,j]| in j-th column -* -* This routine returns minimal magnitude of (non-zero) constraint -* coefficients in j-th column of the constraint matrix. -* -* If the parameter scaled is zero, the original constraint matrix A is -* assumed. Otherwise, the scaled constraint matrix R*A*S is assumed. -* -* If j-th column of the matrix is empty, the routine returns 1. */ - -static double min_col_aij(glp_prob *lp, int j, int scaled) -{ GLPAIJ *aij; - double min_aij, temp; - xassert(1 <= j && j <= lp->n); - min_aij = 1.0; - for (aij = lp->col[j]->ptr; aij != NULL; aij = aij->c_next) - { temp = fabs(aij->val); - if (scaled) temp *= (aij->row->rii * aij->col->sjj); - if (aij->c_prev == NULL || min_aij > temp) - min_aij = temp; - } - return min_aij; -} - -/*********************************************************************** -* max_col_aij - determine maximal |a[i,j]| in j-th column -* -* This routine returns maximal magnitude of (non-zero) constraint -* coefficients in j-th column of the constraint matrix. -* -* If the parameter scaled is zero, the original constraint matrix A is -* assumed. Otherwise, the scaled constraint matrix R*A*S is assumed. -* -* If j-th column of the matrix is empty, the routine returns 1. */ - -static double max_col_aij(glp_prob *lp, int j, int scaled) -{ GLPAIJ *aij; - double max_aij, temp; - xassert(1 <= j && j <= lp->n); - max_aij = 1.0; - for (aij = lp->col[j]->ptr; aij != NULL; aij = aij->c_next) - { temp = fabs(aij->val); - if (scaled) temp *= (aij->row->rii * aij->col->sjj); - if (aij->c_prev == NULL || max_aij < temp) - max_aij = temp; - } - return max_aij; -} - -/*********************************************************************** -* min_mat_aij - determine minimal |a[i,j]| in constraint matrix -* -* This routine returns minimal magnitude of (non-zero) constraint -* coefficients in the constraint matrix. -* -* If the parameter scaled is zero, the original constraint matrix A is -* assumed. Otherwise, the scaled constraint matrix R*A*S is assumed. -* -* If the matrix is empty, the routine returns 1. */ - -static double min_mat_aij(glp_prob *lp, int scaled) -{ int i; - double min_aij, temp; - min_aij = 1.0; - for (i = 1; i <= lp->m; i++) - { temp = min_row_aij(lp, i, scaled); - if (i == 1 || min_aij > temp) - min_aij = temp; - } - return min_aij; -} - -/*********************************************************************** -* max_mat_aij - determine maximal |a[i,j]| in constraint matrix -* -* This routine returns maximal magnitude of (non-zero) constraint -* coefficients in the constraint matrix. -* -* If the parameter scaled is zero, the original constraint matrix A is -* assumed. Otherwise, the scaled constraint matrix R*A*S is assumed. -* -* If the matrix is empty, the routine returns 1. */ - -static double max_mat_aij(glp_prob *lp, int scaled) -{ int i; - double max_aij, temp; - max_aij = 1.0; - for (i = 1; i <= lp->m; i++) - { temp = max_row_aij(lp, i, scaled); - if (i == 1 || max_aij < temp) - max_aij = temp; - } - return max_aij; -} - -/*********************************************************************** -* eq_scaling - perform equilibration scaling -* -* This routine performs equilibration scaling of rows and columns of -* the constraint matrix. -* -* If the parameter flag is zero, the routine scales rows at first and -* then columns. Otherwise, the routine scales columns and then rows. -* -* Rows are scaled as follows: -* -* n -* a'[i,j] = a[i,j] / max |a[i,j]|, i = 1,...,m. -* j=1 -* -* This makes the infinity (maximum) norm of each row of the matrix -* equal to 1. -* -* Columns are scaled as follows: -* -* m -* a'[i,j] = a[i,j] / max |a[i,j]|, j = 1,...,n. -* i=1 -* -* This makes the infinity (maximum) norm of each column of the matrix -* equal to 1. */ - -static void eq_scaling(glp_prob *lp, int flag) -{ int i, j, pass; - double temp; - xassert(flag == 0 || flag == 1); - for (pass = 0; pass <= 1; pass++) - { if (pass == flag) - { /* scale rows */ - for (i = 1; i <= lp->m; i++) - { temp = max_row_aij(lp, i, 1); - glp_set_rii(lp, i, glp_get_rii(lp, i) / temp); - } - } - else - { /* scale columns */ - for (j = 1; j <= lp->n; j++) - { temp = max_col_aij(lp, j, 1); - glp_set_sjj(lp, j, glp_get_sjj(lp, j) / temp); - } - } - } - return; -} - -/*********************************************************************** -* gm_scaling - perform geometric mean scaling -* -* This routine performs geometric mean scaling of rows and columns of -* the constraint matrix. -* -* If the parameter flag is zero, the routine scales rows at first and -* then columns. Otherwise, the routine scales columns and then rows. -* -* Rows are scaled as follows: -* -* a'[i,j] = a[i,j] / sqrt(alfa[i] * beta[i]), i = 1,...,m, -* -* where: -* n n -* alfa[i] = min |a[i,j]|, beta[i] = max |a[i,j]|. -* j=1 j=1 -* -* This allows decreasing the ratio beta[i] / alfa[i] for each row of -* the matrix. -* -* Columns are scaled as follows: -* -* a'[i,j] = a[i,j] / sqrt(alfa[j] * beta[j]), j = 1,...,n, -* -* where: -* m m -* alfa[j] = min |a[i,j]|, beta[j] = max |a[i,j]|. -* i=1 i=1 -* -* This allows decreasing the ratio beta[j] / alfa[j] for each column -* of the matrix. */ - -static void gm_scaling(glp_prob *lp, int flag) -{ int i, j, pass; - double temp; - xassert(flag == 0 || flag == 1); - for (pass = 0; pass <= 1; pass++) - { if (pass == flag) - { /* scale rows */ - for (i = 1; i <= lp->m; i++) - { temp = min_row_aij(lp, i, 1) * max_row_aij(lp, i, 1); - glp_set_rii(lp, i, glp_get_rii(lp, i) / sqrt(temp)); - } - } - else - { /* scale columns */ - for (j = 1; j <= lp->n; j++) - { temp = min_col_aij(lp, j, 1) * max_col_aij(lp, j, 1); - glp_set_sjj(lp, j, glp_get_sjj(lp, j) / sqrt(temp)); - } - } - } - return; -} - -/*********************************************************************** -* max_row_ratio - determine worst scaling "quality" for rows -* -* This routine returns the worst scaling "quality" for rows of the -* currently scaled constraint matrix: -* -* m -* ratio = max ratio[i], -* i=1 -* where: -* n n -* ratio[i] = max |a[i,j]| / min |a[i,j]|, 1 <= i <= m, -* j=1 j=1 -* -* is the scaling "quality" of i-th row. */ - -static double max_row_ratio(glp_prob *lp) -{ int i; - double ratio, temp; - ratio = 1.0; - for (i = 1; i <= lp->m; i++) - { temp = max_row_aij(lp, i, 1) / min_row_aij(lp, i, 1); - if (i == 1 || ratio < temp) ratio = temp; - } - return ratio; -} - -/*********************************************************************** -* max_col_ratio - determine worst scaling "quality" for columns -* -* This routine returns the worst scaling "quality" for columns of the -* currently scaled constraint matrix: -* -* n -* ratio = max ratio[j], -* j=1 -* where: -* m m -* ratio[j] = max |a[i,j]| / min |a[i,j]|, 1 <= j <= n, -* i=1 i=1 -* -* is the scaling "quality" of j-th column. */ - -static double max_col_ratio(glp_prob *lp) -{ int j; - double ratio, temp; - ratio = 1.0; - for (j = 1; j <= lp->n; j++) - { temp = max_col_aij(lp, j, 1) / min_col_aij(lp, j, 1); - if (j == 1 || ratio < temp) ratio = temp; - } - return ratio; -} - -/*********************************************************************** -* gm_iterate - perform iterative geometric mean scaling -* -* This routine performs iterative geometric mean scaling of rows and -* columns of the constraint matrix. -* -* The parameter it_max specifies the maximal number of iterations. -* Recommended value of it_max is 15. -* -* The parameter tau specifies a minimal improvement of the scaling -* "quality" on each iteration, 0 < tau < 1. It means than the scaling -* process continues while the following condition is satisfied: -* -* ratio[k] <= tau * ratio[k-1], -* -* where ratio = max |a[i,j]| / min |a[i,j]| is the scaling "quality" -* to be minimized, k is the iteration number. Recommended value of tau -* is 0.90. */ - -static void gm_iterate(glp_prob *lp, int it_max, double tau) -{ int k, flag; - double ratio = 0.0, r_old; - /* if the scaling "quality" for rows is better than for columns, - the rows are scaled first; otherwise, the columns are scaled - first */ - flag = (max_row_ratio(lp) > max_col_ratio(lp)); - for (k = 1; k <= it_max; k++) - { /* save the scaling "quality" from previous iteration */ - r_old = ratio; - /* determine the current scaling "quality" */ - ratio = max_mat_aij(lp, 1) / min_mat_aij(lp, 1); -#if 0 - xprintf("k = %d; ratio = %g\n", k, ratio); -#endif - /* if improvement is not enough, terminate scaling */ - if (k > 1 && ratio > tau * r_old) break; - /* otherwise, perform another iteration */ - gm_scaling(lp, flag); - } - return; -} - -/*********************************************************************** -* NAME -* -* scale_prob - scale problem data -* -* SYNOPSIS -* -* #include "glpscl.h" -* void scale_prob(glp_prob *lp, int flags); -* -* DESCRIPTION -* -* The routine scale_prob performs automatic scaling of problem data -* for the specified problem object. */ - -static void scale_prob(glp_prob *lp, int flags) -{ static const char *fmt = - "%s: min|aij| = %10.3e max|aij| = %10.3e ratio = %10.3e\n"; - double min_aij, max_aij, ratio; - xprintf("Scaling...\n"); - /* cancel the current scaling effect */ - glp_unscale_prob(lp); - /* report original scaling "quality" */ - min_aij = min_mat_aij(lp, 1); - max_aij = max_mat_aij(lp, 1); - ratio = max_aij / min_aij; - xprintf(fmt, " A", min_aij, max_aij, ratio); - /* check if the problem is well scaled */ - if (min_aij >= 0.10 && max_aij <= 10.0) - { xprintf("Problem data seem to be well scaled\n"); - /* skip scaling, if required */ - if (flags & GLP_SF_SKIP) goto done; - } - /* perform iterative geometric mean scaling, if required */ - if (flags & GLP_SF_GM) - { gm_iterate(lp, 15, 0.90); - min_aij = min_mat_aij(lp, 1); - max_aij = max_mat_aij(lp, 1); - ratio = max_aij / min_aij; - xprintf(fmt, "GM", min_aij, max_aij, ratio); - } - /* perform equilibration scaling, if required */ - if (flags & GLP_SF_EQ) - { eq_scaling(lp, max_row_ratio(lp) > max_col_ratio(lp)); - min_aij = min_mat_aij(lp, 1); - max_aij = max_mat_aij(lp, 1); - ratio = max_aij / min_aij; - xprintf(fmt, "EQ", min_aij, max_aij, ratio); - } - /* round scale factors to nearest power of two, if required */ - if (flags & GLP_SF_2N) - { int i, j; - for (i = 1; i <= lp->m; i++) - glp_set_rii(lp, i, round2n(glp_get_rii(lp, i))); - for (j = 1; j <= lp->n; j++) - glp_set_sjj(lp, j, round2n(glp_get_sjj(lp, j))); - min_aij = min_mat_aij(lp, 1); - max_aij = max_mat_aij(lp, 1); - ratio = max_aij / min_aij; - xprintf(fmt, "2N", min_aij, max_aij, ratio); - } -done: return; -} - -/*********************************************************************** -* NAME -* -* glp_scale_prob - scale problem data -* -* SYNOPSIS -* -* void glp_scale_prob(glp_prob *lp, int flags); -* -* DESCRIPTION -* -* The routine glp_scale_prob performs automatic scaling of problem -* data for the specified problem object. -* -* The parameter flags specifies scaling options used by the routine. -* Options can be combined with the bitwise OR operator and may be the -* following: -* -* GLP_SF_GM perform geometric mean scaling; -* GLP_SF_EQ perform equilibration scaling; -* GLP_SF_2N round scale factors to nearest power of two; -* GLP_SF_SKIP skip scaling, if the problem is well scaled. -* -* The parameter flags may be specified as GLP_SF_AUTO, in which case -* the routine chooses scaling options automatically. */ - -void glp_scale_prob(glp_prob *lp, int flags) -{ if (flags & ~(GLP_SF_GM | GLP_SF_EQ | GLP_SF_2N | GLP_SF_SKIP | - GLP_SF_AUTO)) - xerror("glp_scale_prob: flags = 0x%02X; invalid scaling option" - "s\n", flags); - if (flags & GLP_SF_AUTO) - flags = (GLP_SF_GM | GLP_SF_EQ | GLP_SF_SKIP); - scale_prob(lp, flags); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpspm.c b/code/3rd_glpk/draft/glpspm.c deleted file mode 100644 index c6cfd25d..00000000 --- a/code/3rd_glpk/draft/glpspm.c +++ /dev/null @@ -1,847 +0,0 @@ -/* glpspm.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "glphbm.h" -#include "glprgr.h" -#include "glpspm.h" -#include "env.h" - -/*********************************************************************** -* NAME -* -* spm_create_mat - create general sparse matrix -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPM *spm_create_mat(int m, int n); -* -* DESCRIPTION -* -* The routine spm_create_mat creates a general sparse matrix having -* m rows and n columns. Being created the matrix is zero (empty), i.e. -* has no elements. -* -* RETURNS -* -* The routine returns a pointer to the matrix created. */ - -SPM *spm_create_mat(int m, int n) -{ SPM *A; - xassert(0 <= m && m < INT_MAX); - xassert(0 <= n && n < INT_MAX); - A = xmalloc(sizeof(SPM)); - A->m = m; - A->n = n; - if (m == 0 || n == 0) - { A->pool = NULL; - A->row = NULL; - A->col = NULL; - } - else - { int i, j; - A->pool = dmp_create_pool(); - A->row = xcalloc(1+m, sizeof(SPME *)); - for (i = 1; i <= m; i++) A->row[i] = NULL; - A->col = xcalloc(1+n, sizeof(SPME *)); - for (j = 1; j <= n; j++) A->col[j] = NULL; - } - return A; -} - -/*********************************************************************** -* NAME -* -* spm_new_elem - add new element to sparse matrix -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPME *spm_new_elem(SPM *A, int i, int j, double val); -* -* DESCRIPTION -* -* The routine spm_new_elem adds a new element to the specified sparse -* matrix. Parameters i, j, and val specify the row number, the column -* number, and a numerical value of the element, respectively. -* -* RETURNS -* -* The routine returns a pointer to the new element added. */ - -SPME *spm_new_elem(SPM *A, int i, int j, double val) -{ SPME *e; - xassert(1 <= i && i <= A->m); - xassert(1 <= j && j <= A->n); - e = dmp_get_atom(A->pool, sizeof(SPME)); - e->i = i; - e->j = j; - e->val = val; - e->r_prev = NULL; - e->r_next = A->row[i]; - if (e->r_next != NULL) e->r_next->r_prev = e; - e->c_prev = NULL; - e->c_next = A->col[j]; - if (e->c_next != NULL) e->c_next->c_prev = e; - A->row[i] = A->col[j] = e; - return e; -} - -/*********************************************************************** -* NAME -* -* spm_delete_mat - delete general sparse matrix -* -* SYNOPSIS -* -* #include "glpspm.h" -* void spm_delete_mat(SPM *A); -* -* DESCRIPTION -* -* The routine deletes the specified general sparse matrix freeing all -* the memory allocated to this object. */ - -void spm_delete_mat(SPM *A) -{ /* delete sparse matrix */ - if (A->pool != NULL) dmp_delete_pool(A->pool); - if (A->row != NULL) xfree(A->row); - if (A->col != NULL) xfree(A->col); - xfree(A); - return; -} - -/*********************************************************************** -* NAME -* -* spm_test_mat_e - create test sparse matrix of E(n,c) class -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPM *spm_test_mat_e(int n, int c); -* -* DESCRIPTION -* -* The routine spm_test_mat_e creates a test sparse matrix of E(n,c) -* class as described in the book: Ole 0sterby, Zahari Zlatev. Direct -* Methods for Sparse Matrices. Springer-Verlag, 1983. -* -* Matrix of E(n,c) class is a symmetric positive definite matrix of -* the order n. It has the number 4 on its main diagonal and the number -* -1 on its four co-diagonals, two of which are neighbour to the main -* diagonal and two others are shifted from the main diagonal on the -* distance c. -* -* It is necessary that n >= 3 and 2 <= c <= n-1. -* -* RETURNS -* -* The routine returns a pointer to the matrix created. */ - -SPM *spm_test_mat_e(int n, int c) -{ SPM *A; - int i; - xassert(n >= 3 && 2 <= c && c <= n-1); - A = spm_create_mat(n, n); - for (i = 1; i <= n; i++) - spm_new_elem(A, i, i, 4.0); - for (i = 1; i <= n-1; i++) - { spm_new_elem(A, i, i+1, -1.0); - spm_new_elem(A, i+1, i, -1.0); - } - for (i = 1; i <= n-c; i++) - { spm_new_elem(A, i, i+c, -1.0); - spm_new_elem(A, i+c, i, -1.0); - } - return A; -} - -/*********************************************************************** -* NAME -* -* spm_test_mat_d - create test sparse matrix of D(n,c) class -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPM *spm_test_mat_d(int n, int c); -* -* DESCRIPTION -* -* The routine spm_test_mat_d creates a test sparse matrix of D(n,c) -* class as described in the book: Ole 0sterby, Zahari Zlatev. Direct -* Methods for Sparse Matrices. Springer-Verlag, 1983. -* -* Matrix of D(n,c) class is a non-singular matrix of the order n. It -* has unity main diagonal, three co-diagonals above the main diagonal -* on the distance c, which are cyclically continued below the main -* diagonal, and a triangle block of the size 10x10 in the upper right -* corner. -* -* It is necessary that n >= 14 and 1 <= c <= n-13. -* -* RETURNS -* -* The routine returns a pointer to the matrix created. */ - -SPM *spm_test_mat_d(int n, int c) -{ SPM *A; - int i, j; - xassert(n >= 14 && 1 <= c && c <= n-13); - A = spm_create_mat(n, n); - for (i = 1; i <= n; i++) - spm_new_elem(A, i, i, 1.0); - for (i = 1; i <= n-c; i++) - spm_new_elem(A, i, i+c, (double)(i+1)); - for (i = n-c+1; i <= n; i++) - spm_new_elem(A, i, i-n+c, (double)(i+1)); - for (i = 1; i <= n-c-1; i++) - spm_new_elem(A, i, i+c+1, (double)(-i)); - for (i = n-c; i <= n; i++) - spm_new_elem(A, i, i-n+c+1, (double)(-i)); - for (i = 1; i <= n-c-2; i++) - spm_new_elem(A, i, i+c+2, 16.0); - for (i = n-c-1; i <= n; i++) - spm_new_elem(A, i, i-n+c+2, 16.0); - for (j = 1; j <= 10; j++) - for (i = 1; i <= 11-j; i++) - spm_new_elem(A, i, n-11+i+j, 100.0 * (double)j); - return A; -} - -/*********************************************************************** -* NAME -* -* spm_show_mat - write sparse matrix pattern in BMP file format -* -* SYNOPSIS -* -* #include "glpspm.h" -* int spm_show_mat(const SPM *A, const char *fname); -* -* DESCRIPTION -* -* The routine spm_show_mat writes pattern of the specified sparse -* matrix in uncompressed BMP file format (Windows bitmap) to a binary -* file whose name is specified by the character string fname. -* -* Each pixel corresponds to one matrix element. The pixel colors have -* the following meaning: -* -* Black structurally zero element -* White positive element -* Cyan negative element -* Green zero element -* Red duplicate element -* -* RETURNS -* -* If no error occured, the routine returns zero. Otherwise, it prints -* an appropriate error message and returns non-zero. */ - -int spm_show_mat(const SPM *A, const char *fname) -{ int m = A->m; - int n = A->n; - int i, j, k, ret; - char *map; - xprintf("spm_show_mat: writing matrix pattern to '%s'...\n", - fname); - xassert(1 <= m && m <= 32767); - xassert(1 <= n && n <= 32767); - map = xmalloc(m * n); - memset(map, 0x08, m * n); - for (i = 1; i <= m; i++) - { SPME *e; - for (e = A->row[i]; e != NULL; e = e->r_next) - { j = e->j; - xassert(1 <= j && j <= n); - k = n * (i - 1) + (j - 1); - if (map[k] != 0x08) - map[k] = 0x0C; - else if (e->val > 0.0) - map[k] = 0x0F; - else if (e->val < 0.0) - map[k] = 0x0B; - else - map[k] = 0x0A; - } - } - ret = rgr_write_bmp16(fname, m, n, map); - xfree(map); - return ret; -} - -/*********************************************************************** -* NAME -* -* spm_read_hbm - read sparse matrix in Harwell-Boeing format -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPM *spm_read_hbm(const char *fname); -* -* DESCRIPTION -* -* The routine spm_read_hbm reads a sparse matrix in the Harwell-Boeing -* format from a text file whose name is the character string fname. -* -* Detailed description of the Harwell-Boeing format recognised by this -* routine can be found in the following report: -* -* I.S.Duff, R.G.Grimes, J.G.Lewis. User's Guide for the Harwell-Boeing -* Sparse Matrix Collection (Release I), TR/PA/92/86, October 1992. -* -* NOTE -* -* The routine spm_read_hbm reads the matrix "as is", due to which zero -* and/or duplicate elements can appear in the matrix. -* -* RETURNS -* -* If no error occured, the routine returns a pointer to the matrix -* created. Otherwise, the routine prints an appropriate error message -* and returns NULL. */ - -SPM *spm_read_hbm(const char *fname) -{ SPM *A = NULL; - HBM *hbm; - int nrow, ncol, nnzero, i, j, beg, end, ptr, *colptr, *rowind; - double val, *values; - char *mxtype; - hbm = hbm_read_mat(fname); - if (hbm == NULL) - { xprintf("spm_read_hbm: unable to read matrix\n"); - goto fini; - } - mxtype = hbm->mxtype; - nrow = hbm->nrow; - ncol = hbm->ncol; - nnzero = hbm->nnzero; - colptr = hbm->colptr; - rowind = hbm->rowind; - values = hbm->values; - if (!(strcmp(mxtype, "RSA") == 0 || strcmp(mxtype, "PSA") == 0 || - strcmp(mxtype, "RUA") == 0 || strcmp(mxtype, "PUA") == 0 || - strcmp(mxtype, "RRA") == 0 || strcmp(mxtype, "PRA") == 0)) - { xprintf("spm_read_hbm: matrix type '%s' not supported\n", - mxtype); - goto fini; - } - A = spm_create_mat(nrow, ncol); - if (mxtype[1] == 'S' || mxtype[1] == 'U') - xassert(nrow == ncol); - for (j = 1; j <= ncol; j++) - { beg = colptr[j]; - end = colptr[j+1]; - xassert(1 <= beg && beg <= end && end <= nnzero + 1); - for (ptr = beg; ptr < end; ptr++) - { i = rowind[ptr]; - xassert(1 <= i && i <= nrow); - if (mxtype[0] == 'R') - val = values[ptr]; - else - val = 1.0; - spm_new_elem(A, i, j, val); - if (mxtype[1] == 'S' && i != j) - spm_new_elem(A, j, i, val); - } - } -fini: if (hbm != NULL) hbm_free_mat(hbm); - return A; -} - -/*********************************************************************** -* NAME -* -* spm_count_nnz - determine number of non-zeros in sparse matrix -* -* SYNOPSIS -* -* #include "glpspm.h" -* int spm_count_nnz(const SPM *A); -* -* RETURNS -* -* The routine spm_count_nnz returns the number of structural non-zero -* elements in the specified sparse matrix. */ - -int spm_count_nnz(const SPM *A) -{ SPME *e; - int i, nnz = 0; - for (i = 1; i <= A->m; i++) - for (e = A->row[i]; e != NULL; e = e->r_next) nnz++; - return nnz; -} - -/*********************************************************************** -* NAME -* -* spm_drop_zeros - remove zero elements from sparse matrix -* -* SYNOPSIS -* -* #include "glpspm.h" -* int spm_drop_zeros(SPM *A, double eps); -* -* DESCRIPTION -* -* The routine spm_drop_zeros removes all elements from the specified -* sparse matrix, whose absolute value is less than eps. -* -* If the parameter eps is 0, only zero elements are removed from the -* matrix. -* -* RETURNS -* -* The routine returns the number of elements removed. */ - -int spm_drop_zeros(SPM *A, double eps) -{ SPME *e, *next; - int i, count = 0; - for (i = 1; i <= A->m; i++) - { for (e = A->row[i]; e != NULL; e = next) - { next = e->r_next; - if (e->val == 0.0 || fabs(e->val) < eps) - { /* remove element from the row list */ - if (e->r_prev == NULL) - A->row[e->i] = e->r_next; - else - e->r_prev->r_next = e->r_next; - if (e->r_next == NULL) - ; - else - e->r_next->r_prev = e->r_prev; - /* remove element from the column list */ - if (e->c_prev == NULL) - A->col[e->j] = e->c_next; - else - e->c_prev->c_next = e->c_next; - if (e->c_next == NULL) - ; - else - e->c_next->c_prev = e->c_prev; - /* return element to the memory pool */ - dmp_free_atom(A->pool, e, sizeof(SPME)); - count++; - } - } - } - return count; -} - -/*********************************************************************** -* NAME -* -* spm_read_mat - read sparse matrix from text file -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPM *spm_read_mat(const char *fname); -* -* DESCRIPTION -* -* The routine reads a sparse matrix from a text file whose name is -* specified by the parameter fname. -* -* For the file format see description of the routine spm_write_mat. -* -* RETURNS -* -* On success the routine returns a pointer to the matrix created, -* otherwise NULL. */ - -#if 1 -SPM *spm_read_mat(const char *fname) -{ xassert(fname != fname); - return NULL; -} -#else -SPM *spm_read_mat(const char *fname) -{ SPM *A = NULL; - PDS *pds; - jmp_buf jump; - int i, j, k, m, n, nnz, fail = 0; - double val; - xprintf("spm_read_mat: reading matrix from '%s'...\n", fname); - pds = pds_open_file(fname); - if (pds == NULL) - { xprintf("spm_read_mat: unable to open '%s' - %s\n", fname, - strerror(errno)); - fail = 1; - goto done; - } - if (setjmp(jump)) - { fail = 1; - goto done; - } - pds_set_jump(pds, jump); - /* number of rows, number of columns, number of non-zeros */ - m = pds_scan_int(pds); - if (m < 0) - pds_error(pds, "invalid number of rows\n"); - n = pds_scan_int(pds); - if (n < 0) - pds_error(pds, "invalid number of columns\n"); - nnz = pds_scan_int(pds); - if (nnz < 0) - pds_error(pds, "invalid number of non-zeros\n"); - /* create matrix */ - xprintf("spm_read_mat: %d rows, %d columns, %d non-zeros\n", - m, n, nnz); - A = spm_create_mat(m, n); - /* read matrix elements */ - for (k = 1; k <= nnz; k++) - { /* row index, column index, element value */ - i = pds_scan_int(pds); - if (!(1 <= i && i <= m)) - pds_error(pds, "row index out of range\n"); - j = pds_scan_int(pds); - if (!(1 <= j && j <= n)) - pds_error(pds, "column index out of range\n"); - val = pds_scan_num(pds); - /* add new element to the matrix */ - spm_new_elem(A, i, j, val); - } - xprintf("spm_read_mat: %d lines were read\n", pds->count); -done: if (pds != NULL) pds_close_file(pds); - if (fail && A != NULL) spm_delete_mat(A), A = NULL; - return A; -} -#endif - -/*********************************************************************** -* NAME -* -* spm_write_mat - write sparse matrix to text file -* -* SYNOPSIS -* -* #include "glpspm.h" -* int spm_write_mat(const SPM *A, const char *fname); -* -* DESCRIPTION -* -* The routine spm_write_mat writes the specified sparse matrix to a -* text file whose name is specified by the parameter fname. This file -* can be read back with the routine spm_read_mat. -* -* RETURNS -* -* On success the routine returns zero, otherwise non-zero. -* -* FILE FORMAT -* -* The file created by the routine spm_write_mat is a plain text file, -* which contains the following information: -* -* m n nnz -* row[1] col[1] val[1] -* row[2] col[2] val[2] -* . . . -* row[nnz] col[nnz] val[nnz] -* -* where: -* m is the number of rows; -* n is the number of columns; -* nnz is the number of non-zeros; -* row[k], k = 1,...,nnz, are row indices; -* col[k], k = 1,...,nnz, are column indices; -* val[k], k = 1,...,nnz, are element values. */ - -#if 1 -int spm_write_mat(const SPM *A, const char *fname) -{ xassert(A != A); - xassert(fname != fname); - return 0; -} -#else -int spm_write_mat(const SPM *A, const char *fname) -{ FILE *fp; - int i, nnz, ret = 0; - xprintf("spm_write_mat: writing matrix to '%s'...\n", fname); - fp = fopen(fname, "w"); - if (fp == NULL) - { xprintf("spm_write_mat: unable to create '%s' - %s\n", fname, - strerror(errno)); - ret = 1; - goto done; - } - /* number of rows, number of columns, number of non-zeros */ - nnz = spm_count_nnz(A); - fprintf(fp, "%d %d %d\n", A->m, A->n, nnz); - /* walk through rows of the matrix */ - for (i = 1; i <= A->m; i++) - { SPME *e; - /* walk through elements of i-th row */ - for (e = A->row[i]; e != NULL; e = e->r_next) - { /* row index, column index, element value */ - fprintf(fp, "%d %d %.*g\n", e->i, e->j, DBL_DIG, e->val); - } - } - fflush(fp); - if (ferror(fp)) - { xprintf("spm_write_mat: writing error on '%s' - %s\n", fname, - strerror(errno)); - ret = 1; - goto done; - } - xprintf("spm_write_mat: %d lines were written\n", 1 + nnz); -done: if (fp != NULL) fclose(fp); - return ret; -} -#endif - -/*********************************************************************** -* NAME -* -* spm_transpose - transpose sparse matrix -* -* SYNOPSIS -* -* #include "glpspm.h" -* SPM *spm_transpose(const SPM *A); -* -* RETURNS -* -* The routine computes and returns sparse matrix B, which is a matrix -* transposed to sparse matrix A. */ - -SPM *spm_transpose(const SPM *A) -{ SPM *B; - int i; - B = spm_create_mat(A->n, A->m); - for (i = 1; i <= A->m; i++) - { SPME *e; - for (e = A->row[i]; e != NULL; e = e->r_next) - spm_new_elem(B, e->j, i, e->val); - } - return B; -} - -SPM *spm_add_sym(const SPM *A, const SPM *B) -{ /* add two sparse matrices (symbolic phase) */ - SPM *C; - int i, j, *flag; - xassert(A->m == B->m); - xassert(A->n == B->n); - /* create resultant matrix */ - C = spm_create_mat(A->m, A->n); - /* allocate and clear the flag array */ - flag = xcalloc(1+C->n, sizeof(int)); - for (j = 1; j <= C->n; j++) - flag[j] = 0; - /* compute pattern of C = A + B */ - for (i = 1; i <= C->m; i++) - { SPME *e; - /* at the beginning i-th row of C is empty */ - /* (i-th row of C) := (i-th row of C) union (i-th row of A) */ - for (e = A->row[i]; e != NULL; e = e->r_next) - { /* (note that i-th row of A may have duplicate elements) */ - j = e->j; - if (!flag[j]) - { spm_new_elem(C, i, j, 0.0); - flag[j] = 1; - } - } - /* (i-th row of C) := (i-th row of C) union (i-th row of B) */ - for (e = B->row[i]; e != NULL; e = e->r_next) - { /* (note that i-th row of B may have duplicate elements) */ - j = e->j; - if (!flag[j]) - { spm_new_elem(C, i, j, 0.0); - flag[j] = 1; - } - } - /* reset the flag array */ - for (e = C->row[i]; e != NULL; e = e->r_next) - flag[e->j] = 0; - } - /* check and deallocate the flag array */ - for (j = 1; j <= C->n; j++) - xassert(!flag[j]); - xfree(flag); - return C; -} - -void spm_add_num(SPM *C, double alfa, const SPM *A, double beta, - const SPM *B) -{ /* add two sparse matrices (numeric phase) */ - int i, j; - double *work; - /* allocate and clear the working array */ - work = xcalloc(1+C->n, sizeof(double)); - for (j = 1; j <= C->n; j++) - work[j] = 0.0; - /* compute matrix C = alfa * A + beta * B */ - for (i = 1; i <= C->n; i++) - { SPME *e; - /* work := alfa * (i-th row of A) + beta * (i-th row of B) */ - /* (note that A and/or B may have duplicate elements) */ - for (e = A->row[i]; e != NULL; e = e->r_next) - work[e->j] += alfa * e->val; - for (e = B->row[i]; e != NULL; e = e->r_next) - work[e->j] += beta * e->val; - /* (i-th row of C) := work, work := 0 */ - for (e = C->row[i]; e != NULL; e = e->r_next) - { j = e->j; - e->val = work[j]; - work[j] = 0.0; - } - } - /* check and deallocate the working array */ - for (j = 1; j <= C->n; j++) - xassert(work[j] == 0.0); - xfree(work); - return; -} - -SPM *spm_add_mat(double alfa, const SPM *A, double beta, const SPM *B) -{ /* add two sparse matrices (driver routine) */ - SPM *C; - C = spm_add_sym(A, B); - spm_add_num(C, alfa, A, beta, B); - return C; -} - -SPM *spm_mul_sym(const SPM *A, const SPM *B) -{ /* multiply two sparse matrices (symbolic phase) */ - int i, j, k, *flag; - SPM *C; - xassert(A->n == B->m); - /* create resultant matrix */ - C = spm_create_mat(A->m, B->n); - /* allocate and clear the flag array */ - flag = xcalloc(1+C->n, sizeof(int)); - for (j = 1; j <= C->n; j++) - flag[j] = 0; - /* compute pattern of C = A * B */ - for (i = 1; i <= C->m; i++) - { SPME *e, *ee; - /* compute pattern of i-th row of C */ - for (e = A->row[i]; e != NULL; e = e->r_next) - { k = e->j; - for (ee = B->row[k]; ee != NULL; ee = ee->r_next) - { j = ee->j; - /* if a[i,k] != 0 and b[k,j] != 0 then c[i,j] != 0 */ - if (!flag[j]) - { /* c[i,j] does not exist, so create it */ - spm_new_elem(C, i, j, 0.0); - flag[j] = 1; - } - } - } - /* reset the flag array */ - for (e = C->row[i]; e != NULL; e = e->r_next) - flag[e->j] = 0; - } - /* check and deallocate the flag array */ - for (j = 1; j <= C->n; j++) - xassert(!flag[j]); - xfree(flag); - return C; -} - -void spm_mul_num(SPM *C, const SPM *A, const SPM *B) -{ /* multiply two sparse matrices (numeric phase) */ - int i, j; - double *work; - /* allocate and clear the working array */ - work = xcalloc(1+A->n, sizeof(double)); - for (j = 1; j <= A->n; j++) - work[j] = 0.0; - /* compute matrix C = A * B */ - for (i = 1; i <= C->m; i++) - { SPME *e, *ee; - double temp; - /* work := (i-th row of A) */ - /* (note that A may have duplicate elements) */ - for (e = A->row[i]; e != NULL; e = e->r_next) - work[e->j] += e->val; - /* compute i-th row of C */ - for (e = C->row[i]; e != NULL; e = e->r_next) - { j = e->j; - /* c[i,j] := work * (j-th column of B) */ - temp = 0.0; - for (ee = B->col[j]; ee != NULL; ee = ee->c_next) - temp += work[ee->i] * ee->val; - e->val = temp; - } - /* reset the working array */ - for (e = A->row[i]; e != NULL; e = e->r_next) - work[e->j] = 0.0; - } - /* check and deallocate the working array */ - for (j = 1; j <= A->n; j++) - xassert(work[j] == 0.0); - xfree(work); - return; -} - -SPM *spm_mul_mat(const SPM *A, const SPM *B) -{ /* multiply two sparse matrices (driver routine) */ - SPM *C; - C = spm_mul_sym(A, B); - spm_mul_num(C, A, B); - return C; -} - -PER *spm_create_per(int n) -{ /* create permutation matrix */ - PER *P; - int k; - xassert(n >= 0); - P = xmalloc(sizeof(PER)); - P->n = n; - P->row = xcalloc(1+n, sizeof(int)); - P->col = xcalloc(1+n, sizeof(int)); - /* initially it is identity matrix */ - for (k = 1; k <= n; k++) - P->row[k] = P->col[k] = k; - return P; -} - -void spm_check_per(PER *P) -{ /* check permutation matrix for correctness */ - int i, j; - xassert(P->n >= 0); - for (i = 1; i <= P->n; i++) - { j = P->row[i]; - xassert(1 <= j && j <= P->n); - xassert(P->col[j] == i); - } - return; -} - -void spm_delete_per(PER *P) -{ /* delete permutation matrix */ - xfree(P->row); - xfree(P->col); - xfree(P); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpspm.h b/code/3rd_glpk/draft/glpspm.h deleted file mode 100644 index eda9f98f..00000000 --- a/code/3rd_glpk/draft/glpspm.h +++ /dev/null @@ -1,165 +0,0 @@ -/* glpspm.h (general sparse matrix) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPSPM_H -#define GLPSPM_H - -#include "dmp.h" - -typedef struct SPM SPM; -typedef struct SPME SPME; - -struct SPM -{ /* general sparse matrix */ - int m; - /* number of rows, m >= 0 */ - int n; - /* number of columns, n >= 0 */ - DMP *pool; - /* memory pool to store matrix elements */ - SPME **row; /* SPME *row[1+m]; */ - /* row[i], 1 <= i <= m, is a pointer to i-th row list */ - SPME **col; /* SPME *col[1+n]; */ - /* col[j], 1 <= j <= n, is a pointer to j-th column list */ -}; - -struct SPME -{ /* sparse matrix element */ - int i; - /* row number */ - int j; - /* column number */ - double val; - /* element value */ - SPME *r_prev; - /* pointer to previous element in the same row */ - SPME *r_next; - /* pointer to next element in the same row */ - SPME *c_prev; - /* pointer to previous element in the same column */ - SPME *c_next; - /* pointer to next element in the same column */ -}; - -typedef struct PER PER; - -struct PER -{ /* permutation matrix */ - int n; - /* matrix order, n >= 0 */ - int *row; /* int row[1+n]; */ - /* row[i] = j means p[i,j] = 1 */ - int *col; /* int col[1+n]; */ - /* col[j] = i means p[i,j] = 1 */ -}; - -#define spm_create_mat _glp_spm_create_mat -SPM *spm_create_mat(int m, int n); -/* create general sparse matrix */ - -#define spm_new_elem _glp_spm_new_elem -SPME *spm_new_elem(SPM *A, int i, int j, double val); -/* add new element to sparse matrix */ - -#define spm_delete_mat _glp_spm_delete_mat -void spm_delete_mat(SPM *A); -/* delete general sparse matrix */ - -#define spm_test_mat_e _glp_spm_test_mat_e -SPM *spm_test_mat_e(int n, int c); -/* create test sparse matrix of E(n,c) class */ - -#define spm_test_mat_d _glp_spm_test_mat_d -SPM *spm_test_mat_d(int n, int c); -/* create test sparse matrix of D(n,c) class */ - -#define spm_show_mat _glp_spm_show_mat -int spm_show_mat(const SPM *A, const char *fname); -/* write sparse matrix pattern in BMP file format */ - -#define spm_read_hbm _glp_spm_read_hbm -SPM *spm_read_hbm(const char *fname); -/* read sparse matrix in Harwell-Boeing format */ - -#define spm_count_nnz _glp_spm_count_nnz -int spm_count_nnz(const SPM *A); -/* determine number of non-zeros in sparse matrix */ - -#define spm_drop_zeros _glp_spm_drop_zeros -int spm_drop_zeros(SPM *A, double eps); -/* remove zero elements from sparse matrix */ - -#define spm_read_mat _glp_spm_read_mat -SPM *spm_read_mat(const char *fname); -/* read sparse matrix from text file */ - -#define spm_write_mat _glp_spm_write_mat -int spm_write_mat(const SPM *A, const char *fname); -/* write sparse matrix to text file */ - -#define spm_transpose _glp_spm_transpose -SPM *spm_transpose(const SPM *A); -/* transpose sparse matrix */ - -#define spm_add_sym _glp_spm_add_sym -SPM *spm_add_sym(const SPM *A, const SPM *B); -/* add two sparse matrices (symbolic phase) */ - -#define spm_add_num _glp_spm_add_num -void spm_add_num(SPM *C, double alfa, const SPM *A, double beta, - const SPM *B); -/* add two sparse matrices (numeric phase) */ - -#define spm_add_mat _glp_spm_add_mat -SPM *spm_add_mat(double alfa, const SPM *A, double beta, - const SPM *B); -/* add two sparse matrices (driver routine) */ - -#define spm_mul_sym _glp_spm_mul_sym -SPM *spm_mul_sym(const SPM *A, const SPM *B); -/* multiply two sparse matrices (symbolic phase) */ - -#define spm_mul_num _glp_spm_mul_num -void spm_mul_num(SPM *C, const SPM *A, const SPM *B); -/* multiply two sparse matrices (numeric phase) */ - -#define spm_mul_mat _glp_spm_mul_mat -SPM *spm_mul_mat(const SPM *A, const SPM *B); -/* multiply two sparse matrices (driver routine) */ - -#define spm_create_per _glp_spm_create_per -PER *spm_create_per(int n); -/* create permutation matrix */ - -#define spm_check_per _glp_spm_check_per -void spm_check_per(PER *P); -/* check permutation matrix for correctness */ - -#define spm_delete_per _glp_spm_delete_per -void spm_delete_per(PER *P); -/* delete permutation matrix */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpssx.h b/code/3rd_glpk/draft/glpssx.h deleted file mode 100644 index 3b52b3cc..00000000 --- a/code/3rd_glpk/draft/glpssx.h +++ /dev/null @@ -1,437 +0,0 @@ -/* glpssx.h (simplex method, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPSSX_H -#define GLPSSX_H - -#include "bfx.h" -#include "env.h" -#if 1 /* 25/XI-2017 */ -#include "glpk.h" -#endif - -typedef struct SSX SSX; - -struct SSX -{ /* simplex solver workspace */ -/*---------------------------------------------------------------------- -// LP PROBLEM DATA -// -// It is assumed that LP problem has the following statement: -// -// minimize (or maximize) -// -// z = c[1]*x[1] + ... + c[m+n]*x[m+n] + c[0] (1) -// -// subject to equality constraints -// -// x[1] - a[1,1]*x[m+1] - ... - a[1,n]*x[m+n] = 0 -// -// . . . . . . . (2) -// -// x[m] - a[m,1]*x[m+1] + ... - a[m,n]*x[m+n] = 0 -// -// and bounds of variables -// -// l[1] <= x[1] <= u[1] -// -// . . . . . . . (3) -// -// l[m+n] <= x[m+n] <= u[m+n] -// -// where: -// x[1], ..., x[m] - auxiliary variables; -// x[m+1], ..., x[m+n] - structural variables; -// z - objective function; -// c[1], ..., c[m+n] - coefficients of the objective function; -// c[0] - constant term of the objective function; -// a[1,1], ..., a[m,n] - constraint coefficients; -// l[1], ..., l[m+n] - lower bounds of variables; -// u[1], ..., u[m+n] - upper bounds of variables. -// -// Bounds of variables can be finite as well as inifinite. Besides, -// lower and upper bounds can be equal to each other. So the following -// five types of variables are possible: -// -// Bounds of variable Type of variable -// ------------------------------------------------- -// -inf < x[k] < +inf Free (unbounded) variable -// l[k] <= x[k] < +inf Variable with lower bound -// -inf < x[k] <= u[k] Variable with upper bound -// l[k] <= x[k] <= u[k] Double-bounded variable -// l[k] = x[k] = u[k] Fixed variable -// -// Using vector-matrix notations the LP problem (1)-(3) can be written -// as follows: -// -// minimize (or maximize) -// -// z = c * x + c[0] (4) -// -// subject to equality constraints -// -// xR - A * xS = 0 (5) -// -// and bounds of variables -// -// l <= x <= u (6) -// -// where: -// xR - vector of auxiliary variables; -// xS - vector of structural variables; -// x = (xR, xS) - vector of all variables; -// z - objective function; -// c - vector of objective coefficients; -// c[0] - constant term of the objective function; -// A - matrix of constraint coefficients (has m rows -// and n columns); -// l - vector of lower bounds of variables; -// u - vector of upper bounds of variables. -// -// The simplex method makes no difference between auxiliary and -// structural variables, so it is convenient to think the system of -// equality constraints (5) written in a homogeneous form: -// -// (I | -A) * x = 0, (7) -// -// where (I | -A) is an augmented (m+n)xm constraint matrix, I is mxm -// unity matrix whose columns correspond to auxiliary variables, and A -// is the original mxn constraint matrix whose columns correspond to -// structural variables. Note that only the matrix A is stored. -----------------------------------------------------------------------*/ - int m; - /* number of rows (auxiliary variables), m > 0 */ - int n; - /* number of columns (structural variables), n > 0 */ - int *type; /* int type[1+m+n]; */ - /* type[0] is not used; - type[k], 1 <= k <= m+n, is the type of variable x[k]: */ -#define SSX_FR 0 /* free (unbounded) variable */ -#define SSX_LO 1 /* variable with lower bound */ -#define SSX_UP 2 /* variable with upper bound */ -#define SSX_DB 3 /* double-bounded variable */ -#define SSX_FX 4 /* fixed variable */ - mpq_t *lb; /* mpq_t lb[1+m+n]; alias: l */ - /* lb[0] is not used; - lb[k], 1 <= k <= m+n, is an lower bound of variable x[k]; - if x[k] has no lower bound, lb[k] is zero */ - mpq_t *ub; /* mpq_t ub[1+m+n]; alias: u */ - /* ub[0] is not used; - ub[k], 1 <= k <= m+n, is an upper bound of variable x[k]; - if x[k] has no upper bound, ub[k] is zero; - if x[k] is of fixed type, ub[k] is equal to lb[k] */ - int dir; - /* optimization direction (sense of the objective function): */ -#define SSX_MIN 0 /* minimization */ -#define SSX_MAX 1 /* maximization */ - mpq_t *coef; /* mpq_t coef[1+m+n]; alias: c */ - /* coef[0] is a constant term of the objective function; - coef[k], 1 <= k <= m+n, is a coefficient of the objective - function at variable x[k]; - note that auxiliary variables also may have non-zero objective - coefficients */ - int *A_ptr; /* int A_ptr[1+n+1]; */ - int *A_ind; /* int A_ind[A_ptr[n+1]]; */ - mpq_t *A_val; /* mpq_t A_val[A_ptr[n+1]]; */ - /* constraint matrix A (see (5)) in storage-by-columns format */ -/*---------------------------------------------------------------------- -// LP BASIS AND CURRENT BASIC SOLUTION -// -// The LP basis is defined by the following partition of the augmented -// constraint matrix (7): -// -// (B | N) = (I | -A) * Q, (8) -// -// where B is a mxm non-singular basis matrix whose columns correspond -// to basic variables xB, N is a mxn matrix whose columns correspond to -// non-basic variables xN, and Q is a permutation (m+n)x(m+n) matrix. -// -// From (7) and (8) it follows that -// -// (I | -A) * x = (I | -A) * Q * Q' * x = (B | N) * (xB, xN), -// -// therefore -// -// (xB, xN) = Q' * x, (9) -// -// where x is the vector of all variables in the original order, xB is -// a vector of basic variables, xN is a vector of non-basic variables, -// Q' = inv(Q) is a matrix transposed to Q. -// -// Current values of non-basic variables xN[j], j = 1, ..., n, are not -// stored; they are defined implicitly by their statuses as follows: -// -// 0, if xN[j] is free variable -// lN[j], if xN[j] is on its lower bound (10) -// uN[j], if xN[j] is on its upper bound -// lN[j] = uN[j], if xN[j] is fixed variable -// -// where lN[j] and uN[j] are lower and upper bounds of xN[j]. -// -// Current values of basic variables xB[i], i = 1, ..., m, are computed -// as follows: -// -// beta = - inv(B) * N * xN, (11) -// -// where current values of xN are defined by (10). -// -// Current values of simplex multipliers pi[i], i = 1, ..., m (which -// are values of Lagrange multipliers for equality constraints (7) also -// called shadow prices) are computed as follows: -// -// pi = inv(B') * cB, (12) -// -// where B' is a matrix transposed to B, cB is a vector of objective -// coefficients at basic variables xB. -// -// Current values of reduced costs d[j], j = 1, ..., n, (which are -// values of Langrange multipliers for active inequality constraints -// corresponding to non-basic variables) are computed as follows: -// -// d = cN - N' * pi, (13) -// -// where N' is a matrix transposed to N, cN is a vector of objective -// coefficients at non-basic variables xN. -----------------------------------------------------------------------*/ - int *stat; /* int stat[1+m+n]; */ - /* stat[0] is not used; - stat[k], 1 <= k <= m+n, is the status of variable x[k]: */ -#define SSX_BS 0 /* basic variable */ -#define SSX_NL 1 /* non-basic variable on lower bound */ -#define SSX_NU 2 /* non-basic variable on upper bound */ -#define SSX_NF 3 /* non-basic free variable */ -#define SSX_NS 4 /* non-basic fixed variable */ - int *Q_row; /* int Q_row[1+m+n]; */ - /* matrix Q in row-like format; - Q_row[0] is not used; - Q_row[i] = j means that q[i,j] = 1 */ - int *Q_col; /* int Q_col[1+m+n]; */ - /* matrix Q in column-like format; - Q_col[0] is not used; - Q_col[j] = i means that q[i,j] = 1 */ - /* if k-th column of the matrix (I | A) is k'-th column of the - matrix (B | N), then Q_row[k] = k' and Q_col[k'] = k; - if x[k] is xB[i], then Q_row[k] = i and Q_col[i] = k; - if x[k] is xN[j], then Q_row[k] = m+j and Q_col[m+j] = k */ - BFX *binv; - /* invertable form of the basis matrix B */ - mpq_t *bbar; /* mpq_t bbar[1+m]; alias: beta */ - /* bbar[0] is a value of the objective function; - bbar[i], 1 <= i <= m, is a value of basic variable xB[i] */ - mpq_t *pi; /* mpq_t pi[1+m]; */ - /* pi[0] is not used; - pi[i], 1 <= i <= m, is a simplex multiplier corresponding to - i-th row (equality constraint) */ - mpq_t *cbar; /* mpq_t cbar[1+n]; alias: d */ - /* cbar[0] is not used; - cbar[j], 1 <= j <= n, is a reduced cost of non-basic variable - xN[j] */ -/*---------------------------------------------------------------------- -// SIMPLEX TABLE -// -// Due to (8) and (9) the system of equality constraints (7) for the -// current basis can be written as follows: -// -// xB = A~ * xN, (14) -// -// where -// -// A~ = - inv(B) * N (15) -// -// is a mxn matrix called the simplex table. -// -// The revised simplex method uses only two components of A~, namely, -// pivot column corresponding to non-basic variable xN[q] chosen to -// enter the basis, and pivot row corresponding to basic variable xB[p] -// chosen to leave the basis. -// -// Pivot column alfa_q is q-th column of A~, so -// -// alfa_q = A~ * e[q] = - inv(B) * N * e[q] = - inv(B) * N[q], (16) -// -// where N[q] is q-th column of the matrix N. -// -// Pivot row alfa_p is p-th row of A~ or, equivalently, p-th column of -// A~', a matrix transposed to A~, so -// -// alfa_p = A~' * e[p] = - N' * inv(B') * e[p] = - N' * rho_p, (17) -// -// where (*)' means transposition, and -// -// rho_p = inv(B') * e[p], (18) -// -// is p-th column of inv(B') or, that is the same, p-th row of inv(B). -----------------------------------------------------------------------*/ - int p; - /* number of basic variable xB[p], 1 <= p <= m, chosen to leave - the basis */ - mpq_t *rho; /* mpq_t rho[1+m]; */ - /* p-th row of the inverse inv(B); see (18) */ - mpq_t *ap; /* mpq_t ap[1+n]; */ - /* p-th row of the simplex table; see (17) */ - int q; - /* number of non-basic variable xN[q], 1 <= q <= n, chosen to - enter the basis */ - mpq_t *aq; /* mpq_t aq[1+m]; */ - /* q-th column of the simplex table; see (16) */ -/*--------------------------------------------------------------------*/ - int q_dir; - /* direction in which non-basic variable xN[q] should change on - moving to the adjacent vertex of the polyhedron: - +1 means that xN[q] increases - -1 means that xN[q] decreases */ - int p_stat; - /* non-basic status which should be assigned to basic variable - xB[p] when it has left the basis and become xN[q] */ - mpq_t delta; - /* actual change of xN[q] in the adjacent basis (it has the same - sign as q_dir) */ -/*--------------------------------------------------------------------*/ -#if 1 /* 25/XI-2017 */ - int msg_lev; - /* verbosity level: - GLP_MSG_OFF no output - GLP_MSG_ERR report errors and warnings - GLP_MSG_ON normal output - GLP_MSG_ALL highest verbosity */ -#endif - int it_lim; - /* simplex iterations limit; if this value is positive, it is - decreased by one each time when one simplex iteration has been - performed, and reaching zero value signals the solver to stop - the search; negative value means no iterations limit */ - int it_cnt; - /* simplex iterations count; this count is increased by one each - time when one simplex iteration has been performed */ - double tm_lim; - /* searching time limit, in seconds; if this value is positive, - it is decreased each time when one simplex iteration has been - performed by the amount of time spent for the iteration, and - reaching zero value signals the solver to stop the search; - negative value means no time limit */ - double out_frq; - /* output frequency, in seconds; this parameter specifies how - frequently the solver sends information about the progress of - the search to the standard output */ -#if 0 /* 10/VI-2013 */ - glp_long tm_beg; -#else - double tm_beg; -#endif - /* starting time of the search, in seconds; the total time of the - search is the difference between xtime() and tm_beg */ -#if 0 /* 10/VI-2013 */ - glp_long tm_lag; -#else - double tm_lag; -#endif - /* the most recent time, in seconds, at which the progress of the - the search was displayed */ -}; - -#define ssx_create _glp_ssx_create -#define ssx_factorize _glp_ssx_factorize -#define ssx_get_xNj _glp_ssx_get_xNj -#define ssx_eval_bbar _glp_ssx_eval_bbar -#define ssx_eval_pi _glp_ssx_eval_pi -#define ssx_eval_dj _glp_ssx_eval_dj -#define ssx_eval_cbar _glp_ssx_eval_cbar -#define ssx_eval_rho _glp_ssx_eval_rho -#define ssx_eval_row _glp_ssx_eval_row -#define ssx_eval_col _glp_ssx_eval_col -#define ssx_chuzc _glp_ssx_chuzc -#define ssx_chuzr _glp_ssx_chuzr -#define ssx_update_bbar _glp_ssx_update_bbar -#define ssx_update_pi _glp_ssx_update_pi -#define ssx_update_cbar _glp_ssx_update_cbar -#define ssx_change_basis _glp_ssx_change_basis -#define ssx_delete _glp_ssx_delete - -#define ssx_phase_I _glp_ssx_phase_I -#define ssx_phase_II _glp_ssx_phase_II -#define ssx_driver _glp_ssx_driver - -SSX *ssx_create(int m, int n, int nnz); -/* create simplex solver workspace */ - -int ssx_factorize(SSX *ssx); -/* factorize the current basis matrix */ - -void ssx_get_xNj(SSX *ssx, int j, mpq_t x); -/* determine value of non-basic variable */ - -void ssx_eval_bbar(SSX *ssx); -/* compute values of basic variables */ - -void ssx_eval_pi(SSX *ssx); -/* compute values of simplex multipliers */ - -void ssx_eval_dj(SSX *ssx, int j, mpq_t dj); -/* compute reduced cost of non-basic variable */ - -void ssx_eval_cbar(SSX *ssx); -/* compute reduced costs of all non-basic variables */ - -void ssx_eval_rho(SSX *ssx); -/* compute p-th row of the inverse */ - -void ssx_eval_row(SSX *ssx); -/* compute pivot row of the simplex table */ - -void ssx_eval_col(SSX *ssx); -/* compute pivot column of the simplex table */ - -void ssx_chuzc(SSX *ssx); -/* choose pivot column */ - -void ssx_chuzr(SSX *ssx); -/* choose pivot row */ - -void ssx_update_bbar(SSX *ssx); -/* update values of basic variables */ - -void ssx_update_pi(SSX *ssx); -/* update simplex multipliers */ - -void ssx_update_cbar(SSX *ssx); -/* update reduced costs of non-basic variables */ - -void ssx_change_basis(SSX *ssx); -/* change current basis to adjacent one */ - -void ssx_delete(SSX *ssx); -/* delete simplex solver workspace */ - -int ssx_phase_I(SSX *ssx); -/* find primal feasible solution */ - -int ssx_phase_II(SSX *ssx); -/* find optimal solution */ - -int ssx_driver(SSX *ssx); -/* base driver to exact simplex method */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/glpssx01.c b/code/3rd_glpk/draft/glpssx01.c deleted file mode 100644 index 9b70444e..00000000 --- a/code/3rd_glpk/draft/glpssx01.c +++ /dev/null @@ -1,839 +0,0 @@ -/* glpssx01.c (simplex method, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpssx.h" -#define xfault xerror - -/*---------------------------------------------------------------------- -// ssx_create - create simplex solver workspace. -// -// This routine creates the workspace used by simplex solver routines, -// and returns a pointer to it. -// -// Parameters m, n, and nnz specify, respectively, the number of rows, -// columns, and non-zero constraint coefficients. -// -// This routine only allocates the memory for the workspace components, -// so the workspace needs to be saturated by data. */ - -SSX *ssx_create(int m, int n, int nnz) -{ SSX *ssx; - int i, j, k; - if (m < 1) - xfault("ssx_create: m = %d; invalid number of rows\n", m); - if (n < 1) - xfault("ssx_create: n = %d; invalid number of columns\n", n); - if (nnz < 0) - xfault("ssx_create: nnz = %d; invalid number of non-zero const" - "raint coefficients\n", nnz); - ssx = xmalloc(sizeof(SSX)); - ssx->m = m; - ssx->n = n; - ssx->type = xcalloc(1+m+n, sizeof(int)); - ssx->lb = xcalloc(1+m+n, sizeof(mpq_t)); - for (k = 1; k <= m+n; k++) mpq_init(ssx->lb[k]); - ssx->ub = xcalloc(1+m+n, sizeof(mpq_t)); - for (k = 1; k <= m+n; k++) mpq_init(ssx->ub[k]); - ssx->coef = xcalloc(1+m+n, sizeof(mpq_t)); - for (k = 0; k <= m+n; k++) mpq_init(ssx->coef[k]); - ssx->A_ptr = xcalloc(1+n+1, sizeof(int)); - ssx->A_ptr[n+1] = nnz+1; - ssx->A_ind = xcalloc(1+nnz, sizeof(int)); - ssx->A_val = xcalloc(1+nnz, sizeof(mpq_t)); - for (k = 1; k <= nnz; k++) mpq_init(ssx->A_val[k]); - ssx->stat = xcalloc(1+m+n, sizeof(int)); - ssx->Q_row = xcalloc(1+m+n, sizeof(int)); - ssx->Q_col = xcalloc(1+m+n, sizeof(int)); - ssx->binv = bfx_create_binv(); - ssx->bbar = xcalloc(1+m, sizeof(mpq_t)); - for (i = 0; i <= m; i++) mpq_init(ssx->bbar[i]); - ssx->pi = xcalloc(1+m, sizeof(mpq_t)); - for (i = 1; i <= m; i++) mpq_init(ssx->pi[i]); - ssx->cbar = xcalloc(1+n, sizeof(mpq_t)); - for (j = 1; j <= n; j++) mpq_init(ssx->cbar[j]); - ssx->rho = xcalloc(1+m, sizeof(mpq_t)); - for (i = 1; i <= m; i++) mpq_init(ssx->rho[i]); - ssx->ap = xcalloc(1+n, sizeof(mpq_t)); - for (j = 1; j <= n; j++) mpq_init(ssx->ap[j]); - ssx->aq = xcalloc(1+m, sizeof(mpq_t)); - for (i = 1; i <= m; i++) mpq_init(ssx->aq[i]); - mpq_init(ssx->delta); - return ssx; -} - -/*---------------------------------------------------------------------- -// ssx_factorize - factorize the current basis matrix. -// -// This routine computes factorization of the current basis matrix B -// and returns the singularity flag. If the matrix B is non-singular, -// the flag is zero, otherwise non-zero. */ - -static int basis_col(void *info, int j, int ind[], mpq_t val[]) -{ /* this auxiliary routine provides row indices and numeric values - of non-zero elements in j-th column of the matrix B */ - SSX *ssx = info; - int m = ssx->m; - int n = ssx->n; - int *A_ptr = ssx->A_ptr; - int *A_ind = ssx->A_ind; - mpq_t *A_val = ssx->A_val; - int *Q_col = ssx->Q_col; - int k, len, ptr; - xassert(1 <= j && j <= m); - k = Q_col[j]; /* x[k] = xB[j] */ - xassert(1 <= k && k <= m+n); - /* j-th column of the matrix B is k-th column of the augmented - constraint matrix (I | -A) */ - if (k <= m) - { /* it is a column of the unity matrix I */ - len = 1, ind[1] = k, mpq_set_si(val[1], 1, 1); - } - else - { /* it is a column of the original constraint matrix -A */ - len = 0; - for (ptr = A_ptr[k-m]; ptr < A_ptr[k-m+1]; ptr++) - { len++; - ind[len] = A_ind[ptr]; - mpq_neg(val[len], A_val[ptr]); - } - } - return len; -} - -int ssx_factorize(SSX *ssx) -{ int ret; - ret = bfx_factorize(ssx->binv, ssx->m, basis_col, ssx); - return ret; -} - -/*---------------------------------------------------------------------- -// ssx_get_xNj - determine value of non-basic variable. -// -// This routine determines the value of non-basic variable xN[j] in the -// current basic solution defined as follows: -// -// 0, if xN[j] is free variable -// lN[j], if xN[j] is on its lower bound -// uN[j], if xN[j] is on its upper bound -// lN[j] = uN[j], if xN[j] is fixed variable -// -// where lN[j] and uN[j] are lower and upper bounds of xN[j]. */ - -void ssx_get_xNj(SSX *ssx, int j, mpq_t x) -{ int m = ssx->m; - int n = ssx->n; - mpq_t *lb = ssx->lb; - mpq_t *ub = ssx->ub; - int *stat = ssx->stat; - int *Q_col = ssx->Q_col; - int k; - xassert(1 <= j && j <= n); - k = Q_col[m+j]; /* x[k] = xN[j] */ - xassert(1 <= k && k <= m+n); - switch (stat[k]) - { case SSX_NL: - /* xN[j] is on its lower bound */ - mpq_set(x, lb[k]); break; - case SSX_NU: - /* xN[j] is on its upper bound */ - mpq_set(x, ub[k]); break; - case SSX_NF: - /* xN[j] is free variable */ - mpq_set_si(x, 0, 1); break; - case SSX_NS: - /* xN[j] is fixed variable */ - mpq_set(x, lb[k]); break; - default: - xassert(stat != stat); - } - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_bbar - compute values of basic variables. -// -// This routine computes values of basic variables xB in the current -// basic solution as follows: -// -// beta = - inv(B) * N * xN, -// -// where B is the basis matrix, N is the matrix of non-basic columns, -// xN is a vector of current values of non-basic variables. */ - -void ssx_eval_bbar(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - mpq_t *coef = ssx->coef; - int *A_ptr = ssx->A_ptr; - int *A_ind = ssx->A_ind; - mpq_t *A_val = ssx->A_val; - int *Q_col = ssx->Q_col; - mpq_t *bbar = ssx->bbar; - int i, j, k, ptr; - mpq_t x, temp; - mpq_init(x); - mpq_init(temp); - /* bbar := 0 */ - for (i = 1; i <= m; i++) - mpq_set_si(bbar[i], 0, 1); - /* bbar := - N * xN = - N[1] * xN[1] - ... - N[n] * xN[n] */ - for (j = 1; j <= n; j++) - { ssx_get_xNj(ssx, j, x); - if (mpq_sgn(x) == 0) continue; - k = Q_col[m+j]; /* x[k] = xN[j] */ - if (k <= m) - { /* N[j] is a column of the unity matrix I */ - mpq_sub(bbar[k], bbar[k], x); - } - else - { /* N[j] is a column of the original constraint matrix -A */ - for (ptr = A_ptr[k-m]; ptr < A_ptr[k-m+1]; ptr++) - { mpq_mul(temp, A_val[ptr], x); - mpq_add(bbar[A_ind[ptr]], bbar[A_ind[ptr]], temp); - } - } - } - /* bbar := inv(B) * bbar */ - bfx_ftran(ssx->binv, bbar, 0); -#if 1 - /* compute value of the objective function */ - /* bbar[0] := c[0] */ - mpq_set(bbar[0], coef[0]); - /* bbar[0] := bbar[0] + sum{i in B} cB[i] * xB[i] */ - for (i = 1; i <= m; i++) - { k = Q_col[i]; /* x[k] = xB[i] */ - if (mpq_sgn(coef[k]) == 0) continue; - mpq_mul(temp, coef[k], bbar[i]); - mpq_add(bbar[0], bbar[0], temp); - } - /* bbar[0] := bbar[0] + sum{j in N} cN[j] * xN[j] */ - for (j = 1; j <= n; j++) - { k = Q_col[m+j]; /* x[k] = xN[j] */ - if (mpq_sgn(coef[k]) == 0) continue; - ssx_get_xNj(ssx, j, x); - mpq_mul(temp, coef[k], x); - mpq_add(bbar[0], bbar[0], temp); - } -#endif - mpq_clear(x); - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_pi - compute values of simplex multipliers. -// -// This routine computes values of simplex multipliers (shadow prices) -// pi in the current basic solution as follows: -// -// pi = inv(B') * cB, -// -// where B' is a matrix transposed to the basis matrix B, cB is a vector -// of objective coefficients at basic variables xB. */ - -void ssx_eval_pi(SSX *ssx) -{ int m = ssx->m; - mpq_t *coef = ssx->coef; - int *Q_col = ssx->Q_col; - mpq_t *pi = ssx->pi; - int i; - /* pi := cB */ - for (i = 1; i <= m; i++) mpq_set(pi[i], coef[Q_col[i]]); - /* pi := inv(B') * cB */ - bfx_btran(ssx->binv, pi); - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_dj - compute reduced cost of non-basic variable. -// -// This routine computes reduced cost d[j] of non-basic variable xN[j] -// in the current basic solution as follows: -// -// d[j] = cN[j] - N[j] * pi, -// -// where cN[j] is an objective coefficient at xN[j], N[j] is a column -// of the augmented constraint matrix (I | -A) corresponding to xN[j], -// pi is the vector of simplex multipliers (shadow prices). */ - -void ssx_eval_dj(SSX *ssx, int j, mpq_t dj) -{ int m = ssx->m; - int n = ssx->n; - mpq_t *coef = ssx->coef; - int *A_ptr = ssx->A_ptr; - int *A_ind = ssx->A_ind; - mpq_t *A_val = ssx->A_val; - int *Q_col = ssx->Q_col; - mpq_t *pi = ssx->pi; - int k, ptr, end; - mpq_t temp; - mpq_init(temp); - xassert(1 <= j && j <= n); - k = Q_col[m+j]; /* x[k] = xN[j] */ - xassert(1 <= k && k <= m+n); - /* j-th column of the matrix N is k-th column of the augmented - constraint matrix (I | -A) */ - if (k <= m) - { /* it is a column of the unity matrix I */ - mpq_sub(dj, coef[k], pi[k]); - } - else - { /* it is a column of the original constraint matrix -A */ - mpq_set(dj, coef[k]); - for (ptr = A_ptr[k-m], end = A_ptr[k-m+1]; ptr < end; ptr++) - { mpq_mul(temp, A_val[ptr], pi[A_ind[ptr]]); - mpq_add(dj, dj, temp); - } - } - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_cbar - compute reduced costs of all non-basic variables. -// -// This routine computes the vector of reduced costs pi in the current -// basic solution for all non-basic variables, including fixed ones. */ - -void ssx_eval_cbar(SSX *ssx) -{ int n = ssx->n; - mpq_t *cbar = ssx->cbar; - int j; - for (j = 1; j <= n; j++) - ssx_eval_dj(ssx, j, cbar[j]); - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_rho - compute p-th row of the inverse. -// -// This routine computes p-th row of the matrix inv(B), where B is the -// current basis matrix. -// -// p-th row of the inverse is computed using the following formula: -// -// rho = inv(B') * e[p], -// -// where B' is a matrix transposed to B, e[p] is a unity vector, which -// contains one in p-th position. */ - -void ssx_eval_rho(SSX *ssx) -{ int m = ssx->m; - int p = ssx->p; - mpq_t *rho = ssx->rho; - int i; - xassert(1 <= p && p <= m); - /* rho := 0 */ - for (i = 1; i <= m; i++) mpq_set_si(rho[i], 0, 1); - /* rho := e[p] */ - mpq_set_si(rho[p], 1, 1); - /* rho := inv(B') * rho */ - bfx_btran(ssx->binv, rho); - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_row - compute pivot row of the simplex table. -// -// This routine computes p-th (pivot) row of the current simplex table -// A~ = - inv(B) * N using the following formula: -// -// A~[p] = - N' * inv(B') * e[p] = - N' * rho[p], -// -// where N' is a matrix transposed to the matrix N, rho[p] is p-th row -// of the inverse inv(B). */ - -void ssx_eval_row(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int *A_ptr = ssx->A_ptr; - int *A_ind = ssx->A_ind; - mpq_t *A_val = ssx->A_val; - int *Q_col = ssx->Q_col; - mpq_t *rho = ssx->rho; - mpq_t *ap = ssx->ap; - int j, k, ptr; - mpq_t temp; - mpq_init(temp); - for (j = 1; j <= n; j++) - { /* ap[j] := - N'[j] * rho (inner product) */ - k = Q_col[m+j]; /* x[k] = xN[j] */ - if (k <= m) - mpq_neg(ap[j], rho[k]); - else - { mpq_set_si(ap[j], 0, 1); - for (ptr = A_ptr[k-m]; ptr < A_ptr[k-m+1]; ptr++) - { mpq_mul(temp, A_val[ptr], rho[A_ind[ptr]]); - mpq_add(ap[j], ap[j], temp); - } - } - } - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- -// ssx_eval_col - compute pivot column of the simplex table. -// -// This routine computes q-th (pivot) column of the current simplex -// table A~ = - inv(B) * N using the following formula: -// -// A~[q] = - inv(B) * N[q], -// -// where N[q] is q-th column of the matrix N corresponding to chosen -// non-basic variable xN[q]. */ - -void ssx_eval_col(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int *A_ptr = ssx->A_ptr; - int *A_ind = ssx->A_ind; - mpq_t *A_val = ssx->A_val; - int *Q_col = ssx->Q_col; - int q = ssx->q; - mpq_t *aq = ssx->aq; - int i, k, ptr; - xassert(1 <= q && q <= n); - /* aq := 0 */ - for (i = 1; i <= m; i++) mpq_set_si(aq[i], 0, 1); - /* aq := N[q] */ - k = Q_col[m+q]; /* x[k] = xN[q] */ - if (k <= m) - { /* N[q] is a column of the unity matrix I */ - mpq_set_si(aq[k], 1, 1); - } - else - { /* N[q] is a column of the original constraint matrix -A */ - for (ptr = A_ptr[k-m]; ptr < A_ptr[k-m+1]; ptr++) - mpq_neg(aq[A_ind[ptr]], A_val[ptr]); - } - /* aq := inv(B) * aq */ - bfx_ftran(ssx->binv, aq, 1); - /* aq := - aq */ - for (i = 1; i <= m; i++) mpq_neg(aq[i], aq[i]); - return; -} - -/*---------------------------------------------------------------------- -// ssx_chuzc - choose pivot column. -// -// This routine chooses non-basic variable xN[q] whose reduced cost -// indicates possible improving of the objective function to enter it -// in the basis. -// -// Currently the standard (textbook) pricing is used, i.e. that -// non-basic variable is preferred which has greatest reduced cost (in -// magnitude). -// -// If xN[q] has been chosen, the routine stores its number q and also -// sets the flag q_dir that indicates direction in which xN[q] has to -// change (+1 means increasing, -1 means decreasing). -// -// If the choice cannot be made, because the current basic solution is -// dual feasible, the routine sets the number q to 0. */ - -void ssx_chuzc(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int dir = (ssx->dir == SSX_MIN ? +1 : -1); - int *Q_col = ssx->Q_col; - int *stat = ssx->stat; - mpq_t *cbar = ssx->cbar; - int j, k, s, q, q_dir; - double best, temp; - /* nothing is chosen so far */ - q = 0, q_dir = 0, best = 0.0; - /* look through the list of non-basic variables */ - for (j = 1; j <= n; j++) - { k = Q_col[m+j]; /* x[k] = xN[j] */ - s = dir * mpq_sgn(cbar[j]); - if ((stat[k] == SSX_NF || stat[k] == SSX_NL) && s < 0 || - (stat[k] == SSX_NF || stat[k] == SSX_NU) && s > 0) - { /* reduced cost of xN[j] indicates possible improving of - the objective function */ - temp = fabs(mpq_get_d(cbar[j])); - xassert(temp != 0.0); - if (q == 0 || best < temp) - q = j, q_dir = - s, best = temp; - } - } - ssx->q = q, ssx->q_dir = q_dir; - return; -} - -/*---------------------------------------------------------------------- -// ssx_chuzr - choose pivot row. -// -// This routine looks through elements of q-th column of the simplex -// table and chooses basic variable xB[p] which should leave the basis. -// -// The choice is based on the standard (textbook) ratio test. -// -// If xB[p] has been chosen, the routine stores its number p and also -// sets its non-basic status p_stat which should be assigned to xB[p] -// when it has left the basis and become xN[q]. -// -// Special case p < 0 means that xN[q] is double-bounded variable and -// it reaches its opposite bound before any basic variable does that, -// so the current basis remains unchanged. -// -// If the choice cannot be made, because xN[q] can infinitely change in -// the feasible direction, the routine sets the number p to 0. */ - -void ssx_chuzr(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int *type = ssx->type; - mpq_t *lb = ssx->lb; - mpq_t *ub = ssx->ub; - int *Q_col = ssx->Q_col; - mpq_t *bbar = ssx->bbar; - int q = ssx->q; - mpq_t *aq = ssx->aq; - int q_dir = ssx->q_dir; - int i, k, s, t, p, p_stat; - mpq_t teta, temp; - mpq_init(teta); - mpq_init(temp); - xassert(1 <= q && q <= n); - xassert(q_dir == +1 || q_dir == -1); - /* nothing is chosen so far */ - p = 0, p_stat = 0; - /* look through the list of basic variables */ - for (i = 1; i <= m; i++) - { s = q_dir * mpq_sgn(aq[i]); - if (s < 0) - { /* xB[i] decreases */ - k = Q_col[i]; /* x[k] = xB[i] */ - t = type[k]; - if (t == SSX_LO || t == SSX_DB || t == SSX_FX) - { /* xB[i] has finite lower bound */ - mpq_sub(temp, bbar[i], lb[k]); - mpq_div(temp, temp, aq[i]); - mpq_abs(temp, temp); - if (p == 0 || mpq_cmp(teta, temp) > 0) - { p = i; - p_stat = (t == SSX_FX ? SSX_NS : SSX_NL); - mpq_set(teta, temp); - } - } - } - else if (s > 0) - { /* xB[i] increases */ - k = Q_col[i]; /* x[k] = xB[i] */ - t = type[k]; - if (t == SSX_UP || t == SSX_DB || t == SSX_FX) - { /* xB[i] has finite upper bound */ - mpq_sub(temp, bbar[i], ub[k]); - mpq_div(temp, temp, aq[i]); - mpq_abs(temp, temp); - if (p == 0 || mpq_cmp(teta, temp) > 0) - { p = i; - p_stat = (t == SSX_FX ? SSX_NS : SSX_NU); - mpq_set(teta, temp); - } - } - } - /* if something has been chosen and the ratio test indicates - exact degeneracy, the search can be finished */ - if (p != 0 && mpq_sgn(teta) == 0) break; - } - /* if xN[q] is double-bounded, check if it can reach its opposite - bound before any basic variable */ - k = Q_col[m+q]; /* x[k] = xN[q] */ - if (type[k] == SSX_DB) - { mpq_sub(temp, ub[k], lb[k]); - if (p == 0 || mpq_cmp(teta, temp) > 0) - { p = -1; - p_stat = -1; - mpq_set(teta, temp); - } - } - ssx->p = p; - ssx->p_stat = p_stat; - /* if xB[p] has been chosen, determine its actual change in the - adjacent basis (it has the same sign as q_dir) */ - if (p != 0) - { xassert(mpq_sgn(teta) >= 0); - if (q_dir > 0) - mpq_set(ssx->delta, teta); - else - mpq_neg(ssx->delta, teta); - } - mpq_clear(teta); - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- -// ssx_update_bbar - update values of basic variables. -// -// This routine recomputes the current values of basic variables for -// the adjacent basis. -// -// The simplex table for the current basis is the following: -// -// xB[i] = sum{j in 1..n} alfa[i,j] * xN[q], i = 1,...,m -// -// therefore -// -// delta xB[i] = alfa[i,q] * delta xN[q], i = 1,...,m -// -// where delta xN[q] = xN.new[q] - xN[q] is the change of xN[q] in the -// adjacent basis, and delta xB[i] = xB.new[i] - xB[i] is the change of -// xB[i]. This gives formulae for recomputing values of xB[i]: -// -// xB.new[p] = xN[q] + delta xN[q] -// -// (because xN[q] becomes xB[p] in the adjacent basis), and -// -// xB.new[i] = xB[i] + alfa[i,q] * delta xN[q], i != p -// -// for other basic variables. */ - -void ssx_update_bbar(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - mpq_t *bbar = ssx->bbar; - mpq_t *cbar = ssx->cbar; - int p = ssx->p; - int q = ssx->q; - mpq_t *aq = ssx->aq; - int i; - mpq_t temp; - mpq_init(temp); - xassert(1 <= q && q <= n); - if (p < 0) - { /* xN[q] is double-bounded and goes to its opposite bound */ - /* nop */; - } - else - { /* xN[q] becomes xB[p] in the adjacent basis */ - /* xB.new[p] = xN[q] + delta xN[q] */ - xassert(1 <= p && p <= m); - ssx_get_xNj(ssx, q, temp); - mpq_add(bbar[p], temp, ssx->delta); - } - /* update values of other basic variables depending on xN[q] */ - for (i = 1; i <= m; i++) - { if (i == p) continue; - /* xB.new[i] = xB[i] + alfa[i,q] * delta xN[q] */ - if (mpq_sgn(aq[i]) == 0) continue; - mpq_mul(temp, aq[i], ssx->delta); - mpq_add(bbar[i], bbar[i], temp); - } -#if 1 - /* update value of the objective function */ - /* z.new = z + d[q] * delta xN[q] */ - mpq_mul(temp, cbar[q], ssx->delta); - mpq_add(bbar[0], bbar[0], temp); -#endif - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- --- ssx_update_pi - update simplex multipliers. --- --- This routine recomputes the vector of simplex multipliers for the --- adjacent basis. */ - -void ssx_update_pi(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - mpq_t *pi = ssx->pi; - mpq_t *cbar = ssx->cbar; - int p = ssx->p; - int q = ssx->q; - mpq_t *aq = ssx->aq; - mpq_t *rho = ssx->rho; - int i; - mpq_t new_dq, temp; - mpq_init(new_dq); - mpq_init(temp); - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n); - /* compute d[q] in the adjacent basis */ - mpq_div(new_dq, cbar[q], aq[p]); - /* update the vector of simplex multipliers */ - for (i = 1; i <= m; i++) - { if (mpq_sgn(rho[i]) == 0) continue; - mpq_mul(temp, new_dq, rho[i]); - mpq_sub(pi[i], pi[i], temp); - } - mpq_clear(new_dq); - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- -// ssx_update_cbar - update reduced costs of non-basic variables. -// -// This routine recomputes the vector of reduced costs of non-basic -// variables for the adjacent basis. */ - -void ssx_update_cbar(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - mpq_t *cbar = ssx->cbar; - int p = ssx->p; - int q = ssx->q; - mpq_t *ap = ssx->ap; - int j; - mpq_t temp; - mpq_init(temp); - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n); - /* compute d[q] in the adjacent basis */ - /* d.new[q] = d[q] / alfa[p,q] */ - mpq_div(cbar[q], cbar[q], ap[q]); - /* update reduced costs of other non-basic variables */ - for (j = 1; j <= n; j++) - { if (j == q) continue; - /* d.new[j] = d[j] - (alfa[p,j] / alfa[p,q]) * d[q] */ - if (mpq_sgn(ap[j]) == 0) continue; - mpq_mul(temp, ap[j], cbar[q]); - mpq_sub(cbar[j], cbar[j], temp); - } - mpq_clear(temp); - return; -} - -/*---------------------------------------------------------------------- -// ssx_change_basis - change current basis to adjacent one. -// -// This routine changes the current basis to the adjacent one swapping -// basic variable xB[p] and non-basic variable xN[q]. */ - -void ssx_change_basis(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int *type = ssx->type; - int *stat = ssx->stat; - int *Q_row = ssx->Q_row; - int *Q_col = ssx->Q_col; - int p = ssx->p; - int q = ssx->q; - int p_stat = ssx->p_stat; - int k, kp, kq; - if (p < 0) - { /* special case: xN[q] goes to its opposite bound */ - xassert(1 <= q && q <= n); - k = Q_col[m+q]; /* x[k] = xN[q] */ - xassert(type[k] == SSX_DB); - switch (stat[k]) - { case SSX_NL: - stat[k] = SSX_NU; - break; - case SSX_NU: - stat[k] = SSX_NL; - break; - default: - xassert(stat != stat); - } - } - else - { /* xB[p] leaves the basis, xN[q] enters the basis */ - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n); - kp = Q_col[p]; /* x[kp] = xB[p] */ - kq = Q_col[m+q]; /* x[kq] = xN[q] */ - /* check non-basic status of xB[p] which becomes xN[q] */ - switch (type[kp]) - { case SSX_FR: - xassert(p_stat == SSX_NF); - break; - case SSX_LO: - xassert(p_stat == SSX_NL); - break; - case SSX_UP: - xassert(p_stat == SSX_NU); - break; - case SSX_DB: - xassert(p_stat == SSX_NL || p_stat == SSX_NU); - break; - case SSX_FX: - xassert(p_stat == SSX_NS); - break; - default: - xassert(type != type); - } - /* swap xB[p] and xN[q] */ - stat[kp] = (char)p_stat, stat[kq] = SSX_BS; - Q_row[kp] = m+q, Q_row[kq] = p; - Q_col[p] = kq, Q_col[m+q] = kp; - /* update factorization of the basis matrix */ - if (bfx_update(ssx->binv, p)) - { if (ssx_factorize(ssx)) - xassert(("Internal error: basis matrix is singular", 0)); - } - } - return; -} - -/*---------------------------------------------------------------------- -// ssx_delete - delete simplex solver workspace. -// -// This routine deletes the simplex solver workspace freeing all the -// memory allocated to this object. */ - -void ssx_delete(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int nnz = ssx->A_ptr[n+1]-1; - int i, j, k; - xfree(ssx->type); - for (k = 1; k <= m+n; k++) mpq_clear(ssx->lb[k]); - xfree(ssx->lb); - for (k = 1; k <= m+n; k++) mpq_clear(ssx->ub[k]); - xfree(ssx->ub); - for (k = 0; k <= m+n; k++) mpq_clear(ssx->coef[k]); - xfree(ssx->coef); - xfree(ssx->A_ptr); - xfree(ssx->A_ind); - for (k = 1; k <= nnz; k++) mpq_clear(ssx->A_val[k]); - xfree(ssx->A_val); - xfree(ssx->stat); - xfree(ssx->Q_row); - xfree(ssx->Q_col); - bfx_delete_binv(ssx->binv); - for (i = 0; i <= m; i++) mpq_clear(ssx->bbar[i]); - xfree(ssx->bbar); - for (i = 1; i <= m; i++) mpq_clear(ssx->pi[i]); - xfree(ssx->pi); - for (j = 1; j <= n; j++) mpq_clear(ssx->cbar[j]); - xfree(ssx->cbar); - for (i = 1; i <= m; i++) mpq_clear(ssx->rho[i]); - xfree(ssx->rho); - for (j = 1; j <= n; j++) mpq_clear(ssx->ap[j]); - xfree(ssx->ap); - for (i = 1; i <= m; i++) mpq_clear(ssx->aq[i]); - xfree(ssx->aq); - mpq_clear(ssx->delta); - xfree(ssx); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/glpssx02.c b/code/3rd_glpk/draft/glpssx02.c deleted file mode 100644 index 81db1350..00000000 --- a/code/3rd_glpk/draft/glpssx02.c +++ /dev/null @@ -1,523 +0,0 @@ -/* glpssx02.c (simplex method, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "glpssx.h" - -static void show_progress(SSX *ssx, int phase) -{ /* this auxiliary routine displays information about progress of - the search */ - int i, def = 0; - for (i = 1; i <= ssx->m; i++) - if (ssx->type[ssx->Q_col[i]] == SSX_FX) def++; - xprintf("%s%6d: %s = %22.15g (%d)\n", phase == 1 ? " " : "*", - ssx->it_cnt, phase == 1 ? "infsum" : "objval", - mpq_get_d(ssx->bbar[0]), def); -#if 0 - ssx->tm_lag = utime(); -#else - ssx->tm_lag = xtime(); -#endif - return; -} - -/*---------------------------------------------------------------------- -// ssx_phase_I - find primal feasible solution. -// -// This routine implements phase I of the primal simplex method. -// -// On exit the routine returns one of the following codes: -// -// 0 - feasible solution found; -// 1 - problem has no feasible solution; -// 2 - iterations limit exceeded; -// 3 - time limit exceeded. -----------------------------------------------------------------------*/ - -int ssx_phase_I(SSX *ssx) -{ int m = ssx->m; - int n = ssx->n; - int *type = ssx->type; - mpq_t *lb = ssx->lb; - mpq_t *ub = ssx->ub; - mpq_t *coef = ssx->coef; - int *A_ptr = ssx->A_ptr; - int *A_ind = ssx->A_ind; - mpq_t *A_val = ssx->A_val; - int *Q_col = ssx->Q_col; - mpq_t *bbar = ssx->bbar; - mpq_t *pi = ssx->pi; - mpq_t *cbar = ssx->cbar; - int *orig_type, orig_dir; - mpq_t *orig_lb, *orig_ub, *orig_coef; - int i, k, ret; - /* save components of the original LP problem, which are changed - by the routine */ - orig_type = xcalloc(1+m+n, sizeof(int)); - orig_lb = xcalloc(1+m+n, sizeof(mpq_t)); - orig_ub = xcalloc(1+m+n, sizeof(mpq_t)); - orig_coef = xcalloc(1+m+n, sizeof(mpq_t)); - for (k = 1; k <= m+n; k++) - { orig_type[k] = type[k]; - mpq_init(orig_lb[k]); - mpq_set(orig_lb[k], lb[k]); - mpq_init(orig_ub[k]); - mpq_set(orig_ub[k], ub[k]); - } - orig_dir = ssx->dir; - for (k = 0; k <= m+n; k++) - { mpq_init(orig_coef[k]); - mpq_set(orig_coef[k], coef[k]); - } - /* build an artificial basic solution, which is primal feasible, - and also build an auxiliary objective function to minimize the - sum of infeasibilities for the original problem */ - ssx->dir = SSX_MIN; - for (k = 0; k <= m+n; k++) mpq_set_si(coef[k], 0, 1); - mpq_set_si(bbar[0], 0, 1); - for (i = 1; i <= m; i++) - { int t; - k = Q_col[i]; /* x[k] = xB[i] */ - t = type[k]; - if (t == SSX_LO || t == SSX_DB || t == SSX_FX) - { /* in the original problem x[k] has lower bound */ - if (mpq_cmp(bbar[i], lb[k]) < 0) - { /* which is violated */ - type[k] = SSX_UP; - mpq_set(ub[k], lb[k]); - mpq_set_si(lb[k], 0, 1); - mpq_set_si(coef[k], -1, 1); - mpq_add(bbar[0], bbar[0], ub[k]); - mpq_sub(bbar[0], bbar[0], bbar[i]); - } - } - if (t == SSX_UP || t == SSX_DB || t == SSX_FX) - { /* in the original problem x[k] has upper bound */ - if (mpq_cmp(bbar[i], ub[k]) > 0) - { /* which is violated */ - type[k] = SSX_LO; - mpq_set(lb[k], ub[k]); - mpq_set_si(ub[k], 0, 1); - mpq_set_si(coef[k], +1, 1); - mpq_add(bbar[0], bbar[0], bbar[i]); - mpq_sub(bbar[0], bbar[0], lb[k]); - } - } - } - /* now the initial basic solution should be primal feasible due - to changes of bounds of some basic variables, which turned to - implicit artifical variables */ - /* compute simplex multipliers and reduced costs */ - ssx_eval_pi(ssx); - ssx_eval_cbar(ssx); - /* display initial progress of the search */ -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ON) -#endif - show_progress(ssx, 1); - /* main loop starts here */ - for (;;) - { /* display current progress of the search */ -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ON) -#endif -#if 0 - if (utime() - ssx->tm_lag >= ssx->out_frq - 0.001) -#else - if (xdifftime(xtime(), ssx->tm_lag) >= ssx->out_frq - 0.001) -#endif - show_progress(ssx, 1); - /* we do not need to wait until all artificial variables have - left the basis */ - if (mpq_sgn(bbar[0]) == 0) - { /* the sum of infeasibilities is zero, therefore the current - solution is primal feasible for the original problem */ - ret = 0; - break; - } - /* check if the iterations limit has been exhausted */ - if (ssx->it_lim == 0) - { ret = 2; - break; - } - /* check if the time limit has been exhausted */ -#if 0 - if (ssx->tm_lim >= 0.0 && ssx->tm_lim <= utime() - ssx->tm_beg) -#else - if (ssx->tm_lim >= 0.0 && - ssx->tm_lim <= xdifftime(xtime(), ssx->tm_beg)) -#endif - { ret = 3; - break; - } - /* choose non-basic variable xN[q] */ - ssx_chuzc(ssx); - /* if xN[q] cannot be chosen, the sum of infeasibilities is - minimal but non-zero; therefore the original problem has no - primal feasible solution */ - if (ssx->q == 0) - { ret = 1; - break; - } - /* compute q-th column of the simplex table */ - ssx_eval_col(ssx); - /* choose basic variable xB[p] */ - ssx_chuzr(ssx); - /* the sum of infeasibilities cannot be negative, therefore - the auxiliary lp problem cannot have unbounded solution */ - xassert(ssx->p != 0); - /* update values of basic variables */ - ssx_update_bbar(ssx); - if (ssx->p > 0) - { /* compute p-th row of the inverse inv(B) */ - ssx_eval_rho(ssx); - /* compute p-th row of the simplex table */ - ssx_eval_row(ssx); - xassert(mpq_cmp(ssx->aq[ssx->p], ssx->ap[ssx->q]) == 0); - /* update simplex multipliers */ - ssx_update_pi(ssx); - /* update reduced costs of non-basic variables */ - ssx_update_cbar(ssx); - } - /* xB[p] is leaving the basis; if it is implicit artificial - variable, the corresponding residual vanishes; therefore - bounds of this variable should be restored to the original - values */ - if (ssx->p > 0) - { k = Q_col[ssx->p]; /* x[k] = xB[p] */ - if (type[k] != orig_type[k]) - { /* x[k] is implicit artificial variable */ - type[k] = orig_type[k]; - mpq_set(lb[k], orig_lb[k]); - mpq_set(ub[k], orig_ub[k]); - xassert(ssx->p_stat == SSX_NL || ssx->p_stat == SSX_NU); - ssx->p_stat = (ssx->p_stat == SSX_NL ? SSX_NU : SSX_NL); - if (type[k] == SSX_FX) ssx->p_stat = SSX_NS; - /* nullify the objective coefficient at x[k] */ - mpq_set_si(coef[k], 0, 1); - /* since coef[k] has been changed, we need to compute - new reduced cost of x[k], which it will have in the - adjacent basis */ - /* the formula d[j] = cN[j] - pi' * N[j] is used (note - that the vector pi is not changed, because it depends - on objective coefficients at basic variables, but in - the adjacent basis, for which the vector pi has been - just recomputed, x[k] is non-basic) */ - if (k <= m) - { /* x[k] is auxiliary variable */ - mpq_neg(cbar[ssx->q], pi[k]); - } - else - { /* x[k] is structural variable */ - int ptr; - mpq_t temp; - mpq_init(temp); - mpq_set_si(cbar[ssx->q], 0, 1); - for (ptr = A_ptr[k-m]; ptr < A_ptr[k-m+1]; ptr++) - { mpq_mul(temp, pi[A_ind[ptr]], A_val[ptr]); - mpq_add(cbar[ssx->q], cbar[ssx->q], temp); - } - mpq_clear(temp); - } - } - } - /* jump to the adjacent vertex of the polyhedron */ - ssx_change_basis(ssx); - /* one simplex iteration has been performed */ - if (ssx->it_lim > 0) ssx->it_lim--; - ssx->it_cnt++; - } - /* display final progress of the search */ -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ON) -#endif - show_progress(ssx, 1); - /* restore components of the original problem, which were changed - by the routine */ - for (k = 1; k <= m+n; k++) - { type[k] = orig_type[k]; - mpq_set(lb[k], orig_lb[k]); - mpq_clear(orig_lb[k]); - mpq_set(ub[k], orig_ub[k]); - mpq_clear(orig_ub[k]); - } - ssx->dir = orig_dir; - for (k = 0; k <= m+n; k++) - { mpq_set(coef[k], orig_coef[k]); - mpq_clear(orig_coef[k]); - } - xfree(orig_type); - xfree(orig_lb); - xfree(orig_ub); - xfree(orig_coef); - /* return to the calling program */ - return ret; -} - -/*---------------------------------------------------------------------- -// ssx_phase_II - find optimal solution. -// -// This routine implements phase II of the primal simplex method. -// -// On exit the routine returns one of the following codes: -// -// 0 - optimal solution found; -// 1 - problem has unbounded solution; -// 2 - iterations limit exceeded; -// 3 - time limit exceeded. -----------------------------------------------------------------------*/ - -int ssx_phase_II(SSX *ssx) -{ int ret; - /* display initial progress of the search */ -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ON) -#endif - show_progress(ssx, 2); - /* main loop starts here */ - for (;;) - { /* display current progress of the search */ -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ON) -#endif -#if 0 - if (utime() - ssx->tm_lag >= ssx->out_frq - 0.001) -#else - if (xdifftime(xtime(), ssx->tm_lag) >= ssx->out_frq - 0.001) -#endif - show_progress(ssx, 2); - /* check if the iterations limit has been exhausted */ - if (ssx->it_lim == 0) - { ret = 2; - break; - } - /* check if the time limit has been exhausted */ -#if 0 - if (ssx->tm_lim >= 0.0 && ssx->tm_lim <= utime() - ssx->tm_beg) -#else - if (ssx->tm_lim >= 0.0 && - ssx->tm_lim <= xdifftime(xtime(), ssx->tm_beg)) -#endif - { ret = 3; - break; - } - /* choose non-basic variable xN[q] */ - ssx_chuzc(ssx); - /* if xN[q] cannot be chosen, the current basic solution is - dual feasible and therefore optimal */ - if (ssx->q == 0) - { ret = 0; - break; - } - /* compute q-th column of the simplex table */ - ssx_eval_col(ssx); - /* choose basic variable xB[p] */ - ssx_chuzr(ssx); - /* if xB[p] cannot be chosen, the problem has no dual feasible - solution (i.e. unbounded) */ - if (ssx->p == 0) - { ret = 1; - break; - } - /* update values of basic variables */ - ssx_update_bbar(ssx); - if (ssx->p > 0) - { /* compute p-th row of the inverse inv(B) */ - ssx_eval_rho(ssx); - /* compute p-th row of the simplex table */ - ssx_eval_row(ssx); - xassert(mpq_cmp(ssx->aq[ssx->p], ssx->ap[ssx->q]) == 0); -#if 0 - /* update simplex multipliers */ - ssx_update_pi(ssx); -#endif - /* update reduced costs of non-basic variables */ - ssx_update_cbar(ssx); - } - /* jump to the adjacent vertex of the polyhedron */ - ssx_change_basis(ssx); - /* one simplex iteration has been performed */ - if (ssx->it_lim > 0) ssx->it_lim--; - ssx->it_cnt++; - } - /* display final progress of the search */ -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ON) -#endif - show_progress(ssx, 2); - /* return to the calling program */ - return ret; -} - -/*---------------------------------------------------------------------- -// ssx_driver - base driver to exact simplex method. -// -// This routine is a base driver to a version of the primal simplex -// method using exact (bignum) arithmetic. -// -// On exit the routine returns one of the following codes: -// -// 0 - optimal solution found; -// 1 - problem has no feasible solution; -// 2 - problem has unbounded solution; -// 3 - iterations limit exceeded (phase I); -// 4 - iterations limit exceeded (phase II); -// 5 - time limit exceeded (phase I); -// 6 - time limit exceeded (phase II); -// 7 - initial basis matrix is exactly singular. -----------------------------------------------------------------------*/ - -int ssx_driver(SSX *ssx) -{ int m = ssx->m; - int *type = ssx->type; - mpq_t *lb = ssx->lb; - mpq_t *ub = ssx->ub; - int *Q_col = ssx->Q_col; - mpq_t *bbar = ssx->bbar; - int i, k, ret; - ssx->tm_beg = xtime(); - /* factorize the initial basis matrix */ - if (ssx_factorize(ssx)) -#if 0 /* 25/XI-2017 */ - { xprintf("Initial basis matrix is singular\n"); -#else - { if (ssx->msg_lev >= GLP_MSG_ERR) - xprintf("Initial basis matrix is singular\n"); -#endif - ret = 7; - goto done; - } - /* compute values of basic variables */ - ssx_eval_bbar(ssx); - /* check if the initial basic solution is primal feasible */ - for (i = 1; i <= m; i++) - { int t; - k = Q_col[i]; /* x[k] = xB[i] */ - t = type[k]; - if (t == SSX_LO || t == SSX_DB || t == SSX_FX) - { /* x[k] has lower bound */ - if (mpq_cmp(bbar[i], lb[k]) < 0) - { /* which is violated */ - break; - } - } - if (t == SSX_UP || t == SSX_DB || t == SSX_FX) - { /* x[k] has upper bound */ - if (mpq_cmp(bbar[i], ub[k]) > 0) - { /* which is violated */ - break; - } - } - } - if (i > m) - { /* no basic variable violates its bounds */ - ret = 0; - goto skip; - } - /* phase I: find primal feasible solution */ - ret = ssx_phase_I(ssx); - switch (ret) - { case 0: - ret = 0; - break; - case 1: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("PROBLEM HAS NO FEASIBLE SOLUTION\n"); - ret = 1; - break; - case 2: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("ITERATIONS LIMIT EXCEEDED; SEARCH TERMINATED\n"); - ret = 3; - break; - case 3: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("TIME LIMIT EXCEEDED; SEARCH TERMINATED\n"); - ret = 5; - break; - default: - xassert(ret != ret); - } - /* compute values of basic variables (actually only the objective - value needs to be computed) */ - ssx_eval_bbar(ssx); -skip: /* compute simplex multipliers */ - ssx_eval_pi(ssx); - /* compute reduced costs of non-basic variables */ - ssx_eval_cbar(ssx); - /* if phase I failed, do not start phase II */ - if (ret != 0) goto done; - /* phase II: find optimal solution */ - ret = ssx_phase_II(ssx); - switch (ret) - { case 0: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("OPTIMAL SOLUTION FOUND\n"); - ret = 0; - break; - case 1: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("PROBLEM HAS UNBOUNDED SOLUTION\n"); - ret = 2; - break; - case 2: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("ITERATIONS LIMIT EXCEEDED; SEARCH TERMINATED\n"); - ret = 4; - break; - case 3: -#if 1 /* 25/XI-2017 */ - if (ssx->msg_lev >= GLP_MSG_ALL) -#endif - xprintf("TIME LIMIT EXCEEDED; SEARCH TERMINATED\n"); - ret = 6; - break; - default: - xassert(ret != ret); - } -done: /* decrease the time limit by the spent amount of time */ - if (ssx->tm_lim >= 0.0) -#if 0 - { ssx->tm_lim -= utime() - ssx->tm_beg; -#else - { ssx->tm_lim -= xdifftime(xtime(), ssx->tm_beg); -#endif - if (ssx->tm_lim < 0.0) ssx->tm_lim = 0.0; - } - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/ios.h b/code/3rd_glpk/draft/ios.h deleted file mode 100644 index 1cb07ee0..00000000 --- a/code/3rd_glpk/draft/ios.h +++ /dev/null @@ -1,547 +0,0 @@ -/* ios.h (integer optimization suite) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef IOS_H -#define IOS_H - -#include "prob.h" - -#if 1 /* 02/II-2018 */ -#define NEW_LOCAL 1 -#endif - -#if 1 /* 15/II-2018 */ -#define NEW_COVER 1 -#endif - -typedef struct IOSLOT IOSLOT; -typedef struct IOSNPD IOSNPD; -typedef struct IOSBND IOSBND; -typedef struct IOSTAT IOSTAT; -typedef struct IOSROW IOSROW; -typedef struct IOSAIJ IOSAIJ; -#ifdef NEW_LOCAL /* 02/II-2018 */ -typedef glp_prob IOSPOOL; -typedef GLPROW IOSCUT; -#else -typedef struct IOSPOOL IOSPOOL; -typedef struct IOSCUT IOSCUT; -#endif - -struct glp_tree -{ /* branch-and-bound tree */ - int magic; - /* magic value used for debugging */ - DMP *pool; - /* memory pool to store all IOS components */ - int n; - /* number of columns (variables) */ - /*--------------------------------------------------------------*/ - /* problem components corresponding to the original MIP and its - LP relaxation (used to restore the original problem object on - exit from the solver) */ - int orig_m; - /* number of rows */ - unsigned char *orig_type; /* uchar orig_type[1+orig_m+n]; */ - /* types of all variables */ - double *orig_lb; /* double orig_lb[1+orig_m+n]; */ - /* lower bounds of all variables */ - double *orig_ub; /* double orig_ub[1+orig_m+n]; */ - /* upper bounds of all variables */ - unsigned char *orig_stat; /* uchar orig_stat[1+orig_m+n]; */ - /* statuses of all variables */ - double *orig_prim; /* double orig_prim[1+orig_m+n]; */ - /* primal values of all variables */ - double *orig_dual; /* double orig_dual[1+orig_m+n]; */ - /* dual values of all variables */ - double orig_obj; - /* optimal objective value for LP relaxation */ - /*--------------------------------------------------------------*/ - /* branch-and-bound tree */ - int nslots; - /* length of the array of slots (enlarged automatically) */ - int avail; - /* index of the first free slot; 0 means all slots are in use */ - IOSLOT *slot; /* IOSLOT slot[1+nslots]; */ - /* array of slots: - slot[0] is not used; - slot[p], 1 <= p <= nslots, either contains a pointer to some - node of the branch-and-bound tree, in which case p is used on - API level as the reference number of corresponding subproblem, - or is free; all free slots are linked into single linked list; - slot[1] always contains a pointer to the root node (it is free - only if the tree is empty) */ - IOSNPD *head; - /* pointer to the head of the active list */ - IOSNPD *tail; - /* pointer to the tail of the active list */ - /* the active list is a doubly linked list of active subproblems - which correspond to leaves of the tree; all subproblems in the - active list are ordered chronologically (each a new subproblem - is always added to the tail of the list) */ - int a_cnt; - /* current number of active nodes (including the current one) */ - int n_cnt; - /* current number of all (active and inactive) nodes */ - int t_cnt; - /* total number of nodes including those which have been already - removed from the tree; this count is increased by one whenever - a new node is created and never decreased */ - /*--------------------------------------------------------------*/ - /* problem components corresponding to the root subproblem */ - int root_m; - /* number of rows */ - unsigned char *root_type; /* uchar root_type[1+root_m+n]; */ - /* types of all variables */ - double *root_lb; /* double root_lb[1+root_m+n]; */ - /* lower bounds of all variables */ - double *root_ub; /* double root_ub[1+root_m+n]; */ - /* upper bounds of all variables */ - unsigned char *root_stat; /* uchar root_stat[1+root_m+n]; */ - /* statuses of all variables */ - /*--------------------------------------------------------------*/ - /* current subproblem and its LP relaxation */ - IOSNPD *curr; - /* pointer to the current subproblem (which can be only active); - NULL means the current subproblem does not exist */ - glp_prob *mip; - /* original problem object passed to the solver; if the current - subproblem exists, its LP segment corresponds to LP relaxation - of the current subproblem; if the current subproblem does not - exist, its LP segment corresponds to LP relaxation of the root - subproblem (note that the root subproblem may differ from the - original MIP, because it may be preprocessed and/or may have - additional rows) */ - unsigned char *non_int; /* uchar non_int[1+n]; */ - /* these column flags are set each time when LP relaxation of the - current subproblem has been solved; - non_int[0] is not used; - non_int[j], 1 <= j <= n, is j-th column flag; if this flag is - set, corresponding variable is required to be integer, but its - value in basic solution is fractional */ - /*--------------------------------------------------------------*/ - /* problem components corresponding to the parent (predecessor) - subproblem for the current subproblem; used to inspect changes - on freezing the current subproblem */ - int pred_m; - /* number of rows */ - int pred_max; - /* length of the following four arrays (enlarged automatically), - pred_max >= pred_m + n */ - unsigned char *pred_type; /* uchar pred_type[1+pred_m+n]; */ - /* types of all variables */ - double *pred_lb; /* double pred_lb[1+pred_m+n]; */ - /* lower bounds of all variables */ - double *pred_ub; /* double pred_ub[1+pred_m+n]; */ - /* upper bounds of all variables */ - unsigned char *pred_stat; /* uchar pred_stat[1+pred_m+n]; */ - /* statuses of all variables */ - /****************************************************************/ - /* built-in cut generators segment */ - IOSPOOL *local; - /* local cut pool */ -#if 1 /* 13/II-2018 */ - glp_cov *cov_gen; - /* pointer to working area used by the cover cut generator */ -#endif - glp_mir *mir_gen; - /* pointer to working area used by the MIR cut generator */ - glp_cfg *clq_gen; - /* pointer to conflict graph used by the clique cut generator */ - /*--------------------------------------------------------------*/ - void *pcost; - /* pointer to working area used on pseudocost branching */ - int *iwrk; /* int iwrk[1+n]; */ - /* working array */ - double *dwrk; /* double dwrk[1+n]; */ - /* working array */ - /*--------------------------------------------------------------*/ - /* control parameters and statistics */ - const glp_iocp *parm; - /* copy of control parameters passed to the solver */ - double tm_beg; - /* starting time of the search, in seconds; the total time of the - search is the difference between xtime() and tm_beg */ - double tm_lag; - /* the most recent time, in seconds, at which the progress of the - the search was displayed */ - int sol_cnt; - /* number of integer feasible solutions found */ -#if 1 /* 11/VII-2013 */ - void *P; /* glp_prob *P; */ - /* problem passed to glp_intopt */ - void *npp; /* NPP *npp; */ - /* preprocessor workspace or NULL */ - const char *save_sol; - /* filename (template) to save every new solution */ - int save_cnt; - /* count to generate filename */ -#endif - /*--------------------------------------------------------------*/ - /* advanced solver interface */ - int reason; - /* flag indicating the reason why the callback routine is being - called (see glpk.h) */ - int stop; - /* flag indicating that the callback routine requires premature - termination of the search */ - int next_p; - /* reference number of active subproblem selected to continue - the search; 0 means no subproblem has been selected */ - int reopt; - /* flag indicating that the current LP relaxation needs to be - re-optimized */ - int reinv; - /* flag indicating that some (non-active) rows were removed from - the current LP relaxation, so if there no new rows appear, the - basis must be re-factorized */ - int br_var; - /* the number of variable chosen to branch on */ - int br_sel; - /* flag indicating which branch (subproblem) is suggested to be - selected to continue the search: - GLP_DN_BRNCH - select down-branch - GLP_UP_BRNCH - select up-branch - GLP_NO_BRNCH - use general selection technique */ - int child; - /* subproblem reference number corresponding to br_sel */ -}; - -struct IOSLOT -{ /* node subproblem slot */ - IOSNPD *node; - /* pointer to subproblem descriptor; NULL means free slot */ - int next; - /* index of another free slot (only if this slot is free) */ -}; - -struct IOSNPD -{ /* node subproblem descriptor */ - int p; - /* subproblem reference number (it is the index to corresponding - slot, i.e. slot[p] points to this descriptor) */ - IOSNPD *up; - /* pointer to the parent subproblem; NULL means this node is the - root of the tree, in which case p = 1 */ - int level; - /* node level (the root node has level 0) */ - int count; - /* if count = 0, this subproblem is active; if count > 0, this - subproblem is inactive, in which case count is the number of - its child subproblems */ - /* the following three linked lists are destroyed on reviving and - built anew on freezing the subproblem: */ - IOSBND *b_ptr; - /* linked list of rows and columns of the parent subproblem whose - types and bounds were changed */ - IOSTAT *s_ptr; - /* linked list of rows and columns of the parent subproblem whose - statuses were changed */ - IOSROW *r_ptr; - /* linked list of rows (cuts) added to the parent subproblem */ - int solved; - /* how many times LP relaxation of this subproblem was solved; - for inactive subproblem this count is always non-zero; - for active subproblem, which is not current, this count may be - non-zero, if the subproblem was temporarily suspended */ - double lp_obj; - /* optimal objective value to LP relaxation of this subproblem; - on creating a subproblem this value is inherited from its - parent; for the root subproblem, which has no parent, this - value is initially set to -DBL_MAX (minimization) or +DBL_MAX - (maximization); each time the subproblem is re-optimized, this - value is appropriately changed */ - double bound; - /* local lower (minimization) or upper (maximization) bound for - integer optimal solution to *this* subproblem; this bound is - local in the sense that only subproblems in the subtree rooted - at this node cannot have better integer feasible solutions; - on creating a subproblem its local bound is inherited from its - parent and then can be made stronger (never weaker); for the - root subproblem its local bound is initially set to -DBL_MAX - (minimization) or +DBL_MAX (maximization) and then improved as - the root LP relaxation has been solved */ - /* the following two quantities are defined only if LP relaxation - of this subproblem was solved at least once (solved > 0): */ - int ii_cnt; - /* number of integer variables whose value in optimal solution to - LP relaxation of this subproblem is fractional */ - double ii_sum; - /* sum of integer infeasibilities */ -#if 1 /* 30/XI-2009 */ - int changed; - /* how many times this subproblem was re-formulated (by adding - cutting plane constraints) */ -#endif - int br_var; - /* ordinal number of branching variable, 1 <= br_var <= n, used - to split this subproblem; 0 means that either this subproblem - is active or branching was made on a constraint */ - double br_val; - /* (fractional) value of branching variable in optimal solution - to final LP relaxation of this subproblem */ - void *data; /* char data[tree->cb_size]; */ - /* pointer to the application-specific data */ - IOSNPD *temp; - /* working pointer used by some routines */ - IOSNPD *prev; - /* pointer to previous subproblem in the active list */ - IOSNPD *next; - /* pointer to next subproblem in the active list */ -}; - -struct IOSBND -{ /* bounds change entry */ - int k; - /* ordinal number of corresponding row (1 <= k <= m) or column - (m+1 <= k <= m+n), where m and n are the number of rows and - columns, resp., in the parent subproblem */ - unsigned char type; - /* new type */ - double lb; - /* new lower bound */ - double ub; - /* new upper bound */ - IOSBND *next; - /* pointer to next entry for the same subproblem */ -}; - -struct IOSTAT -{ /* status change entry */ - int k; - /* ordinal number of corresponding row (1 <= k <= m) or column - (m+1 <= k <= m+n), where m and n are the number of rows and - columns, resp., in the parent subproblem */ - unsigned char stat; - /* new status */ - IOSTAT *next; - /* pointer to next entry for the same subproblem */ -}; - -struct IOSROW -{ /* row (constraint) addition entry */ - char *name; - /* row name or NULL */ - unsigned char origin; - /* row origin flag (see glp_attr.origin) */ - unsigned char klass; - /* row class descriptor (see glp_attr.klass) */ - unsigned char type; - /* row type (GLP_LO, GLP_UP, etc.) */ - double lb; - /* row lower bound */ - double ub; - /* row upper bound */ - IOSAIJ *ptr; - /* pointer to the row coefficient list */ - double rii; - /* row scale factor */ - unsigned char stat; - /* row status (GLP_BS, GLP_NL, etc.) */ - IOSROW *next; - /* pointer to next entry for the same subproblem */ -}; - -struct IOSAIJ -{ /* constraint coefficient */ - int j; - /* variable (column) number, 1 <= j <= n */ - double val; - /* non-zero coefficient value */ - IOSAIJ *next; - /* pointer to next coefficient for the same row */ -}; - -#ifndef NEW_LOCAL /* 02/II-2018 */ -struct IOSPOOL -{ /* cut pool */ - int size; - /* pool size = number of cuts in the pool */ - IOSCUT *head; - /* pointer to the first cut */ - IOSCUT *tail; - /* pointer to the last cut */ - int ord; - /* ordinal number of the current cut, 1 <= ord <= size */ - IOSCUT *curr; - /* pointer to the current cut */ -}; -#endif - -#ifndef NEW_LOCAL /* 02/II-2018 */ -struct IOSCUT -{ /* cut (cutting plane constraint) */ - char *name; - /* cut name or NULL */ - unsigned char klass; - /* cut class descriptor (see glp_attr.klass) */ - IOSAIJ *ptr; - /* pointer to the cut coefficient list */ - unsigned char type; - /* cut type: - GLP_LO: sum a[j] * x[j] >= b - GLP_UP: sum a[j] * x[j] <= b - GLP_FX: sum a[j] * x[j] = b */ - double rhs; - /* cut right-hand side */ - IOSCUT *prev; - /* pointer to previous cut */ - IOSCUT *next; - /* pointer to next cut */ -}; -#endif - -#define ios_create_tree _glp_ios_create_tree -glp_tree *ios_create_tree(glp_prob *mip, const glp_iocp *parm); -/* create branch-and-bound tree */ - -#define ios_revive_node _glp_ios_revive_node -void ios_revive_node(glp_tree *tree, int p); -/* revive specified subproblem */ - -#define ios_freeze_node _glp_ios_freeze_node -void ios_freeze_node(glp_tree *tree); -/* freeze current subproblem */ - -#define ios_clone_node _glp_ios_clone_node -void ios_clone_node(glp_tree *tree, int p, int nnn, int ref[]); -/* clone specified subproblem */ - -#define ios_delete_node _glp_ios_delete_node -void ios_delete_node(glp_tree *tree, int p); -/* delete specified subproblem */ - -#define ios_delete_tree _glp_ios_delete_tree -void ios_delete_tree(glp_tree *tree); -/* delete branch-and-bound tree */ - -#define ios_eval_degrad _glp_ios_eval_degrad -void ios_eval_degrad(glp_tree *tree, int j, double *dn, double *up); -/* estimate obj. degrad. for down- and up-branches */ - -#define ios_round_bound _glp_ios_round_bound -double ios_round_bound(glp_tree *tree, double bound); -/* improve local bound by rounding */ - -#define ios_is_hopeful _glp_ios_is_hopeful -int ios_is_hopeful(glp_tree *tree, double bound); -/* check if subproblem is hopeful */ - -#define ios_best_node _glp_ios_best_node -int ios_best_node(glp_tree *tree); -/* find active node with best local bound */ - -#define ios_relative_gap _glp_ios_relative_gap -double ios_relative_gap(glp_tree *tree); -/* compute relative mip gap */ - -#define ios_solve_node _glp_ios_solve_node -int ios_solve_node(glp_tree *tree); -/* solve LP relaxation of current subproblem */ - -#define ios_create_pool _glp_ios_create_pool -IOSPOOL *ios_create_pool(glp_tree *tree); -/* create cut pool */ - -#define ios_add_row _glp_ios_add_row -int ios_add_row(glp_tree *tree, IOSPOOL *pool, - const char *name, int klass, int flags, int len, const int ind[], - const double val[], int type, double rhs); -/* add row (constraint) to the cut pool */ - -#define ios_find_row _glp_ios_find_row -IOSCUT *ios_find_row(IOSPOOL *pool, int i); -/* find row (constraint) in the cut pool */ - -#define ios_del_row _glp_ios_del_row -void ios_del_row(glp_tree *tree, IOSPOOL *pool, int i); -/* remove row (constraint) from the cut pool */ - -#define ios_clear_pool _glp_ios_clear_pool -void ios_clear_pool(glp_tree *tree, IOSPOOL *pool); -/* remove all rows (constraints) from the cut pool */ - -#define ios_delete_pool _glp_ios_delete_pool -void ios_delete_pool(glp_tree *tree, IOSPOOL *pool); -/* delete cut pool */ - -#if 1 /* 11/VII-2013 */ -#define ios_process_sol _glp_ios_process_sol -void ios_process_sol(glp_tree *T); -/* process integer feasible solution just found */ -#endif - -#define ios_preprocess_node _glp_ios_preprocess_node -int ios_preprocess_node(glp_tree *tree, int max_pass); -/* preprocess current subproblem */ - -#define ios_driver _glp_ios_driver -int ios_driver(glp_tree *tree); -/* branch-and-bound driver */ - -#define ios_cov_gen _glp_ios_cov_gen -void ios_cov_gen(glp_tree *tree); -/* generate mixed cover cuts */ - -#define ios_pcost_init _glp_ios_pcost_init -void *ios_pcost_init(glp_tree *tree); -/* initialize working data used on pseudocost branching */ - -#define ios_pcost_branch _glp_ios_pcost_branch -int ios_pcost_branch(glp_tree *T, int *next); -/* choose branching variable with pseudocost branching */ - -#define ios_pcost_update _glp_ios_pcost_update -void ios_pcost_update(glp_tree *tree); -/* update history information for pseudocost branching */ - -#define ios_pcost_free _glp_ios_pcost_free -void ios_pcost_free(glp_tree *tree); -/* free working area used on pseudocost branching */ - -#define ios_feas_pump _glp_ios_feas_pump -void ios_feas_pump(glp_tree *T); -/* feasibility pump heuristic */ - -#if 1 /* 25/V-2013 */ -#define ios_proxy_heur _glp_ios_proxy_heur -void ios_proxy_heur(glp_tree *T); -/* proximity search heuristic */ -#endif - -#define ios_process_cuts _glp_ios_process_cuts -void ios_process_cuts(glp_tree *T); -/* process cuts stored in the local cut pool */ - -#define ios_choose_node _glp_ios_choose_node -int ios_choose_node(glp_tree *T); -/* select subproblem to continue the search */ - -#define ios_choose_var _glp_ios_choose_var -int ios_choose_var(glp_tree *T, int *next); -/* select variable to branch on */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/draft/lux.c b/code/3rd_glpk/draft/lux.c deleted file mode 100644 index 38cb758c..00000000 --- a/code/3rd_glpk/draft/lux.c +++ /dev/null @@ -1,1030 +0,0 @@ -/* lux.c (LU-factorization, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "lux.h" - -#define xfault xerror -#define dmp_create_poolx(size) dmp_create_pool() - -/*********************************************************************** -* lux_create - create LU-factorization -* -* SYNOPSIS -* -* #include "lux.h" -* LUX *lux_create(int n); -* -* DESCRIPTION -* -* The routine lux_create creates LU-factorization data structure for -* a matrix of the order n. Initially the factorization corresponds to -* the unity matrix (F = V = P = Q = I, so A = I). -* -* RETURNS -* -* The routine returns a pointer to the created LU-factorization data -* structure, which represents the unity matrix of the order n. */ - -LUX *lux_create(int n) -{ LUX *lux; - int k; - if (n < 1) - xfault("lux_create: n = %d; invalid parameter\n", n); - lux = xmalloc(sizeof(LUX)); - lux->n = n; - lux->pool = dmp_create_poolx(sizeof(LUXELM)); - lux->F_row = xcalloc(1+n, sizeof(LUXELM *)); - lux->F_col = xcalloc(1+n, sizeof(LUXELM *)); - lux->V_piv = xcalloc(1+n, sizeof(mpq_t)); - lux->V_row = xcalloc(1+n, sizeof(LUXELM *)); - lux->V_col = xcalloc(1+n, sizeof(LUXELM *)); - lux->P_row = xcalloc(1+n, sizeof(int)); - lux->P_col = xcalloc(1+n, sizeof(int)); - lux->Q_row = xcalloc(1+n, sizeof(int)); - lux->Q_col = xcalloc(1+n, sizeof(int)); - for (k = 1; k <= n; k++) - { lux->F_row[k] = lux->F_col[k] = NULL; - mpq_init(lux->V_piv[k]); - mpq_set_si(lux->V_piv[k], 1, 1); - lux->V_row[k] = lux->V_col[k] = NULL; - lux->P_row[k] = lux->P_col[k] = k; - lux->Q_row[k] = lux->Q_col[k] = k; - } - lux->rank = n; - return lux; -} - -/*********************************************************************** -* initialize - initialize LU-factorization data structures -* -* This routine initializes data structures for subsequent computing -* the LU-factorization of a given matrix A, which is specified by the -* formal routine col. On exit V = A and F = P = Q = I, where I is the -* unity matrix. */ - -static void initialize(LUX *lux, int (*col)(void *info, int j, - int ind[], mpq_t val[]), void *info, LUXWKA *wka) -{ int n = lux->n; - DMP *pool = lux->pool; - LUXELM **F_row = lux->F_row; - LUXELM **F_col = lux->F_col; - mpq_t *V_piv = lux->V_piv; - LUXELM **V_row = lux->V_row; - LUXELM **V_col = lux->V_col; - int *P_row = lux->P_row; - int *P_col = lux->P_col; - int *Q_row = lux->Q_row; - int *Q_col = lux->Q_col; - int *R_len = wka->R_len; - int *R_head = wka->R_head; - int *R_prev = wka->R_prev; - int *R_next = wka->R_next; - int *C_len = wka->C_len; - int *C_head = wka->C_head; - int *C_prev = wka->C_prev; - int *C_next = wka->C_next; - LUXELM *fij, *vij; - int i, j, k, len, *ind; - mpq_t *val; - /* F := I */ - for (i = 1; i <= n; i++) - { while (F_row[i] != NULL) - { fij = F_row[i], F_row[i] = fij->r_next; - mpq_clear(fij->val); - dmp_free_atom(pool, fij, sizeof(LUXELM)); - } - } - for (j = 1; j <= n; j++) F_col[j] = NULL; - /* V := 0 */ - for (k = 1; k <= n; k++) mpq_set_si(V_piv[k], 0, 1); - for (i = 1; i <= n; i++) - { while (V_row[i] != NULL) - { vij = V_row[i], V_row[i] = vij->r_next; - mpq_clear(vij->val); - dmp_free_atom(pool, vij, sizeof(LUXELM)); - } - } - for (j = 1; j <= n; j++) V_col[j] = NULL; - /* V := A */ - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(mpq_t)); - for (k = 1; k <= n; k++) mpq_init(val[k]); - for (j = 1; j <= n; j++) - { /* obtain j-th column of matrix A */ - len = col(info, j, ind, val); - if (!(0 <= len && len <= n)) - xfault("lux_decomp: j = %d: len = %d; invalid column length" - "\n", j, len); - /* copy elements of j-th column to matrix V */ - for (k = 1; k <= len; k++) - { /* get row index of a[i,j] */ - i = ind[k]; - if (!(1 <= i && i <= n)) - xfault("lux_decomp: j = %d: i = %d; row index out of ran" - "ge\n", j, i); - /* check for duplicate indices */ - if (V_row[i] != NULL && V_row[i]->j == j) - xfault("lux_decomp: j = %d: i = %d; duplicate row indice" - "s not allowed\n", j, i); - /* check for zero value */ - if (mpq_sgn(val[k]) == 0) - xfault("lux_decomp: j = %d: i = %d; zero elements not al" - "lowed\n", j, i); - /* add new element v[i,j] = a[i,j] to V */ - vij = dmp_get_atom(pool, sizeof(LUXELM)); - vij->i = i, vij->j = j; - mpq_init(vij->val); - mpq_set(vij->val, val[k]); - vij->r_prev = NULL; - vij->r_next = V_row[i]; - vij->c_prev = NULL; - vij->c_next = V_col[j]; - if (vij->r_next != NULL) vij->r_next->r_prev = vij; - if (vij->c_next != NULL) vij->c_next->c_prev = vij; - V_row[i] = V_col[j] = vij; - } - } - xfree(ind); - for (k = 1; k <= n; k++) mpq_clear(val[k]); - xfree(val); - /* P := Q := I */ - for (k = 1; k <= n; k++) - P_row[k] = P_col[k] = Q_row[k] = Q_col[k] = k; - /* the rank of A and V is not determined yet */ - lux->rank = -1; - /* initially the entire matrix V is active */ - /* determine its row lengths */ - for (i = 1; i <= n; i++) - { len = 0; - for (vij = V_row[i]; vij != NULL; vij = vij->r_next) len++; - R_len[i] = len; - } - /* build linked lists of active rows */ - for (len = 0; len <= n; len++) R_head[len] = 0; - for (i = 1; i <= n; i++) - { len = R_len[i]; - R_prev[i] = 0; - R_next[i] = R_head[len]; - if (R_next[i] != 0) R_prev[R_next[i]] = i; - R_head[len] = i; - } - /* determine its column lengths */ - for (j = 1; j <= n; j++) - { len = 0; - for (vij = V_col[j]; vij != NULL; vij = vij->c_next) len++; - C_len[j] = len; - } - /* build linked lists of active columns */ - for (len = 0; len <= n; len++) C_head[len] = 0; - for (j = 1; j <= n; j++) - { len = C_len[j]; - C_prev[j] = 0; - C_next[j] = C_head[len]; - if (C_next[j] != 0) C_prev[C_next[j]] = j; - C_head[len] = j; - } - return; -} - -/*********************************************************************** -* find_pivot - choose a pivot element -* -* This routine chooses a pivot element v[p,q] in the active submatrix -* of matrix U = P*V*Q. -* -* It is assumed that on entry the matrix U has the following partially -* triangularized form: -* -* 1 k n -* 1 x x x x x x x x x x -* . x x x x x x x x x -* . . x x x x x x x x -* . . . x x x x x x x -* k . . . . * * * * * * -* . . . . * * * * * * -* . . . . * * * * * * -* . . . . * * * * * * -* . . . . * * * * * * -* n . . . . * * * * * * -* -* where rows and columns k, k+1, ..., n belong to the active submatrix -* (elements of the active submatrix are marked by '*'). -* -* Since the matrix U = P*V*Q is not stored, the routine works with the -* matrix V. It is assumed that the row-wise representation corresponds -* to the matrix V, but the column-wise representation corresponds to -* the active submatrix of the matrix V, i.e. elements of the matrix V, -* which does not belong to the active submatrix, are missing from the -* column linked lists. It is also assumed that each active row of the -* matrix V is in the set R[len], where len is number of non-zeros in -* the row, and each active column of the matrix V is in the set C[len], -* where len is number of non-zeros in the column (in the latter case -* only elements of the active submatrix are counted; such elements are -* marked by '*' on the figure above). -* -* Due to exact arithmetic any non-zero element of the active submatrix -* can be chosen as a pivot. However, to keep sparsity of the matrix V -* the routine uses Markowitz strategy, trying to choose such element -* v[p,q], which has smallest Markowitz cost (nr[p]-1) * (nc[q]-1), -* where nr[p] and nc[q] are the number of non-zero elements, resp., in -* p-th row and in q-th column of the active submatrix. -* -* In order to reduce the search, i.e. not to walk through all elements -* of the active submatrix, the routine exploits a technique proposed by -* I.Duff. This technique is based on using the sets R[len] and C[len] -* of active rows and columns. -* -* On exit the routine returns a pointer to a pivot v[p,q] chosen, or -* NULL, if the active submatrix is empty. */ - -static LUXELM *find_pivot(LUX *lux, LUXWKA *wka) -{ int n = lux->n; - LUXELM **V_row = lux->V_row; - LUXELM **V_col = lux->V_col; - int *R_len = wka->R_len; - int *R_head = wka->R_head; - int *R_next = wka->R_next; - int *C_len = wka->C_len; - int *C_head = wka->C_head; - int *C_next = wka->C_next; - LUXELM *piv, *some, *vij; - int i, j, len, min_len, ncand, piv_lim = 5; - double best, cost; - /* nothing is chosen so far */ - piv = NULL, best = DBL_MAX, ncand = 0; - /* if in the active submatrix there is a column that has the only - non-zero (column singleton), choose it as a pivot */ - j = C_head[1]; - if (j != 0) - { xassert(C_len[j] == 1); - piv = V_col[j]; - xassert(piv != NULL && piv->c_next == NULL); - goto done; - } - /* if in the active submatrix there is a row that has the only - non-zero (row singleton), choose it as a pivot */ - i = R_head[1]; - if (i != 0) - { xassert(R_len[i] == 1); - piv = V_row[i]; - xassert(piv != NULL && piv->r_next == NULL); - goto done; - } - /* there are no singletons in the active submatrix; walk through - other non-empty rows and columns */ - for (len = 2; len <= n; len++) - { /* consider active columns having len non-zeros */ - for (j = C_head[len]; j != 0; j = C_next[j]) - { /* j-th column has len non-zeros */ - /* find an element in the row of minimal length */ - some = NULL, min_len = INT_MAX; - for (vij = V_col[j]; vij != NULL; vij = vij->c_next) - { if (min_len > R_len[vij->i]) - some = vij, min_len = R_len[vij->i]; - /* if Markowitz cost of this element is not greater than - (len-1)**2, it can be chosen right now; this heuristic - reduces the search and works well in many cases */ - if (min_len <= len) - { piv = some; - goto done; - } - } - /* j-th column has been scanned */ - /* the minimal element found is a next pivot candidate */ - xassert(some != NULL); - ncand++; - /* compute its Markowitz cost */ - cost = (double)(min_len - 1) * (double)(len - 1); - /* choose between the current candidate and this element */ - if (cost < best) piv = some, best = cost; - /* if piv_lim candidates have been considered, there is a - doubt that a much better candidate exists; therefore it - is the time to terminate the search */ - if (ncand == piv_lim) goto done; - } - /* now consider active rows having len non-zeros */ - for (i = R_head[len]; i != 0; i = R_next[i]) - { /* i-th row has len non-zeros */ - /* find an element in the column of minimal length */ - some = NULL, min_len = INT_MAX; - for (vij = V_row[i]; vij != NULL; vij = vij->r_next) - { if (min_len > C_len[vij->j]) - some = vij, min_len = C_len[vij->j]; - /* if Markowitz cost of this element is not greater than - (len-1)**2, it can be chosen right now; this heuristic - reduces the search and works well in many cases */ - if (min_len <= len) - { piv = some; - goto done; - } - } - /* i-th row has been scanned */ - /* the minimal element found is a next pivot candidate */ - xassert(some != NULL); - ncand++; - /* compute its Markowitz cost */ - cost = (double)(len - 1) * (double)(min_len - 1); - /* choose between the current candidate and this element */ - if (cost < best) piv = some, best = cost; - /* if piv_lim candidates have been considered, there is a - doubt that a much better candidate exists; therefore it - is the time to terminate the search */ - if (ncand == piv_lim) goto done; - } - } -done: /* bring the pivot v[p,q] to the factorizing routine */ - return piv; -} - -/*********************************************************************** -* eliminate - perform gaussian elimination -* -* This routine performs elementary gaussian transformations in order -* to eliminate subdiagonal elements in the k-th column of the matrix -* U = P*V*Q using the pivot element u[k,k], where k is the number of -* the current elimination step. -* -* The parameter piv specifies the pivot element v[p,q] = u[k,k]. -* -* Each time when the routine applies the elementary transformation to -* a non-pivot row of the matrix V, it stores the corresponding element -* to the matrix F in order to keep the main equality A = F*V. -* -* The routine assumes that on entry the matrices L = P*F*inv(P) and -* U = P*V*Q are the following: -* -* 1 k 1 k n -* 1 1 . . . . . . . . . 1 x x x x x x x x x x -* x 1 . . . . . . . . . x x x x x x x x x -* x x 1 . . . . . . . . . x x x x x x x x -* x x x 1 . . . . . . . . . x x x x x x x -* k x x x x 1 . . . . . k . . . . * * * * * * -* x x x x _ 1 . . . . . . . . # * * * * * -* x x x x _ . 1 . . . . . . . # * * * * * -* x x x x _ . . 1 . . . . . . # * * * * * -* x x x x _ . . . 1 . . . . . # * * * * * -* n x x x x _ . . . . 1 n . . . . # * * * * * -* -* matrix L matrix U -* -* where rows and columns of the matrix U with numbers k, k+1, ..., n -* form the active submatrix (eliminated elements are marked by '#' and -* other elements of the active submatrix are marked by '*'). Note that -* each eliminated non-zero element u[i,k] of the matrix U gives the -* corresponding element l[i,k] of the matrix L (marked by '_'). -* -* Actually all operations are performed on the matrix V. Should note -* that the row-wise representation corresponds to the matrix V, but the -* column-wise representation corresponds to the active submatrix of the -* matrix V, i.e. elements of the matrix V, which doesn't belong to the -* active submatrix, are missing from the column linked lists. -* -* Let u[k,k] = v[p,q] be the pivot. In order to eliminate subdiagonal -* elements u[i',k] = v[i,q], i' = k+1, k+2, ..., n, the routine applies -* the following elementary gaussian transformations: -* -* (i-th row of V) := (i-th row of V) - f[i,p] * (p-th row of V), -* -* where f[i,p] = v[i,q] / v[p,q] is a gaussian multiplier. -* -* Additionally, in order to keep the main equality A = F*V, each time -* when the routine applies the transformation to i-th row of the matrix -* V, it also adds f[i,p] as a new element to the matrix F. -* -* IMPORTANT: On entry the working arrays flag and work should contain -* zeros. This status is provided by the routine on exit. */ - -static void eliminate(LUX *lux, LUXWKA *wka, LUXELM *piv, int flag[], - mpq_t work[]) -{ DMP *pool = lux->pool; - LUXELM **F_row = lux->F_row; - LUXELM **F_col = lux->F_col; - mpq_t *V_piv = lux->V_piv; - LUXELM **V_row = lux->V_row; - LUXELM **V_col = lux->V_col; - int *R_len = wka->R_len; - int *R_head = wka->R_head; - int *R_prev = wka->R_prev; - int *R_next = wka->R_next; - int *C_len = wka->C_len; - int *C_head = wka->C_head; - int *C_prev = wka->C_prev; - int *C_next = wka->C_next; - LUXELM *fip, *vij, *vpj, *viq, *next; - mpq_t temp; - int i, j, p, q; - mpq_init(temp); - /* determine row and column indices of the pivot v[p,q] */ - xassert(piv != NULL); - p = piv->i, q = piv->j; - /* remove p-th (pivot) row from the active set; it will never - return there */ - if (R_prev[p] == 0) - R_head[R_len[p]] = R_next[p]; - else - R_next[R_prev[p]] = R_next[p]; - if (R_next[p] == 0) - ; - else - R_prev[R_next[p]] = R_prev[p]; - /* remove q-th (pivot) column from the active set; it will never - return there */ - if (C_prev[q] == 0) - C_head[C_len[q]] = C_next[q]; - else - C_next[C_prev[q]] = C_next[q]; - if (C_next[q] == 0) - ; - else - C_prev[C_next[q]] = C_prev[q]; - /* store the pivot value in a separate array */ - mpq_set(V_piv[p], piv->val); - /* remove the pivot from p-th row */ - if (piv->r_prev == NULL) - V_row[p] = piv->r_next; - else - piv->r_prev->r_next = piv->r_next; - if (piv->r_next == NULL) - ; - else - piv->r_next->r_prev = piv->r_prev; - R_len[p]--; - /* remove the pivot from q-th column */ - if (piv->c_prev == NULL) - V_col[q] = piv->c_next; - else - piv->c_prev->c_next = piv->c_next; - if (piv->c_next == NULL) - ; - else - piv->c_next->c_prev = piv->c_prev; - C_len[q]--; - /* free the space occupied by the pivot */ - mpq_clear(piv->val); - dmp_free_atom(pool, piv, sizeof(LUXELM)); - /* walk through p-th (pivot) row, which already does not contain - the pivot v[p,q], and do the following... */ - for (vpj = V_row[p]; vpj != NULL; vpj = vpj->r_next) - { /* get column index of v[p,j] */ - j = vpj->j; - /* store v[p,j] in the working array */ - flag[j] = 1; - mpq_set(work[j], vpj->val); - /* remove j-th column from the active set; it will return there - later with a new length */ - if (C_prev[j] == 0) - C_head[C_len[j]] = C_next[j]; - else - C_next[C_prev[j]] = C_next[j]; - if (C_next[j] == 0) - ; - else - C_prev[C_next[j]] = C_prev[j]; - /* v[p,j] leaves the active submatrix, so remove it from j-th - column; however, v[p,j] is kept in p-th row */ - if (vpj->c_prev == NULL) - V_col[j] = vpj->c_next; - else - vpj->c_prev->c_next = vpj->c_next; - if (vpj->c_next == NULL) - ; - else - vpj->c_next->c_prev = vpj->c_prev; - C_len[j]--; - } - /* now walk through q-th (pivot) column, which already does not - contain the pivot v[p,q], and perform gaussian elimination */ - while (V_col[q] != NULL) - { /* element v[i,q] has to be eliminated */ - viq = V_col[q]; - /* get row index of v[i,q] */ - i = viq->i; - /* remove i-th row from the active set; later it will return - there with a new length */ - if (R_prev[i] == 0) - R_head[R_len[i]] = R_next[i]; - else - R_next[R_prev[i]] = R_next[i]; - if (R_next[i] == 0) - ; - else - R_prev[R_next[i]] = R_prev[i]; - /* compute gaussian multiplier f[i,p] = v[i,q] / v[p,q] and - store it in the matrix F */ - fip = dmp_get_atom(pool, sizeof(LUXELM)); - fip->i = i, fip->j = p; - mpq_init(fip->val); - mpq_div(fip->val, viq->val, V_piv[p]); - fip->r_prev = NULL; - fip->r_next = F_row[i]; - fip->c_prev = NULL; - fip->c_next = F_col[p]; - if (fip->r_next != NULL) fip->r_next->r_prev = fip; - if (fip->c_next != NULL) fip->c_next->c_prev = fip; - F_row[i] = F_col[p] = fip; - /* v[i,q] has to be eliminated, so remove it from i-th row */ - if (viq->r_prev == NULL) - V_row[i] = viq->r_next; - else - viq->r_prev->r_next = viq->r_next; - if (viq->r_next == NULL) - ; - else - viq->r_next->r_prev = viq->r_prev; - R_len[i]--; - /* and also from q-th column */ - V_col[q] = viq->c_next; - C_len[q]--; - /* free the space occupied by v[i,q] */ - mpq_clear(viq->val); - dmp_free_atom(pool, viq, sizeof(LUXELM)); - /* perform gaussian transformation: - (i-th row) := (i-th row) - f[i,p] * (p-th row) - note that now p-th row, which is in the working array, - does not contain the pivot v[p,q], and i-th row does not - contain the element v[i,q] to be eliminated */ - /* walk through i-th row and transform existing non-zero - elements */ - for (vij = V_row[i]; vij != NULL; vij = next) - { next = vij->r_next; - /* get column index of v[i,j] */ - j = vij->j; - /* v[i,j] := v[i,j] - f[i,p] * v[p,j] */ - if (flag[j]) - { /* v[p,j] != 0 */ - flag[j] = 0; - mpq_mul(temp, fip->val, work[j]); - mpq_sub(vij->val, vij->val, temp); - if (mpq_sgn(vij->val) == 0) - { /* new v[i,j] is zero, so remove it from the active - submatrix */ - /* remove v[i,j] from i-th row */ - if (vij->r_prev == NULL) - V_row[i] = vij->r_next; - else - vij->r_prev->r_next = vij->r_next; - if (vij->r_next == NULL) - ; - else - vij->r_next->r_prev = vij->r_prev; - R_len[i]--; - /* remove v[i,j] from j-th column */ - if (vij->c_prev == NULL) - V_col[j] = vij->c_next; - else - vij->c_prev->c_next = vij->c_next; - if (vij->c_next == NULL) - ; - else - vij->c_next->c_prev = vij->c_prev; - C_len[j]--; - /* free the space occupied by v[i,j] */ - mpq_clear(vij->val); - dmp_free_atom(pool, vij, sizeof(LUXELM)); - } - } - } - /* now flag is the pattern of the set v[p,*] \ v[i,*] */ - /* walk through p-th (pivot) row and create new elements in - i-th row, which appear due to fill-in */ - for (vpj = V_row[p]; vpj != NULL; vpj = vpj->r_next) - { j = vpj->j; - if (flag[j]) - { /* create new non-zero v[i,j] = 0 - f[i,p] * v[p,j] and - add it to i-th row and j-th column */ - vij = dmp_get_atom(pool, sizeof(LUXELM)); - vij->i = i, vij->j = j; - mpq_init(vij->val); - mpq_mul(vij->val, fip->val, work[j]); - mpq_neg(vij->val, vij->val); - vij->r_prev = NULL; - vij->r_next = V_row[i]; - vij->c_prev = NULL; - vij->c_next = V_col[j]; - if (vij->r_next != NULL) vij->r_next->r_prev = vij; - if (vij->c_next != NULL) vij->c_next->c_prev = vij; - V_row[i] = V_col[j] = vij; - R_len[i]++, C_len[j]++; - } - else - { /* there is no fill-in, because v[i,j] already exists in - i-th row; restore the flag, which was reset before */ - flag[j] = 1; - } - } - /* now i-th row has been completely transformed and can return - to the active set with a new length */ - R_prev[i] = 0; - R_next[i] = R_head[R_len[i]]; - if (R_next[i] != 0) R_prev[R_next[i]] = i; - R_head[R_len[i]] = i; - } - /* at this point q-th (pivot) column must be empty */ - xassert(C_len[q] == 0); - /* walk through p-th (pivot) row again and do the following... */ - for (vpj = V_row[p]; vpj != NULL; vpj = vpj->r_next) - { /* get column index of v[p,j] */ - j = vpj->j; - /* erase v[p,j] from the working array */ - flag[j] = 0; - mpq_set_si(work[j], 0, 1); - /* now j-th column has been completely transformed, so it can - return to the active list with a new length */ - C_prev[j] = 0; - C_next[j] = C_head[C_len[j]]; - if (C_next[j] != 0) C_prev[C_next[j]] = j; - C_head[C_len[j]] = j; - } - mpq_clear(temp); - /* return to the factorizing routine */ - return; -} - -/*********************************************************************** -* lux_decomp - compute LU-factorization -* -* SYNOPSIS -* -* #include "lux.h" -* int lux_decomp(LUX *lux, int (*col)(void *info, int j, int ind[], -* mpq_t val[]), void *info); -* -* DESCRIPTION -* -* The routine lux_decomp computes LU-factorization of a given square -* matrix A. -* -* The parameter lux specifies LU-factorization data structure built by -* means of the routine lux_create. -* -* The formal routine col specifies the original matrix A. In order to -* obtain j-th column of the matrix A the routine lux_decomp calls the -* routine col with the parameter j (1 <= j <= n, where n is the order -* of A). In response the routine col should store row indices and -* numerical values of non-zero elements of j-th column of A to the -* locations ind[1], ..., ind[len] and val[1], ..., val[len], resp., -* where len is the number of non-zeros in j-th column, which should be -* returned on exit. Neiter zero nor duplicate elements are allowed. -* -* The parameter info is a transit pointer passed to the formal routine -* col; it can be used for various purposes. -* -* RETURNS -* -* The routine lux_decomp returns the singularity flag. Zero flag means -* that the original matrix A is non-singular while non-zero flag means -* that A is (exactly!) singular. -* -* Note that LU-factorization is valid in both cases, however, in case -* of singularity some rows of the matrix V (including pivot elements) -* will be empty. -* -* REPAIRING SINGULAR MATRIX -* -* If the routine lux_decomp returns non-zero flag, it provides all -* necessary information that can be used for "repairing" the matrix A, -* where "repairing" means replacing linearly dependent columns of the -* matrix A by appropriate columns of the unity matrix. This feature is -* needed when the routine lux_decomp is used for reinverting the basis -* matrix within the simplex method procedure. -* -* On exit linearly dependent columns of the matrix U have the numbers -* rank+1, rank+2, ..., n, where rank is the exact rank of the matrix A -* stored by the routine to the member lux->rank. The correspondence -* between columns of A and U is the same as between columns of V and U. -* Thus, linearly dependent columns of the matrix A have the numbers -* Q_col[rank+1], Q_col[rank+2], ..., Q_col[n], where Q_col is an array -* representing the permutation matrix Q in column-like format. It is -* understood that each j-th linearly dependent column of the matrix U -* should be replaced by the unity vector, where all elements are zero -* except the unity diagonal element u[j,j]. On the other hand j-th row -* of the matrix U corresponds to the row of the matrix V (and therefore -* of the matrix A) with the number P_row[j], where P_row is an array -* representing the permutation matrix P in row-like format. Thus, each -* j-th linearly dependent column of the matrix U should be replaced by -* a column of the unity matrix with the number P_row[j]. -* -* The code that repairs the matrix A may look like follows: -* -* for (j = rank+1; j <= n; j++) -* { replace column Q_col[j] of the matrix A by column P_row[j] of -* the unity matrix; -* } -* -* where rank, P_row, and Q_col are members of the structure LUX. */ - -int lux_decomp(LUX *lux, int (*col)(void *info, int j, int ind[], - mpq_t val[]), void *info) -{ int n = lux->n; - LUXELM **V_row = lux->V_row; - LUXELM **V_col = lux->V_col; - int *P_row = lux->P_row; - int *P_col = lux->P_col; - int *Q_row = lux->Q_row; - int *Q_col = lux->Q_col; - LUXELM *piv, *vij; - LUXWKA *wka; - int i, j, k, p, q, t, *flag; - mpq_t *work; - /* allocate working area */ - wka = xmalloc(sizeof(LUXWKA)); - wka->R_len = xcalloc(1+n, sizeof(int)); - wka->R_head = xcalloc(1+n, sizeof(int)); - wka->R_prev = xcalloc(1+n, sizeof(int)); - wka->R_next = xcalloc(1+n, sizeof(int)); - wka->C_len = xcalloc(1+n, sizeof(int)); - wka->C_head = xcalloc(1+n, sizeof(int)); - wka->C_prev = xcalloc(1+n, sizeof(int)); - wka->C_next = xcalloc(1+n, sizeof(int)); - /* initialize LU-factorization data structures */ - initialize(lux, col, info, wka); - /* allocate working arrays */ - flag = xcalloc(1+n, sizeof(int)); - work = xcalloc(1+n, sizeof(mpq_t)); - for (k = 1; k <= n; k++) - { flag[k] = 0; - mpq_init(work[k]); - } - /* main elimination loop */ - for (k = 1; k <= n; k++) - { /* choose a pivot element v[p,q] */ - piv = find_pivot(lux, wka); - if (piv == NULL) - { /* no pivot can be chosen, because the active submatrix is - empty */ - break; - } - /* determine row and column indices of the pivot element */ - p = piv->i, q = piv->j; - /* let v[p,q] correspond to u[i',j']; permute k-th and i'-th - rows and k-th and j'-th columns of the matrix U = P*V*Q to - move the element u[i',j'] to the position u[k,k] */ - i = P_col[p], j = Q_row[q]; - xassert(k <= i && i <= n && k <= j && j <= n); - /* permute k-th and i-th rows of the matrix U */ - t = P_row[k]; - P_row[i] = t, P_col[t] = i; - P_row[k] = p, P_col[p] = k; - /* permute k-th and j-th columns of the matrix U */ - t = Q_col[k]; - Q_col[j] = t, Q_row[t] = j; - Q_col[k] = q, Q_row[q] = k; - /* eliminate subdiagonal elements of k-th column of the matrix - U = P*V*Q using the pivot element u[k,k] = v[p,q] */ - eliminate(lux, wka, piv, flag, work); - } - /* determine the rank of A (and V) */ - lux->rank = k - 1; - /* free working arrays */ - xfree(flag); - for (k = 1; k <= n; k++) mpq_clear(work[k]); - xfree(work); - /* build column lists of the matrix V using its row lists */ - for (j = 1; j <= n; j++) - xassert(V_col[j] == NULL); - for (i = 1; i <= n; i++) - { for (vij = V_row[i]; vij != NULL; vij = vij->r_next) - { j = vij->j; - vij->c_prev = NULL; - vij->c_next = V_col[j]; - if (vij->c_next != NULL) vij->c_next->c_prev = vij; - V_col[j] = vij; - } - } - /* free working area */ - xfree(wka->R_len); - xfree(wka->R_head); - xfree(wka->R_prev); - xfree(wka->R_next); - xfree(wka->C_len); - xfree(wka->C_head); - xfree(wka->C_prev); - xfree(wka->C_next); - xfree(wka); - /* return to the calling program */ - return (lux->rank < n); -} - -/*********************************************************************** -* lux_f_solve - solve system F*x = b or F'*x = b -* -* SYNOPSIS -* -* #include "lux.h" -* void lux_f_solve(LUX *lux, int tr, mpq_t x[]); -* -* DESCRIPTION -* -* The routine lux_f_solve solves either the system F*x = b (if the -* flag tr is zero) or the system F'*x = b (if the flag tr is non-zero), -* where the matrix F is a component of LU-factorization specified by -* the parameter lux, F' is a matrix transposed to F. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix F. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void lux_f_solve(LUX *lux, int tr, mpq_t x[]) -{ int n = lux->n; - LUXELM **F_row = lux->F_row; - LUXELM **F_col = lux->F_col; - int *P_row = lux->P_row; - LUXELM *fik, *fkj; - int i, j, k; - mpq_t temp; - mpq_init(temp); - if (!tr) - { /* solve the system F*x = b */ - for (j = 1; j <= n; j++) - { k = P_row[j]; - if (mpq_sgn(x[k]) != 0) - { for (fik = F_col[k]; fik != NULL; fik = fik->c_next) - { mpq_mul(temp, fik->val, x[k]); - mpq_sub(x[fik->i], x[fik->i], temp); - } - } - } - } - else - { /* solve the system F'*x = b */ - for (i = n; i >= 1; i--) - { k = P_row[i]; - if (mpq_sgn(x[k]) != 0) - { for (fkj = F_row[k]; fkj != NULL; fkj = fkj->r_next) - { mpq_mul(temp, fkj->val, x[k]); - mpq_sub(x[fkj->j], x[fkj->j], temp); - } - } - } - } - mpq_clear(temp); - return; -} - -/*********************************************************************** -* lux_v_solve - solve system V*x = b or V'*x = b -* -* SYNOPSIS -* -* #include "lux.h" -* void lux_v_solve(LUX *lux, int tr, double x[]); -* -* DESCRIPTION -* -* The routine lux_v_solve solves either the system V*x = b (if the -* flag tr is zero) or the system V'*x = b (if the flag tr is non-zero), -* where the matrix V is a component of LU-factorization specified by -* the parameter lux, V' is a matrix transposed to V. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix V. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void lux_v_solve(LUX *lux, int tr, mpq_t x[]) -{ int n = lux->n; - mpq_t *V_piv = lux->V_piv; - LUXELM **V_row = lux->V_row; - LUXELM **V_col = lux->V_col; - int *P_row = lux->P_row; - int *Q_col = lux->Q_col; - LUXELM *vij; - int i, j, k; - mpq_t *b, temp; - b = xcalloc(1+n, sizeof(mpq_t)); - for (k = 1; k <= n; k++) - mpq_init(b[k]), mpq_set(b[k], x[k]), mpq_set_si(x[k], 0, 1); - mpq_init(temp); - if (!tr) - { /* solve the system V*x = b */ - for (k = n; k >= 1; k--) - { i = P_row[k], j = Q_col[k]; - if (mpq_sgn(b[i]) != 0) - { mpq_set(x[j], b[i]); - mpq_div(x[j], x[j], V_piv[i]); - for (vij = V_col[j]; vij != NULL; vij = vij->c_next) - { mpq_mul(temp, vij->val, x[j]); - mpq_sub(b[vij->i], b[vij->i], temp); - } - } - } - } - else - { /* solve the system V'*x = b */ - for (k = 1; k <= n; k++) - { i = P_row[k], j = Q_col[k]; - if (mpq_sgn(b[j]) != 0) - { mpq_set(x[i], b[j]); - mpq_div(x[i], x[i], V_piv[i]); - for (vij = V_row[i]; vij != NULL; vij = vij->r_next) - { mpq_mul(temp, vij->val, x[i]); - mpq_sub(b[vij->j], b[vij->j], temp); - } - } - } - } - for (k = 1; k <= n; k++) mpq_clear(b[k]); - mpq_clear(temp); - xfree(b); - return; -} - -/*********************************************************************** -* lux_solve - solve system A*x = b or A'*x = b -* -* SYNOPSIS -* -* #include "lux.h" -* void lux_solve(LUX *lux, int tr, mpq_t x[]); -* -* DESCRIPTION -* -* The routine lux_solve solves either the system A*x = b (if the flag -* tr is zero) or the system A'*x = b (if the flag tr is non-zero), -* where the parameter lux specifies LU-factorization of the matrix A, -* A' is a matrix transposed to A. -* -* On entry the array x should contain elements of the right-hand side -* vector b in locations x[1], ..., x[n], where n is the order of the -* matrix A. On exit this array will contain elements of the solution -* vector x in the same locations. */ - -void lux_solve(LUX *lux, int tr, mpq_t x[]) -{ if (lux->rank < lux->n) - xfault("lux_solve: LU-factorization has incomplete rank\n"); - if (!tr) - { /* A = F*V, therefore inv(A) = inv(V)*inv(F) */ - lux_f_solve(lux, 0, x); - lux_v_solve(lux, 0, x); - } - else - { /* A' = V'*F', therefore inv(A') = inv(F')*inv(V') */ - lux_v_solve(lux, 1, x); - lux_f_solve(lux, 1, x); - } - return; -} - -/*********************************************************************** -* lux_delete - delete LU-factorization -* -* SYNOPSIS -* -* #include "lux.h" -* void lux_delete(LUX *lux); -* -* DESCRIPTION -* -* The routine lux_delete deletes LU-factorization data structure, -* which the parameter lux points to, freeing all the memory allocated -* to this object. */ - -void lux_delete(LUX *lux) -{ int n = lux->n; - LUXELM *fij, *vij; - int i; - for (i = 1; i <= n; i++) - { for (fij = lux->F_row[i]; fij != NULL; fij = fij->r_next) - mpq_clear(fij->val); - mpq_clear(lux->V_piv[i]); - for (vij = lux->V_row[i]; vij != NULL; vij = vij->r_next) - mpq_clear(vij->val); - } - dmp_delete_pool(lux->pool); - xfree(lux->F_row); - xfree(lux->F_col); - xfree(lux->V_piv); - xfree(lux->V_row); - xfree(lux->V_col); - xfree(lux->P_row); - xfree(lux->P_col); - xfree(lux->Q_row); - xfree(lux->Q_col); - xfree(lux); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/draft/lux.h b/code/3rd_glpk/draft/lux.h deleted file mode 100644 index 8767bb8e..00000000 --- a/code/3rd_glpk/draft/lux.h +++ /dev/null @@ -1,220 +0,0 @@ -/* lux.h (LU-factorization, rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, -* 2009, 2010, 2011, 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef LUX_H -#define LUX_H - -#include "dmp.h" -#include "mygmp.h" - -/*********************************************************************** -* The structure LUX defines LU-factorization of a square matrix A, -* which is the following quartet: -* -* [A] = (F, V, P, Q), (1) -* -* where F and V are such matrices that -* -* A = F * V, (2) -* -* and P and Q are such permutation matrices that the matrix -* -* L = P * F * inv(P) (3) -* -* is lower triangular with unity diagonal, and the matrix -* -* U = P * V * Q (4) -* -* is upper triangular. All the matrices have the order n. -* -* The matrices F and V are stored in row/column-wise sparse format as -* row and column linked lists of non-zero elements. Unity elements on -* the main diagonal of the matrix F are not stored. Pivot elements of -* the matrix V (that correspond to diagonal elements of the matrix U) -* are also missing from the row and column lists and stored separately -* in an ordinary array. -* -* The permutation matrices P and Q are stored as ordinary arrays using -* both row- and column-like formats. -* -* The matrices L and U being completely defined by the matrices F, V, -* P, and Q are not stored explicitly. -* -* It is easy to show that the factorization (1)-(3) is some version of -* LU-factorization. Indeed, from (3) and (4) it follows that: -* -* F = inv(P) * L * P, -* -* V = inv(P) * U * inv(Q), -* -* and substitution into (2) gives: -* -* A = F * V = inv(P) * L * U * inv(Q). -* -* For more details see the program documentation. */ - -typedef struct LUX LUX; -typedef struct LUXELM LUXELM; -typedef struct LUXWKA LUXWKA; - -struct LUX -{ /* LU-factorization of a square matrix */ - int n; - /* the order of matrices A, F, V, P, Q */ - DMP *pool; - /* memory pool for elements of matrices F and V */ - LUXELM **F_row; /* LUXELM *F_row[1+n]; */ - /* F_row[0] is not used; - F_row[i], 1 <= i <= n, is a pointer to the list of elements in - i-th row of matrix F (diagonal elements are not stored) */ - LUXELM **F_col; /* LUXELM *F_col[1+n]; */ - /* F_col[0] is not used; - F_col[j], 1 <= j <= n, is a pointer to the list of elements in - j-th column of matrix F (diagonal elements are not stored) */ - mpq_t *V_piv; /* mpq_t V_piv[1+n]; */ - /* V_piv[0] is not used; - V_piv[p], 1 <= p <= n, is a pivot element v[p,q] corresponding - to a diagonal element u[k,k] of matrix U = P*V*Q (used on k-th - elimination step, k = 1, 2, ..., n) */ - LUXELM **V_row; /* LUXELM *V_row[1+n]; */ - /* V_row[0] is not used; - V_row[i], 1 <= i <= n, is a pointer to the list of elements in - i-th row of matrix V (except pivot elements) */ - LUXELM **V_col; /* LUXELM *V_col[1+n]; */ - /* V_col[0] is not used; - V_col[j], 1 <= j <= n, is a pointer to the list of elements in - j-th column of matrix V (except pivot elements) */ - int *P_row; /* int P_row[1+n]; */ - /* P_row[0] is not used; - P_row[i] = j means that p[i,j] = 1, where p[i,j] is an element - of permutation matrix P */ - int *P_col; /* int P_col[1+n]; */ - /* P_col[0] is not used; - P_col[j] = i means that p[i,j] = 1, where p[i,j] is an element - of permutation matrix P */ - /* if i-th row or column of matrix F is i'-th row or column of - matrix L = P*F*inv(P), or if i-th row of matrix V is i'-th row - of matrix U = P*V*Q, then P_row[i'] = i and P_col[i] = i' */ - int *Q_row; /* int Q_row[1+n]; */ - /* Q_row[0] is not used; - Q_row[i] = j means that q[i,j] = 1, where q[i,j] is an element - of permutation matrix Q */ - int *Q_col; /* int Q_col[1+n]; */ - /* Q_col[0] is not used; - Q_col[j] = i means that q[i,j] = 1, where q[i,j] is an element - of permutation matrix Q */ - /* if j-th column of matrix V is j'-th column of matrix U = P*V*Q, - then Q_row[j] = j' and Q_col[j'] = j */ - int rank; - /* the (exact) rank of matrices A and V */ -}; - -struct LUXELM -{ /* element of matrix F or V */ - int i; - /* row index, 1 <= i <= m */ - int j; - /* column index, 1 <= j <= n */ - mpq_t val; - /* numeric (non-zero) element value */ - LUXELM *r_prev; - /* pointer to previous element in the same row */ - LUXELM *r_next; - /* pointer to next element in the same row */ - LUXELM *c_prev; - /* pointer to previous element in the same column */ - LUXELM *c_next; - /* pointer to next element in the same column */ -}; - -struct LUXWKA -{ /* working area (used only during factorization) */ - /* in order to efficiently implement Markowitz strategy and Duff - search technique there are two families {R[0], R[1], ..., R[n]} - and {C[0], C[1], ..., C[n]}; member R[k] is a set of active - rows of matrix V having k non-zeros, and member C[k] is a set - of active columns of matrix V having k non-zeros (in the active - submatrix); each set R[k] and C[k] is implemented as a separate - doubly linked list */ - int *R_len; /* int R_len[1+n]; */ - /* R_len[0] is not used; - R_len[i], 1 <= i <= n, is the number of non-zero elements in - i-th row of matrix V (that is the length of i-th row) */ - int *R_head; /* int R_head[1+n]; */ - /* R_head[k], 0 <= k <= n, is the number of a first row, which is - active and whose length is k */ - int *R_prev; /* int R_prev[1+n]; */ - /* R_prev[0] is not used; - R_prev[i], 1 <= i <= n, is the number of a previous row, which - is active and has the same length as i-th row */ - int *R_next; /* int R_next[1+n]; */ - /* R_prev[0] is not used; - R_prev[i], 1 <= i <= n, is the number of a next row, which is - active and has the same length as i-th row */ - int *C_len; /* int C_len[1+n]; */ - /* C_len[0] is not used; - C_len[j], 1 <= j <= n, is the number of non-zero elements in - j-th column of the active submatrix of matrix V (that is the - length of j-th column in the active submatrix) */ - int *C_head; /* int C_head[1+n]; */ - /* C_head[k], 0 <= k <= n, is the number of a first column, which - is active and whose length is k */ - int *C_prev; /* int C_prev[1+n]; */ - /* C_prev[0] is not used; - C_prev[j], 1 <= j <= n, is the number of a previous column, - which is active and has the same length as j-th column */ - int *C_next; /* int C_next[1+n]; */ - /* C_next[0] is not used; - C_next[j], 1 <= j <= n, is the number of a next column, which - is active and has the same length as j-th column */ -}; - -#define lux_create _glp_lux_create -LUX *lux_create(int n); -/* create LU-factorization */ - -#define lux_decomp _glp_lux_decomp -int lux_decomp(LUX *lux, int (*col)(void *info, int j, int ind[], - mpq_t val[]), void *info); -/* compute LU-factorization */ - -#define lux_f_solve _glp_lux_f_solve -void lux_f_solve(LUX *lux, int tr, mpq_t x[]); -/* solve system F*x = b or F'*x = b */ - -#define lux_v_solve _glp_lux_v_solve -void lux_v_solve(LUX *lux, int tr, mpq_t x[]); -/* solve system V*x = b or V'*x = b */ - -#define lux_solve _glp_lux_solve -void lux_solve(LUX *lux, int tr, mpq_t x[]); -/* solve system A*x = b or A'*x = b */ - -#define lux_delete _glp_lux_delete -void lux_delete(LUX *lux); -/* delete LU-factorization */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/env/alloc.c b/code/3rd_glpk/env/alloc.c deleted file mode 100644 index 8e2d613d..00000000 --- a/code/3rd_glpk/env/alloc.c +++ /dev/null @@ -1,252 +0,0 @@ -/* alloc.c (dynamic memory allocation) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" - -#define ALIGN 16 -/* some processors need data to be properly aligned, so this macro - * defines the alignment boundary, in bytes, provided by glpk memory - * allocation routines; looks like 16-byte alignment boundary is - * sufficient for all 32- and 64-bit platforms (8-byte boundary is not - * sufficient for some 64-bit platforms because of jmp_buf) */ - -#define MBD_SIZE (((sizeof(MBD) + (ALIGN - 1)) / ALIGN) * ALIGN) -/* size of memory block descriptor, in bytes, rounded up to multiple - * of the alignment boundary */ - -/*********************************************************************** -* dma - dynamic memory allocation (basic routine) -* -* This routine performs dynamic memory allocation. It is similar to -* the standard realloc function, however, it provides every allocated -* memory block with a descriptor, which is used for sanity checks on -* reallocating/freeing previously allocated memory blocks as well as -* for book-keeping the memory usage statistics. */ - -static void *dma(const char *func, void *ptr, size_t size) -{ ENV *env = get_env_ptr(); - MBD *mbd; - if (ptr == NULL) - { /* new memory block will be allocated */ - mbd = NULL; - } - else - { /* allocated memory block will be reallocated or freed */ - /* get pointer to the block descriptor */ - mbd = (MBD *)((char *)ptr - MBD_SIZE); - /* make sure that the block descriptor is valid */ - if (mbd->self != mbd) - xerror("%s: ptr = %p; invalid pointer\n", func, ptr); - /* remove the block from the linked list */ - mbd->self = NULL; - if (mbd->prev == NULL) - env->mem_ptr = mbd->next; - else - mbd->prev->next = mbd->next; - if (mbd->next == NULL) - ; - else - mbd->next->prev = mbd->prev; - /* decrease usage counts */ - if (!(env->mem_count >= 1 && env->mem_total >= mbd->size)) - xerror("%s: memory allocation error\n", func); - env->mem_count--; - env->mem_total -= mbd->size; - if (size == 0) - { /* free the memory block */ - free(mbd); - return NULL; - } - } - /* allocate/reallocate memory block */ - if (size > SIZE_T_MAX - MBD_SIZE) - xerror("%s: block too large\n", func); - size += MBD_SIZE; - if (size > env->mem_limit - env->mem_total) - xerror("%s: memory allocation limit exceeded\n", func); - if (env->mem_count == INT_MAX) - xerror("%s: too many memory blocks allocated\n", func); - mbd = (mbd == NULL ? malloc(size) : realloc(mbd, size)); - if (mbd == NULL) - xerror("%s: no memory available\n", func); - /* setup the block descriptor */ - mbd->size = size; - mbd->self = mbd; - mbd->prev = NULL; - mbd->next = env->mem_ptr; - /* add the block to the beginning of the linked list */ - if (mbd->next != NULL) - mbd->next->prev = mbd; - env->mem_ptr = mbd; - /* increase usage counts */ - env->mem_count++; - if (env->mem_cpeak < env->mem_count) - env->mem_cpeak = env->mem_count; - env->mem_total += size; - if (env->mem_tpeak < env->mem_total) - env->mem_tpeak = env->mem_total; - return (char *)mbd + MBD_SIZE; -} - -/*********************************************************************** -* NAME -* -* glp_alloc - allocate memory block -* -* SYNOPSIS -* -* void *glp_alloc(int n, int size); -* -* DESCRIPTION -* -* The routine glp_alloc allocates a memory block of n * size bytes -* long. -* -* Note that being allocated the memory block contains arbitrary data -* (not binary zeros!). -* -* RETURNS -* -* The routine glp_alloc returns a pointer to the block allocated. -* To free this block the routine glp_free (not free!) must be used. */ - -void *glp_alloc(int n, int size) -{ if (n < 1) - xerror("glp_alloc: n = %d; invalid parameter\n", n); - if (size < 1) - xerror("glp_alloc: size = %d; invalid parameter\n", size); - if ((size_t)n > SIZE_T_MAX / (size_t)size) - xerror("glp_alloc: n = %d, size = %d; block too large\n", - n, size); - return dma("glp_alloc", NULL, (size_t)n * (size_t)size); -} - -/**********************************************************************/ - -void *glp_realloc(void *ptr, int n, int size) -{ /* reallocate memory block */ - if (ptr == NULL) - xerror("glp_realloc: ptr = %p; invalid pointer\n", ptr); - if (n < 1) - xerror("glp_realloc: n = %d; invalid parameter\n", n); - if (size < 1) - xerror("glp_realloc: size = %d; invalid parameter\n", size); - if ((size_t)n > SIZE_T_MAX / (size_t)size) - xerror("glp_realloc: n = %d, size = %d; block too large\n", - n, size); - return dma("glp_realloc", ptr, (size_t)n * (size_t)size); -} - -/*********************************************************************** -* NAME -* -* glp_free - free (deallocate) memory block -* -* SYNOPSIS -* -* void glp_free(void *ptr); -* -* DESCRIPTION -* -* The routine glp_free frees (deallocates) a memory block pointed to -* by ptr, which was previuosly allocated by the routine glp_alloc or -* reallocated by the routine glp_realloc. */ - -void glp_free(void *ptr) -{ if (ptr == NULL) - xerror("glp_free: ptr = %p; invalid pointer\n", ptr); - dma("glp_free", ptr, 0); - return; -} - -/*********************************************************************** -* NAME -* -* glp_mem_limit - set memory usage limit -* -* SYNOPSIS -* -* void glp_mem_limit(int limit); -* -* DESCRIPTION -* -* The routine glp_mem_limit limits the amount of memory available for -* dynamic allocation (in GLPK routines) to limit megabytes. */ - -void glp_mem_limit(int limit) -{ ENV *env = get_env_ptr(); - if (limit < 1) - xerror("glp_mem_limit: limit = %d; invalid parameter\n", - limit); - if ((size_t)limit <= (SIZE_T_MAX >> 20)) - env->mem_limit = (size_t)limit << 20; - else - env->mem_limit = SIZE_T_MAX; - return; -} - -/*********************************************************************** -* NAME -* -* glp_mem_usage - get memory usage information -* -* SYNOPSIS -* -* void glp_mem_usage(int *count, int *cpeak, size_t *total, -* size_t *tpeak); -* -* DESCRIPTION -* -* The routine glp_mem_usage reports some information about utilization -* of the memory by GLPK routines. Information is stored to locations -* specified by corresponding parameters (see below). Any parameter can -* be specified as NULL, in which case its value is not stored. -* -* *count is the number of the memory blocks currently allocated by the -* routines glp_malloc and glp_calloc (one call to glp_malloc or -* glp_calloc results in allocating one memory block). -* -* *cpeak is the peak value of *count reached since the initialization -* of the GLPK library environment. -* -* *total is the total amount, in bytes, of the memory blocks currently -* allocated by the routines glp_malloc and glp_calloc. -* -* *tpeak is the peak value of *total reached since the initialization -* of the GLPK library envirionment. */ - -void glp_mem_usage(int *count, int *cpeak, size_t *total, - size_t *tpeak) -{ ENV *env = get_env_ptr(); - if (count != NULL) - *count = env->mem_count; - if (cpeak != NULL) - *cpeak = env->mem_cpeak; - if (total != NULL) - *total = env->mem_total; - if (tpeak != NULL) - *tpeak = env->mem_tpeak; - return; -} - -/* eof */ diff --git a/code/3rd_glpk/env/dlsup.c b/code/3rd_glpk/env/dlsup.c deleted file mode 100644 index 54c56c6d..00000000 --- a/code/3rd_glpk/env/dlsup.c +++ /dev/null @@ -1,167 +0,0 @@ -/* dlsup.c (dynamic linking support) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifdef HAVE_CONFIG_H -#include -#endif - -#include "env.h" - -/* GNU version ********************************************************/ - -#if defined(HAVE_LTDL) - -#include - -void *xdlopen(const char *module) -{ /* open dynamically linked library */ - void *h = NULL; - if (lt_dlinit() != 0) - { put_err_msg(lt_dlerror()); - goto done; - } - h = lt_dlopen(module); - if (h == NULL) - { put_err_msg(lt_dlerror()); - if (lt_dlexit() != 0) - xerror("xdlopen: %s\n", lt_dlerror()); - } -done: return h; -} - -void *xdlsym(void *h, const char *symbol) -{ /* obtain address of symbol from dynamically linked library */ - void *ptr; - xassert(h != NULL); - ptr = lt_dlsym(h, symbol); - if (ptr == NULL) - xerror("xdlsym: %s: %s\n", symbol, lt_dlerror()); - return ptr; -} - -void xdlclose(void *h) -{ /* close dynamically linked library */ - xassert(h != NULL); - if (lt_dlclose(h) != 0) - xerror("xdlclose: %s\n", lt_dlerror()); - if (lt_dlexit() != 0) - xerror("xdlclose: %s\n", lt_dlerror()); - return; -} - -/* POSIX version ******************************************************/ - -#elif defined(HAVE_DLFCN) - -#include - -void *xdlopen(const char *module) -{ /* open dynamically linked library */ - void *h; - h = dlopen(module, RTLD_NOW); - if (h == NULL) - put_err_msg(dlerror()); - return h; -} - -void *xdlsym(void *h, const char *symbol) -{ /* obtain address of symbol from dynamically linked library */ - void *ptr; - xassert(h != NULL); - ptr = dlsym(h, symbol); - if (ptr == NULL) - xerror("xdlsym: %s: %s\n", symbol, dlerror()); - return ptr; -} - -void xdlclose(void *h) -{ /* close dynamically linked library */ - xassert(h != NULL); - if (dlclose(h) != 0) - xerror("xdlclose: %s\n", dlerror()); - return; -} - -/* MS Windows version *************************************************/ - -#elif defined(__WOE__) - -#include - -void *xdlopen(const char *module) -{ /* open dynamically linked library */ - void *h; - h = LoadLibrary(module); - if (h == NULL) - { char msg[20]; - sprintf(msg, "Error %d", GetLastError()); - put_err_msg(msg); - } - return h; -} - -void *xdlsym(void *h, const char *symbol) -{ /* obtain address of symbol from dynamically linked library */ - void *ptr; - xassert(h != NULL); - ptr = GetProcAddress(h, symbol); - if (ptr == NULL) - xerror("xdlsym: %s: Error %d\n", symbol, GetLastError()); - return ptr; -} - -void xdlclose(void *h) -{ /* close dynamically linked library */ - xassert(h != NULL); - if (!FreeLibrary(h)) - xerror("xdlclose: Error %d\n", GetLastError()); - return; -} - -/* NULL version *******************************************************/ - -#else - -void *xdlopen(const char *module) -{ /* open dynamically linked library */ - xassert(module == module); - put_err_msg("Shared libraries not supported"); - return NULL; -} - -void *xdlsym(void *h, const char *symbol) -{ /* obtain address of symbol from dynamically linked library */ - xassert(h != h); - xassert(symbol != symbol); - return NULL; -} - -void xdlclose(void *h) -{ /* close dynamically linked library */ - xassert(h != h); - return; -} - -#endif - -/* eof */ diff --git a/code/3rd_glpk/env/env.c b/code/3rd_glpk/env/env.c deleted file mode 100644 index 5b901f35..00000000 --- a/code/3rd_glpk/env/env.c +++ /dev/null @@ -1,316 +0,0 @@ -/* env.c (GLPK environment initialization/termination) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifdef HAVE_CONFIG_H -#include -#endif - -#include "glpk.h" -#include "env.h" - -/*********************************************************************** -* NAME -* -* glp_init_env - initialize GLPK environment -* -* SYNOPSIS -* -* int glp_init_env(void); -* -* DESCRIPTION -* -* The routine glp_init_env initializes the GLPK environment. Normally -* the application program does not need to call this routine, because -* it is called automatically on the first call to any API routine. -* -* RETURNS -* -* The routine glp_init_env returns one of the following codes: -* -* 0 - initialization successful; -* 1 - environment has been already initialized; -* 2 - initialization failed (insufficient memory); -* 3 - initialization failed (unsupported programming model). */ - -int glp_init_env(void) -{ ENV *env; - int ok; - /* check if the programming model is supported */ - ok = (CHAR_BIT == 8 && sizeof(char) == 1 && - sizeof(short) == 2 && sizeof(int) == 4 && - (sizeof(void *) == 4 || sizeof(void *) == 8)); - if (!ok) - return 3; - /* check if the environment is already initialized */ - if (tls_get_ptr() != NULL) - return 1; - /* allocate and initialize the environment block */ - env = malloc(sizeof(ENV)); - if (env == NULL) - return 2; - memset(env, 0, sizeof(ENV)); -#if 0 /* 14/I-2017 */ - sprintf(env->version, "%d.%d", - GLP_MAJOR_VERSION, GLP_MINOR_VERSION); -#endif - env->self = env; - env->term_buf = malloc(TBUF_SIZE); - if (env->term_buf == NULL) - { free(env); - return 2; - } - env->term_out = GLP_ON; - env->term_hook = NULL; - env->term_info = NULL; - env->tee_file = NULL; -#if 1 /* 23/XI-2015 */ - env->err_st = 0; -#endif - env->err_file = NULL; - env->err_line = 0; - env->err_hook = NULL; - env->err_info = NULL; - env->err_buf = malloc(EBUF_SIZE); - if (env->err_buf == NULL) - { free(env->term_buf); - free(env); - return 2; - } - env->err_buf[0] = '\0'; - env->mem_limit = SIZE_T_MAX; - env->mem_ptr = NULL; - env->mem_count = env->mem_cpeak = 0; - env->mem_total = env->mem_tpeak = 0; -#if 1 /* 23/XI-2015 */ - env->gmp_pool = NULL; - env->gmp_size = 0; - env->gmp_work = NULL; -#endif - env->h_odbc = env->h_mysql = NULL; - /* save pointer to the environment block */ - tls_set_ptr(env); - /* initialization successful */ - return 0; -} - -/*********************************************************************** -* NAME -* -* get_env_ptr - retrieve pointer to environment block -* -* SYNOPSIS -* -* #include "env.h" -* ENV *get_env_ptr(void); -* -* DESCRIPTION -* -* The routine get_env_ptr retrieves and returns a pointer to the GLPK -* environment block. -* -* If the GLPK environment has not been initialized yet, the routine -* performs initialization. If initialization fails, the routine prints -* an error message to stderr and terminates the program. -* -* RETURNS -* -* The routine returns a pointer to the environment block. */ - -ENV *get_env_ptr(void) -{ ENV *env = tls_get_ptr(); - /* check if the environment has been initialized */ - if (env == NULL) - { /* not initialized yet; perform initialization */ - if (glp_init_env() != 0) - { /* initialization failed; display an error message */ - fprintf(stderr, "GLPK initialization failed\n"); - fflush(stderr); - /* and abnormally terminate the program */ - abort(); - } - /* initialization successful; retrieve the pointer */ - env = tls_get_ptr(); - } - /* check if the environment block is valid */ - if (env->self != env) - { fprintf(stderr, "Invalid GLPK environment\n"); - fflush(stderr); - abort(); - } - return env; -} - -/*********************************************************************** -* NAME -* -* glp_version - determine library version -* -* SYNOPSIS -* -* const char *glp_version(void); -* -* RETURNS -* -* The routine glp_version returns a pointer to a null-terminated -* character string, which specifies the version of the GLPK library in -* the form "X.Y", where X is the major version number, and Y is the -* minor version number, for example, "4.16". */ - -#define str(s) # s -#define xstr(s) str(s) - -const char *glp_version(void) -#if 0 /* 14/I-2017 */ -{ ENV *env = get_env_ptr(); - return env->version; -} -#else /* suggested by Heinrich */ -{ return - xstr(GLP_MAJOR_VERSION) "." xstr(GLP_MINOR_VERSION); -} -#endif - -/*********************************************************************** -* NAME -* -* glp_config - determine library configuration -* -* SYNOPSIS -* -* const char *glp_config(const char *option); -* -* DESCRIPTION -* -* The routine glp_config determines some options which were specified -* on configuring the GLPK library. -* -* RETURNS -* -* The routine glp_config returns a pointer to a null-terminating -* string depending on the option inquired. -* -* For option = "TLS" the routine returns the thread local storage -* class specifier used (e.g. "_Thread_local") if the GLPK library was -* configured to run in multi-threaded environment, or NULL otherwise. -* -* For option = "ODBC_DLNAME" the routine returns the name of ODBC -* shared library if this option was enabled, or NULL otherwise. -* -* For option = "MYSQL_DLNAME" the routine returns the name of MySQL -* shared library if this option was enabled, or NULL otherwise. */ - -const char *glp_config(const char *option) -{ const char *s; - if (strcmp(option, "TLS") == 0) -#ifndef TLS - s = NULL; -#else - s = xstr(TLS); -#endif - else if (strcmp(option, "ODBC_DLNAME") == 0) -#ifndef ODBC_DLNAME - s = NULL; -#else - s = ODBC_DLNAME; -#endif - else if (strcmp(option, "MYSQL_DLNAME") == 0) -#ifndef MYSQL_DLNAME - s = NULL; -#else - s = MYSQL_DLNAME; -#endif - else - { /* invalid option is always disabled */ - s = NULL; - } - return s; -} - -/*********************************************************************** -* NAME -* -* glp_free_env - free GLPK environment -* -* SYNOPSIS -* -* int glp_free_env(void); -* -* DESCRIPTION -* -* The routine glp_free_env frees all resources used by GLPK routines -* (memory blocks, etc.) which are currently still in use. -* -* Normally the application program does not need to call this routine, -* because GLPK routines always free all unused resources. However, if -* the application program even has deleted all problem objects, there -* will be several memory blocks still allocated for the library needs. -* For some reasons the application program may want GLPK to free this -* memory, in which case it should call glp_free_env. -* -* Note that a call to glp_free_env invalidates all problem objects as -* if no GLPK routine were called. -* -* RETURNS -* -* 0 - termination successful; -* 1 - environment is inactive (was not initialized). */ - -int glp_free_env(void) -{ ENV *env = tls_get_ptr(); - MBD *desc; - /* check if the environment is active */ - if (env == NULL) - return 1; - /* check if the environment block is valid */ - if (env->self != env) - { fprintf(stderr, "Invalid GLPK environment\n"); - fflush(stderr); - abort(); - } - /* close handles to shared libraries */ - if (env->h_odbc != NULL) - xdlclose(env->h_odbc); - if (env->h_mysql != NULL) - xdlclose(env->h_mysql); - /* free memory blocks which are still allocated */ - while (env->mem_ptr != NULL) - { desc = env->mem_ptr; - env->mem_ptr = desc->next; - free(desc); - } - /* close text file used for copying terminal output */ - if (env->tee_file != NULL) - fclose(env->tee_file); - /* invalidate the environment block */ - env->self = NULL; - /* free memory allocated to the environment block */ - free(env->term_buf); - free(env->err_buf); - free(env); - /* reset a pointer to the environment block */ - tls_set_ptr(NULL); - /* termination successful */ - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/env/env.h b/code/3rd_glpk/env/env.h deleted file mode 100644 index 67214ef6..00000000 --- a/code/3rd_glpk/env/env.h +++ /dev/null @@ -1,274 +0,0 @@ -/* env.h (GLPK environment) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef ENV_H -#define ENV_H - -#include "stdc.h" - -typedef struct ENV ENV; -typedef struct MBD MBD; - -#define SIZE_T_MAX (~(size_t)0) -/* largest value of size_t type */ - -#define TBUF_SIZE 4096 -/* terminal output buffer size, in bytes */ - -#define EBUF_SIZE 1024 -/* error message buffer size, in bytes */ - -/* enable/disable flag: */ -#define GLP_ON 1 -#define GLP_OFF 0 - -struct ENV -{ /* GLPK environment block */ -#if 0 /* 14/I-2007 */ - char version[7+1]; - /* version string returned by the routine glp_version */ -#endif - ENV *self; - /* pointer to this block to check its validity */ - /*--------------------------------------------------------------*/ - /* terminal output */ - char *term_buf; /* char term_buf[TBUF_SIZE]; */ - /* terminal output buffer */ - int term_out; - /* flag to enable/disable terminal output */ - int (*term_hook)(void *info, const char *s); - /* user-defined routine to intercept terminal output */ - void *term_info; - /* transit pointer (cookie) passed to the routine term_hook */ - FILE *tee_file; - /* output stream used to copy terminal output */ - /*--------------------------------------------------------------*/ - /* error handling */ -#if 1 /* 07/XI-2015 */ - int err_st; - /* error state flag; set on entry to glp_error */ -#endif - const char *err_file; - /* value of the __FILE__ macro passed to glp_error */ - int err_line; - /* value of the __LINE__ macro passed to glp_error */ - void (*err_hook)(void *info); - /* user-defined routine to intercept abnormal termination */ - void *err_info; - /* transit pointer (cookie) passed to the routine err_hook */ - char *err_buf; /* char err_buf[EBUF_SIZE]; */ - /* buffer to store error messages (used by I/O routines) */ - /*--------------------------------------------------------------*/ - /* dynamic memory allocation */ - size_t mem_limit; - /* maximal amount of memory, in bytes, available for dynamic - * allocation */ - MBD *mem_ptr; - /* pointer to the linked list of allocated memory blocks */ - int mem_count; - /* total number of currently allocated memory blocks */ - int mem_cpeak; - /* peak value of mem_count */ - size_t mem_total; - /* total amount of currently allocated memory, in bytes; it is - * the sum of the size field over all memory block descriptors */ - size_t mem_tpeak; - /* peak value of mem_total */ -#if 1 /* 23/XI-2015 */ - /*--------------------------------------------------------------*/ - /* bignum module working area */ - void *gmp_pool; /* DMP *gmp_pool; */ - /* working memory pool */ - int gmp_size; - /* size of working array */ - unsigned short *gmp_work; /* ushort gmp_work[gmp_size]; */ - /* working array */ -#endif - /*--------------------------------------------------------------*/ - /* dynamic linking support (optional) */ - void *h_odbc; - /* handle to ODBC shared library */ - void *h_mysql; - /* handle to MySQL shared library */ -}; - -struct MBD -{ /* memory block descriptor */ - size_t size; - /* size of block, in bytes, including descriptor */ - MBD *self; - /* pointer to this descriptor to check its validity */ - MBD *prev; - /* pointer to previous memory block descriptor */ - MBD *next; - /* pointer to next memory block descriptor */ -}; - -#define get_env_ptr _glp_get_env_ptr -ENV *get_env_ptr(void); -/* retrieve pointer to environment block */ - -#define tls_set_ptr _glp_tls_set_ptr -void tls_set_ptr(void *ptr); -/* store global pointer in TLS */ - -#define tls_get_ptr _glp_tls_get_ptr -void *tls_get_ptr(void); -/* retrieve global pointer from TLS */ - -#define xputs glp_puts -void glp_puts(const char *s); -/* write string on terminal */ - -#define xprintf glp_printf -void glp_printf(const char *fmt, ...); -/* write formatted output on terminal */ - -#define xvprintf glp_vprintf -void glp_vprintf(const char *fmt, va_list arg); -/* write formatted output on terminal */ - -int glp_term_out(int flag); -/* enable/disable terminal output */ - -void glp_term_hook(int (*func)(void *info, const char *s), void *info); -/* install hook to intercept terminal output */ - -int glp_open_tee(const char *fname); -/* start copying terminal output to text file */ - -int glp_close_tee(void); -/* stop copying terminal output to text file */ - -#ifndef GLP_ERRFUNC_DEFINED -#define GLP_ERRFUNC_DEFINED -typedef void (*glp_errfunc)(const char *fmt, ...); -#endif - -#define xerror glp_error_(__FILE__, __LINE__) -glp_errfunc glp_error_(const char *file, int line); -/* display fatal error message and terminate execution */ - -#define xassert(expr) \ - ((void)((expr) || (glp_assert_(#expr, __FILE__, __LINE__), 1))) -void glp_assert_(const char *expr, const char *file, int line); -/* check for logical condition */ - -void glp_error_hook(void (*func)(void *info), void *info); -/* install hook to intercept abnormal termination */ - -#define put_err_msg _glp_put_err_msg -void put_err_msg(const char *msg); -/* provide error message string */ - -#define get_err_msg _glp_get_err_msg -const char *get_err_msg(void); -/* obtain error message string */ - -#define xmalloc(size) glp_alloc(1, size) -/* allocate memory block (obsolete) */ - -#define xcalloc(n, size) glp_alloc(n, size) -/* allocate memory block (obsolete) */ - -#define xalloc(n, size) glp_alloc(n, size) -#define talloc(n, type) ((type *)glp_alloc(n, sizeof(type))) -void *glp_alloc(int n, int size); -/* allocate memory block */ - -#define xrealloc(ptr, n, size) glp_realloc(ptr, n, size) -#define trealloc(ptr, n, type) ((type *)glp_realloc(ptr, n, \ - sizeof(type))) -void *glp_realloc(void *ptr, int n, int size); -/* reallocate memory block */ - -#define xfree(ptr) glp_free(ptr) -#define tfree(ptr) glp_free(ptr) -void glp_free(void *ptr); -/* free memory block */ - -void glp_mem_limit(int limit); -/* set memory usage limit */ - -void glp_mem_usage(int *count, int *cpeak, size_t *total, - size_t *tpeak); -/* get memory usage information */ - -typedef struct glp_file glp_file; -/* sequential stream descriptor */ - -#define glp_open _glp_open -glp_file *glp_open(const char *name, const char *mode); -/* open stream */ - -#define glp_eof _glp_eof -int glp_eof(glp_file *f); -/* test end-of-file indicator */ - -#define glp_ioerr _glp_ioerr -int glp_ioerr(glp_file *f); -/* test I/O error indicator */ - -#define glp_read _glp_read -int glp_read(glp_file *f, void *buf, int nnn); -/* read data from stream */ - -#define glp_getc _glp_getc -int glp_getc(glp_file *f); -/* read character from stream */ - -#define glp_write _glp_write -int glp_write(glp_file *f, const void *buf, int nnn); -/* write data to stream */ - -#define glp_format _glp_format -int glp_format(glp_file *f, const char *fmt, ...); -/* write formatted data to stream */ - -#define glp_close _glp_close -int glp_close(glp_file *f); -/* close stream */ - -#define xtime glp_time -double glp_time(void); -/* determine current universal time */ - -#define xdifftime glp_difftime -double glp_difftime(double t1, double t0); -/* compute difference between two time values */ - -#define xdlopen _glp_dlopen -void *xdlopen(const char *module); -/* open dynamically linked library */ - -#define xdlsym _glp_dlsym -void *xdlsym(void *h, const char *symbol); -/* obtain address of symbol from dynamically linked library */ - -#define xdlclose _glp_dlclose -void xdlclose(void *h); -/* close dynamically linked library */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/env/error.c b/code/3rd_glpk/env/error.c deleted file mode 100644 index a898b768..00000000 --- a/code/3rd_glpk/env/error.c +++ /dev/null @@ -1,200 +0,0 @@ -/* error.c (error handling) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" - -/*********************************************************************** -* NAME -* -* glp_error - display fatal error message and terminate execution -* -* SYNOPSIS -* -* void glp_error(const char *fmt, ...); -* -* DESCRIPTION -* -* The routine glp_error (implemented as a macro) formats its -* parameters using the format control string fmt, writes the formatted -* message on the terminal, and abnormally terminates the program. */ - -static void errfunc(const char *fmt, ...) -{ ENV *env = get_env_ptr(); - va_list arg; -#if 1 /* 07/XI-2015 */ - env->err_st = 1; -#endif - env->term_out = GLP_ON; - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - xprintf("Error detected in file %s at line %d\n", - env->err_file, env->err_line); - if (env->err_hook != NULL) - env->err_hook(env->err_info); - abort(); - exit(EXIT_FAILURE); - /* no return */ -} - -glp_errfunc glp_error_(const char *file, int line) -{ ENV *env = get_env_ptr(); - env->err_file = file; - env->err_line = line; - return errfunc; -} - -#if 1 /* 07/XI-2015 */ -/*********************************************************************** -* NAME -* -* glp_at_error - check for error state -* -* SYNOPSIS -* -* int glp_at_error(void); -* -* DESCRIPTION -* -* The routine glp_at_error checks if the GLPK environment is at error -* state, i.e. if the call to the routine is (indirectly) made from the -* glp_error routine via an user-defined hook routine. -* -* RETURNS -* -* If the GLPK environment is at error state, the routine glp_at_error -* returns non-zero, otherwise zero. */ - -int glp_at_error(void) -{ ENV *env = get_env_ptr(); - return env->err_st; -} -#endif - -/*********************************************************************** -* NAME -* -* glp_assert - check for logical condition -* -* SYNOPSIS -* -* void glp_assert(int expr); -* -* DESCRIPTION -* -* The routine glp_assert (implemented as a macro) checks for a logical -* condition specified by the parameter expr. If the condition is false -* (i.e. the value of expr is zero), the routine writes a message on -* the terminal and abnormally terminates the program. */ - -void glp_assert_(const char *expr, const char *file, int line) -{ glp_error_(file, line)("Assertion failed: %s\n", expr); - /* no return */ -} - -/*********************************************************************** -* NAME -* -* glp_error_hook - install hook to intercept abnormal termination -* -* SYNOPSIS -* -* void glp_error_hook(void (*func)(void *info), void *info); -* -* DESCRIPTION -* -* The routine glp_error_hook installs a user-defined hook routine to -* intercept abnormal termination. -* -* The parameter func specifies the user-defined hook routine. It is -* called from the routine glp_error before the latter calls the abort -* function to abnormally terminate the application program because of -* fatal error. The parameter info is a transit pointer, specified in -* the corresponding call to the routine glp_error_hook; it may be used -* to pass some information to the hook routine. -* -* To uninstall the hook routine the parameters func and info should be -* both specified as NULL. */ - -void glp_error_hook(void (*func)(void *info), void *info) -{ ENV *env = get_env_ptr(); - if (func == NULL) - { env->err_hook = NULL; - env->err_info = NULL; - } - else - { env->err_hook = func; - env->err_info = info; - } - return; -} - -/*********************************************************************** -* NAME -* -* put_err_msg - provide error message string -* -* SYNOPSIS -* -* #include "env.h" -* void put_err_msg(const char *msg); -* -* DESCRIPTION -* -* The routine put_err_msg stores an error message string pointed to by -* msg to the environment block. */ - -void put_err_msg(const char *msg) -{ ENV *env = get_env_ptr(); - int len; - len = strlen(msg); - if (len >= EBUF_SIZE) - len = EBUF_SIZE - 1; - memcpy(env->err_buf, msg, len); - if (len > 0 && env->err_buf[len-1] == '\n') - len--; - env->err_buf[len] = '\0'; - return; -} - -/*********************************************************************** -* NAME -* -* get_err_msg - obtain error message string -* -* SYNOPSIS -* -* #include "env.h" -* const char *get_err_msg(void); -* -* RETURNS -* -* The routine get_err_msg returns a pointer to an error message string -* previously stored by the routine put_err_msg. */ - -const char *get_err_msg(void) -{ ENV *env = get_env_ptr(); - return env->err_buf; -} - -/* eof */ diff --git a/code/3rd_glpk/env/stdc.c b/code/3rd_glpk/env/stdc.c deleted file mode 100644 index 59331e22..00000000 --- a/code/3rd_glpk/env/stdc.c +++ /dev/null @@ -1,98 +0,0 @@ -/* stdc.c (replacements for standard non-thread-safe functions) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifdef HAVE_CONFIG_H -#include -#endif - -/* portable ANSI C version ********************************************/ - -#if !defined(TLS) - -#define ENABLE_NON_SAFE -#include "stdc.h" - -struct tm *xgmtime(const time_t *timer) -{ return - gmtime(timer); -} - -char *xstrerr(int errnum) -{ return - strerror(errnum); -} - -char *xstrtok(char *s1, const char *s2) -{ return - strtok(s1, s2); -} - -/* MS Windows version *************************************************/ - -#elif defined(__WOE__) - -#include "stdc.h" - -struct tm *xgmtime(const time_t *timer) -{ static TLS struct tm result; - gmtime_s(&result, timer); - return &result; -} - -char *xstrerr(int errnum) -{ static TLS char s[1023+1]; - strerror_s(s, sizeof(s), errnum); - return s; -} - -char *xstrtok(char *s1, const char *s2) -{ static TLS char *ptr; - return strtok_s(s1, s2, &ptr); -} - -/* GNU/Linux version **************************************************/ - -#else - -#include "stdc.h" - -struct tm *xgmtime(const time_t *timer) -{ static TLS struct tm result; - gmtime_r(timer, &result); - return &result; -} - -char *xstrerr(int errnum) -{ static TLS char s[1023+1]; - strerror_r(errnum, s, sizeof(s)); - return s; -} - -char *xstrtok(char *s1, const char *s2) -{ static TLS char *ptr; - return strtok_r(s1, s2, &ptr); -} - -#endif - -/* eof */ diff --git a/code/3rd_glpk/env/stdc.h b/code/3rd_glpk/env/stdc.h deleted file mode 100644 index a376f2c9..00000000 --- a/code/3rd_glpk/env/stdc.h +++ /dev/null @@ -1,73 +0,0 @@ -/* stdc.h (standard ANSI C headers) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef STDC_H -#define STDC_H - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#ifndef ENABLE_NON_SAFE /* 29/I-2017 */ -/* disable using non-thread-safe functions directly */ -#undef gmtime -#define gmtime ??? -#undef strerror -#define strerror ??? -#undef strtok -#define strtok ??? -#endif - -#if 1 /* 29/I-2017 */ -/* provide replacements for these functions on a per-thread basis */ -#define xgmtime _glp_xgmtime -struct tm *xgmtime(const time_t *); -#define xstrerr _glp_xstrerr -char *xstrerr(int); -#define xstrtok _glp_xstrtok -char *xstrtok(char *, const char *); -#endif - -#if 1 /* 06/II-2018 */ -#ifdef HAVE_CONFIG_H -#include -#endif -#ifndef __WOE__ -#define CDECL -#else -#define CDECL __cdecl -#endif -#endif - -#endif - -/* eof */ diff --git a/code/3rd_glpk/env/stdout.c b/code/3rd_glpk/env/stdout.c deleted file mode 100644 index 94eee02a..00000000 --- a/code/3rd_glpk/env/stdout.c +++ /dev/null @@ -1,262 +0,0 @@ -/* stdout.c (terminal output) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#undef NDEBUG -#include -#include "env.h" - -/*********************************************************************** -* NAME -* -* glp_puts - write string on terminal -* -* SYNOPSIS -* -* void glp_puts(const char *s); -* -* The routine glp_puts writes the string s on the terminal. */ - -void glp_puts(const char *s) -{ ENV *env = get_env_ptr(); - /* if terminal output is disabled, do nothing */ - if (!env->term_out) - goto skip; - /* pass the string to the hook routine, if defined */ - if (env->term_hook != NULL) - { if (env->term_hook(env->term_info, s) != 0) - goto skip; - } - /* write the string on the terminal */ - fputs(s, stdout); - fflush(stdout); - /* write the string on the tee file, if required */ - if (env->tee_file != NULL) - { fputs(s, env->tee_file); - fflush(env->tee_file); - } -skip: return; -} - -/*********************************************************************** -* NAME -* -* glp_printf - write formatted output on terminal -* -* SYNOPSIS -* -* void glp_printf(const char *fmt, ...); -* -* DESCRIPTION -* -* The routine glp_printf uses the format control string fmt to format -* its parameters and writes the formatted output on the terminal. */ - -void glp_printf(const char *fmt, ...) -{ ENV *env = get_env_ptr(); - va_list arg; - /* if terminal output is disabled, do nothing */ - if (!env->term_out) - goto skip; - /* format the output */ - va_start(arg, fmt); - vsprintf(env->term_buf, fmt, arg); - /* (do not use xassert) */ - assert(strlen(env->term_buf) < TBUF_SIZE); - va_end(arg); - /* write the formatted output on the terminal */ - glp_puts(env->term_buf); -skip: return; -} - -/*********************************************************************** -* NAME -* -* glp_vprintf - write formatted output on terminal -* -* SYNOPSIS -* -* void glp_vprintf(const char *fmt, va_list arg); -* -* DESCRIPTION -* -* The routine glp_vprintf uses the format control string fmt to format -* its parameters specified by the list arg and writes the formatted -* output on the terminal. */ - -void glp_vprintf(const char *fmt, va_list arg) -{ ENV *env = get_env_ptr(); - /* if terminal output is disabled, do nothing */ - if (!env->term_out) - goto skip; - /* format the output */ - vsprintf(env->term_buf, fmt, arg); - /* (do not use xassert) */ - assert(strlen(env->term_buf) < TBUF_SIZE); - /* write the formatted output on the terminal */ - glp_puts(env->term_buf); -skip: return; -} - -/*********************************************************************** -* NAME -* -* glp_term_out - enable/disable terminal output -* -* SYNOPSIS -* -* int glp_term_out(int flag); -* -* DESCRIPTION -* -* Depending on the parameter flag the routine glp_term_out enables or -* disables terminal output performed by glpk routines: -* -* GLP_ON - enable terminal output; -* GLP_OFF - disable terminal output. -* -* RETURNS -* -* The routine glp_term_out returns the previous value of the terminal -* output flag. */ - -int glp_term_out(int flag) -{ ENV *env = get_env_ptr(); - int old = env->term_out; - if (!(flag == GLP_ON || flag == GLP_OFF)) - xerror("glp_term_out: flag = %d; invalid parameter\n", flag); - env->term_out = flag; - return old; -} - -/*********************************************************************** -* NAME -* -* glp_term_hook - install hook to intercept terminal output -* -* SYNOPSIS -* -* void glp_term_hook(int (*func)(void *info, const char *s), -* void *info); -* -* DESCRIPTION -* -* The routine glp_term_hook installs a user-defined hook routine to -* intercept all terminal output performed by glpk routines. -* -* This feature can be used to redirect the terminal output to other -* destination, for example to a file or a text window. -* -* The parameter func specifies the user-defined hook routine. It is -* called from an internal printing routine, which passes to it two -* parameters: info and s. The parameter info is a transit pointer, -* specified in the corresponding call to the routine glp_term_hook; -* it may be used to pass some information to the hook routine. The -* parameter s is a pointer to the null terminated character string, -* which is intended to be written to the terminal. If the hook routine -* returns zero, the printing routine writes the string s to the -* terminal in a usual way; otherwise, if the hook routine returns -* non-zero, no terminal output is performed. -* -* To uninstall the hook routine the parameters func and info should be -* specified as NULL. */ - -void glp_term_hook(int (*func)(void *info, const char *s), void *info) -{ ENV *env = get_env_ptr(); - if (func == NULL) - { env->term_hook = NULL; - env->term_info = NULL; - } - else - { env->term_hook = func; - env->term_info = info; - } - return; -} - -/*********************************************************************** -* NAME -* -* glp_open_tee - start copying terminal output to text file -* -* SYNOPSIS -* -* int glp_open_tee(const char *name); -* -* DESCRIPTION -* -* The routine glp_open_tee starts copying all the terminal output to -* an output text file, whose name is specified by the character string -* name. -* -* RETURNS -* -* 0 - operation successful -* 1 - copying terminal output is already active -* 2 - unable to create output file */ - -int glp_open_tee(const char *name) -{ ENV *env = get_env_ptr(); - if (env->tee_file != NULL) - { /* copying terminal output is already active */ - return 1; - } - env->tee_file = fopen(name, "w"); - if (env->tee_file == NULL) - { /* unable to create output file */ - return 2; - } - return 0; -} - -/*********************************************************************** -* NAME -* -* glp_close_tee - stop copying terminal output to text file -* -* SYNOPSIS -* -* int glp_close_tee(void); -* -* DESCRIPTION -* -* The routine glp_close_tee stops copying the terminal output to the -* output text file previously open by the routine glp_open_tee closing -* that file. -* -* RETURNS -* -* 0 - operation successful -* 1 - copying terminal output was not started */ - -int glp_close_tee(void) -{ ENV *env = get_env_ptr(); - if (env->tee_file == NULL) - { /* copying terminal output was not started */ - return 1; - } - fclose(env->tee_file); - env->tee_file = NULL; - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/env/stream.c b/code/3rd_glpk/env/stream.c deleted file mode 100644 index 906e5b04..00000000 --- a/code/3rd_glpk/env/stream.c +++ /dev/null @@ -1,517 +0,0 @@ -/* stream.c (stream input/output) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "zlib.h" - -struct glp_file -{ /* sequential stream descriptor */ - char *base; - /* pointer to buffer */ - int size; - /* size of buffer, in bytes */ - char *ptr; - /* pointer to next byte in buffer */ - int cnt; - /* count of bytes in buffer */ - int flag; - /* stream flags: */ -#define IONULL 0x01 /* null file */ -#define IOSTD 0x02 /* standard stream */ -#define IOGZIP 0x04 /* gzipped file */ -#define IOWRT 0x08 /* output stream */ -#define IOEOF 0x10 /* end of file */ -#define IOERR 0x20 /* input/output error */ - void *file; - /* pointer to underlying control object */ -}; - -/*********************************************************************** -* NAME -* -* glp_open - open stream -* -* SYNOPSIS -* -* glp_file *glp_open(const char *name, const char *mode); -* -* DESCRIPTION -* -* The routine glp_open opens a file whose name is a string pointed to -* by name and associates a stream with it. -* -* The following special filenames are recognized by the routine (this -* feature is platform independent): -* -* "/dev/null" empty (null) file; -* "/dev/stdin" standard input stream; -* "/dev/stdout" standard output stream; -* "/dev/stderr" standard error stream. -* -* If the specified filename is ended with ".gz", it is assumed that -* the file is in gzipped format. In this case the file is compressed -* or decompressed by the I/O routines "on the fly". -* -* The parameter mode points to a string, which indicates the open mode -* and should be one of the following: -* -* "r" open text file for reading; -* "w" truncate to zero length or create text file for writing; -* "a" append, open or create text file for writing at end-of-file; -* "rb" open binary file for reading; -* "wb" truncate to zero length or create binary file for writing; -* "ab" append, open or create binary file for writing at end-of-file. -* -* RETURNS -* -* The routine glp_open returns a pointer to the object controlling the -* stream. If the operation fails, the routine returns NULL. */ - -glp_file *glp_open(const char *name, const char *mode) -{ glp_file *f; - int flag; - void *file; - if (strcmp(mode, "r") == 0 || strcmp(mode, "rb") == 0) - flag = 0; - else if (strcmp(mode, "w") == 0 || strcmp(mode, "wb") == 0) - flag = IOWRT; -#if 1 /* 08/V-2014 */ - else if (strcmp(mode, "a") == 0 || strcmp(mode, "ab") == 0) - flag = IOWRT; -#endif - else - xerror("glp_open: invalid mode string\n"); - if (strcmp(name, "/dev/null") == 0) - { flag |= IONULL; - file = NULL; - } - else if (strcmp(name, "/dev/stdin") == 0) - { flag |= IOSTD; - file = stdin; - } - else if (strcmp(name, "/dev/stdout") == 0) - { flag |= IOSTD; - file = stdout; - } - else if (strcmp(name, "/dev/stderr") == 0) - { flag |= IOSTD; - file = stderr; - } - else - { char *ext = strrchr(name, '.'); - if (ext == NULL || strcmp(ext, ".gz") != 0) - { file = fopen(name, mode); - if (file == NULL) -#if 0 /* 29/I-2017 */ - { put_err_msg(strerror(errno)); -#else - { put_err_msg(xstrerr(errno)); -#endif - return NULL; - } - } - else - { flag |= IOGZIP; - if (strcmp(mode, "r") == 0) - mode = "rb"; - else if (strcmp(mode, "w") == 0) - mode = "wb"; -#if 1 /* 08/V-2014; this mode seems not to work */ - else if (strcmp(mode, "a") == 0) - mode = "ab"; -#endif - file = gzopen(name, mode); - if (file == NULL) -#if 0 /* 29/I-2017 */ - { put_err_msg(strerror(errno)); -#else - { put_err_msg(xstrerr(errno)); -#endif - return NULL; - } - } - } - f = talloc(1, glp_file); - f->base = talloc(BUFSIZ, char); - f->size = BUFSIZ; - f->ptr = f->base; - f->cnt = 0; - f->flag = flag; - f->file = file; - return f; -} - -/*********************************************************************** -* NAME -* -* glp_eof - test end-of-file indicator -* -* SYNOPSIS -* -* int glp_eof(glp_file *f); -* -* DESCRIPTION -* -* The routine glp_eof tests the end-of-file indicator for the stream -* pointed to by f. -* -* RETURNS -* -* The routine glp_eof returns non-zero if and only if the end-of-file -* indicator is set for the specified stream. */ - -int glp_eof(glp_file *f) -{ return - f->flag & IOEOF; -} - -/*********************************************************************** -* NAME -* -* glp_ioerr - test I/O error indicator -* -* SYNOPSIS -* -* int glp_ioerr(glp_file *f); -* -* DESCRIPTION -* -* The routine glp_ioerr tests the I/O error indicator for the stream -* pointed to by f. -* -* RETURNS -* -* The routine glp_ioerr returns non-zero if and only if the I/O error -* indicator is set for the specified stream. */ - -int glp_ioerr(glp_file *f) -{ return - f->flag & IOERR; -} - -/*********************************************************************** -* NAME -* -* glp_read - read data from stream -* -* SYNOPSIS -* -* int glp_read(glp_file *f, void *buf, int nnn); -* -* DESCRIPTION -* -* The routine glp_read reads, into the buffer pointed to by buf, up to -* nnn bytes, from the stream pointed to by f. -* -* RETURNS -* -* The routine glp_read returns the number of bytes successfully read -* (which may be less than nnn). If an end-of-file is encountered, the -* end-of-file indicator for the stream is set and glp_read returns -* zero. If a read error occurs, the error indicator for the stream is -* set and glp_read returns a negative value. */ - -int glp_read(glp_file *f, void *buf, int nnn) -{ int nrd, cnt; - if (f->flag & IOWRT) - xerror("glp_read: attempt to read from output stream\n"); - if (nnn < 1) - xerror("glp_read: nnn = %d; invalid parameter\n", nnn); - for (nrd = 0; nrd < nnn; nrd += cnt) - { if (f->cnt == 0) - { /* buffer is empty; fill it */ - if (f->flag & IONULL) - cnt = 0; - else if (!(f->flag & IOGZIP)) - { cnt = fread(f->base, 1, f->size, (FILE *)(f->file)); - if (ferror((FILE *)(f->file))) - { f->flag |= IOERR; -#if 0 /* 29/I-2017 */ - put_err_msg(strerror(errno)); -#else - put_err_msg(xstrerr(errno)); -#endif - return EOF; - } - } - else - { int errnum; - const char *msg; - cnt = gzread((gzFile)(f->file), f->base, f->size); - if (cnt < 0) - { f->flag |= IOERR; - msg = gzerror((gzFile)(f->file), &errnum); - if (errnum == Z_ERRNO) -#if 0 /* 29/I-2017 */ - put_err_msg(strerror(errno)); -#else - put_err_msg(xstrerr(errno)); -#endif - else - put_err_msg(msg); - return EOF; - } - } - if (cnt == 0) - { if (nrd == 0) - f->flag |= IOEOF; - break; - } - f->ptr = f->base; - f->cnt = cnt; - } - cnt = nnn - nrd; - if (cnt > f->cnt) - cnt = f->cnt; - memcpy((char *)buf + nrd, f->ptr, cnt); - f->ptr += cnt; - f->cnt -= cnt; - } - return nrd; -} - -/*********************************************************************** -* NAME -* -* glp_getc - read character from stream -* -* SYNOPSIS -* -* int glp_getc(glp_file *f); -* -* DESCRIPTION -* -* The routine glp_getc obtains a next character as an unsigned char -* converted to an int from the input stream pointed to by f. -* -* RETURNS -* -* The routine glp_getc returns the next character obtained. However, -* if an end-of-file is encountered or a read error occurs, the routine -* returns EOF. (An end-of-file and a read error can be distinguished -* by use of the routines glp_eof and glp_ioerr.) */ - -int glp_getc(glp_file *f) -{ unsigned char buf[1]; - if (f->flag & IOWRT) - xerror("glp_getc: attempt to read from output stream\n"); - if (glp_read(f, buf, 1) != 1) - return EOF; - return buf[0]; -} - -/*********************************************************************** -* do_flush - flush output stream -* -* This routine causes buffered data for the specified output stream to -* be written to the associated file. -* -* If the operation was successful, the routine returns zero, otherwise -* non-zero. */ - -static int do_flush(glp_file *f) -{ xassert(f->flag & IOWRT); - if (f->cnt > 0) - { if (f->flag & IONULL) - ; - else if (!(f->flag & IOGZIP)) - { if ((int)fwrite(f->base, 1, f->cnt, (FILE *)(f->file)) - != f->cnt) - { f->flag |= IOERR; -#if 0 /* 29/I-2017 */ - put_err_msg(strerror(errno)); -#else - put_err_msg(xstrerr(errno)); -#endif - return EOF; - } - } - else - { int errnum; - const char *msg; - if (gzwrite((gzFile)(f->file), f->base, f->cnt) != f->cnt) - { f->flag |= IOERR; - msg = gzerror((gzFile)(f->file), &errnum); - if (errnum == Z_ERRNO) -#if 0 /* 29/I-2017 */ - put_err_msg(strerror(errno)); -#else - put_err_msg(xstrerr(errno)); -#endif - else - put_err_msg(msg); - return EOF; - } - } - } - f->ptr = f->base; - f->cnt = 0; - return 0; -} - -/*********************************************************************** -* NAME -* -* glp_write - write data to stream -* -* SYNOPSIS -* -* int glp_write(glp_file *f, const void *buf, int nnn); -* -* DESCRIPTION -* -* The routine glp_write writes, from the buffer pointed to by buf, up -* to nnn bytes, to the stream pointed to by f. -* -* RETURNS -* -* The routine glp_write returns the number of bytes successfully -* written (which is equal to nnn). If a write error occurs, the error -* indicator for the stream is set and glp_write returns a negative -* value. */ - -int glp_write(glp_file *f, const void *buf, int nnn) -{ int nwr, cnt; - if (!(f->flag & IOWRT)) - xerror("glp_write: attempt to write to input stream\n"); - if (nnn < 1) - xerror("glp_write: nnn = %d; invalid parameter\n", nnn); - for (nwr = 0; nwr < nnn; nwr += cnt) - { cnt = nnn - nwr; - if (cnt > f->size - f->cnt) - cnt = f->size - f->cnt; - memcpy(f->ptr, (const char *)buf + nwr, cnt); - f->ptr += cnt; - f->cnt += cnt; - if (f->cnt == f->size) - { /* buffer is full; flush it */ - if (do_flush(f) != 0) - return EOF; - } - } - return nwr; -} - -/*********************************************************************** -* NAME -* -* glp_format - write formatted data to stream -* -* SYNOPSIS -* -* int glp_format(glp_file *f, const char *fmt, ...); -* -* DESCRIPTION -* -* The routine glp_format writes formatted data to the stream pointed -* to by f. The format control string pointed to by fmt specifies how -* subsequent arguments are converted for output. -* -* RETURNS -* -* The routine glp_format returns the number of characters written, or -* a negative value if an output error occurs. */ - -int glp_format(glp_file *f, const char *fmt, ...) -{ ENV *env = get_env_ptr(); - va_list arg; - int nnn; - if (!(f->flag & IOWRT)) - xerror("glp_format: attempt to write to input stream\n"); - va_start(arg, fmt); - nnn = vsprintf(env->term_buf, fmt, arg); - xassert(0 <= nnn && nnn < TBUF_SIZE); - va_end(arg); - return nnn == 0 ? 0 : glp_write(f, env->term_buf, nnn); -} - -/*********************************************************************** -* NAME -* -* glp_close - close stream -* -* SYNOPSIS -* -* int glp_close(glp_file *f); -* -* DESCRIPTION -* -* The routine glp_close closes the stream pointed to by f. -* -* RETURNS -* -* If the operation was successful, the routine returns zero, otherwise -* non-zero. */ - -int glp_close(glp_file *f) -{ int ret = 0; - if (f->flag & IOWRT) - { if (do_flush(f) != 0) - ret = EOF; - } - if (f->flag & (IONULL | IOSTD)) - ; - else if (!(f->flag & IOGZIP)) - { if (fclose((FILE *)(f->file)) != 0) - { if (ret == 0) -#if 0 /* 29/I-2017 */ - { put_err_msg(strerror(errno)); -#else - { put_err_msg(xstrerr(errno)); -#endif - ret = EOF; - } - } - } - else - { int errnum; - errnum = gzclose((gzFile)(f->file)); - if (errnum == Z_OK) - ; - else if (errnum == Z_ERRNO) - { if (ret == 0) -#if 0 /* 29/I-2017 */ - { put_err_msg(strerror(errno)); -#else - { put_err_msg(xstrerr(errno)); -#endif - ret = EOF; - } - } -#if 1 /* FIXME */ - else - { if (ret == 0) - { ENV *env = get_env_ptr(); - sprintf(env->term_buf, "gzclose returned %d", errnum); - put_err_msg(env->term_buf); - ret = EOF; - } - } -#endif - } - tfree(f->base); - tfree(f); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/env/time.c b/code/3rd_glpk/env/time.c deleted file mode 100644 index 1ffb28e9..00000000 --- a/code/3rd_glpk/env/time.c +++ /dev/null @@ -1,150 +0,0 @@ -/* time.c (standard time) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifdef HAVE_CONFIG_H -#include -#endif - -#include "env.h" -#include "jd.h" - -/*********************************************************************** -* NAME -* -* glp_time - determine current universal time -* -* SYNOPSIS -* -* double glp_time(void); -* -* RETURNS -* -* The routine glp_time returns the current universal time (UTC), in -* milliseconds, elapsed since 00:00:00 GMT January 1, 1970. */ - -#define EPOCH 2440588 /* = jday(1, 1, 1970) */ - -/* POSIX version ******************************************************/ - -#if defined(HAVE_SYS_TIME_H) && defined(HAVE_GETTIMEOFDAY) - -#if 0 /* 29/VI-2017 */ -#include -#include - -double glp_time(void) -{ struct timeval tv; - struct tm *tm; - int j; - double t; - gettimeofday(&tv, NULL); -#if 0 /* 29/I-2017 */ - tm = gmtime(&tv.tv_sec); -#else - tm = xgmtime(&tv.tv_sec); -#endif - j = jday(tm->tm_mday, tm->tm_mon + 1, 1900 + tm->tm_year); - xassert(j >= 0); - t = ((((double)(j - EPOCH) * 24.0 + (double)tm->tm_hour) * 60.0 + - (double)tm->tm_min) * 60.0 + (double)tm->tm_sec) * 1000.0 + - (double)(tv.tv_usec / 1000); - return t; -} -#else -#include - -double glp_time(void) -{ struct timeval tv; - double t; - gettimeofday(&tv, NULL); - t = (double)tv.tv_sec + (double)(tv.tv_usec) / 1e6; - xassert(0.0 <= t && t < 4294967296.0); - return 1000.0 * t; -} -#endif - -/* MS Windows version *************************************************/ - -#elif defined(__WOE__) - -#include - -double glp_time(void) -{ SYSTEMTIME st; - int j; - double t; - GetSystemTime(&st); - j = jday(st.wDay, st.wMonth, st.wYear); - xassert(j >= 0); - t = ((((double)(j - EPOCH) * 24.0 + (double)st.wHour) * 60.0 + - (double)st.wMinute) * 60.0 + (double)st.wSecond) * 1000.0 + - (double)st.wMilliseconds; - return t; -} - -/* portable ANSI C version ********************************************/ - -#else - -#include - -double glp_time(void) -{ time_t timer; - struct tm *tm; - int j; - double t; - timer = time(NULL); -#if 0 /* 29/I-2017 */ - tm = gmtime(&timer); -#else - tm = xgmtime(&timer); -#endif - j = jday(tm->tm_mday, tm->tm_mon + 1, 1900 + tm->tm_year); - xassert(j >= 0); - t = ((((double)(j - EPOCH) * 24.0 + (double)tm->tm_hour) * 60.0 + - (double)tm->tm_min) * 60.0 + (double)tm->tm_sec) * 1000.0; - return t; -} - -#endif - -/*********************************************************************** -* NAME -* -* glp_difftime - compute difference between two time values -* -* SYNOPSIS -* -* double glp_difftime(double t1, double t0); -* -* RETURNS -* -* The routine glp_difftime returns the difference between two time -* values t1 and t0, expressed in seconds. */ - -double glp_difftime(double t1, double t0) -{ return - (t1 - t0) / 1000.0; -} - -/* eof */ diff --git a/code/3rd_glpk/env/tls.c b/code/3rd_glpk/env/tls.c deleted file mode 100644 index 4062ee4c..00000000 --- a/code/3rd_glpk/env/tls.c +++ /dev/null @@ -1,128 +0,0 @@ -/* tls.c (thread local storage) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2001-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifdef HAVE_CONFIG_H -#include -#endif - -#include "env.h" - -#ifndef TLS -static void *tls = NULL; -#else -static TLS void *tls = NULL; -/* this option allows running multiple independent instances of GLPK in - * different threads of a multi-threaded application, in which case the - * variable tls should be placed in the Thread Local Storage (TLS); - * it is assumed that the macro TLS is previously defined to something - * like '__thread', '_Thread_local', etc. */ -#endif - -/*********************************************************************** -* NAME -* -* tls_set_ptr - store global pointer in TLS -* -* SYNOPSIS -* -* #include "env.h" -* void tls_set_ptr(void *ptr); -* -* DESCRIPTION -* -* The routine tls_set_ptr stores a pointer specified by the parameter -* ptr in the Thread Local Storage (TLS). */ - -void tls_set_ptr(void *ptr) -{ tls = ptr; - return; -} - -/*********************************************************************** -* NAME -* -* tls_get_ptr - retrieve global pointer from TLS -* -* SYNOPSIS -* -* #include "env.h" -* void *tls_get_ptr(void); -* -* RETURNS -* -* The routine tls_get_ptr returns a pointer previously stored by the -* routine tls_set_ptr. If the latter has not been called yet, NULL is -* returned. */ - -void *tls_get_ptr(void) -{ void *ptr; - ptr = tls; - return ptr; -} - -/**********************************************************************/ - -#ifdef __WOE__ - -/*** Author: Heinrich Schuchardt ***/ - -#pragma comment(lib, "user32.lib") - -#include - -#define VISTA 0x06 - -/* This is the main entry point of the DLL. */ - -BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID - lpvReserved) -{ DWORD version; - DWORD major_version; -#ifdef TLS - switch (fdwReason) - { case DLL_PROCESS_ATTACH: - /* @TODO: - * GetVersion is deprecated but the version help functions are - * not available in Visual Studio 2010. So lets use it until - * we remove the outdated Build files. */ - version = GetVersion(); - major_version = version & 0xff; - if (major_version < VISTA) - { MessageBoxA(NULL, - "The GLPK library called by this application is configur" - "ed to use thread local storage which is not fully suppo" - "rted by your version of Microsoft Windows.\n\n" - "Microsoft Windows Vista or a later version of Windows i" - "s required to run this application.", - "GLPK", MB_OK | MB_ICONERROR); - return FALSE; - } - break; - } -#endif /* TLS */ - return TRUE; -} - -#endif /* __WOE__ */ - -/* eof */ diff --git a/code/3rd_glpk/glpk.h b/code/3rd_glpk/glpk.h deleted file mode 100644 index f4e250f9..00000000 --- a/code/3rd_glpk/glpk.h +++ /dev/null @@ -1,1175 +0,0 @@ -/* glpk.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef GLPK_H -#define GLPK_H - -#include -#include - -#ifdef __cplusplus -extern "C" { -#endif - -/* library version numbers: */ -#define GLP_MAJOR_VERSION 4 -#define GLP_MINOR_VERSION 65 - -typedef struct glp_prob glp_prob; -/* LP/MIP problem object */ - -/* optimization direction flag: */ -#define GLP_MIN 1 /* minimization */ -#define GLP_MAX 2 /* maximization */ - -/* kind of structural variable: */ -#define GLP_CV 1 /* continuous variable */ -#define GLP_IV 2 /* integer variable */ -#define GLP_BV 3 /* binary variable */ - -/* type of auxiliary/structural variable: */ -#define GLP_FR 1 /* free (unbounded) variable */ -#define GLP_LO 2 /* variable with lower bound */ -#define GLP_UP 3 /* variable with upper bound */ -#define GLP_DB 4 /* double-bounded variable */ -#define GLP_FX 5 /* fixed variable */ - -/* status of auxiliary/structural variable: */ -#define GLP_BS 1 /* basic variable */ -#define GLP_NL 2 /* non-basic variable on lower bound */ -#define GLP_NU 3 /* non-basic variable on upper bound */ -#define GLP_NF 4 /* non-basic free (unbounded) variable */ -#define GLP_NS 5 /* non-basic fixed variable */ - -/* scaling options: */ -#define GLP_SF_GM 0x01 /* perform geometric mean scaling */ -#define GLP_SF_EQ 0x10 /* perform equilibration scaling */ -#define GLP_SF_2N 0x20 /* round scale factors to power of two */ -#define GLP_SF_SKIP 0x40 /* skip if problem is well scaled */ -#define GLP_SF_AUTO 0x80 /* choose scaling options automatically */ - -/* solution indicator: */ -#define GLP_SOL 1 /* basic solution */ -#define GLP_IPT 2 /* interior-point solution */ -#define GLP_MIP 3 /* mixed integer solution */ - -/* solution status: */ -#define GLP_UNDEF 1 /* solution is undefined */ -#define GLP_FEAS 2 /* solution is feasible */ -#define GLP_INFEAS 3 /* solution is infeasible */ -#define GLP_NOFEAS 4 /* no feasible solution exists */ -#define GLP_OPT 5 /* solution is optimal */ -#define GLP_UNBND 6 /* solution is unbounded */ - -typedef struct -{ /* basis factorization control parameters */ - int msg_lev; /* (not used) */ - int type; /* factorization type: */ -#if 1 /* 05/III-2014 */ -#define GLP_BF_LUF 0x00 /* plain LU-factorization */ -#define GLP_BF_BTF 0x10 /* block triangular LU-factorization */ -#endif -#define GLP_BF_FT 0x01 /* Forrest-Tomlin (LUF only) */ -#define GLP_BF_BG 0x02 /* Schur compl. + Bartels-Golub */ -#define GLP_BF_GR 0x03 /* Schur compl. + Givens rotation */ - int lu_size; /* (not used) */ - double piv_tol; /* sgf_piv_tol */ - int piv_lim; /* sgf_piv_lim */ - int suhl; /* sgf_suhl */ - double eps_tol; /* sgf_eps_tol */ - double max_gro; /* (not used) */ - int nfs_max; /* fhvint.nfs_max */ - double upd_tol; /* (not used) */ - int nrs_max; /* scfint.nn_max */ - int rs_size; /* (not used) */ - double foo_bar[38]; /* (reserved) */ -} glp_bfcp; - -typedef struct -{ /* simplex solver control parameters */ - int msg_lev; /* message level: */ -#define GLP_MSG_OFF 0 /* no output */ -#define GLP_MSG_ERR 1 /* warning and error messages only */ -#define GLP_MSG_ON 2 /* normal output */ -#define GLP_MSG_ALL 3 /* full output */ -#define GLP_MSG_DBG 4 /* debug output */ - int meth; /* simplex method option: */ -#define GLP_PRIMAL 1 /* use primal simplex */ -#define GLP_DUALP 2 /* use dual; if it fails, use primal */ -#define GLP_DUAL 3 /* use dual simplex */ - int pricing; /* pricing technique: */ -#define GLP_PT_STD 0x11 /* standard (Dantzig's rule) */ -#define GLP_PT_PSE 0x22 /* projected steepest edge */ - int r_test; /* ratio test technique: */ -#define GLP_RT_STD 0x11 /* standard (textbook) */ -#define GLP_RT_HAR 0x22 /* Harris' two-pass ratio test */ -#if 1 /* 16/III-2016 */ -#define GLP_RT_FLIP 0x33 /* long-step (flip-flop) ratio test */ -#endif - double tol_bnd; /* primal feasibility tolerance */ - double tol_dj; /* dual feasibility tolerance */ - double tol_piv; /* pivot tolerance */ - double obj_ll; /* lower objective limit */ - double obj_ul; /* upper objective limit */ - int it_lim; /* simplex iteration limit */ - int tm_lim; /* time limit, ms */ - int out_frq; /* display output frequency, ms */ - int out_dly; /* display output delay, ms */ - int presolve; /* enable/disable using LP presolver */ -#if 1 /* 11/VII-2017 (not documented yet) */ - int excl; /* exclude fixed non-basic variables */ - int shift; /* shift bounds of variables to zero */ - int aorn; /* option to use A or N: */ -#define GLP_USE_AT 1 /* use A matrix in row-wise format */ -#define GLP_USE_NT 2 /* use N matrix in row-wise format */ - double foo_bar[33]; /* (reserved) */ -#endif -} glp_smcp; - -typedef struct -{ /* interior-point solver control parameters */ - int msg_lev; /* message level (see glp_smcp) */ - int ord_alg; /* ordering algorithm: */ -#define GLP_ORD_NONE 0 /* natural (original) ordering */ -#define GLP_ORD_QMD 1 /* quotient minimum degree (QMD) */ -#define GLP_ORD_AMD 2 /* approx. minimum degree (AMD) */ -#define GLP_ORD_SYMAMD 3 /* approx. minimum degree (SYMAMD) */ - double foo_bar[48]; /* (reserved) */ -} glp_iptcp; - -typedef struct glp_tree glp_tree; -/* branch-and-bound tree */ - -typedef struct -{ /* integer optimizer control parameters */ - int msg_lev; /* message level (see glp_smcp) */ - int br_tech; /* branching technique: */ -#define GLP_BR_FFV 1 /* first fractional variable */ -#define GLP_BR_LFV 2 /* last fractional variable */ -#define GLP_BR_MFV 3 /* most fractional variable */ -#define GLP_BR_DTH 4 /* heuristic by Driebeck and Tomlin */ -#define GLP_BR_PCH 5 /* hybrid pseudocost heuristic */ - int bt_tech; /* backtracking technique: */ -#define GLP_BT_DFS 1 /* depth first search */ -#define GLP_BT_BFS 2 /* breadth first search */ -#define GLP_BT_BLB 3 /* best local bound */ -#define GLP_BT_BPH 4 /* best projection heuristic */ - double tol_int; /* mip.tol_int */ - double tol_obj; /* mip.tol_obj */ - int tm_lim; /* mip.tm_lim (milliseconds) */ - int out_frq; /* mip.out_frq (milliseconds) */ - int out_dly; /* mip.out_dly (milliseconds) */ - void (*cb_func)(glp_tree *T, void *info); - /* mip.cb_func */ - void *cb_info; /* mip.cb_info */ - int cb_size; /* mip.cb_size */ - int pp_tech; /* preprocessing technique: */ -#define GLP_PP_NONE 0 /* disable preprocessing */ -#define GLP_PP_ROOT 1 /* preprocessing only on root level */ -#define GLP_PP_ALL 2 /* preprocessing on all levels */ - double mip_gap; /* relative MIP gap tolerance */ - int mir_cuts; /* MIR cuts (GLP_ON/GLP_OFF) */ - int gmi_cuts; /* Gomory's cuts (GLP_ON/GLP_OFF) */ - int cov_cuts; /* cover cuts (GLP_ON/GLP_OFF) */ - int clq_cuts; /* clique cuts (GLP_ON/GLP_OFF) */ - int presolve; /* enable/disable using MIP presolver */ - int binarize; /* try to binarize integer variables */ - int fp_heur; /* feasibility pump heuristic */ - int ps_heur; /* proximity search heuristic */ - int ps_tm_lim; /* proxy time limit, milliseconds */ - int sr_heur; /* simple rounding heuristic */ -#if 1 /* 24/X-2015; not documented--should not be used */ - int use_sol; /* use existing solution */ - const char *save_sol; /* filename to save every new solution */ - int alien; /* use alien solver */ -#endif -#if 1 /* 16/III-2016; not documented--should not be used */ - int flip; /* use long-step dual simplex */ -#endif - double foo_bar[23]; /* (reserved) */ -} glp_iocp; - -typedef struct -{ /* additional row attributes */ - int level; - /* subproblem level at which the row was added */ - int origin; - /* row origin flag: */ -#define GLP_RF_REG 0 /* regular constraint */ -#define GLP_RF_LAZY 1 /* "lazy" constraint */ -#define GLP_RF_CUT 2 /* cutting plane constraint */ - int klass; - /* row class descriptor: */ -#define GLP_RF_GMI 1 /* Gomory's mixed integer cut */ -#define GLP_RF_MIR 2 /* mixed integer rounding cut */ -#define GLP_RF_COV 3 /* mixed cover cut */ -#define GLP_RF_CLQ 4 /* clique cut */ - double foo_bar[7]; - /* (reserved) */ -} glp_attr; - -/* enable/disable flag: */ -#define GLP_ON 1 /* enable something */ -#define GLP_OFF 0 /* disable something */ - -/* reason codes: */ -#define GLP_IROWGEN 0x01 /* request for row generation */ -#define GLP_IBINGO 0x02 /* better integer solution found */ -#define GLP_IHEUR 0x03 /* request for heuristic solution */ -#define GLP_ICUTGEN 0x04 /* request for cut generation */ -#define GLP_IBRANCH 0x05 /* request for branching */ -#define GLP_ISELECT 0x06 /* request for subproblem selection */ -#define GLP_IPREPRO 0x07 /* request for preprocessing */ - -/* branch selection indicator: */ -#define GLP_NO_BRNCH 0 /* select no branch */ -#define GLP_DN_BRNCH 1 /* select down-branch */ -#define GLP_UP_BRNCH 2 /* select up-branch */ - -/* return codes: */ -#define GLP_EBADB 0x01 /* invalid basis */ -#define GLP_ESING 0x02 /* singular matrix */ -#define GLP_ECOND 0x03 /* ill-conditioned matrix */ -#define GLP_EBOUND 0x04 /* invalid bounds */ -#define GLP_EFAIL 0x05 /* solver failed */ -#define GLP_EOBJLL 0x06 /* objective lower limit reached */ -#define GLP_EOBJUL 0x07 /* objective upper limit reached */ -#define GLP_EITLIM 0x08 /* iteration limit exceeded */ -#define GLP_ETMLIM 0x09 /* time limit exceeded */ -#define GLP_ENOPFS 0x0A /* no primal feasible solution */ -#define GLP_ENODFS 0x0B /* no dual feasible solution */ -#define GLP_EROOT 0x0C /* root LP optimum not provided */ -#define GLP_ESTOP 0x0D /* search terminated by application */ -#define GLP_EMIPGAP 0x0E /* relative mip gap tolerance reached */ -#define GLP_ENOFEAS 0x0F /* no primal/dual feasible solution */ -#define GLP_ENOCVG 0x10 /* no convergence */ -#define GLP_EINSTAB 0x11 /* numerical instability */ -#define GLP_EDATA 0x12 /* invalid data */ -#define GLP_ERANGE 0x13 /* result out of range */ - -/* condition indicator: */ -#define GLP_KKT_PE 1 /* primal equalities */ -#define GLP_KKT_PB 2 /* primal bounds */ -#define GLP_KKT_DE 3 /* dual equalities */ -#define GLP_KKT_DB 4 /* dual bounds */ -#define GLP_KKT_CS 5 /* complementary slackness */ - -/* MPS file format: */ -#define GLP_MPS_DECK 1 /* fixed (ancient) */ -#define GLP_MPS_FILE 2 /* free (modern) */ - -typedef struct -{ /* MPS format control parameters */ - int blank; - /* character code to replace blanks in symbolic names */ - char *obj_name; - /* objective row name */ - double tol_mps; - /* zero tolerance for MPS data */ - double foo_bar[17]; - /* (reserved for use in the future) */ -} glp_mpscp; - -typedef struct -{ /* CPLEX LP format control parameters */ - double foo_bar[20]; - /* (reserved for use in the future) */ -} glp_cpxcp; - -#if 1 /* 10/XII-2017 */ -typedef struct glp_prep glp_prep; -/* LP/MIP preprocessor workspace */ -#endif - -typedef struct glp_tran glp_tran; -/* MathProg translator workspace */ - -glp_prob *glp_create_prob(void); -/* create problem object */ - -void glp_set_prob_name(glp_prob *P, const char *name); -/* assign (change) problem name */ - -void glp_set_obj_name(glp_prob *P, const char *name); -/* assign (change) objective function name */ - -void glp_set_obj_dir(glp_prob *P, int dir); -/* set (change) optimization direction flag */ - -int glp_add_rows(glp_prob *P, int nrs); -/* add new rows to problem object */ - -int glp_add_cols(glp_prob *P, int ncs); -/* add new columns to problem object */ - -void glp_set_row_name(glp_prob *P, int i, const char *name); -/* assign (change) row name */ - -void glp_set_col_name(glp_prob *P, int j, const char *name); -/* assign (change) column name */ - -void glp_set_row_bnds(glp_prob *P, int i, int type, double lb, - double ub); -/* set (change) row bounds */ - -void glp_set_col_bnds(glp_prob *P, int j, int type, double lb, - double ub); -/* set (change) column bounds */ - -void glp_set_obj_coef(glp_prob *P, int j, double coef); -/* set (change) obj. coefficient or constant term */ - -void glp_set_mat_row(glp_prob *P, int i, int len, const int ind[], - const double val[]); -/* set (replace) row of the constraint matrix */ - -void glp_set_mat_col(glp_prob *P, int j, int len, const int ind[], - const double val[]); -/* set (replace) column of the constraint matrix */ - -void glp_load_matrix(glp_prob *P, int ne, const int ia[], - const int ja[], const double ar[]); -/* load (replace) the whole constraint matrix */ - -int glp_check_dup(int m, int n, int ne, const int ia[], const int ja[]); -/* check for duplicate elements in sparse matrix */ - -void glp_sort_matrix(glp_prob *P); -/* sort elements of the constraint matrix */ - -void glp_del_rows(glp_prob *P, int nrs, const int num[]); -/* delete specified rows from problem object */ - -void glp_del_cols(glp_prob *P, int ncs, const int num[]); -/* delete specified columns from problem object */ - -void glp_copy_prob(glp_prob *dest, glp_prob *prob, int names); -/* copy problem object content */ - -void glp_erase_prob(glp_prob *P); -/* erase problem object content */ - -void glp_delete_prob(glp_prob *P); -/* delete problem object */ - -const char *glp_get_prob_name(glp_prob *P); -/* retrieve problem name */ - -const char *glp_get_obj_name(glp_prob *P); -/* retrieve objective function name */ - -int glp_get_obj_dir(glp_prob *P); -/* retrieve optimization direction flag */ - -int glp_get_num_rows(glp_prob *P); -/* retrieve number of rows */ - -int glp_get_num_cols(glp_prob *P); -/* retrieve number of columns */ - -const char *glp_get_row_name(glp_prob *P, int i); -/* retrieve row name */ - -const char *glp_get_col_name(glp_prob *P, int j); -/* retrieve column name */ - -int glp_get_row_type(glp_prob *P, int i); -/* retrieve row type */ - -double glp_get_row_lb(glp_prob *P, int i); -/* retrieve row lower bound */ - -double glp_get_row_ub(glp_prob *P, int i); -/* retrieve row upper bound */ - -int glp_get_col_type(glp_prob *P, int j); -/* retrieve column type */ - -double glp_get_col_lb(glp_prob *P, int j); -/* retrieve column lower bound */ - -double glp_get_col_ub(glp_prob *P, int j); -/* retrieve column upper bound */ - -double glp_get_obj_coef(glp_prob *P, int j); -/* retrieve obj. coefficient or constant term */ - -int glp_get_num_nz(glp_prob *P); -/* retrieve number of constraint coefficients */ - -int glp_get_mat_row(glp_prob *P, int i, int ind[], double val[]); -/* retrieve row of the constraint matrix */ - -int glp_get_mat_col(glp_prob *P, int j, int ind[], double val[]); -/* retrieve column of the constraint matrix */ - -void glp_create_index(glp_prob *P); -/* create the name index */ - -int glp_find_row(glp_prob *P, const char *name); -/* find row by its name */ - -int glp_find_col(glp_prob *P, const char *name); -/* find column by its name */ - -void glp_delete_index(glp_prob *P); -/* delete the name index */ - -void glp_set_rii(glp_prob *P, int i, double rii); -/* set (change) row scale factor */ - -void glp_set_sjj(glp_prob *P, int j, double sjj); -/* set (change) column scale factor */ - -double glp_get_rii(glp_prob *P, int i); -/* retrieve row scale factor */ - -double glp_get_sjj(glp_prob *P, int j); -/* retrieve column scale factor */ - -void glp_scale_prob(glp_prob *P, int flags); -/* scale problem data */ - -void glp_unscale_prob(glp_prob *P); -/* unscale problem data */ - -void glp_set_row_stat(glp_prob *P, int i, int stat); -/* set (change) row status */ - -void glp_set_col_stat(glp_prob *P, int j, int stat); -/* set (change) column status */ - -void glp_std_basis(glp_prob *P); -/* construct standard initial LP basis */ - -void glp_adv_basis(glp_prob *P, int flags); -/* construct advanced initial LP basis */ - -void glp_cpx_basis(glp_prob *P); -/* construct Bixby's initial LP basis */ - -int glp_simplex(glp_prob *P, const glp_smcp *parm); -/* solve LP problem with the simplex method */ - -int glp_exact(glp_prob *P, const glp_smcp *parm); -/* solve LP problem in exact arithmetic */ - -void glp_init_smcp(glp_smcp *parm); -/* initialize simplex method control parameters */ - -int glp_get_status(glp_prob *P); -/* retrieve generic status of basic solution */ - -int glp_get_prim_stat(glp_prob *P); -/* retrieve status of primal basic solution */ - -int glp_get_dual_stat(glp_prob *P); -/* retrieve status of dual basic solution */ - -double glp_get_obj_val(glp_prob *P); -/* retrieve objective value (basic solution) */ - -int glp_get_row_stat(glp_prob *P, int i); -/* retrieve row status */ - -double glp_get_row_prim(glp_prob *P, int i); -/* retrieve row primal value (basic solution) */ - -double glp_get_row_dual(glp_prob *P, int i); -/* retrieve row dual value (basic solution) */ - -int glp_get_col_stat(glp_prob *P, int j); -/* retrieve column status */ - -double glp_get_col_prim(glp_prob *P, int j); -/* retrieve column primal value (basic solution) */ - -double glp_get_col_dual(glp_prob *P, int j); -/* retrieve column dual value (basic solution) */ - -int glp_get_unbnd_ray(glp_prob *P); -/* determine variable causing unboundedness */ - -#if 1 /* 08/VIII-2013; not documented yet */ -int glp_get_it_cnt(glp_prob *P); -/* get simplex solver iteration count */ -#endif - -#if 1 /* 08/VIII-2013; not documented yet */ -void glp_set_it_cnt(glp_prob *P, int it_cnt); -/* set simplex solver iteration count */ -#endif - -int glp_interior(glp_prob *P, const glp_iptcp *parm); -/* solve LP problem with the interior-point method */ - -void glp_init_iptcp(glp_iptcp *parm); -/* initialize interior-point solver control parameters */ - -int glp_ipt_status(glp_prob *P); -/* retrieve status of interior-point solution */ - -double glp_ipt_obj_val(glp_prob *P); -/* retrieve objective value (interior point) */ - -double glp_ipt_row_prim(glp_prob *P, int i); -/* retrieve row primal value (interior point) */ - -double glp_ipt_row_dual(glp_prob *P, int i); -/* retrieve row dual value (interior point) */ - -double glp_ipt_col_prim(glp_prob *P, int j); -/* retrieve column primal value (interior point) */ - -double glp_ipt_col_dual(glp_prob *P, int j); -/* retrieve column dual value (interior point) */ - -void glp_set_col_kind(glp_prob *P, int j, int kind); -/* set (change) column kind */ - -int glp_get_col_kind(glp_prob *P, int j); -/* retrieve column kind */ - -int glp_get_num_int(glp_prob *P); -/* retrieve number of integer columns */ - -int glp_get_num_bin(glp_prob *P); -/* retrieve number of binary columns */ - -int glp_intopt(glp_prob *P, const glp_iocp *parm); -/* solve MIP problem with the branch-and-bound method */ - -void glp_init_iocp(glp_iocp *parm); -/* initialize integer optimizer control parameters */ - -int glp_mip_status(glp_prob *P); -/* retrieve status of MIP solution */ - -double glp_mip_obj_val(glp_prob *P); -/* retrieve objective value (MIP solution) */ - -double glp_mip_row_val(glp_prob *P, int i); -/* retrieve row value (MIP solution) */ - -double glp_mip_col_val(glp_prob *P, int j); -/* retrieve column value (MIP solution) */ - -void glp_check_kkt(glp_prob *P, int sol, int cond, double *ae_max, - int *ae_ind, double *re_max, int *re_ind); -/* check feasibility/optimality conditions */ - -int glp_print_sol(glp_prob *P, const char *fname); -/* write basic solution in printable format */ - -int glp_read_sol(glp_prob *P, const char *fname); -/* read basic solution from text file */ - -int glp_write_sol(glp_prob *P, const char *fname); -/* write basic solution to text file */ - -int glp_print_ranges(glp_prob *P, int len, const int list[], - int flags, const char *fname); -/* print sensitivity analysis report */ - -int glp_print_ipt(glp_prob *P, const char *fname); -/* write interior-point solution in printable format */ - -int glp_read_ipt(glp_prob *P, const char *fname); -/* read interior-point solution from text file */ - -int glp_write_ipt(glp_prob *P, const char *fname); -/* write interior-point solution to text file */ - -int glp_print_mip(glp_prob *P, const char *fname); -/* write MIP solution in printable format */ - -int glp_read_mip(glp_prob *P, const char *fname); -/* read MIP solution from text file */ - -int glp_write_mip(glp_prob *P, const char *fname); -/* write MIP solution to text file */ - -int glp_bf_exists(glp_prob *P); -/* check if LP basis factorization exists */ - -int glp_factorize(glp_prob *P); -/* compute LP basis factorization */ - -int glp_bf_updated(glp_prob *P); -/* check if LP basis factorization has been updated */ - -void glp_get_bfcp(glp_prob *P, glp_bfcp *parm); -/* retrieve LP basis factorization control parameters */ - -void glp_set_bfcp(glp_prob *P, const glp_bfcp *parm); -/* change LP basis factorization control parameters */ - -int glp_get_bhead(glp_prob *P, int k); -/* retrieve LP basis header information */ - -int glp_get_row_bind(glp_prob *P, int i); -/* retrieve row index in the basis header */ - -int glp_get_col_bind(glp_prob *P, int j); -/* retrieve column index in the basis header */ - -void glp_ftran(glp_prob *P, double x[]); -/* perform forward transformation (solve system B*x = b) */ - -void glp_btran(glp_prob *P, double x[]); -/* perform backward transformation (solve system B'*x = b) */ - -int glp_warm_up(glp_prob *P); -/* "warm up" LP basis */ - -int glp_eval_tab_row(glp_prob *P, int k, int ind[], double val[]); -/* compute row of the simplex tableau */ - -int glp_eval_tab_col(glp_prob *P, int k, int ind[], double val[]); -/* compute column of the simplex tableau */ - -int glp_transform_row(glp_prob *P, int len, int ind[], double val[]); -/* transform explicitly specified row */ - -int glp_transform_col(glp_prob *P, int len, int ind[], double val[]); -/* transform explicitly specified column */ - -int glp_prim_rtest(glp_prob *P, int len, const int ind[], - const double val[], int dir, double eps); -/* perform primal ratio test */ - -int glp_dual_rtest(glp_prob *P, int len, const int ind[], - const double val[], int dir, double eps); -/* perform dual ratio test */ - -void glp_analyze_bound(glp_prob *P, int k, double *value1, int *var1, - double *value2, int *var2); -/* analyze active bound of non-basic variable */ - -void glp_analyze_coef(glp_prob *P, int k, double *coef1, int *var1, - double *value1, double *coef2, int *var2, double *value2); -/* analyze objective coefficient at basic variable */ - -#if 1 /* 10/XII-2017 */ -glp_prep *glp_npp_alloc_wksp(void); -/* allocate the preprocessor workspace */ - -void glp_npp_load_prob(glp_prep *prep, glp_prob *P, int sol, - int names); -/* load original problem instance */ - -int glp_npp_preprocess1(glp_prep *prep, int hard); -/* perform basic LP/MIP preprocessing */ - -void glp_npp_build_prob(glp_prep *prep, glp_prob *Q); -/* build resultant problem instance */ - -void glp_npp_postprocess(glp_prep *prep, glp_prob *Q); -/* postprocess solution to resultant problem */ - -void glp_npp_obtain_sol(glp_prep *prep, glp_prob *P); -/* obtain solution to original problem */ - -void glp_npp_free_wksp(glp_prep *prep); -/* free the preprocessor workspace */ -#endif - -int glp_ios_reason(glp_tree *T); -/* determine reason for calling the callback routine */ - -glp_prob *glp_ios_get_prob(glp_tree *T); -/* access the problem object */ - -void glp_ios_tree_size(glp_tree *T, int *a_cnt, int *n_cnt, - int *t_cnt); -/* determine size of the branch-and-bound tree */ - -int glp_ios_curr_node(glp_tree *T); -/* determine current active subproblem */ - -int glp_ios_next_node(glp_tree *T, int p); -/* determine next active subproblem */ - -int glp_ios_prev_node(glp_tree *T, int p); -/* determine previous active subproblem */ - -int glp_ios_up_node(glp_tree *T, int p); -/* determine parent subproblem */ - -int glp_ios_node_level(glp_tree *T, int p); -/* determine subproblem level */ - -double glp_ios_node_bound(glp_tree *T, int p); -/* determine subproblem local bound */ - -int glp_ios_best_node(glp_tree *T); -/* find active subproblem with best local bound */ - -double glp_ios_mip_gap(glp_tree *T); -/* compute relative MIP gap */ - -void *glp_ios_node_data(glp_tree *T, int p); -/* access subproblem application-specific data */ - -void glp_ios_row_attr(glp_tree *T, int i, glp_attr *attr); -/* retrieve additional row attributes */ - -int glp_ios_pool_size(glp_tree *T); -/* determine current size of the cut pool */ - -int glp_ios_add_row(glp_tree *T, - const char *name, int klass, int flags, int len, const int ind[], - const double val[], int type, double rhs); -/* add row (constraint) to the cut pool */ - -void glp_ios_del_row(glp_tree *T, int i); -/* remove row (constraint) from the cut pool */ - -void glp_ios_clear_pool(glp_tree *T); -/* remove all rows (constraints) from the cut pool */ - -int glp_ios_can_branch(glp_tree *T, int j); -/* check if can branch upon specified variable */ - -void glp_ios_branch_upon(glp_tree *T, int j, int sel); -/* choose variable to branch upon */ - -void glp_ios_select_node(glp_tree *T, int p); -/* select subproblem to continue the search */ - -int glp_ios_heur_sol(glp_tree *T, const double x[]); -/* provide solution found by heuristic */ - -void glp_ios_terminate(glp_tree *T); -/* terminate the solution process */ - -#ifdef GLP_UNDOC -int glp_gmi_cut(glp_prob *P, int j, int ind[], double val[], double - phi[]); -/* generate Gomory's mixed integer cut (core routine) */ - -int glp_gmi_gen(glp_prob *P, glp_prob *pool, int max_cuts); -/* generate Gomory's mixed integer cuts */ - -typedef struct glp_cov glp_cov; -/* cover cur generator workspace */ - -glp_cov *glp_cov_init(glp_prob *P); -/* create and initialize cover cut generator */ - -void glp_cov_gen1(glp_prob *P, glp_cov *cov, glp_prob *pool); -/* generate locally valid simple cover cuts */ - -void glp_cov_free(glp_cov *cov); -/* delete cover cut generator workspace */ - -typedef struct glp_mir glp_mir; -/* MIR cut generator workspace */ - -glp_mir *glp_mir_init(glp_prob *P); -/* create and initialize MIR cut generator */ - -int glp_mir_gen(glp_prob *P, glp_mir *mir, glp_prob *pool); -/* generate mixed integer rounding (MIR) cuts */ - -void glp_mir_free(glp_mir *mir); -/* delete MIR cut generator workspace */ - -typedef struct glp_cfg glp_cfg; -/* conflict graph descriptor */ - -glp_cfg *glp_cfg_init(glp_prob *P); -/* create and initialize conflict graph */ - -void glp_cfg_free(glp_cfg *G); -/* delete conflict graph descriptor */ - -int glp_clq_cut(glp_prob *P, glp_cfg *G, int ind[], double val[]); -/* generate clique cut from conflict graph */ -#endif /* GLP_UNDOC */ - -void glp_init_mpscp(glp_mpscp *parm); -/* initialize MPS format control parameters */ - -int glp_read_mps(glp_prob *P, int fmt, const glp_mpscp *parm, - const char *fname); -/* read problem data in MPS format */ - -int glp_write_mps(glp_prob *P, int fmt, const glp_mpscp *parm, - const char *fname); -/* write problem data in MPS format */ - -void glp_init_cpxcp(glp_cpxcp *parm); -/* initialize CPLEX LP format control parameters */ - -int glp_read_lp(glp_prob *P, const glp_cpxcp *parm, const char *fname); -/* read problem data in CPLEX LP format */ - -int glp_write_lp(glp_prob *P, const glp_cpxcp *parm, const char *fname); -/* write problem data in CPLEX LP format */ - -int glp_read_prob(glp_prob *P, int flags, const char *fname); -/* read problem data in GLPK format */ - -int glp_write_prob(glp_prob *P, int flags, const char *fname); -/* write problem data in GLPK format */ - -glp_tran *glp_mpl_alloc_wksp(void); -/* allocate the MathProg translator workspace */ - -void glp_mpl_init_rand(glp_tran *tran, int seed); -/* initialize pseudo-random number generator */ - -int glp_mpl_read_model(glp_tran *tran, const char *fname, int skip); -/* read and translate model section */ - -int glp_mpl_read_data(glp_tran *tran, const char *fname); -/* read and translate data section */ - -int glp_mpl_generate(glp_tran *tran, const char *fname); -/* generate the model */ - -void glp_mpl_build_prob(glp_tran *tran, glp_prob *prob); -/* build LP/MIP problem instance from the model */ - -int glp_mpl_postsolve(glp_tran *tran, glp_prob *prob, int sol); -/* postsolve the model */ - -void glp_mpl_free_wksp(glp_tran *tran); -/* free the MathProg translator workspace */ - -int glp_read_cnfsat(glp_prob *P, const char *fname); -/* read CNF-SAT problem data in DIMACS format */ - -int glp_check_cnfsat(glp_prob *P); -/* check for CNF-SAT problem instance */ - -int glp_write_cnfsat(glp_prob *P, const char *fname); -/* write CNF-SAT problem data in DIMACS format */ - -int glp_minisat1(glp_prob *P); -/* solve CNF-SAT problem with MiniSat solver */ - -int glp_intfeas1(glp_prob *P, int use_bound, int obj_bound); -/* solve integer feasibility problem */ - -int glp_init_env(void); -/* initialize GLPK environment */ - -const char *glp_version(void); -/* determine library version */ - -const char *glp_config(const char *option); -/* determine library configuration */ - -int glp_free_env(void); -/* free GLPK environment */ - -void glp_puts(const char *s); -/* write string on terminal */ - -void glp_printf(const char *fmt, ...); -/* write formatted output on terminal */ - -void glp_vprintf(const char *fmt, va_list arg); -/* write formatted output on terminal */ - -int glp_term_out(int flag); -/* enable/disable terminal output */ - -void glp_term_hook(int (*func)(void *info, const char *s), void *info); -/* install hook to intercept terminal output */ - -int glp_open_tee(const char *name); -/* start copying terminal output to text file */ - -int glp_close_tee(void); -/* stop copying terminal output to text file */ - -#ifndef GLP_ERRFUNC_DEFINED -#define GLP_ERRFUNC_DEFINED -typedef void (*glp_errfunc)(const char *fmt, ...); -#endif - -#define glp_error glp_error_(__FILE__, __LINE__) -glp_errfunc glp_error_(const char *file, int line); -/* display fatal error message and terminate execution */ - -#if 1 /* 07/XI-2015 */ -int glp_at_error(void); -/* check for error state */ -#endif - -#define glp_assert(expr) \ - ((void)((expr) || (glp_assert_(#expr, __FILE__, __LINE__), 1))) -void glp_assert_(const char *expr, const char *file, int line); -/* check for logical condition */ - -void glp_error_hook(void (*func)(void *info), void *info); -/* install hook to intercept abnormal termination */ - -#define glp_malloc(size) glp_alloc(1, size) -/* allocate memory block (obsolete) */ - -#define glp_calloc(n, size) glp_alloc(n, size) -/* allocate memory block (obsolete) */ - -void *glp_alloc(int n, int size); -/* allocate memory block */ - -void *glp_realloc(void *ptr, int n, int size); -/* reallocate memory block */ - -void glp_free(void *ptr); -/* free (deallocate) memory block */ - -void glp_mem_limit(int limit); -/* set memory usage limit */ - -void glp_mem_usage(int *count, int *cpeak, size_t *total, - size_t *tpeak); -/* get memory usage information */ - -double glp_time(void); -/* determine current universal time */ - -double glp_difftime(double t1, double t0); -/* compute difference between two time values */ - -typedef struct glp_graph glp_graph; -typedef struct glp_vertex glp_vertex; -typedef struct glp_arc glp_arc; - -struct glp_graph -{ /* graph descriptor */ - void *pool; /* DMP *pool; */ - /* memory pool to store graph components */ - char *name; - /* graph name (1 to 255 chars); NULL means no name is assigned - to the graph */ - int nv_max; - /* length of the vertex list (enlarged automatically) */ - int nv; - /* number of vertices in the graph, 0 <= nv <= nv_max */ - int na; - /* number of arcs in the graph, na >= 0 */ - glp_vertex **v; /* glp_vertex *v[1+nv_max]; */ - /* v[i], 1 <= i <= nv, is a pointer to i-th vertex */ - void *index; /* AVL *index; */ - /* vertex index to find vertices by their names; NULL means the - index does not exist */ - int v_size; - /* size of data associated with each vertex (0 to 256 bytes) */ - int a_size; - /* size of data associated with each arc (0 to 256 bytes) */ -}; - -struct glp_vertex -{ /* vertex descriptor */ - int i; - /* vertex ordinal number, 1 <= i <= nv */ - char *name; - /* vertex name (1 to 255 chars); NULL means no name is assigned - to the vertex */ - void *entry; /* AVLNODE *entry; */ - /* pointer to corresponding entry in the vertex index; NULL means - that either the index does not exist or the vertex has no name - assigned */ - void *data; - /* pointer to data associated with the vertex */ - void *temp; - /* working pointer */ - glp_arc *in; - /* pointer to the (unordered) list of incoming arcs */ - glp_arc *out; - /* pointer to the (unordered) list of outgoing arcs */ -}; - -struct glp_arc -{ /* arc descriptor */ - glp_vertex *tail; - /* pointer to the tail endpoint */ - glp_vertex *head; - /* pointer to the head endpoint */ - void *data; - /* pointer to data associated with the arc */ - void *temp; - /* working pointer */ - glp_arc *t_prev; - /* pointer to previous arc having the same tail endpoint */ - glp_arc *t_next; - /* pointer to next arc having the same tail endpoint */ - glp_arc *h_prev; - /* pointer to previous arc having the same head endpoint */ - glp_arc *h_next; - /* pointer to next arc having the same head endpoint */ -}; - -glp_graph *glp_create_graph(int v_size, int a_size); -/* create graph */ - -void glp_set_graph_name(glp_graph *G, const char *name); -/* assign (change) graph name */ - -int glp_add_vertices(glp_graph *G, int nadd); -/* add new vertices to graph */ - -void glp_set_vertex_name(glp_graph *G, int i, const char *name); -/* assign (change) vertex name */ - -glp_arc *glp_add_arc(glp_graph *G, int i, int j); -/* add new arc to graph */ - -void glp_del_vertices(glp_graph *G, int ndel, const int num[]); -/* delete vertices from graph */ - -void glp_del_arc(glp_graph *G, glp_arc *a); -/* delete arc from graph */ - -void glp_erase_graph(glp_graph *G, int v_size, int a_size); -/* erase graph content */ - -void glp_delete_graph(glp_graph *G); -/* delete graph */ - -void glp_create_v_index(glp_graph *G); -/* create vertex name index */ - -int glp_find_vertex(glp_graph *G, const char *name); -/* find vertex by its name */ - -void glp_delete_v_index(glp_graph *G); -/* delete vertex name index */ - -int glp_read_graph(glp_graph *G, const char *fname); -/* read graph from plain text file */ - -int glp_write_graph(glp_graph *G, const char *fname); -/* write graph to plain text file */ - -void glp_mincost_lp(glp_prob *P, glp_graph *G, int names, int v_rhs, - int a_low, int a_cap, int a_cost); -/* convert minimum cost flow problem to LP */ - -int glp_mincost_okalg(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, double *sol, int a_x, int v_pi); -/* find minimum-cost flow with out-of-kilter algorithm */ - -int glp_mincost_relax4(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, int crash, double *sol, int a_x, int a_rc); -/* find minimum-cost flow with Bertsekas-Tseng relaxation method */ - -void glp_maxflow_lp(glp_prob *P, glp_graph *G, int names, int s, - int t, int a_cap); -/* convert maximum flow problem to LP */ - -int glp_maxflow_ffalg(glp_graph *G, int s, int t, int a_cap, - double *sol, int a_x, int v_cut); -/* find maximal flow with Ford-Fulkerson algorithm */ - -int glp_check_asnprob(glp_graph *G, int v_set); -/* check correctness of assignment problem data */ - -/* assignment problem formulation: */ -#define GLP_ASN_MIN 1 /* perfect matching (minimization) */ -#define GLP_ASN_MAX 2 /* perfect matching (maximization) */ -#define GLP_ASN_MMP 3 /* maximum matching */ - -int glp_asnprob_lp(glp_prob *P, int form, glp_graph *G, int names, - int v_set, int a_cost); -/* convert assignment problem to LP */ - -int glp_asnprob_okalg(int form, glp_graph *G, int v_set, int a_cost, - double *sol, int a_x); -/* solve assignment problem with out-of-kilter algorithm */ - -int glp_asnprob_hall(glp_graph *G, int v_set, int a_x); -/* find bipartite matching of maximum cardinality */ - -double glp_cpp(glp_graph *G, int v_t, int v_es, int v_ls); -/* solve critical path problem */ - -int glp_read_mincost(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, const char *fname); -/* read min-cost flow problem data in DIMACS format */ - -int glp_write_mincost(glp_graph *G, int v_rhs, int a_low, int a_cap, - int a_cost, const char *fname); -/* write min-cost flow problem data in DIMACS format */ - -int glp_read_maxflow(glp_graph *G, int *s, int *t, int a_cap, - const char *fname); -/* read maximum flow problem data in DIMACS format */ - -int glp_write_maxflow(glp_graph *G, int s, int t, int a_cap, - const char *fname); -/* write maximum flow problem data in DIMACS format */ - -int glp_read_asnprob(glp_graph *G, int v_set, int a_cost, const char - *fname); -/* read assignment problem data in DIMACS format */ - -int glp_write_asnprob(glp_graph *G, int v_set, int a_cost, const char - *fname); -/* write assignment problem data in DIMACS format */ - -int glp_read_ccdata(glp_graph *G, int v_wgt, const char *fname); -/* read graph in DIMACS clique/coloring format */ - -int glp_write_ccdata(glp_graph *G, int v_wgt, const char *fname); -/* write graph in DIMACS clique/coloring format */ - -int glp_netgen(glp_graph *G, int v_rhs, int a_cap, int a_cost, - const int parm[1+15]); -/* Klingman's network problem generator */ - -void glp_netgen_prob(int nprob, int parm[1+15]); -/* Klingman's standard network problem instance */ - -int glp_gridgen(glp_graph *G, int v_rhs, int a_cap, int a_cost, - const int parm[1+14]); -/* grid-like network problem generator */ - -int glp_rmfgen(glp_graph *G, int *s, int *t, int a_cap, - const int parm[1+5]); -/* Goldfarb's maximum flow problem generator */ - -int glp_weak_comp(glp_graph *G, int v_num); -/* find all weakly connected components of graph */ - -int glp_strong_comp(glp_graph *G, int v_num); -/* find all strongly connected components of graph */ - -int glp_top_sort(glp_graph *G, int v_num); -/* topological sorting of acyclic digraph */ - -int glp_wclique_exact(glp_graph *G, int v_wgt, double *sol, int v_set); -/* find maximum weight clique with exact algorithm */ - -#ifdef __cplusplus -} -#endif - -#endif - -/* eof */ diff --git a/code/3rd_glpk/intopt/cfg.c b/code/3rd_glpk/intopt/cfg.c deleted file mode 100644 index ab73b2da..00000000 --- a/code/3rd_glpk/intopt/cfg.c +++ /dev/null @@ -1,409 +0,0 @@ -/* cfg.c (conflict graph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "cfg.h" -#include "env.h" - -/*********************************************************************** -* cfg_create_graph - create conflict graph -* -* This routine creates the conflict graph, which initially is empty, -* and returns a pointer to the graph descriptor. -* -* The parameter n specifies the number of *all* variables in MIP, for -* which the conflict graph will be built. -* -* The parameter nv_max specifies maximal number of vertices in the -* conflict graph. It should be the double number of binary variables -* in corresponding MIP. */ - -CFG *cfg_create_graph(int n, int nv_max) -{ CFG *G; - xassert(n >= 0); - xassert(0 <= nv_max && nv_max <= n + n); - G = talloc(1, CFG); - G->n = n; - G->pos = talloc(1+n, int); - memset(&G->pos[1], 0, n * sizeof(int)); - G->neg = talloc(1+n, int); - memset(&G->neg[1], 0, n * sizeof(int)); - G->pool = dmp_create_pool(); - G->nv_max = nv_max; - G->nv = 0; - G->ref = talloc(1+nv_max, int); - G->vptr = talloc(1+nv_max, CFGVLE *); - G->cptr = talloc(1+nv_max, CFGCLE *); - return G; -} - -/*********************************************************************** -* cfg_add_clique - add clique to conflict graph -* -* This routine adds a clique to the conflict graph. -* -* The parameter size specifies the clique size, size >= 2. Note that -* any edge can be considered as a clique of size 2. -* -* The array ind specifies vertices constituting the clique in elements -* ind[k], 1 <= k <= size: -* -* ind[k] = +j means a vertex of the conflict graph that corresponds to -* original binary variable x[j], 1 <= j <= n. -* -* ind[k] = -j means a vertex of the conflict graph that corresponds to -* complement of original binary variable x[j], 1 <= j <= n. -* -* Note that if both vertices for x[j] and (1 - x[j]) have appeared in -* the conflict graph, the routine automatically adds an edge incident -* to these vertices. */ - -static void add_edge(CFG *G, int v, int w) -{ /* add clique of size 2 */ - DMP *pool = G->pool; - int nv = G->nv; - CFGVLE **vptr = G->vptr; - CFGVLE *vle; - xassert(1 <= v && v <= nv); - xassert(1 <= w && w <= nv); - xassert(v != w); - vle = dmp_talloc(pool, CFGVLE); - vle->v = w; - vle->next = vptr[v]; - vptr[v] = vle; - vle = dmp_talloc(pool, CFGVLE); - vle->v = v; - vle->next = vptr[w]; - vptr[w] = vle; - return; -} - -void cfg_add_clique(CFG *G, int size, const int ind[]) -{ int n = G->n; - int *pos = G->pos; - int *neg = G->neg; - DMP *pool = G->pool; - int nv_max = G->nv_max; - int *ref = G->ref; - CFGVLE **vptr = G->vptr; - CFGCLE **cptr = G->cptr; - int j, k, v; - xassert(2 <= size && size <= nv_max); - /* add new vertices to the conflict graph */ - for (k = 1; k <= size; k++) - { j = ind[k]; - if (j > 0) - { /* vertex corresponds to x[j] */ - xassert(1 <= j && j <= n); - if (pos[j] == 0) - { /* no such vertex exists; add it */ - v = pos[j] = ++(G->nv); - xassert(v <= nv_max); - ref[v] = j; - vptr[v] = NULL; - cptr[v] = NULL; - if (neg[j] != 0) - { /* now both vertices for x[j] and (1 - x[j]) exist */ - add_edge(G, v, neg[j]); - } - } - } - else - { /* vertex corresponds to (1 - x[j]) */ - j = -j; - xassert(1 <= j && j <= n); - if (neg[j] == 0) - { /* no such vertex exists; add it */ - v = neg[j] = ++(G->nv); - xassert(v <= nv_max); - ref[v] = j; - vptr[v] = NULL; - cptr[v] = NULL; - if (pos[j] != 0) - { /* now both vertices for x[j] and (1 - x[j]) exist */ - add_edge(G, v, pos[j]); - } - } - } - } - /* add specified clique to the conflict graph */ - if (size == 2) - add_edge(G, - ind[1] > 0 ? pos[+ind[1]] : neg[-ind[1]], - ind[2] > 0 ? pos[+ind[2]] : neg[-ind[2]]); - else - { CFGVLE *vp, *vle; - CFGCLE *cle; - /* build list of clique vertices */ - vp = NULL; - for (k = 1; k <= size; k++) - { vle = dmp_talloc(pool, CFGVLE); - vle->v = ind[k] > 0 ? pos[+ind[k]] : neg[-ind[k]]; - vle->next = vp; - vp = vle; - } - /* attach the clique to all its vertices */ - for (k = 1; k <= size; k++) - { cle = dmp_talloc(pool, CFGCLE); - cle->vptr = vp; - v = ind[k] > 0 ? pos[+ind[k]] : neg[-ind[k]]; - cle->next = cptr[v]; - cptr[v] = cle; - } - } - return; -} - -/*********************************************************************** -* cfg_get_adjacent - get vertices adjacent to specified vertex -* -* This routine stores numbers of all vertices adjacent to specified -* vertex v of the conflict graph in locations ind[1], ..., ind[len], -* and returns len, 1 <= len <= nv-1, where nv is the total number of -* vertices in the conflict graph. -* -* Note that the conflict graph defined by this routine has neither -* self-loops nor multiple edges. */ - -int cfg_get_adjacent(CFG *G, int v, int ind[]) -{ int nv = G->nv; - int *ref = G->ref; - CFGVLE **vptr = G->vptr; - CFGCLE **cptr = G->cptr; - CFGVLE *vle; - CFGCLE *cle; - int k, w, len; - xassert(1 <= v && v <= nv); - len = 0; - /* walk thru the list of adjacent vertices */ - for (vle = vptr[v]; vle != NULL; vle = vle->next) - { w = vle->v; - xassert(1 <= w && w <= nv); - xassert(w != v); - if (ref[w] > 0) - { ind[++len] = w; - ref[w] = -ref[w]; - } - } - /* walk thru the list of incident cliques */ - for (cle = cptr[v]; cle != NULL; cle = cle->next) - { /* walk thru the list of clique vertices */ - for (vle = cle->vptr; vle != NULL; vle = vle->next) - { w = vle->v; - xassert(1 <= w && w <= nv); - if (w != v && ref[w] > 0) - { ind[++len] = w; - ref[w] = -ref[w]; - } - } - } - xassert(1 <= len && len < nv); - /* unmark vertices included in the resultant adjacency list */ - for (k = 1; k <= len; k++) - { w = ind[k]; - ref[w] = -ref[w]; - } - return len; -} - -/*********************************************************************** -* cfg_expand_clique - expand specified clique to maximal clique -* -* Given some clique in the conflict graph this routine expands it to -* a maximal clique by including in it new vertices. -* -* On entry vertex indices constituting the initial clique should be -* stored in locations c_ind[1], ..., c_ind[c_len], where c_len is the -* initial clique size. On exit the routine stores new vertex indices -* to locations c_ind[c_len+1], ..., c_ind[c_len'], where c_len' is the -* size of the maximal clique found, and returns c_len'. -* -* ALGORITHM -* -* Let G = (V, E) be a graph, C within V be a current clique to be -* expanded, and D within V \ C be a subset of vertices adjacent to all -* vertices from C. On every iteration the routine chooses some vertex -* v in D, includes it into C, and removes from D the vertex v as well -* as all vertices not adjacent to v. Initially C is empty and D = V. -* Iterations repeat until D becomes an empty set. Obviously, the final -* set C is a maximal clique in G. -* -* Now let C0 be an initial clique, and we want C0 to be a subset of -* the final maximal clique C. To provide this condition the routine -* starts constructing C by choosing only such vertices v in D, which -* are in C0, until all vertices from C0 have been included in C. May -* note that if on some iteration C0 \ C is non-empty (i.e. if not all -* vertices from C0 have been included in C), C0 \ C is a subset of D, -* because C0 is a clique. */ - -static int intersection(int d_len, int d_ind[], int d_pos[], int len, - const int ind[]) -{ /* compute intersection D := D inter W, where W is some specified - * set of vertices */ - int k, t, v, new_len; - /* walk thru vertices in W and mark vertices in D */ - for (t = 1; t <= len; t++) - { /* v in W */ - v = ind[t]; - /* determine position of v in D */ - k = d_pos[v]; - if (k != 0) - { /* v in D */ - xassert(d_ind[k] == v); - /* mark v to keep it in D */ - d_ind[k] = -v; - } - } - /* remove all unmarked vertices from D */ - new_len = 0; - for (k = 1; k <= d_len; k++) - { /* v in D */ - v = d_ind[k]; - if (v < 0) - { /* v is marked; keep it */ - v = -v; - new_len++; - d_ind[new_len] = v; - d_pos[v] = new_len; - } - else - { /* v is not marked; remove it */ - d_pos[v] = 0; - } - } - return new_len; -} - -int cfg_expand_clique(CFG *G, int c_len, int c_ind[]) -{ int nv = G->nv; - int d_len, *d_ind, *d_pos, len, *ind; - int k, v; - xassert(0 <= c_len && c_len <= nv); - /* allocate working arrays */ - d_ind = talloc(1+nv, int); - d_pos = talloc(1+nv, int); - ind = talloc(1+nv, int); - /* initialize C := 0, D := V */ - d_len = nv; - for (k = 1; k <= nv; k++) - d_ind[k] = d_pos[k] = k; - /* expand C by vertices of specified initial clique C0 */ - for (k = 1; k <= c_len; k++) - { /* v in C0 */ - v = c_ind[k]; - xassert(1 <= v && v <= nv); - /* since C0 is clique, v should be in D */ - xassert(d_pos[v] != 0); - /* W := set of vertices adjacent to v */ - len = cfg_get_adjacent(G, v, ind); - /* D := D inter W */ - d_len = intersection(d_len, d_ind, d_pos, len, ind); - /* since v not in W, now v should be not in D */ - xassert(d_pos[v] == 0); - } - /* expand C by some other vertices until D is empty */ - while (d_len > 0) - { /* v in D */ - v = d_ind[1]; - xassert(1 <= v && v <= nv); - /* note that v is adjacent to all vertices in C (by design), - * so add v to C */ - c_ind[++c_len] = v; - /* W := set of vertices adjacent to v */ - len = cfg_get_adjacent(G, v, ind); - /* D := D inter W */ - d_len = intersection(d_len, d_ind, d_pos, len, ind); - /* since v not in W, now v should be not in D */ - xassert(d_pos[v] == 0); - } - /* free working arrays */ - tfree(d_ind); - tfree(d_pos); - tfree(ind); - /* bring maximal clique to calling routine */ - return c_len; -} - -/*********************************************************************** -* cfg_check_clique - check clique in conflict graph -* -* This routine checks that vertices of the conflict graph specified -* in locations c_ind[1], ..., c_ind[c_len] constitute a clique. -* -* NOTE: for testing/debugging only. */ - -void cfg_check_clique(CFG *G, int c_len, const int c_ind[]) -{ int nv = G->nv; - int k, kk, v, w, len, *ind; - char *flag; - ind = talloc(1+nv, int); - flag = talloc(1+nv, char); - memset(&flag[1], 0, nv); - /* walk thru clique vertices */ - xassert(c_len >= 0); - for (k = 1; k <= c_len; k++) - { /* get clique vertex v */ - v = c_ind[k]; - xassert(1 <= v && v <= nv); - /* get vertices adjacent to vertex v */ - len = cfg_get_adjacent(G, v, ind); - for (kk = 1; kk <= len; kk++) - { w = ind[kk]; - xassert(1 <= w && w <= nv); - xassert(w != v); - flag[w] = 1; - } - /* check that all clique vertices other than v are adjacent - to v */ - for (kk = 1; kk <= c_len; kk++) - { w = c_ind[kk]; - xassert(1 <= w && w <= nv); - if (w != v) - xassert(flag[w]); - } - /* reset vertex flags */ - for (kk = 1; kk <= len; kk++) - flag[ind[kk]] = 0; - } - tfree(ind); - tfree(flag); - return; -} - -/*********************************************************************** -* cfg_delete_graph - delete conflict graph -* -* This routine deletes the conflict graph by freeing all the memory -* allocated to this program object. */ - -void cfg_delete_graph(CFG *G) -{ tfree(G->pos); - tfree(G->neg); - dmp_delete_pool(G->pool); - tfree(G->ref); - tfree(G->vptr); - tfree(G->cptr); - tfree(G); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/cfg.h b/code/3rd_glpk/intopt/cfg.h deleted file mode 100644 index d478f6c0..00000000 --- a/code/3rd_glpk/intopt/cfg.h +++ /dev/null @@ -1,138 +0,0 @@ -/* cfg.h (conflict graph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef CFG_H -#define CFG_H - -#include "dmp.h" - -/*********************************************************************** -* The structure CFG describes the conflict graph. -* -* Conflict graph is an undirected graph G = (V, E), where V is a set -* of vertices, E <= V x V is a set of edges. Each vertex v in V of the -* conflict graph corresponds to a binary variable z[v], which is -* either an original binary variable x[j] or its complement 1 - x[j]. -* Edge (v,w) in E means that z[v] and z[w] cannot take the value 1 at -* the same time, i.e. it defines an inequality z[v] + z[w] <= 1, which -* is assumed to be valid for original MIP. -* -* Since the conflict graph may be dense, it is stored as an union of -* its cliques rather than explicitly. */ - -#if 0 /* 08/III-2016 */ -typedef struct CFG CFG; -#else -typedef struct glp_cfg CFG; -#endif -typedef struct CFGVLE CFGVLE; -typedef struct CFGCLE CFGCLE; - -#if 0 /* 08/III-2016 */ -struct CFG -#else -struct glp_cfg -#endif -{ /* conflict graph descriptor */ - int n; - /* number of *all* variables (columns) in corresponding MIP */ - int *pos; /* int pos[1+n]; */ - /* pos[0] is not used; - * pos[j] = v, 1 <= j <= n, means that vertex v corresponds to - * original binary variable x[j], and pos[j] = 0 means that the - * conflict graph has no such vertex */ - int *neg; /* int neg[1+n]; */ - /* neg[0] is not used; - * neg[j] = v, 1 <= j <= n, means that vertex v corresponds to - * complement of original binary variable x[j], and neg[j] = 0 - * means that the conflict graph has no such vertex */ - DMP *pool; - /* memory pool to allocate elements of the conflict graph */ - int nv_max; - /* maximal number of vertices in the conflict graph */ - int nv; - /* current number of vertices in the conflict graph */ - int *ref; /* int ref[1+nv_max]; */ - /* ref[v] = j, 1 <= v <= nv, means that vertex v corresponds - * either to original binary variable x[j] or to its complement, - * i.e. either pos[j] = v or neg[j] = v */ - CFGVLE **vptr; /* CFGVLE *vptr[1+nv_max]; */ - /* vptr[v], 1 <= v <= nv, is an initial pointer to the list of - * vertices adjacent to vertex v */ - CFGCLE **cptr; /* CFGCLE *cptr[1+nv_max]; */ - /* cptr[v], 1 <= v <= nv, is an initial pointer to the list of - * cliques that contain vertex v */ -}; - -struct CFGVLE -{ /* vertex list element */ - int v; - /* vertex number, 1 <= v <= nv */ - CFGVLE *next; - /* pointer to next vertex list element */ -}; - -struct CFGCLE -{ /* clique list element */ - CFGVLE *vptr; - /* initial pointer to the list of clique vertices */ - CFGCLE *next; - /* pointer to next clique list element */ -}; - -#define cfg_create_graph _glp_cfg_create_graph -CFG *cfg_create_graph(int n, int nv_max); -/* create conflict graph */ - -#define cfg_add_clique _glp_cfg_add_clique -void cfg_add_clique(CFG *G, int size, const int ind[]); -/* add clique to conflict graph */ - -#define cfg_get_adjacent _glp_cfg_get_adjacent -int cfg_get_adjacent(CFG *G, int v, int ind[]); -/* get vertices adjacent to specified vertex */ - -#define cfg_expand_clique _glp_cfg_expand_clique -int cfg_expand_clique(CFG *G, int c_len, int c_ind[]); -/* expand specified clique to maximal clique */ - -#define cfg_check_clique _glp_cfg_check_clique -void cfg_check_clique(CFG *G, int c_len, const int c_ind[]); -/* check clique in conflict graph */ - -#define cfg_delete_graph _glp_cfg_delete_graph -void cfg_delete_graph(CFG *G); -/* delete conflict graph */ - -#define cfg_build_graph _glp_cfg_build_graph -CFG *cfg_build_graph(void /* glp_prob */ *P); -/* build conflict graph */ - -#define cfg_find_clique _glp_cfg_find_clique -int cfg_find_clique(void /* glp_prob */ *P, CFG *G, int ind[], - double *sum); -/* find maximum weight clique in conflict graph */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/intopt/cfg1.c b/code/3rd_glpk/intopt/cfg1.c deleted file mode 100644 index 80a2e834..00000000 --- a/code/3rd_glpk/intopt/cfg1.c +++ /dev/null @@ -1,703 +0,0 @@ -/* cfg1.c (conflict graph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "cfg.h" -#include "env.h" -#include "prob.h" -#include "wclique.h" -#include "wclique1.h" - -/*********************************************************************** -* cfg_build_graph - build conflict graph -* -* This routine builds the conflict graph. It analyzes the specified -* problem object to discover original and implied packing inequalities -* and adds corresponding cliques to the conflict graph. -* -* Packing inequality has the form: -* -* sum z[j] <= 1, (1) -* j in J -* -* where z[j] = x[j] or z[j] = 1 - x[j], x[j] is an original binary -* variable. Every packing inequality (1) is equivalent to a set of -* edge inequalities: -* -* z[i] + z[j] <= 1 for all i, j in J, i != j, (2) -* -* and since every edge inequality (2) defines an edge in the conflict -* graph, corresponding packing inequality (1) defines a clique. -* -* To discover packing inequalities the routine analyzes constraints -* of the specified MIP. To simplify the analysis each constraint is -* analyzed separately. The analysis is performed as follows. -* -* Let some original constraint be the following: -* -* L <= sum a[j] x[j] <= U. (3) -* -* To analyze it the routine analyzes two constraints of "not greater -* than" type: -* -* sum (-a[j]) x[j] <= -L, (4) -* -* sum (+a[j]) x[j] <= +U, (5) -* -* which are relaxations of the original constraint (3). (If, however, -* L = -oo, or U = +oo, corresponding constraint being redundant is not -* analyzed.) -* -* Let a constraint of "not greater than" type be the following: -* -* sum a[j] x[j] + sum a[j] x[j] <= b, (6) -* j in J j in J' -* -* where J is a subset of binary variables, J' is a subset of other -* (continues and non-binary integer) variables. The constraint (6) is -* is relaxed as follows, to eliminate non-binary variables: -* -* sum a[j] x[j] <= b - sum a[j] x[j] <= b', (7) -* j in J j in J' -* -* b' = sup(b - sum a[j] x[j]) = -* j in J' -* -* = b - inf(sum a[j] x[j]) = -* -* = b - sum inf(a[j] x[j]) = (8) -* -* = b - sum a[j] inf(x[j]) - sum a[j] sup(x[j]) = -* a[j]>0 a[j]<0 -* -* = b - sum a[j] l[j] - sum a[j] u[j], -* a[j]>0 a[j]<0 -* -* where l[j] and u[j] are, resp., lower and upper bounds of x[j]. -* -* Then the routine transforms the relaxed constraint containing only -* binary variables: -* -* sum a[j] x[j] <= b (9) -* -* to an equivalent 0-1 knapsack constraint as follows: -* -* sum a[j] x[j] + sum a[j] x[j] <= b ==> -* a[j]>0 a[j]<0 -* -* sum a[j] x[j] + sum a[j] (1 - x[j]) <= b ==> -* a[j]>0 a[j]<0 (10) -* -* sum (+a[j]) x[j] + sum (-a[j]) x[j] <= b + sum (-a[j]) ==> -* a[j]>0 a[j]<0 a[j]<0 -* -* sum a'[j] z[j] <= b', -* -* where a'[j] = |a[j]| > 0, and -* -* ( x[j] if a[j] > 0 -* z[j] = < -* ( 1 - x[j] if a[j] < 0 -* -* is a binary variable, which is either original binary variable x[j] -* or its complement. -* -* Finally, the routine analyzes the resultant 0-1 knapsack inequality: -* -* sum a[j] z[j] <= b, (11) -* j in J -* -* where all a[j] are positive, to discover clique inequalities (1), -* which are valid for (11) and therefore valid for (3). (It is assumed -* that the original MIP has been preprocessed, so it is not checked, -* for example, that b > 0 or that a[j] <= b.) -* -* In principle, to discover any edge inequalities valid for (11) it -* is sufficient to check whether a[i] + a[j] > b for all i, j in J, -* i < j. However, this way requires O(|J|^2) checks, so the routine -* analyses (11) in the following way, which is much more efficient in -* many practical cases. -* -* 1. Let a[p] and a[q] be two minimal coefficients: -* -* a[p] = min a[j], (12) -* -* a[q] = min a[j], j != p, (13) -* -* such that -* -* a[p] + a[q] > b. (14) -* -* This means that a[i] + a[j] > b for any i, j in J, i != j, so -* -* z[i] + z[j] <= 1 (15) -* -* are valid for (11) for any i, j in J, i != j. This case means that -* J define a clique in the conflict graph. -* -* 2. Otherwise, let a[p] and [q] be two maximal coefficients: -* -* a[p] = max a[j], (16) -* -* a[q] = max a[j], j != p, (17) -* -* such that -* -* a[p] + a[q] <= b. (18) -* -* This means that a[i] + a[j] <= b for any i, j in J, i != j, so in -* this case no valid edge inequalities for (11) exist. -* -* 3. Otherwise, let all a[j] be ordered by descending their values: -* -* a[1] >= a[2] >= ... >= a[p-1] >= a[p] >= a[p+1] >= ... (19) -* -* where p is such that -* -* a[p-1] + a[p] > b, (20) -* -* a[p] + a[p+1] <= b. (21) -* -* (May note that due to the former two cases in this case we always -* have 2 <= p <= |J|-1.) -* -* Since a[p] and a[p-1] are two minimal coefficients in the set -* J' = {1, ..., p}, J' define a clique in the conflict graph for the -* same reason as in the first case. Similarly, since a[p] and a[p+1] -* are two maximal coefficients in the set J" = {p, ..., |J|}, no edge -* inequalities exist for all i, j in J" for the same reason as in the -* second case. Thus, to discover other edge inequalities (15) valid -* for (11), the routine checks if a[i] + a[j] > b for all i in J', -* j in J", i != j. */ - -#define is_binary(j) \ - (P->col[j]->kind == GLP_IV && P->col[j]->type == GLP_DB && \ - P->col[j]->lb == 0.0 && P->col[j]->ub == 1.0) -/* check if x[j] is binary variable */ - -struct term { int ind; double val; }; -/* term a[j] * z[j] used to sort a[j]'s */ - -static int CDECL fcmp(const void *e1, const void *e2) -{ /* auxiliary routine called from qsort */ - const struct term *t1 = e1, *t2 = e2; - if (t1->val > t2->val) - return -1; - else if (t1->val < t2->val) - return +1; - else - return 0; -} - -static void analyze_ineq(glp_prob *P, CFG *G, int len, int ind[], - double val[], double rhs, struct term t[]) -{ /* analyze inequality constraint (6) */ - /* P is the original MIP - * G is the conflict graph to be built - * len is the number of terms in the constraint - * ind[1], ..., ind[len] are indices of variables x[j] - * val[1], ..., val[len] are constraint coefficients a[j] - * rhs is the right-hand side b - * t[1+len] is a working array */ - int j, k, kk, p, q, type, new_len; - /* eliminate non-binary variables; see (7) and (8) */ - new_len = 0; - for (k = 1; k <= len; k++) - { /* get index of variable x[j] */ - j = ind[k]; - if (is_binary(j)) - { /* x[j] remains in relaxed constraint */ - new_len++; - ind[new_len] = j; - val[new_len] = val[k]; - } - else if (val[k] > 0.0) - { /* eliminate non-binary x[j] in case a[j] > 0 */ - /* b := b - a[j] * l[j]; see (8) */ - type = P->col[j]->type; - if (type == GLP_FR || type == GLP_UP) - { /* x[j] has no lower bound */ - goto done; - } - rhs -= val[k] * P->col[j]->lb; - } - else /* val[j] < 0.0 */ - { /* eliminate non-binary x[j] in case a[j] < 0 */ - /* b := b - a[j] * u[j]; see (8) */ - type = P->col[j]->type; - if (type == GLP_FR || type == GLP_LO) - { /* x[j] has no upper bound */ - goto done; - } - rhs -= val[k] * P->col[j]->ub; - } - } - len = new_len; - /* now we have the constraint (9) */ - if (len <= 1) - { /* at least two terms are needed */ - goto done; - } - /* make all constraint coefficients positive; see (10) */ - for (k = 1; k <= len; k++) - { if (val[k] < 0.0) - { /* a[j] < 0; substitute x[j] = 1 - x'[j], where x'[j] is - * a complement binary variable */ - ind[k] = -ind[k]; - val[k] = -val[k]; - rhs += val[k]; - } - } - /* now we have 0-1 knapsack inequality (11) */ - /* increase the right-hand side a bit to avoid false checks due - * to rounding errors */ - rhs += 0.001 * (1.0 + fabs(rhs)); - /*** first case ***/ - /* find two minimal coefficients a[p] and a[q] */ - p = 0; - for (k = 1; k <= len; k++) - { if (p == 0 || val[p] > val[k]) - p = k; - } - q = 0; - for (k = 1; k <= len; k++) - { if (k != p && (q == 0 || val[q] > val[k])) - q = k; - } - xassert(p != 0 && q != 0 && p != q); - /* check condition (14) */ - if (val[p] + val[q] > rhs) - { /* all z[j] define a clique in the conflict graph */ - cfg_add_clique(G, len, ind); - goto done; - } - /*** second case ***/ - /* find two maximal coefficients a[p] and a[q] */ - p = 0; - for (k = 1; k <= len; k++) - { if (p == 0 || val[p] < val[k]) - p = k; - } - q = 0; - for (k = 1; k <= len; k++) - { if (k != p && (q == 0 || val[q] < val[k])) - q = k; - } - xassert(p != 0 && q != 0 && p != q); - /* check condition (18) */ - if (val[p] + val[q] <= rhs) - { /* no valid edge inequalities exist */ - goto done; - } - /*** third case ***/ - xassert(len >= 3); - /* sort terms in descending order of coefficient values */ - for (k = 1; k <= len; k++) - { t[k].ind = ind[k]; - t[k].val = val[k]; - } - qsort(&t[1], len, sizeof(struct term), fcmp); - for (k = 1; k <= len; k++) - { ind[k] = t[k].ind; - val[k] = t[k].val; - } - /* now a[1] >= a[2] >= ... >= a[len-1] >= a[len] */ - /* note that a[1] + a[2] > b and a[len-1] + a[len] <= b due two - * the former two cases */ - xassert(val[1] + val[2] > rhs); - xassert(val[len-1] + val[len] <= rhs); - /* find p according to conditions (20) and (21) */ - for (p = 2; p < len; p++) - { if (val[p] + val[p+1] <= rhs) - break; - } - xassert(p < len); - /* z[1], ..., z[p] define a clique in the conflict graph */ - cfg_add_clique(G, p, ind); - /* discover other edge inequalities */ - for (k = 1; k <= p; k++) - { for (kk = p; kk <= len; kk++) - { if (k != kk && val[k] + val[kk] > rhs) - { int iii[1+2]; - iii[1] = ind[k]; - iii[2] = ind[kk]; - cfg_add_clique(G, 2, iii); - } - } - } -done: return; -} - -CFG *cfg_build_graph(void *P_) -{ glp_prob *P = P_; - int m = P->m; - int n = P->n; - CFG *G; - int i, k, type, len, *ind; - double *val; - struct term *t; - /* create the conflict graph (number of its vertices cannot be - * greater than double number of binary variables) */ - G = cfg_create_graph(n, 2 * glp_get_num_bin(P)); - /* allocate working arrays */ - ind = talloc(1+n, int); - val = talloc(1+n, double); - t = talloc(1+n, struct term); - /* analyze constraints to discover edge inequalities */ - for (i = 1; i <= m; i++) - { type = P->row[i]->type; - if (type == GLP_LO || type == GLP_DB || type == GLP_FX) - { /* i-th row has lower bound */ - /* analyze inequality sum (-a[j]) * x[j] <= -lb */ - len = glp_get_mat_row(P, i, ind, val); - for (k = 1; k <= len; k++) - val[k] = -val[k]; - analyze_ineq(P, G, len, ind, val, -P->row[i]->lb, t); - } - if (type == GLP_UP || type == GLP_DB || type == GLP_FX) - { /* i-th row has upper bound */ - /* analyze inequality sum (+a[j]) * x[j] <= +ub */ - len = glp_get_mat_row(P, i, ind, val); - analyze_ineq(P, G, len, ind, val, +P->row[i]->ub, t); - } - } - /* free working arrays */ - tfree(ind); - tfree(val); - tfree(t); - return G; -} - -/*********************************************************************** -* cfg_find_clique - find maximum weight clique in conflict graph -* -* This routine finds a maximum weight clique in the conflict graph -* G = (V, E), where the weight of vertex v in V is the value of -* corresponding binary variable z (which is either an original binary -* variable or its complement) in the optimal solution to LP relaxation -* provided in the problem object. The goal is to find a clique in G, -* whose weight is greater than 1, in which case corresponding packing -* inequality is violated at the optimal point. -* -* On exit the routine stores vertex indices of the conflict graph -* included in the clique found to locations ind[1], ..., ind[len], and -* returns len, which is the clique size. The clique weight is stored -* in location pointed to by the parameter sum. If no clique has been -* found, the routine returns 0. -* -* Since the conflict graph may have a big number of vertices and be -* quite dense, the routine uses an induced subgraph G' = (V', E'), -* which is constructed as follows: -* -* 1. If the weight of some vertex v in V is zero (close to zero), it -* is not included in V'. Obviously, including in a clique -* zero-weight vertices does not change its weight, so if in G there -* exist a clique of a non-zero weight, in G' exists a clique of the -* same weight. This point is extremely important, because dropping -* out zero-weight vertices can be done without retrieving lists of -* adjacent vertices whose size may be very large. -* -* 2. Cumulative weight of vertex v in V is the sum of the weight of v -* and weights of all vertices in V adjacent to v. Obviously, if -* a clique includes a vertex v, the clique weight cannot be greater -* than the cumulative weight of v. Since we are interested only in -* cliques whose weight is greater than 1, vertices of V, whose -* cumulative weight is not greater than 1, are not included in V'. -* -* May note that in many practical cases the size of the induced -* subgraph G' is much less than the size of the original conflict -* graph G due to many binary variables, whose optimal values are zero -* or close to zero. For example, it may happen that |V| = 100,000 and -* |E| = 1e9 while |V'| = 50 and |E'| = 1000. */ - -struct csa -{ /* common storage area */ - glp_prob *P; - /* original MIP */ - CFG *G; - /* original conflict graph G = (V, E), |V| = nv */ - int *ind; /* int ind[1+nv]; */ - /* working array */ - /*--------------------------------------------------------------*/ - /* induced subgraph G' = (V', E') of original conflict graph */ - int nn; - /* number of vertices in V' */ - int *vtoi; /* int vtoi[1+nv]; */ - /* vtoi[v] = i, 1 <= v <= nv, means that vertex v in V is vertex - * i in V'; vtoi[v] = 0 means that vertex v is not included in - * the subgraph */ - int *itov; /* int itov[1+nv]; */ - /* itov[i] = v, 1 <= i <= nn, means that vertex i in V' is vertex - * v in V */ - double *wgt; /* double wgt[1+nv]; */ - /* wgt[i], 1 <= i <= nn, is a weight of vertex i in V', which is - * the value of corresponding binary variable in optimal solution - * to LP relaxation */ -}; - -static void build_subgraph(struct csa *csa) -{ /* build induced subgraph */ - glp_prob *P = csa->P; - int n = P->n; - CFG *G = csa->G; - int *ind = csa->ind; - int *pos = G->pos; - int *neg = G->neg; - int nv = G->nv; - int *ref = G->ref; - int *vtoi = csa->vtoi; - int *itov = csa->itov; - double *wgt = csa->wgt; - int j, k, v, w, nn, len; - double z, sum; - /* initially induced subgraph is empty */ - nn = 0; - /* walk thru vertices of original conflict graph */ - for (v = 1; v <= nv; v++) - { /* determine value of binary variable z[j] that corresponds to - * vertex v */ - j = ref[v]; - xassert(1 <= j && j <= n); - if (pos[j] == v) - { /* z[j] = x[j], where x[j] is original variable */ - z = P->col[j]->prim; - } - else if (neg[j] == v) - { /* z[j] = 1 - x[j], where x[j] is original variable */ - z = 1.0 - P->col[j]->prim; - } - else - xassert(v != v); - /* if z[j] is close to zero, do not include v in the induced - * subgraph */ - if (z < 0.001) - { vtoi[v] = 0; - continue; - } - /* calculate cumulative weight of vertex v */ - sum = z; - /* walk thru all vertices adjacent to v */ - len = cfg_get_adjacent(G, v, ind); - for (k = 1; k <= len; k++) - { /* there is an edge (v,w) in the conflict graph */ - w = ind[k]; - xassert(w != v); - /* add value of z[j] that corresponds to vertex w */ - j = ref[w]; - xassert(1 <= j && j <= n); - if (pos[j] == w) - sum += P->col[j]->prim; - else if (neg[j] == w) - sum += 1.0 - P->col[j]->prim; - else - xassert(w != w); - } - /* cumulative weight of vertex v is an upper bound of weight - * of any clique containing v; so if it not greater than 1, do - * not include v in the induced subgraph */ - if (sum < 1.010) - { vtoi[v] = 0; - continue; - } - /* include vertex v in the induced subgraph */ - nn++; - vtoi[v] = nn; - itov[nn] = v; - wgt[nn] = z; - } - /* induced subgraph has been built */ - csa->nn = nn; - return; -} - -static int sub_adjacent(struct csa *csa, int i, int adj[]) -{ /* retrieve vertices of induced subgraph adjacent to specified - * vertex */ - CFG *G = csa->G; - int nv = G->nv; - int *ind = csa->ind; - int nn = csa->nn; - int *vtoi = csa->vtoi; - int *itov = csa->itov; - int j, k, v, w, len, len1; - /* determine original vertex v corresponding to vertex i */ - xassert(1 <= i && i <= nn); - v = itov[i]; - /* retrieve vertices adjacent to vertex v in original graph */ - len1 = cfg_get_adjacent(G, v, ind); - /* keep only adjacent vertices which are in induced subgraph and - * change their numbers appropriately */ - len = 0; - for (k = 1; k <= len1; k++) - { /* there exists edge (v, w) in original graph */ - w = ind[k]; - xassert(1 <= w && w <= nv && w != v); - j = vtoi[w]; - if (j != 0) - { /* vertex w is vertex j in induced subgraph */ - xassert(1 <= j && j <= nn && j != i); - adj[++len] = j; - } - } - return len; -} - -static int find_clique(struct csa *csa, int c_ind[]) -{ /* find maximum weight clique in induced subgraph with exact - * Ostergard's algorithm */ - int nn = csa->nn; - double *wgt = csa->wgt; - int i, j, k, p, q, t, ne, nb, len, *iwt, *ind; - unsigned char *a; - xassert(nn >= 2); - /* allocate working array */ - ind = talloc(1+nn, int); - /* calculate the number of elements in lower triangle (without - * diagonal) of adjacency matrix of induced subgraph */ - ne = (nn * (nn - 1)) / 2; - /* calculate the number of bytes needed to store lower triangle - * of adjacency matrix */ - nb = (ne + (CHAR_BIT - 1)) / CHAR_BIT; - /* allocate lower triangle of adjacency matrix */ - a = talloc(nb, unsigned char); - /* fill lower triangle of adjacency matrix */ - memset(a, 0, nb); - for (p = 1; p <= nn; p++) - { /* retrieve vertices adjacent to vertex p */ - len = sub_adjacent(csa, p, ind); - for (k = 1; k <= len; k++) - { /* there exists edge (p, q) in induced subgraph */ - q = ind[k]; - xassert(1 <= q && q <= nn && q != p); - /* determine row and column indices of this edge in lower - * triangle of adjacency matrix */ - if (p > q) - i = p, j = q; - else /* p < q */ - i = q, j = p; - /* set bit a[i,j] to 1, i > j */ - t = ((i - 1) * (i - 2)) / 2 + (j - 1); - a[t / CHAR_BIT] |= - (unsigned char)(1 << ((CHAR_BIT - 1) - t % CHAR_BIT)); - } - } - /* scale vertex weights by 1000 and convert them to integers as - * required by Ostergard's algorithm */ - iwt = ind; - for (i = 1; i <= nn; i++) - { /* it is assumed that 0 <= wgt[i] <= 1 */ - t = (int)(1000.0 * wgt[i] + 0.5); - if (t < 0) - t = 0; - else if (t > 1000) - t = 1000; - iwt[i] = t; - } - /* find maximum weight clique */ - len = wclique(nn, iwt, a, c_ind); - /* free working arrays */ - tfree(ind); - tfree(a); - /* return clique size to calling routine */ - return len; -} - -static int func(void *info, int i, int ind[]) -{ /* auxiliary routine used by routine find_clique1 */ - struct csa *csa = info; - xassert(1 <= i && i <= csa->nn); - return sub_adjacent(csa, i, ind); -} - -static int find_clique1(struct csa *csa, int c_ind[]) -{ /* find maximum weight clique in induced subgraph with greedy - * heuristic */ - int nn = csa->nn; - double *wgt = csa->wgt; - int len; - xassert(nn >= 2); - len = wclique1(nn, wgt, func, csa, c_ind); - /* return clique size to calling routine */ - return len; -} - -int cfg_find_clique(void *P, CFG *G, int ind[], double *sum_) -{ int nv = G->nv; - struct csa csa; - int i, k, len; - double sum; - /* initialize common storage area */ - csa.P = P; - csa.G = G; - csa.ind = talloc(1+nv, int); - csa.nn = -1; - csa.vtoi = talloc(1+nv, int); - csa.itov = talloc(1+nv, int); - csa.wgt = talloc(1+nv, double); - /* build induced subgraph */ - build_subgraph(&csa); -#ifdef GLP_DEBUG - xprintf("nn = %d\n", csa.nn); -#endif - /* if subgraph has less than two vertices, do nothing */ - if (csa.nn < 2) - { len = 0; - sum = 0.0; - goto skip; - } - /* find maximum weight clique in induced subgraph */ -#if 1 /* FIXME */ - if (csa.nn <= 50) -#endif - { /* induced subgraph is small; use exact algorithm */ - len = find_clique(&csa, ind); - } - else - { /* induced subgraph is large; use greedy heuristic */ - len = find_clique1(&csa, ind); - } - /* do not report clique, if it has less than two vertices */ - if (len < 2) - { len = 0; - sum = 0.0; - goto skip; - } - /* convert indices of clique vertices from induced subgraph to - * original conflict graph and compute clique weight */ - sum = 0.0; - for (k = 1; k <= len; k++) - { i = ind[k]; - xassert(1 <= i && i <= csa.nn); - sum += csa.wgt[i]; - ind[k] = csa.itov[i]; - } -skip: /* free working arrays */ - tfree(csa.ind); - tfree(csa.vtoi); - tfree(csa.itov); - tfree(csa.wgt); - /* return to calling routine */ - *sum_ = sum; - return len; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/cfg2.c b/code/3rd_glpk/intopt/cfg2.c deleted file mode 100644 index 85c0705e..00000000 --- a/code/3rd_glpk/intopt/cfg2.c +++ /dev/null @@ -1,91 +0,0 @@ -/* cfg2.c (conflict graph) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "cfg.h" -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_cfg_init - create and initialize conflict graph -* -* SYNOPSIS -* -* glp_cfg *glp_cfg_init(glp_prob *P); -* -* DESCRIPTION -* -* This routine creates and initializes the conflict graph for the -* specified problem object. -* -* RETURNS -* -* The routine returns a pointer to the conflict graph descriptor. -* However, if the conflict graph is empty (no conflicts have been -* found), the routine returns NULL. */ - -glp_cfg *glp_cfg_init(glp_prob *P) -{ glp_cfg *G; - int j, n1, n2; - xprintf("Constructing conflict graph...\n"); - G = cfg_build_graph(P); - n1 = n2 = 0; - for (j = 1; j <= P->n; j++) - { if (G->pos[j]) - n1 ++; - if (G->neg[j]) - n2++; - } - if (n1 == 0 && n2 == 0) - { xprintf("No conflicts found\n"); - cfg_delete_graph(G); - G = NULL; - } - else - xprintf("Conflict graph has %d + %d = %d vertices\n", - n1, n2, G->nv); - return G; -} - -/*********************************************************************** -* NAME -* -* glp_cfg_free - delete conflict graph descriptor -* -* SYNOPSIS -* -* void glp_cfg_free(glp_cfg *G); -* -* DESCRIPTION -* -* This routine deletes the conflict graph descriptor and frees all the -* memory allocated to it. */ - -void glp_cfg_free(glp_cfg *G) -{ xassert(G != NULL); - cfg_delete_graph(G); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/clqcut.c b/code/3rd_glpk/intopt/clqcut.c deleted file mode 100644 index d3db5b39..00000000 --- a/code/3rd_glpk/intopt/clqcut.c +++ /dev/null @@ -1,134 +0,0 @@ -/* clqcut.c (clique cut generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "cfg.h" -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_clq_cut - generate clique cut from conflict graph -* -* SYNOPSIS -* -* int glp_clq_cut(glp_prob *P, glp_cfg *G, int ind[], double val[]); -* -* DESCRIPTION -* -* This routine attempts to generate a clique cut. -* -* The cut generated by the routine is the following inequality: -* -* sum a[j] * x[j] <= b, -* -* which is expected to be violated at the current basic solution. -* -* If the cut has been successfully generated, the routine stores its -* non-zero coefficients a[j] and corresponding column indices j in the -* array locations val[1], ..., val[len] and ind[1], ..., ind[len], -* where 1 <= len <= n is the number of non-zero coefficients. The -* right-hand side value b is stored in val[0], and ind[0] is set to 0. -* -* RETURNS -* -* If the cut has been successfully generated, the routine returns -* len, the number of non-zero coefficients in the cut, 1 <= len <= n. -* Otherwise, the routine returns a non-positive value. */ - -int glp_clq_cut(glp_prob *P, glp_cfg *G, int ind[], double val[]) -{ int n = P->n; - int *pos = G->pos; - int *neg = G->neg; - int nv = G->nv; - int *ref = G->ref; - int j, k, v, len; - double rhs, sum; - xassert(G->n == n); - /* find maximum weight clique in conflict graph */ - len = cfg_find_clique(P, G, ind, &sum); -#ifdef GLP_DEBUG - xprintf("len = %d; sum = %g\n", len, sum); - cfg_check_clique(G, len, ind); -#endif - /* check if clique inequality is violated */ - if (sum < 1.07) - return 0; - /* expand clique to maximal one */ - len = cfg_expand_clique(G, len, ind); -#ifdef GLP_DEBUG - xprintf("maximal clique size = %d\n", len); - cfg_check_clique(G, len, ind); -#endif - /* construct clique cut (fixed binary variables are removed, so - this cut is only locally valid) */ - rhs = 1.0; - for (j = 1; j <= n; j++) - val[j] = 0.0; - for (k = 1; k <= len; k++) - { /* v is clique vertex */ - v = ind[k]; - xassert(1 <= v && v <= nv); - /* j is number of corresponding binary variable */ - j = ref[v]; - xassert(1 <= j && j <= n); - if (pos[j] == v) - { /* v corresponds to x[j] */ - if (P->col[j]->type == GLP_FX) - { /* x[j] is fixed */ - rhs -= P->col[j]->prim; - } - else - { /* x[j] is not fixed */ - val[j] += 1.0; - } - } - else if (neg[j] == v) - { /* v corresponds to (1 - x[j]) */ - if (P->col[j]->type == GLP_FX) - { /* x[j] is fixed */ - rhs -= (1.0 - P->col[j]->prim); - } - else - { /* x[j] is not fixed */ - val[j] -= 1.0; - rhs -= 1.0; - } - } - else - xassert(v != v); - } - /* convert cut inequality to sparse format */ - len = 0; - for (j = 1; j <= n; j++) - { if (val[j] != 0.0) - { len++; - ind[len] = j; - val[len] = val[j]; - } - } - ind[0] = 0, val[0] = rhs; - return len; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/covgen.c b/code/3rd_glpk/intopt/covgen.c deleted file mode 100644 index 427c3aa8..00000000 --- a/code/3rd_glpk/intopt/covgen.c +++ /dev/null @@ -1,885 +0,0 @@ -/* covgen.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2017-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "fvs.h" -#include "ks.h" -#include "prob.h" - -struct glp_cov -{ /* cover cut generator working area */ - int n; - /* number of columns (variables) */ - glp_prob *set; - /* set of globally valid 0-1 knapsack inequalities chosen from - * the root problem; each inequality is either original row or - * its relaxation (surrogate 0-1 knapsack) which is constructed - * by substitution of lower/upper single/variable bounds for - * continuous and general integer (non-binary) variables */ -}; - -struct bnd -{ /* simple or variable bound */ - /* if z = 0, it is a simple bound x >= or <= b; if b = -DBL_MAX - * (b = +DBL_MAX), x has no lower (upper) bound; otherwise, if - * z != 0, it is a variable bound x >= or <= a * z + b */ - int z; - /* number of binary variable or 0 */ - double a, b; - /* bound parameters */ -}; - -struct csa -{ /* common storage area */ - glp_prob *P; - /* original (root) MIP */ - struct bnd *l; /* struct bnd l[1+P->n]; */ - /* lower simple/variable bounds of variables */ - struct bnd *u; /* struct bnd u[1+P->n]; */ - /* upper simple/variable bounds of variables */ - glp_prob *set; - /* see struct glp_cov above */ -}; - -/*********************************************************************** -* init_bounds - initialize bounds of variables with simple bounds -* -* This routine initializes lower and upper bounds of all variables -* with simple bounds specified in the original mip. */ - -static void init_bounds(struct csa *csa) -{ glp_prob *P = csa->P; - struct bnd *l = csa->l, *u = csa->u; - int j; - for (j = 1; j <= P->n; j++) - { l[j].z = u[j].z = 0; - l[j].a = u[j].a = 0; - l[j].b = glp_get_col_lb(P, j); - u[j].b = glp_get_col_ub(P, j); - } - return; -} - -/*********************************************************************** -* check_vb - check variable bound -* -* This routine checks if the specified i-th row has the form -* -* a1 * x + a2 * z >= or <= rhs, (1) -* -* where x is a non-fixed continuous or general integer variable, and -* z is a binary variable. If it is, the routine converts the row to -* the following variable lower/upper bound (VLB/VUB) of x: -* -* x >= or <= a * z + b, (2) -* -* where a = - a2 / a1, b = rhs / a1. Note that the inequality type is -* changed to opposite one when a1 < 0. -* -* If the row is identified as a variable bound, the routine returns -* GLP_LO for VLB or GLP_UP for VUB and provides the reference numbers -* of variables x and z and values of a and b. Otherwise, the routine -* returns zero. */ - -static int check_vb(struct csa *csa, int i, int *x, int *z, double *a, - double *b) -{ glp_prob *P = csa->P; - GLPROW *row; - GLPAIJ *a1, *a2; - int type; - double rhs; - xassert(1 <= i && i <= P->m); - row = P->row[i]; - /* check row type */ - switch (row->type) - { case GLP_LO: - case GLP_UP: - break; - default: - return 0; - } - /* take first term of the row */ - a1 = row->ptr; - if (a1 == NULL) - return 0; - /* take second term of the row */ - a2 = a1->r_next; - if (a2 == NULL) - return 0; - /* there should be exactly two terms in the row */ - if (a2->r_next != NULL) - return 0; - /* if first term is a binary variable, swap the terms */ - if (glp_get_col_kind(P, a1->col->j) == GLP_BV) - { GLPAIJ *a; - a = a1, a1 = a2, a2 = a; - } - /* now first term should be a non-fixed continuous or general - * integer variable */ - if (a1->col->type == GLP_FX) - return 0; - if (glp_get_col_kind(P, a1->col->j) == GLP_BV) - return 0; - /* and second term should be a binary variable */ - if (glp_get_col_kind(P, a2->col->j) != GLP_BV) - return 0; - /* VLB/VUB row has been identified */ - switch (row->type) - { case GLP_LO: - type = a1->val > 0 ? GLP_LO : GLP_UP; - rhs = row->lb; - break; - case GLP_UP: - type = a1->val > 0 ? GLP_UP : GLP_LO; - rhs = row->ub; - break; - default: - xassert(type != type); - } - *x = a1->col->j; - *z = a2->col->j; - *a = - a2->val / a1->val; - *b = rhs / a1->val; - return type; -} - -/*********************************************************************** -* set_vb - set variable bound -* -* This routine sets lower or upper variable bound specified as -* -* x >= a * z + b (type = GLP_LO) -* -* x <= a * z + b (type = GLP_UP) */ - -static void set_vb(struct csa *csa, int type, int x, int z, double a, - double b) -{ glp_prob *P = csa->P; - struct bnd *l = csa->l, *u = csa->u; - xassert(glp_get_col_type(P, x) != GLP_FX); - xassert(glp_get_col_kind(P, x) != GLP_BV); - xassert(glp_get_col_kind(P, z) == GLP_BV); - xassert(a != 0); - switch (type) - { case GLP_LO: - /* FIXME: check existing simple lower bound? */ - l[x].z = z, l[x].a = a, l[x].b = b; - break; - case GLP_UP: - /* FIXME: check existing simple upper bound? */ - u[x].z = z, u[x].a = a, u[x].b = b; - break; - default: - xassert(type != type); - } - return; -} - -/*********************************************************************** -* obtain_vbs - obtain and set variable bounds -* -* This routine walks thru all rows of the original mip, identifies -* rows specifying variable lower/upper bounds, and sets these bounds -* for corresponding (non-binary) variables. */ - -static void obtain_vbs(struct csa *csa) -{ glp_prob *P = csa->P; - int i, x, z, type, save; - double a, b; - for (i = 1; i <= P->m; i++) - { switch (P->row[i]->type) - { case GLP_FR: - break; - case GLP_LO: - case GLP_UP: - type = check_vb(csa, i, &x, &z, &a, &b); - if (type) - set_vb(csa, type, x, z, a, b); - break; - case GLP_DB: - case GLP_FX: - /* double-side inequality l <= ... <= u and equality - * ... = l = u are considered as two single inequalities - * ... >= l and ... <= u */ - save = P->row[i]->type; - P->row[i]->type = GLP_LO; - type = check_vb(csa, i, &x, &z, &a, &b); - if (type) - set_vb(csa, type, x, z, a, b); - P->row[i]->type = GLP_UP; - type = check_vb(csa, i, &x, &z, &a, &b); - if (type) - set_vb(csa, type, x, z, a, b); - P->row[i]->type = save; - break; - default: - xassert(P != P); - } - } - return; -} - -/*********************************************************************** -* add_term - add term to sparse vector -* -* This routine computes the following linear combination: -* -* v := v + a * e[j], -* -* where v is a sparse vector in full storage format, a is a non-zero -* scalar, e[j] is j-th column of unity matrix. */ - -static void add_term(FVS *v, int j, double a) -{ xassert(1 <= j && j <= v->n); - xassert(a != 0); - if (v->vec[j] == 0) - { /* create j-th component */ - v->nnz++; - xassert(v->nnz <= v->n); - v->ind[v->nnz] = j; - } - /* perform addition */ - v->vec[j] += a; - if (fabs(v->vec[j]) < 1e-9 * (1 + fabs(a))) - { /* remove j-th component */ - v->vec[j] = DBL_MIN; - } - return; -} - -/*********************************************************************** -* build_ks - build "0-1 knapsack" inequality -* -* Given an inequality of "not greater" type: -* -* sum{j in 1..n} a[j]*x[j] <= b, (1) -* -* this routine attempts to transform it to equivalent or relaxed "0-1 -* knapsack" inequality that contains only binary variables. -* -* If x[j] is a binary variable, the term a[j]*x[j] is not changed. -* Otherwise, if x[j] is a continuous or integer non-binary variable, -* it is replaced by its lower (if a[j] > 0) or upper (if a[j] < 0) -* single or variable bound. In the latter case, if x[j] is a non-fixed -* variable, this results in a relaxation of original inequality known -* as "surrogate knapsack". Thus, if the specified inequality is valid -* for the original mip, the resulting inequality is also valid. -* -* Note that in both source and resulting inequalities coefficients -* a[j] can have any sign. -* -* On entry to the routine the source inequality is specified by the -* parameters n, ind (contains original numbers of x[j]), a, and b. The -* parameter v is a working sparse vector whose components are assumed -* to be zero. -* -* On exit the routine stores the resulting "0-1 knapsack" inequality -* in the parameters ind, a, and b, and returns n which is the number -* of terms in the resulting inequality. Zero content of the vector v -* is restored before exit. -* -* If the resulting inequality cannot be constructed due to missing -* lower/upper bounds of some variable, the routine returns a negative -* value. */ - -static int build_ks(struct csa *csa, int n, int ind[], double a[], - double *b, FVS *v) -{ glp_prob *P = csa->P; - struct bnd *l = csa->l, *u = csa->u; - int j, k; - /* check that v = 0 */ -#ifdef GLP_DEBUG - fvs_check_vec(v); -#endif - xassert(v->nnz == 0); - /* walk thru terms of original inequality */ - for (j = 1; j <= n; j++) - { /* process term a[j]*x[j] */ - k = ind[j]; /* original number of x[j] in mip */ - if (glp_get_col_kind(P, k) == GLP_BV) - { /* x[j] is a binary variable */ - /* include its term into resulting inequality */ - add_term(v, k, a[j]); - } - else if (a[j] > 0) - { /* substitute x[j] by its lower bound */ - if (l[k].b == -DBL_MAX) - { /* x[j] has no lower bound */ - n = -1; - goto skip; - } - else if (l[k].z == 0) - { /* x[j] has simple lower bound */ - *b -= a[j] * l[k].b; - } - else - { /* x[j] has variable lower bound (a * z + b) */ - add_term(v, l[k].z, a[j] * l[k].a); - *b -= a[j] * l[k].b; - } - } - else /* a[j] < 0 */ - { /* substitute x[j] by its upper bound */ - if (u[k].b == +DBL_MAX) - { /* x[j] has no upper bound */ - n = -1; - goto skip; - } - else if (u[k].z == 0) - { /* x[j] has simple upper bound */ - *b -= a[j] * u[k].b; - } - else - { /* x[j] has variable upper bound (a * z + b) */ - add_term(v, u[k].z, a[j] * u[k].a); - *b -= a[j] * u[k].b; - } - } - } - /* replace tiny coefficients by exact zeros (see add_term) */ - fvs_adjust_vec(v, 2 * DBL_MIN); - /* copy terms of resulting inequality */ - xassert(v->nnz <= n); - n = v->nnz; - for (j = 1; j <= n; j++) - { ind[j] = v->ind[j]; - a[j] = v->vec[ind[j]]; - } -skip: /* restore zero content of v */ - fvs_clear_vec(v); - return n; -} - -/*********************************************************************** -* can_be_active - check if inequality can be active -* -* This routine checks if the specified "0-1 knapsack" inequality -* -* sum{j in 1..n} a[j]*x[j] <= b -* -* can be active. If so, the routine returns true, otherwise false. */ - -static int can_be_active(int n, const double a[], double b) -{ int j; - double s; - s = 0; - for (j = 1; j <= n; j++) - { if (a[j] > 0) - s += a[j]; - } - return s > b + .001 * (1 + fabs(b)); -} - -/*********************************************************************** -* is_sos_ineq - check if inequality is packing (SOS) constraint -* -* This routine checks if the specified "0-1 knapsack" inequality -* -* sum{j in 1..n} a[j]*x[j] <= b (1) -* -* is equivalent to packing inequality (Padberg calls such inequalities -* special ordered set or SOS constraints) -* -* sum{j in J'} x[j] - sum{j in J"} x[j] <= 1 - |J"|. (2) -* -* If so, the routine returns true, otherwise false. -* -* Note that if X is a set of feasible binary points satisfying to (2), -* its convex hull conv(X) equals to the set of feasible points of LP -* relaxation of (2), which is a n-dimensional simplex, so inequalities -* (2) are useless for generating cover cuts (due to unimodularity). -* -* ALGORITHM -* -* First, we make all a[j] positive by complementing x[j] = 1 - x'[j] -* in (1). This is performed implicitly (i.e. actually the array a is -* not changed), but b is replaced by b - sum{j : a[j] < 0}. -* -* Then we find two smallest coefficients a[p] = min{j in 1..n} a[j] -* and a[q] = min{j in 1..n : j != p} a[j]. It is obvious that if -* a[p] + a[q] > b, then a[i] + a[j] > b for all i != j, from which it -* follows that x[i] + x[j] <= 1 for all i != j. But the latter means -* that the original inequality (with all a[j] > 0) is equivalent to -* packing inequality -* -* sum{j in 1..n} x[j] <= 1. (3) -* -* Returning to original (uncomplemented) variables x'[j] = 1 - x[j] -* we have that the original inequality is equivalent to (2), where -* J' = {j : a[j] > 0} and J" = {j : a[j] < 0}. */ - -static int is_sos_ineq(int n, const double a[], double b) -{ int j, p, q; - xassert(n >= 2); - /* compute b := b - sum{j : a[j] < 0} */ - for (j = 1; j <= n; j++) - { if (a[j] < 0) - b -= a[j]; - } - /* find a[p] = min{j in 1..n} a[j] */ - p = 1; - for (j = 2; j <= n; j++) - { if (fabs(a[p]) > fabs(a[j])) - p = j; - } - /* find a[q] = min{j in 1..n : j != p} a[j] */ - q = 0; - for (j = 1; j <= n; j++) - { if (j != p) - { if (q == 0 || fabs(a[q]) > fabs(a[j])) - q = j; - } - } - xassert(q != 0); - /* check condition a[p] + a[q] > b */ - return fabs(a[p]) + fabs(a[q]) > b + .001 * (1 + fabs(b)); -} - -/*********************************************************************** -* process_ineq - basic inequality processing -* -* This routine performs basic processing of an inequality of "not -* greater" type -* -* sum{j in 1..n} a[j]*x[j] <= b -* -* specified by the parameters, n, ind, a, and b. -* -* If the inequality can be transformed to "0-1 knapsack" ineqiality -* suitable for generating cover cuts, the routine adds it to the set -* of "0-1 knapsack" inequalities. -* -* Note that the arrays ind and a are not saved on exit. */ - -static void process_ineq(struct csa *csa, int n, int ind[], double a[], - double b, FVS *v) -{ int i; - /* attempt to transform the specified inequality to equivalent or - * relaxed "0-1 knapsack" inequality */ - n = build_ks(csa, n, ind, a, &b, v); - if (n <= 1) - { /* uninteresting inequality (in principle, such inequalities - * should be removed by the preprocessor) */ - goto done; - } - if (!can_be_active(n, a, b)) - { /* inequality is redundant (i.e. cannot be active) */ - goto done; - } - if (is_sos_ineq(n, a, b)) - { /* packing (SOS) inequality is useless for generating cover - * cuts; currently such inequalities are just ignored */ - goto done; - } - /* add resulting "0-1 knapsack" inequality to the set */ - i = glp_add_rows(csa->set, 1); - glp_set_mat_row(csa->set, i, n, ind, a); - glp_set_row_bnds(csa->set, i, GLP_UP, b, b); -done: return; -} - -/**********************************************************************/ - -glp_cov *glp_cov_init(glp_prob *P) -{ /* create and initialize cover cut generator */ - glp_cov *cov; - struct csa csa; - int i, k, len, *ind; - double rhs, *val; - FVS fvs; - csa.P = P; - csa.l = talloc(1+P->n, struct bnd); - csa.u = talloc(1+P->n, struct bnd); - csa.set = glp_create_prob(); - glp_add_cols(csa.set, P->n); - /* initialize bounds of variables with simple bounds */ - init_bounds(&csa); - /* obtain and set variable bounds */ - obtain_vbs(&csa); - /* allocate working arrays */ - ind = talloc(1+P->n, int); - val = talloc(1+P->n, double); - fvs_alloc_vec(&fvs, P->n); - /* process all rows of the root mip */ - for (i = 1; i <= P->m; i++) - { switch (P->row[i]->type) - { case GLP_FR: - break; - case GLP_LO: - /* obtain row of ">=" type */ - len = glp_get_mat_row(P, i, ind, val); - rhs = P->row[i]->lb; - /* transforms it to row of "<=" type */ - for (k = 1; k <= len; k++) - val[k] = - val[k]; - rhs = - rhs; - /* process the row */ - process_ineq(&csa, len, ind, val, rhs, &fvs); - break; - case GLP_UP: - /* obtain row of "<=" type */ - len = glp_get_mat_row(P, i, ind, val); - rhs = P->row[i]->ub; - /* and process it */ - process_ineq(&csa, len, ind, val, rhs, &fvs); - break; - case GLP_DB: - case GLP_FX: - /* double-sided inequalitiy and equality constraints are - * processed as two separate inequalities */ - /* obtain row as if it were of ">=" type */ - len = glp_get_mat_row(P, i, ind, val); - rhs = P->row[i]->lb; - /* transforms it to row of "<=" type */ - for (k = 1; k <= len; k++) - val[k] = - val[k]; - rhs = - rhs; - /* and process it */ - process_ineq(&csa, len, ind, val, rhs, &fvs); - /* obtain the same row as if it were of "<=" type */ - len = glp_get_mat_row(P, i, ind, val); - rhs = P->row[i]->ub; - /* and process it */ - process_ineq(&csa, len, ind, val, rhs, &fvs); - break; - default: - xassert(P != P); - } - } - /* free working arrays */ - tfree(ind); - tfree(val); - fvs_check_vec(&fvs); - fvs_free_vec(&fvs); - /* the set of "0-1 knapsack" inequalities has been built */ - if (csa.set->m == 0) - { /* the set is empty */ - xprintf("No 0-1 knapsack inequalities detected\n"); - cov = NULL; - glp_delete_prob(csa.set); - } - else - { /* create the cover cut generator working area */ - xprintf("Number of 0-1 knapsack inequalities = %d\n", - csa.set->m); - cov = talloc(1, glp_cov); - cov->n = P->n; - cov->set = csa.set; -#if 0 - glp_write_lp(cov->set, 0, "set.lp"); -#endif - } - tfree(csa.l); - tfree(csa.u); - return cov; -} - -/*********************************************************************** -* solve_ks - solve 0-1 knapsack problem -* -* This routine finds (sub)optimal solution to 0-1 knapsack problem: -* -* maximize z = sum{j in 1..n} c[j]x[j] (1) -* -* s.t. sum{j in 1..n} a[j]x[j] <= b (2) -* -* x[j] in {0, 1} for all j in 1..n (3) -* -* It is assumed that the instance is non-normalized, i.e. parameters -* a, b, and c may have any sign. -* -* On exit the routine stores the (sub)optimal point found in locations -* x[1], ..., x[n] and returns the optimal objective value. However, if -* the instance is infeasible, the routine returns INT_MIN. */ - -static int solve_ks(int n, const int a[], int b, const int c[], - char x[]) -{ int z; - /* surprisingly, even for some small instances (n = 50-100) - * MT1 routine takes too much time, so it is used only for tiny - * instances */ - if (n <= 16) -#if 0 - z = ks_enum(n, a, b, c, x); -#else - z = ks_mt1(n, a, b, c, x); -#endif - else - z = ks_greedy(n, a, b, c, x); - return z; -} - -/*********************************************************************** -* simple_cover - find simple cover cut -* -* Given a 0-1 knapsack inequality (which may be globally as well as -* locally valid) -* -* sum{j in 1..n} a[j]x[j] <= b, (1) -* -* where all x[j] are binary variables and all a[j] are positive, and -* a fractional point x~{j in 1..n}, which is feasible to LP relaxation -* of (1), this routine attempts to find a simple cover inequality -* -* sum{j in C} (1 - x[j]) >= 1, (2) -* -* which is valid for (1) and violated at x~. -* -* Actually, the routine finds a cover C, i.e. a subset of {1, ..., n} -* such that -* -* sum{j in C} a[j] > b, (3) -* -* and which minimizes the left-hand side of (2) at x~ -* -* zeta = sum{j in C} (1 - x~[j]). (4) -* -* On exit the routine stores the characteritic vector z{j in 1..n} -* of the cover found (i.e. z[j] = 1 means j in C, and z[j] = 0 means -* j not in C), and returns corresponding minimal value of zeta (4). -* However, if no cover is found, the routine returns DBL_MAX. -* -* ALGORITHM -* -* The separation problem (3)-(4) is converted to 0-1 knapsack problem -* as follows. -* -* First, note that the constraint (3) is equivalent to -* -* sum{j in 1..n} a[j]z[j] >= b + eps, (5) -* -* where eps > 0 is a sufficiently small number (in case of integral -* a and b we may take eps = 1). Multiplying both sides of (5) by (-1) -* gives -* -* sum{j in 1..n} (-a[j])z[j] <= - b - eps. (6) -* -* To make all coefficients in (6) positive, z[j] is complemented by -* substitution z[j] = 1 - z'[j] that finally gives -* -* sum{j in 1..n} a[j]z'[j] <= sum{j in 1..n} a[j] - b - eps. (7) -* -* Minimization of zeta (4) is equivalent to maximization of -* -* -zeta = sum{j in 1..n} (x~[j] - 1)z[j]. (8) -* -* Substitution z[j] = 1 - z'[j] gives -* -* -zeta = sum{j in 1..n} (1 - x~[j])z'[j] - zeta0, (9) -* -* where zeta0 = sum{j in 1..n} (1 - x~[j]) is a constant term. -* -* Thus, the 0-1 knapsack problem to be solved is the following: -* -* maximize -* -* -zeta = sum{j in 1..n} (1 - x~[j])z'[j] - zeta0 (10) -* -* subject to -* -* sum{j in 1..n} a[j]z'[j] <= sum{j in 1..n} a[j] - b - eps (11) -* -* z'[j] in {0,1} for all j = 1,...,n (12) -* -* (The constant term zeta0 doesn't affect the solution, so it can be -* dropped.) */ - -static double simple_cover(int n, const double a[], double b, const - double x[], char z[]) -{ int j, *aa, bb, *cc; - double max_aj, min_aj, s, eps; - xassert(n >= 3); - /* allocate working arrays */ - aa = talloc(1+n, int); - cc = talloc(1+n, int); - /* compute max{j in 1..n} a[j] and min{j in 1..n} a[j] */ - max_aj = 0, min_aj = DBL_MAX; - for (j = 1; j <= n; j++) - { xassert(a[j] > 0); - if (max_aj < a[j]) - max_aj = a[j]; - if (min_aj > a[j]) - min_aj = a[j]; - } - /* scale and round constraint parameters to make them integral; - * note that we make the resulting inequality stronger than (11), - * so a[j]'s are rounded up while rhs is rounded down */ - s = 0; - for (j = 1; j <= n; j++) - { s += a[j]; - aa[j] = ceil(a[j] / max_aj * 1000); - } - bb = floor((s - b) / max_aj * 1000) - 1; - /* scale and round obj. coefficients to make them integral; - * again we make the objective function stronger than (10), so - * the coefficients are rounded down */ - for (j = 1; j <= n; j++) - { xassert(0 <= x[j] && x[j] <= 1); - cc[j] = floor((1 - x[j]) * 1000); - } - /* solve separation problem */ - if (solve_ks(n, aa, bb, cc, z) == INT_MIN) - { /* no cover exists */ - s = DBL_MAX; - goto skip; - } - /* determine z[j] = 1 - z'[j] */ - for (j = 1; j <= n; j++) - { xassert(z[j] == 0 || z[j] == 1); - z[j] ^= 1; - } - /* check condition (11) for original (non-scaled) parameters */ - s = 0; - for (j = 1; j <= n; j++) - { if (z[j]) - s += a[j]; - } - eps = 0.01 * (min_aj >= 1 ? min_aj : 1); - if (!(s >= b + eps)) - { /* no cover found within a precision req'd */ - s = DBL_MAX; - goto skip; - } - /* compute corresponding zeta (4) for cover found */ - s = 0; - for (j = 1; j <= n; j++) - { if (z[j]) - s += 1 - x[j]; - } -skip: /* free working arrays */ - tfree(aa); - tfree(cc); - return s; -} - -/**********************************************************************/ - -void glp_cov_gen1(glp_prob *P, glp_cov *cov, glp_prob *pool) -{ /* generate locally valid simple cover cuts */ - int i, k, len, new_len, *ind; - double *val, rhs, *x, zeta; - char *z; - xassert(P->n == cov->n && P->n == cov->set->n); - xassert(glp_get_status(P) == GLP_OPT); - /* allocate working arrays */ - ind = talloc(1+P->n, int); - val = talloc(1+P->n, double); - x = talloc(1+P->n, double); - z = talloc(1+P->n, char); - /* walk thru 0-1 knapsack inequalities */ - for (i = 1; i <= cov->set->m; i++) - { /* retrieve 0-1 knapsack inequality */ - len = glp_get_mat_row(cov->set, i, ind, val); - rhs = glp_get_row_ub(cov->set, i); - xassert(rhs != +DBL_MAX); - /* FIXME: skip, if slack is too large? */ - /* substitute and eliminate binary variables which have been - * fixed in the current subproblem (this makes the inequality - * only locally valid) */ - new_len = 0; - for (k = 1; k <= len; k++) - { if (glp_get_col_type(P, ind[k]) == GLP_FX) - rhs -= val[k] * glp_get_col_prim(P, ind[k]); - else - { new_len++; - ind[new_len] = ind[k]; - val[new_len] = val[k]; - } - } - len = new_len; - /* we need at least 3 binary variables in the inequality */ - if (len <= 2) - continue; - /* obtain values of binary variables from optimal solution to - * LP relaxation of current subproblem */ - for (k = 1; k <= len; k++) - { xassert(glp_get_col_kind(P, ind[k]) == GLP_BV); - x[k] = glp_get_col_prim(P, ind[k]); - if (x[k] < 0.00001) - x[k] = 0; - else if (x[k] > 0.99999) - x[k] = 1; - /* if val[k] < 0, perform substitution x[k] = 1 - x'[k] to - * make all coefficients positive */ - if (val[k] < 0) - { ind[k] = - ind[k]; /* x[k] is complemented */ - val[k] = - val[k]; - rhs += val[k]; - x[k] = 1 - x[k]; - } - } - /* find locally valid simple cover cut */ - zeta = simple_cover(len, val, rhs, x, z); - if (zeta > 0.95) - { /* no violation or insufficient violation; see (2) */ - continue; - } - /* construct cover inequality (2) for the cover found, which - * for original binary variables x[k] is equivalent to: - * sum{k in C'} x[k] + sum{k in C"} x'[k] <= |C| - 1 - * or - * sum{k in C'} x[k] + sum{k in C"} (1 - x[k]) <= |C| - 1 - * or - * sum{k in C'} x[k] - sum{k in C"} x[k] <= |C'| - 1 - * since |C| - |C"| = |C'| */ - new_len = 0; - rhs = -1; - for (k = 1; k <= len; k++) - { if (z[k]) - { new_len++; - if (ind[k] > 0) - { ind[new_len] = +ind[k]; - val[new_len] = +1; - rhs++; - } - else /* ind[k] < 0 */ - { ind[new_len] = -ind[k]; - val[new_len] = -1; - } - } - } - len = new_len; - /* add the cover inequality to the local cut pool */ - k = glp_add_rows(pool, 1); - glp_set_mat_row(pool, k, len, ind, val); - glp_set_row_bnds(pool, k, GLP_UP, rhs, rhs); - } - /* free working arrays */ - tfree(ind); - tfree(val); - tfree(x); - tfree(z); - return; -} - -/**********************************************************************/ - -void glp_cov_free(glp_cov *cov) -{ /* delete cover cut generator workspace */ - xassert(cov != NULL); - glp_delete_prob(cov->set); - tfree(cov); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/fpump.c b/code/3rd_glpk/intopt/fpump.c deleted file mode 100644 index 0bdd6d4e..00000000 --- a/code/3rd_glpk/intopt/fpump.c +++ /dev/null @@ -1,360 +0,0 @@ -/* fpump.c (feasibility pump heuristic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" -#include "rng.h" - -/*********************************************************************** -* NAME -* -* ios_feas_pump - feasibility pump heuristic -* -* SYNOPSIS -* -* #include "glpios.h" -* void ios_feas_pump(glp_tree *T); -* -* DESCRIPTION -* -* The routine ios_feas_pump is a simple implementation of the Feasi- -* bility Pump heuristic. -* -* REFERENCES -* -* M.Fischetti, F.Glover, and A.Lodi. "The feasibility pump." Math. -* Program., Ser. A 104, pp. 91-104 (2005). */ - -struct VAR -{ /* binary variable */ - int j; - /* ordinal number */ - int x; - /* value in the rounded solution (0 or 1) */ - double d; - /* sorting key */ -}; - -static int CDECL fcmp(const void *x, const void *y) -{ /* comparison routine */ - const struct VAR *vx = x, *vy = y; - if (vx->d > vy->d) - return -1; - else if (vx->d < vy->d) - return +1; - else - return 0; -} - -void ios_feas_pump(glp_tree *T) -{ glp_prob *P = T->mip; - int n = P->n; - glp_prob *lp = NULL; - struct VAR *var = NULL; - RNG *rand = NULL; - GLPCOL *col; - glp_smcp parm; - int j, k, new_x, nfail, npass, nv, ret, stalling; - double dist, tol; - xassert(glp_get_status(P) == GLP_OPT); - /* this heuristic is applied only once on the root level */ - if (!(T->curr->level == 0 && T->curr->solved == 1)) goto done; - /* determine number of binary variables */ - nv = 0; - for (j = 1; j <= n; j++) - { col = P->col[j]; - /* if x[j] is continuous, skip it */ - if (col->kind == GLP_CV) continue; - /* if x[j] is fixed, skip it */ - if (col->type == GLP_FX) continue; - /* x[j] is non-fixed integer */ - xassert(col->kind == GLP_IV); - if (col->type == GLP_DB && col->lb == 0.0 && col->ub == 1.0) - { /* x[j] is binary */ - nv++; - } - else - { /* x[j] is general integer */ - if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("FPUMP heuristic cannot be applied due to genera" - "l integer variables\n"); - goto done; - } - } - /* there must be at least one binary variable */ - if (nv == 0) goto done; - if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Applying FPUMP heuristic...\n"); - /* build the list of binary variables */ - var = xcalloc(1+nv, sizeof(struct VAR)); - k = 0; - for (j = 1; j <= n; j++) - { col = P->col[j]; - if (col->kind == GLP_IV && col->type == GLP_DB) - var[++k].j = j; - } - xassert(k == nv); - /* create working problem object */ - lp = glp_create_prob(); -more: /* copy the original problem object to keep it intact */ - glp_copy_prob(lp, P, GLP_OFF); - /* we are interested to find an integer feasible solution, which - is better than the best known one */ - if (P->mip_stat == GLP_FEAS) - { int *ind; - double *val, bnd; - /* add a row and make it identical to the objective row */ - glp_add_rows(lp, 1); - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) - { ind[j] = j; - val[j] = P->col[j]->coef; - } - glp_set_mat_row(lp, lp->m, n, ind, val); - xfree(ind); - xfree(val); - /* introduce upper (minimization) or lower (maximization) - bound to the original objective function; note that this - additional constraint is not violated at the optimal point - to LP relaxation */ -#if 0 /* modified by xypron */ - if (P->dir == GLP_MIN) - { bnd = P->mip_obj - 0.10 * (1.0 + fabs(P->mip_obj)); - if (bnd < P->obj_val) bnd = P->obj_val; - glp_set_row_bnds(lp, lp->m, GLP_UP, 0.0, bnd - P->c0); - } - else if (P->dir == GLP_MAX) - { bnd = P->mip_obj + 0.10 * (1.0 + fabs(P->mip_obj)); - if (bnd > P->obj_val) bnd = P->obj_val; - glp_set_row_bnds(lp, lp->m, GLP_LO, bnd - P->c0, 0.0); - } - else - xassert(P != P); -#else - bnd = 0.1 * P->obj_val + 0.9 * P->mip_obj; - /* xprintf("bnd = %f\n", bnd); */ - if (P->dir == GLP_MIN) - glp_set_row_bnds(lp, lp->m, GLP_UP, 0.0, bnd - P->c0); - else if (P->dir == GLP_MAX) - glp_set_row_bnds(lp, lp->m, GLP_LO, bnd - P->c0, 0.0); - else - xassert(P != P); -#endif - } - /* reset pass count */ - npass = 0; - /* invalidate the rounded point */ - for (k = 1; k <= nv; k++) - var[k].x = -1; -pass: /* next pass starts here */ - npass++; - if (T->parm->msg_lev >= GLP_MSG_ALL) - xprintf("Pass %d\n", npass); - /* initialize minimal distance between the basic point and the - rounded one obtained during this pass */ - dist = DBL_MAX; - /* reset failure count (the number of succeeded iterations failed - to improve the distance) */ - nfail = 0; - /* if it is not the first pass, perturb the last rounded point - rather than construct it from the basic solution */ - if (npass > 1) - { double rho, temp; - if (rand == NULL) - rand = rng_create_rand(); - for (k = 1; k <= nv; k++) - { j = var[k].j; - col = lp->col[j]; - rho = rng_uniform(rand, -0.3, 0.7); - if (rho < 0.0) rho = 0.0; - temp = fabs((double)var[k].x - col->prim); - if (temp + rho > 0.5) var[k].x = 1 - var[k].x; - } - goto skip; - } -loop: /* innermost loop begins here */ - /* round basic solution (which is assumed primal feasible) */ - stalling = 1; - for (k = 1; k <= nv; k++) - { col = lp->col[var[k].j]; - if (col->prim < 0.5) - { /* rounded value is 0 */ - new_x = 0; - } - else - { /* rounded value is 1 */ - new_x = 1; - } - if (var[k].x != new_x) - { stalling = 0; - var[k].x = new_x; - } - } - /* if the rounded point has not changed (stalling), choose and - flip some its entries heuristically */ - if (stalling) - { /* compute d[j] = |x[j] - round(x[j])| */ - for (k = 1; k <= nv; k++) - { col = lp->col[var[k].j]; - var[k].d = fabs(col->prim - (double)var[k].x); - } - /* sort the list of binary variables by descending d[j] */ - qsort(&var[1], nv, sizeof(struct VAR), fcmp); - /* choose and flip some rounded components */ - for (k = 1; k <= nv; k++) - { if (k >= 5 && var[k].d < 0.35 || k >= 10) break; - var[k].x = 1 - var[k].x; - } - } -skip: /* check if the time limit has been exhausted */ - if (T->parm->tm_lim < INT_MAX && - (double)(T->parm->tm_lim - 1) <= - 1000.0 * xdifftime(xtime(), T->tm_beg)) goto done; - /* build the objective, which is the distance between the current - (basic) point and the rounded one */ - lp->dir = GLP_MIN; - lp->c0 = 0.0; - for (j = 1; j <= n; j++) - lp->col[j]->coef = 0.0; - for (k = 1; k <= nv; k++) - { j = var[k].j; - if (var[k].x == 0) - lp->col[j]->coef = +1.0; - else - { lp->col[j]->coef = -1.0; - lp->c0 += 1.0; - } - } - /* minimize the distance with the simplex method */ - glp_init_smcp(&parm); - if (T->parm->msg_lev <= GLP_MSG_ERR) - parm.msg_lev = T->parm->msg_lev; - else if (T->parm->msg_lev <= GLP_MSG_ALL) - { parm.msg_lev = GLP_MSG_ON; - parm.out_dly = 10000; - } - ret = glp_simplex(lp, &parm); - if (ret != 0) - { if (T->parm->msg_lev >= GLP_MSG_ERR) - xprintf("Warning: glp_simplex returned %d\n", ret); - goto done; - } - ret = glp_get_status(lp); - if (ret != GLP_OPT) - { if (T->parm->msg_lev >= GLP_MSG_ERR) - xprintf("Warning: glp_get_status returned %d\n", ret); - goto done; - } - if (T->parm->msg_lev >= GLP_MSG_DBG) - xprintf("delta = %g\n", lp->obj_val); - /* check if the basic solution is integer feasible; note that it - may be so even if the minimial distance is positive */ - tol = 0.3 * T->parm->tol_int; - for (k = 1; k <= nv; k++) - { col = lp->col[var[k].j]; - if (tol < col->prim && col->prim < 1.0 - tol) break; - } - if (k > nv) - { /* okay; the basic solution seems to be integer feasible */ - double *x = xcalloc(1+n, sizeof(double)); - for (j = 1; j <= n; j++) - { x[j] = lp->col[j]->prim; - if (P->col[j]->kind == GLP_IV) x[j] = floor(x[j] + 0.5); - } -#if 1 /* modified by xypron */ - /* reset direction and right-hand side of objective */ - lp->c0 = P->c0; - lp->dir = P->dir; - /* fix integer variables */ - for (k = 1; k <= nv; k++) -#if 0 /* 18/VI-2013; fixed by mao - * this bug causes numerical instability, because column statuses - * are not changed appropriately */ - { lp->col[var[k].j]->lb = x[var[k].j]; - lp->col[var[k].j]->ub = x[var[k].j]; - lp->col[var[k].j]->type = GLP_FX; - } -#else - glp_set_col_bnds(lp, var[k].j, GLP_FX, x[var[k].j], 0.); -#endif - /* copy original objective function */ - for (j = 1; j <= n; j++) - lp->col[j]->coef = P->col[j]->coef; - /* solve original LP and copy result */ - ret = glp_simplex(lp, &parm); - if (ret != 0) - { if (T->parm->msg_lev >= GLP_MSG_ERR) - xprintf("Warning: glp_simplex returned %d\n", ret); -#if 1 /* 17/III-2016: fix memory leak */ - xfree(x); -#endif - goto done; - } - ret = glp_get_status(lp); - if (ret != GLP_OPT) - { if (T->parm->msg_lev >= GLP_MSG_ERR) - xprintf("Warning: glp_get_status returned %d\n", ret); -#if 1 /* 17/III-2016: fix memory leak */ - xfree(x); -#endif - goto done; - } - for (j = 1; j <= n; j++) - if (P->col[j]->kind != GLP_IV) x[j] = lp->col[j]->prim; -#endif - ret = glp_ios_heur_sol(T, x); - xfree(x); - if (ret == 0) - { /* the integer solution is accepted */ - if (ios_is_hopeful(T, T->curr->bound)) - { /* it is reasonable to apply the heuristic once again */ - goto more; - } - else - { /* the best known integer feasible solution just found - is close to optimal solution to LP relaxation */ - goto done; - } - } - } - /* the basic solution is fractional */ - if (dist == DBL_MAX || - lp->obj_val <= dist - 1e-6 * (1.0 + dist)) - { /* the distance is reducing */ - nfail = 0, dist = lp->obj_val; - } - else - { /* improving the distance failed */ - nfail++; - } - if (nfail < 3) goto loop; - if (npass < 5) goto pass; -done: /* delete working objects */ - if (lp != NULL) glp_delete_prob(lp); - if (var != NULL) xfree(var); - if (rand != NULL) rng_delete_rand(rand); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/gmicut.c b/code/3rd_glpk/intopt/gmicut.c deleted file mode 100644 index 4ef0b746..00000000 --- a/code/3rd_glpk/intopt/gmicut.c +++ /dev/null @@ -1,284 +0,0 @@ -/* gmicut.c (Gomory's mixed integer cut generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2002-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_gmi_cut - generate Gomory's mixed integer cut (core routine) -* -* SYNOPSIS -* -* int glp_gmi_cut(glp_prob *P, int j, int ind[], double val[], double -* phi[]); -* -* DESCRIPTION -* -* This routine attempts to generate a Gomory's mixed integer cut for -* specified integer column (structural variable), whose primal value -* in current basic solution is integer infeasible (fractional). -* -* On entry to the routine the basic solution contained in the problem -* object P should be optimal, and the basis factorization should be -* valid. The parameter j should specify the ordinal number of column -* (structural variable x[j]), for which the cut should be generated, -* 1 <= j <= n, where n is the number of columns in the problem object. -* This column should be integer, non-fixed, and basic, and its primal -* value should be fractional. -* -* The cut generated by the routine is the following inequality: -* -* sum a[j] * x[j] >= b, -* -* which is expected to be violated at the current basic solution. -* -* If the cut has been successfully generated, the routine stores its -* non-zero coefficients a[j] and corresponding column indices j in the -* array locations val[1], ..., val[len] and ind[1], ..., ind[len], -* where 1 <= len <= n is the number of non-zero coefficients. The -* right-hand side value b is stored in val[0], and ind[0] is set to 0. -* -* The working array phi should have 1+m+n locations (location phi[0] -* is not used), where m and n is the number of rows and columns in the -* problem object, resp. -* -* RETURNS -* -* If the cut has been successfully generated, the routine returns -* len, the number of non-zero coefficients in the cut, 1 <= len <= n. -* -* Otherwise, the routine returns one of the following codes: -* -* -1 current basis factorization is not valid; -* -* -2 current basic solution is not optimal; -* -* -3 column ordinal number j is out of range; -* -* -4 variable x[j] is not of integral kind; -* -* -5 variable x[j] is either fixed or non-basic; -* -* -6 primal value of variable x[j] in basic solution is too close -* to nearest integer; -* -* -7 some coefficients in the simplex table row corresponding to -* variable x[j] are too large in magnitude; -* -* -8 some free (unbounded) variables have non-zero coefficients in -* the simplex table row corresponding to variable x[j]. -* -* ALGORITHM -* -* See glpk/doc/notes/gomory (in Russian). */ - -#define f(x) ((x) - floor(x)) -/* compute fractional part of x */ - -int glp_gmi_cut(glp_prob *P, int j, - int ind[/*1+n*/], double val[/*1+n*/], double phi[/*1+m+n*/]) -{ int m = P->m; - int n = P->n; - GLPROW *row; - GLPCOL *col; - GLPAIJ *aij; - int i, k, len, kind, stat; - double lb, ub, alfa, beta, ksi, phi1, rhs; - /* sanity checks */ - if (!(P->m == 0 || P->valid)) - { /* current basis factorization is not valid */ - return -1; - } - if (!(P->pbs_stat == GLP_FEAS && P->dbs_stat == GLP_FEAS)) - { /* current basic solution is not optimal */ - return -2; - } - if (!(1 <= j && j <= n)) - { /* column ordinal number is out of range */ - return -3; - } - col = P->col[j]; - if (col->kind != GLP_IV) - { /* x[j] is not of integral kind */ - return -4; - } - if (col->type == GLP_FX || col->stat != GLP_BS) - { /* x[j] is either fixed or non-basic */ - return -5; - } - if (fabs(col->prim - floor(col->prim + 0.5)) < 0.001) - { /* primal value of x[j] is too close to nearest integer */ - return -6; - } - /* compute row of the simplex tableau, which (row) corresponds - * to specified basic variable xB[i] = x[j]; see (23) */ - len = glp_eval_tab_row(P, m+j, ind, val); - /* determine beta[i], which a value of xB[i] in optimal solution - * to current LP relaxation; note that this value is the same as - * if it would be computed with formula (27); it is assumed that - * beta[i] is fractional enough */ - beta = P->col[j]->prim; - /* compute cut coefficients phi and right-hand side rho, which - * correspond to formula (30); dense format is used, because rows - * of the simplex tableau are usually dense */ - for (k = 1; k <= m+n; k++) - phi[k] = 0.0; - rhs = f(beta); /* initial value of rho; see (28), (32) */ - for (j = 1; j <= len; j++) - { /* determine original number of non-basic variable xN[j] */ - k = ind[j]; - xassert(1 <= k && k <= m+n); - /* determine the kind, bounds and current status of xN[j] in - * optimal solution to LP relaxation */ - if (k <= m) - { /* auxiliary variable */ - row = P->row[k]; - kind = GLP_CV; - lb = row->lb; - ub = row->ub; - stat = row->stat; - } - else - { /* structural variable */ - col = P->col[k-m]; - kind = col->kind; - lb = col->lb; - ub = col->ub; - stat = col->stat; - } - /* xN[j] cannot be basic */ - xassert(stat != GLP_BS); - /* determine row coefficient ksi[i,j] at xN[j]; see (23) */ - ksi = val[j]; - /* if ksi[i,j] is too large in magnitude, report failure */ - if (fabs(ksi) > 1e+05) - return -7; - /* if ksi[i,j] is too small in magnitude, skip it */ - if (fabs(ksi) < 1e-10) - goto skip; - /* compute row coefficient alfa[i,j] at y[j]; see (26) */ - switch (stat) - { case GLP_NF: - /* xN[j] is free (unbounded) having non-zero ksi[i,j]; - * report failure */ - return -8; - case GLP_NL: - /* xN[j] has active lower bound */ - alfa = - ksi; - break; - case GLP_NU: - /* xN[j] has active upper bound */ - alfa = + ksi; - break; - case GLP_NS: - /* xN[j] is fixed; skip it */ - goto skip; - default: - xassert(stat != stat); - } - /* compute cut coefficient phi'[j] at y[j]; see (21), (28) */ - switch (kind) - { case GLP_IV: - /* y[j] is integer */ - if (fabs(alfa - floor(alfa + 0.5)) < 1e-10) - { /* alfa[i,j] is close to nearest integer; skip it */ - goto skip; - } - else if (f(alfa) <= f(beta)) - phi1 = f(alfa); - else - phi1 = (f(beta) / (1.0 - f(beta))) * (1.0 - f(alfa)); - break; - case GLP_CV: - /* y[j] is continuous */ - if (alfa >= 0.0) - phi1 = + alfa; - else - phi1 = (f(beta) / (1.0 - f(beta))) * (- alfa); - break; - default: - xassert(kind != kind); - } - /* compute cut coefficient phi[j] at xN[j] and update right- - * hand side rho; see (31), (32) */ - switch (stat) - { case GLP_NL: - /* xN[j] has active lower bound */ - phi[k] = + phi1; - rhs += phi1 * lb; - break; - case GLP_NU: - /* xN[j] has active upper bound */ - phi[k] = - phi1; - rhs -= phi1 * ub; - break; - default: - xassert(stat != stat); - } -skip: ; - } - /* now the cut has the form sum_k phi[k] * x[k] >= rho, where cut - * coefficients are stored in the array phi in dense format; - * x[1,...,m] are auxiliary variables, x[m+1,...,m+n] are struc- - * tural variables; see (30) */ - /* eliminate auxiliary variables in order to express the cut only - * through structural variables; see (33) */ - for (i = 1; i <= m; i++) - { if (fabs(phi[i]) < 1e-10) - continue; - /* auxiliary variable x[i] has non-zero cut coefficient */ - row = P->row[i]; - /* x[i] cannot be fixed variable */ - xassert(row->type != GLP_FX); - /* substitute x[i] = sum_j a[i,j] * x[m+j] */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - phi[m+aij->col->j] += phi[i] * aij->val; - } - /* convert the final cut to sparse format and substitute fixed - * (structural) variables */ - len = 0; - for (j = 1; j <= n; j++) - { if (fabs(phi[m+j]) < 1e-10) - continue; - /* structural variable x[m+j] has non-zero cut coefficient */ - col = P->col[j]; - if (col->type == GLP_FX) - { /* eliminate x[m+j] */ - rhs -= phi[m+j] * col->lb; - } - else - { len++; - ind[len] = j; - val[len] = phi[m+j]; - } - } - if (fabs(rhs) < 1e-12) - rhs = 0.0; - ind[0] = 0, val[0] = rhs; - /* the cut has been successfully generated */ - return len; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/gmigen.c b/code/3rd_glpk/intopt/gmigen.c deleted file mode 100644 index 627682cb..00000000 --- a/code/3rd_glpk/intopt/gmigen.c +++ /dev/null @@ -1,142 +0,0 @@ -/* gmigen.c (Gomory's mixed integer cuts generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2002-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "prob.h" - -/*********************************************************************** -* NAME -* -* glp_gmi_gen - generate Gomory's mixed integer cuts -* -* SYNOPSIS -* -* int glp_gmi_gen(glp_prob *P, glp_prob *pool, int max_cuts); -* -* DESCRIPTION -* -* This routine attempts to generate Gomory's mixed integer cuts for -* integer variables, whose primal values in current basic solution are -* integer infeasible (fractional). -* -* On entry to the routine the basic solution contained in the problem -* object P should be optimal, and the basis factorization should be -* valid. -* -* The cutting plane inequalities generated by the routine are added to -* the specified cut pool. -* -* The parameter max_cuts specifies the maximal number of cuts to be -* generated. Note that the number of cuts cannot exceed the number of -* basic variables, which is the number of rows in the problem object. -* -* RETURNS -* -* The routine returns the number of cuts that have been generated and -* added to the cut pool. */ - -#define f(x) ((x) - floor(x)) -/* compute fractional part of x */ - -struct var { int j; double f; }; - -static int CDECL fcmp(const void *p1, const void *p2) -{ const struct var *v1 = p1, *v2 = p2; - if (v1->f > v2->f) return -1; - if (v1->f < v2->f) return +1; - return 0; -} - -int glp_gmi_gen(glp_prob *P, glp_prob *pool, int max_cuts) -{ int m = P->m; - int n = P->n; - GLPCOL *col; - struct var *var; - int i, j, k, t, len, nv, nnn, *ind; - double frac, *val, *phi; - /* sanity checks */ - if (!(P->m == 0 || P->valid)) - xerror("glp_gmi_gen: basis factorization does not exist\n"); - if (!(P->pbs_stat == GLP_FEAS && P->dbs_stat == GLP_FEAS)) - xerror("glp_gmi_gen: optimal basic solution required\n"); - if (pool->n != n) - xerror("glp_gmi_gen: cut pool has wrong number of columns\n"); - /* allocate working arrays */ - var = xcalloc(1+n, sizeof(struct var)); - ind = xcalloc(1+n, sizeof(int)); - val = xcalloc(1+n, sizeof(double)); - phi = xcalloc(1+m+n, sizeof(double)); - /* build the list of integer structural variables, which are - * basic and have integer infeasible (fractional) primal values - * in optimal solution to specified LP */ - nv = 0; - for (j = 1; j <= n; j++) - { col = P->col[j]; - if (col->kind != GLP_IV) - continue; - if (col->type == GLP_FX) - continue; - if (col->stat != GLP_BS) - continue; - frac = f(col->prim); - if (!(0.05 <= frac && frac <= 0.95)) - continue; - /* add variable to the list */ - nv++, var[nv].j = j, var[nv].f = frac; - } - /* sort the list by descending fractionality */ - qsort(&var[1], nv, sizeof(struct var), fcmp); - /* try to generate cuts by one for each variable in the list, but - * not more than max_cuts cuts */ - nnn = 0; - for (t = 1; t <= nv; t++) - { len = glp_gmi_cut(P, var[t].j, ind, val, phi); - if (len < 1) - goto skip; - /* if the cut inequality seems to be badly scaled, reject it - * to avoid numerical difficulties */ - for (k = 1; k <= len; k++) - { if (fabs(val[k]) < 1e-03) - goto skip; - if (fabs(val[k]) > 1e+03) - goto skip; - } - /* add the cut to the cut pool for further consideration */ - i = glp_add_rows(pool, 1); - glp_set_row_bnds(pool, i, GLP_LO, val[0], 0); - glp_set_mat_row(pool, i, len, ind, val); - /* one cut has been generated */ - nnn++; - if (nnn == max_cuts) - break; -skip: ; - } - /* free working arrays */ - xfree(var); - xfree(ind); - xfree(val); - xfree(phi); - return nnn; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/mirgen.c b/code/3rd_glpk/intopt/mirgen.c deleted file mode 100644 index 45a0a55d..00000000 --- a/code/3rd_glpk/intopt/mirgen.c +++ /dev/null @@ -1,1529 +0,0 @@ -/* mirgen.c (mixed integer rounding cuts generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2007-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#if 1 /* 29/II-2016 by Chris */ -/*---------------------------------------------------------------------- -Subject: Mir cut generation performance improvement -From: Chris Matrakidis -To: Andrew Makhorin , help-glpk - -Andrew, - -I noticed that mir cut generation takes considerable time on some large -problems (like rocII-4-11 from miplib). The attached patch makes two -improvements that considerably improve performance in such instances: -1. A lot of time was spent on generating a temporary vector in function -aggregate_row. It is a lot faster to reuse an existing vector. -2. A search for an element in the same function was done in row order, -where using the elements in the order they are in the column is more -efficient. This changes the generated cuts in some cases, but seems -neutral overall (0.3% less cuts in a test set of 64 miplib instances). - -Best Regards, - -Chris Matrakidis -----------------------------------------------------------------------*/ -#endif - -#include "env.h" -#include "prob.h" -#include "spv.h" - -#define MIR_DEBUG 0 - -#define MAXAGGR 5 -/* maximal number of rows that can be aggregated */ - -struct glp_mir -{ /* MIR cut generator working area */ - /*--------------------------------------------------------------*/ - /* global information valid for the root subproblem */ - int m; - /* number of rows (in the root subproblem) */ - int n; - /* number of columns */ - char *skip; /* char skip[1+m]; */ - /* skip[i], 1 <= i <= m, is a flag that means that row i should - not be used because (1) it is not suitable, or (2) because it - has been used in the aggregated constraint */ - char *isint; /* char isint[1+m+n]; */ - /* isint[k], 1 <= k <= m+n, is a flag that means that variable - x[k] is integer (otherwise, continuous) */ - double *lb; /* double lb[1+m+n]; */ - /* lb[k], 1 <= k <= m+n, is lower bound of x[k]; -DBL_MAX means - that x[k] has no lower bound */ - int *vlb; /* int vlb[1+m+n]; */ - /* vlb[k] = k', 1 <= k <= m+n, is the number of integer variable, - which defines variable lower bound x[k] >= lb[k] * x[k']; zero - means that x[k] has simple lower bound */ - double *ub; /* double ub[1+m+n]; */ - /* ub[k], 1 <= k <= m+n, is upper bound of x[k]; +DBL_MAX means - that x[k] has no upper bound */ - int *vub; /* int vub[1+m+n]; */ - /* vub[k] = k', 1 <= k <= m+n, is the number of integer variable, - which defines variable upper bound x[k] <= ub[k] * x[k']; zero - means that x[k] has simple upper bound */ - /*--------------------------------------------------------------*/ - /* current (fractional) point to be separated */ - double *x; /* double x[1+m+n]; */ - /* x[k] is current value of auxiliary (1 <= k <= m) or structural - (m+1 <= k <= m+n) variable */ - /*--------------------------------------------------------------*/ - /* aggregated constraint sum a[k] * x[k] = b, which is a linear - combination of original constraints transformed to equalities - by introducing auxiliary variables */ - int agg_cnt; - /* number of rows (original constraints) used to build aggregated - constraint, 1 <= agg_cnt <= MAXAGGR */ - int *agg_row; /* int agg_row[1+MAXAGGR]; */ - /* agg_row[k], 1 <= k <= agg_cnt, is the row number used to build - aggregated constraint */ - SPV *agg_vec; /* SPV agg_vec[1:m+n]; */ - /* sparse vector of aggregated constraint coefficients, a[k] */ - double agg_rhs; - /* right-hand side of the aggregated constraint, b */ - /*--------------------------------------------------------------*/ - /* bound substitution flags for modified constraint */ - char *subst; /* char subst[1+m+n]; */ - /* subst[k], 1 <= k <= m+n, is a bound substitution flag used for - variable x[k]: - '?' - x[k] is missing in modified constraint - 'L' - x[k] = (lower bound) + x'[k] - 'U' - x[k] = (upper bound) - x'[k] */ - /*--------------------------------------------------------------*/ - /* modified constraint sum a'[k] * x'[k] = b', where x'[k] >= 0, - derived from aggregated constraint by substituting bounds; - note that due to substitution of variable bounds there may be - additional terms in the modified constraint */ - SPV *mod_vec; /* SPV mod_vec[1:m+n]; */ - /* sparse vector of modified constraint coefficients, a'[k] */ - double mod_rhs; - /* right-hand side of the modified constraint, b' */ - /*--------------------------------------------------------------*/ - /* cutting plane sum alpha[k] * x[k] <= beta */ - SPV *cut_vec; /* SPV cut_vec[1:m+n]; */ - /* sparse vector of cutting plane coefficients, alpha[k] */ - double cut_rhs; - /* right-hand size of the cutting plane, beta */ -}; - -/*********************************************************************** -* NAME -* -* glp_mir_init - create and initialize MIR cut generator -* -* SYNOPSIS -* -* glp_mir *glp_mir_init(glp_prob *P); -* -* DESCRIPTION -* -* This routine creates and initializes the MIR cut generator for the -* specified problem object. -* -* RETURNS -* -* The routine returns a pointer to the MIR cut generator workspace. */ - -static void set_row_attrib(glp_prob *mip, glp_mir *mir) -{ /* set global row attributes */ - int m = mir->m; - int k; - for (k = 1; k <= m; k++) - { GLPROW *row = mip->row[k]; - mir->skip[k] = 0; - mir->isint[k] = 0; - switch (row->type) - { case GLP_FR: - mir->lb[k] = -DBL_MAX, mir->ub[k] = +DBL_MAX; break; - case GLP_LO: - mir->lb[k] = row->lb, mir->ub[k] = +DBL_MAX; break; - case GLP_UP: - mir->lb[k] = -DBL_MAX, mir->ub[k] = row->ub; break; - case GLP_DB: - mir->lb[k] = row->lb, mir->ub[k] = row->ub; break; - case GLP_FX: - mir->lb[k] = mir->ub[k] = row->lb; break; - default: - xassert(row != row); - } - mir->vlb[k] = mir->vub[k] = 0; - } - return; -} - -static void set_col_attrib(glp_prob *mip, glp_mir *mir) -{ /* set global column attributes */ - int m = mir->m; - int n = mir->n; - int k; - for (k = m+1; k <= m+n; k++) - { GLPCOL *col = mip->col[k-m]; - switch (col->kind) - { case GLP_CV: - mir->isint[k] = 0; break; - case GLP_IV: - mir->isint[k] = 1; break; - default: - xassert(col != col); - } - switch (col->type) - { case GLP_FR: - mir->lb[k] = -DBL_MAX, mir->ub[k] = +DBL_MAX; break; - case GLP_LO: - mir->lb[k] = col->lb, mir->ub[k] = +DBL_MAX; break; - case GLP_UP: - mir->lb[k] = -DBL_MAX, mir->ub[k] = col->ub; break; - case GLP_DB: - mir->lb[k] = col->lb, mir->ub[k] = col->ub; break; - case GLP_FX: - mir->lb[k] = mir->ub[k] = col->lb; break; - default: - xassert(col != col); - } - mir->vlb[k] = mir->vub[k] = 0; - } - return; -} - -static void set_var_bounds(glp_prob *mip, glp_mir *mir) -{ /* set variable bounds */ - int m = mir->m; - GLPAIJ *aij; - int i, k1, k2; - double a1, a2; - for (i = 1; i <= m; i++) - { /* we need the row to be '>= 0' or '<= 0' */ - if (!(mir->lb[i] == 0.0 && mir->ub[i] == +DBL_MAX || - mir->lb[i] == -DBL_MAX && mir->ub[i] == 0.0)) continue; - /* take first term */ - aij = mip->row[i]->ptr; - if (aij == NULL) continue; - k1 = m + aij->col->j, a1 = aij->val; - /* take second term */ - aij = aij->r_next; - if (aij == NULL) continue; - k2 = m + aij->col->j, a2 = aij->val; - /* there must be only two terms */ - if (aij->r_next != NULL) continue; - /* interchange terms, if needed */ - if (!mir->isint[k1] && mir->isint[k2]) - ; - else if (mir->isint[k1] && !mir->isint[k2]) - { k2 = k1, a2 = a1; - k1 = m + aij->col->j, a1 = aij->val; - } - else - { /* both terms are either continuous or integer */ - continue; - } - /* x[k2] should be double-bounded */ - if (mir->lb[k2] == -DBL_MAX || mir->ub[k2] == +DBL_MAX || - mir->lb[k2] == mir->ub[k2]) continue; - /* change signs, if necessary */ - if (mir->ub[i] == 0.0) a1 = - a1, a2 = - a2; - /* now the row has the form a1 * x1 + a2 * x2 >= 0, where x1 - is continuous, x2 is integer */ - if (a1 > 0.0) - { /* x1 >= - (a2 / a1) * x2 */ - if (mir->vlb[k1] == 0) - { /* set variable lower bound for x1 */ - mir->lb[k1] = - a2 / a1; - mir->vlb[k1] = k2; - /* the row should not be used */ - mir->skip[i] = 1; - } - } - else /* a1 < 0.0 */ - { /* x1 <= - (a2 / a1) * x2 */ - if (mir->vub[k1] == 0) - { /* set variable upper bound for x1 */ - mir->ub[k1] = - a2 / a1; - mir->vub[k1] = k2; - /* the row should not be used */ - mir->skip[i] = 1; - } - } - } - return; -} - -static void mark_useless_rows(glp_prob *mip, glp_mir *mir) -{ /* mark rows which should not be used */ - int m = mir->m; - GLPAIJ *aij; - int i, k, nv; - for (i = 1; i <= m; i++) - { /* free rows should not be used */ - if (mir->lb[i] == -DBL_MAX && mir->ub[i] == +DBL_MAX) - { mir->skip[i] = 1; - continue; - } - nv = 0; - for (aij = mip->row[i]->ptr; aij != NULL; aij = aij->r_next) - { k = m + aij->col->j; - /* rows with free variables should not be used */ - if (mir->lb[k] == -DBL_MAX && mir->ub[k] == +DBL_MAX) - { mir->skip[i] = 1; - break; - } - /* rows with integer variables having infinite (lower or - upper) bound should not be used */ - if (mir->isint[k] && mir->lb[k] == -DBL_MAX || - mir->isint[k] && mir->ub[k] == +DBL_MAX) - { mir->skip[i] = 1; - break; - } - /* count non-fixed variables */ - if (!(mir->vlb[k] == 0 && mir->vub[k] == 0 && - mir->lb[k] == mir->ub[k])) nv++; - } - /* rows with all variables fixed should not be used */ - if (nv == 0) - { mir->skip[i] = 1; - continue; - } - } - return; -} - -glp_mir *glp_mir_init(glp_prob *mip) -{ /* create and initialize MIR cut generator */ - int m = mip->m; - int n = mip->n; - glp_mir *mir; -#if MIR_DEBUG - xprintf("ios_mir_init: warning: debug mode enabled\n"); -#endif - /* allocate working area */ - mir = xmalloc(sizeof(glp_mir)); - mir->m = m; - mir->n = n; - mir->skip = xcalloc(1+m, sizeof(char)); - mir->isint = xcalloc(1+m+n, sizeof(char)); - mir->lb = xcalloc(1+m+n, sizeof(double)); - mir->vlb = xcalloc(1+m+n, sizeof(int)); - mir->ub = xcalloc(1+m+n, sizeof(double)); - mir->vub = xcalloc(1+m+n, sizeof(int)); - mir->x = xcalloc(1+m+n, sizeof(double)); - mir->agg_row = xcalloc(1+MAXAGGR, sizeof(int)); - mir->agg_vec = spv_create_vec(m+n); - mir->subst = xcalloc(1+m+n, sizeof(char)); - mir->mod_vec = spv_create_vec(m+n); - mir->cut_vec = spv_create_vec(m+n); - /* set global row attributes */ - set_row_attrib(mip, mir); - /* set global column attributes */ - set_col_attrib(mip, mir); - /* set variable bounds */ - set_var_bounds(mip, mir); - /* mark rows which should not be used */ - mark_useless_rows(mip, mir); - return mir; -} - -/*********************************************************************** -* NAME -* -* glp_mir_gen - generate mixed integer rounding (MIR) cuts -* -* SYNOPSIS -* -* int glp_mir_gen(glp_prob *P, glp_mir *mir, glp_prob *pool); -* -* DESCRIPTION -* -* This routine attempts to generate mixed integer rounding (MIR) cuts -* for current basic solution to the specified problem object. -* -* The cutting plane inequalities generated by the routine are added to -* the specified cut pool. -* -* RETURNS -* -* The routine returns the number of cuts that have been generated and -* added to the cut pool. */ - -static void get_current_point(glp_prob *mip, glp_mir *mir) -{ /* obtain current point */ - int m = mir->m; - int n = mir->n; - int k; - for (k = 1; k <= m; k++) - mir->x[k] = mip->row[k]->prim; - for (k = m+1; k <= m+n; k++) - mir->x[k] = mip->col[k-m]->prim; - return; -} - -#if MIR_DEBUG -static void check_current_point(glp_mir *mir) -{ /* check current point */ - int m = mir->m; - int n = mir->n; - int k, kk; - double lb, ub, eps; - for (k = 1; k <= m+n; k++) - { /* determine lower bound */ - lb = mir->lb[k]; - kk = mir->vlb[k]; - if (kk != 0) - { xassert(lb != -DBL_MAX); - xassert(!mir->isint[k]); - xassert(mir->isint[kk]); - lb *= mir->x[kk]; - } - /* check lower bound */ - if (lb != -DBL_MAX) - { eps = 1e-6 * (1.0 + fabs(lb)); - xassert(mir->x[k] >= lb - eps); - } - /* determine upper bound */ - ub = mir->ub[k]; - kk = mir->vub[k]; - if (kk != 0) - { xassert(ub != +DBL_MAX); - xassert(!mir->isint[k]); - xassert(mir->isint[kk]); - ub *= mir->x[kk]; - } - /* check upper bound */ - if (ub != +DBL_MAX) - { eps = 1e-6 * (1.0 + fabs(ub)); - xassert(mir->x[k] <= ub + eps); - } - } - return; -} -#endif - -static void initial_agg_row(glp_prob *mip, glp_mir *mir, int i) -{ /* use original i-th row as initial aggregated constraint */ - int m = mir->m; - GLPAIJ *aij; - xassert(1 <= i && i <= m); - xassert(!mir->skip[i]); - /* mark i-th row in order not to use it in the same aggregated - constraint */ - mir->skip[i] = 2; - mir->agg_cnt = 1; - mir->agg_row[1] = i; - /* use x[i] - sum a[i,j] * x[m+j] = 0, where x[i] is auxiliary - variable of row i, x[m+j] are structural variables */ - spv_clear_vec(mir->agg_vec); - spv_set_vj(mir->agg_vec, i, 1.0); - for (aij = mip->row[i]->ptr; aij != NULL; aij = aij->r_next) - spv_set_vj(mir->agg_vec, m + aij->col->j, - aij->val); - mir->agg_rhs = 0.0; -#if MIR_DEBUG - spv_check_vec(mir->agg_vec); -#endif - return; -} - -#if MIR_DEBUG -static void check_agg_row(glp_mir *mir) -{ /* check aggregated constraint */ - int m = mir->m; - int n = mir->n; - int j, k; - double r, big; - /* compute the residual r = sum a[k] * x[k] - b and determine - big = max(1, |a[k]|, |b|) */ - r = 0.0, big = 1.0; - for (j = 1; j <= mir->agg_vec->nnz; j++) - { k = mir->agg_vec->ind[j]; - xassert(1 <= k && k <= m+n); - r += mir->agg_vec->val[j] * mir->x[k]; - if (big < fabs(mir->agg_vec->val[j])) - big = fabs(mir->agg_vec->val[j]); - } - r -= mir->agg_rhs; - if (big < fabs(mir->agg_rhs)) - big = fabs(mir->agg_rhs); - /* the residual must be close to zero */ - xassert(fabs(r) <= 1e-6 * big); - return; -} -#endif - -static void subst_fixed_vars(glp_mir *mir) -{ /* substitute fixed variables into aggregated constraint */ - int m = mir->m; - int n = mir->n; - int j, k; - for (j = 1; j <= mir->agg_vec->nnz; j++) - { k = mir->agg_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->vlb[k] == 0 && mir->vub[k] == 0 && - mir->lb[k] == mir->ub[k]) - { /* x[k] is fixed */ - mir->agg_rhs -= mir->agg_vec->val[j] * mir->lb[k]; - mir->agg_vec->val[j] = 0.0; - } - } - /* remove terms corresponding to fixed variables */ - spv_clean_vec(mir->agg_vec, DBL_EPSILON); -#if MIR_DEBUG - spv_check_vec(mir->agg_vec); -#endif - return; -} - -static void bound_subst_heur(glp_mir *mir) -{ /* bound substitution heuristic */ - int m = mir->m; - int n = mir->n; - int j, k, kk; - double d1, d2; - for (j = 1; j <= mir->agg_vec->nnz; j++) - { k = mir->agg_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->isint[k]) continue; /* skip integer variable */ - /* compute distance from x[k] to its lower bound */ - kk = mir->vlb[k]; - if (kk == 0) - { if (mir->lb[k] == -DBL_MAX) - d1 = DBL_MAX; - else - d1 = mir->x[k] - mir->lb[k]; - } - else - { xassert(1 <= kk && kk <= m+n); - xassert(mir->isint[kk]); - xassert(mir->lb[k] != -DBL_MAX); - d1 = mir->x[k] - mir->lb[k] * mir->x[kk]; - } - /* compute distance from x[k] to its upper bound */ - kk = mir->vub[k]; - if (kk == 0) - { if (mir->vub[k] == +DBL_MAX) - d2 = DBL_MAX; - else - d2 = mir->ub[k] - mir->x[k]; - } - else - { xassert(1 <= kk && kk <= m+n); - xassert(mir->isint[kk]); - xassert(mir->ub[k] != +DBL_MAX); - d2 = mir->ub[k] * mir->x[kk] - mir->x[k]; - } - /* x[k] cannot be free */ - xassert(d1 != DBL_MAX || d2 != DBL_MAX); - /* choose the bound which is closer to x[k] */ - xassert(mir->subst[k] == '?'); - if (d1 <= d2) - mir->subst[k] = 'L'; - else - mir->subst[k] = 'U'; - } - return; -} - -static void build_mod_row(glp_mir *mir) -{ /* substitute bounds and build modified constraint */ - int m = mir->m; - int n = mir->n; - int j, jj, k, kk; - /* initially modified constraint is aggregated constraint */ - spv_copy_vec(mir->mod_vec, mir->agg_vec); - mir->mod_rhs = mir->agg_rhs; -#if MIR_DEBUG - spv_check_vec(mir->mod_vec); -#endif - /* substitute bounds for continuous variables; note that due to - substitution of variable bounds additional terms may appear in - modified constraint */ - for (j = mir->mod_vec->nnz; j >= 1; j--) - { k = mir->mod_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->isint[k]) continue; /* skip integer variable */ - if (mir->subst[k] == 'L') - { /* x[k] = (lower bound) + x'[k] */ - xassert(mir->lb[k] != -DBL_MAX); - kk = mir->vlb[k]; - if (kk == 0) - { /* x[k] = lb[k] + x'[k] */ - mir->mod_rhs -= mir->mod_vec->val[j] * mir->lb[k]; - } - else - { /* x[k] = lb[k] * x[kk] + x'[k] */ - xassert(mir->isint[kk]); - jj = mir->mod_vec->pos[kk]; - if (jj == 0) - { spv_set_vj(mir->mod_vec, kk, 1.0); - jj = mir->mod_vec->pos[kk]; - mir->mod_vec->val[jj] = 0.0; - } - mir->mod_vec->val[jj] += - mir->mod_vec->val[j] * mir->lb[k]; - } - } - else if (mir->subst[k] == 'U') - { /* x[k] = (upper bound) - x'[k] */ - xassert(mir->ub[k] != +DBL_MAX); - kk = mir->vub[k]; - if (kk == 0) - { /* x[k] = ub[k] - x'[k] */ - mir->mod_rhs -= mir->mod_vec->val[j] * mir->ub[k]; - } - else - { /* x[k] = ub[k] * x[kk] - x'[k] */ - xassert(mir->isint[kk]); - jj = mir->mod_vec->pos[kk]; - if (jj == 0) - { spv_set_vj(mir->mod_vec, kk, 1.0); - jj = mir->mod_vec->pos[kk]; - mir->mod_vec->val[jj] = 0.0; - } - mir->mod_vec->val[jj] += - mir->mod_vec->val[j] * mir->ub[k]; - } - mir->mod_vec->val[j] = - mir->mod_vec->val[j]; - } - else - xassert(k != k); - } -#if MIR_DEBUG - spv_check_vec(mir->mod_vec); -#endif - /* substitute bounds for integer variables */ - for (j = 1; j <= mir->mod_vec->nnz; j++) - { k = mir->mod_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (!mir->isint[k]) continue; /* skip continuous variable */ - xassert(mir->subst[k] == '?'); - xassert(mir->vlb[k] == 0 && mir->vub[k] == 0); - xassert(mir->lb[k] != -DBL_MAX && mir->ub[k] != +DBL_MAX); - if (fabs(mir->lb[k]) <= fabs(mir->ub[k])) - { /* x[k] = lb[k] + x'[k] */ - mir->subst[k] = 'L'; - mir->mod_rhs -= mir->mod_vec->val[j] * mir->lb[k]; - } - else - { /* x[k] = ub[k] - x'[k] */ - mir->subst[k] = 'U'; - mir->mod_rhs -= mir->mod_vec->val[j] * mir->ub[k]; - mir->mod_vec->val[j] = - mir->mod_vec->val[j]; - } - } -#if MIR_DEBUG - spv_check_vec(mir->mod_vec); -#endif - return; -} - -#if MIR_DEBUG -static void check_mod_row(glp_mir *mir) -{ /* check modified constraint */ - int m = mir->m; - int n = mir->n; - int j, k, kk; - double r, big, x; - /* compute the residual r = sum a'[k] * x'[k] - b' and determine - big = max(1, |a[k]|, |b|) */ - r = 0.0, big = 1.0; - for (j = 1; j <= mir->mod_vec->nnz; j++) - { k = mir->mod_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->subst[k] == 'L') - { /* x'[k] = x[k] - (lower bound) */ - xassert(mir->lb[k] != -DBL_MAX); - kk = mir->vlb[k]; - if (kk == 0) - x = mir->x[k] - mir->lb[k]; - else - x = mir->x[k] - mir->lb[k] * mir->x[kk]; - } - else if (mir->subst[k] == 'U') - { /* x'[k] = (upper bound) - x[k] */ - xassert(mir->ub[k] != +DBL_MAX); - kk = mir->vub[k]; - if (kk == 0) - x = mir->ub[k] - mir->x[k]; - else - x = mir->ub[k] * mir->x[kk] - mir->x[k]; - } - else - xassert(k != k); - r += mir->mod_vec->val[j] * x; - if (big < fabs(mir->mod_vec->val[j])) - big = fabs(mir->mod_vec->val[j]); - } - r -= mir->mod_rhs; - if (big < fabs(mir->mod_rhs)) - big = fabs(mir->mod_rhs); - /* the residual must be close to zero */ - xassert(fabs(r) <= 1e-6 * big); - return; -} -#endif - -/*********************************************************************** -* mir_ineq - construct MIR inequality -* -* Given the single constraint mixed integer set -* -* |N| -* X = {(x,s) in Z x R : sum a[j] * x[j] <= b + s}, -* + + j in N -* -* this routine constructs the mixed integer rounding (MIR) inequality -* -* sum alpha[j] * x[j] <= beta + gamma * s, -* j in N -* -* which is valid for X. -* -* If the MIR inequality has been successfully constructed, the routine -* returns zero. Otherwise, if b is close to nearest integer, there may -* be numeric difficulties due to big coefficients; so in this case the -* routine returns non-zero. */ - -static int mir_ineq(const int n, const double a[], const double b, - double alpha[], double *beta, double *gamma) -{ int j; - double f, t; - if (fabs(b - floor(b + .5)) < 0.01) - return 1; - f = b - floor(b); - for (j = 1; j <= n; j++) - { t = (a[j] - floor(a[j])) - f; - if (t <= 0.0) - alpha[j] = floor(a[j]); - else - alpha[j] = floor(a[j]) + t / (1.0 - f); - } - *beta = floor(b); - *gamma = 1.0 / (1.0 - f); - return 0; -} - -/*********************************************************************** -* cmir_ineq - construct c-MIR inequality -* -* Given the mixed knapsack set -* -* MK |N| -* X = {(x,s) in Z x R : sum a[j] * x[j] <= b + s, -* + + j in N -* -* x[j] <= u[j]}, -* -* a subset C of variables to be complemented, and a divisor delta > 0, -* this routine constructs the complemented MIR (c-MIR) inequality -* -* sum alpha[j] * x[j] <= beta + gamma * s, -* j in N -* MK -* which is valid for X . -* -* If the c-MIR inequality has been successfully constructed, the -* routine returns zero. Otherwise, if there is a risk of numerical -* difficulties due to big coefficients (see comments to the routine -* mir_ineq), the routine cmir_ineq returns non-zero. */ - -static int cmir_ineq(const int n, const double a[], const double b, - const double u[], const char cset[], const double delta, - double alpha[], double *beta, double *gamma) -{ int j; - double *aa, bb; - aa = alpha, bb = b; - for (j = 1; j <= n; j++) - { aa[j] = a[j] / delta; - if (cset[j]) - aa[j] = - aa[j], bb -= a[j] * u[j]; - } - bb /= delta; - if (mir_ineq(n, aa, bb, alpha, beta, gamma)) return 1; - for (j = 1; j <= n; j++) - { if (cset[j]) - alpha[j] = - alpha[j], *beta += alpha[j] * u[j]; - } - *gamma /= delta; - return 0; -} - -/*********************************************************************** -* cmir_sep - c-MIR separation heuristic -* -* Given the mixed knapsack set -* -* MK |N| -* X = {(x,s) in Z x R : sum a[j] * x[j] <= b + s, -* + + j in N -* -* x[j] <= u[j]} -* -* * * -* and a fractional point (x , s ), this routine tries to construct -* c-MIR inequality -* -* sum alpha[j] * x[j] <= beta + gamma * s, -* j in N -* MK -* which is valid for X and has (desirably maximal) violation at the -* fractional point given. This is attained by choosing an appropriate -* set C of variables to be complemented and a divisor delta > 0, which -* together define corresponding c-MIR inequality. -* -* If a violated c-MIR inequality has been successfully constructed, -* the routine returns its violation: -* -* * * -* sum alpha[j] * x [j] - beta - gamma * s , -* j in N -* -* which is positive. In case of failure the routine returns zero. */ - -struct vset { int j; double v; }; - -static int CDECL cmir_cmp(const void *p1, const void *p2) -{ const struct vset *v1 = p1, *v2 = p2; - if (v1->v < v2->v) return -1; - if (v1->v > v2->v) return +1; - return 0; -} - -static double cmir_sep(const int n, const double a[], const double b, - const double u[], const double x[], const double s, - double alpha[], double *beta, double *gamma) -{ int fail, j, k, nv, v; - double delta, eps, d_try[1+3], r, r_best; - char *cset; - struct vset *vset; - /* allocate working arrays */ - cset = xcalloc(1+n, sizeof(char)); - vset = xcalloc(1+n, sizeof(struct vset)); - /* choose initial C */ - for (j = 1; j <= n; j++) - cset[j] = (char)(x[j] >= 0.5 * u[j]); - /* choose initial delta */ - r_best = delta = 0.0; - for (j = 1; j <= n; j++) - { xassert(a[j] != 0.0); - /* if x[j] is close to its bounds, skip it */ - eps = 1e-9 * (1.0 + fabs(u[j])); - if (x[j] < eps || x[j] > u[j] - eps) continue; - /* try delta = |a[j]| to construct c-MIR inequality */ - fail = cmir_ineq(n, a, b, u, cset, fabs(a[j]), alpha, beta, - gamma); - if (fail) continue; - /* compute violation */ - r = - (*beta) - (*gamma) * s; - for (k = 1; k <= n; k++) r += alpha[k] * x[k]; - if (r_best < r) r_best = r, delta = fabs(a[j]); - } - if (r_best < 0.001) r_best = 0.0; - if (r_best == 0.0) goto done; - xassert(delta > 0.0); - /* try to increase violation by dividing delta by 2, 4, and 8, - respectively */ - d_try[1] = delta / 2.0; - d_try[2] = delta / 4.0; - d_try[3] = delta / 8.0; - for (j = 1; j <= 3; j++) - { /* construct c-MIR inequality */ - fail = cmir_ineq(n, a, b, u, cset, d_try[j], alpha, beta, - gamma); - if (fail) continue; - /* compute violation */ - r = - (*beta) - (*gamma) * s; - for (k = 1; k <= n; k++) r += alpha[k] * x[k]; - if (r_best < r) r_best = r, delta = d_try[j]; - } - /* build subset of variables lying strictly between their bounds - and order it by nondecreasing values of |x[j] - u[j]/2| */ - nv = 0; - for (j = 1; j <= n; j++) - { /* if x[j] is close to its bounds, skip it */ - eps = 1e-9 * (1.0 + fabs(u[j])); - if (x[j] < eps || x[j] > u[j] - eps) continue; - /* add x[j] to the subset */ - nv++; - vset[nv].j = j; - vset[nv].v = fabs(x[j] - 0.5 * u[j]); - } - qsort(&vset[1], nv, sizeof(struct vset), cmir_cmp); - /* try to increase violation by successively complementing each - variable in the subset */ - for (v = 1; v <= nv; v++) - { j = vset[v].j; - /* replace x[j] by its complement or vice versa */ - cset[j] = (char)!cset[j]; - /* construct c-MIR inequality */ - fail = cmir_ineq(n, a, b, u, cset, delta, alpha, beta, gamma); - /* restore the variable */ - cset[j] = (char)!cset[j]; - /* do not replace the variable in case of failure */ - if (fail) continue; - /* compute violation */ - r = - (*beta) - (*gamma) * s; - for (k = 1; k <= n; k++) r += alpha[k] * x[k]; - if (r_best < r) r_best = r, cset[j] = (char)!cset[j]; - } - /* construct the best c-MIR inequality chosen */ - fail = cmir_ineq(n, a, b, u, cset, delta, alpha, beta, gamma); - xassert(!fail); -done: /* free working arrays */ - xfree(cset); - xfree(vset); - /* return to the calling routine */ - return r_best; -} - -static double generate(glp_mir *mir) -{ /* try to generate violated c-MIR cut for modified constraint */ - int m = mir->m; - int n = mir->n; - int j, k, kk, nint; - double s, *u, *x, *alpha, r_best = 0.0, b, beta, gamma; - spv_copy_vec(mir->cut_vec, mir->mod_vec); - mir->cut_rhs = mir->mod_rhs; - /* remove small terms, which can appear due to substitution of - variable bounds */ - spv_clean_vec(mir->cut_vec, DBL_EPSILON); -#if MIR_DEBUG - spv_check_vec(mir->cut_vec); -#endif - /* remove positive continuous terms to obtain MK relaxation */ - for (j = 1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (!mir->isint[k] && mir->cut_vec->val[j] > 0.0) - mir->cut_vec->val[j] = 0.0; - } - spv_clean_vec(mir->cut_vec, 0.0); -#if MIR_DEBUG - spv_check_vec(mir->cut_vec); -#endif - /* move integer terms to the beginning of the sparse vector and - determine the number of integer variables */ - nint = 0; - for (j = 1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->isint[k]) - { double temp; - nint++; - /* interchange elements [nint] and [j] */ - kk = mir->cut_vec->ind[nint]; - mir->cut_vec->pos[k] = nint; - mir->cut_vec->pos[kk] = j; - mir->cut_vec->ind[nint] = k; - mir->cut_vec->ind[j] = kk; - temp = mir->cut_vec->val[nint]; - mir->cut_vec->val[nint] = mir->cut_vec->val[j]; - mir->cut_vec->val[j] = temp; - } - } -#if MIR_DEBUG - spv_check_vec(mir->cut_vec); -#endif - /* if there is no integer variable, nothing to generate */ - if (nint == 0) goto done; - /* allocate working arrays */ - u = xcalloc(1+nint, sizeof(double)); - x = xcalloc(1+nint, sizeof(double)); - alpha = xcalloc(1+nint, sizeof(double)); - /* determine u and x */ - for (j = 1; j <= nint; j++) - { k = mir->cut_vec->ind[j]; - xassert(m+1 <= k && k <= m+n); - xassert(mir->isint[k]); - u[j] = mir->ub[k] - mir->lb[k]; - xassert(u[j] >= 1.0); - if (mir->subst[k] == 'L') - x[j] = mir->x[k] - mir->lb[k]; - else if (mir->subst[k] == 'U') - x[j] = mir->ub[k] - mir->x[k]; - else - xassert(k != k); -#if 0 /* 06/III-2016; notorious bug reported many times */ - xassert(x[j] >= -0.001); -#else - if (x[j] < -0.001) - { xprintf("glp_mir_gen: warning: x[%d] = %g\n", j, x[j]); - r_best = 0.0; - goto skip; - } -#endif - if (x[j] < 0.0) x[j] = 0.0; - } - /* compute s = - sum of continuous terms */ - s = 0.0; - for (j = nint+1; j <= mir->cut_vec->nnz; j++) - { double x; - k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - /* must be continuous */ - xassert(!mir->isint[k]); - if (mir->subst[k] == 'L') - { xassert(mir->lb[k] != -DBL_MAX); - kk = mir->vlb[k]; - if (kk == 0) - x = mir->x[k] - mir->lb[k]; - else - x = mir->x[k] - mir->lb[k] * mir->x[kk]; - } - else if (mir->subst[k] == 'U') - { xassert(mir->ub[k] != +DBL_MAX); - kk = mir->vub[k]; - if (kk == 0) - x = mir->ub[k] - mir->x[k]; - else - x = mir->ub[k] * mir->x[kk] - mir->x[k]; - } - else - xassert(k != k); -#if 0 /* 06/III-2016; notorious bug reported many times */ - xassert(x >= -0.001); -#else - if (x < -0.001) - { xprintf("glp_mir_gen: warning: x = %g\n", x); - r_best = 0.0; - goto skip; - } -#endif - if (x < 0.0) x = 0.0; - s -= mir->cut_vec->val[j] * x; - } - xassert(s >= 0.0); - /* apply heuristic to obtain most violated c-MIR inequality */ - b = mir->cut_rhs; - r_best = cmir_sep(nint, mir->cut_vec->val, b, u, x, s, alpha, - &beta, &gamma); - if (r_best == 0.0) goto skip; - xassert(r_best > 0.0); - /* convert to raw cut */ - /* sum alpha[j] * x[j] <= beta + gamma * s */ - for (j = 1; j <= nint; j++) - mir->cut_vec->val[j] = alpha[j]; - for (j = nint+1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - if (k <= m+n) mir->cut_vec->val[j] *= gamma; - } - mir->cut_rhs = beta; -#if MIR_DEBUG - spv_check_vec(mir->cut_vec); -#endif -skip: /* free working arrays */ - xfree(u); - xfree(x); - xfree(alpha); -done: return r_best; -} - -#if MIR_DEBUG -static void check_raw_cut(glp_mir *mir, double r_best) -{ /* check raw cut before back bound substitution */ - int m = mir->m; - int n = mir->n; - int j, k, kk; - double r, big, x; - /* compute the residual r = sum a[k] * x[k] - b and determine - big = max(1, |a[k]|, |b|) */ - r = 0.0, big = 1.0; - for (j = 1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->subst[k] == 'L') - { xassert(mir->lb[k] != -DBL_MAX); - kk = mir->vlb[k]; - if (kk == 0) - x = mir->x[k] - mir->lb[k]; - else - x = mir->x[k] - mir->lb[k] * mir->x[kk]; - } - else if (mir->subst[k] == 'U') - { xassert(mir->ub[k] != +DBL_MAX); - kk = mir->vub[k]; - if (kk == 0) - x = mir->ub[k] - mir->x[k]; - else - x = mir->ub[k] * mir->x[kk] - mir->x[k]; - } - else - xassert(k != k); - r += mir->cut_vec->val[j] * x; - if (big < fabs(mir->cut_vec->val[j])) - big = fabs(mir->cut_vec->val[j]); - } - r -= mir->cut_rhs; - if (big < fabs(mir->cut_rhs)) - big = fabs(mir->cut_rhs); - /* the residual must be close to r_best */ - xassert(fabs(r - r_best) <= 1e-6 * big); - return; -} -#endif - -static void back_subst(glp_mir *mir) -{ /* back substitution of original bounds */ - int m = mir->m; - int n = mir->n; - int j, jj, k, kk; - /* at first, restore bounds of integer variables (because on - restoring variable bounds of continuous variables we need - original, not shifted, bounds of integer variables) */ - for (j = 1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (!mir->isint[k]) continue; /* skip continuous */ - if (mir->subst[k] == 'L') - { /* x'[k] = x[k] - lb[k] */ - xassert(mir->lb[k] != -DBL_MAX); - xassert(mir->vlb[k] == 0); - mir->cut_rhs += mir->cut_vec->val[j] * mir->lb[k]; - } - else if (mir->subst[k] == 'U') - { /* x'[k] = ub[k] - x[k] */ - xassert(mir->ub[k] != +DBL_MAX); - xassert(mir->vub[k] == 0); - mir->cut_rhs -= mir->cut_vec->val[j] * mir->ub[k]; - mir->cut_vec->val[j] = - mir->cut_vec->val[j]; - } - else - xassert(k != k); - } - /* now restore bounds of continuous variables */ - for (j = 1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (mir->isint[k]) continue; /* skip integer */ - if (mir->subst[k] == 'L') - { /* x'[k] = x[k] - (lower bound) */ - xassert(mir->lb[k] != -DBL_MAX); - kk = mir->vlb[k]; - if (kk == 0) - { /* x'[k] = x[k] - lb[k] */ - mir->cut_rhs += mir->cut_vec->val[j] * mir->lb[k]; - } - else - { /* x'[k] = x[k] - lb[k] * x[kk] */ - jj = mir->cut_vec->pos[kk]; -#if 0 - xassert(jj != 0); -#else - if (jj == 0) - { spv_set_vj(mir->cut_vec, kk, 1.0); - jj = mir->cut_vec->pos[kk]; - xassert(jj != 0); - mir->cut_vec->val[jj] = 0.0; - } -#endif - mir->cut_vec->val[jj] -= mir->cut_vec->val[j] * - mir->lb[k]; - } - } - else if (mir->subst[k] == 'U') - { /* x'[k] = (upper bound) - x[k] */ - xassert(mir->ub[k] != +DBL_MAX); - kk = mir->vub[k]; - if (kk == 0) - { /* x'[k] = ub[k] - x[k] */ - mir->cut_rhs -= mir->cut_vec->val[j] * mir->ub[k]; - } - else - { /* x'[k] = ub[k] * x[kk] - x[k] */ - jj = mir->cut_vec->pos[kk]; - if (jj == 0) - { spv_set_vj(mir->cut_vec, kk, 1.0); - jj = mir->cut_vec->pos[kk]; - xassert(jj != 0); - mir->cut_vec->val[jj] = 0.0; - } - mir->cut_vec->val[jj] += mir->cut_vec->val[j] * - mir->ub[k]; - } - mir->cut_vec->val[j] = - mir->cut_vec->val[j]; - } - else - xassert(k != k); - } -#if MIR_DEBUG - spv_check_vec(mir->cut_vec); -#endif - return; -} - -#if MIR_DEBUG -static void check_cut_row(glp_mir *mir, double r_best) -{ /* check the cut after back bound substitution or elimination of - auxiliary variables */ - int m = mir->m; - int n = mir->n; - int j, k; - double r, big; - /* compute the residual r = sum a[k] * x[k] - b and determine - big = max(1, |a[k]|, |b|) */ - r = 0.0, big = 1.0; - for (j = 1; j <= mir->cut_vec->nnz; j++) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - r += mir->cut_vec->val[j] * mir->x[k]; - if (big < fabs(mir->cut_vec->val[j])) - big = fabs(mir->cut_vec->val[j]); - } - r -= mir->cut_rhs; - if (big < fabs(mir->cut_rhs)) - big = fabs(mir->cut_rhs); - /* the residual must be close to r_best */ - xassert(fabs(r - r_best) <= 1e-6 * big); - return; -} -#endif - -static void subst_aux_vars(glp_prob *mip, glp_mir *mir) -{ /* final substitution to eliminate auxiliary variables */ - int m = mir->m; - int n = mir->n; - GLPAIJ *aij; - int j, k, kk, jj; - for (j = mir->cut_vec->nnz; j >= 1; j--) - { k = mir->cut_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (k > m) continue; /* skip structurals */ - for (aij = mip->row[k]->ptr; aij != NULL; aij = aij->r_next) - { kk = m + aij->col->j; /* structural */ - jj = mir->cut_vec->pos[kk]; - if (jj == 0) - { spv_set_vj(mir->cut_vec, kk, 1.0); - jj = mir->cut_vec->pos[kk]; - mir->cut_vec->val[jj] = 0.0; - } - mir->cut_vec->val[jj] += mir->cut_vec->val[j] * aij->val; - } - mir->cut_vec->val[j] = 0.0; - } - spv_clean_vec(mir->cut_vec, 0.0); - return; -} - -static void add_cut(glp_mir *mir, glp_prob *pool) -{ /* add constructed cut inequality to the cut pool */ - int m = mir->m; - int n = mir->n; - int j, k, len; - int *ind = xcalloc(1+n, sizeof(int)); - double *val = xcalloc(1+n, sizeof(double)); - len = 0; - for (j = mir->cut_vec->nnz; j >= 1; j--) - { k = mir->cut_vec->ind[j]; - xassert(m+1 <= k && k <= m+n); - len++, ind[len] = k - m, val[len] = mir->cut_vec->val[j]; - } -#if 0 -#if 0 - ios_add_cut_row(tree, pool, GLP_RF_MIR, len, ind, val, GLP_UP, - mir->cut_rhs); -#else - glp_ios_add_row(tree, NULL, GLP_RF_MIR, 0, len, ind, val, GLP_UP, - mir->cut_rhs); -#endif -#else - { int i; - i = glp_add_rows(pool, 1); - glp_set_row_bnds(pool, i, GLP_UP, 0, mir->cut_rhs); - glp_set_mat_row(pool, i, len, ind, val); - } -#endif - xfree(ind); - xfree(val); - return; -} - -#if 0 /* 29/II-2016 by Chris */ -static int aggregate_row(glp_prob *mip, glp_mir *mir) -#else -static int aggregate_row(glp_prob *mip, glp_mir *mir, SPV *v) -#endif -{ /* try to aggregate another row */ - int m = mir->m; - int n = mir->n; - GLPAIJ *aij; -#if 0 /* 29/II-2016 by Chris */ - SPV *v; -#endif - int ii, j, jj, k, kk, kappa = 0, ret = 0; - double d1, d2, d, d_max = 0.0; - /* choose appropriate structural variable in the aggregated row - to be substituted */ - for (j = 1; j <= mir->agg_vec->nnz; j++) - { k = mir->agg_vec->ind[j]; - xassert(1 <= k && k <= m+n); - if (k <= m) continue; /* skip auxiliary var */ - if (mir->isint[k]) continue; /* skip integer var */ - if (fabs(mir->agg_vec->val[j]) < 0.001) continue; - /* compute distance from x[k] to its lower bound */ - kk = mir->vlb[k]; - if (kk == 0) - { if (mir->lb[k] == -DBL_MAX) - d1 = DBL_MAX; - else - d1 = mir->x[k] - mir->lb[k]; - } - else - { xassert(1 <= kk && kk <= m+n); - xassert(mir->isint[kk]); - xassert(mir->lb[k] != -DBL_MAX); - d1 = mir->x[k] - mir->lb[k] * mir->x[kk]; - } - /* compute distance from x[k] to its upper bound */ - kk = mir->vub[k]; - if (kk == 0) - { if (mir->vub[k] == +DBL_MAX) - d2 = DBL_MAX; - else - d2 = mir->ub[k] - mir->x[k]; - } - else - { xassert(1 <= kk && kk <= m+n); - xassert(mir->isint[kk]); - xassert(mir->ub[k] != +DBL_MAX); - d2 = mir->ub[k] * mir->x[kk] - mir->x[k]; - } - /* x[k] cannot be free */ - xassert(d1 != DBL_MAX || d2 != DBL_MAX); - /* d = min(d1, d2) */ - d = (d1 <= d2 ? d1 : d2); - xassert(d != DBL_MAX); - /* should not be close to corresponding bound */ - if (d < 0.001) continue; - if (d_max < d) d_max = d, kappa = k; - } - if (kappa == 0) - { /* nothing chosen */ - ret = 1; - goto done; - } - /* x[kappa] has been chosen */ - xassert(m+1 <= kappa && kappa <= m+n); - xassert(!mir->isint[kappa]); - /* find another row, which have not been used yet, to eliminate - x[kappa] from the aggregated row */ -#if 0 /* 29/II-2016 by Chris */ - for (ii = 1; ii <= m; ii++) - { if (mir->skip[ii]) continue; - for (aij = mip->row[ii]->ptr; aij != NULL; aij = aij->r_next) - if (aij->col->j == kappa - m) break; - if (aij != NULL && fabs(aij->val) >= 0.001) break; -#else - ii = 0; - for (aij = mip->col[kappa - m]->ptr; aij != NULL; - aij = aij->c_next) - { if (aij->row->i > m) continue; - if (mir->skip[aij->row->i]) continue; - if (fabs(aij->val) >= 0.001) - { ii = aij->row->i; - break; - } -#endif - } -#if 0 /* 29/II-2016 by Chris */ - if (ii > m) -#else - if (ii == 0) -#endif - { /* nothing found */ - ret = 2; - goto done; - } - /* row ii has been found; include it in the aggregated list */ - mir->agg_cnt++; - xassert(mir->agg_cnt <= MAXAGGR); - mir->agg_row[mir->agg_cnt] = ii; - mir->skip[ii] = 2; - /* v := new row */ -#if 0 /* 29/II-2016 by Chris */ - v = ios_create_vec(m+n); -#else - spv_clear_vec(v); -#endif - spv_set_vj(v, ii, 1.0); - for (aij = mip->row[ii]->ptr; aij != NULL; aij = aij->r_next) - spv_set_vj(v, m + aij->col->j, - aij->val); -#if MIR_DEBUG - spv_check_vec(v); -#endif - /* perform gaussian elimination to remove x[kappa] */ - j = mir->agg_vec->pos[kappa]; - xassert(j != 0); - jj = v->pos[kappa]; - xassert(jj != 0); - spv_linear_comb(mir->agg_vec, - - mir->agg_vec->val[j] / v->val[jj], v); -#if 0 /* 29/II-2016 by Chris */ - ios_delete_vec(v); -#endif - spv_set_vj(mir->agg_vec, kappa, 0.0); -#if MIR_DEBUG - spv_check_vec(mir->agg_vec); -#endif -done: return ret; -} - -int glp_mir_gen(glp_prob *mip, glp_mir *mir, glp_prob *pool) -{ /* main routine to generate MIR cuts */ - int m = mir->m; - int n = mir->n; - int i, nnn = 0; - double r_best; -#if 1 /* 29/II-2016 by Chris */ - SPV *work; -#endif - xassert(mip->m >= m); - xassert(mip->n == n); - /* obtain current point */ - get_current_point(mip, mir); -#if MIR_DEBUG - /* check current point */ - check_current_point(mir); -#endif - /* reset bound substitution flags */ - memset(&mir->subst[1], '?', m+n); -#if 1 /* 29/II-2016 by Chris */ - work = spv_create_vec(m+n); -#endif - /* try to generate a set of violated MIR cuts */ - for (i = 1; i <= m; i++) - { if (mir->skip[i]) continue; - /* use original i-th row as initial aggregated constraint */ - initial_agg_row(mip, mir, i); -loop: ; -#if MIR_DEBUG - /* check aggregated row */ - check_agg_row(mir); -#endif - /* substitute fixed variables into aggregated constraint */ - subst_fixed_vars(mir); -#if MIR_DEBUG - /* check aggregated row */ - check_agg_row(mir); -#endif -#if MIR_DEBUG - /* check bound substitution flags */ - { int k; - for (k = 1; k <= m+n; k++) - xassert(mir->subst[k] == '?'); - } -#endif - /* apply bound substitution heuristic */ - bound_subst_heur(mir); - /* substitute bounds and build modified constraint */ - build_mod_row(mir); -#if MIR_DEBUG - /* check modified row */ - check_mod_row(mir); -#endif - /* try to generate violated c-MIR cut for modified row */ - r_best = generate(mir); - if (r_best > 0.0) - { /* success */ -#if MIR_DEBUG - /* check raw cut before back bound substitution */ - check_raw_cut(mir, r_best); -#endif - /* back substitution of original bounds */ - back_subst(mir); -#if MIR_DEBUG - /* check the cut after back bound substitution */ - check_cut_row(mir, r_best); -#endif - /* final substitution to eliminate auxiliary variables */ - subst_aux_vars(mip, mir); -#if MIR_DEBUG - /* check the cut after elimination of auxiliaries */ - check_cut_row(mir, r_best); -#endif - /* add constructed cut inequality to the cut pool */ - add_cut(mir, pool), nnn++; - } - /* reset bound substitution flags */ - { int j, k; - for (j = 1; j <= mir->mod_vec->nnz; j++) - { k = mir->mod_vec->ind[j]; - xassert(1 <= k && k <= m+n); - xassert(mir->subst[k] != '?'); - mir->subst[k] = '?'; - } - } - if (r_best == 0.0) - { /* failure */ - if (mir->agg_cnt < MAXAGGR) - { /* try to aggregate another row */ -#if 0 /* 29/II-2016 by Chris */ - if (aggregate_row(mip, mir) == 0) goto loop; -#else - if (aggregate_row(mip, mir, work) == 0) goto loop; -#endif - } - } - /* unmark rows used in the aggregated constraint */ - { int k, ii; - for (k = 1; k <= mir->agg_cnt; k++) - { ii = mir->agg_row[k]; - xassert(1 <= ii && ii <= m); - xassert(mir->skip[ii] == 2); - mir->skip[ii] = 0; - } - } - } -#if 1 /* 29/II-2016 by Chris */ - spv_delete_vec(work); -#endif - return nnn; -} - -/*********************************************************************** -* NAME -* -* glp_mir_free - delete MIR cut generator workspace -* -* SYNOPSIS -* -* void glp_mir_free(glp_mir *mir); -* -* DESCRIPTION -* -* This routine deletes the MIR cut generator workspace and frees all -* the memory allocated to it. */ - -void glp_mir_free(glp_mir *mir) -{ xfree(mir->skip); - xfree(mir->isint); - xfree(mir->lb); - xfree(mir->vlb); - xfree(mir->ub); - xfree(mir->vub); - xfree(mir->x); - xfree(mir->agg_row); - spv_delete_vec(mir->agg_vec); - xfree(mir->subst); - spv_delete_vec(mir->mod_vec); - spv_delete_vec(mir->cut_vec); - xfree(mir); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/spv.c b/code/3rd_glpk/intopt/spv.c deleted file mode 100644 index 502f3cd0..00000000 --- a/code/3rd_glpk/intopt/spv.c +++ /dev/null @@ -1,303 +0,0 @@ -/* spv.c (operations on sparse vectors) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2007-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spv.h" - -/*********************************************************************** -* NAME -* -* spv_create_vec - create sparse vector -* -* SYNOPSIS -* -* #include "glpios.h" -* SPV *spv_create_vec(int n); -* -* DESCRIPTION -* -* The routine spv_create_vec creates a sparse vector of dimension n, -* which initially is a null vector. -* -* RETURNS -* -* The routine returns a pointer to the vector created. */ - -SPV *spv_create_vec(int n) -{ SPV *v; - xassert(n >= 0); - v = xmalloc(sizeof(SPV)); - v->n = n; - v->nnz = 0; - v->pos = xcalloc(1+n, sizeof(int)); - memset(&v->pos[1], 0, n * sizeof(int)); - v->ind = xcalloc(1+n, sizeof(int)); - v->val = xcalloc(1+n, sizeof(double)); - return v; -} - -/*********************************************************************** -* NAME -* -* spv_check_vec - check that sparse vector has correct representation -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_check_vec(SPV *v); -* -* DESCRIPTION -* -* The routine spv_check_vec checks that a sparse vector specified by -* the parameter v has correct representation. -* -* NOTE -* -* Complexity of this operation is O(n). */ - -void spv_check_vec(SPV *v) -{ int j, k, nnz; - xassert(v->n >= 0); - nnz = 0; - for (j = v->n; j >= 1; j--) - { k = v->pos[j]; - xassert(0 <= k && k <= v->nnz); - if (k != 0) - { xassert(v->ind[k] == j); - nnz++; - } - } - xassert(v->nnz == nnz); - return; -} - -/*********************************************************************** -* NAME -* -* spv_get_vj - retrieve component of sparse vector -* -* SYNOPSIS -* -* #include "glpios.h" -* double spv_get_vj(SPV *v, int j); -* -* RETURNS -* -* The routine spv_get_vj returns j-th component of a sparse vector -* specified by the parameter v. */ - -double spv_get_vj(SPV *v, int j) -{ int k; - xassert(1 <= j && j <= v->n); - k = v->pos[j]; - xassert(0 <= k && k <= v->nnz); - return (k == 0 ? 0.0 : v->val[k]); -} - -/*********************************************************************** -* NAME -* -* spv_set_vj - set/change component of sparse vector -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_set_vj(SPV *v, int j, double val); -* -* DESCRIPTION -* -* The routine spv_set_vj assigns val to j-th component of a sparse -* vector specified by the parameter v. */ - -void spv_set_vj(SPV *v, int j, double val) -{ int k; - xassert(1 <= j && j <= v->n); - k = v->pos[j]; - if (val == 0.0) - { if (k != 0) - { /* remove j-th component */ - v->pos[j] = 0; - if (k < v->nnz) - { v->pos[v->ind[v->nnz]] = k; - v->ind[k] = v->ind[v->nnz]; - v->val[k] = v->val[v->nnz]; - } - v->nnz--; - } - } - else - { if (k == 0) - { /* create j-th component */ - k = ++(v->nnz); - v->pos[j] = k; - v->ind[k] = j; - } - v->val[k] = val; - } - return; -} - -/*********************************************************************** -* NAME -* -* spv_clear_vec - set all components of sparse vector to zero -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_clear_vec(SPV *v); -* -* DESCRIPTION -* -* The routine spv_clear_vec sets all components of a sparse vector -* specified by the parameter v to zero. */ - -void spv_clear_vec(SPV *v) -{ int k; - for (k = 1; k <= v->nnz; k++) - v->pos[v->ind[k]] = 0; - v->nnz = 0; - return; -} - -/*********************************************************************** -* NAME -* -* spv_clean_vec - remove zero or small components from sparse vector -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_clean_vec(SPV *v, double eps); -* -* DESCRIPTION -* -* The routine spv_clean_vec removes zero components and components -* whose magnitude is less than eps from a sparse vector specified by -* the parameter v. If eps is 0.0, only zero components are removed. */ - -void spv_clean_vec(SPV *v, double eps) -{ int k, nnz; - nnz = 0; - for (k = 1; k <= v->nnz; k++) - { if (fabs(v->val[k]) == 0.0 || fabs(v->val[k]) < eps) - { /* remove component */ - v->pos[v->ind[k]] = 0; - } - else - { /* keep component */ - nnz++; - v->pos[v->ind[k]] = nnz; - v->ind[nnz] = v->ind[k]; - v->val[nnz] = v->val[k]; - } - } - v->nnz = nnz; - return; -} - -/*********************************************************************** -* NAME -* -* spv_copy_vec - copy sparse vector (x := y) -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_copy_vec(SPV *x, SPV *y); -* -* DESCRIPTION -* -* The routine spv_copy_vec copies a sparse vector specified by the -* parameter y to a sparse vector specified by the parameter x. */ - -void spv_copy_vec(SPV *x, SPV *y) -{ int j; - xassert(x != y); - xassert(x->n == y->n); - spv_clear_vec(x); - x->nnz = y->nnz; - memcpy(&x->ind[1], &y->ind[1], x->nnz * sizeof(int)); - memcpy(&x->val[1], &y->val[1], x->nnz * sizeof(double)); - for (j = 1; j <= x->nnz; j++) - x->pos[x->ind[j]] = j; - return; -} - -/*********************************************************************** -* NAME -* -* spv_linear_comb - compute linear combination (x := x + a * y) -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_linear_comb(SPV *x, double a, SPV *y); -* -* DESCRIPTION -* -* The routine spv_linear_comb computes the linear combination -* -* x := x + a * y, -* -* where x and y are sparse vectors, a is a scalar. */ - -void spv_linear_comb(SPV *x, double a, SPV *y) -{ int j, k; - double xj, yj; - xassert(x != y); - xassert(x->n == y->n); - for (k = 1; k <= y->nnz; k++) - { j = y->ind[k]; - xj = spv_get_vj(x, j); - yj = y->val[k]; - spv_set_vj(x, j, xj + a * yj); - } - return; -} - -/*********************************************************************** -* NAME -* -* spv_delete_vec - delete sparse vector -* -* SYNOPSIS -* -* #include "glpios.h" -* void spv_delete_vec(SPV *v); -* -* DESCRIPTION -* -* The routine spv_delete_vec deletes a sparse vector specified by the -* parameter v freeing all the memory allocated to this object. */ - -void spv_delete_vec(SPV *v) -{ /* delete sparse vector */ - xfree(v->pos); - xfree(v->ind); - xfree(v->val); - xfree(v); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/intopt/spv.h b/code/3rd_glpk/intopt/spv.h deleted file mode 100644 index d7d4699f..00000000 --- a/code/3rd_glpk/intopt/spv.h +++ /dev/null @@ -1,83 +0,0 @@ -/* spv.h (operations on sparse vectors) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2007-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPV_H -#define SPV_H - -typedef struct SPV SPV; - -struct SPV -{ /* sparse vector v = (v[j]) */ - int n; - /* dimension, n >= 0 */ - int nnz; - /* number of non-zero components, 0 <= nnz <= n */ - int *pos; /* int pos[1+n]; */ - /* pos[j] = k, 1 <= j <= n, is position of (non-zero) v[j] in the - * arrays ind and val, where 1 <= k <= nnz; pos[j] = 0 means that - * v[j] is structural zero */ - int *ind; /* int ind[1+n]; */ - /* ind[k] = j, 1 <= k <= nnz, is index of v[j] */ - double *val; /* double val[1+n]; */ - /* val[k], 1 <= k <= nnz, is a numeric value of v[j] */ -}; - -#define spv_create_vec _glp_spv_create_vec -SPV *spv_create_vec(int n); -/* create sparse vector */ - -#define spv_check_vec _glp_spv_check_vec -void spv_check_vec(SPV *v); -/* check that sparse vector has correct representation */ - -#define spv_get_vj _glp_spv_get_vj -double spv_get_vj(SPV *v, int j); -/* retrieve component of sparse vector */ - -#define spv_set_vj _glp_spv_set_vj -void spv_set_vj(SPV *v, int j, double val); -/* set/change component of sparse vector */ - -#define spv_clear_vec _glp_spv_clear_vec -void spv_clear_vec(SPV *v); -/* set all components of sparse vector to zero */ - -#define spv_clean_vec _glp_spv_clean_vec -void spv_clean_vec(SPV *v, double eps); -/* remove zero or small components from sparse vector */ - -#define spv_copy_vec _glp_spv_copy_vec -void spv_copy_vec(SPV *x, SPV *y); -/* copy sparse vector (x := y) */ - -#define spv_linear_comb _glp_spv_linear_comb -void spv_linear_comb(SPV *x, double a, SPV *y); -/* compute linear combination (x := x + a * y) */ - -#define spv_delete_vec _glp_spv_delete_vec -void spv_delete_vec(SPV *v); -/* delete sparse vector */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/minisat/LICENSE b/code/3rd_glpk/minisat/LICENSE deleted file mode 100644 index 8a6b9f36..00000000 --- a/code/3rd_glpk/minisat/LICENSE +++ /dev/null @@ -1,20 +0,0 @@ -MiniSat -- Copyright (c) 2005, Niklas Sorensson - -Permission is hereby granted, free of charge, to any person obtaining a -copy of this software and associated documentation files (the -"Software"), to deal in the Software without restriction, including -without limitation the rights to use, copy, modify, merge, publish, -distribute, sublicense, and/or sell copies of the Software, and to -permit persons to whom the Software is furnished to do so, subject to -the following conditions: - -The above copyright notice and this permission notice shall be included -in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/code/3rd_glpk/minisat/README b/code/3rd_glpk/minisat/README deleted file mode 100644 index c183c7d6..00000000 --- a/code/3rd_glpk/minisat/README +++ /dev/null @@ -1,22 +0,0 @@ -NOTE: Files in this subdirectory are NOT part of the GLPK package, but - are used with GLPK. - - The original code was modified according to GLPK requirements by - Andrew Makhorin . -************************************************************************ -MiniSat-C v1.14.1 -======================================== - -* Fixed some serious bugs. -* Tweaked to be Visual Studio friendly (by Alan Mishchenko). - This disabled reading of gzipped DIMACS files and signal handling, - but none of these features are essential (and easy to re-enable, if - wanted). - -MiniSat-C v1.14 -======================================== - -Ok, we get it. You hate C++. You hate templates. We agree; C++ is a -seriously messed up language. Although we are more pragmatic about the -quirks and maldesigns in C++, we sympathize with you. So here is a -pure C version of MiniSat, put together by Niklas Sorensson. diff --git a/code/3rd_glpk/minisat/minisat.c b/code/3rd_glpk/minisat/minisat.c deleted file mode 100644 index 2432d650..00000000 --- a/code/3rd_glpk/minisat/minisat.c +++ /dev/null @@ -1,1315 +0,0 @@ -/* minisat.c */ - -/* Modified by Andrew Makhorin , August 2011 */ -/* May 2017: Changes were made to provide 64-bit portability; thanks to - * Chris Matrakidis for patch */ - -/*********************************************************************** -* MiniSat -- Copyright (c) 2005, Niklas Sorensson -* http://www.cs.chalmers.se/Cs/Research/FormalMethods/MiniSat/ -* -* Permission is hereby granted, free of charge, to any person -* obtaining a copy of this software and associated documentation files -* (the "Software"), to deal in the Software without restriction, -* including without limitation the rights to use, copy, modify, merge, -* publish, distribute, sublicense, and/or sell copies of the Software, -* and to permit persons to whom the Software is furnished to do so, -* subject to the following conditions: -* -* The above copyright notice and this permission notice shall be -* included in all copies or substantial portions of the Software. -* -* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS -* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN -* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN -* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -* SOFTWARE. -***********************************************************************/ -/* Modified to compile with MS Visual Studio 6.0 by Alan Mishchenko */ - -#include "env.h" -#include "minisat.h" - -#if 1 /* by mao */ -static void *ymalloc(int size) -{ void *ptr; - xassert(size > 0); - ptr = malloc(size); - if (ptr == NULL) - xerror("MiniSat: no memory available\n"); - return ptr; -} - -static void *yrealloc(void *ptr, int size) -{ xassert(size > 0); - if (ptr == NULL) - ptr = malloc(size); - else - ptr = realloc(ptr, size); - if (ptr == NULL) - xerror("MiniSat: no memory available\n"); - return ptr; -} - -static void yfree(void *ptr) -{ xassert(ptr != NULL); - free(ptr); - return; -} - -#define assert xassert -#define printf xprintf -#define fflush(f) /* nop */ -#define malloc ymalloc -#define realloc yrealloc -#define free yfree -#define inline /* empty */ -#endif - -/*====================================================================*/ -/* Debug: */ - -#if 0 -#define VERBOSEDEBUG 1 -#endif - -/* For derivation output (verbosity level 2) */ -#define L_IND "%-*d" -#define L_ind solver_dlevel(s)*3+3,solver_dlevel(s) -#define L_LIT "%sx%d" -#define L_lit(p) lit_sign(p)?"~":"", (lit_var(p)) - -#if 0 /* by mao */ -/* Just like 'assert()' but expression will be evaluated in the release - version as well. */ -static inline void check(int expr) { assert(expr); } -#endif - -#if 0 /* by mao */ -static void printlits(lit* begin, lit* end) -{ - int i; - for (i = 0; i < end - begin; i++) - printf(L_LIT" ",L_lit(begin[i])); -} -#endif - -/*====================================================================*/ -/* Random numbers: */ - -/* Returns a random float 0 <= x < 1. Seed must never be 0. */ -static inline double drand(double* seed) { - int q; - *seed *= 1389796; - q = (int)(*seed / 2147483647); - *seed -= (double)q * 2147483647; - return *seed / 2147483647; } - -/* Returns a random integer 0 <= x < size. Seed must never be 0. */ -static inline int irand(double* seed, int size) { - return (int)(drand(seed) * size); } - -/*====================================================================*/ -/* Predeclarations: */ - -static void sort(void** array, int size, - int(*comp)(const void *, const void *)); - -/*====================================================================*/ -/* Clause datatype + minor functions: */ - -#if 0 /* by mao; see minisat.h */ -struct clause_t -{ - int size_learnt; - lit lits[0]; -}; -#endif - -#define clause_size(c) ((c)->size_learnt >> 1) - -#define clause_begin(c) ((c)->lits) - -#define clause_learnt(c) ((c)->size_learnt & 1) - -#define clause_activity(c) \ - (*((float*)&(c)->lits[(c)->size_learnt>>1])) - -#define clause_setactivity(c, a) \ - (void)(*((float*)&(c)->lits[(c)->size_learnt>>1]) = (a)) - -/*====================================================================*/ -/* Encode literals in clause pointers: */ - -#if 0 /* 8/I-2017 by cmatraki (64-bit portability) */ -#define clause_from_lit(l) \ - (clause*)((unsigned long)(l) + (unsigned long)(l) + 1) - -#define clause_is_lit(c) \ - ((unsigned long)(c) & 1) - -#define clause_read_lit(c) \ - (lit)((unsigned long)(c) >> 1) -#else -#define clause_from_lit(l) \ - (clause*)((size_t)(l) + (size_t)(l) + 1) - -#define clause_is_lit(c) \ - ((size_t)(c) & 1) - -#define clause_read_lit(c) \ - (lit)((size_t)(c) >> 1) -#endif - -/*====================================================================*/ -/* Simple helpers: */ - -#define solver_dlevel(s) \ - (int)veci_size(&(s)->trail_lim) - -#define solver_read_wlist(s, l) \ - (vecp *)(&(s)->wlists[l]) - -static inline void vecp_remove(vecp* v, void* e) -{ - void** ws = vecp_begin(v); - int j = 0; - - for (; ws[j] != e ; j++); - assert(j < vecp_size(v)); - for (; j < vecp_size(v)-1; j++) ws[j] = ws[j+1]; - vecp_resize(v,vecp_size(v)-1); -} - -/*====================================================================*/ -/* Variable order functions: */ - -static inline void order_update(solver* s, int v) -{ /* updateorder */ - int* orderpos = s->orderpos; - double* activity = s->activity; - int* heap = veci_begin(&s->order); - int i = orderpos[v]; - int x = heap[i]; - int parent = (i - 1) / 2; - - assert(s->orderpos[v] != -1); - - while (i != 0 && activity[x] > activity[heap[parent]]){ - heap[i] = heap[parent]; - orderpos[heap[i]] = i; - i = parent; - parent = (i - 1) / 2; - } - heap[i] = x; - orderpos[x] = i; -} - -#define order_assigned(s, v) /* nop */ - -static inline void order_unassigned(solver* s, int v) -{ /* undoorder */ - int* orderpos = s->orderpos; - if (orderpos[v] == -1){ - orderpos[v] = veci_size(&s->order); - veci_push(&s->order,v); - order_update(s,v); - } -} - -static int order_select(solver* s, float random_var_freq) -{ /* selectvar */ - int* heap; - double* activity; - int* orderpos; - - lbool* values = s->assigns; - - /* Random decision: */ - if (drand(&s->random_seed) < random_var_freq){ - int next = irand(&s->random_seed,s->size); - assert(next >= 0 && next < s->size); - if (values[next] == l_Undef) - return next; - } - - /* Activity based decision: */ - - heap = veci_begin(&s->order); - activity = s->activity; - orderpos = s->orderpos; - - while (veci_size(&s->order) > 0){ - int next = heap[0]; - int size = veci_size(&s->order)-1; - int x = heap[size]; - - veci_resize(&s->order,size); - - orderpos[next] = -1; - - if (size > 0){ - double act = activity[x]; - - int i = 0; - int child = 1; - - while (child < size){ - if (child+1 < size - && activity[heap[child]] < activity[heap[child+1]]) - child++; - - assert(child < size); - - if (act >= activity[heap[child]]) - break; - - heap[i] = heap[child]; - orderpos[heap[i]] = i; - i = child; - child = 2 * child + 1; - } - heap[i] = x; - orderpos[heap[i]] = i; - } - - if (values[next] == l_Undef) - return next; - } - - return var_Undef; -} - -/*====================================================================*/ -/* Activity functions: */ - -static inline void act_var_rescale(solver* s) { - double* activity = s->activity; - int i; - for (i = 0; i < s->size; i++) - activity[i] *= 1e-100; - s->var_inc *= 1e-100; -} - -static inline void act_var_bump(solver* s, int v) { - double* activity = s->activity; - if ((activity[v] += s->var_inc) > 1e100) - act_var_rescale(s); - - /* printf("bump %d %f\n", v-1, activity[v]); */ - - if (s->orderpos[v] != -1) - order_update(s,v); - -} - -static inline void act_var_decay(solver* s) - { s->var_inc *= s->var_decay; } - -static inline void act_clause_rescale(solver* s) { - clause** cs = (clause**)vecp_begin(&s->learnts); - int i; - for (i = 0; i < vecp_size(&s->learnts); i++){ - float a = clause_activity(cs[i]); - clause_setactivity(cs[i], a * (float)1e-20); - } - s->cla_inc *= (float)1e-20; -} - -static inline void act_clause_bump(solver* s, clause *c) { - float a = clause_activity(c) + s->cla_inc; - clause_setactivity(c,a); - if (a > 1e20) act_clause_rescale(s); -} - -static inline void act_clause_decay(solver* s) - { s->cla_inc *= s->cla_decay; } - -/*====================================================================*/ -/* Clause functions: */ - -/* pre: size > 1 && no variable occurs twice - */ -static clause* clause_new(solver* s, lit* begin, lit* end, int learnt) -{ - int size; - clause* c; - int i; - - assert(end - begin > 1); - assert(learnt >= 0 && learnt < 2); - size = end - begin; - c = (clause*)malloc(sizeof(clause) - + sizeof(lit) * size + learnt * sizeof(float)); - c->size_learnt = (size << 1) | learnt; -#if 1 /* by mao & cmatraki; non-portable check that is a fundamental \ - * assumption of minisat code: bit 0 is used as a flag (zero \ - * for pointer, one for shifted int) so allocated memory should \ - * be at least 16-bit aligned */ - assert(((size_t)c & 1) == 0); -#endif - - for (i = 0; i < size; i++) - c->lits[i] = begin[i]; - - if (learnt) - *((float*)&c->lits[size]) = 0.0; - - assert(begin[0] >= 0); - assert(begin[0] < s->size*2); - assert(begin[1] >= 0); - assert(begin[1] < s->size*2); - - assert(lit_neg(begin[0]) < s->size*2); - assert(lit_neg(begin[1]) < s->size*2); - - /* vecp_push(solver_read_wlist(s,lit_neg(begin[0])),(void*)c); */ - /* vecp_push(solver_read_wlist(s,lit_neg(begin[1])),(void*)c); */ - - vecp_push(solver_read_wlist(s,lit_neg(begin[0])), - (void*)(size > 2 ? c : clause_from_lit(begin[1]))); - vecp_push(solver_read_wlist(s,lit_neg(begin[1])), - (void*)(size > 2 ? c : clause_from_lit(begin[0]))); - - return c; -} - -static void clause_remove(solver* s, clause* c) -{ - lit* lits = clause_begin(c); - assert(lit_neg(lits[0]) < s->size*2); - assert(lit_neg(lits[1]) < s->size*2); - - /* vecp_remove(solver_read_wlist(s,lit_neg(lits[0])),(void*)c); */ - /* vecp_remove(solver_read_wlist(s,lit_neg(lits[1])),(void*)c); */ - - assert(lits[0] < s->size*2); - vecp_remove(solver_read_wlist(s,lit_neg(lits[0])), - (void*)(clause_size(c) > 2 ? c : clause_from_lit(lits[1]))); - vecp_remove(solver_read_wlist(s,lit_neg(lits[1])), - (void*)(clause_size(c) > 2 ? c : clause_from_lit(lits[0]))); - - if (clause_learnt(c)){ - s->stats.learnts--; - s->stats.learnts_literals -= clause_size(c); - }else{ - s->stats.clauses--; - s->stats.clauses_literals -= clause_size(c); - } - - free(c); -} - -static lbool clause_simplify(solver* s, clause* c) -{ - lit* lits = clause_begin(c); - lbool* values = s->assigns; - int i; - - assert(solver_dlevel(s) == 0); - - for (i = 0; i < clause_size(c); i++){ - lbool sig = !lit_sign(lits[i]); sig += sig - 1; - if (values[lit_var(lits[i])] == sig) - return l_True; - } - return l_False; -} - -/*====================================================================*/ -/* Minor (solver) functions: */ - -void solver_setnvars(solver* s,int n) -{ - int var; - - if (s->cap < n){ - - while (s->cap < n) s->cap = s->cap*2+1; - - s->wlists = (vecp*) realloc(s->wlists, - sizeof(vecp)*s->cap*2); - s->activity = (double*) realloc(s->activity, - sizeof(double)*s->cap); - s->assigns = (lbool*) realloc(s->assigns, - sizeof(lbool)*s->cap); - s->orderpos = (int*) realloc(s->orderpos, - sizeof(int)*s->cap); - s->reasons = (clause**)realloc(s->reasons, - sizeof(clause*)*s->cap); - s->levels = (int*) realloc(s->levels, - sizeof(int)*s->cap); - s->tags = (lbool*) realloc(s->tags, - sizeof(lbool)*s->cap); - s->trail = (lit*) realloc(s->trail, - sizeof(lit)*s->cap); - } - - for (var = s->size; var < n; var++){ - vecp_new(&s->wlists[2*var]); - vecp_new(&s->wlists[2*var+1]); - s->activity [var] = 0; - s->assigns [var] = l_Undef; - s->orderpos [var] = veci_size(&s->order); - s->reasons [var] = (clause*)0; - s->levels [var] = 0; - s->tags [var] = l_Undef; - - /* does not hold because variables enqueued at top level will - not be reinserted in the heap - assert(veci_size(&s->order) == var); - */ - veci_push(&s->order,var); - order_update(s, var); - } - - s->size = n > s->size ? n : s->size; -} - -static inline bool enqueue(solver* s, lit l, clause* from) -{ - lbool* values = s->assigns; - int v = lit_var(l); - lbool val = values[v]; - lbool sig; -#ifdef VERBOSEDEBUG - printf(L_IND"enqueue("L_LIT")\n", L_ind, L_lit(l)); -#endif - - /* lbool */ sig = !lit_sign(l); sig += sig - 1; - if (val != l_Undef){ - return val == sig; - }else{ - /* New fact -- store it. */ - int* levels; - clause** reasons; -#ifdef VERBOSEDEBUG - printf(L_IND"bind("L_LIT")\n", L_ind, L_lit(l)); -#endif - /* int* */ levels = s->levels; - /* clause** */ reasons = s->reasons; - - values [v] = sig; - levels [v] = solver_dlevel(s); - reasons[v] = from; - s->trail[s->qtail++] = l; - - order_assigned(s, v); - return true; - } -} - -static inline void assume(solver* s, lit l){ - assert(s->qtail == s->qhead); - assert(s->assigns[lit_var(l)] == l_Undef); -#ifdef VERBOSEDEBUG - printf(L_IND"assume("L_LIT")\n", L_ind, L_lit(l)); -#endif - veci_push(&s->trail_lim,s->qtail); - enqueue(s,l,(clause*)0); -} - -static inline void solver_canceluntil(solver* s, int level) { - lit* trail; - lbool* values; - clause** reasons; - int bound; - int c; - - if (solver_dlevel(s) <= level) - return; - - trail = s->trail; - values = s->assigns; - reasons = s->reasons; - bound = (veci_begin(&s->trail_lim))[level]; - - for (c = s->qtail-1; c >= bound; c--) { - int x = lit_var(trail[c]); - values [x] = l_Undef; - reasons[x] = (clause*)0; - } - - for (c = s->qhead-1; c >= bound; c--) - order_unassigned(s,lit_var(trail[c])); - - s->qhead = s->qtail = bound; - veci_resize(&s->trail_lim,level); -} - -static void solver_record(solver* s, veci* cls) -{ - lit* begin = veci_begin(cls); - lit* end = begin + veci_size(cls); - clause* c = (veci_size(cls) > 1) ? clause_new(s,begin,end,1) - : (clause*)0; - enqueue(s,*begin,c); - - assert(veci_size(cls) > 0); - - if (c != 0) { - vecp_push(&s->learnts,c); - act_clause_bump(s,c); - s->stats.learnts++; - s->stats.learnts_literals += veci_size(cls); - } -} - -static double solver_progress(solver* s) -{ - lbool* values = s->assigns; - int* levels = s->levels; - int i; - - double progress = 0; - double F = 1.0 / s->size; - for (i = 0; i < s->size; i++) - if (values[i] != l_Undef) - progress += pow(F, levels[i]); - return progress / s->size; -} - -/*====================================================================*/ -/* Major methods: */ - -static bool solver_lit_removable(solver* s, lit l, int minl) -{ - lbool* tags = s->tags; - clause** reasons = s->reasons; - int* levels = s->levels; - int top = veci_size(&s->tagged); - - assert(lit_var(l) >= 0 && lit_var(l) < s->size); - assert(reasons[lit_var(l)] != 0); - veci_resize(&s->stack,0); - veci_push(&s->stack,lit_var(l)); - - while (veci_size(&s->stack) > 0){ - clause* c; - int v = veci_begin(&s->stack)[veci_size(&s->stack)-1]; - assert(v >= 0 && v < s->size); - veci_resize(&s->stack,veci_size(&s->stack)-1); - assert(reasons[v] != 0); - c = reasons[v]; - - if (clause_is_lit(c)){ - int v = lit_var(clause_read_lit(c)); - if (tags[v] == l_Undef && levels[v] != 0){ - if (reasons[v] != 0 - && ((1 << (levels[v] & 31)) & minl)){ - veci_push(&s->stack,v); - tags[v] = l_True; - veci_push(&s->tagged,v); - }else{ - int* tagged = veci_begin(&s->tagged); - int j; - for (j = top; j < veci_size(&s->tagged); j++) - tags[tagged[j]] = l_Undef; - veci_resize(&s->tagged,top); - return false; - } - } - }else{ - lit* lits = clause_begin(c); - int i, j; - - for (i = 1; i < clause_size(c); i++){ - int v = lit_var(lits[i]); - if (tags[v] == l_Undef && levels[v] != 0){ - if (reasons[v] != 0 - && ((1 << (levels[v] & 31)) & minl)){ - - veci_push(&s->stack,lit_var(lits[i])); - tags[v] = l_True; - veci_push(&s->tagged,v); - }else{ - int* tagged = veci_begin(&s->tagged); - for (j = top; j < veci_size(&s->tagged); j++) - tags[tagged[j]] = l_Undef; - veci_resize(&s->tagged,top); - return false; - } - } - } - } - } - - return true; -} - -static void solver_analyze(solver* s, clause* c, veci* learnt) -{ - lit* trail = s->trail; - lbool* tags = s->tags; - clause** reasons = s->reasons; - int* levels = s->levels; - int cnt = 0; - lit p = lit_Undef; - int ind = s->qtail-1; - lit* lits; - int i, j, minl; - int* tagged; - - veci_push(learnt,lit_Undef); - - do{ - assert(c != 0); - - if (clause_is_lit(c)){ - lit q = clause_read_lit(c); - assert(lit_var(q) >= 0 && lit_var(q) < s->size); - if (tags[lit_var(q)] == l_Undef && levels[lit_var(q)] > 0){ - tags[lit_var(q)] = l_True; - veci_push(&s->tagged,lit_var(q)); - act_var_bump(s,lit_var(q)); - if (levels[lit_var(q)] == solver_dlevel(s)) - cnt++; - else - veci_push(learnt,q); - } - }else{ - - if (clause_learnt(c)) - act_clause_bump(s,c); - - lits = clause_begin(c); - /* printlits(lits,lits+clause_size(c)); printf("\n"); */ - for (j = (p == lit_Undef ? 0 : 1); j < clause_size(c); j++){ - lit q = lits[j]; - assert(lit_var(q) >= 0 && lit_var(q) < s->size); - if (tags[lit_var(q)] == l_Undef - && levels[lit_var(q)] > 0){ - tags[lit_var(q)] = l_True; - veci_push(&s->tagged,lit_var(q)); - act_var_bump(s,lit_var(q)); - if (levels[lit_var(q)] == solver_dlevel(s)) - cnt++; - else - veci_push(learnt,q); - } - } - } - - while (tags[lit_var(trail[ind--])] == l_Undef); - - p = trail[ind+1]; - c = reasons[lit_var(p)]; - cnt--; - - }while (cnt > 0); - - *veci_begin(learnt) = lit_neg(p); - - lits = veci_begin(learnt); - minl = 0; - for (i = 1; i < veci_size(learnt); i++){ - int lev = levels[lit_var(lits[i])]; - minl |= 1 << (lev & 31); - } - - /* simplify (full) */ - for (i = j = 1; i < veci_size(learnt); i++){ - if (reasons[lit_var(lits[i])] == 0 - || !solver_lit_removable(s,lits[i],minl)) - lits[j++] = lits[i]; - } - - /* update size of learnt + statistics */ - s->stats.max_literals += veci_size(learnt); - veci_resize(learnt,j); - s->stats.tot_literals += j; - - /* clear tags */ - tagged = veci_begin(&s->tagged); - for (i = 0; i < veci_size(&s->tagged); i++) - tags[tagged[i]] = l_Undef; - veci_resize(&s->tagged,0); - -#ifdef DEBUG - for (i = 0; i < s->size; i++) - assert(tags[i] == l_Undef); -#endif - -#ifdef VERBOSEDEBUG - printf(L_IND"Learnt {", L_ind); - for (i = 0; i < veci_size(learnt); i++) - printf(" "L_LIT, L_lit(lits[i])); -#endif - if (veci_size(learnt) > 1){ - int max_i = 1; - int max = levels[lit_var(lits[1])]; - lit tmp; - - for (i = 2; i < veci_size(learnt); i++) - if (levels[lit_var(lits[i])] > max){ - max = levels[lit_var(lits[i])]; - max_i = i; - } - - tmp = lits[1]; - lits[1] = lits[max_i]; - lits[max_i] = tmp; - } -#ifdef VERBOSEDEBUG - { - int lev = veci_size(learnt) > 1 ? levels[lit_var(lits[1])] : 0; - printf(" } at level %d\n", lev); - } -#endif -} - -clause* solver_propagate(solver* s) -{ - lbool* values = s->assigns; - clause* confl = (clause*)0; - lit* lits; - - /* printf("solver_propagate\n"); */ - while (confl == 0 && s->qtail - s->qhead > 0){ - lit p = s->trail[s->qhead++]; - vecp* ws = solver_read_wlist(s,p); - clause **begin = (clause**)vecp_begin(ws); - clause **end = begin + vecp_size(ws); - clause **i, **j; - - s->stats.propagations++; - s->simpdb_props--; - - /* printf("checking lit %d: "L_LIT"\n", veci_size(ws), - L_lit(p)); */ - for (i = j = begin; i < end; ){ - if (clause_is_lit(*i)){ - *j++ = *i; - if (!enqueue(s,clause_read_lit(*i),clause_from_lit(p))){ - confl = s->binary; - (clause_begin(confl))[1] = lit_neg(p); - (clause_begin(confl))[0] = clause_read_lit(*i++); - - /* Copy the remaining watches: */ - while (i < end) - *j++ = *i++; - } - }else{ - lit false_lit; - lbool sig; - - lits = clause_begin(*i); - - /* Make sure the false literal is data[1]: */ - false_lit = lit_neg(p); - if (lits[0] == false_lit){ - lits[0] = lits[1]; - lits[1] = false_lit; - } - assert(lits[1] == false_lit); - /* printf("checking clause: "); - printlits(lits, lits+clause_size(*i)); - printf("\n"); */ - - /* If 0th watch is true, then clause is already - satisfied. */ - sig = !lit_sign(lits[0]); sig += sig - 1; - if (values[lit_var(lits[0])] == sig){ - *j++ = *i; - }else{ - /* Look for new watch: */ - lit* stop = lits + clause_size(*i); - lit* k; - for (k = lits + 2; k < stop; k++){ - lbool sig = lit_sign(*k); sig += sig - 1; - if (values[lit_var(*k)] != sig){ - lits[1] = *k; - *k = false_lit; - vecp_push(solver_read_wlist(s, - lit_neg(lits[1])),*i); - goto next; } - } - - *j++ = *i; - /* Clause is unit under assignment: */ - if (!enqueue(s,lits[0], *i)){ - confl = *i++; - /* Copy the remaining watches: */ - while (i < end) - *j++ = *i++; - } - } - } - next: - i++; - } - - s->stats.inspects += j - (clause**)vecp_begin(ws); - vecp_resize(ws,j - (clause**)vecp_begin(ws)); - } - - return confl; -} - -static inline int clause_cmp (const void* x, const void* y) { - return clause_size((clause*)x) > 2 - && (clause_size((clause*)y) == 2 - || clause_activity((clause*)x) - < clause_activity((clause*)y)) ? -1 : 1; } - -void solver_reducedb(solver* s) -{ - int i, j; - double extra_lim = s->cla_inc / vecp_size(&s->learnts); - /* Remove any clause below this activity */ - clause** learnts = (clause**)vecp_begin(&s->learnts); - clause** reasons = s->reasons; - - sort(vecp_begin(&s->learnts), vecp_size(&s->learnts), clause_cmp); - - for (i = j = 0; i < vecp_size(&s->learnts) / 2; i++){ - if (clause_size(learnts[i]) > 2 - && reasons[lit_var(*clause_begin(learnts[i]))] - != learnts[i]) - clause_remove(s,learnts[i]); - else - learnts[j++] = learnts[i]; - } - for (; i < vecp_size(&s->learnts); i++){ - if (clause_size(learnts[i]) > 2 - && reasons[lit_var(*clause_begin(learnts[i]))] - != learnts[i] - && clause_activity(learnts[i]) < extra_lim) - clause_remove(s,learnts[i]); - else - learnts[j++] = learnts[i]; - } - - /* printf("reducedb deleted %d\n", vecp_size(&s->learnts) - j); */ - - vecp_resize(&s->learnts,j); -} - -static lbool solver_search(solver* s, int nof_conflicts, - int nof_learnts) -{ - int* levels = s->levels; - double var_decay = 0.95; - double clause_decay = 0.999; - double random_var_freq = 0.02; - - int conflictC = 0; - veci learnt_clause; - - assert(s->root_level == solver_dlevel(s)); - - s->stats.starts++; - s->var_decay = (float)(1 / var_decay ); - s->cla_decay = (float)(1 / clause_decay); - veci_resize(&s->model,0); - veci_new(&learnt_clause); - - for (;;){ - clause* confl = solver_propagate(s); - if (confl != 0){ - /* CONFLICT */ - int blevel; - -#ifdef VERBOSEDEBUG - printf(L_IND"**CONFLICT**\n", L_ind); -#endif - s->stats.conflicts++; conflictC++; - if (solver_dlevel(s) == s->root_level){ - veci_delete(&learnt_clause); - return l_False; - } - - veci_resize(&learnt_clause,0); - solver_analyze(s, confl, &learnt_clause); - blevel = veci_size(&learnt_clause) > 1 - ? levels[lit_var(veci_begin(&learnt_clause)[1])] - : s->root_level; - blevel = s->root_level > blevel ? s->root_level : blevel; - solver_canceluntil(s,blevel); - solver_record(s,&learnt_clause); - act_var_decay(s); - act_clause_decay(s); - - }else{ - /* NO CONFLICT */ - int next; - - if (nof_conflicts >= 0 && conflictC >= nof_conflicts){ - /* Reached bound on number of conflicts: */ - s->progress_estimate = solver_progress(s); - solver_canceluntil(s,s->root_level); - veci_delete(&learnt_clause); - return l_Undef; } - - if (solver_dlevel(s) == 0) - /* Simplify the set of problem clauses: */ - solver_simplify(s); - - if (nof_learnts >= 0 - && vecp_size(&s->learnts) - s->qtail >= nof_learnts) - /* Reduce the set of learnt clauses: */ - solver_reducedb(s); - - /* New variable decision: */ - s->stats.decisions++; - next = order_select(s,(float)random_var_freq); - - if (next == var_Undef){ - /* Model found: */ - lbool* values = s->assigns; - int i; - for (i = 0; i < s->size; i++) - veci_push(&s->model,(int)values[i]); - solver_canceluntil(s,s->root_level); - veci_delete(&learnt_clause); - - /* - veci apa; veci_new(&apa); - for (i = 0; i < s->size; i++) - veci_push(&apa,(int)(s->model.ptr[i] == l_True - ? toLit(i) : lit_neg(toLit(i)))); - printf("model: "); - printlits((lit*)apa.ptr, - (lit*)apa.ptr + veci_size(&apa)); printf("\n"); - veci_delete(&apa); - */ - - return l_True; - } - - assume(s,lit_neg(toLit(next))); - } - } - -#if 0 /* by mao; unreachable code */ - return l_Undef; /* cannot happen */ -#endif -} - -/*====================================================================*/ -/* External solver functions: */ - -solver* solver_new(void) -{ - solver* s = (solver*)malloc(sizeof(solver)); - - /* initialize vectors */ - vecp_new(&s->clauses); - vecp_new(&s->learnts); - veci_new(&s->order); - veci_new(&s->trail_lim); - veci_new(&s->tagged); - veci_new(&s->stack); - veci_new(&s->model); - - /* initialize arrays */ - s->wlists = 0; - s->activity = 0; - s->assigns = 0; - s->orderpos = 0; - s->reasons = 0; - s->levels = 0; - s->tags = 0; - s->trail = 0; - - /* initialize other vars */ - s->size = 0; - s->cap = 0; - s->qhead = 0; - s->qtail = 0; - s->cla_inc = 1; - s->cla_decay = 1; - s->var_inc = 1; - s->var_decay = 1; - s->root_level = 0; - s->simpdb_assigns = 0; - s->simpdb_props = 0; - s->random_seed = 91648253; - s->progress_estimate = 0; - s->binary = (clause*)malloc(sizeof(clause) - + sizeof(lit)*2); - s->binary->size_learnt = (2 << 1); - s->verbosity = 0; - - s->stats.starts = 0; - s->stats.decisions = 0; - s->stats.propagations = 0; - s->stats.inspects = 0; - s->stats.conflicts = 0; - s->stats.clauses = 0; - s->stats.clauses_literals = 0; - s->stats.learnts = 0; - s->stats.learnts_literals = 0; - s->stats.max_literals = 0; - s->stats.tot_literals = 0; - - return s; -} - -void solver_delete(solver* s) -{ - int i; - for (i = 0; i < vecp_size(&s->clauses); i++) - free(vecp_begin(&s->clauses)[i]); - - for (i = 0; i < vecp_size(&s->learnts); i++) - free(vecp_begin(&s->learnts)[i]); - - /* delete vectors */ - vecp_delete(&s->clauses); - vecp_delete(&s->learnts); - veci_delete(&s->order); - veci_delete(&s->trail_lim); - veci_delete(&s->tagged); - veci_delete(&s->stack); - veci_delete(&s->model); - free(s->binary); - - /* delete arrays */ - if (s->wlists != 0){ - int i; - for (i = 0; i < s->size*2; i++) - vecp_delete(&s->wlists[i]); - - /* if one is different from null, all are */ - free(s->wlists); - free(s->activity ); - free(s->assigns ); - free(s->orderpos ); - free(s->reasons ); - free(s->levels ); - free(s->trail ); - free(s->tags ); - } - - free(s); -} - -bool solver_addclause(solver* s, lit* begin, lit* end) -{ - lit *i,*j; - int maxvar; - lbool* values; - lit last; - - if (begin == end) return false; - - /* printlits(begin,end); printf("\n"); */ - /* insertion sort */ - maxvar = lit_var(*begin); - for (i = begin + 1; i < end; i++){ - lit l = *i; - maxvar = lit_var(l) > maxvar ? lit_var(l) : maxvar; - for (j = i; j > begin && *(j-1) > l; j--) - *j = *(j-1); - *j = l; - } - solver_setnvars(s,maxvar+1); - - /* printlits(begin,end); printf("\n"); */ - values = s->assigns; - - /* delete duplicates */ - last = lit_Undef; - for (i = j = begin; i < end; i++){ - /* printf("lit: "L_LIT", value = %d\n", L_lit(*i), - (lit_sign(*i) ? -values[lit_var(*i)] : values[lit_var(*i)])); */ - lbool sig = !lit_sign(*i); sig += sig - 1; - if (*i == lit_neg(last) || sig == values[lit_var(*i)]) - return true; /* tautology */ - else if (*i != last && values[lit_var(*i)] == l_Undef) - last = *j++ = *i; - } - - /* printf("final: "); printlits(begin,j); printf("\n"); */ - - if (j == begin) /* empty clause */ - return false; - else if (j - begin == 1) /* unit clause */ - return enqueue(s,*begin,(clause*)0); - - /* create new clause */ - vecp_push(&s->clauses,clause_new(s,begin,j,0)); - - s->stats.clauses++; - s->stats.clauses_literals += j - begin; - - return true; -} - -bool solver_simplify(solver* s) -{ - clause** reasons; - int type; - - assert(solver_dlevel(s) == 0); - - if (solver_propagate(s) != 0) - return false; - - if (s->qhead == s->simpdb_assigns || s->simpdb_props > 0) - return true; - - reasons = s->reasons; - for (type = 0; type < 2; type++){ - vecp* cs = type ? &s->learnts : &s->clauses; - clause** cls = (clause**)vecp_begin(cs); - - int i, j; - for (j = i = 0; i < vecp_size(cs); i++){ - if (reasons[lit_var(*clause_begin(cls[i]))] != cls[i] && - clause_simplify(s,cls[i]) == l_True) - clause_remove(s,cls[i]); - else - cls[j++] = cls[i]; - } - vecp_resize(cs,j); - } - - s->simpdb_assigns = s->qhead; - /* (shouldn't depend on 'stats' really, but it will do for now) */ - s->simpdb_props = (int)(s->stats.clauses_literals - + s->stats.learnts_literals); - - return true; -} - -bool solver_solve(solver* s, lit* begin, lit* end) -{ - double nof_conflicts = 100; - double nof_learnts = solver_nclauses(s) / 3; - lbool status = l_Undef; - lbool* values = s->assigns; - lit* i; - - /* printf("solve: "); printlits(begin, end); printf("\n"); */ - for (i = begin; i < end; i++){ - switch (lit_sign(*i) ? -values[lit_var(*i)] - : values[lit_var(*i)]){ - case 1: /* l_True: */ - break; - case 0: /* l_Undef */ - assume(s, *i); - if (solver_propagate(s) == NULL) - break; - /* falltrough */ - case -1: /* l_False */ - solver_canceluntil(s, 0); - return false; - } - } - - s->root_level = solver_dlevel(s); - - if (s->verbosity >= 1){ - printf("==================================[MINISAT]============" - "=======================\n"); - printf("| Conflicts | ORIGINAL | LEARNT " - " | Progress |\n"); - printf("| | Clauses Literals | Limit Clauses Litera" - "ls Lit/Cl | |\n"); - printf("=======================================================" - "=======================\n"); - } - - while (status == l_Undef){ - double Ratio = (s->stats.learnts == 0)? 0.0 : - s->stats.learnts_literals / (double)s->stats.learnts; - - if (s->verbosity >= 1){ - printf("| %9.0f | %7.0f %8.0f | %7.0f %7.0f %8.0f %7.1f | %" - "6.3f %% |\n", - (double)s->stats.conflicts, - (double)s->stats.clauses, - (double)s->stats.clauses_literals, - (double)nof_learnts, - (double)s->stats.learnts, - (double)s->stats.learnts_literals, - Ratio, - s->progress_estimate*100); - fflush(stdout); - } - status = solver_search(s,(int)nof_conflicts, (int)nof_learnts); - nof_conflicts *= 1.5; - nof_learnts *= 1.1; - } - if (s->verbosity >= 1) - printf("=======================================================" - "=======================\n"); - - solver_canceluntil(s,0); - return status != l_False; -} - -int solver_nvars(solver* s) -{ - return s->size; -} - -int solver_nclauses(solver* s) -{ - return vecp_size(&s->clauses); -} - -int solver_nconflicts(solver* s) -{ - return (int)s->stats.conflicts; -} - -/*====================================================================*/ -/* Sorting functions (sigh): */ - -static inline void selectionsort(void** array, int size, - int(*comp)(const void *, const void *)) -{ - int i, j, best_i; - void* tmp; - - for (i = 0; i < size-1; i++){ - best_i = i; - for (j = i+1; j < size; j++){ - if (comp(array[j], array[best_i]) < 0) - best_i = j; - } - tmp = array[i]; array[i] = array[best_i]; array[best_i] = tmp; - } -} - -static void sortrnd(void** array, int size, - int(*comp)(const void *, const void *), - double* seed) -{ - if (size <= 15) - selectionsort(array, size, comp); - - else{ - void* pivot = array[irand(seed, size)]; - void* tmp; - int i = -1; - int j = size; - - for(;;){ - do i++; while(comp(array[i], pivot)<0); - do j--; while(comp(pivot, array[j])<0); - - if (i >= j) break; - - tmp = array[i]; array[i] = array[j]; array[j] = tmp; - } - - sortrnd(array , i , comp, seed); - sortrnd(&array[i], size-i, comp, seed); - } -} - -static void sort(void** array, int size, - int(*comp)(const void *, const void *)) -{ - double seed = 91648253; - sortrnd(array,size,comp,&seed); -} - -/* eof */ diff --git a/code/3rd_glpk/minisat/minisat.h b/code/3rd_glpk/minisat/minisat.h deleted file mode 100644 index 2733e8d6..00000000 --- a/code/3rd_glpk/minisat/minisat.h +++ /dev/null @@ -1,230 +0,0 @@ -/* minisat.h */ - -/* Modified by Andrew Makhorin , August 2011 */ - -/*********************************************************************** -* MiniSat -- Copyright (c) 2005, Niklas Sorensson -* http://www.cs.chalmers.se/Cs/Research/FormalMethods/MiniSat/ -* -* Permission is hereby granted, free of charge, to any person -* obtaining a copy of this software and associated documentation files -* (the "Software"), to deal in the Software without restriction, -* including without limitation the rights to use, copy, modify, merge, -* publish, distribute, sublicense, and/or sell copies of the Software, -* and to permit persons to whom the Software is furnished to do so, -* subject to the following conditions: -* -* The above copyright notice and this permission notice shall be -* included in all copies or substantial portions of the Software. -* -* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS -* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN -* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN -* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -* SOFTWARE. -***********************************************************************/ -/* Modified to compile with MS Visual Studio 6.0 by Alan Mishchenko */ - -#ifndef MINISAT_H -#define MINISAT_H - -/*====================================================================*/ -/* Simple types: */ - -typedef int bool; - -#define true 1 -#define false 0 - -typedef int lit; -#if 0 /* by mao */ -typedef char lbool; -#else -typedef int lbool; -#endif - -#define var_Undef (int)(-1) -#define lit_Undef (lit)(-2) - -#define l_Undef (lbool)0 -#define l_True (lbool)1 -#define l_False (lbool)(-1) - -#define toLit(v) (lit)((v) + (v)) -#define lit_neg(l) (lit)((l) ^ 1) -#define lit_var(l) (int)((l) >> 1) -#define lit_sign(l) (int)((l) & 1) - -/*====================================================================*/ -/* Vectors: */ - -/* vector of 32-bit intergers (added for 64-bit portability) */ -typedef struct /* veci_t */ { - int size; - int cap; - int* ptr; -} veci; - -#define veci_new(v) \ -{ (v)->size = 0; \ - (v)->cap = 4; \ - (v)->ptr = (int*)malloc(sizeof(int)*(v)->cap); \ -} - -#define veci_delete(v) free((v)->ptr) - -#define veci_begin(v) ((v)->ptr) - -#define veci_size(v) ((v)->size) - -#define veci_resize(v, k) (void)((v)->size = (k)) -/* only safe to shrink !! */ - -#define veci_push(v, e) \ -{ if ((v)->size == (v)->cap) \ - { int newsize = (v)->cap * 2+1; \ - (v)->ptr = (int*)realloc((v)->ptr,sizeof(int)*newsize); \ - (v)->cap = newsize; \ - } \ - (v)->ptr[(v)->size++] = (e); \ -} - -/* vector of 32- or 64-bit pointers */ -typedef struct /* vecp_t */ { - int size; - int cap; - void** ptr; -} vecp; - -#define vecp_new(v) \ -{ (v)->size = 0; \ - (v)->cap = 4; \ - (v)->ptr = (void**)malloc(sizeof(void*)*(v)->cap); \ -} - -#define vecp_delete(v) free((v)->ptr) - -#define vecp_begin(v) ((v)->ptr) - -#define vecp_size(v) ((v)->size) - -#define vecp_resize(v, k) (void)((v)->size = (k)) -/* only safe to shrink !! */ - -#define vecp_push(v, e) \ -{ if ((v)->size == (v)->cap) \ - { int newsize = (v)->cap * 2+1; \ - (v)->ptr = (void**)realloc((v)->ptr,sizeof(void*)*newsize); \ - (v)->cap = newsize; \ - } \ - (v)->ptr[(v)->size++] = (e); \ -} - -/*====================================================================*/ -/* Solver representation: */ - -typedef struct /* clause_t */ -{ - int size_learnt; - lit lits[1]; -} clause; - -typedef struct /* stats_t */ -{ - double starts, decisions, propagations, inspects, conflicts; - double clauses, clauses_literals, learnts, learnts_literals, - max_literals, tot_literals; -} stats; - -typedef struct /* solver_t */ -{ - int size; /* nof variables */ - int cap; /* size of varmaps */ - int qhead; /* Head index of queue. */ - int qtail; /* Tail index of queue. */ - - /* clauses */ - vecp clauses; /* List of problem constraints. - (contains: clause*) */ - vecp learnts; /* List of learnt clauses. - (contains: clause*) */ - - /* activities */ - double var_inc; /* Amount to bump next variable with. */ - double var_decay; /* INVERSE decay factor for variable - activity: stores 1/decay. */ - float cla_inc; /* Amount to bump next clause with. */ - float cla_decay; /* INVERSE decay factor for clause - activity: stores 1/decay. */ - - vecp* wlists; - double* activity; /* A heuristic measurement of the activity - of a variable. */ - lbool* assigns; /* Current values of variables. */ - int* orderpos; /* Index in variable order. */ - clause** reasons; - int* levels; - lit* trail; - - clause* binary; /* A temporary binary clause */ - lbool* tags; - veci tagged; /* (contains: var) */ - veci stack; /* (contains: var) */ - - veci order; /* Variable order. (heap) (contains: var) */ - veci trail_lim; /* Separator indices for different decision - levels in 'trail'. (contains: int) */ - veci model; /* If problem is solved, this vector - contains the model (contains: lbool). */ - - int root_level; /* Level of first proper decision. */ - int simpdb_assigns;/* Number of top-level assignments at last - 'simplifyDB()'. */ - int simpdb_props; /* Number of propagations before next - 'simplifyDB()'. */ - double random_seed; - double progress_estimate; - int verbosity; /* Verbosity level. - 0=silent, - 1=some progress report, - 2=everything */ - - stats stats; -} solver; - -/*====================================================================*/ -/* Public interface: */ - -#if 1 /* by mao; to keep namespace clean */ -#define solver_new _glp_minisat_new -#define solver_delete _glp_minisat_delete -#define solver_addclause _glp_minisat_addclause -#define solver_simplify _glp_minisat_simplify -#define solver_solve _glp_minisat_solve -#define solver_nvars _glp_minisat_nvars -#define solver_nclauses _glp_minisat_nclauses -#define solver_nconflicts _glp_minisat_nconflicts -#define solver_setnvars _glp_minisat_setnvars -#define solver_propagate _glp_minisat_propagate -#define solver_reducedb _glp_minisat_reducedb -#endif - -solver* solver_new(void); -void solver_delete(solver* s); - -bool solver_addclause(solver* s, lit* begin, lit* end); -bool solver_simplify(solver* s); -bool solver_solve(solver* s, lit* begin, lit* end); - -int solver_nvars(solver* s); -int solver_nclauses(solver* s); -int solver_nconflicts(solver* s); - -void solver_setnvars(solver* s,int n); - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/avl.c b/code/3rd_glpk/misc/avl.c deleted file mode 100644 index c97cf13a..00000000 --- a/code/3rd_glpk/misc/avl.c +++ /dev/null @@ -1,405 +0,0 @@ -/* avl.c (binary search tree) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "avl.h" -#include "dmp.h" -#include "env.h" - -struct AVL -{ /* AVL tree (Adelson-Velsky & Landis binary search tree) */ - DMP *pool; - /* memory pool for allocating nodes */ - AVLNODE *root; - /* pointer to the root node */ - int (*fcmp)(void *info, const void *key1, const void *key2); - /* application-defined key comparison routine */ - void *info; - /* transit pointer passed to the routine fcmp */ - int size; - /* the tree size (the total number of nodes) */ - int height; - /* the tree height */ -}; - -struct AVLNODE -{ /* node of AVL tree */ - const void *key; - /* pointer to the node key (data structure for representing keys - is supplied by the application) */ - int rank; - /* node rank = relative position of the node in its own subtree = - the number of nodes in the left subtree plus one */ - int type; - /* reserved for the application specific information */ - void *link; - /* reserved for the application specific information */ - AVLNODE *up; - /* pointer to the parent node */ - short int flag; - /* node flag: - 0 - this node is the left child of its parent (or this node is - the root of the tree and has no parent) - 1 - this node is the right child of its parent */ - short int bal; - /* node balance = the difference between heights of the right and - left subtrees: - -1 - the left subtree is higher than the right one; - 0 - the left and right subtrees have the same height; - +1 - the left subtree is lower than the right one */ - AVLNODE *left; - /* pointer to the root of the left subtree */ - AVLNODE *right; - /* pointer to the root of the right subtree */ -}; - -AVL *avl_create_tree(int (*fcmp)(void *info, const void *key1, - const void *key2), void *info) -{ /* create AVL tree */ - AVL *tree; - tree = xmalloc(sizeof(AVL)); - tree->pool = dmp_create_pool(); - tree->root = NULL; - tree->fcmp = fcmp; - tree->info = info; - tree->size = 0; - tree->height = 0; - return tree; -} - -int avl_strcmp(void *info, const void *key1, const void *key2) -{ /* compare character string keys */ - xassert(info == info); - return strcmp(key1, key2); -} - -static AVLNODE *rotate_subtree(AVL *tree, AVLNODE *node); - -AVLNODE *avl_insert_node(AVL *tree, const void *key) -{ /* insert new node into AVL tree */ - AVLNODE *p, *q, *r; - short int flag; - /* find an appropriate point for insertion */ - p = NULL; q = tree->root; - while (q != NULL) - { p = q; - if (tree->fcmp(tree->info, key, p->key) <= 0) - { flag = 0; - q = p->left; - p->rank++; - } - else - { flag = 1; - q = p->right; - } - } - /* create new node and insert it into the tree */ - r = dmp_get_atom(tree->pool, sizeof(AVLNODE)); - r->key = key; r->type = 0; r->link = NULL; - r->rank = 1; r->up = p; - r->flag = (short int)(p == NULL ? 0 : flag); - r->bal = 0; r->left = NULL; r->right = NULL; - tree->size++; - if (p == NULL) - tree->root = r; - else - if (flag == 0) p->left = r; else p->right = r; - /* go upstairs to the root and correct all subtrees affected by - insertion */ - while (p != NULL) - { if (flag == 0) - { /* the height of the left subtree of [p] is increased */ - if (p->bal > 0) - { p->bal = 0; - break; - } - if (p->bal < 0) - { rotate_subtree(tree, p); - break; - } - p->bal = -1; flag = p->flag; p = p->up; - } - else - { /* the height of the right subtree of [p] is increased */ - if (p->bal < 0) - { p->bal = 0; - break; - } - if (p->bal > 0) - { rotate_subtree(tree, p); - break; - } - p->bal = +1; flag = p->flag; p = p->up; - } - } - /* if the root has been reached, the height of the entire tree is - increased */ - if (p == NULL) tree->height++; - return r; -} - -void avl_set_node_type(AVLNODE *node, int type) -{ /* assign the type field of specified node */ - node->type = type; - return; -} - -void avl_set_node_link(AVLNODE *node, void *link) -{ /* assign the link field of specified node */ - node->link = link; - return; -} - -AVLNODE *avl_find_node(AVL *tree, const void *key) -{ /* find node in AVL tree */ - AVLNODE *p; - int c; - p = tree->root; - while (p != NULL) - { c = tree->fcmp(tree->info, key, p->key); - if (c == 0) break; - p = (c < 0 ? p->left : p->right); - } - return p; -} - -int avl_get_node_type(AVLNODE *node) -{ /* retrieve the type field of specified node */ - return node->type; -} - -void *avl_get_node_link(AVLNODE *node) -{ /* retrieve the link field of specified node */ - return node->link; -} - -static AVLNODE *find_next_node(AVL *tree, AVLNODE *node) -{ /* find next node in AVL tree */ - AVLNODE *p, *q; - if (tree->root == NULL) return NULL; - p = node; - q = (p == NULL ? tree->root : p->right); - if (q == NULL) - { /* go upstairs from the left subtree */ - for (;;) - { q = p->up; - if (q == NULL) break; - if (p->flag == 0) break; - p = q; - } - } - else - { /* go downstairs into the right subtree */ - for (;;) - { p = q->left; - if (p == NULL) break; - q = p; - } - } - return q; -} - -void avl_delete_node(AVL *tree, AVLNODE *node) -{ /* delete specified node from AVL tree */ - AVLNODE *f, *p, *q, *r, *s, *x, *y; - short int flag; - p = node; - /* if both subtrees of the specified node are non-empty, the node - should be interchanged with the next one, at least one subtree - of which is always empty */ - if (p->left == NULL || p->right == NULL) goto skip; - f = p->up; q = p->left; - r = find_next_node(tree, p); s = r->right; - if (p->right == r) - { if (f == NULL) - tree->root = r; - else - if (p->flag == 0) f->left = r; else f->right = r; - r->rank = p->rank; r->up = f; - r->flag = p->flag; r->bal = p->bal; - r->left = q; r->right = p; - q->up = r; - p->rank = 1; p->up = r; p->flag = 1; - p->bal = (short int)(s == NULL ? 0 : +1); - p->left = NULL; p->right = s; - if (s != NULL) s->up = p; - } - else - { x = p->right; y = r->up; - if (f == NULL) - tree->root = r; - else - if (p->flag == 0) f->left = r; else f->right = r; - r->rank = p->rank; r->up = f; - r->flag = p->flag; r->bal = p->bal; - r->left = q; r->right = x; - q->up = r; x->up = r; y->left = p; - p->rank = 1; p->up = y; p->flag = 0; - p->bal = (short int)(s == NULL ? 0 : +1); - p->left = NULL; p->right = s; - if (s != NULL) s->up = p; - } -skip: /* now the specified node [p] has at least one empty subtree; - go upstairs to the root and adjust the rank field of all nodes - affected by deletion */ - q = p; f = q->up; - while (f != NULL) - { if (q->flag == 0) f->rank--; - q = f; f = q->up; - } - /* delete the specified node from the tree */ - f = p->up; flag = p->flag; - q = p->left != NULL ? p->left : p->right; - if (f == NULL) - tree->root = q; - else - if (flag == 0) f->left = q; else f->right = q; - if (q != NULL) q->up = f, q->flag = flag; - tree->size--; - /* go upstairs to the root and correct all subtrees affected by - deletion */ - while (f != NULL) - { if (flag == 0) - { /* the height of the left subtree of [f] is decreased */ - if (f->bal == 0) - { f->bal = +1; - break; - } - if (f->bal < 0) - f->bal = 0; - else - { f = rotate_subtree(tree, f); - if (f->bal < 0) break; - } - flag = f->flag; f = f->up; - } - else - { /* the height of the right subtree of [f] is decreased */ - if (f->bal == 0) - { f->bal = -1; - break; - } - if (f->bal > 0) - f->bal = 0; - else - { f = rotate_subtree(tree, f); - if (f->bal > 0) break; - } - flag = f->flag; f = f->up; - } - } - /* if the root has been reached, the height of the entire tree is - decreased */ - if (f == NULL) tree->height--; - /* returns the deleted node to the memory pool */ - dmp_free_atom(tree->pool, p, sizeof(AVLNODE)); - return; -} - -static AVLNODE *rotate_subtree(AVL *tree, AVLNODE *node) -{ /* restore balance of AVL subtree */ - AVLNODE *f, *p, *q, *r, *x, *y; - xassert(node != NULL); - p = node; - if (p->bal < 0) - { /* perform negative (left) rotation */ - f = p->up; q = p->left; r = q->right; - if (q->bal <= 0) - { /* perform single negative rotation */ - if (f == NULL) - tree->root = q; - else - if (p->flag == 0) f->left = q; else f->right = q; - p->rank -= q->rank; - q->up = f; q->flag = p->flag; q->bal++; q->right = p; - p->up = q; p->flag = 1; - p->bal = (short int)(-q->bal); p->left = r; - if (r != NULL) r->up = p, r->flag = 0; - node = q; - } - else - { /* perform double negative rotation */ - x = r->left; y = r->right; - if (f == NULL) - tree->root = r; - else - if (p->flag == 0) f->left = r; else f->right = r; - p->rank -= (q->rank + r->rank); - r->rank += q->rank; - p->bal = (short int)(r->bal >= 0 ? 0 : +1); - q->bal = (short int)(r->bal <= 0 ? 0 : -1); - r->up = f; r->flag = p->flag; r->bal = 0; - r->left = q; r->right = p; - p->up = r; p->flag = 1; p->left = y; - q->up = r; q->flag = 0; q->right = x; - if (x != NULL) x->up = q, x->flag = 1; - if (y != NULL) y->up = p, y->flag = 0; - node = r; - } - } - else - { /* perform positive (right) rotation */ - f = p->up; q = p->right; r = q->left; - if (q->bal >= 0) - { /* perform single positive rotation */ - if (f == NULL) - tree->root = q; - else - if (p->flag == 0) f->left = q; else f->right = q; - q->rank += p->rank; - q->up = f; q->flag = p->flag; q->bal--; q->left = p; - p->up = q; p->flag = 0; - p->bal = (short int)(-q->bal); p->right = r; - if (r != NULL) r->up = p, r->flag = 1; - node = q; - } - else - { /* perform double positive rotation */ - x = r->left; y = r->right; - if (f == NULL) - tree->root = r; - else - if (p->flag == 0) f->left = r; else f->right = r; - q->rank -= r->rank; - r->rank += p->rank; - p->bal = (short int)(r->bal <= 0 ? 0 : -1); - q->bal = (short int)(r->bal >= 0 ? 0 : +1); - r->up = f; r->flag = p->flag; r->bal = 0; - r->left = p; r->right = q; - p->up = r; p->flag = 0; p->right = x; - q->up = r; q->flag = 1; q->left = y; - if (x != NULL) x->up = p, x->flag = 1; - if (y != NULL) y->up = q, y->flag = 0; - node = r; - } - } - return node; -} - -void avl_delete_tree(AVL *tree) -{ /* delete AVL tree */ - dmp_delete_pool(tree->pool); - xfree(tree); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/avl.h b/code/3rd_glpk/misc/avl.h deleted file mode 100644 index b0aaef61..00000000 --- a/code/3rd_glpk/misc/avl.h +++ /dev/null @@ -1,73 +0,0 @@ -/* avl.h (binary search tree) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef AVL_H -#define AVL_H - -typedef struct AVL AVL; -typedef struct AVLNODE AVLNODE; - -#define avl_create_tree _glp_avl_create_tree -AVL *avl_create_tree(int (*fcmp)(void *info, const void *key1, - const void *key2), void *info); -/* create AVL tree */ - -#define avl_strcmp _glp_avl_strcmp -int avl_strcmp(void *info, const void *key1, const void *key2); -/* compare character string keys */ - -#define avl_insert_node _glp_avl_insert_node -AVLNODE *avl_insert_node(AVL *tree, const void *key); -/* insert new node into AVL tree */ - -#define avl_set_node_type _glp_avl_set_node_type -void avl_set_node_type(AVLNODE *node, int type); -/* assign the type field of specified node */ - -#define avl_set_node_link _glp_avl_set_node_link -void avl_set_node_link(AVLNODE *node, void *link); -/* assign the link field of specified node */ - -#define avl_find_node _glp_avl_find_node -AVLNODE *avl_find_node(AVL *tree, const void *key); -/* find node in AVL tree */ - -#define avl_get_node_type _glp_avl_get_node_type -int avl_get_node_type(AVLNODE *node); -/* retrieve the type field of specified node */ - -#define avl_get_node_link _glp_avl_get_node_link -void *avl_get_node_link(AVLNODE *node); -/* retrieve the link field of specified node */ - -#define avl_delete_node _glp_avl_delete_node -void avl_delete_node(AVL *tree, AVLNODE *node); -/* delete specified node from AVL tree */ - -#define avl_delete_tree _glp_avl_delete_tree -void avl_delete_tree(AVL *tree); -/* delete AVL tree */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/bignum.c b/code/3rd_glpk/misc/bignum.c deleted file mode 100644 index 540dd9fd..00000000 --- a/code/3rd_glpk/misc/bignum.c +++ /dev/null @@ -1,286 +0,0 @@ -/* bignum.c (bignum arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2006-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "bignum.h" - -/*********************************************************************** -* Two routines below are intended to multiply and divide unsigned -* integer numbers of arbitrary precision. -* -* The routines assume that an unsigned integer number is represented in -* the positional numeral system with the base 2^16 = 65536, i.e. each -* "digit" of the number is in the range [0, 65535] and represented as -* a 16-bit value of the unsigned short type. In other words, a number x -* has the following representation: -* -* n-1 -* x = sum d[j] * 65536^j, -* j=0 -* -* where n is the number of places (positions), and d[j] is j-th "digit" -* of x, 0 <= d[j] <= 65535. -***********************************************************************/ - -/*********************************************************************** -* NAME -* -* bigmul - multiply unsigned integer numbers of arbitrary precision -* -* SYNOPSIS -* -* #include "bignum.h" -* void bigmul(int n, int m, unsigned short x[], unsigned short y[]); -* -* DESCRIPTION -* -* The routine bigmul multiplies unsigned integer numbers of arbitrary -* precision. -* -* n is the number of digits of multiplicand, n >= 1; -* -* m is the number of digits of multiplier, m >= 1; -* -* x is an array containing digits of the multiplicand in elements -* x[m], x[m+1], ..., x[n+m-1]. Contents of x[0], x[1], ..., x[m-1] are -* ignored on entry. -* -* y is an array containing digits of the multiplier in elements y[0], -* y[1], ..., y[m-1]. -* -* On exit digits of the product are stored in elements x[0], x[1], ..., -* x[n+m-1]. The array y is not changed. */ - -void bigmul(int n, int m, unsigned short x[], unsigned short y[]) -{ int i, j; - unsigned int t; - xassert(n >= 1); - xassert(m >= 1); - for (j = 0; j < m; j++) x[j] = 0; - for (i = 0; i < n; i++) - { if (x[i+m]) - { t = 0; - for (j = 0; j < m; j++) - { t += (unsigned int)x[i+m] * (unsigned int)y[j] + - (unsigned int)x[i+j]; - x[i+j] = (unsigned short)t; - t >>= 16; - } - x[i+m] = (unsigned short)t; - } - } - return; -} - -/*********************************************************************** -* NAME -* -* bigdiv - divide unsigned integer numbers of arbitrary precision -* -* SYNOPSIS -* -* #include "bignum.h" -* void bigdiv(int n, int m, unsigned short x[], unsigned short y[]); -* -* DESCRIPTION -* -* The routine bigdiv divides one unsigned integer number of arbitrary -* precision by another with the algorithm described in [1]. -* -* n is the difference between the number of digits of dividend and the -* number of digits of divisor, n >= 0. -* -* m is the number of digits of divisor, m >= 1. -* -* x is an array containing digits of the dividend in elements x[0], -* x[1], ..., x[n+m-1]. -* -* y is an array containing digits of the divisor in elements y[0], -* y[1], ..., y[m-1]. The highest digit y[m-1] must be non-zero. -* -* On exit n+1 digits of the quotient are stored in elements x[m], -* x[m+1], ..., x[n+m], and m digits of the remainder are stored in -* elements x[0], x[1], ..., x[m-1]. The array y is changed but then -* restored. -* -* REFERENCES -* -* 1. D. Knuth. The Art of Computer Programming. Vol. 2: Seminumerical -* Algorithms. Stanford University, 1969. */ - -void bigdiv(int n, int m, unsigned short x[], unsigned short y[]) -{ int i, j; - unsigned int t; - unsigned short d, q, r; - xassert(n >= 0); - xassert(m >= 1); - xassert(y[m-1] != 0); - /* special case when divisor has the only digit */ - if (m == 1) - { d = 0; - for (i = n; i >= 0; i--) - { t = ((unsigned int)d << 16) + (unsigned int)x[i]; - x[i+1] = (unsigned short)(t / y[0]); - d = (unsigned short)(t % y[0]); - } - x[0] = d; - goto done; - } - /* multiply dividend and divisor by a normalizing coefficient in - * order to provide the condition y[m-1] >= base / 2 */ - d = (unsigned short)(0x10000 / ((unsigned int)y[m-1] + 1)); - if (d == 1) - x[n+m] = 0; - else - { t = 0; - for (i = 0; i < n+m; i++) - { t += (unsigned int)x[i] * (unsigned int)d; - x[i] = (unsigned short)t; - t >>= 16; - } - x[n+m] = (unsigned short)t; - t = 0; - for (j = 0; j < m; j++) - { t += (unsigned int)y[j] * (unsigned int)d; - y[j] = (unsigned short)t; - t >>= 16; - } - } - /* main loop */ - for (i = n; i >= 0; i--) - { /* estimate and correct the current digit of quotient */ - if (x[i+m] < y[m-1]) - { t = ((unsigned int)x[i+m] << 16) + (unsigned int)x[i+m-1]; - q = (unsigned short)(t / (unsigned int)y[m-1]); - r = (unsigned short)(t % (unsigned int)y[m-1]); - if (q == 0) goto putq; else goto test; - } - q = 0; - r = x[i+m-1]; -decr: q--; /* if q = 0 then q-- = 0xFFFF */ - t = (unsigned int)r + (unsigned int)y[m-1]; - r = (unsigned short)t; - if (t > 0xFFFF) goto msub; -test: t = (unsigned int)y[m-2] * (unsigned int)q; - if ((unsigned short)(t >> 16) > r) goto decr; - if ((unsigned short)(t >> 16) < r) goto msub; - if ((unsigned short)t > x[i+m-2]) goto decr; -msub: /* now subtract divisor multiplied by the current digit of - * quotient from the current dividend */ - if (q == 0) goto putq; - t = 0; - for (j = 0; j < m; j++) - { t += (unsigned int)y[j] * (unsigned int)q; - if (x[i+j] < (unsigned short)t) t += 0x10000; - x[i+j] -= (unsigned short)t; - t >>= 16; - } - if (x[i+m] >= (unsigned short)t) goto putq; - /* perform correcting addition, because the current digit of - * quotient is greater by one than its correct value */ - q--; - t = 0; - for (j = 0; j < m; j++) - { t += (unsigned int)x[i+j] + (unsigned int)y[j]; - x[i+j] = (unsigned short)t; - t >>= 16; - } -putq: /* store the current digit of quotient */ - x[i+m] = q; - } - /* divide divisor and remainder by the normalizing coefficient in - * order to restore their original values */ - if (d > 1) - { t = 0; - for (i = m-1; i >= 0; i--) - { t = (t << 16) + (unsigned int)x[i]; - x[i] = (unsigned short)(t / (unsigned int)d); - t %= (unsigned int)d; - } - t = 0; - for (j = m-1; j >= 0; j--) - { t = (t << 16) + (unsigned int)y[j]; - y[j] = (unsigned short)(t / (unsigned int)d); - t %= (unsigned int)d; - } - } -done: return; -} - -/**********************************************************************/ - -#ifdef GLP_TEST -#include -#include -#include -#include "rng.h" - -#define N_MAX 7 -/* maximal number of digits in multiplicand */ - -#define M_MAX 5 -/* maximal number of digits in multiplier */ - -#define N_TEST 1000000 -/* number of tests */ - -int main(void) -{ RNG *rand; - int d, j, n, m, test; - unsigned short x[N_MAX], y[M_MAX], z[N_MAX+M_MAX]; - rand = rng_create_rand(); - for (test = 1; test <= N_TEST; test++) - { /* x[0,...,n-1] := multiplicand */ - n = 1 + rng_unif_rand(rand, N_MAX-1); - assert(1 <= n && n <= N_MAX); - for (j = 0; j < n; j++) - { d = rng_unif_rand(rand, 65536); - assert(0 <= d && d <= 65535); - x[j] = (unsigned short)d; - } - /* y[0,...,m-1] := multiplier */ - m = 1 + rng_unif_rand(rand, M_MAX-1); - assert(1 <= m && m <= M_MAX); - for (j = 0; j < m; j++) - { d = rng_unif_rand(rand, 65536); - assert(0 <= d && d <= 65535); - y[j] = (unsigned short)d; - } - if (y[m-1] == 0) y[m-1] = 1; - /* z[0,...,n+m-1] := x * y */ - for (j = 0; j < n; j++) z[m+j] = x[j]; - bigmul(n, m, z, y); - /* z[0,...,m-1] := z mod y, z[m,...,n+m-1] := z div y */ - bigdiv(n, m, z, y); - /* z mod y must be 0 */ - for (j = 0; j < m; j++) assert(z[j] == 0); - /* z div y must be x */ - for (j = 0; j < n; j++) assert(z[m+j] == x[j]); - } - fprintf(stderr, "%d tests successfully passed\n", N_TEST); - rng_delete_rand(rand); - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/bignum.h b/code/3rd_glpk/misc/bignum.h deleted file mode 100644 index 8567519b..00000000 --- a/code/3rd_glpk/misc/bignum.h +++ /dev/null @@ -1,37 +0,0 @@ -/* bignum.h (bignum arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2006-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef BIGNUM_H -#define BIGNUM_H - -#define bigmul _glp_bigmul -void bigmul(int n, int m, unsigned short x[], unsigned short y[]); -/* multiply unsigned integer numbers of arbitrary precision */ - -#define bigdiv _glp_bigdiv -void bigdiv(int n, int m, unsigned short x[], unsigned short y[]); -/* divide unsigned integer numbers of arbitrary precision */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/dimacs.c b/code/3rd_glpk/misc/dimacs.c deleted file mode 100644 index 6aa630a5..00000000 --- a/code/3rd_glpk/misc/dimacs.c +++ /dev/null @@ -1,147 +0,0 @@ -/* dimacs.c (reading data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "dimacs.h" - -void dmx_error(DMX *csa, const char *fmt, ...) -{ /* print error message and terminate processing */ - va_list arg; - xprintf("%s:%d: error: ", csa->fname, csa->count); - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - xprintf("\n"); - longjmp(csa->jump, 1); - /* no return */ -} - -void dmx_warning(DMX *csa, const char *fmt, ...) -{ /* print warning message and continue processing */ - va_list arg; - xprintf("%s:%d: warning: ", csa->fname, csa->count); - va_start(arg, fmt); - xvprintf(fmt, arg); - va_end(arg); - xprintf("\n"); - return; -} - -void dmx_read_char(DMX *csa) -{ /* read character from input text file */ - int c; - if (csa->c == '\n') csa->count++; - c = glp_getc(csa->fp); - if (c < 0) - { if (glp_ioerr(csa->fp)) - dmx_error(csa, "read error - %s", get_err_msg()); - else if (csa->c == '\n') - dmx_error(csa, "unexpected end of file"); - else - { dmx_warning(csa, "missing final end of line"); - c = '\n'; - } - } - else if (c == '\n') - ; - else if (isspace(c)) - c = ' '; - else if (iscntrl(c)) - dmx_error(csa, "invalid control character 0x%02X", c); - csa->c = c; - return; -} - -void dmx_read_designator(DMX *csa) -{ /* read one-character line designator */ - xassert(csa->c == '\n'); - dmx_read_char(csa); - for (;;) - { /* skip preceding white-space characters */ - while (csa->c == ' ') - dmx_read_char(csa); - if (csa->c == '\n') - { /* ignore empty line */ - if (!csa->empty) - { dmx_warning(csa, "empty line ignored"); - csa->empty = 1; - } - dmx_read_char(csa); - } - else if (csa->c == 'c') - { /* skip comment line */ - while (csa->c != '\n') - dmx_read_char(csa); - dmx_read_char(csa); - } - else - { /* hmm... looks like a line designator */ - csa->field[0] = (char)csa->c, csa->field[1] = '\0'; - /* check that it is followed by a white-space character */ - dmx_read_char(csa); - if (!(csa->c == ' ' || csa->c == '\n')) - dmx_error(csa, "line designator missing or invalid"); - break; - } - } - return; -} - -void dmx_read_field(DMX *csa) -{ /* read data field */ - int len = 0; - /* skip preceding white-space characters */ - while (csa->c == ' ') - dmx_read_char(csa); - /* scan data field */ - if (csa->c == '\n') - dmx_error(csa, "unexpected end of line"); - while (!(csa->c == ' ' || csa->c == '\n')) - { if (len == sizeof(csa->field)-1) - dmx_error(csa, "data field '%.15s...' too long", - csa->field); - csa->field[len++] = (char)csa->c; - dmx_read_char(csa); - } - csa->field[len] = '\0'; - return; -} - -void dmx_end_of_line(DMX *csa) -{ /* skip white-space characters until end of line */ - while (csa->c == ' ') - dmx_read_char(csa); - if (csa->c != '\n') - dmx_error(csa, "too many data fields specified"); - return; -} - -void dmx_check_int(DMX *csa, double num) -{ /* print a warning if non-integer data are detected */ - if (!csa->nonint && num != floor(num)) - { dmx_warning(csa, "non-integer data detected"); - csa->nonint = 1; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/dimacs.h b/code/3rd_glpk/misc/dimacs.h deleted file mode 100644 index 42fb9996..00000000 --- a/code/3rd_glpk/misc/dimacs.h +++ /dev/null @@ -1,81 +0,0 @@ -/* dimacs.h (reading data in DIMACS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef DIMACS_H -#define DIMACS_H - -#include "env.h" - -typedef struct DMX DMX; - -struct DMX -{ /* DIMACS data reader */ - jmp_buf jump; - /* label for go to in case of error */ - const char *fname; - /* name of input text file */ - glp_file *fp; - /* stream assigned to input text file */ - int count; - /* line count */ - int c; - /* current character */ - char field[255+1]; - /* data field */ - int empty; - /* warning 'empty line ignored' was printed */ - int nonint; - /* warning 'non-integer data detected' was printed */ -}; - -#define dmx_error _glp_dmx_error -void dmx_error(DMX *csa, const char *fmt, ...); -/* print error message and terminate processing */ - -#define dmx_warning _glp_dmx_warning -void dmx_warning(DMX *csa, const char *fmt, ...); -/* print warning message and continue processing */ - -#define dmx_read_char _glp_dmx_read_char -void dmx_read_char(DMX *csa); -/* read character from input text file */ - -#define dmx_read_designator _glp_dmx_read_designator -void dmx_read_designator(DMX *csa); -/* read one-character line designator */ - -#define dmx_read_field _glp_dmx_read_field -void dmx_read_field(DMX *csa); -/* read data field */ - -#define dmx_end_of_line _glp_dmx_end_of_line -void dmx_end_of_line(DMX *csa); -/* skip white-space characters until end of line */ - -#define dmx_check_int _glp_dmx_check_int -void dmx_check_int(DMX *csa, double num); -/* print a warning if non-integer data are detected */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/dmp.c b/code/3rd_glpk/misc/dmp.c deleted file mode 100644 index a4882c86..00000000 --- a/code/3rd_glpk/misc/dmp.c +++ /dev/null @@ -1,243 +0,0 @@ -/* dmp.c (dynamic memory pool) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "dmp.h" - -struct DMP -{ /* dynamic memory pool */ - void *avail[32]; - /* avail[k], 0 <= k <= 31, is a pointer to first available (free) - * atom of (k+1)*8 bytes long; at the beginning of each free atom - * there is a pointer to another free atom of the same size */ - void *block; - /* pointer to most recently allocated memory block; at the - * beginning of each allocated memory block there is a pointer to - * previously allocated memory block */ - int used; - /* number of bytes used in most recently allocated memory block */ - size_t count; - /* number of atoms which are currently in use */ -}; - -#define DMP_BLK_SIZE 8000 -/* size of memory blocks, in bytes, allocated for memory pools */ - -struct prefix -{ /* atom prefix (for debugging only) */ - DMP *pool; - /* dynamic memory pool */ - int size; - /* original atom size, in bytes */ -}; - -#define prefix_size ((sizeof(struct prefix) + 7) & ~7) -/* size of atom prefix rounded up to multiple of 8 bytes */ - -int dmp_debug; -/* debug mode flag */ - -/*********************************************************************** -* NAME -* -* dmp_create_pool - create dynamic memory pool -* -* SYNOPSIS -* -* #include "dmp.h" -* DMP *dmp_create_pool(void); -* -* DESCRIPTION -* -* The routine dmp_create_pool creates a dynamic memory pool. -* -* RETURNS -* -* The routine returns a pointer to the memory pool created. */ - -DMP *dmp_create_pool(void) -{ DMP *pool; - int k; - xassert(sizeof(void *) <= 8); - if (dmp_debug) - xprintf("dmp_create_pool: warning: debug mode is on\n"); - pool = talloc(1, DMP); - for (k = 0; k <= 31; k++) - pool->avail[k] = NULL; - pool->block = NULL; - pool->used = DMP_BLK_SIZE; - pool->count = 0; - return pool; -} - -/*********************************************************************** -* NAME -* -* dmp_get_atom - get free atom from dynamic memory pool -* -* SYNOPSIS -* -* #include "dmp.h" -* void *dmp_get_atom(DMP *pool, int size); -* -* DESCRIPTION -* -* The routine dmp_get_atom obtains a free atom (memory space) from the -* specified memory pool. -* -* The parameter size is the atom size, in bytes, 1 <= size <= 256. -* -* Note that the free atom contains arbitrary data, not binary zeros. -* -* RETURNS -* -* The routine returns a pointer to the free atom obtained. */ - -void *dmp_get_atom(DMP *pool, int size) -{ void *atom; - int k, need; - xassert(1 <= size && size <= 256); - /* round up atom size to multiple of 8 bytes */ - need = (size + 7) & ~7; - /* determine number of corresponding list of free atoms */ - k = (need >> 3) - 1; - /* obtain free atom */ - if (pool->avail[k] == NULL) - { /* corresponding list of free atoms is empty */ - /* if debug mode is on, add atom prefix size */ - if (dmp_debug) - need += prefix_size; - if (pool->used + need > DMP_BLK_SIZE) - { /* allocate new memory block */ - void *block = talloc(DMP_BLK_SIZE, char); - *(void **)block = pool->block; - pool->block = block; - pool->used = 8; /* sufficient to store pointer */ - } - /* allocate new atom in current memory block */ - atom = (char *)pool->block + pool->used; - pool->used += need; - } - else - { /* obtain atom from corresponding list of free atoms */ - atom = pool->avail[k]; - pool->avail[k] = *(void **)atom; - } - /* if debug mode is on, fill atom prefix */ - if (dmp_debug) - { ((struct prefix *)atom)->pool = pool; - ((struct prefix *)atom)->size = size; - atom = (char *)atom + prefix_size; - } - /* increase number of allocated atoms */ - pool->count++; - return atom; -} - -/*********************************************************************** -* NAME -* -* dmp_free_atom - return atom to dynamic memory pool -* -* SYNOPSIS -* -* #include "dmp.h" -* void dmp_free_atom(DMP *pool, void *atom, int size); -* -* DESCRIPTION -* -* The routine dmp_free_atom returns the specified atom (memory space) -* to the specified memory pool, making the atom free. -* -* The parameter size is the atom size, in bytes, 1 <= size <= 256. -* -* Note that the atom can be returned only to the pool, from which it -* was obtained, and its size must be exactly the same as on obtaining -* it from the pool. */ - -void dmp_free_atom(DMP *pool, void *atom, int size) -{ int k; - xassert(1 <= size && size <= 256); - /* determine number of corresponding list of free atoms */ - k = ((size + 7) >> 3) - 1; - /* if debug mode is on, check atom prefix */ - if (dmp_debug) - { atom = (char *)atom - prefix_size; - xassert(((struct prefix *)atom)->pool == pool); - xassert(((struct prefix *)atom)->size == size); - } - /* return atom to corresponding list of free atoms */ - *(void **)atom = pool->avail[k]; - pool->avail[k] = atom; - /* decrease number of allocated atoms */ - xassert(pool->count > 0); - pool->count--; - return; -} - -/*********************************************************************** -* NAME -* -* dmp_in_use - determine how many atoms are still in use -* -* SYNOPSIS -* -* #include "dmp.h" -* size_t dmp_in_use(DMP *pool); -* -* RETURNS -* -* The routine returns the number of atoms of the specified memory pool -* which are still in use. */ - -size_t dmp_in_use(DMP *pool) -{ return - pool->count; -} - -/*********************************************************************** -* NAME -* -* dmp_delete_pool - delete dynamic memory pool -* -* SYNOPSIS -* -* #include "dmp.h" -* void dmp_delete_pool(DMP *pool); -* -* DESCRIPTION -* -* The routine dmp_delete_pool deletes the specified dynamic memory -* pool freeing all the memory allocated to this object. */ - -void dmp_delete_pool(DMP *pool) -{ while (pool->block != NULL) - { void *block = pool->block; - pool->block = *(void **)block; - tfree(block); - } - tfree(pool); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/dmp.h b/code/3rd_glpk/misc/dmp.h deleted file mode 100644 index 85fe7176..00000000 --- a/code/3rd_glpk/misc/dmp.h +++ /dev/null @@ -1,63 +0,0 @@ -/* dmp.h (dynamic memory pool) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef DMP_H -#define DMP_H - -#include "stdc.h" - -typedef struct DMP DMP; - -#define dmp_debug _glp_dmp_debug -extern int dmp_debug; -/* debug mode flag */ - -#define dmp_create_pool _glp_dmp_create_pool -DMP *dmp_create_pool(void); -/* create dynamic memory pool */ - -#define dmp_talloc(pool, type) \ - ((type *)dmp_get_atom(pool, sizeof(type))) - -#define dmp_get_atom _glp_dmp_get_atom -void *dmp_get_atom(DMP *pool, int size); -/* get free atom from dynamic memory pool */ - -#define dmp_tfree(pool, atom) \ - dmp_free_atom(pool, atom, sizeof(*(atom))) - -#define dmp_free_atom _glp_dmp_free_atom -void dmp_free_atom(DMP *pool, void *atom, int size); -/* return atom to dynamic memory pool */ - -#define dmp_in_use _glp_dmp_in_use -size_t dmp_in_use(DMP *pool); -/* determine how many atoms are still in use */ - -#define dmp_delete_pool _glp_dmp_delete_pool -void dmp_delete_pool(DMP *pool); -/* delete dynamic memory pool */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/ffalg.c b/code/3rd_glpk/misc/ffalg.c deleted file mode 100644 index 4ea2913d..00000000 --- a/code/3rd_glpk/misc/ffalg.c +++ /dev/null @@ -1,221 +0,0 @@ -/* ffalg.c (Ford-Fulkerson algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ffalg.h" - -/*********************************************************************** -* NAME -* -* ffalg - Ford-Fulkerson algorithm -* -* SYNOPSIS -* -* #include "ffalg.h" -* void ffalg(int nv, int na, const int tail[], const int head[], -* int s, int t, const int cap[], int x[], char cut[]); -* -* DESCRIPTION -* -* The routine ffalg implements the Ford-Fulkerson algorithm to find a -* maximal flow in the specified flow network. -* -* INPUT PARAMETERS -* -* nv is the number of nodes, nv >= 2. -* -* na is the number of arcs, na >= 0. -* -* tail[a], a = 1,...,na, is the index of tail node of arc a. -* -* head[a], a = 1,...,na, is the index of head node of arc a. -* -* s is the source node index, 1 <= s <= nv. -* -* t is the sink node index, 1 <= t <= nv, t != s. -* -* cap[a], a = 1,...,na, is the capacity of arc a, cap[a] >= 0. -* -* NOTE: Multiple arcs are allowed, but self-loops are not allowed. -* -* OUTPUT PARAMETERS -* -* x[a], a = 1,...,na, is optimal value of the flow through arc a. -* -* cut[i], i = 1,...,nv, is 1 if node i is labelled, and 0 otherwise. -* The set of arcs, whose one endpoint is labelled and other is not, -* defines the minimal cut corresponding to the maximal flow found. -* If the parameter cut is NULL, the cut information are not stored. -* -* REFERENCES -* -* L.R.Ford, Jr., and D.R.Fulkerson, "Flows in Networks," The RAND -* Corp., Report R-375-PR (August 1962), Chap. I "Static Maximal Flow," -* pp.30-33. */ - -void ffalg(int nv, int na, const int tail[], const int head[], - int s, int t, const int cap[], int x[], char cut[]) -{ int a, delta, i, j, k, pos1, pos2, temp, - *ptr, *arc, *link, *list; - /* sanity checks */ - xassert(nv >= 2); - xassert(na >= 0); - xassert(1 <= s && s <= nv); - xassert(1 <= t && t <= nv); - xassert(s != t); - for (a = 1; a <= na; a++) - { i = tail[a], j = head[a]; - xassert(1 <= i && i <= nv); - xassert(1 <= j && j <= nv); - xassert(i != j); - xassert(cap[a] >= 0); - } - /* allocate working arrays */ - ptr = xcalloc(1+nv+1, sizeof(int)); - arc = xcalloc(1+na+na, sizeof(int)); - link = xcalloc(1+nv, sizeof(int)); - list = xcalloc(1+nv, sizeof(int)); - /* ptr[i] := (degree of node i) */ - for (i = 1; i <= nv; i++) - ptr[i] = 0; - for (a = 1; a <= na; a++) - { ptr[tail[a]]++; - ptr[head[a]]++; - } - /* initialize arc pointers */ - ptr[1]++; - for (i = 1; i < nv; i++) - ptr[i+1] += ptr[i]; - ptr[nv+1] = ptr[nv]; - /* build arc lists */ - for (a = 1; a <= na; a++) - { arc[--ptr[tail[a]]] = a; - arc[--ptr[head[a]]] = a; - } - xassert(ptr[1] == 1); - xassert(ptr[nv+1] == na+na+1); - /* now the indices of arcs incident to node i are stored in - * locations arc[ptr[i]], arc[ptr[i]+1], ..., arc[ptr[i+1]-1] */ - /* initialize arc flows */ - for (a = 1; a <= na; a++) - x[a] = 0; -loop: /* main loop starts here */ - /* build augmenting tree rooted at s */ - /* link[i] = 0 means that node i is not labelled yet; - * link[i] = a means that arc a immediately precedes node i */ - /* initially node s is labelled as the root */ - for (i = 1; i <= nv; i++) - link[i] = 0; - link[s] = -1, list[1] = s, pos1 = pos2 = 1; - /* breadth first search */ - while (pos1 <= pos2) - { /* dequeue node i */ - i = list[pos1++]; - /* consider all arcs incident to node i */ - for (k = ptr[i]; k < ptr[i+1]; k++) - { a = arc[k]; - if (tail[a] == i) - { /* a = i->j is a forward arc from s to t */ - j = head[a]; - /* if node j has been labelled, skip the arc */ - if (link[j] != 0) continue; - /* if the arc does not allow increasing the flow through - * it, skip the arc */ - if (x[a] == cap[a]) continue; - } - else if (head[a] == i) - { /* a = i<-j is a backward arc from s to t */ - j = tail[a]; - /* if node j has been labelled, skip the arc */ - if (link[j] != 0) continue; - /* if the arc does not allow decreasing the flow through - * it, skip the arc */ - if (x[a] == 0) continue; - } - else - xassert(a != a); - /* label node j and enqueue it */ - link[j] = a, list[++pos2] = j; - /* check for breakthrough */ - if (j == t) goto brkt; - } - } - /* NONBREAKTHROUGH */ - /* no augmenting path exists; current flow is maximal */ - /* store minimal cut information, if necessary */ - if (cut != NULL) - { for (i = 1; i <= nv; i++) - cut[i] = (char)(link[i] != 0); - } - goto done; -brkt: /* BREAKTHROUGH */ - /* walk through arcs of the augmenting path (s, ..., t) found in - * the reverse order and determine maximal change of the flow */ - delta = 0; - for (j = t; j != s; j = i) - { /* arc a immediately precedes node j in the path */ - a = link[j]; - if (head[a] == j) - { /* a = i->j is a forward arc of the cycle */ - i = tail[a]; - /* x[a] may be increased until its upper bound */ - temp = cap[a] - x[a]; - } - else if (tail[a] == j) - { /* a = i<-j is a backward arc of the cycle */ - i = head[a]; - /* x[a] may be decreased until its lower bound */ - temp = x[a]; - } - else - xassert(a != a); - if (delta == 0 || delta > temp) delta = temp; - } - xassert(delta > 0); - /* increase the flow along the path */ - for (j = t; j != s; j = i) - { /* arc a immediately precedes node j in the path */ - a = link[j]; - if (head[a] == j) - { /* a = i->j is a forward arc of the cycle */ - i = tail[a]; - x[a] += delta; - } - else if (tail[a] == j) - { /* a = i<-j is a backward arc of the cycle */ - i = head[a]; - x[a] -= delta; - } - else - xassert(a != a); - } - goto loop; -done: /* free working arrays */ - xfree(ptr); - xfree(arc); - xfree(link); - xfree(list); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/ffalg.h b/code/3rd_glpk/misc/ffalg.h deleted file mode 100644 index 7016f8fa..00000000 --- a/code/3rd_glpk/misc/ffalg.h +++ /dev/null @@ -1,34 +0,0 @@ -/* ffalg.h (Ford-Fulkerson algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef FFALG_H -#define FFALG_H - -#define ffalg _glp_ffalg -void ffalg(int nv, int na, const int tail[], const int head[], - int s, int t, const int cap[], int x[], char cut[]); -/* Ford-Fulkerson algorithm */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/fp2rat.c b/code/3rd_glpk/misc/fp2rat.c deleted file mode 100644 index 4699bbd1..00000000 --- a/code/3rd_glpk/misc/fp2rat.c +++ /dev/null @@ -1,164 +0,0 @@ -/* fp2rat.c (convert floating-point number to rational number) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "misc.h" - -/*********************************************************************** -* NAME -* -* fp2rat - convert floating-point number to rational number -* -* SYNOPSIS -* -* #include "misc.h" -* int fp2rat(double x, double eps, double *p, double *q); -* -* DESCRIPTION -* -* Given a floating-point number 0 <= x < 1 the routine fp2rat finds -* its "best" rational approximation p / q, where p >= 0 and q > 0 are -* integer numbers, such that |x - p / q| <= eps. -* -* RETURNS -* -* The routine fp2rat returns the number of iterations used to achieve -* the specified precision eps. -* -* EXAMPLES -* -* For x = sqrt(2) - 1 = 0.414213562373095 and eps = 1e-6 the routine -* gives p = 408 and q = 985, where 408 / 985 = 0.414213197969543. -* -* BACKGROUND -* -* It is well known that every positive real number x can be expressed -* as the following continued fraction: -* -* x = b[0] + a[1] -* ------------------------ -* b[1] + a[2] -* ----------------- -* b[2] + a[3] -* ---------- -* b[3] + ... -* -* where: -* -* a[k] = 1, k = 0, 1, 2, ... -* -* b[k] = floor(x[k]), k = 0, 1, 2, ... -* -* x[0] = x, -* -* x[k] = 1 / frac(x[k-1]), k = 1, 2, 3, ... -* -* To find the "best" rational approximation of x the routine computes -* partial fractions f[k] by dropping after k terms as follows: -* -* f[k] = A[k] / B[k], -* -* where: -* -* A[-1] = 1, A[0] = b[0], B[-1] = 0, B[0] = 1, -* -* A[k] = b[k] * A[k-1] + a[k] * A[k-2], -* -* B[k] = b[k] * B[k-1] + a[k] * B[k-2]. -* -* Once the condition -* -* |x - f[k]| <= eps -* -* has been satisfied, the routine reports p = A[k] and q = B[k] as the -* final answer. -* -* In the table below here is some statistics obtained for one million -* random numbers uniformly distributed in the range [0, 1). -* -* eps max p mean p max q mean q max k mean k -* ------------------------------------------------------------- -* 1e-1 8 1.6 9 3.2 3 1.4 -* 1e-2 98 6.2 99 12.4 5 2.4 -* 1e-3 997 20.7 998 41.5 8 3.4 -* 1e-4 9959 66.6 9960 133.5 10 4.4 -* 1e-5 97403 211.7 97404 424.2 13 5.3 -* 1e-6 479669 669.9 479670 1342.9 15 6.3 -* 1e-7 1579030 2127.3 3962146 4257.8 16 7.3 -* 1e-8 26188823 6749.4 26188824 13503.4 19 8.2 -* -* REFERENCES -* -* W. B. Jones and W. J. Thron, "Continued Fractions: Analytic Theory -* and Applications," Encyclopedia on Mathematics and Its Applications, -* Addison-Wesley, 1980. */ - -int fp2rat(double x, double eps, double *p, double *q) -{ int k; - double xk, Akm1, Ak, Bkm1, Bk, ak, bk, fk, temp; - xassert(0.0 <= x && x < 1.0); - for (k = 0; ; k++) - { xassert(k <= 100); - if (k == 0) - { /* x[0] = x */ - xk = x; - /* A[-1] = 1 */ - Akm1 = 1.0; - /* A[0] = b[0] = floor(x[0]) = 0 */ - Ak = 0.0; - /* B[-1] = 0 */ - Bkm1 = 0.0; - /* B[0] = 1 */ - Bk = 1.0; - } - else - { /* x[k] = 1 / frac(x[k-1]) */ - temp = xk - floor(xk); - xassert(temp != 0.0); - xk = 1.0 / temp; - /* a[k] = 1 */ - ak = 1.0; - /* b[k] = floor(x[k]) */ - bk = floor(xk); - /* A[k] = b[k] * A[k-1] + a[k] * A[k-2] */ - temp = bk * Ak + ak * Akm1; - Akm1 = Ak, Ak = temp; - /* B[k] = b[k] * B[k-1] + a[k] * B[k-2] */ - temp = bk * Bk + ak * Bkm1; - Bkm1 = Bk, Bk = temp; - } - /* f[k] = A[k] / B[k] */ - fk = Ak / Bk; -#if 0 - print("%.*g / %.*g = %.*g", - DBL_DIG, Ak, DBL_DIG, Bk, DBL_DIG, fk); -#endif - if (fabs(x - fk) <= eps) - break; - } - *p = Ak; - *q = Bk; - return k; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/fvs.c b/code/3rd_glpk/misc/fvs.c deleted file mode 100644 index 916a1bf9..00000000 --- a/code/3rd_glpk/misc/fvs.c +++ /dev/null @@ -1,137 +0,0 @@ -/* fvs.c (sparse vector in FVS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "fvs.h" - -void fvs_alloc_vec(FVS *x, int n) -{ /* allocate sparse vector */ - int j; - xassert(n >= 0); - x->n = n; - x->nnz = 0; - x->ind = talloc(1+n, int); - x->vec = talloc(1+n, double); - for (j = 1; j <= n; j++) - x->vec[j] = 0.0; - return; -} - -void fvs_check_vec(const FVS *x) -{ /* check sparse vector */ - /* NOTE: for testing/debugging only */ - int n = x->n; - int nnz = x->nnz; - int *ind = x->ind; - double *vec = x->vec; - char *map; - int j, k; - xassert(n >= 0); - xassert(0 <= nnz && nnz <= n); - map = talloc(1+n, char); - for (j = 1; j <= n; j++) - map[j] = (vec[j] != 0.0); - for (k = 1; k <= nnz; k++) - { j = ind[k]; - xassert(1 <= j && j <= n); - xassert(map[j]); - map[j] = 0; - } - for (j = 1; j <= n; j++) - xassert(!map[j]); - tfree(map); - return; -} - -void fvs_gather_vec(FVS *x, double eps) -{ /* gather sparse vector */ - int n = x->n; - int *ind = x->ind; - double *vec = x->vec; - int j, nnz = 0; - for (j = n; j >= 1; j--) - { if (-eps < vec[j] && vec[j] < +eps) - vec[j] = 0.0; - else - ind[++nnz] = j; - } - x->nnz = nnz; - return; -} - -void fvs_clear_vec(FVS *x) -{ /* clear sparse vector */ - int *ind = x->ind; - double *vec = x->vec; - int k; - for (k = x->nnz; k >= 1; k--) - vec[ind[k]] = 0.0; - x->nnz = 0; - return; -} - -void fvs_copy_vec(FVS *x, const FVS *y) -{ /* copy sparse vector */ - int *x_ind = x->ind; - double *x_vec = x->vec; - int *y_ind = y->ind; - double *y_vec = y->vec; - int j, k; - xassert(x != y); - xassert(x->n == y->n); - fvs_clear_vec(x); - for (k = x->nnz = y->nnz; k >= 1; k--) - { j = x_ind[k] = y_ind[k]; - x_vec[j] = y_vec[j]; - } - return; -} - -void fvs_adjust_vec(FVS *x, double eps) -{ /* replace tiny vector elements by exact zeros */ - int nnz = x->nnz; - int *ind = x->ind; - double *vec = x->vec; - int j, k, cnt = 0; - for (k = 1; k <= nnz; k++) - { j = ind[k]; - if (-eps < vec[j] && vec[j] < +eps) - vec[j] = 0.0; - else - ind[++cnt] = j; - } - x->nnz = cnt; - return; -} - -void fvs_free_vec(FVS *x) -{ /* deallocate sparse vector */ - tfree(x->ind); - tfree(x->vec); - x->n = x->nnz = -1; - x->ind = NULL; - x->vec = NULL; - return; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/fvs.h b/code/3rd_glpk/misc/fvs.h deleted file mode 100644 index abfed8cc..00000000 --- a/code/3rd_glpk/misc/fvs.h +++ /dev/null @@ -1,76 +0,0 @@ -/* fvs.h (sparse vector in FVS format) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef FVS_H -#define FVS_H - -typedef struct FVS FVS; - -struct FVS -{ /* sparse vector in FVS (Full Vector Storage) format */ - int n; - /* vector dimension (total number of elements) */ - int nnz; - /* number of non-zero elements, 0 <= nnz <= n */ - int *ind; /* int ind[1+n]; */ - /* ind[0] is not used; - * ind[k] = j, 1 <= k <= nnz, means that vec[j] != 0 - * non-zero indices in the array ind are stored in arbitrary - * order; if vec[j] = 0, its index j SHOULD NOT be presented in - * the array ind */ - double *vec; /* double vec[1+n]; */ - /* vec[0] is not used; - * vec[j], 1 <= j <= n, is a numeric value of j-th element */ -}; - -#define fvs_alloc_vec _glp_fvs_alloc_vec -void fvs_alloc_vec(FVS *x, int n); -/* allocate sparse vector */ - -#define fvs_check_vec _glp_fvs_check_vec -void fvs_check_vec(const FVS *x); -/* check sparse vector */ - -#define fvs_gather_vec _glp_fvs_gather_vec -void fvs_gather_vec(FVS *x, double eps); -/* gather sparse vector */ - -#define fvs_clear_vec _glp_fvs_clear_vec -void fvs_clear_vec(FVS *x); -/* clear sparse vector */ - -#define fvs_copy_vec _glp_fvs_copy_vec -void fvs_copy_vec(FVS *x, const FVS *y); -/* copy sparse vector */ - -#define fvs_adjust_vec _glp_fvs_adjust_vec -void fvs_adjust_vec(FVS *x, double eps); -/* replace tiny vector elements by exact zeros */ - -#define fvs_free_vec _glp_fvs_free_vec -void fvs_free_vec(FVS *x); -/* deallocate sparse vector */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/gcd.c b/code/3rd_glpk/misc/gcd.c deleted file mode 100644 index 95c48cc0..00000000 --- a/code/3rd_glpk/misc/gcd.c +++ /dev/null @@ -1,102 +0,0 @@ -/* gcd.c (greatest common divisor) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "misc.h" - -/*********************************************************************** -* NAME -* -* gcd - find greatest common divisor of two integers -* -* SYNOPSIS -* -* #include "misc.h" -* int gcd(int x, int y); -* -* RETURNS -* -* The routine gcd returns gcd(x, y), the greatest common divisor of -* the two positive integers given. -* -* ALGORITHM -* -* The routine gcd is based on Euclid's algorithm. -* -* REFERENCES -* -* Don Knuth, The Art of Computer Programming, Vol.2: Seminumerical -* Algorithms, 3rd Edition, Addison-Wesley, 1997. Section 4.5.2: The -* Greatest Common Divisor, pp. 333-56. */ - -int gcd(int x, int y) -{ int r; - xassert(x > 0 && y > 0); - while (y > 0) - r = x % y, x = y, y = r; - return x; -} - -/*********************************************************************** -* NAME -* -* gcdn - find greatest common divisor of n integers -* -* SYNOPSIS -* -* #include "misc.h" -* int gcdn(int n, int x[]); -* -* RETURNS -* -* The routine gcdn returns gcd(x[1], x[2], ..., x[n]), the greatest -* common divisor of n positive integers given, n > 0. -* -* BACKGROUND -* -* The routine gcdn is based on the following identity: -* -* gcd(x, y, z) = gcd(gcd(x, y), z). -* -* REFERENCES -* -* Don Knuth, The Art of Computer Programming, Vol.2: Seminumerical -* Algorithms, 3rd Edition, Addison-Wesley, 1997. Section 4.5.2: The -* Greatest Common Divisor, pp. 333-56. */ - -int gcdn(int n, int x[]) -{ int d, j; - xassert(n > 0); - for (j = 1; j <= n; j++) - { xassert(x[j] > 0); - if (j == 1) - d = x[1]; - else - d = gcd(d, x[j]); - if (d == 1) - break; - } - return d; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/jd.c b/code/3rd_glpk/misc/jd.c deleted file mode 100644 index c9d63171..00000000 --- a/code/3rd_glpk/misc/jd.c +++ /dev/null @@ -1,152 +0,0 @@ -/* jd.c (conversions between calendar date and Julian day number) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include -#include "jd.h" - -/*********************************************************************** -* NAME -* -* jday - convert calendar date to Julian day number -* -* SYNOPSIS -* -* #include "jd.h" -* int jday(int d, int m, int y); -* -* DESCRIPTION -* -* The routine jday converts a calendar date, Gregorian calendar, to -* corresponding Julian day number j. -* -* From the given day d, month m, and year y, the Julian day number j -* is computed without using tables. -* -* The routine is valid for 1 <= y <= 4000. -* -* RETURNS -* -* The routine jday returns the Julian day number, or negative value if -* the specified date is incorrect. -* -* REFERENCES -* -* R. G. Tantzen, Algorithm 199: conversions between calendar date and -* Julian day number, Communications of the ACM, vol. 6, no. 8, p. 444, -* Aug. 1963. */ - -int jday(int d, int m, int y) -{ int c, ya, j, dd; - if (!(1 <= d && d <= 31 && - 1 <= m && m <= 12 && - 1 <= y && y <= 4000)) - return -1; - if (m >= 3) - m -= 3; - else - m += 9, y--; - c = y / 100; - ya = y - 100 * c; - j = (146097 * c) / 4 + (1461 * ya) / 4 + (153 * m + 2) / 5 + d + - 1721119; - jdate(j, &dd, NULL, NULL); - if (d != dd) - return -1; - return j; -} - -/*********************************************************************** -* NAME -* -* jdate - convert Julian day number to calendar date -* -* SYNOPSIS -* -* #include "jd.h" -* int jdate(int j, int *d, int *m, int *y); -* -* DESCRIPTION -* -* The routine jdate converts a Julian day number j to corresponding -* calendar date, Gregorian calendar. -* -* The day d, month m, and year y are computed without using tables and -* stored in corresponding locations. -* -* The routine is valid for 1721426 <= j <= 3182395. -* -* RETURNS -* -* If the conversion is successful, the routine returns zero, otherwise -* non-zero. -* -* REFERENCES -* -* R. G. Tantzen, Algorithm 199: conversions between calendar date and -* Julian day number, Communications of the ACM, vol. 6, no. 8, p. 444, -* Aug. 1963. */ - -int jdate(int j, int *d_, int *m_, int *y_) -{ int d, m, y; - if (!(1721426 <= j && j <= 3182395)) - return 1; - j -= 1721119; - y = (4 * j - 1) / 146097; - j = (4 * j - 1) % 146097; - d = j / 4; - j = (4 * d + 3) / 1461; - d = (4 * d + 3) % 1461; - d = (d + 4) / 4; - m = (5 * d - 3) / 153; - d = (5 * d - 3) % 153; - d = (d + 5) / 5; - y = 100 * y + j; - if (m <= 9) - m += 3; - else m -= 9, - y++; - if (d_ != NULL) *d_ = d; - if (m_ != NULL) *m_ = m; - if (y_ != NULL) *y_ = y; - return 0; -} - -#ifdef GLP_TEST -#include -#include -#include - -int main(void) -{ int jbeg, jend, j, d, m, y; - jbeg = jday(1, 1, 1); - jend = jday(31, 12, 4000); - for (j = jbeg; j <= jend; j++) - { assert(jdate(j, &d, &m, &y) == 0); - assert(jday(d, m, y) == j); - } - printf("Routines jday and jdate work correctly.\n"); - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/jd.h b/code/3rd_glpk/misc/jd.h deleted file mode 100644 index 009d2daa..00000000 --- a/code/3rd_glpk/misc/jd.h +++ /dev/null @@ -1,32 +0,0 @@ -/* jd.h (conversions between calendar date and Julian day number) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#define jday _glp_jday -int jday(int d, int m, int y); -/* convert calendar date to Julian day number */ - -#define jdate _glp_jdate -int jdate(int j, int *d, int *m, int *y); -/* convert Julian day number to calendar date */ - -/* eof */ diff --git a/code/3rd_glpk/misc/keller.c b/code/3rd_glpk/misc/keller.c deleted file mode 100644 index d64d3c1e..00000000 --- a/code/3rd_glpk/misc/keller.c +++ /dev/null @@ -1,235 +0,0 @@ -/* keller.c (cover edges by cliques, Kellerman's heuristic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "glpk.h" -#include "env.h" -#include "keller.h" - -/*********************************************************************** -* NAME -* -* kellerman - cover edges by cliques with Kellerman's heuristic -* -* SYNOPSIS -* -* #include "keller.h" -* int kellerman(int n, int (*func)(void *info, int i, int ind[]), -* void *info, glp_graph *H); -* -* DESCRIPTION -* -* The routine kellerman implements Kellerman's heuristic algorithm -* to find a minimal set of cliques which cover all edges of specified -* graph G = (V, E). -* -* The parameter n specifies the number of vertices |V|, n >= 0. -* -* Formal routine func specifies the set of edges E in the following -* way. Running the routine kellerman calls the routine func and passes -* to it parameter i, which is the number of some vertex, 1 <= i <= n. -* In response the routine func should store numbers of all vertices -* adjacent to vertex i to locations ind[1], ind[2], ..., ind[len] and -* return the value of len, which is the number of adjacent vertices, -* 0 <= len <= n. Self-loops are allowed, but ignored. Multiple edges -* are not allowed. -* -* The parameter info is a transit pointer (magic cookie) passed to the -* formal routine func as its first parameter. -* -* The result provided by the routine kellerman is the bipartite graph -* H = (V union C, F), which defines the covering found. (The program -* object of type glp_graph specified by the parameter H should be -* previously created with the routine glp_create_graph. On entry the -* routine kellerman erases the content of this object with the routine -* glp_erase_graph.) Vertices of first part V correspond to vertices of -* the graph G and have the same ordinal numbers 1, 2, ..., n. Vertices -* of second part C correspond to cliques and have ordinal numbers -* n+1, n+2, ..., n+k, where k is the total number of cliques in the -* edge covering found. Every edge f in F in the program object H is -* represented as arc f = (i->j), where i in V and j in C, which means -* that vertex i of the graph G is in clique C[j], 1 <= j <= k. (Thus, -* if two vertices of the graph G are in the same clique, these vertices -* are adjacent in G, and corresponding edge is covered by that clique.) -* -* RETURNS -* -* The routine Kellerman returns k, the total number of cliques in the -* edge covering found. -* -* REFERENCE -* -* For more details see: glpk/doc/notes/keller.pdf (in Russian). */ - -struct set -{ /* set of vertices */ - int size; - /* size (cardinality) of the set, 0 <= card <= n */ - int *list; /* int list[1+n]; */ - /* the set contains vertices list[1,...,size] */ - int *pos; /* int pos[1+n]; */ - /* pos[i] > 0 means that vertex i is in the set and - * list[pos[i]] = i; pos[i] = 0 means that vertex i is not in - * the set */ -}; - -int kellerman(int n, int (*func)(void *info, int i, int ind[]), - void *info, void /* glp_graph */ *H_) -{ glp_graph *H = H_; - struct set W_, *W = &W_, V_, *V = &V_; - glp_arc *a; - int i, j, k, m, t, len, card, best; - xassert(n >= 0); - /* H := (V, 0; 0), where V is the set of vertices of graph G */ - glp_erase_graph(H, H->v_size, H->a_size); - glp_add_vertices(H, n); - /* W := 0 */ - W->size = 0; - W->list = xcalloc(1+n, sizeof(int)); - W->pos = xcalloc(1+n, sizeof(int)); - memset(&W->pos[1], 0, sizeof(int) * n); - /* V := 0 */ - V->size = 0; - V->list = xcalloc(1+n, sizeof(int)); - V->pos = xcalloc(1+n, sizeof(int)); - memset(&V->pos[1], 0, sizeof(int) * n); - /* main loop */ - for (i = 1; i <= n; i++) - { /* W must be empty */ - xassert(W->size == 0); - /* W := { j : i > j and (i,j) in E } */ - len = func(info, i, W->list); - xassert(0 <= len && len <= n); - for (t = 1; t <= len; t++) - { j = W->list[t]; - xassert(1 <= j && j <= n); - if (j >= i) continue; - xassert(W->pos[j] == 0); - W->list[++W->size] = j, W->pos[j] = W->size; - } - /* on i-th iteration we need to cover edges (i,j) for all - * j in W */ - /* if W is empty, it is a special case */ - if (W->size == 0) - { /* set k := k + 1 and create new clique C[k] = { i } */ - k = glp_add_vertices(H, 1) - n; - glp_add_arc(H, i, n + k); - continue; - } - /* try to include vertex i into existing cliques */ - /* V must be empty */ - xassert(V->size == 0); - /* k is the number of cliques found so far */ - k = H->nv - n; - for (m = 1; m <= k; m++) - { /* do while V != W; since here V is within W, we can use - * equivalent condition: do while |V| < |W| */ - if (V->size == W->size) break; - /* check if C[m] is within W */ - for (a = H->v[n + m]->in; a != NULL; a = a->h_next) - { j = a->tail->i; - if (W->pos[j] == 0) break; - } - if (a != NULL) continue; - /* C[m] is within W, expand clique C[m] with vertex i */ - /* C[m] := C[m] union {i} */ - glp_add_arc(H, i, n + m); - /* V is a set of vertices whose incident edges are already - * covered by existing cliques */ - /* V := V union C[m] */ - for (a = H->v[n + m]->in; a != NULL; a = a->h_next) - { j = a->tail->i; - if (V->pos[j] == 0) - V->list[++V->size] = j, V->pos[j] = V->size; - } - } - /* remove from set W the vertices whose incident edges are - * already covered by existing cliques */ - /* W := W \ V, V := 0 */ - for (t = 1; t <= V->size; t++) - { j = V->list[t], V->pos[j] = 0; - if (W->pos[j] != 0) - { /* remove vertex j from W */ - if (W->pos[j] != W->size) - { int jj = W->list[W->size]; - W->list[W->pos[j]] = jj; - W->pos[jj] = W->pos[j]; - } - W->size--, W->pos[j] = 0; - } - } - V->size = 0; - /* now set W contains only vertices whose incident edges are - * still not covered by existing cliques; create new cliques - * to cover remaining edges until set W becomes empty */ - while (W->size > 0) - { /* find clique C[m], 1 <= m <= k, which shares maximal - * number of vertices with W; to break ties choose clique - * having smallest number m */ - m = 0, best = -1; - k = H->nv - n; - for (t = 1; t <= k; t++) - { /* compute cardinality of intersection of W and C[t] */ - card = 0; - for (a = H->v[n + t]->in; a != NULL; a = a->h_next) - { j = a->tail->i; - if (W->pos[j] != 0) card++; - } - if (best < card) - m = t, best = card; - } - xassert(m > 0); - /* set k := k + 1 and create new clique: - * C[k] := (W intersect C[m]) union { i }, which covers all - * edges incident to vertices from (W intersect C[m]) */ - k = glp_add_vertices(H, 1) - n; - for (a = H->v[n + m]->in; a != NULL; a = a->h_next) - { j = a->tail->i; - if (W->pos[j] != 0) - { /* vertex j is in both W and C[m]; include it in new - * clique C[k] */ - glp_add_arc(H, j, n + k); - /* remove vertex j from W, since edge (i,j) will be - * covered by new clique C[k] */ - if (W->pos[j] != W->size) - { int jj = W->list[W->size]; - W->list[W->pos[j]] = jj; - W->pos[jj] = W->pos[j]; - } - W->size--, W->pos[j] = 0; - } - } - /* include vertex i to new clique C[k] to cover edges (i,j) - * incident to all vertices j just removed from W */ - glp_add_arc(H, i, n + k); - } - } - /* free working arrays */ - xfree(W->list); - xfree(W->pos); - xfree(V->list); - xfree(V->pos); - /* return the number of cliques in the edge covering found */ - return H->nv - n; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/keller.h b/code/3rd_glpk/misc/keller.h deleted file mode 100644 index d7a5b343..00000000 --- a/code/3rd_glpk/misc/keller.h +++ /dev/null @@ -1,34 +0,0 @@ -/* keller.h (cover edges by cliques, Kellerman's heuristic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef KELLER_H -#define KELLER_H - -#define kellerman _glp_kellerman -int kellerman(int n, int (*func)(void *info, int i, int ind[]), - void *info, void /* glp_graph */ *H); -/* cover edges by cliques with Kellerman's heuristic */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/ks.c b/code/3rd_glpk/misc/ks.c deleted file mode 100644 index 0720cc90..00000000 --- a/code/3rd_glpk/misc/ks.c +++ /dev/null @@ -1,466 +0,0 @@ -/* ks.c (0-1 knapsack problem) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2017-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ks.h" -#include "mt1.h" - -/*********************************************************************** -* 0-1 knapsack problem has the following formulation: -* -* maximize z = sum{j in 1..n} c[j]x[j] (1) -* -* s.t. sum{j in 1..n} a[j]x[j] <= b (2) -* -* x[j] in {0, 1} for all j in 1..n (3) -* -* In general case it is assumed that the instance is non-normalized, -* i.e. parameters a, b, and c may have any sign. -***********************************************************************/ - -/*********************************************************************** -* ks_enum - solve 0-1 knapsack problem by complete enumeration -* -* This routine finds optimal solution to 0-1 knapsack problem (1)-(3) -* by complete enumeration. It is intended mainly for testing purposes. -* -* The instance to be solved is specified by parameters n, a, b, and c. -* Note that these parameters can have any sign, i.e. normalization is -* not needed. -* -* On exit the routine stores the optimal point found in locations -* x[1], ..., x[n] and returns the optimal objective value. However, if -* the instance is infeasible, the routine returns INT_MIN. -* -* Since the complete enumeration is inefficient, this routine can be -* used only for small instances (n <= 20-30). */ - -#define N_MAX 40 - -int ks_enum(int n, const int a[/*1+n*/], int b, const int c[/*1+n*/], - char x[/*1+n*/]) -{ int j, s, z, z_best; - char x_best[1+N_MAX]; - xassert(0 <= n && n <= N_MAX); - /* initialization */ - memset(&x[1], 0, n * sizeof(char)); - z_best = INT_MIN; -loop: /* compute constraint and objective at current x */ - s = z = 0; - for (j = 1; j <= n; j++) - { if (x[j]) - s += a[j], z += c[j]; - } - /* check constraint violation */ - if (s > b) - goto next; - /* check objective function */ - if (z_best < z) - { /* better solution has been found */ - memcpy(&x_best[1], &x[1], n * sizeof(char)); - z_best = z; - } -next: /* generate next x */ - for (j = 1; j <= n; j++) - { if (!x[j]) - { x[j] = 1; - goto loop; - } - x[j] = 0; - } - /* report best (optimal) solution */ - memcpy(&x[1], &x_best[1], n * sizeof(char)); - return z_best; -} - -/*********************************************************************** -* reduce - prepare reduced instance of 0-1 knapsack -* -* Given original instance of 0-1 knapsack (1)-(3) specified by the -* parameters n, a, b, and c this routine transforms it to equivalent -* reduced instance in the same format. The reduced instance is -* normalized, i.e. the following additional conditions are met: -* -* n >= 2 (4) -* -* 1 <= a[j] <= b for all j in 1..n (5) -* -* sum{j in 1..n} a[j] >= b+1 (6) -* -* c[j] >= 1 for all j in 1..n (7) -* -* The routine creates the structure ks and stores there parameters n, -* a, b, and c of the reduced instance as well as template of solution -* to original instance. -* -* Normally the routine returns a pointer to the structure ks created. -* However, if the original instance is infeasible, the routine returns -* a null pointer. */ - -struct ks -{ int orig_n; - /* original problem dimension */ - int n; - /* reduced problem dimension */ - int *a; /* int a[1+orig_n]; */ - /* a{j in 1..n} are constraint coefficients (2) */ - int b; - /* b is constraint right-hand side (2) */ - int *c; /* int c[1+orig_n]; */ - /* c{j in 1..n} are objective coefficients (1) */ - int c0; - /* c0 is objective constant term */ - char *x; /* char x[1+orig_n]; */ - /* x{j in 1..orig_n} is solution template to original instance: - * x[j] = 0 x[j] is fixed at 0 - * x[j] = 1 x[j] is fixed at 1 - * x[j] = 0x10 x[j] = x[j'] - * x[j] = 0x11 x[j] = 1 - x[j'] - * where x[j'] is corresponding solution to reduced instance */ -}; - -static void free_ks(struct ks *ks); - -static struct ks *reduce(const int n, const int a[/*1+n*/], int b, - const int c[/*1+n*/]) -{ struct ks *ks; - int j, s; - xassert(n >= 0); - /* initially reduced instance is the same as original one */ - ks = talloc(1, struct ks); - ks->orig_n = n; - ks->n = 0; - ks->a = talloc(1+n, int); - memcpy(&ks->a[1], &a[1], n * sizeof(int)); - ks->b = b; - ks->c = talloc(1+n, int); - memcpy(&ks->c[1], &c[1], n * sizeof(int)); - ks->c0 = 0; - ks->x = talloc(1+n, char); - /* make all a[j] non-negative */ - for (j = 1; j <= n; j++) - { if (a[j] >= 0) - { /* keep original x[j] */ - ks->x[j] = 0x10; - } - else /* a[j] < 0 */ - { /* substitute x[j] = 1 - x'[j] */ - ks->x[j] = 0x11; - /* ... + a[j]x[j] + ... <= b - * ... + a[j](1 - x'[j]) + ... <= b - * ... - a[j]x'[j] + ... <= b - a[j] */ - ks->a[j] = - ks->a[j]; - ks->b += ks->a[j]; - /* z = ... + c[j]x[j] + ... + c0 = - * = ... + c[j](1 - x'[j]) + ... + c0 = - * = ... - c[j]x'[j] + ... + (c0 + c[j]) */ - ks->c0 += ks->c[j]; - ks->c[j] = - ks->c[j]; - } - } - /* now a[j] >= 0 for all j in 1..n */ - if (ks->b < 0) - { /* instance is infeasible */ - free_ks(ks); - return NULL; - } - /* build reduced instance */ - for (j = 1; j <= n; j++) - { if (ks->a[j] == 0) - { if (ks->c[j] <= 0) - { /* fix x[j] at 0 */ - ks->x[j] ^= 0x10; - } - else - { /* fix x[j] at 1 */ - ks->x[j] ^= 0x11; - ks->c0 += ks->c[j]; - } - } - else if (ks->a[j] > ks->b || ks->c[j] <= 0) - { /* fix x[j] at 0 */ - ks->x[j] ^= 0x10; - } - else - { /* include x[j] in reduced instance */ - ks->n++; - ks->a[ks->n] = ks->a[j]; - ks->c[ks->n] = ks->c[j]; - } - } - /* now conditions (5) and (7) are met */ - /* check condition (6) */ - s = 0; - for (j = 1; j <= ks->n; j++) - { xassert(1 <= ks->a[j] && ks->a[j] <= ks->b); - xassert(ks->c[j] >= 1); - s += ks->a[j]; - } - if (s <= ks->b) - { /* sum{j in 1..n} a[j] <= b */ - /* fix all remaining x[j] at 1 to obtain trivial solution */ - for (j = 1; j <= n; j++) - { if (ks->x[j] & 0x10) - ks->x[j] ^= 0x11; - } - for (j = 1; j <= ks->n; j++) - ks->c0 += ks->c[j]; - /* reduced instance is empty */ - ks->n = 0; - } - /* here n = 0 or n >= 2 due to condition (6) */ - xassert(ks->n == 0 || ks->n >= 2); - return ks; -} - -/*********************************************************************** -* restore - restore solution to original 0-1 knapsack instance -* -* Given optimal solution x{j in 1..ks->n} to the reduced 0-1 knapsack -* instance (previously prepared by the routine reduce) this routine -* constructs optimal solution to the original instance and stores it -* in the array ks->x{j in 1..ks->orig_n}. -* -* On exit the routine returns optimal objective value for the original -* instance. -* -* NOTE: This operation should be performed only once. */ - -static int restore(struct ks *ks, char x[]) -{ int j, k, z; - z = ks->c0; - for (j = 1, k = 0; j <= ks->orig_n; j++) - { if (ks->x[j] & 0x10) - { k++; - xassert(k <= ks->n); - xassert(x[k] == 0 || x[k] == 1); - if (ks->x[j] & 1) - ks->x[j] = 1 - x[k]; - else - ks->x[j] = x[k]; - if (x[k]) - z += ks->c[k]; - } - } - xassert(k == ks->n); - return z; -} - -/*********************************************************************** -* free_ks - deallocate structure ks -* -* This routine frees memory previously allocated to the structure ks -* and all its components. */ - -static void free_ks(struct ks *ks) -{ xassert(ks != NULL); - tfree(ks->a); - tfree(ks->c); - tfree(ks->x); - tfree(ks); -} - -/*********************************************************************** -* ks_mt1 - solve 0-1 knapsack problem with Martello & Toth algorithm -* -* This routine finds optimal solution to 0-1 knapsack problem (1)-(3) -* with Martello & Toth algorithm MT1. -* -* The instance to be solved is specified by parameters n, a, b, and c. -* Note that these parameters can have any sign, i.e. normalization is -* not needed. -* -* On exit the routine stores the optimal point found in locations -* x[1], ..., x[n] and returns the optimal objective value. However, if -* the instance is infeasible, the routine returns INT_MIN. -* -* REFERENCES -* -* S.Martello, P.Toth. Knapsack Problems: Algorithms and Computer Imp- -* lementations. John Wiley & Sons, 1990. */ - -struct mt -{ int j; - float r; /* r[j] = c[j] / a[j] */ -}; - -static int CDECL fcmp(const void *p1, const void *p2) -{ if (((struct mt *)p1)->r > ((struct mt *)p2)->r) - return -1; - else if (((struct mt *)p1)->r < ((struct mt *)p2)->r) - return +1; - else - return 0; -} - -static int mt1a(int n, const int a[], int b, const int c[], char x[]) -{ /* interface routine to MT1 */ - struct mt *mt; - int j, z, *p, *w, *x1, *xx, *min, *psign, *wsign, *zsign; - xassert(n >= 2); - /* allocate working arrays */ - mt = talloc(1+n, struct mt); - p = talloc(1+n+1, int); - w = talloc(1+n+1, int); - x1 = talloc(1+n+1, int); - xx = talloc(1+n+1, int); - min = talloc(1+n+1, int); - psign = talloc(1+n+1, int); - wsign = talloc(1+n+1, int); - zsign = talloc(1+n+1, int); - /* reorder items to provide c[j] / a[j] >= a[j+1] / a[j+1] */ - for (j = 1; j <= n; j++) - { mt[j].j = j; - mt[j].r = (float)c[j] / (float)a[j]; - } - qsort(&mt[1], n, sizeof(struct mt), fcmp); - /* load instance parameters */ - for (j = 1; j <= n; j++) - { p[j] = c[mt[j].j]; - w[j] = a[mt[j].j]; - } - /* find optimal solution */ - z = mt1(n, p, w, b, x1, 1, xx, min, psign, wsign, zsign); - xassert(z >= 0); - /* store optimal point found */ - for (j = 1; j <= n; j++) - { xassert(x1[j] == 0 || x1[j] == 1); - x[mt[j].j] = x1[j]; - } - /* free working arrays */ - tfree(mt); - tfree(p); - tfree(w); - tfree(x1); - tfree(xx); - tfree(min); - tfree(psign); - tfree(wsign); - tfree(zsign); - return z; -} - -int ks_mt1(int n, const int a[/*1+n*/], int b, const int c[/*1+n*/], - char x[/*1+n*/]) -{ struct ks *ks; - int j, s1, s2, z; - xassert(n >= 0); - /* prepare reduced instance */ - ks = reduce(n, a, b, c); - if (ks == NULL) - { /* original instance is infeasible */ - return INT_MIN; - } - /* find optimal solution to reduced instance */ - if (ks->n > 0) - mt1a(ks->n, ks->a, ks->b, ks->c, x); - /* restore solution to original instance */ - z = restore(ks, x); - memcpy(&x[1], &ks->x[1], n * sizeof(char)); - free_ks(ks); - /* check solution found */ - s1 = s2 = 0; - for (j = 1; j <= n; j++) - { xassert(x[j] == 0 || x[j] == 1); - if (x[j]) - s1 += a[j], s2 += c[j]; - } - xassert(s1 <= b); - xassert(s2 == z); - return z; -} - -/*********************************************************************** -* ks_greedy - solve 0-1 knapsack problem with greedy heuristic -* -* This routine finds (sub)optimal solution to 0-1 knapsack problem -* (1)-(3) with greedy heuristic. -* -* The instance to be solved is specified by parameters n, a, b, and c. -* Note that these parameters can have any sign, i.e. normalization is -* not needed. -* -* On exit the routine stores the optimal point found in locations -* x[1], ..., x[n] and returns the optimal objective value. However, if -* the instance is infeasible, the routine returns INT_MIN. */ - -static int greedy(int n, const int a[], int b, const int c[], char x[]) -{ /* core routine for normalized 0-1 knapsack instance */ - struct mt *mt; - int j, s, z; - xassert(n >= 2); - /* reorder items to provide c[j] / a[j] >= a[j+1] / a[j+1] */ - mt = talloc(1+n, struct mt); - for (j = 1; j <= n; j++) - { mt[j].j = j; - mt[j].r = (float)c[j] / (float)a[j]; - } - qsort(&mt[1], n, sizeof(struct mt), fcmp); - /* take items starting from most valuable ones until the knapsack - * is full */ - s = z = 0; - for (j = 1; j <= n; j++) - { if (s + a[mt[j].j] > b) - break; - x[mt[j].j] = 1; - s += a[mt[j].j]; - z += c[mt[j].j]; - } - /* don't take remaining items */ - for (j = j; j <= n; j++) - x[mt[j].j] = 0; - tfree(mt); - return z; -} - -int ks_greedy(int n, const int a[/*1+n*/], int b, const int c[/*1+n*/], - char x[/*1+n*/]) -{ struct ks *ks; - int j, s1, s2, z; - xassert(n >= 0); - /* prepare reduced instance */ - ks = reduce(n, a, b, c); - if (ks == NULL) - { /* original instance is infeasible */ - return INT_MIN; - } - /* find suboptimal solution to reduced instance */ - if (ks->n > 0) - greedy(ks->n, ks->a, ks->b, ks->c, x); - /* restore solution to original instance */ - z = restore(ks, x); - memcpy(&x[1], &ks->x[1], n * sizeof(char)); - free_ks(ks); - /* check solution found */ - s1 = s2 = 0; - for (j = 1; j <= n; j++) - { xassert(x[j] == 0 || x[j] == 1); - if (x[j]) - s1 += a[j], s2 += c[j]; - } - xassert(s1 <= b); - xassert(s2 == z); - return z; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/ks.h b/code/3rd_glpk/misc/ks.h deleted file mode 100644 index d607dc44..00000000 --- a/code/3rd_glpk/misc/ks.h +++ /dev/null @@ -1,44 +0,0 @@ -/* ks.h (0-1 knapsack problem) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2017-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef KS_H -#define KS_H - -#define ks_enum _glp_ks_enum -int ks_enum(int n, const int a[/*1+n*/], int b, const int c[/*1+n*/], - char x[/*1+n*/]); -/* solve 0-1 knapsack problem by complete enumeration */ - -#define ks_mt1 _glp_ks_mt1 -int ks_mt1(int n, const int a[/*1+n*/], int b, const int c[/*1+n*/], - char x[/*1+n*/]); -/* solve 0-1 knapsack problem with Martello & Toth algorithm */ - -#define ks_greedy _glp_ks_greedy -int ks_greedy(int n, const int a[/*1+n*/], int b, const int c[/*1+n*/], - char x[/*1+n*/]); -/* solve 0-1 knapsack problem with greedy heuristic */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mc13d.c b/code/3rd_glpk/misc/mc13d.c deleted file mode 100644 index d8bab398..00000000 --- a/code/3rd_glpk/misc/mc13d.c +++ /dev/null @@ -1,314 +0,0 @@ -/* mc13d.c (permutations to block triangular form) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* This code is the result of translation of the Fortran subroutines -* MC13D and MC13E associated with the following paper: -* -* I.S.Duff, J.K.Reid, Algorithm 529: Permutations to block triangular -* form, ACM Trans. on Math. Softw. 4 (1978), 189-192. -* -* Use of ACM Algorithms is subject to the ACM Software Copyright and -* License Agreement. See . -* -* The translation was made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mc13d.h" - -/*********************************************************************** -* NAME -* -* mc13d - permutations to block triangular form -* -* SYNOPSIS -* -* #include "mc13d.h" -* int mc13d(int n, const int icn[], const int ip[], const int lenr[], -* int ior[], int ib[], int lowl[], int numb[], int prev[]); -* -* DESCRIPTION -* -* Given the column numbers of the nonzeros in each row of the sparse -* matrix, the routine mc13d finds a symmetric permutation that makes -* the matrix block lower triangular. -* -* INPUT PARAMETERS -* -* n order of the matrix. -* -* icn array containing the column indices of the non-zeros. Those -* belonging to a single row must be contiguous but the ordering -* of column indices within each row is unimportant and wasted -* space between rows is permitted. -* -* ip ip[i], i = 1,2,...,n, is the position in array icn of the -* first column index of a non-zero in row i. -* -* lenr lenr[i], i = 1,2,...,n, is the number of non-zeros in row i. -* -* OUTPUT PARAMETERS -* -* ior ior[i], i = 1,2,...,n, gives the position on the original -* ordering of the row or column which is in position i in the -* permuted form. -* -* ib ib[i], i = 1,2,...,num, is the row number in the permuted -* matrix of the beginning of block i, 1 <= num <= n. -* -* WORKING ARRAYS -* -* arp working array of length [1+n], where arp[0] is not used. -* arp[i] is one less than the number of unsearched edges leaving -* node i. At the end of the algorithm it is set to a permutation -* which puts the matrix in block lower triangular form. -* -* ib working array of length [1+n], where ib[0] is not used. -* ib[i] is the position in the ordering of the start of the ith -* block. ib[n+1-i] holds the node number of the ith node on the -* stack. -* -* lowl working array of length [1+n], where lowl[0] is not used. -* lowl[i] is the smallest stack position of any node to which a -* path from node i has been found. It is set to n+1 when node i -* is removed from the stack. -* -* numb working array of length [1+n], where numb[0] is not used. -* numb[i] is the position of node i in the stack if it is on it, -* is the permuted order of node i for those nodes whose final -* position has been found and is otherwise zero. -* -* prev working array of length [1+n], where prev[0] is not used. -* prev[i] is the node at the end of the path when node i was -* placed on the stack. -* -* RETURNS -* -* The routine mc13d returns num, the number of blocks found. */ - -int mc13d(int n, const int icn[], const int ip[], const int lenr[], - int ior[], int ib[], int lowl[], int numb[], int prev[]) -{ int *arp = ior; - int dummy, i, i1, i2, icnt, ii, isn, ist, ist1, iv, iw, j, lcnt, - nnm1, num, stp; - /* icnt is the number of nodes whose positions in final ordering - * have been found. */ - icnt = 0; - /* num is the number of blocks that have been found. */ - num = 0; - nnm1 = n + n - 1; - /* Initialization of arrays. */ - for (j = 1; j <= n; j++) - { numb[j] = 0; - arp[j] = lenr[j] - 1; - } - for (isn = 1; isn <= n; isn++) - { /* Look for a starting node. */ - if (numb[isn] != 0) continue; - iv = isn; - /* ist is the number of nodes on the stack ... it is the stack - * pointer. */ - ist = 1; - /* Put node iv at beginning of stack. */ - lowl[iv] = numb[iv] = 1; - ib[n] = iv; - /* The body of this loop puts a new node on the stack or - * backtracks. */ - for (dummy = 1; dummy <= nnm1; dummy++) - { i1 = arp[iv]; - /* Have all edges leaving node iv been searched? */ - if (i1 >= 0) - { i2 = ip[iv] + lenr[iv] - 1; - i1 = i2 - i1; - /* Look at edges leaving node iv until one enters a new - * node or all edges are exhausted. */ - for (ii = i1; ii <= i2; ii++) - { iw = icn[ii]; - /* Has node iw been on stack already? */ - if (numb[iw] == 0) goto L70; - /* Update value of lowl[iv] if necessary. */ - if (lowl[iw] < lowl[iv]) lowl[iv] = lowl[iw]; - } - /* There are no more edges leaving node iv. */ - arp[iv] = -1; - } - /* Is node iv the root of a block? */ - if (lowl[iv] < numb[iv]) goto L60; - /* Order nodes in a block. */ - num++; - ist1 = n + 1 - ist; - lcnt = icnt + 1; - /* Peel block off the top of the stack starting at the top - * and working down to the root of the block. */ - for (stp = ist1; stp <= n; stp++) - { iw = ib[stp]; - lowl[iw] = n + 1; - numb[iw] = ++icnt; - if (iw == iv) break; - } - ist = n - stp; - ib[num] = lcnt; - /* Are there any nodes left on the stack? */ - if (ist != 0) goto L60; - /* Have all the nodes been ordered? */ - if (icnt < n) break; - goto L100; -L60: /* Backtrack to previous node on path. */ - iw = iv; - iv = prev[iv]; - /* Update value of lowl[iv] if necessary. */ - if (lowl[iw] < lowl[iv]) lowl[iv] = lowl[iw]; - continue; -L70: /* Put new node on the stack. */ - arp[iv] = i2 - ii - 1; - prev[iw] = iv; - iv = iw; - lowl[iv] = numb[iv] = ++ist; - ib[n+1-ist] = iv; - } - } -L100: /* Put permutation in the required form. */ - for (i = 1; i <= n; i++) - arp[numb[i]] = i; - return num; -} - -/**********************************************************************/ - -#ifdef GLP_TEST -#include "env.h" - -void test(int n, int ipp); - -int main(void) -{ /* test program for routine mc13d */ - test( 1, 0); - test( 2, 1); - test( 2, 2); - test( 3, 3); - test( 4, 4); - test( 5, 10); - test(10, 10); - test(10, 20); - test(20, 20); - test(20, 50); - test(50, 50); - test(50, 200); - return 0; -} - -void fa01bs(int max, int *nrand); - -void setup(int n, char a[1+50][1+50], int ip[], int icn[], int lenr[]); - -void test(int n, int ipp) -{ int ip[1+50], icn[1+1000], ior[1+50], ib[1+51], iw[1+150], - lenr[1+50]; - char a[1+50][1+50], hold[1+100]; - int i, ii, iblock, ij, index, j, jblock, jj, k9, num; - xprintf("\n\n\nMatrix is of order %d and has %d off-diagonal non-" - "zeros\n", n, ipp); - for (j = 1; j <= n; j++) - { for (i = 1; i <= n; i++) - a[i][j] = 0; - a[j][j] = 1; - } - for (k9 = 1; k9 <= ipp; k9++) - { /* these statements should be replaced by calls to your - * favorite random number generator to place two pseudo-random - * numbers between 1 and n in the variables i and j */ - for (;;) - { fa01bs(n, &i); - fa01bs(n, &j); - if (!a[i][j]) break; - } - a[i][j] = 1; - } - /* setup converts matrix a[i,j] to required sparsity-oriented - * storage format */ - setup(n, a, ip, icn, lenr); - num = mc13d(n, icn, ip, lenr, ior, ib, &iw[0], &iw[n], &iw[n+n]); - /* output reordered matrix with blocking to improve clarity */ - xprintf("\nThe reordered matrix which has %d block%s is of the fo" - "rm\n", num, num == 1 ? "" : "s"); - ib[num+1] = n + 1; - index = 100; - iblock = 1; - for (i = 1; i <= n; i++) - { for (ij = 1; ij <= index; ij++) - hold[ij] = ' '; - if (i == ib[iblock]) - { xprintf("\n"); - iblock++; - } - jblock = 1; - index = 0; - for (j = 1; j <= n; j++) - { if (j == ib[jblock]) - { hold[++index] = ' '; - jblock++; - } - ii = ior[i]; - jj = ior[j]; - hold[++index] = (char)(a[ii][jj] ? 'X' : '0'); - } - xprintf("%.*s\n", index, &hold[1]); - } - xprintf("\nThe starting point for each block is given by\n"); - for (i = 1; i <= num; i++) - { if ((i - 1) % 12 == 0) xprintf("\n"); - xprintf(" %4d", ib[i]); - } - xprintf("\n"); - return; -} - -void setup(int n, char a[1+50][1+50], int ip[], int icn[], int lenr[]) -{ int i, j, ind; - for (i = 1; i <= n; i++) - lenr[i] = 0; - ind = 1; - for (i = 1; i <= n; i++) - { ip[i] = ind; - for (j = 1; j <= n; j++) - { if (a[i][j]) - { lenr[i]++; - icn[ind++] = j; - } - } - } - return; -} - -double g = 1431655765.0; - -double fa01as(int i) -{ /* random number generator */ - g = fmod(g * 9228907.0, 4294967296.0); - if (i >= 0) - return g / 4294967296.0; - else - return 2.0 * g / 4294967296.0 - 1.0; -} - -void fa01bs(int max, int *nrand) -{ *nrand = (int)(fa01as(1) * (double)max) + 1; - return; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mc13d.h b/code/3rd_glpk/misc/mc13d.h deleted file mode 100644 index bdd57a19..00000000 --- a/code/3rd_glpk/misc/mc13d.h +++ /dev/null @@ -1,34 +0,0 @@ -/* mc13d.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MC13D_H -#define MC13D_H - -#define mc13d _glp_mc13d -int mc13d(int n, const int icn[], const int ip[], const int lenr[], - int ior[], int ib[], int lowl[], int numb[], int prev[]); -/* permutations to block triangular form */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mc21a.c b/code/3rd_glpk/misc/mc21a.c deleted file mode 100644 index 700d0f4e..00000000 --- a/code/3rd_glpk/misc/mc21a.c +++ /dev/null @@ -1,301 +0,0 @@ -/* mc21a.c (permutations for zero-free diagonal) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* This code is the result of translation of the Fortran subroutines -* MC21A and MC21B associated with the following paper: -* -* I.S.Duff, Algorithm 575: Permutations for zero-free diagonal, ACM -* Trans. on Math. Softw. 7 (1981), 387-390. -* -* Use of ACM Algorithms is subject to the ACM Software Copyright and -* License Agreement. See . -* -* The translation was made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mc21a.h" - -/*********************************************************************** -* NAME -* -* mc21a - permutations for zero-free diagonal -* -* SYNOPSIS -* -* #include "mc21a.h" -* int mc21a(int n, const int icn[], const int ip[], const int lenr[], -* int iperm[], int pr[], int arp[], int cv[], int out[]); -* -* DESCRIPTION -* -* Given the pattern of nonzeros of a sparse matrix, the routine mc21a -* attempts to find a permutation of its rows that makes the matrix have -* no zeros on its diagonal. -* -* INPUT PARAMETERS -* -* n order of matrix. -* -* icn array containing the column indices of the non-zeros. Those -* belonging to a single row must be contiguous but the ordering -* of column indices within each row is unimportant and wasted -* space between rows is permitted. -* -* ip ip[i], i = 1,2,...,n, is the position in array icn of the -* first column index of a non-zero in row i. -* -* lenr lenr[i], i = 1,2,...,n, is the number of non-zeros in row i. -* -* OUTPUT PARAMETER -* -* iperm contains permutation to make diagonal have the smallest -* number of zeros on it. Elements (iperm[i], i), i = 1,2,...,n, -* are non-zero at the end of the algorithm unless the matrix is -* structurally singular. In this case, (iperm[i], i) will be -* zero for n - numnz entries. -* -* WORKING ARRAYS -* -* pr working array of length [1+n], where pr[0] is not used. -* pr[i] is the previous row to i in the depth first search. -* -* arp working array of length [1+n], where arp[0] is not used. -* arp[i] is one less than the number of non-zeros in row i which -* have not been scanned when looking for a cheap assignment. -* -* cv working array of length [1+n], where cv[0] is not used. -* cv[i] is the most recent row extension at which column i was -* visited. -* -* out working array of length [1+n], where out[0] is not used. -* out[i] is one less than the number of non-zeros in row i -* which have not been scanned during one pass through the main -* loop. -* -* RETURNS -* -* The routine mc21a returns numnz, the number of non-zeros on diagonal -* of permuted matrix. */ - -int mc21a(int n, const int icn[], const int ip[], const int lenr[], - int iperm[], int pr[], int arp[], int cv[], int out[]) -{ int i, ii, in1, in2, j, j1, jord, k, kk, numnz; - /* Initialization of arrays. */ - for (i = 1; i <= n; i++) - { arp[i] = lenr[i] - 1; - cv[i] = iperm[i] = 0; - } - numnz = 0; - /* Main loop. */ - /* Each pass round this loop either results in a new assignment - * or gives a row with no assignment. */ - for (jord = 1; jord <= n; jord++) - { j = jord; - pr[j] = -1; - for (k = 1; k <= jord; k++) - { /* Look for a cheap assignment. */ - in1 = arp[j]; - if (in1 >= 0) - { in2 = ip[j] + lenr[j] - 1; - in1 = in2 - in1; - for (ii = in1; ii <= in2; ii++) - { i = icn[ii]; - if (iperm[i] == 0) goto L110; - } - /* No cheap assignment in row. */ - arp[j] = -1; - } - /* Begin looking for assignment chain starting with row j.*/ - out[j] = lenr[j] - 1; - /* Inner loop. Extends chain by one or backtracks. */ - for (kk = 1; kk <= jord; kk++) - { in1 = out[j]; - if (in1 >= 0) - { in2 = ip[j] + lenr[j] - 1; - in1 = in2 - in1; - /* Forward scan. */ - for (ii = in1; ii <= in2; ii++) - { i = icn[ii]; - if (cv[i] != jord) - { /* Column i has not yet been accessed during - * this pass. */ - j1 = j; - j = iperm[i]; - cv[i] = jord; - pr[j] = j1; - out[j1] = in2 - ii - 1; - goto L100; - } - } - } - /* Backtracking step. */ - j = pr[j]; - if (j == -1) goto L130; - } -L100: ; - } -L110: /* New assignment is made. */ - iperm[i] = j; - arp[j] = in2 - ii - 1; - numnz++; - for (k = 1; k <= jord; k++) - { j = pr[j]; - if (j == -1) break; - ii = ip[j] + lenr[j] - out[j] - 2; - i = icn[ii]; - iperm[i] = j; - } -L130: ; - } - /* If matrix is structurally singular, we now complete the - * permutation iperm. */ - if (numnz < n) - { for (i = 1; i <= n; i++) - arp[i] = 0; - k = 0; - for (i = 1; i <= n; i++) - { if (iperm[i] == 0) - out[++k] = i; - else - arp[iperm[i]] = i; - } - k = 0; - for (i = 1; i <= n; i++) - { if (arp[i] == 0) - iperm[out[++k]] = i; - } - } - return numnz; -} - -/**********************************************************************/ - -#ifdef GLP_TEST -#include "env.h" - -int sing; - -void ranmat(int m, int n, int icn[], int iptr[], int nnnp1, int *knum, - int iw[]); - -void fa01bs(int max, int *nrand); - -int main(void) -{ /* test program for the routine mc21a */ - /* these runs on random matrices cause all possible statements in - * mc21a to be executed */ - int i, iold, j, j1, j2, jj, knum, l, licn, n, nov4, num, numnz; - int ip[1+21], icn[1+1000], iperm[1+20], lenr[1+20], iw1[1+80]; - licn = 1000; - /* run on random matrices of orders 1 through 20 */ - for (n = 1; n <= 20; n++) - { nov4 = n / 4; - if (nov4 < 1) nov4 = 1; -L10: fa01bs(nov4, &l); - knum = l * n; - /* knum is requested number of non-zeros in random matrix */ - if (knum > licn) goto L10; - /* if sing is false, matrix is guaranteed structurally - * non-singular */ - sing = ((n / 2) * 2 == n); - /* call to subroutine to generate random matrix */ - ranmat(n, n, icn, ip, n+1, &knum, iw1); - /* knum is now actual number of non-zeros in random matrix */ - if (knum > licn) goto L10; - xprintf("n = %2d; nz = %4d; sing = %d\n", n, knum, sing); - /* set up array of row lengths */ - for (i = 1; i <= n; i++) - lenr[i] = ip[i+1] - ip[i]; - /* call to mc21a */ - numnz = mc21a(n, icn, ip, lenr, iperm, &iw1[0], &iw1[n], - &iw1[n+n], &iw1[n+n+n]); - /* testing to see if there are numnz non-zeros on the diagonal - * of the permuted matrix. */ - num = 0; - for (i = 1; i <= n; i++) - { iold = iperm[i]; - j1 = ip[iold]; - j2 = j1 + lenr[iold] - 1; - if (j2 < j1) continue; - for (jj = j1; jj <= j2; jj++) - { j = icn[jj]; - if (j == i) - { num++; - break; - } - } - } - if (num != numnz) - xprintf("Failure in mc21a, numnz = %d instead of %d\n", - numnz, num); - } - return 0; -} - -void ranmat(int m, int n, int icn[], int iptr[], int nnnp1, int *knum, - int iw[]) -{ /* subroutine to generate random matrix */ - int i, ii, inum, j, lrow, matnum; - inum = (*knum / n) * 2; - if (inum > n-1) inum = n-1; - matnum = 1; - /* each pass through this loop generates a row of the matrix */ - for (j = 1; j <= m; j++) - { iptr[j] = matnum; - if (!(sing || j > n)) - icn[matnum++] = j; - if (n == 1) continue; - for (i = 1; i <= n; i++) iw[i] = 0; - if (!sing) iw[j] = 1; - fa01bs(inum, &lrow); - lrow--; - if (lrow == 0) continue; - /* lrow off-diagonal non-zeros in row j of the matrix */ - for (ii = 1; ii <= lrow; ii++) - { for (;;) - { fa01bs(n, &i); - if (iw[i] != 1) break; - } - iw[i] = 1; - icn[matnum++] = i; - } - } - for (i = m+1; i <= nnnp1; i++) - iptr[i] = matnum; - *knum = matnum - 1; - return; -} - -double g = 1431655765.0; - -double fa01as(int i) -{ /* random number generator */ - g = fmod(g * 9228907.0, 4294967296.0); - if (i >= 0) - return g / 4294967296.0; - else - return 2.0 * g / 4294967296.0 - 1.0; -} - -void fa01bs(int max, int *nrand) -{ *nrand = (int)(fa01as(1) * (double)max) + 1; - return; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mc21a.h b/code/3rd_glpk/misc/mc21a.h deleted file mode 100644 index 755f28b2..00000000 --- a/code/3rd_glpk/misc/mc21a.h +++ /dev/null @@ -1,34 +0,0 @@ -/* mc21a.h (permutations for zero-free diagonal) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MC21A_H -#define MC21A_H - -#define mc21a _glp_mc21a -int mc21a(int n, const int icn[], const int ip[], const int lenr[], - int iperm[], int pr[], int arp[], int cv[], int out[]); -/* permutations for zero-free diagonal */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/misc.h b/code/3rd_glpk/misc/misc.h deleted file mode 100644 index 1ba1dc50..00000000 --- a/code/3rd_glpk/misc/misc.h +++ /dev/null @@ -1,61 +0,0 @@ -/* misc.h (miscellaneous routines) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MISC_H -#define MISC_H - -#define str2int _glp_str2int -int str2int(const char *str, int *val); -/* convert character string to value of int type */ - -#define str2num _glp_str2num -int str2num(const char *str, double *val); -/* convert character string to value of double type */ - -#define strspx _glp_strspx -char *strspx(char *str); -/* remove all spaces from character string */ - -#define strtrim _glp_strtrim -char *strtrim(char *str); -/* remove trailing spaces from character string */ - -#define gcd _glp_gcd -int gcd(int x, int y); -/* find greatest common divisor of two integers */ - -#define gcdn _glp_gcdn -int gcdn(int n, int x[]); -/* find greatest common divisor of n integers */ - -#define round2n _glp_round2n -double round2n(double x); -/* round floating-point number to nearest power of two */ - -#define fp2rat _glp_fp2rat -int fp2rat(double x, double eps, double *p, double *q); -/* convert floating-point number to rational number */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mt1.c b/code/3rd_glpk/misc/mt1.c deleted file mode 100644 index 63a0f80e..00000000 --- a/code/3rd_glpk/misc/mt1.c +++ /dev/null @@ -1,1110 +0,0 @@ -/* mt1.c (0-1 knapsack problem; Martello & Toth algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* THIS CODE IS THE RESULT OF TRANSLATION OF THE FORTRAN SUBROUTINES -* MT1 FROM THE BOOK: -* -* SILVANO MARTELLO, PAOLO TOTH. KNAPSACK PROBLEMS: ALGORITHMS AND -* COMPUTER IMPLEMENTATIONS. JOHN WILEY & SONS, 1990. -* -* THE TRANSLATION HAS BEEN DONE WITH THE PERMISSION OF THE AUTHORS OF -* THE ORIGINAL FORTRAN SUBROUTINES: SILVANO MARTELLO AND PAOLO TOTH. -* -* The translation was made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#line 1 "" -/* -- translated by f2c (version 20100827). - You must link the resulting object file with libf2c: - on Microsoft Windows system, link with libf2c.lib; - on Linux or Unix systems, link with .../path/to/libf2c.a -lm - or, if you install libf2c.a in a standard place, with -lf2c -lm - -- in that order, at the end of the command line, as in - cc *.o -lf2c -lm - Source for libf2c is in /netlib/f2c/libf2c.zip, e.g., - - http://www.netlib.org/f2c/libf2c.zip -*/ - -#if 0 /* by mao */ -#include "f2c.h" -#else -#include "env.h" -#include "mt1.h" - -typedef int integer; -typedef float real; -#endif - -#line 1 "" -/*< SUBROUTINE MT1(N,P,W,C,Z,X,JDIM,JCK,XX,MIN,PSIGN,WSIGN,ZSIGN) >*/ -#if 1 /* by mao */ -static int chmt1_(int *, int *, int *, int *, int *, int *); - -static -#endif -/* Subroutine */ int mt1_(integer *n, integer *p, integer *w, integer *c__, - integer *z__, integer *x, integer *jdim, integer *jck, integer *xx, - integer *min__, integer *psign, integer *wsign, integer *zsign) -{ - /* System generated locals */ - integer i__1; - - /* Local variables */ - static real a, b; - static integer j, r__, t, j1, n1, ch, ii, jj, kk, in, ll, ip, nn, iu, ii1, - chs, lim, lim1, diff, lold, mink; - extern /* Subroutine */ int chmt1_(integer *, integer *, integer *, - integer *, integer *, integer *); - static integer profit; - - -/* THIS SUBROUTINE SOLVES THE 0-1 SINGLE KNAPSACK PROBLEM */ - -/* MAXIMIZE Z = P(1)*X(1) + ... + P(N)*X(N) */ - -/* SUBJECT TO: W(1)*X(1) + ... + W(N)*X(N) .LE. C , */ -/* X(J) = 0 OR 1 FOR J=1,...,N. */ - -/* THE PROGRAM IS INCLUDED IN THE VOLUME */ -/* S. MARTELLO, P. TOTH, "KNAPSACK PROBLEMS: ALGORITHMS */ -/* AND COMPUTER IMPLEMENTATIONS", JOHN WILEY, 1990 */ -/* AND IMPLEMENTS THE BRANCH-AND-BOUND ALGORITHM DESCRIBED IN */ -/* SECTION 2.5.2 . */ -/* THE PROGRAM DERIVES FROM AN EARLIER CODE PRESENTED IN */ -/* S. MARTELLO, P. TOTH, "ALGORITHM FOR THE SOLUTION OF THE 0-1 SINGLE */ -/* KNAPSACK PROBLEM", COMPUTING, 1978. */ - -/* THE INPUT PROBLEM MUST SATISFY THE CONDITIONS */ - -/* 1) 2 .LE. N .LE. JDIM - 1 ; */ -/* 2) P(J), W(J), C POSITIVE INTEGERS; */ -/* 3) MAX (W(J)) .LE. C ; */ -/* 4) W(1) + ... + W(N) .GT. C ; */ -/* 5) P(J)/W(J) .GE. P(J+1)/W(J+1) FOR J=1,...,N-1. */ - -/* MT1 CALLS 1 PROCEDURE: CHMT1. */ - -/* THE PROGRAM IS COMPLETELY SELF-CONTAINED AND COMMUNICATION TO IT IS */ -/* ACHIEVED SOLELY THROUGH THE PARAMETER LIST OF MT1. */ -/* NO MACHINE-DEPENDENT CONSTANT IS USED. */ -/* THE PROGRAM IS WRITTEN IN 1967 AMERICAN NATIONAL STANDARD FORTRAN */ -/* AND IS ACCEPTED BY THE PFORT VERIFIER (PFORT IS THE PORTABLE */ -/* SUBSET OF ANSI DEFINED BY THE ASSOCIATION FOR COMPUTING MACHINERY). */ -/* THE PROGRAM HAS BEEN TESTED ON A DIGITAL VAX 11/780 AND AN H.P. */ -/* 9000/840. */ - -/* MT1 NEEDS 8 ARRAYS ( P , W , X , XX , MIN , PSIGN , WSIGN */ -/* AND ZSIGN ) OF LENGTH AT LEAST N + 1 . */ - -/* MEANING OF THE INPUT PARAMETERS: */ -/* N = NUMBER OF ITEMS; */ -/* P(J) = PROFIT OF ITEM J (J=1,...,N); */ -/* W(J) = WEIGHT OF ITEM J (J=1,...,N); */ -/* C = CAPACITY OF THE KNAPSACK; */ -/* JDIM = DIMENSION OF THE 8 ARRAYS; */ -/* JCK = 1 IF CHECK ON THE INPUT DATA IS DESIRED, */ -/* = 0 OTHERWISE. */ - -/* MEANING OF THE OUTPUT PARAMETERS: */ -/* Z = VALUE OF THE OPTIMAL SOLUTION IF Z .GT. 0 , */ -/* = ERROR IN THE INPUT DATA (WHEN JCK=1) IF Z .LT. 0 : CONDI- */ -/* TION - Z IS VIOLATED; */ -/* X(J) = 1 IF ITEM J IS IN THE OPTIMAL SOLUTION, */ -/* = 0 OTHERWISE. */ - -/* ARRAYS XX, MIN, PSIGN, WSIGN AND ZSIGN ARE DUMMY. */ - -/* ALL THE PARAMETERS ARE INTEGER. ON RETURN OF MT1 ALL THE INPUT */ -/* PARAMETERS ARE UNCHANGED. */ - -/*< INTEGER P(JDIM),W(JDIM),X(JDIM),C,Z >*/ -/*< INTEGER XX(JDIM),MIN(JDIM),PSIGN(JDIM),WSIGN(JDIM),ZSIGN(JDIM) >*/ -/*< INTEGER CH,CHS,DIFF,PROFIT,R,T >*/ -/*< Z = 0 >*/ -#line 65 "" - /* Parameter adjustments */ -#line 65 "" - --zsign; -#line 65 "" - --wsign; -#line 65 "" - --psign; -#line 65 "" - --min__; -#line 65 "" - --xx; -#line 65 "" - --x; -#line 65 "" - --w; -#line 65 "" - --p; -#line 65 "" - -#line 65 "" - /* Function Body */ -#line 65 "" - *z__ = 0; -/*< IF ( JCK .EQ. 1 ) CALL CHMT1(N,P,W,C,Z,JDIM) >*/ -#line 66 "" - if (*jck == 1) { -#line 66 "" - chmt1_(n, &p[1], &w[1], c__, z__, jdim); -#line 66 "" - } -/*< IF ( Z .LT. 0 ) RETURN >*/ -#line 67 "" - if (*z__ < 0) { -#line 67 "" - return 0; -#line 67 "" - } -/* INITIALIZE. */ -/*< CH = C >*/ -#line 69 "" - ch = *c__; -/*< IP = 0 >*/ -#line 70 "" - ip = 0; -/*< CHS = CH >*/ -#line 71 "" - chs = ch; -/*< DO 10 LL=1,N >*/ -#line 72 "" - i__1 = *n; -#line 72 "" - for (ll = 1; ll <= i__1; ++ll) { -/*< IF ( W(LL) .GT. CHS ) GO TO 20 >*/ -#line 73 "" - if (w[ll] > chs) { -#line 73 "" - goto L20; -#line 73 "" - } -/*< IP = IP + P(LL) >*/ -#line 74 "" - ip += p[ll]; -/*< CHS = CHS - W(LL) >*/ -#line 75 "" - chs -= w[ll]; -/*< 10 CONTINUE >*/ -#line 76 "" -/* L10: */ -#line 76 "" - } -/*< 20 LL = LL - 1 >*/ -#line 77 "" -L20: -#line 77 "" - --ll; -/*< IF ( CHS .EQ. 0 ) GO TO 50 >*/ -#line 78 "" - if (chs == 0) { -#line 78 "" - goto L50; -#line 78 "" - } -/*< P(N+1) = 0 >*/ -#line 79 "" - p[*n + 1] = 0; -/*< W(N+1) = CH + 1 >*/ -#line 80 "" - w[*n + 1] = ch + 1; -/*< LIM = IP + CHS*P(LL+2)/W(LL+2) >*/ -#line 81 "" - lim = ip + chs * p[ll + 2] / w[ll + 2]; -/*< A = W(LL+1) - CHS >*/ -#line 82 "" - a = (real) (w[ll + 1] - chs); -/*< B = IP + P(LL+1) >*/ -#line 83 "" - b = (real) (ip + p[ll + 1]); -/*< LIM1 = B - A*FLOAT(P(LL))/FLOAT(W(LL)) >*/ -#line 84 "" - lim1 = b - a * (real) p[ll] / (real) w[ll]; -/*< IF ( LIM1 .GT. LIM ) LIM = LIM1 >*/ -#line 85 "" - if (lim1 > lim) { -#line 85 "" - lim = lim1; -#line 85 "" - } -/*< MINK = CH + 1 >*/ -#line 86 "" - mink = ch + 1; -/*< MIN(N) = MINK >*/ -#line 87 "" - min__[*n] = mink; -/*< DO 30 J=2,N >*/ -#line 88 "" - i__1 = *n; -#line 88 "" - for (j = 2; j <= i__1; ++j) { -/*< KK = N + 2 - J >*/ -#line 89 "" - kk = *n + 2 - j; -/*< IF ( W(KK) .LT. MINK ) MINK = W(KK) >*/ -#line 90 "" - if (w[kk] < mink) { -#line 90 "" - mink = w[kk]; -#line 90 "" - } -/*< MIN(KK-1) = MINK >*/ -#line 91 "" - min__[kk - 1] = mink; -/*< 30 CONTINUE >*/ -#line 92 "" -/* L30: */ -#line 92 "" - } -/*< DO 40 J=1,N >*/ -#line 93 "" - i__1 = *n; -#line 93 "" - for (j = 1; j <= i__1; ++j) { -/*< XX(J) = 0 >*/ -#line 94 "" - xx[j] = 0; -/*< 40 CONTINUE >*/ -#line 95 "" -/* L40: */ -#line 95 "" - } -/*< Z = 0 >*/ -#line 96 "" - *z__ = 0; -/*< PROFIT = 0 >*/ -#line 97 "" - profit = 0; -/*< LOLD = N >*/ -#line 98 "" - lold = *n; -/*< II = 1 >*/ -#line 99 "" - ii = 1; -/*< GO TO 170 >*/ -#line 100 "" - goto L170; -/*< 50 Z = IP >*/ -#line 101 "" -L50: -#line 101 "" - *z__ = ip; -/*< DO 60 J=1,LL >*/ -#line 102 "" - i__1 = ll; -#line 102 "" - for (j = 1; j <= i__1; ++j) { -/*< X(J) = 1 >*/ -#line 103 "" - x[j] = 1; -/*< 60 CONTINUE >*/ -#line 104 "" -/* L60: */ -#line 104 "" - } -/*< NN = LL + 1 >*/ -#line 105 "" - nn = ll + 1; -/*< DO 70 J=NN,N >*/ -#line 106 "" - i__1 = *n; -#line 106 "" - for (j = nn; j <= i__1; ++j) { -/*< X(J) = 0 >*/ -#line 107 "" - x[j] = 0; -/*< 70 CONTINUE >*/ -#line 108 "" -/* L70: */ -#line 108 "" - } -/*< RETURN >*/ -#line 109 "" - return 0; -/* TRY TO INSERT THE II-TH ITEM INTO THE CURRENT SOLUTION. */ -/*< 80 IF ( W(II) .LE. CH ) GO TO 90 >*/ -#line 111 "" -L80: -#line 111 "" - if (w[ii] <= ch) { -#line 111 "" - goto L90; -#line 111 "" - } -/*< II1 = II + 1 >*/ -#line 112 "" - ii1 = ii + 1; -/*< IF ( Z .GE. CH*P(II1)/W(II1) + PROFIT ) GO TO 280 >*/ -#line 113 "" - if (*z__ >= ch * p[ii1] / w[ii1] + profit) { -#line 113 "" - goto L280; -#line 113 "" - } -/*< II = II1 >*/ -#line 114 "" - ii = ii1; -/*< GO TO 80 >*/ -#line 115 "" - goto L80; -/* BUILD A NEW CURRENT SOLUTION. */ -/*< 90 IP = PSIGN(II) >*/ -#line 117 "" -L90: -#line 117 "" - ip = psign[ii]; -/*< CHS = CH - WSIGN(II) >*/ -#line 118 "" - chs = ch - wsign[ii]; -/*< IN = ZSIGN(II) >*/ -#line 119 "" - in = zsign[ii]; -/*< DO 100 LL=IN,N >*/ -#line 120 "" - i__1 = *n; -#line 120 "" - for (ll = in; ll <= i__1; ++ll) { -/*< IF ( W(LL) .GT. CHS ) GO TO 160 >*/ -#line 121 "" - if (w[ll] > chs) { -#line 121 "" - goto L160; -#line 121 "" - } -/*< IP = IP + P(LL) >*/ -#line 122 "" - ip += p[ll]; -/*< CHS = CHS - W(LL) >*/ -#line 123 "" - chs -= w[ll]; -/*< 100 CONTINUE >*/ -#line 124 "" -/* L100: */ -#line 124 "" - } -/*< LL = N >*/ -#line 125 "" - ll = *n; -/*< 110 IF ( Z .GE. IP + PROFIT ) GO TO 280 >*/ -#line 126 "" -L110: -#line 126 "" - if (*z__ >= ip + profit) { -#line 126 "" - goto L280; -#line 126 "" - } -/*< Z = IP + PROFIT >*/ -#line 127 "" - *z__ = ip + profit; -/*< NN = II - 1 >*/ -#line 128 "" - nn = ii - 1; -/*< DO 120 J=1,NN >*/ -#line 129 "" - i__1 = nn; -#line 129 "" - for (j = 1; j <= i__1; ++j) { -/*< X(J) = XX(J) >*/ -#line 130 "" - x[j] = xx[j]; -/*< 120 CONTINUE >*/ -#line 131 "" -/* L120: */ -#line 131 "" - } -/*< DO 130 J=II,LL >*/ -#line 132 "" - i__1 = ll; -#line 132 "" - for (j = ii; j <= i__1; ++j) { -/*< X(J) = 1 >*/ -#line 133 "" - x[j] = 1; -/*< 130 CONTINUE >*/ -#line 134 "" -/* L130: */ -#line 134 "" - } -/*< IF ( LL .EQ. N ) GO TO 150 >*/ -#line 135 "" - if (ll == *n) { -#line 135 "" - goto L150; -#line 135 "" - } -/*< NN = LL + 1 >*/ -#line 136 "" - nn = ll + 1; -/*< DO 140 J=NN,N >*/ -#line 137 "" - i__1 = *n; -#line 137 "" - for (j = nn; j <= i__1; ++j) { -/*< X(J) = 0 >*/ -#line 138 "" - x[j] = 0; -/*< 140 CONTINUE >*/ -#line 139 "" -/* L140: */ -#line 139 "" - } -/*< 150 IF ( Z .NE. LIM ) GO TO 280 >*/ -#line 140 "" -L150: -#line 140 "" - if (*z__ != lim) { -#line 140 "" - goto L280; -#line 140 "" - } -/*< RETURN >*/ -#line 141 "" - return 0; -/*< 160 IU = CHS*P(LL)/W(LL) >*/ -#line 142 "" -L160: -#line 142 "" - iu = chs * p[ll] / w[ll]; -/*< LL = LL - 1 >*/ -#line 143 "" - --ll; -/*< IF ( IU .EQ. 0 ) GO TO 110 >*/ -#line 144 "" - if (iu == 0) { -#line 144 "" - goto L110; -#line 144 "" - } -/*< IF ( Z .GE. PROFIT + IP + IU ) GO TO 280 >*/ -#line 145 "" - if (*z__ >= profit + ip + iu) { -#line 145 "" - goto L280; -#line 145 "" - } -/* SAVE THE CURRENT SOLUTION. */ -/*< 170 WSIGN(II) = CH - CHS >*/ -#line 147 "" -L170: -#line 147 "" - wsign[ii] = ch - chs; -/*< PSIGN(II) = IP >*/ -#line 148 "" - psign[ii] = ip; -/*< ZSIGN(II) = LL + 1 >*/ -#line 149 "" - zsign[ii] = ll + 1; -/*< XX(II) = 1 >*/ -#line 150 "" - xx[ii] = 1; -/*< NN = LL - 1 >*/ -#line 151 "" - nn = ll - 1; -/*< IF ( NN .LT. II) GO TO 190 >*/ -#line 152 "" - if (nn < ii) { -#line 152 "" - goto L190; -#line 152 "" - } -/*< DO 180 J=II,NN >*/ -#line 153 "" - i__1 = nn; -#line 153 "" - for (j = ii; j <= i__1; ++j) { -/*< WSIGN(J+1) = WSIGN(J) - W(J) >*/ -#line 154 "" - wsign[j + 1] = wsign[j] - w[j]; -/*< PSIGN(J+1) = PSIGN(J) - P(J) >*/ -#line 155 "" - psign[j + 1] = psign[j] - p[j]; -/*< ZSIGN(J+1) = LL + 1 >*/ -#line 156 "" - zsign[j + 1] = ll + 1; -/*< XX(J+1) = 1 >*/ -#line 157 "" - xx[j + 1] = 1; -/*< 180 CONTINUE >*/ -#line 158 "" -/* L180: */ -#line 158 "" - } -/*< 190 J1 = LL + 1 >*/ -#line 159 "" -L190: -#line 159 "" - j1 = ll + 1; -/*< DO 200 J=J1,LOLD >*/ -#line 160 "" - i__1 = lold; -#line 160 "" - for (j = j1; j <= i__1; ++j) { -/*< WSIGN(J) = 0 >*/ -#line 161 "" - wsign[j] = 0; -/*< PSIGN(J) = 0 >*/ -#line 162 "" - psign[j] = 0; -/*< ZSIGN(J) = J >*/ -#line 163 "" - zsign[j] = j; -/*< 200 CONTINUE >*/ -#line 164 "" -/* L200: */ -#line 164 "" - } -/*< LOLD = LL >*/ -#line 165 "" - lold = ll; -/*< CH = CHS >*/ -#line 166 "" - ch = chs; -/*< PROFIT = PROFIT + IP >*/ -#line 167 "" - profit += ip; -/*< IF ( LL - (N - 2) ) 240, 220, 210 >*/ -#line 168 "" - if ((i__1 = ll - (*n - 2)) < 0) { -#line 168 "" - goto L240; -#line 168 "" - } else if (i__1 == 0) { -#line 168 "" - goto L220; -#line 168 "" - } else { -#line 168 "" - goto L210; -#line 168 "" - } -/*< 210 II = N >*/ -#line 169 "" -L210: -#line 169 "" - ii = *n; -/*< GO TO 250 >*/ -#line 170 "" - goto L250; -/*< 220 IF ( CH .LT. W(N) ) GO TO 230 >*/ -#line 171 "" -L220: -#line 171 "" - if (ch < w[*n]) { -#line 171 "" - goto L230; -#line 171 "" - } -/*< CH = CH - W(N) >*/ -#line 172 "" - ch -= w[*n]; -/*< PROFIT = PROFIT + P(N) >*/ -#line 173 "" - profit += p[*n]; -/*< XX(N) = 1 >*/ -#line 174 "" - xx[*n] = 1; -/*< 230 II = N - 1 >*/ -#line 175 "" -L230: -#line 175 "" - ii = *n - 1; -/*< GO TO 250 >*/ -#line 176 "" - goto L250; -/*< 240 II = LL + 2 >*/ -#line 177 "" -L240: -#line 177 "" - ii = ll + 2; -/*< IF ( CH .GE. MIN(II-1) ) GO TO 80 >*/ -#line 178 "" - if (ch >= min__[ii - 1]) { -#line 178 "" - goto L80; -#line 178 "" - } -/* SAVE THE CURRENT OPTIMAL SOLUTION. */ -/*< 250 IF ( Z .GE. PROFIT ) GO TO 270 >*/ -#line 180 "" -L250: -#line 180 "" - if (*z__ >= profit) { -#line 180 "" - goto L270; -#line 180 "" - } -/*< Z = PROFIT >*/ -#line 181 "" - *z__ = profit; -/*< DO 260 J=1,N >*/ -#line 182 "" - i__1 = *n; -#line 182 "" - for (j = 1; j <= i__1; ++j) { -/*< X(J) = XX(J) >*/ -#line 183 "" - x[j] = xx[j]; -/*< 260 CONTINUE >*/ -#line 184 "" -/* L260: */ -#line 184 "" - } -/*< IF ( Z .EQ. LIM ) RETURN >*/ -#line 185 "" - if (*z__ == lim) { -#line 185 "" - return 0; -#line 185 "" - } -/*< 270 IF ( XX(N) .EQ. 0 ) GO TO 280 >*/ -#line 186 "" -L270: -#line 186 "" - if (xx[*n] == 0) { -#line 186 "" - goto L280; -#line 186 "" - } -/*< XX(N) = 0 >*/ -#line 187 "" - xx[*n] = 0; -/*< CH = CH + W(N) >*/ -#line 188 "" - ch += w[*n]; -/*< PROFIT = PROFIT - P(N) >*/ -#line 189 "" - profit -= p[*n]; -/* BACKTRACK. */ -/*< 280 NN = II - 1 >*/ -#line 191 "" -L280: -#line 191 "" - nn = ii - 1; -/*< IF ( NN .EQ. 0 ) RETURN >*/ -#line 192 "" - if (nn == 0) { -#line 192 "" - return 0; -#line 192 "" - } -/*< DO 290 J=1,NN >*/ -#line 193 "" - i__1 = nn; -#line 193 "" - for (j = 1; j <= i__1; ++j) { -/*< KK = II - J >*/ -#line 194 "" - kk = ii - j; -/*< IF ( XX(KK) .EQ. 1 ) GO TO 300 >*/ -#line 195 "" - if (xx[kk] == 1) { -#line 195 "" - goto L300; -#line 195 "" - } -/*< 290 CONTINUE >*/ -#line 196 "" -/* L290: */ -#line 196 "" - } -/*< RETURN >*/ -#line 197 "" - return 0; -/*< 300 R = CH >*/ -#line 198 "" -L300: -#line 198 "" - r__ = ch; -/*< CH = CH + W(KK) >*/ -#line 199 "" - ch += w[kk]; -/*< PROFIT = PROFIT - P(KK) >*/ -#line 200 "" - profit -= p[kk]; -/*< XX(KK) = 0 >*/ -#line 201 "" - xx[kk] = 0; -/*< IF ( R .LT. MIN(KK) ) GO TO 310 >*/ -#line 202 "" - if (r__ < min__[kk]) { -#line 202 "" - goto L310; -#line 202 "" - } -/*< II = KK + 1 >*/ -#line 203 "" - ii = kk + 1; -/*< GO TO 80 >*/ -#line 204 "" - goto L80; -/*< 310 NN = KK + 1 >*/ -#line 205 "" -L310: -#line 205 "" - nn = kk + 1; -/*< II = KK >*/ -#line 206 "" - ii = kk; -/* TRY TO SUBSTITUTE THE NN-TH ITEM FOR THE KK-TH. */ -/*< 320 IF ( Z .GE. PROFIT + CH*P(NN)/W(NN) ) GO TO 280 >*/ -#line 208 "" -L320: -#line 208 "" - if (*z__ >= profit + ch * p[nn] / w[nn]) { -#line 208 "" - goto L280; -#line 208 "" - } -/*< DIFF = W(NN) - W(KK) >*/ -#line 209 "" - diff = w[nn] - w[kk]; -/*< IF ( DIFF ) 370, 330, 340 >*/ -#line 210 "" - if (diff < 0) { -#line 210 "" - goto L370; -#line 210 "" - } else if (diff == 0) { -#line 210 "" - goto L330; -#line 210 "" - } else { -#line 210 "" - goto L340; -#line 210 "" - } -/*< 330 NN = NN + 1 >*/ -#line 211 "" -L330: -#line 211 "" - ++nn; -/*< GO TO 320 >*/ -#line 212 "" - goto L320; -/*< 340 IF ( DIFF .GT. R ) GO TO 330 >*/ -#line 213 "" -L340: -#line 213 "" - if (diff > r__) { -#line 213 "" - goto L330; -#line 213 "" - } -/*< IF ( Z .GE. PROFIT + P(NN) ) GO TO 330 >*/ -#line 214 "" - if (*z__ >= profit + p[nn]) { -#line 214 "" - goto L330; -#line 214 "" - } -/*< Z = PROFIT + P(NN) >*/ -#line 215 "" - *z__ = profit + p[nn]; -/*< DO 350 J=1,KK >*/ -#line 216 "" - i__1 = kk; -#line 216 "" - for (j = 1; j <= i__1; ++j) { -/*< X(J) = XX(J) >*/ -#line 217 "" - x[j] = xx[j]; -/*< 350 CONTINUE >*/ -#line 218 "" -/* L350: */ -#line 218 "" - } -/*< JJ = KK + 1 >*/ -#line 219 "" - jj = kk + 1; -/*< DO 360 J=JJ,N >*/ -#line 220 "" - i__1 = *n; -#line 220 "" - for (j = jj; j <= i__1; ++j) { -/*< X(J) = 0 >*/ -#line 221 "" - x[j] = 0; -/*< 360 CONTINUE >*/ -#line 222 "" -/* L360: */ -#line 222 "" - } -/*< X(NN) = 1 >*/ -#line 223 "" - x[nn] = 1; -/*< IF ( Z .EQ. LIM ) RETURN >*/ -#line 224 "" - if (*z__ == lim) { -#line 224 "" - return 0; -#line 224 "" - } -/*< R = R - DIFF >*/ -#line 225 "" - r__ -= diff; -/*< KK = NN >*/ -#line 226 "" - kk = nn; -/*< NN = NN + 1 >*/ -#line 227 "" - ++nn; -/*< GO TO 320 >*/ -#line 228 "" - goto L320; -/*< 370 T = R - DIFF >*/ -#line 229 "" -L370: -#line 229 "" - t = r__ - diff; -/*< IF ( T .LT. MIN(NN) ) GO TO 330 >*/ -#line 230 "" - if (t < min__[nn]) { -#line 230 "" - goto L330; -#line 230 "" - } -/*< IF ( Z .GE. PROFIT + P(NN) + T*P(NN+1)/W(NN+1)) GO TO 280 >*/ -#line 231 "" - if (*z__ >= profit + p[nn] + t * p[nn + 1] / w[nn + 1]) { -#line 231 "" - goto L280; -#line 231 "" - } -/*< CH = CH - W(NN) >*/ -#line 232 "" - ch -= w[nn]; -/*< PROFIT = PROFIT + P(NN) >*/ -#line 233 "" - profit += p[nn]; -/*< XX(NN) = 1 >*/ -#line 234 "" - xx[nn] = 1; -/*< II = NN + 1 >*/ -#line 235 "" - ii = nn + 1; -/*< WSIGN(NN) = W(NN) >*/ -#line 236 "" - wsign[nn] = w[nn]; -/*< PSIGN(NN) = P(NN) >*/ -#line 237 "" - psign[nn] = p[nn]; -/*< ZSIGN(NN) = II >*/ -#line 238 "" - zsign[nn] = ii; -/*< N1 = NN + 1 >*/ -#line 239 "" - n1 = nn + 1; -/*< DO 380 J=N1,LOLD >*/ -#line 240 "" - i__1 = lold; -#line 240 "" - for (j = n1; j <= i__1; ++j) { -/*< WSIGN(J) = 0 >*/ -#line 241 "" - wsign[j] = 0; -/*< PSIGN(J) = 0 >*/ -#line 242 "" - psign[j] = 0; -/*< ZSIGN(J) = J >*/ -#line 243 "" - zsign[j] = j; -/*< 380 CONTINUE >*/ -#line 244 "" -/* L380: */ -#line 244 "" - } -/*< LOLD = NN >*/ -#line 245 "" - lold = nn; -/*< GO TO 80 >*/ -#line 246 "" - goto L80; -/*< END >*/ -} /* mt1_ */ - -/*< SUBROUTINE CHMT1(N,P,W,C,Z,JDIM) >*/ -#if 1 /* by mao */ -static -#endif -/* Subroutine */ int chmt1_(integer *n, integer *p, integer *w, integer *c__, - integer *z__, integer *jdim) -{ - /* System generated locals */ - integer i__1; - - /* Local variables */ - static integer j; - static real r__, rr; - static integer jsw; - - -/* CHECK THE INPUT DATA. */ - -/*< INTEGER P(JDIM),W(JDIM),C,Z >*/ -/*< IF ( N .GE. 2 .AND. N .LE. JDIM - 1 ) GO TO 10 >*/ -#line 253 "" - /* Parameter adjustments */ -#line 253 "" - --w; -#line 253 "" - --p; -#line 253 "" - -#line 253 "" - /* Function Body */ -#line 253 "" - if (*n >= 2 && *n <= *jdim - 1) { -#line 253 "" - goto L10; -#line 253 "" - } -/*< Z = - 1 >*/ -#line 254 "" - *z__ = -1; -/*< RETURN >*/ -#line 255 "" - return 0; -/*< 10 IF ( C .GT. 0 ) GO TO 30 >*/ -#line 256 "" -L10: -#line 256 "" - if (*c__ > 0) { -#line 256 "" - goto L30; -#line 256 "" - } -/*< 20 Z = - 2 >*/ -#line 257 "" -L20: -#line 257 "" - *z__ = -2; -/*< RETURN >*/ -#line 258 "" - return 0; -/*< 30 JSW = 0 >*/ -#line 259 "" -L30: -#line 259 "" - jsw = 0; -/*< RR = FLOAT(P(1))/FLOAT(W(1)) >*/ -#line 260 "" - rr = (real) p[1] / (real) w[1]; -/*< DO 50 J=1,N >*/ -#line 261 "" - i__1 = *n; -#line 261 "" - for (j = 1; j <= i__1; ++j) { -/*< R = RR >*/ -#line 262 "" - r__ = rr; -/*< IF ( P(J) .LE. 0 ) GO TO 20 >*/ -#line 263 "" - if (p[j] <= 0) { -#line 263 "" - goto L20; -#line 263 "" - } -/*< IF ( W(J) .LE. 0 ) GO TO 20 >*/ -#line 264 "" - if (w[j] <= 0) { -#line 264 "" - goto L20; -#line 264 "" - } -/*< JSW = JSW + W(J) >*/ -#line 265 "" - jsw += w[j]; -/*< IF ( W(J) .LE. C ) GO TO 40 >*/ -#line 266 "" - if (w[j] <= *c__) { -#line 266 "" - goto L40; -#line 266 "" - } -/*< Z = - 3 >*/ -#line 267 "" - *z__ = -3; -/*< RETURN >*/ -#line 268 "" - return 0; -/*< 40 RR = FLOAT(P(J))/FLOAT(W(J)) >*/ -#line 269 "" -L40: -#line 269 "" - rr = (real) p[j] / (real) w[j]; -/*< IF ( RR .LE. R ) GO TO 50 >*/ -#line 270 "" - if (rr <= r__) { -#line 270 "" - goto L50; -#line 270 "" - } -/*< Z = - 5 >*/ -#line 271 "" - *z__ = -5; -/*< RETURN >*/ -#line 272 "" - return 0; -/*< 50 CONTINUE >*/ -#line 273 "" -L50: -#line 273 "" - ; -#line 273 "" - } -/*< IF ( JSW .GT. C ) RETURN >*/ -#line 274 "" - if (jsw > *c__) { -#line 274 "" - return 0; -#line 274 "" - } -/*< Z = - 4 >*/ -#line 275 "" - *z__ = -4; -/*< RETURN >*/ -#line 276 "" - return 0; -/*< END >*/ -} /* chmt1_ */ - -#if 1 /* by mao */ -int mt1(int n, int p[], int w[], int c, int x[], int jck, int xx[], - int min[], int psign[], int wsign[], int zsign[]) -{ /* solve 0-1 knapsack problem */ - int z, jdim = n+1, j, s1, s2; - mt1_(&n, &p[1], &w[1], &c, &z, &x[1], &jdim, &jck, &xx[1], - &min[1], &psign[1], &wsign[1], &zsign[1]); - /* check solution found */ - s1 = s2 = 0; - for (j = 1; j <= n; j++) - { xassert(x[j] == 0 || x[j] == 1); - if (x[j]) - s1 += p[j], s2 += w[j]; - } - xassert(s1 == z); - xassert(s2 <= c); - return z; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mt1.f b/code/3rd_glpk/misc/mt1.f deleted file mode 100644 index 82cc4a1b..00000000 --- a/code/3rd_glpk/misc/mt1.f +++ /dev/null @@ -1,277 +0,0 @@ - SUBROUTINE MT1(N,P,W,C,Z,X,JDIM,JCK,XX,MIN,PSIGN,WSIGN,ZSIGN) -C -C THIS SUBROUTINE SOLVES THE 0-1 SINGLE KNAPSACK PROBLEM -C -C MAXIMIZE Z = P(1)*X(1) + ... + P(N)*X(N) -C -C SUBJECT TO: W(1)*X(1) + ... + W(N)*X(N) .LE. C , -C X(J) = 0 OR 1 FOR J=1,...,N. -C -C THE PROGRAM IS INCLUDED IN THE VOLUME -C S. MARTELLO, P. TOTH, "KNAPSACK PROBLEMS: ALGORITHMS -C AND COMPUTER IMPLEMENTATIONS", JOHN WILEY, 1990 -C AND IMPLEMENTS THE BRANCH-AND-BOUND ALGORITHM DESCRIBED IN -C SECTION 2.5.2 . -C THE PROGRAM DERIVES FROM AN EARLIER CODE PRESENTED IN -C S. MARTELLO, P. TOTH, "ALGORITHM FOR THE SOLUTION OF THE 0-1 SINGLE -C KNAPSACK PROBLEM", COMPUTING, 1978. -C -C THE INPUT PROBLEM MUST SATISFY THE CONDITIONS -C -C 1) 2 .LE. N .LE. JDIM - 1 ; -C 2) P(J), W(J), C POSITIVE INTEGERS; -C 3) MAX (W(J)) .LE. C ; -C 4) W(1) + ... + W(N) .GT. C ; -C 5) P(J)/W(J) .GE. P(J+1)/W(J+1) FOR J=1,...,N-1. -C -C MT1 CALLS 1 PROCEDURE: CHMT1. -C -C THE PROGRAM IS COMPLETELY SELF-CONTAINED AND COMMUNICATION TO IT IS -C ACHIEVED SOLELY THROUGH THE PARAMETER LIST OF MT1. -C NO MACHINE-DEPENDENT CONSTANT IS USED. -C THE PROGRAM IS WRITTEN IN 1967 AMERICAN NATIONAL STANDARD FORTRAN -C AND IS ACCEPTED BY THE PFORT VERIFIER (PFORT IS THE PORTABLE -C SUBSET OF ANSI DEFINED BY THE ASSOCIATION FOR COMPUTING MACHINERY). -C THE PROGRAM HAS BEEN TESTED ON A DIGITAL VAX 11/780 AND AN H.P. -C 9000/840. -C -C MT1 NEEDS 8 ARRAYS ( P , W , X , XX , MIN , PSIGN , WSIGN -C AND ZSIGN ) OF LENGTH AT LEAST N + 1 . -C -C MEANING OF THE INPUT PARAMETERS: -C N = NUMBER OF ITEMS; -C P(J) = PROFIT OF ITEM J (J=1,...,N); -C W(J) = WEIGHT OF ITEM J (J=1,...,N); -C C = CAPACITY OF THE KNAPSACK; -C JDIM = DIMENSION OF THE 8 ARRAYS; -C JCK = 1 IF CHECK ON THE INPUT DATA IS DESIRED, -C = 0 OTHERWISE. -C -C MEANING OF THE OUTPUT PARAMETERS: -C Z = VALUE OF THE OPTIMAL SOLUTION IF Z .GT. 0 , -C = ERROR IN THE INPUT DATA (WHEN JCK=1) IF Z .LT. 0 : CONDI- -C TION - Z IS VIOLATED; -C X(J) = 1 IF ITEM J IS IN THE OPTIMAL SOLUTION, -C = 0 OTHERWISE. -C -C ARRAYS XX, MIN, PSIGN, WSIGN AND ZSIGN ARE DUMMY. -C -C ALL THE PARAMETERS ARE INTEGER. ON RETURN OF MT1 ALL THE INPUT -C PARAMETERS ARE UNCHANGED. -C - INTEGER P(JDIM),W(JDIM),X(JDIM),C,Z - INTEGER XX(JDIM),MIN(JDIM),PSIGN(JDIM),WSIGN(JDIM),ZSIGN(JDIM) - INTEGER CH,CHS,DIFF,PROFIT,R,T - Z = 0 - IF ( JCK .EQ. 1 ) CALL CHMT1(N,P,W,C,Z,JDIM) - IF ( Z .LT. 0 ) RETURN -C INITIALIZE. - CH = C - IP = 0 - CHS = CH - DO 10 LL=1,N - IF ( W(LL) .GT. CHS ) GO TO 20 - IP = IP + P(LL) - CHS = CHS - W(LL) - 10 CONTINUE - 20 LL = LL - 1 - IF ( CHS .EQ. 0 ) GO TO 50 - P(N+1) = 0 - W(N+1) = CH + 1 - LIM = IP + CHS*P(LL+2)/W(LL+2) - A = W(LL+1) - CHS - B = IP + P(LL+1) - LIM1 = B - A*FLOAT(P(LL))/FLOAT(W(LL)) - IF ( LIM1 .GT. LIM ) LIM = LIM1 - MINK = CH + 1 - MIN(N) = MINK - DO 30 J=2,N - KK = N + 2 - J - IF ( W(KK) .LT. MINK ) MINK = W(KK) - MIN(KK-1) = MINK - 30 CONTINUE - DO 40 J=1,N - XX(J) = 0 - 40 CONTINUE - Z = 0 - PROFIT = 0 - LOLD = N - II = 1 - GO TO 170 - 50 Z = IP - DO 60 J=1,LL - X(J) = 1 - 60 CONTINUE - NN = LL + 1 - DO 70 J=NN,N - X(J) = 0 - 70 CONTINUE - RETURN -C TRY TO INSERT THE II-TH ITEM INTO THE CURRENT SOLUTION. - 80 IF ( W(II) .LE. CH ) GO TO 90 - II1 = II + 1 - IF ( Z .GE. CH*P(II1)/W(II1) + PROFIT ) GO TO 280 - II = II1 - GO TO 80 -C BUILD A NEW CURRENT SOLUTION. - 90 IP = PSIGN(II) - CHS = CH - WSIGN(II) - IN = ZSIGN(II) - DO 100 LL=IN,N - IF ( W(LL) .GT. CHS ) GO TO 160 - IP = IP + P(LL) - CHS = CHS - W(LL) - 100 CONTINUE - LL = N - 110 IF ( Z .GE. IP + PROFIT ) GO TO 280 - Z = IP + PROFIT - NN = II - 1 - DO 120 J=1,NN - X(J) = XX(J) - 120 CONTINUE - DO 130 J=II,LL - X(J) = 1 - 130 CONTINUE - IF ( LL .EQ. N ) GO TO 150 - NN = LL + 1 - DO 140 J=NN,N - X(J) = 0 - 140 CONTINUE - 150 IF ( Z .NE. LIM ) GO TO 280 - RETURN - 160 IU = CHS*P(LL)/W(LL) - LL = LL - 1 - IF ( IU .EQ. 0 ) GO TO 110 - IF ( Z .GE. PROFIT + IP + IU ) GO TO 280 -C SAVE THE CURRENT SOLUTION. - 170 WSIGN(II) = CH - CHS - PSIGN(II) = IP - ZSIGN(II) = LL + 1 - XX(II) = 1 - NN = LL - 1 - IF ( NN .LT. II) GO TO 190 - DO 180 J=II,NN - WSIGN(J+1) = WSIGN(J) - W(J) - PSIGN(J+1) = PSIGN(J) - P(J) - ZSIGN(J+1) = LL + 1 - XX(J+1) = 1 - 180 CONTINUE - 190 J1 = LL + 1 - DO 200 J=J1,LOLD - WSIGN(J) = 0 - PSIGN(J) = 0 - ZSIGN(J) = J - 200 CONTINUE - LOLD = LL - CH = CHS - PROFIT = PROFIT + IP - IF ( LL - (N - 2) ) 240, 220, 210 - 210 II = N - GO TO 250 - 220 IF ( CH .LT. W(N) ) GO TO 230 - CH = CH - W(N) - PROFIT = PROFIT + P(N) - XX(N) = 1 - 230 II = N - 1 - GO TO 250 - 240 II = LL + 2 - IF ( CH .GE. MIN(II-1) ) GO TO 80 -C SAVE THE CURRENT OPTIMAL SOLUTION. - 250 IF ( Z .GE. PROFIT ) GO TO 270 - Z = PROFIT - DO 260 J=1,N - X(J) = XX(J) - 260 CONTINUE - IF ( Z .EQ. LIM ) RETURN - 270 IF ( XX(N) .EQ. 0 ) GO TO 280 - XX(N) = 0 - CH = CH + W(N) - PROFIT = PROFIT - P(N) -C BACKTRACK. - 280 NN = II - 1 - IF ( NN .EQ. 0 ) RETURN - DO 290 J=1,NN - KK = II - J - IF ( XX(KK) .EQ. 1 ) GO TO 300 - 290 CONTINUE - RETURN - 300 R = CH - CH = CH + W(KK) - PROFIT = PROFIT - P(KK) - XX(KK) = 0 - IF ( R .LT. MIN(KK) ) GO TO 310 - II = KK + 1 - GO TO 80 - 310 NN = KK + 1 - II = KK -C TRY TO SUBSTITUTE THE NN-TH ITEM FOR THE KK-TH. - 320 IF ( Z .GE. PROFIT + CH*P(NN)/W(NN) ) GO TO 280 - DIFF = W(NN) - W(KK) - IF ( DIFF ) 370, 330, 340 - 330 NN = NN + 1 - GO TO 320 - 340 IF ( DIFF .GT. R ) GO TO 330 - IF ( Z .GE. PROFIT + P(NN) ) GO TO 330 - Z = PROFIT + P(NN) - DO 350 J=1,KK - X(J) = XX(J) - 350 CONTINUE - JJ = KK + 1 - DO 360 J=JJ,N - X(J) = 0 - 360 CONTINUE - X(NN) = 1 - IF ( Z .EQ. LIM ) RETURN - R = R - DIFF - KK = NN - NN = NN + 1 - GO TO 320 - 370 T = R - DIFF - IF ( T .LT. MIN(NN) ) GO TO 330 - IF ( Z .GE. PROFIT + P(NN) + T*P(NN+1)/W(NN+1)) GO TO 280 - CH = CH - W(NN) - PROFIT = PROFIT + P(NN) - XX(NN) = 1 - II = NN + 1 - WSIGN(NN) = W(NN) - PSIGN(NN) = P(NN) - ZSIGN(NN) = II - N1 = NN + 1 - DO 380 J=N1,LOLD - WSIGN(J) = 0 - PSIGN(J) = 0 - ZSIGN(J) = J - 380 CONTINUE - LOLD = NN - GO TO 80 - END - SUBROUTINE CHMT1(N,P,W,C,Z,JDIM) -C -C CHECK THE INPUT DATA. -C - INTEGER P(JDIM),W(JDIM),C,Z - IF ( N .GE. 2 .AND. N .LE. JDIM - 1 ) GO TO 10 - Z = - 1 - RETURN - 10 IF ( C .GT. 0 ) GO TO 30 - 20 Z = - 2 - RETURN - 30 JSW = 0 - RR = FLOAT(P(1))/FLOAT(W(1)) - DO 50 J=1,N - R = RR - IF ( P(J) .LE. 0 ) GO TO 20 - IF ( W(J) .LE. 0 ) GO TO 20 - JSW = JSW + W(J) - IF ( W(J) .LE. C ) GO TO 40 - Z = - 3 - RETURN - 40 RR = FLOAT(P(J))/FLOAT(W(J)) - IF ( RR .LE. R ) GO TO 50 - Z = - 5 - RETURN - 50 CONTINUE - IF ( JSW .GT. C ) RETURN - Z = - 4 - RETURN - END diff --git a/code/3rd_glpk/misc/mt1.h b/code/3rd_glpk/misc/mt1.h deleted file mode 100644 index cceebba9..00000000 --- a/code/3rd_glpk/misc/mt1.h +++ /dev/null @@ -1,34 +0,0 @@ -/* mt1.h (0-1 knapsack problem; Martello & Toth algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2017-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MT1_H -#define MT1_H - -#define mt1 _glp_mt1 -int mt1(int n, int p[], int w[], int c, int x[], int jck, int xx[], - int min[], int psign[], int wsign[], int zsign[]); -/* solve 0-1 single knapsack problem */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mygmp.c b/code/3rd_glpk/misc/mygmp.c deleted file mode 100644 index 89d053ae..00000000 --- a/code/3rd_glpk/misc/mygmp.c +++ /dev/null @@ -1,1162 +0,0 @@ -/* mygmp.c (integer and rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mygmp.h" - -#ifdef HAVE_GMP /* use GNU MP library */ - -/* nothing is needed */ - -#else /* use GLPK MP module */ - -#include "bignum.h" -#include "dmp.h" -#include "env.h" - -#define gmp_pool env->gmp_pool -#define gmp_size env->gmp_size -#define gmp_work env->gmp_work - -void *gmp_get_atom(int size) -{ ENV *env = get_env_ptr(); - if (gmp_pool == NULL) - gmp_pool = dmp_create_pool(); - return dmp_get_atom(gmp_pool, size); -} - -void gmp_free_atom(void *ptr, int size) -{ ENV *env = get_env_ptr(); - xassert(gmp_pool != NULL); - dmp_free_atom(gmp_pool, ptr, size); - return; -} - -int gmp_pool_count(void) -{ ENV *env = get_env_ptr(); - if (gmp_pool == NULL) - return 0; - else - return dmp_in_use(gmp_pool); -} - -unsigned short *gmp_get_work(int size) -{ ENV *env = get_env_ptr(); - xassert(size > 0); - if (gmp_size < size) - { if (gmp_size == 0) - { xassert(gmp_work == NULL); - gmp_size = 100; - } - else - { xassert(gmp_work != NULL); - xfree(gmp_work); - } - while (gmp_size < size) - gmp_size += gmp_size; - gmp_work = xcalloc(gmp_size, sizeof(unsigned short)); - } - return gmp_work; -} - -void gmp_free_mem(void) -{ ENV *env = get_env_ptr(); - if (gmp_pool != NULL) - dmp_delete_pool(gmp_pool); - if (gmp_work != NULL) - xfree(gmp_work); - gmp_pool = NULL; - gmp_size = 0; - gmp_work = NULL; - return; -} - -/*--------------------------------------------------------------------*/ - -mpz_t _mpz_init(void) -{ /* initialize x and set its value to 0 */ - mpz_t x; - x = gmp_get_atom(sizeof(struct mpz)); - x->val = 0; - x->ptr = NULL; - return x; -} - -void mpz_clear(mpz_t x) -{ /* free the space occupied by x */ - mpz_set_si(x, 0); - xassert(x->ptr == NULL); - /* free the number descriptor */ - gmp_free_atom(x, sizeof(struct mpz)); - return; -} - -void mpz_set(mpz_t z, mpz_t x) -{ /* set the value of z from x */ - struct mpz_seg *e, *ee, *es; - if (z != x) - { mpz_set_si(z, 0); - z->val = x->val; - xassert(z->ptr == NULL); - for (e = x->ptr, es = NULL; e != NULL; e = e->next) - { ee = gmp_get_atom(sizeof(struct mpz_seg)); - memcpy(ee->d, e->d, 12); - ee->next = NULL; - if (z->ptr == NULL) - z->ptr = ee; - else - es->next = ee; - es = ee; - } - } - return; -} - -void mpz_set_si(mpz_t x, int val) -{ /* set the value of x to val */ - struct mpz_seg *e; - /* free existing segments, if any */ - while (x->ptr != NULL) - { e = x->ptr; - x->ptr = e->next; - gmp_free_atom(e, sizeof(struct mpz_seg)); - } - /* assign new value */ - if (val == 0x80000000) - { /* long format is needed */ - x->val = -1; - x->ptr = e = gmp_get_atom(sizeof(struct mpz_seg)); - memset(e->d, 0, 12); - e->d[1] = 0x8000; - e->next = NULL; - } - else - { /* short format is enough */ - x->val = val; - } - return; -} - -double mpz_get_d(mpz_t x) -{ /* convert x to a double, truncating if necessary */ - struct mpz_seg *e; - int j; - double val, deg; - if (x->ptr == NULL) - val = (double)x->val; - else - { xassert(x->val != 0); - val = 0.0; - deg = 1.0; - for (e = x->ptr; e != NULL; e = e->next) - { for (j = 0; j <= 5; j++) - { val += deg * (double)((int)e->d[j]); - deg *= 65536.0; - } - } - if (x->val < 0) - val = - val; - } - return val; -} - -double mpz_get_d_2exp(int *exp, mpz_t x) -{ /* convert x to a double, truncating if necessary (i.e. rounding - * towards zero), and returning the exponent separately; - * the return value is in the range 0.5 <= |d| < 1 and the - * exponent is stored to *exp; d*2^exp is the (truncated) x value; - * if x is zero, the return is 0.0 and 0 is stored to *exp; - * this is similar to the standard C frexp function */ - struct mpz_seg *e; - int j, n, n1; - double val; - if (x->ptr == NULL) - val = (double)x->val, n = 0; - else - { xassert(x->val != 0); - val = 0.0, n = 0; - for (e = x->ptr; e != NULL; e = e->next) - { for (j = 0; j <= 5; j++) - { val += (double)((int)e->d[j]); - val /= 65536.0, n += 16; - } - } - if (x->val < 0) - val = - val; - } - val = frexp(val, &n1); - *exp = n + n1; - return val; -} - -void mpz_swap(mpz_t x, mpz_t y) -{ /* swap the values x and y efficiently */ - int val; - void *ptr; - val = x->val, ptr = x->ptr; - x->val = y->val, x->ptr = y->ptr; - y->val = val, y->ptr = ptr; - return; -} - -static void normalize(mpz_t x) -{ /* normalize integer x that includes removing non-significant - * (leading) zeros and converting to short format, if possible */ - struct mpz_seg *es, *e; - /* if the integer is in short format, it remains unchanged */ - if (x->ptr == NULL) - { xassert(x->val != 0x80000000); - goto done; - } - xassert(x->val == +1 || x->val == -1); - /* find the last (most significant) non-zero segment */ - es = NULL; - for (e = x->ptr; e != NULL; e = e->next) - { if (e->d[0] || e->d[1] || e->d[2] || - e->d[3] || e->d[4] || e->d[5]) - es = e; - } - /* if all segments contain zeros, the integer is zero */ - if (es == NULL) - { mpz_set_si(x, 0); - goto done; - } - /* remove non-significant (leading) zero segments */ - while (es->next != NULL) - { e = es->next; - es->next = e->next; - gmp_free_atom(e, sizeof(struct mpz_seg)); - } - /* convert the integer to short format, if possible */ - e = x->ptr; - if (e->next == NULL && e->d[1] <= 0x7FFF && - !e->d[2] && !e->d[3] && !e->d[4] && !e->d[5]) - { int val; - val = (int)e->d[0] + ((int)e->d[1] << 16); - if (x->val < 0) - val = - val; - mpz_set_si(x, val); - } -done: return; -} - -void mpz_add(mpz_t z, mpz_t x, mpz_t y) -{ /* set z to x + y */ - static struct mpz_seg zero = { { 0, 0, 0, 0, 0, 0 }, NULL }; - struct mpz_seg dumx, dumy, *ex, *ey, *ez, *es, *ee; - int k, sx, sy, sz; - unsigned int t; - /* if [x] = 0 then [z] = [y] */ - if (x->val == 0) - { xassert(x->ptr == NULL); - mpz_set(z, y); - goto done; - } - /* if [y] = 0 then [z] = [x] */ - if (y->val == 0) - { xassert(y->ptr == NULL); - mpz_set(z, x); - goto done; - } - /* special case when both [x] and [y] are in short format */ - if (x->ptr == NULL && y->ptr == NULL) - { int xval = x->val, yval = y->val, zval = x->val + y->val; - xassert(xval != 0x80000000 && yval != 0x80000000); - if (!(xval > 0 && yval > 0 && zval <= 0 || - xval < 0 && yval < 0 && zval >= 0)) - { mpz_set_si(z, zval); - goto done; - } - } - /* convert [x] to long format, if necessary */ - if (x->ptr == NULL) - { xassert(x->val != 0x80000000); - if (x->val >= 0) - { sx = +1; - t = (unsigned int)(+ x->val); - } - else - { sx = -1; - t = (unsigned int)(- x->val); - } - ex = &dumx; - ex->d[0] = (unsigned short)t; - ex->d[1] = (unsigned short)(t >> 16); - ex->d[2] = ex->d[3] = ex->d[4] = ex->d[5] = 0; - ex->next = NULL; - } - else - { sx = x->val; - xassert(sx == +1 || sx == -1); - ex = x->ptr; - } - /* convert [y] to long format, if necessary */ - if (y->ptr == NULL) - { xassert(y->val != 0x80000000); - if (y->val >= 0) - { sy = +1; - t = (unsigned int)(+ y->val); - } - else - { sy = -1; - t = (unsigned int)(- y->val); - } - ey = &dumy; - ey->d[0] = (unsigned short)t; - ey->d[1] = (unsigned short)(t >> 16); - ey->d[2] = ey->d[3] = ey->d[4] = ey->d[5] = 0; - ey->next = NULL; - } - else - { sy = y->val; - xassert(sy == +1 || sy == -1); - ey = y->ptr; - } - /* main fragment */ - sz = sx; - ez = es = NULL; - if (sx > 0 && sy > 0 || sx < 0 && sy < 0) - { /* [x] and [y] have identical signs -- addition */ - t = 0; - for (; ex || ey; ex = ex->next, ey = ey->next) - { if (ex == NULL) - ex = &zero; - if (ey == NULL) - ey = &zero; - ee = gmp_get_atom(sizeof(struct mpz_seg)); - for (k = 0; k <= 5; k++) - { t += (unsigned int)ex->d[k]; - t += (unsigned int)ey->d[k]; - ee->d[k] = (unsigned short)t; - t >>= 16; - } - ee->next = NULL; - if (ez == NULL) - ez = ee; - else - es->next = ee; - es = ee; - } - if (t) - { /* overflow -- one extra digit is needed */ - ee = gmp_get_atom(sizeof(struct mpz_seg)); - ee->d[0] = 1; - ee->d[1] = ee->d[2] = ee->d[3] = ee->d[4] = ee->d[5] = 0; - ee->next = NULL; - xassert(es != NULL); - es->next = ee; - } - } - else - { /* [x] and [y] have different signs -- subtraction */ - t = 1; - for (; ex || ey; ex = ex->next, ey = ey->next) - { if (ex == NULL) - ex = &zero; - if (ey == NULL) - ey = &zero; - ee = gmp_get_atom(sizeof(struct mpz_seg)); - for (k = 0; k <= 5; k++) - { t += (unsigned int)ex->d[k]; - t += (0xFFFF - (unsigned int)ey->d[k]); - ee->d[k] = (unsigned short)t; - t >>= 16; - } - ee->next = NULL; - if (ez == NULL) - ez = ee; - else - es->next = ee; - es = ee; - } - if (!t) - { /* |[x]| < |[y]| -- result in complement coding */ - sz = - sz; - t = 1; - for (ee = ez; ee != NULL; ee = ee->next) - { for (k = 0; k <= 5; k++) - { t += (0xFFFF - (unsigned int)ee->d[k]); - ee->d[k] = (unsigned short)t; - t >>= 16; - } - } - } - } - /* contruct and normalize result */ - mpz_set_si(z, 0); - z->val = sz; - z->ptr = ez; - normalize(z); -done: return; -} - -void mpz_sub(mpz_t z, mpz_t x, mpz_t y) -{ /* set z to x - y */ - if (x == y) - mpz_set_si(z, 0); - else - { y->val = - y->val; - mpz_add(z, x, y); - if (y != z) - y->val = - y->val; - } - return; -} - -void mpz_mul(mpz_t z, mpz_t x, mpz_t y) -{ /* set z to x * y */ - struct mpz_seg dumx, dumy, *ex, *ey, *es, *e; - int sx, sy, k, nx, ny, n; - unsigned int t; - unsigned short *work, *wx, *wy; - /* if [x] = 0 then [z] = 0 */ - if (x->val == 0) - { xassert(x->ptr == NULL); - mpz_set_si(z, 0); - goto done; - } - /* if [y] = 0 then [z] = 0 */ - if (y->val == 0) - { xassert(y->ptr == NULL); - mpz_set_si(z, 0); - goto done; - } - /* special case when both [x] and [y] are in short format */ - if (x->ptr == NULL && y->ptr == NULL) - { int xval = x->val, yval = y->val, sz = +1; - xassert(xval != 0x80000000 && yval != 0x80000000); - if (xval < 0) - xval = - xval, sz = - sz; - if (yval < 0) - yval = - yval, sz = - sz; - if (xval <= 0x7FFFFFFF / yval) - { mpz_set_si(z, sz * (xval * yval)); - goto done; - } - } - /* convert [x] to long format, if necessary */ - if (x->ptr == NULL) - { xassert(x->val != 0x80000000); - if (x->val >= 0) - { sx = +1; - t = (unsigned int)(+ x->val); - } - else - { sx = -1; - t = (unsigned int)(- x->val); - } - ex = &dumx; - ex->d[0] = (unsigned short)t; - ex->d[1] = (unsigned short)(t >> 16); - ex->d[2] = ex->d[3] = ex->d[4] = ex->d[5] = 0; - ex->next = NULL; - } - else - { sx = x->val; - xassert(sx == +1 || sx == -1); - ex = x->ptr; - } - /* convert [y] to long format, if necessary */ - if (y->ptr == NULL) - { xassert(y->val != 0x80000000); - if (y->val >= 0) - { sy = +1; - t = (unsigned int)(+ y->val); - } - else - { sy = -1; - t = (unsigned int)(- y->val); - } - ey = &dumy; - ey->d[0] = (unsigned short)t; - ey->d[1] = (unsigned short)(t >> 16); - ey->d[2] = ey->d[3] = ey->d[4] = ey->d[5] = 0; - ey->next = NULL; - } - else - { sy = y->val; - xassert(sy == +1 || sy == -1); - ey = y->ptr; - } - /* determine the number of digits of [x] */ - nx = n = 0; - for (e = ex; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++) - { n++; - if (e->d[k]) - nx = n; - } - } - xassert(nx > 0); - /* determine the number of digits of [y] */ - ny = n = 0; - for (e = ey; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++) - { n++; - if (e->d[k]) - ny = n; - } - } - xassert(ny > 0); - /* we need working array containing at least nx+ny+ny places */ - work = gmp_get_work(nx+ny+ny); - /* load digits of [x] */ - wx = &work[0]; - for (n = 0; n < nx; n++) - wx[ny+n] = 0; - for (n = 0, e = ex; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++, n++) - { if (e->d[k]) - wx[ny+n] = e->d[k]; - } - } - /* load digits of [y] */ - wy = &work[nx+ny]; - for (n = 0; n < ny; n++) wy[n] = 0; - for (n = 0, e = ey; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++, n++) - { if (e->d[k]) - wy[n] = e->d[k]; - } - } - /* compute [x] * [y] */ - bigmul(nx, ny, wx, wy); - /* construct and normalize result */ - mpz_set_si(z, 0); - z->val = sx * sy; - es = NULL; - k = 6; - for (n = 0; n < nx+ny; n++) - { if (k > 5) - { e = gmp_get_atom(sizeof(struct mpz_seg)); - e->d[0] = e->d[1] = e->d[2] = 0; - e->d[3] = e->d[4] = e->d[5] = 0; - e->next = NULL; - if (z->ptr == NULL) - z->ptr = e; - else - es->next = e; - es = e; - k = 0; - } - es->d[k++] = wx[n]; - } - normalize(z); -done: return; -} - -void mpz_neg(mpz_t z, mpz_t x) -{ /* set z to 0 - x */ - mpz_set(z, x); - z->val = - z->val; - return; -} - -void mpz_abs(mpz_t z, mpz_t x) -{ /* set z to the absolute value of x */ - mpz_set(z, x); - if (z->val < 0) - z->val = - z->val; - return; -} - -void mpz_div(mpz_t q, mpz_t r, mpz_t x, mpz_t y) -{ /* divide x by y, forming quotient q and/or remainder r - * if q = NULL then quotient is not stored; if r = NULL then - * remainder is not stored - * the sign of quotient is determined as in algebra while the - * sign of remainder is the same as the sign of dividend: - * +26 : +7 = +3, remainder is +5 - * -26 : +7 = -3, remainder is -5 - * +26 : -7 = -3, remainder is +5 - * -26 : -7 = +3, remainder is -5 */ - struct mpz_seg dumx, dumy, *ex, *ey, *es, *e; - int sx, sy, k, nx, ny, n; - unsigned int t; - unsigned short *work, *wx, *wy; - /* divide by zero is not allowed */ - if (y->val == 0) - { xassert(y->ptr == NULL); - xerror("mpz_div: divide by zero not allowed\n"); - } - /* if [x] = 0 then [q] = [r] = 0 */ - if (x->val == 0) - { xassert(x->ptr == NULL); - if (q != NULL) - mpz_set_si(q, 0); - if (r != NULL) - mpz_set_si(r, 0); - goto done; - } - /* special case when both [x] and [y] are in short format */ - if (x->ptr == NULL && y->ptr == NULL) - { int xval = x->val, yval = y->val; - xassert(xval != 0x80000000 && yval != 0x80000000); - /* FIXME: use div function */ - if (q != NULL) - mpz_set_si(q, xval / yval); - if (r != NULL) - mpz_set_si(r, xval % yval); - goto done; - } - /* convert [x] to long format, if necessary */ - if (x->ptr == NULL) - { xassert(x->val != 0x80000000); - if (x->val >= 0) - { sx = +1; - t = (unsigned int)(+ x->val); - } - else - { sx = -1; - t = (unsigned int)(- x->val); - } - ex = &dumx; - ex->d[0] = (unsigned short)t; - ex->d[1] = (unsigned short)(t >> 16); - ex->d[2] = ex->d[3] = ex->d[4] = ex->d[5] = 0; - ex->next = NULL; - } - else - { sx = x->val; - xassert(sx == +1 || sx == -1); - ex = x->ptr; - } - /* convert [y] to long format, if necessary */ - if (y->ptr == NULL) - { xassert(y->val != 0x80000000); - if (y->val >= 0) - { sy = +1; - t = (unsigned int)(+ y->val); - } - else - { sy = -1; - t = (unsigned int)(- y->val); - } - ey = &dumy; - ey->d[0] = (unsigned short)t; - ey->d[1] = (unsigned short)(t >> 16); - ey->d[2] = ey->d[3] = ey->d[4] = ey->d[5] = 0; - ey->next = NULL; - } - else - { sy = y->val; - xassert(sy == +1 || sy == -1); - ey = y->ptr; - } - /* determine the number of digits of [x] */ - nx = n = 0; - for (e = ex; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++) - { n++; - if (e->d[k]) - nx = n; - } - } - xassert(nx > 0); - /* determine the number of digits of [y] */ - ny = n = 0; - for (e = ey; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++) - { n++; - if (e->d[k]) - ny = n; - } - } - xassert(ny > 0); - /* if nx < ny then [q] = 0 and [r] = [x] */ - if (nx < ny) - { if (r != NULL) - mpz_set(r, x); - if (q != NULL) - mpz_set_si(q, 0); - goto done; - } - /* we need working array containing at least nx+ny+1 places */ - work = gmp_get_work(nx+ny+1); - /* load digits of [x] */ - wx = &work[0]; - for (n = 0; n < nx; n++) - wx[n] = 0; - for (n = 0, e = ex; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++, n++) - if (e->d[k]) wx[n] = e->d[k]; - } - /* load digits of [y] */ - wy = &work[nx+1]; - for (n = 0; n < ny; n++) - wy[n] = 0; - for (n = 0, e = ey; e != NULL; e = e->next) - { for (k = 0; k <= 5; k++, n++) - if (e->d[k]) wy[n] = e->d[k]; - } - /* compute quotient and remainder */ - xassert(wy[ny-1] != 0); - bigdiv(nx-ny, ny, wx, wy); - /* construct and normalize quotient */ - if (q != NULL) - { mpz_set_si(q, 0); - q->val = sx * sy; - es = NULL; - k = 6; - for (n = ny; n <= nx; n++) - { if (k > 5) - { e = gmp_get_atom(sizeof(struct mpz_seg)); - e->d[0] = e->d[1] = e->d[2] = 0; - e->d[3] = e->d[4] = e->d[5] = 0; - e->next = NULL; - if (q->ptr == NULL) - q->ptr = e; - else - es->next = e; - es = e; - k = 0; - } - es->d[k++] = wx[n]; - } - normalize(q); - } - /* construct and normalize remainder */ - if (r != NULL) - { mpz_set_si(r, 0); - r->val = sx; - es = NULL; - k = 6; - for (n = 0; n < ny; n++) - { if (k > 5) - { e = gmp_get_atom(sizeof(struct mpz_seg)); - e->d[0] = e->d[1] = e->d[2] = 0; - e->d[3] = e->d[4] = e->d[5] = 0; - e->next = NULL; - if (r->ptr == NULL) - r->ptr = e; - else - es->next = e; - es = e; - k = 0; - } - es->d[k++] = wx[n]; - } - normalize(r); - } -done: return; -} - -void mpz_gcd(mpz_t z, mpz_t x, mpz_t y) -{ /* set z to the greatest common divisor of x and y */ - /* in case of arbitrary integers GCD(x, y) = GCD(|x|, |y|), and, - * in particular, GCD(0, 0) = 0 */ - mpz_t u, v, r; - mpz_init(u); - mpz_init(v); - mpz_init(r); - mpz_abs(u, x); - mpz_abs(v, y); - while (mpz_sgn(v)) - { mpz_div(NULL, r, u, v); - mpz_set(u, v); - mpz_set(v, r); - } - mpz_set(z, u); - mpz_clear(u); - mpz_clear(v); - mpz_clear(r); - return; -} - -int mpz_cmp(mpz_t x, mpz_t y) -{ /* compare x and y; return a positive value if x > y, zero if - * x = y, or a nefative value if x < y */ - static struct mpz_seg zero = { { 0, 0, 0, 0, 0, 0 }, NULL }; - struct mpz_seg dumx, dumy, *ex, *ey; - int cc, sx, sy, k; - unsigned int t; - if (x == y) - { cc = 0; - goto done; - } - /* special case when both [x] and [y] are in short format */ - if (x->ptr == NULL && y->ptr == NULL) - { int xval = x->val, yval = y->val; - xassert(xval != 0x80000000 && yval != 0x80000000); - cc = (xval > yval ? +1 : xval < yval ? -1 : 0); - goto done; - } - /* special case when [x] and [y] have different signs */ - if (x->val > 0 && y->val <= 0 || x->val == 0 && y->val < 0) - { cc = +1; - goto done; - } - if (x->val < 0 && y->val >= 0 || x->val == 0 && y->val > 0) - { cc = -1; - goto done; - } - /* convert [x] to long format, if necessary */ - if (x->ptr == NULL) - { xassert(x->val != 0x80000000); - if (x->val >= 0) - { sx = +1; - t = (unsigned int)(+ x->val); - } - else - { sx = -1; - t = (unsigned int)(- x->val); - } - ex = &dumx; - ex->d[0] = (unsigned short)t; - ex->d[1] = (unsigned short)(t >> 16); - ex->d[2] = ex->d[3] = ex->d[4] = ex->d[5] = 0; - ex->next = NULL; - } - else - { sx = x->val; - xassert(sx == +1 || sx == -1); - ex = x->ptr; - } - /* convert [y] to long format, if necessary */ - if (y->ptr == NULL) - { xassert(y->val != 0x80000000); - if (y->val >= 0) - { sy = +1; - t = (unsigned int)(+ y->val); - } - else - { sy = -1; - t = (unsigned int)(- y->val); - } - ey = &dumy; - ey->d[0] = (unsigned short)t; - ey->d[1] = (unsigned short)(t >> 16); - ey->d[2] = ey->d[3] = ey->d[4] = ey->d[5] = 0; - ey->next = NULL; - } - else - { sy = y->val; - xassert(sy == +1 || sy == -1); - ey = y->ptr; - } - /* main fragment */ - xassert(sx > 0 && sy > 0 || sx < 0 && sy < 0); - cc = 0; - for (; ex || ey; ex = ex->next, ey = ey->next) - { if (ex == NULL) - ex = &zero; - if (ey == NULL) - ey = &zero; - for (k = 0; k <= 5; k++) - { if (ex->d[k] > ey->d[k]) - cc = +1; - if (ex->d[k] < ey->d[k]) - cc = -1; - } - } - if (sx < 0) cc = - cc; -done: return cc; -} - -int mpz_sgn(mpz_t x) -{ /* return +1 if x > 0, 0 if x = 0, and -1 if x < 0 */ - int s; - s = (x->val > 0 ? +1 : x->val < 0 ? -1 : 0); - return s; -} - -int mpz_out_str(void *_fp, int base, mpz_t x) -{ /* output x on stream fp, as a string in given base; the base - * may vary from 2 to 36; - * return the number of bytes written, or if an error occurred, - * return 0 */ - FILE *fp = _fp; - mpz_t b, y, r; - int n, j, nwr = 0; - unsigned char *d; - static char *set = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"; - if (!(2 <= base && base <= 36)) - xerror("mpz_out_str: base = %d; invalid base\n", base); - mpz_init(b); - mpz_set_si(b, base); - mpz_init(y); - mpz_init(r); - /* determine the number of digits */ - mpz_abs(y, x); - for (n = 0; mpz_sgn(y) != 0; n++) - mpz_div(y, NULL, y, b); - if (n == 0) n = 1; - /* compute the digits */ - d = xmalloc(n); - mpz_abs(y, x); - for (j = 0; j < n; j++) - { mpz_div(y, r, y, b); - xassert(0 <= r->val && r->val < base && r->ptr == NULL); - d[j] = (unsigned char)r->val; - } - /* output the integer to the stream */ - if (fp == NULL) - fp = stdout; - if (mpz_sgn(x) < 0) - fputc('-', fp), nwr++; - for (j = n-1; j >= 0; j--) - fputc(set[d[j]], fp), nwr++; - if (ferror(fp)) - nwr = 0; - mpz_clear(b); - mpz_clear(y); - mpz_clear(r); - xfree(d); - return nwr; -} - -/*--------------------------------------------------------------------*/ - -mpq_t _mpq_init(void) -{ /* initialize x, and set its value to 0/1 */ - mpq_t x; - x = gmp_get_atom(sizeof(struct mpq)); - x->p.val = 0; - x->p.ptr = NULL; - x->q.val = 1; - x->q.ptr = NULL; - return x; -} - -void mpq_clear(mpq_t x) -{ /* free the space occupied by x */ - mpz_set_si(&x->p, 0); - xassert(x->p.ptr == NULL); - mpz_set_si(&x->q, 0); - xassert(x->q.ptr == NULL); - /* free the number descriptor */ - gmp_free_atom(x, sizeof(struct mpq)); - return; -} - -void mpq_canonicalize(mpq_t x) -{ /* remove any factors that are common to the numerator and - * denominator of x, and make the denominator positive */ - mpz_t f; - xassert(x->q.val != 0); - if (x->q.val < 0) - { mpz_neg(&x->p, &x->p); - mpz_neg(&x->q, &x->q); - } - mpz_init(f); - mpz_gcd(f, &x->p, &x->q); - if (!(f->val == 1 && f->ptr == NULL)) - { mpz_div(&x->p, NULL, &x->p, f); - mpz_div(&x->q, NULL, &x->q, f); - } - mpz_clear(f); - return; -} - -void mpq_set(mpq_t z, mpq_t x) -{ /* set the value of z from x */ - if (z != x) - { mpz_set(&z->p, &x->p); - mpz_set(&z->q, &x->q); - } - return; -} - -void mpq_set_si(mpq_t x, int p, unsigned int q) -{ /* set the value of x to p/q */ - if (q == 0) - xerror("mpq_set_si: zero denominator not allowed\n"); - mpz_set_si(&x->p, p); - xassert(q <= 0x7FFFFFFF); - mpz_set_si(&x->q, q); - return; -} - -double mpq_get_d(mpq_t x) -{ /* convert x to a double, truncating if necessary */ - int np, nq; - double p, q; - p = mpz_get_d_2exp(&np, &x->p); - q = mpz_get_d_2exp(&nq, &x->q); - return ldexp(p / q, np - nq); -} - -void mpq_set_d(mpq_t x, double val) -{ /* set x to val; there is no rounding, the conversion is exact */ - int s, n, d, j; - double f; - mpz_t temp; - xassert(-DBL_MAX <= val && val <= +DBL_MAX); - mpq_set_si(x, 0, 1); - if (val > 0.0) - s = +1; - else if (val < 0.0) - s = -1; - else - goto done; - f = frexp(fabs(val), &n); - /* |val| = f * 2^n, where 0.5 <= f < 1.0 */ - mpz_init(temp); - while (f != 0.0) - { f *= 16.0, n -= 4; - d = (int)f; - xassert(0 <= d && d <= 15); - f -= (double)d; - /* x := 16 * x + d */ - mpz_set_si(temp, 16); - mpz_mul(&x->p, &x->p, temp); - mpz_set_si(temp, d); - mpz_add(&x->p, &x->p, temp); - } - mpz_clear(temp); - /* x := x * 2^n */ - if (n > 0) - { for (j = 1; j <= n; j++) - mpz_add(&x->p, &x->p, &x->p); - } - else if (n < 0) - { for (j = 1; j <= -n; j++) - mpz_add(&x->q, &x->q, &x->q); - mpq_canonicalize(x); - } - if (s < 0) - mpq_neg(x, x); -done: return; -} - -void mpq_add(mpq_t z, mpq_t x, mpq_t y) -{ /* set z to x + y */ - mpz_t p, q; - mpz_init(p); - mpz_init(q); - mpz_mul(p, &x->p, &y->q); - mpz_mul(q, &x->q, &y->p); - mpz_add(p, p, q); - mpz_mul(q, &x->q, &y->q); - mpz_set(&z->p, p); - mpz_set(&z->q, q); - mpz_clear(p); - mpz_clear(q); - mpq_canonicalize(z); - return; -} - -void mpq_sub(mpq_t z, mpq_t x, mpq_t y) -{ /* set z to x - y */ - mpz_t p, q; - mpz_init(p); - mpz_init(q); - mpz_mul(p, &x->p, &y->q); - mpz_mul(q, &x->q, &y->p); - mpz_sub(p, p, q); - mpz_mul(q, &x->q, &y->q); - mpz_set(&z->p, p); - mpz_set(&z->q, q); - mpz_clear(p); - mpz_clear(q); - mpq_canonicalize(z); - return; -} - -void mpq_mul(mpq_t z, mpq_t x, mpq_t y) -{ /* set z to x * y */ - mpz_mul(&z->p, &x->p, &y->p); - mpz_mul(&z->q, &x->q, &y->q); - mpq_canonicalize(z); - return; -} - -void mpq_div(mpq_t z, mpq_t x, mpq_t y) -{ /* set z to x / y */ - mpz_t p, q; - if (mpq_sgn(y) == 0) - xerror("mpq_div: zero divisor not allowed\n"); - mpz_init(p); - mpz_init(q); - mpz_mul(p, &x->p, &y->q); - mpz_mul(q, &x->q, &y->p); - mpz_set(&z->p, p); - mpz_set(&z->q, q); - mpz_clear(p); - mpz_clear(q); - mpq_canonicalize(z); - return; -} - -void mpq_neg(mpq_t z, mpq_t x) -{ /* set z to 0 - x */ - mpq_set(z, x); - mpz_neg(&z->p, &z->p); - return; -} - -void mpq_abs(mpq_t z, mpq_t x) -{ /* set z to the absolute value of x */ - mpq_set(z, x); - mpz_abs(&z->p, &z->p); - xassert(mpz_sgn(&x->q) > 0); - return; -} - -int mpq_cmp(mpq_t x, mpq_t y) -{ /* compare x and y; return a positive value if x > y, zero if - * x = y, or a negative value if x < y */ - mpq_t temp; - int s; - mpq_init(temp); - mpq_sub(temp, x, y); - s = mpq_sgn(temp); - mpq_clear(temp); - return s; -} - -int mpq_sgn(mpq_t x) -{ /* return +1 if x > 0, 0 if x = 0, and -1 if x < 0 */ - int s; - s = mpz_sgn(&x->p); - xassert(mpz_sgn(&x->q) > 0); - return s; -} - -int mpq_out_str(void *_fp, int base, mpq_t x) -{ /* output x on stream fp, as a string in given base; the base - * may vary from 2 to 36; output is in the form 'num/den' or if - * the denominator is 1 then just 'num'; - * if the parameter fp is a null pointer, stdout is assumed; - * return the number of bytes written, or if an error occurred, - * return 0 */ - FILE *fp = _fp; - int nwr; - if (!(2 <= base && base <= 36)) - xerror("mpq_out_str: base = %d; invalid base\n", base); - if (fp == NULL) - fp = stdout; - nwr = mpz_out_str(fp, base, &x->p); - if (x->q.val == 1 && x->q.ptr == NULL) - ; - else - { fputc('/', fp), nwr++; - nwr += mpz_out_str(fp, base, &x->q); - } - if (ferror(fp)) - nwr = 0; - return nwr; -} - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/mygmp.h b/code/3rd_glpk/misc/mygmp.h deleted file mode 100644 index 31d2024d..00000000 --- a/code/3rd_glpk/misc/mygmp.h +++ /dev/null @@ -1,254 +0,0 @@ -/* mygmp.h (integer and rational arithmetic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2008-2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MYGMP_H -#define MYGMP_H - -#ifdef HAVE_CONFIG_H -#include -#endif - -#ifdef HAVE_GMP /* use GNU MP library */ - -#include - -#define gmp_pool_count() 0 - -#define gmp_free_mem() ((void)0) - -#else /* use GLPK MP module */ - -/*********************************************************************** -* INTEGER NUMBERS -* --------------- -* Depending on its magnitude an integer number of arbitrary precision -* is represented either in short format or in long format. -* -* Short format corresponds to the int type and allows representing -* integer numbers in the range [-(2^31-1), +(2^31-1)]. Note that for -* the most negative number of int type the short format is not used. -* -* In long format integer numbers are represented using the positional -* system with the base (radix) 2^16 = 65536: -* -* x = (-1)^s sum{j in 0..n-1} d[j] * 65536^j, -* -* where x is the integer to be represented, s is its sign (+1 or -1), -* d[j] are its digits (0 <= d[j] <= 65535). -* -* RATIONAL NUMBERS -* ---------------- -* A rational number is represented as an irreducible fraction: -* -* p / q, -* -* where p (numerator) and q (denominator) are integer numbers (q > 0) -* having no common divisors. */ - -struct mpz -{ /* integer number */ - int val; - /* if ptr is a null pointer, the number is in short format, and - val is its value; otherwise, the number is in long format, and - val is its sign (+1 or -1) */ - struct mpz_seg *ptr; - /* pointer to the linked list of the number segments ordered in - ascending of powers of the base */ -}; - -struct mpz_seg -{ /* integer number segment */ - unsigned short d[6]; - /* six digits of the number ordered in ascending of powers of the - base */ - struct mpz_seg *next; - /* pointer to the next number segment */ -}; - -struct mpq -{ /* rational number (p / q) */ - struct mpz p; - /* numerator */ - struct mpz q; - /* denominator */ -}; - -typedef struct mpz *mpz_t; -typedef struct mpq *mpq_t; - -#define gmp_get_atom _glp_gmp_get_atom -void *gmp_get_atom(int size); - -#define gmp_free_atom _glp_gmp_free_atom -void gmp_free_atom(void *ptr, int size); - -#define gmp_pool_count _glp_gmp_pool_count -int gmp_pool_count(void); - -#define gmp_get_work _glp_gmp_get_work -unsigned short *gmp_get_work(int size); - -#define gmp_free_mem _glp_gmp_free_mem -void gmp_free_mem(void); - -#define mpz_init(x) (void)((x) = _mpz_init()) - -#define _mpz_init _glp_mpz_init -mpz_t _mpz_init(void); -/* initialize x and set its value to 0 */ - -#define mpz_clear _glp_mpz_clear -void mpz_clear(mpz_t x); -/* free the space occupied by x */ - -#define mpz_set _glp_mpz_set -void mpz_set(mpz_t z, mpz_t x); -/* set the value of z from x */ - -#define mpz_set_si _glp_mpz_set_si -void mpz_set_si(mpz_t x, int val); -/* set the value of x to val */ - -#define mpz_get_d _glp_mpz_get_d -double mpz_get_d(mpz_t x); -/* convert x to a double, truncating if necessary */ - -#define mpz_get_d_2exp _glp_mpz_get_d_2exp -double mpz_get_d_2exp(int *exp, mpz_t x); -/* convert x to a double, returning the exponent separately */ - -#define mpz_swap _glp_mpz_swap -void mpz_swap(mpz_t x, mpz_t y); -/* swap the values x and y efficiently */ - -#define mpz_add _glp_mpz_add -void mpz_add(mpz_t, mpz_t, mpz_t); -/* set z to x + y */ - -#define mpz_sub _glp_mpz_sub -void mpz_sub(mpz_t, mpz_t, mpz_t); -/* set z to x - y */ - -#define mpz_mul _glp_mpz_mul -void mpz_mul(mpz_t, mpz_t, mpz_t); -/* set z to x * y */ - -#define mpz_neg _glp_mpz_neg -void mpz_neg(mpz_t z, mpz_t x); -/* set z to 0 - x */ - -#define mpz_abs _glp_mpz_abs -void mpz_abs(mpz_t z, mpz_t x); -/* set z to the absolute value of x */ - -#define mpz_div _glp_mpz_div -void mpz_div(mpz_t q, mpz_t r, mpz_t x, mpz_t y); -/* divide x by y, forming quotient q and/or remainder r */ - -#define mpz_gcd _glp_mpz_gcd -void mpz_gcd(mpz_t z, mpz_t x, mpz_t y); -/* set z to the greatest common divisor of x and y */ - -#define mpz_cmp _glp_mpz_cmp -int mpz_cmp(mpz_t x, mpz_t y); -/* compare x and y */ - -#define mpz_sgn _glp_mpz_sgn -int mpz_sgn(mpz_t x); -/* return +1 if x > 0, 0 if x = 0, and -1 if x < 0 */ - -#define mpz_out_str _glp_mpz_out_str -int mpz_out_str(void *fp, int base, mpz_t x); -/* output x on stream fp, as a string in given base */ - -#define mpq_init(x) (void)((x) = _mpq_init()) - -#define _mpq_init _glp_mpq_init -mpq_t _mpq_init(void); -/* initialize x, and set its value to 0/1 */ - -#define mpq_clear _glp_mpq_clear -void mpq_clear(mpq_t x); -/* free the space occupied by x */ - -#define mpq_canonicalize _glp_mpq_canonicalize -void mpq_canonicalize(mpq_t x); -/* canonicalize x */ - -#define mpq_set _glp_mpq_set -void mpq_set(mpq_t z, mpq_t x); -/* set the value of z from x */ - -#define mpq_set_si _glp_mpq_set_si -void mpq_set_si(mpq_t x, int p, unsigned int q); -/* set the value of x to p/q */ - -#define mpq_get_d _glp_mpq_get_d -double mpq_get_d(mpq_t x); -/* convert x to a double, truncating if necessary */ - -#define mpq_set_d _glp_mpq_set_d -void mpq_set_d(mpq_t x, double val); -/* set x to val; there is no rounding, the conversion is exact */ - -#define mpq_add _glp_mpq_add -void mpq_add(mpq_t z, mpq_t x, mpq_t y); -/* set z to x + y */ - -#define mpq_sub _glp_mpq_sub -void mpq_sub(mpq_t z, mpq_t x, mpq_t y); -/* set z to x - y */ - -#define mpq_mul _glp_mpq_mul -void mpq_mul(mpq_t z, mpq_t x, mpq_t y); -/* set z to x * y */ - -#define mpq_div _glp_mpq_div -void mpq_div(mpq_t z, mpq_t x, mpq_t y); -/* set z to x / y */ - -#define mpq_neg _glp_mpq_neg -void mpq_neg(mpq_t z, mpq_t x); -/* set z to 0 - x */ - -#define mpq_abs _glp_mpq_abs -void mpq_abs(mpq_t z, mpq_t x); -/* set z to the absolute value of x */ - -#define mpq_cmp _glp_mpq_cmp -int mpq_cmp(mpq_t x, mpq_t y); -/* compare x and y */ - -#define mpq_sgn _glp_mpq_sgn -int mpq_sgn(mpq_t x); -/* return +1 if x > 0, 0 if x = 0, and -1 if x < 0 */ - -#define mpq_out_str _glp_mpq_out_str -int mpq_out_str(void *fp, int base, mpq_t x); -/* output x on stream fp, as a string in given base */ - -#endif - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/okalg.c b/code/3rd_glpk/misc/okalg.c deleted file mode 100644 index 8eecd6df..00000000 --- a/code/3rd_glpk/misc/okalg.c +++ /dev/null @@ -1,382 +0,0 @@ -/* okalg.c (out-of-kilter algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "okalg.h" - -/*********************************************************************** -* NAME -* -* okalg - out-of-kilter algorithm -* -* SYNOPSIS -* -* #include "okalg.h" -* int okalg(int nv, int na, const int tail[], const int head[], -* const int low[], const int cap[], const int cost[], int x[], -* int pi[]); -* -* DESCRIPTION -* -* The routine okalg implements the out-of-kilter algorithm to find a -* minimal-cost circulation in the specified flow network. -* -* INPUT PARAMETERS -* -* nv is the number of nodes, nv >= 0. -* -* na is the number of arcs, na >= 0. -* -* tail[a], a = 1,...,na, is the index of tail node of arc a. -* -* head[a], a = 1,...,na, is the index of head node of arc a. -* -* low[a], a = 1,...,na, is an lower bound to the flow through arc a. -* -* cap[a], a = 1,...,na, is an upper bound to the flow through arc a, -* which is the capacity of the arc. -* -* cost[a], a = 1,...,na, is a per-unit cost of the flow through arc a. -* -* NOTES -* -* 1. Multiple arcs are allowed, but self-loops are not allowed. -* -* 2. It is required that 0 <= low[a] <= cap[a] for all arcs. -* -* 3. Arc costs may have any sign. -* -* OUTPUT PARAMETERS -* -* x[a], a = 1,...,na, is optimal value of the flow through arc a. -* -* pi[i], i = 1,...,nv, is Lagrange multiplier for flow conservation -* equality constraint corresponding to node i (the node potential). -* -* RETURNS -* -* 0 optimal circulation found; -* -* 1 there is no feasible circulation; -* -* 2 integer overflow occured; -* -* 3 optimality test failed (logic error). -* -* REFERENCES -* -* L.R.Ford, Jr., and D.R.Fulkerson, "Flows in Networks," The RAND -* Corp., Report R-375-PR (August 1962), Chap. III "Minimal Cost Flow -* Problems," pp.113-26. */ - -static int overflow(int u, int v) -{ /* check for integer overflow on computing u + v */ - if (u > 0 && v > 0 && u + v < 0) return 1; - if (u < 0 && v < 0 && u + v > 0) return 1; - return 0; -} - -int okalg(int nv, int na, const int tail[], const int head[], - const int low[], const int cap[], const int cost[], int x[], - int pi[]) -{ int a, aok, delta, i, j, k, lambda, pos1, pos2, s, t, temp, ret, - *ptr, *arc, *link, *list; - /* sanity checks */ - xassert(nv >= 0); - xassert(na >= 0); - for (a = 1; a <= na; a++) - { i = tail[a], j = head[a]; - xassert(1 <= i && i <= nv); - xassert(1 <= j && j <= nv); - xassert(i != j); - xassert(0 <= low[a] && low[a] <= cap[a]); - } - /* allocate working arrays */ - ptr = xcalloc(1+nv+1, sizeof(int)); - arc = xcalloc(1+na+na, sizeof(int)); - link = xcalloc(1+nv, sizeof(int)); - list = xcalloc(1+nv, sizeof(int)); - /* ptr[i] := (degree of node i) */ - for (i = 1; i <= nv; i++) - ptr[i] = 0; - for (a = 1; a <= na; a++) - { ptr[tail[a]]++; - ptr[head[a]]++; - } - /* initialize arc pointers */ - ptr[1]++; - for (i = 1; i < nv; i++) - ptr[i+1] += ptr[i]; - ptr[nv+1] = ptr[nv]; - /* build arc lists */ - for (a = 1; a <= na; a++) - { arc[--ptr[tail[a]]] = a; - arc[--ptr[head[a]]] = a; - } - xassert(ptr[1] == 1); - xassert(ptr[nv+1] == na+na+1); - /* now the indices of arcs incident to node i are stored in - * locations arc[ptr[i]], arc[ptr[i]+1], ..., arc[ptr[i+1]-1] */ - /* initialize arc flows and node potentials */ - for (a = 1; a <= na; a++) - x[a] = 0; - for (i = 1; i <= nv; i++) - pi[i] = 0; -loop: /* main loop starts here */ - /* find out-of-kilter arc */ - aok = 0; - for (a = 1; a <= na; a++) - { i = tail[a], j = head[a]; - if (overflow(cost[a], pi[i] - pi[j])) - { ret = 2; - goto done; - } - lambda = cost[a] + (pi[i] - pi[j]); - if (x[a] < low[a] || (lambda < 0 && x[a] < cap[a])) - { /* arc a = i->j is out of kilter, and we need to increase - * the flow through this arc */ - aok = a, s = j, t = i; - break; - } - if (x[a] > cap[a] || (lambda > 0 && x[a] > low[a])) - { /* arc a = i->j is out of kilter, and we need to decrease - * the flow through this arc */ - aok = a, s = i, t = j; - break; - } - } - if (aok == 0) - { /* all arcs are in kilter */ - /* check for feasibility */ - for (a = 1; a <= na; a++) - { if (!(low[a] <= x[a] && x[a] <= cap[a])) - { ret = 3; - goto done; - } - } - for (i = 1; i <= nv; i++) - { temp = 0; - for (k = ptr[i]; k < ptr[i+1]; k++) - { a = arc[k]; - if (tail[a] == i) - { /* a is outgoing arc */ - temp += x[a]; - } - else if (head[a] == i) - { /* a is incoming arc */ - temp -= x[a]; - } - else - xassert(a != a); - } - if (temp != 0) - { ret = 3; - goto done; - } - } - /* check for optimality */ - for (a = 1; a <= na; a++) - { i = tail[a], j = head[a]; - lambda = cost[a] + (pi[i] - pi[j]); - if ((lambda > 0 && x[a] != low[a]) || - (lambda < 0 && x[a] != cap[a])) - { ret = 3; - goto done; - } - } - /* current circulation is optimal */ - ret = 0; - goto done; - } - /* now we need to find a cycle (t, a, s, ..., t), which allows - * increasing the flow along it, where a is the out-of-kilter arc - * just found */ - /* link[i] = 0 means that node i is not labelled yet; - * link[i] = a means that arc a immediately precedes node i */ - /* initially only node s is labelled */ - for (i = 1; i <= nv; i++) - link[i] = 0; - link[s] = aok, list[1] = s, pos1 = pos2 = 1; - /* breadth first search */ - while (pos1 <= pos2) - { /* dequeue node i */ - i = list[pos1++]; - /* consider all arcs incident to node i */ - for (k = ptr[i]; k < ptr[i+1]; k++) - { a = arc[k]; - if (tail[a] == i) - { /* a = i->j is a forward arc from s to t */ - j = head[a]; - /* if node j has been labelled, skip the arc */ - if (link[j] != 0) continue; - /* if the arc does not allow increasing the flow through - * it, skip the arc */ - if (x[a] >= cap[a]) continue; - if (overflow(cost[a], pi[i] - pi[j])) - { ret = 2; - goto done; - } - lambda = cost[a] + (pi[i] - pi[j]); - if (lambda > 0 && x[a] >= low[a]) continue; - } - else if (head[a] == i) - { /* a = i<-j is a backward arc from s to t */ - j = tail[a]; - /* if node j has been labelled, skip the arc */ - if (link[j] != 0) continue; - /* if the arc does not allow decreasing the flow through - * it, skip the arc */ - if (x[a] <= low[a]) continue; - if (overflow(cost[a], pi[j] - pi[i])) - { ret = 2; - goto done; - } - lambda = cost[a] + (pi[j] - pi[i]); - if (lambda < 0 && x[a] <= cap[a]) continue; - } - else - xassert(a != a); - /* label node j and enqueue it */ - link[j] = a, list[++pos2] = j; - /* check for breakthrough */ - if (j == t) goto brkt; - } - } - /* NONBREAKTHROUGH */ - /* consider all arcs, whose one endpoint is labelled and other is - * not, and determine maximal change of node potentials */ - delta = 0; - for (a = 1; a <= na; a++) - { i = tail[a], j = head[a]; - if (link[i] != 0 && link[j] == 0) - { /* a = i->j, where node i is labelled, node j is not */ - if (overflow(cost[a], pi[i] - pi[j])) - { ret = 2; - goto done; - } - lambda = cost[a] + (pi[i] - pi[j]); - if (x[a] <= cap[a] && lambda > 0) - if (delta == 0 || delta > + lambda) delta = + lambda; - } - else if (link[i] == 0 && link[j] != 0) - { /* a = j<-i, where node j is labelled, node i is not */ - if (overflow(cost[a], pi[i] - pi[j])) - { ret = 2; - goto done; - } - lambda = cost[a] + (pi[i] - pi[j]); - if (x[a] >= low[a] && lambda < 0) - if (delta == 0 || delta > - lambda) delta = - lambda; - } - } - if (delta == 0) - { /* there is no feasible circulation */ - ret = 1; - goto done; - } - /* increase potentials of all unlabelled nodes */ - for (i = 1; i <= nv; i++) - { if (link[i] == 0) - { if (overflow(pi[i], delta)) - { ret = 2; - goto done; - } - pi[i] += delta; - } - } - goto loop; -brkt: /* BREAKTHROUGH */ - /* walk through arcs of the cycle (t, a, s, ..., t) found in the - * reverse order and determine maximal change of the flow */ - delta = 0; - for (j = t;; j = i) - { /* arc a immediately precedes node j in the cycle */ - a = link[j]; - if (head[a] == j) - { /* a = i->j is a forward arc of the cycle */ - i = tail[a]; - lambda = cost[a] + (pi[i] - pi[j]); - if (lambda > 0 && x[a] < low[a]) - { /* x[a] may be increased until its lower bound */ - temp = low[a] - x[a]; - } - else if (lambda <= 0 && x[a] < cap[a]) - { /* x[a] may be increased until its upper bound */ - temp = cap[a] - x[a]; - } - else - xassert(a != a); - } - else if (tail[a] == j) - { /* a = i<-j is a backward arc of the cycle */ - i = head[a]; - lambda = cost[a] + (pi[j] - pi[i]); - if (lambda < 0 && x[a] > cap[a]) - { /* x[a] may be decreased until its upper bound */ - temp = x[a] - cap[a]; - } - else if (lambda >= 0 && x[a] > low[a]) - { /* x[a] may be decreased until its lower bound */ - temp = x[a] - low[a]; - } - else - xassert(a != a); - } - else - xassert(a != a); - if (delta == 0 || delta > temp) delta = temp; - /* check for end of the cycle */ - if (i == t) break; - } - xassert(delta > 0); - /* increase the flow along the cycle */ - for (j = t;; j = i) - { /* arc a immediately precedes node j in the cycle */ - a = link[j]; - if (head[a] == j) - { /* a = i->j is a forward arc of the cycle */ - i = tail[a]; - /* overflow cannot occur */ - x[a] += delta; - } - else if (tail[a] == j) - { /* a = i<-j is a backward arc of the cycle */ - i = head[a]; - /* overflow cannot occur */ - x[a] -= delta; - } - else - xassert(a != a); - /* check for end of the cycle */ - if (i == t) break; - } - goto loop; -done: /* free working arrays */ - xfree(ptr); - xfree(arc); - xfree(link); - xfree(list); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/okalg.h b/code/3rd_glpk/misc/okalg.h deleted file mode 100644 index 2f2d9740..00000000 --- a/code/3rd_glpk/misc/okalg.h +++ /dev/null @@ -1,35 +0,0 @@ -/* okalg.h (out-of-kilter algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef OKALG_H -#define OKALG_H - -#define okalg _glp_okalg -int okalg(int nv, int na, const int tail[], const int head[], - const int low[], const int cap[], const int cost[], int x[], - int pi[]); -/* out-of-kilter algorithm */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/qmd.c b/code/3rd_glpk/misc/qmd.c deleted file mode 100644 index a3397dcf..00000000 --- a/code/3rd_glpk/misc/qmd.c +++ /dev/null @@ -1,584 +0,0 @@ -/* qmd.c (quotient minimum degree algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* THIS CODE IS THE RESULT OF TRANSLATION OF THE FORTRAN SUBROUTINES -* GENQMD, QMDRCH, QMDQT, QMDUPD, AND QMDMRG FROM THE BOOK: -* -* ALAN GEORGE, JOSEPH W-H LIU. COMPUTER SOLUTION OF LARGE SPARSE -* POSITIVE DEFINITE SYSTEMS. PRENTICE-HALL, 1981. -* -* THE TRANSLATION HAS BEEN DONE WITH THE PERMISSION OF THE AUTHORS -* OF THE ORIGINAL FORTRAN SUBROUTINES: ALAN GEORGE AND JOSEPH LIU, -* UNIVERSITY OF WATERLOO, WATERLOO, ONTARIO, CANADA. -* -* The translation was made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "qmd.h" - -/*********************************************************************** -* NAME -* -* genqmd - GENeral Quotient Minimum Degree algorithm -* -* SYNOPSIS -* -* #include "qmd.h" -* void genqmd(int *neqns, int xadj[], int adjncy[], int perm[], -* int invp[], int deg[], int marker[], int rchset[], int nbrhd[], -* int qsize[], int qlink[], int *nofsub); -* -* PURPOSE -* -* This routine implements the minimum degree algorithm. It makes use -* of the implicit representation of the elimination graph by quotient -* graphs, and the notion of indistinguishable nodes. -* -* CAUTION -* -* The adjancy vector adjncy will be destroyed. -* -* INPUT PARAMETERS -* -* neqns - number of equations; -* (xadj, adjncy) - -* the adjancy structure. -* -* OUTPUT PARAMETERS -* -* perm - the minimum degree ordering; -* invp - the inverse of perm. -* -* WORKING PARAMETERS -* -* deg - the degree vector. deg[i] is negative means node i has been -* numbered; -* marker - a marker vector, where marker[i] is negative means node i -* has been merged with another nodeand thus can be ignored; -* rchset - vector used for the reachable set; -* nbrhd - vector used for neighborhood set; -* qsize - vector used to store the size of indistinguishable -* supernodes; -* qlink - vector used to store indistinguishable nodes, i, qlink[i], -* qlink[qlink[i]], ... are the members of the supernode -* represented by i. -* -* PROGRAM SUBROUTINES -* -* qmdrch, qmdqt, qmdupd. -***********************************************************************/ - -void genqmd(int *_neqns, int xadj[], int adjncy[], int perm[], - int invp[], int deg[], int marker[], int rchset[], int nbrhd[], - int qsize[], int qlink[], int *_nofsub) -{ int inode, ip, irch, j, mindeg, ndeg, nhdsze, node, np, num, - nump1, nxnode, rchsze, search, thresh; -# define neqns (*_neqns) -# define nofsub (*_nofsub) - /* Initialize degree vector and other working variables. */ - mindeg = neqns; - nofsub = 0; - for (node = 1; node <= neqns; node++) - { perm[node] = node; - invp[node] = node; - marker[node] = 0; - qsize[node] = 1; - qlink[node] = 0; - ndeg = xadj[node+1] - xadj[node]; - deg[node] = ndeg; - if (ndeg < mindeg) mindeg = ndeg; - } - num = 0; - /* Perform threshold search to get a node of min degree. - * Variable search point to where search should start. */ -s200: search = 1; - thresh = mindeg; - mindeg = neqns; -s300: nump1 = num + 1; - if (nump1 > search) search = nump1; - for (j = search; j <= neqns; j++) - { node = perm[j]; - if (marker[node] >= 0) - { ndeg = deg[node]; - if (ndeg <= thresh) goto s500; - if (ndeg < mindeg) mindeg = ndeg; - } - } - goto s200; - /* Node has minimum degree. Find its reachable sets by calling - * qmdrch. */ -s500: search = j; - nofsub += deg[node]; - marker[node] = 1; - qmdrch(&node, xadj, adjncy, deg, marker, &rchsze, rchset, &nhdsze, - nbrhd); - /* Eliminate all nodes indistinguishable from node. They are given - * by node, qlink[node], ... . */ - nxnode = node; -s600: num++; - np = invp[nxnode]; - ip = perm[num]; - perm[np] = ip; - invp[ip] = np; - perm[num] = nxnode; - invp[nxnode] = num; - deg[nxnode] = -1; - nxnode = qlink[nxnode]; - if (nxnode > 0) goto s600; - if (rchsze > 0) - { /* Update the degrees of the nodes in the reachable set and - * identify indistinguishable nodes. */ - qmdupd(xadj, adjncy, &rchsze, rchset, deg, qsize, qlink, - marker, &rchset[rchsze+1], &nbrhd[nhdsze+1]); - /* Reset marker value of nodes in reach set. Update threshold - * value for cyclic search. Also call qmdqt to form new - * quotient graph. */ - marker[node] = 0; - for (irch = 1; irch <= rchsze; irch++) - { inode = rchset[irch]; - if (marker[inode] >= 0) - { marker[inode] = 0; - ndeg = deg[inode]; - if (ndeg < mindeg) mindeg = ndeg; - if (ndeg <= thresh) - { mindeg = thresh; - thresh = ndeg; - search = invp[inode]; - } - } - } - if (nhdsze > 0) - qmdqt(&node, xadj, adjncy, marker, &rchsze, rchset, nbrhd); - } - if (num < neqns) goto s300; - return; -# undef neqns -# undef nofsub -} - -/*********************************************************************** -* NAME -* -* qmdrch - Quotient MD ReaCHable set -* -* SYNOPSIS -* -* #include "qmd.h" -* void qmdrch(int *root, int xadj[], int adjncy[], int deg[], -* int marker[], int *rchsze, int rchset[], int *nhdsze, -* int nbrhd[]); -* -* PURPOSE -* -* This subroutine determines the reachable set of a node through a -* given subset. The adjancy structure is assumed to be stored in a -* quotient graph format. -* -* INPUT PARAMETERS -* -* root - the given node not in the subset; -* (xadj, adjncy) - -* the adjancy structure pair; -* deg - the degree vector. deg[i] < 0 means the node belongs to the -* given subset. -* -* OUTPUT PARAMETERS -* -* (rchsze, rchset) - -* the reachable set; -* (nhdsze, nbrhd) - -* the neighborhood set. -* -* UPDATED PARAMETERS -* -* marker - the marker vector for reach and nbrhd sets. > 0 means the -* node is in reach set. < 0 means the node has been merged -* with others in the quotient or it is in nbrhd set. -***********************************************************************/ - -void qmdrch(int *_root, int xadj[], int adjncy[], int deg[], - int marker[], int *_rchsze, int rchset[], int *_nhdsze, - int nbrhd[]) -{ int i, istop, istrt, j, jstop, jstrt, nabor, node; -# define root (*_root) -# define rchsze (*_rchsze) -# define nhdsze (*_nhdsze) - /* Loop through the neighbors of root in the quotient graph. */ - nhdsze = 0; - rchsze = 0; - istrt = xadj[root]; - istop = xadj[root+1] - 1; - if (istop < istrt) return; - for (i = istrt; i <= istop; i++) - { nabor = adjncy[i]; - if (nabor == 0) return; - if (marker[nabor] == 0) - { if (deg[nabor] >= 0) - { /* Include nabor into the reachable set. */ - rchsze++; - rchset[rchsze] = nabor; - marker[nabor] = 1; - goto s600; - } - /* nabor has been eliminated. Find nodes reachable from - * it. */ - marker[nabor] = -1; - nhdsze++; - nbrhd[nhdsze] = nabor; -s300: jstrt = xadj[nabor]; - jstop = xadj[nabor+1] - 1; - for (j = jstrt; j <= jstop; j++) - { node = adjncy[j]; - nabor = - node; - if (node < 0) goto s300; - if (node == 0) goto s600; - if (marker[node] == 0) - { rchsze++; - rchset[rchsze] = node; - marker[node] = 1; - } - } - } -s600: ; - } - return; -# undef root -# undef rchsze -# undef nhdsze -} - -/*********************************************************************** -* NAME -* -* qmdqt - Quotient MD Quotient graph Transformation -* -* SYNOPSIS -* -* #include "qmd.h" -* void qmdqt(int *root, int xadj[], int adjncy[], int marker[], -* int *rchsze, int rchset[], int nbrhd[]); -* -* PURPOSE -* -* This subroutine performs the quotient graph transformation after a -* node has been eliminated. -* -* INPUT PARAMETERS -* -* root - the node just eliminated. It becomes the representative of -* the new supernode; -* (xadj, adjncy) - -* the adjancy structure; -* (rchsze, rchset) - -* the reachable set of root in the old quotient graph; -* nbrhd - the neighborhood set which will be merged with root to form -* the new supernode; -* marker - the marker vector. -* -* UPDATED PARAMETERS -* -* adjncy - becomes the adjncy of the quotient graph. -***********************************************************************/ - -void qmdqt(int *_root, int xadj[], int adjncy[], int marker[], - int *_rchsze, int rchset[], int nbrhd[]) -{ int inhd, irch, j, jstop, jstrt, link, nabor, node; -# define root (*_root) -# define rchsze (*_rchsze) - irch = 0; - inhd = 0; - node = root; -s100: jstrt = xadj[node]; - jstop = xadj[node+1] - 2; - if (jstop >= jstrt) - { /* Place reach nodes into the adjacent list of node. */ - for (j = jstrt; j <= jstop; j++) - { irch++; - adjncy[j] = rchset[irch]; - if (irch >= rchsze) goto s400; - } - } - /* Link to other space provided by the nbrhd set. */ - link = adjncy[jstop+1]; - node = - link; - if (link >= 0) - { inhd++; - node = nbrhd[inhd]; - adjncy[jstop+1] = - node; - } - goto s100; - /* All reachable nodes have been saved. End the adjacent list. - * Add root to the neighborhood list of each node in the reach - * set. */ -s400: adjncy[j+1] = 0; - for (irch = 1; irch <= rchsze; irch++) - { node = rchset[irch]; - if (marker[node] >= 0) - { jstrt = xadj[node]; - jstop = xadj[node+1] - 1; - for (j = jstrt; j <= jstop; j++) - { nabor = adjncy[j]; - if (marker[nabor] < 0) - { adjncy[j] = root; - goto s600; - } - } - } -s600: ; - } - return; -# undef root -# undef rchsze -} - -/*********************************************************************** -* NAME -* -* qmdupd - Quotient MD UPDate -* -* SYNOPSIS -* -* #include "qmd.h" -* void qmdupd(int xadj[], int adjncy[], int *nlist, int list[], -* int deg[], int qsize[], int qlink[], int marker[], int rchset[], -* int nbrhd[]); -* -* PURPOSE -* -* This routine performs degree update for a set of nodes in the minimum -* degree algorithm. -* -* INPUT PARAMETERS -* -* (xadj, adjncy) - -* the adjancy structure; -* (nlist, list) - -* the list of nodes whose degree has to be updated. -* -* UPDATED PARAMETERS -* -* deg - the degree vector; -* qsize - size of indistinguishable supernodes; -* qlink - linked list for indistinguishable nodes; -* marker - used to mark those nodes in reach/nbrhd sets. -* -* WORKING PARAMETERS -* -* rchset - the reachable set; -* nbrhd - the neighborhood set. -* -* PROGRAM SUBROUTINES -* -* qmdmrg. -***********************************************************************/ - -void qmdupd(int xadj[], int adjncy[], int *_nlist, int list[], - int deg[], int qsize[], int qlink[], int marker[], int rchset[], - int nbrhd[]) -{ int deg0, deg1, il, inhd, inode, irch, j, jstop, jstrt, mark, - nabor, nhdsze, node, rchsze; -# define nlist (*_nlist) - /* Find all eliminated supernodes that are adjacent to some nodes - * in the given list. Put them into (nhdsze, nbrhd). deg0 contains - * the number of nodes in the list. */ - if (nlist <= 0) return; - deg0 = 0; - nhdsze = 0; - for (il = 1; il <= nlist; il++) - { node = list[il]; - deg0 += qsize[node]; - jstrt = xadj[node]; - jstop = xadj[node+1] - 1; - for (j = jstrt; j <= jstop; j++) - { nabor = adjncy[j]; - if (marker[nabor] == 0 && deg[nabor] < 0) - { marker[nabor] = -1; - nhdsze++; - nbrhd[nhdsze] = nabor; - } - } - } - /* Merge indistinguishable nodes in the list by calling the - * subroutine qmdmrg. */ - if (nhdsze > 0) - qmdmrg(xadj, adjncy, deg, qsize, qlink, marker, °0, &nhdsze, - nbrhd, rchset, &nbrhd[nhdsze+1]); - /* Find the new degrees of the nodes that have not been merged. */ - for (il = 1; il <= nlist; il++) - { node = list[il]; - mark = marker[node]; - if (mark == 0 || mark == 1) - { marker[node] = 2; - qmdrch(&node, xadj, adjncy, deg, marker, &rchsze, rchset, - &nhdsze, nbrhd); - deg1 = deg0; - if (rchsze > 0) - { for (irch = 1; irch <= rchsze; irch++) - { inode = rchset[irch]; - deg1 += qsize[inode]; - marker[inode] = 0; - } - } - deg[node] = deg1 - 1; - if (nhdsze > 0) - { for (inhd = 1; inhd <= nhdsze; inhd++) - { inode = nbrhd[inhd]; - marker[inode] = 0; - } - } - } - } - return; -# undef nlist -} - -/*********************************************************************** -* NAME -* -* qmdmrg - Quotient MD MeRGe -* -* SYNOPSIS -* -* #include "qmd.h" -* void qmdmrg(int xadj[], int adjncy[], int deg[], int qsize[], -* int qlink[], int marker[], int *deg0, int *nhdsze, int nbrhd[], -* int rchset[], int ovrlp[]); -* -* PURPOSE -* -* This routine merges indistinguishable nodes in the minimum degree -* ordering algorithm. It also computes the new degrees of these new -* supernodes. -* -* INPUT PARAMETERS -* -* (xadj, adjncy) - -* the adjancy structure; -* deg0 - the number of nodes in the given set; -* (nhdsze, nbrhd) - -* the set of eliminated supernodes adjacent to some nodes in -* the set. -* -* UPDATED PARAMETERS -* -* deg - the degree vector; -* qsize - size of indistinguishable nodes; -* qlink - linked list for indistinguishable nodes; -* marker - the given set is given by those nodes with marker value set -* to 1. Those nodes with degree updated will have marker value -* set to 2. -* -* WORKING PARAMETERS -* -* rchset - the reachable set; -* ovrlp - temp vector to store the intersection of two reachable sets. -***********************************************************************/ - -void qmdmrg(int xadj[], int adjncy[], int deg[], int qsize[], - int qlink[], int marker[], int *_deg0, int *_nhdsze, int nbrhd[], - int rchset[], int ovrlp[]) -{ int deg1, head, inhd, iov, irch, j, jstop, jstrt, link, lnode, - mark, mrgsze, nabor, node, novrlp, rchsze, root; -# define deg0 (*_deg0) -# define nhdsze (*_nhdsze) - /* Initialization. */ - if (nhdsze <= 0) return; - for (inhd = 1; inhd <= nhdsze; inhd++) - { root = nbrhd[inhd]; - marker[root] = 0; - } - /* Loop through each eliminated supernode in the set - * (nhdsze, nbrhd). */ - for (inhd = 1; inhd <= nhdsze; inhd++) - { root = nbrhd[inhd]; - marker[root] = -1; - rchsze = 0; - novrlp = 0; - deg1 = 0; -s200: jstrt = xadj[root]; - jstop = xadj[root+1] - 1; - /* Determine the reachable set and its intersection with the - * input reachable set. */ - for (j = jstrt; j <= jstop; j++) - { nabor = adjncy[j]; - root = - nabor; - if (nabor < 0) goto s200; - if (nabor == 0) break; - mark = marker[nabor]; - if (mark == 0) - { rchsze++; - rchset[rchsze] = nabor; - deg1 += qsize[nabor]; - marker[nabor] = 1; - } - else if (mark == 1) - { novrlp++; - ovrlp[novrlp] = nabor; - marker[nabor] = 2; - } - } - /* From the overlapped set, determine the nodes that can be - * merged together. */ - head = 0; - mrgsze = 0; - for (iov = 1; iov <= novrlp; iov++) - { node = ovrlp[iov]; - jstrt = xadj[node]; - jstop = xadj[node+1] - 1; - for (j = jstrt; j <= jstop; j++) - { nabor = adjncy[j]; - if (marker[nabor] == 0) - { marker[node] = 1; - goto s1100; - } - } - /* Node belongs to the new merged supernode. Update the - * vectors qlink and qsize. */ - mrgsze += qsize[node]; - marker[node] = -1; - lnode = node; -s900: link = qlink[lnode]; - if (link > 0) - { lnode = link; - goto s900; - } - qlink[lnode] = head; - head = node; -s1100: ; - } - if (head > 0) - { qsize[head] = mrgsze; - deg[head] = deg0 + deg1 - 1; - marker[head] = 2; - } - /* Reset marker values. */ - root = nbrhd[inhd]; - marker[root] = 0; - if (rchsze > 0) - { for (irch = 1; irch <= rchsze; irch++) - { node = rchset[irch]; - marker[node] = 0; - } - } - } - return; -# undef deg0 -# undef nhdsze -} - -/* eof */ diff --git a/code/3rd_glpk/misc/qmd.h b/code/3rd_glpk/misc/qmd.h deleted file mode 100644 index e55d50f5..00000000 --- a/code/3rd_glpk/misc/qmd.h +++ /dev/null @@ -1,58 +0,0 @@ -/* qmd.h (quotient minimum degree algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2001-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef QMD_H -#define QMD_H - -#define genqmd _glp_genqmd -void genqmd(int *neqns, int xadj[], int adjncy[], int perm[], - int invp[], int deg[], int marker[], int rchset[], int nbrhd[], - int qsize[], int qlink[], int *nofsub); -/* GENeral Quotient Minimum Degree algorithm */ - -#define qmdrch _glp_qmdrch -void qmdrch(int *root, int xadj[], int adjncy[], int deg[], - int marker[], int *rchsze, int rchset[], int *nhdsze, - int nbrhd[]); -/* Quotient MD ReaCHable set */ - -#define qmdqt _glp_qmdqt -void qmdqt(int *root, int xadj[], int adjncy[], int marker[], - int *rchsze, int rchset[], int nbrhd[]); -/* Quotient MD Quotient graph Transformation */ - -#define qmdupd _glp_qmdupd -void qmdupd(int xadj[], int adjncy[], int *nlist, int list[], - int deg[], int qsize[], int qlink[], int marker[], int rchset[], - int nbrhd[]); -/* Quotient MD UPDate */ - -#define qmdmrg _glp_qmdmrg -void qmdmrg(int xadj[], int adjncy[], int deg[], int qsize[], - int qlink[], int marker[], int *deg0, int *nhdsze, int nbrhd[], - int rchset[], int ovrlp[]); -/* Quotient MD MeRGe */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/relax4.c b/code/3rd_glpk/misc/relax4.c deleted file mode 100644 index f0a47d6d..00000000 --- a/code/3rd_glpk/misc/relax4.c +++ /dev/null @@ -1,2850 +0,0 @@ -/* relax4.c (relaxation method of Bertsekas and Tseng) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* THIS CODE IS THE RESULT OF TRANSLATION OF THE FORTRAN CODE RELAX4. -* -* THE TRANSLATION HAS BEEN DONE WITH THE PERMISSION OF THE AUTHOR OF -* THE ORIGINAL FORTRAN CODE PROF. DIMITRI P. BERTSEKAS, MASSACHUSETTS -* INSTITUTE OF TECHNOLOGY, CAMBRIDGE, MASSACHUSETTS, USA. -* -* The translation was made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "relax4.h" - -/*********************************************************************** -* WARNING -* -* A serious bug was *tentatively* fixed in this code (see #if/#endif -* marked by 'mao'). -* -* This bug is inherited from the original Fortran version of the -* RELAX-IV code. Unfortunately, the code is very intricate, so this -* bug is still under investigation. Thanks to Sylvain Fournier for bug -* report. -* -* RELAX-IV bug details -* -------------------- -* In the original RELAX-IV code there are four similar fragments in -* subroutines ascnt1 and ascnt2 like this: -* -* C -* C DECREASE THE PRICES OF THE SCANNED NODES BY DELPRC. -* C ADJUST FLOW TO MAINTAIN COMPLEMENTARY SLACKNESS WITH -* C THE PRICES. -* C -* NB = 0 -* DO 6 I=1,NSAVE -* . . . -* IF (RC(ARC).EQ.0) THEN -* DELX=DELX+U(ARC) -* NB = NB + 1 -* PRDCSR(NB) = ARC -* END IF -* . . . -* -* On some instances the variable NB becomes greater than N (the number -* of nodes) that leads to indexing error, because the array PRDCSR is -* declared as array of N elements (more precisely, as array of MAXNN -* elements, however, NB becomes even much greater than MAXNN). -***********************************************************************/ - -#define false 0 -#define true 1 - -/*********************************************************************** -* NAME -* -* RELAX-IV (version of October 1994) -* -* PURPOSE -* -* This routine implements the relaxation method of Bertsekas and Tseng -* (see [1], [2]) for linear cost ordinary network flow problems. -* -* [1] Bertsekas, D. P., "A Unified Framework for Primal-Dual Methods" -* Mathematical Programming, Vol. 32, 1985, pp. 125-145. -* [2] Bertsekas, D. P., and Tseng, P., "Relaxation Methods for -* Minimum Cost" Operations Research, Vol. 26, 1988, pp. 93-114. -* -* The relaxation method is also described in the books: -* -* [3] Bertsekas, D. P., "Linear Network Optimization: Algorithms and -* Codes" MIT Press, 1991. -* [4] Bertsekas, D. P. and Tsitsiklis, J. N., "Parallel and Distributed -* Computation: Numerical Methods", Prentice-Hall, 1989. -* [5] Bertsekas, D. P., "Network Optimization: Continuous and Discrete -* Models", Athena Scientific, 1998. -* -* RELEASE NOTE -* -* This version of relaxation code has option for a special crash -* procedure for the initial price-flow pair. This is recommended for -* difficult problems where the default initialization results in long -* running times. crash = 1 corresponds to an auction/shortest path -* method -* -* These initializations are recommended in the absence of any prior -* information on a favorable initial flow-price vector pair that -* satisfies complementary slackness. -* -* The relaxation portion of the code differs from the code RELAXT-III -* and other earlier relaxation codes in that it maintains the set of -* nodes with nonzero deficit in a fifo queue. Like its predecessor -* RELAXT-III, this code maintains a linked list of balanced (i.e., of -* zero reduced cost) arcs so to reduce the work in labeling and -* scanning. Unlike RELAXT-III, it does not use selectively shortest -* path iterations for initialization. -* -* SOURCE -* -* The original Fortran code was written by Dimitri P. Bertsekas and -* Paul Tseng, with a contribution by Jonathan Eckstein in the phase II -* initialization. The original Fortran routine AUCTION was written by -* Dimitri P. Bertsekas and is based on the method described in the -* paper: -* -* [6] Bertsekas, D. P., "An Auction/Sequential Shortest Path Algorithm -* for the Minimum Cost Flow Problem", LIDS Report P-2146, MIT, -* Nov. 1992. -* -* For inquiries about the original Fortran code, please contact: -* -* Dimitri P. Bertsekas -* Laboratory for information and decision systems -* Massachusetts Institute of Technology -* Cambridge, MA 02139 -* (617) 253-7267, dimitrib@mit.edu -* -* This code is the result of translation of the original Fortran code. -* The translation was made by Andrew Makhorin . -* -* USER GUIDELINES -* -* This routine is in the public domain to be used only for research -* purposes. It cannot be used as part of a commercial product, or to -* satisfy in any part commercial delivery requirements to government -* or industry, without prior agreement with the authors. Users are -* requested to acknowledge the authorship of the code, and the -* relaxation method. -* -* No modification should be made to this code other than the minimal -* necessary to make it compatible with specific platforms. -* -* INPUT PARAMETERS (see notes 1, 2, 4) -* -* n = number of nodes -* na = number of arcs -* large = a very large integer to represent infinity -* (see note 3) -* repeat = true if initialization is to be skipped -* (false otherwise) -* crash = 0 if default initialization is used -* 1 if auction initialization is used -* startn[j] = starting node for arc j, j = 1,...,na -* endn[j] = ending node for arc j, j = 1,...,na -* fou[i] = first arc out of node i, i = 1,...,n -* nxtou[j] = next arc out of the starting node of arc j, j = 1,...,na -* fin[i] = first arc into node i, i = 1,...,n -* nxtin[j] = next arc into the ending node of arc j, j = 1,...,na -* -* UPDATED PARAMETERS (see notes 1, 3, 4) -* -* rc[j] = reduced cost of arc j, j = 1,...,na -* u[j] = capacity of arc j on input -* and (capacity of arc j) - x[j] on output, j = 1,...,na -* dfct[i] = demand at node i on input -* and zero on output, i = 1,...,n -* -* OUTPUT PARAMETERS (see notes 1, 3, 4) -* -* x[j] = flow on arc j, j = 1,...,na -* nmultinode = number of multinode relaxation iterations in RELAX4 -* iter = number of relaxation iterations in RELAX4 -* num_augm = number of flow augmentation steps in RELAX4 -* num_ascnt = number of multinode ascent steps in RELAX4 -* nsp = number of auction/shortest path iterations -* -* WORKING PARAMETERS (see notes 1, 4, 5) -* -* label[1+n], prdcsr[1+n], save[1+na], tfstou[1+n], tnxtou[1+na], -* tfstin[1+n], tnxtin[1+na], nxtqueue[1+n], scan[1+n], mark[1+n], -* extend_arc[1+n], sb_level[1+n], sb_arc[1+n] -* -* RETURNS -* -* 0 = normal return -* 1,...,8 = problem is found to be infeasible -* -* NOTE 1 -* -* To run in limited memory systems, declare the arrays startn, endn, -* nxtin, nxtou, fin, fou, label, prdcsr, save, tfstou, tnxtou, tfstin, -* tnxtin, ddpos, ddneg, nxtqueue as short instead. -* -* NOTE 2 -* -* This routine makes no effort to initialize with a favorable x from -* amongst those flow vectors that satisfy complementary slackness with -* the initial reduced cost vector rc. If a favorable x is known, then -* it can be passed, together with the corresponding arrays u and dfct, -* to this routine directly. This, however, requires that the capacity -* tightening portion and the flow initialization portion of this -* routine (up to line labeled 90) be skipped. -* -* NOTE 3 -* -* All problem data should be less than large in magnitude, and large -* should be less than, say, 1/4 the largest int of the machine used. -* This will guard primarily against overflow in uncapacitated problems -* where the arc capacities are taken finite but very large. Note, -* however, that as in all codes operating with integers, overflow may -* occur if some of the problem data takes very large values. -* -* NOTE 4 -* -* [This note being specific to Fortran was removed.-A.M.] -* -* NOTE 5 -* -* ddpos and ddneg are arrays that give the directional derivatives for -* all positive and negative single-node price changes. These are used -* only in phase II of the initialization procedure, before the linked -* list of balanced arcs comes to play. Therefore, to reduce storage, -* they are equivalence to tfstou and tfstin, which are of the same size -* (number of nodes) and are used only after the tree comes into use. */ - -static void ascnt1(struct relax4_csa *csa, int dm, int *delx, - int *nlabel, int *feasbl, int *svitch, int nscan, int curnode, - int *prevnode); - -static void ascnt2(struct relax4_csa *csa, int dm, int *delx, - int *nlabel, int *feasbl, int *svitch, int nscan, int curnode, - int *prevnode); - -static int auction(struct relax4_csa *csa); - -int relax4(struct relax4_csa *csa) -{ /* input parameters */ - int n = csa->n; - int na = csa->na; - int large = csa->large; - int repeat = csa->repeat; - int crash = csa->crash; - int *startn = csa->startn; - int *endn = csa->endn; - int *fou = csa->fou; - int *nxtou = csa->nxtou; - int *fin = csa->fin; - int *nxtin = csa->nxtin; - /* updated parameters */ - int *rc = csa->rc; - int *u = csa->u; - int *dfct = csa->dfct; - /* output parameters */ - int *x = csa->x; -# define nmultinode (csa->nmultinode) -# define iter (csa->iter) -# define num_augm (csa->num_augm) -# define num_ascnt (csa->num_ascnt) -# define nsp (csa->nsp) - /* working parameters */ - int *label = csa->label; - int *prdcsr = csa->prdcsr; - int *save = csa->save; - int *tfstou = csa->tfstou; - int *tnxtou = csa->tnxtou; - int *tfstin = csa->tfstin; - int *tnxtin = csa->tnxtin; - int *nxtqueue = csa->nxtqueue; - char *scan = csa->scan; - char *mark = csa->mark; - int *ddpos = tfstou; - int *ddneg = tfstin; - /* local variables */ - int arc, augnod, capin, capout, defcit, delprc, delx, dm, dp, - dx, feasbl, i, ib, indef, j, lastqueue, maxcap, narc, nb, - nlabel, node, node2, node_def, naugnod, nscan, num_passes, - numnz, numnz_new, numpasses, nxtarc, nxtbrk, nxtnode, passes, - pchange, posit, prevnode, prvarc, quit, rdcost, scapin, - scapou, svitch, t, t1, t2, tmparc, tp, trc, ts; - /*--------------------------------------------------------------*/ - /* Initialization phase I */ - /* In this phase, we reduce the arc capacities by as much as - * possible without changing the problem; then we set the initial - * flow array x, together with the corresponding arrays u and - * dfct. */ - /* This phase and phase II (from here up to line labeled 90) can - * be skipped (by setting repeat to true) if the calling program - * places in common user-chosen values for the arc flows, the - * residual arc capacities, and the nodal deficits. When this is - * done, it is critical that the flow and the reduced cost for - * each arc satisfy complementary slackness and the dfct array - * properly correspond to the initial arc/flows. */ - if (repeat) - goto L90; - for (node = 1; node <= n; node++) - { node_def = dfct[node]; - ddpos[node] = node_def; - ddneg[node] = -node_def; - maxcap = 0; - scapou = 0; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { if (scapou <= large - u[arc]) - scapou += u[arc]; - else - goto L10; - } - if (scapou <= large - node_def) - capout = scapou + node_def; - else - goto L10; - if (capout < 0) - { /* problem is infeasible */ - /* exogenous flow into node exceeds out capacity */ - return 1; - } - scapin = 0; - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { if (u[arc] > capout) - u[arc] = capout; - if (maxcap < u[arc]) - maxcap = u[arc]; - if (scapin <= large - u[arc]) - scapin += u[arc]; - else - goto L10; - } - if (scapin <= large + node_def) - capin = scapin - node_def; - else - goto L10; - if (capin < 0) - { /* problem is infeasible */ - /* exogenous flow out of node exceeds in capacity */ - return 2; - } - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { if (u[arc] > capin) - u[arc] = capin; - } -L10: ; - } - /*--------------------------------------------------------------*/ - /* Initialization phase II */ - /* In this phase, we initialize the prices and flows by either - * calling the routine auction or by performing only single node - * (coordinate) relaxation iterations. */ - if (crash == 1) - { nsp = 0; - if (auction(csa) != 0) - { /* problem is found to be infeasible */ - return 3; - } - goto L70; - } - /* Initialize the arc flows to satisfy complementary slackness - * with the prices. u[arc] is the residual capacity of arc, and - * x[arc] is the flow. These two always add up to the total - * capacity for arc. Also compute the directional derivatives for - * each coordinate and compute the actual deficits. */ - for (arc = 1; arc <= na; arc++) - { x[arc] = 0; - if (rc[arc] <= 0) - { t = u[arc]; - t1 = startn[arc]; - t2 = endn[arc]; - ddpos[t1] += t; - ddneg[t2] += t; - if (rc[arc] < 0) - { x[arc] = t; - u[arc] = 0; - dfct[t1] += t; - dfct[t2] -= t; - ddneg[t1] -= t; - ddpos[t2] -= t; - } - } - } - /* Make 2 or 3 passes through all nodes, performing only single - * node relaxation iterations. The number of passes depends on the - * density of the network. */ - if (na > n * 10) - numpasses = 2; - else - numpasses = 3; - for (passes = 1; passes <= numpasses; passes++) - for (node = 1; node <= n; node++) - { if (dfct[node] == 0) - continue; - if (ddpos[node] <= 0) - { /* Compute delprc, the stepsize to the next breakpoint in - * the dual cost as the price of node is increased. - * [Since the reduced cost of all outgoing (resp., incoming) - * arcs will decrease (resp., increase) as the price of node - * is increased, the next breakpoint is the minimum of the - * positive reduced cost on outgoing arcs and of the - * negative reduced cost on incoming arcs.] */ - delprc = large; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { trc = rc[arc]; - if ((trc > 0) && (trc < delprc)) - delprc = trc; - } - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { trc = rc[arc]; - if ((trc < 0) && (trc > -delprc)) - delprc = -trc; - } - /* If no breakpoint is left and dual ascent is still - * possible, the problem is infeasible. */ - if (delprc >= large) - { if (ddpos[node] == 0) - continue; - return 4; - } - /* delprc is the stepsize to next breakpoint. Increase - * price of node by delprc and compute the stepsize to the - * next breakpoint in the dual cost. */ -L53: nxtbrk = large; - /* Look at all arcs out of node. */ - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { trc = rc[arc]; - if (trc == 0) - { t1 = endn[arc]; - t = u[arc]; - if (t > 0) - { dfct[node] += t; - dfct[t1] -= t; - x[arc] = t; - u[arc] = 0; - } - else - t = x[arc]; - ddneg[node] -= t; - ddpos[t1] -= t; - } - /* Decrease the reduced costs on all outgoing arcs. */ - trc -= delprc; - if ((trc > 0) && (trc < nxtbrk)) - nxtbrk = trc; - else if (trc == 0) - { /* Arc goes from inactive to balanced. Update the rate - * of dual ascent at node and at its neighbor. */ - ddpos[node] += u[arc]; - ddneg[endn[arc]] += u[arc]; - } - rc[arc] = trc; - } - /* Look at all arcs into node. */ - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { trc = rc[arc]; - if (trc == 0) - { t1 = startn[arc]; - t = x[arc]; - if (t > 0) - { dfct[node] += t; - dfct[t1] -= t; - u[arc] = t; - x[arc] = 0; - } - else - t = u[arc]; - ddpos[t1] -= t; - ddneg[node] -= t; - } - /* Increase the reduced cost on all incoming arcs. */ - trc += delprc; - if ((trc < 0) && (trc > -nxtbrk)) - nxtbrk = -trc; - else if (trc == 0) - { /* Arc goes from active to balanced. Update the rate - * of dual ascent at node and at its neighbor. */ - ddneg[startn[arc]] += x[arc]; - ddpos[node] += x[arc]; - } - rc[arc] = trc; - } - /* If price of node can be increased further without - * decreasing the dual cost (even the dual cost doesn't - * increase), return to increase the price further. */ - if ((ddpos[node] <= 0) && (nxtbrk < large)) - { delprc = nxtbrk; - goto L53; - } - } - else if (ddneg[node] <= 0) - { /* Compute delprc, the stepsize to the next breakpoint in - * the dual cost as the price of node is decreased. - * [Since the reduced cost of all outgoing (resp., incoming) - * arcs will increase (resp., decrease) as the price of node - * is decreased, the next breakpoint is the minimum of the - * negative reduced cost on outgoing arcs and of the - * positive reduced cost on incoming arcs.] */ - delprc = large; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { trc = rc[arc]; - if ((trc < 0) && (trc > -delprc)) - delprc = -trc; - } - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { trc = rc[arc]; - if ((trc > 0) && (trc < delprc)) - delprc = trc; - } - /* If no breakpoint is left and dual ascent is still - * possible, the problem is infeasible. */ - if (delprc == large) - { if (ddneg[node] == 0) - continue; - return 5; - } - /* delprc is the stepsize to next breakpoint. Decrease - * price of node by delprc and compute the stepsize to the - * next breakpoint in the dual cost. */ -L63: nxtbrk = large; - /* Look at all arcs out of node. */ - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { trc = rc[arc]; - if (trc == 0) - { t1 = endn[arc]; - t = x[arc]; - if (t > 0) - { dfct[node] -= t; - dfct[t1] += t; - u[arc] = t; - x[arc] = 0; - } - else - t = u[arc]; - ddpos[node] -= t; - ddneg[t1] -= t; - } - /* Increase the reduced cost on all outgoing arcs. */ - trc += delprc; - if ((trc < 0) && (trc > -nxtbrk)) - nxtbrk = -trc; - else if (trc == 0) - { /* Arc goes from active to balanced. Update the rate - * of dual ascent at node and at its neighbor. */ - ddneg[node] += x[arc]; - ddpos[endn[arc]] += x[arc]; - } - rc[arc] = trc; - } - /* Look at all arcs into node. */ - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { trc = rc[arc]; - if (trc == 0) - { t1 = startn[arc]; - t = u[arc]; - if (t > 0) - { dfct[node] -= t; - dfct[t1] += t; - x[arc] = t; - u[arc] = 0; - } - else - t = x[arc]; - ddneg[t1] -= t; - ddpos[node] -= t; - } - /* Decrease the reduced cost on all incoming arcs. */ - trc -= delprc; - if ((trc > 0) && (trc < nxtbrk)) - nxtbrk = trc; - else if (trc == 0) - { /* Arc goes from inactive to balanced. Update the rate - * of dual ascent at node and at its neighbor. */ - ddpos[startn[arc]] += u[arc]; - ddneg[node] += u[arc]; - } - rc[arc] = trc; - } - /* If price of node can be decreased further without - * decreasing the dual cost (even the dual cost doesn't - * increase), return to decrease the price further. */ - if ((ddneg[node] <= 0) && (nxtbrk < large)) - { delprc = nxtbrk; - goto L63; - } - } - } - /*--------------------------------------------------------------*/ -L70: /* Initialize tree data structure. */ - for (i = 1; i <= n; i++) - tfstou[i] = tfstin[i] = 0; - for (i = 1; i <= na; i++) - { tnxtin[i] = tnxtou[i] = -1; - if (rc[i] == 0) - { tnxtou[i] = tfstou[startn[i]]; - tfstou[startn[i]] = i; - tnxtin[i] = tfstin[endn[i]]; - tfstin[endn[i]] = i; - } - } -L90: /* Initialize other variables. */ - feasbl = true; - iter = 0; - nmultinode = 0; - num_augm = 0; - num_ascnt = 0; - num_passes = 0; - numnz = n; - numnz_new = 0; - svitch = false; - for (i = 1; i <= n; i++) - mark[i] = scan[i] = false; - nlabel = 0; - /* RELAX4 uses an adaptive strategy to decide whether to continue - * the scanning process after a multinode price change. - * The threshold parameter tp and ts that control this strategy - * are set in the next two lines. */ - tp = 10; - ts = n / 15; - /* Initialize the queue of nodes with nonzero deficit. */ - for (node = 1; node <= n - 1; node++) - nxtqueue[node] = node + 1; - nxtqueue[n] = 1; - node = lastqueue = n; - /*--------------------------------------------------------------*/ - /* Start the relaxation algorithm. */ -L100: /* Code for advancing the queue of nonzero deficit nodes. */ - prevnode = node; - node = nxtqueue[node]; - defcit = dfct[node]; - if (node == lastqueue) - { numnz = numnz_new; - numnz_new = 0; - lastqueue = prevnode; - num_passes++; - } - /* Code for deleting a node from the queue. */ - if (defcit == 0) - { nxtnode = nxtqueue[node]; - if (node == nxtnode) - return 0; - else - { nxtqueue[prevnode] = nxtnode; - nxtqueue[node] = 0; - node = nxtnode; - goto L100; - } - } - else - posit = (defcit > 0); - iter++; - numnz_new++; - if (posit) - { /* Attempt a single node iteration from node with positive - * deficit. */ - pchange = false; - indef = defcit; - delx = 0; - nb = 0; - /* Check outgoing (probably) balanced arcs from node. */ - for (arc = tfstou[node]; arc > 0; arc = tnxtou[arc]) - { if ((rc[arc] == 0) && (x[arc] > 0)) - { delx += x[arc]; - nb++; - save[nb] = arc; - } - } - /* Check incoming arcs. */ - for (arc = tfstin[node]; arc > 0; arc = tnxtin[arc]) - { if ((rc[arc] == 0) && (u[arc] > 0)) - { delx += u[arc]; - nb++; - save[nb] = -arc; - } - } - /* End of initial node scan. */ -L4018: /* If no price change is possible, exit. */ - if (delx > defcit) - { quit = (defcit < indef); - goto L4016; - } - /* RELAX4 searches along the ascent direction for the best - * price by checking the slope of the dual cost at successive - * break points. First, we compute the distance to the next - * break point. */ - delprc = large; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { rdcost = rc[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { rdcost = rc[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - /* Check if problem is infeasible. */ - if ((delx < defcit) && (delprc == large)) - { /* The dual cost can be decreased without bound. */ - return 6; - } - /* Skip flow adjustment if there is no flow to modify. */ - if (delx == 0) - goto L4014; - /* Adjust the flow on the balanced arcs incident to node to - * maintain complementary slackness after the price change. */ - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc > 0) - { node2 = endn[arc]; - t1 = x[arc]; - dfct[node2] += t1; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - u[arc] += t1; - x[arc] = 0; - } - else - { narc = -arc; - node2 = startn[narc]; - t1 = u[narc]; - dfct[node2] += t1; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - x[narc] += t1; - u[narc] = 0; - } - } - defcit -= delx; -L4014: if (delprc == large) - { quit = true; - goto L4019; - } - /* Node corresponds to a dual ascent direction. Decrease the - * price of node by delprc and compute the stepsize to the next - * breakpoint in the dual cost. */ - nb = 0; - pchange = true; - dp = delprc; - delprc = large; - delx = 0; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { rdcost = rc[arc] + dp; - rc[arc] = rdcost; - if (rdcost == 0) - { nb++; - save[nb] = arc; - delx += x[arc]; - } - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { rdcost = rc[arc] - dp; - rc[arc] = rdcost; - if (rdcost == 0) - { nb++; - save[nb] = -arc; - delx += u[arc]; - } - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - /* Return to check if another price change is possible. */ - goto L4018; -L4016: /* Perform flow augmentation at node. */ - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc > 0) - { /* arc is an outgoing arc from node. */ - node2 = endn[arc]; - t1 = dfct[node2]; - if (t1 < 0) - { /* Decrease the total deficit by decreasing flow of - * arc. */ - quit = true; - t2 = x[arc]; - dx = defcit; - if (dx > -t1) dx = -t1; - if (dx > t2) dx = t2; - defcit -= dx; - dfct[node2] = t1 + dx; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - x[arc] = t2 - dx; - u[arc] += dx; - if (defcit == 0) - break; - } - } - else - { /* -arc is an incoming arc to node. */ - narc = -arc; - node2 = startn[narc]; - t1 = dfct[node2]; - if (t1 < 0) - { /* Decrease the total deficit by increasing flow of - * -arc. */ - quit = true; - t2 = u[narc]; - dx = defcit; - if (dx > -t1) dx = -t1; - if (dx > t2) dx = t2; - defcit -= dx; - dfct[node2] = t1 + dx; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - x[narc] += dx; - u[narc] = t2 - dx; - if (defcit == 0) - break; - } - } - } -L4019: dfct[node] = defcit; - /* Reconstruct the linked list of balance arcs incident to this - * node. For each adjacent node, we add any newly balanced arcs - * to the list, but do not bother removing formerly balanced - * ones (they will be removed the next time each adjacent node - * is scanned). */ - if (pchange) - { arc = tfstou[node]; - tfstou[node] = 0; - while (arc > 0) - { nxtarc = tnxtou[arc]; - tnxtou[arc] = -1; - arc = nxtarc; - } - arc = tfstin[node]; - tfstin[node] = 0; - while (arc > 0) - { nxtarc = tnxtin[arc]; - tnxtin[arc] = -1; - arc = nxtarc; - } - /* Now add the currently balanced arcs to the list for this - * node (which is now empty), and the appropriate adjacent - * ones. */ - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc < 0) - arc = -arc; - if (tnxtou[arc] < 0) - { tnxtou[arc] = tfstou[startn[arc]]; - tfstou[startn[arc]] = arc; - } - if (tnxtin[arc] < 0) - { tnxtin[arc] = tfstin[endn[arc]]; - tfstin[endn[arc]] = arc; - } - } - } - /* End of single node iteration for positive deficit node. */ - } - else - { /* Attempt a single node iteration from node with negative - * deficit. */ - pchange = false; - defcit = -defcit; - indef = defcit; - delx = 0; - nb = 0; - for (arc = tfstin[node]; arc > 0; arc = tnxtin[arc]) - { if ((rc[arc] == 0) && (x[arc] > 0)) - { delx += x[arc]; - nb++; - save[nb] = arc; - } - } - for (arc = tfstou[node]; arc > 0; arc = tnxtou[arc]) - { if ((rc[arc] == 0) && (u[arc] > 0)) - { delx += u[arc]; - nb++; - save[nb] = -arc; - } - } -L4028: if (delx >= defcit) - { quit = (defcit < indef); - goto L4026; - } - /* Compute distance to next breakpoint. */ - delprc = large; - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { rdcost = rc[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { rdcost = rc[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - /* Check if problem is infeasible. */ - if ((delx < defcit) && (delprc == large)) - return 7; - if (delx == 0) - goto L4024; - /* Flow augmentation is possible. */ - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc > 0) - { node2 = startn[arc]; - t1 = x[arc]; - dfct[node2] -= t1; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - u[arc] += t1; - x[arc] = 0; - } - else - { narc = -arc; - node2 = endn[narc]; - t1 = u[narc]; - dfct[node2] -= t1; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - x[narc] += t1; - u[narc] = 0; - } - } - defcit -= delx; -L4024: if (delprc == large) - { quit = true; - goto L4029; - } - /* Price increase at node is possible. */ - nb = 0; - pchange = true; - dp = delprc; - delprc = large; - delx = 0; - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { rdcost = rc[arc] + dp; - rc[arc] = rdcost; - if (rdcost == 0) - { nb++; - save[nb] = arc; - delx += x[arc]; - } - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { rdcost = rc[arc] - dp; - rc[arc] = rdcost; - if (rdcost == 0) - { nb++; - save[nb] = -arc; - delx += u[arc]; - } - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - goto L4028; -L4026: /* Perform flow augmentation at node. */ - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc > 0) - { /* arc is an incoming arc to node. */ - node2 = startn[arc]; - t1 = dfct[node2]; - if (t1 > 0) - { quit = true; - t2 = x[arc]; - dx = defcit; - if (dx > t1) dx = t1; - if (dx > t2) dx = t2; - defcit -= dx; - dfct[node2] = t1 - dx; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - x[arc] = t2 - dx; - u[arc] += dx; - if (defcit == 0) - break; - } - } - else - { /* -arc is an outgoing arc from node. */ - narc = -arc; - node2 = endn[narc]; - t1 = dfct[node2]; - if (t1 > 0) - { quit = true; - t2 = u[narc]; - dx = defcit; - if (dx > t1) dx = t1; - if (dx > t2) dx = t2; - defcit -= dx; - dfct[node2] = t1 - dx; - if (nxtqueue[node2] == 0) - { nxtqueue[prevnode] = node2; - nxtqueue[node2] = node; - prevnode = node2; - } - x[narc] += dx; - u[narc] = t2 - dx; - if (defcit == 0) - break; - } - } - } -L4029: dfct[node] = -defcit; - /* Reconstruct the list of balanced arcs incident to node. */ - if (pchange) - { arc = tfstou[node]; - tfstou[node] = 0; - while (arc > 0) - { nxtarc = tnxtou[arc]; - tnxtou[arc] = -1; - arc = nxtarc; - } - arc = tfstin[node]; - tfstin[node] = 0; - while (arc > 0) - { nxtarc = tnxtin[arc]; - tnxtin[arc] = -1; - arc = nxtarc; - } - /* Now add the currently balanced arcs to the list for this - * node (which is now empty), and the appropriate adjacent - * ones. */ - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc <= 0) - arc = -arc; - if (tnxtou[arc] < 0) - { tnxtou[arc] = tfstou[startn[arc]]; - tfstou[startn[arc]] = arc; - } - if (tnxtin[arc] < 0) - { tnxtin[arc] = tfstin[endn[arc]]; - tfstin[endn[arc]] = arc; - } - } - } - /* End of single node iteration for a negative deficit node. */ - } - if (quit || (num_passes <= 3)) - goto L100; - /* Do a multinode iteration from node. */ - nmultinode++; - /* If number of nonzero deficit nodes is small, continue labeling - * until a flow augmentation is done. */ - svitch = (numnz < tp); - /* Unmark nodes labeled earlier. */ - for (j = 1; j <= nlabel; j++) - { node2 = label[j]; - mark[node2] = scan[node2] = false; - } - /* Initialize labeling. */ - nlabel = 1; - label[1] = node; - mark[node] = true; - prdcsr[node] = 0; - /* Scan starting node. */ - scan[node] = true; - nscan = 1; - dm = dfct[node]; - delx = 0; - for (j = 1; j <= nb; j++) - { arc = save[j]; - if (arc > 0) - { if (posit) - node2 = endn[arc]; - else - node2 = startn[arc]; - if (!mark[node2]) - { nlabel++; - label[nlabel] = node2; - prdcsr[node2] = arc; - mark[node2] = true; - delx += x[arc]; - } - } - else - { narc = -arc; - if (posit) - node2 = startn[narc]; - else - node2 = endn[narc]; - if (!mark[node2]) - { nlabel++; - label[nlabel] = node2; - prdcsr[node2] = arc; - mark[node2] = true; - delx += u[narc]; - } - } - } -L4120:/* Start scanning a labeled but unscanned node. */ - nscan++; - /* Check to see if switch needs to be set to true so to continue - * scanning even after a price change. */ - svitch = svitch || ((nscan > ts) && (numnz < ts)); - /* Scanning will continue until either an overestimate of the - * residual capacity across the cut corresponding to the scanned - * set of nodes (called delx) exceeds the absolute value of the - * total deficit of the scanned nodes (called dm), or else an - * augmenting path is found. Arcs that are in the tree but are not - * balanced are removed as part of the scanning process. */ - i = label[nscan]; - scan[i] = true; - naugnod = 0; - if (posit) - { /* Scanning node i in case of positive deficit. */ - prvarc = 0; - arc = tfstou[i]; - while (arc > 0) - { /* arc is an outgoing arc from node. */ - if (rc[arc] == 0) - { if (x[arc] > 0) - { node2 = endn[arc]; - if (!mark[node2]) - { /* node2 is not labeled, so add node2 to the - labeled set. */ - prdcsr[node2] = arc; - if (dfct[node2] < 0) - { naugnod++; - save[naugnod] = node2; - } - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - delx += x[arc]; - } - } - prvarc = arc; - arc = tnxtou[arc]; - } - else - { tmparc = arc; - arc = tnxtou[arc]; - tnxtou[tmparc] = -1; - if (prvarc == 0) - tfstou[i] = arc; - else - tnxtou[prvarc] = arc; - } - } - prvarc = 0; - arc = tfstin[i]; - while (arc > 0) - { /* arc is an incoming arc into node. */ - if (rc[arc] == 0) - { if (u[arc] > 0) - { node2 = startn[arc]; - if (!mark[node2]) - { /* node2 is not labeled, so add node2 to the - * labeled set. */ - prdcsr[node2] = -arc; - if (dfct[node2] < 0) - { naugnod++; - save[naugnod] = node2; - } - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - delx += u[arc]; - } - } - prvarc = arc; - arc = tnxtin[arc]; - } - else - { tmparc = arc; - arc = tnxtin[arc]; - tnxtin[tmparc] = -1; - if (prvarc == 0) - tfstin[i] = arc; - else - tnxtin[prvarc] = arc; - } - } - /* Correct the residual capacity of the scanned node cut. */ - arc = prdcsr[i]; - if (arc > 0) - delx -= x[arc]; - else - delx -= u[-arc]; - /* End of scanning of node i for positive deficit case. */ - } - else - { /* Scanning node i for negative deficit case. */ - prvarc = 0; - arc = tfstin[i]; - while (arc > 0) - { if (rc[arc] == 0) - { if (x[arc] > 0) - { node2 = startn[arc]; - if (!mark[node2]) - { prdcsr[node2] = arc; - if (dfct[node2] > 0) - { naugnod++; - save[naugnod] = node2; - } - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - delx += x[arc]; - } - } - prvarc = arc; - arc = tnxtin[arc]; - } - else - { tmparc = arc; - arc = tnxtin[arc]; - tnxtin[tmparc] = -1; - if (prvarc == 0) - tfstin[i] = arc; - else - tnxtin[prvarc] = arc; - } - } - prvarc = 0; - arc = tfstou[i]; - while (arc > 0) - { if (rc[arc] == 0) - { if (u[arc] > 0) - { node2 = endn[arc]; - if (!mark[node2]) - { prdcsr[node2] = -arc; - if (dfct[node2] > 0) - { naugnod++; - save[naugnod] = node2; - } - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - delx += u[arc]; - } - } - prvarc = arc; - arc = tnxtou[arc]; - } - else - { tmparc = arc; - arc = tnxtou[arc]; - tnxtou[tmparc] = -1; - if (prvarc == 0) - tfstou[i] = arc; - else - tnxtou[prvarc] = arc; - } - } - arc = prdcsr[i]; - if (arc > 0) - delx -= x[arc]; - else - delx -= u[-arc]; - } - /* Add deficit of node scanned to dm. */ - dm += dfct[i]; - /* Check if the set of scanned nodes correspond to a dual ascent - * direction; if yes, perform a price adjustment step, otherwise - * continue labeling. */ - if (nscan < nlabel) - { if (svitch) - goto L4210; - if ((delx >= dm) && (delx >= -dm)) - goto L4210; - } - /* Try a price change. - * [Note that since delx - abs(dm) is an overestimate of ascent - * slope, we may occasionally try a direction that is not an - * ascent direction. In this case the ascnt routines return with - * quit = false, so we continue labeling nodes.] */ - if (posit) - { ascnt1(csa, dm, &delx, &nlabel, &feasbl, &svitch, nscan, node, - &prevnode); - num_ascnt++; - } - else - { ascnt2(csa, dm, &delx, &nlabel, &feasbl, &svitch, nscan, node, - &prevnode); - num_ascnt++; - } - if (!feasbl) - return 8; - if (!svitch) - goto L100; - /* Store those newly labeled nodes to which flow augmentation is - * possible. */ - naugnod = 0; - for (j = nscan + 1; j <= nlabel; j++) - { node2 = label[j]; - if (posit && (dfct[node2] < 0)) - { naugnod++; - save[naugnod] = node2; - } - else if ((!posit) && (dfct[node2] > 0)) - { naugnod++; - save[naugnod] = node2; - } - } -L4210:/* Check if flow augmentation is possible. If not, return to scan - * another node. */ - if (naugnod == 0) - goto L4120; - for (j = 1; j <= naugnod; j++) - { num_augm++; - augnod = save[j]; - if (posit) - { /* Do the augmentation from node with positive deficit. */ - dx = -dfct[augnod]; - ib = augnod; - while (ib != node) - { arc = prdcsr[ib]; - if (arc > 0) - { if (dx > x[arc]) dx = x[arc]; - ib = startn[arc]; - } - else - { if (dx > u[-arc]) dx = u[-arc]; - ib = endn[-arc]; - } - } - if (dx > dfct[node]) dx = dfct[node]; - if (dx > 0) - { /* Increase (decrease) the flow of all forward (backward) - * arcs in the flow augmenting path. Adjust node deficit - * accordingly. */ - if (nxtqueue[augnod] == 0) - { nxtqueue[prevnode] = augnod; - nxtqueue[augnod] = node; - prevnode = augnod; - } - dfct[augnod] += dx; - dfct[node] -= dx; - ib = augnod; - while (ib != node) - { arc = prdcsr[ib]; - if (arc > 0) - { x[arc] -= dx; - u[arc] += dx; - ib = startn[arc]; - } - else - { narc = -arc; - x[narc] += dx; - u[narc] -= dx; - ib = endn[narc]; - } - } - } - } - else - { /* Do the augmentation from node with negative deficit. */ - dx = dfct[augnod]; - ib = augnod; - while (ib != node) - { arc = prdcsr[ib]; - if (arc > 0) - { if (dx > x[arc]) dx = x[arc]; - ib = endn[arc]; - } - else - { if (dx > u[-arc]) dx = u[-arc]; - ib = startn[-arc]; - } - } - if (dx > -dfct[node]) dx = -dfct[node]; - if (dx > 0) - { /* Update the flow and deficits. */ - if (nxtqueue[augnod] == 0) - { nxtqueue[prevnode] = augnod; - nxtqueue[augnod] = node; - prevnode = augnod; - } - dfct[augnod] -= dx; - dfct[node] += dx; - ib = augnod; - while (ib != node) - { arc = prdcsr[ib]; - if (arc > 0) - { x[arc] -= dx; - u[arc] += dx; - ib = endn[arc]; - } - else - { narc = -arc; - x[narc] += dx; - u[narc] -= dx; - ib = startn[narc]; - } - } - } - } - if (dfct[node] == 0) - goto L100; - if (dfct[augnod] != 0) - svitch = false; - } - /* If node still has nonzero deficit and all newly labeled nodes - * have same sign for their deficit as node, we can continue - * labeling. In this case, continue labeling only when flow - * augmentation is done relatively infrequently. */ - if (svitch && (iter > 8 * num_augm)) - goto L4120; - /* Return to do another relaxation iteration. */ - goto L100; -# undef nmultinode -# undef iter -# undef num_augm -# undef num_ascnt -# undef nsp -} - -/*********************************************************************** -* NAME -* -* relax4_inidat - construct linked lists for network topology -* -* PURPOSE -* -* This routine constructs two linked lists for the network topology: -* one list (given by fou, nxtou) for the outgoing arcs of nodes and -* one list (given by fin, nxtin) for the incoming arcs of nodes. These -* two lists are required by RELAX4. -* -* INPUT PARAMETERS -* -* n = number of nodes -* na = number of arcs -* startn[j] = starting node for arc j, j = 1,...,na -* endn[j] = ending node for arc j, j = 1,...,na -* -* OUTPUT PARAMETERS -* -* fou[i] = first arc out of node i, i = 1,...,n -* nxtou[j] = next arc out of the starting node of arc j, j = 1,...,na -* fin[i] = first arc into node i, i = 1,...,n -* nxtin[j] = next arc into the ending node of arc j, j = 1,...,na -* -* WORKING PARAMETERS -* -* tempin[1+n], tempou[1+n] */ - -void relax4_inidat(struct relax4_csa *csa) -{ /* input parameters */ - int n = csa->n; - int na = csa->na; - int *startn = csa->startn; - int *endn = csa->endn; - /* output parameters */ - int *fou = csa->fou; - int *nxtou = csa->nxtou; - int *fin = csa->fin; - int *nxtin = csa->nxtin; - /* working parameters */ - int *tempin = csa->label; - int *tempou = csa->prdcsr; - /* local variables */ - int i, i1, i2; - for (i = 1; i <= n; i++) - { fin[i] = fou[i] = 0; - tempin[i] = tempou[i] = 0; - } - for (i = 1; i <= na; i++) - { nxtin[i] = nxtou[i] = 0; - i1 = startn[i]; - i2 = endn[i]; - if (fou[i1] != 0) - nxtou[tempou[i1]] = i; - else - fou[i1] = i; - tempou[i1] = i; - if (fin[i2] != 0) - nxtin[tempin[i2]] = i; - else - fin[i2] = i; - tempin[i2] = i; - } - return; -} - -/*********************************************************************** -* NAME -* -* ascnt1 - multi-node price adjustment for positive deficit case -* -* PURPOSE -* -* This subroutine performs the multi-node price adjustment step for -* the case where the scanned nodes have positive deficit. It first -* checks if decreasing the price of the scanned nodes increases the -* dual cost. If yes, then it decreases the price of all scanned nodes. -* There are two possibilities for price decrease: if switch = true, -* then the set of scanned nodes corresponds to an elementary direction -* of maximal rate of ascent, in which case the price of all scanned -* nodes are decreased until the next breakpoint in the dual cost is -* encountered. At this point, some arc becomes balanced and more -* node(s) are added to the labeled set and the subroutine is exited. -* If switch = false, then the price of all scanned nodes are decreased -* until the rate of ascent becomes negative (this corresponds to the -* price adjustment step in which both the line search and the -* degenerate ascent iteration are implemented). -* -* INPUT PARAMETERS -* -* dm = total deficit of scanned nodes -* switch = true if labeling is to continue after price change -* nscan = number of scanned nodes -* curnode = most recently scanned node -* n = number of nodes -* na = number of arcs -* large = a very large integer to represent infinity (see note 3) -* startn[i] = starting node for the i-th arc, i = 1,...,na -* endn[i] = ending node for the i-th arc, i = 1,...,na -* fou[i] = first arc leaving i-th node, i = 1,...,n -* nxtou[i] = next arc leaving the starting node of j-th arc, -* i = 1,...,na -* fin[i] = first arc entering i-th node, i = 1,...,n -* nxtin[i] = next arc entering the ending node of j-th arc, -* i = 1,...,na -* -* UPDATED PARAMETERS -* -* delx = a lower estimate of the total flow on balanced arcs in -* the scanned-nodes cut -* nlabel = number of labeled nodes -* feasbl = false if problem is found to be infeasible -* prevnode = the node before curnode in queue -* rc[j] = reduced cost of arc j, j = 1,...,na -* u[j] = residual capacity of arc j, j = 1,...,na -* x[j] = flow on arc j, j = 1,...,na -* dfct[i] = deficit at node i, i = 1,...,n -* label[k] = k-th node labeled, k = 1,...,nlabel -* prdcsr[i] = predecessor of node i in tree of labeled nodes (0 if i -* is unlabeled), i = 1,...,n -* tfstou[i] = first balanced arc out of node i, i = 1,...,n -* tnxtou[j] = next balanced arc out of the starting node of arc j, -* j = 1,...,na -* tfstin[i] = first balanced arc into node i, i = 1,...,n -* tnxtin[j] = next balanced arc into the ending node of arc j, -* j = 1,...,na -* nxtqueue[i] = node following node i in the fifo queue (0 if node is -* not in the queue), i = 1,...,n -* scan[i] = true if node i is scanned, i = 1,...,n -* mark[i] = true if node i is labeled, i = 1,...,n -* -* WORKING PARAMETERS -* -* save[1+na] */ - -static void ascnt1(struct relax4_csa *csa, int dm, int *delx, - int *nlabel, int *feasbl, int *svitch, int nscan, int curnode, - int *prevnode) -{ /* input parameters */ - int n = csa->n; - /* int na = csa->na; */ - int large = csa->large; - int *startn = csa->startn; - int *endn = csa->endn; - int *fou = csa->fou; - int *nxtou = csa->nxtou; - int *fin = csa->fin; - int *nxtin = csa->nxtin; - /* updated parameters */ -# define delx (*delx) -# define nlabel (*nlabel) -# define feasbl (*feasbl) -# define svitch (*svitch) -# define prevnode (*prevnode) - int *rc = csa->rc; - int *u = csa->u; - int *x = csa->x; - int *dfct = csa->dfct; - int *label = csa->label; - int *prdcsr = csa->prdcsr; - int *tfstou = csa->tfstou; - int *tnxtou = csa->tnxtou; - int *tfstin = csa->tfstin; - int *tnxtin = csa->tnxtin; - int *nxtqueue = csa->nxtqueue; - char *scan = csa->scan; - char *mark = csa->mark; - int *save = csa->save; - /* local variables */ - int arc, delprc, dlx, i, j, nb, node, node2, nsave, rdcost, t1, - t2, t3; - /* Store the arcs between the set of scanned nodes and its - * complement in save and compute delprc, the stepsize to the next - * breakpoint in the dual cost in the direction of decreasing - * prices of the scanned nodes. - * [The arcs are stored into save by looking at the arcs incident - * to either the set of scanned nodes or its complement, depending - * on whether nscan > n/2 or not. This improves the efficiency of - * storing.] */ - delprc = large; - dlx = 0; - nsave = 0; - if (nscan <= n / 2) - { for (i = 1; i <= nscan; i++) - { node = label[i]; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { /* arc points from scanned node to an unscanned node. */ - node2 = endn[arc]; - if (!scan[node2]) - { nsave++; - save[nsave] = arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node2] != arc)) - dlx += x[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - } - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { /* arc points from unscanned node to scanned node. */ - node2 = startn[arc]; - if (!scan[node2]) - { nsave++; - save[nsave] = -arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node2] != -arc)) - dlx += u[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - } - } - } - else - { for (node = 1; node <= n; node++) - { if (scan[node]) - continue; - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { node2 = startn[arc]; - if (scan[node2]) - { nsave++; - save[nsave] = arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node] != arc)) - dlx += x[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - } - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { node2 = endn[arc]; - if (scan[node2]) - { nsave++; - save[nsave] = -arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node] != -arc)) - dlx += u[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - } - } - } - /* Check if the set of scanned nodes truly corresponds to a dual - * ascent direction. [Here delx + dlx is the exact sum of the flow - * on arcs from the scanned set to the unscanned set plus the - * (capacity - flow) on arcs from the unscanned set to the scanned - * set.] If this were not the case, set switch to true and exit - * subroutine. */ - if ((!svitch) && (delx + dlx >= dm)) - { svitch = true; - return; - } - delx += dlx; -L4: /* Check that the problem is feasible. */ - if (delprc == large) - { /* We can increase the dual cost without bound, so the primal - * problem is infeasible. */ - feasbl = false; - return; - } - /* Decrease the prices of the scanned nodes, add more nodes to - * the labeled set and check if a newly labeled node has negative - * deficit. */ - if (svitch) - { for (i = 1; i <= nsave; i++) - { arc = save[i]; - if (arc > 0) - { rc[arc] += delprc; - if (rc[arc] == 0) - { node2 = endn[arc]; - if (tnxtou[arc] < 0) - { tnxtou[arc] = tfstou[startn[arc]]; - tfstou[startn[arc]] = arc; - } - if (tnxtin[arc] < 0) - { tnxtin[arc] = tfstin[node2]; - tfstin[node2] = arc; - } - if (!mark[node2]) - { prdcsr[node2] = arc; - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - } - } - } - else - { arc = -arc; - rc[arc] -= delprc; - if (rc[arc] == 0) - { node2 = startn[arc]; - if (tnxtou[arc] < 0) - { tnxtou[arc] = tfstou[node2]; - tfstou[node2] = arc; - } - if (tnxtin[arc] < 0) - { tnxtin[arc] = tfstin[endn[arc]]; - tfstin[endn[arc]] = arc; - } - if (!mark[node2]) - { prdcsr[node2] = -arc; - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - } - } - } - } - return; - } - else - { /* Decrease the prices of the scanned nodes by delprc. Adjust - * flow to maintain complementary slackness with the prices. */ - nb = 0; - for (i = 1; i <= nsave; i++) - { arc = save[i]; - if (arc > 0) - { t1 = rc[arc]; - if (t1 == 0) - { t2 = x[arc]; - t3 = startn[arc]; - dfct[t3] -= t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - t3 = endn[arc]; - dfct[t3] += t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - u[arc] += t2; - x[arc] = 0; - } - rc[arc] = t1 + delprc; -#if 0 /* by mao; 26/IV-2013 */ - if (rc[arc] == 0) -#else - if (rc[arc] == 0 && nb < n) -#endif - { delx += x[arc]; - nb++; - prdcsr[nb] = arc; - } - } - else - { arc = -arc; - t1 = rc[arc]; - if (t1 == 0) - { t2 = u[arc]; - t3 = startn[arc]; - dfct[t3] += t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - t3 = endn[arc]; - dfct[t3] -= t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - x[arc] += t2; - u[arc] = 0; - } - rc[arc] = t1 - delprc; -#if 0 /* by mao; 26/IV-2013 */ - if (rc[arc] == 0) -#else - if (rc[arc] == 0 && nb < n) -#endif - { delx += u[arc]; - nb++; - prdcsr[nb] = arc; - } - } - } - } - if (delx <= dm) - { /* The set of scanned nodes still corresponds to a dual - * (possibly degenerate) ascent direction. Compute the stepsize - * delprc to the next breakpoint in the dual cost. */ - delprc = large; - for (i = 1; i <= nsave; i++) - { arc = save[i]; - if (arc > 0) - { rdcost = rc[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - else - { arc = -arc; - rdcost = rc[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - } - if ((delprc != large) || (delx < dm)) - goto L4; - } - /* Add new balanced arcs to the superset of balanced arcs. */ - for (i = 1; i <= nb; i++) - { arc = prdcsr[i]; - if (tnxtin[arc] == -1) - { j = endn[arc]; - tnxtin[arc] = tfstin[j]; - tfstin[j] = arc; - } - if (tnxtou[arc] == -1) - { j = startn[arc]; - tnxtou[arc] = tfstou[j]; - tfstou[j] = arc; - } - } - return; -# undef delx -# undef nlabel -# undef feasbl -# undef svitch -# undef prevnode -} - -/*********************************************************************** -* NAME -* -* ascnt2 - multi-node price adjustment for negative deficit case -* -* PURPOSE -* -* This routine is analogous to ascnt1 but for the case where the -* scanned nodes have negative deficit. */ - -static void ascnt2(struct relax4_csa *csa, int dm, int *delx, - int *nlabel, int *feasbl, int *svitch, int nscan, int curnode, - int *prevnode) -{ /* input parameters */ - int n = csa->n; - /* int na = csa->na; */ - int large = csa->large; - int *startn = csa->startn; - int *endn = csa->endn; - int *fou = csa->fou; - int *nxtou = csa->nxtou; - int *fin = csa->fin; - int *nxtin = csa->nxtin; - /* updated parameters */ -# define delx (*delx) -# define nlabel (*nlabel) -# define feasbl (*feasbl) -# define svitch (*svitch) -# define prevnode (*prevnode) - int *rc = csa->rc; - int *u = csa->u; - int *x = csa->x; - int *dfct = csa->dfct; - int *label = csa->label; - int *prdcsr = csa->prdcsr; - int *tfstou = csa->tfstou; - int *tnxtou = csa->tnxtou; - int *tfstin = csa->tfstin; - int *tnxtin = csa->tnxtin; - int *nxtqueue = csa->nxtqueue; - char *scan = csa->scan; - char *mark = csa->mark; - int *save = csa->save; - /* local variables */ - int arc, delprc, dlx, i, j, nb, node, node2, nsave, rdcost, t1, - t2, t3; - /* Store the arcs between the set of scanned nodes and its - * complement in save and compute delprc, the stepsize to the next - * breakpoint in the dual cost in the direction of increasing - * prices of the scanned nodes. */ - delprc = large; - dlx = 0; - nsave = 0; - if (nscan <= n / 2) - { for (i = 1; i <= nscan; i++) - { node = label[i]; - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { node2 = startn[arc]; - if (!scan[node2]) - { nsave++; - save[nsave] = arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node2] != arc)) - dlx += x[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - } - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { node2 = endn[arc]; - if (!scan[node2]) - { nsave++; - save[nsave] = -arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node2] != -arc)) - dlx += u[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - } - } - } - else - { for (node = 1; node <= n; node++) - { if (scan[node]) - continue; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { node2 = endn[arc]; - if (scan[node2]) - { nsave++; - save[nsave] = arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node] != arc)) - dlx += x[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - } - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { node2 = startn[arc]; - if (scan[node2]) - { nsave++; - save[nsave] = -arc; - rdcost = rc[arc]; - if ((rdcost == 0) && (prdcsr[node] != -arc)) - dlx += u[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - } - } - } - if ((!svitch) && (delx + dlx >= -dm)) - { svitch = true; - return; - } - delx += dlx; - /* Check that the problem is feasible. */ -L4: if (delprc == large) - { feasbl = false; - return; - } - /* Increase the prices of the scanned nodes, add more nodes to - * the labeled set and check if a newly labeled node has positive - * deficit. */ - if (svitch) - { for (i = 1; i <= nsave; i++) - { arc = save[i]; - if (arc > 0) - { rc[arc] += delprc; - if (rc[arc] == 0) - { node2 = startn[arc]; - if (tnxtou[arc] < 0) - { tnxtou[arc] = tfstou[node2]; - tfstou[node2] = arc; - } - if (tnxtin[arc] < 0) - { tnxtin[arc] = tfstin[endn[arc]]; - tfstin[endn[arc]] = arc; - } - if (!mark[node2]) - { prdcsr[node2] = arc; - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - } - } - } - else - { arc = -arc; - rc[arc] -= delprc; - if (rc[arc] == 0) - { node2 = endn[arc]; - if (tnxtou[arc] < 0) - { tnxtou[arc] = tfstou[startn[arc]]; - tfstou[startn[arc]] = arc; - } - if (tnxtin[arc] < 0) - { tnxtin[arc] = tfstin[node2]; - tfstin[node2] = arc; - } - if (!mark[node2]) - { prdcsr[node2] = -arc; - nlabel++; - label[nlabel] = node2; - mark[node2] = true; - } - } - } - } - return; - } - else - { nb = 0; - for (i = 1; i <= nsave; i++) - { arc = save[i]; - if (arc > 0) - { t1 = rc[arc]; - if (t1 == 0) - { t2 = x[arc]; - t3 = startn[arc]; - dfct[t3] -= t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - t3 = endn[arc]; - dfct[t3] += t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - u[arc] += t2; - x[arc] = 0; - } - rc[arc] = t1 + delprc; -#if 0 /* by mao; 26/IV-2013 */ - if (rc[arc] == 0) -#else - if (rc[arc] == 0 && nb < n) -#endif - { delx += x[arc]; - nb++; - prdcsr[nb] = arc; - } - } - else - { arc = -arc; - t1 = rc[arc]; - if (t1 == 0) - { t2 = u[arc]; - t3 = startn[arc]; - dfct[t3] += t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - t3 = endn[arc]; - dfct[t3] -= t2; - if (nxtqueue[t3] == 0) - { nxtqueue[prevnode] = t3; - nxtqueue[t3] = curnode; - prevnode = t3; - } - x[arc] += t2; - u[arc] = 0; - } - rc[arc] = t1 - delprc; -#if 0 /* by mao; 26/IV-2013 */ - if (rc[arc] == 0) -#else - if (rc[arc] == 0 && nb < n) -#endif - { delx += u[arc]; - nb++; - prdcsr[nb] = arc; - } - } - } - } - if (delx <= -dm) - { delprc = large; - for (i = 1; i <= nsave; i++) - { arc = save[i]; - if (arc > 0) - { rdcost = rc[arc]; - if ((rdcost < 0) && (rdcost > -delprc)) - delprc = -rdcost; - } - else - { arc = -arc; - rdcost = rc[arc]; - if ((rdcost > 0) && (rdcost < delprc)) - delprc = rdcost; - } - } - if ((delprc != large) || (delx < -dm)) - goto L4; - } - /* Add new balanced arcs to the superset of balanced arcs. */ - for (i = 1; i <= nb; i++) - { arc = prdcsr[i]; - if (tnxtin[arc] == -1) - { j = endn[arc]; - tnxtin[arc] = tfstin[j]; - tfstin[j] = arc; - } - if (tnxtou[arc] == -1) - { j = startn[arc]; - tnxtou[arc] = tfstou[j]; - tfstou[j] = arc; - } - } - return; -# undef delx -# undef nlabel -# undef feasbl -# undef svitch -# undef prevnode -} - -/*********************************************************************** -* NAME -* -* auction - compute good initial flow and prices -* -* PURPOSE -* -* This subroutine uses a version of the auction algorithm for min -* cost network flow to compute a good initial flow and prices for the -* problem. -* -* INPUT PARAMETERS -* -* n = number of nodes -* na = number of arcs -* large = a very large integer to represent infinity (see note 3) -* startn[i] = starting node for the i-th arc, i = 1,...,na -* endn[i] = ending node for the i-th arc, i = 1,...,na -* fou[i] = first arc leaving i-th node, i = 1,...,n -* nxtou[i] = next arc leaving the starting node of j-th arc, -* i = 1,...,na -* fin[i] = first arc entering i-th node, i = 1,...,n -* nxtin[i] = next arc entering the ending node of j-th arc, -* i = 1,...,na -* -* UPDATED PARAMETERS -* -* rc[j] = reduced cost of arc j, j = 1,...,na -* u[j] = residual capacity of arc j, j = 1,...,na -* x[j] = flow on arc j, j = 1,...,na -* dfct[i] = deficit at node i, i = 1,...,n -* -* OUTPUT PARAMETERS -* -* nsp = number of auction/shortest path iterations -* -* WORKING PARAMETERS -* -* p[1+n], prdcsr[1+n], save[1+na], fpushf[1+n], nxtpushf[1+na], -* fpushb[1+n], nxtpushb[1+na], nxtqueue[1+n], extend_arc[1+n], -* sb_level[1+n], sb_arc[1+n], path_id[1+n] -* -* RETURNS -* -* 0 = normal return -* 1 = problem is found to be infeasible */ - -static int auction(struct relax4_csa *csa) -{ /* input parameters */ - int n = csa->n; - int na = csa->na; - int large = csa->large; - int *startn = csa->startn; - int *endn = csa->endn; - int *fou = csa->fou; - int *nxtou = csa->nxtou; - int *fin = csa->fin; - int *nxtin = csa->nxtin; - /* updated parameters */ -# define crash (csa->crash) - int *rc = csa->rc; - int *u = csa->u; - int *x = csa->x; - int *dfct = csa->dfct; - /* output parameters */ -# define nsp (csa->nsp) - /* working parameters */ - int *p = csa->label; - int *prdcsr = csa->prdcsr; - int *save = csa->save; - int *fpushf = csa->tfstou; - int *nxtpushf = csa->tnxtou; - int *fpushb = csa->tfstin; - int *nxtpushb = csa->tnxtin; - int *nxtqueue = csa->nxtqueue; - int *extend_arc = csa->extend_arc; - int *sb_level = csa->sb_level; - int *sb_arc = csa->sb_arc; - char *path_id = csa->mark; - /* local variables */ - int arc, bstlevel, end, eps, extarc, factor, flow, i, incr, - last, lastqueue, maxcost, mincost, nas, naug, new_level, node, - nolist, num_passes, nxtnode, pass, pend, pr_term, prd, - prevarc, prevlevel, prevnode, pstart, pterm, rdcost, red_cost, - resid, root, secarc, seclevel, start, term, thresh_dfct; - /* start initialization using auction */ - naug = 0; - pass = 0; - thresh_dfct = 0; - /* factor determines by how much epsilon is reduced at each - * minimization */ - factor = 3; - /* num_passes determines how many auction scaling phases are - * performed */ - num_passes = 1; - /* set arc flows to satisfy cs and calculate maxcost and - * mincost */ - maxcost = -large; - mincost = large; - for (arc = 1; arc <= na; arc++) - { start = startn[arc]; - end = endn[arc]; - rdcost = rc[arc]; - if (maxcost < rdcost) - maxcost = rdcost; - if (mincost > rdcost) - mincost = rdcost; - if (rdcost < 0) - { dfct[start] += u[arc]; - dfct[end] -= u[arc]; - x[arc] = u[arc]; - u[arc] = 0; - } - else - x[arc] = 0; - } - /* set initial epsilon */ - if ((maxcost - mincost) >= 8) - eps = (maxcost - mincost) / 8; - else - eps = 1; - /* set initial prices to zero */ - for (node = 1; node <= n; node++) - p[node] = 0; - /* Initialization using auction/shortest paths. */ -L100: /* Start of the first scaling phase. */ - pass++; - if ((pass == num_passes) || (eps == 1)) - crash = 0; - nolist = 0; - /* construct list of positive surplus nodes and queue of negative - * surplus nodes */ - for (node = 1; node <= n; node++) - { prdcsr[node] = 0; - path_id[node] = false; - extend_arc[node] = 0; - sb_level[node] = -large; - nxtqueue[node] = node + 1; - if (dfct[node] > 0) - { nolist++; - save[nolist] = node; - } - } - nxtqueue[n] = 1; - root = 1; - prevnode = lastqueue = n; - /* initialization with down iterations for negative surplus - * nodes */ - for (i = 1; i <= nolist; i++) - { node = save[i]; - nsp++; - /* build the list of arcs w/ room for pushing flow and find - * proper price for down iteration */ - bstlevel = -large; - fpushf[node] = 0; - for (arc = fou[node]; arc > 0; arc = nxtou[arc]) - { if (u[arc] > 0) - { if (fpushf[node] == 0) - { fpushf[node] = arc; - nxtpushf[arc] = 0; - last = arc; - } - else - { nxtpushf[last] = arc; - nxtpushf[arc] = 0; - last = arc; - } - } - if (x[arc] > 0) - { new_level = p[endn[arc]] + rc[arc]; - if (new_level > bstlevel) - { bstlevel = new_level; - extarc = arc; - } - } - } - fpushb[node] = 0; - for (arc = fin[node]; arc > 0; arc = nxtin[arc]) - { if (x[arc] > 0) - { if (fpushb[node] == 0) - { fpushb[node] = arc; - nxtpushb[arc] = 0; - last = arc; - } - else - { nxtpushb[last] = arc; - nxtpushb[arc] = 0; - last = arc; - } - } - if (u[arc] > 0) - { new_level = p[startn[arc]] - rc[arc]; - if (new_level > bstlevel) - { bstlevel = new_level; - extarc = -arc; - } - } - } - extend_arc[node] = extarc; - p[node] = bstlevel - eps; - } -L200: /* Start the augmentation cycles of the new scaling phase. */ - if (dfct[root] >= thresh_dfct) - goto L3000; - term = root; - path_id[root] = true; -L500: /* Main forward algorithm with root as origin. */ - /* start of a new forward iteration */ - pterm = p[term]; - extarc = extend_arc[term]; - if (extarc == 0) - { /* build the list of arcs w/ room for pushing flow */ - fpushf[term] = 0; - for (arc = fou[term]; arc > 0; arc = nxtou[arc]) - { if (u[arc] > 0) - { if (fpushf[term] == 0) - { fpushf[term] = arc; - nxtpushf[arc] = 0; - last = arc; - } - else - { nxtpushf[last] = arc; - nxtpushf[arc] = 0; - last = arc; - } - } - } - fpushb[term] = 0; - for (arc = fin[term]; arc > 0; arc = nxtin[arc]) - { if (x[arc] > 0) - { if (fpushb[term] == 0) - { fpushb[term] = arc; - nxtpushb[arc] = 0; - last = arc; - } - else - { nxtpushb[last] = arc; - nxtpushb[arc] = 0; - last = arc; - } - } - } - goto L600; - } - /* speculative path extension attempt */ - /* note: arc > 0 means that arc is oriented from the root to the - * destinations - * arc < 0 means that arc is oriented from the destinations to the - * root - * extarc = 0 or prdarc = 0, means the extension arc or the - * predecessor arc, respectively, has not been established */ - if (extarc > 0) - { if (u[extarc] == 0) - { seclevel = sb_level[term]; - goto L580; - } - end = endn[extarc]; - bstlevel = p[end] + rc[extarc]; - if (pterm >= bstlevel) - { if (path_id[end]) - goto L1200; - term = end; - prdcsr[term] = extarc; - path_id[term] = true; - /* if negative surplus node is found, do an augmentation */ - if (dfct[term] > 0) - goto L2000; - /* return for another iteration */ - goto L500; - } - } - else - { extarc = -extarc; - if (x[extarc] == 0) - { seclevel = sb_level[term]; - goto L580; - } - start = startn[extarc]; - bstlevel = p[start] - rc[extarc]; - if (pterm >= bstlevel) - { if (path_id[start]) - goto L1200; - term = start; - prdcsr[term] = -extarc; - path_id[term] = true; - /* if negative surplus node is found, do an augmentation */ - if (dfct[term] > 0) - goto L2000; - /* return for another iteration */ - goto L500; - } - } -L550: /* second best logic test applied to save a full node scan - * if old best level continues to be best go for another - * contraction */ - seclevel = sb_level[term]; - if (bstlevel <= seclevel) - goto L800; -L580: /* if second best can be used do either a contraction or start - * over with a speculative extension */ - if (seclevel > -large) - { extarc = sb_arc[term]; - if (extarc > 0) - { if (u[extarc] == 0) - goto L600; - bstlevel = p[endn[extarc]] + rc[extarc]; - } - else - { if (x[-extarc] == 0) - goto L600; - bstlevel = p[startn[-extarc]] - rc[-extarc]; - } - if (bstlevel == seclevel) - { sb_level[term] = -large; - extend_arc[term] = extarc; - goto L800; - } - } -L600: /* extension/contraction attempt was unsuccessful, so scan - * terminal node */ - nsp++; - bstlevel = seclevel = large; - for (arc = fpushf[term]; arc > 0; arc = nxtpushf[arc]) - { new_level = p[endn[arc]] + rc[arc]; - if (new_level < seclevel) - { if (new_level < bstlevel) - { seclevel = bstlevel; - bstlevel = new_level; - secarc = extarc; - extarc = arc; - } - else - { seclevel = new_level; - secarc = arc; - } - } - } - for (arc = fpushb[term]; arc > 0; arc = nxtpushb[arc]) - { new_level = p[startn[arc]] - rc[arc]; - if (new_level < seclevel) - { if (new_level < bstlevel) - { seclevel = bstlevel; - bstlevel = new_level; - secarc = extarc; - extarc = -arc; - } - else - { seclevel = new_level; - secarc = -arc; - } - } - } - sb_level[term] = seclevel; - sb_arc[term] = secarc; - extend_arc[term] = extarc; -L800: /* End of node scan. */ - /* if the terminal node is the root, adjust its price and change - * root */ - if (term == root) - { p[term] = bstlevel + eps; - if (p[term] >= large) - { /* no path to the destination */ - /* problem is found to be infeasible */ - return 1; - } - path_id[root] = false; - prevnode = root; - root = nxtqueue[root]; - goto L200; - } - /* check whether extension or contraction */ - prd = prdcsr[term]; - if (prd > 0) - { pr_term = startn[prd]; - prevlevel = p[pr_term] - rc[prd]; - } - else - { pr_term = endn[-prd]; - prevlevel = p[pr_term] + rc[-prd]; - } - if (prevlevel > bstlevel) - { /* path extension */ - if (prevlevel >= bstlevel + eps) - p[term] = bstlevel + eps; - else - p[term] = prevlevel; - if (extarc > 0) - { end = endn[extarc]; - if (path_id[end]) - goto L1200; - term = end; - } - else - { start = startn[-extarc]; - if (path_id[start]) - goto L1200; - term = start; - } - prdcsr[term] = extarc; - path_id[term] = true; - /* if negative surplus node is found, do an augmentation */ - if (dfct[term] > 0) - goto L2000; - /* return for another iteration */ - goto L500; - } - else - { /* path contraction */ - p[term] = bstlevel + eps; - path_id[term] = false; - term = pr_term; - if (pr_term != root) - { if (bstlevel <= pterm + eps) - goto L2000; - } - pterm = p[term]; - extarc = prd; - if (prd > 0) - bstlevel += eps + rc[prd]; - else - bstlevel += eps - rc[-prd]; - /* do a second best test and if that fails, do a full node - * scan */ - goto L550; - } -L1200:/* A cycle is about to form; do a retreat sequence. */ - node = term; -L1600:if (node != root) - { path_id[node] = false; - prd = prdcsr[node]; - if (prd > 0) - { pr_term = startn[prd]; - if (p[pr_term] == p[node] + rc[prd] + eps) - { node = pr_term; - goto L1600; - } - } - else - { pr_term = endn[-prd]; - if (p[pr_term] == p[node] - rc[-prd] + eps) - { node = pr_term; - goto L1600; - } - } - /* do a full scan and price rise at pr_term */ - nsp++; - bstlevel = seclevel = large; - for (arc = fpushf[pr_term]; arc > 0; arc = nxtpushf[arc]) - { new_level = p[endn[arc]] + rc[arc]; - if (new_level < seclevel) - { if (new_level < bstlevel) - { seclevel = bstlevel; - bstlevel = new_level; - secarc = extarc; - extarc = arc; - } - else - { seclevel = new_level; - secarc = arc; - } - } - } - for (arc = fpushb[pr_term]; arc > 0; arc = nxtpushb[arc]) - { new_level = p[startn[arc]] - rc[arc]; - if (new_level < seclevel) - { if (new_level < bstlevel) - { seclevel = bstlevel; - bstlevel = new_level; - secarc = extarc; - extarc = -arc; - } - else - { seclevel = new_level; - secarc = -arc; - } - } - } - sb_level[pr_term] = seclevel; - sb_arc[pr_term] = secarc; - extend_arc[pr_term] = extarc; - p[pr_term] = bstlevel + eps; - if (pr_term == root) - { prevnode = root; - path_id[root] = false; - root = nxtqueue[root]; - goto L200; - } - path_id[pr_term] = false; - prd = prdcsr[pr_term]; - if (prd > 0) - term = startn[prd]; - else - term = endn[-prd]; - if (term == root) - { prevnode = root; - path_id[root] = false; - root = nxtqueue[root]; - goto L200; - } - else - goto L2000; - } -L2000:/* End of auction/shortest path routine. */ - /* do augmentation from root and correct the push lists */ - incr = -dfct[root]; - for (node = root;;) - { extarc = extend_arc[node]; - path_id[node] = false; - if (extarc > 0) - { node = endn[extarc]; - if (incr > u[extarc]) - incr = u[extarc]; - } - else - { node = startn[-extarc]; - if (incr > x[-extarc]) - incr = x[-extarc]; - } - if (node == term) - break; - } - path_id[term] = false; - if (dfct[term] > 0) - { if (incr > dfct[term]) - incr = dfct[term]; - } - for (node = root;;) - { extarc = extend_arc[node]; - if (extarc > 0) - { end = endn[extarc]; - /* add arc to the reduced graph */ - if (x[extarc] == 0) - { nxtpushb[extarc] = fpushb[end]; - fpushb[end] = extarc; - new_level = p[node] - rc[extarc]; - if (sb_level[end] > new_level) - { sb_level[end] = new_level; - sb_arc[end] = -extarc; - } - } - x[extarc] += incr; - u[extarc] -= incr; - /* remove arc from the reduced graph */ - if (u[extarc] == 0) - { nas++; - arc = fpushf[node]; - if (arc == extarc) - fpushf[node] = nxtpushf[arc]; - else - { prevarc = arc; - arc = nxtpushf[arc]; - while (arc > 0) - { if (arc == extarc) - { nxtpushf[prevarc] = nxtpushf[arc]; - break; - } - prevarc = arc; - arc = nxtpushf[arc]; - } - } - } - node = end; - } - else - { extarc = -extarc; - start = startn[extarc]; - /* add arc to the reduced graph */ - if (u[extarc] == 0) - { nxtpushf[extarc] = fpushf[start]; - fpushf[start] = extarc; - new_level = p[node] + rc[extarc]; - if (sb_level[start] > new_level) - { sb_level[start] = new_level; - sb_arc[start] = extarc; - } - } - u[extarc] += incr; - x[extarc] -= incr; - /* remove arc from the reduced graph */ - if (x[extarc] == 0) - { nas++; - arc = fpushb[node]; - if (arc == extarc) - fpushb[node] = nxtpushb[arc]; - else - { prevarc = arc; - arc = nxtpushb[arc]; - while (arc > 0) - { if (arc == extarc) - { nxtpushb[prevarc] = nxtpushb[arc]; - break; - } - prevarc = arc; - arc = nxtpushb[arc]; - } - } - } - node = start; - } - if (node == term) - break; - } - dfct[term] -= incr; - dfct[root] += incr; - /* insert term in the queue if it has a large enough surplus */ - if (dfct[term] < thresh_dfct) - { if (nxtqueue[term] == 0) - { nxtnode = nxtqueue[root]; - if ((p[term] >= p[nxtnode]) && (root != nxtnode)) - { nxtqueue[root] = term; - nxtqueue[term] = nxtnode; - } - else - { nxtqueue[prevnode] = term; - nxtqueue[term] = root; - prevnode = term; - } - } - } - /* if root has a large enough surplus, keep it in the queue and - * return for another iteration */ - if (dfct[root] < thresh_dfct) - { prevnode = root; - root = nxtqueue[root]; - goto L200; - } -L3000:/* end of augmentation cycle */ - /* Check for termination of scaling phase. If scaling phase is not - * finished, advance the queue and return to take another node. */ - nxtnode = nxtqueue[root]; - if (root != nxtnode) - { nxtqueue[root] = 0; - nxtqueue[prevnode] = nxtnode; - root = nxtnode; - goto L200; - } - /* End of subproblem (scaling phase). */ - /* Reduce epsilon. */ - eps /= factor; - if (eps < 1) eps = 1; - thresh_dfct /= factor; - if (eps == 1) thresh_dfct = 0; - /* if another auction scaling phase remains, reset the flows & - * the push lists; else reset arc flows to satisfy cs and compute - * reduced costs */ - if (crash == 1) - { for (arc = 1; arc <= na; arc++) - { start = startn[arc]; - end = endn[arc]; - pstart = p[start]; - pend = p[end]; - if (pstart > pend + eps + rc[arc]) - { resid = u[arc]; - if (resid > 0) - { dfct[start] += resid; - dfct[end] -= resid; - x[arc] += resid; - u[arc] = 0; - } - } - else if (pstart < pend - eps + rc[arc]) - { flow = x[arc]; - if (flow > 0) - { dfct[start] -= flow; - dfct[end] += flow; - x[arc] = 0; - u[arc] += flow; - } - } - } - /* return for another phase */ - goto L100; - } - else - { crash = 1; - for (arc = 1; arc <= na; arc++) - { start = startn[arc]; - end = endn[arc]; - red_cost = rc[arc] + p[end] - p[start]; - if (red_cost < 0) - { resid = u[arc]; - if (resid > 0) - { dfct[start] += resid; - dfct[end] -= resid; - x[arc] += resid; - u[arc] = 0; - } - } - else if (red_cost > 0) - { flow = x[arc]; - if (flow > 0) - { dfct[start] -= flow; - dfct[end] += flow; - x[arc] = 0; - u[arc] += flow; - } - } - rc[arc] = red_cost; - } - } - return 0; -# undef crash -# undef nsp -} - -/* eof */ diff --git a/code/3rd_glpk/misc/relax4.h b/code/3rd_glpk/misc/relax4.h deleted file mode 100644 index f48b8508..00000000 --- a/code/3rd_glpk/misc/relax4.h +++ /dev/null @@ -1,102 +0,0 @@ -/* relax4.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef RELAX4_H -#define RELAX4_H - -struct relax4_csa -{ /* common storage area */ - /* input parameters --------------------------------------------*/ - int n; - /* number of nodes */ - int na; - /* number of arcs */ - int large; - /* very large int to represent infinity */ - int repeat; - /* true if initialization is to be skipped (false otherwise) */ - int crash; - /* 0 if default initialization is used - * 1 if auction initialization is used */ - int *startn; /* int startn[1+na]; */ - /* startn[j] = starting node for arc j, j = 1,...,na */ - int *endn; /* int endn[1+na] */ - /* endn[j] = ending node for arc j, j = 1,...,na */ - int *fou; /* int fou[1+n]; */ - /* fou[i] = first arc out of node i, i = 1,...,n */ - int *nxtou; /* int nxtou[1+na]; */ - /* nxtou[j] = next arc out of the starting node of arc j, - * j = 1,...,na */ - int *fin; /* int fin[1+n]; */ - /* fin[i] = first arc into node i, i = 1,...,n */ - int *nxtin; /* int nxtin[1+na]; */ - /* nxtin[j] = next arc into the ending node of arc j, - * j = 1,...,na */ - /* updated parameters ------------------------------------------*/ - int *rc; /* int rc[1+na]; */ - /* rc[j] = reduced cost of arc j, j = 1,...,na */ - int *u; /* int u[1+na]; */ - /* u[j] = capacity of arc j on input - * and (capacity of arc j) - x(j) on output, j = 1,...,na */ - int *dfct; /* int dfct[1+n]; */ - /* dfct[i] = demand at node i on input - * and zero on output, i = 1,...,n */ - /* output parameters -------------------------------------------*/ - int *x; /* int x[1+na]; */ - /* x[j] = flow on arc j, j = 1,...,na */ - int nmultinode; - /* number of multinode relaxation iterations in RELAX4 */ - int iter; - /* number of relaxation iterations in RELAX4 */ - int num_augm; - /* number of flow augmentation steps in RELAX4 */ - int num_ascnt; - /* number of multinode ascent steps in RELAX4 */ - int nsp; - /* number of auction/shortest path iterations */ - /* working parameters ------------------------------------------*/ - int *label; /* int label, tempin, p[1+n]; */ - int *prdcsr; /* int prdcsr, tempou, price[1+n]; */ - int *save; /* int save[1+na]; */ - int *tfstou; /* int tfstou, fpushf[1+n]; */ - int *tnxtou; /* int tnxtou, nxtpushf[1+na]; */ - int *tfstin; /* int tfstin, fpushb[1+n]; */ - int *tnxtin; /* int tnxtin, nxtpushb[1+na]; */ - int *nxtqueue; /* int nxtqueue[1+n]; */ - char *scan; /* bool scan[1+n]; */ - char *mark; /* bool mark, path_id[1+n]; */ - /* working parameters used by routine auction only -------------*/ - int *extend_arc; /* int extend_arc[1+n]; */ - int *sb_level; /* int sb_level[1+n]; */ - int *sb_arc; /* int sb_arc[1+n]; */ -}; - -#define relax4 _glp_relax4 -int relax4(struct relax4_csa *csa); - -#define relax4_inidat _glp_relax4_inidat -void relax4_inidat(struct relax4_csa *csa); - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/rng.c b/code/3rd_glpk/misc/rng.c deleted file mode 100644 index e0acb53a..00000000 --- a/code/3rd_glpk/misc/rng.c +++ /dev/null @@ -1,227 +0,0 @@ -/* rng.c (pseudo-random number generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* This code is a modified version of the module GB_FLIP, a portable -* pseudo-random number generator. The original version of GB_FLIP is -* a part of The Stanford GraphBase developed by Donald E. Knuth (see -* http://www-cs-staff.stanford.edu/~knuth/sgb.html). -* -* Note that all changes concern only external names, so this modified -* version produces exactly the same results as the original version. -* -* Changes were made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "rng.h" - -#if 0 -int A[56] = { -1 }; -#else -#define A (rand->A) -#endif -/* pseudo-random values */ - -#if 0 -int *fptr = A; -#else -#define fptr (rand->fptr) -#endif -/* the next A value to be exported */ - -#define mod_diff(x, y) (((x) - (y)) & 0x7FFFFFFF) -/* difference modulo 2^31 */ - -static int flip_cycle(RNG *rand) -{ /* this is an auxiliary routine to do 55 more steps of the basic - * recurrence, at high speed, and to reset fptr */ - int *ii, *jj; - for (ii = &A[1], jj = &A[32]; jj <= &A[55]; ii++, jj++) - *ii = mod_diff(*ii, *jj); - for (jj = &A[1]; ii <= &A[55]; ii++, jj++) - *ii = mod_diff(*ii, *jj); - fptr = &A[54]; - return A[55]; -} - -/*********************************************************************** -* NAME -* -* rng_create_rand - create pseudo-random number generator -* -* SYNOPSIS -* -* #include "rng.h" -* RNG *rng_create_rand(void); -* -* DESCRIPTION -* -* The routine rng_create_rand creates and initializes a pseudo-random -* number generator. -* -* RETURNS -* -* The routine returns a pointer to the generator created. */ - -RNG *rng_create_rand(void) -{ RNG *rand; - int i; - rand = talloc(1, RNG); - A[0] = -1; - for (i = 1; i <= 55; i++) A[i] = 0; - fptr = A; - rng_init_rand(rand, 1); - return rand; -} - -/*********************************************************************** -* NAME -* -* rng_init_rand - initialize pseudo-random number generator -* -* SYNOPSIS -* -* #include "rng.h" -* void rng_init_rand(RNG *rand, int seed); -* -* DESCRIPTION -* -* The routine rng_init_rand initializes the pseudo-random number -* generator. The parameter seed may be any integer number. Note that -* on creating the generator this routine is called with the parameter -* seed equal to 1. */ - -void rng_init_rand(RNG *rand, int seed) -{ int i; - int prev = seed, next = 1; - seed = prev = mod_diff(prev, 0); - A[55] = prev; - for (i = 21; i; i = (i + 21) % 55) - { A[i] = next; - next = mod_diff(prev, next); - if (seed & 1) - seed = 0x40000000 + (seed >> 1); - else - seed >>= 1; - next = mod_diff(next, seed); - prev = A[i]; - } - flip_cycle(rand); - flip_cycle(rand); - flip_cycle(rand); - flip_cycle(rand); - flip_cycle(rand); - return; -} - -/*********************************************************************** -* NAME -* -* rng_next_rand - obtain pseudo-random integer in the range [0, 2^31-1] -* -* SYNOPSIS -* -* #include "rng.h" -* int rng_next_rand(RNG *rand); -* -* RETURNS -* -* The routine rng_next_rand returns a next pseudo-random integer which -* is uniformly distributed between 0 and 2^31-1, inclusive. The period -* length of the generated numbers is 2^85 - 2^30. The low order bits of -* the generated numbers are just as random as the high-order bits. */ - -int rng_next_rand(RNG *rand) -{ return - *fptr >= 0 ? *fptr-- : flip_cycle(rand); -} - -/*********************************************************************** -* NAME -* -* rng_unif_rand - obtain pseudo-random integer in the range [0, m-1] -* -* SYNOPSIS -* -* #include "rng.h" -* int rng_unif_rand(RNG *rand, int m); -* -* RETURNS -* -* The routine rng_unif_rand returns a next pseudo-random integer which -* is uniformly distributed between 0 and m-1, inclusive, where m is any -* positive integer less than 2^31. */ - -#define two_to_the_31 ((unsigned int)0x80000000) - -int rng_unif_rand(RNG *rand, int m) -{ unsigned int t = two_to_the_31 - (two_to_the_31 % m); - int r; - xassert(m > 0); - do { r = rng_next_rand(rand); } while (t <= (unsigned int)r); - return r % m; -} - -/*********************************************************************** -* NAME -* -* rng_delete_rand - delete pseudo-random number generator -* -* SYNOPSIS -* -* #include "rng.h" -* void rng_delete_rand(RNG *rand); -* -* DESCRIPTION -* -* The routine rng_delete_rand frees all the memory allocated to the -* specified pseudo-random number generator. */ - -void rng_delete_rand(RNG *rand) -{ tfree(rand); - return; -} - -/**********************************************************************/ - -#ifdef GLP_TEST -/* To be sure that this modified version produces the same results as - * the original version, run this validation program. */ - -int main(void) -{ RNG *rand; - int j; - rand = rng_create_rand(); - rng_init_rand(rand, -314159); - if (rng_next_rand(rand) != 119318998) - { fprintf(stderr, "Failure on the first try!\n"); - return -1; - } - for (j = 1; j <= 133; j++) rng_next_rand(rand); - if (rng_unif_rand(rand, 0x55555555) != 748103812) - { fprintf(stderr, "Failure on the second try!\n"); - return -2; - } - fprintf(stderr, "OK, the random-number generator routines seem to" - " work!\n"); - rng_delete_rand(rand); - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/rng.h b/code/3rd_glpk/misc/rng.h deleted file mode 100644 index 49725e05..00000000 --- a/code/3rd_glpk/misc/rng.h +++ /dev/null @@ -1,67 +0,0 @@ -/* rng.h (pseudo-random number generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef RNG_H -#define RNG_H - -typedef struct RNG RNG; - -struct RNG -{ /* Knuth's portable pseudo-random number generator */ - int A[56]; - /* pseudo-random values */ - int *fptr; - /* the next A value to be exported */ -}; - -#define rng_create_rand _glp_rng_create_rand -RNG *rng_create_rand(void); -/* create pseudo-random number generator */ - -#define rng_init_rand _glp_rng_init_rand -void rng_init_rand(RNG *rand, int seed); -/* initialize pseudo-random number generator */ - -#define rng_next_rand _glp_rng_next_rand -int rng_next_rand(RNG *rand); -/* obtain pseudo-random integer in the range [0, 2^31-1] */ - -#define rng_unif_rand _glp_rng_unif_rand -int rng_unif_rand(RNG *rand, int m); -/* obtain pseudo-random integer in the range [0, m-1] */ - -#define rng_delete_rand _glp_rng_delete_rand -void rng_delete_rand(RNG *rand); -/* delete pseudo-random number generator */ - -#define rng_unif_01 _glp_rng_unif_01 -double rng_unif_01(RNG *rand); -/* obtain pseudo-random number in the range [0, 1] */ - -#define rng_uniform _glp_rng_uniform -double rng_uniform(RNG *rand, double a, double b); -/* obtain pseudo-random number in the range [a, b] */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/rng1.c b/code/3rd_glpk/misc/rng1.c deleted file mode 100644 index b89f676f..00000000 --- a/code/3rd_glpk/misc/rng1.c +++ /dev/null @@ -1,73 +0,0 @@ -/* rng1.c (pseudo-random number generator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "rng.h" - -/*********************************************************************** -* NAME -* -* rng_unif_01 - obtain pseudo-random number in the range [0, 1] -* -* SYNOPSIS -* -* #include "rng.h" -* double rng_unif_01(RNG *rand); -* -* RETURNS -* -* The routine rng_unif_01 returns a next pseudo-random number which is -* uniformly distributed in the range [0, 1]. */ - -double rng_unif_01(RNG *rand) -{ double x; - x = (double)rng_next_rand(rand) / 2147483647.0; - xassert(0.0 <= x && x <= 1.0); - return x; -} - -/*********************************************************************** -* NAME -* -* rng_uniform - obtain pseudo-random number in the range [a, b] -* -* SYNOPSIS -* -* #include "rng.h" -* double rng_uniform(RNG *rand, double a, double b); -* -* RETURNS -* -* The routine rng_uniform returns a next pseudo-random number which is -* uniformly distributed in the range [a, b]. */ - -double rng_uniform(RNG *rand, double a, double b) -{ double x; - xassert(a < b); - x = rng_unif_01(rand); - x = a * (1.0 - x) + b * x; - xassert(a <= x && x <= b); - return x; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/round2n.c b/code/3rd_glpk/misc/round2n.c deleted file mode 100644 index 8a94c616..00000000 --- a/code/3rd_glpk/misc/round2n.c +++ /dev/null @@ -1,64 +0,0 @@ -/* round2n.c (round floating-point number to nearest power of two) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "misc.h" - -/*********************************************************************** -* NAME -* -* round2n - round floating-point number to nearest power of two -* -* SYNOPSIS -* -* #include "misc.h" -* double round2n(double x); -* -* RETURNS -* -* Given a positive floating-point value x the routine round2n returns -* 2^n such that |x - 2^n| is minimal. -* -* EXAMPLES -* -* round2n(10.1) = 2^3 = 8 -* round2n(15.3) = 2^4 = 16 -* round2n(0.01) = 2^(-7) = 0.0078125 -* -* BACKGROUND -* -* Let x = f * 2^e, where 0.5 <= f < 1 is a normalized fractional part, -* e is an integer exponent. Then, obviously, 0.5 * 2^e <= x < 2^e, so -* if x - 0.5 * 2^e <= 2^e - x, we choose 0.5 * 2^e = 2^(e-1), and 2^e -* otherwise. The latter condition can be written as 2 * x <= 1.5 * 2^e -* or 2 * f * 2^e <= 1.5 * 2^e or, finally, f <= 0.75. */ - -double round2n(double x) -{ int e; - double f; - xassert(x > 0.0); - f = frexp(x, &e); - return ldexp(1.0, f <= 0.75 ? e-1 : e); -} - -/* eof */ diff --git a/code/3rd_glpk/misc/str2int.c b/code/3rd_glpk/misc/str2int.c deleted file mode 100644 index cbd6e953..00000000 --- a/code/3rd_glpk/misc/str2int.c +++ /dev/null @@ -1,92 +0,0 @@ -/* str2int.c (convert string to int) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "misc.h" -#include "stdc.h" - -/*********************************************************************** -* NAME -* -* str2int - convert character string to value of int type -* -* SYNOPSIS -* -* #include "misc.h" -* int str2int(const char *str, int *val); -* -* DESCRIPTION -* -* The routine str2int converts the character string str to a value of -* integer type and stores the value into location, which the parameter -* val points to (in the case of error content of this location is not -* changed). -* -* RETURNS -* -* The routine returns one of the following error codes: -* -* 0 - no error; -* 1 - value out of range; -* 2 - character string is syntactically incorrect. */ - -int str2int(const char *str, int *val_) -{ int d, k, s, val = 0; - /* scan optional sign */ - if (str[0] == '+') - s = +1, k = 1; - else if (str[0] == '-') - s = -1, k = 1; - else - s = +1, k = 0; - /* check for the first digit */ - if (!isdigit((unsigned char)str[k])) - return 2; - /* scan digits */ - while (isdigit((unsigned char)str[k])) - { d = str[k++] - '0'; - if (s > 0) - { if (val > INT_MAX / 10) - return 1; - val *= 10; - if (val > INT_MAX - d) - return 1; - val += d; - } - else /* s < 0 */ - { if (val < INT_MIN / 10) - return 1; - val *= 10; - if (val < INT_MIN + d) - return 1; - val -= d; - } - } - /* check for terminator */ - if (str[k] != '\0') - return 2; - /* conversion has been done */ - *val_ = val; - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/str2num.c b/code/3rd_glpk/misc/str2num.c deleted file mode 100644 index 26c2f68f..00000000 --- a/code/3rd_glpk/misc/str2num.c +++ /dev/null @@ -1,110 +0,0 @@ -/* str2num.c (convert string to double) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "misc.h" -#include "stdc.h" - -/*********************************************************************** -* NAME -* -* str2num - convert character string to value of double type -* -* SYNOPSIS -* -* #include "misc.h" -* int str2num(const char *str, double *val); -* -* DESCRIPTION -* -* The routine str2num converts the character string str to a value of -* double type and stores the value into location, which the parameter -* val points to (in the case of error content of this location is not -* changed). -* -* RETURNS -* -* The routine returns one of the following error codes: -* -* 0 - no error; -* 1 - value out of range; -* 2 - character string is syntactically incorrect. */ - -int str2num(const char *str, double *val_) -{ int k; - double val; - /* scan optional sign */ - k = (str[0] == '+' || str[0] == '-' ? 1 : 0); - /* check for decimal point */ - if (str[k] == '.') - { k++; - /* a digit should follow it */ - if (!isdigit((unsigned char)str[k])) - return 2; - k++; - goto frac; - } - /* integer part should start with a digit */ - if (!isdigit((unsigned char)str[k])) - return 2; - /* scan integer part */ - while (isdigit((unsigned char)str[k])) - k++; - /* check for decimal point */ - if (str[k] == '.') k++; -frac: /* scan optional fraction part */ - while (isdigit((unsigned char)str[k])) - k++; - /* check for decimal exponent */ - if (str[k] == 'E' || str[k] == 'e') - { k++; - /* scan optional sign */ - if (str[k] == '+' || str[k] == '-') - k++; - /* a digit should follow E, E+ or E- */ - if (!isdigit((unsigned char)str[k])) - return 2; - } - /* scan optional exponent part */ - while (isdigit((unsigned char)str[k])) - k++; - /* check for terminator */ - if (str[k] != '\0') - return 2; - /* perform conversion */ - { char *endptr; - val = strtod(str, &endptr); - if (*endptr != '\0') - return 2; - } - /* check for overflow */ - if (!(-DBL_MAX <= val && val <= +DBL_MAX)) - return 1; - /* check for underflow */ - if (-DBL_MIN < val && val < +DBL_MIN) - val = 0.0; - /* conversion has been done */ - *val_ = val; - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/strspx.c b/code/3rd_glpk/misc/strspx.c deleted file mode 100644 index fe8a2a10..00000000 --- a/code/3rd_glpk/misc/strspx.c +++ /dev/null @@ -1,60 +0,0 @@ -/* strspx.c (remove all spaces from string) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "misc.h" - -/*********************************************************************** -* NAME -* -* strspx - remove all spaces from character string -* -* SYNOPSIS -* -* #include "misc.h" -* char *strspx(char *str); -* -* DESCRIPTION -* -* The routine strspx removes all spaces from the character string str. -* -* RETURNS -* -* The routine returns a pointer to the character string. -* -* EXAMPLES -* -* strspx(" Errare humanum est ") => "Errarehumanumest" -* -* strspx(" ") => "" */ - -char *strspx(char *str) -{ char *s, *t; - for (s = t = str; *s; s++) - { if (*s != ' ') - *t++ = *s; - } - *t = '\0'; - return str; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/strtrim.c b/code/3rd_glpk/misc/strtrim.c deleted file mode 100644 index 9992c4b0..00000000 --- a/code/3rd_glpk/misc/strtrim.c +++ /dev/null @@ -1,62 +0,0 @@ -/* strtrim.c (remove trailing spaces from string) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2000-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "misc.h" -#include "stdc.h" - -/*********************************************************************** -* NAME -* -* strtrim - remove trailing spaces from character string -* -* SYNOPSIS -* -* #include "misc.h" -* char *strtrim(char *str); -* -* DESCRIPTION -* -* The routine strtrim removes trailing spaces from the character -* string str. -* -* RETURNS -* -* The routine returns a pointer to the character string. -* -* EXAMPLES -* -* strtrim("Errare humanum est ") => "Errare humanum est" -* -* strtrim(" ") => "" */ - -char *strtrim(char *str) -{ char *t; - for (t = strrchr(str, '\0') - 1; t >= str; t--) - { if (*t != ' ') - break; - *t = '\0'; - } - return str; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/triang.c b/code/3rd_glpk/misc/triang.c deleted file mode 100644 index 99ba4d60..00000000 --- a/code/3rd_glpk/misc/triang.c +++ /dev/null @@ -1,311 +0,0 @@ -/* triang.c (find maximal triangular part of rectangular matrix) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "triang.h" - -/*********************************************************************** -* triang - find maximal triangular part of rectangular matrix -* -* Given a mxn sparse matrix A this routine finds permutation matrices -* P and Q such that matrix A' = P * A * Q has the following structure: -* -* 1 s n -* 1 * . . . . . x x x x x -* * * . . . . x x x x x -* * * * . . . x x x x x -* * * * * . . x x x x x -* * * * * * . x x x x x -* s * * * * * * x x x x x -* x x x x x x x x x x x -* x x x x x x x x x x x -* m x x x x x x x x x x x -* -* where '*' are elements of the triangular part, '.' are structural -* zeros, 'x' are other elements. -* -* The formal routine mat specifies the original matrix A in both row- -* and column-wise format. If the routine mat is called with k = +i, -* 1 <= i <= m, it should store column indices and values of non-zero -* elements of i-th row of A in locations ind[1], ..., ind[len] and -* val[1], ..., val[len], resp., where len is the returned number of -* non-zeros in the row, 0 <= len <= n. Similarly, if the routine mat -* is called with k = -j, 1 <= j <= n, it should store row indices and -* values of non-zero elements of j-th column of A and return len, the -* number of non-zeros in the column, 0 <= len <= m. Should note that -* duplicate indices are not allowed. -* -* The parameter info is a transit pointer passed to the routine mat. -* -* The parameter tol is a tolerance. The routine triang guarantees that -* each diagonal element in the triangular part of matrix A' is not -* less in magnitude than tol * max, where max is the maximal magnitude -* of elements in corresponding column. -* -* On exit the routine triang stores information on the triangular part -* found in the arrays rn and cn. Elements rn[1], ..., rn[s] specify -* row numbers and elements cn[1], ..., cn[s] specify column numbers -* of the original matrix A, which correspond to rows/columns 1, ..., s -* of matrix A', where s is the size of the triangular part returned by -* the routine, 0 <= s <= min(m, n). The order of rows and columns that -* are not included in the triangular part remains unspecified. -* -* ALGORITHM -* -* The routine triang uses a simple greedy heuristic. -* -* At some step the matrix A' = P * A * Q has the following structure: -* -* 1 n -* 1 * . . . . . . . x x x -* * * . . . . . . x x x -* * * * . . . . . x x x -* * * * * . . . . x x x -* x x x x # # # # x x x -* x x x x # # # # x x x -* x x x x # # # # x x x -* x x x x # # # # x x x -* m x x x x # # # # x x x -* -* where '#' are elements of active submatrix. Initially P = Q = I, so -* the active submatrix is the original matrix A = A'. -* -* If some row has exactly one non-zero in the active submatrix (row -* singleton), the routine includes this row and corresponding column -* in the triangular part, and removes the column from the active -* submatrix. Otherwise, the routine simply removes a column having -* maximal number of non-zeros from the active submatrix in the hope -* that new row singleton(s) will appear. -* -* COMPLEXITY -* -* The time complexity of the routine triang is O(nnz), where nnz is -* number of non-zeros in the original matrix A. */ - -int triang(int m, int n, int (*mat)(void *info, int k, int ind[], - double val[]), void *info, double tol, int rn[], int cn[]) -{ int head, i, j, jj, k, kk, ks, len, len2, next_j, ns, size; - int *cind, *rind, *cnt, *ptr, *list, *prev, *next; - double *cval, *rval, *big; - char *flag; - /* allocate working arrays */ - cind = talloc(1+m, int); - cval = talloc(1+m, double); - rind = talloc(1+n, int); - rval = talloc(1+n, double); - cnt = ptr = talloc(1+m, int); - list = talloc(1+n, int); - prev = talloc(1+n, int); - next = talloc(1+n, int); - big = talloc(1+n, double); - flag = talloc(1+n, char); - /*--------------------------------------------------------------*/ - /* build linked lists of columns having equal lengths */ - /*--------------------------------------------------------------*/ - /* ptr[len], 0 <= len <= m, is number of first column of length - * len; - * next[j], 1 <= j <= n, is number of next column having the same - * length as column j; - * big[j], 1 <= j <= n, is maximal magnitude of elements in j-th - * column */ - for (len = 0; len <= m; len++) - ptr[len] = 0; - for (j = 1; j <= n; j++) - { /* get j-th column */ - len = mat(info, -j, cind, cval); - xassert(0 <= len && len <= m); - /* add this column to beginning of list ptr[len] */ - next[j] = ptr[len]; - ptr[len] = j; - /* determine maximal magnitude of elements in this column */ - big[j] = 0.0; - for (k = 1; k <= len; k++) - { if (big[j] < fabs(cval[k])) - big[j] = fabs(cval[k]); - } - } - /*--------------------------------------------------------------*/ - /* build doubly linked list of columns ordered by decreasing */ - /* column lengths */ - /*--------------------------------------------------------------*/ - /* head is number of first column in the list; - * prev[j], 1 <= j <= n, is number of column that precedes j-th - * column in the list; - * next[j], 1 <= j <= n, is number of column that follows j-th - * column in the list */ - head = 0; - for (len = 0; len <= m; len++) - { /* walk thru list of columns of length len */ - for (j = ptr[len]; j != 0; j = next_j) - { next_j = next[j]; - /* add j-th column to beginning of the column list */ - prev[j] = 0; - next[j] = head; - if (head != 0) - prev[head] = j; - head = j; - } - } - /*--------------------------------------------------------------*/ - /* build initial singleton list */ - /*--------------------------------------------------------------*/ - /* there are used two list of columns: - * 1) doubly linked list of active columns, in which all columns - * are ordered by decreasing column lengths; - * 2) singleton list; an active column is included in this list - * if it has at least one row singleton in active submatrix */ - /* flag[j], 1 <= j <= n, is a flag of j-th column: - * 0 j-th column is inactive; - * 1 j-th column is active; - * 2 j-th column is active and has row singleton(s) */ - /* initially all columns are active */ - for (j = 1; j <= n; j++) - flag[j] = 1; - /* initialize row counts and build initial singleton list */ - /* cnt[i], 1 <= i <= m, is number of non-zeros, which i-th row - * has in active submatrix; - * ns is size of singleton list; - * list[1], ..., list[ns] are numbers of active columns included - * in the singleton list */ - ns = 0; - for (i = 1; i <= m; i++) - { /* get i-th row */ - len = cnt[i] = mat(info, +i, rind, rval); - xassert(0 <= len && len <= n); - if (len == 1) - { /* a[i,j] is row singleton */ - j = rind[1]; - xassert(1 <= j && j <= n); - if (flag[j] != 2) - { /* include j-th column in singleton list */ - flag[j] = 2; - list[++ns] = j; - } - } - } - /*--------------------------------------------------------------*/ - /* main loop */ - /*--------------------------------------------------------------*/ - size = 0; /* size of triangular part */ - /* loop until active column list is non-empty, i.e. until the - * active submatrix has at least one column */ - while (head != 0) - { if (ns == 0) - { /* singleton list is empty */ - /* remove from the active submatrix a column of maximal - * length in the hope that some row singletons appear */ - j = head; - len = mat(info, -j, cind, cval); - xassert(0 <= len && len <= m); - goto drop; - } - /* take column j from the singleton list */ - j = list[ns--]; - xassert(flag[j] == 2); - /* j-th column has at least one row singleton in the active - * submatrix; choose one having maximal magnitude */ - len = mat(info, -j, cind, cval); - xassert(0 <= len && len <= m); - kk = 0; - for (k = 1; k <= len; k++) - { i = cind[k]; - xassert(1 <= i && i <= m); - if (cnt[i] == 1) - { /* a[i,j] is row singleton */ - if (kk == 0 || fabs(cval[kk]) < fabs(cval[k])) - kk = k; - } - } - xassert(kk > 0); - /* check magnitude of the row singleton chosen */ - if (fabs(cval[kk]) < tol * big[j]) - { /* all row singletons are too small in magnitude; drop j-th - * column */ - goto drop; - } - /* row singleton a[i,j] is ok; add i-th row and j-th column to - * the triangular part */ - size++; - rn[size] = cind[kk]; - cn[size] = j; -drop: /* remove j-th column from the active submatrix */ - xassert(flag[j]); - flag[j] = 0; - if (prev[j] == 0) - head = next[j]; - else - next[prev[j]] = next[j]; - if (next[j] == 0) - ; - else - prev[next[j]] = prev[j]; - /* decrease row counts */ - for (k = 1; k <= len; k++) - { i = cind[k]; - xassert(1 <= i && i <= m); - xassert(cnt[i] > 0); - cnt[i]--; - if (cnt[i] == 1) - { /* new singleton appeared in i-th row; determine number - * of corresponding column (it is the only active column - * in this row) */ - len2 = mat(info, +i, rind, rval); - xassert(0 <= len2 && len2 <= n); - ks = 0; - for (kk = 1; kk <= len2; kk++) - { jj = rind[kk]; - xassert(1 <= jj && jj <= n); - if (flag[jj]) - { xassert(ks == 0); - ks = kk; - } - } - xassert(ks > 0); - /* a[i,jj] is new row singleton */ - jj = rind[ks]; - if (flag[jj] != 2) - { /* include jj-th column in the singleton list */ - flag[jj] = 2; - list[++ns] = jj; - } - } - } - } - /* now all row counts should be zero */ - for (i = 1; i <= m; i++) - xassert(cnt[i] == 0); - /* deallocate working arrays */ - tfree(cind); - tfree(cval); - tfree(rind); - tfree(rval); - tfree(ptr); - tfree(list); - tfree(prev); - tfree(next); - tfree(big); - tfree(flag); - return size; -} - -/* eof */ diff --git a/code/3rd_glpk/misc/triang.h b/code/3rd_glpk/misc/triang.h deleted file mode 100644 index 1e50d44d..00000000 --- a/code/3rd_glpk/misc/triang.h +++ /dev/null @@ -1,34 +0,0 @@ -/* triang.h (find maximal triangular part of rectangular matrix) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef TRIANG_H -#define TRIANG_H - -#define triang _glp_triang -int triang(int m, int n, int (*mat)(void *info, int k, int ind[], - double val[]), void *info, double tol, int rn[], int cn[]); -/* find maximal triangular part of rectangular matrix */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/wclique.c b/code/3rd_glpk/misc/wclique.c deleted file mode 100644 index 5daa69cf..00000000 --- a/code/3rd_glpk/misc/wclique.c +++ /dev/null @@ -1,242 +0,0 @@ -/* wclique.c (maximum weight clique, Ostergard's algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Two subroutines sub() and wclique() below are intended to find a -* maximum weight clique in a given undirected graph. These subroutines -* are slightly modified version of the program WCLIQUE developed by -* Patric Ostergard and based -* on ideas from the article "P. R. J. Ostergard, A new algorithm for -* the maximum-weight clique problem, submitted for publication", which -* in turn is a generalization of the algorithm for unweighted graphs -* presented in "P. R. J. Ostergard, A fast algorithm for the maximum -* clique problem, submitted for publication". -* -* USED WITH PERMISSION OF THE AUTHOR OF THE ORIGINAL CODE. -* -* Changes were made by Andrew Makhorin . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "wclique.h" - -/*********************************************************************** -* NAME -* -* wclique - find maximum weight clique with Ostergard's algorithm -* -* SYNOPSIS -* -* #include "wclique.h" -* int wclique(int n, const int w[], const unsigned char a[], -* int ind[]); -* -* DESCRIPTION -* -* The routine wclique finds a maximum weight clique in an undirected -* graph with Ostergard's algorithm. -* -* INPUT PARAMETERS -* -* n is the number of vertices, n > 0. -* -* w[i], i = 1,...,n, is a weight of vertex i. -* -* a[*] is the strict (without main diagonal) lower triangle of the -* graph adjacency matrix in packed format. -* -* OUTPUT PARAMETER -* -* ind[k], k = 1,...,size, is the number of a vertex included in the -* clique found, 1 <= ind[k] <= n, where size is the number of vertices -* in the clique returned on exit. -* -* RETURNS -* -* The routine returns the clique size, i.e. the number of vertices in -* the clique. */ - -struct csa -{ /* common storage area */ - int n; - /* number of vertices */ - const int *wt; /* int wt[0:n-1]; */ - /* weights */ - const unsigned char *a; - /* adjacency matrix (packed lower triangle without main diag.) */ - int record; - /* weight of best clique */ - int rec_level; - /* number of vertices in best clique */ - int *rec; /* int rec[0:n-1]; */ - /* best clique so far */ - int *clique; /* int clique[0:n-1]; */ - /* table for pruning */ - int *set; /* int set[0:n-1]; */ - /* current clique */ -}; - -#define n (csa->n) -#define wt (csa->wt) -#define a (csa->a) -#define record (csa->record) -#define rec_level (csa->rec_level) -#define rec (csa->rec) -#define clique (csa->clique) -#define set (csa->set) - -#if 0 -static int is_edge(struct csa *csa, int i, int j) -{ /* if there is arc (i,j), the routine returns true; otherwise - * false; 0 <= i, j < n */ - int k; - xassert(0 <= i && i < n); - xassert(0 <= j && j < n); - if (i == j) return 0; - if (i < j) k = i, i = j, j = k; - k = (i * (i - 1)) / 2 + j; - return a[k / CHAR_BIT] & - (unsigned char)(1 << ((CHAR_BIT - 1) - k % CHAR_BIT)); -} -#else -#define is_edge(csa, i, j) ((i) == (j) ? 0 : \ - (i) > (j) ? is_edge1(i, j) : is_edge1(j, i)) -#define is_edge1(i, j) is_edge2(((i) * ((i) - 1)) / 2 + (j)) -#define is_edge2(k) (a[(k) / CHAR_BIT] & \ - (unsigned char)(1 << ((CHAR_BIT - 1) - (k) % CHAR_BIT))) -#endif - -static void sub(struct csa *csa, int ct, int table[], int level, - int weight, int l_weight) -{ int i, j, k, curr_weight, left_weight, *p1, *p2, *newtable; - newtable = xcalloc(n, sizeof(int)); - if (ct <= 0) - { /* 0 or 1 elements left; include these */ - if (ct == 0) - { set[level++] = table[0]; - weight += l_weight; - } - if (weight > record) - { record = weight; - rec_level = level; - for (i = 0; i < level; i++) rec[i] = set[i]; - } - goto done; - } - for (i = ct; i >= 0; i--) - { if ((level == 0) && (i < ct)) goto done; - k = table[i]; - if ((level > 0) && (clique[k] <= (record - weight))) - goto done; /* prune */ - set[level] = k; - curr_weight = weight + wt[k]; - l_weight -= wt[k]; - if (l_weight <= (record - curr_weight)) - goto done; /* prune */ - p1 = newtable; - p2 = table; - left_weight = 0; - while (p2 < table + i) - { j = *p2++; - if (is_edge(csa, j, k)) - { *p1++ = j; - left_weight += wt[j]; - } - } - if (left_weight <= (record - curr_weight)) continue; - sub(csa, p1 - newtable - 1, newtable, level + 1, curr_weight, - left_weight); - } -done: xfree(newtable); - return; -} - -int wclique(int n_, const int w[], const unsigned char a_[], int ind[]) -{ struct csa csa_, *csa = &csa_; - int i, j, p, max_wt, max_nwt, wth, *used, *nwt, *pos; - double timer; - n = n_; - xassert(n > 0); - wt = &w[1]; - a = a_; - record = 0; - rec_level = 0; - rec = &ind[1]; - clique = xcalloc(n, sizeof(int)); - set = xcalloc(n, sizeof(int)); - used = xcalloc(n, sizeof(int)); - nwt = xcalloc(n, sizeof(int)); - pos = xcalloc(n, sizeof(int)); - /* start timer */ - timer = xtime(); - /* order vertices */ - for (i = 0; i < n; i++) - { nwt[i] = 0; - for (j = 0; j < n; j++) - if (is_edge(csa, i, j)) nwt[i] += wt[j]; - } - for (i = 0; i < n; i++) - used[i] = 0; - for (i = n-1; i >= 0; i--) - { max_wt = -1; - max_nwt = -1; - for (j = 0; j < n; j++) - { if ((!used[j]) && ((wt[j] > max_wt) || (wt[j] == max_wt - && nwt[j] > max_nwt))) - { max_wt = wt[j]; - max_nwt = nwt[j]; - p = j; - } - } - pos[i] = p; - used[p] = 1; - for (j = 0; j < n; j++) - if ((!used[j]) && (j != p) && (is_edge(csa, p, j))) - nwt[j] -= wt[p]; - } - /* main routine */ - wth = 0; - for (i = 0; i < n; i++) - { wth += wt[pos[i]]; - sub(csa, i, pos, 0, 0, wth); - clique[pos[i]] = record; - if (xdifftime(xtime(), timer) >= 5.0 - 0.001) - { /* print current record and reset timer */ - xprintf("level = %d (%d); best = %d\n", i+1, n, record); - timer = xtime(); - } - } - xfree(clique); - xfree(set); - xfree(used); - xfree(nwt); - xfree(pos); - /* return the solution found */ - for (i = 1; i <= rec_level; i++) ind[i]++; - return rec_level; -} - -#undef n -#undef wt -#undef a -#undef record -#undef rec_level -#undef rec -#undef clique -#undef set - -/* eof */ diff --git a/code/3rd_glpk/misc/wclique.h b/code/3rd_glpk/misc/wclique.h deleted file mode 100644 index d52dc805..00000000 --- a/code/3rd_glpk/misc/wclique.h +++ /dev/null @@ -1,33 +0,0 @@ -/* wclique.h (maximum weight clique, Ostergard's algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef WCLIQUE_H -#define WCLIQUE_H - -#define wclique _glp_wclique -int wclique(int n, const int w[], const unsigned char a[], int ind[]); -/* find maximum weight clique with Ostergard's algorithm */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/wclique1.c b/code/3rd_glpk/misc/wclique1.c deleted file mode 100644 index a3d89542..00000000 --- a/code/3rd_glpk/misc/wclique1.c +++ /dev/null @@ -1,317 +0,0 @@ -/* wclique1.c (maximum weight clique, greedy heuristic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "wclique1.h" - -/*********************************************************************** -* NAME -* -* wclique1 - find maximum weight clique with greedy heuristic -* -* SYNOPSIS -* -* #include "wclique1.h" -* int wclique1(int n, const double w[], -* int (*func)(void *info, int i, int ind[]), void *info, int c[]); -* -* DESCRIPTION -* -* The routine wclique1 implements a sequential greedy heuristic to -* find maximum weight clique in a given (undirected) graph G = (V, E). -* -* The parameter n specifies the number of vertices |V| in the graph, -* n >= 0. -* -* The array w specifies vertex weights in locations w[i], i = 1,...,n. -* All weights must be non-negative. -* -* The formal routine func specifies the graph. For a given vertex i, -* 1 <= i <= n, it stores indices of all vertices adjacent to vertex i -* in locations ind[1], ..., ind[deg], where deg is the degree of -* vertex i, 0 <= deg < n, returned on exit. Note that self-loops and -* multiple edges are not allowed. -* -* The parameter info is a cookie passed to the routine func. -* -* On exit the routine wclique1 stores vertex indices included in -* the clique found to locations c[1], ..., c[size], where size is the -* clique size returned by the routine, 0 <= size <= n. -* -* RETURNS -* -* The routine wclique1 returns the size of the clique found. */ - -struct vertex { int i; double cw; }; - -static int CDECL fcmp(const void *xx, const void *yy) -{ const struct vertex *x = xx, *y = yy; - if (x->cw > y->cw) return -1; - if (x->cw < y->cw) return +1; - return 0; -} - -int wclique1(int n, const double w[], - int (*func)(void *info, int i, int ind[]), void *info, int c[]) -{ struct vertex *v_list; - int deg, c_size, d_size, i, j, k, kk, l, *ind, *c_list, *d_list, - size = 0; - double c_wght, d_wght, *sw, best = 0.0; - char *d_flag, *skip; - /* perform sanity checks */ - xassert(n >= 0); - for (i = 1; i <= n; i++) - xassert(w[i] >= 0.0); - /* if the graph is empty, nothing to do */ - if (n == 0) goto done; - /* allocate working arrays */ - ind = xcalloc(1+n, sizeof(int)); - v_list = xcalloc(1+n, sizeof(struct vertex)); - c_list = xcalloc(1+n, sizeof(int)); - d_list = xcalloc(1+n, sizeof(int)); - d_flag = xcalloc(1+n, sizeof(char)); - skip = xcalloc(1+n, sizeof(char)); - sw = xcalloc(1+n, sizeof(double)); - /* build the vertex list */ - for (i = 1; i <= n; i++) - { v_list[i].i = i; - /* compute the cumulative weight of each vertex i, which is - * cw[i] = w[i] + sum{j : (i,j) in E} w[j] */ - v_list[i].cw = w[i]; - deg = func(info, i, ind); - xassert(0 <= deg && deg < n); - for (k = 1; k <= deg; k++) - { j = ind[k]; - xassert(1 <= j && j <= n && j != i); - v_list[i].cw += w[j]; - } - } - /* sort the vertex list to access vertices in descending order of - * cumulative weights */ - qsort(&v_list[1], n, sizeof(struct vertex), fcmp); - /* initially all vertices are unmarked */ - memset(&skip[1], 0, sizeof(char) * n); - /* clear flags of all vertices */ - memset(&d_flag[1], 0, sizeof(char) * n); - /* look through all vertices of the graph */ - for (l = 1; l <= n; l++) - { /* take vertex i */ - i = v_list[l].i; - /* if this vertex was already included in one of previosuly - * constructed cliques, skip it */ - if (skip[i]) continue; - /* use vertex i as the initial clique vertex */ - c_size = 1; /* size of current clique */ - c_list[1] = i; /* list of vertices in current clique */ - c_wght = w[i]; /* weight of current clique */ - /* determine the candidate set D = { j : (i,j) in E } */ - d_size = func(info, i, d_list); - xassert(0 <= d_size && d_size < n); - d_wght = 0.0; /* weight of set D */ - for (k = 1; k <= d_size; k++) - { j = d_list[k]; - xassert(1 <= j && j <= n && j != i); - xassert(!d_flag[j]); - d_flag[j] = 1; - d_wght += w[j]; - } - /* check an upper bound to the final clique weight */ - if (c_wght + d_wght < best + 1e-5 * (1.0 + fabs(best))) - { /* skip constructing the current clique */ - goto next; - } - /* compute the summary weight of each vertex i in D, which is - * sw[i] = w[i] + sum{j in D and (i,j) in E} w[j] */ - for (k = 1; k <= d_size; k++) - { i = d_list[k]; - sw[i] = w[i]; - /* consider vertices adjacent to vertex i */ - deg = func(info, i, ind); - xassert(0 <= deg && deg < n); - for (kk = 1; kk <= deg; kk++) - { j = ind[kk]; - xassert(1 <= j && j <= n && j != i); - if (d_flag[j]) sw[i] += w[j]; - } - } - /* grow the current clique by adding vertices from D */ - while (d_size > 0) - { /* check an upper bound to the final clique weight */ - if (c_wght + d_wght < best + 1e-5 * (1.0 + fabs(best))) - { /* skip constructing the current clique */ - goto next; - } - /* choose vertex i in D having maximal summary weight */ - i = d_list[1]; - for (k = 2; k <= d_size; k++) - { j = d_list[k]; - if (sw[i] < sw[j]) i = j; - } - /* include vertex i in the current clique */ - c_size++; - c_list[c_size] = i; - c_wght += w[i]; - /* remove all vertices not adjacent to vertex i, including - * vertex i itself, from the candidate set D */ - deg = func(info, i, ind); - xassert(0 <= deg && deg < n); - for (k = 1; k <= deg; k++) - { j = ind[k]; - xassert(1 <= j && j <= n && j != i); - /* vertex j is adjacent to vertex i */ - if (d_flag[j]) - { xassert(d_flag[j] == 1); - /* mark vertex j to keep it in D */ - d_flag[j] = 2; - } - } - kk = d_size, d_size = 0; - for (k = 1; k <= kk; k++) - { j = d_list[k]; - if (d_flag[j] == 1) - { /* remove vertex j from D */ - d_flag[j] = 0; - d_wght -= w[j]; - } - else if (d_flag[j] == 2) - { /* keep vertex j in D */ - d_list[++d_size] = j; - d_flag[j] = 1; - } - else - xassert(d_flag != d_flag); - } - } - /* the current clique has been completely constructed */ - if (best < c_wght) - { best = c_wght; - size = c_size; - xassert(1 <= size && size <= n); - memcpy(&c[1], &c_list[1], size * sizeof(int)); - } -next: /* mark the current clique vertices in order not to use them - * as initial vertices anymore */ - for (k = 1; k <= c_size; k++) - skip[c_list[k]] = 1; - /* set D can be non-empty, so clean up vertex flags */ - for (k = 1; k <= d_size; k++) - d_flag[d_list[k]] = 0; - } - /* free working arrays */ - xfree(ind); - xfree(v_list); - xfree(c_list); - xfree(d_list); - xfree(d_flag); - xfree(skip); - xfree(sw); -done: /* return to the calling program */ - return size; -} - -/**********************************************************************/ - -#ifdef GLP_TEST -#include "glpk.h" -#include "rng.h" - -typedef struct { double w; } v_data; - -#define weight(v) (((v_data *)((v)->data))->w) - -glp_graph *G; - -char *flag; - -int func(void *info, int i, int ind[]) -{ glp_arc *e; - int j, k, deg = 0; - xassert(info == NULL); - xassert(1 <= i && i <= G->nv); - /* look through incoming arcs */ - for (e = G->v[i]->in; e != NULL; e = e->h_next) - { j = e->tail->i; /* j->i */ - if (j != i && !flag[j]) ind[++deg] = j, flag[j] = 1; - } - /* look through outgoing arcs */ - for (e = G->v[i]->out; e != NULL; e = e->t_next) - { j = e->head->i; /* i->j */ - if (j != i && !flag[j]) ind[++deg] = j, flag[j] = 1; - } - /* clear the flag array */ - xassert(deg < G->nv); - for (k = 1; k <= deg; k++) flag[ind[k]] = 0; - return deg; -} - -int main(int argc, char *argv[]) -{ RNG *rand; - int i, k, kk, size, *c, *ind, deg; - double *w, sum, t; - /* read graph in DIMACS format */ - G = glp_create_graph(sizeof(v_data), 0); - xassert(argc == 2); - xassert(glp_read_ccdata(G, offsetof(v_data, w), argv[1]) == 0); - /* print the number of connected components */ - xprintf("nc = %d\n", glp_weak_comp(G, -1)); - /* assign random weights unformly distributed in [1,100] */ - w = xcalloc(1+G->nv, sizeof(double)); - rand = rng_create_rand(); - for (i = 1; i <= G->nv; i++) -#if 0 - w[i] = weight(G->v[i]) = 1.0; -#else - w[i] = weight(G->v[i]) = rng_unif_rand(rand, 100) + 1; -#endif - /* write graph in DIMACS format */ - xassert(glp_write_ccdata(G, offsetof(v_data, w), "graph") == 0); - /* find maximum weight clique */ - c = xcalloc(1+G->nv, sizeof(int)); - flag = xcalloc(1+G->nv, sizeof(char)); - memset(&flag[1], 0, G->nv); - t = xtime(); - size = wclique1(G->nv, w, func, NULL, c); - xprintf("Time used: %.1f s\n", xdifftime(xtime(), t)); - /* check the clique found */ - ind = xcalloc(1+G->nv, sizeof(int)); - for (k = 1; k <= size; k++) - { i = c[k]; - deg = func(NULL, i, ind); - for (kk = 1; kk <= size; kk++) - flag[c[kk]] = 1; - flag[i] = 0; - for (kk = 1; kk <= deg; kk++) - flag[ind[kk]] = 0; - for (kk = 1; kk <= size; kk++) - xassert(flag[c[kk]] == 0); - } - /* compute the clique weight */ - sum = 0.0; - for (i = 1; i <= size; i++) - sum += w[c[i]]; - xprintf("size = %d; sum = %g\n", size, sum); - return 0; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/misc/wclique1.h b/code/3rd_glpk/misc/wclique1.h deleted file mode 100644 index 588f3257..00000000 --- a/code/3rd_glpk/misc/wclique1.h +++ /dev/null @@ -1,34 +0,0 @@ -/* wclique1.h (maximum weight clique, greedy heuristic) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2012-2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef WCLIQUE1_H -#define WCLIQUE1_H - -#define wclique1 _glp_wclique1 -int wclique1(int n, const double w[], - int (*func)(void *info, int i, int ind[]), void *info, int c[]); -/* find maximum weight clique with greedy heuristic */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl.h b/code/3rd_glpk/mpl/mpl.h deleted file mode 100644 index ddd31543..00000000 --- a/code/3rd_glpk/mpl/mpl.h +++ /dev/null @@ -1,2598 +0,0 @@ -/* mpl.h (GNU MathProg translator) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MPL_H -#define MPL_H - -#include "avl.h" -#include "dmp.h" -#include "env.h" -#include "misc.h" -#include "rng.h" - -#if 0 /* 22/I-2013 */ -typedef struct MPL MPL; -#else -typedef struct glp_tran MPL; -#endif -typedef char STRING; -typedef struct SYMBOL SYMBOL; -typedef struct TUPLE TUPLE; -typedef struct ARRAY ELEMSET; -typedef struct ELEMVAR ELEMVAR; -typedef struct FORMULA FORMULA; -typedef struct ELEMCON ELEMCON; -typedef union VALUE VALUE; -typedef struct ARRAY ARRAY; -typedef struct MEMBER MEMBER; -#if 1 -/* many C compilers have DOMAIN declared in :( */ -#undef DOMAIN -#define DOMAIN DOMAIN1 -#endif -typedef struct DOMAIN DOMAIN; -typedef struct DOMAIN_BLOCK DOMAIN_BLOCK; -typedef struct DOMAIN_SLOT DOMAIN_SLOT; -typedef struct SET SET; -typedef struct WITHIN WITHIN; -typedef struct GADGET GADGET; -typedef struct PARAMETER PARAMETER; -typedef struct CONDITION CONDITION; -typedef struct VARIABLE VARIABLE; -typedef struct CONSTRAINT CONSTRAINT; -typedef struct TABLE TABLE; -typedef struct TABARG TABARG; -typedef struct TABFLD TABFLD; -typedef struct TABIN TABIN; -typedef struct TABOUT TABOUT; -typedef struct TABDCA TABDCA; -typedef union OPERANDS OPERANDS; -typedef struct ARG_LIST ARG_LIST; -typedef struct CODE CODE; -typedef struct CHECK CHECK; -typedef struct DISPLAY DISPLAY; -typedef struct DISPLAY1 DISPLAY1; -typedef struct PRINTF PRINTF; -typedef struct PRINTF1 PRINTF1; -typedef struct FOR FOR; -typedef struct STATEMENT STATEMENT; -typedef struct TUPLE SLICE; - -/**********************************************************************/ -/* * * TRANSLATOR DATABASE * * */ -/**********************************************************************/ - -#define A_BINARY 101 /* something binary */ -#define A_CHECK 102 /* check statement */ -#define A_CONSTRAINT 103 /* model constraint */ -#define A_DISPLAY 104 /* display statement */ -#define A_ELEMCON 105 /* elemental constraint/objective */ -#define A_ELEMSET 106 /* elemental set */ -#define A_ELEMVAR 107 /* elemental variable */ -#define A_EXPRESSION 108 /* expression */ -#define A_FOR 109 /* for statement */ -#define A_FORMULA 110 /* formula */ -#define A_INDEX 111 /* dummy index */ -#define A_INPUT 112 /* input table */ -#define A_INTEGER 113 /* something integer */ -#define A_LOGICAL 114 /* something logical */ -#define A_MAXIMIZE 115 /* objective has to be maximized */ -#define A_MINIMIZE 116 /* objective has to be minimized */ -#define A_NONE 117 /* nothing */ -#define A_NUMERIC 118 /* something numeric */ -#define A_OUTPUT 119 /* output table */ -#define A_PARAMETER 120 /* model parameter */ -#define A_PRINTF 121 /* printf statement */ -#define A_SET 122 /* model set */ -#define A_SOLVE 123 /* solve statement */ -#define A_SYMBOLIC 124 /* something symbolic */ -#define A_TABLE 125 /* data table */ -#define A_TUPLE 126 /* n-tuple */ -#define A_VARIABLE 127 /* model variable */ - -#define MAX_LENGTH 100 -/* maximal length of any symbolic value (this includes symbolic names, - numeric and string literals, and all symbolic values that may appear - during the evaluation phase) */ - -#define CONTEXT_SIZE 60 -/* size of the context queue, in characters */ - -#define OUTBUF_SIZE 1024 -/* size of the output buffer, in characters */ - -#if 0 /* 22/I-2013 */ -struct MPL -#else -struct glp_tran -#endif -{ /* translator database */ - /*--------------------------------------------------------------*/ - /* scanning segment */ - int line; - /* number of the current text line */ - int c; - /* the current character or EOF */ - int token; - /* the current token: */ -#define T_EOF 201 /* end of file */ -#define T_NAME 202 /* symbolic name (model section only) */ -#define T_SYMBOL 203 /* symbol (data section only) */ -#define T_NUMBER 204 /* numeric literal */ -#define T_STRING 205 /* string literal */ -#define T_AND 206 /* and && */ -#define T_BY 207 /* by */ -#define T_CROSS 208 /* cross */ -#define T_DIFF 209 /* diff */ -#define T_DIV 210 /* div */ -#define T_ELSE 211 /* else */ -#define T_IF 212 /* if */ -#define T_IN 213 /* in */ -#define T_INFINITY 214 /* Infinity */ -#define T_INTER 215 /* inter */ -#define T_LESS 216 /* less */ -#define T_MOD 217 /* mod */ -#define T_NOT 218 /* not ! */ -#define T_OR 219 /* or || */ -#define T_SPTP 220 /* s.t. */ -#define T_SYMDIFF 221 /* symdiff */ -#define T_THEN 222 /* then */ -#define T_UNION 223 /* union */ -#define T_WITHIN 224 /* within */ -#define T_PLUS 225 /* + */ -#define T_MINUS 226 /* - */ -#define T_ASTERISK 227 /* * */ -#define T_SLASH 228 /* / */ -#define T_POWER 229 /* ^ ** */ -#define T_LT 230 /* < */ -#define T_LE 231 /* <= */ -#define T_EQ 232 /* = == */ -#define T_GE 233 /* >= */ -#define T_GT 234 /* > */ -#define T_NE 235 /* <> != */ -#define T_CONCAT 236 /* & */ -#define T_BAR 237 /* | */ -#define T_POINT 238 /* . */ -#define T_COMMA 239 /* , */ -#define T_COLON 240 /* : */ -#define T_SEMICOLON 241 /* ; */ -#define T_ASSIGN 242 /* := */ -#define T_DOTS 243 /* .. */ -#define T_LEFT 244 /* ( */ -#define T_RIGHT 245 /* ) */ -#define T_LBRACKET 246 /* [ */ -#define T_RBRACKET 247 /* ] */ -#define T_LBRACE 248 /* { */ -#define T_RBRACE 249 /* } */ -#define T_APPEND 250 /* >> */ -#define T_TILDE 251 /* ~ */ -#define T_INPUT 252 /* <- */ - int imlen; - /* length of the current token */ - char *image; /* char image[MAX_LENGTH+1]; */ - /* image of the current token */ - double value; - /* value of the current token (for T_NUMBER only) */ - int b_token; - /* the previous token */ - int b_imlen; - /* length of the previous token */ - char *b_image; /* char b_image[MAX_LENGTH+1]; */ - /* image of the previous token */ - double b_value; - /* value of the previous token (if token is T_NUMBER) */ - int f_dots; - /* if this flag is set, the next token should be recognized as - T_DOTS, not as T_POINT */ - int f_scan; - /* if this flag is set, the next token is already scanned */ - int f_token; - /* the next token */ - int f_imlen; - /* length of the next token */ - char *f_image; /* char f_image[MAX_LENGTH+1]; */ - /* image of the next token */ - double f_value; - /* value of the next token (if token is T_NUMBER) */ - char *context; /* char context[CONTEXT_SIZE]; */ - /* context circular queue (not null-terminated!) */ - int c_ptr; - /* pointer to the current position in the context queue */ - int flag_d; - /* if this flag is set, the data section is being processed */ - /*--------------------------------------------------------------*/ - /* translating segment */ - DMP *pool; - /* memory pool used to allocate all data instances created during - the translation phase */ - AVL *tree; - /* symbolic name table: - node.type = A_INDEX => node.link -> DOMAIN_SLOT - node.type = A_SET => node.link -> SET - node.type = A_PARAMETER => node.link -> PARAMETER - node.type = A_VARIABLE => node.link -> VARIABLE - node.type = A_CONSTRANT => node.link -> CONSTRAINT */ - STATEMENT *model; - /* linked list of model statements in the original order */ - int flag_x; - /* if this flag is set, the current token being left parenthesis - begins a slice that allows recognizing any undeclared symbolic - names as dummy indices; this flag is automatically reset once - the next token has been scanned */ - int as_within; - /* the warning "in understood as within" has been issued */ - int as_in; - /* the warning "within understood as in" has been issued */ - int as_binary; - /* the warning "logical understood as binary" has been issued */ - int flag_s; - /* if this flag is set, the solve statement has been parsed */ - /*--------------------------------------------------------------*/ - /* common segment */ - DMP *strings; - /* memory pool to allocate STRING data structures */ - DMP *symbols; - /* memory pool to allocate SYMBOL data structures */ - DMP *tuples; - /* memory pool to allocate TUPLE data structures */ - DMP *arrays; - /* memory pool to allocate ARRAY data structures */ - DMP *members; - /* memory pool to allocate MEMBER data structures */ - DMP *elemvars; - /* memory pool to allocate ELEMVAR data structures */ - DMP *formulae; - /* memory pool to allocate FORMULA data structures */ - DMP *elemcons; - /* memory pool to allocate ELEMCON data structures */ - ARRAY *a_list; - /* linked list of all arrays in the database */ - char *sym_buf; /* char sym_buf[255+1]; */ - /* working buffer used by the routine format_symbol */ - char *tup_buf; /* char tup_buf[255+1]; */ - /* working buffer used by the routine format_tuple */ - /*--------------------------------------------------------------*/ - /* generating/postsolving segment */ - RNG *rand; - /* pseudo-random number generator */ - int flag_p; - /* if this flag is set, the postsolving phase is in effect */ - STATEMENT *stmt; - /* model statement being currently executed */ - TABDCA *dca; - /* pointer to table driver communication area for table statement - currently executed */ - int m; - /* number of rows in the problem, m >= 0 */ - int n; - /* number of columns in the problem, n >= 0 */ - ELEMCON **row; /* ELEMCON *row[1+m]; */ - /* row[0] is not used; - row[i] is elemental constraint or objective, which corresponds - to i-th row of the problem, 1 <= i <= m */ - ELEMVAR **col; /* ELEMVAR *col[1+n]; */ - /* col[0] is not used; - col[j] is elemental variable, which corresponds to j-th column - of the problem, 1 <= j <= n */ - /*--------------------------------------------------------------*/ - /* input/output segment */ - glp_file *in_fp; - /* stream assigned to the input text file */ - char *in_file; - /* name of the input text file */ - glp_file *out_fp; - /* stream assigned to the output text file used to write all data - produced by display and printf statements; NULL means the data - should be sent to stdout via the routine xprintf */ - char *out_file; - /* name of the output text file */ -#if 0 /* 08/XI-2009 */ - char *out_buf; /* char out_buf[OUTBUF_SIZE] */ - /* buffer to accumulate output data */ - int out_cnt; - /* count of data bytes stored in the output buffer */ -#endif - glp_file *prt_fp; - /* stream assigned to the print text file; may be NULL */ - char *prt_file; - /* name of the output print file */ - /*--------------------------------------------------------------*/ - /* solver interface segment */ - jmp_buf jump; - /* jump address for non-local go to in case of error */ - int phase; - /* phase of processing: - 0 - database is being or has been initialized - 1 - model section is being or has been read - 2 - data section is being or has been read - 3 - model is being or has been generated/postsolved - 4 - model processing error has occurred */ - char *mod_file; - /* name of the input text file, which contains model section */ - char *mpl_buf; /* char mpl_buf[255+1]; */ - /* working buffer used by some interface routines */ -}; - -/**********************************************************************/ -/* * * PROCESSING MODEL SECTION * * */ -/**********************************************************************/ - -#define alloc(type) ((type *)dmp_get_atomv(mpl->pool, sizeof(type))) -/* allocate atom of given type */ - -#define enter_context _glp_mpl_enter_context -void enter_context(MPL *mpl); -/* enter current token into context queue */ - -#define print_context _glp_mpl_print_context -void print_context(MPL *mpl); -/* print current content of context queue */ - -#define get_char _glp_mpl_get_char -void get_char(MPL *mpl); -/* scan next character from input text file */ - -#define append_char _glp_mpl_append_char -void append_char(MPL *mpl); -/* append character to current token */ - -#define get_token _glp_mpl_get_token -void get_token(MPL *mpl); -/* scan next token from input text file */ - -#define unget_token _glp_mpl_unget_token -void unget_token(MPL *mpl); -/* return current token back to input stream */ - -#define is_keyword _glp_mpl_is_keyword -int is_keyword(MPL *mpl, char *keyword); -/* check if current token is given non-reserved keyword */ - -#define is_reserved _glp_mpl_is_reserved -int is_reserved(MPL *mpl); -/* check if current token is reserved keyword */ - -#define make_code _glp_mpl_make_code -CODE *make_code(MPL *mpl, int op, OPERANDS *arg, int type, int dim); -/* generate pseudo-code (basic routine) */ - -#define make_unary _glp_mpl_make_unary -CODE *make_unary(MPL *mpl, int op, CODE *x, int type, int dim); -/* generate pseudo-code for unary operation */ - -#define make_binary _glp_mpl_make_binary -CODE *make_binary(MPL *mpl, int op, CODE *x, CODE *y, int type, - int dim); -/* generate pseudo-code for binary operation */ - -#define make_ternary _glp_mpl_make_ternary -CODE *make_ternary(MPL *mpl, int op, CODE *x, CODE *y, CODE *z, - int type, int dim); -/* generate pseudo-code for ternary operation */ - -#define numeric_literal _glp_mpl_numeric_literal -CODE *numeric_literal(MPL *mpl); -/* parse reference to numeric literal */ - -#define string_literal _glp_mpl_string_literal -CODE *string_literal(MPL *mpl); -/* parse reference to string literal */ - -#define create_arg_list _glp_mpl_create_arg_list -ARG_LIST *create_arg_list(MPL *mpl); -/* create empty operands list */ - -#define expand_arg_list _glp_mpl_expand_arg_list -ARG_LIST *expand_arg_list(MPL *mpl, ARG_LIST *list, CODE *x); -/* append operand to operands list */ - -#define arg_list_len _glp_mpl_arg_list_len -int arg_list_len(MPL *mpl, ARG_LIST *list); -/* determine length of operands list */ - -#define subscript_list _glp_mpl_subscript_list -ARG_LIST *subscript_list(MPL *mpl); -/* parse subscript list */ - -#define object_reference _glp_mpl_object_reference -CODE *object_reference(MPL *mpl); -/* parse reference to named object */ - -#define numeric_argument _glp_mpl_numeric_argument -CODE *numeric_argument(MPL *mpl, char *func); -/* parse argument passed to built-in function */ - -#define symbolic_argument _glp_mpl_symbolic_argument -CODE *symbolic_argument(MPL *mpl, char *func); - -#define elemset_argument _glp_mpl_elemset_argument -CODE *elemset_argument(MPL *mpl, char *func); - -#define function_reference _glp_mpl_function_reference -CODE *function_reference(MPL *mpl); -/* parse reference to built-in function */ - -#define create_domain _glp_mpl_create_domain -DOMAIN *create_domain(MPL *mpl); -/* create empty domain */ - -#define create_block _glp_mpl_create_block -DOMAIN_BLOCK *create_block(MPL *mpl); -/* create empty domain block */ - -#define append_block _glp_mpl_append_block -void append_block(MPL *mpl, DOMAIN *domain, DOMAIN_BLOCK *block); -/* append domain block to specified domain */ - -#define append_slot _glp_mpl_append_slot -DOMAIN_SLOT *append_slot(MPL *mpl, DOMAIN_BLOCK *block, char *name, - CODE *code); -/* create and append new slot to domain block */ - -#define expression_list _glp_mpl_expression_list -CODE *expression_list(MPL *mpl); -/* parse expression list */ - -#define literal_set _glp_mpl_literal_set -CODE *literal_set(MPL *mpl, CODE *code); -/* parse literal set */ - -#define indexing_expression _glp_mpl_indexing_expression -DOMAIN *indexing_expression(MPL *mpl); -/* parse indexing expression */ - -#define close_scope _glp_mpl_close_scope -void close_scope(MPL *mpl, DOMAIN *domain); -/* close scope of indexing expression */ - -#define iterated_expression _glp_mpl_iterated_expression -CODE *iterated_expression(MPL *mpl); -/* parse iterated expression */ - -#define domain_arity _glp_mpl_domain_arity -int domain_arity(MPL *mpl, DOMAIN *domain); -/* determine arity of domain */ - -#define set_expression _glp_mpl_set_expression -CODE *set_expression(MPL *mpl); -/* parse set expression */ - -#define branched_expression _glp_mpl_branched_expression -CODE *branched_expression(MPL *mpl); -/* parse conditional expression */ - -#define primary_expression _glp_mpl_primary_expression -CODE *primary_expression(MPL *mpl); -/* parse primary expression */ - -#define error_preceding _glp_mpl_error_preceding -void error_preceding(MPL *mpl, char *opstr); -/* raise error if preceding operand has wrong type */ - -#define error_following _glp_mpl_error_following -void error_following(MPL *mpl, char *opstr); -/* raise error if following operand has wrong type */ - -#define error_dimension _glp_mpl_error_dimension -void error_dimension(MPL *mpl, char *opstr, int dim1, int dim2); -/* raise error if operands have different dimension */ - -#define expression_0 _glp_mpl_expression_0 -CODE *expression_0(MPL *mpl); -/* parse expression of level 0 */ - -#define expression_1 _glp_mpl_expression_1 -CODE *expression_1(MPL *mpl); -/* parse expression of level 1 */ - -#define expression_2 _glp_mpl_expression_2 -CODE *expression_2(MPL *mpl); -/* parse expression of level 2 */ - -#define expression_3 _glp_mpl_expression_3 -CODE *expression_3(MPL *mpl); -/* parse expression of level 3 */ - -#define expression_4 _glp_mpl_expression_4 -CODE *expression_4(MPL *mpl); -/* parse expression of level 4 */ - -#define expression_5 _glp_mpl_expression_5 -CODE *expression_5(MPL *mpl); -/* parse expression of level 5 */ - -#define expression_6 _glp_mpl_expression_6 -CODE *expression_6(MPL *mpl); -/* parse expression of level 6 */ - -#define expression_7 _glp_mpl_expression_7 -CODE *expression_7(MPL *mpl); -/* parse expression of level 7 */ - -#define expression_8 _glp_mpl_expression_8 -CODE *expression_8(MPL *mpl); -/* parse expression of level 8 */ - -#define expression_9 _glp_mpl_expression_9 -CODE *expression_9(MPL *mpl); -/* parse expression of level 9 */ - -#define expression_10 _glp_mpl_expression_10 -CODE *expression_10(MPL *mpl); -/* parse expression of level 10 */ - -#define expression_11 _glp_mpl_expression_11 -CODE *expression_11(MPL *mpl); -/* parse expression of level 11 */ - -#define expression_12 _glp_mpl_expression_12 -CODE *expression_12(MPL *mpl); -/* parse expression of level 12 */ - -#define expression_13 _glp_mpl_expression_13 -CODE *expression_13(MPL *mpl); -/* parse expression of level 13 */ - -#define set_statement _glp_mpl_set_statement -SET *set_statement(MPL *mpl); -/* parse set statement */ - -#define parameter_statement _glp_mpl_parameter_statement -PARAMETER *parameter_statement(MPL *mpl); -/* parse parameter statement */ - -#define variable_statement _glp_mpl_variable_statement -VARIABLE *variable_statement(MPL *mpl); -/* parse variable statement */ - -#define constraint_statement _glp_mpl_constraint_statement -CONSTRAINT *constraint_statement(MPL *mpl); -/* parse constraint statement */ - -#define objective_statement _glp_mpl_objective_statement -CONSTRAINT *objective_statement(MPL *mpl); -/* parse objective statement */ - -#define table_statement _glp_mpl_table_statement -TABLE *table_statement(MPL *mpl); -/* parse table statement */ - -#define solve_statement _glp_mpl_solve_statement -void *solve_statement(MPL *mpl); -/* parse solve statement */ - -#define check_statement _glp_mpl_check_statement -CHECK *check_statement(MPL *mpl); -/* parse check statement */ - -#define display_statement _glp_mpl_display_statement -DISPLAY *display_statement(MPL *mpl); -/* parse display statement */ - -#define printf_statement _glp_mpl_printf_statement -PRINTF *printf_statement(MPL *mpl); -/* parse printf statement */ - -#define for_statement _glp_mpl_for_statement -FOR *for_statement(MPL *mpl); -/* parse for statement */ - -#define end_statement _glp_mpl_end_statement -void end_statement(MPL *mpl); -/* parse end statement */ - -#define simple_statement _glp_mpl_simple_statement -STATEMENT *simple_statement(MPL *mpl, int spec); -/* parse simple statement */ - -#define model_section _glp_mpl_model_section -void model_section(MPL *mpl); -/* parse model section */ - -/**********************************************************************/ -/* * * PROCESSING DATA SECTION * * */ -/**********************************************************************/ - -#if 2 + 2 == 5 -struct SLICE /* see TUPLE */ -{ /* component of slice; the slice itself is associated with its - first component; slices are similar to n-tuples with exception - that some slice components (which are indicated by asterisks) - don't refer to any symbols */ - SYMBOL *sym; - /* symbol, which this component refers to; can be NULL */ - SLICE *next; - /* the next component of slice */ -}; -#endif - -#define create_slice _glp_mpl_create_slice -SLICE *create_slice(MPL *mpl); -/* create slice */ - -#define expand_slice _glp_mpl_expand_slice -SLICE *expand_slice -( MPL *mpl, - SLICE *slice, /* destroyed */ - SYMBOL *sym /* destroyed */ -); -/* append new component to slice */ - -#define slice_dimen _glp_mpl_slice_dimen -int slice_dimen -( MPL *mpl, - SLICE *slice /* not changed */ -); -/* determine dimension of slice */ - -#define slice_arity _glp_mpl_slice_arity -int slice_arity -( MPL *mpl, - SLICE *slice /* not changed */ -); -/* determine arity of slice */ - -#define fake_slice _glp_mpl_fake_slice -SLICE *fake_slice(MPL *mpl, int dim); -/* create fake slice of all asterisks */ - -#define delete_slice _glp_mpl_delete_slice -void delete_slice -( MPL *mpl, - SLICE *slice /* destroyed */ -); -/* delete slice */ - -#define is_number _glp_mpl_is_number -int is_number(MPL *mpl); -/* check if current token is number */ - -#define is_symbol _glp_mpl_is_symbol -int is_symbol(MPL *mpl); -/* check if current token is symbol */ - -#define is_literal _glp_mpl_is_literal -int is_literal(MPL *mpl, char *literal); -/* check if current token is given symbolic literal */ - -#define read_number _glp_mpl_read_number -double read_number(MPL *mpl); -/* read number */ - -#define read_symbol _glp_mpl_read_symbol -SYMBOL *read_symbol(MPL *mpl); -/* read symbol */ - -#define read_slice _glp_mpl_read_slice -SLICE *read_slice -( MPL *mpl, - char *name, /* not changed */ - int dim -); -/* read slice */ - -#define select_set _glp_mpl_select_set -SET *select_set -( MPL *mpl, - char *name /* not changed */ -); -/* select set to saturate it with elemental sets */ - -#define simple_format _glp_mpl_simple_format -void simple_format -( MPL *mpl, - SET *set, /* not changed */ - MEMBER *memb, /* modified */ - SLICE *slice /* not changed */ -); -/* read set data block in simple format */ - -#define matrix_format _glp_mpl_matrix_format -void matrix_format -( MPL *mpl, - SET *set, /* not changed */ - MEMBER *memb, /* modified */ - SLICE *slice, /* not changed */ - int tr -); -/* read set data block in matrix format */ - -#define set_data _glp_mpl_set_data -void set_data(MPL *mpl); -/* read set data */ - -#define select_parameter _glp_mpl_select_parameter -PARAMETER *select_parameter -( MPL *mpl, - char *name /* not changed */ -); -/* select parameter to saturate it with data */ - -#define set_default _glp_mpl_set_default -void set_default -( MPL *mpl, - PARAMETER *par, /* not changed */ - SYMBOL *altval /* destroyed */ -); -/* set default parameter value */ - -#define read_value _glp_mpl_read_value -MEMBER *read_value -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* destroyed */ -); -/* read value and assign it to parameter member */ - -#define plain_format _glp_mpl_plain_format -void plain_format -( MPL *mpl, - PARAMETER *par, /* not changed */ - SLICE *slice /* not changed */ -); -/* read parameter data block in plain format */ - -#define tabular_format _glp_mpl_tabular_format -void tabular_format -( MPL *mpl, - PARAMETER *par, /* not changed */ - SLICE *slice, /* not changed */ - int tr -); -/* read parameter data block in tabular format */ - -#define tabbing_format _glp_mpl_tabbing_format -void tabbing_format -( MPL *mpl, - SYMBOL *altval /* not changed */ -); -/* read parameter data block in tabbing format */ - -#define parameter_data _glp_mpl_parameter_data -void parameter_data(MPL *mpl); -/* read parameter data */ - -#define data_section _glp_mpl_data_section -void data_section(MPL *mpl); -/* read data section */ - -/**********************************************************************/ -/* * * FLOATING-POINT NUMBERS * * */ -/**********************************************************************/ - -#define fp_add _glp_mpl_fp_add -double fp_add(MPL *mpl, double x, double y); -/* floating-point addition */ - -#define fp_sub _glp_mpl_fp_sub -double fp_sub(MPL *mpl, double x, double y); -/* floating-point subtraction */ - -#define fp_less _glp_mpl_fp_less -double fp_less(MPL *mpl, double x, double y); -/* floating-point non-negative subtraction */ - -#define fp_mul _glp_mpl_fp_mul -double fp_mul(MPL *mpl, double x, double y); -/* floating-point multiplication */ - -#define fp_div _glp_mpl_fp_div -double fp_div(MPL *mpl, double x, double y); -/* floating-point division */ - -#define fp_idiv _glp_mpl_fp_idiv -double fp_idiv(MPL *mpl, double x, double y); -/* floating-point quotient of exact division */ - -#define fp_mod _glp_mpl_fp_mod -double fp_mod(MPL *mpl, double x, double y); -/* floating-point remainder of exact division */ - -#define fp_power _glp_mpl_fp_power -double fp_power(MPL *mpl, double x, double y); -/* floating-point exponentiation (raise to power) */ - -#define fp_exp _glp_mpl_fp_exp -double fp_exp(MPL *mpl, double x); -/* floating-point base-e exponential */ - -#define fp_log _glp_mpl_fp_log -double fp_log(MPL *mpl, double x); -/* floating-point natural logarithm */ - -#define fp_log10 _glp_mpl_fp_log10 -double fp_log10(MPL *mpl, double x); -/* floating-point common (decimal) logarithm */ - -#define fp_sqrt _glp_mpl_fp_sqrt -double fp_sqrt(MPL *mpl, double x); -/* floating-point square root */ - -#define fp_sin _glp_mpl_fp_sin -double fp_sin(MPL *mpl, double x); -/* floating-point trigonometric sine */ - -#define fp_cos _glp_mpl_fp_cos -double fp_cos(MPL *mpl, double x); -/* floating-point trigonometric cosine */ - -#define fp_tan _glp_mpl_fp_tan -double fp_tan(MPL *mpl, double x); -/* floating-point trigonometric tangent */ - -#define fp_atan _glp_mpl_fp_atan -double fp_atan(MPL *mpl, double x); -/* floating-point trigonometric arctangent */ - -#define fp_atan2 _glp_mpl_fp_atan2 -double fp_atan2(MPL *mpl, double y, double x); -/* floating-point trigonometric arctangent */ - -#define fp_round _glp_mpl_fp_round -double fp_round(MPL *mpl, double x, double n); -/* round floating-point value to n fractional digits */ - -#define fp_trunc _glp_mpl_fp_trunc -double fp_trunc(MPL *mpl, double x, double n); -/* truncate floating-point value to n fractional digits */ - -/**********************************************************************/ -/* * * PSEUDO-RANDOM NUMBER GENERATORS * * */ -/**********************************************************************/ - -#define fp_irand224 _glp_mpl_fp_irand224 -double fp_irand224(MPL *mpl); -/* pseudo-random integer in the range [0, 2^24) */ - -#define fp_uniform01 _glp_mpl_fp_uniform01 -double fp_uniform01(MPL *mpl); -/* pseudo-random number in the range [0, 1) */ - -#define fp_uniform _glp_mpl_uniform -double fp_uniform(MPL *mpl, double a, double b); -/* pseudo-random number in the range [a, b) */ - -#define fp_normal01 _glp_mpl_fp_normal01 -double fp_normal01(MPL *mpl); -/* Gaussian random variate with mu = 0 and sigma = 1 */ - -#define fp_normal _glp_mpl_fp_normal -double fp_normal(MPL *mpl, double mu, double sigma); -/* Gaussian random variate with specified mu and sigma */ - -/**********************************************************************/ -/* * * DATE/TIME * * */ -/**********************************************************************/ - -#define fn_gmtime _glp_mpl_fn_gmtime -double fn_gmtime(MPL *mpl); -/* obtain the current calendar time (UTC) */ - -#define fn_str2time _glp_mpl_fn_str2time -double fn_str2time(MPL *mpl, const char *str, const char *fmt); -/* convert character string to the calendar time */ - -#define fn_time2str _glp_mpl_fn_time2str -void fn_time2str(MPL *mpl, char *str, double t, const char *fmt); -/* convert the calendar time to character string */ - -/**********************************************************************/ -/* * * CHARACTER STRINGS * * */ -/**********************************************************************/ - -#define create_string _glp_mpl_create_string -STRING *create_string -( MPL *mpl, - char buf[MAX_LENGTH+1] /* not changed */ -); -/* create character string */ - -#define copy_string _glp_mpl_copy_string -STRING *copy_string -( MPL *mpl, - STRING *str /* not changed */ -); -/* make copy of character string */ - -#define compare_strings _glp_mpl_compare_strings -int compare_strings -( MPL *mpl, - STRING *str1, /* not changed */ - STRING *str2 /* not changed */ -); -/* compare one character string with another */ - -#define fetch_string _glp_mpl_fetch_string -char *fetch_string -( MPL *mpl, - STRING *str, /* not changed */ - char buf[MAX_LENGTH+1] /* modified */ -); -/* extract content of character string */ - -#define delete_string _glp_mpl_delete_string -void delete_string -( MPL *mpl, - STRING *str /* destroyed */ -); -/* delete character string */ - -/**********************************************************************/ -/* * * SYMBOLS * * */ -/**********************************************************************/ - -struct SYMBOL -{ /* symbol (numeric or abstract quantity) */ - double num; - /* numeric value of symbol (used only if str == NULL) */ - STRING *str; - /* abstract value of symbol (used only if str != NULL) */ -}; - -#define create_symbol_num _glp_mpl_create_symbol_num -SYMBOL *create_symbol_num(MPL *mpl, double num); -/* create symbol of numeric type */ - -#define create_symbol_str _glp_mpl_create_symbol_str -SYMBOL *create_symbol_str -( MPL *mpl, - STRING *str /* destroyed */ -); -/* create symbol of abstract type */ - -#define copy_symbol _glp_mpl_copy_symbol -SYMBOL *copy_symbol -( MPL *mpl, - SYMBOL *sym /* not changed */ -); -/* make copy of symbol */ - -#define compare_symbols _glp_mpl_compare_symbols -int compare_symbols -( MPL *mpl, - SYMBOL *sym1, /* not changed */ - SYMBOL *sym2 /* not changed */ -); -/* compare one symbol with another */ - -#define delete_symbol _glp_mpl_delete_symbol -void delete_symbol -( MPL *mpl, - SYMBOL *sym /* destroyed */ -); -/* delete symbol */ - -#define format_symbol _glp_mpl_format_symbol -char *format_symbol -( MPL *mpl, - SYMBOL *sym /* not changed */ -); -/* format symbol for displaying or printing */ - -#define concat_symbols _glp_mpl_concat_symbols -SYMBOL *concat_symbols -( MPL *mpl, - SYMBOL *sym1, /* destroyed */ - SYMBOL *sym2 /* destroyed */ -); -/* concatenate one symbol with another */ - -/**********************************************************************/ -/* * * N-TUPLES * * */ -/**********************************************************************/ - -struct TUPLE -{ /* component of n-tuple; the n-tuple itself is associated with - its first component; (note that 0-tuple has no components) */ - SYMBOL *sym; - /* symbol, which the component refers to; cannot be NULL */ - TUPLE *next; - /* the next component of n-tuple */ -}; - -#define create_tuple _glp_mpl_create_tuple -TUPLE *create_tuple(MPL *mpl); -/* create n-tuple */ - -#define expand_tuple _glp_mpl_expand_tuple -TUPLE *expand_tuple -( MPL *mpl, - TUPLE *tuple, /* destroyed */ - SYMBOL *sym /* destroyed */ -); -/* append symbol to n-tuple */ - -#define tuple_dimen _glp_mpl_tuple_dimen -int tuple_dimen -( MPL *mpl, - TUPLE *tuple /* not changed */ -); -/* determine dimension of n-tuple */ - -#define copy_tuple _glp_mpl_copy_tuple -TUPLE *copy_tuple -( MPL *mpl, - TUPLE *tuple /* not changed */ -); -/* make copy of n-tuple */ - -#define compare_tuples _glp_mpl_compare_tuples -int compare_tuples -( MPL *mpl, - TUPLE *tuple1, /* not changed */ - TUPLE *tuple2 /* not changed */ -); -/* compare one n-tuple with another */ - -#define build_subtuple _glp_mpl_build_subtuple -TUPLE *build_subtuple -( MPL *mpl, - TUPLE *tuple, /* not changed */ - int dim -); -/* build subtuple of given n-tuple */ - -#define delete_tuple _glp_mpl_delete_tuple -void delete_tuple -( MPL *mpl, - TUPLE *tuple /* destroyed */ -); -/* delete n-tuple */ - -#define format_tuple _glp_mpl_format_tuple -char *format_tuple -( MPL *mpl, - int c, - TUPLE *tuple /* not changed */ -); -/* format n-tuple for displaying or printing */ - -/**********************************************************************/ -/* * * ELEMENTAL SETS * * */ -/**********************************************************************/ - -#if 2 + 2 == 5 -struct ELEMSET /* see ARRAY */ -{ /* elemental set of n-tuples; formally it is a "value" assigned - to members of model sets (like numbers and symbols, which are - values assigned to members of model parameters); note that a - simple model set is not an elemental set, it is 0-dimensional - array, the only member of which (if it exists) is assigned an - elemental set */ -#endif - -#define create_elemset _glp_mpl_create_elemset -ELEMSET *create_elemset(MPL *mpl, int dim); -/* create elemental set */ - -#define find_tuple _glp_mpl_find_tuple -MEMBER *find_tuple -( MPL *mpl, - ELEMSET *set, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* check if elemental set contains given n-tuple */ - -#define add_tuple _glp_mpl_add_tuple -MEMBER *add_tuple -( MPL *mpl, - ELEMSET *set, /* modified */ - TUPLE *tuple /* destroyed */ -); -/* add new n-tuple to elemental set */ - -#define check_then_add _glp_mpl_check_then_add -MEMBER *check_then_add -( MPL *mpl, - ELEMSET *set, /* modified */ - TUPLE *tuple /* destroyed */ -); -/* check and add new n-tuple to elemental set */ - -#define copy_elemset _glp_mpl_copy_elemset -ELEMSET *copy_elemset -( MPL *mpl, - ELEMSET *set /* not changed */ -); -/* make copy of elemental set */ - -#define delete_elemset _glp_mpl_delete_elemset -void delete_elemset -( MPL *mpl, - ELEMSET *set /* destroyed */ -); -/* delete elemental set */ - -#define arelset_size _glp_mpl_arelset_size -int arelset_size(MPL *mpl, double t0, double tf, double dt); -/* compute size of "arithmetic" elemental set */ - -#define arelset_member _glp_mpl_arelset_member -double arelset_member(MPL *mpl, double t0, double tf, double dt, int j); -/* compute member of "arithmetic" elemental set */ - -#define create_arelset _glp_mpl_create_arelset -ELEMSET *create_arelset(MPL *mpl, double t0, double tf, double dt); -/* create "arithmetic" elemental set */ - -#define set_union _glp_mpl_set_union -ELEMSET *set_union -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -); -/* union of two elemental sets */ - -#define set_diff _glp_mpl_set_diff -ELEMSET *set_diff -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -); -/* difference between two elemental sets */ - -#define set_symdiff _glp_mpl_set_symdiff -ELEMSET *set_symdiff -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -); -/* symmetric difference between two elemental sets */ - -#define set_inter _glp_mpl_set_inter -ELEMSET *set_inter -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -); -/* intersection of two elemental sets */ - -#define set_cross _glp_mpl_set_cross -ELEMSET *set_cross -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -); -/* cross (Cartesian) product of two elemental sets */ - -/**********************************************************************/ -/* * * ELEMENTAL VARIABLES * * */ -/**********************************************************************/ - -struct ELEMVAR -{ /* elemental variable; formally it is a "value" assigned to - members of model variables (like numbers and symbols, which - are values assigned to members of model parameters) */ - int j; - /* LP column number assigned to this elemental variable */ - VARIABLE *var; - /* model variable, which contains this elemental variable */ - MEMBER *memb; - /* array member, which is assigned this elemental variable */ - double lbnd; - /* lower bound */ - double ubnd; - /* upper bound */ - double temp; - /* working quantity used in operations on linear forms; normally - it contains floating-point zero */ -#if 1 /* 15/V-2010 */ - int stat; - double prim, dual; - /* solution components provided by the solver */ -#endif -}; - -/**********************************************************************/ -/* * * LINEAR FORMS * * */ -/**********************************************************************/ - -struct FORMULA -{ /* term of linear form c * x, where c is a coefficient, x is an - elemental variable; the linear form itself is the sum of terms - and is associated with its first term; (note that the linear - form may be empty that means the sum is equal to zero) */ - double coef; - /* coefficient at elemental variable or constant term */ - ELEMVAR *var; - /* reference to elemental variable; NULL means constant term */ - FORMULA *next; - /* the next term of linear form */ -}; - -#define constant_term _glp_mpl_constant_term -FORMULA *constant_term(MPL *mpl, double coef); -/* create constant term */ - -#define single_variable _glp_mpl_single_variable -FORMULA *single_variable -( MPL *mpl, - ELEMVAR *var /* referenced */ -); -/* create single variable */ - -#define copy_formula _glp_mpl_copy_formula -FORMULA *copy_formula -( MPL *mpl, - FORMULA *form /* not changed */ -); -/* make copy of linear form */ - -#define delete_formula _glp_mpl_delete_formula -void delete_formula -( MPL *mpl, - FORMULA *form /* destroyed */ -); -/* delete linear form */ - -#define linear_comb _glp_mpl_linear_comb -FORMULA *linear_comb -( MPL *mpl, - double a, FORMULA *fx, /* destroyed */ - double b, FORMULA *fy /* destroyed */ -); -/* linear combination of two linear forms */ - -#define remove_constant _glp_mpl_remove_constant -FORMULA *remove_constant -( MPL *mpl, - FORMULA *form, /* destroyed */ - double *coef /* modified */ -); -/* remove constant term from linear form */ - -#define reduce_terms _glp_mpl_reduce_terms -FORMULA *reduce_terms -( MPL *mpl, - FORMULA *form /* destroyed */ -); -/* reduce identical terms in linear form */ - -/**********************************************************************/ -/* * * ELEMENTAL CONSTRAINTS * * */ -/**********************************************************************/ - -struct ELEMCON -{ /* elemental constraint; formally it is a "value" assigned to - members of model constraints (like numbers or symbols, which - are values assigned to members of model parameters) */ - int i; - /* LP row number assigned to this elemental constraint */ - CONSTRAINT *con; - /* model constraint, which contains this elemental constraint */ - MEMBER *memb; - /* array member, which is assigned this elemental constraint */ - FORMULA *form; - /* linear form */ - double lbnd; - /* lower bound */ - double ubnd; - /* upper bound */ -#if 1 /* 15/V-2010 */ - int stat; - double prim, dual; - /* solution components provided by the solver */ -#endif -}; - -/**********************************************************************/ -/* * * GENERIC VALUES * * */ -/**********************************************************************/ - -union VALUE -{ /* generic value, which can be assigned to object member or be a - result of evaluation of expression */ - /* indicator that specifies the particular type of generic value - is stored in the corresponding array or pseudo-code descriptor - and can be one of the following: - A_NONE - no value - A_NUMERIC - floating-point number - A_SYMBOLIC - symbol - A_LOGICAL - logical value - A_TUPLE - n-tuple - A_ELEMSET - elemental set - A_ELEMVAR - elemental variable - A_FORMULA - linear form - A_ELEMCON - elemental constraint */ - void *none; /* null */ - double num; /* value */ - SYMBOL *sym; /* value */ - int bit; /* value */ - TUPLE *tuple; /* value */ - ELEMSET *set; /* value */ - ELEMVAR *var; /* reference */ - FORMULA *form; /* value */ - ELEMCON *con; /* reference */ -}; - -#define delete_value _glp_mpl_delete_value -void delete_value -( MPL *mpl, - int type, - VALUE *value /* content destroyed */ -); -/* delete generic value */ - -/**********************************************************************/ -/* * * SYMBOLICALLY INDEXED ARRAYS * * */ -/**********************************************************************/ - -struct ARRAY -{ /* multi-dimensional array, a set of members indexed over simple - or compound sets of symbols; arrays are used to represent the - contents of model objects (i.e. sets, parameters, variables, - constraints, and objectives); arrays also are used as "values" - that are assigned to members of set objects, in which case the - array itself represents an elemental set */ - int type; - /* type of generic values assigned to the array members: - A_NONE - none (members have no assigned values) - A_NUMERIC - floating-point numbers - A_SYMBOLIC - symbols - A_ELEMSET - elemental sets - A_ELEMVAR - elemental variables - A_ELEMCON - elemental constraints */ - int dim; - /* dimension of the array that determines number of components in - n-tuples for all members of the array, dim >= 0; dim = 0 means - the array is 0-dimensional */ - int size; - /* size of the array, i.e. number of its members */ - MEMBER *head; - /* the first array member; NULL means the array is empty */ - MEMBER *tail; - /* the last array member; NULL means the array is empty */ - AVL *tree; - /* the search tree intended to find array members for logarithmic - time; NULL means the search tree doesn't exist */ - ARRAY *prev; - /* the previous array in the translator database */ - ARRAY *next; - /* the next array in the translator database */ -}; - -struct MEMBER -{ /* array member */ - TUPLE *tuple; - /* n-tuple, which identifies the member; number of its components - is the same for all members within the array and determined by - the array dimension; duplicate members are not allowed */ - MEMBER *next; - /* the next array member */ - VALUE value; - /* generic value assigned to the member */ -}; - -#define create_array _glp_mpl_create_array -ARRAY *create_array(MPL *mpl, int type, int dim); -/* create array */ - -#define find_member _glp_mpl_find_member -MEMBER *find_member -( MPL *mpl, - ARRAY *array, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* find array member with given n-tuple */ - -#define add_member _glp_mpl_add_member -MEMBER *add_member -( MPL *mpl, - ARRAY *array, /* modified */ - TUPLE *tuple /* destroyed */ -); -/* add new member to array */ - -#define delete_array _glp_mpl_delete_array -void delete_array -( MPL *mpl, - ARRAY *array /* destroyed */ -); -/* delete array */ - -/**********************************************************************/ -/* * * DOMAINS AND DUMMY INDICES * * */ -/**********************************************************************/ - -struct DOMAIN -{ /* domain (a simple or compound set); syntactically domain looks - like '{ i in I, (j,k) in S, t in T : }'; domains - are used to define sets, over which model objects are indexed, - and also as constituents of iterated operators */ - DOMAIN_BLOCK *list; - /* linked list of domain blocks (in the example above such blocks - are 'i in I', '(j,k) in S', and 't in T'); this list cannot be - empty */ - CODE *code; - /* pseudo-code for computing the logical predicate, which follows - the colon; NULL means no predicate is specified */ -}; - -struct DOMAIN_BLOCK -{ /* domain block; syntactically domain blocks look like 'i in I', - '(j,k) in S', and 't in T' in the example above (in the sequel - sets like I, S, and T are called basic sets) */ - DOMAIN_SLOT *list; - /* linked list of domain slots (i.e. indexing positions); number - of slots in this list is the same as dimension of n-tuples in - the basic set; this list cannot be empty */ - CODE *code; - /* pseudo-code for computing basic set; cannot be NULL */ - TUPLE *backup; - /* if this n-tuple is not empty, current values of dummy indices - in the domain block are the same as components of this n-tuple - (note that this n-tuple may have larger dimension than number - of dummy indices in this block, in which case extra components - are ignored); this n-tuple is used to restore former values of - dummy indices, if they were changed due to recursive calls to - the domain block */ - DOMAIN_BLOCK *next; - /* the next block in the same domain */ -}; - -struct DOMAIN_SLOT -{ /* domain slot; it specifies an individual indexing position and - defines the corresponding dummy index */ - char *name; - /* symbolic name of the dummy index; null pointer means the dummy - index is not explicitly specified */ - CODE *code; - /* pseudo-code for computing symbolic value, at which the dummy - index is bound; NULL means the dummy index is free within the - domain scope */ - SYMBOL *value; - /* current value assigned to the dummy index; NULL means no value - is assigned at the moment */ - CODE *list; - /* linked list of pseudo-codes with operation O_INDEX referring - to this slot; this linked list is used to invalidate resultant - values of the operation, which depend on this dummy index */ - DOMAIN_SLOT *next; - /* the next slot in the same domain block */ -}; - -#define assign_dummy_index _glp_mpl_assign_dummy_index -void assign_dummy_index -( MPL *mpl, - DOMAIN_SLOT *slot, /* modified */ - SYMBOL *value /* not changed */ -); -/* assign new value to dummy index */ - -#define update_dummy_indices _glp_mpl_update_dummy_indices -void update_dummy_indices -( MPL *mpl, - DOMAIN_BLOCK *block /* not changed */ -); -/* update current values of dummy indices */ - -#define enter_domain_block _glp_mpl_enter_domain_block -int enter_domain_block -( MPL *mpl, - DOMAIN_BLOCK *block, /* not changed */ - TUPLE *tuple, /* not changed */ - void *info, void (*func)(MPL *mpl, void *info) -); -/* enter domain block */ - -#define eval_within_domain _glp_mpl_eval_within_domain -int eval_within_domain -( MPL *mpl, - DOMAIN *domain, /* not changed */ - TUPLE *tuple, /* not changed */ - void *info, void (*func)(MPL *mpl, void *info) -); -/* perform evaluation within domain scope */ - -#define loop_within_domain _glp_mpl_loop_within_domain -void loop_within_domain -( MPL *mpl, - DOMAIN *domain, /* not changed */ - void *info, int (*func)(MPL *mpl, void *info) -); -/* perform iterations within domain scope */ - -#define out_of_domain _glp_mpl_out_of_domain -void out_of_domain -( MPL *mpl, - char *name, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* raise domain exception */ - -#define get_domain_tuple _glp_mpl_get_domain_tuple -TUPLE *get_domain_tuple -( MPL *mpl, - DOMAIN *domain /* not changed */ -); -/* obtain current n-tuple from domain */ - -#define clean_domain _glp_mpl_clean_domain -void clean_domain(MPL *mpl, DOMAIN *domain); -/* clean domain */ - -/**********************************************************************/ -/* * * MODEL SETS * * */ -/**********************************************************************/ - -struct SET -{ /* model set */ - char *name; - /* symbolic name; cannot be NULL */ - char *alias; - /* alias; NULL means alias is not specified */ - int dim; /* aka arity */ - /* dimension (number of subscripts); dim = 0 means 0-dimensional - (unsubscripted) set, dim > 0 means set of sets */ - DOMAIN *domain; - /* subscript domain; NULL for 0-dimensional set */ - int dimen; - /* dimension of n-tuples, which members of this set consist of - (note that the model set itself is an array of elemental sets, - which are its members; so, don't confuse this dimension with - dimension of the model set); always non-zero */ - WITHIN *within; - /* list of supersets, which restrict each member of the set to be - in every superset from this list; this list can be empty */ - CODE *assign; - /* pseudo-code for computing assigned value; can be NULL */ - CODE *option; - /* pseudo-code for computing default value; can be NULL */ - GADGET *gadget; - /* plain set used to initialize the array of sets; can be NULL */ - int data; - /* data status flag: - 0 - no data are provided in the data section - 1 - data are provided, but not checked yet - 2 - data are provided and have been checked */ - ARRAY *array; - /* array of members, which are assigned elemental sets */ -}; - -struct WITHIN -{ /* restricting superset list entry */ - CODE *code; - /* pseudo-code for computing the superset; cannot be NULL */ - WITHIN *next; - /* the next entry for the same set or parameter */ -}; - -struct GADGET -{ /* plain set used to initialize the array of sets with data */ - SET *set; - /* pointer to plain set; cannot be NULL */ - int ind[20]; /* ind[dim+dimen]; */ - /* permutation of integers 1, 2, ..., dim+dimen */ -}; - -#define check_elem_set _glp_mpl_check_elem_set -void check_elem_set -( MPL *mpl, - SET *set, /* not changed */ - TUPLE *tuple, /* not changed */ - ELEMSET *refer /* not changed */ -); -/* check elemental set assigned to set member */ - -#define take_member_set _glp_mpl_take_member_set -ELEMSET *take_member_set /* returns reference, not value */ -( MPL *mpl, - SET *set, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* obtain elemental set assigned to set member */ - -#define eval_member_set _glp_mpl_eval_member_set -ELEMSET *eval_member_set /* returns reference, not value */ -( MPL *mpl, - SET *set, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* evaluate elemental set assigned to set member */ - -#define eval_whole_set _glp_mpl_eval_whole_set -void eval_whole_set(MPL *mpl, SET *set); -/* evaluate model set over entire domain */ - -#define clean_set _glp_mpl_clean_set -void clean_set(MPL *mpl, SET *set); -/* clean model set */ - -/**********************************************************************/ -/* * * MODEL PARAMETERS * * */ -/**********************************************************************/ - -struct PARAMETER -{ /* model parameter */ - char *name; - /* symbolic name; cannot be NULL */ - char *alias; - /* alias; NULL means alias is not specified */ - int dim; /* aka arity */ - /* dimension (number of subscripts); dim = 0 means 0-dimensional - (unsubscripted) parameter */ - DOMAIN *domain; - /* subscript domain; NULL for 0-dimensional parameter */ - int type; - /* parameter type: - A_NUMERIC - numeric - A_INTEGER - integer - A_BINARY - binary - A_SYMBOLIC - symbolic */ - CONDITION *cond; - /* list of conditions, which restrict each parameter member to - satisfy to every condition from this list; this list is used - only for numeric parameters and can be empty */ - WITHIN *in; - /* list of supersets, which restrict each parameter member to be - in every superset from this list; this list is used only for - symbolic parameters and can be empty */ - CODE *assign; - /* pseudo-code for computing assigned value; can be NULL */ - CODE *option; - /* pseudo-code for computing default value; can be NULL */ - int data; - /* data status flag: - 0 - no data are provided in the data section - 1 - data are provided, but not checked yet - 2 - data are provided and have been checked */ - SYMBOL *defval; - /* default value provided in the data section; can be NULL */ - ARRAY *array; - /* array of members, which are assigned numbers or symbols */ -}; - -struct CONDITION -{ /* restricting condition list entry */ - int rho; - /* flag that specifies the form of the condition: - O_LT - less than - O_LE - less than or equal to - O_EQ - equal to - O_GE - greater than or equal to - O_GT - greater than - O_NE - not equal to */ - CODE *code; - /* pseudo-code for computing the reference value */ - CONDITION *next; - /* the next entry for the same parameter */ -}; - -#define check_value_num _glp_mpl_check_value_num -void check_value_num -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple, /* not changed */ - double value -); -/* check numeric value assigned to parameter member */ - -#define take_member_num _glp_mpl_take_member_num -double take_member_num -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* obtain numeric value assigned to parameter member */ - -#define eval_member_num _glp_mpl_eval_member_num -double eval_member_num -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* evaluate numeric value assigned to parameter member */ - -#define check_value_sym _glp_mpl_check_value_sym -void check_value_sym -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple, /* not changed */ - SYMBOL *value /* not changed */ -); -/* check symbolic value assigned to parameter member */ - -#define take_member_sym _glp_mpl_take_member_sym -SYMBOL *take_member_sym /* returns value, not reference */ -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* obtain symbolic value assigned to parameter member */ - -#define eval_member_sym _glp_mpl_eval_member_sym -SYMBOL *eval_member_sym /* returns value, not reference */ -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* evaluate symbolic value assigned to parameter member */ - -#define eval_whole_par _glp_mpl_eval_whole_par -void eval_whole_par(MPL *mpl, PARAMETER *par); -/* evaluate model parameter over entire domain */ - -#define clean_parameter _glp_mpl_clean_parameter -void clean_parameter(MPL *mpl, PARAMETER *par); -/* clean model parameter */ - -/**********************************************************************/ -/* * * MODEL VARIABLES * * */ -/**********************************************************************/ - -struct VARIABLE -{ /* model variable */ - char *name; - /* symbolic name; cannot be NULL */ - char *alias; - /* alias; NULL means alias is not specified */ - int dim; /* aka arity */ - /* dimension (number of subscripts); dim = 0 means 0-dimensional - (unsubscripted) variable */ - DOMAIN *domain; - /* subscript domain; NULL for 0-dimensional variable */ - int type; - /* variable type: - A_NUMERIC - continuous - A_INTEGER - integer - A_BINARY - binary */ - CODE *lbnd; - /* pseudo-code for computing lower bound; NULL means lower bound - is not specified */ - CODE *ubnd; - /* pseudo-code for computing upper bound; NULL means upper bound - is not specified */ - /* if both the pointers lbnd and ubnd refer to the same code, the - variable is fixed at the corresponding value */ - ARRAY *array; - /* array of members, which are assigned elemental variables */ -}; - -#define take_member_var _glp_mpl_take_member_var -ELEMVAR *take_member_var /* returns reference */ -( MPL *mpl, - VARIABLE *var, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* obtain reference to elemental variable */ - -#define eval_member_var _glp_mpl_eval_member_var -ELEMVAR *eval_member_var /* returns reference */ -( MPL *mpl, - VARIABLE *var, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* evaluate reference to elemental variable */ - -#define eval_whole_var _glp_mpl_eval_whole_var -void eval_whole_var(MPL *mpl, VARIABLE *var); -/* evaluate model variable over entire domain */ - -#define clean_variable _glp_mpl_clean_variable -void clean_variable(MPL *mpl, VARIABLE *var); -/* clean model variable */ - -/**********************************************************************/ -/* * * MODEL CONSTRAINTS AND OBJECTIVES * * */ -/**********************************************************************/ - -struct CONSTRAINT -{ /* model constraint or objective */ - char *name; - /* symbolic name; cannot be NULL */ - char *alias; - /* alias; NULL means alias is not specified */ - int dim; /* aka arity */ - /* dimension (number of subscripts); dim = 0 means 0-dimensional - (unsubscripted) constraint */ - DOMAIN *domain; - /* subscript domain; NULL for 0-dimensional constraint */ - int type; - /* constraint type: - A_CONSTRAINT - constraint - A_MINIMIZE - objective (minimization) - A_MAXIMIZE - objective (maximization) */ - CODE *code; - /* pseudo-code for computing main linear form; cannot be NULL */ - CODE *lbnd; - /* pseudo-code for computing lower bound; NULL means lower bound - is not specified */ - CODE *ubnd; - /* pseudo-code for computing upper bound; NULL means upper bound - is not specified */ - /* if both the pointers lbnd and ubnd refer to the same code, the - constraint has the form of equation */ - ARRAY *array; - /* array of members, which are assigned elemental constraints */ -}; - -#define take_member_con _glp_mpl_take_member_con -ELEMCON *take_member_con /* returns reference */ -( MPL *mpl, - CONSTRAINT *con, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* obtain reference to elemental constraint */ - -#define eval_member_con _glp_mpl_eval_member_con -ELEMCON *eval_member_con /* returns reference */ -( MPL *mpl, - CONSTRAINT *con, /* not changed */ - TUPLE *tuple /* not changed */ -); -/* evaluate reference to elemental constraint */ - -#define eval_whole_con _glp_mpl_eval_whole_con -void eval_whole_con(MPL *mpl, CONSTRAINT *con); -/* evaluate model constraint over entire domain */ - -#define clean_constraint _glp_mpl_clean_constraint -void clean_constraint(MPL *mpl, CONSTRAINT *con); -/* clean model constraint */ - -/**********************************************************************/ -/* * * DATA TABLES * * */ -/**********************************************************************/ - -struct TABLE -{ /* data table */ - char *name; - /* symbolic name; cannot be NULL */ - char *alias; - /* alias; NULL means alias is not specified */ - int type; - /* table type: - A_INPUT - input table - A_OUTPUT - output table */ - TABARG *arg; - /* argument list; cannot be empty */ - union - { struct - { SET *set; - /* input set; NULL means the set is not specified */ - TABFLD *fld; - /* field list; cannot be empty */ - TABIN *list; - /* input list; can be empty */ - } in; - struct - { DOMAIN *domain; - /* subscript domain; cannot be NULL */ - TABOUT *list; - /* output list; cannot be empty */ - } out; - } u; -}; - -struct TABARG -{ /* table argument list entry */ - CODE *code; - /* pseudo-code for computing the argument */ - TABARG *next; - /* next entry for the same table */ -}; - -struct TABFLD -{ /* table field list entry */ - char *name; - /* field name; cannot be NULL */ - TABFLD *next; - /* next entry for the same table */ -}; - -struct TABIN -{ /* table input list entry */ - PARAMETER *par; - /* parameter to be read; cannot be NULL */ - char *name; - /* column name; cannot be NULL */ - TABIN *next; - /* next entry for the same table */ -}; - -struct TABOUT -{ /* table output list entry */ - CODE *code; - /* pseudo-code for computing the value to be written */ - char *name; - /* column name; cannot be NULL */ - TABOUT *next; - /* next entry for the same table */ -}; - -struct TABDCA -{ /* table driver communication area */ - int id; - /* driver identifier (set by mpl_tab_drv_open) */ - void *link; - /* driver link pointer (set by mpl_tab_drv_open) */ - int na; - /* number of arguments */ - char **arg; /* char *arg[1+ns]; */ - /* arg[k], 1 <= k <= ns, is pointer to k-th argument */ - int nf; - /* number of fields */ - char **name; /* char *name[1+nc]; */ - /* name[k], 1 <= k <= nc, is name of k-th field */ - int *type; /* int type[1+nc]; */ - /* type[k], 1 <= k <= nc, is type of k-th field: - '?' - value not assigned - 'N' - number - 'S' - character string */ - double *num; /* double num[1+nc]; */ - /* num[k], 1 <= k <= nc, is numeric value of k-th field */ - char **str; - /* str[k], 1 <= k <= nc, is string value of k-th field */ -}; - -#define mpl_tab_num_args _glp_mpl_tab_num_args -int mpl_tab_num_args(TABDCA *dca); - -#define mpl_tab_get_arg _glp_mpl_tab_get_arg -const char *mpl_tab_get_arg(TABDCA *dca, int k); - -#define mpl_tab_num_flds _glp_mpl_tab_num_flds -int mpl_tab_num_flds(TABDCA *dca); - -#define mpl_tab_get_name _glp_mpl_tab_get_name -const char *mpl_tab_get_name(TABDCA *dca, int k); - -#define mpl_tab_get_type _glp_mpl_tab_get_type -int mpl_tab_get_type(TABDCA *dca, int k); - -#define mpl_tab_get_num _glp_mpl_tab_get_num -double mpl_tab_get_num(TABDCA *dca, int k); - -#define mpl_tab_get_str _glp_mpl_tab_get_str -const char *mpl_tab_get_str(TABDCA *dca, int k); - -#define mpl_tab_set_num _glp_mpl_tab_set_num -void mpl_tab_set_num(TABDCA *dca, int k, double num); - -#define mpl_tab_set_str _glp_mpl_tab_set_str -void mpl_tab_set_str(TABDCA *dca, int k, const char *str); - -#define mpl_tab_drv_open _glp_mpl_tab_drv_open -void mpl_tab_drv_open(MPL *mpl, int mode); - -#define mpl_tab_drv_read _glp_mpl_tab_drv_read -int mpl_tab_drv_read(MPL *mpl); - -#define mpl_tab_drv_write _glp_mpl_tab_drv_write -void mpl_tab_drv_write(MPL *mpl); - -#define mpl_tab_drv_close _glp_mpl_tab_drv_close -void mpl_tab_drv_close(MPL *mpl); - -/**********************************************************************/ -/* * * PSEUDO-CODE * * */ -/**********************************************************************/ - -union OPERANDS -{ /* operands that participate in pseudo-code operation (choice of - particular operands depends on the operation code) */ - /*--------------------------------------------------------------*/ - double num; /* O_NUMBER */ - /* floaing-point number to be taken */ - /*--------------------------------------------------------------*/ - char *str; /* O_STRING */ - /* character string to be taken */ - /*--------------------------------------------------------------*/ - struct /* O_INDEX */ - { DOMAIN_SLOT *slot; - /* domain slot, which contains dummy index to be taken */ - CODE *next; - /* the next pseudo-code with op = O_INDEX, which refers to the - same slot as this one; pointer to the beginning of this list - is stored in the corresponding domain slot */ - } index; - /*--------------------------------------------------------------*/ - struct /* O_MEMNUM, O_MEMSYM */ - { PARAMETER *par; - /* model parameter, which contains member to be taken */ - ARG_LIST *list; - /* list of subscripts; NULL for 0-dimensional parameter */ - } par; - /*--------------------------------------------------------------*/ - struct /* O_MEMSET */ - { SET *set; - /* model set, which contains member to be taken */ - ARG_LIST *list; - /* list of subscripts; NULL for 0-dimensional set */ - } set; - /*--------------------------------------------------------------*/ - struct /* O_MEMVAR */ - { VARIABLE *var; - /* model variable, which contains member to be taken */ - ARG_LIST *list; - /* list of subscripts; NULL for 0-dimensional variable */ -#if 1 /* 15/V-2010 */ - int suff; - /* suffix specified: */ -#define DOT_NONE 0x00 /* none (means variable itself) */ -#define DOT_LB 0x01 /* .lb (lower bound) */ -#define DOT_UB 0x02 /* .ub (upper bound) */ -#define DOT_STATUS 0x03 /* .status (status) */ -#define DOT_VAL 0x04 /* .val (primal value) */ -#define DOT_DUAL 0x05 /* .dual (dual value) */ -#endif - } var; -#if 1 /* 15/V-2010 */ - /*--------------------------------------------------------------*/ - struct /* O_MEMCON */ - { CONSTRAINT *con; - /* model constraint, which contains member to be taken */ - ARG_LIST *list; - /* list of subscripys; NULL for 0-dimensional constraint */ - int suff; - /* suffix specified (see O_MEMVAR above) */ - } con; -#endif - /*--------------------------------------------------------------*/ - ARG_LIST *list; /* O_TUPLE, O_MAKE, n-ary operations */ - /* list of operands */ - /*--------------------------------------------------------------*/ - DOMAIN_BLOCK *slice; /* O_SLICE */ - /* domain block, which specifies slice (i.e. n-tuple that contains - free dummy indices); this operation is never evaluated */ - /*--------------------------------------------------------------*/ - struct /* unary, binary, ternary operations */ - { CODE *x; - /* pseudo-code for computing first operand */ - CODE *y; - /* pseudo-code for computing second operand */ - CODE *z; - /* pseudo-code for computing third operand */ - } arg; - /*--------------------------------------------------------------*/ - struct /* iterated operations */ - { DOMAIN *domain; - /* domain, over which the operation is performed */ - CODE *x; - /* pseudo-code for computing "integrand" */ - } loop; - /*--------------------------------------------------------------*/ -}; - -struct ARG_LIST -{ /* operands list entry */ - CODE *x; - /* pseudo-code for computing operand */ - ARG_LIST *next; - /* the next operand of the same operation */ -}; - -struct CODE -{ /* pseudo-code (internal form of expressions) */ - int op; - /* operation code: */ -#define O_NUMBER 301 /* take floating-point number */ -#define O_STRING 302 /* take character string */ -#define O_INDEX 303 /* take dummy index */ -#define O_MEMNUM 304 /* take member of numeric parameter */ -#define O_MEMSYM 305 /* take member of symbolic parameter */ -#define O_MEMSET 306 /* take member of set */ -#define O_MEMVAR 307 /* take member of variable */ -#define O_MEMCON 308 /* take member of constraint */ -#define O_TUPLE 309 /* make n-tuple */ -#define O_MAKE 310 /* make elemental set of n-tuples */ -#define O_SLICE 311 /* define domain block (dummy op) */ - /* 0-ary operations --------------------*/ -#define O_IRAND224 312 /* pseudo-random in [0, 2^24-1] */ -#define O_UNIFORM01 313 /* pseudo-random in [0, 1) */ -#define O_NORMAL01 314 /* gaussian random, mu = 0, sigma = 1 */ -#define O_GMTIME 315 /* current calendar time (UTC) */ - /* unary operations --------------------*/ -#define O_CVTNUM 316 /* conversion to numeric */ -#define O_CVTSYM 317 /* conversion to symbolic */ -#define O_CVTLOG 318 /* conversion to logical */ -#define O_CVTTUP 319 /* conversion to 1-tuple */ -#define O_CVTLFM 320 /* conversion to linear form */ -#define O_PLUS 321 /* unary plus */ -#define O_MINUS 322 /* unary minus */ -#define O_NOT 323 /* negation (logical "not") */ -#define O_ABS 324 /* absolute value */ -#define O_CEIL 325 /* round upward ("ceiling of x") */ -#define O_FLOOR 326 /* round downward ("floor of x") */ -#define O_EXP 327 /* base-e exponential */ -#define O_LOG 328 /* natural logarithm */ -#define O_LOG10 329 /* common (decimal) logarithm */ -#define O_SQRT 330 /* square root */ -#define O_SIN 331 /* trigonometric sine */ -#define O_COS 332 /* trigonometric cosine */ -#define O_TAN 333 /* trigonometric tangent */ -#define O_ATAN 334 /* trigonometric arctangent */ -#define O_ROUND 335 /* round to nearest integer */ -#define O_TRUNC 336 /* truncate to nearest integer */ -#define O_CARD 337 /* cardinality of set */ -#define O_LENGTH 338 /* length of symbolic value */ - /* binary operations -------------------*/ -#define O_ADD 339 /* addition */ -#define O_SUB 340 /* subtraction */ -#define O_LESS 341 /* non-negative subtraction */ -#define O_MUL 342 /* multiplication */ -#define O_DIV 343 /* division */ -#define O_IDIV 344 /* quotient of exact division */ -#define O_MOD 345 /* remainder of exact division */ -#define O_POWER 346 /* exponentiation (raise to power) */ -#define O_ATAN2 347 /* trigonometric arctangent */ -#define O_ROUND2 348 /* round to n fractional digits */ -#define O_TRUNC2 349 /* truncate to n fractional digits */ -#define O_UNIFORM 350 /* pseudo-random in [a, b) */ -#define O_NORMAL 351 /* gaussian random, given mu and sigma */ -#define O_CONCAT 352 /* concatenation */ -#define O_LT 353 /* comparison on 'less than' */ -#define O_LE 354 /* comparison on 'not greater than' */ -#define O_EQ 355 /* comparison on 'equal to' */ -#define O_GE 356 /* comparison on 'not less than' */ -#define O_GT 357 /* comparison on 'greater than' */ -#define O_NE 358 /* comparison on 'not equal to' */ -#define O_AND 359 /* conjunction (logical "and") */ -#define O_OR 360 /* disjunction (logical "or") */ -#define O_UNION 361 /* union */ -#define O_DIFF 362 /* difference */ -#define O_SYMDIFF 363 /* symmetric difference */ -#define O_INTER 364 /* intersection */ -#define O_CROSS 365 /* cross (Cartesian) product */ -#define O_IN 366 /* test on 'x in Y' */ -#define O_NOTIN 367 /* test on 'x not in Y' */ -#define O_WITHIN 368 /* test on 'X within Y' */ -#define O_NOTWITHIN 369 /* test on 'X not within Y' */ -#define O_SUBSTR 370 /* substring */ -#define O_STR2TIME 371 /* convert string to time */ -#define O_TIME2STR 372 /* convert time to string */ - /* ternary operations ------------------*/ -#define O_DOTS 373 /* build "arithmetic" set */ -#define O_FORK 374 /* if-then-else */ -#define O_SUBSTR3 375 /* substring */ - /* n-ary operations --------------------*/ -#define O_MIN 376 /* minimal value (n-ary) */ -#define O_MAX 377 /* maximal value (n-ary) */ - /* iterated operations -----------------*/ -#define O_SUM 378 /* summation */ -#define O_PROD 379 /* multiplication */ -#define O_MINIMUM 380 /* minimum */ -#define O_MAXIMUM 381 /* maximum */ -#define O_FORALL 382 /* conjunction (A-quantification) */ -#define O_EXISTS 383 /* disjunction (E-quantification) */ -#define O_SETOF 384 /* compute elemental set */ -#define O_BUILD 385 /* build elemental set */ - OPERANDS arg; - /* operands that participate in the operation */ - int type; - /* type of the resultant value: - A_NUMERIC - numeric - A_SYMBOLIC - symbolic - A_LOGICAL - logical - A_TUPLE - n-tuple - A_ELEMSET - elemental set - A_FORMULA - linear form */ - int dim; - /* dimension of the resultant value; for A_TUPLE and A_ELEMSET it - is the dimension of the corresponding n-tuple(s) and cannot be - zero; for other resultant types it is always zero */ - CODE *up; - /* parent pseudo-code, which refers to this pseudo-code as to its - operand; NULL means this pseudo-code has no parent and defines - an expression, which is not contained in another expression */ - int vflag; - /* volatile flag; being set this flag means that this operation - has a side effect; for primary expressions this flag is set - directly by corresponding parsing routines (for example, if - primary expression is a reference to a function that generates - pseudo-random numbers); in other cases this flag is inherited - from operands */ - int valid; - /* if this flag is set, the resultant value, which is a temporary - result of evaluating this operation on particular values of - operands, is valid; if this flag is clear, the resultant value - doesn't exist and therefore not valid; having been evaluated - the resultant value is stored here and not destroyed until the - dummy indices, which this value depends on, have been changed - (and if it doesn't depend on dummy indices at all, it is never - destroyed); thus, if the resultant value is valid, evaluating - routine can immediately take its copy not computing the result - from scratch; this mechanism is similar to moving invariants - out of loops and allows improving efficiency at the expense of - some extra memory needed to keep temporary results */ - /* however, if the volatile flag (see above) is set, even if the - resultant value is valid, evaluating routine computes it as if - it were not valid, i.e. caching is not used in this case */ - VALUE value; - /* resultant value in generic format */ -}; - -#define eval_numeric _glp_mpl_eval_numeric -double eval_numeric(MPL *mpl, CODE *code); -/* evaluate pseudo-code to determine numeric value */ - -#define eval_symbolic _glp_mpl_eval_symbolic -SYMBOL *eval_symbolic(MPL *mpl, CODE *code); -/* evaluate pseudo-code to determine symbolic value */ - -#define eval_logical _glp_mpl_eval_logical -int eval_logical(MPL *mpl, CODE *code); -/* evaluate pseudo-code to determine logical value */ - -#define eval_tuple _glp_mpl_eval_tuple -TUPLE *eval_tuple(MPL *mpl, CODE *code); -/* evaluate pseudo-code to construct n-tuple */ - -#define eval_elemset _glp_mpl_eval_elemset -ELEMSET *eval_elemset(MPL *mpl, CODE *code); -/* evaluate pseudo-code to construct elemental set */ - -#define is_member _glp_mpl_is_member -int is_member(MPL *mpl, CODE *code, TUPLE *tuple); -/* check if n-tuple is in set specified by pseudo-code */ - -#define eval_formula _glp_mpl_eval_formula -FORMULA *eval_formula(MPL *mpl, CODE *code); -/* evaluate pseudo-code to construct linear form */ - -#define clean_code _glp_mpl_clean_code -void clean_code(MPL *mpl, CODE *code); -/* clean pseudo-code */ - -/**********************************************************************/ -/* * * MODEL STATEMENTS * * */ -/**********************************************************************/ - -struct CHECK -{ /* check statement */ - DOMAIN *domain; - /* subscript domain; NULL means domain is not used */ - CODE *code; - /* code for computing the predicate to be checked */ -}; - -struct DISPLAY -{ /* display statement */ - DOMAIN *domain; - /* subscript domain; NULL means domain is not used */ - DISPLAY1 *list; - /* display list; cannot be empty */ -}; - -struct DISPLAY1 -{ /* display list entry */ - int type; - /* item type: - A_INDEX - dummy index - A_SET - model set - A_PARAMETER - model parameter - A_VARIABLE - model variable - A_CONSTRAINT - model constraint/objective - A_EXPRESSION - expression */ - union - { DOMAIN_SLOT *slot; - SET *set; - PARAMETER *par; - VARIABLE *var; - CONSTRAINT *con; - CODE *code; - } u; - /* item to be displayed */ -#if 0 /* 15/V-2010 */ - ARG_LIST *list; - /* optional subscript list (for constraint/objective only) */ -#endif - DISPLAY1 *next; - /* the next entry for the same statement */ -}; - -struct PRINTF -{ /* printf statement */ - DOMAIN *domain; - /* subscript domain; NULL means domain is not used */ - CODE *fmt; - /* pseudo-code for computing format string */ - PRINTF1 *list; - /* printf list; can be empty */ - CODE *fname; - /* pseudo-code for computing filename to redirect the output; - NULL means the output goes to stdout */ - int app; - /* if this flag is set, the output is appended */ -}; - -struct PRINTF1 -{ /* printf list entry */ - CODE *code; - /* pseudo-code for computing value to be printed */ - PRINTF1 *next; - /* the next entry for the same statement */ -}; - -struct FOR -{ /* for statement */ - DOMAIN *domain; - /* subscript domain; cannot be NULL */ - STATEMENT *list; - /* linked list of model statements within this for statement in - the original order */ -}; - -struct STATEMENT -{ /* model statement */ - int line; - /* number of source text line, where statement begins */ - int type; - /* statement type: - A_SET - set statement - A_PARAMETER - parameter statement - A_VARIABLE - variable statement - A_CONSTRAINT - constraint/objective statement - A_TABLE - table statement - A_SOLVE - solve statement - A_CHECK - check statement - A_DISPLAY - display statement - A_PRINTF - printf statement - A_FOR - for statement */ - union - { SET *set; - PARAMETER *par; - VARIABLE *var; - CONSTRAINT *con; - TABLE *tab; - void *slv; /* currently not used (set to NULL) */ - CHECK *chk; - DISPLAY *dpy; - PRINTF *prt; - FOR *fur; - } u; - /* specific part of statement */ - STATEMENT *next; - /* the next statement; in this list statements follow in the same - order as they appear in the model section */ -}; - -#define execute_table _glp_mpl_execute_table -void execute_table(MPL *mpl, TABLE *tab); -/* execute table statement */ - -#define free_dca _glp_mpl_free_dca -void free_dca(MPL *mpl); -/* free table driver communucation area */ - -#define clean_table _glp_mpl_clean_table -void clean_table(MPL *mpl, TABLE *tab); -/* clean table statement */ - -#define execute_check _glp_mpl_execute_check -void execute_check(MPL *mpl, CHECK *chk); -/* execute check statement */ - -#define clean_check _glp_mpl_clean_check -void clean_check(MPL *mpl, CHECK *chk); -/* clean check statement */ - -#define execute_display _glp_mpl_execute_display -void execute_display(MPL *mpl, DISPLAY *dpy); -/* execute display statement */ - -#define clean_display _glp_mpl_clean_display -void clean_display(MPL *mpl, DISPLAY *dpy); -/* clean display statement */ - -#define execute_printf _glp_mpl_execute_printf -void execute_printf(MPL *mpl, PRINTF *prt); -/* execute printf statement */ - -#define clean_printf _glp_mpl_clean_printf -void clean_printf(MPL *mpl, PRINTF *prt); -/* clean printf statement */ - -#define execute_for _glp_mpl_execute_for -void execute_for(MPL *mpl, FOR *fur); -/* execute for statement */ - -#define clean_for _glp_mpl_clean_for -void clean_for(MPL *mpl, FOR *fur); -/* clean for statement */ - -#define execute_statement _glp_mpl_execute_statement -void execute_statement(MPL *mpl, STATEMENT *stmt); -/* execute specified model statement */ - -#define clean_statement _glp_mpl_clean_statement -void clean_statement(MPL *mpl, STATEMENT *stmt); -/* clean specified model statement */ - -/**********************************************************************/ -/* * * GENERATING AND POSTSOLVING MODEL * * */ -/**********************************************************************/ - -#define alloc_content _glp_mpl_alloc_content -void alloc_content(MPL *mpl); -/* allocate content arrays for all model objects */ - -#define generate_model _glp_mpl_generate_model -void generate_model(MPL *mpl); -/* generate model */ - -#define build_problem _glp_mpl_build_problem -void build_problem(MPL *mpl); -/* build problem instance */ - -#define postsolve_model _glp_mpl_postsolve_model -void postsolve_model(MPL *mpl); -/* postsolve model */ - -#define clean_model _glp_mpl_clean_model -void clean_model(MPL *mpl); -/* clean model content */ - -/**********************************************************************/ -/* * * INPUT/OUTPUT * * */ -/**********************************************************************/ - -#define open_input _glp_mpl_open_input -void open_input(MPL *mpl, char *file); -/* open input text file */ - -#define read_char _glp_mpl_read_char -int read_char(MPL *mpl); -/* read next character from input text file */ - -#define close_input _glp_mpl_close_input -void close_input(MPL *mpl); -/* close input text file */ - -#define open_output _glp_mpl_open_output -void open_output(MPL *mpl, char *file); -/* open output text file */ - -#define write_char _glp_mpl_write_char -void write_char(MPL *mpl, int c); -/* write next character to output text file */ - -#define write_text _glp_mpl_write_text -void write_text(MPL *mpl, char *fmt, ...); -/* format and write text to output text file */ - -#define flush_output _glp_mpl_flush_output -void flush_output(MPL *mpl); -/* finalize writing data to output text file */ - -/**********************************************************************/ -/* * * SOLVER INTERFACE * * */ -/**********************************************************************/ - -#define MPL_FR 401 /* free (unbounded) */ -#define MPL_LO 402 /* lower bound */ -#define MPL_UP 403 /* upper bound */ -#define MPL_DB 404 /* both lower and upper bounds */ -#define MPL_FX 405 /* fixed */ - -#define MPL_ST 411 /* constraint */ -#define MPL_MIN 412 /* objective (minimization) */ -#define MPL_MAX 413 /* objective (maximization) */ - -#define MPL_NUM 421 /* continuous */ -#define MPL_INT 422 /* integer */ -#define MPL_BIN 423 /* binary */ - -#define error _glp_mpl_error -void error(MPL *mpl, char *fmt, ...); -/* print error message and terminate model processing */ - -#define warning _glp_mpl_warning -void warning(MPL *mpl, char *fmt, ...); -/* print warning message and continue model processing */ - -#define mpl_initialize _glp_mpl_initialize -MPL *mpl_initialize(void); -/* create and initialize translator database */ - -#define mpl_read_model _glp_mpl_read_model -int mpl_read_model(MPL *mpl, char *file, int skip_data); -/* read model section and optional data section */ - -#define mpl_read_data _glp_mpl_read_data -int mpl_read_data(MPL *mpl, char *file); -/* read data section */ - -#define mpl_generate _glp_mpl_generate -int mpl_generate(MPL *mpl, char *file); -/* generate model */ - -#define mpl_get_prob_name _glp_mpl_get_prob_name -char *mpl_get_prob_name(MPL *mpl); -/* obtain problem (model) name */ - -#define mpl_get_num_rows _glp_mpl_get_num_rows -int mpl_get_num_rows(MPL *mpl); -/* determine number of rows */ - -#define mpl_get_num_cols _glp_mpl_get_num_cols -int mpl_get_num_cols(MPL *mpl); -/* determine number of columns */ - -#define mpl_get_row_name _glp_mpl_get_row_name -char *mpl_get_row_name(MPL *mpl, int i); -/* obtain row name */ - -#define mpl_get_row_kind _glp_mpl_get_row_kind -int mpl_get_row_kind(MPL *mpl, int i); -/* determine row kind */ - -#define mpl_get_row_bnds _glp_mpl_get_row_bnds -int mpl_get_row_bnds(MPL *mpl, int i, double *lb, double *ub); -/* obtain row bounds */ - -#define mpl_get_mat_row _glp_mpl_get_mat_row -int mpl_get_mat_row(MPL *mpl, int i, int ndx[], double val[]); -/* obtain row of the constraint matrix */ - -#define mpl_get_row_c0 _glp_mpl_get_row_c0 -double mpl_get_row_c0(MPL *mpl, int i); -/* obtain constant term of free row */ - -#define mpl_get_col_name _glp_mpl_get_col_name -char *mpl_get_col_name(MPL *mpl, int j); -/* obtain column name */ - -#define mpl_get_col_kind _glp_mpl_get_col_kind -int mpl_get_col_kind(MPL *mpl, int j); -/* determine column kind */ - -#define mpl_get_col_bnds _glp_mpl_get_col_bnds -int mpl_get_col_bnds(MPL *mpl, int j, double *lb, double *ub); -/* obtain column bounds */ - -#define mpl_has_solve_stmt _glp_mpl_has_solve_stmt -int mpl_has_solve_stmt(MPL *mpl); -/* check if model has solve statement */ - -#if 1 /* 15/V-2010 */ -#define mpl_put_row_soln _glp_mpl_put_row_soln -void mpl_put_row_soln(MPL *mpl, int i, int stat, double prim, - double dual); -/* store row (constraint/objective) solution components */ -#endif - -#if 1 /* 15/V-2010 */ -#define mpl_put_col_soln _glp_mpl_put_col_soln -void mpl_put_col_soln(MPL *mpl, int j, int stat, double prim, - double dual); -/* store column (variable) solution components */ -#endif - -#if 0 /* 15/V-2010 */ -#define mpl_put_col_value _glp_mpl_put_col_value -void mpl_put_col_value(MPL *mpl, int j, double val); -/* store column value */ -#endif - -#define mpl_postsolve _glp_mpl_postsolve -int mpl_postsolve(MPL *mpl); -/* postsolve model */ - -#define mpl_terminate _glp_mpl_terminate -void mpl_terminate(MPL *mpl); -/* free all resources used by translator */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl1.c b/code/3rd_glpk/mpl/mpl1.c deleted file mode 100644 index 7dc3cd79..00000000 --- a/code/3rd_glpk/mpl/mpl1.c +++ /dev/null @@ -1,4718 +0,0 @@ -/* mpl1.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mpl.h" - -#define dmp_get_atomv dmp_get_atom - -/**********************************************************************/ -/* * * PROCESSING MODEL SECTION * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- enter_context - enter current token into context queue. --- --- This routine enters the current token into the context queue. */ - -void enter_context(MPL *mpl) -{ char *image, *s; - if (mpl->token == T_EOF) - image = "_|_"; - else if (mpl->token == T_STRING) - image = "'...'"; - else - image = mpl->image; - xassert(0 <= mpl->c_ptr && mpl->c_ptr < CONTEXT_SIZE); - mpl->context[mpl->c_ptr++] = ' '; - if (mpl->c_ptr == CONTEXT_SIZE) mpl->c_ptr = 0; - for (s = image; *s != '\0'; s++) - { mpl->context[mpl->c_ptr++] = *s; - if (mpl->c_ptr == CONTEXT_SIZE) mpl->c_ptr = 0; - } - return; -} - -/*---------------------------------------------------------------------- --- print_context - print current content of context queue. --- --- This routine prints current content of the context queue. */ - -void print_context(MPL *mpl) -{ int c; - while (mpl->c_ptr > 0) - { mpl->c_ptr--; - c = mpl->context[0]; - memmove(mpl->context, mpl->context+1, CONTEXT_SIZE-1); - mpl->context[CONTEXT_SIZE-1] = (char)c; - } - xprintf("Context: %s%.*s\n", mpl->context[0] == ' ' ? "" : "...", - CONTEXT_SIZE, mpl->context); - return; -} - -/*---------------------------------------------------------------------- --- get_char - scan next character from input text file. --- --- This routine scans a next ASCII character from the input text file. --- In case of end-of-file, the character is assigned EOF. */ - -void get_char(MPL *mpl) -{ int c; - if (mpl->c == EOF) goto done; - if (mpl->c == '\n') mpl->line++; - c = read_char(mpl); - if (c == EOF) - { if (mpl->c == '\n') - mpl->line--; - else - warning(mpl, "final NL missing before end of file"); - } - else if (c == '\n') - ; - else if (isspace(c)) - c = ' '; - else if (iscntrl(c)) - { enter_context(mpl); - error(mpl, "control character 0x%02X not allowed", c); - } - mpl->c = c; -done: return; -} - -/*---------------------------------------------------------------------- --- append_char - append character to current token. --- --- This routine appends the current character to the current token and --- then scans a next character. */ - -void append_char(MPL *mpl) -{ xassert(0 <= mpl->imlen && mpl->imlen <= MAX_LENGTH); - if (mpl->imlen == MAX_LENGTH) - { switch (mpl->token) - { case T_NAME: - enter_context(mpl); - error(mpl, "symbolic name %s... too long", mpl->image); - case T_SYMBOL: - enter_context(mpl); - error(mpl, "symbol %s... too long", mpl->image); - case T_NUMBER: - enter_context(mpl); - error(mpl, "numeric literal %s... too long", mpl->image); - case T_STRING: - enter_context(mpl); - error(mpl, "string literal too long"); - default: - xassert(mpl != mpl); - } - } - mpl->image[mpl->imlen++] = (char)mpl->c; - mpl->image[mpl->imlen] = '\0'; - get_char(mpl); - return; -} - -/*---------------------------------------------------------------------- --- get_token - scan next token from input text file. --- --- This routine scans a next token from the input text file using the --- standard finite automation technique. */ - -void get_token(MPL *mpl) -{ /* save the current token */ - mpl->b_token = mpl->token; - mpl->b_imlen = mpl->imlen; - strcpy(mpl->b_image, mpl->image); - mpl->b_value = mpl->value; - /* if the next token is already scanned, make it current */ - if (mpl->f_scan) - { mpl->f_scan = 0; - mpl->token = mpl->f_token; - mpl->imlen = mpl->f_imlen; - strcpy(mpl->image, mpl->f_image); - mpl->value = mpl->f_value; - goto done; - } -loop: /* nothing has been scanned so far */ - mpl->token = 0; - mpl->imlen = 0; - mpl->image[0] = '\0'; - mpl->value = 0.0; - /* skip any uninteresting characters */ - while (mpl->c == ' ' || mpl->c == '\n') get_char(mpl); - /* recognize and construct the token */ - if (mpl->c == EOF) - { /* end-of-file reached */ - mpl->token = T_EOF; - } - else if (mpl->c == '#') - { /* comment; skip anything until end-of-line */ - while (mpl->c != '\n' && mpl->c != EOF) get_char(mpl); - goto loop; - } - else if (!mpl->flag_d && (isalpha(mpl->c) || mpl->c == '_')) - { /* symbolic name or reserved keyword */ - mpl->token = T_NAME; - while (isalnum(mpl->c) || mpl->c == '_') append_char(mpl); - if (strcmp(mpl->image, "and") == 0) - mpl->token = T_AND; - else if (strcmp(mpl->image, "by") == 0) - mpl->token = T_BY; - else if (strcmp(mpl->image, "cross") == 0) - mpl->token = T_CROSS; - else if (strcmp(mpl->image, "diff") == 0) - mpl->token = T_DIFF; - else if (strcmp(mpl->image, "div") == 0) - mpl->token = T_DIV; - else if (strcmp(mpl->image, "else") == 0) - mpl->token = T_ELSE; - else if (strcmp(mpl->image, "if") == 0) - mpl->token = T_IF; - else if (strcmp(mpl->image, "in") == 0) - mpl->token = T_IN; -#if 1 /* 21/VII-2006 */ - else if (strcmp(mpl->image, "Infinity") == 0) - mpl->token = T_INFINITY; -#endif - else if (strcmp(mpl->image, "inter") == 0) - mpl->token = T_INTER; - else if (strcmp(mpl->image, "less") == 0) - mpl->token = T_LESS; - else if (strcmp(mpl->image, "mod") == 0) - mpl->token = T_MOD; - else if (strcmp(mpl->image, "not") == 0) - mpl->token = T_NOT; - else if (strcmp(mpl->image, "or") == 0) - mpl->token = T_OR; - else if (strcmp(mpl->image, "s") == 0 && mpl->c == '.') - { mpl->token = T_SPTP; - append_char(mpl); - if (mpl->c != 't') -sptp: { enter_context(mpl); - error(mpl, "keyword s.t. incomplete"); - } - append_char(mpl); - if (mpl->c != '.') goto sptp; - append_char(mpl); - } - else if (strcmp(mpl->image, "symdiff") == 0) - mpl->token = T_SYMDIFF; - else if (strcmp(mpl->image, "then") == 0) - mpl->token = T_THEN; - else if (strcmp(mpl->image, "union") == 0) - mpl->token = T_UNION; - else if (strcmp(mpl->image, "within") == 0) - mpl->token = T_WITHIN; - } - else if (!mpl->flag_d && isdigit(mpl->c)) - { /* numeric literal */ - mpl->token = T_NUMBER; - /* scan integer part */ - while (isdigit(mpl->c)) append_char(mpl); - /* scan optional fractional part */ - if (mpl->c == '.') - { append_char(mpl); - if (mpl->c == '.') - { /* hmm, it is not the fractional part, it is dots that - follow the integer part */ - mpl->imlen--; - mpl->image[mpl->imlen] = '\0'; - mpl->f_dots = 1; - goto conv; - } -frac: while (isdigit(mpl->c)) append_char(mpl); - } - /* scan optional decimal exponent */ - if (mpl->c == 'e' || mpl->c == 'E') - { append_char(mpl); - if (mpl->c == '+' || mpl->c == '-') append_char(mpl); - if (!isdigit(mpl->c)) - { enter_context(mpl); - error(mpl, "numeric literal %s incomplete", mpl->image); - } - while (isdigit(mpl->c)) append_char(mpl); - } - /* there must be no letter following the numeric literal */ - if (isalpha(mpl->c) || mpl->c == '_') - { enter_context(mpl); - error(mpl, "symbol %s%c... should be enclosed in quotes", - mpl->image, mpl->c); - } -conv: /* convert numeric literal to floating-point */ - if (str2num(mpl->image, &mpl->value)) -err: { enter_context(mpl); - error(mpl, "cannot convert numeric literal %s to floating-p" - "oint number", mpl->image); - } - } - else if (mpl->c == '\'' || mpl->c == '"') - { /* character string */ - int quote = mpl->c; - mpl->token = T_STRING; - get_char(mpl); - for (;;) - { if (mpl->c == '\n' || mpl->c == EOF) - { enter_context(mpl); - error(mpl, "unexpected end of line; string literal incom" - "plete"); - } - if (mpl->c == quote) - { get_char(mpl); - if (mpl->c != quote) break; - } - append_char(mpl); - } - } - else if (!mpl->flag_d && mpl->c == '+') - mpl->token = T_PLUS, append_char(mpl); - else if (!mpl->flag_d && mpl->c == '-') - mpl->token = T_MINUS, append_char(mpl); - else if (mpl->c == '*') - { mpl->token = T_ASTERISK, append_char(mpl); - if (mpl->c == '*') - mpl->token = T_POWER, append_char(mpl); - } - else if (mpl->c == '/') - { mpl->token = T_SLASH, append_char(mpl); - if (mpl->c == '*') - { /* comment sequence */ - get_char(mpl); - for (;;) - { if (mpl->c == EOF) - { /* do not call enter_context at this point */ - error(mpl, "unexpected end of file; comment sequence " - "incomplete"); - } - else if (mpl->c == '*') - { get_char(mpl); - if (mpl->c == '/') break; - } - else - get_char(mpl); - } - get_char(mpl); - goto loop; - } - } - else if (mpl->c == '^') - mpl->token = T_POWER, append_char(mpl); - else if (mpl->c == '<') - { mpl->token = T_LT, append_char(mpl); - if (mpl->c == '=') - mpl->token = T_LE, append_char(mpl); - else if (mpl->c == '>') - mpl->token = T_NE, append_char(mpl); -#if 1 /* 11/II-2008 */ - else if (mpl->c == '-') - mpl->token = T_INPUT, append_char(mpl); -#endif - } - else if (mpl->c == '=') - { mpl->token = T_EQ, append_char(mpl); - if (mpl->c == '=') append_char(mpl); - } - else if (mpl->c == '>') - { mpl->token = T_GT, append_char(mpl); - if (mpl->c == '=') - mpl->token = T_GE, append_char(mpl); -#if 1 /* 14/VII-2006 */ - else if (mpl->c == '>') - mpl->token = T_APPEND, append_char(mpl); -#endif - } - else if (mpl->c == '!') - { mpl->token = T_NOT, append_char(mpl); - if (mpl->c == '=') - mpl->token = T_NE, append_char(mpl); - } - else if (mpl->c == '&') - { mpl->token = T_CONCAT, append_char(mpl); - if (mpl->c == '&') - mpl->token = T_AND, append_char(mpl); - } - else if (mpl->c == '|') - { mpl->token = T_BAR, append_char(mpl); - if (mpl->c == '|') - mpl->token = T_OR, append_char(mpl); - } - else if (!mpl->flag_d && mpl->c == '.') - { mpl->token = T_POINT, append_char(mpl); - if (mpl->f_dots) - { /* dots; the first dot was read on the previous call to the - scanner, so the current character is the second dot */ - mpl->token = T_DOTS; - mpl->imlen = 2; - strcpy(mpl->image, ".."); - mpl->f_dots = 0; - } - else if (mpl->c == '.') - mpl->token = T_DOTS, append_char(mpl); - else if (isdigit(mpl->c)) - { /* numeric literal that begins with the decimal point */ - mpl->token = T_NUMBER, append_char(mpl); - goto frac; - } - } - else if (mpl->c == ',') - mpl->token = T_COMMA, append_char(mpl); - else if (mpl->c == ':') - { mpl->token = T_COLON, append_char(mpl); - if (mpl->c == '=') - mpl->token = T_ASSIGN, append_char(mpl); - } - else if (mpl->c == ';') - mpl->token = T_SEMICOLON, append_char(mpl); - else if (mpl->c == '(') - mpl->token = T_LEFT, append_char(mpl); - else if (mpl->c == ')') - mpl->token = T_RIGHT, append_char(mpl); - else if (mpl->c == '[') - mpl->token = T_LBRACKET, append_char(mpl); - else if (mpl->c == ']') - mpl->token = T_RBRACKET, append_char(mpl); - else if (mpl->c == '{') - mpl->token = T_LBRACE, append_char(mpl); - else if (mpl->c == '}') - mpl->token = T_RBRACE, append_char(mpl); -#if 1 /* 11/II-2008 */ - else if (mpl->c == '~') - mpl->token = T_TILDE, append_char(mpl); -#endif - else if (isalnum(mpl->c) || strchr("+-._", mpl->c) != NULL) - { /* symbol */ - xassert(mpl->flag_d); - mpl->token = T_SYMBOL; - while (isalnum(mpl->c) || strchr("+-._", mpl->c) != NULL) - append_char(mpl); - switch (str2num(mpl->image, &mpl->value)) - { case 0: - mpl->token = T_NUMBER; - break; - case 1: - goto err; - case 2: - break; - default: - xassert(mpl != mpl); - } - } - else - { enter_context(mpl); - error(mpl, "character %c not allowed", mpl->c); - } - /* enter the current token into the context queue */ - enter_context(mpl); - /* reset the flag, which may be set by indexing_expression() and - is used by expression_list() */ - mpl->flag_x = 0; -done: return; -} - -/*---------------------------------------------------------------------- --- unget_token - return current token back to input stream. --- --- This routine returns the current token back to the input stream, so --- the previously scanned token becomes the current one. */ - -void unget_token(MPL *mpl) -{ /* save the current token, which becomes the next one */ - xassert(!mpl->f_scan); - mpl->f_scan = 1; - mpl->f_token = mpl->token; - mpl->f_imlen = mpl->imlen; - strcpy(mpl->f_image, mpl->image); - mpl->f_value = mpl->value; - /* restore the previous token, which becomes the current one */ - mpl->token = mpl->b_token; - mpl->imlen = mpl->b_imlen; - strcpy(mpl->image, mpl->b_image); - mpl->value = mpl->b_value; - return; -} - -/*---------------------------------------------------------------------- --- is_keyword - check if current token is given non-reserved keyword. --- --- If the current token is given (non-reserved) keyword, this routine --- returns non-zero. Otherwise zero is returned. */ - -int is_keyword(MPL *mpl, char *keyword) -{ return - mpl->token == T_NAME && strcmp(mpl->image, keyword) == 0; -} - -/*---------------------------------------------------------------------- --- is_reserved - check if current token is reserved keyword. --- --- If the current token is a reserved keyword, this routine returns --- non-zero. Otherwise zero is returned. */ - -int is_reserved(MPL *mpl) -{ return - mpl->token == T_AND && mpl->image[0] == 'a' || - mpl->token == T_BY || - mpl->token == T_CROSS || - mpl->token == T_DIFF || - mpl->token == T_DIV || - mpl->token == T_ELSE || - mpl->token == T_IF || - mpl->token == T_IN || - mpl->token == T_INTER || - mpl->token == T_LESS || - mpl->token == T_MOD || - mpl->token == T_NOT && mpl->image[0] == 'n' || - mpl->token == T_OR && mpl->image[0] == 'o' || - mpl->token == T_SYMDIFF || - mpl->token == T_THEN || - mpl->token == T_UNION || - mpl->token == T_WITHIN; -} - -/*---------------------------------------------------------------------- --- make_code - generate pseudo-code (basic routine). --- --- This routine generates specified pseudo-code. It is assumed that all --- other translator routines use this basic routine. */ - -CODE *make_code(MPL *mpl, int op, OPERANDS *arg, int type, int dim) -{ CODE *code; - DOMAIN *domain; - DOMAIN_BLOCK *block; - ARG_LIST *e; - /* generate pseudo-code */ - code = alloc(CODE); - code->op = op; - code->vflag = 0; /* is inherited from operand(s) */ - /* copy operands and also make them referring to the pseudo-code - being generated, because the latter becomes the parent for all - its operands */ - memset(&code->arg, '?', sizeof(OPERANDS)); - switch (op) - { case O_NUMBER: - code->arg.num = arg->num; - break; - case O_STRING: - code->arg.str = arg->str; - break; - case O_INDEX: - code->arg.index.slot = arg->index.slot; - code->arg.index.next = arg->index.next; - break; - case O_MEMNUM: - case O_MEMSYM: - for (e = arg->par.list; e != NULL; e = e->next) - { xassert(e->x != NULL); - xassert(e->x->up == NULL); - e->x->up = code; - code->vflag |= e->x->vflag; - } - code->arg.par.par = arg->par.par; - code->arg.par.list = arg->par.list; - break; - case O_MEMSET: - for (e = arg->set.list; e != NULL; e = e->next) - { xassert(e->x != NULL); - xassert(e->x->up == NULL); - e->x->up = code; - code->vflag |= e->x->vflag; - } - code->arg.set.set = arg->set.set; - code->arg.set.list = arg->set.list; - break; - case O_MEMVAR: - for (e = arg->var.list; e != NULL; e = e->next) - { xassert(e->x != NULL); - xassert(e->x->up == NULL); - e->x->up = code; - code->vflag |= e->x->vflag; - } - code->arg.var.var = arg->var.var; - code->arg.var.list = arg->var.list; -#if 1 /* 15/V-2010 */ - code->arg.var.suff = arg->var.suff; -#endif - break; -#if 1 /* 15/V-2010 */ - case O_MEMCON: - for (e = arg->con.list; e != NULL; e = e->next) - { xassert(e->x != NULL); - xassert(e->x->up == NULL); - e->x->up = code; - code->vflag |= e->x->vflag; - } - code->arg.con.con = arg->con.con; - code->arg.con.list = arg->con.list; - code->arg.con.suff = arg->con.suff; - break; -#endif - case O_TUPLE: - case O_MAKE: - for (e = arg->list; e != NULL; e = e->next) - { xassert(e->x != NULL); - xassert(e->x->up == NULL); - e->x->up = code; - code->vflag |= e->x->vflag; - } - code->arg.list = arg->list; - break; - case O_SLICE: - xassert(arg->slice != NULL); - code->arg.slice = arg->slice; - break; - case O_IRAND224: - case O_UNIFORM01: - case O_NORMAL01: - case O_GMTIME: - code->vflag = 1; - break; - case O_CVTNUM: - case O_CVTSYM: - case O_CVTLOG: - case O_CVTTUP: - case O_CVTLFM: - case O_PLUS: - case O_MINUS: - case O_NOT: - case O_ABS: - case O_CEIL: - case O_FLOOR: - case O_EXP: - case O_LOG: - case O_LOG10: - case O_SQRT: - case O_SIN: - case O_COS: - case O_TAN: - case O_ATAN: - case O_ROUND: - case O_TRUNC: - case O_CARD: - case O_LENGTH: - /* unary operation */ - xassert(arg->arg.x != NULL); - xassert(arg->arg.x->up == NULL); - arg->arg.x->up = code; - code->vflag |= arg->arg.x->vflag; - code->arg.arg.x = arg->arg.x; - break; - case O_ADD: - case O_SUB: - case O_LESS: - case O_MUL: - case O_DIV: - case O_IDIV: - case O_MOD: - case O_POWER: - case O_ATAN2: - case O_ROUND2: - case O_TRUNC2: - case O_UNIFORM: - if (op == O_UNIFORM) code->vflag = 1; - case O_NORMAL: - if (op == O_NORMAL) code->vflag = 1; - case O_CONCAT: - case O_LT: - case O_LE: - case O_EQ: - case O_GE: - case O_GT: - case O_NE: - case O_AND: - case O_OR: - case O_UNION: - case O_DIFF: - case O_SYMDIFF: - case O_INTER: - case O_CROSS: - case O_IN: - case O_NOTIN: - case O_WITHIN: - case O_NOTWITHIN: - case O_SUBSTR: - case O_STR2TIME: - case O_TIME2STR: - /* binary operation */ - xassert(arg->arg.x != NULL); - xassert(arg->arg.x->up == NULL); - arg->arg.x->up = code; - code->vflag |= arg->arg.x->vflag; - xassert(arg->arg.y != NULL); - xassert(arg->arg.y->up == NULL); - arg->arg.y->up = code; - code->vflag |= arg->arg.y->vflag; - code->arg.arg.x = arg->arg.x; - code->arg.arg.y = arg->arg.y; - break; - case O_DOTS: - case O_FORK: - case O_SUBSTR3: - /* ternary operation */ - xassert(arg->arg.x != NULL); - xassert(arg->arg.x->up == NULL); - arg->arg.x->up = code; - code->vflag |= arg->arg.x->vflag; - xassert(arg->arg.y != NULL); - xassert(arg->arg.y->up == NULL); - arg->arg.y->up = code; - code->vflag |= arg->arg.y->vflag; - if (arg->arg.z != NULL) - { xassert(arg->arg.z->up == NULL); - arg->arg.z->up = code; - code->vflag |= arg->arg.z->vflag; - } - code->arg.arg.x = arg->arg.x; - code->arg.arg.y = arg->arg.y; - code->arg.arg.z = arg->arg.z; - break; - case O_MIN: - case O_MAX: - /* n-ary operation */ - for (e = arg->list; e != NULL; e = e->next) - { xassert(e->x != NULL); - xassert(e->x->up == NULL); - e->x->up = code; - code->vflag |= e->x->vflag; - } - code->arg.list = arg->list; - break; - case O_SUM: - case O_PROD: - case O_MINIMUM: - case O_MAXIMUM: - case O_FORALL: - case O_EXISTS: - case O_SETOF: - case O_BUILD: - /* iterated operation */ - domain = arg->loop.domain; - xassert(domain != NULL); - if (domain->code != NULL) - { xassert(domain->code->up == NULL); - domain->code->up = code; - code->vflag |= domain->code->vflag; - } - for (block = domain->list; block != NULL; block = - block->next) - { xassert(block->code != NULL); - xassert(block->code->up == NULL); - block->code->up = code; - code->vflag |= block->code->vflag; - } - if (arg->loop.x != NULL) - { xassert(arg->loop.x->up == NULL); - arg->loop.x->up = code; - code->vflag |= arg->loop.x->vflag; - } - code->arg.loop.domain = arg->loop.domain; - code->arg.loop.x = arg->loop.x; - break; - default: - xassert(op != op); - } - /* set other attributes of the pseudo-code */ - code->type = type; - code->dim = dim; - code->up = NULL; - code->valid = 0; - memset(&code->value, '?', sizeof(VALUE)); - return code; -} - -/*---------------------------------------------------------------------- --- make_unary - generate pseudo-code for unary operation. --- --- This routine generates pseudo-code for unary operation. */ - -CODE *make_unary(MPL *mpl, int op, CODE *x, int type, int dim) -{ CODE *code; - OPERANDS arg; - xassert(x != NULL); - arg.arg.x = x; - code = make_code(mpl, op, &arg, type, dim); - return code; -} - -/*---------------------------------------------------------------------- --- make_binary - generate pseudo-code for binary operation. --- --- This routine generates pseudo-code for binary operation. */ - -CODE *make_binary(MPL *mpl, int op, CODE *x, CODE *y, int type, - int dim) -{ CODE *code; - OPERANDS arg; - xassert(x != NULL); - xassert(y != NULL); - arg.arg.x = x; - arg.arg.y = y; - code = make_code(mpl, op, &arg, type, dim); - return code; -} - -/*---------------------------------------------------------------------- --- make_ternary - generate pseudo-code for ternary operation. --- --- This routine generates pseudo-code for ternary operation. */ - -CODE *make_ternary(MPL *mpl, int op, CODE *x, CODE *y, CODE *z, - int type, int dim) -{ CODE *code; - OPERANDS arg; - xassert(x != NULL); - xassert(y != NULL); - /* third operand can be NULL */ - arg.arg.x = x; - arg.arg.y = y; - arg.arg.z = z; - code = make_code(mpl, op, &arg, type, dim); - return code; -} - -/*---------------------------------------------------------------------- --- numeric_literal - parse reference to numeric literal. --- --- This routine parses primary expression using the syntax: --- --- ::= */ - -CODE *numeric_literal(MPL *mpl) -{ CODE *code; - OPERANDS arg; - xassert(mpl->token == T_NUMBER); - arg.num = mpl->value; - code = make_code(mpl, O_NUMBER, &arg, A_NUMERIC, 0); - get_token(mpl /* */); - return code; -} - -/*---------------------------------------------------------------------- --- string_literal - parse reference to string literal. --- --- This routine parses primary expression using the syntax: --- --- ::= */ - -CODE *string_literal(MPL *mpl) -{ CODE *code; - OPERANDS arg; - xassert(mpl->token == T_STRING); - arg.str = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(arg.str, mpl->image); - code = make_code(mpl, O_STRING, &arg, A_SYMBOLIC, 0); - get_token(mpl /* */); - return code; -} - -/*---------------------------------------------------------------------- --- create_arg_list - create empty operands list. --- --- This routine creates operands list, which is initially empty. */ - -ARG_LIST *create_arg_list(MPL *mpl) -{ ARG_LIST *list; - xassert(mpl == mpl); - list = NULL; - return list; -} - -/*---------------------------------------------------------------------- --- expand_arg_list - append operand to operands list. --- --- This routine appends new operand to specified operands list. */ - -ARG_LIST *expand_arg_list(MPL *mpl, ARG_LIST *list, CODE *x) -{ ARG_LIST *tail, *temp; - xassert(x != NULL); - /* create new operands list entry */ - tail = alloc(ARG_LIST); - tail->x = x; - tail->next = NULL; - /* and append it to the operands list */ - if (list == NULL) - list = tail; - else - { for (temp = list; temp->next != NULL; temp = temp->next); - temp->next = tail; - } - return list; -} - -/*---------------------------------------------------------------------- --- arg_list_len - determine length of operands list. --- --- This routine returns the number of operands in operands list. */ - -int arg_list_len(MPL *mpl, ARG_LIST *list) -{ ARG_LIST *temp; - int len; - xassert(mpl == mpl); - len = 0; - for (temp = list; temp != NULL; temp = temp->next) len++; - return len; -} - -/*---------------------------------------------------------------------- --- subscript_list - parse subscript list. --- --- This routine parses subscript list using the syntax: --- --- ::= --- ::= , --- ::= */ - -ARG_LIST *subscript_list(MPL *mpl) -{ ARG_LIST *list; - CODE *x; - list = create_arg_list(mpl); - for (;;) - { /* parse subscript expression */ - x = expression_5(mpl); - /* convert it to symbolic type, if necessary */ - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0); - /* check that now the expression is of symbolic type */ - if (x->type != A_SYMBOLIC) - error(mpl, "subscript expression has invalid type"); - xassert(x->dim == 0); - /* and append it to the subscript list */ - list = expand_arg_list(mpl, list, x); - /* check a token that follows the subscript expression */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RBRACKET) - break; - else - error(mpl, "syntax error in subscript list"); - } - return list; -} - -#if 1 /* 15/V-2010 */ -/*---------------------------------------------------------------------- --- object_reference - parse reference to named object. --- --- This routine parses primary expression using the syntax: --- --- ::= --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- --- ::= --- ::= [ ] --- --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= | .lb | .ub | .status | .val | .dual */ - -CODE *object_reference(MPL *mpl) -{ AVLNODE *node; - DOMAIN_SLOT *slot; - SET *set; - PARAMETER *par; - VARIABLE *var; - CONSTRAINT *con; - ARG_LIST *list; - OPERANDS arg; - CODE *code; - char *name; - int dim, suff; - /* find the object in the symbolic name table */ - xassert(mpl->token == T_NAME); - node = avl_find_node(mpl->tree, mpl->image); - if (node == NULL) - error(mpl, "%s not defined", mpl->image); - /* check the object type and obtain its dimension */ - switch (avl_get_node_type(node)) - { case A_INDEX: - /* dummy index */ - slot = (DOMAIN_SLOT *)avl_get_node_link(node); - name = slot->name; - dim = 0; - break; - case A_SET: - /* model set */ - set = (SET *)avl_get_node_link(node); - name = set->name; - dim = set->dim; - /* if a set object is referenced in its own declaration and - the dimen attribute is not specified yet, use dimen 1 by - default */ - if (set->dimen == 0) set->dimen = 1; - break; - case A_PARAMETER: - /* model parameter */ - par = (PARAMETER *)avl_get_node_link(node); - name = par->name; - dim = par->dim; - break; - case A_VARIABLE: - /* model variable */ - var = (VARIABLE *)avl_get_node_link(node); - name = var->name; - dim = var->dim; - break; - case A_CONSTRAINT: - /* model constraint or objective */ - con = (CONSTRAINT *)avl_get_node_link(node); - name = con->name; - dim = con->dim; - break; - default: - xassert(node != node); - } - get_token(mpl /* */); - /* parse optional subscript list */ - if (mpl->token == T_LBRACKET) - { /* subscript list is specified */ - if (dim == 0) - error(mpl, "%s cannot be subscripted", name); - get_token(mpl /* [ */); - list = subscript_list(mpl); - if (dim != arg_list_len(mpl, list)) - error(mpl, "%s must have %d subscript%s rather than %d", - name, dim, dim == 1 ? "" : "s", arg_list_len(mpl, list)); - xassert(mpl->token == T_RBRACKET); - get_token(mpl /* ] */); - } - else - { /* subscript list is not specified */ - if (dim != 0) - error(mpl, "%s must be subscripted", name); - list = create_arg_list(mpl); - } - /* parse optional suffix */ - if (!mpl->flag_s && avl_get_node_type(node) == A_VARIABLE) - suff = DOT_NONE; - else - suff = DOT_VAL; - if (mpl->token == T_POINT) - { get_token(mpl /* . */); - if (mpl->token != T_NAME) - error(mpl, "invalid use of period"); - if (!(avl_get_node_type(node) == A_VARIABLE || - avl_get_node_type(node) == A_CONSTRAINT)) - error(mpl, "%s cannot have a suffix", name); - if (strcmp(mpl->image, "lb") == 0) - suff = DOT_LB; - else if (strcmp(mpl->image, "ub") == 0) - suff = DOT_UB; - else if (strcmp(mpl->image, "status") == 0) - suff = DOT_STATUS; - else if (strcmp(mpl->image, "val") == 0) - suff = DOT_VAL; - else if (strcmp(mpl->image, "dual") == 0) - suff = DOT_DUAL; - else - error(mpl, "suffix .%s invalid", mpl->image); - get_token(mpl /* suffix */); - } - /* generate pseudo-code to take value of the object */ - switch (avl_get_node_type(node)) - { case A_INDEX: - arg.index.slot = slot; - arg.index.next = slot->list; - code = make_code(mpl, O_INDEX, &arg, A_SYMBOLIC, 0); - slot->list = code; - break; - case A_SET: - arg.set.set = set; - arg.set.list = list; - code = make_code(mpl, O_MEMSET, &arg, A_ELEMSET, - set->dimen); - break; - case A_PARAMETER: - arg.par.par = par; - arg.par.list = list; - if (par->type == A_SYMBOLIC) - code = make_code(mpl, O_MEMSYM, &arg, A_SYMBOLIC, 0); - else - code = make_code(mpl, O_MEMNUM, &arg, A_NUMERIC, 0); - break; - case A_VARIABLE: - if (!mpl->flag_s && (suff == DOT_STATUS || suff == DOT_VAL - || suff == DOT_DUAL)) - error(mpl, "invalid reference to status, primal value, o" - "r dual value of variable %s above solve statement", - var->name); - arg.var.var = var; - arg.var.list = list; - arg.var.suff = suff; - code = make_code(mpl, O_MEMVAR, &arg, suff == DOT_NONE ? - A_FORMULA : A_NUMERIC, 0); - break; - case A_CONSTRAINT: - if (!mpl->flag_s && (suff == DOT_STATUS || suff == DOT_VAL - || suff == DOT_DUAL)) - error(mpl, "invalid reference to status, primal value, o" - "r dual value of %s %s above solve statement", - con->type == A_CONSTRAINT ? "constraint" : "objective" - , con->name); - arg.con.con = con; - arg.con.list = list; - arg.con.suff = suff; - code = make_code(mpl, O_MEMCON, &arg, A_NUMERIC, 0); - break; - default: - xassert(node != node); - } - return code; -} -#endif - -/*---------------------------------------------------------------------- --- numeric_argument - parse argument passed to built-in function. --- --- This routine parses an argument passed to numeric built-in function --- using the syntax: --- --- ::= */ - -CODE *numeric_argument(MPL *mpl, char *func) -{ CODE *x; - x = expression_5(mpl); - /* convert the argument to numeric type, if necessary */ - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - /* check that now the argument is of numeric type */ - if (x->type != A_NUMERIC) - error(mpl, "argument for %s has invalid type", func); - xassert(x->dim == 0); - return x; -} - -#if 1 /* 15/VII-2006 */ -CODE *symbolic_argument(MPL *mpl, char *func) -{ CODE *x; - x = expression_5(mpl); - /* convert the argument to symbolic type, if necessary */ - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0); - /* check that now the argument is of symbolic type */ - if (x->type != A_SYMBOLIC) - error(mpl, "argument for %s has invalid type", func); - xassert(x->dim == 0); - return x; -} -#endif - -#if 1 /* 15/VII-2006 */ -CODE *elemset_argument(MPL *mpl, char *func) -{ CODE *x; - x = expression_9(mpl); - if (x->type != A_ELEMSET) - error(mpl, "argument for %s has invalid type", func); - xassert(x->dim > 0); - return x; -} -#endif - -/*---------------------------------------------------------------------- --- function_reference - parse reference to built-in function. --- --- This routine parses primary expression using the syntax: --- --- ::= abs ( ) --- ::= ceil ( ) --- ::= floor ( ) --- ::= exp ( ) --- ::= log ( ) --- ::= log10 ( ) --- ::= max ( ) --- ::= min ( ) --- ::= sqrt ( ) --- ::= sin ( ) --- ::= cos ( ) --- ::= tan ( ) --- ::= atan ( ) --- ::= atan2 ( , ) --- ::= round ( ) --- ::= round ( , ) --- ::= trunc ( ) --- ::= trunc ( , ) --- ::= Irand224 ( ) --- ::= Uniform01 ( ) --- ::= Uniform ( , ) --- ::= Normal01 ( ) --- ::= Normal ( , ) --- ::= card ( ) --- ::= length ( ) --- ::= substr ( , ) --- ::= substr ( , , ) --- ::= str2time ( , ) --- ::= time2str ( , ) --- ::= gmtime ( ) --- ::= --- ::= , */ - -CODE *function_reference(MPL *mpl) -{ CODE *code; - OPERANDS arg; - int op; - char func[15+1]; - /* determine operation code */ - xassert(mpl->token == T_NAME); - if (strcmp(mpl->image, "abs") == 0) - op = O_ABS; - else if (strcmp(mpl->image, "ceil") == 0) - op = O_CEIL; - else if (strcmp(mpl->image, "floor") == 0) - op = O_FLOOR; - else if (strcmp(mpl->image, "exp") == 0) - op = O_EXP; - else if (strcmp(mpl->image, "log") == 0) - op = O_LOG; - else if (strcmp(mpl->image, "log10") == 0) - op = O_LOG10; - else if (strcmp(mpl->image, "sqrt") == 0) - op = O_SQRT; - else if (strcmp(mpl->image, "sin") == 0) - op = O_SIN; - else if (strcmp(mpl->image, "cos") == 0) - op = O_COS; - else if (strcmp(mpl->image, "tan") == 0) - op = O_TAN; - else if (strcmp(mpl->image, "atan") == 0) - op = O_ATAN; - else if (strcmp(mpl->image, "min") == 0) - op = O_MIN; - else if (strcmp(mpl->image, "max") == 0) - op = O_MAX; - else if (strcmp(mpl->image, "round") == 0) - op = O_ROUND; - else if (strcmp(mpl->image, "trunc") == 0) - op = O_TRUNC; - else if (strcmp(mpl->image, "Irand224") == 0) - op = O_IRAND224; - else if (strcmp(mpl->image, "Uniform01") == 0) - op = O_UNIFORM01; - else if (strcmp(mpl->image, "Uniform") == 0) - op = O_UNIFORM; - else if (strcmp(mpl->image, "Normal01") == 0) - op = O_NORMAL01; - else if (strcmp(mpl->image, "Normal") == 0) - op = O_NORMAL; - else if (strcmp(mpl->image, "card") == 0) - op = O_CARD; - else if (strcmp(mpl->image, "length") == 0) - op = O_LENGTH; - else if (strcmp(mpl->image, "substr") == 0) - op = O_SUBSTR; - else if (strcmp(mpl->image, "str2time") == 0) - op = O_STR2TIME; - else if (strcmp(mpl->image, "time2str") == 0) - op = O_TIME2STR; - else if (strcmp(mpl->image, "gmtime") == 0) - op = O_GMTIME; - else - error(mpl, "function %s unknown", mpl->image); - /* save symbolic name of the function */ - strcpy(func, mpl->image); - xassert(strlen(func) < sizeof(func)); - get_token(mpl /* */); - /* check the left parenthesis that follows the function name */ - xassert(mpl->token == T_LEFT); - get_token(mpl /* ( */); - /* parse argument list */ - if (op == O_MIN || op == O_MAX) - { /* min and max allow arbitrary number of arguments */ - arg.list = create_arg_list(mpl); - /* parse argument list */ - for (;;) - { /* parse argument and append it to the operands list */ - arg.list = expand_arg_list(mpl, arg.list, - numeric_argument(mpl, func)); - /* check a token that follows the argument */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RIGHT) - break; - else - error(mpl, "syntax error in argument list for %s", func); - } - } - else if (op == O_IRAND224 || op == O_UNIFORM01 || op == - O_NORMAL01 || op == O_GMTIME) - { /* Irand224, Uniform01, Normal01, gmtime need no arguments */ - if (mpl->token != T_RIGHT) - error(mpl, "%s needs no arguments", func); - } - else if (op == O_UNIFORM || op == O_NORMAL) - { /* Uniform and Normal need two arguments */ - /* parse the first argument */ - arg.arg.x = numeric_argument(mpl, func); - /* check a token that follows the first argument */ - if (mpl->token == T_COMMA) - ; - else if (mpl->token == T_RIGHT) - error(mpl, "%s needs two arguments", func); - else - error(mpl, "syntax error in argument for %s", func); - get_token(mpl /* , */); - /* parse the second argument */ - arg.arg.y = numeric_argument(mpl, func); - /* check a token that follows the second argument */ - if (mpl->token == T_COMMA) - error(mpl, "%s needs two argument", func); - else if (mpl->token == T_RIGHT) - ; - else - error(mpl, "syntax error in argument for %s", func); - } - else if (op == O_ATAN || op == O_ROUND || op == O_TRUNC) - { /* atan, round, and trunc need one or two arguments */ - /* parse the first argument */ - arg.arg.x = numeric_argument(mpl, func); - /* parse the second argument, if specified */ - if (mpl->token == T_COMMA) - { switch (op) - { case O_ATAN: op = O_ATAN2; break; - case O_ROUND: op = O_ROUND2; break; - case O_TRUNC: op = O_TRUNC2; break; - default: xassert(op != op); - } - get_token(mpl /* , */); - arg.arg.y = numeric_argument(mpl, func); - } - /* check a token that follows the last argument */ - if (mpl->token == T_COMMA) - error(mpl, "%s needs one or two arguments", func); - else if (mpl->token == T_RIGHT) - ; - else - error(mpl, "syntax error in argument for %s", func); - } - else if (op == O_SUBSTR) - { /* substr needs two or three arguments */ - /* parse the first argument */ - arg.arg.x = symbolic_argument(mpl, func); - /* check a token that follows the first argument */ - if (mpl->token == T_COMMA) - ; - else if (mpl->token == T_RIGHT) - error(mpl, "%s needs two or three arguments", func); - else - error(mpl, "syntax error in argument for %s", func); - get_token(mpl /* , */); - /* parse the second argument */ - arg.arg.y = numeric_argument(mpl, func); - /* parse the third argument, if specified */ - if (mpl->token == T_COMMA) - { op = O_SUBSTR3; - get_token(mpl /* , */); - arg.arg.z = numeric_argument(mpl, func); - } - /* check a token that follows the last argument */ - if (mpl->token == T_COMMA) - error(mpl, "%s needs two or three arguments", func); - else if (mpl->token == T_RIGHT) - ; - else - error(mpl, "syntax error in argument for %s", func); - } - else if (op == O_STR2TIME) - { /* str2time needs two arguments, both symbolic */ - /* parse the first argument */ - arg.arg.x = symbolic_argument(mpl, func); - /* check a token that follows the first argument */ - if (mpl->token == T_COMMA) - ; - else if (mpl->token == T_RIGHT) - error(mpl, "%s needs two arguments", func); - else - error(mpl, "syntax error in argument for %s", func); - get_token(mpl /* , */); - /* parse the second argument */ - arg.arg.y = symbolic_argument(mpl, func); - /* check a token that follows the second argument */ - if (mpl->token == T_COMMA) - error(mpl, "%s needs two argument", func); - else if (mpl->token == T_RIGHT) - ; - else - error(mpl, "syntax error in argument for %s", func); - } - else if (op == O_TIME2STR) - { /* time2str needs two arguments, numeric and symbolic */ - /* parse the first argument */ - arg.arg.x = numeric_argument(mpl, func); - /* check a token that follows the first argument */ - if (mpl->token == T_COMMA) - ; - else if (mpl->token == T_RIGHT) - error(mpl, "%s needs two arguments", func); - else - error(mpl, "syntax error in argument for %s", func); - get_token(mpl /* , */); - /* parse the second argument */ - arg.arg.y = symbolic_argument(mpl, func); - /* check a token that follows the second argument */ - if (mpl->token == T_COMMA) - error(mpl, "%s needs two argument", func); - else if (mpl->token == T_RIGHT) - ; - else - error(mpl, "syntax error in argument for %s", func); - } - else - { /* other functions need one argument */ - if (op == O_CARD) - arg.arg.x = elemset_argument(mpl, func); - else if (op == O_LENGTH) - arg.arg.x = symbolic_argument(mpl, func); - else - arg.arg.x = numeric_argument(mpl, func); - /* check a token that follows the argument */ - if (mpl->token == T_COMMA) - error(mpl, "%s needs one argument", func); - else if (mpl->token == T_RIGHT) - ; - else - error(mpl, "syntax error in argument for %s", func); - } - /* make pseudo-code to call the built-in function */ - if (op == O_SUBSTR || op == O_SUBSTR3 || op == O_TIME2STR) - code = make_code(mpl, op, &arg, A_SYMBOLIC, 0); - else - code = make_code(mpl, op, &arg, A_NUMERIC, 0); - /* the reference ends with the right parenthesis */ - xassert(mpl->token == T_RIGHT); - get_token(mpl /* ) */); - return code; -} - -/*---------------------------------------------------------------------- --- create_domain - create empty domain. --- --- This routine creates empty domain, which is initially empty, i.e. --- has no domain blocks. */ - -DOMAIN *create_domain(MPL *mpl) -{ DOMAIN *domain; - domain = alloc(DOMAIN); - domain->list = NULL; - domain->code = NULL; - return domain; -} - -/*---------------------------------------------------------------------- --- create_block - create empty domain block. --- --- This routine creates empty domain block, which is initially empty, --- i.e. has no domain slots. */ - -DOMAIN_BLOCK *create_block(MPL *mpl) -{ DOMAIN_BLOCK *block; - block = alloc(DOMAIN_BLOCK); - block->list = NULL; - block->code = NULL; - block->backup = NULL; - block->next = NULL; - return block; -} - -/*---------------------------------------------------------------------- --- append_block - append domain block to specified domain. --- --- This routine adds given domain block to the end of the block list of --- specified domain. */ - -void append_block(MPL *mpl, DOMAIN *domain, DOMAIN_BLOCK *block) -{ DOMAIN_BLOCK *temp; - xassert(mpl == mpl); - xassert(domain != NULL); - xassert(block != NULL); - xassert(block->next == NULL); - if (domain->list == NULL) - domain->list = block; - else - { for (temp = domain->list; temp->next != NULL; temp = - temp->next); - temp->next = block; - } - return; -} - -/*---------------------------------------------------------------------- --- append_slot - create and append new slot to domain block. --- --- This routine creates new domain slot and adds it to the end of slot --- list of specified domain block. --- --- The parameter name is symbolic name of the dummy index associated --- with the slot (the character string must be allocated). NULL means --- the dummy index is not explicitly specified. --- --- The parameter code is pseudo-code for computing symbolic value, at --- which the dummy index is bounded. NULL means the dummy index is free --- in the domain scope. */ - -DOMAIN_SLOT *append_slot(MPL *mpl, DOMAIN_BLOCK *block, char *name, - CODE *code) -{ DOMAIN_SLOT *slot, *temp; - xassert(block != NULL); - slot = alloc(DOMAIN_SLOT); - slot->name = name; - slot->code = code; - slot->value = NULL; - slot->list = NULL; - slot->next = NULL; - if (block->list == NULL) - block->list = slot; - else - { for (temp = block->list; temp->next != NULL; temp = - temp->next); - temp->next = slot; - } - return slot; -} - -/*---------------------------------------------------------------------- --- expression_list - parse expression list. --- --- This routine parses a list of one or more expressions enclosed into --- the parentheses using the syntax: --- --- ::= ( ) --- ::= --- ::= , --- --- Note that this construction may have three different meanings: --- --- 1. If consists of only one expression, is a parenthesized expression, which may be of any --- valid type (not necessarily 1-tuple). --- --- 2. If consists of several expressions separated by --- commae, where no expression is undeclared symbolic name, is a n-tuple. --- --- 3. If consists of several expressions separated by --- commae, where at least one expression is undeclared symbolic name --- (that denotes a dummy index), is a slice and --- can be only used as constituent of indexing expression. */ - -#define max_dim 20 -/* maximal number of components allowed within parentheses */ - -CODE *expression_list(MPL *mpl) -{ CODE *code; - OPERANDS arg; - struct { char *name; CODE *code; } list[1+max_dim]; - int flag_x, next_token, dim, j, slice = 0; - xassert(mpl->token == T_LEFT); - /* the flag, which allows recognizing undeclared symbolic names - as dummy indices, will be automatically reset by get_token(), - so save it before scanning the next token */ - flag_x = mpl->flag_x; - get_token(mpl /* ( */); - /* parse */ - for (dim = 1; ; dim++) - { if (dim > max_dim) - error(mpl, "too many components within parentheses"); - /* current component of can be either dummy - index or expression */ - if (mpl->token == T_NAME) - { /* symbolic name is recognized as dummy index only if: - the flag, which allows that, is set, and - the name is followed by comma or right parenthesis, and - the name is undeclared */ - get_token(mpl /* */); - next_token = mpl->token; - unget_token(mpl); - if (!(flag_x && - (next_token == T_COMMA || next_token == T_RIGHT) && - avl_find_node(mpl->tree, mpl->image) == NULL)) - { /* this is not dummy index */ - goto expr; - } - /* all dummy indices within the same slice must have unique - symbolic names */ - for (j = 1; j < dim; j++) - { if (list[j].name != NULL && strcmp(list[j].name, - mpl->image) == 0) - error(mpl, "duplicate dummy index %s not allowed", - mpl->image); - } - /* current component of is dummy index */ - list[dim].name - = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(list[dim].name, mpl->image); - list[dim].code = NULL; - get_token(mpl /* */); - /* is a slice, because at least one dummy - index has appeared */ - slice = 1; - /* note that the context ( ) is not allowed, - i.e. in this case is considered as - a parenthesized expression */ - if (dim == 1 && mpl->token == T_RIGHT) - error(mpl, "%s not defined", list[dim].name); - } - else -expr: { /* current component of is expression */ - code = expression_13(mpl); - /* if the current expression is followed by comma or it is - not the very first expression, entire - is n-tuple or slice, in which case the current expression - should be converted to symbolic type, if necessary */ - if (mpl->token == T_COMMA || dim > 1) - { if (code->type == A_NUMERIC) - code = make_unary(mpl, O_CVTSYM, code, A_SYMBOLIC, 0); - /* now the expression must be of symbolic type */ - if (code->type != A_SYMBOLIC) - error(mpl, "component expression has invalid type"); - xassert(code->dim == 0); - } - list[dim].name = NULL; - list[dim].code = code; - } - /* check a token that follows the current component */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RIGHT) - break; - else - error(mpl, "right parenthesis missing where expected"); - } - /* generate pseudo-code for */ - if (dim == 1 && !slice) - { /* is a parenthesized expression */ - code = list[1].code; - } - else if (!slice) - { /* is a n-tuple */ - arg.list = create_arg_list(mpl); - for (j = 1; j <= dim; j++) - arg.list = expand_arg_list(mpl, arg.list, list[j].code); - code = make_code(mpl, O_TUPLE, &arg, A_TUPLE, dim); - } - else - { /* is a slice */ - arg.slice = create_block(mpl); - for (j = 1; j <= dim; j++) - append_slot(mpl, arg.slice, list[j].name, list[j].code); - /* note that actually pseudo-codes with op = O_SLICE are never - evaluated */ - code = make_code(mpl, O_SLICE, &arg, A_TUPLE, dim); - } - get_token(mpl /* ) */); - /* if is a slice, there must be the keyword - 'in', which follows the right parenthesis */ - if (slice && mpl->token != T_IN) - error(mpl, "keyword in missing where expected"); - /* if the slice flag is set and there is the keyword 'in', which - follows , the latter must be a slice */ - if (flag_x && mpl->token == T_IN && !slice) - { if (dim == 1) - error(mpl, "syntax error in indexing expression"); - else - error(mpl, "0-ary slice not allowed"); - } - return code; -} - -/*---------------------------------------------------------------------- --- literal set - parse literal set. --- --- This routine parses literal set using the syntax: --- --- ::= { } --- ::= --- ::= , --- ::= --- --- It is assumed that the left curly brace and the very first member --- expression that follows it are already parsed. The right curly brace --- remains unscanned on exit. */ - -CODE *literal_set(MPL *mpl, CODE *code) -{ OPERANDS arg; - int j; - xassert(code != NULL); - arg.list = create_arg_list(mpl); - /* parse */ - for (j = 1; ; j++) - { /* all member expressions must be n-tuples; so, if the current - expression is not n-tuple, convert it to 1-tuple */ - if (code->type == A_NUMERIC) - code = make_unary(mpl, O_CVTSYM, code, A_SYMBOLIC, 0); - if (code->type == A_SYMBOLIC) - code = make_unary(mpl, O_CVTTUP, code, A_TUPLE, 1); - /* now the expression must be n-tuple */ - if (code->type != A_TUPLE) - error(mpl, "member expression has invalid type"); - /* all member expressions must have identical dimension */ - if (arg.list != NULL && arg.list->x->dim != code->dim) - error(mpl, "member %d has %d component%s while member %d ha" - "s %d component%s", - j-1, arg.list->x->dim, arg.list->x->dim == 1 ? "" : "s", - j, code->dim, code->dim == 1 ? "" : "s"); - /* append the current expression to the member list */ - arg.list = expand_arg_list(mpl, arg.list, code); - /* check a token that follows the current expression */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RBRACE) - break; - else - error(mpl, "syntax error in literal set"); - /* parse the next expression that follows the comma */ - code = expression_5(mpl); - } - /* generate pseudo-code for */ - code = make_code(mpl, O_MAKE, &arg, A_ELEMSET, arg.list->x->dim); - return code; -} - -/*---------------------------------------------------------------------- --- indexing_expression - parse indexing expression. --- --- This routine parses indexing expression using the syntax: --- --- ::= --- ::= { } --- ::= { : } --- ::= --- ::= , --- ::= --- ::= in --- ::= in --- ::= --- ::= ( ) --- ::= --- ::= --- --- This routine creates domain for , where each --- domain block corresponds to , and each domain slot --- corresponds to individual indexing position. */ - -DOMAIN *indexing_expression(MPL *mpl) -{ DOMAIN *domain; - DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - CODE *code; - xassert(mpl->token == T_LBRACE); - get_token(mpl /* { */); - if (mpl->token == T_RBRACE) - error(mpl, "empty indexing expression not allowed"); - /* create domain to be constructed */ - domain = create_domain(mpl); - /* parse either or that follows the - left brace */ - for (;;) - { /* domain block for is not created yet */ - block = NULL; - /* pseudo-code for is not generated yet */ - code = NULL; - /* check a token, which begins with */ - if (mpl->token == T_NAME) - { /* it is a symbolic name */ - int next_token; - char *name; - /* symbolic name is recognized as dummy index only if it is - followed by the keyword 'in' and not declared */ - get_token(mpl /* */); - next_token = mpl->token; - unget_token(mpl); - if (!(next_token == T_IN && - avl_find_node(mpl->tree, mpl->image) == NULL)) - { /* this is not dummy index; the symbolic name begins an - expression, which is either or the - very first in */ - goto expr; - } - /* create domain block with one slot, which is assigned the - dummy index */ - block = create_block(mpl); - name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(name, mpl->image); - append_slot(mpl, block, name, NULL); - get_token(mpl /* */); - /* the keyword 'in' is already checked above */ - xassert(mpl->token == T_IN); - get_token(mpl /* in */); - /* that follows the keyword 'in' will be - parsed below */ - } - else if (mpl->token == T_LEFT) - { /* it is the left parenthesis; parse expression that begins - with this parenthesis (the flag is set in order to allow - recognizing slices; see the routine expression_list) */ - mpl->flag_x = 1; - code = expression_9(mpl); - if (code->op != O_SLICE) - { /* this is either or the very first - in */ - goto expr; - } - /* this is a slice; besides the corresponding domain block - is already created by expression_list() */ - block = code->arg.slice; - code = NULL; /* is not parsed yet */ - /* the keyword 'in' following the slice is already checked - by expression_list() */ - xassert(mpl->token == T_IN); - get_token(mpl /* in */); - /* that follows the keyword 'in' will be - parsed below */ - } -expr: /* parse expression that follows either the keyword 'in' (in - which case it can be as well as the - very first in ); note that - this expression can be already parsed above */ - if (code == NULL) code = expression_9(mpl); - /* check the type of the expression just parsed */ - if (code->type != A_ELEMSET) - { /* it is not and therefore it can only - be the very first in ; - however, then there must be no dummy index neither slice - between the left brace and this expression */ - if (block != NULL) - error(mpl, "domain expression has invalid type"); - /* parse the rest part of and make this set - be , i.e. the construction {a, b, c} - is parsed as it were written as {A}, where A = {a, b, c} - is a temporary elemental set */ - code = literal_set(mpl, code); - } - /* now pseudo-code for has been built */ - xassert(code != NULL); - xassert(code->type == A_ELEMSET); - xassert(code->dim > 0); - /* if domain block for the current is still - not created, create it for fake slice of the same dimension - as */ - if (block == NULL) - { int j; - block = create_block(mpl); - for (j = 1; j <= code->dim; j++) - append_slot(mpl, block, NULL, NULL); - } - /* number of indexing positions in must be - the same as dimension of n-tuples in basic set */ - { int dim = 0; - for (slot = block->list; slot != NULL; slot = slot->next) - dim++; - if (dim != code->dim) - error(mpl,"%d %s specified for set of dimension %d", - dim, dim == 1 ? "index" : "indices", code->dim); - } - /* store pseudo-code for in the domain block */ - xassert(block->code == NULL); - block->code = code; - /* and append the domain block to the domain */ - append_block(mpl, domain, block); - /* the current has been completely parsed; - include all its dummy indices into the symbolic name table - to make them available for referencing from expressions; - implicit declarations of dummy indices remain valid while - the corresponding domain scope is valid */ - for (slot = block->list; slot != NULL; slot = slot->next) - if (slot->name != NULL) - { AVLNODE *node; - xassert(avl_find_node(mpl->tree, slot->name) == NULL); - node = avl_insert_node(mpl->tree, slot->name); - avl_set_node_type(node, A_INDEX); - avl_set_node_link(node, (void *)slot); - } - /* check a token that follows */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_COLON || mpl->token == T_RBRACE) - break; - else - error(mpl, "syntax error in indexing expression"); - } - /* parse that follows the colon */ - if (mpl->token == T_COLON) - { get_token(mpl /* : */); - code = expression_13(mpl); - /* convert the expression to logical type, if necessary */ - if (code->type == A_SYMBOLIC) - code = make_unary(mpl, O_CVTNUM, code, A_NUMERIC, 0); - if (code->type == A_NUMERIC) - code = make_unary(mpl, O_CVTLOG, code, A_LOGICAL, 0); - /* now the expression must be of logical type */ - if (code->type != A_LOGICAL) - error(mpl, "expression following colon has invalid type"); - xassert(code->dim == 0); - domain->code = code; - /* the right brace must follow the logical expression */ - if (mpl->token != T_RBRACE) - error(mpl, "syntax error in indexing expression"); - } - get_token(mpl /* } */); - return domain; -} - -/*---------------------------------------------------------------------- --- close_scope - close scope of indexing expression. --- --- The routine closes the scope of indexing expression specified by its --- domain and thereby makes all dummy indices introduced in the indexing --- expression no longer available for referencing. */ - -void close_scope(MPL *mpl, DOMAIN *domain) -{ DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - AVLNODE *node; - xassert(domain != NULL); - /* remove all dummy indices from the symbolic names table */ - for (block = domain->list; block != NULL; block = block->next) - { for (slot = block->list; slot != NULL; slot = slot->next) - { if (slot->name != NULL) - { node = avl_find_node(mpl->tree, slot->name); - xassert(node != NULL); - xassert(avl_get_node_type(node) == A_INDEX); - avl_delete_node(mpl->tree, node); - } - } - } - return; -} - -/*---------------------------------------------------------------------- --- iterated_expression - parse iterated expression. --- --- This routine parses primary expression using the syntax: --- --- ::= --- ::= sum --- ::= prod --- ::= min --- ::= max --- ::= exists --- --- ::= forall --- --- ::= setof --- --- Note that parsing "integrand" depends on the iterated operator. */ - -#if 1 /* 07/IX-2008 */ -static void link_up(CODE *code) -{ /* if we have something like sum{(i+1,j,k-1) in E} x[i,j,k], - where i and k are dummy indices defined out of the iterated - expression, we should link up pseudo-code for computing i+1 - and k-1 to pseudo-code for computing the iterated expression; - this is needed to invalidate current value of the iterated - expression once i or k have been changed */ - DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - for (block = code->arg.loop.domain->list; block != NULL; - block = block->next) - { for (slot = block->list; slot != NULL; slot = slot->next) - { if (slot->code != NULL) - { xassert(slot->code->up == NULL); - slot->code->up = code; - } - } - } - return; -} -#endif - -CODE *iterated_expression(MPL *mpl) -{ CODE *code; - OPERANDS arg; - int op; - char opstr[8]; - /* determine operation code */ - xassert(mpl->token == T_NAME); - if (strcmp(mpl->image, "sum") == 0) - op = O_SUM; - else if (strcmp(mpl->image, "prod") == 0) - op = O_PROD; - else if (strcmp(mpl->image, "min") == 0) - op = O_MINIMUM; - else if (strcmp(mpl->image, "max") == 0) - op = O_MAXIMUM; - else if (strcmp(mpl->image, "forall") == 0) - op = O_FORALL; - else if (strcmp(mpl->image, "exists") == 0) - op = O_EXISTS; - else if (strcmp(mpl->image, "setof") == 0) - op = O_SETOF; - else - error(mpl, "operator %s unknown", mpl->image); - strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - get_token(mpl /* */); - /* check the left brace that follows the operator name */ - xassert(mpl->token == T_LBRACE); - /* parse indexing expression that controls iterating */ - arg.loop.domain = indexing_expression(mpl); - /* parse "integrand" expression and generate pseudo-code */ - switch (op) - { case O_SUM: - case O_PROD: - case O_MINIMUM: - case O_MAXIMUM: - arg.loop.x = expression_3(mpl); - /* convert the integrand to numeric type, if necessary */ - if (arg.loop.x->type == A_SYMBOLIC) - arg.loop.x = make_unary(mpl, O_CVTNUM, arg.loop.x, - A_NUMERIC, 0); - /* now the integrand must be of numeric type or linear form - (the latter is only allowed for the sum operator) */ - if (!(arg.loop.x->type == A_NUMERIC || - op == O_SUM && arg.loop.x->type == A_FORMULA)) -err: error(mpl, "integrand following %s{...} has invalid type" - , opstr); - xassert(arg.loop.x->dim == 0); - /* generate pseudo-code */ - code = make_code(mpl, op, &arg, arg.loop.x->type, 0); - break; - case O_FORALL: - case O_EXISTS: - arg.loop.x = expression_12(mpl); - /* convert the integrand to logical type, if necessary */ - if (arg.loop.x->type == A_SYMBOLIC) - arg.loop.x = make_unary(mpl, O_CVTNUM, arg.loop.x, - A_NUMERIC, 0); - if (arg.loop.x->type == A_NUMERIC) - arg.loop.x = make_unary(mpl, O_CVTLOG, arg.loop.x, - A_LOGICAL, 0); - /* now the integrand must be of logical type */ - if (arg.loop.x->type != A_LOGICAL) goto err; - xassert(arg.loop.x->dim == 0); - /* generate pseudo-code */ - code = make_code(mpl, op, &arg, A_LOGICAL, 0); - break; - case O_SETOF: - arg.loop.x = expression_5(mpl); - /* convert the integrand to 1-tuple, if necessary */ - if (arg.loop.x->type == A_NUMERIC) - arg.loop.x = make_unary(mpl, O_CVTSYM, arg.loop.x, - A_SYMBOLIC, 0); - if (arg.loop.x->type == A_SYMBOLIC) - arg.loop.x = make_unary(mpl, O_CVTTUP, arg.loop.x, - A_TUPLE, 1); - /* now the integrand must be n-tuple */ - if (arg.loop.x->type != A_TUPLE) goto err; - xassert(arg.loop.x->dim > 0); - /* generate pseudo-code */ - code = make_code(mpl, op, &arg, A_ELEMSET, arg.loop.x->dim); - break; - default: - xassert(op != op); - } - /* close the scope of the indexing expression */ - close_scope(mpl, arg.loop.domain); -#if 1 /* 07/IX-2008 */ - link_up(code); -#endif - return code; -} - -/*---------------------------------------------------------------------- --- domain_arity - determine arity of domain. --- --- This routine returns arity of specified domain, which is number of --- its free dummy indices. */ - -int domain_arity(MPL *mpl, DOMAIN *domain) -{ DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - int arity; - xassert(mpl == mpl); - arity = 0; - for (block = domain->list; block != NULL; block = block->next) - for (slot = block->list; slot != NULL; slot = slot->next) - if (slot->code == NULL) arity++; - return arity; -} - -/*---------------------------------------------------------------------- --- set_expression - parse set expression. --- --- This routine parses primary expression using the syntax: --- --- ::= { } --- ::= */ - -CODE *set_expression(MPL *mpl) -{ CODE *code; - OPERANDS arg; - xassert(mpl->token == T_LBRACE); - get_token(mpl /* { */); - /* check a token that follows the left brace */ - if (mpl->token == T_RBRACE) - { /* it is the right brace, so the resultant is an empty set of - dimension 1 */ - arg.list = NULL; - /* generate pseudo-code to build the resultant set */ - code = make_code(mpl, O_MAKE, &arg, A_ELEMSET, 1); - get_token(mpl /* } */); - } - else - { /* the next token begins an indexing expression */ - unget_token(mpl); - arg.loop.domain = indexing_expression(mpl); - arg.loop.x = NULL; /* integrand is not used */ - /* close the scope of the indexing expression */ - close_scope(mpl, arg.loop.domain); - /* generate pseudo-code to build the resultant set */ - code = make_code(mpl, O_BUILD, &arg, A_ELEMSET, - domain_arity(mpl, arg.loop.domain)); -#if 1 /* 07/IX-2008 */ - link_up(code); -#endif - } - return code; -} - -/*---------------------------------------------------------------------- --- branched_expression - parse conditional expression. --- --- This routine parses primary expression using the syntax: --- --- ::= --- ::= if then --- ::= if then --- else --- ::= */ - -CODE *branched_expression(MPL *mpl) -{ CODE *code, *x, *y, *z; - xassert(mpl->token == T_IF); - get_token(mpl /* if */); - /* parse that follows 'if' */ - x = expression_13(mpl); - /* convert the expression to logical type, if necessary */ - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0); - /* now the expression must be of logical type */ - if (x->type != A_LOGICAL) - error(mpl, "expression following if has invalid type"); - xassert(x->dim == 0); - /* the keyword 'then' must follow the logical expression */ - if (mpl->token != T_THEN) - error(mpl, "keyword then missing where expected"); - get_token(mpl /* then */); - /* parse that follows 'then' and check its type */ - y = expression_9(mpl); - if (!(y->type == A_NUMERIC || y->type == A_SYMBOLIC || - y->type == A_ELEMSET || y->type == A_FORMULA)) - error(mpl, "expression following then has invalid type"); - /* if the expression that follows the keyword 'then' is elemental - set, the keyword 'else' cannot be omitted; otherwise else-part - is optional */ - if (mpl->token != T_ELSE) - { if (y->type == A_ELEMSET) - error(mpl, "keyword else missing where expected"); - z = NULL; - goto skip; - } - get_token(mpl /* else */); - /* parse that follow 'else' and check its type */ - z = expression_9(mpl); - if (!(z->type == A_NUMERIC || z->type == A_SYMBOLIC || - z->type == A_ELEMSET || z->type == A_FORMULA)) - error(mpl, "expression following else has invalid type"); - /* convert to identical types, if necessary */ - if (y->type == A_FORMULA || z->type == A_FORMULA) - { if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTLFM, y, A_FORMULA, 0); - if (z->type == A_SYMBOLIC) - z = make_unary(mpl, O_CVTNUM, z, A_NUMERIC, 0); - if (z->type == A_NUMERIC) - z = make_unary(mpl, O_CVTLFM, z, A_FORMULA, 0); - } - if (y->type == A_SYMBOLIC || z->type == A_SYMBOLIC) - { if (y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTSYM, y, A_SYMBOLIC, 0); - if (z->type == A_NUMERIC) - z = make_unary(mpl, O_CVTSYM, z, A_SYMBOLIC, 0); - } - /* now both expressions must have identical types */ - if (y->type != z->type) - error(mpl, "expressions following then and else have incompati" - "ble types"); - /* and identical dimensions */ - if (y->dim != z->dim) - error(mpl, "expressions following then and else have different" - " dimensions %d and %d, respectively", y->dim, z->dim); -skip: /* generate pseudo-code to perform branching */ - code = make_ternary(mpl, O_FORK, x, y, z, y->type, y->dim); - return code; -} - -/*---------------------------------------------------------------------- --- primary_expression - parse primary expression. --- --- This routine parses primary expression using the syntax: --- --- ::= --- ::= Infinity --- ::= --- ::= --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- ::= ( ) --- ::= ( ) --- ::= --- ::= { } --- ::= --- ::= --- --- For complete list of syntactic rules for see --- comments to the corresponding parsing routines. */ - -CODE *primary_expression(MPL *mpl) -{ CODE *code; - if (mpl->token == T_NUMBER) - { /* parse numeric literal */ - code = numeric_literal(mpl); - } -#if 1 /* 21/VII-2006 */ - else if (mpl->token == T_INFINITY) - { /* parse "infinity" */ - OPERANDS arg; - arg.num = DBL_MAX; - code = make_code(mpl, O_NUMBER, &arg, A_NUMERIC, 0); - get_token(mpl /* Infinity */); - } -#endif - else if (mpl->token == T_STRING) - { /* parse string literal */ - code = string_literal(mpl); - } - else if (mpl->token == T_NAME) - { int next_token; - get_token(mpl /* */); - next_token = mpl->token; - unget_token(mpl); - /* check a token that follows */ - switch (next_token) - { case T_LBRACKET: - /* parse reference to subscripted object */ - code = object_reference(mpl); - break; - case T_LEFT: - /* parse reference to built-in function */ - code = function_reference(mpl); - break; - case T_LBRACE: - /* parse iterated expression */ - code = iterated_expression(mpl); - break; - default: - /* parse reference to unsubscripted object */ - code = object_reference(mpl); - break; - } - } - else if (mpl->token == T_LEFT) - { /* parse parenthesized expression */ - code = expression_list(mpl); - } - else if (mpl->token == T_LBRACE) - { /* parse set expression */ - code = set_expression(mpl); - } - else if (mpl->token == T_IF) - { /* parse conditional expression */ - code = branched_expression(mpl); - } - else if (is_reserved(mpl)) - { /* other reserved keywords cannot be used here */ - error(mpl, "invalid use of reserved keyword %s", mpl->image); - } - else - error(mpl, "syntax error in expression"); - return code; -} - -/*---------------------------------------------------------------------- --- error_preceding - raise error if preceding operand has wrong type. --- --- This routine is called to raise error if operand that precedes some --- infix operator has invalid type. */ - -void error_preceding(MPL *mpl, char *opstr) -{ error(mpl, "operand preceding %s has invalid type", opstr); - /* no return */ -} - -/*---------------------------------------------------------------------- --- error_following - raise error if following operand has wrong type. --- --- This routine is called to raise error if operand that follows some --- infix operator has invalid type. */ - -void error_following(MPL *mpl, char *opstr) -{ error(mpl, "operand following %s has invalid type", opstr); - /* no return */ -} - -/*---------------------------------------------------------------------- --- error_dimension - raise error if operands have different dimension. --- --- This routine is called to raise error if two operands of some infix --- operator have different dimension. */ - -void error_dimension(MPL *mpl, char *opstr, int dim1, int dim2) -{ error(mpl, "operands preceding and following %s have different di" - "mensions %d and %d, respectively", opstr, dim1, dim2); - /* no return */ -} - -/*---------------------------------------------------------------------- --- expression_0 - parse expression of level 0. --- --- This routine parses expression of level 0 using the syntax: --- --- ::= */ - -CODE *expression_0(MPL *mpl) -{ CODE *code; - code = primary_expression(mpl); - return code; -} - -/*---------------------------------------------------------------------- --- expression_1 - parse expression of level 1. --- --- This routine parses expression of level 1 using the syntax: --- --- ::= --- ::= --- ::= --- ::= ^ | ** */ - -CODE *expression_1(MPL *mpl) -{ CODE *x, *y; - char opstr[8]; - x = expression_0(mpl); - if (mpl->token == T_POWER) - { strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type != A_NUMERIC) - error_preceding(mpl, opstr); - get_token(mpl /* ^ | ** */); - if (mpl->token == T_PLUS || mpl->token == T_MINUS) - y = expression_2(mpl); - else - y = expression_1(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, opstr); - x = make_binary(mpl, O_POWER, x, y, A_NUMERIC, 0); - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_2 - parse expression of level 2. --- --- This routine parses expression of level 2 using the syntax: --- --- ::= --- ::= + --- ::= - */ - -CODE *expression_2(MPL *mpl) -{ CODE *x; - if (mpl->token == T_PLUS) - { get_token(mpl /* + */); - x = expression_1(mpl); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (!(x->type == A_NUMERIC || x->type == A_FORMULA)) - error_following(mpl, "+"); - x = make_unary(mpl, O_PLUS, x, x->type, 0); - } - else if (mpl->token == T_MINUS) - { get_token(mpl /* - */); - x = expression_1(mpl); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (!(x->type == A_NUMERIC || x->type == A_FORMULA)) - error_following(mpl, "-"); - x = make_unary(mpl, O_MINUS, x, x->type, 0); - } - else - x = expression_1(mpl); - return x; -} - -/*---------------------------------------------------------------------- --- expression_3 - parse expression of level 3. --- --- This routine parses expression of level 3 using the syntax: --- --- ::= --- ::= * --- ::= / --- ::= div --- ::= mod */ - -CODE *expression_3(MPL *mpl) -{ CODE *x, *y; - x = expression_2(mpl); - for (;;) - { if (mpl->token == T_ASTERISK) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (!(x->type == A_NUMERIC || x->type == A_FORMULA)) - error_preceding(mpl, "*"); - get_token(mpl /* * */); - y = expression_2(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (!(y->type == A_NUMERIC || y->type == A_FORMULA)) - error_following(mpl, "*"); - if (x->type == A_FORMULA && y->type == A_FORMULA) - error(mpl, "multiplication of linear forms not allowed"); - if (x->type == A_NUMERIC && y->type == A_NUMERIC) - x = make_binary(mpl, O_MUL, x, y, A_NUMERIC, 0); - else - x = make_binary(mpl, O_MUL, x, y, A_FORMULA, 0); - } - else if (mpl->token == T_SLASH) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (!(x->type == A_NUMERIC || x->type == A_FORMULA)) - error_preceding(mpl, "/"); - get_token(mpl /* / */); - y = expression_2(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, "/"); - if (x->type == A_NUMERIC) - x = make_binary(mpl, O_DIV, x, y, A_NUMERIC, 0); - else - x = make_binary(mpl, O_DIV, x, y, A_FORMULA, 0); - } - else if (mpl->token == T_DIV) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type != A_NUMERIC) - error_preceding(mpl, "div"); - get_token(mpl /* div */); - y = expression_2(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, "div"); - x = make_binary(mpl, O_IDIV, x, y, A_NUMERIC, 0); - } - else if (mpl->token == T_MOD) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type != A_NUMERIC) - error_preceding(mpl, "mod"); - get_token(mpl /* mod */); - y = expression_2(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, "mod"); - x = make_binary(mpl, O_MOD, x, y, A_NUMERIC, 0); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_4 - parse expression of level 4. --- --- This routine parses expression of level 4 using the syntax: --- --- ::= --- ::= + --- ::= - --- ::= less */ - -CODE *expression_4(MPL *mpl) -{ CODE *x, *y; - x = expression_3(mpl); - for (;;) - { if (mpl->token == T_PLUS) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (!(x->type == A_NUMERIC || x->type == A_FORMULA)) - error_preceding(mpl, "+"); - get_token(mpl /* + */); - y = expression_3(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (!(y->type == A_NUMERIC || y->type == A_FORMULA)) - error_following(mpl, "+"); - if (x->type == A_NUMERIC && y->type == A_FORMULA) - x = make_unary(mpl, O_CVTLFM, x, A_FORMULA, 0); - if (x->type == A_FORMULA && y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTLFM, y, A_FORMULA, 0); - x = make_binary(mpl, O_ADD, x, y, x->type, 0); - } - else if (mpl->token == T_MINUS) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (!(x->type == A_NUMERIC || x->type == A_FORMULA)) - error_preceding(mpl, "-"); - get_token(mpl /* - */); - y = expression_3(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (!(y->type == A_NUMERIC || y->type == A_FORMULA)) - error_following(mpl, "-"); - if (x->type == A_NUMERIC && y->type == A_FORMULA) - x = make_unary(mpl, O_CVTLFM, x, A_FORMULA, 0); - if (x->type == A_FORMULA && y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTLFM, y, A_FORMULA, 0); - x = make_binary(mpl, O_SUB, x, y, x->type, 0); - } - else if (mpl->token == T_LESS) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type != A_NUMERIC) - error_preceding(mpl, "less"); - get_token(mpl /* less */); - y = expression_3(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, "less"); - x = make_binary(mpl, O_LESS, x, y, A_NUMERIC, 0); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_5 - parse expression of level 5. --- --- This routine parses expression of level 5 using the syntax: --- --- ::= --- ::= & */ - -CODE *expression_5(MPL *mpl) -{ CODE *x, *y; - x = expression_4(mpl); - for (;;) - { if (mpl->token == T_CONCAT) - { if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0); - if (x->type != A_SYMBOLIC) - error_preceding(mpl, "&"); - get_token(mpl /* & */); - y = expression_4(mpl); - if (y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTSYM, y, A_SYMBOLIC, 0); - if (y->type != A_SYMBOLIC) - error_following(mpl, "&"); - x = make_binary(mpl, O_CONCAT, x, y, A_SYMBOLIC, 0); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_6 - parse expression of level 6. --- --- This routine parses expression of level 6 using the syntax: --- --- ::= --- ::= .. --- ::= .. by --- */ - -CODE *expression_6(MPL *mpl) -{ CODE *x, *y, *z; - x = expression_5(mpl); - if (mpl->token == T_DOTS) - { if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type != A_NUMERIC) - error_preceding(mpl, ".."); - get_token(mpl /* .. */); - y = expression_5(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, ".."); - if (mpl->token == T_BY) - { get_token(mpl /* by */); - z = expression_5(mpl); - if (z->type == A_SYMBOLIC) - z = make_unary(mpl, O_CVTNUM, z, A_NUMERIC, 0); - if (z->type != A_NUMERIC) - error_following(mpl, "by"); - } - else - z = NULL; - x = make_ternary(mpl, O_DOTS, x, y, z, A_ELEMSET, 1); - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_7 - parse expression of level 7. --- --- This routine parses expression of level 7 using the syntax: --- --- ::= --- ::= cross */ - -CODE *expression_7(MPL *mpl) -{ CODE *x, *y; - x = expression_6(mpl); - for (;;) - { if (mpl->token == T_CROSS) - { if (x->type != A_ELEMSET) - error_preceding(mpl, "cross"); - get_token(mpl /* cross */); - y = expression_6(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, "cross"); - x = make_binary(mpl, O_CROSS, x, y, A_ELEMSET, - x->dim + y->dim); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_8 - parse expression of level 8. --- --- This routine parses expression of level 8 using the syntax: --- --- ::= --- ::= inter */ - -CODE *expression_8(MPL *mpl) -{ CODE *x, *y; - x = expression_7(mpl); - for (;;) - { if (mpl->token == T_INTER) - { if (x->type != A_ELEMSET) - error_preceding(mpl, "inter"); - get_token(mpl /* inter */); - y = expression_7(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, "inter"); - if (x->dim != y->dim) - error_dimension(mpl, "inter", x->dim, y->dim); - x = make_binary(mpl, O_INTER, x, y, A_ELEMSET, x->dim); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_9 - parse expression of level 9. --- --- This routine parses expression of level 9 using the syntax: --- --- ::= --- ::= union --- ::= diff --- ::= symdiff */ - -CODE *expression_9(MPL *mpl) -{ CODE *x, *y; - x = expression_8(mpl); - for (;;) - { if (mpl->token == T_UNION) - { if (x->type != A_ELEMSET) - error_preceding(mpl, "union"); - get_token(mpl /* union */); - y = expression_8(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, "union"); - if (x->dim != y->dim) - error_dimension(mpl, "union", x->dim, y->dim); - x = make_binary(mpl, O_UNION, x, y, A_ELEMSET, x->dim); - } - else if (mpl->token == T_DIFF) - { if (x->type != A_ELEMSET) - error_preceding(mpl, "diff"); - get_token(mpl /* diff */); - y = expression_8(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, "diff"); - if (x->dim != y->dim) - error_dimension(mpl, "diff", x->dim, y->dim); - x = make_binary(mpl, O_DIFF, x, y, A_ELEMSET, x->dim); - } - else if (mpl->token == T_SYMDIFF) - { if (x->type != A_ELEMSET) - error_preceding(mpl, "symdiff"); - get_token(mpl /* symdiff */); - y = expression_8(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, "symdiff"); - if (x->dim != y->dim) - error_dimension(mpl, "symdiff", x->dim, y->dim); - x = make_binary(mpl, O_SYMDIFF, x, y, A_ELEMSET, x->dim); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_10 - parse expression of level 10. --- --- This routine parses expression of level 10 using the syntax: --- --- ::= --- ::= --- ::= < | <= | = | == | >= | > | <> | != | in | not in | ! in | --- within | not within | ! within */ - -CODE *expression_10(MPL *mpl) -{ CODE *x, *y; - int op = -1; - char opstr[16]; - x = expression_9(mpl); - strcpy(opstr, ""); - switch (mpl->token) - { case T_LT: - op = O_LT; break; - case T_LE: - op = O_LE; break; - case T_EQ: - op = O_EQ; break; - case T_GE: - op = O_GE; break; - case T_GT: - op = O_GT; break; - case T_NE: - op = O_NE; break; - case T_IN: - op = O_IN; break; - case T_WITHIN: - op = O_WITHIN; break; - case T_NOT: - strcpy(opstr, mpl->image); - get_token(mpl /* not | ! */); - if (mpl->token == T_IN) - op = O_NOTIN; - else if (mpl->token == T_WITHIN) - op = O_NOTWITHIN; - else - error(mpl, "invalid use of %s", opstr); - strcat(opstr, " "); - break; - default: - goto done; - } - strcat(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - switch (op) - { case O_EQ: - case O_NE: -#if 1 /* 02/VIII-2008 */ - case O_LT: - case O_LE: - case O_GT: - case O_GE: -#endif - if (!(x->type == A_NUMERIC || x->type == A_SYMBOLIC)) - error_preceding(mpl, opstr); - get_token(mpl /* */); - y = expression_9(mpl); - if (!(y->type == A_NUMERIC || y->type == A_SYMBOLIC)) - error_following(mpl, opstr); - if (x->type == A_NUMERIC && y->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0); - if (x->type == A_SYMBOLIC && y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTSYM, y, A_SYMBOLIC, 0); - x = make_binary(mpl, op, x, y, A_LOGICAL, 0); - break; -#if 0 /* 02/VIII-2008 */ - case O_LT: - case O_LE: - case O_GT: - case O_GE: - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type != A_NUMERIC) - error_preceding(mpl, opstr); - get_token(mpl /* */); - y = expression_9(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type != A_NUMERIC) - error_following(mpl, opstr); - x = make_binary(mpl, op, x, y, A_LOGICAL, 0); - break; -#endif - case O_IN: - case O_NOTIN: - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTSYM, x, A_SYMBOLIC, 0); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTTUP, x, A_TUPLE, 1); - if (x->type != A_TUPLE) - error_preceding(mpl, opstr); - get_token(mpl /* */); - y = expression_9(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, opstr); - if (x->dim != y->dim) - error_dimension(mpl, opstr, x->dim, y->dim); - x = make_binary(mpl, op, x, y, A_LOGICAL, 0); - break; - case O_WITHIN: - case O_NOTWITHIN: - if (x->type != A_ELEMSET) - error_preceding(mpl, opstr); - get_token(mpl /* */); - y = expression_9(mpl); - if (y->type != A_ELEMSET) - error_following(mpl, opstr); - if (x->dim != y->dim) - error_dimension(mpl, opstr, x->dim, y->dim); - x = make_binary(mpl, op, x, y, A_LOGICAL, 0); - break; - default: - xassert(op != op); - } -done: return x; -} - -/*---------------------------------------------------------------------- --- expression_11 - parse expression of level 11. --- --- This routine parses expression of level 11 using the syntax: --- --- ::= --- ::= not --- ::= ! */ - -CODE *expression_11(MPL *mpl) -{ CODE *x; - char opstr[8]; - if (mpl->token == T_NOT) - { strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - get_token(mpl /* not | ! */); - x = expression_10(mpl); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0); - if (x->type != A_LOGICAL) - error_following(mpl, opstr); - x = make_unary(mpl, O_NOT, x, A_LOGICAL, 0); - } - else - x = expression_10(mpl); - return x; -} - -/*---------------------------------------------------------------------- --- expression_12 - parse expression of level 12. --- --- This routine parses expression of level 12 using the syntax: --- --- ::= --- ::= and --- ::= && */ - -CODE *expression_12(MPL *mpl) -{ CODE *x, *y; - char opstr[8]; - x = expression_11(mpl); - for (;;) - { if (mpl->token == T_AND) - { strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0); - if (x->type != A_LOGICAL) - error_preceding(mpl, opstr); - get_token(mpl /* and | && */); - y = expression_11(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTLOG, y, A_LOGICAL, 0); - if (y->type != A_LOGICAL) - error_following(mpl, opstr); - x = make_binary(mpl, O_AND, x, y, A_LOGICAL, 0); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- expression_13 - parse expression of level 13. --- --- This routine parses expression of level 13 using the syntax: --- --- ::= --- ::= or --- ::= || */ - -CODE *expression_13(MPL *mpl) -{ CODE *x, *y; - char opstr[8]; - x = expression_12(mpl); - for (;;) - { if (mpl->token == T_OR) - { strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - if (x->type == A_SYMBOLIC) - x = make_unary(mpl, O_CVTNUM, x, A_NUMERIC, 0); - if (x->type == A_NUMERIC) - x = make_unary(mpl, O_CVTLOG, x, A_LOGICAL, 0); - if (x->type != A_LOGICAL) - error_preceding(mpl, opstr); - get_token(mpl /* or | || */); - y = expression_12(mpl); - if (y->type == A_SYMBOLIC) - y = make_unary(mpl, O_CVTNUM, y, A_NUMERIC, 0); - if (y->type == A_NUMERIC) - y = make_unary(mpl, O_CVTLOG, y, A_LOGICAL, 0); - if (y->type != A_LOGICAL) - error_following(mpl, opstr); - x = make_binary(mpl, O_OR, x, y, A_LOGICAL, 0); - } - else - break; - } - return x; -} - -/*---------------------------------------------------------------------- --- set_statement - parse set statement. --- --- This routine parses set statement using the syntax: --- --- ::= set --- ; --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= , dimen --- ::= , within --- ::= , := --- ::= , default --- --- Commae in are optional and may be omitted anywhere. */ - -SET *set_statement(MPL *mpl) -{ SET *set; - int dimen_used = 0; - xassert(is_keyword(mpl, "set")); - get_token(mpl /* set */); - /* symbolic name must follow the keyword 'set' */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "symbolic name missing where expected"); - /* there must be no other object with the same name */ - if (avl_find_node(mpl->tree, mpl->image) != NULL) - error(mpl, "%s multiply declared", mpl->image); - /* create model set */ - set = alloc(SET); - set->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(set->name, mpl->image); - set->alias = NULL; - set->dim = 0; - set->domain = NULL; - set->dimen = 0; - set->within = NULL; - set->assign = NULL; - set->option = NULL; - set->gadget = NULL; - set->data = 0; - set->array = NULL; - get_token(mpl /* */); - /* parse optional alias */ - if (mpl->token == T_STRING) - { set->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(set->alias, mpl->image); - get_token(mpl /* */); - } - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { set->domain = indexing_expression(mpl); - set->dim = domain_arity(mpl, set->domain); - } - /* include the set name in the symbolic names table */ - { AVLNODE *node; - node = avl_insert_node(mpl->tree, set->name); - avl_set_node_type(node, A_SET); - avl_set_node_link(node, (void *)set); - } - /* parse the list of optional attributes */ - for (;;) - { if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_SEMICOLON) - break; - if (is_keyword(mpl, "dimen")) - { /* dimension of set members */ - int dimen; - get_token(mpl /* dimen */); - if (!(mpl->token == T_NUMBER && - 1.0 <= mpl->value && mpl->value <= 20.0 && - floor(mpl->value) == mpl->value)) - error(mpl, "dimension must be integer between 1 and 20"); - dimen = (int)(mpl->value + 0.5); - if (dimen_used) - error(mpl, "at most one dimension attribute allowed"); - if (set->dimen > 0) - error(mpl, "dimension %d conflicts with dimension %d alr" - "eady determined", dimen, set->dimen); - set->dimen = dimen; - dimen_used = 1; - get_token(mpl /* */); - } - else if (mpl->token == T_WITHIN || mpl->token == T_IN) - { /* restricting superset */ - WITHIN *within, *temp; - if (mpl->token == T_IN && !mpl->as_within) - { warning(mpl, "keyword in understood as within"); - mpl->as_within = 1; - } - get_token(mpl /* within */); - /* create new restricting superset list entry and append it - to the within-list */ - within = alloc(WITHIN); - within->code = NULL; - within->next = NULL; - if (set->within == NULL) - set->within = within; - else - { for (temp = set->within; temp->next != NULL; temp = - temp->next); - temp->next = within; - } - /* parse an expression that follows 'within' */ - within->code = expression_9(mpl); - if (within->code->type != A_ELEMSET) - error(mpl, "expression following within has invalid type" - ); - xassert(within->code->dim > 0); - /* check/set dimension of set members */ - if (set->dimen == 0) set->dimen = within->code->dim; - if (set->dimen != within->code->dim) - error(mpl, "set expression following within must have di" - "mension %d rather than %d", - set->dimen, within->code->dim); - } - else if (mpl->token == T_ASSIGN) - { /* assignment expression */ - if (!(set->assign == NULL && set->option == NULL && - set->gadget == NULL)) -err: error(mpl, "at most one := or default/data allowed"); - get_token(mpl /* := */); - /* parse an expression that follows ':=' */ - set->assign = expression_9(mpl); - if (set->assign->type != A_ELEMSET) - error(mpl, "expression following := has invalid type"); - xassert(set->assign->dim > 0); - /* check/set dimension of set members */ - if (set->dimen == 0) set->dimen = set->assign->dim; - if (set->dimen != set->assign->dim) - error(mpl, "set expression following := must have dimens" - "ion %d rather than %d", - set->dimen, set->assign->dim); - } - else if (is_keyword(mpl, "default")) - { /* expression for default value */ - if (!(set->assign == NULL && set->option == NULL)) goto err; - get_token(mpl /* := */); - /* parse an expression that follows 'default' */ - set->option = expression_9(mpl); - if (set->option->type != A_ELEMSET) - error(mpl, "expression following default has invalid typ" - "e"); - xassert(set->option->dim > 0); - /* check/set dimension of set members */ - if (set->dimen == 0) set->dimen = set->option->dim; - if (set->dimen != set->option->dim) - error(mpl, "set expression following default must have d" - "imension %d rather than %d", - set->dimen, set->option->dim); - } -#if 1 /* 12/XII-2008 */ - else if (is_keyword(mpl, "data")) - { /* gadget to initialize the set by data from plain set */ - GADGET *gadget; - AVLNODE *node; - int i, k, fff[20]; - if (!(set->assign == NULL && set->gadget == NULL)) goto err; - get_token(mpl /* data */); - set->gadget = gadget = alloc(GADGET); - /* set name must follow the keyword 'data' */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", - mpl->image); - else - error(mpl, "set name missing where expected"); - /* find the set in the symbolic name table */ - node = avl_find_node(mpl->tree, mpl->image); - if (node == NULL) - error(mpl, "%s not defined", mpl->image); - if (avl_get_node_type(node) != A_SET) -err1: error(mpl, "%s not a plain set", mpl->image); - gadget->set = avl_get_node_link(node); - if (gadget->set->dim != 0) goto err1; - if (gadget->set == set) - error(mpl, "set cannot be initialized by itself"); - /* check and set dimensions */ - if (set->dim >= gadget->set->dimen) -err2: error(mpl, "dimension of %s too small", mpl->image); - if (set->dimen == 0) - set->dimen = gadget->set->dimen - set->dim; - if (set->dim + set->dimen > gadget->set->dimen) - goto err2; - else if (set->dim + set->dimen < gadget->set->dimen) - error(mpl, "dimension of %s too big", mpl->image); - get_token(mpl /* set name */); - /* left parenthesis must follow the set name */ - if (mpl->token == T_LEFT) - get_token(mpl /* ( */); - else - error(mpl, "left parenthesis missing where expected"); - /* parse permutation of component numbers */ - for (k = 0; k < gadget->set->dimen; k++) fff[k] = 0; - k = 0; - for (;;) - { if (mpl->token != T_NUMBER) - error(mpl, "component number missing where expected"); - if (str2int(mpl->image, &i) != 0) -err3: error(mpl, "component number must be integer between " - "1 and %d", gadget->set->dimen); - if (!(1 <= i && i <= gadget->set->dimen)) goto err3; - if (fff[i-1] != 0) - error(mpl, "component %d multiply specified", i); - gadget->ind[k++] = i, fff[i-1] = 1; - xassert(k <= gadget->set->dimen); - get_token(mpl /* number */); - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RIGHT) - break; - else - error(mpl, "syntax error in data attribute"); - } - if (k < gadget->set->dimen) - error(mpl, "there are must be %d components rather than " - "%d", gadget->set->dimen, k); - get_token(mpl /* ) */); - } -#endif - else - error(mpl, "syntax error in set statement"); - } - /* close the domain scope */ - if (set->domain != NULL) close_scope(mpl, set->domain); - /* if dimension of set members is still unknown, set it to 1 */ - if (set->dimen == 0) set->dimen = 1; - /* the set statement has been completely parsed */ - xassert(mpl->token == T_SEMICOLON); - get_token(mpl /* ; */); - return set; -} - -/*---------------------------------------------------------------------- --- parameter_statement - parse parameter statement. --- --- This routine parses parameter statement using the syntax: --- --- ::= param --- ; --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= , integer --- ::= , binary --- ::= , symbolic --- ::= , --- ::= , in --- ::= , := --- ::= , default --- ::= < | <= | = | == | >= | > | <> | != --- --- Commae in are optional and may be omitted anywhere. */ - -PARAMETER *parameter_statement(MPL *mpl) -{ PARAMETER *par; - int integer_used = 0, binary_used = 0, symbolic_used = 0; - xassert(is_keyword(mpl, "param")); - get_token(mpl /* param */); - /* symbolic name must follow the keyword 'param' */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "symbolic name missing where expected"); - /* there must be no other object with the same name */ - if (avl_find_node(mpl->tree, mpl->image) != NULL) - error(mpl, "%s multiply declared", mpl->image); - /* create model parameter */ - par = alloc(PARAMETER); - par->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(par->name, mpl->image); - par->alias = NULL; - par->dim = 0; - par->domain = NULL; - par->type = A_NUMERIC; - par->cond = NULL; - par->in = NULL; - par->assign = NULL; - par->option = NULL; - par->data = 0; - par->defval = NULL; - par->array = NULL; - get_token(mpl /* */); - /* parse optional alias */ - if (mpl->token == T_STRING) - { par->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(par->alias, mpl->image); - get_token(mpl /* */); - } - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { par->domain = indexing_expression(mpl); - par->dim = domain_arity(mpl, par->domain); - } - /* include the parameter name in the symbolic names table */ - { AVLNODE *node; - node = avl_insert_node(mpl->tree, par->name); - avl_set_node_type(node, A_PARAMETER); - avl_set_node_link(node, (void *)par); - } - /* parse the list of optional attributes */ - for (;;) - { if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_SEMICOLON) - break; - if (is_keyword(mpl, "integer")) - { if (integer_used) - error(mpl, "at most one integer allowed"); - if (par->type == A_SYMBOLIC) - error(mpl, "symbolic parameter cannot be integer"); - if (par->type != A_BINARY) par->type = A_INTEGER; - integer_used = 1; - get_token(mpl /* integer */); - } - else if (is_keyword(mpl, "binary")) -bin: { if (binary_used) - error(mpl, "at most one binary allowed"); - if (par->type == A_SYMBOLIC) - error(mpl, "symbolic parameter cannot be binary"); - par->type = A_BINARY; - binary_used = 1; - get_token(mpl /* binary */); - } - else if (is_keyword(mpl, "logical")) - { if (!mpl->as_binary) - { warning(mpl, "keyword logical understood as binary"); - mpl->as_binary = 1; - } - goto bin; - } - else if (is_keyword(mpl, "symbolic")) - { if (symbolic_used) - error(mpl, "at most one symbolic allowed"); - if (par->type != A_NUMERIC) - error(mpl, "integer or binary parameter cannot be symbol" - "ic"); - /* the parameter may be referenced from expressions given - in the same parameter declaration, so its type must be - completed before parsing that expressions */ - if (!(par->cond == NULL && par->in == NULL && - par->assign == NULL && par->option == NULL)) - error(mpl, "keyword symbolic must precede any other para" - "meter attributes"); - par->type = A_SYMBOLIC; - symbolic_used = 1; - get_token(mpl /* symbolic */); - } - else if (mpl->token == T_LT || mpl->token == T_LE || - mpl->token == T_EQ || mpl->token == T_GE || - mpl->token == T_GT || mpl->token == T_NE) - { /* restricting condition */ - CONDITION *cond, *temp; - char opstr[8]; - /* create new restricting condition list entry and append - it to the conditions list */ - cond = alloc(CONDITION); - switch (mpl->token) - { case T_LT: - cond->rho = O_LT, strcpy(opstr, mpl->image); break; - case T_LE: - cond->rho = O_LE, strcpy(opstr, mpl->image); break; - case T_EQ: - cond->rho = O_EQ, strcpy(opstr, mpl->image); break; - case T_GE: - cond->rho = O_GE, strcpy(opstr, mpl->image); break; - case T_GT: - cond->rho = O_GT, strcpy(opstr, mpl->image); break; - case T_NE: - cond->rho = O_NE, strcpy(opstr, mpl->image); break; - default: - xassert(mpl->token != mpl->token); - } - xassert(strlen(opstr) < sizeof(opstr)); - cond->code = NULL; - cond->next = NULL; - if (par->cond == NULL) - par->cond = cond; - else - { for (temp = par->cond; temp->next != NULL; temp = - temp->next); - temp->next = cond; - } -#if 0 /* 13/VIII-2008 */ - if (par->type == A_SYMBOLIC && - !(cond->rho == O_EQ || cond->rho == O_NE)) - error(mpl, "inequality restriction not allowed"); -#endif - get_token(mpl /* rho */); - /* parse an expression that follows relational operator */ - cond->code = expression_5(mpl); - if (!(cond->code->type == A_NUMERIC || - cond->code->type == A_SYMBOLIC)) - error(mpl, "expression following %s has invalid type", - opstr); - xassert(cond->code->dim == 0); - /* convert to the parameter type, if necessary */ - if (par->type != A_SYMBOLIC && cond->code->type == - A_SYMBOLIC) - cond->code = make_unary(mpl, O_CVTNUM, cond->code, - A_NUMERIC, 0); - if (par->type == A_SYMBOLIC && cond->code->type != - A_SYMBOLIC) - cond->code = make_unary(mpl, O_CVTSYM, cond->code, - A_SYMBOLIC, 0); - } - else if (mpl->token == T_IN || mpl->token == T_WITHIN) - { /* restricting superset */ - WITHIN *in, *temp; - if (mpl->token == T_WITHIN && !mpl->as_in) - { warning(mpl, "keyword within understood as in"); - mpl->as_in = 1; - } - get_token(mpl /* in */); - /* create new restricting superset list entry and append it - to the in-list */ - in = alloc(WITHIN); - in->code = NULL; - in->next = NULL; - if (par->in == NULL) - par->in = in; - else - { for (temp = par->in; temp->next != NULL; temp = - temp->next); - temp->next = in; - } - /* parse an expression that follows 'in' */ - in->code = expression_9(mpl); - if (in->code->type != A_ELEMSET) - error(mpl, "expression following in has invalid type"); - xassert(in->code->dim > 0); - if (in->code->dim != 1) - error(mpl, "set expression following in must have dimens" - "ion 1 rather than %d", in->code->dim); - } - else if (mpl->token == T_ASSIGN) - { /* assignment expression */ - if (!(par->assign == NULL && par->option == NULL)) -err: error(mpl, "at most one := or default allowed"); - get_token(mpl /* := */); - /* parse an expression that follows ':=' */ - par->assign = expression_5(mpl); - /* the expression must be of numeric/symbolic type */ - if (!(par->assign->type == A_NUMERIC || - par->assign->type == A_SYMBOLIC)) - error(mpl, "expression following := has invalid type"); - xassert(par->assign->dim == 0); - /* convert to the parameter type, if necessary */ - if (par->type != A_SYMBOLIC && par->assign->type == - A_SYMBOLIC) - par->assign = make_unary(mpl, O_CVTNUM, par->assign, - A_NUMERIC, 0); - if (par->type == A_SYMBOLIC && par->assign->type != - A_SYMBOLIC) - par->assign = make_unary(mpl, O_CVTSYM, par->assign, - A_SYMBOLIC, 0); - } - else if (is_keyword(mpl, "default")) - { /* expression for default value */ - if (!(par->assign == NULL && par->option == NULL)) goto err; - get_token(mpl /* default */); - /* parse an expression that follows 'default' */ - par->option = expression_5(mpl); - if (!(par->option->type == A_NUMERIC || - par->option->type == A_SYMBOLIC)) - error(mpl, "expression following default has invalid typ" - "e"); - xassert(par->option->dim == 0); - /* convert to the parameter type, if necessary */ - if (par->type != A_SYMBOLIC && par->option->type == - A_SYMBOLIC) - par->option = make_unary(mpl, O_CVTNUM, par->option, - A_NUMERIC, 0); - if (par->type == A_SYMBOLIC && par->option->type != - A_SYMBOLIC) - par->option = make_unary(mpl, O_CVTSYM, par->option, - A_SYMBOLIC, 0); - } - else - error(mpl, "syntax error in parameter statement"); - } - /* close the domain scope */ - if (par->domain != NULL) close_scope(mpl, par->domain); - /* the parameter statement has been completely parsed */ - xassert(mpl->token == T_SEMICOLON); - get_token(mpl /* ; */); - return par; -} - -/*---------------------------------------------------------------------- --- variable_statement - parse variable statement. --- --- This routine parses variable statement using the syntax: --- --- ::= var --- ; --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= , integer --- ::= , binary --- ::= , --- ::= >= | <= | = | == --- --- Commae in are optional and may be omitted anywhere. */ - -VARIABLE *variable_statement(MPL *mpl) -{ VARIABLE *var; - int integer_used = 0, binary_used = 0; - xassert(is_keyword(mpl, "var")); - if (mpl->flag_s) - error(mpl, "variable statement must precede solve statement"); - get_token(mpl /* var */); - /* symbolic name must follow the keyword 'var' */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "symbolic name missing where expected"); - /* there must be no other object with the same name */ - if (avl_find_node(mpl->tree, mpl->image) != NULL) - error(mpl, "%s multiply declared", mpl->image); - /* create model variable */ - var = alloc(VARIABLE); - var->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(var->name, mpl->image); - var->alias = NULL; - var->dim = 0; - var->domain = NULL; - var->type = A_NUMERIC; - var->lbnd = NULL; - var->ubnd = NULL; - var->array = NULL; - get_token(mpl /* */); - /* parse optional alias */ - if (mpl->token == T_STRING) - { var->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(var->alias, mpl->image); - get_token(mpl /* */); - } - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { var->domain = indexing_expression(mpl); - var->dim = domain_arity(mpl, var->domain); - } - /* include the variable name in the symbolic names table */ - { AVLNODE *node; - node = avl_insert_node(mpl->tree, var->name); - avl_set_node_type(node, A_VARIABLE); - avl_set_node_link(node, (void *)var); - } - /* parse the list of optional attributes */ - for (;;) - { if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_SEMICOLON) - break; - if (is_keyword(mpl, "integer")) - { if (integer_used) - error(mpl, "at most one integer allowed"); - if (var->type != A_BINARY) var->type = A_INTEGER; - integer_used = 1; - get_token(mpl /* integer */); - } - else if (is_keyword(mpl, "binary")) -bin: { if (binary_used) - error(mpl, "at most one binary allowed"); - var->type = A_BINARY; - binary_used = 1; - get_token(mpl /* binary */); - } - else if (is_keyword(mpl, "logical")) - { if (!mpl->as_binary) - { warning(mpl, "keyword logical understood as binary"); - mpl->as_binary = 1; - } - goto bin; - } - else if (is_keyword(mpl, "symbolic")) - error(mpl, "variable cannot be symbolic"); - else if (mpl->token == T_GE) - { /* lower bound */ - if (var->lbnd != NULL) - { if (var->lbnd == var->ubnd) - error(mpl, "both fixed value and lower bound not allo" - "wed"); - else - error(mpl, "at most one lower bound allowed"); - } - get_token(mpl /* >= */); - /* parse an expression that specifies the lower bound */ - var->lbnd = expression_5(mpl); - if (var->lbnd->type == A_SYMBOLIC) - var->lbnd = make_unary(mpl, O_CVTNUM, var->lbnd, - A_NUMERIC, 0); - if (var->lbnd->type != A_NUMERIC) - error(mpl, "expression following >= has invalid type"); - xassert(var->lbnd->dim == 0); - } - else if (mpl->token == T_LE) - { /* upper bound */ - if (var->ubnd != NULL) - { if (var->ubnd == var->lbnd) - error(mpl, "both fixed value and upper bound not allo" - "wed"); - else - error(mpl, "at most one upper bound allowed"); - } - get_token(mpl /* <= */); - /* parse an expression that specifies the upper bound */ - var->ubnd = expression_5(mpl); - if (var->ubnd->type == A_SYMBOLIC) - var->ubnd = make_unary(mpl, O_CVTNUM, var->ubnd, - A_NUMERIC, 0); - if (var->ubnd->type != A_NUMERIC) - error(mpl, "expression following <= has invalid type"); - xassert(var->ubnd->dim == 0); - } - else if (mpl->token == T_EQ) - { /* fixed value */ - char opstr[8]; - if (!(var->lbnd == NULL && var->ubnd == NULL)) - { if (var->lbnd == var->ubnd) - error(mpl, "at most one fixed value allowed"); - else if (var->lbnd != NULL) - error(mpl, "both lower bound and fixed value not allo" - "wed"); - else - error(mpl, "both upper bound and fixed value not allo" - "wed"); - } - strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - get_token(mpl /* = | == */); - /* parse an expression that specifies the fixed value */ - var->lbnd = expression_5(mpl); - if (var->lbnd->type == A_SYMBOLIC) - var->lbnd = make_unary(mpl, O_CVTNUM, var->lbnd, - A_NUMERIC, 0); - if (var->lbnd->type != A_NUMERIC) - error(mpl, "expression following %s has invalid type", - opstr); - xassert(var->lbnd->dim == 0); - /* indicate that the variable is fixed, not bounded */ - var->ubnd = var->lbnd; - } - else if (mpl->token == T_LT || mpl->token == T_GT || - mpl->token == T_NE) - error(mpl, "strict bound not allowed"); - else - error(mpl, "syntax error in variable statement"); - } - /* close the domain scope */ - if (var->domain != NULL) close_scope(mpl, var->domain); - /* the variable statement has been completely parsed */ - xassert(mpl->token == T_SEMICOLON); - get_token(mpl /* ; */); - return var; -} - -/*---------------------------------------------------------------------- --- constraint_statement - parse constraint statement. --- --- This routine parses constraint statement using the syntax: --- --- ::= --- : ; --- ::= --- ::= subject to --- ::= subj to --- ::= s.t. --- ::= --- ::= --- ::= --- ::= --- ::= , >= --- ::= , <= --- ::= , = --- ::= , <= , <= --- ::= , >= , >= --- ::= --- --- Commae in are optional and may be omitted anywhere. */ - -CONSTRAINT *constraint_statement(MPL *mpl) -{ CONSTRAINT *con; - CODE *first, *second, *third; - int rho; - char opstr[8]; - if (mpl->flag_s) - error(mpl, "constraint statement must precede solve statement") - ; - if (is_keyword(mpl, "subject")) - { get_token(mpl /* subject */); - if (!is_keyword(mpl, "to")) - error(mpl, "keyword subject to incomplete"); - get_token(mpl /* to */); - } - else if (is_keyword(mpl, "subj")) - { get_token(mpl /* subj */); - if (!is_keyword(mpl, "to")) - error(mpl, "keyword subj to incomplete"); - get_token(mpl /* to */); - } - else if (mpl->token == T_SPTP) - get_token(mpl /* s.t. */); - /* the current token must be symbolic name of constraint */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "symbolic name missing where expected"); - /* there must be no other object with the same name */ - if (avl_find_node(mpl->tree, mpl->image) != NULL) - error(mpl, "%s multiply declared", mpl->image); - /* create model constraint */ - con = alloc(CONSTRAINT); - con->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(con->name, mpl->image); - con->alias = NULL; - con->dim = 0; - con->domain = NULL; - con->type = A_CONSTRAINT; - con->code = NULL; - con->lbnd = NULL; - con->ubnd = NULL; - con->array = NULL; - get_token(mpl /* */); - /* parse optional alias */ - if (mpl->token == T_STRING) - { con->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(con->alias, mpl->image); - get_token(mpl /* */); - } - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { con->domain = indexing_expression(mpl); - con->dim = domain_arity(mpl, con->domain); - } - /* include the constraint name in the symbolic names table */ - { AVLNODE *node; - node = avl_insert_node(mpl->tree, con->name); - avl_set_node_type(node, A_CONSTRAINT); - avl_set_node_link(node, (void *)con); - } - /* the colon must precede the first expression */ - if (mpl->token != T_COLON) - error(mpl, "colon missing where expected"); - get_token(mpl /* : */); - /* parse the first expression */ - first = expression_5(mpl); - if (first->type == A_SYMBOLIC) - first = make_unary(mpl, O_CVTNUM, first, A_NUMERIC, 0); - if (!(first->type == A_NUMERIC || first->type == A_FORMULA)) - error(mpl, "expression following colon has invalid type"); - xassert(first->dim == 0); - /* relational operator must follow the first expression */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - switch (mpl->token) - { case T_LE: - case T_GE: - case T_EQ: - break; - case T_LT: - case T_GT: - case T_NE: - error(mpl, "strict inequality not allowed"); - case T_SEMICOLON: - error(mpl, "constraint must be equality or inequality"); - default: - goto err; - } - rho = mpl->token; - strcpy(opstr, mpl->image); - xassert(strlen(opstr) < sizeof(opstr)); - get_token(mpl /* rho */); - /* parse the second expression */ - second = expression_5(mpl); - if (second->type == A_SYMBOLIC) - second = make_unary(mpl, O_CVTNUM, second, A_NUMERIC, 0); - if (!(second->type == A_NUMERIC || second->type == A_FORMULA)) - error(mpl, "expression following %s has invalid type", opstr); - xassert(second->dim == 0); - /* check a token that follow the second expression */ - if (mpl->token == T_COMMA) - { get_token(mpl /* , */); - if (mpl->token == T_SEMICOLON) goto err; - } - if (mpl->token == T_LT || mpl->token == T_LE || - mpl->token == T_EQ || mpl->token == T_GE || - mpl->token == T_GT || mpl->token == T_NE) - { /* it is another relational operator, therefore the constraint - is double inequality */ - if (rho == T_EQ || mpl->token != rho) - error(mpl, "double inequality must be ... <= ... <= ... or " - "... >= ... >= ..."); - /* the first expression cannot be linear form */ - if (first->type == A_FORMULA) - error(mpl, "leftmost expression in double inequality cannot" - " be linear form"); - get_token(mpl /* rho */); - /* parse the third expression */ - third = expression_5(mpl); - if (third->type == A_SYMBOLIC) - third = make_unary(mpl, O_CVTNUM, second, A_NUMERIC, 0); - if (!(third->type == A_NUMERIC || third->type == A_FORMULA)) - error(mpl, "rightmost expression in double inequality const" - "raint has invalid type"); - xassert(third->dim == 0); - /* the third expression also cannot be linear form */ - if (third->type == A_FORMULA) - error(mpl, "rightmost expression in double inequality canno" - "t be linear form"); - } - else - { /* the constraint is equality or single inequality */ - third = NULL; - } - /* close the domain scope */ - if (con->domain != NULL) close_scope(mpl, con->domain); - /* convert all expressions to linear form, if necessary */ - if (first->type != A_FORMULA) - first = make_unary(mpl, O_CVTLFM, first, A_FORMULA, 0); - if (second->type != A_FORMULA) - second = make_unary(mpl, O_CVTLFM, second, A_FORMULA, 0); - if (third != NULL) - third = make_unary(mpl, O_CVTLFM, third, A_FORMULA, 0); - /* arrange expressions in the constraint */ - if (third == NULL) - { /* the constraint is equality or single inequality */ - switch (rho) - { case T_LE: - /* first <= second */ - con->code = first; - con->lbnd = NULL; - con->ubnd = second; - break; - case T_GE: - /* first >= second */ - con->code = first; - con->lbnd = second; - con->ubnd = NULL; - break; - case T_EQ: - /* first = second */ - con->code = first; - con->lbnd = second; - con->ubnd = second; - break; - default: - xassert(rho != rho); - } - } - else - { /* the constraint is double inequality */ - switch (rho) - { case T_LE: - /* first <= second <= third */ - con->code = second; - con->lbnd = first; - con->ubnd = third; - break; - case T_GE: - /* first >= second >= third */ - con->code = second; - con->lbnd = third; - con->ubnd = first; - break; - default: - xassert(rho != rho); - } - } - /* the constraint statement has been completely parsed */ - if (mpl->token != T_SEMICOLON) -err: error(mpl, "syntax error in constraint statement"); - get_token(mpl /* ; */); - return con; -} - -/*---------------------------------------------------------------------- --- objective_statement - parse objective statement. --- --- This routine parses objective statement using the syntax: --- --- ::= : --- ; --- ::= minimize --- ::= maximize --- ::= --- ::= --- ::= --- ::= --- ::= */ - -CONSTRAINT *objective_statement(MPL *mpl) -{ CONSTRAINT *obj; - int type; - if (is_keyword(mpl, "minimize")) - type = A_MINIMIZE; - else if (is_keyword(mpl, "maximize")) - type = A_MAXIMIZE; - else - xassert(mpl != mpl); - if (mpl->flag_s) - error(mpl, "objective statement must precede solve statement"); - get_token(mpl /* minimize | maximize */); - /* symbolic name must follow the verb 'minimize' or 'maximize' */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "symbolic name missing where expected"); - /* there must be no other object with the same name */ - if (avl_find_node(mpl->tree, mpl->image) != NULL) - error(mpl, "%s multiply declared", mpl->image); - /* create model objective */ - obj = alloc(CONSTRAINT); - obj->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(obj->name, mpl->image); - obj->alias = NULL; - obj->dim = 0; - obj->domain = NULL; - obj->type = type; - obj->code = NULL; - obj->lbnd = NULL; - obj->ubnd = NULL; - obj->array = NULL; - get_token(mpl /* */); - /* parse optional alias */ - if (mpl->token == T_STRING) - { obj->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(obj->alias, mpl->image); - get_token(mpl /* */); - } - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { obj->domain = indexing_expression(mpl); - obj->dim = domain_arity(mpl, obj->domain); - } - /* include the constraint name in the symbolic names table */ - { AVLNODE *node; - node = avl_insert_node(mpl->tree, obj->name); - avl_set_node_type(node, A_CONSTRAINT); - avl_set_node_link(node, (void *)obj); - } - /* the colon must precede the objective expression */ - if (mpl->token != T_COLON) - error(mpl, "colon missing where expected"); - get_token(mpl /* : */); - /* parse the objective expression */ - obj->code = expression_5(mpl); - if (obj->code->type == A_SYMBOLIC) - obj->code = make_unary(mpl, O_CVTNUM, obj->code, A_NUMERIC, 0); - if (obj->code->type == A_NUMERIC) - obj->code = make_unary(mpl, O_CVTLFM, obj->code, A_FORMULA, 0); - if (obj->code->type != A_FORMULA) - error(mpl, "expression following colon has invalid type"); - xassert(obj->code->dim == 0); - /* close the domain scope */ - if (obj->domain != NULL) close_scope(mpl, obj->domain); - /* the objective statement has been completely parsed */ - if (mpl->token != T_SEMICOLON) - error(mpl, "syntax error in objective statement"); - get_token(mpl /* ; */); - return obj; -} - -#if 1 /* 11/II-2008 */ -/*********************************************************************** -* table_statement - parse table statement -* -* This routine parses table statement using the syntax: -* -* ::= -*
::= -* -* ::= -* table
IN : -* [ ] , ; -* ::= -* ::= -* ::= -* ::= -* ::= , -* ::= -* ::= <- -* ::= -* ::= , -* ::= -* ::= , -* ::= -* ::= ~ -* -* ::= -* table
OUT : -* ; -* ::= -* ::= -* ::= , -* ::= -* ::= ~ */ - -TABLE *table_statement(MPL *mpl) -{ TABLE *tab; - TABARG *last_arg, *arg; - TABFLD *last_fld, *fld; - TABIN *last_in, *in; - TABOUT *last_out, *out; - AVLNODE *node; - int nflds; - char name[MAX_LENGTH+1]; - xassert(is_keyword(mpl, "table")); - get_token(mpl /* solve */); - /* symbolic name must follow the keyword table */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "symbolic name missing where expected"); - /* there must be no other object with the same name */ - if (avl_find_node(mpl->tree, mpl->image) != NULL) - error(mpl, "%s multiply declared", mpl->image); - /* create data table */ - tab = alloc(TABLE); - tab->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(tab->name, mpl->image); - get_token(mpl /* */); - /* parse optional alias */ - if (mpl->token == T_STRING) - { tab->alias = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(tab->alias, mpl->image); - get_token(mpl /* */); - } - else - tab->alias = NULL; - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { /* this is output table */ - tab->type = A_OUTPUT; - tab->u.out.domain = indexing_expression(mpl); - if (!is_keyword(mpl, "OUT")) - error(mpl, "keyword OUT missing where expected"); - get_token(mpl /* OUT */); - } - else - { /* this is input table */ - tab->type = A_INPUT; - if (!is_keyword(mpl, "IN")) - error(mpl, "keyword IN missing where expected"); - get_token(mpl /* IN */); - } - /* parse argument list */ - tab->arg = last_arg = NULL; - for (;;) - { /* create argument list entry */ - arg = alloc(TABARG); - /* parse argument expression */ - if (mpl->token == T_COMMA || mpl->token == T_COLON || - mpl->token == T_SEMICOLON) - error(mpl, "argument expression missing where expected"); - arg->code = expression_5(mpl); - /* convert the result to symbolic type, if necessary */ - if (arg->code->type == A_NUMERIC) - arg->code = - make_unary(mpl, O_CVTSYM, arg->code, A_SYMBOLIC, 0); - /* check that now the result is of symbolic type */ - if (arg->code->type != A_SYMBOLIC) - error(mpl, "argument expression has invalid type"); - /* add the entry to the end of the list */ - arg->next = NULL; - if (last_arg == NULL) - tab->arg = arg; - else - last_arg->next = arg; - last_arg = arg; - /* argument expression has been parsed */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_COLON || mpl->token == T_SEMICOLON) - break; - } - xassert(tab->arg != NULL); - /* argument list must end with colon */ - if (mpl->token == T_COLON) - get_token(mpl /* : */); - else - error(mpl, "colon missing where expected"); - /* parse specific part of the table statement */ - switch (tab->type) - { case A_INPUT: goto input_table; - case A_OUTPUT: goto output_table; - default: xassert(tab != tab); - } -input_table: - /* parse optional set name */ - if (mpl->token == T_NAME) - { node = avl_find_node(mpl->tree, mpl->image); - if (node == NULL) - error(mpl, "%s not defined", mpl->image); - if (avl_get_node_type(node) != A_SET) - error(mpl, "%s not a set", mpl->image); - tab->u.in.set = (SET *)avl_get_node_link(node); - if (tab->u.in.set->assign != NULL) - error(mpl, "%s needs no data", mpl->image); - if (tab->u.in.set->dim != 0) - error(mpl, "%s must be a simple set", mpl->image); - get_token(mpl /* */); - if (mpl->token == T_INPUT) - get_token(mpl /* <- */); - else - error(mpl, "delimiter <- missing where expected"); - } - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - tab->u.in.set = NULL; - /* parse field list */ - tab->u.in.fld = last_fld = NULL; - nflds = 0; - if (mpl->token == T_LBRACKET) - get_token(mpl /* [ */); - else - error(mpl, "field list missing where expected"); - for (;;) - { /* create field list entry */ - fld = alloc(TABFLD); - /* parse field name */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, - "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "field name missing where expected"); - fld->name = dmp_get_atomv(mpl->pool, strlen(mpl->image)+1); - strcpy(fld->name, mpl->image); - get_token(mpl /* */); - /* add the entry to the end of the list */ - fld->next = NULL; - if (last_fld == NULL) - tab->u.in.fld = fld; - else - last_fld->next = fld; - last_fld = fld; - nflds++; - /* field name has been parsed */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RBRACKET) - break; - else - error(mpl, "syntax error in field list"); - } - /* check that the set dimen is equal to the number of fields */ - if (tab->u.in.set != NULL && tab->u.in.set->dimen != nflds) - error(mpl, "there must be %d field%s rather than %d", - tab->u.in.set->dimen, tab->u.in.set->dimen == 1 ? "" : "s", - nflds); - get_token(mpl /* ] */); - /* parse optional input list */ - tab->u.in.list = last_in = NULL; - while (mpl->token == T_COMMA) - { get_token(mpl /* , */); - /* create input list entry */ - in = alloc(TABIN); - /* parse parameter name */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, - "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "parameter name missing where expected"); - node = avl_find_node(mpl->tree, mpl->image); - if (node == NULL) - error(mpl, "%s not defined", mpl->image); - if (avl_get_node_type(node) != A_PARAMETER) - error(mpl, "%s not a parameter", mpl->image); - in->par = (PARAMETER *)avl_get_node_link(node); - if (in->par->dim != nflds) - error(mpl, "%s must have %d subscript%s rather than %d", - mpl->image, nflds, nflds == 1 ? "" : "s", in->par->dim); - if (in->par->assign != NULL) - error(mpl, "%s needs no data", mpl->image); - get_token(mpl /* */); - /* parse optional field name */ - if (mpl->token == T_TILDE) - { get_token(mpl /* ~ */); - /* parse field name */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, - "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "field name missing where expected"); - xassert(strlen(mpl->image) < sizeof(name)); - strcpy(name, mpl->image); - get_token(mpl /* */); - } - else - { /* field name is the same as the parameter name */ - xassert(strlen(in->par->name) < sizeof(name)); - strcpy(name, in->par->name); - } - /* assign field name */ - in->name = dmp_get_atomv(mpl->pool, strlen(name)+1); - strcpy(in->name, name); - /* add the entry to the end of the list */ - in->next = NULL; - if (last_in == NULL) - tab->u.in.list = in; - else - last_in->next = in; - last_in = in; - } - goto end_of_table; -output_table: - /* parse output list */ - tab->u.out.list = last_out = NULL; - for (;;) - { /* create output list entry */ - out = alloc(TABOUT); - /* parse expression */ - if (mpl->token == T_COMMA || mpl->token == T_SEMICOLON) - error(mpl, "expression missing where expected"); - if (mpl->token == T_NAME) - { xassert(strlen(mpl->image) < sizeof(name)); - strcpy(name, mpl->image); - } - else - name[0] = '\0'; - out->code = expression_5(mpl); - /* parse optional field name */ - if (mpl->token == T_TILDE) - { get_token(mpl /* ~ */); - /* parse field name */ - if (mpl->token == T_NAME) - ; - else if (is_reserved(mpl)) - error(mpl, - "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "field name missing where expected"); - xassert(strlen(mpl->image) < sizeof(name)); - strcpy(name, mpl->image); - get_token(mpl /* */); - } - /* assign field name */ - if (name[0] == '\0') - error(mpl, "field name required"); - out->name = dmp_get_atomv(mpl->pool, strlen(name)+1); - strcpy(out->name, name); - /* add the entry to the end of the list */ - out->next = NULL; - if (last_out == NULL) - tab->u.out.list = out; - else - last_out->next = out; - last_out = out; - /* output item has been parsed */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_SEMICOLON) - break; - else - error(mpl, "syntax error in output list"); - } - /* close the domain scope */ - close_scope(mpl,tab->u.out.domain); -end_of_table: - /* the table statement must end with semicolon */ - if (mpl->token != T_SEMICOLON) - error(mpl, "syntax error in table statement"); - get_token(mpl /* ; */); - return tab; -} -#endif - -/*---------------------------------------------------------------------- --- solve_statement - parse solve statement. --- --- This routine parses solve statement using the syntax: --- --- ::= solve ; --- --- The solve statement can be used at most once. */ - -void *solve_statement(MPL *mpl) -{ xassert(is_keyword(mpl, "solve")); - if (mpl->flag_s) - error(mpl, "at most one solve statement allowed"); - mpl->flag_s = 1; - get_token(mpl /* solve */); - /* semicolon must follow solve statement */ - if (mpl->token != T_SEMICOLON) - error(mpl, "syntax error in solve statement"); - get_token(mpl /* ; */); - return NULL; -} - -/*---------------------------------------------------------------------- --- check_statement - parse check statement. --- --- This routine parses check statement using the syntax: --- --- ::= check : ; --- ::= --- ::= --- --- If is omitted, colon following it may also be omitted. */ - -CHECK *check_statement(MPL *mpl) -{ CHECK *chk; - xassert(is_keyword(mpl, "check")); - /* create check descriptor */ - chk = alloc(CHECK); - chk->domain = NULL; - chk->code = NULL; - get_token(mpl /* check */); - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { chk->domain = indexing_expression(mpl); -#if 0 - if (mpl->token != T_COLON) - error(mpl, "colon missing where expected"); -#endif - } - /* skip optional colon */ - if (mpl->token == T_COLON) get_token(mpl /* : */); - /* parse logical expression */ - chk->code = expression_13(mpl); - if (chk->code->type != A_LOGICAL) - error(mpl, "expression has invalid type"); - xassert(chk->code->dim == 0); - /* close the domain scope */ - if (chk->domain != NULL) close_scope(mpl, chk->domain); - /* the check statement has been completely parsed */ - if (mpl->token != T_SEMICOLON) - error(mpl, "syntax error in check statement"); - get_token(mpl /* ; */); - return chk; -} - -#if 1 /* 15/V-2010 */ -/*---------------------------------------------------------------------- --- display_statement - parse display statement. --- --- This routine parses display statement using the syntax: --- --- ::= display : ; --- ::= display ; --- ::= --- ::= --- ::= --- ::= , --- ::= --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- ::= --- ::= [ ] --- ::= */ - -DISPLAY *display_statement(MPL *mpl) -{ DISPLAY *dpy; - DISPLAY1 *entry, *last_entry; - xassert(is_keyword(mpl, "display")); - /* create display descriptor */ - dpy = alloc(DISPLAY); - dpy->domain = NULL; - dpy->list = last_entry = NULL; - get_token(mpl /* display */); - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - dpy->domain = indexing_expression(mpl); - /* skip optional colon */ - if (mpl->token == T_COLON) get_token(mpl /* : */); - /* parse display list */ - for (;;) - { /* create new display entry */ - entry = alloc(DISPLAY1); - entry->type = 0; - entry->next = NULL; - /* and append it to the display list */ - if (dpy->list == NULL) - dpy->list = entry; - else - last_entry->next = entry; - last_entry = entry; - /* parse display entry */ - if (mpl->token == T_NAME) - { AVLNODE *node; - int next_token; - get_token(mpl /* */); - next_token = mpl->token; - unget_token(mpl); - if (!(next_token == T_COMMA || next_token == T_SEMICOLON)) - { /* symbolic name begins expression */ - goto expr; - } - /* display entry is dummy index or model object */ - node = avl_find_node(mpl->tree, mpl->image); - if (node == NULL) - error(mpl, "%s not defined", mpl->image); - entry->type = avl_get_node_type(node); - switch (avl_get_node_type(node)) - { case A_INDEX: - entry->u.slot = - (DOMAIN_SLOT *)avl_get_node_link(node); - break; - case A_SET: - entry->u.set = (SET *)avl_get_node_link(node); - break; - case A_PARAMETER: - entry->u.par = (PARAMETER *)avl_get_node_link(node); - break; - case A_VARIABLE: - entry->u.var = (VARIABLE *)avl_get_node_link(node); - if (!mpl->flag_s) - error(mpl, "invalid reference to variable %s above" - " solve statement", entry->u.var->name); - break; - case A_CONSTRAINT: - entry->u.con = (CONSTRAINT *)avl_get_node_link(node); - if (!mpl->flag_s) - error(mpl, "invalid reference to %s %s above solve" - " statement", - entry->u.con->type == A_CONSTRAINT ? - "constraint" : "objective", entry->u.con->name); - break; - default: - xassert(node != node); - } - get_token(mpl /* */); - } - else -expr: { /* display entry is expression */ - entry->type = A_EXPRESSION; - entry->u.code = expression_13(mpl); - } - /* check a token that follows the entry parsed */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else - break; - } - /* close the domain scope */ - if (dpy->domain != NULL) close_scope(mpl, dpy->domain); - /* the display statement has been completely parsed */ - if (mpl->token != T_SEMICOLON) - error(mpl, "syntax error in display statement"); - get_token(mpl /* ; */); - return dpy; -} -#endif - -/*---------------------------------------------------------------------- --- printf_statement - parse printf statement. --- --- This routine parses print statement using the syntax: --- --- ::= ; --- ::= > ; --- ::= >> ; --- ::= printf : --- ::= printf --- ::= --- ::= --- ::= --- ::= --- ::= , --- ::= --- ::= */ - -PRINTF *printf_statement(MPL *mpl) -{ PRINTF *prt; - PRINTF1 *entry, *last_entry; - xassert(is_keyword(mpl, "printf")); - /* create printf descriptor */ - prt = alloc(PRINTF); - prt->domain = NULL; - prt->fmt = NULL; - prt->list = last_entry = NULL; - get_token(mpl /* printf */); - /* parse optional indexing expression */ - if (mpl->token == T_LBRACE) - { prt->domain = indexing_expression(mpl); -#if 0 - if (mpl->token != T_COLON) - error(mpl, "colon missing where expected"); -#endif - } - /* skip optional colon */ - if (mpl->token == T_COLON) get_token(mpl /* : */); - /* parse expression for format string */ - prt->fmt = expression_5(mpl); - /* convert it to symbolic type, if necessary */ - if (prt->fmt->type == A_NUMERIC) - prt->fmt = make_unary(mpl, O_CVTSYM, prt->fmt, A_SYMBOLIC, 0); - /* check that now the expression is of symbolic type */ - if (prt->fmt->type != A_SYMBOLIC) - error(mpl, "format expression has invalid type"); - /* parse printf list */ - while (mpl->token == T_COMMA) - { get_token(mpl /* , */); - /* create new printf entry */ - entry = alloc(PRINTF1); - entry->code = NULL; - entry->next = NULL; - /* and append it to the printf list */ - if (prt->list == NULL) - prt->list = entry; - else - last_entry->next = entry; - last_entry = entry; - /* parse printf entry */ - entry->code = expression_9(mpl); - if (!(entry->code->type == A_NUMERIC || - entry->code->type == A_SYMBOLIC || - entry->code->type == A_LOGICAL)) - error(mpl, "only numeric, symbolic, or logical expression a" - "llowed"); - } - /* close the domain scope */ - if (prt->domain != NULL) close_scope(mpl, prt->domain); -#if 1 /* 14/VII-2006 */ - /* parse optional redirection */ - prt->fname = NULL, prt->app = 0; - if (mpl->token == T_GT || mpl->token == T_APPEND) - { prt->app = (mpl->token == T_APPEND); - get_token(mpl /* > or >> */); - /* parse expression for file name string */ - prt->fname = expression_5(mpl); - /* convert it to symbolic type, if necessary */ - if (prt->fname->type == A_NUMERIC) - prt->fname = make_unary(mpl, O_CVTSYM, prt->fname, - A_SYMBOLIC, 0); - /* check that now the expression is of symbolic type */ - if (prt->fname->type != A_SYMBOLIC) - error(mpl, "file name expression has invalid type"); - } -#endif - /* the printf statement has been completely parsed */ - if (mpl->token != T_SEMICOLON) - error(mpl, "syntax error in printf statement"); - get_token(mpl /* ; */); - return prt; -} - -/*---------------------------------------------------------------------- --- for_statement - parse for statement. --- --- This routine parses for statement using the syntax: --- --- ::= for --- ::= for { } --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= */ - -FOR *for_statement(MPL *mpl) -{ FOR *fur; - STATEMENT *stmt, *last_stmt; - xassert(is_keyword(mpl, "for")); - /* create for descriptor */ - fur = alloc(FOR); - fur->domain = NULL; - fur->list = last_stmt = NULL; - get_token(mpl /* for */); - /* parse indexing expression */ - if (mpl->token != T_LBRACE) - error(mpl, "indexing expression missing where expected"); - fur->domain = indexing_expression(mpl); - /* skip optional colon */ - if (mpl->token == T_COLON) get_token(mpl /* : */); - /* parse for statement body */ - if (mpl->token != T_LBRACE) - { /* parse simple statement */ - fur->list = simple_statement(mpl, 1); - } - else - { /* parse compound statement */ - get_token(mpl /* { */); - while (mpl->token != T_RBRACE) - { /* parse statement */ - stmt = simple_statement(mpl, 1); - /* and append it to the end of the statement list */ - if (last_stmt == NULL) - fur->list = stmt; - else - last_stmt->next = stmt; - last_stmt = stmt; - } - get_token(mpl /* } */); - } - /* close the domain scope */ - xassert(fur->domain != NULL); - close_scope(mpl, fur->domain); - /* the for statement has been completely parsed */ - return fur; -} - -/*---------------------------------------------------------------------- --- end_statement - parse end statement. --- --- This routine parses end statement using the syntax: --- --- ::= end ; */ - -void end_statement(MPL *mpl) -{ if (!mpl->flag_d && is_keyword(mpl, "end") || - mpl->flag_d && is_literal(mpl, "end")) - { get_token(mpl /* end */); - if (mpl->token == T_SEMICOLON) - get_token(mpl /* ; */); - else - warning(mpl, "no semicolon following end statement; missing" - " semicolon inserted"); - } - else - warning(mpl, "unexpected end of file; missing end statement in" - "serted"); - if (mpl->token != T_EOF) - warning(mpl, "some text detected beyond end statement; text ig" - "nored"); - return; -} - -/*---------------------------------------------------------------------- --- simple_statement - parse simple statement. --- --- This routine parses simple statement using the syntax: --- --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= --- ::= --- --- If the flag spec is set, some statements cannot be used. */ - -STATEMENT *simple_statement(MPL *mpl, int spec) -{ STATEMENT *stmt; - stmt = alloc(STATEMENT); - stmt->line = mpl->line; - stmt->next = NULL; - if (is_keyword(mpl, "set")) - { if (spec) - error(mpl, "set statement not allowed here"); - stmt->type = A_SET; - stmt->u.set = set_statement(mpl); - } - else if (is_keyword(mpl, "param")) - { if (spec) - error(mpl, "parameter statement not allowed here"); - stmt->type = A_PARAMETER; - stmt->u.par = parameter_statement(mpl); - } - else if (is_keyword(mpl, "var")) - { if (spec) - error(mpl, "variable statement not allowed here"); - stmt->type = A_VARIABLE; - stmt->u.var = variable_statement(mpl); - } - else if (is_keyword(mpl, "subject") || - is_keyword(mpl, "subj") || - mpl->token == T_SPTP) - { if (spec) - error(mpl, "constraint statement not allowed here"); - stmt->type = A_CONSTRAINT; - stmt->u.con = constraint_statement(mpl); - } - else if (is_keyword(mpl, "minimize") || - is_keyword(mpl, "maximize")) - { if (spec) - error(mpl, "objective statement not allowed here"); - stmt->type = A_CONSTRAINT; - stmt->u.con = objective_statement(mpl); - } -#if 1 /* 11/II-2008 */ - else if (is_keyword(mpl, "table")) - { if (spec) - error(mpl, "table statement not allowed here"); - stmt->type = A_TABLE; - stmt->u.tab = table_statement(mpl); - } -#endif - else if (is_keyword(mpl, "solve")) - { if (spec) - error(mpl, "solve statement not allowed here"); - stmt->type = A_SOLVE; - stmt->u.slv = solve_statement(mpl); - } - else if (is_keyword(mpl, "check")) - { stmt->type = A_CHECK; - stmt->u.chk = check_statement(mpl); - } - else if (is_keyword(mpl, "display")) - { stmt->type = A_DISPLAY; - stmt->u.dpy = display_statement(mpl); - } - else if (is_keyword(mpl, "printf")) - { stmt->type = A_PRINTF; - stmt->u.prt = printf_statement(mpl); - } - else if (is_keyword(mpl, "for")) - { stmt->type = A_FOR; - stmt->u.fur = for_statement(mpl); - } - else if (mpl->token == T_NAME) - { if (spec) - error(mpl, "constraint statement not allowed here"); - stmt->type = A_CONSTRAINT; - stmt->u.con = constraint_statement(mpl); - } - else if (is_reserved(mpl)) - error(mpl, "invalid use of reserved keyword %s", mpl->image); - else - error(mpl, "syntax error in model section"); - return stmt; -} - -/*---------------------------------------------------------------------- --- model_section - parse model section. --- --- This routine parses model section using the syntax: --- --- ::= --- ::= --- --- Parsing model section is terminated by either the keyword 'data', or --- the keyword 'end', or the end of file. */ - -void model_section(MPL *mpl) -{ STATEMENT *stmt, *last_stmt; - xassert(mpl->model == NULL); - last_stmt = NULL; - while (!(mpl->token == T_EOF || is_keyword(mpl, "data") || - is_keyword(mpl, "end"))) - { /* parse statement */ - stmt = simple_statement(mpl, 0); - /* and append it to the end of the statement list */ - if (last_stmt == NULL) - mpl->model = stmt; - else - last_stmt->next = stmt; - last_stmt = stmt; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl2.c b/code/3rd_glpk/mpl/mpl2.c deleted file mode 100644 index 0f99528b..00000000 --- a/code/3rd_glpk/mpl/mpl2.c +++ /dev/null @@ -1,1202 +0,0 @@ -/* mpl2.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mpl.h" - -/**********************************************************************/ -/* * * PROCESSING DATA SECTION * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- create_slice - create slice. --- --- This routine creates a slice, which initially has no components. */ - -SLICE *create_slice(MPL *mpl) -{ SLICE *slice; - xassert(mpl == mpl); - slice = NULL; - return slice; -} - -/*---------------------------------------------------------------------- --- expand_slice - append new component to slice. --- --- This routine expands slice appending to it either a given symbol or --- null component, which becomes the last component of the slice. */ - -SLICE *expand_slice -( MPL *mpl, - SLICE *slice, /* destroyed */ - SYMBOL *sym /* destroyed */ -) -{ SLICE *tail, *temp; - /* create a new component */ - tail = dmp_get_atom(mpl->tuples, sizeof(SLICE)); - tail->sym = sym; - tail->next = NULL; - /* and append it to the component list */ - if (slice == NULL) - slice = tail; - else - { for (temp = slice; temp->next != NULL; temp = temp->next); - temp->next = tail; - } - return slice; -} - -/*---------------------------------------------------------------------- --- slice_dimen - determine dimension of slice. --- --- This routine returns dimension of slice, which is number of all its --- components including null ones. */ - -int slice_dimen -( MPL *mpl, - SLICE *slice /* not changed */ -) -{ SLICE *temp; - int dim; - xassert(mpl == mpl); - dim = 0; - for (temp = slice; temp != NULL; temp = temp->next) dim++; - return dim; -} - -/*---------------------------------------------------------------------- --- slice_arity - determine arity of slice. --- --- This routine returns arity of slice, i.e. number of null components --- (indicated by asterisks) in the slice. */ - -int slice_arity -( MPL *mpl, - SLICE *slice /* not changed */ -) -{ SLICE *temp; - int arity; - xassert(mpl == mpl); - arity = 0; - for (temp = slice; temp != NULL; temp = temp->next) - if (temp->sym == NULL) arity++; - return arity; -} - -/*---------------------------------------------------------------------- --- fake_slice - create fake slice of all asterisks. --- --- This routine creates a fake slice of given dimension, which contains --- asterisks in all components. Zero dimension is allowed. */ - -SLICE *fake_slice(MPL *mpl, int dim) -{ SLICE *slice; - slice = create_slice(mpl); - while (dim-- > 0) slice = expand_slice(mpl, slice, NULL); - return slice; -} - -/*---------------------------------------------------------------------- --- delete_slice - delete slice. --- --- This routine deletes specified slice. */ - -void delete_slice -( MPL *mpl, - SLICE *slice /* destroyed */ -) -{ SLICE *temp; - while (slice != NULL) - { temp = slice; - slice = temp->next; - if (temp->sym != NULL) delete_symbol(mpl, temp->sym); -xassert(sizeof(SLICE) == sizeof(TUPLE)); - dmp_free_atom(mpl->tuples, temp, sizeof(TUPLE)); - } - return; -} - -/*---------------------------------------------------------------------- --- is_number - check if current token is number. --- --- If the current token is a number, this routine returns non-zero. --- Otherwise zero is returned. */ - -int is_number(MPL *mpl) -{ return - mpl->token == T_NUMBER; -} - -/*---------------------------------------------------------------------- --- is_symbol - check if current token is symbol. --- --- If the current token is suitable to be a symbol, the routine returns --- non-zero. Otherwise zero is returned. */ - -int is_symbol(MPL *mpl) -{ return - mpl->token == T_NUMBER || - mpl->token == T_SYMBOL || - mpl->token == T_STRING; -} - -/*---------------------------------------------------------------------- --- is_literal - check if current token is given symbolic literal. --- --- If the current token is given symbolic literal, this routine returns --- non-zero. Otherwise zero is returned. --- --- This routine is used on processing the data section in the same way --- as the routine is_keyword on processing the model section. */ - -int is_literal(MPL *mpl, char *literal) -{ return - is_symbol(mpl) && strcmp(mpl->image, literal) == 0; -} - -/*---------------------------------------------------------------------- --- read_number - read number. --- --- This routine reads the current token, which must be a number, and --- returns its numeric value. */ - -double read_number(MPL *mpl) -{ double num; - xassert(is_number(mpl)); - num = mpl->value; - get_token(mpl /* */); - return num; -} - -/*---------------------------------------------------------------------- --- read_symbol - read symbol. --- --- This routine reads the current token, which must be a symbol, and --- returns its symbolic value. */ - -SYMBOL *read_symbol(MPL *mpl) -{ SYMBOL *sym; - xassert(is_symbol(mpl)); - if (is_number(mpl)) - sym = create_symbol_num(mpl, mpl->value); - else - sym = create_symbol_str(mpl, create_string(mpl, mpl->image)); - get_token(mpl /* */); - return sym; -} - -/*---------------------------------------------------------------------- --- read_slice - read slice. --- --- This routine reads slice using the syntax: --- --- ::= [ ] --- ::= ( ) --- ::= --- ::= , --- ::= --- ::= * --- --- The bracketed form of slice is used for members of multi-dimensional --- objects while the parenthesized form is used for elemental sets. */ - -SLICE *read_slice -( MPL *mpl, - char *name, /* not changed */ - int dim -) -{ SLICE *slice; - int close; - xassert(name != NULL); - switch (mpl->token) - { case T_LBRACKET: - close = T_RBRACKET; - break; - case T_LEFT: - xassert(dim > 0); - close = T_RIGHT; - break; - default: - xassert(mpl != mpl); - } - if (dim == 0) - error(mpl, "%s cannot be subscripted", name); - get_token(mpl /* ( | [ */); - /* read slice components */ - slice = create_slice(mpl); - for (;;) - { /* the current token must be a symbol or asterisk */ - if (is_symbol(mpl)) - slice = expand_slice(mpl, slice, read_symbol(mpl)); - else if (mpl->token == T_ASTERISK) - { slice = expand_slice(mpl, slice, NULL); - get_token(mpl /* * */); - } - else - error(mpl, "number, symbol, or asterisk missing where expec" - "ted"); - /* check a token that follows the symbol */ - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == close) - break; - else - error(mpl, "syntax error in slice"); - } - /* number of slice components must be the same as the appropriate - dimension */ - if (slice_dimen(mpl, slice) != dim) - { switch (close) - { case T_RBRACKET: - error(mpl, "%s must have %d subscript%s, not %d", name, - dim, dim == 1 ? "" : "s", slice_dimen(mpl, slice)); - break; - case T_RIGHT: - error(mpl, "%s has dimension %d, not %d", name, dim, - slice_dimen(mpl, slice)); - break; - default: - xassert(close != close); - } - } - get_token(mpl /* ) | ] */); - return slice; -} - -/*---------------------------------------------------------------------- --- select_set - select set to saturate it with elemental sets. --- --- This routine selects set to saturate it with elemental sets provided --- in the data section. */ - -SET *select_set -( MPL *mpl, - char *name /* not changed */ -) -{ SET *set; - AVLNODE *node; - xassert(name != NULL); - node = avl_find_node(mpl->tree, name); - if (node == NULL || avl_get_node_type(node) != A_SET) - error(mpl, "%s not a set", name); - set = (SET *)avl_get_node_link(node); - if (set->assign != NULL || set->gadget != NULL) - error(mpl, "%s needs no data", name); - set->data = 1; - return set; -} - -/*---------------------------------------------------------------------- --- simple_format - read set data block in simple format. --- --- This routine reads set data block using the syntax: --- --- ::= , , ... , --- --- where are used to construct a complete n-tuple, which is --- included in elemental set assigned to the set member. Commae between --- symbols are optional and may be omitted anywhere. --- --- Number of components in the slice must be the same as dimension of --- n-tuples in elemental sets assigned to the set members. To construct --- complete n-tuple the routine replaces null positions in the slice by --- corresponding . --- --- If the slice contains at least one null position, the current token --- must be symbol. Otherwise, the routine reads no symbols to construct --- the n-tuple, so the current token is not checked. */ - -void simple_format -( MPL *mpl, - SET *set, /* not changed */ - MEMBER *memb, /* modified */ - SLICE *slice /* not changed */ -) -{ TUPLE *tuple; - SLICE *temp; - SYMBOL *sym, *with = NULL; - xassert(set != NULL); - xassert(memb != NULL); - xassert(slice != NULL); - xassert(set->dimen == slice_dimen(mpl, slice)); - xassert(memb->value.set->dim == set->dimen); - if (slice_arity(mpl, slice) > 0) xassert(is_symbol(mpl)); - /* read symbols and construct complete n-tuple */ - tuple = create_tuple(mpl); - for (temp = slice; temp != NULL; temp = temp->next) - { if (temp->sym == NULL) - { /* substitution is needed; read symbol */ - if (!is_symbol(mpl)) - { int lack = slice_arity(mpl, temp); - /* with cannot be null due to assertion above */ - xassert(with != NULL); - if (lack == 1) - error(mpl, "one item missing in data group beginning " - "with %s", format_symbol(mpl, with)); - else - error(mpl, "%d items missing in data group beginning " - "with %s", lack, format_symbol(mpl, with)); - } - sym = read_symbol(mpl); - if (with == NULL) with = sym; - } - else - { /* copy symbol from the slice */ - sym = copy_symbol(mpl, temp->sym); - } - /* append the symbol to the n-tuple */ - tuple = expand_tuple(mpl, tuple, sym); - /* skip optional comma *between* */ - if (temp->next != NULL && mpl->token == T_COMMA) - get_token(mpl /* , */); - } - /* add constructed n-tuple to elemental set */ - check_then_add(mpl, memb->value.set, tuple); - return; -} - -/*---------------------------------------------------------------------- --- matrix_format - read set data block in matrix format. --- --- This routine reads set data block using the syntax: --- --- ::= ... := --- +/- +/- ... +/- --- +/- +/- ... +/- --- . . . . . . . . . . . --- +/- +/- ... +/- --- --- where are symbols that denote rows of the matrix, --- are symbols that denote columns of the matrix, "+" and "-" indicate --- whether corresponding n-tuple needs to be included in the elemental --- set or not, respectively. --- --- Number of the slice components must be the same as dimension of the --- elemental set. The slice must have two null positions. To construct --- complete n-tuple for particular element of the matrix the routine --- replaces first null position of the slice by the corresponding --- (or , if the flag tr is on) and second null position by the --- corresponding (or by , if the flag tr is on). */ - -void matrix_format -( MPL *mpl, - SET *set, /* not changed */ - MEMBER *memb, /* modified */ - SLICE *slice, /* not changed */ - int tr -) -{ SLICE *list, *col, *temp; - TUPLE *tuple; - SYMBOL *row; - xassert(set != NULL); - xassert(memb != NULL); - xassert(slice != NULL); - xassert(set->dimen == slice_dimen(mpl, slice)); - xassert(memb->value.set->dim == set->dimen); - xassert(slice_arity(mpl, slice) == 2); - /* read the matrix heading that contains column symbols (there - may be no columns at all) */ - list = create_slice(mpl); - while (mpl->token != T_ASSIGN) - { /* read column symbol and append it to the column list */ - if (!is_symbol(mpl)) - error(mpl, "number, symbol, or := missing where expected"); - list = expand_slice(mpl, list, read_symbol(mpl)); - } - get_token(mpl /* := */); - /* read zero or more rows that contain matrix data */ - while (is_symbol(mpl)) - { /* read row symbol (if the matrix has no columns, row symbols - are just ignored) */ - row = read_symbol(mpl); - /* read the matrix row accordingly to the column list */ - for (col = list; col != NULL; col = col->next) - { int which = 0; - /* check indicator */ - if (is_literal(mpl, "+")) - ; - else if (is_literal(mpl, "-")) - { get_token(mpl /* - */); - continue; - } - else - { int lack = slice_dimen(mpl, col); - if (lack == 1) - error(mpl, "one item missing in data group beginning " - "with %s", format_symbol(mpl, row)); - else - error(mpl, "%d items missing in data group beginning " - "with %s", lack, format_symbol(mpl, row)); - } - /* construct complete n-tuple */ - tuple = create_tuple(mpl); - for (temp = slice; temp != NULL; temp = temp->next) - { if (temp->sym == NULL) - { /* substitution is needed */ - switch (++which) - { case 1: - /* substitute in the first null position */ - tuple = expand_tuple(mpl, tuple, - copy_symbol(mpl, tr ? col->sym : row)); - break; - case 2: - /* substitute in the second null position */ - tuple = expand_tuple(mpl, tuple, - copy_symbol(mpl, tr ? row : col->sym)); - break; - default: - xassert(which != which); - } - } - else - { /* copy symbol from the slice */ - tuple = expand_tuple(mpl, tuple, copy_symbol(mpl, - temp->sym)); - } - } - xassert(which == 2); - /* add constructed n-tuple to elemental set */ - check_then_add(mpl, memb->value.set, tuple); - get_token(mpl /* + */); - } - /* delete the row symbol */ - delete_symbol(mpl, row); - } - /* delete the column list */ - delete_slice(mpl, list); - return; -} - -/*---------------------------------------------------------------------- --- set_data - read set data. --- --- This routine reads set data using the syntax: --- --- ::= set ; --- ::= set [ ] ; --- ::= --- ::= --- ::= , := --- ::= , ( ) --- ::= , --- ::= , : --- ::= , (tr) --- ::= , (tr) : --- --- Commae in are optional and may be omitted anywhere. */ - -void set_data(MPL *mpl) -{ SET *set; - TUPLE *tuple; - MEMBER *memb; - SLICE *slice; - int tr = 0; - xassert(is_literal(mpl, "set")); - get_token(mpl /* set */); - /* symbolic name of set must follows the keyword 'set' */ - if (!is_symbol(mpl)) - error(mpl, "set name missing where expected"); - /* select the set to saturate it with data */ - set = select_set(mpl, mpl->image); - get_token(mpl /* */); - /* read optional subscript list, which identifies member of the - set to be read */ - tuple = create_tuple(mpl); - if (mpl->token == T_LBRACKET) - { /* subscript list is specified */ - if (set->dim == 0) - error(mpl, "%s cannot be subscripted", set->name); - get_token(mpl /* [ */); - /* read symbols and construct subscript list */ - for (;;) - { if (!is_symbol(mpl)) - error(mpl, "number or symbol missing where expected"); - tuple = expand_tuple(mpl, tuple, read_symbol(mpl)); - if (mpl->token == T_COMMA) - get_token(mpl /* , */); - else if (mpl->token == T_RBRACKET) - break; - else - error(mpl, "syntax error in subscript list"); - } - if (set->dim != tuple_dimen(mpl, tuple)) - error(mpl, "%s must have %d subscript%s rather than %d", - set->name, set->dim, set->dim == 1 ? "" : "s", - tuple_dimen(mpl, tuple)); - get_token(mpl /* ] */); - } - else - { /* subscript list is not specified */ - if (set->dim != 0) - error(mpl, "%s must be subscripted", set->name); - } - /* there must be no member with the same subscript list */ - if (find_member(mpl, set->array, tuple) != NULL) - error(mpl, "%s%s already defined", - set->name, format_tuple(mpl, '[', tuple)); - /* add new member to the set and assign it empty elemental set */ - memb = add_member(mpl, set->array, tuple); - memb->value.set = create_elemset(mpl, set->dimen); - /* create an initial fake slice of all asterisks */ - slice = fake_slice(mpl, set->dimen); - /* read zero or more data assignments */ - for (;;) - { /* skip optional comma */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - /* process assignment element */ - if (mpl->token == T_ASSIGN) - { /* assignment ligature is non-significant element */ - get_token(mpl /* := */); - } - else if (mpl->token == T_LEFT) - { /* left parenthesis begins either new slice or "transpose" - indicator */ - int is_tr; - get_token(mpl /* ( */); - is_tr = is_literal(mpl, "tr"); - unget_token(mpl /* ( */); - if (is_tr) goto left; - /* delete the current slice and read new one */ - delete_slice(mpl, slice); - slice = read_slice(mpl, set->name, set->dimen); - /* each new slice resets the "transpose" indicator */ - tr = 0; - /* if the new slice is 0-ary, formally there is one 0-tuple - (in the simple format) that follows it */ - if (slice_arity(mpl, slice) == 0) - simple_format(mpl, set, memb, slice); - } - else if (is_symbol(mpl)) - { /* number or symbol begins data in the simple format */ - simple_format(mpl, set, memb, slice); - } - else if (mpl->token == T_COLON) - { /* colon begins data in the matrix format */ - if (slice_arity(mpl, slice) != 2) -err1: error(mpl, "slice currently used must specify 2 asterisk" - "s, not %d", slice_arity(mpl, slice)); - get_token(mpl /* : */); - /* read elemental set data in the matrix format */ - matrix_format(mpl, set, memb, slice, tr); - } - else if (mpl->token == T_LEFT) -left: { /* left parenthesis begins the "transpose" indicator, which - is followed by data in the matrix format */ - get_token(mpl /* ( */); - if (!is_literal(mpl, "tr")) -err2: error(mpl, "transpose indicator (tr) incomplete"); - if (slice_arity(mpl, slice) != 2) goto err1; - get_token(mpl /* tr */); - if (mpl->token != T_RIGHT) goto err2; - get_token(mpl /* ) */); - /* in this case the colon is optional */ - if (mpl->token == T_COLON) get_token(mpl /* : */); - /* set the "transpose" indicator */ - tr = 1; - /* read elemental set data in the matrix format */ - matrix_format(mpl, set, memb, slice, tr); - } - else if (mpl->token == T_SEMICOLON) - { /* semicolon terminates the data block */ - get_token(mpl /* ; */); - break; - } - else - error(mpl, "syntax error in set data block"); - } - /* delete the current slice */ - delete_slice(mpl, slice); - return; -} - -/*---------------------------------------------------------------------- --- select_parameter - select parameter to saturate it with data. --- --- This routine selects parameter to saturate it with data provided in --- the data section. */ - -PARAMETER *select_parameter -( MPL *mpl, - char *name /* not changed */ -) -{ PARAMETER *par; - AVLNODE *node; - xassert(name != NULL); - node = avl_find_node(mpl->tree, name); - if (node == NULL || avl_get_node_type(node) != A_PARAMETER) - error(mpl, "%s not a parameter", name); - par = (PARAMETER *)avl_get_node_link(node); - if (par->assign != NULL) - error(mpl, "%s needs no data", name); - if (par->data) - error(mpl, "%s already provided with data", name); - par->data = 1; - return par; -} - -/*---------------------------------------------------------------------- --- set_default - set default parameter value. --- --- This routine sets default value for specified parameter. */ - -void set_default -( MPL *mpl, - PARAMETER *par, /* not changed */ - SYMBOL *altval /* destroyed */ -) -{ xassert(par != NULL); - xassert(altval != NULL); - if (par->option != NULL) - error(mpl, "default value for %s already specified in model se" - "ction", par->name); - xassert(par->defval == NULL); - par->defval = altval; - return; -} - -/*---------------------------------------------------------------------- --- read_value - read value and assign it to parameter member. --- --- This routine reads numeric or symbolic value from the input stream --- and assigns to new parameter member specified by its n-tuple, which --- (the member) is created and added to the parameter array. */ - -MEMBER *read_value -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* destroyed */ -) -{ MEMBER *memb; - xassert(par != NULL); - xassert(is_symbol(mpl)); - /* there must be no member with the same n-tuple */ - if (find_member(mpl, par->array, tuple) != NULL) - error(mpl, "%s%s already defined", - par->name, format_tuple(mpl, '[', tuple)); - /* create new parameter member with given n-tuple */ - memb = add_member(mpl, par->array, tuple); - /* read value and assigns it to the new parameter member */ - switch (par->type) - { case A_NUMERIC: - case A_INTEGER: - case A_BINARY: - if (!is_number(mpl)) - error(mpl, "%s requires numeric data", par->name); - memb->value.num = read_number(mpl); - break; - case A_SYMBOLIC: - memb->value.sym = read_symbol(mpl); - break; - default: - xassert(par != par); - } - return memb; -} - -/*---------------------------------------------------------------------- --- plain_format - read parameter data block in plain format. --- --- This routine reads parameter data block using the syntax: --- --- ::= , , ... , , --- --- where are used to determine a complete subscript list for --- parameter member, is a numeric or symbolic value assigned to --- the parameter member. Commae between data items are optional and may --- be omitted anywhere. --- --- Number of components in the slice must be the same as dimension of --- the parameter. To construct the complete subscript list the routine --- replaces null positions in the slice by corresponding . */ - -void plain_format -( MPL *mpl, - PARAMETER *par, /* not changed */ - SLICE *slice /* not changed */ -) -{ TUPLE *tuple; - SLICE *temp; - SYMBOL *sym, *with = NULL; - xassert(par != NULL); - xassert(par->dim == slice_dimen(mpl, slice)); - xassert(is_symbol(mpl)); - /* read symbols and construct complete subscript list */ - tuple = create_tuple(mpl); - for (temp = slice; temp != NULL; temp = temp->next) - { if (temp->sym == NULL) - { /* substitution is needed; read symbol */ - if (!is_symbol(mpl)) - { int lack = slice_arity(mpl, temp) + 1; - xassert(with != NULL); - xassert(lack > 1); - error(mpl, "%d items missing in data group beginning wit" - "h %s", lack, format_symbol(mpl, with)); - } - sym = read_symbol(mpl); - if (with == NULL) with = sym; - } - else - { /* copy symbol from the slice */ - sym = copy_symbol(mpl, temp->sym); - } - /* append the symbol to the subscript list */ - tuple = expand_tuple(mpl, tuple, sym); - /* skip optional comma */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - } - /* read value and assign it to new parameter member */ - if (!is_symbol(mpl)) - { xassert(with != NULL); - error(mpl, "one item missing in data group beginning with %s", - format_symbol(mpl, with)); - } - read_value(mpl, par, tuple); - return; -} - -/*---------------------------------------------------------------------- --- tabular_format - read parameter data block in tabular format. --- --- This routine reads parameter data block using the syntax: --- --- ::= ... := --- ... --- ... --- . . . . . . . . . . . --- ... --- --- where are symbols that denote rows of the table, --- are symbols that denote columns of the table, are numeric --- or symbolic values assigned to the corresponding parameter members. --- If is specified as single point, no value is provided. --- --- Number of components in the slice must be the same as dimension of --- the parameter. The slice must have two null positions. To construct --- complete subscript list for particular the routine replaces --- the first null position of the slice by the corresponding (or --- , if the flag tr is on) and the second null position by the --- corresponding (or by , if the flag tr is on). */ - -void tabular_format -( MPL *mpl, - PARAMETER *par, /* not changed */ - SLICE *slice, /* not changed */ - int tr -) -{ SLICE *list, *col, *temp; - TUPLE *tuple; - SYMBOL *row; - xassert(par != NULL); - xassert(par->dim == slice_dimen(mpl, slice)); - xassert(slice_arity(mpl, slice) == 2); - /* read the table heading that contains column symbols (the table - may have no columns) */ - list = create_slice(mpl); - while (mpl->token != T_ASSIGN) - { /* read column symbol and append it to the column list */ - if (!is_symbol(mpl)) - error(mpl, "number, symbol, or := missing where expected"); - list = expand_slice(mpl, list, read_symbol(mpl)); - } - get_token(mpl /* := */); - /* read zero or more rows that contain tabular data */ - while (is_symbol(mpl)) - { /* read row symbol (if the table has no columns, these symbols - are just ignored) */ - row = read_symbol(mpl); - /* read values accordingly to the column list */ - for (col = list; col != NULL; col = col->next) - { int which = 0; - /* if the token is single point, no value is provided */ - if (is_literal(mpl, ".")) - { get_token(mpl /* . */); - continue; - } - /* construct complete subscript list */ - tuple = create_tuple(mpl); - for (temp = slice; temp != NULL; temp = temp->next) - { if (temp->sym == NULL) - { /* substitution is needed */ - switch (++which) - { case 1: - /* substitute in the first null position */ - tuple = expand_tuple(mpl, tuple, - copy_symbol(mpl, tr ? col->sym : row)); - break; - case 2: - /* substitute in the second null position */ - tuple = expand_tuple(mpl, tuple, - copy_symbol(mpl, tr ? row : col->sym)); - break; - default: - xassert(which != which); - } - } - else - { /* copy symbol from the slice */ - tuple = expand_tuple(mpl, tuple, copy_symbol(mpl, - temp->sym)); - } - } - xassert(which == 2); - /* read value and assign it to new parameter member */ - if (!is_symbol(mpl)) - { int lack = slice_dimen(mpl, col); - if (lack == 1) - error(mpl, "one item missing in data group beginning " - "with %s", format_symbol(mpl, row)); - else - error(mpl, "%d items missing in data group beginning " - "with %s", lack, format_symbol(mpl, row)); - } - read_value(mpl, par, tuple); - } - /* delete the row symbol */ - delete_symbol(mpl, row); - } - /* delete the column list */ - delete_slice(mpl, list); - return; -} - -/*---------------------------------------------------------------------- --- tabbing_format - read parameter data block in tabbing format. --- --- This routine reads parameter data block using the syntax: --- --- ::= , ... , , := , --- , ... , , , ... , , --- , ... , , , ... , , --- . . . . . . . . . . . . . . . . . --- , ... , , , ... , --- ::= --- ::= : --- --- where are names of parameters (all the parameters must be --- subscripted and have identical dimensions), are symbols --- used to define subscripts of parameter members, are numeric --- or symbolic values assigned to the corresponding parameter members. --- Optional may specify a simple set, in which case n-tuples --- built of for each row of the data table (i.e. subscripts --- of parameter members) are added to the specified set. Commae between --- data items are optional and may be omitted anywhere. --- --- If the parameter altval is not NULL, it specifies a default value --- provided for all the parameters specified in the data block. */ - -void tabbing_format -( MPL *mpl, - SYMBOL *altval /* not changed */ -) -{ SET *set = NULL; - PARAMETER *par; - SLICE *list, *col; - TUPLE *tuple; - int next_token, j, dim = 0; - char *last_name = NULL; - /* read the optional */ - if (is_symbol(mpl)) - { get_token(mpl /* */); - next_token = mpl->token; - unget_token(mpl /* */); - if (next_token == T_COLON) - { /* select the set to saturate it with data */ - set = select_set(mpl, mpl->image); - /* the set must be simple (i.e. not set of sets) */ - if (set->dim != 0) - error(mpl, "%s must be a simple set", set->name); - /* and must not be defined yet */ - if (set->array->head != NULL) - error(mpl, "%s already defined", set->name); - /* add new (the only) member to the set and assign it empty - elemental set */ - add_member(mpl, set->array, NULL)->value.set = - create_elemset(mpl, set->dimen); - last_name = set->name, dim = set->dimen; - get_token(mpl /* */); - xassert(mpl->token == T_COLON); - get_token(mpl /* : */); - } - } - /* read the table heading that contains parameter names */ - list = create_slice(mpl); - while (mpl->token != T_ASSIGN) - { /* there must be symbolic name of parameter */ - if (!is_symbol(mpl)) - error(mpl, "parameter name or := missing where expected"); - /* select the parameter to saturate it with data */ - par = select_parameter(mpl, mpl->image); - /* the parameter must be subscripted */ - if (par->dim == 0) - error(mpl, "%s not a subscripted parameter", mpl->image); - /* the set (if specified) and all the parameters in the data - block must have identical dimension */ - if (dim != 0 && par->dim != dim) - { xassert(last_name != NULL); - error(mpl, "%s has dimension %d while %s has dimension %d", - last_name, dim, par->name, par->dim); - } - /* set default value for the parameter (if specified) */ - if (altval != NULL) - set_default(mpl, par, copy_symbol(mpl, altval)); - /* append the parameter to the column list */ - list = expand_slice(mpl, list, (SYMBOL *)par); - last_name = par->name, dim = par->dim; - get_token(mpl /* */); - /* skip optional comma */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - } - if (slice_dimen(mpl, list) == 0) - error(mpl, "at least one parameter name required"); - get_token(mpl /* := */); - /* skip optional comma */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - /* read rows that contain tabbing data */ - while (is_symbol(mpl)) - { /* read subscript list */ - tuple = create_tuple(mpl); - for (j = 1; j <= dim; j++) - { /* read j-th subscript */ - if (!is_symbol(mpl)) - { int lack = slice_dimen(mpl, list) + dim - j + 1; - xassert(tuple != NULL); - xassert(lack > 1); - error(mpl, "%d items missing in data group beginning wit" - "h %s", lack, format_symbol(mpl, tuple->sym)); - } - /* read and append j-th subscript to the n-tuple */ - tuple = expand_tuple(mpl, tuple, read_symbol(mpl)); - /* skip optional comma *between* */ - if (j < dim && mpl->token == T_COMMA) - get_token(mpl /* , */); - } - /* if the set is specified, add to it new n-tuple, which is a - copy of the subscript list just read */ - if (set != NULL) - check_then_add(mpl, set->array->head->value.set, - copy_tuple(mpl, tuple)); - /* skip optional comma between and */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - /* read values accordingly to the column list */ - for (col = list; col != NULL; col = col->next) - { /* if the token is single point, no value is provided */ - if (is_literal(mpl, ".")) - { get_token(mpl /* . */); - continue; - } - /* read value and assign it to new parameter member */ - if (!is_symbol(mpl)) - { int lack = slice_dimen(mpl, col); - xassert(tuple != NULL); - if (lack == 1) - error(mpl, "one item missing in data group beginning " - "with %s", format_symbol(mpl, tuple->sym)); - else - error(mpl, "%d items missing in data group beginning " - "with %s", lack, format_symbol(mpl, tuple->sym)); - } - read_value(mpl, (PARAMETER *)col->sym, copy_tuple(mpl, - tuple)); - /* skip optional comma preceding the next value */ - if (col->next != NULL && mpl->token == T_COMMA) - get_token(mpl /* , */); - } - /* delete the original subscript list */ - delete_tuple(mpl, tuple); - /* skip optional comma (only if there is next data group) */ - if (mpl->token == T_COMMA) - { get_token(mpl /* , */); - if (!is_symbol(mpl)) unget_token(mpl /* , */); - } - } - /* delete the column list (it contains parameters, not symbols, - so nullify it before) */ - for (col = list; col != NULL; col = col->next) col->sym = NULL; - delete_slice(mpl, list); - return; -} - -/*---------------------------------------------------------------------- --- parameter_data - read parameter data. --- --- This routine reads parameter data using the syntax: --- --- ::= param : ; --- ::= param --- ; --- ::= --- ::= --- ::= default --- ::= --- ::= , := --- ::= , [ ] --- ::= , --- ::= , : --- ::= , (tr) --- ::= , (tr) : --- --- Commae in are optional and may be omitted anywhere. */ - -void parameter_data(MPL *mpl) -{ PARAMETER *par; - SYMBOL *altval = NULL; - SLICE *slice; - int tr = 0; - xassert(is_literal(mpl, "param")); - get_token(mpl /* param */); - /* read optional default value */ - if (is_literal(mpl, "default")) - { get_token(mpl /* default */); - if (!is_symbol(mpl)) - error(mpl, "default value missing where expected"); - altval = read_symbol(mpl); - /* if the default value follows the keyword 'param', the next - token must be only the colon */ - if (mpl->token != T_COLON) - error(mpl, "colon missing where expected"); - } - /* being used after the keyword 'param' or the optional default - value the colon begins data in the tabbing format */ - if (mpl->token == T_COLON) - { get_token(mpl /* : */); - /* skip optional comma */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - /* read parameter data in the tabbing format */ - tabbing_format(mpl, altval); - /* on reading data in the tabbing format the default value is - always copied, so delete the original symbol */ - if (altval != NULL) delete_symbol(mpl, altval); - /* the next token must be only semicolon */ - if (mpl->token != T_SEMICOLON) - error(mpl, "symbol, number, or semicolon missing where expe" - "cted"); - get_token(mpl /* ; */); - goto done; - } - /* in other cases there must be symbolic name of parameter, which - follows the keyword 'param' */ - if (!is_symbol(mpl)) - error(mpl, "parameter name missing where expected"); - /* select the parameter to saturate it with data */ - par = select_parameter(mpl, mpl->image); - get_token(mpl /* */); - /* read optional default value */ - if (is_literal(mpl, "default")) - { get_token(mpl /* default */); - if (!is_symbol(mpl)) - error(mpl, "default value missing where expected"); - altval = read_symbol(mpl); - /* set default value for the parameter */ - set_default(mpl, par, altval); - } - /* create initial fake slice of all asterisks */ - slice = fake_slice(mpl, par->dim); - /* read zero or more data assignments */ - for (;;) - { /* skip optional comma */ - if (mpl->token == T_COMMA) get_token(mpl /* , */); - /* process current assignment */ - if (mpl->token == T_ASSIGN) - { /* assignment ligature is non-significant element */ - get_token(mpl /* := */); - } - else if (mpl->token == T_LBRACKET) - { /* left bracket begins new slice; delete the current slice - and read new one */ - delete_slice(mpl, slice); - slice = read_slice(mpl, par->name, par->dim); - /* each new slice resets the "transpose" indicator */ - tr = 0; - } - else if (is_symbol(mpl)) - { /* number or symbol begins data in the plain format */ - plain_format(mpl, par, slice); - } - else if (mpl->token == T_COLON) - { /* colon begins data in the tabular format */ - if (par->dim == 0) -err1: error(mpl, "%s not a subscripted parameter", - par->name); - if (slice_arity(mpl, slice) != 2) -err2: error(mpl, "slice currently used must specify 2 asterisk" - "s, not %d", slice_arity(mpl, slice)); - get_token(mpl /* : */); - /* read parameter data in the tabular format */ - tabular_format(mpl, par, slice, tr); - } - else if (mpl->token == T_LEFT) - { /* left parenthesis begins the "transpose" indicator, which - is followed by data in the tabular format */ - get_token(mpl /* ( */); - if (!is_literal(mpl, "tr")) -err3: error(mpl, "transpose indicator (tr) incomplete"); - if (par->dim == 0) goto err1; - if (slice_arity(mpl, slice) != 2) goto err2; - get_token(mpl /* tr */); - if (mpl->token != T_RIGHT) goto err3; - get_token(mpl /* ) */); - /* in this case the colon is optional */ - if (mpl->token == T_COLON) get_token(mpl /* : */); - /* set the "transpose" indicator */ - tr = 1; - /* read parameter data in the tabular format */ - tabular_format(mpl, par, slice, tr); - } - else if (mpl->token == T_SEMICOLON) - { /* semicolon terminates the data block */ - get_token(mpl /* ; */); - break; - } - else - error(mpl, "syntax error in parameter data block"); - } - /* delete the current slice */ - delete_slice(mpl, slice); -done: return; -} - -/*---------------------------------------------------------------------- --- data_section - read data section. --- --- This routine reads data section using the syntax: --- --- ::= --- ::= ; --- ::= --- ::= --- --- Reading data section is terminated by either the keyword 'end' or --- the end of file. */ - -void data_section(MPL *mpl) -{ while (!(mpl->token == T_EOF || is_literal(mpl, "end"))) - { if (is_literal(mpl, "set")) - set_data(mpl); - else if (is_literal(mpl, "param")) - parameter_data(mpl); - else - error(mpl, "syntax error in data section"); - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl3.c b/code/3rd_glpk/mpl/mpl3.c deleted file mode 100644 index 2489db27..00000000 --- a/code/3rd_glpk/mpl/mpl3.c +++ /dev/null @@ -1,6100 +0,0 @@ -/* mpl3.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mpl.h" - -/**********************************************************************/ -/* * * FLOATING-POINT NUMBERS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- fp_add - floating-point addition. --- --- This routine computes the sum x + y. */ - -double fp_add(MPL *mpl, double x, double y) -{ if (x > 0.0 && y > 0.0 && x > + 0.999 * DBL_MAX - y || - x < 0.0 && y < 0.0 && x < - 0.999 * DBL_MAX - y) - error(mpl, "%.*g + %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - return x + y; -} - -/*---------------------------------------------------------------------- --- fp_sub - floating-point subtraction. --- --- This routine computes the difference x - y. */ - -double fp_sub(MPL *mpl, double x, double y) -{ if (x > 0.0 && y < 0.0 && x > + 0.999 * DBL_MAX + y || - x < 0.0 && y > 0.0 && x < - 0.999 * DBL_MAX + y) - error(mpl, "%.*g - %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - return x - y; -} - -/*---------------------------------------------------------------------- --- fp_less - floating-point non-negative subtraction. --- --- This routine computes the non-negative difference max(0, x - y). */ - -double fp_less(MPL *mpl, double x, double y) -{ if (x < y) return 0.0; - if (x > 0.0 && y < 0.0 && x > + 0.999 * DBL_MAX + y) - error(mpl, "%.*g less %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - return x - y; -} - -/*---------------------------------------------------------------------- --- fp_mul - floating-point multiplication. --- --- This routine computes the product x * y. */ - -double fp_mul(MPL *mpl, double x, double y) -{ if (fabs(y) > 1.0 && fabs(x) > (0.999 * DBL_MAX) / fabs(y)) - error(mpl, "%.*g * %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - return x * y; -} - -/*---------------------------------------------------------------------- --- fp_div - floating-point division. --- --- This routine computes the quotient x / y. */ - -double fp_div(MPL *mpl, double x, double y) -{ if (fabs(y) < DBL_MIN) - error(mpl, "%.*g / %.*g; floating-point zero divide", - DBL_DIG, x, DBL_DIG, y); - if (fabs(y) < 1.0 && fabs(x) > (0.999 * DBL_MAX) * fabs(y)) - error(mpl, "%.*g / %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - return x / y; -} - -/*---------------------------------------------------------------------- --- fp_idiv - floating-point quotient of exact division. --- --- This routine computes the quotient of exact division x div y. */ - -double fp_idiv(MPL *mpl, double x, double y) -{ if (fabs(y) < DBL_MIN) - error(mpl, "%.*g div %.*g; floating-point zero divide", - DBL_DIG, x, DBL_DIG, y); - if (fabs(y) < 1.0 && fabs(x) > (0.999 * DBL_MAX) * fabs(y)) - error(mpl, "%.*g div %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - x /= y; - return x > 0.0 ? floor(x) : x < 0.0 ? ceil(x) : 0.0; -} - -/*---------------------------------------------------------------------- --- fp_mod - floating-point remainder of exact division. --- --- This routine computes the remainder of exact division x mod y. --- --- NOTE: By definition x mod y = x - y * floor(x / y). */ - -double fp_mod(MPL *mpl, double x, double y) -{ double r; - xassert(mpl == mpl); - if (x == 0.0) - r = 0.0; - else if (y == 0.0) - r = x; - else - { r = fmod(fabs(x), fabs(y)); - if (r != 0.0) - { if (x < 0.0) r = - r; - if (x > 0.0 && y < 0.0 || x < 0.0 && y > 0.0) r += y; - } - } - return r; -} - -/*---------------------------------------------------------------------- --- fp_power - floating-point exponentiation (raise to power). --- --- This routine computes the exponentiation x ** y. */ - -double fp_power(MPL *mpl, double x, double y) -{ double r; - if (x == 0.0 && y <= 0.0 || x < 0.0 && y != floor(y)) - error(mpl, "%.*g ** %.*g; result undefined", - DBL_DIG, x, DBL_DIG, y); - if (x == 0.0) goto eval; - if (fabs(x) > 1.0 && y > +1.0 && - +log(fabs(x)) > (0.999 * log(DBL_MAX)) / y || - fabs(x) < 1.0 && y < -1.0 && - +log(fabs(x)) < (0.999 * log(DBL_MAX)) / y) - error(mpl, "%.*g ** %.*g; floating-point overflow", - DBL_DIG, x, DBL_DIG, y); - if (fabs(x) > 1.0 && y < -1.0 && - -log(fabs(x)) < (0.999 * log(DBL_MAX)) / y || - fabs(x) < 1.0 && y > +1.0 && - -log(fabs(x)) > (0.999 * log(DBL_MAX)) / y) - r = 0.0; - else -eval: r = pow(x, y); - return r; -} - -/*---------------------------------------------------------------------- --- fp_exp - floating-point base-e exponential. --- --- This routine computes the base-e exponential e ** x. */ - -double fp_exp(MPL *mpl, double x) -{ if (x > 0.999 * log(DBL_MAX)) - error(mpl, "exp(%.*g); floating-point overflow", DBL_DIG, x); - return exp(x); -} - -/*---------------------------------------------------------------------- --- fp_log - floating-point natural logarithm. --- --- This routine computes the natural logarithm log x. */ - -double fp_log(MPL *mpl, double x) -{ if (x <= 0.0) - error(mpl, "log(%.*g); non-positive argument", DBL_DIG, x); - return log(x); -} - -/*---------------------------------------------------------------------- --- fp_log10 - floating-point common (decimal) logarithm. --- --- This routine computes the common (decimal) logarithm lg x. */ - -double fp_log10(MPL *mpl, double x) -{ if (x <= 0.0) - error(mpl, "log10(%.*g); non-positive argument", DBL_DIG, x); - return log10(x); -} - -/*---------------------------------------------------------------------- --- fp_sqrt - floating-point square root. --- --- This routine computes the square root x ** 0.5. */ - -double fp_sqrt(MPL *mpl, double x) -{ if (x < 0.0) - error(mpl, "sqrt(%.*g); negative argument", DBL_DIG, x); - return sqrt(x); -} - -/*---------------------------------------------------------------------- --- fp_sin - floating-point trigonometric sine. --- --- This routine computes the trigonometric sine sin(x). */ - -double fp_sin(MPL *mpl, double x) -{ if (!(-1e6 <= x && x <= +1e6)) - error(mpl, "sin(%.*g); argument too large", DBL_DIG, x); - return sin(x); -} - -/*---------------------------------------------------------------------- --- fp_cos - floating-point trigonometric cosine. --- --- This routine computes the trigonometric cosine cos(x). */ - -double fp_cos(MPL *mpl, double x) -{ if (!(-1e6 <= x && x <= +1e6)) - error(mpl, "cos(%.*g); argument too large", DBL_DIG, x); - return cos(x); -} - -/*---------------------------------------------------------------------- --- fp_tan - floating-point trigonometric tangent. --- --- This routine computes the trigonometric tangent tan(x). */ - -double fp_tan(MPL *mpl, double x) -{ if (!(-1e6 <= x && x <= +1e6)) - error(mpl, "tan(%.*g); argument too large", DBL_DIG, x); - return tan(x); -} - -/*---------------------------------------------------------------------- --- fp_atan - floating-point trigonometric arctangent. --- --- This routine computes the trigonometric arctangent atan(x). */ - -double fp_atan(MPL *mpl, double x) -{ xassert(mpl == mpl); - return atan(x); -} - -/*---------------------------------------------------------------------- --- fp_atan2 - floating-point trigonometric arctangent. --- --- This routine computes the trigonometric arctangent atan(y / x). */ - -double fp_atan2(MPL *mpl, double y, double x) -{ xassert(mpl == mpl); - return atan2(y, x); -} - -/*---------------------------------------------------------------------- --- fp_round - round floating-point value to n fractional digits. --- --- This routine rounds given floating-point value x to n fractional --- digits with the formula: --- --- round(x, n) = floor(x * 10^n + 0.5) / 10^n. --- --- The parameter n is assumed to be integer. */ - -double fp_round(MPL *mpl, double x, double n) -{ double ten_to_n; - if (n != floor(n)) - error(mpl, "round(%.*g, %.*g); non-integer second argument", - DBL_DIG, x, DBL_DIG, n); - if (n <= DBL_DIG + 2) - { ten_to_n = pow(10.0, n); - if (fabs(x) < (0.999 * DBL_MAX) / ten_to_n) - { x = floor(x * ten_to_n + 0.5); - if (x != 0.0) x /= ten_to_n; - } - } - return x; -} - -/*---------------------------------------------------------------------- --- fp_trunc - truncate floating-point value to n fractional digits. --- --- This routine truncates given floating-point value x to n fractional --- digits with the formula: --- --- ( floor(x * 10^n) / 10^n, if x >= 0 --- trunc(x, n) = < --- ( ceil(x * 10^n) / 10^n, if x < 0 --- --- The parameter n is assumed to be integer. */ - -double fp_trunc(MPL *mpl, double x, double n) -{ double ten_to_n; - if (n != floor(n)) - error(mpl, "trunc(%.*g, %.*g); non-integer second argument", - DBL_DIG, x, DBL_DIG, n); - if (n <= DBL_DIG + 2) - { ten_to_n = pow(10.0, n); - if (fabs(x) < (0.999 * DBL_MAX) / ten_to_n) - { x = (x >= 0.0 ? floor(x * ten_to_n) : ceil(x * ten_to_n)); - if (x != 0.0) x /= ten_to_n; - } - } - return x; -} - -/**********************************************************************/ -/* * * PSEUDO-RANDOM NUMBER GENERATORS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- fp_irand224 - pseudo-random integer in the range [0, 2^24). --- --- This routine returns a next pseudo-random integer (converted to --- floating-point) which is uniformly distributed between 0 and 2^24-1, --- inclusive. */ - -#define two_to_the_24 0x1000000 - -double fp_irand224(MPL *mpl) -{ return - (double)rng_unif_rand(mpl->rand, two_to_the_24); -} - -/*---------------------------------------------------------------------- --- fp_uniform01 - pseudo-random number in the range [0, 1). --- --- This routine returns a next pseudo-random number which is uniformly --- distributed in the range [0, 1). */ - -#define two_to_the_31 ((unsigned int)0x80000000) - -double fp_uniform01(MPL *mpl) -{ return - (double)rng_next_rand(mpl->rand) / (double)two_to_the_31; -} - -/*---------------------------------------------------------------------- --- fp_uniform - pseudo-random number in the range [a, b). --- --- This routine returns a next pseudo-random number which is uniformly --- distributed in the range [a, b). */ - -double fp_uniform(MPL *mpl, double a, double b) -{ double x; - if (a >= b) - error(mpl, "Uniform(%.*g, %.*g); invalid range", - DBL_DIG, a, DBL_DIG, b); - x = fp_uniform01(mpl); -#if 0 - x = a * (1.0 - x) + b * x; -#else - x = fp_add(mpl, a * (1.0 - x), b * x); -#endif - return x; -} - -/*---------------------------------------------------------------------- --- fp_normal01 - Gaussian random variate with mu = 0 and sigma = 1. --- --- This routine returns a Gaussian random variate with zero mean and --- unit standard deviation. The polar (Box-Mueller) method is used. --- --- This code is a modified version of the routine gsl_ran_gaussian from --- the GNU Scientific Library Version 1.0. */ - -double fp_normal01(MPL *mpl) -{ double x, y, r2; - do - { /* choose x, y in uniform square (-1,-1) to (+1,+1) */ - x = -1.0 + 2.0 * fp_uniform01(mpl); - y = -1.0 + 2.0 * fp_uniform01(mpl); - /* see if it is in the unit circle */ - r2 = x * x + y * y; - } while (r2 > 1.0 || r2 == 0.0); - /* Box-Muller transform */ - return y * sqrt(-2.0 * log (r2) / r2); -} - -/*---------------------------------------------------------------------- --- fp_normal - Gaussian random variate with specified mu and sigma. --- --- This routine returns a Gaussian random variate with mean mu and --- standard deviation sigma. */ - -double fp_normal(MPL *mpl, double mu, double sigma) -{ double x; -#if 0 - x = mu + sigma * fp_normal01(mpl); -#else - x = fp_add(mpl, mu, fp_mul(mpl, sigma, fp_normal01(mpl))); -#endif - return x; -} - -/**********************************************************************/ -/* * * SEGMENTED CHARACTER STRINGS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- create_string - create character string. --- --- This routine creates a segmented character string, which is exactly --- equivalent to specified character string. */ - -STRING *create_string -( MPL *mpl, - char buf[MAX_LENGTH+1] /* not changed */ -) -#if 0 -{ STRING *head, *tail; - int i, j; - xassert(buf != NULL); - xassert(strlen(buf) <= MAX_LENGTH); - head = tail = dmp_get_atom(mpl->strings, sizeof(STRING)); - for (i = j = 0; ; i++) - { if ((tail->seg[j++] = buf[i]) == '\0') break; - if (j == STRSEG_SIZE) -tail = (tail->next = dmp_get_atom(mpl->strings, sizeof(STRING))), j = 0; - } - tail->next = NULL; - return head; -} -#else -{ STRING *str; - xassert(strlen(buf) <= MAX_LENGTH); - str = dmp_get_atom(mpl->strings, strlen(buf)+1); - strcpy(str, buf); - return str; -} -#endif - -/*---------------------------------------------------------------------- --- copy_string - make copy of character string. --- --- This routine returns an exact copy of segmented character string. */ - -STRING *copy_string -( MPL *mpl, - STRING *str /* not changed */ -) -#if 0 -{ STRING *head, *tail; - xassert(str != NULL); - head = tail = dmp_get_atom(mpl->strings, sizeof(STRING)); - for (; str != NULL; str = str->next) - { memcpy(tail->seg, str->seg, STRSEG_SIZE); - if (str->next != NULL) -tail = (tail->next = dmp_get_atom(mpl->strings, sizeof(STRING))); - } - tail->next = NULL; - return head; -} -#else -{ xassert(mpl == mpl); - return create_string(mpl, str); -} -#endif - -/*---------------------------------------------------------------------- --- compare_strings - compare one character string with another. --- --- This routine compares one segmented character strings with another --- and returns the result of comparison as follows: --- --- = 0 - both strings are identical; --- < 0 - the first string precedes the second one; --- > 0 - the first string follows the second one. */ - -int compare_strings -( MPL *mpl, - STRING *str1, /* not changed */ - STRING *str2 /* not changed */ -) -#if 0 -{ int j, c1, c2; - xassert(mpl == mpl); - for (;; str1 = str1->next, str2 = str2->next) - { xassert(str1 != NULL); - xassert(str2 != NULL); - for (j = 0; j < STRSEG_SIZE; j++) - { c1 = (unsigned char)str1->seg[j]; - c2 = (unsigned char)str2->seg[j]; - if (c1 < c2) return -1; - if (c1 > c2) return +1; - if (c1 == '\0') goto done; - } - } -done: return 0; -} -#else -{ xassert(mpl == mpl); - return strcmp(str1, str2); -} -#endif - -/*---------------------------------------------------------------------- --- fetch_string - extract content of character string. --- --- This routine returns a character string, which is exactly equivalent --- to specified segmented character string. */ - -char *fetch_string -( MPL *mpl, - STRING *str, /* not changed */ - char buf[MAX_LENGTH+1] /* modified */ -) -#if 0 -{ int i, j; - xassert(mpl == mpl); - xassert(buf != NULL); - for (i = 0; ; str = str->next) - { xassert(str != NULL); - for (j = 0; j < STRSEG_SIZE; j++) - if ((buf[i++] = str->seg[j]) == '\0') goto done; - } -done: xassert(strlen(buf) <= MAX_LENGTH); - return buf; -} -#else -{ xassert(mpl == mpl); - return strcpy(buf, str); -} -#endif - -/*---------------------------------------------------------------------- --- delete_string - delete character string. --- --- This routine deletes specified segmented character string. */ - -void delete_string -( MPL *mpl, - STRING *str /* destroyed */ -) -#if 0 -{ STRING *temp; - xassert(str != NULL); - while (str != NULL) - { temp = str; - str = str->next; - dmp_free_atom(mpl->strings, temp, sizeof(STRING)); - } - return; -} -#else -{ dmp_free_atom(mpl->strings, str, strlen(str)+1); - return; -} -#endif - -/**********************************************************************/ -/* * * SYMBOLS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- create_symbol_num - create symbol of numeric type. --- --- This routine creates a symbol, which has a numeric value specified --- as floating-point number. */ - -SYMBOL *create_symbol_num(MPL *mpl, double num) -{ SYMBOL *sym; - sym = dmp_get_atom(mpl->symbols, sizeof(SYMBOL)); - sym->num = num; - sym->str = NULL; - return sym; -} - -/*---------------------------------------------------------------------- --- create_symbol_str - create symbol of abstract type. --- --- This routine creates a symbol, which has an abstract value specified --- as segmented character string. */ - -SYMBOL *create_symbol_str -( MPL *mpl, - STRING *str /* destroyed */ -) -{ SYMBOL *sym; - xassert(str != NULL); - sym = dmp_get_atom(mpl->symbols, sizeof(SYMBOL)); - sym->num = 0.0; - sym->str = str; - return sym; -} - -/*---------------------------------------------------------------------- --- copy_symbol - make copy of symbol. --- --- This routine returns an exact copy of symbol. */ - -SYMBOL *copy_symbol -( MPL *mpl, - SYMBOL *sym /* not changed */ -) -{ SYMBOL *copy; - xassert(sym != NULL); - copy = dmp_get_atom(mpl->symbols, sizeof(SYMBOL)); - if (sym->str == NULL) - { copy->num = sym->num; - copy->str = NULL; - } - else - { copy->num = 0.0; - copy->str = copy_string(mpl, sym->str); - } - return copy; -} - -/*---------------------------------------------------------------------- --- compare_symbols - compare one symbol with another. --- --- This routine compares one symbol with another and returns the result --- of comparison as follows: --- --- = 0 - both symbols are identical; --- < 0 - the first symbol precedes the second one; --- > 0 - the first symbol follows the second one. --- --- Note that the linear order, in which symbols follow each other, is --- implementation-dependent. It may be not an alphabetical order. */ - -int compare_symbols -( MPL *mpl, - SYMBOL *sym1, /* not changed */ - SYMBOL *sym2 /* not changed */ -) -{ xassert(sym1 != NULL); - xassert(sym2 != NULL); - /* let all numeric quantities precede all symbolic quantities */ - if (sym1->str == NULL && sym2->str == NULL) - { if (sym1->num < sym2->num) return -1; - if (sym1->num > sym2->num) return +1; - return 0; - } - if (sym1->str == NULL) return -1; - if (sym2->str == NULL) return +1; - return compare_strings(mpl, sym1->str, sym2->str); -} - -/*---------------------------------------------------------------------- --- delete_symbol - delete symbol. --- --- This routine deletes specified symbol. */ - -void delete_symbol -( MPL *mpl, - SYMBOL *sym /* destroyed */ -) -{ xassert(sym != NULL); - if (sym->str != NULL) delete_string(mpl, sym->str); - dmp_free_atom(mpl->symbols, sym, sizeof(SYMBOL)); - return; -} - -/*---------------------------------------------------------------------- --- format_symbol - format symbol for displaying or printing. --- --- This routine converts specified symbol to a charater string, which --- is suitable for displaying or printing. --- --- The resultant string is never longer than 255 characters. If it gets --- longer, it is truncated from the right and appended by dots. */ - -char *format_symbol -( MPL *mpl, - SYMBOL *sym /* not changed */ -) -{ char *buf = mpl->sym_buf; - xassert(sym != NULL); - if (sym->str == NULL) - sprintf(buf, "%.*g", DBL_DIG, sym->num); - else - { char str[MAX_LENGTH+1]; - int quoted, j, len; - fetch_string(mpl, sym->str, str); - if (!(isalpha((unsigned char)str[0]) || str[0] == '_')) - quoted = 1; - else - { quoted = 0; - for (j = 1; str[j] != '\0'; j++) - { if (!(isalnum((unsigned char)str[j]) || - strchr("+-._", (unsigned char)str[j]) != NULL)) - { quoted = 1; - break; - } - } - } -# define safe_append(c) \ - (void)(len < 255 ? (buf[len++] = (char)(c)) : 0) - buf[0] = '\0', len = 0; - if (quoted) safe_append('\''); - for (j = 0; str[j] != '\0'; j++) - { if (quoted && str[j] == '\'') safe_append('\''); - safe_append(str[j]); - } - if (quoted) safe_append('\''); -# undef safe_append - buf[len] = '\0'; - if (len == 255) strcpy(buf+252, "..."); - } - xassert(strlen(buf) <= 255); - return buf; -} - -/*---------------------------------------------------------------------- --- concat_symbols - concatenate one symbol with another. --- --- This routine concatenates values of two given symbols and assigns --- the resultant character string to a new symbol, which is returned on --- exit. Both original symbols are destroyed. */ - -SYMBOL *concat_symbols -( MPL *mpl, - SYMBOL *sym1, /* destroyed */ - SYMBOL *sym2 /* destroyed */ -) -{ char str1[MAX_LENGTH+1], str2[MAX_LENGTH+1]; - xassert(MAX_LENGTH >= DBL_DIG + DBL_DIG); - if (sym1->str == NULL) - sprintf(str1, "%.*g", DBL_DIG, sym1->num); - else - fetch_string(mpl, sym1->str, str1); - if (sym2->str == NULL) - sprintf(str2, "%.*g", DBL_DIG, sym2->num); - else - fetch_string(mpl, sym2->str, str2); - if (strlen(str1) + strlen(str2) > MAX_LENGTH) - { char buf[255+1]; - strcpy(buf, format_symbol(mpl, sym1)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s & %s; resultant symbol exceeds %d characters", - buf, format_symbol(mpl, sym2), MAX_LENGTH); - } - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - return create_symbol_str(mpl, create_string(mpl, strcat(str1, - str2))); -} - -/**********************************************************************/ -/* * * N-TUPLES * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- create_tuple - create n-tuple. --- --- This routine creates a n-tuple, which initially has no components, --- i.e. which is 0-tuple. */ - -TUPLE *create_tuple(MPL *mpl) -{ TUPLE *tuple; - xassert(mpl == mpl); - tuple = NULL; - return tuple; -} - -/*---------------------------------------------------------------------- --- expand_tuple - append symbol to n-tuple. --- --- This routine expands n-tuple appending to it a given symbol, which --- becomes its new last component. */ - -TUPLE *expand_tuple -( MPL *mpl, - TUPLE *tuple, /* destroyed */ - SYMBOL *sym /* destroyed */ -) -{ TUPLE *tail, *temp; - xassert(sym != NULL); - /* create a new component */ - tail = dmp_get_atom(mpl->tuples, sizeof(TUPLE)); - tail->sym = sym; - tail->next = NULL; - /* and append it to the component list */ - if (tuple == NULL) - tuple = tail; - else - { for (temp = tuple; temp->next != NULL; temp = temp->next); - temp->next = tail; - } - return tuple; -} - -/*---------------------------------------------------------------------- --- tuple_dimen - determine dimension of n-tuple. --- --- This routine returns dimension of n-tuple, i.e. number of components --- in the n-tuple. */ - -int tuple_dimen -( MPL *mpl, - TUPLE *tuple /* not changed */ -) -{ TUPLE *temp; - int dim = 0; - xassert(mpl == mpl); - for (temp = tuple; temp != NULL; temp = temp->next) dim++; - return dim; -} - -/*---------------------------------------------------------------------- --- copy_tuple - make copy of n-tuple. --- --- This routine returns an exact copy of n-tuple. */ - -TUPLE *copy_tuple -( MPL *mpl, - TUPLE *tuple /* not changed */ -) -{ TUPLE *head, *tail; - if (tuple == NULL) - head = NULL; - else - { head = tail = dmp_get_atom(mpl->tuples, sizeof(TUPLE)); - for (; tuple != NULL; tuple = tuple->next) - { xassert(tuple->sym != NULL); - tail->sym = copy_symbol(mpl, tuple->sym); - if (tuple->next != NULL) -tail = (tail->next = dmp_get_atom(mpl->tuples, sizeof(TUPLE))); - } - tail->next = NULL; - } - return head; -} - -/*---------------------------------------------------------------------- --- compare_tuples - compare one n-tuple with another. --- --- This routine compares two given n-tuples, which must have the same --- dimension (not checked for the sake of efficiency), and returns one --- of the following codes: --- --- = 0 - both n-tuples are identical; --- < 0 - the first n-tuple precedes the second one; --- > 0 - the first n-tuple follows the second one. --- --- Note that the linear order, in which n-tuples follow each other, is --- implementation-dependent. It may be not an alphabetical order. */ - -int compare_tuples -( MPL *mpl, - TUPLE *tuple1, /* not changed */ - TUPLE *tuple2 /* not changed */ -) -{ TUPLE *item1, *item2; - int ret; - xassert(mpl == mpl); - for (item1 = tuple1, item2 = tuple2; item1 != NULL; - item1 = item1->next, item2 = item2->next) - { xassert(item2 != NULL); - xassert(item1->sym != NULL); - xassert(item2->sym != NULL); - ret = compare_symbols(mpl, item1->sym, item2->sym); - if (ret != 0) return ret; - } - xassert(item2 == NULL); - return 0; -} - -/*---------------------------------------------------------------------- --- build_subtuple - build subtuple of given n-tuple. --- --- This routine builds subtuple, which consists of first dim components --- of given n-tuple. */ - -TUPLE *build_subtuple -( MPL *mpl, - TUPLE *tuple, /* not changed */ - int dim -) -{ TUPLE *head, *temp; - int j; - head = create_tuple(mpl); - for (j = 1, temp = tuple; j <= dim; j++, temp = temp->next) - { xassert(temp != NULL); - head = expand_tuple(mpl, head, copy_symbol(mpl, temp->sym)); - } - return head; -} - -/*---------------------------------------------------------------------- --- delete_tuple - delete n-tuple. --- --- This routine deletes specified n-tuple. */ - -void delete_tuple -( MPL *mpl, - TUPLE *tuple /* destroyed */ -) -{ TUPLE *temp; - while (tuple != NULL) - { temp = tuple; - tuple = temp->next; - xassert(temp->sym != NULL); - delete_symbol(mpl, temp->sym); - dmp_free_atom(mpl->tuples, temp, sizeof(TUPLE)); - } - return; -} - -/*---------------------------------------------------------------------- --- format_tuple - format n-tuple for displaying or printing. --- --- This routine converts specified n-tuple to a character string, which --- is suitable for displaying or printing. --- --- The resultant string is never longer than 255 characters. If it gets --- longer, it is truncated from the right and appended by dots. */ - -char *format_tuple -( MPL *mpl, - int c, - TUPLE *tuple /* not changed */ -) -{ TUPLE *temp; - int dim, j, len; - char *buf = mpl->tup_buf, str[255+1], *save; -# define safe_append(c) \ - (void)(len < 255 ? (buf[len++] = (char)(c)) : 0) - buf[0] = '\0', len = 0; - dim = tuple_dimen(mpl, tuple); - if (c == '[' && dim > 0) safe_append('['); - if (c == '(' && dim > 1) safe_append('('); - for (temp = tuple; temp != NULL; temp = temp->next) - { if (temp != tuple) safe_append(','); - xassert(temp->sym != NULL); - save = mpl->sym_buf; - mpl->sym_buf = str; - format_symbol(mpl, temp->sym); - mpl->sym_buf = save; - xassert(strlen(str) < sizeof(str)); - for (j = 0; str[j] != '\0'; j++) safe_append(str[j]); - } - if (c == '[' && dim > 0) safe_append(']'); - if (c == '(' && dim > 1) safe_append(')'); -# undef safe_append - buf[len] = '\0'; - if (len == 255) strcpy(buf+252, "..."); - xassert(strlen(buf) <= 255); - return buf; -} - -/**********************************************************************/ -/* * * ELEMENTAL SETS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- create_elemset - create elemental set. --- --- This routine creates an elemental set, whose members are n-tuples of --- specified dimension. Being created the set is initially empty. */ - -ELEMSET *create_elemset(MPL *mpl, int dim) -{ ELEMSET *set; - xassert(dim > 0); - set = create_array(mpl, A_NONE, dim); - return set; -} - -/*---------------------------------------------------------------------- --- find_tuple - check if elemental set contains given n-tuple. --- --- This routine finds given n-tuple in specified elemental set in order --- to check if the set contains that n-tuple. If the n-tuple is found, --- the routine returns pointer to corresponding array member. Otherwise --- null pointer is returned. */ - -MEMBER *find_tuple -( MPL *mpl, - ELEMSET *set, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ xassert(set != NULL); - xassert(set->type == A_NONE); - xassert(set->dim == tuple_dimen(mpl, tuple)); - return find_member(mpl, set, tuple); -} - -/*---------------------------------------------------------------------- --- add_tuple - add new n-tuple to elemental set. --- --- This routine adds given n-tuple to specified elemental set. --- --- For the sake of efficiency this routine doesn't check whether the --- set already contains the same n-tuple or not. Therefore the calling --- program should use the routine find_tuple (if necessary) in order to --- make sure that the given n-tuple is not contained in the set, since --- duplicate n-tuples within the same set are not allowed. */ - -MEMBER *add_tuple -( MPL *mpl, - ELEMSET *set, /* modified */ - TUPLE *tuple /* destroyed */ -) -{ MEMBER *memb; - xassert(set != NULL); - xassert(set->type == A_NONE); - xassert(set->dim == tuple_dimen(mpl, tuple)); - memb = add_member(mpl, set, tuple); - memb->value.none = NULL; - return memb; -} - -/*---------------------------------------------------------------------- --- check_then_add - check and add new n-tuple to elemental set. --- --- This routine is equivalent to the routine add_tuple except that it --- does check for duplicate n-tuples. */ - -MEMBER *check_then_add -( MPL *mpl, - ELEMSET *set, /* modified */ - TUPLE *tuple /* destroyed */ -) -{ if (find_tuple(mpl, set, tuple) != NULL) - error(mpl, "duplicate tuple %s detected", format_tuple(mpl, - '(', tuple)); - return add_tuple(mpl, set, tuple); -} - -/*---------------------------------------------------------------------- --- copy_elemset - make copy of elemental set. --- --- This routine makes an exact copy of elemental set. */ - -ELEMSET *copy_elemset -( MPL *mpl, - ELEMSET *set /* not changed */ -) -{ ELEMSET *copy; - MEMBER *memb; - xassert(set != NULL); - xassert(set->type == A_NONE); - xassert(set->dim > 0); - copy = create_elemset(mpl, set->dim); - for (memb = set->head; memb != NULL; memb = memb->next) - add_tuple(mpl, copy, copy_tuple(mpl, memb->tuple)); - return copy; -} - -/*---------------------------------------------------------------------- --- delete_elemset - delete elemental set. --- --- This routine deletes specified elemental set. */ - -void delete_elemset -( MPL *mpl, - ELEMSET *set /* destroyed */ -) -{ xassert(set != NULL); - xassert(set->type == A_NONE); - delete_array(mpl, set); - return; -} - -/*---------------------------------------------------------------------- --- arelset_size - compute size of "arithmetic" elemental set. --- --- This routine computes the size of "arithmetic" elemental set, which --- is specified in the form of arithmetic progression: --- --- { t0 .. tf by dt }. --- --- The size is computed using the formula: --- --- n = max(0, floor((tf - t0) / dt) + 1). */ - -int arelset_size(MPL *mpl, double t0, double tf, double dt) -{ double temp; - if (dt == 0.0) - error(mpl, "%.*g .. %.*g by %.*g; zero stride not allowed", - DBL_DIG, t0, DBL_DIG, tf, DBL_DIG, dt); - if (tf > 0.0 && t0 < 0.0 && tf > + 0.999 * DBL_MAX + t0) - temp = +DBL_MAX; - else if (tf < 0.0 && t0 > 0.0 && tf < - 0.999 * DBL_MAX + t0) - temp = -DBL_MAX; - else - temp = tf - t0; - if (fabs(dt) < 1.0 && fabs(temp) > (0.999 * DBL_MAX) * fabs(dt)) - { if (temp > 0.0 && dt > 0.0 || temp < 0.0 && dt < 0.0) - temp = +DBL_MAX; - else - temp = 0.0; - } - else - { temp = floor(temp / dt) + 1.0; - if (temp < 0.0) temp = 0.0; - } - xassert(temp >= 0.0); - if (temp > (double)(INT_MAX - 1)) - error(mpl, "%.*g .. %.*g by %.*g; set too large", - DBL_DIG, t0, DBL_DIG, tf, DBL_DIG, dt); - return (int)(temp + 0.5); -} - -/*---------------------------------------------------------------------- --- arelset_member - compute member of "arithmetic" elemental set. --- --- This routine returns a numeric value of symbol, which is equivalent --- to j-th member of given "arithmetic" elemental set specified in the --- form of arithmetic progression: --- --- { t0 .. tf by dt }. --- --- The symbol value is computed with the formula: --- --- j-th member = t0 + (j - 1) * dt, --- --- The number j must satisfy to the restriction 1 <= j <= n, where n is --- the set size computed by the routine arelset_size. */ - -double arelset_member(MPL *mpl, double t0, double tf, double dt, int j) -{ xassert(1 <= j && j <= arelset_size(mpl, t0, tf, dt)); - return t0 + (double)(j - 1) * dt; -} - -/*---------------------------------------------------------------------- --- create_arelset - create "arithmetic" elemental set. --- --- This routine creates "arithmetic" elemental set, which is specified --- in the form of arithmetic progression: --- --- { t0 .. tf by dt }. --- --- Components of this set are 1-tuples. */ - -ELEMSET *create_arelset(MPL *mpl, double t0, double tf, double dt) -{ ELEMSET *set; - int j, n; - set = create_elemset(mpl, 1); - n = arelset_size(mpl, t0, tf, dt); - for (j = 1; j <= n; j++) - { add_tuple - ( mpl, - set, - expand_tuple - ( mpl, - create_tuple(mpl), - create_symbol_num - ( mpl, - arelset_member(mpl, t0, tf, dt, j) - ) - ) - ); - } - return set; -} - -/*---------------------------------------------------------------------- --- set_union - union of two elemental sets. --- --- This routine computes the union: --- --- X U Y = { j | (j in X) or (j in Y) }, --- --- where X and Y are given elemental sets (destroyed on exit). */ - -ELEMSET *set_union -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -) -{ MEMBER *memb; - xassert(X != NULL); - xassert(X->type == A_NONE); - xassert(X->dim > 0); - xassert(Y != NULL); - xassert(Y->type == A_NONE); - xassert(Y->dim > 0); - xassert(X->dim == Y->dim); - for (memb = Y->head; memb != NULL; memb = memb->next) - { if (find_tuple(mpl, X, memb->tuple) == NULL) - add_tuple(mpl, X, copy_tuple(mpl, memb->tuple)); - } - delete_elemset(mpl, Y); - return X; -} - -/*---------------------------------------------------------------------- --- set_diff - difference between two elemental sets. --- --- This routine computes the difference: --- --- X \ Y = { j | (j in X) and (j not in Y) }, --- --- where X and Y are given elemental sets (destroyed on exit). */ - -ELEMSET *set_diff -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -) -{ ELEMSET *Z; - MEMBER *memb; - xassert(X != NULL); - xassert(X->type == A_NONE); - xassert(X->dim > 0); - xassert(Y != NULL); - xassert(Y->type == A_NONE); - xassert(Y->dim > 0); - xassert(X->dim == Y->dim); - Z = create_elemset(mpl, X->dim); - for (memb = X->head; memb != NULL; memb = memb->next) - { if (find_tuple(mpl, Y, memb->tuple) == NULL) - add_tuple(mpl, Z, copy_tuple(mpl, memb->tuple)); - } - delete_elemset(mpl, X); - delete_elemset(mpl, Y); - return Z; -} - -/*---------------------------------------------------------------------- --- set_symdiff - symmetric difference between two elemental sets. --- --- This routine computes the symmetric difference: --- --- X (+) Y = (X \ Y) U (Y \ X), --- --- where X and Y are given elemental sets (destroyed on exit). */ - -ELEMSET *set_symdiff -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -) -{ ELEMSET *Z; - MEMBER *memb; - xassert(X != NULL); - xassert(X->type == A_NONE); - xassert(X->dim > 0); - xassert(Y != NULL); - xassert(Y->type == A_NONE); - xassert(Y->dim > 0); - xassert(X->dim == Y->dim); - /* Z := X \ Y */ - Z = create_elemset(mpl, X->dim); - for (memb = X->head; memb != NULL; memb = memb->next) - { if (find_tuple(mpl, Y, memb->tuple) == NULL) - add_tuple(mpl, Z, copy_tuple(mpl, memb->tuple)); - } - /* Z := Z U (Y \ X) */ - for (memb = Y->head; memb != NULL; memb = memb->next) - { if (find_tuple(mpl, X, memb->tuple) == NULL) - add_tuple(mpl, Z, copy_tuple(mpl, memb->tuple)); - } - delete_elemset(mpl, X); - delete_elemset(mpl, Y); - return Z; -} - -/*---------------------------------------------------------------------- --- set_inter - intersection of two elemental sets. --- --- This routine computes the intersection: --- --- X ^ Y = { j | (j in X) and (j in Y) }, --- --- where X and Y are given elemental sets (destroyed on exit). */ - -ELEMSET *set_inter -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -) -{ ELEMSET *Z; - MEMBER *memb; - xassert(X != NULL); - xassert(X->type == A_NONE); - xassert(X->dim > 0); - xassert(Y != NULL); - xassert(Y->type == A_NONE); - xassert(Y->dim > 0); - xassert(X->dim == Y->dim); - Z = create_elemset(mpl, X->dim); - for (memb = X->head; memb != NULL; memb = memb->next) - { if (find_tuple(mpl, Y, memb->tuple) != NULL) - add_tuple(mpl, Z, copy_tuple(mpl, memb->tuple)); - } - delete_elemset(mpl, X); - delete_elemset(mpl, Y); - return Z; -} - -/*---------------------------------------------------------------------- --- set_cross - cross (Cartesian) product of two elemental sets. --- --- This routine computes the cross (Cartesian) product: --- --- X x Y = { (i,j) | (i in X) and (j in Y) }, --- --- where X and Y are given elemental sets (destroyed on exit). */ - -ELEMSET *set_cross -( MPL *mpl, - ELEMSET *X, /* destroyed */ - ELEMSET *Y /* destroyed */ -) -{ ELEMSET *Z; - MEMBER *memx, *memy; - TUPLE *tuple, *temp; - xassert(X != NULL); - xassert(X->type == A_NONE); - xassert(X->dim > 0); - xassert(Y != NULL); - xassert(Y->type == A_NONE); - xassert(Y->dim > 0); - Z = create_elemset(mpl, X->dim + Y->dim); - for (memx = X->head; memx != NULL; memx = memx->next) - { for (memy = Y->head; memy != NULL; memy = memy->next) - { tuple = copy_tuple(mpl, memx->tuple); - for (temp = memy->tuple; temp != NULL; temp = temp->next) - tuple = expand_tuple(mpl, tuple, copy_symbol(mpl, - temp->sym)); - add_tuple(mpl, Z, tuple); - } - } - delete_elemset(mpl, X); - delete_elemset(mpl, Y); - return Z; -} - -/**********************************************************************/ -/* * * ELEMENTAL VARIABLES * * */ -/**********************************************************************/ - -/* (there are no specific routines for elemental variables) */ - -/**********************************************************************/ -/* * * LINEAR FORMS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- constant_term - create constant term. --- --- This routine creates the linear form, which is a constant term. */ - -FORMULA *constant_term(MPL *mpl, double coef) -{ FORMULA *form; - if (coef == 0.0) - form = NULL; - else - { form = dmp_get_atom(mpl->formulae, sizeof(FORMULA)); - form->coef = coef; - form->var = NULL; - form->next = NULL; - } - return form; -} - -/*---------------------------------------------------------------------- --- single_variable - create single variable. --- --- This routine creates the linear form, which is a single elemental --- variable. */ - -FORMULA *single_variable -( MPL *mpl, - ELEMVAR *var /* referenced */ -) -{ FORMULA *form; - xassert(var != NULL); - form = dmp_get_atom(mpl->formulae, sizeof(FORMULA)); - form->coef = 1.0; - form->var = var; - form->next = NULL; - return form; -} - -/*---------------------------------------------------------------------- --- copy_formula - make copy of linear form. --- --- This routine returns an exact copy of linear form. */ - -FORMULA *copy_formula -( MPL *mpl, - FORMULA *form /* not changed */ -) -{ FORMULA *head, *tail; - if (form == NULL) - head = NULL; - else - { head = tail = dmp_get_atom(mpl->formulae, sizeof(FORMULA)); - for (; form != NULL; form = form->next) - { tail->coef = form->coef; - tail->var = form->var; - if (form->next != NULL) -tail = (tail->next = dmp_get_atom(mpl->formulae, sizeof(FORMULA))); - } - tail->next = NULL; - } - return head; -} - -/*---------------------------------------------------------------------- --- delete_formula - delete linear form. --- --- This routine deletes specified linear form. */ - -void delete_formula -( MPL *mpl, - FORMULA *form /* destroyed */ -) -{ FORMULA *temp; - while (form != NULL) - { temp = form; - form = form->next; - dmp_free_atom(mpl->formulae, temp, sizeof(FORMULA)); - } - return; -} - -/*---------------------------------------------------------------------- --- linear_comb - linear combination of two linear forms. --- --- This routine computes the linear combination: --- --- a * fx + b * fy, --- --- where a and b are numeric coefficients, fx and fy are linear forms --- (destroyed on exit). */ - -FORMULA *linear_comb -( MPL *mpl, - double a, FORMULA *fx, /* destroyed */ - double b, FORMULA *fy /* destroyed */ -) -{ FORMULA *form = NULL, *term, *temp; - double c0 = 0.0; - for (term = fx; term != NULL; term = term->next) - { if (term->var == NULL) - c0 = fp_add(mpl, c0, fp_mul(mpl, a, term->coef)); - else - term->var->temp = - fp_add(mpl, term->var->temp, fp_mul(mpl, a, term->coef)); - } - for (term = fy; term != NULL; term = term->next) - { if (term->var == NULL) - c0 = fp_add(mpl, c0, fp_mul(mpl, b, term->coef)); - else - term->var->temp = - fp_add(mpl, term->var->temp, fp_mul(mpl, b, term->coef)); - } - for (term = fx; term != NULL; term = term->next) - { if (term->var != NULL && term->var->temp != 0.0) - { temp = dmp_get_atom(mpl->formulae, sizeof(FORMULA)); - temp->coef = term->var->temp, temp->var = term->var; - temp->next = form, form = temp; - term->var->temp = 0.0; - } - } - for (term = fy; term != NULL; term = term->next) - { if (term->var != NULL && term->var->temp != 0.0) - { temp = dmp_get_atom(mpl->formulae, sizeof(FORMULA)); - temp->coef = term->var->temp, temp->var = term->var; - temp->next = form, form = temp; - term->var->temp = 0.0; - } - } - if (c0 != 0.0) - { temp = dmp_get_atom(mpl->formulae, sizeof(FORMULA)); - temp->coef = c0, temp->var = NULL; - temp->next = form, form = temp; - } - delete_formula(mpl, fx); - delete_formula(mpl, fy); - return form; -} - -/*---------------------------------------------------------------------- --- remove_constant - remove constant term from linear form. --- --- This routine removes constant term from linear form and stores its --- value to given location. */ - -FORMULA *remove_constant -( MPL *mpl, - FORMULA *form, /* destroyed */ - double *coef /* modified */ -) -{ FORMULA *head = NULL, *temp; - *coef = 0.0; - while (form != NULL) - { temp = form; - form = form->next; - if (temp->var == NULL) - { /* constant term */ - *coef = fp_add(mpl, *coef, temp->coef); - dmp_free_atom(mpl->formulae, temp, sizeof(FORMULA)); - } - else - { /* linear term */ - temp->next = head; - head = temp; - } - } - return head; -} - -/*---------------------------------------------------------------------- --- reduce_terms - reduce identical terms in linear form. --- --- This routine reduces identical terms in specified linear form. */ - -FORMULA *reduce_terms -( MPL *mpl, - FORMULA *form /* destroyed */ -) -{ FORMULA *term, *next_term; - double c0 = 0.0; - for (term = form; term != NULL; term = term->next) - { if (term->var == NULL) - c0 = fp_add(mpl, c0, term->coef); - else - term->var->temp = fp_add(mpl, term->var->temp, term->coef); - } - next_term = form, form = NULL; - for (term = next_term; term != NULL; term = next_term) - { next_term = term->next; - if (term->var == NULL && c0 != 0.0) - { term->coef = c0, c0 = 0.0; - term->next = form, form = term; - } - else if (term->var != NULL && term->var->temp != 0.0) - { term->coef = term->var->temp, term->var->temp = 0.0; - term->next = form, form = term; - } - else - dmp_free_atom(mpl->formulae, term, sizeof(FORMULA)); - } - return form; -} - -/**********************************************************************/ -/* * * ELEMENTAL CONSTRAINTS * * */ -/**********************************************************************/ - -/* (there are no specific routines for elemental constraints) */ - -/**********************************************************************/ -/* * * GENERIC VALUES * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- delete_value - delete generic value. --- --- This routine deletes specified generic value. --- --- NOTE: The generic value to be deleted must be valid. */ - -void delete_value -( MPL *mpl, - int type, - VALUE *value /* content destroyed */ -) -{ xassert(value != NULL); - switch (type) - { case A_NONE: - value->none = NULL; - break; - case A_NUMERIC: - value->num = 0.0; - break; - case A_SYMBOLIC: - delete_symbol(mpl, value->sym), value->sym = NULL; - break; - case A_LOGICAL: - value->bit = 0; - break; - case A_TUPLE: - delete_tuple(mpl, value->tuple), value->tuple = NULL; - break; - case A_ELEMSET: - delete_elemset(mpl, value->set), value->set = NULL; - break; - case A_ELEMVAR: - value->var = NULL; - break; - case A_FORMULA: - delete_formula(mpl, value->form), value->form = NULL; - break; - case A_ELEMCON: - value->con = NULL; - break; - default: - xassert(type != type); - } - return; -} - -/**********************************************************************/ -/* * * SYMBOLICALLY INDEXED ARRAYS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- create_array - create array. --- --- This routine creates an array of specified type and dimension. Being --- created the array is initially empty. --- --- The type indicator determines generic values, which can be assigned --- to the array members: --- --- A_NONE - none (members have no assigned values) --- A_NUMERIC - floating-point numbers --- A_SYMBOLIC - symbols --- A_ELEMSET - elemental sets --- A_ELEMVAR - elemental variables --- A_ELEMCON - elemental constraints --- --- The dimension may be 0, in which case the array consists of the only --- member (such arrays represent 0-dimensional objects). */ - -ARRAY *create_array(MPL *mpl, int type, int dim) -{ ARRAY *array; - xassert(type == A_NONE || type == A_NUMERIC || - type == A_SYMBOLIC || type == A_ELEMSET || - type == A_ELEMVAR || type == A_ELEMCON); - xassert(dim >= 0); - array = dmp_get_atom(mpl->arrays, sizeof(ARRAY)); - array->type = type; - array->dim = dim; - array->size = 0; - array->head = NULL; - array->tail = NULL; - array->tree = NULL; - array->prev = NULL; - array->next = mpl->a_list; - /* include the array in the global array list */ - if (array->next != NULL) array->next->prev = array; - mpl->a_list = array; - return array; -} - -/*---------------------------------------------------------------------- --- find_member - find array member with given n-tuple. --- --- This routine finds an array member, which has given n-tuple. If the --- array is short, the linear search is used. Otherwise the routine --- autimatically creates the search tree (i.e. the array index) to find --- members for logarithmic time. */ - -static int compare_member_tuples(void *info, const void *key1, - const void *key2) -{ /* this is an auxiliary routine used to compare keys, which are - n-tuples assigned to array members */ - return compare_tuples((MPL *)info, (TUPLE *)key1, (TUPLE *)key2); -} - -MEMBER *find_member -( MPL *mpl, - ARRAY *array, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ MEMBER *memb; - xassert(array != NULL); - /* the n-tuple must have the same dimension as the array */ - xassert(tuple_dimen(mpl, tuple) == array->dim); - /* if the array is large enough, create the search tree and index - all existing members of the array */ - if (array->size > 30 && array->tree == NULL) - { array->tree = avl_create_tree(compare_member_tuples, mpl); - for (memb = array->head; memb != NULL; memb = memb->next) -avl_set_node_link(avl_insert_node(array->tree, memb->tuple), - (void *)memb); - } - /* find a member, which has the given tuple */ - if (array->tree == NULL) - { /* the search tree doesn't exist; use the linear search */ - for (memb = array->head; memb != NULL; memb = memb->next) - if (compare_tuples(mpl, memb->tuple, tuple) == 0) break; - } - else - { /* the search tree exists; use the binary search */ - AVLNODE *node; - node = avl_find_node(array->tree, tuple); -memb = (MEMBER *)(node == NULL ? NULL : avl_get_node_link(node)); - } - return memb; -} - -/*---------------------------------------------------------------------- --- add_member - add new member to array. --- --- This routine creates a new member with given n-tuple and adds it to --- specified array. --- --- For the sake of efficiency this routine doesn't check whether the --- array already contains a member with the given n-tuple or not. Thus, --- if necessary, the calling program should use the routine find_member --- in order to be sure that the array contains no member with the same --- n-tuple, because members with duplicate n-tuples are not allowed. --- --- This routine assigns no generic value to the new member, because the --- calling program must do that. */ - -MEMBER *add_member -( MPL *mpl, - ARRAY *array, /* modified */ - TUPLE *tuple /* destroyed */ -) -{ MEMBER *memb; - xassert(array != NULL); - /* the n-tuple must have the same dimension as the array */ - xassert(tuple_dimen(mpl, tuple) == array->dim); - /* create new member */ - memb = dmp_get_atom(mpl->members, sizeof(MEMBER)); - memb->tuple = tuple; - memb->next = NULL; - memset(&memb->value, '?', sizeof(VALUE)); - /* and append it to the member list */ - array->size++; - if (array->head == NULL) - array->head = memb; - else - array->tail->next = memb; - array->tail = memb; - /* if the search tree exists, index the new member */ - if (array->tree != NULL) -avl_set_node_link(avl_insert_node(array->tree, memb->tuple), - (void *)memb); - return memb; -} - -/*---------------------------------------------------------------------- --- delete_array - delete array. --- --- This routine deletes specified array. --- --- Generic values assigned to the array members are not deleted by this --- routine. The calling program itself must delete all assigned generic --- values before deleting the array. */ - -void delete_array -( MPL *mpl, - ARRAY *array /* destroyed */ -) -{ MEMBER *memb; - xassert(array != NULL); - /* delete all existing array members */ - while (array->head != NULL) - { memb = array->head; - array->head = memb->next; - delete_tuple(mpl, memb->tuple); - dmp_free_atom(mpl->members, memb, sizeof(MEMBER)); - } - /* if the search tree exists, also delete it */ - if (array->tree != NULL) avl_delete_tree(array->tree); - /* remove the array from the global array list */ - if (array->prev == NULL) - mpl->a_list = array->next; - else - array->prev->next = array->next; - if (array->next == NULL) - ; - else - array->next->prev = array->prev; - /* delete the array descriptor */ - dmp_free_atom(mpl->arrays, array, sizeof(ARRAY)); - return; -} - -/**********************************************************************/ -/* * * DOMAINS AND DUMMY INDICES * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- assign_dummy_index - assign new value to dummy index. --- --- This routine assigns new value to specified dummy index and, that is --- important, invalidates all temporary resultant values, which depends --- on that dummy index. */ - -void assign_dummy_index -( MPL *mpl, - DOMAIN_SLOT *slot, /* modified */ - SYMBOL *value /* not changed */ -) -{ CODE *leaf, *code; - xassert(slot != NULL); - xassert(value != NULL); - /* delete the current value assigned to the dummy index */ - if (slot->value != NULL) - { /* if the current value and the new one are identical, actual - assignment is not needed */ - if (compare_symbols(mpl, slot->value, value) == 0) goto done; - /* delete a symbol, which is the current value */ - delete_symbol(mpl, slot->value), slot->value = NULL; - } - /* now walk through all the pseudo-codes with op = O_INDEX, which - refer to the dummy index to be changed (these pseudo-codes are - leaves in the forest of *all* expressions in the database) */ - for (leaf = slot->list; leaf != NULL; leaf = leaf->arg.index. - next) - { xassert(leaf->op == O_INDEX); - /* invalidate all resultant values, which depend on the dummy - index, walking from the current leaf toward the root of the - corresponding expression tree */ - for (code = leaf; code != NULL; code = code->up) - { if (code->valid) - { /* invalidate and delete resultant value */ - code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - } - } - /* assign new value to the dummy index */ - slot->value = copy_symbol(mpl, value); -done: return; -} - -/*---------------------------------------------------------------------- --- update_dummy_indices - update current values of dummy indices. --- --- This routine assigns components of "backup" n-tuple to dummy indices --- of specified domain block. If no "backup" n-tuple is defined for the --- domain block, values of the dummy indices remain untouched. */ - -void update_dummy_indices -( MPL *mpl, - DOMAIN_BLOCK *block /* not changed */ -) -{ DOMAIN_SLOT *slot; - TUPLE *temp; - if (block->backup != NULL) - { for (slot = block->list, temp = block->backup; slot != NULL; - slot = slot->next, temp = temp->next) - { xassert(temp != NULL); - xassert(temp->sym != NULL); - assign_dummy_index(mpl, slot, temp->sym); - } - } - return; -} - -/*---------------------------------------------------------------------- --- enter_domain_block - enter domain block. --- --- Let specified domain block have the form: --- --- { ..., (j1, j2, ..., jn) in J, ... } --- --- where j1, j2, ..., jn are dummy indices, J is a basic set. --- --- This routine does the following: --- --- 1. Checks if the given n-tuple is a member of the basic set J. Note --- that J being *out of the scope* of the domain block cannot depend --- on the dummy indices in the same and inner domain blocks, so it --- can be computed before the dummy indices are assigned new values. --- If this check fails, the routine returns with non-zero code. --- --- 2. Saves current values of the dummy indices j1, j2, ..., jn. --- --- 3. Assigns new values, which are components of the given n-tuple, to --- the dummy indices j1, j2, ..., jn. If dimension of the n-tuple is --- larger than n, its extra components n+1, n+2, ... are not used. --- --- 4. Calls the formal routine func which either enters the next domain --- block or evaluates some code within the domain scope. --- --- 5. Restores former values of the dummy indices j1, j2, ..., jn. --- --- Since current values assigned to the dummy indices on entry to this --- routine are restored on exit, the formal routine func is allowed to --- call this routine recursively. */ - -int enter_domain_block -( MPL *mpl, - DOMAIN_BLOCK *block, /* not changed */ - TUPLE *tuple, /* not changed */ - void *info, void (*func)(MPL *mpl, void *info) -) -{ TUPLE *backup; - int ret = 0; - /* check if the given n-tuple is a member of the basic set */ - xassert(block->code != NULL); - if (!is_member(mpl, block->code, tuple)) - { ret = 1; - goto done; - } - /* save reference to "backup" n-tuple, which was used to assign - current values of the dummy indices (it is sufficient to save - reference, not value, because that n-tuple is defined in some - outer level of recursion and therefore cannot be changed on - this and deeper recursive calls) */ - backup = block->backup; - /* set up new "backup" n-tuple, which defines new values of the - dummy indices */ - block->backup = tuple; - /* assign new values to the dummy indices */ - update_dummy_indices(mpl, block); - /* call the formal routine that does the rest part of the job */ - func(mpl, info); - /* restore reference to the former "backup" n-tuple */ - block->backup = backup; - /* restore former values of the dummy indices; note that if the - domain block just escaped has no other active instances which - may exist due to recursion (it is indicated by a null pointer - to the former n-tuple), former values of the dummy indices are - undefined; therefore in this case the routine keeps currently - assigned values of the dummy indices that involves keeping all - dependent temporary results and thereby, if this domain block - is not used recursively, allows improving efficiency */ - update_dummy_indices(mpl, block); -done: return ret; -} - -/*---------------------------------------------------------------------- --- eval_within_domain - perform evaluation within domain scope. --- --- This routine assigns new values (symbols) to all dummy indices of --- specified domain and calls the formal routine func, which is used to --- evaluate some code in the domain scope. Each free dummy index in the --- domain is assigned a value specified in the corresponding component --- of given n-tuple. Non-free dummy indices are assigned values, which --- are computed by this routine. --- --- Number of components in the given n-tuple must be the same as number --- of free indices in the domain. --- --- If the given n-tuple is not a member of the domain set, the routine --- func is not called, and non-zero code is returned. --- --- For the sake of convenience it is allowed to specify domain as NULL --- (then n-tuple also must be 0-tuple, i.e. empty), in which case this --- routine just calls the routine func and returns zero. --- --- This routine allows recursive calls from the routine func providing --- correct values of dummy indices for each instance. --- --- NOTE: The n-tuple passed to this routine must not be changed by any --- other routines called from the formal routine func until this --- routine has returned. */ - -struct eval_domain_info -{ /* working info used by the routine eval_within_domain */ - DOMAIN *domain; - /* domain, which has to be entered */ - DOMAIN_BLOCK *block; - /* domain block, which is currently processed */ - TUPLE *tuple; - /* tail of original n-tuple, whose components have to be assigned - to free dummy indices in the current domain block */ - void *info; - /* transit pointer passed to the formal routine func */ - void (*func)(MPL *mpl, void *info); - /* routine, which has to be executed in the domain scope */ - int failure; - /* this flag indicates that given n-tuple is not a member of the - domain set */ -}; - -static void eval_domain_func(MPL *mpl, void *_my_info) -{ /* this routine recursively enters into the domain scope and then - calls the routine func */ - struct eval_domain_info *my_info = _my_info; - if (my_info->block != NULL) - { /* the current domain block to be entered exists */ - DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - TUPLE *tuple = NULL, *temp = NULL; - /* save pointer to the current domain block */ - block = my_info->block; - /* and get ready to enter the next block (if it exists) */ - my_info->block = block->next; - /* construct temporary n-tuple, whose components correspond to - dummy indices (slots) of the current domain; components of - the temporary n-tuple that correspond to free dummy indices - are assigned references (not values!) to symbols specified - in the corresponding components of the given n-tuple, while - other components that correspond to non-free dummy indices - are assigned symbolic values computed here */ - for (slot = block->list; slot != NULL; slot = slot->next) - { /* create component that corresponds to the current slot */ - if (tuple == NULL) - tuple = temp = dmp_get_atom(mpl->tuples, sizeof(TUPLE)); - else -temp = (temp->next = dmp_get_atom(mpl->tuples, sizeof(TUPLE))); - if (slot->code == NULL) - { /* dummy index is free; take reference to symbol, which - is specified in the corresponding component of given - n-tuple */ - xassert(my_info->tuple != NULL); - temp->sym = my_info->tuple->sym; - xassert(temp->sym != NULL); - my_info->tuple = my_info->tuple->next; - } - else - { /* dummy index is non-free; compute symbolic value to be - temporarily assigned to the dummy index */ - temp->sym = eval_symbolic(mpl, slot->code); - } - } - temp->next = NULL; - /* enter the current domain block */ - if (enter_domain_block(mpl, block, tuple, my_info, - eval_domain_func)) my_info->failure = 1; - /* delete temporary n-tuple as well as symbols that correspond - to non-free dummy indices (they were computed here) */ - for (slot = block->list; slot != NULL; slot = slot->next) - { xassert(tuple != NULL); - temp = tuple; - tuple = tuple->next; - if (slot->code != NULL) - { /* dummy index is non-free; delete symbolic value */ - delete_symbol(mpl, temp->sym); - } - /* delete component that corresponds to the current slot */ - dmp_free_atom(mpl->tuples, temp, sizeof(TUPLE)); - } - } - else - { /* there are no more domain blocks, i.e. we have reached the - domain scope */ - xassert(my_info->tuple == NULL); - /* check optional predicate specified for the domain */ - if (my_info->domain->code != NULL && !eval_logical(mpl, - my_info->domain->code)) - { /* the predicate is false */ - my_info->failure = 2; - } - else - { /* the predicate is true; do the job */ - my_info->func(mpl, my_info->info); - } - } - return; -} - -int eval_within_domain -( MPL *mpl, - DOMAIN *domain, /* not changed */ - TUPLE *tuple, /* not changed */ - void *info, void (*func)(MPL *mpl, void *info) -) -{ /* this routine performs evaluation within domain scope */ - struct eval_domain_info _my_info, *my_info = &_my_info; - if (domain == NULL) - { xassert(tuple == NULL); - func(mpl, info); - my_info->failure = 0; - } - else - { xassert(tuple != NULL); - my_info->domain = domain; - my_info->block = domain->list; - my_info->tuple = tuple; - my_info->info = info; - my_info->func = func; - my_info->failure = 0; - /* enter the very first domain block */ - eval_domain_func(mpl, my_info); - } - return my_info->failure; -} - -/*---------------------------------------------------------------------- --- loop_within_domain - perform iterations within domain scope. --- --- This routine iteratively assigns new values (symbols) to the dummy --- indices of specified domain by enumerating all n-tuples, which are --- members of the domain set, and for every n-tuple it calls the formal --- routine func to evaluate some code within the domain scope. --- --- If the routine func returns non-zero, enumeration within the domain --- is prematurely terminated. --- --- For the sake of convenience it is allowed to specify domain as NULL, --- in which case this routine just calls the routine func only once and --- returns zero. --- --- This routine allows recursive calls from the routine func providing --- correct values of dummy indices for each instance. */ - -struct loop_domain_info -{ /* working info used by the routine loop_within_domain */ - DOMAIN *domain; - /* domain, which has to be entered */ - DOMAIN_BLOCK *block; - /* domain block, which is currently processed */ - int looping; - /* clearing this flag leads to terminating enumeration */ - void *info; - /* transit pointer passed to the formal routine func */ - int (*func)(MPL *mpl, void *info); - /* routine, which needs to be executed in the domain scope */ -}; - -static void loop_domain_func(MPL *mpl, void *_my_info) -{ /* this routine enumerates all n-tuples in the basic set of the - current domain block, enters recursively into the domain scope - for every n-tuple, and then calls the routine func */ - struct loop_domain_info *my_info = _my_info; - if (my_info->block != NULL) - { /* the current domain block to be entered exists */ - DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - TUPLE *bound; - /* save pointer to the current domain block */ - block = my_info->block; - /* and get ready to enter the next block (if it exists) */ - my_info->block = block->next; - /* compute symbolic values, at which non-free dummy indices of - the current domain block are bound; since that values don't - depend on free dummy indices of the current block, they can - be computed once out of the enumeration loop */ - bound = create_tuple(mpl); - for (slot = block->list; slot != NULL; slot = slot->next) - { if (slot->code != NULL) - bound = expand_tuple(mpl, bound, eval_symbolic(mpl, - slot->code)); - } - /* start enumeration */ - xassert(block->code != NULL); - if (block->code->op == O_DOTS) - { /* the basic set is "arithmetic", in which case it doesn't - need to be computed explicitly */ - TUPLE *tuple; - int n, j; - double t0, tf, dt; - /* compute "parameters" of the basic set */ - t0 = eval_numeric(mpl, block->code->arg.arg.x); - tf = eval_numeric(mpl, block->code->arg.arg.y); - if (block->code->arg.arg.z == NULL) - dt = 1.0; - else - dt = eval_numeric(mpl, block->code->arg.arg.z); - /* determine cardinality of the basic set */ - n = arelset_size(mpl, t0, tf, dt); - /* create dummy 1-tuple for members of the basic set */ - tuple = expand_tuple(mpl, create_tuple(mpl), - create_symbol_num(mpl, 0.0)); - /* in case of "arithmetic" set there is exactly one dummy - index, which cannot be non-free */ - xassert(bound == NULL); - /* walk through 1-tuples of the basic set */ - for (j = 1; j <= n && my_info->looping; j++) - { /* construct dummy 1-tuple for the current member */ - tuple->sym->num = arelset_member(mpl, t0, tf, dt, j); - /* enter the current domain block */ - enter_domain_block(mpl, block, tuple, my_info, - loop_domain_func); - } - /* delete dummy 1-tuple */ - delete_tuple(mpl, tuple); - } - else - { /* the basic set is of general kind, in which case it needs - to be explicitly computed */ - ELEMSET *set; - MEMBER *memb; - TUPLE *temp1, *temp2; - /* compute the basic set */ - set = eval_elemset(mpl, block->code); - /* walk through all n-tuples of the basic set */ - for (memb = set->head; memb != NULL && my_info->looping; - memb = memb->next) - { /* all components of the current n-tuple that correspond - to non-free dummy indices must be feasible; otherwise - the n-tuple is not in the basic set */ - temp1 = memb->tuple; - temp2 = bound; - for (slot = block->list; slot != NULL; slot = slot->next) - { xassert(temp1 != NULL); - if (slot->code != NULL) - { /* non-free dummy index */ - xassert(temp2 != NULL); - if (compare_symbols(mpl, temp1->sym, temp2->sym) - != 0) - { /* the n-tuple is not in the basic set */ - goto skip; - } - temp2 = temp2->next; - } - temp1 = temp1->next; - } - xassert(temp1 == NULL); - xassert(temp2 == NULL); - /* enter the current domain block */ - enter_domain_block(mpl, block, memb->tuple, my_info, - loop_domain_func); -skip: ; - } - /* delete the basic set */ - delete_elemset(mpl, set); - } - /* delete symbolic values binding non-free dummy indices */ - delete_tuple(mpl, bound); - /* restore pointer to the current domain block */ - my_info->block = block; - } - else - { /* there are no more domain blocks, i.e. we have reached the - domain scope */ - /* check optional predicate specified for the domain */ - if (my_info->domain->code != NULL && !eval_logical(mpl, - my_info->domain->code)) - { /* the predicate is false */ - /* nop */; - } - else - { /* the predicate is true; do the job */ - my_info->looping = !my_info->func(mpl, my_info->info); - } - } - return; -} - -void loop_within_domain -( MPL *mpl, - DOMAIN *domain, /* not changed */ - void *info, int (*func)(MPL *mpl, void *info) -) -{ /* this routine performs iterations within domain scope */ - struct loop_domain_info _my_info, *my_info = &_my_info; - if (domain == NULL) - func(mpl, info); - else - { my_info->domain = domain; - my_info->block = domain->list; - my_info->looping = 1; - my_info->info = info; - my_info->func = func; - /* enter the very first domain block */ - loop_domain_func(mpl, my_info); - } - return; -} - -/*---------------------------------------------------------------------- --- out_of_domain - raise domain exception. --- --- This routine is called when a reference is made to a member of some --- model object, but its n-tuple is out of the object domain. */ - -void out_of_domain -( MPL *mpl, - char *name, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ xassert(name != NULL); - xassert(tuple != NULL); - error(mpl, "%s%s out of domain", name, format_tuple(mpl, '[', - tuple)); - /* no return */ -} - -/*---------------------------------------------------------------------- --- get_domain_tuple - obtain current n-tuple from domain. --- --- This routine constructs n-tuple, whose components are current values --- assigned to *free* dummy indices of specified domain. --- --- For the sake of convenience it is allowed to specify domain as NULL, --- in which case this routine returns 0-tuple. --- --- NOTE: This routine must not be called out of domain scope. */ - -TUPLE *get_domain_tuple -( MPL *mpl, - DOMAIN *domain /* not changed */ -) -{ DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - TUPLE *tuple; - tuple = create_tuple(mpl); - if (domain != NULL) - { for (block = domain->list; block != NULL; block = block->next) - { for (slot = block->list; slot != NULL; slot = slot->next) - { if (slot->code == NULL) - { xassert(slot->value != NULL); - tuple = expand_tuple(mpl, tuple, copy_symbol(mpl, - slot->value)); - } - } - } - } - return tuple; -} - -/*---------------------------------------------------------------------- --- clean_domain - clean domain. --- --- This routine cleans specified domain that assumes deleting all stuff --- dynamically allocated during the generation phase. */ - -void clean_domain(MPL *mpl, DOMAIN *domain) -{ DOMAIN_BLOCK *block; - DOMAIN_SLOT *slot; - /* if no domain is specified, do nothing */ - if (domain == NULL) goto done; - /* clean all domain blocks */ - for (block = domain->list; block != NULL; block = block->next) - { /* clean all domain slots */ - for (slot = block->list; slot != NULL; slot = slot->next) - { /* clean pseudo-code for computing bound value */ - clean_code(mpl, slot->code); - /* delete symbolic value assigned to dummy index */ - if (slot->value != NULL) - delete_symbol(mpl, slot->value), slot->value = NULL; - } - /* clean pseudo-code for computing basic set */ - clean_code(mpl, block->code); - } - /* clean pseudo-code for computing domain predicate */ - clean_code(mpl, domain->code); -done: return; -} - -/**********************************************************************/ -/* * * MODEL SETS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- check_elem_set - check elemental set assigned to set member. --- --- This routine checks if given elemental set being assigned to member --- of specified model set satisfies to all restrictions. --- --- NOTE: This routine must not be called out of domain scope. */ - -void check_elem_set -( MPL *mpl, - SET *set, /* not changed */ - TUPLE *tuple, /* not changed */ - ELEMSET *refer /* not changed */ -) -{ WITHIN *within; - MEMBER *memb; - int eqno; - /* elemental set must be within all specified supersets */ - for (within = set->within, eqno = 1; within != NULL; within = - within->next, eqno++) - { xassert(within->code != NULL); - for (memb = refer->head; memb != NULL; memb = memb->next) - { if (!is_member(mpl, within->code, memb->tuple)) - { char buf[255+1]; - strcpy(buf, format_tuple(mpl, '(', memb->tuple)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s contains %s which not within specified " - "set; see (%d)", set->name, format_tuple(mpl, '[', - tuple), buf, eqno); - } - } - } - return; -} - -/*---------------------------------------------------------------------- --- take_member_set - obtain elemental set assigned to set member. --- --- This routine obtains a reference to elemental set assigned to given --- member of specified model set and returns it on exit. --- --- NOTE: This routine must not be called out of domain scope. */ - -ELEMSET *take_member_set /* returns reference, not value */ -( MPL *mpl, - SET *set, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ MEMBER *memb; - ELEMSET *refer; - /* find member in the set array */ - memb = find_member(mpl, set->array, tuple); - if (memb != NULL) - { /* member exists, so just take the reference */ - refer = memb->value.set; - } - else if (set->assign != NULL) - { /* compute value using assignment expression */ - refer = eval_elemset(mpl, set->assign); -add: /* check that the elemental set satisfies to all restrictions, - assign it to new member, and add the member to the array */ - check_elem_set(mpl, set, tuple, refer); - memb = add_member(mpl, set->array, copy_tuple(mpl, tuple)); - memb->value.set = refer; - } - else if (set->option != NULL) - { /* compute default elemental set */ - refer = eval_elemset(mpl, set->option); - goto add; - } - else - { /* no value (elemental set) is provided */ - error(mpl, "no value for %s%s", set->name, format_tuple(mpl, - '[', tuple)); - } - return refer; -} - -/*---------------------------------------------------------------------- --- eval_member_set - evaluate elemental set assigned to set member. --- --- This routine evaluates a reference to elemental set assigned to given --- member of specified model set and returns it on exit. */ - -struct eval_set_info -{ /* working info used by the routine eval_member_set */ - SET *set; - /* model set */ - TUPLE *tuple; - /* n-tuple, which defines set member */ - MEMBER *memb; - /* normally this pointer is NULL; the routine uses this pointer - to check data provided in the data section, in which case it - points to a member currently checked; this check is performed - automatically only once when a reference to any member occurs - for the first time */ - ELEMSET *refer; - /* evaluated reference to elemental set */ -}; - -static void eval_set_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine to work within domain scope */ - struct eval_set_info *info = _info; - if (info->memb != NULL) - { /* checking call; check elemental set being assigned */ - check_elem_set(mpl, info->set, info->memb->tuple, - info->memb->value.set); - } - else - { /* normal call; evaluate member, which has given n-tuple */ - info->refer = take_member_set(mpl, info->set, info->tuple); - } - return; -} - -#if 1 /* 12/XII-2008 */ -static void saturate_set(MPL *mpl, SET *set) -{ GADGET *gadget = set->gadget; - ELEMSET *data; - MEMBER *elem, *memb; - TUPLE *tuple, *work[20]; - int i; - xprintf("Generating %s...\n", set->name); - eval_whole_set(mpl, gadget->set); - /* gadget set must have exactly one member */ - xassert(gadget->set->array != NULL); - xassert(gadget->set->array->head != NULL); - xassert(gadget->set->array->head == gadget->set->array->tail); - data = gadget->set->array->head->value.set; - xassert(data->type == A_NONE); - xassert(data->dim == gadget->set->dimen); - /* walk thru all elements of the plain set */ - for (elem = data->head; elem != NULL; elem = elem->next) - { /* create a copy of n-tuple */ - tuple = copy_tuple(mpl, elem->tuple); - /* rearrange component of the n-tuple */ - for (i = 0; i < gadget->set->dimen; i++) - work[i] = NULL; - for (i = 0; tuple != NULL; tuple = tuple->next) - work[gadget->ind[i++]-1] = tuple; - xassert(i == gadget->set->dimen); - for (i = 0; i < gadget->set->dimen; i++) - { xassert(work[i] != NULL); - work[i]->next = work[i+1]; - } - /* construct subscript list from first set->dim components */ - if (set->dim == 0) - tuple = NULL; - else - tuple = work[0], work[set->dim-1]->next = NULL; - /* find corresponding member of the set to be initialized */ - memb = find_member(mpl, set->array, tuple); - if (memb == NULL) - { /* not found; add new member to the set and assign it empty - elemental set */ - memb = add_member(mpl, set->array, tuple); - memb->value.set = create_elemset(mpl, set->dimen); - } - else - { /* found; free subscript list */ - delete_tuple(mpl, tuple); - } - /* construct new n-tuple from rest set->dimen components */ - tuple = work[set->dim]; - xassert(set->dim + set->dimen == gadget->set->dimen); - work[gadget->set->dimen-1]->next = NULL; - /* and add it to the elemental set assigned to the member - (no check for duplicates is needed) */ - add_tuple(mpl, memb->value.set, tuple); - } - /* the set has been saturated with data */ - set->data = 1; - return; -} -#endif - -ELEMSET *eval_member_set /* returns reference, not value */ -( MPL *mpl, - SET *set, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ /* this routine evaluates set member */ - struct eval_set_info _info, *info = &_info; - xassert(set->dim == tuple_dimen(mpl, tuple)); - info->set = set; - info->tuple = tuple; -#if 1 /* 12/XII-2008 */ - if (set->gadget != NULL && set->data == 0) - { /* initialize the set with data from a plain set */ - saturate_set(mpl, set); - } -#endif - if (set->data == 1) - { /* check data, which are provided in the data section, but not - checked yet */ - /* save pointer to the last array member; note that during the - check new members may be added beyond the last member due to - references to the same parameter from default expression as - well as from expressions that define restricting supersets; - however, values assigned to the new members will be checked - by other routine, so we don't need to check them here */ - MEMBER *tail = set->array->tail; - /* change the data status to prevent infinite recursive loop - due to references to the same set during the check */ - set->data = 2; - /* check elemental sets assigned to array members in the data - section until the marked member has been reached */ - for (info->memb = set->array->head; info->memb != NULL; - info->memb = info->memb->next) - { if (eval_within_domain(mpl, set->domain, info->memb->tuple, - info, eval_set_func)) - out_of_domain(mpl, set->name, info->memb->tuple); - if (info->memb == tail) break; - } - /* the check has been finished */ - } - /* evaluate member, which has given n-tuple */ - info->memb = NULL; - if (eval_within_domain(mpl, info->set->domain, info->tuple, info, - eval_set_func)) - out_of_domain(mpl, set->name, info->tuple); - /* bring evaluated reference to the calling program */ - return info->refer; -} - -/*---------------------------------------------------------------------- --- eval_whole_set - evaluate model set over entire domain. --- --- This routine evaluates all members of specified model set over entire --- domain. */ - -static int whole_set_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - SET *set = (SET *)info; - TUPLE *tuple = get_domain_tuple(mpl, set->domain); - eval_member_set(mpl, set, tuple); - delete_tuple(mpl, tuple); - return 0; -} - -void eval_whole_set(MPL *mpl, SET *set) -{ loop_within_domain(mpl, set->domain, set, whole_set_func); - return; -} - -/*---------------------------------------------------------------------- --- clean set - clean model set. --- --- This routine cleans specified model set that assumes deleting all --- stuff dynamically allocated during the generation phase. */ - -void clean_set(MPL *mpl, SET *set) -{ WITHIN *within; - MEMBER *memb; - /* clean subscript domain */ - clean_domain(mpl, set->domain); - /* clean pseudo-code for computing supersets */ - for (within = set->within; within != NULL; within = within->next) - clean_code(mpl, within->code); - /* clean pseudo-code for computing assigned value */ - clean_code(mpl, set->assign); - /* clean pseudo-code for computing default value */ - clean_code(mpl, set->option); - /* reset data status flag */ - set->data = 0; - /* delete content array */ - for (memb = set->array->head; memb != NULL; memb = memb->next) - delete_value(mpl, set->array->type, &memb->value); - delete_array(mpl, set->array), set->array = NULL; - return; -} - -/**********************************************************************/ -/* * * MODEL PARAMETERS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- check_value_num - check numeric value assigned to parameter member. --- --- This routine checks if numeric value being assigned to some member --- of specified numeric model parameter satisfies to all restrictions. --- --- NOTE: This routine must not be called out of domain scope. */ - -void check_value_num -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple, /* not changed */ - double value -) -{ CONDITION *cond; - WITHIN *in; - int eqno; - /* the value must satisfy to the parameter type */ - switch (par->type) - { case A_NUMERIC: - break; - case A_INTEGER: - if (value != floor(value)) - error(mpl, "%s%s = %.*g not integer", par->name, - format_tuple(mpl, '[', tuple), DBL_DIG, value); - break; - case A_BINARY: - if (!(value == 0.0 || value == 1.0)) - error(mpl, "%s%s = %.*g not binary", par->name, - format_tuple(mpl, '[', tuple), DBL_DIG, value); - break; - default: - xassert(par != par); - } - /* the value must satisfy to all specified conditions */ - for (cond = par->cond, eqno = 1; cond != NULL; cond = cond->next, - eqno++) - { double bound; - char *rho; - xassert(cond->code != NULL); - bound = eval_numeric(mpl, cond->code); - switch (cond->rho) - { case O_LT: - if (!(value < bound)) - { rho = "<"; -err: error(mpl, "%s%s = %.*g not %s %.*g; see (%d)", - par->name, format_tuple(mpl, '[', tuple), DBL_DIG, - value, rho, DBL_DIG, bound, eqno); - } - break; - case O_LE: - if (!(value <= bound)) { rho = "<="; goto err; } - break; - case O_EQ: - if (!(value == bound)) { rho = "="; goto err; } - break; - case O_GE: - if (!(value >= bound)) { rho = ">="; goto err; } - break; - case O_GT: - if (!(value > bound)) { rho = ">"; goto err; } - break; - case O_NE: - if (!(value != bound)) { rho = "<>"; goto err; } - break; - default: - xassert(cond != cond); - } - } - /* the value must be in all specified supersets */ - for (in = par->in, eqno = 1; in != NULL; in = in->next, eqno++) - { TUPLE *dummy; - xassert(in->code != NULL); - xassert(in->code->dim == 1); - dummy = expand_tuple(mpl, create_tuple(mpl), - create_symbol_num(mpl, value)); - if (!is_member(mpl, in->code, dummy)) - error(mpl, "%s%s = %.*g not in specified set; see (%d)", - par->name, format_tuple(mpl, '[', tuple), DBL_DIG, - value, eqno); - delete_tuple(mpl, dummy); - } - return; -} - -/*---------------------------------------------------------------------- --- take_member_num - obtain num. value assigned to parameter member. --- --- This routine obtains a numeric value assigned to member of specified --- numeric model parameter and returns it on exit. --- --- NOTE: This routine must not be called out of domain scope. */ - -double take_member_num -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ MEMBER *memb; - double value; - /* find member in the parameter array */ - memb = find_member(mpl, par->array, tuple); - if (memb != NULL) - { /* member exists, so just take its value */ - value = memb->value.num; - } - else if (par->assign != NULL) - { /* compute value using assignment expression */ - value = eval_numeric(mpl, par->assign); -add: /* check that the value satisfies to all restrictions, assign - it to new member, and add the member to the array */ - check_value_num(mpl, par, tuple, value); - memb = add_member(mpl, par->array, copy_tuple(mpl, tuple)); - memb->value.num = value; - } - else if (par->option != NULL) - { /* compute default value */ - value = eval_numeric(mpl, par->option); - goto add; - } - else if (par->defval != NULL) - { /* take default value provided in the data section */ - if (par->defval->str != NULL) - error(mpl, "cannot convert %s to floating-point number", - format_symbol(mpl, par->defval)); - value = par->defval->num; - goto add; - } - else - { /* no value is provided */ - error(mpl, "no value for %s%s", par->name, format_tuple(mpl, - '[', tuple)); - } - return value; -} - -/*---------------------------------------------------------------------- --- eval_member_num - evaluate num. value assigned to parameter member. --- --- This routine evaluates a numeric value assigned to given member of --- specified numeric model parameter and returns it on exit. */ - -struct eval_num_info -{ /* working info used by the routine eval_member_num */ - PARAMETER *par; - /* model parameter */ - TUPLE *tuple; - /* n-tuple, which defines parameter member */ - MEMBER *memb; - /* normally this pointer is NULL; the routine uses this pointer - to check data provided in the data section, in which case it - points to a member currently checked; this check is performed - automatically only once when a reference to any member occurs - for the first time */ - double value; - /* evaluated numeric value */ -}; - -static void eval_num_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine to work within domain scope */ - struct eval_num_info *info = _info; - if (info->memb != NULL) - { /* checking call; check numeric value being assigned */ - check_value_num(mpl, info->par, info->memb->tuple, - info->memb->value.num); - } - else - { /* normal call; evaluate member, which has given n-tuple */ - info->value = take_member_num(mpl, info->par, info->tuple); - } - return; -} - -double eval_member_num -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ /* this routine evaluates numeric parameter member */ - struct eval_num_info _info, *info = &_info; - xassert(par->type == A_NUMERIC || par->type == A_INTEGER || - par->type == A_BINARY); - xassert(par->dim == tuple_dimen(mpl, tuple)); - info->par = par; - info->tuple = tuple; - if (par->data == 1) - { /* check data, which are provided in the data section, but not - checked yet */ - /* save pointer to the last array member; note that during the - check new members may be added beyond the last member due to - references to the same parameter from default expression as - well as from expressions that define restricting conditions; - however, values assigned to the new members will be checked - by other routine, so we don't need to check them here */ - MEMBER *tail = par->array->tail; - /* change the data status to prevent infinite recursive loop - due to references to the same parameter during the check */ - par->data = 2; - /* check values assigned to array members in the data section - until the marked member has been reached */ - for (info->memb = par->array->head; info->memb != NULL; - info->memb = info->memb->next) - { if (eval_within_domain(mpl, par->domain, info->memb->tuple, - info, eval_num_func)) - out_of_domain(mpl, par->name, info->memb->tuple); - if (info->memb == tail) break; - } - /* the check has been finished */ - } - /* evaluate member, which has given n-tuple */ - info->memb = NULL; - if (eval_within_domain(mpl, info->par->domain, info->tuple, info, - eval_num_func)) - out_of_domain(mpl, par->name, info->tuple); - /* bring evaluated value to the calling program */ - return info->value; -} - -/*---------------------------------------------------------------------- --- check_value_sym - check symbolic value assigned to parameter member. --- --- This routine checks if symbolic value being assigned to some member --- of specified symbolic model parameter satisfies to all restrictions. --- --- NOTE: This routine must not be called out of domain scope. */ - -void check_value_sym -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple, /* not changed */ - SYMBOL *value /* not changed */ -) -{ CONDITION *cond; - WITHIN *in; - int eqno; - /* the value must satisfy to all specified conditions */ - for (cond = par->cond, eqno = 1; cond != NULL; cond = cond->next, - eqno++) - { SYMBOL *bound; - char buf[255+1]; - xassert(cond->code != NULL); - bound = eval_symbolic(mpl, cond->code); - switch (cond->rho) - { -#if 1 /* 13/VIII-2008 */ - case O_LT: - if (!(compare_symbols(mpl, value, bound) < 0)) - { strcpy(buf, format_symbol(mpl, bound)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s = %s not < %s", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), buf, eqno); - } - break; - case O_LE: - if (!(compare_symbols(mpl, value, bound) <= 0)) - { strcpy(buf, format_symbol(mpl, bound)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s = %s not <= %s", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), buf, eqno); - } - break; -#endif - case O_EQ: - if (!(compare_symbols(mpl, value, bound) == 0)) - { strcpy(buf, format_symbol(mpl, bound)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s = %s not = %s", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), buf, eqno); - } - break; -#if 1 /* 13/VIII-2008 */ - case O_GE: - if (!(compare_symbols(mpl, value, bound) >= 0)) - { strcpy(buf, format_symbol(mpl, bound)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s = %s not >= %s", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), buf, eqno); - } - break; - case O_GT: - if (!(compare_symbols(mpl, value, bound) > 0)) - { strcpy(buf, format_symbol(mpl, bound)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s = %s not > %s", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), buf, eqno); - } - break; -#endif - case O_NE: - if (!(compare_symbols(mpl, value, bound) != 0)) - { strcpy(buf, format_symbol(mpl, bound)); - xassert(strlen(buf) < sizeof(buf)); - error(mpl, "%s%s = %s not <> %s", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), buf, eqno); - } - break; - default: - xassert(cond != cond); - } - delete_symbol(mpl, bound); - } - /* the value must be in all specified supersets */ - for (in = par->in, eqno = 1; in != NULL; in = in->next, eqno++) - { TUPLE *dummy; - xassert(in->code != NULL); - xassert(in->code->dim == 1); - dummy = expand_tuple(mpl, create_tuple(mpl), copy_symbol(mpl, - value)); - if (!is_member(mpl, in->code, dummy)) - error(mpl, "%s%s = %s not in specified set; see (%d)", - par->name, format_tuple(mpl, '[', tuple), - format_symbol(mpl, value), eqno); - delete_tuple(mpl, dummy); - } - return; -} - -/*---------------------------------------------------------------------- --- take_member_sym - obtain symb. value assigned to parameter member. --- --- This routine obtains a symbolic value assigned to member of specified --- symbolic model parameter and returns it on exit. --- --- NOTE: This routine must not be called out of domain scope. */ - -SYMBOL *take_member_sym /* returns value, not reference */ -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ MEMBER *memb; - SYMBOL *value; - /* find member in the parameter array */ - memb = find_member(mpl, par->array, tuple); - if (memb != NULL) - { /* member exists, so just take its value */ - value = copy_symbol(mpl, memb->value.sym); - } - else if (par->assign != NULL) - { /* compute value using assignment expression */ - value = eval_symbolic(mpl, par->assign); -add: /* check that the value satisfies to all restrictions, assign - it to new member, and add the member to the array */ - check_value_sym(mpl, par, tuple, value); - memb = add_member(mpl, par->array, copy_tuple(mpl, tuple)); - memb->value.sym = copy_symbol(mpl, value); - } - else if (par->option != NULL) - { /* compute default value */ - value = eval_symbolic(mpl, par->option); - goto add; - } - else if (par->defval != NULL) - { /* take default value provided in the data section */ - value = copy_symbol(mpl, par->defval); - goto add; - } - else - { /* no value is provided */ - error(mpl, "no value for %s%s", par->name, format_tuple(mpl, - '[', tuple)); - } - return value; -} - -/*---------------------------------------------------------------------- --- eval_member_sym - evaluate symb. value assigned to parameter member. --- --- This routine evaluates a symbolic value assigned to given member of --- specified symbolic model parameter and returns it on exit. */ - -struct eval_sym_info -{ /* working info used by the routine eval_member_sym */ - PARAMETER *par; - /* model parameter */ - TUPLE *tuple; - /* n-tuple, which defines parameter member */ - MEMBER *memb; - /* normally this pointer is NULL; the routine uses this pointer - to check data provided in the data section, in which case it - points to a member currently checked; this check is performed - automatically only once when a reference to any member occurs - for the first time */ - SYMBOL *value; - /* evaluated symbolic value */ -}; - -static void eval_sym_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine to work within domain scope */ - struct eval_sym_info *info = _info; - if (info->memb != NULL) - { /* checking call; check symbolic value being assigned */ - check_value_sym(mpl, info->par, info->memb->tuple, - info->memb->value.sym); - } - else - { /* normal call; evaluate member, which has given n-tuple */ - info->value = take_member_sym(mpl, info->par, info->tuple); - } - return; -} - -SYMBOL *eval_member_sym /* returns value, not reference */ -( MPL *mpl, - PARAMETER *par, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ /* this routine evaluates symbolic parameter member */ - struct eval_sym_info _info, *info = &_info; - xassert(par->type == A_SYMBOLIC); - xassert(par->dim == tuple_dimen(mpl, tuple)); - info->par = par; - info->tuple = tuple; - if (par->data == 1) - { /* check data, which are provided in the data section, but not - checked yet */ - /* save pointer to the last array member; note that during the - check new members may be added beyond the last member due to - references to the same parameter from default expression as - well as from expressions that define restricting conditions; - however, values assigned to the new members will be checked - by other routine, so we don't need to check them here */ - MEMBER *tail = par->array->tail; - /* change the data status to prevent infinite recursive loop - due to references to the same parameter during the check */ - par->data = 2; - /* check values assigned to array members in the data section - until the marked member has been reached */ - for (info->memb = par->array->head; info->memb != NULL; - info->memb = info->memb->next) - { if (eval_within_domain(mpl, par->domain, info->memb->tuple, - info, eval_sym_func)) - out_of_domain(mpl, par->name, info->memb->tuple); - if (info->memb == tail) break; - } - /* the check has been finished */ - } - /* evaluate member, which has given n-tuple */ - info->memb = NULL; - if (eval_within_domain(mpl, info->par->domain, info->tuple, info, - eval_sym_func)) - out_of_domain(mpl, par->name, info->tuple); - /* bring evaluated value to the calling program */ - return info->value; -} - -/*---------------------------------------------------------------------- --- eval_whole_par - evaluate model parameter over entire domain. --- --- This routine evaluates all members of specified model parameter over --- entire domain. */ - -static int whole_par_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - PARAMETER *par = (PARAMETER *)info; - TUPLE *tuple = get_domain_tuple(mpl, par->domain); - switch (par->type) - { case A_NUMERIC: - case A_INTEGER: - case A_BINARY: - eval_member_num(mpl, par, tuple); - break; - case A_SYMBOLIC: - delete_symbol(mpl, eval_member_sym(mpl, par, tuple)); - break; - default: - xassert(par != par); - } - delete_tuple(mpl, tuple); - return 0; -} - -void eval_whole_par(MPL *mpl, PARAMETER *par) -{ loop_within_domain(mpl, par->domain, par, whole_par_func); - return; -} - -/*---------------------------------------------------------------------- --- clean_parameter - clean model parameter. --- --- This routine cleans specified model parameter that assumes deleting --- all stuff dynamically allocated during the generation phase. */ - -void clean_parameter(MPL *mpl, PARAMETER *par) -{ CONDITION *cond; - WITHIN *in; - MEMBER *memb; - /* clean subscript domain */ - clean_domain(mpl, par->domain); - /* clean pseudo-code for computing restricting conditions */ - for (cond = par->cond; cond != NULL; cond = cond->next) - clean_code(mpl, cond->code); - /* clean pseudo-code for computing restricting supersets */ - for (in = par->in; in != NULL; in = in->next) - clean_code(mpl, in->code); - /* clean pseudo-code for computing assigned value */ - clean_code(mpl, par->assign); - /* clean pseudo-code for computing default value */ - clean_code(mpl, par->option); - /* reset data status flag */ - par->data = 0; - /* delete default symbolic value */ - if (par->defval != NULL) - delete_symbol(mpl, par->defval), par->defval = NULL; - /* delete content array */ - for (memb = par->array->head; memb != NULL; memb = memb->next) - delete_value(mpl, par->array->type, &memb->value); - delete_array(mpl, par->array), par->array = NULL; - return; -} - -/**********************************************************************/ -/* * * MODEL VARIABLES * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- take_member_var - obtain reference to elemental variable. --- --- This routine obtains a reference to elemental variable assigned to --- given member of specified model variable and returns it on exit. If --- necessary, new elemental variable is created. --- --- NOTE: This routine must not be called out of domain scope. */ - -ELEMVAR *take_member_var /* returns reference */ -( MPL *mpl, - VARIABLE *var, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ MEMBER *memb; - ELEMVAR *refer; - /* find member in the variable array */ - memb = find_member(mpl, var->array, tuple); - if (memb != NULL) - { /* member exists, so just take the reference */ - refer = memb->value.var; - } - else - { /* member is referenced for the first time and therefore does - not exist; create new elemental variable, assign it to new - member, and add the member to the variable array */ - memb = add_member(mpl, var->array, copy_tuple(mpl, tuple)); - refer = (memb->value.var = - dmp_get_atom(mpl->elemvars, sizeof(ELEMVAR))); - refer->j = 0; - refer->var = var; - refer->memb = memb; - /* compute lower bound */ - if (var->lbnd == NULL) - refer->lbnd = 0.0; - else - refer->lbnd = eval_numeric(mpl, var->lbnd); - /* compute upper bound */ - if (var->ubnd == NULL) - refer->ubnd = 0.0; - else if (var->ubnd == var->lbnd) - refer->ubnd = refer->lbnd; - else - refer->ubnd = eval_numeric(mpl, var->ubnd); - /* nullify working quantity */ - refer->temp = 0.0; -#if 1 /* 15/V-2010 */ - /* solution has not been obtained by the solver yet */ - refer->stat = 0; - refer->prim = refer->dual = 0.0; -#endif - } - return refer; -} - -/*---------------------------------------------------------------------- --- eval_member_var - evaluate reference to elemental variable. --- --- This routine evaluates a reference to elemental variable assigned to --- member of specified model variable and returns it on exit. */ - -struct eval_var_info -{ /* working info used by the routine eval_member_var */ - VARIABLE *var; - /* model variable */ - TUPLE *tuple; - /* n-tuple, which defines variable member */ - ELEMVAR *refer; - /* evaluated reference to elemental variable */ -}; - -static void eval_var_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine to work within domain scope */ - struct eval_var_info *info = _info; - info->refer = take_member_var(mpl, info->var, info->tuple); - return; -} - -ELEMVAR *eval_member_var /* returns reference */ -( MPL *mpl, - VARIABLE *var, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ /* this routine evaluates variable member */ - struct eval_var_info _info, *info = &_info; - xassert(var->dim == tuple_dimen(mpl, tuple)); - info->var = var; - info->tuple = tuple; - /* evaluate member, which has given n-tuple */ - if (eval_within_domain(mpl, info->var->domain, info->tuple, info, - eval_var_func)) - out_of_domain(mpl, var->name, info->tuple); - /* bring evaluated reference to the calling program */ - return info->refer; -} - -/*---------------------------------------------------------------------- --- eval_whole_var - evaluate model variable over entire domain. --- --- This routine evaluates all members of specified model variable over --- entire domain. */ - -static int whole_var_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - VARIABLE *var = (VARIABLE *)info; - TUPLE *tuple = get_domain_tuple(mpl, var->domain); - eval_member_var(mpl, var, tuple); - delete_tuple(mpl, tuple); - return 0; -} - -void eval_whole_var(MPL *mpl, VARIABLE *var) -{ loop_within_domain(mpl, var->domain, var, whole_var_func); - return; -} - -/*---------------------------------------------------------------------- --- clean_variable - clean model variable. --- --- This routine cleans specified model variable that assumes deleting --- all stuff dynamically allocated during the generation phase. */ - -void clean_variable(MPL *mpl, VARIABLE *var) -{ MEMBER *memb; - /* clean subscript domain */ - clean_domain(mpl, var->domain); - /* clean code for computing lower bound */ - clean_code(mpl, var->lbnd); - /* clean code for computing upper bound */ - if (var->ubnd != var->lbnd) clean_code(mpl, var->ubnd); - /* delete content array */ - for (memb = var->array->head; memb != NULL; memb = memb->next) - dmp_free_atom(mpl->elemvars, memb->value.var, sizeof(ELEMVAR)); - delete_array(mpl, var->array), var->array = NULL; - return; -} - -/**********************************************************************/ -/* * * MODEL CONSTRAINTS AND OBJECTIVES * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- take_member_con - obtain reference to elemental constraint. --- --- This routine obtains a reference to elemental constraint assigned --- to given member of specified model constraint and returns it on exit. --- If necessary, new elemental constraint is created. --- --- NOTE: This routine must not be called out of domain scope. */ - -ELEMCON *take_member_con /* returns reference */ -( MPL *mpl, - CONSTRAINT *con, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ MEMBER *memb; - ELEMCON *refer; - /* find member in the constraint array */ - memb = find_member(mpl, con->array, tuple); - if (memb != NULL) - { /* member exists, so just take the reference */ - refer = memb->value.con; - } - else - { /* member is referenced for the first time and therefore does - not exist; create new elemental constraint, assign it to new - member, and add the member to the constraint array */ - memb = add_member(mpl, con->array, copy_tuple(mpl, tuple)); - refer = (memb->value.con = - dmp_get_atom(mpl->elemcons, sizeof(ELEMCON))); - refer->i = 0; - refer->con = con; - refer->memb = memb; - /* compute linear form */ - xassert(con->code != NULL); - refer->form = eval_formula(mpl, con->code); - /* compute lower and upper bounds */ - if (con->lbnd == NULL && con->ubnd == NULL) - { /* objective has no bounds */ - double temp; - xassert(con->type == A_MINIMIZE || con->type == A_MAXIMIZE); - /* carry the constant term to the right-hand side */ - refer->form = remove_constant(mpl, refer->form, &temp); - refer->lbnd = refer->ubnd = - temp; - } - else if (con->lbnd != NULL && con->ubnd == NULL) - { /* constraint a * x + b >= c * y + d is transformed to the - standard form a * x - c * y >= d - b */ - double temp; - xassert(con->type == A_CONSTRAINT); - refer->form = linear_comb(mpl, - +1.0, refer->form, - -1.0, eval_formula(mpl, con->lbnd)); - refer->form = remove_constant(mpl, refer->form, &temp); - refer->lbnd = - temp; - refer->ubnd = 0.0; - } - else if (con->lbnd == NULL && con->ubnd != NULL) - { /* constraint a * x + b <= c * y + d is transformed to the - standard form a * x - c * y <= d - b */ - double temp; - xassert(con->type == A_CONSTRAINT); - refer->form = linear_comb(mpl, - +1.0, refer->form, - -1.0, eval_formula(mpl, con->ubnd)); - refer->form = remove_constant(mpl, refer->form, &temp); - refer->lbnd = 0.0; - refer->ubnd = - temp; - } - else if (con->lbnd == con->ubnd) - { /* constraint a * x + b = c * y + d is transformed to the - standard form a * x - c * y = d - b */ - double temp; - xassert(con->type == A_CONSTRAINT); - refer->form = linear_comb(mpl, - +1.0, refer->form, - -1.0, eval_formula(mpl, con->lbnd)); - refer->form = remove_constant(mpl, refer->form, &temp); - refer->lbnd = refer->ubnd = - temp; - } - else - { /* ranged constraint c <= a * x + b <= d is transformed to - the standard form c - b <= a * x <= d - b */ - double temp, temp1, temp2; - xassert(con->type == A_CONSTRAINT); - refer->form = remove_constant(mpl, refer->form, &temp); - xassert(remove_constant(mpl, eval_formula(mpl, con->lbnd), - &temp1) == NULL); - xassert(remove_constant(mpl, eval_formula(mpl, con->ubnd), - &temp2) == NULL); - refer->lbnd = fp_sub(mpl, temp1, temp); - refer->ubnd = fp_sub(mpl, temp2, temp); - } -#if 1 /* 15/V-2010 */ - /* solution has not been obtained by the solver yet */ - refer->stat = 0; - refer->prim = refer->dual = 0.0; -#endif - } - return refer; -} - -/*---------------------------------------------------------------------- --- eval_member_con - evaluate reference to elemental constraint. --- --- This routine evaluates a reference to elemental constraint assigned --- to member of specified model constraint and returns it on exit. */ - -struct eval_con_info -{ /* working info used by the routine eval_member_con */ - CONSTRAINT *con; - /* model constraint */ - TUPLE *tuple; - /* n-tuple, which defines constraint member */ - ELEMCON *refer; - /* evaluated reference to elemental constraint */ -}; - -static void eval_con_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine to work within domain scope */ - struct eval_con_info *info = _info; - info->refer = take_member_con(mpl, info->con, info->tuple); - return; -} - -ELEMCON *eval_member_con /* returns reference */ -( MPL *mpl, - CONSTRAINT *con, /* not changed */ - TUPLE *tuple /* not changed */ -) -{ /* this routine evaluates constraint member */ - struct eval_con_info _info, *info = &_info; - xassert(con->dim == tuple_dimen(mpl, tuple)); - info->con = con; - info->tuple = tuple; - /* evaluate member, which has given n-tuple */ - if (eval_within_domain(mpl, info->con->domain, info->tuple, info, - eval_con_func)) - out_of_domain(mpl, con->name, info->tuple); - /* bring evaluated reference to the calling program */ - return info->refer; -} - -/*---------------------------------------------------------------------- --- eval_whole_con - evaluate model constraint over entire domain. --- --- This routine evaluates all members of specified model constraint over --- entire domain. */ - -static int whole_con_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - CONSTRAINT *con = (CONSTRAINT *)info; - TUPLE *tuple = get_domain_tuple(mpl, con->domain); - eval_member_con(mpl, con, tuple); - delete_tuple(mpl, tuple); - return 0; -} - -void eval_whole_con(MPL *mpl, CONSTRAINT *con) -{ loop_within_domain(mpl, con->domain, con, whole_con_func); - return; -} - -/*---------------------------------------------------------------------- --- clean_constraint - clean model constraint. --- --- This routine cleans specified model constraint that assumes deleting --- all stuff dynamically allocated during the generation phase. */ - -void clean_constraint(MPL *mpl, CONSTRAINT *con) -{ MEMBER *memb; - /* clean subscript domain */ - clean_domain(mpl, con->domain); - /* clean code for computing main linear form */ - clean_code(mpl, con->code); - /* clean code for computing lower bound */ - clean_code(mpl, con->lbnd); - /* clean code for computing upper bound */ - if (con->ubnd != con->lbnd) clean_code(mpl, con->ubnd); - /* delete content array */ - for (memb = con->array->head; memb != NULL; memb = memb->next) - { delete_formula(mpl, memb->value.con->form); - dmp_free_atom(mpl->elemcons, memb->value.con, sizeof(ELEMCON)); - } - delete_array(mpl, con->array), con->array = NULL; - return; -} - -/**********************************************************************/ -/* * * PSEUDO-CODE * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- eval_numeric - evaluate pseudo-code to determine numeric value. --- --- This routine evaluates specified pseudo-code to determine resultant --- numeric value, which is returned on exit. */ - -struct iter_num_info -{ /* working info used by the routine iter_num_func */ - CODE *code; - /* pseudo-code for iterated operation to be performed */ - double value; - /* resultant value */ -}; - -static int iter_num_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine used to perform iterated operation - on numeric "integrand" within domain scope */ - struct iter_num_info *info = _info; - double temp; - temp = eval_numeric(mpl, info->code->arg.loop.x); - switch (info->code->op) - { case O_SUM: - /* summation over domain */ - info->value = fp_add(mpl, info->value, temp); - break; - case O_PROD: - /* multiplication over domain */ - info->value = fp_mul(mpl, info->value, temp); - break; - case O_MINIMUM: - /* minimum over domain */ - if (info->value > temp) info->value = temp; - break; - case O_MAXIMUM: - /* maximum over domain */ - if (info->value < temp) info->value = temp; - break; - default: - xassert(info != info); - } - return 0; -} - -double eval_numeric(MPL *mpl, CODE *code) -{ double value; - xassert(code != NULL); - xassert(code->type == A_NUMERIC); - xassert(code->dim == 0); - /* if the operation has a side effect, invalidate and delete the - resultant value */ - if (code->vflag && code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* if resultant value is valid, no evaluation is needed */ - if (code->valid) - { value = code->value.num; - goto done; - } - /* evaluate pseudo-code recursively */ - switch (code->op) - { case O_NUMBER: - /* take floating-point number */ - value = code->arg.num; - break; - case O_MEMNUM: - /* take member of numeric parameter */ - { TUPLE *tuple; - ARG_LIST *e; - tuple = create_tuple(mpl); - for (e = code->arg.par.list; e != NULL; e = e->next) - tuple = expand_tuple(mpl, tuple, eval_symbolic(mpl, - e->x)); - value = eval_member_num(mpl, code->arg.par.par, tuple); - delete_tuple(mpl, tuple); - } - break; - case O_MEMVAR: - /* take computed value of elemental variable */ - { TUPLE *tuple; - ARG_LIST *e; -#if 1 /* 15/V-2010 */ - ELEMVAR *var; -#endif - tuple = create_tuple(mpl); - for (e = code->arg.var.list; e != NULL; e = e->next) - tuple = expand_tuple(mpl, tuple, eval_symbolic(mpl, - e->x)); -#if 0 /* 15/V-2010 */ - value = eval_member_var(mpl, code->arg.var.var, tuple) - ->value; -#else - var = eval_member_var(mpl, code->arg.var.var, tuple); - switch (code->arg.var.suff) - { case DOT_LB: - if (var->var->lbnd == NULL) - value = -DBL_MAX; - else - value = var->lbnd; - break; - case DOT_UB: - if (var->var->ubnd == NULL) - value = +DBL_MAX; - else - value = var->ubnd; - break; - case DOT_STATUS: - value = var->stat; - break; - case DOT_VAL: - value = var->prim; - break; - case DOT_DUAL: - value = var->dual; - break; - default: - xassert(code != code); - } -#endif - delete_tuple(mpl, tuple); - } - break; -#if 1 /* 15/V-2010 */ - case O_MEMCON: - /* take computed value of elemental constraint */ - { TUPLE *tuple; - ARG_LIST *e; - ELEMCON *con; - tuple = create_tuple(mpl); - for (e = code->arg.con.list; e != NULL; e = e->next) - tuple = expand_tuple(mpl, tuple, eval_symbolic(mpl, - e->x)); - con = eval_member_con(mpl, code->arg.con.con, tuple); - switch (code->arg.con.suff) - { case DOT_LB: - if (con->con->lbnd == NULL) - value = -DBL_MAX; - else - value = con->lbnd; - break; - case DOT_UB: - if (con->con->ubnd == NULL) - value = +DBL_MAX; - else - value = con->ubnd; - break; - case DOT_STATUS: - value = con->stat; - break; - case DOT_VAL: - value = con->prim; - break; - case DOT_DUAL: - value = con->dual; - break; - default: - xassert(code != code); - } - delete_tuple(mpl, tuple); - } - break; -#endif - case O_IRAND224: - /* pseudo-random in [0, 2^24-1] */ - value = fp_irand224(mpl); - break; - case O_UNIFORM01: - /* pseudo-random in [0, 1) */ - value = fp_uniform01(mpl); - break; - case O_NORMAL01: - /* gaussian random, mu = 0, sigma = 1 */ - value = fp_normal01(mpl); - break; - case O_GMTIME: - /* current calendar time */ - value = fn_gmtime(mpl); - break; - case O_CVTNUM: - /* conversion to numeric */ - { SYMBOL *sym; - sym = eval_symbolic(mpl, code->arg.arg.x); -#if 0 /* 23/XI-2008 */ - if (sym->str != NULL) - error(mpl, "cannot convert %s to floating-point numbe" - "r", format_symbol(mpl, sym)); - value = sym->num; -#else - if (sym->str == NULL) - value = sym->num; - else - { if (str2num(sym->str, &value)) - error(mpl, "cannot convert %s to floating-point nu" - "mber", format_symbol(mpl, sym)); - } -#endif - delete_symbol(mpl, sym); - } - break; - case O_PLUS: - /* unary plus */ - value = + eval_numeric(mpl, code->arg.arg.x); - break; - case O_MINUS: - /* unary minus */ - value = - eval_numeric(mpl, code->arg.arg.x); - break; - case O_ABS: - /* absolute value */ - value = fabs(eval_numeric(mpl, code->arg.arg.x)); - break; - case O_CEIL: - /* round upward ("ceiling of x") */ - value = ceil(eval_numeric(mpl, code->arg.arg.x)); - break; - case O_FLOOR: - /* round downward ("floor of x") */ - value = floor(eval_numeric(mpl, code->arg.arg.x)); - break; - case O_EXP: - /* base-e exponential */ - value = fp_exp(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_LOG: - /* natural logarithm */ - value = fp_log(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_LOG10: - /* common (decimal) logarithm */ - value = fp_log10(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_SQRT: - /* square root */ - value = fp_sqrt(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_SIN: - /* trigonometric sine */ - value = fp_sin(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_COS: - /* trigonometric cosine */ - value = fp_cos(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_TAN: - /* trigonometric tangent */ - value = fp_tan(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_ATAN: - /* trigonometric arctangent (one argument) */ - value = fp_atan(mpl, eval_numeric(mpl, code->arg.arg.x)); - break; - case O_ATAN2: - /* trigonometric arctangent (two arguments) */ - value = fp_atan2(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_ROUND: - /* round to nearest integer */ - value = fp_round(mpl, - eval_numeric(mpl, code->arg.arg.x), 0.0); - break; - case O_ROUND2: - /* round to n fractional digits */ - value = fp_round(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_TRUNC: - /* truncate to nearest integer */ - value = fp_trunc(mpl, - eval_numeric(mpl, code->arg.arg.x), 0.0); - break; - case O_TRUNC2: - /* truncate to n fractional digits */ - value = fp_trunc(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_ADD: - /* addition */ - value = fp_add(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_SUB: - /* subtraction */ - value = fp_sub(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_LESS: - /* non-negative subtraction */ - value = fp_less(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_MUL: - /* multiplication */ - value = fp_mul(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_DIV: - /* division */ - value = fp_div(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_IDIV: - /* quotient of exact division */ - value = fp_idiv(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_MOD: - /* remainder of exact division */ - value = fp_mod(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_POWER: - /* exponentiation (raise to power) */ - value = fp_power(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_UNIFORM: - /* pseudo-random in [a, b) */ - value = fp_uniform(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_NORMAL: - /* gaussian random, given mu and sigma */ - value = fp_normal(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y)); - break; - case O_CARD: - { ELEMSET *set; - set = eval_elemset(mpl, code->arg.arg.x); - value = set->size; - delete_array(mpl, set); - } - break; - case O_LENGTH: - { SYMBOL *sym; - char str[MAX_LENGTH+1]; - sym = eval_symbolic(mpl, code->arg.arg.x); - if (sym->str == NULL) - sprintf(str, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, str); - delete_symbol(mpl, sym); - value = strlen(str); - } - break; - case O_STR2TIME: - { SYMBOL *sym; - char str[MAX_LENGTH+1], fmt[MAX_LENGTH+1]; - sym = eval_symbolic(mpl, code->arg.arg.x); - if (sym->str == NULL) - sprintf(str, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, str); - delete_symbol(mpl, sym); - sym = eval_symbolic(mpl, code->arg.arg.y); - if (sym->str == NULL) - sprintf(fmt, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, fmt); - delete_symbol(mpl, sym); - value = fn_str2time(mpl, str, fmt); - } - break; - case O_FORK: - /* if-then-else */ - if (eval_logical(mpl, code->arg.arg.x)) - value = eval_numeric(mpl, code->arg.arg.y); - else if (code->arg.arg.z == NULL) - value = 0.0; - else - value = eval_numeric(mpl, code->arg.arg.z); - break; - case O_MIN: - /* minimal value (n-ary) */ - { ARG_LIST *e; - double temp; - value = +DBL_MAX; - for (e = code->arg.list; e != NULL; e = e->next) - { temp = eval_numeric(mpl, e->x); - if (value > temp) value = temp; - } - } - break; - case O_MAX: - /* maximal value (n-ary) */ - { ARG_LIST *e; - double temp; - value = -DBL_MAX; - for (e = code->arg.list; e != NULL; e = e->next) - { temp = eval_numeric(mpl, e->x); - if (value < temp) value = temp; - } - } - break; - case O_SUM: - /* summation over domain */ - { struct iter_num_info _info, *info = &_info; - info->code = code; - info->value = 0.0; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_num_func); - value = info->value; - } - break; - case O_PROD: - /* multiplication over domain */ - { struct iter_num_info _info, *info = &_info; - info->code = code; - info->value = 1.0; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_num_func); - value = info->value; - } - break; - case O_MINIMUM: - /* minimum over domain */ - { struct iter_num_info _info, *info = &_info; - info->code = code; - info->value = +DBL_MAX; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_num_func); - if (info->value == +DBL_MAX) - error(mpl, "min{} over empty set; result undefined"); - value = info->value; - } - break; - case O_MAXIMUM: - /* maximum over domain */ - { struct iter_num_info _info, *info = &_info; - info->code = code; - info->value = -DBL_MAX; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_num_func); - if (info->value == -DBL_MAX) - error(mpl, "max{} over empty set; result undefined"); - value = info->value; - } - break; - default: - xassert(code != code); - } - /* save resultant value */ - xassert(!code->valid); - code->valid = 1; - code->value.num = value; -done: return value; -} - -/*---------------------------------------------------------------------- --- eval_symbolic - evaluate pseudo-code to determine symbolic value. --- --- This routine evaluates specified pseudo-code to determine resultant --- symbolic value, which is returned on exit. */ - -SYMBOL *eval_symbolic(MPL *mpl, CODE *code) -{ SYMBOL *value; - xassert(code != NULL); - xassert(code->type == A_SYMBOLIC); - xassert(code->dim == 0); - /* if the operation has a side effect, invalidate and delete the - resultant value */ - if (code->vflag && code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* if resultant value is valid, no evaluation is needed */ - if (code->valid) - { value = copy_symbol(mpl, code->value.sym); - goto done; - } - /* evaluate pseudo-code recursively */ - switch (code->op) - { case O_STRING: - /* take character string */ - value = create_symbol_str(mpl, create_string(mpl, - code->arg.str)); - break; - case O_INDEX: - /* take dummy index */ - xassert(code->arg.index.slot->value != NULL); - value = copy_symbol(mpl, code->arg.index.slot->value); - break; - case O_MEMSYM: - /* take member of symbolic parameter */ - { TUPLE *tuple; - ARG_LIST *e; - tuple = create_tuple(mpl); - for (e = code->arg.par.list; e != NULL; e = e->next) - tuple = expand_tuple(mpl, tuple, eval_symbolic(mpl, - e->x)); - value = eval_member_sym(mpl, code->arg.par.par, tuple); - delete_tuple(mpl, tuple); - } - break; - case O_CVTSYM: - /* conversion to symbolic */ - value = create_symbol_num(mpl, eval_numeric(mpl, - code->arg.arg.x)); - break; - case O_CONCAT: - /* concatenation */ - value = concat_symbols(mpl, - eval_symbolic(mpl, code->arg.arg.x), - eval_symbolic(mpl, code->arg.arg.y)); - break; - case O_FORK: - /* if-then-else */ - if (eval_logical(mpl, code->arg.arg.x)) - value = eval_symbolic(mpl, code->arg.arg.y); - else if (code->arg.arg.z == NULL) - value = create_symbol_num(mpl, 0.0); - else - value = eval_symbolic(mpl, code->arg.arg.z); - break; - case O_SUBSTR: - case O_SUBSTR3: - { double pos, len; - char str[MAX_LENGTH+1]; - value = eval_symbolic(mpl, code->arg.arg.x); - if (value->str == NULL) - sprintf(str, "%.*g", DBL_DIG, value->num); - else - fetch_string(mpl, value->str, str); - delete_symbol(mpl, value); - if (code->op == O_SUBSTR) - { pos = eval_numeric(mpl, code->arg.arg.y); - if (pos != floor(pos)) - error(mpl, "substr('...', %.*g); non-integer secon" - "d argument", DBL_DIG, pos); - if (pos < 1 || pos > strlen(str) + 1) - error(mpl, "substr('...', %.*g); substring out of " - "range", DBL_DIG, pos); - } - else - { pos = eval_numeric(mpl, code->arg.arg.y); - len = eval_numeric(mpl, code->arg.arg.z); - if (pos != floor(pos) || len != floor(len)) - error(mpl, "substr('...', %.*g, %.*g); non-integer" - " second and/or third argument", DBL_DIG, pos, - DBL_DIG, len); - if (pos < 1 || len < 0 || pos + len > strlen(str) + 1) - error(mpl, "substr('...', %.*g, %.*g); substring o" - "ut of range", DBL_DIG, pos, DBL_DIG, len); - str[(int)pos + (int)len - 1] = '\0'; - } - value = create_symbol_str(mpl, create_string(mpl, str + - (int)pos - 1)); - } - break; - case O_TIME2STR: - { double num; - SYMBOL *sym; - char str[MAX_LENGTH+1], fmt[MAX_LENGTH+1]; - num = eval_numeric(mpl, code->arg.arg.x); - sym = eval_symbolic(mpl, code->arg.arg.y); - if (sym->str == NULL) - sprintf(fmt, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, fmt); - delete_symbol(mpl, sym); - fn_time2str(mpl, str, num, fmt); - value = create_symbol_str(mpl, create_string(mpl, str)); - } - break; - default: - xassert(code != code); - } - /* save resultant value */ - xassert(!code->valid); - code->valid = 1; - code->value.sym = copy_symbol(mpl, value); -done: return value; -} - -/*---------------------------------------------------------------------- --- eval_logical - evaluate pseudo-code to determine logical value. --- --- This routine evaluates specified pseudo-code to determine resultant --- logical value, which is returned on exit. */ - -struct iter_log_info -{ /* working info used by the routine iter_log_func */ - CODE *code; - /* pseudo-code for iterated operation to be performed */ - int value; - /* resultant value */ -}; - -static int iter_log_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine used to perform iterated operation - on logical "integrand" within domain scope */ - struct iter_log_info *info = _info; - int ret = 0; - switch (info->code->op) - { case O_FORALL: - /* conjunction over domain */ - info->value &= eval_logical(mpl, info->code->arg.loop.x); - if (!info->value) ret = 1; - break; - case O_EXISTS: - /* disjunction over domain */ - info->value |= eval_logical(mpl, info->code->arg.loop.x); - if (info->value) ret = 1; - break; - default: - xassert(info != info); - } - return ret; -} - -int eval_logical(MPL *mpl, CODE *code) -{ int value; - xassert(code->type == A_LOGICAL); - xassert(code->dim == 0); - /* if the operation has a side effect, invalidate and delete the - resultant value */ - if (code->vflag && code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* if resultant value is valid, no evaluation is needed */ - if (code->valid) - { value = code->value.bit; - goto done; - } - /* evaluate pseudo-code recursively */ - switch (code->op) - { case O_CVTLOG: - /* conversion to logical */ - value = (eval_numeric(mpl, code->arg.arg.x) != 0.0); - break; - case O_NOT: - /* negation (logical "not") */ - value = !eval_logical(mpl, code->arg.arg.x); - break; - case O_LT: - /* comparison on 'less than' */ -#if 0 /* 02/VIII-2008 */ - value = (eval_numeric(mpl, code->arg.arg.x) < - eval_numeric(mpl, code->arg.arg.y)); -#else - xassert(code->arg.arg.x != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - value = (eval_numeric(mpl, code->arg.arg.x) < - eval_numeric(mpl, code->arg.arg.y)); - else - { SYMBOL *sym1 = eval_symbolic(mpl, code->arg.arg.x); - SYMBOL *sym2 = eval_symbolic(mpl, code->arg.arg.y); - value = (compare_symbols(mpl, sym1, sym2) < 0); - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - } -#endif - break; - case O_LE: - /* comparison on 'not greater than' */ -#if 0 /* 02/VIII-2008 */ - value = (eval_numeric(mpl, code->arg.arg.x) <= - eval_numeric(mpl, code->arg.arg.y)); -#else - xassert(code->arg.arg.x != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - value = (eval_numeric(mpl, code->arg.arg.x) <= - eval_numeric(mpl, code->arg.arg.y)); - else - { SYMBOL *sym1 = eval_symbolic(mpl, code->arg.arg.x); - SYMBOL *sym2 = eval_symbolic(mpl, code->arg.arg.y); - value = (compare_symbols(mpl, sym1, sym2) <= 0); - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - } -#endif - break; - case O_EQ: - /* comparison on 'equal to' */ - xassert(code->arg.arg.x != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - value = (eval_numeric(mpl, code->arg.arg.x) == - eval_numeric(mpl, code->arg.arg.y)); - else - { SYMBOL *sym1 = eval_symbolic(mpl, code->arg.arg.x); - SYMBOL *sym2 = eval_symbolic(mpl, code->arg.arg.y); - value = (compare_symbols(mpl, sym1, sym2) == 0); - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - } - break; - case O_GE: - /* comparison on 'not less than' */ -#if 0 /* 02/VIII-2008 */ - value = (eval_numeric(mpl, code->arg.arg.x) >= - eval_numeric(mpl, code->arg.arg.y)); -#else - xassert(code->arg.arg.x != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - value = (eval_numeric(mpl, code->arg.arg.x) >= - eval_numeric(mpl, code->arg.arg.y)); - else - { SYMBOL *sym1 = eval_symbolic(mpl, code->arg.arg.x); - SYMBOL *sym2 = eval_symbolic(mpl, code->arg.arg.y); - value = (compare_symbols(mpl, sym1, sym2) >= 0); - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - } -#endif - break; - case O_GT: - /* comparison on 'greater than' */ -#if 0 /* 02/VIII-2008 */ - value = (eval_numeric(mpl, code->arg.arg.x) > - eval_numeric(mpl, code->arg.arg.y)); -#else - xassert(code->arg.arg.x != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - value = (eval_numeric(mpl, code->arg.arg.x) > - eval_numeric(mpl, code->arg.arg.y)); - else - { SYMBOL *sym1 = eval_symbolic(mpl, code->arg.arg.x); - SYMBOL *sym2 = eval_symbolic(mpl, code->arg.arg.y); - value = (compare_symbols(mpl, sym1, sym2) > 0); - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - } -#endif - break; - case O_NE: - /* comparison on 'not equal to' */ - xassert(code->arg.arg.x != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - value = (eval_numeric(mpl, code->arg.arg.x) != - eval_numeric(mpl, code->arg.arg.y)); - else - { SYMBOL *sym1 = eval_symbolic(mpl, code->arg.arg.x); - SYMBOL *sym2 = eval_symbolic(mpl, code->arg.arg.y); - value = (compare_symbols(mpl, sym1, sym2) != 0); - delete_symbol(mpl, sym1); - delete_symbol(mpl, sym2); - } - break; - case O_AND: - /* conjunction (logical "and") */ - value = eval_logical(mpl, code->arg.arg.x) && - eval_logical(mpl, code->arg.arg.y); - break; - case O_OR: - /* disjunction (logical "or") */ - value = eval_logical(mpl, code->arg.arg.x) || - eval_logical(mpl, code->arg.arg.y); - break; - case O_IN: - /* test on 'x in Y' */ - { TUPLE *tuple; - tuple = eval_tuple(mpl, code->arg.arg.x); - value = is_member(mpl, code->arg.arg.y, tuple); - delete_tuple(mpl, tuple); - } - break; - case O_NOTIN: - /* test on 'x not in Y' */ - { TUPLE *tuple; - tuple = eval_tuple(mpl, code->arg.arg.x); - value = !is_member(mpl, code->arg.arg.y, tuple); - delete_tuple(mpl, tuple); - } - break; - case O_WITHIN: - /* test on 'X within Y' */ - { ELEMSET *set; - MEMBER *memb; - set = eval_elemset(mpl, code->arg.arg.x); - value = 1; - for (memb = set->head; memb != NULL; memb = memb->next) - { if (!is_member(mpl, code->arg.arg.y, memb->tuple)) - { value = 0; - break; - } - } - delete_elemset(mpl, set); - } - break; - case O_NOTWITHIN: - /* test on 'X not within Y' */ - { ELEMSET *set; - MEMBER *memb; - set = eval_elemset(mpl, code->arg.arg.x); - value = 1; - for (memb = set->head; memb != NULL; memb = memb->next) - { if (is_member(mpl, code->arg.arg.y, memb->tuple)) - { value = 0; - break; - } - } - delete_elemset(mpl, set); - } - break; - case O_FORALL: - /* conjunction (A-quantification) */ - { struct iter_log_info _info, *info = &_info; - info->code = code; - info->value = 1; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_log_func); - value = info->value; - } - break; - case O_EXISTS: - /* disjunction (E-quantification) */ - { struct iter_log_info _info, *info = &_info; - info->code = code; - info->value = 0; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_log_func); - value = info->value; - } - break; - default: - xassert(code != code); - } - /* save resultant value */ - xassert(!code->valid); - code->valid = 1; - code->value.bit = value; -done: return value; -} - -/*---------------------------------------------------------------------- --- eval_tuple - evaluate pseudo-code to construct n-tuple. --- --- This routine evaluates specified pseudo-code to construct resultant --- n-tuple, which is returned on exit. */ - -TUPLE *eval_tuple(MPL *mpl, CODE *code) -{ TUPLE *value; - xassert(code != NULL); - xassert(code->type == A_TUPLE); - xassert(code->dim > 0); - /* if the operation has a side effect, invalidate and delete the - resultant value */ - if (code->vflag && code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* if resultant value is valid, no evaluation is needed */ - if (code->valid) - { value = copy_tuple(mpl, code->value.tuple); - goto done; - } - /* evaluate pseudo-code recursively */ - switch (code->op) - { case O_TUPLE: - /* make n-tuple */ - { ARG_LIST *e; - value = create_tuple(mpl); - for (e = code->arg.list; e != NULL; e = e->next) - value = expand_tuple(mpl, value, eval_symbolic(mpl, - e->x)); - } - break; - case O_CVTTUP: - /* convert to 1-tuple */ - value = expand_tuple(mpl, create_tuple(mpl), - eval_symbolic(mpl, code->arg.arg.x)); - break; - default: - xassert(code != code); - } - /* save resultant value */ - xassert(!code->valid); - code->valid = 1; - code->value.tuple = copy_tuple(mpl, value); -done: return value; -} - -/*---------------------------------------------------------------------- --- eval_elemset - evaluate pseudo-code to construct elemental set. --- --- This routine evaluates specified pseudo-code to construct resultant --- elemental set, which is returned on exit. */ - -struct iter_set_info -{ /* working info used by the routine iter_set_func */ - CODE *code; - /* pseudo-code for iterated operation to be performed */ - ELEMSET *value; - /* resultant value */ -}; - -static int iter_set_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine used to perform iterated operation - on n-tuple "integrand" within domain scope */ - struct iter_set_info *info = _info; - TUPLE *tuple; - switch (info->code->op) - { case O_SETOF: - /* compute next n-tuple and add it to the set; in this case - duplicate n-tuples are silently ignored */ - tuple = eval_tuple(mpl, info->code->arg.loop.x); - if (find_tuple(mpl, info->value, tuple) == NULL) - add_tuple(mpl, info->value, tuple); - else - delete_tuple(mpl, tuple); - break; - case O_BUILD: - /* construct next n-tuple using current values assigned to - *free* dummy indices as its components and add it to the - set; in this case duplicate n-tuples cannot appear */ - add_tuple(mpl, info->value, get_domain_tuple(mpl, - info->code->arg.loop.domain)); - break; - default: - xassert(info != info); - } - return 0; -} - -ELEMSET *eval_elemset(MPL *mpl, CODE *code) -{ ELEMSET *value; - xassert(code != NULL); - xassert(code->type == A_ELEMSET); - xassert(code->dim > 0); - /* if the operation has a side effect, invalidate and delete the - resultant value */ - if (code->vflag && code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* if resultant value is valid, no evaluation is needed */ - if (code->valid) - { value = copy_elemset(mpl, code->value.set); - goto done; - } - /* evaluate pseudo-code recursively */ - switch (code->op) - { case O_MEMSET: - /* take member of set */ - { TUPLE *tuple; - ARG_LIST *e; - tuple = create_tuple(mpl); - for (e = code->arg.set.list; e != NULL; e = e->next) - tuple = expand_tuple(mpl, tuple, eval_symbolic(mpl, - e->x)); - value = copy_elemset(mpl, - eval_member_set(mpl, code->arg.set.set, tuple)); - delete_tuple(mpl, tuple); - } - break; - case O_MAKE: - /* make elemental set of n-tuples */ - { ARG_LIST *e; - value = create_elemset(mpl, code->dim); - for (e = code->arg.list; e != NULL; e = e->next) - check_then_add(mpl, value, eval_tuple(mpl, e->x)); - } - break; - case O_UNION: - /* union of two elemental sets */ - value = set_union(mpl, - eval_elemset(mpl, code->arg.arg.x), - eval_elemset(mpl, code->arg.arg.y)); - break; - case O_DIFF: - /* difference between two elemental sets */ - value = set_diff(mpl, - eval_elemset(mpl, code->arg.arg.x), - eval_elemset(mpl, code->arg.arg.y)); - break; - case O_SYMDIFF: - /* symmetric difference between two elemental sets */ - value = set_symdiff(mpl, - eval_elemset(mpl, code->arg.arg.x), - eval_elemset(mpl, code->arg.arg.y)); - break; - case O_INTER: - /* intersection of two elemental sets */ - value = set_inter(mpl, - eval_elemset(mpl, code->arg.arg.x), - eval_elemset(mpl, code->arg.arg.y)); - break; - case O_CROSS: - /* cross (Cartesian) product of two elemental sets */ - value = set_cross(mpl, - eval_elemset(mpl, code->arg.arg.x), - eval_elemset(mpl, code->arg.arg.y)); - break; - case O_DOTS: - /* build "arithmetic" elemental set */ - value = create_arelset(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_numeric(mpl, code->arg.arg.y), - code->arg.arg.z == NULL ? 1.0 : eval_numeric(mpl, - code->arg.arg.z)); - break; - case O_FORK: - /* if-then-else */ - if (eval_logical(mpl, code->arg.arg.x)) - value = eval_elemset(mpl, code->arg.arg.y); - else - value = eval_elemset(mpl, code->arg.arg.z); - break; - case O_SETOF: - /* compute elemental set */ - { struct iter_set_info _info, *info = &_info; - info->code = code; - info->value = create_elemset(mpl, code->dim); - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_set_func); - value = info->value; - } - break; - case O_BUILD: - /* build elemental set identical to domain set */ - { struct iter_set_info _info, *info = &_info; - info->code = code; - info->value = create_elemset(mpl, code->dim); - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_set_func); - value = info->value; - } - break; - default: - xassert(code != code); - } - /* save resultant value */ - xassert(!code->valid); - code->valid = 1; - code->value.set = copy_elemset(mpl, value); -done: return value; -} - -/*---------------------------------------------------------------------- --- is_member - check if n-tuple is in set specified by pseudo-code. --- --- This routine checks if given n-tuple is a member of elemental set --- specified in the form of pseudo-code (i.e. by expression). --- --- The n-tuple may have more components that dimension of the elemental --- set, in which case the extra components are ignored. */ - -static void null_func(MPL *mpl, void *info) -{ /* this is dummy routine used to enter the domain scope */ - xassert(mpl == mpl); - xassert(info == NULL); - return; -} - -int is_member(MPL *mpl, CODE *code, TUPLE *tuple) -{ int value; - xassert(code != NULL); - xassert(code->type == A_ELEMSET); - xassert(code->dim > 0); - xassert(tuple != NULL); - switch (code->op) - { case O_MEMSET: - /* check if given n-tuple is member of elemental set, which - is assigned to member of model set */ - { ARG_LIST *e; - TUPLE *temp; - ELEMSET *set; - /* evaluate reference to elemental set */ - temp = create_tuple(mpl); - for (e = code->arg.set.list; e != NULL; e = e->next) - temp = expand_tuple(mpl, temp, eval_symbolic(mpl, - e->x)); - set = eval_member_set(mpl, code->arg.set.set, temp); - delete_tuple(mpl, temp); - /* check if the n-tuple is contained in the set array */ - temp = build_subtuple(mpl, tuple, set->dim); - value = (find_tuple(mpl, set, temp) != NULL); - delete_tuple(mpl, temp); - } - break; - case O_MAKE: - /* check if given n-tuple is member of literal set */ - { ARG_LIST *e; - TUPLE *temp, *that; - value = 0; - temp = build_subtuple(mpl, tuple, code->dim); - for (e = code->arg.list; e != NULL; e = e->next) - { that = eval_tuple(mpl, e->x); - value = (compare_tuples(mpl, temp, that) == 0); - delete_tuple(mpl, that); - if (value) break; - } - delete_tuple(mpl, temp); - } - break; - case O_UNION: - value = is_member(mpl, code->arg.arg.x, tuple) || - is_member(mpl, code->arg.arg.y, tuple); - break; - case O_DIFF: - value = is_member(mpl, code->arg.arg.x, tuple) && - !is_member(mpl, code->arg.arg.y, tuple); - break; - case O_SYMDIFF: - { int in1 = is_member(mpl, code->arg.arg.x, tuple); - int in2 = is_member(mpl, code->arg.arg.y, tuple); - value = (in1 && !in2) || (!in1 && in2); - } - break; - case O_INTER: - value = is_member(mpl, code->arg.arg.x, tuple) && - is_member(mpl, code->arg.arg.y, tuple); - break; - case O_CROSS: - { int j; - value = is_member(mpl, code->arg.arg.x, tuple); - if (value) - { for (j = 1; j <= code->arg.arg.x->dim; j++) - { xassert(tuple != NULL); - tuple = tuple->next; - } - value = is_member(mpl, code->arg.arg.y, tuple); - } - } - break; - case O_DOTS: - /* check if given 1-tuple is member of "arithmetic" set */ - { int j; - double x, t0, tf, dt; - xassert(code->dim == 1); - /* compute "parameters" of the "arithmetic" set */ - t0 = eval_numeric(mpl, code->arg.arg.x); - tf = eval_numeric(mpl, code->arg.arg.y); - if (code->arg.arg.z == NULL) - dt = 1.0; - else - dt = eval_numeric(mpl, code->arg.arg.z); - /* make sure the parameters are correct */ - arelset_size(mpl, t0, tf, dt); - /* if component of 1-tuple is symbolic, not numeric, the - 1-tuple cannot be member of "arithmetic" set */ - xassert(tuple->sym != NULL); - if (tuple->sym->str != NULL) - { value = 0; - break; - } - /* determine numeric value of the component */ - x = tuple->sym->num; - /* if the component value is out of the set range, the - 1-tuple is not in the set */ - if (dt > 0.0 && !(t0 <= x && x <= tf) || - dt < 0.0 && !(tf <= x && x <= t0)) - { value = 0; - break; - } - /* estimate ordinal number of the 1-tuple in the set */ - j = (int)(((x - t0) / dt) + 0.5) + 1; - /* perform the main check */ - value = (arelset_member(mpl, t0, tf, dt, j) == x); - } - break; - case O_FORK: - /* check if given n-tuple is member of conditional set */ - if (eval_logical(mpl, code->arg.arg.x)) - value = is_member(mpl, code->arg.arg.y, tuple); - else - value = is_member(mpl, code->arg.arg.z, tuple); - break; - case O_SETOF: - /* check if given n-tuple is member of computed set */ - /* it is not clear how to efficiently perform the check not - computing the entire elemental set :+( */ - error(mpl, "implementation restriction; in/within setof{} n" - "ot allowed"); - break; - case O_BUILD: - /* check if given n-tuple is member of domain set */ - { TUPLE *temp; - temp = build_subtuple(mpl, tuple, code->dim); - /* try to enter the domain scope; if it is successful, - the n-tuple is in the domain set */ - value = (eval_within_domain(mpl, code->arg.loop.domain, - temp, NULL, null_func) == 0); - delete_tuple(mpl, temp); - } - break; - default: - xassert(code != code); - } - return value; -} - -/*---------------------------------------------------------------------- --- eval_formula - evaluate pseudo-code to construct linear form. --- --- This routine evaluates specified pseudo-code to construct resultant --- linear form, which is returned on exit. */ - -struct iter_form_info -{ /* working info used by the routine iter_form_func */ - CODE *code; - /* pseudo-code for iterated operation to be performed */ - FORMULA *value; - /* resultant value */ - FORMULA *tail; - /* pointer to the last term */ -}; - -static int iter_form_func(MPL *mpl, void *_info) -{ /* this is auxiliary routine used to perform iterated operation - on linear form "integrand" within domain scope */ - struct iter_form_info *info = _info; - switch (info->code->op) - { case O_SUM: - /* summation over domain */ -#if 0 - info->value = - linear_comb(mpl, - +1.0, info->value, - +1.0, eval_formula(mpl, info->code->arg.loop.x)); -#else - /* the routine linear_comb needs to look through all terms - of both linear forms to reduce identical terms, so using - it here is not a good idea (for example, evaluation of - sum{i in 1..n} x[i] required quadratic time); the better - idea is to gather all terms of the integrand in one list - and reduce identical terms only once after all terms of - the resultant linear form have been evaluated */ - { FORMULA *form, *term; - form = eval_formula(mpl, info->code->arg.loop.x); - if (info->value == NULL) - { xassert(info->tail == NULL); - info->value = form; - } - else - { xassert(info->tail != NULL); - info->tail->next = form; - } - for (term = form; term != NULL; term = term->next) - info->tail = term; - } -#endif - break; - default: - xassert(info != info); - } - return 0; -} - -FORMULA *eval_formula(MPL *mpl, CODE *code) -{ FORMULA *value; - xassert(code != NULL); - xassert(code->type == A_FORMULA); - xassert(code->dim == 0); - /* if the operation has a side effect, invalidate and delete the - resultant value */ - if (code->vflag && code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* if resultant value is valid, no evaluation is needed */ - if (code->valid) - { value = copy_formula(mpl, code->value.form); - goto done; - } - /* evaluate pseudo-code recursively */ - switch (code->op) - { case O_MEMVAR: - /* take member of variable */ - { TUPLE *tuple; - ARG_LIST *e; - tuple = create_tuple(mpl); - for (e = code->arg.var.list; e != NULL; e = e->next) - tuple = expand_tuple(mpl, tuple, eval_symbolic(mpl, - e->x)); -#if 1 /* 15/V-2010 */ - xassert(code->arg.var.suff == DOT_NONE); -#endif - value = single_variable(mpl, - eval_member_var(mpl, code->arg.var.var, tuple)); - delete_tuple(mpl, tuple); - } - break; - case O_CVTLFM: - /* convert to linear form */ - value = constant_term(mpl, eval_numeric(mpl, - code->arg.arg.x)); - break; - case O_PLUS: - /* unary plus */ - value = linear_comb(mpl, - 0.0, constant_term(mpl, 0.0), - +1.0, eval_formula(mpl, code->arg.arg.x)); - break; - case O_MINUS: - /* unary minus */ - value = linear_comb(mpl, - 0.0, constant_term(mpl, 0.0), - -1.0, eval_formula(mpl, code->arg.arg.x)); - break; - case O_ADD: - /* addition */ - value = linear_comb(mpl, - +1.0, eval_formula(mpl, code->arg.arg.x), - +1.0, eval_formula(mpl, code->arg.arg.y)); - break; - case O_SUB: - /* subtraction */ - value = linear_comb(mpl, - +1.0, eval_formula(mpl, code->arg.arg.x), - -1.0, eval_formula(mpl, code->arg.arg.y)); - break; - case O_MUL: - /* multiplication */ - xassert(code->arg.arg.x != NULL); - xassert(code->arg.arg.y != NULL); - if (code->arg.arg.x->type == A_NUMERIC) - { xassert(code->arg.arg.y->type == A_FORMULA); - value = linear_comb(mpl, - eval_numeric(mpl, code->arg.arg.x), - eval_formula(mpl, code->arg.arg.y), - 0.0, constant_term(mpl, 0.0)); - } - else - { xassert(code->arg.arg.x->type == A_FORMULA); - xassert(code->arg.arg.y->type == A_NUMERIC); - value = linear_comb(mpl, - eval_numeric(mpl, code->arg.arg.y), - eval_formula(mpl, code->arg.arg.x), - 0.0, constant_term(mpl, 0.0)); - } - break; - case O_DIV: - /* division */ - value = linear_comb(mpl, - fp_div(mpl, 1.0, eval_numeric(mpl, code->arg.arg.y)), - eval_formula(mpl, code->arg.arg.x), - 0.0, constant_term(mpl, 0.0)); - break; - case O_FORK: - /* if-then-else */ - if (eval_logical(mpl, code->arg.arg.x)) - value = eval_formula(mpl, code->arg.arg.y); - else if (code->arg.arg.z == NULL) - value = constant_term(mpl, 0.0); - else - value = eval_formula(mpl, code->arg.arg.z); - break; - case O_SUM: - /* summation over domain */ - { struct iter_form_info _info, *info = &_info; - info->code = code; - info->value = constant_term(mpl, 0.0); - info->tail = NULL; - loop_within_domain(mpl, code->arg.loop.domain, info, - iter_form_func); - value = reduce_terms(mpl, info->value); - } - break; - default: - xassert(code != code); - } - /* save resultant value */ - xassert(!code->valid); - code->valid = 1; - code->value.form = copy_formula(mpl, value); -done: return value; -} - -/*---------------------------------------------------------------------- --- clean_code - clean pseudo-code. --- --- This routine recursively cleans specified pseudo-code that assumes --- deleting all temporary resultant values. */ - -void clean_code(MPL *mpl, CODE *code) -{ ARG_LIST *e; - /* if no pseudo-code is specified, do nothing */ - if (code == NULL) goto done; - /* if resultant value is valid (exists), delete it */ - if (code->valid) - { code->valid = 0; - delete_value(mpl, code->type, &code->value); - } - /* recursively clean pseudo-code for operands */ - switch (code->op) - { case O_NUMBER: - case O_STRING: - case O_INDEX: - break; - case O_MEMNUM: - case O_MEMSYM: - for (e = code->arg.par.list; e != NULL; e = e->next) - clean_code(mpl, e->x); - break; - case O_MEMSET: - for (e = code->arg.set.list; e != NULL; e = e->next) - clean_code(mpl, e->x); - break; - case O_MEMVAR: - for (e = code->arg.var.list; e != NULL; e = e->next) - clean_code(mpl, e->x); - break; -#if 1 /* 15/V-2010 */ - case O_MEMCON: - for (e = code->arg.con.list; e != NULL; e = e->next) - clean_code(mpl, e->x); - break; -#endif - case O_TUPLE: - case O_MAKE: - for (e = code->arg.list; e != NULL; e = e->next) - clean_code(mpl, e->x); - break; - case O_SLICE: - xassert(code != code); - case O_IRAND224: - case O_UNIFORM01: - case O_NORMAL01: - case O_GMTIME: - break; - case O_CVTNUM: - case O_CVTSYM: - case O_CVTLOG: - case O_CVTTUP: - case O_CVTLFM: - case O_PLUS: - case O_MINUS: - case O_NOT: - case O_ABS: - case O_CEIL: - case O_FLOOR: - case O_EXP: - case O_LOG: - case O_LOG10: - case O_SQRT: - case O_SIN: - case O_COS: - case O_TAN: - case O_ATAN: - case O_ROUND: - case O_TRUNC: - case O_CARD: - case O_LENGTH: - /* unary operation */ - clean_code(mpl, code->arg.arg.x); - break; - case O_ADD: - case O_SUB: - case O_LESS: - case O_MUL: - case O_DIV: - case O_IDIV: - case O_MOD: - case O_POWER: - case O_ATAN2: - case O_ROUND2: - case O_TRUNC2: - case O_UNIFORM: - case O_NORMAL: - case O_CONCAT: - case O_LT: - case O_LE: - case O_EQ: - case O_GE: - case O_GT: - case O_NE: - case O_AND: - case O_OR: - case O_UNION: - case O_DIFF: - case O_SYMDIFF: - case O_INTER: - case O_CROSS: - case O_IN: - case O_NOTIN: - case O_WITHIN: - case O_NOTWITHIN: - case O_SUBSTR: - case O_STR2TIME: - case O_TIME2STR: - /* binary operation */ - clean_code(mpl, code->arg.arg.x); - clean_code(mpl, code->arg.arg.y); - break; - case O_DOTS: - case O_FORK: - case O_SUBSTR3: - /* ternary operation */ - clean_code(mpl, code->arg.arg.x); - clean_code(mpl, code->arg.arg.y); - clean_code(mpl, code->arg.arg.z); - break; - case O_MIN: - case O_MAX: - /* n-ary operation */ - for (e = code->arg.list; e != NULL; e = e->next) - clean_code(mpl, e->x); - break; - case O_SUM: - case O_PROD: - case O_MINIMUM: - case O_MAXIMUM: - case O_FORALL: - case O_EXISTS: - case O_SETOF: - case O_BUILD: - /* iterated operation */ - clean_domain(mpl, code->arg.loop.domain); - clean_code(mpl, code->arg.loop.x); - break; - default: - xassert(code->op != code->op); - } -done: return; -} - -#if 1 /* 11/II-2008 */ -/**********************************************************************/ -/* * * DATA TABLES * * */ -/**********************************************************************/ - -int mpl_tab_num_args(TABDCA *dca) -{ /* returns the number of arguments */ - return dca->na; -} - -const char *mpl_tab_get_arg(TABDCA *dca, int k) -{ /* returns pointer to k-th argument */ - xassert(1 <= k && k <= dca->na); - return dca->arg[k]; -} - -int mpl_tab_num_flds(TABDCA *dca) -{ /* returns the number of fields */ - return dca->nf; -} - -const char *mpl_tab_get_name(TABDCA *dca, int k) -{ /* returns pointer to name of k-th field */ - xassert(1 <= k && k <= dca->nf); - return dca->name[k]; -} - -int mpl_tab_get_type(TABDCA *dca, int k) -{ /* returns type of k-th field */ - xassert(1 <= k && k <= dca->nf); - return dca->type[k]; -} - -double mpl_tab_get_num(TABDCA *dca, int k) -{ /* returns numeric value of k-th field */ - xassert(1 <= k && k <= dca->nf); - xassert(dca->type[k] == 'N'); - return dca->num[k]; -} - -const char *mpl_tab_get_str(TABDCA *dca, int k) -{ /* returns pointer to string value of k-th field */ - xassert(1 <= k && k <= dca->nf); - xassert(dca->type[k] == 'S'); - xassert(dca->str[k] != NULL); - return dca->str[k]; -} - -void mpl_tab_set_num(TABDCA *dca, int k, double num) -{ /* assign numeric value to k-th field */ - xassert(1 <= k && k <= dca->nf); - xassert(dca->type[k] == '?'); - dca->type[k] = 'N'; - dca->num[k] = num; - return; -} - -void mpl_tab_set_str(TABDCA *dca, int k, const char *str) -{ /* assign string value to k-th field */ - xassert(1 <= k && k <= dca->nf); - xassert(dca->type[k] == '?'); - xassert(strlen(str) <= MAX_LENGTH); - xassert(dca->str[k] != NULL); - dca->type[k] = 'S'; - strcpy(dca->str[k], str); - return; -} - -static int write_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - TABLE *tab = info; - TABDCA *dca = mpl->dca; - TABOUT *out; - SYMBOL *sym; - int k; - char buf[MAX_LENGTH+1]; - /* evaluate field values */ - k = 0; - for (out = tab->u.out.list; out != NULL; out = out->next) - { k++; - switch (out->code->type) - { case A_NUMERIC: - dca->type[k] = 'N'; - dca->num[k] = eval_numeric(mpl, out->code); - dca->str[k][0] = '\0'; - break; - case A_SYMBOLIC: - sym = eval_symbolic(mpl, out->code); - if (sym->str == NULL) - { dca->type[k] = 'N'; - dca->num[k] = sym->num; - dca->str[k][0] = '\0'; - } - else - { dca->type[k] = 'S'; - dca->num[k] = 0.0; - fetch_string(mpl, sym->str, buf); - strcpy(dca->str[k], buf); - } - delete_symbol(mpl, sym); - break; - default: - xassert(out != out); - } - } - /* write record to output table */ - mpl_tab_drv_write(mpl); - return 0; -} - -void execute_table(MPL *mpl, TABLE *tab) -{ /* execute table statement */ - TABARG *arg; - TABFLD *fld; - TABIN *in; - TABOUT *out; - TABDCA *dca; - SET *set; - int k; - char buf[MAX_LENGTH+1]; - /* allocate table driver communication area */ - xassert(mpl->dca == NULL); - mpl->dca = dca = xmalloc(sizeof(TABDCA)); - dca->id = 0; - dca->link = NULL; - dca->na = 0; - dca->arg = NULL; - dca->nf = 0; - dca->name = NULL; - dca->type = NULL; - dca->num = NULL; - dca->str = NULL; - /* allocate arguments */ - xassert(dca->na == 0); - for (arg = tab->arg; arg != NULL; arg = arg->next) - dca->na++; - dca->arg = xcalloc(1+dca->na, sizeof(char *)); -#if 1 /* 28/IX-2008 */ - for (k = 1; k <= dca->na; k++) dca->arg[k] = NULL; -#endif - /* evaluate argument values */ - k = 0; - for (arg = tab->arg; arg != NULL; arg = arg->next) - { SYMBOL *sym; - k++; - xassert(arg->code->type == A_SYMBOLIC); - sym = eval_symbolic(mpl, arg->code); - if (sym->str == NULL) - sprintf(buf, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, buf); - delete_symbol(mpl, sym); - dca->arg[k] = xmalloc(strlen(buf)+1); - strcpy(dca->arg[k], buf); - } - /* perform table input/output */ - switch (tab->type) - { case A_INPUT: goto read_table; - case A_OUTPUT: goto write_table; - default: xassert(tab != tab); - } -read_table: - /* read data from input table */ - /* add the only member to the control set and assign it empty - elemental set */ - set = tab->u.in.set; - if (set != NULL) - { if (set->data) - error(mpl, "%s already provided with data", set->name); - xassert(set->array->head == NULL); - add_member(mpl, set->array, NULL)->value.set = - create_elemset(mpl, set->dimen); - set->data = 1; - } - /* check parameters specified in the input list */ - for (in = tab->u.in.list; in != NULL; in = in->next) - { if (in->par->data) - error(mpl, "%s already provided with data", in->par->name); - in->par->data = 1; - } - /* allocate and initialize fields */ - xassert(dca->nf == 0); - for (fld = tab->u.in.fld; fld != NULL; fld = fld->next) - dca->nf++; - for (in = tab->u.in.list; in != NULL; in = in->next) - dca->nf++; - dca->name = xcalloc(1+dca->nf, sizeof(char *)); - dca->type = xcalloc(1+dca->nf, sizeof(int)); - dca->num = xcalloc(1+dca->nf, sizeof(double)); - dca->str = xcalloc(1+dca->nf, sizeof(char *)); - k = 0; - for (fld = tab->u.in.fld; fld != NULL; fld = fld->next) - { k++; - dca->name[k] = fld->name; - dca->type[k] = '?'; - dca->num[k] = 0.0; - dca->str[k] = xmalloc(MAX_LENGTH+1); - dca->str[k][0] = '\0'; - } - for (in = tab->u.in.list; in != NULL; in = in->next) - { k++; - dca->name[k] = in->name; - dca->type[k] = '?'; - dca->num[k] = 0.0; - dca->str[k] = xmalloc(MAX_LENGTH+1); - dca->str[k][0] = '\0'; - } - /* open input table */ - mpl_tab_drv_open(mpl, 'R'); - /* read and process records */ - for (;;) - { TUPLE *tup; - /* reset field types */ - for (k = 1; k <= dca->nf; k++) - dca->type[k] = '?'; - /* read next record */ - if (mpl_tab_drv_read(mpl)) break; - /* all fields must be set by the driver */ - for (k = 1; k <= dca->nf; k++) - { if (dca->type[k] == '?') - error(mpl, "field %s missing in input table", - dca->name[k]); - } - /* construct n-tuple */ - tup = create_tuple(mpl); - k = 0; - for (fld = tab->u.in.fld; fld != NULL; fld = fld->next) - { k++; - xassert(k <= dca->nf); - switch (dca->type[k]) - { case 'N': - tup = expand_tuple(mpl, tup, create_symbol_num(mpl, - dca->num[k])); - break; - case 'S': - xassert(strlen(dca->str[k]) <= MAX_LENGTH); - tup = expand_tuple(mpl, tup, create_symbol_str(mpl, - create_string(mpl, dca->str[k]))); - break; - default: - xassert(dca != dca); - } - } - /* add n-tuple just read to the control set */ - if (tab->u.in.set != NULL) - check_then_add(mpl, tab->u.in.set->array->head->value.set, - copy_tuple(mpl, tup)); - /* assign values to the parameters in the input list */ - for (in = tab->u.in.list; in != NULL; in = in->next) - { MEMBER *memb; - k++; - xassert(k <= dca->nf); - /* there must be no member with the same n-tuple */ - if (find_member(mpl, in->par->array, tup) != NULL) - error(mpl, "%s%s already defined", in->par->name, - format_tuple(mpl, '[', tup)); - /* create new parameter member with given n-tuple */ - memb = add_member(mpl, in->par->array, copy_tuple(mpl, tup)) - ; - /* assign value to the parameter member */ - switch (in->par->type) - { case A_NUMERIC: - case A_INTEGER: - case A_BINARY: - if (dca->type[k] != 'N') - error(mpl, "%s requires numeric data", - in->par->name); - memb->value.num = dca->num[k]; - break; - case A_SYMBOLIC: - switch (dca->type[k]) - { case 'N': - memb->value.sym = create_symbol_num(mpl, - dca->num[k]); - break; - case 'S': - xassert(strlen(dca->str[k]) <= MAX_LENGTH); - memb->value.sym = create_symbol_str(mpl, - create_string(mpl,dca->str[k])); - break; - default: - xassert(dca != dca); - } - break; - default: - xassert(in != in); - } - } - /* n-tuple is no more needed */ - delete_tuple(mpl, tup); - } - /* close input table */ - mpl_tab_drv_close(mpl); - goto done; -write_table: - /* write data to output table */ - /* allocate and initialize fields */ - xassert(dca->nf == 0); - for (out = tab->u.out.list; out != NULL; out = out->next) - dca->nf++; - dca->name = xcalloc(1+dca->nf, sizeof(char *)); - dca->type = xcalloc(1+dca->nf, sizeof(int)); - dca->num = xcalloc(1+dca->nf, sizeof(double)); - dca->str = xcalloc(1+dca->nf, sizeof(char *)); - k = 0; - for (out = tab->u.out.list; out != NULL; out = out->next) - { k++; - dca->name[k] = out->name; - dca->type[k] = '?'; - dca->num[k] = 0.0; - dca->str[k] = xmalloc(MAX_LENGTH+1); - dca->str[k][0] = '\0'; - } - /* open output table */ - mpl_tab_drv_open(mpl, 'W'); - /* evaluate fields and write records */ - loop_within_domain(mpl, tab->u.out.domain, tab, write_func); - /* close output table */ - mpl_tab_drv_close(mpl); -done: /* free table driver communication area */ - free_dca(mpl); - return; -} - -void free_dca(MPL *mpl) -{ /* free table driver communucation area */ - TABDCA *dca = mpl->dca; - int k; - if (dca != NULL) - { if (dca->link != NULL) - mpl_tab_drv_close(mpl); - if (dca->arg != NULL) - { for (k = 1; k <= dca->na; k++) -#if 1 /* 28/IX-2008 */ - if (dca->arg[k] != NULL) -#endif - xfree(dca->arg[k]); - xfree(dca->arg); - } - if (dca->name != NULL) xfree(dca->name); - if (dca->type != NULL) xfree(dca->type); - if (dca->num != NULL) xfree(dca->num); - if (dca->str != NULL) - { for (k = 1; k <= dca->nf; k++) - xfree(dca->str[k]); - xfree(dca->str); - } - xfree(dca), mpl->dca = NULL; - } - return; -} - -void clean_table(MPL *mpl, TABLE *tab) -{ /* clean table statement */ - TABARG *arg; - TABOUT *out; - /* clean string list */ - for (arg = tab->arg; arg != NULL; arg = arg->next) - clean_code(mpl, arg->code); - switch (tab->type) - { case A_INPUT: - break; - case A_OUTPUT: - /* clean subscript domain */ - clean_domain(mpl, tab->u.out.domain); - /* clean output list */ - for (out = tab->u.out.list; out != NULL; out = out->next) - clean_code(mpl, out->code); - break; - default: - xassert(tab != tab); - } - return; -} -#endif - -/**********************************************************************/ -/* * * MODEL STATEMENTS * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- execute_check - execute check statement. --- --- This routine executes specified check statement. */ - -static int check_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - CHECK *chk = (CHECK *)info; - if (!eval_logical(mpl, chk->code)) - error(mpl, "check%s failed", format_tuple(mpl, '[', - get_domain_tuple(mpl, chk->domain))); - return 0; -} - -void execute_check(MPL *mpl, CHECK *chk) -{ loop_within_domain(mpl, chk->domain, chk, check_func); - return; -} - -/*---------------------------------------------------------------------- --- clean_check - clean check statement. --- --- This routine cleans specified check statement that assumes deleting --- all stuff dynamically allocated on generating/postsolving phase. */ - -void clean_check(MPL *mpl, CHECK *chk) -{ /* clean subscript domain */ - clean_domain(mpl, chk->domain); - /* clean pseudo-code for computing predicate */ - clean_code(mpl, chk->code); - return; -} - -/*---------------------------------------------------------------------- --- execute_display - execute display statement. --- --- This routine executes specified display statement. */ - -static void display_set(MPL *mpl, SET *set, MEMBER *memb) -{ /* display member of model set */ - ELEMSET *s = memb->value.set; - MEMBER *m; - write_text(mpl, "%s%s%s\n", set->name, - format_tuple(mpl, '[', memb->tuple), - s->head == NULL ? " is empty" : ":"); - for (m = s->head; m != NULL; m = m->next) - write_text(mpl, " %s\n", format_tuple(mpl, '(', m->tuple)); - return; -} - -static void display_par(MPL *mpl, PARAMETER *par, MEMBER *memb) -{ /* display member of model parameter */ - switch (par->type) - { case A_NUMERIC: - case A_INTEGER: - case A_BINARY: - write_text(mpl, "%s%s = %.*g\n", par->name, - format_tuple(mpl, '[', memb->tuple), - DBL_DIG, memb->value.num); - break; - case A_SYMBOLIC: - write_text(mpl, "%s%s = %s\n", par->name, - format_tuple(mpl, '[', memb->tuple), - format_symbol(mpl, memb->value.sym)); - break; - default: - xassert(par != par); - } - return; -} - -#if 1 /* 15/V-2010 */ -static void display_var(MPL *mpl, VARIABLE *var, MEMBER *memb, - int suff) -{ /* display member of model variable */ - if (suff == DOT_NONE || suff == DOT_VAL) - write_text(mpl, "%s%s.val = %.*g\n", var->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.var->prim); - else if (suff == DOT_LB) - write_text(mpl, "%s%s.lb = %.*g\n", var->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.var->var->lbnd == NULL ? -DBL_MAX : - memb->value.var->lbnd); - else if (suff == DOT_UB) - write_text(mpl, "%s%s.ub = %.*g\n", var->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.var->var->ubnd == NULL ? +DBL_MAX : - memb->value.var->ubnd); - else if (suff == DOT_STATUS) - write_text(mpl, "%s%s.status = %d\n", var->name, format_tuple - (mpl, '[', memb->tuple), memb->value.var->stat); - else if (suff == DOT_DUAL) - write_text(mpl, "%s%s.dual = %.*g\n", var->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.var->dual); - else - xassert(suff != suff); - return; -} -#endif - -#if 1 /* 15/V-2010 */ -static void display_con(MPL *mpl, CONSTRAINT *con, MEMBER *memb, - int suff) -{ /* display member of model constraint */ - if (suff == DOT_NONE || suff == DOT_VAL) - write_text(mpl, "%s%s.val = %.*g\n", con->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.con->prim); - else if (suff == DOT_LB) - write_text(mpl, "%s%s.lb = %.*g\n", con->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.con->con->lbnd == NULL ? -DBL_MAX : - memb->value.con->lbnd); - else if (suff == DOT_UB) - write_text(mpl, "%s%s.ub = %.*g\n", con->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.con->con->ubnd == NULL ? +DBL_MAX : - memb->value.con->ubnd); - else if (suff == DOT_STATUS) - write_text(mpl, "%s%s.status = %d\n", con->name, format_tuple - (mpl, '[', memb->tuple), memb->value.con->stat); - else if (suff == DOT_DUAL) - write_text(mpl, "%s%s.dual = %.*g\n", con->name, - format_tuple(mpl, '[', memb->tuple), DBL_DIG, - memb->value.con->dual); - else - xassert(suff != suff); - return; -} -#endif - -static void display_memb(MPL *mpl, CODE *code) -{ /* display member specified by pseudo-code */ - MEMBER memb; - ARG_LIST *e; - xassert(code->op == O_MEMNUM || code->op == O_MEMSYM - || code->op == O_MEMSET || code->op == O_MEMVAR - || code->op == O_MEMCON); - memb.tuple = create_tuple(mpl); - for (e = code->arg.par.list; e != NULL; e = e->next) - memb.tuple = expand_tuple(mpl, memb.tuple, eval_symbolic(mpl, - e->x)); - switch (code->op) - { case O_MEMNUM: - memb.value.num = eval_member_num(mpl, code->arg.par.par, - memb.tuple); - display_par(mpl, code->arg.par.par, &memb); - break; - case O_MEMSYM: - memb.value.sym = eval_member_sym(mpl, code->arg.par.par, - memb.tuple); - display_par(mpl, code->arg.par.par, &memb); - delete_symbol(mpl, memb.value.sym); - break; - case O_MEMSET: - memb.value.set = eval_member_set(mpl, code->arg.set.set, - memb.tuple); - display_set(mpl, code->arg.set.set, &memb); - break; - case O_MEMVAR: - memb.value.var = eval_member_var(mpl, code->arg.var.var, - memb.tuple); - display_var - (mpl, code->arg.var.var, &memb, code->arg.var.suff); - break; - case O_MEMCON: - memb.value.con = eval_member_con(mpl, code->arg.con.con, - memb.tuple); - display_con - (mpl, code->arg.con.con, &memb, code->arg.con.suff); - break; - default: - xassert(code != code); - } - delete_tuple(mpl, memb.tuple); - return; -} - -static void display_code(MPL *mpl, CODE *code) -{ /* display value of expression */ - switch (code->type) - { case A_NUMERIC: - /* numeric value */ - { double num; - num = eval_numeric(mpl, code); - write_text(mpl, "%.*g\n", DBL_DIG, num); - } - break; - case A_SYMBOLIC: - /* symbolic value */ - { SYMBOL *sym; - sym = eval_symbolic(mpl, code); - write_text(mpl, "%s\n", format_symbol(mpl, sym)); - delete_symbol(mpl, sym); - } - break; - case A_LOGICAL: - /* logical value */ - { int bit; - bit = eval_logical(mpl, code); - write_text(mpl, "%s\n", bit ? "true" : "false"); - } - break; - case A_TUPLE: - /* n-tuple */ - { TUPLE *tuple; - tuple = eval_tuple(mpl, code); - write_text(mpl, "%s\n", format_tuple(mpl, '(', tuple)); - delete_tuple(mpl, tuple); - } - break; - case A_ELEMSET: - /* elemental set */ - { ELEMSET *set; - MEMBER *memb; - set = eval_elemset(mpl, code); - if (set->head == 0) - write_text(mpl, "set is empty\n"); - for (memb = set->head; memb != NULL; memb = memb->next) - write_text(mpl, " %s\n", format_tuple(mpl, '(', - memb->tuple)); - delete_elemset(mpl, set); - } - break; - case A_FORMULA: - /* linear form */ - { FORMULA *form, *term; - form = eval_formula(mpl, code); - if (form == NULL) - write_text(mpl, "linear form is empty\n"); - for (term = form; term != NULL; term = term->next) - { if (term->var == NULL) - write_text(mpl, " %.*g\n", term->coef); - else - write_text(mpl, " %.*g %s%s\n", DBL_DIG, - term->coef, term->var->var->name, - format_tuple(mpl, '[', term->var->memb->tuple)); - } - delete_formula(mpl, form); - } - break; - default: - xassert(code != code); - } - return; -} - -static int display_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - DISPLAY *dpy = (DISPLAY *)info; - DISPLAY1 *entry; - for (entry = dpy->list; entry != NULL; entry = entry->next) - { if (entry->type == A_INDEX) - { /* dummy index */ - DOMAIN_SLOT *slot = entry->u.slot; - write_text(mpl, "%s = %s\n", slot->name, - format_symbol(mpl, slot->value)); - } - else if (entry->type == A_SET) - { /* model set */ - SET *set = entry->u.set; - MEMBER *memb; - if (set->assign != NULL) - { /* the set has assignment expression; evaluate all its - members over entire domain */ - eval_whole_set(mpl, set); - } - else - { /* the set has no assignment expression; refer to its - any existing member ignoring resultant value to check - the data provided the data section */ -#if 1 /* 12/XII-2008 */ - if (set->gadget != NULL && set->data == 0) - { /* initialize the set with data from a plain set */ - saturate_set(mpl, set); - } -#endif - if (set->array->head != NULL) - eval_member_set(mpl, set, set->array->head->tuple); - } - /* display all members of the set array */ - if (set->array->head == NULL) - write_text(mpl, "%s has empty content\n", set->name); - for (memb = set->array->head; memb != NULL; memb = - memb->next) display_set(mpl, set, memb); - } - else if (entry->type == A_PARAMETER) - { /* model parameter */ - PARAMETER *par = entry->u.par; - MEMBER *memb; - if (par->assign != NULL) - { /* the parameter has an assignment expression; evaluate - all its member over entire domain */ - eval_whole_par(mpl, par); - } - else - { /* the parameter has no assignment expression; refer to - its any existing member ignoring resultant value to - check the data provided in the data section */ - if (par->array->head != NULL) - { if (par->type != A_SYMBOLIC) - eval_member_num(mpl, par, par->array->head->tuple); - else - delete_symbol(mpl, eval_member_sym(mpl, par, - par->array->head->tuple)); - } - } - /* display all members of the parameter array */ - if (par->array->head == NULL) - write_text(mpl, "%s has empty content\n", par->name); - for (memb = par->array->head; memb != NULL; memb = - memb->next) display_par(mpl, par, memb); - } - else if (entry->type == A_VARIABLE) - { /* model variable */ - VARIABLE *var = entry->u.var; - MEMBER *memb; - xassert(mpl->flag_p); - /* display all members of the variable array */ - if (var->array->head == NULL) - write_text(mpl, "%s has empty content\n", var->name); - for (memb = var->array->head; memb != NULL; memb = - memb->next) display_var(mpl, var, memb, DOT_NONE); - } - else if (entry->type == A_CONSTRAINT) - { /* model constraint */ - CONSTRAINT *con = entry->u.con; - MEMBER *memb; - xassert(mpl->flag_p); - /* display all members of the constraint array */ - if (con->array->head == NULL) - write_text(mpl, "%s has empty content\n", con->name); - for (memb = con->array->head; memb != NULL; memb = - memb->next) display_con(mpl, con, memb, DOT_NONE); - } - else if (entry->type == A_EXPRESSION) - { /* expression */ - CODE *code = entry->u.code; - if (code->op == O_MEMNUM || code->op == O_MEMSYM || - code->op == O_MEMSET || code->op == O_MEMVAR || - code->op == O_MEMCON) - display_memb(mpl, code); - else - display_code(mpl, code); - } - else - xassert(entry != entry); - } - return 0; -} - -void execute_display(MPL *mpl, DISPLAY *dpy) -{ loop_within_domain(mpl, dpy->domain, dpy, display_func); - return; -} - -/*---------------------------------------------------------------------- --- clean_display - clean display statement. --- --- This routine cleans specified display statement that assumes deleting --- all stuff dynamically allocated on generating/postsolving phase. */ - -void clean_display(MPL *mpl, DISPLAY *dpy) -{ DISPLAY1 *d; -#if 0 /* 15/V-2010 */ - ARG_LIST *e; -#endif - /* clean subscript domain */ - clean_domain(mpl, dpy->domain); - /* clean display list */ - for (d = dpy->list; d != NULL; d = d->next) - { /* clean pseudo-code for computing expression */ - if (d->type == A_EXPRESSION) - clean_code(mpl, d->u.code); -#if 0 /* 15/V-2010 */ - /* clean pseudo-code for computing subscripts */ - for (e = d->list; e != NULL; e = e->next) - clean_code(mpl, e->x); -#endif - } - return; -} - -/*---------------------------------------------------------------------- --- execute_printf - execute printf statement. --- --- This routine executes specified printf statement. */ - -#if 1 /* 14/VII-2006 */ -static void print_char(MPL *mpl, int c) -{ if (mpl->prt_fp == NULL) - write_char(mpl, c); - else -#if 0 /* 04/VIII-2013 */ - xfputc(c, mpl->prt_fp); -#else - { unsigned char buf[1]; - buf[0] = (unsigned char)c; - glp_write(mpl->prt_fp, buf, 1); - } -#endif - return; -} - -static void print_text(MPL *mpl, char *fmt, ...) -{ va_list arg; - char buf[OUTBUF_SIZE], *c; - va_start(arg, fmt); - vsprintf(buf, fmt, arg); - xassert(strlen(buf) < sizeof(buf)); - va_end(arg); - for (c = buf; *c != '\0'; c++) print_char(mpl, *c); - return; -} -#endif - -static int printf_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - PRINTF *prt = (PRINTF *)info; - PRINTF1 *entry; - SYMBOL *sym; - char fmt[MAX_LENGTH+1], *c, *from, save; - /* evaluate format control string */ - sym = eval_symbolic(mpl, prt->fmt); - if (sym->str == NULL) - sprintf(fmt, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, fmt); - delete_symbol(mpl, sym); - /* scan format control string and perform formatting output */ - entry = prt->list; - for (c = fmt; *c != '\0'; c++) - { if (*c == '%') - { /* scan format specifier */ - from = c++; - if (*c == '%') - { print_char(mpl, '%'); - continue; - } - if (entry == NULL) break; - /* scan optional flags */ - while (*c == '-' || *c == '+' || *c == ' ' || *c == '#' || - *c == '0') c++; - /* scan optional minimum field width */ - while (isdigit((unsigned char)*c)) c++; - /* scan optional precision */ - if (*c == '.') - { c++; - while (isdigit((unsigned char)*c)) c++; - } - /* scan conversion specifier and perform formatting */ - save = *(c+1), *(c+1) = '\0'; - if (*c == 'd' || *c == 'i' || *c == 'e' || *c == 'E' || - *c == 'f' || *c == 'F' || *c == 'g' || *c == 'G') - { /* the specifier requires numeric value */ - double value; - xassert(entry != NULL); - switch (entry->code->type) - { case A_NUMERIC: - value = eval_numeric(mpl, entry->code); - break; - case A_SYMBOLIC: - sym = eval_symbolic(mpl, entry->code); - if (sym->str != NULL) - error(mpl, "cannot convert %s to floating-point" - " number", format_symbol(mpl, sym)); - value = sym->num; - delete_symbol(mpl, sym); - break; - case A_LOGICAL: - if (eval_logical(mpl, entry->code)) - value = 1.0; - else - value = 0.0; - break; - default: - xassert(entry != entry); - } - if (*c == 'd' || *c == 'i') - { double int_max = (double)INT_MAX; - if (!(-int_max <= value && value <= +int_max)) - error(mpl, "cannot convert %.*g to integer", - DBL_DIG, value); - print_text(mpl, from, (int)floor(value + 0.5)); - } - else - print_text(mpl, from, value); - } - else if (*c == 's') - { /* the specifier requires symbolic value */ - char value[MAX_LENGTH+1]; - switch (entry->code->type) - { case A_NUMERIC: - sprintf(value, "%.*g", DBL_DIG, eval_numeric(mpl, - entry->code)); - break; - case A_LOGICAL: - if (eval_logical(mpl, entry->code)) - strcpy(value, "T"); - else - strcpy(value, "F"); - break; - case A_SYMBOLIC: - sym = eval_symbolic(mpl, entry->code); - if (sym->str == NULL) - sprintf(value, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, value); - delete_symbol(mpl, sym); - break; - default: - xassert(entry != entry); - } - print_text(mpl, from, value); - } - else - error(mpl, "format specifier missing or invalid"); - *(c+1) = save; - entry = entry->next; - } - else if (*c == '\\') - { /* write some control character */ - c++; - if (*c == 't') - print_char(mpl, '\t'); - else if (*c == 'n') - print_char(mpl, '\n'); -#if 1 /* 28/X-2010 */ - else if (*c == '\0') - { /* format string ends with backslash */ - error(mpl, "invalid use of escape character \\ in format" - " control string"); - } -#endif - else - print_char(mpl, *c); - } - else - { /* write character without formatting */ - print_char(mpl, *c); - } - } - return 0; -} - -#if 0 /* 14/VII-2006 */ -void execute_printf(MPL *mpl, PRINTF *prt) -{ loop_within_domain(mpl, prt->domain, prt, printf_func); - return; -} -#else -void execute_printf(MPL *mpl, PRINTF *prt) -{ if (prt->fname == NULL) - { /* switch to the standard output */ - if (mpl->prt_fp != NULL) - { glp_close(mpl->prt_fp), mpl->prt_fp = NULL; - xfree(mpl->prt_file), mpl->prt_file = NULL; - } - } - else - { /* evaluate file name string */ - SYMBOL *sym; - char fname[MAX_LENGTH+1]; - sym = eval_symbolic(mpl, prt->fname); - if (sym->str == NULL) - sprintf(fname, "%.*g", DBL_DIG, sym->num); - else - fetch_string(mpl, sym->str, fname); - delete_symbol(mpl, sym); - /* close the current print file, if necessary */ - if (mpl->prt_fp != NULL && - (!prt->app || strcmp(mpl->prt_file, fname) != 0)) - { glp_close(mpl->prt_fp), mpl->prt_fp = NULL; - xfree(mpl->prt_file), mpl->prt_file = NULL; - } - /* open the specified print file, if necessary */ - if (mpl->prt_fp == NULL) - { mpl->prt_fp = glp_open(fname, prt->app ? "a" : "w"); - if (mpl->prt_fp == NULL) - error(mpl, "unable to open '%s' for writing - %s", - fname, get_err_msg()); - mpl->prt_file = xmalloc(strlen(fname)+1); - strcpy(mpl->prt_file, fname); - } - } - loop_within_domain(mpl, prt->domain, prt, printf_func); - if (mpl->prt_fp != NULL) - { -#if 0 /* FIXME */ - xfflush(mpl->prt_fp); -#endif - if (glp_ioerr(mpl->prt_fp)) - error(mpl, "writing error to '%s' - %s", mpl->prt_file, - get_err_msg()); - } - return; -} -#endif - -/*---------------------------------------------------------------------- --- clean_printf - clean printf statement. --- --- This routine cleans specified printf statement that assumes deleting --- all stuff dynamically allocated on generating/postsolving phase. */ - -void clean_printf(MPL *mpl, PRINTF *prt) -{ PRINTF1 *p; - /* clean subscript domain */ - clean_domain(mpl, prt->domain); - /* clean pseudo-code for computing format string */ - clean_code(mpl, prt->fmt); - /* clean printf list */ - for (p = prt->list; p != NULL; p = p->next) - { /* clean pseudo-code for computing value to be printed */ - clean_code(mpl, p->code); - } -#if 1 /* 14/VII-2006 */ - /* clean pseudo-code for computing file name string */ - clean_code(mpl, prt->fname); -#endif - return; -} - -/*---------------------------------------------------------------------- --- execute_for - execute for statement. --- --- This routine executes specified for statement. */ - -static int for_func(MPL *mpl, void *info) -{ /* this is auxiliary routine to work within domain scope */ - FOR *fur = (FOR *)info; - STATEMENT *stmt, *save; - save = mpl->stmt; - for (stmt = fur->list; stmt != NULL; stmt = stmt->next) - execute_statement(mpl, stmt); - mpl->stmt = save; - return 0; -} - -void execute_for(MPL *mpl, FOR *fur) -{ loop_within_domain(mpl, fur->domain, fur, for_func); - return; -} - -/*---------------------------------------------------------------------- --- clean_for - clean for statement. --- --- This routine cleans specified for statement that assumes deleting all --- stuff dynamically allocated on generating/postsolving phase. */ - -void clean_for(MPL *mpl, FOR *fur) -{ STATEMENT *stmt; - /* clean subscript domain */ - clean_domain(mpl, fur->domain); - /* clean all sub-statements */ - for (stmt = fur->list; stmt != NULL; stmt = stmt->next) - clean_statement(mpl, stmt); - return; -} - -/*---------------------------------------------------------------------- --- execute_statement - execute specified model statement. --- --- This routine executes specified model statement. */ - -void execute_statement(MPL *mpl, STATEMENT *stmt) -{ mpl->stmt = stmt; - switch (stmt->type) - { case A_SET: - case A_PARAMETER: - case A_VARIABLE: - break; - case A_CONSTRAINT: - xprintf("Generating %s...\n", stmt->u.con->name); - eval_whole_con(mpl, stmt->u.con); - break; - case A_TABLE: - switch (stmt->u.tab->type) - { case A_INPUT: - xprintf("Reading %s...\n", stmt->u.tab->name); - break; - case A_OUTPUT: - xprintf("Writing %s...\n", stmt->u.tab->name); - break; - default: - xassert(stmt != stmt); - } - execute_table(mpl, stmt->u.tab); - break; - case A_SOLVE: - break; - case A_CHECK: - xprintf("Checking (line %d)...\n", stmt->line); - execute_check(mpl, stmt->u.chk); - break; - case A_DISPLAY: - write_text(mpl, "Display statement at line %d\n", - stmt->line); - execute_display(mpl, stmt->u.dpy); - break; - case A_PRINTF: - execute_printf(mpl, stmt->u.prt); - break; - case A_FOR: - execute_for(mpl, stmt->u.fur); - break; - default: - xassert(stmt != stmt); - } - return; -} - -/*---------------------------------------------------------------------- --- clean_statement - clean specified model statement. --- --- This routine cleans specified model statement that assumes deleting --- all stuff dynamically allocated on generating/postsolving phase. */ - -void clean_statement(MPL *mpl, STATEMENT *stmt) -{ switch(stmt->type) - { case A_SET: - clean_set(mpl, stmt->u.set); break; - case A_PARAMETER: - clean_parameter(mpl, stmt->u.par); break; - case A_VARIABLE: - clean_variable(mpl, stmt->u.var); break; - case A_CONSTRAINT: - clean_constraint(mpl, stmt->u.con); break; -#if 1 /* 11/II-2008 */ - case A_TABLE: - clean_table(mpl, stmt->u.tab); break; -#endif - case A_SOLVE: - break; - case A_CHECK: - clean_check(mpl, stmt->u.chk); break; - case A_DISPLAY: - clean_display(mpl, stmt->u.dpy); break; - case A_PRINTF: - clean_printf(mpl, stmt->u.prt); break; - case A_FOR: - clean_for(mpl, stmt->u.fur); break; - default: - xassert(stmt != stmt); - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl4.c b/code/3rd_glpk/mpl/mpl4.c deleted file mode 100644 index 6e80499c..00000000 --- a/code/3rd_glpk/mpl/mpl4.c +++ /dev/null @@ -1,1426 +0,0 @@ -/* mpl4.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mpl.h" - -#define xfault xerror -#define xfprintf glp_format -#define dmp_create_poolx(size) dmp_create_pool() - -/**********************************************************************/ -/* * * GENERATING AND POSTSOLVING MODEL * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- alloc_content - allocate content arrays for all model objects. --- --- This routine allocates content arrays for all existing model objects --- and thereby finalizes creating model. --- --- This routine must be called immediately after reading model section, --- i.e. before reading data section or generating model. */ - -void alloc_content(MPL *mpl) -{ STATEMENT *stmt; - /* walk through all model statements */ - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { switch (stmt->type) - { case A_SET: - /* model set */ - xassert(stmt->u.set->array == NULL); - stmt->u.set->array = create_array(mpl, A_ELEMSET, - stmt->u.set->dim); - break; - case A_PARAMETER: - /* model parameter */ - xassert(stmt->u.par->array == NULL); - switch (stmt->u.par->type) - { case A_NUMERIC: - case A_INTEGER: - case A_BINARY: - stmt->u.par->array = create_array(mpl, A_NUMERIC, - stmt->u.par->dim); - break; - case A_SYMBOLIC: - stmt->u.par->array = create_array(mpl, A_SYMBOLIC, - stmt->u.par->dim); - break; - default: - xassert(stmt != stmt); - } - break; - case A_VARIABLE: - /* model variable */ - xassert(stmt->u.var->array == NULL); - stmt->u.var->array = create_array(mpl, A_ELEMVAR, - stmt->u.var->dim); - break; - case A_CONSTRAINT: - /* model constraint/objective */ - xassert(stmt->u.con->array == NULL); - stmt->u.con->array = create_array(mpl, A_ELEMCON, - stmt->u.con->dim); - break; -#if 1 /* 11/II-2008 */ - case A_TABLE: -#endif - case A_SOLVE: - case A_CHECK: - case A_DISPLAY: - case A_PRINTF: - case A_FOR: - /* functional statements have no content array */ - break; - default: - xassert(stmt != stmt); - } - } - return; -} - -/*---------------------------------------------------------------------- --- generate_model - generate model. --- --- This routine executes the model statements which precede the solve --- statement. */ - -void generate_model(MPL *mpl) -{ STATEMENT *stmt; - xassert(!mpl->flag_p); - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { execute_statement(mpl, stmt); - if (mpl->stmt->type == A_SOLVE) break; - } - mpl->stmt = stmt; - return; -} - -/*---------------------------------------------------------------------- --- build_problem - build problem instance. --- --- This routine builds lists of rows and columns for problem instance, --- which corresponds to the generated model. */ - -void build_problem(MPL *mpl) -{ STATEMENT *stmt; - MEMBER *memb; - VARIABLE *v; - CONSTRAINT *c; - FORMULA *t; - int i, j; - xassert(mpl->m == 0); - xassert(mpl->n == 0); - xassert(mpl->row == NULL); - xassert(mpl->col == NULL); - /* check that all elemental variables has zero column numbers */ - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { if (stmt->type == A_VARIABLE) - { v = stmt->u.var; - for (memb = v->array->head; memb != NULL; memb = memb->next) - xassert(memb->value.var->j == 0); - } - } - /* assign row numbers to elemental constraints and objectives */ - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { if (stmt->type == A_CONSTRAINT) - { c = stmt->u.con; - for (memb = c->array->head; memb != NULL; memb = memb->next) - { xassert(memb->value.con->i == 0); - memb->value.con->i = ++mpl->m; - /* walk through linear form and mark elemental variables, - which are referenced at least once */ - for (t = memb->value.con->form; t != NULL; t = t->next) - { xassert(t->var != NULL); - t->var->memb->value.var->j = -1; - } - } - } - } - /* assign column numbers to marked elemental variables */ - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { if (stmt->type == A_VARIABLE) - { v = stmt->u.var; - for (memb = v->array->head; memb != NULL; memb = memb->next) - if (memb->value.var->j != 0) memb->value.var->j = - ++mpl->n; - } - } - /* build list of rows */ - mpl->row = xcalloc(1+mpl->m, sizeof(ELEMCON *)); - for (i = 1; i <= mpl->m; i++) mpl->row[i] = NULL; - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { if (stmt->type == A_CONSTRAINT) - { c = stmt->u.con; - for (memb = c->array->head; memb != NULL; memb = memb->next) - { i = memb->value.con->i; - xassert(1 <= i && i <= mpl->m); - xassert(mpl->row[i] == NULL); - mpl->row[i] = memb->value.con; - } - } - } - for (i = 1; i <= mpl->m; i++) xassert(mpl->row[i] != NULL); - /* build list of columns */ - mpl->col = xcalloc(1+mpl->n, sizeof(ELEMVAR *)); - for (j = 1; j <= mpl->n; j++) mpl->col[j] = NULL; - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - { if (stmt->type == A_VARIABLE) - { v = stmt->u.var; - for (memb = v->array->head; memb != NULL; memb = memb->next) - { j = memb->value.var->j; - if (j == 0) continue; - xassert(1 <= j && j <= mpl->n); - xassert(mpl->col[j] == NULL); - mpl->col[j] = memb->value.var; - } - } - } - for (j = 1; j <= mpl->n; j++) xassert(mpl->col[j] != NULL); - return; -} - -/*---------------------------------------------------------------------- --- postsolve_model - postsolve model. --- --- This routine executes the model statements which follow the solve --- statement. */ - -void postsolve_model(MPL *mpl) -{ STATEMENT *stmt; - xassert(!mpl->flag_p); - mpl->flag_p = 1; - for (stmt = mpl->stmt; stmt != NULL; stmt = stmt->next) - execute_statement(mpl, stmt); - mpl->stmt = NULL; - return; -} - -/*---------------------------------------------------------------------- --- clean_model - clean model content. --- --- This routine cleans the model content that assumes deleting all stuff --- dynamically allocated on generating/postsolving phase. --- --- Actually cleaning model content is not needed. This function is used --- mainly to be sure that there were no logical errors on using dynamic --- memory pools during the generation phase. --- --- NOTE: This routine must not be called if any errors were detected on --- the generation phase. */ - -void clean_model(MPL *mpl) -{ STATEMENT *stmt; - for (stmt = mpl->model; stmt != NULL; stmt = stmt->next) - clean_statement(mpl, stmt); - /* check that all atoms have been returned to their pools */ - if (dmp_in_use(mpl->strings) != 0) - error(mpl, "internal logic error: %d string segment(s) were lo" - "st", dmp_in_use(mpl->strings)); - if (dmp_in_use(mpl->symbols) != 0) - error(mpl, "internal logic error: %d symbol(s) were lost", - dmp_in_use(mpl->symbols)); - if (dmp_in_use(mpl->tuples) != 0) - error(mpl, "internal logic error: %d n-tuple component(s) were" - " lost", dmp_in_use(mpl->tuples)); - if (dmp_in_use(mpl->arrays) != 0) - error(mpl, "internal logic error: %d array(s) were lost", - dmp_in_use(mpl->arrays)); - if (dmp_in_use(mpl->members) != 0) - error(mpl, "internal logic error: %d array member(s) were lost" - , dmp_in_use(mpl->members)); - if (dmp_in_use(mpl->elemvars) != 0) - error(mpl, "internal logic error: %d elemental variable(s) wer" - "e lost", dmp_in_use(mpl->elemvars)); - if (dmp_in_use(mpl->formulae) != 0) - error(mpl, "internal logic error: %d linear term(s) were lost", - dmp_in_use(mpl->formulae)); - if (dmp_in_use(mpl->elemcons) != 0) - error(mpl, "internal logic error: %d elemental constraint(s) w" - "ere lost", dmp_in_use(mpl->elemcons)); - return; -} - -/**********************************************************************/ -/* * * INPUT/OUTPUT * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- open_input - open input text file. --- --- This routine opens the input text file for scanning. */ - -void open_input(MPL *mpl, char *file) -{ mpl->line = 0; - mpl->c = '\n'; - mpl->token = 0; - mpl->imlen = 0; - mpl->image[0] = '\0'; - mpl->value = 0.0; - mpl->b_token = T_EOF; - mpl->b_imlen = 0; - mpl->b_image[0] = '\0'; - mpl->b_value = 0.0; - mpl->f_dots = 0; - mpl->f_scan = 0; - mpl->f_token = 0; - mpl->f_imlen = 0; - mpl->f_image[0] = '\0'; - mpl->f_value = 0.0; - memset(mpl->context, ' ', CONTEXT_SIZE); - mpl->c_ptr = 0; - xassert(mpl->in_fp == NULL); - mpl->in_fp = glp_open(file, "r"); - if (mpl->in_fp == NULL) - error(mpl, "unable to open %s - %s", file, get_err_msg()); - mpl->in_file = file; - /* scan the very first character */ - get_char(mpl); - /* scan the very first token */ - get_token(mpl); - return; -} - -/*---------------------------------------------------------------------- --- read_char - read next character from input text file. --- --- This routine returns a next ASCII character read from the input text --- file. If the end of file has been reached, EOF is returned. */ - -int read_char(MPL *mpl) -{ int c; - xassert(mpl->in_fp != NULL); - c = glp_getc(mpl->in_fp); - if (c < 0) - { if (glp_ioerr(mpl->in_fp)) - error(mpl, "read error on %s - %s", mpl->in_file, - get_err_msg()); - c = EOF; - } - return c; -} - -/*---------------------------------------------------------------------- --- close_input - close input text file. --- --- This routine closes the input text file. */ - -void close_input(MPL *mpl) -{ xassert(mpl->in_fp != NULL); - glp_close(mpl->in_fp); - mpl->in_fp = NULL; - mpl->in_file = NULL; - return; -} - -/*---------------------------------------------------------------------- --- open_output - open output text file. --- --- This routine opens the output text file for writing data produced by --- display and printf statements. */ - -void open_output(MPL *mpl, char *file) -{ xassert(mpl->out_fp == NULL); - if (file == NULL) - { file = ""; - mpl->out_fp = (void *)stdout; - } - else - { mpl->out_fp = glp_open(file, "w"); - if (mpl->out_fp == NULL) - error(mpl, "unable to create %s - %s", file, get_err_msg()); - } - mpl->out_file = xmalloc(strlen(file)+1); - strcpy(mpl->out_file, file); - return; -} - -/*---------------------------------------------------------------------- --- write_char - write next character to output text file. --- --- This routine writes an ASCII character to the output text file. */ - -void write_char(MPL *mpl, int c) -{ xassert(mpl->out_fp != NULL); - if (mpl->out_fp == (void *)stdout) - xprintf("%c", c); - else - xfprintf(mpl->out_fp, "%c", c); - return; -} - -/*---------------------------------------------------------------------- --- write_text - format and write text to output text file. --- --- This routine formats a text using the format control string and then --- writes this text to the output text file. */ - -void write_text(MPL *mpl, char *fmt, ...) -{ va_list arg; - char buf[OUTBUF_SIZE], *c; - va_start(arg, fmt); - vsprintf(buf, fmt, arg); - xassert(strlen(buf) < sizeof(buf)); - va_end(arg); - for (c = buf; *c != '\0'; c++) write_char(mpl, *c); - return; -} - -/*---------------------------------------------------------------------- --- flush_output - finalize writing data to output text file. --- --- This routine finalizes writing data to the output text file. */ - -void flush_output(MPL *mpl) -{ xassert(mpl->out_fp != NULL); - if (mpl->out_fp != (void *)stdout) - { -#if 0 /* FIXME */ - xfflush(mpl->out_fp); -#endif - if (glp_ioerr(mpl->out_fp)) - error(mpl, "write error on %s - %s", mpl->out_file, - get_err_msg()); - } - return; -} - -/**********************************************************************/ -/* * * SOLVER INTERFACE * * */ -/**********************************************************************/ - -/*---------------------------------------------------------------------- --- error - print error message and terminate model processing. --- --- This routine formats and prints an error message and then terminates --- model processing. */ - -void error(MPL *mpl, char *fmt, ...) -{ va_list arg; - char msg[4095+1]; - va_start(arg, fmt); - vsprintf(msg, fmt, arg); - xassert(strlen(msg) < sizeof(msg)); - va_end(arg); - switch (mpl->phase) - { case 1: - case 2: - /* translation phase */ - xprintf("%s:%d: %s\n", - mpl->in_file == NULL ? "(unknown)" : mpl->in_file, - mpl->line, msg); - print_context(mpl); - break; - case 3: - /* generation/postsolve phase */ - xprintf("%s:%d: %s\n", - mpl->mod_file == NULL ? "(unknown)" : mpl->mod_file, - mpl->stmt == NULL ? 0 : mpl->stmt->line, msg); - break; - default: - xassert(mpl != mpl); - } - mpl->phase = 4; - longjmp(mpl->jump, 1); - /* no return */ -} - -/*---------------------------------------------------------------------- --- warning - print warning message and continue model processing. --- --- This routine formats and prints a warning message and returns to the --- calling program. */ - -void warning(MPL *mpl, char *fmt, ...) -{ va_list arg; - char msg[4095+1]; - va_start(arg, fmt); - vsprintf(msg, fmt, arg); - xassert(strlen(msg) < sizeof(msg)); - va_end(arg); - switch (mpl->phase) - { case 1: - case 2: - /* translation phase */ - xprintf("%s:%d: warning: %s\n", - mpl->in_file == NULL ? "(unknown)" : mpl->in_file, - mpl->line, msg); - break; - case 3: - /* generation/postsolve phase */ - xprintf("%s:%d: warning: %s\n", - mpl->mod_file == NULL ? "(unknown)" : mpl->mod_file, - mpl->stmt == NULL ? 0 : mpl->stmt->line, msg); - break; - default: - xassert(mpl != mpl); - } - return; -} - -/*---------------------------------------------------------------------- --- mpl_initialize - create and initialize translator database. --- --- *Synopsis* --- --- #include "glpmpl.h" --- MPL *mpl_initialize(void); --- --- *Description* --- --- The routine mpl_initialize creates and initializes the database used --- by the GNU MathProg translator. --- --- *Returns* --- --- The routine returns a pointer to the database created. */ - -MPL *mpl_initialize(void) -{ MPL *mpl; - mpl = xmalloc(sizeof(MPL)); - /* scanning segment */ - mpl->line = 0; - mpl->c = 0; - mpl->token = 0; - mpl->imlen = 0; - mpl->image = xcalloc(MAX_LENGTH+1, sizeof(char)); - mpl->image[0] = '\0'; - mpl->value = 0.0; - mpl->b_token = 0; - mpl->b_imlen = 0; - mpl->b_image = xcalloc(MAX_LENGTH+1, sizeof(char)); - mpl->b_image[0] = '\0'; - mpl->b_value = 0.0; - mpl->f_dots = 0; - mpl->f_scan = 0; - mpl->f_token = 0; - mpl->f_imlen = 0; - mpl->f_image = xcalloc(MAX_LENGTH+1, sizeof(char)); - mpl->f_image[0] = '\0'; - mpl->f_value = 0.0; - mpl->context = xcalloc(CONTEXT_SIZE, sizeof(char)); - memset(mpl->context, ' ', CONTEXT_SIZE); - mpl->c_ptr = 0; - mpl->flag_d = 0; - /* translating segment */ - mpl->pool = dmp_create_poolx(0); - mpl->tree = avl_create_tree(avl_strcmp, NULL); - mpl->model = NULL; - mpl->flag_x = 0; - mpl->as_within = 0; - mpl->as_in = 0; - mpl->as_binary = 0; - mpl->flag_s = 0; - /* common segment */ - mpl->strings = dmp_create_poolx(sizeof(STRING)); - mpl->symbols = dmp_create_poolx(sizeof(SYMBOL)); - mpl->tuples = dmp_create_poolx(sizeof(TUPLE)); - mpl->arrays = dmp_create_poolx(sizeof(ARRAY)); - mpl->members = dmp_create_poolx(sizeof(MEMBER)); - mpl->elemvars = dmp_create_poolx(sizeof(ELEMVAR)); - mpl->formulae = dmp_create_poolx(sizeof(FORMULA)); - mpl->elemcons = dmp_create_poolx(sizeof(ELEMCON)); - mpl->a_list = NULL; - mpl->sym_buf = xcalloc(255+1, sizeof(char)); - mpl->sym_buf[0] = '\0'; - mpl->tup_buf = xcalloc(255+1, sizeof(char)); - mpl->tup_buf[0] = '\0'; - /* generating/postsolving segment */ - mpl->rand = rng_create_rand(); - mpl->flag_p = 0; - mpl->stmt = NULL; -#if 1 /* 11/II-2008 */ - mpl->dca = NULL; -#endif - mpl->m = 0; - mpl->n = 0; - mpl->row = NULL; - mpl->col = NULL; - /* input/output segment */ - mpl->in_fp = NULL; - mpl->in_file = NULL; - mpl->out_fp = NULL; - mpl->out_file = NULL; - mpl->prt_fp = NULL; - mpl->prt_file = NULL; - /* solver interface segment */ - if (setjmp(mpl->jump)) xassert(mpl != mpl); - mpl->phase = 0; - mpl->mod_file = NULL; - mpl->mpl_buf = xcalloc(255+1, sizeof(char)); - mpl->mpl_buf[0] = '\0'; - return mpl; -} - -/*---------------------------------------------------------------------- --- mpl_read_model - read model section and optional data section. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_read_model(MPL *mpl, char *file, int skip_data); --- --- *Description* --- --- The routine mpl_read_model reads model section and optionally data --- section, which may follow the model section, from the text file, --- whose name is the character string file, performs translating model --- statements and data blocks, and stores all the information in the --- translator database. --- --- The parameter skip_data is a flag. If the input file contains the --- data section and this flag is set, the data section is not read as --- if there were no data section and a warning message is issued. This --- allows reading the data section from another input file. --- --- This routine should be called once after the routine mpl_initialize --- and before other API routines. --- --- *Returns* --- --- The routine mpl_read_model returns one the following codes: --- --- 1 - translation successful. The input text file contains only model --- section. In this case the calling program may call the routine --- mpl_read_data to read data section from another file. --- 2 - translation successful. The input text file contains both model --- and data section. --- 4 - processing failed due to some errors. In this case the calling --- program should call the routine mpl_terminate to terminate model --- processing. */ - -int mpl_read_model(MPL *mpl, char *file, int skip_data) -{ if (mpl->phase != 0) - xfault("mpl_read_model: invalid call sequence\n"); - if (file == NULL) - xfault("mpl_read_model: no input filename specified\n"); - /* set up error handler */ - if (setjmp(mpl->jump)) goto done; - /* translate model section */ - mpl->phase = 1; - xprintf("Reading model section from %s...\n", file); - open_input(mpl, file); - model_section(mpl); - if (mpl->model == NULL) - error(mpl, "empty model section not allowed"); - /* save name of the input text file containing model section for - error diagnostics during the generation phase */ - mpl->mod_file = xcalloc(strlen(file)+1, sizeof(char)); - strcpy(mpl->mod_file, mpl->in_file); - /* allocate content arrays for all model objects */ - alloc_content(mpl); - /* optional data section may begin with the keyword 'data' */ - if (is_keyword(mpl, "data")) - { if (skip_data) - { warning(mpl, "data section ignored"); - goto skip; - } - mpl->flag_d = 1; - get_token(mpl /* data */); - if (mpl->token != T_SEMICOLON) - error(mpl, "semicolon missing where expected"); - get_token(mpl /* ; */); - /* translate data section */ - mpl->phase = 2; - xprintf("Reading data section from %s...\n", file); - data_section(mpl); - } - /* process end statement */ - end_statement(mpl); -skip: xprintf("%d line%s were read\n", - mpl->line, mpl->line == 1 ? "" : "s"); - close_input(mpl); -done: /* return to the calling program */ - return mpl->phase; -} - -/*---------------------------------------------------------------------- --- mpl_read_data - read data section. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_read_data(MPL *mpl, char *file); --- --- *Description* --- --- The routine mpl_read_data reads data section from the text file, --- whose name is the character string file, performs translating data --- blocks, and stores the data read in the translator database. --- --- If this routine is used, it should be called once after the routine --- mpl_read_model and if the latter returned the code 1. --- --- *Returns* --- --- The routine mpl_read_data returns one of the following codes: --- --- 2 - data section has been successfully processed. --- 4 - processing failed due to some errors. In this case the calling --- program should call the routine mpl_terminate to terminate model --- processing. */ - -int mpl_read_data(MPL *mpl, char *file) -#if 0 /* 02/X-2008 */ -{ if (mpl->phase != 1) -#else -{ if (!(mpl->phase == 1 || mpl->phase == 2)) -#endif - xfault("mpl_read_data: invalid call sequence\n"); - if (file == NULL) - xfault("mpl_read_data: no input filename specified\n"); - /* set up error handler */ - if (setjmp(mpl->jump)) goto done; - /* process data section */ - mpl->phase = 2; - xprintf("Reading data section from %s...\n", file); - mpl->flag_d = 1; - open_input(mpl, file); - /* in this case the keyword 'data' is optional */ - if (is_literal(mpl, "data")) - { get_token(mpl /* data */); - if (mpl->token != T_SEMICOLON) - error(mpl, "semicolon missing where expected"); - get_token(mpl /* ; */); - } - data_section(mpl); - /* process end statement */ - end_statement(mpl); - xprintf("%d line%s were read\n", - mpl->line, mpl->line == 1 ? "" : "s"); - close_input(mpl); -done: /* return to the calling program */ - return mpl->phase; -} - -/*---------------------------------------------------------------------- --- mpl_generate - generate model. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_generate(MPL *mpl, char *file); --- --- *Description* --- --- The routine mpl_generate generates the model using its description --- stored in the translator database. This phase means generating all --- variables, constraints, and objectives, executing check and display --- statements, which precede the solve statement (if it is presented), --- and building the problem instance. --- --- The character string file specifies the name of output text file, to --- which output produced by display statements should be written. It is --- allowed to specify NULL, in which case the output goes to stdout via --- the routine print. --- --- This routine should be called once after the routine mpl_read_model --- or mpl_read_data and if one of the latters returned the code 2. --- --- *Returns* --- --- The routine mpl_generate returns one of the following codes: --- --- 3 - model has been successfully generated. In this case the calling --- program may call other api routines to obtain components of the --- problem instance from the translator database. --- 4 - processing failed due to some errors. In this case the calling --- program should call the routine mpl_terminate to terminate model --- processing. */ - -int mpl_generate(MPL *mpl, char *file) -{ if (!(mpl->phase == 1 || mpl->phase == 2)) - xfault("mpl_generate: invalid call sequence\n"); - /* set up error handler */ - if (setjmp(mpl->jump)) goto done; - /* generate model */ - mpl->phase = 3; - open_output(mpl, file); - generate_model(mpl); - flush_output(mpl); - /* build problem instance */ - build_problem(mpl); - /* generation phase has been finished */ - xprintf("Model has been successfully generated\n"); -done: /* return to the calling program */ - return mpl->phase; -} - -/*---------------------------------------------------------------------- --- mpl_get_prob_name - obtain problem (model) name. --- --- *Synopsis* --- --- #include "glpmpl.h" --- char *mpl_get_prob_name(MPL *mpl); --- --- *Returns* --- --- The routine mpl_get_prob_name returns a pointer to internal buffer, --- which contains symbolic name of the problem (model). --- --- *Note* --- --- Currently MathProg has no feature to assign a symbolic name to the --- model. Therefore the routine mpl_get_prob_name tries to construct --- such name using the name of input text file containing model section, --- although this is not a good idea (due to portability problems). */ - -char *mpl_get_prob_name(MPL *mpl) -{ char *name = mpl->mpl_buf; - char *file = mpl->mod_file; - int k; - if (mpl->phase != 3) - xfault("mpl_get_prob_name: invalid call sequence\n"); - for (;;) - { if (strchr(file, '/') != NULL) - file = strchr(file, '/') + 1; - else if (strchr(file, '\\') != NULL) - file = strchr(file, '\\') + 1; - else if (strchr(file, ':') != NULL) - file = strchr(file, ':') + 1; - else - break; - } - for (k = 0; ; k++) - { if (k == 255) break; - if (!(isalnum((unsigned char)*file) || *file == '_')) break; - name[k] = *file++; - } - if (k == 0) - strcpy(name, "Unknown"); - else - name[k] = '\0'; - xassert(strlen(name) <= 255); - return name; -} - -/*---------------------------------------------------------------------- --- mpl_get_num_rows - determine number of rows. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_num_rows(MPL *mpl); --- --- *Returns* --- --- The routine mpl_get_num_rows returns total number of rows in the --- problem, where each row is an individual constraint or objective. */ - -int mpl_get_num_rows(MPL *mpl) -{ if (mpl->phase != 3) - xfault("mpl_get_num_rows: invalid call sequence\n"); - return mpl->m; -} - -/*---------------------------------------------------------------------- --- mpl_get_num_cols - determine number of columns. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_num_cols(MPL *mpl); --- --- *Returns* --- --- The routine mpl_get_num_cols returns total number of columns in the --- problem, where each column is an individual variable. */ - -int mpl_get_num_cols(MPL *mpl) -{ if (mpl->phase != 3) - xfault("mpl_get_num_cols: invalid call sequence\n"); - return mpl->n; -} - -/*---------------------------------------------------------------------- --- mpl_get_row_name - obtain row name. --- --- *Synopsis* --- --- #include "glpmpl.h" --- char *mpl_get_row_name(MPL *mpl, int i); --- --- *Returns* --- --- The routine mpl_get_row_name returns a pointer to internal buffer, --- which contains symbolic name of i-th row of the problem. */ - -char *mpl_get_row_name(MPL *mpl, int i) -{ char *name = mpl->mpl_buf, *t; - int len; - if (mpl->phase != 3) - xfault("mpl_get_row_name: invalid call sequence\n"); - if (!(1 <= i && i <= mpl->m)) - xfault("mpl_get_row_name: i = %d; row number out of range\n", - i); - strcpy(name, mpl->row[i]->con->name); - len = strlen(name); - xassert(len <= 255); - t = format_tuple(mpl, '[', mpl->row[i]->memb->tuple); - while (*t) - { if (len == 255) break; - name[len++] = *t++; - } - name[len] = '\0'; - if (len == 255) strcpy(name+252, "..."); - xassert(strlen(name) <= 255); - return name; -} - -/*---------------------------------------------------------------------- --- mpl_get_row_kind - determine row kind. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_row_kind(MPL *mpl, int i); --- --- *Returns* --- --- The routine mpl_get_row_kind returns the kind of i-th row, which can --- be one of the following: --- --- MPL_ST - non-free (constraint) row; --- MPL_MIN - free (objective) row to be minimized; --- MPL_MAX - free (objective) row to be maximized. */ - -int mpl_get_row_kind(MPL *mpl, int i) -{ int kind; - if (mpl->phase != 3) - xfault("mpl_get_row_kind: invalid call sequence\n"); - if (!(1 <= i && i <= mpl->m)) - xfault("mpl_get_row_kind: i = %d; row number out of range\n", - i); - switch (mpl->row[i]->con->type) - { case A_CONSTRAINT: - kind = MPL_ST; break; - case A_MINIMIZE: - kind = MPL_MIN; break; - case A_MAXIMIZE: - kind = MPL_MAX; break; - default: - xassert(mpl != mpl); - } - return kind; -} - -/*---------------------------------------------------------------------- --- mpl_get_row_bnds - obtain row bounds. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_row_bnds(MPL *mpl, int i, double *lb, double *ub); --- --- *Description* --- --- The routine mpl_get_row_bnds stores lower and upper bounds of i-th --- row of the problem to the locations, which the parameters lb and ub --- point to, respectively. Besides the routine returns the type of the --- i-th row. --- --- If some of the parameters lb and ub is NULL, the corresponding bound --- value is not stored. --- --- Types and bounds have the following meaning: --- --- Type Bounds Note --- ----------------------------------------------------------- --- MPL_FR -inf < f(x) < +inf Free linear form --- MPL_LO lb <= f(x) < +inf Inequality f(x) >= lb --- MPL_UP -inf < f(x) <= ub Inequality f(x) <= ub --- MPL_DB lb <= f(x) <= ub Inequality lb <= f(x) <= ub --- MPL_FX f(x) = lb Equality f(x) = lb --- --- where f(x) is the corresponding linear form of the i-th row. --- --- If the row has no lower bound, *lb is set to zero; if the row has --- no upper bound, *ub is set to zero; and if the row is of fixed type, --- both *lb and *ub are set to the same value. --- --- *Returns* --- --- The routine returns the type of the i-th row as it is stated in the --- table above. */ - -int mpl_get_row_bnds(MPL *mpl, int i, double *_lb, double *_ub) -{ ELEMCON *con; - int type; - double lb, ub; - if (mpl->phase != 3) - xfault("mpl_get_row_bnds: invalid call sequence\n"); - if (!(1 <= i && i <= mpl->m)) - xfault("mpl_get_row_bnds: i = %d; row number out of range\n", - i); - con = mpl->row[i]; -#if 0 /* 21/VII-2006 */ - if (con->con->lbnd == NULL && con->con->ubnd == NULL) - type = MPL_FR, lb = ub = 0.0; - else if (con->con->ubnd == NULL) - type = MPL_LO, lb = con->lbnd, ub = 0.0; - else if (con->con->lbnd == NULL) - type = MPL_UP, lb = 0.0, ub = con->ubnd; - else if (con->con->lbnd != con->con->ubnd) - type = MPL_DB, lb = con->lbnd, ub = con->ubnd; - else - type = MPL_FX, lb = ub = con->lbnd; -#else - lb = (con->con->lbnd == NULL ? -DBL_MAX : con->lbnd); - ub = (con->con->ubnd == NULL ? +DBL_MAX : con->ubnd); - if (lb == -DBL_MAX && ub == +DBL_MAX) - type = MPL_FR, lb = ub = 0.0; - else if (ub == +DBL_MAX) - type = MPL_LO, ub = 0.0; - else if (lb == -DBL_MAX) - type = MPL_UP, lb = 0.0; - else if (con->con->lbnd != con->con->ubnd) - type = MPL_DB; - else - type = MPL_FX; -#endif - if (_lb != NULL) *_lb = lb; - if (_ub != NULL) *_ub = ub; - return type; -} - -/*---------------------------------------------------------------------- --- mpl_get_mat_row - obtain row of the constraint matrix. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_mat_row(MPL *mpl, int i, int ndx[], double val[]); --- --- *Description* --- --- The routine mpl_get_mat_row stores column indices and numeric values --- of constraint coefficients for the i-th row to locations ndx[1], ..., --- ndx[len] and val[1], ..., val[len], respectively, where 0 <= len <= n --- is number of (structural) non-zero constraint coefficients, and n is --- number of columns in the problem. --- --- If the parameter ndx is NULL, column indices are not stored. If the --- parameter val is NULL, numeric values are not stored. --- --- Note that free rows may have constant terms, which are not part of --- the constraint matrix and therefore not reported by this routine. The --- constant term of a particular row can be obtained, if necessary, via --- the routine mpl_get_row_c0. --- --- *Returns* --- --- The routine mpl_get_mat_row returns len, which is length of i-th row --- of the constraint matrix (i.e. number of non-zero coefficients). */ - -int mpl_get_mat_row(MPL *mpl, int i, int ndx[], double val[]) -{ FORMULA *term; - int len = 0; - if (mpl->phase != 3) - xfault("mpl_get_mat_row: invalid call sequence\n"); - if (!(1 <= i && i <= mpl->m)) - xfault("mpl_get_mat_row: i = %d; row number out of range\n", - i); - for (term = mpl->row[i]->form; term != NULL; term = term->next) - { xassert(term->var != NULL); - len++; - xassert(len <= mpl->n); - if (ndx != NULL) ndx[len] = term->var->j; - if (val != NULL) val[len] = term->coef; - } - return len; -} - -/*---------------------------------------------------------------------- --- mpl_get_row_c0 - obtain constant term of free row. --- --- *Synopsis* --- --- #include "glpmpl.h" --- double mpl_get_row_c0(MPL *mpl, int i); --- --- *Returns* --- --- The routine mpl_get_row_c0 returns numeric value of constant term of --- i-th row. --- --- Note that only free rows may have non-zero constant terms. Therefore --- if i-th row is not free, the routine returns zero. */ - -double mpl_get_row_c0(MPL *mpl, int i) -{ ELEMCON *con; - double c0; - if (mpl->phase != 3) - xfault("mpl_get_row_c0: invalid call sequence\n"); - if (!(1 <= i && i <= mpl->m)) - xfault("mpl_get_row_c0: i = %d; row number out of range\n", - i); - con = mpl->row[i]; - if (con->con->lbnd == NULL && con->con->ubnd == NULL) - c0 = - con->lbnd; - else - c0 = 0.0; - return c0; -} - -/*---------------------------------------------------------------------- --- mpl_get_col_name - obtain column name. --- --- *Synopsis* --- --- #include "glpmpl.h" --- char *mpl_get_col_name(MPL *mpl, int j); --- --- *Returns* --- --- The routine mpl_get_col_name returns a pointer to internal buffer, --- which contains symbolic name of j-th column of the problem. */ - -char *mpl_get_col_name(MPL *mpl, int j) -{ char *name = mpl->mpl_buf, *t; - int len; - if (mpl->phase != 3) - xfault("mpl_get_col_name: invalid call sequence\n"); - if (!(1 <= j && j <= mpl->n)) - xfault("mpl_get_col_name: j = %d; column number out of range\n" - , j); - strcpy(name, mpl->col[j]->var->name); - len = strlen(name); - xassert(len <= 255); - t = format_tuple(mpl, '[', mpl->col[j]->memb->tuple); - while (*t) - { if (len == 255) break; - name[len++] = *t++; - } - name[len] = '\0'; - if (len == 255) strcpy(name+252, "..."); - xassert(strlen(name) <= 255); - return name; -} - -/*---------------------------------------------------------------------- --- mpl_get_col_kind - determine column kind. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_col_kind(MPL *mpl, int j); --- --- *Returns* --- --- The routine mpl_get_col_kind returns the kind of j-th column, which --- can be one of the following: --- --- MPL_NUM - continuous variable; --- MPL_INT - integer variable; --- MPL_BIN - binary variable. --- --- Note that column kinds are defined independently on type and bounds --- (reported by the routine mpl_get_col_bnds) of corresponding columns. --- This means, in particular, that bounds of an integer column may be --- fractional, or a binary column may have lower and upper bounds that --- are not 0 and 1 (or it may have no lower/upper bound at all). */ - -int mpl_get_col_kind(MPL *mpl, int j) -{ int kind; - if (mpl->phase != 3) - xfault("mpl_get_col_kind: invalid call sequence\n"); - if (!(1 <= j && j <= mpl->n)) - xfault("mpl_get_col_kind: j = %d; column number out of range\n" - , j); - switch (mpl->col[j]->var->type) - { case A_NUMERIC: - kind = MPL_NUM; break; - case A_INTEGER: - kind = MPL_INT; break; - case A_BINARY: - kind = MPL_BIN; break; - default: - xassert(mpl != mpl); - } - return kind; -} - -/*---------------------------------------------------------------------- --- mpl_get_col_bnds - obtain column bounds. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_get_col_bnds(MPL *mpl, int j, double *lb, double *ub); --- --- *Description* --- --- The routine mpl_get_col_bnds stores lower and upper bound of j-th --- column of the problem to the locations, which the parameters lb and --- ub point to, respectively. Besides the routine returns the type of --- the j-th column. --- --- If some of the parameters lb and ub is NULL, the corresponding bound --- value is not stored. --- --- Types and bounds have the following meaning: --- --- Type Bounds Note --- ------------------------------------------------------ --- MPL_FR -inf < x < +inf Free (unbounded) variable --- MPL_LO lb <= x < +inf Variable with lower bound --- MPL_UP -inf < x <= ub Variable with upper bound --- MPL_DB lb <= x <= ub Double-bounded variable --- MPL_FX x = lb Fixed variable --- --- where x is individual variable corresponding to the j-th column. --- --- If the column has no lower bound, *lb is set to zero; if the column --- has no upper bound, *ub is set to zero; and if the column is of fixed --- type, both *lb and *ub are set to the same value. --- --- *Returns* --- --- The routine returns the type of the j-th column as it is stated in --- the table above. */ - -int mpl_get_col_bnds(MPL *mpl, int j, double *_lb, double *_ub) -{ ELEMVAR *var; - int type; - double lb, ub; - if (mpl->phase != 3) - xfault("mpl_get_col_bnds: invalid call sequence\n"); - if (!(1 <= j && j <= mpl->n)) - xfault("mpl_get_col_bnds: j = %d; column number out of range\n" - , j); - var = mpl->col[j]; -#if 0 /* 21/VII-2006 */ - if (var->var->lbnd == NULL && var->var->ubnd == NULL) - type = MPL_FR, lb = ub = 0.0; - else if (var->var->ubnd == NULL) - type = MPL_LO, lb = var->lbnd, ub = 0.0; - else if (var->var->lbnd == NULL) - type = MPL_UP, lb = 0.0, ub = var->ubnd; - else if (var->var->lbnd != var->var->ubnd) - type = MPL_DB, lb = var->lbnd, ub = var->ubnd; - else - type = MPL_FX, lb = ub = var->lbnd; -#else - lb = (var->var->lbnd == NULL ? -DBL_MAX : var->lbnd); - ub = (var->var->ubnd == NULL ? +DBL_MAX : var->ubnd); - if (lb == -DBL_MAX && ub == +DBL_MAX) - type = MPL_FR, lb = ub = 0.0; - else if (ub == +DBL_MAX) - type = MPL_LO, ub = 0.0; - else if (lb == -DBL_MAX) - type = MPL_UP, lb = 0.0; - else if (var->var->lbnd != var->var->ubnd) - type = MPL_DB; - else - type = MPL_FX; -#endif - if (_lb != NULL) *_lb = lb; - if (_ub != NULL) *_ub = ub; - return type; -} - -/*---------------------------------------------------------------------- --- mpl_has_solve_stmt - check if model has solve statement. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_has_solve_stmt(MPL *mpl); --- --- *Returns* --- --- If the model has the solve statement, the routine returns non-zero, --- otherwise zero is returned. */ - -int mpl_has_solve_stmt(MPL *mpl) -{ if (mpl->phase != 3) - xfault("mpl_has_solve_stmt: invalid call sequence\n"); - return mpl->flag_s; -} - -#if 1 /* 15/V-2010 */ -void mpl_put_row_soln(MPL *mpl, int i, int stat, double prim, - double dual) -{ /* store row (constraint/objective) solution components */ - xassert(mpl->phase == 3); - xassert(1 <= i && i <= mpl->m); - mpl->row[i]->stat = stat; - mpl->row[i]->prim = prim; - mpl->row[i]->dual = dual; - return; -} -#endif - -#if 1 /* 15/V-2010 */ -void mpl_put_col_soln(MPL *mpl, int j, int stat, double prim, - double dual) -{ /* store column (variable) solution components */ - xassert(mpl->phase == 3); - xassert(1 <= j && j <= mpl->n); - mpl->col[j]->stat = stat; - mpl->col[j]->prim = prim; - mpl->col[j]->dual = dual; - return; -} -#endif - -#if 0 /* 15/V-2010 */ -/*---------------------------------------------------------------------- --- mpl_put_col_value - store column value. --- --- *Synopsis* --- --- #include "glpmpl.h" --- void mpl_put_col_value(MPL *mpl, int j, double val); --- --- *Description* --- --- The routine mpl_put_col_value stores numeric value of j-th column --- into the translator database. It is assumed that the column value is --- provided by the solver. */ - -void mpl_put_col_value(MPL *mpl, int j, double val) -{ if (mpl->phase != 3) - xfault("mpl_put_col_value: invalid call sequence\n"); - if (!(1 <= j && j <= mpl->n)) - xfault( - "mpl_put_col_value: j = %d; column number out of range\n", j); - mpl->col[j]->prim = val; - return; -} -#endif - -/*---------------------------------------------------------------------- --- mpl_postsolve - postsolve model. --- --- *Synopsis* --- --- #include "glpmpl.h" --- int mpl_postsolve(MPL *mpl); --- --- *Description* --- --- The routine mpl_postsolve performs postsolving of the model using --- its description stored in the translator database. This phase means --- executing statements, which follow the solve statement. --- --- If this routine is used, it should be called once after the routine --- mpl_generate and if the latter returned the code 3. --- --- *Returns* --- --- The routine mpl_postsolve returns one of the following codes: --- --- 3 - model has been successfully postsolved. --- 4 - processing failed due to some errors. In this case the calling --- program should call the routine mpl_terminate to terminate model --- processing. */ - -int mpl_postsolve(MPL *mpl) -{ if (!(mpl->phase == 3 && !mpl->flag_p)) - xfault("mpl_postsolve: invalid call sequence\n"); - /* set up error handler */ - if (setjmp(mpl->jump)) goto done; - /* perform postsolving */ - postsolve_model(mpl); - flush_output(mpl); - /* postsolving phase has been finished */ - xprintf("Model has been successfully processed\n"); -done: /* return to the calling program */ - return mpl->phase; -} - -/*---------------------------------------------------------------------- --- mpl_terminate - free all resources used by translator. --- --- *Synopsis* --- --- #include "glpmpl.h" --- void mpl_terminate(MPL *mpl); --- --- *Description* --- --- The routine mpl_terminate frees all the resources used by the GNU --- MathProg translator. */ - -void mpl_terminate(MPL *mpl) -{ if (setjmp(mpl->jump)) xassert(mpl != mpl); - switch (mpl->phase) - { case 0: - case 1: - case 2: - case 3: - /* there were no errors; clean the model content */ - clean_model(mpl); - xassert(mpl->a_list == NULL); -#if 1 /* 11/II-2008 */ - xassert(mpl->dca == NULL); -#endif - break; - case 4: - /* model processing has been finished due to error; delete - search trees, which may be created for some arrays */ - { ARRAY *a; - for (a = mpl->a_list; a != NULL; a = a->next) - if (a->tree != NULL) avl_delete_tree(a->tree); - } -#if 1 /* 11/II-2008 */ - free_dca(mpl); -#endif - break; - default: - xassert(mpl != mpl); - } - /* delete the translator database */ - xfree(mpl->image); - xfree(mpl->b_image); - xfree(mpl->f_image); - xfree(mpl->context); - dmp_delete_pool(mpl->pool); - avl_delete_tree(mpl->tree); - dmp_delete_pool(mpl->strings); - dmp_delete_pool(mpl->symbols); - dmp_delete_pool(mpl->tuples); - dmp_delete_pool(mpl->arrays); - dmp_delete_pool(mpl->members); - dmp_delete_pool(mpl->elemvars); - dmp_delete_pool(mpl->formulae); - dmp_delete_pool(mpl->elemcons); - xfree(mpl->sym_buf); - xfree(mpl->tup_buf); - rng_delete_rand(mpl->rand); - if (mpl->row != NULL) xfree(mpl->row); - if (mpl->col != NULL) xfree(mpl->col); - if (mpl->in_fp != NULL) glp_close(mpl->in_fp); - if (mpl->out_fp != NULL && mpl->out_fp != (void *)stdout) - glp_close(mpl->out_fp); - if (mpl->out_file != NULL) xfree(mpl->out_file); - if (mpl->prt_fp != NULL) glp_close(mpl->prt_fp); - if (mpl->prt_file != NULL) xfree(mpl->prt_file); - if (mpl->mod_file != NULL) xfree(mpl->mod_file); - xfree(mpl->mpl_buf); - xfree(mpl); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl5.c b/code/3rd_glpk/mpl/mpl5.c deleted file mode 100644 index c5374c9c..00000000 --- a/code/3rd_glpk/mpl/mpl5.c +++ /dev/null @@ -1,566 +0,0 @@ -/* mpl5.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Authors: Andrew Makhorin -* Heinrich Schuchardt -* -* Copyright (C) 2003-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#if 1 /* 11/VI-2013 */ -#include "jd.h" -#endif -#include "mpl.h" - -double fn_gmtime(MPL *mpl) -{ /* obtain the current calendar time (UTC) */ - time_t timer; - struct tm *tm; - int j; - time(&timer); - if (timer == (time_t)(-1)) -err: error(mpl, "gmtime(); unable to obtain current calendar time"); -#if 0 /* 29/I-2017 */ - tm = gmtime(&timer); -#else - tm = xgmtime(&timer); -#endif - if (tm == NULL) goto err; - j = jday(tm->tm_mday, tm->tm_mon + 1, 1900 + tm->tm_year); - if (j < 0) goto err; - return (((double)(j - jday(1, 1, 1970)) * 24.0 + - (double)tm->tm_hour) * 60.0 + (double)tm->tm_min) * 60.0 + - (double)tm->tm_sec; -} - -static char *week[] = { "Monday", "Tuesday", "Wednesday", "Thursday", - "Friday", "Saturday", "Sunday" }; - -static char *moon[] = { "January", "February", "March", "April", "May", - "June", "July", "August", "September", "October", "November", - "December" }; - -static void error1(MPL *mpl, const char *str, const char *s, - const char *fmt, const char *f, const char *msg) -{ xprintf("Input string passed to str2time:\n"); - xprintf("%s\n", str); - xprintf("%*s\n", (s - str) + 1, "^"); - xprintf("Format string passed to str2time:\n"); - xprintf("%s\n", fmt); - xprintf("%*s\n", (f - fmt) + 1, "^"); - error(mpl, "%s", msg); - /* no return */ -} - -double fn_str2time(MPL *mpl, const char *str, const char *fmt) -{ /* convert character string to the calendar time */ - int j, year, month, day, hh, mm, ss, zone; - const char *s, *f; - year = month = day = hh = mm = ss = -1, zone = INT_MAX; - s = str; - for (f = fmt; *f != '\0'; f++) - { if (*f == '%') - { f++; - if (*f == 'b' || *f == 'h') - { /* the abbreviated month name */ - int k; - char *name; - if (month >= 0) - error1(mpl, str, s, fmt, f, "month multiply specified" - ); - while (*s == ' ') s++; - for (month = 1; month <= 12; month++) - { name = moon[month-1]; - for (k = 0; k <= 2; k++) - { if (toupper((unsigned char)s[k]) != - toupper((unsigned char)name[k])) goto next; - } - s += 3; - for (k = 3; name[k] != '\0'; k++) - { if (toupper((unsigned char)*s) != - toupper((unsigned char)name[k])) break; - s++; - } - break; -next: ; - } - if (month > 12) - error1(mpl, str, s, fmt, f, "abbreviated month name m" - "issing or invalid"); - } - else if (*f == 'd') - { /* the day of the month as a decimal number (01..31) */ - if (day >= 0) - error1(mpl, str, s, fmt, f, "day multiply specified"); - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "day missing or invalid"); - day = (*s++) - '0'; - if ('0' <= *s && *s <= '9') - day = 10 * day + ((*s++) - '0'); - if (!(1 <= day && day <= 31)) - error1(mpl, str, s, fmt, f, "day out of range"); - } - else if (*f == 'H') - { /* the hour as a decimal number, using a 24-hour clock - (00..23) */ - if (hh >= 0) - error1(mpl, str, s, fmt, f, "hour multiply specified") - ; - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "hour missing or invalid") - ; - hh = (*s++) - '0'; - if ('0' <= *s && *s <= '9') - hh = 10 * hh + ((*s++) - '0'); - if (!(0 <= hh && hh <= 23)) - error1(mpl, str, s, fmt, f, "hour out of range"); - } - else if (*f == 'm') - { /* the month as a decimal number (01..12) */ - if (month >= 0) - error1(mpl, str, s, fmt, f, "month multiply specified" - ); - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "month missing or invalid" - ); - month = (*s++) - '0'; - if ('0' <= *s && *s <= '9') - month = 10 * month + ((*s++) - '0'); - if (!(1 <= month && month <= 12)) - error1(mpl, str, s, fmt, f, "month out of range"); - } - else if (*f == 'M') - { /* the minute as a decimal number (00..59) */ - if (mm >= 0) - error1(mpl, str, s, fmt, f, "minute multiply specifie" - "d"); - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "minute missing or invali" - "d"); - mm = (*s++) - '0'; - if ('0' <= *s && *s <= '9') - mm = 10 * mm + ((*s++) - '0'); - if (!(0 <= mm && mm <= 59)) - error1(mpl, str, s, fmt, f, "minute out of range"); - } - else if (*f == 'S') - { /* the second as a decimal number (00..60) */ - if (ss >= 0) - error1(mpl, str, s, fmt, f, "second multiply specifie" - "d"); - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "second missing or invali" - "d"); - ss = (*s++) - '0'; - if ('0' <= *s && *s <= '9') - ss = 10 * ss + ((*s++) - '0'); - if (!(0 <= ss && ss <= 60)) - error1(mpl, str, s, fmt, f, "second out of range"); - } - else if (*f == 'y') - { /* the year without a century as a decimal number - (00..99); the values 00 to 68 mean the years 2000 to - 2068 while the values 69 to 99 mean the years 1969 to - 1999 */ - if (year >= 0) - error1(mpl, str, s, fmt, f, "year multiply specified") - ; - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "year missing or invalid") - ; - year = (*s++) - '0'; - if ('0' <= *s && *s <= '9') - year = 10 * year + ((*s++) - '0'); - year += (year >= 69 ? 1900 : 2000); - } - else if (*f == 'Y') - { /* the year as a decimal number, using the Gregorian - calendar */ - if (year >= 0) - error1(mpl, str, s, fmt, f, "year multiply specified") - ; - while (*s == ' ') s++; - if (!('0' <= *s && *s <= '9')) - error1(mpl, str, s, fmt, f, "year missing or invalid") - ; - year = 0; - for (j = 1; j <= 4; j++) - { if (!('0' <= *s && *s <= '9')) break; - year = 10 * year + ((*s++) - '0'); - } - if (!(1 <= year && year <= 4000)) - error1(mpl, str, s, fmt, f, "year out of range"); - } - else if (*f == 'z') - { /* time zone offset in the form zhhmm */ - int z, hh, mm; - if (zone != INT_MAX) - error1(mpl, str, s, fmt, f, "time zone offset multipl" - "y specified"); - while (*s == ' ') s++; - if (*s == 'Z') - { z = hh = mm = 0, s++; - goto skip; - } - if (*s == '+') - z = +1, s++; - else if (*s == '-') - z = -1, s++; - else - error1(mpl, str, s, fmt, f, "time zone offset sign mi" - "ssing"); - hh = 0; - for (j = 1; j <= 2; j++) - { if (!('0' <= *s && *s <= '9')) -err1: error1(mpl, str, s, fmt, f, "time zone offset valu" - "e incomplete or invalid"); - hh = 10 * hh + ((*s++) - '0'); - } - if (hh > 23) -err2: error1(mpl, str, s, fmt, f, "time zone offset value o" - "ut of range"); - if (*s == ':') - { s++; - if (!('0' <= *s && *s <= '9')) goto err1; - } - mm = 0; - if (!('0' <= *s && *s <= '9')) goto skip; - for (j = 1; j <= 2; j++) - { if (!('0' <= *s && *s <= '9')) goto err1; - mm = 10 * mm + ((*s++) - '0'); - } - if (mm > 59) goto err2; -skip: zone = z * (60 * hh + mm); - } - else if (*f == '%') - { /* literal % character */ - goto test; - } - else - error1(mpl, str, s, fmt, f, "invalid conversion specifie" - "r"); - } - else if (*f == ' ') - ; - else -test: { /* check a matching character in the input string */ - if (*s != *f) - error1(mpl, str, s, fmt, f, "character mismatch"); - s++; - } - } - if (year < 0) year = 1970; - if (month < 0) month = 1; - if (day < 0) day = 1; - if (hh < 0) hh = 0; - if (mm < 0) mm = 0; - if (ss < 0) ss = 0; - if (zone == INT_MAX) zone = 0; - j = jday(day, month, year); - xassert(j >= 0); - return (((double)(j - jday(1, 1, 1970)) * 24.0 + (double)hh) * - 60.0 + (double)mm) * 60.0 + (double)ss - 60.0 * (double)zone; -} - -static void error2(MPL *mpl, const char *fmt, const char *f, - const char *msg) -{ xprintf("Format string passed to time2str:\n"); - xprintf("%s\n", fmt); - xprintf("%*s\n", (f - fmt) + 1, "^"); - error(mpl, "%s", msg); - /* no return */ -} - -static int weekday(int j) -{ /* determine weekday number (1 = Mon, ..., 7 = Sun) */ - return (j + jday(1, 1, 1970)) % 7 + 1; -} - -static int firstday(int year) -{ /* determine the first day of the first week for a specified year - according to ISO 8601 */ - int j; - /* if 1 January is Monday, Tuesday, Wednesday or Thursday, it is - in week 01; if 1 January is Friday, Saturday or Sunday, it is - in week 52 or 53 of the previous year */ - j = jday(1, 1, year) - jday(1, 1, 1970); - switch (weekday(j)) - { case 1: /* 1 Jan is Mon */ j += 0; break; - case 2: /* 1 Jan is Tue */ j -= 1; break; - case 3: /* 1 Jan is Wed */ j -= 2; break; - case 4: /* 1 Jan is Thu */ j -= 3; break; - case 5: /* 1 Jan is Fri */ j += 3; break; - case 6: /* 1 Jan is Sat */ j += 2; break; - case 7: /* 1 Jan is Sun */ j += 1; break; - default: xassert(j != j); - } - /* the first day of the week must be Monday */ - xassert(weekday(j) == 1); - return j; -} - -void fn_time2str(MPL *mpl, char *str, double t, const char *fmt) -{ /* convert the calendar time to character string */ - int j, year, month, day, hh, mm, ss, len; - double temp; - const char *f; - char buf[MAX_LENGTH+1]; - if (!(-62135596800.0 <= t && t <= 64092211199.0)) - error(mpl, "time2str(%.*g,...); argument out of range", - DBL_DIG, t); - t = floor(t + 0.5); - temp = fabs(t) / 86400.0; - j = (int)floor(temp); - if (t < 0.0) - { if (temp == floor(temp)) - j = - j; - else - j = - (j + 1); - } - xassert(jdate(j + jday(1, 1, 1970), &day, &month, &year) == 0); - ss = (int)(t - 86400.0 * (double)j); - xassert(0 <= ss && ss < 86400); - mm = ss / 60, ss %= 60; - hh = mm / 60, mm %= 60; - len = 0; - for (f = fmt; *f != '\0'; f++) - { if (*f == '%') - { f++; - if (*f == 'a') - { /* the abbreviated weekday name */ - memcpy(buf, week[weekday(j)-1], 3), buf[3] = '\0'; - } - else if (*f == 'A') - { /* the full weekday name */ - strcpy(buf, week[weekday(j)-1]); - } - else if (*f == 'b' || *f == 'h') - { /* the abbreviated month name */ - memcpy(buf, moon[month-1], 3), buf[3] = '\0'; - } - else if (*f == 'B') - { /* the full month name */ - strcpy(buf, moon[month-1]); - } - else if (*f == 'C') - { /* the century of the year */ - sprintf(buf, "%02d", year / 100); - } - else if (*f == 'd') - { /* the day of the month as a decimal number (01..31) */ - sprintf(buf, "%02d", day); - } - else if (*f == 'D') - { /* the date using the format %m/%d/%y */ - sprintf(buf, "%02d/%02d/%02d", month, day, year % 100); - } - else if (*f == 'e') - { /* the day of the month like with %d, but padded with - blank (1..31) */ - sprintf(buf, "%2d", day); - } - else if (*f == 'F') - { /* the date using the format %Y-%m-%d */ - sprintf(buf, "%04d-%02d-%02d", year, month, day); - } - else if (*f == 'g') - { /* the year corresponding to the ISO week number, but - without the century (range 00 through 99); this has - the same format and value as %y, except that if the - ISO week number (see %V) belongs to the previous or - next year, that year is used instead */ - int iso; - if (j < firstday(year)) - iso = year - 1; - else if (j < firstday(year + 1)) - iso = year; - else - iso = year + 1; - sprintf(buf, "%02d", iso % 100); - } - else if (*f == 'G') - { /* the year corresponding to the ISO week number; this - has the same format and value as %Y, excepth that if - the ISO week number (see %V) belongs to the previous - or next year, that year is used instead */ - int iso; - if (j < firstday(year)) - iso = year - 1; - else if (j < firstday(year + 1)) - iso = year; - else - iso = year + 1; - sprintf(buf, "%04d", iso); - } - else if (*f == 'H') - { /* the hour as a decimal number, using a 24-hour clock - (00..23) */ - sprintf(buf, "%02d", hh); - } - else if (*f == 'I') - { /* the hour as a decimal number, using a 12-hour clock - (01..12) */ - sprintf(buf, "%02d", - hh == 0 ? 12 : hh <= 12 ? hh : hh - 12); - } - else if (*f == 'j') - { /* the day of the year as a decimal number (001..366) */ - sprintf(buf, "%03d", - jday(day, month, year) - jday(1, 1, year) + 1); - } - else if (*f == 'k') - { /* the hour as a decimal number, using a 24-hour clock - like %H, but padded with blank (0..23) */ - sprintf(buf, "%2d", hh); - } - else if (*f == 'l') - { /* the hour as a decimal number, using a 12-hour clock - like %I, but padded with blank (1..12) */ - sprintf(buf, "%2d", - hh == 0 ? 12 : hh <= 12 ? hh : hh - 12); - } - else if (*f == 'm') - { /* the month as a decimal number (01..12) */ - sprintf(buf, "%02d", month); - } - else if (*f == 'M') - { /* the minute as a decimal number (00..59) */ - sprintf(buf, "%02d", mm); - } - else if (*f == 'p') - { /* either AM or PM, according to the given time value; - noon is treated as PM and midnight as AM */ - strcpy(buf, hh <= 11 ? "AM" : "PM"); - } - else if (*f == 'P') - { /* either am or pm, according to the given time value; - noon is treated as pm and midnight as am */ - strcpy(buf, hh <= 11 ? "am" : "pm"); - } - else if (*f == 'r') - { /* the calendar time using the format %I:%M:%S %p */ - sprintf(buf, "%02d:%02d:%02d %s", - hh == 0 ? 12 : hh <= 12 ? hh : hh - 12, - mm, ss, hh <= 11 ? "AM" : "PM"); - } - else if (*f == 'R') - { /* the hour and minute using the format %H:%M */ - sprintf(buf, "%02d:%02d", hh, mm); - } - else if (*f == 'S') - { /* the second as a decimal number (00..59) */ - sprintf(buf, "%02d", ss); - } - else if (*f == 'T') - { /* the time of day using the format %H:%M:%S */ - sprintf(buf, "%02d:%02d:%02d", hh, mm, ss); - } - else if (*f == 'u') - { /* the day of the week as a decimal number (1..7), - Monday being 1 */ - sprintf(buf, "%d", weekday(j)); - } - else if (*f == 'U') - { /* the week number of the current year as a decimal - number (range 00 through 53), starting with the first - Sunday as the first day of the first week; days - preceding the first Sunday in the year are considered - to be in week 00 */ -#if 1 /* 09/I-2009 */ -#undef sun -/* causes compilation error in SunOS */ -#endif - int sun; - /* sun = the first Sunday of the year */ - sun = jday(1, 1, year) - jday(1, 1, 1970); - sun += (7 - weekday(sun)); - sprintf(buf, "%02d", (j + 7 - sun) / 7); - } - else if (*f == 'V') - { /* the ISO week number as a decimal number (range 01 - through 53); ISO weeks start with Monday and end with - Sunday; week 01 of a year is the first week which has - the majority of its days in that year; week 01 of - a year can contain days from the previous year; the - week before week 01 of a year is the last week (52 or - 53) of the previous year even if it contains days - from the new year */ - int iso; - if (j < firstday(year)) - iso = j - firstday(year - 1); - else if (j < firstday(year + 1)) - iso = j - firstday(year); - else - iso = j - firstday(year + 1); - sprintf(buf, "%02d", iso / 7 + 1); - } - else if (*f == 'w') - { /* the day of the week as a decimal number (0..6), - Sunday being 0 */ - sprintf(buf, "%d", weekday(j) % 7); - } - else if (*f == 'W') - { /* the week number of the current year as a decimal - number (range 00 through 53), starting with the first - Monday as the first day of the first week; days - preceding the first Monday in the year are considered - to be in week 00 */ - int mon; - /* mon = the first Monday of the year */ - mon = jday(1, 1, year) - jday(1, 1, 1970); - mon += (8 - weekday(mon)) % 7; - sprintf(buf, "%02d", (j + 7 - mon) / 7); - } - else if (*f == 'y') - { /* the year without a century as a decimal number - (00..99) */ - sprintf(buf, "%02d", year % 100); - } - else if (*f == 'Y') - { /* the year as a decimal number, using the Gregorian - calendar */ - sprintf(buf, "%04d", year); - } - else if (*f == '%') - { /* a literal % character */ - buf[0] = '%', buf[1] = '\0'; - } - else - error2(mpl, fmt, f, "invalid conversion specifier"); - } - else - buf[0] = *f, buf[1] = '\0'; - if (len + strlen(buf) > MAX_LENGTH) - error(mpl, "time2str; output string length exceeds %d chara" - "cters", MAX_LENGTH); - memcpy(str+len, buf, strlen(buf)); - len += strlen(buf); - } - str[len] = '\0'; - return; -} - -/* eof */ diff --git a/code/3rd_glpk/mpl/mpl6.c b/code/3rd_glpk/mpl/mpl6.c deleted file mode 100644 index ac2a0393..00000000 --- a/code/3rd_glpk/mpl/mpl6.c +++ /dev/null @@ -1,1039 +0,0 @@ -/* mpl6.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2003-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "mpl.h" -#include "mplsql.h" - -/**********************************************************************/ - -#define CSV_FIELD_MAX 50 -/* maximal number of fields in record */ - -#define CSV_FDLEN_MAX 100 -/* maximal field length */ - -struct csv -{ /* comma-separated values file */ - int mode; - /* 'R' = reading; 'W' = writing */ - char *fname; - /* name of csv file */ - FILE *fp; - /* stream assigned to csv file */ - jmp_buf jump; - /* address for non-local go to in case of error */ - int count; - /* record count */ - /*--------------------------------------------------------------*/ - /* used only for input csv file */ - int c; - /* current character or EOF */ - int what; - /* current marker: */ -#define CSV_EOF 0 /* end-of-file */ -#define CSV_EOR 1 /* end-of-record */ -#define CSV_NUM 2 /* floating-point number */ -#define CSV_STR 3 /* character string */ - char field[CSV_FDLEN_MAX+1]; - /* current field just read */ - int nf; - /* number of fields in the csv file */ - int ref[1+CSV_FIELD_MAX]; - /* ref[k] = k', if k-th field of the csv file corresponds to - k'-th field in the table statement; if ref[k] = 0, k-th field - of the csv file is ignored */ -#if 1 /* 01/VI-2010 */ - int nskip; - /* number of comment records preceding the header record */ -#endif -}; - -#undef read_char - -static void read_char(struct csv *csv) -{ /* read character from csv data file */ - int c; - xassert(csv->c != EOF); - if (csv->c == '\n') csv->count++; -loop: c = fgetc(csv->fp); - if (ferror(csv->fp)) - { xprintf("%s:%d: read error - %s\n", csv->fname, csv->count, -#if 0 /* 29/I-2017 */ - strerror(errno)); -#else - xstrerr(errno)); -#endif - longjmp(csv->jump, 0); - } - if (feof(csv->fp)) - { if (csv->c == '\n') - { csv->count--; - c = EOF; - } - else - { xprintf("%s:%d: warning: missing final end-of-line\n", - csv->fname, csv->count); - c = '\n'; - } - } - else if (c == '\r') - goto loop; - else if (c == '\n') - ; - else if (iscntrl(c)) - { xprintf("%s:%d: invalid control character 0x%02X\n", - csv->fname, csv->count, c); - longjmp(csv->jump, 0); - } - csv->c = c; - return; -} - -static void read_field(struct csv *csv) -{ /* read field from csv data file */ - /* check for end of file */ - if (csv->c == EOF) - { csv->what = CSV_EOF; - strcpy(csv->field, "EOF"); - goto done; - } - /* check for end of record */ - if (csv->c == '\n') - { csv->what = CSV_EOR; - strcpy(csv->field, "EOR"); - read_char(csv); - if (csv->c == ',') -err1: { xprintf("%s:%d: empty field not allowed\n", csv->fname, - csv->count); - longjmp(csv->jump, 0); - } - if (csv->c == '\n') - { xprintf("%s:%d: empty record not allowed\n", csv->fname, - csv->count); - longjmp(csv->jump, 0); - } -#if 1 /* 01/VI-2010 */ - /* skip comment records; may appear only before the very first - record containing field names */ - if (csv->c == '#' && csv->count == 1) - { while (csv->c == '#') - { while (csv->c != '\n') - read_char(csv); - read_char(csv); - csv->nskip++; - } - } -#endif - goto done; - } - /* skip comma before next field */ - if (csv->c == ',') - read_char(csv); - /* read field */ - if (csv->c == '\'' || csv->c == '"') - { /* read a field enclosed in quotes */ - int quote = csv->c, len = 0; - csv->what = CSV_STR; - /* skip opening quote */ - read_char(csv); - /* read field characters within quotes */ - for (;;) - { /* check for closing quote and read it */ - if (csv->c == quote) - { read_char(csv); - if (csv->c == quote) - ; - else if (csv->c == ',' || csv->c == '\n') - break; - else - { xprintf("%s:%d: invalid field\n", csv->fname, - csv->count); - longjmp(csv->jump, 0); - } - } - /* check the current field length */ - if (len == CSV_FDLEN_MAX) -err2: { xprintf("%s:%d: field too long\n", csv->fname, - csv->count); - longjmp(csv->jump, 0); - } - /* add the current character to the field */ - csv->field[len++] = (char)csv->c; - /* read the next character */ - read_char(csv); - } - /* the field has been read */ - if (len == 0) goto err1; - csv->field[len] = '\0'; - } - else - { /* read a field not enclosed in quotes */ - int len = 0; - double temp; - csv->what = CSV_NUM; - while (!(csv->c == ',' || csv->c == '\n')) - { /* quotes within the field are not allowed */ - if (csv->c == '\'' || csv->c == '"') - { xprintf("%s:%d: invalid use of single or double quote wi" - "thin field\n", csv->fname, csv->count); - longjmp(csv->jump, 0); - } - /* check the current field length */ - if (len == CSV_FDLEN_MAX) goto err2; - /* add the current character to the field */ - csv->field[len++] = (char)csv->c; - /* read the next character */ - read_char(csv); - } - /* the field has been read */ - if (len == 0) goto err1; - csv->field[len] = '\0'; - /* check the field type */ - if (str2num(csv->field, &temp)) csv->what = CSV_STR; - } -done: return; -} - -static struct csv *csv_open_file(TABDCA *dca, int mode) -{ /* open csv data file */ - struct csv *csv; - /* create control structure */ - csv = xmalloc(sizeof(struct csv)); - csv->mode = mode; - csv->fname = NULL; - csv->fp = NULL; - if (setjmp(csv->jump)) goto fail; - csv->count = 0; - csv->c = '\n'; - csv->what = 0; - csv->field[0] = '\0'; - csv->nf = 0; - /* try to open the csv data file */ - if (mpl_tab_num_args(dca) < 2) - { xprintf("csv_driver: file name not specified\n"); - longjmp(csv->jump, 0); - } - csv->fname = xmalloc(strlen(mpl_tab_get_arg(dca, 2))+1); - strcpy(csv->fname, mpl_tab_get_arg(dca, 2)); - if (mode == 'R') - { /* open the file for reading */ - int k; - csv->fp = fopen(csv->fname, "r"); - if (csv->fp == NULL) - { xprintf("csv_driver: unable to open %s - %s\n", -#if 0 /* 29/I-2017 */ - csv->fname, strerror(errno)); -#else - csv->fname, xstrerr(errno)); -#endif - longjmp(csv->jump, 0); - } -#if 1 /* 01/VI-2010 */ - csv->nskip = 0; -#endif - /* skip fake new-line */ - read_field(csv); - xassert(csv->what == CSV_EOR); - /* read field names */ - xassert(csv->nf == 0); - for (;;) - { read_field(csv); - if (csv->what == CSV_EOR) - break; - if (csv->what != CSV_STR) - { xprintf("%s:%d: invalid field name\n", csv->fname, - csv->count); - longjmp(csv->jump, 0); - } - if (csv->nf == CSV_FIELD_MAX) - { xprintf("%s:%d: too many fields\n", csv->fname, - csv->count); - longjmp(csv->jump, 0); - } - csv->nf++; - /* find corresponding field in the table statement */ - for (k = mpl_tab_num_flds(dca); k >= 1; k--) - { if (strcmp(mpl_tab_get_name(dca, k), csv->field) == 0) - break; - } - csv->ref[csv->nf] = k; - } - /* find dummy RECNO field in the table statement */ - for (k = mpl_tab_num_flds(dca); k >= 1; k--) - if (strcmp(mpl_tab_get_name(dca, k), "RECNO") == 0) break; - csv->ref[0] = k; - } - else if (mode == 'W') - { /* open the file for writing */ - int k, nf; - csv->fp = fopen(csv->fname, "w"); - if (csv->fp == NULL) - { xprintf("csv_driver: unable to create %s - %s\n", -#if 0 /* 29/I-2017 */ - csv->fname, strerror(errno)); -#else - csv->fname, xstrerr(errno)); -#endif - longjmp(csv->jump, 0); - } - /* write field names */ - nf = mpl_tab_num_flds(dca); - for (k = 1; k <= nf; k++) - fprintf(csv->fp, "%s%c", mpl_tab_get_name(dca, k), - k < nf ? ',' : '\n'); - csv->count++; - } - else - xassert(mode != mode); - /* the file has been open */ - return csv; -fail: /* the file cannot be open */ - if (csv->fname != NULL) xfree(csv->fname); - if (csv->fp != NULL) fclose(csv->fp); - xfree(csv); - return NULL; -} - -static int csv_read_record(TABDCA *dca, struct csv *csv) -{ /* read next record from csv data file */ - int k, ret = 0; - xassert(csv->mode == 'R'); - if (setjmp(csv->jump)) - { ret = 1; - goto done; - } - /* read dummy RECNO field */ - if (csv->ref[0] > 0) -#if 0 /* 01/VI-2010 */ - mpl_tab_set_num(dca, csv->ref[0], csv->count-1); -#else - mpl_tab_set_num(dca, csv->ref[0], csv->count-csv->nskip-1); -#endif - /* read fields */ - for (k = 1; k <= csv->nf; k++) - { read_field(csv); - if (csv->what == CSV_EOF) - { /* end-of-file reached */ - xassert(k == 1); - ret = -1; - goto done; - } - else if (csv->what == CSV_EOR) - { /* end-of-record reached */ - int lack = csv->nf - k + 1; - if (lack == 1) - xprintf("%s:%d: one field missing\n", csv->fname, - csv->count); - else - xprintf("%s:%d: %d fields missing\n", csv->fname, - csv->count, lack); - longjmp(csv->jump, 0); - } - else if (csv->what == CSV_NUM) - { /* floating-point number */ - if (csv->ref[k] > 0) - { double num; - xassert(str2num(csv->field, &num) == 0); - mpl_tab_set_num(dca, csv->ref[k], num); - } - } - else if (csv->what == CSV_STR) - { /* character string */ - if (csv->ref[k] > 0) - mpl_tab_set_str(dca, csv->ref[k], csv->field); - } - else - xassert(csv != csv); - } - /* now there must be NL */ - read_field(csv); - xassert(csv->what != CSV_EOF); - if (csv->what != CSV_EOR) - { xprintf("%s:%d: too many fields\n", csv->fname, csv->count); - longjmp(csv->jump, 0); - } -done: return ret; -} - -static int csv_write_record(TABDCA *dca, struct csv *csv) -{ /* write next record to csv data file */ - int k, nf, ret = 0; - const char *c; - xassert(csv->mode == 'W'); - nf = mpl_tab_num_flds(dca); - for (k = 1; k <= nf; k++) - { switch (mpl_tab_get_type(dca, k)) - { case 'N': - fprintf(csv->fp, "%.*g", DBL_DIG, - mpl_tab_get_num(dca, k)); - break; - case 'S': - fputc('"', csv->fp); - for (c = mpl_tab_get_str(dca, k); *c != '\0'; c++) - { if (*c == '"') - fputc('"', csv->fp), fputc('"', csv->fp); - else - fputc(*c, csv->fp); - } - fputc('"', csv->fp); - break; - default: - xassert(dca != dca); - } - fputc(k < nf ? ',' : '\n', csv->fp); - } - csv->count++; - if (ferror(csv->fp)) - { xprintf("%s:%d: write error - %s\n", csv->fname, csv->count, -#if 0 /* 29/I-2017 */ - strerror(errno)); -#else - xstrerr(errno)); -#endif - ret = 1; - } - return ret; -} - -static int csv_close_file(TABDCA *dca, struct csv *csv) -{ /* close csv data file */ - int ret = 0; - xassert(dca == dca); - if (csv->mode == 'W') - { fflush(csv->fp); - if (ferror(csv->fp)) - { xprintf("%s:%d: write error - %s\n", csv->fname, -#if 0 /* 29/I-2017 */ - csv->count, strerror(errno)); -#else - csv->count, xstrerr(errno)); -#endif - ret = 1; - } - } - xfree(csv->fname); - fclose(csv->fp); - xfree(csv); - return ret; -} - -/**********************************************************************/ - -#define DBF_FIELD_MAX 50 -/* maximal number of fields in record */ - -#define DBF_FDLEN_MAX 100 -/* maximal field length */ - -struct dbf -{ /* xBASE data file */ - int mode; - /* 'R' = reading; 'W' = writing */ - char *fname; - /* name of xBASE file */ - FILE *fp; - /* stream assigned to xBASE file */ - jmp_buf jump; - /* address for non-local go to in case of error */ - int offset; - /* offset of a byte to be read next */ - int count; - /* record count */ - int nf; - /* number of fields */ - int ref[1+DBF_FIELD_MAX]; - /* ref[k] = k', if k-th field of the csv file corresponds to - k'-th field in the table statement; if ref[k] = 0, k-th field - of the csv file is ignored */ - int type[1+DBF_FIELD_MAX]; - /* type[k] is type of k-th field */ - int len[1+DBF_FIELD_MAX]; - /* len[k] is length of k-th field */ - int prec[1+DBF_FIELD_MAX]; - /* prec[k] is precision of k-th field */ -}; - -static int read_byte(struct dbf *dbf) -{ /* read byte from xBASE data file */ - int b; - b = fgetc(dbf->fp); - if (ferror(dbf->fp)) - { xprintf("%s:0x%X: read error - %s\n", dbf->fname, -#if 0 /* 29/I-2017 */ - dbf->offset, strerror(errno)); -#else - dbf->offset, xstrerr(errno)); -#endif - longjmp(dbf->jump, 0); - } - if (feof(dbf->fp)) - { xprintf("%s:0x%X: unexpected end of file\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - xassert(0x00 <= b && b <= 0xFF); - dbf->offset++; - return b; -} - -static void read_header(TABDCA *dca, struct dbf *dbf) -{ /* read xBASE data file header */ - int b, j, k, recl; - char name[10+1]; - /* (ignored) */ - for (j = 1; j <= 10; j++) - read_byte(dbf); - /* length of each record, in bytes */ - recl = read_byte(dbf); - recl += read_byte(dbf) << 8; - /* (ignored) */ - for (j = 1; j <= 20; j++) - read_byte(dbf); - /* field descriptor array */ - xassert(dbf->nf == 0); - for (;;) - { /* check for end of array */ - b = read_byte(dbf); - if (b == 0x0D) break; - if (dbf->nf == DBF_FIELD_MAX) - { xprintf("%s:0x%X: too many fields\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - dbf->nf++; - /* field name */ - name[0] = (char)b; - for (j = 1; j < 10; j++) - { b = read_byte(dbf); - name[j] = (char)b; - } - name[10] = '\0'; - b = read_byte(dbf); - if (b != 0x00) - { xprintf("%s:0x%X: invalid field name\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - /* find corresponding field in the table statement */ - for (k = mpl_tab_num_flds(dca); k >= 1; k--) - if (strcmp(mpl_tab_get_name(dca, k), name) == 0) break; - dbf->ref[dbf->nf] = k; - /* field type */ - b = read_byte(dbf); - if (!(b == 'C' || b == 'N')) - { xprintf("%s:0x%X: invalid field type\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - dbf->type[dbf->nf] = b; - /* (ignored) */ - for (j = 1; j <= 4; j++) - read_byte(dbf); - /* field length */ - b = read_byte(dbf); - if (b == 0) - { xprintf("%s:0x%X: invalid field length\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - if (b > DBF_FDLEN_MAX) - { xprintf("%s:0x%X: field too long\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - dbf->len[dbf->nf] = b; - recl -= b; - /* (ignored) */ - for (j = 1; j <= 15; j++) - read_byte(dbf); - } - if (recl != 1) - { xprintf("%s:0x%X: invalid file header\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - /* find dummy RECNO field in the table statement */ - for (k = mpl_tab_num_flds(dca); k >= 1; k--) - if (strcmp(mpl_tab_get_name(dca, k), "RECNO") == 0) break; - dbf->ref[0] = k; - return; -} - -static void parse_third_arg(TABDCA *dca, struct dbf *dbf) -{ /* parse xBASE file format (third argument) */ - int j, k, temp; - const char *arg; - dbf->nf = mpl_tab_num_flds(dca); - arg = mpl_tab_get_arg(dca, 3), j = 0; - for (k = 1; k <= dbf->nf; k++) - { /* parse specification of k-th field */ - if (arg[j] == '\0') - { xprintf("xBASE driver: field %s: specification missing\n", - mpl_tab_get_name(dca, k)); - longjmp(dbf->jump, 0); - } - /* parse field type */ - if (arg[j] == 'C' || arg[j] == 'N') - dbf->type[k] = arg[j], j++; - else - { xprintf("xBASE driver: field %s: invalid field type\n", - mpl_tab_get_name(dca, k)); - longjmp(dbf->jump, 0); - } - /* check for left parenthesis */ - if (arg[j] == '(') - j++; - else -err: { xprintf("xBASE driver: field %s: invalid field format\n", - mpl_tab_get_name(dca, k)); - longjmp(dbf->jump, 0); - } - /* parse field length */ - temp = 0; - while (isdigit(arg[j])) - { if (temp > DBF_FDLEN_MAX) break; - temp = 10 * temp + (arg[j] - '0'), j++; - } - if (!(1 <= temp && temp <= DBF_FDLEN_MAX)) - { xprintf("xBASE driver: field %s: invalid field length\n", - mpl_tab_get_name(dca, k)); - longjmp(dbf->jump, 0); - } - dbf->len[k] = temp; - /* parse optional field precision */ - if (dbf->type[k] == 'N' && arg[j] == ',') - { j++; - temp = 0; - while (isdigit(arg[j])) - { if (temp > dbf->len[k]) break; - temp = 10 * temp + (arg[j] - '0'), j++; - } - if (temp > dbf->len[k]) - { xprintf("xBASE driver: field %s: invalid field precision" - "\n", mpl_tab_get_name(dca, k)); - longjmp(dbf->jump, 0); - } - dbf->prec[k] = temp; - } - else - dbf->prec[k] = 0; - /* check for right parenthesis */ - if (arg[j] == ')') - j++; - else - goto err; - } - /* ignore other specifications */ - return; -} - -static void write_byte(struct dbf *dbf, int b) -{ /* write byte to xBASE data file */ - fputc(b, dbf->fp); - dbf->offset++; - return; -} - -static void write_header(TABDCA *dca, struct dbf *dbf) -{ /* write xBASE data file header */ - int j, k, temp; - const char *name; - /* version number */ - write_byte(dbf, 0x03 /* file without DBT */); - /* date of last update (YYMMDD) */ - write_byte(dbf, 70 /* 1970 */); - write_byte(dbf, 1 /* January */); - write_byte(dbf, 1 /* 1st */); - /* number of records (unknown so far) */ - for (j = 1; j <= 4; j++) - write_byte(dbf, 0xFF); - /* length of the header, in bytes */ - temp = 32 + dbf->nf * 32 + 1; - write_byte(dbf, temp); - write_byte(dbf, temp >> 8); - /* length of each record, in bytes */ - temp = 1; - for (k = 1; k <= dbf->nf; k++) - temp += dbf->len[k]; - write_byte(dbf, temp); - write_byte(dbf, temp >> 8); - /* (reserved) */ - for (j = 1; j <= 20; j++) - write_byte(dbf, 0x00); - /* field descriptor array */ - for (k = 1; k <= dbf->nf; k++) - { /* field name (terminated by 0x00) */ - name = mpl_tab_get_name(dca, k); - for (j = 0; j < 10 && name[j] != '\0'; j++) - write_byte(dbf, name[j]); - for (j = j; j < 11; j++) - write_byte(dbf, 0x00); - /* field type */ - write_byte(dbf, dbf->type[k]); - /* (reserved) */ - for (j = 1; j <= 4; j++) - write_byte(dbf, 0x00); - /* field length */ - write_byte(dbf, dbf->len[k]); - /* field precision */ - write_byte(dbf, dbf->prec[k]); - /* (reserved) */ - for (j = 1; j <= 14; j++) - write_byte(dbf, 0x00); - } - /* end of header */ - write_byte(dbf, 0x0D); - return; -} - -static struct dbf *dbf_open_file(TABDCA *dca, int mode) -{ /* open xBASE data file */ - struct dbf *dbf; - /* create control structure */ - dbf = xmalloc(sizeof(struct dbf)); - dbf->mode = mode; - dbf->fname = NULL; - dbf->fp = NULL; - if (setjmp(dbf->jump)) goto fail; - dbf->offset = 0; - dbf->count = 0; - dbf->nf = 0; - /* try to open the xBASE data file */ - if (mpl_tab_num_args(dca) < 2) - { xprintf("xBASE driver: file name not specified\n"); - longjmp(dbf->jump, 0); - } - dbf->fname = xmalloc(strlen(mpl_tab_get_arg(dca, 2))+1); - strcpy(dbf->fname, mpl_tab_get_arg(dca, 2)); - if (mode == 'R') - { /* open the file for reading */ - dbf->fp = fopen(dbf->fname, "rb"); - if (dbf->fp == NULL) - { xprintf("xBASE driver: unable to open %s - %s\n", -#if 0 /* 29/I-2017 */ - dbf->fname, strerror(errno)); -#else - dbf->fname, xstrerr(errno)); -#endif - longjmp(dbf->jump, 0); - } - read_header(dca, dbf); - } - else if (mode == 'W') - { /* open the file for writing */ - if (mpl_tab_num_args(dca) < 3) - { xprintf("xBASE driver: file format not specified\n"); - longjmp(dbf->jump, 0); - } - parse_third_arg(dca, dbf); - dbf->fp = fopen(dbf->fname, "wb"); - if (dbf->fp == NULL) - { xprintf("xBASE driver: unable to create %s - %s\n", -#if 0 /* 29/I-2017 */ - dbf->fname, strerror(errno)); -#else - dbf->fname, xstrerr(errno)); -#endif - longjmp(dbf->jump, 0); - } - write_header(dca, dbf); - } - else - xassert(mode != mode); - /* the file has been open */ - return dbf; -fail: /* the file cannot be open */ - if (dbf->fname != NULL) xfree(dbf->fname); - if (dbf->fp != NULL) fclose(dbf->fp); - xfree(dbf); - return NULL; -} - -static int dbf_read_record(TABDCA *dca, struct dbf *dbf) -{ /* read next record from xBASE data file */ - int b, j, k, ret = 0; - char buf[DBF_FDLEN_MAX+1]; - xassert(dbf->mode == 'R'); - if (setjmp(dbf->jump)) - { ret = 1; - goto done; - } - /* check record flag */ - b = read_byte(dbf); - if (b == 0x1A) - { /* end of data */ - ret = -1; - goto done; - } - if (b != 0x20) - { xprintf("%s:0x%X: invalid record flag\n", dbf->fname, - dbf->offset); - longjmp(dbf->jump, 0); - } - /* read dummy RECNO field */ - if (dbf->ref[0] > 0) - mpl_tab_set_num(dca, dbf->ref[0], dbf->count+1); - /* read fields */ - for (k = 1; k <= dbf->nf; k++) - { /* read k-th field */ - for (j = 0; j < dbf->len[k]; j++) - buf[j] = (char)read_byte(dbf); - buf[dbf->len[k]] = '\0'; - /* set field value */ - if (dbf->type[k] == 'C') - { /* character field */ - if (dbf->ref[k] > 0) - mpl_tab_set_str(dca, dbf->ref[k], strtrim(buf)); - } - else if (dbf->type[k] == 'N') - { /* numeric field */ - if (dbf->ref[k] > 0) - { double num; - strspx(buf); - xassert(str2num(buf, &num) == 0); - mpl_tab_set_num(dca, dbf->ref[k], num); - } - } - else - xassert(dbf != dbf); - } - /* increase record count */ - dbf->count++; -done: return ret; -} - -static int dbf_write_record(TABDCA *dca, struct dbf *dbf) -{ /* write next record to xBASE data file */ - int j, k, ret = 0; - char buf[255+1]; - xassert(dbf->mode == 'W'); - if (setjmp(dbf->jump)) - { ret = 1; - goto done; - } - /* record flag */ - write_byte(dbf, 0x20); - xassert(dbf->nf == mpl_tab_num_flds(dca)); - for (k = 1; k <= dbf->nf; k++) - { if (dbf->type[k] == 'C') - { /* character field */ - const char *str; - if (mpl_tab_get_type(dca, k) == 'N') - { sprintf(buf, "%.*g", DBL_DIG, mpl_tab_get_num(dca, k)); - str = buf; - } - else if (mpl_tab_get_type(dca, k) == 'S') - str = mpl_tab_get_str(dca, k); - else - xassert(dca != dca); - if ((int)strlen(str) > dbf->len[k]) - { xprintf("xBASE driver: field %s: cannot convert %.15s..." - " to field format\n", mpl_tab_get_name(dca, k), str); - longjmp(dbf->jump, 0); - } - for (j = 0; j < dbf->len[k] && str[j] != '\0'; j++) - write_byte(dbf, str[j]); - for (j = j; j < dbf->len[k]; j++) - write_byte(dbf, ' '); - } - else if (dbf->type[k] == 'N') - { /* numeric field */ - double num = mpl_tab_get_num(dca, k); - if (fabs(num) > 1e20) -err: { xprintf("xBASE driver: field %s: cannot convert %g to fi" - "eld format\n", mpl_tab_get_name(dca, k), num); - longjmp(dbf->jump, 0); - } - sprintf(buf, "%*.*f", dbf->len[k], dbf->prec[k], num); - xassert(strlen(buf) < sizeof(buf)); - if ((int)strlen(buf) != dbf->len[k]) goto err; - for (j = 0; j < dbf->len[k]; j++) - write_byte(dbf, buf[j]); - } - else - xassert(dbf != dbf); - } - /* increase record count */ - dbf->count++; -done: return ret; -} - -static int dbf_close_file(TABDCA *dca, struct dbf *dbf) -{ /* close xBASE data file */ - int ret = 0; - xassert(dca == dca); - if (dbf->mode == 'W') - { if (setjmp(dbf->jump)) - { ret = 1; - goto skip; - } - /* end-of-file flag */ - write_byte(dbf, 0x1A); - /* number of records */ - dbf->offset = 4; - if (fseek(dbf->fp, dbf->offset, SEEK_SET)) - { xprintf("%s:0x%X: seek error - %s\n", dbf->fname, -#if 0 /* 29/I-2017 */ - dbf->offset, strerror(errno)); -#else - dbf->offset, xstrerr(errno)); -#endif - longjmp(dbf->jump, 0); - } - write_byte(dbf, dbf->count); - write_byte(dbf, dbf->count >> 8); - write_byte(dbf, dbf->count >> 16); - write_byte(dbf, dbf->count >> 24); - fflush(dbf->fp); - if (ferror(dbf->fp)) - { xprintf("%s:0x%X: write error - %s\n", dbf->fname, -#if 0 /* 29/I-2017 */ - dbf->offset, strerror(errno)); -#else - dbf->offset, xstrerr(errno)); -#endif - longjmp(dbf->jump, 0); - } -skip: ; - } - xfree(dbf->fname); - fclose(dbf->fp); - xfree(dbf); - return ret; -} - -/**********************************************************************/ - -#define TAB_CSV 1 -#define TAB_XBASE 2 -#define TAB_ODBC 3 -#define TAB_MYSQL 4 - -void mpl_tab_drv_open(MPL *mpl, int mode) -{ TABDCA *dca = mpl->dca; - xassert(dca->id == 0); - xassert(dca->link == NULL); - xassert(dca->na >= 1); - if (strcmp(dca->arg[1], "CSV") == 0) - { dca->id = TAB_CSV; - dca->link = csv_open_file(dca, mode); - } - else if (strcmp(dca->arg[1], "xBASE") == 0) - { dca->id = TAB_XBASE; - dca->link = dbf_open_file(dca, mode); - } - else if (strcmp(dca->arg[1], "ODBC") == 0 || - strcmp(dca->arg[1], "iODBC") == 0) - { dca->id = TAB_ODBC; - dca->link = db_iodbc_open(dca, mode); - } - else if (strcmp(dca->arg[1], "MySQL") == 0) - { dca->id = TAB_MYSQL; - dca->link = db_mysql_open(dca, mode); - } - else - xprintf("Invalid table driver '%s'\n", dca->arg[1]); - if (dca->link == NULL) - error(mpl, "error on opening table %s", - mpl->stmt->u.tab->name); - return; -} - -int mpl_tab_drv_read(MPL *mpl) -{ TABDCA *dca = mpl->dca; - int ret; - switch (dca->id) - { case TAB_CSV: - ret = csv_read_record(dca, dca->link); - break; - case TAB_XBASE: - ret = dbf_read_record(dca, dca->link); - break; - case TAB_ODBC: - ret = db_iodbc_read(dca, dca->link); - break; - case TAB_MYSQL: - ret = db_mysql_read(dca, dca->link); - break; - default: - xassert(dca != dca); - } - if (ret > 0) - error(mpl, "error on reading data from table %s", - mpl->stmt->u.tab->name); - return ret; -} - -void mpl_tab_drv_write(MPL *mpl) -{ TABDCA *dca = mpl->dca; - int ret; - switch (dca->id) - { case TAB_CSV: - ret = csv_write_record(dca, dca->link); - break; - case TAB_XBASE: - ret = dbf_write_record(dca, dca->link); - break; - case TAB_ODBC: - ret = db_iodbc_write(dca, dca->link); - break; - case TAB_MYSQL: - ret = db_mysql_write(dca, dca->link); - break; - default: - xassert(dca != dca); - } - if (ret) - error(mpl, "error on writing data to table %s", - mpl->stmt->u.tab->name); - return; -} - -void mpl_tab_drv_close(MPL *mpl) -{ TABDCA *dca = mpl->dca; - int ret; - switch (dca->id) - { case TAB_CSV: - ret = csv_close_file(dca, dca->link); - break; - case TAB_XBASE: - ret = dbf_close_file(dca, dca->link); - break; - case TAB_ODBC: - ret = db_iodbc_close(dca, dca->link); - break; - case TAB_MYSQL: - ret = db_mysql_close(dca, dca->link); - break; - default: - xassert(dca != dca); - } - dca->id = 0; - dca->link = NULL; - if (ret) - error(mpl, "error on closing table %s", - mpl->stmt->u.tab->name); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/mpl/mplsql.c b/code/3rd_glpk/mpl/mplsql.c deleted file mode 100644 index fcd2afa6..00000000 --- a/code/3rd_glpk/mpl/mplsql.c +++ /dev/null @@ -1,1659 +0,0 @@ -/* mplsql.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Author: Heinrich Schuchardt . -* -* Copyright (C) 2003-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifdef HAVE_CONFIG_H -#include -#endif - -#include "mpl.h" -#include "mplsql.h" - -#ifdef ODBC_DLNAME -#define HAVE_ODBC -#define libodbc ODBC_DLNAME -#define h_odbc (get_env_ptr()->h_odbc) -#endif - -#ifdef MYSQL_DLNAME -#define HAVE_MYSQL -#define libmysql MYSQL_DLNAME -#define h_mysql (get_env_ptr()->h_mysql) -#endif - -static void *db_iodbc_open_int(TABDCA *dca, int mode, const char - **sqllines); -static void *db_mysql_open_int(TABDCA *dca, int mode, const char - **sqllines); - -/**********************************************************************/ - -#if defined(HAVE_ODBC) || defined(HAVE_MYSQL) - -#define SQL_FIELD_MAX 100 -/* maximal field count */ - -#define SQL_FDLEN_MAX 255 -/* maximal field length */ - -/*********************************************************************** -* NAME -* -* args_concat - concatenate arguments -* -* SYNOPSIS -* -* static char **args_concat(TABDCA *dca); -* -* DESCRIPTION -* -* The arguments passed in dca are SQL statements. A SQL statement may -* be split over multiple arguments. The last argument of a SQL -* statement will be terminated with a semilocon. Each SQL statement is -* merged into a single zero terminated string. Boundaries between -* arguments are replaced by space. -* -* RETURNS -* -* Buffer with SQL statements */ - -static char **args_concat(TABDCA *dca) -{ - const char *arg; - int i; - int j; - int j0; - int j1; - size_t len; - int lentot; - int narg; - int nline = 0; - char **sqllines = NULL; - - narg = mpl_tab_num_args(dca); - /* The SQL statements start with argument 3. */ - if (narg < 3) - return NULL; - /* Count the SQL statements */ - for (j = 3; j <= narg; j++) - { - arg = mpl_tab_get_arg(dca, j); - len = strlen(arg); - if (arg[len-1] == ';' || j == narg) - nline ++; - } - /* Allocate string buffer. */ - sqllines = (char **) xmalloc((nline+1) * sizeof(char **)); - /* Join arguments */ - sqllines[0] = NULL; - j0 = 3; - i = 0; - lentot = 0; - for (j = 3; j <= narg; j++) - { - arg = mpl_tab_get_arg(dca, j); - len = strlen(arg); - /* add length of part */ - lentot += len; - /* add length of space separating parts or 0x00 at end of SQL - statement */ - lentot++; - if (arg[len-1] == ';' || j == narg) - { /* Join arguments for a single SQL statement */ - sqllines[i] = xmalloc(lentot); - sqllines[i+1] = NULL; - sqllines[i][0] = 0x00; - for (j1 = j0; j1 <= j; j1++) - { if(j1>j0) - strcat(sqllines[i], " "); - strcat(sqllines[i], mpl_tab_get_arg(dca, j1)); - } - len = strlen(sqllines[i]); - if (sqllines[i][len-1] == ';') - sqllines[i][len-1] = 0x00; - j0 = j+1; - i++; - lentot = 0; - } - } - return sqllines; -} - -/*********************************************************************** -* NAME -* -* free_buffer - free multiline string buffer -* -* SYNOPSIS -* -* static void free_buffer(char **buf); -* -* DESCRIPTION -* -* buf is a list of strings terminated by NULL. -* The memory for the strings and for the list is released. */ - -static void free_buffer(char **buf) -{ int i; - - for(i = 0; buf[i] != NULL; i++) - xfree(buf[i]); - xfree(buf); -} - -static int db_escaped_string_length(const char* from) -/* length of escaped string */ -{ - int count; - const char *pointer; - - for (pointer = from, count = 0; *pointer != (char) '\0'; pointer++, - count++) - { - switch (*pointer) - { - case '\'': - count++; - break; - } - } - - return count; -} - -static void db_escape_string (char *to, const char *from) -/* escape string*/ -{ - const char *source = from; - char *target = to; - size_t remaining; - - remaining = strlen(from); - - if (to == NULL) - to = (char *) (from + remaining); - - while (remaining > 0) - { - switch (*source) - { - case '\'': - *target = '\''; - target++; - *target = '\''; - break; - - default: - *target = *source; - } - source++; - target++; - remaining--; - } - - /* Write the terminating NUL character. */ - *target = '\0'; -} - -static char *db_generate_select_stmt(TABDCA *dca) -/* generate select statement */ -{ - char *arg; - char const *field; - char *query; - int j; - int narg; - int nf; - int total; - - total = 50; - nf = mpl_tab_num_flds(dca); - narg = mpl_tab_num_args(dca); - for (j=1; j <= nf && j <= SQL_FIELD_MAX; j++) - { - field = mpl_tab_get_name(dca, j); - total += strlen(field); - total += 2; - } - arg = (char *) mpl_tab_get_arg(dca, narg); - total += strlen(arg); - query = xmalloc( total * sizeof(char)); - strcpy (query, "SELECT "); - for (j=1; j <= nf && j <= SQL_FIELD_MAX; j++) - { - field = mpl_tab_get_name(dca, j); - strcat(query, field); - if ( j < nf ) - strcat(query, ", "); - } - strcat(query, " FROM "); - strcat(query, arg); - return query; -} - -static char *db_generate_insert_stmt(TABDCA *dca) -/* generate insert statement */ -{ - char *arg; - char const *field; - char *query; - int j; - int narg; - int nf; - int total; - - total = 50; - nf = mpl_tab_num_flds(dca); - narg = mpl_tab_num_args(dca); - for (j=1; j <= nf && j <= SQL_FIELD_MAX; j++) - { - field = mpl_tab_get_name(dca, j); - total += strlen(field); - total += 5; - } - arg = (char *) mpl_tab_get_arg(dca, narg); - total += strlen(arg); - query = xmalloc( (total+1) * sizeof(char)); - strcpy (query, "INSERT INTO "); - strcat(query, arg); - strcat(query, " ( "); - for (j=1; j <= nf && j <= SQL_FIELD_MAX; j++) - { - field = mpl_tab_get_name(dca, j); - strcat(query, field); - if ( j < nf ) - strcat(query, ", "); - } - strcat(query, " ) VALUES ( "); - for (j=1; j <= nf && j <= SQL_FIELD_MAX; j++) - { - strcat(query, "?"); - if ( j < nf ) - strcat(query, ", "); - } - strcat(query, " )"); - return query; -} - -#endif - -/**********************************************************************/ - -#ifndef HAVE_ODBC - -void *db_iodbc_open(TABDCA *dca, int mode) -{ xassert(dca == dca); - xassert(mode == mode); - xprintf("iODBC table driver not supported\n"); - return NULL; -} - -int db_iodbc_read(TABDCA *dca, void *link) -{ xassert(dca != dca); - xassert(link != link); - return 0; -} - -int db_iodbc_write(TABDCA *dca, void *link) -{ xassert(dca != dca); - xassert(link != link); - return 0; -} - -int db_iodbc_close(TABDCA *dca, void *link) -{ xassert(dca != dca); - xassert(link != link); - return 0; -} - -#else - -#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__WOE__) -#include -#endif - -#include -#include - -struct db_odbc -{ - int mode; /*'R' = Read, 'W' = Write*/ - SQLHDBC hdbc; /*connection handle*/ - SQLHENV henv; /*environment handle*/ - SQLHSTMT hstmt; /*statement handle*/ - SQLSMALLINT nresultcols; /* columns in result*/ - SQLULEN collen[SQL_FIELD_MAX+1]; - SQLLEN outlen[SQL_FIELD_MAX+1]; - SQLSMALLINT coltype[SQL_FIELD_MAX+1]; - SQLCHAR data[SQL_FIELD_MAX+1][SQL_FDLEN_MAX+1]; -#if 1 /* 12/I-2014 */ - SQLDOUBLE datanum[SQL_FIELD_MAX+1]; -#endif - SQLCHAR colname[SQL_FIELD_MAX+1][SQL_FDLEN_MAX+1]; - int isnumeric[SQL_FIELD_MAX+1]; - int nf; - /* number of fields in the csv file */ - int ref[1+SQL_FIELD_MAX]; - /* ref[k] = k', if k-th field of the csv file corresponds to - k'-th field in the table statement; if ref[k] = 0, k-th field - of the csv file is ignored */ - SQLCHAR *query; - /* query generated by db_iodbc_open */ -}; - -SQLRETURN SQL_API dl_SQLAllocHandle ( - SQLSMALLINT HandleType, - SQLHANDLE InputHandle, - SQLHANDLE *OutputHandle) -{ - typedef SQLRETURN SQL_API ep_SQLAllocHandle( - SQLSMALLINT HandleType, - SQLHANDLE InputHandle, - SQLHANDLE *OutputHandle); - - ep_SQLAllocHandle *fn; - fn = (ep_SQLAllocHandle *) xdlsym(h_odbc, "SQLAllocHandle"); - xassert(fn != NULL); - return (*fn)(HandleType, InputHandle, OutputHandle); -} - -SQLRETURN SQL_API dl_SQLBindCol ( - SQLHSTMT StatementHandle, - SQLUSMALLINT ColumnNumber, - SQLSMALLINT TargetType, - SQLPOINTER TargetValue, - SQLLEN BufferLength, - SQLLEN *StrLen_or_Ind) -{ - typedef SQLRETURN SQL_API ep_SQLBindCol( - SQLHSTMT StatementHandle, - SQLUSMALLINT ColumnNumber, - SQLSMALLINT TargetType, - SQLPOINTER TargetValue, - SQLLEN BufferLength, - SQLLEN *StrLen_or_Ind); - ep_SQLBindCol *fn; - fn = (ep_SQLBindCol *) xdlsym(h_odbc, "SQLBindCol"); - xassert(fn != NULL); - return (*fn)(StatementHandle, ColumnNumber, TargetType, - TargetValue, BufferLength, StrLen_or_Ind); -} - -SQLRETURN SQL_API dl_SQLCloseCursor ( - SQLHSTMT StatementHandle) -{ - typedef SQLRETURN SQL_API ep_SQLCloseCursor ( - SQLHSTMT StatementHandle); - - ep_SQLCloseCursor *fn; - fn = (ep_SQLCloseCursor *) xdlsym(h_odbc, "SQLCloseCursor"); - xassert(fn != NULL); - return (*fn)(StatementHandle); -} - - -SQLRETURN SQL_API dl_SQLDisconnect ( - SQLHDBC ConnectionHandle) -{ - typedef SQLRETURN SQL_API ep_SQLDisconnect( - SQLHDBC ConnectionHandle); - - ep_SQLDisconnect *fn; - fn = (ep_SQLDisconnect *) xdlsym(h_odbc, "SQLDisconnect"); - xassert(fn != NULL); - return (*fn)(ConnectionHandle); -} - -SQLRETURN SQL_API dl_SQLDriverConnect ( - SQLHDBC hdbc, - SQLHWND hwnd, - SQLCHAR *szConnStrIn, - SQLSMALLINT cbConnStrIn, - SQLCHAR *szConnStrOut, - SQLSMALLINT cbConnStrOutMax, - SQLSMALLINT *pcbConnStrOut, - SQLUSMALLINT fDriverCompletion) -{ - typedef SQLRETURN SQL_API ep_SQLDriverConnect( - SQLHDBC hdbc, - SQLHWND hwnd, - SQLCHAR * szConnStrIn, - SQLSMALLINT cbConnStrIn, - SQLCHAR * szConnStrOut, - SQLSMALLINT cbConnStrOutMax, - SQLSMALLINT * pcbConnStrOut, - SQLUSMALLINT fDriverCompletion); - - ep_SQLDriverConnect *fn; - fn = (ep_SQLDriverConnect *) xdlsym(h_odbc, "SQLDriverConnect"); - xassert(fn != NULL); - return (*fn)(hdbc, hwnd, szConnStrIn, cbConnStrIn, szConnStrOut, - cbConnStrOutMax, pcbConnStrOut, fDriverCompletion); -} - -SQLRETURN SQL_API dl_SQLEndTran ( - SQLSMALLINT HandleType, - SQLHANDLE Handle, - SQLSMALLINT CompletionType) -{ - typedef SQLRETURN SQL_API ep_SQLEndTran ( - SQLSMALLINT HandleType, - SQLHANDLE Handle, - SQLSMALLINT CompletionType); - - ep_SQLEndTran *fn; - fn = (ep_SQLEndTran *) xdlsym(h_odbc, "SQLEndTran"); - xassert(fn != NULL); - return (*fn)(HandleType, Handle, CompletionType); -} - -SQLRETURN SQL_API dl_SQLExecDirect ( - SQLHSTMT StatementHandle, - SQLCHAR * StatementText, - SQLINTEGER TextLength) -{ - typedef SQLRETURN SQL_API ep_SQLExecDirect ( - SQLHSTMT StatementHandle, - SQLCHAR * StatementText, - SQLINTEGER TextLength); - - ep_SQLExecDirect *fn; - fn = (ep_SQLExecDirect *) xdlsym(h_odbc, "SQLExecDirect"); - xassert(fn != NULL); - return (*fn)(StatementHandle, StatementText, TextLength); -} - -SQLRETURN SQL_API dl_SQLFetch ( - SQLHSTMT StatementHandle) -{ - typedef SQLRETURN SQL_API ep_SQLFetch ( - SQLHSTMT StatementHandle); - - ep_SQLFetch *fn; - fn = (ep_SQLFetch*) xdlsym(h_odbc, "SQLFetch"); - xassert(fn != NULL); - return (*fn)(StatementHandle); -} - -SQLRETURN SQL_API dl_SQLFreeHandle ( - SQLSMALLINT HandleType, - SQLHANDLE Handle) -{ - typedef SQLRETURN SQL_API ep_SQLFreeHandle ( - SQLSMALLINT HandleType, - SQLHANDLE Handle); - - ep_SQLFreeHandle *fn; - fn = (ep_SQLFreeHandle *) xdlsym(h_odbc, "SQLFreeHandle"); - xassert(fn != NULL); - return (*fn)(HandleType, Handle); -} - -SQLRETURN SQL_API dl_SQLDescribeCol ( - SQLHSTMT StatementHandle, - SQLUSMALLINT ColumnNumber, - SQLCHAR * ColumnName, - SQLSMALLINT BufferLength, - SQLSMALLINT * NameLength, - SQLSMALLINT * DataType, - SQLULEN * ColumnSize, - SQLSMALLINT * DecimalDigits, - SQLSMALLINT * Nullable) -{ - typedef SQLRETURN SQL_API ep_SQLDescribeCol ( - SQLHSTMT StatementHandle, - SQLUSMALLINT ColumnNumber, - SQLCHAR *ColumnName, - SQLSMALLINT BufferLength, - SQLSMALLINT *NameLength, - SQLSMALLINT *DataType, - SQLULEN *ColumnSize, - SQLSMALLINT *DecimalDigits, - SQLSMALLINT *Nullable); - - ep_SQLDescribeCol *fn; - fn = (ep_SQLDescribeCol *) xdlsym(h_odbc, "SQLDescribeCol"); - xassert(fn != NULL); - return (*fn)(StatementHandle, ColumnNumber, ColumnName, - BufferLength, NameLength, - DataType, ColumnSize, DecimalDigits, Nullable); -} - -SQLRETURN SQL_API dl_SQLGetDiagRec ( - SQLSMALLINT HandleType, - SQLHANDLE Handle, - SQLSMALLINT RecNumber, - SQLCHAR *Sqlstate, - SQLINTEGER *NativeError, - SQLCHAR *MessageText, - SQLSMALLINT BufferLength, - SQLSMALLINT *TextLength) -{ - typedef SQLRETURN SQL_API ep_SQLGetDiagRec ( - SQLSMALLINT HandleType, - SQLHANDLE Handle, - SQLSMALLINT RecNumber, - SQLCHAR *Sqlstate, - SQLINTEGER *NativeError, - SQLCHAR *MessageText, - SQLSMALLINT BufferLength, - SQLSMALLINT *TextLength); - - ep_SQLGetDiagRec *fn; - fn = (ep_SQLGetDiagRec *) xdlsym(h_odbc, "SQLGetDiagRec"); - xassert(fn != NULL); - return (*fn)(HandleType, Handle, RecNumber, Sqlstate, - NativeError, MessageText, BufferLength, TextLength); -} - -SQLRETURN SQL_API dl_SQLGetInfo ( - SQLHDBC ConnectionHandle, - SQLUSMALLINT InfoType, - SQLPOINTER InfoValue, - SQLSMALLINT BufferLength, - SQLSMALLINT *StringLength) -{ - typedef SQLRETURN SQL_API ep_SQLGetInfo ( - SQLHDBC ConnectionHandle, - SQLUSMALLINT InfoType, - SQLPOINTER InfoValue, - SQLSMALLINT BufferLength, - SQLSMALLINT *StringLength); - - ep_SQLGetInfo *fn; - fn = (ep_SQLGetInfo *) xdlsym(h_odbc, "SQLGetInfo"); - xassert(fn != NULL); - return (*fn)(ConnectionHandle, InfoType, InfoValue, BufferLength, - StringLength); -} - -SQLRETURN SQL_API dl_SQLNumResultCols ( - SQLHSTMT StatementHandle, - SQLSMALLINT *ColumnCount) -{ - typedef SQLRETURN SQL_API ep_SQLNumResultCols ( - SQLHSTMT StatementHandle, - SQLSMALLINT *ColumnCount); - - ep_SQLNumResultCols *fn; - fn = (ep_SQLNumResultCols *) xdlsym(h_odbc, "SQLNumResultCols"); - xassert(fn != NULL); - return (*fn)(StatementHandle, ColumnCount); -} - -SQLRETURN SQL_API dl_SQLSetConnectAttr ( - SQLHDBC ConnectionHandle, - SQLINTEGER Attribute, - SQLPOINTER Value, - SQLINTEGER StringLength) -{ - typedef SQLRETURN SQL_API ep_SQLSetConnectAttr ( - SQLHDBC ConnectionHandle, - SQLINTEGER Attribute, - SQLPOINTER Value, - SQLINTEGER StringLength); - - ep_SQLSetConnectAttr *fn; - fn = (ep_SQLSetConnectAttr *) xdlsym(h_odbc, "SQLSetConnectAttr"); - xassert(fn != NULL); - return (*fn)(ConnectionHandle, Attribute, Value, StringLength); -} - -SQLRETURN SQL_API dl_SQLSetEnvAttr ( - SQLHENV EnvironmentHandle, - SQLINTEGER Attribute, - SQLPOINTER Value, - SQLINTEGER StringLength) -{ - typedef SQLRETURN SQL_API ep_SQLSetEnvAttr ( - SQLHENV EnvironmentHandle, - SQLINTEGER Attribute, - SQLPOINTER Value, - SQLINTEGER StringLength); - - ep_SQLSetEnvAttr *fn; - fn = (ep_SQLSetEnvAttr *) xdlsym(h_odbc, "SQLSetEnvAttr"); - xassert(fn != NULL); - return (*fn)(EnvironmentHandle, Attribute, Value, StringLength); -} - -static void extract_error( - char *fn, - SQLHANDLE handle, - SQLSMALLINT type); - -static int is_numeric( - SQLSMALLINT coltype); - -/*********************************************************************** -* NAME -* -* db_iodbc_open - open connection to ODBC data base -* -* SYNOPSIS -* -* #include "mplsql.h" -* void *db_iodbc_open(TABDCA *dca, int mode); -* -* DESCRIPTION -* -* The routine db_iodbc_open opens a connection to an ODBC data base. -* It then executes the sql statements passed. -* -* In the case of table read the SELECT statement is executed. -* -* In the case of table write the INSERT statement is prepared. -* RETURNS -* -* The routine returns a pointer to data storage area created. */ -void *db_iodbc_open(TABDCA *dca, int mode) -{ void *ret; - char **sqllines; - - sqllines = args_concat(dca); - if (sqllines == NULL) - { xprintf("Missing arguments in table statement.\n" - "Please, supply table driver, dsn, and query.\n"); - return NULL; - } - ret = db_iodbc_open_int(dca, mode, (const char **) sqllines); - free_buffer(sqllines); - return ret; -} - -static void *db_iodbc_open_int(TABDCA *dca, int mode, const char - **sqllines) -{ - struct db_odbc *sql; - SQLRETURN ret; - SQLCHAR FAR *dsn; - SQLCHAR info[256]; - SQLSMALLINT colnamelen; - SQLSMALLINT nullable; - SQLSMALLINT scale; - const char *arg; - int narg; - int i, j; - int total; - - if (libodbc == NULL) - { - xprintf("No loader for shared ODBC library available\n"); - return NULL; - } - - if (h_odbc == NULL) - { - h_odbc = xdlopen(libodbc); - if (h_odbc == NULL) - { xprintf("unable to open library %s\n", libodbc); - xprintf("%s\n", get_err_msg()); - return NULL; - } - } - - sql = (struct db_odbc *) xmalloc(sizeof(struct db_odbc)); - if (sql == NULL) - return NULL; - - sql->mode = mode; - sql->hdbc = NULL; - sql->henv = NULL; - sql->hstmt = NULL; - sql->query = NULL; - narg = mpl_tab_num_args(dca); - - dsn = (SQLCHAR FAR *) mpl_tab_get_arg(dca, 2); - /* allocate an environment handle */ - ret = dl_SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, - &(sql->henv)); - /* set attribute to enable application to run as ODBC 3.0 - application */ - ret = dl_SQLSetEnvAttr(sql->henv, SQL_ATTR_ODBC_VERSION, - (void *) SQL_OV_ODBC3, 0); - /* allocate a connection handle */ - ret = dl_SQLAllocHandle(SQL_HANDLE_DBC, sql->henv, &(sql->hdbc)); - /* connect */ - ret = dl_SQLDriverConnect(sql->hdbc, NULL, dsn, SQL_NTS, NULL, 0, - NULL, SQL_DRIVER_COMPLETE); - if (SQL_SUCCEEDED(ret)) - { /* output information about data base connection */ - xprintf("Connected to "); - dl_SQLGetInfo(sql->hdbc, SQL_DBMS_NAME, (SQLPOINTER)info, - sizeof(info), NULL); - xprintf("%s ", info); - dl_SQLGetInfo(sql->hdbc, SQL_DBMS_VER, (SQLPOINTER)info, - sizeof(info), NULL); - xprintf("%s - ", info); - dl_SQLGetInfo(sql->hdbc, SQL_DATABASE_NAME, (SQLPOINTER)info, - sizeof(info), NULL); - xprintf("%s\n", info); - } - else - { /* describe error */ - xprintf("Failed to connect\n"); - extract_error("SQLDriverConnect", sql->hdbc, SQL_HANDLE_DBC); - dl_SQLFreeHandle(SQL_HANDLE_DBC, sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_ENV, sql->henv); - xfree(sql); - return NULL; - } - /* set AUTOCOMMIT on*/ - ret = dl_SQLSetConnectAttr(sql->hdbc, SQL_ATTR_AUTOCOMMIT, - (SQLPOINTER)SQL_AUTOCOMMIT_ON, 0); - /* allocate a statement handle */ - ret = dl_SQLAllocHandle(SQL_HANDLE_STMT, sql->hdbc, &(sql->hstmt)); - - /* initialization queries */ - for(j = 0; sqllines[j+1] != NULL; j++) - { - sql->query = (SQLCHAR *) sqllines[j]; - xprintf("%s\n", sql->query); - ret = dl_SQLExecDirect(sql->hstmt, sql->query, SQL_NTS); - switch (ret) - { - case SQL_SUCCESS: - case SQL_SUCCESS_WITH_INFO: - case SQL_NO_DATA_FOUND: - break; - default: - xprintf("db_iodbc_open: Query\n\"%s\"\nfailed.\n", - sql->query); - extract_error("SQLExecDirect", sql->hstmt, SQL_HANDLE_STMT); - dl_SQLFreeHandle(SQL_HANDLE_STMT, sql->hstmt); - dl_SQLDisconnect(sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_DBC, sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_ENV, sql->henv); - xfree(sql); - return NULL; - } - /* commit statement */ - dl_SQLEndTran(SQL_HANDLE_ENV, sql->henv, SQL_COMMIT); - } - - if ( sql->mode == 'R' ) - { sql->nf = mpl_tab_num_flds(dca); - for(j = 0; sqllines[j] != NULL; j++) - arg = sqllines[j]; - total = strlen(arg); - if (total > 7 && 0 == strncmp(arg, "SELECT ", 7)) - { - total = strlen(arg); - sql->query = xmalloc( (total+1) * sizeof(char)); - strcpy (sql->query, arg); - } - else - { - sql->query = db_generate_select_stmt(dca); - } - xprintf("%s\n", sql->query); - if (dl_SQLExecDirect(sql->hstmt, sql->query, SQL_NTS) != - SQL_SUCCESS) - { - xprintf("db_iodbc_open: Query\n\"%s\"\nfailed.\n", sql->query); - extract_error("SQLExecDirect", sql->hstmt, SQL_HANDLE_STMT); - dl_SQLFreeHandle(SQL_HANDLE_STMT, sql->hstmt); - dl_SQLDisconnect(sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_DBC, sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_ENV, sql->henv); - xfree(sql->query); - xfree(sql); - return NULL; - } - xfree(sql->query); - /* determine number of result columns */ - ret = dl_SQLNumResultCols(sql->hstmt, &sql->nresultcols); - total = sql->nresultcols; - if (total > SQL_FIELD_MAX) - { xprintf("db_iodbc_open: Too many fields (> %d) in query.\n" - "\"%s\"\n", SQL_FIELD_MAX, sql->query); - dl_SQLFreeHandle(SQL_HANDLE_STMT, sql->hstmt); - dl_SQLDisconnect(sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_DBC, sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_ENV, sql->henv); - xfree(sql->query); - return NULL; - } - for (i = 1; i <= total; i++) - { /* return a set of attributes for a column */ - ret = dl_SQLDescribeCol(sql->hstmt, (SQLSMALLINT) i, - sql->colname[i], SQL_FDLEN_MAX, - &colnamelen, &(sql->coltype[i]), &(sql->collen[i]), &scale, - &nullable); - sql->isnumeric[i] = is_numeric(sql->coltype[i]); - /* bind columns to program vars, converting all types to CHAR*/ - if (sql->isnumeric[i]) -#if 0 /* 12/I-2014 */ - { dl_SQLBindCol(sql->hstmt, i, SQL_DOUBLE, sql->data[i], -#else - { dl_SQLBindCol(sql->hstmt, i, SQL_DOUBLE, &sql->datanum[i], -#endif - SQL_FDLEN_MAX, &(sql->outlen[i])); - } else - { dl_SQLBindCol(sql->hstmt, i, SQL_CHAR, sql->data[i], - SQL_FDLEN_MAX, &(sql->outlen[i])); - } - for (j = sql->nf; j >= 1; j--) - { if (strcmp(mpl_tab_get_name(dca, j), sql->colname[i]) == 0) - break; - } - sql->ref[i] = j; - } - } - else if ( sql->mode == 'W' ) - { for(j = 0; sqllines[j] != NULL; j++) - arg = sqllines[j]; - if ( NULL != strchr(arg, '?') ) - { - total = strlen(arg); - sql->query = xmalloc( (total+1) * sizeof(char)); - strcpy (sql->query, arg); - } - else - { - sql->query = db_generate_insert_stmt(dca); - } - xprintf("%s\n", sql->query); - } - return sql; -} - -int db_iodbc_read(TABDCA *dca, void *link) -{ - struct db_odbc *sql; - SQLRETURN ret; - char buf[SQL_FDLEN_MAX+1]; - int i; - int len; - double num; - - sql = (struct db_odbc *) link; - - xassert(sql != NULL); - xassert(sql->mode == 'R'); - - ret=dl_SQLFetch(sql->hstmt); - if (ret== SQL_ERROR) - return -1; - if (ret== SQL_NO_DATA_FOUND) - return -1; /*EOF*/ - for (i=1; i <= sql->nresultcols; i++) - { - if (sql->ref[i] > 0) - { - len = sql->outlen[i]; - if (len != SQL_NULL_DATA) - { - if (sql->isnumeric[i]) - { mpl_tab_set_num(dca, sql->ref[i], -#if 0 /* 12/I-2014 */ - *((const double *) sql->data[i])); -#else - (const double) sql->datanum[i]); -#endif - } - else - { if (len > SQL_FDLEN_MAX) - len = SQL_FDLEN_MAX; - else if (len < 0) - len = 0; - strncpy(buf, (const char *) sql->data[i], len); - buf[len] = 0x00; - mpl_tab_set_str(dca, sql->ref[i], strtrim(buf)); - } - } - } - } - return 0; -} - -int db_iodbc_write(TABDCA *dca, void *link) -{ - struct db_odbc *sql; - char *part; - char *query; - char *template; - char num[50]; - int k; - int len; - int nf; - - sql = (struct db_odbc *) link; - xassert(sql != NULL); - xassert(sql->mode == 'W'); - - len = strlen(sql->query); - template = (char *) xmalloc( (len + 1) * sizeof(char) ); - strcpy(template, sql->query); - - nf = mpl_tab_num_flds(dca); - for (k = 1; k <= nf; k++) - { switch (mpl_tab_get_type(dca, k)) - { case 'N': - len += 20; - break; - case 'S': - len += db_escaped_string_length(mpl_tab_get_str(dca, k)); - len += 2; - break; - default: - xassert(dca != dca); - } - } - query = xmalloc( (len + 1 ) * sizeof(char) ); - query[0] = 0x00; -#if 0 /* 29/I-2017 */ - for (k = 1, part = strtok (template, "?"); (part != NULL); - part = strtok (NULL, "?"), k++) -#else - for (k = 1, part = xstrtok (template, "?"); (part != NULL); - part = xstrtok (NULL, "?"), k++) -#endif - { - if (k > nf) break; - strcat( query, part ); - switch (mpl_tab_get_type(dca, k)) - { case 'N': -#if 0 /* 02/XI-2010 by xypron */ - sprintf(num, "%-18g",mpl_tab_get_num(dca, k)); -#else - sprintf(num, "%.*g", DBL_DIG, mpl_tab_get_num(dca, k)); -#endif - strcat( query, num ); - break; - case 'S': - strcat( query, "'"); - db_escape_string( query + strlen(query), - mpl_tab_get_str(dca, k) ); - strcat( query, "'"); - break; - default: - xassert(dca != dca); - } - } - if (part != NULL) - strcat(query, part); - if (dl_SQLExecDirect(sql->hstmt, (SQLCHAR *) query, SQL_NTS) - != SQL_SUCCESS) - { - xprintf("db_iodbc_write: Query\n\"%s\"\nfailed.\n", query); - extract_error("SQLExecDirect", sql->hdbc, SQL_HANDLE_DBC); - xfree(query); - xfree(template); - return 1; - } - - xfree(query); - xfree(template); - return 0; -} - -int db_iodbc_close(TABDCA *dca, void *link) -{ - struct db_odbc *sql; - - sql = (struct db_odbc *) link; - xassert(sql != NULL); - /* Commit */ - if ( sql->mode == 'W' ) - dl_SQLEndTran(SQL_HANDLE_ENV, sql->henv, SQL_COMMIT); - if ( sql->mode == 'R' ) - dl_SQLCloseCursor(sql->hstmt); - - dl_SQLFreeHandle(SQL_HANDLE_STMT, sql->hstmt); - dl_SQLDisconnect(sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_DBC, sql->hdbc); - dl_SQLFreeHandle(SQL_HANDLE_ENV, sql->henv); - if ( sql->mode == 'W' ) - xfree(sql->query); - xfree(sql); - dca->link = NULL; - return 0; -} - -static void extract_error( - char *fn, - SQLHANDLE handle, - SQLSMALLINT type) -{ - SQLINTEGER i = 0; - SQLINTEGER native; - SQLCHAR state[ 7 ]; - SQLCHAR text[256]; - SQLSMALLINT len; - SQLRETURN ret; - - xprintf("\nThe driver reported the following diagnostics whilst " - "running %s\n", fn); - - do - { - ret = dl_SQLGetDiagRec(type, handle, ++i, state, &native, text, - sizeof(text), &len ); - if (SQL_SUCCEEDED(ret)) - xprintf("%s:%ld:%ld:%s\n", state, i, native, text); - } - while( ret == SQL_SUCCESS ); -} - -static int is_numeric(SQLSMALLINT coltype) -{ - int ret = 0; - switch (coltype) - { - case SQL_DECIMAL: - case SQL_NUMERIC: - case SQL_SMALLINT: - case SQL_INTEGER: - case SQL_REAL: - case SQL_FLOAT: - case SQL_DOUBLE: - case SQL_TINYINT: - case SQL_BIGINT: - ret = 1; - break; - } - return ret; -} - -#endif - -/**********************************************************************/ - -#ifndef HAVE_MYSQL - -void *db_mysql_open(TABDCA *dca, int mode) -{ xassert(dca == dca); - xassert(mode == mode); - xprintf("MySQL table driver not supported\n"); - return NULL; -} - -int db_mysql_read(TABDCA *dca, void *link) -{ xassert(dca != dca); - xassert(link != link); - return 0; -} - -int db_mysql_write(TABDCA *dca, void *link) -{ xassert(dca != dca); - xassert(link != link); - return 0; -} - -int db_mysql_close(TABDCA *dca, void *link) -{ xassert(dca != dca); - xassert(link != link); - return 0; -} - -#else - -#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__WOE__) -#include -#endif - -#ifdef __CYGWIN__ -#define byte_defined 1 -#endif - -#if 0 /* 12/II-2014; to fix namespace bug */ -#include -#include -#endif -#include - -struct db_mysql -{ - int mode; /*'R' = Read, 'W' = Write*/ - MYSQL *con; /*connection*/ - MYSQL_RES *res; /*result*/ - int nf; - /* number of fields in the csv file */ - int ref[1+SQL_FIELD_MAX]; - /* ref[k] = k', if k-th field of the csv file corresponds to - k'-th field in the table statement; if ref[k] = 0, k-th field - of the csv file is ignored */ - char *query; - /* query generated by db_mysql_open */ -}; - -void STDCALL dl_mysql_close(MYSQL *sock) -{ - typedef void STDCALL ep_mysql_close(MYSQL *sock); - - ep_mysql_close *fn; - fn = (ep_mysql_close *) xdlsym(h_mysql, "mysql_close"); - xassert(fn != NULL); - return (*fn)(sock); -} - -const char * STDCALL dl_mysql_error(MYSQL *mysql) -{ - typedef const char * STDCALL ep_mysql_error(MYSQL *mysql); - - ep_mysql_error *fn; - fn = (ep_mysql_error *) xdlsym(h_mysql, "mysql_error"); - xassert(fn != NULL); - return (*fn)(mysql); -} - -MYSQL_FIELD * STDCALL dl_mysql_fetch_fields(MYSQL_RES *res) -{ - typedef MYSQL_FIELD * STDCALL - ep_mysql_fetch_fields(MYSQL_RES *res); - - ep_mysql_fetch_fields *fn; - fn = (ep_mysql_fetch_fields *) xdlsym(h_mysql, "mysql_fetch_fields"); - xassert(fn != NULL); - return (*fn)(res); -} - -unsigned long * STDCALL dl_mysql_fetch_lengths(MYSQL_RES *result) -{ - typedef unsigned long * STDCALL - ep_mysql_fetch_lengths(MYSQL_RES *result); - - ep_mysql_fetch_lengths *fn; - fn = (ep_mysql_fetch_lengths *) xdlsym(h_mysql, - "mysql_fetch_lengths"); - xassert(fn != NULL); - return (*fn)(result); -} - -MYSQL_ROW STDCALL dl_mysql_fetch_row(MYSQL_RES *result) -{ - typedef MYSQL_ROW STDCALL ep_mysql_fetch_row(MYSQL_RES *result); - - ep_mysql_fetch_row *fn; - fn = (ep_mysql_fetch_row *) xdlsym(h_mysql, "mysql_fetch_row"); - xassert(fn != NULL); - return (*fn)(result); -} - -unsigned int STDCALL dl_mysql_field_count(MYSQL *mysql) -{ - typedef unsigned int STDCALL ep_mysql_field_count(MYSQL *mysql); - - ep_mysql_field_count *fn; - fn = (ep_mysql_field_count *) xdlsym(h_mysql, "mysql_field_count"); - xassert(fn != NULL); - return (*fn)(mysql); -} - -MYSQL * STDCALL dl_mysql_init(MYSQL *mysql) -{ - typedef MYSQL * STDCALL ep_mysql_init(MYSQL *mysql); - - ep_mysql_init *fn; - fn = (ep_mysql_init *) xdlsym(h_mysql, "mysql_init"); - xassert(fn != NULL); - return (*fn)(mysql); -} - -unsigned int STDCALL dl_mysql_num_fields(MYSQL_RES *res) -{ - typedef unsigned int STDCALL ep_mysql_num_fields(MYSQL_RES *res); - - ep_mysql_num_fields *fn; - fn = (ep_mysql_num_fields *) xdlsym(h_mysql, "mysql_num_fields"); - xassert(fn != NULL); - return (*fn)(res); -} - -int STDCALL dl_mysql_query(MYSQL *mysql, const char *q) -{ - typedef int STDCALL ep_mysql_query(MYSQL *mysql, const char *q); - - ep_mysql_query *fn; - fn = (ep_mysql_query *) xdlsym(h_mysql, "mysql_query"); - xassert(fn != NULL); - return (*fn)(mysql, q); -} - -MYSQL * STDCALL dl_mysql_real_connect(MYSQL *mysql, const char *host, - const char *user, - const char *passwd, - const char *db, - unsigned int port, - const char *unix_socket, - unsigned long clientflag) -{ - typedef MYSQL * STDCALL ep_mysql_real_connect(MYSQL *mysql, - const char *host, - const char *user, - const char *passwd, - const char *db, - unsigned int port, - const char *unix_socket, - unsigned long clientflag); - - ep_mysql_real_connect *fn; - fn = (ep_mysql_real_connect *) xdlsym(h_mysql, - "mysql_real_connect"); - xassert(fn != NULL); - return (*fn)(mysql, host, user, passwd, db, port, unix_socket, - clientflag); -} - -MYSQL_RES * STDCALL dl_mysql_use_result(MYSQL *mysql) -{ - typedef MYSQL_RES * STDCALL ep_mysql_use_result(MYSQL *mysql); - ep_mysql_use_result *fn; - fn = (ep_mysql_use_result *) xdlsym(h_mysql, "mysql_use_result"); - xassert(fn != NULL); - return (*fn)(mysql); -} - -/*********************************************************************** -* NAME -* -* db_mysql_open - open connection to ODBC data base -* -* SYNOPSIS -* -* #include "mplsql.h" -* void *db_mysql_open(TABDCA *dca, int mode); -* -* DESCRIPTION -* -* The routine db_mysql_open opens a connection to a MySQL data base. -* It then executes the sql statements passed. -* -* In the case of table read the SELECT statement is executed. -* -* In the case of table write the INSERT statement is prepared. -* RETURNS -* -* The routine returns a pointer to data storage area created. */ - -void *db_mysql_open(TABDCA *dca, int mode) -{ void *ret; - char **sqllines; - - sqllines = args_concat(dca); - if (sqllines == NULL) - { xprintf("Missing arguments in table statement.\n" - "Please, supply table driver, dsn, and query.\n"); - return NULL; - } - ret = db_mysql_open_int(dca, mode, (const char **) sqllines); - free_buffer(sqllines); - return ret; -} - -static void *db_mysql_open_int(TABDCA *dca, int mode, const char - **sqllines) -{ - struct db_mysql *sql = NULL; - char *arg = NULL; - const char *field; - MYSQL_FIELD *fields; - char *keyword; - char *value; - char *query; - char *dsn; -/* "Server=[server_name];Database=[database_name];UID=[username];*/ -/* PWD=[password];Port=[port]"*/ - char *server = NULL; /* Server */ - char *user = NULL; /* UID */ - char *password = NULL; /* PWD */ - char *database = NULL; /* Database */ - unsigned int port = 0; /* Port */ - int narg; - int i, j, total; - - if (libmysql == NULL) - { - xprintf("No loader for shared MySQL library available\n"); - return NULL; - } - - if (h_mysql == NULL) - { - h_mysql = xdlopen(libmysql); - if (h_mysql == NULL) - { xprintf("unable to open library %s\n", libmysql); - xprintf("%s\n", get_err_msg()); - return NULL; - } - } - - sql = (struct db_mysql *) xmalloc(sizeof(struct db_mysql)); - if (sql == NULL) - return NULL; - sql->mode = mode; - sql->res = NULL; - sql->query = NULL; - sql->nf = mpl_tab_num_flds(dca); - - narg = mpl_tab_num_args(dca); - if (narg < 3 ) - xprintf("MySQL driver: string list too short \n"); - - /* get connection string*/ - dsn = (char *) mpl_tab_get_arg(dca, 2); - /* copy connection string*/ - i = strlen(dsn); - i++; - arg = xmalloc(i * sizeof(char)); - strcpy(arg, dsn); - /*tokenize connection string*/ -#if 0 /* 29/I-2017 */ - for (i = 1, keyword = strtok (arg, "="); (keyword != NULL); - keyword = strtok (NULL, "="), i++) -#else - for (i = 1, keyword = xstrtok (arg, "="); (keyword != NULL); - keyword = xstrtok (NULL, "="), i++) -#endif - { -#if 0 /* 29/I-2017 */ - value = strtok (NULL, ";"); -#else - value = xstrtok (NULL, ";"); -#endif - if (value==NULL) - { - xprintf("db_mysql_open: Missing value for keyword %s\n", - keyword); - xfree(arg); - xfree(sql); - return NULL; - } - if (0 == strcmp(keyword, "Server")) - server = value; - else if (0 == strcmp(keyword, "Database")) - database = value; - else if (0 == strcmp(keyword, "UID")) - user = value; - else if (0 == strcmp(keyword, "PWD")) - password = value; - else if (0 == strcmp(keyword, "Port")) - port = (unsigned int) atol(value); - } - /* Connect to database */ - sql->con = dl_mysql_init(NULL); - if (!dl_mysql_real_connect(sql->con, server, user, password, database, - port, NULL, 0)) - { - xprintf("db_mysql_open: Connect failed\n"); - xprintf("%s\n", dl_mysql_error(sql->con)); - xfree(arg); - xfree(sql); - return NULL; - } - xfree(arg); - - for(j = 0; sqllines[j+1] != NULL; j++) - { query = (char *) sqllines[j]; - xprintf("%s\n", query); - if (dl_mysql_query(sql->con, query)) - { - xprintf("db_mysql_open: Query\n\"%s\"\nfailed.\n", query); - xprintf("%s\n",dl_mysql_error(sql->con)); - dl_mysql_close(sql->con); - xfree(sql); - return NULL; - } - } - - if ( sql->mode == 'R' ) - { sql->nf = mpl_tab_num_flds(dca); - for(j = 0; sqllines[j] != NULL; j++) - arg = (char *) sqllines[j]; - total = strlen(arg); - if (total > 7 && 0 == strncmp(arg, "SELECT ", 7)) - { - total = strlen(arg); - query = xmalloc( (total+1) * sizeof(char)); - strcpy (query, arg); - } - else - { - query = db_generate_select_stmt(dca); - } - xprintf("%s\n", query); - if (dl_mysql_query(sql->con, query)) - { - xprintf("db_mysql_open: Query\n\"%s\"\nfailed.\n", query); - xprintf("%s\n",dl_mysql_error(sql->con)); - dl_mysql_close(sql->con); - xfree(query); - xfree(sql); - return NULL; - } - xfree(query); - sql->res = dl_mysql_use_result(sql->con); - if (sql->res) - { - /* create references between query results and table fields*/ - total = dl_mysql_num_fields(sql->res); - if (total > SQL_FIELD_MAX) - { xprintf("db_mysql_open: Too many fields (> %d) in query.\n" - "\"%s\"\n", SQL_FIELD_MAX, query); - xprintf("%s\n",dl_mysql_error(sql->con)); - dl_mysql_close(sql->con); - xfree(query); - xfree(sql); - return NULL; - } - fields = dl_mysql_fetch_fields(sql->res); - for (i = 1; i <= total; i++) - { - for (j = sql->nf; j >= 1; j--) - { - if (strcmp(mpl_tab_get_name(dca, j), fields[i-1].name) - == 0) - break; - } - sql->ref[i] = j; - } - } - else - { - if(dl_mysql_field_count(sql->con) == 0) - { - xprintf("db_mysql_open: Query was not a SELECT\n\"%s\"\n", - query); - xprintf("%s\n",dl_mysql_error(sql->con)); - xfree(query); - xfree(sql); - return NULL; - } - else - { - xprintf("db_mysql_open: Query\n\"%s\"\nfailed.\n", query); - xprintf("%s\n",dl_mysql_error(sql->con)); - xfree(query); - xfree(sql); - return NULL; - } - } - } - else if ( sql->mode == 'W' ) - { for(j = 0; sqllines[j] != NULL; j++) - arg = (char *) sqllines[j]; - if ( NULL != strchr(arg, '?') ) - { - total = strlen(arg); - query = xmalloc( (total+1) * sizeof(char)); - strcpy (query, arg); - } - else - query = db_generate_insert_stmt(dca); - sql->query = query; - xprintf("%s\n", query); - } - return sql; -} - -int db_mysql_read(TABDCA *dca, void *link) -{ struct db_mysql *sql; - char buf[255+1]; - char **row; - unsigned long *lengths; - MYSQL_FIELD *fields; - double num; - int len; - unsigned long num_fields; - int i; - - sql = (struct db_mysql *) link; - - xassert(sql != NULL); - xassert(sql->mode == 'R'); - if (NULL == sql->res) - { - xprintf("db_mysql_read: no result set available"); - return 1; - } - if (NULL==(row = (char **)dl_mysql_fetch_row(sql->res))) { - return -1; /*EOF*/ - } - lengths = dl_mysql_fetch_lengths(sql->res); - fields = dl_mysql_fetch_fields(sql->res); - num_fields = dl_mysql_num_fields(sql->res); - for (i=1; i <= num_fields; i++) - { - if (row[i-1] != NULL) - { len = (size_t) lengths[i-1]; - if (len > 255) - len = 255; - strncpy(buf, (const char *) row[i-1], len); - buf[len] = 0x00; - if (0 != (fields[i-1].flags & NUM_FLAG)) - { strspx(buf); /* remove spaces*/ - if (str2num(buf, &num) != 0) - { xprintf("'%s' cannot be converted to a number.\n", buf); - return 1; - } - if (sql->ref[i] > 0) - mpl_tab_set_num(dca, sql->ref[i], num); - } - else - { if (sql->ref[i] > 0) - mpl_tab_set_str(dca, sql->ref[i], strtrim(buf)); - } - } - } - return 0; -} - -int db_mysql_write(TABDCA *dca, void *link) -{ - struct db_mysql *sql; - char *part; - char *query; - char *template; - char num[50]; - int k; - int len; - int nf; - - sql = (struct db_mysql *) link; - xassert(sql != NULL); - xassert(sql->mode == 'W'); - - len = strlen(sql->query); - template = (char *) xmalloc( (len + 1) * sizeof(char) ); - strcpy(template, sql->query); - - nf = mpl_tab_num_flds(dca); - for (k = 1; k <= nf; k++) - { switch (mpl_tab_get_type(dca, k)) - { case 'N': - len += 20; - break; - case 'S': - len += db_escaped_string_length(mpl_tab_get_str(dca, k)); - len += 2; - break; - default: - xassert(dca != dca); - } - } - query = xmalloc( (len + 1 ) * sizeof(char) ); - query[0] = 0x00; -#if 0 /* 29/I-2017 */ - for (k = 1, part = strtok (template, "?"); (part != NULL); - part = strtok (NULL, "?"), k++) -#else - for (k = 1, part = xstrtok (template, "?"); (part != NULL); - part = xstrtok (NULL, "?"), k++) -#endif - { - if (k > nf) break; - strcat( query, part ); - switch (mpl_tab_get_type(dca, k)) - { case 'N': -#if 0 /* 02/XI-2010 by xypron */ - sprintf(num, "%-18g",mpl_tab_get_num(dca, k)); -#else - sprintf(num, "%.*g", DBL_DIG, mpl_tab_get_num(dca, k)); -#endif - strcat( query, num ); - break; - case 'S': - strcat( query, "'"); - db_escape_string( query + strlen(query), - mpl_tab_get_str(dca, k) ); - strcat( query, "'"); - break; - default: - xassert(dca != dca); - } - } - if (part != NULL) - strcat(query, part); - if (dl_mysql_query(sql->con, query)) - { - xprintf("db_mysql_write: Query\n\"%s\"\nfailed.\n", query); - xprintf("%s\n",dl_mysql_error(sql->con)); - xfree(query); - xfree(template); - return 1; - } - - xfree(query); - xfree(template); - return 0; - } - -int db_mysql_close(TABDCA *dca, void *link) -{ - struct db_mysql *sql; - - sql = (struct db_mysql *) link; - xassert(sql != NULL); - dl_mysql_close(sql->con); - if ( sql->mode == 'W' ) - xfree(sql->query); - xfree(sql); - dca->link = NULL; - return 0; -} - -#endif - -/* eof */ diff --git a/code/3rd_glpk/mpl/mplsql.h b/code/3rd_glpk/mpl/mplsql.h deleted file mode 100644 index 11d438bb..00000000 --- a/code/3rd_glpk/mpl/mplsql.h +++ /dev/null @@ -1,63 +0,0 @@ -/* mplsql.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Author: Heinrich Schuchardt . -* -* Copyright (C) 2003-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef MPLSQL_H -#define MPLSQL_H - -#define db_iodbc_open _glp_db_iodbc_open -void *db_iodbc_open(TABDCA *dca, int mode); -/* open iODBC database connection */ - -#define db_iodbc_read _glp_db_iodbc_read -int db_iodbc_read(TABDCA *dca, void *link); -/* read data from iODBC */ - -#define db_iodbc_write _glp_db_iodbc_write -int db_iodbc_write(TABDCA *dca, void *link); -/* write data to iODBC */ - -#define db_iodbc_close _glp_db_iodbc_close -int db_iodbc_close(TABDCA *dca, void *link); -/* close iODBC database connection */ - -#define db_mysql_open _glp_db_mysql_open -void *db_mysql_open(TABDCA *dca, int mode); -/* open MySQL database connection */ - -#define db_mysql_read _glp_db_mysql_read -int db_mysql_read(TABDCA *dca, void *link); -/* read data from MySQL */ - -#define db_mysql_write _glp_db_mysql_write -int db_mysql_write(TABDCA *dca, void *link); -/* write data to MySQL */ - -#define db_mysql_close _glp_db_mysql_close -int db_mysql_close(TABDCA *dca, void *link); -/* close MySQL database connection */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/npp/npp.h b/code/3rd_glpk/npp/npp.h deleted file mode 100644 index 428cb23c..00000000 --- a/code/3rd_glpk/npp/npp.h +++ /dev/null @@ -1,645 +0,0 @@ -/* npp.h (LP/MIP preprocessor) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef NPP_H -#define NPP_H - -#include "prob.h" - -#if 0 /* 20/XI-2017 */ -typedef struct NPP NPP; -#else -typedef struct glp_prep NPP; -#endif -typedef struct NPPROW NPPROW; -typedef struct NPPCOL NPPCOL; -typedef struct NPPAIJ NPPAIJ; -typedef struct NPPTSE NPPTSE; -typedef struct NPPLFE NPPLFE; - -#if 0 /* 20/XI-2017 */ -struct NPP -#else -struct glp_prep -#endif -{ /* LP/MIP preprocessor workspace */ - /*--------------------------------------------------------------*/ - /* original problem segment */ - int orig_dir; - /* optimization direction flag: - GLP_MIN - minimization - GLP_MAX - maximization */ - int orig_m; - /* number of rows */ - int orig_n; - /* number of columns */ - int orig_nnz; - /* number of non-zero constraint coefficients */ - /*--------------------------------------------------------------*/ - /* transformed problem segment (always minimization) */ - DMP *pool; - /* memory pool to store problem components */ - char *name; - /* problem name (1 to 255 chars); NULL means no name is assigned - to the problem */ - char *obj; - /* objective function name (1 to 255 chars); NULL means no name - is assigned to the objective function */ - double c0; - /* constant term of the objective function */ - int nrows; - /* number of rows introduced into the problem; this count - increases by one every time a new row is added and never - decreases; thus, actual number of rows may be less than nrows - due to row deletions */ - int ncols; - /* number of columns introduced into the problem; this count - increases by one every time a new column is added and never - decreases; thus, actual number of column may be less than - ncols due to column deletions */ - NPPROW *r_head; - /* pointer to the beginning of the row list */ - NPPROW *r_tail; - /* pointer to the end of the row list */ - NPPCOL *c_head; - /* pointer to the beginning of the column list */ - NPPCOL *c_tail; - /* pointer to the end of the column list */ - /*--------------------------------------------------------------*/ - /* transformation history */ - DMP *stack; - /* memory pool to store transformation entries */ - NPPTSE *top; - /* pointer to most recent transformation entry */ -#if 0 /* 16/XII-2009 */ - int count[1+25]; - /* transformation statistics */ -#endif - /*--------------------------------------------------------------*/ - /* resultant (preprocessed) problem segment */ - int m; - /* number of rows */ - int n; - /* number of columns */ - int nnz; - /* number of non-zero constraint coefficients */ - int *row_ref; /* int row_ref[1+m]; */ - /* row_ref[i], 1 <= i <= m, is the reference number assigned to - a row, which is i-th row of the resultant problem */ - int *col_ref; /* int col_ref[1+n]; */ - /* col_ref[j], 1 <= j <= n, is the reference number assigned to - a column, which is j-th column of the resultant problem */ - /*--------------------------------------------------------------*/ - /* recovered solution segment */ - int sol; - /* solution indicator: - GLP_SOL - basic solution - GLP_IPT - interior-point solution - GLP_MIP - mixed integer solution */ - int scaling; - /* scaling option: - GLP_OFF - scaling is disabled - GLP_ON - scaling is enabled */ - int p_stat; - /* status of primal basic solution: - GLP_UNDEF - primal solution is undefined - GLP_FEAS - primal solution is feasible - GLP_INFEAS - primal solution is infeasible - GLP_NOFEAS - no primal feasible solution exists */ - int d_stat; - /* status of dual basic solution: - GLP_UNDEF - dual solution is undefined - GLP_FEAS - dual solution is feasible - GLP_INFEAS - dual solution is infeasible - GLP_NOFEAS - no dual feasible solution exists */ - int t_stat; - /* status of interior-point solution: - GLP_UNDEF - interior solution is undefined - GLP_OPT - interior solution is optimal */ - int i_stat; - /* status of mixed integer solution: - GLP_UNDEF - integer solution is undefined - GLP_OPT - integer solution is optimal - GLP_FEAS - integer solution is feasible - GLP_NOFEAS - no integer solution exists */ - char *r_stat; /* char r_stat[1+nrows]; */ - /* r_stat[i], 1 <= i <= nrows, is status of i-th row: - GLP_BS - inactive constraint - GLP_NL - active constraint on lower bound - GLP_NU - active constraint on upper bound - GLP_NF - active free row - GLP_NS - active equality constraint */ - char *c_stat; /* char c_stat[1+nrows]; */ - /* c_stat[j], 1 <= j <= nrows, is status of j-th column: - GLP_BS - basic variable - GLP_NL - non-basic variable on lower bound - GLP_NU - non-basic variable on upper bound - GLP_NF - non-basic free variable - GLP_NS - non-basic fixed variable */ - double *r_pi; /* double r_pi[1+nrows]; */ - /* r_pi[i], 1 <= i <= nrows, is Lagrange multiplier (dual value) - for i-th row (constraint) */ - double *c_value; /* double c_value[1+ncols]; */ - /* c_value[j], 1 <= j <= ncols, is primal value of j-th column - (structural variable) */ -}; - -struct NPPROW -{ /* row (constraint) */ - int i; - /* reference number assigned to the row, 1 <= i <= nrows */ - char *name; - /* row name (1 to 255 chars); NULL means no name is assigned to - the row */ - double lb; - /* lower bound; -DBL_MAX means the row has no lower bound */ - double ub; - /* upper bound; +DBL_MAX means the row has no upper bound */ - NPPAIJ *ptr; - /* pointer to the linked list of constraint coefficients */ - int temp; - /* working field used by preprocessor routines */ - NPPROW *prev; - /* pointer to previous row in the row list */ - NPPROW *next; - /* pointer to next row in the row list */ -}; - -struct NPPCOL -{ /* column (variable) */ - int j; - /* reference number assigned to the column, 1 <= j <= ncols */ - char *name; - /* column name (1 to 255 chars); NULL means no name is assigned - to the column */ - char is_int; - /* 0 means continuous variable; 1 means integer variable */ - double lb; - /* lower bound; -DBL_MAX means the column has no lower bound */ - double ub; - /* upper bound; +DBL_MAX means the column has no upper bound */ - double coef; - /* objective coefficient */ - NPPAIJ *ptr; - /* pointer to the linked list of constraint coefficients */ - int temp; - /* working field used by preprocessor routines */ -#if 1 /* 28/XII-2009 */ - union - { double ll; - /* implied column lower bound */ - int pos; - /* vertex ordinal number corresponding to this binary column - in the conflict graph (0, if the vertex does not exist) */ - } ll; - union - { double uu; - /* implied column upper bound */ - int neg; - /* vertex ordinal number corresponding to complement of this - binary column in the conflict graph (0, if the vertex does - not exist) */ - } uu; -#endif - NPPCOL *prev; - /* pointer to previous column in the column list */ - NPPCOL *next; - /* pointer to next column in the column list */ -}; - -struct NPPAIJ -{ /* constraint coefficient */ - NPPROW *row; - /* pointer to corresponding row */ - NPPCOL *col; - /* pointer to corresponding column */ - double val; - /* (non-zero) coefficient value */ - NPPAIJ *r_prev; - /* pointer to previous coefficient in the same row */ - NPPAIJ *r_next; - /* pointer to next coefficient in the same row */ - NPPAIJ *c_prev; - /* pointer to previous coefficient in the same column */ - NPPAIJ *c_next; - /* pointer to next coefficient in the same column */ -}; - -struct NPPTSE -{ /* transformation stack entry */ - int (*func)(NPP *npp, void *info); - /* pointer to routine performing back transformation */ - void *info; - /* pointer to specific info (depends on the transformation) */ - NPPTSE *link; - /* pointer to another entry created *before* this entry */ -}; - -struct NPPLFE -{ /* linear form element */ - int ref; - /* row/column reference number */ - double val; - /* (non-zero) coefficient value */ - NPPLFE *next; - /* pointer to another element */ -}; - -#define npp_create_wksp _glp_npp_create_wksp -NPP *npp_create_wksp(void); -/* create LP/MIP preprocessor workspace */ - -#define npp_insert_row _glp_npp_insert_row -void npp_insert_row(NPP *npp, NPPROW *row, int where); -/* insert row to the row list */ - -#define npp_remove_row _glp_npp_remove_row -void npp_remove_row(NPP *npp, NPPROW *row); -/* remove row from the row list */ - -#define npp_activate_row _glp_npp_activate_row -void npp_activate_row(NPP *npp, NPPROW *row); -/* make row active */ - -#define npp_deactivate_row _glp_npp_deactivate_row -void npp_deactivate_row(NPP *npp, NPPROW *row); -/* make row inactive */ - -#define npp_insert_col _glp_npp_insert_col -void npp_insert_col(NPP *npp, NPPCOL *col, int where); -/* insert column to the column list */ - -#define npp_remove_col _glp_npp_remove_col -void npp_remove_col(NPP *npp, NPPCOL *col); -/* remove column from the column list */ - -#define npp_activate_col _glp_npp_activate_col -void npp_activate_col(NPP *npp, NPPCOL *col); -/* make column active */ - -#define npp_deactivate_col _glp_npp_deactivate_col -void npp_deactivate_col(NPP *npp, NPPCOL *col); -/* make column inactive */ - -#define npp_add_row _glp_npp_add_row -NPPROW *npp_add_row(NPP *npp); -/* add new row to the current problem */ - -#define npp_add_col _glp_npp_add_col -NPPCOL *npp_add_col(NPP *npp); -/* add new column to the current problem */ - -#define npp_add_aij _glp_npp_add_aij -NPPAIJ *npp_add_aij(NPP *npp, NPPROW *row, NPPCOL *col, double val); -/* add new element to the constraint matrix */ - -#define npp_row_nnz _glp_npp_row_nnz -int npp_row_nnz(NPP *npp, NPPROW *row); -/* count number of non-zero coefficients in row */ - -#define npp_col_nnz _glp_npp_col_nnz -int npp_col_nnz(NPP *npp, NPPCOL *col); -/* count number of non-zero coefficients in column */ - -#define npp_push_tse _glp_npp_push_tse -void *npp_push_tse(NPP *npp, int (*func)(NPP *npp, void *info), - int size); -/* push new entry to the transformation stack */ - -#define npp_erase_row _glp_npp_erase_row -void npp_erase_row(NPP *npp, NPPROW *row); -/* erase row content to make it empty */ - -#define npp_del_row _glp_npp_del_row -void npp_del_row(NPP *npp, NPPROW *row); -/* remove row from the current problem */ - -#define npp_del_col _glp_npp_del_col -void npp_del_col(NPP *npp, NPPCOL *col); -/* remove column from the current problem */ - -#define npp_del_aij _glp_npp_del_aij -void npp_del_aij(NPP *npp, NPPAIJ *aij); -/* remove element from the constraint matrix */ - -#define npp_load_prob _glp_npp_load_prob -void npp_load_prob(NPP *npp, glp_prob *orig, int names, int sol, - int scaling); -/* load original problem into the preprocessor workspace */ - -#define npp_build_prob _glp_npp_build_prob -void npp_build_prob(NPP *npp, glp_prob *prob); -/* build resultant (preprocessed) problem */ - -#define npp_postprocess _glp_npp_postprocess -void npp_postprocess(NPP *npp, glp_prob *prob); -/* postprocess solution from the resultant problem */ - -#define npp_unload_sol _glp_npp_unload_sol -void npp_unload_sol(NPP *npp, glp_prob *orig); -/* store solution to the original problem */ - -#define npp_delete_wksp _glp_npp_delete_wksp -void npp_delete_wksp(NPP *npp); -/* delete LP/MIP preprocessor workspace */ - -#define npp_error() - -#define npp_free_row _glp_npp_free_row -void npp_free_row(NPP *npp, NPPROW *p); -/* process free (unbounded) row */ - -#define npp_geq_row _glp_npp_geq_row -void npp_geq_row(NPP *npp, NPPROW *p); -/* process row of 'not less than' type */ - -#define npp_leq_row _glp_npp_leq_row -void npp_leq_row(NPP *npp, NPPROW *p); -/* process row of 'not greater than' type */ - -#define npp_free_col _glp_npp_free_col -void npp_free_col(NPP *npp, NPPCOL *q); -/* process free (unbounded) column */ - -#define npp_lbnd_col _glp_npp_lbnd_col -void npp_lbnd_col(NPP *npp, NPPCOL *q); -/* process column with (non-zero) lower bound */ - -#define npp_ubnd_col _glp_npp_ubnd_col -void npp_ubnd_col(NPP *npp, NPPCOL *q); -/* process column with upper bound */ - -#define npp_dbnd_col _glp_npp_dbnd_col -void npp_dbnd_col(NPP *npp, NPPCOL *q); -/* process non-negative column with upper bound */ - -#define npp_fixed_col _glp_npp_fixed_col -void npp_fixed_col(NPP *npp, NPPCOL *q); -/* process fixed column */ - -#define npp_make_equality _glp_npp_make_equality -int npp_make_equality(NPP *npp, NPPROW *p); -/* process row with almost identical bounds */ - -#define npp_make_fixed _glp_npp_make_fixed -int npp_make_fixed(NPP *npp, NPPCOL *q); -/* process column with almost identical bounds */ - -#define npp_empty_row _glp_npp_empty_row -int npp_empty_row(NPP *npp, NPPROW *p); -/* process empty row */ - -#define npp_empty_col _glp_npp_empty_col -int npp_empty_col(NPP *npp, NPPCOL *q); -/* process empty column */ - -#define npp_implied_value _glp_npp_implied_value -int npp_implied_value(NPP *npp, NPPCOL *q, double s); -/* process implied column value */ - -#define npp_eq_singlet _glp_npp_eq_singlet -int npp_eq_singlet(NPP *npp, NPPROW *p); -/* process row singleton (equality constraint) */ - -#define npp_implied_lower _glp_npp_implied_lower -int npp_implied_lower(NPP *npp, NPPCOL *q, double l); -/* process implied column lower bound */ - -#define npp_implied_upper _glp_npp_implied_upper -int npp_implied_upper(NPP *npp, NPPCOL *q, double u); -/* process implied upper bound of column */ - -#define npp_ineq_singlet _glp_npp_ineq_singlet -int npp_ineq_singlet(NPP *npp, NPPROW *p); -/* process row singleton (inequality constraint) */ - -#define npp_implied_slack _glp_npp_implied_slack -void npp_implied_slack(NPP *npp, NPPCOL *q); -/* process column singleton (implied slack variable) */ - -#define npp_implied_free _glp_npp_implied_free -int npp_implied_free(NPP *npp, NPPCOL *q); -/* process column singleton (implied free variable) */ - -#define npp_eq_doublet _glp_npp_eq_doublet -NPPCOL *npp_eq_doublet(NPP *npp, NPPROW *p); -/* process row doubleton (equality constraint) */ - -#define npp_forcing_row _glp_npp_forcing_row -int npp_forcing_row(NPP *npp, NPPROW *p, int at); -/* process forcing row */ - -#define npp_analyze_row _glp_npp_analyze_row -int npp_analyze_row(NPP *npp, NPPROW *p); -/* perform general row analysis */ - -#define npp_inactive_bound _glp_npp_inactive_bound -void npp_inactive_bound(NPP *npp, NPPROW *p, int which); -/* remove row lower/upper inactive bound */ - -#define npp_implied_bounds _glp_npp_implied_bounds -void npp_implied_bounds(NPP *npp, NPPROW *p); -/* determine implied column bounds */ - -#define npp_binarize_prob _glp_npp_binarize_prob -int npp_binarize_prob(NPP *npp); -/* binarize MIP problem */ - -#define npp_is_packing _glp_npp_is_packing -int npp_is_packing(NPP *npp, NPPROW *row); -/* test if constraint is packing inequality */ - -#define npp_hidden_packing _glp_npp_hidden_packing -int npp_hidden_packing(NPP *npp, NPPROW *row); -/* identify hidden packing inequality */ - -#define npp_implied_packing _glp_npp_implied_packing -int npp_implied_packing(NPP *npp, NPPROW *row, int which, - NPPCOL *var[], char set[]); -/* identify implied packing inequality */ - -#define npp_is_covering _glp_npp_is_covering -int npp_is_covering(NPP *npp, NPPROW *row); -/* test if constraint is covering inequality */ - -#define npp_hidden_covering _glp_npp_hidden_covering -int npp_hidden_covering(NPP *npp, NPPROW *row); -/* identify hidden covering inequality */ - -#define npp_is_partitioning _glp_npp_is_partitioning -int npp_is_partitioning(NPP *npp, NPPROW *row); -/* test if constraint is partitioning equality */ - -#define npp_reduce_ineq_coef _glp_npp_reduce_ineq_coef -int npp_reduce_ineq_coef(NPP *npp, NPPROW *row); -/* reduce inequality constraint coefficients */ - -#define npp_clean_prob _glp_npp_clean_prob -void npp_clean_prob(NPP *npp); -/* perform initial LP/MIP processing */ - -#define npp_process_row _glp_npp_process_row -int npp_process_row(NPP *npp, NPPROW *row, int hard); -/* perform basic row processing */ - -#define npp_improve_bounds _glp_npp_improve_bounds -int npp_improve_bounds(NPP *npp, NPPROW *row, int flag); -/* improve current column bounds */ - -#define npp_process_col _glp_npp_process_col -int npp_process_col(NPP *npp, NPPCOL *col); -/* perform basic column processing */ - -#define npp_process_prob _glp_npp_process_prob -int npp_process_prob(NPP *npp, int hard); -/* perform basic LP/MIP processing */ - -#define npp_simplex _glp_npp_simplex -int npp_simplex(NPP *npp, const glp_smcp *parm); -/* process LP prior to applying primal/dual simplex method */ - -#define npp_integer _glp_npp_integer -int npp_integer(NPP *npp, const glp_iocp *parm); -/* process MIP prior to applying branch-and-bound method */ - -/**********************************************************************/ - -#define npp_sat_free_row _glp_npp_sat_free_row -void npp_sat_free_row(NPP *npp, NPPROW *p); -/* process free (unbounded) row */ - -#define npp_sat_fixed_col _glp_npp_sat_fixed_col -int npp_sat_fixed_col(NPP *npp, NPPCOL *q); -/* process fixed column */ - -#define npp_sat_is_bin_comb _glp_npp_sat_is_bin_comb -int npp_sat_is_bin_comb(NPP *npp, NPPROW *row); -/* test if row is binary combination */ - -#define npp_sat_num_pos_coef _glp_npp_sat_num_pos_coef -int npp_sat_num_pos_coef(NPP *npp, NPPROW *row); -/* determine number of positive coefficients */ - -#define npp_sat_num_neg_coef _glp_npp_sat_num_neg_coef -int npp_sat_num_neg_coef(NPP *npp, NPPROW *row); -/* determine number of negative coefficients */ - -#define npp_sat_is_cover_ineq _glp_npp_sat_is_cover_ineq -int npp_sat_is_cover_ineq(NPP *npp, NPPROW *row); -/* test if row is covering inequality */ - -#define npp_sat_is_pack_ineq _glp_npp_sat_is_pack_ineq -int npp_sat_is_pack_ineq(NPP *npp, NPPROW *row); -/* test if row is packing inequality */ - -#define npp_sat_is_partn_eq _glp_npp_sat_is_partn_eq -int npp_sat_is_partn_eq(NPP *npp, NPPROW *row); -/* test if row is partitioning equality */ - -#define npp_sat_reverse_row _glp_npp_sat_reverse_row -int npp_sat_reverse_row(NPP *npp, NPPROW *row); -/* multiply both sides of row by -1 */ - -#define npp_sat_split_pack _glp_npp_sat_split_pack -NPPROW *npp_sat_split_pack(NPP *npp, NPPROW *row, int nnn); -/* split packing inequality */ - -#define npp_sat_encode_pack _glp_npp_sat_encode_pack -void npp_sat_encode_pack(NPP *npp, NPPROW *row); -/* encode packing inequality */ - -typedef struct NPPLIT NPPLIT; -typedef struct NPPLSE NPPLSE; -typedef struct NPPSED NPPSED; - -struct NPPLIT -{ /* literal (binary variable or its negation) */ - NPPCOL *col; - /* pointer to binary variable; NULL means constant false */ - int neg; - /* negation flag: - 0 - literal is variable (or constant false) - 1 - literal is negation of variable (or constant true) */ -}; - -struct NPPLSE -{ /* literal set element */ - NPPLIT lit; - /* literal */ - NPPLSE *next; - /* pointer to another element */ -}; - -struct NPPSED -{ /* summation encoding descriptor */ - /* this struct describes the equality - x + y + z = s + 2 * c, - which was encoded as CNF and included into the transformed - problem; here x and y are literals, z is either a literal or - constant zero, s and c are binary variables modeling, resp., - the low and high (carry) sum bits */ - NPPLIT x, y, z; - /* literals; if z.col = NULL, z is constant zero */ - NPPCOL *s, *c; - /* binary variables modeling the sum bits */ -}; - -#define npp_sat_encode_sum2 _glp_npp_sat_encode_sum2 -void npp_sat_encode_sum2(NPP *npp, NPPLSE *set, NPPSED *sed); -/* encode 2-bit summation */ - -#define npp_sat_encode_sum3 _glp_npp_sat_encode_sum3 -void npp_sat_encode_sum3(NPP *npp, NPPLSE *set, NPPSED *sed); -/* encode 3-bit summation */ - -#define npp_sat_encode_sum_ax _glp_npp_sat_encode_sum_ax -int npp_sat_encode_sum_ax(NPP *npp, NPPROW *row, NPPLIT y[]); -/* encode linear combination of 0-1 variables */ - -#define npp_sat_normalize_clause _glp_npp_sat_normalize_clause -int npp_sat_normalize_clause(NPP *npp, int size, NPPLIT lit[]); -/* normalize clause */ - -#define npp_sat_encode_clause _glp_npp_sat_encode_clause -NPPROW *npp_sat_encode_clause(NPP *npp, int size, NPPLIT lit[]); -/* translate clause to cover inequality */ - -#define npp_sat_encode_geq _glp_npp_sat_encode_geq -int npp_sat_encode_geq(NPP *npp, int n, NPPLIT y[], int rhs); -/* encode "not less than" constraint */ - -#define npp_sat_encode_leq _glp_npp_sat_encode_leq -int npp_sat_encode_leq(NPP *npp, int n, NPPLIT y[], int rhs); -/* encode "not greater than" constraint */ - -#define npp_sat_encode_row _glp_npp_sat_encode_row -int npp_sat_encode_row(NPP *npp, NPPROW *row); -/* encode constraint (row) of general type */ - -#define npp_sat_encode_prob _glp_npp_sat_encode_prob -int npp_sat_encode_prob(NPP *npp); -/* encode 0-1 feasibility problem */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/npp/npp1.c b/code/3rd_glpk/npp/npp1.c deleted file mode 100644 index 51758bad..00000000 --- a/code/3rd_glpk/npp/npp1.c +++ /dev/null @@ -1,937 +0,0 @@ -/* npp1.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -NPP *npp_create_wksp(void) -{ /* create LP/MIP preprocessor workspace */ - NPP *npp; - npp = xmalloc(sizeof(NPP)); - npp->orig_dir = 0; - npp->orig_m = npp->orig_n = npp->orig_nnz = 0; - npp->pool = dmp_create_pool(); - npp->name = npp->obj = NULL; - npp->c0 = 0.0; - npp->nrows = npp->ncols = 0; - npp->r_head = npp->r_tail = NULL; - npp->c_head = npp->c_tail = NULL; - npp->stack = dmp_create_pool(); - npp->top = NULL; -#if 0 /* 16/XII-2009 */ - memset(&npp->count, 0, sizeof(npp->count)); -#endif - npp->m = npp->n = npp->nnz = 0; - npp->row_ref = npp->col_ref = NULL; - npp->sol = npp->scaling = 0; - npp->p_stat = npp->d_stat = npp->t_stat = npp->i_stat = 0; - npp->r_stat = NULL; - /*npp->r_prim =*/ npp->r_pi = NULL; - npp->c_stat = NULL; - npp->c_value = /*npp->c_dual =*/ NULL; - return npp; -} - -void npp_insert_row(NPP *npp, NPPROW *row, int where) -{ /* insert row to the row list */ - if (where == 0) - { /* insert row to the beginning of the row list */ - row->prev = NULL; - row->next = npp->r_head; - if (row->next == NULL) - npp->r_tail = row; - else - row->next->prev = row; - npp->r_head = row; - } - else - { /* insert row to the end of the row list */ - row->prev = npp->r_tail; - row->next = NULL; - if (row->prev == NULL) - npp->r_head = row; - else - row->prev->next = row; - npp->r_tail = row; - } - return; -} - -void npp_remove_row(NPP *npp, NPPROW *row) -{ /* remove row from the row list */ - if (row->prev == NULL) - npp->r_head = row->next; - else - row->prev->next = row->next; - if (row->next == NULL) - npp->r_tail = row->prev; - else - row->next->prev = row->prev; - return; -} - -void npp_activate_row(NPP *npp, NPPROW *row) -{ /* make row active */ - if (!row->temp) - { row->temp = 1; - /* move the row to the beginning of the row list */ - npp_remove_row(npp, row); - npp_insert_row(npp, row, 0); - } - return; -} - -void npp_deactivate_row(NPP *npp, NPPROW *row) -{ /* make row inactive */ - if (row->temp) - { row->temp = 0; - /* move the row to the end of the row list */ - npp_remove_row(npp, row); - npp_insert_row(npp, row, 1); - } - return; -} - -void npp_insert_col(NPP *npp, NPPCOL *col, int where) -{ /* insert column to the column list */ - if (where == 0) - { /* insert column to the beginning of the column list */ - col->prev = NULL; - col->next = npp->c_head; - if (col->next == NULL) - npp->c_tail = col; - else - col->next->prev = col; - npp->c_head = col; - } - else - { /* insert column to the end of the column list */ - col->prev = npp->c_tail; - col->next = NULL; - if (col->prev == NULL) - npp->c_head = col; - else - col->prev->next = col; - npp->c_tail = col; - } - return; -} - -void npp_remove_col(NPP *npp, NPPCOL *col) -{ /* remove column from the column list */ - if (col->prev == NULL) - npp->c_head = col->next; - else - col->prev->next = col->next; - if (col->next == NULL) - npp->c_tail = col->prev; - else - col->next->prev = col->prev; - return; -} - -void npp_activate_col(NPP *npp, NPPCOL *col) -{ /* make column active */ - if (!col->temp) - { col->temp = 1; - /* move the column to the beginning of the column list */ - npp_remove_col(npp, col); - npp_insert_col(npp, col, 0); - } - return; -} - -void npp_deactivate_col(NPP *npp, NPPCOL *col) -{ /* make column inactive */ - if (col->temp) - { col->temp = 0; - /* move the column to the end of the column list */ - npp_remove_col(npp, col); - npp_insert_col(npp, col, 1); - } - return; -} - -NPPROW *npp_add_row(NPP *npp) -{ /* add new row to the current problem */ - NPPROW *row; - row = dmp_get_atom(npp->pool, sizeof(NPPROW)); - row->i = ++(npp->nrows); - row->name = NULL; - row->lb = -DBL_MAX, row->ub = +DBL_MAX; - row->ptr = NULL; - row->temp = 0; - npp_insert_row(npp, row, 1); - return row; -} - -NPPCOL *npp_add_col(NPP *npp) -{ /* add new column to the current problem */ - NPPCOL *col; - col = dmp_get_atom(npp->pool, sizeof(NPPCOL)); - col->j = ++(npp->ncols); - col->name = NULL; -#if 0 - col->kind = GLP_CV; -#else - col->is_int = 0; -#endif - col->lb = col->ub = col->coef = 0.0; - col->ptr = NULL; - col->temp = 0; - npp_insert_col(npp, col, 1); - return col; -} - -NPPAIJ *npp_add_aij(NPP *npp, NPPROW *row, NPPCOL *col, double val) -{ /* add new element to the constraint matrix */ - NPPAIJ *aij; - aij = dmp_get_atom(npp->pool, sizeof(NPPAIJ)); - aij->row = row; - aij->col = col; - aij->val = val; - aij->r_prev = NULL; - aij->r_next = row->ptr; - aij->c_prev = NULL; - aij->c_next = col->ptr; - if (aij->r_next != NULL) - aij->r_next->r_prev = aij; - if (aij->c_next != NULL) - aij->c_next->c_prev = aij; - row->ptr = col->ptr = aij; - return aij; -} - -int npp_row_nnz(NPP *npp, NPPROW *row) -{ /* count number of non-zero coefficients in row */ - NPPAIJ *aij; - int nnz; - xassert(npp == npp); - nnz = 0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - nnz++; - return nnz; -} - -int npp_col_nnz(NPP *npp, NPPCOL *col) -{ /* count number of non-zero coefficients in column */ - NPPAIJ *aij; - int nnz; - xassert(npp == npp); - nnz = 0; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - nnz++; - return nnz; -} - -void *npp_push_tse(NPP *npp, int (*func)(NPP *npp, void *info), - int size) -{ /* push new entry to the transformation stack */ - NPPTSE *tse; - tse = dmp_get_atom(npp->stack, sizeof(NPPTSE)); - tse->func = func; - tse->info = dmp_get_atom(npp->stack, size); - tse->link = npp->top; - npp->top = tse; - return tse->info; -} - -#if 1 /* 23/XII-2009 */ -void npp_erase_row(NPP *npp, NPPROW *row) -{ /* erase row content to make it empty */ - NPPAIJ *aij; - while (row->ptr != NULL) - { aij = row->ptr; - row->ptr = aij->r_next; - if (aij->c_prev == NULL) - aij->col->ptr = aij->c_next; - else - aij->c_prev->c_next = aij->c_next; - if (aij->c_next == NULL) - ; - else - aij->c_next->c_prev = aij->c_prev; - dmp_free_atom(npp->pool, aij, sizeof(NPPAIJ)); - } - return; -} -#endif - -void npp_del_row(NPP *npp, NPPROW *row) -{ /* remove row from the current problem */ -#if 0 /* 23/XII-2009 */ - NPPAIJ *aij; -#endif - if (row->name != NULL) - dmp_free_atom(npp->pool, row->name, strlen(row->name)+1); -#if 0 /* 23/XII-2009 */ - while (row->ptr != NULL) - { aij = row->ptr; - row->ptr = aij->r_next; - if (aij->c_prev == NULL) - aij->col->ptr = aij->c_next; - else - aij->c_prev->c_next = aij->c_next; - if (aij->c_next == NULL) - ; - else - aij->c_next->c_prev = aij->c_prev; - dmp_free_atom(npp->pool, aij, sizeof(NPPAIJ)); - } -#else - npp_erase_row(npp, row); -#endif - npp_remove_row(npp, row); - dmp_free_atom(npp->pool, row, sizeof(NPPROW)); - return; -} - -void npp_del_col(NPP *npp, NPPCOL *col) -{ /* remove column from the current problem */ - NPPAIJ *aij; - if (col->name != NULL) - dmp_free_atom(npp->pool, col->name, strlen(col->name)+1); - while (col->ptr != NULL) - { aij = col->ptr; - col->ptr = aij->c_next; - if (aij->r_prev == NULL) - aij->row->ptr = aij->r_next; - else - aij->r_prev->r_next = aij->r_next; - if (aij->r_next == NULL) - ; - else - aij->r_next->r_prev = aij->r_prev; - dmp_free_atom(npp->pool, aij, sizeof(NPPAIJ)); - } - npp_remove_col(npp, col); - dmp_free_atom(npp->pool, col, sizeof(NPPCOL)); - return; -} - -void npp_del_aij(NPP *npp, NPPAIJ *aij) -{ /* remove element from the constraint matrix */ - if (aij->r_prev == NULL) - aij->row->ptr = aij->r_next; - else - aij->r_prev->r_next = aij->r_next; - if (aij->r_next == NULL) - ; - else - aij->r_next->r_prev = aij->r_prev; - if (aij->c_prev == NULL) - aij->col->ptr = aij->c_next; - else - aij->c_prev->c_next = aij->c_next; - if (aij->c_next == NULL) - ; - else - aij->c_next->c_prev = aij->c_prev; - dmp_free_atom(npp->pool, aij, sizeof(NPPAIJ)); - return; -} - -void npp_load_prob(NPP *npp, glp_prob *orig, int names, int sol, - int scaling) -{ /* load original problem into the preprocessor workspace */ - int m = orig->m; - int n = orig->n; - NPPROW **link; - int i, j; - double dir; - xassert(names == GLP_OFF || names == GLP_ON); - xassert(sol == GLP_SOL || sol == GLP_IPT || sol == GLP_MIP); - xassert(scaling == GLP_OFF || scaling == GLP_ON); - if (sol == GLP_MIP) xassert(!scaling); - npp->orig_dir = orig->dir; - if (npp->orig_dir == GLP_MIN) - dir = +1.0; - else if (npp->orig_dir == GLP_MAX) - dir = -1.0; - else - xassert(npp != npp); - npp->orig_m = m; - npp->orig_n = n; - npp->orig_nnz = orig->nnz; - if (names && orig->name != NULL) - { npp->name = dmp_get_atom(npp->pool, strlen(orig->name)+1); - strcpy(npp->name, orig->name); - } - if (names && orig->obj != NULL) - { npp->obj = dmp_get_atom(npp->pool, strlen(orig->obj)+1); - strcpy(npp->obj, orig->obj); - } - npp->c0 = dir * orig->c0; - /* load rows */ - link = xcalloc(1+m, sizeof(NPPROW *)); - for (i = 1; i <= m; i++) - { GLPROW *rrr = orig->row[i]; - NPPROW *row; - link[i] = row = npp_add_row(npp); - xassert(row->i == i); - if (names && rrr->name != NULL) - { row->name = dmp_get_atom(npp->pool, strlen(rrr->name)+1); - strcpy(row->name, rrr->name); - } - if (!scaling) - { if (rrr->type == GLP_FR) - row->lb = -DBL_MAX, row->ub = +DBL_MAX; - else if (rrr->type == GLP_LO) - row->lb = rrr->lb, row->ub = +DBL_MAX; - else if (rrr->type == GLP_UP) - row->lb = -DBL_MAX, row->ub = rrr->ub; - else if (rrr->type == GLP_DB) - row->lb = rrr->lb, row->ub = rrr->ub; - else if (rrr->type == GLP_FX) - row->lb = row->ub = rrr->lb; - else - xassert(rrr != rrr); - } - else - { double rii = rrr->rii; - if (rrr->type == GLP_FR) - row->lb = -DBL_MAX, row->ub = +DBL_MAX; - else if (rrr->type == GLP_LO) - row->lb = rrr->lb * rii, row->ub = +DBL_MAX; - else if (rrr->type == GLP_UP) - row->lb = -DBL_MAX, row->ub = rrr->ub * rii; - else if (rrr->type == GLP_DB) - row->lb = rrr->lb * rii, row->ub = rrr->ub * rii; - else if (rrr->type == GLP_FX) - row->lb = row->ub = rrr->lb * rii; - else - xassert(rrr != rrr); - } - } - /* load columns and constraint coefficients */ - for (j = 1; j <= n; j++) - { GLPCOL *ccc = orig->col[j]; - GLPAIJ *aaa; - NPPCOL *col; - col = npp_add_col(npp); - xassert(col->j == j); - if (names && ccc->name != NULL) - { col->name = dmp_get_atom(npp->pool, strlen(ccc->name)+1); - strcpy(col->name, ccc->name); - } - if (sol == GLP_MIP) -#if 0 - col->kind = ccc->kind; -#else - col->is_int = (char)(ccc->kind == GLP_IV); -#endif - if (!scaling) - { if (ccc->type == GLP_FR) - col->lb = -DBL_MAX, col->ub = +DBL_MAX; - else if (ccc->type == GLP_LO) - col->lb = ccc->lb, col->ub = +DBL_MAX; - else if (ccc->type == GLP_UP) - col->lb = -DBL_MAX, col->ub = ccc->ub; - else if (ccc->type == GLP_DB) - col->lb = ccc->lb, col->ub = ccc->ub; - else if (ccc->type == GLP_FX) - col->lb = col->ub = ccc->lb; - else - xassert(ccc != ccc); - col->coef = dir * ccc->coef; - for (aaa = ccc->ptr; aaa != NULL; aaa = aaa->c_next) - npp_add_aij(npp, link[aaa->row->i], col, aaa->val); - } - else - { double sjj = ccc->sjj; - if (ccc->type == GLP_FR) - col->lb = -DBL_MAX, col->ub = +DBL_MAX; - else if (ccc->type == GLP_LO) - col->lb = ccc->lb / sjj, col->ub = +DBL_MAX; - else if (ccc->type == GLP_UP) - col->lb = -DBL_MAX, col->ub = ccc->ub / sjj; - else if (ccc->type == GLP_DB) - col->lb = ccc->lb / sjj, col->ub = ccc->ub / sjj; - else if (ccc->type == GLP_FX) - col->lb = col->ub = ccc->lb / sjj; - else - xassert(ccc != ccc); - col->coef = dir * ccc->coef * sjj; - for (aaa = ccc->ptr; aaa != NULL; aaa = aaa->c_next) - npp_add_aij(npp, link[aaa->row->i], col, - aaa->row->rii * aaa->val * sjj); - } - } - xfree(link); - /* keep solution indicator and scaling option */ - npp->sol = sol; - npp->scaling = scaling; - return; -} - -void npp_build_prob(NPP *npp, glp_prob *prob) -{ /* build resultant (preprocessed) problem */ - NPPROW *row; - NPPCOL *col; - NPPAIJ *aij; - int i, j, type, len, *ind; - double dir, *val; - glp_erase_prob(prob); - glp_set_prob_name(prob, npp->name); - glp_set_obj_name(prob, npp->obj); - glp_set_obj_dir(prob, npp->orig_dir); - if (npp->orig_dir == GLP_MIN) - dir = +1.0; - else if (npp->orig_dir == GLP_MAX) - dir = -1.0; - else - xassert(npp != npp); - glp_set_obj_coef(prob, 0, dir * npp->c0); - /* build rows */ - for (row = npp->r_head; row != NULL; row = row->next) - { row->temp = i = glp_add_rows(prob, 1); - glp_set_row_name(prob, i, row->name); - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) - type = GLP_FR; - else if (row->ub == +DBL_MAX) - type = GLP_LO; - else if (row->lb == -DBL_MAX) - type = GLP_UP; - else if (row->lb != row->ub) - type = GLP_DB; - else - type = GLP_FX; - glp_set_row_bnds(prob, i, type, row->lb, row->ub); - } - /* build columns and the constraint matrix */ - ind = xcalloc(1+prob->m, sizeof(int)); - val = xcalloc(1+prob->m, sizeof(double)); - for (col = npp->c_head; col != NULL; col = col->next) - { j = glp_add_cols(prob, 1); - glp_set_col_name(prob, j, col->name); -#if 0 - glp_set_col_kind(prob, j, col->kind); -#else - glp_set_col_kind(prob, j, col->is_int ? GLP_IV : GLP_CV); -#endif - if (col->lb == -DBL_MAX && col->ub == +DBL_MAX) - type = GLP_FR; - else if (col->ub == +DBL_MAX) - type = GLP_LO; - else if (col->lb == -DBL_MAX) - type = GLP_UP; - else if (col->lb != col->ub) - type = GLP_DB; - else - type = GLP_FX; - glp_set_col_bnds(prob, j, type, col->lb, col->ub); - glp_set_obj_coef(prob, j, dir * col->coef); - len = 0; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - { len++; - ind[len] = aij->row->temp; - val[len] = aij->val; - } - glp_set_mat_col(prob, j, len, ind, val); - } - xfree(ind); - xfree(val); - /* resultant problem has been built */ - npp->m = prob->m; - npp->n = prob->n; - npp->nnz = prob->nnz; - npp->row_ref = xcalloc(1+npp->m, sizeof(int)); - npp->col_ref = xcalloc(1+npp->n, sizeof(int)); - for (row = npp->r_head, i = 0; row != NULL; row = row->next) - npp->row_ref[++i] = row->i; - for (col = npp->c_head, j = 0; col != NULL; col = col->next) - npp->col_ref[++j] = col->j; - /* transformed problem segment is no longer needed */ - dmp_delete_pool(npp->pool), npp->pool = NULL; - npp->name = npp->obj = NULL; - npp->c0 = 0.0; - npp->r_head = npp->r_tail = NULL; - npp->c_head = npp->c_tail = NULL; - return; -} - -void npp_postprocess(NPP *npp, glp_prob *prob) -{ /* postprocess solution from the resultant problem */ - GLPROW *row; - GLPCOL *col; - NPPTSE *tse; - int i, j, k; - double dir; - xassert(npp->orig_dir == prob->dir); - if (npp->orig_dir == GLP_MIN) - dir = +1.0; - else if (npp->orig_dir == GLP_MAX) - dir = -1.0; - else - xassert(npp != npp); -#if 0 /* 11/VII-2013; due to call from ios_main */ - xassert(npp->m == prob->m); -#else - if (npp->sol != GLP_MIP) - xassert(npp->m == prob->m); -#endif - xassert(npp->n == prob->n); -#if 0 /* 11/VII-2013; due to call from ios_main */ - xassert(npp->nnz == prob->nnz); -#else - if (npp->sol != GLP_MIP) - xassert(npp->nnz == prob->nnz); -#endif - /* copy solution status */ - if (npp->sol == GLP_SOL) - { npp->p_stat = prob->pbs_stat; - npp->d_stat = prob->dbs_stat; - } - else if (npp->sol == GLP_IPT) - npp->t_stat = prob->ipt_stat; - else if (npp->sol == GLP_MIP) - npp->i_stat = prob->mip_stat; - else - xassert(npp != npp); - /* allocate solution arrays */ - if (npp->sol == GLP_SOL) - { if (npp->r_stat == NULL) - npp->r_stat = xcalloc(1+npp->nrows, sizeof(char)); - for (i = 1; i <= npp->nrows; i++) - npp->r_stat[i] = 0; - if (npp->c_stat == NULL) - npp->c_stat = xcalloc(1+npp->ncols, sizeof(char)); - for (j = 1; j <= npp->ncols; j++) - npp->c_stat[j] = 0; - } -#if 0 - if (npp->r_prim == NULL) - npp->r_prim = xcalloc(1+npp->nrows, sizeof(double)); - for (i = 1; i <= npp->nrows; i++) - npp->r_prim[i] = DBL_MAX; -#endif - if (npp->c_value == NULL) - npp->c_value = xcalloc(1+npp->ncols, sizeof(double)); - for (j = 1; j <= npp->ncols; j++) - npp->c_value[j] = DBL_MAX; - if (npp->sol != GLP_MIP) - { if (npp->r_pi == NULL) - npp->r_pi = xcalloc(1+npp->nrows, sizeof(double)); - for (i = 1; i <= npp->nrows; i++) - npp->r_pi[i] = DBL_MAX; -#if 0 - if (npp->c_dual == NULL) - npp->c_dual = xcalloc(1+npp->ncols, sizeof(double)); - for (j = 1; j <= npp->ncols; j++) - npp->c_dual[j] = DBL_MAX; -#endif - } - /* copy solution components from the resultant problem */ - if (npp->sol == GLP_SOL) - { for (i = 1; i <= npp->m; i++) - { row = prob->row[i]; - k = npp->row_ref[i]; - npp->r_stat[k] = (char)row->stat; - /*npp->r_prim[k] = row->prim;*/ - npp->r_pi[k] = dir * row->dual; - } - for (j = 1; j <= npp->n; j++) - { col = prob->col[j]; - k = npp->col_ref[j]; - npp->c_stat[k] = (char)col->stat; - npp->c_value[k] = col->prim; - /*npp->c_dual[k] = dir * col->dual;*/ - } - } - else if (npp->sol == GLP_IPT) - { for (i = 1; i <= npp->m; i++) - { row = prob->row[i]; - k = npp->row_ref[i]; - /*npp->r_prim[k] = row->pval;*/ - npp->r_pi[k] = dir * row->dval; - } - for (j = 1; j <= npp->n; j++) - { col = prob->col[j]; - k = npp->col_ref[j]; - npp->c_value[k] = col->pval; - /*npp->c_dual[k] = dir * col->dval;*/ - } - } - else if (npp->sol == GLP_MIP) - { -#if 0 - for (i = 1; i <= npp->m; i++) - { row = prob->row[i]; - k = npp->row_ref[i]; - /*npp->r_prim[k] = row->mipx;*/ - } -#endif - for (j = 1; j <= npp->n; j++) - { col = prob->col[j]; - k = npp->col_ref[j]; - npp->c_value[k] = col->mipx; - } - } - else - xassert(npp != npp); - /* perform postprocessing to construct solution to the original - problem */ - for (tse = npp->top; tse != NULL; tse = tse->link) - { xassert(tse->func != NULL); - xassert(tse->func(npp, tse->info) == 0); - } - return; -} - -void npp_unload_sol(NPP *npp, glp_prob *orig) -{ /* store solution to the original problem */ - GLPROW *row; - GLPCOL *col; - int i, j; - double dir; - xassert(npp->orig_dir == orig->dir); - if (npp->orig_dir == GLP_MIN) - dir = +1.0; - else if (npp->orig_dir == GLP_MAX) - dir = -1.0; - else - xassert(npp != npp); - xassert(npp->orig_m == orig->m); - xassert(npp->orig_n == orig->n); - xassert(npp->orig_nnz == orig->nnz); - if (npp->sol == GLP_SOL) - { /* store basic solution */ - orig->valid = 0; - orig->pbs_stat = npp->p_stat; - orig->dbs_stat = npp->d_stat; - orig->obj_val = orig->c0; - orig->some = 0; - for (i = 1; i <= orig->m; i++) - { row = orig->row[i]; - row->stat = npp->r_stat[i]; - if (!npp->scaling) - { /*row->prim = npp->r_prim[i];*/ - row->dual = dir * npp->r_pi[i]; - } - else - { /*row->prim = npp->r_prim[i] / row->rii;*/ - row->dual = dir * npp->r_pi[i] * row->rii; - } - if (row->stat == GLP_BS) - row->dual = 0.0; - else if (row->stat == GLP_NL) - { xassert(row->type == GLP_LO || row->type == GLP_DB); - row->prim = row->lb; - } - else if (row->stat == GLP_NU) - { xassert(row->type == GLP_UP || row->type == GLP_DB); - row->prim = row->ub; - } - else if (row->stat == GLP_NF) - { xassert(row->type == GLP_FR); - row->prim = 0.0; - } - else if (row->stat == GLP_NS) - { xassert(row->type == GLP_FX); - row->prim = row->lb; - } - else - xassert(row != row); - } - for (j = 1; j <= orig->n; j++) - { col = orig->col[j]; - col->stat = npp->c_stat[j]; - if (!npp->scaling) - { col->prim = npp->c_value[j]; - /*col->dual = dir * npp->c_dual[j];*/ - } - else - { col->prim = npp->c_value[j] * col->sjj; - /*col->dual = dir * npp->c_dual[j] / col->sjj;*/ - } - if (col->stat == GLP_BS) - col->dual = 0.0; -#if 1 - else if (col->stat == GLP_NL) - { xassert(col->type == GLP_LO || col->type == GLP_DB); - col->prim = col->lb; - } - else if (col->stat == GLP_NU) - { xassert(col->type == GLP_UP || col->type == GLP_DB); - col->prim = col->ub; - } - else if (col->stat == GLP_NF) - { xassert(col->type == GLP_FR); - col->prim = 0.0; - } - else if (col->stat == GLP_NS) - { xassert(col->type == GLP_FX); - col->prim = col->lb; - } - else - xassert(col != col); -#endif - orig->obj_val += col->coef * col->prim; - } -#if 1 - /* compute primal values of inactive rows */ - for (i = 1; i <= orig->m; i++) - { row = orig->row[i]; - if (row->stat == GLP_BS) - { GLPAIJ *aij; - double temp; - temp = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - temp += aij->val * aij->col->prim; - row->prim = temp; - } - } - /* compute reduced costs of active columns */ - for (j = 1; j <= orig->n; j++) - { col = orig->col[j]; - if (col->stat != GLP_BS) - { GLPAIJ *aij; - double temp; - temp = col->coef; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - temp -= aij->val * aij->row->dual; - col->dual = temp; - } - } -#endif - } - else if (npp->sol == GLP_IPT) - { /* store interior-point solution */ - orig->ipt_stat = npp->t_stat; - orig->ipt_obj = orig->c0; - for (i = 1; i <= orig->m; i++) - { row = orig->row[i]; - if (!npp->scaling) - { /*row->pval = npp->r_prim[i];*/ - row->dval = dir * npp->r_pi[i]; - } - else - { /*row->pval = npp->r_prim[i] / row->rii;*/ - row->dval = dir * npp->r_pi[i] * row->rii; - } - } - for (j = 1; j <= orig->n; j++) - { col = orig->col[j]; - if (!npp->scaling) - { col->pval = npp->c_value[j]; - /*col->dval = dir * npp->c_dual[j];*/ - } - else - { col->pval = npp->c_value[j] * col->sjj; - /*col->dval = dir * npp->c_dual[j] / col->sjj;*/ - } - orig->ipt_obj += col->coef * col->pval; - } -#if 1 - /* compute row primal values */ - for (i = 1; i <= orig->m; i++) - { row = orig->row[i]; - { GLPAIJ *aij; - double temp; - temp = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - temp += aij->val * aij->col->pval; - row->pval = temp; - } - } - /* compute column dual values */ - for (j = 1; j <= orig->n; j++) - { col = orig->col[j]; - { GLPAIJ *aij; - double temp; - temp = col->coef; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - temp -= aij->val * aij->row->dval; - col->dval = temp; - } - } -#endif - } - else if (npp->sol == GLP_MIP) - { /* store MIP solution */ - xassert(!npp->scaling); - orig->mip_stat = npp->i_stat; - orig->mip_obj = orig->c0; -#if 0 - for (i = 1; i <= orig->m; i++) - { row = orig->row[i]; - /*row->mipx = npp->r_prim[i];*/ - } -#endif - for (j = 1; j <= orig->n; j++) - { col = orig->col[j]; - col->mipx = npp->c_value[j]; - if (col->kind == GLP_IV) - xassert(col->mipx == floor(col->mipx)); - orig->mip_obj += col->coef * col->mipx; - } -#if 1 - /* compute row primal values */ - for (i = 1; i <= orig->m; i++) - { row = orig->row[i]; - { GLPAIJ *aij; - double temp; - temp = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - temp += aij->val * aij->col->mipx; - row->mipx = temp; - } - } -#endif - } - else - xassert(npp != npp); - return; -} - -void npp_delete_wksp(NPP *npp) -{ /* delete LP/MIP preprocessor workspace */ - if (npp->pool != NULL) - dmp_delete_pool(npp->pool); - if (npp->stack != NULL) - dmp_delete_pool(npp->stack); - if (npp->row_ref != NULL) - xfree(npp->row_ref); - if (npp->col_ref != NULL) - xfree(npp->col_ref); - if (npp->r_stat != NULL) - xfree(npp->r_stat); -#if 0 - if (npp->r_prim != NULL) - xfree(npp->r_prim); -#endif - if (npp->r_pi != NULL) - xfree(npp->r_pi); - if (npp->c_stat != NULL) - xfree(npp->c_stat); - if (npp->c_value != NULL) - xfree(npp->c_value); -#if 0 - if (npp->c_dual != NULL) - xfree(npp->c_dual); -#endif - xfree(npp); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/npp/npp2.c b/code/3rd_glpk/npp/npp2.c deleted file mode 100644 index 4efcf1d1..00000000 --- a/code/3rd_glpk/npp/npp2.c +++ /dev/null @@ -1,1433 +0,0 @@ -/* npp2.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -/*********************************************************************** -* NAME -* -* npp_free_row - process free (unbounded) row -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_free_row(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_free_row processes row p, which is free (i.e. has -* no finite bounds): -* -* -inf < sum a[p,j] x[j] < +inf. (1) -* j -* -* PROBLEM TRANSFORMATION -* -* Constraint (1) cannot be active, so it is redundant and can be -* removed from the original problem. -* -* Removing row p leads to removing a column of multiplier pi[p] for -* this row in the dual system. Since row p has no bounds, pi[p] = 0, -* so removing the column does not affect the dual solution. -* -* RECOVERING BASIC SOLUTION -* -* In solution to the original problem row p is inactive constraint, -* so it is assigned status GLP_BS, and multiplier pi[p] is assigned -* zero value. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* In solution to the original problem row p is inactive constraint, -* so its multiplier pi[p] is assigned zero value. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct free_row -{ /* free (unbounded) row */ - int p; - /* row reference number */ -}; - -static int rcv_free_row(NPP *npp, void *info); - -void npp_free_row(NPP *npp, NPPROW *p) -{ /* process free (unbounded) row */ - struct free_row *info; - /* the row must be free */ - xassert(p->lb == -DBL_MAX && p->ub == +DBL_MAX); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_free_row, sizeof(struct free_row)); - info->p = p->i; - /* remove the row from the problem */ - npp_del_row(npp, p); - return; -} - -static int rcv_free_row(NPP *npp, void *_info) -{ /* recover free (unbounded) row */ - struct free_row *info = _info; - if (npp->sol == GLP_SOL) - npp->r_stat[info->p] = GLP_BS; - if (npp->sol != GLP_MIP) - npp->r_pi[info->p] = 0.0; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_geq_row - process row of 'not less than' type -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_geq_row(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_geq_row processes row p, which is 'not less than' -* inequality constraint: -* -* L[p] <= sum a[p,j] x[j] (<= U[p]), (1) -* j -* -* where L[p] < U[p], and upper bound may not exist (U[p] = +oo). -* -* PROBLEM TRANSFORMATION -* -* Constraint (1) can be replaced by equality constraint: -* -* sum a[p,j] x[j] - s = L[p], (2) -* j -* -* where -* -* 0 <= s (<= U[p] - L[p]) (3) -* -* is a non-negative surplus variable. -* -* Since in the primal system there appears column s having the only -* non-zero coefficient in row p, in the dual system there appears a -* new row: -* -* (-1) pi[p] + lambda = 0, (4) -* -* where (-1) is coefficient of column s in row p, pi[p] is multiplier -* of row p, lambda is multiplier of column q, 0 is coefficient of -* column s in the objective row. -* -* RECOVERING BASIC SOLUTION -* -* Status of row p in solution to the original problem is determined -* by its status and status of column q in solution to the transformed -* problem as follows: -* -* +--------------------------------------+------------------+ -* | Transformed problem | Original problem | -* +-----------------+--------------------+------------------+ -* | Status of row p | Status of column s | Status of row p | -* +-----------------+--------------------+------------------+ -* | GLP_BS | GLP_BS | N/A | -* | GLP_BS | GLP_NL | GLP_BS | -* | GLP_BS | GLP_NU | GLP_BS | -* | GLP_NS | GLP_BS | GLP_BS | -* | GLP_NS | GLP_NL | GLP_NL | -* | GLP_NS | GLP_NU | GLP_NU | -* +-----------------+--------------------+------------------+ -* -* Value of row multiplier pi[p] in solution to the original problem -* is the same as in solution to the transformed problem. -* -* 1. In solution to the transformed problem row p and column q cannot -* be basic at the same time; otherwise the basis matrix would have -* two linear dependent columns: unity column of auxiliary variable -* of row p and unity column of variable s. -* -* 2. Though in the transformed problem row p is equality constraint, -* it may be basic due to primal degenerate solution. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of row multiplier pi[p] in solution to the original problem -* is the same as in solution to the transformed problem. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct ineq_row -{ /* inequality constraint row */ - int p; - /* row reference number */ - int s; - /* column reference number for slack/surplus variable */ -}; - -static int rcv_geq_row(NPP *npp, void *info); - -void npp_geq_row(NPP *npp, NPPROW *p) -{ /* process row of 'not less than' type */ - struct ineq_row *info; - NPPCOL *s; - /* the row must have lower bound */ - xassert(p->lb != -DBL_MAX); - xassert(p->lb < p->ub); - /* create column for surplus variable */ - s = npp_add_col(npp); - s->lb = 0.0; - s->ub = (p->ub == +DBL_MAX ? +DBL_MAX : p->ub - p->lb); - /* and add it to the transformed problem */ - npp_add_aij(npp, p, s, -1.0); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_geq_row, sizeof(struct ineq_row)); - info->p = p->i; - info->s = s->j; - /* replace the row by equality constraint */ - p->ub = p->lb; - return; -} - -static int rcv_geq_row(NPP *npp, void *_info) -{ /* recover row of 'not less than' type */ - struct ineq_row *info = _info; - if (npp->sol == GLP_SOL) - { if (npp->r_stat[info->p] == GLP_BS) - { if (npp->c_stat[info->s] == GLP_BS) - { npp_error(); - return 1; - } - else if (npp->c_stat[info->s] == GLP_NL || - npp->c_stat[info->s] == GLP_NU) - npp->r_stat[info->p] = GLP_BS; - else - { npp_error(); - return 1; - } - } - else if (npp->r_stat[info->p] == GLP_NS) - { if (npp->c_stat[info->s] == GLP_BS) - npp->r_stat[info->p] = GLP_BS; - else if (npp->c_stat[info->s] == GLP_NL) - npp->r_stat[info->p] = GLP_NL; - else if (npp->c_stat[info->s] == GLP_NU) - npp->r_stat[info->p] = GLP_NU; - else - { npp_error(); - return 1; - } - } - else - { npp_error(); - return 1; - } - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_leq_row - process row of 'not greater than' type -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_leq_row(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_leq_row processes row p, which is 'not greater than' -* inequality constraint: -* -* (L[p] <=) sum a[p,j] x[j] <= U[p], (1) -* j -* -* where L[p] < U[p], and lower bound may not exist (L[p] = +oo). -* -* PROBLEM TRANSFORMATION -* -* Constraint (1) can be replaced by equality constraint: -* -* sum a[p,j] x[j] + s = L[p], (2) -* j -* -* where -* -* 0 <= s (<= U[p] - L[p]) (3) -* -* is a non-negative slack variable. -* -* Since in the primal system there appears column s having the only -* non-zero coefficient in row p, in the dual system there appears a -* new row: -* -* (+1) pi[p] + lambda = 0, (4) -* -* where (+1) is coefficient of column s in row p, pi[p] is multiplier -* of row p, lambda is multiplier of column q, 0 is coefficient of -* column s in the objective row. -* -* RECOVERING BASIC SOLUTION -* -* Status of row p in solution to the original problem is determined -* by its status and status of column q in solution to the transformed -* problem as follows: -* -* +--------------------------------------+------------------+ -* | Transformed problem | Original problem | -* +-----------------+--------------------+------------------+ -* | Status of row p | Status of column s | Status of row p | -* +-----------------+--------------------+------------------+ -* | GLP_BS | GLP_BS | N/A | -* | GLP_BS | GLP_NL | GLP_BS | -* | GLP_BS | GLP_NU | GLP_BS | -* | GLP_NS | GLP_BS | GLP_BS | -* | GLP_NS | GLP_NL | GLP_NU | -* | GLP_NS | GLP_NU | GLP_NL | -* +-----------------+--------------------+------------------+ -* -* Value of row multiplier pi[p] in solution to the original problem -* is the same as in solution to the transformed problem. -* -* 1. In solution to the transformed problem row p and column q cannot -* be basic at the same time; otherwise the basis matrix would have -* two linear dependent columns: unity column of auxiliary variable -* of row p and unity column of variable s. -* -* 2. Though in the transformed problem row p is equality constraint, -* it may be basic due to primal degeneracy. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of row multiplier pi[p] in solution to the original problem -* is the same as in solution to the transformed problem. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -static int rcv_leq_row(NPP *npp, void *info); - -void npp_leq_row(NPP *npp, NPPROW *p) -{ /* process row of 'not greater than' type */ - struct ineq_row *info; - NPPCOL *s; - /* the row must have upper bound */ - xassert(p->ub != +DBL_MAX); - xassert(p->lb < p->ub); - /* create column for slack variable */ - s = npp_add_col(npp); - s->lb = 0.0; - s->ub = (p->lb == -DBL_MAX ? +DBL_MAX : p->ub - p->lb); - /* and add it to the transformed problem */ - npp_add_aij(npp, p, s, +1.0); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_leq_row, sizeof(struct ineq_row)); - info->p = p->i; - info->s = s->j; - /* replace the row by equality constraint */ - p->lb = p->ub; - return; -} - -static int rcv_leq_row(NPP *npp, void *_info) -{ /* recover row of 'not greater than' type */ - struct ineq_row *info = _info; - if (npp->sol == GLP_SOL) - { if (npp->r_stat[info->p] == GLP_BS) - { if (npp->c_stat[info->s] == GLP_BS) - { npp_error(); - return 1; - } - else if (npp->c_stat[info->s] == GLP_NL || - npp->c_stat[info->s] == GLP_NU) - npp->r_stat[info->p] = GLP_BS; - else - { npp_error(); - return 1; - } - } - else if (npp->r_stat[info->p] == GLP_NS) - { if (npp->c_stat[info->s] == GLP_BS) - npp->r_stat[info->p] = GLP_BS; - else if (npp->c_stat[info->s] == GLP_NL) - npp->r_stat[info->p] = GLP_NU; - else if (npp->c_stat[info->s] == GLP_NU) - npp->r_stat[info->p] = GLP_NL; - else - { npp_error(); - return 1; - } - } - else - { npp_error(); - return 1; - } - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_free_col - process free (unbounded) column -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_free_col(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_free_col processes column q, which is free (i.e. has -* no finite bounds): -* -* -oo < x[q] < +oo. (1) -* -* PROBLEM TRANSFORMATION -* -* Free (unbounded) variable can be replaced by the difference of two -* non-negative variables: -* -* x[q] = s' - s'', s', s'' >= 0. (2) -* -* Assuming that in the transformed problem x[q] becomes s', -* transformation (2) causes new column s'' to appear, which differs -* from column s' only in the sign of coefficients in constraint and -* objective rows. Thus, if in the dual system the following row -* corresponds to column s': -* -* sum a[i,q] pi[i] + lambda' = c[q], (3) -* i -* -* the row which corresponds to column s'' is the following: -* -* sum (-a[i,q]) pi[i] + lambda'' = -c[q]. (4) -* i -* -* Then from (3) and (4) it follows that: -* -* lambda' + lambda'' = 0 => lambda' = lmabda'' = 0, (5) -* -* where lambda' and lambda'' are multipliers for columns s' and s'', -* resp. -* -* RECOVERING BASIC SOLUTION -* -* With respect to (5) status of column q in solution to the original -* problem is determined by statuses of columns s' and s'' in solution -* to the transformed problem as follows: -* -* +--------------------------------------+------------------+ -* | Transformed problem | Original problem | -* +------------------+-------------------+------------------+ -* | Status of col s' | Status of col s'' | Status of col q | -* +------------------+-------------------+------------------+ -* | GLP_BS | GLP_BS | N/A | -* | GLP_BS | GLP_NL | GLP_BS | -* | GLP_NL | GLP_BS | GLP_BS | -* | GLP_NL | GLP_NL | GLP_NF | -* +------------------+-------------------+------------------+ -* -* Value of column q is computed with formula (2). -* -* 1. In solution to the transformed problem columns s' and s'' cannot -* be basic at the same time, because they differ only in the sign, -* hence, are linear dependent. -* -* 2. Though column q is free, it can be non-basic due to dual -* degeneracy. -* -* 3. If column q is integral, columns s' and s'' are also integral. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of column q is computed with formula (2). -* -* RECOVERING MIP SOLUTION -* -* Value of column q is computed with formula (2). */ - -struct free_col -{ /* free (unbounded) column */ - int q; - /* column reference number for variables x[q] and s' */ - int s; - /* column reference number for variable s'' */ -}; - -static int rcv_free_col(NPP *npp, void *info); - -void npp_free_col(NPP *npp, NPPCOL *q) -{ /* process free (unbounded) column */ - struct free_col *info; - NPPCOL *s; - NPPAIJ *aij; - /* the column must be free */ - xassert(q->lb == -DBL_MAX && q->ub == +DBL_MAX); - /* variable x[q] becomes s' */ - q->lb = 0.0, q->ub = +DBL_MAX; - /* create variable s'' */ - s = npp_add_col(npp); - s->is_int = q->is_int; - s->lb = 0.0, s->ub = +DBL_MAX; - /* duplicate objective coefficient */ - s->coef = -q->coef; - /* duplicate column of the constraint matrix */ - for (aij = q->ptr; aij != NULL; aij = aij->c_next) - npp_add_aij(npp, aij->row, s, -aij->val); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_free_col, sizeof(struct free_col)); - info->q = q->j; - info->s = s->j; - return; -} - -static int rcv_free_col(NPP *npp, void *_info) -{ /* recover free (unbounded) column */ - struct free_col *info = _info; - if (npp->sol == GLP_SOL) - { if (npp->c_stat[info->q] == GLP_BS) - { if (npp->c_stat[info->s] == GLP_BS) - { npp_error(); - return 1; - } - else if (npp->c_stat[info->s] == GLP_NL) - npp->c_stat[info->q] = GLP_BS; - else - { npp_error(); - return -1; - } - } - else if (npp->c_stat[info->q] == GLP_NL) - { if (npp->c_stat[info->s] == GLP_BS) - npp->c_stat[info->q] = GLP_BS; - else if (npp->c_stat[info->s] == GLP_NL) - npp->c_stat[info->q] = GLP_NF; - else - { npp_error(); - return -1; - } - } - else - { npp_error(); - return -1; - } - } - /* compute value of x[q] with formula (2) */ - npp->c_value[info->q] -= npp->c_value[info->s]; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_lbnd_col - process column with (non-zero) lower bound -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_lbnd_col(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_lbnd_col processes column q, which has (non-zero) -* lower bound: -* -* l[q] <= x[q] (<= u[q]), (1) -* -* where l[q] < u[q], and upper bound may not exist (u[q] = +oo). -* -* PROBLEM TRANSFORMATION -* -* Column q can be replaced as follows: -* -* x[q] = l[q] + s, (2) -* -* where -* -* 0 <= s (<= u[q] - l[q]) (3) -* -* is a non-negative variable. -* -* Substituting x[q] from (2) into the objective row, we have: -* -* z = sum c[j] x[j] + c0 = -* j -* -* = sum c[j] x[j] + c[q] x[q] + c0 = -* j!=q -* -* = sum c[j] x[j] + c[q] (l[q] + s) + c0 = -* j!=q -* -* = sum c[j] x[j] + c[q] s + c~0, -* -* where -* -* c~0 = c0 + c[q] l[q] (4) -* -* is the constant term of the objective in the transformed problem. -* Similarly, substituting x[q] into constraint row i, we have: -* -* L[i] <= sum a[i,j] x[j] <= U[i] ==> -* j -* -* L[i] <= sum a[i,j] x[j] + a[i,q] x[q] <= U[i] ==> -* j!=q -* -* L[i] <= sum a[i,j] x[j] + a[i,q] (l[q] + s) <= U[i] ==> -* j!=q -* -* L~[i] <= sum a[i,j] x[j] + a[i,q] s <= U~[i], -* j!=q -* -* where -* -* L~[i] = L[i] - a[i,q] l[q], U~[i] = U[i] - a[i,q] l[q] (5) -* -* are lower and upper bounds of row i in the transformed problem, -* resp. -* -* Transformation (2) does not affect the dual system. -* -* RECOVERING BASIC SOLUTION -* -* Status of column q in solution to the original problem is the same -* as in solution to the transformed problem (GLP_BS, GLP_NL or GLP_NU). -* Value of column q is computed with formula (2). -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of column q is computed with formula (2). -* -* RECOVERING MIP SOLUTION -* -* Value of column q is computed with formula (2). */ - -struct bnd_col -{ /* bounded column */ - int q; - /* column reference number for variables x[q] and s */ - double bnd; - /* lower/upper bound l[q] or u[q] */ -}; - -static int rcv_lbnd_col(NPP *npp, void *info); - -void npp_lbnd_col(NPP *npp, NPPCOL *q) -{ /* process column with (non-zero) lower bound */ - struct bnd_col *info; - NPPROW *i; - NPPAIJ *aij; - /* the column must have non-zero lower bound */ - xassert(q->lb != 0.0); - xassert(q->lb != -DBL_MAX); - xassert(q->lb < q->ub); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_lbnd_col, sizeof(struct bnd_col)); - info->q = q->j; - info->bnd = q->lb; - /* substitute x[q] into objective row */ - npp->c0 += q->coef * q->lb; - /* substitute x[q] into constraint rows */ - for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { i = aij->row; - if (i->lb == i->ub) - i->ub = (i->lb -= aij->val * q->lb); - else - { if (i->lb != -DBL_MAX) - i->lb -= aij->val * q->lb; - if (i->ub != +DBL_MAX) - i->ub -= aij->val * q->lb; - } - } - /* column x[q] becomes column s */ - if (q->ub != +DBL_MAX) - q->ub -= q->lb; - q->lb = 0.0; - return; -} - -static int rcv_lbnd_col(NPP *npp, void *_info) -{ /* recover column with (non-zero) lower bound */ - struct bnd_col *info = _info; - if (npp->sol == GLP_SOL) - { if (npp->c_stat[info->q] == GLP_BS || - npp->c_stat[info->q] == GLP_NL || - npp->c_stat[info->q] == GLP_NU) - npp->c_stat[info->q] = npp->c_stat[info->q]; - else - { npp_error(); - return 1; - } - } - /* compute value of x[q] with formula (2) */ - npp->c_value[info->q] = info->bnd + npp->c_value[info->q]; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_ubnd_col - process column with upper bound -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_ubnd_col(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_ubnd_col processes column q, which has upper bound: -* -* (l[q] <=) x[q] <= u[q], (1) -* -* where l[q] < u[q], and lower bound may not exist (l[q] = -oo). -* -* PROBLEM TRANSFORMATION -* -* Column q can be replaced as follows: -* -* x[q] = u[q] - s, (2) -* -* where -* -* 0 <= s (<= u[q] - l[q]) (3) -* -* is a non-negative variable. -* -* Substituting x[q] from (2) into the objective row, we have: -* -* z = sum c[j] x[j] + c0 = -* j -* -* = sum c[j] x[j] + c[q] x[q] + c0 = -* j!=q -* -* = sum c[j] x[j] + c[q] (u[q] - s) + c0 = -* j!=q -* -* = sum c[j] x[j] - c[q] s + c~0, -* -* where -* -* c~0 = c0 + c[q] u[q] (4) -* -* is the constant term of the objective in the transformed problem. -* Similarly, substituting x[q] into constraint row i, we have: -* -* L[i] <= sum a[i,j] x[j] <= U[i] ==> -* j -* -* L[i] <= sum a[i,j] x[j] + a[i,q] x[q] <= U[i] ==> -* j!=q -* -* L[i] <= sum a[i,j] x[j] + a[i,q] (u[q] - s) <= U[i] ==> -* j!=q -* -* L~[i] <= sum a[i,j] x[j] - a[i,q] s <= U~[i], -* j!=q -* -* where -* -* L~[i] = L[i] - a[i,q] u[q], U~[i] = U[i] - a[i,q] u[q] (5) -* -* are lower and upper bounds of row i in the transformed problem, -* resp. -* -* Note that in the transformed problem coefficients c[q] and a[i,q] -* change their sign. Thus, the row of the dual system corresponding to -* column q: -* -* sum a[i,q] pi[i] + lambda[q] = c[q] (6) -* i -* -* in the transformed problem becomes the following: -* -* sum (-a[i,q]) pi[i] + lambda[s] = -c[q]. (7) -* i -* -* Therefore: -* -* lambda[q] = - lambda[s], (8) -* -* where lambda[q] is multiplier for column q, lambda[s] is multiplier -* for column s. -* -* RECOVERING BASIC SOLUTION -* -* With respect to (8) status of column q in solution to the original -* problem is determined by status of column s in solution to the -* transformed problem as follows: -* -* +-----------------------+--------------------+ -* | Status of column s | Status of column q | -* | (transformed problem) | (original problem) | -* +-----------------------+--------------------+ -* | GLP_BS | GLP_BS | -* | GLP_NL | GLP_NU | -* | GLP_NU | GLP_NL | -* +-----------------------+--------------------+ -* -* Value of column q is computed with formula (2). -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of column q is computed with formula (2). -* -* RECOVERING MIP SOLUTION -* -* Value of column q is computed with formula (2). */ - -static int rcv_ubnd_col(NPP *npp, void *info); - -void npp_ubnd_col(NPP *npp, NPPCOL *q) -{ /* process column with upper bound */ - struct bnd_col *info; - NPPROW *i; - NPPAIJ *aij; - /* the column must have upper bound */ - xassert(q->ub != +DBL_MAX); - xassert(q->lb < q->ub); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_ubnd_col, sizeof(struct bnd_col)); - info->q = q->j; - info->bnd = q->ub; - /* substitute x[q] into objective row */ - npp->c0 += q->coef * q->ub; - q->coef = -q->coef; - /* substitute x[q] into constraint rows */ - for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { i = aij->row; - if (i->lb == i->ub) - i->ub = (i->lb -= aij->val * q->ub); - else - { if (i->lb != -DBL_MAX) - i->lb -= aij->val * q->ub; - if (i->ub != +DBL_MAX) - i->ub -= aij->val * q->ub; - } - aij->val = -aij->val; - } - /* column x[q] becomes column s */ - if (q->lb != -DBL_MAX) - q->ub -= q->lb; - else - q->ub = +DBL_MAX; - q->lb = 0.0; - return; -} - -static int rcv_ubnd_col(NPP *npp, void *_info) -{ /* recover column with upper bound */ - struct bnd_col *info = _info; - if (npp->sol == GLP_BS) - { if (npp->c_stat[info->q] == GLP_BS) - npp->c_stat[info->q] = GLP_BS; - else if (npp->c_stat[info->q] == GLP_NL) - npp->c_stat[info->q] = GLP_NU; - else if (npp->c_stat[info->q] == GLP_NU) - npp->c_stat[info->q] = GLP_NL; - else - { npp_error(); - return 1; - } - } - /* compute value of x[q] with formula (2) */ - npp->c_value[info->q] = info->bnd - npp->c_value[info->q]; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_dbnd_col - process non-negative column with upper bound -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_dbnd_col(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_dbnd_col processes column q, which is non-negative -* and has upper bound: -* -* 0 <= x[q] <= u[q], (1) -* -* where u[q] > 0. -* -* PROBLEM TRANSFORMATION -* -* Upper bound of column q can be replaced by the following equality -* constraint: -* -* x[q] + s = u[q], (2) -* -* where s >= 0 is a non-negative complement variable. -* -* Since in the primal system along with new row (2) there appears a -* new column s having the only non-zero coefficient in this row, in -* the dual system there appears a new row: -* -* (+1)pi + lambda[s] = 0, (3) -* -* where (+1) is coefficient at column s in row (2), pi is multiplier -* for row (2), lambda[s] is multiplier for column s, 0 is coefficient -* at column s in the objective row. -* -* RECOVERING BASIC SOLUTION -* -* Status of column q in solution to the original problem is determined -* by its status and status of column s in solution to the transformed -* problem as follows: -* -* +-----------------------------------+------------------+ -* | Transformed problem | Original problem | -* +-----------------+-----------------+------------------+ -* | Status of col q | Status of col s | Status of col q | -* +-----------------+-----------------+------------------+ -* | GLP_BS | GLP_BS | GLP_BS | -* | GLP_BS | GLP_NL | GLP_NU | -* | GLP_NL | GLP_BS | GLP_NL | -* | GLP_NL | GLP_NL | GLP_NL (*) | -* +-----------------+-----------------+------------------+ -* -* Value of column q in solution to the original problem is the same as -* in solution to the transformed problem. -* -* 1. Formally, in solution to the transformed problem columns q and s -* cannot be non-basic at the same time, since the constraint (2) -* would be violated. However, if u[q] is close to zero, violation -* may be less than a working precision even if both columns q and s -* are non-basic. In this degenerate case row (2) can be only basic, -* i.e. non-active constraint (otherwise corresponding row of the -* basis matrix would be zero). This allows to pivot out auxiliary -* variable and pivot in column s, in which case the row becomes -* active while column s becomes basic. -* -* 2. If column q is integral, column s is also integral. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of column q in solution to the original problem is the same as -* in solution to the transformed problem. -* -* RECOVERING MIP SOLUTION -* -* Value of column q in solution to the original problem is the same as -* in solution to the transformed problem. */ - -struct dbnd_col -{ /* double-bounded column */ - int q; - /* column reference number for variable x[q] */ - int s; - /* column reference number for complement variable s */ -}; - -static int rcv_dbnd_col(NPP *npp, void *info); - -void npp_dbnd_col(NPP *npp, NPPCOL *q) -{ /* process non-negative column with upper bound */ - struct dbnd_col *info; - NPPROW *p; - NPPCOL *s; - /* the column must be non-negative with upper bound */ - xassert(q->lb == 0.0); - xassert(q->ub > 0.0); - xassert(q->ub != +DBL_MAX); - /* create variable s */ - s = npp_add_col(npp); - s->is_int = q->is_int; - s->lb = 0.0, s->ub = +DBL_MAX; - /* create equality constraint (2) */ - p = npp_add_row(npp); - p->lb = p->ub = q->ub; - npp_add_aij(npp, p, q, +1.0); - npp_add_aij(npp, p, s, +1.0); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_dbnd_col, sizeof(struct dbnd_col)); - info->q = q->j; - info->s = s->j; - /* remove upper bound of x[q] */ - q->ub = +DBL_MAX; - return; -} - -static int rcv_dbnd_col(NPP *npp, void *_info) -{ /* recover non-negative column with upper bound */ - struct dbnd_col *info = _info; - if (npp->sol == GLP_BS) - { if (npp->c_stat[info->q] == GLP_BS) - { if (npp->c_stat[info->s] == GLP_BS) - npp->c_stat[info->q] = GLP_BS; - else if (npp->c_stat[info->s] == GLP_NL) - npp->c_stat[info->q] = GLP_NU; - else - { npp_error(); - return 1; - } - } - else if (npp->c_stat[info->q] == GLP_NL) - { if (npp->c_stat[info->s] == GLP_BS || - npp->c_stat[info->s] == GLP_NL) - npp->c_stat[info->q] = GLP_NL; - else - { npp_error(); - return 1; - } - } - else - { npp_error(); - return 1; - } - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_fixed_col - process fixed column -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_fixed_col(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_fixed_col processes column q, which is fixed: -* -* x[q] = s[q], (1) -* -* where s[q] is a fixed column value. -* -* PROBLEM TRANSFORMATION -* -* The value of a fixed column can be substituted into the objective -* and constraint rows that allows removing the column from the problem. -* -* Substituting x[q] = s[q] into the objective row, we have: -* -* z = sum c[j] x[j] + c0 = -* j -* -* = sum c[j] x[j] + c[q] x[q] + c0 = -* j!=q -* -* = sum c[j] x[j] + c[q] s[q] + c0 = -* j!=q -* -* = sum c[j] x[j] + c~0, -* j!=q -* -* where -* -* c~0 = c0 + c[q] s[q] (2) -* -* is the constant term of the objective in the transformed problem. -* Similarly, substituting x[q] = s[q] into constraint row i, we have: -* -* L[i] <= sum a[i,j] x[j] <= U[i] ==> -* j -* -* L[i] <= sum a[i,j] x[j] + a[i,q] x[q] <= U[i] ==> -* j!=q -* -* L[i] <= sum a[i,j] x[j] + a[i,q] s[q] <= U[i] ==> -* j!=q -* -* L~[i] <= sum a[i,j] x[j] + a[i,q] s <= U~[i], -* j!=q -* -* where -* -* L~[i] = L[i] - a[i,q] s[q], U~[i] = U[i] - a[i,q] s[q] (3) -* -* are lower and upper bounds of row i in the transformed problem, -* resp. -* -* RECOVERING BASIC SOLUTION -* -* Column q is assigned status GLP_NS and its value is assigned s[q]. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of column q is assigned s[q]. -* -* RECOVERING MIP SOLUTION -* -* Value of column q is assigned s[q]. */ - -struct fixed_col -{ /* fixed column */ - int q; - /* column reference number for variable x[q] */ - double s; - /* value, at which x[q] is fixed */ -}; - -static int rcv_fixed_col(NPP *npp, void *info); - -void npp_fixed_col(NPP *npp, NPPCOL *q) -{ /* process fixed column */ - struct fixed_col *info; - NPPROW *i; - NPPAIJ *aij; - /* the column must be fixed */ - xassert(q->lb == q->ub); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_fixed_col, sizeof(struct fixed_col)); - info->q = q->j; - info->s = q->lb; - /* substitute x[q] = s[q] into objective row */ - npp->c0 += q->coef * q->lb; - /* substitute x[q] = s[q] into constraint rows */ - for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { i = aij->row; - if (i->lb == i->ub) - i->ub = (i->lb -= aij->val * q->lb); - else - { if (i->lb != -DBL_MAX) - i->lb -= aij->val * q->lb; - if (i->ub != +DBL_MAX) - i->ub -= aij->val * q->lb; - } - } - /* remove the column from the problem */ - npp_del_col(npp, q); - return; -} - -static int rcv_fixed_col(NPP *npp, void *_info) -{ /* recover fixed column */ - struct fixed_col *info = _info; - if (npp->sol == GLP_SOL) - npp->c_stat[info->q] = GLP_NS; - npp->c_value[info->q] = info->s; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_make_equality - process row with almost identical bounds -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_make_equality(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_make_equality processes row p: -* -* L[p] <= sum a[p,j] x[j] <= U[p], (1) -* j -* -* where -oo < L[p] < U[p] < +oo, i.e. which is double-sided inequality -* constraint. -* -* RETURNS -* -* 0 - row bounds have not been changed; -* -* 1 - row has been replaced by equality constraint. -* -* PROBLEM TRANSFORMATION -* -* If bounds of row (1) are very close to each other: -* -* U[p] - L[p] <= eps, (2) -* -* where eps is an absolute tolerance for row value, the row can be -* replaced by the following almost equivalent equiality constraint: -* -* sum a[p,j] x[j] = b, (3) -* j -* -* where b = (L[p] + U[p]) / 2. If the right-hand side in (3) happens -* to be very close to its nearest integer: -* -* |b - floor(b + 0.5)| <= eps, (4) -* -* it is reasonable to use this nearest integer as the right-hand side. -* -* RECOVERING BASIC SOLUTION -* -* Status of row p in solution to the original problem is determined -* by its status and the sign of its multiplier pi[p] in solution to -* the transformed problem as follows: -* -* +-----------------------+---------+--------------------+ -* | Status of row p | Sign of | Status of row p | -* | (transformed problem) | pi[p] | (original problem) | -* +-----------------------+---------+--------------------+ -* | GLP_BS | + / - | GLP_BS | -* | GLP_NS | + | GLP_NL | -* | GLP_NS | - | GLP_NU | -* +-----------------------+---------+--------------------+ -* -* Value of row multiplier pi[p] in solution to the original problem is -* the same as in solution to the transformed problem. -* -* RECOVERING INTERIOR POINT SOLUTION -* -* Value of row multiplier pi[p] in solution to the original problem is -* the same as in solution to the transformed problem. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct make_equality -{ /* row with almost identical bounds */ - int p; - /* row reference number */ -}; - -static int rcv_make_equality(NPP *npp, void *info); - -int npp_make_equality(NPP *npp, NPPROW *p) -{ /* process row with almost identical bounds */ - struct make_equality *info; - double b, eps, nint; - /* the row must be double-sided inequality */ - xassert(p->lb != -DBL_MAX); - xassert(p->ub != +DBL_MAX); - xassert(p->lb < p->ub); - /* check row bounds */ - eps = 1e-9 + 1e-12 * fabs(p->lb); - if (p->ub - p->lb > eps) return 0; - /* row bounds are very close to each other */ - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_make_equality, sizeof(struct make_equality)); - info->p = p->i; - /* compute right-hand side */ - b = 0.5 * (p->ub + p->lb); - nint = floor(b + 0.5); - if (fabs(b - nint) <= eps) b = nint; - /* replace row p by almost equivalent equality constraint */ - p->lb = p->ub = b; - return 1; -} - -int rcv_make_equality(NPP *npp, void *_info) -{ /* recover row with almost identical bounds */ - struct make_equality *info = _info; - if (npp->sol == GLP_SOL) - { if (npp->r_stat[info->p] == GLP_BS) - npp->r_stat[info->p] = GLP_BS; - else if (npp->r_stat[info->p] == GLP_NS) - { if (npp->r_pi[info->p] >= 0.0) - npp->r_stat[info->p] = GLP_NL; - else - npp->r_stat[info->p] = GLP_NU; - } - else - { npp_error(); - return 1; - } - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_make_fixed - process column with almost identical bounds -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_make_fixed(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_make_fixed processes column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* where -oo < l[q] < u[q] < +oo, i.e. which has both lower and upper -* bounds. -* -* RETURNS -* -* 0 - column bounds have not been changed; -* -* 1 - column has been fixed. -* -* PROBLEM TRANSFORMATION -* -* If bounds of column (1) are very close to each other: -* -* u[q] - l[q] <= eps, (2) -* -* where eps is an absolute tolerance for column value, the column can -* be fixed: -* -* x[q] = s[q], (3) -* -* where s[q] = (l[q] + u[q]) / 2. And if the fixed column value s[q] -* happens to be very close to its nearest integer: -* -* |s[q] - floor(s[q] + 0.5)| <= eps, (4) -* -* it is reasonable to use this nearest integer as the fixed value. -* -* RECOVERING BASIC SOLUTION -* -* In the dual system of the original (as well as transformed) problem -* column q corresponds to the following row: -* -* sum a[i,q] pi[i] + lambda[q] = c[q]. (5) -* i -* -* Since multipliers pi[i] are known for all rows from solution to the -* transformed problem, formula (5) allows computing value of multiplier -* (reduced cost) for column q: -* -* lambda[q] = c[q] - sum a[i,q] pi[i]. (6) -* i -* -* Status of column q in solution to the original problem is determined -* by its status and the sign of its multiplier lambda[q] in solution to -* the transformed problem as follows: -* -* +-----------------------+-----------+--------------------+ -* | Status of column q | Sign of | Status of column q | -* | (transformed problem) | lambda[q] | (original problem) | -* +-----------------------+-----------+--------------------+ -* | GLP_BS | + / - | GLP_BS | -* | GLP_NS | + | GLP_NL | -* | GLP_NS | - | GLP_NU | -* +-----------------------+-----------+--------------------+ -* -* Value of column q in solution to the original problem is the same as -* in solution to the transformed problem. -* -* RECOVERING INTERIOR POINT SOLUTION -* -* Value of column q in solution to the original problem is the same as -* in solution to the transformed problem. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct make_fixed -{ /* column with almost identical bounds */ - int q; - /* column reference number */ - double c; - /* objective coefficient at x[q] */ - NPPLFE *ptr; - /* list of non-zero coefficients a[i,q] */ -}; - -static int rcv_make_fixed(NPP *npp, void *info); - -int npp_make_fixed(NPP *npp, NPPCOL *q) -{ /* process column with almost identical bounds */ - struct make_fixed *info; - NPPAIJ *aij; - NPPLFE *lfe; - double s, eps, nint; - /* the column must be double-bounded */ - xassert(q->lb != -DBL_MAX); - xassert(q->ub != +DBL_MAX); - xassert(q->lb < q->ub); - /* check column bounds */ - eps = 1e-9 + 1e-12 * fabs(q->lb); - if (q->ub - q->lb > eps) return 0; - /* column bounds are very close to each other */ - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_make_fixed, sizeof(struct make_fixed)); - info->q = q->j; - info->c = q->coef; - info->ptr = NULL; - /* save column coefficients a[i,q] (needed for basic solution - only) */ - if (npp->sol == GLP_SOL) - { for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { lfe = dmp_get_atom(npp->stack, sizeof(NPPLFE)); - lfe->ref = aij->row->i; - lfe->val = aij->val; - lfe->next = info->ptr; - info->ptr = lfe; - } - } - /* compute column fixed value */ - s = 0.5 * (q->ub + q->lb); - nint = floor(s + 0.5); - if (fabs(s - nint) <= eps) s = nint; - /* make column q fixed */ - q->lb = q->ub = s; - return 1; -} - -static int rcv_make_fixed(NPP *npp, void *_info) -{ /* recover column with almost identical bounds */ - struct make_fixed *info = _info; - NPPLFE *lfe; - double lambda; - if (npp->sol == GLP_SOL) - { if (npp->c_stat[info->q] == GLP_BS) - npp->c_stat[info->q] = GLP_BS; - else if (npp->c_stat[info->q] == GLP_NS) - { /* compute multiplier for column q with formula (6) */ - lambda = info->c; - for (lfe = info->ptr; lfe != NULL; lfe = lfe->next) - lambda -= lfe->val * npp->r_pi[lfe->ref]; - /* assign status to non-basic column */ - if (lambda >= 0.0) - npp->c_stat[info->q] = GLP_NL; - else - npp->c_stat[info->q] = GLP_NU; - } - else - { npp_error(); - return 1; - } - } - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/npp/npp3.c b/code/3rd_glpk/npp/npp3.c deleted file mode 100644 index 883af127..00000000 --- a/code/3rd_glpk/npp/npp3.c +++ /dev/null @@ -1,2861 +0,0 @@ -/* npp3.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -/*********************************************************************** -* NAME -* -* npp_empty_row - process empty row -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_empty_row(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_empty_row processes row p, which is empty, i.e. -* coefficients at all columns in this row are zero: -* -* L[p] <= sum 0 x[j] <= U[p], (1) -* -* where L[p] <= U[p]. -* -* RETURNS -* -* 0 - success; -* -* 1 - problem has no primal feasible solution. -* -* PROBLEM TRANSFORMATION -* -* If the following conditions hold: -* -* L[p] <= +eps, U[p] >= -eps, (2) -* -* where eps is an absolute tolerance for row value, the row p is -* redundant. In this case it can be replaced by equivalent redundant -* row, which is free (unbounded), and then removed from the problem. -* Otherwise, the row p is infeasible and, thus, the problem has no -* primal feasible solution. -* -* RECOVERING BASIC SOLUTION -* -* See the routine npp_free_row. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* See the routine npp_free_row. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -int npp_empty_row(NPP *npp, NPPROW *p) -{ /* process empty row */ - double eps = 1e-3; - /* the row must be empty */ - xassert(p->ptr == NULL); - /* check primal feasibility */ - if (p->lb > +eps || p->ub < -eps) - return 1; - /* replace the row by equivalent free (unbounded) row */ - p->lb = -DBL_MAX, p->ub = +DBL_MAX; - /* and process it */ - npp_free_row(npp, p); - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_empty_col - process empty column -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_empty_col(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_empty_col processes column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* where l[q] <= u[q], which is empty, i.e. has zero coefficients in -* all constraint rows. -* -* RETURNS -* -* 0 - success; -* -* 1 - problem has no dual feasible solution. -* -* PROBLEM TRANSFORMATION -* -* The row of the dual system corresponding to the empty column is the -* following: -* -* sum 0 pi[i] + lambda[q] = c[q], (2) -* i -* -* from which it follows that: -* -* lambda[q] = c[q]. (3) -* -* If the following condition holds: -* -* c[q] < - eps, (4) -* -* where eps is an absolute tolerance for column multiplier, the lower -* column bound l[q] must be active to provide dual feasibility (note -* that being preprocessed the problem is always minimization). In this -* case the column can be fixed on its lower bound and removed from the -* problem (if the column is integral, its bounds are also assumed to -* be integral). And if the column has no lower bound (l[q] = -oo), the -* problem has no dual feasible solution. -* -* If the following condition holds: -* -* c[q] > + eps, (5) -* -* the upper column bound u[q] must be active to provide dual -* feasibility. In this case the column can be fixed on its upper bound -* and removed from the problem. And if the column has no upper bound -* (u[q] = +oo), the problem has no dual feasible solution. -* -* Finally, if the following condition holds: -* -* - eps <= c[q] <= +eps, (6) -* -* dual feasibility does not depend on a particular value of column q. -* In this case the column can be fixed either on its lower bound (if -* l[q] > -oo) or on its upper bound (if u[q] < +oo) or at zero (if the -* column is unbounded) and then removed from the problem. -* -* RECOVERING BASIC SOLUTION -* -* See the routine npp_fixed_col. Having been recovered the column -* is assigned status GLP_NS. However, if actually it is not fixed -* (l[q] < u[q]), its status should be changed to GLP_NL, GLP_NU, or -* GLP_NF depending on which bound it was fixed on transformation stage. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* See the routine npp_fixed_col. -* -* RECOVERING MIP SOLUTION -* -* See the routine npp_fixed_col. */ - -struct empty_col -{ /* empty column */ - int q; - /* column reference number */ - char stat; - /* status in basic solution */ -}; - -static int rcv_empty_col(NPP *npp, void *info); - -int npp_empty_col(NPP *npp, NPPCOL *q) -{ /* process empty column */ - struct empty_col *info; - double eps = 1e-3; - /* the column must be empty */ - xassert(q->ptr == NULL); - /* check dual feasibility */ - if (q->coef > +eps && q->lb == -DBL_MAX) - return 1; - if (q->coef < -eps && q->ub == +DBL_MAX) - return 1; - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_empty_col, sizeof(struct empty_col)); - info->q = q->j; - /* fix the column */ - if (q->lb == -DBL_MAX && q->ub == +DBL_MAX) - { /* free column */ - info->stat = GLP_NF; - q->lb = q->ub = 0.0; - } - else if (q->ub == +DBL_MAX) -lo: { /* column with lower bound */ - info->stat = GLP_NL; - q->ub = q->lb; - } - else if (q->lb == -DBL_MAX) -up: { /* column with upper bound */ - info->stat = GLP_NU; - q->lb = q->ub; - } - else if (q->lb != q->ub) - { /* double-bounded column */ - if (q->coef >= +DBL_EPSILON) goto lo; - if (q->coef <= -DBL_EPSILON) goto up; - if (fabs(q->lb) <= fabs(q->ub)) goto lo; else goto up; - } - else - { /* fixed column */ - info->stat = GLP_NS; - } - /* process fixed column */ - npp_fixed_col(npp, q); - return 0; -} - -static int rcv_empty_col(NPP *npp, void *_info) -{ /* recover empty column */ - struct empty_col *info = _info; - if (npp->sol == GLP_SOL) - npp->c_stat[info->q] = info->stat; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_implied_value - process implied column value -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_implied_value(NPP *npp, NPPCOL *q, double s); -* -* DESCRIPTION -* -* For column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* where l[q] < u[q], the routine npp_implied_value processes its -* implied value s[q]. If this implied value satisfies to the current -* column bounds and integrality condition, the routine fixes column q -* at the given point. Note that the column is kept in the problem in -* any case. -* -* RETURNS -* -* 0 - column has been fixed; -* -* 1 - implied value violates to current column bounds; -* -* 2 - implied value violates integrality condition. -* -* ALGORITHM -* -* Implied column value s[q] satisfies to the current column bounds if -* the following condition holds: -* -* l[q] - eps <= s[q] <= u[q] + eps, (2) -* -* where eps is an absolute tolerance for column value. If the column -* is integral, the following condition also must hold: -* -* |s[q] - floor(s[q]+0.5)| <= eps, (3) -* -* where floor(s[q]+0.5) is the nearest integer to s[q]. -* -* If both condition (2) and (3) are satisfied, the column can be fixed -* at the value s[q], or, if it is integral, at floor(s[q]+0.5). -* Otherwise, if s[q] violates (2) or (3), the problem has no feasible -* solution. -* -* Note: If s[q] is close to l[q] or u[q], it seems to be reasonable to -* fix the column at its lower or upper bound, resp. rather than at the -* implied value. */ - -int npp_implied_value(NPP *npp, NPPCOL *q, double s) -{ /* process implied column value */ - double eps, nint; - xassert(npp == npp); - /* column must not be fixed */ - xassert(q->lb < q->ub); - /* check integrality */ - if (q->is_int) - { nint = floor(s + 0.5); - if (fabs(s - nint) <= 1e-5) - s = nint; - else - return 2; - } - /* check current column lower bound */ - if (q->lb != -DBL_MAX) - { eps = (q->is_int ? 1e-5 : 1e-5 + 1e-8 * fabs(q->lb)); - if (s < q->lb - eps) return 1; - /* if s[q] is close to l[q], fix column at its lower bound - rather than at the implied value */ - if (s < q->lb + 1e-3 * eps) - { q->ub = q->lb; - return 0; - } - } - /* check current column upper bound */ - if (q->ub != +DBL_MAX) - { eps = (q->is_int ? 1e-5 : 1e-5 + 1e-8 * fabs(q->ub)); - if (s > q->ub + eps) return 1; - /* if s[q] is close to u[q], fix column at its upper bound - rather than at the implied value */ - if (s > q->ub - 1e-3 * eps) - { q->lb = q->ub; - return 0; - } - } - /* fix column at the implied value */ - q->lb = q->ub = s; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_eq_singlet - process row singleton (equality constraint) -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_eq_singlet(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_eq_singlet processes row p, which is equiality -* constraint having the only non-zero coefficient: -* -* a[p,q] x[q] = b. (1) -* -* RETURNS -* -* 0 - success; -* -* 1 - problem has no primal feasible solution; -* -* 2 - problem has no integer feasible solution. -* -* PROBLEM TRANSFORMATION -* -* The equality constraint defines implied value of column q: -* -* x[q] = s[q] = b / a[p,q]. (2) -* -* If the implied value s[q] satisfies to the column bounds (see the -* routine npp_implied_value), the column can be fixed at s[q] and -* removed from the problem. In this case row p becomes redundant, so -* it can be replaced by equivalent free row and also removed from the -* problem. -* -* Note that the routine removes from the problem only row p. Column q -* becomes fixed, however, it is kept in the problem. -* -* RECOVERING BASIC SOLUTION -* -* In solution to the original problem row p is assigned status GLP_NS -* (active equality constraint), and column q is assigned status GLP_BS -* (basic column). -* -* Multiplier for row p can be computed as follows. In the dual system -* of the original problem column q corresponds to the following row: -* -* sum a[i,q] pi[i] + lambda[q] = c[q] ==> -* i -* -* sum a[i,q] pi[i] + a[p,q] pi[p] + lambda[q] = c[q]. -* i!=p -* -* Therefore: -* -* 1 -* pi[p] = ------ (c[q] - lambda[q] - sum a[i,q] pi[i]), (3) -* a[p,q] i!=q -* -* where lambda[q] = 0 (since column[q] is basic), and pi[i] for all -* i != p are known in solution to the transformed problem. -* -* Value of column q in solution to the original problem is assigned -* its implied value s[q]. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Multiplier for row p is computed with formula (3). Value of column -* q is assigned its implied value s[q]. -* -* RECOVERING MIP SOLUTION -* -* Value of column q is assigned its implied value s[q]. */ - -struct eq_singlet -{ /* row singleton (equality constraint) */ - int p; - /* row reference number */ - int q; - /* column reference number */ - double apq; - /* constraint coefficient a[p,q] */ - double c; - /* objective coefficient at x[q] */ - NPPLFE *ptr; - /* list of non-zero coefficients a[i,q], i != p */ -}; - -static int rcv_eq_singlet(NPP *npp, void *info); - -int npp_eq_singlet(NPP *npp, NPPROW *p) -{ /* process row singleton (equality constraint) */ - struct eq_singlet *info; - NPPCOL *q; - NPPAIJ *aij; - NPPLFE *lfe; - int ret; - double s; - /* the row must be singleton equality constraint */ - xassert(p->lb == p->ub); - xassert(p->ptr != NULL && p->ptr->r_next == NULL); - /* compute and process implied column value */ - aij = p->ptr; - q = aij->col; - s = p->lb / aij->val; - ret = npp_implied_value(npp, q, s); - xassert(0 <= ret && ret <= 2); - if (ret != 0) return ret; - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_eq_singlet, sizeof(struct eq_singlet)); - info->p = p->i; - info->q = q->j; - info->apq = aij->val; - info->c = q->coef; - info->ptr = NULL; - /* save column coefficients a[i,q], i != p (not needed for MIP - solution) */ - if (npp->sol != GLP_MIP) - { for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { if (aij->row == p) continue; /* skip a[p,q] */ - lfe = dmp_get_atom(npp->stack, sizeof(NPPLFE)); - lfe->ref = aij->row->i; - lfe->val = aij->val; - lfe->next = info->ptr; - info->ptr = lfe; - } - } - /* remove the row from the problem */ - npp_del_row(npp, p); - return 0; -} - -static int rcv_eq_singlet(NPP *npp, void *_info) -{ /* recover row singleton (equality constraint) */ - struct eq_singlet *info = _info; - NPPLFE *lfe; - double temp; - if (npp->sol == GLP_SOL) - { /* column q must be already recovered as GLP_NS */ - if (npp->c_stat[info->q] != GLP_NS) - { npp_error(); - return 1; - } - npp->r_stat[info->p] = GLP_NS; - npp->c_stat[info->q] = GLP_BS; - } - if (npp->sol != GLP_MIP) - { /* compute multiplier for row p with formula (3) */ - temp = info->c; - for (lfe = info->ptr; lfe != NULL; lfe = lfe->next) - temp -= lfe->val * npp->r_pi[lfe->ref]; - npp->r_pi[info->p] = temp / info->apq; - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_implied_lower - process implied column lower bound -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_implied_lower(NPP *npp, NPPCOL *q, double l); -* -* DESCRIPTION -* -* For column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* where l[q] < u[q], the routine npp_implied_lower processes its -* implied lower bound l'[q]. As the result the current column lower -* bound may increase. Note that the column is kept in the problem in -* any case. -* -* RETURNS -* -* 0 - current column lower bound has not changed; -* -* 1 - current column lower bound has changed, but not significantly; -* -* 2 - current column lower bound has significantly changed; -* -* 3 - column has been fixed on its upper bound; -* -* 4 - implied lower bound violates current column upper bound. -* -* ALGORITHM -* -* If column q is integral, before processing its implied lower bound -* should be rounded up: -* -* ( floor(l'[q]+0.5), if |l'[q] - floor(l'[q]+0.5)| <= eps -* l'[q] := < (2) -* ( ceil(l'[q]), otherwise -* -* where floor(l'[q]+0.5) is the nearest integer to l'[q], ceil(l'[q]) -* is smallest integer not less than l'[q], and eps is an absolute -* tolerance for column value. -* -* Processing implied column lower bound l'[q] includes the following -* cases: -* -* 1) if l'[q] < l[q] + eps, implied lower bound is redundant; -* -* 2) if l[q] + eps <= l[q] <= u[q] + eps, current column lower bound -* l[q] can be strengthened by replacing it with l'[q]. If in this -* case new column lower bound becomes close to current column upper -* bound u[q], the column can be fixed on its upper bound; -* -* 3) if l'[q] > u[q] + eps, implied lower bound violates current -* column upper bound u[q], in which case the problem has no primal -* feasible solution. */ - -int npp_implied_lower(NPP *npp, NPPCOL *q, double l) -{ /* process implied column lower bound */ - int ret; - double eps, nint; - xassert(npp == npp); - /* column must not be fixed */ - xassert(q->lb < q->ub); - /* implied lower bound must be finite */ - xassert(l != -DBL_MAX); - /* if column is integral, round up l'[q] */ - if (q->is_int) - { nint = floor(l + 0.5); - if (fabs(l - nint) <= 1e-5) - l = nint; - else - l = ceil(l); - } - /* check current column lower bound */ - if (q->lb != -DBL_MAX) - { eps = (q->is_int ? 1e-3 : 1e-3 + 1e-6 * fabs(q->lb)); - if (l < q->lb + eps) - { ret = 0; /* redundant */ - goto done; - } - } - /* check current column upper bound */ - if (q->ub != +DBL_MAX) - { eps = (q->is_int ? 1e-5 : 1e-5 + 1e-8 * fabs(q->ub)); - if (l > q->ub + eps) - { ret = 4; /* infeasible */ - goto done; - } - /* if l'[q] is close to u[q], fix column at its upper bound */ - if (l > q->ub - 1e-3 * eps) - { q->lb = q->ub; - ret = 3; /* fixed */ - goto done; - } - } - /* check if column lower bound changes significantly */ - if (q->lb == -DBL_MAX) - ret = 2; /* significantly */ - else if (q->is_int && l > q->lb + 0.5) - ret = 2; /* significantly */ - else if (l > q->lb + 0.30 * (1.0 + fabs(q->lb))) - ret = 2; /* significantly */ - else - ret = 1; /* not significantly */ - /* set new column lower bound */ - q->lb = l; -done: return ret; -} - -/*********************************************************************** -* NAME -* -* npp_implied_upper - process implied column upper bound -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_implied_upper(NPP *npp, NPPCOL *q, double u); -* -* DESCRIPTION -* -* For column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* where l[q] < u[q], the routine npp_implied_upper processes its -* implied upper bound u'[q]. As the result the current column upper -* bound may decrease. Note that the column is kept in the problem in -* any case. -* -* RETURNS -* -* 0 - current column upper bound has not changed; -* -* 1 - current column upper bound has changed, but not significantly; -* -* 2 - current column upper bound has significantly changed; -* -* 3 - column has been fixed on its lower bound; -* -* 4 - implied upper bound violates current column lower bound. -* -* ALGORITHM -* -* If column q is integral, before processing its implied upper bound -* should be rounded down: -* -* ( floor(u'[q]+0.5), if |u'[q] - floor(l'[q]+0.5)| <= eps -* u'[q] := < (2) -* ( floor(l'[q]), otherwise -* -* where floor(u'[q]+0.5) is the nearest integer to u'[q], -* floor(u'[q]) is largest integer not greater than u'[q], and eps is -* an absolute tolerance for column value. -* -* Processing implied column upper bound u'[q] includes the following -* cases: -* -* 1) if u'[q] > u[q] - eps, implied upper bound is redundant; -* -* 2) if l[q] - eps <= u[q] <= u[q] - eps, current column upper bound -* u[q] can be strengthened by replacing it with u'[q]. If in this -* case new column upper bound becomes close to current column lower -* bound, the column can be fixed on its lower bound; -* -* 3) if u'[q] < l[q] - eps, implied upper bound violates current -* column lower bound l[q], in which case the problem has no primal -* feasible solution. */ - -int npp_implied_upper(NPP *npp, NPPCOL *q, double u) -{ int ret; - double eps, nint; - xassert(npp == npp); - /* column must not be fixed */ - xassert(q->lb < q->ub); - /* implied upper bound must be finite */ - xassert(u != +DBL_MAX); - /* if column is integral, round down u'[q] */ - if (q->is_int) - { nint = floor(u + 0.5); - if (fabs(u - nint) <= 1e-5) - u = nint; - else - u = floor(u); - } - /* check current column upper bound */ - if (q->ub != +DBL_MAX) - { eps = (q->is_int ? 1e-3 : 1e-3 + 1e-6 * fabs(q->ub)); - if (u > q->ub - eps) - { ret = 0; /* redundant */ - goto done; - } - } - /* check current column lower bound */ - if (q->lb != -DBL_MAX) - { eps = (q->is_int ? 1e-5 : 1e-5 + 1e-8 * fabs(q->lb)); - if (u < q->lb - eps) - { ret = 4; /* infeasible */ - goto done; - } - /* if u'[q] is close to l[q], fix column at its lower bound */ - if (u < q->lb + 1e-3 * eps) - { q->ub = q->lb; - ret = 3; /* fixed */ - goto done; - } - } - /* check if column upper bound changes significantly */ - if (q->ub == +DBL_MAX) - ret = 2; /* significantly */ - else if (q->is_int && u < q->ub - 0.5) - ret = 2; /* significantly */ - else if (u < q->ub - 0.30 * (1.0 + fabs(q->ub))) - ret = 2; /* significantly */ - else - ret = 1; /* not significantly */ - /* set new column upper bound */ - q->ub = u; -done: return ret; -} - -/*********************************************************************** -* NAME -* -* npp_ineq_singlet - process row singleton (inequality constraint) -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_ineq_singlet(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_ineq_singlet processes row p, which is inequality -* constraint having the only non-zero coefficient: -* -* L[p] <= a[p,q] * x[q] <= U[p], (1) -* -* where L[p] < U[p], L[p] > -oo and/or U[p] < +oo. -* -* RETURNS -* -* 0 - current column bounds have not changed; -* -* 1 - current column bounds have changed, but not significantly; -* -* 2 - current column bounds have significantly changed; -* -* 3 - column has been fixed on its lower or upper bound; -* -* 4 - problem has no primal feasible solution. -* -* PROBLEM TRANSFORMATION -* -* Inequality constraint (1) defines implied bounds of column q: -* -* ( L[p] / a[p,q], if a[p,q] > 0 -* l'[q] = < (2) -* ( U[p] / a[p,q], if a[p,q] < 0 -* -* ( U[p] / a[p,q], if a[p,q] > 0 -* u'[q] = < (3) -* ( L[p] / a[p,q], if a[p,q] < 0 -* -* If these implied bounds do not violate current bounds of column q: -* -* l[q] <= x[q] <= u[q], (4) -* -* they can be used to strengthen the current column bounds: -* -* l[q] := max(l[q], l'[q]), (5) -* -* u[q] := min(u[q], u'[q]). (6) -* -* (See the routines npp_implied_lower and npp_implied_upper.) -* -* Once bounds of row p (1) have been carried over column q, the row -* becomes redundant, so it can be replaced by equivalent free row and -* removed from the problem. -* -* Note that the routine removes from the problem only row p. Column q, -* even it has been fixed, is kept in the problem. -* -* RECOVERING BASIC SOLUTION -* -* Note that the row in the dual system corresponding to column q is -* the following: -* -* sum a[i,q] pi[i] + lambda[q] = c[q] ==> -* i -* (7) -* sum a[i,q] pi[i] + a[p,q] pi[p] + lambda[q] = c[q], -* i!=p -* -* where pi[i] for all i != p are known in solution to the transformed -* problem. Row p does not exist in the transformed problem, so it has -* zero multiplier there. This allows computing multiplier for column q -* in solution to the transformed problem: -* -* lambda~[q] = c[q] - sum a[i,q] pi[i]. (8) -* i!=p -* -* Let in solution to the transformed problem column q be non-basic -* with lower bound active (GLP_NL, lambda~[q] >= 0), and this lower -* bound be implied one l'[q]. From the original problem's standpoint -* this then means that actually the original column lower bound l[q] -* is inactive, and active is that row bound L[p] or U[p] that defines -* the implied bound l'[q] (2). In this case in solution to the -* original problem column q is assigned status GLP_BS while row p is -* assigned status GLP_NL (if a[p,q] > 0) or GLP_NU (if a[p,q] < 0). -* Since now column q is basic, its multiplier lambda[q] is zero. This -* allows using (7) and (8) to find multiplier for row p in solution to -* the original problem: -* -* 1 -* pi[p] = ------ (c[q] - sum a[i,q] pi[i]) = lambda~[q] / a[p,q] (9) -* a[p,q] i!=p -* -* Now let in solution to the transformed problem column q be non-basic -* with upper bound active (GLP_NU, lambda~[q] <= 0), and this upper -* bound be implied one u'[q]. As in the previous case this then means -* that from the original problem's standpoint actually the original -* column upper bound u[q] is inactive, and active is that row bound -* L[p] or U[p] that defines the implied bound u'[q] (3). In this case -* in solution to the original problem column q is assigned status -* GLP_BS, row p is assigned status GLP_NU (if a[p,q] > 0) or GLP_NL -* (if a[p,q] < 0), and its multiplier is computed with formula (9). -* -* Strengthening bounds of column q according to (5) and (6) may make -* it fixed. Thus, if in solution to the transformed problem column q is -* non-basic and fixed (GLP_NS), we can suppose that if lambda~[q] > 0, -* column q has active lower bound (GLP_NL), and if lambda~[q] < 0, -* column q has active upper bound (GLP_NU), reducing this case to two -* previous ones. If, however, lambda~[q] is close to zero or -* corresponding bound of row p does not exist (this may happen if -* lambda~[q] has wrong sign due to round-off errors, in which case it -* is expected to be close to zero, since solution is assumed to be dual -* feasible), column q can be assigned status GLP_BS (basic), and row p -* can be made active on its existing bound. In the latter case row -* multiplier pi[p] computed with formula (9) will be also close to -* zero, and dual feasibility will be kept. -* -* In all other cases, namely, if in solution to the transformed -* problem column q is basic (GLP_BS), or non-basic with original lower -* bound l[q] active (GLP_NL), or non-basic with original upper bound -* u[q] active (GLP_NU), constraint (1) is inactive. So in solution to -* the original problem status of column q remains unchanged, row p is -* assigned status GLP_BS, and its multiplier pi[p] is assigned zero -* value. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* First, value of multiplier for column q in solution to the original -* problem is computed with formula (8). If lambda~[q] > 0 and column q -* has implied lower bound, or if lambda~[q] < 0 and column q has -* implied upper bound, this means that from the original problem's -* standpoint actually row p has corresponding active bound, in which -* case its multiplier pi[p] is computed with formula (9). In other -* cases, when the sign of lambda~[q] corresponds to original bound of -* column q, or when lambda~[q] =~ 0, value of row multiplier pi[p] is -* assigned zero value. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct ineq_singlet -{ /* row singleton (inequality constraint) */ - int p; - /* row reference number */ - int q; - /* column reference number */ - double apq; - /* constraint coefficient a[p,q] */ - double c; - /* objective coefficient at x[q] */ - double lb; - /* row lower bound */ - double ub; - /* row upper bound */ - char lb_changed; - /* this flag is set if column lower bound was changed */ - char ub_changed; - /* this flag is set if column upper bound was changed */ - NPPLFE *ptr; - /* list of non-zero coefficients a[i,q], i != p */ -}; - -static int rcv_ineq_singlet(NPP *npp, void *info); - -int npp_ineq_singlet(NPP *npp, NPPROW *p) -{ /* process row singleton (inequality constraint) */ - struct ineq_singlet *info; - NPPCOL *q; - NPPAIJ *apq, *aij; - NPPLFE *lfe; - int lb_changed, ub_changed; - double ll, uu; - /* the row must be singleton inequality constraint */ - xassert(p->lb != -DBL_MAX || p->ub != +DBL_MAX); - xassert(p->lb < p->ub); - xassert(p->ptr != NULL && p->ptr->r_next == NULL); - /* compute implied column bounds */ - apq = p->ptr; - q = apq->col; - xassert(q->lb < q->ub); - if (apq->val > 0.0) - { ll = (p->lb == -DBL_MAX ? -DBL_MAX : p->lb / apq->val); - uu = (p->ub == +DBL_MAX ? +DBL_MAX : p->ub / apq->val); - } - else - { ll = (p->ub == +DBL_MAX ? -DBL_MAX : p->ub / apq->val); - uu = (p->lb == -DBL_MAX ? +DBL_MAX : p->lb / apq->val); - } - /* process implied column lower bound */ - if (ll == -DBL_MAX) - lb_changed = 0; - else - { lb_changed = npp_implied_lower(npp, q, ll); - xassert(0 <= lb_changed && lb_changed <= 4); - if (lb_changed == 4) return 4; /* infeasible */ - } - /* process implied column upper bound */ - if (uu == +DBL_MAX) - ub_changed = 0; - else if (lb_changed == 3) - { /* column was fixed on its upper bound due to l'[q] = u[q] */ - /* note that L[p] < U[p], so l'[q] = u[q] < u'[q] */ - ub_changed = 0; - } - else - { ub_changed = npp_implied_upper(npp, q, uu); - xassert(0 <= ub_changed && ub_changed <= 4); - if (ub_changed == 4) return 4; /* infeasible */ - } - /* if neither lower nor upper column bound was changed, the row - is originally redundant and can be replaced by free row */ - if (!lb_changed && !ub_changed) - { p->lb = -DBL_MAX, p->ub = +DBL_MAX; - npp_free_row(npp, p); - return 0; - } - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_ineq_singlet, sizeof(struct ineq_singlet)); - info->p = p->i; - info->q = q->j; - info->apq = apq->val; - info->c = q->coef; - info->lb = p->lb; - info->ub = p->ub; - info->lb_changed = (char)lb_changed; - info->ub_changed = (char)ub_changed; - info->ptr = NULL; - /* save column coefficients a[i,q], i != p (not needed for MIP - solution) */ - if (npp->sol != GLP_MIP) - { for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { if (aij == apq) continue; /* skip a[p,q] */ - lfe = dmp_get_atom(npp->stack, sizeof(NPPLFE)); - lfe->ref = aij->row->i; - lfe->val = aij->val; - lfe->next = info->ptr; - info->ptr = lfe; - } - } - /* remove the row from the problem */ - npp_del_row(npp, p); - return lb_changed >= ub_changed ? lb_changed : ub_changed; -} - -static int rcv_ineq_singlet(NPP *npp, void *_info) -{ /* recover row singleton (inequality constraint) */ - struct ineq_singlet *info = _info; - NPPLFE *lfe; - double lambda; - if (npp->sol == GLP_MIP) goto done; - /* compute lambda~[q] in solution to the transformed problem - with formula (8) */ - lambda = info->c; - for (lfe = info->ptr; lfe != NULL; lfe = lfe->next) - lambda -= lfe->val * npp->r_pi[lfe->ref]; - if (npp->sol == GLP_SOL) - { /* recover basic solution */ - if (npp->c_stat[info->q] == GLP_BS) - { /* column q is basic, so row p is inactive */ - npp->r_stat[info->p] = GLP_BS; - npp->r_pi[info->p] = 0.0; - } - else if (npp->c_stat[info->q] == GLP_NL) -nl: { /* column q is non-basic with lower bound active */ - if (info->lb_changed) - { /* it is implied bound, so actually row p is active - while column q is basic */ - npp->r_stat[info->p] = - (char)(info->apq > 0.0 ? GLP_NL : GLP_NU); - npp->c_stat[info->q] = GLP_BS; - npp->r_pi[info->p] = lambda / info->apq; - } - else - { /* it is original bound, so row p is inactive */ - npp->r_stat[info->p] = GLP_BS; - npp->r_pi[info->p] = 0.0; - } - } - else if (npp->c_stat[info->q] == GLP_NU) -nu: { /* column q is non-basic with upper bound active */ - if (info->ub_changed) - { /* it is implied bound, so actually row p is active - while column q is basic */ - npp->r_stat[info->p] = - (char)(info->apq > 0.0 ? GLP_NU : GLP_NL); - npp->c_stat[info->q] = GLP_BS; - npp->r_pi[info->p] = lambda / info->apq; - } - else - { /* it is original bound, so row p is inactive */ - npp->r_stat[info->p] = GLP_BS; - npp->r_pi[info->p] = 0.0; - } - } - else if (npp->c_stat[info->q] == GLP_NS) - { /* column q is non-basic and fixed; note, however, that in - in the original problem it is non-fixed */ - if (lambda > +1e-7) - { if (info->apq > 0.0 && info->lb != -DBL_MAX || - info->apq < 0.0 && info->ub != +DBL_MAX || - !info->lb_changed) - { /* either corresponding bound of row p exists or - column q remains non-basic with its original lower - bound active */ - npp->c_stat[info->q] = GLP_NL; - goto nl; - } - } - if (lambda < -1e-7) - { if (info->apq > 0.0 && info->ub != +DBL_MAX || - info->apq < 0.0 && info->lb != -DBL_MAX || - !info->ub_changed) - { /* either corresponding bound of row p exists or - column q remains non-basic with its original upper - bound active */ - npp->c_stat[info->q] = GLP_NU; - goto nu; - } - } - /* either lambda~[q] is close to zero, or corresponding - bound of row p does not exist, because lambda~[q] has - wrong sign due to round-off errors; in the latter case - lambda~[q] is also assumed to be close to zero; so, we - can make row p active on its existing bound and column q - basic; pi[p] will have wrong sign, but it also will be - close to zero (rarus casus of dual degeneracy) */ - if (info->lb != -DBL_MAX && info->ub == +DBL_MAX) - { /* row lower bound exists, but upper bound doesn't */ - npp->r_stat[info->p] = GLP_NL; - } - else if (info->lb == -DBL_MAX && info->ub != +DBL_MAX) - { /* row upper bound exists, but lower bound doesn't */ - npp->r_stat[info->p] = GLP_NU; - } - else if (info->lb != -DBL_MAX && info->ub != +DBL_MAX) - { /* both row lower and upper bounds exist */ - /* to choose proper active row bound we should not use - lambda~[q], because its value being close to zero is - unreliable; so we choose that bound which provides - primal feasibility for original constraint (1) */ - if (info->apq * npp->c_value[info->q] <= - 0.5 * (info->lb + info->ub)) - npp->r_stat[info->p] = GLP_NL; - else - npp->r_stat[info->p] = GLP_NU; - } - else - { npp_error(); - return 1; - } - npp->c_stat[info->q] = GLP_BS; - npp->r_pi[info->p] = lambda / info->apq; - } - else - { npp_error(); - return 1; - } - } - if (npp->sol == GLP_IPT) - { /* recover interior-point solution */ - if (lambda > +DBL_EPSILON && info->lb_changed || - lambda < -DBL_EPSILON && info->ub_changed) - { /* actually row p has corresponding active bound */ - npp->r_pi[info->p] = lambda / info->apq; - } - else - { /* either bounds of column q are both inactive or its - original bound is active */ - npp->r_pi[info->p] = 0.0; - } - } -done: return 0; -} - -/*********************************************************************** -* NAME -* -* npp_implied_slack - process column singleton (implied slack variable) -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_implied_slack(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_implied_slack processes column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* where l[q] < u[q], having the only non-zero coefficient in row p, -* which is equality constraint: -* -* sum a[p,j] x[j] + a[p,q] x[q] = b. (2) -* j!=q -* -* PROBLEM TRANSFORMATION -* -* (If x[q] is integral, this transformation must not be used.) -* -* The term a[p,q] x[q] in constraint (2) can be considered as a slack -* variable that allows to carry bounds of column q over row p and then -* remove column q from the problem. -* -* Constraint (2) can be written as follows: -* -* sum a[p,j] x[j] = b - a[p,q] x[q]. (3) -* j!=q -* -* According to (1) constraint (3) is equivalent to the following -* inequality constraint: -* -* L[p] <= sum a[p,j] x[j] <= U[p], (4) -* j!=q -* -* where -* -* ( b - a[p,q] u[q], if a[p,q] > 0 -* L[p] = < (5) -* ( b - a[p,q] l[q], if a[p,q] < 0 -* -* ( b - a[p,q] l[q], if a[p,q] > 0 -* U[p] = < (6) -* ( b - a[p,q] u[q], if a[p,q] < 0 -* -* From (2) it follows that: -* -* 1 -* x[q] = ------ (b - sum a[p,j] x[j]). (7) -* a[p,q] j!=q -* -* In order to eliminate x[q] from the objective row we substitute it -* from (6) to that row: -* -* z = sum c[j] x[j] + c[q] x[q] + c[0] = -* j!=q -* 1 -* = sum c[j] x[j] + c[q] [------ (b - sum a[p,j] x[j])] + c0 = -* j!=q a[p,q] j!=q -* -* = sum c~[j] x[j] + c~[0], -* j!=q -* a[p,j] b -* c~[j] = c[j] - c[q] ------, c~0 = c0 - c[q] ------ (8) -* a[p,q] a[p,q] -* -* are values of objective coefficients and constant term, resp., in -* the transformed problem. -* -* Note that column q is column singleton, so in the dual system of the -* original problem it corresponds to the following row singleton: -* -* a[p,q] pi[p] + lambda[q] = c[q]. (9) -* -* In the transformed problem row (9) would be the following: -* -* a[p,q] pi~[p] + lambda[q] = c~[q] = 0. (10) -* -* Subtracting (10) from (9) we have: -* -* a[p,q] (pi[p] - pi~[p]) = c[q] -* -* that gives the following formula to compute multiplier for row p in -* solution to the original problem using its value in solution to the -* transformed problem: -* -* pi[p] = pi~[p] + c[q] / a[p,q]. (11) -* -* RECOVERING BASIC SOLUTION -* -* Status of column q in solution to the original problem is defined -* by status of row p in solution to the transformed problem and the -* sign of coefficient a[p,q] in the original inequality constraint (2) -* as follows: -* -* +-----------------------+---------+--------------------+ -* | Status of row p | Sign of | Status of column q | -* | (transformed problem) | a[p,q] | (original problem) | -* +-----------------------+---------+--------------------+ -* | GLP_BS | + / - | GLP_BS | -* | GLP_NL | + | GLP_NU | -* | GLP_NL | - | GLP_NL | -* | GLP_NU | + | GLP_NL | -* | GLP_NU | - | GLP_NU | -* | GLP_NF | + / - | GLP_NF | -* +-----------------------+---------+--------------------+ -* -* Value of column q is computed with formula (7). Since originally row -* p is equality constraint, its status is assigned GLP_NS, and value of -* its multiplier pi[p] is computed with formula (11). -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of column q is computed with formula (7). Row multiplier value -* pi[p] is computed with formula (11). -* -* RECOVERING MIP SOLUTION -* -* Value of column q is computed with formula (7). */ - -struct implied_slack -{ /* column singleton (implied slack variable) */ - int p; - /* row reference number */ - int q; - /* column reference number */ - double apq; - /* constraint coefficient a[p,q] */ - double b; - /* right-hand side of original equality constraint */ - double c; - /* original objective coefficient at x[q] */ - NPPLFE *ptr; - /* list of non-zero coefficients a[p,j], j != q */ -}; - -static int rcv_implied_slack(NPP *npp, void *info); - -void npp_implied_slack(NPP *npp, NPPCOL *q) -{ /* process column singleton (implied slack variable) */ - struct implied_slack *info; - NPPROW *p; - NPPAIJ *aij; - NPPLFE *lfe; - /* the column must be non-integral non-fixed singleton */ - xassert(!q->is_int); - xassert(q->lb < q->ub); - xassert(q->ptr != NULL && q->ptr->c_next == NULL); - /* corresponding row must be equality constraint */ - aij = q->ptr; - p = aij->row; - xassert(p->lb == p->ub); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_implied_slack, sizeof(struct implied_slack)); - info->p = p->i; - info->q = q->j; - info->apq = aij->val; - info->b = p->lb; - info->c = q->coef; - info->ptr = NULL; - /* save row coefficients a[p,j], j != q, and substitute x[q] - into the objective row */ - for (aij = p->ptr; aij != NULL; aij = aij->r_next) - { if (aij->col == q) continue; /* skip a[p,q] */ - lfe = dmp_get_atom(npp->stack, sizeof(NPPLFE)); - lfe->ref = aij->col->j; - lfe->val = aij->val; - lfe->next = info->ptr; - info->ptr = lfe; - aij->col->coef -= info->c * (aij->val / info->apq); - } - npp->c0 += info->c * (info->b / info->apq); - /* compute new row bounds */ - if (info->apq > 0.0) - { p->lb = (q->ub == +DBL_MAX ? - -DBL_MAX : info->b - info->apq * q->ub); - p->ub = (q->lb == -DBL_MAX ? - +DBL_MAX : info->b - info->apq * q->lb); - } - else - { p->lb = (q->lb == -DBL_MAX ? - -DBL_MAX : info->b - info->apq * q->lb); - p->ub = (q->ub == +DBL_MAX ? - +DBL_MAX : info->b - info->apq * q->ub); - } - /* remove the column from the problem */ - npp_del_col(npp, q); - return; -} - -static int rcv_implied_slack(NPP *npp, void *_info) -{ /* recover column singleton (implied slack variable) */ - struct implied_slack *info = _info; - NPPLFE *lfe; - double temp; - if (npp->sol == GLP_SOL) - { /* assign statuses to row p and column q */ - if (npp->r_stat[info->p] == GLP_BS || - npp->r_stat[info->p] == GLP_NF) - npp->c_stat[info->q] = npp->r_stat[info->p]; - else if (npp->r_stat[info->p] == GLP_NL) - npp->c_stat[info->q] = - (char)(info->apq > 0.0 ? GLP_NU : GLP_NL); - else if (npp->r_stat[info->p] == GLP_NU) - npp->c_stat[info->q] = - (char)(info->apq > 0.0 ? GLP_NL : GLP_NU); - else - { npp_error(); - return 1; - } - npp->r_stat[info->p] = GLP_NS; - } - if (npp->sol != GLP_MIP) - { /* compute multiplier for row p */ - npp->r_pi[info->p] += info->c / info->apq; - } - /* compute value of column q */ - temp = info->b; - for (lfe = info->ptr; lfe != NULL; lfe = lfe->next) - temp -= lfe->val * npp->c_value[lfe->ref]; - npp->c_value[info->q] = temp / info->apq; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_implied_free - process column singleton (implied free variable) -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_implied_free(NPP *npp, NPPCOL *q); -* -* DESCRIPTION -* -* The routine npp_implied_free processes column q: -* -* l[q] <= x[q] <= u[q], (1) -* -* having non-zero coefficient in the only row p, which is inequality -* constraint: -* -* L[p] <= sum a[p,j] x[j] + a[p,q] x[q] <= U[p], (2) -* j!=q -* -* where l[q] < u[q], L[p] < U[p], L[p] > -oo and/or U[p] < +oo. -* -* RETURNS -* -* 0 - success; -* -* 1 - column lower and/or upper bound(s) can be active; -* -* 2 - problem has no dual feasible solution. -* -* PROBLEM TRANSFORMATION -* -* Constraint (2) can be written as follows: -* -* L[p] - sum a[p,j] x[j] <= a[p,q] x[q] <= U[p] - sum a[p,j] x[j], -* j!=q j!=q -* -* from which it follows that: -* -* alfa <= a[p,q] x[q] <= beta, (3) -* -* where -* -* alfa = inf(L[p] - sum a[p,j] x[j]) = -* j!=q -* -* = L[p] - sup sum a[p,j] x[j] = (4) -* j!=q -* -* = L[p] - sum a[p,j] u[j] - sum a[p,j] l[j], -* j in Jp j in Jn -* -* beta = sup(L[p] - sum a[p,j] x[j]) = -* j!=q -* -* = L[p] - inf sum a[p,j] x[j] = (5) -* j!=q -* -* = L[p] - sum a[p,j] l[j] - sum a[p,j] u[j], -* j in Jp j in Jn -* -* Jp = {j != q: a[p,j] > 0}, Jn = {j != q: a[p,j] < 0}. (6) -* -* Inequality (3) defines implied bounds of variable x[q]: -* -* l'[q] <= x[q] <= u'[q], (7) -* -* where -* -* ( alfa / a[p,q], if a[p,q] > 0 -* l'[q] = < (8a) -* ( beta / a[p,q], if a[p,q] < 0 -* -* ( beta / a[p,q], if a[p,q] > 0 -* u'[q] = < (8b) -* ( alfa / a[p,q], if a[p,q] < 0 -* -* Thus, if l'[q] > l[q] - eps and u'[q] < u[q] + eps, where eps is -* an absolute tolerance for column value, column bounds (1) cannot be -* active, in which case column q can be replaced by equivalent free -* (unbounded) column. -* -* Note that column q is column singleton, so in the dual system of the -* original problem it corresponds to the following row singleton: -* -* a[p,q] pi[p] + lambda[q] = c[q], (9) -* -* from which it follows that: -* -* pi[p] = (c[q] - lambda[q]) / a[p,q]. (10) -* -* Let x[q] be implied free (unbounded) variable. Then column q can be -* only basic, so its multiplier lambda[q] is equal to zero, and from -* (10) we have: -* -* pi[p] = c[q] / a[p,q]. (11) -* -* There are possible three cases: -* -* 1) pi[p] < -eps, where eps is an absolute tolerance for row -* multiplier. In this case, to provide dual feasibility of the -* original problem, row p must be active on its lower bound, and -* if its lower bound does not exist (L[p] = -oo), the problem has -* no dual feasible solution; -* -* 2) pi[p] > +eps. In this case row p must be active on its upper -* bound, and if its upper bound does not exist (U[p] = +oo), the -* problem has no dual feasible solution; -* -* 3) -eps <= pi[p] <= +eps. In this case any (either lower or upper) -* bound of row p can be active, because this does not affect dual -* feasibility. -* -* Thus, in all three cases original inequality constraint (2) can be -* replaced by equality constraint, where the right-hand side is either -* lower or upper bound of row p, and bounds of column q can be removed -* that makes it free (unbounded). (May note that this transformation -* can be followed by transformation "Column singleton (implied slack -* variable)" performed by the routine npp_implied_slack.) -* -* RECOVERING BASIC SOLUTION -* -* Status of row p in solution to the original problem is determined -* by its status in solution to the transformed problem and its bound, -* which was choosen to be active: -* -* +-----------------------+--------+--------------------+ -* | Status of row p | Active | Status of row p | -* | (transformed problem) | bound | (original problem) | -* +-----------------------+--------+--------------------+ -* | GLP_BS | L[p] | GLP_BS | -* | GLP_BS | U[p] | GLP_BS | -* | GLP_NS | L[p] | GLP_NL | -* | GLP_NS | U[p] | GLP_NU | -* +-----------------------+--------+--------------------+ -* -* Value of row multiplier pi[p] (as well as value of column q) in -* solution to the original problem is the same as in solution to the -* transformed problem. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of row multiplier pi[p] in solution to the original problem is -* the same as in solution to the transformed problem. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct implied_free -{ /* column singleton (implied free variable) */ - int p; - /* row reference number */ - char stat; - /* row status: - GLP_NL - active constraint on lower bound - GLP_NU - active constraint on upper bound */ -}; - -static int rcv_implied_free(NPP *npp, void *info); - -int npp_implied_free(NPP *npp, NPPCOL *q) -{ /* process column singleton (implied free variable) */ - struct implied_free *info; - NPPROW *p; - NPPAIJ *apq, *aij; - double alfa, beta, l, u, pi, eps; - /* the column must be non-fixed singleton */ - xassert(q->lb < q->ub); - xassert(q->ptr != NULL && q->ptr->c_next == NULL); - /* corresponding row must be inequality constraint */ - apq = q->ptr; - p = apq->row; - xassert(p->lb != -DBL_MAX || p->ub != +DBL_MAX); - xassert(p->lb < p->ub); - /* compute alfa */ - alfa = p->lb; - if (alfa != -DBL_MAX) - { for (aij = p->ptr; aij != NULL; aij = aij->r_next) - { if (aij == apq) continue; /* skip a[p,q] */ - if (aij->val > 0.0) - { if (aij->col->ub == +DBL_MAX) - { alfa = -DBL_MAX; - break; - } - alfa -= aij->val * aij->col->ub; - } - else /* < 0.0 */ - { if (aij->col->lb == -DBL_MAX) - { alfa = -DBL_MAX; - break; - } - alfa -= aij->val * aij->col->lb; - } - } - } - /* compute beta */ - beta = p->ub; - if (beta != +DBL_MAX) - { for (aij = p->ptr; aij != NULL; aij = aij->r_next) - { if (aij == apq) continue; /* skip a[p,q] */ - if (aij->val > 0.0) - { if (aij->col->lb == -DBL_MAX) - { beta = +DBL_MAX; - break; - } - beta -= aij->val * aij->col->lb; - } - else /* < 0.0 */ - { if (aij->col->ub == +DBL_MAX) - { beta = +DBL_MAX; - break; - } - beta -= aij->val * aij->col->ub; - } - } - } - /* compute implied column lower bound l'[q] */ - if (apq->val > 0.0) - l = (alfa == -DBL_MAX ? -DBL_MAX : alfa / apq->val); - else /* < 0.0 */ - l = (beta == +DBL_MAX ? -DBL_MAX : beta / apq->val); - /* compute implied column upper bound u'[q] */ - if (apq->val > 0.0) - u = (beta == +DBL_MAX ? +DBL_MAX : beta / apq->val); - else - u = (alfa == -DBL_MAX ? +DBL_MAX : alfa / apq->val); - /* check if column lower bound l[q] can be active */ - if (q->lb != -DBL_MAX) - { eps = 1e-9 + 1e-12 * fabs(q->lb); - if (l < q->lb - eps) return 1; /* yes, it can */ - } - /* check if column upper bound u[q] can be active */ - if (q->ub != +DBL_MAX) - { eps = 1e-9 + 1e-12 * fabs(q->ub); - if (u > q->ub + eps) return 1; /* yes, it can */ - } - /* okay; make column q free (unbounded) */ - q->lb = -DBL_MAX, q->ub = +DBL_MAX; - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_implied_free, sizeof(struct implied_free)); - info->p = p->i; - info->stat = -1; - /* compute row multiplier pi[p] */ - pi = q->coef / apq->val; - /* check dual feasibility for row p */ - if (pi > +DBL_EPSILON) - { /* lower bound L[p] must be active */ - if (p->lb != -DBL_MAX) -nl: { info->stat = GLP_NL; - p->ub = p->lb; - } - else - { if (pi > +1e-5) return 2; /* dual infeasibility */ - /* take a chance on U[p] */ - xassert(p->ub != +DBL_MAX); - goto nu; - } - } - else if (pi < -DBL_EPSILON) - { /* upper bound U[p] must be active */ - if (p->ub != +DBL_MAX) -nu: { info->stat = GLP_NU; - p->lb = p->ub; - } - else - { if (pi < -1e-5) return 2; /* dual infeasibility */ - /* take a chance on L[p] */ - xassert(p->lb != -DBL_MAX); - goto nl; - } - } - else - { /* any bound (either L[p] or U[p]) can be made active */ - if (p->ub == +DBL_MAX) - { xassert(p->lb != -DBL_MAX); - goto nl; - } - if (p->lb == -DBL_MAX) - { xassert(p->ub != +DBL_MAX); - goto nu; - } - if (fabs(p->lb) <= fabs(p->ub)) goto nl; else goto nu; - } - return 0; -} - -static int rcv_implied_free(NPP *npp, void *_info) -{ /* recover column singleton (implied free variable) */ - struct implied_free *info = _info; - if (npp->sol == GLP_SOL) - { if (npp->r_stat[info->p] == GLP_BS) - npp->r_stat[info->p] = GLP_BS; - else if (npp->r_stat[info->p] == GLP_NS) - { xassert(info->stat == GLP_NL || info->stat == GLP_NU); - npp->r_stat[info->p] = info->stat; - } - else - { npp_error(); - return 1; - } - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_eq_doublet - process row doubleton (equality constraint) -* -* SYNOPSIS -* -* #include "glpnpp.h" -* NPPCOL *npp_eq_doublet(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_eq_doublet processes row p, which is equality -* constraint having exactly two non-zero coefficients: -* -* a[p,q] x[q] + a[p,r] x[r] = b. (1) -* -* As the result of processing one of columns q or r is eliminated from -* all other rows and, thus, becomes column singleton of type "implied -* slack variable". Row p is not changed and along with column q and r -* remains in the problem. -* -* RETURNS -* -* The routine npp_eq_doublet returns pointer to the descriptor of that -* column q or r which has been eliminated. If, due to some reason, the -* elimination was not performed, the routine returns NULL. -* -* PROBLEM TRANSFORMATION -* -* First, we decide which column q or r will be eliminated. Let it be -* column q. Consider i-th constraint row, where column q has non-zero -* coefficient a[i,q] != 0: -* -* L[i] <= sum a[i,j] x[j] <= U[i]. (2) -* j -* -* In order to eliminate column q from row (2) we subtract from it row -* (1) multiplied by gamma[i] = a[i,q] / a[p,q], i.e. we replace in the -* transformed problem row (2) by its linear combination with row (1). -* This transformation changes only coefficients in columns q and r, -* and bounds of row i as follows: -* -* a~[i,q] = a[i,q] - gamma[i] a[p,q] = 0, (3) -* -* a~[i,r] = a[i,r] - gamma[i] a[p,r], (4) -* -* L~[i] = L[i] - gamma[i] b, (5) -* -* U~[i] = U[i] - gamma[i] b. (6) -* -* RECOVERING BASIC SOLUTION -* -* The transformation of the primal system of the original problem: -* -* L <= A x <= U (7) -* -* is equivalent to multiplying from the left a transformation matrix F -* by components of this primal system, which in the transformed problem -* becomes the following: -* -* F L <= F A x <= F U ==> L~ <= A~x <= U~. (8) -* -* The matrix F has the following structure: -* -* ( 1 -gamma[1] ) -* ( ) -* ( 1 -gamma[2] ) -* ( ) -* ( ... ... ) -* ( ) -* F = ( 1 -gamma[p-1] ) (9) -* ( ) -* ( 1 ) -* ( ) -* ( -gamma[p+1] 1 ) -* ( ) -* ( ... ... ) -* -* where its column containing elements -gamma[i] corresponds to row p -* of the primal system. -* -* From (8) it follows that the dual system of the original problem: -* -* A'pi + lambda = c, (10) -* -* in the transformed problem becomes the following: -* -* A'F'inv(F')pi + lambda = c ==> (A~)'pi~ + lambda = c, (11) -* -* where: -* -* pi~ = inv(F')pi (12) -* -* is the vector of row multipliers in the transformed problem. Thus: -* -* pi = F'pi~. (13) -* -* Therefore, as it follows from (13), value of multiplier for row p in -* solution to the original problem can be computed as follows: -* -* pi[p] = pi~[p] - sum gamma[i] pi~[i], (14) -* i -* -* where pi~[i] = pi[i] is multiplier for row i (i != p). -* -* Note that the statuses of all rows and columns are not changed. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Multiplier for row p in solution to the original problem is computed -* with formula (14). -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct eq_doublet -{ /* row doubleton (equality constraint) */ - int p; - /* row reference number */ - double apq; - /* constraint coefficient a[p,q] */ - NPPLFE *ptr; - /* list of non-zero coefficients a[i,q], i != p */ -}; - -static int rcv_eq_doublet(NPP *npp, void *info); - -NPPCOL *npp_eq_doublet(NPP *npp, NPPROW *p) -{ /* process row doubleton (equality constraint) */ - struct eq_doublet *info; - NPPROW *i; - NPPCOL *q, *r; - NPPAIJ *apq, *apr, *aiq, *air, *next; - NPPLFE *lfe; - double gamma; - /* the row must be doubleton equality constraint */ - xassert(p->lb == p->ub); - xassert(p->ptr != NULL && p->ptr->r_next != NULL && - p->ptr->r_next->r_next == NULL); - /* choose column to be eliminated */ - { NPPAIJ *a1, *a2; - a1 = p->ptr, a2 = a1->r_next; - if (fabs(a2->val) < 0.001 * fabs(a1->val)) - { /* only first column can be eliminated, because second one - has too small constraint coefficient */ - apq = a1, apr = a2; - } - else if (fabs(a1->val) < 0.001 * fabs(a2->val)) - { /* only second column can be eliminated, because first one - has too small constraint coefficient */ - apq = a2, apr = a1; - } - else - { /* both columns are appropriate; choose that one which is - shorter to minimize fill-in */ - if (npp_col_nnz(npp, a1->col) <= npp_col_nnz(npp, a2->col)) - { /* first column is shorter */ - apq = a1, apr = a2; - } - else - { /* second column is shorter */ - apq = a2, apr = a1; - } - } - } - /* now columns q and r have been chosen */ - q = apq->col, r = apr->col; - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_eq_doublet, sizeof(struct eq_doublet)); - info->p = p->i; - info->apq = apq->val; - info->ptr = NULL; - /* transform each row i (i != p), where a[i,q] != 0, to eliminate - column q */ - for (aiq = q->ptr; aiq != NULL; aiq = next) - { next = aiq->c_next; - if (aiq == apq) continue; /* skip row p */ - i = aiq->row; /* row i to be transformed */ - /* save constraint coefficient a[i,q] */ - if (npp->sol != GLP_MIP) - { lfe = dmp_get_atom(npp->stack, sizeof(NPPLFE)); - lfe->ref = i->i; - lfe->val = aiq->val; - lfe->next = info->ptr; - info->ptr = lfe; - } - /* find coefficient a[i,r] in row i */ - for (air = i->ptr; air != NULL; air = air->r_next) - if (air->col == r) break; - /* if a[i,r] does not exist, create a[i,r] = 0 */ - if (air == NULL) - air = npp_add_aij(npp, i, r, 0.0); - /* compute gamma[i] = a[i,q] / a[p,q] */ - gamma = aiq->val / apq->val; - /* (row i) := (row i) - gamma[i] * (row p); see (3)-(6) */ - /* new a[i,q] is exact zero due to elimnation; remove it from - row i */ - npp_del_aij(npp, aiq); - /* compute new a[i,r] */ - air->val -= gamma * apr->val; - /* if new a[i,r] is close to zero due to numeric cancelation, - remove it from row i */ - if (fabs(air->val) <= 1e-10) - npp_del_aij(npp, air); - /* compute new lower and upper bounds of row i */ - if (i->lb == i->ub) - i->lb = i->ub = (i->lb - gamma * p->lb); - else - { if (i->lb != -DBL_MAX) - i->lb -= gamma * p->lb; - if (i->ub != +DBL_MAX) - i->ub -= gamma * p->lb; - } - } - return q; -} - -static int rcv_eq_doublet(NPP *npp, void *_info) -{ /* recover row doubleton (equality constraint) */ - struct eq_doublet *info = _info; - NPPLFE *lfe; - double gamma, temp; - /* we assume that processing row p is followed by processing - column q as singleton of type "implied slack variable", in - which case row p must always be active equality constraint */ - if (npp->sol == GLP_SOL) - { if (npp->r_stat[info->p] != GLP_NS) - { npp_error(); - return 1; - } - } - if (npp->sol != GLP_MIP) - { /* compute value of multiplier for row p; see (14) */ - temp = npp->r_pi[info->p]; - for (lfe = info->ptr; lfe != NULL; lfe = lfe->next) - { gamma = lfe->val / info->apq; /* a[i,q] / a[p,q] */ - temp -= gamma * npp->r_pi[lfe->ref]; - } - npp->r_pi[info->p] = temp; - } - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_forcing_row - process forcing row -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_forcing_row(NPP *npp, NPPROW *p, int at); -* -* DESCRIPTION -* -* The routine npp_forcing row processes row p of general format: -* -* L[p] <= sum a[p,j] x[j] <= U[p], (1) -* j -* -* l[j] <= x[j] <= u[j], (2) -* -* where L[p] <= U[p] and l[j] < u[j] for all a[p,j] != 0. It is also -* assumed that: -* -* 1) if at = 0 then |L[p] - U'[p]| <= eps, where U'[p] is implied -* row upper bound (see below), eps is an absolute tolerance for row -* value; -* -* 2) if at = 1 then |U[p] - L'[p]| <= eps, where L'[p] is implied -* row lower bound (see below). -* -* RETURNS -* -* 0 - success; -* -* 1 - cannot fix columns due to too small constraint coefficients. -* -* PROBLEM TRANSFORMATION -* -* Implied lower and upper bounds of row (1) are determined by bounds -* of corresponding columns (variables) as follows: -* -* L'[p] = inf sum a[p,j] x[j] = -* j -* (3) -* = sum a[p,j] l[j] + sum a[p,j] u[j], -* j in Jp j in Jn -* -* U'[p] = sup sum a[p,j] x[j] = -* (4) -* = sum a[p,j] u[j] + sum a[p,j] l[j], -* j in Jp j in Jn -* -* Jp = {j: a[p,j] > 0}, Jn = {j: a[p,j] < 0}. (5) -* -* If L[p] =~ U'[p] (at = 0), solution can be primal feasible only when -* all variables take their boundary values as defined by (4): -* -* ( u[j], if j in Jp -* x[j] = < (6) -* ( l[j], if j in Jn -* -* Similarly, if U[p] =~ L'[p] (at = 1), solution can be primal feasible -* only when all variables take their boundary values as defined by (3): -* -* ( l[j], if j in Jp -* x[j] = < (7) -* ( u[j], if j in Jn -* -* Condition (6) or (7) allows fixing all columns (variables x[j]) -* in row (1) on their bounds and then removing them from the problem -* (see the routine npp_fixed_col). Due to this row p becomes redundant, -* so it can be replaced by equivalent free (unbounded) row and also -* removed from the problem (see the routine npp_free_row). -* -* 1. To apply this transformation row (1) should not have coefficients -* whose magnitude is too small, i.e. all a[p,j] should satisfy to -* the following condition: -* -* |a[p,j]| >= eps * max(1, |a[p,k]|), (8) -* k -* where eps is a relative tolerance for constraint coefficients. -* Otherwise, fixing columns may be numerically unreliable and may -* lead to wrong solution. -* -* 2. The routine fixes columns and remove bounds of row p, however, -* it does not remove the row and columns from the problem. -* -* RECOVERING BASIC SOLUTION -* -* In the transformed problem row p being inactive constraint is -* assigned status GLP_BS (as the result of transformation of free -* row), and all columns in this row are assigned status GLP_NS (as the -* result of transformation of fixed columns). -* -* Note that in the dual system of the transformed (as well as original) -* problem every column j in row p corresponds to the following row: -* -* sum a[i,j] pi[i] + a[p,j] pi[p] + lambda[j] = c[j], (9) -* i!=p -* -* from which it follows that: -* -* lambda[j] = c[j] - sum a[i,j] pi[i] - a[p,j] pi[p]. (10) -* i!=p -* -* In the transformed problem values of all multipliers pi[i] are known -* (including pi[i], whose value is zero, since row p is inactive). -* Thus, using formula (10) it is possible to compute values of -* multipliers lambda[j] for all columns in row p. -* -* Note also that in the original problem all columns in row p are -* bounded, not fixed. So status GLP_NS assigned to every such column -* must be changed to GLP_NL or GLP_NU depending on which bound the -* corresponding column has been fixed. This status change may lead to -* dual feasibility violation for solution of the original problem, -* because now column multipliers must satisfy to the following -* condition: -* -* ( >= 0, if status of column j is GLP_NL, -* lambda[j] < (11) -* ( <= 0, if status of column j is GLP_NU. -* -* If this condition holds, solution to the original problem is the -* same as to the transformed problem. Otherwise, we have to perform -* one degenerate pivoting step of the primal simplex method to obtain -* dual feasible (hence, optimal) solution to the original problem as -* follows. If, on problem transformation, row p was made active on its -* lower bound (case at = 0), we change its status to GLP_NL (or GLP_NS) -* and start increasing its multiplier pi[p]. Otherwise, if row p was -* made active on its upper bound (case at = 1), we change its status -* to GLP_NU (or GLP_NS) and start decreasing pi[p]. From (10) it -* follows that: -* -* delta lambda[j] = - a[p,j] * delta pi[p] = - a[p,j] pi[p]. (12) -* -* Simple analysis of formulae (3)-(5) shows that changing pi[p] in the -* specified direction causes increasing lambda[j] for every column j -* assigned status GLP_NL (delta lambda[j] > 0) and decreasing lambda[j] -* for every column j assigned status GLP_NU (delta lambda[j] < 0). It -* is understood that once the last lambda[q], which violates condition -* (11), has reached zero, multipliers lambda[j] for all columns get -* valid signs. Such column q can be determined as follows. Let d[j] be -* initial value of lambda[j] (i.e. reduced cost of column j) in the -* transformed problem computed with formula (10) when pi[p] = 0. Then -* lambda[j] = d[j] + delta lambda[j], and from (12) it follows that -* lambda[j] becomes zero if: -* -* delta lambda[j] = - a[p,j] pi[p] = - d[j] ==> -* (13) -* pi[p] = d[j] / a[p,j]. -* -* Therefore, the last column q, for which lambda[q] becomes zero, can -* be determined from the following condition: -* -* |d[q] / a[p,q]| = max |pi[p]| = max |d[j] / a[p,j]|, (14) -* j in D j in D -* -* where D is a set of columns j whose, reduced costs d[j] have invalid -* signs, i.e. violate condition (11). (Thus, if D is empty, solution -* to the original problem is the same as solution to the transformed -* problem, and no correction is needed as was noticed above.) In -* solution to the original problem column q is assigned status GLP_BS, -* since it replaces column of auxiliary variable of row p (becoming -* active) in the basis, and multiplier for row p is assigned its new -* value, which is pi[p] = d[q] / a[p,q]. Note that due to primal -* degeneracy values of all columns having non-zero coefficients in row -* p remain unchanged. -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* Value of multiplier pi[p] in solution to the original problem is -* corrected in the same way as for basic solution. Values of all -* columns having non-zero coefficients in row p remain unchanged. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct forcing_col -{ /* column fixed on its bound by forcing row */ - int j; - /* column reference number */ - char stat; - /* original column status: - GLP_NL - fixed on lower bound - GLP_NU - fixed on upper bound */ - double a; - /* constraint coefficient a[p,j] */ - double c; - /* objective coefficient c[j] */ - NPPLFE *ptr; - /* list of non-zero coefficients a[i,j], i != p */ - struct forcing_col *next; - /* pointer to another column fixed by forcing row */ -}; - -struct forcing_row -{ /* forcing row */ - int p; - /* row reference number */ - char stat; - /* status assigned to the row if it becomes active: - GLP_NS - active equality constraint - GLP_NL - inequality constraint with lower bound active - GLP_NU - inequality constraint with upper bound active */ - struct forcing_col *ptr; - /* list of all columns having non-zero constraint coefficient - a[p,j] in the forcing row */ -}; - -static int rcv_forcing_row(NPP *npp, void *info); - -int npp_forcing_row(NPP *npp, NPPROW *p, int at) -{ /* process forcing row */ - struct forcing_row *info; - struct forcing_col *col = NULL; - NPPCOL *j; - NPPAIJ *apj, *aij; - NPPLFE *lfe; - double big; - xassert(at == 0 || at == 1); - /* determine maximal magnitude of the row coefficients */ - big = 1.0; - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - if (big < fabs(apj->val)) big = fabs(apj->val); - /* if there are too small coefficients in the row, transformation - should not be applied */ - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - if (fabs(apj->val) < 1e-7 * big) return 1; - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_forcing_row, sizeof(struct forcing_row)); - info->p = p->i; - if (p->lb == p->ub) - { /* equality constraint */ - info->stat = GLP_NS; - } - else if (at == 0) - { /* inequality constraint; case L[p] = U'[p] */ - info->stat = GLP_NL; - xassert(p->lb != -DBL_MAX); - } - else /* at == 1 */ - { /* inequality constraint; case U[p] = L'[p] */ - info->stat = GLP_NU; - xassert(p->ub != +DBL_MAX); - } - info->ptr = NULL; - /* scan the forcing row, fix columns at corresponding bounds, and - save column information (the latter is not needed for MIP) */ - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { /* column j has non-zero coefficient in the forcing row */ - j = apj->col; - /* it must be non-fixed */ - xassert(j->lb < j->ub); - /* allocate stack entry to save column information */ - if (npp->sol != GLP_MIP) - { col = dmp_get_atom(npp->stack, sizeof(struct forcing_col)); - col->j = j->j; - col->stat = -1; /* will be set below */ - col->a = apj->val; - col->c = j->coef; - col->ptr = NULL; - col->next = info->ptr; - info->ptr = col; - } - /* fix column j */ - if (at == 0 && apj->val < 0.0 || at != 0 && apj->val > 0.0) - { /* at its lower bound */ - if (npp->sol != GLP_MIP) - col->stat = GLP_NL; - xassert(j->lb != -DBL_MAX); - j->ub = j->lb; - } - else - { /* at its upper bound */ - if (npp->sol != GLP_MIP) - col->stat = GLP_NU; - xassert(j->ub != +DBL_MAX); - j->lb = j->ub; - } - /* save column coefficients a[i,j], i != p */ - if (npp->sol != GLP_MIP) - { for (aij = j->ptr; aij != NULL; aij = aij->c_next) - { if (aij == apj) continue; /* skip a[p,j] */ - lfe = dmp_get_atom(npp->stack, sizeof(NPPLFE)); - lfe->ref = aij->row->i; - lfe->val = aij->val; - lfe->next = col->ptr; - col->ptr = lfe; - } - } - } - /* make the row free (unbounded) */ - p->lb = -DBL_MAX, p->ub = +DBL_MAX; - return 0; -} - -static int rcv_forcing_row(NPP *npp, void *_info) -{ /* recover forcing row */ - struct forcing_row *info = _info; - struct forcing_col *col, *piv; - NPPLFE *lfe; - double d, big, temp; - if (npp->sol == GLP_MIP) goto done; - /* initially solution to the original problem is the same as - to the transformed problem, where row p is inactive constraint - with pi[p] = 0, and all columns are non-basic */ - if (npp->sol == GLP_SOL) - { if (npp->r_stat[info->p] != GLP_BS) - { npp_error(); - return 1; - } - for (col = info->ptr; col != NULL; col = col->next) - { if (npp->c_stat[col->j] != GLP_NS) - { npp_error(); - return 1; - } - npp->c_stat[col->j] = col->stat; /* original status */ - } - } - /* compute reduced costs d[j] for all columns with formula (10) - and store them in col.c instead objective coefficients */ - for (col = info->ptr; col != NULL; col = col->next) - { d = col->c; - for (lfe = col->ptr; lfe != NULL; lfe = lfe->next) - d -= lfe->val * npp->r_pi[lfe->ref]; - col->c = d; - } - /* consider columns j, whose multipliers lambda[j] has wrong - sign in solution to the transformed problem (where lambda[j] = - d[j]), and choose column q, whose multipler lambda[q] reaches - zero last on changing row multiplier pi[p]; see (14) */ - piv = NULL, big = 0.0; - for (col = info->ptr; col != NULL; col = col->next) - { d = col->c; /* d[j] */ - temp = fabs(d / col->a); - if (col->stat == GLP_NL) - { /* column j has active lower bound */ - if (d < 0.0 && big < temp) - piv = col, big = temp; - } - else if (col->stat == GLP_NU) - { /* column j has active upper bound */ - if (d > 0.0 && big < temp) - piv = col, big = temp; - } - else - { npp_error(); - return 1; - } - } - /* if column q does not exist, no correction is needed */ - if (piv != NULL) - { /* correct solution; row p becomes active constraint while - column q becomes basic */ - if (npp->sol == GLP_SOL) - { npp->r_stat[info->p] = info->stat; - npp->c_stat[piv->j] = GLP_BS; - } - /* assign new value to row multiplier pi[p] = d[p] / a[p,q] */ - npp->r_pi[info->p] = piv->c / piv->a; - } -done: return 0; -} - -/*********************************************************************** -* NAME -* -* npp_analyze_row - perform general row analysis -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_analyze_row(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_analyze_row performs analysis of row p of general -* format: -* -* L[p] <= sum a[p,j] x[j] <= U[p], (1) -* j -* -* l[j] <= x[j] <= u[j], (2) -* -* where L[p] <= U[p] and l[j] <= u[j] for all a[p,j] != 0. -* -* RETURNS -* -* 0x?0 - row lower bound does not exist or is redundant; -* -* 0x?1 - row lower bound can be active; -* -* 0x?2 - row lower bound is a forcing bound; -* -* 0x0? - row upper bound does not exist or is redundant; -* -* 0x1? - row upper bound can be active; -* -* 0x2? - row upper bound is a forcing bound; -* -* 0x33 - row bounds are inconsistent with column bounds. -* -* ALGORITHM -* -* Analysis of row (1) is based on analysis of its implied lower and -* upper bounds, which are determined by bounds of corresponding columns -* (variables) as follows: -* -* L'[p] = inf sum a[p,j] x[j] = -* j -* (3) -* = sum a[p,j] l[j] + sum a[p,j] u[j], -* j in Jp j in Jn -* -* U'[p] = sup sum a[p,j] x[j] = -* (4) -* = sum a[p,j] u[j] + sum a[p,j] l[j], -* j in Jp j in Jn -* -* Jp = {j: a[p,j] > 0}, Jn = {j: a[p,j] < 0}. (5) -* -* (Note that bounds of all columns in row p are assumed to be correct, -* so L'[p] <= U'[p].) -* -* Analysis of row lower bound L[p] includes the following cases: -* -* 1) if L[p] > U'[p] + eps, where eps is an absolute tolerance for row -* value, row lower bound L[p] and implied row upper bound U'[p] are -* inconsistent, ergo, the problem has no primal feasible solution; -* -* 2) if U'[p] - eps <= L[p] <= U'[p] + eps, i.e. if L[p] =~ U'[p], -* the row is a forcing row on its lower bound (see description of -* the routine npp_forcing_row); -* -* 3) if L[p] > L'[p] + eps, row lower bound L[p] can be active (this -* conclusion does not account other rows in the problem); -* -* 4) if L[p] <= L'[p] + eps, row lower bound L[p] cannot be active, so -* it is redundant and can be removed (replaced by -oo). -* -* Analysis of row upper bound U[p] is performed in a similar way and -* includes the following cases: -* -* 1) if U[p] < L'[p] - eps, row upper bound U[p] and implied row lower -* bound L'[p] are inconsistent, ergo the problem has no primal -* feasible solution; -* -* 2) if L'[p] - eps <= U[p] <= L'[p] + eps, i.e. if U[p] =~ L'[p], -* the row is a forcing row on its upper bound (see description of -* the routine npp_forcing_row); -* -* 3) if U[p] < U'[p] - eps, row upper bound U[p] can be active (this -* conclusion does not account other rows in the problem); -* -* 4) if U[p] >= U'[p] - eps, row upper bound U[p] cannot be active, so -* it is redundant and can be removed (replaced by +oo). */ - -int npp_analyze_row(NPP *npp, NPPROW *p) -{ /* perform general row analysis */ - NPPAIJ *aij; - int ret = 0x00; - double l, u, eps; - xassert(npp == npp); - /* compute implied lower bound L'[p]; see (3) */ - l = 0.0; - for (aij = p->ptr; aij != NULL; aij = aij->r_next) - { if (aij->val > 0.0) - { if (aij->col->lb == -DBL_MAX) - { l = -DBL_MAX; - break; - } - l += aij->val * aij->col->lb; - } - else /* aij->val < 0.0 */ - { if (aij->col->ub == +DBL_MAX) - { l = -DBL_MAX; - break; - } - l += aij->val * aij->col->ub; - } - } - /* compute implied upper bound U'[p]; see (4) */ - u = 0.0; - for (aij = p->ptr; aij != NULL; aij = aij->r_next) - { if (aij->val > 0.0) - { if (aij->col->ub == +DBL_MAX) - { u = +DBL_MAX; - break; - } - u += aij->val * aij->col->ub; - } - else /* aij->val < 0.0 */ - { if (aij->col->lb == -DBL_MAX) - { u = +DBL_MAX; - break; - } - u += aij->val * aij->col->lb; - } - } - /* column bounds are assumed correct, so L'[p] <= U'[p] */ - /* check if row lower bound is consistent */ - if (p->lb != -DBL_MAX) - { eps = 1e-3 + 1e-6 * fabs(p->lb); - if (p->lb - eps > u) - { ret = 0x33; - goto done; - } - } - /* check if row upper bound is consistent */ - if (p->ub != +DBL_MAX) - { eps = 1e-3 + 1e-6 * fabs(p->ub); - if (p->ub + eps < l) - { ret = 0x33; - goto done; - } - } - /* check if row lower bound can be active/forcing */ - if (p->lb != -DBL_MAX) - { eps = 1e-9 + 1e-12 * fabs(p->lb); - if (p->lb - eps > l) - { if (p->lb + eps <= u) - ret |= 0x01; - else - ret |= 0x02; - } - } - /* check if row upper bound can be active/forcing */ - if (p->ub != +DBL_MAX) - { eps = 1e-9 + 1e-12 * fabs(p->ub); - if (p->ub + eps < u) - { /* check if the upper bound is forcing */ - if (p->ub - eps >= l) - ret |= 0x10; - else - ret |= 0x20; - } - } -done: return ret; -} - -/*********************************************************************** -* NAME -* -* npp_inactive_bound - remove row lower/upper inactive bound -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_inactive_bound(NPP *npp, NPPROW *p, int which); -* -* DESCRIPTION -* -* The routine npp_inactive_bound removes lower (if which = 0) or upper -* (if which = 1) bound of row p: -* -* L[p] <= sum a[p,j] x[j] <= U[p], -* -* which (bound) is assumed to be redundant. -* -* PROBLEM TRANSFORMATION -* -* If which = 0, current lower bound L[p] of row p is assigned -oo. -* If which = 1, current upper bound U[p] of row p is assigned +oo. -* -* RECOVERING BASIC SOLUTION -* -* If in solution to the transformed problem row p is inactive -* constraint (GLP_BS), its status is not changed in solution to the -* original problem. Otherwise, status of row p in solution to the -* original problem is defined by its type before transformation and -* its status in solution to the transformed problem as follows: -* -* +---------------------+-------+---------------+---------------+ -* | Row | Flag | Row status in | Row status in | -* | type | which | transfmd soln | original soln | -* +---------------------+-------+---------------+---------------+ -* | sum >= L[p] | 0 | GLP_NF | GLP_NL | -* | sum <= U[p] | 1 | GLP_NF | GLP_NU | -* | L[p] <= sum <= U[p] | 0 | GLP_NU | GLP_NU | -* | L[p] <= sum <= U[p] | 1 | GLP_NL | GLP_NL | -* | sum = L[p] = U[p] | 0 | GLP_NU | GLP_NS | -* | sum = L[p] = U[p] | 1 | GLP_NL | GLP_NS | -* +---------------------+-------+---------------+---------------+ -* -* RECOVERING INTERIOR-POINT SOLUTION -* -* None needed. -* -* RECOVERING MIP SOLUTION -* -* None needed. */ - -struct inactive_bound -{ /* row inactive bound */ - int p; - /* row reference number */ - char stat; - /* row status (if active constraint) */ -}; - -static int rcv_inactive_bound(NPP *npp, void *info); - -void npp_inactive_bound(NPP *npp, NPPROW *p, int which) -{ /* remove row lower/upper inactive bound */ - struct inactive_bound *info; - if (npp->sol == GLP_SOL) - { /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_inactive_bound, sizeof(struct inactive_bound)); - info->p = p->i; - if (p->ub == +DBL_MAX) - info->stat = GLP_NL; - else if (p->lb == -DBL_MAX) - info->stat = GLP_NU; - else if (p->lb != p->ub) - info->stat = (char)(which == 0 ? GLP_NU : GLP_NL); - else - info->stat = GLP_NS; - } - /* remove row inactive bound */ - if (which == 0) - { xassert(p->lb != -DBL_MAX); - p->lb = -DBL_MAX; - } - else if (which == 1) - { xassert(p->ub != +DBL_MAX); - p->ub = +DBL_MAX; - } - else - xassert(which != which); - return; -} - -static int rcv_inactive_bound(NPP *npp, void *_info) -{ /* recover row status */ - struct inactive_bound *info = _info; - if (npp->sol != GLP_SOL) - { npp_error(); - return 1; - } - if (npp->r_stat[info->p] == GLP_BS) - npp->r_stat[info->p] = GLP_BS; - else - npp->r_stat[info->p] = info->stat; - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_implied_bounds - determine implied column bounds -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_implied_bounds(NPP *npp, NPPROW *p); -* -* DESCRIPTION -* -* The routine npp_implied_bounds inspects general row (constraint) p: -* -* L[p] <= sum a[p,j] x[j] <= U[p], (1) -* -* l[j] <= x[j] <= u[j], (2) -* -* where L[p] <= U[p] and l[j] <= u[j] for all a[p,j] != 0, to compute -* implied bounds of columns (variables x[j]) in this row. -* -* The routine stores implied column bounds l'[j] and u'[j] in column -* descriptors (NPPCOL); it does not change current column bounds l[j] -* and u[j]. (Implied column bounds can be then used to strengthen the -* current column bounds; see the routines npp_implied_lower and -* npp_implied_upper). -* -* ALGORITHM -* -* Current column bounds (2) define implied lower and upper bounds of -* row (1) as follows: -* -* L'[p] = inf sum a[p,j] x[j] = -* j -* (3) -* = sum a[p,j] l[j] + sum a[p,j] u[j], -* j in Jp j in Jn -* -* U'[p] = sup sum a[p,j] x[j] = -* (4) -* = sum a[p,j] u[j] + sum a[p,j] l[j], -* j in Jp j in Jn -* -* Jp = {j: a[p,j] > 0}, Jn = {j: a[p,j] < 0}. (5) -* -* (Note that bounds of all columns in row p are assumed to be correct, -* so L'[p] <= U'[p].) -* -* If L[p] > L'[p] and/or U[p] < U'[p], the lower and/or upper bound of -* row (1) can be active, in which case such row defines implied bounds -* of its variables. -* -* Let x[k] be some variable having in row (1) coefficient a[p,k] != 0. -* Consider a case when row lower bound can be active (L[p] > L'[p]): -* -* sum a[p,j] x[j] >= L[p] ==> -* j -* -* sum a[p,j] x[j] + a[p,k] x[k] >= L[p] ==> -* j!=k -* (6) -* a[p,k] x[k] >= L[p] - sum a[p,j] x[j] ==> -* j!=k -* -* a[p,k] x[k] >= L[p,k], -* -* where -* -* L[p,k] = inf(L[p] - sum a[p,j] x[j]) = -* j!=k -* -* = L[p] - sup sum a[p,j] x[j] = (7) -* j!=k -* -* = L[p] - sum a[p,j] u[j] - sum a[p,j] l[j]. -* j in Jp\{k} j in Jn\{k} -* -* Thus: -* -* x[k] >= l'[k] = L[p,k] / a[p,k], if a[p,k] > 0, (8) -* -* x[k] <= u'[k] = L[p,k] / a[p,k], if a[p,k] < 0. (9) -* -* where l'[k] and u'[k] are implied lower and upper bounds of variable -* x[k], resp. -* -* Now consider a similar case when row upper bound can be active -* (U[p] < U'[p]): -* -* sum a[p,j] x[j] <= U[p] ==> -* j -* -* sum a[p,j] x[j] + a[p,k] x[k] <= U[p] ==> -* j!=k -* (10) -* a[p,k] x[k] <= U[p] - sum a[p,j] x[j] ==> -* j!=k -* -* a[p,k] x[k] <= U[p,k], -* -* where: -* -* U[p,k] = sup(U[p] - sum a[p,j] x[j]) = -* j!=k -* -* = U[p] - inf sum a[p,j] x[j] = (11) -* j!=k -* -* = U[p] - sum a[p,j] l[j] - sum a[p,j] u[j]. -* j in Jp\{k} j in Jn\{k} -* -* Thus: -* -* x[k] <= u'[k] = U[p,k] / a[p,k], if a[p,k] > 0, (12) -* -* x[k] >= l'[k] = U[p,k] / a[p,k], if a[p,k] < 0. (13) -* -* Note that in formulae (8), (9), (12), and (13) coefficient a[p,k] -* must not be too small in magnitude relatively to other non-zero -* coefficients in row (1), i.e. the following condition must hold: -* -* |a[p,k]| >= eps * max(1, |a[p,j]|), (14) -* j -* -* where eps is a relative tolerance for constraint coefficients. -* Otherwise the implied column bounds can be numerical inreliable. For -* example, using formula (8) for the following inequality constraint: -* -* 1e-12 x1 - x2 - x3 >= 0, -* -* where x1 >= -1, x2, x3, >= 0, may lead to numerically unreliable -* conclusion that x1 >= 0. -* -* Using formulae (8), (9), (12), and (13) to compute implied bounds -* for one variable requires |J| operations, where J = {j: a[p,j] != 0}, -* because this needs computing L[p,k] and U[p,k]. Thus, computing -* implied bounds for all variables in row (1) would require |J|^2 -* operations, that is not a good technique. However, the total number -* of operations can be reduced to |J| as follows. -* -* Let a[p,k] > 0. Then from (7) and (11) we have: -* -* L[p,k] = L[p] - (U'[p] - a[p,k] u[k]) = -* -* = L[p] - U'[p] + a[p,k] u[k], -* -* U[p,k] = U[p] - (L'[p] - a[p,k] l[k]) = -* -* = U[p] - L'[p] + a[p,k] l[k], -* -* where L'[p] and U'[p] are implied row lower and upper bounds defined -* by formulae (3) and (4). Substituting these expressions into (8) and -* (12) gives: -* -* l'[k] = L[p,k] / a[p,k] = u[k] + (L[p] - U'[p]) / a[p,k], (15) -* -* u'[k] = U[p,k] / a[p,k] = l[k] + (U[p] - L'[p]) / a[p,k]. (16) -* -* Similarly, if a[p,k] < 0, according to (7) and (11) we have: -* -* L[p,k] = L[p] - (U'[p] - a[p,k] l[k]) = -* -* = L[p] - U'[p] + a[p,k] l[k], -* -* U[p,k] = U[p] - (L'[p] - a[p,k] u[k]) = -* -* = U[p] - L'[p] + a[p,k] u[k], -* -* and substituting these expressions into (8) and (12) gives: -* -* l'[k] = U[p,k] / a[p,k] = u[k] + (U[p] - L'[p]) / a[p,k], (17) -* -* u'[k] = L[p,k] / a[p,k] = l[k] + (L[p] - U'[p]) / a[p,k]. (18) -* -* Note that formulae (15)-(18) can be used only if L'[p] and U'[p] -* exist. However, if for some variable x[j] it happens that l[j] = -oo -* and/or u[j] = +oo, values of L'[p] (if a[p,j] > 0) and/or U'[p] (if -* a[p,j] < 0) are undefined. Consider, therefore, the most general -* situation, when some column bounds (2) may not exist. -* -* Let: -* -* J' = {j : (a[p,j] > 0 and l[j] = -oo) or -* (19) -* (a[p,j] < 0 and u[j] = +oo)}. -* -* Then (assuming that row upper bound U[p] can be active) the following -* three cases are possible: -* -* 1) |J'| = 0. In this case L'[p] exists, thus, for all variables x[j] -* in row (1) we can use formulae (16) and (17); -* -* 2) J' = {k}. In this case L'[p] = -oo, however, U[p,k] (11) exists, -* so for variable x[k] we can use formulae (12) and (13). Note that -* for all other variables x[j] (j != k) l'[j] = -oo (if a[p,j] < 0) -* or u'[j] = +oo (if a[p,j] > 0); -* -* 3) |J'| > 1. In this case for all variables x[j] in row [1] we have -* l'[j] = -oo (if a[p,j] < 0) or u'[j] = +oo (if a[p,j] > 0). -* -* Similarly, let: -* -* J'' = {j : (a[p,j] > 0 and u[j] = +oo) or -* (20) -* (a[p,j] < 0 and l[j] = -oo)}. -* -* Then (assuming that row lower bound L[p] can be active) the following -* three cases are possible: -* -* 1) |J''| = 0. In this case U'[p] exists, thus, for all variables x[j] -* in row (1) we can use formulae (15) and (18); -* -* 2) J'' = {k}. In this case U'[p] = +oo, however, L[p,k] (7) exists, -* so for variable x[k] we can use formulae (8) and (9). Note that -* for all other variables x[j] (j != k) l'[j] = -oo (if a[p,j] > 0) -* or u'[j] = +oo (if a[p,j] < 0); -* -* 3) |J''| > 1. In this case for all variables x[j] in row (1) we have -* l'[j] = -oo (if a[p,j] > 0) or u'[j] = +oo (if a[p,j] < 0). */ - -void npp_implied_bounds(NPP *npp, NPPROW *p) -{ NPPAIJ *apj, *apk; - double big, eps, temp; - xassert(npp == npp); - /* initialize implied bounds for all variables and determine - maximal magnitude of row coefficients a[p,j] */ - big = 1.0; - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { apj->col->ll.ll = -DBL_MAX, apj->col->uu.uu = +DBL_MAX; - if (big < fabs(apj->val)) big = fabs(apj->val); - } - eps = 1e-6 * big; - /* process row lower bound (assuming that it can be active) */ - if (p->lb != -DBL_MAX) - { apk = NULL; - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { if (apj->val > 0.0 && apj->col->ub == +DBL_MAX || - apj->val < 0.0 && apj->col->lb == -DBL_MAX) - { if (apk == NULL) - apk = apj; - else - goto skip1; - } - } - /* if a[p,k] = NULL then |J'| = 0 else J' = { k } */ - temp = p->lb; - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { if (apj == apk) - /* skip a[p,k] */; - else if (apj->val > 0.0) - temp -= apj->val * apj->col->ub; - else /* apj->val < 0.0 */ - temp -= apj->val * apj->col->lb; - } - /* compute column implied bounds */ - if (apk == NULL) - { /* temp = L[p] - U'[p] */ - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { if (apj->val >= +eps) - { /* l'[j] := u[j] + (L[p] - U'[p]) / a[p,j] */ - apj->col->ll.ll = apj->col->ub + temp / apj->val; - } - else if (apj->val <= -eps) - { /* u'[j] := l[j] + (L[p] - U'[p]) / a[p,j] */ - apj->col->uu.uu = apj->col->lb + temp / apj->val; - } - } - } - else - { /* temp = L[p,k] */ - if (apk->val >= +eps) - { /* l'[k] := L[p,k] / a[p,k] */ - apk->col->ll.ll = temp / apk->val; - } - else if (apk->val <= -eps) - { /* u'[k] := L[p,k] / a[p,k] */ - apk->col->uu.uu = temp / apk->val; - } - } -skip1: ; - } - /* process row upper bound (assuming that it can be active) */ - if (p->ub != +DBL_MAX) - { apk = NULL; - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { if (apj->val > 0.0 && apj->col->lb == -DBL_MAX || - apj->val < 0.0 && apj->col->ub == +DBL_MAX) - { if (apk == NULL) - apk = apj; - else - goto skip2; - } - } - /* if a[p,k] = NULL then |J''| = 0 else J'' = { k } */ - temp = p->ub; - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { if (apj == apk) - /* skip a[p,k] */; - else if (apj->val > 0.0) - temp -= apj->val * apj->col->lb; - else /* apj->val < 0.0 */ - temp -= apj->val * apj->col->ub; - } - /* compute column implied bounds */ - if (apk == NULL) - { /* temp = U[p] - L'[p] */ - for (apj = p->ptr; apj != NULL; apj = apj->r_next) - { if (apj->val >= +eps) - { /* u'[j] := l[j] + (U[p] - L'[p]) / a[p,j] */ - apj->col->uu.uu = apj->col->lb + temp / apj->val; - } - else if (apj->val <= -eps) - { /* l'[j] := u[j] + (U[p] - L'[p]) / a[p,j] */ - apj->col->ll.ll = apj->col->ub + temp / apj->val; - } - } - } - else - { /* temp = U[p,k] */ - if (apk->val >= +eps) - { /* u'[k] := U[p,k] / a[p,k] */ - apk->col->uu.uu = temp / apk->val; - } - else if (apk->val <= -eps) - { /* l'[k] := U[p,k] / a[p,k] */ - apk->col->ll.ll = temp / apk->val; - } - } -skip2: ; - } - return; -} - -/* eof */ diff --git a/code/3rd_glpk/npp/npp4.c b/code/3rd_glpk/npp/npp4.c deleted file mode 100644 index d7dd0e86..00000000 --- a/code/3rd_glpk/npp/npp4.c +++ /dev/null @@ -1,1414 +0,0 @@ -/* npp4.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -/*********************************************************************** -* NAME -* -* npp_binarize_prob - binarize MIP problem -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_binarize_prob(NPP *npp); -* -* DESCRIPTION -* -* The routine npp_binarize_prob replaces in the original MIP problem -* every integer variable: -* -* l[q] <= x[q] <= u[q], (1) -* -* where l[q] < u[q], by an equivalent sum of binary variables. -* -* RETURNS -* -* The routine returns the number of integer variables for which the -* transformation failed, because u[q] - l[q] > d_max. -* -* PROBLEM TRANSFORMATION -* -* If variable x[q] has non-zero lower bound, it is first processed -* with the routine npp_lbnd_col. Thus, we can assume that: -* -* 0 <= x[q] <= u[q]. (2) -* -* If u[q] = 1, variable x[q] is already binary, so further processing -* is not needed. Let, therefore, that 2 <= u[q] <= d_max, and n be a -* smallest integer such that u[q] <= 2^n - 1 (n >= 2, since u[q] >= 2). -* Then variable x[q] can be replaced by the following sum: -* -* n-1 -* x[q] = sum 2^k x[k], (3) -* k=0 -* -* where x[k] are binary columns (variables). If u[q] < 2^n - 1, the -* following additional inequality constraint must be also included in -* the transformed problem: -* -* n-1 -* sum 2^k x[k] <= u[q]. (4) -* k=0 -* -* Note: Assuming that in the transformed problem x[q] becomes binary -* variable x[0], this transformation causes new n-1 binary variables -* to appear. -* -* Substituting x[q] from (3) to the objective row gives: -* -* z = sum c[j] x[j] + c[0] = -* j -* -* = sum c[j] x[j] + c[q] x[q] + c[0] = -* j!=q -* n-1 -* = sum c[j] x[j] + c[q] sum 2^k x[k] + c[0] = -* j!=q k=0 -* n-1 -* = sum c[j] x[j] + sum c[k] x[k] + c[0], -* j!=q k=0 -* -* where: -* -* c[k] = 2^k c[q], k = 0, ..., n-1. (5) -* -* And substituting x[q] from (3) to i-th constraint row i gives: -* -* L[i] <= sum a[i,j] x[j] <= U[i] ==> -* j -* -* L[i] <= sum a[i,j] x[j] + a[i,q] x[q] <= U[i] ==> -* j!=q -* n-1 -* L[i] <= sum a[i,j] x[j] + a[i,q] sum 2^k x[k] <= U[i] ==> -* j!=q k=0 -* n-1 -* L[i] <= sum a[i,j] x[j] + sum a[i,k] x[k] <= U[i], -* j!=q k=0 -* -* where: -* -* a[i,k] = 2^k a[i,q], k = 0, ..., n-1. (6) -* -* RECOVERING SOLUTION -* -* Value of variable x[q] is computed with formula (3). */ - -struct binarize -{ int q; - /* column reference number for x[q] = x[0] */ - int j; - /* column reference number for x[1]; x[2] has reference number - j+1, x[3] - j+2, etc. */ - int n; - /* total number of binary variables, n >= 2 */ -}; - -static int rcv_binarize_prob(NPP *npp, void *info); - -int npp_binarize_prob(NPP *npp) -{ /* binarize MIP problem */ - struct binarize *info; - NPPROW *row; - NPPCOL *col, *bin; - NPPAIJ *aij; - int u, n, k, temp, nfails, nvars, nbins, nrows; - /* new variables will be added to the end of the column list, so - we go from the end to beginning of the column list */ - nfails = nvars = nbins = nrows = 0; - for (col = npp->c_tail; col != NULL; col = col->prev) - { /* skip continuous variable */ - if (!col->is_int) continue; - /* skip fixed variable */ - if (col->lb == col->ub) continue; - /* skip binary variable */ - if (col->lb == 0.0 && col->ub == 1.0) continue; - /* check if the transformation is applicable */ - if (col->lb < -1e6 || col->ub > +1e6 || - col->ub - col->lb > 4095.0) - { /* unfortunately, not */ - nfails++; - continue; - } - /* process integer non-binary variable x[q] */ - nvars++; - /* make x[q] non-negative, if its lower bound is non-zero */ - if (col->lb != 0.0) - npp_lbnd_col(npp, col); - /* now 0 <= x[q] <= u[q] */ - xassert(col->lb == 0.0); - u = (int)col->ub; - xassert(col->ub == (double)u); - /* if x[q] is binary, further processing is not needed */ - if (u == 1) continue; - /* determine smallest n such that u <= 2^n - 1 (thus, n is the - number of binary variables needed) */ - n = 2, temp = 4; - while (u >= temp) - n++, temp += temp; - nbins += n; - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_binarize_prob, sizeof(struct binarize)); - info->q = col->j; - info->j = 0; /* will be set below */ - info->n = n; - /* if u < 2^n - 1, we need one additional row for (4) */ - if (u < temp - 1) - { row = npp_add_row(npp), nrows++; - row->lb = -DBL_MAX, row->ub = u; - } - else - row = NULL; - /* in the transformed problem variable x[q] becomes binary - variable x[0], so its objective and constraint coefficients - are not changed */ - col->ub = 1.0; - /* include x[0] into constraint (4) */ - if (row != NULL) - npp_add_aij(npp, row, col, 1.0); - /* add other binary variables x[1], ..., x[n-1] */ - for (k = 1, temp = 2; k < n; k++, temp += temp) - { /* add new binary variable x[k] */ - bin = npp_add_col(npp); - bin->is_int = 1; - bin->lb = 0.0, bin->ub = 1.0; - bin->coef = (double)temp * col->coef; - /* store column reference number for x[1] */ - if (info->j == 0) - info->j = bin->j; - else - xassert(info->j + (k-1) == bin->j); - /* duplicate constraint coefficients for x[k]; this also - automatically includes x[k] into constraint (4) */ - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - npp_add_aij(npp, aij->row, bin, (double)temp * aij->val); - } - } - if (nvars > 0) - xprintf("%d integer variable(s) were replaced by %d binary one" - "s\n", nvars, nbins); - if (nrows > 0) - xprintf("%d row(s) were added due to binarization\n", nrows); - if (nfails > 0) - xprintf("Binarization failed for %d integer variable(s)\n", - nfails); - return nfails; -} - -static int rcv_binarize_prob(NPP *npp, void *_info) -{ /* recovery binarized variable */ - struct binarize *info = _info; - int k, temp; - double sum; - /* compute value of x[q]; see formula (3) */ - sum = npp->c_value[info->q]; - for (k = 1, temp = 2; k < info->n; k++, temp += temp) - sum += (double)temp * npp->c_value[info->j + (k-1)]; - npp->c_value[info->q] = sum; - return 0; -} - -/**********************************************************************/ - -struct elem -{ /* linear form element a[j] x[j] */ - double aj; - /* non-zero coefficient value */ - NPPCOL *xj; - /* pointer to variable (column) */ - struct elem *next; - /* pointer to another term */ -}; - -static struct elem *copy_form(NPP *npp, NPPROW *row, double s) -{ /* copy linear form */ - NPPAIJ *aij; - struct elem *ptr, *e; - ptr = NULL; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { e = dmp_get_atom(npp->pool, sizeof(struct elem)); - e->aj = s * aij->val; - e->xj = aij->col; - e->next = ptr; - ptr = e; - } - return ptr; -} - -static void drop_form(NPP *npp, struct elem *ptr) -{ /* drop linear form */ - struct elem *e; - while (ptr != NULL) - { e = ptr; - ptr = e->next; - dmp_free_atom(npp->pool, e, sizeof(struct elem)); - } - return; -} - -/*********************************************************************** -* NAME -* -* npp_is_packing - test if constraint is packing inequality -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_is_packing(NPP *npp, NPPROW *row); -* -* RETURNS -* -* If the specified row (constraint) is packing inequality (see below), -* the routine npp_is_packing returns non-zero. Otherwise, it returns -* zero. -* -* PACKING INEQUALITIES -* -* In canonical format the packing inequality is the following: -* -* sum x[j] <= 1, (1) -* j in J -* -* where all variables x[j] are binary. This inequality expresses the -* condition that in any integer feasible solution at most one variable -* from set J can take non-zero (unity) value while other variables -* must be equal to zero. W.l.o.g. it is assumed that |J| >= 2, because -* if J is empty or |J| = 1, the inequality (1) is redundant. -* -* In general case the packing inequality may include original variables -* x[j] as well as their complements x~[j]: -* -* sum x[j] + sum x~[j] <= 1, (2) -* j in Jp j in Jn -* -* where Jp and Jn are not intersected. Therefore, using substitution -* x~[j] = 1 - x[j] gives the packing inequality in generalized format: -* -* sum x[j] - sum x[j] <= 1 - |Jn|. (3) -* j in Jp j in Jn */ - -int npp_is_packing(NPP *npp, NPPROW *row) -{ /* test if constraint is packing inequality */ - NPPCOL *col; - NPPAIJ *aij; - int b; - xassert(npp == npp); - if (!(row->lb == -DBL_MAX && row->ub != +DBL_MAX)) - return 0; - b = 1; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col = aij->col; - if (!(col->is_int && col->lb == 0.0 && col->ub == 1.0)) - return 0; - if (aij->val == +1.0) - ; - else if (aij->val == -1.0) - b--; - else - return 0; - } - if (row->ub != (double)b) return 0; - return 1; -} - -/*********************************************************************** -* NAME -* -* npp_hidden_packing - identify hidden packing inequality -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_hidden_packing(NPP *npp, NPPROW *row); -* -* DESCRIPTION -* -* The routine npp_hidden_packing processes specified inequality -* constraint, which includes only binary variables, and the number of -* the variables is not less than two. If the original inequality is -* equivalent to a packing inequality, the routine replaces it by this -* equivalent inequality. If the original constraint is double-sided -* inequality, it is replaced by a pair of single-sided inequalities, -* if necessary. -* -* RETURNS -* -* If the original inequality constraint was replaced by equivalent -* packing inequality, the routine npp_hidden_packing returns non-zero. -* Otherwise, it returns zero. -* -* PROBLEM TRANSFORMATION -* -* Consider an inequality constraint: -* -* sum a[j] x[j] <= b, (1) -* j in J -* -* where all variables x[j] are binary, and |J| >= 2. (In case of '>=' -* inequality it can be transformed to '<=' format by multiplying both -* its sides by -1.) -* -* Let Jp = {j: a[j] > 0}, Jn = {j: a[j] < 0}. Performing substitution -* x[j] = 1 - x~[j] for all j in Jn, we have: -* -* sum a[j] x[j] <= b ==> -* j in J -* -* sum a[j] x[j] + sum a[j] x[j] <= b ==> -* j in Jp j in Jn -* -* sum a[j] x[j] + sum a[j] (1 - x~[j]) <= b ==> -* j in Jp j in Jn -* -* sum a[j] x[j] - sum a[j] x~[j] <= b - sum a[j]. -* j in Jp j in Jn j in Jn -* -* Thus, meaning the transformation above, we can assume that in -* inequality (1) all coefficients a[j] are positive. Moreover, we can -* assume that a[j] <= b. In fact, let a[j] > b; then the following -* three cases are possible: -* -* 1) b < 0. In this case inequality (1) is infeasible, so the problem -* has no feasible solution (see the routine npp_analyze_row); -* -* 2) b = 0. In this case inequality (1) is a forcing inequality on its -* upper bound (see the routine npp_forcing row), from which it -* follows that all variables x[j] should be fixed at zero; -* -* 3) b > 0. In this case inequality (1) defines an implied zero upper -* bound for variable x[j] (see the routine npp_implied_bounds), from -* which it follows that x[j] should be fixed at zero. -* -* It is assumed that all three cases listed above have been recognized -* by the routine npp_process_prob, which performs basic MIP processing -* prior to a call the routine npp_hidden_packing. So, if one of these -* cases occurs, we should just skip processing such constraint. -* -* Thus, let 0 < a[j] <= b. Then it is obvious that constraint (1) is -* equivalent to packing inquality only if: -* -* a[j] + a[k] > b + eps (2) -* -* for all j, k in J, j != k, where eps is an absolute tolerance for -* row (linear form) value. Checking the condition (2) for all j and k, -* j != k, requires time O(|J|^2). However, this time can be reduced to -* O(|J|), if use minimal a[j] and a[k], in which case it is sufficient -* to check the condition (2) only once. -* -* Once the original inequality (1) is replaced by equivalent packing -* inequality, we need to perform back substitution x~[j] = 1 - x[j] for -* all j in Jn (see above). -* -* RECOVERING SOLUTION -* -* None needed. */ - -static int hidden_packing(NPP *npp, struct elem *ptr, double *_b) -{ /* process inequality constraint: sum a[j] x[j] <= b; - 0 - specified row is NOT hidden packing inequality; - 1 - specified row is packing inequality; - 2 - specified row is hidden packing inequality. */ - struct elem *e, *ej, *ek; - int neg; - double b = *_b, eps; - xassert(npp == npp); - /* a[j] must be non-zero, x[j] must be binary, for all j in J */ - for (e = ptr; e != NULL; e = e->next) - { xassert(e->aj != 0.0); - xassert(e->xj->is_int); - xassert(e->xj->lb == 0.0 && e->xj->ub == 1.0); - } - /* check if the specified inequality constraint already has the - form of packing inequality */ - neg = 0; /* neg is |Jn| */ - for (e = ptr; e != NULL; e = e->next) - { if (e->aj == +1.0) - ; - else if (e->aj == -1.0) - neg++; - else - break; - } - if (e == NULL) - { /* all coefficients a[j] are +1 or -1; check rhs b */ - if (b == (double)(1 - neg)) - { /* it is packing inequality; no processing is needed */ - return 1; - } - } - /* substitute x[j] = 1 - x~[j] for all j in Jn to make all a[j] - positive; the result is a~[j] = |a[j]| and new rhs b */ - for (e = ptr; e != NULL; e = e->next) - if (e->aj < 0) b -= e->aj; - /* now a[j] > 0 for all j in J (actually |a[j]| are used) */ - /* if a[j] > b, skip processing--this case must not appear */ - for (e = ptr; e != NULL; e = e->next) - if (fabs(e->aj) > b) return 0; - /* now 0 < a[j] <= b for all j in J */ - /* find two minimal coefficients a[j] and a[k], j != k */ - ej = NULL; - for (e = ptr; e != NULL; e = e->next) - if (ej == NULL || fabs(ej->aj) > fabs(e->aj)) ej = e; - xassert(ej != NULL); - ek = NULL; - for (e = ptr; e != NULL; e = e->next) - if (e != ej) - if (ek == NULL || fabs(ek->aj) > fabs(e->aj)) ek = e; - xassert(ek != NULL); - /* the specified constraint is equivalent to packing inequality - iff a[j] + a[k] > b + eps */ - eps = 1e-3 + 1e-6 * fabs(b); - if (fabs(ej->aj) + fabs(ek->aj) <= b + eps) return 0; - /* perform back substitution x~[j] = 1 - x[j] and construct the - final equivalent packing inequality in generalized format */ - b = 1.0; - for (e = ptr; e != NULL; e = e->next) - { if (e->aj > 0.0) - e->aj = +1.0; - else /* e->aj < 0.0 */ - e->aj = -1.0, b -= 1.0; - } - *_b = b; - return 2; -} - -int npp_hidden_packing(NPP *npp, NPPROW *row) -{ /* identify hidden packing inequality */ - NPPROW *copy; - NPPAIJ *aij; - struct elem *ptr, *e; - int kase, ret, count = 0; - double b; - /* the row must be inequality constraint */ - xassert(row->lb < row->ub); - for (kase = 0; kase <= 1; kase++) - { if (kase == 0) - { /* process row upper bound */ - if (row->ub == +DBL_MAX) continue; - ptr = copy_form(npp, row, +1.0); - b = + row->ub; - } - else - { /* process row lower bound */ - if (row->lb == -DBL_MAX) continue; - ptr = copy_form(npp, row, -1.0); - b = - row->lb; - } - /* now the inequality has the form "sum a[j] x[j] <= b" */ - ret = hidden_packing(npp, ptr, &b); - xassert(0 <= ret && ret <= 2); - if (kase == 1 && ret == 1 || ret == 2) - { /* the original inequality has been identified as hidden - packing inequality */ - count++; -#ifdef GLP_DEBUG - xprintf("Original constraint:\n"); - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - xprintf(" %+g x%d", aij->val, aij->col->j); - if (row->lb != -DBL_MAX) xprintf(", >= %g", row->lb); - if (row->ub != +DBL_MAX) xprintf(", <= %g", row->ub); - xprintf("\n"); - xprintf("Equivalent packing inequality:\n"); - for (e = ptr; e != NULL; e = e->next) - xprintf(" %sx%d", e->aj > 0.0 ? "+" : "-", e->xj->j); - xprintf(", <= %g\n", b); -#endif - if (row->lb == -DBL_MAX || row->ub == +DBL_MAX) - { /* the original row is single-sided inequality; no copy - is needed */ - copy = NULL; - } - else - { /* the original row is double-sided inequality; we need - to create its copy for other bound before replacing it - with the equivalent inequality */ - copy = npp_add_row(npp); - if (kase == 0) - { /* the copy is for lower bound */ - copy->lb = row->lb, copy->ub = +DBL_MAX; - } - else - { /* the copy is for upper bound */ - copy->lb = -DBL_MAX, copy->ub = row->ub; - } - /* copy original row coefficients */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - npp_add_aij(npp, copy, aij->col, aij->val); - } - /* replace the original inequality by equivalent one */ - npp_erase_row(npp, row); - row->lb = -DBL_MAX, row->ub = b; - for (e = ptr; e != NULL; e = e->next) - npp_add_aij(npp, row, e->xj, e->aj); - /* continue processing lower bound for the copy */ - if (copy != NULL) row = copy; - } - drop_form(npp, ptr); - } - return count; -} - -/*********************************************************************** -* NAME -* -* npp_implied_packing - identify implied packing inequality -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_implied_packing(NPP *npp, NPPROW *row, int which, -* NPPCOL *var[], char set[]); -* -* DESCRIPTION -* -* The routine npp_implied_packing processes specified row (constraint) -* of general format: -* -* L <= sum a[j] x[j] <= U. (1) -* j -* -* If which = 0, only lower bound L, which must exist, is considered, -* while upper bound U is ignored. Similarly, if which = 1, only upper -* bound U, which must exist, is considered, while lower bound L is -* ignored. Thus, if the specified row is a double-sided inequality or -* equality constraint, this routine should be called twice for both -* lower and upper bounds. -* -* The routine npp_implied_packing attempts to find a non-trivial (i.e. -* having not less than two binary variables) packing inequality: -* -* sum x[j] - sum x[j] <= 1 - |Jn|, (2) -* j in Jp j in Jn -* -* which is relaxation of the constraint (1) in the sense that any -* solution satisfying to that constraint also satisfies to the packing -* inequality (2). If such relaxation exists, the routine stores -* pointers to descriptors of corresponding binary variables and their -* flags, resp., to locations var[1], var[2], ..., var[len] and set[1], -* set[2], ..., set[len], where set[j] = 0 means that j in Jp and -* set[j] = 1 means that j in Jn. -* -* RETURNS -* -* The routine npp_implied_packing returns len, which is the total -* number of binary variables in the packing inequality found, len >= 2. -* However, if the relaxation does not exist, the routine returns zero. -* -* ALGORITHM -* -* If which = 0, the constraint coefficients (1) are multiplied by -1 -* and b is assigned -L; if which = 1, the constraint coefficients (1) -* are not changed and b is assigned +U. In both cases the specified -* constraint gets the following format: -* -* sum a[j] x[j] <= b. (3) -* j -* -* (Note that (3) is a relaxation of (1), because one of bounds L or U -* is ignored.) -* -* Let J be set of binary variables, Kp be set of non-binary (integer -* or continuous) variables with a[j] > 0, and Kn be set of non-binary -* variables with a[j] < 0. Then the inequality (3) can be written as -* follows: -* -* sum a[j] x[j] <= b - sum a[j] x[j] - sum a[j] x[j]. (4) -* j in J j in Kp j in Kn -* -* To get rid of non-binary variables we can replace the inequality (4) -* by the following relaxed inequality: -* -* sum a[j] x[j] <= b~, (5) -* j in J -* -* where: -* -* b~ = sup(b - sum a[j] x[j] - sum a[j] x[j]) = -* j in Kp j in Kn -* -* = b - inf sum a[j] x[j] - inf sum a[j] x[j] = (6) -* j in Kp j in Kn -* -* = b - sum a[j] l[j] - sum a[j] u[j]. -* j in Kp j in Kn -* -* Note that if lower bound l[j] (if j in Kp) or upper bound u[j] -* (if j in Kn) of some non-binary variable x[j] does not exist, then -* formally b = +oo, in which case further analysis is not performed. -* -* Let Bp = {j in J: a[j] > 0}, Bn = {j in J: a[j] < 0}. To make all -* the inequality coefficients in (5) positive, we replace all x[j] in -* Bn by their complementaries, substituting x[j] = 1 - x~[j] for all -* j in Bn, that gives: -* -* sum a[j] x[j] - sum a[j] x~[j] <= b~ - sum a[j]. (7) -* j in Bp j in Bn j in Bn -* -* This inequality is a relaxation of the original constraint (1), and -* it is a binary knapsack inequality. Writing it in the standard format -* we have: -* -* sum alfa[j] z[j] <= beta, (8) -* j in J -* -* where: -* ( + a[j], if j in Bp, -* alfa[j] = < (9) -* ( - a[j], if j in Bn, -* -* ( x[j], if j in Bp, -* z[j] = < (10) -* ( 1 - x[j], if j in Bn, -* -* beta = b~ - sum a[j]. (11) -* j in Bn -* -* In the inequality (8) all coefficients are positive, therefore, the -* packing relaxation to be found for this inequality is the following: -* -* sum z[j] <= 1. (12) -* j in P -* -* It is obvious that set P within J, which we would like to find, must -* satisfy to the following condition: -* -* alfa[j] + alfa[k] > beta + eps for all j, k in P, j != k, (13) -* -* where eps is an absolute tolerance for value of the linear form. -* Thus, it is natural to take P = {j: alpha[j] > (beta + eps) / 2}. -* Moreover, if in the equality (8) there exist coefficients alfa[k], -* for which alfa[k] <= (beta + eps) / 2, but which, nevertheless, -* satisfies to the condition (13) for all j in P, *one* corresponding -* variable z[k] (having, for example, maximal coefficient alfa[k]) can -* be included in set P, that allows increasing the number of binary -* variables in (12) by one. -* -* Once the set P has been built, for the inequality (12) we need to -* perform back substitution according to (10) in order to express it -* through the original binary variables. As the result of such back -* substitution the relaxed packing inequality get its final format (2), -* where Jp = J intersect Bp, and Jn = J intersect Bn. */ - -int npp_implied_packing(NPP *npp, NPPROW *row, int which, - NPPCOL *var[], char set[]) -{ struct elem *ptr, *e, *i, *k; - int len = 0; - double b, eps; - /* build inequality (3) */ - if (which == 0) - { ptr = copy_form(npp, row, -1.0); - xassert(row->lb != -DBL_MAX); - b = - row->lb; - } - else if (which == 1) - { ptr = copy_form(npp, row, +1.0); - xassert(row->ub != +DBL_MAX); - b = + row->ub; - } - /* remove non-binary variables to build relaxed inequality (5); - compute its right-hand side b~ with formula (6) */ - for (e = ptr; e != NULL; e = e->next) - { if (!(e->xj->is_int && e->xj->lb == 0.0 && e->xj->ub == 1.0)) - { /* x[j] is non-binary variable */ - if (e->aj > 0.0) - { if (e->xj->lb == -DBL_MAX) goto done; - b -= e->aj * e->xj->lb; - } - else /* e->aj < 0.0 */ - { if (e->xj->ub == +DBL_MAX) goto done; - b -= e->aj * e->xj->ub; - } - /* a[j] = 0 means that variable x[j] is removed */ - e->aj = 0.0; - } - } - /* substitute x[j] = 1 - x~[j] to build knapsack inequality (8); - compute its right-hand side beta with formula (11) */ - for (e = ptr; e != NULL; e = e->next) - if (e->aj < 0.0) b -= e->aj; - /* if beta is close to zero, the knapsack inequality is either - infeasible or forcing inequality; this must never happen, so - we skip further analysis */ - if (b < 1e-3) goto done; - /* build set P as well as sets Jp and Jn, and determine x[k] as - explained above in comments to the routine */ - eps = 1e-3 + 1e-6 * b; - i = k = NULL; - for (e = ptr; e != NULL; e = e->next) - { /* note that alfa[j] = |a[j]| */ - if (fabs(e->aj) > 0.5 * (b + eps)) - { /* alfa[j] > (b + eps) / 2; include x[j] in set P, i.e. in - set Jp or Jn */ - var[++len] = e->xj; - set[len] = (char)(e->aj > 0.0 ? 0 : 1); - /* alfa[i] = min alfa[j] over all j included in set P */ - if (i == NULL || fabs(i->aj) > fabs(e->aj)) i = e; - } - else if (fabs(e->aj) >= 1e-3) - { /* alfa[k] = max alfa[j] over all j not included in set P; - we skip coefficient a[j] if it is close to zero to avoid - numerically unreliable results */ - if (k == NULL || fabs(k->aj) < fabs(e->aj)) k = e; - } - } - /* if alfa[k] satisfies to condition (13) for all j in P, include - x[k] in P */ - if (i != NULL && k != NULL && fabs(i->aj) + fabs(k->aj) > b + eps) - { var[++len] = k->xj; - set[len] = (char)(k->aj > 0.0 ? 0 : 1); - } - /* trivial packing inequality being redundant must never appear, - so we just ignore it */ - if (len < 2) len = 0; -done: drop_form(npp, ptr); - return len; -} - -/*********************************************************************** -* NAME -* -* npp_is_covering - test if constraint is covering inequality -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_is_covering(NPP *npp, NPPROW *row); -* -* RETURNS -* -* If the specified row (constraint) is covering inequality (see below), -* the routine npp_is_covering returns non-zero. Otherwise, it returns -* zero. -* -* COVERING INEQUALITIES -* -* In canonical format the covering inequality is the following: -* -* sum x[j] >= 1, (1) -* j in J -* -* where all variables x[j] are binary. This inequality expresses the -* condition that in any integer feasible solution variables in set J -* cannot be all equal to zero at the same time, i.e. at least one -* variable must take non-zero (unity) value. W.l.o.g. it is assumed -* that |J| >= 2, because if J is empty, the inequality (1) is -* infeasible, and if |J| = 1, the inequality (1) is a forcing row. -* -* In general case the covering inequality may include original -* variables x[j] as well as their complements x~[j]: -* -* sum x[j] + sum x~[j] >= 1, (2) -* j in Jp j in Jn -* -* where Jp and Jn are not intersected. Therefore, using substitution -* x~[j] = 1 - x[j] gives the packing inequality in generalized format: -* -* sum x[j] - sum x[j] >= 1 - |Jn|. (3) -* j in Jp j in Jn -* -* (May note that the inequality (3) cuts off infeasible solutions, -* where x[j] = 0 for all j in Jp and x[j] = 1 for all j in Jn.) -* -* NOTE: If |J| = 2, the inequality (3) is equivalent to packing -* inequality (see the routine npp_is_packing). */ - -int npp_is_covering(NPP *npp, NPPROW *row) -{ /* test if constraint is covering inequality */ - NPPCOL *col; - NPPAIJ *aij; - int b; - xassert(npp == npp); - if (!(row->lb != -DBL_MAX && row->ub == +DBL_MAX)) - return 0; - b = 1; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col = aij->col; - if (!(col->is_int && col->lb == 0.0 && col->ub == 1.0)) - return 0; - if (aij->val == +1.0) - ; - else if (aij->val == -1.0) - b--; - else - return 0; - } - if (row->lb != (double)b) return 0; - return 1; -} - -/*********************************************************************** -* NAME -* -* npp_hidden_covering - identify hidden covering inequality -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_hidden_covering(NPP *npp, NPPROW *row); -* -* DESCRIPTION -* -* The routine npp_hidden_covering processes specified inequality -* constraint, which includes only binary variables, and the number of -* the variables is not less than three. If the original inequality is -* equivalent to a covering inequality (see below), the routine -* replaces it by the equivalent inequality. If the original constraint -* is double-sided inequality, it is replaced by a pair of single-sided -* inequalities, if necessary. -* -* RETURNS -* -* If the original inequality constraint was replaced by equivalent -* covering inequality, the routine npp_hidden_covering returns -* non-zero. Otherwise, it returns zero. -* -* PROBLEM TRANSFORMATION -* -* Consider an inequality constraint: -* -* sum a[j] x[j] >= b, (1) -* j in J -* -* where all variables x[j] are binary, and |J| >= 3. (In case of '<=' -* inequality it can be transformed to '>=' format by multiplying both -* its sides by -1.) -* -* Let Jp = {j: a[j] > 0}, Jn = {j: a[j] < 0}. Performing substitution -* x[j] = 1 - x~[j] for all j in Jn, we have: -* -* sum a[j] x[j] >= b ==> -* j in J -* -* sum a[j] x[j] + sum a[j] x[j] >= b ==> -* j in Jp j in Jn -* -* sum a[j] x[j] + sum a[j] (1 - x~[j]) >= b ==> -* j in Jp j in Jn -* -* sum m a[j] x[j] - sum a[j] x~[j] >= b - sum a[j]. -* j in Jp j in Jn j in Jn -* -* Thus, meaning the transformation above, we can assume that in -* inequality (1) all coefficients a[j] are positive. Moreover, we can -* assume that b > 0, because otherwise the inequality (1) would be -* redundant (see the routine npp_analyze_row). It is then obvious that -* constraint (1) is equivalent to covering inequality only if: -* -* a[j] >= b, (2) -* -* for all j in J. -* -* Once the original inequality (1) is replaced by equivalent covering -* inequality, we need to perform back substitution x~[j] = 1 - x[j] for -* all j in Jn (see above). -* -* RECOVERING SOLUTION -* -* None needed. */ - -static int hidden_covering(NPP *npp, struct elem *ptr, double *_b) -{ /* process inequality constraint: sum a[j] x[j] >= b; - 0 - specified row is NOT hidden covering inequality; - 1 - specified row is covering inequality; - 2 - specified row is hidden covering inequality. */ - struct elem *e; - int neg; - double b = *_b, eps; - xassert(npp == npp); - /* a[j] must be non-zero, x[j] must be binary, for all j in J */ - for (e = ptr; e != NULL; e = e->next) - { xassert(e->aj != 0.0); - xassert(e->xj->is_int); - xassert(e->xj->lb == 0.0 && e->xj->ub == 1.0); - } - /* check if the specified inequality constraint already has the - form of covering inequality */ - neg = 0; /* neg is |Jn| */ - for (e = ptr; e != NULL; e = e->next) - { if (e->aj == +1.0) - ; - else if (e->aj == -1.0) - neg++; - else - break; - } - if (e == NULL) - { /* all coefficients a[j] are +1 or -1; check rhs b */ - if (b == (double)(1 - neg)) - { /* it is covering inequality; no processing is needed */ - return 1; - } - } - /* substitute x[j] = 1 - x~[j] for all j in Jn to make all a[j] - positive; the result is a~[j] = |a[j]| and new rhs b */ - for (e = ptr; e != NULL; e = e->next) - if (e->aj < 0) b -= e->aj; - /* now a[j] > 0 for all j in J (actually |a[j]| are used) */ - /* if b <= 0, skip processing--this case must not appear */ - if (b < 1e-3) return 0; - /* now a[j] > 0 for all j in J, and b > 0 */ - /* the specified constraint is equivalent to covering inequality - iff a[j] >= b for all j in J */ - eps = 1e-9 + 1e-12 * fabs(b); - for (e = ptr; e != NULL; e = e->next) - if (fabs(e->aj) < b - eps) return 0; - /* perform back substitution x~[j] = 1 - x[j] and construct the - final equivalent covering inequality in generalized format */ - b = 1.0; - for (e = ptr; e != NULL; e = e->next) - { if (e->aj > 0.0) - e->aj = +1.0; - else /* e->aj < 0.0 */ - e->aj = -1.0, b -= 1.0; - } - *_b = b; - return 2; -} - -int npp_hidden_covering(NPP *npp, NPPROW *row) -{ /* identify hidden covering inequality */ - NPPROW *copy; - NPPAIJ *aij; - struct elem *ptr, *e; - int kase, ret, count = 0; - double b; - /* the row must be inequality constraint */ - xassert(row->lb < row->ub); - for (kase = 0; kase <= 1; kase++) - { if (kase == 0) - { /* process row lower bound */ - if (row->lb == -DBL_MAX) continue; - ptr = copy_form(npp, row, +1.0); - b = + row->lb; - } - else - { /* process row upper bound */ - if (row->ub == +DBL_MAX) continue; - ptr = copy_form(npp, row, -1.0); - b = - row->ub; - } - /* now the inequality has the form "sum a[j] x[j] >= b" */ - ret = hidden_covering(npp, ptr, &b); - xassert(0 <= ret && ret <= 2); - if (kase == 1 && ret == 1 || ret == 2) - { /* the original inequality has been identified as hidden - covering inequality */ - count++; -#ifdef GLP_DEBUG - xprintf("Original constraint:\n"); - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - xprintf(" %+g x%d", aij->val, aij->col->j); - if (row->lb != -DBL_MAX) xprintf(", >= %g", row->lb); - if (row->ub != +DBL_MAX) xprintf(", <= %g", row->ub); - xprintf("\n"); - xprintf("Equivalent covering inequality:\n"); - for (e = ptr; e != NULL; e = e->next) - xprintf(" %sx%d", e->aj > 0.0 ? "+" : "-", e->xj->j); - xprintf(", >= %g\n", b); -#endif - if (row->lb == -DBL_MAX || row->ub == +DBL_MAX) - { /* the original row is single-sided inequality; no copy - is needed */ - copy = NULL; - } - else - { /* the original row is double-sided inequality; we need - to create its copy for other bound before replacing it - with the equivalent inequality */ - copy = npp_add_row(npp); - if (kase == 0) - { /* the copy is for upper bound */ - copy->lb = -DBL_MAX, copy->ub = row->ub; - } - else - { /* the copy is for lower bound */ - copy->lb = row->lb, copy->ub = +DBL_MAX; - } - /* copy original row coefficients */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - npp_add_aij(npp, copy, aij->col, aij->val); - } - /* replace the original inequality by equivalent one */ - npp_erase_row(npp, row); - row->lb = b, row->ub = +DBL_MAX; - for (e = ptr; e != NULL; e = e->next) - npp_add_aij(npp, row, e->xj, e->aj); - /* continue processing upper bound for the copy */ - if (copy != NULL) row = copy; - } - drop_form(npp, ptr); - } - return count; -} - -/*********************************************************************** -* NAME -* -* npp_is_partitioning - test if constraint is partitioning equality -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_is_partitioning(NPP *npp, NPPROW *row); -* -* RETURNS -* -* If the specified row (constraint) is partitioning equality (see -* below), the routine npp_is_partitioning returns non-zero. Otherwise, -* it returns zero. -* -* PARTITIONING EQUALITIES -* -* In canonical format the partitioning equality is the following: -* -* sum x[j] = 1, (1) -* j in J -* -* where all variables x[j] are binary. This equality expresses the -* condition that in any integer feasible solution exactly one variable -* in set J must take non-zero (unity) value while other variables must -* be equal to zero. W.l.o.g. it is assumed that |J| >= 2, because if -* J is empty, the inequality (1) is infeasible, and if |J| = 1, the -* inequality (1) is a fixing row. -* -* In general case the partitioning equality may include original -* variables x[j] as well as their complements x~[j]: -* -* sum x[j] + sum x~[j] = 1, (2) -* j in Jp j in Jn -* -* where Jp and Jn are not intersected. Therefore, using substitution -* x~[j] = 1 - x[j] leads to the partitioning equality in generalized -* format: -* -* sum x[j] - sum x[j] = 1 - |Jn|. (3) -* j in Jp j in Jn */ - -int npp_is_partitioning(NPP *npp, NPPROW *row) -{ /* test if constraint is partitioning equality */ - NPPCOL *col; - NPPAIJ *aij; - int b; - xassert(npp == npp); - if (row->lb != row->ub) return 0; - b = 1; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col = aij->col; - if (!(col->is_int && col->lb == 0.0 && col->ub == 1.0)) - return 0; - if (aij->val == +1.0) - ; - else if (aij->val == -1.0) - b--; - else - return 0; - } - if (row->lb != (double)b) return 0; - return 1; -} - -/*********************************************************************** -* NAME -* -* npp_reduce_ineq_coef - reduce inequality constraint coefficients -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_reduce_ineq_coef(NPP *npp, NPPROW *row); -* -* DESCRIPTION -* -* The routine npp_reduce_ineq_coef processes specified inequality -* constraint attempting to replace it by an equivalent constraint, -* where magnitude of coefficients at binary variables is smaller than -* in the original constraint. If the inequality is double-sided, it is -* replaced by a pair of single-sided inequalities, if necessary. -* -* RETURNS -* -* The routine npp_reduce_ineq_coef returns the number of coefficients -* reduced. -* -* BACKGROUND -* -* Consider an inequality constraint: -* -* sum a[j] x[j] >= b. (1) -* j in J -* -* (In case of '<=' inequality it can be transformed to '>=' format by -* multiplying both its sides by -1.) Let x[k] be a binary variable; -* other variables can be integer as well as continuous. We can write -* constraint (1) as follows: -* -* a[k] x[k] + t[k] >= b, (2) -* -* where: -* -* t[k] = sum a[j] x[j]. (3) -* j in J\{k} -* -* Since x[k] is binary, constraint (2) is equivalent to disjunction of -* the following two constraints: -* -* x[k] = 0, t[k] >= b (4) -* -* OR -* -* x[k] = 1, t[k] >= b - a[k]. (5) -* -* Let also that for the partial sum t[k] be known some its implied -* lower bound inf t[k]. -* -* Case a[k] > 0. Let inf t[k] < b, since otherwise both constraints -* (4) and (5) and therefore constraint (2) are redundant. -* If inf t[k] > b - a[k], only constraint (5) is redundant, in which -* case it can be replaced with the following redundant and therefore -* equivalent constraint: -* -* t[k] >= b - a'[k] = inf t[k], (6) -* -* where: -* -* a'[k] = b - inf t[k]. (7) -* -* Thus, the original constraint (2) is equivalent to the following -* constraint with coefficient at variable x[k] changed: -* -* a'[k] x[k] + t[k] >= b. (8) -* -* From inf t[k] < b it follows that a'[k] > 0, i.e. the coefficient -* at x[k] keeps its sign. And from inf t[k] > b - a[k] it follows that -* a'[k] < a[k], i.e. the coefficient reduces in magnitude. -* -* Case a[k] < 0. Let inf t[k] < b - a[k], since otherwise both -* constraints (4) and (5) and therefore constraint (2) are redundant. -* If inf t[k] > b, only constraint (4) is redundant, in which case it -* can be replaced with the following redundant and therefore equivalent -* constraint: -* -* t[k] >= b' = inf t[k]. (9) -* -* Rewriting constraint (5) as follows: -* -* t[k] >= b - a[k] = b' - a'[k], (10) -* -* where: -* -* a'[k] = a[k] + b' - b = a[k] + inf t[k] - b, (11) -* -* we can see that disjunction of constraint (9) and (10) is equivalent -* to disjunction of constraint (4) and (5), from which it follows that -* the original constraint (2) is equivalent to the following constraint -* with both coefficient at variable x[k] and right-hand side changed: -* -* a'[k] x[k] + t[k] >= b'. (12) -* -* From inf t[k] < b - a[k] it follows that a'[k] < 0, i.e. the -* coefficient at x[k] keeps its sign. And from inf t[k] > b it follows -* that a'[k] > a[k], i.e. the coefficient reduces in magnitude. -* -* PROBLEM TRANSFORMATION -* -* In the routine npp_reduce_ineq_coef the following implied lower -* bound of the partial sum (3) is used: -* -* inf t[k] = sum a[j] l[j] + sum a[j] u[j], (13) -* j in Jp\{k} k in Jn\{k} -* -* where Jp = {j : a[j] > 0}, Jn = {j : a[j] < 0}, l[j] and u[j] are -* lower and upper bounds, resp., of variable x[j]. -* -* In order to compute inf t[k] more efficiently, the following formula, -* which is equivalent to (13), is actually used: -* -* ( h - a[k] l[k] = h, if a[k] > 0, -* inf t[k] = < (14) -* ( h - a[k] u[k] = h - a[k], if a[k] < 0, -* -* where: -* -* h = sum a[j] l[j] + sum a[j] u[j] (15) -* j in Jp j in Jn -* -* is the implied lower bound of row (1). -* -* Reduction of positive coefficient (a[k] > 0) does not change value -* of h, since l[k] = 0. In case of reduction of negative coefficient -* (a[k] < 0) from (11) it follows that: -* -* delta a[k] = a'[k] - a[k] = inf t[k] - b (> 0), (16) -* -* so new value of h (accounting that u[k] = 1) can be computed as -* follows: -* -* h := h + delta a[k] = h + (inf t[k] - b). (17) -* -* RECOVERING SOLUTION -* -* None needed. */ - -static int reduce_ineq_coef(NPP *npp, struct elem *ptr, double *_b) -{ /* process inequality constraint: sum a[j] x[j] >= b */ - /* returns: the number of coefficients reduced */ - struct elem *e; - int count = 0; - double h, inf_t, new_a, b = *_b; - xassert(npp == npp); - /* compute h; see (15) */ - h = 0.0; - for (e = ptr; e != NULL; e = e->next) - { if (e->aj > 0.0) - { if (e->xj->lb == -DBL_MAX) goto done; - h += e->aj * e->xj->lb; - } - else /* e->aj < 0.0 */ - { if (e->xj->ub == +DBL_MAX) goto done; - h += e->aj * e->xj->ub; - } - } - /* perform reduction of coefficients at binary variables */ - for (e = ptr; e != NULL; e = e->next) - { /* skip non-binary variable */ - if (!(e->xj->is_int && e->xj->lb == 0.0 && e->xj->ub == 1.0)) - continue; - if (e->aj > 0.0) - { /* compute inf t[k]; see (14) */ - inf_t = h; - if (b - e->aj < inf_t && inf_t < b) - { /* compute reduced coefficient a'[k]; see (7) */ - new_a = b - inf_t; - if (new_a >= +1e-3 && - e->aj - new_a >= 0.01 * (1.0 + e->aj)) - { /* accept a'[k] */ -#ifdef GLP_DEBUG - xprintf("+"); -#endif - e->aj = new_a; - count++; - } - } - } - else /* e->aj < 0.0 */ - { /* compute inf t[k]; see (14) */ - inf_t = h - e->aj; - if (b < inf_t && inf_t < b - e->aj) - { /* compute reduced coefficient a'[k]; see (11) */ - new_a = e->aj + (inf_t - b); - if (new_a <= -1e-3 && - new_a - e->aj >= 0.01 * (1.0 - e->aj)) - { /* accept a'[k] */ -#ifdef GLP_DEBUG - xprintf("-"); -#endif - e->aj = new_a; - /* update h; see (17) */ - h += (inf_t - b); - /* compute b'; see (9) */ - b = inf_t; - count++; - } - } - } - } - *_b = b; -done: return count; -} - -int npp_reduce_ineq_coef(NPP *npp, NPPROW *row) -{ /* reduce inequality constraint coefficients */ - NPPROW *copy; - NPPAIJ *aij; - struct elem *ptr, *e; - int kase, count[2]; - double b; - /* the row must be inequality constraint */ - xassert(row->lb < row->ub); - count[0] = count[1] = 0; - for (kase = 0; kase <= 1; kase++) - { if (kase == 0) - { /* process row lower bound */ - if (row->lb == -DBL_MAX) continue; -#ifdef GLP_DEBUG - xprintf("L"); -#endif - ptr = copy_form(npp, row, +1.0); - b = + row->lb; - } - else - { /* process row upper bound */ - if (row->ub == +DBL_MAX) continue; -#ifdef GLP_DEBUG - xprintf("U"); -#endif - ptr = copy_form(npp, row, -1.0); - b = - row->ub; - } - /* now the inequality has the form "sum a[j] x[j] >= b" */ - count[kase] = reduce_ineq_coef(npp, ptr, &b); - if (count[kase] > 0) - { /* the original inequality has been replaced by equivalent - one with coefficients reduced */ - if (row->lb == -DBL_MAX || row->ub == +DBL_MAX) - { /* the original row is single-sided inequality; no copy - is needed */ - copy = NULL; - } - else - { /* the original row is double-sided inequality; we need - to create its copy for other bound before replacing it - with the equivalent inequality */ -#ifdef GLP_DEBUG - xprintf("*"); -#endif - copy = npp_add_row(npp); - if (kase == 0) - { /* the copy is for upper bound */ - copy->lb = -DBL_MAX, copy->ub = row->ub; - } - else - { /* the copy is for lower bound */ - copy->lb = row->lb, copy->ub = +DBL_MAX; - } - /* copy original row coefficients */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - npp_add_aij(npp, copy, aij->col, aij->val); - } - /* replace the original inequality by equivalent one */ - npp_erase_row(npp, row); - row->lb = b, row->ub = +DBL_MAX; - for (e = ptr; e != NULL; e = e->next) - npp_add_aij(npp, row, e->xj, e->aj); - /* continue processing upper bound for the copy */ - if (copy != NULL) row = copy; - } - drop_form(npp, ptr); - } - return count[0] + count[1]; -} - -/* eof */ diff --git a/code/3rd_glpk/npp/npp5.c b/code/3rd_glpk/npp/npp5.c deleted file mode 100644 index 2fad496d..00000000 --- a/code/3rd_glpk/npp/npp5.c +++ /dev/null @@ -1,809 +0,0 @@ -/* npp5.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2009-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -/*********************************************************************** -* NAME -* -* npp_clean_prob - perform initial LP/MIP processing -* -* SYNOPSIS -* -* #include "glpnpp.h" -* void npp_clean_prob(NPP *npp); -* -* DESCRIPTION -* -* The routine npp_clean_prob performs initial LP/MIP processing that -* currently includes: -* -* 1) removing free rows; -* -* 2) replacing double-sided constraint rows with almost identical -* bounds, by equality constraint rows; -* -* 3) removing fixed columns; -* -* 4) replacing double-bounded columns with almost identical bounds by -* fixed columns and removing those columns; -* -* 5) initial processing constraint coefficients (not implemented); -* -* 6) initial processing objective coefficients (not implemented). */ - -void npp_clean_prob(NPP *npp) -{ /* perform initial LP/MIP processing */ - NPPROW *row, *next_row; - NPPCOL *col, *next_col; - int ret; - xassert(npp == npp); - /* process rows which originally are free */ - for (row = npp->r_head; row != NULL; row = next_row) - { next_row = row->next; - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) - { /* process free row */ -#ifdef GLP_DEBUG - xprintf("1"); -#endif - npp_free_row(npp, row); - /* row was deleted */ - } - } - /* process rows which originally are double-sided inequalities */ - for (row = npp->r_head; row != NULL; row = next_row) - { next_row = row->next; - if (row->lb != -DBL_MAX && row->ub != +DBL_MAX && - row->lb < row->ub) - { ret = npp_make_equality(npp, row); - if (ret == 0) - ; - else if (ret == 1) - { /* row was replaced by equality constraint */ -#ifdef GLP_DEBUG - xprintf("2"); -#endif - } - else - xassert(ret != ret); - } - } - /* process columns which are originally fixed */ - for (col = npp->c_head; col != NULL; col = next_col) - { next_col = col->next; - if (col->lb == col->ub) - { /* process fixed column */ -#ifdef GLP_DEBUG - xprintf("3"); -#endif - npp_fixed_col(npp, col); - /* column was deleted */ - } - } - /* process columns which are originally double-bounded */ - for (col = npp->c_head; col != NULL; col = next_col) - { next_col = col->next; - if (col->lb != -DBL_MAX && col->ub != +DBL_MAX && - col->lb < col->ub) - { ret = npp_make_fixed(npp, col); - if (ret == 0) - ; - else if (ret == 1) - { /* column was replaced by fixed column; process it */ -#ifdef GLP_DEBUG - xprintf("4"); -#endif - npp_fixed_col(npp, col); - /* column was deleted */ - } - } - } - return; -} - -/*********************************************************************** -* NAME -* -* npp_process_row - perform basic row processing -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_process_row(NPP *npp, NPPROW *row, int hard); -* -* DESCRIPTION -* -* The routine npp_process_row performs basic row processing that -* currently includes: -* -* 1) removing empty row; -* -* 2) removing equality constraint row singleton and corresponding -* column; -* -* 3) removing inequality constraint row singleton and corresponding -* column if it was fixed; -* -* 4) performing general row analysis; -* -* 5) removing redundant row bounds; -* -* 6) removing forcing row and corresponding columns; -* -* 7) removing row which becomes free due to redundant bounds; -* -* 8) computing implied bounds for all columns in the row and using -* them to strengthen current column bounds (MIP only, optional, -* performed if the flag hard is on). -* -* Additionally the routine may activate affected rows and/or columns -* for further processing. -* -* RETURNS -* -* 0 success; -* -* GLP_ENOPFS primal/integer infeasibility detected; -* -* GLP_ENODFS dual infeasibility detected. */ - -int npp_process_row(NPP *npp, NPPROW *row, int hard) -{ /* perform basic row processing */ - NPPCOL *col; - NPPAIJ *aij, *next_aij, *aaa; - int ret; - /* row must not be free */ - xassert(!(row->lb == -DBL_MAX && row->ub == +DBL_MAX)); - /* start processing row */ - if (row->ptr == NULL) - { /* empty row */ - ret = npp_empty_row(npp, row); - if (ret == 0) - { /* row was deleted */ -#ifdef GLP_DEBUG - xprintf("A"); -#endif - return 0; - } - else if (ret == 1) - { /* primal infeasibility */ - return GLP_ENOPFS; - } - else - xassert(ret != ret); - } - if (row->ptr->r_next == NULL) - { /* row singleton */ - col = row->ptr->col; - if (row->lb == row->ub) - { /* equality constraint */ - ret = npp_eq_singlet(npp, row); - if (ret == 0) - { /* column was fixed, row was deleted */ -#ifdef GLP_DEBUG - xprintf("B"); -#endif - /* activate rows affected by column */ - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - npp_activate_row(npp, aij->row); - /* process fixed column */ - npp_fixed_col(npp, col); - /* column was deleted */ - return 0; - } - else if (ret == 1 || ret == 2) - { /* primal/integer infeasibility */ - return GLP_ENOPFS; - } - else - xassert(ret != ret); - } - else - { /* inequality constraint */ - ret = npp_ineq_singlet(npp, row); - if (0 <= ret && ret <= 3) - { /* row was deleted */ -#ifdef GLP_DEBUG - xprintf("C"); -#endif - /* activate column, since its length was changed due to - row deletion */ - npp_activate_col(npp, col); - if (ret >= 2) - { /* column bounds changed significantly or column was - fixed */ - /* activate rows affected by column */ - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - npp_activate_row(npp, aij->row); - } - if (ret == 3) - { /* column was fixed; process it */ -#ifdef GLP_DEBUG - xprintf("D"); -#endif - npp_fixed_col(npp, col); - /* column was deleted */ - } - return 0; - } - else if (ret == 4) - { /* primal infeasibility */ - return GLP_ENOPFS; - } - else - xassert(ret != ret); - } - } -#if 0 - /* sometimes this causes too large round-off errors; probably - pivot coefficient should be chosen more carefully */ - if (row->ptr->r_next->r_next == NULL) - { /* row doubleton */ - if (row->lb == row->ub) - { /* equality constraint */ - if (!(row->ptr->col->is_int || - row->ptr->r_next->col->is_int)) - { /* both columns are continuous */ - NPPCOL *q; - q = npp_eq_doublet(npp, row); - if (q != NULL) - { /* column q was eliminated */ -#ifdef GLP_DEBUG - xprintf("E"); -#endif - /* now column q is singleton of type "implied slack - variable"; we process it here to make sure that on - recovering basic solution the row is always active - equality constraint (as required by the routine - rcv_eq_doublet) */ - xassert(npp_process_col(npp, q) == 0); - /* column q was deleted; note that row p also may be - deleted */ - return 0; - } - } - } - } -#endif - /* general row analysis */ - ret = npp_analyze_row(npp, row); - xassert(0x00 <= ret && ret <= 0xFF); - if (ret == 0x33) - { /* row bounds are inconsistent with column bounds */ - return GLP_ENOPFS; - } - if ((ret & 0x0F) == 0x00) - { /* row lower bound does not exist or redundant */ - if (row->lb != -DBL_MAX) - { /* remove redundant row lower bound */ -#ifdef GLP_DEBUG - xprintf("F"); -#endif - npp_inactive_bound(npp, row, 0); - } - } - else if ((ret & 0x0F) == 0x01) - { /* row lower bound can be active */ - /* see below */ - } - else if ((ret & 0x0F) == 0x02) - { /* row lower bound is a forcing bound */ -#ifdef GLP_DEBUG - xprintf("G"); -#endif - /* process forcing row */ - if (npp_forcing_row(npp, row, 0) == 0) -fixup: { /* columns were fixed, row was made free */ - for (aij = row->ptr; aij != NULL; aij = next_aij) - { /* process column fixed by forcing row */ -#ifdef GLP_DEBUG - xprintf("H"); -#endif - col = aij->col; - next_aij = aij->r_next; - /* activate rows affected by column */ - for (aaa = col->ptr; aaa != NULL; aaa = aaa->c_next) - npp_activate_row(npp, aaa->row); - /* process fixed column */ - npp_fixed_col(npp, col); - /* column was deleted */ - } - /* process free row (which now is empty due to deletion of - all its columns) */ - npp_free_row(npp, row); - /* row was deleted */ - return 0; - } - } - else - xassert(ret != ret); - if ((ret & 0xF0) == 0x00) - { /* row upper bound does not exist or redundant */ - if (row->ub != +DBL_MAX) - { /* remove redundant row upper bound */ -#ifdef GLP_DEBUG - xprintf("I"); -#endif - npp_inactive_bound(npp, row, 1); - } - } - else if ((ret & 0xF0) == 0x10) - { /* row upper bound can be active */ - /* see below */ - } - else if ((ret & 0xF0) == 0x20) - { /* row upper bound is a forcing bound */ -#ifdef GLP_DEBUG - xprintf("J"); -#endif - /* process forcing row */ - if (npp_forcing_row(npp, row, 1) == 0) goto fixup; - } - else - xassert(ret != ret); - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) - { /* row became free due to redundant bounds removal */ -#ifdef GLP_DEBUG - xprintf("K"); -#endif - /* activate its columns, since their length will change due - to row deletion */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - npp_activate_col(npp, aij->col); - /* process free row */ - npp_free_row(npp, row); - /* row was deleted */ - return 0; - } -#if 1 /* 23/XII-2009 */ - /* row lower and/or upper bounds can be active */ - if (npp->sol == GLP_MIP && hard) - { /* improve current column bounds (optional) */ - if (npp_improve_bounds(npp, row, 1) < 0) - return GLP_ENOPFS; - } -#endif - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_improve_bounds - improve current column bounds -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_improve_bounds(NPP *npp, NPPROW *row, int flag); -* -* DESCRIPTION -* -* The routine npp_improve_bounds analyzes specified row (inequality -* or equality constraint) to determine implied column bounds and then -* uses these bounds to improve (strengthen) current column bounds. -* -* If the flag is on and current column bounds changed significantly -* or the column was fixed, the routine activate rows affected by the -* column for further processing. (This feature is intended to be used -* in the main loop of the routine npp_process_row.) -* -* NOTE: This operation can be used for MIP problem only. -* -* RETURNS -* -* The routine npp_improve_bounds returns the number of significantly -* changed bounds plus the number of column having been fixed due to -* bound improvements. However, if the routine detects primal/integer -* infeasibility, it returns a negative value. */ - -int npp_improve_bounds(NPP *npp, NPPROW *row, int flag) -{ /* improve current column bounds */ - NPPCOL *col; - NPPAIJ *aij, *next_aij, *aaa; - int kase, ret, count = 0; - double lb, ub; - xassert(npp->sol == GLP_MIP); - /* row must not be free */ - xassert(!(row->lb == -DBL_MAX && row->ub == +DBL_MAX)); - /* determine implied column bounds */ - npp_implied_bounds(npp, row); - /* and use these bounds to strengthen current column bounds */ - for (aij = row->ptr; aij != NULL; aij = next_aij) - { col = aij->col; - next_aij = aij->r_next; - for (kase = 0; kase <= 1; kase++) - { /* save current column bounds */ - lb = col->lb, ub = col->ub; - if (kase == 0) - { /* process implied column lower bound */ - if (col->ll.ll == -DBL_MAX) continue; - ret = npp_implied_lower(npp, col, col->ll.ll); - } - else - { /* process implied column upper bound */ - if (col->uu.uu == +DBL_MAX) continue; - ret = npp_implied_upper(npp, col, col->uu.uu); - } - if (ret == 0 || ret == 1) - { /* current column bounds did not change or changed, but - not significantly; restore current column bounds */ - col->lb = lb, col->ub = ub; - } - else if (ret == 2 || ret == 3) - { /* current column bounds changed significantly or column - was fixed */ -#ifdef GLP_DEBUG - xprintf("L"); -#endif - count++; - /* activate other rows affected by column, if required */ - if (flag) - { for (aaa = col->ptr; aaa != NULL; aaa = aaa->c_next) - { if (aaa->row != row) - npp_activate_row(npp, aaa->row); - } - } - if (ret == 3) - { /* process fixed column */ -#ifdef GLP_DEBUG - xprintf("M"); -#endif - npp_fixed_col(npp, col); - /* column was deleted */ - break; /* for kase */ - } - } - else if (ret == 4) - { /* primal/integer infeasibility */ - return -1; - } - else - xassert(ret != ret); - } - } - return count; -} - -/*********************************************************************** -* NAME -* -* npp_process_col - perform basic column processing -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_process_col(NPP *npp, NPPCOL *col); -* -* DESCRIPTION -* -* The routine npp_process_col performs basic column processing that -* currently includes: -* -* 1) fixing and removing empty column; -* -* 2) removing column singleton, which is implied slack variable, and -* corresponding row if it becomes free; -* -* 3) removing bounds of column, which is implied free variable, and -* replacing corresponding row by equality constraint. -* -* Additionally the routine may activate affected rows and/or columns -* for further processing. -* -* RETURNS -* -* 0 success; -* -* GLP_ENOPFS primal/integer infeasibility detected; -* -* GLP_ENODFS dual infeasibility detected. */ - -int npp_process_col(NPP *npp, NPPCOL *col) -{ /* perform basic column processing */ - NPPROW *row; - NPPAIJ *aij; - int ret; - /* column must not be fixed */ - xassert(col->lb < col->ub); - /* start processing column */ - if (col->ptr == NULL) - { /* empty column */ - ret = npp_empty_col(npp, col); - if (ret == 0) - { /* column was fixed and deleted */ -#ifdef GLP_DEBUG - xprintf("N"); -#endif - return 0; - } - else if (ret == 1) - { /* dual infeasibility */ - return GLP_ENODFS; - } - else - xassert(ret != ret); - } - if (col->ptr->c_next == NULL) - { /* column singleton */ - row = col->ptr->row; - if (row->lb == row->ub) - { /* equality constraint */ - if (!col->is_int) -slack: { /* implied slack variable */ -#ifdef GLP_DEBUG - xprintf("O"); -#endif - npp_implied_slack(npp, col); - /* column was deleted */ - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) - { /* row became free due to implied slack variable */ -#ifdef GLP_DEBUG - xprintf("P"); -#endif - /* activate columns affected by row */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - npp_activate_col(npp, aij->col); - /* process free row */ - npp_free_row(npp, row); - /* row was deleted */ - } - else - { /* row became inequality constraint; activate it - since its length changed due to column deletion */ - npp_activate_row(npp, row); - } - return 0; - } - } - else - { /* inequality constraint */ - if (!col->is_int) - { ret = npp_implied_free(npp, col); - if (ret == 0) - { /* implied free variable */ -#ifdef GLP_DEBUG - xprintf("Q"); -#endif - /* column bounds were removed, row was replaced by - equality constraint */ - goto slack; - } - else if (ret == 1) - { /* column is not implied free variable, because its - lower and/or upper bounds can be active */ - } - else if (ret == 2) - { /* dual infeasibility */ - return GLP_ENODFS; - } - } - } - } - /* column still exists */ - return 0; -} - -/*********************************************************************** -* NAME -* -* npp_process_prob - perform basic LP/MIP processing -* -* SYNOPSIS -* -* #include "glpnpp.h" -* int npp_process_prob(NPP *npp, int hard); -* -* DESCRIPTION -* -* The routine npp_process_prob performs basic LP/MIP processing that -* currently includes: -* -* 1) initial LP/MIP processing (see the routine npp_clean_prob), -* -* 2) basic row processing (see the routine npp_process_row), and -* -* 3) basic column processing (see the routine npp_process_col). -* -* If the flag hard is on, the routine attempts to improve current -* column bounds multiple times within the main processing loop, in -* which case this feature may take a time. Otherwise, if the flag hard -* is off, improving column bounds is performed only once at the end of -* the main loop. (Note that this feature is used for MIP only.) -* -* The routine uses two sets: the set of active rows and the set of -* active columns. Rows/columns are marked by a flag (the field temp in -* NPPROW/NPPCOL). If the flag is non-zero, the row/column is active, -* in which case it is placed in the beginning of the row/column list; -* otherwise, if the flag is zero, the row/column is inactive, in which -* case it is placed in the end of the row/column list. If a row/column -* being currently processed may affect other rows/columns, the latters -* are activated for further processing. -* -* RETURNS -* -* 0 success; -* -* GLP_ENOPFS primal/integer infeasibility detected; -* -* GLP_ENODFS dual infeasibility detected. */ - -int npp_process_prob(NPP *npp, int hard) -{ /* perform basic LP/MIP processing */ - NPPROW *row; - NPPCOL *col; - int processing, ret; - /* perform initial LP/MIP processing */ - npp_clean_prob(npp); - /* activate all remaining rows and columns */ - for (row = npp->r_head; row != NULL; row = row->next) - row->temp = 1; - for (col = npp->c_head; col != NULL; col = col->next) - col->temp = 1; - /* main processing loop */ - processing = 1; - while (processing) - { processing = 0; - /* process all active rows */ - for (;;) - { row = npp->r_head; - if (row == NULL || !row->temp) break; - npp_deactivate_row(npp, row); - ret = npp_process_row(npp, row, hard); - if (ret != 0) goto done; - processing = 1; - } - /* process all active columns */ - for (;;) - { col = npp->c_head; - if (col == NULL || !col->temp) break; - npp_deactivate_col(npp, col); - ret = npp_process_col(npp, col); - if (ret != 0) goto done; - processing = 1; - } - } -#if 1 /* 23/XII-2009 */ - if (npp->sol == GLP_MIP && !hard) - { /* improve current column bounds (optional) */ - for (row = npp->r_head; row != NULL; row = row->next) - { if (npp_improve_bounds(npp, row, 0) < 0) - { ret = GLP_ENOPFS; - goto done; - } - } - } -#endif - /* all seems ok */ - ret = 0; -done: xassert(ret == 0 || ret == GLP_ENOPFS || ret == GLP_ENODFS); -#ifdef GLP_DEBUG - xprintf("\n"); -#endif - return ret; -} - -/**********************************************************************/ - -int npp_simplex(NPP *npp, const glp_smcp *parm) -{ /* process LP prior to applying primal/dual simplex method */ - int ret; - xassert(npp->sol == GLP_SOL); - xassert(parm == parm); - ret = npp_process_prob(npp, 0); - return ret; -} - -/**********************************************************************/ - -int npp_integer(NPP *npp, const glp_iocp *parm) -{ /* process MIP prior to applying branch-and-bound method */ - NPPROW *row, *prev_row; - NPPCOL *col; - NPPAIJ *aij; - int count, ret; - xassert(npp->sol == GLP_MIP); - xassert(parm == parm); - /*==============================================================*/ - /* perform basic MIP processing */ - ret = npp_process_prob(npp, 1); - if (ret != 0) goto done; - /*==============================================================*/ - /* binarize problem, if required */ - if (parm->binarize) - npp_binarize_prob(npp); - /*==============================================================*/ - /* identify hidden packing inequalities */ - count = 0; - /* new rows will be added to the end of the row list, so we go - from the end to beginning of the row list */ - for (row = npp->r_tail; row != NULL; row = prev_row) - { prev_row = row->prev; - /* skip free row */ - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) continue; - /* skip equality constraint */ - if (row->lb == row->ub) continue; - /* skip row having less than two variables */ - if (row->ptr == NULL || row->ptr->r_next == NULL) continue; - /* skip row having non-binary variables */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col = aij->col; - if (!(col->is_int && col->lb == 0.0 && col->ub == 1.0)) - break; - } - if (aij != NULL) continue; - count += npp_hidden_packing(npp, row); - } - if (count > 0) - xprintf("%d hidden packing inequaliti(es) were detected\n", - count); - /*==============================================================*/ - /* identify hidden covering inequalities */ - count = 0; - /* new rows will be added to the end of the row list, so we go - from the end to beginning of the row list */ - for (row = npp->r_tail; row != NULL; row = prev_row) - { prev_row = row->prev; - /* skip free row */ - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) continue; - /* skip equality constraint */ - if (row->lb == row->ub) continue; - /* skip row having less than three variables */ - if (row->ptr == NULL || row->ptr->r_next == NULL || - row->ptr->r_next->r_next == NULL) continue; - /* skip row having non-binary variables */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { col = aij->col; - if (!(col->is_int && col->lb == 0.0 && col->ub == 1.0)) - break; - } - if (aij != NULL) continue; - count += npp_hidden_covering(npp, row); - } - if (count > 0) - xprintf("%d hidden covering inequaliti(es) were detected\n", - count); - /*==============================================================*/ - /* reduce inequality constraint coefficients */ - count = 0; - /* new rows will be added to the end of the row list, so we go - from the end to beginning of the row list */ - for (row = npp->r_tail; row != NULL; row = prev_row) - { prev_row = row->prev; - /* skip equality constraint */ - if (row->lb == row->ub) continue; - count += npp_reduce_ineq_coef(npp, row); - } - if (count > 0) - xprintf("%d constraint coefficient(s) were reduced\n", count); - /*==============================================================*/ -#ifdef GLP_DEBUG - routine(npp); -#endif - /*==============================================================*/ - /* all seems ok */ - ret = 0; -done: return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/npp/npp6.c b/code/3rd_glpk/npp/npp6.c deleted file mode 100644 index b57f8615..00000000 --- a/code/3rd_glpk/npp/npp6.c +++ /dev/null @@ -1,1500 +0,0 @@ -/* npp6.c (translate feasibility problem to CNF-SAT) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2011-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "npp.h" - -/*********************************************************************** -* npp_sat_free_row - process free (unbounded) row -* -* This routine processes row p, which is free (i.e. has no finite -* bounds): -* -* -inf < sum a[p,j] x[j] < +inf. (1) -* -* The constraint (1) cannot be active and therefore it is redundant, -* so the routine simply removes it from the original problem. */ - -void npp_sat_free_row(NPP *npp, NPPROW *p) -{ /* the row should be free */ - xassert(p->lb == -DBL_MAX && p->ub == +DBL_MAX); - /* remove the row from the problem */ - npp_del_row(npp, p); - return; -} - -/*********************************************************************** -* npp_sat_fixed_col - process fixed column -* -* This routine processes column q, which is fixed: -* -* x[q] = s[q], (1) -* -* where s[q] is a fixed column value. -* -* The routine substitutes fixed value s[q] into constraint rows and -* then removes column x[q] from the original problem. -* -* Substitution of x[q] = s[q] into row i gives: -* -* L[i] <= sum a[i,j] x[j] <= U[i] ==> -* j -* -* L[i] <= sum a[i,j] x[j] + a[i,q] x[q] <= U[i] ==> -* j!=q -* -* L[i] <= sum a[i,j] x[j] + a[i,q] s[q] <= U[i] ==> -* j!=q -* -* L~[i] <= sum a[i,j] x[j] <= U~[i], -* j!=q -* -* where -* -* L~[i] = L[i] - a[i,q] s[q], (2) -* -* U~[i] = U[i] - a[i,q] s[q] (3) -* -* are, respectively, lower and upper bound of row i in the transformed -* problem. -* -* On recovering solution x[q] is assigned the value of s[q]. */ - -struct sat_fixed_col -{ /* fixed column */ - int q; - /* column reference number for variable x[q] */ - int s; - /* value, at which x[q] is fixed */ -}; - -static int rcv_sat_fixed_col(NPP *, void *); - -int npp_sat_fixed_col(NPP *npp, NPPCOL *q) -{ struct sat_fixed_col *info; - NPPROW *i; - NPPAIJ *aij; - int temp; - /* the column should be fixed */ - xassert(q->lb == q->ub); - /* create transformation stack entry */ - info = npp_push_tse(npp, - rcv_sat_fixed_col, sizeof(struct sat_fixed_col)); - info->q = q->j; - info->s = (int)q->lb; - xassert((double)info->s == q->lb); - /* substitute x[q] = s[q] into constraint rows */ - if (info->s == 0) - goto skip; - for (aij = q->ptr; aij != NULL; aij = aij->c_next) - { i = aij->row; - if (i->lb != -DBL_MAX) - { i->lb -= aij->val * (double)info->s; - temp = (int)i->lb; - if ((double)temp != i->lb) - return 1; /* integer arithmetic error */ - } - if (i->ub != +DBL_MAX) - { i->ub -= aij->val * (double)info->s; - temp = (int)i->ub; - if ((double)temp != i->ub) - return 2; /* integer arithmetic error */ - } - } -skip: /* remove the column from the problem */ - npp_del_col(npp, q); - return 0; -} - -static int rcv_sat_fixed_col(NPP *npp, void *info_) -{ struct sat_fixed_col *info = info_; - npp->c_value[info->q] = (double)info->s; - return 0; -} - -/*********************************************************************** -* npp_sat_is_bin_comb - test if row is binary combination -* -* This routine tests if the specified row is a binary combination, -* i.e. all its constraint coefficients are +1 and -1 and all variables -* are binary. If the test was passed, the routine returns non-zero, -* otherwise zero. */ - -int npp_sat_is_bin_comb(NPP *npp, NPPROW *row) -{ NPPCOL *col; - NPPAIJ *aij; - xassert(npp == npp); - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { if (!(aij->val == +1.0 || aij->val == -1.0)) - return 0; /* non-unity coefficient */ - col = aij->col; - if (!(col->is_int && col->lb == 0.0 && col->ub == 1.0)) - return 0; /* non-binary column */ - } - return 1; /* test was passed */ -} - -/*********************************************************************** -* npp_sat_num_pos_coef - determine number of positive coefficients -* -* This routine returns the number of positive coefficients in the -* specified row. */ - -int npp_sat_num_pos_coef(NPP *npp, NPPROW *row) -{ NPPAIJ *aij; - int num = 0; - xassert(npp == npp); - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { if (aij->val > 0.0) - num++; - } - return num; -} - -/*********************************************************************** -* npp_sat_num_neg_coef - determine number of negative coefficients -* -* This routine returns the number of negative coefficients in the -* specified row. */ - -int npp_sat_num_neg_coef(NPP *npp, NPPROW *row) -{ NPPAIJ *aij; - int num = 0; - xassert(npp == npp); - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { if (aij->val < 0.0) - num++; - } - return num; -} - -/*********************************************************************** -* npp_sat_is_cover_ineq - test if row is covering inequality -* -* The canonical form of a covering inequality is the following: -* -* sum x[j] >= 1, (1) -* j in J -* -* where all x[j] are binary variables. -* -* In general case a covering inequality may have one of the following -* two forms: -* -* sum x[j] - sum x[j] >= 1 - |J-|, (2) -* j in J+ j in J- -* -* -* sum x[j] - sum x[j] <= |J+| - 1. (3) -* j in J+ j in J- -* -* Obviously, the inequality (2) can be transformed to the form (1) by -* substitution x[j] = 1 - x'[j] for all j in J-, where x'[j] is the -* negation of variable x[j]. And the inequality (3) can be transformed -* to (2) by multiplying both left- and right-hand sides by -1. -* -* This routine returns one of the following codes: -* -* 0, if the specified row is not a covering inequality; -* -* 1, if the specified row has the form (2); -* -* 2, if the specified row has the form (3). */ - -int npp_sat_is_cover_ineq(NPP *npp, NPPROW *row) -{ xassert(npp == npp); - if (row->lb != -DBL_MAX && row->ub == +DBL_MAX) - { /* row is inequality of '>=' type */ - if (npp_sat_is_bin_comb(npp, row)) - { /* row is a binary combination */ - if (row->lb == 1.0 - npp_sat_num_neg_coef(npp, row)) - { /* row has the form (2) */ - return 1; - } - } - } - else if (row->lb == -DBL_MAX && row->ub != +DBL_MAX) - { /* row is inequality of '<=' type */ - if (npp_sat_is_bin_comb(npp, row)) - { /* row is a binary combination */ - if (row->ub == npp_sat_num_pos_coef(npp, row) - 1.0) - { /* row has the form (3) */ - return 2; - } - } - } - /* row is not a covering inequality */ - return 0; -} - -/*********************************************************************** -* npp_sat_is_pack_ineq - test if row is packing inequality -* -* The canonical form of a packing inequality is the following: -* -* sum x[j] <= 1, (1) -* j in J -* -* where all x[j] are binary variables. -* -* In general case a packing inequality may have one of the following -* two forms: -* -* sum x[j] - sum x[j] <= 1 - |J-|, (2) -* j in J+ j in J- -* -* -* sum x[j] - sum x[j] >= |J+| - 1. (3) -* j in J+ j in J- -* -* Obviously, the inequality (2) can be transformed to the form (1) by -* substitution x[j] = 1 - x'[j] for all j in J-, where x'[j] is the -* negation of variable x[j]. And the inequality (3) can be transformed -* to (2) by multiplying both left- and right-hand sides by -1. -* -* This routine returns one of the following codes: -* -* 0, if the specified row is not a packing inequality; -* -* 1, if the specified row has the form (2); -* -* 2, if the specified row has the form (3). */ - -int npp_sat_is_pack_ineq(NPP *npp, NPPROW *row) -{ xassert(npp == npp); - if (row->lb == -DBL_MAX && row->ub != +DBL_MAX) - { /* row is inequality of '<=' type */ - if (npp_sat_is_bin_comb(npp, row)) - { /* row is a binary combination */ - if (row->ub == 1.0 - npp_sat_num_neg_coef(npp, row)) - { /* row has the form (2) */ - return 1; - } - } - } - else if (row->lb != -DBL_MAX && row->ub == +DBL_MAX) - { /* row is inequality of '>=' type */ - if (npp_sat_is_bin_comb(npp, row)) - { /* row is a binary combination */ - if (row->lb == npp_sat_num_pos_coef(npp, row) - 1.0) - { /* row has the form (3) */ - return 2; - } - } - } - /* row is not a packing inequality */ - return 0; -} - -/*********************************************************************** -* npp_sat_is_partn_eq - test if row is partitioning equality -* -* The canonical form of a partitioning equality is the following: -* -* sum x[j] = 1, (1) -* j in J -* -* where all x[j] are binary variables. -* -* In general case a partitioning equality may have one of the following -* two forms: -* -* sum x[j] - sum x[j] = 1 - |J-|, (2) -* j in J+ j in J- -* -* -* sum x[j] - sum x[j] = |J+| - 1. (3) -* j in J+ j in J- -* -* Obviously, the equality (2) can be transformed to the form (1) by -* substitution x[j] = 1 - x'[j] for all j in J-, where x'[j] is the -* negation of variable x[j]. And the equality (3) can be transformed -* to (2) by multiplying both left- and right-hand sides by -1. -* -* This routine returns one of the following codes: -* -* 0, if the specified row is not a partitioning equality; -* -* 1, if the specified row has the form (2); -* -* 2, if the specified row has the form (3). */ - -int npp_sat_is_partn_eq(NPP *npp, NPPROW *row) -{ xassert(npp == npp); - if (row->lb == row->ub) - { /* row is equality constraint */ - if (npp_sat_is_bin_comb(npp, row)) - { /* row is a binary combination */ - if (row->lb == 1.0 - npp_sat_num_neg_coef(npp, row)) - { /* row has the form (2) */ - return 1; - } - if (row->ub == npp_sat_num_pos_coef(npp, row) - 1.0) - { /* row has the form (3) */ - return 2; - } - } - } - /* row is not a partitioning equality */ - return 0; -} - -/*********************************************************************** -* npp_sat_reverse_row - multiply both sides of row by -1 -* -* This routines multiplies by -1 both left- and right-hand sides of -* the specified row: -* -* L <= sum x[j] <= U, -* -* that results in the following row: -* -* -U <= sum (-x[j]) <= -L. -* -* If no integer overflow occured, the routine returns zero, otherwise -* non-zero. */ - -int npp_sat_reverse_row(NPP *npp, NPPROW *row) -{ NPPAIJ *aij; - int temp, ret = 0; - double old_lb, old_ub; - xassert(npp == npp); - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { aij->val = -aij->val; - temp = (int)aij->val; - if ((double)temp != aij->val) - ret = 1; - } - old_lb = row->lb, old_ub = row->ub; - if (old_ub == +DBL_MAX) - row->lb = -DBL_MAX; - else - { row->lb = -old_ub; - temp = (int)row->lb; - if ((double)temp != row->lb) - ret = 2; - } - if (old_lb == -DBL_MAX) - row->ub = +DBL_MAX; - else - { row->ub = -old_lb; - temp = (int)row->ub; - if ((double)temp != row->ub) - ret = 3; - } - return ret; -} - -/*********************************************************************** -* npp_sat_split_pack - split packing inequality -* -* Let there be given a packing inequality in canonical form: -* -* sum t[j] <= 1, (1) -* j in J -* -* where t[j] = x[j] or t[j] = 1 - x[j], x[j] is a binary variable. -* And let J = J1 U J2 is a partition of the set of literals. Then the -* inequality (1) is obviously equivalent to the following two packing -* inequalities: -* -* sum t[j] <= y <--> sum t[j] + (1 - y) <= 1, (2) -* j in J1 j in J1 -* -* sum t[j] <= 1 - y <--> sum t[j] + y <= 1, (3) -* j in J2 j in J2 -* -* where y is a new binary variable added to the transformed problem. -* -* Assuming that the specified row is a packing inequality (1), this -* routine constructs the set J1 by including there first nlit literals -* (terms) from the specified row, and the set J2 = J \ J1. Then the -* routine creates a new row, which corresponds to inequality (2), and -* replaces the specified row with inequality (3). */ - -NPPROW *npp_sat_split_pack(NPP *npp, NPPROW *row, int nlit) -{ NPPROW *rrr; - NPPCOL *col; - NPPAIJ *aij; - int k; - /* original row should be packing inequality (1) */ - xassert(npp_sat_is_pack_ineq(npp, row) == 1); - /* and nlit should be less than the number of literals (terms) - in the original row */ - xassert(0 < nlit && nlit < npp_row_nnz(npp, row)); - /* create new row corresponding to inequality (2) */ - rrr = npp_add_row(npp); - rrr->lb = -DBL_MAX, rrr->ub = 1.0; - /* move first nlit literals (terms) from the original row to the - new row; the original row becomes inequality (3) */ - for (k = 1; k <= nlit; k++) - { aij = row->ptr; - xassert(aij != NULL); - /* add literal to the new row */ - npp_add_aij(npp, rrr, aij->col, aij->val); - /* correct rhs */ - if (aij->val < 0.0) - rrr->ub -= 1.0, row->ub += 1.0; - /* remove literal from the original row */ - npp_del_aij(npp, aij); - } - /* create new binary variable y */ - col = npp_add_col(npp); - col->is_int = 1, col->lb = 0.0, col->ub = 1.0; - /* include literal (1 - y) in the new row */ - npp_add_aij(npp, rrr, col, -1.0); - rrr->ub -= 1.0; - /* include literal y in the original row */ - npp_add_aij(npp, row, col, +1.0); - return rrr; -} - -/*********************************************************************** -* npp_sat_encode_pack - encode packing inequality -* -* Given a packing inequality in canonical form: -* -* sum t[j] <= 1, (1) -* j in J -* -* where t[j] = x[j] or t[j] = 1 - x[j], x[j] is a binary variable, -* this routine translates it to CNF by replacing it with the following -* equivalent set of edge packing inequalities: -* -* t[j] + t[k] <= 1 for all j, k in J, j != k. (2) -* -* Then the routine transforms each edge packing inequality (2) to -* corresponding covering inequality (that encodes two-literal clause) -* by multiplying both its part by -1: -* -* - t[j] - t[k] >= -1 <--> (1 - t[j]) + (1 - t[k]) >= 1. (3) -* -* On exit the routine removes the original row from the problem. */ - -void npp_sat_encode_pack(NPP *npp, NPPROW *row) -{ NPPROW *rrr; - NPPAIJ *aij, *aik; - /* original row should be packing inequality (1) */ - xassert(npp_sat_is_pack_ineq(npp, row) == 1); - /* create equivalent system of covering inequalities (3) */ - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { /* due to symmetry only one of inequalities t[j] + t[k] <= 1 - and t[k] <= t[j] <= 1 can be considered */ - for (aik = aij->r_next; aik != NULL; aik = aik->r_next) - { /* create edge packing inequality (2) */ - rrr = npp_add_row(npp); - rrr->lb = -DBL_MAX, rrr->ub = 1.0; - npp_add_aij(npp, rrr, aij->col, aij->val); - if (aij->val < 0.0) - rrr->ub -= 1.0; - npp_add_aij(npp, rrr, aik->col, aik->val); - if (aik->val < 0.0) - rrr->ub -= 1.0; - /* and transform it to covering inequality (3) */ - npp_sat_reverse_row(npp, rrr); - xassert(npp_sat_is_cover_ineq(npp, rrr) == 1); - } - } - /* remove the original row from the problem */ - npp_del_row(npp, row); - return; -} - -/*********************************************************************** -* npp_sat_encode_sum2 - encode 2-bit summation -* -* Given a set containing two literals x and y this routine encodes -* the equality -* -* x + y = s + 2 * c, (1) -* -* where -* -* s = (x + y) % 2 (2) -* -* is a binary variable modeling the low sum bit, and -* -* c = (x + y) / 2 (3) -* -* is a binary variable modeling the high (carry) sum bit. */ - -void npp_sat_encode_sum2(NPP *npp, NPPLSE *set, NPPSED *sed) -{ NPPROW *row; - int x, y, s, c; - /* the set should contain exactly two literals */ - xassert(set != NULL); - xassert(set->next != NULL); - xassert(set->next->next == NULL); - sed->x = set->lit; - xassert(sed->x.neg == 0 || sed->x.neg == 1); - sed->y = set->next->lit; - xassert(sed->y.neg == 0 || sed->y.neg == 1); - sed->z.col = NULL, sed->z.neg = 0; - /* perform encoding s = (x + y) % 2 */ - sed->s = npp_add_col(npp); - sed->s->is_int = 1, sed->s->lb = 0.0, sed->s->ub = 1.0; - for (x = 0; x <= 1; x++) - { for (y = 0; y <= 1; y++) - { for (s = 0; s <= 1; s++) - { if ((x + y) % 2 != s) - { /* generate CNF clause to disable infeasible - combination */ - row = npp_add_row(npp); - row->lb = 1.0, row->ub = +DBL_MAX; - if (x == sed->x.neg) - npp_add_aij(npp, row, sed->x.col, +1.0); - else - { npp_add_aij(npp, row, sed->x.col, -1.0); - row->lb -= 1.0; - } - if (y == sed->y.neg) - npp_add_aij(npp, row, sed->y.col, +1.0); - else - { npp_add_aij(npp, row, sed->y.col, -1.0); - row->lb -= 1.0; - } - if (s == 0) - npp_add_aij(npp, row, sed->s, +1.0); - else - { npp_add_aij(npp, row, sed->s, -1.0); - row->lb -= 1.0; - } - } - } - } - } - /* perform encoding c = (x + y) / 2 */ - sed->c = npp_add_col(npp); - sed->c->is_int = 1, sed->c->lb = 0.0, sed->c->ub = 1.0; - for (x = 0; x <= 1; x++) - { for (y = 0; y <= 1; y++) - { for (c = 0; c <= 1; c++) - { if ((x + y) / 2 != c) - { /* generate CNF clause to disable infeasible - combination */ - row = npp_add_row(npp); - row->lb = 1.0, row->ub = +DBL_MAX; - if (x == sed->x.neg) - npp_add_aij(npp, row, sed->x.col, +1.0); - else - { npp_add_aij(npp, row, sed->x.col, -1.0); - row->lb -= 1.0; - } - if (y == sed->y.neg) - npp_add_aij(npp, row, sed->y.col, +1.0); - else - { npp_add_aij(npp, row, sed->y.col, -1.0); - row->lb -= 1.0; - } - if (c == 0) - npp_add_aij(npp, row, sed->c, +1.0); - else - { npp_add_aij(npp, row, sed->c, -1.0); - row->lb -= 1.0; - } - } - } - } - } - return; -} - -/*********************************************************************** -* npp_sat_encode_sum3 - encode 3-bit summation -* -* Given a set containing at least three literals this routine chooses -* some literals x, y, z from that set and encodes the equality -* -* x + y + z = s + 2 * c, (1) -* -* where -* -* s = (x + y + z) % 2 (2) -* -* is a binary variable modeling the low sum bit, and -* -* c = (x + y + z) / 2 (3) -* -* is a binary variable modeling the high (carry) sum bit. */ - -void npp_sat_encode_sum3(NPP *npp, NPPLSE *set, NPPSED *sed) -{ NPPROW *row; - int x, y, z, s, c; - /* the set should contain at least three literals */ - xassert(set != NULL); - xassert(set->next != NULL); - xassert(set->next->next != NULL); - sed->x = set->lit; - xassert(sed->x.neg == 0 || sed->x.neg == 1); - sed->y = set->next->lit; - xassert(sed->y.neg == 0 || sed->y.neg == 1); - sed->z = set->next->next->lit; - xassert(sed->z.neg == 0 || sed->z.neg == 1); - /* perform encoding s = (x + y + z) % 2 */ - sed->s = npp_add_col(npp); - sed->s->is_int = 1, sed->s->lb = 0.0, sed->s->ub = 1.0; - for (x = 0; x <= 1; x++) - { for (y = 0; y <= 1; y++) - { for (z = 0; z <= 1; z++) - { for (s = 0; s <= 1; s++) - { if ((x + y + z) % 2 != s) - { /* generate CNF clause to disable infeasible - combination */ - row = npp_add_row(npp); - row->lb = 1.0, row->ub = +DBL_MAX; - if (x == sed->x.neg) - npp_add_aij(npp, row, sed->x.col, +1.0); - else - { npp_add_aij(npp, row, sed->x.col, -1.0); - row->lb -= 1.0; - } - if (y == sed->y.neg) - npp_add_aij(npp, row, sed->y.col, +1.0); - else - { npp_add_aij(npp, row, sed->y.col, -1.0); - row->lb -= 1.0; - } - if (z == sed->z.neg) - npp_add_aij(npp, row, sed->z.col, +1.0); - else - { npp_add_aij(npp, row, sed->z.col, -1.0); - row->lb -= 1.0; - } - if (s == 0) - npp_add_aij(npp, row, sed->s, +1.0); - else - { npp_add_aij(npp, row, sed->s, -1.0); - row->lb -= 1.0; - } - } - } - } - } - } - /* perform encoding c = (x + y + z) / 2 */ - sed->c = npp_add_col(npp); - sed->c->is_int = 1, sed->c->lb = 0.0, sed->c->ub = 1.0; - for (x = 0; x <= 1; x++) - { for (y = 0; y <= 1; y++) - { for (z = 0; z <= 1; z++) - { for (c = 0; c <= 1; c++) - { if ((x + y + z) / 2 != c) - { /* generate CNF clause to disable infeasible - combination */ - row = npp_add_row(npp); - row->lb = 1.0, row->ub = +DBL_MAX; - if (x == sed->x.neg) - npp_add_aij(npp, row, sed->x.col, +1.0); - else - { npp_add_aij(npp, row, sed->x.col, -1.0); - row->lb -= 1.0; - } - if (y == sed->y.neg) - npp_add_aij(npp, row, sed->y.col, +1.0); - else - { npp_add_aij(npp, row, sed->y.col, -1.0); - row->lb -= 1.0; - } - if (z == sed->z.neg) - npp_add_aij(npp, row, sed->z.col, +1.0); - else - { npp_add_aij(npp, row, sed->z.col, -1.0); - row->lb -= 1.0; - } - if (c == 0) - npp_add_aij(npp, row, sed->c, +1.0); - else - { npp_add_aij(npp, row, sed->c, -1.0); - row->lb -= 1.0; - } - } - } - } - } - } - return; -} - -/*********************************************************************** -* npp_sat_encode_sum_ax - encode linear combination of 0-1 variables -* -* PURPOSE -* -* Given a linear combination of binary variables: -* -* sum a[j] x[j], (1) -* j -* -* which is the linear form of the specified row, this routine encodes -* (i.e. translates to CNF) the following equality: -* -* n -* sum |a[j]| t[j] = sum 2**(k-1) * y[k], (2) -* j k=1 -* -* where t[j] = x[j] (if a[j] > 0) or t[j] = 1 - x[j] (if a[j] < 0), -* and y[k] is either t[j] or a new literal created by the routine or -* a constant zero. Note that the sum in the right-hand side of (2) can -* be thought as a n-bit representation of the sum in the left-hand -* side, which is a non-negative integer number. -* -* ALGORITHM -* -* First, the number of bits, n, sufficient to represent any value in -* the left-hand side of (2) is determined. Obviously, n is the number -* of bits sufficient to represent the sum (sum |a[j]|). -* -* Let -* -* n -* |a[j]| = sum 2**(k-1) b[j,k], (3) -* k=1 -* -* where b[j,k] is k-th bit in a n-bit representation of |a[j]|. Then -* -* m n -* sum |a[j]| * t[j] = sum 2**(k-1) sum b[j,k] * t[j]. (4) -* j k=1 j=1 -* -* Introducing the set -* -* J[k] = { j : b[j,k] = 1 } (5) -* -* allows rewriting (4) as follows: -* -* n -* sum |a[j]| * t[j] = sum 2**(k-1) sum t[j]. (6) -* j k=1 j in J[k] -* -* Thus, our goal is to provide |J[k]| <= 1 for all k, in which case -* we will have the representation (1). -* -* Let |J[k]| = 2, i.e. J[k] has exactly two literals u and v. In this -* case we can apply the following transformation: -* -* u + v = s + 2 * c, (7) -* -* where s and c are, respectively, low (sum) and high (carry) bits of -* the sum of two bits. This allows to replace two literals u and v in -* J[k] by one literal s, and carry out literal c to J[k+1]. -* -* If |J[k]| >= 3, i.e. J[k] has at least three literals u, v, and w, -* we can apply the following transformation: -* -* u + v + w = s + 2 * c. (8) -* -* Again, literal s replaces literals u, v, and w in J[k], and literal -* c goes into J[k+1]. -* -* On exit the routine stores each literal from J[k] in element y[k], -* 1 <= k <= n. If J[k] is empty, y[k] is set to constant false. -* -* RETURNS -* -* The routine returns n, the number of literals in the right-hand side -* of (2), 0 <= n <= NBIT_MAX. If the sum (sum |a[j]|) is too large, so -* more than NBIT_MAX (= 31) literals are needed to encode the original -* linear combination, the routine returns a negative value. */ - -#define NBIT_MAX 31 -/* maximal number of literals in the right hand-side of (2) */ - -static NPPLSE *remove_lse(NPP *npp, NPPLSE *set, NPPCOL *col) -{ /* remove specified literal from specified literal set */ - NPPLSE *lse, *prev = NULL; - for (lse = set; lse != NULL; prev = lse, lse = lse->next) - if (lse->lit.col == col) break; - xassert(lse != NULL); - if (prev == NULL) - set = lse->next; - else - prev->next = lse->next; - dmp_free_atom(npp->pool, lse, sizeof(NPPLSE)); - return set; -} - -int npp_sat_encode_sum_ax(NPP *npp, NPPROW *row, NPPLIT y[]) -{ NPPAIJ *aij; - NPPLSE *set[1+NBIT_MAX], *lse; - NPPSED sed; - int k, n, temp; - double sum; - /* compute the sum (sum |a[j]|) */ - sum = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - sum += fabs(aij->val); - /* determine n, the number of bits in the sum */ - temp = (int)sum; - if ((double)temp != sum) - return -1; /* integer arithmetic error */ - for (n = 0; temp > 0; n++, temp >>= 1); - xassert(0 <= n && n <= NBIT_MAX); - /* build initial sets J[k], 1 <= k <= n; see (5) */ - /* set[k] is a pointer to the list of literals in J[k] */ - for (k = 1; k <= n; k++) - set[k] = NULL; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { temp = (int)fabs(aij->val); - xassert((int)temp == fabs(aij->val)); - for (k = 1; temp > 0; k++, temp >>= 1) - { if (temp & 1) - { xassert(k <= n); - lse = dmp_get_atom(npp->pool, sizeof(NPPLSE)); - lse->lit.col = aij->col; - lse->lit.neg = (aij->val > 0.0 ? 0 : 1); - lse->next = set[k]; - set[k] = lse; - } - } - } - /* main transformation loop */ - for (k = 1; k <= n; k++) - { /* reduce J[k] and set y[k] */ - for (;;) - { if (set[k] == NULL) - { /* J[k] is empty */ - /* set y[k] to constant false */ - y[k].col = NULL, y[k].neg = 0; - break; - } - if (set[k]->next == NULL) - { /* J[k] contains one literal */ - /* set y[k] to that literal */ - y[k] = set[k]->lit; - dmp_free_atom(npp->pool, set[k], sizeof(NPPLSE)); - break; - } - if (set[k]->next->next == NULL) - { /* J[k] contains two literals */ - /* apply transformation (7) */ - npp_sat_encode_sum2(npp, set[k], &sed); - } - else - { /* J[k] contains at least three literals */ - /* apply transformation (8) */ - npp_sat_encode_sum3(npp, set[k], &sed); - /* remove third literal from set[k] */ - set[k] = remove_lse(npp, set[k], sed.z.col); - } - /* remove second literal from set[k] */ - set[k] = remove_lse(npp, set[k], sed.y.col); - /* remove first literal from set[k] */ - set[k] = remove_lse(npp, set[k], sed.x.col); - /* include new literal s to set[k] */ - lse = dmp_get_atom(npp->pool, sizeof(NPPLSE)); - lse->lit.col = sed.s, lse->lit.neg = 0; - lse->next = set[k]; - set[k] = lse; - /* include new literal c to set[k+1] */ - xassert(k < n); /* FIXME: can "overflow" happen? */ - lse = dmp_get_atom(npp->pool, sizeof(NPPLSE)); - lse->lit.col = sed.c, lse->lit.neg = 0; - lse->next = set[k+1]; - set[k+1] = lse; - } - } - return n; -} - -/*********************************************************************** -* npp_sat_normalize_clause - normalize clause -* -* This routine normalizes the specified clause, which is a disjunction -* of literals, by replacing multiple literals, which refer to the same -* binary variable, with a single literal. -* -* On exit the routine returns the number of literals in the resulting -* clause. However, if the specified clause includes both a literal and -* its negation, the routine returns a negative value meaning that the -* clause is equivalent to the value true. */ - -int npp_sat_normalize_clause(NPP *npp, int size, NPPLIT lit[]) -{ int j, k, new_size; - xassert(npp == npp); - xassert(size >= 0); - new_size = 0; - for (k = 1; k <= size; k++) - { for (j = 1; j <= new_size; j++) - { if (lit[k].col == lit[j].col) - { /* lit[k] refers to the same variable as lit[j], which - is already included in the resulting clause */ - if (lit[k].neg == lit[j].neg) - { /* ignore lit[k] due to the idempotent law */ - goto skip; - } - else - { /* lit[k] is NOT lit[j]; the clause is equivalent to - the value true */ - return -1; - } - } - } - /* include lit[k] in the resulting clause */ - lit[++new_size] = lit[k]; -skip: ; - } - return new_size; -} - -/*********************************************************************** -* npp_sat_encode_clause - translate clause to cover inequality -* -* Given a clause -* -* OR t[j], (1) -* j in J -* -* where t[j] is a literal, i.e. t[j] = x[j] or t[j] = NOT x[j], this -* routine translates it to the following equivalent cover inequality, -* which is added to the transformed problem: -* -* sum t[j] >= 1, (2) -* j in J -* -* where t[j] = x[j] or t[j] = 1 - x[j]. -* -* If necessary, the clause should be normalized before a call to this -* routine. */ - -NPPROW *npp_sat_encode_clause(NPP *npp, int size, NPPLIT lit[]) -{ NPPROW *row; - int k; - xassert(size >= 1); - row = npp_add_row(npp); - row->lb = 1.0, row->ub = +DBL_MAX; - for (k = 1; k <= size; k++) - { xassert(lit[k].col != NULL); - if (lit[k].neg == 0) - npp_add_aij(npp, row, lit[k].col, +1.0); - else if (lit[k].neg == 1) - { npp_add_aij(npp, row, lit[k].col, -1.0); - row->lb -= 1.0; - } - else - xassert(lit != lit); - } - return row; -} - -/*********************************************************************** -* npp_sat_encode_geq - encode "not less than" constraint -* -* PURPOSE -* -* This routine translates to CNF the following constraint: -* -* n -* sum 2**(k-1) * y[k] >= b, (1) -* k=1 -* -* where y[k] is either a literal (i.e. y[k] = x[k] or y[k] = 1 - x[k]) -* or constant false (zero), b is a given lower bound. -* -* ALGORITHM -* -* If b < 0, the constraint is redundant, so assume that b >= 0. Let -* -* n -* b = sum 2**(k-1) b[k], (2) -* k=1 -* -* where b[k] is k-th binary digit of b. (Note that if b >= 2**n and -* therefore cannot be represented in the form (2), the constraint (1) -* is infeasible.) In this case the condition (1) is equivalent to the -* following condition: -* -* y[n] y[n-1] ... y[2] y[1] >= b[n] b[n-1] ... b[2] b[1], (3) -* -* where ">=" is understood lexicographically. -* -* Algorithmically the condition (3) can be tested as follows: -* -* for (k = n; k >= 1; k--) -* { if (y[k] < b[k]) -* y is less than b; -* if (y[k] > b[k]) -* y is greater than b; -* } -* y is equal to b; -* -* Thus, y is less than b iff there exists k, 1 <= k <= n, for which -* the following condition is satisfied: -* -* y[n] = b[n] AND ... AND y[k+1] = b[k+1] AND y[k] < b[k]. (4) -* -* Negating the condition (4) we have that y is not less than b iff for -* all k, 1 <= k <= n, the following condition is satisfied: -* -* y[n] != b[n] OR ... OR y[k+1] != b[k+1] OR y[k] >= b[k]. (5) -* -* Note that if b[k] = 0, the literal y[k] >= b[k] is always true, in -* which case the entire clause (5) is true and can be omitted. -* -* RETURNS -* -* Normally the routine returns zero. However, if the constraint (1) is -* infeasible, the routine returns non-zero. */ - -int npp_sat_encode_geq(NPP *npp, int n, NPPLIT y[], int rhs) -{ NPPLIT lit[1+NBIT_MAX]; - int j, k, size, temp, b[1+NBIT_MAX]; - xassert(0 <= n && n <= NBIT_MAX); - /* if the constraint (1) is redundant, do nothing */ - if (rhs < 0) - return 0; - /* determine binary digits of b according to (2) */ - for (k = 1, temp = rhs; k <= n; k++, temp >>= 1) - b[k] = temp & 1; - if (temp != 0) - { /* b >= 2**n; the constraint (1) is infeasible */ - return 1; - } - /* main transformation loop */ - for (k = 1; k <= n; k++) - { /* build the clause (5) for current k */ - size = 0; /* clause size = number of literals */ - /* add literal y[k] >= b[k] */ - if (b[k] == 0) - { /* b[k] = 0 -> the literal is true */ - goto skip; - } - else if (y[k].col == NULL) - { /* y[k] = 0, b[k] = 1 -> the literal is false */ - xassert(y[k].neg == 0); - } - else - { /* add literal y[k] = 1 */ - lit[++size] = y[k]; - } - for (j = k+1; j <= n; j++) - { /* add literal y[j] != b[j] */ - if (y[j].col == NULL) - { xassert(y[j].neg == 0); - if (b[j] == 0) - { /* y[j] = 0, b[j] = 0 -> the literal is false */ - continue; - } - else - { /* y[j] = 0, b[j] = 1 -> the literal is true */ - goto skip; - } - } - else - { lit[++size] = y[j]; - if (b[j] != 0) - lit[size].neg = 1 - lit[size].neg; - } - } - /* normalize the clause */ - size = npp_sat_normalize_clause(npp, size, lit); - if (size < 0) - { /* the clause is equivalent to the value true */ - goto skip; - } - if (size == 0) - { /* the clause is equivalent to the value false; this means - that the constraint (1) is infeasible */ - return 2; - } - /* translate the clause to corresponding cover inequality */ - npp_sat_encode_clause(npp, size, lit); -skip: ; - } - return 0; -} - -/*********************************************************************** -* npp_sat_encode_leq - encode "not greater than" constraint -* -* PURPOSE -* -* This routine translates to CNF the following constraint: -* -* n -* sum 2**(k-1) * y[k] <= b, (1) -* k=1 -* -* where y[k] is either a literal (i.e. y[k] = x[k] or y[k] = 1 - x[k]) -* or constant false (zero), b is a given upper bound. -* -* ALGORITHM -* -* If b < 0, the constraint is infeasible, so assume that b >= 0. Let -* -* n -* b = sum 2**(k-1) b[k], (2) -* k=1 -* -* where b[k] is k-th binary digit of b. (Note that if b >= 2**n and -* therefore cannot be represented in the form (2), the constraint (1) -* is redundant.) In this case the condition (1) is equivalent to the -* following condition: -* -* y[n] y[n-1] ... y[2] y[1] <= b[n] b[n-1] ... b[2] b[1], (3) -* -* where "<=" is understood lexicographically. -* -* Algorithmically the condition (3) can be tested as follows: -* -* for (k = n; k >= 1; k--) -* { if (y[k] < b[k]) -* y is less than b; -* if (y[k] > b[k]) -* y is greater than b; -* } -* y is equal to b; -* -* Thus, y is greater than b iff there exists k, 1 <= k <= n, for which -* the following condition is satisfied: -* -* y[n] = b[n] AND ... AND y[k+1] = b[k+1] AND y[k] > b[k]. (4) -* -* Negating the condition (4) we have that y is not greater than b iff -* for all k, 1 <= k <= n, the following condition is satisfied: -* -* y[n] != b[n] OR ... OR y[k+1] != b[k+1] OR y[k] <= b[k]. (5) -* -* Note that if b[k] = 1, the literal y[k] <= b[k] is always true, in -* which case the entire clause (5) is true and can be omitted. -* -* RETURNS -* -* Normally the routine returns zero. However, if the constraint (1) is -* infeasible, the routine returns non-zero. */ - -int npp_sat_encode_leq(NPP *npp, int n, NPPLIT y[], int rhs) -{ NPPLIT lit[1+NBIT_MAX]; - int j, k, size, temp, b[1+NBIT_MAX]; - xassert(0 <= n && n <= NBIT_MAX); - /* check if the constraint (1) is infeasible */ - if (rhs < 0) - return 1; - /* determine binary digits of b according to (2) */ - for (k = 1, temp = rhs; k <= n; k++, temp >>= 1) - b[k] = temp & 1; - if (temp != 0) - { /* b >= 2**n; the constraint (1) is redundant */ - return 0; - } - /* main transformation loop */ - for (k = 1; k <= n; k++) - { /* build the clause (5) for current k */ - size = 0; /* clause size = number of literals */ - /* add literal y[k] <= b[k] */ - if (b[k] == 1) - { /* b[k] = 1 -> the literal is true */ - goto skip; - } - else if (y[k].col == NULL) - { /* y[k] = 0, b[k] = 0 -> the literal is true */ - xassert(y[k].neg == 0); - goto skip; - } - else - { /* add literal y[k] = 0 */ - lit[++size] = y[k]; - lit[size].neg = 1 - lit[size].neg; - } - for (j = k+1; j <= n; j++) - { /* add literal y[j] != b[j] */ - if (y[j].col == NULL) - { xassert(y[j].neg == 0); - if (b[j] == 0) - { /* y[j] = 0, b[j] = 0 -> the literal is false */ - continue; - } - else - { /* y[j] = 0, b[j] = 1 -> the literal is true */ - goto skip; - } - } - else - { lit[++size] = y[j]; - if (b[j] != 0) - lit[size].neg = 1 - lit[size].neg; - } - } - /* normalize the clause */ - size = npp_sat_normalize_clause(npp, size, lit); - if (size < 0) - { /* the clause is equivalent to the value true */ - goto skip; - } - if (size == 0) - { /* the clause is equivalent to the value false; this means - that the constraint (1) is infeasible */ - return 2; - } - /* translate the clause to corresponding cover inequality */ - npp_sat_encode_clause(npp, size, lit); -skip: ; - } - return 0; -} - -/*********************************************************************** -* npp_sat_encode_row - encode constraint (row) of general type -* -* PURPOSE -* -* This routine translates to CNF the following constraint (row): -* -* L <= sum a[j] x[j] <= U, (1) -* j -* -* where all x[j] are binary variables. -* -* ALGORITHM -* -* First, the routine performs substitution x[j] = t[j] for j in J+ -* and x[j] = 1 - t[j] for j in J-, where J+ = { j : a[j] > 0 } and -* J- = { j : a[j] < 0 }. This gives: -* -* L <= sum a[j] t[j] + sum a[j] (1 - t[j]) <= U ==> -* j in J+ j in J- -* -* L' <= sum |a[j]| t[j] <= U', (2) -* j -* -* where -* -* L' = L - sum a[j], U' = U - sum a[j]. (3) -* j in J- j in J- -* -* (Actually only new bounds L' and U' are computed.) -* -* Then the routine translates to CNF the following equality: -* -* n -* sum |a[j]| t[j] = sum 2**(k-1) * y[k], (4) -* j k=1 -* -* where y[k] is either some t[j] or a new literal or a constant zero -* (see the routine npp_sat_encode_sum_ax). -* -* Finally, the routine translates to CNF the following conditions: -* -* n -* sum 2**(k-1) * y[k] >= L' (5) -* k=1 -* -* and -* -* n -* sum 2**(k-1) * y[k] <= U' (6) -* k=1 -* -* (see the routines npp_sat_encode_geq and npp_sat_encode_leq). -* -* All resulting clauses are encoded as cover inequalities and included -* into the transformed problem. -* -* Note that on exit the routine removes the specified constraint (row) -* from the original problem. -* -* RETURNS -* -* The routine returns one of the following codes: -* -* 0 - translation was successful; -* 1 - constraint (1) was found infeasible; -* 2 - integer arithmetic error occured. */ - -int npp_sat_encode_row(NPP *npp, NPPROW *row) -{ NPPAIJ *aij; - NPPLIT y[1+NBIT_MAX]; - int n, rhs; - double lb, ub; - /* the row should not be free */ - xassert(!(row->lb == -DBL_MAX && row->ub == +DBL_MAX)); - /* compute new bounds L' and U' (3) */ - lb = row->lb; - ub = row->ub; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - { if (aij->val < 0.0) - { if (lb != -DBL_MAX) - lb -= aij->val; - if (ub != -DBL_MAX) - ub -= aij->val; - } - } - /* encode the equality (4) */ - n = npp_sat_encode_sum_ax(npp, row, y); - if (n < 0) - return 2; /* integer arithmetic error */ - /* encode the condition (5) */ - if (lb != -DBL_MAX) - { rhs = (int)lb; - if ((double)rhs != lb) - return 2; /* integer arithmetic error */ - if (npp_sat_encode_geq(npp, n, y, rhs) != 0) - return 1; /* original constraint is infeasible */ - } - /* encode the condition (6) */ - if (ub != +DBL_MAX) - { rhs = (int)ub; - if ((double)rhs != ub) - return 2; /* integer arithmetic error */ - if (npp_sat_encode_leq(npp, n, y, rhs) != 0) - return 1; /* original constraint is infeasible */ - } - /* remove the specified row from the problem */ - npp_del_row(npp, row); - return 0; -} - -/*********************************************************************** -* npp_sat_encode_prob - encode 0-1 feasibility problem -* -* This routine translates the specified 0-1 feasibility problem to an -* equivalent SAT-CNF problem. -* -* N.B. Currently this is a very crude implementation. -* -* RETURNS -* -* 0 success; -* -* GLP_ENOPFS primal/integer infeasibility detected; -* -* GLP_ERANGE integer overflow occured. */ - -int npp_sat_encode_prob(NPP *npp) -{ NPPROW *row, *next_row, *prev_row; - NPPCOL *col, *next_col; - int cover = 0, pack = 0, partn = 0, ret; - /* process and remove free rows */ - for (row = npp->r_head; row != NULL; row = next_row) - { next_row = row->next; - if (row->lb == -DBL_MAX && row->ub == +DBL_MAX) - npp_sat_free_row(npp, row); - } - /* process and remove fixed columns */ - for (col = npp->c_head; col != NULL; col = next_col) - { next_col = col->next; - if (col->lb == col->ub) - xassert(npp_sat_fixed_col(npp, col) == 0); - } - /* only binary variables should remain */ - for (col = npp->c_head; col != NULL; col = col->next) - xassert(col->is_int && col->lb == 0.0 && col->ub == 1.0); - /* new rows may be added to the end of the row list, so we walk - from the end to beginning of the list */ - for (row = npp->r_tail; row != NULL; row = prev_row) - { prev_row = row->prev; - /* process special cases */ - ret = npp_sat_is_cover_ineq(npp, row); - if (ret != 0) - { /* row is covering inequality */ - cover++; - /* since it already encodes a clause, just transform it to - canonical form */ - if (ret == 2) - { xassert(npp_sat_reverse_row(npp, row) == 0); - ret = npp_sat_is_cover_ineq(npp, row); - } - xassert(ret == 1); - continue; - } - ret = npp_sat_is_partn_eq(npp, row); - if (ret != 0) - { /* row is partitioning equality */ - NPPROW *cov; - NPPAIJ *aij; - partn++; - /* transform it to canonical form */ - if (ret == 2) - { xassert(npp_sat_reverse_row(npp, row) == 0); - ret = npp_sat_is_partn_eq(npp, row); - } - xassert(ret == 1); - /* and split it into covering and packing inequalities, - both in canonical forms */ - cov = npp_add_row(npp); - cov->lb = row->lb, cov->ub = +DBL_MAX; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - npp_add_aij(npp, cov, aij->col, aij->val); - xassert(npp_sat_is_cover_ineq(npp, cov) == 1); - /* the cover inequality already encodes a clause and do - not need any further processing */ - row->lb = -DBL_MAX; - xassert(npp_sat_is_pack_ineq(npp, row) == 1); - /* the packing inequality will be processed below */ - pack--; - } - ret = npp_sat_is_pack_ineq(npp, row); - if (ret != 0) - { /* row is packing inequality */ - NPPROW *rrr; - int nlit, desired_nlit = 4; - pack++; - /* transform it to canonical form */ - if (ret == 2) - { xassert(npp_sat_reverse_row(npp, row) == 0); - ret = npp_sat_is_pack_ineq(npp, row); - } - xassert(ret == 1); - /* process the packing inequality */ - for (;;) - { /* determine the number of literals in the remaining - inequality */ - nlit = npp_row_nnz(npp, row); - if (nlit <= desired_nlit) - break; - /* split the current inequality into one having not more - than desired_nlit literals and remaining one */ - rrr = npp_sat_split_pack(npp, row, desired_nlit-1); - /* translate the former inequality to CNF and remove it - from the original problem */ - npp_sat_encode_pack(npp, rrr); - } - /* translate the remaining inequality to CNF and remove it - from the original problem */ - npp_sat_encode_pack(npp, row); - continue; - } - /* translate row of general type to CNF and remove it from the - original problem */ - ret = npp_sat_encode_row(npp, row); - if (ret == 0) - ; - else if (ret == 1) - ret = GLP_ENOPFS; - else if (ret == 2) - ret = GLP_ERANGE; - else - xassert(ret != ret); - if (ret != 0) - goto done; - } - ret = 0; - if (cover != 0) - xprintf("%d covering inequalities\n", cover); - if (pack != 0) - xprintf("%d packing inequalities\n", pack); - if (partn != 0) - xprintf("%d partitioning equalities\n", partn); -done: return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/proxy/main.c b/code/3rd_glpk/proxy/main.c deleted file mode 100644 index a7d1e2b8..00000000 --- a/code/3rd_glpk/proxy/main.c +++ /dev/null @@ -1,87 +0,0 @@ -/* Last update: 08-May-2013 */ - -#include -#include -#include - -#include "glpk.h" -#include "proxy.h" - -/**********************************************************************/ -int main(int argc, char **argv) -/**********************************************************************/ -{ - glp_prob *lp; - int ncols, status; - double *initsol, zstar, *xstar; - - /* check arguments */ - if ( (argc == 1) || (argc > 3) ) { - printf("ERROR: Usage: ts <(possibly) xml initsols>\n" - ); - exit(1); - } - - /* creating the problem */ - lp = glp_create_prob(); - glp_set_prob_name(lp, "Proxy"); - - /* reading the problem */ - glp_term_out(GLP_OFF); -#if 0 /* by mao */ - status = glp_read_lp(lp, NULL, argv[1]); -#else - status = glp_read_mps(lp, GLP_MPS_FILE, NULL, argv[1]); -#endif - glp_term_out(GLP_ON); - if ( status ) { - printf("Problem %s does not exist!!!, status %d\n", - argv[1], status); - exit(1); - } - - ncols = glp_get_num_cols(lp); - - initsol = (double *) calloc(ncols+1, sizeof(double)); - - if (argc == 3) { - FILE *fp=fopen(argv[2],"r"); - char tmp[256]={0x0}; - int counter = 1; - while(fp!=NULL && fgets(tmp, sizeof(tmp),fp)!=NULL) - { - char *valini = strstr(tmp, "value"); - if (valini!=NULL){ - int num; - double dnum; - valini +=7; - sscanf(valini, "%d%*s",&num); - dnum = (double)num; - initsol[counter] = dnum; - counter++; - } - } - fclose(fp); - } - - xstar = (double *) calloc(ncols+1, sizeof(double)); - - if (argc == 3) { - status = proxy(lp, &zstar, xstar, initsol, 0.0, 0, 1); - } - else { - status = proxy(lp, &zstar, xstar, NULL, 0.0, 0, 1); - } - - printf("Status = %d; ZSTAR = %f\n",status,zstar); - /* - int i; - for (i=1; i< ncols+1; i++) { - printf("XSTAR[%d] = %f\n",i, xstar[i]); - } - */ - - glp_delete_prob(lp); - - return 0; -} diff --git a/code/3rd_glpk/proxy/proxy.c b/code/3rd_glpk/proxy/proxy.c deleted file mode 100644 index 7d890003..00000000 --- a/code/3rd_glpk/proxy/proxy.c +++ /dev/null @@ -1,1073 +0,0 @@ -/* proxy.c (proximity search heuristic algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Author: Giorgio Sartor <0gioker0@gmail.com>. -* -* Copyright (C) 2013, 2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -* -************************************************************************ -* -* THIS CODE IS AN IMPLEMENTATION OF THE ALGORITHM PROPOSED IN -* -* M. Fischetti, M. Monaci, -* "Proximity Search for 0-1 Mixed-Integer Convex Programming" -* Technical Report DEI, University of Padua, March 2013. -* -* AVAILABLE AT -* http://www.dei.unipd.it/~fisch/papers/proximity_search.pdf -* -* THE CODE HAS BEEN WRITTEN BY GIORGIO SARTOR, " 0gioker0@gmail.com " -* -* BASIC IDEA: -* -* The initial feasible solution x_tilde is defined. This initial -* solution can be found by an ad-hoc heuristic and proxy can be used to -* refine it by exploiting an underlying MIP model whose solution from -* scratch turned out to be problematic. Otherwise, x_tilde can be found -* by running the GLPK mip solver until a first feasible solution is -* found, setting a conservative time limit of 10 minutes (by default). -* Time limit can be modified passing variable tlim [ms]. -* -* Then the cutoff tolerance "delta" is defined. The default tolerance -* is 1% of the last feasible solution obj value--rounded to integer if -* all the variables and obj coefficients are integer. -* -* Next, the objective function c' x is replaced by the Hamming distance -* between x (the actual obj coefficients) and x_tilde (the given -* solution). Distance is only computed wrt the binary variables. -* -* The GLPK solver is then invoked to hopefully find a new incumbent -* x_star with cost c' x_star <= c' x_tilde - delta. A crucial property -* here is that the root-node solution of the LP relaxation is expected -* to be not too different from x_tilde, as this latter solution would -* be optimal without the cutoff constraint, that for a small delta can -* typically be fulfilled with just local adjustments. -* -* If no new solution x_star is found within the time limit the -* algorithm stops. Of course, if the MIP solver proved infeasibility -* for the given delta, we have that c' x_tilde - delta is a valid lower -* bound (in case of minimazation) on the optimal value of the original -* MIP. -* -* The new solution x_star, if any, is possibly improved by solving a -* simplified problem (refinement) where all binary variables have been -* fixed to their value in x_star so as to find the best solution within -* the neighborhood. -* -* Finally, the approach is reapplied on x_star (that replaces x_tilde) -* so as to recenter the distance Hamming function and by modifying the -* cutoff tolerance delta. -* -* In this way, there will be a series of hopefully not-too-difficult -* sub-MIPs to solve, each leading to an improvement of the incumbent. -* More aggressive policies on the definition of tolerance delta can -* lead to a better performance, but would require an ad-hoc tuning. -* -************************************************************************ -* -* int proxy(glp_prob *lp, double *zstar, double *xstar, -* const double[] initsol, double rel_impr, int tlim, -* int verbose) -* -* lp : GLPK problem pointer to a MIP with binary variables -* -* zstar : the value of objective function of the best solution found -* -* xstar : best solution with components xstar[1],...,xstar[ncols] -* -* initsol : pointer to a initial feasible solution, see -* glp_ios_heur_sol -* If initsol = NULL, the procedure finds the first solution -* by itself. -* -* rel_impr : minimum relative obj improvement to be achieved at each -* internal step; if <= 0.0 a default value of 0.01 (1%) is -* used; for some problems (e.g., set covering with small -* integer costs) a more-conservative choice of 0.001 (0.1%) -* can lead to a better final solution; values larger than -* 0.05 (5%) are typically too aggressive and do not work -* well. -* -* tlim : time limit to find a new solution, in ms. -* If tlim = 0, it is set to its default value, 600000 ms -* -* verbose : if 1 the output is activated. If 0 only errors are -* displayed -* -* The procedure returns -1 if an error occurred, 0 otherwise (possibly, -* time limit) -* -***********************************************************************/ - -/**********************************************************************/ -/* 1. INCLUDE */ -/**********************************************************************/ - -#include "glpk.h" -#include "env.h" -#include "proxy.h" - -/**********************************************************************/ -/* 2. PARAMETERS AND CONSTANTS */ -/**********************************************************************/ - -#define TDAY 86400.0 -#define TRUE 1 -#define FALSE 0 -#define EPS 1e-6 -#define RINF 1e38 -#define MAXVAL 1e20 -#define MINVAL -1e20 -#if 0 /* by gioker */ - #define PROXY_DEBUG -#endif - -/**********************************************************************/ -/* 3. GLOBAL VARIABLES */ -/**********************************************************************/ - -struct csa { - -int integer_obj; /* TRUE if each feasible solution has an - integral cost */ -int b_vars_exist; /* TRUE if there is at least one binary - variable in the problem */ -int i_vars_exist; /* TRUE if there is at least one general - integer variable in the problem */ -const double *startsol; /* Pointer to the initial solution */ - -int *ckind; /* Store the kind of the structural variables - of the problem */ -double *clb; /* Store the lower bound on the structural - variables of the problem */ -double *cub; /* Store the upper bound on the structural - variables of the problem */ -double *true_obj; /* Store the obj coefficients of the problem */ - -int dir; /* Minimization or maximization problem */ -int ncols; /* Number of structural variables of the - problem */ - -time_t GLOtstart; /* starting time of the algorithm */ - -glp_prob *lp_ref; /* glp problem for refining only*/ - -}; - -/**********************************************************************/ -/* 4. FUNCTIONS PROTOTYPES */ -/**********************************************************************/ - -static void callback(glp_tree *tree, void *info); -static void get_info(struct csa *csa, glp_prob *lp); -static int is_integer(struct csa *csa); -static void check_integrality(struct csa *csa); -static int check_ref(struct csa *csa, glp_prob *lp, double *xref); -static double second(void); -static int add_cutoff(struct csa *csa, glp_prob *lp); -static void get_sol(struct csa *csa, glp_prob *lp, double *xstar); -static double elapsed_time(struct csa *csa); -static void redefine_obj(glp_prob *lp, double *xtilde, int ncols, - int *ckind, double *clb, double *cub); -static double update_cutoff(struct csa *csa, glp_prob *lp, - double zstar, int index, double rel_impr); -static double compute_delta(struct csa *csa, double z, - double rel_impr); -static double objval(int ncols, double *x, double *true_obj); -static void array_copy(int begin, int end, double *source, - double *destination); -static int do_refine(struct csa *csa, glp_prob *lp_ref, int ncols, - int *ckind, double *xref, int *tlim, int tref_lim, - int verbose); -static void deallocate(struct csa *csa, int refine); - -/**********************************************************************/ -/* 5. FUNCTIONS */ -/**********************************************************************/ - -int proxy(glp_prob *lp, double *zfinal, double *xfinal, - const double initsol[], double rel_impr, int tlim, - int verbose) - -{ struct csa csa_, *csa = &csa_; - glp_iocp parm; - glp_smcp parm_lp; - size_t tpeak; - int refine, tref_lim, err, cutoff_row, niter, status, i, tout; - double *xref, *xstar, zstar, tela, cutoff, zz; - - memset(csa, 0, sizeof(struct csa)); - - - /********** **********/ - /********** RETRIEVING PROBLEM INFO **********/ - /********** **********/ - - /* getting problem direction (min or max) */ - csa->dir = glp_get_obj_dir(lp); - - /* getting number of variables */ - csa->ncols = glp_get_num_cols(lp); - - /* getting kind, bounds and obj coefficient of each variable - information is stored in ckind, cub, clb, true_obj */ - get_info(csa, lp); - - /* checking if the objective function is always integral */ - check_integrality(csa); - - /* Proximity search cannot be used if there are no binary - variables */ - if (csa->b_vars_exist == FALSE) { - if (verbose) { - xprintf("The problem has not binary variables. Proximity se" - "arch cannot be used.\n"); - } - tfree(csa->ckind); - tfree(csa->clb); - tfree(csa->cub); - tfree(csa->true_obj); - return -1; - } - - /* checking if the problem needs refinement, i.e., not all - variables are binary. If so, the routine creates a copy of the - lp problem named lp_ref and initializes the solution xref to - zero. */ - xref = talloc(csa->ncols+1, double); -#if 0 /* by mao */ - memset(xref, 0, sizeof(double)*(csa->ncols+1)); -#endif - refine = check_ref(csa, lp, xref); -#ifdef PROXY_DEBUG - xprintf("REFINE = %d\n",refine); -#endif - - /* Initializing the solution */ - xstar = talloc(csa->ncols+1, double); -#if 0 /* by mao */ - memset(xstar, 0, sizeof(double)*(csa->ncols+1)); -#endif - - /********** **********/ - /********** FINDING FIRST SOLUTION **********/ - /********** **********/ - - if (verbose) { - xprintf("Applying PROXY heuristic...\n"); - } - - /* get the initial time */ - csa->GLOtstart = second(); - - /* setting the optimization parameters */ - glp_init_iocp(&parm); - glp_init_smcp(&parm_lp); -#if 0 /* by gioker */ - /* Preprocessing should be disabled because the mip passed - to proxy is already preprocessed */ - parm.presolve = GLP_ON; -#endif -#if 1 /* by mao */ - /* best projection backtracking seems to be more efficient to find - any integer feasible solution */ - parm.bt_tech = GLP_BT_BPH; -#endif - - /* Setting the default value of the minimum relative improvement - to 1% */ - if ( rel_impr <= 0.0 ) { - rel_impr = 0.01; - } - - /* Setting the default value of time limit to 10 minutes */ - if (tlim <= 0) { - tlim = INT_MAX; - } - if (verbose) { - xprintf("Proxy's time limit set to %d seconds.\n",tlim/1000); - xprintf("Proxy's relative improvement " - "set to %2.2lf %c.\n",rel_impr*100,37); - } - - parm_lp.tm_lim = tlim; - - parm.mip_gap = 9999999.9; /* to stop the optimization at the first - feasible solution found */ - - /* finding the first solution */ - if (verbose) { - xprintf("Searching for a feasible solution...\n"); - } - - /* verifying the existence of an input starting solution */ - if (initsol != NULL) { - csa->startsol = initsol; - parm.cb_func = callback; - parm.cb_info = csa; - if (verbose) { - xprintf("Input solution found.\n"); - } - } - - tout = glp_term_out(GLP_OFF); - err = glp_simplex(lp,&parm_lp); - glp_term_out(tout); - - status = glp_get_status(lp); - - if (status != GLP_OPT) { - if (verbose) { - xprintf("Proxy heuristic terminated.\n"); - } -#ifdef PROXY_DEBUG - /* For debug only */ - xprintf("GLP_SIMPLEX status = %d\n",status); - xprintf("GLP_SIMPLEX error code = %d\n",err); -#endif - tfree(xref); - tfree(xstar); - deallocate(csa, refine); - return -1; - } - - tela = elapsed_time(csa); - if (tlim-tela*1000 <= 0) { - if (verbose) { - xprintf("Time limit exceeded. Proxy could not " - "find optimal solution to LP relaxation.\n"); - xprintf("Proxy heuristic aborted.\n"); - } - tfree(xref); - tfree(xstar); - deallocate(csa, refine); - return -1; - } - - parm.tm_lim = tlim - tela*1000; - tref_lim = (tlim - tela *1000) / 20; - - tout = glp_term_out(GLP_OFF); - err = glp_intopt(lp, &parm); - glp_term_out(tout); - - status = glp_mip_status(lp); - - /***** If no solution was found *****/ - - if (status == GLP_NOFEAS || status == GLP_UNDEF) { - if (err == GLP_ETMLIM) { - if (verbose) { - xprintf("Time limit exceeded. Proxy could not " - "find an initial integer feasible solution.\n"); - xprintf("Proxy heuristic aborted.\n"); - } - } - else { - if (verbose) { - xprintf("Proxy could not " - "find an initial integer feasible solution.\n"); - xprintf("Proxy heuristic aborted.\n"); - } - } - tfree(xref); - tfree(xstar); - deallocate(csa, refine); - return -1; - } - - /* getting the first solution and its value */ - get_sol(csa, lp,xstar); - zstar = glp_mip_obj_val(lp); - - if (verbose) { - xprintf(">>>>> first solution = %e;\n", zstar); - } - - /* If a feasible solution was found but the time limit is - exceeded */ - if (err == GLP_ETMLIM) { - if (verbose) { - xprintf("Time limit exceeded. Proxy heuristic terminated.\n"); - } - goto done; - } - - tela = elapsed_time(csa); - tpeak = 0; - glp_mem_usage(NULL, NULL, NULL, &tpeak); - if (verbose) { - xprintf("Time used: %3.1lf secs. Memory used: %2.1lf Mb\n", - tela,(double)tpeak/1048576); - xprintf("Starting proximity search...\n"); - } - - /********** **********/ - /********** PREPARING THE PROBLEM FOR PROXY **********/ - /********** **********/ - - /* adding a dummy cutoff constraint */ - cutoff_row = add_cutoff(csa, lp); - - /* proximity search needs minimization direction - even if the problem is a maximization one */ - if (csa->dir == GLP_MAX) { - glp_set_obj_dir(lp, GLP_MIN); - } - - /********** **********/ - /********** STARTING PROXIMITY SEARCH **********/ - /********** **********/ - - - niter = 0; - - while (TRUE) { - niter++; - - /********** CHANGING THE OBJ FUNCTION **********/ - - redefine_obj(lp,xstar, csa->ncols, csa->ckind, csa->clb, - csa->cub); - - /********** UPDATING THE CUTOFF CONSTRAINT **********/ - - cutoff = update_cutoff(csa, lp,zstar, cutoff_row, rel_impr); - -#ifdef PROXY_DEBUG - xprintf("TRUE_OBJ[0] = %f\n",csa->true_obj[0]); - xprintf("ZSTAR = %f\n",zstar); - xprintf("CUTOFF = %f\n",cutoff); -#endif - - /********** SEARCHING FOR A BETTER SOLUTION **********/ - - tela = elapsed_time(csa); - if (tlim-tela*1000 <= 0) { - if (verbose) { - xprintf("Time limit exceeded. Proxy heuristic " - "terminated.\n"); - } - goto done; - } -#ifdef PROXY_DEBUG - xprintf("TELA = %3.1lf\n",tela*1000); - xprintf("TLIM = %3.1lf\n",tlim - tela*1000); -#endif - parm_lp.tm_lim = tlim -tela*1000; - - tout = glp_term_out(GLP_OFF); - err = glp_simplex(lp,&parm_lp); - glp_term_out(tout); - - status = glp_get_status(lp); - - if (status != GLP_OPT) { - if (status == GLP_NOFEAS) { - if (verbose) { - xprintf("Bound exceeded = %f. ",cutoff); - } - } - if (verbose) { - xprintf("Proxy heuristic terminated.\n"); - } -#ifdef PROXY_DEBUG - xprintf("GLP_SIMPLEX status = %d\n",status); - xprintf("GLP_SIMPLEX error code = %d\n",err); -#endif - goto done; - } - - tela = elapsed_time(csa); - if (tlim-tela*1000 <= 0) { - if (verbose) { - xprintf("Time limit exceeded. Proxy heuristic " - "terminated.\n"); - } - goto done; - } - parm.tm_lim = tlim - tela*1000; - parm.cb_func = NULL; -#if 0 /* by gioker */ - /* Preprocessing should be disabled because the mip passed - to proxy is already preprocessed */ - parm.presolve = GLP_ON; -#endif - tout = glp_term_out(GLP_OFF); - err = glp_intopt(lp, &parm); - glp_term_out(tout); - - /********** MANAGEMENT OF THE SOLUTION **********/ - - status = glp_mip_status(lp); - - /***** No feasible solutions *****/ - - if (status == GLP_NOFEAS) { - if (verbose) { - xprintf("Bound exceeded = %f. Proxy heuristic " - "terminated.\n",cutoff); - } - goto done; - } - - /***** Undefined solution *****/ - - if (status == GLP_UNDEF) { - if (err == GLP_ETMLIM) { - if (verbose) { - xprintf("Time limit exceeded. Proxy heuristic " - "terminated.\n"); - } - } - else { - if (verbose) { - xprintf("Proxy terminated unexpectedly.\n"); -#ifdef PROXY_DEBUG - xprintf("GLP_INTOPT error code = %d\n",err); -#endif - } - } - goto done; - } - - /***** Feasible solution *****/ - - if ((status == GLP_FEAS) || (status == GLP_OPT)) { - - /* getting the solution and computing its value */ - get_sol(csa, lp,xstar); - zz = objval(csa->ncols, xstar, csa->true_obj); - - /* Comparing the incumbent solution with the current best - one */ -#ifdef PROXY_DEBUG - xprintf("ZZ = %f\n",zz); - xprintf("ZSTAR = %f\n",zstar); - xprintf("REFINE = %d\n",refine); -#endif - if (((zzdir == GLP_MIN)) || - ((zz>zstar) && (csa->dir == GLP_MAX))) { - - /* refining (possibly) the solution */ - if (refine) { - - /* copying the incumbent solution in the refinement - one */ - array_copy(1, csa->ncols +1, xstar, xref); - err = do_refine(csa, csa->lp_ref, csa->ncols, - csa->ckind, xref, &tlim, tref_lim, verbose); - if (!err) { - double zref = objval(csa->ncols, xref, - csa->true_obj); - if (((zrefdir == GLP_MIN)) || - ((zref>zz) && (csa->dir == GLP_MAX))) { - zz = zref; - /* copying the refinement solution in the - incumbent one */ - array_copy(1, csa->ncols +1, xref, xstar); - } - } - } - zstar = zz; - tela = elapsed_time(csa); - if (verbose) { - xprintf(">>>>> it: %3d: mip = %e; elapsed time " - "%3.1lf sec.s\n", niter,zstar,tela); - } - } - } - } - -done: - tela = elapsed_time(csa); - glp_mem_usage(NULL, NULL, NULL, &tpeak); - if (verbose) { - xprintf("Time used: %3.1lf. Memory used: %2.1lf Mb\n", - tela,(double)tpeak/1048576); - } - - - /* Exporting solution and obj val */ - *zfinal = zstar; - - for (i=1; i < (csa->ncols + 1); i++) { - xfinal[i]=xstar[i]; - } - - /* Freeing allocated memory */ - tfree(xref); - tfree(xstar); - deallocate(csa, refine); - - return 0; -} - -/**********************************************************************/ -static void callback(glp_tree *tree, void *info){ -/**********************************************************************/ - struct csa *csa = info; - switch(glp_ios_reason(tree)) { - case GLP_IHEUR: - glp_ios_heur_sol(tree, csa->startsol); - break; - default: break; - } -} - -/**********************************************************************/ -static void get_info(struct csa *csa, glp_prob *lp) -/**********************************************************************/ -{ - int i; - - /* Storing helpful info of the problem */ - - csa->ckind = talloc(csa->ncols+1, int); -#if 0 /* by mao */ - memset(csa->ckind, 0, sizeof(int)*(csa->ncols+1)); -#endif - csa->clb = talloc(csa->ncols+1, double); -#if 0 /* by mao */ - memset(csa->clb, 0, sizeof(double)*(csa->ncols+1)); -#endif - csa->cub = talloc(csa->ncols+1, double); -#if 0 /* by mao */ - memset(csa->cub, 0, sizeof(double)*(csa->ncols+1)); -#endif - csa->true_obj = talloc(csa->ncols+1, double); -#if 0 /* by mao */ - memset(csa->true_obj, 0, sizeof(double)*(csa->ncols+1)); -#endif - for( i = 1 ; i < (csa->ncols + 1); i++ ) { - csa->ckind[i] = glp_get_col_kind(lp, i); - csa->clb[i] = glp_get_col_lb(lp, i); - csa->cub[i] = glp_get_col_ub(lp, i); - csa->true_obj[i] = glp_get_obj_coef(lp, i); - } - csa->true_obj[0] = glp_get_obj_coef(lp, 0); -} - -/**********************************************************************/ -static int is_integer(struct csa *csa) -/**********************************************************************/ -{ - int i; - csa->integer_obj = TRUE; - for ( i = 1; i < (csa->ncols + 1); i++ ) { - if (fabs(csa->true_obj[i]) > INT_MAX ) { - csa->integer_obj = FALSE; - } - if (fabs(csa->true_obj[i]) <= INT_MAX) { - double tmp, rem; - if (fabs(csa->true_obj[i]) - floor(fabs(csa->true_obj[i])) - < 0.5) { - tmp = floor(fabs(csa->true_obj[i])); - } - else { - tmp = ceil(fabs(csa->true_obj[i])); - } - rem = fabs(csa->true_obj[i]) - tmp; - rem = fabs(rem); - if (rem > EPS) { - csa->integer_obj = FALSE; - } - - } - } - return csa->integer_obj; -} - -/**********************************************************************/ -static void check_integrality(struct csa *csa) -/**********************************************************************/ -{ - /* - Checking if the problem has binary, integer or continuos variables. - integer_obj is TRUE if the problem has no continuous variables - and all the obj coefficients are integer (and < INT_MAX). - */ - - int i; - csa->integer_obj = is_integer(csa); - csa->b_vars_exist = FALSE; - csa->i_vars_exist = FALSE; - for ( i = 1; i < (csa->ncols + 1); i++ ) { - if ( csa->ckind[i] == GLP_IV ){ - csa->i_vars_exist = TRUE; - continue; - } - if ( csa->ckind[i] == GLP_BV ){ - csa->b_vars_exist =TRUE; - continue; - } - csa->integer_obj = FALSE; - } -} - -/**********************************************************************/ -static int check_ref(struct csa *csa, glp_prob *lp, double *xref) -/**********************************************************************/ -{ - /* - checking if the problem has continuos or integer variables. If so, - refinement is prepared. - */ - int refine = FALSE; - int i; - for ( i = 1; i < (csa->ncols + 1); i++ ) { - if ( csa->ckind[i] != GLP_BV) { - refine = TRUE; - break; - } - } - - /* possibly creating a mip clone for refinement only */ - if ( refine ) { - csa->lp_ref = glp_create_prob(); - glp_copy_prob(csa->lp_ref, lp, GLP_ON); - } - - return refine; -} - -/**********************************************************************/ -static double second(void) -/**********************************************************************/ -{ -#if 0 /* by mao */ - return ((double)clock()/(double)CLOCKS_PER_SEC); -#else - return xtime() / 1000.0; -#endif -} - -/**********************************************************************/ -static int add_cutoff(struct csa *csa, glp_prob *lp) -/**********************************************************************/ -{ - /* - Adding a cutoff constraint to set an upper bound (in case of - minimaztion) on the obj value of the next solution, i.e., the next - value of the true obj function that we would like to find - */ - - /* store non-zero coefficients in the objective function */ - int *obj_index = talloc(csa->ncols+1, int); -#if 0 /* by mao */ - memset(obj_index, 0, sizeof(int)*(csa->ncols+1)); -#endif - double *obj_value = talloc(csa->ncols+1, double); -#if 0 /* by mao */ - memset(obj_value, 0, sizeof(double)*(csa->ncols+1)); -#endif - int obj_nzcnt = 0; - int i, irow; - const char *rowname; - for ( i = 1; i < (csa->ncols + 1); i++ ) { - if ( fabs(csa->true_obj[i]) > EPS ) { - obj_nzcnt++; - obj_index[obj_nzcnt] = i; - obj_value[obj_nzcnt] = csa->true_obj[i]; - } - } - - irow = glp_add_rows(lp, 1); - rowname = "Cutoff"; - glp_set_row_name(lp, irow, rowname); - if (csa->dir == GLP_MIN) { - /* minimization problem */ - glp_set_row_bnds(lp, irow, GLP_UP, MAXVAL, MAXVAL); - } - else { - /* maximization problem */ - glp_set_row_bnds(lp, irow, GLP_LO, MINVAL, MINVAL); - } - - glp_set_mat_row(lp, irow, obj_nzcnt, obj_index, obj_value); - - tfree(obj_index); - tfree(obj_value); - - return irow; -} - -/**********************************************************************/ -static void get_sol(struct csa *csa, glp_prob *lp, double *xstar) -/**********************************************************************/ -{ - /* Retrieving and storing the coefficients of the solution */ - - int i; - for (i = 1; i < (csa->ncols +1); i++) { - xstar[i] = glp_mip_col_val(lp, i); - } -} - -/**********************************************************************/ -static double elapsed_time(struct csa *csa) -/**********************************************************************/ -{ - double tela = second() - csa->GLOtstart; - if ( tela < 0 ) tela += TDAY; - return(tela); -} - -/**********************************************************************/ -static void redefine_obj(glp_prob *lp, double *xtilde, int ncols, - int *ckind, double *clb, double *cub) -/**********************************************************************/ - -/* - Redefine the lp objective function obj as the distance-to-integrality - (Hamming distance) from xtilde (the incumbent feasible solution), wrt - to binary vars only - */ - -{ - int j; - double *delta = talloc(ncols+1, double); -#if 0 /* by mao */ - memset(delta, 0, sizeof(double)*(ncols+1)); -#endif - - for ( j = 1; j < (ncols +1); j++ ) { - delta[j] = 0.0; - /* skip continuous variables */ - if ( ckind[j] == GLP_CV ) continue; - - /* skip integer variables that have been fixed */ - if ( cub[j]-clb[j] < 0.5 ) continue; - - /* binary variable */ - if ( ckind[j] == GLP_BV ) { - if ( xtilde[j] > 0.5 ) { - delta[j] = -1.0; - } - else { - delta[j] = 1.0; - } - } - } - - /* changing the obj coeff. for all variables, including continuous - ones */ - for ( j = 1; j < (ncols +1); j++ ) { - glp_set_obj_coef(lp, j, delta[j]); - } - glp_set_obj_coef(lp, 0, 0.0); - - tfree(delta); -} - -/**********************************************************************/ -static double update_cutoff(struct csa *csa, glp_prob *lp, - double zstar, int cutoff_row, - double rel_impr) -/**********************************************************************/ -{ - /* - Updating the cutoff constraint with the value we would like to - find during the next optimization - */ - double cutoff; - zstar -= csa->true_obj[0]; - if (csa->dir == GLP_MIN) { - cutoff = zstar - compute_delta(csa, zstar, rel_impr); - glp_set_row_bnds(lp, cutoff_row, GLP_UP, cutoff, cutoff); - } - else { - cutoff = zstar + compute_delta(csa, zstar, rel_impr); - glp_set_row_bnds(lp, cutoff_row, GLP_LO, cutoff, cutoff); - } - - return cutoff; -} - -/**********************************************************************/ -static double compute_delta(struct csa *csa, double z, double rel_impr) -/**********************************************************************/ -{ - /* Computing the offset for the next best solution */ - - double delta = rel_impr * fabs(z); - if ( csa->integer_obj ) delta = ceil(delta); - - return(delta); -} - -/**********************************************************************/ -static double objval(int ncols, double *x, double *true_obj) -/**********************************************************************/ -{ - /* Computing the true cost of x (using the original obj coeff.s) */ - - int j; - double z = 0.0; - for ( j = 1; j < (ncols +1); j++ ) { - z += x[j] * true_obj[j]; - } - return z + true_obj[0]; -} - -/**********************************************************************/ -static void array_copy(int begin, int end, double *source, - double *destination) -/**********************************************************************/ -{ - int i; - for (i = begin; i < end; i++) { - destination[i] = source[i]; - } -} -/**********************************************************************/ -static int do_refine(struct csa *csa, glp_prob *lp_ref, int ncols, - int *ckind, double *xref, int *tlim, int tref_lim, - int verbose) -/**********************************************************************/ -{ - /* - Refinement is applied when the variables of the problem are not - all binary. Binary variables are fixed to their value and - remaining ones are optimized. If there are only continuos - variables (in addition to those binary) the problem becomes just - an LP. Otherwise, it remains a MIP but of smaller size. - */ - - int j, tout; - double refineStart = second(); - double val, tela, tlimit; - - if ( glp_get_num_cols(lp_ref) != ncols ) { - if (verbose) { - xprintf("Error in Proxy refinement: "); - xprintf("wrong number of columns (%d vs %d).\n", - ncols, glp_get_num_cols(lp_ref)); - } - return 1; - } - - val = -1.0; - - /* fixing all binary variables to their current value in xref */ - for ( j = 1; j < (ncols + 1); j++ ) { - if ( ckind[j] == GLP_BV ) { - val = 0.0; - if ( xref[j] > 0.5 ) val = 1.0; - glp_set_col_bnds(lp_ref, j, GLP_FX, val, val); - } - } - - /* re-optimizing (refining) if some bound has been changed */ - if ( val > -1.0 ) { - glp_iocp parm_ref; - glp_smcp parm_ref_lp; - int err, status; - - glp_init_iocp(&parm_ref); - parm_ref.presolve = GLP_ON; - glp_init_smcp(&parm_ref_lp); - /* - If there are no general integer variable the problem becomes - an LP (after fixing the binary variables) and can be solved - quickly. Otherwise the problem is still a MIP problem and a - timelimit has to be set. - */ - parm_ref.tm_lim = tref_lim; - if (parm_ref.tm_lim > *tlim) { - parm_ref.tm_lim = *tlim; - } - parm_ref_lp.tm_lim = parm_ref.tm_lim; -#ifdef PROXY_DEBUG - xprintf("***** REFINING *****\n"); -#endif - tout = glp_term_out(GLP_OFF); - if (csa->i_vars_exist == TRUE) { - err = glp_intopt(lp_ref, &parm_ref); - } - else { - err = glp_simplex(lp_ref, &parm_ref_lp); - } - glp_term_out(tout); - - if (csa->i_vars_exist == TRUE) { - status = glp_mip_status(lp_ref); - } - else { - status = glp_get_status(lp_ref); - } - -#if 1 /* 29/II-2016 by mao as reported by Chris */ - switch (status) - { case GLP_OPT: - case GLP_FEAS: - break; - default: - status = GLP_UNDEF; - break; - } -#endif - -#ifdef PROXY_DEBUG - xprintf("STATUS REFINING = %d\n",status); -#endif - if (status == GLP_UNDEF) { - if (err == GLP_ETMLIM) { -#ifdef PROXY_DEBUG - xprintf("Time limit exceeded on Proxy refining.\n"); -#endif - return 1; - } - } - for( j = 1 ; j < (ncols + 1); j++ ){ - if (ckind[j] != GLP_BV) { - if (csa->i_vars_exist == TRUE) { - xref[j] = glp_mip_col_val(lp_ref, j); - } - else{ - xref[j] = glp_get_col_prim(lp_ref, j); - } - } - } - } - tela = second() - refineStart; -#ifdef PROXY_DEBUG - xprintf("REFINE TELA = %3.1lf\n",tela*1000); -#endif - return 0; -} -/**********************************************************************/ -static void deallocate(struct csa *csa, int refine) -/**********************************************************************/ -{ - /* Deallocating routine */ - - if (refine) { - glp_delete_prob(csa->lp_ref); - } - - tfree(csa->ckind); - tfree(csa->clb); - tfree(csa->cub); - tfree(csa->true_obj); - -} - -/* eof */ diff --git a/code/3rd_glpk/proxy/proxy.h b/code/3rd_glpk/proxy/proxy.h deleted file mode 100644 index a91e36f2..00000000 --- a/code/3rd_glpk/proxy/proxy.h +++ /dev/null @@ -1,36 +0,0 @@ -/* proxy.h (proximity search heuristic algorithm) */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Author: Giorgio Sartor <0gioker0@gmail.com>. -* -* Copyright (C) 2013 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef PROXY_H -#define PROXY_H - -#define proxy _glp_proxy -int proxy(glp_prob *lp, double *zstar, double *xstar, - const double initsol[], double rel_impr, int tlim, - int verbose); - -#endif - -/* eof */ diff --git a/code/3rd_glpk/proxy/proxy1.c b/code/3rd_glpk/proxy/proxy1.c deleted file mode 100644 index 5f9850d4..00000000 --- a/code/3rd_glpk/proxy/proxy1.c +++ /dev/null @@ -1,88 +0,0 @@ -/* proxy1.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2013, 2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "ios.h" -#include "proxy.h" - -void ios_proxy_heur(glp_tree *T) -{ glp_prob *prob; - int j, status; - double *xstar, zstar; - /* this heuristic is applied only once on the root level */ - if (!(T->curr->level == 0 && T->curr->solved == 1)) - goto done; - prob = glp_create_prob(); - glp_copy_prob(prob, T->mip, 0); - xstar = xcalloc(1+prob->n, sizeof(double)); - for (j = 1; j <= prob->n; j++) - xstar[j] = 0.0; - if (T->mip->mip_stat != GLP_FEAS) - status = proxy(prob, &zstar, xstar, NULL, 0.0, - T->parm->ps_tm_lim, 1); - else - { double *xinit = xcalloc(1+prob->n, sizeof(double)); - for (j = 1; j <= prob->n; j++) - xinit[j] = T->mip->col[j]->mipx; - status = proxy(prob, &zstar, xstar, xinit, 0.0, - T->parm->ps_tm_lim, 1); - xfree(xinit); - } - if (status == 0) -#if 0 /* 17/III-2016 */ - glp_ios_heur_sol(T, xstar); -#else - { /* sometimes the proxy heuristic reports a wrong solution, so - * make sure that the solution is really integer feasible */ - int i, feas1, feas2, ae_ind, re_ind; - double ae_max, re_max; - glp_copy_prob(prob, T->mip, 0); - for (j = 1; j <= prob->n; j++) - prob->col[j]->mipx = xstar[j]; - for (i = 1; i <= prob->m; i++) - { GLPROW *row; - GLPAIJ *aij; - row = prob->row[i]; - row->mipx = 0.0; - for (aij = row->ptr; aij != NULL; aij = aij->r_next) - row->mipx += aij->val * aij->col->mipx; - } - glp_check_kkt(prob, GLP_MIP, GLP_KKT_PE, &ae_max, &ae_ind, - &re_max, &re_ind); - feas1 = (re_max <= 1e-6); - glp_check_kkt(prob, GLP_MIP, GLP_KKT_PB, &ae_max, &ae_ind, - &re_max, &re_ind); - feas2 = (re_max <= 1e-6); - if (feas1 && feas2) - glp_ios_heur_sol(T, xstar); - else - xprintf("WARNING: PROXY HEURISTIC REPORTED WRONG SOLUTION; " - "SOLUTION REJECTED\n"); - } -#endif - xfree(xstar); - glp_delete_prob(prob); -done: return; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/simplex.h b/code/3rd_glpk/simplex/simplex.h deleted file mode 100644 index 9a5acdb2..00000000 --- a/code/3rd_glpk/simplex/simplex.h +++ /dev/null @@ -1,39 +0,0 @@ -/* simplex.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SIMPLEX_H -#define SIMPLEX_H - -#include "prob.h" - -#define spx_primal _glp_spx_primal -int spx_primal(glp_prob *P, const glp_smcp *parm); -/* driver to the primal simplex method */ - -#define spy_dual _glp_spy_dual -int spy_dual(glp_prob *P, const glp_smcp *parm); -/* driver to the dual simplex method */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxat.c b/code/3rd_glpk/simplex/spxat.c deleted file mode 100644 index 3570a18c..00000000 --- a/code/3rd_glpk/simplex/spxat.c +++ /dev/null @@ -1,265 +0,0 @@ -/* spxat.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spxat.h" - -/*********************************************************************** -* spx_alloc_at - allocate constraint matrix in sparse row-wise format -* -* This routine allocates the memory for arrays needed to represent the -* constraint matrix in sparse row-wise format. */ - -void spx_alloc_at(SPXLP *lp, SPXAT *at) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - at->ptr = talloc(1+m+1, int); - at->ind = talloc(1+nnz, int); - at->val = talloc(1+nnz, double); - at->work = talloc(1+n, double); - return; -} - -/*********************************************************************** -* spx_build_at - build constraint matrix in sparse row-wise format -* -* This routine builds sparse row-wise representation of the constraint -* matrix A using its sparse column-wise representation stored in the -* lp object, and stores the result in the at object. */ - -void spx_build_at(SPXLP *lp, SPXAT *at) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int *AT_ptr = at->ptr; - int *AT_ind = at->ind; - double *AT_val = at->val; - int i, k, ptr, end, pos; - /* calculate AT_ptr[i] = number of non-zeros in i-th row */ - memset(&AT_ptr[1], 0, m * sizeof(int)); - for (k = 1; k <= n; k++) - { ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - AT_ptr[A_ind[ptr]]++; - } - /* set AT_ptr[i] to position after last element in i-th row */ - AT_ptr[1]++; - for (i = 2; i <= m; i++) - AT_ptr[i] += AT_ptr[i-1]; - xassert(AT_ptr[m] == nnz+1); - AT_ptr[m+1] = nnz+1; - /* build row-wise representation and re-arrange AT_ptr[i] */ - for (k = n; k >= 1; k--) - { /* copy elements from k-th column to corresponding rows */ - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - { pos = --AT_ptr[A_ind[ptr]]; - AT_ind[pos] = k; - AT_val[pos] = A_val[ptr]; - } - } - xassert(AT_ptr[1] == 1); - return; -} - -/*********************************************************************** -* spx_at_prod - compute product y := y + s * A'* x -* -* This routine computes the product: -* -* y := y + s * A'* x, -* -* where A' is a matrix transposed to the mxn-matrix A of constraint -* coefficients, x is a m-vector, s is a scalar, y is a n-vector. -* -* The routine uses the row-wise representation of the matrix A and -* computes the product as a linear combination: -* -* y := y + s * (A'[1] * x[1] + ... + A'[m] * x[m]), -* -* where A'[i] is i-th row of A, 1 <= i <= m. */ - -void spx_at_prod(SPXLP *lp, SPXAT *at, double y[/*1+n*/], double s, - const double x[/*1+m*/]) -{ int m = lp->m; - int *AT_ptr = at->ptr; - int *AT_ind = at->ind; - double *AT_val = at->val; - int i, ptr, end; - double t; - for (i = 1; i <= m; i++) - { if (x[i] != 0.0) - { /* y := y + s * (i-th row of A) * x[i] */ - t = s * x[i]; - ptr = AT_ptr[i]; - end = AT_ptr[i+1]; - for (; ptr < end; ptr++) - y[AT_ind[ptr]] += AT_val[ptr] * t; - } - } - return; -} - -/*********************************************************************** -* spx_nt_prod1 - compute product y := y + s * N'* x -* -* This routine computes the product: -* -* y := y + s * N'* x, -* -* where N' is a matrix transposed to the mx(n-m)-matrix N composed -* from non-basic columns of the constraint matrix A, x is a m-vector, -* s is a scalar, y is (n-m)-vector. -* -* If the flag ign is non-zero, the routine ignores the input content -* of the array y assuming that y = 0. */ - -void spx_nt_prod1(SPXLP *lp, SPXAT *at, double y[/*1+n-m*/], int ign, - double s, const double x[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - double *work = at->work; - int j, k; - for (k = 1; k <= n; k++) - work[k] = 0.0; - if (!ign) - { for (j = 1; j <= n-m; j++) - work[head[m+j]] = y[j]; - } - spx_at_prod(lp, at, work, s, x); - for (j = 1; j <= n-m; j++) - y[j] = work[head[m+j]]; - return; -} - -/*********************************************************************** -* spx_eval_trow1 - compute i-th row of simplex table -* -* This routine computes i-th row of the current simplex table -* T = (T[i,j]) = - inv(B) * N, 1 <= i <= m, using representation of -* the constraint matrix A in row-wise format. -* -* The vector rho = (rho[j]), which is i-th row of the basis inverse -* inv(B), should be previously computed with the routine spx_eval_rho. -* It is assumed that elements of this vector are stored in the array -* locations rho[1], ..., rho[m]. -* -* There exist two ways to compute the simplex table row. -* -* 1. T[i,j], j = 1,...,n-m, is computed as inner product: -* -* m -* T[i,j] = - sum a[i,k] * rho[i], -* i=1 -* -* where N[j] = A[k] is a column of the constraint matrix corresponding -* to non-basic variable xN[j]. The estimated number of operations in -* this case is: -* -* n1 = (n - m) * (nnz(A) / n), -* -* (n - m) is the number of columns of N, nnz(A) / n is the average -* number of non-zeros in one column of A and, therefore, of N. -* -* 2. The simplex table row is computed as part of a linear combination -* of rows of A with coefficients rho[i] != 0. The estimated number -* of operations in this case is: -* -* n2 = nnz(rho) * (nnz(A) / m), -* -* where nnz(rho) is the number of non-zeros in the vector rho, -* nnz(A) / m is the average number of non-zeros in one row of A. -* -* If n1 < n2, the routine computes the simples table row using the -* first way (like the routine spx_eval_trow). Otherwise, the routine -* uses the second way calling the routine spx_nt_prod1. -* -* On exit components of the simplex table row are stored in the array -* locations trow[1], ... trow[n-m]. */ - -void spx_eval_trow1(SPXLP *lp, SPXAT *at, const double rho[/*1+m*/], - double trow[/*1+n-m*/]) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - int i, j, nnz_rho; - double cnt1, cnt2; - /* determine nnz(rho) */ - nnz_rho = 0; - for (i = 1; i <= m; i++) - { if (rho[i] != 0.0) - nnz_rho++; - } - /* estimate the number of operations for both ways */ - cnt1 = (double)(n - m) * ((double)nnz / (double)n); - cnt2 = (double)nnz_rho * ((double)nnz / (double)m); - /* compute i-th row of simplex table */ - if (cnt1 < cnt2) - { /* as inner products */ - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int *head = lp->head; - int k, ptr, end; - double tij; - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* compute t[i,j] = - N'[j] * pi */ - tij = 0.0; - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - tij -= A_val[ptr] * rho[A_ind[ptr]]; - trow[j] = tij; - } - } - else - { /* as linear combination */ - spx_nt_prod1(lp, at, trow, 1, -1.0, rho); - } - return; -} - -/*********************************************************************** -* spx_free_at - deallocate constraint matrix in sparse row-wise format -* -* This routine deallocates the memory used for arrays of the program -* object at. */ - -void spx_free_at(SPXLP *lp, SPXAT *at) -{ xassert(lp == lp); - tfree(at->ptr); - tfree(at->ind); - tfree(at->val); - tfree(at->work); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxat.h b/code/3rd_glpk/simplex/spxat.h deleted file mode 100644 index 98d5b003..00000000 --- a/code/3rd_glpk/simplex/spxat.h +++ /dev/null @@ -1,80 +0,0 @@ -/* spxat.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPXAT_H -#define SPXAT_H - -#include "spxlp.h" - -typedef struct SPXAT SPXAT; - -struct SPXAT -{ /* mxn-matrix A of constraint coefficients in sparse row-wise - * format */ - int *ptr; /* int ptr[1+m+1]; */ - /* ptr[0] is not used; - * ptr[i], 1 <= i <= m, is starting position of i-th row in - * arrays ind and val; note that ptr[1] is always 1; - * ptr[m+1] indicates the position after the last element in - * arrays ind and val, i.e. ptr[m+1] = nnz+1, where nnz is the - * number of non-zero elements in matrix A; - * the length of i-th row (the number of non-zero elements in - * that row) can be calculated as ptr[i+1] - ptr[i] */ - int *ind; /* int ind[1+nnz]; */ - /* column indices */ - double *val; /* double val[1+nnz]; */ - /* non-zero element values */ - double *work; /* double work[1+n]; */ - /* working array */ -}; - -#define spx_alloc_at _glp_spx_alloc_at -void spx_alloc_at(SPXLP *lp, SPXAT *at); -/* allocate constraint matrix in sparse row-wise format */ - -#define spx_build_at _glp_spx_build_at -void spx_build_at(SPXLP *lp, SPXAT *at); -/* build constraint matrix in sparse row-wise format */ - -#define spx_at_prod _glp_spx_at_prod -void spx_at_prod(SPXLP *lp, SPXAT *at, double y[/*1+n*/], double s, - const double x[/*1+m*/]); -/* compute product y := y + s * A'* x */ - -#define spx_nt_prod1 _glp_spx_nt_prod1 -void spx_nt_prod1(SPXLP *lp, SPXAT *at, double y[/*1+n-m*/], int ign, - double s, const double x[/*1+m*/]); -/* compute product y := y + s * N'* x */ - -#define spx_eval_trow1 _glp_spx_eval_trow1 -void spx_eval_trow1(SPXLP *lp, SPXAT *at, const double rho[/*1+m*/], - double trow[/*1+n-m*/]); -/* compute i-th row of simplex table */ - -#define spx_free_at _glp_spx_free_at -void spx_free_at(SPXLP *lp, SPXAT *at); -/* deallocate constraint matrix in sparse row-wise format */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxchuzc.c b/code/3rd_glpk/simplex/spxchuzc.c deleted file mode 100644 index c60ccabc..00000000 --- a/code/3rd_glpk/simplex/spxchuzc.c +++ /dev/null @@ -1,381 +0,0 @@ -/* spxchuzc.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spxchuzc.h" - -/*********************************************************************** -* spx_chuzc_sel - select eligible non-basic variables -* -* This routine selects eligible non-basic variables xN[j], whose -* reduced costs d[j] have "wrong" sign, i.e. changing such xN[j] in -* feasible direction improves (decreases) the objective function. -* -* Reduced costs of non-basic variables should be placed in the array -* locations d[1], ..., d[n-m]. -* -* Non-basic variable xN[j] is considered eligible if: -* -* d[j] <= -eps[j] and xN[j] can increase -* -* d[j] >= +eps[j] and xN[j] can decrease -* -* for -* -* eps[j] = tol + tol1 * |cN[j]|, -* -* where cN[j] is the objective coefficient at xN[j], tol and tol1 are -* specified tolerances. -* -* On exit the routine stores indices j of eligible non-basic variables -* xN[j] to the array locations list[1], ..., list[num] and returns the -* number of such variables 0 <= num <= n-m. (If the parameter list is -* specified as NULL, no indices are stored.) */ - -int spx_chuzc_sel(SPXLP *lp, const double d[/*1+n-m*/], double tol, - double tol1, int list[/*1+n-m*/]) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, k, num; - double ck, eps; - num = 0; - /* walk thru list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == u[k]) - { /* xN[j] is fixed variable; skip it */ - continue; - } - /* determine absolute tolerance eps[j] */ - ck = c[k]; - eps = tol + tol1 * (ck >= 0.0 ? +ck : -ck); - /* check if xN[j] is eligible */ - if (d[j] <= -eps) - { /* xN[j] should be able to increase */ - if (flag[j]) - { /* but its upper bound is active */ - continue; - } - } - else if (d[j] >= +eps) - { /* xN[j] should be able to decrease */ - if (!flag[j] && l[k] != -DBL_MAX) - { /* but its lower bound is active */ - continue; - } - } - else /* -eps < d[j] < +eps */ - { /* xN[j] does not affect the objective function within the - * specified tolerance */ - continue; - } - /* xN[j] is eligible non-basic variable */ - num++; - if (list != NULL) - list[num] = j; - } - return num; -} - -/*********************************************************************** -* spx_chuzc_std - choose non-basic variable (Dantzig's rule) -* -* This routine chooses most eligible non-basic variable xN[q] -* according to Dantzig's ("standard") rule: -* -* d[q] = max |d[j]|, -* j in J -* -* where J <= {1, ..., n-m} is the set of indices of eligible non-basic -* variables, d[j] is the reduced cost of non-basic variable xN[j] in -* the current basis. -* -* Reduced costs of non-basic variables should be placed in the array -* locations d[1], ..., d[n-m]. -* -* Indices of eligible non-basic variables j in J should be placed in -* the array locations list[1], ..., list[num], where num = |J| > 0 is -* the total number of such variables. -* -* On exit the routine returns q, the index of the non-basic variable -* xN[q] chosen. */ - -int spx_chuzc_std(SPXLP *lp, const double d[/*1+n-m*/], int num, - const int list[]) -{ int m = lp->m; - int n = lp->n; - int j, q, t; - double abs_dj, abs_dq; - xassert(0 < num && num <= n-m); - q = 0, abs_dq = -1.0; - for (t = 1; t <= num; t++) - { j = list[t]; - abs_dj = (d[j] >= 0.0 ? +d[j] : -d[j]); - if (abs_dq < abs_dj) - q = j, abs_dq = abs_dj; - } - xassert(q != 0); - return q; -} - -/*********************************************************************** -* spx_alloc_se - allocate pricing data block -* -* This routine allocates the memory for arrays used in the pricing -* data block. */ - -void spx_alloc_se(SPXLP *lp, SPXSE *se) -{ int m = lp->m; - int n = lp->n; - se->valid = 0; - se->refsp = talloc(1+n, char); - se->gamma = talloc(1+n-m, double); - se->work = talloc(1+m, double); - return; -} - -/*********************************************************************** -* spx_reset_refsp - reset reference space -* -* This routine resets (re-initializes) the reference space composing -* it from variables which are non-basic in the current basis, and sets -* all weights gamma[j] to 1. */ - -void spx_reset_refsp(SPXLP *lp, SPXSE *se) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *gamma = se->gamma; - int j, k; - se->valid = 1; - memset(&refsp[1], 0, n * sizeof(char)); - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - refsp[k] = 1; - gamma[j] = 1.0; - } - return; -} - -/*********************************************************************** -* spx_eval_gamma_j - compute projected steepest edge weight directly -* -* This routine computes projected steepest edge weight gamma[j], -* 1 <= j <= n-m, for the current basis directly with the formula: -* -* m -* gamma[j] = delta[j] + sum eta[i] * T[i,j]**2, -* i=1 -* -* where T[i,j] is element of the current simplex table, and -* -* ( 1, if xB[i] is in the reference space -* eta[i] = { -* ( 0, otherwise -* -* ( 1, if xN[j] is in the reference space -* delta[j] = { -* ( 0, otherwise -* -* NOTE: For testing/debugging only. */ - -double spx_eval_gamma_j(SPXLP *lp, SPXSE *se, int j) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *tcol = se->work; - int i, k; - double gamma_j; - xassert(se->valid); - xassert(1 <= j && j <= n-m); - k = head[m+j]; /* x[k] = xN[j] */ - gamma_j = (refsp[k] ? 1.0 : 0.0); - spx_eval_tcol(lp, j, tcol); - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - if (refsp[k]) - gamma_j += tcol[i] * tcol[i]; - } - return gamma_j; -} - -/*********************************************************************** -* spx_chuzc_pse - choose non-basic variable (projected steepest edge) -* -* This routine chooses most eligible non-basic variable xN[q] -* according to the projected steepest edge method: -* -* d[q]**2 d[j]**2 -* -------- = max -------- , -* gamma[q] j in J gamma[j] -* -* where J <= {1, ..., n-m} is the set of indices of eligible non-basic -* variable, d[j] is the reduced cost of non-basic variable xN[j] in -* the current basis, gamma[j] is the projected steepest edge weight. -* -* Reduced costs of non-basic variables should be placed in the array -* locations d[1], ..., d[n-m]. -* -* Indices of eligible non-basic variables j in J should be placed in -* the array locations list[1], ..., list[num], where num = |J| > 0 is -* the total number of such variables. -* -* On exit the routine returns q, the index of the non-basic variable -* xN[q] chosen. */ - -int spx_chuzc_pse(SPXLP *lp, SPXSE *se, const double d[/*1+n-m*/], - int num, const int list[]) -{ int m = lp->m; - int n = lp->n; - double *gamma = se->gamma; - int j, q, t; - double best, temp; - xassert(se->valid); - xassert(0 < num && num <= n-m); - q = 0, best = -1.0; - for (t = 1; t <= num; t++) - { j = list[t]; - /* FIXME */ - if (gamma[j] < DBL_EPSILON) - temp = 0.0; - else - temp = (d[j] * d[j]) / gamma[j]; - if (best < temp) - q = j, best = temp; - } - xassert(q != 0); - return q; -} - -/*********************************************************************** -* spx_update_gamma - update projected steepest edge weights exactly -* -* This routine updates the vector gamma = (gamma[j]) of projected -* steepest edge weights exactly, for the adjacent basis. -* -* On entry to the routine the content of the se object should be valid -* and should correspond to the current basis. -* -* The parameter 1 <= p <= m specifies basic variable xB[p] which -* becomes non-basic variable xN[q] in the adjacent basis. -* -* The parameter 1 <= q <= n-m specified non-basic variable xN[q] which -* becomes basic variable xB[p] in the adjacent basis. -* -* It is assumed that the array trow contains elements of p-th (pivot) -* row T'[p] of the simplex table in locations trow[1], ..., trow[n-m]. -* It is also assumed that the array tcol contains elements of q-th -* (pivot) column T[q] of the simple table in locations tcol[1], ..., -* tcol[m]. (These row and column should be computed for the current -* basis.) -* -* For details about the formulae used see the program documentation. -* -* The routine also computes the relative error: -* -* e = |gamma[q] - gamma'[q]| / (1 + |gamma[q]|), -* -* where gamma'[q] is the weight for xN[q] on entry to the routine, -* and returns e on exit. (If e happens to be large enough, the calling -* program may reset the reference space, since other weights also may -* be inaccurate.) */ - -double spx_update_gamma(SPXLP *lp, SPXSE *se, int p, int q, - const double trow[/*1+n-m*/], const double tcol[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *gamma = se->gamma; - double *u = se->work; - int i, j, k, ptr, end; - double gamma_q, delta_q, e, r, s, t1, t2; - xassert(se->valid); - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - /* compute gamma[q] in current basis more accurately; also - * compute auxiliary vector u */ - k = head[m+q]; /* x[k] = xN[q] */ - gamma_q = delta_q = (refsp[k] ? 1.0 : 0.0); - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - if (refsp[k]) - { gamma_q += tcol[i] * tcol[i]; - u[i] = tcol[i]; - } - else - u[i] = 0.0; - } - bfd_btran(lp->bfd, u); - /* compute relative error in gamma[q] */ - e = fabs(gamma_q - gamma[q]) / (1.0 + gamma_q); - /* compute new gamma[q] */ - gamma[q] = gamma_q / (tcol[p] * tcol[p]); - /* compute new gamma[j] for all j != q */ - for (j = 1; j <= n-m; j++) - { if (j == q) - continue; - if (-1e-9 < trow[j] && trow[j] < +1e-9) - { /* T[p,j] is close to zero; gamma[j] is not changed */ - continue; - } - /* compute r[j] = T[p,j] / T[p,q] */ - r = trow[j] / tcol[p]; - /* compute inner product s[j] = N'[j] * u, where N[j] = A[k] - * is constraint matrix column corresponding to xN[j] */ - s = 0.0; - k = head[m+j]; /* x[k] = xN[j] */ - ptr = lp->A_ptr[k]; - end = lp->A_ptr[k+1]; - for (; ptr < end; ptr++) - s += lp->A_val[ptr] * u[lp->A_ind[ptr]]; - /* compute new gamma[j] */ - t1 = gamma[j] + r * (r * gamma_q + s + s); - t2 = (refsp[k] ? 1.0 : 0.0) + delta_q * r * r; - gamma[j] = (t1 >= t2 ? t1 : t2); - } - return e; -} - -/*********************************************************************** -* spx_free_se - deallocate pricing data block -* -* This routine deallocates the memory used for arrays in the pricing -* data block. */ - -void spx_free_se(SPXLP *lp, SPXSE *se) -{ xassert(lp == lp); - tfree(se->refsp); - tfree(se->gamma); - tfree(se->work); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxchuzc.h b/code/3rd_glpk/simplex/spxchuzc.h deleted file mode 100644 index c09cca9a..00000000 --- a/code/3rd_glpk/simplex/spxchuzc.h +++ /dev/null @@ -1,85 +0,0 @@ -/* spxchuzc.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPXCHUZC_H -#define SPXCHUZC_H - -#include "spxlp.h" - -#define spx_chuzc_sel _glp_spx_chuzc_sel -int spx_chuzc_sel(SPXLP *lp, const double d[/*1+n-m*/], double tol, - double tol1, int list[/*1+n-m*/]); -/* select eligible non-basic variables */ - -#define spx_chuzc_std _glp_spx_chuzc_std -int spx_chuzc_std(SPXLP *lp, const double d[/*1+n-m*/], int num, - const int list[]); -/* choose non-basic variable (Dantzig's rule) */ - -typedef struct SPXSE SPXSE; - -struct SPXSE -{ /* projected steepest edge and Devex pricing data block */ - int valid; - /* content validity flag */ - char *refsp; /* char refsp[1+n]; */ - /* refsp[0] is not used; - * refsp[k], 1 <= k <= n, is the flag meaning that variable x[k] - * is in the reference space */ - double *gamma; /* double gamma[1+n-m]; */ - /* gamma[0] is not used; - * gamma[j], 1 <= j <= n-m, is the weight for reduced cost d[j] - * of non-basic variable xN[j] in the current basis */ - double *work; /* double work[1+m]; */ - /* working array */ -}; - -#define spx_alloc_se _glp_spx_alloc_se -void spx_alloc_se(SPXLP *lp, SPXSE *se); -/* allocate pricing data block */ - -#define spx_reset_refsp _glp_spx_reset_refsp -void spx_reset_refsp(SPXLP *lp, SPXSE *se); -/* reset reference space */ - -#define spx_eval_gamma_j _glp_spx_eval_gamma_j -double spx_eval_gamma_j(SPXLP *lp, SPXSE *se, int j); -/* compute projeted steepest edge weight directly */ - -#define spx_chuzc_pse _glp_spx_chuzc_pse -int spx_chuzc_pse(SPXLP *lp, SPXSE *se, const double d[/*1+n-m*/], - int num, const int list[]); -/* choose non-basic variable (projected steepest edge) */ - -#define spx_update_gamma _glp_spx_update_gamma -double spx_update_gamma(SPXLP *lp, SPXSE *se, int p, int q, - const double trow[/*1+n-m*/], const double tcol[/*1+m*/]); -/* update projected steepest edge weights exactly */ - -#define spx_free_se _glp_spx_free_se -void spx_free_se(SPXLP *lp, SPXSE *se); -/* deallocate pricing data block */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxchuzr.c b/code/3rd_glpk/simplex/spxchuzr.c deleted file mode 100644 index 8bef77ba..00000000 --- a/code/3rd_glpk/simplex/spxchuzr.c +++ /dev/null @@ -1,594 +0,0 @@ -/* spxchuzr.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spxchuzr.h" - -/*********************************************************************** -* spx_chuzr_std - choose basic variable (textbook ratio test) -* -* This routine implements an improved textbook ratio test to choose -* basic variable xB[p]. -* -* The parameter phase specifies the search phase: -* -* 1 - searching for feasible basic solution. In this case the routine -* uses artificial bounds of basic variables that correspond to -* breakpoints of the penalty function: -* -* ( lB[i], if cB[i] = 0 -* ( -* lB'[i] = { uB[i], if cB[i] > 0 -* ( -* ( -inf, if cB[i] < 0 -* -* ( uB[i], if cB[i] = 0 -* ( -* uB'[i] = { +inf, if cB[i] > 0 -* ( -* ( lB[i], if cB[i] < 0 -* -* where lB[i] and uB[i] are original bounds of variable xB[i], -* cB[i] is the penalty (objective) coefficient of that variable. -* -* 2 - searching for optimal basic solution. In this case the routine -* uses original bounds of basic variables. -* -* Current values of basic variables should be placed in the array -* locations beta[1], ..., beta[m]. -* -* The parameter 1 <= q <= n-m specifies the index of non-basic -* variable xN[q] chosen. -* -* The parameter s specifies the direction in which xN[q] changes: -* s = +1.0 means xN[q] increases, and s = -1.0 means xN[q] decreases. -* (Thus, the corresponding ray parameter is theta = s (xN[q] - f[q]), -* where f[q] is the active bound of xN[q] in the current basis.) -* -* Elements of q-th simplex table column T[q] = (t[i,q]) corresponding -* to non-basic variable xN[q] should be placed in the array locations -* tcol[1], ..., tcol[m]. -* -* The parameter tol_piv specifies a tolerance for elements of the -* simplex table column T[q]. If |t[i,q]| < tol_piv, basic variable -* xB[i] is skipped, i.e. it is assumed that it does not depend on the -* ray parameter theta. -* -* The parameters tol and tol1 specify tolerances used to increase the -* choice freedom by simulating an artificial degeneracy as follows. -* If beta[i] <= lB[i] + delta[i], where delta[i] = tol + tol1 |lB[i]|, -* it is assumed that beta[i] is exactly the same as lB[i]. Similarly, -* if beta[i] >= uB[i] - delta[i], where delta[i] = tol + tol1 |uB[i]|, -* it is assumed that beta[i] is exactly the same as uB[i]. -* -* The routine determines the index 1 <= p <= m of basic variable xB[p] -* that reaches its (lower or upper) bound first on increasing the ray -* parameter theta, stores the bound flag (0 - lower bound or fixed -* value, 1 - upper bound) to the location pointed to by the pointer -* p_flag, and returns the index p. If non-basic variable xN[q] is -* double-bounded and reaches its opposite bound first, the routine -* returns (-1). And if the ray parameter may increase unlimitedly, the -* routine returns zero. -* -* Should note that the bound flag stored to the location pointed to by -* p_flag corresponds to the original (not artficial) bound of variable -* xB[p] and defines the active bound flag lp->flag[q] to be set in the -* adjacent basis for that basic variable. */ - -int spx_chuzr_std(SPXLP *lp, int phase, const double beta[/*1+m*/], - int q, double s, const double tcol[/*1+m*/], int *p_flag, - double tol_piv, double tol, double tol1) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int i, i_flag, k, p; - double alfa, biga, delta, lk, uk, teta, teta_min; - xassert(phase == 1 || phase == 2); - xassert(1 <= q && q <= n-m); - xassert(s == +1.0 || s == -1.0); - /* determine initial teta_min */ - k = head[m+q]; /* x[k] = xN[q] */ - if (l[k] == -DBL_MAX || u[k] == +DBL_MAX) - { /* xN[q] has no opposite bound */ - p = 0, *p_flag = 0, teta_min = DBL_MAX, biga = 0.0; - } - else - { /* xN[q] have both lower and upper bounds */ - p = -1, *p_flag = 0, teta_min = fabs(l[k] - u[k]), biga = 1.0; - } - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - /* determine alfa such that delta xB[i] = alfa * teta */ - alfa = s * tcol[i]; - if (alfa <= -tol_piv) - { /* xB[i] decreases */ - /* determine actual lower bound of xB[i] */ - if (phase == 1 && c[k] < 0.0) - { /* xB[i] has no actual lower bound */ - continue; - } - else if (phase == 1 && c[k] > 0.0) - { /* actual lower bound of xB[i] is its upper bound */ - lk = u[k]; - xassert(lk != +DBL_MAX); - i_flag = 1; - } - else - { /* actual lower bound of xB[i] is its original bound */ - lk = l[k]; - if (lk == -DBL_MAX) - continue; - i_flag = 0; - } - /* determine teta on which xB[i] reaches its lower bound */ - delta = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] <= lk + delta) - teta = 0.0; - else - teta = (lk - beta[i]) / alfa; - } - else if (alfa >= +tol_piv) - { /* xB[i] increases */ - /* determine actual upper bound of xB[i] */ - if (phase == 1 && c[k] < 0.0) - { /* actual upper bound of xB[i] is its lower bound */ - uk = l[k]; - xassert(uk != -DBL_MAX); - i_flag = 0; - } - else if (phase == 1 && c[k] > 0.0) - { /* xB[i] has no actual upper bound */ - continue; - } - else - { /* actual upper bound of xB[i] is its original bound */ - uk = u[k]; - if (uk == +DBL_MAX) - continue; - i_flag = 1; - } - /* determine teta on which xB[i] reaches its upper bound */ - delta = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] >= uk - delta) - teta = 0.0; - else - teta = (uk - beta[i]) / alfa; - } - else - { /* xB[i] does not depend on teta */ - continue; - } - /* choose basic variable xB[p] for which teta is minimal */ - xassert(teta >= 0.0); - alfa = (alfa >= 0.0 ? +alfa : -alfa); - if (teta_min > teta || (teta_min == teta && biga < alfa)) - p = i, *p_flag = i_flag, teta_min = teta, biga = alfa; - } - /* if xB[p] is fixed variable, adjust its bound flag */ - if (p > 0) - { k = head[p]; - if (l[k] == u[k]) - *p_flag = 0; - } - return p; -} - -/*********************************************************************** -* spx_chuzr_harris - choose basic variable (Harris' ratio test) -* -* This routine implements Harris' ratio test to choose basic variable -* xB[p]. -* -* All the parameters, except tol and tol1, as well as the returned -* value have the same meaning as for the routine spx_chuzr_std (see -* above). -* -* The parameters tol and tol1 specify tolerances on bound violations -* for basic variables. For the lower bound of basic variable xB[i] the -* tolerance is delta[i] = tol + tol1 |lB[i]|, and for the upper bound -* the tolerance is delta[i] = tol + tol1 |uB[i]|. */ - -int spx_chuzr_harris(SPXLP *lp, int phase, const double beta[/*1+m*/], - int q, double s, const double tcol[/*1+m*/], int *p_flag, - double tol_piv, double tol, double tol1) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int i, i_flag, k, p; - double alfa, biga, delta, lk, uk, teta, teta_min; - xassert(phase == 1 || phase == 2); - xassert(1 <= q && q <= n-m); - xassert(s == +1.0 || s == -1.0); - /*--------------------------------------------------------------*/ - /* first pass: determine teta_min for relaxed bounds */ - /*--------------------------------------------------------------*/ - teta_min = DBL_MAX; - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - /* determine alfa such that delta xB[i] = alfa * teta */ - alfa = s * tcol[i]; - if (alfa <= -tol_piv) - { /* xB[i] decreases */ - /* determine actual lower bound of xB[i] */ - if (phase == 1 && c[k] < 0.0) - { /* xB[i] has no actual lower bound */ - continue; - } - else if (phase == 1 && c[k] > 0.0) - { /* actual lower bound of xB[i] is its upper bound */ - lk = u[k]; - xassert(lk != +DBL_MAX); - } - else - { /* actual lower bound of xB[i] is its original bound */ - lk = l[k]; - if (lk == -DBL_MAX) - continue; - } - /* determine teta on which xB[i] reaches its relaxed lower - * bound */ - delta = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] < lk) - teta = - delta / alfa; - else - teta = ((lk - delta) - beta[i]) / alfa; - } - else if (alfa >= +tol_piv) - { /* xB[i] increases */ - /* determine actual upper bound of xB[i] */ - if (phase == 1 && c[k] < 0.0) - { /* actual upper bound of xB[i] is its lower bound */ - uk = l[k]; - xassert(uk != -DBL_MAX); - } - else if (phase == 1 && c[k] > 0.0) - { /* xB[i] has no actual upper bound */ - continue; - } - else - { /* actual upper bound of xB[i] is its original bound */ - uk = u[k]; - if (uk == +DBL_MAX) - continue; - } - /* determine teta on which xB[i] reaches its relaxed upper - * bound */ - delta = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] > uk) - teta = + delta / alfa; - else - teta = ((uk + delta) - beta[i]) / alfa; - } - else - { /* xB[i] does not depend on teta */ - continue; - } - xassert(teta >= 0.0); - if (teta_min > teta) - teta_min = teta; - } - /*--------------------------------------------------------------*/ - /* second pass: choose basic variable xB[p] */ - /*--------------------------------------------------------------*/ - k = head[m+q]; /* x[k] = xN[q] */ - if (l[k] != -DBL_MAX && u[k] != +DBL_MAX) - { /* xN[q] has both lower and upper bounds */ - if (fabs(l[k] - u[k]) <= teta_min) - { /* and reaches its opposite bound */ - p = -1, *p_flag = 0; - goto done; - } - } - if (teta_min == DBL_MAX) - { /* teta may increase unlimitedly */ - p = 0, *p_flag = 0; - goto done; - } - /* nothing is chosen so far */ - p = 0, *p_flag = 0, biga = 0.0; - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - /* determine alfa such that delta xB[i] = alfa * teta */ - alfa = s * tcol[i]; - if (alfa <= -tol_piv) - { /* xB[i] decreases */ - /* determine actual lower bound of xB[i] */ - if (phase == 1 && c[k] < 0.0) - { /* xB[i] has no actual lower bound */ - continue; - } - else if (phase == 1 && c[k] > 0.0) - { /* actual lower bound of xB[i] is its upper bound */ - lk = u[k]; - xassert(lk != +DBL_MAX); - i_flag = 1; - } - else - { /* actual lower bound of xB[i] is its original bound */ - lk = l[k]; - if (lk == -DBL_MAX) - continue; - i_flag = 0; - } - /* determine teta on which xB[i] reaches its lower bound */ - teta = (lk - beta[i]) / alfa; - } - else if (alfa >= +tol_piv) - { /* xB[i] increases */ - /* determine actual upper bound of xB[i] */ - if (phase == 1 && c[k] < 0.0) - { /* actual upper bound of xB[i] is its lower bound */ - uk = l[k]; - xassert(uk != -DBL_MAX); - i_flag = 0; - } - else if (phase == 1 && c[k] > 0.0) - { /* xB[i] has no actual upper bound */ - continue; - } - else - { /* actual upper bound of xB[i] is its original bound */ - uk = u[k]; - if (uk == +DBL_MAX) - continue; - i_flag = 1; - } - /* determine teta on which xB[i] reaches its upper bound */ - teta = (uk - beta[i]) / alfa; - } - else - { /* xB[i] does not depend on teta */ - continue; - } - /* choose basic variable for which teta is not greater than - * teta_min determined for relaxed bounds and which has best - * (largest in magnitude) pivot */ - alfa = (alfa >= 0.0 ? +alfa : -alfa); - if (teta <= teta_min && biga < alfa) - p = i, *p_flag = i_flag, biga = alfa; - } - /* something must be chosen */ - xassert(1 <= p && p <= m); - /* if xB[p] is fixed variable, adjust its bound flag */ - k = head[p]; - if (l[k] == u[k]) - *p_flag = 0; -done: return p; -} - -#if 1 /* 22/VI-2017 */ -/*********************************************************************** -* spx_ls_eval_bp - determine penalty function break points -* -* This routine determines break points of the penalty function (which -* is the sum of primal infeasibilities). -* -* The parameters lp, beta, q, dq, tcol, and tol_piv have the same -* meaning as for the routine spx_chuzr_std (see above). -* -* The routine stores the break-points determined to the array elements -* bp[1], ..., bp[nbp] in *arbitrary* order, where 0 <= nbp <= 2*m+1 is -* the number of break-points returned by the routine on exit. */ - -int spx_ls_eval_bp(SPXLP *lp, const double beta[/*1+m*/], - int q, double dq, const double tcol[/*1+m*/], double tol_piv, - SPXBP bp[/*1+2*m+1*/]) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int i, k, nbp; - double s, alfa; - xassert(1 <= q && q <= n-m); - xassert(dq != 0.0); - s = (dq < 0.0 ? +1.0 : -1.0); - nbp = 0; - /* if chosen non-basic variable xN[q] is double-bounded, include - * it in the list, because it can cross its opposite bound */ - k = head[m+q]; /* x[k] = xN[q] */ - if (l[k] != -DBL_MAX && u[k] != +DBL_MAX) - { nbp++; - bp[nbp].i = 0; - xassert(l[k] < u[k]); /* xN[q] cannot be fixed */ - bp[nbp].teta = u[k] - l[k]; - bp[nbp].dc = s; - } - /* build the list of all basic variables xB[i] that can cross - * their bound(s) for the ray parameter 0 <= teta < teta_max */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - xassert(l[k] <= u[k]); - /* determine alfa such that (delta xB[i]) = alfa * teta */ - alfa = s * tcol[i]; - if (alfa >= +tol_piv) - { /* xB[i] increases on increasing teta */ - if (l[k] == u[k]) - { /* xB[i] is fixed at lB[i] = uB[i] */ - if (c[k] <= 0.0) - { /* increasing xB[i] can cross its fixed value lB[i], - * because currently xB[i] <= lB[i] */ - nbp++; - bp[nbp].i = +i; - bp[nbp].teta = (l[k] - beta[i]) / alfa; - /* if xB[i] > lB[i] then cB[i] = +1 */ - bp[nbp].dc = +1.0 - c[k]; - } - } - else - { if (l[k] != -DBL_MAX && c[k] < 0.0) - { /* increasing xB[i] can cross its lower bound lB[i], - * because currently xB[i] < lB[i] */ - nbp++; - bp[nbp].i = +i; - bp[nbp].teta = (l[k] - beta[i]) / alfa; - bp[nbp].dc = +1.0; - } - if (u[k] != +DBL_MAX && c[k] <= 0.0) - { /* increasing xB[i] can cross its upper bound uB[i], - * because currently xB[i] does not violate it */ - nbp++; - bp[nbp].i = -i; - bp[nbp].teta = (u[k] - beta[i]) / alfa; - bp[nbp].dc = +1.0; - } - } - } - else if (alfa <= -tol_piv) - { /* xB[i] decreases on increasing teta */ - if (l[k] == u[k]) - { /* xB[i] is fixed at lB[i] = uB[i] */ - if (c[k] >= 0.0) - { /* decreasing xB[i] can cross its fixed value lB[i], - * because currently xB[i] >= lB[i] */ - nbp++; - bp[nbp].i = +i; - bp[nbp].teta = (l[k] - beta[i]) / alfa; - /* if xB[i] < lB[i] then cB[i] = -1 */ - bp[nbp].dc = -1.0 - c[k]; - } - } - else - { if (l[k] != -DBL_MAX && c[k] >= 0.0) - { /* decreasing xB[i] can cross its lower bound lB[i], - * because currently xB[i] does not violate it */ - nbp++; - bp[nbp].i = +i; - bp[nbp].teta = (l[k] - beta[i]) / alfa; - bp[nbp].dc = -1.0; - } - if (u[k] != +DBL_MAX && c[k] > 0.0) - { /* decreasing xB[i] can cross its upper bound uB[i], - * because currently xB[i] > uB[i] */ - nbp++; - bp[nbp].i = -i; - bp[nbp].teta = (u[k] - beta[i]) / alfa; - bp[nbp].dc = -1.0; - } - } - } - else - { /* xB[i] does not depend on teta within a tolerance */ - continue; - } - /* teta < 0 may happen only due to round-off errors when the - * current value of xB[i] is *close* to its (lower or upper) - * bound; in this case we replace teta by exact zero */ - if (bp[nbp].teta < 0.0) - bp[nbp].teta = 0.0; - } - xassert(nbp <= 2*m+1); - return nbp; -} -#endif - -#if 1 /* 22/VI-2017 */ -/*********************************************************************** -* spx_ls_select_bp - select and process penalty function break points -* -* This routine selects a next portion of the penalty function break -* points and processes them. -* -* On entry to the routine it is assumed that break points bp[1], ..., -* bp[num] are already processed, and slope is the penalty function -* slope to the right of the last processed break point bp[num]. -* (Initially, when num = 0, slope should be specified as -fabs(d[q]), -* where d[q] is the reduced cost of chosen non-basic variable xN[q].) -* -* The routine selects break points among bp[num+1], ..., bp[nbp], for -* which teta <= teta_lim, and moves these break points to the array -* elements bp[num+1], ..., bp[num1], where num <= num1 <= 2*m+1 is the -* new number of processed break points returned by the routine on -* exit. Then the routine sorts the break points by ascending teta and -* computes the change of the penalty function relative to its value at -* teta = 0. -* -* On exit the routine also replaces the parameter slope with a new -* value that corresponds to the new last break-point bp[num1]. */ - -static int CDECL fcmp(const void *v1, const void *v2) -{ const SPXBP *p1 = v1, *p2 = v2; - if (p1->teta < p2->teta) - return -1; - else if (p1->teta > p2->teta) - return +1; - else - return 0; -} - -int spx_ls_select_bp(SPXLP *lp, const double tcol[/*1+m*/], - int nbp, SPXBP bp[/*1+m+m+1*/], int num, double *slope, double - teta_lim) -{ int m = lp->m; - int i, t, num1; - double teta, dz; - xassert(0 <= num && num <= nbp && nbp <= m+m+1); - /* select a new portion of break points */ - num1 = num; - for (t = num+1; t <= nbp; t++) - { if (bp[t].teta <= teta_lim) - { /* move break point to the beginning of the new portion */ - num1++; - i = bp[num1].i, teta = bp[num1].teta, dz = bp[num1].dc; - bp[num1].i = bp[t].i, bp[num1].teta = bp[t].teta, - bp[num1].dc = bp[t].dc; - bp[t].i = i, bp[t].teta = teta, bp[t].dc = dz; - } - } - /* sort new break points bp[num+1], ..., bp[num1] by ascending - * the ray parameter teta */ - if (num1 - num > 1) - qsort(&bp[num+1], num1 - num, sizeof(SPXBP), fcmp); - /* calculate the penalty function change at the new break points - * selected */ - for (t = num+1; t <= num1; t++) - { /* calculate the penalty function change relative to its value - * at break point bp[t-1] */ - dz = (*slope) * (bp[t].teta - (t == 1 ? 0.0 : bp[t-1].teta)); - /* calculate the penalty function change relative to its value - * at teta = 0 */ - bp[t].dz = (t == 1 ? 0.0 : bp[t-1].dz) + dz; - /* calculate a new slope of the penalty function to the right - * of the current break point bp[t] */ - i = (bp[t].i >= 0 ? bp[t].i : -bp[t].i); - xassert(0 <= i && i <= m); - if (i == 0) - *slope += fabs(1.0 * bp[t].dc); - else - *slope += fabs(tcol[i] * bp[t].dc); - } - return num1; -} -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxchuzr.h b/code/3rd_glpk/simplex/spxchuzr.h deleted file mode 100644 index 3ec90050..00000000 --- a/code/3rd_glpk/simplex/spxchuzr.h +++ /dev/null @@ -1,77 +0,0 @@ -/* spxchuzr.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPXCHUZR_H -#define SPXCHUZR_H - -#include "spxlp.h" - -#define spx_chuzr_std _glp_spx_chuzr_std -int spx_chuzr_std(SPXLP *lp, int phase, const double beta[/*1+m*/], - int q, double s, const double tcol[/*1+m*/], int *p_flag, - double tol_piv, double tol, double tol1); -/* choose basic variable (textbook ratio test) */ - -#define spx_chuzr_harris _glp_spx_chuzr_harris -int spx_chuzr_harris(SPXLP *lp, int phase, const double beta[/*1+m*/], - int q, double s, const double tcol[/*1+m*/], int *p_flag, - double tol_piv, double tol, double tol1); -/* choose basic variable (Harris' ratio test) */ - -#if 1 /* 22/VI-2017 */ -typedef struct SPXBP SPXBP; - -struct SPXBP -{ /* penalty function (sum of infeasibilities) break point */ - int i; - /* basic variable xB[i], 1 <= i <= m, that intersects its bound - * at this break point - * i > 0 if xB[i] intersects its lower bound (or fixed value) - * i < 0 if xB[i] intersects its upper bound - * i = 0 if xN[q] intersects its opposite bound */ - double teta; - /* ray parameter value, teta >= 0, at this break point */ - double dc; - /* increment of the penalty function coefficient cB[i] at this - * break point */ - double dz; - /* increment, z[t] - z[0], of the penalty function at this break - * point */ -}; - -#define spx_ls_eval_bp _glp_spx_ls_eval_bp -int spx_ls_eval_bp(SPXLP *lp, const double beta[/*1+m*/], - int q, double dq, const double tcol[/*1+m*/], double tol_piv, - SPXBP bp[/*1+2*m+1*/]); -/* determine penalty function break points */ - -#define spx_ls_select_bp _glp_spx_ls_select_bp -int spx_ls_select_bp(SPXLP *lp, const double tcol[/*1+m*/], - int nbp, SPXBP bp[/*1+m+m+1*/], int num, double *slope, double - teta_lim); -/* select and process penalty function break points */ -#endif - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxlp.c b/code/3rd_glpk/simplex/spxlp.c deleted file mode 100644 index 90ce2636..00000000 --- a/code/3rd_glpk/simplex/spxlp.c +++ /dev/null @@ -1,819 +0,0 @@ -/* spxlp.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spxlp.h" - -/*********************************************************************** -* spx_factorize - compute factorization of current basis matrix -* -* This routine computes factorization of the current basis matrix B. -* -* If the factorization has been successfully computed, the routine -* validates it and returns zero. Otherwise, the routine invalidates -* the factorization and returns the code provided by the factorization -* driver (bfd_factorize). */ - -static int jth_col(void *info, int j, int ind[], double val[]) -{ /* provide column B[j] */ - SPXLP *lp = info; - int m = lp->m; - int *A_ptr = lp->A_ptr; - int *head = lp->head; - int k, ptr, len; - xassert(1 <= j && j <= m); - k = head[j]; /* x[k] = xB[j] */ - ptr = A_ptr[k]; - len = A_ptr[k+1] - ptr; - memcpy(&ind[1], &lp->A_ind[ptr], len * sizeof(int)); - memcpy(&val[1], &lp->A_val[ptr], len * sizeof(double)); - return len; -} - -int spx_factorize(SPXLP *lp) -{ int ret; - ret = bfd_factorize(lp->bfd, lp->m, jth_col, lp); - lp->valid = (ret == 0); - return ret; -} - -/*********************************************************************** -* spx_eval_beta - compute current values of basic variables -* -* This routine computes vector beta = (beta[i]) of current values of -* basic variables xB = (xB[i]). (Factorization of the current basis -* matrix should be valid.) -* -* First the routine computes a modified vector of right-hand sides: -* -* n-m -* y = b - N * f = b - sum N[j] * f[j], -* j=1 -* -* where b = (b[i]) is the original vector of right-hand sides, N is -* a matrix composed from columns of the original constraint matrix A, -* which (columns) correspond to non-basic variables, f = (f[j]) is the -* vector of active bounds of non-basic variables xN = (xN[j]), -* N[j] = A[k] is a column of matrix A corresponding to non-basic -* variable xN[j] = x[k], f[j] is current active bound lN[j] = l[k] or -* uN[j] = u[k] of non-basic variable xN[j] = x[k]. The matrix-vector -* product N * f is computed as a linear combination of columns of N, -* so if f[j] = 0, column N[j] can be skipped. -* -* Then the routine performs FTRAN to compute the vector beta: -* -* beta = inv(B) * y. -* -* On exit the routine stores components of the vector beta to array -* locations beta[1], ..., beta[m]. */ - -void spx_eval_beta(SPXLP *lp, double beta[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - double *b = lp->b; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, k, ptr, end; - double fj, *y; - /* compute y = b - N * xN */ - /* y := b */ - y = beta; - memcpy(&y[1], &b[1], m * sizeof(double)); - /* y := y - N * f */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* f[j] := active bound of xN[j] */ - fj = flag[j] ? u[k] : l[k]; - if (fj == 0.0 || fj == -DBL_MAX) - { /* either xN[j] has zero active bound or it is unbounded; - * in the latter case its value is assumed to be zero */ - continue; - } - /* y := y - N[j] * f[j] */ - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - y[A_ind[ptr]] -= A_val[ptr] * fj; - } - /* compute beta = inv(B) * y */ - xassert(lp->valid); - bfd_ftran(lp->bfd, beta); - return; -} - -/*********************************************************************** -* spx_eval_obj - compute current value of objective function -* -* This routine computes the value of the objective function in the -* current basic solution: -* -* z = cB'* beta + cN'* f + c[0] = -* -* m n-m -* = sum cB[i] * beta[i] + sum cN[j] * f[j] + c[0], -* i=1 j=1 -* -* where cB = (cB[i]) is the vector of objective coefficients at basic -* variables, beta = (beta[i]) is the vector of current values of basic -* variables, cN = (cN[j]) is the vector of objective coefficients at -* non-basic variables, f = (f[j]) is the vector of current active -* bounds of non-basic variables, c[0] is the constant term of the -* objective function. -* -* It as assumed that components of the vector beta are stored in the -* array locations beta[1], ..., beta[m]. */ - -double spx_eval_obj(SPXLP *lp, const double beta[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int i, j, k; - double fj, z; - /* compute z = cB'* beta + cN'* f + c0 */ - /* z := c0 */ - z = c[0]; - /* z := z + cB'* beta */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - z += c[k] * beta[i]; - } - /* z := z + cN'* f */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* f[j] := active bound of xN[j] */ - fj = flag[j] ? u[k] : l[k]; - if (fj == 0.0 || fj == -DBL_MAX) - { /* either xN[j] has zero active bound or it is unbounded; - * in the latter case its value is assumed to be zero */ - continue; - } - z += c[k] * fj; - } - return z; -} - -/*********************************************************************** -* spx_eval_pi - compute simplex multipliers in current basis -* -* This routine computes vector pi = (pi[i]) of simplex multipliers in -* the current basis. (Factorization of the current basis matrix should -* be valid.) -* -* The vector pi is computed by performing BTRAN: -* -* pi = inv(B') * cB, -* -* where cB = (cB[i]) is the vector of objective coefficients at basic -* variables xB = (xB[i]). -* -* On exit components of vector pi are stored in the array locations -* pi[1], ..., pi[m]. */ - -void spx_eval_pi(SPXLP *lp, double pi[/*1+m*/]) -{ int m = lp->m; - double *c = lp->c; - int *head = lp->head; - int i; - double *cB; - /* construct cB */ - cB = pi; - for (i = 1; i <= m; i++) - cB[i] = c[head[i]]; - /* compute pi = inv(B) * cB */ - bfd_btran(lp->bfd, pi); - return; -} - -/*********************************************************************** -* spx_eval_dj - compute reduced cost of j-th non-basic variable -* -* This routine computes reduced cost d[j] of non-basic variable -* xN[j] = x[k], 1 <= j <= n-m, in the current basic solution: -* -* d[j] = c[k] - A'[k] * pi, -* -* where c[k] is the objective coefficient at x[k], A[k] is k-th column -* of the constraint matrix, pi is the vector of simplex multipliers in -* the current basis. -* -* It as assumed that components of the vector pi are stored in the -* array locations pi[1], ..., pi[m]. */ - -double spx_eval_dj(SPXLP *lp, const double pi[/*1+m*/], int j) -{ int m = lp->m; - int n = lp->n; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int k, ptr, end; - double dj; - xassert(1 <= j && j <= n-m); - k = lp->head[m+j]; /* x[k] = xN[j] */ - /* dj := c[k] */ - dj = lp->c[k]; - /* dj := dj - A'[k] * pi */ - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - dj -= A_val[ptr] * pi[A_ind[ptr]]; - return dj; -} - -/*********************************************************************** -* spx_eval_tcol - compute j-th column of simplex table -* -* This routine computes j-th column of the current simplex table -* T = (T[i,j]) = - inv(B) * N, 1 <= j <= n-m. (Factorization of the -* current basis matrix should be valid.) -* -* The simplex table column is computed by performing FTRAN: -* -* tcol = - inv(B) * N[j], -* -* where B is the current basis matrix, N[j] = A[k] is a column of the -* constraint matrix corresponding to non-basic variable xN[j] = x[k]. -* -* On exit components of the simplex table column are stored in the -* array locations tcol[1], ... tcol[m]. */ - -void spx_eval_tcol(SPXLP *lp, int j, double tcol[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int *head = lp->head; - int i, k, ptr, end; - xassert(1 <= j && j <= n-m); - k = head[m+j]; /* x[k] = xN[j] */ - /* compute tcol = - inv(B) * N[j] */ - for (i = 1; i <= m; i++) - tcol[i] = 0.0; - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - tcol[A_ind[ptr]] = -A_val[ptr]; - bfd_ftran(lp->bfd, tcol); - return; -} - -/*********************************************************************** -* spx_eval_rho - compute i-th row of basis matrix inverse -* -* This routine computes i-th row of the matrix inv(B), where B is -* the current basis matrix, 1 <= i <= m. (Factorization of the current -* basis matrix should be valid.) -* -* The inverse row is computed by performing BTRAN: -* -* rho = inv(B') * e[i], -* -* where e[i] is i-th column of unity matrix. -* -* On exit components of the row are stored in the array locations -* row[1], ..., row[m]. */ - -void spx_eval_rho(SPXLP *lp, int i, double rho[/*1+m*/]) -{ int m = lp->m; - int j; - xassert(1 <= i && i <= m); - /* compute rho = inv(B') * e[i] */ - for (j = 1; j <= m; j++) - rho[j] = 0.0; - rho[i] = 1.0; - bfd_btran(lp->bfd, rho); - return; -} - -#if 1 /* 31/III-2016 */ -void spx_eval_rho_s(SPXLP *lp, int i, FVS *rho) -{ /* sparse version of spx_eval_rho */ - int m = lp->m; - xassert(1 <= i && i <= m); - /* compute rho = inv(B') * e[i] */ - xassert(rho->n == m); - fvs_clear_vec(rho); - rho->nnz = 1; - rho->ind[1] = i; - rho->vec[i] = 1.0; - bfd_btran_s(lp->bfd, rho); - return; -} -#endif - -/*********************************************************************** -* spx_eval_tij - compute element T[i,j] of simplex table -* -* This routine computes element T[i,j] of the current simplex table -* T = - inv(B) * N, 1 <= i <= m, 1 <= j <= n-m, with the following -* formula: -* -* T[i,j] = - N'[j] * rho, (1) -* -* where N[j] = A[k] is a column of the constraint matrix corresponding -* to non-basic variable xN[j] = x[k], rho is i-th row of the inverse -* matrix inv(B). -* -* It as assumed that components of the inverse row rho = (rho[j]) are -* stored in the array locations rho[1], ..., rho[m]. */ - -double spx_eval_tij(SPXLP *lp, const double rho[/*1+m*/], int j) -{ int m = lp->m; - int n = lp->n; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int k, ptr, end; - double tij; - xassert(1 <= j && j <= n-m); - k = lp->head[m+j]; /* x[k] = xN[j] */ - /* compute t[i,j] = - N'[j] * pi */ - tij = 0.0; - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - tij -= A_val[ptr] * rho[A_ind[ptr]]; - return tij; -} - -/*********************************************************************** -* spx_eval_trow - compute i-th row of simplex table -* -* This routine computes i-th row of the current simplex table -* T = (T[i,j]) = - inv(B) * N, 1 <= i <= m. -* -* Elements of the row T[i] = (T[i,j]), j = 1, ..., n-m, are computed -* directly with the routine spx_eval_tij. -* -* The vector rho = (rho[j]), which is i-th row of the basis inverse -* inv(B), should be previously computed with the routine spx_eval_rho. -* It is assumed that elements of this vector are stored in the array -* locations rho[1], ..., rho[m]. -* -* On exit components of the simplex table row are stored in the array -* locations trow[1], ... trow[n-m]. -* -* NOTE: For testing/debugging only. */ - -void spx_eval_trow(SPXLP *lp, const double rho[/*1+m*/], double - trow[/*1+n-m*/]) -{ int m = lp->m; - int n = lp->n; - int j; - for (j = 1; j <= n-m; j++) - trow[j] = spx_eval_tij(lp, rho, j); - return; -} - -/*********************************************************************** -* spx_update_beta - update values of basic variables -* -* This routine updates the vector beta = (beta[i]) of values of basic -* variables xB = (xB[i]) for the adjacent basis. -* -* On entry to the routine components of the vector beta in the current -* basis should be placed in array locations beta[1], ..., beta[m]. -* -* The parameter 1 <= p <= m specifies basic variable xB[p] which -* becomes non-basic variable xN[q] in the adjacent basis. The special -* case p < 0 means that non-basic variable xN[q] goes from its current -* active bound to opposite one in the adjacent basis. -* -* If the flag p_flag is set, the active bound of xB[p] in the adjacent -* basis is set to its upper bound. (In this case xB[p] should have its -* upper bound and should not be fixed.) -* -* The parameter 1 <= q <= n-m specifies non-basic variable xN[q] which -* becomes basic variable xB[p] in the adjacent basis (if 1 <= p <= m), -* or goes to its opposite bound (if p < 0). (In the latter case xN[q] -* should have both lower and upper bounds and should not be fixed.) -* -* It is assumed that the array tcol contains elements of q-th (pivot) -* column T[q] of the simple table in locations tcol[1], ..., tcol[m]. -* (This column should be computed for the current basis.) -* -* First, the routine determines the increment of basic variable xB[p] -* in the adjacent basis (but only if 1 <= p <= m): -* -* ( - beta[p], if -inf < xB[p] < +inf -* ( -* delta xB[p] = { lB[p] - beta[p], if p_flag = 0 -* ( -* ( uB[p] - beta[p], if p_flag = 1 -* -* where beta[p] is the value of xB[p] in the current basis, lB[p] and -* uB[p] are its lower and upper bounds. Then, the routine determines -* the increment of non-basic variable xN[q] in the adjacent basis: -* -* ( delta xB[p] / T[p,q], if 1 <= p <= m -* ( -* delta xN[q] = { uN[q] - lN[q], if p < 0 and f[q] = lN[q] -* ( -* ( lN[q] - uN[q], if p < 0 and f[q] = uN[q] -* -* where T[p,q] is the pivot element of the simplex table, f[q] is the -* active bound of xN[q] in the current basis. -* -* If 1 <= p <= m, in the adjacent basis xN[q] becomes xB[p], so: -* -* new beta[p] = f[q] + delta xN[q]. -* -* Values of other basic variables xB[i] for 1 <= i <= m, i != p, are -* updated as follows: -* -* new beta[i] = beta[i] + T[i,q] * delta xN[q]. -* -* On exit the routine stores updated components of the vector beta to -* the same locations, where the input vector beta was stored. */ - -void spx_update_beta(SPXLP *lp, double beta[/*1+m*/], int p, - int p_flag, int q, const double tcol[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int i, k; - double delta_p, delta_q; - if (p < 0) - { /* special case: xN[q] goes to its opposite bound */ - xassert(1 <= q && q <= n-m); - /* xN[q] should be double-bounded variable */ - k = head[m+q]; /* x[k] = xN[q] */ - xassert(l[k] != -DBL_MAX && u[k] != +DBL_MAX && l[k] != u[k]); - /* determine delta xN[q] */ - if (flag[q]) - { /* xN[q] goes from its upper bound to its lower bound */ - delta_q = l[k] - u[k]; - } - else - { /* xN[q] goes from its lower bound to its upper bound */ - delta_q = u[k] - l[k]; - } - } - else - { /* xB[p] leaves the basis, xN[q] enters the basis */ - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - /* determine delta xB[p] */ - k = head[p]; /* x[k] = xB[p] */ - if (p_flag) - { /* xB[p] goes to its upper bound */ - xassert(l[k] != u[k] && u[k] != +DBL_MAX); - delta_p = u[k] - beta[p]; - } - else if (l[k] == -DBL_MAX) - { /* unbounded xB[p] becomes non-basic (unusual case) */ - xassert(u[k] == +DBL_MAX); - delta_p = 0.0 - beta[p]; - } - else - { /* xB[p] goes to its lower bound or becomes fixed */ - delta_p = l[k] - beta[p]; - } - /* determine delta xN[q] */ - delta_q = delta_p / tcol[p]; - /* compute new beta[p], which is the value of xN[q] in the - * adjacent basis */ - k = head[m+q]; /* x[k] = xN[q] */ - if (flag[q]) - { /* xN[q] has its upper bound active */ - xassert(l[k] != u[k] && u[k] != +DBL_MAX); - beta[p] = u[k] + delta_q; - } - else if (l[k] == -DBL_MAX) - { /* xN[q] is non-basic unbounded variable */ - xassert(u[k] == +DBL_MAX); - beta[p] = 0.0 + delta_q; - } - else - { /* xN[q] has its lower bound active or is fixed (latter - * case is unusual) */ - beta[p] = l[k] + delta_q; - } - } - /* compute new beta[i] for all i != p */ - for (i = 1; i <= m; i++) - { if (i != p) - beta[i] += tcol[i] * delta_q; - } - return; -} - -#if 1 /* 30/III-2016 */ -void spx_update_beta_s(SPXLP *lp, double beta[/*1+m*/], int p, - int p_flag, int q, const FVS *tcol) -{ /* sparse version of spx_update_beta */ - int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int nnz = tcol->nnz; - int *ind = tcol->ind; - double *vec = tcol->vec; - int i, k; - double delta_p, delta_q; - xassert(tcol->n == m); - if (p < 0) - { /* special case: xN[q] goes to its opposite bound */ -#if 0 /* 11/VI-2017 */ - /* FIXME: not tested yet */ - xassert(0); -#endif - xassert(1 <= q && q <= n-m); - /* xN[q] should be double-bounded variable */ - k = head[m+q]; /* x[k] = xN[q] */ - xassert(l[k] != -DBL_MAX && u[k] != +DBL_MAX && l[k] != u[k]); - /* determine delta xN[q] */ - if (flag[q]) - { /* xN[q] goes from its upper bound to its lower bound */ - delta_q = l[k] - u[k]; - } - else - { /* xN[q] goes from its lower bound to its upper bound */ - delta_q = u[k] - l[k]; - } - } - else - { /* xB[p] leaves the basis, xN[q] enters the basis */ - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - /* determine delta xB[p] */ - k = head[p]; /* x[k] = xB[p] */ - if (p_flag) - { /* xB[p] goes to its upper bound */ - xassert(l[k] != u[k] && u[k] != +DBL_MAX); - delta_p = u[k] - beta[p]; - } - else if (l[k] == -DBL_MAX) - { /* unbounded xB[p] becomes non-basic (unusual case) */ - xassert(u[k] == +DBL_MAX); - delta_p = 0.0 - beta[p]; - } - else - { /* xB[p] goes to its lower bound or becomes fixed */ - delta_p = l[k] - beta[p]; - } - /* determine delta xN[q] */ - delta_q = delta_p / vec[p]; - /* compute new beta[p], which is the value of xN[q] in the - * adjacent basis */ - k = head[m+q]; /* x[k] = xN[q] */ - if (flag[q]) - { /* xN[q] has its upper bound active */ - xassert(l[k] != u[k] && u[k] != +DBL_MAX); - beta[p] = u[k] + delta_q; - } - else if (l[k] == -DBL_MAX) - { /* xN[q] is non-basic unbounded variable */ - xassert(u[k] == +DBL_MAX); - beta[p] = 0.0 + delta_q; - } - else - { /* xN[q] has its lower bound active or is fixed (latter - * case is unusual) */ - beta[p] = l[k] + delta_q; - } - } - /* compute new beta[i] for all i != p */ - for (k = 1; k <= nnz; k++) - { i = ind[k]; - if (i != p) - beta[i] += vec[i] * delta_q; - } - return; -} -#endif - -/*********************************************************************** -* spx_update_d - update reduced costs of non-basic variables -* -* This routine updates the vector d = (d[j]) of reduced costs of -* non-basic variables xN = (xN[j]) for the adjacent basis. -* -* On entry to the routine components of the vector d in the current -* basis should be placed in locations d[1], ..., d[n-m]. -* -* The parameter 1 <= p <= m specifies basic variable xB[p] which -* becomes non-basic variable xN[q] in the adjacent basis. -* -* The parameter 1 <= q <= n-m specified non-basic variable xN[q] which -* becomes basic variable xB[p] in the adjacent basis. -* -* It is assumed that the array trow contains elements of p-th (pivot) -* row T'[p] of the simplex table in locations trow[1], ..., trow[n-m]. -* It is also assumed that the array tcol contains elements of q-th -* (pivot) column T[q] of the simple table in locations tcol[1], ..., -* tcol[m]. (These row and column should be computed for the current -* basis.) -* -* First, the routine computes more accurate reduced cost d[q] in the -* current basis using q-th column of the simplex table: -* -* n-m -* d[q] = cN[q] + sum t[i,q] * cB[i], -* i=1 -* -* where cN[q] and cB[i] are objective coefficients at variables xN[q] -* and xB[i], resp. The routine also computes the relative error: -* -* e = |d[q] - d'[q]| / (1 + |d[q]|), -* -* where d'[q] is the reduced cost of xN[q] on entry to the routine, -* and returns e on exit. (If e happens to be large enough, the calling -* program may compute the reduced costs directly, since other reduced -* costs also may be inaccurate.) -* -* In the adjacent basis xB[p] becomes xN[q], so: -* -* new d[q] = d[q] / T[p,q], -* -* where T[p,q] is the pivot element of the simplex table (it is taken -* from column T[q] as more accurate). Reduced costs of other non-basic -* variables xN[j] for 1 <= j <= n-m, j != q, are updated as follows: -* -* new d[j] = d[j] + T[p,j] * new d[q]. -* -* On exit the routine stores updated components of the vector d to the -* same locations, where the input vector d was stored. */ - -double spx_update_d(SPXLP *lp, double d[/*1+n-m*/], int p, int q, - const double trow[/*1+n-m*/], const double tcol[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - int *head = lp->head; - int i, j, k; - double dq, e; - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n); - /* compute d[q] in current basis more accurately */ - k = head[m+q]; /* x[k] = xN[q] */ - dq = c[k]; - for (i = 1; i <= m; i++) - dq += tcol[i] * c[head[i]]; - /* compute relative error in d[q] */ - e = fabs(dq - d[q]) / (1.0 + fabs(dq)); - /* compute new d[q], which is the reduced cost of xB[p] in the - * adjacent basis */ - d[q] = (dq /= tcol[p]); - /* compute new d[j] for all j != q */ - for (j = 1; j <= n-m; j++) - { if (j != q) - d[j] -= trow[j] * dq; - } - return e; -} - -#if 1 /* 30/III-2016 */ -double spx_update_d_s(SPXLP *lp, double d[/*1+n-m*/], int p, int q, - const FVS *trow, const FVS *tcol) -{ /* sparse version of spx_update_d */ - int m = lp->m; - int n = lp->n; - double *c = lp->c; - int *head = lp->head; - int trow_nnz = trow->nnz; - int *trow_ind = trow->ind; - double *trow_vec = trow->vec; - int tcol_nnz = tcol->nnz; - int *tcol_ind = tcol->ind; - double *tcol_vec = tcol->vec; - int i, j, k; - double dq, e; - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n); - xassert(trow->n == n-m); - xassert(tcol->n == m); - /* compute d[q] in current basis more accurately */ - k = head[m+q]; /* x[k] = xN[q] */ - dq = c[k]; - for (k = 1; k <= tcol_nnz; k++) - { i = tcol_ind[k]; - dq += tcol_vec[i] * c[head[i]]; - } - /* compute relative error in d[q] */ - e = fabs(dq - d[q]) / (1.0 + fabs(dq)); - /* compute new d[q], which is the reduced cost of xB[p] in the - * adjacent basis */ - d[q] = (dq /= tcol_vec[p]); - /* compute new d[j] for all j != q */ - for (k = 1; k <= trow_nnz; k++) - { j = trow_ind[k]; - if (j != q) - d[j] -= trow_vec[j] * dq; - } - return e; -} -#endif - -/*********************************************************************** -* spx_change_basis - change current basis to adjacent one -* -* This routine changes the current basis to the adjacent one making -* necessary changes in lp->head and lp->flag members. -* -* The parameters p, p_flag, and q have the same meaning as for the -* routine spx_update_beta. */ - -void spx_change_basis(SPXLP *lp, int p, int p_flag, int q) -{ int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int k; - if (p < 0) - { /* special case: xN[q] goes to its opposite bound */ - xassert(1 <= q && q <= n-m); - /* xN[q] should be double-bounded variable */ - k = head[m+q]; /* x[k] = xN[q] */ - xassert(l[k] != -DBL_MAX && u[k] != +DBL_MAX && l[k] != u[k]); - /* change active bound flag */ - flag[q] = 1 - flag[q]; - } - else - { /* xB[p] leaves the basis, xN[q] enters the basis */ - xassert(1 <= p && p <= m); - xassert(p_flag == 0 || p_flag == 1); - xassert(1 <= q && q <= n-m); - k = head[p]; /* xB[p] = x[k] */ - if (p_flag) - { /* xB[p] goes to its upper bound */ - xassert(l[k] != u[k] && u[k] != +DBL_MAX); - } - /* swap xB[p] and xN[q] in the basis */ - head[p] = head[m+q], head[m+q] = k; - /* and set active bound flag for new xN[q] */ - lp->flag[q] = p_flag; - } - return; -} - -/*********************************************************************** -* spx_update_invb - update factorization of basis matrix -* -* This routine updates factorization of the basis matrix B when i-th -* column of B is replaced by k-th column of the constraint matrix A. -* -* The parameter 1 <= i <= m specifies the number of column of matrix B -* to be replaced by a new column. -* -* The parameter 1 <= k <= n specifies the number of column of matrix A -* to be used for replacement. -* -* If the factorization has been successfully updated, the routine -* validates it and returns zero. Otherwise, the routine invalidates -* the factorization and returns the code provided by the factorization -* driver (bfd_update). */ - -int spx_update_invb(SPXLP *lp, int i, int k) -{ int m = lp->m; - int n = lp->n; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int ptr, len, ret; - xassert(1 <= i && i <= m); - xassert(1 <= k && k <= n); - ptr = A_ptr[k]; - len = A_ptr[k+1] - ptr; - ret = bfd_update(lp->bfd, i, len, &A_ind[ptr-1], &A_val[ptr-1]); - lp->valid = (ret == 0); - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxlp.h b/code/3rd_glpk/simplex/spxlp.h deleted file mode 100644 index 29a135fe..00000000 --- a/code/3rd_glpk/simplex/spxlp.h +++ /dev/null @@ -1,234 +0,0 @@ -/* spxlp.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPXLP_H -#define SPXLP_H - -#include "bfd.h" - -/*********************************************************************** -* The structure SPXLP describes LP problem and its current basis. -* -* It is assumed that LP problem has the following formulation (this is -* so called "working format"): -* -* z = c'* x + c0 -> min (1) -* -* A * x = b (2) -* -* l <= x <= u (3) -* -* where: -* -* x = (x[k]) is a n-vector of variables; -* -* z is an objective function; -* -* c = (c[k]) is a n-vector of objective coefficients; -* -* c0 is a constant term of the objective function; -* -* A = (a[i,k]) is a mxn-matrix of constraint coefficients; -* -* b = (b[i]) is a m-vector of right-hand sides; -* -* l = (l[k]) is a n-vector of lower bounds of variables; -* -* u = (u[k]) is a n-vector of upper bounds of variables. -* -* If variable x[k] has no lower (upper) bound, it is formally assumed -* that l[k] = -inf (u[k] = +inf). Variable having no bounds is called -* free (unbounded) variable. If l[k] = u[k], variable x[k] is assumed -* to be fixed. -* -* It is also assumed that matrix A has full row rank: rank(A) = m, -* i.e. all its rows are linearly independent, so m <= n. -* -* The (current) basis is defined by an appropriate permutation matrix -* P of order n such that: -* -* ( xB ) -* P * x = ( ), (4) -* ( xN ) -* -* where xB = (xB[i]) is a m-vector of basic variables, xN = (xN[j]) is -* a (n-m)-vector of non-basic variables. If a non-basic variable xN[j] -* has both lower and upper bounds, there is used an additional flag to -* indicate which bound is active. -* -* From (2) and (4) it follows that: -* -* A * P'* P * x = b <=> B * xB + N * xN = b, (5) -* -* where P' is a matrix transposed to P, and -* -* A * P' = (B | N). (6) -* -* Here B is the basis matrix, which is a square non-singular matrix -* of order m composed from columns of matrix A that correspond to -* basic variables xB, and N is a mx(n-m) matrix composed from columns -* of matrix A that correspond to non-basic variables xN. */ - -typedef struct SPXLP SPXLP; - -struct SPXLP -{ /* LP problem data and its (current) basis */ - int m; - /* number of equality constraints, m > 0 */ - int n; - /* number of variables, n >= m */ - int nnz; - /* number of non-zeros in constraint matrix A */ - /*--------------------------------------------------------------*/ - /* mxn-matrix A of constraint coefficients in sparse column-wise - * format */ - int *A_ptr; /* int A_ptr[1+n+1]; */ - /* A_ptr[0] is not used; - * A_ptr[k], 1 <= k <= n, is starting position of k-th column in - * arrays A_ind and A_val; note that A_ptr[1] is always 1; - * A_ptr[n+1] indicates the position after the last element in - * arrays A_ind and A_val, i.e. A_ptr[n+1] = nnz+1, where nnz is - * the number of non-zero elements in matrix A; - * the length of k-th column (the number of non-zero elements in - * that column) can be calculated as A_ptr[k+1] - A_ptr[k] */ - int *A_ind; /* int A_ind[1+nnz]; */ - /* row indices */ - double *A_val; /* double A_val[1+nnz]; */ - /* non-zero element values (constraint coefficients) */ - /*--------------------------------------------------------------*/ - /* principal vectors of LP formulation */ - double *b; /* double b[1+m]; */ - /* b[0] is not used; - * b[i], 1 <= i <= m, is the right-hand side of i-th equality - * constraint */ - double *c; /* double c[1+n]; */ - /* c[0] is the constant term of the objective function; - * c[k], 1 <= k <= n, is the objective function coefficient at - * variable x[k] */ - double *l; /* double l[1+n]; */ - /* l[0] is not used; - * l[k], 1 <= k <= n, is the lower bound of variable x[k]; - * if x[k] has no lower bound, l[k] = -DBL_MAX */ - double *u; /* double u[1+n]; */ - /* u[0] is not used; - * u[k], 1 <= k <= n, is the upper bound of variable u[k]; - * if x[k] has no upper bound, u[k] = +DBL_MAX; - * note that l[k] = u[k] means that x[k] is fixed variable */ - /*--------------------------------------------------------------*/ - /* LP basis */ - int *head; /* int head[1+n]; */ - /* basis header, which is permutation matrix P (4): - * head[0] is not used; - * head[i] = k means that xB[i] = x[k], 1 <= i <= m; - * head[m+j] = k, means that xN[j] = x[k], 1 <= j <= n-m */ - char *flag; /* char flag[1+n-m]; */ - /* flags of non-basic variables: - * flag[0] is not used; - * flag[j], 1 <= j <= n-m, indicates that non-basic variable - * xN[j] is non-fixed and has its upper bound active */ - /*--------------------------------------------------------------*/ - /* basis matrix B of order m stored in factorized form */ - int valid; - /* factorization validity flag */ - BFD *bfd; - /* driver to factorization of the basis matrix */ -}; - -#define spx_factorize _glp_spx_factorize -int spx_factorize(SPXLP *lp); -/* compute factorization of current basis matrix */ - -#define spx_eval_beta _glp_spx_eval_beta -void spx_eval_beta(SPXLP *lp, double beta[/*1+m*/]); -/* compute values of basic variables */ - -#define spx_eval_obj _glp_spx_eval_obj -double spx_eval_obj(SPXLP *lp, const double beta[/*1+m*/]); -/* compute value of objective function */ - -#define spx_eval_pi _glp_spx_eval_pi -void spx_eval_pi(SPXLP *lp, double pi[/*1+m*/]); -/* compute simplex multipliers */ - -#define spx_eval_dj _glp_spx_eval_dj -double spx_eval_dj(SPXLP *lp, const double pi[/*1+m*/], int j); -/* compute reduced cost of j-th non-basic variable */ - -#define spx_eval_tcol _glp_spx_eval_tcol -void spx_eval_tcol(SPXLP *lp, int j, double tcol[/*1+m*/]); -/* compute j-th column of simplex table */ - -#define spx_eval_rho _glp_spx_eval_rho -void spx_eval_rho(SPXLP *lp, int i, double rho[/*1+m*/]); -/* compute i-th row of basis matrix inverse */ - -#if 1 /* 31/III-2016 */ -#define spx_eval_rho_s _glp_spx_eval_rho_s -void spx_eval_rho_s(SPXLP *lp, int i, FVS *rho); -/* sparse version of spx_eval_rho */ -#endif - -#define spx_eval_tij _glp_spx_eval_tij -double spx_eval_tij(SPXLP *lp, const double rho[/*1+m*/], int j); -/* compute element T[i,j] of simplex table */ - -#define spx_eval_trow _glp_spx_eval_trow -void spx_eval_trow(SPXLP *lp, const double rho[/*1+m*/], double - trow[/*1+n-m*/]); -/* compute i-th row of simplex table */ - -#define spx_update_beta _glp_spx_update_beta -void spx_update_beta(SPXLP *lp, double beta[/*1+m*/], int p, - int p_flag, int q, const double tcol[/*1+m*/]); -/* update values of basic variables */ - -#if 1 /* 30/III-2016 */ -#define spx_update_beta_s _glp_spx_update_beta_s -void spx_update_beta_s(SPXLP *lp, double beta[/*1+m*/], int p, - int p_flag, int q, const FVS *tcol); -/* sparse version of spx_update_beta */ -#endif - -#define spx_update_d _glp_spx_update_d -double spx_update_d(SPXLP *lp, double d[/*1+n-m*/], int p, int q, - const double trow[/*1+n-m*/], const double tcol[/*1+m*/]); -/* update reduced costs of non-basic variables */ - -#if 1 /* 30/III-2016 */ -#define spx_update_d_s _glp_spx_update_d_s -double spx_update_d_s(SPXLP *lp, double d[/*1+n-m*/], int p, int q, - const FVS *trow, const FVS *tcol); -/* sparse version of spx_update_d */ -#endif - -#define spx_change_basis _glp_spx_change_basis -void spx_change_basis(SPXLP *lp, int p, int p_flag, int q); -/* change current basis to adjacent one */ - -#define spx_update_invb _glp_spx_update_invb -int spx_update_invb(SPXLP *lp, int i, int k); -/* update factorization of basis matrix */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxnt.c b/code/3rd_glpk/simplex/spxnt.c deleted file mode 100644 index 7eaac852..00000000 --- a/code/3rd_glpk/simplex/spxnt.c +++ /dev/null @@ -1,303 +0,0 @@ -/* spxnt.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spxnt.h" - -/*********************************************************************** -* spx_alloc_nt - allocate matrix N in sparse row-wise format -* -* This routine allocates the memory for arrays needed to represent the -* matrix N composed of non-basic columns of the constraint matrix A. */ - -void spx_alloc_nt(SPXLP *lp, SPXNT *nt) -{ int m = lp->m; - int nnz = lp->nnz; - nt->ptr = talloc(1+m, int); - nt->len = talloc(1+m, int); - nt->ind = talloc(1+nnz, int); - nt->val = talloc(1+nnz, double); - return; -} - -/*********************************************************************** -* spx_init_nt - initialize row pointers for matrix N -* -* This routine initializes (sets up) row pointers for the matrix N -* using column-wise representation of the constraint matrix A. -* -* This routine needs to be called only once. */ - -void spx_init_nt(SPXLP *lp, SPXNT *nt) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - int *NT_ptr = nt->ptr; - int *NT_len = nt->len; - int i, k, ptr, end; - /* calculate NT_len[i] = maximal number of non-zeros in i-th row - * of N = number of non-zeros in i-th row of A */ - memset(&NT_len[1], 0, m * sizeof(int)); - for (k = 1; k <= n; k++) - { ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - NT_len[A_ind[ptr]]++; - } - /* initialize row pointers NT_ptr[i], i = 1,...,n-m */ - NT_ptr[1] = 1; - for (i = 2; i <= m; i++) - NT_ptr[i] = NT_ptr[i-1] + NT_len[i-1]; - xassert(NT_ptr[m] + NT_len[m] == nnz+1); - return; -} - -/*********************************************************************** -* spx_nt_add_col - add column N[j] = A[k] to matrix N -* -* This routine adds elements of column N[j] = A[k], 1 <= j <= n-m, -* 1 <= k <= n, to the row-wise represntation of the matrix N. It is -* assumed (with no check) that elements of the specified column are -* missing in the row-wise represntation of N. */ - -void spx_nt_add_col(SPXLP *lp, SPXNT *nt, int j, int k) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - int *NT_ptr = nt->ptr; - int *NT_len = nt->len; - int *NT_ind = nt->ind; - double *NT_val = nt->val; - int i, ptr, end, pos; - xassert(1 <= j && j <= n-m); - xassert(1 <= k && k <= n); - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - { i = A_ind[ptr]; - /* add element N[i,j] = A[i,k] to i-th row of matrix N */ - pos = NT_ptr[i] + (NT_len[i]++); - if (i < m) - xassert(pos < NT_ptr[i+1]); - else - xassert(pos <= nnz); - NT_ind[pos] = j; - NT_val[pos] = A_val[ptr]; - } - return; -} - -/*********************************************************************** -* spx_build_nt - build matrix N for current basis -* -* This routine builds the row-wise represntation of the matrix N -* for the current basis by adding columns of the constraint matrix A -* corresponding to non-basic variables. */ - -void spx_build_nt(SPXLP *lp, SPXNT *nt) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - int *NT_len = nt->len; - int j, k; - /* N := 0 */ - memset(&NT_len[1], 0, m * sizeof(int)); - /* add non-basic columns N[j] = A[k] */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - spx_nt_add_col(lp, nt, j, k); - } - return; -} - -/*********************************************************************** -* spx_nt_del_col - remove column N[j] = A[k] from matrix N -* -* This routine removes elements of column N[j] = A[k], 1 <= j <= n-m, -* 1 <= k <= n, from the row-wise representation of the matrix N. It is -* assumed (with no check) that elements of the specified column are -* present in the row-wise representation of N. */ - -void spx_nt_del_col(SPXLP *lp, SPXNT *nt, int j, int k) -{ int m = lp->m; - int n = lp->n; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - int *NT_ptr = nt->ptr; - int *NT_len = nt->len; - int *NT_ind = nt->ind; - double *NT_val = nt->val; - int i, ptr, end, ptr1, end1; - xassert(1 <= j && j <= n-m); - xassert(1 <= k && k <= n); - ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - { i = A_ind[ptr]; - /* find element N[i,j] = A[i,k] in i-th row of matrix N */ - ptr1 = NT_ptr[i]; - end1 = ptr1 + NT_len[i]; - for (; NT_ind[ptr1] != j; ptr1++) - /* nop */; - xassert(ptr1 < end1); - /* and remove it from i-th row element list */ - NT_len[i]--; - NT_ind[ptr1] = NT_ind[end1-1]; - NT_val[ptr1] = NT_val[end1-1]; - } - return; -} - -/*********************************************************************** -* spx_update_nt - update matrix N for adjacent basis -* -* This routine updates the row-wise represntation of matrix N for -* the adjacent basis, where column N[q], 1 <= q <= n-m, is replaced by -* column B[p], 1 <= p <= m, of the current basis matrix B. */ - -void spx_update_nt(SPXLP *lp, SPXNT *nt, int p, int q) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - /* remove old column N[q] corresponding to variable xN[q] */ - spx_nt_del_col(lp, nt, q, head[m+q]); - /* add new column N[q] corresponding to variable xB[p] */ - spx_nt_add_col(lp, nt, q, head[p]); - return; -} - -/*********************************************************************** -* spx_nt_prod - compute product y := y + s * N'* x -* -* This routine computes the product: -* -* y := y + s * N'* x, -* -* where N' is a matrix transposed to the mx(n-m)-matrix N composed -* from non-basic columns of the constraint matrix A, x is a m-vector, -* s is a scalar, y is (n-m)-vector. -* -* If the flag ign is non-zero, the routine ignores the input content -* of the array y assuming that y = 0. -* -* The routine uses the row-wise representation of the matrix N and -* computes the product as a linear combination: -* -* y := y + s * (N'[1] * x[1] + ... + N'[m] * x[m]), -* -* where N'[i] is i-th row of N, 1 <= i <= m. */ - -void spx_nt_prod(SPXLP *lp, SPXNT *nt, double y[/*1+n-m*/], int ign, - double s, const double x[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - int *NT_ptr = nt->ptr; - int *NT_len = nt->len; - int *NT_ind = nt->ind; - double *NT_val = nt->val; - int i, j, ptr, end; - double t; - if (ign) - { /* y := 0 */ - for (j = 1; j <= n-m; j++) - y[j] = 0.0; - } - for (i = 1; i <= m; i++) - { if (x[i] != 0.0) - { /* y := y + s * (i-th row of N) * x[i] */ - t = s * x[i]; - ptr = NT_ptr[i]; - end = ptr + NT_len[i]; - for (; ptr < end; ptr++) - y[NT_ind[ptr]] += NT_val[ptr] * t; - } - } - return; -} - -#if 1 /* 31/III-2016 */ -void spx_nt_prod_s(SPXLP *lp, SPXNT *nt, FVS *y, int ign, double s, - const FVS *x, double eps) -{ /* sparse version of spx_nt_prod */ - int *NT_ptr = nt->ptr; - int *NT_len = nt->len; - int *NT_ind = nt->ind; - double *NT_val = nt->val; - int *x_ind = x->ind; - double *x_vec = x->vec; - int *y_ind = y->ind; - double *y_vec = y->vec; - int i, j, k, nnz, ptr, end; - double t; - xassert(x->n == lp->m); - xassert(y->n == lp->n-lp->m); - if (ign) - { /* y := 0 */ - fvs_clear_vec(y); - } - nnz = y->nnz; - for (k = x->nnz; k >= 1; k--) - { i = x_ind[k]; - /* y := y + s * (i-th row of N) * x[i] */ - t = s * x_vec[i]; - ptr = NT_ptr[i]; - end = ptr + NT_len[i]; - for (; ptr < end; ptr++) - { j = NT_ind[ptr]; - if (y_vec[j] == 0.0) - y_ind[++nnz] = j; - y_vec[j] += NT_val[ptr] * t; - /* don't forget about numeric cancellation */ - if (y_vec[j] == 0.0) - y_vec[j] = DBL_MIN; - } - } - y->nnz = nnz; - fvs_adjust_vec(y, eps); - return; -} -#endif - -/*********************************************************************** -* spx_free_nt - deallocate matrix N in sparse row-wise format -* -* This routine deallocates the memory used for arrays of the program -* object nt. */ - -void spx_free_nt(SPXLP *lp, SPXNT *nt) -{ xassert(lp == lp); - tfree(nt->ptr); - tfree(nt->len); - tfree(nt->ind); - tfree(nt->val); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxnt.h b/code/3rd_glpk/simplex/spxnt.h deleted file mode 100644 index 857917b8..00000000 --- a/code/3rd_glpk/simplex/spxnt.h +++ /dev/null @@ -1,96 +0,0 @@ -/* spxnt.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPXNT_H -#define SPXNT_H - -#include "spxlp.h" - -typedef struct SPXNT SPXNT; - -struct SPXNT -{ /* mx(n-m)-matrix N composed of non-basic columns of constraint - * matrix A, in sparse row-wise format */ - int *ptr; /* int ptr[1+m]; */ - /* ptr[0] is not used; - * ptr[i], 1 <= i <= m, is starting position of i-th row in - * arrays ind and val; note that ptr[1] is always 1; - * these starting positions are set up *once* as if they would - * correspond to rows of matrix A stored without gaps, i.e. - * ptr[i+1] - ptr[i] is the number of non-zeros in i-th (i < m) - * row of matrix A, and (nnz+1) - ptr[m] is the number of - * non-zero in m-th (last) row of matrix A, where nnz is the - * total number of non-zeros in matrix A */ - int *len; /* int len[1+m]; */ - /* len[0] is not used; - * len[i], 1 <= i <= m, is the number of non-zeros in i-th row - * of current matrix N */ - int *ind; /* int ind[1+nnz]; */ - /* column indices */ - double *val; /* double val[1+nnz]; */ - /* non-zero element values */ -}; - -#define spx_alloc_nt _glp_spx_alloc_nt -void spx_alloc_nt(SPXLP *lp, SPXNT *nt); -/* allocate matrix N in sparse row-wise format */ - -#define spx_init_nt _glp_spx_init_nt -void spx_init_nt(SPXLP *lp, SPXNT *nt); -/* initialize row pointers for matrix N */ - -#define spx_nt_add_col _glp_spx_nt_add_col -void spx_nt_add_col(SPXLP *lp, SPXNT *nt, int j, int k); -/* add column N[j] = A[k] */ - -#define spx_build_nt _glp_spx_build_nt -void spx_build_nt(SPXLP *lp, SPXNT *nt); -/* build matrix N for current basis */ - -#define spx_nt_del_col _glp_spx_nt_del_col -void spx_nt_del_col(SPXLP *lp, SPXNT *nt, int j, int k); -/* remove column N[j] = A[k] from matrix N */ - -#define spx_update_nt _glp_spx_update_nt -void spx_update_nt(SPXLP *lp, SPXNT *nt, int p, int q); -/* update matrix N for adjacent basis */ - -#define spx_nt_prod _glp_spx_nt_prod -void spx_nt_prod(SPXLP *lp, SPXNT *nt, double y[/*1+n-m*/], int ign, - double s, const double x[/*1+m*/]); -/* compute product y := y + s * N'* x */ - -#if 1 /* 31/III-2016 */ -#define spx_nt_prod_s _glp_spx_nt_prod_s -void spx_nt_prod_s(SPXLP *lp, SPXNT *nt, FVS *y, int ign, double s, - const FVS *x, double eps); -/* sparse version of spx_nt_prod */ -#endif - -#define spx_free_nt _glp_spx_free_nt -void spx_free_nt(SPXLP *lp, SPXNT *nt); -/* deallocate matrix N in sparse row-wise format */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxprim.c b/code/3rd_glpk/simplex/spxprim.c deleted file mode 100644 index e1cdfb5a..00000000 --- a/code/3rd_glpk/simplex/spxprim.c +++ /dev/null @@ -1,1860 +0,0 @@ -/* spxprim.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#if 1 /* 18/VII-2017 */ -#define SCALE_Z 1 -#endif - -#include "env.h" -#include "simplex.h" -#include "spxat.h" -#include "spxnt.h" -#include "spxchuzc.h" -#include "spxchuzr.h" -#include "spxprob.h" - -#define CHECK_ACCURACY 0 -/* (for debugging) */ - -struct csa -{ /* common storage area */ - SPXLP *lp; - /* LP problem data and its (current) basis; this LP has m rows - * and n columns */ - int dir; - /* original optimization direction: - * +1 - minimization - * -1 - maximization */ -#if SCALE_Z - double fz; - /* factor used to scale original objective */ -#endif - double *orig_c; /* double orig_c[1+n]; */ - /* copy of original objective coefficients */ - double *orig_l; /* double orig_l[1+n]; */ - /* copy of original lower bounds */ - double *orig_u; /* double orig_u[1+n]; */ - /* copy of original upper bounds */ - SPXAT *at; - /* mxn-matrix A of constraint coefficients, in sparse row-wise - * format (NULL if not used) */ - SPXNT *nt; - /* mx(n-m)-matrix N composed of non-basic columns of constraint - * matrix A, in sparse row-wise format (NULL if not used) */ - int phase; - /* search phase: - * 0 - not determined yet - * 1 - searching for primal feasible solution - * 2 - searching for optimal solution */ - double *beta; /* double beta[1+m]; */ - /* beta[i] is a primal value of basic variable xB[i] */ - int beta_st; - /* status of the vector beta: - * 0 - undefined - * 1 - just computed - * 2 - updated */ - double *d; /* double d[1+n-m]; */ - /* d[j] is a reduced cost of non-basic variable xN[j] */ - int d_st; - /* status of the vector d: - * 0 - undefined - * 1 - just computed - * 2 - updated */ - SPXSE *se; - /* projected steepest edge and Devex pricing data block (NULL if - * not used) */ - int num; - /* number of eligible non-basic variables */ - int *list; /* int list[1+n-m]; */ - /* list[1], ..., list[num] are indices j of eligible non-basic - * variables xN[j] */ - int q; - /* xN[q] is a non-basic variable chosen to enter the basis */ -#if 0 /* 11/VI-2017 */ - double *tcol; /* double tcol[1+m]; */ -#else - FVS tcol; /* FVS tcol[1:m]; */ -#endif - /* q-th (pivot) column of the simplex table */ -#if 1 /* 23/VI-2017 */ - SPXBP *bp; /* SPXBP bp[1+2*m+1]; */ - /* penalty function break points */ -#endif - int p; - /* xB[p] is a basic variable chosen to leave the basis; - * p = 0 means that no basic variable reaches its bound; - * p < 0 means that non-basic variable xN[q] reaches its opposite - * bound before any basic variable */ - int p_flag; - /* if this flag is set, the active bound of xB[p] in the adjacent - * basis should be set to the upper bound */ -#if 0 /* 11/VI-2017 */ - double *trow; /* double trow[1+n-m]; */ -#else - FVS trow; /* FVS trow[1:n-m]; */ -#endif - /* p-th (pivot) row of the simplex table */ -#if 0 /* 09/VII-2017 */ - double *work; /* double work[1+m]; */ - /* working array */ -#else - FVS work; /* FVS work[1:m]; */ - /* working vector */ -#endif - int p_stat, d_stat; - /* primal and dual solution statuses */ - /*--------------------------------------------------------------*/ - /* control parameters (see struct glp_smcp) */ - int msg_lev; - /* message level */ -#if 0 /* 23/VI-2017 */ - int harris; - /* ratio test technique: - * 0 - textbook ratio test - * 1 - Harris' two pass ratio test */ -#else - int r_test; - /* ratio test technique: - * GLP_RT_STD - textbook ratio test - * GLP_RT_HAR - Harris' two pass ratio test - * GLP_RT_FLIP - long-step ratio test (only for phase I) */ -#endif - double tol_bnd, tol_bnd1; - /* primal feasibility tolerances */ - double tol_dj, tol_dj1; - /* dual feasibility tolerances */ - double tol_piv; - /* pivot tolerance */ - int it_lim; - /* iteration limit */ - int tm_lim; - /* time limit, milliseconds */ - int out_frq; -#if 0 /* 15/VII-2017 */ - /* display output frequency, iterations */ -#else - /* display output frequency, milliseconds */ -#endif - int out_dly; - /* display output delay, milliseconds */ - /*--------------------------------------------------------------*/ - /* working parameters */ - double tm_beg; - /* time value at the beginning of the search */ - int it_beg; - /* simplex iteration count at the beginning of the search */ - int it_cnt; - /* simplex iteration count; it increases by one every time the - * basis changes (including the case when a non-basic variable - * jumps to its opposite bound) */ - int it_dpy; - /* simplex iteration count at most recent display output */ -#if 1 /* 15/VII-2017 */ - double tm_dpy; - /* time value at most recent display output */ -#endif - int inv_cnt; - /* basis factorization count since most recent display output */ -#if 1 /* 01/VII-2017 */ - int degen; - /* count of successive degenerate iterations; this count is used - * to detect stalling */ -#endif -#if 1 /* 23/VI-2017 */ - int ns_cnt, ls_cnt; - /* normal and long-step iteration counts */ -#endif -}; - -/*********************************************************************** -* set_penalty - set penalty function coefficients -* -* This routine sets up objective coefficients of the penalty function, -* which is the sum of primal infeasibilities, as follows: -* -* if beta[i] < l[k] - eps1, set c[k] = -1, -* -* if beta[i] > u[k] + eps2, set c[k] = +1, -* -* otherwise, set c[k] = 0, -* -* where beta[i] is current value of basic variable xB[i] = x[k], l[k] -* and u[k] are original bounds of x[k], and -* -* eps1 = tol + tol1 * |l[k]|, -* -* eps2 = tol + tol1 * |u[k]|. -* -* The routine returns the number of non-zero objective coefficients, -* which is the number of basic variables violating their bounds. Thus, -* if the value returned is zero, the current basis is primal feasible -* within the specified tolerances. */ - -static int set_penalty(struct csa *csa, double tol, double tol1) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - double *beta = csa->beta; - int i, k, count = 0; - double t, eps; - /* reset objective coefficients */ - for (k = 0; k <= n; k++) - c[k] = 0.0; - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - /* check lower bound */ - if ((t = l[k]) != -DBL_MAX) - { eps = tol + tol1 * (t >= 0.0 ? +t : -t); - if (beta[i] < t - eps) - { /* lower bound is violated */ - c[k] = -1.0, count++; - } - } - /* check upper bound */ - if ((t = u[k]) != +DBL_MAX) - { eps = tol + tol1 * (t >= 0.0 ? +t : -t); - if (beta[i] > t + eps) - { /* upper bound is violated */ - c[k] = +1.0, count++; - } - } - } - return count; -} - -/*********************************************************************** -* check_feas - check primal feasibility of basic solution -* -* This routine checks if the specified values of all basic variables -* beta = (beta[i]) are within their bounds. -* -* Let l[k] and u[k] be original bounds of basic variable xB[i] = x[k]. -* The actual bounds of x[k] are determined as follows: -* -* 1) if phase = 1 and c[k] < 0, x[k] violates its lower bound, so its -* actual bounds are artificial: -inf < x[k] <= l[k]; -* -* 2) if phase = 1 and c[k] > 0, x[k] violates its upper bound, so its -* actual bounds are artificial: u[k] <= x[k] < +inf; -* -* 3) in all other cases (if phase = 1 and c[k] = 0, or if phase = 2) -* actual bounds are original: l[k] <= x[k] <= u[k]. -* -* The parameters tol and tol1 are bound violation tolerances. The -* actual bounds l'[k] and u'[k] are considered as non-violated within -* the specified tolerance if -* -* l'[k] - eps1 <= beta[i] <= u'[k] + eps2, -* -* where eps1 = tol + tol1 * |l'[k]|, eps2 = tol + tol1 * |u'[k]|. -* -* The routine returns one of the following codes: -* -* 0 - solution is feasible (no actual bounds are violated); -* -* 1 - solution is infeasible, however, only artificial bounds are -* violated (this is possible only if phase = 1); -* -* 2 - solution is infeasible and at least one original bound is -* violated. */ - -static int check_feas(struct csa *csa, int phase, double tol, double - tol1) -{ SPXLP *lp = csa->lp; - int m = lp->m; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - double *beta = csa->beta; - int i, k, orig, ret = 0; - double lk, uk, eps; - xassert(phase == 1 || phase == 2); - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - /* determine actual bounds of x[k] */ - if (phase == 1 && c[k] < 0.0) - { /* -inf < x[k] <= l[k] */ - lk = -DBL_MAX, uk = l[k]; - orig = 0; /* artificial bounds */ - } - else if (phase == 1 && c[k] > 0.0) - { /* u[k] <= x[k] < +inf */ - lk = u[k], uk = +DBL_MAX; - orig = 0; /* artificial bounds */ - } - else - { /* l[k] <= x[k] <= u[k] */ - lk = l[k], uk = u[k]; - orig = 1; /* original bounds */ - } - /* check actual lower bound */ - if (lk != -DBL_MAX) - { eps = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] < lk - eps) - { /* actual lower bound is violated */ - if (orig) - { ret = 2; - break; - } - ret = 1; - } - } - /* check actual upper bound */ - if (uk != +DBL_MAX) - { eps = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] > uk + eps) - { /* actual upper bound is violated */ - if (orig) - { ret = 2; - break; - } - ret = 1; - } - } - } - return ret; -} - -/*********************************************************************** -* adjust_penalty - adjust penalty function coefficients -* -* On searching for primal feasible solution it may happen that some -* basic variable xB[i] = x[k] has non-zero objective coefficient c[k] -* indicating that xB[i] violates its lower (if c[k] < 0) or upper (if -* c[k] > 0) original bound, but due to primal degenarcy the violation -* is close to zero. -* -* This routine identifies such basic variables and sets objective -* coefficients at these variables to zero that allows avoiding zero- -* step simplex iterations. -* -* The parameters tol and tol1 are bound violation tolerances. The -* original bounds l[k] and u[k] are considered as non-violated within -* the specified tolerance if -* -* l[k] - eps1 <= beta[i] <= u[k] + eps2, -* -* where beta[i] is value of basic variable xB[i] = x[k] in the current -* basis, eps1 = tol + tol1 * |l[k]|, eps2 = tol + tol1 * |u[k]|. -* -* The routine returns the number of objective coefficients which were -* set to zero. */ - -#if 0 -static int adjust_penalty(struct csa *csa, double tol, double tol1) -{ SPXLP *lp = csa->lp; - int m = lp->m; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - double *beta = csa->beta; - int i, k, count = 0; - double t, eps; - xassert(csa->phase == 1); - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - if (c[k] < 0.0) - { /* x[k] violates its original lower bound l[k] */ - xassert((t = l[k]) != -DBL_MAX); - eps = tol + tol1 * (t >= 0.0 ? +t : -t); - if (beta[i] >= t - eps) - { /* however, violation is close to zero */ - c[k] = 0.0, count++; - } - } - else if (c[k] > 0.0) - { /* x[k] violates its original upper bound u[k] */ - xassert((t = u[k]) != +DBL_MAX); - eps = tol + tol1 * (t >= 0.0 ? +t : -t); - if (beta[i] <= t + eps) - { /* however, violation is close to zero */ - c[k] = 0.0, count++; - } - } - } - return count; -} -#else -static int adjust_penalty(struct csa *csa, int num, const int - ind[/*1+num*/], double tol, double tol1) -{ SPXLP *lp = csa->lp; - int m = lp->m; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - double *beta = csa->beta; - int i, k, t, cnt = 0; - double lk, uk, eps; - xassert(csa->phase == 1); - /* walk thru the specified list of basic variables */ - for (t = 1; t <= num; t++) - { i = ind[t]; - xassert(1 <= i && i <= m); - k = head[i]; /* x[k] = xB[i] */ - if (c[k] < 0.0) - { /* x[k] violates its original lower bound */ - lk = l[k]; - xassert(lk != -DBL_MAX); - eps = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] >= lk - eps) - { /* however, violation is close to zero */ - c[k] = 0.0, cnt++; - } - } - else if (c[k] > 0.0) - { /* x[k] violates its original upper bound */ - uk = u[k]; - xassert(uk != +DBL_MAX); - eps = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] <= uk + eps) - { /* however, violation is close to zero */ - c[k] = 0.0, cnt++; - } - } - } - return cnt; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_vec - compute maximal relative error between two vectors -* -* This routine computes and returns maximal relative error between -* n-vectors x and y: -* -* err_max = max |x[i] - y[i]| / (1 + |x[i]|). -* -* NOTE: This routine is intended only for debugginig purposes. */ - -static double err_in_vec(int n, const double x[], const double y[]) -{ int i; - double err, err_max; - err_max = 0.0; - for (i = 1; i <= n; i++) - { err = fabs(x[i] - y[i]) / (1.0 + fabs(x[i])); - if (err_max < err) - err_max = err; - } - return err_max; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_beta - compute maximal relative error in vector beta -* -* This routine computes and returns maximal relative error in vector -* of values of basic variables beta = (beta[i]). -* -* NOTE: This routine is intended only for debugginig purposes. */ - -static double err_in_beta(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - double err, *beta; - beta = talloc(1+m, double); - spx_eval_beta(lp, beta); - err = err_in_vec(m, beta, csa->beta); - tfree(beta); - return err; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_d - compute maximal relative error in vector d -* -* This routine computes and returns maximal relative error in vector -* of reduced costs of non-basic variables d = (d[j]). -* -* NOTE: This routine is intended only for debugginig purposes. */ - -static double err_in_d(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - int j; - double err, *pi, *d; - pi = talloc(1+m, double); - d = talloc(1+n-m, double); - spx_eval_pi(lp, pi); - for (j = 1; j <= n-m; j++) - d[j] = spx_eval_dj(lp, pi, j); - err = err_in_vec(n-m, d, csa->d); - tfree(pi); - tfree(d); - return err; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_gamma - compute maximal relative error in vector gamma -* -* This routine computes and returns maximal relative error in vector -* of projected steepest edge weights gamma = (gamma[j]). -* -* NOTE: This routine is intended only for debugginig purposes. */ - -static double err_in_gamma(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - SPXSE *se = csa->se; - int j; - double err, *gamma; - xassert(se != NULL); - gamma = talloc(1+n-m, double); - for (j = 1; j <= n-m; j++) - gamma[j] = spx_eval_gamma_j(lp, se, j); - err = err_in_vec(n-m, gamma, se->gamma); - tfree(gamma); - return err; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* check_accuracy - check accuracy of basic solution components -* -* This routine checks accuracy of current basic solution components. -* -* NOTE: This routine is intended only for debugginig purposes. */ - -static void check_accuracy(struct csa *csa) -{ double e_beta, e_d, e_gamma; - e_beta = err_in_beta(csa); - e_d = err_in_d(csa); - if (csa->se == NULL) - e_gamma = 0.; - else - e_gamma = err_in_gamma(csa); - xprintf("e_beta = %10.3e; e_d = %10.3e; e_gamma = %10.3e\n", - e_beta, e_d, e_gamma); - xassert(e_beta <= 1e-5 && e_d <= 1e-5 && e_gamma <= 1e-3); - return; -} -#endif - -/*********************************************************************** -* choose_pivot - choose xN[q] and xB[p] -* -* Given the list of eligible non-basic variables this routine first -* chooses non-basic variable xN[q]. This choice is always possible, -* because the list is assumed to be non-empty. Then the routine -* computes q-th column T[*,q] of the simplex table T[i,j] and chooses -* basic variable xB[p]. If the pivot T[p,q] is small in magnitude, -* the routine attempts to choose another xN[q] and xB[p] in order to -* avoid badly conditioned adjacent bases. */ - -#if 1 /* 17/III-2016 */ -#define MIN_RATIO 0.0001 - -static int choose_pivot(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *beta = csa->beta; - double *d = csa->d; - SPXSE *se = csa->se; - int *list = csa->list; -#if 0 /* 09/VII-2017 */ - double *tcol = csa->work; -#else - double *tcol = csa->work.vec; -#endif - double tol_piv = csa->tol_piv; - int try, nnn, /*i,*/ p, p_flag, q, t; - double big, /*temp,*/ best_ratio; -#if 1 /* 23/VI-2017 */ - double *c = lp->c; - int *head = lp->head; - SPXBP *bp = csa->bp; - int nbp, t_best, ret, k; - double dz_best; -#endif - xassert(csa->beta_st); - xassert(csa->d_st); -more: /* initial number of eligible non-basic variables */ - nnn = csa->num; - /* nothing has been chosen so far */ - csa->q = 0; - best_ratio = 0.0; -#if 0 /* 23/VI-2017 */ - try = 0; -#else - try = ret = 0; -#endif -try: /* choose non-basic variable xN[q] */ - xassert(nnn > 0); - try++; - if (se == NULL) - { /* Dantzig's rule */ - q = spx_chuzc_std(lp, d, nnn, list); - } - else - { /* projected steepest edge */ - q = spx_chuzc_pse(lp, se, d, nnn, list); - } - xassert(1 <= q && q <= n-m); - /* compute q-th column of the simplex table */ - spx_eval_tcol(lp, q, tcol); -#if 0 - /* big := max(1, |tcol[1]|, ..., |tcol[m]|) */ - big = 1.0; - for (i = 1; i <= m; i++) - { temp = tcol[i]; - if (temp < 0.0) - temp = - temp; - if (big < temp) - big = temp; - } -#else - /* this still puzzles me */ - big = 1.0; -#endif - /* choose basic variable xB[p] */ -#if 1 /* 23/VI-2017 */ - if (csa->phase == 1 && csa->r_test == GLP_RT_FLIP && try <= 2) - { /* long-step ratio test */ - int t, num, num1; - double slope, teta_lim; - /* determine penalty function break points */ - nbp = spx_ls_eval_bp(lp, beta, q, d[q], tcol, tol_piv, bp); - if (nbp < 2) - goto skip; - /* set initial slope */ - slope = - fabs(d[q]); - /* estimate initial teta_lim */ - teta_lim = DBL_MAX; - for (t = 1; t <= nbp; t++) - { if (teta_lim > bp[t].teta) - teta_lim = bp[t].teta; - } - xassert(teta_lim >= 0.0); - if (teta_lim < 1e-3) - teta_lim = 1e-3; - /* nothing has been chosen so far */ - t_best = 0, dz_best = 0.0, num = 0; - /* choose appropriate break point */ - while (num < nbp) - { /* select and process a new portion of break points */ - num1 = spx_ls_select_bp(lp, tcol, nbp, bp, num, &slope, - teta_lim); - for (t = num+1; t <= num1; t++) - { int i = (bp[t].i >= 0 ? bp[t].i : -bp[t].i); - xassert(0 <= i && i <= m); - if (i == 0 || fabs(tcol[i]) / big >= MIN_RATIO) - { if (dz_best > bp[t].dz) - t_best = t, dz_best = bp[t].dz; - } -#if 0 - if (i == 0) - { /* do not consider further break points beyond this - * point, where xN[q] reaches its opposite bound; - * in principle (see spx_ls_eval_bp), this break - * point should be the last one, however, due to - * round-off errors there may be other break points - * with the same teta beyond this one */ - slope = +1.0; - } -#endif - } - if (slope > 0.0) - { /* penalty function starts increasing */ - break; - } - /* penalty function continues decreasing */ - num = num1; - teta_lim += teta_lim; - } - if (dz_best == 0.0) - goto skip; - /* the choice has been made */ - xassert(1 <= t_best && t_best <= num1); - if (t_best == 1) - { /* the very first break point was chosen; it is reasonable - * to use the short-step ratio test */ - goto skip; - } - csa->q = q; - memcpy(&csa->tcol.vec[1], &tcol[1], m * sizeof(double)); - fvs_gather_vec(&csa->tcol, DBL_EPSILON); - if (bp[t_best].i == 0) - { /* xN[q] goes to its opposite bound */ - csa->p = -1; - csa->p_flag = 0; - best_ratio = 1.0; - } - else if (bp[t_best].i > 0) - { /* xB[p] leaves the basis and goes to its lower bound */ - csa->p = + bp[t_best].i; - xassert(1 <= csa->p && csa->p <= m); - csa->p_flag = 0; - best_ratio = fabs(tcol[csa->p]) / big; - } - else - { /* xB[p] leaves the basis and goes to its upper bound */ - csa->p = - bp[t_best].i; - xassert(1 <= csa->p && csa->p <= m); - csa->p_flag = 1; - best_ratio = fabs(tcol[csa->p]) / big; - } -#if 0 - xprintf("num1 = %d; t_best = %d; dz = %g\n", num1, t_best, - bp[t_best].dz); -#endif - ret = 1; - goto done; -skip: ; - } -#endif -#if 0 /* 23/VI-2017 */ - if (!csa->harris) -#else - if (csa->r_test == GLP_RT_STD) -#endif - { /* textbook ratio test */ - p = spx_chuzr_std(lp, csa->phase, beta, q, - d[q] < 0.0 ? +1. : -1., tcol, &p_flag, tol_piv, - .30 * csa->tol_bnd, .30 * csa->tol_bnd1); - } - else - { /* Harris' two-pass ratio test */ - p = spx_chuzr_harris(lp, csa->phase, beta, q, - d[q] < 0.0 ? +1. : -1., tcol, &p_flag , tol_piv, - .50 * csa->tol_bnd, .50 * csa->tol_bnd1); - } - if (p <= 0) - { /* primal unboundedness or special case */ - csa->q = q; -#if 0 /* 11/VI-2017 */ - memcpy(&csa->tcol[1], &tcol[1], m * sizeof(double)); -#else - memcpy(&csa->tcol.vec[1], &tcol[1], m * sizeof(double)); - fvs_gather_vec(&csa->tcol, DBL_EPSILON); -#endif - csa->p = p; - csa->p_flag = p_flag; - best_ratio = 1.0; - goto done; - } - /* either keep previous choice or accept new choice depending on - * which one is better */ - if (best_ratio < fabs(tcol[p]) / big) - { csa->q = q; -#if 0 /* 11/VI-2017 */ - memcpy(&csa->tcol[1], &tcol[1], m * sizeof(double)); -#else - memcpy(&csa->tcol.vec[1], &tcol[1], m * sizeof(double)); - fvs_gather_vec(&csa->tcol, DBL_EPSILON); -#endif - csa->p = p; - csa->p_flag = p_flag; - best_ratio = fabs(tcol[p]) / big; - } - /* check if the current choice is acceptable */ - if (best_ratio >= MIN_RATIO || nnn == 1 || try == 5) - goto done; - /* try to choose other xN[q] and xB[p] */ - /* find xN[q] in the list */ - for (t = 1; t <= nnn; t++) - if (list[t] == q) break; - xassert(t <= nnn); - /* move xN[q] to the end of the list */ - list[t] = list[nnn], list[nnn] = q; - /* and exclude it from consideration */ - nnn--; - /* repeat the choice */ - goto try; -done: /* the choice has been made */ -#if 1 /* FIXME: currently just to avoid badly conditioned basis */ - if (best_ratio < .001 * MIN_RATIO) - { /* looks like this helps */ - if (bfd_get_count(lp->bfd) > 0) - return -1; - /* didn't help; last chance to improve the choice */ - if (tol_piv == csa->tol_piv) - { tol_piv *= 1000.; - goto more; - } - } -#endif -#if 0 /* 23/VI-2017 */ - return 0; -#else /* FIXME */ - if (ret) - { /* invalidate dual basic solution components */ - csa->d_st = 0; - /* change penalty function coefficients at basic variables for - * all break points preceding the chosen one */ - for (t = 1; t < t_best; t++) - { int i = (bp[t].i >= 0 ? bp[t].i : -bp[t].i); - xassert(0 <= i && i <= m); - if (i == 0) - { /* xN[q] crosses its opposite bound */ - xassert(1 <= csa->q && csa->q <= n-m); - k = head[m+csa->q]; - } - else - { /* xB[i] crosses its (lower or upper) bound */ - k = head[i]; /* x[k] = xB[i] */ - } - c[k] += bp[t].dc; - xassert(c[k] == 0.0 || c[k] == +1.0 || c[k] == -1.0); - } - } - return ret; -#endif -} -#endif - -/*********************************************************************** -* play_bounds - play bounds of primal variables -* -* This routine is called after the primal values of basic variables -* beta[i] were updated and the basis was changed to the adjacent one. -* -* It is assumed that before updating all the primal values beta[i] -* were strongly feasible, so in the adjacent basis beta[i] remain -* feasible within a tolerance, i.e. if some beta[i] violates its lower -* or upper bound, the violation is insignificant. -* -* If some beta[i] violates its lower or upper bound, this routine -* changes (perturbs) the bound to remove such violation, i.e. to make -* all beta[i] strongly feasible. Otherwise, if beta[i] has a feasible -* value, this routine attempts to reduce (or remove) perturbation of -* corresponding lower/upper bound keeping strong feasibility. */ - -/* FIXME: what to do if l[k] = u[k]? */ - -/* FIXME: reduce/remove perturbation if x[k] becomes non-basic? */ - -static void play_bounds(struct csa *csa, int all) -{ SPXLP *lp = csa->lp; - int m = lp->m; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - double *orig_l = csa->orig_l; - double *orig_u = csa->orig_u; - double *beta = csa->beta; -#if 0 /* 11/VI-2017 */ - const double *tcol = csa->tcol; /* was used to update beta */ -#else - const double *tcol = csa->tcol.vec; -#endif - int i, k; - xassert(csa->phase == 1 || csa->phase == 2); - /* primal values beta = (beta[i]) should be valid */ - xassert(csa->beta_st); - /* walk thru the list of basic variables xB = (xB[i]) */ - for (i = 1; i <= m; i++) - { if (all || tcol[i] != 0.0) - { /* beta[i] has changed in the adjacent basis */ - k = head[i]; /* x[k] = xB[i] */ - if (csa->phase == 1 && c[k] < 0.0) - { /* -inf < xB[i] <= lB[i] (artificial bounds) */ - if (beta[i] < l[k] - 1e-9) - continue; - /* restore actual bounds */ - c[k] = 0.0; - csa->d_st = 0; /* since c[k] = cB[i] has changed */ - } - if (csa->phase == 1 && c[k] > 0.0) - { /* uB[i] <= xB[i] < +inf (artificial bounds) */ - if (beta[i] > u[k] + 1e-9) - continue; - /* restore actual bounds */ - c[k] = 0.0; - csa->d_st = 0; /* since c[k] = cB[i] has changed */ - } - /* lB[i] <= xB[i] <= uB[i] */ - if (csa->phase == 1) - xassert(c[k] == 0.0); - if (l[k] != -DBL_MAX) - { /* xB[i] has lower bound */ - if (beta[i] < l[k]) - { /* strong feasibility means xB[i] >= lB[i] */ -#if 0 /* 11/VI-2017 */ - l[k] = beta[i]; -#else - l[k] = beta[i] - 1e-9; -#endif - } - else if (l[k] < orig_l[k]) - { /* remove/reduce perturbation of lB[i] */ - if (beta[i] >= orig_l[k]) - l[k] = orig_l[k]; - else - l[k] = beta[i]; - } - } - if (u[k] != +DBL_MAX) - { /* xB[i] has upper bound */ - if (beta[i] > u[k]) - { /* strong feasibility means xB[i] <= uB[i] */ -#if 0 /* 11/VI-2017 */ - u[k] = beta[i]; -#else - u[k] = beta[i] + 1e-9; -#endif - } - else if (u[k] > orig_u[k]) - { /* remove/reduce perturbation of uB[i] */ - if (beta[i] <= orig_u[k]) - u[k] = orig_u[k]; - else - u[k] = beta[i]; - } - } - } - } - return; -} - -static void remove_perturb(struct csa *csa) -{ /* remove perturbation */ - SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - double *orig_l = csa->orig_l; - double *orig_u = csa->orig_u; - int j, k; - /* restore original bounds of variables */ - memcpy(l, orig_l, (1+n) * sizeof(double)); - memcpy(u, orig_u, (1+n) * sizeof(double)); - /* adjust flags of fixed non-basic variables, because in the - * perturbed problem such variables might be changed to double- - * bounded type */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == u[k]) - flag[j] = 0; - } - /* removing perturbation changes primal solution components */ - csa->phase = csa->beta_st = 0; -#if 1 - if (csa->msg_lev >= GLP_MSG_ALL) - xprintf("Removing LP perturbation [%d]...\n", - csa->it_cnt); -#endif - return; -} - -/*********************************************************************** -* sum_infeas - compute sum of primal infeasibilities -* -* This routine compute the sum of primal infeasibilities, which is the -* current penalty function value. */ - -static double sum_infeas(SPXLP *lp, const double beta[/*1+m*/]) -{ int m = lp->m; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int i, k; - double sum = 0.0; - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - if (l[k] != -DBL_MAX && beta[i] < l[k]) - sum += l[k] - beta[i]; - if (u[k] != +DBL_MAX && beta[i] > u[k]) - sum += beta[i] - u[k]; - } - return sum; -} - -/*********************************************************************** -* display - display search progress -* -* This routine displays some information about the search progress -* that includes: -* -* search phase; -* -* number of simplex iterations performed by the solver; -* -* original objective value; -* -* sum of (scaled) primal infeasibilities; -* -* number of infeasibilities (phase I) or non-optimalities (phase II); -* -* number of basic factorizations since last display output. */ - -static void display(struct csa *csa, int spec) -{ int nnn, k; - double obj, sum, *save, *save1; -#if 1 /* 15/VII-2017 */ - double tm_cur; -#endif - /* check if the display output should be skipped */ - if (csa->msg_lev < GLP_MSG_ON) goto skip; -#if 1 /* 15/VII-2017 */ - tm_cur = xtime(); -#endif - if (csa->out_dly > 0 && -#if 0 /* 15/VII-2017 */ - 1000.0 * xdifftime(xtime(), csa->tm_beg) < csa->out_dly) -#else - 1000.0 * xdifftime(tm_cur, csa->tm_beg) < csa->out_dly) -#endif - goto skip; - if (csa->it_cnt == csa->it_dpy) goto skip; -#if 0 /* 15/VII-2017 */ - if (!spec && csa->it_cnt % csa->out_frq != 0) goto skip; -#else - if (!spec && - 1000.0 * xdifftime(tm_cur, csa->tm_dpy) < csa->out_frq) - goto skip; -#endif - /* compute original objective value */ - save = csa->lp->c; - csa->lp->c = csa->orig_c; - obj = csa->dir * spx_eval_obj(csa->lp, csa->beta); - csa->lp->c = save; -#if SCALE_Z - obj *= csa->fz; -#endif - /* compute sum of (scaled) primal infeasibilities */ -#if 1 /* 01/VII-2017 */ - save = csa->lp->l; - save1 = csa->lp->u; - csa->lp->l = csa->orig_l; - csa->lp->u = csa->orig_u; -#endif - sum = sum_infeas(csa->lp, csa->beta); -#if 1 /* 01/VII-2017 */ - csa->lp->l = save; - csa->lp->u = save1; -#endif - /* compute number of infeasibilities/non-optimalities */ - switch (csa->phase) - { case 1: - nnn = 0; - for (k = 1; k <= csa->lp->n; k++) - if (csa->lp->c[k] != 0.0) nnn++; - break; - case 2: - xassert(csa->d_st); - nnn = spx_chuzc_sel(csa->lp, csa->d, csa->tol_dj, - csa->tol_dj1, NULL); - break; - default: - xassert(csa != csa); - } - /* display search progress */ - xprintf("%c%6d: obj = %17.9e inf = %11.3e (%d)", - csa->phase == 2 ? '*' : ' ', csa->it_cnt, obj, sum, nnn); - if (csa->inv_cnt) - { /* number of basis factorizations performed */ - xprintf(" %d", csa->inv_cnt); - csa->inv_cnt = 0; - } -#if 1 /* 23/VI-2017 */ - if (csa->phase == 1 && csa->r_test == GLP_RT_FLIP) - { /*xprintf(" %d,%d", csa->ns_cnt, csa->ls_cnt);*/ - if (csa->ns_cnt + csa->ls_cnt) - xprintf(" %d%%", - (100 * csa->ls_cnt) / (csa->ns_cnt + csa->ls_cnt)); - csa->ns_cnt = csa->ls_cnt = 0; - } -#endif - xprintf("\n"); - csa->it_dpy = csa->it_cnt; -#if 1 /* 15/VII-2017 */ - csa->tm_dpy = tm_cur; -#endif -skip: return; -} - -/*********************************************************************** -* spx_primal - driver to the primal simplex method -* -* This routine is a driver to the two-phase primal simplex method. -* -* On exit this routine returns one of the following codes: -* -* 0 LP instance has been successfully solved. -* -* GLP_EITLIM -* Iteration limit has been exhausted. -* -* GLP_ETMLIM -* Time limit has been exhausted. -* -* GLP_EFAIL -* The solver failed to solve LP instance. */ - -static int primal_simplex(struct csa *csa) -{ /* primal simplex method main logic routine */ - SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *c = lp->c; - int *head = lp->head; - SPXAT *at = csa->at; - SPXNT *nt = csa->nt; - double *beta = csa->beta; - double *d = csa->d; - SPXSE *se = csa->se; - int *list = csa->list; -#if 0 /* 11/VI-2017 */ - double *tcol = csa->tcol; - double *trow = csa->trow; -#endif -#if 0 /* 09/VII-2017 */ - double *pi = csa->work; - double *rho = csa->work; -#else - double *pi = csa->work.vec; - double *rho = csa->work.vec; -#endif - int msg_lev = csa->msg_lev; - double tol_bnd = csa->tol_bnd; - double tol_bnd1 = csa->tol_bnd1; - double tol_dj = csa->tol_dj; - double tol_dj1 = csa->tol_dj1; - int perturb = -1; - /* -1 = perturbation is not used, but enabled - * 0 = perturbation is not used and disabled - * +1 = perturbation is being used */ - int j, refct, ret; -loop: /* main loop starts here */ - /* compute factorization of the basis matrix */ - if (!lp->valid) - { double cond; - ret = spx_factorize(lp); - csa->inv_cnt++; - if (ret != 0) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: unable to factorize the basis matrix (%d" - ")\n", ret); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - } - /* check condition of the basis matrix */ - cond = bfd_condest(lp->bfd); - if (cond > 1.0 / DBL_EPSILON) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: basis matrix is singular to working prec" - "ision (cond = %.3g)\n", cond); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - } - if (cond > 0.001 / DBL_EPSILON) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Warning: basis matrix is ill-conditioned (cond " - "= %.3g)\n", cond); - } - /* invalidate basic solution components */ - csa->beta_st = csa->d_st = 0; - } - /* compute values of basic variables beta = (beta[i]) */ - if (!csa->beta_st) - { spx_eval_beta(lp, beta); - csa->beta_st = 1; /* just computed */ - /* determine the search phase, if not determined yet */ - if (!csa->phase) - { if (set_penalty(csa, 0.97 * tol_bnd, 0.97 * tol_bnd1)) - { /* current basic solution is primal infeasible */ - /* start to minimize the sum of infeasibilities */ - csa->phase = 1; - } - else - { /* current basic solution is primal feasible */ - /* start to minimize the original objective function */ - csa->phase = 2; - memcpy(c, csa->orig_c, (1+n) * sizeof(double)); - } - /* working objective coefficients have been changed, so - * invalidate reduced costs */ - csa->d_st = 0; - } - /* make sure that the current basic solution remains primal - * feasible (or pseudo-feasible on phase I) */ - if (perturb <= 0) - { if (check_feas(csa, csa->phase, tol_bnd, tol_bnd1)) - { /* excessive bound violations due to round-off errors */ -#if 1 /* 01/VII-2017 */ - if (perturb < 0) - { if (msg_lev >= GLP_MSG_ALL) - xprintf("Perturbing LP to avoid instability [%d].." - ".\n", csa->it_cnt); - perturb = 1; - goto loop; - } -#endif - if (msg_lev >= GLP_MSG_ERR) - xprintf("Warning: numerical instability (primal simpl" - "ex, phase %s)\n", csa->phase == 1 ? "I" : "II"); - /* restart the search */ - lp->valid = 0; - csa->phase = 0; - goto loop; - } - if (csa->phase == 1) - { int i, cnt; - for (i = 1; i <= m; i++) - csa->tcol.ind[i] = i; - cnt = adjust_penalty(csa, m, csa->tcol.ind, - 0.99 * tol_bnd, 0.99 * tol_bnd1); - if (cnt) - { /*xprintf("*** cnt = %d\n", cnt);*/ - csa->d_st = 0; - } - } - } - else - { /* FIXME */ - play_bounds(csa, 1); - } - } - /* at this point the search phase is determined */ - xassert(csa->phase == 1 || csa->phase == 2); - /* compute reduced costs of non-basic variables d = (d[j]) */ - if (!csa->d_st) - { spx_eval_pi(lp, pi); - for (j = 1; j <= n-m; j++) - d[j] = spx_eval_dj(lp, pi, j); - csa->d_st = 1; /* just computed */ - } - /* reset the reference space, if necessary */ - if (se != NULL && !se->valid) - spx_reset_refsp(lp, se), refct = 1000; - /* at this point the basis factorization and all basic solution - * components are valid */ - xassert(lp->valid && csa->beta_st && csa->d_st); -#if CHECK_ACCURACY - /* check accuracy of current basic solution components (only for - * debugging) */ - check_accuracy(csa); -#endif - /* check if the iteration limit has been exhausted */ - if (csa->it_cnt - csa->it_beg >= csa->it_lim) - { if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - if (msg_lev >= GLP_MSG_ALL) - xprintf("ITERATION LIMIT EXCEEDED; SEARCH TERMINATED\n"); - csa->p_stat = (csa->phase == 2 ? GLP_FEAS : GLP_INFEAS); - csa->d_stat = GLP_UNDEF; /* will be set below */ - ret = GLP_EITLIM; - goto fini; - } - /* check if the time limit has been exhausted */ - if (1000.0 * xdifftime(xtime(), csa->tm_beg) >= csa->tm_lim) - { if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - if (msg_lev >= GLP_MSG_ALL) - xprintf("TIME LIMIT EXCEEDED; SEARCH TERMINATED\n"); - csa->p_stat = (csa->phase == 2 ? GLP_FEAS : GLP_INFEAS); - csa->d_stat = GLP_UNDEF; /* will be set below */ - ret = GLP_ETMLIM; - goto fini; - } - /* display the search progress */ - display(csa, 0); - /* select eligible non-basic variables */ - switch (csa->phase) - { case 1: - csa->num = spx_chuzc_sel(lp, d, 1e-8, 0.0, list); - break; - case 2: - csa->num = spx_chuzc_sel(lp, d, tol_dj, tol_dj1, list); - break; - default: - xassert(csa != csa); - } - /* check for optimality */ - if (csa->num == 0) - { if (perturb > 0 && csa->phase == 2) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - /* current basis is optimal */ - display(csa, 1); - switch (csa->phase) - { case 1: - /* check for primal feasibility */ - if (!check_feas(csa, 2, tol_bnd, tol_bnd1)) - { /* feasible solution found; switch to phase II */ - memcpy(c, csa->orig_c, (1+n) * sizeof(double)); - csa->phase = 2; - csa->d_st = 0; - goto loop; - } - /* no feasible solution exists */ -#if 1 /* 09/VII-2017 */ - /* FIXME: remove perturbation */ -#endif - if (msg_lev >= GLP_MSG_ALL) - xprintf("LP HAS NO PRIMAL FEASIBLE SOLUTION\n"); - csa->p_stat = GLP_NOFEAS; - csa->d_stat = GLP_UNDEF; /* will be set below */ - ret = 0; - goto fini; - case 2: - /* optimal solution found */ - if (msg_lev >= GLP_MSG_ALL) - xprintf("OPTIMAL LP SOLUTION FOUND\n"); - csa->p_stat = csa->d_stat = GLP_FEAS; - ret = 0; - goto fini; - default: - xassert(csa != csa); - } - } - /* choose xN[q] and xB[p] */ -#if 0 /* 23/VI-2017 */ -#if 0 /* 17/III-2016 */ - choose_pivot(csa); -#else - if (choose_pivot(csa) < 0) - { lp->valid = 0; - goto loop; - } -#endif -#else - ret = choose_pivot(csa); - if (ret < 0) - { lp->valid = 0; - goto loop; - } - if (ret == 0) - csa->ns_cnt++; - else - csa->ls_cnt++; -#endif - /* check for unboundedness */ - if (csa->p == 0) - { if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - switch (csa->phase) - { case 1: - /* this should never happen */ - if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: primal simplex failed\n"); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - case 2: - /* primal unboundedness detected */ - if (msg_lev >= GLP_MSG_ALL) - xprintf("LP HAS UNBOUNDED PRIMAL SOLUTION\n"); - csa->p_stat = GLP_FEAS; - csa->d_stat = GLP_NOFEAS; - ret = 0; - goto fini; - default: - xassert(csa != csa); - } - } -#if 1 /* 01/VII-2017 */ - /* check for stalling */ - if (csa->p > 0) - { int k; - xassert(1 <= csa->p && csa->p <= m); - k = head[csa->p]; /* x[k] = xB[p] */ - if (lp->l[k] != lp->u[k]) - { if (csa->p_flag) - { /* xB[p] goes to its upper bound */ - xassert(lp->u[k] != +DBL_MAX); - if (fabs(beta[csa->p] - lp->u[k]) >= 1e-6) - { csa->degen = 0; - goto skip1; - } - } - else if (lp->l[k] == -DBL_MAX) - { /* unusual case */ - goto skip1; - } - else - { /* xB[p] goes to its lower bound */ - xassert(lp->l[k] != -DBL_MAX); - if (fabs(beta[csa->p] - lp->l[k]) >= 1e-6) - { csa->degen = 0; - goto skip1; - } - } - /* degenerate iteration has been detected */ - csa->degen++; - if (perturb < 0 && csa->degen >= 200) - { if (msg_lev >= GLP_MSG_ALL) - xprintf("Perturbing LP to avoid stalling [%d]...\n", - csa->it_cnt); - perturb = 1; - } -skip1: ; - } - } -#endif - /* update values of basic variables for adjacent basis */ -#if 0 /* 11/VI-2017 */ - spx_update_beta(lp, beta, csa->p, csa->p_flag, csa->q, tcol); -#else - spx_update_beta_s(lp, beta, csa->p, csa->p_flag, csa->q, - &csa->tcol); -#endif - csa->beta_st = 2; - /* p < 0 means that xN[q] jumps to its opposite bound */ - if (csa->p < 0) - goto skip; - /* xN[q] enters and xB[p] leaves the basis */ - /* compute p-th row of inv(B) */ - spx_eval_rho(lp, csa->p, rho); - /* compute p-th (pivot) row of the simplex table */ -#if 0 /* 11/VI-2017 */ - if (at != NULL) - spx_eval_trow1(lp, at, rho, trow); - else - spx_nt_prod(lp, nt, trow, 1, -1.0, rho); -#else - if (at != NULL) - spx_eval_trow1(lp, at, rho, csa->trow.vec); - else - spx_nt_prod(lp, nt, csa->trow.vec, 1, -1.0, rho); - fvs_gather_vec(&csa->trow, DBL_EPSILON); -#endif - /* FIXME: tcol[p] and trow[q] should be close to each other */ -#if 0 /* 26/V-2017 by cmatraki */ - xassert(trow[csa->q] != 0.0); -#else - if (csa->trow.vec[csa->q] == 0.0) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: trow[q] = 0.0\n"); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - } -#endif - /* update reduced costs of non-basic variables for adjacent - * basis */ -#if 1 /* 23/VI-2017 */ - /* dual solution may be invalidated due to long step */ - if (csa->d_st) -#endif -#if 0 /* 11/VI-2017 */ - if (spx_update_d(lp, d, csa->p, csa->q, trow, tcol) <= 1e-9) -#else - if (spx_update_d_s(lp, d, csa->p, csa->q, &csa->trow, &csa->tcol) - <= 1e-9) -#endif - { /* successful updating */ - csa->d_st = 2; - if (csa->phase == 1) - { /* adjust reduced cost of xN[q] in adjacent basis, since - * its penalty coefficient changes (see below) */ - d[csa->q] -= c[head[csa->p]]; - } - } - else - { /* new reduced costs are inaccurate */ - csa->d_st = 0; - } - if (csa->phase == 1) - { /* xB[p] leaves the basis replacing xN[q], so set its penalty - * coefficient to zero */ - c[head[csa->p]] = 0.0; - } - /* update steepest edge weights for adjacent basis, if used */ - if (se != NULL) - { if (refct > 0) -#if 0 /* 11/VI-2017 */ - { if (spx_update_gamma(lp, se, csa->p, csa->q, trow, tcol) - <= 1e-3) -#else /* FIXME: spx_update_gamma_s */ - { if (spx_update_gamma(lp, se, csa->p, csa->q, csa->trow.vec, - csa->tcol.vec) <= 1e-3) -#endif - { /* successful updating */ - refct--; - } - else - { /* new weights are inaccurate; reset reference space */ - se->valid = 0; - } - } - else - { /* too many updates; reset reference space */ - se->valid = 0; - } - } - /* update matrix N for adjacent basis, if used */ - if (nt != NULL) - spx_update_nt(lp, nt, csa->p, csa->q); -skip: /* change current basis header to adjacent one */ - spx_change_basis(lp, csa->p, csa->p_flag, csa->q); - /* and update factorization of the basis matrix */ - if (csa->p > 0) - spx_update_invb(lp, csa->p, head[csa->p]); -#if 1 - if (perturb <= 0) - { if (csa->phase == 1) - { int cnt; - /* adjust penalty function coefficients */ - cnt = adjust_penalty(csa, csa->tcol.nnz, csa->tcol.ind, - 0.99 * tol_bnd, 0.99 * tol_bnd1); - if (cnt) - { /* some coefficients were changed, so invalidate reduced - * costs of non-basic variables */ - /*xprintf("... cnt = %d\n", cnt);*/ - csa->d_st = 0; - } - } - } - else - { /* FIXME */ - play_bounds(csa, 0); - } -#endif - /* simplex iteration complete */ - csa->it_cnt++; - goto loop; -fini: /* restore original objective function */ - memcpy(c, csa->orig_c, (1+n) * sizeof(double)); - /* compute reduced costs of non-basic variables and determine - * solution dual status, if necessary */ - if (csa->p_stat != GLP_UNDEF && csa->d_stat == GLP_UNDEF) - { xassert(ret != GLP_EFAIL); - spx_eval_pi(lp, pi); - for (j = 1; j <= n-m; j++) - d[j] = spx_eval_dj(lp, pi, j); - csa->num = spx_chuzc_sel(lp, d, tol_dj, tol_dj1, NULL); - csa->d_stat = (csa->num == 0 ? GLP_FEAS : GLP_INFEAS); - } - return ret; -} - -int spx_primal(glp_prob *P, const glp_smcp *parm) -{ /* driver to the primal simplex method */ - struct csa csa_, *csa = &csa_; - SPXLP lp; - SPXAT at; - SPXNT nt; - SPXSE se; - int ret, *map, *daeh; -#if SCALE_Z - int i, j, k; -#endif - /* build working LP and its initial basis */ - memset(csa, 0, sizeof(struct csa)); - csa->lp = &lp; - spx_init_lp(csa->lp, P, parm->excl); - spx_alloc_lp(csa->lp); - map = talloc(1+P->m+P->n, int); - spx_build_lp(csa->lp, P, parm->excl, parm->shift, map); - spx_build_basis(csa->lp, P, map); - switch (P->dir) - { case GLP_MIN: - csa->dir = +1; - break; - case GLP_MAX: - csa->dir = -1; - break; - default: - xassert(P != P); - } -#if SCALE_Z - csa->fz = 0.0; - for (k = 1; k <= csa->lp->n; k++) - { double t = fabs(csa->lp->c[k]); - if (csa->fz < t) - csa->fz = t; - } - if (csa->fz <= 1000.0) - csa->fz = 1.0; - else - csa->fz /= 1000.0; - /*xprintf("csa->fz = %g\n", csa->fz);*/ - for (k = 0; k <= csa->lp->n; k++) - csa->lp->c[k] /= csa->fz; -#endif - csa->orig_c = talloc(1+csa->lp->n, double); - memcpy(csa->orig_c, csa->lp->c, (1+csa->lp->n) * sizeof(double)); -#if 1 /*PERTURB*/ - csa->orig_l = talloc(1+csa->lp->n, double); - memcpy(csa->orig_l, csa->lp->l, (1+csa->lp->n) * sizeof(double)); - csa->orig_u = talloc(1+csa->lp->n, double); - memcpy(csa->orig_u, csa->lp->u, (1+csa->lp->n) * sizeof(double)); -#else - csa->orig_l = csa->orig_u = NULL; -#endif - switch (parm->aorn) - { case GLP_USE_AT: - /* build matrix A in row-wise format */ - csa->at = &at; - csa->nt = NULL; - spx_alloc_at(csa->lp, csa->at); - spx_build_at(csa->lp, csa->at); - break; - case GLP_USE_NT: - /* build matrix N in row-wise format for initial basis */ - csa->at = NULL; - csa->nt = &nt; - spx_alloc_nt(csa->lp, csa->nt); - spx_init_nt(csa->lp, csa->nt); - spx_build_nt(csa->lp, csa->nt); - break; - default: - xassert(parm != parm); - } - /* allocate and initialize working components */ - csa->phase = 0; - csa->beta = talloc(1+csa->lp->m, double); - csa->beta_st = 0; - csa->d = talloc(1+csa->lp->n-csa->lp->m, double); - csa->d_st = 0; - switch (parm->pricing) - { case GLP_PT_STD: - csa->se = NULL; - break; - case GLP_PT_PSE: - csa->se = &se; - spx_alloc_se(csa->lp, csa->se); - break; - default: - xassert(parm != parm); - } - csa->list = talloc(1+csa->lp->n-csa->lp->m, int); -#if 0 /* 11/VI-2017 */ - csa->tcol = talloc(1+csa->lp->m, double); - csa->trow = talloc(1+csa->lp->n-csa->lp->m, double); -#else - fvs_alloc_vec(&csa->tcol, csa->lp->m); - fvs_alloc_vec(&csa->trow, csa->lp->n-csa->lp->m); -#endif -#if 1 /* 23/VI-2017 */ - csa->bp = NULL; -#endif -#if 0 /* 09/VII-2017 */ - csa->work = talloc(1+csa->lp->m, double); -#else - fvs_alloc_vec(&csa->work, csa->lp->m); -#endif - /* initialize control parameters */ - csa->msg_lev = parm->msg_lev; -#if 0 /* 23/VI-2017 */ - switch (parm->r_test) - { case GLP_RT_STD: - csa->harris = 0; - break; - case GLP_RT_HAR: -#if 1 /* 16/III-2016 */ - case GLP_RT_FLIP: - /* FIXME */ - /* currently for primal simplex GLP_RT_FLIP is equivalent - * to GLP_RT_HAR */ -#endif - csa->harris = 1; - break; - default: - xassert(parm != parm); - } -#else - switch (parm->r_test) - { case GLP_RT_STD: - case GLP_RT_HAR: - break; - case GLP_RT_FLIP: - csa->bp = talloc(1+2*csa->lp->m+1, SPXBP); - break; - default: - xassert(parm != parm); - } - csa->r_test = parm->r_test; -#endif - csa->tol_bnd = parm->tol_bnd; - csa->tol_bnd1 = .001 * parm->tol_bnd; - csa->tol_dj = parm->tol_dj; - csa->tol_dj1 = .001 * parm->tol_dj; - csa->tol_piv = parm->tol_piv; - csa->it_lim = parm->it_lim; - csa->tm_lim = parm->tm_lim; - csa->out_frq = parm->out_frq; - csa->out_dly = parm->out_dly; - /* initialize working parameters */ - csa->tm_beg = xtime(); - csa->it_beg = csa->it_cnt = P->it_cnt; - csa->it_dpy = -1; -#if 1 /* 15/VII-2017 */ - csa->tm_dpy = 0.0; -#endif - csa->inv_cnt = 0; -#if 1 /* 01/VII-2017 */ - csa->degen = 0; -#endif -#if 1 /* 23/VI-2017 */ - csa->ns_cnt = csa->ls_cnt = 0; -#endif - /* try to solve working LP */ - ret = primal_simplex(csa); - /* return basis factorization back to problem object */ - P->valid = csa->lp->valid; - P->bfd = csa->lp->bfd; - /* set solution status */ - P->pbs_stat = csa->p_stat; - P->dbs_stat = csa->d_stat; - /* if the solver failed, do not store basis header and basic - * solution components to problem object */ - if (ret == GLP_EFAIL) - goto skip; - /* convert working LP basis to original LP basis and store it to - * problem object */ - daeh = talloc(1+csa->lp->n, int); - spx_store_basis(csa->lp, P, map, daeh); - /* compute simplex multipliers for final basic solution found by - * the solver */ -#if 0 /* 09/VII-2017 */ - spx_eval_pi(csa->lp, csa->work); -#else - spx_eval_pi(csa->lp, csa->work.vec); -#endif - /* convert working LP solution to original LP solution and store - * it into the problem object */ -#if SCALE_Z - for (i = 1; i <= csa->lp->m; i++) - csa->work.vec[i] *= csa->fz; - for (j = 1; j <= csa->lp->n-csa->lp->m; j++) - csa->d[j] *= csa->fz; -#endif -#if 0 /* 09/VII-2017 */ - spx_store_sol(csa->lp, P, SHIFT, map, daeh, csa->beta, csa->work, - csa->d); -#else - spx_store_sol(csa->lp, P, parm->shift, map, daeh, csa->beta, - csa->work.vec, csa->d); -#endif - tfree(daeh); - /* save simplex iteration count */ - P->it_cnt = csa->it_cnt; - /* report auxiliary/structural variable causing unboundedness */ - P->some = 0; - if (csa->p_stat == GLP_FEAS && csa->d_stat == GLP_NOFEAS) - { int k, kk; - /* xN[q] = x[k] causes unboundedness */ - xassert(1 <= csa->q && csa->q <= csa->lp->n - csa->lp->m); - k = csa->lp->head[csa->lp->m + csa->q]; - xassert(1 <= k && k <= csa->lp->n); - /* convert to number of original variable */ - for (kk = 1; kk <= P->m + P->n; kk++) - { if (abs(map[kk]) == k) - { P->some = kk; - break; - } - } - xassert(P->some != 0); - } -skip: /* deallocate working objects and arrays */ - spx_free_lp(csa->lp); - tfree(map); - tfree(csa->orig_c); -#if 1 /*PERTURB*/ - tfree(csa->orig_l); - tfree(csa->orig_u); -#endif - if (csa->at != NULL) - spx_free_at(csa->lp, csa->at); - if (csa->nt != NULL) - spx_free_nt(csa->lp, csa->nt); - tfree(csa->beta); - tfree(csa->d); - if (csa->se != NULL) - spx_free_se(csa->lp, csa->se); - tfree(csa->list); -#if 0 /* 11/VI-2017 */ - tfree(csa->tcol); - tfree(csa->trow); -#else - fvs_free_vec(&csa->tcol); - fvs_free_vec(&csa->trow); -#endif -#if 1 /* 23/VI-2017 */ - if (csa->bp != NULL) - tfree(csa->bp); -#endif -#if 0 /* 09/VII-2017 */ - tfree(csa->work); -#else - fvs_free_vec(&csa->work); -#endif - /* return to calling program */ - return ret; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxprob.c b/code/3rd_glpk/simplex/spxprob.c deleted file mode 100644 index 4bebe2e7..00000000 --- a/code/3rd_glpk/simplex/spxprob.c +++ /dev/null @@ -1,679 +0,0 @@ -/* spxprob.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spxprob.h" - -/*********************************************************************** -* spx_init_lp - initialize working LP object -* -* This routine determines the number of equality constraints m, the -* number of variables n, and the number of non-zero elements nnz in -* the constraint matrix for the working LP, which corresponds to the -* original LP, and stores these dimensions to the working LP object. -* (The working LP object should be allocated by the calling routine.) -* -* If the flag excl is set, the routine assumes that non-basic fixed -* variables will be excluded from the working LP. */ - -void spx_init_lp(SPXLP *lp, glp_prob *P, int excl) -{ int i, j, m, n, nnz; - m = P->m; - xassert(m > 0); - n = 0; - nnz = P->nnz; - xassert(P->valid); - /* scan rows of original LP */ - for (i = 1; i <= m; i++) - { GLPROW *row = P->row[i]; - if (excl && row->stat == GLP_NS) - { /* skip non-basic fixed auxiliary variable */ - /* nop */ - } - else - { /* include auxiliary variable in working LP */ - n++; - nnz++; /* unity column */ - } - } - /* scan columns of original LP */ - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - if (excl && col->stat == GLP_NS) - { /* skip non-basic fixed structural variable */ - GLPAIJ *aij; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - nnz--; - } - else - { /* include structural variable in working LP */ - n++; - } - } - /* initialize working LP data block */ - memset(lp, 0, sizeof(SPXLP)); - lp->m = m; - xassert(n > 0); - lp->n = n; - lp->nnz = nnz; - return; -} - -/*********************************************************************** -* spx_alloc_lp - allocate working LP arrays -* -* This routine allocates the memory for all arrays in the working LP -* object. */ - -void spx_alloc_lp(SPXLP *lp) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - lp->A_ptr = talloc(1+n+1, int); - lp->A_ind = talloc(1+nnz, int); - lp->A_val = talloc(1+nnz, double); - lp->b = talloc(1+m, double); - lp->c = talloc(1+n, double); - lp->l = talloc(1+n, double); - lp->u = talloc(1+n, double); - lp->head = talloc(1+n, int); - lp->flag = talloc(1+n-m, char); - return; -} - -/*********************************************************************** -* spx_build_lp - convert original LP to working LP -* -* This routine converts components (except the current basis) of the -* original LP to components of the working LP and perform scaling of -* these components. Also, if the original LP is maximization, the -* routine changes the signs of the objective coefficients and constant -* term to opposite ones. -* -* If the flag excl is set, original non-basic fixed variables are -* *not* included in the working LP. Otherwise, all (auxiliary and -* structural) original variables are included in the working LP. Note -* that this flag should have the same value as it has in a call to the -* routine spx_init_lp. -* -* If the flag shift is set, the routine shift bounds of variables -* included in the working LP to make at least one bound to be zero. -* If a variable has both lower and upper bounds, the bound having -* smaller magnitude is shifted to zero. -* -* On exit the routine stores information about correspondence between -* numbers of variables in the original and working LPs to the array -* map, which should have 1+P->m+P->n locations (location [0] is not -* used), where P->m is the numbers of rows and P->n is the number of -* columns in the original LP: -* -* map[i] = +k, 1 <= i <= P->m, means that i-th auxiliary variable of -* the original LP corresponds to variable x[k] of the working LP; -* -* map[i] = -k, 1 <= i <= P->m, means that i-th auxiliary variable of -* the original LP corresponds to variable x[k] of the working LP, and -* the upper bound of that variable was shifted to zero; -* -* map[i] = 0, 1 <= i <= P->m, means that i-th auxiliary variable of -* the original LP was excluded from the working LP; -* -* map[P->m+j], 1 <= j <= P->n, has the same sense as above, however, -* for j-th structural variable of the original LP. */ - -void spx_build_lp(SPXLP *lp, glp_prob *P, int excl, int shift, - int map[/*1+P->m+P->n*/]) -{ int m = lp->m; - int n = lp->n; - int nnz = lp->nnz; - int *A_ptr = lp->A_ptr; - int *A_ind = lp->A_ind; - double *A_val = lp->A_val; - double *b = lp->b; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int i, j, k, kk, ptr, end; - double dir, delta; - /* working LP is always minimization */ - switch (P->dir) - { case GLP_MIN: - dir = +1.0; - break; - case GLP_MAX: - dir = -1.0; - break; - default: - xassert(P != P); - } - /* initialize constant term of the objective */ - c[0] = dir * P->c0; - k = 0; /* number of variable in working LP */ - ptr = 1; /* current available position in A_ind/A_val */ - /* process rows of original LP */ - xassert(P->m == m); - for (i = 1; i <= m; i++) - { GLPROW *row = P->row[i]; - if (excl && row->stat == GLP_NS) - { /* i-th auxiliary variable is non-basic and fixed */ - /* substitute its scaled value in working LP */ - xassert(row->type == GLP_FX); - map[i] = 0; - b[i] = - row->lb * row->rii; - } - else - { /* include i-th auxiliary variable in working LP */ - map[i] = ++k; - /* setup k-th column of working constraint matrix which is - * i-th column of unity matrix */ - A_ptr[k] = ptr; - A_ind[ptr] = i; - A_val[ptr] = 1.0; - ptr++; - /* initialize right-hand side of i-th equality constraint - * and setup zero objective coefficient at variable x[k] */ - b[i] = c[k] = 0.0; - /* setup scaled bounds of variable x[k] */ - switch (row->type) - { case GLP_FR: - l[k] = -DBL_MAX, u[k] = +DBL_MAX; - break; - case GLP_LO: - l[k] = row->lb * row->rii, u[k] = +DBL_MAX; - break; - case GLP_UP: - l[k] = -DBL_MAX, u[k] = row->ub * row->rii; - break; - case GLP_DB: - l[k] = row->lb * row->rii, u[k] = row->ub * row->rii; - xassert(l[k] != u[k]); - break; - case GLP_FX: - l[k] = u[k] = row->lb * row->rii; - break; - default: - xassert(row != row); - } - } - } - /* process columns of original LP */ - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - GLPAIJ *aij; - if (excl && col->stat == GLP_NS) - { /* j-th structural variable is non-basic and fixed */ - /* substitute its scaled value in working LP */ - xassert(col->type == GLP_FX); - map[m+j] = 0; - if (col->lb != 0.0) - { /* (note that sjj scale factor is cancelled) */ - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - b[aij->row->i] += - (aij->row->rii * aij->val) * col->lb; - c[0] += (dir * col->coef) * col->lb; - } - } - else - { /* include j-th structural variable in working LP */ - map[m+j] = ++k; - /* setup k-th column of working constraint matrix which is - * scaled j-th column of original constraint matrix (-A) */ - A_ptr[k] = ptr; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - { A_ind[ptr] = aij->row->i; - A_val[ptr] = - aij->row->rii * aij->val * col->sjj; - ptr++; - } - /* setup scaled objective coefficient at variable x[k] */ - c[k] = dir * col->coef * col->sjj; - /* setup scaled bounds of variable x[k] */ - switch (col->type) - { case GLP_FR: - l[k] = -DBL_MAX, u[k] = +DBL_MAX; - break; - case GLP_LO: - l[k] = col->lb / col->sjj, u[k] = +DBL_MAX; - break; - case GLP_UP: - l[k] = -DBL_MAX, u[k] = col->ub / col->sjj; - break; - case GLP_DB: - l[k] = col->lb / col->sjj, u[k] = col->ub / col->sjj; - xassert(l[k] != u[k]); - break; - case GLP_FX: - l[k] = u[k] = col->lb / col->sjj; - break; - default: - xassert(col != col); - } - } - } - xassert(k == n); - xassert(ptr == nnz+1); - A_ptr[n+1] = ptr; - /* shift bounds of all variables of working LP (optionally) */ - if (shift) - { for (kk = 1; kk <= m+P->n; kk++) - { k = map[kk]; - if (k == 0) - { /* corresponding original variable was excluded */ - continue; - } - /* shift bounds of variable x[k] */ - if (l[k] == -DBL_MAX && u[k] == +DBL_MAX) - { /* x[k] is unbounded variable */ - delta = 0.0; - } - else if (l[k] != -DBL_MAX && u[k] == +DBL_MAX) - { /* shift lower bound to zero */ - delta = l[k]; - l[k] = 0.0; - } - else if (l[k] == -DBL_MAX && u[k] != +DBL_MAX) - { /* shift upper bound to zero */ - map[kk] = -k; - delta = u[k]; - u[k] = 0.0; - } - else if (l[k] != u[k]) - { /* x[k] is double bounded variable */ - if (fabs(l[k]) <= fabs(u[k])) - { /* shift lower bound to zero */ - delta = l[k]; - l[k] = 0.0, u[k] -= delta; - } - else - { /* shift upper bound to zero */ - map[kk] = -k; - delta = u[k]; - l[k] -= delta, u[k] = 0.0; - } - xassert(l[k] != u[k]); - } - else - { /* shift fixed value to zero */ - delta = l[k]; - l[k] = u[k] = 0.0; - } - /* substitute x[k] = x'[k] + delta into all constraints - * and the objective function of working LP */ - if (delta != 0.0) - { ptr = A_ptr[k]; - end = A_ptr[k+1]; - for (; ptr < end; ptr++) - b[A_ind[ptr]] -= A_val[ptr] * delta; - c[0] += c[k] * delta; - } - } - } - return; -} - -/*********************************************************************** -* spx_build_basis - convert original LP basis to working LP basis -* -* This routine converts the current basis of the original LP to -* corresponding initial basis of the working LP, and moves the basis -* factorization driver from the original LP object to the working LP -* object. -* -* The array map should contain information provided by the routine -* spx_build_lp. */ - -void spx_build_basis(SPXLP *lp, glp_prob *P, const int map[]) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *flag = lp->flag; - int i, j, k, ii, jj; - /* original basis factorization should be valid that guarantees - * the basis is correct */ - xassert(P->m == m); - xassert(P->valid); - /* initialize basis header for working LP */ - memset(&head[1], 0, m * sizeof(int)); - jj = 0; - /* scan rows of original LP */ - xassert(P->m == m); - for (i = 1; i <= m; i++) - { GLPROW *row = P->row[i]; - /* determine ordinal number of x[k] in working LP */ - if ((k = map[i]) < 0) - k = -k; - if (k == 0) - { /* corresponding original variable was excluded */ - continue; - } - xassert(1 <= k && k <= n); - if (row->stat == GLP_BS) - { /* x[k] is basic variable xB[ii] */ - ii = row->bind; - xassert(1 <= ii && ii <= m); - xassert(head[ii] == 0); - head[ii] = k; - } - else - { /* x[k] is non-basic variable xN[jj] */ - jj++; - head[m+jj] = k; - flag[jj] = (row->stat == GLP_NU); - } - } - /* scan columns of original LP */ - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - /* determine ordinal number of x[k] in working LP */ - if ((k = map[m+j]) < 0) - k = -k; - if (k == 0) - { /* corresponding original variable was excluded */ - continue; - } - xassert(1 <= k && k <= n); - if (col->stat == GLP_BS) - { /* x[k] is basic variable xB[ii] */ - ii = col->bind; - xassert(1 <= ii && ii <= m); - xassert(head[ii] == 0); - head[ii] = k; - } - else - { /* x[k] is non-basic variable xN[jj] */ - jj++; - head[m+jj] = k; - flag[jj] = (col->stat == GLP_NU); - } - } - xassert(m+jj == n); - /* acquire basis factorization */ - lp->valid = 1; - lp->bfd = P->bfd; - P->valid = 0; - P->bfd = NULL; - return; -} - -/*********************************************************************** -* spx_store_basis - convert working LP basis to original LP basis -* -* This routine converts the current working LP basis to corresponding -* original LP basis. This operations includes determining and setting -* statuses of all rows (auxiliary variables) and columns (structural -* variables), and building the basis header. -* -* The array map should contain information provided by the routine -* spx_build_lp. -* -* On exit the routine fills the array daeh. This array should have -* 1+lp->n locations (location [0] is not used) and contain the inverse -* of the working basis header lp->head, i.e. head[k'] = k means that -* daeh[k] = k'. */ - -void spx_store_basis(SPXLP *lp, glp_prob *P, const int map[], - int daeh[/*1+n*/]) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *flag = lp->flag; - int i, j, k, kk; - /* determine inverse of working basis header */ - for (kk = 1; kk <= n; kk++) - daeh[head[kk]] = kk; - /* set row statuses */ - xassert(P->m == m); - for (i = 1; i <= m; i++) - { GLPROW *row = P->row[i]; - if ((k = map[i]) < 0) - k = -k; - if (k == 0) - { /* non-basic fixed auxiliary variable was excluded */ - xassert(row->type == GLP_FX); - row->stat = GLP_NS; - row->bind = 0; - } - else - { /* auxiliary variable corresponds to variable x[k] */ - kk = daeh[k]; - if (kk <= m) - { /* x[k] = xB[kk] */ - P->head[kk] = i; - row->stat = GLP_BS; - row->bind = kk; - } - else - { /* x[k] = xN[kk-m] */ - switch (row->type) - { case GLP_FR: - row->stat = GLP_NF; - break; - case GLP_LO: - row->stat = GLP_NL; - break; - case GLP_UP: - row->stat = GLP_NU; - break; - case GLP_DB: - row->stat = (flag[kk-m] ? GLP_NU : GLP_NL); - break; - case GLP_FX: - row->stat = GLP_NS; - break; - default: - xassert(row != row); - } - row->bind = 0; - } - } - } - /* set column statuses */ - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - if ((k = map[m+j]) < 0) - k = -k; - if (k == 0) - { /* non-basic fixed structural variable was excluded */ - xassert(col->type == GLP_FX); - col->stat = GLP_NS; - col->bind = 0; - } - else - { /* structural variable corresponds to variable x[k] */ - kk = daeh[k]; - if (kk <= m) - { /* x[k] = xB[kk] */ - P->head[kk] = m+j; - col->stat = GLP_BS; - col->bind = kk; - } - else - { /* x[k] = xN[kk-m] */ - switch (col->type) - { case GLP_FR: - col->stat = GLP_NF; - break; - case GLP_LO: - col->stat = GLP_NL; - break; - case GLP_UP: - col->stat = GLP_NU; - break; - case GLP_DB: - col->stat = (flag[kk-m] ? GLP_NU : GLP_NL); - break; - case GLP_FX: - col->stat = GLP_NS; - break; - default: - xassert(col != col); - } - col->bind = 0; - } - } - } - return; -} - -/*********************************************************************** -* spx_store_sol - convert working LP solution to original LP solution -* -* This routine converts the current basic solution of the working LP -* (values of basic variables, simplex multipliers, reduced costs of -* non-basic variables) to corresponding basic solution of the original -* LP (values and reduced costs of auxiliary and structural variables). -* This conversion includes unscaling all basic solution components, -* computing reduced costs of excluded non-basic variables, recovering -* unshifted values of basic variables, changing the signs of reduced -* costs (if the original LP is maximization), and computing the value -* of the objective function. -* -* The flag shift should have the same value as it has in a call to the -* routine spx_build_lp. -* -* The array map should contain information provided by the routine -* spx_build_lp. -* -* The array daeh should contain information provided by the routine -* spx_store_basis. -* -* The arrays beta, pi, and d should contain basic solution components -* for the working LP: -* -* array locations beta[1], ..., beta[m] should contain values of basic -* variables beta = (beta[i]); -* -* array locations pi[1], ..., pi[m] should contain simplex multipliers -* pi = (pi[i]); -* -* array locations d[1], ..., d[n-m] should contain reduced costs of -* non-basic variables d = (d[j]). */ - -void spx_store_sol(SPXLP *lp, glp_prob *P, int shift, - const int map[], const int daeh[], const double beta[], - const double pi[], const double d[]) -{ int m = lp->m; - char *flag = lp->flag; - int i, j, k, kk; - double dir; - /* working LP is always minimization */ - switch (P->dir) - { case GLP_MIN: - dir = +1.0; - break; - case GLP_MAX: - dir = -1.0; - break; - default: - xassert(P != P); - } - /* compute row solution components */ - xassert(P->m == m); - for (i = 1; i <= m; i++) - { GLPROW *row = P->row[i]; - if ((k = map[i]) < 0) - k = -k; - if (k == 0) - { /* non-basic fixed auxiliary variable was excluded */ - xassert(row->type == GLP_FX); - row->prim = row->lb; - /* compute reduced cost d[k] = c[k] - A'[k] * pi as if x[k] - * would be non-basic in working LP */ - row->dual = - dir * pi[i] * row->rii; - } - else - { /* auxiliary variable corresponds to variable x[k] */ - kk = daeh[k]; - if (kk <= m) - { /* x[k] = xB[kk] */ - row->prim = beta[kk] / row->rii; - if (shift) - row->prim += (map[i] < 0 ? row->ub : row->lb); - row->dual = 0.0; - } - else - { /* x[k] = xN[kk-m] */ - row->prim = (flag[kk-m] ? row->ub : row->lb); - row->dual = (dir * d[kk-m]) * row->rii; - } - } - } - /* compute column solution components and objective value */ - P->obj_val = P->c0; - for (j = 1; j <= P->n; j++) - { GLPCOL *col = P->col[j]; - if ((k = map[m+j]) < 0) - k = -k; - if (k == 0) - { /* non-basic fixed structural variable was excluded */ - GLPAIJ *aij; - double dk; - xassert(col->type == GLP_FX); - col->prim = col->lb; - /* compute reduced cost d[k] = c[k] - A'[k] * pi as if x[k] - * would be non-basic in working LP */ - /* (note that sjj scale factor is cancelled) */ - dk = dir * col->coef; - for (aij = col->ptr; aij != NULL; aij = aij->c_next) - dk += (aij->row->rii * aij->val) * pi[aij->row->i]; - col->dual = dir * dk; - } - else - { /* structural variable corresponds to variable x[k] */ - kk = daeh[k]; - if (kk <= m) - { /* x[k] = xB[kk] */ - col->prim = beta[kk] * col->sjj; - if (shift) - col->prim += (map[m+j] < 0 ? col->ub : col->lb); - col->dual = 0.0; - } - else - { /* x[k] = xN[kk-m] */ - col->prim = (flag[kk-m] ? col->ub : col->lb); - col->dual = (dir * d[kk-m]) / col->sjj; - } - } - P->obj_val += col->coef * col->prim; - } - return; -} - -/*********************************************************************** -* spx_free_lp - deallocate working LP arrays -* -* This routine deallocates the memory used for arrays of the working -* LP object. */ - -void spx_free_lp(SPXLP *lp) -{ tfree(lp->A_ptr); - tfree(lp->A_ind); - tfree(lp->A_val); - tfree(lp->b); - tfree(lp->c); - tfree(lp->l); - tfree(lp->u); - tfree(lp->head); - tfree(lp->flag); - return; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spxprob.h b/code/3rd_glpk/simplex/spxprob.h deleted file mode 100644 index b7d87fa7..00000000 --- a/code/3rd_glpk/simplex/spxprob.h +++ /dev/null @@ -1,64 +0,0 @@ -/* spxprob.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPXPROB_H -#define SPXPROB_H - -#include "prob.h" -#include "spxlp.h" - -#define spx_init_lp _glp_spx_init_lp -void spx_init_lp(SPXLP *lp, glp_prob *P, int excl); -/* initialize working LP object */ - -#define spx_alloc_lp _glp_spx_alloc_lp -void spx_alloc_lp(SPXLP *lp); -/* allocate working LP arrays */ - -#define spx_build_lp _glp_spx_build_lp -void spx_build_lp(SPXLP *lp, glp_prob *P, int excl, int shift, - int map[/*1+P->m+P->n*/]); -/* convert original LP to working LP */ - -#define spx_build_basis _glp_spx_build_basis -void spx_build_basis(SPXLP *lp, glp_prob *P, const int map[]); -/* convert original LP basis to working LP basis */ - -#define spx_store_basis _glp_spx_store_basis -void spx_store_basis(SPXLP *lp, glp_prob *P, const int map[], - int daeh[/*1+n*/]); -/* convert working LP basis to original LP basis */ - -#define spx_store_sol _glp_spx_store_sol -void spx_store_sol(SPXLP *lp, glp_prob *P, int shift, - const int map[], const int daeh[], const double beta[], - const double pi[], const double d[]); -/* convert working LP solution to original LP solution */ - -#define spx_free_lp _glp_spx_free_lp -void spx_free_lp(SPXLP *lp); -/* deallocate working LP arrays */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spychuzc.c b/code/3rd_glpk/simplex/spychuzc.c deleted file mode 100644 index b9221298..00000000 --- a/code/3rd_glpk/simplex/spychuzc.c +++ /dev/null @@ -1,567 +0,0 @@ -/* spychuzc.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2018 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spychuzc.h" - -/*********************************************************************** -* spy_chuzc_std - choose non-basic variable (dual textbook ratio test) -* -* This routine implements an improved dual textbook ratio test to -* choose non-basic variable xN[q]. -* -* Current reduced costs of non-basic variables should be placed in the -* array locations d[1], ..., d[n-m]. Note that d[j] is a value of dual -* basic variable lambdaN[j] in the current basis. -* -#if 0 (* 14/III-2016 *) -* The parameter s specifies the sign of bound violation for basic -* variable xB[p] chosen: s = +1.0 means that xB[p] violates its lower -* bound, so dual non-basic variable lambdaB[p] = lambda^+B[p] -* increases, and s = -1.0 means that xB[p] violates its upper bound, -* so dual non-basic variable lambdaB[p] = lambda^-B[p] decreases. -* (Thus, the dual ray parameter theta = s * lambdaB[p] >= 0.) -#else -* The parameter r specifies the bound violation for basic variable -* xB[p] chosen: -* -* r = lB[p] - beta[p] > 0 means that xB[p] violates its lower bound, -* so dual non-basic variable lambdaB[p] = lambda^+B[p] increases; and -* -* r = uB[p] - beta[p] < 0 means that xB[p] violates its upper bound, -* so dual non-basic variable lambdaB[p] = lambda^-B[p] decreases. -* -* (Note that r is the dual reduced cost of lambdaB[p].) -#endif -* -* Elements of p-th simplex table row t[p] = (t[p,j]) corresponding -* to basic variable xB[p] should be placed in the array locations -* trow[1], ..., trow[n-m]. -* -* The parameter tol_piv specifies a tolerance for elements of the -* simplex table row t[p]. If |t[p,j]| < tol_piv, dual basic variable -* lambdaN[j] is skipped, i.e. it is assumed that it does not depend on -* the dual ray parameter theta. -* -* The parameters tol and tol1 specify tolerances used to increase the -* choice freedom by simulating an artificial degeneracy as follows. -* If lambdaN[j] = lambda^+N[j] >= 0 and d[j] <= +delta[j], or if -* lambdaN[j] = lambda^-N[j] <= 0 and d[j] >= -delta[j], where -* delta[j] = tol + tol1 * |cN[j]|, cN[j] is objective coefficient at -* xN[j], then it is assumed that reduced cost d[j] is equal to zero. -* -* The routine determines the index 1 <= q <= n-m of non-basic variable -* xN[q], for which corresponding dual basic variable lambda^+N[j] or -* lambda^-N[j] reaches its zero bound first on increasing the dual ray -* parameter theta, and returns p on exit. And if theta may increase -* unlimitedly, the routine returns zero. */ - -int spy_chuzc_std(SPXLP *lp, const double d[/*1+n-m*/], -#if 0 /* 14/III-2016 */ - double s, const double trow[/*1+n-m*/], double tol_piv, -#else - double r, const double trow[/*1+n-m*/], double tol_piv, -#endif - double tol, double tol1) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, k, q; - double alfa, biga, delta, teta, teta_min; -#if 0 /* 14/III-2016 */ - xassert(s == +1.0 || s == -1.0); -#else - double s; - xassert(r != 0.0); - s = (r > 0.0 ? +1.0 : -1.0); -#endif - /* nothing is chosen so far */ - q = 0, teta_min = DBL_MAX, biga = 0.0; - /* walk thru the list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* if xN[j] is fixed variable, skip it */ - if (l[k] == u[k]) - continue; - alfa = s * trow[j]; - if (alfa >= +tol_piv && !flag[j]) - { /* xN[j] is either free or has its lower bound active, so - * lambdaN[j] = d[j] >= 0 decreases down to zero */ - delta = tol + tol1 * (c[k] >= 0.0 ? +c[k] : -c[k]); - /* determine theta on which lambdaN[j] reaches zero */ - teta = (d[j] < +delta ? 0.0 : d[j] / alfa); - } - else if (alfa <= -tol_piv && (l[k] == -DBL_MAX || flag[j])) - { /* xN[j] is either free or has its upper bound active, so - * lambdaN[j] = d[j] <= 0 increases up to zero */ - delta = tol + tol1 * (c[k] >= 0.0 ? +c[k] : -c[k]); - /* determine theta on which lambdaN[j] reaches zero */ - teta = (d[j] > -delta ? 0.0 : d[j] / alfa); - } - else - { /* lambdaN[j] cannot reach zero on increasing theta */ - continue; - } - /* choose non-basic variable xN[q] by corresponding dual basic - * variable lambdaN[q] for which theta is minimal */ - xassert(teta >= 0.0); - alfa = (alfa >= 0.0 ? +alfa : -alfa); - if (teta_min > teta || (teta_min == teta && biga < alfa)) - q = j, teta_min = teta, biga = alfa; - } - return q; -} - -/*********************************************************************** -* spy_chuzc_harris - choose non-basic var. (dual Harris' ratio test) -* -* This routine implements dual Harris' ratio test to choose non-basic -* variable xN[q]. -* -* All the parameters, except tol and tol1, as well as the returned -* value have the same meaning as for the routine spx_chuzr_std (see -* above). -* -* The parameters tol and tol1 specify tolerances on zero bound -* violations for reduced costs of non-basic variables. For reduced -* cost d[j] the tolerance is delta[j] = tol + tol1 |cN[j]|, where -* cN[j] is objective coefficient at non-basic variable xN[j]. */ - -int spy_chuzc_harris(SPXLP *lp, const double d[/*1+n-m*/], -#if 0 /* 14/III-2016 */ - double s, const double trow[/*1+n-m*/], double tol_piv, -#else - double r, const double trow[/*1+n-m*/], double tol_piv, -#endif - double tol, double tol1) -{ int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, k, q; - double alfa, biga, delta, teta, teta_min; -#if 0 /* 14/III-2016 */ - xassert(s == +1.0 || s == -1.0); -#else - double s; - xassert(r != 0.0); - s = (r > 0.0 ? +1.0 : -1.0); -#endif - /*--------------------------------------------------------------*/ - /* first pass: determine teta_min for relaxed bounds */ - /*--------------------------------------------------------------*/ - teta_min = DBL_MAX; - /* walk thru the list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* if xN[j] is fixed variable, skip it */ - if (l[k] == u[k]) - continue; - alfa = s * trow[j]; - if (alfa >= +tol_piv && !flag[j]) - { /* xN[j] is either free or has its lower bound active, so - * lambdaN[j] = d[j] >= 0 decreases down to zero */ - delta = tol + tol1 * (c[k] >= 0.0 ? +c[k] : -c[k]); - /* determine theta on which lambdaN[j] reaches -delta */ - teta = ((d[j] < 0.0 ? 0.0 : d[j]) + delta) / alfa; - } - else if (alfa <= -tol_piv && (l[k] == -DBL_MAX || flag[j])) - { /* xN[j] is either free or has its upper bound active, so - * lambdaN[j] = d[j] <= 0 increases up to zero */ - delta = tol + tol1 * (c[k] >= 0.0 ? +c[k] : -c[k]); - /* determine theta on which lambdaN[j] reaches +delta */ - teta = ((d[j] > 0.0 ? 0.0 : d[j]) - delta) / alfa; - } - else - { /* lambdaN[j] cannot reach zero on increasing theta */ - continue; - } - xassert(teta >= 0.0); - if (teta_min > teta) - teta_min = teta; - } - /*--------------------------------------------------------------*/ - /* second pass: choose non-basic variable xN[q] */ - /*--------------------------------------------------------------*/ - if (teta_min == DBL_MAX) - { /* theta may increase unlimitedly */ - q = 0; - goto done; - } - /* nothing is chosen so far */ - q = 0, biga = 0.0; - /* walk thru the list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* if xN[j] is fixed variable, skip it */ - if (l[k] == u[k]) - continue; - alfa = s * trow[j]; - if (alfa >= +tol_piv && !flag[j]) - { /* xN[j] is either free or has its lower bound active, so - * lambdaN[j] = d[j] >= 0 decreases down to zero */ - /* determine theta on which lambdaN[j] reaches zero */ - teta = d[j] / alfa; - } - else if (alfa <= -tol_piv && (l[k] == -DBL_MAX || flag[j])) - { /* xN[j] is either free or has its upper bound active, so - * lambdaN[j] = d[j] <= 0 increases up to zero */ - /* determine theta on which lambdaN[j] reaches zero */ - teta = d[j] / alfa; - } - else - { /* lambdaN[j] cannot reach zero on increasing theta */ - continue; - } - /* choose non-basic variable for which theta is not greater - * than theta_min determined for relaxed bounds and which has - * best (largest in magnitude) pivot */ - alfa = (alfa >= 0.0 ? +alfa : -alfa); - if (teta <= teta_min && biga < alfa) - q = j, biga = alfa; - } - /* something must be chosen */ - xassert(1 <= q && q <= n-m); -done: return q; -} - -#if 0 /* 23/III-2016 */ -/*********************************************************************** -* spy_eval_bp - determine dual objective function break-points -* -* This routine determines the dual objective function break-points. -* -* The parameters lp, d, r, trow, and tol_piv have the same meaning as -* for the routine spx_chuzc_std (see above). -* -* On exit the routine stores the break-points determined to the array -* elements bp[1], ..., bp[num], where 0 <= num <= n-m is the number of -* break-points returned by the routine. -* -* The break-points stored in the array bp are ordered by ascending -* the ray parameter teta >= 0. The break-points numbered 1, ..., num-1 -* always correspond to non-basic non-fixed variables xN[j] of primal -* LP having both lower and upper bounds while the last break-point -* numbered num may correspond to a non-basic variable having only one -* lower or upper bound, if such variable prevents further increasing -* of the ray parameter teta. Besides, the routine includes in the -* array bp only the break-points that correspond to positive increment -* of the dual objective. */ - -static int CDECL fcmp(const void *v1, const void *v2) -{ const SPYBP *p1 = v1, *p2 = v2; - if (p1->teta < p2->teta) - return -1; - else if (p1->teta > p2->teta) - return +1; - else - return 0; -} - -int spy_eval_bp(SPXLP *lp, const double d[/*1+n-m*/], - double r, const double trow[/*1+n-m*/], double tol_piv, - SPYBP bp[/*1+n-m*/]) -{ int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, j_max, k, t, nnn, num; - double s, alfa, teta, teta_max, dz, v; - xassert(r != 0.0); - s = (r > 0.0 ? +1.0 : -1.0); - /* build the list of all dual basic variables lambdaN[j] that - * can reach zero on increasing the ray parameter teta >= 0 */ - num = 0; - /* walk thru the list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* if xN[j] is fixed variable, skip it */ - if (l[k] == u[k]) - continue; - alfa = s * trow[j]; - if (alfa >= +tol_piv && !flag[j]) - { /* xN[j] is either free or has its lower bound active, so - * lambdaN[j] = d[j] >= 0 decreases down to zero */ - /* determine teta[j] on which lambdaN[j] reaches zero */ - teta = (d[j] < 0.0 ? 0.0 : d[j] / alfa); - } - else if (alfa <= -tol_piv && (l[k] == -DBL_MAX || flag[j])) - { /* xN[j] is either free or has its upper bound active, so - * lambdaN[j] = d[j] <= 0 increases up to zero */ - /* determine teta[j] on which lambdaN[j] reaches zero */ - teta = (d[j] > 0.0 ? 0.0 : d[j] / alfa); - } - else - { /* lambdaN[j] cannot reach zero on increasing teta */ - continue; - } - /* add lambdaN[j] to the list */ - num++; - bp[num].j = j; - bp[num].teta = teta; - } - if (num == 0) - { /* dual unboundedness */ - goto done; - } - /* determine "blocking" dual basic variable lambdaN[j_max] that - * prevents increasing teta more than teta_max */ - j_max = 0, teta_max = DBL_MAX; - for (t = 1; t <= num; t++) - { j = bp[t].j; - k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == -DBL_MAX || u[k] == +DBL_MAX) - { /* lambdaN[j] cannot intersect zero */ - if (j_max == 0 - || teta_max > bp[t].teta - || (teta_max == bp[t].teta - && fabs(trow[j_max]) < fabs(trow[j]))) - j_max = j, teta_max = bp[t].teta; - } - } - /* keep in the list only dual basic variables lambdaN[j] that - * correspond to primal double-bounded variables xN[j] and whose - * teta[j] is not greater than teta_max */ - nnn = 0; - for (t = 1; t <= num; t++) - { j = bp[t].j; - k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] != -DBL_MAX && u[k] != +DBL_MAX - && bp[t].teta <= teta_max) - { nnn++; - bp[nnn].j = j; - bp[nnn].teta = bp[t].teta; - } - } - num = nnn; - /* sort break-points by ascending teta[j] */ - qsort(&bp[1], num, sizeof(SPYBP), fcmp); - /* add lambdaN[j_max] to the end of the list */ - if (j_max != 0) - { xassert(num < n-m); - num++; - bp[num].j = j_max; - bp[num].teta = teta_max; - } - /* compute increments of the dual objective at all break-points - * (relative to its value at teta = 0) */ - dz = 0.0; /* dual objective increment */ - v = fabs(r); /* dual objective slope d zeta / d teta */ - for (t = 1; t <= num; t++) - { /* compute increment at current break-point */ - dz += v * (bp[t].teta - (t == 1 ? 0.0 : bp[t-1].teta)); - if (dz < 0.001) - { /* break-point with non-positive increment reached */ - num = t - 1; - break; - } - bp[t].dz = dz; - /* compute next slope on the right to current break-point */ - if (t < num) - { j = bp[t].j; - k = head[m+j]; /* x[k] = xN[j] */ - xassert(-DBL_MAX < l[k] && l[k] < u[k] && u[k] < +DBL_MAX); - v -= fabs(trow[j]) * (u[k] - l[k]); - } - } -done: return num; -} -#endif - -/*********************************************************************** -* spy_ls_eval_bp - determine dual objective function break-points -* -* This routine determines the dual objective function break-points. -* -* The parameters lp, d, r, trow, and tol_piv have the same meaning as -* for the routine spx_chuzc_std (see above). -* -* The routine stores the break-points determined to the array elements -* bp[1], ..., bp[nbp] in *arbitrary* order, where 0 <= nbp <= n-m is -* the number of break-points returned by the routine on exit. */ - -int spy_ls_eval_bp(SPXLP *lp, const double d[/*1+n-m*/], - double r, const double trow[/*1+n-m*/], double tol_piv, - SPYBP bp[/*1+n-m*/]) -{ int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, k, t, nnn, nbp; - double s, alfa, teta, teta_max; - xassert(r != 0.0); - s = (r > 0.0 ? +1.0 : -1.0); - /* build the list of all dual basic variables lambdaN[j] that - * can reach zero on increasing the ray parameter teta >= 0 */ - nnn = 0, teta_max = DBL_MAX; - /* walk thru the list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - /* if xN[j] is fixed variable, skip it */ - if (l[k] == u[k]) - continue; - alfa = s * trow[j]; - if (alfa >= +tol_piv && !flag[j]) - { /* xN[j] is either free or has its lower bound active, so - * lambdaN[j] = d[j] >= 0 decreases down to zero */ - /* determine teta[j] on which lambdaN[j] reaches zero */ - teta = (d[j] < 0.0 ? 0.0 : d[j] / alfa); - /* if xN[j] has no upper bound, lambdaN[j] cannot become - * negative and thereby blocks further increasing teta */ - if (u[k] == +DBL_MAX && teta_max > teta) - teta_max = teta; - } - else if (alfa <= -tol_piv && (l[k] == -DBL_MAX || flag[j])) - { /* xN[j] is either free or has its upper bound active, so - * lambdaN[j] = d[j] <= 0 increases up to zero */ - /* determine teta[j] on which lambdaN[j] reaches zero */ - teta = (d[j] > 0.0 ? 0.0 : d[j] / alfa); - /* if xN[j] has no lower bound, lambdaN[j] cannot become - * positive and thereby blocks further increasing teta */ - if (l[k] == -DBL_MAX && teta_max > teta) - teta_max = teta; - } - else - { /* lambdaN[j] cannot reach zero on increasing teta */ - continue; - } - /* add lambdaN[j] to the list */ - nnn++; - bp[nnn].j = j; - bp[nnn].teta = teta; - } - /* remove from the list all dual basic variables lambdaN[j], for - * which teta[j] > teta_max */ - nbp = 0; - for (t = 1; t <= nnn; t++) - { if (bp[t].teta <= teta_max + 1e-6) - { nbp++; - bp[nbp].j = bp[t].j; - bp[nbp].teta = bp[t].teta; - } - } - return nbp; -} - -/*********************************************************************** -* spy_ls_select_bp - select and process dual objective break-points -* -* This routine selects a next portion of the dual objective function -* break-points and processes them. -* -* On entry to the routine it is assumed that break-points bp[1], ..., -* bp[num] are already processed, and slope is the dual objective slope -* to the right of the last processed break-point bp[num]. (Initially, -* when num = 0, slope should be specified as fabs(r), where r has the -* same meaning as above.) -* -* The routine selects break-points among bp[num+1], ..., bp[nbp], for -* which teta <= teta_lim, and moves these break-points to the array -* elements bp[num+1], ..., bp[num1], where num <= num1 <= n-m is the -* new number of processed break-points returned by the routine on -* exit. Then the routine sorts these break-points by ascending teta -* and computes the change of the dual objective function relative to -* its value at teta = 0. -* -* On exit the routine also replaces the parameter slope with a new -* value that corresponds to the new last break-point bp[num1]. */ - -static int CDECL fcmp(const void *v1, const void *v2) -{ const SPYBP *p1 = v1, *p2 = v2; - if (p1->teta < p2->teta) - return -1; - else if (p1->teta > p2->teta) - return +1; - else - return 0; -} - -int spy_ls_select_bp(SPXLP *lp, const double trow[/*1+n-m*/], - int nbp, SPYBP bp[/*1+n-m*/], int num, double *slope, double - teta_lim) -{ int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int j, k, t, num1; - double teta, dz; - xassert(0 <= num && num <= nbp && nbp <= n-m); - /* select a new portion of break-points */ - num1 = num; - for (t = num+1; t <= nbp; t++) - { if (bp[t].teta <= teta_lim) - { /* move break-point to the beginning of the new portion */ - num1++; - j = bp[num1].j, teta = bp[num1].teta; - bp[num1].j = bp[t].j, bp[num1].teta = bp[t].teta; - bp[t].j = j, bp[t].teta = teta; - } - } - /* sort new break-points bp[num+1], ..., bp[num1] by ascending - * the ray parameter teta */ - if (num1 - num > 1) - qsort(&bp[num+1], num1 - num, sizeof(SPYBP), fcmp); - /* calculate the dual objective change at the new break-points */ - for (t = num+1; t <= num1; t++) - { /* calculate the dual objective change relative to its value - * at break-point bp[t-1] */ - if (*slope == -DBL_MAX) - dz = -DBL_MAX; - else - dz = (*slope) * - (bp[t].teta - (t == 1 ? 0.0 : bp[t-1].teta)); - /* calculate the dual objective change relative to its value - * at teta = 0 */ - if (dz == -DBL_MAX) - bp[t].dz = -DBL_MAX; - else - bp[t].dz = (t == 1 ? 0.0 : bp[t-1].dz) + dz; - /* calculate a new slope of the dual objective to the right of - * the current break-point bp[t] */ - if (*slope != -DBL_MAX) - { j = bp[t].j; - k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == -DBL_MAX || u[k] == +DBL_MAX) - *slope = -DBL_MAX; /* blocking break-point reached */ - else - { xassert(l[k] < u[k]); - *slope -= fabs(trow[j]) * (u[k] - l[k]); - } - } - } - return num1; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spychuzc.h b/code/3rd_glpk/simplex/spychuzc.h deleted file mode 100644 index 8aa45a07..00000000 --- a/code/3rd_glpk/simplex/spychuzc.h +++ /dev/null @@ -1,85 +0,0 @@ -/* spychuzc.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2016 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPYCHUZC_H -#define SPYCHUZC_H - -#include "spxlp.h" - -#define spy_chuzc_std _glp_spy_chuzc_std -int spy_chuzc_std(SPXLP *lp, const double d[/*1+n-m*/], -#if 0 /* 14/III-2016 */ - double s, const double trow[/*1+n-m*/], double tol_piv, -#else - double r, const double trow[/*1+n-m*/], double tol_piv, -#endif - double tol, double tol1); -/* choose non-basic variable (dual textbook ratio test) */ - -#define spy_chuzc_harris _glp_spy_chuzc_harris -int spy_chuzc_harris(SPXLP *lp, const double d[/*1+n-m*/], -#if 0 /* 14/III-2016 */ - double s, const double trow[/*1+n-m*/], double tol_piv, -#else - double r, const double trow[/*1+n-m*/], double tol_piv, -#endif - double tol, double tol1); -/* choose non-basic variable (dual Harris' ratio test) */ - -typedef struct SPYBP SPYBP; - -struct SPYBP -{ /* dual objective function break point */ - int j; - /* dual basic variable lambdaN[j], 1 <= j <= n-m, that intersects - * zero at this break point */ - double teta; - /* ray parameter value, teta[j] >= 0, at this break point */ - double dz; - /* increment, zeta[j] - zeta[0], of the dual objective function - * at this break point */ -}; - -#if 0 /* 23/III-2016 */ -#define spy_eval_bp _glp_spy_eval_bp -int spy_eval_bp(SPXLP *lp, const double d[/*1+n-m*/], - double r, const double trow[/*1+n-m*/], double tol_piv, - SPYBP bp[/*1+n-m*/]); -/* determine dual objective function break-points */ -#endif - -#define spy_ls_eval_bp _glp_spy_ls_eval_bp -int spy_ls_eval_bp(SPXLP *lp, const double d[/*1+n-m*/], - double r, const double trow[/*1+n-m*/], double tol_piv, - SPYBP bp[/*1+n-m*/]); -/* determine dual objective function break-points */ - -#define spy_ls_select_bp _glp_spy_ls_select_bp -int spy_ls_select_bp(SPXLP *lp, const double trow[/*1+n-m*/], - int nbp, SPYBP bp[/*1+n-m*/], int num, double *slope, double - teta_lim); -/* select and process dual objective break-points */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spychuzr.c b/code/3rd_glpk/simplex/spychuzr.c deleted file mode 100644 index 63079c17..00000000 --- a/code/3rd_glpk/simplex/spychuzr.c +++ /dev/null @@ -1,483 +0,0 @@ -/* spychuzr.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#include "env.h" -#include "spychuzr.h" - -/*********************************************************************** -* spy_chuzr_sel - select eligible basic variables -* -* This routine selects eligible basic variables xB[i], whose value -* beta[i] violates corresponding lower lB[i] or upper uB[i] bound. -* Positive bound violation rp[i] = lb[i] - beta[i] > 0 is the reduced -* cost of non-basic dual variable lambda^+B[i] >= 0, so increasing it -* increases the dual objective. Similarly, negative bound violation -* rn[i] = ub[i] - beta[i] < 0 is the reduced cost of non-basic dual -* variable lambda^-B[i] <= 0, so decreasing it also increases the dual -* objective. -* -* Current values of basic variables should be placed in the array -* locations beta[1], ..., beta[m]. -* -* Basic variable xB[i] is considered eligible, if: -* -* beta[i] <= lB[i] - eps1[i], or -* -* beta[i] >= uB[i] + eps2[i], -* -* for -* -* eps1[i] = tol + tol1 * |lB[i]|, -* -* eps2[i] = tol + tol2 * |uB[i]|, -* -* where lB[i] and uB[i] are, resp., lower and upper bounds of xB[i], -* tol and tol1 are specified tolerances. -* -* On exit the routine stores indices i of eligible basic variables -* xB[i] to the array locations list[1], ..., list[num] and returns the -* number of such variables 0 <= num <= m. (If the parameter list is -* specified as NULL, no indices are stored.) */ - -int spy_chuzr_sel(SPXLP *lp, const double beta[/*1+m*/], double tol, - double tol1, int list[/*1+m*/]) -{ int m = lp->m; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int i, k, num; - double lk, uk, eps; - num = 0; - /* walk thru list of basic variables */ - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - lk = l[k], uk = u[k]; - /* check if xB[i] is eligible */ - if (beta[i] < lk) - { /* determine absolute tolerance eps1[i] */ - eps = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] < lk - eps) - { /* lower bound is violated */ - num++; - if (list != NULL) - list[num] = i; - } - } - else if (beta[i] > uk) - { /* determine absolute tolerance eps2[i] */ - eps = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] > uk + eps) - { /* upper bound is violated */ - num++; - if (list != NULL) - list[num] = i; - } - } - } - return num; -} - -/*********************************************************************** -* spy_chuzr_std - choose basic variable (dual Dantzig's rule) -* -* This routine chooses most eligible basic variable xB[p] according -* to dual Dantzig's ("standard") rule: -* -* r[p] = max |r[i]|, -* i in I -* -* ( lB[i] - beta[i], if beta[i] < lB[i] -* ( -* r[i] = { 0, if lB[i] <= beta[i] <= uB[i] -* ( -* ( uB[i] - beta[i], if beta[i] > uB[i] -* -* where I <= {1, ..., m} is the set of indices of eligible basic -* variables, beta[i] is current value of xB[i], lB[i] and uB[i] are, -* resp., lower and upper bounds of xB[i], r[i] is bound violation. -* -* Current values of basic variables should be placed in the array -* locations beta[1], ..., beta[m]. -* -* Indices of eligible basic variables i in I should be placed in the -* array locations list[1], ..., list[num], where num = |J| > 0 is the -* total number of such variables. -* -* On exit the routine returns p, the index of the basic variable xB[p] -* chosen. */ - -int spy_chuzr_std(SPXLP *lp, const double beta[/*1+m*/], int num, - const int list[]) -{ int m = lp->m; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int i, k, p, t; - double abs_ri, abs_rp; - xassert(0 < num && num <= m); - p = 0, abs_rp = -1.0; - for (t = 1; t <= num; t++) - { i = list[t]; - k = head[i]; /* x[k] = xB[i] */ - if (beta[i] < l[k]) - abs_ri = l[k] - beta[i]; - else if (beta[i] > u[k]) - abs_ri = beta[i] - u[k]; - else - xassert(t != t); - if (abs_rp < abs_ri) - p = i, abs_rp = abs_ri; - } - xassert(p != 0); - return p; -} - -/*********************************************************************** -* spy_alloc_se - allocate dual pricing data block -* -* This routine allocates the memory for arrays used in the dual -* pricing data block. */ - -void spy_alloc_se(SPXLP *lp, SPYSE *se) -{ int m = lp->m; - int n = lp->n; -#if 1 /* 30/III-2016 */ - int i; -#endif - se->valid = 0; - se->refsp = talloc(1+n, char); - se->gamma = talloc(1+m, double); - se->work = talloc(1+m, double); -#if 1 /* 30/III-2016 */ - se->u.n = m; - se->u.nnz = 0; - se->u.ind = talloc(1+m, int); - se->u.vec = talloc(1+m, double); - for (i = 1; i <= m; i++) - se->u.vec[i] = 0.0; -#endif - return; -} - -/*********************************************************************** -* spy_reset_refsp - reset dual reference space -* -* This routine resets (re-initializes) the dual reference space -* composing it from dual variables which are non-basic (corresponding -* to basic primal variables) in the current basis, and sets all -* weights gamma[i] to 1. */ - -void spy_reset_refsp(SPXLP *lp, SPYSE *se) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *gamma = se->gamma; - int i, k; - se->valid = 1; - memset(&refsp[1], 0, n * sizeof(char)); - for (i = 1; i <= m; i++) - { k = head[i]; /* x[k] = xB[i] */ - refsp[k] = 1; - gamma[i] = 1.0; - } - return; -} - -/*********************************************************************** -* spy_eval_gamma_i - compute dual proj. steepest edge weight directly -* -* This routine computes dual projected steepest edge weight gamma[i], -* 1 <= i <= m, for the current basis directly with the formula: -* -* n-m -* gamma[i] = delta[i] + sum eta[j] * T[i,j]**2, -* j=1 -* -* where T[i,j] is element of the current simplex table, and -* -* ( 1, if lambdaN[j] is in the reference space -* eta[j] = { -* ( 0, otherwise -* -* ( 1, if lambdaB[i] is in the reference space -* delta[i] = { -* ( 0, otherwise -* -* Dual basic variable lambdaN[j] corresponds to primal non-basic -* variable xN[j], and dual non-basic variable lambdaB[j] corresponds -* to primal basic variable xB[i]. -* -* NOTE: For testing/debugging only. */ - -double spy_eval_gamma_i(SPXLP *lp, SPYSE *se, int i) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *rho = se->work; - int j, k; - double gamma_i, t_ij; - xassert(se->valid); - xassert(1 <= i && i <= m); - k = head[i]; /* x[k] = xB[i] */ - gamma_i = (refsp[k] ? 1.0 : 0.0); - spx_eval_rho(lp, i, rho); - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (refsp[k]) - { t_ij = spx_eval_tij(lp, rho, j); - gamma_i += t_ij * t_ij; - } - } - return gamma_i; -} - -/*********************************************************************** -* spy_chuzr_pse - choose basic variable (dual projected steepest edge) -* -* This routine chooses most eligible basic variable xB[p] according -* to the dual projected steepest edge method: -* -* r[p]**2 r[i]**2 -* -------- = max -------- , -* gamma[p] i in I gamma[i] -* -* ( lB[i] - beta[i], if beta[i] < lB[i] -* ( -* r[i] = { 0, if lB[i] <= beta[i] <= uB[i] -* ( -* ( uB[i] - beta[i], if beta[i] > uB[i] -* -* where I <= {1, ..., m} is the set of indices of eligible basic -* variables, beta[i] is current value of xB[i], lB[i] and uB[i] are, -* resp., lower and upper bounds of xB[i], r[i] is bound violation. -* -* Current values of basic variables should be placed in the array -* locations beta[1], ..., beta[m]. -* -* Indices of eligible basic variables i in I should be placed in the -* array locations list[1], ..., list[num], where num = |J| > 0 is the -* total number of such variables. -* -* On exit the routine returns p, the index of the basic variable xB[p] -* chosen. */ - -int spy_chuzr_pse(SPXLP *lp, SPYSE *se, const double beta[/*1+m*/], - int num, const int list[]) -{ int m = lp->m; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - double *gamma = se->gamma; - int i, k, p, t; - double best, ri, temp; - xassert(0 < num && num <= m); - p = 0, best = -1.0; - for (t = 1; t <= num; t++) - { i = list[t]; - k = head[i]; /* x[k] = xB[i] */ - if (beta[i] < l[k]) - ri = l[k] - beta[i]; - else if (beta[i] > u[k]) - ri = u[k] - beta[i]; - else - xassert(t != t); - /* FIXME */ - if (gamma[i] < DBL_EPSILON) - temp = 0.0; - else - temp = (ri * ri) / gamma[i]; - if (best < temp) - p = i, best = temp; - } - xassert(p != 0); - return p; -} - -/*********************************************************************** -* spy_update_gamma - update dual proj. steepest edge weights exactly -* -* This routine updates the vector gamma = (gamma[i]) of dual projected -* steepest edge weights exactly, for the adjacent basis. -* -* On entry to the routine the content of the se object should be valid -* and should correspond to the current basis. -* -* The parameter 1 <= p <= m specifies basic variable xB[p] which -* becomes non-basic variable xN[q] in the adjacent basis. -* -* The parameter 1 <= q <= n-m specified non-basic variable xN[q] which -* becomes basic variable xB[p] in the adjacent basis. -* -* It is assumed that the array trow contains elements of p-th (pivot) -* row T'[p] of the simplex table in locations trow[1], ..., trow[n-m]. -* It is also assumed that the array tcol contains elements of q-th -* (pivot) column T[q] of the simple table in locations tcol[1], ..., -* tcol[m]. (These row and column should be computed for the current -* basis.) -* -* For details about the formulae used see the program documentation. -* -* The routine also computes the relative error: -* -* e = |gamma[p] - gamma'[p]| / (1 + |gamma[p]|), -* -* where gamma'[p] is the weight for lambdaB[p] (which is dual -* non-basic variable corresponding to xB[p]) on entry to the routine, -* and returns e on exit. (If e happens to be large enough, the calling -* program may reset the reference space, since other weights also may -* be inaccurate.) */ - -double spy_update_gamma(SPXLP *lp, SPYSE *se, int p, int q, - const double trow[/*1+n-m*/], const double tcol[/*1+m*/]) -{ int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *gamma = se->gamma; - double *u = se->work; - int i, j, k, ptr, end; - double gamma_p, delta_p, e, r, t1, t2; - xassert(se->valid); - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - /* compute gamma[p] in current basis more accurately; also - * compute auxiliary vector u */ - k = head[p]; /* x[k] = xB[p] */ - gamma_p = delta_p = (refsp[k] ? 1.0 : 0.0); - for (i = 1; i <= m; i++) - u[i] = 0.0; - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (refsp[k] && trow[j] != 0.0) - { gamma_p += trow[j] * trow[j]; - /* u := u + T[p,j] * N[j], where N[j] = A[k] is constraint - * matrix column corresponding to xN[j] */ - ptr = lp->A_ptr[k]; - end = lp->A_ptr[k+1]; - for (; ptr < end; ptr++) - u[lp->A_ind[ptr]] += trow[j] * lp->A_val[ptr]; - } - } - bfd_ftran(lp->bfd, u); - /* compute relative error in gamma[p] */ - e = fabs(gamma_p - gamma[p]) / (1.0 + gamma_p); - /* compute new gamma[p] */ - gamma[p] = gamma_p / (tcol[p] * tcol[p]); - /* compute new gamma[i] for all i != p */ - for (i = 1; i <= m; i++) - { if (i == p) - continue; - /* compute r[i] = T[i,q] / T[p,q] */ - r = tcol[i] / tcol[p]; - /* compute new gamma[i] */ - t1 = gamma[i] + r * (r * gamma_p + u[i] + u[i]); - k = head[i]; /* x[k] = xB[i] */ - t2 = (refsp[k] ? 1.0 : 0.0) + delta_p * r * r; - gamma[i] = (t1 >= t2 ? t1 : t2); - } - return e; -} - -#if 1 /* 30/III-2016 */ -double spy_update_gamma_s(SPXLP *lp, SPYSE *se, int p, int q, - const FVS *trow, const FVS *tcol) -{ /* sparse version of spy_update_gamma */ - int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *refsp = se->refsp; - double *gamma = se->gamma; - double *u = se->work; - int trow_nnz = trow->nnz; - int *trow_ind = trow->ind; - double *trow_vec = trow->vec; - int tcol_nnz = tcol->nnz; - int *tcol_ind = tcol->ind; - double *tcol_vec = tcol->vec; - int i, j, k, t, ptr, end; - double gamma_p, delta_p, e, r, t1, t2; - xassert(se->valid); - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - /* compute gamma[p] in current basis more accurately; also - * compute auxiliary vector u */ - k = head[p]; /* x[k] = xB[p] */ - gamma_p = delta_p = (refsp[k] ? 1.0 : 0.0); - for (i = 1; i <= m; i++) - u[i] = 0.0; - for (t = 1; t <= trow_nnz; t++) - { j = trow_ind[t]; - k = head[m+j]; /* x[k] = xN[j] */ - if (refsp[k]) - { gamma_p += trow_vec[j] * trow_vec[j]; - /* u := u + T[p,j] * N[j], where N[j] = A[k] is constraint - * matrix column corresponding to xN[j] */ - ptr = lp->A_ptr[k]; - end = lp->A_ptr[k+1]; - for (; ptr < end; ptr++) - u[lp->A_ind[ptr]] += trow_vec[j] * lp->A_val[ptr]; - } - } - bfd_ftran(lp->bfd, u); - /* compute relative error in gamma[p] */ - e = fabs(gamma_p - gamma[p]) / (1.0 + gamma_p); - /* compute new gamma[p] */ - gamma[p] = gamma_p / (tcol_vec[p] * tcol_vec[p]); - /* compute new gamma[i] for all i != p */ - for (t = 1; t <= tcol_nnz; t++) - { i = tcol_ind[t]; - if (i == p) - continue; - /* compute r[i] = T[i,q] / T[p,q] */ - r = tcol_vec[i] / tcol_vec[p]; - /* compute new gamma[i] */ - t1 = gamma[i] + r * (r * gamma_p + u[i] + u[i]); - k = head[i]; /* x[k] = xB[i] */ - t2 = (refsp[k] ? 1.0 : 0.0) + delta_p * r * r; - gamma[i] = (t1 >= t2 ? t1 : t2); - } - return e; -} -#endif - -/*********************************************************************** -* spy_free_se - deallocate dual pricing data block -* -* This routine deallocates the memory used for arrays in the dual -* pricing data block. */ - -void spy_free_se(SPXLP *lp, SPYSE *se) -{ xassert(lp == lp); - tfree(se->refsp); - tfree(se->gamma); - tfree(se->work); -#if 1 /* 30/III-2016 */ - tfree(se->u.ind); - tfree(se->u.vec); -#endif - return; -} - -/* eof */ diff --git a/code/3rd_glpk/simplex/spychuzr.h b/code/3rd_glpk/simplex/spychuzr.h deleted file mode 100644 index 31f01b78..00000000 --- a/code/3rd_glpk/simplex/spychuzr.h +++ /dev/null @@ -1,97 +0,0 @@ -/* spychuzr.h */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#ifndef SPYCHUZR_H -#define SPYCHUZR_H - -#include "spxlp.h" - -#define spy_chuzr_sel _glp_spy_chuzr_sel -int spy_chuzr_sel(SPXLP *lp, const double beta[/*1+m*/], double tol, - double tol1, int list[/*1+m*/]); -/* select eligible basic variables */ - -#define spy_chuzr_std _glp_spy_chuzr_std -int spy_chuzr_std(SPXLP *lp, const double beta[/*1+m*/], int num, - const int list[]); -/* choose basic variable (dual Dantzig's rule) */ - -typedef struct SPYSE SPYSE; - -struct SPYSE -{ /* dual projected steepest edge and Devex pricing data block */ - int valid; - /* content validity flag */ - char *refsp; /* char refsp[1+n]; */ - /* refsp[0] is not used; - * refsp[k], 1 <= k <= n, is the flag meaning that dual variable - * lambda[k] is in the dual reference space */ - double *gamma; /* double gamma[1+m]; */ - /* gamma[0] is not used; - * gamma[i], 1 <= i <= m, is the weight for reduced cost r[i] - * of dual non-basic variable lambdaB[j] in the current basis - * (r[i] is bound violation for basic variable xB[i]) */ - double *work; /* double work[1+m]; */ - /* working array */ -#if 1 /* 30/III-2016 */ - FVS u; /* FVS u[1:m]; */ - /* working vector */ -#endif -}; - -#define spy_alloc_se _glp_spy_alloc_se -void spy_alloc_se(SPXLP *lp, SPYSE *se); -/* allocate dual pricing data block */ - -#define spy_reset_refsp _glp_spy_reset_refsp -void spy_reset_refsp(SPXLP *lp, SPYSE *se); -/* reset dual reference space */ - -#define spy_eval_gamma_i _glp_spy_eval_gamma_i -double spy_eval_gamma_i(SPXLP *lp, SPYSE *se, int i); -/* compute dual projected steepest edge weight directly */ - -#define spy_chuzr_pse _glp_spy_chuzr_pse -int spy_chuzr_pse(SPXLP *lp, SPYSE *se, const double beta[/*1+m*/], - int num, const int list[]); -/* choose basic variable (dual projected steepest edge) */ - -#define spy_update_gamma _glp_spy_update_gamma -double spy_update_gamma(SPXLP *lp, SPYSE *se, int p, int q, - const double trow[/*1+n-m*/], const double tcol[/*1+m*/]); -/* update dual projected steepest edge weights exactly */ - -#if 1 /* 30/III-2016 */ -#define spy_update_gamma_s _glp_spy_update_gamma_s -double spy_update_gamma_s(SPXLP *lp, SPYSE *se, int p, int q, - const FVS *trow, const FVS *tcol); -/* sparse version of spy_update_gamma */ -#endif - -#define spy_free_se _glp_spy_free_se -void spy_free_se(SPXLP *lp, SPYSE *se); -/* deallocate dual pricing data block */ - -#endif - -/* eof */ diff --git a/code/3rd_glpk/simplex/spydual.c b/code/3rd_glpk/simplex/spydual.c deleted file mode 100644 index 89d98db9..00000000 --- a/code/3rd_glpk/simplex/spydual.c +++ /dev/null @@ -1,2101 +0,0 @@ -/* spydual.c */ - -/*********************************************************************** -* This code is part of GLPK (GNU Linear Programming Kit). -* -* Copyright (C) 2015-2017 Andrew Makhorin, Department for Applied -* Informatics, Moscow Aviation Institute, Moscow, Russia. All rights -* reserved. E-mail: . -* -* GLPK is free software: you can redistribute it and/or modify it -* under the terms of the GNU General Public License as published by -* the Free Software Foundation, either version 3 of the License, or -* (at your option) any later version. -* -* GLPK is distributed in the hope that it will be useful, but WITHOUT -* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public -* License for more details. -* -* You should have received a copy of the GNU General Public License -* along with GLPK. If not, see . -***********************************************************************/ - -#if 1 /* 18/VII-2017 */ -#define SCALE_Z 1 -#endif - -#include "env.h" -#include "simplex.h" -#include "spxat.h" -#include "spxnt.h" -#include "spxprob.h" -#include "spychuzc.h" -#include "spychuzr.h" -#if 0 /* 11/VI-2017 */ -#if 1 /* 29/III-2016 */ -#include "fvs.h" -#endif -#endif - -#define CHECK_ACCURACY 0 -/* (for debugging) */ - -struct csa -{ /* common storage area */ - SPXLP *lp; - /* LP problem data and its (current) basis; this LP has m rows - * and n columns */ - int dir; - /* original optimization direction: - * +1 - minimization - * -1 - maximization */ -#if SCALE_Z - double fz; - /* factor used to scale original objective */ -#endif - double *orig_b; /* double orig_b[1+m]; */ - /* copy of original right-hand sides */ - double *orig_c; /* double orig_c[1+n]; */ - /* copy of original objective coefficients */ - double *orig_l; /* double orig_l[1+n]; */ - /* copy of original lower bounds */ - double *orig_u; /* double orig_u[1+n]; */ - /* copy of original upper bounds */ - SPXAT *at; - /* mxn-matrix A of constraint coefficients, in sparse row-wise - * format (NULL if not used) */ - SPXNT *nt; - /* mx(n-m)-matrix N composed of non-basic columns of constraint - * matrix A, in sparse row-wise format (NULL if not used) */ - int phase; - /* search phase: - * 0 - not determined yet - * 1 - searching for dual feasible solution - * 2 - searching for optimal solution */ - double *beta; /* double beta[1+m]; */ - /* beta[i] is primal value of basic variable xB[i] */ - int beta_st; - /* status of the vector beta: - * 0 - undefined - * 1 - just computed - * 2 - updated */ - double *d; /* double d[1+n-m]; */ - /* d[j] is reduced cost of non-basic variable xN[j] */ - int d_st; - /* status of the vector d: - * 0 - undefined - * 1 - just computed - * 2 - updated */ - SPYSE *se; - /* dual projected steepest edge and Devex pricing data block - * (NULL if not used) */ -#if 0 /* 30/III-2016 */ - int num; - /* number of eligible basic variables */ - int *list; /* int list[1+m]; */ - /* list[1], ..., list[num] are indices i of eligible basic - * variables xB[i] */ -#else - FVS r; /* FVS r[1:m]; */ - /* vector of primal infeasibilities */ - /* r->nnz = num; r->ind = list */ - /* vector r has the same status as vector beta (see above) */ -#endif - int p; - /* xB[p] is a basic variable chosen to leave the basis */ -#if 0 /* 29/III-2016 */ - double *trow; /* double trow[1+n-m]; */ -#else - FVS trow; /* FVS trow[1:n-m]; */ -#endif - /* p-th (pivot) row of the simplex table */ -#if 1 /* 16/III-2016 */ - SPYBP *bp; /* SPYBP bp[1+n-m]; */ - /* dual objective break-points */ -#endif - int q; - /* xN[q] is a non-basic variable chosen to enter the basis */ -#if 0 /* 29/III-2016 */ - double *tcol; /* double tcol[1+m]; */ -#else - FVS tcol; /* FVS tcol[1:m]; */ -#endif - /* q-th (pivot) column of the simplex table */ - double *work; /* double work[1+m]; */ - /* working array */ - double *work1; /* double work1[1+n-m]; */ - /* another working array */ -#if 0 /* 11/VI-2017 */ -#if 1 /* 31/III-2016 */ - FVS wrow; /* FVS wrow[1:n-m]; */ - FVS wcol; /* FVS wcol[1:m]; */ - /* working sparse vectors */ -#endif -#endif - int p_stat, d_stat; - /* primal and dual solution statuses */ - /*--------------------------------------------------------------*/ - /* control parameters (see struct glp_smcp) */ - int msg_lev; - /* message level */ - int dualp; - /* if this flag is set, report failure in case of instability */ -#if 0 /* 16/III-2016 */ - int harris; - /* dual ratio test technique: - * 0 - textbook ratio test - * 1 - Harris' two pass ratio test */ -#else - int r_test; - /* dual ratio test technique: - * GLP_RT_STD - textbook ratio test - * GLP_RT_HAR - Harris' two pass ratio test - * GLP_RT_FLIP - long-step (flip-flop) ratio test */ -#endif - double tol_bnd, tol_bnd1; - /* primal feasibility tolerances */ - double tol_dj, tol_dj1; - /* dual feasibility tolerances */ - double tol_piv; - /* pivot tolerance */ - double obj_lim; - /* objective limit */ - int it_lim; - /* iteration limit */ - int tm_lim; - /* time limit, milliseconds */ - int out_frq; -#if 0 /* 15/VII-2017 */ - /* display output frequency, iterations */ -#else - /* display output frequency, milliseconds */ -#endif - int out_dly; - /* display output delay, milliseconds */ - /*--------------------------------------------------------------*/ - /* working parameters */ - double tm_beg; - /* time value at the beginning of the search */ - int it_beg; - /* simplex iteration count at the beginning of the search */ - int it_cnt; - /* simplex iteration count; it increases by one every time the - * basis changes */ - int it_dpy; - /* simplex iteration count at most recent display output */ -#if 1 /* 15/VII-2017 */ - double tm_dpy; - /* time value at most recent display output */ -#endif - int inv_cnt; - /* basis factorization count since most recent display output */ -#if 1 /* 11/VII-2017 */ - int degen; - /* count of successive degenerate iterations; this count is used - * to detect stalling */ -#endif -#if 1 /* 23/III-2016 */ - int ns_cnt, ls_cnt; - /* normal and long-step iteration count */ -#endif -}; - -/*********************************************************************** -* check_flags - check correctness of active bound flags -* -* This routine checks that flags specifying active bounds of all -* non-basic variables are correct. -* -* NOTE: It is important to note that if bounds of variables have been -* changed, active bound flags should be corrected accordingly. */ - -static void check_flags(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - int j, k; - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == -DBL_MAX && u[k] == +DBL_MAX) - xassert(!flag[j]); - else if (l[k] != -DBL_MAX && u[k] == +DBL_MAX) - xassert(!flag[j]); - else if (l[k] == -DBL_MAX && u[k] != +DBL_MAX) - xassert(flag[j]); - else if (l[k] == u[k]) - xassert(!flag[j]); - } - return; -} - -/*********************************************************************** -* set_art_bounds - set artificial right-hand sides and bounds -* -* This routine sets artificial right-hand sides and artificial bounds -* for all variables to minimize the sum of dual infeasibilities on -* phase I. Given current reduced costs d = (d[j]) this routine also -* sets active artificial bounds of non-basic variables to provide dual -* feasibility (this is always possible because all variables have both -* lower and upper artificial bounds). */ - -static void set_art_bounds(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *b = lp->b; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - double *d = csa->d; - int i, j, k; -#if 1 /* 31/III-2016: FIXME */ - /* set artificial right-hand sides */ - for (i = 1; i <= m; i++) - b[i] = 0.0; - /* set artificial bounds depending on types of variables */ - for (k = 1; k <= n; k++) - { if (csa->orig_l[k] == -DBL_MAX && csa->orig_u[k] == +DBL_MAX) - { /* force free variables to enter the basis */ - l[k] = -1e3, u[k] = +1e3; - } - else if (csa->orig_l[k] != -DBL_MAX && csa->orig_u[k] == +DBL_MAX) - l[k] = 0.0, u[k] = +1.0; - else if (csa->orig_l[k] == -DBL_MAX && csa->orig_u[k] != +DBL_MAX) - l[k] = -1.0, u[k] = 0.0; - else - l[k] = u[k] = 0.0; - } -#endif - /* set active artificial bounds for non-basic variables */ - xassert(csa->d_st == 1); - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - flag[j] = (l[k] != u[k] && d[j] < 0.0); - } - /* invalidate values of basic variables, since active bounds of - * non-basic variables have been changed */ - csa->beta_st = 0; - return; -} - -/*********************************************************************** -* set_orig_bounds - restore original right-hand sides and bounds -* -* This routine restores original right-hand sides and original bounds -* for all variables. This routine also sets active original bounds for -* non-basic variables; for double-bounded non-basic variables current -* reduced costs d = (d[j]) are used to decide which bound (lower or -* upper) should be made active. */ - -static void set_orig_bounds(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *b = lp->b; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - double *d = csa->d; - int j, k; - /* restore original right-hand sides */ - memcpy(b, csa->orig_b, (1+m) * sizeof(double)); - /* restore original bounds of all variables */ - memcpy(l, csa->orig_l, (1+n) * sizeof(double)); - memcpy(u, csa->orig_u, (1+n) * sizeof(double)); - /* set active original bounds for non-basic variables */ - xassert(csa->d_st == 1); - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == -DBL_MAX && u[k] == +DBL_MAX) - flag[j] = 0; - else if (l[k] != -DBL_MAX && u[k] == +DBL_MAX) - flag[j] = 0; - else if (l[k] == -DBL_MAX && u[k] != +DBL_MAX) - flag[j] = 1; - else if (l[k] != u[k]) - flag[j] = (d[j] < 0.0); - else - flag[j] = 0; - } - /* invalidate values of basic variables, since active bounds of - * non-basic variables have been changed */ - csa->beta_st = 0; - return; -} - -/*********************************************************************** -* check_feas - check dual feasibility of basic solution -* -* This routine checks that reduced costs of all non-basic variables -* d = (d[j]) have correct signs. -* -* Reduced cost d[j] is considered as having correct sign within the -* specified tolerance depending on status of non-basic variable xN[j] -* if one of the following conditions is met: -* -* xN[j] is free -eps <= d[j] <= +eps -* -* xN[j] has its lower bound active d[j] >= -eps -* -* xN[j] has its upper bound active d[j] <= +eps -* -* xN[j] is fixed d[j] has any value -* -* where eps = tol + tol1 * |cN[j]|, cN[j] is the objective coefficient -* at xN[j]. (See also the routine spx_chuzc_sel.) -* -* The flag recov allows the routine to recover dual feasibility by -* changing active bounds of non-basic variables. (For example, if -* xN[j] has its lower bound active and d[j] < -eps, the feasibility -* can be recovered by making xN[j] active on its upper bound.) -* -* If the basic solution is dual feasible, the routine returns zero. -* If the basic solution is dual infeasible, but its dual feasibility -* can be recovered (or has been recovered, if the flag recov is set), -* the routine returns a negative value. Otherwise, the routine returns -* the number j of some non-basic variable xN[j], whose reduced cost -* d[j] is dual infeasible and cannot be recovered. */ - -static int check_feas(struct csa *csa, double tol, double tol1, - int recov) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - double *d = csa->d; - int j, k, ret = 0; - double eps; - /* reduced costs should be just computed */ - xassert(csa->d_st == 1); - /* walk thru list of non-basic variables */ - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == u[k]) - { /* xN[j] is fixed variable; skip it */ - continue; - } - /* determine absolute tolerance eps[j] */ - eps = tol + tol1 * (c[k] >= 0.0 ? +c[k] : -c[k]); - /* check dual feasibility of xN[j] */ - if (d[j] > +eps) - { /* xN[j] should have its lower bound active */ - if (l[k] == -DBL_MAX || flag[j]) - { /* but it either has no lower bound or its lower bound - * is inactive */ - if (l[k] == -DBL_MAX) - { /* cannot recover, since xN[j] has no lower bound */ - ret = j; - break; - } - /* recovering is possible */ - if (recov) - flag[j] = 0; - ret = -1; - } - } - else if (d[j] < -eps) - { /* xN[j] should have its upper bound active */ - if (!flag[j]) - { /* but it either has no upper bound or its upper bound - * is inactive */ - if (u[k] == +DBL_MAX) - { /* cannot recover, since xN[j] has no upper bound */ - ret = j; - break; - } - /* recovering is possible */ - if (recov) - flag[j] = 1; - ret = -1; - } - } - } - if (recov && ret) - { /* invalidate values of basic variables, since active bounds - * of non-basic variables have been changed */ - csa->beta_st = 0; - } - return ret; -} - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_vec - compute maximal relative error between two vectors -* -* This routine computes and returns maximal relative error between -* n-vectors x and y: -* -* err_max = max |x[i] - y[i]| / (1 + |x[i]|). -* -* NOTE: This routine is intended only for debugging purposes. */ - -static double err_in_vec(int n, const double x[], const double y[]) -{ int i; - double err, err_max; - err_max = 0.0; - for (i = 1; i <= n; i++) - { err = fabs(x[i] - y[i]) / (1.0 + fabs(x[i])); - if (err_max < err) - err_max = err; - } - return err_max; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_beta - compute maximal relative error in vector beta -* -* This routine computes and returns maximal relative error in vector -* of values of basic variables beta = (beta[i]). -* -* NOTE: This routine is intended only for debugging purposes. */ - -static double err_in_beta(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - double err, *beta; - beta = talloc(1+m, double); - spx_eval_beta(lp, beta); - err = err_in_vec(m, beta, csa->beta); - tfree(beta); - return err; -} -#endif - -#if CHECK_ACCURACY -static double err_in_r(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int i, k; - double err, *r; - r = talloc(1+m, double); - for (i = 1; i <= m; i++) - { k = lp->head[i]; - if (csa->beta[i] < lp->l[k]) - r[i] = lp->l[k] - csa->beta[i]; - else if (csa->beta[i] > lp->u[k]) - r[i] = lp->u[k] - csa->beta[i]; - else - r[i] = 0.0; - -if (fabs(r[i] - csa->r.vec[i]) > 1e-6) -printf("i = %d; r = %g; csa->r = %g\n", i, r[i], csa->r.vec[i]); - - - } - err = err_in_vec(m, r, csa->r.vec); - tfree(r); - return err; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_d - compute maximal relative error in vector d -* -* This routine computes and returns maximal relative error in vector -* of reduced costs of non-basic variables d = (d[j]). -* -* NOTE: This routine is intended only for debugging purposes. */ - -static double err_in_d(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - int j; - double err, *pi, *d; - pi = talloc(1+m, double); - d = talloc(1+n-m, double); - spx_eval_pi(lp, pi); - for (j = 1; j <= n-m; j++) - d[j] = spx_eval_dj(lp, pi, j); - err = err_in_vec(n-m, d, csa->d); - tfree(pi); - tfree(d); - return err; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* err_in_gamma - compute maximal relative error in vector gamma -* -* This routine computes and returns maximal relative error in vector -* of projected steepest edge weights gamma = (gamma[j]). -* -* NOTE: This routine is intended only for debugging purposes. */ - -static double err_in_gamma(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - SPYSE *se = csa->se; - int i; - double err, *gamma; - xassert(se != NULL); -gamma = talloc(1+m, double); - for (i = 1; i <= m; i++) - gamma[i] = spy_eval_gamma_i(lp, se, i); - err = err_in_vec(m, gamma, se->gamma); - tfree(gamma); - return err; -} -#endif - -#if CHECK_ACCURACY -/*********************************************************************** -* check_accuracy - check accuracy of basic solution components -* -* This routine checks accuracy of current basic solution components. -* -* NOTE: This routine is intended only for debugging purposes. */ - -static void check_accuracy(struct csa *csa) -{ double e_beta, e_r, e_d, e_gamma; - e_beta = err_in_beta(csa); - e_r = err_in_r(csa); - e_d = err_in_d(csa); - if (csa->se == NULL) - e_gamma = 0.; - else - e_gamma = err_in_gamma(csa); - xprintf("e_beta = %10.3e; e_r = %10.3e; e_d = %10.3e; e_gamma = %" - "10.3e\n", e_beta, e_r, e_d, e_gamma); - xassert(e_beta <= 1e-5 && e_d <= 1e-5 && e_gamma <= 1e-3); - return; -} -#endif - -#if 1 /* 30/III-2016 */ -static -void spy_eval_r(SPXLP *lp, const double beta[/*1+m*/], double tol, - double tol1, FVS *r) -{ /* this routine computes the vector of primal infeasibilities: - * - * ( lB[i] - beta[i] > 0, if beta[i] < lb[i] - * r[i] = { 0, if lb[i] <= beta[i] <= ub[i] - * ( ub[i] - beta[i] < 0, if beta[i] > ub[i] - * - * (this routine replaces spy_chuzr_sel) */ - int m = lp->m; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int *ind = r->ind; - double *vec = r->vec; - int i, k, nnz = 0; - double lk, uk, eps; - xassert(r->n == m); - /* walk thru the list of basic variables */ - for (i = 1; i <= m; i++) - { vec[i] = 0.0; - k = head[i]; /* x[k] = xB[i] */ - lk = l[k], uk = u[k]; - /* check primal feasibility */ - if (beta[i] < lk) - { /* determine absolute tolerance eps1[i] */ - eps = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] < lk - eps) - { /* lower bound is violated */ - ind[++nnz] = i; - vec[i] = lk - beta[i]; - } - } - else if (beta[i] > uk) - { /* determine absolute tolerance eps2[i] */ - eps = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] > uk + eps) - { /* upper bound is violated */ - ind[++nnz] = i; - vec[i] = uk - beta[i]; - } - } - } - r->nnz = nnz; - return; -} -#endif - -/*********************************************************************** -* choose_pivot - choose xB[p] and xN[q] -* -* Given the list of eligible basic variables this routine first -* chooses basic variable xB[p]. This choice is always possible, -* because the list is assumed to be non-empty. Then the routine -* computes p-th row T[p,*] of the simplex table T[i,j] and chooses -* non-basic variable xN[q]. If the pivot T[p,q] is small in magnitude, -* the routine attempts to choose another xB[p] and xN[q] in order to -* avoid badly conditioned adjacent bases. -* -* If the normal choice was made, the routine returns zero. Otherwise, -* if the long-step choice was made, the routine returns non-zero. */ - -#ifdef TIMING /* 31/III-2016 */ - -#include "choose_pivot.c" - -#else - -#define MIN_RATIO 0.0001 - -static int choose_pivot(struct csa *csa) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - SPXAT *at = csa->at; - SPXNT *nt = csa->nt; - double *beta = csa->beta; - double *d = csa->d; - SPYSE *se = csa->se; -#if 0 /* 30/III-2016 */ - int *list = csa->list; -#else - int *list = csa->r.ind; -#endif - double *rho = csa->work; - double *trow = csa->work1; - SPYBP *bp = csa->bp; - double tol_piv = csa->tol_piv; - int try, nnn, j, k, p, q, t, t_best, nbp, ret; - double big, temp, r, best_ratio, dz_best; - xassert(csa->beta_st); - xassert(csa->d_st); -more: /* initial number of eligible basic variables */ -#if 0 /* 30/III-2016 */ - nnn = csa->num; -#else - nnn = csa->r.nnz; -#endif - /* nothing has been chosen so far */ - csa->p = 0; - best_ratio = 0.0; - try = ret = 0; -try: /* choose basic variable xB[p] */ - xassert(nnn > 0); - try++; - if (se == NULL) - { /* dual Dantzig's rule */ - p = spy_chuzr_std(lp, beta, nnn, list); - } - else - { /* dual projected steepest edge */ - p = spy_chuzr_pse(lp, se, beta, nnn, list); - } - xassert(1 <= p && p <= m); - /* compute p-th row of inv(B) */ - spx_eval_rho(lp, p, rho); - /* compute p-th row of the simplex table */ - if (at != NULL) - spx_eval_trow1(lp, at, rho, trow); - else - spx_nt_prod(lp, nt, trow, 1, -1.0, rho); -#if 1 /* 23/III-2016 */ - /* big := max(1, |trow[1]|, ..., |trow[n-m]|) */ - big = 1.0; - for (j = 1; j <= n-m; j++) - { temp = trow[j]; - if (temp < 0.0) - temp = - temp; - if (big < temp) - big = temp; - } -#else - /* this still puzzles me */ - big = 1.0; -#endif - /* choose non-basic variable xN[q] */ - k = head[p]; /* x[k] = xB[p] */ - xassert(beta[p] < l[k] || beta[p] > u[k]); - r = beta[p] < l[k] ? l[k] - beta[p] : u[k] - beta[p]; - if (csa->r_test == GLP_RT_FLIP && try <= 2) - { /* long-step ratio test */ -#if 0 /* 23/III-2016 */ - /* determine dual objective break-points */ - nbp = spy_eval_bp(lp, d, r, trow, tol_piv, bp); - if (nbp <= 1) - goto skip; - /* choose appropriate break-point */ - t_best = 0, dz_best = -DBL_MAX; - for (t = 1; t <= nbp; t++) - { if (fabs(trow[bp[t].j]) / big >= MIN_RATIO) - { if (dz_best < bp[t].dz) - t_best = t, dz_best = bp[t].dz; - } - } - if (t_best == 0) - goto skip; -#else - int t, num, num1; - double slope, teta_lim; - /* determine dual objective break-points */ - nbp = spy_ls_eval_bp(lp, d, r, trow, tol_piv, bp); - if (nbp < 2) - goto skip; - /* set initial slope */ - slope = fabs(r); - /* estimate initial teta_lim */ - teta_lim = DBL_MAX; - for (t = 1; t <= nbp; t++) - { if (teta_lim > bp[t].teta) - teta_lim = bp[t].teta; - } - xassert(teta_lim >= 0.0); - if (teta_lim < 1e-6) - teta_lim = 1e-6; - /* nothing has been chosen so far */ - t_best = 0, dz_best = 0.0, num = 0; - /* choose appropriate break-point */ - while (num < nbp) - { /* select and process a new portion of break-points */ - num1 = spy_ls_select_bp(lp, trow, nbp, bp, num, &slope, - teta_lim); - for (t = num+1; t <= num1; t++) - { if (fabs(trow[bp[t].j]) / big >= MIN_RATIO) - { if (dz_best < bp[t].dz) - t_best = t, dz_best = bp[t].dz; - } - } - if (slope < 0.0) - { /* the dual objective starts decreasing */ - break; - } - /* the dual objective continues increasing */ - num = num1; - teta_lim += teta_lim; - } - if (dz_best == 0.0) - goto skip; - xassert(1 <= t_best && t_best <= num1); -#endif - /* the choice has been made */ - csa->p = p; -#if 0 /* 29/III-2016 */ - memcpy(&csa->trow[1], &trow[1], (n-m) * sizeof(double)); -#else - memcpy(&csa->trow.vec[1], &trow[1], (n-m) * sizeof(double)); - fvs_gather_vec(&csa->trow, DBL_EPSILON); -#endif - csa->q = bp[t_best].j; - best_ratio = fabs(trow[bp[t_best].j]) / big; -#if 0 - xprintf("num = %d; t_best = %d; dz = %g\n", num, t_best, - bp[t_best].dz); -#endif - ret = 1; - goto done; -skip: ; - } - if (csa->r_test == GLP_RT_STD) - { /* textbook dual ratio test */ - q = spy_chuzc_std(lp, d, r, trow, tol_piv, - .30 * csa->tol_dj, .30 * csa->tol_dj1); - } - else - { /* Harris' two-pass dual ratio test */ - q = spy_chuzc_harris(lp, d, r, trow, tol_piv, - .35 * csa->tol_dj, .35 * csa->tol_dj1); - } - if (q == 0) - { /* dual unboundedness */ - csa->p = p; -#if 0 /* 29/III-2016 */ - memcpy(&csa->trow[1], &trow[1], (n-m) * sizeof(double)); -#else - memcpy(&csa->trow.vec[1], &trow[1], (n-m) * sizeof(double)); - fvs_gather_vec(&csa->trow, DBL_EPSILON); -#endif - csa->q = q; - best_ratio = 1.0; - goto done; - } - /* either keep previous choice or accept new choice depending on - * which one is better */ - if (best_ratio < fabs(trow[q]) / big) - { csa->p = p; -#if 0 /* 29/III-2016 */ - memcpy(&csa->trow[1], &trow[1], (n-m) * sizeof(double)); -#else - memcpy(&csa->trow.vec[1], &trow[1], (n-m) * sizeof(double)); - fvs_gather_vec(&csa->trow, DBL_EPSILON); -#endif - csa->q = q; - best_ratio = fabs(trow[q]) / big; - } - /* check if the current choice is acceptable */ - if (best_ratio >= MIN_RATIO || nnn == 1 || try == 5) - goto done; - /* try to choose other xB[p] and xN[q] */ - /* find xB[p] in the list */ - for (t = 1; t <= nnn; t++) - if (list[t] == p) break; - xassert(t <= nnn); - /* move xB[p] to the end of the list */ - list[t] = list[nnn], list[nnn] = p; - /* and exclude it from consideration */ - nnn--; - /* repeat the choice */ - goto try; -done: /* the choice has been made */ -#if 1 /* FIXME: currently just to avoid badly conditioned basis */ - if (best_ratio < .001 * MIN_RATIO) - { /* looks like this helps */ - if (bfd_get_count(lp->bfd) > 0) - return -1; - /* didn't help; last chance to improve the choice */ - if (tol_piv == csa->tol_piv) - { tol_piv *= 1000.; - goto more; - } - } -#endif -#if 1 /* FIXME */ - if (ret) - { /* invalidate basic solution components */ -#if 0 /* 28/III-2016 */ - csa->beta_st = csa->d_st = 0; -#else - /* dual solution remains valid */ - csa->beta_st = 0; -#endif - /* set double-bounded non-basic variables to opposite bounds - * for all break-points preceding the chosen one */ - for (t = 1; t < t_best; t++) - { k = head[m + bp[t].j]; - xassert(-DBL_MAX < l[k] && l[k] < u[k] && u[k] < +DBL_MAX); - lp->flag[bp[t].j] = !(lp->flag[bp[t].j]); - } - } -#endif - return ret; -} - -#endif - -/*********************************************************************** -* play_coef - play objective coefficients -* -* This routine is called after the reduced costs d[j] was updated and -* the basis was changed to the adjacent one. -* -* It is assumed that before updating all the reduced costs d[j] were -* strongly feasible, so in the adjacent basis d[j] remain feasible -* within a tolerance, i.e. if some d[j] violates its zero bound, the -* violation is insignificant. -* -* If some d[j] violates its zero bound, the routine changes (perturbs) -* objective coefficient cN[j] to provide d[j] = 0, i.e. to make all -* d[j] strongly feasible. Otherwise, if d[j] has a feasible value, the -* routine attempts to reduce (or remove) perturbation in cN[j] by -* shifting d[j] to its zero bound keeping strong feasibility. */ - -static void play_coef(struct csa *csa, int all) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *c = lp->c; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - char *flag = lp->flag; - double *orig_c = csa->orig_c; - double *d = csa->d; - const double *trow = csa->trow.vec; - /* this vector was used to update d = (d[j]) */ - int j, k; - static const double eps = 1e-9; - /* reduced costs d = (d[j]) should be valid */ - xassert(csa->d_st); - /* walk thru the list of non-basic variables xN = (xN[j]) */ - for (j = 1; j <= n-m; j++) - { if (all || trow[j] != 0.0) - { /* d[j] has changed in the adjacent basis */ - k = head[m+j]; /* x[k] = xN[j] */ - if (l[k] == u[k]) - { /* xN[j] is fixed variable */ - /* d[j] may have any sign */ - } - else if (l[k] == -DBL_MAX && u[k] == +DBL_MAX) - { /* xN[j] is free (unbounded) variable */ - /* strong feasibility means d[j] = 0 */ - c[k] -= d[j], d[j] = 0.0; - /* in this case dual degeneracy is not critical, since - * if xN[j] enters the basis, it never leaves it */ - } - else if (!flag[j]) - { /* xN[j] has its lower bound active */ - xassert(l[k] != -DBL_MAX); - /* first, we remove current perturbation to provide - * c[k] = orig_c[k] */ - d[j] -= c[k] - orig_c[k], c[k] = orig_c[k]; - /* strong feasibility means d[j] >= 0, but we provide - * d[j] >= +eps to prevent dual degeneracy */ - if (d[j] < +eps) - c[k] -= d[j] - eps, d[j] = +eps; - } - else - { /* xN[j] has its upper bound active */ - xassert(u[k] != +DBL_MAX); - /* similarly, we remove current perturbation to provide - * c[k] = orig_c[k] */ - d[j] -= c[k] - orig_c[k], c[k] = orig_c[k]; - /* strong feasibility means d[j] <= 0, but we provide - * d[j] <= -eps to prevent dual degeneracy */ - if (d[j] > -eps) - c[k] -= d[j] + eps, d[j] = -eps; - } - } - } - return; -} - -#if 1 /* 11/VII-2017 */ -static void remove_perturb(struct csa *csa) -{ /* remove perturbation */ - SPXLP *lp = csa->lp; - int n = lp->n; - double *c = lp->c; - double *orig_c = csa->orig_c; - memcpy(c, orig_c, (1+n) * sizeof(double)); - /* removing perturbation changes dual solution components */ - csa->phase = csa->d_st = 0; -#if 1 - if (csa->msg_lev >= GLP_MSG_ALL) - xprintf("Removing LP perturbation [%d]...\n", - csa->it_cnt); -#endif - return; -} -#endif - -/*********************************************************************** -* display - display search progress -* -* This routine displays some information about the search progress -* that includes: -* -* search phase; -* -* number of simplex iterations performed by the solver; -* -* original objective value (only on phase II); -* -* sum of (scaled) dual infeasibilities for original bounds; -* -* number of dual infeasibilities (phase I) or primal infeasibilities -* (phase II); -* -* number of basic factorizations since last display output. */ - -static void display(struct csa *csa, int spec) -{ SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - int *head = lp->head; - char *flag = lp->flag; - double *l = csa->orig_l; /* original lower bounds */ - double *u = csa->orig_u; /* original upper bounds */ - double *beta = csa->beta; - double *d = csa->d; - int j, k, nnn; - double sum; -#if 1 /* 15/VII-2017 */ - double tm_cur; -#endif - /* check if the display output should be skipped */ - if (csa->msg_lev < GLP_MSG_ON) goto skip; -#if 1 /* 15/VII-2017 */ - tm_cur = xtime(); -#endif - if (csa->out_dly > 0 && -#if 0 /* 15/VII-2017 */ - 1000.0 * xdifftime(xtime(), csa->tm_beg) < csa->out_dly) -#else - 1000.0 * xdifftime(tm_cur, csa->tm_beg) < csa->out_dly) -#endif - goto skip; - if (csa->it_cnt == csa->it_dpy) goto skip; -#if 0 /* 15/VII-2017 */ - if (!spec && csa->it_cnt % csa->out_frq != 0) goto skip; -#else - if (!spec && - 1000.0 * xdifftime(tm_cur, csa->tm_dpy) < csa->out_frq) - goto skip; -#endif - /* display search progress depending on search phase */ - switch (csa->phase) - { case 1: - /* compute sum and number of (scaled) dual infeasibilities - * for original bounds */ - sum = 0.0, nnn = 0; - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (d[j] > 0.0) - { /* xN[j] should have lower bound */ - if (l[k] == -DBL_MAX) - { sum += d[j]; - if (d[j] > +1e-7) - nnn++; - } - } - else if (d[j] < 0.0) - { /* xN[j] should have upper bound */ - if (u[k] == +DBL_MAX) - { sum -= d[j]; - if (d[j] < -1e-7) - nnn++; - } - } - } - /* on phase I variables have artificial bounds which are - * meaningless for original LP, so corresponding objective - * function value is also meaningless */ -#if 0 /* 27/III-2016 */ - xprintf(" %6d: %23s inf = %11.3e (%d)", - csa->it_cnt, "", sum, nnn); -#else - xprintf(" %6d: sum = %17.9e inf = %11.3e (%d)", - csa->it_cnt, lp->c[0] - spx_eval_obj(lp, beta), - sum, nnn); -#endif - break; - case 2: - /* compute sum of (scaled) dual infeasibilities */ - sum = 0.0, nnn = 0; - for (j = 1; j <= n-m; j++) - { k = head[m+j]; /* x[k] = xN[j] */ - if (d[j] > 0.0) - { /* xN[j] should have its lower bound active */ - if (l[k] == -DBL_MAX || flag[j]) - sum += d[j]; - } - else if (d[j] < 0.0) - { /* xN[j] should have its upper bound active */ - if (l[k] != u[k] && !flag[j]) - sum -= d[j]; - } - } - /* compute number of primal infeasibilities */ - nnn = spy_chuzr_sel(lp, beta, csa->tol_bnd, csa->tol_bnd1, - NULL); - xprintf("#%6d: obj = %17.9e inf = %11.3e (%d)", -#if SCALE_Z - csa->it_cnt, - (double)csa->dir * csa->fz * spx_eval_obj(lp, beta), -#else - csa->it_cnt, (double)csa->dir * spx_eval_obj(lp, beta), -#endif - sum, nnn); - break; - default: - xassert(csa != csa); - } - if (csa->inv_cnt) - { /* number of basis factorizations performed */ - xprintf(" %d", csa->inv_cnt); - csa->inv_cnt = 0; - } -#if 1 /* 23/III-2016 */ - if (csa->r_test == GLP_RT_FLIP) - { /*xprintf(" %d,%d", csa->ns_cnt, csa->ls_cnt);*/ - if (csa->ns_cnt + csa->ls_cnt) - xprintf(" %d%%", - (100 * csa->ls_cnt) / (csa->ns_cnt + csa->ls_cnt)); - csa->ns_cnt = csa->ls_cnt = 0; - } -#endif - xprintf("\n"); - csa->it_dpy = csa->it_cnt; -#if 1 /* 15/VII-2017 */ - csa->tm_dpy = tm_cur; -#endif -skip: return; -} - -#if 1 /* 31/III-2016 */ -static -void spy_update_r(SPXLP *lp, int p, int q, const double beta[/*1+m*/], - const FVS *tcol, double tol, double tol1, FVS *r) -{ /* update vector r of primal infeasibilities */ - /* it is assumed that xB[p] leaves the basis, xN[q] enters the - * basis, and beta corresponds to the adjacent basis (i.e. this - * routine should be called after spx_update_beta) */ - int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - int *tcol_ind = tcol->ind; - int *ind = r->ind; - double *vec = r->vec; - int i, k, t, nnz; - double lk, uk, ri, eps; - xassert(1 <= p && p <= m); - xassert(1 <= q && q <= n-m); - nnz = r->nnz; - for (t = tcol->nnz; t >= 1; t--) - { i = tcol_ind[t]; - /* xB[i] changes in the adjacent basis to beta[i], so only - * r[i] should be updated */ - if (i == p) - k = head[m+q]; /* x[k] = new xB[p] = old xN[q] */ - else - k = head[i]; /* x[k] = new xB[i] = old xB[i] */ - lk = l[k], uk = u[k]; - /* determine new value of r[i]; see spy_eval_r */ - ri = 0.0; - if (beta[i] < lk) - { /* determine absolute tolerance eps1[i] */ - eps = tol + tol1 * (lk >= 0.0 ? +lk : -lk); - if (beta[i] < lk - eps) - { /* lower bound is violated */ - ri = lk - beta[i]; - } - } - else if (beta[i] > uk) - { /* determine absolute tolerance eps2[i] */ - eps = tol + tol1 * (uk >= 0.0 ? +uk : -uk); - if (beta[i] > uk + eps) - { /* upper bound is violated */ - ri = uk - beta[i]; - } - } - if (ri == 0.0) - { if (vec[i] != 0.0) - vec[i] = DBL_MIN; /* will be removed */ - } - else - { if (vec[i] == 0.0) - ind[++nnz] = i; - vec[i] = ri; - } - - } - r->nnz = nnz; - /* remove zero elements */ - fvs_adjust_vec(r, DBL_MIN + DBL_MIN); - return; -} -#endif - -/*********************************************************************** -* spy_dual - driver to the dual simplex method -* -* This routine is a driver to the two-phase dual simplex method. -* -* On exit this routine returns one of the following codes: -* -* 0 LP instance has been successfully solved. -* -* GLP_EOBJLL -* Objective lower limit has been reached (maximization). -* -* GLP_EOBJUL -* Objective upper limit has been reached (minimization). -* -* GLP_EITLIM -* Iteration limit has been exhausted. -* -* GLP_ETMLIM -* Time limit has been exhausted. -* -* GLP_EFAIL -* The solver failed to solve LP instance. */ - -static int dual_simplex(struct csa *csa) -{ /* dual simplex method main logic routine */ - SPXLP *lp = csa->lp; - int m = lp->m; - int n = lp->n; - double *l = lp->l; - double *u = lp->u; - int *head = lp->head; - SPXNT *nt = csa->nt; - double *beta = csa->beta; - double *d = csa->d; - SPYSE *se = csa->se; -#if 0 /* 30/III-2016 */ - int *list = csa->list; -#endif -#if 0 /* 31/III-2016 */ - double *trow = csa->trow; - double *tcol = csa->tcol; -#endif - double *pi = csa->work; - int msg_lev = csa->msg_lev; - double tol_bnd = csa->tol_bnd; - double tol_bnd1 = csa->tol_bnd1; - double tol_dj = csa->tol_dj; - double tol_dj1 = csa->tol_dj1; - int j, k, p_flag, refct, ret; - int perturb = -1; - /* -1 = perturbation is not used, but enabled - * 0 = perturbation is not used and disabled - * +1 = perturbation is being used */ -#if 1 /* 27/III-2016 */ - int instab = 0; /* instability count */ -#endif -#ifdef TIMING - double t_total = timer(); /* total time */ - double t_fact = 0.0; /* computing factorization */ - double t_rtest = 0.0; /* performing ratio test */ - double t_pivcol = 0.0; /* computing pivot column */ - double t_upd1 = 0.0; /* updating primal values */ - double t_upd2 = 0.0; /* updating dual values */ - double t_upd3 = 0.0; /* updating se weights */ - double t_upd4 = 0.0; /* updating matrix N */ - double t_upd5 = 0.0; /* updating factorization */ - double t_start; -#endif - check_flags(csa); -loop: /* main loop starts here */ - /* compute factorization of the basis matrix */ - if (!lp->valid) - { double cond; -#ifdef TIMING - t_start = timer(); -#endif - ret = spx_factorize(lp); -#ifdef TIMING - t_fact += timer() - t_start; -#endif - csa->inv_cnt++; - if (ret != 0) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: unable to factorize the basis matrix (%d" - ")\n", ret); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - } - /* check condition of the basis matrix */ - cond = bfd_condest(lp->bfd); - if (cond > 1.0 / DBL_EPSILON) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: basis matrix is singular to working prec" - "ision (cond = %.3g)\n", cond); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - } - if (cond > 0.001 / DBL_EPSILON) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Warning: basis matrix is ill-conditioned (cond " - "= %.3g)\n", cond); - } - /* invalidate basic solution components */ - csa->beta_st = csa->d_st = 0; - } - /* compute reduced costs of non-basic variables d = (d[j]) */ - if (!csa->d_st) - { spx_eval_pi(lp, pi); - for (j = 1; j <= n-m; j++) - d[j] = spx_eval_dj(lp, pi, j); - csa->d_st = 1; /* just computed */ - /* determine the search phase, if not determined yet (this is - * performed only once at the beginning of the search for the - * original bounds) */ - if (!csa->phase) - { j = check_feas(csa, 0.97 * tol_dj, 0.97 * tol_dj1, 1); - if (j > 0) - { /* initial basic solution is dual infeasible and cannot - * be recovered */ - /* start to search for dual feasible solution */ - set_art_bounds(csa); - csa->phase = 1; - } - else - { /* initial basic solution is either dual feasible or its - * dual feasibility has been recovered */ - /* start to search for optimal solution */ - csa->phase = 2; - } - } - /* make sure that current basic solution is dual feasible */ -#if 1 /* 11/VII-2017 */ - if (perturb <= 0) - { if (check_feas(csa, tol_dj, tol_dj1, 0)) - { /* dual feasibility is broken due to excessive round-off - * errors */ - if (perturb < 0) - { if (msg_lev >= GLP_MSG_ALL) - xprintf("Perturbing LP to avoid instability [%d].." - ".\n", csa->it_cnt); - perturb = 1; - goto loop; - } - if (msg_lev >= GLP_MSG_ERR) - xprintf("Warning: numerical instability (dual simplex" - ", phase %s)\n", csa->phase == 1 ? "I" : "II"); - instab++; - if (csa->dualp && instab >= 10) - { /* do not continue the search; report failure */ - if (msg_lev >= GLP_MSG_ERR) - xprintf("Warning: dual simplex failed due to exces" - "sive numerical instability\n"); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = -1; /* special case of GLP_EFAIL */ - goto fini; - } - /* try to recover dual feasibility */ - j = check_feas(csa, 0.97 * tol_dj, 0.97 * tol_dj1, 1); - if (j > 0) - { /* dual feasibility cannot be recovered (this may - * happen only on phase II) */ - xassert(csa->phase == 2); - /* restart to search for dual feasible solution */ - set_art_bounds(csa); - csa->phase = 1; - } - } - } - else - { /* FIXME */ - play_coef(csa, 1); - } - } -#endif - /* at this point the search phase is determined */ - xassert(csa->phase == 1 || csa->phase == 2); - /* compute values of basic variables beta = (beta[i]) */ - if (!csa->beta_st) - { spx_eval_beta(lp, beta); -#if 1 /* 31/III-2016 */ - /* also compute vector r of primal infeasibilities */ - switch (csa->phase) - { case 1: - spy_eval_r(lp, beta, 1e-8, 0.0, &csa->r); - break; - case 2: - spy_eval_r(lp, beta, tol_bnd, tol_bnd1, &csa->r); - break; - default: - xassert(csa != csa); - } -#endif - csa->beta_st = 1; /* just computed */ - } - /* reset the dual reference space, if necessary */ - if (se != NULL && !se->valid) - spy_reset_refsp(lp, se), refct = 1000; - /* at this point the basis factorization and all basic solution - * components are valid */ - xassert(lp->valid && csa->beta_st && csa->d_st); -#ifdef GLP_DEBUG - check_flags(csa); -#endif -#if CHECK_ACCURACY - /* check accuracy of current basic solution components (only for - * debugging) */ - check_accuracy(csa); -#endif - /* check if the objective limit has been reached */ - if (csa->phase == 2 && csa->obj_lim != DBL_MAX - && spx_eval_obj(lp, beta) >= csa->obj_lim) - { -#if 1 /* 26/V-2017 by mao */ - if (perturb > 0) - { /* remove perturbation */ - /* [Should note that perturbing of objective coefficients - * implemented in play_coef is equivalent to *relaxing* of - * (zero) bounds of dual variables, so the perturbed - * objective is always better (*greater*) that the original - * one at the same basic point.] */ - remove_perturb(csa); - perturb = 0; - } -#endif - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - if (msg_lev >= GLP_MSG_ALL) - xprintf("OBJECTIVE %s LIMIT REACHED; SEARCH TERMINATED\n", - csa->dir > 0 ? "UPPER" : "LOWER"); -#if 0 /* 30/III-2016 */ - csa->num = spy_chuzr_sel(lp, beta, tol_bnd, tol_bnd1, list); - csa->p_stat = (csa->num == 0 ? GLP_FEAS : GLP_INFEAS); -#else - spy_eval_r(lp, beta, tol_bnd, tol_bnd1, &csa->r); - csa->p_stat = (csa->r.nnz == 0 ? GLP_FEAS : GLP_INFEAS); -#endif - csa->d_stat = GLP_FEAS; - ret = (csa->dir > 0 ? GLP_EOBJUL : GLP_EOBJLL); - goto fini; - } - /* check if the iteration limit has been exhausted */ - if (csa->it_cnt - csa->it_beg >= csa->it_lim) - { if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - if (msg_lev >= GLP_MSG_ALL) - xprintf("ITERATION LIMIT EXCEEDED; SEARCH TERMINATED\n"); - if (csa->phase == 1) - { set_orig_bounds(csa); - check_flags(csa); - spx_eval_beta(lp, beta); - } -#if 0 /* 30/III-2016 */ - csa->num = spy_chuzr_sel(lp, beta, tol_bnd, tol_bnd1, list); - csa->p_stat = (csa->num == 0 ? GLP_FEAS : GLP_INFEAS); -#else - spy_eval_r(lp, beta, tol_bnd, tol_bnd1, &csa->r); - csa->p_stat = (csa->r.nnz == 0 ? GLP_FEAS : GLP_INFEAS); -#endif - csa->d_stat = (csa->phase == 1 ? GLP_INFEAS : GLP_FEAS); - ret = GLP_EITLIM; - goto fini; - } - /* check if the time limit has been exhausted */ - if (1000.0 * xdifftime(xtime(), csa->tm_beg) >= csa->tm_lim) - { if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - if (msg_lev >= GLP_MSG_ALL) - xprintf("TIME LIMIT EXCEEDED; SEARCH TERMINATED\n"); - if (csa->phase == 1) - { set_orig_bounds(csa); - check_flags(csa); - spx_eval_beta(lp, beta); - } -#if 0 /* 30/III-2016 */ - csa->num = spy_chuzr_sel(lp, beta, tol_bnd, tol_bnd1, list); - csa->p_stat = (csa->num == 0 ? GLP_FEAS : GLP_INFEAS); -#else - spy_eval_r(lp, beta, tol_bnd, tol_bnd1, &csa->r); - csa->p_stat = (csa->r.nnz == 0 ? GLP_FEAS : GLP_INFEAS); -#endif - csa->d_stat = (csa->phase == 1 ? GLP_INFEAS : GLP_FEAS); - ret = GLP_ETMLIM; - goto fini; - } - /* display the search progress */ - display(csa, 0); - /* select eligible basic variables */ -#if 0 /* 31/III-2016; not needed because r is valid */ - switch (csa->phase) - { case 1: -#if 0 /* 30/III-2016 */ - csa->num = spy_chuzr_sel(lp, beta, 1e-8, 0.0, list); -#else - spy_eval_r(lp, beta, 1e-8, 0.0, &csa->r); -#endif - break; - case 2: -#if 0 /* 30/III-2016 */ - csa->num = spy_chuzr_sel(lp, beta, tol_bnd, tol_bnd1, list); -#else - spy_eval_r(lp, beta, tol_bnd, tol_bnd1, &csa->r); -#endif - break; - default: - xassert(csa != csa); - } -#endif - /* check for optimality */ -#if 0 /* 30/III-2016 */ - if (csa->num == 0) -#else - if (csa->r.nnz == 0) -#endif - { if (perturb > 0 && csa->phase == 2) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - /* current basis is optimal */ - display(csa, 1); - switch (csa->phase) - { case 1: - /* check for dual feasibility */ - set_orig_bounds(csa); - check_flags(csa); - if (check_feas(csa, tol_dj, tol_dj1, 0) == 0) - { /* dual feasible solution found; switch to phase II */ - csa->phase = 2; - xassert(!csa->beta_st); - goto loop; - } -#if 1 /* 26/V-2017 by cmatraki */ - if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - goto loop; - } -#endif - /* no dual feasible solution exists */ - if (msg_lev >= GLP_MSG_ALL) - xprintf("LP HAS NO DUAL FEASIBLE SOLUTION\n"); - spx_eval_beta(lp, beta); -#if 0 /* 30/III-2016 */ - csa->num = spy_chuzr_sel(lp, beta, tol_bnd, tol_bnd1, - list); - csa->p_stat = (csa->num == 0 ? GLP_FEAS : GLP_INFEAS); -#else - spy_eval_r(lp, beta, tol_bnd, tol_bnd1, &csa->r); - csa->p_stat = (csa->r.nnz == 0 ? GLP_FEAS : GLP_INFEAS); -#endif - csa->d_stat = GLP_NOFEAS; - ret = 0; - goto fini; - case 2: - /* optimal solution found */ - if (msg_lev >= GLP_MSG_ALL) - xprintf("OPTIMAL LP SOLUTION FOUND\n"); - csa->p_stat = csa->d_stat = GLP_FEAS; - ret = 0; - goto fini; - default: - xassert(csa != csa); - } - } - /* choose xB[p] and xN[q] */ -#if 0 /* 23/III-2016 */ - choose_pivot(csa); -#else -#ifdef TIMING - t_start = timer(); -#endif -#if 1 /* 31/III-2016 */ - ret = choose_pivot(csa); -#endif -#ifdef TIMING - t_rtest += timer() - t_start; -#endif - if (ret < 0) - { lp->valid = 0; - goto loop; - } - if (ret == 0) - csa->ns_cnt++; - else - csa->ls_cnt++; -#endif - /* check for dual unboundedness */ - if (csa->q == 0) - { if (perturb > 0) - { /* remove perturbation */ - remove_perturb(csa); - perturb = 0; - } - if (csa->beta_st != 1) - csa->beta_st = 0; - if (csa->d_st != 1) - csa->d_st = 0; - if (!(csa->beta_st && csa->d_st)) - goto loop; - display(csa, 1); - switch (csa->phase) - { case 1: - /* this should never happen */ - if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: dual simplex failed\n"); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - case 2: - /* dual unboundedness detected */ - if (msg_lev >= GLP_MSG_ALL) - xprintf("LP HAS NO PRIMAL FEASIBLE SOLUTION\n"); - csa->p_stat = GLP_NOFEAS; - csa->d_stat = GLP_FEAS; - ret = 0; - goto fini; - default: - xassert(csa != csa); - } - } - /* compute q-th column of the simplex table */ -#ifdef TIMING - t_start = timer(); -#endif -#if 0 /* 31/III-2016 */ - spx_eval_tcol(lp, csa->q, tcol); -#else - spx_eval_tcol(lp, csa->q, csa->tcol.vec); - fvs_gather_vec(&csa->tcol, DBL_EPSILON); -#endif -#ifdef TIMING - t_pivcol += timer() - t_start; -#endif - /* FIXME: tcol[p] and trow[q] should be close to each other */ -#if 0 /* 26/V-2017 by cmatraki */ - xassert(csa->tcol.vec[csa->p] != 0.0); -#else - if (csa->tcol.vec[csa->p] == 0.0) - { if (msg_lev >= GLP_MSG_ERR) - xprintf("Error: tcol[p] = 0.0\n"); - csa->p_stat = csa->d_stat = GLP_UNDEF; - ret = GLP_EFAIL; - goto fini; - } -#endif - /* update values of basic variables for adjacent basis */ - k = head[csa->p]; /* x[k] = xB[p] */ - p_flag = (l[k] != u[k] && beta[csa->p] > u[k]); -#if 0 /* 16/III-2016 */ - spx_update_beta(lp, beta, csa->p, p_flag, csa->q, tcol); - csa->beta_st = 2; -#else - /* primal solution may be invalidated due to long step */ -#ifdef TIMING - t_start = timer(); -#endif - if (csa->beta_st) -#if 0 /* 30/III-2016 */ - { spx_update_beta(lp, beta, csa->p, p_flag, csa->q, tcol); -#else - { spx_update_beta_s(lp, beta, csa->p, p_flag, csa->q, - &csa->tcol); - /* also update vector r of primal infeasibilities */ - /*fvs_check_vec(&csa->r);*/ - switch (csa->phase) - { case 1: - spy_update_r(lp, csa->p, csa->q, beta, &csa->tcol, - 1e-8, 0.0, &csa->r); - break; - case 2: - spy_update_r(lp, csa->p, csa->q, beta, &csa->tcol, - tol_bnd, tol_bnd1, &csa->r); - break; - default: - xassert(csa != csa); - } - /*fvs_check_vec(&csa->r);*/ -#endif - csa->beta_st = 2; - } -#ifdef TIMING - t_upd1 += timer() - t_start; -#endif -#endif -#if 1 /* 11/VII-2017 */ - /* check for stalling */ - { int k; - xassert(1 <= csa->p && csa->p <= m); - xassert(1 <= csa->q && csa->q <= n-m); - /* FIXME: recompute d[q]; see spx_update_d */ - k = head[m+csa->q]; /* x[k] = xN[q] */ - if (!(lp->l[k] == -DBL_MAX && lp->u[k] == +DBL_MAX)) - { if (fabs(d[csa->q]) >= 1e-6) - { csa->degen = 0; - goto skip1; - } - /* degenerate iteration has been detected */ - csa->degen++; - if (perturb < 0 && csa->degen >= 200) - { if (msg_lev >= GLP_MSG_ALL) - xprintf("Perturbing LP to avoid stalling [%d]...\n", - csa->it_cnt); - perturb = 1; - } -skip1: ; - } - } -#endif - /* update reduced costs of non-basic variables for adjacent - * basis */ -#if 1 /* 28/III-2016 */ - xassert(csa->d_st); -#endif -#ifdef TIMING - t_start = timer(); -#endif -#if 0 /* 30/III-2016 */ - if (spx_update_d(lp, d, csa->p, csa->q, trow, tcol) <= 1e-9) -#else - if (spx_update_d_s(lp, d, csa->p, csa->q, &csa->trow, &csa->tcol) - <= 1e-9) -#endif - { /* successful updating */ - csa->d_st = 2; - } - else - { /* new reduced costs are inaccurate */ - csa->d_st = 0; - } -#ifdef TIMING - t_upd2 += timer() - t_start; -#endif - /* update steepest edge weights for adjacent basis, if used */ -#ifdef TIMING - t_start = timer(); -#endif - if (se != NULL) - { if (refct > 0) -#if 0 /* 30/III-2016 */ - { if (spy_update_gamma(lp, se, csa->p, csa->q, trow, tcol) - <= 1e-3) -#else - { if (spy_update_gamma_s(lp, se, csa->p, csa->q, &csa->trow, - &csa->tcol) <= 1e-3) -#endif - { /* successful updating */ - refct--; - } - else - { /* new weights are inaccurate; reset reference space */ - se->valid = 0; - } - } - else - { /* too many updates; reset reference space */ - se->valid = 0; - } - } -#ifdef TIMING - t_upd3 += timer() - t_start; -#endif -#ifdef TIMING - t_start = timer(); -#endif - /* update matrix N for adjacent basis, if used */ - if (nt != NULL) - spx_update_nt(lp, nt, csa->p, csa->q); -#ifdef TIMING - t_upd4 += timer() - t_start; -#endif - /* change current basis header to adjacent one */ - spx_change_basis(lp, csa->p, p_flag, csa->q); - /* and update factorization of the basis matrix */ -#ifdef TIMING - t_start = timer(); -#endif -#if 0 /* 16/III-2016 */ - if (csa->p > 0) -#endif - spx_update_invb(lp, csa->p, head[csa->p]); -#ifdef TIMING - t_upd5 += timer() - t_start; -#endif - if (perturb > 0 && csa->d_st) - play_coef(csa, 0); - /* dual simplex iteration complete */ - csa->it_cnt++; - goto loop; -fini: -#ifdef TIMING - t_total = timer() - t_total; - xprintf("Total time = %10.3f\n", t_total); - xprintf("Factorization = %10.3f\n", t_fact); - xprintf("Ratio test = %10.3f\n", t_rtest); - xprintf("Pivot column = %10.3f\n", t_pivcol); - xprintf("Updating beta = %10.3f\n", t_upd1); - xprintf("Updating d = %10.3f\n", t_upd2); - xprintf("Updating gamma = %10.3f\n", t_upd3); - xprintf("Updating N = %10.3f\n", t_upd4); - xprintf("Updating inv(B) = %10.3f\n", t_upd5); -#endif - return ret; -} - -int spy_dual(glp_prob *P, const glp_smcp *parm) -{ /* driver to the dual simplex method */ - struct csa csa_, *csa = &csa_; - SPXLP lp; - SPXAT at; - SPXNT nt; - SPYSE se; - int ret, *map, *daeh; -#if SCALE_Z - int i, j, k; -#endif - /* build working LP and its initial basis */ - memset(csa, 0, sizeof(struct csa)); - csa->lp = &lp; - spx_init_lp(csa->lp, P, parm->excl); - spx_alloc_lp(csa->lp); - map = talloc(1+P->m+P->n, int); - spx_build_lp(csa->lp, P, parm->excl, parm->shift, map); - spx_build_basis(csa->lp, P, map); - switch (P->dir) - { case GLP_MIN: - csa->dir = +1; - break; - case GLP_MAX: - csa->dir = -1; - break; - default: - xassert(P != P); - } -#if SCALE_Z - csa->fz = 0.0; - for (k = 1; k <= csa->lp->n; k++) - { double t = fabs(csa->lp->c[k]); - if (csa->fz < t) - csa->fz = t; - } - if (csa->fz <= 1000.0) - csa->fz = 1.0; - else - csa->fz /= 1000.0; - /*xprintf("csa->fz = %g\n", csa->fz);*/ - for (k = 0; k <= csa->lp->n; k++) - csa->lp->c[k] /= csa->fz; -#endif - csa->orig_b = talloc(1+csa->lp->m, double); - memcpy(csa->orig_b, csa->lp->b, (1+csa->lp->m) * sizeof(double)); - csa->orig_c = talloc(1+csa->lp->n, double); - memcpy(csa->orig_c, csa->lp->c, (1+csa->lp->n) * sizeof(double)); - csa->orig_l = talloc(1+csa->lp->n, double); - memcpy(csa->orig_l, csa->lp->l, (1+csa->lp->n) * sizeof(double)); - csa->orig_u = talloc(1+csa->lp->n, double); - memcpy(csa->orig_u, csa->lp->u, (1+csa->lp->n) * sizeof(double)); - switch (parm->aorn) - { case GLP_USE_AT: - /* build matrix A in row-wise format */ - csa->at = &at; - csa->nt = NULL; - spx_alloc_at(csa->lp, csa->at); - spx_build_at(csa->lp, csa->at); - break; - case GLP_USE_NT: - /* build matrix N in row-wise format for initial basis */ - csa->at = NULL; - csa->nt = &nt; - spx_alloc_nt(csa->lp, csa->nt); - spx_init_nt(csa->lp, csa->nt); - spx_build_nt(csa->lp, csa->nt); - break; - default: - xassert(parm != parm); - } - /* allocate and initialize working components */ - csa->phase = 0; - csa->beta = talloc(1+csa->lp->m, double); - csa->beta_st = 0; - csa->d = talloc(1+csa->lp->n-csa->lp->m, double); - csa->d_st = 0; - switch (parm->pricing) - { case GLP_PT_STD: - csa->se = NULL; - break; - case GLP_PT_PSE: - csa->se = &se; - spy_alloc_se(csa->lp, csa->se); - break; - default: - xassert(parm != parm); - } -#if 0 /* 30/III-2016 */ - csa->list = talloc(1+csa->lp->m, int); - csa->trow = talloc(1+csa->lp->n-csa->lp->m, double); - csa->tcol = talloc(1+csa->lp->m, double); -#else - fvs_alloc_vec(&csa->r, csa->lp->m); - fvs_alloc_vec(&csa->trow, csa->lp->n-csa->lp->m); - fvs_alloc_vec(&csa->tcol, csa->lp->m); -#endif -#if 1 /* 16/III-2016 */ - csa->bp = NULL; -#endif - csa->work = talloc(1+csa->lp->m, double); - csa->work1 = talloc(1+csa->lp->n-csa->lp->m, double); -#if 0 /* 11/VI-2017 */ -#if 1 /* 31/III-2016 */ - fvs_alloc_vec(&csa->wrow, csa->lp->n-csa->lp->m); - fvs_alloc_vec(&csa->wcol, csa->lp->m); -#endif -#endif - /* initialize control parameters */ - csa->msg_lev = parm->msg_lev; - csa->dualp = (parm->meth == GLP_DUALP); -#if 0 /* 16/III-2016 */ - switch (parm->r_test) - { case GLP_RT_STD: - csa->harris = 0; - break; - case GLP_RT_HAR: - csa->harris = 1; - break; - default: - xassert(parm != parm); - } -#else - switch (parm->r_test) - { case GLP_RT_STD: - case GLP_RT_HAR: - break; - case GLP_RT_FLIP: - csa->bp = talloc(1+csa->lp->n-csa->lp->m, SPYBP); - break; - default: - xassert(parm != parm); - } - csa->r_test = parm->r_test; -#endif - csa->tol_bnd = parm->tol_bnd; - csa->tol_bnd1 = .001 * parm->tol_bnd; - csa->tol_dj = parm->tol_dj; - csa->tol_dj1 = .001 * parm->tol_dj; -#if 0 - csa->tol_dj1 = 1e-9 * parm->tol_dj; -#endif - csa->tol_piv = parm->tol_piv; - switch (P->dir) - { case GLP_MIN: - csa->obj_lim = + parm->obj_ul; - break; - case GLP_MAX: - csa->obj_lim = - parm->obj_ll; - break; - default: - xassert(parm != parm); - } -#if SCALE_Z - if (csa->obj_lim != DBL_MAX) - csa->obj_lim /= csa->fz; -#endif - csa->it_lim = parm->it_lim; - csa->tm_lim = parm->tm_lim; - csa->out_frq = parm->out_frq; - csa->out_dly = parm->out_dly; - /* initialize working parameters */ - csa->tm_beg = xtime(); - csa->it_beg = csa->it_cnt = P->it_cnt; - csa->it_dpy = -1; -#if 1 /* 15/VII-2017 */ - csa->tm_dpy = 0.0; -#endif - csa->inv_cnt = 0; -#if 1 /* 11/VII-2017 */ - csa->degen = 0; -#endif -#if 1 /* 23/III-2016 */ - csa->ns_cnt = csa->ls_cnt = 0; -#endif - /* try to solve working LP */ - ret = dual_simplex(csa); - /* return basis factorization back to problem object */ - P->valid = csa->lp->valid; - P->bfd = csa->lp->bfd; - /* set solution status */ - P->pbs_stat = csa->p_stat; - P->dbs_stat = csa->d_stat; - /* if the solver failed, do not store basis header and basic - * solution components to problem object */ - if (ret == GLP_EFAIL) - goto skip; - /* convert working LP basis to original LP basis and store it to - * problem object */ - daeh = talloc(1+csa->lp->n, int); - spx_store_basis(csa->lp, P, map, daeh); - /* compute simplex multipliers for final basic solution found by - * the solver */ - spx_eval_pi(csa->lp, csa->work); - /* convert working LP solution to original LP solution and store - * it to problem object */ -#if SCALE_Z - for (i = 1; i <= csa->lp->m; i++) - csa->work[i] *= csa->fz; - for (j = 1; j <= csa->lp->n-csa->lp->m; j++) - csa->d[j] *= csa->fz; -#endif - spx_store_sol(csa->lp, P, parm->shift, map, daeh, csa->beta, - csa->work, csa->d); - tfree(daeh); - /* save simplex iteration count */ - P->it_cnt = csa->it_cnt; - /* report auxiliary/structural variable causing unboundedness */ - P->some = 0; - if (csa->p_stat == GLP_NOFEAS && csa->d_stat == GLP_FEAS) - { int k, kk; - /* xB[p] = x[k] causes dual unboundedness */ - xassert(1 <= csa->p && csa->p <= csa->lp->m); - k = csa->lp->head[csa->p]; - xassert(1 <= k && k <= csa->lp->n); - /* convert to number of original variable */ - for (kk = 1; kk <= P->m + P->n; kk++) - { if (abs(map[kk]) == k) - { P->some = kk; - break; - } - } - xassert(P->some != 0); - } -skip: /* deallocate working objects and arrays */ - spx_free_lp(csa->lp); - tfree(map); - tfree(csa->orig_b); - tfree(csa->orig_c); - tfree(csa->orig_l); - tfree(csa->orig_u); - if (csa->at != NULL) - spx_free_at(csa->lp, csa->at); - if (csa->nt != NULL) - spx_free_nt(csa->lp, csa->nt); - tfree(csa->beta); - tfree(csa->d); - if (csa->se != NULL) - spy_free_se(csa->lp, csa->se); -#if 0 /* 30/III-2016 */ - tfree(csa->list); - tfree(csa->trow); -#else - fvs_free_vec(&csa->r); - fvs_free_vec(&csa->trow); -#endif -#if 1 /* 16/III-2016 */ - if (csa->bp != NULL) - tfree(csa->bp); -#endif -#if 0 /* 29/III-2016 */ - tfree(csa->tcol); -#else - fvs_free_vec(&csa->tcol); -#endif - tfree(csa->work); - tfree(csa->work1); -#if 0 /* 11/VI-2017 */ -#if 1 /* 31/III-2016 */ - fvs_free_vec(&csa->wrow); - fvs_free_vec(&csa->wcol); -#endif -#endif - /* return to calling program */ - return ret >= 0 ? ret : GLP_EFAIL; -} - -/* eof */ diff --git a/code/3rd_glpk/zlib/README b/code/3rd_glpk/zlib/README deleted file mode 100644 index 2796312f..00000000 --- a/code/3rd_glpk/zlib/README +++ /dev/null @@ -1,45 +0,0 @@ -NOTE: Files in this subdirectory are NOT part of the GLPK package, but - are used with GLPK. - - The original code was modified according to GLPK requirements by - Andrew Makhorin . - - The following files were rewritten: - gzguts.h, zconf.h, zutil.h. - - The following files were added: - zio.h, zio.c. - - Other files were not changed. -************************************************************************ -zlib general purpose compression library -version 1.2.5, April 19th, 2010 - -Copyright (C) 1995-2010 Jean-loup Gailly and Mark Adler - -This software is provided 'as-is', without any express or implied -warranty. In no event will the authors be held liable for any damages -arising from the use of this software. - -Permission is granted to anyone to use this software for any purpose, -including commercial applications, and to alter it and redistribute it -freely, subject to the following restrictions: - -1. The origin of this software must not be misrepresented; you must not - claim that you wrote the original software. If you use this software - in a product, an acknowledgment in the product documentation would - be appreciated but is not required. - -2. Altered source versions must be plainly marked as such, and must not - be misrepresented as being the original software. - -3. This notice may not be removed or altered from any source - distribution. - -Jean-loup Gailly Mark Adler -jloup@gzip.org madler@alumni.caltech.edu - -The data format used by the zlib library is described by RFCs (Request -for Comments) 1950 to 1952 in the files -http://www.ietf.org/rfc/rfc1950.txt (zlib format), rfc1951.txt (deflate -format) and rfc1952.txt (gzip format). diff --git a/code/3rd_glpk/zlib/adler32.c b/code/3rd_glpk/zlib/adler32.c deleted file mode 100644 index 65ad6a5a..00000000 --- a/code/3rd_glpk/zlib/adler32.c +++ /dev/null @@ -1,169 +0,0 @@ -/* adler32.c -- compute the Adler-32 checksum of a data stream - * Copyright (C) 1995-2007 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* @(#) $Id$ */ - -#include "zutil.h" - -#define local static - -local uLong adler32_combine_(uLong adler1, uLong adler2, z_off64_t len2); - -#define BASE 65521UL /* largest prime smaller than 65536 */ -#define NMAX 5552 -/* NMAX is the largest n such that 255n(n+1)/2 + (n+1)(BASE-1) <= 2^32-1 */ - -#define DO1(buf,i) {adler += (buf)[i]; sum2 += adler;} -#define DO2(buf,i) DO1(buf,i); DO1(buf,i+1); -#define DO4(buf,i) DO2(buf,i); DO2(buf,i+2); -#define DO8(buf,i) DO4(buf,i); DO4(buf,i+4); -#define DO16(buf) DO8(buf,0); DO8(buf,8); - -/* use NO_DIVIDE if your processor does not do division in hardware */ -#ifdef NO_DIVIDE -# define MOD(a) \ - do { \ - if (a >= (BASE << 16)) a -= (BASE << 16); \ - if (a >= (BASE << 15)) a -= (BASE << 15); \ - if (a >= (BASE << 14)) a -= (BASE << 14); \ - if (a >= (BASE << 13)) a -= (BASE << 13); \ - if (a >= (BASE << 12)) a -= (BASE << 12); \ - if (a >= (BASE << 11)) a -= (BASE << 11); \ - if (a >= (BASE << 10)) a -= (BASE << 10); \ - if (a >= (BASE << 9)) a -= (BASE << 9); \ - if (a >= (BASE << 8)) a -= (BASE << 8); \ - if (a >= (BASE << 7)) a -= (BASE << 7); \ - if (a >= (BASE << 6)) a -= (BASE << 6); \ - if (a >= (BASE << 5)) a -= (BASE << 5); \ - if (a >= (BASE << 4)) a -= (BASE << 4); \ - if (a >= (BASE << 3)) a -= (BASE << 3); \ - if (a >= (BASE << 2)) a -= (BASE << 2); \ - if (a >= (BASE << 1)) a -= (BASE << 1); \ - if (a >= BASE) a -= BASE; \ - } while (0) -# define MOD4(a) \ - do { \ - if (a >= (BASE << 4)) a -= (BASE << 4); \ - if (a >= (BASE << 3)) a -= (BASE << 3); \ - if (a >= (BASE << 2)) a -= (BASE << 2); \ - if (a >= (BASE << 1)) a -= (BASE << 1); \ - if (a >= BASE) a -= BASE; \ - } while (0) -#else -# define MOD(a) a %= BASE -# define MOD4(a) a %= BASE -#endif - -/* ========================================================================= */ -uLong ZEXPORT adler32(adler, buf, len) - uLong adler; - const Bytef *buf; - uInt len; -{ - unsigned long sum2; - unsigned n; - - /* split Adler-32 into component sums */ - sum2 = (adler >> 16) & 0xffff; - adler &= 0xffff; - - /* in case user likes doing a byte at a time, keep it fast */ - if (len == 1) { - adler += buf[0]; - if (adler >= BASE) - adler -= BASE; - sum2 += adler; - if (sum2 >= BASE) - sum2 -= BASE; - return adler | (sum2 << 16); - } - - /* initial Adler-32 value (deferred check for len == 1 speed) */ - if (buf == Z_NULL) - return 1L; - - /* in case short lengths are provided, keep it somewhat fast */ - if (len < 16) { - while (len--) { - adler += *buf++; - sum2 += adler; - } - if (adler >= BASE) - adler -= BASE; - MOD4(sum2); /* only added so many BASE's */ - return adler | (sum2 << 16); - } - - /* do length NMAX blocks -- requires just one modulo operation */ - while (len >= NMAX) { - len -= NMAX; - n = NMAX / 16; /* NMAX is divisible by 16 */ - do { - DO16(buf); /* 16 sums unrolled */ - buf += 16; - } while (--n); - MOD(adler); - MOD(sum2); - } - - /* do remaining bytes (less than NMAX, still just one modulo) */ - if (len) { /* avoid modulos if none remaining */ - while (len >= 16) { - len -= 16; - DO16(buf); - buf += 16; - } - while (len--) { - adler += *buf++; - sum2 += adler; - } - MOD(adler); - MOD(sum2); - } - - /* return recombined sums */ - return adler | (sum2 << 16); -} - -/* ========================================================================= */ -local uLong adler32_combine_(adler1, adler2, len2) - uLong adler1; - uLong adler2; - z_off64_t len2; -{ - unsigned long sum1; - unsigned long sum2; - unsigned rem; - - /* the derivation of this formula is left as an exercise for the reader */ - rem = (unsigned)(len2 % BASE); - sum1 = adler1 & 0xffff; - sum2 = rem * sum1; - MOD(sum2); - sum1 += (adler2 & 0xffff) + BASE - 1; - sum2 += ((adler1 >> 16) & 0xffff) + ((adler2 >> 16) & 0xffff) + BASE - rem; - if (sum1 >= BASE) sum1 -= BASE; - if (sum1 >= BASE) sum1 -= BASE; - if (sum2 >= (BASE << 1)) sum2 -= (BASE << 1); - if (sum2 >= BASE) sum2 -= BASE; - return sum1 | (sum2 << 16); -} - -/* ========================================================================= */ -uLong ZEXPORT adler32_combine(adler1, adler2, len2) - uLong adler1; - uLong adler2; - z_off_t len2; -{ - return adler32_combine_(adler1, adler2, len2); -} - -uLong ZEXPORT adler32_combine64(adler1, adler2, len2) - uLong adler1; - uLong adler2; - z_off64_t len2; -{ - return adler32_combine_(adler1, adler2, len2); -} diff --git a/code/3rd_glpk/zlib/compress.c b/code/3rd_glpk/zlib/compress.c deleted file mode 100644 index ea4dfbe9..00000000 --- a/code/3rd_glpk/zlib/compress.c +++ /dev/null @@ -1,80 +0,0 @@ -/* compress.c -- compress a memory buffer - * Copyright (C) 1995-2005 Jean-loup Gailly. - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* @(#) $Id$ */ - -#define ZLIB_INTERNAL -#include "zlib.h" - -/* =========================================================================== - Compresses the source buffer into the destination buffer. The level - parameter has the same meaning as in deflateInit. sourceLen is the byte - length of the source buffer. Upon entry, destLen is the total size of the - destination buffer, which must be at least 0.1% larger than sourceLen plus - 12 bytes. Upon exit, destLen is the actual size of the compressed buffer. - - compress2 returns Z_OK if success, Z_MEM_ERROR if there was not enough - memory, Z_BUF_ERROR if there was not enough room in the output buffer, - Z_STREAM_ERROR if the level parameter is invalid. -*/ -int ZEXPORT compress2 (dest, destLen, source, sourceLen, level) - Bytef *dest; - uLongf *destLen; - const Bytef *source; - uLong sourceLen; - int level; -{ - z_stream stream; - int err; - - stream.next_in = (Bytef*)source; - stream.avail_in = (uInt)sourceLen; -#ifdef MAXSEG_64K - /* Check for source > 64K on 16-bit machine: */ - if ((uLong)stream.avail_in != sourceLen) return Z_BUF_ERROR; -#endif - stream.next_out = dest; - stream.avail_out = (uInt)*destLen; - if ((uLong)stream.avail_out != *destLen) return Z_BUF_ERROR; - - stream.zalloc = (alloc_func)0; - stream.zfree = (free_func)0; - stream.opaque = (voidpf)0; - - err = deflateInit(&stream, level); - if (err != Z_OK) return err; - - err = deflate(&stream, Z_FINISH); - if (err != Z_STREAM_END) { - deflateEnd(&stream); - return err == Z_OK ? Z_BUF_ERROR : err; - } - *destLen = stream.total_out; - - err = deflateEnd(&stream); - return err; -} - -/* =========================================================================== - */ -int ZEXPORT compress (dest, destLen, source, sourceLen) - Bytef *dest; - uLongf *destLen; - const Bytef *source; - uLong sourceLen; -{ - return compress2(dest, destLen, source, sourceLen, Z_DEFAULT_COMPRESSION); -} - -/* =========================================================================== - If the default memLevel or windowBits for deflateInit() is changed, then - this function needs to be updated. - */ -uLong ZEXPORT compressBound (sourceLen) - uLong sourceLen; -{ - return sourceLen + (sourceLen >> 12) + (sourceLen >> 14) + - (sourceLen >> 25) + 13; -} diff --git a/code/3rd_glpk/zlib/crc32.c b/code/3rd_glpk/zlib/crc32.c deleted file mode 100644 index 91be372d..00000000 --- a/code/3rd_glpk/zlib/crc32.c +++ /dev/null @@ -1,442 +0,0 @@ -/* crc32.c -- compute the CRC-32 of a data stream - * Copyright (C) 1995-2006, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - * - * Thanks to Rodney Brown for his contribution of faster - * CRC methods: exclusive-oring 32 bits of data at a time, and pre-computing - * tables for updating the shift register in one step with three exclusive-ors - * instead of four steps with four exclusive-ors. This results in about a - * factor of two increase in speed on a Power PC G4 (PPC7455) using gcc -O3. - */ - -/* @(#) $Id$ */ - -/* - Note on the use of DYNAMIC_CRC_TABLE: there is no mutex or semaphore - protection on the static variables used to control the first-use generation - of the crc tables. Therefore, if you #define DYNAMIC_CRC_TABLE, you should - first call get_crc_table() to initialize the tables before allowing more than - one thread to use crc32(). - */ - -#ifdef MAKECRCH -# include -# ifndef DYNAMIC_CRC_TABLE -# define DYNAMIC_CRC_TABLE -# endif /* !DYNAMIC_CRC_TABLE */ -#endif /* MAKECRCH */ - -#include "zutil.h" /* for STDC and FAR definitions */ - -#define local static - -/* Find a four-byte integer type for crc32_little() and crc32_big(). */ -#ifndef NOBYFOUR -# ifdef STDC /* need ANSI C limits.h to determine sizes */ -# include -# define BYFOUR -# if (UINT_MAX == 0xffffffffUL) - typedef unsigned int u4; -# else -# if (ULONG_MAX == 0xffffffffUL) - typedef unsigned long u4; -# else -# if (USHRT_MAX == 0xffffffffUL) - typedef unsigned short u4; -# else -# undef BYFOUR /* can't find a four-byte integer type! */ -# endif -# endif -# endif -# endif /* STDC */ -#endif /* !NOBYFOUR */ - -/* Definitions for doing the crc four data bytes at a time. */ -#ifdef BYFOUR -# define REV(w) ((((w)>>24)&0xff)+(((w)>>8)&0xff00)+ \ - (((w)&0xff00)<<8)+(((w)&0xff)<<24)) - local unsigned long crc32_little OF((unsigned long, - const unsigned char FAR *, unsigned)); - local unsigned long crc32_big OF((unsigned long, - const unsigned char FAR *, unsigned)); -# define TBLS 8 -#else -# define TBLS 1 -#endif /* BYFOUR */ - -/* Local functions for crc concatenation */ -local unsigned long gf2_matrix_times OF((unsigned long *mat, - unsigned long vec)); -local void gf2_matrix_square OF((unsigned long *square, unsigned long *mat)); -local uLong crc32_combine_(uLong crc1, uLong crc2, z_off64_t len2); - - -#ifdef DYNAMIC_CRC_TABLE - -local volatile int crc_table_empty = 1; -local unsigned long FAR crc_table[TBLS][256]; -local void make_crc_table OF((void)); -#ifdef MAKECRCH - local void write_table OF((FILE *, const unsigned long FAR *)); -#endif /* MAKECRCH */ -/* - Generate tables for a byte-wise 32-bit CRC calculation on the polynomial: - x^32+x^26+x^23+x^22+x^16+x^12+x^11+x^10+x^8+x^7+x^5+x^4+x^2+x+1. - - Polynomials over GF(2) are represented in binary, one bit per coefficient, - with the lowest powers in the most significant bit. Then adding polynomials - is just exclusive-or, and multiplying a polynomial by x is a right shift by - one. If we call the above polynomial p, and represent a byte as the - polynomial q, also with the lowest power in the most significant bit (so the - byte 0xb1 is the polynomial x^7+x^3+x+1), then the CRC is (q*x^32) mod p, - where a mod b means the remainder after dividing a by b. - - This calculation is done using the shift-register method of multiplying and - taking the remainder. The register is initialized to zero, and for each - incoming bit, x^32 is added mod p to the register if the bit is a one (where - x^32 mod p is p+x^32 = x^26+...+1), and the register is multiplied mod p by - x (which is shifting right by one and adding x^32 mod p if the bit shifted - out is a one). We start with the highest power (least significant bit) of - q and repeat for all eight bits of q. - - The first table is simply the CRC of all possible eight bit values. This is - all the information needed to generate CRCs on data a byte at a time for all - combinations of CRC register values and incoming bytes. The remaining tables - allow for word-at-a-time CRC calculation for both big-endian and little- - endian machines, where a word is four bytes. -*/ -local void make_crc_table() -{ - unsigned long c; - int n, k; - unsigned long poly; /* polynomial exclusive-or pattern */ - /* terms of polynomial defining this crc (except x^32): */ - static volatile int first = 1; /* flag to limit concurrent making */ - static const unsigned char p[] = {0,1,2,4,5,7,8,10,11,12,16,22,23,26}; - - /* See if another task is already doing this (not thread-safe, but better - than nothing -- significantly reduces duration of vulnerability in - case the advice about DYNAMIC_CRC_TABLE is ignored) */ - if (first) { - first = 0; - - /* make exclusive-or pattern from polynomial (0xedb88320UL) */ - poly = 0UL; - for (n = 0; n < sizeof(p)/sizeof(unsigned char); n++) - poly |= 1UL << (31 - p[n]); - - /* generate a crc for every 8-bit value */ - for (n = 0; n < 256; n++) { - c = (unsigned long)n; - for (k = 0; k < 8; k++) - c = c & 1 ? poly ^ (c >> 1) : c >> 1; - crc_table[0][n] = c; - } - -#ifdef BYFOUR - /* generate crc for each value followed by one, two, and three zeros, - and then the byte reversal of those as well as the first table */ - for (n = 0; n < 256; n++) { - c = crc_table[0][n]; - crc_table[4][n] = REV(c); - for (k = 1; k < 4; k++) { - c = crc_table[0][c & 0xff] ^ (c >> 8); - crc_table[k][n] = c; - crc_table[k + 4][n] = REV(c); - } - } -#endif /* BYFOUR */ - - crc_table_empty = 0; - } - else { /* not first */ - /* wait for the other guy to finish (not efficient, but rare) */ - while (crc_table_empty) - ; - } - -#ifdef MAKECRCH - /* write out CRC tables to crc32.h */ - { - FILE *out; - - out = fopen("crc32.h", "w"); - if (out == NULL) return; - fprintf(out, "/* crc32.h -- tables for rapid CRC calculation\n"); - fprintf(out, " * Generated automatically by crc32.c\n */\n\n"); - fprintf(out, "local const unsigned long FAR "); - fprintf(out, "crc_table[TBLS][256] =\n{\n {\n"); - write_table(out, crc_table[0]); -# ifdef BYFOUR - fprintf(out, "#ifdef BYFOUR\n"); - for (k = 1; k < 8; k++) { - fprintf(out, " },\n {\n"); - write_table(out, crc_table[k]); - } - fprintf(out, "#endif\n"); -# endif /* BYFOUR */ - fprintf(out, " }\n};\n"); - fclose(out); - } -#endif /* MAKECRCH */ -} - -#ifdef MAKECRCH -local void write_table(out, table) - FILE *out; - const unsigned long FAR *table; -{ - int n; - - for (n = 0; n < 256; n++) - fprintf(out, "%s0x%08lxUL%s", n % 5 ? "" : " ", table[n], - n == 255 ? "\n" : (n % 5 == 4 ? ",\n" : ", ")); -} -#endif /* MAKECRCH */ - -#else /* !DYNAMIC_CRC_TABLE */ -/* ======================================================================== - * Tables of CRC-32s of all single-byte values, made by make_crc_table(). - */ -#include "crc32.h" -#endif /* DYNAMIC_CRC_TABLE */ - -/* ========================================================================= - * This function can be used by asm versions of crc32() - */ -const unsigned long FAR * ZEXPORT get_crc_table() -{ -#ifdef DYNAMIC_CRC_TABLE - if (crc_table_empty) - make_crc_table(); -#endif /* DYNAMIC_CRC_TABLE */ - return (const unsigned long FAR *)crc_table; -} - -/* ========================================================================= */ -#define DO1 crc = crc_table[0][((int)crc ^ (*buf++)) & 0xff] ^ (crc >> 8) -#define DO8 DO1; DO1; DO1; DO1; DO1; DO1; DO1; DO1 - -/* ========================================================================= */ -unsigned long ZEXPORT crc32(crc, buf, len) - unsigned long crc; - const unsigned char FAR *buf; - uInt len; -{ - if (buf == Z_NULL) return 0UL; - -#ifdef DYNAMIC_CRC_TABLE - if (crc_table_empty) - make_crc_table(); -#endif /* DYNAMIC_CRC_TABLE */ - -#ifdef BYFOUR - if (sizeof(void *) == sizeof(ptrdiff_t)) { - u4 endian; - - endian = 1; - if (*((unsigned char *)(&endian))) - return crc32_little(crc, buf, len); - else - return crc32_big(crc, buf, len); - } -#endif /* BYFOUR */ - crc = crc ^ 0xffffffffUL; - while (len >= 8) { - DO8; - len -= 8; - } - if (len) do { - DO1; - } while (--len); - return crc ^ 0xffffffffUL; -} - -#ifdef BYFOUR - -/* ========================================================================= */ -#define DOLIT4 c ^= *buf4++; \ - c = crc_table[3][c & 0xff] ^ crc_table[2][(c >> 8) & 0xff] ^ \ - crc_table[1][(c >> 16) & 0xff] ^ crc_table[0][c >> 24] -#define DOLIT32 DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4 - -/* ========================================================================= */ -local unsigned long crc32_little(crc, buf, len) - unsigned long crc; - const unsigned char FAR *buf; - unsigned len; -{ - register u4 c; - register const u4 FAR *buf4; - - c = (u4)crc; - c = ~c; - while (len && ((ptrdiff_t)buf & 3)) { - c = crc_table[0][(c ^ *buf++) & 0xff] ^ (c >> 8); - len--; - } - - buf4 = (const u4 FAR *)(const void FAR *)buf; - while (len >= 32) { - DOLIT32; - len -= 32; - } - while (len >= 4) { - DOLIT4; - len -= 4; - } - buf = (const unsigned char FAR *)buf4; - - if (len) do { - c = crc_table[0][(c ^ *buf++) & 0xff] ^ (c >> 8); - } while (--len); - c = ~c; - return (unsigned long)c; -} - -/* ========================================================================= */ -#define DOBIG4 c ^= *++buf4; \ - c = crc_table[4][c & 0xff] ^ crc_table[5][(c >> 8) & 0xff] ^ \ - crc_table[6][(c >> 16) & 0xff] ^ crc_table[7][c >> 24] -#define DOBIG32 DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4 - -/* ========================================================================= */ -local unsigned long crc32_big(crc, buf, len) - unsigned long crc; - const unsigned char FAR *buf; - unsigned len; -{ - register u4 c; - register const u4 FAR *buf4; - - c = REV((u4)crc); - c = ~c; - while (len && ((ptrdiff_t)buf & 3)) { - c = crc_table[4][(c >> 24) ^ *buf++] ^ (c << 8); - len--; - } - - buf4 = (const u4 FAR *)(const void FAR *)buf; - buf4--; - while (len >= 32) { - DOBIG32; - len -= 32; - } - while (len >= 4) { - DOBIG4; - len -= 4; - } - buf4++; - buf = (const unsigned char FAR *)buf4; - - if (len) do { - c = crc_table[4][(c >> 24) ^ *buf++] ^ (c << 8); - } while (--len); - c = ~c; - return (unsigned long)(REV(c)); -} - -#endif /* BYFOUR */ - -#define GF2_DIM 32 /* dimension of GF(2) vectors (length of CRC) */ - -/* ========================================================================= */ -local unsigned long gf2_matrix_times(mat, vec) - unsigned long *mat; - unsigned long vec; -{ - unsigned long sum; - - sum = 0; - while (vec) { - if (vec & 1) - sum ^= *mat; - vec >>= 1; - mat++; - } - return sum; -} - -/* ========================================================================= */ -local void gf2_matrix_square(square, mat) - unsigned long *square; - unsigned long *mat; -{ - int n; - - for (n = 0; n < GF2_DIM; n++) - square[n] = gf2_matrix_times(mat, mat[n]); -} - -/* ========================================================================= */ -local uLong crc32_combine_(crc1, crc2, len2) - uLong crc1; - uLong crc2; - z_off64_t len2; -{ - int n; - unsigned long row; - unsigned long even[GF2_DIM]; /* even-power-of-two zeros operator */ - unsigned long odd[GF2_DIM]; /* odd-power-of-two zeros operator */ - - /* degenerate case (also disallow negative lengths) */ - if (len2 <= 0) - return crc1; - - /* put operator for one zero bit in odd */ - odd[0] = 0xedb88320UL; /* CRC-32 polynomial */ - row = 1; - for (n = 1; n < GF2_DIM; n++) { - odd[n] = row; - row <<= 1; - } - - /* put operator for two zero bits in even */ - gf2_matrix_square(even, odd); - - /* put operator for four zero bits in odd */ - gf2_matrix_square(odd, even); - - /* apply len2 zeros to crc1 (first square will put the operator for one - zero byte, eight zero bits, in even) */ - do { - /* apply zeros operator for this bit of len2 */ - gf2_matrix_square(even, odd); - if (len2 & 1) - crc1 = gf2_matrix_times(even, crc1); - len2 >>= 1; - - /* if no more bits set, then done */ - if (len2 == 0) - break; - - /* another iteration of the loop with odd and even swapped */ - gf2_matrix_square(odd, even); - if (len2 & 1) - crc1 = gf2_matrix_times(odd, crc1); - len2 >>= 1; - - /* if no more bits set, then done */ - } while (len2 != 0); - - /* return combined crc */ - crc1 ^= crc2; - return crc1; -} - -/* ========================================================================= */ -uLong ZEXPORT crc32_combine(crc1, crc2, len2) - uLong crc1; - uLong crc2; - z_off_t len2; -{ - return crc32_combine_(crc1, crc2, len2); -} - -uLong ZEXPORT crc32_combine64(crc1, crc2, len2) - uLong crc1; - uLong crc2; - z_off64_t len2; -{ - return crc32_combine_(crc1, crc2, len2); -} diff --git a/code/3rd_glpk/zlib/crc32.h b/code/3rd_glpk/zlib/crc32.h deleted file mode 100644 index 8053b611..00000000 --- a/code/3rd_glpk/zlib/crc32.h +++ /dev/null @@ -1,441 +0,0 @@ -/* crc32.h -- tables for rapid CRC calculation - * Generated automatically by crc32.c - */ - -local const unsigned long FAR crc_table[TBLS][256] = -{ - { - 0x00000000UL, 0x77073096UL, 0xee0e612cUL, 0x990951baUL, 0x076dc419UL, - 0x706af48fUL, 0xe963a535UL, 0x9e6495a3UL, 0x0edb8832UL, 0x79dcb8a4UL, - 0xe0d5e91eUL, 0x97d2d988UL, 0x09b64c2bUL, 0x7eb17cbdUL, 0xe7b82d07UL, - 0x90bf1d91UL, 0x1db71064UL, 0x6ab020f2UL, 0xf3b97148UL, 0x84be41deUL, - 0x1adad47dUL, 0x6ddde4ebUL, 0xf4d4b551UL, 0x83d385c7UL, 0x136c9856UL, - 0x646ba8c0UL, 0xfd62f97aUL, 0x8a65c9ecUL, 0x14015c4fUL, 0x63066cd9UL, - 0xfa0f3d63UL, 0x8d080df5UL, 0x3b6e20c8UL, 0x4c69105eUL, 0xd56041e4UL, - 0xa2677172UL, 0x3c03e4d1UL, 0x4b04d447UL, 0xd20d85fdUL, 0xa50ab56bUL, - 0x35b5a8faUL, 0x42b2986cUL, 0xdbbbc9d6UL, 0xacbcf940UL, 0x32d86ce3UL, - 0x45df5c75UL, 0xdcd60dcfUL, 0xabd13d59UL, 0x26d930acUL, 0x51de003aUL, - 0xc8d75180UL, 0xbfd06116UL, 0x21b4f4b5UL, 0x56b3c423UL, 0xcfba9599UL, - 0xb8bda50fUL, 0x2802b89eUL, 0x5f058808UL, 0xc60cd9b2UL, 0xb10be924UL, - 0x2f6f7c87UL, 0x58684c11UL, 0xc1611dabUL, 0xb6662d3dUL, 0x76dc4190UL, - 0x01db7106UL, 0x98d220bcUL, 0xefd5102aUL, 0x71b18589UL, 0x06b6b51fUL, - 0x9fbfe4a5UL, 0xe8b8d433UL, 0x7807c9a2UL, 0x0f00f934UL, 0x9609a88eUL, - 0xe10e9818UL, 0x7f6a0dbbUL, 0x086d3d2dUL, 0x91646c97UL, 0xe6635c01UL, - 0x6b6b51f4UL, 0x1c6c6162UL, 0x856530d8UL, 0xf262004eUL, 0x6c0695edUL, - 0x1b01a57bUL, 0x8208f4c1UL, 0xf50fc457UL, 0x65b0d9c6UL, 0x12b7e950UL, - 0x8bbeb8eaUL, 0xfcb9887cUL, 0x62dd1ddfUL, 0x15da2d49UL, 0x8cd37cf3UL, - 0xfbd44c65UL, 0x4db26158UL, 0x3ab551ceUL, 0xa3bc0074UL, 0xd4bb30e2UL, - 0x4adfa541UL, 0x3dd895d7UL, 0xa4d1c46dUL, 0xd3d6f4fbUL, 0x4369e96aUL, - 0x346ed9fcUL, 0xad678846UL, 0xda60b8d0UL, 0x44042d73UL, 0x33031de5UL, - 0xaa0a4c5fUL, 0xdd0d7cc9UL, 0x5005713cUL, 0x270241aaUL, 0xbe0b1010UL, - 0xc90c2086UL, 0x5768b525UL, 0x206f85b3UL, 0xb966d409UL, 0xce61e49fUL, - 0x5edef90eUL, 0x29d9c998UL, 0xb0d09822UL, 0xc7d7a8b4UL, 0x59b33d17UL, - 0x2eb40d81UL, 0xb7bd5c3bUL, 0xc0ba6cadUL, 0xedb88320UL, 0x9abfb3b6UL, - 0x03b6e20cUL, 0x74b1d29aUL, 0xead54739UL, 0x9dd277afUL, 0x04db2615UL, - 0x73dc1683UL, 0xe3630b12UL, 0x94643b84UL, 0x0d6d6a3eUL, 0x7a6a5aa8UL, - 0xe40ecf0bUL, 0x9309ff9dUL, 0x0a00ae27UL, 0x7d079eb1UL, 0xf00f9344UL, - 0x8708a3d2UL, 0x1e01f268UL, 0x6906c2feUL, 0xf762575dUL, 0x806567cbUL, - 0x196c3671UL, 0x6e6b06e7UL, 0xfed41b76UL, 0x89d32be0UL, 0x10da7a5aUL, - 0x67dd4accUL, 0xf9b9df6fUL, 0x8ebeeff9UL, 0x17b7be43UL, 0x60b08ed5UL, - 0xd6d6a3e8UL, 0xa1d1937eUL, 0x38d8c2c4UL, 0x4fdff252UL, 0xd1bb67f1UL, - 0xa6bc5767UL, 0x3fb506ddUL, 0x48b2364bUL, 0xd80d2bdaUL, 0xaf0a1b4cUL, - 0x36034af6UL, 0x41047a60UL, 0xdf60efc3UL, 0xa867df55UL, 0x316e8eefUL, - 0x4669be79UL, 0xcb61b38cUL, 0xbc66831aUL, 0x256fd2a0UL, 0x5268e236UL, - 0xcc0c7795UL, 0xbb0b4703UL, 0x220216b9UL, 0x5505262fUL, 0xc5ba3bbeUL, - 0xb2bd0b28UL, 0x2bb45a92UL, 0x5cb36a04UL, 0xc2d7ffa7UL, 0xb5d0cf31UL, - 0x2cd99e8bUL, 0x5bdeae1dUL, 0x9b64c2b0UL, 0xec63f226UL, 0x756aa39cUL, - 0x026d930aUL, 0x9c0906a9UL, 0xeb0e363fUL, 0x72076785UL, 0x05005713UL, - 0x95bf4a82UL, 0xe2b87a14UL, 0x7bb12baeUL, 0x0cb61b38UL, 0x92d28e9bUL, - 0xe5d5be0dUL, 0x7cdcefb7UL, 0x0bdbdf21UL, 0x86d3d2d4UL, 0xf1d4e242UL, - 0x68ddb3f8UL, 0x1fda836eUL, 0x81be16cdUL, 0xf6b9265bUL, 0x6fb077e1UL, - 0x18b74777UL, 0x88085ae6UL, 0xff0f6a70UL, 0x66063bcaUL, 0x11010b5cUL, - 0x8f659effUL, 0xf862ae69UL, 0x616bffd3UL, 0x166ccf45UL, 0xa00ae278UL, - 0xd70dd2eeUL, 0x4e048354UL, 0x3903b3c2UL, 0xa7672661UL, 0xd06016f7UL, - 0x4969474dUL, 0x3e6e77dbUL, 0xaed16a4aUL, 0xd9d65adcUL, 0x40df0b66UL, - 0x37d83bf0UL, 0xa9bcae53UL, 0xdebb9ec5UL, 0x47b2cf7fUL, 0x30b5ffe9UL, - 0xbdbdf21cUL, 0xcabac28aUL, 0x53b39330UL, 0x24b4a3a6UL, 0xbad03605UL, - 0xcdd70693UL, 0x54de5729UL, 0x23d967bfUL, 0xb3667a2eUL, 0xc4614ab8UL, - 0x5d681b02UL, 0x2a6f2b94UL, 0xb40bbe37UL, 0xc30c8ea1UL, 0x5a05df1bUL, - 0x2d02ef8dUL -#ifdef BYFOUR - }, - { - 0x00000000UL, 0x191b3141UL, 0x32366282UL, 0x2b2d53c3UL, 0x646cc504UL, - 0x7d77f445UL, 0x565aa786UL, 0x4f4196c7UL, 0xc8d98a08UL, 0xd1c2bb49UL, - 0xfaefe88aUL, 0xe3f4d9cbUL, 0xacb54f0cUL, 0xb5ae7e4dUL, 0x9e832d8eUL, - 0x87981ccfUL, 0x4ac21251UL, 0x53d92310UL, 0x78f470d3UL, 0x61ef4192UL, - 0x2eaed755UL, 0x37b5e614UL, 0x1c98b5d7UL, 0x05838496UL, 0x821b9859UL, - 0x9b00a918UL, 0xb02dfadbUL, 0xa936cb9aUL, 0xe6775d5dUL, 0xff6c6c1cUL, - 0xd4413fdfUL, 0xcd5a0e9eUL, 0x958424a2UL, 0x8c9f15e3UL, 0xa7b24620UL, - 0xbea97761UL, 0xf1e8e1a6UL, 0xe8f3d0e7UL, 0xc3de8324UL, 0xdac5b265UL, - 0x5d5daeaaUL, 0x44469febUL, 0x6f6bcc28UL, 0x7670fd69UL, 0x39316baeUL, - 0x202a5aefUL, 0x0b07092cUL, 0x121c386dUL, 0xdf4636f3UL, 0xc65d07b2UL, - 0xed705471UL, 0xf46b6530UL, 0xbb2af3f7UL, 0xa231c2b6UL, 0x891c9175UL, - 0x9007a034UL, 0x179fbcfbUL, 0x0e848dbaUL, 0x25a9de79UL, 0x3cb2ef38UL, - 0x73f379ffUL, 0x6ae848beUL, 0x41c51b7dUL, 0x58de2a3cUL, 0xf0794f05UL, - 0xe9627e44UL, 0xc24f2d87UL, 0xdb541cc6UL, 0x94158a01UL, 0x8d0ebb40UL, - 0xa623e883UL, 0xbf38d9c2UL, 0x38a0c50dUL, 0x21bbf44cUL, 0x0a96a78fUL, - 0x138d96ceUL, 0x5ccc0009UL, 0x45d73148UL, 0x6efa628bUL, 0x77e153caUL, - 0xbabb5d54UL, 0xa3a06c15UL, 0x888d3fd6UL, 0x91960e97UL, 0xded79850UL, - 0xc7cca911UL, 0xece1fad2UL, 0xf5facb93UL, 0x7262d75cUL, 0x6b79e61dUL, - 0x4054b5deUL, 0x594f849fUL, 0x160e1258UL, 0x0f152319UL, 0x243870daUL, - 0x3d23419bUL, 0x65fd6ba7UL, 0x7ce65ae6UL, 0x57cb0925UL, 0x4ed03864UL, - 0x0191aea3UL, 0x188a9fe2UL, 0x33a7cc21UL, 0x2abcfd60UL, 0xad24e1afUL, - 0xb43fd0eeUL, 0x9f12832dUL, 0x8609b26cUL, 0xc94824abUL, 0xd05315eaUL, - 0xfb7e4629UL, 0xe2657768UL, 0x2f3f79f6UL, 0x362448b7UL, 0x1d091b74UL, - 0x04122a35UL, 0x4b53bcf2UL, 0x52488db3UL, 0x7965de70UL, 0x607eef31UL, - 0xe7e6f3feUL, 0xfefdc2bfUL, 0xd5d0917cUL, 0xcccba03dUL, 0x838a36faUL, - 0x9a9107bbUL, 0xb1bc5478UL, 0xa8a76539UL, 0x3b83984bUL, 0x2298a90aUL, - 0x09b5fac9UL, 0x10aecb88UL, 0x5fef5d4fUL, 0x46f46c0eUL, 0x6dd93fcdUL, - 0x74c20e8cUL, 0xf35a1243UL, 0xea412302UL, 0xc16c70c1UL, 0xd8774180UL, - 0x9736d747UL, 0x8e2de606UL, 0xa500b5c5UL, 0xbc1b8484UL, 0x71418a1aUL, - 0x685abb5bUL, 0x4377e898UL, 0x5a6cd9d9UL, 0x152d4f1eUL, 0x0c367e5fUL, - 0x271b2d9cUL, 0x3e001cddUL, 0xb9980012UL, 0xa0833153UL, 0x8bae6290UL, - 0x92b553d1UL, 0xddf4c516UL, 0xc4eff457UL, 0xefc2a794UL, 0xf6d996d5UL, - 0xae07bce9UL, 0xb71c8da8UL, 0x9c31de6bUL, 0x852aef2aUL, 0xca6b79edUL, - 0xd37048acUL, 0xf85d1b6fUL, 0xe1462a2eUL, 0x66de36e1UL, 0x7fc507a0UL, - 0x54e85463UL, 0x4df36522UL, 0x02b2f3e5UL, 0x1ba9c2a4UL, 0x30849167UL, - 0x299fa026UL, 0xe4c5aeb8UL, 0xfdde9ff9UL, 0xd6f3cc3aUL, 0xcfe8fd7bUL, - 0x80a96bbcUL, 0x99b25afdUL, 0xb29f093eUL, 0xab84387fUL, 0x2c1c24b0UL, - 0x350715f1UL, 0x1e2a4632UL, 0x07317773UL, 0x4870e1b4UL, 0x516bd0f5UL, - 0x7a468336UL, 0x635db277UL, 0xcbfad74eUL, 0xd2e1e60fUL, 0xf9ccb5ccUL, - 0xe0d7848dUL, 0xaf96124aUL, 0xb68d230bUL, 0x9da070c8UL, 0x84bb4189UL, - 0x03235d46UL, 0x1a386c07UL, 0x31153fc4UL, 0x280e0e85UL, 0x674f9842UL, - 0x7e54a903UL, 0x5579fac0UL, 0x4c62cb81UL, 0x8138c51fUL, 0x9823f45eUL, - 0xb30ea79dUL, 0xaa1596dcUL, 0xe554001bUL, 0xfc4f315aUL, 0xd7626299UL, - 0xce7953d8UL, 0x49e14f17UL, 0x50fa7e56UL, 0x7bd72d95UL, 0x62cc1cd4UL, - 0x2d8d8a13UL, 0x3496bb52UL, 0x1fbbe891UL, 0x06a0d9d0UL, 0x5e7ef3ecUL, - 0x4765c2adUL, 0x6c48916eUL, 0x7553a02fUL, 0x3a1236e8UL, 0x230907a9UL, - 0x0824546aUL, 0x113f652bUL, 0x96a779e4UL, 0x8fbc48a5UL, 0xa4911b66UL, - 0xbd8a2a27UL, 0xf2cbbce0UL, 0xebd08da1UL, 0xc0fdde62UL, 0xd9e6ef23UL, - 0x14bce1bdUL, 0x0da7d0fcUL, 0x268a833fUL, 0x3f91b27eUL, 0x70d024b9UL, - 0x69cb15f8UL, 0x42e6463bUL, 0x5bfd777aUL, 0xdc656bb5UL, 0xc57e5af4UL, - 0xee530937UL, 0xf7483876UL, 0xb809aeb1UL, 0xa1129ff0UL, 0x8a3fcc33UL, - 0x9324fd72UL - }, - { - 0x00000000UL, 0x01c26a37UL, 0x0384d46eUL, 0x0246be59UL, 0x0709a8dcUL, - 0x06cbc2ebUL, 0x048d7cb2UL, 0x054f1685UL, 0x0e1351b8UL, 0x0fd13b8fUL, - 0x0d9785d6UL, 0x0c55efe1UL, 0x091af964UL, 0x08d89353UL, 0x0a9e2d0aUL, - 0x0b5c473dUL, 0x1c26a370UL, 0x1de4c947UL, 0x1fa2771eUL, 0x1e601d29UL, - 0x1b2f0bacUL, 0x1aed619bUL, 0x18abdfc2UL, 0x1969b5f5UL, 0x1235f2c8UL, - 0x13f798ffUL, 0x11b126a6UL, 0x10734c91UL, 0x153c5a14UL, 0x14fe3023UL, - 0x16b88e7aUL, 0x177ae44dUL, 0x384d46e0UL, 0x398f2cd7UL, 0x3bc9928eUL, - 0x3a0bf8b9UL, 0x3f44ee3cUL, 0x3e86840bUL, 0x3cc03a52UL, 0x3d025065UL, - 0x365e1758UL, 0x379c7d6fUL, 0x35dac336UL, 0x3418a901UL, 0x3157bf84UL, - 0x3095d5b3UL, 0x32d36beaUL, 0x331101ddUL, 0x246be590UL, 0x25a98fa7UL, - 0x27ef31feUL, 0x262d5bc9UL, 0x23624d4cUL, 0x22a0277bUL, 0x20e69922UL, - 0x2124f315UL, 0x2a78b428UL, 0x2bbade1fUL, 0x29fc6046UL, 0x283e0a71UL, - 0x2d711cf4UL, 0x2cb376c3UL, 0x2ef5c89aUL, 0x2f37a2adUL, 0x709a8dc0UL, - 0x7158e7f7UL, 0x731e59aeUL, 0x72dc3399UL, 0x7793251cUL, 0x76514f2bUL, - 0x7417f172UL, 0x75d59b45UL, 0x7e89dc78UL, 0x7f4bb64fUL, 0x7d0d0816UL, - 0x7ccf6221UL, 0x798074a4UL, 0x78421e93UL, 0x7a04a0caUL, 0x7bc6cafdUL, - 0x6cbc2eb0UL, 0x6d7e4487UL, 0x6f38fadeUL, 0x6efa90e9UL, 0x6bb5866cUL, - 0x6a77ec5bUL, 0x68315202UL, 0x69f33835UL, 0x62af7f08UL, 0x636d153fUL, - 0x612bab66UL, 0x60e9c151UL, 0x65a6d7d4UL, 0x6464bde3UL, 0x662203baUL, - 0x67e0698dUL, 0x48d7cb20UL, 0x4915a117UL, 0x4b531f4eUL, 0x4a917579UL, - 0x4fde63fcUL, 0x4e1c09cbUL, 0x4c5ab792UL, 0x4d98dda5UL, 0x46c49a98UL, - 0x4706f0afUL, 0x45404ef6UL, 0x448224c1UL, 0x41cd3244UL, 0x400f5873UL, - 0x4249e62aUL, 0x438b8c1dUL, 0x54f16850UL, 0x55330267UL, 0x5775bc3eUL, - 0x56b7d609UL, 0x53f8c08cUL, 0x523aaabbUL, 0x507c14e2UL, 0x51be7ed5UL, - 0x5ae239e8UL, 0x5b2053dfUL, 0x5966ed86UL, 0x58a487b1UL, 0x5deb9134UL, - 0x5c29fb03UL, 0x5e6f455aUL, 0x5fad2f6dUL, 0xe1351b80UL, 0xe0f771b7UL, - 0xe2b1cfeeUL, 0xe373a5d9UL, 0xe63cb35cUL, 0xe7fed96bUL, 0xe5b86732UL, - 0xe47a0d05UL, 0xef264a38UL, 0xeee4200fUL, 0xeca29e56UL, 0xed60f461UL, - 0xe82fe2e4UL, 0xe9ed88d3UL, 0xebab368aUL, 0xea695cbdUL, 0xfd13b8f0UL, - 0xfcd1d2c7UL, 0xfe976c9eUL, 0xff5506a9UL, 0xfa1a102cUL, 0xfbd87a1bUL, - 0xf99ec442UL, 0xf85cae75UL, 0xf300e948UL, 0xf2c2837fUL, 0xf0843d26UL, - 0xf1465711UL, 0xf4094194UL, 0xf5cb2ba3UL, 0xf78d95faUL, 0xf64fffcdUL, - 0xd9785d60UL, 0xd8ba3757UL, 0xdafc890eUL, 0xdb3ee339UL, 0xde71f5bcUL, - 0xdfb39f8bUL, 0xddf521d2UL, 0xdc374be5UL, 0xd76b0cd8UL, 0xd6a966efUL, - 0xd4efd8b6UL, 0xd52db281UL, 0xd062a404UL, 0xd1a0ce33UL, 0xd3e6706aUL, - 0xd2241a5dUL, 0xc55efe10UL, 0xc49c9427UL, 0xc6da2a7eUL, 0xc7184049UL, - 0xc25756ccUL, 0xc3953cfbUL, 0xc1d382a2UL, 0xc011e895UL, 0xcb4dafa8UL, - 0xca8fc59fUL, 0xc8c97bc6UL, 0xc90b11f1UL, 0xcc440774UL, 0xcd866d43UL, - 0xcfc0d31aUL, 0xce02b92dUL, 0x91af9640UL, 0x906dfc77UL, 0x922b422eUL, - 0x93e92819UL, 0x96a63e9cUL, 0x976454abUL, 0x9522eaf2UL, 0x94e080c5UL, - 0x9fbcc7f8UL, 0x9e7eadcfUL, 0x9c381396UL, 0x9dfa79a1UL, 0x98b56f24UL, - 0x99770513UL, 0x9b31bb4aUL, 0x9af3d17dUL, 0x8d893530UL, 0x8c4b5f07UL, - 0x8e0de15eUL, 0x8fcf8b69UL, 0x8a809decUL, 0x8b42f7dbUL, 0x89044982UL, - 0x88c623b5UL, 0x839a6488UL, 0x82580ebfUL, 0x801eb0e6UL, 0x81dcdad1UL, - 0x8493cc54UL, 0x8551a663UL, 0x8717183aUL, 0x86d5720dUL, 0xa9e2d0a0UL, - 0xa820ba97UL, 0xaa6604ceUL, 0xaba46ef9UL, 0xaeeb787cUL, 0xaf29124bUL, - 0xad6fac12UL, 0xacadc625UL, 0xa7f18118UL, 0xa633eb2fUL, 0xa4755576UL, - 0xa5b73f41UL, 0xa0f829c4UL, 0xa13a43f3UL, 0xa37cfdaaUL, 0xa2be979dUL, - 0xb5c473d0UL, 0xb40619e7UL, 0xb640a7beUL, 0xb782cd89UL, 0xb2cddb0cUL, - 0xb30fb13bUL, 0xb1490f62UL, 0xb08b6555UL, 0xbbd72268UL, 0xba15485fUL, - 0xb853f606UL, 0xb9919c31UL, 0xbcde8ab4UL, 0xbd1ce083UL, 0xbf5a5edaUL, - 0xbe9834edUL - }, - { - 0x00000000UL, 0xb8bc6765UL, 0xaa09c88bUL, 0x12b5afeeUL, 0x8f629757UL, - 0x37def032UL, 0x256b5fdcUL, 0x9dd738b9UL, 0xc5b428efUL, 0x7d084f8aUL, - 0x6fbde064UL, 0xd7018701UL, 0x4ad6bfb8UL, 0xf26ad8ddUL, 0xe0df7733UL, - 0x58631056UL, 0x5019579fUL, 0xe8a530faUL, 0xfa109f14UL, 0x42acf871UL, - 0xdf7bc0c8UL, 0x67c7a7adUL, 0x75720843UL, 0xcdce6f26UL, 0x95ad7f70UL, - 0x2d111815UL, 0x3fa4b7fbUL, 0x8718d09eUL, 0x1acfe827UL, 0xa2738f42UL, - 0xb0c620acUL, 0x087a47c9UL, 0xa032af3eUL, 0x188ec85bUL, 0x0a3b67b5UL, - 0xb28700d0UL, 0x2f503869UL, 0x97ec5f0cUL, 0x8559f0e2UL, 0x3de59787UL, - 0x658687d1UL, 0xdd3ae0b4UL, 0xcf8f4f5aUL, 0x7733283fUL, 0xeae41086UL, - 0x525877e3UL, 0x40edd80dUL, 0xf851bf68UL, 0xf02bf8a1UL, 0x48979fc4UL, - 0x5a22302aUL, 0xe29e574fUL, 0x7f496ff6UL, 0xc7f50893UL, 0xd540a77dUL, - 0x6dfcc018UL, 0x359fd04eUL, 0x8d23b72bUL, 0x9f9618c5UL, 0x272a7fa0UL, - 0xbafd4719UL, 0x0241207cUL, 0x10f48f92UL, 0xa848e8f7UL, 0x9b14583dUL, - 0x23a83f58UL, 0x311d90b6UL, 0x89a1f7d3UL, 0x1476cf6aUL, 0xaccaa80fUL, - 0xbe7f07e1UL, 0x06c36084UL, 0x5ea070d2UL, 0xe61c17b7UL, 0xf4a9b859UL, - 0x4c15df3cUL, 0xd1c2e785UL, 0x697e80e0UL, 0x7bcb2f0eUL, 0xc377486bUL, - 0xcb0d0fa2UL, 0x73b168c7UL, 0x6104c729UL, 0xd9b8a04cUL, 0x446f98f5UL, - 0xfcd3ff90UL, 0xee66507eUL, 0x56da371bUL, 0x0eb9274dUL, 0xb6054028UL, - 0xa4b0efc6UL, 0x1c0c88a3UL, 0x81dbb01aUL, 0x3967d77fUL, 0x2bd27891UL, - 0x936e1ff4UL, 0x3b26f703UL, 0x839a9066UL, 0x912f3f88UL, 0x299358edUL, - 0xb4446054UL, 0x0cf80731UL, 0x1e4da8dfUL, 0xa6f1cfbaUL, 0xfe92dfecUL, - 0x462eb889UL, 0x549b1767UL, 0xec277002UL, 0x71f048bbUL, 0xc94c2fdeUL, - 0xdbf98030UL, 0x6345e755UL, 0x6b3fa09cUL, 0xd383c7f9UL, 0xc1366817UL, - 0x798a0f72UL, 0xe45d37cbUL, 0x5ce150aeUL, 0x4e54ff40UL, 0xf6e89825UL, - 0xae8b8873UL, 0x1637ef16UL, 0x048240f8UL, 0xbc3e279dUL, 0x21e91f24UL, - 0x99557841UL, 0x8be0d7afUL, 0x335cb0caUL, 0xed59b63bUL, 0x55e5d15eUL, - 0x47507eb0UL, 0xffec19d5UL, 0x623b216cUL, 0xda874609UL, 0xc832e9e7UL, - 0x708e8e82UL, 0x28ed9ed4UL, 0x9051f9b1UL, 0x82e4565fUL, 0x3a58313aUL, - 0xa78f0983UL, 0x1f336ee6UL, 0x0d86c108UL, 0xb53aa66dUL, 0xbd40e1a4UL, - 0x05fc86c1UL, 0x1749292fUL, 0xaff54e4aUL, 0x322276f3UL, 0x8a9e1196UL, - 0x982bbe78UL, 0x2097d91dUL, 0x78f4c94bUL, 0xc048ae2eUL, 0xd2fd01c0UL, - 0x6a4166a5UL, 0xf7965e1cUL, 0x4f2a3979UL, 0x5d9f9697UL, 0xe523f1f2UL, - 0x4d6b1905UL, 0xf5d77e60UL, 0xe762d18eUL, 0x5fdeb6ebUL, 0xc2098e52UL, - 0x7ab5e937UL, 0x680046d9UL, 0xd0bc21bcUL, 0x88df31eaUL, 0x3063568fUL, - 0x22d6f961UL, 0x9a6a9e04UL, 0x07bda6bdUL, 0xbf01c1d8UL, 0xadb46e36UL, - 0x15080953UL, 0x1d724e9aUL, 0xa5ce29ffUL, 0xb77b8611UL, 0x0fc7e174UL, - 0x9210d9cdUL, 0x2aacbea8UL, 0x38191146UL, 0x80a57623UL, 0xd8c66675UL, - 0x607a0110UL, 0x72cfaefeUL, 0xca73c99bUL, 0x57a4f122UL, 0xef189647UL, - 0xfdad39a9UL, 0x45115eccUL, 0x764dee06UL, 0xcef18963UL, 0xdc44268dUL, - 0x64f841e8UL, 0xf92f7951UL, 0x41931e34UL, 0x5326b1daUL, 0xeb9ad6bfUL, - 0xb3f9c6e9UL, 0x0b45a18cUL, 0x19f00e62UL, 0xa14c6907UL, 0x3c9b51beUL, - 0x842736dbUL, 0x96929935UL, 0x2e2efe50UL, 0x2654b999UL, 0x9ee8defcUL, - 0x8c5d7112UL, 0x34e11677UL, 0xa9362eceUL, 0x118a49abUL, 0x033fe645UL, - 0xbb838120UL, 0xe3e09176UL, 0x5b5cf613UL, 0x49e959fdUL, 0xf1553e98UL, - 0x6c820621UL, 0xd43e6144UL, 0xc68bceaaUL, 0x7e37a9cfUL, 0xd67f4138UL, - 0x6ec3265dUL, 0x7c7689b3UL, 0xc4caeed6UL, 0x591dd66fUL, 0xe1a1b10aUL, - 0xf3141ee4UL, 0x4ba87981UL, 0x13cb69d7UL, 0xab770eb2UL, 0xb9c2a15cUL, - 0x017ec639UL, 0x9ca9fe80UL, 0x241599e5UL, 0x36a0360bUL, 0x8e1c516eUL, - 0x866616a7UL, 0x3eda71c2UL, 0x2c6fde2cUL, 0x94d3b949UL, 0x090481f0UL, - 0xb1b8e695UL, 0xa30d497bUL, 0x1bb12e1eUL, 0x43d23e48UL, 0xfb6e592dUL, - 0xe9dbf6c3UL, 0x516791a6UL, 0xccb0a91fUL, 0x740cce7aUL, 0x66b96194UL, - 0xde0506f1UL - }, - { - 0x00000000UL, 0x96300777UL, 0x2c610eeeUL, 0xba510999UL, 0x19c46d07UL, - 0x8ff46a70UL, 0x35a563e9UL, 0xa395649eUL, 0x3288db0eUL, 0xa4b8dc79UL, - 0x1ee9d5e0UL, 0x88d9d297UL, 0x2b4cb609UL, 0xbd7cb17eUL, 0x072db8e7UL, - 0x911dbf90UL, 0x6410b71dUL, 0xf220b06aUL, 0x4871b9f3UL, 0xde41be84UL, - 0x7dd4da1aUL, 0xebe4dd6dUL, 0x51b5d4f4UL, 0xc785d383UL, 0x56986c13UL, - 0xc0a86b64UL, 0x7af962fdUL, 0xecc9658aUL, 0x4f5c0114UL, 0xd96c0663UL, - 0x633d0ffaUL, 0xf50d088dUL, 0xc8206e3bUL, 0x5e10694cUL, 0xe44160d5UL, - 0x727167a2UL, 0xd1e4033cUL, 0x47d4044bUL, 0xfd850dd2UL, 0x6bb50aa5UL, - 0xfaa8b535UL, 0x6c98b242UL, 0xd6c9bbdbUL, 0x40f9bcacUL, 0xe36cd832UL, - 0x755cdf45UL, 0xcf0dd6dcUL, 0x593dd1abUL, 0xac30d926UL, 0x3a00de51UL, - 0x8051d7c8UL, 0x1661d0bfUL, 0xb5f4b421UL, 0x23c4b356UL, 0x9995bacfUL, - 0x0fa5bdb8UL, 0x9eb80228UL, 0x0888055fUL, 0xb2d90cc6UL, 0x24e90bb1UL, - 0x877c6f2fUL, 0x114c6858UL, 0xab1d61c1UL, 0x3d2d66b6UL, 0x9041dc76UL, - 0x0671db01UL, 0xbc20d298UL, 0x2a10d5efUL, 0x8985b171UL, 0x1fb5b606UL, - 0xa5e4bf9fUL, 0x33d4b8e8UL, 0xa2c90778UL, 0x34f9000fUL, 0x8ea80996UL, - 0x18980ee1UL, 0xbb0d6a7fUL, 0x2d3d6d08UL, 0x976c6491UL, 0x015c63e6UL, - 0xf4516b6bUL, 0x62616c1cUL, 0xd8306585UL, 0x4e0062f2UL, 0xed95066cUL, - 0x7ba5011bUL, 0xc1f40882UL, 0x57c40ff5UL, 0xc6d9b065UL, 0x50e9b712UL, - 0xeab8be8bUL, 0x7c88b9fcUL, 0xdf1ddd62UL, 0x492dda15UL, 0xf37cd38cUL, - 0x654cd4fbUL, 0x5861b24dUL, 0xce51b53aUL, 0x7400bca3UL, 0xe230bbd4UL, - 0x41a5df4aUL, 0xd795d83dUL, 0x6dc4d1a4UL, 0xfbf4d6d3UL, 0x6ae96943UL, - 0xfcd96e34UL, 0x468867adUL, 0xd0b860daUL, 0x732d0444UL, 0xe51d0333UL, - 0x5f4c0aaaUL, 0xc97c0dddUL, 0x3c710550UL, 0xaa410227UL, 0x10100bbeUL, - 0x86200cc9UL, 0x25b56857UL, 0xb3856f20UL, 0x09d466b9UL, 0x9fe461ceUL, - 0x0ef9de5eUL, 0x98c9d929UL, 0x2298d0b0UL, 0xb4a8d7c7UL, 0x173db359UL, - 0x810db42eUL, 0x3b5cbdb7UL, 0xad6cbac0UL, 0x2083b8edUL, 0xb6b3bf9aUL, - 0x0ce2b603UL, 0x9ad2b174UL, 0x3947d5eaUL, 0xaf77d29dUL, 0x1526db04UL, - 0x8316dc73UL, 0x120b63e3UL, 0x843b6494UL, 0x3e6a6d0dUL, 0xa85a6a7aUL, - 0x0bcf0ee4UL, 0x9dff0993UL, 0x27ae000aUL, 0xb19e077dUL, 0x44930ff0UL, - 0xd2a30887UL, 0x68f2011eUL, 0xfec20669UL, 0x5d5762f7UL, 0xcb676580UL, - 0x71366c19UL, 0xe7066b6eUL, 0x761bd4feUL, 0xe02bd389UL, 0x5a7ada10UL, - 0xcc4add67UL, 0x6fdfb9f9UL, 0xf9efbe8eUL, 0x43beb717UL, 0xd58eb060UL, - 0xe8a3d6d6UL, 0x7e93d1a1UL, 0xc4c2d838UL, 0x52f2df4fUL, 0xf167bbd1UL, - 0x6757bca6UL, 0xdd06b53fUL, 0x4b36b248UL, 0xda2b0dd8UL, 0x4c1b0aafUL, - 0xf64a0336UL, 0x607a0441UL, 0xc3ef60dfUL, 0x55df67a8UL, 0xef8e6e31UL, - 0x79be6946UL, 0x8cb361cbUL, 0x1a8366bcUL, 0xa0d26f25UL, 0x36e26852UL, - 0x95770cccUL, 0x03470bbbUL, 0xb9160222UL, 0x2f260555UL, 0xbe3bbac5UL, - 0x280bbdb2UL, 0x925ab42bUL, 0x046ab35cUL, 0xa7ffd7c2UL, 0x31cfd0b5UL, - 0x8b9ed92cUL, 0x1daede5bUL, 0xb0c2649bUL, 0x26f263ecUL, 0x9ca36a75UL, - 0x0a936d02UL, 0xa906099cUL, 0x3f360eebUL, 0x85670772UL, 0x13570005UL, - 0x824abf95UL, 0x147ab8e2UL, 0xae2bb17bUL, 0x381bb60cUL, 0x9b8ed292UL, - 0x0dbed5e5UL, 0xb7efdc7cUL, 0x21dfdb0bUL, 0xd4d2d386UL, 0x42e2d4f1UL, - 0xf8b3dd68UL, 0x6e83da1fUL, 0xcd16be81UL, 0x5b26b9f6UL, 0xe177b06fUL, - 0x7747b718UL, 0xe65a0888UL, 0x706a0fffUL, 0xca3b0666UL, 0x5c0b0111UL, - 0xff9e658fUL, 0x69ae62f8UL, 0xd3ff6b61UL, 0x45cf6c16UL, 0x78e20aa0UL, - 0xeed20dd7UL, 0x5483044eUL, 0xc2b30339UL, 0x612667a7UL, 0xf71660d0UL, - 0x4d476949UL, 0xdb776e3eUL, 0x4a6ad1aeUL, 0xdc5ad6d9UL, 0x660bdf40UL, - 0xf03bd837UL, 0x53aebca9UL, 0xc59ebbdeUL, 0x7fcfb247UL, 0xe9ffb530UL, - 0x1cf2bdbdUL, 0x8ac2bacaUL, 0x3093b353UL, 0xa6a3b424UL, 0x0536d0baUL, - 0x9306d7cdUL, 0x2957de54UL, 0xbf67d923UL, 0x2e7a66b3UL, 0xb84a61c4UL, - 0x021b685dUL, 0x942b6f2aUL, 0x37be0bb4UL, 0xa18e0cc3UL, 0x1bdf055aUL, - 0x8def022dUL - }, - { - 0x00000000UL, 0x41311b19UL, 0x82623632UL, 0xc3532d2bUL, 0x04c56c64UL, - 0x45f4777dUL, 0x86a75a56UL, 0xc796414fUL, 0x088ad9c8UL, 0x49bbc2d1UL, - 0x8ae8effaUL, 0xcbd9f4e3UL, 0x0c4fb5acUL, 0x4d7eaeb5UL, 0x8e2d839eUL, - 0xcf1c9887UL, 0x5112c24aUL, 0x1023d953UL, 0xd370f478UL, 0x9241ef61UL, - 0x55d7ae2eUL, 0x14e6b537UL, 0xd7b5981cUL, 0x96848305UL, 0x59981b82UL, - 0x18a9009bUL, 0xdbfa2db0UL, 0x9acb36a9UL, 0x5d5d77e6UL, 0x1c6c6cffUL, - 0xdf3f41d4UL, 0x9e0e5acdUL, 0xa2248495UL, 0xe3159f8cUL, 0x2046b2a7UL, - 0x6177a9beUL, 0xa6e1e8f1UL, 0xe7d0f3e8UL, 0x2483dec3UL, 0x65b2c5daUL, - 0xaaae5d5dUL, 0xeb9f4644UL, 0x28cc6b6fUL, 0x69fd7076UL, 0xae6b3139UL, - 0xef5a2a20UL, 0x2c09070bUL, 0x6d381c12UL, 0xf33646dfUL, 0xb2075dc6UL, - 0x715470edUL, 0x30656bf4UL, 0xf7f32abbUL, 0xb6c231a2UL, 0x75911c89UL, - 0x34a00790UL, 0xfbbc9f17UL, 0xba8d840eUL, 0x79dea925UL, 0x38efb23cUL, - 0xff79f373UL, 0xbe48e86aUL, 0x7d1bc541UL, 0x3c2ade58UL, 0x054f79f0UL, - 0x447e62e9UL, 0x872d4fc2UL, 0xc61c54dbUL, 0x018a1594UL, 0x40bb0e8dUL, - 0x83e823a6UL, 0xc2d938bfUL, 0x0dc5a038UL, 0x4cf4bb21UL, 0x8fa7960aUL, - 0xce968d13UL, 0x0900cc5cUL, 0x4831d745UL, 0x8b62fa6eUL, 0xca53e177UL, - 0x545dbbbaUL, 0x156ca0a3UL, 0xd63f8d88UL, 0x970e9691UL, 0x5098d7deUL, - 0x11a9ccc7UL, 0xd2fae1ecUL, 0x93cbfaf5UL, 0x5cd76272UL, 0x1de6796bUL, - 0xdeb55440UL, 0x9f844f59UL, 0x58120e16UL, 0x1923150fUL, 0xda703824UL, - 0x9b41233dUL, 0xa76bfd65UL, 0xe65ae67cUL, 0x2509cb57UL, 0x6438d04eUL, - 0xa3ae9101UL, 0xe29f8a18UL, 0x21cca733UL, 0x60fdbc2aUL, 0xafe124adUL, - 0xeed03fb4UL, 0x2d83129fUL, 0x6cb20986UL, 0xab2448c9UL, 0xea1553d0UL, - 0x29467efbUL, 0x687765e2UL, 0xf6793f2fUL, 0xb7482436UL, 0x741b091dUL, - 0x352a1204UL, 0xf2bc534bUL, 0xb38d4852UL, 0x70de6579UL, 0x31ef7e60UL, - 0xfef3e6e7UL, 0xbfc2fdfeUL, 0x7c91d0d5UL, 0x3da0cbccUL, 0xfa368a83UL, - 0xbb07919aUL, 0x7854bcb1UL, 0x3965a7a8UL, 0x4b98833bUL, 0x0aa99822UL, - 0xc9fab509UL, 0x88cbae10UL, 0x4f5def5fUL, 0x0e6cf446UL, 0xcd3fd96dUL, - 0x8c0ec274UL, 0x43125af3UL, 0x022341eaUL, 0xc1706cc1UL, 0x804177d8UL, - 0x47d73697UL, 0x06e62d8eUL, 0xc5b500a5UL, 0x84841bbcUL, 0x1a8a4171UL, - 0x5bbb5a68UL, 0x98e87743UL, 0xd9d96c5aUL, 0x1e4f2d15UL, 0x5f7e360cUL, - 0x9c2d1b27UL, 0xdd1c003eUL, 0x120098b9UL, 0x533183a0UL, 0x9062ae8bUL, - 0xd153b592UL, 0x16c5f4ddUL, 0x57f4efc4UL, 0x94a7c2efUL, 0xd596d9f6UL, - 0xe9bc07aeUL, 0xa88d1cb7UL, 0x6bde319cUL, 0x2aef2a85UL, 0xed796bcaUL, - 0xac4870d3UL, 0x6f1b5df8UL, 0x2e2a46e1UL, 0xe136de66UL, 0xa007c57fUL, - 0x6354e854UL, 0x2265f34dUL, 0xe5f3b202UL, 0xa4c2a91bUL, 0x67918430UL, - 0x26a09f29UL, 0xb8aec5e4UL, 0xf99fdefdUL, 0x3accf3d6UL, 0x7bfde8cfUL, - 0xbc6ba980UL, 0xfd5ab299UL, 0x3e099fb2UL, 0x7f3884abUL, 0xb0241c2cUL, - 0xf1150735UL, 0x32462a1eUL, 0x73773107UL, 0xb4e17048UL, 0xf5d06b51UL, - 0x3683467aUL, 0x77b25d63UL, 0x4ed7facbUL, 0x0fe6e1d2UL, 0xccb5ccf9UL, - 0x8d84d7e0UL, 0x4a1296afUL, 0x0b238db6UL, 0xc870a09dUL, 0x8941bb84UL, - 0x465d2303UL, 0x076c381aUL, 0xc43f1531UL, 0x850e0e28UL, 0x42984f67UL, - 0x03a9547eUL, 0xc0fa7955UL, 0x81cb624cUL, 0x1fc53881UL, 0x5ef42398UL, - 0x9da70eb3UL, 0xdc9615aaUL, 0x1b0054e5UL, 0x5a314ffcUL, 0x996262d7UL, - 0xd85379ceUL, 0x174fe149UL, 0x567efa50UL, 0x952dd77bUL, 0xd41ccc62UL, - 0x138a8d2dUL, 0x52bb9634UL, 0x91e8bb1fUL, 0xd0d9a006UL, 0xecf37e5eUL, - 0xadc26547UL, 0x6e91486cUL, 0x2fa05375UL, 0xe836123aUL, 0xa9070923UL, - 0x6a542408UL, 0x2b653f11UL, 0xe479a796UL, 0xa548bc8fUL, 0x661b91a4UL, - 0x272a8abdUL, 0xe0bccbf2UL, 0xa18dd0ebUL, 0x62defdc0UL, 0x23efe6d9UL, - 0xbde1bc14UL, 0xfcd0a70dUL, 0x3f838a26UL, 0x7eb2913fUL, 0xb924d070UL, - 0xf815cb69UL, 0x3b46e642UL, 0x7a77fd5bUL, 0xb56b65dcUL, 0xf45a7ec5UL, - 0x370953eeUL, 0x763848f7UL, 0xb1ae09b8UL, 0xf09f12a1UL, 0x33cc3f8aUL, - 0x72fd2493UL - }, - { - 0x00000000UL, 0x376ac201UL, 0x6ed48403UL, 0x59be4602UL, 0xdca80907UL, - 0xebc2cb06UL, 0xb27c8d04UL, 0x85164f05UL, 0xb851130eUL, 0x8f3bd10fUL, - 0xd685970dUL, 0xe1ef550cUL, 0x64f91a09UL, 0x5393d808UL, 0x0a2d9e0aUL, - 0x3d475c0bUL, 0x70a3261cUL, 0x47c9e41dUL, 0x1e77a21fUL, 0x291d601eUL, - 0xac0b2f1bUL, 0x9b61ed1aUL, 0xc2dfab18UL, 0xf5b56919UL, 0xc8f23512UL, - 0xff98f713UL, 0xa626b111UL, 0x914c7310UL, 0x145a3c15UL, 0x2330fe14UL, - 0x7a8eb816UL, 0x4de47a17UL, 0xe0464d38UL, 0xd72c8f39UL, 0x8e92c93bUL, - 0xb9f80b3aUL, 0x3cee443fUL, 0x0b84863eUL, 0x523ac03cUL, 0x6550023dUL, - 0x58175e36UL, 0x6f7d9c37UL, 0x36c3da35UL, 0x01a91834UL, 0x84bf5731UL, - 0xb3d59530UL, 0xea6bd332UL, 0xdd011133UL, 0x90e56b24UL, 0xa78fa925UL, - 0xfe31ef27UL, 0xc95b2d26UL, 0x4c4d6223UL, 0x7b27a022UL, 0x2299e620UL, - 0x15f32421UL, 0x28b4782aUL, 0x1fdeba2bUL, 0x4660fc29UL, 0x710a3e28UL, - 0xf41c712dUL, 0xc376b32cUL, 0x9ac8f52eUL, 0xada2372fUL, 0xc08d9a70UL, - 0xf7e75871UL, 0xae591e73UL, 0x9933dc72UL, 0x1c259377UL, 0x2b4f5176UL, - 0x72f11774UL, 0x459bd575UL, 0x78dc897eUL, 0x4fb64b7fUL, 0x16080d7dUL, - 0x2162cf7cUL, 0xa4748079UL, 0x931e4278UL, 0xcaa0047aUL, 0xfdcac67bUL, - 0xb02ebc6cUL, 0x87447e6dUL, 0xdefa386fUL, 0xe990fa6eUL, 0x6c86b56bUL, - 0x5bec776aUL, 0x02523168UL, 0x3538f369UL, 0x087faf62UL, 0x3f156d63UL, - 0x66ab2b61UL, 0x51c1e960UL, 0xd4d7a665UL, 0xe3bd6464UL, 0xba032266UL, - 0x8d69e067UL, 0x20cbd748UL, 0x17a11549UL, 0x4e1f534bUL, 0x7975914aUL, - 0xfc63de4fUL, 0xcb091c4eUL, 0x92b75a4cUL, 0xa5dd984dUL, 0x989ac446UL, - 0xaff00647UL, 0xf64e4045UL, 0xc1248244UL, 0x4432cd41UL, 0x73580f40UL, - 0x2ae64942UL, 0x1d8c8b43UL, 0x5068f154UL, 0x67023355UL, 0x3ebc7557UL, - 0x09d6b756UL, 0x8cc0f853UL, 0xbbaa3a52UL, 0xe2147c50UL, 0xd57ebe51UL, - 0xe839e25aUL, 0xdf53205bUL, 0x86ed6659UL, 0xb187a458UL, 0x3491eb5dUL, - 0x03fb295cUL, 0x5a456f5eUL, 0x6d2fad5fUL, 0x801b35e1UL, 0xb771f7e0UL, - 0xeecfb1e2UL, 0xd9a573e3UL, 0x5cb33ce6UL, 0x6bd9fee7UL, 0x3267b8e5UL, - 0x050d7ae4UL, 0x384a26efUL, 0x0f20e4eeUL, 0x569ea2ecUL, 0x61f460edUL, - 0xe4e22fe8UL, 0xd388ede9UL, 0x8a36abebUL, 0xbd5c69eaUL, 0xf0b813fdUL, - 0xc7d2d1fcUL, 0x9e6c97feUL, 0xa90655ffUL, 0x2c101afaUL, 0x1b7ad8fbUL, - 0x42c49ef9UL, 0x75ae5cf8UL, 0x48e900f3UL, 0x7f83c2f2UL, 0x263d84f0UL, - 0x115746f1UL, 0x944109f4UL, 0xa32bcbf5UL, 0xfa958df7UL, 0xcdff4ff6UL, - 0x605d78d9UL, 0x5737bad8UL, 0x0e89fcdaUL, 0x39e33edbUL, 0xbcf571deUL, - 0x8b9fb3dfUL, 0xd221f5ddUL, 0xe54b37dcUL, 0xd80c6bd7UL, 0xef66a9d6UL, - 0xb6d8efd4UL, 0x81b22dd5UL, 0x04a462d0UL, 0x33cea0d1UL, 0x6a70e6d3UL, - 0x5d1a24d2UL, 0x10fe5ec5UL, 0x27949cc4UL, 0x7e2adac6UL, 0x494018c7UL, - 0xcc5657c2UL, 0xfb3c95c3UL, 0xa282d3c1UL, 0x95e811c0UL, 0xa8af4dcbUL, - 0x9fc58fcaUL, 0xc67bc9c8UL, 0xf1110bc9UL, 0x740744ccUL, 0x436d86cdUL, - 0x1ad3c0cfUL, 0x2db902ceUL, 0x4096af91UL, 0x77fc6d90UL, 0x2e422b92UL, - 0x1928e993UL, 0x9c3ea696UL, 0xab546497UL, 0xf2ea2295UL, 0xc580e094UL, - 0xf8c7bc9fUL, 0xcfad7e9eUL, 0x9613389cUL, 0xa179fa9dUL, 0x246fb598UL, - 0x13057799UL, 0x4abb319bUL, 0x7dd1f39aUL, 0x3035898dUL, 0x075f4b8cUL, - 0x5ee10d8eUL, 0x698bcf8fUL, 0xec9d808aUL, 0xdbf7428bUL, 0x82490489UL, - 0xb523c688UL, 0x88649a83UL, 0xbf0e5882UL, 0xe6b01e80UL, 0xd1dadc81UL, - 0x54cc9384UL, 0x63a65185UL, 0x3a181787UL, 0x0d72d586UL, 0xa0d0e2a9UL, - 0x97ba20a8UL, 0xce0466aaUL, 0xf96ea4abUL, 0x7c78ebaeUL, 0x4b1229afUL, - 0x12ac6fadUL, 0x25c6adacUL, 0x1881f1a7UL, 0x2feb33a6UL, 0x765575a4UL, - 0x413fb7a5UL, 0xc429f8a0UL, 0xf3433aa1UL, 0xaafd7ca3UL, 0x9d97bea2UL, - 0xd073c4b5UL, 0xe71906b4UL, 0xbea740b6UL, 0x89cd82b7UL, 0x0cdbcdb2UL, - 0x3bb10fb3UL, 0x620f49b1UL, 0x55658bb0UL, 0x6822d7bbUL, 0x5f4815baUL, - 0x06f653b8UL, 0x319c91b9UL, 0xb48adebcUL, 0x83e01cbdUL, 0xda5e5abfUL, - 0xed3498beUL - }, - { - 0x00000000UL, 0x6567bcb8UL, 0x8bc809aaUL, 0xeeafb512UL, 0x5797628fUL, - 0x32f0de37UL, 0xdc5f6b25UL, 0xb938d79dUL, 0xef28b4c5UL, 0x8a4f087dUL, - 0x64e0bd6fUL, 0x018701d7UL, 0xb8bfd64aUL, 0xddd86af2UL, 0x3377dfe0UL, - 0x56106358UL, 0x9f571950UL, 0xfa30a5e8UL, 0x149f10faUL, 0x71f8ac42UL, - 0xc8c07bdfUL, 0xada7c767UL, 0x43087275UL, 0x266fcecdUL, 0x707fad95UL, - 0x1518112dUL, 0xfbb7a43fUL, 0x9ed01887UL, 0x27e8cf1aUL, 0x428f73a2UL, - 0xac20c6b0UL, 0xc9477a08UL, 0x3eaf32a0UL, 0x5bc88e18UL, 0xb5673b0aUL, - 0xd00087b2UL, 0x6938502fUL, 0x0c5fec97UL, 0xe2f05985UL, 0x8797e53dUL, - 0xd1878665UL, 0xb4e03addUL, 0x5a4f8fcfUL, 0x3f283377UL, 0x8610e4eaUL, - 0xe3775852UL, 0x0dd8ed40UL, 0x68bf51f8UL, 0xa1f82bf0UL, 0xc49f9748UL, - 0x2a30225aUL, 0x4f579ee2UL, 0xf66f497fUL, 0x9308f5c7UL, 0x7da740d5UL, - 0x18c0fc6dUL, 0x4ed09f35UL, 0x2bb7238dUL, 0xc518969fUL, 0xa07f2a27UL, - 0x1947fdbaUL, 0x7c204102UL, 0x928ff410UL, 0xf7e848a8UL, 0x3d58149bUL, - 0x583fa823UL, 0xb6901d31UL, 0xd3f7a189UL, 0x6acf7614UL, 0x0fa8caacUL, - 0xe1077fbeUL, 0x8460c306UL, 0xd270a05eUL, 0xb7171ce6UL, 0x59b8a9f4UL, - 0x3cdf154cUL, 0x85e7c2d1UL, 0xe0807e69UL, 0x0e2fcb7bUL, 0x6b4877c3UL, - 0xa20f0dcbUL, 0xc768b173UL, 0x29c70461UL, 0x4ca0b8d9UL, 0xf5986f44UL, - 0x90ffd3fcUL, 0x7e5066eeUL, 0x1b37da56UL, 0x4d27b90eUL, 0x284005b6UL, - 0xc6efb0a4UL, 0xa3880c1cUL, 0x1ab0db81UL, 0x7fd76739UL, 0x9178d22bUL, - 0xf41f6e93UL, 0x03f7263bUL, 0x66909a83UL, 0x883f2f91UL, 0xed589329UL, - 0x546044b4UL, 0x3107f80cUL, 0xdfa84d1eUL, 0xbacff1a6UL, 0xecdf92feUL, - 0x89b82e46UL, 0x67179b54UL, 0x027027ecUL, 0xbb48f071UL, 0xde2f4cc9UL, - 0x3080f9dbUL, 0x55e74563UL, 0x9ca03f6bUL, 0xf9c783d3UL, 0x176836c1UL, - 0x720f8a79UL, 0xcb375de4UL, 0xae50e15cUL, 0x40ff544eUL, 0x2598e8f6UL, - 0x73888baeUL, 0x16ef3716UL, 0xf8408204UL, 0x9d273ebcUL, 0x241fe921UL, - 0x41785599UL, 0xafd7e08bUL, 0xcab05c33UL, 0x3bb659edUL, 0x5ed1e555UL, - 0xb07e5047UL, 0xd519ecffUL, 0x6c213b62UL, 0x094687daUL, 0xe7e932c8UL, - 0x828e8e70UL, 0xd49eed28UL, 0xb1f95190UL, 0x5f56e482UL, 0x3a31583aUL, - 0x83098fa7UL, 0xe66e331fUL, 0x08c1860dUL, 0x6da63ab5UL, 0xa4e140bdUL, - 0xc186fc05UL, 0x2f294917UL, 0x4a4ef5afUL, 0xf3762232UL, 0x96119e8aUL, - 0x78be2b98UL, 0x1dd99720UL, 0x4bc9f478UL, 0x2eae48c0UL, 0xc001fdd2UL, - 0xa566416aUL, 0x1c5e96f7UL, 0x79392a4fUL, 0x97969f5dUL, 0xf2f123e5UL, - 0x05196b4dUL, 0x607ed7f5UL, 0x8ed162e7UL, 0xebb6de5fUL, 0x528e09c2UL, - 0x37e9b57aUL, 0xd9460068UL, 0xbc21bcd0UL, 0xea31df88UL, 0x8f566330UL, - 0x61f9d622UL, 0x049e6a9aUL, 0xbda6bd07UL, 0xd8c101bfUL, 0x366eb4adUL, - 0x53090815UL, 0x9a4e721dUL, 0xff29cea5UL, 0x11867bb7UL, 0x74e1c70fUL, - 0xcdd91092UL, 0xa8beac2aUL, 0x46111938UL, 0x2376a580UL, 0x7566c6d8UL, - 0x10017a60UL, 0xfeaecf72UL, 0x9bc973caUL, 0x22f1a457UL, 0x479618efUL, - 0xa939adfdUL, 0xcc5e1145UL, 0x06ee4d76UL, 0x6389f1ceUL, 0x8d2644dcUL, - 0xe841f864UL, 0x51792ff9UL, 0x341e9341UL, 0xdab12653UL, 0xbfd69aebUL, - 0xe9c6f9b3UL, 0x8ca1450bUL, 0x620ef019UL, 0x07694ca1UL, 0xbe519b3cUL, - 0xdb362784UL, 0x35999296UL, 0x50fe2e2eUL, 0x99b95426UL, 0xfcdee89eUL, - 0x12715d8cUL, 0x7716e134UL, 0xce2e36a9UL, 0xab498a11UL, 0x45e63f03UL, - 0x208183bbUL, 0x7691e0e3UL, 0x13f65c5bUL, 0xfd59e949UL, 0x983e55f1UL, - 0x2106826cUL, 0x44613ed4UL, 0xaace8bc6UL, 0xcfa9377eUL, 0x38417fd6UL, - 0x5d26c36eUL, 0xb389767cUL, 0xd6eecac4UL, 0x6fd61d59UL, 0x0ab1a1e1UL, - 0xe41e14f3UL, 0x8179a84bUL, 0xd769cb13UL, 0xb20e77abUL, 0x5ca1c2b9UL, - 0x39c67e01UL, 0x80fea99cUL, 0xe5991524UL, 0x0b36a036UL, 0x6e511c8eUL, - 0xa7166686UL, 0xc271da3eUL, 0x2cde6f2cUL, 0x49b9d394UL, 0xf0810409UL, - 0x95e6b8b1UL, 0x7b490da3UL, 0x1e2eb11bUL, 0x483ed243UL, 0x2d596efbUL, - 0xc3f6dbe9UL, 0xa6916751UL, 0x1fa9b0ccUL, 0x7ace0c74UL, 0x9461b966UL, - 0xf10605deUL -#endif - } -}; diff --git a/code/3rd_glpk/zlib/deflate.c b/code/3rd_glpk/zlib/deflate.c deleted file mode 100644 index 5c4022f3..00000000 --- a/code/3rd_glpk/zlib/deflate.c +++ /dev/null @@ -1,1834 +0,0 @@ -/* deflate.c -- compress data using the deflation algorithm - * Copyright (C) 1995-2010 Jean-loup Gailly and Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* - * ALGORITHM - * - * The "deflation" process depends on being able to identify portions - * of the input text which are identical to earlier input (within a - * sliding window trailing behind the input currently being processed). - * - * The most straightforward technique turns out to be the fastest for - * most input files: try all possible matches and select the longest. - * The key feature of this algorithm is that insertions into the string - * dictionary are very simple and thus fast, and deletions are avoided - * completely. Insertions are performed at each input character, whereas - * string matches are performed only when the previous match ends. So it - * is preferable to spend more time in matches to allow very fast string - * insertions and avoid deletions. The matching algorithm for small - * strings is inspired from that of Rabin & Karp. A brute force approach - * is used to find longer strings when a small match has been found. - * A similar algorithm is used in comic (by Jan-Mark Wams) and freeze - * (by Leonid Broukhis). - * A previous version of this file used a more sophisticated algorithm - * (by Fiala and Greene) which is guaranteed to run in linear amortized - * time, but has a larger average cost, uses more memory and is patented. - * However the F&G algorithm may be faster for some highly redundant - * files if the parameter max_chain_length (described below) is too large. - * - * ACKNOWLEDGEMENTS - * - * The idea of lazy evaluation of matches is due to Jan-Mark Wams, and - * I found it in 'freeze' written by Leonid Broukhis. - * Thanks to many people for bug reports and testing. - * - * REFERENCES - * - * Deutsch, L.P.,"DEFLATE Compressed Data Format Specification". - * Available in http://www.ietf.org/rfc/rfc1951.txt - * - * A description of the Rabin and Karp algorithm is given in the book - * "Algorithms" by R. Sedgewick, Addison-Wesley, p252. - * - * Fiala,E.R., and Greene,D.H. - * Data Compression with Finite Windows, Comm.ACM, 32,4 (1989) 490-595 - * - */ - -/* @(#) $Id$ */ - -#include "deflate.h" - -const char deflate_copyright[] = - " deflate 1.2.5 Copyright 1995-2010 Jean-loup Gailly and Mark Adler "; -/* - If you use the zlib library in a product, an acknowledgment is welcome - in the documentation of your product. If for some reason you cannot - include such an acknowledgment, I would appreciate that you keep this - copyright string in the executable of your product. - */ - -/* =========================================================================== - * Function prototypes. - */ -typedef enum { - need_more, /* block not completed, need more input or more output */ - block_done, /* block flush performed */ - finish_started, /* finish started, need only more output at next deflate */ - finish_done /* finish done, accept no more input or output */ -} block_state; - -typedef block_state (*compress_func) OF((deflate_state *s, int flush)); -/* Compression function. Returns the block state after the call. */ - -local void fill_window OF((deflate_state *s)); -local block_state deflate_stored OF((deflate_state *s, int flush)); -local block_state deflate_fast OF((deflate_state *s, int flush)); -#ifndef FASTEST -local block_state deflate_slow OF((deflate_state *s, int flush)); -#endif -local block_state deflate_rle OF((deflate_state *s, int flush)); -local block_state deflate_huff OF((deflate_state *s, int flush)); -local void lm_init OF((deflate_state *s)); -local void putShortMSB OF((deflate_state *s, uInt b)); -local void flush_pending OF((z_streamp strm)); -local int read_buf OF((z_streamp strm, Bytef *buf, unsigned size)); -#ifdef ASMV - void match_init OF((void)); /* asm code initialization */ - uInt longest_match OF((deflate_state *s, IPos cur_match)); -#else -local uInt longest_match OF((deflate_state *s, IPos cur_match)); -#endif - -#ifdef DEBUG -local void check_match OF((deflate_state *s, IPos start, IPos match, - int length)); -#endif - -/* =========================================================================== - * Local data - */ - -#define NIL 0 -/* Tail of hash chains */ - -#ifndef TOO_FAR -# define TOO_FAR 4096 -#endif -/* Matches of length 3 are discarded if their distance exceeds TOO_FAR */ - -/* Values for max_lazy_match, good_match and max_chain_length, depending on - * the desired pack level (0..9). The values given below have been tuned to - * exclude worst case performance for pathological files. Better values may be - * found for specific files. - */ -typedef struct config_s { - ush good_length; /* reduce lazy search above this match length */ - ush max_lazy; /* do not perform lazy search above this match length */ - ush nice_length; /* quit search above this match length */ - ush max_chain; - compress_func func; -} config; - -#ifdef FASTEST -local const config configuration_table[2] = { -/* good lazy nice chain */ -/* 0 */ {0, 0, 0, 0, deflate_stored}, /* store only */ -/* 1 */ {4, 4, 8, 4, deflate_fast}}; /* max speed, no lazy matches */ -#else -local const config configuration_table[10] = { -/* good lazy nice chain */ -/* 0 */ {0, 0, 0, 0, deflate_stored}, /* store only */ -/* 1 */ {4, 4, 8, 4, deflate_fast}, /* max speed, no lazy matches */ -/* 2 */ {4, 5, 16, 8, deflate_fast}, -/* 3 */ {4, 6, 32, 32, deflate_fast}, - -/* 4 */ {4, 4, 16, 16, deflate_slow}, /* lazy matches */ -/* 5 */ {8, 16, 32, 32, deflate_slow}, -/* 6 */ {8, 16, 128, 128, deflate_slow}, -/* 7 */ {8, 32, 128, 256, deflate_slow}, -/* 8 */ {32, 128, 258, 1024, deflate_slow}, -/* 9 */ {32, 258, 258, 4096, deflate_slow}}; /* max compression */ -#endif - -/* Note: the deflate() code requires max_lazy >= MIN_MATCH and max_chain >= 4 - * For deflate_fast() (levels <= 3) good is ignored and lazy has a different - * meaning. - */ - -#define EQUAL 0 -/* result of memcmp for equal strings */ - -#ifndef NO_DUMMY_DECL -struct static_tree_desc_s {int dummy;}; /* for buggy compilers */ -#endif - -/* =========================================================================== - * Update a hash value with the given input byte - * IN assertion: all calls to to UPDATE_HASH are made with consecutive - * input characters, so that a running hash key can be computed from the - * previous key instead of complete recalculation each time. - */ -#define UPDATE_HASH(s,h,c) (h = (((h)<hash_shift) ^ (c)) & s->hash_mask) - - -/* =========================================================================== - * Insert string str in the dictionary and set match_head to the previous head - * of the hash chain (the most recent string with same hash key). Return - * the previous length of the hash chain. - * If this file is compiled with -DFASTEST, the compression level is forced - * to 1, and no hash chains are maintained. - * IN assertion: all calls to to INSERT_STRING are made with consecutive - * input characters and the first MIN_MATCH bytes of str are valid - * (except for the last MIN_MATCH-1 bytes of the input file). - */ -#ifdef FASTEST -#define INSERT_STRING(s, str, match_head) \ - (UPDATE_HASH(s, s->ins_h, s->window[(str) + (MIN_MATCH-1)]), \ - match_head = s->head[s->ins_h], \ - s->head[s->ins_h] = (Pos)(str)) -#else -#define INSERT_STRING(s, str, match_head) \ - (UPDATE_HASH(s, s->ins_h, s->window[(str) + (MIN_MATCH-1)]), \ - match_head = s->prev[(str) & s->w_mask] = s->head[s->ins_h], \ - s->head[s->ins_h] = (Pos)(str)) -#endif - -/* =========================================================================== - * Initialize the hash table (avoiding 64K overflow for 16 bit systems). - * prev[] will be initialized on the fly. - */ -#define CLEAR_HASH(s) \ - s->head[s->hash_size-1] = NIL; \ - zmemzero((Bytef *)s->head, (unsigned)(s->hash_size-1)*sizeof(*s->head)); - -/* ========================================================================= */ -int ZEXPORT deflateInit_(strm, level, version, stream_size) - z_streamp strm; - int level; - const char *version; - int stream_size; -{ - return deflateInit2_(strm, level, Z_DEFLATED, MAX_WBITS, DEF_MEM_LEVEL, - Z_DEFAULT_STRATEGY, version, stream_size); - /* To do: ignore strm->next_in if we use it as window */ -} - -/* ========================================================================= */ -int ZEXPORT deflateInit2_(strm, level, method, windowBits, memLevel, strategy, - version, stream_size) - z_streamp strm; - int level; - int method; - int windowBits; - int memLevel; - int strategy; - const char *version; - int stream_size; -{ - deflate_state *s; - int wrap = 1; - static const char my_version[] = ZLIB_VERSION; - - ushf *overlay; - /* We overlay pending_buf and d_buf+l_buf. This works since the average - * output size for (length,distance) codes is <= 24 bits. - */ - - if (version == Z_NULL || version[0] != my_version[0] || - stream_size != sizeof(z_stream)) { - return Z_VERSION_ERROR; - } - if (strm == Z_NULL) return Z_STREAM_ERROR; - - strm->msg = Z_NULL; - if (strm->zalloc == (alloc_func)0) { - strm->zalloc = zcalloc; - strm->opaque = (voidpf)0; - } - if (strm->zfree == (free_func)0) strm->zfree = zcfree; - -#ifdef FASTEST - if (level != 0) level = 1; -#else - if (level == Z_DEFAULT_COMPRESSION) level = 6; -#endif - - if (windowBits < 0) { /* suppress zlib wrapper */ - wrap = 0; - windowBits = -windowBits; - } -#ifdef GZIP - else if (windowBits > 15) { - wrap = 2; /* write gzip wrapper instead */ - windowBits -= 16; - } -#endif - if (memLevel < 1 || memLevel > MAX_MEM_LEVEL || method != Z_DEFLATED || - windowBits < 8 || windowBits > 15 || level < 0 || level > 9 || - strategy < 0 || strategy > Z_FIXED) { - return Z_STREAM_ERROR; - } - if (windowBits == 8) windowBits = 9; /* until 256-byte window bug fixed */ - s = (deflate_state *) ZALLOC(strm, 1, sizeof(deflate_state)); - if (s == Z_NULL) return Z_MEM_ERROR; - strm->state = (struct internal_state FAR *)s; - s->strm = strm; - - s->wrap = wrap; - s->gzhead = Z_NULL; - s->w_bits = windowBits; - s->w_size = 1 << s->w_bits; - s->w_mask = s->w_size - 1; - - s->hash_bits = memLevel + 7; - s->hash_size = 1 << s->hash_bits; - s->hash_mask = s->hash_size - 1; - s->hash_shift = ((s->hash_bits+MIN_MATCH-1)/MIN_MATCH); - - s->window = (Bytef *) ZALLOC(strm, s->w_size, 2*sizeof(Byte)); - s->prev = (Posf *) ZALLOC(strm, s->w_size, sizeof(Pos)); - s->head = (Posf *) ZALLOC(strm, s->hash_size, sizeof(Pos)); - - s->high_water = 0; /* nothing written to s->window yet */ - - s->lit_bufsize = 1 << (memLevel + 6); /* 16K elements by default */ - - overlay = (ushf *) ZALLOC(strm, s->lit_bufsize, sizeof(ush)+2); - s->pending_buf = (uchf *) overlay; - s->pending_buf_size = (ulg)s->lit_bufsize * (sizeof(ush)+2L); - - if (s->window == Z_NULL || s->prev == Z_NULL || s->head == Z_NULL || - s->pending_buf == Z_NULL) { - s->status = FINISH_STATE; - strm->msg = (char*)ERR_MSG(Z_MEM_ERROR); - deflateEnd (strm); - return Z_MEM_ERROR; - } - s->d_buf = overlay + s->lit_bufsize/sizeof(ush); - s->l_buf = s->pending_buf + (1+sizeof(ush))*s->lit_bufsize; - - s->level = level; - s->strategy = strategy; - s->method = (Byte)method; - - return deflateReset(strm); -} - -/* ========================================================================= */ -int ZEXPORT deflateSetDictionary (strm, dictionary, dictLength) - z_streamp strm; - const Bytef *dictionary; - uInt dictLength; -{ - deflate_state *s; - uInt length = dictLength; - uInt n; - IPos hash_head = 0; - - if (strm == Z_NULL || strm->state == Z_NULL || dictionary == Z_NULL || - strm->state->wrap == 2 || - (strm->state->wrap == 1 && strm->state->status != INIT_STATE)) - return Z_STREAM_ERROR; - - s = strm->state; - if (s->wrap) - strm->adler = adler32(strm->adler, dictionary, dictLength); - - if (length < MIN_MATCH) return Z_OK; - if (length > s->w_size) { - length = s->w_size; - dictionary += dictLength - length; /* use the tail of the dictionary */ - } - zmemcpy(s->window, dictionary, length); - s->strstart = length; - s->block_start = (long)length; - - /* Insert all strings in the hash table (except for the last two bytes). - * s->lookahead stays null, so s->ins_h will be recomputed at the next - * call of fill_window. - */ - s->ins_h = s->window[0]; - UPDATE_HASH(s, s->ins_h, s->window[1]); - for (n = 0; n <= length - MIN_MATCH; n++) { - INSERT_STRING(s, n, hash_head); - } - if (hash_head) hash_head = 0; /* to make compiler happy */ - return Z_OK; -} - -/* ========================================================================= */ -int ZEXPORT deflateReset (strm) - z_streamp strm; -{ - deflate_state *s; - - if (strm == Z_NULL || strm->state == Z_NULL || - strm->zalloc == (alloc_func)0 || strm->zfree == (free_func)0) { - return Z_STREAM_ERROR; - } - - strm->total_in = strm->total_out = 0; - strm->msg = Z_NULL; /* use zfree if we ever allocate msg dynamically */ - strm->data_type = Z_UNKNOWN; - - s = (deflate_state *)strm->state; - s->pending = 0; - s->pending_out = s->pending_buf; - - if (s->wrap < 0) { - s->wrap = -s->wrap; /* was made negative by deflate(..., Z_FINISH); */ - } - s->status = s->wrap ? INIT_STATE : BUSY_STATE; - strm->adler = -#ifdef GZIP - s->wrap == 2 ? crc32(0L, Z_NULL, 0) : -#endif - adler32(0L, Z_NULL, 0); - s->last_flush = Z_NO_FLUSH; - - _tr_init(s); - lm_init(s); - - return Z_OK; -} - -/* ========================================================================= */ -int ZEXPORT deflateSetHeader (strm, head) - z_streamp strm; - gz_headerp head; -{ - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - if (strm->state->wrap != 2) return Z_STREAM_ERROR; - strm->state->gzhead = head; - return Z_OK; -} - -/* ========================================================================= */ -int ZEXPORT deflatePrime (strm, bits, value) - z_streamp strm; - int bits; - int value; -{ - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - strm->state->bi_valid = bits; - strm->state->bi_buf = (ush)(value & ((1 << bits) - 1)); - return Z_OK; -} - -/* ========================================================================= */ -int ZEXPORT deflateParams(strm, level, strategy) - z_streamp strm; - int level; - int strategy; -{ - deflate_state *s; - compress_func func; - int err = Z_OK; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - s = strm->state; - -#ifdef FASTEST - if (level != 0) level = 1; -#else - if (level == Z_DEFAULT_COMPRESSION) level = 6; -#endif - if (level < 0 || level > 9 || strategy < 0 || strategy > Z_FIXED) { - return Z_STREAM_ERROR; - } - func = configuration_table[s->level].func; - - if ((strategy != s->strategy || func != configuration_table[level].func) && - strm->total_in != 0) { - /* Flush the last buffer: */ - err = deflate(strm, Z_BLOCK); - } - if (s->level != level) { - s->level = level; - s->max_lazy_match = configuration_table[level].max_lazy; - s->good_match = configuration_table[level].good_length; - s->nice_match = configuration_table[level].nice_length; - s->max_chain_length = configuration_table[level].max_chain; - } - s->strategy = strategy; - return err; -} - -/* ========================================================================= */ -int ZEXPORT deflateTune(strm, good_length, max_lazy, nice_length, max_chain) - z_streamp strm; - int good_length; - int max_lazy; - int nice_length; - int max_chain; -{ - deflate_state *s; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - s = strm->state; - s->good_match = good_length; - s->max_lazy_match = max_lazy; - s->nice_match = nice_length; - s->max_chain_length = max_chain; - return Z_OK; -} - -/* ========================================================================= - * For the default windowBits of 15 and memLevel of 8, this function returns - * a close to exact, as well as small, upper bound on the compressed size. - * They are coded as constants here for a reason--if the #define's are - * changed, then this function needs to be changed as well. The return - * value for 15 and 8 only works for those exact settings. - * - * For any setting other than those defaults for windowBits and memLevel, - * the value returned is a conservative worst case for the maximum expansion - * resulting from using fixed blocks instead of stored blocks, which deflate - * can emit on compressed data for some combinations of the parameters. - * - * This function could be more sophisticated to provide closer upper bounds for - * every combination of windowBits and memLevel. But even the conservative - * upper bound of about 14% expansion does not seem onerous for output buffer - * allocation. - */ -uLong ZEXPORT deflateBound(strm, sourceLen) - z_streamp strm; - uLong sourceLen; -{ - deflate_state *s; - uLong complen, wraplen; - Bytef *str; - - /* conservative upper bound for compressed data */ - complen = sourceLen + - ((sourceLen + 7) >> 3) + ((sourceLen + 63) >> 6) + 5; - - /* if can't get parameters, return conservative bound plus zlib wrapper */ - if (strm == Z_NULL || strm->state == Z_NULL) - return complen + 6; - - /* compute wrapper length */ - s = strm->state; - switch (s->wrap) { - case 0: /* raw deflate */ - wraplen = 0; - break; - case 1: /* zlib wrapper */ - wraplen = 6 + (s->strstart ? 4 : 0); - break; - case 2: /* gzip wrapper */ - wraplen = 18; - if (s->gzhead != Z_NULL) { /* user-supplied gzip header */ - if (s->gzhead->extra != Z_NULL) - wraplen += 2 + s->gzhead->extra_len; - str = s->gzhead->name; - if (str != Z_NULL) - do { - wraplen++; - } while (*str++); - str = s->gzhead->comment; - if (str != Z_NULL) - do { - wraplen++; - } while (*str++); - if (s->gzhead->hcrc) - wraplen += 2; - } - break; - default: /* for compiler happiness */ - wraplen = 6; - } - - /* if not default parameters, return conservative bound */ - if (s->w_bits != 15 || s->hash_bits != 8 + 7) - return complen + wraplen; - - /* default settings: return tight bound for that case */ - return sourceLen + (sourceLen >> 12) + (sourceLen >> 14) + - (sourceLen >> 25) + 13 - 6 + wraplen; -} - -/* ========================================================================= - * Put a short in the pending buffer. The 16-bit value is put in MSB order. - * IN assertion: the stream state is correct and there is enough room in - * pending_buf. - */ -local void putShortMSB (s, b) - deflate_state *s; - uInt b; -{ - put_byte(s, (Byte)(b >> 8)); - put_byte(s, (Byte)(b & 0xff)); -} - -/* ========================================================================= - * Flush as much pending output as possible. All deflate() output goes - * through this function so some applications may wish to modify it - * to avoid allocating a large strm->next_out buffer and copying into it. - * (See also read_buf()). - */ -local void flush_pending(strm) - z_streamp strm; -{ - unsigned len = strm->state->pending; - - if (len > strm->avail_out) len = strm->avail_out; - if (len == 0) return; - - zmemcpy(strm->next_out, strm->state->pending_out, len); - strm->next_out += len; - strm->state->pending_out += len; - strm->total_out += len; - strm->avail_out -= len; - strm->state->pending -= len; - if (strm->state->pending == 0) { - strm->state->pending_out = strm->state->pending_buf; - } -} - -/* ========================================================================= */ -int ZEXPORT deflate (strm, flush) - z_streamp strm; - int flush; -{ - int old_flush; /* value of flush param for previous deflate call */ - deflate_state *s; - - if (strm == Z_NULL || strm->state == Z_NULL || - flush > Z_BLOCK || flush < 0) { - return Z_STREAM_ERROR; - } - s = strm->state; - - if (strm->next_out == Z_NULL || - (strm->next_in == Z_NULL && strm->avail_in != 0) || - (s->status == FINISH_STATE && flush != Z_FINISH)) { - ERR_RETURN(strm, Z_STREAM_ERROR); - } - if (strm->avail_out == 0) ERR_RETURN(strm, Z_BUF_ERROR); - - s->strm = strm; /* just in case */ - old_flush = s->last_flush; - s->last_flush = flush; - - /* Write the header */ - if (s->status == INIT_STATE) { -#ifdef GZIP - if (s->wrap == 2) { - strm->adler = crc32(0L, Z_NULL, 0); - put_byte(s, 31); - put_byte(s, 139); - put_byte(s, 8); - if (s->gzhead == Z_NULL) { - put_byte(s, 0); - put_byte(s, 0); - put_byte(s, 0); - put_byte(s, 0); - put_byte(s, 0); - put_byte(s, s->level == 9 ? 2 : - (s->strategy >= Z_HUFFMAN_ONLY || s->level < 2 ? - 4 : 0)); - put_byte(s, OS_CODE); - s->status = BUSY_STATE; - } - else { - put_byte(s, (s->gzhead->text ? 1 : 0) + - (s->gzhead->hcrc ? 2 : 0) + - (s->gzhead->extra == Z_NULL ? 0 : 4) + - (s->gzhead->name == Z_NULL ? 0 : 8) + - (s->gzhead->comment == Z_NULL ? 0 : 16) - ); - put_byte(s, (Byte)(s->gzhead->time & 0xff)); - put_byte(s, (Byte)((s->gzhead->time >> 8) & 0xff)); - put_byte(s, (Byte)((s->gzhead->time >> 16) & 0xff)); - put_byte(s, (Byte)((s->gzhead->time >> 24) & 0xff)); - put_byte(s, s->level == 9 ? 2 : - (s->strategy >= Z_HUFFMAN_ONLY || s->level < 2 ? - 4 : 0)); - put_byte(s, s->gzhead->os & 0xff); - if (s->gzhead->extra != Z_NULL) { - put_byte(s, s->gzhead->extra_len & 0xff); - put_byte(s, (s->gzhead->extra_len >> 8) & 0xff); - } - if (s->gzhead->hcrc) - strm->adler = crc32(strm->adler, s->pending_buf, - s->pending); - s->gzindex = 0; - s->status = EXTRA_STATE; - } - } - else -#endif - { - uInt header = (Z_DEFLATED + ((s->w_bits-8)<<4)) << 8; - uInt level_flags; - - if (s->strategy >= Z_HUFFMAN_ONLY || s->level < 2) - level_flags = 0; - else if (s->level < 6) - level_flags = 1; - else if (s->level == 6) - level_flags = 2; - else - level_flags = 3; - header |= (level_flags << 6); - if (s->strstart != 0) header |= PRESET_DICT; - header += 31 - (header % 31); - - s->status = BUSY_STATE; - putShortMSB(s, header); - - /* Save the adler32 of the preset dictionary: */ - if (s->strstart != 0) { - putShortMSB(s, (uInt)(strm->adler >> 16)); - putShortMSB(s, (uInt)(strm->adler & 0xffff)); - } - strm->adler = adler32(0L, Z_NULL, 0); - } - } -#ifdef GZIP - if (s->status == EXTRA_STATE) { - if (s->gzhead->extra != Z_NULL) { - uInt beg = s->pending; /* start of bytes to update crc */ - - while (s->gzindex < (s->gzhead->extra_len & 0xffff)) { - if (s->pending == s->pending_buf_size) { - if (s->gzhead->hcrc && s->pending > beg) - strm->adler = crc32(strm->adler, s->pending_buf + beg, - s->pending - beg); - flush_pending(strm); - beg = s->pending; - if (s->pending == s->pending_buf_size) - break; - } - put_byte(s, s->gzhead->extra[s->gzindex]); - s->gzindex++; - } - if (s->gzhead->hcrc && s->pending > beg) - strm->adler = crc32(strm->adler, s->pending_buf + beg, - s->pending - beg); - if (s->gzindex == s->gzhead->extra_len) { - s->gzindex = 0; - s->status = NAME_STATE; - } - } - else - s->status = NAME_STATE; - } - if (s->status == NAME_STATE) { - if (s->gzhead->name != Z_NULL) { - uInt beg = s->pending; /* start of bytes to update crc */ - int val; - - do { - if (s->pending == s->pending_buf_size) { - if (s->gzhead->hcrc && s->pending > beg) - strm->adler = crc32(strm->adler, s->pending_buf + beg, - s->pending - beg); - flush_pending(strm); - beg = s->pending; - if (s->pending == s->pending_buf_size) { - val = 1; - break; - } - } - val = s->gzhead->name[s->gzindex++]; - put_byte(s, val); - } while (val != 0); - if (s->gzhead->hcrc && s->pending > beg) - strm->adler = crc32(strm->adler, s->pending_buf + beg, - s->pending - beg); - if (val == 0) { - s->gzindex = 0; - s->status = COMMENT_STATE; - } - } - else - s->status = COMMENT_STATE; - } - if (s->status == COMMENT_STATE) { - if (s->gzhead->comment != Z_NULL) { - uInt beg = s->pending; /* start of bytes to update crc */ - int val; - - do { - if (s->pending == s->pending_buf_size) { - if (s->gzhead->hcrc && s->pending > beg) - strm->adler = crc32(strm->adler, s->pending_buf + beg, - s->pending - beg); - flush_pending(strm); - beg = s->pending; - if (s->pending == s->pending_buf_size) { - val = 1; - break; - } - } - val = s->gzhead->comment[s->gzindex++]; - put_byte(s, val); - } while (val != 0); - if (s->gzhead->hcrc && s->pending > beg) - strm->adler = crc32(strm->adler, s->pending_buf + beg, - s->pending - beg); - if (val == 0) - s->status = HCRC_STATE; - } - else - s->status = HCRC_STATE; - } - if (s->status == HCRC_STATE) { - if (s->gzhead->hcrc) { - if (s->pending + 2 > s->pending_buf_size) - flush_pending(strm); - if (s->pending + 2 <= s->pending_buf_size) { - put_byte(s, (Byte)(strm->adler & 0xff)); - put_byte(s, (Byte)((strm->adler >> 8) & 0xff)); - strm->adler = crc32(0L, Z_NULL, 0); - s->status = BUSY_STATE; - } - } - else - s->status = BUSY_STATE; - } -#endif - - /* Flush as much pending output as possible */ - if (s->pending != 0) { - flush_pending(strm); - if (strm->avail_out == 0) { - /* Since avail_out is 0, deflate will be called again with - * more output space, but possibly with both pending and - * avail_in equal to zero. There won't be anything to do, - * but this is not an error situation so make sure we - * return OK instead of BUF_ERROR at next call of deflate: - */ - s->last_flush = -1; - return Z_OK; - } - - /* Make sure there is something to do and avoid duplicate consecutive - * flushes. For repeated and useless calls with Z_FINISH, we keep - * returning Z_STREAM_END instead of Z_BUF_ERROR. - */ - } else if (strm->avail_in == 0 && flush <= old_flush && - flush != Z_FINISH) { - ERR_RETURN(strm, Z_BUF_ERROR); - } - - /* User must not provide more input after the first FINISH: */ - if (s->status == FINISH_STATE && strm->avail_in != 0) { - ERR_RETURN(strm, Z_BUF_ERROR); - } - - /* Start a new block or continue the current one. - */ - if (strm->avail_in != 0 || s->lookahead != 0 || - (flush != Z_NO_FLUSH && s->status != FINISH_STATE)) { - block_state bstate; - - bstate = s->strategy == Z_HUFFMAN_ONLY ? deflate_huff(s, flush) : - (s->strategy == Z_RLE ? deflate_rle(s, flush) : - (*(configuration_table[s->level].func))(s, flush)); - - if (bstate == finish_started || bstate == finish_done) { - s->status = FINISH_STATE; - } - if (bstate == need_more || bstate == finish_started) { - if (strm->avail_out == 0) { - s->last_flush = -1; /* avoid BUF_ERROR next call, see above */ - } - return Z_OK; - /* If flush != Z_NO_FLUSH && avail_out == 0, the next call - * of deflate should use the same flush parameter to make sure - * that the flush is complete. So we don't have to output an - * empty block here, this will be done at next call. This also - * ensures that for a very small output buffer, we emit at most - * one empty block. - */ - } - if (bstate == block_done) { - if (flush == Z_PARTIAL_FLUSH) { - _tr_align(s); - } else if (flush != Z_BLOCK) { /* FULL_FLUSH or SYNC_FLUSH */ - _tr_stored_block(s, (char*)0, 0L, 0); - /* For a full flush, this empty block will be recognized - * as a special marker by inflate_sync(). - */ - if (flush == Z_FULL_FLUSH) { - CLEAR_HASH(s); /* forget history */ - if (s->lookahead == 0) { - s->strstart = 0; - s->block_start = 0L; - } - } - } - flush_pending(strm); - if (strm->avail_out == 0) { - s->last_flush = -1; /* avoid BUF_ERROR at next call, see above */ - return Z_OK; - } - } - } - Assert(strm->avail_out > 0, "bug2"); - - if (flush != Z_FINISH) return Z_OK; - if (s->wrap <= 0) return Z_STREAM_END; - - /* Write the trailer */ -#ifdef GZIP - if (s->wrap == 2) { - put_byte(s, (Byte)(strm->adler & 0xff)); - put_byte(s, (Byte)((strm->adler >> 8) & 0xff)); - put_byte(s, (Byte)((strm->adler >> 16) & 0xff)); - put_byte(s, (Byte)((strm->adler >> 24) & 0xff)); - put_byte(s, (Byte)(strm->total_in & 0xff)); - put_byte(s, (Byte)((strm->total_in >> 8) & 0xff)); - put_byte(s, (Byte)((strm->total_in >> 16) & 0xff)); - put_byte(s, (Byte)((strm->total_in >> 24) & 0xff)); - } - else -#endif - { - putShortMSB(s, (uInt)(strm->adler >> 16)); - putShortMSB(s, (uInt)(strm->adler & 0xffff)); - } - flush_pending(strm); - /* If avail_out is zero, the application will call deflate again - * to flush the rest. - */ - if (s->wrap > 0) s->wrap = -s->wrap; /* write the trailer only once! */ - return s->pending != 0 ? Z_OK : Z_STREAM_END; -} - -/* ========================================================================= */ -int ZEXPORT deflateEnd (strm) - z_streamp strm; -{ - int status; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - - status = strm->state->status; - if (status != INIT_STATE && - status != EXTRA_STATE && - status != NAME_STATE && - status != COMMENT_STATE && - status != HCRC_STATE && - status != BUSY_STATE && - status != FINISH_STATE) { - return Z_STREAM_ERROR; - } - - /* Deallocate in reverse order of allocations: */ - TRY_FREE(strm, strm->state->pending_buf); - TRY_FREE(strm, strm->state->head); - TRY_FREE(strm, strm->state->prev); - TRY_FREE(strm, strm->state->window); - - ZFREE(strm, strm->state); - strm->state = Z_NULL; - - return status == BUSY_STATE ? Z_DATA_ERROR : Z_OK; -} - -/* ========================================================================= - * Copy the source state to the destination state. - * To simplify the source, this is not supported for 16-bit MSDOS (which - * doesn't have enough memory anyway to duplicate compression states). - */ -int ZEXPORT deflateCopy (dest, source) - z_streamp dest; - z_streamp source; -{ -#ifdef MAXSEG_64K - return Z_STREAM_ERROR; -#else - deflate_state *ds; - deflate_state *ss; - ushf *overlay; - - - if (source == Z_NULL || dest == Z_NULL || source->state == Z_NULL) { - return Z_STREAM_ERROR; - } - - ss = source->state; - - zmemcpy(dest, source, sizeof(z_stream)); - - ds = (deflate_state *) ZALLOC(dest, 1, sizeof(deflate_state)); - if (ds == Z_NULL) return Z_MEM_ERROR; - dest->state = (struct internal_state FAR *) ds; - zmemcpy(ds, ss, sizeof(deflate_state)); - ds->strm = dest; - - ds->window = (Bytef *) ZALLOC(dest, ds->w_size, 2*sizeof(Byte)); - ds->prev = (Posf *) ZALLOC(dest, ds->w_size, sizeof(Pos)); - ds->head = (Posf *) ZALLOC(dest, ds->hash_size, sizeof(Pos)); - overlay = (ushf *) ZALLOC(dest, ds->lit_bufsize, sizeof(ush)+2); - ds->pending_buf = (uchf *) overlay; - - if (ds->window == Z_NULL || ds->prev == Z_NULL || ds->head == Z_NULL || - ds->pending_buf == Z_NULL) { - deflateEnd (dest); - return Z_MEM_ERROR; - } - /* following zmemcpy do not work for 16-bit MSDOS */ - zmemcpy(ds->window, ss->window, ds->w_size * 2 * sizeof(Byte)); - zmemcpy(ds->prev, ss->prev, ds->w_size * sizeof(Pos)); - zmemcpy(ds->head, ss->head, ds->hash_size * sizeof(Pos)); - zmemcpy(ds->pending_buf, ss->pending_buf, (uInt)ds->pending_buf_size); - - ds->pending_out = ds->pending_buf + (ss->pending_out - ss->pending_buf); - ds->d_buf = overlay + ds->lit_bufsize/sizeof(ush); - ds->l_buf = ds->pending_buf + (1+sizeof(ush))*ds->lit_bufsize; - - ds->l_desc.dyn_tree = ds->dyn_ltree; - ds->d_desc.dyn_tree = ds->dyn_dtree; - ds->bl_desc.dyn_tree = ds->bl_tree; - - return Z_OK; -#endif /* MAXSEG_64K */ -} - -/* =========================================================================== - * Read a new buffer from the current input stream, update the adler32 - * and total number of bytes read. All deflate() input goes through - * this function so some applications may wish to modify it to avoid - * allocating a large strm->next_in buffer and copying from it. - * (See also flush_pending()). - */ -local int read_buf(strm, buf, size) - z_streamp strm; - Bytef *buf; - unsigned size; -{ - unsigned len = strm->avail_in; - - if (len > size) len = size; - if (len == 0) return 0; - - strm->avail_in -= len; - - if (strm->state->wrap == 1) { - strm->adler = adler32(strm->adler, strm->next_in, len); - } -#ifdef GZIP - else if (strm->state->wrap == 2) { - strm->adler = crc32(strm->adler, strm->next_in, len); - } -#endif - zmemcpy(buf, strm->next_in, len); - strm->next_in += len; - strm->total_in += len; - - return (int)len; -} - -/* =========================================================================== - * Initialize the "longest match" routines for a new zlib stream - */ -local void lm_init (s) - deflate_state *s; -{ - s->window_size = (ulg)2L*s->w_size; - - CLEAR_HASH(s); - - /* Set the default configuration parameters: - */ - s->max_lazy_match = configuration_table[s->level].max_lazy; - s->good_match = configuration_table[s->level].good_length; - s->nice_match = configuration_table[s->level].nice_length; - s->max_chain_length = configuration_table[s->level].max_chain; - - s->strstart = 0; - s->block_start = 0L; - s->lookahead = 0; - s->match_length = s->prev_length = MIN_MATCH-1; - s->match_available = 0; - s->ins_h = 0; -#ifndef FASTEST -#ifdef ASMV - match_init(); /* initialize the asm code */ -#endif -#endif -} - -#ifndef FASTEST -/* =========================================================================== - * Set match_start to the longest match starting at the given string and - * return its length. Matches shorter or equal to prev_length are discarded, - * in which case the result is equal to prev_length and match_start is - * garbage. - * IN assertions: cur_match is the head of the hash chain for the current - * string (strstart) and its distance is <= MAX_DIST, and prev_length >= 1 - * OUT assertion: the match length is not greater than s->lookahead. - */ -#ifndef ASMV -/* For 80x86 and 680x0, an optimized version will be provided in match.asm or - * match.S. The code will be functionally equivalent. - */ -local uInt longest_match(s, cur_match) - deflate_state *s; - IPos cur_match; /* current match */ -{ - unsigned chain_length = s->max_chain_length;/* max hash chain length */ - register Bytef *scan = s->window + s->strstart; /* current string */ - register Bytef *match; /* matched string */ - register int len; /* length of current match */ - int best_len = s->prev_length; /* best match length so far */ - int nice_match = s->nice_match; /* stop if match long enough */ - IPos limit = s->strstart > (IPos)MAX_DIST(s) ? - s->strstart - (IPos)MAX_DIST(s) : NIL; - /* Stop when cur_match becomes <= limit. To simplify the code, - * we prevent matches with the string of window index 0. - */ - Posf *prev = s->prev; - uInt wmask = s->w_mask; - -#ifdef UNALIGNED_OK - /* Compare two bytes at a time. Note: this is not always beneficial. - * Try with and without -DUNALIGNED_OK to check. - */ - register Bytef *strend = s->window + s->strstart + MAX_MATCH - 1; - register ush scan_start = *(ushf*)scan; - register ush scan_end = *(ushf*)(scan+best_len-1); -#else - register Bytef *strend = s->window + s->strstart + MAX_MATCH; - register Byte scan_end1 = scan[best_len-1]; - register Byte scan_end = scan[best_len]; -#endif - - /* The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of 16. - * It is easy to get rid of this optimization if necessary. - */ - Assert(s->hash_bits >= 8 && MAX_MATCH == 258, "Code too clever"); - - /* Do not waste too much time if we already have a good match: */ - if (s->prev_length >= s->good_match) { - chain_length >>= 2; - } - /* Do not look for matches beyond the end of the input. This is necessary - * to make deflate deterministic. - */ - if ((uInt)nice_match > s->lookahead) nice_match = s->lookahead; - - Assert((ulg)s->strstart <= s->window_size-MIN_LOOKAHEAD, "need lookahead"); - - do { - Assert(cur_match < s->strstart, "no future"); - match = s->window + cur_match; - - /* Skip to next match if the match length cannot increase - * or if the match length is less than 2. Note that the checks below - * for insufficient lookahead only occur occasionally for performance - * reasons. Therefore uninitialized memory will be accessed, and - * conditional jumps will be made that depend on those values. - * However the length of the match is limited to the lookahead, so - * the output of deflate is not affected by the uninitialized values. - */ -#if (defined(UNALIGNED_OK) && MAX_MATCH == 258) - /* This code assumes sizeof(unsigned short) == 2. Do not use - * UNALIGNED_OK if your compiler uses a different size. - */ - if (*(ushf*)(match+best_len-1) != scan_end || - *(ushf*)match != scan_start) continue; - - /* It is not necessary to compare scan[2] and match[2] since they are - * always equal when the other bytes match, given that the hash keys - * are equal and that HASH_BITS >= 8. Compare 2 bytes at a time at - * strstart+3, +5, ... up to strstart+257. We check for insufficient - * lookahead only every 4th comparison; the 128th check will be made - * at strstart+257. If MAX_MATCH-2 is not a multiple of 8, it is - * necessary to put more guard bytes at the end of the window, or - * to check more often for insufficient lookahead. - */ - Assert(scan[2] == match[2], "scan[2]?"); - scan++, match++; - do { - } while (*(ushf*)(scan+=2) == *(ushf*)(match+=2) && - *(ushf*)(scan+=2) == *(ushf*)(match+=2) && - *(ushf*)(scan+=2) == *(ushf*)(match+=2) && - *(ushf*)(scan+=2) == *(ushf*)(match+=2) && - scan < strend); - /* The funny "do {}" generates better code on most compilers */ - - /* Here, scan <= window+strstart+257 */ - Assert(scan <= s->window+(unsigned)(s->window_size-1), "wild scan"); - if (*scan == *match) scan++; - - len = (MAX_MATCH - 1) - (int)(strend-scan); - scan = strend - (MAX_MATCH-1); - -#else /* UNALIGNED_OK */ - - if (match[best_len] != scan_end || - match[best_len-1] != scan_end1 || - *match != *scan || - *++match != scan[1]) continue; - - /* The check at best_len-1 can be removed because it will be made - * again later. (This heuristic is not always a win.) - * It is not necessary to compare scan[2] and match[2] since they - * are always equal when the other bytes match, given that - * the hash keys are equal and that HASH_BITS >= 8. - */ - scan += 2, match++; - Assert(*scan == *match, "match[2]?"); - - /* We check for insufficient lookahead only every 8th comparison; - * the 256th check will be made at strstart+258. - */ - do { - } while (*++scan == *++match && *++scan == *++match && - *++scan == *++match && *++scan == *++match && - *++scan == *++match && *++scan == *++match && - *++scan == *++match && *++scan == *++match && - scan < strend); - - Assert(scan <= s->window+(unsigned)(s->window_size-1), "wild scan"); - - len = MAX_MATCH - (int)(strend - scan); - scan = strend - MAX_MATCH; - -#endif /* UNALIGNED_OK */ - - if (len > best_len) { - s->match_start = cur_match; - best_len = len; - if (len >= nice_match) break; -#ifdef UNALIGNED_OK - scan_end = *(ushf*)(scan+best_len-1); -#else - scan_end1 = scan[best_len-1]; - scan_end = scan[best_len]; -#endif - } - } while ((cur_match = prev[cur_match & wmask]) > limit - && --chain_length != 0); - - if ((uInt)best_len <= s->lookahead) return (uInt)best_len; - return s->lookahead; -} -#endif /* ASMV */ - -#else /* FASTEST */ - -/* --------------------------------------------------------------------------- - * Optimized version for FASTEST only - */ -local uInt longest_match(s, cur_match) - deflate_state *s; - IPos cur_match; /* current match */ -{ - register Bytef *scan = s->window + s->strstart; /* current string */ - register Bytef *match; /* matched string */ - register int len; /* length of current match */ - register Bytef *strend = s->window + s->strstart + MAX_MATCH; - - /* The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of 16. - * It is easy to get rid of this optimization if necessary. - */ - Assert(s->hash_bits >= 8 && MAX_MATCH == 258, "Code too clever"); - - Assert((ulg)s->strstart <= s->window_size-MIN_LOOKAHEAD, "need lookahead"); - - Assert(cur_match < s->strstart, "no future"); - - match = s->window + cur_match; - - /* Return failure if the match length is less than 2: - */ - if (match[0] != scan[0] || match[1] != scan[1]) return MIN_MATCH-1; - - /* The check at best_len-1 can be removed because it will be made - * again later. (This heuristic is not always a win.) - * It is not necessary to compare scan[2] and match[2] since they - * are always equal when the other bytes match, given that - * the hash keys are equal and that HASH_BITS >= 8. - */ - scan += 2, match += 2; - Assert(*scan == *match, "match[2]?"); - - /* We check for insufficient lookahead only every 8th comparison; - * the 256th check will be made at strstart+258. - */ - do { - } while (*++scan == *++match && *++scan == *++match && - *++scan == *++match && *++scan == *++match && - *++scan == *++match && *++scan == *++match && - *++scan == *++match && *++scan == *++match && - scan < strend); - - Assert(scan <= s->window+(unsigned)(s->window_size-1), "wild scan"); - - len = MAX_MATCH - (int)(strend - scan); - - if (len < MIN_MATCH) return MIN_MATCH - 1; - - s->match_start = cur_match; - return (uInt)len <= s->lookahead ? (uInt)len : s->lookahead; -} - -#endif /* FASTEST */ - -#ifdef DEBUG -/* =========================================================================== - * Check that the match at match_start is indeed a match. - */ -local void check_match(s, start, match, length) - deflate_state *s; - IPos start, match; - int length; -{ - /* check that the match is indeed a match */ - if (zmemcmp(s->window + match, - s->window + start, length) != EQUAL) { - fprintf(stderr, " start %u, match %u, length %d\n", - start, match, length); - do { - fprintf(stderr, "%c%c", s->window[match++], s->window[start++]); - } while (--length != 0); - z_error("invalid match"); - } - if (z_verbose > 1) { - fprintf(stderr,"\\[%d,%d]", start-match, length); - do { putc(s->window[start++], stderr); } while (--length != 0); - } -} -#else -# define check_match(s, start, match, length) -#endif /* DEBUG */ - -/* =========================================================================== - * Fill the window when the lookahead becomes insufficient. - * Updates strstart and lookahead. - * - * IN assertion: lookahead < MIN_LOOKAHEAD - * OUT assertions: strstart <= window_size-MIN_LOOKAHEAD - * At least one byte has been read, or avail_in == 0; reads are - * performed for at least two bytes (required for the zip translate_eol - * option -- not supported here). - */ -local void fill_window(s) - deflate_state *s; -{ - register unsigned n, m; - register Posf *p; - unsigned more; /* Amount of free space at the end of the window. */ - uInt wsize = s->w_size; - - do { - more = (unsigned)(s->window_size -(ulg)s->lookahead -(ulg)s->strstart); - - /* Deal with !@#$% 64K limit: */ - if (sizeof(int) <= 2) { - if (more == 0 && s->strstart == 0 && s->lookahead == 0) { - more = wsize; - - } else if (more == (unsigned)(-1)) { - /* Very unlikely, but possible on 16 bit machine if - * strstart == 0 && lookahead == 1 (input done a byte at time) - */ - more--; - } - } - - /* If the window is almost full and there is insufficient lookahead, - * move the upper half to the lower one to make room in the upper half. - */ - if (s->strstart >= wsize+MAX_DIST(s)) { - - zmemcpy(s->window, s->window+wsize, (unsigned)wsize); - s->match_start -= wsize; - s->strstart -= wsize; /* we now have strstart >= MAX_DIST */ - s->block_start -= (long) wsize; - - /* Slide the hash table (could be avoided with 32 bit values - at the expense of memory usage). We slide even when level == 0 - to keep the hash table consistent if we switch back to level > 0 - later. (Using level 0 permanently is not an optimal usage of - zlib, so we don't care about this pathological case.) - */ - n = s->hash_size; - p = &s->head[n]; - do { - m = *--p; - *p = (Pos)(m >= wsize ? m-wsize : NIL); - } while (--n); - - n = wsize; -#ifndef FASTEST - p = &s->prev[n]; - do { - m = *--p; - *p = (Pos)(m >= wsize ? m-wsize : NIL); - /* If n is not on any hash chain, prev[n] is garbage but - * its value will never be used. - */ - } while (--n); -#endif - more += wsize; - } - if (s->strm->avail_in == 0) return; - - /* If there was no sliding: - * strstart <= WSIZE+MAX_DIST-1 && lookahead <= MIN_LOOKAHEAD - 1 && - * more == window_size - lookahead - strstart - * => more >= window_size - (MIN_LOOKAHEAD-1 + WSIZE + MAX_DIST-1) - * => more >= window_size - 2*WSIZE + 2 - * In the BIG_MEM or MMAP case (not yet supported), - * window_size == input_size + MIN_LOOKAHEAD && - * strstart + s->lookahead <= input_size => more >= MIN_LOOKAHEAD. - * Otherwise, window_size == 2*WSIZE so more >= 2. - * If there was sliding, more >= WSIZE. So in all cases, more >= 2. - */ - Assert(more >= 2, "more < 2"); - - n = read_buf(s->strm, s->window + s->strstart + s->lookahead, more); - s->lookahead += n; - - /* Initialize the hash value now that we have some input: */ - if (s->lookahead >= MIN_MATCH) { - s->ins_h = s->window[s->strstart]; - UPDATE_HASH(s, s->ins_h, s->window[s->strstart+1]); -#if MIN_MATCH != 3 - Call UPDATE_HASH() MIN_MATCH-3 more times -#endif - } - /* If the whole input has less than MIN_MATCH bytes, ins_h is garbage, - * but this is not important since only literal bytes will be emitted. - */ - - } while (s->lookahead < MIN_LOOKAHEAD && s->strm->avail_in != 0); - - /* If the WIN_INIT bytes after the end of the current data have never been - * written, then zero those bytes in order to avoid memory check reports of - * the use of uninitialized (or uninitialised as Julian writes) bytes by - * the longest match routines. Update the high water mark for the next - * time through here. WIN_INIT is set to MAX_MATCH since the longest match - * routines allow scanning to strstart + MAX_MATCH, ignoring lookahead. - */ - if (s->high_water < s->window_size) { - ulg curr = s->strstart + (ulg)(s->lookahead); - ulg init; - - if (s->high_water < curr) { - /* Previous high water mark below current data -- zero WIN_INIT - * bytes or up to end of window, whichever is less. - */ - init = s->window_size - curr; - if (init > WIN_INIT) - init = WIN_INIT; - zmemzero(s->window + curr, (unsigned)init); - s->high_water = curr + init; - } - else if (s->high_water < (ulg)curr + WIN_INIT) { - /* High water mark at or above current data, but below current data - * plus WIN_INIT -- zero out to current data plus WIN_INIT, or up - * to end of window, whichever is less. - */ - init = (ulg)curr + WIN_INIT - s->high_water; - if (init > s->window_size - s->high_water) - init = s->window_size - s->high_water; - zmemzero(s->window + s->high_water, (unsigned)init); - s->high_water += init; - } - } -} - -/* =========================================================================== - * Flush the current block, with given end-of-file flag. - * IN assertion: strstart is set to the end of the current match. - */ -#define FLUSH_BLOCK_ONLY(s, last) { \ - _tr_flush_block(s, (s->block_start >= 0L ? \ - (charf *)&s->window[(unsigned)s->block_start] : \ - (charf *)Z_NULL), \ - (ulg)((long)s->strstart - s->block_start), \ - (last)); \ - s->block_start = s->strstart; \ - flush_pending(s->strm); \ - Tracev((stderr,"[FLUSH]")); \ -} - -/* Same but force premature exit if necessary. */ -#define FLUSH_BLOCK(s, last) { \ - FLUSH_BLOCK_ONLY(s, last); \ - if (s->strm->avail_out == 0) return (last) ? finish_started : need_more; \ -} - -/* =========================================================================== - * Copy without compression as much as possible from the input stream, return - * the current block state. - * This function does not insert new strings in the dictionary since - * uncompressible data is probably not useful. This function is used - * only for the level=0 compression option. - * NOTE: this function should be optimized to avoid extra copying from - * window to pending_buf. - */ -local block_state deflate_stored(s, flush) - deflate_state *s; - int flush; -{ - /* Stored blocks are limited to 0xffff bytes, pending_buf is limited - * to pending_buf_size, and each stored block has a 5 byte header: - */ - ulg max_block_size = 0xffff; - ulg max_start; - - if (max_block_size > s->pending_buf_size - 5) { - max_block_size = s->pending_buf_size - 5; - } - - /* Copy as much as possible from input to output: */ - for (;;) { - /* Fill the window as much as possible: */ - if (s->lookahead <= 1) { - - Assert(s->strstart < s->w_size+MAX_DIST(s) || - s->block_start >= (long)s->w_size, "slide too late"); - - fill_window(s); - if (s->lookahead == 0 && flush == Z_NO_FLUSH) return need_more; - - if (s->lookahead == 0) break; /* flush the current block */ - } - Assert(s->block_start >= 0L, "block gone"); - - s->strstart += s->lookahead; - s->lookahead = 0; - - /* Emit a stored block if pending_buf will be full: */ - max_start = s->block_start + max_block_size; - if (s->strstart == 0 || (ulg)s->strstart >= max_start) { - /* strstart == 0 is possible when wraparound on 16-bit machine */ - s->lookahead = (uInt)(s->strstart - max_start); - s->strstart = (uInt)max_start; - FLUSH_BLOCK(s, 0); - } - /* Flush if we may have to slide, otherwise block_start may become - * negative and the data will be gone: - */ - if (s->strstart - (uInt)s->block_start >= MAX_DIST(s)) { - FLUSH_BLOCK(s, 0); - } - } - FLUSH_BLOCK(s, flush == Z_FINISH); - return flush == Z_FINISH ? finish_done : block_done; -} - -/* =========================================================================== - * Compress as much as possible from the input stream, return the current - * block state. - * This function does not perform lazy evaluation of matches and inserts - * new strings in the dictionary only for unmatched strings or for short - * matches. It is used only for the fast compression options. - */ -local block_state deflate_fast(s, flush) - deflate_state *s; - int flush; -{ - IPos hash_head; /* head of the hash chain */ - int bflush; /* set if current block must be flushed */ - - for (;;) { - /* Make sure that we always have enough lookahead, except - * at the end of the input file. We need MAX_MATCH bytes - * for the next match, plus MIN_MATCH bytes to insert the - * string following the next match. - */ - if (s->lookahead < MIN_LOOKAHEAD) { - fill_window(s); - if (s->lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) { - return need_more; - } - if (s->lookahead == 0) break; /* flush the current block */ - } - - /* Insert the string window[strstart .. strstart+2] in the - * dictionary, and set hash_head to the head of the hash chain: - */ - hash_head = NIL; - if (s->lookahead >= MIN_MATCH) { - INSERT_STRING(s, s->strstart, hash_head); - } - - /* Find the longest match, discarding those <= prev_length. - * At this point we have always match_length < MIN_MATCH - */ - if (hash_head != NIL && s->strstart - hash_head <= MAX_DIST(s)) { - /* To simplify the code, we prevent matches with the string - * of window index 0 (in particular we have to avoid a match - * of the string with itself at the start of the input file). - */ - s->match_length = longest_match (s, hash_head); - /* longest_match() sets match_start */ - } - if (s->match_length >= MIN_MATCH) { - check_match(s, s->strstart, s->match_start, s->match_length); - - _tr_tally_dist(s, s->strstart - s->match_start, - s->match_length - MIN_MATCH, bflush); - - s->lookahead -= s->match_length; - - /* Insert new strings in the hash table only if the match length - * is not too large. This saves time but degrades compression. - */ -#ifndef FASTEST - if (s->match_length <= s->max_insert_length && - s->lookahead >= MIN_MATCH) { - s->match_length--; /* string at strstart already in table */ - do { - s->strstart++; - INSERT_STRING(s, s->strstart, hash_head); - /* strstart never exceeds WSIZE-MAX_MATCH, so there are - * always MIN_MATCH bytes ahead. - */ - } while (--s->match_length != 0); - s->strstart++; - } else -#endif - { - s->strstart += s->match_length; - s->match_length = 0; - s->ins_h = s->window[s->strstart]; - UPDATE_HASH(s, s->ins_h, s->window[s->strstart+1]); -#if MIN_MATCH != 3 - Call UPDATE_HASH() MIN_MATCH-3 more times -#endif - /* If lookahead < MIN_MATCH, ins_h is garbage, but it does not - * matter since it will be recomputed at next deflate call. - */ - } - } else { - /* No match, output a literal byte */ - Tracevv((stderr,"%c", s->window[s->strstart])); - _tr_tally_lit (s, s->window[s->strstart], bflush); - s->lookahead--; - s->strstart++; - } - if (bflush) FLUSH_BLOCK(s, 0); - } - FLUSH_BLOCK(s, flush == Z_FINISH); - return flush == Z_FINISH ? finish_done : block_done; -} - -#ifndef FASTEST -/* =========================================================================== - * Same as above, but achieves better compression. We use a lazy - * evaluation for matches: a match is finally adopted only if there is - * no better match at the next window position. - */ -local block_state deflate_slow(s, flush) - deflate_state *s; - int flush; -{ - IPos hash_head; /* head of hash chain */ - int bflush; /* set if current block must be flushed */ - - /* Process the input block. */ - for (;;) { - /* Make sure that we always have enough lookahead, except - * at the end of the input file. We need MAX_MATCH bytes - * for the next match, plus MIN_MATCH bytes to insert the - * string following the next match. - */ - if (s->lookahead < MIN_LOOKAHEAD) { - fill_window(s); - if (s->lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) { - return need_more; - } - if (s->lookahead == 0) break; /* flush the current block */ - } - - /* Insert the string window[strstart .. strstart+2] in the - * dictionary, and set hash_head to the head of the hash chain: - */ - hash_head = NIL; - if (s->lookahead >= MIN_MATCH) { - INSERT_STRING(s, s->strstart, hash_head); - } - - /* Find the longest match, discarding those <= prev_length. - */ - s->prev_length = s->match_length, s->prev_match = s->match_start; - s->match_length = MIN_MATCH-1; - - if (hash_head != NIL && s->prev_length < s->max_lazy_match && - s->strstart - hash_head <= MAX_DIST(s)) { - /* To simplify the code, we prevent matches with the string - * of window index 0 (in particular we have to avoid a match - * of the string with itself at the start of the input file). - */ - s->match_length = longest_match (s, hash_head); - /* longest_match() sets match_start */ - - if (s->match_length <= 5 && (s->strategy == Z_FILTERED -#if TOO_FAR <= 32767 - || (s->match_length == MIN_MATCH && - s->strstart - s->match_start > TOO_FAR) -#endif - )) { - - /* If prev_match is also MIN_MATCH, match_start is garbage - * but we will ignore the current match anyway. - */ - s->match_length = MIN_MATCH-1; - } - } - /* If there was a match at the previous step and the current - * match is not better, output the previous match: - */ - if (s->prev_length >= MIN_MATCH && s->match_length <= s->prev_length) { - uInt max_insert = s->strstart + s->lookahead - MIN_MATCH; - /* Do not insert strings in hash table beyond this. */ - - check_match(s, s->strstart-1, s->prev_match, s->prev_length); - - _tr_tally_dist(s, s->strstart -1 - s->prev_match, - s->prev_length - MIN_MATCH, bflush); - - /* Insert in hash table all strings up to the end of the match. - * strstart-1 and strstart are already inserted. If there is not - * enough lookahead, the last two strings are not inserted in - * the hash table. - */ - s->lookahead -= s->prev_length-1; - s->prev_length -= 2; - do { - if (++s->strstart <= max_insert) { - INSERT_STRING(s, s->strstart, hash_head); - } - } while (--s->prev_length != 0); - s->match_available = 0; - s->match_length = MIN_MATCH-1; - s->strstart++; - - if (bflush) FLUSH_BLOCK(s, 0); - - } else if (s->match_available) { - /* If there was no match at the previous position, output a - * single literal. If there was a match but the current match - * is longer, truncate the previous match to a single literal. - */ - Tracevv((stderr,"%c", s->window[s->strstart-1])); - _tr_tally_lit(s, s->window[s->strstart-1], bflush); - if (bflush) { - FLUSH_BLOCK_ONLY(s, 0); - } - s->strstart++; - s->lookahead--; - if (s->strm->avail_out == 0) return need_more; - } else { - /* There is no previous match to compare with, wait for - * the next step to decide. - */ - s->match_available = 1; - s->strstart++; - s->lookahead--; - } - } - Assert (flush != Z_NO_FLUSH, "no flush?"); - if (s->match_available) { - Tracevv((stderr,"%c", s->window[s->strstart-1])); - _tr_tally_lit(s, s->window[s->strstart-1], bflush); - s->match_available = 0; - } - FLUSH_BLOCK(s, flush == Z_FINISH); - return flush == Z_FINISH ? finish_done : block_done; -} -#endif /* FASTEST */ - -/* =========================================================================== - * For Z_RLE, simply look for runs of bytes, generate matches only of distance - * one. Do not maintain a hash table. (It will be regenerated if this run of - * deflate switches away from Z_RLE.) - */ -local block_state deflate_rle(s, flush) - deflate_state *s; - int flush; -{ - int bflush; /* set if current block must be flushed */ - uInt prev; /* byte at distance one to match */ - Bytef *scan, *strend; /* scan goes up to strend for length of run */ - - for (;;) { - /* Make sure that we always have enough lookahead, except - * at the end of the input file. We need MAX_MATCH bytes - * for the longest encodable run. - */ - if (s->lookahead < MAX_MATCH) { - fill_window(s); - if (s->lookahead < MAX_MATCH && flush == Z_NO_FLUSH) { - return need_more; - } - if (s->lookahead == 0) break; /* flush the current block */ - } - - /* See how many times the previous byte repeats */ - s->match_length = 0; - if (s->lookahead >= MIN_MATCH && s->strstart > 0) { - scan = s->window + s->strstart - 1; - prev = *scan; - if (prev == *++scan && prev == *++scan && prev == *++scan) { - strend = s->window + s->strstart + MAX_MATCH; - do { - } while (prev == *++scan && prev == *++scan && - prev == *++scan && prev == *++scan && - prev == *++scan && prev == *++scan && - prev == *++scan && prev == *++scan && - scan < strend); - s->match_length = MAX_MATCH - (int)(strend - scan); - if (s->match_length > s->lookahead) - s->match_length = s->lookahead; - } - } - - /* Emit match if have run of MIN_MATCH or longer, else emit literal */ - if (s->match_length >= MIN_MATCH) { - check_match(s, s->strstart, s->strstart - 1, s->match_length); - - _tr_tally_dist(s, 1, s->match_length - MIN_MATCH, bflush); - - s->lookahead -= s->match_length; - s->strstart += s->match_length; - s->match_length = 0; - } else { - /* No match, output a literal byte */ - Tracevv((stderr,"%c", s->window[s->strstart])); - _tr_tally_lit (s, s->window[s->strstart], bflush); - s->lookahead--; - s->strstart++; - } - if (bflush) FLUSH_BLOCK(s, 0); - } - FLUSH_BLOCK(s, flush == Z_FINISH); - return flush == Z_FINISH ? finish_done : block_done; -} - -/* =========================================================================== - * For Z_HUFFMAN_ONLY, do not look for matches. Do not maintain a hash table. - * (It will be regenerated if this run of deflate switches away from Huffman.) - */ -local block_state deflate_huff(s, flush) - deflate_state *s; - int flush; -{ - int bflush; /* set if current block must be flushed */ - - for (;;) { - /* Make sure that we have a literal to write. */ - if (s->lookahead == 0) { - fill_window(s); - if (s->lookahead == 0) { - if (flush == Z_NO_FLUSH) - return need_more; - break; /* flush the current block */ - } - } - - /* Output a literal byte */ - s->match_length = 0; - Tracevv((stderr,"%c", s->window[s->strstart])); - _tr_tally_lit (s, s->window[s->strstart], bflush); - s->lookahead--; - s->strstart++; - if (bflush) FLUSH_BLOCK(s, 0); - } - FLUSH_BLOCK(s, flush == Z_FINISH); - return flush == Z_FINISH ? finish_done : block_done; -} diff --git a/code/3rd_glpk/zlib/deflate.h b/code/3rd_glpk/zlib/deflate.h deleted file mode 100644 index cbf0d1ea..00000000 --- a/code/3rd_glpk/zlib/deflate.h +++ /dev/null @@ -1,342 +0,0 @@ -/* deflate.h -- internal compression state - * Copyright (C) 1995-2010 Jean-loup Gailly - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. - */ - -/* @(#) $Id$ */ - -#ifndef DEFLATE_H -#define DEFLATE_H - -#include "zutil.h" - -/* define NO_GZIP when compiling if you want to disable gzip header and - trailer creation by deflate(). NO_GZIP would be used to avoid linking in - the crc code when it is not needed. For shared libraries, gzip encoding - should be left enabled. */ -#ifndef NO_GZIP -# define GZIP -#endif - -/* =========================================================================== - * Internal compression state. - */ - -#define LENGTH_CODES 29 -/* number of length codes, not counting the special END_BLOCK code */ - -#define LITERALS 256 -/* number of literal bytes 0..255 */ - -#define L_CODES (LITERALS+1+LENGTH_CODES) -/* number of Literal or Length codes, including the END_BLOCK code */ - -#define D_CODES 30 -/* number of distance codes */ - -#define BL_CODES 19 -/* number of codes used to transfer the bit lengths */ - -#define HEAP_SIZE (2*L_CODES+1) -/* maximum heap size */ - -#define MAX_BITS 15 -/* All codes must not exceed MAX_BITS bits */ - -#define INIT_STATE 42 -#define EXTRA_STATE 69 -#define NAME_STATE 73 -#define COMMENT_STATE 91 -#define HCRC_STATE 103 -#define BUSY_STATE 113 -#define FINISH_STATE 666 -/* Stream status */ - - -/* Data structure describing a single value and its code string. */ -typedef struct ct_data_s { - union { - ush freq; /* frequency count */ - ush code; /* bit string */ - } fc; - union { - ush dad; /* father node in Huffman tree */ - ush len; /* length of bit string */ - } dl; -} FAR ct_data; - -#define Freq fc.freq -#define Code fc.code -#define Dad dl.dad -#define Len dl.len - -typedef struct static_tree_desc_s static_tree_desc; - -typedef struct tree_desc_s { - ct_data *dyn_tree; /* the dynamic tree */ - int max_code; /* largest code with non zero frequency */ - static_tree_desc *stat_desc; /* the corresponding static tree */ -} FAR tree_desc; - -typedef ush Pos; -typedef Pos FAR Posf; -typedef unsigned IPos; - -/* A Pos is an index in the character window. We use short instead of int to - * save space in the various tables. IPos is used only for parameter passing. - */ - -typedef struct internal_state { - z_streamp strm; /* pointer back to this zlib stream */ - int status; /* as the name implies */ - Bytef *pending_buf; /* output still pending */ - ulg pending_buf_size; /* size of pending_buf */ - Bytef *pending_out; /* next pending byte to output to the stream */ - uInt pending; /* nb of bytes in the pending buffer */ - int wrap; /* bit 0 true for zlib, bit 1 true for gzip */ - gz_headerp gzhead; /* gzip header information to write */ - uInt gzindex; /* where in extra, name, or comment */ - Byte method; /* STORED (for zip only) or DEFLATED */ - int last_flush; /* value of flush param for previous deflate call */ - - /* used by deflate.c: */ - - uInt w_size; /* LZ77 window size (32K by default) */ - uInt w_bits; /* log2(w_size) (8..16) */ - uInt w_mask; /* w_size - 1 */ - - Bytef *window; - /* Sliding window. Input bytes are read into the second half of the window, - * and move to the first half later to keep a dictionary of at least wSize - * bytes. With this organization, matches are limited to a distance of - * wSize-MAX_MATCH bytes, but this ensures that IO is always - * performed with a length multiple of the block size. Also, it limits - * the window size to 64K, which is quite useful on MSDOS. - * To do: use the user input buffer as sliding window. - */ - - ulg window_size; - /* Actual size of window: 2*wSize, except when the user input buffer - * is directly used as sliding window. - */ - - Posf *prev; - /* Link to older string with same hash index. To limit the size of this - * array to 64K, this link is maintained only for the last 32K strings. - * An index in this array is thus a window index modulo 32K. - */ - - Posf *head; /* Heads of the hash chains or NIL. */ - - uInt ins_h; /* hash index of string to be inserted */ - uInt hash_size; /* number of elements in hash table */ - uInt hash_bits; /* log2(hash_size) */ - uInt hash_mask; /* hash_size-1 */ - - uInt hash_shift; - /* Number of bits by which ins_h must be shifted at each input - * step. It must be such that after MIN_MATCH steps, the oldest - * byte no longer takes part in the hash key, that is: - * hash_shift * MIN_MATCH >= hash_bits - */ - - long block_start; - /* Window position at the beginning of the current output block. Gets - * negative when the window is moved backwards. - */ - - uInt match_length; /* length of best match */ - IPos prev_match; /* previous match */ - int match_available; /* set if previous match exists */ - uInt strstart; /* start of string to insert */ - uInt match_start; /* start of matching string */ - uInt lookahead; /* number of valid bytes ahead in window */ - - uInt prev_length; - /* Length of the best match at previous step. Matches not greater than this - * are discarded. This is used in the lazy match evaluation. - */ - - uInt max_chain_length; - /* To speed up deflation, hash chains are never searched beyond this - * length. A higher limit improves compression ratio but degrades the - * speed. - */ - - uInt max_lazy_match; - /* Attempt to find a better match only when the current match is strictly - * smaller than this value. This mechanism is used only for compression - * levels >= 4. - */ -# define max_insert_length max_lazy_match - /* Insert new strings in the hash table only if the match length is not - * greater than this length. This saves time but degrades compression. - * max_insert_length is used only for compression levels <= 3. - */ - - int level; /* compression level (1..9) */ - int strategy; /* favor or force Huffman coding*/ - - uInt good_match; - /* Use a faster search when the previous match is longer than this */ - - int nice_match; /* Stop searching when current match exceeds this */ - - /* used by trees.c: */ - /* Didn't use ct_data typedef below to supress compiler warning */ - struct ct_data_s dyn_ltree[HEAP_SIZE]; /* literal and length tree */ - struct ct_data_s dyn_dtree[2*D_CODES+1]; /* distance tree */ - struct ct_data_s bl_tree[2*BL_CODES+1]; /* Huffman tree for bit lengths */ - - struct tree_desc_s l_desc; /* desc. for literal tree */ - struct tree_desc_s d_desc; /* desc. for distance tree */ - struct tree_desc_s bl_desc; /* desc. for bit length tree */ - - ush bl_count[MAX_BITS+1]; - /* number of codes at each bit length for an optimal tree */ - - int heap[2*L_CODES+1]; /* heap used to build the Huffman trees */ - int heap_len; /* number of elements in the heap */ - int heap_max; /* element of largest frequency */ - /* The sons of heap[n] are heap[2*n] and heap[2*n+1]. heap[0] is not used. - * The same heap array is used to build all trees. - */ - - uch depth[2*L_CODES+1]; - /* Depth of each subtree used as tie breaker for trees of equal frequency - */ - - uchf *l_buf; /* buffer for literals or lengths */ - - uInt lit_bufsize; - /* Size of match buffer for literals/lengths. There are 4 reasons for - * limiting lit_bufsize to 64K: - * - frequencies can be kept in 16 bit counters - * - if compression is not successful for the first block, all input - * data is still in the window so we can still emit a stored block even - * when input comes from standard input. (This can also be done for - * all blocks if lit_bufsize is not greater than 32K.) - * - if compression is not successful for a file smaller than 64K, we can - * even emit a stored file instead of a stored block (saving 5 bytes). - * This is applicable only for zip (not gzip or zlib). - * - creating new Huffman trees less frequently may not provide fast - * adaptation to changes in the input data statistics. (Take for - * example a binary file with poorly compressible code followed by - * a highly compressible string table.) Smaller buffer sizes give - * fast adaptation but have of course the overhead of transmitting - * trees more frequently. - * - I can't count above 4 - */ - - uInt last_lit; /* running index in l_buf */ - - ushf *d_buf; - /* Buffer for distances. To simplify the code, d_buf and l_buf have - * the same number of elements. To use different lengths, an extra flag - * array would be necessary. - */ - - ulg opt_len; /* bit length of current block with optimal trees */ - ulg static_len; /* bit length of current block with static trees */ - uInt matches; /* number of string matches in current block */ - int last_eob_len; /* bit length of EOB code for last block */ - -#ifdef DEBUG - ulg compressed_len; /* total bit length of compressed file mod 2^32 */ - ulg bits_sent; /* bit length of compressed data sent mod 2^32 */ -#endif - - ush bi_buf; - /* Output buffer. bits are inserted starting at the bottom (least - * significant bits). - */ - int bi_valid; - /* Number of valid bits in bi_buf. All bits above the last valid bit - * are always zero. - */ - - ulg high_water; - /* High water mark offset in window for initialized bytes -- bytes above - * this are set to zero in order to avoid memory check warnings when - * longest match routines access bytes past the input. This is then - * updated to the new high water mark. - */ - -} FAR deflate_state; - -/* Output a byte on the stream. - * IN assertion: there is enough room in pending_buf. - */ -#define put_byte(s, c) {s->pending_buf[s->pending++] = (c);} - - -#define MIN_LOOKAHEAD (MAX_MATCH+MIN_MATCH+1) -/* Minimum amount of lookahead, except at the end of the input file. - * See deflate.c for comments about the MIN_MATCH+1. - */ - -#define MAX_DIST(s) ((s)->w_size-MIN_LOOKAHEAD) -/* In order to simplify the code, particularly on 16 bit machines, match - * distances are limited to MAX_DIST instead of WSIZE. - */ - -#define WIN_INIT MAX_MATCH -/* Number of bytes after end of data in window to initialize in order to avoid - memory checker errors from longest match routines */ - - /* in trees.c */ -void ZLIB_INTERNAL _tr_init OF((deflate_state *s)); -int ZLIB_INTERNAL _tr_tally OF((deflate_state *s, unsigned dist, unsigned lc)); -void ZLIB_INTERNAL _tr_flush_block OF((deflate_state *s, charf *buf, - ulg stored_len, int last)); -void ZLIB_INTERNAL _tr_align OF((deflate_state *s)); -void ZLIB_INTERNAL _tr_stored_block OF((deflate_state *s, charf *buf, - ulg stored_len, int last)); - -#define d_code(dist) \ - ((dist) < 256 ? _dist_code[dist] : _dist_code[256+((dist)>>7)]) -/* Mapping from a distance to a distance code. dist is the distance - 1 and - * must not have side effects. _dist_code[256] and _dist_code[257] are never - * used. - */ - -#ifndef DEBUG -/* Inline versions of _tr_tally for speed: */ - -#if defined(GEN_TREES_H) || !defined(STDC) - extern uch ZLIB_INTERNAL _length_code[]; - extern uch ZLIB_INTERNAL _dist_code[]; -#else - extern const uch ZLIB_INTERNAL _length_code[]; - extern const uch ZLIB_INTERNAL _dist_code[]; -#endif - -# define _tr_tally_lit(s, c, flush) \ - { uch cc = (c); \ - s->d_buf[s->last_lit] = 0; \ - s->l_buf[s->last_lit++] = cc; \ - s->dyn_ltree[cc].Freq++; \ - flush = (s->last_lit == s->lit_bufsize-1); \ - } -# define _tr_tally_dist(s, distance, length, flush) \ - { uch len = (length); \ - ush dist = (distance); \ - s->d_buf[s->last_lit] = dist; \ - s->l_buf[s->last_lit++] = len; \ - dist--; \ - s->dyn_ltree[_length_code[len]+LITERALS+1].Freq++; \ - s->dyn_dtree[d_code(dist)].Freq++; \ - flush = (s->last_lit == s->lit_bufsize-1); \ - } -#else -# define _tr_tally_lit(s, c, flush) flush = _tr_tally(s, 0, c) -# define _tr_tally_dist(s, distance, length, flush) \ - flush = _tr_tally(s, distance, length) -#endif - -#endif /* DEFLATE_H */ diff --git a/code/3rd_glpk/zlib/gzclose.c b/code/3rd_glpk/zlib/gzclose.c deleted file mode 100644 index caeb99a3..00000000 --- a/code/3rd_glpk/zlib/gzclose.c +++ /dev/null @@ -1,25 +0,0 @@ -/* gzclose.c -- zlib gzclose() function - * Copyright (C) 2004, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -#include "gzguts.h" - -/* gzclose() is in a separate file so that it is linked in only if it is used. - That way the other gzclose functions can be used instead to avoid linking in - unneeded compression or decompression routines. */ -int ZEXPORT gzclose(file) - gzFile file; -{ -#ifndef NO_GZCOMPRESS - gz_statep state; - - if (file == NULL) - return Z_STREAM_ERROR; - state = (gz_statep)file; - - return state->mode == GZ_READ ? gzclose_r(file) : gzclose_w(file); -#else - return gzclose_r(file); -#endif -} diff --git a/code/3rd_glpk/zlib/gzguts.h b/code/3rd_glpk/zlib/gzguts.h deleted file mode 100644 index 9d01ac7b..00000000 --- a/code/3rd_glpk/zlib/gzguts.h +++ /dev/null @@ -1,74 +0,0 @@ -/* gzguts.h (zlib internal header definitions for gz* operations) */ - -/* Modified by Andrew Makhorin , April 2011 */ - -/* Copyright (C) 2004, 2005, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in - * zlib.h */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. */ - -#ifndef GZGUTS_H -#define GZGUTS_H - -#define ZLIB_INTERNAL - -#include -#include -#include -#include -#include -#include "zio.h" -#include "zlib.h" - -#define local static - -#define zstrerror() strerror(errno) - -#define GZBUFSIZE 8192 - -#define GZ_NONE 0 -#define GZ_READ 7247 -#define GZ_WRITE 31153 -#define GZ_APPEND 1 - -#define LOOK 0 -#define COPY 1 -#define GZIP 2 - -typedef struct -{ int mode; - int fd; - char *path; - z_off64_t pos; - unsigned size; - unsigned want; - unsigned char *in; - unsigned char *out; - unsigned char *next; - unsigned have; - int eof; - z_off64_t start; - z_off64_t raw; - int how; - int direct; - int level; - int strategy; - z_off64_t skip; - int seek; - int err; - char *msg; - z_stream strm; -} gz_state; - -typedef gz_state *gz_statep; - -void ZLIB_INTERNAL gz_error OF((gz_statep, int, const char *)); - -#define GT_OFF(x) (sizeof(int) == sizeof(z_off64_t) && (x) > INT_MAX) - -#endif - -/* eof */ diff --git a/code/3rd_glpk/zlib/gzlib.c b/code/3rd_glpk/zlib/gzlib.c deleted file mode 100644 index 603e60ed..00000000 --- a/code/3rd_glpk/zlib/gzlib.c +++ /dev/null @@ -1,537 +0,0 @@ -/* gzlib.c -- zlib functions common to reading and writing gzip files - * Copyright (C) 2004, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -#include "gzguts.h" - -#if defined(_LARGEFILE64_SOURCE) && _LFS64_LARGEFILE-0 -# define LSEEK lseek64 -#else -# define LSEEK lseek -#endif - -/* Local functions */ -local void gz_reset OF((gz_statep)); -local gzFile gz_open OF((const char *, int, const char *)); - -#if defined UNDER_CE - -/* Map the Windows error number in ERROR to a locale-dependent error message - string and return a pointer to it. Typically, the values for ERROR come - from GetLastError. - - The string pointed to shall not be modified by the application, but may be - overwritten by a subsequent call to gz_strwinerror - - The gz_strwinerror function does not change the current setting of - GetLastError. */ -char ZLIB_INTERNAL *gz_strwinerror (error) - DWORD error; -{ - static char buf[1024]; - - wchar_t *msgbuf; - DWORD lasterr = GetLastError(); - DWORD chars = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM - | FORMAT_MESSAGE_ALLOCATE_BUFFER, - NULL, - error, - 0, /* Default language */ - (LPVOID)&msgbuf, - 0, - NULL); - if (chars != 0) { - /* If there is an \r\n appended, zap it. */ - if (chars >= 2 - && msgbuf[chars - 2] == '\r' && msgbuf[chars - 1] == '\n') { - chars -= 2; - msgbuf[chars] = 0; - } - - if (chars > sizeof (buf) - 1) { - chars = sizeof (buf) - 1; - msgbuf[chars] = 0; - } - - wcstombs(buf, msgbuf, chars + 1); - LocalFree(msgbuf); - } - else { - sprintf(buf, "unknown win32 error (%ld)", error); - } - - SetLastError(lasterr); - return buf; -} - -#endif /* UNDER_CE */ - -/* Reset gzip file state */ -local void gz_reset(state) - gz_statep state; -{ - if (state->mode == GZ_READ) { /* for reading ... */ - state->have = 0; /* no output data available */ - state->eof = 0; /* not at end of file */ - state->how = LOOK; /* look for gzip header */ - state->direct = 1; /* default for empty file */ - } - state->seek = 0; /* no seek request pending */ - gz_error(state, Z_OK, NULL); /* clear error */ - state->pos = 0; /* no uncompressed data yet */ - state->strm.avail_in = 0; /* no input data yet */ -} - -/* Open a gzip file either by name or file descriptor. */ -local gzFile gz_open(path, fd, mode) - const char *path; - int fd; - const char *mode; -{ - gz_statep state; - - /* allocate gzFile structure to return */ - state = malloc(sizeof(gz_state)); - if (state == NULL) - return NULL; - state->size = 0; /* no buffers allocated yet */ - state->want = GZBUFSIZE; /* requested buffer size */ - state->msg = NULL; /* no error message yet */ - - /* interpret mode */ - state->mode = GZ_NONE; - state->level = Z_DEFAULT_COMPRESSION; - state->strategy = Z_DEFAULT_STRATEGY; - while (*mode) { - if (*mode >= '0' && *mode <= '9') - state->level = *mode - '0'; - else - switch (*mode) { - case 'r': - state->mode = GZ_READ; - break; -#ifndef NO_GZCOMPRESS - case 'w': - state->mode = GZ_WRITE; - break; - case 'a': - state->mode = GZ_APPEND; - break; -#endif - case '+': /* can't read and write at the same time */ - free(state); - return NULL; - case 'b': /* ignore -- will request binary anyway */ - break; - case 'f': - state->strategy = Z_FILTERED; - break; - case 'h': - state->strategy = Z_HUFFMAN_ONLY; - break; - case 'R': - state->strategy = Z_RLE; - break; - case 'F': - state->strategy = Z_FIXED; - default: /* could consider as an error, but just ignore */ - ; - } - mode++; - } - - /* must provide an "r", "w", or "a" */ - if (state->mode == GZ_NONE) { - free(state); - return NULL; - } - - /* save the path name for error messages */ - state->path = malloc(strlen(path) + 1); - if (state->path == NULL) { - free(state); - return NULL; - } - strcpy(state->path, path); - - /* open the file with the appropriate mode (or just use fd) */ - state->fd = fd != -1 ? fd : - open(path, -#ifdef O_LARGEFILE - O_LARGEFILE | -#endif -#ifdef O_BINARY - O_BINARY | -#endif - (state->mode == GZ_READ ? - O_RDONLY : - (O_WRONLY | O_CREAT | ( - state->mode == GZ_WRITE ? - O_TRUNC : - O_APPEND))), - 0666); - if (state->fd == -1) { - free(state->path); - free(state); - return NULL; - } - if (state->mode == GZ_APPEND) - state->mode = GZ_WRITE; /* simplify later checks */ - - /* save the current position for rewinding (only if reading) */ - if (state->mode == GZ_READ) { - state->start = LSEEK(state->fd, 0, SEEK_CUR); - if (state->start == -1) state->start = 0; - } - - /* initialize stream */ - gz_reset(state); - - /* return stream */ - return (gzFile)state; -} - -/* -- see zlib.h -- */ -gzFile ZEXPORT gzopen(path, mode) - const char *path; - const char *mode; -{ - return gz_open(path, -1, mode); -} - -/* -- see zlib.h -- */ -gzFile ZEXPORT gzopen64(path, mode) - const char *path; - const char *mode; -{ - return gz_open(path, -1, mode); -} - -/* -- see zlib.h -- */ -gzFile ZEXPORT gzdopen(fd, mode) - int fd; - const char *mode; -{ - char *path; /* identifier for error messages */ - gzFile gz; - - if (fd == -1 || (path = malloc(7 + 3 * sizeof(int))) == NULL) - return NULL; - sprintf(path, "", fd); /* for debugging */ - gz = gz_open(path, fd, mode); - free(path); - return gz; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzbuffer(file, size) - gzFile file; - unsigned size; -{ - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return -1; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return -1; - - /* make sure we haven't already allocated memory */ - if (state->size != 0) - return -1; - - /* check and set requested size */ - if (size == 0) - return -1; - state->want = size; - return 0; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzrewind(file) - gzFile file; -{ - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - - /* check that we're reading and that there's no error */ - if (state->mode != GZ_READ || state->err != Z_OK) - return -1; - - /* back up and start over */ - if (LSEEK(state->fd, state->start, SEEK_SET) == -1) - return -1; - gz_reset(state); - return 0; -} - -/* -- see zlib.h -- */ -z_off64_t ZEXPORT gzseek64(file, offset, whence) - gzFile file; - z_off64_t offset; - int whence; -{ - unsigned n; - z_off64_t ret; - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return -1; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return -1; - - /* check that there's no error */ - if (state->err != Z_OK) - return -1; - - /* can only seek from start or relative to current position */ - if (whence != SEEK_SET && whence != SEEK_CUR) - return -1; - - /* normalize offset to a SEEK_CUR specification */ - if (whence == SEEK_SET) - offset -= state->pos; - else if (state->seek) - offset += state->skip; - state->seek = 0; - - /* if within raw area while reading, just go there */ - if (state->mode == GZ_READ && state->how == COPY && - state->pos + offset >= state->raw) { - ret = LSEEK(state->fd, offset - state->have, SEEK_CUR); - if (ret == -1) - return -1; - state->have = 0; - state->eof = 0; - state->seek = 0; - gz_error(state, Z_OK, NULL); - state->strm.avail_in = 0; - state->pos += offset; - return state->pos; - } - - /* calculate skip amount, rewinding if needed for back seek when reading */ - if (offset < 0) { - if (state->mode != GZ_READ) /* writing -- can't go backwards */ - return -1; - offset += state->pos; - if (offset < 0) /* before start of file! */ - return -1; - if (gzrewind(file) == -1) /* rewind, then skip to offset */ - return -1; - } - - /* if reading, skip what's in output buffer (one less gzgetc() check) */ - if (state->mode == GZ_READ) { - n = GT_OFF(state->have) || (z_off64_t)state->have > offset ? - (unsigned)offset : state->have; - state->have -= n; - state->next += n; - state->pos += n; - offset -= n; - } - - /* request skip (if not zero) */ - if (offset) { - state->seek = 1; - state->skip = offset; - } - return state->pos + offset; -} - -/* -- see zlib.h -- */ -z_off_t ZEXPORT gzseek(file, offset, whence) - gzFile file; - z_off_t offset; - int whence; -{ - z_off64_t ret; - - ret = gzseek64(file, (z_off64_t)offset, whence); - return ret == (z_off_t)ret ? (z_off_t)ret : -1; -} - -/* -- see zlib.h -- */ -z_off64_t ZEXPORT gztell64(file) - gzFile file; -{ - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return -1; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return -1; - - /* return position */ - return state->pos + (state->seek ? state->skip : 0); -} - -/* -- see zlib.h -- */ -z_off_t ZEXPORT gztell(file) - gzFile file; -{ - z_off64_t ret; - - ret = gztell64(file); - return ret == (z_off_t)ret ? (z_off_t)ret : -1; -} - -/* -- see zlib.h -- */ -z_off64_t ZEXPORT gzoffset64(file) - gzFile file; -{ - z_off64_t offset; - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return -1; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return -1; - - /* compute and return effective offset in file */ - offset = LSEEK(state->fd, 0, SEEK_CUR); - if (offset == -1) - return -1; - if (state->mode == GZ_READ) /* reading */ - offset -= state->strm.avail_in; /* don't count buffered input */ - return offset; -} - -/* -- see zlib.h -- */ -z_off_t ZEXPORT gzoffset(file) - gzFile file; -{ - z_off64_t ret; - - ret = gzoffset64(file); - return ret == (z_off_t)ret ? (z_off_t)ret : -1; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzeof(file) - gzFile file; -{ - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return 0; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return 0; - - /* return end-of-file state */ - return state->mode == GZ_READ ? - (state->eof && state->strm.avail_in == 0 && state->have == 0) : 0; -} - -/* -- see zlib.h -- */ -const char * ZEXPORT gzerror(file, errnum) - gzFile file; - int *errnum; -{ - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return NULL; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return NULL; - - /* return error information */ - if (errnum != NULL) - *errnum = state->err; - return state->msg == NULL ? "" : state->msg; -} - -/* -- see zlib.h -- */ -void ZEXPORT gzclearerr(file) - gzFile file; -{ - gz_statep state; - - /* get internal structure and check integrity */ - if (file == NULL) - return; - state = (gz_statep)file; - if (state->mode != GZ_READ && state->mode != GZ_WRITE) - return; - - /* clear error and end-of-file */ - if (state->mode == GZ_READ) - state->eof = 0; - gz_error(state, Z_OK, NULL); -} - -/* Create an error message in allocated memory and set state->err and - state->msg accordingly. Free any previous error message already there. Do - not try to free or allocate space if the error is Z_MEM_ERROR (out of - memory). Simply save the error message as a static string. If there is an - allocation failure constructing the error message, then convert the error to - out of memory. */ -void ZLIB_INTERNAL gz_error(state, err, msg) - gz_statep state; - int err; - const char *msg; -{ - /* free previously allocated message and clear */ - if (state->msg != NULL) { - if (state->err != Z_MEM_ERROR) - free(state->msg); - state->msg = NULL; - } - - /* set error code, and if no message, then done */ - state->err = err; - if (msg == NULL) - return; - - /* for an out of memory error, save as static string */ - if (err == Z_MEM_ERROR) { - state->msg = (char *)msg; - return; - } - - /* construct error message with path */ - if ((state->msg = malloc(strlen(state->path) + strlen(msg) + 3)) == NULL) { - state->err = Z_MEM_ERROR; - state->msg = (char *)"out of memory"; - return; - } - strcpy(state->msg, state->path); - strcat(state->msg, ": "); - strcat(state->msg, msg); - return; -} - -#ifndef INT_MAX -/* portably return maximum value for an int (when limits.h presumed not - available) -- we need to do this to cover cases where 2's complement not - used, since C standard permits 1's complement and sign-bit representations, - otherwise we could just use ((unsigned)-1) >> 1 */ -unsigned ZLIB_INTERNAL gz_intmax() -{ - unsigned p, q; - - p = 1; - do { - q = p; - p <<= 1; - p++; - } while (p > q); - return q >> 1; -} -#endif diff --git a/code/3rd_glpk/zlib/gzread.c b/code/3rd_glpk/zlib/gzread.c deleted file mode 100644 index 548201ab..00000000 --- a/code/3rd_glpk/zlib/gzread.c +++ /dev/null @@ -1,653 +0,0 @@ -/* gzread.c -- zlib functions for reading gzip files - * Copyright (C) 2004, 2005, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -#include "gzguts.h" - -/* Local functions */ -local int gz_load OF((gz_statep, unsigned char *, unsigned, unsigned *)); -local int gz_avail OF((gz_statep)); -local int gz_next4 OF((gz_statep, unsigned long *)); -local int gz_head OF((gz_statep)); -local int gz_decomp OF((gz_statep)); -local int gz_make OF((gz_statep)); -local int gz_skip OF((gz_statep, z_off64_t)); - -/* Use read() to load a buffer -- return -1 on error, otherwise 0. Read from - state->fd, and update state->eof, state->err, and state->msg as appropriate. - This function needs to loop on read(), since read() is not guaranteed to - read the number of bytes requested, depending on the type of descriptor. */ -local int gz_load(state, buf, len, have) - gz_statep state; - unsigned char *buf; - unsigned len; - unsigned *have; -{ - int ret; - - *have = 0; - do { - ret = read(state->fd, buf + *have, len - *have); - if (ret <= 0) - break; - *have += ret; - } while (*have < len); - if (ret < 0) { - gz_error(state, Z_ERRNO, zstrerror()); - return -1; - } - if (ret == 0) - state->eof = 1; - return 0; -} - -/* Load up input buffer and set eof flag if last data loaded -- return -1 on - error, 0 otherwise. Note that the eof flag is set when the end of the input - file is reached, even though there may be unused data in the buffer. Once - that data has been used, no more attempts will be made to read the file. - gz_avail() assumes that strm->avail_in == 0. */ -local int gz_avail(state) - gz_statep state; -{ - z_streamp strm = &(state->strm); - - if (state->err != Z_OK) - return -1; - if (state->eof == 0) { - if (gz_load(state, state->in, state->size, - (unsigned *)&(strm->avail_in)) == -1) - return -1; - strm->next_in = state->in; - } - return 0; -} - -/* Get next byte from input, or -1 if end or error. */ -#define NEXT() ((strm->avail_in == 0 && gz_avail(state) == -1) ? -1 : \ - (strm->avail_in == 0 ? -1 : \ - (strm->avail_in--, *(strm->next_in)++))) - -/* Get a four-byte little-endian integer and return 0 on success and the value - in *ret. Otherwise -1 is returned and *ret is not modified. */ -local int gz_next4(state, ret) - gz_statep state; - unsigned long *ret; -{ - int ch; - unsigned long val; - z_streamp strm = &(state->strm); - - val = NEXT(); - val += (unsigned)NEXT() << 8; - val += (unsigned long)NEXT() << 16; - ch = NEXT(); - if (ch == -1) - return -1; - val += (unsigned long)ch << 24; - *ret = val; - return 0; -} - -/* Look for gzip header, set up for inflate or copy. state->have must be zero. - If this is the first time in, allocate required memory. state->how will be - left unchanged if there is no more input data available, will be set to COPY - if there is no gzip header and direct copying will be performed, or it will - be set to GZIP for decompression, and the gzip header will be skipped so - that the next available input data is the raw deflate stream. If direct - copying, then leftover input data from the input buffer will be copied to - the output buffer. In that case, all further file reads will be directly to - either the output buffer or a user buffer. If decompressing, the inflate - state and the check value will be initialized. gz_head() will return 0 on - success or -1 on failure. Failures may include read errors or gzip header - errors. */ -local int gz_head(state) - gz_statep state; -{ - z_streamp strm = &(state->strm); - int flags; - unsigned len; - - /* allocate read buffers and inflate memory */ - if (state->size == 0) { - /* allocate buffers */ - state->in = malloc(state->want); - state->out = malloc(state->want << 1); - if (state->in == NULL || state->out == NULL) { - if (state->out != NULL) - free(state->out); - if (state->in != NULL) - free(state->in); - gz_error(state, Z_MEM_ERROR, "out of memory"); - return -1; - } - state->size = state->want; - - /* allocate inflate memory */ - state->strm.zalloc = Z_NULL; - state->strm.zfree = Z_NULL; - state->strm.opaque = Z_NULL; - state->strm.avail_in = 0; - state->strm.next_in = Z_NULL; - if (inflateInit2(&(state->strm), -15) != Z_OK) { /* raw inflate */ - free(state->out); - free(state->in); - state->size = 0; - gz_error(state, Z_MEM_ERROR, "out of memory"); - return -1; - } - } - - /* get some data in the input buffer */ - if (strm->avail_in == 0) { - if (gz_avail(state) == -1) - return -1; - if (strm->avail_in == 0) - return 0; - } - - /* look for the gzip magic header bytes 31 and 139 */ - if (strm->next_in[0] == 31) { - strm->avail_in--; - strm->next_in++; - if (strm->avail_in == 0 && gz_avail(state) == -1) - return -1; - if (strm->avail_in && strm->next_in[0] == 139) { - /* we have a gzip header, woo hoo! */ - strm->avail_in--; - strm->next_in++; - - /* skip rest of header */ - if (NEXT() != 8) { /* compression method */ - gz_error(state, Z_DATA_ERROR, "unknown compression method"); - return -1; - } - flags = NEXT(); - if (flags & 0xe0) { /* reserved flag bits */ - gz_error(state, Z_DATA_ERROR, "unknown header flags set"); - return -1; - } - NEXT(); /* modification time */ - NEXT(); - NEXT(); - NEXT(); - NEXT(); /* extra flags */ - NEXT(); /* operating system */ - if (flags & 4) { /* extra field */ - len = (unsigned)NEXT(); - len += (unsigned)NEXT() << 8; - while (len--) - if (NEXT() < 0) - break; - } - if (flags & 8) /* file name */ - while (NEXT() > 0) - ; - if (flags & 16) /* comment */ - while (NEXT() > 0) - ; - if (flags & 2) { /* header crc */ - NEXT(); - NEXT(); - } - /* an unexpected end of file is not checked for here -- it will be - noticed on the first request for uncompressed data */ - - /* set up for decompression */ - inflateReset(strm); - strm->adler = crc32(0L, Z_NULL, 0); - state->how = GZIP; - state->direct = 0; - return 0; - } - else { - /* not a gzip file -- save first byte (31) and fall to raw i/o */ - state->out[0] = 31; - state->have = 1; - } - } - - /* doing raw i/o, save start of raw data for seeking, copy any leftover - input to output -- this assumes that the output buffer is larger than - the input buffer, which also assures space for gzungetc() */ - state->raw = state->pos; - state->next = state->out; - if (strm->avail_in) { - memcpy(state->next + state->have, strm->next_in, strm->avail_in); - state->have += strm->avail_in; - strm->avail_in = 0; - } - state->how = COPY; - state->direct = 1; - return 0; -} - -/* Decompress from input to the provided next_out and avail_out in the state. - If the end of the compressed data is reached, then verify the gzip trailer - check value and length (modulo 2^32). state->have and state->next are set - to point to the just decompressed data, and the crc is updated. If the - trailer is verified, state->how is reset to LOOK to look for the next gzip - stream or raw data, once state->have is depleted. Returns 0 on success, -1 - on failure. Failures may include invalid compressed data or a failed gzip - trailer verification. */ -local int gz_decomp(state) - gz_statep state; -{ - int ret; - unsigned had; - unsigned long crc, len; - z_streamp strm = &(state->strm); - - /* fill output buffer up to end of deflate stream */ - had = strm->avail_out; - do { - /* get more input for inflate() */ - if (strm->avail_in == 0 && gz_avail(state) == -1) - return -1; - if (strm->avail_in == 0) { - gz_error(state, Z_DATA_ERROR, "unexpected end of file"); - return -1; - } - - /* decompress and handle errors */ - ret = inflate(strm, Z_NO_FLUSH); - if (ret == Z_STREAM_ERROR || ret == Z_NEED_DICT) { - gz_error(state, Z_STREAM_ERROR, - "internal error: inflate stream corrupt"); - return -1; - } - if (ret == Z_MEM_ERROR) { - gz_error(state, Z_MEM_ERROR, "out of memory"); - return -1; - } - if (ret == Z_DATA_ERROR) { /* deflate stream invalid */ - gz_error(state, Z_DATA_ERROR, - strm->msg == NULL ? "compressed data error" : strm->msg); - return -1; - } - } while (strm->avail_out && ret != Z_STREAM_END); - - /* update available output and crc check value */ - state->have = had - strm->avail_out; - state->next = strm->next_out - state->have; - strm->adler = crc32(strm->adler, state->next, state->have); - - /* check gzip trailer if at end of deflate stream */ - if (ret == Z_STREAM_END) { - if (gz_next4(state, &crc) == -1 || gz_next4(state, &len) == -1) { - gz_error(state, Z_DATA_ERROR, "unexpected end of file"); - return -1; - } - if (crc != strm->adler) { - gz_error(state, Z_DATA_ERROR, "incorrect data check"); - return -1; - } - if (len != (strm->total_out & 0xffffffffL)) { - gz_error(state, Z_DATA_ERROR, "incorrect length check"); - return -1; - } - state->how = LOOK; /* ready for next stream, once have is 0 (leave - state->direct unchanged to remember how) */ - } - - /* good decompression */ - return 0; -} - -/* Make data and put in the output buffer. Assumes that state->have == 0. - Data is either copied from the input file or decompressed from the input - file depending on state->how. If state->how is LOOK, then a gzip header is - looked for (and skipped if found) to determine wither to copy or decompress. - Returns -1 on error, otherwise 0. gz_make() will leave state->have as COPY - or GZIP unless the end of the input file has been reached and all data has - been processed. */ -local int gz_make(state) - gz_statep state; -{ - z_streamp strm = &(state->strm); - - if (state->how == LOOK) { /* look for gzip header */ - if (gz_head(state) == -1) - return -1; - if (state->have) /* got some data from gz_head() */ - return 0; - } - if (state->how == COPY) { /* straight copy */ - if (gz_load(state, state->out, state->size << 1, &(state->have)) == -1) - return -1; - state->next = state->out; - } - else if (state->how == GZIP) { /* decompress */ - strm->avail_out = state->size << 1; - strm->next_out = state->out; - if (gz_decomp(state) == -1) - return -1; - } - return 0; -} - -/* Skip len uncompressed bytes of output. Return -1 on error, 0 on success. */ -local int gz_skip(state, len) - gz_statep state; - z_off64_t len; -{ - unsigned n; - - /* skip over len bytes or reach end-of-file, whichever comes first */ - while (len) - /* skip over whatever is in output buffer */ - if (state->have) { - n = GT_OFF(state->have) || (z_off64_t)state->have > len ? - (unsigned)len : state->have; - state->have -= n; - state->next += n; - state->pos += n; - len -= n; - } - - /* output buffer empty -- return if we're at the end of the input */ - else if (state->eof && state->strm.avail_in == 0) - break; - - /* need more data to skip -- load up output buffer */ - else { - /* get more output, looking for header if required */ - if (gz_make(state) == -1) - return -1; - } - return 0; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzread(file, buf, len) - gzFile file; - voidp buf; - unsigned len; -{ - unsigned got, n; - gz_statep state; - z_streamp strm; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - strm = &(state->strm); - - /* check that we're reading and that there's no error */ - if (state->mode != GZ_READ || state->err != Z_OK) - return -1; - - /* since an int is returned, make sure len fits in one, otherwise return - with an error (this avoids the flaw in the interface) */ - if ((int)len < 0) { - gz_error(state, Z_BUF_ERROR, "requested length does not fit in int"); - return -1; - } - - /* if len is zero, avoid unnecessary operations */ - if (len == 0) - return 0; - - /* process a skip request */ - if (state->seek) { - state->seek = 0; - if (gz_skip(state, state->skip) == -1) - return -1; - } - - /* get len bytes to buf, or less than len if at the end */ - got = 0; - do { - /* first just try copying data from the output buffer */ - if (state->have) { - n = state->have > len ? len : state->have; - memcpy(buf, state->next, n); - state->next += n; - state->have -= n; - } - - /* output buffer empty -- return if we're at the end of the input */ - else if (state->eof && strm->avail_in == 0) - break; - - /* need output data -- for small len or new stream load up our output - buffer */ - else if (state->how == LOOK || len < (state->size << 1)) { - /* get more output, looking for header if required */ - if (gz_make(state) == -1) - return -1; - continue; /* no progress yet -- go back to memcpy() above */ - /* the copy above assures that we will leave with space in the - output buffer, allowing at least one gzungetc() to succeed */ - } - - /* large len -- read directly into user buffer */ - else if (state->how == COPY) { /* read directly */ - if (gz_load(state, buf, len, &n) == -1) - return -1; - } - - /* large len -- decompress directly into user buffer */ - else { /* state->how == GZIP */ - strm->avail_out = len; - strm->next_out = buf; - if (gz_decomp(state) == -1) - return -1; - n = state->have; - state->have = 0; - } - - /* update progress */ - len -= n; - buf = (char *)buf + n; - got += n; - state->pos += n; - } while (len); - - /* return number of bytes read into user buffer (will fit in int) */ - return (int)got; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzgetc(file) - gzFile file; -{ - int ret; - unsigned char buf[1]; - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - - /* check that we're reading and that there's no error */ - if (state->mode != GZ_READ || state->err != Z_OK) - return -1; - - /* try output buffer (no need to check for skip request) */ - if (state->have) { - state->have--; - state->pos++; - return *(state->next)++; - } - - /* nothing there -- try gzread() */ - ret = gzread(file, buf, 1); - return ret < 1 ? -1 : buf[0]; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzungetc(c, file) - int c; - gzFile file; -{ - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - - /* check that we're reading and that there's no error */ - if (state->mode != GZ_READ || state->err != Z_OK) - return -1; - - /* process a skip request */ - if (state->seek) { - state->seek = 0; - if (gz_skip(state, state->skip) == -1) - return -1; - } - - /* can't push EOF */ - if (c < 0) - return -1; - - /* if output buffer empty, put byte at end (allows more pushing) */ - if (state->have == 0) { - state->have = 1; - state->next = state->out + (state->size << 1) - 1; - state->next[0] = c; - state->pos--; - return c; - } - - /* if no room, give up (must have already done a gzungetc()) */ - if (state->have == (state->size << 1)) { - gz_error(state, Z_BUF_ERROR, "out of room to push characters"); - return -1; - } - - /* slide output data if needed and insert byte before existing data */ - if (state->next == state->out) { - unsigned char *src = state->out + state->have; - unsigned char *dest = state->out + (state->size << 1); - while (src > state->out) - *--dest = *--src; - state->next = dest; - } - state->have++; - state->next--; - state->next[0] = c; - state->pos--; - return c; -} - -/* -- see zlib.h -- */ -char * ZEXPORT gzgets(file, buf, len) - gzFile file; - char *buf; - int len; -{ - unsigned left, n; - char *str; - unsigned char *eol; - gz_statep state; - - /* check parameters and get internal structure */ - if (file == NULL || buf == NULL || len < 1) - return NULL; - state = (gz_statep)file; - - /* check that we're reading and that there's no error */ - if (state->mode != GZ_READ || state->err != Z_OK) - return NULL; - - /* process a skip request */ - if (state->seek) { - state->seek = 0; - if (gz_skip(state, state->skip) == -1) - return NULL; - } - - /* copy output bytes up to new line or len - 1, whichever comes first -- - append a terminating zero to the string (we don't check for a zero in - the contents, let the user worry about that) */ - str = buf; - left = (unsigned)len - 1; - if (left) do { - /* assure that something is in the output buffer */ - if (state->have == 0) { - if (gz_make(state) == -1) - return NULL; /* error */ - if (state->have == 0) { /* end of file */ - if (buf == str) /* got bupkus */ - return NULL; - break; /* got something -- return it */ - } - } - - /* look for end-of-line in current output buffer */ - n = state->have > left ? left : state->have; - eol = memchr(state->next, '\n', n); - if (eol != NULL) - n = (unsigned)(eol - state->next) + 1; - - /* copy through end-of-line, or remainder if not found */ - memcpy(buf, state->next, n); - state->have -= n; - state->next += n; - state->pos += n; - left -= n; - buf += n; - } while (left && eol == NULL); - - /* found end-of-line or out of space -- terminate string and return it */ - buf[0] = 0; - return str; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzdirect(file) - gzFile file; -{ - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return 0; - state = (gz_statep)file; - - /* check that we're reading */ - if (state->mode != GZ_READ) - return 0; - - /* if the state is not known, but we can find out, then do so (this is - mainly for right after a gzopen() or gzdopen()) */ - if (state->how == LOOK && state->have == 0) - (void)gz_head(state); - - /* return 1 if reading direct, 0 if decompressing a gzip stream */ - return state->direct; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzclose_r(file) - gzFile file; -{ - int ret; - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return Z_STREAM_ERROR; - state = (gz_statep)file; - - /* check that we're reading */ - if (state->mode != GZ_READ) - return Z_STREAM_ERROR; - - /* free memory and close file */ - if (state->size) { - inflateEnd(&(state->strm)); - free(state->out); - free(state->in); - } - gz_error(state, Z_OK, NULL); - free(state->path); - ret = close(state->fd); - free(state); - return ret ? Z_ERRNO : Z_OK; -} diff --git a/code/3rd_glpk/zlib/gzwrite.c b/code/3rd_glpk/zlib/gzwrite.c deleted file mode 100644 index 13c5558e..00000000 --- a/code/3rd_glpk/zlib/gzwrite.c +++ /dev/null @@ -1,531 +0,0 @@ -/* gzwrite.c -- zlib functions for writing gzip files - * Copyright (C) 2004, 2005, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -#include "gzguts.h" - -/* Local functions */ -local int gz_init OF((gz_statep)); -local int gz_comp OF((gz_statep, int)); -local int gz_zero OF((gz_statep, z_off64_t)); - -/* Initialize state for writing a gzip file. Mark initialization by setting - state->size to non-zero. Return -1 on failure or 0 on success. */ -local int gz_init(state) - gz_statep state; -{ - int ret; - z_streamp strm = &(state->strm); - - /* allocate input and output buffers */ - state->in = malloc(state->want); - state->out = malloc(state->want); - if (state->in == NULL || state->out == NULL) { - if (state->out != NULL) - free(state->out); - if (state->in != NULL) - free(state->in); - gz_error(state, Z_MEM_ERROR, "out of memory"); - return -1; - } - - /* allocate deflate memory, set up for gzip compression */ - strm->zalloc = Z_NULL; - strm->zfree = Z_NULL; - strm->opaque = Z_NULL; - ret = deflateInit2(strm, state->level, Z_DEFLATED, - 15 + 16, 8, state->strategy); - if (ret != Z_OK) { - free(state->in); - gz_error(state, Z_MEM_ERROR, "out of memory"); - return -1; - } - - /* mark state as initialized */ - state->size = state->want; - - /* initialize write buffer */ - strm->avail_out = state->size; - strm->next_out = state->out; - state->next = strm->next_out; - return 0; -} - -/* Compress whatever is at avail_in and next_in and write to the output file. - Return -1 if there is an error writing to the output file, otherwise 0. - flush is assumed to be a valid deflate() flush value. If flush is Z_FINISH, - then the deflate() state is reset to start a new gzip stream. */ -local int gz_comp(state, flush) - gz_statep state; - int flush; -{ - int ret, got; - unsigned have; - z_streamp strm = &(state->strm); - - /* allocate memory if this is the first time through */ - if (state->size == 0 && gz_init(state) == -1) - return -1; - - /* run deflate() on provided input until it produces no more output */ - ret = Z_OK; - do { - /* write out current buffer contents if full, or if flushing, but if - doing Z_FINISH then don't write until we get to Z_STREAM_END */ - if (strm->avail_out == 0 || (flush != Z_NO_FLUSH && - (flush != Z_FINISH || ret == Z_STREAM_END))) { - have = (unsigned)(strm->next_out - state->next); - if (have && ((got = write(state->fd, state->next, have)) < 0 || - (unsigned)got != have)) { - gz_error(state, Z_ERRNO, zstrerror()); - return -1; - } - if (strm->avail_out == 0) { - strm->avail_out = state->size; - strm->next_out = state->out; - } - state->next = strm->next_out; - } - - /* compress */ - have = strm->avail_out; - ret = deflate(strm, flush); - if (ret == Z_STREAM_ERROR) { - gz_error(state, Z_STREAM_ERROR, - "internal error: deflate stream corrupt"); - return -1; - } - have -= strm->avail_out; - } while (have); - - /* if that completed a deflate stream, allow another to start */ - if (flush == Z_FINISH) - deflateReset(strm); - - /* all done, no errors */ - return 0; -} - -/* Compress len zeros to output. Return -1 on error, 0 on success. */ -local int gz_zero(state, len) - gz_statep state; - z_off64_t len; -{ - int first; - unsigned n; - z_streamp strm = &(state->strm); - - /* consume whatever's left in the input buffer */ - if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) - return -1; - - /* compress len zeros (len guaranteed > 0) */ - first = 1; - while (len) { - n = GT_OFF(state->size) || (z_off64_t)state->size > len ? - (unsigned)len : state->size; - if (first) { - memset(state->in, 0, n); - first = 0; - } - strm->avail_in = n; - strm->next_in = state->in; - state->pos += n; - if (gz_comp(state, Z_NO_FLUSH) == -1) - return -1; - len -= n; - } - return 0; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzwrite(file, buf, len) - gzFile file; - voidpc buf; - unsigned len; -{ - unsigned put = len; - unsigned n; - gz_statep state; - z_streamp strm; - - /* get internal structure */ - if (file == NULL) - return 0; - state = (gz_statep)file; - strm = &(state->strm); - - /* check that we're writing and that there's no error */ - if (state->mode != GZ_WRITE || state->err != Z_OK) - return 0; - - /* since an int is returned, make sure len fits in one, otherwise return - with an error (this avoids the flaw in the interface) */ - if ((int)len < 0) { - gz_error(state, Z_BUF_ERROR, "requested length does not fit in int"); - return 0; - } - - /* if len is zero, avoid unnecessary operations */ - if (len == 0) - return 0; - - /* allocate memory if this is the first time through */ - if (state->size == 0 && gz_init(state) == -1) - return 0; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - if (gz_zero(state, state->skip) == -1) - return 0; - } - - /* for small len, copy to input buffer, otherwise compress directly */ - if (len < state->size) { - /* copy to input buffer, compress when full */ - do { - if (strm->avail_in == 0) - strm->next_in = state->in; - n = state->size - strm->avail_in; - if (n > len) - n = len; - memcpy(strm->next_in + strm->avail_in, buf, n); - strm->avail_in += n; - state->pos += n; - buf = (char *)buf + n; - len -= n; - if (len && gz_comp(state, Z_NO_FLUSH) == -1) - return 0; - } while (len); - } - else { - /* consume whatever's left in the input buffer */ - if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) - return 0; - - /* directly compress user buffer to file */ - strm->avail_in = len; - strm->next_in = (voidp)buf; - state->pos += len; - if (gz_comp(state, Z_NO_FLUSH) == -1) - return 0; - } - - /* input was all buffered or compressed (put will fit in int) */ - return (int)put; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzputc(file, c) - gzFile file; - int c; -{ - unsigned char buf[1]; - gz_statep state; - z_streamp strm; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - strm = &(state->strm); - - /* check that we're writing and that there's no error */ - if (state->mode != GZ_WRITE || state->err != Z_OK) - return -1; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - if (gz_zero(state, state->skip) == -1) - return -1; - } - - /* try writing to input buffer for speed (state->size == 0 if buffer not - initialized) */ - if (strm->avail_in < state->size) { - if (strm->avail_in == 0) - strm->next_in = state->in; - strm->next_in[strm->avail_in++] = c; - state->pos++; - return c; - } - - /* no room in buffer or not initialized, use gz_write() */ - buf[0] = c; - if (gzwrite(file, buf, 1) != 1) - return -1; - return c; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzputs(file, str) - gzFile file; - const char *str; -{ - int ret; - unsigned len; - - /* write string */ - len = (unsigned)strlen(str); - ret = gzwrite(file, str, len); - return ret == 0 && len != 0 ? -1 : ret; -} - -#ifdef STDC -#include - -/* -- see zlib.h -- */ -int ZEXPORTVA gzprintf (gzFile file, const char *format, ...) -{ - int size, len; - gz_statep state; - z_streamp strm; - va_list va; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - strm = &(state->strm); - - /* check that we're writing and that there's no error */ - if (state->mode != GZ_WRITE || state->err != Z_OK) - return 0; - - /* make sure we have some buffer space */ - if (state->size == 0 && gz_init(state) == -1) - return 0; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - if (gz_zero(state, state->skip) == -1) - return 0; - } - - /* consume whatever's left in the input buffer */ - if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) - return 0; - - /* do the printf() into the input buffer, put length in len */ - size = (int)(state->size); - state->in[size - 1] = 0; - va_start(va, format); -#ifdef NO_vsnprintf -# ifdef HAS_vsprintf_void - (void)vsprintf(state->in, format, va); - va_end(va); - for (len = 0; len < size; len++) - if (state->in[len] == 0) break; -# else - len = vsprintf((char *)state->in, format, va); - va_end(va); -# endif -#else -# ifdef HAS_vsnprintf_void - (void)vsnprintf(state->in, size, format, va); - va_end(va); - len = strlen(state->in); -# else - len = vsnprintf((char *)(state->in), size, format, va); - va_end(va); -# endif -#endif - - /* check that printf() results fit in buffer */ - if (len <= 0 || len >= (int)size || state->in[size - 1] != 0) - return 0; - - /* update buffer and position, defer compression until needed */ - strm->avail_in = (unsigned)len; - strm->next_in = state->in; - state->pos += len; - return len; -} - -#else /* !STDC */ - -/* -- see zlib.h -- */ -int ZEXPORTVA gzprintf (file, format, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, - a11, a12, a13, a14, a15, a16, a17, a18, a19, a20) - gzFile file; - const char *format; - int a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, - a11, a12, a13, a14, a15, a16, a17, a18, a19, a20; -{ - int size, len; - gz_statep state; - z_streamp strm; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - strm = &(state->strm); - - /* check that we're writing and that there's no error */ - if (state->mode != GZ_WRITE || state->err != Z_OK) - return 0; - - /* make sure we have some buffer space */ - if (state->size == 0 && gz_init(state) == -1) - return 0; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - if (gz_zero(state, state->skip) == -1) - return 0; - } - - /* consume whatever's left in the input buffer */ - if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) - return 0; - - /* do the printf() into the input buffer, put length in len */ - size = (int)(state->size); - state->in[size - 1] = 0; -#ifdef NO_snprintf -# ifdef HAS_sprintf_void - sprintf(state->in, format, a1, a2, a3, a4, a5, a6, a7, a8, - a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); - for (len = 0; len < size; len++) - if (state->in[len] == 0) break; -# else - len = sprintf(state->in, format, a1, a2, a3, a4, a5, a6, a7, a8, - a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); -# endif -#else -# ifdef HAS_snprintf_void - snprintf(state->in, size, format, a1, a2, a3, a4, a5, a6, a7, a8, - a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); - len = strlen(state->in); -# else - len = snprintf(state->in, size, format, a1, a2, a3, a4, a5, a6, a7, a8, - a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); -# endif -#endif - - /* check that printf() results fit in buffer */ - if (len <= 0 || len >= (int)size || state->in[size - 1] != 0) - return 0; - - /* update buffer and position, defer compression until needed */ - strm->avail_in = (unsigned)len; - strm->next_in = state->in; - state->pos += len; - return len; -} - -#endif - -/* -- see zlib.h -- */ -int ZEXPORT gzflush(file, flush) - gzFile file; - int flush; -{ - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return -1; - state = (gz_statep)file; - - /* check that we're writing and that there's no error */ - if (state->mode != GZ_WRITE || state->err != Z_OK) - return Z_STREAM_ERROR; - - /* check flush parameter */ - if (flush < 0 || flush > Z_FINISH) - return Z_STREAM_ERROR; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - if (gz_zero(state, state->skip) == -1) - return -1; - } - - /* compress remaining data with requested flush */ - gz_comp(state, flush); - return state->err; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzsetparams(file, level, strategy) - gzFile file; - int level; - int strategy; -{ - gz_statep state; - z_streamp strm; - - /* get internal structure */ - if (file == NULL) - return Z_STREAM_ERROR; - state = (gz_statep)file; - strm = &(state->strm); - - /* check that we're writing and that there's no error */ - if (state->mode != GZ_WRITE || state->err != Z_OK) - return Z_STREAM_ERROR; - - /* if no change is requested, then do nothing */ - if (level == state->level && strategy == state->strategy) - return Z_OK; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - if (gz_zero(state, state->skip) == -1) - return -1; - } - - /* change compression parameters for subsequent input */ - if (state->size) { - /* flush previous input with previous parameters before changing */ - if (strm->avail_in && gz_comp(state, Z_PARTIAL_FLUSH) == -1) - return state->err; - deflateParams(strm, level, strategy); - } - state->level = level; - state->strategy = strategy; - return Z_OK; -} - -/* -- see zlib.h -- */ -int ZEXPORT gzclose_w(file) - gzFile file; -{ - int ret = 0; - gz_statep state; - - /* get internal structure */ - if (file == NULL) - return Z_STREAM_ERROR; - state = (gz_statep)file; - - /* check that we're writing */ - if (state->mode != GZ_WRITE) - return Z_STREAM_ERROR; - - /* check for seek request */ - if (state->seek) { - state->seek = 0; - ret += gz_zero(state, state->skip); - } - - /* flush, free memory, and close file */ - ret += gz_comp(state, Z_FINISH); - (void)deflateEnd(&(state->strm)); - free(state->out); - free(state->in); - gz_error(state, Z_OK, NULL); - free(state->path); - ret += close(state->fd); - free(state); - return ret ? Z_ERRNO : Z_OK; -} diff --git a/code/3rd_glpk/zlib/inffast.c b/code/3rd_glpk/zlib/inffast.c deleted file mode 100644 index 2f1d60b4..00000000 --- a/code/3rd_glpk/zlib/inffast.c +++ /dev/null @@ -1,340 +0,0 @@ -/* inffast.c -- fast decoding - * Copyright (C) 1995-2008, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -#include "zutil.h" -#include "inftrees.h" -#include "inflate.h" -#include "inffast.h" - -#ifndef ASMINF - -/* Allow machine dependent optimization for post-increment or pre-increment. - Based on testing to date, - Pre-increment preferred for: - - PowerPC G3 (Adler) - - MIPS R5000 (Randers-Pehrson) - Post-increment preferred for: - - none - No measurable difference: - - Pentium III (Anderson) - - M68060 (Nikl) - */ -#ifdef POSTINC -# define OFF 0 -# define PUP(a) *(a)++ -#else -# define OFF 1 -# define PUP(a) *++(a) -#endif - -/* - Decode literal, length, and distance codes and write out the resulting - literal and match bytes until either not enough input or output is - available, an end-of-block is encountered, or a data error is encountered. - When large enough input and output buffers are supplied to inflate(), for - example, a 16K input buffer and a 64K output buffer, more than 95% of the - inflate execution time is spent in this routine. - - Entry assumptions: - - state->mode == LEN - strm->avail_in >= 6 - strm->avail_out >= 258 - start >= strm->avail_out - state->bits < 8 - - On return, state->mode is one of: - - LEN -- ran out of enough output space or enough available input - TYPE -- reached end of block code, inflate() to interpret next block - BAD -- error in block data - - Notes: - - - The maximum input bits used by a length/distance pair is 15 bits for the - length code, 5 bits for the length extra, 15 bits for the distance code, - and 13 bits for the distance extra. This totals 48 bits, or six bytes. - Therefore if strm->avail_in >= 6, then there is enough input to avoid - checking for available input while decoding. - - - The maximum bytes that a single length/distance pair can output is 258 - bytes, which is the maximum length that can be coded. inflate_fast() - requires strm->avail_out >= 258 for each loop to avoid checking for - output space. - */ -void ZLIB_INTERNAL inflate_fast(strm, start) -z_streamp strm; -unsigned start; /* inflate()'s starting value for strm->avail_out */ -{ - struct inflate_state FAR *state; - unsigned char FAR *in; /* local strm->next_in */ - unsigned char FAR *last; /* while in < last, enough input available */ - unsigned char FAR *out; /* local strm->next_out */ - unsigned char FAR *beg; /* inflate()'s initial strm->next_out */ - unsigned char FAR *end; /* while out < end, enough space available */ -#ifdef INFLATE_STRICT - unsigned dmax; /* maximum distance from zlib header */ -#endif - unsigned wsize; /* window size or zero if not using window */ - unsigned whave; /* valid bytes in the window */ - unsigned wnext; /* window write index */ - unsigned char FAR *window; /* allocated sliding window, if wsize != 0 */ - unsigned long hold; /* local strm->hold */ - unsigned bits; /* local strm->bits */ - code const FAR *lcode; /* local strm->lencode */ - code const FAR *dcode; /* local strm->distcode */ - unsigned lmask; /* mask for first level of length codes */ - unsigned dmask; /* mask for first level of distance codes */ - code here; /* retrieved table entry */ - unsigned op; /* code bits, operation, extra bits, or */ - /* window position, window bytes to copy */ - unsigned len; /* match length, unused bytes */ - unsigned dist; /* match distance */ - unsigned char FAR *from; /* where to copy match from */ - - /* copy state to local variables */ - state = (struct inflate_state FAR *)strm->state; - in = strm->next_in - OFF; - last = in + (strm->avail_in - 5); - out = strm->next_out - OFF; - beg = out - (start - strm->avail_out); - end = out + (strm->avail_out - 257); -#ifdef INFLATE_STRICT - dmax = state->dmax; -#endif - wsize = state->wsize; - whave = state->whave; - wnext = state->wnext; - window = state->window; - hold = state->hold; - bits = state->bits; - lcode = state->lencode; - dcode = state->distcode; - lmask = (1U << state->lenbits) - 1; - dmask = (1U << state->distbits) - 1; - - /* decode literals and length/distances until end-of-block or not enough - input data or output space */ - do { - if (bits < 15) { - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - } - here = lcode[hold & lmask]; - dolen: - op = (unsigned)(here.bits); - hold >>= op; - bits -= op; - op = (unsigned)(here.op); - if (op == 0) { /* literal */ - Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ? - "inflate: literal '%c'\n" : - "inflate: literal 0x%02x\n", here.val)); - PUP(out) = (unsigned char)(here.val); - } - else if (op & 16) { /* length base */ - len = (unsigned)(here.val); - op &= 15; /* number of extra bits */ - if (op) { - if (bits < op) { - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - } - len += (unsigned)hold & ((1U << op) - 1); - hold >>= op; - bits -= op; - } - Tracevv((stderr, "inflate: length %u\n", len)); - if (bits < 15) { - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - } - here = dcode[hold & dmask]; - dodist: - op = (unsigned)(here.bits); - hold >>= op; - bits -= op; - op = (unsigned)(here.op); - if (op & 16) { /* distance base */ - dist = (unsigned)(here.val); - op &= 15; /* number of extra bits */ - if (bits < op) { - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - if (bits < op) { - hold += (unsigned long)(PUP(in)) << bits; - bits += 8; - } - } - dist += (unsigned)hold & ((1U << op) - 1); -#ifdef INFLATE_STRICT - if (dist > dmax) { - strm->msg = (char *)"invalid distance too far back"; - state->mode = BAD; - break; - } -#endif - hold >>= op; - bits -= op; - Tracevv((stderr, "inflate: distance %u\n", dist)); - op = (unsigned)(out - beg); /* max distance in output */ - if (dist > op) { /* see if copy from window */ - op = dist - op; /* distance back in window */ - if (op > whave) { - if (state->sane) { - strm->msg = - (char *)"invalid distance too far back"; - state->mode = BAD; - break; - } -#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR - if (len <= op - whave) { - do { - PUP(out) = 0; - } while (--len); - continue; - } - len -= op - whave; - do { - PUP(out) = 0; - } while (--op > whave); - if (op == 0) { - from = out - dist; - do { - PUP(out) = PUP(from); - } while (--len); - continue; - } -#endif - } - from = window - OFF; - if (wnext == 0) { /* very common case */ - from += wsize - op; - if (op < len) { /* some from window */ - len -= op; - do { - PUP(out) = PUP(from); - } while (--op); - from = out - dist; /* rest from output */ - } - } - else if (wnext < op) { /* wrap around window */ - from += wsize + wnext - op; - op -= wnext; - if (op < len) { /* some from end of window */ - len -= op; - do { - PUP(out) = PUP(from); - } while (--op); - from = window - OFF; - if (wnext < len) { /* some from start of window */ - op = wnext; - len -= op; - do { - PUP(out) = PUP(from); - } while (--op); - from = out - dist; /* rest from output */ - } - } - } - else { /* contiguous in window */ - from += wnext - op; - if (op < len) { /* some from window */ - len -= op; - do { - PUP(out) = PUP(from); - } while (--op); - from = out - dist; /* rest from output */ - } - } - while (len > 2) { - PUP(out) = PUP(from); - PUP(out) = PUP(from); - PUP(out) = PUP(from); - len -= 3; - } - if (len) { - PUP(out) = PUP(from); - if (len > 1) - PUP(out) = PUP(from); - } - } - else { - from = out - dist; /* copy direct from output */ - do { /* minimum length is three */ - PUP(out) = PUP(from); - PUP(out) = PUP(from); - PUP(out) = PUP(from); - len -= 3; - } while (len > 2); - if (len) { - PUP(out) = PUP(from); - if (len > 1) - PUP(out) = PUP(from); - } - } - } - else if ((op & 64) == 0) { /* 2nd level distance code */ - here = dcode[here.val + (hold & ((1U << op) - 1))]; - goto dodist; - } - else { - strm->msg = (char *)"invalid distance code"; - state->mode = BAD; - break; - } - } - else if ((op & 64) == 0) { /* 2nd level length code */ - here = lcode[here.val + (hold & ((1U << op) - 1))]; - goto dolen; - } - else if (op & 32) { /* end-of-block */ - Tracevv((stderr, "inflate: end of block\n")); - state->mode = TYPE; - break; - } - else { - strm->msg = (char *)"invalid literal/length code"; - state->mode = BAD; - break; - } - } while (in < last && out < end); - - /* return unused bytes (on entry, bits < 8, so in won't go too far back) */ - len = bits >> 3; - in -= len; - bits -= len << 3; - hold &= (1U << bits) - 1; - - /* update state and return */ - strm->next_in = in + OFF; - strm->next_out = out + OFF; - strm->avail_in = (unsigned)(in < last ? 5 + (last - in) : 5 - (in - last)); - strm->avail_out = (unsigned)(out < end ? - 257 + (end - out) : 257 - (out - end)); - state->hold = hold; - state->bits = bits; - return; -} - -/* - inflate_fast() speedups that turned out slower (on a PowerPC G3 750CXe): - - Using bit fields for code structure - - Different op definition to avoid & for extra bits (do & for table bits) - - Three separate decoding do-loops for direct, window, and wnext == 0 - - Special case for distance > 1 copies to do overlapped load and store copy - - Explicit branch predictions (based on measured branch probabilities) - - Deferring match copy and interspersed it with decoding subsequent codes - - Swapping literal/length else - - Swapping window/direct else - - Larger unrolled copy loops (three is about right) - - Moving len -= 3 statement into middle of loop - */ - -#endif /* !ASMINF */ diff --git a/code/3rd_glpk/zlib/inffast.h b/code/3rd_glpk/zlib/inffast.h deleted file mode 100644 index e5c1aa4c..00000000 --- a/code/3rd_glpk/zlib/inffast.h +++ /dev/null @@ -1,11 +0,0 @@ -/* inffast.h -- header to use inffast.c - * Copyright (C) 1995-2003, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. - */ - -void ZLIB_INTERNAL inflate_fast OF((z_streamp strm, unsigned start)); diff --git a/code/3rd_glpk/zlib/inffixed.h b/code/3rd_glpk/zlib/inffixed.h deleted file mode 100644 index 75ed4b59..00000000 --- a/code/3rd_glpk/zlib/inffixed.h +++ /dev/null @@ -1,94 +0,0 @@ - /* inffixed.h -- table for decoding fixed codes - * Generated automatically by makefixed(). - */ - - /* WARNING: this file should *not* be used by applications. It - is part of the implementation of the compression library and - is subject to change. Applications should only use zlib.h. - */ - - static const code lenfix[512] = { - {96,7,0},{0,8,80},{0,8,16},{20,8,115},{18,7,31},{0,8,112},{0,8,48}, - {0,9,192},{16,7,10},{0,8,96},{0,8,32},{0,9,160},{0,8,0},{0,8,128}, - {0,8,64},{0,9,224},{16,7,6},{0,8,88},{0,8,24},{0,9,144},{19,7,59}, - {0,8,120},{0,8,56},{0,9,208},{17,7,17},{0,8,104},{0,8,40},{0,9,176}, - {0,8,8},{0,8,136},{0,8,72},{0,9,240},{16,7,4},{0,8,84},{0,8,20}, - {21,8,227},{19,7,43},{0,8,116},{0,8,52},{0,9,200},{17,7,13},{0,8,100}, - {0,8,36},{0,9,168},{0,8,4},{0,8,132},{0,8,68},{0,9,232},{16,7,8}, - {0,8,92},{0,8,28},{0,9,152},{20,7,83},{0,8,124},{0,8,60},{0,9,216}, - {18,7,23},{0,8,108},{0,8,44},{0,9,184},{0,8,12},{0,8,140},{0,8,76}, - {0,9,248},{16,7,3},{0,8,82},{0,8,18},{21,8,163},{19,7,35},{0,8,114}, - {0,8,50},{0,9,196},{17,7,11},{0,8,98},{0,8,34},{0,9,164},{0,8,2}, - {0,8,130},{0,8,66},{0,9,228},{16,7,7},{0,8,90},{0,8,26},{0,9,148}, - {20,7,67},{0,8,122},{0,8,58},{0,9,212},{18,7,19},{0,8,106},{0,8,42}, - {0,9,180},{0,8,10},{0,8,138},{0,8,74},{0,9,244},{16,7,5},{0,8,86}, - {0,8,22},{64,8,0},{19,7,51},{0,8,118},{0,8,54},{0,9,204},{17,7,15}, - {0,8,102},{0,8,38},{0,9,172},{0,8,6},{0,8,134},{0,8,70},{0,9,236}, - {16,7,9},{0,8,94},{0,8,30},{0,9,156},{20,7,99},{0,8,126},{0,8,62}, - {0,9,220},{18,7,27},{0,8,110},{0,8,46},{0,9,188},{0,8,14},{0,8,142}, - {0,8,78},{0,9,252},{96,7,0},{0,8,81},{0,8,17},{21,8,131},{18,7,31}, - {0,8,113},{0,8,49},{0,9,194},{16,7,10},{0,8,97},{0,8,33},{0,9,162}, - {0,8,1},{0,8,129},{0,8,65},{0,9,226},{16,7,6},{0,8,89},{0,8,25}, - {0,9,146},{19,7,59},{0,8,121},{0,8,57},{0,9,210},{17,7,17},{0,8,105}, - {0,8,41},{0,9,178},{0,8,9},{0,8,137},{0,8,73},{0,9,242},{16,7,4}, - {0,8,85},{0,8,21},{16,8,258},{19,7,43},{0,8,117},{0,8,53},{0,9,202}, - {17,7,13},{0,8,101},{0,8,37},{0,9,170},{0,8,5},{0,8,133},{0,8,69}, - {0,9,234},{16,7,8},{0,8,93},{0,8,29},{0,9,154},{20,7,83},{0,8,125}, - {0,8,61},{0,9,218},{18,7,23},{0,8,109},{0,8,45},{0,9,186},{0,8,13}, - {0,8,141},{0,8,77},{0,9,250},{16,7,3},{0,8,83},{0,8,19},{21,8,195}, - {19,7,35},{0,8,115},{0,8,51},{0,9,198},{17,7,11},{0,8,99},{0,8,35}, - {0,9,166},{0,8,3},{0,8,131},{0,8,67},{0,9,230},{16,7,7},{0,8,91}, - {0,8,27},{0,9,150},{20,7,67},{0,8,123},{0,8,59},{0,9,214},{18,7,19}, - {0,8,107},{0,8,43},{0,9,182},{0,8,11},{0,8,139},{0,8,75},{0,9,246}, - {16,7,5},{0,8,87},{0,8,23},{64,8,0},{19,7,51},{0,8,119},{0,8,55}, - {0,9,206},{17,7,15},{0,8,103},{0,8,39},{0,9,174},{0,8,7},{0,8,135}, - {0,8,71},{0,9,238},{16,7,9},{0,8,95},{0,8,31},{0,9,158},{20,7,99}, - {0,8,127},{0,8,63},{0,9,222},{18,7,27},{0,8,111},{0,8,47},{0,9,190}, - {0,8,15},{0,8,143},{0,8,79},{0,9,254},{96,7,0},{0,8,80},{0,8,16}, - {20,8,115},{18,7,31},{0,8,112},{0,8,48},{0,9,193},{16,7,10},{0,8,96}, - {0,8,32},{0,9,161},{0,8,0},{0,8,128},{0,8,64},{0,9,225},{16,7,6}, - {0,8,88},{0,8,24},{0,9,145},{19,7,59},{0,8,120},{0,8,56},{0,9,209}, - {17,7,17},{0,8,104},{0,8,40},{0,9,177},{0,8,8},{0,8,136},{0,8,72}, - {0,9,241},{16,7,4},{0,8,84},{0,8,20},{21,8,227},{19,7,43},{0,8,116}, - {0,8,52},{0,9,201},{17,7,13},{0,8,100},{0,8,36},{0,9,169},{0,8,4}, - {0,8,132},{0,8,68},{0,9,233},{16,7,8},{0,8,92},{0,8,28},{0,9,153}, - {20,7,83},{0,8,124},{0,8,60},{0,9,217},{18,7,23},{0,8,108},{0,8,44}, - {0,9,185},{0,8,12},{0,8,140},{0,8,76},{0,9,249},{16,7,3},{0,8,82}, - {0,8,18},{21,8,163},{19,7,35},{0,8,114},{0,8,50},{0,9,197},{17,7,11}, - {0,8,98},{0,8,34},{0,9,165},{0,8,2},{0,8,130},{0,8,66},{0,9,229}, - {16,7,7},{0,8,90},{0,8,26},{0,9,149},{20,7,67},{0,8,122},{0,8,58}, - {0,9,213},{18,7,19},{0,8,106},{0,8,42},{0,9,181},{0,8,10},{0,8,138}, - {0,8,74},{0,9,245},{16,7,5},{0,8,86},{0,8,22},{64,8,0},{19,7,51}, - {0,8,118},{0,8,54},{0,9,205},{17,7,15},{0,8,102},{0,8,38},{0,9,173}, - {0,8,6},{0,8,134},{0,8,70},{0,9,237},{16,7,9},{0,8,94},{0,8,30}, - {0,9,157},{20,7,99},{0,8,126},{0,8,62},{0,9,221},{18,7,27},{0,8,110}, - {0,8,46},{0,9,189},{0,8,14},{0,8,142},{0,8,78},{0,9,253},{96,7,0}, - {0,8,81},{0,8,17},{21,8,131},{18,7,31},{0,8,113},{0,8,49},{0,9,195}, - {16,7,10},{0,8,97},{0,8,33},{0,9,163},{0,8,1},{0,8,129},{0,8,65}, - {0,9,227},{16,7,6},{0,8,89},{0,8,25},{0,9,147},{19,7,59},{0,8,121}, - {0,8,57},{0,9,211},{17,7,17},{0,8,105},{0,8,41},{0,9,179},{0,8,9}, - {0,8,137},{0,8,73},{0,9,243},{16,7,4},{0,8,85},{0,8,21},{16,8,258}, - {19,7,43},{0,8,117},{0,8,53},{0,9,203},{17,7,13},{0,8,101},{0,8,37}, - {0,9,171},{0,8,5},{0,8,133},{0,8,69},{0,9,235},{16,7,8},{0,8,93}, - {0,8,29},{0,9,155},{20,7,83},{0,8,125},{0,8,61},{0,9,219},{18,7,23}, - {0,8,109},{0,8,45},{0,9,187},{0,8,13},{0,8,141},{0,8,77},{0,9,251}, - {16,7,3},{0,8,83},{0,8,19},{21,8,195},{19,7,35},{0,8,115},{0,8,51}, - {0,9,199},{17,7,11},{0,8,99},{0,8,35},{0,9,167},{0,8,3},{0,8,131}, - {0,8,67},{0,9,231},{16,7,7},{0,8,91},{0,8,27},{0,9,151},{20,7,67}, - {0,8,123},{0,8,59},{0,9,215},{18,7,19},{0,8,107},{0,8,43},{0,9,183}, - {0,8,11},{0,8,139},{0,8,75},{0,9,247},{16,7,5},{0,8,87},{0,8,23}, - {64,8,0},{19,7,51},{0,8,119},{0,8,55},{0,9,207},{17,7,15},{0,8,103}, - {0,8,39},{0,9,175},{0,8,7},{0,8,135},{0,8,71},{0,9,239},{16,7,9}, - {0,8,95},{0,8,31},{0,9,159},{20,7,99},{0,8,127},{0,8,63},{0,9,223}, - {18,7,27},{0,8,111},{0,8,47},{0,9,191},{0,8,15},{0,8,143},{0,8,79}, - {0,9,255} - }; - - static const code distfix[32] = { - {16,5,1},{23,5,257},{19,5,17},{27,5,4097},{17,5,5},{25,5,1025}, - {21,5,65},{29,5,16385},{16,5,3},{24,5,513},{20,5,33},{28,5,8193}, - {18,5,9},{26,5,2049},{22,5,129},{64,5,0},{16,5,2},{23,5,385}, - {19,5,25},{27,5,6145},{17,5,7},{25,5,1537},{21,5,97},{29,5,24577}, - {16,5,4},{24,5,769},{20,5,49},{28,5,12289},{18,5,13},{26,5,3073}, - {22,5,193},{64,5,0} - }; diff --git a/code/3rd_glpk/zlib/inflate.c b/code/3rd_glpk/zlib/inflate.c deleted file mode 100644 index a8431abe..00000000 --- a/code/3rd_glpk/zlib/inflate.c +++ /dev/null @@ -1,1480 +0,0 @@ -/* inflate.c -- zlib decompression - * Copyright (C) 1995-2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* - * Change history: - * - * 1.2.beta0 24 Nov 2002 - * - First version -- complete rewrite of inflate to simplify code, avoid - * creation of window when not needed, minimize use of window when it is - * needed, make inffast.c even faster, implement gzip decoding, and to - * improve code readability and style over the previous zlib inflate code - * - * 1.2.beta1 25 Nov 2002 - * - Use pointers for available input and output checking in inffast.c - * - Remove input and output counters in inffast.c - * - Change inffast.c entry and loop from avail_in >= 7 to >= 6 - * - Remove unnecessary second byte pull from length extra in inffast.c - * - Unroll direct copy to three copies per loop in inffast.c - * - * 1.2.beta2 4 Dec 2002 - * - Change external routine names to reduce potential conflicts - * - Correct filename to inffixed.h for fixed tables in inflate.c - * - Make hbuf[] unsigned char to match parameter type in inflate.c - * - Change strm->next_out[-state->offset] to *(strm->next_out - state->offset) - * to avoid negation problem on Alphas (64 bit) in inflate.c - * - * 1.2.beta3 22 Dec 2002 - * - Add comments on state->bits assertion in inffast.c - * - Add comments on op field in inftrees.h - * - Fix bug in reuse of allocated window after inflateReset() - * - Remove bit fields--back to byte structure for speed - * - Remove distance extra == 0 check in inflate_fast()--only helps for lengths - * - Change post-increments to pre-increments in inflate_fast(), PPC biased? - * - Add compile time option, POSTINC, to use post-increments instead (Intel?) - * - Make MATCH copy in inflate() much faster for when inflate_fast() not used - * - Use local copies of stream next and avail values, as well as local bit - * buffer and bit count in inflate()--for speed when inflate_fast() not used - * - * 1.2.beta4 1 Jan 2003 - * - Split ptr - 257 statements in inflate_table() to avoid compiler warnings - * - Move a comment on output buffer sizes from inffast.c to inflate.c - * - Add comments in inffast.c to introduce the inflate_fast() routine - * - Rearrange window copies in inflate_fast() for speed and simplification - * - Unroll last copy for window match in inflate_fast() - * - Use local copies of window variables in inflate_fast() for speed - * - Pull out common wnext == 0 case for speed in inflate_fast() - * - Make op and len in inflate_fast() unsigned for consistency - * - Add FAR to lcode and dcode declarations in inflate_fast() - * - Simplified bad distance check in inflate_fast() - * - Added inflateBackInit(), inflateBack(), and inflateBackEnd() in new - * source file infback.c to provide a call-back interface to inflate for - * programs like gzip and unzip -- uses window as output buffer to avoid - * window copying - * - * 1.2.beta5 1 Jan 2003 - * - Improved inflateBack() interface to allow the caller to provide initial - * input in strm. - * - Fixed stored blocks bug in inflateBack() - * - * 1.2.beta6 4 Jan 2003 - * - Added comments in inffast.c on effectiveness of POSTINC - * - Typecasting all around to reduce compiler warnings - * - Changed loops from while (1) or do {} while (1) to for (;;), again to - * make compilers happy - * - Changed type of window in inflateBackInit() to unsigned char * - * - * 1.2.beta7 27 Jan 2003 - * - Changed many types to unsigned or unsigned short to avoid warnings - * - Added inflateCopy() function - * - * 1.2.0 9 Mar 2003 - * - Changed inflateBack() interface to provide separate opaque descriptors - * for the in() and out() functions - * - Changed inflateBack() argument and in_func typedef to swap the length - * and buffer address return values for the input function - * - Check next_in and next_out for Z_NULL on entry to inflate() - * - * The history for versions after 1.2.0 are in ChangeLog in zlib distribution. - */ - -#include "zutil.h" -#include "inftrees.h" -#include "inflate.h" -#include "inffast.h" - -#ifdef MAKEFIXED -# ifndef BUILDFIXED -# define BUILDFIXED -# endif -#endif - -/* function prototypes */ -local void fixedtables OF((struct inflate_state FAR *state)); -local int updatewindow OF((z_streamp strm, unsigned out)); -#ifdef BUILDFIXED - void makefixed OF((void)); -#endif -local unsigned syncsearch OF((unsigned FAR *have, unsigned char FAR *buf, - unsigned len)); - -int ZEXPORT inflateReset(strm) -z_streamp strm; -{ - struct inflate_state FAR *state; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - strm->total_in = strm->total_out = state->total = 0; - strm->msg = Z_NULL; - strm->adler = 1; /* to support ill-conceived Java test suite */ - state->mode = HEAD; - state->last = 0; - state->havedict = 0; - state->dmax = 32768U; - state->head = Z_NULL; - state->wsize = 0; - state->whave = 0; - state->wnext = 0; - state->hold = 0; - state->bits = 0; - state->lencode = state->distcode = state->next = state->codes; - state->sane = 1; - state->back = -1; - Tracev((stderr, "inflate: reset\n")); - return Z_OK; -} - -int ZEXPORT inflateReset2(strm, windowBits) -z_streamp strm; -int windowBits; -{ - int wrap; - struct inflate_state FAR *state; - - /* get the state */ - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - - /* extract wrap request from windowBits parameter */ - if (windowBits < 0) { - wrap = 0; - windowBits = -windowBits; - } - else { - wrap = (windowBits >> 4) + 1; -#ifdef GUNZIP - if (windowBits < 48) - windowBits &= 15; -#endif - } - - /* set number of window bits, free window if different */ - if (windowBits && (windowBits < 8 || windowBits > 15)) - return Z_STREAM_ERROR; - if (state->window != Z_NULL && state->wbits != (unsigned)windowBits) { - ZFREE(strm, state->window); - state->window = Z_NULL; - } - - /* update state and reset the rest of it */ - state->wrap = wrap; - state->wbits = (unsigned)windowBits; - return inflateReset(strm); -} - -int ZEXPORT inflateInit2_(strm, windowBits, version, stream_size) -z_streamp strm; -int windowBits; -const char *version; -int stream_size; -{ - int ret; - struct inflate_state FAR *state; - - if (version == Z_NULL || version[0] != ZLIB_VERSION[0] || - stream_size != (int)(sizeof(z_stream))) - return Z_VERSION_ERROR; - if (strm == Z_NULL) return Z_STREAM_ERROR; - strm->msg = Z_NULL; /* in case we return an error */ - if (strm->zalloc == (alloc_func)0) { - strm->zalloc = zcalloc; - strm->opaque = (voidpf)0; - } - if (strm->zfree == (free_func)0) strm->zfree = zcfree; - state = (struct inflate_state FAR *) - ZALLOC(strm, 1, sizeof(struct inflate_state)); - if (state == Z_NULL) return Z_MEM_ERROR; - Tracev((stderr, "inflate: allocated\n")); - strm->state = (struct internal_state FAR *)state; - state->window = Z_NULL; - ret = inflateReset2(strm, windowBits); - if (ret != Z_OK) { - ZFREE(strm, state); - strm->state = Z_NULL; - } - return ret; -} - -int ZEXPORT inflateInit_(strm, version, stream_size) -z_streamp strm; -const char *version; -int stream_size; -{ - return inflateInit2_(strm, DEF_WBITS, version, stream_size); -} - -int ZEXPORT inflatePrime(strm, bits, value) -z_streamp strm; -int bits; -int value; -{ - struct inflate_state FAR *state; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - if (bits < 0) { - state->hold = 0; - state->bits = 0; - return Z_OK; - } - if (bits > 16 || state->bits + bits > 32) return Z_STREAM_ERROR; - value &= (1L << bits) - 1; - state->hold += value << state->bits; - state->bits += bits; - return Z_OK; -} - -/* - Return state with length and distance decoding tables and index sizes set to - fixed code decoding. Normally this returns fixed tables from inffixed.h. - If BUILDFIXED is defined, then instead this routine builds the tables the - first time it's called, and returns those tables the first time and - thereafter. This reduces the size of the code by about 2K bytes, in - exchange for a little execution time. However, BUILDFIXED should not be - used for threaded applications, since the rewriting of the tables and virgin - may not be thread-safe. - */ -local void fixedtables(state) -struct inflate_state FAR *state; -{ -#ifdef BUILDFIXED - static int virgin = 1; - static code *lenfix, *distfix; - static code fixed[544]; - - /* build fixed huffman tables if first call (may not be thread safe) */ - if (virgin) { - unsigned sym, bits; - static code *next; - - /* literal/length table */ - sym = 0; - while (sym < 144) state->lens[sym++] = 8; - while (sym < 256) state->lens[sym++] = 9; - while (sym < 280) state->lens[sym++] = 7; - while (sym < 288) state->lens[sym++] = 8; - next = fixed; - lenfix = next; - bits = 9; - inflate_table(LENS, state->lens, 288, &(next), &(bits), state->work); - - /* distance table */ - sym = 0; - while (sym < 32) state->lens[sym++] = 5; - distfix = next; - bits = 5; - inflate_table(DISTS, state->lens, 32, &(next), &(bits), state->work); - - /* do this just once */ - virgin = 0; - } -#else /* !BUILDFIXED */ -# include "inffixed.h" -#endif /* BUILDFIXED */ - state->lencode = lenfix; - state->lenbits = 9; - state->distcode = distfix; - state->distbits = 5; -} - -#ifdef MAKEFIXED -#include - -/* - Write out the inffixed.h that is #include'd above. Defining MAKEFIXED also - defines BUILDFIXED, so the tables are built on the fly. makefixed() writes - those tables to stdout, which would be piped to inffixed.h. A small program - can simply call makefixed to do this: - - void makefixed(void); - - int main(void) - { - makefixed(); - return 0; - } - - Then that can be linked with zlib built with MAKEFIXED defined and run: - - a.out > inffixed.h - */ -void makefixed() -{ - unsigned low, size; - struct inflate_state state; - - fixedtables(&state); - puts(" /* inffixed.h -- table for decoding fixed codes"); - puts(" * Generated automatically by makefixed()."); - puts(" */"); - puts(""); - puts(" /* WARNING: this file should *not* be used by applications."); - puts(" It is part of the implementation of this library and is"); - puts(" subject to change. Applications should only use zlib.h."); - puts(" */"); - puts(""); - size = 1U << 9; - printf(" static const code lenfix[%u] = {", size); - low = 0; - for (;;) { - if ((low % 7) == 0) printf("\n "); - printf("{%u,%u,%d}", state.lencode[low].op, state.lencode[low].bits, - state.lencode[low].val); - if (++low == size) break; - putchar(','); - } - puts("\n };"); - size = 1U << 5; - printf("\n static const code distfix[%u] = {", size); - low = 0; - for (;;) { - if ((low % 6) == 0) printf("\n "); - printf("{%u,%u,%d}", state.distcode[low].op, state.distcode[low].bits, - state.distcode[low].val); - if (++low == size) break; - putchar(','); - } - puts("\n };"); -} -#endif /* MAKEFIXED */ - -/* - Update the window with the last wsize (normally 32K) bytes written before - returning. If window does not exist yet, create it. This is only called - when a window is already in use, or when output has been written during this - inflate call, but the end of the deflate stream has not been reached yet. - It is also called to create a window for dictionary data when a dictionary - is loaded. - - Providing output buffers larger than 32K to inflate() should provide a speed - advantage, since only the last 32K of output is copied to the sliding window - upon return from inflate(), and since all distances after the first 32K of - output will fall in the output data, making match copies simpler and faster. - The advantage may be dependent on the size of the processor's data caches. - */ -local int updatewindow(strm, out) -z_streamp strm; -unsigned out; -{ - struct inflate_state FAR *state; - unsigned copy, dist; - - state = (struct inflate_state FAR *)strm->state; - - /* if it hasn't been done already, allocate space for the window */ - if (state->window == Z_NULL) { - state->window = (unsigned char FAR *) - ZALLOC(strm, 1U << state->wbits, - sizeof(unsigned char)); - if (state->window == Z_NULL) return 1; - } - - /* if window not in use yet, initialize */ - if (state->wsize == 0) { - state->wsize = 1U << state->wbits; - state->wnext = 0; - state->whave = 0; - } - - /* copy state->wsize or less output bytes into the circular window */ - copy = out - strm->avail_out; - if (copy >= state->wsize) { - zmemcpy(state->window, strm->next_out - state->wsize, state->wsize); - state->wnext = 0; - state->whave = state->wsize; - } - else { - dist = state->wsize - state->wnext; - if (dist > copy) dist = copy; - zmemcpy(state->window + state->wnext, strm->next_out - copy, dist); - copy -= dist; - if (copy) { - zmemcpy(state->window, strm->next_out - copy, copy); - state->wnext = copy; - state->whave = state->wsize; - } - else { - state->wnext += dist; - if (state->wnext == state->wsize) state->wnext = 0; - if (state->whave < state->wsize) state->whave += dist; - } - } - return 0; -} - -/* Macros for inflate(): */ - -/* check function to use adler32() for zlib or crc32() for gzip */ -#ifdef GUNZIP -# define UPDATE(check, buf, len) \ - (state->flags ? crc32(check, buf, len) : adler32(check, buf, len)) -#else -# define UPDATE(check, buf, len) adler32(check, buf, len) -#endif - -/* check macros for header crc */ -#ifdef GUNZIP -# define CRC2(check, word) \ - do { \ - hbuf[0] = (unsigned char)(word); \ - hbuf[1] = (unsigned char)((word) >> 8); \ - check = crc32(check, hbuf, 2); \ - } while (0) - -# define CRC4(check, word) \ - do { \ - hbuf[0] = (unsigned char)(word); \ - hbuf[1] = (unsigned char)((word) >> 8); \ - hbuf[2] = (unsigned char)((word) >> 16); \ - hbuf[3] = (unsigned char)((word) >> 24); \ - check = crc32(check, hbuf, 4); \ - } while (0) -#endif - -/* Load registers with state in inflate() for speed */ -#define LOAD() \ - do { \ - put = strm->next_out; \ - left = strm->avail_out; \ - next = strm->next_in; \ - have = strm->avail_in; \ - hold = state->hold; \ - bits = state->bits; \ - } while (0) - -/* Restore state from registers in inflate() */ -#define RESTORE() \ - do { \ - strm->next_out = put; \ - strm->avail_out = left; \ - strm->next_in = next; \ - strm->avail_in = have; \ - state->hold = hold; \ - state->bits = bits; \ - } while (0) - -/* Clear the input bit accumulator */ -#define INITBITS() \ - do { \ - hold = 0; \ - bits = 0; \ - } while (0) - -/* Get a byte of input into the bit accumulator, or return from inflate() - if there is no input available. */ -#define PULLBYTE() \ - do { \ - if (have == 0) goto inf_leave; \ - have--; \ - hold += (unsigned long)(*next++) << bits; \ - bits += 8; \ - } while (0) - -/* Assure that there are at least n bits in the bit accumulator. If there is - not enough available input to do that, then return from inflate(). */ -#define NEEDBITS(n) \ - do { \ - while (bits < (unsigned)(n)) \ - PULLBYTE(); \ - } while (0) - -/* Return the low n bits of the bit accumulator (n < 16) */ -#define BITS(n) \ - ((unsigned)hold & ((1U << (n)) - 1)) - -/* Remove n bits from the bit accumulator */ -#define DROPBITS(n) \ - do { \ - hold >>= (n); \ - bits -= (unsigned)(n); \ - } while (0) - -/* Remove zero to seven bits as needed to go to a byte boundary */ -#define BYTEBITS() \ - do { \ - hold >>= bits & 7; \ - bits -= bits & 7; \ - } while (0) - -/* Reverse the bytes in a 32-bit value */ -#define REVERSE(q) \ - ((((q) >> 24) & 0xff) + (((q) >> 8) & 0xff00) + \ - (((q) & 0xff00) << 8) + (((q) & 0xff) << 24)) - -/* - inflate() uses a state machine to process as much input data and generate as - much output data as possible before returning. The state machine is - structured roughly as follows: - - for (;;) switch (state) { - ... - case STATEn: - if (not enough input data or output space to make progress) - return; - ... make progress ... - state = STATEm; - break; - ... - } - - so when inflate() is called again, the same case is attempted again, and - if the appropriate resources are provided, the machine proceeds to the - next state. The NEEDBITS() macro is usually the way the state evaluates - whether it can proceed or should return. NEEDBITS() does the return if - the requested bits are not available. The typical use of the BITS macros - is: - - NEEDBITS(n); - ... do something with BITS(n) ... - DROPBITS(n); - - where NEEDBITS(n) either returns from inflate() if there isn't enough - input left to load n bits into the accumulator, or it continues. BITS(n) - gives the low n bits in the accumulator. When done, DROPBITS(n) drops - the low n bits off the accumulator. INITBITS() clears the accumulator - and sets the number of available bits to zero. BYTEBITS() discards just - enough bits to put the accumulator on a byte boundary. After BYTEBITS() - and a NEEDBITS(8), then BITS(8) would return the next byte in the stream. - - NEEDBITS(n) uses PULLBYTE() to get an available byte of input, or to return - if there is no input available. The decoding of variable length codes uses - PULLBYTE() directly in order to pull just enough bytes to decode the next - code, and no more. - - Some states loop until they get enough input, making sure that enough - state information is maintained to continue the loop where it left off - if NEEDBITS() returns in the loop. For example, want, need, and keep - would all have to actually be part of the saved state in case NEEDBITS() - returns: - - case STATEw: - while (want < need) { - NEEDBITS(n); - keep[want++] = BITS(n); - DROPBITS(n); - } - state = STATEx; - case STATEx: - - As shown above, if the next state is also the next case, then the break - is omitted. - - A state may also return if there is not enough output space available to - complete that state. Those states are copying stored data, writing a - literal byte, and copying a matching string. - - When returning, a "goto inf_leave" is used to update the total counters, - update the check value, and determine whether any progress has been made - during that inflate() call in order to return the proper return code. - Progress is defined as a change in either strm->avail_in or strm->avail_out. - When there is a window, goto inf_leave will update the window with the last - output written. If a goto inf_leave occurs in the middle of decompression - and there is no window currently, goto inf_leave will create one and copy - output to the window for the next call of inflate(). - - In this implementation, the flush parameter of inflate() only affects the - return code (per zlib.h). inflate() always writes as much as possible to - strm->next_out, given the space available and the provided input--the effect - documented in zlib.h of Z_SYNC_FLUSH. Furthermore, inflate() always defers - the allocation of and copying into a sliding window until necessary, which - provides the effect documented in zlib.h for Z_FINISH when the entire input - stream available. So the only thing the flush parameter actually does is: - when flush is set to Z_FINISH, inflate() cannot return Z_OK. Instead it - will return Z_BUF_ERROR if it has not reached the end of the stream. - */ - -int ZEXPORT inflate(strm, flush) -z_streamp strm; -int flush; -{ - struct inflate_state FAR *state; - unsigned char FAR *next; /* next input */ - unsigned char FAR *put; /* next output */ - unsigned have, left; /* available input and output */ - unsigned long hold; /* bit buffer */ - unsigned bits; /* bits in bit buffer */ - unsigned in, out; /* save starting available input and output */ - unsigned copy; /* number of stored or match bytes to copy */ - unsigned char FAR *from; /* where to copy match bytes from */ - code here; /* current decoding table entry */ - code last; /* parent table entry */ - unsigned len; /* length to copy for repeats, bits to drop */ - int ret; /* return code */ -#ifdef GUNZIP - unsigned char hbuf[4]; /* buffer for gzip header crc calculation */ -#endif - static const unsigned short order[19] = /* permutation of code lengths */ - {16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15}; - - if (strm == Z_NULL || strm->state == Z_NULL || strm->next_out == Z_NULL || - (strm->next_in == Z_NULL && strm->avail_in != 0)) - return Z_STREAM_ERROR; - - state = (struct inflate_state FAR *)strm->state; - if (state->mode == TYPE) state->mode = TYPEDO; /* skip check */ - LOAD(); - in = have; - out = left; - ret = Z_OK; - for (;;) - switch (state->mode) { - case HEAD: - if (state->wrap == 0) { - state->mode = TYPEDO; - break; - } - NEEDBITS(16); -#ifdef GUNZIP - if ((state->wrap & 2) && hold == 0x8b1f) { /* gzip header */ - state->check = crc32(0L, Z_NULL, 0); - CRC2(state->check, hold); - INITBITS(); - state->mode = FLAGS; - break; - } - state->flags = 0; /* expect zlib header */ - if (state->head != Z_NULL) - state->head->done = -1; - if (!(state->wrap & 1) || /* check if zlib header allowed */ -#else - if ( -#endif - ((BITS(8) << 8) + (hold >> 8)) % 31) { - strm->msg = (char *)"incorrect header check"; - state->mode = BAD; - break; - } - if (BITS(4) != Z_DEFLATED) { - strm->msg = (char *)"unknown compression method"; - state->mode = BAD; - break; - } - DROPBITS(4); - len = BITS(4) + 8; - if (state->wbits == 0) - state->wbits = len; - else if (len > state->wbits) { - strm->msg = (char *)"invalid window size"; - state->mode = BAD; - break; - } - state->dmax = 1U << len; - Tracev((stderr, "inflate: zlib header ok\n")); - strm->adler = state->check = adler32(0L, Z_NULL, 0); - state->mode = hold & 0x200 ? DICTID : TYPE; - INITBITS(); - break; -#ifdef GUNZIP - case FLAGS: - NEEDBITS(16); - state->flags = (int)(hold); - if ((state->flags & 0xff) != Z_DEFLATED) { - strm->msg = (char *)"unknown compression method"; - state->mode = BAD; - break; - } - if (state->flags & 0xe000) { - strm->msg = (char *)"unknown header flags set"; - state->mode = BAD; - break; - } - if (state->head != Z_NULL) - state->head->text = (int)((hold >> 8) & 1); - if (state->flags & 0x0200) CRC2(state->check, hold); - INITBITS(); - state->mode = TIME; - case TIME: - NEEDBITS(32); - if (state->head != Z_NULL) - state->head->time = hold; - if (state->flags & 0x0200) CRC4(state->check, hold); - INITBITS(); - state->mode = OS; - case OS: - NEEDBITS(16); - if (state->head != Z_NULL) { - state->head->xflags = (int)(hold & 0xff); - state->head->os = (int)(hold >> 8); - } - if (state->flags & 0x0200) CRC2(state->check, hold); - INITBITS(); - state->mode = EXLEN; - case EXLEN: - if (state->flags & 0x0400) { - NEEDBITS(16); - state->length = (unsigned)(hold); - if (state->head != Z_NULL) - state->head->extra_len = (unsigned)hold; - if (state->flags & 0x0200) CRC2(state->check, hold); - INITBITS(); - } - else if (state->head != Z_NULL) - state->head->extra = Z_NULL; - state->mode = EXTRA; - case EXTRA: - if (state->flags & 0x0400) { - copy = state->length; - if (copy > have) copy = have; - if (copy) { - if (state->head != Z_NULL && - state->head->extra != Z_NULL) { - len = state->head->extra_len - state->length; - zmemcpy(state->head->extra + len, next, - len + copy > state->head->extra_max ? - state->head->extra_max - len : copy); - } - if (state->flags & 0x0200) - state->check = crc32(state->check, next, copy); - have -= copy; - next += copy; - state->length -= copy; - } - if (state->length) goto inf_leave; - } - state->length = 0; - state->mode = NAME; - case NAME: - if (state->flags & 0x0800) { - if (have == 0) goto inf_leave; - copy = 0; - do { - len = (unsigned)(next[copy++]); - if (state->head != Z_NULL && - state->head->name != Z_NULL && - state->length < state->head->name_max) - state->head->name[state->length++] = len; - } while (len && copy < have); - if (state->flags & 0x0200) - state->check = crc32(state->check, next, copy); - have -= copy; - next += copy; - if (len) goto inf_leave; - } - else if (state->head != Z_NULL) - state->head->name = Z_NULL; - state->length = 0; - state->mode = COMMENT; - case COMMENT: - if (state->flags & 0x1000) { - if (have == 0) goto inf_leave; - copy = 0; - do { - len = (unsigned)(next[copy++]); - if (state->head != Z_NULL && - state->head->comment != Z_NULL && - state->length < state->head->comm_max) - state->head->comment[state->length++] = len; - } while (len && copy < have); - if (state->flags & 0x0200) - state->check = crc32(state->check, next, copy); - have -= copy; - next += copy; - if (len) goto inf_leave; - } - else if (state->head != Z_NULL) - state->head->comment = Z_NULL; - state->mode = HCRC; - case HCRC: - if (state->flags & 0x0200) { - NEEDBITS(16); - if (hold != (state->check & 0xffff)) { - strm->msg = (char *)"header crc mismatch"; - state->mode = BAD; - break; - } - INITBITS(); - } - if (state->head != Z_NULL) { - state->head->hcrc = (int)((state->flags >> 9) & 1); - state->head->done = 1; - } - strm->adler = state->check = crc32(0L, Z_NULL, 0); - state->mode = TYPE; - break; -#endif - case DICTID: - NEEDBITS(32); - strm->adler = state->check = REVERSE(hold); - INITBITS(); - state->mode = DICT; - case DICT: - if (state->havedict == 0) { - RESTORE(); - return Z_NEED_DICT; - } - strm->adler = state->check = adler32(0L, Z_NULL, 0); - state->mode = TYPE; - case TYPE: - if (flush == Z_BLOCK || flush == Z_TREES) goto inf_leave; - case TYPEDO: - if (state->last) { - BYTEBITS(); - state->mode = CHECK; - break; - } - NEEDBITS(3); - state->last = BITS(1); - DROPBITS(1); - switch (BITS(2)) { - case 0: /* stored block */ - Tracev((stderr, "inflate: stored block%s\n", - state->last ? " (last)" : "")); - state->mode = STORED; - break; - case 1: /* fixed block */ - fixedtables(state); - Tracev((stderr, "inflate: fixed codes block%s\n", - state->last ? " (last)" : "")); - state->mode = LEN_; /* decode codes */ - if (flush == Z_TREES) { - DROPBITS(2); - goto inf_leave; - } - break; - case 2: /* dynamic block */ - Tracev((stderr, "inflate: dynamic codes block%s\n", - state->last ? " (last)" : "")); - state->mode = TABLE; - break; - case 3: - strm->msg = (char *)"invalid block type"; - state->mode = BAD; - } - DROPBITS(2); - break; - case STORED: - BYTEBITS(); /* go to byte boundary */ - NEEDBITS(32); - if ((hold & 0xffff) != ((hold >> 16) ^ 0xffff)) { - strm->msg = (char *)"invalid stored block lengths"; - state->mode = BAD; - break; - } - state->length = (unsigned)hold & 0xffff; - Tracev((stderr, "inflate: stored length %u\n", - state->length)); - INITBITS(); - state->mode = COPY_; - if (flush == Z_TREES) goto inf_leave; - case COPY_: - state->mode = COPY; - case COPY: - copy = state->length; - if (copy) { - if (copy > have) copy = have; - if (copy > left) copy = left; - if (copy == 0) goto inf_leave; - zmemcpy(put, next, copy); - have -= copy; - next += copy; - left -= copy; - put += copy; - state->length -= copy; - break; - } - Tracev((stderr, "inflate: stored end\n")); - state->mode = TYPE; - break; - case TABLE: - NEEDBITS(14); - state->nlen = BITS(5) + 257; - DROPBITS(5); - state->ndist = BITS(5) + 1; - DROPBITS(5); - state->ncode = BITS(4) + 4; - DROPBITS(4); -#ifndef PKZIP_BUG_WORKAROUND - if (state->nlen > 286 || state->ndist > 30) { - strm->msg = (char *)"too many length or distance symbols"; - state->mode = BAD; - break; - } -#endif - Tracev((stderr, "inflate: table sizes ok\n")); - state->have = 0; - state->mode = LENLENS; - case LENLENS: - while (state->have < state->ncode) { - NEEDBITS(3); - state->lens[order[state->have++]] = (unsigned short)BITS(3); - DROPBITS(3); - } - while (state->have < 19) - state->lens[order[state->have++]] = 0; - state->next = state->codes; - state->lencode = (code const FAR *)(state->next); - state->lenbits = 7; - ret = inflate_table(CODES, state->lens, 19, &(state->next), - &(state->lenbits), state->work); - if (ret) { - strm->msg = (char *)"invalid code lengths set"; - state->mode = BAD; - break; - } - Tracev((stderr, "inflate: code lengths ok\n")); - state->have = 0; - state->mode = CODELENS; - case CODELENS: - while (state->have < state->nlen + state->ndist) { - for (;;) { - here = state->lencode[BITS(state->lenbits)]; - if ((unsigned)(here.bits) <= bits) break; - PULLBYTE(); - } - if (here.val < 16) { - NEEDBITS(here.bits); - DROPBITS(here.bits); - state->lens[state->have++] = here.val; - } - else { - if (here.val == 16) { - NEEDBITS(here.bits + 2); - DROPBITS(here.bits); - if (state->have == 0) { - strm->msg = (char *)"invalid bit length repeat"; - state->mode = BAD; - break; - } - len = state->lens[state->have - 1]; - copy = 3 + BITS(2); - DROPBITS(2); - } - else if (here.val == 17) { - NEEDBITS(here.bits + 3); - DROPBITS(here.bits); - len = 0; - copy = 3 + BITS(3); - DROPBITS(3); - } - else { - NEEDBITS(here.bits + 7); - DROPBITS(here.bits); - len = 0; - copy = 11 + BITS(7); - DROPBITS(7); - } - if (state->have + copy > state->nlen + state->ndist) { - strm->msg = (char *)"invalid bit length repeat"; - state->mode = BAD; - break; - } - while (copy--) - state->lens[state->have++] = (unsigned short)len; - } - } - - /* handle error breaks in while */ - if (state->mode == BAD) break; - - /* check for end-of-block code (better have one) */ - if (state->lens[256] == 0) { - strm->msg = (char *)"invalid code -- missing end-of-block"; - state->mode = BAD; - break; - } - - /* build code tables -- note: do not change the lenbits or distbits - values here (9 and 6) without reading the comments in inftrees.h - concerning the ENOUGH constants, which depend on those values */ - state->next = state->codes; - state->lencode = (code const FAR *)(state->next); - state->lenbits = 9; - ret = inflate_table(LENS, state->lens, state->nlen, &(state->next), - &(state->lenbits), state->work); - if (ret) { - strm->msg = (char *)"invalid literal/lengths set"; - state->mode = BAD; - break; - } - state->distcode = (code const FAR *)(state->next); - state->distbits = 6; - ret = inflate_table(DISTS, state->lens + state->nlen, state->ndist, - &(state->next), &(state->distbits), state->work); - if (ret) { - strm->msg = (char *)"invalid distances set"; - state->mode = BAD; - break; - } - Tracev((stderr, "inflate: codes ok\n")); - state->mode = LEN_; - if (flush == Z_TREES) goto inf_leave; - case LEN_: - state->mode = LEN; - case LEN: - if (have >= 6 && left >= 258) { - RESTORE(); - inflate_fast(strm, out); - LOAD(); - if (state->mode == TYPE) - state->back = -1; - break; - } - state->back = 0; - for (;;) { - here = state->lencode[BITS(state->lenbits)]; - if ((unsigned)(here.bits) <= bits) break; - PULLBYTE(); - } - if (here.op && (here.op & 0xf0) == 0) { - last = here; - for (;;) { - here = state->lencode[last.val + - (BITS(last.bits + last.op) >> last.bits)]; - if ((unsigned)(last.bits + here.bits) <= bits) break; - PULLBYTE(); - } - DROPBITS(last.bits); - state->back += last.bits; - } - DROPBITS(here.bits); - state->back += here.bits; - state->length = (unsigned)here.val; - if ((int)(here.op) == 0) { - Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ? - "inflate: literal '%c'\n" : - "inflate: literal 0x%02x\n", here.val)); - state->mode = LIT; - break; - } - if (here.op & 32) { - Tracevv((stderr, "inflate: end of block\n")); - state->back = -1; - state->mode = TYPE; - break; - } - if (here.op & 64) { - strm->msg = (char *)"invalid literal/length code"; - state->mode = BAD; - break; - } - state->extra = (unsigned)(here.op) & 15; - state->mode = LENEXT; - case LENEXT: - if (state->extra) { - NEEDBITS(state->extra); - state->length += BITS(state->extra); - DROPBITS(state->extra); - state->back += state->extra; - } - Tracevv((stderr, "inflate: length %u\n", state->length)); - state->was = state->length; - state->mode = DIST; - case DIST: - for (;;) { - here = state->distcode[BITS(state->distbits)]; - if ((unsigned)(here.bits) <= bits) break; - PULLBYTE(); - } - if ((here.op & 0xf0) == 0) { - last = here; - for (;;) { - here = state->distcode[last.val + - (BITS(last.bits + last.op) >> last.bits)]; - if ((unsigned)(last.bits + here.bits) <= bits) break; - PULLBYTE(); - } - DROPBITS(last.bits); - state->back += last.bits; - } - DROPBITS(here.bits); - state->back += here.bits; - if (here.op & 64) { - strm->msg = (char *)"invalid distance code"; - state->mode = BAD; - break; - } - state->offset = (unsigned)here.val; - state->extra = (unsigned)(here.op) & 15; - state->mode = DISTEXT; - case DISTEXT: - if (state->extra) { - NEEDBITS(state->extra); - state->offset += BITS(state->extra); - DROPBITS(state->extra); - state->back += state->extra; - } -#ifdef INFLATE_STRICT - if (state->offset > state->dmax) { - strm->msg = (char *)"invalid distance too far back"; - state->mode = BAD; - break; - } -#endif - Tracevv((stderr, "inflate: distance %u\n", state->offset)); - state->mode = MATCH; - case MATCH: - if (left == 0) goto inf_leave; - copy = out - left; - if (state->offset > copy) { /* copy from window */ - copy = state->offset - copy; - if (copy > state->whave) { - if (state->sane) { - strm->msg = (char *)"invalid distance too far back"; - state->mode = BAD; - break; - } -#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR - Trace((stderr, "inflate.c too far\n")); - copy -= state->whave; - if (copy > state->length) copy = state->length; - if (copy > left) copy = left; - left -= copy; - state->length -= copy; - do { - *put++ = 0; - } while (--copy); - if (state->length == 0) state->mode = LEN; - break; -#endif - } - if (copy > state->wnext) { - copy -= state->wnext; - from = state->window + (state->wsize - copy); - } - else - from = state->window + (state->wnext - copy); - if (copy > state->length) copy = state->length; - } - else { /* copy from output */ - from = put - state->offset; - copy = state->length; - } - if (copy > left) copy = left; - left -= copy; - state->length -= copy; - do { - *put++ = *from++; - } while (--copy); - if (state->length == 0) state->mode = LEN; - break; - case LIT: - if (left == 0) goto inf_leave; - *put++ = (unsigned char)(state->length); - left--; - state->mode = LEN; - break; - case CHECK: - if (state->wrap) { - NEEDBITS(32); - out -= left; - strm->total_out += out; - state->total += out; - if (out) - strm->adler = state->check = - UPDATE(state->check, put - out, out); - out = left; - if (( -#ifdef GUNZIP - state->flags ? hold : -#endif - REVERSE(hold)) != state->check) { - strm->msg = (char *)"incorrect data check"; - state->mode = BAD; - break; - } - INITBITS(); - Tracev((stderr, "inflate: check matches trailer\n")); - } -#ifdef GUNZIP - state->mode = LENGTH; - case LENGTH: - if (state->wrap && state->flags) { - NEEDBITS(32); - if (hold != (state->total & 0xffffffffUL)) { - strm->msg = (char *)"incorrect length check"; - state->mode = BAD; - break; - } - INITBITS(); - Tracev((stderr, "inflate: length matches trailer\n")); - } -#endif - state->mode = DONE; - case DONE: - ret = Z_STREAM_END; - goto inf_leave; - case BAD: - ret = Z_DATA_ERROR; - goto inf_leave; - case MEM: - return Z_MEM_ERROR; - case SYNC: - default: - return Z_STREAM_ERROR; - } - - /* - Return from inflate(), updating the total counts and the check value. - If there was no progress during the inflate() call, return a buffer - error. Call updatewindow() to create and/or update the window state. - Note: a memory error from inflate() is non-recoverable. - */ - inf_leave: - RESTORE(); - if (state->wsize || (state->mode < CHECK && out != strm->avail_out)) - if (updatewindow(strm, out)) { - state->mode = MEM; - return Z_MEM_ERROR; - } - in -= strm->avail_in; - out -= strm->avail_out; - strm->total_in += in; - strm->total_out += out; - state->total += out; - if (state->wrap && out) - strm->adler = state->check = - UPDATE(state->check, strm->next_out - out, out); - strm->data_type = state->bits + (state->last ? 64 : 0) + - (state->mode == TYPE ? 128 : 0) + - (state->mode == LEN_ || state->mode == COPY_ ? 256 : 0); - if (((in == 0 && out == 0) || flush == Z_FINISH) && ret == Z_OK) - ret = Z_BUF_ERROR; - return ret; -} - -int ZEXPORT inflateEnd(strm) -z_streamp strm; -{ - struct inflate_state FAR *state; - if (strm == Z_NULL || strm->state == Z_NULL || strm->zfree == (free_func)0) - return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - if (state->window != Z_NULL) ZFREE(strm, state->window); - ZFREE(strm, strm->state); - strm->state = Z_NULL; - Tracev((stderr, "inflate: end\n")); - return Z_OK; -} - -int ZEXPORT inflateSetDictionary(strm, dictionary, dictLength) -z_streamp strm; -const Bytef *dictionary; -uInt dictLength; -{ - struct inflate_state FAR *state; - unsigned long id; - - /* check state */ - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - if (state->wrap != 0 && state->mode != DICT) - return Z_STREAM_ERROR; - - /* check for correct dictionary id */ - if (state->mode == DICT) { - id = adler32(0L, Z_NULL, 0); - id = adler32(id, dictionary, dictLength); - if (id != state->check) - return Z_DATA_ERROR; - } - - /* copy dictionary to window */ - if (updatewindow(strm, strm->avail_out)) { - state->mode = MEM; - return Z_MEM_ERROR; - } - if (dictLength > state->wsize) { - zmemcpy(state->window, dictionary + dictLength - state->wsize, - state->wsize); - state->whave = state->wsize; - } - else { - zmemcpy(state->window + state->wsize - dictLength, dictionary, - dictLength); - state->whave = dictLength; - } - state->havedict = 1; - Tracev((stderr, "inflate: dictionary set\n")); - return Z_OK; -} - -int ZEXPORT inflateGetHeader(strm, head) -z_streamp strm; -gz_headerp head; -{ - struct inflate_state FAR *state; - - /* check state */ - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - if ((state->wrap & 2) == 0) return Z_STREAM_ERROR; - - /* save header structure */ - state->head = head; - head->done = 0; - return Z_OK; -} - -/* - Search buf[0..len-1] for the pattern: 0, 0, 0xff, 0xff. Return when found - or when out of input. When called, *have is the number of pattern bytes - found in order so far, in 0..3. On return *have is updated to the new - state. If on return *have equals four, then the pattern was found and the - return value is how many bytes were read including the last byte of the - pattern. If *have is less than four, then the pattern has not been found - yet and the return value is len. In the latter case, syncsearch() can be - called again with more data and the *have state. *have is initialized to - zero for the first call. - */ -local unsigned syncsearch(have, buf, len) -unsigned FAR *have; -unsigned char FAR *buf; -unsigned len; -{ - unsigned got; - unsigned next; - - got = *have; - next = 0; - while (next < len && got < 4) { - if ((int)(buf[next]) == (got < 2 ? 0 : 0xff)) - got++; - else if (buf[next]) - got = 0; - else - got = 4 - got; - next++; - } - *have = got; - return next; -} - -int ZEXPORT inflateSync(strm) -z_streamp strm; -{ - unsigned len; /* number of bytes to look at or looked at */ - unsigned long in, out; /* temporary to save total_in and total_out */ - unsigned char buf[4]; /* to restore bit buffer to byte string */ - struct inflate_state FAR *state; - - /* check parameters */ - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - if (strm->avail_in == 0 && state->bits < 8) return Z_BUF_ERROR; - - /* if first time, start search in bit buffer */ - if (state->mode != SYNC) { - state->mode = SYNC; - state->hold <<= state->bits & 7; - state->bits -= state->bits & 7; - len = 0; - while (state->bits >= 8) { - buf[len++] = (unsigned char)(state->hold); - state->hold >>= 8; - state->bits -= 8; - } - state->have = 0; - syncsearch(&(state->have), buf, len); - } - - /* search available input */ - len = syncsearch(&(state->have), strm->next_in, strm->avail_in); - strm->avail_in -= len; - strm->next_in += len; - strm->total_in += len; - - /* return no joy or set up to restart inflate() on a new block */ - if (state->have != 4) return Z_DATA_ERROR; - in = strm->total_in; out = strm->total_out; - inflateReset(strm); - strm->total_in = in; strm->total_out = out; - state->mode = TYPE; - return Z_OK; -} - -/* - Returns true if inflate is currently at the end of a block generated by - Z_SYNC_FLUSH or Z_FULL_FLUSH. This function is used by one PPP - implementation to provide an additional safety check. PPP uses - Z_SYNC_FLUSH but removes the length bytes of the resulting empty stored - block. When decompressing, PPP checks that at the end of input packet, - inflate is waiting for these length bytes. - */ -int ZEXPORT inflateSyncPoint(strm) -z_streamp strm; -{ - struct inflate_state FAR *state; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - return state->mode == STORED && state->bits == 0; -} - -int ZEXPORT inflateCopy(dest, source) -z_streamp dest; -z_streamp source; -{ - struct inflate_state FAR *state; - struct inflate_state FAR *copy; - unsigned char FAR *window; - unsigned wsize; - - /* check input */ - if (dest == Z_NULL || source == Z_NULL || source->state == Z_NULL || - source->zalloc == (alloc_func)0 || source->zfree == (free_func)0) - return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)source->state; - - /* allocate space */ - copy = (struct inflate_state FAR *) - ZALLOC(source, 1, sizeof(struct inflate_state)); - if (copy == Z_NULL) return Z_MEM_ERROR; - window = Z_NULL; - if (state->window != Z_NULL) { - window = (unsigned char FAR *) - ZALLOC(source, 1U << state->wbits, sizeof(unsigned char)); - if (window == Z_NULL) { - ZFREE(source, copy); - return Z_MEM_ERROR; - } - } - - /* copy state */ - zmemcpy(dest, source, sizeof(z_stream)); - zmemcpy(copy, state, sizeof(struct inflate_state)); - if (state->lencode >= state->codes && - state->lencode <= state->codes + ENOUGH - 1) { - copy->lencode = copy->codes + (state->lencode - state->codes); - copy->distcode = copy->codes + (state->distcode - state->codes); - } - copy->next = copy->codes + (state->next - state->codes); - if (window != Z_NULL) { - wsize = 1U << state->wbits; - zmemcpy(window, state->window, wsize); - } - copy->window = window; - dest->state = (struct internal_state FAR *)copy; - return Z_OK; -} - -int ZEXPORT inflateUndermine(strm, subvert) -z_streamp strm; -int subvert; -{ - struct inflate_state FAR *state; - - if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR; - state = (struct inflate_state FAR *)strm->state; - state->sane = !subvert; -#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR - return Z_OK; -#else - state->sane = 1; - return Z_DATA_ERROR; -#endif -} - -long ZEXPORT inflateMark(strm) -z_streamp strm; -{ - struct inflate_state FAR *state; - - if (strm == Z_NULL || strm->state == Z_NULL) return -1L << 16; - state = (struct inflate_state FAR *)strm->state; - return ((long)(state->back) << 16) + - (state->mode == COPY ? state->length : - (state->mode == MATCH ? state->was - state->length : 0)); -} diff --git a/code/3rd_glpk/zlib/inflate.h b/code/3rd_glpk/zlib/inflate.h deleted file mode 100644 index 95f4986d..00000000 --- a/code/3rd_glpk/zlib/inflate.h +++ /dev/null @@ -1,122 +0,0 @@ -/* inflate.h -- internal inflate state definition - * Copyright (C) 1995-2009 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. - */ - -/* define NO_GZIP when compiling if you want to disable gzip header and - trailer decoding by inflate(). NO_GZIP would be used to avoid linking in - the crc code when it is not needed. For shared libraries, gzip decoding - should be left enabled. */ -#ifndef NO_GZIP -# define GUNZIP -#endif - -/* Possible inflate modes between inflate() calls */ -typedef enum { - HEAD, /* i: waiting for magic header */ - FLAGS, /* i: waiting for method and flags (gzip) */ - TIME, /* i: waiting for modification time (gzip) */ - OS, /* i: waiting for extra flags and operating system (gzip) */ - EXLEN, /* i: waiting for extra length (gzip) */ - EXTRA, /* i: waiting for extra bytes (gzip) */ - NAME, /* i: waiting for end of file name (gzip) */ - COMMENT, /* i: waiting for end of comment (gzip) */ - HCRC, /* i: waiting for header crc (gzip) */ - DICTID, /* i: waiting for dictionary check value */ - DICT, /* waiting for inflateSetDictionary() call */ - TYPE, /* i: waiting for type bits, including last-flag bit */ - TYPEDO, /* i: same, but skip check to exit inflate on new block */ - STORED, /* i: waiting for stored size (length and complement) */ - COPY_, /* i/o: same as COPY below, but only first time in */ - COPY, /* i/o: waiting for input or output to copy stored block */ - TABLE, /* i: waiting for dynamic block table lengths */ - LENLENS, /* i: waiting for code length code lengths */ - CODELENS, /* i: waiting for length/lit and distance code lengths */ - LEN_, /* i: same as LEN below, but only first time in */ - LEN, /* i: waiting for length/lit/eob code */ - LENEXT, /* i: waiting for length extra bits */ - DIST, /* i: waiting for distance code */ - DISTEXT, /* i: waiting for distance extra bits */ - MATCH, /* o: waiting for output space to copy string */ - LIT, /* o: waiting for output space to write literal */ - CHECK, /* i: waiting for 32-bit check value */ - LENGTH, /* i: waiting for 32-bit length (gzip) */ - DONE, /* finished check, done -- remain here until reset */ - BAD, /* got a data error -- remain here until reset */ - MEM, /* got an inflate() memory error -- remain here until reset */ - SYNC /* looking for synchronization bytes to restart inflate() */ -} inflate_mode; - -/* - State transitions between above modes - - - (most modes can go to BAD or MEM on error -- not shown for clarity) - - Process header: - HEAD -> (gzip) or (zlib) or (raw) - (gzip) -> FLAGS -> TIME -> OS -> EXLEN -> EXTRA -> NAME -> COMMENT -> - HCRC -> TYPE - (zlib) -> DICTID or TYPE - DICTID -> DICT -> TYPE - (raw) -> TYPEDO - Read deflate blocks: - TYPE -> TYPEDO -> STORED or TABLE or LEN_ or CHECK - STORED -> COPY_ -> COPY -> TYPE - TABLE -> LENLENS -> CODELENS -> LEN_ - LEN_ -> LEN - Read deflate codes in fixed or dynamic block: - LEN -> LENEXT or LIT or TYPE - LENEXT -> DIST -> DISTEXT -> MATCH -> LEN - LIT -> LEN - Process trailer: - CHECK -> LENGTH -> DONE - */ - -/* state maintained between inflate() calls. Approximately 10K bytes. */ -struct inflate_state { - inflate_mode mode; /* current inflate mode */ - int last; /* true if processing last block */ - int wrap; /* bit 0 true for zlib, bit 1 true for gzip */ - int havedict; /* true if dictionary provided */ - int flags; /* gzip header method and flags (0 if zlib) */ - unsigned dmax; /* zlib header max distance (INFLATE_STRICT) */ - unsigned long check; /* protected copy of check value */ - unsigned long total; /* protected copy of output count */ - gz_headerp head; /* where to save gzip header information */ - /* sliding window */ - unsigned wbits; /* log base 2 of requested window size */ - unsigned wsize; /* window size or zero if not using window */ - unsigned whave; /* valid bytes in the window */ - unsigned wnext; /* window write index */ - unsigned char FAR *window; /* allocated sliding window, if needed */ - /* bit accumulator */ - unsigned long hold; /* input bit accumulator */ - unsigned bits; /* number of bits in "in" */ - /* for string and stored block copying */ - unsigned length; /* literal or length of data to copy */ - unsigned offset; /* distance back to copy string from */ - /* for table and code decoding */ - unsigned extra; /* extra bits needed */ - /* fixed and dynamic code tables */ - code const FAR *lencode; /* starting table for length/literal codes */ - code const FAR *distcode; /* starting table for distance codes */ - unsigned lenbits; /* index bits for lencode */ - unsigned distbits; /* index bits for distcode */ - /* dynamic table building */ - unsigned ncode; /* number of code length code lengths */ - unsigned nlen; /* number of length code lengths */ - unsigned ndist; /* number of distance code lengths */ - unsigned have; /* number of code lengths in lens[] */ - code FAR *next; /* next available space in codes[] */ - unsigned short lens[320]; /* temporary storage for code lengths */ - unsigned short work[288]; /* work area for code table building */ - code codes[ENOUGH]; /* space for code tables */ - int sane; /* if false, allow invalid distance too far */ - int back; /* bits back of last unprocessed length/lit */ - unsigned was; /* initial length of match */ -}; diff --git a/code/3rd_glpk/zlib/inftrees.c b/code/3rd_glpk/zlib/inftrees.c deleted file mode 100644 index 11e9c52a..00000000 --- a/code/3rd_glpk/zlib/inftrees.c +++ /dev/null @@ -1,330 +0,0 @@ -/* inftrees.c -- generate Huffman trees for efficient decoding - * Copyright (C) 1995-2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -#include "zutil.h" -#include "inftrees.h" - -#define MAXBITS 15 - -const char inflate_copyright[] = - " inflate 1.2.5 Copyright 1995-2010 Mark Adler "; -/* - If you use the zlib library in a product, an acknowledgment is welcome - in the documentation of your product. If for some reason you cannot - include such an acknowledgment, I would appreciate that you keep this - copyright string in the executable of your product. - */ - -/* - Build a set of tables to decode the provided canonical Huffman code. - The code lengths are lens[0..codes-1]. The result starts at *table, - whose indices are 0..2^bits-1. work is a writable array of at least - lens shorts, which is used as a work area. type is the type of code - to be generated, CODES, LENS, or DISTS. On return, zero is success, - -1 is an invalid code, and +1 means that ENOUGH isn't enough. table - on return points to the next available entry's address. bits is the - requested root table index bits, and on return it is the actual root - table index bits. It will differ if the request is greater than the - longest code or if it is less than the shortest code. - */ -int ZLIB_INTERNAL inflate_table(type, lens, codes, table, bits, work) -codetype type; -unsigned short FAR *lens; -unsigned codes; -code FAR * FAR *table; -unsigned FAR *bits; -unsigned short FAR *work; -{ - unsigned len; /* a code's length in bits */ - unsigned sym; /* index of code symbols */ - unsigned min, max; /* minimum and maximum code lengths */ - unsigned root; /* number of index bits for root table */ - unsigned curr; /* number of index bits for current table */ - unsigned drop; /* code bits to drop for sub-table */ - int left; /* number of prefix codes available */ - unsigned used; /* code entries in table used */ - unsigned huff; /* Huffman code */ - unsigned incr; /* for incrementing code, index */ - unsigned fill; /* index for replicating entries */ - unsigned low; /* low bits for current root entry */ - unsigned mask; /* mask for low root bits */ - code here; /* table entry for duplication */ - code FAR *next; /* next available space in table */ - const unsigned short FAR *base; /* base value table to use */ - const unsigned short FAR *extra; /* extra bits table to use */ - int end; /* use base and extra for symbol > end */ - unsigned short count[MAXBITS+1]; /* number of codes of each length */ - unsigned short offs[MAXBITS+1]; /* offsets in table for each length */ - static const unsigned short lbase[31] = { /* Length codes 257..285 base */ - 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31, - 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0}; - static const unsigned short lext[31] = { /* Length codes 257..285 extra */ - 16, 16, 16, 16, 16, 16, 16, 16, 17, 17, 17, 17, 18, 18, 18, 18, - 19, 19, 19, 19, 20, 20, 20, 20, 21, 21, 21, 21, 16, 73, 195}; - static const unsigned short dbase[32] = { /* Distance codes 0..29 base */ - 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, - 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, - 8193, 12289, 16385, 24577, 0, 0}; - static const unsigned short dext[32] = { /* Distance codes 0..29 extra */ - 16, 16, 16, 16, 17, 17, 18, 18, 19, 19, 20, 20, 21, 21, 22, 22, - 23, 23, 24, 24, 25, 25, 26, 26, 27, 27, - 28, 28, 29, 29, 64, 64}; - - /* - Process a set of code lengths to create a canonical Huffman code. The - code lengths are lens[0..codes-1]. Each length corresponds to the - symbols 0..codes-1. The Huffman code is generated by first sorting the - symbols by length from short to long, and retaining the symbol order - for codes with equal lengths. Then the code starts with all zero bits - for the first code of the shortest length, and the codes are integer - increments for the same length, and zeros are appended as the length - increases. For the deflate format, these bits are stored backwards - from their more natural integer increment ordering, and so when the - decoding tables are built in the large loop below, the integer codes - are incremented backwards. - - This routine assumes, but does not check, that all of the entries in - lens[] are in the range 0..MAXBITS. The caller must assure this. - 1..MAXBITS is interpreted as that code length. zero means that that - symbol does not occur in this code. - - The codes are sorted by computing a count of codes for each length, - creating from that a table of starting indices for each length in the - sorted table, and then entering the symbols in order in the sorted - table. The sorted table is work[], with that space being provided by - the caller. - - The length counts are used for other purposes as well, i.e. finding - the minimum and maximum length codes, determining if there are any - codes at all, checking for a valid set of lengths, and looking ahead - at length counts to determine sub-table sizes when building the - decoding tables. - */ - - /* accumulate lengths for codes (assumes lens[] all in 0..MAXBITS) */ - for (len = 0; len <= MAXBITS; len++) - count[len] = 0; - for (sym = 0; sym < codes; sym++) - count[lens[sym]]++; - - /* bound code lengths, force root to be within code lengths */ - root = *bits; - for (max = MAXBITS; max >= 1; max--) - if (count[max] != 0) break; - if (root > max) root = max; - if (max == 0) { /* no symbols to code at all */ - here.op = (unsigned char)64; /* invalid code marker */ - here.bits = (unsigned char)1; - here.val = (unsigned short)0; - *(*table)++ = here; /* make a table to force an error */ - *(*table)++ = here; - *bits = 1; - return 0; /* no symbols, but wait for decoding to report error */ - } - for (min = 1; min < max; min++) - if (count[min] != 0) break; - if (root < min) root = min; - - /* check for an over-subscribed or incomplete set of lengths */ - left = 1; - for (len = 1; len <= MAXBITS; len++) { - left <<= 1; - left -= count[len]; - if (left < 0) return -1; /* over-subscribed */ - } - if (left > 0 && (type == CODES || max != 1)) - return -1; /* incomplete set */ - - /* generate offsets into symbol table for each length for sorting */ - offs[1] = 0; - for (len = 1; len < MAXBITS; len++) - offs[len + 1] = offs[len] + count[len]; - - /* sort symbols by length, by symbol order within each length */ - for (sym = 0; sym < codes; sym++) - if (lens[sym] != 0) work[offs[lens[sym]]++] = (unsigned short)sym; - - /* - Create and fill in decoding tables. In this loop, the table being - filled is at next and has curr index bits. The code being used is huff - with length len. That code is converted to an index by dropping drop - bits off of the bottom. For codes where len is less than drop + curr, - those top drop + curr - len bits are incremented through all values to - fill the table with replicated entries. - - root is the number of index bits for the root table. When len exceeds - root, sub-tables are created pointed to by the root entry with an index - of the low root bits of huff. This is saved in low to check for when a - new sub-table should be started. drop is zero when the root table is - being filled, and drop is root when sub-tables are being filled. - - When a new sub-table is needed, it is necessary to look ahead in the - code lengths to determine what size sub-table is needed. The length - counts are used for this, and so count[] is decremented as codes are - entered in the tables. - - used keeps track of how many table entries have been allocated from the - provided *table space. It is checked for LENS and DIST tables against - the constants ENOUGH_LENS and ENOUGH_DISTS to guard against changes in - the initial root table size constants. See the comments in inftrees.h - for more information. - - sym increments through all symbols, and the loop terminates when - all codes of length max, i.e. all codes, have been processed. This - routine permits incomplete codes, so another loop after this one fills - in the rest of the decoding tables with invalid code markers. - */ - - /* set up for code type */ - switch (type) { - case CODES: - base = extra = work; /* dummy value--not used */ - end = 19; - break; - case LENS: - base = lbase; - base -= 257; - extra = lext; - extra -= 257; - end = 256; - break; - default: /* DISTS */ - base = dbase; - extra = dext; - end = -1; - } - - /* initialize state for loop */ - huff = 0; /* starting code */ - sym = 0; /* starting code symbol */ - len = min; /* starting code length */ - next = *table; /* current table to fill in */ - curr = root; /* current table index bits */ - drop = 0; /* current bits to drop from code for index */ - low = (unsigned)(-1); /* trigger new sub-table when len > root */ - used = 1U << root; /* use root table entries */ - mask = used - 1; /* mask for comparing low */ - - /* check available table space */ - if ((type == LENS && used >= ENOUGH_LENS) || - (type == DISTS && used >= ENOUGH_DISTS)) - return 1; - - /* process all codes and make table entries */ - for (;;) { - /* create table entry */ - here.bits = (unsigned char)(len - drop); - if ((int)(work[sym]) < end) { - here.op = (unsigned char)0; - here.val = work[sym]; - } - else if ((int)(work[sym]) > end) { - here.op = (unsigned char)(extra[work[sym]]); - here.val = base[work[sym]]; - } - else { - here.op = (unsigned char)(32 + 64); /* end of block */ - here.val = 0; - } - - /* replicate for those indices with low len bits equal to huff */ - incr = 1U << (len - drop); - fill = 1U << curr; - min = fill; /* save offset to next table */ - do { - fill -= incr; - next[(huff >> drop) + fill] = here; - } while (fill != 0); - - /* backwards increment the len-bit code huff */ - incr = 1U << (len - 1); - while (huff & incr) - incr >>= 1; - if (incr != 0) { - huff &= incr - 1; - huff += incr; - } - else - huff = 0; - - /* go to next symbol, update count, len */ - sym++; - if (--(count[len]) == 0) { - if (len == max) break; - len = lens[work[sym]]; - } - - /* create new sub-table if needed */ - if (len > root && (huff & mask) != low) { - /* if first time, transition to sub-tables */ - if (drop == 0) - drop = root; - - /* increment past last table */ - next += min; /* here min is 1 << curr */ - - /* determine length of next table */ - curr = len - drop; - left = (int)(1 << curr); - while (curr + drop < max) { - left -= count[curr + drop]; - if (left <= 0) break; - curr++; - left <<= 1; - } - - /* check for enough space */ - used += 1U << curr; - if ((type == LENS && used >= ENOUGH_LENS) || - (type == DISTS && used >= ENOUGH_DISTS)) - return 1; - - /* point entry in root table to sub-table */ - low = huff & mask; - (*table)[low].op = (unsigned char)curr; - (*table)[low].bits = (unsigned char)root; - (*table)[low].val = (unsigned short)(next - *table); - } - } - - /* - Fill in rest of table for incomplete codes. This loop is similar to the - loop above in incrementing huff for table indices. It is assumed that - len is equal to curr + drop, so there is no loop needed to increment - through high index bits. When the current sub-table is filled, the loop - drops back to the root table to fill in any remaining entries there. - */ - here.op = (unsigned char)64; /* invalid code marker */ - here.bits = (unsigned char)(len - drop); - here.val = (unsigned short)0; - while (huff != 0) { - /* when done with sub-table, drop back to root table */ - if (drop != 0 && (huff & mask) != low) { - drop = 0; - len = root; - next = *table; - here.bits = (unsigned char)len; - } - - /* put invalid code marker in table */ - next[huff >> drop] = here; - - /* backwards increment the len-bit code huff */ - incr = 1U << (len - 1); - while (huff & incr) - incr >>= 1; - if (incr != 0) { - huff &= incr - 1; - huff += incr; - } - else - huff = 0; - } - - /* set return parameters */ - *table += used; - *bits = root; - return 0; -} diff --git a/code/3rd_glpk/zlib/inftrees.h b/code/3rd_glpk/zlib/inftrees.h deleted file mode 100644 index baa53a0b..00000000 --- a/code/3rd_glpk/zlib/inftrees.h +++ /dev/null @@ -1,62 +0,0 @@ -/* inftrees.h -- header to use inftrees.c - * Copyright (C) 1995-2005, 2010 Mark Adler - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. - */ - -/* Structure for decoding tables. Each entry provides either the - information needed to do the operation requested by the code that - indexed that table entry, or it provides a pointer to another - table that indexes more bits of the code. op indicates whether - the entry is a pointer to another table, a literal, a length or - distance, an end-of-block, or an invalid code. For a table - pointer, the low four bits of op is the number of index bits of - that table. For a length or distance, the low four bits of op - is the number of extra bits to get after the code. bits is - the number of bits in this code or part of the code to drop off - of the bit buffer. val is the actual byte to output in the case - of a literal, the base length or distance, or the offset from - the current table to the next table. Each entry is four bytes. */ -typedef struct { - unsigned char op; /* operation, extra bits, table bits */ - unsigned char bits; /* bits in this part of the code */ - unsigned short val; /* offset in table or code value */ -} code; - -/* op values as set by inflate_table(): - 00000000 - literal - 0000tttt - table link, tttt != 0 is the number of table index bits - 0001eeee - length or distance, eeee is the number of extra bits - 01100000 - end of block - 01000000 - invalid code - */ - -/* Maximum size of the dynamic table. The maximum number of code structures is - 1444, which is the sum of 852 for literal/length codes and 592 for distance - codes. These values were found by exhaustive searches using the program - examples/enough.c found in the zlib distribtution. The arguments to that - program are the number of symbols, the initial root table size, and the - maximum bit length of a code. "enough 286 9 15" for literal/length codes - returns returns 852, and "enough 30 6 15" for distance codes returns 592. - The initial root table size (9 or 6) is found in the fifth argument of the - inflate_table() calls in inflate.c and infback.c. If the root table size is - changed, then these maximum sizes would be need to be recalculated and - updated. */ -#define ENOUGH_LENS 852 -#define ENOUGH_DISTS 592 -#define ENOUGH (ENOUGH_LENS+ENOUGH_DISTS) - -/* Type of code to build for inflate_table() */ -typedef enum { - CODES, - LENS, - DISTS -} codetype; - -int ZLIB_INTERNAL inflate_table OF((codetype type, unsigned short FAR *lens, - unsigned codes, code FAR * FAR *table, - unsigned FAR *bits, unsigned short FAR *work)); diff --git a/code/3rd_glpk/zlib/trees.c b/code/3rd_glpk/zlib/trees.c deleted file mode 100644 index 56e9bb1c..00000000 --- a/code/3rd_glpk/zlib/trees.c +++ /dev/null @@ -1,1244 +0,0 @@ -/* trees.c -- output deflated data using Huffman coding - * Copyright (C) 1995-2010 Jean-loup Gailly - * detect_data_type() function provided freely by Cosmin Truta, 2006 - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* - * ALGORITHM - * - * The "deflation" process uses several Huffman trees. The more - * common source values are represented by shorter bit sequences. - * - * Each code tree is stored in a compressed form which is itself - * a Huffman encoding of the lengths of all the code strings (in - * ascending order by source values). The actual code strings are - * reconstructed from the lengths in the inflate process, as described - * in the deflate specification. - * - * REFERENCES - * - * Deutsch, L.P.,"'Deflate' Compressed Data Format Specification". - * Available in ftp.uu.net:/pub/archiving/zip/doc/deflate-1.1.doc - * - * Storer, James A. - * Data Compression: Methods and Theory, pp. 49-50. - * Computer Science Press, 1988. ISBN 0-7167-8156-5. - * - * Sedgewick, R. - * Algorithms, p290. - * Addison-Wesley, 1983. ISBN 0-201-06672-6. - */ - -/* @(#) $Id$ */ - -/* #define GEN_TREES_H */ - -#include "deflate.h" - -#ifdef DEBUG -# include -#endif - -/* =========================================================================== - * Constants - */ - -#define MAX_BL_BITS 7 -/* Bit length codes must not exceed MAX_BL_BITS bits */ - -#define END_BLOCK 256 -/* end of block literal code */ - -#define REP_3_6 16 -/* repeat previous bit length 3-6 times (2 bits of repeat count) */ - -#define REPZ_3_10 17 -/* repeat a zero length 3-10 times (3 bits of repeat count) */ - -#define REPZ_11_138 18 -/* repeat a zero length 11-138 times (7 bits of repeat count) */ - -local const int extra_lbits[LENGTH_CODES] /* extra bits for each length code */ - = {0,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,0}; - -local const int extra_dbits[D_CODES] /* extra bits for each distance code */ - = {0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13}; - -local const int extra_blbits[BL_CODES]/* extra bits for each bit length code */ - = {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,2,3,7}; - -local const uch bl_order[BL_CODES] - = {16,17,18,0,8,7,9,6,10,5,11,4,12,3,13,2,14,1,15}; -/* The lengths of the bit length codes are sent in order of decreasing - * probability, to avoid transmitting the lengths for unused bit length codes. - */ - -#define Buf_size (8 * 2*sizeof(char)) -/* Number of bits used within bi_buf. (bi_buf might be implemented on - * more than 16 bits on some systems.) - */ - -/* =========================================================================== - * Local data. These are initialized only once. - */ - -#define DIST_CODE_LEN 512 /* see definition of array dist_code below */ - -#if defined(GEN_TREES_H) || !defined(STDC) -/* non ANSI compilers may not accept trees.h */ - -local ct_data static_ltree[L_CODES+2]; -/* The static literal tree. Since the bit lengths are imposed, there is no - * need for the L_CODES extra codes used during heap construction. However - * The codes 286 and 287 are needed to build a canonical tree (see _tr_init - * below). - */ - -local ct_data static_dtree[D_CODES]; -/* The static distance tree. (Actually a trivial tree since all codes use - * 5 bits.) - */ - -uch _dist_code[DIST_CODE_LEN]; -/* Distance codes. The first 256 values correspond to the distances - * 3 .. 258, the last 256 values correspond to the top 8 bits of - * the 15 bit distances. - */ - -uch _length_code[MAX_MATCH-MIN_MATCH+1]; -/* length code for each normalized match length (0 == MIN_MATCH) */ - -local int base_length[LENGTH_CODES]; -/* First normalized length for each code (0 = MIN_MATCH) */ - -local int base_dist[D_CODES]; -/* First normalized distance for each code (0 = distance of 1) */ - -#else -# include "trees.h" -#endif /* GEN_TREES_H */ - -struct static_tree_desc_s { - const ct_data *static_tree; /* static tree or NULL */ - const intf *extra_bits; /* extra bits for each code or NULL */ - int extra_base; /* base index for extra_bits */ - int elems; /* max number of elements in the tree */ - int max_length; /* max bit length for the codes */ -}; - -local static_tree_desc static_l_desc = -{static_ltree, extra_lbits, LITERALS+1, L_CODES, MAX_BITS}; - -local static_tree_desc static_d_desc = -{static_dtree, extra_dbits, 0, D_CODES, MAX_BITS}; - -local static_tree_desc static_bl_desc = -{(const ct_data *)0, extra_blbits, 0, BL_CODES, MAX_BL_BITS}; - -/* =========================================================================== - * Local (static) routines in this file. - */ - -local void tr_static_init OF((void)); -local void init_block OF((deflate_state *s)); -local void pqdownheap OF((deflate_state *s, ct_data *tree, int k)); -local void gen_bitlen OF((deflate_state *s, tree_desc *desc)); -local void gen_codes OF((ct_data *tree, int max_code, ushf *bl_count)); -local void build_tree OF((deflate_state *s, tree_desc *desc)); -local void scan_tree OF((deflate_state *s, ct_data *tree, int max_code)); -local void send_tree OF((deflate_state *s, ct_data *tree, int max_code)); -local int build_bl_tree OF((deflate_state *s)); -local void send_all_trees OF((deflate_state *s, int lcodes, int dcodes, - int blcodes)); -local void compress_block OF((deflate_state *s, ct_data *ltree, - ct_data *dtree)); -local int detect_data_type OF((deflate_state *s)); -local unsigned bi_reverse OF((unsigned value, int length)); -local void bi_windup OF((deflate_state *s)); -local void bi_flush OF((deflate_state *s)); -local void copy_block OF((deflate_state *s, charf *buf, unsigned len, - int header)); - -#ifdef GEN_TREES_H -local void gen_trees_header OF((void)); -#endif - -#ifndef DEBUG -# define send_code(s, c, tree) send_bits(s, tree[c].Code, tree[c].Len) - /* Send a code of the given tree. c and tree must not have side effects */ - -#else /* DEBUG */ -# define send_code(s, c, tree) \ - { if (z_verbose>2) fprintf(stderr,"\ncd %3d ",(c)); \ - send_bits(s, tree[c].Code, tree[c].Len); } -#endif - -/* =========================================================================== - * Output a short LSB first on the stream. - * IN assertion: there is enough room in pendingBuf. - */ -#define put_short(s, w) { \ - put_byte(s, (uch)((w) & 0xff)); \ - put_byte(s, (uch)((ush)(w) >> 8)); \ -} - -/* =========================================================================== - * Send a value on a given number of bits. - * IN assertion: length <= 16 and value fits in length bits. - */ -#ifdef DEBUG -local void send_bits OF((deflate_state *s, int value, int length)); - -local void send_bits(s, value, length) - deflate_state *s; - int value; /* value to send */ - int length; /* number of bits */ -{ - Tracevv((stderr," l %2d v %4x ", length, value)); - Assert(length > 0 && length <= 15, "invalid length"); - s->bits_sent += (ulg)length; - - /* If not enough room in bi_buf, use (valid) bits from bi_buf and - * (16 - bi_valid) bits from value, leaving (width - (16-bi_valid)) - * unused bits in value. - */ - if (s->bi_valid > (int)Buf_size - length) { - s->bi_buf |= (ush)value << s->bi_valid; - put_short(s, s->bi_buf); - s->bi_buf = (ush)value >> (Buf_size - s->bi_valid); - s->bi_valid += length - Buf_size; - } else { - s->bi_buf |= (ush)value << s->bi_valid; - s->bi_valid += length; - } -} -#else /* !DEBUG */ - -#define send_bits(s, value, length) \ -{ int len = length;\ - if (s->bi_valid > (int)Buf_size - len) {\ - int val = value;\ - s->bi_buf |= (ush)val << s->bi_valid;\ - put_short(s, s->bi_buf);\ - s->bi_buf = (ush)val >> (Buf_size - s->bi_valid);\ - s->bi_valid += len - Buf_size;\ - } else {\ - s->bi_buf |= (ush)(value) << s->bi_valid;\ - s->bi_valid += len;\ - }\ -} -#endif /* DEBUG */ - - -/* the arguments must not have side effects */ - -/* =========================================================================== - * Initialize the various 'constant' tables. - */ -local void tr_static_init() -{ -#if defined(GEN_TREES_H) || !defined(STDC) - static int static_init_done = 0; - int n; /* iterates over tree elements */ - int bits; /* bit counter */ - int length; /* length value */ - int code; /* code value */ - int dist; /* distance index */ - ush bl_count[MAX_BITS+1]; - /* number of codes at each bit length for an optimal tree */ - - if (static_init_done) return; - - /* For some embedded targets, global variables are not initialized: */ -#ifdef NO_INIT_GLOBAL_POINTERS - static_l_desc.static_tree = static_ltree; - static_l_desc.extra_bits = extra_lbits; - static_d_desc.static_tree = static_dtree; - static_d_desc.extra_bits = extra_dbits; - static_bl_desc.extra_bits = extra_blbits; -#endif - - /* Initialize the mapping length (0..255) -> length code (0..28) */ - length = 0; - for (code = 0; code < LENGTH_CODES-1; code++) { - base_length[code] = length; - for (n = 0; n < (1< dist code (0..29) */ - dist = 0; - for (code = 0 ; code < 16; code++) { - base_dist[code] = dist; - for (n = 0; n < (1<>= 7; /* from now on, all distances are divided by 128 */ - for ( ; code < D_CODES; code++) { - base_dist[code] = dist << 7; - for (n = 0; n < (1<<(extra_dbits[code]-7)); n++) { - _dist_code[256 + dist++] = (uch)code; - } - } - Assert (dist == 256, "tr_static_init: 256+dist != 512"); - - /* Construct the codes of the static literal tree */ - for (bits = 0; bits <= MAX_BITS; bits++) bl_count[bits] = 0; - n = 0; - while (n <= 143) static_ltree[n++].Len = 8, bl_count[8]++; - while (n <= 255) static_ltree[n++].Len = 9, bl_count[9]++; - while (n <= 279) static_ltree[n++].Len = 7, bl_count[7]++; - while (n <= 287) static_ltree[n++].Len = 8, bl_count[8]++; - /* Codes 286 and 287 do not exist, but we must include them in the - * tree construction to get a canonical Huffman tree (longest code - * all ones) - */ - gen_codes((ct_data *)static_ltree, L_CODES+1, bl_count); - - /* The static distance tree is trivial: */ - for (n = 0; n < D_CODES; n++) { - static_dtree[n].Len = 5; - static_dtree[n].Code = bi_reverse((unsigned)n, 5); - } - static_init_done = 1; - -# ifdef GEN_TREES_H - gen_trees_header(); -# endif -#endif /* defined(GEN_TREES_H) || !defined(STDC) */ -} - -/* =========================================================================== - * Genererate the file trees.h describing the static trees. - */ -#ifdef GEN_TREES_H -# ifndef DEBUG -# include -# endif - -# define SEPARATOR(i, last, width) \ - ((i) == (last)? "\n};\n\n" : \ - ((i) % (width) == (width)-1 ? ",\n" : ", ")) - -void gen_trees_header() -{ - FILE *header = fopen("trees.h", "w"); - int i; - - Assert (header != NULL, "Can't open trees.h"); - fprintf(header, - "/* header created automatically with -DGEN_TREES_H */\n\n"); - - fprintf(header, "local const ct_data static_ltree[L_CODES+2] = {\n"); - for (i = 0; i < L_CODES+2; i++) { - fprintf(header, "{{%3u},{%3u}}%s", static_ltree[i].Code, - static_ltree[i].Len, SEPARATOR(i, L_CODES+1, 5)); - } - - fprintf(header, "local const ct_data static_dtree[D_CODES] = {\n"); - for (i = 0; i < D_CODES; i++) { - fprintf(header, "{{%2u},{%2u}}%s", static_dtree[i].Code, - static_dtree[i].Len, SEPARATOR(i, D_CODES-1, 5)); - } - - fprintf(header, "const uch ZLIB_INTERNAL _dist_code[DIST_CODE_LEN] = {\n"); - for (i = 0; i < DIST_CODE_LEN; i++) { - fprintf(header, "%2u%s", _dist_code[i], - SEPARATOR(i, DIST_CODE_LEN-1, 20)); - } - - fprintf(header, - "const uch ZLIB_INTERNAL _length_code[MAX_MATCH-MIN_MATCH+1]= {\n"); - for (i = 0; i < MAX_MATCH-MIN_MATCH+1; i++) { - fprintf(header, "%2u%s", _length_code[i], - SEPARATOR(i, MAX_MATCH-MIN_MATCH, 20)); - } - - fprintf(header, "local const int base_length[LENGTH_CODES] = {\n"); - for (i = 0; i < LENGTH_CODES; i++) { - fprintf(header, "%1u%s", base_length[i], - SEPARATOR(i, LENGTH_CODES-1, 20)); - } - - fprintf(header, "local const int base_dist[D_CODES] = {\n"); - for (i = 0; i < D_CODES; i++) { - fprintf(header, "%5u%s", base_dist[i], - SEPARATOR(i, D_CODES-1, 10)); - } - - fclose(header); -} -#endif /* GEN_TREES_H */ - -/* =========================================================================== - * Initialize the tree data structures for a new zlib stream. - */ -void ZLIB_INTERNAL _tr_init(s) - deflate_state *s; -{ - tr_static_init(); - - s->l_desc.dyn_tree = s->dyn_ltree; - s->l_desc.stat_desc = &static_l_desc; - - s->d_desc.dyn_tree = s->dyn_dtree; - s->d_desc.stat_desc = &static_d_desc; - - s->bl_desc.dyn_tree = s->bl_tree; - s->bl_desc.stat_desc = &static_bl_desc; - - s->bi_buf = 0; - s->bi_valid = 0; - s->last_eob_len = 8; /* enough lookahead for inflate */ -#ifdef DEBUG - s->compressed_len = 0L; - s->bits_sent = 0L; -#endif - - /* Initialize the first block of the first file: */ - init_block(s); -} - -/* =========================================================================== - * Initialize a new block. - */ -local void init_block(s) - deflate_state *s; -{ - int n; /* iterates over tree elements */ - - /* Initialize the trees. */ - for (n = 0; n < L_CODES; n++) s->dyn_ltree[n].Freq = 0; - for (n = 0; n < D_CODES; n++) s->dyn_dtree[n].Freq = 0; - for (n = 0; n < BL_CODES; n++) s->bl_tree[n].Freq = 0; - - s->dyn_ltree[END_BLOCK].Freq = 1; - s->opt_len = s->static_len = 0L; - s->last_lit = s->matches = 0; -} - -#define SMALLEST 1 -/* Index within the heap array of least frequent node in the Huffman tree */ - - -/* =========================================================================== - * Remove the smallest element from the heap and recreate the heap with - * one less element. Updates heap and heap_len. - */ -#define pqremove(s, tree, top) \ -{\ - top = s->heap[SMALLEST]; \ - s->heap[SMALLEST] = s->heap[s->heap_len--]; \ - pqdownheap(s, tree, SMALLEST); \ -} - -/* =========================================================================== - * Compares to subtrees, using the tree depth as tie breaker when - * the subtrees have equal frequency. This minimizes the worst case length. - */ -#define smaller(tree, n, m, depth) \ - (tree[n].Freq < tree[m].Freq || \ - (tree[n].Freq == tree[m].Freq && depth[n] <= depth[m])) - -/* =========================================================================== - * Restore the heap property by moving down the tree starting at node k, - * exchanging a node with the smallest of its two sons if necessary, stopping - * when the heap property is re-established (each father smaller than its - * two sons). - */ -local void pqdownheap(s, tree, k) - deflate_state *s; - ct_data *tree; /* the tree to restore */ - int k; /* node to move down */ -{ - int v = s->heap[k]; - int j = k << 1; /* left son of k */ - while (j <= s->heap_len) { - /* Set j to the smallest of the two sons: */ - if (j < s->heap_len && - smaller(tree, s->heap[j+1], s->heap[j], s->depth)) { - j++; - } - /* Exit if v is smaller than both sons */ - if (smaller(tree, v, s->heap[j], s->depth)) break; - - /* Exchange v with the smallest son */ - s->heap[k] = s->heap[j]; k = j; - - /* And continue down the tree, setting j to the left son of k */ - j <<= 1; - } - s->heap[k] = v; -} - -/* =========================================================================== - * Compute the optimal bit lengths for a tree and update the total bit length - * for the current block. - * IN assertion: the fields freq and dad are set, heap[heap_max] and - * above are the tree nodes sorted by increasing frequency. - * OUT assertions: the field len is set to the optimal bit length, the - * array bl_count contains the frequencies for each bit length. - * The length opt_len is updated; static_len is also updated if stree is - * not null. - */ -local void gen_bitlen(s, desc) - deflate_state *s; - tree_desc *desc; /* the tree descriptor */ -{ - ct_data *tree = desc->dyn_tree; - int max_code = desc->max_code; - const ct_data *stree = desc->stat_desc->static_tree; - const intf *extra = desc->stat_desc->extra_bits; - int base = desc->stat_desc->extra_base; - int max_length = desc->stat_desc->max_length; - int h; /* heap index */ - int n, m; /* iterate over the tree elements */ - int bits; /* bit length */ - int xbits; /* extra bits */ - ush f; /* frequency */ - int overflow = 0; /* number of elements with bit length too large */ - - for (bits = 0; bits <= MAX_BITS; bits++) s->bl_count[bits] = 0; - - /* In a first pass, compute the optimal bit lengths (which may - * overflow in the case of the bit length tree). - */ - tree[s->heap[s->heap_max]].Len = 0; /* root of the heap */ - - for (h = s->heap_max+1; h < HEAP_SIZE; h++) { - n = s->heap[h]; - bits = tree[tree[n].Dad].Len + 1; - if (bits > max_length) bits = max_length, overflow++; - tree[n].Len = (ush)bits; - /* We overwrite tree[n].Dad which is no longer needed */ - - if (n > max_code) continue; /* not a leaf node */ - - s->bl_count[bits]++; - xbits = 0; - if (n >= base) xbits = extra[n-base]; - f = tree[n].Freq; - s->opt_len += (ulg)f * (bits + xbits); - if (stree) s->static_len += (ulg)f * (stree[n].Len + xbits); - } - if (overflow == 0) return; - - Trace((stderr,"\nbit length overflow\n")); - /* This happens for example on obj2 and pic of the Calgary corpus */ - - /* Find the first bit length which could increase: */ - do { - bits = max_length-1; - while (s->bl_count[bits] == 0) bits--; - s->bl_count[bits]--; /* move one leaf down the tree */ - s->bl_count[bits+1] += 2; /* move one overflow item as its brother */ - s->bl_count[max_length]--; - /* The brother of the overflow item also moves one step up, - * but this does not affect bl_count[max_length] - */ - overflow -= 2; - } while (overflow > 0); - - /* Now recompute all bit lengths, scanning in increasing frequency. - * h is still equal to HEAP_SIZE. (It is simpler to reconstruct all - * lengths instead of fixing only the wrong ones. This idea is taken - * from 'ar' written by Haruhiko Okumura.) - */ - for (bits = max_length; bits != 0; bits--) { - n = s->bl_count[bits]; - while (n != 0) { - m = s->heap[--h]; - if (m > max_code) continue; - if ((unsigned) tree[m].Len != (unsigned) bits) { - Trace((stderr,"code %d bits %d->%d\n", m, tree[m].Len, bits)); - s->opt_len += ((long)bits - (long)tree[m].Len) - *(long)tree[m].Freq; - tree[m].Len = (ush)bits; - } - n--; - } - } -} - -/* =========================================================================== - * Generate the codes for a given tree and bit counts (which need not be - * optimal). - * IN assertion: the array bl_count contains the bit length statistics for - * the given tree and the field len is set for all tree elements. - * OUT assertion: the field code is set for all tree elements of non - * zero code length. - */ -local void gen_codes (tree, max_code, bl_count) - ct_data *tree; /* the tree to decorate */ - int max_code; /* largest code with non zero frequency */ - ushf *bl_count; /* number of codes at each bit length */ -{ - ush next_code[MAX_BITS+1]; /* next code value for each bit length */ - ush code = 0; /* running code value */ - int bits; /* bit index */ - int n; /* code index */ - - /* The distribution counts are first used to generate the code values - * without bit reversal. - */ - for (bits = 1; bits <= MAX_BITS; bits++) { - next_code[bits] = code = (code + bl_count[bits-1]) << 1; - } - /* Check that the bit counts in bl_count are consistent. The last code - * must be all ones. - */ - Assert (code + bl_count[MAX_BITS]-1 == (1<dyn_tree; - const ct_data *stree = desc->stat_desc->static_tree; - int elems = desc->stat_desc->elems; - int n, m; /* iterate over heap elements */ - int max_code = -1; /* largest code with non zero frequency */ - int node; /* new node being created */ - - /* Construct the initial heap, with least frequent element in - * heap[SMALLEST]. The sons of heap[n] are heap[2*n] and heap[2*n+1]. - * heap[0] is not used. - */ - s->heap_len = 0, s->heap_max = HEAP_SIZE; - - for (n = 0; n < elems; n++) { - if (tree[n].Freq != 0) { - s->heap[++(s->heap_len)] = max_code = n; - s->depth[n] = 0; - } else { - tree[n].Len = 0; - } - } - - /* The pkzip format requires that at least one distance code exists, - * and that at least one bit should be sent even if there is only one - * possible code. So to avoid special checks later on we force at least - * two codes of non zero frequency. - */ - while (s->heap_len < 2) { - node = s->heap[++(s->heap_len)] = (max_code < 2 ? ++max_code : 0); - tree[node].Freq = 1; - s->depth[node] = 0; - s->opt_len--; if (stree) s->static_len -= stree[node].Len; - /* node is 0 or 1 so it does not have extra bits */ - } - desc->max_code = max_code; - - /* The elements heap[heap_len/2+1 .. heap_len] are leaves of the tree, - * establish sub-heaps of increasing lengths: - */ - for (n = s->heap_len/2; n >= 1; n--) pqdownheap(s, tree, n); - - /* Construct the Huffman tree by repeatedly combining the least two - * frequent nodes. - */ - node = elems; /* next internal node of the tree */ - do { - pqremove(s, tree, n); /* n = node of least frequency */ - m = s->heap[SMALLEST]; /* m = node of next least frequency */ - - s->heap[--(s->heap_max)] = n; /* keep the nodes sorted by frequency */ - s->heap[--(s->heap_max)] = m; - - /* Create a new node father of n and m */ - tree[node].Freq = tree[n].Freq + tree[m].Freq; - s->depth[node] = (uch)((s->depth[n] >= s->depth[m] ? - s->depth[n] : s->depth[m]) + 1); - tree[n].Dad = tree[m].Dad = (ush)node; -#ifdef DUMP_BL_TREE - if (tree == s->bl_tree) { - fprintf(stderr,"\nnode %d(%d), sons %d(%d) %d(%d)", - node, tree[node].Freq, n, tree[n].Freq, m, tree[m].Freq); - } -#endif - /* and insert the new node in the heap */ - s->heap[SMALLEST] = node++; - pqdownheap(s, tree, SMALLEST); - - } while (s->heap_len >= 2); - - s->heap[--(s->heap_max)] = s->heap[SMALLEST]; - - /* At this point, the fields freq and dad are set. We can now - * generate the bit lengths. - */ - gen_bitlen(s, (tree_desc *)desc); - - /* The field len is now set, we can generate the bit codes */ - gen_codes ((ct_data *)tree, max_code, s->bl_count); -} - -/* =========================================================================== - * Scan a literal or distance tree to determine the frequencies of the codes - * in the bit length tree. - */ -local void scan_tree (s, tree, max_code) - deflate_state *s; - ct_data *tree; /* the tree to be scanned */ - int max_code; /* and its largest code of non zero frequency */ -{ - int n; /* iterates over all tree elements */ - int prevlen = -1; /* last emitted length */ - int curlen; /* length of current code */ - int nextlen = tree[0].Len; /* length of next code */ - int count = 0; /* repeat count of the current code */ - int max_count = 7; /* max repeat count */ - int min_count = 4; /* min repeat count */ - - if (nextlen == 0) max_count = 138, min_count = 3; - tree[max_code+1].Len = (ush)0xffff; /* guard */ - - for (n = 0; n <= max_code; n++) { - curlen = nextlen; nextlen = tree[n+1].Len; - if (++count < max_count && curlen == nextlen) { - continue; - } else if (count < min_count) { - s->bl_tree[curlen].Freq += count; - } else if (curlen != 0) { - if (curlen != prevlen) s->bl_tree[curlen].Freq++; - s->bl_tree[REP_3_6].Freq++; - } else if (count <= 10) { - s->bl_tree[REPZ_3_10].Freq++; - } else { - s->bl_tree[REPZ_11_138].Freq++; - } - count = 0; prevlen = curlen; - if (nextlen == 0) { - max_count = 138, min_count = 3; - } else if (curlen == nextlen) { - max_count = 6, min_count = 3; - } else { - max_count = 7, min_count = 4; - } - } -} - -/* =========================================================================== - * Send a literal or distance tree in compressed form, using the codes in - * bl_tree. - */ -local void send_tree (s, tree, max_code) - deflate_state *s; - ct_data *tree; /* the tree to be scanned */ - int max_code; /* and its largest code of non zero frequency */ -{ - int n; /* iterates over all tree elements */ - int prevlen = -1; /* last emitted length */ - int curlen; /* length of current code */ - int nextlen = tree[0].Len; /* length of next code */ - int count = 0; /* repeat count of the current code */ - int max_count = 7; /* max repeat count */ - int min_count = 4; /* min repeat count */ - - /* tree[max_code+1].Len = -1; */ /* guard already set */ - if (nextlen == 0) max_count = 138, min_count = 3; - - for (n = 0; n <= max_code; n++) { - curlen = nextlen; nextlen = tree[n+1].Len; - if (++count < max_count && curlen == nextlen) { - continue; - } else if (count < min_count) { - do { send_code(s, curlen, s->bl_tree); } while (--count != 0); - - } else if (curlen != 0) { - if (curlen != prevlen) { - send_code(s, curlen, s->bl_tree); count--; - } - Assert(count >= 3 && count <= 6, " 3_6?"); - send_code(s, REP_3_6, s->bl_tree); send_bits(s, count-3, 2); - - } else if (count <= 10) { - send_code(s, REPZ_3_10, s->bl_tree); send_bits(s, count-3, 3); - - } else { - send_code(s, REPZ_11_138, s->bl_tree); send_bits(s, count-11, 7); - } - count = 0; prevlen = curlen; - if (nextlen == 0) { - max_count = 138, min_count = 3; - } else if (curlen == nextlen) { - max_count = 6, min_count = 3; - } else { - max_count = 7, min_count = 4; - } - } -} - -/* =========================================================================== - * Construct the Huffman tree for the bit lengths and return the index in - * bl_order of the last bit length code to send. - */ -local int build_bl_tree(s) - deflate_state *s; -{ - int max_blindex; /* index of last bit length code of non zero freq */ - - /* Determine the bit length frequencies for literal and distance trees */ - scan_tree(s, (ct_data *)s->dyn_ltree, s->l_desc.max_code); - scan_tree(s, (ct_data *)s->dyn_dtree, s->d_desc.max_code); - - /* Build the bit length tree: */ - build_tree(s, (tree_desc *)(&(s->bl_desc))); - /* opt_len now includes the length of the tree representations, except - * the lengths of the bit lengths codes and the 5+5+4 bits for the counts. - */ - - /* Determine the number of bit length codes to send. The pkzip format - * requires that at least 4 bit length codes be sent. (appnote.txt says - * 3 but the actual value used is 4.) - */ - for (max_blindex = BL_CODES-1; max_blindex >= 3; max_blindex--) { - if (s->bl_tree[bl_order[max_blindex]].Len != 0) break; - } - /* Update opt_len to include the bit length tree and counts */ - s->opt_len += 3*(max_blindex+1) + 5+5+4; - Tracev((stderr, "\ndyn trees: dyn %ld, stat %ld", - s->opt_len, s->static_len)); - - return max_blindex; -} - -/* =========================================================================== - * Send the header for a block using dynamic Huffman trees: the counts, the - * lengths of the bit length codes, the literal tree and the distance tree. - * IN assertion: lcodes >= 257, dcodes >= 1, blcodes >= 4. - */ -local void send_all_trees(s, lcodes, dcodes, blcodes) - deflate_state *s; - int lcodes, dcodes, blcodes; /* number of codes for each tree */ -{ - int rank; /* index in bl_order */ - - Assert (lcodes >= 257 && dcodes >= 1 && blcodes >= 4, "not enough codes"); - Assert (lcodes <= L_CODES && dcodes <= D_CODES && blcodes <= BL_CODES, - "too many codes"); - Tracev((stderr, "\nbl counts: ")); - send_bits(s, lcodes-257, 5); /* not +255 as stated in appnote.txt */ - send_bits(s, dcodes-1, 5); - send_bits(s, blcodes-4, 4); /* not -3 as stated in appnote.txt */ - for (rank = 0; rank < blcodes; rank++) { - Tracev((stderr, "\nbl code %2d ", bl_order[rank])); - send_bits(s, s->bl_tree[bl_order[rank]].Len, 3); - } - Tracev((stderr, "\nbl tree: sent %ld", s->bits_sent)); - - send_tree(s, (ct_data *)s->dyn_ltree, lcodes-1); /* literal tree */ - Tracev((stderr, "\nlit tree: sent %ld", s->bits_sent)); - - send_tree(s, (ct_data *)s->dyn_dtree, dcodes-1); /* distance tree */ - Tracev((stderr, "\ndist tree: sent %ld", s->bits_sent)); -} - -/* =========================================================================== - * Send a stored block - */ -void ZLIB_INTERNAL _tr_stored_block(s, buf, stored_len, last) - deflate_state *s; - charf *buf; /* input block */ - ulg stored_len; /* length of input block */ - int last; /* one if this is the last block for a file */ -{ - send_bits(s, (STORED_BLOCK<<1)+last, 3); /* send block type */ -#ifdef DEBUG - s->compressed_len = (s->compressed_len + 3 + 7) & (ulg)~7L; - s->compressed_len += (stored_len + 4) << 3; -#endif - copy_block(s, buf, (unsigned)stored_len, 1); /* with header */ -} - -/* =========================================================================== - * Send one empty static block to give enough lookahead for inflate. - * This takes 10 bits, of which 7 may remain in the bit buffer. - * The current inflate code requires 9 bits of lookahead. If the - * last two codes for the previous block (real code plus EOB) were coded - * on 5 bits or less, inflate may have only 5+3 bits of lookahead to decode - * the last real code. In this case we send two empty static blocks instead - * of one. (There are no problems if the previous block is stored or fixed.) - * To simplify the code, we assume the worst case of last real code encoded - * on one bit only. - */ -void ZLIB_INTERNAL _tr_align(s) - deflate_state *s; -{ - send_bits(s, STATIC_TREES<<1, 3); - send_code(s, END_BLOCK, static_ltree); -#ifdef DEBUG - s->compressed_len += 10L; /* 3 for block type, 7 for EOB */ -#endif - bi_flush(s); - /* Of the 10 bits for the empty block, we have already sent - * (10 - bi_valid) bits. The lookahead for the last real code (before - * the EOB of the previous block) was thus at least one plus the length - * of the EOB plus what we have just sent of the empty static block. - */ - if (1 + s->last_eob_len + 10 - s->bi_valid < 9) { - send_bits(s, STATIC_TREES<<1, 3); - send_code(s, END_BLOCK, static_ltree); -#ifdef DEBUG - s->compressed_len += 10L; -#endif - bi_flush(s); - } - s->last_eob_len = 7; -} - -/* =========================================================================== - * Determine the best encoding for the current block: dynamic trees, static - * trees or store, and output the encoded block to the zip file. - */ -void ZLIB_INTERNAL _tr_flush_block(s, buf, stored_len, last) - deflate_state *s; - charf *buf; /* input block, or NULL if too old */ - ulg stored_len; /* length of input block */ - int last; /* one if this is the last block for a file */ -{ - ulg opt_lenb, static_lenb; /* opt_len and static_len in bytes */ - int max_blindex = 0; /* index of last bit length code of non zero freq */ - - /* Build the Huffman trees unless a stored block is forced */ - if (s->level > 0) { - - /* Check if the file is binary or text */ - if (s->strm->data_type == Z_UNKNOWN) - s->strm->data_type = detect_data_type(s); - - /* Construct the literal and distance trees */ - build_tree(s, (tree_desc *)(&(s->l_desc))); - Tracev((stderr, "\nlit data: dyn %ld, stat %ld", s->opt_len, - s->static_len)); - - build_tree(s, (tree_desc *)(&(s->d_desc))); - Tracev((stderr, "\ndist data: dyn %ld, stat %ld", s->opt_len, - s->static_len)); - /* At this point, opt_len and static_len are the total bit lengths of - * the compressed block data, excluding the tree representations. - */ - - /* Build the bit length tree for the above two trees, and get the index - * in bl_order of the last bit length code to send. - */ - max_blindex = build_bl_tree(s); - - /* Determine the best encoding. Compute the block lengths in bytes. */ - opt_lenb = (s->opt_len+3+7)>>3; - static_lenb = (s->static_len+3+7)>>3; - - Tracev((stderr, "\nopt %lu(%lu) stat %lu(%lu) stored %lu lit %u ", - opt_lenb, s->opt_len, static_lenb, s->static_len, stored_len, - s->last_lit)); - - if (static_lenb <= opt_lenb) opt_lenb = static_lenb; - - } else { - Assert(buf != (char*)0, "lost buf"); - opt_lenb = static_lenb = stored_len + 5; /* force a stored block */ - } - -#ifdef FORCE_STORED - if (buf != (char*)0) { /* force stored block */ -#else - if (stored_len+4 <= opt_lenb && buf != (char*)0) { - /* 4: two words for the lengths */ -#endif - /* The test buf != NULL is only necessary if LIT_BUFSIZE > WSIZE. - * Otherwise we can't have processed more than WSIZE input bytes since - * the last block flush, because compression would have been - * successful. If LIT_BUFSIZE <= WSIZE, it is never too late to - * transform a block into a stored block. - */ - _tr_stored_block(s, buf, stored_len, last); - -#ifdef FORCE_STATIC - } else if (static_lenb >= 0) { /* force static trees */ -#else - } else if (s->strategy == Z_FIXED || static_lenb == opt_lenb) { -#endif - send_bits(s, (STATIC_TREES<<1)+last, 3); - compress_block(s, (ct_data *)static_ltree, (ct_data *)static_dtree); -#ifdef DEBUG - s->compressed_len += 3 + s->static_len; -#endif - } else { - send_bits(s, (DYN_TREES<<1)+last, 3); - send_all_trees(s, s->l_desc.max_code+1, s->d_desc.max_code+1, - max_blindex+1); - compress_block(s, (ct_data *)s->dyn_ltree, (ct_data *)s->dyn_dtree); -#ifdef DEBUG - s->compressed_len += 3 + s->opt_len; -#endif - } - Assert (s->compressed_len == s->bits_sent, "bad compressed size"); - /* The above check is made mod 2^32, for files larger than 512 MB - * and uLong implemented on 32 bits. - */ - init_block(s); - - if (last) { - bi_windup(s); -#ifdef DEBUG - s->compressed_len += 7; /* align on byte boundary */ -#endif - } - Tracev((stderr,"\ncomprlen %lu(%lu) ", s->compressed_len>>3, - s->compressed_len-7*last)); -} - -/* =========================================================================== - * Save the match info and tally the frequency counts. Return true if - * the current block must be flushed. - */ -int ZLIB_INTERNAL _tr_tally (s, dist, lc) - deflate_state *s; - unsigned dist; /* distance of matched string */ - unsigned lc; /* match length-MIN_MATCH or unmatched char (if dist==0) */ -{ - s->d_buf[s->last_lit] = (ush)dist; - s->l_buf[s->last_lit++] = (uch)lc; - if (dist == 0) { - /* lc is the unmatched char */ - s->dyn_ltree[lc].Freq++; - } else { - s->matches++; - /* Here, lc is the match length - MIN_MATCH */ - dist--; /* dist = match distance - 1 */ - Assert((ush)dist < (ush)MAX_DIST(s) && - (ush)lc <= (ush)(MAX_MATCH-MIN_MATCH) && - (ush)d_code(dist) < (ush)D_CODES, "_tr_tally: bad match"); - - s->dyn_ltree[_length_code[lc]+LITERALS+1].Freq++; - s->dyn_dtree[d_code(dist)].Freq++; - } - -#ifdef TRUNCATE_BLOCK - /* Try to guess if it is profitable to stop the current block here */ - if ((s->last_lit & 0x1fff) == 0 && s->level > 2) { - /* Compute an upper bound for the compressed length */ - ulg out_length = (ulg)s->last_lit*8L; - ulg in_length = (ulg)((long)s->strstart - s->block_start); - int dcode; - for (dcode = 0; dcode < D_CODES; dcode++) { - out_length += (ulg)s->dyn_dtree[dcode].Freq * - (5L+extra_dbits[dcode]); - } - out_length >>= 3; - Tracev((stderr,"\nlast_lit %u, in %ld, out ~%ld(%ld%%) ", - s->last_lit, in_length, out_length, - 100L - out_length*100L/in_length)); - if (s->matches < s->last_lit/2 && out_length < in_length/2) return 1; - } -#endif - return (s->last_lit == s->lit_bufsize-1); - /* We avoid equality with lit_bufsize because of wraparound at 64K - * on 16 bit machines and because stored blocks are restricted to - * 64K-1 bytes. - */ -} - -/* =========================================================================== - * Send the block data compressed using the given Huffman trees - */ -local void compress_block(s, ltree, dtree) - deflate_state *s; - ct_data *ltree; /* literal tree */ - ct_data *dtree; /* distance tree */ -{ - unsigned dist; /* distance of matched string */ - int lc; /* match length or unmatched char (if dist == 0) */ - unsigned lx = 0; /* running index in l_buf */ - unsigned code; /* the code to send */ - int extra; /* number of extra bits to send */ - - if (s->last_lit != 0) do { - dist = s->d_buf[lx]; - lc = s->l_buf[lx++]; - if (dist == 0) { - send_code(s, lc, ltree); /* send a literal byte */ - Tracecv(isgraph(lc), (stderr," '%c' ", lc)); - } else { - /* Here, lc is the match length - MIN_MATCH */ - code = _length_code[lc]; - send_code(s, code+LITERALS+1, ltree); /* send the length code */ - extra = extra_lbits[code]; - if (extra != 0) { - lc -= base_length[code]; - send_bits(s, lc, extra); /* send the extra length bits */ - } - dist--; /* dist is now the match distance - 1 */ - code = d_code(dist); - Assert (code < D_CODES, "bad d_code"); - - send_code(s, code, dtree); /* send the distance code */ - extra = extra_dbits[code]; - if (extra != 0) { - dist -= base_dist[code]; - send_bits(s, dist, extra); /* send the extra distance bits */ - } - } /* literal or match pair ? */ - - /* Check that the overlay between pending_buf and d_buf+l_buf is ok: */ - Assert((uInt)(s->pending) < s->lit_bufsize + 2*lx, - "pendingBuf overflow"); - - } while (lx < s->last_lit); - - send_code(s, END_BLOCK, ltree); - s->last_eob_len = ltree[END_BLOCK].Len; -} - -/* =========================================================================== - * Check if the data type is TEXT or BINARY, using the following algorithm: - * - TEXT if the two conditions below are satisfied: - * a) There are no non-portable control characters belonging to the - * "black list" (0..6, 14..25, 28..31). - * b) There is at least one printable character belonging to the - * "white list" (9 {TAB}, 10 {LF}, 13 {CR}, 32..255). - * - BINARY otherwise. - * - The following partially-portable control characters form a - * "gray list" that is ignored in this detection algorithm: - * (7 {BEL}, 8 {BS}, 11 {VT}, 12 {FF}, 26 {SUB}, 27 {ESC}). - * IN assertion: the fields Freq of dyn_ltree are set. - */ -local int detect_data_type(s) - deflate_state *s; -{ - /* black_mask is the bit mask of black-listed bytes - * set bits 0..6, 14..25, and 28..31 - * 0xf3ffc07f = binary 11110011111111111100000001111111 - */ - unsigned long black_mask = 0xf3ffc07fUL; - int n; - - /* Check for non-textual ("black-listed") bytes. */ - for (n = 0; n <= 31; n++, black_mask >>= 1) - if ((black_mask & 1) && (s->dyn_ltree[n].Freq != 0)) - return Z_BINARY; - - /* Check for textual ("white-listed") bytes. */ - if (s->dyn_ltree[9].Freq != 0 || s->dyn_ltree[10].Freq != 0 - || s->dyn_ltree[13].Freq != 0) - return Z_TEXT; - for (n = 32; n < LITERALS; n++) - if (s->dyn_ltree[n].Freq != 0) - return Z_TEXT; - - /* There are no "black-listed" or "white-listed" bytes: - * this stream either is empty or has tolerated ("gray-listed") bytes only. - */ - return Z_BINARY; -} - -/* =========================================================================== - * Reverse the first len bits of a code, using straightforward code (a faster - * method would use a table) - * IN assertion: 1 <= len <= 15 - */ -local unsigned bi_reverse(code, len) - unsigned code; /* the value to invert */ - int len; /* its bit length */ -{ - register unsigned res = 0; - do { - res |= code & 1; - code >>= 1, res <<= 1; - } while (--len > 0); - return res >> 1; -} - -/* =========================================================================== - * Flush the bit buffer, keeping at most 7 bits in it. - */ -local void bi_flush(s) - deflate_state *s; -{ - if (s->bi_valid == 16) { - put_short(s, s->bi_buf); - s->bi_buf = 0; - s->bi_valid = 0; - } else if (s->bi_valid >= 8) { - put_byte(s, (Byte)s->bi_buf); - s->bi_buf >>= 8; - s->bi_valid -= 8; - } -} - -/* =========================================================================== - * Flush the bit buffer and align the output on a byte boundary - */ -local void bi_windup(s) - deflate_state *s; -{ - if (s->bi_valid > 8) { - put_short(s, s->bi_buf); - } else if (s->bi_valid > 0) { - put_byte(s, (Byte)s->bi_buf); - } - s->bi_buf = 0; - s->bi_valid = 0; -#ifdef DEBUG - s->bits_sent = (s->bits_sent+7) & ~7; -#endif -} - -/* =========================================================================== - * Copy a stored block, storing first the length and its - * one's complement if requested. - */ -local void copy_block(s, buf, len, header) - deflate_state *s; - charf *buf; /* the input data */ - unsigned len; /* its length */ - int header; /* true if block header must be written */ -{ - bi_windup(s); /* align on byte boundary */ - s->last_eob_len = 8; /* enough lookahead for inflate */ - - if (header) { - put_short(s, (ush)len); - put_short(s, (ush)~len); -#ifdef DEBUG - s->bits_sent += 2*16; -#endif - } -#ifdef DEBUG - s->bits_sent += (ulg)len<<3; -#endif - while (len--) { - put_byte(s, *buf++); - } -} diff --git a/code/3rd_glpk/zlib/trees.h b/code/3rd_glpk/zlib/trees.h deleted file mode 100644 index d35639d8..00000000 --- a/code/3rd_glpk/zlib/trees.h +++ /dev/null @@ -1,128 +0,0 @@ -/* header created automatically with -DGEN_TREES_H */ - -local const ct_data static_ltree[L_CODES+2] = { -{{ 12},{ 8}}, {{140},{ 8}}, {{ 76},{ 8}}, {{204},{ 8}}, {{ 44},{ 8}}, -{{172},{ 8}}, {{108},{ 8}}, {{236},{ 8}}, {{ 28},{ 8}}, {{156},{ 8}}, -{{ 92},{ 8}}, {{220},{ 8}}, {{ 60},{ 8}}, {{188},{ 8}}, {{124},{ 8}}, -{{252},{ 8}}, {{ 2},{ 8}}, {{130},{ 8}}, {{ 66},{ 8}}, {{194},{ 8}}, -{{ 34},{ 8}}, {{162},{ 8}}, {{ 98},{ 8}}, {{226},{ 8}}, {{ 18},{ 8}}, -{{146},{ 8}}, {{ 82},{ 8}}, {{210},{ 8}}, {{ 50},{ 8}}, {{178},{ 8}}, -{{114},{ 8}}, {{242},{ 8}}, {{ 10},{ 8}}, {{138},{ 8}}, {{ 74},{ 8}}, -{{202},{ 8}}, {{ 42},{ 8}}, {{170},{ 8}}, {{106},{ 8}}, {{234},{ 8}}, -{{ 26},{ 8}}, {{154},{ 8}}, {{ 90},{ 8}}, {{218},{ 8}}, {{ 58},{ 8}}, -{{186},{ 8}}, {{122},{ 8}}, {{250},{ 8}}, {{ 6},{ 8}}, {{134},{ 8}}, -{{ 70},{ 8}}, {{198},{ 8}}, {{ 38},{ 8}}, {{166},{ 8}}, {{102},{ 8}}, -{{230},{ 8}}, {{ 22},{ 8}}, {{150},{ 8}}, {{ 86},{ 8}}, {{214},{ 8}}, -{{ 54},{ 8}}, {{182},{ 8}}, {{118},{ 8}}, {{246},{ 8}}, {{ 14},{ 8}}, -{{142},{ 8}}, {{ 78},{ 8}}, {{206},{ 8}}, {{ 46},{ 8}}, {{174},{ 8}}, -{{110},{ 8}}, {{238},{ 8}}, {{ 30},{ 8}}, {{158},{ 8}}, {{ 94},{ 8}}, -{{222},{ 8}}, {{ 62},{ 8}}, {{190},{ 8}}, {{126},{ 8}}, {{254},{ 8}}, -{{ 1},{ 8}}, {{129},{ 8}}, {{ 65},{ 8}}, {{193},{ 8}}, {{ 33},{ 8}}, -{{161},{ 8}}, {{ 97},{ 8}}, {{225},{ 8}}, {{ 17},{ 8}}, {{145},{ 8}}, -{{ 81},{ 8}}, {{209},{ 8}}, {{ 49},{ 8}}, {{177},{ 8}}, {{113},{ 8}}, -{{241},{ 8}}, {{ 9},{ 8}}, {{137},{ 8}}, {{ 73},{ 8}}, {{201},{ 8}}, -{{ 41},{ 8}}, {{169},{ 8}}, {{105},{ 8}}, {{233},{ 8}}, {{ 25},{ 8}}, -{{153},{ 8}}, {{ 89},{ 8}}, {{217},{ 8}}, {{ 57},{ 8}}, {{185},{ 8}}, -{{121},{ 8}}, {{249},{ 8}}, {{ 5},{ 8}}, {{133},{ 8}}, {{ 69},{ 8}}, -{{197},{ 8}}, {{ 37},{ 8}}, {{165},{ 8}}, {{101},{ 8}}, {{229},{ 8}}, -{{ 21},{ 8}}, {{149},{ 8}}, {{ 85},{ 8}}, {{213},{ 8}}, {{ 53},{ 8}}, -{{181},{ 8}}, {{117},{ 8}}, {{245},{ 8}}, {{ 13},{ 8}}, {{141},{ 8}}, -{{ 77},{ 8}}, {{205},{ 8}}, {{ 45},{ 8}}, {{173},{ 8}}, {{109},{ 8}}, -{{237},{ 8}}, {{ 29},{ 8}}, {{157},{ 8}}, {{ 93},{ 8}}, {{221},{ 8}}, -{{ 61},{ 8}}, {{189},{ 8}}, {{125},{ 8}}, {{253},{ 8}}, {{ 19},{ 9}}, -{{275},{ 9}}, {{147},{ 9}}, {{403},{ 9}}, {{ 83},{ 9}}, {{339},{ 9}}, -{{211},{ 9}}, {{467},{ 9}}, {{ 51},{ 9}}, {{307},{ 9}}, {{179},{ 9}}, -{{435},{ 9}}, {{115},{ 9}}, {{371},{ 9}}, {{243},{ 9}}, {{499},{ 9}}, -{{ 11},{ 9}}, {{267},{ 9}}, {{139},{ 9}}, {{395},{ 9}}, {{ 75},{ 9}}, -{{331},{ 9}}, {{203},{ 9}}, {{459},{ 9}}, {{ 43},{ 9}}, {{299},{ 9}}, -{{171},{ 9}}, {{427},{ 9}}, {{107},{ 9}}, {{363},{ 9}}, {{235},{ 9}}, -{{491},{ 9}}, {{ 27},{ 9}}, {{283},{ 9}}, {{155},{ 9}}, {{411},{ 9}}, -{{ 91},{ 9}}, {{347},{ 9}}, {{219},{ 9}}, {{475},{ 9}}, {{ 59},{ 9}}, -{{315},{ 9}}, {{187},{ 9}}, {{443},{ 9}}, {{123},{ 9}}, {{379},{ 9}}, -{{251},{ 9}}, {{507},{ 9}}, {{ 7},{ 9}}, {{263},{ 9}}, {{135},{ 9}}, -{{391},{ 9}}, {{ 71},{ 9}}, {{327},{ 9}}, {{199},{ 9}}, {{455},{ 9}}, -{{ 39},{ 9}}, {{295},{ 9}}, {{167},{ 9}}, {{423},{ 9}}, {{103},{ 9}}, -{{359},{ 9}}, {{231},{ 9}}, {{487},{ 9}}, {{ 23},{ 9}}, {{279},{ 9}}, -{{151},{ 9}}, {{407},{ 9}}, {{ 87},{ 9}}, {{343},{ 9}}, {{215},{ 9}}, -{{471},{ 9}}, {{ 55},{ 9}}, {{311},{ 9}}, {{183},{ 9}}, {{439},{ 9}}, -{{119},{ 9}}, {{375},{ 9}}, {{247},{ 9}}, {{503},{ 9}}, {{ 15},{ 9}}, -{{271},{ 9}}, {{143},{ 9}}, {{399},{ 9}}, {{ 79},{ 9}}, {{335},{ 9}}, -{{207},{ 9}}, {{463},{ 9}}, {{ 47},{ 9}}, {{303},{ 9}}, {{175},{ 9}}, -{{431},{ 9}}, {{111},{ 9}}, {{367},{ 9}}, {{239},{ 9}}, {{495},{ 9}}, -{{ 31},{ 9}}, {{287},{ 9}}, {{159},{ 9}}, {{415},{ 9}}, {{ 95},{ 9}}, -{{351},{ 9}}, {{223},{ 9}}, {{479},{ 9}}, {{ 63},{ 9}}, {{319},{ 9}}, -{{191},{ 9}}, {{447},{ 9}}, {{127},{ 9}}, {{383},{ 9}}, {{255},{ 9}}, -{{511},{ 9}}, {{ 0},{ 7}}, {{ 64},{ 7}}, {{ 32},{ 7}}, {{ 96},{ 7}}, -{{ 16},{ 7}}, {{ 80},{ 7}}, {{ 48},{ 7}}, {{112},{ 7}}, {{ 8},{ 7}}, -{{ 72},{ 7}}, {{ 40},{ 7}}, {{104},{ 7}}, {{ 24},{ 7}}, {{ 88},{ 7}}, -{{ 56},{ 7}}, {{120},{ 7}}, {{ 4},{ 7}}, {{ 68},{ 7}}, {{ 36},{ 7}}, -{{100},{ 7}}, {{ 20},{ 7}}, {{ 84},{ 7}}, {{ 52},{ 7}}, {{116},{ 7}}, -{{ 3},{ 8}}, {{131},{ 8}}, {{ 67},{ 8}}, {{195},{ 8}}, {{ 35},{ 8}}, -{{163},{ 8}}, {{ 99},{ 8}}, {{227},{ 8}} -}; - -local const ct_data static_dtree[D_CODES] = { -{{ 0},{ 5}}, {{16},{ 5}}, {{ 8},{ 5}}, {{24},{ 5}}, {{ 4},{ 5}}, -{{20},{ 5}}, {{12},{ 5}}, {{28},{ 5}}, {{ 2},{ 5}}, {{18},{ 5}}, -{{10},{ 5}}, {{26},{ 5}}, {{ 6},{ 5}}, {{22},{ 5}}, {{14},{ 5}}, -{{30},{ 5}}, {{ 1},{ 5}}, {{17},{ 5}}, {{ 9},{ 5}}, {{25},{ 5}}, -{{ 5},{ 5}}, {{21},{ 5}}, {{13},{ 5}}, {{29},{ 5}}, {{ 3},{ 5}}, -{{19},{ 5}}, {{11},{ 5}}, {{27},{ 5}}, {{ 7},{ 5}}, {{23},{ 5}} -}; - -const uch ZLIB_INTERNAL _dist_code[DIST_CODE_LEN] = { - 0, 1, 2, 3, 4, 4, 5, 5, 6, 6, 6, 6, 7, 7, 7, 7, 8, 8, 8, 8, - 8, 8, 8, 8, 9, 9, 9, 9, 9, 9, 9, 9, 10, 10, 10, 10, 10, 10, 10, 10, -10, 10, 10, 10, 10, 10, 10, 10, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, -11, 11, 11, 11, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, -12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 13, 13, 13, 13, -13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, -13, 13, 13, 13, 13, 13, 13, 13, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, -14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, -14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, -14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 15, 15, 15, 15, 15, 15, 15, 15, -15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, -15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, -15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 0, 0, 16, 17, -18, 18, 19, 19, 20, 20, 20, 20, 21, 21, 21, 21, 22, 22, 22, 22, 22, 22, 22, 22, -23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, -24, 24, 24, 24, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, -26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, -26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27, -27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, -27, 27, 27, 27, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, -28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, -28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, -28, 28, 28, 28, 28, 28, 28, 28, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, -29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, -29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, -29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29 -}; - -const uch ZLIB_INTERNAL _length_code[MAX_MATCH-MIN_MATCH+1]= { - 0, 1, 2, 3, 4, 5, 6, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 12, 12, -13, 13, 13, 13, 14, 14, 14, 14, 15, 15, 15, 15, 16, 16, 16, 16, 16, 16, 16, 16, -17, 17, 17, 17, 17, 17, 17, 17, 18, 18, 18, 18, 18, 18, 18, 18, 19, 19, 19, 19, -19, 19, 19, 19, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, -21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 22, 22, 22, 22, -22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 23, 23, 23, 23, 23, 23, 23, 23, -23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, -24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, -25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, -25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 26, 26, 26, 26, 26, 26, 26, 26, -26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, -26, 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, -27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 28 -}; - -local const int base_length[LENGTH_CODES] = { -0, 1, 2, 3, 4, 5, 6, 7, 8, 10, 12, 14, 16, 20, 24, 28, 32, 40, 48, 56, -64, 80, 96, 112, 128, 160, 192, 224, 0 -}; - -local const int base_dist[D_CODES] = { - 0, 1, 2, 3, 4, 6, 8, 12, 16, 24, - 32, 48, 64, 96, 128, 192, 256, 384, 512, 768, - 1024, 1536, 2048, 3072, 4096, 6144, 8192, 12288, 16384, 24576 -}; - diff --git a/code/3rd_glpk/zlib/uncompr.c b/code/3rd_glpk/zlib/uncompr.c deleted file mode 100644 index ad98be3a..00000000 --- a/code/3rd_glpk/zlib/uncompr.c +++ /dev/null @@ -1,59 +0,0 @@ -/* uncompr.c -- decompress a memory buffer - * Copyright (C) 1995-2003, 2010 Jean-loup Gailly. - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* @(#) $Id$ */ - -#define ZLIB_INTERNAL -#include "zlib.h" - -/* =========================================================================== - Decompresses the source buffer into the destination buffer. sourceLen is - the byte length of the source buffer. Upon entry, destLen is the total - size of the destination buffer, which must be large enough to hold the - entire uncompressed data. (The size of the uncompressed data must have - been saved previously by the compressor and transmitted to the decompressor - by some mechanism outside the scope of this compression library.) - Upon exit, destLen is the actual size of the compressed buffer. - - uncompress returns Z_OK if success, Z_MEM_ERROR if there was not - enough memory, Z_BUF_ERROR if there was not enough room in the output - buffer, or Z_DATA_ERROR if the input data was corrupted. -*/ -int ZEXPORT uncompress (dest, destLen, source, sourceLen) - Bytef *dest; - uLongf *destLen; - const Bytef *source; - uLong sourceLen; -{ - z_stream stream; - int err; - - stream.next_in = (Bytef*)source; - stream.avail_in = (uInt)sourceLen; - /* Check for source > 64K on 16-bit machine: */ - if ((uLong)stream.avail_in != sourceLen) return Z_BUF_ERROR; - - stream.next_out = dest; - stream.avail_out = (uInt)*destLen; - if ((uLong)stream.avail_out != *destLen) return Z_BUF_ERROR; - - stream.zalloc = (alloc_func)0; - stream.zfree = (free_func)0; - - err = inflateInit(&stream); - if (err != Z_OK) return err; - - err = inflate(&stream, Z_FINISH); - if (err != Z_STREAM_END) { - inflateEnd(&stream); - if (err == Z_NEED_DICT || (err == Z_BUF_ERROR && stream.avail_in == 0)) - return Z_DATA_ERROR; - return err; - } - *destLen = stream.total_out; - - err = inflateEnd(&stream); - return err; -} diff --git a/code/3rd_glpk/zlib/zconf.h b/code/3rd_glpk/zlib/zconf.h deleted file mode 100644 index af6a4f0f..00000000 --- a/code/3rd_glpk/zlib/zconf.h +++ /dev/null @@ -1,168 +0,0 @@ -/* zconf.h (configuration of the zlib compression library) */ - -/* Modified by Andrew Makhorin , April 2011 */ - -/* Copyright (C) 1995-2010 Jean-loup Gailly - * For conditions of distribution and use, see copyright notice in - * zlib.h */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. */ - -#ifndef ZCONF_H -#define ZCONF_H - -/* (file adler32.c) */ -#define adler32 _glp_zlib_adler32 -#define adler32_combine _glp_zlib_adler32_combine -#define adler32_combine64 _glp_zlib_adler32_combine64 - -/* (file compress.c) */ -#define compress2 _glp_zlib_compress2 -#define compress _glp_zlib_compress -#define compressBound _glp_zlib_compressBound - -/* (file crc32.c) */ -#define get_crc_table _glp_zlib_get_crc_table -#define crc32 _glp_zlib_crc32 -#define crc32_combine _glp_zlib_crc32_combine -#define crc32_combine64 _glp_zlib_crc32_combine64 - -/* (file deflate.c) */ -#define deflateInit_ _glp_zlib_deflateInit_ -#define deflateInit2_ _glp_zlib_deflateInit2_ -#define deflateSetDictionary _glp_zlib_deflateSetDictionary -#define deflateReset _glp_zlib_deflateReset -#define deflateSetHeader _glp_zlib_deflateSetHeader -#define deflatePrime _glp_zlib_deflatePrime -#define deflateParams _glp_zlib_deflateParams -#define deflateTune _glp_zlib_deflateTune -#define deflateBound _glp_zlib_deflateBound -#define deflate _glp_zlib_deflate -#define deflateEnd _glp_zlib_deflateEnd -#define deflateCopy _glp_zlib_deflateCopy -#define deflate_copyright _glp_zlib_deflate_copyright - -/* (file gzclose.c) */ -#define gzclose _glp_zlib_gzclose - -/* (file gzlib.c) */ -#define gzopen _glp_zlib_gzopen -#define gzopen64 _glp_zlib_gzopen64 -#define gzdopen _glp_zlib_gzdopen -#define gzbuffer _glp_zlib_gzbuffer -#define gzrewind _glp_zlib_gzrewind -#define gzseek64 _glp_zlib_gzseek64 -#define gzseek _glp_zlib_gzseek -#define gztell64 _glp_zlib_gztell64 -#define gztell _glp_zlib_gztell -#define gzoffset64 _glp_zlib_gzoffset64 -#define gzoffset _glp_zlib_gzoffset -#define gzeof _glp_zlib_gzeof -#define gzerror _glp_zlib_gzerror -#define gzclearerr _glp_zlib_gzclearerr -#define gz_error _glp_zlib_gz_error - -/* (file gzread.c) */ -#define gzread _glp_zlib_gzread -#define gzgetc _glp_zlib_gzgetc -#define gzungetc _glp_zlib_gzungetc -#define gzgets _glp_zlib_gzgets -#define gzdirect _glp_zlib_gzdirect -#define gzclose_r _glp_zlib_gzclose_r - -/* (file gzwrite.c) */ -#define gzwrite _glp_zlib_gzwrite -#define gzputc _glp_zlib_gzputc -#define gzputs _glp_zlib_gzputs -#define gzprintf _glp_zlib_gzprintf -#define gzflush _glp_zlib_gzflush -#define gzsetparams _glp_zlib_gzsetparams -#define gzclose_w _glp_zlib_gzclose_w - -/* (file infback.c) */ -#define inflateBackInit_ _glp_zlib_inflateBackInit_ -#define inflateBack _glp_zlib_inflateBack -#define inflateBackEnd _glp_zlib_inflateBackEnd - -/* (file inffast.c) */ -#define inflate_fast _glp_zlib_inflate_fast - -/* (file inflate.c) */ -#define inflateReset _glp_zlib_inflateReset -#define inflateReset2 _glp_zlib_inflateReset2 -#define inflateInit2_ _glp_zlib_inflateInit2_ -#define inflateInit_ _glp_zlib_inflateInit_ -#define inflatePrime _glp_zlib_inflatePrime -#define inflate _glp_zlib_inflate -#define inflateEnd _glp_zlib_inflateEnd -#define inflateSetDictionary _glp_zlib_inflateSetDictionary -#define inflateGetHeader _glp_zlib_inflateGetHeader -#define inflateSync _glp_zlib_inflateSync -#define inflateSyncPoint _glp_zlib_inflateSyncPoint -#define inflateCopy _glp_zlib_inflateCopy -#define inflateUndermine _glp_zlib_inflateUndermine -#define inflateMark _glp_zlib_inflateMark - -/* (file inftrees.c) */ -#define inflate_table _glp_zlib_inflate_table -#define inflate_copyright _glp_zlib_inflate_copyright - -/* (file trees.c) */ -#define _tr_init _glp_zlib_tr_init -#define _tr_stored_block _glp_zlib_tr_stored_block -#define _tr_align _glp_zlib_tr_align -#define _tr_flush_block _glp_zlib_tr_flush_block -#define _tr_tally _glp_zlib_tr_tally -#define _dist_code _glp_zlib_dist_code -#define _length_code _glp_zlib_length_code - -/* (file uncompr.c) */ -#define uncompress _glp_zlib_uncompress - -/* (file zutil.c) */ -#define zlibVersion _glp_zlib_zlibVersion -#define zlibCompileFlags _glp_zlib_zlibCompileFlags -#define zError _glp_zlib_zError -#define zcalloc _glp_zlib_zcalloc -#define zcfree _glp_zlib_zcfree -#define z_errmsg _glp_zlib_z_errmsg - -#define STDC 1 - -#define MAX_MEM_LEVEL 9 - -#define MAX_WBITS 15 - -#define OF(args) args - -#define ZEXTERN extern -#define ZEXPORT -#define ZEXPORTVA - -#define FAR - -typedef unsigned char Byte; -typedef unsigned int uInt; -typedef unsigned long uLong; - -typedef Byte Bytef; -typedef char charf; -typedef int intf; -typedef uInt uIntf; -typedef uLong uLongf; - -typedef void const *voidpc; -typedef void *voidpf; -typedef void *voidp; - -#define z_off_t long - -#define z_off64_t z_off_t - -#define NO_vsnprintf 1 - -#endif - -/* eof */ diff --git a/code/3rd_glpk/zlib/zio.c b/code/3rd_glpk/zlib/zio.c deleted file mode 100644 index a55b258a..00000000 --- a/code/3rd_glpk/zlib/zio.c +++ /dev/null @@ -1,92 +0,0 @@ -/* zio.c (simulation of non-standard low-level i/o functions) */ - -/* Written by Andrew Makhorin , April 2011 - * For conditions of distribution and use, see copyright notice in - * zlib.h */ - -/* (reserved for copyright notice) */ - -#include -#include -#include "zio.h" - -static FILE *file[FOPEN_MAX]; -static int initialized = 0; - -static void initialize(void) -{ int fd; - assert(!initialized); - file[0] = stdin; - file[1] = stdout; - file[2] = stderr; - for (fd = 3; fd < FOPEN_MAX; fd++) - file[fd] = NULL; - initialized = 1; - return; -} - -int open(const char *path, int oflag, ...) -{ FILE *fp; - int fd; - if (!initialized) initialize(); - /* see file gzlib.c, function gz_open */ - if (oflag == O_RDONLY) - fp = fopen(path, "rb"); - else if (oflag == (O_WRONLY | O_CREAT | O_TRUNC)) - fp = fopen(path, "wb"); - else if (oflag == (O_WRONLY | O_CREAT | O_APPEND)) - fp = fopen(path, "ab"); - else - assert(oflag != oflag); - if (fp == NULL) - return -1; - for (fd = 0; fd < FOPEN_MAX; fd++) - if (file[fd] == NULL) break; - assert(fd < FOPEN_MAX); - file[fd] = fp; - return fd; -} - -long read(int fd, void *buf, unsigned long nbyte) -{ unsigned long count; - if (!initialized) initialize(); - assert(0 <= fd && fd < FOPEN_MAX); - assert(file[fd] != NULL); - count = fread(buf, 1, nbyte, file[fd]); - if (ferror(file[fd])) - return -1; - return count; -} - -long write(int fd, const void *buf, unsigned long nbyte) -{ unsigned long count; - if (!initialized) initialize(); - assert(0 <= fd && fd < FOPEN_MAX); - assert(file[fd] != NULL); - count = fwrite(buf, 1, nbyte, file[fd]); - if (count != nbyte) - return -1; - if (fflush(file[fd]) != 0) - return -1; - return count; -} - -long lseek(int fd, long offset, int whence) -{ if (!initialized) initialize(); - assert(0 <= fd && fd < FOPEN_MAX); - assert(file[fd] != NULL); - if (fseek(file[fd], offset, whence) != 0) - return -1; - return ftell(file[fd]); -} - -int close(int fd) -{ if (!initialized) initialize(); - assert(0 <= fd && fd < FOPEN_MAX); - assert(file[fd] != NULL); - fclose(file[fd]); - file[fd] = NULL; - return 0; -} - -/* eof */ diff --git a/code/3rd_glpk/zlib/zio.h b/code/3rd_glpk/zlib/zio.h deleted file mode 100644 index 1626c4ae..00000000 --- a/code/3rd_glpk/zlib/zio.h +++ /dev/null @@ -1,37 +0,0 @@ -/* zio.h (simulation of non-standard low-level i/o functions) */ - -/* Written by Andrew Makhorin , April 2011 - * For conditions of distribution and use, see copyright notice in - * zlib.h */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. */ - -#ifndef ZIO_H -#define ZIO_H - -#define O_RDONLY 0x00 -#define O_WRONLY 0x01 -#define O_CREAT 0x10 -#define O_TRUNC 0x20 -#define O_APPEND 0x30 - -#define open _glp_zlib_open -int open(const char *path, int oflag, ...); - -#define read _glp_zlib_read -long read(int fd, void *buf, unsigned long nbyte); - -#define write _glp_zlib_write -long write(int fd, const void *buf, unsigned long nbyte); - -#define lseek _glp_zlib_lseek -long lseek(int fd, long offset, int whence); - -#define close _glp_zlib_close -int close(int fd); - -#endif - -/* eof */ diff --git a/code/3rd_glpk/zlib/zlib.h b/code/3rd_glpk/zlib/zlib.h deleted file mode 100644 index bfbba83e..00000000 --- a/code/3rd_glpk/zlib/zlib.h +++ /dev/null @@ -1,1613 +0,0 @@ -/* zlib.h -- interface of the 'zlib' general purpose compression library - version 1.2.5, April 19th, 2010 - - Copyright (C) 1995-2010 Jean-loup Gailly and Mark Adler - - This software is provided 'as-is', without any express or implied - warranty. In no event will the authors be held liable for any damages - arising from the use of this software. - - Permission is granted to anyone to use this software for any purpose, - including commercial applications, and to alter it and redistribute it - freely, subject to the following restrictions: - - 1. The origin of this software must not be misrepresented; you must not - claim that you wrote the original software. If you use this software - in a product, an acknowledgment in the product documentation would be - appreciated but is not required. - 2. Altered source versions must be plainly marked as such, and must not be - misrepresented as being the original software. - 3. This notice may not be removed or altered from any source distribution. - - Jean-loup Gailly Mark Adler - jloup@gzip.org madler@alumni.caltech.edu - - - The data format used by the zlib library is described by RFCs (Request for - Comments) 1950 to 1952 in the files http://www.ietf.org/rfc/rfc1950.txt - (zlib format), rfc1951.txt (deflate format) and rfc1952.txt (gzip format). -*/ - -#ifndef ZLIB_H -#define ZLIB_H - -#include "zconf.h" - -#ifdef __cplusplus -extern "C" { -#endif - -#define ZLIB_VERSION "1.2.5" -#define ZLIB_VERNUM 0x1250 -#define ZLIB_VER_MAJOR 1 -#define ZLIB_VER_MINOR 2 -#define ZLIB_VER_REVISION 5 -#define ZLIB_VER_SUBREVISION 0 - -/* - The 'zlib' compression library provides in-memory compression and - decompression functions, including integrity checks of the uncompressed data. - This version of the library supports only one compression method (deflation) - but other algorithms will be added later and will have the same stream - interface. - - Compression can be done in a single step if the buffers are large enough, - or can be done by repeated calls of the compression function. In the latter - case, the application must provide more input and/or consume the output - (providing more output space) before each call. - - The compressed data format used by default by the in-memory functions is - the zlib format, which is a zlib wrapper documented in RFC 1950, wrapped - around a deflate stream, which is itself documented in RFC 1951. - - The library also supports reading and writing files in gzip (.gz) format - with an interface similar to that of stdio using the functions that start - with "gz". The gzip format is different from the zlib format. gzip is a - gzip wrapper, documented in RFC 1952, wrapped around a deflate stream. - - This library can optionally read and write gzip streams in memory as well. - - The zlib format was designed to be compact and fast for use in memory - and on communications channels. The gzip format was designed for single- - file compression on file systems, has a larger header than zlib to maintain - directory information, and uses a different, slower check method than zlib. - - The library does not install any signal handler. The decoder checks - the consistency of the compressed data, so the library should never crash - even in case of corrupted input. -*/ - -typedef voidpf (*alloc_func) OF((voidpf opaque, uInt items, uInt size)); -typedef void (*free_func) OF((voidpf opaque, voidpf address)); - -struct internal_state; - -typedef struct z_stream_s { - Bytef *next_in; /* next input byte */ - uInt avail_in; /* number of bytes available at next_in */ - uLong total_in; /* total nb of input bytes read so far */ - - Bytef *next_out; /* next output byte should be put there */ - uInt avail_out; /* remaining free space at next_out */ - uLong total_out; /* total nb of bytes output so far */ - - char *msg; /* last error message, NULL if no error */ - struct internal_state FAR *state; /* not visible by applications */ - - alloc_func zalloc; /* used to allocate the internal state */ - free_func zfree; /* used to free the internal state */ - voidpf opaque; /* private data object passed to zalloc and zfree */ - - int data_type; /* best guess about the data type: binary or text */ - uLong adler; /* adler32 value of the uncompressed data */ - uLong reserved; /* reserved for future use */ -} z_stream; - -typedef z_stream FAR *z_streamp; - -/* - gzip header information passed to and from zlib routines. See RFC 1952 - for more details on the meanings of these fields. -*/ -typedef struct gz_header_s { - int text; /* true if compressed data believed to be text */ - uLong time; /* modification time */ - int xflags; /* extra flags (not used when writing a gzip file) */ - int os; /* operating system */ - Bytef *extra; /* pointer to extra field or Z_NULL if none */ - uInt extra_len; /* extra field length (valid if extra != Z_NULL) */ - uInt extra_max; /* space at extra (only when reading header) */ - Bytef *name; /* pointer to zero-terminated file name or Z_NULL */ - uInt name_max; /* space at name (only when reading header) */ - Bytef *comment; /* pointer to zero-terminated comment or Z_NULL */ - uInt comm_max; /* space at comment (only when reading header) */ - int hcrc; /* true if there was or will be a header crc */ - int done; /* true when done reading gzip header (not used - when writing a gzip file) */ -} gz_header; - -typedef gz_header FAR *gz_headerp; - -/* - The application must update next_in and avail_in when avail_in has dropped - to zero. It must update next_out and avail_out when avail_out has dropped - to zero. The application must initialize zalloc, zfree and opaque before - calling the init function. All other fields are set by the compression - library and must not be updated by the application. - - The opaque value provided by the application will be passed as the first - parameter for calls of zalloc and zfree. This can be useful for custom - memory management. The compression library attaches no meaning to the - opaque value. - - zalloc must return Z_NULL if there is not enough memory for the object. - If zlib is used in a multi-threaded application, zalloc and zfree must be - thread safe. - - On 16-bit systems, the functions zalloc and zfree must be able to allocate - exactly 65536 bytes, but will not be required to allocate more than this if - the symbol MAXSEG_64K is defined (see zconf.h). WARNING: On MSDOS, pointers - returned by zalloc for objects of exactly 65536 bytes *must* have their - offset normalized to zero. The default allocation function provided by this - library ensures this (see zutil.c). To reduce memory requirements and avoid - any allocation of 64K objects, at the expense of compression ratio, compile - the library with -DMAX_WBITS=14 (see zconf.h). - - The fields total_in and total_out can be used for statistics or progress - reports. After compression, total_in holds the total size of the - uncompressed data and may be saved for use in the decompressor (particularly - if the decompressor wants to decompress everything in a single step). -*/ - - /* constants */ - -#define Z_NO_FLUSH 0 -#define Z_PARTIAL_FLUSH 1 -#define Z_SYNC_FLUSH 2 -#define Z_FULL_FLUSH 3 -#define Z_FINISH 4 -#define Z_BLOCK 5 -#define Z_TREES 6 -/* Allowed flush values; see deflate() and inflate() below for details */ - -#define Z_OK 0 -#define Z_STREAM_END 1 -#define Z_NEED_DICT 2 -#define Z_ERRNO (-1) -#define Z_STREAM_ERROR (-2) -#define Z_DATA_ERROR (-3) -#define Z_MEM_ERROR (-4) -#define Z_BUF_ERROR (-5) -#define Z_VERSION_ERROR (-6) -/* Return codes for the compression/decompression functions. Negative values - * are errors, positive values are used for special but normal events. - */ - -#define Z_NO_COMPRESSION 0 -#define Z_BEST_SPEED 1 -#define Z_BEST_COMPRESSION 9 -#define Z_DEFAULT_COMPRESSION (-1) -/* compression levels */ - -#define Z_FILTERED 1 -#define Z_HUFFMAN_ONLY 2 -#define Z_RLE 3 -#define Z_FIXED 4 -#define Z_DEFAULT_STRATEGY 0 -/* compression strategy; see deflateInit2() below for details */ - -#define Z_BINARY 0 -#define Z_TEXT 1 -#define Z_ASCII Z_TEXT /* for compatibility with 1.2.2 and earlier */ -#define Z_UNKNOWN 2 -/* Possible values of the data_type field (though see inflate()) */ - -#define Z_DEFLATED 8 -/* The deflate compression method (the only one supported in this version) */ - -#define Z_NULL 0 /* for initializing zalloc, zfree, opaque */ - -#define zlib_version zlibVersion() -/* for compatibility with versions < 1.0.2 */ - - - /* basic functions */ - -ZEXTERN const char * ZEXPORT zlibVersion OF((void)); -/* The application can compare zlibVersion and ZLIB_VERSION for consistency. - If the first character differs, the library code actually used is not - compatible with the zlib.h header file used by the application. This check - is automatically made by deflateInit and inflateInit. - */ - -/* -ZEXTERN int ZEXPORT deflateInit OF((z_streamp strm, int level)); - - Initializes the internal stream state for compression. The fields - zalloc, zfree and opaque must be initialized before by the caller. If - zalloc and zfree are set to Z_NULL, deflateInit updates them to use default - allocation functions. - - The compression level must be Z_DEFAULT_COMPRESSION, or between 0 and 9: - 1 gives best speed, 9 gives best compression, 0 gives no compression at all - (the input data is simply copied a block at a time). Z_DEFAULT_COMPRESSION - requests a default compromise between speed and compression (currently - equivalent to level 6). - - deflateInit returns Z_OK if success, Z_MEM_ERROR if there was not enough - memory, Z_STREAM_ERROR if level is not a valid compression level, or - Z_VERSION_ERROR if the zlib library version (zlib_version) is incompatible - with the version assumed by the caller (ZLIB_VERSION). msg is set to null - if there is no error message. deflateInit does not perform any compression: - this will be done by deflate(). -*/ - - -ZEXTERN int ZEXPORT deflate OF((z_streamp strm, int flush)); -/* - deflate compresses as much data as possible, and stops when the input - buffer becomes empty or the output buffer becomes full. It may introduce - some output latency (reading input without producing any output) except when - forced to flush. - - The detailed semantics are as follows. deflate performs one or both of the - following actions: - - - Compress more input starting at next_in and update next_in and avail_in - accordingly. If not all input can be processed (because there is not - enough room in the output buffer), next_in and avail_in are updated and - processing will resume at this point for the next call of deflate(). - - - Provide more output starting at next_out and update next_out and avail_out - accordingly. This action is forced if the parameter flush is non zero. - Forcing flush frequently degrades the compression ratio, so this parameter - should be set only when necessary (in interactive applications). Some - output may be provided even if flush is not set. - - Before the call of deflate(), the application should ensure that at least - one of the actions is possible, by providing more input and/or consuming more - output, and updating avail_in or avail_out accordingly; avail_out should - never be zero before the call. The application can consume the compressed - output when it wants, for example when the output buffer is full (avail_out - == 0), or after each call of deflate(). If deflate returns Z_OK and with - zero avail_out, it must be called again after making room in the output - buffer because there might be more output pending. - - Normally the parameter flush is set to Z_NO_FLUSH, which allows deflate to - decide how much data to accumulate before producing output, in order to - maximize compression. - - If the parameter flush is set to Z_SYNC_FLUSH, all pending output is - flushed to the output buffer and the output is aligned on a byte boundary, so - that the decompressor can get all input data available so far. (In - particular avail_in is zero after the call if enough output space has been - provided before the call.) Flushing may degrade compression for some - compression algorithms and so it should be used only when necessary. This - completes the current deflate block and follows it with an empty stored block - that is three bits plus filler bits to the next byte, followed by four bytes - (00 00 ff ff). - - If flush is set to Z_PARTIAL_FLUSH, all pending output is flushed to the - output buffer, but the output is not aligned to a byte boundary. All of the - input data so far will be available to the decompressor, as for Z_SYNC_FLUSH. - This completes the current deflate block and follows it with an empty fixed - codes block that is 10 bits long. This assures that enough bytes are output - in order for the decompressor to finish the block before the empty fixed code - block. - - If flush is set to Z_BLOCK, a deflate block is completed and emitted, as - for Z_SYNC_FLUSH, but the output is not aligned on a byte boundary, and up to - seven bits of the current block are held to be written as the next byte after - the next deflate block is completed. In this case, the decompressor may not - be provided enough bits at this point in order to complete decompression of - the data provided so far to the compressor. It may need to wait for the next - block to be emitted. This is for advanced applications that need to control - the emission of deflate blocks. - - If flush is set to Z_FULL_FLUSH, all output is flushed as with - Z_SYNC_FLUSH, and the compression state is reset so that decompression can - restart from this point if previous compressed data has been damaged or if - random access is desired. Using Z_FULL_FLUSH too often can seriously degrade - compression. - - If deflate returns with avail_out == 0, this function must be called again - with the same value of the flush parameter and more output space (updated - avail_out), until the flush is complete (deflate returns with non-zero - avail_out). In the case of a Z_FULL_FLUSH or Z_SYNC_FLUSH, make sure that - avail_out is greater than six to avoid repeated flush markers due to - avail_out == 0 on return. - - If the parameter flush is set to Z_FINISH, pending input is processed, - pending output is flushed and deflate returns with Z_STREAM_END if there was - enough output space; if deflate returns with Z_OK, this function must be - called again with Z_FINISH and more output space (updated avail_out) but no - more input data, until it returns with Z_STREAM_END or an error. After - deflate has returned Z_STREAM_END, the only possible operations on the stream - are deflateReset or deflateEnd. - - Z_FINISH can be used immediately after deflateInit if all the compression - is to be done in a single step. In this case, avail_out must be at least the - value returned by deflateBound (see below). If deflate does not return - Z_STREAM_END, then it must be called again as described above. - - deflate() sets strm->adler to the adler32 checksum of all input read - so far (that is, total_in bytes). - - deflate() may update strm->data_type if it can make a good guess about - the input data type (Z_BINARY or Z_TEXT). In doubt, the data is considered - binary. This field is only for information purposes and does not affect the - compression algorithm in any manner. - - deflate() returns Z_OK if some progress has been made (more input - processed or more output produced), Z_STREAM_END if all input has been - consumed and all output has been produced (only when flush is set to - Z_FINISH), Z_STREAM_ERROR if the stream state was inconsistent (for example - if next_in or next_out was Z_NULL), Z_BUF_ERROR if no progress is possible - (for example avail_in or avail_out was zero). Note that Z_BUF_ERROR is not - fatal, and deflate() can be called again with more input and more output - space to continue compressing. -*/ - - -ZEXTERN int ZEXPORT deflateEnd OF((z_streamp strm)); -/* - All dynamically allocated data structures for this stream are freed. - This function discards any unprocessed input and does not flush any pending - output. - - deflateEnd returns Z_OK if success, Z_STREAM_ERROR if the - stream state was inconsistent, Z_DATA_ERROR if the stream was freed - prematurely (some input or output was discarded). In the error case, msg - may be set but then points to a static string (which must not be - deallocated). -*/ - - -/* -ZEXTERN int ZEXPORT inflateInit OF((z_streamp strm)); - - Initializes the internal stream state for decompression. The fields - next_in, avail_in, zalloc, zfree and opaque must be initialized before by - the caller. If next_in is not Z_NULL and avail_in is large enough (the - exact value depends on the compression method), inflateInit determines the - compression method from the zlib header and allocates all data structures - accordingly; otherwise the allocation will be deferred to the first call of - inflate. If zalloc and zfree are set to Z_NULL, inflateInit updates them to - use default allocation functions. - - inflateInit returns Z_OK if success, Z_MEM_ERROR if there was not enough - memory, Z_VERSION_ERROR if the zlib library version is incompatible with the - version assumed by the caller, or Z_STREAM_ERROR if the parameters are - invalid, such as a null pointer to the structure. msg is set to null if - there is no error message. inflateInit does not perform any decompression - apart from possibly reading the zlib header if present: actual decompression - will be done by inflate(). (So next_in and avail_in may be modified, but - next_out and avail_out are unused and unchanged.) The current implementation - of inflateInit() does not process any header information -- that is deferred - until inflate() is called. -*/ - - -ZEXTERN int ZEXPORT inflate OF((z_streamp strm, int flush)); -/* - inflate decompresses as much data as possible, and stops when the input - buffer becomes empty or the output buffer becomes full. It may introduce - some output latency (reading input without producing any output) except when - forced to flush. - - The detailed semantics are as follows. inflate performs one or both of the - following actions: - - - Decompress more input starting at next_in and update next_in and avail_in - accordingly. If not all input can be processed (because there is not - enough room in the output buffer), next_in is updated and processing will - resume at this point for the next call of inflate(). - - - Provide more output starting at next_out and update next_out and avail_out - accordingly. inflate() provides as much output as possible, until there is - no more input data or no more space in the output buffer (see below about - the flush parameter). - - Before the call of inflate(), the application should ensure that at least - one of the actions is possible, by providing more input and/or consuming more - output, and updating the next_* and avail_* values accordingly. The - application can consume the uncompressed output when it wants, for example - when the output buffer is full (avail_out == 0), or after each call of - inflate(). If inflate returns Z_OK and with zero avail_out, it must be - called again after making room in the output buffer because there might be - more output pending. - - The flush parameter of inflate() can be Z_NO_FLUSH, Z_SYNC_FLUSH, Z_FINISH, - Z_BLOCK, or Z_TREES. Z_SYNC_FLUSH requests that inflate() flush as much - output as possible to the output buffer. Z_BLOCK requests that inflate() - stop if and when it gets to the next deflate block boundary. When decoding - the zlib or gzip format, this will cause inflate() to return immediately - after the header and before the first block. When doing a raw inflate, - inflate() will go ahead and process the first block, and will return when it - gets to the end of that block, or when it runs out of data. - - The Z_BLOCK option assists in appending to or combining deflate streams. - Also to assist in this, on return inflate() will set strm->data_type to the - number of unused bits in the last byte taken from strm->next_in, plus 64 if - inflate() is currently decoding the last block in the deflate stream, plus - 128 if inflate() returned immediately after decoding an end-of-block code or - decoding the complete header up to just before the first byte of the deflate - stream. The end-of-block will not be indicated until all of the uncompressed - data from that block has been written to strm->next_out. The number of - unused bits may in general be greater than seven, except when bit 7 of - data_type is set, in which case the number of unused bits will be less than - eight. data_type is set as noted here every time inflate() returns for all - flush options, and so can be used to determine the amount of currently - consumed input in bits. - - The Z_TREES option behaves as Z_BLOCK does, but it also returns when the - end of each deflate block header is reached, before any actual data in that - block is decoded. This allows the caller to determine the length of the - deflate block header for later use in random access within a deflate block. - 256 is added to the value of strm->data_type when inflate() returns - immediately after reaching the end of the deflate block header. - - inflate() should normally be called until it returns Z_STREAM_END or an - error. However if all decompression is to be performed in a single step (a - single call of inflate), the parameter flush should be set to Z_FINISH. In - this case all pending input is processed and all pending output is flushed; - avail_out must be large enough to hold all the uncompressed data. (The size - of the uncompressed data may have been saved by the compressor for this - purpose.) The next operation on this stream must be inflateEnd to deallocate - the decompression state. The use of Z_FINISH is never required, but can be - used to inform inflate that a faster approach may be used for the single - inflate() call. - - In this implementation, inflate() always flushes as much output as - possible to the output buffer, and always uses the faster approach on the - first call. So the only effect of the flush parameter in this implementation - is on the return value of inflate(), as noted below, or when it returns early - because Z_BLOCK or Z_TREES is used. - - If a preset dictionary is needed after this call (see inflateSetDictionary - below), inflate sets strm->adler to the adler32 checksum of the dictionary - chosen by the compressor and returns Z_NEED_DICT; otherwise it sets - strm->adler to the adler32 checksum of all output produced so far (that is, - total_out bytes) and returns Z_OK, Z_STREAM_END or an error code as described - below. At the end of the stream, inflate() checks that its computed adler32 - checksum is equal to that saved by the compressor and returns Z_STREAM_END - only if the checksum is correct. - - inflate() can decompress and check either zlib-wrapped or gzip-wrapped - deflate data. The header type is detected automatically, if requested when - initializing with inflateInit2(). Any information contained in the gzip - header is not retained, so applications that need that information should - instead use raw inflate, see inflateInit2() below, or inflateBack() and - perform their own processing of the gzip header and trailer. - - inflate() returns Z_OK if some progress has been made (more input processed - or more output produced), Z_STREAM_END if the end of the compressed data has - been reached and all uncompressed output has been produced, Z_NEED_DICT if a - preset dictionary is needed at this point, Z_DATA_ERROR if the input data was - corrupted (input stream not conforming to the zlib format or incorrect check - value), Z_STREAM_ERROR if the stream structure was inconsistent (for example - next_in or next_out was Z_NULL), Z_MEM_ERROR if there was not enough memory, - Z_BUF_ERROR if no progress is possible or if there was not enough room in the - output buffer when Z_FINISH is used. Note that Z_BUF_ERROR is not fatal, and - inflate() can be called again with more input and more output space to - continue decompressing. If Z_DATA_ERROR is returned, the application may - then call inflateSync() to look for a good compression block if a partial - recovery of the data is desired. -*/ - - -ZEXTERN int ZEXPORT inflateEnd OF((z_streamp strm)); -/* - All dynamically allocated data structures for this stream are freed. - This function discards any unprocessed input and does not flush any pending - output. - - inflateEnd returns Z_OK if success, Z_STREAM_ERROR if the stream state - was inconsistent. In the error case, msg may be set but then points to a - static string (which must not be deallocated). -*/ - - - /* Advanced functions */ - -/* - The following functions are needed only in some special applications. -*/ - -/* -ZEXTERN int ZEXPORT deflateInit2 OF((z_streamp strm, - int level, - int method, - int windowBits, - int memLevel, - int strategy)); - - This is another version of deflateInit with more compression options. The - fields next_in, zalloc, zfree and opaque must be initialized before by the - caller. - - The method parameter is the compression method. It must be Z_DEFLATED in - this version of the library. - - The windowBits parameter is the base two logarithm of the window size - (the size of the history buffer). It should be in the range 8..15 for this - version of the library. Larger values of this parameter result in better - compression at the expense of memory usage. The default value is 15 if - deflateInit is used instead. - - windowBits can also be -8..-15 for raw deflate. In this case, -windowBits - determines the window size. deflate() will then generate raw deflate data - with no zlib header or trailer, and will not compute an adler32 check value. - - windowBits can also be greater than 15 for optional gzip encoding. Add - 16 to windowBits to write a simple gzip header and trailer around the - compressed data instead of a zlib wrapper. The gzip header will have no - file name, no extra data, no comment, no modification time (set to zero), no - header crc, and the operating system will be set to 255 (unknown). If a - gzip stream is being written, strm->adler is a crc32 instead of an adler32. - - The memLevel parameter specifies how much memory should be allocated - for the internal compression state. memLevel=1 uses minimum memory but is - slow and reduces compression ratio; memLevel=9 uses maximum memory for - optimal speed. The default value is 8. See zconf.h for total memory usage - as a function of windowBits and memLevel. - - The strategy parameter is used to tune the compression algorithm. Use the - value Z_DEFAULT_STRATEGY for normal data, Z_FILTERED for data produced by a - filter (or predictor), Z_HUFFMAN_ONLY to force Huffman encoding only (no - string match), or Z_RLE to limit match distances to one (run-length - encoding). Filtered data consists mostly of small values with a somewhat - random distribution. In this case, the compression algorithm is tuned to - compress them better. The effect of Z_FILTERED is to force more Huffman - coding and less string matching; it is somewhat intermediate between - Z_DEFAULT_STRATEGY and Z_HUFFMAN_ONLY. Z_RLE is designed to be almost as - fast as Z_HUFFMAN_ONLY, but give better compression for PNG image data. The - strategy parameter only affects the compression ratio but not the - correctness of the compressed output even if it is not set appropriately. - Z_FIXED prevents the use of dynamic Huffman codes, allowing for a simpler - decoder for special applications. - - deflateInit2 returns Z_OK if success, Z_MEM_ERROR if there was not enough - memory, Z_STREAM_ERROR if any parameter is invalid (such as an invalid - method), or Z_VERSION_ERROR if the zlib library version (zlib_version) is - incompatible with the version assumed by the caller (ZLIB_VERSION). msg is - set to null if there is no error message. deflateInit2 does not perform any - compression: this will be done by deflate(). -*/ - -ZEXTERN int ZEXPORT deflateSetDictionary OF((z_streamp strm, - const Bytef *dictionary, - uInt dictLength)); -/* - Initializes the compression dictionary from the given byte sequence - without producing any compressed output. This function must be called - immediately after deflateInit, deflateInit2 or deflateReset, before any call - of deflate. The compressor and decompressor must use exactly the same - dictionary (see inflateSetDictionary). - - The dictionary should consist of strings (byte sequences) that are likely - to be encountered later in the data to be compressed, with the most commonly - used strings preferably put towards the end of the dictionary. Using a - dictionary is most useful when the data to be compressed is short and can be - predicted with good accuracy; the data can then be compressed better than - with the default empty dictionary. - - Depending on the size of the compression data structures selected by - deflateInit or deflateInit2, a part of the dictionary may in effect be - discarded, for example if the dictionary is larger than the window size - provided in deflateInit or deflateInit2. Thus the strings most likely to be - useful should be put at the end of the dictionary, not at the front. In - addition, the current implementation of deflate will use at most the window - size minus 262 bytes of the provided dictionary. - - Upon return of this function, strm->adler is set to the adler32 value - of the dictionary; the decompressor may later use this value to determine - which dictionary has been used by the compressor. (The adler32 value - applies to the whole dictionary even if only a subset of the dictionary is - actually used by the compressor.) If a raw deflate was requested, then the - adler32 value is not computed and strm->adler is not set. - - deflateSetDictionary returns Z_OK if success, or Z_STREAM_ERROR if a - parameter is invalid (e.g. dictionary being Z_NULL) or the stream state is - inconsistent (for example if deflate has already been called for this stream - or if the compression method is bsort). deflateSetDictionary does not - perform any compression: this will be done by deflate(). -*/ - -ZEXTERN int ZEXPORT deflateCopy OF((z_streamp dest, - z_streamp source)); -/* - Sets the destination stream as a complete copy of the source stream. - - This function can be useful when several compression strategies will be - tried, for example when there are several ways of pre-processing the input - data with a filter. The streams that will be discarded should then be freed - by calling deflateEnd. Note that deflateCopy duplicates the internal - compression state which can be quite large, so this strategy is slow and can - consume lots of memory. - - deflateCopy returns Z_OK if success, Z_MEM_ERROR if there was not - enough memory, Z_STREAM_ERROR if the source stream state was inconsistent - (such as zalloc being Z_NULL). msg is left unchanged in both source and - destination. -*/ - -ZEXTERN int ZEXPORT deflateReset OF((z_streamp strm)); -/* - This function is equivalent to deflateEnd followed by deflateInit, - but does not free and reallocate all the internal compression state. The - stream will keep the same compression level and any other attributes that - may have been set by deflateInit2. - - deflateReset returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent (such as zalloc or state being Z_NULL). -*/ - -ZEXTERN int ZEXPORT deflateParams OF((z_streamp strm, - int level, - int strategy)); -/* - Dynamically update the compression level and compression strategy. The - interpretation of level and strategy is as in deflateInit2. This can be - used to switch between compression and straight copy of the input data, or - to switch to a different kind of input data requiring a different strategy. - If the compression level is changed, the input available so far is - compressed with the old level (and may be flushed); the new level will take - effect only at the next call of deflate(). - - Before the call of deflateParams, the stream state must be set as for - a call of deflate(), since the currently available input may have to be - compressed and flushed. In particular, strm->avail_out must be non-zero. - - deflateParams returns Z_OK if success, Z_STREAM_ERROR if the source - stream state was inconsistent or if a parameter was invalid, Z_BUF_ERROR if - strm->avail_out was zero. -*/ - -ZEXTERN int ZEXPORT deflateTune OF((z_streamp strm, - int good_length, - int max_lazy, - int nice_length, - int max_chain)); -/* - Fine tune deflate's internal compression parameters. This should only be - used by someone who understands the algorithm used by zlib's deflate for - searching for the best matching string, and even then only by the most - fanatic optimizer trying to squeeze out the last compressed bit for their - specific input data. Read the deflate.c source code for the meaning of the - max_lazy, good_length, nice_length, and max_chain parameters. - - deflateTune() can be called after deflateInit() or deflateInit2(), and - returns Z_OK on success, or Z_STREAM_ERROR for an invalid deflate stream. - */ - -ZEXTERN uLong ZEXPORT deflateBound OF((z_streamp strm, - uLong sourceLen)); -/* - deflateBound() returns an upper bound on the compressed size after - deflation of sourceLen bytes. It must be called after deflateInit() or - deflateInit2(), and after deflateSetHeader(), if used. This would be used - to allocate an output buffer for deflation in a single pass, and so would be - called before deflate(). -*/ - -ZEXTERN int ZEXPORT deflatePrime OF((z_streamp strm, - int bits, - int value)); -/* - deflatePrime() inserts bits in the deflate output stream. The intent - is that this function is used to start off the deflate output with the bits - leftover from a previous deflate stream when appending to it. As such, this - function can only be used for raw deflate, and must be used before the first - deflate() call after a deflateInit2() or deflateReset(). bits must be less - than or equal to 16, and that many of the least significant bits of value - will be inserted in the output. - - deflatePrime returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent. -*/ - -ZEXTERN int ZEXPORT deflateSetHeader OF((z_streamp strm, - gz_headerp head)); -/* - deflateSetHeader() provides gzip header information for when a gzip - stream is requested by deflateInit2(). deflateSetHeader() may be called - after deflateInit2() or deflateReset() and before the first call of - deflate(). The text, time, os, extra field, name, and comment information - in the provided gz_header structure are written to the gzip header (xflag is - ignored -- the extra flags are set according to the compression level). The - caller must assure that, if not Z_NULL, name and comment are terminated with - a zero byte, and that if extra is not Z_NULL, that extra_len bytes are - available there. If hcrc is true, a gzip header crc is included. Note that - the current versions of the command-line version of gzip (up through version - 1.3.x) do not support header crc's, and will report that it is a "multi-part - gzip file" and give up. - - If deflateSetHeader is not used, the default gzip header has text false, - the time set to zero, and os set to 255, with no extra, name, or comment - fields. The gzip header is returned to the default state by deflateReset(). - - deflateSetHeader returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent. -*/ - -/* -ZEXTERN int ZEXPORT inflateInit2 OF((z_streamp strm, - int windowBits)); - - This is another version of inflateInit with an extra parameter. The - fields next_in, avail_in, zalloc, zfree and opaque must be initialized - before by the caller. - - The windowBits parameter is the base two logarithm of the maximum window - size (the size of the history buffer). It should be in the range 8..15 for - this version of the library. The default value is 15 if inflateInit is used - instead. windowBits must be greater than or equal to the windowBits value - provided to deflateInit2() while compressing, or it must be equal to 15 if - deflateInit2() was not used. If a compressed stream with a larger window - size is given as input, inflate() will return with the error code - Z_DATA_ERROR instead of trying to allocate a larger window. - - windowBits can also be zero to request that inflate use the window size in - the zlib header of the compressed stream. - - windowBits can also be -8..-15 for raw inflate. In this case, -windowBits - determines the window size. inflate() will then process raw deflate data, - not looking for a zlib or gzip header, not generating a check value, and not - looking for any check values for comparison at the end of the stream. This - is for use with other formats that use the deflate compressed data format - such as zip. Those formats provide their own check values. If a custom - format is developed using the raw deflate format for compressed data, it is - recommended that a check value such as an adler32 or a crc32 be applied to - the uncompressed data as is done in the zlib, gzip, and zip formats. For - most applications, the zlib format should be used as is. Note that comments - above on the use in deflateInit2() applies to the magnitude of windowBits. - - windowBits can also be greater than 15 for optional gzip decoding. Add - 32 to windowBits to enable zlib and gzip decoding with automatic header - detection, or add 16 to decode only the gzip format (the zlib format will - return a Z_DATA_ERROR). If a gzip stream is being decoded, strm->adler is a - crc32 instead of an adler32. - - inflateInit2 returns Z_OK if success, Z_MEM_ERROR if there was not enough - memory, Z_VERSION_ERROR if the zlib library version is incompatible with the - version assumed by the caller, or Z_STREAM_ERROR if the parameters are - invalid, such as a null pointer to the structure. msg is set to null if - there is no error message. inflateInit2 does not perform any decompression - apart from possibly reading the zlib header if present: actual decompression - will be done by inflate(). (So next_in and avail_in may be modified, but - next_out and avail_out are unused and unchanged.) The current implementation - of inflateInit2() does not process any header information -- that is - deferred until inflate() is called. -*/ - -ZEXTERN int ZEXPORT inflateSetDictionary OF((z_streamp strm, - const Bytef *dictionary, - uInt dictLength)); -/* - Initializes the decompression dictionary from the given uncompressed byte - sequence. This function must be called immediately after a call of inflate, - if that call returned Z_NEED_DICT. The dictionary chosen by the compressor - can be determined from the adler32 value returned by that call of inflate. - The compressor and decompressor must use exactly the same dictionary (see - deflateSetDictionary). For raw inflate, this function can be called - immediately after inflateInit2() or inflateReset() and before any call of - inflate() to set the dictionary. The application must insure that the - dictionary that was used for compression is provided. - - inflateSetDictionary returns Z_OK if success, Z_STREAM_ERROR if a - parameter is invalid (e.g. dictionary being Z_NULL) or the stream state is - inconsistent, Z_DATA_ERROR if the given dictionary doesn't match the - expected one (incorrect adler32 value). inflateSetDictionary does not - perform any decompression: this will be done by subsequent calls of - inflate(). -*/ - -ZEXTERN int ZEXPORT inflateSync OF((z_streamp strm)); -/* - Skips invalid compressed data until a full flush point (see above the - description of deflate with Z_FULL_FLUSH) can be found, or until all - available input is skipped. No output is provided. - - inflateSync returns Z_OK if a full flush point has been found, Z_BUF_ERROR - if no more input was provided, Z_DATA_ERROR if no flush point has been - found, or Z_STREAM_ERROR if the stream structure was inconsistent. In the - success case, the application may save the current current value of total_in - which indicates where valid compressed data was found. In the error case, - the application may repeatedly call inflateSync, providing more input each - time, until success or end of the input data. -*/ - -ZEXTERN int ZEXPORT inflateCopy OF((z_streamp dest, - z_streamp source)); -/* - Sets the destination stream as a complete copy of the source stream. - - This function can be useful when randomly accessing a large stream. The - first pass through the stream can periodically record the inflate state, - allowing restarting inflate at those points when randomly accessing the - stream. - - inflateCopy returns Z_OK if success, Z_MEM_ERROR if there was not - enough memory, Z_STREAM_ERROR if the source stream state was inconsistent - (such as zalloc being Z_NULL). msg is left unchanged in both source and - destination. -*/ - -ZEXTERN int ZEXPORT inflateReset OF((z_streamp strm)); -/* - This function is equivalent to inflateEnd followed by inflateInit, - but does not free and reallocate all the internal decompression state. The - stream will keep attributes that may have been set by inflateInit2. - - inflateReset returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent (such as zalloc or state being Z_NULL). -*/ - -ZEXTERN int ZEXPORT inflateReset2 OF((z_streamp strm, - int windowBits)); -/* - This function is the same as inflateReset, but it also permits changing - the wrap and window size requests. The windowBits parameter is interpreted - the same as it is for inflateInit2. - - inflateReset2 returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent (such as zalloc or state being Z_NULL), or if - the windowBits parameter is invalid. -*/ - -ZEXTERN int ZEXPORT inflatePrime OF((z_streamp strm, - int bits, - int value)); -/* - This function inserts bits in the inflate input stream. The intent is - that this function is used to start inflating at a bit position in the - middle of a byte. The provided bits will be used before any bytes are used - from next_in. This function should only be used with raw inflate, and - should be used before the first inflate() call after inflateInit2() or - inflateReset(). bits must be less than or equal to 16, and that many of the - least significant bits of value will be inserted in the input. - - If bits is negative, then the input stream bit buffer is emptied. Then - inflatePrime() can be called again to put bits in the buffer. This is used - to clear out bits leftover after feeding inflate a block description prior - to feeding inflate codes. - - inflatePrime returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent. -*/ - -ZEXTERN long ZEXPORT inflateMark OF((z_streamp strm)); -/* - This function returns two values, one in the lower 16 bits of the return - value, and the other in the remaining upper bits, obtained by shifting the - return value down 16 bits. If the upper value is -1 and the lower value is - zero, then inflate() is currently decoding information outside of a block. - If the upper value is -1 and the lower value is non-zero, then inflate is in - the middle of a stored block, with the lower value equaling the number of - bytes from the input remaining to copy. If the upper value is not -1, then - it is the number of bits back from the current bit position in the input of - the code (literal or length/distance pair) currently being processed. In - that case the lower value is the number of bytes already emitted for that - code. - - A code is being processed if inflate is waiting for more input to complete - decoding of the code, or if it has completed decoding but is waiting for - more output space to write the literal or match data. - - inflateMark() is used to mark locations in the input data for random - access, which may be at bit positions, and to note those cases where the - output of a code may span boundaries of random access blocks. The current - location in the input stream can be determined from avail_in and data_type - as noted in the description for the Z_BLOCK flush parameter for inflate. - - inflateMark returns the value noted above or -1 << 16 if the provided - source stream state was inconsistent. -*/ - -ZEXTERN int ZEXPORT inflateGetHeader OF((z_streamp strm, - gz_headerp head)); -/* - inflateGetHeader() requests that gzip header information be stored in the - provided gz_header structure. inflateGetHeader() may be called after - inflateInit2() or inflateReset(), and before the first call of inflate(). - As inflate() processes the gzip stream, head->done is zero until the header - is completed, at which time head->done is set to one. If a zlib stream is - being decoded, then head->done is set to -1 to indicate that there will be - no gzip header information forthcoming. Note that Z_BLOCK or Z_TREES can be - used to force inflate() to return immediately after header processing is - complete and before any actual data is decompressed. - - The text, time, xflags, and os fields are filled in with the gzip header - contents. hcrc is set to true if there is a header CRC. (The header CRC - was valid if done is set to one.) If extra is not Z_NULL, then extra_max - contains the maximum number of bytes to write to extra. Once done is true, - extra_len contains the actual extra field length, and extra contains the - extra field, or that field truncated if extra_max is less than extra_len. - If name is not Z_NULL, then up to name_max characters are written there, - terminated with a zero unless the length is greater than name_max. If - comment is not Z_NULL, then up to comm_max characters are written there, - terminated with a zero unless the length is greater than comm_max. When any - of extra, name, or comment are not Z_NULL and the respective field is not - present in the header, then that field is set to Z_NULL to signal its - absence. This allows the use of deflateSetHeader() with the returned - structure to duplicate the header. However if those fields are set to - allocated memory, then the application will need to save those pointers - elsewhere so that they can be eventually freed. - - If inflateGetHeader is not used, then the header information is simply - discarded. The header is always checked for validity, including the header - CRC if present. inflateReset() will reset the process to discard the header - information. The application would need to call inflateGetHeader() again to - retrieve the header from the next gzip stream. - - inflateGetHeader returns Z_OK if success, or Z_STREAM_ERROR if the source - stream state was inconsistent. -*/ - -/* -ZEXTERN int ZEXPORT inflateBackInit OF((z_streamp strm, int windowBits, - unsigned char FAR *window)); - - Initialize the internal stream state for decompression using inflateBack() - calls. The fields zalloc, zfree and opaque in strm must be initialized - before the call. If zalloc and zfree are Z_NULL, then the default library- - derived memory allocation routines are used. windowBits is the base two - logarithm of the window size, in the range 8..15. window is a caller - supplied buffer of that size. Except for special applications where it is - assured that deflate was used with small window sizes, windowBits must be 15 - and a 32K byte window must be supplied to be able to decompress general - deflate streams. - - See inflateBack() for the usage of these routines. - - inflateBackInit will return Z_OK on success, Z_STREAM_ERROR if any of - the paramaters are invalid, Z_MEM_ERROR if the internal state could not be - allocated, or Z_VERSION_ERROR if the version of the library does not match - the version of the header file. -*/ - -typedef unsigned (*in_func) OF((void FAR *, unsigned char FAR * FAR *)); -typedef int (*out_func) OF((void FAR *, unsigned char FAR *, unsigned)); - -ZEXTERN int ZEXPORT inflateBack OF((z_streamp strm, - in_func in, void FAR *in_desc, - out_func out, void FAR *out_desc)); -/* - inflateBack() does a raw inflate with a single call using a call-back - interface for input and output. This is more efficient than inflate() for - file i/o applications in that it avoids copying between the output and the - sliding window by simply making the window itself the output buffer. This - function trusts the application to not change the output buffer passed by - the output function, at least until inflateBack() returns. - - inflateBackInit() must be called first to allocate the internal state - and to initialize the state with the user-provided window buffer. - inflateBack() may then be used multiple times to inflate a complete, raw - deflate stream with each call. inflateBackEnd() is then called to free the - allocated state. - - A raw deflate stream is one with no zlib or gzip header or trailer. - This routine would normally be used in a utility that reads zip or gzip - files and writes out uncompressed files. The utility would decode the - header and process the trailer on its own, hence this routine expects only - the raw deflate stream to decompress. This is different from the normal - behavior of inflate(), which expects either a zlib or gzip header and - trailer around the deflate stream. - - inflateBack() uses two subroutines supplied by the caller that are then - called by inflateBack() for input and output. inflateBack() calls those - routines until it reads a complete deflate stream and writes out all of the - uncompressed data, or until it encounters an error. The function's - parameters and return types are defined above in the in_func and out_func - typedefs. inflateBack() will call in(in_desc, &buf) which should return the - number of bytes of provided input, and a pointer to that input in buf. If - there is no input available, in() must return zero--buf is ignored in that - case--and inflateBack() will return a buffer error. inflateBack() will call - out(out_desc, buf, len) to write the uncompressed data buf[0..len-1]. out() - should return zero on success, or non-zero on failure. If out() returns - non-zero, inflateBack() will return with an error. Neither in() nor out() - are permitted to change the contents of the window provided to - inflateBackInit(), which is also the buffer that out() uses to write from. - The length written by out() will be at most the window size. Any non-zero - amount of input may be provided by in(). - - For convenience, inflateBack() can be provided input on the first call by - setting strm->next_in and strm->avail_in. If that input is exhausted, then - in() will be called. Therefore strm->next_in must be initialized before - calling inflateBack(). If strm->next_in is Z_NULL, then in() will be called - immediately for input. If strm->next_in is not Z_NULL, then strm->avail_in - must also be initialized, and then if strm->avail_in is not zero, input will - initially be taken from strm->next_in[0 .. strm->avail_in - 1]. - - The in_desc and out_desc parameters of inflateBack() is passed as the - first parameter of in() and out() respectively when they are called. These - descriptors can be optionally used to pass any information that the caller- - supplied in() and out() functions need to do their job. - - On return, inflateBack() will set strm->next_in and strm->avail_in to - pass back any unused input that was provided by the last in() call. The - return values of inflateBack() can be Z_STREAM_END on success, Z_BUF_ERROR - if in() or out() returned an error, Z_DATA_ERROR if there was a format error - in the deflate stream (in which case strm->msg is set to indicate the nature - of the error), or Z_STREAM_ERROR if the stream was not properly initialized. - In the case of Z_BUF_ERROR, an input or output error can be distinguished - using strm->next_in which will be Z_NULL only if in() returned an error. If - strm->next_in is not Z_NULL, then the Z_BUF_ERROR was due to out() returning - non-zero. (in() will always be called before out(), so strm->next_in is - assured to be defined if out() returns non-zero.) Note that inflateBack() - cannot return Z_OK. -*/ - -ZEXTERN int ZEXPORT inflateBackEnd OF((z_streamp strm)); -/* - All memory allocated by inflateBackInit() is freed. - - inflateBackEnd() returns Z_OK on success, or Z_STREAM_ERROR if the stream - state was inconsistent. -*/ - -ZEXTERN uLong ZEXPORT zlibCompileFlags OF((void)); -/* Return flags indicating compile-time options. - - Type sizes, two bits each, 00 = 16 bits, 01 = 32, 10 = 64, 11 = other: - 1.0: size of uInt - 3.2: size of uLong - 5.4: size of voidpf (pointer) - 7.6: size of z_off_t - - Compiler, assembler, and debug options: - 8: DEBUG - 9: ASMV or ASMINF -- use ASM code - 10: ZLIB_WINAPI -- exported functions use the WINAPI calling convention - 11: 0 (reserved) - - One-time table building (smaller code, but not thread-safe if true): - 12: BUILDFIXED -- build static block decoding tables when needed - 13: DYNAMIC_CRC_TABLE -- build CRC calculation tables when needed - 14,15: 0 (reserved) - - Library content (indicates missing functionality): - 16: NO_GZCOMPRESS -- gz* functions cannot compress (to avoid linking - deflate code when not needed) - 17: NO_GZIP -- deflate can't write gzip streams, and inflate can't detect - and decode gzip streams (to avoid linking crc code) - 18-19: 0 (reserved) - - Operation variations (changes in library functionality): - 20: PKZIP_BUG_WORKAROUND -- slightly more permissive inflate - 21: FASTEST -- deflate algorithm with only one, lowest compression level - 22,23: 0 (reserved) - - The sprintf variant used by gzprintf (zero is best): - 24: 0 = vs*, 1 = s* -- 1 means limited to 20 arguments after the format - 25: 0 = *nprintf, 1 = *printf -- 1 means gzprintf() not secure! - 26: 0 = returns value, 1 = void -- 1 means inferred string length returned - - Remainder: - 27-31: 0 (reserved) - */ - - - /* utility functions */ - -/* - The following utility functions are implemented on top of the basic - stream-oriented functions. To simplify the interface, some default options - are assumed (compression level and memory usage, standard memory allocation - functions). The source code of these utility functions can be modified if - you need special options. -*/ - -ZEXTERN int ZEXPORT compress OF((Bytef *dest, uLongf *destLen, - const Bytef *source, uLong sourceLen)); -/* - Compresses the source buffer into the destination buffer. sourceLen is - the byte length of the source buffer. Upon entry, destLen is the total size - of the destination buffer, which must be at least the value returned by - compressBound(sourceLen). Upon exit, destLen is the actual size of the - compressed buffer. - - compress returns Z_OK if success, Z_MEM_ERROR if there was not - enough memory, Z_BUF_ERROR if there was not enough room in the output - buffer. -*/ - -ZEXTERN int ZEXPORT compress2 OF((Bytef *dest, uLongf *destLen, - const Bytef *source, uLong sourceLen, - int level)); -/* - Compresses the source buffer into the destination buffer. The level - parameter has the same meaning as in deflateInit. sourceLen is the byte - length of the source buffer. Upon entry, destLen is the total size of the - destination buffer, which must be at least the value returned by - compressBound(sourceLen). Upon exit, destLen is the actual size of the - compressed buffer. - - compress2 returns Z_OK if success, Z_MEM_ERROR if there was not enough - memory, Z_BUF_ERROR if there was not enough room in the output buffer, - Z_STREAM_ERROR if the level parameter is invalid. -*/ - -ZEXTERN uLong ZEXPORT compressBound OF((uLong sourceLen)); -/* - compressBound() returns an upper bound on the compressed size after - compress() or compress2() on sourceLen bytes. It would be used before a - compress() or compress2() call to allocate the destination buffer. -*/ - -ZEXTERN int ZEXPORT uncompress OF((Bytef *dest, uLongf *destLen, - const Bytef *source, uLong sourceLen)); -/* - Decompresses the source buffer into the destination buffer. sourceLen is - the byte length of the source buffer. Upon entry, destLen is the total size - of the destination buffer, which must be large enough to hold the entire - uncompressed data. (The size of the uncompressed data must have been saved - previously by the compressor and transmitted to the decompressor by some - mechanism outside the scope of this compression library.) Upon exit, destLen - is the actual size of the uncompressed buffer. - - uncompress returns Z_OK if success, Z_MEM_ERROR if there was not - enough memory, Z_BUF_ERROR if there was not enough room in the output - buffer, or Z_DATA_ERROR if the input data was corrupted or incomplete. -*/ - - - /* gzip file access functions */ - -/* - This library supports reading and writing files in gzip (.gz) format with - an interface similar to that of stdio, using the functions that start with - "gz". The gzip format is different from the zlib format. gzip is a gzip - wrapper, documented in RFC 1952, wrapped around a deflate stream. -*/ - -typedef voidp gzFile; /* opaque gzip file descriptor */ - -/* -ZEXTERN gzFile ZEXPORT gzopen OF((const char *path, const char *mode)); - - Opens a gzip (.gz) file for reading or writing. The mode parameter is as - in fopen ("rb" or "wb") but can also include a compression level ("wb9") or - a strategy: 'f' for filtered data as in "wb6f", 'h' for Huffman-only - compression as in "wb1h", 'R' for run-length encoding as in "wb1R", or 'F' - for fixed code compression as in "wb9F". (See the description of - deflateInit2 for more information about the strategy parameter.) Also "a" - can be used instead of "w" to request that the gzip stream that will be - written be appended to the file. "+" will result in an error, since reading - and writing to the same gzip file is not supported. - - gzopen can be used to read a file which is not in gzip format; in this - case gzread will directly read from the file without decompression. - - gzopen returns NULL if the file could not be opened, if there was - insufficient memory to allocate the gzFile state, or if an invalid mode was - specified (an 'r', 'w', or 'a' was not provided, or '+' was provided). - errno can be checked to determine if the reason gzopen failed was that the - file could not be opened. -*/ - -ZEXTERN gzFile ZEXPORT gzdopen OF((int fd, const char *mode)); -/* - gzdopen associates a gzFile with the file descriptor fd. File descriptors - are obtained from calls like open, dup, creat, pipe or fileno (if the file - has been previously opened with fopen). The mode parameter is as in gzopen. - - The next call of gzclose on the returned gzFile will also close the file - descriptor fd, just like fclose(fdopen(fd, mode)) closes the file descriptor - fd. If you want to keep fd open, use fd = dup(fd_keep); gz = gzdopen(fd, - mode);. The duplicated descriptor should be saved to avoid a leak, since - gzdopen does not close fd if it fails. - - gzdopen returns NULL if there was insufficient memory to allocate the - gzFile state, if an invalid mode was specified (an 'r', 'w', or 'a' was not - provided, or '+' was provided), or if fd is -1. The file descriptor is not - used until the next gz* read, write, seek, or close operation, so gzdopen - will not detect if fd is invalid (unless fd is -1). -*/ - -ZEXTERN int ZEXPORT gzbuffer OF((gzFile file, unsigned size)); -/* - Set the internal buffer size used by this library's functions. The - default buffer size is 8192 bytes. This function must be called after - gzopen() or gzdopen(), and before any other calls that read or write the - file. The buffer memory allocation is always deferred to the first read or - write. Two buffers are allocated, either both of the specified size when - writing, or one of the specified size and the other twice that size when - reading. A larger buffer size of, for example, 64K or 128K bytes will - noticeably increase the speed of decompression (reading). - - The new buffer size also affects the maximum length for gzprintf(). - - gzbuffer() returns 0 on success, or -1 on failure, such as being called - too late. -*/ - -ZEXTERN int ZEXPORT gzsetparams OF((gzFile file, int level, int strategy)); -/* - Dynamically update the compression level or strategy. See the description - of deflateInit2 for the meaning of these parameters. - - gzsetparams returns Z_OK if success, or Z_STREAM_ERROR if the file was not - opened for writing. -*/ - -ZEXTERN int ZEXPORT gzread OF((gzFile file, voidp buf, unsigned len)); -/* - Reads the given number of uncompressed bytes from the compressed file. If - the input file was not in gzip format, gzread copies the given number of - bytes into the buffer. - - After reaching the end of a gzip stream in the input, gzread will continue - to read, looking for another gzip stream, or failing that, reading the rest - of the input file directly without decompression. The entire input file - will be read if gzread is called until it returns less than the requested - len. - - gzread returns the number of uncompressed bytes actually read, less than - len for end of file, or -1 for error. -*/ - -ZEXTERN int ZEXPORT gzwrite OF((gzFile file, - voidpc buf, unsigned len)); -/* - Writes the given number of uncompressed bytes into the compressed file. - gzwrite returns the number of uncompressed bytes written or 0 in case of - error. -*/ - -ZEXTERN int ZEXPORTVA gzprintf OF((gzFile file, const char *format, ...)); -/* - Converts, formats, and writes the arguments to the compressed file under - control of the format string, as in fprintf. gzprintf returns the number of - uncompressed bytes actually written, or 0 in case of error. The number of - uncompressed bytes written is limited to 8191, or one less than the buffer - size given to gzbuffer(). The caller should assure that this limit is not - exceeded. If it is exceeded, then gzprintf() will return an error (0) with - nothing written. In this case, there may also be a buffer overflow with - unpredictable consequences, which is possible only if zlib was compiled with - the insecure functions sprintf() or vsprintf() because the secure snprintf() - or vsnprintf() functions were not available. This can be determined using - zlibCompileFlags(). -*/ - -ZEXTERN int ZEXPORT gzputs OF((gzFile file, const char *s)); -/* - Writes the given null-terminated string to the compressed file, excluding - the terminating null character. - - gzputs returns the number of characters written, or -1 in case of error. -*/ - -ZEXTERN char * ZEXPORT gzgets OF((gzFile file, char *buf, int len)); -/* - Reads bytes from the compressed file until len-1 characters are read, or a - newline character is read and transferred to buf, or an end-of-file - condition is encountered. If any characters are read or if len == 1, the - string is terminated with a null character. If no characters are read due - to an end-of-file or len < 1, then the buffer is left untouched. - - gzgets returns buf which is a null-terminated string, or it returns NULL - for end-of-file or in case of error. If there was an error, the contents at - buf are indeterminate. -*/ - -ZEXTERN int ZEXPORT gzputc OF((gzFile file, int c)); -/* - Writes c, converted to an unsigned char, into the compressed file. gzputc - returns the value that was written, or -1 in case of error. -*/ - -ZEXTERN int ZEXPORT gzgetc OF((gzFile file)); -/* - Reads one byte from the compressed file. gzgetc returns this byte or -1 - in case of end of file or error. -*/ - -ZEXTERN int ZEXPORT gzungetc OF((int c, gzFile file)); -/* - Push one character back onto the stream to be read as the first character - on the next read. At least one character of push-back is allowed. - gzungetc() returns the character pushed, or -1 on failure. gzungetc() will - fail if c is -1, and may fail if a character has been pushed but not read - yet. If gzungetc is used immediately after gzopen or gzdopen, at least the - output buffer size of pushed characters is allowed. (See gzbuffer above.) - The pushed character will be discarded if the stream is repositioned with - gzseek() or gzrewind(). -*/ - -ZEXTERN int ZEXPORT gzflush OF((gzFile file, int flush)); -/* - Flushes all pending output into the compressed file. The parameter flush - is as in the deflate() function. The return value is the zlib error number - (see function gzerror below). gzflush is only permitted when writing. - - If the flush parameter is Z_FINISH, the remaining data is written and the - gzip stream is completed in the output. If gzwrite() is called again, a new - gzip stream will be started in the output. gzread() is able to read such - concatented gzip streams. - - gzflush should be called only when strictly necessary because it will - degrade compression if called too often. -*/ - -/* -ZEXTERN z_off_t ZEXPORT gzseek OF((gzFile file, - z_off_t offset, int whence)); - - Sets the starting position for the next gzread or gzwrite on the given - compressed file. The offset represents a number of bytes in the - uncompressed data stream. The whence parameter is defined as in lseek(2); - the value SEEK_END is not supported. - - If the file is opened for reading, this function is emulated but can be - extremely slow. If the file is opened for writing, only forward seeks are - supported; gzseek then compresses a sequence of zeroes up to the new - starting position. - - gzseek returns the resulting offset location as measured in bytes from - the beginning of the uncompressed stream, or -1 in case of error, in - particular if the file is opened for writing and the new starting position - would be before the current position. -*/ - -ZEXTERN int ZEXPORT gzrewind OF((gzFile file)); -/* - Rewinds the given file. This function is supported only for reading. - - gzrewind(file) is equivalent to (int)gzseek(file, 0L, SEEK_SET) -*/ - -/* -ZEXTERN z_off_t ZEXPORT gztell OF((gzFile file)); - - Returns the starting position for the next gzread or gzwrite on the given - compressed file. This position represents a number of bytes in the - uncompressed data stream, and is zero when starting, even if appending or - reading a gzip stream from the middle of a file using gzdopen(). - - gztell(file) is equivalent to gzseek(file, 0L, SEEK_CUR) -*/ - -/* -ZEXTERN z_off_t ZEXPORT gzoffset OF((gzFile file)); - - Returns the current offset in the file being read or written. This offset - includes the count of bytes that precede the gzip stream, for example when - appending or when using gzdopen() for reading. When reading, the offset - does not include as yet unused buffered input. This information can be used - for a progress indicator. On error, gzoffset() returns -1. -*/ - -ZEXTERN int ZEXPORT gzeof OF((gzFile file)); -/* - Returns true (1) if the end-of-file indicator has been set while reading, - false (0) otherwise. Note that the end-of-file indicator is set only if the - read tried to go past the end of the input, but came up short. Therefore, - just like feof(), gzeof() may return false even if there is no more data to - read, in the event that the last read request was for the exact number of - bytes remaining in the input file. This will happen if the input file size - is an exact multiple of the buffer size. - - If gzeof() returns true, then the read functions will return no more data, - unless the end-of-file indicator is reset by gzclearerr() and the input file - has grown since the previous end of file was detected. -*/ - -ZEXTERN int ZEXPORT gzdirect OF((gzFile file)); -/* - Returns true (1) if file is being copied directly while reading, or false - (0) if file is a gzip stream being decompressed. This state can change from - false to true while reading the input file if the end of a gzip stream is - reached, but is followed by data that is not another gzip stream. - - If the input file is empty, gzdirect() will return true, since the input - does not contain a gzip stream. - - If gzdirect() is used immediately after gzopen() or gzdopen() it will - cause buffers to be allocated to allow reading the file to determine if it - is a gzip file. Therefore if gzbuffer() is used, it should be called before - gzdirect(). -*/ - -ZEXTERN int ZEXPORT gzclose OF((gzFile file)); -/* - Flushes all pending output if necessary, closes the compressed file and - deallocates the (de)compression state. Note that once file is closed, you - cannot call gzerror with file, since its structures have been deallocated. - gzclose must not be called more than once on the same file, just as free - must not be called more than once on the same allocation. - - gzclose will return Z_STREAM_ERROR if file is not valid, Z_ERRNO on a - file operation error, or Z_OK on success. -*/ - -ZEXTERN int ZEXPORT gzclose_r OF((gzFile file)); -ZEXTERN int ZEXPORT gzclose_w OF((gzFile file)); -/* - Same as gzclose(), but gzclose_r() is only for use when reading, and - gzclose_w() is only for use when writing or appending. The advantage to - using these instead of gzclose() is that they avoid linking in zlib - compression or decompression code that is not used when only reading or only - writing respectively. If gzclose() is used, then both compression and - decompression code will be included the application when linking to a static - zlib library. -*/ - -ZEXTERN const char * ZEXPORT gzerror OF((gzFile file, int *errnum)); -/* - Returns the error message for the last error which occurred on the given - compressed file. errnum is set to zlib error number. If an error occurred - in the file system and not in the compression library, errnum is set to - Z_ERRNO and the application may consult errno to get the exact error code. - - The application must not modify the returned string. Future calls to - this function may invalidate the previously returned string. If file is - closed, then the string previously returned by gzerror will no longer be - available. - - gzerror() should be used to distinguish errors from end-of-file for those - functions above that do not distinguish those cases in their return values. -*/ - -ZEXTERN void ZEXPORT gzclearerr OF((gzFile file)); -/* - Clears the error and end-of-file flags for file. This is analogous to the - clearerr() function in stdio. This is useful for continuing to read a gzip - file that is being written concurrently. -*/ - - - /* checksum functions */ - -/* - These functions are not related to compression but are exported - anyway because they might be useful in applications using the compression - library. -*/ - -ZEXTERN uLong ZEXPORT adler32 OF((uLong adler, const Bytef *buf, uInt len)); -/* - Update a running Adler-32 checksum with the bytes buf[0..len-1] and - return the updated checksum. If buf is Z_NULL, this function returns the - required initial value for the checksum. - - An Adler-32 checksum is almost as reliable as a CRC32 but can be computed - much faster. - - Usage example: - - uLong adler = adler32(0L, Z_NULL, 0); - - while (read_buffer(buffer, length) != EOF) { - adler = adler32(adler, buffer, length); - } - if (adler != original_adler) error(); -*/ - -/* -ZEXTERN uLong ZEXPORT adler32_combine OF((uLong adler1, uLong adler2, - z_off_t len2)); - - Combine two Adler-32 checksums into one. For two sequences of bytes, seq1 - and seq2 with lengths len1 and len2, Adler-32 checksums were calculated for - each, adler1 and adler2. adler32_combine() returns the Adler-32 checksum of - seq1 and seq2 concatenated, requiring only adler1, adler2, and len2. -*/ - -ZEXTERN uLong ZEXPORT crc32 OF((uLong crc, const Bytef *buf, uInt len)); -/* - Update a running CRC-32 with the bytes buf[0..len-1] and return the - updated CRC-32. If buf is Z_NULL, this function returns the required - initial value for the for the crc. Pre- and post-conditioning (one's - complement) is performed within this function so it shouldn't be done by the - application. - - Usage example: - - uLong crc = crc32(0L, Z_NULL, 0); - - while (read_buffer(buffer, length) != EOF) { - crc = crc32(crc, buffer, length); - } - if (crc != original_crc) error(); -*/ - -/* -ZEXTERN uLong ZEXPORT crc32_combine OF((uLong crc1, uLong crc2, z_off_t len2)); - - Combine two CRC-32 check values into one. For two sequences of bytes, - seq1 and seq2 with lengths len1 and len2, CRC-32 check values were - calculated for each, crc1 and crc2. crc32_combine() returns the CRC-32 - check value of seq1 and seq2 concatenated, requiring only crc1, crc2, and - len2. -*/ - - - /* various hacks, don't look :) */ - -/* deflateInit and inflateInit are macros to allow checking the zlib version - * and the compiler's view of z_stream: - */ -ZEXTERN int ZEXPORT deflateInit_ OF((z_streamp strm, int level, - const char *version, int stream_size)); -ZEXTERN int ZEXPORT inflateInit_ OF((z_streamp strm, - const char *version, int stream_size)); -ZEXTERN int ZEXPORT deflateInit2_ OF((z_streamp strm, int level, int method, - int windowBits, int memLevel, - int strategy, const char *version, - int stream_size)); -ZEXTERN int ZEXPORT inflateInit2_ OF((z_streamp strm, int windowBits, - const char *version, int stream_size)); -ZEXTERN int ZEXPORT inflateBackInit_ OF((z_streamp strm, int windowBits, - unsigned char FAR *window, - const char *version, - int stream_size)); -#define deflateInit(strm, level) \ - deflateInit_((strm), (level), ZLIB_VERSION, sizeof(z_stream)) -#define inflateInit(strm) \ - inflateInit_((strm), ZLIB_VERSION, sizeof(z_stream)) -#define deflateInit2(strm, level, method, windowBits, memLevel, strategy) \ - deflateInit2_((strm),(level),(method),(windowBits),(memLevel),\ - (strategy), ZLIB_VERSION, sizeof(z_stream)) -#define inflateInit2(strm, windowBits) \ - inflateInit2_((strm), (windowBits), ZLIB_VERSION, sizeof(z_stream)) -#define inflateBackInit(strm, windowBits, window) \ - inflateBackInit_((strm), (windowBits), (window), \ - ZLIB_VERSION, sizeof(z_stream)) - -/* provide 64-bit offset functions if _LARGEFILE64_SOURCE defined, and/or - * change the regular functions to 64 bits if _FILE_OFFSET_BITS is 64 (if - * both are true, the application gets the *64 functions, and the regular - * functions are changed to 64 bits) -- in case these are set on systems - * without large file support, _LFS64_LARGEFILE must also be true - */ -#if defined(_LARGEFILE64_SOURCE) && _LFS64_LARGEFILE-0 - ZEXTERN gzFile ZEXPORT gzopen64 OF((const char *, const char *)); - ZEXTERN z_off64_t ZEXPORT gzseek64 OF((gzFile, z_off64_t, int)); - ZEXTERN z_off64_t ZEXPORT gztell64 OF((gzFile)); - ZEXTERN z_off64_t ZEXPORT gzoffset64 OF((gzFile)); - ZEXTERN uLong ZEXPORT adler32_combine64 OF((uLong, uLong, z_off64_t)); - ZEXTERN uLong ZEXPORT crc32_combine64 OF((uLong, uLong, z_off64_t)); -#endif - -#if !defined(ZLIB_INTERNAL) && _FILE_OFFSET_BITS-0 == 64 && _LFS64_LARGEFILE-0 -# define gzopen gzopen64 -# define gzseek gzseek64 -# define gztell gztell64 -# define gzoffset gzoffset64 -# define adler32_combine adler32_combine64 -# define crc32_combine crc32_combine64 -# ifdef _LARGEFILE64_SOURCE - ZEXTERN gzFile ZEXPORT gzopen64 OF((const char *, const char *)); - ZEXTERN z_off_t ZEXPORT gzseek64 OF((gzFile, z_off_t, int)); - ZEXTERN z_off_t ZEXPORT gztell64 OF((gzFile)); - ZEXTERN z_off_t ZEXPORT gzoffset64 OF((gzFile)); - ZEXTERN uLong ZEXPORT adler32_combine64 OF((uLong, uLong, z_off_t)); - ZEXTERN uLong ZEXPORT crc32_combine64 OF((uLong, uLong, z_off_t)); -# endif -#else - ZEXTERN gzFile ZEXPORT gzopen OF((const char *, const char *)); - ZEXTERN z_off_t ZEXPORT gzseek OF((gzFile, z_off_t, int)); - ZEXTERN z_off_t ZEXPORT gztell OF((gzFile)); - ZEXTERN z_off_t ZEXPORT gzoffset OF((gzFile)); - ZEXTERN uLong ZEXPORT adler32_combine OF((uLong, uLong, z_off_t)); - ZEXTERN uLong ZEXPORT crc32_combine OF((uLong, uLong, z_off_t)); -#endif - -/* hack for buggy compilers */ -#if !defined(ZUTIL_H) && !defined(NO_DUMMY_DECL) - struct internal_state {int dummy;}; -#endif - -/* undocumented functions */ -ZEXTERN const char * ZEXPORT zError OF((int)); -ZEXTERN int ZEXPORT inflateSyncPoint OF((z_streamp)); -ZEXTERN const uLongf * ZEXPORT get_crc_table OF((void)); -ZEXTERN int ZEXPORT inflateUndermine OF((z_streamp, int)); - -#ifdef __cplusplus -} -#endif - -#endif /* ZLIB_H */ diff --git a/code/3rd_glpk/zlib/zutil.c b/code/3rd_glpk/zlib/zutil.c deleted file mode 100644 index 898ed345..00000000 --- a/code/3rd_glpk/zlib/zutil.c +++ /dev/null @@ -1,318 +0,0 @@ -/* zutil.c -- target dependent utility functions for the compression library - * Copyright (C) 1995-2005, 2010 Jean-loup Gailly. - * For conditions of distribution and use, see copyright notice in zlib.h - */ - -/* @(#) $Id$ */ - -#include "zutil.h" - -#ifndef NO_DUMMY_DECL -struct internal_state {int dummy;}; /* for buggy compilers */ -#endif - -const char * const z_errmsg[10] = { -"need dictionary", /* Z_NEED_DICT 2 */ -"stream end", /* Z_STREAM_END 1 */ -"", /* Z_OK 0 */ -"file error", /* Z_ERRNO (-1) */ -"stream error", /* Z_STREAM_ERROR (-2) */ -"data error", /* Z_DATA_ERROR (-3) */ -"insufficient memory", /* Z_MEM_ERROR (-4) */ -"buffer error", /* Z_BUF_ERROR (-5) */ -"incompatible version",/* Z_VERSION_ERROR (-6) */ -""}; - - -const char * ZEXPORT zlibVersion() -{ - return ZLIB_VERSION; -} - -uLong ZEXPORT zlibCompileFlags() -{ - uLong flags; - - flags = 0; - switch ((int)(sizeof(uInt))) { - case 2: break; - case 4: flags += 1; break; - case 8: flags += 2; break; - default: flags += 3; - } - switch ((int)(sizeof(uLong))) { - case 2: break; - case 4: flags += 1 << 2; break; - case 8: flags += 2 << 2; break; - default: flags += 3 << 2; - } - switch ((int)(sizeof(voidpf))) { - case 2: break; - case 4: flags += 1 << 4; break; - case 8: flags += 2 << 4; break; - default: flags += 3 << 4; - } - switch ((int)(sizeof(z_off_t))) { - case 2: break; - case 4: flags += 1 << 6; break; - case 8: flags += 2 << 6; break; - default: flags += 3 << 6; - } -#ifdef DEBUG - flags += 1 << 8; -#endif -#if defined(ASMV) || defined(ASMINF) - flags += 1 << 9; -#endif -#ifdef ZLIB_WINAPI - flags += 1 << 10; -#endif -#ifdef BUILDFIXED - flags += 1 << 12; -#endif -#ifdef DYNAMIC_CRC_TABLE - flags += 1 << 13; -#endif -#ifdef NO_GZCOMPRESS - flags += 1L << 16; -#endif -#ifdef NO_GZIP - flags += 1L << 17; -#endif -#ifdef PKZIP_BUG_WORKAROUND - flags += 1L << 20; -#endif -#ifdef FASTEST - flags += 1L << 21; -#endif -#ifdef STDC -# ifdef NO_vsnprintf - flags += 1L << 25; -# ifdef HAS_vsprintf_void - flags += 1L << 26; -# endif -# else -# ifdef HAS_vsnprintf_void - flags += 1L << 26; -# endif -# endif -#else - flags += 1L << 24; -# ifdef NO_snprintf - flags += 1L << 25; -# ifdef HAS_sprintf_void - flags += 1L << 26; -# endif -# else -# ifdef HAS_snprintf_void - flags += 1L << 26; -# endif -# endif -#endif - return flags; -} - -#ifdef DEBUG - -# ifndef verbose -# define verbose 0 -# endif -int ZLIB_INTERNAL z_verbose = verbose; - -void ZLIB_INTERNAL z_error (m) - char *m; -{ - fprintf(stderr, "%s\n", m); - exit(1); -} -#endif - -/* exported to allow conversion of error code to string for compress() and - * uncompress() - */ -const char * ZEXPORT zError(err) - int err; -{ - return ERR_MSG(err); -} - -#if defined(_WIN32_WCE) - /* The Microsoft C Run-Time Library for Windows CE doesn't have - * errno. We define it as a global variable to simplify porting. - * Its value is always 0 and should not be used. - */ - int errno = 0; -#endif - -#ifndef HAVE_MEMCPY - -void ZLIB_INTERNAL zmemcpy(dest, source, len) - Bytef* dest; - const Bytef* source; - uInt len; -{ - if (len == 0) return; - do { - *dest++ = *source++; /* ??? to be unrolled */ - } while (--len != 0); -} - -int ZLIB_INTERNAL zmemcmp(s1, s2, len) - const Bytef* s1; - const Bytef* s2; - uInt len; -{ - uInt j; - - for (j = 0; j < len; j++) { - if (s1[j] != s2[j]) return 2*(s1[j] > s2[j])-1; - } - return 0; -} - -void ZLIB_INTERNAL zmemzero(dest, len) - Bytef* dest; - uInt len; -{ - if (len == 0) return; - do { - *dest++ = 0; /* ??? to be unrolled */ - } while (--len != 0); -} -#endif - - -#ifdef SYS16BIT - -#ifdef __TURBOC__ -/* Turbo C in 16-bit mode */ - -# define MY_ZCALLOC - -/* Turbo C malloc() does not allow dynamic allocation of 64K bytes - * and farmalloc(64K) returns a pointer with an offset of 8, so we - * must fix the pointer. Warning: the pointer must be put back to its - * original form in order to free it, use zcfree(). - */ - -#define MAX_PTR 10 -/* 10*64K = 640K */ - -local int next_ptr = 0; - -typedef struct ptr_table_s { - voidpf org_ptr; - voidpf new_ptr; -} ptr_table; - -local ptr_table table[MAX_PTR]; -/* This table is used to remember the original form of pointers - * to large buffers (64K). Such pointers are normalized with a zero offset. - * Since MSDOS is not a preemptive multitasking OS, this table is not - * protected from concurrent access. This hack doesn't work anyway on - * a protected system like OS/2. Use Microsoft C instead. - */ - -voidpf ZLIB_INTERNAL zcalloc (voidpf opaque, unsigned items, unsigned size) -{ - voidpf buf = opaque; /* just to make some compilers happy */ - ulg bsize = (ulg)items*size; - - /* If we allocate less than 65520 bytes, we assume that farmalloc - * will return a usable pointer which doesn't have to be normalized. - */ - if (bsize < 65520L) { - buf = farmalloc(bsize); - if (*(ush*)&buf != 0) return buf; - } else { - buf = farmalloc(bsize + 16L); - } - if (buf == NULL || next_ptr >= MAX_PTR) return NULL; - table[next_ptr].org_ptr = buf; - - /* Normalize the pointer to seg:0 */ - *((ush*)&buf+1) += ((ush)((uch*)buf-0) + 15) >> 4; - *(ush*)&buf = 0; - table[next_ptr++].new_ptr = buf; - return buf; -} - -void ZLIB_INTERNAL zcfree (voidpf opaque, voidpf ptr) -{ - int n; - if (*(ush*)&ptr != 0) { /* object < 64K */ - farfree(ptr); - return; - } - /* Find the original pointer */ - for (n = 0; n < next_ptr; n++) { - if (ptr != table[n].new_ptr) continue; - - farfree(table[n].org_ptr); - while (++n < next_ptr) { - table[n-1] = table[n]; - } - next_ptr--; - return; - } - ptr = opaque; /* just to make some compilers happy */ - Assert(0, "zcfree: ptr not found"); -} - -#endif /* __TURBOC__ */ - - -#ifdef M_I86 -/* Microsoft C in 16-bit mode */ - -# define MY_ZCALLOC - -#if (!defined(_MSC_VER) || (_MSC_VER <= 600)) -# define _halloc halloc -# define _hfree hfree -#endif - -voidpf ZLIB_INTERNAL zcalloc (voidpf opaque, uInt items, uInt size) -{ - if (opaque) opaque = 0; /* to make compiler happy */ - return _halloc((long)items, size); -} - -void ZLIB_INTERNAL zcfree (voidpf opaque, voidpf ptr) -{ - if (opaque) opaque = 0; /* to make compiler happy */ - _hfree(ptr); -} - -#endif /* M_I86 */ - -#endif /* SYS16BIT */ - - -#ifndef MY_ZCALLOC /* Any system without a special alloc function */ - -#ifndef STDC -extern voidp malloc OF((uInt size)); -extern voidp calloc OF((uInt items, uInt size)); -extern void free OF((voidpf ptr)); -#endif - -voidpf ZLIB_INTERNAL zcalloc (opaque, items, size) - voidpf opaque; - unsigned items; - unsigned size; -{ - if (opaque) items += size - size; /* make compiler happy */ - return sizeof(uInt) > 2 ? (voidpf)malloc(items * size) : - (voidpf)calloc(items, size); -} - -void ZLIB_INTERNAL zcfree (opaque, ptr) - voidpf opaque; - voidpf ptr; -{ - free(ptr); - if (opaque) return; /* make compiler happy */ -} - -#endif /* MY_ZCALLOC */ diff --git a/code/3rd_glpk/zlib/zutil.h b/code/3rd_glpk/zlib/zutil.h deleted file mode 100644 index 737bd38f..00000000 --- a/code/3rd_glpk/zlib/zutil.h +++ /dev/null @@ -1,93 +0,0 @@ -/* zutil.h (internal interface of the zlib compression library) */ - -/* Modified by Andrew Makhorin , April 2011 */ - -/* Copyright (C) 1995-2010 Jean-loup Gailly - * For conditions of distribution and use, see copyright notice in - * zlib.h */ - -/* WARNING: this file should *not* be used by applications. It is - part of the implementation of the compression library and is - subject to change. Applications should only use zlib.h. */ - -#ifndef ZUTIL_H -#define ZUTIL_H - -#define ZLIB_INTERNAL - -#include "zlib.h" - -#include -#include -#include - -#define local static - -typedef unsigned char uch; -typedef uch uchf; -typedef unsigned short ush; -typedef ush ushf; -typedef unsigned long ulg; - -extern const char * const z_errmsg[10]; - -#define ERR_MSG(err) z_errmsg[Z_NEED_DICT-(err)] - -#define ERR_RETURN(strm, err) \ - return (strm->msg = (char *)ERR_MSG(err), (err)) - -#define DEF_WBITS MAX_WBITS - -#if MAX_MEM_LEVEL >= 8 -#define DEF_MEM_LEVEL 8 -#else -#define DEF_MEM_LEVEL MAX_MEM_LEVEL -#endif - -#define STORED_BLOCK 0 -#define STATIC_TREES 1 -#define DYN_TREES 2 - -#define MIN_MATCH 3 -#define MAX_MATCH 258 - -#define PRESET_DICT 0x20 - -#define OS_CODE 0x03 /* assume Unix */ - -#define HAVE_MEMCPY 1 -#define zmemcpy memcpy -#define zmemzero(dest, len) memset(dest, 0, len) - -#ifdef DEBUG -#include -extern int ZLIB_INTERNAL z_verbose; -extern void ZLIB_INTERNAL z_error OF((char *m)); -#define Assert(cond, msg) { if(!(cond)) z_error(msg); } -#define Trace(x) { if (z_verbose >= 0) fprintf x; } -#define Tracev(x) { if (z_verbose > 0) fprintf x; } -#define Tracevv(x) {if (z_verbose > 1) fprintf x; } -#define Tracec(c, x) {if (z_verbose > 0 && (c)) fprintf x; } -#define Tracecv(c, x) {if (z_verbose > 1 && (c)) fprintf x; } -#else -#define Assert(cond, msg) -#define Trace(x) -#define Tracev(x) -#define Tracevv(x) -#define Tracec(c, x) -#define Tracecv(c, x) -#endif - -voidpf ZLIB_INTERNAL zcalloc OF((voidpf opaque, unsigned items, - unsigned size)); -void ZLIB_INTERNAL zcfree OF((voidpf opaque, voidpf ptr)); - -#define ZALLOC(strm, items, size) \ - (*((strm)->zalloc))((strm)->opaque, (items), (size)) -#define ZFREE(strm, addr) \ - (*((strm)->zfree))((strm)->opaque, (voidpf)(addr)) -#define TRY_FREE(s, p) { if (p) ZFREE(s, p); } - -#endif - -/* eof */ diff --git a/code/3rd_lpsolve/CMakeLists.txt b/code/3rd_lpsolve/CMakeLists.txt deleted file mode 100644 index c5b8eed2..00000000 --- a/code/3rd_lpsolve/CMakeLists.txt +++ /dev/null @@ -1,100 +0,0 @@ -get_filename_component(PROJECT_NAME ${CMAKE_CURRENT_SOURCE_DIR} NAME) -project(${PROJECT_NAME}) - - -set(lpsolve_HEADERS - fortify.h - ini.h - lp_bit.h - lp_crash.h - lp_explicit.h - lp_Hash.h - lp_lib.h - lp_matrix.h - lp_MDO.h - lp_mipbb.h - lp_MPS.h - lp_presolve.h - lp_price.h - lp_pricePSE.h - lp_report.h - lp_rlp.h - lp_scale.h - lp_simplex.h - lp_SOS.h - lp_types.h - lp_utils.h - lp_wlp.h - lpkit.h - yacc_read.h - shared/commonlib.h - shared/mmio.h - shared/myblas.h - bfp/lp_BFP.h - bfp/bfp_LUSOL/LUSOL/lusol.h - bfp/bfp_LUSOL/bfp_LUSOL.h - bfp/bfp_LUSOL/lp_LUSOL.h - colamd/colamd.h - ) - -set(lpsolve_SOURCES - fortify.c - ini.c - lp_crash.c - lp_Hash.c - lp_lib.c - lp_matrix.c - lp_MDO.c - lp_mipbb.c - lp_MPS.c - lp_params.c - lp_presolve.c - lp_price.c - lp_pricePSE.c - lp_report.c - lp_rlp.c - lp_scale.c - lp_simplex.c - lp_SOS.c - lp_utils.c - lp_wlp.c - yacc_read.c - shared/commonlib.c - shared/mmio.c - shared/myblas.c - bfp/bfp_LUSOL/LUSOL/lusol.c -# bfp/bfp_LUSOL/LUSOL/lusol1.c -# bfp/bfp_LUSOL/LUSOL/lusol2.c -# bfp/bfp_LUSOL/LUSOL/lusol6a.c -# bfp/bfp_LUSOL/LUSOL/lusol6l0.c -# bfp/bfp_LUSOL/LUSOL/lusol6u.c -# bfp/bfp_LUSOL/LUSOL/lusol7a.c -# bfp/bfp_LUSOL/LUSOL/lusol8a.c - bfp/bfp_LUSOL/lp_LUSOL.c - colamd/colamd.c - ) - - -add_library(${PROJECT_NAME} STATIC ${lpsolve_SOURCES} ${lpsolve_HEADERS}) -set_target_properties(${PROJECT_NAME} PROPERTIES - FOLDER "3rd_party") - -set_target_properties(${PROJECT_NAME} PROPERTIES POSITION_INDEPENDENT_CODE ON) - -target_include_directories(${PROJECT_NAME} PRIVATE - ${POLYFIT_lpsolve_DIR} - ${POLYFIT_lpsolve_DIR}/shared - ${POLYFIT_lpsolve_DIR}/bfp - ${POLYFIT_lpsolve_DIR}/bfp/bfp_LUSOL - ${POLYFIT_lpsolve_DIR}/bfp/bfp_LUSOL/LUSOL - ${POLYFIT_lpsolve_DIR}/colamd - ) - -target_compile_definitions(${PROJECT_NAME} PRIVATE CHECK_SOLUTION YY_NEVER_INTERACTIVE PARSER_LP INVERSE_ACTIVE=INVERSE_LUSOL RoleIsExternalInvEngine) - -if (MSVC) - target_compile_definitions(${PROJECT_NAME} PRIVATE - _CRT_SECURE_NO_WARNINGS - _CRT_SECURE_NO_DEPRECATE - ) -endif() diff --git a/code/3rd_lpsolve/README.txt b/code/3rd_lpsolve/README.txt deleted file mode 100644 index 7304d3eb..00000000 --- a/code/3rd_lpsolve/README.txt +++ /dev/null @@ -1,527 +0,0 @@ -Introduction ------------- -What is lp_solve and what is it not? -The simple answer is, lp_solve is a Mixed Integer Linear Programming (MILP) solver. - -It is a free (see LGPL for the GNU lesser general public license) linear (integer) programming solver -based on the revised simplex method and the Branch-and-bound method for the integers. -It contains full source, examples and manuals. -lp_solve solves pure linear, (mixed) integer/binary, semi-continuous and -special ordered sets (SOS) models. - -See the reference guide for more information. - - -lp_solve 5.5 ------------- - -Why a jump from version numbers 5.1 to 5.5 ? -This is done to indicate that this is more than just another update. -The solver engine was revised and optimised in such a way that performance has improved considerably. -Numerical stability is also better resulting in more models that can be solved. -The LUSOL bfp is now also the default. In the past, the etaPFI bfp package was the default, -but for larger models this leads faster to numerical instabilities and performance problems. - -Overall, the v5.5 code is faster and more robust than v5.1. -This robustness is for example proven by the fact that many more models can now be solved even without scaling. - -The API hasn't changed very much. -There are a couple of new routines and one routine has an extra argument. -Some constants got new values. - - * Fundamental internal change to the solver engine resulting in better performance and numerical stability. - Both the LP solver and the B&B solvers are enhanced. - * Optimised MILP branch truncation, with reduced cost fixing. - * LUSOL bfp is now the default. - * Presolve is improved in functionality and performance. - * Better handling of degeneracy, with more options. - * Store and read options from a file make it easier to set options. - * Partial pricing for the primal simplex now works. - * Full support for xli_ZIMPL v2.0.3. - * The objective function is no longer stored as part of the constraint matrix. - * Dual-long step code is in place, but not fully activated yet. - * General code cleanup. - * Added OBJSENSE and OBJNAME headers in the free MPS format (See MPS file format). - * The MathProg xli driver has now the ability to generate a model. - * New API routines - -Start by taking a look at 'Changes compared to version 4', 'Changes from version 5.1 to version 5.5' -and 'lp_solve usage' -This gives a good starting point. - - -BFP's ------ - -BFP stands for Basis Factorization Package, which is a unique lp_solve feature. Considerable -effort has been put in this new feature and we have big expectations for this. BFP is a generic -interface model and users can develop their own implementations based on the provided templates. -We are very interested in providing as many different BFPs as possible to the community. - -lp_solve 5.5 has the LUSOL BFP built in as default engine. In addition two other -BFPs are included for both Windows and Linux: bfp_etaPFI.dll, bfp_GLPK.dll for Windows and -libbfp_etaPFI.so, libbfp_GLPK.so for Linux. The bfp_etaPFI includes -advanced column ordering using the COLAMD library, as well as better pivot management for -stability. For complex models, however, the LU factorization approach is much better, and -lp_solve now includes LUSOL as one of the most stable engines available anywhere. LUSOL was -originally developed by Prof. Saunders at Stanford, and it has now been ported to C -and enhanced by Kjell. - -If you compile BFPs yourself, make sure that under Windows, you use __stdcall convention and -use 8 byte alignments. This is needed for the BFPs to work correctly with the general -distribution of lp_solve and also to make sharing BFPs as uncomplicated as possible. - -See the reference guide for more information. - - -XLI's ------ - -XLI stands for eXternal Language Interface, also a unique lp_solve feature. XLI's are stand-alone -libraries used as add-on to lp_solve to make it possible to read and write lp models in a format -not natively supported by lp_solve. Examples are CPLEX lp format, LINDO lp format, MathProg format, -XML format... - -See the reference guide for more information. - - -lpsolve API ------------ - -Don't forget that the API has changed compared to previous versions of lpsolve and that you just -can't use the version 5 lpsolve library with your version 4 or older code. That is also the -reason why the library is now called lpsolve55.dll/lpsolve55.a. lpsolve55.dll or lpsolve55.a are -only needed when you call the lpsolve library via the API interface from your program. -The lp_solve program is still a stand-alone executable program. - -There are examples interfaces for different language like C, VB, C#, VB.NET, Java, -Delphi, and there is now also even a COM object to access the lpsolve library. This means that -the door is wide-open for using lp_solve in many different situations. Thus everything that is -available in version 4 is now also available in version 5 and already much more! - -See the reference guide for more information. - - -Conversion between lp modeling formats --------------------------------------- - -Note that lp2mps and mps2lp don't exist anymore. However this functionality is now implemented -in lp_solve: - -lp2mps can be simulated as following: -lp_solve -parse_only -lp infile -wmps outfile - -mps2lp can be simulated as following: -lp_solve -parse_only -mps infile -wlp outfile - - -via the -rxli option, a model can be read via an XLI library and via the -wxli option, a model -can be written via an XLI library. - - -How to build the executables yourself. ---------------------------------------- - -At this time, there are no Makefiles yet. However for the time being, there are batch files/scripts -to build. For the Microsoft compiler under Windows, use cvc6.bat, for the gnu compiler under Windows, -use cgcc.bat and for Unix/Linux, use the ccc shell script (sh ccc). - -See the reference guide for more information. - - -IDE ---- - -Under Windows, there is now also a very user friendly lpsolve IDE. Check out LPSolveIDE - -See the reference guide for more information. - - -Documentation (reference guide) -------------------------------- - -See lp_solve55.chm for a Windows HTML help documentation file. -The html files are also in lp_solve_5.5_doc.tar.gz. Start with index.htm -Also see http://lpsolve.sourceforge.net/ for a on-line documentation - - -Change history: ---------------- - -17/05/05 version 5.5.0.0 -- Beta release of version 5.5 - -??/??/05 version 5.5.0.1 -- ? - -26/06/05 version 5.5.0.2 -- ? - -29/06/05 version 5.5.0.3 -- ? - -16/08/05 version 5.5.0.4 -- There are no API changes -- The LUSOL message routine could generate a crash under some cicumstances. Fixed -- A crash could occur when building the model in add_row_mode. Fixed. -- write_params didn't write the PRESOLVE and PRESOLVELOOPS correctly. Fixed. -- write_params didn't write constants with value 0. Fixed. -- The library did not compile under msdev 2002 (VC 7.0 _MSC_VER 1300). Fixed. -- There were some problems with printing long long variables which could generate a crash. Fixed. -- An overflow error could occur because memory was sometimes overwritten. Fixed. -- Presolve routines are revised. They are again improved and made faster. - Also some problems with it are fixed (possible crashes). -- Solver revised. Again made faster and more stable. -- get_row/get_column returned FALSE if the row/column is empty. Fixed. -- get_rowex/get_columnex now returns -1 if and error is detected. This instead of 0. - This to know the distinction between an empty row/column and an error. -- set_bounds had a possible problem when min and max are equal. Fixed. -- A crash/damage error could occur when rows/columns are added after a solve. Fixed. -- The my_chsign macro in lp_types.h gave warnings with some compilers. Fixed. -- The lp_solve program now returns 255 if an unexpected error occurs. Before this was 1 - But this interferes with the lpsolve library return codes. -- With the lp_solve program, debug and print modes were not written correctly in a - specified parameter file. Fixed. -- With the lp_solve program, presolveloops was not set correctly. Fixed. - -17/09/05 version 5.5.0.5 -- In some cases, SOS restrictions were not optimized till optimality. Fixed. -- Presolve sometimes generated 'Column -xxx out of range during presolve' with a possible crash. -- Presolve sometimes removed integer and/or semi-cont variables that should not be deleted. Fixed. -- B&B sometimes didn't find the most optimal solution. Fixed. -- Internal constant COMP_EQUAL renamed to COMP_PREFERNONE because this could interfere with a define - in the standard header files. -- The lp parser had problems with variables starting with INF and there is a + or - sign just before it. - Fixed. -- Added options -presolvem, -presolvefd, -presolvebnd, -presolved, -presolveslk -- Updated documentation. put_bb_branchfunc, put_bb_nodefunc, set_epslevel, dualize_lp, set_basisvar - -16/11/05 version 5.5.0.6 -- set_add_rowmode should not be called after a solve. There is now a test in this routine when this is - done and it then returns FALSE. -- When an empty string ("") as filename is provided to set_outputfile, then output is completely - ignored. -- In some cases, SOS models did not solve to their most optimal solution. -- There was as problem with get_sensitivity_objex. Calling it (sometimes only after multiple times) - resulted in protection errors/core dumps. -- When a model has no constraints, it did not solve most of the times. -- column_in_lp didn't work anymore. -- Large upper bounds could make the model unstable. A change was made to fix this. -- set_improve could let crash the model. -- lp_params.c used the non-ANSI function unlink(). Changed to ANSI function remove(). -- Presolve is again revised considerably. -- SOS handling is improved when there are a lot of SOS constraints. -- Limited constraint-to-SOS1 presolve to constraints with at least 4 variables. -- Limited bound tightening presolve loops. - -12/02/06 version 5.5.0.7 -- When SOS restrictions are added after a previous solve, a crash could occur. -- Optimized renaming a variable when the new name is equal to the old name. -- A possible crash could occur when presolve was enabled -- The constant ANTIDEGEN_DEFAULT is changed. ANTIDEGEN_INFEASIBLE is removed from it. - This constant should not be used unless you have some very tight and hard to solve - models where the optimal solution numerically straddles infeasibility. -- There was a possible problem with set_row(ex). It sometimes wrongfully changed the row. -- When integer variables were scaled, it could happen that because of rounding errors, - a loop was created. -- Sometimes integer models kept on looping in the B&B algorithm. -- A memory overrun could occur when an initial basis is set. This when variable names - are in Rnnn format and constraint names in Cnnn format. -- Some fixes are made in presolve. -- On 64-bit systems, compiler warnings were given and some code worked wrong resulting in - wrong results. -- lp_solve.c didn't compile with some compilers because if a very deep nested if statement. -- The distributed files now have the version number include in the filename. - For example lp_solve_5.5.0.7_exe.zip - This for a possible move to SourceForge in the (near?) future. -- When illegal bounds are specified in the MPS format (lower bound larger than upper bound) - then a warning was given but the illegal bound was just ignored and the model was solved. - This resulted in a solution that did not comply to the original model. Now the message is - seen as an error and solving is aborted. - - -06/09/06 version 5.5.0.8 -- When presolve is active and columns are removed and there are SOS constraints, then presolve - had an error which could result in hanging while solve or maybe wrong solutions. -- set_row(ex) set wrong values when used after a previous solve and scaling is active. -- disabled PRESOLVE_REDUCEMIP since it is very rare that this is effective, and also that it adds - code complications and delayed presolve effects that are not captured properly. -- made routine guess_basis available for all languages (now exported by the dll). - The routine is now also documented. -- some bug corrections in guess_basis. -- Corrected a problem with add_column(ex) when add_rowmode is enabled. -- write_lp now wraps long lines over multiple lines to make it more readable. -- A compilation warning/error sometimes occured on is_fixedvar in lp_lusol.c with some compilers. -- Added options -wxlisol and -wxlisolopt to lp_solve program to write a solution file for those - XLIs that support it. -- Updated CPLEX XLI to support constants in objective. -- Added documentation on infeasible models, guess_basis, DIMACS models, CPLEX format, Zimpl, GNU Mathprog. - Corrected/updated documentation on get_col_name, get_row_name, get_nameindex, write_xli, - External Language Interfaces. -- The mps reader was improved for the rarely cases where the same elements are provided multiple - times. These are now added. -- Revised the java unittest example because it gave some errors that it shouldn't give. - -07/10/06 version 5.5.0.9 -- set_row(ex) could sometimes set wrong values in the model. -- Sometimes models with semi-cont variables which are also integer and scaling is active, a solution - was returned that is not most optimal or it returns infeasible. -- write_mps didn't write semi-cont variables with an infinite upper bound. -- When presolve can solve the model on its own and objective direction is maximize then a wrong sign - was shown in the reported price on screen. -- write_lp writes constraint and bounds in the same way if a constraint is not named. If a constraint - only has one variable then it looks like a bound. This can give problems because when a constraint - is interpreted as bound and it is negative then the problem definition changes. - Therefore a constraint which is not named and having only one variable in it is getting a name to - make sure it is interpreted as a constraint. -- The lp_solve program didn't interprete the PRESOLVED solve return code very well. Fixed. -- bfp_GLPK and xli_MathProg are now compiled against GLPK 4.11 -- When an integer model is aborted before the most optimal solution is found (timeout or - break at first, ...) solve returned OPTIMAL (0) instead of SUBOPTIMAL (1). This is now corrected. - -14/01/07 version 5.5.0.10 -- If a model has integer variables, but the model is already integer optimal in the simplex fase, - then it was reported as suboptimal feasible. -- get_objective, get_variables, get_ptr_variables, get_constraints, get_ptr_constraints, get_primal_solution - reported 'not a valid basis' when presolve is active and the model is entirely solved by presolve. -- presolve on a model with SOS variables sometimes went wrong. -- presolve on a model with SOS variables where infeasibility is detected crashed. -- read_bas could fail when not all constraints had names or had names like default variable names. -- A crash could occur with set_row(ex) in rowmode. -- The lp format has been extended with a free section to define free variables. -- bfp_GLPK and xli_MathProg are now compiled against GLPK 4.13 -- fixed bug in the pseudocost logic that can blow up subsequent pseudocost values in that - branch and make them almost random. -- In some rare cases a memory overrun could occur when constraints are added after a previous solve. -- Made the copy_lp routine work. Note that information about the optimisation of the original model - is not copied (at this time). Only the model is. -- Fixed a bug in the hashing routines that had only influence in some rare cases in the - MATLAB, O-Matrix, Scilab, Octave, Python drivers. -- coldual sometimes worked on a uninitialised variable value with unpredictable results. - -27/12/07 version 5.5.0.11 -- Fixed a problem in presolve. Sometimes an array-index-out-of-bounds error occured. -- Added a makefile for Linux. -- When adding constraints, in some rare cases a memory overrun could occur resulting in a crash. -- add_constraintex with count=0 and row=colno=NULL gave a protection error. - several XLIs didn't work anymore because of this. -- set_constr_type sometimes set wrong signs for the coefficient matrix when add_rowmode is on. -- presolve did an unnecessary extra loop. Also a test is added to stop presolve when very few - changes are done. -- for very large models, a request of much more memory than is reasonable could occur. Fixed. -- Modified LINDO XLI to read keywords also not at column 1 and to accept an empty objective function. - Previously this wat not possible. -- In some rare cases, numbers very close to their integer values (for example 11276.999999999998) - were truncated to their ceiling value (for example 11276) instead of rounded - (for example 11277). -- Solved a problem with presolve with an all-int constraint. -- Solved a problem with presolve coldominate -- Added stronger checking of the MPS format. - Fields that must be blank are now also tested accordingly so that if data is there that it is - not ignored as before. -- FREE MPS format was not read ok if row/column name were all numbers - or a FR, MI, PL, BV bound was defined. Fixed. -- The lp-format now also supports a bin(ary) section to define binary variables. -- When an integer model is aborted before the most optimal solution is found - via break at first or break at value, solve returned OPTIMAL (0) instead of SUBOPTIMAL (1). - This is now corrected. Problem occured in version 5.5.0.10 -- Fixed a problem with del_constraint. Constraints names were not shifted and reported variable result was incorrect. -- read_XLI failed with MathProg if datamodel was provided with "" when there is no datamodel. - NULL was expected in the past. "" is now also accepted. -- Added an XLI to read Xpress lp files. -- Added routines MPS_writefileex, write_lpex. -- Added options -o0, -o1 to lp_solve command driven program to specify if objective is in basis or not. -- Added new information in the reference guide: - - Linear programming basics - - Presolve - - Xpress lp files - -04/01/08 version 5.5.0.11 -- There was an error in the lp-parser resulting is misreading the model in many cases. This was fixed in the existing release and an update of following files is posted: - lp_solve_5.5.0.11_exe.tar.gz - lp_solve_5.5.0.11_dev.tar.gz - lp_solve_5.5.0.11_source.tar.gz - lp_solve_5.5.0.11_exe.zip - lp_solve_5.5.0.11_dev.zip - -13/03/08 version 5.5.0.12 -- When NODE_RCOSTFIXING (bound tightening during B&B) (enabled by default) was enabled some models were not solved - to optimality. -- In following situation a (sub-optimal) solution was returned while no integer - solution isn't found yet at this time. - - The model contains integers - - A break at first is set - - A timeout is set - - The timeout occurs before a first integer solution is found - - When the timeout occurs, the simplex algorithm is in phase 2 and has a feasible (but non-integer) solution, but not optimal yet. -- If an integer solution is found with the same objective value as the relaxed solution then - B&B is stopped. However this resulted in a report of a SUBOPTIMAL solution while this is not true. - The found solution is then optimal. -- On 64-bit Linux systems crashes occured when information is printed via variable argument routines. -- There was a warning about redefinition of MAXINT32, MAXUINT32, MAXINT64, MAXUINT64 when compiled - with MS Visual C 2008. -- The reference guide now also contains the Java documentation. - -03/08/08 version 5.5.0.13 -- The code should now completely be reentrant and thread safe. - Especially the lp-parser, but also the solver code itself used static variables which made it - not reentrant. - The lp parser code has changed considerably for this. Also the lex and yacc code had to be made - reentrant. This is only tested with GNU flex and bison. Not with lex and yacc. - This should be no problem since the corresponding c files are included with the source so people - don't have to build these themselves. -- commonlib.c/h were also under BFP/BFP_LUSOL/LUSOL while a more advanced version is also - under the shared directory. The one from LUSOL is removed. -- If objective coefficients are all integer and the columns are integer or objective coefficients - are zero, then B&B can stop when the objective function equals ceil/floor(real objective) - which results in a faster solving process. This did not work anymore and is now fixed. -- The lpsolve IDE is now also on sourceforge (previously it was a link to another site). - Also the sources are included. - -02/02/09 version 5.5.0.14 -- The lp format now allows spaces between the row label and the colon. -- Added better error handling in the copy_lp API -- Timeout in a MIP model sometimes resulted in an endless loop -- Revised isnan call for gnu compilers -- Removed some extra static variables for better reentrance support. -- REPORT_Objective now reports large objective values (>1e-5) with better precision -- On Windows the binaries are again compiled with compiler optimization /O2 for better performance. -- Compiled and tested on Windows and Linux 64-bit. 64-bit binaries are now provided. - Changed project files and compile scripts for 64-bit compilation. - Binaries are now located in a subdirectory bin\platform -- Added PHP driver so that lpsolve can be called from PHP -- When an MPS file is created, upper bounds were written before lower bounds. - Now it is the other way around. First lower bounds and then upper bounds. -- The MPS reader could not handle negative lower bounds where the lower bound was specified after the upper bound. -- The MPS write now writes the implicit lower bound of zero if the variable has no lower bound and it is integer. - This to make sure that other solvers (like CPLEX) interprete the variable as integer and not binary. -- The LINDO XLI parser interpreted negative lower and upper bounds as positive -- Added the option -stat to the lp_solve driver program. It prints statistics of the model like number of - rows/columns used and smalled and largest value in objective, RHS and coefficient matrix. -- Added the option -plp to the lp_solve driver program. It prints the model via the print_lp API function. - -09/09/09 version 5.5.0.15 -- Improved guess_basis -- set_row(ex) corrections. Actually completely revised the routine. - Sometimes the matrix was corrupted after the call. -- When in set_add_row mode, all API calls can now be used. For example printing or writing the model, - retrieving data from the model and so on are now all possible in that mode. - Will help greatly when debugging your models. -- Sometimes an integer model is reported infeasible while it isn't because the integer solution - is much different from the relaxed (real) model. Then API call set_bb_depthlimit can be used - to increase the default B&B depth. However its new value was not used for the depth limit of an - individual variabled which could result in still infeasible results. -- modified the demo program not to use the str_ versions of API calls, because they are not - performant and not intended to be used in real programs. -- Added an lpsolve driver to Sysquake. -- Added an lpsolve driver to FreeMat. -- Added an lpsolve driver to Euler. -- Added an Access example. -- Added documentation to use lpsolve from Sage via the existing Python driver. -- Changed the second parameter to read_mps, read_MPS, read_freemps and read_freeMPS from verbose to options. - These routines now supports via this options parameter the native IBM interpretation of - integer variables and allows to negate the objective constant. - This is also supported by the lp_solve command line program via the -mps_ibm and -mps_negobjconst options. - This is also supported by the IDE in the options tab. -- Removed read_LPhandle and read_MPShandle from the lprec structure. - On their place there is now read_LP and read_MPS. - The handle functions are not usable anyway under windows as dll because the FILE structure - is searched in the dll space and therefore not working. - read_LP and read_MPS work on the file name and always work. - Since these handle routines were not documented anyway and the lp structure should not be - accessed by applications (except XLI and BFP drivers) this change should not give any (compatibility) problem at all. -- write_lp/write_LP write just a + or - if the factor to a variable is +1 or -1 - However when the number was not exactly one, for example 1.0000000000001, then the test - on equal to one gave false, but the writing was a 1 because only 12 significant digits are written. - The result was that the 1 was written in that case. Not an error, but when lp files are compared, - this is enoying. Now this is solved. When a +1 or -1 would be written then alyways only + and - - is written and +1 or -1 will never occur again. -- When a message function is set via put_msgfunc, - in message MSG_LPOPTIMAL it was not possible to get the solution. Now it is. - Also when messages MSG_MILPFEASIBLE, MSG_MILPBETTER, MSG_MILPEQUAL were catched the - returned solution was the one from the previous MIP solution. This is now also fixed. - With this fix, the IDE now also shows the relaxed solution (column LP Optimal). -- Extended the MATLAB, O-Matrix, Scilab, Octave, Python, PHP and of course the new - Sysquake, FreeMat and Euler drivers to support string constants. See the reference guide. -- Compiled and tested lpsolve on MAC OSX 10.4.8 (Intel). Compilation scripts for this platform - were added and revised and binaries are provided to download. -- Revised the c# and vb.net lpsolve code to pass arrays back and forth to the dll to make it also work on 64 bit systems. -- IDE enhancements: - - Can talk to the lpsolve55.dll large address aware version so access to 3.4G 32 bit memory. - - Can write and read lp_solve parameter files ( type .lpi) via Options, Solver params, so - it is easy to save a readable file of parameters for a particular problem class when it is found. - The file can then be collected by the -rpar exe param or the read_params() API. - - Added Options, Reset to solver defaults to go back to all lp_solve defaults. - - Added in Help a link to the online help files. - - Resized the help, main and statistics windows for Vista. - - Added the Xpress XLI to the setup. - - Use the latest dlls, especially lpsolve55.dll, in the setup. - - Added the two new MPS options for IBM integer variables and negate objective constant. - - Added the option to ignore integer restrictions. -17/09/09 version 5.5.0.15b -- objfrom values where not correct if scaling is active. - Routine to calculate these values and the objective from-till values is revised - and made a bit more performant. -- write_lp did not write all column data. For objective function and constraints. - -28/10/09 version 5.5.1.0 -- In some cases with a model with integer variables lp_solve did not find the most optimal solution. Fixed. -- The reported relative gap for the optimal solution was not always ok. -- Sensitivity on a zero upper bounded variable was not calculated. Now it is. - -08/03/10 version 5.5.1.2 -- When there are integer variables in the model then sensitivity is not correct. - -12/08/10 version 5.5.2.0 -- add_SOS did a wrong test when SOS type is bigger than 2 if this SOS is allowed in lp_solve. Fixed -- new and improved MIP_stepOF function to find integer solutions. -- fixes in set_row. -- replaces code - w *= 2*lp->is_lower[i] - 1; - by - w = my_chsign(!lp->is_lower[i], w); - because the old failed on AIX - Also changed the definition of my_chsign in lp_types.h in the same way. -- Added the option -BR in the lp_solve driver program to select branch rule NODE_PSEUDORATIOSELECT -- lpsolve.pas updated for Delphi 2009. -- Python driver updated to allow also to provide numpy arrays to the lpsolve driver. -- xli_MathProg is now compiled against glpk 4.44 meaning that it now supports the MathProg Tables feature - as described in http://www.cs.unb.ca/~bremner/docs/glpk/tables.pdf - csv, ODBC and MySQL are supported on all platforms. -- Added MSF support. MSF (Microsoft Solver Foundation) is a microsoft .NET library to access solvers via an OO way - MSF has a default lpsolve driver, but the one on the sourceforge site is enhanced in functionality and performance. - There is also documentation in the lpsolve reference guide. - -../../.. version 5.5.2.1 -- fixed a small error in new and improved MIP_stepOF function to find integer solutions. - -../../.. version 5.5.2.2 -- For integer models with semi-cont variables it happened sometimes that a message - "fillbranches_BB: Inconsistent equal-valued bounds for ..." occured and that the semi-cont condition - was not respected. -- New functions added: get_accuracy to get the numeric accuracy after solve. -- New functions added: set_break_numeric_accuracy, get_break_numeric_accuracy to let lp_solve return ACCURACYERROR - instead of FEASIBLE when numerical accuracy if worse then the provided values. - In the past, lp_solve only returned a non-optimal status in case of very severe numerical instability. - Now it will return already ACCURACYERROR when it finds a relative inaccuracy of 5e-7 -- When reading a model from the lp-format and important issues are detected such as already bounds on variables being overruled - later with for example a bin keyword, this is now reported in the default verbose level such that this is seen easier. - -08/05/16 version 5.5.2.3 -- For some models with integer variables, lp_solve did not find the most optimal solution. - -15/09/16 version 5.5.2.4 -- When using set_lowbo and set_upbo to set bounds on a variable and the new low/up bounds are very close to each other - but not equal then they are set equal for numerical stability. - -18/09/16 version 5.5.2.5 -- When all variables in the model are integer, but not all binary (in fact difference between upper and lower bound 1), - then it could happen that not the most optimal integer solution was found. -- Updated/added scaling options to the lp_solve command line program - -We are thrilled to hear from you and your experiences with this new version. The good and the bad. -Also we would be pleased to hear about your experiences with the different BFPs on your models. - -Please send reactions to: -Peter Notebaert: lpsolve@peno.be -Kjell Eikland: kjell.eikland@broadpark.no diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/LUSOL_LGPL.txt b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/LUSOL_LGPL.txt deleted file mode 100644 index 223ede7d..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/LUSOL_LGPL.txt +++ /dev/null @@ -1,504 +0,0 @@ - GNU LESSER GENERAL PUBLIC LICENSE - Version 2.1, February 1999 - - Copyright (C) 1991, 1999 Free Software Foundation, Inc. - 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - -[This is the first released version of the Lesser GPL. It also counts - as the successor of the GNU Library Public License, version 2, hence - the version number 2.1.] - - Preamble - - The licenses for most software are designed to take away your -freedom to share and change it. By contrast, the GNU General Public -Licenses are intended to guarantee your freedom to share and change -free software--to make sure the software is free for all its users. - - This license, the Lesser General Public License, applies to some -specially designated software packages--typically libraries--of the -Free Software Foundation and other authors who decide to use it. You -can use it too, but we suggest you first think carefully about whether -this license or the ordinary General Public License is the better -strategy to use in any particular case, based on the explanations below. - - When we speak of free software, we are referring to freedom of use, -not price. Our General Public Licenses are designed to make sure that -you have the freedom to distribute copies of free software (and charge -for this service if you wish); that you receive source code or can get -it if you want it; that you can change the software and use pieces of -it in new free programs; and that you are informed that you can do -these things. - - To protect your rights, we need to make restrictions that forbid -distributors to deny you these rights or to ask you to surrender these -rights. These restrictions translate to certain responsibilities for -you if you distribute copies of the library or if you modify it. - - For example, if you distribute copies of the library, whether gratis -or for a fee, you must give the recipients all the rights that we gave -you. You must make sure that they, too, receive or can get the source -code. If you link other code with the library, you must provide -complete object files to the recipients, so that they can relink them -with the library after making changes to the library and recompiling -it. And you must show them these terms so they know their rights. - - We protect your rights with a two-step method: (1) we copyright the -library, and (2) we offer you this license, which gives you legal -permission to copy, distribute and/or modify the library. - - To protect each distributor, we want to make it very clear that -there is no warranty for the free library. Also, if the library is -modified by someone else and passed on, the recipients should know -that what they have is not the original version, so that the original -author's reputation will not be affected by problems that might be -introduced by others. - - Finally, software patents pose a constant threat to the existence of -any free program. We wish to make sure that a company cannot -effectively restrict the users of a free program by obtaining a -restrictive license from a patent holder. Therefore, we insist that -any patent license obtained for a version of the library must be -consistent with the full freedom of use specified in this license. - - Most GNU software, including some libraries, is covered by the -ordinary GNU General Public License. This license, the GNU Lesser -General Public License, applies to certain designated libraries, and -is quite different from the ordinary General Public License. We use -this license for certain libraries in order to permit linking those -libraries into non-free programs. - - When a program is linked with a library, whether statically or using -a shared library, the combination of the two is legally speaking a -combined work, a derivative of the original library. The ordinary -General Public License therefore permits such linking only if the -entire combination fits its criteria of freedom. The Lesser General -Public License permits more lax criteria for linking other code with -the library. - - We call this license the "Lesser" General Public License because it -does Less to protect the user's freedom than the ordinary General -Public License. It also provides other free software developers Less -of an advantage over competing non-free programs. These disadvantages -are the reason we use the ordinary General Public License for many -libraries. However, the Lesser license provides advantages in certain -special circumstances. - - For example, on rare occasions, there may be a special need to -encourage the widest possible use of a certain library, so that it becomes -a de-facto standard. To achieve this, non-free programs must be -allowed to use the library. A more frequent case is that a free -library does the same job as widely used non-free libraries. In this -case, there is little to gain by limiting the free library to free -software only, so we use the Lesser General Public License. - - In other cases, permission to use a particular library in non-free -programs enables a greater number of people to use a large body of -free software. For example, permission to use the GNU C Library in -non-free programs enables many more people to use the whole GNU -operating system, as well as its variant, the GNU/Linux operating -system. - - Although the Lesser General Public License is Less protective of the -users' freedom, it does ensure that the user of a program that is -linked with the Library has the freedom and the wherewithal to run -that program using a modified version of the Library. - - The precise terms and conditions for copying, distribution and -modification follow. Pay close attention to the difference between a -"work based on the library" and a "work that uses the library". The -former contains code derived from the library, whereas the latter must -be combined with the library in order to run. - - GNU LESSER GENERAL PUBLIC LICENSE - TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION - - 0. This License Agreement applies to any software library or other -program which contains a notice placed by the copyright holder or -other authorized party saying it may be distributed under the terms of -this Lesser General Public License (also called "this License"). -Each licensee is addressed as "you". - - A "library" means a collection of software functions and/or data -prepared so as to be conveniently linked with application programs -(which use some of those functions and data) to form executables. - - The "Library", below, refers to any such software library or work -which has been distributed under these terms. A "work based on the -Library" means either the Library or any derivative work under -copyright law: that is to say, a work containing the Library or a -portion of it, either verbatim or with modifications and/or translated -straightforwardly into another language. (Hereinafter, translation is -included without limitation in the term "modification".) - - "Source code" for a work means the preferred form of the work for -making modifications to it. For a library, complete source code means -all the source code for all modules it contains, plus any associated -interface definition files, plus the scripts used to control compilation -and installation of the library. - - Activities other than copying, distribution and modification are not -covered by this License; they are outside its scope. The act of -running a program using the Library is not restricted, and output from -such a program is covered only if its contents constitute a work based -on the Library (independent of the use of the Library in a tool for -writing it). Whether that is true depends on what the Library does -and what the program that uses the Library does. - - 1. You may copy and distribute verbatim copies of the Library's -complete source code as you receive it, in any medium, provided that -you conspicuously and appropriately publish on each copy an -appropriate copyright notice and disclaimer of warranty; keep intact -all the notices that refer to this License and to the absence of any -warranty; and distribute a copy of this License along with the -Library. - - You may charge a fee for the physical act of transferring a copy, -and you may at your option offer warranty protection in exchange for a -fee. - - 2. You may modify your copy or copies of the Library or any portion -of it, thus forming a work based on the Library, and copy and -distribute such modifications or work under the terms of Section 1 -above, provided that you also meet all of these conditions: - - a) The modified work must itself be a software library. - - b) You must cause the files modified to carry prominent notices - stating that you changed the files and the date of any change. - - c) You must cause the whole of the work to be licensed at no - charge to all third parties under the terms of this License. - - d) If a facility in the modified Library refers to a function or a - table of data to be supplied by an application program that uses - the facility, other than as an argument passed when the facility - is invoked, then you must make a good faith effort to ensure that, - in the event an application does not supply such function or - table, the facility still operates, and performs whatever part of - its purpose remains meaningful. - - (For example, a function in a library to compute square roots has - a purpose that is entirely well-defined independent of the - application. Therefore, Subsection 2d requires that any - application-supplied function or table used by this function must - be optional: if the application does not supply it, the square - root function must still compute square roots.) - -These requirements apply to the modified work as a whole. If -identifiable sections of that work are not derived from the Library, -and can be reasonably considered independent and separate works in -themselves, then this License, and its terms, do not apply to those -sections when you distribute them as separate works. But when you -distribute the same sections as part of a whole which is a work based -on the Library, the distribution of the whole must be on the terms of -this License, whose permissions for other licensees extend to the -entire whole, and thus to each and every part regardless of who wrote -it. - -Thus, it is not the intent of this section to claim rights or contest -your rights to work written entirely by you; rather, the intent is to -exercise the right to control the distribution of derivative or -collective works based on the Library. - -In addition, mere aggregation of another work not based on the Library -with the Library (or with a work based on the Library) on a volume of -a storage or distribution medium does not bring the other work under -the scope of this License. - - 3. You may opt to apply the terms of the ordinary GNU General Public -License instead of this License to a given copy of the Library. To do -this, you must alter all the notices that refer to this License, so -that they refer to the ordinary GNU General Public License, version 2, -instead of to this License. (If a newer version than version 2 of the -ordinary GNU General Public License has appeared, then you can specify -that version instead if you wish.) Do not make any other change in -these notices. - - Once this change is made in a given copy, it is irreversible for -that copy, so the ordinary GNU General Public License applies to all -subsequent copies and derivative works made from that copy. - - This option is useful when you wish to copy part of the code of -the Library into a program that is not a library. - - 4. You may copy and distribute the Library (or a portion or -derivative of it, under Section 2) in object code or executable form -under the terms of Sections 1 and 2 above provided that you accompany -it with the complete corresponding machine-readable source code, which -must be distributed under the terms of Sections 1 and 2 above on a -medium customarily used for software interchange. - - If distribution of object code is made by offering access to copy -from a designated place, then offering equivalent access to copy the -source code from the same place satisfies the requirement to -distribute the source code, even though third parties are not -compelled to copy the source along with the object code. - - 5. A program that contains no derivative of any portion of the -Library, but is designed to work with the Library by being compiled or -linked with it, is called a "work that uses the Library". Such a -work, in isolation, is not a derivative work of the Library, and -therefore falls outside the scope of this License. - - However, linking a "work that uses the Library" with the Library -creates an executable that is a derivative of the Library (because it -contains portions of the Library), rather than a "work that uses the -library". The executable is therefore covered by this License. -Section 6 states terms for distribution of such executables. - - When a "work that uses the Library" uses material from a header file -that is part of the Library, the object code for the work may be a -derivative work of the Library even though the source code is not. -Whether this is true is especially significant if the work can be -linked without the Library, or if the work is itself a library. The -threshold for this to be true is not precisely defined by law. - - If such an object file uses only numerical parameters, data -structure layouts and accessors, and small macros and small inline -functions (ten lines or less in length), then the use of the object -file is unrestricted, regardless of whether it is legally a derivative -work. (Executables containing this object code plus portions of the -Library will still fall under Section 6.) - - Otherwise, if the work is a derivative of the Library, you may -distribute the object code for the work under the terms of Section 6. -Any executables containing that work also fall under Section 6, -whether or not they are linked directly with the Library itself. - - 6. As an exception to the Sections above, you may also combine or -link a "work that uses the Library" with the Library to produce a -work containing portions of the Library, and distribute that work -under terms of your choice, provided that the terms permit -modification of the work for the customer's own use and reverse -engineering for debugging such modifications. - - You must give prominent notice with each copy of the work that the -Library is used in it and that the Library and its use are covered by -this License. You must supply a copy of this License. If the work -during execution displays copyright notices, you must include the -copyright notice for the Library among them, as well as a reference -directing the user to the copy of this License. Also, you must do one -of these things: - - a) Accompany the work with the complete corresponding - machine-readable source code for the Library including whatever - changes were used in the work (which must be distributed under - Sections 1 and 2 above); and, if the work is an executable linked - with the Library, with the complete machine-readable "work that - uses the Library", as object code and/or source code, so that the - user can modify the Library and then relink to produce a modified - executable containing the modified Library. (It is understood - that the user who changes the contents of definitions files in the - Library will not necessarily be able to recompile the application - to use the modified definitions.) - - b) Use a suitable shared library mechanism for linking with the - Library. A suitable mechanism is one that (1) uses at run time a - copy of the library already present on the user's computer system, - rather than copying library functions into the executable, and (2) - will operate properly with a modified version of the library, if - the user installs one, as long as the modified version is - interface-compatible with the version that the work was made with. - - c) Accompany the work with a written offer, valid for at - least three years, to give the same user the materials - specified in Subsection 6a, above, for a charge no more - than the cost of performing this distribution. - - d) If distribution of the work is made by offering access to copy - from a designated place, offer equivalent access to copy the above - specified materials from the same place. - - e) Verify that the user has already received a copy of these - materials or that you have already sent this user a copy. - - For an executable, the required form of the "work that uses the -Library" must include any data and utility programs needed for -reproducing the executable from it. However, as a special exception, -the materials to be distributed need not include anything that is -normally distributed (in either source or binary form) with the major -components (compiler, kernel, and so on) of the operating system on -which the executable runs, unless that component itself accompanies -the executable. - - It may happen that this requirement contradicts the license -restrictions of other proprietary libraries that do not normally -accompany the operating system. Such a contradiction means you cannot -use both them and the Library together in an executable that you -distribute. - - 7. You may place library facilities that are a work based on the -Library side-by-side in a single library together with other library -facilities not covered by this License, and distribute such a combined -library, provided that the separate distribution of the work based on -the Library and of the other library facilities is otherwise -permitted, and provided that you do these two things: - - a) Accompany the combined library with a copy of the same work - based on the Library, uncombined with any other library - facilities. This must be distributed under the terms of the - Sections above. - - b) Give prominent notice with the combined library of the fact - that part of it is a work based on the Library, and explaining - where to find the accompanying uncombined form of the same work. - - 8. You may not copy, modify, sublicense, link with, or distribute -the Library except as expressly provided under this License. Any -attempt otherwise to copy, modify, sublicense, link with, or -distribute the Library is void, and will automatically terminate your -rights under this License. However, parties who have received copies, -or rights, from you under this License will not have their licenses -terminated so long as such parties remain in full compliance. - - 9. You are not required to accept this License, since you have not -signed it. However, nothing else grants you permission to modify or -distribute the Library or its derivative works. These actions are -prohibited by law if you do not accept this License. Therefore, by -modifying or distributing the Library (or any work based on the -Library), you indicate your acceptance of this License to do so, and -all its terms and conditions for copying, distributing or modifying -the Library or works based on it. - - 10. Each time you redistribute the Library (or any work based on the -Library), the recipient automatically receives a license from the -original licensor to copy, distribute, link with or modify the Library -subject to these terms and conditions. You may not impose any further -restrictions on the recipients' exercise of the rights granted herein. -You are not responsible for enforcing compliance by third parties with -this License. - - 11. If, as a consequence of a court judgment or allegation of patent -infringement or for any other reason (not limited to patent issues), -conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot -distribute so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you -may not distribute the Library at all. For example, if a patent -license would not permit royalty-free redistribution of the Library by -all those who receive copies directly or indirectly through you, then -the only way you could satisfy both it and this License would be to -refrain entirely from distribution of the Library. - -If any portion of this section is held invalid or unenforceable under any -particular circumstance, the balance of the section is intended to apply, -and the section as a whole is intended to apply in other circumstances. - -It is not the purpose of this section to induce you to infringe any -patents or other property right claims or to contest validity of any -such claims; this section has the sole purpose of protecting the -integrity of the free software distribution system which is -implemented by public license practices. Many people have made -generous contributions to the wide range of software distributed -through that system in reliance on consistent application of that -system; it is up to the author/donor to decide if he or she is willing -to distribute software through any other system and a licensee cannot -impose that choice. - -This section is intended to make thoroughly clear what is believed to -be a consequence of the rest of this License. - - 12. If the distribution and/or use of the Library is restricted in -certain countries either by patents or by copyrighted interfaces, the -original copyright holder who places the Library under this License may add -an explicit geographical distribution limitation excluding those countries, -so that distribution is permitted only in or among countries not thus -excluded. In such case, this License incorporates the limitation as if -written in the body of this License. - - 13. The Free Software Foundation may publish revised and/or new -versions of the Lesser General Public License from time to time. -Such new versions will be similar in spirit to the present version, -but may differ in detail to address new problems or concerns. - -Each version is given a distinguishing version number. If the Library -specifies a version number of this License which applies to it and -"any later version", you have the option of following the terms and -conditions either of that version or of any later version published by -the Free Software Foundation. If the Library does not specify a -license version number, you may choose any version ever published by -the Free Software Foundation. - - 14. If you wish to incorporate parts of the Library into other free -programs whose distribution conditions are incompatible with these, -write to the author to ask for permission. For software which is -copyrighted by the Free Software Foundation, write to the Free -Software Foundation; we sometimes make exceptions for this. Our -decision will be guided by the two goals of preserving the free status -of all derivatives of our free software and of promoting the sharing -and reuse of software generally. - - NO WARRANTY - - 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO -WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. -EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR -OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY -KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE -IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE -LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME -THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN -WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY -AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU -FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR -CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE -LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING -RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A -FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF -SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH -DAMAGES. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Libraries - - If you develop a new library, and you want it to be of the greatest -possible use to the public, we recommend making it free software that -everyone can redistribute and change. You can do so by permitting -redistribution under these terms (or, alternatively, under the terms of the -ordinary General Public License). - - To apply these terms, attach the following notices to the library. It is -safest to attach them to the start of each source file to most effectively -convey the exclusion of warranty; and each file should have at least the -"copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This library is free software; you can redistribute it and/or - modify it under the terms of the GNU Lesser General Public - License as published by the Free Software Foundation; either - version 2 of the License, or (at your option) any later version. - - This library is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - Lesser General Public License for more details. - - You should have received a copy of the GNU Lesser General Public - License along with this library; if not, write to the Free Software - Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - -Also add information on how to contact you by electronic and paper mail. - -You should also get your employer (if you work as a programmer) or your -school, if any, to sign a "copyright disclaimer" for the library, if -necessary. Here is a sample; alter the names: - - Yoyodyne, Inc., hereby disclaims all copyright interest in the - library `Frob' (a library for tweaking knobs) written by James Random Hacker. - - , 1 April 1990 - Ty Coon, President of Vice - -That's all there is to it! - - diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.c deleted file mode 100644 index b7ce41da..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.c +++ /dev/null @@ -1,808 +0,0 @@ - -/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - LUSOL routines from the Stanford Optimization Laboratory - The parts included are: - lusol1 Factor a given matrix A from scratch (lu1fac). - lusol2 Heap-management routines for lu1fac. - lusol6 Solve with the current LU factors. - lusol7 Utilities for all update routines. - lusol8 Replace a column (Bartels-Golub update). - ------------------------------------------------------------------ - 26 Apr 2002: TCP implemented using heap data structure. - 01 May 2002: lu1DCP implemented. - 07 May 2002: lu1mxc must put 0.0 at top of empty columns. - 09 May 2002: lu1mCP implements Markowitz with cols searched - in heap order. - Often faster (searching 20 or 40 cols) but more dense. - 11 Jun 2002: TRP implemented. - lu1mRP implements Markowitz with Threshold Rook - Pivoting. - lu1mxc maintains max col elements (was lu1max.) - lu1mxr maintains max row elements. - 12 Jun 2002: lu1mCP seems too slow on big problems (e.g. memplus). - Disabled it for the moment. (Use lu1mar + TCP.) - 14 Dec 2002: TSP implemented. - lu1mSP implements Markowitz with TSP. - 07 Mar 2003: character*1, character*2 changed to f90 form. - Comments changed from * in column to ! in column 1. - Comments kept within column 72 to avoid compiler - warning. - 06 Mar 2004: Translation to C by Kjell Eikland with the addition - of data wrappers, parametric constants, various - helper routines, and dynamic memory reallocation. - 26 May 2004: Added LUSOL_IP_UPDATELIMIT parameter and provided - for dynamic memory expansion based on possible - forward requirements. - 08 Jul 2004: Revised logic in lu6chk based on new code from - Michael Saunders. - 01 Dec 2005: Add support for CMEX interface (disable by undef MATLAB) - Also include various bug fixes (disable by undef YZHANG) - Yin Zhang - 01 Jan 2006: Added storage of singular indeces, not only the last. - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */ - -#include -#include -#include -#include -#include -#include -#include "lusol.h" -#include "myblas.h" -#ifdef MATLAB - #include "mex.h" -#endif - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* LUSOL Object creation and destruction */ - -void *clean_realloc(void *oldptr, int width, int newsize, int oldsize) -{ - newsize *= width; - oldsize *= width; - oldptr = LUSOL_REALLOC(oldptr, newsize); - if(newsize > oldsize) -/* MEMCLEAR(oldptr+oldsize, newsize-oldsize); */ - memset((char *)oldptr+oldsize, '\0', newsize-oldsize); - return(oldptr); -} - -MYBOOL LUSOL_realloc_a(LUSOLrec *LUSOL, int newsize) -{ - int oldsize; - - if(newsize < 0) - newsize = LUSOL->lena + MAX(abs(newsize), LUSOL_MINDELTA_a); - - oldsize = LUSOL->lena; - LUSOL->lena = newsize; - if(newsize > 0) - newsize++; - if(oldsize > 0) - oldsize++; - - LUSOL->a = (REAL *) clean_realloc(LUSOL->a, sizeof(*(LUSOL->a)), - newsize, oldsize); - LUSOL->indc = (int *) clean_realloc(LUSOL->indc, sizeof(*(LUSOL->indc)), - newsize, oldsize); - LUSOL->indr = (int *) clean_realloc(LUSOL->indr, sizeof(*(LUSOL->indr)), - newsize, oldsize); - if((newsize == 0) || - ((LUSOL->a != NULL) && (LUSOL->indc != NULL) && (LUSOL->indr != NULL))) - return( TRUE ); - else - return( FALSE ); -} - -MYBOOL LUSOL_expand_a(LUSOLrec *LUSOL, int *delta_lena, int *right_shift) -{ -#ifdef StaticMemAlloc - return( FALSE ); -#else - int LENA, NFREE, LFREE; - - /* Add expansion factor to avoid having to resize too often/too much; - (exponential formula suggested by Michael A. Saunders) */ - LENA = LUSOL->lena; - *delta_lena = DELTA_SIZE(*delta_lena, LENA); - - /* Expand it! */ - if((*delta_lena <= 0) || !LUSOL_realloc_a(LUSOL, LENA+(*delta_lena))) - return( FALSE ); - - /* Make sure we return the actual memory increase of a */ - *delta_lena = LUSOL->lena-LENA; - - /* Shift the used memory area to the right */ - LFREE = *right_shift; - NFREE = LFREE+*delta_lena; - LENA -= LFREE-1; - MEMMOVE(LUSOL->a+NFREE, LUSOL->a+LFREE, LENA); - MEMMOVE(LUSOL->indr+NFREE, LUSOL->indr+LFREE, LENA); - MEMMOVE(LUSOL->indc+NFREE, LUSOL->indc+LFREE, LENA); - - /* Also return the new starting position for the used memory area of a */ - *right_shift = NFREE; - - LUSOL->expanded_a++; - return( TRUE ); -#endif -} - -MYBOOL LUSOL_realloc_r(LUSOLrec *LUSOL, int newsize) -{ - int oldsize; - - if(newsize < 0) - newsize = LUSOL->maxm + MAX(abs(newsize), LUSOL_MINDELTA_rc); - - oldsize = LUSOL->maxm; - LUSOL->maxm = newsize; - if(newsize > 0) - newsize++; - if(oldsize > 0) - oldsize++; - - LUSOL->lenr = (int *) clean_realloc(LUSOL->lenr, sizeof(*(LUSOL->lenr)), - newsize, oldsize); - LUSOL->ip = (int *) clean_realloc(LUSOL->ip, sizeof(*(LUSOL->ip)), - newsize, oldsize); - LUSOL->iqloc = (int *) clean_realloc(LUSOL->iqloc, sizeof(*(LUSOL->iqloc)), - newsize, oldsize); - LUSOL->ipinv = (int *) clean_realloc(LUSOL->ipinv, sizeof(*(LUSOL->ipinv)), - newsize, oldsize); - LUSOL->locr = (int *) clean_realloc(LUSOL->locr, sizeof(*(LUSOL->locr)), - newsize, oldsize); - - if((newsize == 0) || - ((LUSOL->lenr != NULL) && - (LUSOL->ip != NULL) && (LUSOL->iqloc != NULL) && - (LUSOL->ipinv != NULL) && (LUSOL->locr != NULL))) { - -#ifndef ClassicHamaxR -#ifdef AlwaysSeparateHamaxR - if(LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TRP) -#endif - { - LUSOL->amaxr = (REAL *) clean_realloc(LUSOL->amaxr, sizeof(*(LUSOL->amaxr)), - newsize, oldsize); - if((newsize > 0) && (LUSOL->amaxr == NULL)) - return( FALSE ); - } -#endif - return( TRUE ); - } - else - return( FALSE ); -} - -MYBOOL LUSOL_realloc_c(LUSOLrec *LUSOL, int newsize) -{ - int oldsize; - - if(newsize < 0) - newsize = LUSOL->maxn + MAX(abs(newsize), LUSOL_MINDELTA_rc); - - oldsize = LUSOL->maxn; - LUSOL->maxn = newsize; - if(newsize > 0) - newsize++; - if(oldsize > 0) - oldsize++; - - LUSOL->lenc = (int *) clean_realloc(LUSOL->lenc, sizeof(*(LUSOL->lenc)), - newsize, oldsize); - LUSOL->iq = (int *) clean_realloc(LUSOL->iq, sizeof(*(LUSOL->iq)), - newsize, oldsize); - LUSOL->iploc = (int *) clean_realloc(LUSOL->iploc, sizeof(*(LUSOL->iploc)), - newsize, oldsize); - LUSOL->iqinv = (int *) clean_realloc(LUSOL->iqinv, sizeof(*(LUSOL->iqinv)), - newsize, oldsize); - LUSOL->locc = (int *) clean_realloc(LUSOL->locc, sizeof(*(LUSOL->locc)), - newsize, oldsize); - LUSOL->w = (REAL *) clean_realloc(LUSOL->w, sizeof(*(LUSOL->w)), - newsize, oldsize); -#ifdef LUSOLSafeFastUpdate - LUSOL->vLU6L = (REAL *) clean_realloc(LUSOL->vLU6L, sizeof(*(LUSOL->vLU6L)), - newsize, oldsize); -#else - LUSOL->vLU6L = LUSOL->w; -#endif - - if((newsize == 0) || - ((LUSOL->w != NULL) && (LUSOL->lenc != NULL) && - (LUSOL->iq != NULL) && (LUSOL->iploc != NULL) && - (LUSOL->iqinv != NULL) && (LUSOL->locc != NULL))) { - -#ifndef ClassicHamaxR - if(LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TCP) { - LUSOL->Ha = (REAL *) clean_realloc(LUSOL->Ha, sizeof(*(LUSOL->Ha)), - newsize, oldsize); - LUSOL->Hj = (int *) clean_realloc(LUSOL->Hj, sizeof(*(LUSOL->Hj)), - newsize, oldsize); - LUSOL->Hk = (int *) clean_realloc(LUSOL->Hk, sizeof(*(LUSOL->Hk)), - newsize, oldsize); - if((newsize > 0) && - ((LUSOL->Ha == NULL) || (LUSOL->Hj == NULL) || (LUSOL->Hk == NULL))) - return( FALSE ); - } -#endif -#ifndef ClassicdiagU - if(LUSOL->luparm[LUSOL_IP_KEEPLU] == FALSE) { - LUSOL->diagU = (REAL *) clean_realloc(LUSOL->diagU, sizeof(*(LUSOL->diagU)), - newsize, oldsize); - if((newsize > 0) && (LUSOL->diagU == NULL)) - return( FALSE ); - } -#endif - - return( TRUE ); - } - else - return( FALSE ); -} - -LUSOLrec *LUSOL_create(FILE *outstream, int msgfil, int pivotmodel, int updatelimit) -{ - LUSOLrec *newLU; - - newLU = (LUSOLrec *) LUSOL_CALLOC(1, sizeof(*newLU)); - if(newLU == NULL) - return( newLU ); - - newLU->luparm[LUSOL_IP_SCALAR_NZA] = LUSOL_MULT_nz_a; - newLU->outstream = outstream; - newLU->luparm[LUSOL_IP_PRINTUNIT] = msgfil; - newLU->luparm[LUSOL_IP_PRINTLEVEL] = LUSOL_MSG_SINGULARITY; - - LUSOL_setpivotmodel(newLU, pivotmodel, LUSOL_PIVTOL_DEFAULT); - - newLU->parmlu[LUSOL_RP_GAMMA] = LUSOL_DEFAULT_GAMMA; - - newLU->parmlu[LUSOL_RP_ZEROTOLERANCE] = 3.0e-13; - - newLU->parmlu[LUSOL_RP_SMALLDIAG_U] = /*3.7e-11;*/ - newLU->parmlu[LUSOL_RP_EPSDIAG_U] = 3.7e-11; - - newLU->parmlu[LUSOL_RP_COMPSPACE_U] = 3.0e+0; - - newLU->luparm[LUSOL_IP_MARKOWITZ_MAXCOL] = 5; - newLU->parmlu[LUSOL_RP_MARKOWITZ_CONLY] = 0.3e+0; - newLU->parmlu[LUSOL_RP_MARKOWITZ_DENSE] = 0.5e+0; - - newLU->parmlu[LUSOL_RP_SMARTRATIO] = LUSOL_DEFAULT_SMARTRATIO; -#ifdef ForceRowBasedL0 - newLU->luparm[LUSOL_IP_ACCELERATION] = LUSOL_BASEORDER; -#endif - newLU->luparm[LUSOL_IP_KEEPLU] = TRUE; - newLU->luparm[LUSOL_IP_UPDATELIMIT] = updatelimit; - - init_BLAS(); - - return( newLU ); -} - -MYBOOL LUSOL_sizeto(LUSOLrec *LUSOL, int init_r, int init_c, int init_a) -{ - if(init_c == 0) - LUSOL_FREE(LUSOL->isingular); - if(LUSOL_realloc_a(LUSOL, init_a) && - LUSOL_realloc_r(LUSOL, init_r) && - LUSOL_realloc_c(LUSOL, init_c)) - return( TRUE ); - else - return( FALSE ); -} - -char *LUSOL_pivotLabel(LUSOLrec *LUSOL) -{ - static /*const*/ char *pivotText[LUSOL_PIVMOD_MAX+1] = - {"TPP", "TRP", "TCP", "TSP"}; - return(pivotText[LUSOL->luparm[LUSOL_IP_PIVOTTYPE]]); -} - -void LUSOL_setpivotmodel(LUSOLrec *LUSOL, int pivotmodel, int initlevel) -{ - REAL newFM, newUM; - - /* Set pivotmodel if specified */ - if(pivotmodel > LUSOL_PIVMOD_NOCHANGE) { - if((pivotmodel <= LUSOL_PIVMOD_DEFAULT) || (pivotmodel > LUSOL_PIVMOD_MAX)) - pivotmodel = LUSOL_PIVMOD_TPP; - LUSOL->luparm[LUSOL_IP_PIVOTTYPE] = pivotmodel; - } - - /* Check if we need bother about changing tolerances */ - if((initlevel <= LUSOL_PIVTOL_NOCHANGE) || (initlevel > LUSOL_PIVTOL_MAX)) - return; - - /* Set default pivot tolerances - (note that UPDATEMAX should always be <= FACTORMAX) */ - if(initlevel == LUSOL_PIVTOL_BAGGY) { /* Extra-loose pivot thresholds */ - newFM = 500.0; - newUM = newFM / 20; - } - else if(initlevel == LUSOL_PIVTOL_LOOSE) { /* Moderately tight pivot tolerances */ - newFM = 100.0; - newUM = newFM / 10; - } - else if(initlevel == LUSOL_PIVTOL_NORMAL) { /* Standard pivot tolerances */ - newFM = 28.0; - newUM = newFM / 4; - } - else if(initlevel == LUSOL_PIVTOL_SLIM) { /* Better accuracy pivot tolerances */ - newFM = 10.0; - newUM = newFM / 2; - } - else if(initlevel == LUSOL_PIVTOL_TIGHT) { /* Enhanced accuracy pivot tolerances */ - newFM = 5.0; - newUM = newFM / 2; - } - else if(initlevel == LUSOL_PIVTOL_SUPER) { /* Very tight pivot tolerances for extra accuracy */ - newFM = 2.5; - newUM = 1.99; - } - else { /* Extremely tight pivot tolerances for extra accuracy */ - newFM = 1.99; - newUM = newFM / 1.49; - } - - /* Set the tolerances */ - LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij] = newFM; - LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij] = newUM; -} - -MYBOOL LUSOL_tightenpivot(LUSOLrec *LUSOL) -{ -#if 0 - REAL newvalue; -#endif - - /* Give up tightening if we are already less than limit and we cannot change strategy */ - if(MIN(LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij], - LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij]) < 1.1) { - if(LUSOL->luparm[LUSOL_IP_PIVOTTYPE] >= LUSOL_PIVMOD_TRP) - return( FALSE ); - LUSOL_setpivotmodel(LUSOL, LUSOL->luparm[LUSOL_IP_PIVOTTYPE]+1, LUSOL_PIVTOL_DEFAULT+1); - return( 2 ); - } - - /* Otherwise tighten according to defined schedule */ -#if 0 /* This is Michael Saunder's proposed tightening procedure */ - newvalue = sqrt(LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij]); - LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij] = newvalue; - SETMIN(LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij], newvalue); -#elif 0 /* This is Kjell Eikland's schedule #1 */ - newvalue = sqrt(LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij]); - LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij] = newvalue; - LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij] = 1.0 + (newvalue - 1.0) / 2; -#else /* This was Kjell Eikland's schedule #2 */ - LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij] = 1.0 + LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij]/3.0; - LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij] = 1.0 + LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij]/3.0; -#endif - return( TRUE ); -} - -MYBOOL LUSOL_addSingularity(LUSOLrec *LUSOL, int singcol, int *inform) -{ - int NSING = LUSOL->luparm[LUSOL_IP_SINGULARITIES], - ASING = LUSOL->luparm[LUSOL_IP_SINGULARLISTSIZE]; - - /* Check if we need to allocated list memory to store multiple singularities */ - if((NSING > 0) && (NSING >= ASING)) { - - /* Increase list in "reasonable" steps */ - ASING += (int) (10.0 * (log10((REAL) LUSOL->m)+1.0)); - LUSOL->isingular = (int *) LUSOL_REALLOC(LUSOL->isingular, sizeof(*LUSOL->isingular)*(ASING+1)); - if(LUSOL->isingular == NULL) { - LUSOL->luparm[LUSOL_IP_SINGULARLISTSIZE] = 0; - *inform = LUSOL_INFORM_NOMEMLEFT; - return( FALSE ); - } - LUSOL->luparm[LUSOL_IP_SINGULARLISTSIZE] = ASING; - - /* Transfer the first singularity if the list was just created */ - if(NSING == 1) - LUSOL->isingular[NSING] = LUSOL->luparm[LUSOL_IP_SINGULARINDEX]; - } - - /* Update singularity count and store its index */ - NSING++; - if(NSING > 1) { - LUSOL->isingular[0] = NSING; - LUSOL->isingular[NSING] = singcol; - } - LUSOL->luparm[LUSOL_IP_SINGULARITIES] = NSING; - - /* Mimic old logic by keeping the last singularity stored */ - LUSOL->luparm[LUSOL_IP_SINGULARINDEX] = singcol; - - return( TRUE ); -} - -int LUSOL_getSingularity(LUSOLrec *LUSOL, int singitem) -{ - if((singitem > LUSOL->luparm[LUSOL_IP_SINGULARITIES]) || (singitem < 0)) - singitem = -1; - else if(singitem == 0) - singitem = LUSOL->luparm[LUSOL_IP_SINGULARITIES]; - else if(singitem > 1) - singitem = LUSOL->isingular[singitem]; - else - singitem = LUSOL->luparm[LUSOL_IP_SINGULARINDEX]; - return( singitem ); -} - -int LUSOL_findSingularityPosition(LUSOLrec *LUSOL, int singcol) -/* The purpose of this routine is to find the slack row/column in - user-index that was singular in the last unsuccessful column - update; zero is returned if the search was unsuccessful. - By adding a slack at this position this particular singularity - should disappear. - (Source: Michael A. Saunders; private communication to KE) */ -{ -#if 0 /* Michael Saunders version */ - int j; - for(j = LUSOL->m; j > 0; j--) - if(LUSOL->iq[j] == singcol) - break; - singcol = j; -#else /* Kjell Eikland version (note that iqinv could be invalid in early versions of LUSOL) */ - singcol = LUSOL->iqinv[singcol]; -#endif - return( LUSOL->ip[singcol] ); -} - -char *LUSOL_informstr(LUSOLrec *LUSOL, int inform) -{ - static char *informText[LUSOL_INFORM_MAX-LUSOL_INFORM_MIN+1] = - {"LUSOL_RANKLOSS: Lost rank", - "LUSOL_LUSUCCESS: Success", - "LUSOL_LUSINGULAR: Singular A", - "LUSOL_LUUNSTABLE: Unstable factorization", - "LUSOL_ADIMERR: Row or column count exceeded", - "LUSOL_ADUPLICATE: Duplicate A matrix entry found", - "(Undefined message)", - "(Undefined message)", - "LUSOL_ANEEDMEM: Insufficient memory for factorization", - "LUSOL_FATALERR: Fatal internal error", - "LUSOL_NOPIVOT: Found no suitable pivot", - "LUSOL_NOMEMLEFT: Could not obtain more memory"}; - if((inform < LUSOL_INFORM_MIN) || (inform > LUSOL_INFORM_MAX)) - inform = LUSOL->luparm[LUSOL_IP_INFORM]; - return(informText[inform-LUSOL_INFORM_MIN]); -} - -void LUSOL_clear(LUSOLrec *LUSOL, MYBOOL nzonly) -{ - int len; - - LUSOL->nelem = 0; - if(!nzonly) { - - /* lena arrays */ - len = LUSOL->lena + LUSOL_ARRAYOFFSET; - MEMCLEAR(LUSOL->a, len); - MEMCLEAR(LUSOL->indc, len); - MEMCLEAR(LUSOL->indr, len); - - /* maxm arrays */ - len = LUSOL->maxm + LUSOL_ARRAYOFFSET; - MEMCLEAR(LUSOL->lenr, len); - MEMCLEAR(LUSOL->ip, len); - MEMCLEAR(LUSOL->iqloc, len); - MEMCLEAR(LUSOL->ipinv, len); - MEMCLEAR(LUSOL->locr, len); - -#ifndef ClassicHamaxR - if((LUSOL->amaxr != NULL) -#ifdef AlwaysSeparateHamaxR - && (LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TRP) -#endif - ) - MEMCLEAR(LUSOL->amaxr, len); -#endif - - /* maxn arrays */ - len = LUSOL->maxn + LUSOL_ARRAYOFFSET; - MEMCLEAR(LUSOL->lenc, len); - MEMCLEAR(LUSOL->iq, len); - MEMCLEAR(LUSOL->iploc, len); - MEMCLEAR(LUSOL->iqinv, len); - MEMCLEAR(LUSOL->locc, len); - MEMCLEAR(LUSOL->w, len); - - if(LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TCP) { - MEMCLEAR(LUSOL->Ha, len); - MEMCLEAR(LUSOL->Hj, len); - MEMCLEAR(LUSOL->Hk, len); - } -#ifndef ClassicdiagU - if(LUSOL->luparm[LUSOL_IP_KEEPLU] == FALSE) { - MEMCLEAR(LUSOL->diagU, len); - } -#endif - - } -} - -MYBOOL LUSOL_assign(LUSOLrec *LUSOL, int iA[], int jA[], REAL Aij[], int nzcount, MYBOOL istriplet) -{ - int k, m, n, ij, kol; - - /* Adjust the size of the a structure */ - if(nzcount > (LUSOL->lena/LUSOL->luparm[LUSOL_IP_SCALAR_NZA]) && - !LUSOL_realloc_a(LUSOL, nzcount*LUSOL->luparm[LUSOL_IP_SCALAR_NZA])) - return( FALSE ); - - m = 0; - n = 0; - kol = 1; - for(k = 1; k <= nzcount; k++) { - /* First the row indicator */ - ij = iA[k]; - if(ij > m) { - m = ij; - if(m > LUSOL->maxm && - !LUSOL_realloc_r(LUSOL, -(m / LUSOL_MINDELTA_FACTOR + 1))) - return( FALSE ); - } - LUSOL->indc[k] = ij; - - /* Then the column indicator; - Handle both triplet and column count formats */ - if(istriplet) - ij = jA[k]; - else { - if(k >= jA[kol]) - kol++; - ij = kol; - } - if(ij > n) { - n = ij; - if(n > LUSOL->maxn && - !LUSOL_realloc_c(LUSOL, -(n / LUSOL_MINDELTA_FACTOR + 1))) - return( FALSE ); - } - LUSOL->indr[k] = ij; - - /* Lastly the matrix value itself */ - LUSOL->a[k] = Aij[k]; - } - LUSOL->m = m; - LUSOL->n = n; - LUSOL->nelem = nzcount; - return( TRUE ); -} - -int LUSOL_loadColumn(LUSOLrec *LUSOL, int iA[], int jA, REAL Aij[], int nzcount, int offset1) -{ - int i, ii, nz, k; - - nz = LUSOL->nelem; - i = nz + nzcount; - if(i > (LUSOL->lena/LUSOL->luparm[LUSOL_IP_SCALAR_NZA]) && - !LUSOL_realloc_a(LUSOL, i*LUSOL->luparm[LUSOL_IP_SCALAR_NZA])) - return( -1 ); - - k = 0; - for(ii = 1; ii <= nzcount; ii++) { - i = ii + offset1; - if(Aij[i] == 0) - continue; - if(iA[i] <= 0 || iA[i] > LUSOL->m || - jA <= 0 || jA > LUSOL->n) { - LUSOL_report(LUSOL, 0, "Variable index outside of set bounds (r:%d/%d, c:%d/%d)\n", - iA[i], LUSOL->m, jA, LUSOL->n); - continue; - } - k++; - nz++; - LUSOL->a[nz] = Aij[i]; - LUSOL->indc[nz] = iA[i]; - LUSOL->indr[nz] = jA; - } - LUSOL->nelem = nz; - return( k ); -} - -void LUSOL_free(LUSOLrec *LUSOL) -{ - LUSOL_realloc_a(LUSOL, 0); - LUSOL_realloc_r(LUSOL, 0); - LUSOL_realloc_c(LUSOL, 0); - if(LUSOL->L0 != NULL) - LUSOL_matfree(&(LUSOL->L0)); - if(LUSOL->U != NULL) - LUSOL_matfree(&(LUSOL->U)); - if(!is_nativeBLAS()) - unload_BLAS(); - LUSOL_FREE(LUSOL); -} - -void LUSOL_report(LUSOLrec *LUSOL, int msglevel, char *format, ...) -{ - va_list ap; - - if(LUSOL == NULL) { - va_start(ap, format); - vfprintf(stderr, format, ap); - va_end(ap); - } - else if(msglevel >= 0 /*LUSOL->luparm[2]*/) { - if(LUSOL->writelog != NULL) { - char buff[255]; - - va_start(ap, format); - vsprintf(buff, format, ap); - va_end(ap); - LUSOL->writelog(LUSOL, LUSOL->loghandle, buff); - } - if(LUSOL->outstream != NULL) { - va_start(ap, format); - vfprintf(LUSOL->outstream, format, ap); - va_end(ap); - fflush(LUSOL->outstream); - } - } -} - -void LUSOL_timer(LUSOLrec *LUSOL, int timerid, char *text) -{ - LUSOL_report(LUSOL, -1, "TimerID %d at %s - %s\n", - timerid, "", text); -} - -int LUSOL_factorize(LUSOLrec *LUSOL) -{ - int inform; - - LU1FAC( LUSOL, &inform ); - return( inform ); -} - -int LUSOL_ftran(LUSOLrec *LUSOL, REAL b[], int NZidx[], MYBOOL prepareupdate) -{ - int inform; - REAL *vector; - - if(prepareupdate) - vector = LUSOL->vLU6L; - else - vector = LUSOL->w; - - /* Copy RHS vector, but make adjustment for offset since this - can create a memory error when the calling program uses - a 0-base vector offset back to comply with LUSOL. */ - MEMCOPY(vector+1, b+1, LUSOL->n); - if (vector != NULL) - vector[0] = 0; - - LU6SOL(LUSOL, LUSOL_SOLVE_Aw_v, vector, b, NZidx, &inform); - LUSOL->luparm[LUSOL_IP_FTRANCOUNT]++; - - return(inform); -} - - -int LUSOL_btran(LUSOLrec *LUSOL, REAL b[], int NZidx[]) -{ - int inform; - - /* Copy RHS vector, but make adjustment for offset since this - can create a memory error when the calling program uses - a 0-base vector offset back to comply with LUSOL. */ - MEMCOPY(LUSOL->w+1, b+1, LUSOL->m); - if (LUSOL->w != NULL) - LUSOL->w[0] = 0; - - LU6SOL(LUSOL, LUSOL_SOLVE_Atv_w, b, LUSOL->w, NZidx, &inform); - LUSOL->luparm[LUSOL_IP_BTRANCOUNT]++; - - return(inform); -} - - -int LUSOL_replaceColumn(LUSOLrec *LUSOL, int jcol, REAL v[]) -{ - int inform; - REAL DIAG, VNORM; - - LU8RPC(LUSOL, LUSOL_UPDATE_OLDNONEMPTY, LUSOL_UPDATE_NEWNONEMPTY, - jcol, v, NULL, - &inform, &DIAG, &VNORM); - - LUSOL->replaced_c++; - return( inform ); -} - -REAL LUSOL_vecdensity(LUSOLrec *LUSOL, REAL V[]) -{ - int I, N = 0; - - for(I = 1; I <= LUSOL->m; I++) - if(fabs(V[I]) > 0) - N++; - return( (REAL) N / (REAL) LUSOL->m ); -} - -char relationChar(REAL left, REAL right) -{ - if(left > right) - return('>'); - else if(left == right) - return('='); - else - return('<'); -} - -/* Retrieve the core modules ordered by order of dependency */ - -#include "lusol2.c" /* Heap management */ -#include "lusol6a.c" /* Singularity checking and equation solving */ -#include "lusol1.c" /* Factorization and core components */ -#include "lusol7a.c" /* Utility routines for updates */ -#include "lusol8a.c" /* Column update */ - - -void LUSOL_dump(FILE *output, LUSOLrec *LUSOL) -{ - MYBOOL userfile = (MYBOOL) (output != NULL); - - if(!userfile) - output = fopen("LUSOL.dbg", "w"); - - blockWriteREAL(output, "a", LUSOL->a, 1, LUSOL->lena); - blockWriteINT(output, "indc", LUSOL->indc, 1, LUSOL->lena); - blockWriteINT(output, "indr", LUSOL->indr, 1, LUSOL->lena); - - blockWriteINT(output, "ip", LUSOL->ip, 1, LUSOL->m); - blockWriteINT(output, "iq", LUSOL->iq, 1, LUSOL->n); - blockWriteINT(output, "lenc", LUSOL->lenc, 1, LUSOL->n); - blockWriteINT(output, "lenr", LUSOL->lenr, 1, LUSOL->m); - - blockWriteINT(output, "locc", LUSOL->locc, 1, LUSOL->n); - blockWriteINT(output, "locr", LUSOL->locr, 1, LUSOL->m); - blockWriteINT(output, "iploc", LUSOL->iploc, 1, LUSOL->n); - blockWriteINT(output, "iqloc", LUSOL->iqloc, 1, LUSOL->m); - - blockWriteINT(output, "ipinv", LUSOL->ipinv, 1, LUSOL->m); - blockWriteINT(output, "iqinv", LUSOL->iqinv, 1, LUSOL->n); - - if(!userfile) - fclose(output); -} - -LUSOLmat *LUSOL_matcreate(int dim, int nz) -{ - LUSOLmat *newm; - - newm = (LUSOLmat *) LUSOL_CALLOC(1, sizeof(*newm)); - if(newm != NULL) { - newm->a = (REAL *) LUSOL_MALLOC((nz+1)*sizeof(REAL)); - newm->lenx = (int *) LUSOL_MALLOC((dim+1)*sizeof(int)); - newm->indx = (int *) LUSOL_MALLOC((dim+1)*sizeof(int)); - newm->indr = (int *) LUSOL_MALLOC((nz+1)*sizeof(int)); - newm->indc = (int *) LUSOL_MALLOC((nz+1)*sizeof(int)); - if((newm->a == NULL) || - (newm->lenx == NULL) || (newm->indx == NULL) || - (newm->indr == NULL) || (newm->indc == NULL)) - LUSOL_matfree(&newm); - } - return(newm); -} -void LUSOL_matfree(LUSOLmat **mat) -{ - if((mat == NULL) || (*mat == NULL)) - return; - LUSOL_FREE((*mat)->a); - LUSOL_FREE((*mat)->indc); - LUSOL_FREE((*mat)->indr); - LUSOL_FREE((*mat)->lenx); - LUSOL_FREE((*mat)->indx); - LUSOL_FREE(*mat); -} - diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.h b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.h deleted file mode 100644 index 81ee30dc..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol.h +++ /dev/null @@ -1,357 +0,0 @@ -#ifndef HEADER_LUSOL -#define HEADER_LUSOL - -/* Include necessary libraries */ -/* ------------------------------------------------------------------------- */ -#include -#include "commonlib.h" - -/* Version information */ -/* ------------------------------------------------------------------------- */ -#define LUSOL_VERMAJOR 2 -#define LUSOL_VERMINOR 2 -#define LUSOL_RELEASE 2 -#define LUSOL_BUILD 0 - -/* Dynamic memory management macros */ -/* ------------------------------------------------------------------------- */ -#ifdef MATLAB - #define LUSOL_MALLOC(bytesize) mxMalloc(bytesize) - #define LUSOL_CALLOC(count, recsize) mxCalloc(count, recsize) - #define LUSOL_REALLOC(ptr, bytesize) mxRealloc((void *) ptr, bytesize) - #define LUSOL_FREE(ptr) {mxFree(ptr); ptr=NULL;} -#else - #define LUSOL_MALLOC(bytesize) malloc(bytesize) - #define LUSOL_CALLOC(count, recsize) calloc(count, recsize) - #define LUSOL_REALLOC(ptr, bytesize) realloc((void *) ptr, bytesize) - #define LUSOL_FREE(ptr) {free(ptr); ptr=NULL;} -#endif - -/* Performance compiler options */ -/* ------------------------------------------------------------------------- */ -#if 1 - #define ForceInitialization /* Satisfy compilers, check during debugging! */ - #define LUSOLFastDenseIndex /* Increment the linearized dense address */ - #define LUSOLFastClear /* Use intrinsic functions for memory zeroing */ - #define LUSOLFastMove /* Use intrinsic functions for memory moves */ - #define LUSOLFastCopy /* Use intrinsic functions for memory copy */ - #define LUSOLFastSolve /* Use pointer operations in equation solving */ - #define LUSOLSafeFastUpdate /* Use separate array for LU6L result storage */ -/*#define UseOld_LU6CHK_20040510 */ -/*#define AlwaysSeparateHamaxR */ /* Enabled when the pivot model is fixed */ - #if 0 - #define ForceRowBasedL0 /* Create a row-sorted version of L0 */ - #endif -/* #define SetSmallToZero*/ -/* #define DoTraceL0 */ -#endif -/*#define UseTimer */ - - -/* Legacy compatibility and testing options (Fortran-LUSOL) */ -/* ------------------------------------------------------------------------- */ -#if 0 - #define LegacyTesting - #define StaticMemAlloc /* Preallocated vs. dynamic memory allocation */ - #define ClassicdiagU /* Store diagU at end of a */ - #define ClassicHamaxR /* Store H+AmaxR at end of a/indc/indr */ -#endif - - -/* General constants and data type definitions */ -/* ------------------------------------------------------------------------- */ -#define LUSOL_ARRAYOFFSET 1 -#ifndef ZERO - #define ZERO 0 -#endif -#ifndef ONE - #define ONE 1 -#endif -#ifndef FALSE - #define FALSE 0 -#endif -#ifndef TRUE - #define TRUE 1 -#endif -#ifndef NULL - #define NULL 0 -#endif -#ifndef REAL - #define REAL double -#endif -#ifndef REALXP - #define REALXP long double -#endif -#ifndef MYBOOL - #define MYBOOL unsigned char -#endif - - -/* User-settable default parameter values */ -/* ------------------------------------------------------------------------- */ -#define LUSOL_DEFAULT_GAMMA 2.0 -#define LUSOL_SMALLNUM 1.0e-20 /* IAEE doubles have precision 2.22e-16 */ -#define LUSOL_BIGNUM 1.0e+20 -#define LUSOL_MINDELTA_FACTOR 4 -#define LUSOL_MINDELTA_a 10000 -#if 1 - #define LUSOL_MULT_nz_a 2 /* Suggested by Yin Zhang */ -#else - #define LUSOL_MULT_nz_a 5 /* Could consider 6 or 7 */ -#endif -#define LUSOL_MINDELTA_rc 1000 -#define LUSOL_DEFAULT_SMARTRATIO 0.667 - -/* Fixed system parameters (changeable only by developers) */ -/* ------------------------------------------------------------------------- */ - -/* parmlu INPUT parameters: */ -#define LUSOL_RP_SMARTRATIO 0 -#define LUSOL_RP_FACTORMAX_Lij 1 -#define LUSOL_RP_UPDATEMAX_Lij 2 -#define LUSOL_RP_ZEROTOLERANCE 3 -#define LUSOL_RP_SMALLDIAG_U 4 -#define LUSOL_RP_EPSDIAG_U 5 -#define LUSOL_RP_COMPSPACE_U 6 -#define LUSOL_RP_MARKOWITZ_CONLY 7 -#define LUSOL_RP_MARKOWITZ_DENSE 8 -#define LUSOL_RP_GAMMA 9 - -/* parmlu OUPUT parameters: */ -#define LUSOL_RP_MAXELEM_A 10 -#define LUSOL_RP_MAXMULT_L 11 -#define LUSOL_RP_MAXELEM_U 12 -#define LUSOL_RP_MAXELEM_DIAGU 13 -#define LUSOL_RP_MINELEM_DIAGU 14 -#define LUSOL_RP_MAXELEM_TCP 15 -#define LUSOL_RP_GROWTHRATE 16 -#define LUSOL_RP_USERDATA_1 17 -#define LUSOL_RP_USERDATA_2 18 -#define LUSOL_RP_USERDATA_3 19 -#define LUSOL_RP_RESIDUAL_U 20 -#define LUSOL_RP_LASTITEM LUSOL_RP_RESIDUAL_U - -/* luparm INPUT parameters: */ -#define LUSOL_IP_USERDATA_0 0 -#define LUSOL_IP_PRINTUNIT 1 -#define LUSOL_IP_PRINTLEVEL 2 -#define LUSOL_IP_MARKOWITZ_MAXCOL 3 -#define LUSOL_IP_SCALAR_NZA 4 -#define LUSOL_IP_UPDATELIMIT 5 -#define LUSOL_IP_PIVOTTYPE 6 -#define LUSOL_IP_ACCELERATION 7 -#define LUSOL_IP_KEEPLU 8 -#define LUSOL_IP_SINGULARLISTSIZE 9 - -/* luparm OUTPUT parameters: */ -#define LUSOL_IP_INFORM 10 -#define LUSOL_IP_SINGULARITIES 11 -#define LUSOL_IP_SINGULARINDEX 12 -#define LUSOL_IP_MINIMUMLENA 13 -#define LUSOL_IP_MAXLEN 14 -#define LUSOL_IP_UPDATECOUNT 15 -#define LUSOL_IP_RANK_U 16 -#define LUSOL_IP_COLCOUNT_DENSE1 17 -#define LUSOL_IP_COLCOUNT_DENSE2 18 -#define LUSOL_IP_COLINDEX_DUMIN 19 -#define LUSOL_IP_COLCOUNT_L0 20 -#define LUSOL_IP_NONZEROS_L0 21 -#define LUSOL_IP_NONZEROS_U0 22 -#define LUSOL_IP_NONZEROS_L 23 -#define LUSOL_IP_NONZEROS_U 24 -#define LUSOL_IP_NONZEROS_ROW 25 -#define LUSOL_IP_COMPRESSIONS_LU 26 -#define LUSOL_IP_MARKOWITZ_MERIT 27 -#define LUSOL_IP_TRIANGROWS_U 28 -#define LUSOL_IP_TRIANGROWS_L 29 -#define LUSOL_IP_FTRANCOUNT 30 -#define LUSOL_IP_BTRANCOUNT 31 -#define LUSOL_IP_ROWCOUNT_L0 32 -#define LUSOL_IP_LASTITEM LUSOL_IP_ROWCOUNT_L0 - - -/* Macros for matrix-based access for dense part of A and timer mapping */ -/* ------------------------------------------------------------------------- */ -#define DAPOS(row, col) (row + (col-1)*LDA) -#define timer(text, id) LUSOL_timer(LUSOL, id, text) - - -/* Parameter/option defines */ -/* ------------------------------------------------------------------------- */ -#define LUSOL_MSG_NONE -1 -#define LUSOL_MSG_SINGULARITY 0 -#define LUSOL_MSG_STATISTICS 10 -#define LUSOL_MSG_PIVOT 50 - -#define LUSOL_BASEORDER 0 -#define LUSOL_OTHERORDER 1 -#define LUSOL_AUTOORDER 2 -#define LUSOL_ACCELERATE_L0 4 -#define LUSOL_ACCELERATE_U 8 - -#define LUSOL_PIVMOD_NOCHANGE -2 /* Don't change active pivoting model */ -#define LUSOL_PIVMOD_DEFAULT -1 /* Set pivoting model to default */ -#define LUSOL_PIVMOD_TPP 0 /* Threshold Partial pivoting (normal) */ -#define LUSOL_PIVMOD_TRP 1 /* Threshold Rook pivoting */ -#define LUSOL_PIVMOD_TCP 2 /* Threshold Complete pivoting */ -#define LUSOL_PIVMOD_TSP 3 /* Threshold Symmetric pivoting */ -#define LUSOL_PIVMOD_MAX LUSOL_PIVMOD_TSP - -#define LUSOL_PIVTOL_NOCHANGE 0 -#define LUSOL_PIVTOL_BAGGY 1 -#define LUSOL_PIVTOL_LOOSE 2 -#define LUSOL_PIVTOL_NORMAL 3 -#define LUSOL_PIVTOL_SLIM 4 -#define LUSOL_PIVTOL_TIGHT 5 -#define LUSOL_PIVTOL_SUPER 6 -#define LUSOL_PIVTOL_CORSET 7 -#define LUSOL_PIVTOL_DEFAULT LUSOL_PIVTOL_SLIM -#define LUSOL_PIVTOL_MAX LUSOL_PIVTOL_CORSET - -#define LUSOL_UPDATE_OLDEMPTY 0 /* No/empty current column. */ -#define LUSOL_UPDATE_OLDNONEMPTY 1 /* Current column need not have been empty. */ -#define LUSOL_UPDATE_NEWEMPTY 0 /* New column is taken to be zero. */ -#define LUSOL_UPDATE_NEWNONEMPTY 1 /* v(*) contains the new column; - on exit, v(*) satisfies L*v = a(new). */ -#define LUSOL_UPDATE_USEPREPARED 2 /* v(*) must satisfy L*v = a(new). */ - -#define LUSOL_SOLVE_Lv_v 1 /* v solves L v = v(input). w is not touched. */ -#define LUSOL_SOLVE_Ltv_v 2 /* v solves L'v = v(input). w is not touched. */ -#define LUSOL_SOLVE_Uw_v 3 /* w solves U w = v. v is not altered. */ -#define LUSOL_SOLVE_Utv_w 4 /* v solves U'v = w. w is destroyed. */ -#define LUSOL_SOLVE_Aw_v 5 /* w solves A w = v. v is altered as in 1. */ -#define LUSOL_FTRAN LUSOL_SOLVE_Aw_v -#define LUSOL_SOLVE_Atv_w 6 /* v solves A'v = w. w is destroyed. */ -#define LUSOL_BTRAN LUSOL_SOLVE_Atv_w - -/* If mode = 3,4,5,6, v and w must not be the same arrays. - If lu1fac has just been used to factorize a symmetric matrix A - (which must be definite or quasi-definite), the factors A = L U - may be regarded as A = LDL', where D = diag(U). In such cases, - the following (faster) solve codes may be used: */ -#define LUSOL_SOLVE_Av_v 7 /* v solves A v = L D L'v = v(input). w is not touched. */ -#define LUSOL_SOLVE_LDLtv_v 8 /* v solves L |D| L'v = v(input). w is not touched. */ - -#define LUSOL_INFORM_RANKLOSS -1 -#define LUSOL_INFORM_LUSUCCESS 0 -#define LUSOL_INFORM_LUSINGULAR 1 -#define LUSOL_INFORM_LUUNSTABLE 2 -#define LUSOL_INFORM_ADIMERR 3 -#define LUSOL_INFORM_ADUPLICATE 4 -#define LUSOL_INFORM_ANEEDMEM 7 /* Set lena >= luparm[LUSOL_IP_MINIMUMLENA] */ -#define LUSOL_INFORM_FATALERR 8 -#define LUSOL_INFORM_NOPIVOT 9 /* No diagonal pivot found with TSP or TDP. */ -#define LUSOL_INFORM_NOMEMLEFT 10 - -#define LUSOL_INFORM_MIN LUSOL_INFORM_RANKLOSS -#define LUSOL_INFORM_MAX LUSOL_INFORM_NOMEMLEFT - -#define LUSOL_INFORM_GETLAST 10 /* Code for LUSOL_informstr. */ -#define LUSOL_INFORM_SERIOUS LUSOL_INFORM_LUUNSTABLE - - -/* Prototypes for call-back functions */ -/* ------------------------------------------------------------------------- */ -typedef void LUSOLlogfunc(void *lp, void *userhandle, char *buf); - - -/* Sparse matrix data */ -typedef struct _LUSOLmat { - REAL *a; - int *lenx, *indr, *indc, *indx; -} LUSOLmat; - - -/* The main LUSOL data record */ -/* ------------------------------------------------------------------------- */ -typedef struct _LUSOLrec { - - /* General data */ - FILE *outstream; /* Output stream, initialized to STDOUT */ - LUSOLlogfunc *writelog; - void *loghandle; - LUSOLlogfunc *debuginfo; - - /* Parameter storage arrays */ - int luparm[LUSOL_IP_LASTITEM + 1]; - REAL parmlu[LUSOL_RP_LASTITEM + 1]; - - /* Arrays of length lena+1 */ - int lena, nelem; - int *indc, *indr; - REAL *a; - - /* Arrays of length maxm+1 (row storage) */ - int maxm, m; - int *lenr, *ip, *iqloc, *ipinv, *locr; - - /* Arrays of length maxn+1 (column storage) */ - int maxn, n; - int *lenc, *iq, *iploc, *iqinv, *locc; - REAL *w, *vLU6L; - - /* List of singular columns, with dynamic size allocation */ - int *isingular; - - /* Extra arrays of length n for TCP and keepLU == FALSE */ - REAL *Ha, *diagU; - int *Hj, *Hk; - - /* Extra arrays of length m for TRP*/ - REAL *amaxr; - - /* Extra array for L0 and U stored by row/column for faster btran/ftran */ - LUSOLmat *L0; - LUSOLmat *U; - - /* Miscellaneous data */ - int expanded_a; - int replaced_c; - int replaced_r; - -} LUSOLrec; - - -LUSOLrec *LUSOL_create(FILE *outstream, int msgfil, int pivotmodel, int updatelimit); -MYBOOL LUSOL_sizeto(LUSOLrec *LUSOL, int init_r, int init_c, int init_a); -MYBOOL LUSOL_assign(LUSOLrec *LUSOL, int iA[], int jA[], REAL Aij[], - int nzcount, MYBOOL istriplet); -void LUSOL_clear(LUSOLrec *LUSOL, MYBOOL nzonly); -void LUSOL_free(LUSOLrec *LUSOL); - -LUSOLmat *LUSOL_matcreate(int dim, int nz); -void LUSOL_matfree(LUSOLmat **mat); - -int LUSOL_loadColumn(LUSOLrec *LUSOL, int iA[], int jA, REAL Aij[], int nzcount, int offset1); -void LUSOL_setpivotmodel(LUSOLrec *LUSOL, int pivotmodel, int initlevel); -int LUSOL_factorize(LUSOLrec *LUSOL); -int LUSOL_replaceColumn(LUSOLrec *LUSOL, int jcol, REAL v[]); - -MYBOOL LUSOL_tightenpivot(LUSOLrec *LUSOL); -MYBOOL LUSOL_addSingularity(LUSOLrec *LUSOL, int singcol, int *inform); -int LUSOL_getSingularity(LUSOLrec *LUSOL, int singitem); -int LUSOL_findSingularityPosition(LUSOLrec *LUSOL, int singcol); - -char *LUSOL_pivotLabel(LUSOLrec *LUSOL); -char *LUSOL_informstr(LUSOLrec *LUSOL, int inform); -REAL LUSOL_vecdensity(LUSOLrec *LUSOL, REAL V[]); -void LUSOL_report(LUSOLrec *LUSOL, int msglevel, char *format, ...); -void LUSOL_timer(LUSOLrec *LUSOL, int timerid, char *text); - -int LUSOL_ftran(LUSOLrec *LUSOL, REAL b[], int NZidx[], MYBOOL prepareupdate); -int LUSOL_btran(LUSOLrec *LUSOL, REAL b[], int NZidx[]); - -void LU1FAC(LUSOLrec *LUSOL, int *INFORM); -MYBOOL LU1L0(LUSOLrec *LUSOL, LUSOLmat **mat, int *inform); -void LU6SOL(LUSOLrec *LUSOL, int MODE, REAL V[], REAL W[], int NZidx[], int *INFORM); -void LU8RPC(LUSOLrec *LUSOL, int MODE1, int MODE2, - int JREP, REAL V[], REAL W[], - int *INFORM, REAL *DIAG, REAL *VNORM); - -void LUSOL_dump(FILE *output, LUSOLrec *LUSOL); - - -void print_L0(LUSOLrec *LUSOL); - - -#endif /* HEADER_LUSOL */ diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol1.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol1.c deleted file mode 100644 index 1180c43f..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol1.c +++ /dev/null @@ -1,3725 +0,0 @@ - -/* ================================================================== - lu1DCP factors a dense m x n matrix A by Gaussian elimination, - using Complete Pivoting (row and column interchanges) for stability. - This version also uses column interchanges if all elements in a - pivot column are smaller than (or equal to) "small". Such columns - are changed to zero and permuted to the right-hand end. - As in LINPACK's dgefa, ipvt(!) keeps track of pivot rows. - Rows of U are interchanged, but we don't have to physically - permute rows of L. In contrast, column interchanges are applied - directly to the columns of both L and U, and to the column - permutation vector iq(*). - ------------------------------------------------------------------ - On entry: - a Array holding the matrix A to be factored. - lda The leading dimension of the array a. - m The number of rows in A. - n The number of columns in A. - small A drop tolerance. Must be zero or positive. - - On exit: - a An upper triangular matrix and the multipliers - which were used to obtain it. - The factorization can be written A = L*U where - L is a product of permutation and unit lower - triangular matrices and U is upper triangular. - nsing Number of singularities detected. - ipvt Records the pivot rows. - iq A vector to which column interchanges are applied. - ------------------------------------------------------------------ - 01 May 2002: First dense Complete Pivoting, derived from lu1DPP. - 07 May 2002: Another break needed at end of first loop. - 07 May 2002: Current version of lu1DCP. - ================================================================== */ -void LU1DCP(LUSOLrec *LUSOL, REAL DA[], int LDA, int M, int N, REAL SMALL, - int *NSING, int IPVT[], int IX[]) -{ - - int I, J, K, KP1, L, LAST, LENCOL, IMAX, JMAX, JLAST, JNEW; - REAL AIJMAX, AJMAX; - register REAL T; -#ifdef LUSOLFastDenseIndex - register REAL *DA1, *DA2; - int IDA1, IDA2; -#else - register int IDA1, IDA2; -#endif - - *NSING = 0; - LENCOL = M+1; - LAST = N; -/* ----------------------------------------------------------------- - Start of elimination loop. - ----------------------------------------------------------------- */ - for(K = 1; K <= N; K++) { - KP1 = K+1; - LENCOL--; -/* Find the biggest aij in row imax and column jmax. */ - AIJMAX = ZERO; - IMAX = K; - JMAX = K; - JLAST = LAST; - for(J = K; J <= JLAST; J++) { -x10: - L = idamax(LENCOL,DA+DAPOS(K,J)-LUSOL_ARRAYOFFSET,1)+K-1; - AJMAX = fabs(DA[DAPOS(L,J)]); - if(AJMAX<=SMALL) { -/* ======================================================== - Do column interchange, changing old column to zero. - Reduce "last" and try again with same j. - ======================================================== */ - (*NSING)++; - JNEW = IX[LAST]; - IX[LAST] = IX[J]; - IX[J] = JNEW; -#ifdef LUSOLFastDenseIndex - DA1 = DA+DAPOS(0,LAST); - DA2 = DA+DAPOS(0,J); - for(I = 1; I <= K-1; I++) { - DA1++; - DA2++; - T = *DA1; - *DA1 = *DA2; - *DA2 = T; -#else - for(I = 1; I <= K-1; I++) { - IDA1 = DAPOS(I,LAST); - IDA2 = DAPOS(I,J); - T = DA[IDA1]; - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; -#endif - } -#ifdef LUSOLFastDenseIndex - for(I = K; I <= M; I++) { - DA1++; - DA2++; - T = *DA1; - *DA1 = ZERO; - *DA2 = T; -#else - for(I = K; I <= M; I++) { - IDA1 = DAPOS(I,LAST); - IDA2 = DAPOS(I,J); - T = DA[IDA1]; - DA[IDA1] = ZERO; - DA[IDA2] = T; -#endif - } - LAST--; - if(J<=LAST) - goto x10; - break; - } -/* Check if this column has biggest aij so far. */ - if(AIJMAX=LAST) - break; - } - IPVT[K] = IMAX; - if(JMAX!=K) { -/* ========================================================== - Do column interchange (k and jmax). - ========================================================== */ - JNEW = IX[JMAX]; - IX[JMAX] = IX[K]; - IX[K] = JNEW; -#ifdef LUSOLFastDenseIndex - DA1 = DA+DAPOS(0,JMAX); - DA2 = DA+DAPOS(0,K); - for(I = 1; I <= M; I++) { - DA1++; - DA2++; - T = *DA1; - *DA1 = *DA2; - *DA2 = T; -#else - for(I = 1; I <= M; I++) { - IDA1 = DAPOS(I,JMAX); - IDA2 = DAPOS(I,K); - T = DA[IDA1]; - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; -#endif - } - } - if(M>K) { -/* =========================================================== - Do row interchange if necessary. - =========================================================== */ - if(IMAX!=K) { - IDA1 = DAPOS(IMAX,K); - IDA2 = DAPOS(K,K); - T = DA[IDA1]; - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; - } -/* =========================================================== - Compute multipliers. - Do row elimination with column indexing. - =========================================================== */ - T = -ONE/DA[DAPOS(K,K)]; - dscal(M-K,T,DA+DAPOS(KP1,K)-LUSOL_ARRAYOFFSET,1); - for(J = KP1; J <= LAST; J++) { - IDA1 = DAPOS(IMAX,J); - T = DA[IDA1]; - if(IMAX!=K) { - IDA2 = DAPOS(K,J); - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; - } - daxpy(M-K,T,DA+DAPOS(KP1,K)-LUSOL_ARRAYOFFSET,1, - DA+DAPOS(KP1,J)-LUSOL_ARRAYOFFSET,1); - } - } - else - break; - if(K>=LAST) - break; - } -/* Set ipvt(*) for singular rows. */ - for(K = LAST+1; K <= M; K++) - IPVT[K] = K; - -} - -/* ================================================================== - lu1DPP factors a dense m x n matrix A by Gaussian elimination, - using row interchanges for stability, as in dgefa from LINPACK. - This version also uses column interchanges if all elements in a - pivot column are smaller than (or equal to) "small". Such columns - are changed to zero and permuted to the right-hand end. - As in LINPACK, ipvt(*) keeps track of pivot rows. - Rows of U are interchanged, but we don't have to physically - permute rows of L. In contrast, column interchanges are applied - directly to the columns of both L and U, and to the column - permutation vector iq(*). - ------------------------------------------------------------------ - On entry: - a Array holding the matrix A to be factored. - lda The leading dimension of the array a. - m The number of rows in A. - n The number of columns in A. - small A drop tolerance. Must be zero or positive. - - On exit: - a An upper triangular matrix and the multipliers - which were used to obtain it. - The factorization can be written A = L*U where - L is a product of permutation and unit lower - triangular matrices and U is upper triangular. - nsing Number of singularities detected. - ipvt Records the pivot rows. - iq A vector to which column interchanges are applied. - ------------------------------------------------------------------ - 02 May 1989: First version derived from dgefa - in LINPACK (version dated 08/14/78). - 05 Feb 1994: Generalized to treat rectangular matrices - and use column interchanges when necessary. - ipvt is retained, but column permutations are applied - directly to iq(*). - 21 Dec 1994: Bug found via example from Steve Dirkse. - Loop 100 added to set ipvt(*) for singular rows. - ================================================================== */ -void LU1DPP(LUSOLrec *LUSOL, REAL DA[], int LDA, int M, int N, REAL SMALL, - int *NSING, int IPVT[], int IX[]) -{ - int I, J, K, KP1, L, LAST, LENCOL; - register REAL T; -#ifdef LUSOLFastDenseIndex - register REAL *DA1, *DA2; - int IDA1, IDA2; -#else - register int IDA1, IDA2; -#endif - - *NSING = 0; - K = 1; - LAST = N; -/* ------------------------------------------------------------------ - Start of elimination loop. - ------------------------------------------------------------------ */ -x10: - KP1 = K+1; - LENCOL = (M-K)+1; -/* Find l, the pivot row. */ - L = (idamax(LENCOL,DA+DAPOS(K,K)-LUSOL_ARRAYOFFSET,1)+K)-1; - IPVT[K] = L; - if(fabs(DA[DAPOS(L,K)])<=SMALL) { -/* =============================================================== - Do column interchange, changing old pivot column to zero. - Reduce "last" and try again with same k. - =============================================================== */ - (*NSING)++; - J = IX[LAST]; - IX[LAST] = IX[K]; - IX[K] = J; -#ifdef LUSOLFastDenseIndex - DA1 = DA+DAPOS(0,LAST); - DA2 = DA+DAPOS(0,K); - for(I = 1; I <= K-1; I++) { - DA1++; - DA2++; - T = *DA1; - *DA1 = *DA2; - *DA2 = T; -#else - for(I = 1; I <= K-1; I++) { - IDA1 = DAPOS(I,LAST); - IDA2 = DAPOS(I,K); - T = DA[IDA1]; - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; -#endif - } -#ifdef LUSOLFastDenseIndex - for(I = K; I <= M; I++) { - DA1++; - DA2++; - T = *DA1; - *DA1 = ZERO; - *DA2 = T; -#else - for(I = K; I <= M; I++) { - IDA1 = DAPOS(I,LAST); - IDA2 = DAPOS(I,K); - T = DA[IDA1]; - DA[IDA1] = ZERO; - DA[IDA2] = T; -#endif - } - LAST = LAST-1; - if(K<=LAST) - goto x10; - } - else if(M>K) { -/* =============================================================== - Do row interchange if necessary. - =============================================================== */ - if(L!=K) { - IDA1 = DAPOS(L,K); - IDA2 = DAPOS(K,K); - T = DA[IDA1]; - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; - } -/* =============================================================== - Compute multipliers. - Do row elimination with column indexing. - =============================================================== */ - T = -ONE/DA[DAPOS(K,K)]; - dscal(M-K,T,DA+DAPOS(KP1,K)-LUSOL_ARRAYOFFSET,1); - for(J = KP1; J <= LAST; J++) { - IDA1 = DAPOS(L,J); - T = DA[IDA1]; - if(L!=K) { - IDA2 = DAPOS(K,J); - DA[IDA1] = DA[IDA2]; - DA[IDA2] = T; - } - daxpy(M-K,T,DA+DAPOS(KP1,K)-LUSOL_ARRAYOFFSET,1, - DA+DAPOS(KP1,J)-LUSOL_ARRAYOFFSET,1); - } - K++; - if(K<=LAST) - goto x10; - } -/* Set ipvt(*) for singular rows. */ - for(K = LAST+1; K <= M; K++) - IPVT[K] = K; - -} - - -/* ================================================================== - lu1pq1 constructs a permutation iperm from the array len. - ------------------------------------------------------------------ - On entry: - len(i) holds the number of nonzeros in the i-th row (say) - of an m by n matrix. - num(*) can be anything (workspace). - - On exit: - iperm contains a list of row numbers in the order - rows of length 0, rows of length 1,..., rows of length n. - loc(nz) points to the first row containing nz nonzeros, - nz = 1, n. - inv(i) points to the position of row i within iperm(*). - ================================================================== */ -void LU1PQ1(LUSOLrec *LUSOL, int M, int N, int LEN[], - int IPERM[], int LOC[], int INV[], int NUM[]) -{ - int NZEROS, NZ, I, L; - -/* Count the number of rows of each length. */ - NZEROS = 0; - for(NZ = 1; NZ <= N; NZ++) { - NUM[NZ] = 0; - LOC[NZ] = 0; - } - for(I = 1; I <= M; I++) { - NZ = LEN[I]; - if(NZ==0) - NZEROS++; - else - NUM[NZ]++; - } -/* Set starting locations for each length. */ - L = NZEROS+1; - for(NZ = 1; NZ <= N; NZ++) { - LOC[NZ] = L; - L += NUM[NZ]; - NUM[NZ] = 0; - } -/* Form the list. */ - NZEROS = 0; - for(I = 1; I <= M; I++) { - NZ = LEN[I]; - if(NZ==0) { - NZEROS++; - IPERM[NZEROS] = I; - } - else { - L = LOC[NZ]+NUM[NZ]; - IPERM[L] = I; - NUM[NZ]++; - } - } -/* Define the inverse of iperm. */ - for(L = 1; L <= M; L++) { - I = IPERM[L]; - INV[I] = L; - } -} - -/* ================================================================== - lu1pq2 frees the space occupied by the pivot row, - and updates the column permutation iq. - Also used to free the pivot column and update the row perm ip. - ------------------------------------------------------------------ - nzpiv (input) is the length of the pivot row (or column). - nzchng (output) is the net change in total nonzeros. - ------------------------------------------------------------------ - 14 Apr 1989 First version. - ================================================================== */ -void LU1PQ2(LUSOLrec *LUSOL, int NZPIV, int *NZCHNG, - int IND[], int LENOLD[], int LENNEW[], int IXLOC[], int IX[], int IXINV[]) -{ - int LR, J, NZ, NZNEW, L, NEXT, LNEW, JNEW; - - *NZCHNG = 0; - for(LR = 1; LR <= NZPIV; LR++) { - J = IND[LR]; - IND[LR] = 0; - NZ = LENOLD[LR]; - NZNEW = LENNEW[J]; - if(NZ!=NZNEW) { - L = IXINV[J]; - *NZCHNG = (*NZCHNG+NZNEW)-NZ; -/* l above is the position of column j in iq (so j = iq(l)). */ - if(NZNZNEW) - goto x120; - } - IX[LNEW] = J; - IXINV[J] = LNEW; - } - } -} - -/* ================================================================== - lu1pq3 looks at the permutation iperm(*) and moves any entries - to the end whose corresponding length len(*) is zero. - ------------------------------------------------------------------ - 09 Feb 1994: Added work array iw(*) to improve efficiency. - ================================================================== */ -void LU1PQ3(LUSOLrec *LUSOL, int MN, int LEN[], int IPERM[], int IW[], int *NRANK) -{ - int NZEROS, K, I; - - *NRANK = 0; - NZEROS = 0; - for(K = 1; K <= MN; K++) { - I = IPERM[K]; - if(LEN[I]==0) { - NZEROS++; - IW[NZEROS] = I; - } - else { - (*NRANK)++; - IPERM[*NRANK] = I; - } - } - for(K = 1; K <= NZEROS; K++) - IPERM[(*NRANK)+K] = IW[K]; -} - -/* ================================================================== - lu1rec - ------------------------------------------------------------------ - On exit: - ltop is the length of useful entries in ind(*), a(*). - ind(ltop+1) is "i" such that len(i), loc(i) belong to the last - item in ind(*), a(*). - ------------------------------------------------------------------ - 00 Jun 1983: Original version of lu1rec followed John Reid's - compression routine in LA05. It recovered - space in ind(*) and optionally a(*) - by eliminating entries with ind(l) = 0. - The elements of ind(*) could not be negative. - If len(i) was positive, entry i contained - that many elements, starting at loc(i). - Otherwise, entry i was eliminated. - 23 Mar 2001: Realised we could have len(i) = 0 in rare cases! - (Mostly during TCP when the pivot row contains - a column of length 1 that couldn't be a pivot.) - Revised storage scheme to - keep entries with ind(l) > 0, - squeeze out entries with -n <= ind(l) <= 0, - and to allow len(i) = 0. - Empty items are moved to the end of the compressed - ind(*) and/or a(*) arrays are given one empty space. - Items with len(i) < 0 are still eliminated. - 27 Mar 2001: Decided to use only ind(l) > 0 and = 0 in lu1fad. - Still have to keep entries with len(i) = 0. - ================================================================== */ -void LU1REC(LUSOLrec *LUSOL, int N, MYBOOL REALS, int *LTOP, - int IND[], int LEN[], int LOC[]) -{ - int NEMPTY, I, LENI, L, LEND, K, KLAST, ILAST, LPRINT; - - NEMPTY = 0; - for(I = 1; I <= N; I++) { - LENI = LEN[I]; - if(LENI>0) { - L = (LOC[I]+LENI)-1; - LEN[I] = IND[L]; - IND[L] = -(N+I); - } - else if(LENI==0) - NEMPTY++; - } - K = 0; -/* Previous k */ - KLAST = 0; -/* Last entry moved. */ - ILAST = 0; - LEND = *LTOP; - for(L = 1; L <= LEND; L++) { - I = IND[L]; - if(I>0) { - K++; - IND[K] = I; - if(REALS) - LUSOL->a[K] = LUSOL->a[L]; - } - else if(I<-N) { -/* This is the end of entry i. */ - I = -(N+I); - ILAST = I; - K++; - IND[K] = LEN[I]; - if(REALS) - LUSOL->a[K] = LUSOL->a[L]; - LOC[I] = KLAST+1; - LEN[I] = K-KLAST; - KLAST = K; - } - } -/* Move any empty items to the end, adding 1 free entry for each. */ - if(NEMPTY>0) { - for(I = 1; I <= N; I++) { - if(LEN[I]==0) { - K++; - LOC[I] = K; - IND[K] = 0; - ILAST = I; - } - } - } - LPRINT = LUSOL->luparm[LUSOL_IP_PRINTLEVEL]; - if(LPRINT>=LUSOL_MSG_PIVOT) - LUSOL_report(LUSOL, 0, "lu1rec. File compressed from %d to %d\n", - *LTOP,K,REALS,NEMPTY); -/* ncp */ - LUSOL->luparm[LUSOL_IP_COMPRESSIONS_LU]++; -/* Return ilast in ind(ltop + 1). */ - *LTOP = K; - IND[(*LTOP)+1] = ILAST; -} - -/* ================================================================== - lu1slk sets w(j) > 0 if column j is a unit vector. - ------------------------------------------------------------------ - 21 Nov 2000: First version. lu1fad needs it for TCP. - Note that w(*) is nominally an integer array, - but the only spare space is the double array w(*). - ================================================================== */ -void LU1SLK(LUSOLrec *LUSOL) -{ - int J, LC1, LQ, LQ1, LQ2; - - for(J = 1; J <= LUSOL->n; J++) { - LUSOL->w[J] = 0; - } - LQ1 = (LUSOL->iqloc ? LUSOL->iqloc[1] : LUSOL->n+1); -/* LQ1 = LUSOL->iqloc[1]; This is the original version; correction above by Yin Zhang */ - LQ2 = LUSOL->n; - if(LUSOL->m>1) - LQ2 = LUSOL->iqloc[2]-1; - for(LQ = LQ1; LQ <= LQ2; LQ++) { - J = LUSOL->iq[LQ]; - LC1 = LUSOL->locc[J]; - if(fabs(LUSOL->a[LC1])==1) { - LUSOL->w[J] = 1; - } - } -} - -/* ================================================================== - lu1gau does most of the work for each step of Gaussian elimination. - A multiple of the pivot column is added to each other column j - in the pivot row. The column list is fully updated. - The row list is updated if there is room, but some fill-ins may - remain, as indicated by ifill and jfill. - ------------------------------------------------------------------ - Input: - ilast is the row at the end of the row list. - jlast is the column at the end of the column list. - lfirst is the first column to be processed. - lu + 1 is the corresponding element of U in au(*). - nfill keeps track of pending fill-in. - a(*) contains the nonzeros for each column j. - indc(*) contains the row indices for each column j. - al(*) contains the new column of L. A multiple of it is - used to modify each column. - mark(*) has been set to -1, -2, -3, ... in the rows - corresponding to nonzero 1, 2, 3, ... of the col of L. - au(*) contains the new row of U. Each nonzero gives the - required multiple of the column of L. - - Workspace: - markl(*) marks the nonzeros of L actually used. - (A different mark, namely j, is used for each column.) - - Output: - ilast New last row in the row list. - jlast New last column in the column list. - lfirst = 0 if all columns were completed, - > 0 otherwise. - lu returns the position of the last nonzero of U - actually used, in case we come back in again. - nfill keeps track of the total extra space needed in the - row file. - ifill(ll) counts pending fill-in for rows involved in the new - column of L. - jfill(lu) marks the first pending fill-in stored in columns - involved in the new row of U. - ------------------------------------------------------------------ - 16 Apr 1989: First version of lu1gau. - 23 Apr 1989: lfirst, lu, nfill are now input and output - to allow re-entry if elimination is interrupted. - 23 Mar 2001: Introduced ilast, jlast. - 27 Mar 2001: Allow fill-in "in situ" if there is already room - up to but NOT INCLUDING the end of the - row or column file. - Seems safe way to avoid overwriting empty rows/cols - at the end. (May not be needed though, now that we - have ilast and jlast.) - ================================================================== */ -void LU1GAU(LUSOLrec *LUSOL, int MELIM, int NSPARE, - REAL SMALL, int LPIVC1, int LPIVC2, int *LFIRST, int LPIVR2, - int LFREE, int MINFRE, int ILAST, int *JLAST, int *LROW, int *LCOL, - int *LU, int *NFILL, - int MARK[], REAL AL[], int MARKL[], REAL AU[], int IFILL[], int JFILL[]) -{ - MYBOOL ATEND; - int LR, J, LENJ, NFREE, LC1, LC2, NDONE, NDROP, L, I, LL, K, - LR1, LAST, LREP, L1, L2, LC, LENI; - register REAL UJ; - REAL AIJ; - - for(LR = *LFIRST; LR <= LPIVR2; LR++) { - J = LUSOL->indr[LR]; - LENJ = LUSOL->lenc[J]; - NFREE = LFREE - *LCOL; - if(NFREElocc[J]; - LC2 = (LC1+LENJ)-1; - ATEND = (MYBOOL) (J==*JLAST); - NDONE = 0; - if(LENJ==0) - goto x500; - NDROP = 0; - for(L = LC1; L <= LC2; L++) { - I = LUSOL->indc[L]; - LL = -MARK[I]; - if(LL>0) { - NDONE++; - MARKL[LL] = J; - LUSOL->a[L] += AL[LL]*UJ; - if(fabs(LUSOL->a[L])<=SMALL) { - NDROP++; - } - } - } -/* --------------------------------------------------------------- - Remove any negligible modified nonzeros from both - the column file and the row file. - --------------------------------------------------------------- */ - if(NDROP==0) - goto x500; - K = LC1; - for(L = LC1; L <= LC2; L++) { - I = LUSOL->indc[L]; - if(fabs(LUSOL->a[L])<=SMALL) - goto x460; - LUSOL->a[K] = LUSOL->a[L]; - LUSOL->indc[K] = I; - K++; - continue; -/* Delete the nonzero from the row file. */ -x460: - LENJ--; - LUSOL->lenr[I]--; - LR1 = LUSOL->locr[I]; - LAST = LR1+LUSOL->lenr[I]; - for(LREP = LR1; LREP <= LAST; LREP++) { - if(LUSOL->indr[LREP]==J) - break; - } - LUSOL->indr[LREP] = LUSOL->indr[LAST]; - LUSOL->indr[LAST] = 0; - if(I==ILAST) - (*LROW)--; - } -/* Free the deleted elements from the column file. */ -#ifdef LUSOLFastClear - MEMCLEAR(LUSOL->indc+K, LC2-K+1); -#else - for(L = K; L <= LC2; L++) - LUSOL->indc[L] = ZERO; -#endif - if(ATEND) - *LCOL = K-1; -/* --------------------------------------------------------------- - Deal with the fill-in in column j. - --------------------------------------------------------------- */ -x500: - if(NDONE==MELIM) - goto x590; -/* See if column j already has room for the fill-in. */ - if(ATEND) - goto x540; - LAST = (LC1+LENJ)-1; - L1 = LAST+1; - L2 = (LAST+MELIM)-NDONE; -/* 27 Mar 2001: Be sure it's not at or past end of the col file. */ - if(L2>=*LCOL) - goto x520; - for(L = L1; L <= L2; L++) { - if(LUSOL->indc[L]!=0) - goto x520; - } - goto x540; -/* We must move column j to the end of the column file. - First, leave some spare room at the end of the - current last column. */ -x520: -#if 1 - L1 = (*LCOL)+1; - L2 = (*LCOL)+NSPARE; - *LCOL = L2; - for(L = L1; L <= L2; L++) { -#else - for(L = (*LCOL)+1; L <= (*LCOL)+NSPARE; L++) { - *LCOL = L; /* ****** ERROR ???? */ -#endif -/* Spare space is free. */ - LUSOL->indc[L] = 0; - } - ATEND = TRUE; - *JLAST = J; - L1 = LC1; - L2 = *LCOL; - LC1 = L2+1; - LUSOL->locc[J] = LC1; - for(L = L1; L <= LAST; L++) { - L2++; - LUSOL->a[L2] = LUSOL->a[L]; - LUSOL->indc[L2] = LUSOL->indc[L]; -/* Free space. */ - LUSOL->indc[L] = 0; - } - *LCOL = L2; -/* --------------------------------------------------------------- - Inner loop for the fill-in in column j. - This is usually not very expensive. - --------------------------------------------------------------- */ -x540: - LAST = (LC1+LENJ)-1; - LL = 0; - for(LC = LPIVC1; LC <= LPIVC2; LC++) { - LL++; - if(MARKL[LL]==J) - continue; - AIJ = AL[LL]*UJ; - if(fabs(AIJ)<=SMALL) - continue; - LENJ++; - LAST++; - LUSOL->a[LAST] = AIJ; - I = LUSOL->indc[LC]; - LUSOL->indc[LAST] = I; - LENI = LUSOL->lenr[I]; -/* Add 1 fill-in to row i if there is already room. - 27 Mar 2001: Be sure it's not at or past the } - of the row file. */ - L = LUSOL->locr[I]+LENI; - if(L>=*LROW) - goto x550; - if(LUSOL->indr[L]>0) - goto x550; - LUSOL->indr[L] = J; - LUSOL->lenr[I] = LENI+1; - continue; -/* Row i does not have room for the fill-in. - Increment ifill(ll) to count how often this has - happened to row i. Also, add m to the row index - indc(last) in column j to mark it as a fill-in that is - still pending. - If this is the first pending fill-in for row i, - nfill includes the current length of row i - (since the whole row has to be moved later). - If this is the first pending fill-in for column j, - jfill(lu) records the current length of column j - (to shorten the search for pending fill-ins later). */ -x550: - if(IFILL[LL]==0) - (*NFILL) += LENI+NSPARE; - if(JFILL[*LU]==0) - JFILL[*LU] = LENJ; - (*NFILL)++; - IFILL[LL]++; - LUSOL->indc[LAST] = LUSOL->m+I; - } - if(ATEND) - *LCOL = LAST; -/* End loop for column j. Store its final length. */ -x590: - LUSOL->lenc[J] = LENJ; - } -/* Successful completion. */ - *LFIRST = 0; - return; -/* Interruption. We have to come back in after the - column file is compressed. Give lfirst a new value. - lu and nfill will retain their current values. */ -x900: - *LFIRST = LR; -} - -/* ================================================================== - lu1mar uses a Markowitz criterion to select a pivot element - for the next stage of a sparse LU factorization, - subject to a Threshold Partial Pivoting stability criterion (TPP) - that bounds the elements of L. - ------------------------------------------------------------------ - gamma is "gamma" in the tie-breaking rule TB4 in the LUSOL paper. - ------------------------------------------------------------------ - Search cols of length nz = 1, then rows of length nz = 1, - then cols of length nz = 2, then rows of length nz = 2, etc. - ------------------------------------------------------------------ - 00 Jan 1986 Version documented in LUSOL paper: - Gill, Murray, Saunders and Wright (1987), - Maintaining LU factors of a general sparse matrix, - Linear algebra and its applications 88/89, 239-270. - 02 Feb 1989 Following Suhl and Aittoniemi (1987), the largest - element in each column is now kept at the start of - the column, i.e. in position locc(j) of a and indc. - This should speed up the Markowitz searches. - 26 Apr 1989 Both columns and rows searched during spars1 phase. - Only columns searched during spars2 phase. - maxtie replaced by maxcol and maxrow. - 05 Nov 1993 Initializing "mbest = m * n" wasn't big enough when - m = 10, n = 3, and last column had 7 nonzeros. - 09 Feb 1994 Realised that "mbest = maxmn * maxmn" might overflow. - Changed to "mbest = maxmn * 1000". - 27 Apr 2000 On large example from Todd Munson, - that allowed "if (mbest .le. nz1**2) go to 900" - to exit before any pivot had been found. - Introduced kbest = mbest / nz1. - Most pivots can be rejected with no integer multiply. - TRUE merit is evaluated only if it's as good as the - best so far (or better). There should be no danger - of integer overflow unless A is incredibly - large and dense. - 10 Sep 2000 TCP, aijtol added for Threshold Complete Pivoting. - ================================================================== */ -void LU1MAR(LUSOLrec *LUSOL, int MAXMN, MYBOOL TCP, REAL AIJTOL, REAL LTOL, - int MAXCOL, int MAXROW, int *IBEST, int *JBEST, int *MBEST) -{ - int KBEST, NCOL, NROW, NZ1, NZ, LQ1, LQ2, LQ, J, LC1, LC2, LC, I, LEN1, MERIT, LP1, - LP2, LP, LR1, LR2, LR; - REAL ABEST, LBEST, AMAX, AIJ, CMAX; - - ABEST = ZERO; - LBEST = ZERO; - *IBEST = 0; - *MBEST = -1; - KBEST = MAXMN+1; - NCOL = 0; - NROW = 0; - NZ1 = 0; - for(NZ = 1; NZ <= MAXMN; NZ++) { -/* nz1 = nz - 1 - if (mbest .le. nz1**2) go to 900 */ - if(KBEST<=NZ1) - goto x900; - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x200; - } - if(NZ>LUSOL->m) - goto x200; -/* --------------------------------------------------------------- - Search the set of columns of length nz. - --------------------------------------------------------------- */ - LQ1 = LUSOL->iqloc[NZ]; - LQ2 = LUSOL->n; - if(NZm) - LQ2 = LUSOL->iqloc[NZ+1]-1; - for(LQ = LQ1; LQ <= LQ2; LQ++) { - NCOL = NCOL+1; - J = LUSOL->iq[LQ]; - LC1 = LUSOL->locc[J]; - LC2 = LC1+NZ1; - AMAX = fabs(LUSOL->a[LC1]); -/* Test all aijs in this column. - amax is the largest element (the first in the column). - cmax is the largest multiplier if aij becomes pivot. */ - if(TCP) { -/* Nothing in whole column */ - if(AMAXindc[LC]; - LEN1 = LUSOL->lenr[I]-1; -/* merit = nz1 * len1 - if (merit > mbest) continue; */ - if(LEN1>KBEST) - continue; -/* aij has a promising merit. - Apply the stability test. - We require aij to be sufficiently large compared to - all other nonzeros in column j. This is equivalent - to requiring cmax to be bounded by Ltol. */ - if(LC==LC1) { -/* This is the maximum element, amax. - Find the biggest element in the rest of the column - and hence get cmax. We know cmax .le. 1, but - we still want it exactly in order to break ties. - 27 Apr 2002: Settle for cmax = 1. */ - AIJ = AMAX; - CMAX = ONE; -/* cmax = zero - for (l = lc1 + 1; l <= lc2; l++) - cmax = max( cmax, abs( a(l) ) ); - cmax = cmax / amax; */ - } - else { -/* aij is not the biggest element, so cmax .ge. 1. - Bail out if cmax will be too big. */ - AIJ = fabs(LUSOL->a[LC]); -/* Absolute test for Complete Pivoting */ - if(TCP) { - if(AIJparmlu[LUSOL_RP_GAMMA] && - CMAX<=LUSOL->parmlu[LUSOL_RP_GAMMA]) { - if(ABEST>=AIJ) - continue; - } - else { - if(LBEST<=CMAX) - continue; - } - } -/* aij is the best pivot so far. */ - *IBEST = I; - *JBEST = J; - KBEST = LEN1; - *MBEST = MERIT; - ABEST = AIJ; - LBEST = CMAX; - if(NZ==1) - goto x900; - } -/* Finished with that column. */ - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x200; - } - } -/* --------------------------------------------------------------- - Search the set of rows of length nz. - --------------------------------------------------------------- */ -x200: -/* if (mbest .le. nz*nz1) go to 900 */ - if(KBEST<=NZ) - goto x900; - if(*IBEST>0) { - if(NROW>=MAXROW) - goto x290; - } - if(NZ>LUSOL->n) - goto x290; - LP1 = LUSOL->iploc[NZ]; - LP2 = LUSOL->m; - if(NZn) - LP2 = LUSOL->iploc[NZ+1]-1; - for(LP = LP1; LP <= LP2; LP++) { - NROW++; - I = LUSOL->ip[LP]; - LR1 = LUSOL->locr[I]; - LR2 = LR1+NZ1; - for(LR = LR1; LR <= LR2; LR++) { - J = LUSOL->indr[LR]; - LEN1 = LUSOL->lenc[J]-1; -/* merit = nz1 * len1 - if (merit .gt. mbest) continue */ - if(LEN1>KBEST) - continue; -/* aij has a promising merit. - Find where aij is in column j. */ - LC1 = LUSOL->locc[J]; - LC2 = LC1+LEN1; - AMAX = fabs(LUSOL->a[LC1]); - for(LC = LC1; LC <= LC2; LC++) { - if(LUSOL->indc[LC]==I) - break; - } -/* Apply the same stability test as above. */ - AIJ = fabs(LUSOL->a[LC]); -/* Absolute test for Complete Pivoting */ - if(TCP) { - if(AIJparmlu[LUSOL_RP_GAMMA] && - CMAX<=LUSOL->parmlu[LUSOL_RP_GAMMA]) { - if(ABEST>=AIJ) - continue; - } - else { - if(LBEST<=CMAX) - continue; - } - } -/* aij is the best pivot so far. */ - *IBEST = I; - *JBEST = J; - *MBEST = MERIT; - KBEST = LEN1; - ABEST = AIJ; - LBEST = CMAX; - if(NZ==1) - goto x900; - } -/* Finished with that row. */ - if(*IBEST>0) { - if(NROW>=MAXROW) - goto x290; - } - } -/* See if it's time to quit. */ -x290: - if(*IBEST>0) { - if(NROW>=MAXROW && NCOL>=MAXCOL) - goto x900; - } -/* Press on with next nz. */ - NZ1 = NZ; - if(*IBEST>0) - KBEST = *MBEST/NZ1; - } -x900: -; -} - -/* ================================================================== - lu1mCP uses a Markowitz criterion to select a pivot element - for the next stage of a sparse LU factorization, - subject to a Threshold Complete Pivoting stability criterion (TCP) - that bounds the elements of L and U. - ------------------------------------------------------------------ - gamma is "gamma" in the tie-breaking rule TB4 in the LUSOL paper. - ------------------------------------------------------------------ - 09 May 2002: First version of lu1mCP. - It searches columns only, using the heap that - holds the largest element in each column. - 09 May 2002: Current version of lu1mCP. - ================================================================== */ -void LU1MCP(LUSOLrec *LUSOL, REAL AIJTOL, int *IBEST, int *JBEST, int *MBEST, - int HLEN, REAL HA[], int HJ[]) -{ - int J, KHEAP, LC, LC1, LC2, LENJ, MAXCOL, NCOL, NZ1, I, LEN1, MERIT; - REAL ABEST, AIJ, AMAX, CMAX, LBEST; - -/* ------------------------------------------------------------------ - Search up to maxcol columns stored at the top of the heap. - The very top column helps initialize mbest. - ------------------------------------------------------------------ */ - ABEST = ZERO; - LBEST = ZERO; - *IBEST = 0; -/* Column at the top of the heap */ - *JBEST = HJ[1]; - LENJ = LUSOL->lenc[*JBEST]; -/* Bigger than any possible merit */ - *MBEST = LENJ*HLEN; -/* ??? Big question */ - MAXCOL = 40; -/* No. of columns searched */ - NCOL = 0; - for(KHEAP = 1; KHEAP <= HLEN; KHEAP++) { - AMAX = HA[KHEAP]; - if(AMAXlenc[J]; - NZ1 = LENJ-1; - LC1 = LUSOL->locc[J]; - LC2 = LC1+NZ1; -/* -- amax = abs( a(lc1) ) - Test all aijs in this column. - amax is the largest element (the first in the column). - cmax is the largest multiplier if aij becomes pivot. */ - for(LC = LC1; LC <= LC2; LC++) { - I = LUSOL->indc[LC]; - LEN1 = LUSOL->lenr[I]-1; - MERIT = NZ1*LEN1; - if(MERIT>*MBEST) - continue; -/* aij has a promising merit. */ - if(LC==LC1) { -/* This is the maximum element, amax. - Find the biggest element in the rest of the column - and hence get cmax. We know cmax .le. 1, but - we still want it exactly in order to break ties. - 27 Apr 2002: Settle for cmax = 1. */ - AIJ = AMAX; - CMAX = ONE; -/* cmax = ZERO; - for(l = lc1 + 1; l <= lc2; l++) - cmax = max( cmax, abs( a(l) ) ) - cmax = cmax / amax; */ - } - else { -/* aij is not the biggest element, so cmax .ge. 1. - Bail out if cmax will be too big. */ - AIJ = fabs(LUSOL->a[LC]); - if(AIJparmlu[LUSOL_RP_GAMMA] && - CMAX<=LUSOL->parmlu[LUSOL_RP_GAMMA]) { - if(ABEST>=AIJ) - continue; - } - else { - if(LBEST<=CMAX) - continue; - } - } -/* aij is the best pivot so far. */ - *IBEST = I; - *JBEST = J; - *MBEST = MERIT; - ABEST = AIJ; - LBEST = CMAX; -/* Col or row of length 1 */ - if(MERIT==0) - goto x900; - } - if(NCOL>=MAXCOL) - goto x900; - } -x900: -; -} - -/* ================================================================== - lu1mRP uses a Markowitz criterion to select a pivot element - for the next stage of a sparse LU factorization, - subject to a Threshold Rook Pivoting stability criterion (TRP) - that bounds the elements of L and U. - ------------------------------------------------------------------ - 11 Jun 2002: First version of lu1mRP derived from lu1mar. - 11 Jun 2002: Current version of lu1mRP. - ================================================================== */ -void LU1MRP(LUSOLrec *LUSOL, int MAXMN, REAL LTOL, int MAXCOL, int MAXROW, - int *IBEST, int *JBEST, int *MBEST, REAL AMAXR[]) -{ - int I, J, KBEST, LC, LC1, LC2, LEN1, LP, LP1, LP2, LQ, LQ1, - LQ2, LR, LR1, LR2, MERIT, NCOL, NROW, NZ, NZ1; - REAL ABEST, AIJ, AMAX, ATOLI, ATOLJ; - -/* ------------------------------------------------------------------ - Search cols of length nz = 1, then rows of length nz = 1, - then cols of length nz = 2, then rows of length nz = 2, etc. - ------------------------------------------------------------------ */ - ABEST = ZERO; - *IBEST = 0; - KBEST = MAXMN+1; - *MBEST = -1; - NCOL = 0; - NROW = 0; - NZ1 = 0; - for(NZ = 1; NZ <= MAXMN; NZ++) { -/* nz1 = nz - 1 - if (mbest .le. nz1**2) go to 900 */ - if(KBEST<=NZ1) - goto x900; - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x200; - } - if(NZ>LUSOL->m) - goto x200; -/* --------------------------------------------------------------- - Search the set of columns of length nz. - --------------------------------------------------------------- */ - LQ1 = LUSOL->iqloc[NZ]; - LQ2 = LUSOL->n; - if(NZm) - LQ2 = LUSOL->iqloc[NZ+1]-1; - for(LQ = LQ1; LQ <= LQ2; LQ++) { - NCOL = NCOL+1; - J = LUSOL->iq[LQ]; - LC1 = LUSOL->locc[J]; - LC2 = LC1+NZ1; - AMAX = fabs(LUSOL->a[LC1]); -/* Min size of pivots in col j */ - ATOLJ = AMAX/LTOL; -/* Test all aijs in this column. */ - for(LC = LC1; LC <= LC2; LC++) { - I = LUSOL->indc[LC]; - LEN1 = LUSOL->lenr[I]-1; -/* merit = nz1 * len1 - if (merit .gt. mbest) continue; */ - if(LEN1>KBEST) - continue; -/* aij has a promising merit. - Apply the Threshold Rook Pivoting stability test. - First we require aij to be sufficiently large - compared to other nonzeros in column j. - Then we require aij to be sufficiently large - compared to other nonzeros in row i. */ - AIJ = fabs(LUSOL->a[LC]); - if(AIJ=AIJ) - continue; - } -/* aij is the best pivot so far. */ - *IBEST = I; - *JBEST = J; - KBEST = LEN1; - *MBEST = MERIT; - ABEST = AIJ; - if(NZ==1) - goto x900; - } -/* Finished with that column. */ - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x200; - } - } -/* --------------------------------------------------------------- - Search the set of rows of length nz. - --------------------------------------------------------------- */ -x200: -/* if (mbest .le. nz*nz1) go to 900 */ - if(KBEST<=NZ) - goto x900; - if(*IBEST>0) { - if(NROW>=MAXROW) - goto x290; - } - if(NZ>LUSOL->n) - goto x290; - LP1 = LUSOL->iploc[NZ]; - LP2 = LUSOL->m; - if(NZn) - LP2 = LUSOL->iploc[NZ+1]-1; - for(LP = LP1; LP <= LP2; LP++) { - NROW = NROW+1; - I = LUSOL->ip[LP]; - LR1 = LUSOL->locr[I]; - LR2 = LR1+NZ1; -/* Min size of pivots in row i */ - ATOLI = AMAXR[I]/LTOL; - for(LR = LR1; LR <= LR2; LR++) { - J = LUSOL->indr[LR]; - LEN1 = LUSOL->lenc[J]-1; -/* merit = nz1 * len1 - if (merit .gt. mbest) continue; */ - if(LEN1>KBEST) - continue; -/* aij has a promising merit. - Find where aij is in column j. */ - LC1 = LUSOL->locc[J]; - LC2 = LC1+LEN1; - AMAX = fabs(LUSOL->a[LC1]); - for(LC = LC1; LC <= LC2; LC++) { - if(LUSOL->indc[LC]==I) - break; - } -/* Apply the Threshold Rook Pivoting stability test. - First we require aij to be sufficiently large - compared to other nonzeros in row i. - Then we require aij to be sufficiently large - compared to other nonzeros in column j. */ - AIJ = fabs(LUSOL->a[LC]); - if(AIJ=AIJ) - continue; - } -/* aij is the best pivot so far. */ - *IBEST = I; - *JBEST = J; - KBEST = LEN1; - *MBEST = MERIT; - ABEST = AIJ; - if(NZ==1) - goto x900; - } -/* Finished with that row. */ - if(*IBEST>0) { - if(NROW>=MAXROW) - goto x290; - } - } -/* See if it's time to quit. */ -x290: - if(*IBEST>0) { - if(NROW>=MAXROW && NCOL>=MAXCOL) - goto x900; - } -/* Press on with next nz. */ - NZ1 = NZ; - if(*IBEST>0) - KBEST = *MBEST/NZ1; - } -x900: -; -} - -/* ================================================================== - lu1mSP is intended for symmetric matrices that are either - definite or quasi-definite. - lu1mSP uses a Markowitz criterion to select a pivot element for - the next stage of a sparse LU factorization of a symmetric matrix, - subject to a Threshold Symmetric Pivoting stability criterion - (TSP) restricted to diagonal elements to preserve symmetry. - This bounds the elements of L and U and should have rank-revealing - properties analogous to Threshold Rook Pivoting for unsymmetric - matrices. - ------------------------------------------------------------------ - 14 Dec 2002: First version of lu1mSP derived from lu1mRP. - There is no safeguard to ensure that A is symmetric. - 14 Dec 2002: Current version of lu1mSP. - ================================================================== */ -void LU1MSP(LUSOLrec *LUSOL, int MAXMN, REAL LTOL, int MAXCOL, - int *IBEST, int *JBEST, int *MBEST) -{ - int I, J, KBEST, LC, LC1, LC2, LQ, LQ1, LQ2, MERIT, NCOL, NZ, NZ1; - REAL ABEST, AIJ, AMAX, ATOLJ; - -/* ------------------------------------------------------------------ - Search cols of length nz = 1, then cols of length nz = 2, etc. - ------------------------------------------------------------------ */ - ABEST = ZERO; - *IBEST = 0; - *MBEST = -1; - KBEST = MAXMN+1; - NCOL = 0; - NZ1 = 0; - for(NZ = 1; NZ <= MAXMN; NZ++) { -/* nz1 = nz - 1 - if (mbest .le. nz1**2) go to 900 */ - if(KBEST<=NZ1) - goto x900; - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x200; - } - if(NZ>LUSOL->m) - goto x200; -/* --------------------------------------------------------------- - Search the set of columns of length nz. - --------------------------------------------------------------- */ - LQ1 = LUSOL->iqloc[NZ]; - LQ2 = LUSOL->n; - if(NZm) - LQ2 = LUSOL->iqloc[NZ+1]-1; - for(LQ = LQ1; LQ <= LQ2; LQ++) { - NCOL++; - J = LUSOL->iq[LQ]; - LC1 = LUSOL->locc[J]; - LC2 = LC1+NZ1; - AMAX = fabs(LUSOL->a[LC1]); -/* Min size of pivots in col j */ - ATOLJ = AMAX/LTOL; -/* Test all aijs in this column. - Ignore everything except the diagonal. */ - for(LC = LC1; LC <= LC2; LC++) { - I = LUSOL->indc[LC]; -/* Skip off-diagonals. */ - if(I!=J) - continue; -/* merit = nz1 * nz1 - if (merit .gt. mbest) continue; */ - if(NZ1>KBEST) - continue; -/* aij has a promising merit. - Apply the Threshold Partial Pivoting stability test - (which is equivalent to Threshold Rook Pivoting for - symmetric matrices). - We require aij to be sufficiently large - compared to other nonzeros in column j. */ - AIJ = fabs(LUSOL->a[LC]); - if(AIJ=AIJ) - continue; - } -/* aij is the best pivot so far. */ - *IBEST = I; - *JBEST = J; - KBEST = NZ1; - *MBEST = MERIT; - ABEST = AIJ; - if(NZ==1) - goto x900; - } -/* Finished with that column. */ - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x200; - } - } -/* See if it's time to quit. */ -x200: - if(*IBEST>0) { - if(NCOL>=MAXCOL) - goto x900; - } -/* Press on with next nz. */ - NZ1 = NZ; - if(*IBEST>0) - KBEST = *MBEST/NZ1; - } -x900: -; -} - -/* ================================================================== - lu1mxc moves the largest element in each of columns iq(k1:k2) - to the top of its column. - If k1 > k2, nothing happens. - ------------------------------------------------------------------ - 06 May 2002: (and earlier) - All columns k1:k2 must have one or more elements. - 07 May 2002: Allow for empty columns. The heap routines need to - find 0.0 as the "largest element". - 29 Nov 2005: Bug fix - avoiding overwriting the next column when - the current column is empty (i.e. LENJ==0) - Yin Zhang - ================================================================== */ -void LU1MXC(LUSOLrec *LUSOL, int K1, int K2, int IX[]) -{ - int I, J, K, L, LC, LENJ; - REAL AMAX; - - for(K = K1; K <= K2; K++) { - J = IX[K]; - LC = LUSOL->locc[J]; - LENJ = LUSOL->lenc[J]; - if(LENJ==0) -/* LUSOL->a[LC] = ZERO; Removal suggested by Yin Zhang to avoid overwriting next column when current is empty */ - ; - else { - L = idamax(LUSOL->lenc[J], LUSOL->a + LC - LUSOL_ARRAYOFFSET,1) + LC - 1; - if(L>LC) { - AMAX = LUSOL->a[L]; - LUSOL->a[L] = LUSOL->a[LC]; - LUSOL->a[LC] = AMAX; - I = LUSOL->indc[L]; - LUSOL->indc[L] = LUSOL->indc[LC]; - LUSOL->indc[LC] = I; - } - } - } -} - -/* ================================================================== - lu1mxr finds the largest element in each of row ip(k1:k2) - and stores it in Amaxr(*). The nonzeros are stored column-wise - in (a,indc,lenc,locc) and their structure is row-wise - in ( indr,lenr,locr). - If k1 > k2, nothing happens. - ------------------------------------------------------------------ - 11 Jun 2002: First version of lu1mxr. - Allow for empty columns. - ================================================================== */ -void LU1MXR(LUSOLrec *LUSOL, int K1, int K2, int IX[], REAL AMAXR[]) -{ -#define FastMXR -#ifdef FastMXR - static int I, *J, *IC, K, LC, LC1, LC2, LR, LR1, LR2; - static REAL AMAX; -#else - int I, J, K, LC, LC1, LC2, LR, LR1, LR2; - REAL AMAX; -#endif - - for(K = K1; K <= K2; K++) { - AMAX = ZERO; - I = IX[K]; -/* Find largest element in row i. */ - LR1 = LUSOL->locr[I]; - LR2 = (LR1+LUSOL->lenr[I])-1; -#ifdef FastMXR - for(LR = LR1, J = LUSOL->indr + LR1; - LR <= LR2; LR++, J++) { -/* Find where aij is in column j. */ - LC1 = LUSOL->locc[*J]; - LC2 = LC1+LUSOL->lenc[*J]; - for(LC = LC1, IC = LUSOL->indc + LC1; - LC < LC2; LC++, IC++) { - if(*IC==I) - break; - } - SETMAX(AMAX,fabs(LUSOL->a[LC])); - } -#else - for(LR = LR1; LR <= LR2; LR++) { - J = LUSOL->indr[LR]; -/* Find where aij is in column j. */ - LC1 = LUSOL->locc[J]; - LC2 = (LC1+LUSOL->lenc[J])-1; - for(LC = LC1; LC <= LC2; LC++) { - if(LUSOL->indc[LC]==I) - break; - } - SETMAX(AMAX,fabs(LUSOL->a[LC])); - } -#endif - AMAXR[I] = AMAX; - } -} - - -/* ================================================================== - lu1ful computes a dense (full) LU factorization of the - mleft by nleft matrix that remains to be factored at the - beginning of the nrowu-th pass through the main loop of lu1fad. - ------------------------------------------------------------------ - 02 May 1989: First version. - 05 Feb 1994: Column interchanges added to lu1DPP. - 08 Feb 1994: ipinv reconstructed, since lu1pq3 may alter ip. - ================================================================== */ -void LU1FUL(LUSOLrec *LUSOL, int LEND, int LU1, MYBOOL TPP, - int MLEFT, int NLEFT, int NRANK, int NROWU, - int *LENL, int *LENU, int *NSING, - MYBOOL KEEPLU, REAL SMALL, REAL D[], int IPVT[]) -{ - int L, I, J, IPBASE, LDBASE, LQ, LC1, LC2, LC, LD, LKK, LKN, LU, K, L1, - L2, IBEST, JBEST, LA, LL, NROWD, NCOLD; - REAL AI, AJ; - -/* ------------------------------------------------------------------ - If lu1pq3 moved any empty rows, reset ipinv = inverse of ip. - ------------------------------------------------------------------ */ - if(NRANKm) { - for(L = 1; L <= LUSOL->m; L++) { - I = LUSOL->ip[L]; - LUSOL->ipinv[I] = L; - } - } -/* ------------------------------------------------------------------ - Copy the remaining matrix into the dense matrix D. - ------------------------------------------------------------------ */ -#ifdef LUSOLFastClear - MEMCLEAR((D+1), LEND); -#else -/* dload(LEND, ZERO, D, 1); */ - for(J = 1; J <= LEND; J++) - D[J] = ZERO; -#endif - - IPBASE = NROWU-1; - LDBASE = 1-NROWU; - for(LQ = NROWU; LQ <= LUSOL->n; LQ++) { - J = LUSOL->iq[LQ]; - LC1 = LUSOL->locc[J]; - LC2 = (LC1+LUSOL->lenc[J])-1; - for(LC = LC1; LC <= LC2; LC++) { - I = LUSOL->indc[LC]; - LD = LDBASE+LUSOL->ipinv[I]; - D[LD] = LUSOL->a[LC]; - } - LDBASE += MLEFT; - } -/* ------------------------------------------------------------------ - Call our favorite dense LU factorizer. - ------------------------------------------------------------------ */ - if(TPP) - LU1DPP(LUSOL, D,MLEFT,MLEFT,NLEFT,SMALL,NSING,IPVT,LUSOL->iq+NROWU-LUSOL_ARRAYOFFSET); - else - LU1DCP(LUSOL, D,MLEFT,MLEFT,NLEFT,SMALL,NSING,IPVT,LUSOL->iq+NROWU-LUSOL_ARRAYOFFSET); - -/* ------------------------------------------------------------------ - Move D to the beginning of A, - and pack L and U at the top of a, indc, indr. - In the process, apply the row permutation to ip. - lkk points to the diagonal of U. - ------------------------------------------------------------------ */ -#ifdef LUSOLFastCopy - MEMCOPY(LUSOL->a+1,D+1,LEND); -#else - dcopy(LEND,D,1,LUSOL->a,1); -#endif -#ifdef ClassicdiagU - LUSOL->diagU = LUSOL->a + (LUSOL->lena-LUSOL->n); -#endif - LKK = 1; - LKN = (LEND-MLEFT)+1; - LU = LU1; - for(K = 1; K <= MIN(MLEFT,NLEFT); K++) { - L1 = IPBASE+K; - L2 = IPBASE+IPVT[K]; - if(L1!=L2) { - I = LUSOL->ip[L1]; - LUSOL->ip[L1] = LUSOL->ip[L2]; - LUSOL->ip[L2] = I; - } - IBEST = LUSOL->ip[L1]; - JBEST = LUSOL->iq[L1]; - if(KEEPLU) { -/* =========================================================== - Pack the next column of L. - =========================================================== */ - LA = LKK; - LL = LU; - NROWD = 1; - for(I = K+1; I <= MLEFT; I++) { - LA++; - AI = LUSOL->a[LA]; - if(fabs(AI)>SMALL) { - NROWD = NROWD+1; - LL--; - LUSOL->a[LL] = AI; - LUSOL->indc[LL] = LUSOL->ip[IPBASE+I]; - LUSOL->indr[LL] = IBEST; - } - } -/* =========================================================== - Pack the next row of U. - We go backwards through the row of D - so the diagonal ends up at the front of the row of U. - Beware -- the diagonal may be zero. - =========================================================== */ - LA = LKN+MLEFT; - LU = LL; - NCOLD = 0; - for(J = NLEFT; J >= K; J--) { - LA = LA-MLEFT; - AJ = LUSOL->a[LA]; - if(fabs(AJ)>SMALL || J==K) { - NCOLD++; - LU--; - LUSOL->a[LU] = AJ; - LUSOL->indr[LU] = LUSOL->iq[IPBASE+J]; - } - } - LUSOL->lenr[IBEST] = -NCOLD; - LUSOL->lenc[JBEST] = -NROWD; - *LENL = ((*LENL)+NROWD)-1; - *LENU = (*LENU)+NCOLD; - LKN++; - } - else { -/* =========================================================== - Store just the diagonal of U, in natural order. - =========================================================== */ - LUSOL->diagU[JBEST] = LUSOL->a[LKK]; - } - LKK += MLEFT+1; - } -} - - -/* ================================================================== - lu1or1 organizes the elements of an m by n matrix A as - follows. On entry, the parallel arrays a, indc, indr, - contain nelem entries of the form aij, i, j, - in any order. nelem must be positive. - Entries not larger than the input parameter small are treated as - zero and removed from a, indc, indr. The remaining entries are - defined to be nonzero. numnz returns the number of such nonzeros - and Amax returns the magnitude of the largest nonzero. - The arrays lenc, lenr return the number of nonzeros in each - column and row of A. - inform = 0 on exit, except inform = 1 if any of the indices in - indc, indr imply that the element aij lies outside the m by n - dimensions of A. - ------------------------------------------------------------------ - xx Feb 1985: Original version. - 17 Oct 2000: a, indc, indr now have size lena to allow nelem = 0. - ================================================================== */ -void LU1OR1(LUSOLrec *LUSOL, REAL SMALL, - REAL *AMAX, int *NUMNZ, int *LERR, int *INFORM) -{ - int I, J, L, LDUMMY; - -#ifdef LUSOLFastClear - MEMCLEAR((LUSOL->lenr+1), LUSOL->m); - MEMCLEAR((LUSOL->lenc+1), LUSOL->n); -#else - for(I = 1; I <= LUSOL->m; I++) - LUSOL->lenr[I] = ZERO; - for(I = 1; I <= LUSOL->n; I++) - LUSOL->lenc[I] = ZERO; -#endif - - *AMAX = 0; - *NUMNZ = LUSOL->nelem; - L = LUSOL->nelem+1; - for(LDUMMY = 1; LDUMMY <= LUSOL->nelem; LDUMMY++) { - L--; - if(fabs(LUSOL->a[L])>SMALL) { - I = LUSOL->indc[L]; - J = LUSOL->indr[L]; - SETMAX(*AMAX,fabs(LUSOL->a[L])); - if(I<1 || I>LUSOL->m) - goto x910; - if(J<1 || J>LUSOL->n) - goto x910; - LUSOL->lenr[I]++; - LUSOL->lenc[J]++; - } - else { -/* Replace a negligible element by last element. Since - we are going backwards, we know the last element is ok. */ - LUSOL->a[L] = LUSOL->a[*NUMNZ]; - LUSOL->indc[L] = LUSOL->indc[*NUMNZ]; - LUSOL->indr[L] = LUSOL->indr[*NUMNZ]; - (*NUMNZ)--; - } - } - *LERR = 0; - *INFORM = LUSOL_INFORM_LUSUCCESS; - return; - -x910: - *LERR = L; - *INFORM = LUSOL_INFORM_LUSINGULAR; -} - -/* ================================================================== - lu1or2 sorts a list of matrix elements a(i,j) into column - order, given numa entries a(i,j), i, j in the parallel - arrays a, inum, jnum respectively. The matrix is assumed - to have n columns and an arbitrary number of rows. - On entry, len(*) must contain the length of each column. - On exit, a(*) and inum(*) are sorted, jnum(*) = 0, and - loc(j) points to the start of column j. - lu1or2 is derived from mc20ad, a routine in the Harwell - Subroutine Library, author J. K. Reid. - ------------------------------------------------------------------ - xx Feb 1985: Original version. - 17 Oct 2000: a, inum, jnum now have size lena to allow nelem = 0. - ================================================================== */ -void LU1OR2(LUSOLrec *LUSOL) -{ - REAL ACE, ACEP; - int L, J, I, JCE, ICE, ICEP, JCEP, JA, JB; - -/* Set loc(j) to point to the beginning of column j. */ - L = 1; - for(J = 1; J <= LUSOL->n; J++) { - LUSOL->locc[J] = L; - L += LUSOL->lenc[J]; - } -/* Sort the elements into column order. - The algorithm is an in-place sort and is of order numa. */ - for(I = 1; I <= LUSOL->nelem; I++) { -/* Establish the current entry. */ - JCE = LUSOL->indr[I]; - if(JCE==0) - continue; - ACE = LUSOL->a[I]; - ICE = LUSOL->indc[I]; - LUSOL->indr[I] = 0; -/* Chain from current entry. */ - for(J = 1; J <= LUSOL->nelem; J++) { -/* The current entry is not in the correct position. - Determine where to store it. */ - L = LUSOL->locc[JCE]; - LUSOL->locc[JCE]++; -/* Save the contents of that location. */ - ACEP = LUSOL->a[L]; - ICEP = LUSOL->indc[L]; - JCEP = LUSOL->indr[L]; -/* Store current entry. */ - LUSOL->a[L] = ACE; - LUSOL->indc[L] = ICE; - LUSOL->indr[L] = 0; -/* If next current entry needs to be processed, - copy it into current entry. */ - if(JCEP==0) - break; - ACE = ACEP; - ICE = ICEP; - JCE = JCEP; - } - } -/* Reset loc(j) to point to the start of column j. */ - JA = 1; - for(J = 1; J <= LUSOL->n; J++) { - JB = LUSOL->locc[J]; - LUSOL->locc[J] = JA; - JA = JB; - } -} - -/* ================================================================== - lu1or3 looks for duplicate elements in an m by n matrix A - defined by the column list indc, lenc, locc. - iw is used as a work vector of length m. - ------------------------------------------------------------------ - xx Feb 1985: Original version. - 17 Oct 2000: indc, indr now have size lena to allow nelem = 0. - ================================================================== */ -void LU1OR3(LUSOLrec *LUSOL, int *LERR, int *INFORM) -{ - int I, J, L1, L2, L; - -#ifdef LUSOLFastClear - MEMCLEAR((LUSOL->ip+1), LUSOL->m); -#else - for(I = 1; I <= LUSOL->m; I++) - LUSOL->ip[I] = ZERO; -#endif - - for(J = 1; J <= LUSOL->n; J++) { - if(LUSOL->lenc[J]>0) { - L1 = LUSOL->locc[J]; - L2 = (L1+LUSOL->lenc[J])-1; - for(L = L1; L <= L2; L++) { - I = LUSOL->indc[L]; - if(LUSOL->ip[I]==J) - goto x910; - LUSOL->ip[I] = J; - } - } - } - *INFORM = LUSOL_INFORM_LUSUCCESS; - return; -x910: - *LERR = L; - *INFORM = LUSOL_INFORM_LUSINGULAR; -} - -/* ================================================================== - lu1or4 constructs a row list indr, locr - from a corresponding column list indc, locc, - given the lengths of both columns and rows in lenc, lenr. - ------------------------------------------------------------------ - xx Feb 1985: Original version. - 17 Oct 2000: indc, indr now have size lena to allow nelem = 0. - ================================================================== */ -void LU1OR4(LUSOLrec *LUSOL) -{ - int L, I, L2, J, JDUMMY, L1, LR; - -/* Initialize locr(i) to point just beyond where the - last component of row i will be stored. */ - L = 1; - for(I = 1; I <= LUSOL->m; I++) { - L += LUSOL->lenr[I]; - LUSOL->locr[I] = L; - } -/* By processing the columns backwards and decreasing locr(i) - each time it is accessed, it will end up pointing to the - beginning of row i as required. */ - L2 = LUSOL->nelem; - J = LUSOL->n+1; - for(JDUMMY = 1; JDUMMY <= LUSOL->n; JDUMMY++) { - J = J-1; - if(LUSOL->lenc[J]>0) { - L1 = LUSOL->locc[J]; - for(L = L1; L <= L2; L++) { - I = LUSOL->indc[L]; - LR = LUSOL->locr[I]-1; - LUSOL->locr[I] = LR; - LUSOL->indr[LR] = J; - } - L2 = L1-1; - } - } -} - -/* ================================================================== - lu1pen deals with pending fill-in in the row file. - ------------------------------------------------------------------ - ifill(ll) says if a row involved in the new column of L - has to be updated. If positive, it is the total - length of the final updated row. - jfill(lu) says if a column involved in the new row of U - contains any pending fill-ins. If positive, it points - to the first fill-in in the column that has yet to be - added to the row file. - ------------------------------------------------------------------ - 16 Apr 1989: First version of lu1pen. - 23 Mar 2001: ilast used and updated. - ================================================================== */ -void LU1PEN(LUSOLrec *LUSOL, int NSPARE, int *ILAST, - int LPIVC1, int LPIVC2, int LPIVR1, int LPIVR2, - int *LROW, int IFILL[], int JFILL[]) -{ - int LL, LC, L, I, LR1, LR2, LR, LU, J, LC1, LC2, LAST; - - LL = 0; - for(LC = LPIVC1; LC <= LPIVC2; LC++) { - LL++; - if(IFILL[LL]==0) - continue; -/* Another row has pending fill. - First, add some spare space at the } - of the current last row. */ -#if 1 - LC1 = (*LROW)+1; - LC2 = (*LROW)+NSPARE; - *LROW = LC2; - for(L = LC1; L <= LC2; L++) { -#else - for(L = (*LROW)+1; L <= (*LROW)+NSPARE; L++) { - *LROW = L; /* ******* ERROR ???? */ -#endif - LUSOL->indr[L] = 0; - } -/* Now move row i to the end of the row file. */ - I = LUSOL->indc[LC]; - *ILAST = I; - LR1 = LUSOL->locr[I]; - LR2 = (LR1+LUSOL->lenr[I])-1; - LUSOL->locr[I] = (*LROW)+1; - for(LR = LR1; LR <= LR2; LR++) { - (*LROW)++; - LUSOL->indr[*LROW] = LUSOL->indr[LR]; - LUSOL->indr[LR] = 0; - } - (*LROW) += IFILL[LL]; - } -/* Scan all columns of D and insert the pending fill-in - into the row file. */ - LU = 1; - for(LR = LPIVR1; LR <= LPIVR2; LR++) { - LU++; - if(JFILL[LU]==0) - continue; - J = LUSOL->indr[LR]; - LC1 = (LUSOL->locc[J]+JFILL[LU])-1; - LC2 = (LUSOL->locc[J]+LUSOL->lenc[J])-1; - for(LC = LC1; LC <= LC2; LC++) { - I = LUSOL->indc[LC]-LUSOL->m; - if(I>0) { - LUSOL->indc[LC] = I; - LAST = LUSOL->locr[I]+LUSOL->lenr[I]; - LUSOL->indr[LAST] = J; - LUSOL->lenr[I]++; - } - } - } -} - - -/* ================================================================== - lu1fad is a driver for the numerical phase of lu1fac. - At each stage it computes a column of L and a row of U, - using a Markowitz criterion to select the pivot element, - subject to a stability criterion that bounds the elements of L. - ------------------------------------------------------------------ - Local variables - --------------- - lcol is the length of the column file. It points to the last - nonzero in the column list. - lrow is the analogous quantity for the row file. - lfile is the file length (lcol or lrow) after the most recent - compression of the column list or row list. - nrowd and ncold are the number of rows and columns in the - matrix defined by the pivot column and row. They are the - dimensions of the submatrix D being altered at this stage. - melim and nelim are the number of rows and columns in the - same matrix D, excluding the pivot column and row. - mleft and nleft are the number of rows and columns - still left to be factored. - nzchng is the increase in nonzeros in the matrix that remains - to be factored after the current elimination - (usually negative). - nzleft is the number of nonzeros still left to be factored. - nspare is the space we leave at the end of the last row or - column whenever a row or column is being moved to the } - of its file. nspare = 1 or 2 might help reduce the - number of file compressions when storage is tight. - The row and column ordering permutes A into the form - ------------------------ - \ | - \ U1 | - \ | - -------------------- - |\ - | \ - | \ - P A Q = | \ - | \ - | -------------- - | | | - | | | - | L1 | A2 | - | | | - | | | - -------------------- - where the block A2 is factored as A2 = L2 U2. - The phases of the factorization are as follows. - Utri is true when U1 is being determined. - Any column of length 1 is accepted immediately (if TPP). - Ltri is true when L1 is being determined. - lu1mar exits as soon as an acceptable pivot is found - in a row of length 1. - spars1 is true while the density of the (modified) A2 is less - than the parameter dens1 = parmlu(7) = 0.3 say. - lu1mar searches maxcol columns and maxrow rows, - where maxcol = luparm(3), maxrow = maxcol - 1. - lu1mxc is used to keep the biggest element at the top - of all remaining columns. - spars2 is true while the density of the modified A2 is less - than the parameter dens2 = parmlu(8) = 0.6 say. - lu1mar searches maxcol columns and no rows. - lu1mxc could fix up only the first maxcol cols (with TPP). - 22 Sep 2000: For simplicity, lu1mxc fixes all - modified cols. - dense is true once the density of A2 reaches dens2. - lu1mar searches only 1 column (the shortest). - lu1mxc could fix up only the first column (with TPP). - ------------------------------------------------------------------ - 00 Jan 1986 Version documented in LUSOL paper: - Gill, Murray, Saunders and Wright (1987), - Maintaining LU factors of a general sparse matrix, - Linear algebra and its applications 88/89, 239-270. - 02 Feb 1989 Following Suhl and Aittoniemi (1987), the largest - element in each column is now kept at the start of - the column, i.e. in position locc(j) of a and indc. - This should speed up the Markowitz searches. - To save time on highly triangular matrices, we wait - until there are no further columns of length 1 - before setting and maintaining that property. - 12 Apr 1989 ipinv and iqinv added (inverses of ip and iq) - to save searching ip and iq for rows and columns - altered in each elimination step. (Used in lu1pq2) - 19 Apr 1989 Code segmented to reduce its size. - lu1gau does most of the Gaussian elimination work. - lu1mar does just the Markowitz search. - lu1mxc moves biggest elements to top of columns. - lu1pen deals with pending fill-in in the row list. - lu1pq2 updates the row and column permutations. - 26 Apr 1989 maxtie replaced by maxcol, maxrow in the Markowitz - search. maxcol, maxrow change as density increases. - 25 Oct 1993 keepLU implemented. - 07 Feb 1994 Exit main loop early to finish off with a dense LU. - densLU tells lu1fad whether to do it. - 21 Dec 1994 Bug fixed. nrank was wrong after the call to lu1ful. - 12 Nov 1999 A parallel version of dcopy gave trouble in lu1ful - during left-shift of dense matrix D within a(*). - Fixed this unexpected problem here in lu1fad - by making sure the first and second D don't overlap. - 13 Sep 2000 TCP (Threshold Complete Pivoting) implemented. - lu2max added - (finds aijmax from biggest elems in each col). - Utri, Ltri and Spars1 phases apply. - No switch to Dense CP yet. (Only TPP switches.) - 14 Sep 2000 imax needed to remember row containing aijmax. - 22 Sep 2000 For simplicity, lu1mxc always fixes all modified cols. - (TPP spars2 used to fix just the first maxcol cols.) - 08 Nov 2000: Speed up search for aijmax. - Don't need to search all columns if the elimination - didn't alter the col containing the current aijmax. - 21 Nov 2000: lu1slk implemented for Utri phase with TCP - to guard against deceptive triangular matrices. - (Utri used to have aijtol >= 0.9999 to include - slacks, but this allows other 1s to be accepted.) - Utri now accepts slacks, but applies normal aijtol - test to other pivots. - 28 Nov 2000: TCP with empty cols must call lu1mxc and lu2max - with ( lq1, n, ... ), not just ( 1, n, ... ). - 23 Mar 2001: lu1fad bug with TCP. - A col of length 1 might not be accepted as a pivot. - Later it appears in a pivot row and temporarily - has length 0 (when pivot row is removed - but before the column is filled in). If it is the - last column in storage, the preceding col also thinks - it is "last". Trouble arises when the preceding col - needs fill-in -- it overlaps the real "last" column. - (Very rarely, same trouble might have happened if - the drop tolerance caused columns to have length 0.) - Introduced ilast to record the last row in row file, - jlast to record the last col in col file. - lu1rec returns ilast = indr(lrow + 1) - or jlast = indc(lcol + 1). - (Should be an output parameter, but didn't want to - alter lu1rec's parameter list.) - lu1rec also treats empty rows or cols safely. - (Doesn't eliminate them!) - 26 Apr 2002: Heap routines added for TCP. - lu2max no longer needed. - imax, jmax used only for printing. - 01 May 2002: lu1DCP implemented (dense complete pivoting). - Both TPP and TCP now switch to dense LU - when density exceeds dens2. - 06 May 2002: In dense mode, store diag(U) in natural order. - 09 May 2002: lu1mCP implemented (Markowitz TCP via heap). - 11 Jun 2002: lu1mRP implemented (Markowitz TRP). - 28 Jun 2002: Fixed call to lu1mxr. - 14 Dec 2002: lu1mSP implemented (Markowitz TSP). - 15 Dec 2002: Both TPP and TSP can grab cols of length 1 - during Utri. - ================================================================== */ -void LU1FAD(LUSOLrec *LUSOL, -#ifdef ClassicHamaxR - int LENA2, int LENH, REAL HA[], int HJ[], int HK[], REAL AMAXR[], -#endif - int *INFORM, int *LENL, int *LENU, int *MINLEN, - int *MERSUM, int *NUTRI, int *NLTRI, - int *NDENS1, int *NDENS2, int *NRANK, - REAL *LMAX, REAL *UMAX, REAL *DUMAX, REAL *DUMIN, REAL *AKMAX) -{ - MYBOOL UTRI, LTRI, SPARS1, SPARS2, DENSE, DENSLU, KEEPLU, TCP, TPP, TRP,TSP; - int HLEN, HOPS, H, LPIV, LPRINT, MAXCOL, MAXROW, ILAST, JLAST, LFILE, LROW, LCOL, - MINMN, MAXMN, NZLEFT, NSPARE, LU1, KK, J, LC, MLEFT, NLEFT, NROWU, - LQ1, LQ2, JBEST, LQ, I, IBEST, MBEST, LEND, NFREE, LD, NCOLD, NROWD, - MELIM, NELIM, JMAX, IMAX, LL1, LSAVE, LFREE, LIMIT, MINFRE, LPIVR, LPIVR1, LPIVR2, - L, LPIVC, LPIVC1, LPIVC2, KBEST, LU, LR, LENJ, LC1, LAST, LL, LS, - LENI, LR1, LFIRST, NFILL, NZCHNG, K, MRANK, NSING; - REAL LIJ, LTOL, SMALL, USPACE, DENS1, DENS2, AIJMAX, AIJTOL, AMAX, ABEST, DIAG, V; -#ifdef ClassicHamaxR - int LDIAGU; -#else - int LENA2 = LUSOL->lena; -#endif - -#ifdef UseTimer - int eltime, mktime, ntime; - timer ( "start", 3 ); - ntime = LUSOL->n / 4; -#endif - -#ifdef ForceInitialization - AIJMAX = 0; - AIJTOL = 0; - HLEN = 0; - JBEST = 0; - IBEST = 0; - MBEST = 0; - LEND = 0; - LD = 0; -#endif - - LPRINT = LUSOL->luparm[LUSOL_IP_PRINTLEVEL]; - MAXCOL = LUSOL->luparm[LUSOL_IP_MARKOWITZ_MAXCOL]; - LPIV = LUSOL->luparm[LUSOL_IP_PIVOTTYPE]; - KEEPLU = (MYBOOL) (LUSOL->luparm[LUSOL_IP_KEEPLU]!=FALSE); -/* Threshold Partial Pivoting (normal). */ - TPP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TPP); -/* Threshold Rook Pivoting */ - TRP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TRP); -/* Threshold Complete Pivoting. */ - TCP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TCP); -/* Threshold Symmetric Pivoting. */ - TSP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TSP); - DENSLU = FALSE; - MAXROW = MAXCOL-1; -/* Assume row m is last in the row file. */ - ILAST = LUSOL->m; -/* Assume col n is last in the col file. */ - JLAST = LUSOL->n; - LFILE = LUSOL->nelem; - LROW = LUSOL->nelem; - LCOL = LUSOL->nelem; - MINMN = MIN(LUSOL->m,LUSOL->n); - MAXMN = MAX(LUSOL->m,LUSOL->n); - NZLEFT = LUSOL->nelem; - NSPARE = 1; - - if(KEEPLU) - LU1 = LENA2+1; - else { -/* Store only the diagonals of U in the top of memory. */ -#ifdef ClassicdiagU - LDIAGU = LENA2-LUSOL->n; - LU1 = LDIAGU+1; - LUSOL->diagU = LUSOL->a+LDIAGU; -#else - LU1 = LENA2+1; -#endif - } - - LTOL = LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - USPACE = LUSOL->parmlu[LUSOL_RP_COMPSPACE_U]; - DENS1 = LUSOL->parmlu[LUSOL_RP_MARKOWITZ_CONLY]; - DENS2 = LUSOL->parmlu[LUSOL_RP_MARKOWITZ_DENSE]; - UTRI = TRUE; - LTRI = FALSE; - SPARS1 = FALSE; - SPARS2 = FALSE; - DENSE = FALSE; -/* Check parameters. */ - SETMAX(LTOL,1.0001E+0); - SETMIN(DENS1,DENS2); -/* Initialize output parameters. - lenL, lenU, minlen, mersum, nUtri, nLtri, ndens1, ndens2, nrank - are already initialized by lu1fac. */ - *LMAX = ZERO; - *UMAX = ZERO; - *DUMAX = ZERO; - *DUMIN = LUSOL_BIGNUM; - if(LUSOL->nelem==0) - *DUMIN = ZERO; - *AKMAX = ZERO; - HOPS = 0; -/* More initialization. - Don't worry yet about lu1mxc. */ - if(TPP || TSP) { - AIJMAX = ZERO; - AIJTOL = ZERO; - HLEN = 1; -/* TRP or TCP */ - } - else { -/* Move biggest element to top of each column. - Set w(*) to mark slack columns (unit vectors). */ - LU1MXC(LUSOL, 1,LUSOL->n,LUSOL->iq); - LU1SLK(LUSOL); - } - if(TRP) -/* Find biggest element in each row. */ -#ifdef ClassicHamaxR - LU1MXR(LUSOL, 1,LUSOL->m,LUSOL->ip,AMAXR); -#else - LU1MXR(LUSOL, 1,LUSOL->m,LUSOL->ip,LUSOL->amaxr); -#endif - - if(TCP) { -/* Set Ha(1:Hlen) = biggest element in each column, - Hj(1:Hlen) = corresponding column indices. */ - HLEN = 0; - for(KK = 1; KK <= LUSOL->n; KK++) { - HLEN++; - J = LUSOL->iq[KK]; - LC = LUSOL->locc[J]; -#ifdef ClassicHamaxR - HA[HLEN] = fabs(LUSOL->a[LC]); - HJ[HLEN] = J; - HK[J] = HLEN; -#else - LUSOL->Ha[HLEN] = fabs(LUSOL->a[LC]); - LUSOL->Hj[HLEN] = J; - LUSOL->Hk[J] = HLEN; -#endif - } -/* Build the heap, creating new Ha, Hj and setting Hk(1:Hlen). */ -#ifdef ClassicHamaxR - HBUILD(HA,HJ,HK,HLEN,&HOPS); -#else - HBUILD(LUSOL->Ha,LUSOL->Hj,LUSOL->Hk,HLEN,&HOPS); -#endif - } -/* ------------------------------------------------------------------ - Start of main loop. - ------------------------------------------------------------------ */ - MLEFT = LUSOL->m+1; - NLEFT = LUSOL->n+1; - for(NROWU = 1; NROWU <= MINMN; NROWU++) { -#ifdef UseTimer - mktime = (nrowu / ntime) + 4; - eltime = (nrowu / ntime) + 9; -#endif - MLEFT--; - NLEFT--; -/* Bail out if there are no nonzero rows left. */ - if(LUSOL->iploc[1]>LUSOL->m) - goto x900; -/* For TCP, the largest Aij is at the top of the heap. */ - if(TCP) { -/* - Marvelously easy */ -#ifdef ClassicHamaxR - AIJMAX = HA[1]; -#else - AIJMAX = LUSOL->Ha[1]; -#endif - SETMAX(*AKMAX,AIJMAX); - AIJTOL = AIJMAX/LTOL; - } -/* =============================================================== - Find a suitable pivot element. - =============================================================== */ - if(UTRI) { -/* ------------------------------------------------------------ - So far all columns have had length 1. - We are still looking for the (backward) triangular part of A - that forms the first rows and columns of U. - ------------------------------------------------------------ */ - LQ1 = LUSOL->iqloc[1]; - LQ2 = LUSOL->n; - if(LUSOL->m>1) - LQ2 = LUSOL->iqloc[2]-1; -/* There are more cols of length 1. */ - if(LQ1<=LQ2) { - if(TPP || TSP) { -/* Grab the first one. */ - JBEST = LUSOL->iq[LQ1]; -/* Scan all columns of length 1 ... TRP or TCP */ - } - else { - JBEST = 0; - for(LQ = LQ1; LQ <= LQ2; LQ++) { - J = LUSOL->iq[LQ]; -/* Accept a slack */ - if(LUSOL->w[J]>ZERO) { - JBEST = J; - goto x250; - } - LC = LUSOL->locc[J]; - AMAX = fabs(LUSOL->a[LC]); - if(TRP) { - I = LUSOL->indc[LC]; -#ifdef ClassicHamaxR - AIJTOL = AMAXR[I]/LTOL; -#else - AIJTOL = LUSOL->amaxr[I]/LTOL; -#endif - } - if(AMAX>=AIJTOL) { - JBEST = J; - goto x250; - } - } - } -x250: - if(JBEST>0) { - LC = LUSOL->locc[JBEST]; - IBEST = LUSOL->indc[LC]; - MBEST = 0; - goto x300; - } - } -/* This is the end of the U triangle. - We will not return to this part of the code. - TPP and TSP call lu1mxc for the first time - (to move biggest element to top of each column). */ - if(LPRINT>=LUSOL_MSG_PIVOT) - LUSOL_report(LUSOL, 0, "Utri ended. spars1 = TRUE\n"); - UTRI = FALSE; - LTRI = TRUE; - SPARS1 = TRUE; - *NUTRI = NROWU-1; - if(TPP || TSP) - LU1MXC(LUSOL, LQ1,LUSOL->n,LUSOL->iq); - } - if(SPARS1) { -/* ------------------------------------------------------------ - Perform a Markowitz search. - Search cols of length 1, then rows of length 1, - then cols of length 2, then rows of length 2, etc. - ------------------------------------------------------------ */ -#ifdef UseTimer - timer ( "start", mktime ); -#endif -/* 12 Jun 2002: Next line disables lu1mCP below - if (TPP) then */ - if(TPP || TCP) { - LU1MAR(LUSOL, MAXMN,TCP,AIJTOL,LTOL,MAXCOL,MAXROW,&IBEST,&JBEST,&MBEST); - } - else if(TRP) { -#ifdef ClassicHamaxR - LU1MRP(LUSOL, MAXMN,LTOL,MAXCOL,MAXROW,&IBEST,&JBEST,&MBEST,AMAXR); -#else - LU1MRP(LUSOL, MAXMN,LTOL,MAXCOL,MAXROW,&IBEST,&JBEST,&MBEST,LUSOL->amaxr); -#endif -/* else if (TCP) { - lu1mCP( m , n , lena , aijtol, - ibest, jbest , mbest , - a , indc , indr , - lenc , lenr , locc , - Hlen , Ha , Hj ) */ - } - else if(TSP) { - LU1MSP(LUSOL, MAXMN,LTOL,MAXCOL,&IBEST,&JBEST,&MBEST); - if(IBEST==0) - goto x990; - } -#ifdef UseTimer - timer ( "finish", mktime ); -#endif - if(LTRI) { -/* So far all rows have had length 1. - We are still looking for the (forward) triangle of A - that forms the first rows and columns of L. */ - if(MBEST>0) { - LTRI = FALSE; - *NLTRI = NROWU-1-*NUTRI; - if(LPRINT>=LUSOL_MSG_PIVOT) - LUSOL_report(LUSOL, 0, "Ltri ended.\n"); - } - } - else { -/* See if what's left is as dense as dens1. */ - if(NZLEFT>=(DENS1*MLEFT)*NLEFT) { - SPARS1 = FALSE; - SPARS2 = TRUE; - *NDENS1 = NLEFT; - MAXROW = 0; - if(LPRINT>=LUSOL_MSG_PIVOT) - LUSOL_report(LUSOL, 0, "spars1 ended. spars2 = TRUE\n"); - } - } - } - else if(SPARS2 || DENSE) { -/* ------------------------------------------------------------ - Perform a restricted Markowitz search, - looking at only the first maxcol columns. (maxrow = 0.) - ------------------------------------------------------------ */ -#ifdef UseTimer - timer ( "start", mktime ); -#endif -/* 12 Jun 2002: Next line disables lu1mCP below - if (TPP) then */ - if(TPP || TCP) { - LU1MAR(LUSOL, MAXMN,TCP,AIJTOL,LTOL,MAXCOL,MAXROW,&IBEST,&JBEST,&MBEST); - } - else if(TRP) { -#ifdef ClassicHamaxR - LU1MRP(LUSOL, MAXMN,LTOL,MAXCOL,MAXROW,&IBEST,&JBEST,&MBEST,AMAXR); -#else - LU1MRP(LUSOL, MAXMN,LTOL,MAXCOL,MAXROW,&IBEST,&JBEST,&MBEST,LUSOL->amaxr); -#endif -/* else if (TCP) { - lu1mCP( m , n , lena , aijtol, - ibest, jbest , mbest , - a , indc , indr , - lenc , lenr , locc , - Hlen , Ha , Hj ) */ - } - else if(TSP) { - LU1MSP(LUSOL, MAXMN,LTOL,MAXCOL,&IBEST,&JBEST,&MBEST); - if(IBEST==0) - goto x985; - } -#ifdef UseTimer - timer ( "finish", mktime ); -#endif -/* See if what's left is as dense as dens2. */ - if(SPARS2) { - if(NZLEFT>=(DENS2*MLEFT)*NLEFT) { - SPARS2 = FALSE; - DENSE = TRUE; - *NDENS2 = NLEFT; - MAXCOL = 1; - if(LPRINT>=LUSOL_MSG_PIVOT) - LUSOL_report(LUSOL, 0, "spars2 ended. dense = TRUE\n"); - } - } - } -/* --------------------------------------------------------------- - See if we can finish quickly. - --------------------------------------------------------------- */ - if(DENSE) { - LEND = MLEFT*NLEFT; - NFREE = LU1-1; - if(NFREE>=2*LEND) { -/* There is room to treat the remaining matrix as - a dense matrix D. - We may have to compress the column file first. - 12 Nov 1999: D used to be put at the - beginning of free storage (lD = lcol + 1). - Now put it at the end (lD = lu1 - lenD) - so the left-shift in lu1ful will not - involve overlapping storage - (fatal with parallel dcopy). - */ - DENSLU = TRUE; - *NDENS2 = NLEFT; - LD = LU1-LEND; - if(LCOL>=LD) { - LU1REC(LUSOL, LUSOL->n,TRUE,&LCOL, - LUSOL->indc,LUSOL->lenc,LUSOL->locc); - LFILE = LCOL; - JLAST = LUSOL->indc[LCOL+1]; - } - goto x900; - } - } -/* =============================================================== - The best aij has been found. - The pivot row ibest and the pivot column jbest - Define a dense matrix D of size nrowd by ncold. - =============================================================== */ -x300: - NCOLD = LUSOL->lenr[IBEST]; - NROWD = LUSOL->lenc[JBEST]; - MELIM = NROWD-1; - NELIM = NCOLD-1; - (*MERSUM) += MBEST; - (*LENL) += MELIM; - (*LENU) += NCOLD; - if(LPRINT>=LUSOL_MSG_PIVOT) { - if(NROWU==1) - LUSOL_report(LUSOL, 0, "lu1fad debug:\n"); - if(TPP || TRP || TSP) { - LUSOL_report(LUSOL, 0, "nrowu:%7d i,jbest:%7d,%7d nrowd,ncold:%6d,%6d\n", - NROWU, IBEST,JBEST, NROWD,NCOLD); -/* TCP */ - } - else { -#ifdef ClassicHamaxR - JMAX = HJ[1]; -#else - JMAX = LUSOL->Hj[1]; -#endif - IMAX = LUSOL->indc[LUSOL->locc[JMAX]]; - LUSOL_report(LUSOL, 0, "nrowu:%7d i,jbest:%7d,%7d nrowd,ncold:%6d,%6d i,jmax:%7d,%7d aijmax:%g\n", - NROWU, IBEST,JBEST, NROWD,NCOLD, IMAX,JMAX, AIJMAX); - } - } -/* =============================================================== - Allocate storage for the next column of L and next row of U. - Initially the top of a, indc, indr are used as follows: - ncold melim ncold melim - a |...........|...........|ujbest..ujn|li1......lim| - indc |...........| lenr(i) | lenc(j) | markl(i) | - indr |...........| iqloc(i) | jfill(j) | ifill(i) | - ^ ^ ^ ^ ^ - lfree lsave lu1 ll1 oldlu1 - Later the correct indices are inserted: - indc | | | |i1........im| - indr | | |jbest....jn|ibest..ibest| - =============================================================== */ - if(!KEEPLU) { -/* Always point to the top spot. - Only the current column of L and row of U will - take up space, overwriting the previous ones. */ -#ifdef ClassicHamaxR - LU1 = LDIAGU+1; -#else - LU1 = LENA2+1; -#endif - } - /* Update (left-shift) pointers to make room for the new data */ - LL1 = LU1-MELIM; - LU1 = LL1-NCOLD; - LSAVE = LU1-NROWD; - LFREE = LSAVE-NCOLD; - - /* Check if we need to allocate more memory, and allocate if necessary */ -#if 0 /* Proposal by Michael A. Saunders (logic based on Markowitz' rule) */ - L = NROWD*NCOLD; - - /* Try to avoid future expansions by anticipating further updates - KE extension */ - if(LUSOL->luparm[LUSOL_IP_UPDATELIMIT] > 0) -#if 1 - L *= (int) (log(LUSOL->luparm[LUSOL_IP_UPDATELIMIT]-LUSOL->luparm[LUSOL_IP_UPDATECOUNT]+2.0) + 1); -#else - L *= (LUSOL->luparm[LUSOL_IP_UPDATELIMIT]-LUSOL->luparm[LUSOL_IP_UPDATECOUNT]) / 2 + 1; -#endif - -#else /* Version by Kjell Eikland (from luparm[LUSOL_IP_MINIMUMLENA] and safety margin) */ - L = (KEEPLU ? MAX(LROW, LCOL) + 2*(LUSOL->m+LUSOL->n) : 0); - L *= LUSOL_MULT_nz_a; - SETMAX(L, NROWD*NCOLD); -#endif - - /* Do the memory expansion */ - if((L > LFREE-LCOL) && LUSOL_expand_a(LUSOL, &L, &LFREE)) { - LL1 += L; - LU1 += L; - LSAVE += L; -#ifdef ClassicdiagU - LUSOL->diagU += L; -#endif -#ifdef ClassicHamaxR - HA += L; - HJ += L; - HK += L; - AMAXR += L; -#endif - } - LIMIT = (int) (USPACE*LFILE)+LUSOL->m+LUSOL->n+1000; - -/* Make sure the column file has room. - Also force a compression if its length exceeds a certain limit. */ -#ifdef StaticMemAlloc - MINFRE = NCOLD+MELIM; -#else - MINFRE = NROWD*NCOLD; -#endif - NFREE = LFREE-LCOL; - if(NFREELIMIT) { - LU1REC(LUSOL, LUSOL->n,TRUE,&LCOL, - LUSOL->indc,LUSOL->lenc,LUSOL->locc); - LFILE = LCOL; - JLAST = LUSOL->indc[LCOL+1]; - NFREE = LFREE-LCOL; - if(NFREELIMIT) { - LU1REC(LUSOL, LUSOL->m,FALSE,&LROW, - LUSOL->indr,LUSOL->lenr,LUSOL->locr); - LFILE = LROW; - ILAST = LUSOL->indr[LROW+1]; - NFREE = LFREE-LROW; - if(NFREElocr[IBEST]; - LPIVR1 = LPIVR+1; - LPIVR2 = LPIVR+NELIM; - for(L = LPIVR; L <= LPIVR2; L++) { - if(LUSOL->indr[L]==JBEST) - break; - } - - LUSOL->indr[L] = LUSOL->indr[LPIVR]; - LUSOL->indr[LPIVR] = JBEST; - LPIVC = LUSOL->locc[JBEST]; - LPIVC1 = LPIVC+1; - LPIVC2 = LPIVC+MELIM; - for(L = LPIVC; L <= LPIVC2; L++) { - if(LUSOL->indc[L]==IBEST) - break; - } - LUSOL->indc[L] = LUSOL->indc[LPIVC]; - LUSOL->indc[LPIVC] = IBEST; - ABEST = LUSOL->a[L]; - LUSOL->a[L] = LUSOL->a[LPIVC]; - LUSOL->a[LPIVC] = ABEST; - if(!KEEPLU) -/* Store just the diagonal of U, in natural order. - !! a[ldiagU + nrowu] = abest ! This was in pivot order. */ - LUSOL->diagU[JBEST] = ABEST; - -/* ============================================================== - Delete pivot col from heap. - Hk tells us where it is in the heap. - ============================================================== */ - if(TCP) { -#ifdef ClassicHamaxR - KBEST = HK[JBEST]; - HDELETE(HA,HJ,HK,&HLEN,KBEST,&H); -#else - KBEST = LUSOL->Hk[JBEST]; - HDELETE(LUSOL->Ha,LUSOL->Hj,LUSOL->Hk,&HLEN,KBEST,&H); -#endif - HOPS += H; - } -/* =============================================================== - Delete the pivot row from the column file - and store it as the next row of U. - set indr(lu) = 0 to initialize jfill ptrs on columns of D, - indc(lu) = lenj to save the original column lengths. - =============================================================== */ - LUSOL->a[LU1] = ABEST; - LUSOL->indr[LU1] = JBEST; - LUSOL->indc[LU1] = NROWD; - LU = LU1; - DIAG = fabs(ABEST); - SETMAX(*UMAX,DIAG); - SETMAX(*DUMAX,DIAG); - SETMIN(*DUMIN,DIAG); - for(LR = LPIVR1; LR <= LPIVR2; LR++) { - LU++; - J = LUSOL->indr[LR]; - LENJ = LUSOL->lenc[J]; - LUSOL->lenc[J] = LENJ-1; - LC1 = LUSOL->locc[J]; - LAST = LC1+LUSOL->lenc[J]; - for(L = LC1; L <= LAST; L++) { - if(LUSOL->indc[L]==IBEST) - break; - } - LUSOL->a[LU] = LUSOL->a[L]; - LUSOL->indr[LU] = 0; - LUSOL->indc[LU] = LENJ; - SETMAX(*UMAX,fabs(LUSOL->a[LU])); - LUSOL->a[L] = LUSOL->a[LAST]; - LUSOL->indc[L] = LUSOL->indc[LAST]; -/* Free entry */ - LUSOL->indc[LAST] = 0; -/* ??? if (j .eq. jlast) lcol = lcol - 1 */ - } -/* =============================================================== - Delete the pivot column from the row file - and store the nonzeros of the next column of L. - Set indc(ll) = 0 to initialize markl(*) markers, - indr(ll) = 0 to initialize ifill(*) row fill-in cntrs, - indc(ls) = leni to save the original row lengths, - indr(ls) = iqloc(i) to save parts of iqloc(*), - iqloc(i) = lsave - ls to point to the nonzeros of L - = -1, -2, -3, ... in mark(*). - =============================================================== */ - LUSOL->indc[LSAVE] = NCOLD; - if(MELIM==0) - goto x700; - LL = LL1-1; - LS = LSAVE; - ABEST = ONE/ABEST; - for(LC = LPIVC1; LC <= LPIVC2; LC++) { - LL++; - LS++; - I = LUSOL->indc[LC]; - LENI = LUSOL->lenr[I]; - LUSOL->lenr[I] = LENI-1; - LR1 = LUSOL->locr[I]; - LAST = LR1+LUSOL->lenr[I]; - for(L = LR1; L <= LAST; L++) { - if(LUSOL->indr[L]==JBEST) - break; - } - LUSOL->indr[L] = LUSOL->indr[LAST]; -/* Free entry */ - LUSOL->indr[LAST] = 0; - LUSOL->a[LL] = -LUSOL->a[LC]*ABEST; - LIJ = fabs(LUSOL->a[LL]); - SETMAX(*LMAX,LIJ); - LUSOL->indc[LL] = 0; - LUSOL->indr[LL] = 0; - LUSOL->indc[LS] = LENI; - LUSOL->indr[LS] = LUSOL->iqloc[I]; - LUSOL->iqloc[I] = LSAVE-LS; - } -/* =============================================================== - Do the Gaussian elimination. - This involves adding a multiple of the pivot column - to all other columns in the pivot row. - Sometimes more than one call to lu1gau is needed to allow - compression of the column file. - lfirst says which column the elimination should start with. - minfre is a bound on the storage needed for any one column. - lu points to off-diagonals of u. - nfill keeps track of pending fill-in in the row file. - =============================================================== */ - if(NELIM==0) - goto x700; - LFIRST = LPIVR1; - MINFRE = MLEFT+NSPARE; - LU = 1; - NFILL = 0; - -x400: -#ifdef UseTimer - timer ( "start", eltime ); -#endif - LU1GAU(LUSOL, MELIM,NSPARE,SMALL,LPIVC1,LPIVC2,&LFIRST,LPIVR2, - LFREE,MINFRE,ILAST,&JLAST,&LROW,&LCOL,&LU,&NFILL, - LUSOL->iqloc, LUSOL->a+LL1-LUSOL_ARRAYOFFSET, - LUSOL->indc+LL1-LUSOL_ARRAYOFFSET, LUSOL->a+LU1-LUSOL_ARRAYOFFSET, - LUSOL->indr+LL1-LUSOL_ARRAYOFFSET, LUSOL->indr+LU1-LUSOL_ARRAYOFFSET); -#ifdef UseTimer - timer ( "finish", eltime ); -#endif - if(LFIRST>0) { -/* The elimination was interrupted. - Compress the column file and try again. - lfirst, lu and nfill have appropriate new values. */ - LU1REC(LUSOL, LUSOL->n,TRUE,&LCOL, - LUSOL->indc,LUSOL->lenc,LUSOL->locc); - LFILE = LCOL; - JLAST = LUSOL->indc[LCOL+1]; - LPIVC = LUSOL->locc[JBEST]; - LPIVC1 = LPIVC+1; - LPIVC2 = LPIVC+MELIM; - NFREE = LFREE-LCOL; - if(NFREE0) { -/* Compress the row file if necessary. - lu1gau has set nfill to be the number of pending fill-ins - plus the current length of any rows that need to be moved. */ - MINFRE = NFILL; - NFREE = LFREE-LROW; - if(NFREEm,FALSE,&LROW, - LUSOL->indr,LUSOL->lenr,LUSOL->locr); - LFILE = LROW; - ILAST = LUSOL->indr[LROW+1]; - LPIVR = LUSOL->locr[IBEST]; - LPIVR1 = LPIVR+1; - LPIVR2 = LPIVR+NELIM; - NFREE = LFREE-LROW; - if(NFREEindr+LL1-LUSOL_ARRAYOFFSET,LUSOL->indr+LU1-LUSOL_ARRAYOFFSET); - } -/* =============================================================== - Restore the saved values of iqloc. - Insert the correct indices for the col of L and the row of U. - =============================================================== */ -x700: - LUSOL->lenr[IBEST] = 0; - LUSOL->lenc[JBEST] = 0; - LL = LL1-1; - LS = LSAVE; - for(LC = LPIVC1; LC <= LPIVC2; LC++) { - LL++; - LS++; - I = LUSOL->indc[LC]; - LUSOL->iqloc[I] = LUSOL->indr[LS]; - LUSOL->indc[LL] = I; - LUSOL->indr[LL] = IBEST; - } - LU = LU1-1; - for(LR = LPIVR; LR <= LPIVR2; LR++) { - LU++; - LUSOL->indr[LU] = LUSOL->indr[LR]; - } -/* =============================================================== - Free the space occupied by the pivot row - and update the column permutation. - Then free the space occupied by the pivot column - and update the row permutation. - nzchng is found in both calls to lu1pq2, but we use it only - after the second. - =============================================================== */ - LU1PQ2(LUSOL, NCOLD, &NZCHNG, - LUSOL->indr+LPIVR-LUSOL_ARRAYOFFSET, - LUSOL->indc+LU1-LUSOL_ARRAYOFFSET, LUSOL->lenc, - LUSOL->iqloc, LUSOL->iq, LUSOL->iqinv); - LU1PQ2(LUSOL, NROWD, &NZCHNG, - LUSOL->indc+LPIVC-LUSOL_ARRAYOFFSET, - LUSOL->indc+LSAVE-LUSOL_ARRAYOFFSET, LUSOL->lenr, - LUSOL->iploc, LUSOL->ip, LUSOL->ipinv); - NZLEFT += NZCHNG; - -/* =============================================================== - lu1mxr resets Amaxr(i) in each modified row i. - lu1mxc moves the largest aij to the top of each modified col j. - 28 Jun 2002: Note that cols of L have an implicit diag of 1.0, - so lu1mxr is called with ll1, not ll1+1, whereas - lu1mxc is called with lu1+1. - =============================================================== */ - if(UTRI && TPP) { -/* Relax -- we're not keeping big elements at the top yet. */ - } - else { - if(TRP && MELIM>0) -#ifdef ClassicHamaxR - LU1MXR(LUSOL, LL1,LL,LUSOL->indc,AMAXR); -#else - LU1MXR(LUSOL, LL1,LL,LUSOL->indc,LUSOL->amaxr); -#endif - - if(NELIM>0) { - LU1MXC(LUSOL, LU1+1,LU,LUSOL->indr); -/* Update modified columns in heap */ - if(TCP) { - for(KK = LU1+1; KK <= LU; KK++) { - J = LUSOL->indr[KK]; -#ifdef ClassicHamaxR - K = HK[J]; -#else - K = LUSOL->Hk[J]; -#endif -/* Biggest aij in column j */ - V = fabs(LUSOL->a[LUSOL->locc[J]]); -#ifdef ClassicHamaxR - HCHANGE(HA,HJ,HK,HLEN,K,V,J,&H); -#else - HCHANGE(LUSOL->Ha,LUSOL->Hj,LUSOL->Hk,HLEN,K,V,J,&H); -#endif - HOPS += H; - } - } - } - } -/* =============================================================== - Negate lengths of pivot row and column so they will be - eliminated during compressions. - =============================================================== */ - LUSOL->lenr[IBEST] = -NCOLD; - LUSOL->lenc[JBEST] = -NROWD; - -/* Test for fatal bug: row or column lists overwriting L and U. */ - if(LROW>LSAVE || LCOL>LSAVE) - goto x980; - -/* Reset the file lengths if pivot row or col was at the end. */ - if(IBEST==ILAST) - LROW = LUSOL->locr[IBEST]; - - if(JBEST==JLAST) - LCOL = LUSOL->locc[JBEST]; - - } -/* ------------------------------------------------------------------ - End of main loop. - ------------------------------------------------------------------ - ------------------------------------------------------------------ - Normal exit. - Move empty rows and cols to the end of ip, iq. - Then finish with a dense LU if necessary. - ------------------------------------------------------------------ */ -x900: - *INFORM = LUSOL_INFORM_LUSUCCESS; - LU1PQ3(LUSOL, LUSOL->m,LUSOL->lenr,LUSOL->ip,LUSOL->ipinv,&MRANK); - LU1PQ3(LUSOL, LUSOL->n,LUSOL->lenc,LUSOL->iq,LUSOL->iqinv,NRANK); - SETMIN(*NRANK, MRANK); - if(DENSLU) { -#ifdef UseTimer - timer ( "start", 17 ); -#endif - LU1FUL(LUSOL, LEND,LU1,TPP,MLEFT,NLEFT,*NRANK,NROWU,LENL,LENU, - &NSING,KEEPLU,SMALL,LUSOL->a+LD-LUSOL_ARRAYOFFSET,LUSOL->locr); -/* *** 21 Dec 1994: Bug in next line. - *** nrank = nrank - nsing */ - *NRANK = MINMN-NSING; -#ifdef UseTimer - timer ( "finish", 17 ); -#endif - } - *MINLEN = (*LENL)+(*LENU)+2*(LUSOL->m+LUSOL->n); - goto x990; -/* Not enough space free after a compress. - Set minlen to an estimate of the necessary value of lena. */ -x970: - *INFORM = LUSOL_INFORM_ANEEDMEM; - *MINLEN = LENA2+LFILE+2*(LUSOL->m+LUSOL->n); - goto x990; -/* Fatal error. This will never happen! - (Famous last words.) */ -x980: - *INFORM = LUSOL_INFORM_FATALERR; - goto x990; -/* Fatal error with TSP. Diagonal pivot not found. */ -x985: - *INFORM = LUSOL_INFORM_NOPIVOT; -/* Exit. */ -x990: -#ifdef UseTimer - timer ( "finish", 3 ); -#endif -; -} - - -/* ================================================================== - lu1fac computes a factorization A = L*U, where A is a sparse - matrix with m rows and n columns, P*L*P' is lower triangular - and P*U*Q is upper triangular for certain permutations P, Q - (which are returned in the arrays ip, iq). - Stability is ensured by limiting the size of the elements of L. - The nonzeros of A are input via the parallel arrays a, indc, indr, - which should contain nelem entries of the form aij, i, j - in any order. There should be no duplicate pairs i, j. - - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - + Beware !!! The row indices i must be in indc, + - + and the column indices j must be in indr. + - + (Not the other way round!) + - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - - It does not matter if some of the entries in a(*) are zero. - Entries satisfying abs( a(i) ) .le. parmlu(3) are ignored. - Other parameters in luparm and parmlu are described below. - The matrix A may be singular. On exit, nsing = luparm(11) gives - the number of apparent singularities. This is the number of - "small" diagonals of the permuted factor U, as judged by - the input tolerances Utol1 = parmlu(4) and Utol2 = parmlu(5). - The diagonal element diagj associated with column j of A is - "small" if - abs( diagj ) .le. Utol1 - or - abs( diagj ) .le. Utol2 * max( uj ), - where max( uj ) is the maximum element in the j-th column of U. - The position of such elements is returned in w(*). In general, - w(j) = + max( uj ), but if column j is a singularity, - w(j) = - max( uj ). Thus, w(j) .le. 0 if column j appears to be - dependent on the other columns of A. - NOTE: lu1fac (like certain other sparse LU packages) does not - treat dense columns efficiently. This means it will be slow - on "arrow matrices" of the form - A = (x a) - ( x b) - ( x c) - ( x d) - (x x x x e) - if the numerical values in the dense column allow it to be - chosen LATE in the pivot order. - With TPP (Threshold Partial Pivoting), the dense column is - likely to be chosen late. - With TCP (Threshold Complete Pivoting), if any of a,b,c,d - is significantly larger than other elements of A, it will - be chosen as the first pivot and the dense column will be - eliminated, giving reasonably sparse factors. - However, if element e is so big that TCP chooses it, the factors - will become dense. (It's hard to win on these examples!) - ------------------------------------------------------------------ - - Notes on the array names - ------------------------ - During the LU factorization, the sparsity pattern of the matrix - being factored is stored twice: in a column list and a row list. - The column list is ( a, indc, locc, lenc ) - where - a(*) holds the nonzeros, - indc(*) holds the indices for the column list, - locc(j) points to the start of column j in a(*) and indc(*), - lenc(j) is the number of nonzeros in column j. - The row list is ( indr, locr, lenr ) - where - indr(*) holds the indices for the row list, - locr(i) points to the start of row i in indr(*), - lenr(i) is the number of nonzeros in row i. - At all stages of the LU factorization, ip contains a complete - row permutation. At the start of stage k, ip(1), ..., ip(k-1) - are the first k-1 rows of the final row permutation P. - The remaining rows are stored in an ordered list - ( ip, iploc, ipinv ) - where - iploc(nz) points to the start in ip(*) of the set of rows - that currently contain nz nonzeros, - ipinv(i) points to the position of row i in ip(*). - For example, - iploc(1) = k (and this is where rows of length 1 {), - iploc(2) = k+p if there are p rows of length 1 - (and this is where rows of length 2 {). - Similarly for iq, iqloc, iqinv. - --------------------------------------------------------------------- - INPUT PARAMETERS - m (not altered) is the number of rows in A. - n (not altered) is the number of columns in A. - nelem (not altered) is the number of matrix entries given in - the arrays a, indc, indr. - lena (not altered) is the dimension of a, indc, indr. - This should be significantly larger than nelem. - Typically one should have - lena > max( 2*nelem, 10*m, 10*n, 10000 ) - but some applications may need more. - On machines with virtual memory it is safe to have - lena "far bigger than necessary", since not all of the - arrays will be used. - a (overwritten) contains entries Aij in a(1:nelem). - indc (overwritten) contains the indices i in indc(1:nelem). - indr (overwritten) contains the indices j in indr(1:nelem). - luparm input parameters: Typical value - luparm( 1) = nout File number for printed messages. 6 - luparm( 2) = lprint Print level. 0 - < 0 suppresses output. - = 0 gives error messages. - >= 10 gives statistics about the LU factors. - >= 50 gives debug output from lu1fac - (the pivot row and column and the - no. of rows and columns involved at - each elimination step). - luparm( 3) = maxcol lu1fac: maximum number of columns 5 - searched allowed in a Markowitz-type - search for the next pivot element. - For some of the factorization, the - number of rows searched is - maxrow = maxcol - 1. - luparm( 6) = 0 => TPP: Threshold Partial Pivoting. 0 - = 1 => TRP: Threshold Rook Pivoting. - = 2 => TCP: Threshold Complete Pivoting. - = 3 => TSP: Threshold Symmetric Pivoting. - = 4 => TDP: Threshold Diagonal Pivoting. - (TDP not yet implemented). - TRP and TCP are more expensive than TPP but - more stable and better at revealing rank. - Take care with setting parmlu(1), especially - with TCP. - NOTE: TSP and TDP are for symmetric matrices - that are either definite or quasi-definite. - TSP is effectively TRP for symmetric matrices. - TDP is effectively TCP for symmetric matrices. - luparm( 8) = keepLU lu1fac: keepLU = 1 means the numerical 1 - factors will be computed if possible. - keepLU = 0 means L and U will be discarded - but other information such as the row and - column permutations will be returned. - The latter option requires less storage. - parmlu input parameters: Typical value - parmlu( 1) = Ltol1 Max Lij allowed during Factor. - TPP 10.0 or 100.0 - TRP 4.0 or 10.0 - TCP 5.0 or 10.0 - TSP 4.0 or 10.0 - With TRP and TCP (Rook and Complete Pivoting), - values less than 25.0 may be expensive - on badly scaled data. However, - values less than 10.0 may be needed - to obtain a reliable rank-revealing - factorization. - parmlu( 2) = Ltol2 Max Lij allowed during Updates. 10.0 - during updates. - parmlu( 3) = small Absolute tolerance for eps**0.8 = 3.0d-13 - treating reals as zero. - parmlu( 4) = Utol1 Absolute tol for flagging eps**0.67= 3.7d-11 - small diagonals of U. - parmlu( 5) = Utol2 Relative tol for flagging eps**0.67= 3.7d-11 - small diagonals of U. - (eps = machine precision) - parmlu( 6) = Uspace Factor limiting waste space in U. 3.0 - In lu1fac, the row or column lists - are compressed if their length - exceeds Uspace times the length of - either file after the last compression. - parmlu( 7) = dens1 The density at which the Markowitz 0.3 - pivot strategy should search maxcol - columns and no rows. - (Use 0.3 unless you are experimenting - with the pivot strategy.) - parmlu( 8) = dens2 the density at which the Markowitz 0.5 - strategy should search only 1 column, - or (if storage is available) - the density at which all remaining - rows and columns will be processed - by a dense LU code. - For example, if dens2 = 0.1 and lena is - large enough, a dense LU will be used - once more than 10 per cent of the - remaining matrix is nonzero. - - OUTPUT PARAMETERS - a, indc, indr contain the nonzero entries in the LU factors of A. - If keepLU = 1, they are in a form suitable for use - by other parts of the LUSOL package, such as lu6sol. - U is stored by rows at the start of a, indr. - L is stored by cols at the end of a, indc. - If keepLU = 0, only the diagonals of U are stored, at the - end of a. - ip, iq are the row and column permutations defining the - pivot order. For example, row ip(1) and column iq(1) - defines the first diagonal of U. - lenc(1:numl0) contains the number of entries in nontrivial - columns of L (in pivot order). - lenr(1:m) contains the number of entries in each row of U - (in original order). - locc(1:n) = 0 (ready for the LU update routines). - locr(1:m) points to the beginning of the rows of U in a, indr. - iploc, iqloc, ipinv, iqinv are undefined. - w indicates singularity as described above. - inform = 0 if the LU factors were obtained successfully. - = 1 if U appears to be singular, as judged by lu6chk. - = 3 if some index pair indc(l), indr(l) lies outside - the matrix dimensions 1:m , 1:n. - = 4 if some index pair indc(l), indr(l) duplicates - another such pair. - = 7 if the arrays a, indc, indr were not large enough. - Their length "lena" should be increase to at least - the value "minlen" given in luparm(13). - = 8 if there was some other fatal error. (Shouldn't happen!) - = 9 if no diagonal pivot could be found with TSP or TDP. - The matrix must not be sufficiently definite - or quasi-definite. - luparm output parameters: - luparm(10) = inform Return code from last call to any LU routine. - luparm(11) = nsing No. of singularities marked in the - output array w(*). - luparm(12) = jsing Column index of last singularity. - luparm(13) = minlen Minimum recommended value for lena. - luparm(14) = maxlen ? - luparm(15) = nupdat No. of updates performed by the lu8 routines. - luparm(16) = nrank No. of nonempty rows of U. - luparm(17) = ndens1 No. of columns remaining when the density of - the matrix being factorized reached dens1. - luparm(18) = ndens2 No. of columns remaining when the density of - the matrix being factorized reached dens2. - luparm(19) = jumin The column index associated with DUmin. - luparm(20) = numL0 No. of columns in initial L. - luparm(21) = lenL0 Size of initial L (no. of nonzeros). - luparm(22) = lenU0 Size of initial U. - luparm(23) = lenL Size of current L. - luparm(24) = lenU Size of current U. - luparm(25) = lrow Length of row file. - luparm(26) = ncp No. of compressions of LU data structures. - luparm(27) = mersum lu1fac: sum of Markowitz merit counts. - luparm(28) = nUtri lu1fac: triangular rows in U. - luparm(29) = nLtri lu1fac: triangular rows in L. - luparm(30) = - parmlu output parameters: - parmlu(10) = Amax Maximum element in A. - parmlu(11) = Lmax Maximum multiplier in current L. - parmlu(12) = Umax Maximum element in current U. - parmlu(13) = DUmax Maximum diagonal in U. - parmlu(14) = DUmin Minimum diagonal in U. - parmlu(15) = Akmax Maximum element generated at any stage - during TCP factorization. - parmlu(16) = growth TPP: Umax/Amax TRP, TCP, TSP: Akmax/Amax - parmlu(17) = - parmlu(18) = - parmlu(19) = - parmlu(20) = resid lu6sol: residual after solve with U or U'. - ... - parmlu(30) = - ------------------------------------------------------------------ - 00 Jun 1983 Original version. - 00 Jul 1987 nrank saved in luparm(16). - 12 Apr 1989 ipinv, iqinv added as workspace. - 26 Apr 1989 maxtie replaced by maxcol in Markowitz search. - 16 Mar 1992 jumin saved in luparm(19). - 10 Jun 1992 lu1fad has to move empty rows and cols to the bottom - (via lu1pq3) before doing the dense LU. - 12 Jun 1992 Deleted dense LU (lu1ful, lu1vlu). - 25 Oct 1993 keepLU implemented. - 07 Feb 1994 Added new dense LU (lu1ful, lu1den). - 21 Dec 1994 Bugs fixed in lu1fad (nrank) and lu1ful (ipvt). - 08 Aug 1995 Use ip instead of w as parameter to lu1or3 (for F90). - 13 Sep 2000 TPP and TCP options implemented. - 17 Oct 2000 Fixed troubles due to A = empty matrix (Todd Munson). - 01 Dec 2000 Save Lmax, Umax, etc. after both lu1fad and lu6chk. - lu1fad sets them when keepLU = false. - lu6chk sets them otherwise, and includes items - from the dense LU. - 11 Mar 2001 lu6chk now looks at diag(U) when keepLU = false. - 26 Apr 2002 New TCP implementation using heap routines to - store largest element in each column. - New workspace arrays Ha, Hj, Hk required. - For compatibility, borrow space from a, indc, indr - rather than adding new input parameters. - 01 May 2002 lu1den changed to lu1DPP (dense partial pivoting). - lu1DCP implemented (dense complete pivoting). - Both TPP and TCP now switch to dense mode and end. - ================================================================== */ -void LU1FAC(LUSOLrec *LUSOL, int *INFORM) -{ - MYBOOL KEEPLU, TCP, TPP, TRP, TSP; - int LPIV, NELEM0, LPRINT, MINLEN, NUML0, LENL, LENU, LROW, MERSUM, - NUTRI, NLTRI, NDENS1, NDENS2, NRANK, NSING, JSING, JUMIN, NUMNZ, LERR, - LU, LL, LM, LTOPL, K, I, LENUK, J, LENLK, IDUMMY, LLSAVE, NMOVE, L2, L, NCP, NBUMP; -#ifdef ClassicHamaxR - int LENH, LENA2, LOCH, LMAXR; -#endif - - REAL LMAX, LTOL, SMALL, AMAX, UMAX, DUMAX, DUMIN, AKMAX, DM, DN, DELEM, DENSTY, - AGRWTH, UGRWTH, GROWTH, CONDU, DINCR, AVGMER; - -/* Free row-based version of L0 (regenerated by LUSOL_btran). */ - if(LUSOL->L0 != NULL) - LUSOL_matfree(&(LUSOL->L0)); - -/* Grab relevant input parameters. */ - NELEM0 = LUSOL->nelem; - LPRINT = LUSOL->luparm[LUSOL_IP_PRINTLEVEL]; - LPIV = LUSOL->luparm[LUSOL_IP_PIVOTTYPE]; - KEEPLU = (MYBOOL) (LUSOL->luparm[LUSOL_IP_KEEPLU]!=FALSE); -/* Limit on size of Lij */ - LTOL = LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij]; -/* Drop tolerance */ - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - TPP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TPP); - TRP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TRP); - TCP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TCP); - TSP = (MYBOOL) (LPIV==LUSOL_PIVMOD_TSP); -/* Initialize output parameters. */ - *INFORM = LUSOL_INFORM_LUSUCCESS; - LERR = 0; - MINLEN = LUSOL->nelem + 2*(LUSOL->m+LUSOL->n); - NUML0 = 0; - LENL = 0; - LENU = 0; - LROW = 0; - MERSUM = 0; - NUTRI = LUSOL->m; - NLTRI = 0; - NDENS1 = 0; - NDENS2 = 0; - NRANK = 0; - NSING = 0; - JSING = 0; - JUMIN = 0; - AMAX = ZERO; - LMAX = ZERO; - UMAX = ZERO; - DUMAX = ZERO; - DUMIN = ZERO; - AKMAX = ZERO; - -/* Float version of dimensions. */ - DM = LUSOL->m; - DN = LUSOL->n; - DELEM = LUSOL->nelem; - -/* Initialize workspace parameters. */ - LUSOL->luparm[LUSOL_IP_COMPRESSIONS_LU] = 0; - if(LUSOL->lena < MINLEN) { - if(!LUSOL_realloc_a(LUSOL, MINLEN)) - goto x970; - } - -/* ------------------------------------------------------------------ - Organize the aij's in a, indc, indr. - lu1or1 deletes small entries, tests for illegal i,j's, - and counts the nonzeros in each row and column. - lu1or2 reorders the elements of A by columns. - lu1or3 uses the column list to test for duplicate entries - (same indices i,j). - lu1or4 constructs a row list from the column list. - ------------------------------------------------------------------ */ - LU1OR1(LUSOL, SMALL,&AMAX,&NUMNZ,&LERR,INFORM); - if(LPRINT>=LUSOL_MSG_STATISTICS) { - DENSTY = (100*DELEM)/(DM*DN); - LUSOL_report(LUSOL, 0, "m:%6d %c n:%6d nzcount:%9d Amax:%g Density:%g\n", - LUSOL->m, relationChar(LUSOL->m, LUSOL->n), LUSOL->n, - LUSOL->nelem, AMAX, DENSTY); - } - if(*INFORM!=LUSOL_INFORM_LUSUCCESS) - goto x930; - LUSOL->nelem = NUMNZ; - LU1OR2(LUSOL); - LU1OR3(LUSOL, &LERR,INFORM); - if(*INFORM!=LUSOL_INFORM_LUSUCCESS) - goto x940; - LU1OR4(LUSOL); -/* ------------------------------------------------------------------ - Set up lists of rows and columns with equal numbers of nonzeros, - using indc(*) as workspace. - ------------------------------------------------------------------ */ - LU1PQ1(LUSOL, LUSOL->m,LUSOL->n,LUSOL->lenr, - LUSOL->ip,LUSOL->iploc,LUSOL->ipinv, - LUSOL->indc+LUSOL->nelem); /* LUSOL_ARRAYOFFSET implied */ - LU1PQ1(LUSOL, LUSOL->n,LUSOL->m,LUSOL->lenc, - LUSOL->iq,LUSOL->iqloc,LUSOL->iqinv, - LUSOL->indc+LUSOL->nelem); /* LUSOL_ARRAYOFFSET implied */ -/* ------------------------------------------------------------------ - For TCP, Ha, Hj, Hk are allocated separately, similarly amaxr - for TRP. Then compute the factorization A = L*U. - ------------------------------------------------------------------ */ -#ifdef ClassicHamaxR - if(TPP || TSP) { - LENH = 1; - LENA2 = LUSOL->lena; - LOCH = LUSOL->lena; - LMAXR = 1; - } - else if(TRP) { - LENH = 1; /* Dummy */ - LENA2 = LUSOL->lena-LUSOL->m; /* Reduced length of a */ - LOCH = LUSOL->lena; /* Dummy */ - LMAXR = LENA2+1; /* Start of Amaxr in a */ - } - else if(TCP) { - LENH = LUSOL->n; /* Length of heap */ - LENA2 = LUSOL->lena-LENH; /* Reduced length of a, indc, indr */ - LOCH = LENA2+1; /* Start of Ha, Hj, Hk in a, indc, indr */ - LMAXR = 1; /* Dummy */ - } - LU1FAD(LUSOL, - LENA2,LENH, - LUSOL->a+LOCH-LUSOL_ARRAYOFFSET, - LUSOL->indc+LOCH-LUSOL_ARRAYOFFSET, - LUSOL->indr+LOCH-LUSOL_ARRAYOFFSET, - LUSOL->a+LMAXR-LUSOL_ARRAYOFFSET, - INFORM,&LENL,&LENU, - &MINLEN,&MERSUM,&NUTRI,&NLTRI,&NDENS1,&NDENS2, - &NRANK,&LMAX,&UMAX,&DUMAX,&DUMIN,&AKMAX); -#else - LU1FAD(LUSOL, - INFORM,&LENL,&LENU, - &MINLEN,&MERSUM,&NUTRI,&NLTRI,&NDENS1,&NDENS2, - &NRANK,&LMAX,&UMAX,&DUMAX,&DUMIN,&AKMAX); -#endif - LUSOL->luparm[LUSOL_IP_RANK_U] = NRANK; - LUSOL->luparm[LUSOL_IP_NONZEROS_L] = LENL; - if(*INFORM==LUSOL_INFORM_ANEEDMEM) - goto x970; - if(*INFORM==LUSOL_INFORM_NOPIVOT) - goto x985; - if(*INFORM>LUSOL_INFORM_LUSUCCESS) - goto x980; - if(KEEPLU) { -/* --------------------------------------------------------------- - The LU factors are at the top of a, indc, indr, - with the columns of L and the rows of U in the order - ( free ) ... ( u3 ) ( l3 ) ( u2 ) ( l2 ) ( u1 ) ( l1 ). - Starting with ( l1 ) and ( u1 ), move the rows of U to the - left and the columns of L to the right, giving - ( u1 ) ( u2 ) ( u3 ) ... ( free ) ... ( l3 ) ( l2 ) ( l1 ). - Also, set numl0 = the number of nonempty columns of U. - --------------------------------------------------------------- */ - LU = 0; - LL = LUSOL->lena+1; -#ifdef ClassicHamaxR - LM = LENA2+1; -#else - LM = LL; -#endif - LTOPL = LL-LENL-LENU; - LROW = LENU; - for(K = 1; K <= NRANK; K++) { - I = LUSOL->ip[K]; - LENUK = -LUSOL->lenr[I]; - LUSOL->lenr[I] = LENUK; - J = LUSOL->iq[K]; - LENLK = -LUSOL->lenc[J]-1; - if(LENLK>0) { - NUML0++; - LUSOL->iqloc[NUML0] = LENLK; - } - if(LU+LENUKa[LL] = LUSOL->a[LM]; - LUSOL->indc[LL] = LUSOL->indc[LM]; - LUSOL->indr[LL] = LUSOL->indr[LM]; - } - } - else { -/* ========================================================= - There is no room for ( uk ) yet. We have to - right-shift the whole of the remaining LU file. - Note that ( lk ) ends up in the correct place. - ========================================================= */ - LLSAVE = LL-LENLK; - NMOVE = LM-LTOPL; - for(IDUMMY = 1; IDUMMY <= NMOVE; IDUMMY++) { - LL--; - LM--; - LUSOL->a[LL] = LUSOL->a[LM]; - LUSOL->indc[LL] = LUSOL->indc[LM]; - LUSOL->indr[LL] = LUSOL->indr[LM]; - } - LTOPL = LL; - LL = LLSAVE; - LM = LL; - } -/* ====================================================== - Left-shift ( uk ). - ====================================================== */ - LUSOL->locr[I] = LU+1; - L2 = LM-1; - LM = LM-LENUK; - for(L = LM; L <= L2; L++) { - LU = LU+1; - LUSOL->a[LU] = LUSOL->a[L]; - LUSOL->indr[LU] = LUSOL->indr[L]; - } - } -/* --------------------------------------------------------------- - Save the lengths of the nonempty columns of L, - and initialize locc(j) for the LU update routines. - --------------------------------------------------------------- */ - for(K = 1; K <= NUML0; K++) { - LUSOL->lenc[K] = LUSOL->iqloc[K]; - } - for(J = 1; J <= LUSOL->n; J++) { - LUSOL->locc[J] = 0; - } -/* --------------------------------------------------------------- - Test for singularity. - lu6chk sets nsing, jsing, jumin, Lmax, Umax, DUmax, DUmin - (including entries from the dense LU). - inform = 1 if there are singularities (nsing gt 0). - --------------------------------------------------------------- */ - LU6CHK(LUSOL, 1,LUSOL->lena,INFORM); - NSING = LUSOL->luparm[LUSOL_IP_SINGULARITIES]; - JSING = LUSOL->luparm[LUSOL_IP_SINGULARINDEX]; - JUMIN = LUSOL->luparm[LUSOL_IP_COLINDEX_DUMIN]; - LMAX = LUSOL->parmlu[LUSOL_RP_MAXMULT_L]; - UMAX = LUSOL->parmlu[LUSOL_RP_MAXELEM_U]; - DUMAX = LUSOL->parmlu[LUSOL_RP_MAXELEM_DIAGU]; - DUMIN = LUSOL->parmlu[LUSOL_RP_MINELEM_DIAGU]; - } - else { -/* --------------------------------------------------------------- - keepLU = 0. L and U were not kept, just the diagonals of U. - lu1fac will probably be called again soon with keepLU = .true. - 11 Mar 2001: lu6chk revised. We can call it with keepLU = 0, - but we want to keep Lmax, Umax from lu1fad. - 05 May 2002: Allow for TCP with new lu1DCP. Diag(U) starts - below lena2, not lena. Need lena2 in next line. - --------------------------------------------------------------- */ -#ifdef ClassicHamaxR - LU6CHK(LUSOL, 1,LENA2,INFORM); -#else - LU6CHK(LUSOL, 1,LUSOL->lena,INFORM); -#endif - NSING = LUSOL->luparm[LUSOL_IP_SINGULARITIES]; - JSING = LUSOL->luparm[LUSOL_IP_SINGULARINDEX]; - JUMIN = LUSOL->luparm[LUSOL_IP_COLINDEX_DUMIN]; - DUMAX = LUSOL->parmlu[LUSOL_RP_MAXELEM_DIAGU]; - DUMIN = LUSOL->parmlu[LUSOL_RP_MINELEM_DIAGU]; - } - goto x990; -/* ------------ - Error exits. - ------------ */ -x930: - *INFORM = LUSOL_INFORM_ADIMERR; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu1fac error...\nentry a[%d] has an illegal row (%d) or column (%d) index\n", - LERR,LUSOL->indc[LERR],LUSOL->indr[LERR]); - goto x990; -x940: - *INFORM = LUSOL_INFORM_ADUPLICATE; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu1fac error...\nentry a[%d] is a duplicate with indeces indc=%d, indr=%d\n", - LERR,LUSOL->indc[LERR],LUSOL->indr[LERR]); - goto x990; -x970: - *INFORM = LUSOL_INFORM_ANEEDMEM; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu1fac error...\ninsufficient storage; increase lena from %d to at least %d\n", - LUSOL->lena, MINLEN); - goto x990; -x980: - *INFORM = LUSOL_INFORM_FATALERR; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu1fac error...\nfatal bug (sorry --- this should never happen)\n"); - goto x990; -x985: - *INFORM = LUSOL_INFORM_NOPIVOT; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu1fac error...\nTSP used but diagonal pivot could not be found\n"); - -/* Finalize and store output parameters. */ -x990: - LUSOL->nelem = NELEM0; - LUSOL->luparm[LUSOL_IP_SINGULARITIES] = NSING; - LUSOL->luparm[LUSOL_IP_SINGULARINDEX] = JSING; - LUSOL->luparm[LUSOL_IP_MINIMUMLENA] = MINLEN; - LUSOL->luparm[LUSOL_IP_UPDATECOUNT] = 0; - LUSOL->luparm[LUSOL_IP_RANK_U] = NRANK; - LUSOL->luparm[LUSOL_IP_COLCOUNT_DENSE1] = NDENS1; - LUSOL->luparm[LUSOL_IP_COLCOUNT_DENSE2] = NDENS2; - LUSOL->luparm[LUSOL_IP_COLINDEX_DUMIN] = JUMIN; - LUSOL->luparm[LUSOL_IP_COLCOUNT_L0] = NUML0; - LUSOL->luparm[LUSOL_IP_ROWCOUNT_L0] = 0; - LUSOL->luparm[LUSOL_IP_NONZEROS_L0] = LENL; - LUSOL->luparm[LUSOL_IP_NONZEROS_U0] = LENU; - LUSOL->luparm[LUSOL_IP_NONZEROS_L] = LENL; - LUSOL->luparm[LUSOL_IP_NONZEROS_U] = LENU; - LUSOL->luparm[LUSOL_IP_NONZEROS_ROW] = LROW; - LUSOL->luparm[LUSOL_IP_MARKOWITZ_MERIT] = MERSUM; - LUSOL->luparm[LUSOL_IP_TRIANGROWS_U] = NUTRI; - LUSOL->luparm[LUSOL_IP_TRIANGROWS_L] = NLTRI; - LUSOL->parmlu[LUSOL_RP_MAXELEM_A] = AMAX; - LUSOL->parmlu[LUSOL_RP_MAXMULT_L] = LMAX; - LUSOL->parmlu[LUSOL_RP_MAXELEM_U] = UMAX; - LUSOL->parmlu[LUSOL_RP_MAXELEM_DIAGU] = DUMAX; - LUSOL->parmlu[LUSOL_RP_MINELEM_DIAGU] = DUMIN; - LUSOL->parmlu[LUSOL_RP_MAXELEM_TCP] = AKMAX; - AGRWTH = AKMAX/(AMAX+LUSOL_SMALLNUM); - UGRWTH = UMAX/(AMAX+LUSOL_SMALLNUM); - if(TPP) - GROWTH = UGRWTH; -/* TRP or TCP or TSP */ - else - GROWTH = AGRWTH; - LUSOL->parmlu[LUSOL_RP_GROWTHRATE] = GROWTH; - - LUSOL->luparm[LUSOL_IP_FTRANCOUNT] = 0; - LUSOL->luparm[LUSOL_IP_BTRANCOUNT] = 0; - -/* ------------------------------------------------------------------ - Set overall status variable. - ------------------------------------------------------------------ */ - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; - if(*INFORM == LUSOL_INFORM_NOMEMLEFT) - LUSOL_report(LUSOL, 0, "lu1fac error...\ninsufficient memory available\n"); - -/* ------------------------------------------------------------------ - Print statistics for the LU factors. - ------------------------------------------------------------------ */ - NCP = LUSOL->luparm[LUSOL_IP_COMPRESSIONS_LU]; - CONDU = DUMAX/MAX(DUMIN,LUSOL_SMALLNUM); - DINCR = (LENL+LENU)-LUSOL->nelem; - DINCR = (DINCR*100)/MAX(DELEM,ONE); - AVGMER = MERSUM; - AVGMER = AVGMER/DM; - NBUMP = LUSOL->m-NUTRI-NLTRI; - if(LPRINT>=LUSOL_MSG_STATISTICS) { - if(TPP) { - LUSOL_report(LUSOL, 0, "Merit %g %d %d %d %g %d %d %g %g %d %d %d\n", - AVGMER,LENL,LENL+LENU,NCP,DINCR,NUTRI,LENU, - LTOL,UMAX,UGRWTH,NLTRI,NDENS1,LMAX); - } - else { - LUSOL_report(LUSOL, 0, "Merit %s %g %d %d %d %g %d %d %g %g %d %d %d %g %g\n", - LUSOL_pivotLabel(LUSOL), - AVGMER,LENL,LENL+LENU,NCP,DINCR,NUTRI,LENU, - LTOL,UMAX,UGRWTH,NLTRI,NDENS1,LMAX,AKMAX,AGRWTH); - } - LUSOL_report(LUSOL, 0, "bump%9d dense2%7d DUmax%g DUmin%g conDU%g\n", - NBUMP,NDENS2,DUMAX,DUMIN,CONDU); - } -} - - diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol2.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol2.c deleted file mode 100644 index 9fdcbb46..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol2.c +++ /dev/null @@ -1,204 +0,0 @@ - -/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - File lusol2 LUSOL heap management routines - Hbuild Hchange Hdelete Hdown Hinsert Hup - Heap-management routines for LUSOL's lu1fac. - May be useful for other applications. - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - For LUSOL, the heap structure involves three arrays of length N. - N is the current number of entries in the heap. - Ha(1:N) contains the values that the heap is partially sorting. - For LUSOL they are double precision values -- the largest - element in each remaining column of the updated matrix. - The biggest entry is in Ha(1), the top of the heap. - Hj(1:N) contains column numbers j. - Ha(k) is the biggest entry in column j = Hj(k). - Hk(1:N) contains indices within the heap. It is the - inverse of Hj(1:N), so k = Hk(j) <=> j = Hj(k). - Column j is entry k in the heap. - hops is the number of heap operations, - i.e., the number of times an entry is moved - (the number of "hops" up or down the heap). - Together, Hj and Hk let us find values inside the heap - whenever we want to change one of the values in Ha. - For other applications, Ha may need to be some other data type, - like the keys that sort routines operate on. - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - 11 Feb 2002: MATLAB version derived from "Algorithms" by - R. Sedgewick - 03 Mar 2002: F77 version derived from MATLAB version. - 07 May 2002: Safeguard input parameters k, N, Nk. - We don't want them to be output! - 07 May 2002: Current version of lusol2.f. - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */ - -/* ================================================================== - Hdown updates heap by moving down tree from node k. - ------------------------------------------------------------------ - 01 May 2002: Need Nk for length of Hk. - 05 May 2002: Change input paramter k to kk to stop k being output. - 05 May 2002: Current version of Hdown. - ================================================================== */ -void HDOWN(REAL HA[], int HJ[], int HK[], int N, int K, int *HOPS) -{ - int J, JJ, JV, N2; - REAL V; - - *HOPS = 0; - V = HA[K]; - JV = HJ[K]; - N2 = N/2; -/* while 1 - break */ -x100: - if(K>N2) - goto x200; - (*HOPS)++; - J = K+K; - if(J=HA[J]) - goto x200; - HA[K] = HA[J]; - JJ = HJ[J]; - HJ[K] = JJ; - HK[JJ] = K; - K = J; - goto x100; -/* end while */ -x200: - HA[K] = V; - HJ[K] = JV; - HK[JV] = K; -} - -/* ================================================================== - Hup updates heap by moving up tree from node k. - ------------------------------------------------------------------ - 01 May 2002: Need Nk for length of Hk. - 05 May 2002: Change input paramter k to kk to stop k being output. - 05 May 2002: Current version of Hup. - ================================================================== */ -void HUP(REAL HA[], int HJ[], int HK[], int K, int *HOPS) -{ - int J, JV, K2; - REAL V; - - *HOPS = 0; - V = HA[K]; - JV = HJ[K]; -/* while 1 - break */ -x100: - if(K<2) - goto x200; - K2 = K/2; -/* break */ - if(V n - nrank may occur. - If keepLU = 0, - Lmax and Umax are already set by lu1fac. - The diagonals of U are in the top of A. - Only Utol1 is used in the singularity test to set w(*). - inform = 0 if A appears to have full column rank (nsing = 0). - inform = 1 otherwise (nsing .gt. 0). - ------------------------------------------------------------------ - 00 Jul 1987: Early version. - 09 May 1988: f77 version. - 11 Mar 2001: Allow for keepLU = 0. - 17 Nov 2001: Briefer output for singular factors. - 05 May 2002: Comma needed in format 1100 (via Kenneth Holmstrom). - 06 May 2002: With keepLU = 0, diags of U are in natural order. - They were not being extracted correctly. - 23 Apr 2004: TRP can judge singularity better by comparing - all diagonals to DUmax. - 27 Jun 2004: (PEG) Allow write only if nout .gt. 0. - ================================================================== */ -#ifdef UseOld_LU6CHK_20040510 -void LU6CHK(LUSOLrec *LUSOL, int MODE, int LENA2, int *INFORM) -{ - MYBOOL KEEPLU; - int I, J, JUMIN, K, L, L1, L2, LENL, LPRINT, NDEFIC, NRANK; - REAL AIJ, DIAG, DUMAX, DUMIN, LMAX, UMAX, UTOL1, UTOL2; - - LPRINT = LUSOL->luparm[LUSOL_IP_PRINTLEVEL]; - KEEPLU = (MYBOOL) (LUSOL->luparm[LUSOL_IP_KEEPLU]!=0); - NRANK = LUSOL->luparm[LUSOL_IP_RANK_U]; - LENL = LUSOL->luparm[LUSOL_IP_NONZEROS_L]; - UTOL1 = LUSOL->parmlu[LUSOL_RP_SMALLDIAG_U]; - UTOL2 = LUSOL->parmlu[LUSOL_RP_EPSDIAG_U]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - LMAX = ZERO; - UMAX = ZERO; - LUSOL->luparm[LUSOL_IP_SINGULARITIES] = 0; - LUSOL->luparm[LUSOL_IP_SINGULARINDEX] = 0; - JUMIN = 0; - DUMAX = ZERO; - DUMIN = LUSOL_BIGNUM; - -#ifdef LUSOLFastClear - MEMCLEAR(LUSOL->w+1, LUSOL->n); -#else - for(I = 1; I <= LUSOL->n; I++) - LUSOL->w[I] = ZERO; -#endif - - if(KEEPLU) { -/* -------------------------------------------------------------- - Find Lmax. - -------------------------------------------------------------- */ - for(L = (LENA2+1)-LENL; L <= LENA2; L++) { - SETMAX(LMAX,fabs(LUSOL->a[L])); - } -/* -------------------------------------------------------------- - Find Umax and set w(j) = maximum element in j-th column of U. - -------------------------------------------------------------- */ - for(K = 1; K <= NRANK; K++) { - I = LUSOL->ip[K]; - L1 = LUSOL->locr[I]; - L2 = (L1+LUSOL->lenr[I])-1; - for(L = L1; L <= L2; L++) { - J = LUSOL->indr[L]; - AIJ = fabs(LUSOL->a[L]); - SETMAX(LUSOL->w[J],AIJ); - SETMAX(UMAX,AIJ); - } - } -/* -------------------------------------------------------------- - Negate w(j) if the corresponding diagonal of U is - too small in absolute terms or relative to the other elements - in the same column of U. - Also find DUmax and DUmin, the extreme diagonals of U. - -------------------------------------------------------------- */ - for(K = 1; K <= LUSOL->n; K++) { - J = LUSOL->iq[K]; - if(K>NRANK) - DIAG = ZERO; - else { - I = LUSOL->ip[K]; - L1 = LUSOL->locr[I]; - DIAG = fabs(LUSOL->a[L1]); - SETMAX(DUMAX,DIAG); - if(DUMIN>DIAG) { - DUMIN = DIAG; - JUMIN = J; - } - } - if((DIAG<=UTOL1) || (DIAG<=UTOL2*LUSOL->w[J])) { - LUSOL_addSingularity(LUSOL, J, INFORM); - LUSOL->w[J] = -LUSOL->w[J]; - } - } - LUSOL->parmlu[LUSOL_RP_MAXMULT_L] = LMAX; - LUSOL->parmlu[LUSOL_RP_MAXELEM_U] = UMAX; - } - else { -/* -------------------------------------------------------------- - keepLU = 0. - Only diag(U) is stored. Set w(*) accordingly. - -------------------------------------------------------------- */ - for(K = 1; K <= LUSOL->n; K++) { - J = LUSOL->iq[K]; - if(K>NRANK) - DIAG = ZERO; - else { -/* ! diag = abs( diagU(k) ) ! 06 May 2002: Diags are in natural order */ - DIAG = fabs(LUSOL->diagU[J]); - LUSOL->w[J] = DIAG; - SETMAX(DUMAX,DIAG); - if(DUMIN>DIAG) { - DUMIN = DIAG; - JUMIN = J; - } - } - if(DIAG<=UTOL1) { - LUSOL_addSingularity(LUSOL, J, INFORM); - LUSOL->w[J] = -LUSOL->w[J]; - } - } - } -/* ----------------------------------------------------------------- - Set output parameters. - ----------------------------------------------------------------- */ - if(JUMIN==0) - DUMIN = ZERO; - LUSOL->luparm[LUSOL_IP_COLINDEX_DUMIN] = JUMIN; - LUSOL->parmlu[LUSOL_RP_MAXELEM_DIAGU] = DUMAX; - LUSOL->parmlu[LUSOL_RP_MINELEM_DIAGU] = DUMIN; -/* The matrix has been judged singular. */ - if(LUSOL->luparm[LUSOL_IP_SINGULARITIES]>0) { - *INFORM = LUSOL_INFORM_LUSINGULAR; - NDEFIC = LUSOL->n-NRANK; - if(LPRINT>=LUSOL_MSG_SINGULARITY) { - LUSOL_report(LUSOL, 0, "Singular(m%cn) rank:%9d n-rank:%8d nsing:%9d\n", - relationChar(LUSOL->m, LUSOL->n),NRANK,NDEFIC, - LUSOL->luparm[LUSOL_IP_SINGULARITIES]); - } - } -/* Exit. */ - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; -} -#else -void LU6CHK(LUSOLrec *LUSOL, int MODE, int LENA2, int *INFORM) -{ - MYBOOL KEEPLU, TRP; - int I, J, JUMIN, K, L, L1, L2, LENL, LDIAGU, LPRINT, NDEFIC, NRANK; - REAL AIJ, DIAG, DUMAX, DUMIN, LMAX, UMAX, UTOL1, UTOL2; - - LPRINT = LUSOL->luparm[LUSOL_IP_PRINTLEVEL]; - KEEPLU = (MYBOOL) (LUSOL->luparm[LUSOL_IP_KEEPLU] != 0); - TRP = (MYBOOL) (LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TRP); - NRANK = LUSOL->luparm[LUSOL_IP_RANK_U]; - LENL = LUSOL->luparm[LUSOL_IP_NONZEROS_L]; - UTOL1 = LUSOL->parmlu[LUSOL_RP_SMALLDIAG_U]; - UTOL2 = LUSOL->parmlu[LUSOL_RP_EPSDIAG_U]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - LMAX = ZERO; - UMAX = ZERO; - LUSOL->luparm[LUSOL_IP_SINGULARITIES] = 0; - LUSOL->luparm[LUSOL_IP_SINGULARINDEX] = 0; - JUMIN = 0; - DUMAX = ZERO; - DUMIN = LUSOL_BIGNUM; - -#ifdef LUSOLFastClear - MEMCLEAR(LUSOL->w+1, LUSOL->n); -#else - for(I = 1; I <= LUSOL->n; I++) - LUSOL->w[I] = ZERO; -#endif - - if(KEEPLU) { -/* -------------------------------------------------------------- - Find Lmax. - -------------------------------------------------------------- */ - for(L = (LENA2+1)-LENL; L <= LENA2; L++) { - SETMAX(LMAX,fabs(LUSOL->a[L])); - } -/* -------------------------------------------------------------- - Find Umax and set w(j) = maximum element in j-th column of U. - -------------------------------------------------------------- */ - for(K = 1; K <= NRANK; K++) { - I = LUSOL->ip[K]; - L1 = LUSOL->locr[I]; - L2 = (L1+LUSOL->lenr[I])-1; - for(L = L1; L <= L2; L++) { - J = LUSOL->indr[L]; - AIJ = fabs(LUSOL->a[L]); - SETMAX(LUSOL->w[J],AIJ); - SETMAX(UMAX,AIJ); - } - } - LUSOL->parmlu[LUSOL_RP_MAXMULT_L] = LMAX; - LUSOL->parmlu[LUSOL_RP_MAXELEM_U] = UMAX; -/* -------------------------------------------------------------- - Find DUmax and DUmin, the extreme diagonals of U. - -------------------------------------------------------------- */ - for(K = 1; K <= NRANK; K++) { - J = LUSOL->iq[K]; - I = LUSOL->ip[K]; - L1 = LUSOL->locr[I]; - DIAG = fabs(LUSOL->a[L1]); - SETMAX( DUMAX, DIAG ); - if(DUMIN > DIAG) { - DUMIN = DIAG; - JUMIN = J; - } - } - } - else { -/* -------------------------------------------------------------- - keepLU = 0. - Only diag(U) is stored. Set w(*) accordingly. - Find DUmax and DUmin, the extreme diagonals of U. - -------------------------------------------------------------- */ - LDIAGU = LENA2 - LUSOL->n; - for(K = 1; K <= NRANK; K++) { - J = LUSOL->iq[K]; - DIAG = fabs( LUSOL->a[LDIAGU + J] ); /* are in natural order */ - LUSOL->w[J] = DIAG; - SETMAX( DUMAX, DIAG ); - if(DUMIN > DIAG) { - DUMIN = DIAG; - JUMIN = J; - } - } - } -/* -------------------------------------------------------------- - Negate w(j) if the corresponding diagonal of U is - too small in absolute terms or relative to the other elements - in the same column of U. - - 23 Apr 2004: TRP ensures that diags are NOT small relative to - other elements in their own column. - Much better, we can compare all diags to DUmax. - -------------------------------------------------------------- */ - if((MODE == 1) && TRP) { - SETMAX( UTOL1, UTOL2*DUMAX ); - } - - if(KEEPLU) { - for(K = 1; K <= LUSOL->n; K++) { - J = LUSOL->iq[K]; - if(K>NRANK) - DIAG = ZERO; - else { - I = LUSOL->ip[K]; - L1 = LUSOL->locr[I]; - DIAG = fabs(LUSOL->a[L1]); - } - if((DIAG<=UTOL1) || (DIAG<=UTOL2*LUSOL->w[J])) { - LUSOL_addSingularity(LUSOL, J, INFORM); - LUSOL->w[J] = -LUSOL->w[J]; - } - } - } - else { /* keepLU = FALSE */ - for(K = 1; K <= LUSOL->n; K++) { - J = LUSOL->iq[K]; - DIAG = LUSOL->w[J]; - if(DIAG<=UTOL1) { - LUSOL_addSingularity(LUSOL, J, INFORM); - LUSOL->w[J] = -LUSOL->w[J]; - } - } - } -/* ----------------------------------------------------------------- - Set output parameters. - ----------------------------------------------------------------- */ - if(JUMIN==0) - DUMIN = ZERO; - LUSOL->luparm[LUSOL_IP_COLINDEX_DUMIN] = JUMIN; - LUSOL->parmlu[LUSOL_RP_MAXELEM_DIAGU] = DUMAX; - LUSOL->parmlu[LUSOL_RP_MINELEM_DIAGU] = DUMIN; -/* The matrix has been judged singular. */ - if(LUSOL->luparm[LUSOL_IP_SINGULARITIES]>0) { - *INFORM = LUSOL_INFORM_LUSINGULAR; - NDEFIC = LUSOL->n-NRANK; - if((LUSOL->outstream!=NULL) && (LPRINT>=LUSOL_MSG_SINGULARITY)) { - LUSOL_report(LUSOL, 0, "Singular(m%cn) rank:%9d n-rank:%8d nsing:%9d\n", - relationChar(LUSOL->m, LUSOL->n),NRANK,NDEFIC, - LUSOL->luparm[LUSOL_IP_SINGULARITIES]); - } - } -/* Exit. */ - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; -} -#endif - - -/* ------------------------------------------------------------------ - Include routines for row-based L0. - 20 Apr 2005 Current version - KE. - ------------------------------------------------------------------ */ -#include "lusol6l0.c" - - -/* ------------------------------------------------------------------ - lu6L solves L v = v(input). - ------------------------------------------------------------------ - 15 Dec 2002: First version derived from lu6sol. - 15 Dec 2002: Current version. - ------------------------------------------------------------------ */ -void LU6L(LUSOLrec *LUSOL, int *INFORM, REAL V[], int NZidx[]) -{ - int JPIV, K, L, L1, LEN, LENL, LENL0, NUML, NUML0; - REAL SMALL; - register REAL VPIV; -#ifdef LUSOLFastSolve - REAL *aptr; - int *iptr, *jptr; -#else - int I, J; -#endif - - NUML0 = LUSOL->luparm[LUSOL_IP_COLCOUNT_L0]; - LENL0 = LUSOL->luparm[LUSOL_IP_NONZEROS_L0]; - LENL = LUSOL->luparm[LUSOL_IP_NONZEROS_L]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - L1 = LUSOL->lena+1; - for(K = 1; K <= NUML0; K++) { - LEN = LUSOL->lenc[K]; - L = L1; - L1 -= LEN; - JPIV = LUSOL->indr[L1]; - VPIV = V[JPIV]; - if(fabs(VPIV)>SMALL) { -/* ***** This loop could be coded specially. */ -#ifdef LUSOLFastSolve - L--; - for(aptr = LUSOL->a+L, iptr = LUSOL->indc+L; - LEN > 0; LEN--, aptr--, iptr--) - V[*iptr] += (*aptr) * VPIV; -#else - for(; LEN > 0; LEN--) { - L--; - I = LUSOL->indc[L]; - V[I] += LUSOL->a[L]*VPIV; - } -#endif - } -#ifdef SetSmallToZero - else - V[JPIV] = 0; -#endif - } - L = (LUSOL->lena-LENL0)+1; - NUML = LENL-LENL0; -/* ***** This loop could be coded specially. */ -#ifdef LUSOLFastSolve - L--; - for(aptr = LUSOL->a+L, jptr = LUSOL->indr+L, iptr = LUSOL->indc+L; - NUML > 0; NUML--, aptr--, jptr--, iptr--) { - if(fabs(V[*jptr])>SMALL) - V[*iptr] += (*aptr) * V[*jptr]; -#ifdef SetSmallToZero - else - V[*jptr] = 0; -#endif - } -#else - for(; NUML > 0; NUML--) { - L--; - J = LUSOL->indr[L]; - if(fabs(V[J])>SMALL) { - I = LUSOL->indc[L]; - V[I] += LUSOL->a[L]*V[J]; - } -#ifdef SetSmallToZero - else - V[J] = 0; -#endif - } -#endif -/* Exit. */ - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; -} - -/* ================================================================== - lu6LD assumes lu1fac has computed factors A = LU of a - symmetric definite or quasi-definite matrix A, - using Threshold Symmetric Pivoting (TSP), luparm(6) = 3, - or Threshold Diagonal Pivoting (TDP), luparm(6) = 4. - It also assumes that no updates have been performed. - In such cases, U = D L', where D = diag(U). - lu6LDL returns v as follows: - - mode - 1 v solves L D v = v(input). - 2 v solves L|D|v = v(input). - ------------------------------------------------------------------ - 15 Dec 2002: First version of lu6LD. - 15 Dec 2002: Current version. - ================================================================== */ -void LU6LD(LUSOLrec *LUSOL, int *INFORM, int MODE, REAL V[], int NZidx[]) -{ - int IPIV, K, L, L1, LEN, NUML0; - REAL DIAG, SMALL; - register REAL VPIV; -#ifdef LUSOLFastSolve - REAL *aptr; - int *jptr; -#else - int J; -#endif - -/* Solve L D v(new) = v or L|D|v(new) = v, depending on mode. - The code for L is the same as in lu6L, - but when a nonzero entry of v arises, we divide by - the corresponding entry of D or |D|. */ - NUML0 = LUSOL->luparm[LUSOL_IP_COLCOUNT_L0]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - L1 = LUSOL->lena+1; - for(K = 1; K <= NUML0; K++) { - LEN = LUSOL->lenc[K]; - L = L1; - L1 -= LEN; - IPIV = LUSOL->indr[L1]; - VPIV = V[IPIV]; - if(fabs(VPIV)>SMALL) { -/* ***** This loop could be coded specially. */ -#ifdef LUSOLFastSolve - L--; - for(aptr = LUSOL->a+L, jptr = LUSOL->indc+L; - LEN > 0; LEN--, aptr--, jptr--) - V[*jptr] += (*aptr)*VPIV; -#else - for(; LEN > 0; LEN--) { - L--; - J = LUSOL->indc[L]; - V[J] += LUSOL->a[L]*VPIV; - } -#endif -/* Find diag = U(ipiv,ipiv) and divide by diag or |diag|. */ - L = LUSOL->locr[IPIV]; - DIAG = LUSOL->a[L]; - if(MODE==2) - DIAG = fabs(DIAG); - V[IPIV] = VPIV/DIAG; - } -#ifdef SetSmallToZero - else - V[IPIV] = 0; -#endif - } -} - - -/* ================================================================== - lu6Lt solves L'v = v(input). - ------------------------------------------------------------------ - 15 Dec 2002: First version derived from lu6sol. - 15 Dec 2002: Current version. - ================================================================== */ -void LU6LT(LUSOLrec *LUSOL, int *INFORM, REAL V[], int NZidx[]) -{ -#ifdef DoTraceL0 - REAL TEMP; -#endif - int K, L, L1, L2, LEN, LENL, LENL0, NUML0; - REAL SMALL; - register REALXP SUM; - register REAL HOLD; -#if (defined LUSOLFastSolve) && !(defined DoTraceL0) - REAL *aptr; - int *iptr, *jptr; -#else - int I, J; -#endif - - NUML0 = LUSOL->luparm[LUSOL_IP_COLCOUNT_L0]; - LENL0 = LUSOL->luparm[LUSOL_IP_NONZEROS_L0]; - LENL = LUSOL->luparm[LUSOL_IP_NONZEROS_L]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - L1 = (LUSOL->lena-LENL)+1; - L2 = LUSOL->lena-LENL0; - -/* ***** This loop could be coded specially. */ -#if (defined LUSOLFastSolve) && !(defined DoTraceL0) - for(L = L1, aptr = LUSOL->a+L1, iptr = LUSOL->indr+L1, jptr = LUSOL->indc+L1; - L <= L2; L++, aptr++, iptr++, jptr++) { - HOLD = V[*jptr]; - if(fabs(HOLD)>SMALL) - V[*iptr] += (*aptr)*HOLD; -#ifdef SetSmallToZero - else - V[*jptr] = 0; -#endif - } -#else - for(L = L1; L <= L2; L++) { - J = LUSOL->indc[L]; - HOLD = V[J]; - if(fabs(HOLD)>SMALL) { - I = LUSOL->indr[L]; - V[I] += LUSOL->a[L]*HOLD; - } -#ifdef SetSmallToZero - else - V[J] = 0; -#endif - } -#endif - - /* Do row-based L0 version, if available */ - if((LUSOL->L0 != NULL) || - ((LUSOL->luparm[LUSOL_IP_BTRANCOUNT] == 0) && LU1L0(LUSOL, &(LUSOL->L0), INFORM))) { - LU6L0T_v(LUSOL, LUSOL->L0, V, NZidx, INFORM); - } - - /* Alternatively, do the standard column-based L0 version */ - else { - /* Perform loop over columns */ - for(K = NUML0; K >= 1; K--) { - SUM = ZERO; - LEN = LUSOL->lenc[K]; - L1 = L2+1; - L2 += LEN; -/* ***** This loop could be coded specially. */ -#if (defined LUSOLFastSolve) && !(defined DoTraceL0) - for(L = L1, aptr = LUSOL->a+L1, jptr = LUSOL->indc+L1; - L <= L2; L++, aptr++, jptr++) - SUM += (*aptr) * V[*jptr]; -#else - for(L = L1; L <= L2; L++) { - J = LUSOL->indc[L]; -#ifndef DoTraceL0 - SUM += LUSOL->a[L]*V[J]; -#else - TEMP = V[LUSOL->indr[L1]] + SUM; - SUM += LUSOL->a[L]*V[J]; - printf("V[%3d] = V[%3d] + L[%d,%d]*V[%3d]\n", LUSOL->indr[L1], LUSOL->indr[L1], J,LUSOL->indr[L1], J); - printf("%6g = %6g + %6g*%6g\n", V[LUSOL->indr[L1]] + SUM, TEMP, LUSOL->a[L], V[J]); -#endif - } -#endif - V[LUSOL->indr[L1]] += (REAL) SUM; - } - } - -/* Exit. */ - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; -} - -void print_L0(LUSOLrec *LUSOL) -{ - int I, J, K, L, L1, L2, LEN, LENL0, NUML0; - REAL *denseL0 = (REAL*) calloc(LUSOL->m+1, (LUSOL->n+1)*sizeof(*denseL0)); - - NUML0 = LUSOL->luparm[LUSOL_IP_COLCOUNT_L0]; - LENL0 = LUSOL->luparm[LUSOL_IP_NONZEROS_L0]; - - L2 = LUSOL->lena-LENL0; - for(K = NUML0; K >= 1; K--) { - LEN = LUSOL->lenc[K]; - L1 = L2+1; - L2 += LEN; - for(L = L1; L <= L2; L++) { - I = LUSOL->indc[L]; - I = LUSOL->ipinv[I]; /* Undo row mapping */ - J = LUSOL->indr[L]; - denseL0[(LUSOL->n+1)*(J-1) + I] = LUSOL->a[L]; - } - } - - for(I = 1; I <= LUSOL->n; I++) { - for(J = 1; J <= LUSOL->m; J++) - fprintf(stdout, "%10g", denseL0[(LUSOL->n+1)*(J-1) + I]); - fprintf(stdout, "\n"); - } - LUSOL_FREE(denseL0); -} - - -/* ------------------------------------------------------------------ - Include routines for column-based U. - 5 Feb 2006 Current version - KE. - ------------------------------------------------------------------ */ -#include "lusol6u.c" - - -/* ================================================================== - lu6U solves U w = v. v is not altered. - ------------------------------------------------------------------ - 15 Dec 2002: First version derived from lu6sol. - 15 Dec 2002: Current version. - ================================================================== */ -void LU6U(LUSOLrec *LUSOL, int *INFORM, REAL V[], REAL W[], int NZidx[]) -{ - /* Do column-based U version, if available */ - if((LUSOL->U != NULL) || - ((LUSOL->luparm[LUSOL_IP_FTRANCOUNT] == 0) && LU1U0(LUSOL, &(LUSOL->U), INFORM))) { - LU6U0_v(LUSOL, LUSOL->U, V, W, NZidx, INFORM); - } - /* Alternatively, do the standard column-based L0 version */ - else { - int I, J, K, KLAST, L, L1, L2, L3, NRANK, NRANK1; - REAL SMALL; - register REALXP T; -#ifdef LUSOLFastSolve - REAL *aptr; - int *jptr; -#endif - NRANK = LUSOL->luparm[LUSOL_IP_RANK_U]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - NRANK1 = NRANK+1; -/* Find the first nonzero in v(1:nrank), counting backwards. */ - for(KLAST = NRANK; KLAST >= 1; KLAST--) { - I = LUSOL->ip[KLAST]; - if(fabs(V[I])>SMALL) - break; - } - L = LUSOL->n; -#ifdef LUSOLFastSolve - for(K = KLAST+1, jptr = LUSOL->iq+K; K <= L; K++, jptr++) - W[*jptr] = ZERO; -#else - for(K = KLAST+1; K <= L; K++) { - J = LUSOL->iq[K]; - W[J] = ZERO; - } -#endif -/* Do the back-substitution, using rows 1:klast of U. */ - for(K = KLAST; K >= 1; K--) { - I = LUSOL->ip[K]; - T = V[I]; - L1 = LUSOL->locr[I]; - L2 = L1+1; - L3 = (L1+LUSOL->lenr[I])-1; -/* ***** This loop could be coded specially. */ -#ifdef LUSOLFastSolve - for(L = L2, aptr = LUSOL->a+L2, jptr = LUSOL->indr+L2; - L <= L3; L++, aptr++, jptr++) - T -= (*aptr) * W[*jptr]; -#else - for(L = L2; L <= L3; L++) { - J = LUSOL->indr[L]; - T -= LUSOL->a[L]*W[J]; - } -#endif - J = LUSOL->iq[K]; - if(fabs((REAL) T)<=SMALL) - T = ZERO; - else - T /= LUSOL->a[L1]; - W[J] = (REAL) T; - } -/* Compute residual for overdetermined systems. */ - T = ZERO; - for(K = NRANK1; K <= LUSOL->m; K++) { - I = LUSOL->ip[K]; - T += fabs(V[I]); - } -/* Exit. */ - if(T>ZERO) - *INFORM = LUSOL_INFORM_LUSINGULAR; - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; - LUSOL->parmlu[LUSOL_RP_RESIDUAL_U] = (REAL) T; - } -} - -/* ================================================================== - lu6Ut solves U'v = w. w is destroyed. - ------------------------------------------------------------------ - 15 Dec 2002: First version derived from lu6sol. - 15 Dec 2002: Current version. - ================================================================== */ -void LU6UT(LUSOLrec *LUSOL, int *INFORM, REAL V[], REAL W[], int NZidx[]) -{ - int I, J, K, L, L1, L2, NRANK, NRANK1, - *ip = LUSOL->ip + 1, *iq = LUSOL->iq + 1; - REAL SMALL; - register REAL T; -#ifdef LUSOLFastSolve - REAL *aptr; - int *jptr; -#endif - - NRANK = LUSOL->luparm[LUSOL_IP_RANK_U]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - NRANK1 = NRANK+1; - L = LUSOL->m; -#ifdef LUSOLFastSolve - for(K = NRANK1, jptr = LUSOL->ip+K; K <= L; K++, jptr++) - V[*jptr] = ZERO; -#else - for(K = NRANK1; K <= L; K++) { - I = LUSOL->ip[K]; - V[I] = ZERO; - } -#endif -/* Do the forward-substitution, skipping columns of U(transpose) - when the associated element of w(*) is negligible. */ -#if 0 - for(K = 1; K <= NRANK; K++) { - I = LUSOL->ip[K]; - J = LUSOL->iq[K]; -#else - for(K = 1; K <= NRANK; K++, ip++, iq++) { - I = *ip; - J = *iq; -#endif - T = W[J]; - if(fabs(T)<=SMALL) { - V[I] = ZERO; - continue; - } - L1 = LUSOL->locr[I]; - T /= LUSOL->a[L1]; - V[I] = T; - L2 = (L1+LUSOL->lenr[I])-1; - L1++; -/* ***** This loop could be coded specially. */ -#ifdef LUSOLFastSolve - for(L = L1, aptr = LUSOL->a+L1, jptr = LUSOL->indr+L1; - L <= L2; L++, aptr++, jptr++) - W[*jptr] -= T * (*aptr); -#else - for(L = L1; L <= L2; L++) { - J = LUSOL->indr[L]; - W[J] -= T*LUSOL->a[L]; - } -#endif - } -/* Compute residual for overdetermined systems. */ - T = ZERO; - for(K = NRANK1; K <= LUSOL->n; K++) { - J = LUSOL->iq[K]; - T += fabs(W[J]); - } -/* Exit. */ - if(T>ZERO) - *INFORM = LUSOL_INFORM_LUSINGULAR; - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; - LUSOL->parmlu[LUSOL_RP_RESIDUAL_U] = T; -} - -/* ================================================================== - lu6sol uses the factorization A = L U as follows: - ------------------------------------------------------------------ - mode - 1 v solves L v = v(input). w is not touched. - 2 v solves L'v = v(input). w is not touched. - 3 w solves U w = v. v is not altered. - 4 v solves U'v = w. w is destroyed. - 5 w solves A w = v. v is altered as in 1. - 6 v solves A'v = w. w is destroyed. - - If mode = 3,4,5,6, v and w must not be the same arrays. - If lu1fac has just been used to factorize a symmetric matrix A - (which must be definite or quasi-definite), the factors A = L U - may be regarded as A = LDL', where D = diag(U). In such cases, - - mode - 7 v solves A v = L D L'v = v(input). w is not touched. - 8 v solves L |D| L'v = v(input). w is not touched. - - ip(*), iq(*) hold row and column numbers in pivotal order. - lenc(k) is the length of the k-th column of initial L. - lenr(i) is the length of the i-th row of U. - locc(*) is not used. - locr(i) is the start of the i-th row of U. - - U is assumed to be in upper-trapezoidal form (nrank by n). - The first entry for each row is the diagonal element - (according to the permutations ip, iq). It is stored at - location locr(i) in a(*), indr(*). - - On exit, inform = 0 except as follows. - if(mode = 3,4,5,6 and if U (and hence A) is singular,) - inform = 1 if there is a nonzero residual in solving the system - involving U. parmlu(20) returns the norm of the residual. - ------------------------------------------------------------------ - July 1987: Early version. - 09 May 1988: f77 version. - 27 Apr 2000: Abolished the dreaded "computed go to". - But hard to change other "go to"s to "if then else". - 15 Dec 2002: lu6L, lu6Lt, lu6U, lu6Ut added to modularize lu6sol. - ================================================================== */ -void LU6SOL(LUSOLrec *LUSOL, int MODE, REAL V[], REAL W[], int NZidx[], int *INFORM) -{ - if(MODE==LUSOL_SOLVE_Lv_v) { /* Solve L v(new) = v. */ - LU6L(LUSOL, INFORM,V, NZidx); - } - else if(MODE==LUSOL_SOLVE_Ltv_v) { /* Solve L'v(new) = v. */ - LU6LT(LUSOL, INFORM,V, NZidx); - } - else if(MODE==LUSOL_SOLVE_Uw_v) { /* Solve U w = v. */ - LU6U(LUSOL, INFORM,V,W, NZidx); - } - else if(MODE==LUSOL_SOLVE_Utv_w) { /* Solve U'v = w. */ - LU6UT(LUSOL, INFORM,V,W, NZidx); - } - else if(MODE==LUSOL_SOLVE_Aw_v) { /* Solve A w = v (i.e. FTRAN) */ - LU6L(LUSOL, INFORM,V, NZidx); /* via L v(new) = v */ - LU6U(LUSOL, INFORM,V,W, NULL); /* ... and U w = v(new). */ - } - else if(MODE==LUSOL_SOLVE_Atv_w) { /* Solve A'v = w (i.e. BTRAN) */ - LU6UT(LUSOL, INFORM,V,W, NZidx); /* via U'v = w */ - LU6LT(LUSOL, INFORM,V, NULL); /* ... and L'v(new) = v. */ - } - else if(MODE==LUSOL_SOLVE_Av_v) { /* Solve LDv(bar) = v */ - LU6LD(LUSOL, INFORM,1,V, NZidx); /* and L'v(new) = v(bar). */ - LU6LT(LUSOL, INFORM,V, NULL); - } - else if(MODE==LUSOL_SOLVE_LDLtv_v) { /* Solve L|D|v(bar) = v */ - LU6LD(LUSOL, INFORM,2,V, NZidx); /* and L'v(new) = v(bar). */ - LU6LT(LUSOL, INFORM,V, NULL); - } -} - diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6l0.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6l0.c deleted file mode 100644 index 2a961e03..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6l0.c +++ /dev/null @@ -1,163 +0,0 @@ - -/* Create a row-based version of L0. - This makes it possible to solve L0'x=h (btran) faster for sparse h, - since we only run down the columns of L0' (rows of LO) for which - the corresponding entry in h is non-zero. */ -MYBOOL LU1L0(LUSOLrec *LUSOL, LUSOLmat **mat, int *inform) -{ - MYBOOL status = FALSE; - int K, L, LL, L1, L2, LENL0, NUML0, I; - int *lsumr; - - /* Assume success */ - *inform = LUSOL_INFORM_LUSUCCESS; - - /* Check if there is anything worth doing */ - if(mat == NULL) - return( status ); - if(*mat != NULL) - LUSOL_matfree(mat); - NUML0 = LUSOL->luparm[LUSOL_IP_COLCOUNT_L0]; - LENL0 = LUSOL->luparm[LUSOL_IP_NONZEROS_L0]; - if((NUML0 == 0) || (LENL0 == 0) || - (LUSOL->luparm[LUSOL_IP_ACCELERATION] == LUSOL_BASEORDER) || - ((LUSOL->luparm[LUSOL_IP_ACCELERATION] & LUSOL_ACCELERATE_L0) == 0)) - return( status ); - - /* Allocate temporary array */ - lsumr = (int *) LUSOL_CALLOC((LUSOL->m+1), sizeof(*lsumr)); - if(lsumr == NULL) { - *inform = LUSOL_INFORM_NOMEMLEFT; - return( status ); - } - - /* Compute non-zero counts by permuted row index (order is unimportant) */ - K = 0; - L2 = LUSOL->lena; - L1 = L2-LENL0+1; - for(L = L1; L <= L2; L++) { - I = LUSOL->indc[L]; - lsumr[I]++; - if(lsumr[I] == 1) - K++; - } - LUSOL->luparm[LUSOL_IP_ROWCOUNT_L0] = K; - - /* Check if we should apply "smarts" before proceeding to the row matrix creation */ - if((LUSOL->luparm[LUSOL_IP_ACCELERATION] & LUSOL_AUTOORDER) && - ((REAL) LUSOL->luparm[LUSOL_IP_ROWCOUNT_L0] / -#if 0 - LUSOL->luparm[LUSOL_IP_COLCOUNT_L0] -#else - LUSOL->m -#endif - > LUSOL->parmlu[LUSOL_RP_SMARTRATIO])) - goto Finish; - - /* We are Ok to create the new matrix object */ - *mat = LUSOL_matcreate(LUSOL->m, LENL0); - if(*mat == NULL) { - *inform = LUSOL_INFORM_NOMEMLEFT; - goto Finish; - } - - /* Cumulate row counts to get vector offsets; first row is leftmost - (stick with Fortran array offset for consistency) */ - (*mat)->lenx[0] = 1; - for(K = 1; K <= LUSOL->m; K++) { - (*mat)->lenx[K] = (*mat)->lenx[K-1] + lsumr[K]; - lsumr[K] = (*mat)->lenx[K-1]; - } - - /* Map the matrix into row order by permuted index; - Note: The first permuted row is located leftmost in the array. - The column order is irrelevant, since the indeces will - refer to constant / resolved values of V[] during solve. */ - L2 = LUSOL->lena; - L1 = L2-LENL0+1; - for(L = L1; L <= L2; L++) { - I = LUSOL->indc[L]; - LL = lsumr[I]++; - (*mat)->a[LL] = LUSOL->a[L]; - (*mat)->indr[LL] = LUSOL->indr[L]; - (*mat)->indc[LL] = I; - } - - /* Pack row starting positions, and set mapper from original index to packed */ - I = 0; - for(L = 1; L <= LUSOL->m; L++) { - K = LUSOL->ip[L]; - if((*mat)->lenx[K] > (*mat)->lenx[K-1]) { - I++; - (*mat)->indx[I] = K; - } - } - - /* Confirm that everything went well */ - status = TRUE; - - /* Clean up */ -Finish: - FREE(lsumr); - return( status ); -} - -/* Solve L0' v = v based on row-based version of L0, constructed by LU1L0 */ -void LU6L0T_v(LUSOLrec *LUSOL, LUSOLmat *mat, REAL V[], int NZidx[], int *INFORM) -{ -#ifdef DoTraceL0 - REAL TEMP; -#endif - int LEN, K, KK, L, L1, NUML0; - REAL SMALL; - register REAL VPIV; -#if (defined LUSOLFastSolve) && !(defined DoTraceL0) - REAL *aptr; - int *jptr; -#else - int J; -#endif - - NUML0 = LUSOL->luparm[LUSOL_IP_ROWCOUNT_L0]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - - /* Loop over the nz columns of L0' - from the end, going forward. */ - for(K = NUML0; K > 0; K--) { - KK = mat->indx[K]; - L = mat->lenx[KK]; - L1 = mat->lenx[KK-1]; - LEN = L - L1; - if(LEN == 0) - continue; - /* Get value of the corresponding active entry of V[] */ - VPIV = V[KK]; - /* Only process the column of L0' if the value of V[] is non-zero */ - if(fabs(VPIV)>SMALL) { -/* ***** This loop could be coded specially. */ -#if (defined LUSOLFastSolve) && !(defined DoTraceL0) - L--; - for(aptr = mat->a+L, jptr = mat->indr+L; - LEN > 0; LEN--, aptr--, jptr--) - V[*jptr] += VPIV * (*aptr); -#else - for(; LEN > 0; LEN--) { - L--; - J = mat->indr[L]; -#ifndef DoTraceL0 - V[J] += VPIV * mat->a[L]; -#else - TEMP = V[J]; - V[J] += VPIV * mat->a[L]; - printf("V[%3d] = V[%3d] + L[%d,%d]*V[%3d]\n", J, J, KK,J, KK); - printf("%6g = %6g + %6g*%6g\n", V[J], TEMP, mat->a[L], VPIV); -#endif - } -#endif - } -#ifdef SetSmallToZero - else - V[KK] = 0; -#endif - } - -} diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6u.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6u.c deleted file mode 100644 index e85db5c4..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol6u.c +++ /dev/null @@ -1,176 +0,0 @@ - -/* Create a column-based version of U. - This makes it possible to solve Ux=h (ftran) faster for sparse h, - since we only run down the rows of U (columns of U') for which - the corresponding entry in h is non-zero. */ -MYBOOL LU1U0(LUSOLrec *LUSOL, LUSOLmat **mat, int *inform) -{ - MYBOOL status = FALSE; - int K, L, LL, LENU, NUMU, J; - int *lsumc; - - /* Assume success */ - *inform = LUSOL_INFORM_LUSUCCESS; - - /* Check if there is anything worth doing */ - if(mat == NULL) - return( status ); - if(*mat != NULL) - LUSOL_matfree(mat); - NUMU = LUSOL->luparm[LUSOL_IP_RANK_U]; - LENU = LUSOL->luparm[LUSOL_IP_NONZEROS_U]; - if((NUMU == 0) || (LENU == 0) || - (LUSOL->luparm[LUSOL_IP_ACCELERATION] == LUSOL_BASEORDER) || - ((LUSOL->luparm[LUSOL_IP_ACCELERATION] & LUSOL_ACCELERATE_U) == 0)) - return( status ); - - /* Allocate temporary array */ - lsumc = (int *) LUSOL_CALLOC((LUSOL->n+1), sizeof(*lsumc)); - if(lsumc == NULL) { - *inform = LUSOL_INFORM_NOMEMLEFT; - return( status ); - } - - /* Compute non-zero counts by permuted column index (order is unimportant) */ - for(L = 1; L <= LENU; L++) { - J = LUSOL->indr[L]; - lsumc[J]++; - } - - /* Check if we should apply "smarts" before proceeding to the column matrix creation */ - if((LUSOL->luparm[LUSOL_IP_ACCELERATION] & LUSOL_AUTOORDER) && - ((REAL) sqrt((REAL) NUMU/LENU) > LUSOL->parmlu[LUSOL_RP_SMARTRATIO])) - goto Finish; - - /* We are Ok to create the new matrix object */ - *mat = LUSOL_matcreate(LUSOL->n, LENU); - if(*mat == NULL) { - *inform = LUSOL_INFORM_NOMEMLEFT; - goto Finish; - } - - /* Cumulate row counts to get vector offsets; first column is leftmost - (stick with Fortran array offset for consistency) */ - (*mat)->lenx[0] = 1; - for(K = 1; K <= LUSOL->n; K++) { - (*mat)->lenx[K] = (*mat)->lenx[K-1] + lsumc[K]; - lsumc[K] = (*mat)->lenx[K-1]; - } - - /* Map the matrix into column order by permuted index; - Note: The first permuted column is located leftmost in the array. - The row order is irrelevant, since the indeces will - refer to constant / resolved values of V[] during solve. */ - for(L = 1; L <= LENU; L++) { - J = LUSOL->indr[L]; - LL = lsumc[J]++; - (*mat)->a[LL] = LUSOL->a[L]; - (*mat)->indr[LL] = J; - (*mat)->indc[LL] = LUSOL->indc[L]; - } - - /* Pack column starting positions, and set mapper from original index to packed */ - J = 0; - for(L = 1; L <= LUSOL->n; L++) { - K = LUSOL->iq[L]; -#if 1 /* Deactivate to produce a full-rank version (implicit unit diagonals) */ - if((*mat)->lenx[K] > (*mat)->lenx[K-1]) -#endif - { - J++; - (*mat)->indx[J] = K; - } - } - - /* Confirm that everything went well */ - status = TRUE; - - /* Clean up */ -Finish: - FREE(lsumc); - return( status ); -} - -/* Solve U w = v based on column-based version of U, constructed by LU1U0 */ -void LU6U0_v(LUSOLrec *LUSOL, LUSOLmat *mat, REAL V[], REAL W[], int NZidx[], int *INFORM) -{ -#ifdef DoTraceU0 - REAL TEMP; -#endif - int LEN, I, K, L, L1, NRANK, NRANK1, KLAST; - REAL SMALL; - register REAL T; -#if (defined xxLUSOLFastSolve) && !(defined DoTraceU0) - REAL *aptr; - int *jptr; -#else - int J; -#endif - - NRANK = LUSOL->luparm[LUSOL_IP_RANK_U]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *INFORM = LUSOL_INFORM_LUSUCCESS; - NRANK1 = NRANK+1; -/* Find the first nonzero in v(1:nrank), counting backwards. */ - for(KLAST = NRANK; KLAST >= 1; KLAST--) { - I = LUSOL->ip[KLAST]; - if(fabs(V[I])>SMALL) - break; - } - L = LUSOL->n; -#ifdef xxLUSOLFastSolve - for(K = KLAST+1, jptr = LUSOL->iq+K; K <= L; K++, jptr++) - W[*jptr] = ZERO; -#else - for(K = KLAST+1; K <= L; K++) { - J = LUSOL->iq[K]; - W[J] = ZERO; - } -#endif - /* Loop over the nz columns of U - from the right, going left. */ - for(K = NRANK; K > 0; K--) { - I = mat->indx[K]; - L = mat->lenx[I]; - L1 = mat->lenx[I-1]; - LEN = L - L1; - T = V[I]; - if(fabs(T)<=SMALL) { - W[K] = ZERO; - continue; - } - T /= mat->a[L1]; /* Should it be L or L1 ? */ - W[K] = T; - LEN--; -/* ***** This loop could be coded specially. */ -#ifdef xxLUSOLFastSolve - L--; - for(aptr = mat->a+L, jptr = mat->indc+L; - LEN > 0; LEN--, aptr--, jptr--) - V[*jptr] -= T * (*aptr); -#else - for(; LEN > 0; LEN--) { - L--; - J = mat->indc[L]; -#ifndef DoTraceL0 - V[J] -= T * mat->a[L]; -#else - TEMP = V[J]; - V[J] += T * mat->a[L]; - printf("V[%3d] = V[%3d] + L[%d,%d]*V[%3d]\n", J, J, I,J, I); - printf("%6g = %6g + %6g*%6g\n", V[J], TEMP, mat->a[L], T); -#endif - } -#endif - } -/* Compute residual for overdetermined systems. */ - T = ZERO; - for(K = NRANK1; K <= LUSOL->m; K++) { - I = LUSOL->ip[K]; - T += fabs(V[I]); - } -/* Exit. */ - if(T>ZERO) - *INFORM = LUSOL_INFORM_LUSINGULAR; - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; - LUSOL->parmlu[LUSOL_RP_RESIDUAL_U] = (REAL) T; -} diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol7a.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol7a.c deleted file mode 100644 index c5b69ad9..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol7a.c +++ /dev/null @@ -1,704 +0,0 @@ - -/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - File lusol7a - lu7add lu7cyc lu7elm lu7for lu7rnk lu7zap - Utilities for LUSOL's update routines. - lu7for is the most important -- the forward sweep. - 01 May 2002: Derived from LUSOL's original lu7a.f file. - 01 May 2002: Current version of lusol7a.f. - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */ - -/* ================================================================== - lu7add inserts the first nrank elements of the vector v(*) - as column jadd of U. We assume that U does not yet have any - entries in this column. - Elements no larger than parmlu(3) are treated as zero. - klast will be set so that the last row to be affected - (in pivotal order) is row ip(klast). - ------------------------------------------------------------------ - 09 May 1988: First f77 version. - ================================================================== */ -void LU7ADD(LUSOLrec *LUSOL, int JADD, REAL V[], int LENL, int *LENU, - int *LROW, int NRANK, int *INFORM, int *KLAST, REAL *VNORM) -{ - REAL SMALL; - int K, I, LENI, MINFRE, NFREE, LR1, LR2, L; -#ifndef LUSOLFastMove - int J; -#endif - - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - *VNORM = ZERO; - *KLAST = 0; - for(K = 1; K <= NRANK; K++) { - I = LUSOL->ip[K]; - if(fabs(V[I])<=SMALL) - continue; - *KLAST = K; - (*VNORM) += fabs(V[I]); - LENI = LUSOL->lenr[I]; -/* Compress row file if necessary. */ - MINFRE = LENI+1; - NFREE = LUSOL->lena - LENL - *LROW; - if(NFREEm, TRUE,LROW,LUSOL->indr,LUSOL->lenr,LUSOL->locr); - NFREE = LUSOL->lena - LENL - *LROW; - if(NFREElocr[I] = (*LROW) + 1; - LR1 = LUSOL->locr[I]; - LR2 = (LR1+LENI)-1; - if(LR2==*LROW) - goto x150; - if(LUSOL->indr[LR2+1]==0) - goto x180; - LUSOL->locr[I] = (*LROW) + 1; -#ifdef LUSOLFastMove - L = LR2-LR1+1; - if(L > 0) { - LR2 = (*LROW)+1; - MEMMOVE(LUSOL->a+LR2, LUSOL->a+LR1, L); - MEMMOVE(LUSOL->indr+LR2, LUSOL->indr+LR1, L); - MEMCLEAR(LUSOL->indr+LR1, L); - *LROW += L; - } -#else - for(L = LR1; L <= LR2; L++) { - (*LROW)++; - LUSOL->a[*LROW] = LUSOL->a[L]; - J = LUSOL->indr[L]; - LUSOL->indr[L] = 0; - LUSOL->indr[*LROW] = J; - } -#endif -x150: - LR2 = *LROW; - (*LROW)++; -/* Add the element of v. */ -x180: - LR2++; - LUSOL->a[LR2] = V[I]; - LUSOL->indr[LR2] = JADD; - LUSOL->lenr[I] = LENI+1; - (*LENU)++; - } -/* Normal exit. */ - *INFORM = LUSOL_INFORM_LUSUCCESS; - goto x990; -/* Not enough storage. */ -x970: - *INFORM = LUSOL_INFORM_ANEEDMEM; -x990: -; -} - -/* ================================================================== - lu7cyc performs a cyclic permutation on the row or column ordering - stored in ip, moving entry kfirst down to klast. - If kfirst .ge. klast, lu7cyc should not be called. - Sometimes klast = 0 and nothing should happen. - ------------------------------------------------------------------ - 09 May 1988: First f77 version. - ================================================================== */ -void LU7CYC(LUSOLrec *LUSOL, int KFIRST, int KLAST, int IX[]) -{ - if(KFIRST 0, y has just become column jelm of the matrix A. - lu7elm should not be called unless m is greater than nrank. - inform = 0 if y contained no subdiagonal nonzeros to eliminate. - inform = 1 if y contained at least one nontrivial subdiagonal. - inform = 7 if there is insufficient storage. - ------------------------------------------------------------------ - 09 May 1988: First f77 version. - No longer calls lu7for at end. lu8rpc, lu8mod do so. - ================================================================== */ -void LU7ELM(LUSOLrec *LUSOL, int JELM, REAL V[], int *LENL, - int *LROW, int NRANK, int *INFORM, REAL *DIAG) -{ - REAL VI, VMAX, SMALL; - int NRANK1, MINFRE, NFREE, KMAX, L, K, I, LMAX, IMAX, L1, L2; - -#ifdef ForceInitialization - LMAX = 0; -#endif - - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - NRANK1 = NRANK+1; - *DIAG = ZERO; -/* Compress row file if necessary. */ - MINFRE = LUSOL->m-NRANK; - NFREE = LUSOL->lena-(*LENL)-(*LROW); - if(NFREE>=MINFRE) - goto x100; - LU1REC(LUSOL, LUSOL->m,TRUE,LROW,LUSOL->indr,LUSOL->lenr,LUSOL->locr); - NFREE = LUSOL->lena-(*LENL)-(*LROW); - if(NFREElena-(*LENL))+1; - for(K = NRANK1; K <= LUSOL->m; K++) { - I = LUSOL->ip[K]; - VI = fabs(V[I]); - if(VI<=SMALL) - continue; - L--; - LUSOL->a[L] = V[I]; - LUSOL->indc[L] = I; - if(VMAX>=VI) - continue; - VMAX = VI; - KMAX = K; - LMAX = L; - } - if(KMAX==0) - goto x900; -/* ------------------------------------------------------------------ - Remove vmax by overwriting it with the last packed v(i). - Then set the multipliers in L for the other elements. - ------------------------------------------------------------------ */ - IMAX = LUSOL->ip[KMAX]; - VMAX = LUSOL->a[LMAX]; - LUSOL->a[LMAX] = LUSOL->a[L]; - LUSOL->indc[LMAX] = LUSOL->indc[L]; - L1 = L+1; - L2 = LUSOL->lena-(*LENL); - *LENL = ((*LENL)+L2)-L; - for(L = L1; L <= L2; L++) { - LUSOL->a[L] /= -VMAX; - LUSOL->indr[L] = IMAX; - } -/* Move the row containing vmax to pivotal position nrank + 1. */ - LUSOL->ip[KMAX] = LUSOL->ip[NRANK1]; - LUSOL->ip[NRANK1] = IMAX; - *DIAG = VMAX; -/* ------------------------------------------------------------------ - If jelm is positive, insert vmax into a new row of U. - This is now the only subdiagonal element. - ------------------------------------------------------------------ */ - if(JELM>0) { - (*LROW)++; - LUSOL->locr[IMAX] = *LROW; - LUSOL->lenr[IMAX] = 1; - LUSOL->a[*LROW] = VMAX; - LUSOL->indr[*LROW] = JELM; - } - *INFORM = LUSOL_INFORM_LUSINGULAR; - goto x990; -/* No elements to eliminate. */ -x900: - *INFORM = LUSOL_INFORM_LUSUCCESS; - goto x990; -/* Not enough storage. */ -x970: - *INFORM = LUSOL_INFORM_ANEEDMEM; -x990: -; -} - -/* ================================================================== - lu7for (forward sweep) updates the LU factorization A = L*U - when row iw = ip(klast) of U is eliminated by a forward - sweep of stabilized row operations, leaving ip * U * iq upper - triangular. - The row permutation ip is updated to preserve stability and/or - sparsity. The column permutation iq is not altered. - kfirst is such that row ip(kfirst) is the first row involved - in eliminating row iw. (Hence, kfirst marks the first nonzero - in row iw in pivotal order.) If kfirst is unknown it may be - input as 1. - klast is such that row ip(klast) is the row being eliminated. - klast is not altered. - lu7for should be called only if kfirst .le. klast. - If kfirst = klast, there are no nonzeros to eliminate, but the - diagonal element of row ip(klast) may need to be moved to the - front of the row. - ------------------------------------------------------------------ - On entry, locc(*) must be zero. - - On exit: - inform = 0 if row iw has a nonzero diagonal (could be small). - inform = 1 if row iw has no diagonal. - inform = 7 if there is not enough storage to finish the update. - - On a successful exit (inform le 1), locc(*) will again be zero. - ------------------------------------------------------------------ - Jan 1985: Final f66 version. - 09 May 1988: First f77 version. - ================================================================== */ -void LU7FOR(LUSOLrec *LUSOL, int KFIRST, int KLAST, int *LENL, int *LENU, - int *LROW, int *INFORM, REAL *DIAG) -{ - MYBOOL SWAPPD; - int KBEGIN, IW, LENW, LW1, LW2, JFIRST, MINFRE, NFREE, L, J, KSTART, KSTOP, K, - LFIRST, IV, LENV, LV1, JLAST, LV2, LV3, LV, JV, LW, LDIAG, LIMIT; - REAL AMULT, LTOL, USPACE, SMALL, VJ, WJ; - - LTOL = LUSOL->parmlu[LUSOL_RP_UPDATEMAX_Lij]; - SMALL = LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE]; - USPACE = LUSOL->parmlu[LUSOL_RP_COMPSPACE_U]; - KBEGIN = KFIRST; - SWAPPD = FALSE; - -/* We come back here from below if a row interchange is performed. */ -x100: - IW = LUSOL->ip[KLAST]; - LENW = LUSOL->lenr[IW]; - if(LENW==0) - goto x910; - LW1 = LUSOL->locr[IW]; - LW2 = (LW1+LENW)-1; - JFIRST = LUSOL->iq[KBEGIN]; - if(KBEGIN>=KLAST) - goto x700; -/* Make sure there is room at the end of the row file - in case row iw is moved there and fills in completely. */ - MINFRE = LUSOL->n+1; - NFREE = LUSOL->lena-(*LENL)-(*LROW); - if(NFREEm,TRUE,LROW,LUSOL->indr,LUSOL->lenr,LUSOL->locr); - LW1 = LUSOL->locr[IW]; - LW2 = (LW1+LENW)-1; - NFREE = LUSOL->lena-(*LENL)-(*LROW); - if(NFREEindr[L]; - LUSOL->locc[J] = L; - } -/* ================================================================== - Main elimination loop. - ================================================================== */ - KSTART = KBEGIN; - KSTOP = MIN(KLAST,LUSOL->n); - for(K = KSTART; K <= KSTOP; K++) { - JFIRST = LUSOL->iq[K]; - LFIRST = LUSOL->locc[JFIRST]; - if(LFIRST==0) - goto x490; -/* Row iw has its first element in column jfirst. */ - WJ = LUSOL->a[LFIRST]; - if(K==KLAST) - goto x490; -/* --------------------------------------------------------------- - We are about to use the first element of row iv - to eliminate the first element of row iw. - However, we may wish to interchange the rows instead, - to preserve stability and/or sparsity. - --------------------------------------------------------------- */ - IV = LUSOL->ip[K]; - LENV = LUSOL->lenr[IV]; - LV1 = LUSOL->locr[IV]; - VJ = ZERO; - if(LENV==0) - goto x150; - if(LUSOL->indr[LV1]!=JFIRST) - goto x150; - VJ = LUSOL->a[LV1]; - if(SWAPPD) - goto x200; - if(LTOL*fabs(WJ)ip[KLAST] = IV; - LUSOL->ip[K] = IW; - KBEGIN = K; - SWAPPD = TRUE; - goto x600; -/* --------------------------------------------------------------- - Delete the eliminated element from row iw - by overwriting it with the last element. - --------------------------------------------------------------- */ -x200: - LUSOL->a[LFIRST] = LUSOL->a[LW2]; - JLAST = LUSOL->indr[LW2]; - LUSOL->indr[LFIRST] = JLAST; - LUSOL->indr[LW2] = 0; - LUSOL->locc[JLAST] = LFIRST; - LUSOL->locc[JFIRST] = 0; - LENW--; - (*LENU)--; - if(*LROW==LW2) - (*LROW)--; - LW2 = LW2-1; -/* --------------------------------------------------------------- - Form the multiplier and store it in the L file. - --------------------------------------------------------------- */ - if(fabs(WJ)<=SMALL) - goto x490; - AMULT = -WJ/VJ; - L = LUSOL->lena-(*LENL); - LUSOL->a[L] = AMULT; - LUSOL->indr[L] = IV; - LUSOL->indc[L] = IW; - (*LENL)++; -/* --------------------------------------------------------------- - Add the appropriate multiple of row iv to row iw. - We use two different inner loops. The first one is for the - case where row iw is not at the end of storage. - --------------------------------------------------------------- */ - if(LENV==1) - goto x490; - LV2 = LV1+1; - LV3 = (LV1+LENV)-1; - if(LW2==*LROW) - goto x400; -/* ............................................................... - This inner loop will be interrupted only if - fill-in occurs enough to bump into the next row. - ............................................................... */ - for(LV = LV2; LV <= LV3; LV++) { - JV = LUSOL->indr[LV]; - LW = LUSOL->locc[JV]; - if(LW>0) { -/* No fill-in. */ - LUSOL->a[LW] += AMULT*LUSOL->a[LV]; - if(fabs(LUSOL->a[LW])<=SMALL) { -/* Delete small computed element. */ - LUSOL->a[LW] = LUSOL->a[LW2]; - J = LUSOL->indr[LW2]; - LUSOL->indr[LW] = J; - LUSOL->indr[LW2] = 0; - LUSOL->locc[J] = LW; - LUSOL->locc[JV] = 0; - (*LENU)--; - LENW--; - LW2--; - } - } - else { -/* Row iw doesn't have an element in column jv yet - so there is a fill-in. */ - if(LUSOL->indr[LW2+1]!=0) - goto x360; - (*LENU)++; - LENW++; - LW2++; - LUSOL->a[LW2] = AMULT*LUSOL->a[LV]; - LUSOL->indr[LW2] = JV; - LUSOL->locc[JV] = LW2; - } - } - goto x490; -/* Fill-in interrupted the previous loop. - Move row iw to the end of the row file. */ -x360: - LV2 = LV; - LUSOL->locr[IW] = (*LROW)+1; - -#ifdef LUSOLFastMove - L = LW2-LW1+1; - if(L > 0) { - int loci, *locp; - for(loci = LW1, locp = LUSOL->indr+LW1; - loci <= LW2; loci++, locp++) { - (*LROW)++; - LUSOL->locc[*locp] = *LROW; - } - LW2 = (*LROW)-L+1; - MEMMOVE(LUSOL->a+LW2, LUSOL->a+LW1, L); - MEMMOVE(LUSOL->indr+LW2, LUSOL->indr+LW1, L); - MEMCLEAR(LUSOL->indr+LW1, L); - } -#else - for(L = LW1; L <= LW2; L++) { - (*LROW)++; - LUSOL->a[*LROW] = LUSOL->a[L]; - J = LUSOL->indr[L]; - LUSOL->indr[L] = 0; - LUSOL->indr[*LROW] = J; - LUSOL->locc[J] = *LROW; - } -#endif - LW1 = LUSOL->locr[IW]; - LW2 = *LROW; -/* ............................................................... - Inner loop with row iw at the end of storage. - ............................................................... */ -x400: - for(LV = LV2; LV <= LV3; LV++) { - JV = LUSOL->indr[LV]; - LW = LUSOL->locc[JV]; - if(LW>0) { -/* No fill-in. */ - LUSOL->a[LW] += AMULT*LUSOL->a[LV]; - if(fabs(LUSOL->a[LW])<=SMALL) { -/* Delete small computed element. */ - LUSOL->a[LW] = LUSOL->a[LW2]; - J = LUSOL->indr[LW2]; - LUSOL->indr[LW] = J; - LUSOL->indr[LW2] = 0; - LUSOL->locc[J] = LW; - LUSOL->locc[JV] = 0; - (*LENU)--; - LENW--; - LW2--; - } - } - else { -/* Row iw doesn't have an element in column jv yet - so there is a fill-in. */ - (*LENU)++; - LENW++; - LW2++; - LUSOL->a[LW2] = AMULT*LUSOL->a[LV]; - LUSOL->indr[LW2] = JV; - LUSOL->locc[JV] = LW2; - } - } - *LROW = LW2; -/* The k-th element of row iw has been processed. - Reset swappd before looking at the next element. */ -x490: - SWAPPD = FALSE; - } -/* ================================================================== - End of main elimination loop. - ================================================================== - - Cancel markers on row iw. */ -x600: - LUSOL->lenr[IW] = LENW; - if(LENW==0) - goto x910; - for(L = LW1; L <= LW2; L++) { - J = LUSOL->indr[L]; - LUSOL->locc[J] = 0; - } -/* Move the diagonal element to the front of row iw. - At this stage, lenw gt 0 and klast le n. */ -x700: - for(L = LW1; L <= LW2; L++) { - LDIAG = L; - if(LUSOL->indr[L]==JFIRST) - goto x730; - } - goto x910; - -x730: - *DIAG = LUSOL->a[LDIAG]; - LUSOL->a[LDIAG] = LUSOL->a[LW1]; - LUSOL->a[LW1] = *DIAG; - LUSOL->indr[LDIAG] = LUSOL->indr[LW1]; - LUSOL->indr[LW1] = JFIRST; -/* If an interchange is needed, repeat from the beginning with the - new row iw, knowing that the opposite interchange cannot occur. */ - if(SWAPPD) - goto x100; - *INFORM = LUSOL_INFORM_LUSUCCESS; - goto x950; -/* Singular. */ -x910: - *DIAG = ZERO; - *INFORM = LUSOL_INFORM_LUSINGULAR; -/* Force a compression if the file for U is much longer than the - no. of nonzeros in U (i.e. if lrow is much bigger than lenU). - This should prevent memory fragmentation when there is far more - memory than necessary (i.e. when lena is huge). */ -x950: - LIMIT = (int) (USPACE*(*LENU))+LUSOL->m+LUSOL->n+1000; - if(*LROW>LIMIT) - LU1REC(LUSOL, LUSOL->m,TRUE,LROW,LUSOL->indr,LUSOL->lenr,LUSOL->locr); - goto x990; -/* Not enough storage. */ -x970: - *INFORM = LUSOL_INFORM_ANEEDMEM; -/* Exit. */ -x990: -; -} - -/* ================================================================== - lu7rnk (check rank) assumes U is currently nrank by n - and determines if row nrank contains an acceptable pivot. - If not, the row is deleted and nrank is decreased by 1. - jsing is an input parameter (not altered). If jsing is positive, - column jsing has already been judged dependent. A substitute - (if any) must be some other column. - ------------------------------------------------------------------ - -- Jul 1987: First version. - 09 May 1988: First f77 version. - ================================================================== */ -void LU7RNK(LUSOLrec *LUSOL, int JSING, int *LENU, - int *LROW, int *NRANK, int *INFORM, REAL *DIAG) -{ - REAL UTOL1, UMAX; - int IW, LENW, L1, L2, LMAX, L, JMAX, KMAX; - -#ifdef ForceInitialization - L1 = 0; - L2 = 0; -#endif - - UTOL1 = LUSOL->parmlu[LUSOL_RP_SMALLDIAG_U]; - *DIAG = ZERO; -/* Find Umax, the largest element in row nrank. */ - IW = LUSOL->ip[*NRANK]; - LENW = LUSOL->lenr[IW]; - if(LENW==0) - goto x400; - L1 = LUSOL->locr[IW]; - L2 = (L1+LENW)-1; - UMAX = ZERO; - LMAX = L1; - for(L = L1; L <= L2; L++) { - if(UMAXa[L])) { - UMAX = fabs(LUSOL->a[L]); - LMAX = L; - } - } -/* Find which column that guy is in (in pivotal order). - Interchange him with column nrank, then move him to be - the new diagonal at the front of row nrank. */ - *DIAG = LUSOL->a[LMAX]; - JMAX = LUSOL->indr[LMAX]; - for(KMAX = *NRANK; KMAX <= LUSOL->n; KMAX++) { - if(LUSOL->iq[KMAX]==JMAX) - break; - } - LUSOL->iq[KMAX] = LUSOL->iq[*NRANK]; - LUSOL->iq[*NRANK] = JMAX; - LUSOL->a[LMAX] = LUSOL->a[L1]; - LUSOL->a[L1] = *DIAG; - LUSOL->indr[LMAX] = LUSOL->indr[L1]; - LUSOL->indr[L1] = JMAX; -/* See if the new diagonal is big enough. */ - if(UMAX<=UTOL1) - goto x400; - if(JMAX==JSING) - goto x400; -/* ------------------------------------------------------------------ - The rank stays the same. - ------------------------------------------------------------------ */ - *INFORM = LUSOL_INFORM_LUSUCCESS; - return; -/* ------------------------------------------------------------------ - The rank decreases by one. - ------------------------------------------------------------------ */ -x400: - *INFORM = LUSOL_INFORM_RANKLOSS; - (*NRANK)--; - if(LENW>0) { -/* Delete row nrank from U. */ - LENU = LENU-LENW; - LUSOL->lenr[IW] = 0; - for(L = L1; L <= L2; L++) { - LUSOL->indr[L] = 0; - } - if(L2==*LROW) { -/* This row was at the end of the data structure. - We have to reset lrow. - Preceding rows might already have been deleted, so we - have to be prepared to go all the way back to 1. */ - for(L = 1; L <= L2; L++) { - if(LUSOL->indr[*LROW]>0) - goto x900; - (*LROW)--; - } - } - } -x900: -; -} - -/* ================================================================== - lu7zap eliminates all nonzeros in column jzap of U. - It also sets kzap to the position of jzap in pivotal order. - Thus, on exit we have iq(kzap) = jzap. - ------------------------------------------------------------------ - -- Jul 1987: nrank added. - 10 May 1988: First f77 version. - ================================================================== */ -void LU7ZAP(LUSOLrec *LUSOL, int JZAP, int *KZAP, int *LENU, int *LROW, - int NRANK) -{ - int K, I, LENI, LR1, LR2, L; - - for(K = 1; K <= NRANK; K++) { - I = LUSOL->ip[K]; - LENI = LUSOL->lenr[I]; - if(LENI==0) - goto x90; - LR1 = LUSOL->locr[I]; - LR2 = (LR1+LENI)-1; - for(L = LR1; L <= LR2; L++) { - if(LUSOL->indr[L]==JZAP) - goto x60; - } - goto x90; -/* Delete the old element. */ -x60: - LUSOL->a[L] = LUSOL->a[LR2]; - LUSOL->indr[L] = LUSOL->indr[LR2]; - LUSOL->indr[LR2] = 0; - LUSOL->lenr[I] = LENI-1; - (*LENU)--; -/* Stop if we know there are no more rows containing jzap. */ -x90: - *KZAP = K; - if(LUSOL->iq[K]==JZAP) - goto x800; - } -/* nrank must be smaller than n because we haven't found kzap yet. */ - L = LUSOL->n; - for(K = NRANK+1; K <= L; K++) { - *KZAP = K; - if(LUSOL->iq[K]==JZAP) - break; - } -/* See if we zapped the last element in the file. */ -x800: - if(*LROW>0) { - if(LUSOL->indr[*LROW]==0) - (*LROW)--; - } - -} - diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol8a.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol8a.c deleted file mode 100644 index 979c5af2..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/LUSOL/lusol8a.c +++ /dev/null @@ -1,279 +0,0 @@ - -/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ - File lusol8a - lu8rpc - Sparse LU update: Replace Column - LUSOL's sparse implementation of the Bartels-Golub update. - - 01 May 2002: Derived from LUSOL's original lu8a.f file. - 01 May 2002: Current version of lusol8a.f. - ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */ - -/* ================================================================== - lu8rpc updates the LU factorization A = L*U when column jrep - is replaced by some vector a(new). - lu8rpc is an implementation of the Bartels-Golub update, - designed for the case where A is rectangular and/or singular. - L is a product of stabilized eliminations (m x m, nonsingular). - P U Q is upper trapezoidal (m x n, rank nrank). - - If mode1 = 0, the old column is taken to be zero - (so it does not have to be removed from U). - If mode1 = 1, the old column need not have been zero. - If mode2 = 0, the new column is taken to be zero. - v(*) is not used or altered. - If mode2 = 1, v(*) must contain the new column a(new). - On exit, v(*) will satisfy L*v = a(new). - If mode2 = 2, v(*) must satisfy L*v = a(new). - - The array w(*) is not used or altered. - On entry, all elements of locc are assumed to be zero. - On a successful exit (inform != 7), this will again be true. - On exit: - - inform = -1 if the rank of U decreased by 1. - inform = 0 if the rank of U stayed the same. - inform = 1 if the rank of U increased by 1. - inform = 2 if the update seemed to be unstable - (diag much bigger than vnorm). - inform = 7 if the update was not completed (lack of storage). - inform = 8 if jrep is not between 1 and n. - ------------------------------------------------------------------ - -- Jan 1985: Original F66 version. - -- Jul 1987: Modified to maintain U in trapezoidal form. - 10 May 1988: First f77 version. - 16 Oct 2000: Added test for instability (inform = 2). - ================================================================== */ -void LU8RPC(LUSOLrec *LUSOL, int MODE1, int MODE2, - int JREP, REAL V[], REAL W[], - int *INFORM, REAL *DIAG, REAL *VNORM) -{ - MYBOOL SINGLR; - int LPRINT, NRANK, LENL, LENU, LROW, NRANK0, KREP, KLAST, IW, L1, J1, JSING; - REAL UTOL1, UTOL2; - - LPRINT = LUSOL->luparm[LUSOL_IP_PRINTLEVEL]; - NRANK = LUSOL->luparm[LUSOL_IP_RANK_U]; - LENL = LUSOL->luparm[LUSOL_IP_NONZEROS_L]; - LENU = LUSOL->luparm[LUSOL_IP_NONZEROS_U]; - LROW = LUSOL->luparm[LUSOL_IP_NONZEROS_ROW]; - UTOL1 = LUSOL->parmlu[LUSOL_RP_SMALLDIAG_U]; - UTOL2 = LUSOL->parmlu[LUSOL_RP_EPSDIAG_U]; - NRANK0 = NRANK; - *DIAG = ZERO; - *VNORM = ZERO; - if(JREP<1) - goto x980; - if(JREP>LUSOL->n) - goto x980; - -/* ------------------------------------------------------------------ - If mode1 = 0, there are no elements to be removed from U - but we still have to set krep (using a backward loop). - Otherwise, use lu7zap to remove column jrep from U - and set krep at the same time. - ------------------------------------------------------------------ */ - if(MODE1==LUSOL_UPDATE_OLDEMPTY) { - KREP = LUSOL->n+1; -x10: - KREP--; - if(LUSOL->iq[KREP]!=JREP) - goto x10; - } - else - LU7ZAP(LUSOL, JREP,&KREP,&LENU,&LROW,NRANK); - -/* ------------------------------------------------------------------ - Insert a new column of u and find klast. - ------------------------------------------------------------------ */ - if(MODE2==LUSOL_UPDATE_NEWEMPTY) { - KLAST = 0; - } - else { - if(MODE2==LUSOL_UPDATE_NEWNONEMPTY) { -/* Transform v = a(new) to satisfy L*v = a(new). */ - LU6SOL(LUSOL, LUSOL_SOLVE_Lv_v, V,W, NULL, INFORM); - } - else if(V==NULL) -/* Otherwise, the V vector is taken to satisfy this already, or stored earlier. */ - V=LUSOL->vLU6L; - - -/* Insert into U any nonzeros in the top of v. - row ip(klast) will contain the last nonzero in pivotal order. - Note that klast will be in the range ( 0, nrank ). */ - LU7ADD(LUSOL, JREP,V,LENL,&LENU,&LROW,NRANK,INFORM,&KLAST,VNORM); - if(*INFORM==LUSOL_INFORM_ANEEDMEM) - goto x970; - } -/* ------------------------------------------------------------------ - In general, the new column causes U to look like this: - krep n krep n - ....a......... ..........a... - . a . . a . - . a . . a . - .a . . a . - P U Q = a . or . a . - b. . . a . - b . . . a . - b . . . a . - b ...... ..a... nrank - c c - c c - c c m - klast points to the last nonzero "a" or "b". - klast = 0 means all "a" and "b" entries are zero. - ------------------------------------------------------------------ */ - if(MODE2==LUSOL_UPDATE_NEWEMPTY) { - if(KREP>NRANK) - goto x900; - } - else if(NRANKm) { -/* Eliminate any "c"s (in either case). - Row nrank + 1 may end up containing one nonzero. */ - LU7ELM(LUSOL, JREP,V,&LENL,&LROW,NRANK,INFORM,DIAG); - if(*INFORM==LUSOL_INFORM_ANEEDMEM) - goto x970; - if(*INFORM==LUSOL_INFORM_LUSINGULAR) { -/* The nonzero is apparently significant. - Increase nrank by 1 and make klast point to the bottom. */ - NRANK++; - KLAST = NRANK; - } - } - if(NRANKn) { -/* The column rank is low. - In the first case, we want the new column to end up in - position nrank, so the trapezoidal columns will have a chance - later on (in lu7rnk) to pivot in that position. - Otherwise the new column is not part of the triangle. We - swap it into position nrank so we can judge it for singularity. - lu7rnk might choose some other trapezoidal column later. */ - if(KREPiq[KREP] = LUSOL->iq[NRANK]; - LUSOL->iq[NRANK] = JREP; - KREP = NRANK; - } - } -/* ------------------------------------------------------------------ - If krep .lt. klast, there are some "b"s to eliminate: - krep - ....a......... - . a . - . a . - .a . - P U Q = a . krep - b. . - b . . - b . . - b ...... nrank - If krep .eq. klast, there are no "b"s, but the last "a" still - has to be moved to the front of row krep (by lu7for). - ------------------------------------------------------------------ */ - if(KREP<=KLAST) { -/* Perform a cyclic permutation on the current pivotal order, - and eliminate the resulting row spike. krep becomes klast. - The final diagonal (if any) will be correctly positioned at - the front of the new krep-th row. nrank stays the same. */ - LU7CYC(LUSOL, KREP,KLAST,LUSOL->ip); - LU7CYC(LUSOL, KREP,KLAST,LUSOL->iq); - LU7FOR(LUSOL, KREP,KLAST,&LENL,&LENU,&LROW,INFORM,DIAG); - if(*INFORM==LUSOL_INFORM_ANEEDMEM) - goto x970; - KREP = KLAST; -/* Test for instability (diag much bigger than vnorm). */ - SINGLR = (MYBOOL) ((*VNORM)ip[KREP]; - SINGLR = (MYBOOL) (LUSOL->lenr[IW]==0); - if(!SINGLR) { - L1 = LUSOL->locr[IW]; - J1 = LUSOL->indr[L1]; - SINGLR = (MYBOOL) (J1!=JREP); - if(!SINGLR) { - *DIAG = LUSOL->a[L1]; - SINGLR = (MYBOOL) (fabs(*DIAG)<=UTOL1 || fabs(*DIAG)<=UTOL2*(*VNORM)); - } - } - if(SINGLR && (KREPip); - LU7CYC(LUSOL, KREP,LUSOL->n,LUSOL->iq); - LU7FOR(LUSOL, KREP,NRANK,&LENL,&LENU,&LROW,INFORM,DIAG); - if(*INFORM==LUSOL_INFORM_ANEEDMEM) - goto x970; - } -/* Find the best column to be in position nrank. - If singlr, it can't be the new column, jrep. - If nothing satisfactory exists, nrank will be decreased. */ - if(SINGLR || (NRANKn)) { - JSING = 0; - if(SINGLR) - JSING = JREP; - LU7RNK(LUSOL, JSING,&LENU,&LROW,&NRANK,INFORM,DIAG); - } - -/* ------------------------------------------------------------------ - Update indeces of optional row-based version of L0. - ------------------------------------------------------------------ */ -#if 0 - if(LUSOL->L0 != NULL) - LU1L0UPD(LUSOL, INFORM); -#endif - -/* ------------------------------------------------------------------ - Set inform for exit. - ------------------------------------------------------------------ */ -x900: - if(NRANK==NRANK0) - *INFORM = LUSOL_INFORM_LUSUCCESS; - else if(NRANKn) { - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu8rpc warning...\nSingularity after replacing column. jrep=%8d diag=%g\n", - JREP,DIAG); - } - } - else - *INFORM = LUSOL_INFORM_LUSINGULAR; - goto x990; -/* Instability. */ -x920: - *INFORM = LUSOL_INFORM_LUUNSTABLE; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu8rpc warning...\nInstability after replacing column. jrep=%8d diag=%g\n", - JREP,DIAG); - goto x990; -/* Not enough storage. */ -x970: - *INFORM = LUSOL_INFORM_ANEEDMEM; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu8rpc error...\nInsufficient memory. lena=%8d\n", - LUSOL->lena); - goto x990; -/* jrep is out of range. */ -x980: - *INFORM = LUSOL_INFORM_FATALERR; - if(LPRINT>=LUSOL_MSG_SINGULARITY) - LUSOL_report(LUSOL, 0, "lu8rpc error...\njrep is out of range. m=%8d n=%8d jrep=%8d\n", - LUSOL->m,LUSOL->n,JREP); -/* Exit. */ -x990: - LUSOL->luparm[LUSOL_IP_UPDATECOUNT]++; - LUSOL->luparm[LUSOL_IP_RANK_U] = NRANK; - LUSOL->luparm[LUSOL_IP_NONZEROS_L] = LENL; - LUSOL->luparm[LUSOL_IP_NONZEROS_U] = LENU; - LUSOL->luparm[LUSOL_IP_NONZEROS_ROW] = LROW; - LUSOL->luparm[LUSOL_IP_INFORM] = *INFORM; -} diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.c deleted file mode 100644 index 18f5f219..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.c +++ /dev/null @@ -1,27 +0,0 @@ - -#include "bfp_LUSOL.h" -#include "lp_lib.h" -#include "lp_LUSOL.h" - -BOOL APIENTRY DllMain( HANDLE hModule, - DWORD ul_reason_for_call, - LPVOID lpReserved - ) -{ - switch (ul_reason_for_call) - { - case DLL_PROCESS_ATTACH: - case DLL_THREAD_ATTACH: - case DLL_THREAD_DETACH: - case DLL_PROCESS_DETACH: - break; - } - return TRUE; -} - -#if defined FORTIFY -int EndOfPgr(int i) -{ - exit(i); -} -#endif diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.h b/code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.h deleted file mode 100644 index 517626e8..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/bfp_LUSOL.h +++ /dev/null @@ -1,15 +0,0 @@ -// stdafx.h : include file for standard system include files, -// or project specific include files that are used frequently, but -// are changed infrequently -// - -//#pragma once - -//RoleIsExternalInvEngine;INVERSE_ACTIVE=INVERSE_LUSOL - -#define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers -// Windows Header Files: -#include - -// TODO: reference additional headers your program requires here - diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.c b/code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.c deleted file mode 100644 index 25327063..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.c +++ /dev/null @@ -1,735 +0,0 @@ - -/* Modularized simplex basis factorization module - w/interface for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lusol.h, lp_lib.h, myblas.h - - Release notes: - v2.0.0 1 March 2004 First implementation of the LUSOL v2.0 C translation. - v2.0.1 1 April 2004 Added singularity recovery and fast/reuse update logic. - v2.0.2 23 May 2004 Moved mustrefact() function into the BFP structure. - v2.0.3 5 September 2004 Reworked pivot threshold tightening logic and default - values. - v2.1.0 18 June 2005 Made changes to allow for "pure" factorization; - i.e. without the objective function included. - - ---------------------------------------------------------------------------------- */ - -/* Generic include libraries */ -#include -#include -#include "lp_lib.h" - -/* Include libraries for this factorization system */ -#include "myblas.h" -#include "commonlib.h" -#include "lp_LUSOL.h" -#include "lusol.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* Include routines common to factorization engine implementations */ -#include "lp_BFP1.c" -#include "lp_BFP2.c" - - -/* MUST MODIFY */ -char * BFP_CALLMODEL bfp_name(void) -{ - return( "LUSOL v2.2.1.0" ); -} - - -/* MUST MODIFY */ -MYBOOL BFP_CALLMODEL bfp_resize(lprec *lp, int newsize) -{ - INVrec *lu; - - lu = lp->invB; - - /* Increment dimensionality since we put the objective row at the top */ - newsize = newsize + bfp_rowoffset(lp); - lu->dimalloc = newsize; - - /* Allocate index tracker arrays, LU matrices and various work vectors */ - if(!allocREAL(lp, &(lu->value), newsize+MATINDEXBASE, AUTOMATIC)) - return( FALSE ); - - /* Data specific to the factorization engine */ - if(lu->LUSOL != NULL) { - if(newsize > 0 || 1) - LUSOL_sizeto(lu->LUSOL, newsize, newsize, 0); - else { - LUSOL_free(lu->LUSOL); - lu->LUSOL = NULL; - } - } - else if(newsize > 0 || 1) { - int asize; - REAL bsize; - - lu->LUSOL = LUSOL_create(NULL, 0, LUSOL_PIVMOD_TPP, bfp_pivotmax(lp)*0); - -#if 1 - lu->LUSOL->luparm[LUSOL_IP_ACCELERATION] = LUSOL_AUTOORDER; - lu->LUSOL->parmlu[LUSOL_RP_SMARTRATIO] = 0.50; -#endif -#if 0 - lu->timed_refact = DEF_TIMEDREFACT; -#else - lu->timed_refact = FALSE; -#endif - - /* The following adjustments seem necessary to make the really tough NETLIB - models perform reliably and still performant (e.g. cycle.mps) */ -#if 0 - lu->LUSOL->parmlu[LUSOL_RP_SMALLDIAG_U] = - lu->LUSOL->parmlu[LUSOL_RP_EPSDIAG_U] = lp->epsprimal; -#endif -#if 0 - lu->LUSOL->parmlu[LUSOL_RP_ZEROTOLERANCE] = lp->epsvalue; -#endif - -#if 1 - LUSOL_setpivotmodel(lu->LUSOL, LUSOL_PIVMOD_NOCHANGE, LUSOL_PIVTOL_SLIM); -#else - LUSOL_setpivotmodel(lu->LUSOL, LUSOL_PIVMOD_NOCHANGE, LUSOL_PIVTOL_TIGHT); -#endif - -#ifdef LUSOL_UseBLAS -/* if(fileSearchPath("PATH", "myBLAS.DLL", NULL) && load_BLAS("myBLAS")) */ - if(is_nativeBLAS() && load_BLAS(libnameBLAS)) - lp->report(lp, NORMAL, "Optimized BLAS was successfully loaded for bfp_LUSOL.\n"); -#endif - - /* Try to minimize memory allocation if we have a large number of unit columns */ - bsize = (REAL) lp->get_nonzeros(lp); - if(newsize > lp->columns) - bsize += newsize; - else - bsize = bsize/lp->columns*newsize; - /* Add a "reasonable" delta to allow for B and associated factorizations - that are denser than average; this makes reallocations less frequent. - Values between 1.2 and 1.5 appear to be reasonable. */ - asize = (int) (bsize*MAX_DELTAFILLIN*1.3333); - if(!LUSOL_sizeto(lu->LUSOL, newsize, newsize, asize)) - return( FALSE ); - } - lu->dimcount = newsize; - return( TRUE ); -} - - -/* MUST MODIFY */ -void BFP_CALLMODEL bfp_free(lprec *lp) -{ - INVrec *lu; - - lu = lp->invB; - if(lu == NULL) - return; - - /* General arrays */ - FREE(lu->opts); - FREE(lu->value); - - /* Data specific to the factorization engine */ - LUSOL_free(lu->LUSOL); - - FREE(lu); - lp->invB = NULL; -} - - -/* MUST MODIFY */ -int BFP_CALLMODEL bfp_nonzeros(lprec *lp, MYBOOL maximum) -{ - INVrec *lu; - - lu = lp->invB; - if(maximum == TRUE) - return(lu->max_LUsize); - else if(maximum == AUTOMATIC) - return(lu->max_Bsize); - else - return(lu->LUSOL->luparm[LUSOL_IP_NONZEROS_L0]+lu->LUSOL->luparm[LUSOL_IP_NONZEROS_U0]); -/* return(lu->LUSOL->luparm[LUSOL_IP_NONZEROS_ROW]); */ -} - - -/* MUST MODIFY (or ignore) */ -int BFP_CALLMODEL bfp_memallocated(lprec *lp) -{ - int mem; - LUSOLrec *LUSOL = lp->invB->LUSOL; - - mem = sizeof(REAL) * (LUSOL->lena+LUSOL->maxm+LUSOL_RP_LASTITEM); - mem += sizeof(int) * (2*LUSOL->lena+5*LUSOL->maxm+5*LUSOL->maxn+LUSOL_IP_LASTITEM); - if(LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TCP) - mem += sizeof(REAL) * LUSOL->maxn + 2*sizeof(REAL)*LUSOL->maxn; - else if(LUSOL->luparm[LUSOL_IP_PIVOTTYPE] == LUSOL_PIVMOD_TRP) - mem += sizeof(REAL) * LUSOL->maxn; - if(!LUSOL->luparm[LUSOL_IP_KEEPLU]) - mem += sizeof(REAL) * LUSOL->maxn; - return( mem ); -} - - -/* MUST MODIFY */ -int BFP_CALLMODEL bfp_preparefactorization(lprec *lp) -{ - INVrec *lu = lp->invB; - - /* Finish any outstanding business */ - if(lu->is_dirty == AUTOMATIC) - lp->bfp_finishfactorization(lp); - - /* Clear or resize the existing LU matrices - specific for the factorization engine */ - LUSOL_clear(lu->LUSOL, TRUE); - if(lu->dimcount != lp->rows + bfp_rowoffset(lp)) - lp->bfp_resize(lp, lp->rows); - - /* Reset additional indicators */ - lp->bfp_updaterefactstats(lp); - lu->col_pos = 0; - - return(0); - -} - - -/* LOCAL HELPER ROUTINE - Replace a basis column with corresponding slack */ -int bfp_LUSOLsetcolumn(lprec *lp, int posnr, int colnr) -{ - int nz, inform; - - nz = lp->get_lpcolumn(lp, colnr, lp->invB->LUSOL->w + bfp_rowoffset(lp), NULL, NULL); - inform = LUSOL_replaceColumn(lp->invB->LUSOL, posnr, lp->invB->LUSOL->w); - return( inform ); -} - - -/* LOCAL HELPER ROUTINE - force the basis to be the identity matrix */ -int bfp_LUSOLidentity(lprec *lp, int *rownum) -{ - int i, nz; - INVrec *invB = lp->invB; - - /* Reset the factorization engine */ - LUSOL_clear(invB->LUSOL, TRUE); - - /* Add the basis columns */ - lp->invB->set_Bidentity = TRUE; - for(i = 1; i <= invB->dimcount; i++) { - nz = lp->get_basiscolumn(lp, i, rownum, invB->value); - LUSOL_loadColumn(invB->LUSOL, rownum, i, invB->value, nz, 0); - } - lp->invB->set_Bidentity = FALSE; - - /* Factorize */ - i = LUSOL_factorize(invB->LUSOL); - - return( i ); -} - - -/* LOCAL HELPER ROUTINE */ -int bfp_LUSOLfactorize(lprec *lp, MYBOOL *usedpos, int *rownum, int *singular) -{ - int i, j, nz, deltarows = bfp_rowoffset(lp); - INVrec *invB = lp->invB; - - /* Handle normal, presumed nonsingular case */ - if(singular == NULL) { - - /* Optionally do a symbolic minimum degree ordering; - not that slack variables should not be processed */ -/*#define UsePreprocessMDO*/ -#ifdef UsePreprocessMDO - int *mdo; - mdo = lp->bfp_createMDO(lp, usedpos, lp->rows, TRUE); - if(mdo != NULL) { - for(i = 1; i <= lp->rows; i++) - lp->set_basisvar(lp, i, mdo[i]); - FREE(mdo); - } -#endif - - /* Reset the factorization engine */ - LUSOL_clear(invB->LUSOL, TRUE); - - /* Add the basis columns in the original order */ - for(i = 1; i <= invB->dimcount; i++) { - nz = lp->get_basiscolumn(lp, i, rownum, invB->value); - LUSOL_loadColumn(invB->LUSOL, rownum, i, invB->value, nz, 0); - if((i > deltarows) && (lp->var_basic[i-deltarows] > lp->rows)) - lp->invB->user_colcount++; - } - - /* Factorize */ - i = LUSOL_factorize(invB->LUSOL); - } - - /* Handle case where a column may be singular */ - else { - LLrec *map; - - /* Reset the factorization engine */ - i = bfp_LUSOLidentity(lp, rownum); - - /* Build map of available columns */ - nz = createLink(lp->rows, &map, NULL); - for(i = 1; i <= lp->rows; i++) { - if(lp->var_basic[i] <= lp->rows) - removeLink(map, i); - } - - /* Rebuild the basis, column by column, while skipping slack columns */ - j = firstActiveLink(map); - for(i = 1; i <= lp->rows; i++) { - if(lp->var_basic[i] <= lp->rows) - continue; - nz = bfp_LUSOLsetcolumn(lp, j+deltarows, lp->var_basic[i]); - if(nz == LUSOL_INFORM_LUSUCCESS) - lp->invB->user_colcount++; - else { - nz = bfp_LUSOLsetcolumn(lp, j+deltarows, i); - lp->set_basisvar(lp, i, i); - } - j = nextActiveLink(map, j); - } - - /* Sort the basis list */ - MEMCOPY(rownum, lp->var_basic, lp->rows+1); - sortByINT(lp->var_basic, rownum, lp->rows, 1, TRUE); - - } - - return( i ); -} -/* LOCAL HELPER ROUTINE */ -void bfp_LUSOLtighten(lprec *lp) -{ - int infolevel = DETAILED; - - switch(LUSOL_tightenpivot(lp->invB->LUSOL)) { - case FALSE: lp->report(lp, infolevel, "bfp_factorize: Very hard numerics, but cannot tighten LUSOL thresholds further.\n"); - break; - case TRUE: lp->report(lp, infolevel, "bfp_factorize: Frequent refact pivot count %d at iter %.0f; tightened thresholds.\n", - lp->invB->num_pivots, (REAL) lp->get_total_iter(lp)); - break; - default: lp->report(lp, infolevel, "bfp_factorize: LUSOL switched to %s pivoting model to enhance stability.\n", - LUSOL_pivotLabel(lp->invB->LUSOL)); - } -} - -#define is_fixedvar is_fixedvar_ /* resolves a compiler warning/error conflict with lp_lib.h */ - -static MYBOOL is_fixedvar(lprec *lp, int variable) -{ - if((lp->bb_bounds != NULL && lp->bb_bounds->UBzerobased) || (variable <= lp->rows)) - return( (MYBOOL) (lp->upbo[variable] < lp->epsprimal) ); - else - return( (MYBOOL) (lp->upbo[variable]-lp->lowbo[variable] < lp->epsprimal) ); -} /* is_fixedvar */ - -/* MUST MODIFY */ -int BFP_CALLMODEL bfp_factorize(lprec *lp, int uservars, int Bsize, MYBOOL *usedpos, MYBOOL final) -{ - int kcol, inform, - *rownum = NULL, - singularities = 0, - dimsize = lp->invB->dimcount; - LUSOLrec *LUSOL = lp->invB->LUSOL; - - /* Set dimensions and create work array */ - SETMAX(lp->invB->max_Bsize, Bsize+(1+lp->rows-uservars)); - kcol = lp->invB->dimcount; - LUSOL->m = kcol; - LUSOL->n = kcol; - allocINT(lp, &rownum, kcol+1, FALSE); - - /* Check if the refactorization frequency is low; - tighten pivot thresholds if appropriate */ - inform = lp->bfp_pivotcount(lp); - if(!final && /* No solution update-based refactorization */ - !lp->invB->force_refact && /* No sparsity-based refactorization */ - !lp->is_action(lp->spx_action, - ACTION_TIMEDREINVERT) && /* No optimal time-based refactorization */ - (inform > 5) && (inform < 0.25*lp->bfp_pivotmax(lp))) - bfp_LUSOLtighten(lp); - - - /* Reload B and factorize */ - inform = bfp_LUSOLfactorize(lp, usedpos, rownum, NULL); - - /* Do some checks */ -#ifdef Paranoia - if(uservars != lp->invB->user_colcount) { - lp->report(lp, SEVERE, "bfp_factorize: User variable count reconciliation failed\n"); - return( singularities ); - } -#endif - - /* Check result and do further remedial action if necessary */ - if(inform != LUSOL_INFORM_LUSUCCESS) { - int singularcols, - replacedcols = 0; - REAL hold; - - /* Make sure we do not tighten factorization pivot criteria too often, and simply - accept the substitution of slack columns into the basis */ - if((lp->invB->num_singular+1) % TIGHTENAFTER == 0) - bfp_LUSOLtighten(lp); - - /* Try to restore a non-singular basis by substituting singular columns with slacks */ - while((inform == LUSOL_INFORM_LUSINGULAR) && (replacedcols < dimsize)) { - int iLeave, jLeave, iEnter; - MYBOOL isfixed; - - singularities++; - singularcols = LUSOL->luparm[LUSOL_IP_SINGULARITIES]; - hold = (REAL) lp->get_total_iter(lp); - lp->report(lp, NORMAL, "bfp_factorize: Resolving %d singularit%s at refact %d, iter %.0f\n", - singularcols, my_plural_y(singularcols), lp->invB->num_refact, hold); - - /* Find the failing / singular column(s) and make slack substitutions */ - for(kcol = 1; kcol <= singularcols; kcol++) { - - /* Determine leaving and entering columns. */ - iLeave = LUSOL_getSingularity(LUSOL, kcol); /* This is the singular column as natural index */ - iEnter = iLeave; /* This is the target replacement slack */ -#if 1 - iEnter = LUSOL->iqinv[iEnter]; - iEnter = LUSOL->ip[iEnter]; -#endif - iLeave-= bfp_rowextra(lp); /* This is the original B column/basis index */ - jLeave = lp->var_basic[iLeave]; /* This is the IA column index in lp_solve */ - - /* Express the slack index in original lp_solve [1..rows] reference and check validity */ - /* if(B4 != NULL) iEnter = B4->B4_row[iEnter]; v6 FUNCTIONALITY */ - iEnter -= bfp_rowextra(lp); - if(lp->is_basic[iEnter]) { - lp->report(lp, DETAILED, "bfp_factorize: Replacement slack %d is already basic!\n", iEnter); - - /* See if we can find a good alternative slack variable to enter */ - iEnter = 0; - for(inform = 1; inform <= lp->rows; inform++) - if(!lp->is_basic[inform]) { - if((iEnter == 0) || (lp->upbo[inform] > lp->upbo[iEnter])) { - iEnter = inform; - if(my_infinite(lp, lp->upbo[iEnter])) - break; - } - } - if(iEnter == 0) { - lp->report(lp, SEVERE, "bfp_factorize: Could not find replacement slack variable!\n"); - break; - } - } - - /* We should update bound states for both the entering and leaving variables. - Note that this may cause (primal or dual) infeasibility, but I assume that - lp_solve traps this and takes necessary corrective action. */ - isfixed = is_fixedvar(lp, iEnter); - if(isfixed) - lp->fixedvars++; - hold = lp->upbo[jLeave]; - lp->is_lower[jLeave] = isfixed || (fabs(hold)>=lp->infinite) || (lp->rhs[iLeave] < hold); - lp->is_lower[iEnter] = TRUE; - - /* Do the basis replacement */ - lp->set_basisvar(lp, iLeave, iEnter); - - } - - /* Refactorize with slack substitutions */ - inform = bfp_LUSOLfactorize(lp, NULL, rownum, NULL); - replacedcols += singularcols; - } - - /* Check if we had a fundamental problem */ - if(singularities >= dimsize) { - lp->report(lp, IMPORTANT, "bfp_factorize: LUSOL was unable to recover from a singular basis\n"); - lp->spx_status = NUMFAILURE; - } - } - - /* Clean up before returning */ - FREE(rownum); - - /* Update statistics */ - /* SETMAX(lp->invB->max_Bsize, (*Bsize)); */ - lp->invB->num_singular += singularities; /* The total number of singular updates */ - - return( singularities ); -} - -/* MUST MODIFY */ -MYBOOL BFP_CALLMODEL bfp_finishupdate(lprec *lp, MYBOOL changesign) -/* Was addetacol() in versions of lp_solve before 4.0.1.8 - KE */ -{ - int i, k, kcol, deltarows = bfp_rowoffset(lp); - REAL DIAG, VNORM; - INVrec *lu = lp->invB; - LUSOLrec *LUSOL = lu->LUSOL; - - if(!lu->is_dirty) - return( FALSE ); - if(lu->is_dirty != AUTOMATIC) - lu->is_dirty = FALSE; - - /* Perform the update */ - k = lu->col_pos+deltarows; - lu->num_pivots++; - if(lu->col_leave > lu->dimcount-deltarows) - lu->user_colcount--; - if(lu->col_enter > lu->dimcount-deltarows) - lu->user_colcount++; - kcol = lu->col_pos; - lu->col_pos = 0; - - /* Do standard update */ -#ifdef LUSOLSafeFastUpdate /* NB! Defined in lusol.h */ - if(TRUE || !changesign) { - if(changesign) { - REAL *temp = LUSOL->vLU6L; - for(i = 1, temp++; i <= lp->rows+deltarows; i++, temp++) - if(*temp != 0) - *temp = -(*temp); - } - /* Execute the update using data prepared earlier */ - LU8RPC(LUSOL, LUSOL_UPDATE_OLDNONEMPTY, LUSOL_UPDATE_USEPREPARED, - k, NULL, NULL, &i, &DIAG, &VNORM); - } - else -#endif - { - /* Retrieve the data for the entering column (base 0) */ - i = lp->get_lpcolumn(lp, lu->col_enter, lu->value+deltarows, NULL, NULL); - lu->value[0] = 0; - /* Execute the update */ - LU8RPC(LUSOL, LUSOL_UPDATE_OLDNONEMPTY, LUSOL_UPDATE_NEWNONEMPTY, - k, lu->value, NULL, &i, &DIAG, &VNORM); - } - - if(i == LUSOL_INFORM_LUSUCCESS) { - - /* Check if we should refactorize based on accumulation of fill-in */ - DIAG = LUSOL->luparm[LUSOL_IP_NONZEROS_L]+LUSOL->luparm[LUSOL_IP_NONZEROS_U]; - VNORM = LUSOL->luparm[LUSOL_IP_NONZEROS_L0]+LUSOL->luparm[LUSOL_IP_NONZEROS_U0]; -#if 0 - /* This is Michael Saunder's fixed parameter */ - VNORM *= MAX_DELTAFILLIN; -#else - /* This is Kjell Eikland's dynamic error accumulation measure */ - VNORM *= pow(MAX_DELTAFILLIN, pow((0.5*LUSOL->nelem/VNORM), 0.25)); -#endif - lu->force_refact = (MYBOOL) ((DIAG > VNORM) && (lu->num_pivots > 20)); - -#if 0 - /* Additional KE logic to reduce maximum pivot count based on the density of B */ - if(!lu->force_refact) { - VNORM = lp->rows+1; - VNORM = 1.0 - pow((REAL) LUSOL->nelem/VNORM/VNORM, 0.2); - lu->force_refact = (MYBOOL) (lu->num_pivots > VNORM*lp->bfp_pivotmax(lp)); - } -#endif - } - - /* Handle errors */ - else { -/* int infolevel = NORMAL; */ - int infolevel = DETAILED; - lp->report(lp, infolevel, "bfp_finishupdate: Failed at iter %.0f, pivot %d;\n%s\n", - (REAL) (lp->total_iter+lp->current_iter), lu->num_pivots, LUSOL_informstr(LUSOL, i)); - if(i == LUSOL_INFORM_ANEEDMEM) { /* To compress used memory and realloc, if necessary */ - lp->invert(lp, INITSOL_USEZERO, FALSE); - if(i != LUSOL_INFORM_LUSUCCESS) - lp->report(lp, NORMAL, "bfp_finishupdate: Insufficient memory at iter %.0f;\n%s\n", - (REAL) (lp->total_iter+lp->current_iter), LUSOL_informstr(LUSOL, i)); - } - else if(i == LUSOL_INFORM_RANKLOSS) { /* To fix rank loss and clear cumulative errors */ -#if 0 - /* This is test code to do pivot in slack BEFORE refactorization (pessimistic approach); - assumes that LUSOL returns correct information about the source of the singularity */ - kcol = LUSOL->luparm[LUSOL_IP_SINGULARINDEX]; -#ifdef MAPSINGULARCOLUMN - kcol = LUSOL_findColumnPosition(LUSOL, kcol); -#endif - lp->set_basisvar(lp, kcol-deltarows, kcol-deltarows); -#endif - lp->invert(lp, INITSOL_USEZERO, FALSE); - i = LUSOL->luparm[LUSOL_IP_INFORM]; - if(i != LUSOL_INFORM_LUSUCCESS) - lp->report(lp, NORMAL, "bfp_finishupdate: Recovery attempt unsuccessful at iter %.0f;\n%s\n", - (REAL) (lp->total_iter+lp->current_iter), LUSOL_informstr(LUSOL, i)); - else - lp->report(lp, infolevel, "bfp_finishupdate: Correction or recovery was successful.\n"); - } - } - return( (MYBOOL) (i == LUSOL_INFORM_LUSUCCESS) ); - -} /* bfp_finishupdate */ - - -/* MUST MODIFY */ -void BFP_CALLMODEL bfp_ftran_normal(lprec *lp, REAL *pcol, int *nzidx) -{ - int i; - INVrec *lu; - - lu = lp->invB; - - /* Do the LUSOL ftran */ - i = LUSOL_ftran(lu->LUSOL, pcol-bfp_rowoffset(lp), nzidx, FALSE); - if(i != LUSOL_INFORM_LUSUCCESS) { - lu->status = BFP_STATUS_ERROR; - lp->report(lp, NORMAL, "bfp_ftran_normal: Failed at iter %.0f, pivot %d;\n%s\n", - (REAL) (lp->total_iter+lp->current_iter), lu->num_pivots, LUSOL_informstr(lu->LUSOL, i)); - } -} - - -/* MAY MODIFY */ -void BFP_CALLMODEL bfp_ftran_prepare(lprec *lp, REAL *pcol, int *nzidx) -{ - int i; - INVrec *lu; - - lu = lp->invB; - - /* Do the LUSOL ftran */ - i = LUSOL_ftran(lu->LUSOL, pcol-bfp_rowoffset(lp), nzidx, TRUE); - if(i != LUSOL_INFORM_LUSUCCESS) { - lu->status = BFP_STATUS_ERROR; - lp->report(lp, NORMAL, "bfp_ftran_prepare: Failed at iter %.0f, pivot %d;\n%s\n", - (REAL) (lp->total_iter+lp->current_iter), lu->num_pivots, LUSOL_informstr(lu->LUSOL, i)); - } -} - - -/* MUST MODIFY */ -void BFP_CALLMODEL bfp_btran_normal(lprec *lp, REAL *prow, int *nzidx) -{ - int i; - INVrec *lu; - - lu = lp->invB; - - /* Do the LUSOL btran */ - i = LUSOL_btran(lu->LUSOL, prow-bfp_rowoffset(lp), nzidx); - if(i != LUSOL_INFORM_LUSUCCESS) { - lu->status = BFP_STATUS_ERROR; - lp->report(lp, NORMAL, "bfp_btran_normal: Failed at iter %.0f, pivot %d;\n%s\n", - (REAL) (lp->total_iter+lp->current_iter), lu->num_pivots, LUSOL_informstr(lu->LUSOL, i)); - } - - /* Check performance data */ -#if 0 - if(lu->num_pivots == 1) { - if(lu->LUSOL->luparm[LUSOL_IP_ACCELERATION] > 0) - lp->report(lp, NORMAL, "RowL0 R:%10.7f C:%10.7f NZ:%10.7f\n", - (REAL) lu->LUSOL->luparm[LUSOL_IP_ROWCOUNT_L0] / lu->LUSOL->m, - (REAL) lu->LUSOL->luparm[LUSOL_IP_COLCOUNT_L0] / lu->LUSOL->m, - (REAL) lu->LUSOL->luparm[LUSOL_IP_NONZEROS_L0] / pow((REAL) lu->LUSOL->m, 2)); - else - lp->report(lp, NORMAL, "ColL0 C:%10.7f NZ:%10.7f\n", - (REAL) lu->LUSOL->luparm[LUSOL_IP_COLCOUNT_L0] / lu->LUSOL->m, - (REAL) lu->LUSOL->luparm[LUSOL_IP_NONZEROS_L0] / pow((REAL) lu->LUSOL->m, 2)); - } -#endif - -} - -/* MUST MODIFY - Routine to find maximum rank of equality constraints */ -int BFP_CALLMODEL bfp_findredundant(lprec *lp, int items, getcolumnex_func cb, int *maprow, int *mapcol) -{ - int i, j, nz = 0, m = 0, n = 0, *nzrows = NULL; - REAL *nzvalues = NULL, *arraymax = NULL; - LUSOLrec *LUSOL; - - /* Are we capable of finding redundancy with this BFP? */ - if((maprow == NULL) && (mapcol == NULL)) - return( n ); - - /* If so, initialize memory structures */ - if(!allocINT(lp, &nzrows, items, FALSE) || - !allocREAL(lp, &nzvalues, items, FALSE)) - return( n ); - - /* Compute the number of non-empty columns */ - m = 0; - for(j = 1; j <= mapcol[0]; j++) { - n = cb(lp, mapcol[j], NULL, NULL, maprow); - if(n > 0) { - m++; - mapcol[m] = mapcol[j]; - nz += n; - } - } - mapcol[0] = m; - - /* Instantiate a LUSOL object */ - LUSOL = LUSOL_create(NULL, 0, LUSOL_PIVMOD_TRP, 0); - if((LUSOL == NULL) || !LUSOL_sizeto(LUSOL, items, m, nz*LUSOL_MULT_nz_a)) - goto Finish; - - /* Modify relevant LUSOL parameters */ - LUSOL->m = items; - LUSOL->n = m; -#if 0 - LUSOL->luparm[LUSOL_IP_KEEPLU] = FALSE; - LUSOL->luparm[LUSOL_IP_PIVOTTYPE] = LUSOL_PIVMOD_TRP; - LUSOL->parmlu[LUSOL_RP_FACTORMAX_Lij] = 2.0; -#endif - - /* Load the columns into LUSOL */ - for(j = 1; j <= m; j++) { - n = cb(lp, mapcol[j], nzvalues, nzrows, maprow); - i = LUSOL_loadColumn(LUSOL, nzrows, j, nzvalues, n, -1); - if(n != i) { - lp->report(lp, IMPORTANT, "bfp_findredundant: Error %d while loading column %d with %d nz\n", - i, j, n); - n = 0; - goto Finish; - } - } - - /* Scale rows to prevent numerical problems */ - if((lp->scalemode != SCALE_NONE) && allocREAL(lp, &arraymax, items+1, TRUE)) { - for(i = 1; i <= nz; i++) { - SETMAX(arraymax[LUSOL->indc[i]], fabs(LUSOL->a[i])); - } - for(i = 1; i <= nz; i++) - LUSOL->a[i] /= arraymax[LUSOL->indc[i]]; - FREE(arraymax); - } - - /* Factorize for maximum rank */ - n = 0; - i = LUSOL_factorize(LUSOL); - /* lp->report(lp, NORMAL, "bfp_findredundant: r=%d c=%d - %s\n", items, m, LUSOL_informstr(LUSOL, i));*/ - if((i == LUSOL_INFORM_LUSUCCESS) || (i != LUSOL_INFORM_LUSINGULAR)) - goto Finish; - - /* We have a singular matrix, obtain the indeces of the singular rows */ - for(i = LUSOL->luparm[LUSOL_IP_RANK_U] + 1; i <= items; i++) { - n++; - maprow[n] = LUSOL->ip[i]; - } - maprow[0] = n; - - /* Clean up */ -Finish: - LUSOL_free(LUSOL); - FREE(nzrows); - FREE(nzvalues); - - return( n ); -} diff --git a/code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.h b/code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.h deleted file mode 100644 index 6bf4157e..00000000 --- a/code/3rd_lpsolve/bfp/bfp_LUSOL/lp_LUSOL.h +++ /dev/null @@ -1,63 +0,0 @@ -#ifndef HEADER_lp_LUSOL -#define HEADER_lp_LUSOL - -/* Include libraries for this inverse system */ -#include "lp_types.h" -#include "lusol.h" - -/* LUSOL defines */ -#ifdef WIN32 -# define LUSOL_UseBLAS -#endif -/*#define MAPSINGULARCOLUMN*/ -#define MATINDEXBASE LUSOL_ARRAYOFFSET /* Inversion engine index start for arrays */ -#define LU_START_SIZE 10000 /* Start size of LU; realloc'ed if needed */ -#define DEF_MAXPIVOT 250 /* Maximum number of pivots before refactorization */ -#define MAX_DELTAFILLIN 2.0 /* Do refactorizations based on sparsity considerations */ -#define TIGHTENAFTER 10 /* Tighten LU pivot criteria only after this number of singularities */ - -/* typedef */ struct _INVrec -{ - int status; /* Last operation status code */ - int dimcount; /* The actual number of LU rows/columns */ - int dimalloc; /* The allocated LU rows/columns size */ - int user_colcount; /* The number of user LU columns */ - LUSOLrec *LUSOL; - int col_enter; /* The full index of the entering column */ - int col_leave; /* The full index of the leaving column */ - int col_pos; /* The B column to be changed at the next update using data in value[.]*/ - REAL *value; - REAL *pcol; /* Reference to the elimination vector */ - REAL theta_enter; /* Value of the entering column theta */ - - int max_Bsize; /* The largest B matrix of user variables */ - int max_colcount; /* The maximum number of user columns in LU */ - int max_LUsize; /* The largest NZ-count of LU-files generated */ - int num_refact; /* Number of times the basis was refactored */ - int num_timed_refact; - int num_dense_refact; - double time_refactstart; /* Time since start of last refactorization-pivots cyle */ - double time_refactnext; /* Time estimated to next refactorization */ - int num_pivots; /* Number of pivots since last refactorization */ - int num_singular; /* The total number of singular updates */ - char *opts; - MYBOOL is_dirty; /* Specifies if a column is incompletely processed */ - MYBOOL force_refact; /* Force refactorization at the next opportunity */ - MYBOOL timed_refact; /* Set if timer-driven refactorization should be active */ - MYBOOL set_Bidentity; /* Force B to be the identity matrix at the next refactorization */ -} /* INVrec */; - - -#ifdef __cplusplus -/* namespace LUSOL */ -extern "C" { -#endif - -/* Put function headers here */ -#include "lp_BFP.h" - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_LUSOL */ diff --git a/code/3rd_lpsolve/bfp/lp_BFP.h b/code/3rd_lpsolve/bfp/lp_BFP.h deleted file mode 100644 index 36e1fbb9..00000000 --- a/code/3rd_lpsolve/bfp/lp_BFP.h +++ /dev/null @@ -1,82 +0,0 @@ - -/* ---------------------------------------------------------------------------------- */ -/* lp_solve v5+ headers for basis inversion / factorization libraries */ -/* ---------------------------------------------------------------------------------- */ -#define BFP_STATUS_RANKLOSS -1 -#define BFP_STATUS_SUCCESS 0 -#define BFP_STATUS_SINGULAR 1 -#define BFP_STATUS_UNSTABLE 2 -#define BFP_STATUS_NOPIVOT 3 -#define BFP_STATUS_DIMERROR 4 -#define BFP_STATUS_DUPLICATE 5 -#define BFP_STATUS_NOMEMORY 6 -#define BFP_STATUS_ERROR 7 /* Unspecified, command-related error */ -#define BFP_STATUS_FATAL 8 - -#define BFP_STAT_ERROR -1 -#define BFP_STAT_REFACT_TOTAL 0 -#define BFP_STAT_REFACT_TIMED 1 -#define BFP_STAT_REFACT_DENSE 2 - -#ifndef BFP_CALLMODEL - #ifdef WIN32 - #define BFP_CALLMODEL __stdcall /* "Standard" call model */ - #else - #define BFP_CALLMODEL - #endif -#endif - -#ifdef RoleIsExternalInvEngine - #define __BFP_EXPORT_TYPE __EXPORT_TYPE -#else - #define __BFP_EXPORT_TYPE -#endif - - -/* Routines with UNIQUE implementations for each inversion engine */ -/* ---------------------------------------------------------------------------------- */ -char __BFP_EXPORT_TYPE *(BFP_CALLMODEL bfp_name)(void); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_free)(lprec *lp); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_resize)(lprec *lp, int newsize); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_nonzeros)(lprec *lp, MYBOOL maximum); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_memallocated)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_preparefactorization)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_factorize)(lprec *lp, int uservars, int Bsize, MYBOOL *usedpos, MYBOOL final); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_finishupdate)(lprec *lp, MYBOOL changesign); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_ftran_normal)(lprec *lp, REAL *pcol, int *nzidx); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_ftran_prepare)(lprec *lp, REAL *pcol, int *nzidx); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_btran_normal)(lprec *lp, REAL *prow, int *nzidx); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_status)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_findredundant)(lprec *lp, int items, getcolumnex_func cb, int *maprow, int*mapcol); - - -/* Routines SHARED for all inverse implementations; located in lp_BFP1.c */ -/* ---------------------------------------------------------------------------------- */ -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_compatible)(lprec *lp, int bfpversion, int lpversion, int sizeofvar); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_indexbase)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_rowoffset)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_pivotmax)(lprec *lp); -REAL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_efficiency)(lprec *lp); -REAL __BFP_EXPORT_TYPE *(BFP_CALLMODEL bfp_pivotvector)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_pivotcount)(lprec *lp); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_mustrefactorize)(lprec *lp); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_refactcount)(lprec *lp, int kind); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_isSetI)(lprec *lp); -int *bfp_createMDO(lprec *lp, MYBOOL *usedpos, int count, MYBOOL doMDO); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_updaterefactstats)(lprec *lp); -int BFP_CALLMODEL bfp_rowextra(lprec *lp); - -/* Routines with OPTIONAL SHARED code; template routines suitable for canned */ -/* inverse engines are located in lp_BFP2.c */ -/* ---------------------------------------------------------------------------------- */ -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_init)(lprec *lp, int size, int deltasize, char *options); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_restart)(lprec *lp); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_implicitslack)(lprec *lp); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_pivotalloc)(lprec *lp, int newsize); -int __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_colcount)(lprec *lp); -MYBOOL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_canresetbasis)(lprec *lp); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_finishfactorization)(lprec *lp); -LREAL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_prepareupdate)(lprec *lp, int row_nr, int col_nr, REAL *pcol); -REAL __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_pivotRHS)(lprec *lp, LREAL theta, REAL *pcol); -void __BFP_EXPORT_TYPE (BFP_CALLMODEL bfp_btran_double)(lprec *lp, REAL *prow, int *pnzidx, REAL *drow, int *dnzidx); - diff --git a/code/3rd_lpsolve/bfp/lp_BFP1.c b/code/3rd_lpsolve/bfp/lp_BFP1.c deleted file mode 100644 index 663336c7..00000000 --- a/code/3rd_lpsolve/bfp/lp_BFP1.c +++ /dev/null @@ -1,206 +0,0 @@ - -/* Routines located in lp_BFP1.cpp; common for all factorization engines */ -/* Cfr. lp_BFP.h for definitions */ -/* ---------------------------------------------------------------------------------- */ -/* Changes: */ -/* 29 May 2004 Corrected calculation of bfp_efficiency(), which required */ -/* modifying the max_Bsize to include slack variables. KE. */ -/* 16 June 2004 Make the symbolic minimum degree ordering routine available */ -/* to BFPs as a routine internal to the library. KE */ -/* 1 July 2004 Change due to change in MDO naming. */ -/* ---------------------------------------------------------------------------------- */ - - -/* MUST MODIFY */ -MYBOOL BFP_CALLMODEL bfp_compatible(lprec *lp, int bfpversion, int lpversion, int sizeofvar) -{ - MYBOOL status = FALSE; - - if((lp != NULL) && (bfpversion == BFPVERSION) && (sizeof(REAL) == sizeofvar)) { -#if 0 - if(lpversion == MAJORVERSION) /* Forces BFP renewal at lp_solve major version changes */ -#endif - status = TRUE; - } - return( status ); -} - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_status(lprec *lp) -{ - return(lp->invB->status); -} - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_indexbase(lprec *lp) -{ - return( MATINDEXBASE ); -} - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_rowoffset(lprec *lp) -{ - if(lp->obj_in_basis) - return( 1 ); - else - return( 0 ); -} - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_pivotmax(lprec *lp) -{ - if(lp->max_pivots > 0) - return( lp->max_pivots ); - else - return( DEF_MAXPIVOT ); -} - -/* DON'T MODIFY */ -REAL * BFP_CALLMODEL bfp_pivotvector(lprec *lp) -{ - return( lp->invB->pcol ); -} - -/* DON'T MODIFY */ -REAL BFP_CALLMODEL bfp_efficiency(lprec *lp) -{ - REAL hold; - - hold = lp->bfp_nonzeros(lp, AUTOMATIC); - if(hold == 0) - hold = 1 + lp->rows; - hold = lp->bfp_nonzeros(lp, TRUE)/hold; - - return(hold); -} - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_pivotcount(lprec *lp) -{ - return(lp->invB->num_pivots); -} - - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_refactcount(lprec *lp, int kind) -{ - if(kind == BFP_STAT_REFACT_TOTAL) - return(lp->invB->num_refact); - else if(kind == BFP_STAT_REFACT_TIMED) - return(lp->invB->num_timed_refact); - else if(kind == BFP_STAT_REFACT_DENSE) - return(lp->invB->num_dense_refact); - else - return( BFP_STAT_ERROR ); -} - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_mustrefactorize(lprec *lp) -{ - MYBOOL test = lp->is_action(lp->spx_action, ACTION_REINVERT | ACTION_TIMEDREINVERT); - if(!test) { - REAL f; - INVrec *lu = lp->invB; - - if(lu->num_pivots > 0) - f = (timeNow()-lu->time_refactstart) / (REAL) lu->num_pivots; - else - f = 0; - - /* Always refactorize if we are above the set pivot limit */ - if(lu->force_refact || - (lu->num_pivots >= lp->bfp_pivotmax(lp))) - lp->set_action(&lp->spx_action, ACTION_REINVERT); - - /* Check if we should do an optimal time-based refactorization */ - else if(lu->timed_refact && (lu->num_pivots > 1) && - (f > MIN_TIMEPIVOT) && (f > lu->time_refactnext)) { - /* If we have excessive time usage in automatic mode then - treat as untimed case and update optimal time metric, ... */ - if((lu->timed_refact == AUTOMATIC) && - (lu->num_pivots < 0.4*lp->bfp_pivotmax(lp))) - lu->time_refactnext = f; - /* ... otherwise set flag for the optimal time-based refactorization */ - else - lp->set_action(&lp->spx_action, ACTION_TIMEDREINVERT); - } - - /* Otherwise simply update the optimal time metric */ - else - lu->time_refactnext = f; -#if 0 - if(lu->num_pivots % 10 == 0) - lp->report(lp, NORMAL, "bfp pivot %d - start %f - timestat %f", - lu->num_pivots, lu->time_refactstart, f); -#endif - } - - test = lp->is_action(lp->spx_action, ACTION_REINVERT | ACTION_TIMEDREINVERT); - return(test); -} - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_isSetI(lprec *lp) -{ - return( (MYBOOL) lp->invB->set_Bidentity ); -} - -/* DON'T MODIFY */ -int *bfp_createMDO(lprec *lp, MYBOOL *usedpos, int count, MYBOOL doMDO) -{ - int *mdo, i, j, kk; - - mdo = (int *) malloc((count + 1)*sizeof(*mdo)); -/* allocINT(lp, &mdo, count + 1, FALSE); */ - - /* Fill the mdo[] array with remaining full-pivot basic user variables */ - kk = 0; - for(j = 1; j <= lp->columns; j++) { - i = lp->rows + j; - if(usedpos[i] == TRUE) { - kk++; - mdo[kk] = i; - } - } - mdo[0] = kk; - if(kk == 0) - goto Process; - - /* Calculate the approximate minimum degree column ordering */ - if(doMDO) { - i = lp->getMDO(lp, usedpos, mdo, NULL, FALSE); - if(i != 0) { - lp->report(lp, CRITICAL, "bfp_createMDO: Internal error %d in minimum degree ordering routine", i); - FREE(mdo); - } - } -Process: - return( mdo ); -} -void BFP_CALLMODEL bfp_updaterefactstats(lprec *lp) -{ - INVrec *lu = lp->invB; - - /* Signal that we are refactorizing */ - lu->is_dirty = AUTOMATIC; - - /* Set time of start of current refactorization cycle */ - lu->time_refactstart = timeNow(); - lu->time_refactnext = 0; - lu->user_colcount = 0; - - /* Do the numbers */ - if(lu->force_refact) - lu->num_dense_refact++; - else if(lu->timed_refact && lp->is_action(lp->spx_action, ACTION_TIMEDREINVERT)) - lu->num_timed_refact++; - lu->num_refact++; -} - -int BFP_CALLMODEL bfp_rowextra(lprec *lp) -{ - if(lp->is_obj_in_basis(lp)) - return( 1 ); - else - return( 0 ); -} diff --git a/code/3rd_lpsolve/bfp/lp_BFP2.c b/code/3rd_lpsolve/bfp/lp_BFP2.c deleted file mode 100644 index f5232fa9..00000000 --- a/code/3rd_lpsolve/bfp/lp_BFP2.c +++ /dev/null @@ -1,177 +0,0 @@ - - -/* Routines located in lp_BFP2.cpp; optional shared for canned implementations */ -/* Cfr. lp_BFP.h for definitions */ -/* ---------------------------------------------------------------------------------- */ - - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_init(lprec *lp, int size, int delta, char *options) -{ - INVrec *lu; - - lp->invB = (INVrec *) calloc(1, sizeof(*(lp->invB))); - lu = lp->invB; - if((lu == NULL) || - !lp->bfp_resize(lp, size) || - !lp->bfp_restart(lp)) - return( FALSE ); - - /* Store any passed options */ - if(options != NULL) { - size_t len = strlen(options); - lu->opts = (char *) malloc(len + 1); - strcpy(lu->opts, options); - } - - /* Prepare for factorization and undo values reset by bfp_preparefactorization */ - lp->bfp_preparefactorization(lp); - lu->num_refact = 0; - - return( TRUE ); -} - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_restart(lprec *lp) -{ - INVrec *lu; - - lu = lp->invB; - if(lu == NULL) - return( FALSE ); - - lu->status = BFP_STATUS_SUCCESS; - lu->max_Bsize = 0; /* The largest NZ-count of the B matrix */ - lu->max_colcount = 0; /* The maximum number of user columns in B */ - lu->max_LUsize = 0; /* The largest NZ-count of LU-files generated */ - lu->num_refact = 0; /* The number of times the basis has been factored */ - lu->num_timed_refact = 0; - lu->num_dense_refact = 0; - lu->num_pivots = 0; /* The number of pivots since last factorization */ - lu->pcol = NULL; - lu->set_Bidentity = FALSE; - - return( TRUE ); -} - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_implicitslack(lprec *lp) -{ - return( FALSE ); -} - -/* DON'T MODIFY */ -int BFP_CALLMODEL bfp_colcount(lprec *lp) -{ - return(lp->invB->user_colcount); -} - - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_canresetbasis(lprec *lp) -{ - return( FALSE ); -} - - -/* DON'T MODIFY */ -MYBOOL BFP_CALLMODEL bfp_pivotalloc(lprec *lp, int newsize) -{ - /* Does nothing in the default implementation */ - return( TRUE ); -} - - -/* DON'T MODIFY */ -void BFP_CALLMODEL bfp_finishfactorization(lprec *lp) -{ - INVrec *lu; - - lu = lp->invB; - - SETMAX(lu->max_colcount, lp->bfp_colcount(lp)); - SETMAX(lu->max_LUsize, lp->bfp_nonzeros(lp, FALSE)); - - /* Signal that we done factorizing/reinverting */ - lu->is_dirty = FALSE; - lp->clear_action(&lp->spx_action, ACTION_REINVERT | ACTION_TIMEDREINVERT); - lu->force_refact = FALSE; - - /* Store information about the current inverse */ - lu->num_pivots = 0; - -} - - -/* DON'T MODIFY */ -LREAL BFP_CALLMODEL bfp_prepareupdate(lprec *lp, int row_nr, int col_nr, REAL *pcol) -/* Was condensecol() in versions of lp_solve before 4.0.1.8 - KE */ -{ - LREAL pivValue; - INVrec *lu; - - lu = lp->invB; - - /* Store the incoming pivot value for RHS update purposes */ - lu->col_enter = col_nr; /* The index of the new data column */ - lu->col_pos = row_nr; /* The basis column to be replaced */ - lu->col_leave = lp->var_basic[row_nr]; - if(pcol == NULL) - pivValue = 0; - else - pivValue = pcol[row_nr]; - lu->theta_enter = pivValue; - - /* Save reference to the elimination vector */ - lu->pcol = pcol; - - /* Set completion status; but hold if we are reinverting */ - if(lu->is_dirty != AUTOMATIC) - lu->is_dirty = TRUE; - - return( pivValue ); -} - - -/* DON'T MODIFY */ -REAL BFP_CALLMODEL bfp_pivotRHS(lprec *lp, LREAL theta, REAL *pcol) -/* This function is used to adjust the RHS in bound swap operations as - well as handling the updating of the RHS for normal basis changes. - Was rhsmincol(), ie. "rhs minus column" in versions of lp_solve before 4.0.1.8 - KE */ -{ - INVrec *lu; - - lu = lp->invB; - - if(pcol == NULL) - pcol = lu->pcol; - - if(theta != 0) { - register int i, n = lp->rows; - register LREAL roundzero = lp->epsvalue; - register LREAL *rhs = lp->rhs, rhsmax = 0; - - for(i = 0; i <= n; i++, rhs++, pcol++) { - (*rhs) -= theta * (*pcol); - my_roundzero(*rhs, roundzero); - SETMAX(rhsmax, fabs(*rhs)); - } - lp->rhsmax = rhsmax; - } - - if(pcol == lu->pcol) - return( lu->theta_enter ); - else - return( 0.0 ); -} - - -/* DON'T MODIFY */ -void BFP_CALLMODEL bfp_btran_double(lprec *lp, REAL *prow, int *pnzidx, REAL *drow, int *dnzidx) -{ - if(prow != NULL) - lp->bfp_btran_normal(lp, prow, pnzidx); - if(drow != NULL) - lp->bfp_btran_normal(lp, drow, dnzidx); -} - diff --git a/code/3rd_lpsolve/colamd/colamd.c b/code/3rd_lpsolve/colamd/colamd.c deleted file mode 100644 index f48c6f56..00000000 --- a/code/3rd_lpsolve/colamd/colamd.c +++ /dev/null @@ -1,3469 +0,0 @@ -/* ========================================================================== */ -/* === colamd/symamd - a sparse matrix column ordering algorithm ============ */ -/* ========================================================================== */ - -/* - colamd: an approximate minimum degree column ordering algorithm, - for LU factorization of symmetric or unsymmetric matrices, - QR factorization, least squares, interior point methods for - linear programming problems, and other related problems. - - symamd: an approximate minimum degree ordering algorithm for Cholesky - factorization of symmetric matrices. - - Purpose: - - Colamd computes a permutation Q such that the Cholesky factorization of - (AQ)'(AQ) has less fill-in and requires fewer floating point operations - than A'A. This also provides a good ordering for sparse partial - pivoting methods, P(AQ) = LU, where Q is computed prior to numerical - factorization, and P is computed during numerical factorization via - conventional partial pivoting with row interchanges. Colamd is the - column ordering method used in SuperLU, part of the ScaLAPACK library. - It is also available as built-in function in Matlab Version 6, - available from MathWorks, Inc. (http://www.mathworks.com). This - routine can be used in place of colmmd in Matlab. By default, the \ - and / operators in Matlab perform a column ordering (using colmmd - or colamd) prior to LU factorization using sparse partial pivoting, - in the built-in Matlab lu(A) routine. - - Symamd computes a permutation P of a symmetric matrix A such that the - Cholesky factorization of PAP' has less fill-in and requires fewer - floating point operations than A. Symamd constructs a matrix M such - that M'M has the same nonzero pattern of A, and then orders the columns - of M using colmmd. The column ordering of M is then returned as the - row and column ordering P of A. - - Authors: - - The authors of the code itself are Stefan I. Larimore and Timothy A. - Davis (davis@cise.ufl.edu), University of Florida. The algorithm was - developed in collaboration with John Gilbert, Xerox PARC, and Esmond - Ng, Oak Ridge National Laboratory. - - Date: - - May 4, 2001. Version 2.1. - - Acknowledgements: - - This work was supported by the National Science Foundation, under - grants DMS-9504974 and DMS-9803599. - - Notice: - - Copyright (c) 1998-2001 by the University of Florida. - All Rights Reserved. - - THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY - EXPRESSED OR IMPLIED. ANY USE IS AT YOUR OWN RISK. - - Permission is hereby granted to use or copy this program for any - purpose, provided the above notices are retained on all copies. - User documentation of any code that uses this code must cite the - Authors, the Copyright, and "Used by permission." If this code is - accessible from within Matlab, then typing "help colamd" and "help - symamd" must cite the Authors. Permission to modify the code and to - distribute modified code is granted, provided the above notices are - retained, and a notice that the code was modified is included with the - above copyright notice. You must also retain the Availability - information below, of the original version. - - This software is provided free of charge. - - Availability: - - The colamd/symamd library is available at - - http://www.cise.ufl.edu/research/sparse/colamd/ - - This is the http://www.cise.ufl.edu/research/sparse/colamd/colamd.c - file. It requires the colamd.h file. It is required by the colamdmex.c - and symamdmex.c files, for the Matlab interface to colamd and symamd. - - Changes to the colamd library since Version 1.0 and 1.1: - - No bugs were found in version 1.1. These changes merely add new - functionality. - - * added the COLAMD_RECOMMENDED (nnz, n_row, n_col) macro. - - * moved the output statistics, from A, to a separate output argument. - The arguments changed for the C-callable routines. - - * added colamd_report and symamd_report. - - * added a C-callable symamd routine. Formerly, symamd was only - available as a mexFunction from Matlab. - - * added error-checking to symamd. Formerly, it assumed its input - was error-free. - - * added the optional stats and knobs arguments to the symamd mexFunction - - * deleted colamd_help. A help message is still available from - "help colamd" and "help symamd" in Matlab. - - * deleted colamdtree.m and symamdtree.m. Now, colamd.m and symamd.m - also do the elimination tree post-ordering. The Version 1.1 - colamd and symamd mexFunctions, which do not do the post- - ordering, are now visible as colamdmex and symamdmex from - Matlab. Essentialy, the post-ordering is now the default - behavior of colamd.m and symamd.m, to match the behavior of - colmmd and symmmd. The post-ordering is only available in the - Matlab interface, not the C-callable interface. - - * made a slight change to the dense row/column detection in symamd, - to match the stated specifications. - - Changes from Version 2.0 to 2.1: - - * TRUE and FALSE are predefined on some systems, so they are defined - here only if not already defined. - - * web site changed - - * UNIX Makefile modified, to handle the case if "." is not in your path. - -*/ - -/* ========================================================================== */ -/* === Description of user-callable routines ================================ */ -/* ========================================================================== */ - -/* - ---------------------------------------------------------------------------- - colamd_recommended: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - int colamd_recommended (int nnz, int n_row, int n_col) ; - - or as a C macro - - #include "colamd.h" - Alen = COLAMD_RECOMMENDED (int nnz, int n_row, int n_col) ; - - Purpose: - - Returns recommended value of Alen for use by colamd. Returns -1 - if any input argument is negative. The use of this routine - or macro is optional. Note that the macro uses its arguments - more than once, so be careful for side effects, if you pass - expressions as arguments to COLAMD_RECOMMENDED. Not needed for - symamd, which dynamically allocates its own memory. - - Arguments (all input arguments): - - int nnz ; Number of nonzeros in the matrix A. This must - be the same value as p [n_col] in the call to - colamd - otherwise you will get a wrong value - of the recommended memory to use. - - int n_row ; Number of rows in the matrix A. - - int n_col ; Number of columns in the matrix A. - - ---------------------------------------------------------------------------- - colamd_set_defaults: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - colamd_set_defaults (int knobs [COLAMD_KNOBS]) ; - - Purpose: - - Sets the default parameters. The use of this routine is optional. - - Arguments: - - double knobs [COLAMD_KNOBS] ; Output only. - - Colamd: rows with more than (knobs [COLAMD_DENSE_ROW] * n_col) - entries are removed prior to ordering. Columns with more than - (knobs [COLAMD_DENSE_COL] * n_row) entries are removed prior to - ordering, and placed last in the output column ordering. - - Symamd: uses only knobs [COLAMD_DENSE_ROW], which is knobs [0]. - Rows and columns with more than (knobs [COLAMD_DENSE_ROW] * n) - entries are removed prior to ordering, and placed last in the - output ordering. - - COLAMD_DENSE_ROW and COLAMD_DENSE_COL are defined as 0 and 1, - respectively, in colamd.h. Default values of these two knobs - are both 0.5. Currently, only knobs [0] and knobs [1] are - used, but future versions may use more knobs. If so, they will - be properly set to their defaults by the future version of - colamd_set_defaults, so that the code that calls colamd will - not need to change, assuming that you either use - colamd_set_defaults, or pass a (double *) NULL pointer as the - knobs array to colamd or symamd. - - ---------------------------------------------------------------------------- - colamd: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - int colamd (int n_row, int n_col, int Alen, int *A, int *p, - double knobs [COLAMD_KNOBS], int stats [COLAMD_STATS]) ; - - Purpose: - - Computes a column ordering (Q) of A such that P(AQ)=LU or - (AQ)'AQ=LL' have less fill-in and require fewer floating point - operations than factorizing the unpermuted matrix A or A'A, - respectively. - - Returns: - - TRUE (1) if successful, FALSE (0) otherwise. - - Arguments: - - int n_row ; Input argument. - - Number of rows in the matrix A. - Restriction: n_row >= 0. - Colamd returns FALSE if n_row is negative. - - int n_col ; Input argument. - - Number of columns in the matrix A. - Restriction: n_col >= 0. - Colamd returns FALSE if n_col is negative. - - int Alen ; Input argument. - - Restriction (see note): - Alen >= 2*nnz + 6*(n_col+1) + 4*(n_row+1) + n_col - Colamd returns FALSE if these conditions are not met. - - Note: this restriction makes an modest assumption regarding - the size of the two typedef's structures in colamd.h. - We do, however, guarantee that - - Alen >= colamd_recommended (nnz, n_row, n_col) - - or equivalently as a C preprocessor macro: - - Alen >= COLAMD_RECOMMENDED (nnz, n_row, n_col) - - will be sufficient. - - int A [Alen] ; Input argument, undefined on output. - - A is an integer array of size Alen. Alen must be at least as - large as the bare minimum value given above, but this is very - low, and can result in excessive run time. For best - performance, we recommend that Alen be greater than or equal to - colamd_recommended (nnz, n_row, n_col), which adds - nnz/5 to the bare minimum value given above. - - On input, the row indices of the entries in column c of the - matrix are held in A [(p [c]) ... (p [c+1]-1)]. The row indices - in a given column c need not be in ascending order, and - duplicate row indices may be be present. However, colamd will - work a little faster if both of these conditions are met - (Colamd puts the matrix into this format, if it finds that the - the conditions are not met). - - The matrix is 0-based. That is, rows are in the range 0 to - n_row-1, and columns are in the range 0 to n_col-1. Colamd - returns FALSE if any row index is out of range. - - The contents of A are modified during ordering, and are - undefined on output. - - int p [n_col+1] ; Both input and output argument. - - p is an integer array of size n_col+1. On input, it holds the - "pointers" for the column form of the matrix A. Column c of - the matrix A is held in A [(p [c]) ... (p [c+1]-1)]. The first - entry, p [0], must be zero, and p [c] <= p [c+1] must hold - for all c in the range 0 to n_col-1. The value p [n_col] is - thus the total number of entries in the pattern of the matrix A. - Colamd returns FALSE if these conditions are not met. - - On output, if colamd returns TRUE, the array p holds the column - permutation (Q, for P(AQ)=LU or (AQ)'(AQ)=LL'), where p [0] is - the first column index in the new ordering, and p [n_col-1] is - the last. That is, p [k] = j means that column j of A is the - kth pivot column, in AQ, where k is in the range 0 to n_col-1 - (p [0] = j means that column j of A is the first column in AQ). - - If colamd returns FALSE, then no permutation is returned, and - p is undefined on output. - - double knobs [COLAMD_KNOBS] ; Input argument. - - See colamd_set_defaults for a description. - - int stats [COLAMD_STATS] ; Output argument. - - Statistics on the ordering, and error status. - See colamd.h for related definitions. - Colamd returns FALSE if stats is not present. - - stats [0]: number of dense or empty rows ignored. - - stats [1]: number of dense or empty columns ignored (and - ordered last in the output permutation p) - Note that a row can become "empty" if it - contains only "dense" and/or "empty" columns, - and similarly a column can become "empty" if it - only contains "dense" and/or "empty" rows. - - stats [2]: number of garbage collections performed. - This can be excessively high if Alen is close - to the minimum required value. - - stats [3]: status code. < 0 is an error code. - > 1 is a warning or notice. - - 0 OK. Each column of the input matrix contained - row indices in increasing order, with no - duplicates. - - 1 OK, but columns of input matrix were jumbled - (unsorted columns or duplicate entries). Colamd - had to do some extra work to sort the matrix - first and remove duplicate entries, but it - still was able to return a valid permutation - (return value of colamd was TRUE). - - stats [4]: highest numbered column that - is unsorted or has duplicate - entries. - stats [5]: last seen duplicate or - unsorted row index. - stats [6]: number of duplicate or - unsorted row indices. - - -1 A is a null pointer - - -2 p is a null pointer - - -3 n_row is negative - - stats [4]: n_row - - -4 n_col is negative - - stats [4]: n_col - - -5 number of nonzeros in matrix is negative - - stats [4]: number of nonzeros, p [n_col] - - -6 p [0] is nonzero - - stats [4]: p [0] - - -7 A is too small - - stats [4]: required size - stats [5]: actual size (Alen) - - -8 a column has a negative number of entries - - stats [4]: column with < 0 entries - stats [5]: number of entries in col - - -9 a row index is out of bounds - - stats [4]: column with bad row index - stats [5]: bad row index - stats [6]: n_row, # of rows of matrx - - -10 (unused; see symamd.c) - - -999 (unused; see symamd.c) - - Future versions may return more statistics in the stats array. - - Example: - - See http://www.cise.ufl.edu/research/sparse/colamd/example.c - for a complete example. - - To order the columns of a 5-by-4 matrix with 11 nonzero entries in - the following nonzero pattern - - x 0 x 0 - x 0 x x - 0 x x 0 - 0 0 x x - x x 0 0 - - with default knobs and no output statistics, do the following: - - #include "colamd.h" - #define ALEN COLAMD_RECOMMENDED (11, 5, 4) - int A [ALEN] = {1, 2, 5, 3, 5, 1, 2, 3, 4, 2, 4} ; - int p [ ] = {0, 3, 5, 9, 11} ; - int stats [COLAMD_STATS] ; - colamd (5, 4, ALEN, A, p, (double *) NULL, stats) ; - - The permutation is returned in the array p, and A is destroyed. - - ---------------------------------------------------------------------------- - symamd: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - int symamd (int n, int *A, int *p, int *perm, - int knobs [COLAMD_KNOBS], int stats [COLAMD_STATS], - void (*allocate) (size_t, size_t), void (*release) (void *)) ; - - Purpose: - - The symamd routine computes an ordering P of a symmetric sparse - matrix A such that the Cholesky factorization PAP' = LL' remains - sparse. It is based on a column ordering of a matrix M constructed - so that the nonzero pattern of M'M is the same as A. The matrix A - is assumed to be symmetric; only the strictly lower triangular part - is accessed. You must pass your selected memory allocator (usually - calloc/free or mxCalloc/mxFree) to symamd, for it to allocate - memory for the temporary matrix M. - - Returns: - - TRUE (1) if successful, FALSE (0) otherwise. - - Arguments: - - int n ; Input argument. - - Number of rows and columns in the symmetrix matrix A. - Restriction: n >= 0. - Symamd returns FALSE if n is negative. - - int A [nnz] ; Input argument. - - A is an integer array of size nnz, where nnz = p [n]. - - The row indices of the entries in column c of the matrix are - held in A [(p [c]) ... (p [c+1]-1)]. The row indices in a - given column c need not be in ascending order, and duplicate - row indices may be present. However, symamd will run faster - if the columns are in sorted order with no duplicate entries. - - The matrix is 0-based. That is, rows are in the range 0 to - n-1, and columns are in the range 0 to n-1. Symamd - returns FALSE if any row index is out of range. - - The contents of A are not modified. - - int p [n+1] ; Input argument. - - p is an integer array of size n+1. On input, it holds the - "pointers" for the column form of the matrix A. Column c of - the matrix A is held in A [(p [c]) ... (p [c+1]-1)]. The first - entry, p [0], must be zero, and p [c] <= p [c+1] must hold - for all c in the range 0 to n-1. The value p [n] is - thus the total number of entries in the pattern of the matrix A. - Symamd returns FALSE if these conditions are not met. - - The contents of p are not modified. - - int perm [n+1] ; Output argument. - - On output, if symamd returns TRUE, the array perm holds the - permutation P, where perm [0] is the first index in the new - ordering, and perm [n-1] is the last. That is, perm [k] = j - means that row and column j of A is the kth column in PAP', - where k is in the range 0 to n-1 (perm [0] = j means - that row and column j of A are the first row and column in - PAP'). The array is used as a workspace during the ordering, - which is why it must be of length n+1, not just n. - - double knobs [COLAMD_KNOBS] ; Input argument. - - See colamd_set_defaults for a description. - - int stats [COLAMD_STATS] ; Output argument. - - Statistics on the ordering, and error status. - See colamd.h for related definitions. - Symamd returns FALSE if stats is not present. - - stats [0]: number of dense or empty row and columns ignored - (and ordered last in the output permutation - perm). Note that a row/column can become - "empty" if it contains only "dense" and/or - "empty" columns/rows. - - stats [1]: (same as stats [0]) - - stats [2]: number of garbage collections performed. - - stats [3]: status code. < 0 is an error code. - > 1 is a warning or notice. - - 0 OK. Each column of the input matrix contained - row indices in increasing order, with no - duplicates. - - 1 OK, but columns of input matrix were jumbled - (unsorted columns or duplicate entries). Symamd - had to do some extra work to sort the matrix - first and remove duplicate entries, but it - still was able to return a valid permutation - (return value of symamd was TRUE). - - stats [4]: highest numbered column that - is unsorted or has duplicate - entries. - stats [5]: last seen duplicate or - unsorted row index. - stats [6]: number of duplicate or - unsorted row indices. - - -1 A is a null pointer - - -2 p is a null pointer - - -3 (unused, see colamd.c) - - -4 n is negative - - stats [4]: n - - -5 number of nonzeros in matrix is negative - - stats [4]: # of nonzeros (p [n]). - - -6 p [0] is nonzero - - stats [4]: p [0] - - -7 (unused) - - -8 a column has a negative number of entries - - stats [4]: column with < 0 entries - stats [5]: number of entries in col - - -9 a row index is out of bounds - - stats [4]: column with bad row index - stats [5]: bad row index - stats [6]: n_row, # of rows of matrx - - -10 out of memory (unable to allocate temporary - workspace for M or count arrays using the - "allocate" routine passed into symamd). - - -999 internal error. colamd failed to order the - matrix M, when it should have succeeded. This - indicates a bug. If this (and *only* this) - error code occurs, please contact the authors. - Don't contact the authors if you get any other - error code. - - Future versions may return more statistics in the stats array. - - void * (*allocate) (size_t, size_t) - - A pointer to a function providing memory allocation. The - allocated memory must be returned initialized to zero. For a - C application, this argument should normally be a pointer to - calloc. For a Matlab mexFunction, the routine mxCalloc is - passed instead. - - void (*release) (size_t, size_t) - - A pointer to a function that frees memory allocated by the - memory allocation routine above. For a C application, this - argument should normally be a pointer to free. For a Matlab - mexFunction, the routine mxFree is passed instead. - - - ---------------------------------------------------------------------------- - colamd_report: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - colamd_report (int stats [COLAMD_STATS]) ; - - Purpose: - - Prints the error status and statistics recorded in the stats - array on the standard error output (for a standard C routine) - or on the Matlab output (for a mexFunction). - - Arguments: - - int stats [COLAMD_STATS] ; Input only. Statistics from colamd. - - - ---------------------------------------------------------------------------- - symamd_report: - ---------------------------------------------------------------------------- - - C syntax: - - #include "colamd.h" - symamd_report (int stats [COLAMD_STATS]) ; - - Purpose: - - Prints the error status and statistics recorded in the stats - array on the standard error output (for a standard C routine) - or on the Matlab output (for a mexFunction). - - Arguments: - - int stats [COLAMD_STATS] ; Input only. Statistics from symamd. - - -*/ - -/* ========================================================================== */ -/* === Scaffolding code definitions ======================================== */ -/* ========================================================================== */ - -/* Ensure that debugging is turned off: */ -#ifndef NDEBUG -#define NDEBUG -#endif /* NDEBUG */ - -/* - Our "scaffolding code" philosophy: In our opinion, well-written library - code should keep its "debugging" code, and just normally have it turned off - by the compiler so as not to interfere with performance. This serves - several purposes: - - (1) assertions act as comments to the reader, telling you what the code - expects at that point. All assertions will always be true (unless - there really is a bug, of course). - - (2) leaving in the scaffolding code assists anyone who would like to modify - the code, or understand the algorithm (by reading the debugging output, - one can get a glimpse into what the code is doing). - - (3) (gasp!) for actually finding bugs. This code has been heavily tested - and "should" be fully functional and bug-free ... but you never know... - - To enable debugging, comment out the "#define NDEBUG" above. For a Matlab - mexFunction, you will also need to modify mexopts.sh to remove the -DNDEBUG - definition. The code will become outrageously slow when debugging is - enabled. To control the level of debugging output, set an environment - variable D to 0 (little), 1 (some), 2, 3, or 4 (lots). When debugging, - you should see the following message on the standard output: - - colamd: debug version, D = 1 (THIS WILL BE SLOW!) - - or a similar message for symamd. If you don't, then debugging has not - been enabled. - -*/ - -/* ========================================================================== */ -/* === Include files ======================================================== */ -/* ========================================================================== */ - -#include "colamd.h" -#include - -#ifdef MATLAB_MEX_FILE -#include "mex.h" -#include "matrix.h" -#else -#include -#include -#endif /* MATLAB_MEX_FILE */ - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* ========================================================================== */ -/* === Definitions ========================================================== */ -/* ========================================================================== */ - -/* Routines are either PUBLIC (user-callable) or PRIVATE (not user-callable) */ -#define PUBLIC -#define PRIVATE static - -#define MAX(a,b) (((a) > (b)) ? (a) : (b)) -#define MIN(a,b) (((a) < (b)) ? (a) : (b)) - -#define ONES_COMPLEMENT(r) (-(r)-1) - -/* -------------------------------------------------------------------------- */ -/* Change for version 2.1: define TRUE and FALSE only if not yet defined */ -/* -------------------------------------------------------------------------- */ - -#ifndef TRUE -#define TRUE (1) -#endif - -#ifndef FALSE -#define FALSE (0) -#endif - -/* -------------------------------------------------------------------------- */ - -#define EMPTY (-1) - -/* Row and column status */ -#define ALIVE (0) -#define DEAD (-1) - -/* Column status */ -#define DEAD_PRINCIPAL (-1) -#define DEAD_NON_PRINCIPAL (-2) - -/* Macros for row and column status update and checking. */ -#define ROW_IS_DEAD(r) ROW_IS_MARKED_DEAD (Row[r].shared2.mark) -#define ROW_IS_MARKED_DEAD(row_mark) (row_mark < ALIVE) -#define ROW_IS_ALIVE(r) (Row [r].shared2.mark >= ALIVE) -#define COL_IS_DEAD(c) (Col [c].start < ALIVE) -#define COL_IS_ALIVE(c) (Col [c].start >= ALIVE) -#define COL_IS_DEAD_PRINCIPAL(c) (Col [c].start == DEAD_PRINCIPAL) -#define KILL_ROW(r) { Row [r].shared2.mark = DEAD ; } -#define KILL_PRINCIPAL_COL(c) { Col [c].start = DEAD_PRINCIPAL ; } -#define KILL_NON_PRINCIPAL_COL(c) { Col [c].start = DEAD_NON_PRINCIPAL ; } - -/* ========================================================================== */ -/* === Colamd reporting mechanism =========================================== */ -/* ========================================================================== */ - -#ifdef MATLAB_MEX_FILE - -/* use mexPrintf in a Matlab mexFunction, for debugging and statistics output */ -#define PRINTF mexPrintf - -/* In Matlab, matrices are 1-based to the user, but 0-based internally */ -#define INDEX(i) ((i)+1) - -#else - -/* Use printf in standard C environment, for debugging and statistics output. */ -/* Output is generated only if debugging is enabled at compile time, or if */ -/* the caller explicitly calls colamd_report or symamd_report. */ -#define PRINTF printf - -/* In C, matrices are 0-based and indices are reported as such in *_report */ -#define INDEX(i) (i) - -#endif /* MATLAB_MEX_FILE */ - -/* ========================================================================== */ -/* === Prototypes of PRIVATE routines ======================================= */ -/* ========================================================================== */ - -PRIVATE int init_rows_cols -( - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [], - int p [], - int stats [COLAMD_STATS] -) ; - -PRIVATE void init_scoring -( - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [], - int head [], - double knobs [COLAMD_KNOBS], - int *p_n_row2, - int *p_n_col2, - int *p_max_deg -) ; - -PRIVATE int find_ordering -( - int n_row, - int n_col, - int Alen, - Colamd_Row Row [], - Colamd_Col Col [], - int A [], - int head [], - int n_col2, - int max_deg, - int pfree -) ; - -PRIVATE void order_children -( - int n_col, - Colamd_Col Col [], - int p [] -) ; - -PRIVATE void detect_super_cols -( - -#ifndef NDEBUG - int n_col, - Colamd_Row Row [], -#endif /* NDEBUG */ - - Colamd_Col Col [], - int A [], - int head [], - int row_start, - int row_length -) ; - -PRIVATE int garbage_collection -( - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [], - int *pfree -) ; - -PRIVATE int clear_mark -( - int n_row, - Colamd_Row Row [] -) ; - -PRIVATE void print_report -( - char *method, - int stats [COLAMD_STATS] -) ; - -/* ========================================================================== */ -/* === Debugging prototypes and definitions ================================= */ -/* ========================================================================== */ - -#ifndef NDEBUG - -/* colamd_debug is the *ONLY* global variable, and is only */ -/* present when debugging */ - -PRIVATE int colamd_debug ; /* debug print level */ - -#define DEBUG0(params) { (void) PRINTF params ; } -#define DEBUG1(params) { if (colamd_debug >= 1) (void) PRINTF params ; } -#define DEBUG2(params) { if (colamd_debug >= 2) (void) PRINTF params ; } -#define DEBUG3(params) { if (colamd_debug >= 3) (void) PRINTF params ; } -#define DEBUG4(params) { if (colamd_debug >= 4) (void) PRINTF params ; } - -#ifdef MATLAB_MEX_FILE -#define ASSERT(expression) (mxAssert ((expression), "")) -#else -#define ASSERT(expression) (assert (expression)) -#endif /* MATLAB_MEX_FILE */ - -PRIVATE void colamd_get_debug /* gets the debug print level from getenv */ -( - char *method -) ; - -PRIVATE void debug_deg_lists -( - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int head [], - int min_score, - int should, - int max_deg -) ; - -PRIVATE void debug_mark -( - int n_row, - Colamd_Row Row [], - int tag_mark, - int max_mark -) ; - -PRIVATE void debug_matrix -( - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [] -) ; - -PRIVATE void debug_structures -( - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [], - int n_col2 -) ; - -#else /* NDEBUG */ - -/* === No debugging ========================================================= */ - -#define DEBUG0(params) ; -#define DEBUG1(params) ; -#define DEBUG2(params) ; -#define DEBUG3(params) ; -#define DEBUG4(params) ; - -#define ASSERT(expression) ((void) 0) - -#endif /* NDEBUG */ - -/* ========================================================================== */ - - - -/* ========================================================================== */ -/* === USER-CALLABLE ROUTINES: ============================================== */ -/* ========================================================================== */ - - -/* ========================================================================== */ -/* === colamd_recommended =================================================== */ -/* ========================================================================== */ - -/* - The colamd_recommended routine returns the suggested size for Alen. This - value has been determined to provide good balance between the number of - garbage collections and the memory requirements for colamd. If any - argument is negative, a -1 is returned as an error condition. This - function is also available as a macro defined in colamd.h, so that you - can use it for a statically-allocated array size. -*/ - -PUBLIC int colamd_recommended /* returns recommended value of Alen. */ -( - /* === Parameters ======================================================= */ - - int nnz, /* number of nonzeros in A */ - int n_row, /* number of rows in A */ - int n_col /* number of columns in A */ -) -{ - return (COLAMD_RECOMMENDED (nnz, n_row, n_col)) ; -} - - -/* ========================================================================== */ -/* === colamd_set_defaults ================================================== */ -/* ========================================================================== */ - -/* - The colamd_set_defaults routine sets the default values of the user- - controllable parameters for colamd: - - knobs [0] rows with knobs[0]*n_col entries or more are removed - prior to ordering in colamd. Rows and columns with - knobs[0]*n_col entries or more are removed prior to - ordering in symamd and placed last in the output - ordering. - - knobs [1] columns with knobs[1]*n_row entries or more are removed - prior to ordering in colamd, and placed last in the - column permutation. Symamd ignores this knob. - - knobs [2..19] unused, but future versions might use this -*/ - -PUBLIC void colamd_set_defaults -( - /* === Parameters ======================================================= */ - - double knobs [COLAMD_KNOBS] /* knob array */ -) -{ - /* === Local variables ================================================== */ - - int i ; - - if (!knobs) - { - return ; /* no knobs to initialize */ - } - for (i = 0 ; i < COLAMD_KNOBS ; i++) - { - knobs [i] = 0 ; - } - knobs [COLAMD_DENSE_ROW] = 0.5 ; /* ignore rows over 50% dense */ - knobs [COLAMD_DENSE_COL] = 0.5 ; /* ignore columns over 50% dense */ -} - - -/* ========================================================================== */ -/* === symamd =============================================================== */ -/* ========================================================================== */ - -PUBLIC int symamd /* return TRUE if OK, FALSE otherwise */ -( - /* === Parameters ======================================================= */ - - int n, /* number of rows and columns of A */ - int A [], /* row indices of A */ - int p [], /* column pointers of A */ - int perm [], /* output permutation, size n+1 */ - double knobs [COLAMD_KNOBS], /* parameters (uses defaults if NULL) */ - int stats [COLAMD_STATS], /* output statistics and error codes */ - void * (*allocate) (size_t, size_t), - /* pointer to calloc (ANSI C) or */ - /* mxCalloc (for Matlab mexFunction) */ - void (*release) (void *) - /* pointer to free (ANSI C) or */ - /* mxFree (for Matlab mexFunction) */ -) -{ - /* === Local variables ================================================== */ - - int *count ; /* length of each column of M, and col pointer*/ - int *mark ; /* mark array for finding duplicate entries */ - int *M ; /* row indices of matrix M */ - int Mlen ; /* length of M */ - int n_row ; /* number of rows in M */ - int nnz ; /* number of entries in A */ - int i ; /* row index of A */ - int j ; /* column index of A */ - int k ; /* row index of M */ - int mnz ; /* number of nonzeros in M */ - int pp ; /* index into a column of A */ - int last_row ; /* last row seen in the current column */ - int length ; /* number of nonzeros in a column */ - - double cknobs [COLAMD_KNOBS] ; /* knobs for colamd */ - double default_knobs [COLAMD_KNOBS] ; /* default knobs for colamd */ - int cstats [COLAMD_STATS] ; /* colamd stats */ - -#ifndef NDEBUG - colamd_get_debug ("symamd") ; -#endif /* NDEBUG */ - - /* === Check the input arguments ======================================== */ - - if (!stats) - { - DEBUG0 (("symamd: stats not present\n")) ; - return (FALSE) ; - } - for (i = 0 ; i < COLAMD_STATS ; i++) - { - stats [i] = 0 ; - } - stats [COLAMD_STATUS] = COLAMD_OK ; - stats [COLAMD_INFO1] = -1 ; - stats [COLAMD_INFO2] = -1 ; - - if (!A) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_A_not_present ; - DEBUG0 (("symamd: A not present\n")) ; - return (FALSE) ; - } - - if (!p) /* p is not present */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p_not_present ; - DEBUG0 (("symamd: p not present\n")) ; - return (FALSE) ; - } - - if (n < 0) /* n must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_ncol_negative ; - stats [COLAMD_INFO1] = n ; - DEBUG0 (("symamd: n negative %d\n", n)) ; - return (FALSE) ; - } - - nnz = p [n] ; - if (nnz < 0) /* nnz must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_nnz_negative ; - stats [COLAMD_INFO1] = nnz ; - DEBUG0 (("symamd: number of entries negative %d\n", nnz)) ; - return (FALSE) ; - } - - if (p [0] != 0) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p0_nonzero ; - stats [COLAMD_INFO1] = p [0] ; - DEBUG0 (("symamd: p[0] not zero %d\n", p [0])) ; - return (FALSE) ; - } - - /* === If no knobs, set default knobs =================================== */ - - if (!knobs) - { - colamd_set_defaults (default_knobs) ; - knobs = default_knobs ; - } - - /* === Allocate count and mark ========================================== */ - - count = (int *) ((*allocate) (n+1, sizeof (int))) ; - if (!count) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_out_of_memory ; - DEBUG0 (("symamd: allocate count (size %d) failed\n", n+1)) ; - return (FALSE) ; - } - - mark = (int *) ((*allocate) (n+1, sizeof (int))) ; - if (!mark) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_out_of_memory ; - (*release) ((void *) count) ; - DEBUG0 (("symamd: allocate mark (size %d) failed\n", n+1)) ; - return (FALSE) ; - } - - /* === Compute column counts of M, check if A is valid ================== */ - - stats [COLAMD_INFO3] = 0 ; /* number of duplicate or unsorted row indices*/ - - for (i = 0 ; i < n ; i++) - { - mark [i] = -1 ; - } - - for (j = 0 ; j < n ; j++) - { - last_row = -1 ; - - length = p [j+1] - p [j] ; - if (length < 0) - { - /* column pointers must be non-decreasing */ - stats [COLAMD_STATUS] = COLAMD_ERROR_col_length_negative ; - stats [COLAMD_INFO1] = j ; - stats [COLAMD_INFO2] = length ; - (*release) ((void *) count) ; - (*release) ((void *) mark) ; - DEBUG0 (("symamd: col %d negative length %d\n", j, length)) ; - return (FALSE) ; - } - - for (pp = p [j] ; pp < p [j+1] ; pp++) - { - i = A [pp] ; - if (i < 0 || i >= n) - { - /* row index i, in column j, is out of bounds */ - stats [COLAMD_STATUS] = COLAMD_ERROR_row_index_out_of_bounds ; - stats [COLAMD_INFO1] = j ; - stats [COLAMD_INFO2] = i ; - stats [COLAMD_INFO3] = n ; - (*release) ((void *) count) ; - (*release) ((void *) mark) ; - DEBUG0 (("symamd: row %d col %d out of bounds\n", i, j)) ; - return (FALSE) ; - } - - if (i <= last_row || mark [i] == j) - { - /* row index is unsorted or repeated (or both), thus col */ - /* is jumbled. This is a notice, not an error condition. */ - stats [COLAMD_STATUS] = COLAMD_OK_BUT_JUMBLED ; - stats [COLAMD_INFO1] = j ; - stats [COLAMD_INFO2] = i ; - (stats [COLAMD_INFO3]) ++ ; - DEBUG1 (("symamd: row %d col %d unsorted/duplicate\n", i, j)) ; - } - - if (i > j && mark [i] != j) - { - /* row k of M will contain column indices i and j */ - count [i]++ ; - count [j]++ ; - } - - /* mark the row as having been seen in this column */ - mark [i] = j ; - - last_row = i ; - } - } - - if (stats [COLAMD_STATUS] == COLAMD_OK) - { - /* if there are no duplicate entries, then mark is no longer needed */ - (*release) ((void *) mark) ; - } - - /* === Compute column pointers of M ===================================== */ - - /* use output permutation, perm, for column pointers of M */ - perm [0] = 0 ; - for (j = 1 ; j <= n ; j++) - { - perm [j] = perm [j-1] + count [j-1] ; - } - for (j = 0 ; j < n ; j++) - { - count [j] = perm [j] ; - } - - /* === Construct M ====================================================== */ - - mnz = perm [n] ; - n_row = mnz / 2 ; - Mlen = colamd_recommended (mnz, n_row, n) ; - M = (int *) ((*allocate) (Mlen, sizeof (int))) ; - DEBUG0 (("symamd: M is %d-by-%d with %d entries, Mlen = %d\n", - n_row, n, mnz, Mlen)) ; - - if (!M) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_out_of_memory ; - (*release) ((void *) count) ; - (*release) ((void *) mark) ; - DEBUG0 (("symamd: allocate M (size %d) failed\n", Mlen)) ; - return (FALSE) ; - } - - k = 0 ; - - if (stats [COLAMD_STATUS] == COLAMD_OK) - { - /* Matrix is OK */ - for (j = 0 ; j < n ; j++) - { - ASSERT (p [j+1] - p [j] >= 0) ; - for (pp = p [j] ; pp < p [j+1] ; pp++) - { - i = A [pp] ; - ASSERT (i >= 0 && i < n) ; - if (i > j) - { - /* row k of M contains column indices i and j */ - M [count [i]++] = k ; - M [count [j]++] = k ; - k++ ; - } - } - } - } - else - { - /* Matrix is jumbled. Do not add duplicates to M. Unsorted cols OK. */ - DEBUG0 (("symamd: Duplicates in A.\n")) ; - for (i = 0 ; i < n ; i++) - { - mark [i] = -1 ; - } - for (j = 0 ; j < n ; j++) - { - ASSERT (p [j+1] - p [j] >= 0) ; - for (pp = p [j] ; pp < p [j+1] ; pp++) - { - i = A [pp] ; - ASSERT (i >= 0 && i < n) ; - if (i > j && mark [i] != j) - { - /* row k of M contains column indices i and j */ - M [count [i]++] = k ; - M [count [j]++] = k ; - k++ ; - mark [i] = j ; - } - } - } - (*release) ((void *) mark) ; - } - - /* count and mark no longer needed */ - (*release) ((void *) count) ; - ASSERT (k == n_row) ; - - /* === Adjust the knobs for M =========================================== */ - - for (i = 0 ; i < COLAMD_KNOBS ; i++) - { - cknobs [i] = knobs [i] ; - } - - /* there are no dense rows in M */ - cknobs [COLAMD_DENSE_ROW] = 1.0 ; - - if (n_row != 0 && n < n_row) - { - /* On input, the knob is a fraction of 1..n, the number of rows of A. */ - /* Convert it to a fraction of 1..n_row, of the number of rows of M. */ - cknobs [COLAMD_DENSE_COL] = (knobs [COLAMD_DENSE_ROW] * n) / n_row ; - } - else - { - /* no dense columns in M */ - cknobs [COLAMD_DENSE_COL] = 1.0 ; - } - - DEBUG0 (("symamd: dense col knob for M: %g\n", cknobs [COLAMD_DENSE_COL])) ; - - /* === Order the columns of M =========================================== */ - - if (!colamd (n_row, n, Mlen, M, perm, cknobs, cstats)) - { - /* This "cannot" happen, unless there is a bug in the code. */ - stats [COLAMD_STATUS] = COLAMD_ERROR_internal_error ; - (*release) ((void *) M) ; - DEBUG0 (("symamd: internal error!\n")) ; - return (FALSE) ; - } - - /* Note that the output permutation is now in perm */ - - /* === get the statistics for symamd from colamd ======================== */ - - /* note that a dense column in colamd means a dense row and col in symamd */ - stats [COLAMD_DENSE_ROW] = cstats [COLAMD_DENSE_COL] ; - stats [COLAMD_DENSE_COL] = cstats [COLAMD_DENSE_COL] ; - stats [COLAMD_DEFRAG_COUNT] = cstats [COLAMD_DEFRAG_COUNT] ; - - /* === Free M =========================================================== */ - - (*release) ((void *) M) ; - DEBUG0 (("symamd: done.\n")) ; - return (TRUE) ; - -} - -/* ========================================================================== */ -/* === colamd =============================================================== */ -/* ========================================================================== */ - -/* - The colamd routine computes a column ordering Q of a sparse matrix - A such that the LU factorization P(AQ) = LU remains sparse, where P is - selected via partial pivoting. The routine can also be viewed as - providing a permutation Q such that the Cholesky factorization - (AQ)'(AQ) = LL' remains sparse. -*/ - -PUBLIC int colamd /* returns TRUE if successful, FALSE otherwise*/ -( - /* === Parameters ======================================================= */ - - int n_row, /* number of rows in A */ - int n_col, /* number of columns in A */ - int Alen, /* length of A */ - int A [], /* row indices of A */ - int p [], /* pointers to columns in A */ - double knobs [COLAMD_KNOBS],/* parameters (uses defaults if NULL) */ - int stats [COLAMD_STATS] /* output statistics and error codes */ -) -{ - /* === Local variables ================================================== */ - - int i ; /* loop index */ - int nnz ; /* nonzeros in A */ - int Row_size ; /* size of Row [], in integers */ - int Col_size ; /* size of Col [], in integers */ - int need ; /* minimum required length of A */ - Colamd_Row *Row ; /* pointer into A of Row [0..n_row] array */ - Colamd_Col *Col ; /* pointer into A of Col [0..n_col] array */ - int n_col2 ; /* number of non-dense, non-empty columns */ - int n_row2 ; /* number of non-dense, non-empty rows */ - int ngarbage ; /* number of garbage collections performed */ - int max_deg ; /* maximum row degree */ - double default_knobs [COLAMD_KNOBS] ; /* default knobs array */ - -#ifndef NDEBUG - colamd_get_debug ("colamd") ; -#endif /* NDEBUG */ - - /* === Check the input arguments ======================================== */ - - if (!stats) - { - DEBUG0 (("colamd: stats not present\n")) ; - return (FALSE) ; - } - for (i = 0 ; i < COLAMD_STATS ; i++) - { - stats [i] = 0 ; - } - stats [COLAMD_STATUS] = COLAMD_OK ; - stats [COLAMD_INFO1] = -1 ; - stats [COLAMD_INFO2] = -1 ; - - if (!A) /* A is not present */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_A_not_present ; - DEBUG0 (("colamd: A not present\n")) ; - return (FALSE) ; - } - - if (!p) /* p is not present */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p_not_present ; - DEBUG0 (("colamd: p not present\n")) ; - return (FALSE) ; - } - - if (n_row < 0) /* n_row must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_nrow_negative ; - stats [COLAMD_INFO1] = n_row ; - DEBUG0 (("colamd: nrow negative %d\n", n_row)) ; - return (FALSE) ; - } - - if (n_col < 0) /* n_col must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_ncol_negative ; - stats [COLAMD_INFO1] = n_col ; - DEBUG0 (("colamd: ncol negative %d\n", n_col)) ; - return (FALSE) ; - } - - nnz = p [n_col] ; - if (nnz < 0) /* nnz must be >= 0 */ - { - stats [COLAMD_STATUS] = COLAMD_ERROR_nnz_negative ; - stats [COLAMD_INFO1] = nnz ; - DEBUG0 (("colamd: number of entries negative %d\n", nnz)) ; - return (FALSE) ; - } - - if (p [0] != 0) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_p0_nonzero ; - stats [COLAMD_INFO1] = p [0] ; - DEBUG0 (("colamd: p[0] not zero %d\n", p [0])) ; - return (FALSE) ; - } - - /* === If no knobs, set default knobs =================================== */ - - if (!knobs) - { - colamd_set_defaults (default_knobs) ; - knobs = default_knobs ; - } - - /* === Allocate the Row and Col arrays from array A ===================== */ - - Col_size = COLAMD_C (n_col) ; - Row_size = COLAMD_R (n_row) ; - need = 2*nnz + n_col + Col_size + Row_size ; - - if (need > Alen) - { - /* not enough space in array A to perform the ordering */ - stats [COLAMD_STATUS] = COLAMD_ERROR_A_too_small ; - stats [COLAMD_INFO1] = need ; - stats [COLAMD_INFO2] = Alen ; - DEBUG0 (("colamd: Need Alen >= %d, given only Alen = %d\n", need,Alen)); - return (FALSE) ; - } - - Alen -= Col_size + Row_size ; - Col = (Colamd_Col *) &A [Alen] ; - Row = (Colamd_Row *) &A [Alen + Col_size] ; - - /* === Construct the row and column data structures ===================== */ - - if (!init_rows_cols (n_row, n_col, Row, Col, A, p, stats)) - { - /* input matrix is invalid */ - DEBUG0 (("colamd: Matrix invalid\n")) ; - return (FALSE) ; - } - - /* === Initialize scores, kill dense rows/columns ======================= */ - - init_scoring (n_row, n_col, Row, Col, A, p, knobs, - &n_row2, &n_col2, &max_deg) ; - - /* === Order the supercolumns =========================================== */ - - ngarbage = find_ordering (n_row, n_col, Alen, Row, Col, A, p, - n_col2, max_deg, 2*nnz) ; - - /* === Order the non-principal columns ================================== */ - - order_children (n_col, Col, p) ; - - /* === Return statistics in stats ======================================= */ - - stats [COLAMD_DENSE_ROW] = n_row - n_row2 ; - stats [COLAMD_DENSE_COL] = n_col - n_col2 ; - stats [COLAMD_DEFRAG_COUNT] = ngarbage ; - DEBUG0 (("colamd: done.\n")) ; - return (TRUE) ; -} - - -/* ========================================================================== */ -/* === colamd_report ======================================================== */ -/* ========================================================================== */ - -PUBLIC void colamd_report -( - int stats [COLAMD_STATS] -) -{ - print_report ("colamd", stats) ; -} - - -/* ========================================================================== */ -/* === symamd_report ======================================================== */ -/* ========================================================================== */ - -PUBLIC void symamd_report -( - int stats [COLAMD_STATS] -) -{ - print_report ("symamd", stats) ; -} - - - -/* ========================================================================== */ -/* === NON-USER-CALLABLE ROUTINES: ========================================== */ -/* ========================================================================== */ - -/* There are no user-callable routines beyond this point in the file */ - - -/* ========================================================================== */ -/* === init_rows_cols ======================================================= */ -/* ========================================================================== */ - -/* - Takes the column form of the matrix in A and creates the row form of the - matrix. Also, row and column attributes are stored in the Col and Row - structs. If the columns are un-sorted or contain duplicate row indices, - this routine will also sort and remove duplicate row indices from the - column form of the matrix. Returns FALSE if the matrix is invalid, - TRUE otherwise. Not user-callable. -*/ - -PRIVATE int init_rows_cols /* returns TRUE if OK, or FALSE otherwise */ -( - /* === Parameters ======================================================= */ - - int n_row, /* number of rows of A */ - int n_col, /* number of columns of A */ - Colamd_Row Row [], /* of size n_row+1 */ - Colamd_Col Col [], /* of size n_col+1 */ - int A [], /* row indices of A, of size Alen */ - int p [], /* pointers to columns in A, of size n_col+1 */ - int stats [COLAMD_STATS] /* colamd statistics */ -) -{ - /* === Local variables ================================================== */ - - int col ; /* a column index */ - int row ; /* a row index */ - int *cp ; /* a column pointer */ - int *cp_end ; /* a pointer to the end of a column */ - int *rp ; /* a row pointer */ - int *rp_end ; /* a pointer to the end of a row */ - int last_row ; /* previous row */ - - /* === Initialize columns, and check column pointers ==================== */ - - for (col = 0 ; col < n_col ; col++) - { - Col [col].start = p [col] ; - Col [col].length = p [col+1] - p [col] ; - - if (Col [col].length < 0) - { - /* column pointers must be non-decreasing */ - stats [COLAMD_STATUS] = COLAMD_ERROR_col_length_negative ; - stats [COLAMD_INFO1] = col ; - stats [COLAMD_INFO2] = Col [col].length ; - DEBUG0 (("colamd: col %d length %d < 0\n", col, Col [col].length)) ; - return (FALSE) ; - } - - Col [col].shared1.thickness = 1 ; - Col [col].shared2.score = 0 ; - Col [col].shared3.prev = EMPTY ; - Col [col].shared4.degree_next = EMPTY ; - } - - /* p [0..n_col] no longer needed, used as "head" in subsequent routines */ - - /* === Scan columns, compute row degrees, and check row indices ========= */ - - stats [COLAMD_INFO3] = 0 ; /* number of duplicate or unsorted row indices*/ - - for (row = 0 ; row < n_row ; row++) - { - Row [row].length = 0 ; - Row [row].shared2.mark = -1 ; - } - - for (col = 0 ; col < n_col ; col++) - { - last_row = -1 ; - - cp = &A [p [col]] ; - cp_end = &A [p [col+1]] ; - - while (cp < cp_end) - { - row = *cp++ ; - - /* make sure row indices within range */ - if (row < 0 || row >= n_row) - { - stats [COLAMD_STATUS] = COLAMD_ERROR_row_index_out_of_bounds ; - stats [COLAMD_INFO1] = col ; - stats [COLAMD_INFO2] = row ; - stats [COLAMD_INFO3] = n_row ; - DEBUG0 (("colamd: row %d col %d out of bounds\n", row, col)) ; - return (FALSE) ; - } - - if (row <= last_row || Row [row].shared2.mark == col) - { - /* row index are unsorted or repeated (or both), thus col */ - /* is jumbled. This is a notice, not an error condition. */ - stats [COLAMD_STATUS] = COLAMD_OK_BUT_JUMBLED ; - stats [COLAMD_INFO1] = col ; - stats [COLAMD_INFO2] = row ; - (stats [COLAMD_INFO3]) ++ ; - DEBUG1 (("colamd: row %d col %d unsorted/duplicate\n",row,col)); - } - - if (Row [row].shared2.mark != col) - { - Row [row].length++ ; - } - else - { - /* this is a repeated entry in the column, */ - /* it will be removed */ - Col [col].length-- ; - } - - /* mark the row as having been seen in this column */ - Row [row].shared2.mark = col ; - - last_row = row ; - } - } - - /* === Compute row pointers ============================================= */ - - /* row form of the matrix starts directly after the column */ - /* form of matrix in A */ - Row [0].start = p [n_col] ; - Row [0].shared1.p = Row [0].start ; - Row [0].shared2.mark = -1 ; - for (row = 1 ; row < n_row ; row++) - { - Row [row].start = Row [row-1].start + Row [row-1].length ; - Row [row].shared1.p = Row [row].start ; - Row [row].shared2.mark = -1 ; - } - - /* === Create row form ================================================== */ - - if (stats [COLAMD_STATUS] == COLAMD_OK_BUT_JUMBLED) - { - /* if cols jumbled, watch for repeated row indices */ - for (col = 0 ; col < n_col ; col++) - { - cp = &A [p [col]] ; - cp_end = &A [p [col+1]] ; - while (cp < cp_end) - { - row = *cp++ ; - if (Row [row].shared2.mark != col) - { - A [(Row [row].shared1.p)++] = col ; - Row [row].shared2.mark = col ; - } - } - } - } - else - { - /* if cols not jumbled, we don't need the mark (this is faster) */ - for (col = 0 ; col < n_col ; col++) - { - cp = &A [p [col]] ; - cp_end = &A [p [col+1]] ; - while (cp < cp_end) - { - A [(Row [*cp++].shared1.p)++] = col ; - } - } - } - - /* === Clear the row marks and set row degrees ========================== */ - - for (row = 0 ; row < n_row ; row++) - { - Row [row].shared2.mark = 0 ; - Row [row].shared1.degree = Row [row].length ; - } - - /* === See if we need to re-create columns ============================== */ - - if (stats [COLAMD_STATUS] == COLAMD_OK_BUT_JUMBLED) - { - DEBUG0 (("colamd: reconstructing column form, matrix jumbled\n")) ; - -#ifndef NDEBUG - /* make sure column lengths are correct */ - for (col = 0 ; col < n_col ; col++) - { - p [col] = Col [col].length ; - } - for (row = 0 ; row < n_row ; row++) - { - rp = &A [Row [row].start] ; - rp_end = rp + Row [row].length ; - while (rp < rp_end) - { - p [*rp++]-- ; - } - } - for (col = 0 ; col < n_col ; col++) - { - ASSERT (p [col] == 0) ; - } - /* now p is all zero (different than when debugging is turned off) */ -#endif /* NDEBUG */ - - /* === Compute col pointers ========================================= */ - - /* col form of the matrix starts at A [0]. */ - /* Note, we may have a gap between the col form and the row */ - /* form if there were duplicate entries, if so, it will be */ - /* removed upon the first garbage collection */ - Col [0].start = 0 ; - p [0] = Col [0].start ; - for (col = 1 ; col < n_col ; col++) - { - /* note that the lengths here are for pruned columns, i.e. */ - /* no duplicate row indices will exist for these columns */ - Col [col].start = Col [col-1].start + Col [col-1].length ; - p [col] = Col [col].start ; - } - - /* === Re-create col form =========================================== */ - - for (row = 0 ; row < n_row ; row++) - { - rp = &A [Row [row].start] ; - rp_end = rp + Row [row].length ; - while (rp < rp_end) - { - A [(p [*rp++])++] = row ; - } - } - } - - /* === Done. Matrix is not (or no longer) jumbled ====================== */ - - return (TRUE) ; -} - - -/* ========================================================================== */ -/* === init_scoring ========================================================= */ -/* ========================================================================== */ - -/* - Kills dense or empty columns and rows, calculates an initial score for - each column, and places all columns in the degree lists. Not user-callable. -*/ - -PRIVATE void init_scoring -( - /* === Parameters ======================================================= */ - - int n_row, /* number of rows of A */ - int n_col, /* number of columns of A */ - Colamd_Row Row [], /* of size n_row+1 */ - Colamd_Col Col [], /* of size n_col+1 */ - int A [], /* column form and row form of A */ - int head [], /* of size n_col+1 */ - double knobs [COLAMD_KNOBS],/* parameters */ - int *p_n_row2, /* number of non-dense, non-empty rows */ - int *p_n_col2, /* number of non-dense, non-empty columns */ - int *p_max_deg /* maximum row degree */ -) -{ - /* === Local variables ================================================== */ - - int c ; /* a column index */ - int r, row ; /* a row index */ - int *cp ; /* a column pointer */ - int deg ; /* degree of a row or column */ - int *cp_end ; /* a pointer to the end of a column */ - int *new_cp ; /* new column pointer */ - int col_length ; /* length of pruned column */ - int score ; /* current column score */ - int n_col2 ; /* number of non-dense, non-empty columns */ - int n_row2 ; /* number of non-dense, non-empty rows */ - int dense_row_count ; /* remove rows with more entries than this */ - int dense_col_count ; /* remove cols with more entries than this */ - int min_score ; /* smallest column score */ - int max_deg ; /* maximum row degree */ - int next_col ; /* Used to add to degree list.*/ - -#ifndef NDEBUG - int debug_count ; /* debug only. */ -#endif /* NDEBUG */ - - /* === Extract knobs ==================================================== */ - - dense_row_count = (int) MAX (0, MIN (knobs [COLAMD_DENSE_ROW] * n_col, n_col)) ; - dense_col_count = (int) MAX (0, MIN (knobs [COLAMD_DENSE_COL] * n_row, n_row)) ; - DEBUG1 (("colamd: densecount: %d %d\n", dense_row_count, dense_col_count)) ; - max_deg = 0 ; - n_col2 = n_col ; - n_row2 = n_row ; - - /* === Kill empty columns =============================================== */ - - /* Put the empty columns at the end in their natural order, so that LU */ - /* factorization can proceed as far as possible. */ - for (c = n_col-1 ; c >= 0 ; c--) - { - deg = Col [c].length ; - if (deg == 0) - { - /* this is a empty column, kill and order it last */ - Col [c].shared2.order = --n_col2 ; - KILL_PRINCIPAL_COL (c) ; - } - } - DEBUG1 (("colamd: null columns killed: %d\n", n_col - n_col2)) ; - - /* === Kill dense columns =============================================== */ - - /* Put the dense columns at the end, in their natural order */ - for (c = n_col-1 ; c >= 0 ; c--) - { - /* skip any dead columns */ - if (COL_IS_DEAD (c)) - { - continue ; - } - deg = Col [c].length ; - if (deg > dense_col_count) - { - /* this is a dense column, kill and order it last */ - Col [c].shared2.order = --n_col2 ; - /* decrement the row degrees */ - cp = &A [Col [c].start] ; - cp_end = cp + Col [c].length ; - while (cp < cp_end) - { - Row [*cp++].shared1.degree-- ; - } - KILL_PRINCIPAL_COL (c) ; - } - } - DEBUG1 (("colamd: Dense and null columns killed: %d\n", n_col - n_col2)) ; - - /* === Kill dense and empty rows ======================================== */ - - for (r = 0 ; r < n_row ; r++) - { - deg = Row [r].shared1.degree ; - ASSERT (deg >= 0 && deg <= n_col) ; - if (deg > dense_row_count || deg == 0) - { - /* kill a dense or empty row */ - KILL_ROW (r) ; - --n_row2 ; - } - else - { - /* keep track of max degree of remaining rows */ - max_deg = MAX (max_deg, deg) ; - } - } - DEBUG1 (("colamd: Dense and null rows killed: %d\n", n_row - n_row2)) ; - - /* === Compute initial column scores ==================================== */ - - /* At this point the row degrees are accurate. They reflect the number */ - /* of "live" (non-dense) columns in each row. No empty rows exist. */ - /* Some "live" columns may contain only dead rows, however. These are */ - /* pruned in the code below. */ - - /* now find the initial matlab score for each column */ - for (c = n_col-1 ; c >= 0 ; c--) - { - /* skip dead column */ - if (COL_IS_DEAD (c)) - { - continue ; - } - score = 0 ; - cp = &A [Col [c].start] ; - new_cp = cp ; - cp_end = cp + Col [c].length ; - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - /* skip if dead */ - if (ROW_IS_DEAD (row)) - { - continue ; - } - /* compact the column */ - *new_cp++ = row ; - /* add row's external degree */ - score += Row [row].shared1.degree - 1 ; - /* guard against integer overflow */ - score = MIN (score, n_col) ; - } - /* determine pruned column length */ - col_length = (int) (new_cp - &A [Col [c].start]) ; - if (col_length == 0) - { - /* a newly-made null column (all rows in this col are "dense" */ - /* and have already been killed) */ - DEBUG2 (("Newly null killed: %d\n", c)) ; - Col [c].shared2.order = --n_col2 ; - KILL_PRINCIPAL_COL (c) ; - } - else - { - /* set column length and set score */ - ASSERT (score >= 0) ; - ASSERT (score <= n_col) ; - Col [c].length = col_length ; - Col [c].shared2.score = score ; - } - } - DEBUG1 (("colamd: Dense, null, and newly-null columns killed: %d\n", - n_col-n_col2)) ; - - /* At this point, all empty rows and columns are dead. All live columns */ - /* are "clean" (containing no dead rows) and simplicial (no supercolumns */ - /* yet). Rows may contain dead columns, but all live rows contain at */ - /* least one live column. */ - -#ifndef NDEBUG - debug_structures (n_row, n_col, Row, Col, A, n_col2) ; -#endif /* NDEBUG */ - - /* === Initialize degree lists ========================================== */ - -#ifndef NDEBUG - debug_count = 0 ; -#endif /* NDEBUG */ - - /* clear the hash buckets */ - for (c = 0 ; c <= n_col ; c++) - { - head [c] = EMPTY ; - } - min_score = n_col ; - /* place in reverse order, so low column indices are at the front */ - /* of the lists. This is to encourage natural tie-breaking */ - for (c = n_col-1 ; c >= 0 ; c--) - { - /* only add principal columns to degree lists */ - if (COL_IS_ALIVE (c)) - { - DEBUG4 (("place %d score %d minscore %d ncol %d\n", - c, Col [c].shared2.score, min_score, n_col)) ; - - /* === Add columns score to DList =============================== */ - - score = Col [c].shared2.score ; - - ASSERT (min_score >= 0) ; - ASSERT (min_score <= n_col) ; - ASSERT (score >= 0) ; - ASSERT (score <= n_col) ; - ASSERT (head [score] >= EMPTY) ; - - /* now add this column to dList at proper score location */ - next_col = head [score] ; - Col [c].shared3.prev = EMPTY ; - Col [c].shared4.degree_next = next_col ; - - /* if there already was a column with the same score, set its */ - /* previous pointer to this new column */ - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = c ; - } - head [score] = c ; - - /* see if this score is less than current min */ - min_score = MIN (min_score, score) ; - -#ifndef NDEBUG - debug_count++ ; -#endif /* NDEBUG */ - - } - } - -#ifndef NDEBUG - DEBUG1 (("colamd: Live cols %d out of %d, non-princ: %d\n", - debug_count, n_col, n_col-debug_count)) ; - ASSERT (debug_count == n_col2) ; - debug_deg_lists (n_row, n_col, Row, Col, head, min_score, n_col2, max_deg) ; -#endif /* NDEBUG */ - - /* === Return number of remaining columns, and max row degree =========== */ - - *p_n_col2 = n_col2 ; - *p_n_row2 = n_row2 ; - *p_max_deg = max_deg ; -} - - -/* ========================================================================== */ -/* === find_ordering ======================================================== */ -/* ========================================================================== */ - -/* - Order the principal columns of the supercolumn form of the matrix - (no supercolumns on input). Uses a minimum approximate column minimum - degree ordering method. Not user-callable. -*/ - -PRIVATE int find_ordering /* return the number of garbage collections */ -( - /* === Parameters ======================================================= */ - - int n_row, /* number of rows of A */ - int n_col, /* number of columns of A */ - int Alen, /* size of A, 2*nnz + n_col or larger */ - Colamd_Row Row [], /* of size n_row+1 */ - Colamd_Col Col [], /* of size n_col+1 */ - int A [], /* column form and row form of A */ - int head [], /* of size n_col+1 */ - int n_col2, /* Remaining columns to order */ - int max_deg, /* Maximum row degree */ - int pfree /* index of first free slot (2*nnz on entry) */ -) -{ - /* === Local variables ================================================== */ - - int k ; /* current pivot ordering step */ - int pivot_col ; /* current pivot column */ - int *cp ; /* a column pointer */ - int *rp ; /* a row pointer */ - int pivot_row ; /* current pivot row */ - int *new_cp ; /* modified column pointer */ - int *new_rp ; /* modified row pointer */ - int pivot_row_start ; /* pointer to start of pivot row */ - int pivot_row_degree ; /* number of columns in pivot row */ - int pivot_row_length ; /* number of supercolumns in pivot row */ - int pivot_col_score ; /* score of pivot column */ - int needed_memory ; /* free space needed for pivot row */ - int *cp_end ; /* pointer to the end of a column */ - int *rp_end ; /* pointer to the end of a row */ - int row ; /* a row index */ - int col ; /* a column index */ - int max_score ; /* maximum possible score */ - int cur_score ; /* score of current column */ - unsigned int hash ; /* hash value for supernode detection */ - int head_column ; /* head of hash bucket */ - int first_col ; /* first column in hash bucket */ - int tag_mark ; /* marker value for mark array */ - int row_mark ; /* Row [row].shared2.mark */ - int set_difference ; /* set difference size of row with pivot row */ - int min_score ; /* smallest column score */ - int col_thickness ; /* "thickness" (no. of columns in a supercol) */ - int max_mark ; /* maximum value of tag_mark */ - int pivot_col_thickness ; /* number of columns represented by pivot col */ - int prev_col ; /* Used by Dlist operations. */ - int next_col ; /* Used by Dlist operations. */ - int ngarbage ; /* number of garbage collections performed */ - -#ifndef NDEBUG - int debug_d ; /* debug loop counter */ - int debug_step = 0 ; /* debug loop counter */ -#endif /* NDEBUG */ - - /* === Initialization and clear mark ==================================== */ - - max_mark = INT_MAX - n_col ; /* INT_MAX defined in */ - tag_mark = clear_mark (n_row, Row) ; - min_score = 0 ; - ngarbage = 0 ; - DEBUG1 (("colamd: Ordering, n_col2=%d\n", n_col2)) ; - - /* === Order the columns ================================================ */ - - for (k = 0 ; k < n_col2 ; /* 'k' is incremented below */) - { - -#ifndef NDEBUG - if (debug_step % 100 == 0) - { - DEBUG2 (("\n... Step k: %d out of n_col2: %d\n", k, n_col2)) ; - } - else - { - DEBUG3 (("\n----------Step k: %d out of n_col2: %d\n", k, n_col2)) ; - } - debug_step++ ; - debug_deg_lists (n_row, n_col, Row, Col, head, - min_score, n_col2-k, max_deg) ; - debug_matrix (n_row, n_col, Row, Col, A) ; -#endif /* NDEBUG */ - - /* === Select pivot column, and order it ============================ */ - - /* make sure degree list isn't empty */ - ASSERT (min_score >= 0) ; - ASSERT (min_score <= n_col) ; - ASSERT (head [min_score] >= EMPTY) ; - -#ifndef NDEBUG - for (debug_d = 0 ; debug_d < min_score ; debug_d++) - { - ASSERT (head [debug_d] == EMPTY) ; - } -#endif /* NDEBUG */ - - /* get pivot column from head of minimum degree list */ - while (head [min_score] == EMPTY && min_score < n_col) - { - min_score++ ; - } - pivot_col = head [min_score] ; - ASSERT (pivot_col >= 0 && pivot_col <= n_col) ; - next_col = Col [pivot_col].shared4.degree_next ; - head [min_score] = next_col ; - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = EMPTY ; - } - - ASSERT (COL_IS_ALIVE (pivot_col)) ; - DEBUG3 (("Pivot col: %d\n", pivot_col)) ; - - /* remember score for defrag check */ - pivot_col_score = Col [pivot_col].shared2.score ; - - /* the pivot column is the kth column in the pivot order */ - Col [pivot_col].shared2.order = k ; - - /* increment order count by column thickness */ - pivot_col_thickness = Col [pivot_col].shared1.thickness ; - k += pivot_col_thickness ; - ASSERT (pivot_col_thickness > 0) ; - - /* === Garbage_collection, if necessary ============================= */ - - needed_memory = MIN (pivot_col_score, n_col - k) ; - if (pfree + needed_memory >= Alen) - { - pfree = garbage_collection (n_row, n_col, Row, Col, A, &A [pfree]) ; - ngarbage++ ; - /* after garbage collection we will have enough */ - ASSERT (pfree + needed_memory < Alen) ; - /* garbage collection has wiped out the Row[].shared2.mark array */ - tag_mark = clear_mark (n_row, Row) ; - -#ifndef NDEBUG - debug_matrix (n_row, n_col, Row, Col, A) ; -#endif /* NDEBUG */ - } - - /* === Compute pivot row pattern ==================================== */ - - /* get starting location for this new merged row */ - pivot_row_start = pfree ; - - /* initialize new row counts to zero */ - pivot_row_degree = 0 ; - - /* tag pivot column as having been visited so it isn't included */ - /* in merged pivot row */ - Col [pivot_col].shared1.thickness = -pivot_col_thickness ; - - /* pivot row is the union of all rows in the pivot column pattern */ - cp = &A [Col [pivot_col].start] ; - cp_end = cp + Col [pivot_col].length ; - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - DEBUG4 (("Pivot col pattern %d %d\n", ROW_IS_ALIVE (row), row)) ; - /* skip if row is dead */ - if (ROW_IS_DEAD (row)) - { - continue ; - } - rp = &A [Row [row].start] ; - rp_end = rp + Row [row].length ; - while (rp < rp_end) - { - /* get a column */ - col = *rp++ ; - /* add the column, if alive and untagged */ - col_thickness = Col [col].shared1.thickness ; - if (col_thickness > 0 && COL_IS_ALIVE (col)) - { - /* tag column in pivot row */ - Col [col].shared1.thickness = -col_thickness ; - ASSERT (pfree < Alen) ; - /* place column in pivot row */ - A [pfree++] = col ; - pivot_row_degree += col_thickness ; - } - } - } - - /* clear tag on pivot column */ - Col [pivot_col].shared1.thickness = pivot_col_thickness ; - max_deg = MAX (max_deg, pivot_row_degree) ; - -#ifndef NDEBUG - DEBUG3 (("check2\n")) ; - debug_mark (n_row, Row, tag_mark, max_mark) ; -#endif /* NDEBUG */ - - /* === Kill all rows used to construct pivot row ==================== */ - - /* also kill pivot row, temporarily */ - cp = &A [Col [pivot_col].start] ; - cp_end = cp + Col [pivot_col].length ; - while (cp < cp_end) - { - /* may be killing an already dead row */ - row = *cp++ ; - DEBUG3 (("Kill row in pivot col: %d\n", row)) ; - KILL_ROW (row) ; - } - - /* === Select a row index to use as the new pivot row =============== */ - - pivot_row_length = pfree - pivot_row_start ; - if (pivot_row_length > 0) - { - /* pick the "pivot" row arbitrarily (first row in col) */ - pivot_row = A [Col [pivot_col].start] ; - DEBUG3 (("Pivotal row is %d\n", pivot_row)) ; - } - else - { - /* there is no pivot row, since it is of zero length */ - pivot_row = EMPTY ; - ASSERT (pivot_row_length == 0) ; - } - ASSERT (Col [pivot_col].length > 0 || pivot_row_length == 0) ; - - /* === Approximate degree computation =============================== */ - - /* Here begins the computation of the approximate degree. The column */ - /* score is the sum of the pivot row "length", plus the size of the */ - /* set differences of each row in the column minus the pattern of the */ - /* pivot row itself. The column ("thickness") itself is also */ - /* excluded from the column score (we thus use an approximate */ - /* external degree). */ - - /* The time taken by the following code (compute set differences, and */ - /* add them up) is proportional to the size of the data structure */ - /* being scanned - that is, the sum of the sizes of each column in */ - /* the pivot row. Thus, the amortized time to compute a column score */ - /* is proportional to the size of that column (where size, in this */ - /* context, is the column "length", or the number of row indices */ - /* in that column). The number of row indices in a column is */ - /* monotonically non-decreasing, from the length of the original */ - /* column on input to colamd. */ - - /* === Compute set differences ====================================== */ - - DEBUG3 (("** Computing set differences phase. **\n")) ; - - /* pivot row is currently dead - it will be revived later. */ - - DEBUG3 (("Pivot row: ")) ; - /* for each column in pivot row */ - rp = &A [pivot_row_start] ; - rp_end = rp + pivot_row_length ; - while (rp < rp_end) - { - col = *rp++ ; - ASSERT (COL_IS_ALIVE (col) && col != pivot_col) ; - DEBUG3 (("Col: %d\n", col)) ; - - /* clear tags used to construct pivot row pattern */ - col_thickness = -Col [col].shared1.thickness ; - ASSERT (col_thickness > 0) ; - Col [col].shared1.thickness = col_thickness ; - - /* === Remove column from degree list =========================== */ - - cur_score = Col [col].shared2.score ; - prev_col = Col [col].shared3.prev ; - next_col = Col [col].shared4.degree_next ; - ASSERT (cur_score >= 0) ; - ASSERT (cur_score <= n_col) ; - ASSERT (cur_score >= EMPTY) ; - if (prev_col == EMPTY) - { - head [cur_score] = next_col ; - } - else - { - Col [prev_col].shared4.degree_next = next_col ; - } - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = prev_col ; - } - - /* === Scan the column ========================================== */ - - cp = &A [Col [col].start] ; - cp_end = cp + Col [col].length ; - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - row_mark = Row [row].shared2.mark ; - /* skip if dead */ - if (ROW_IS_MARKED_DEAD (row_mark)) - { - continue ; - } - ASSERT (row != pivot_row) ; - set_difference = row_mark - tag_mark ; - /* check if the row has been seen yet */ - if (set_difference < 0) - { - ASSERT (Row [row].shared1.degree <= max_deg) ; - set_difference = Row [row].shared1.degree ; - } - /* subtract column thickness from this row's set difference */ - set_difference -= col_thickness ; - ASSERT (set_difference >= 0) ; - /* absorb this row if the set difference becomes zero */ - if (set_difference == 0) - { - DEBUG3 (("aggressive absorption. Row: %d\n", row)) ; - KILL_ROW (row) ; - } - else - { - /* save the new mark */ - Row [row].shared2.mark = set_difference + tag_mark ; - } - } - } - -#ifndef NDEBUG - debug_deg_lists (n_row, n_col, Row, Col, head, - min_score, n_col2-k-pivot_row_degree, max_deg) ; -#endif /* NDEBUG */ - - /* === Add up set differences for each column ======================= */ - - DEBUG3 (("** Adding set differences phase. **\n")) ; - - /* for each column in pivot row */ - rp = &A [pivot_row_start] ; - rp_end = rp + pivot_row_length ; - while (rp < rp_end) - { - /* get a column */ - col = *rp++ ; - ASSERT (COL_IS_ALIVE (col) && col != pivot_col) ; - hash = 0 ; - cur_score = 0 ; - cp = &A [Col [col].start] ; - /* compact the column */ - new_cp = cp ; - cp_end = cp + Col [col].length ; - - DEBUG4 (("Adding set diffs for Col: %d.\n", col)) ; - - while (cp < cp_end) - { - /* get a row */ - row = *cp++ ; - ASSERT(row >= 0 && row < n_row) ; - row_mark = Row [row].shared2.mark ; - /* skip if dead */ - if (ROW_IS_MARKED_DEAD (row_mark)) - { - continue ; - } - ASSERT (row_mark > tag_mark) ; - /* compact the column */ - *new_cp++ = row ; - /* compute hash function */ - hash += row ; - /* add set difference */ - cur_score += row_mark - tag_mark ; - /* integer overflow... */ - cur_score = MIN (cur_score, n_col) ; - } - - /* recompute the column's length */ - Col [col].length = (int) (new_cp - &A [Col [col].start]) ; - - /* === Further mass elimination ================================= */ - - if (Col [col].length == 0) - { - DEBUG4 (("further mass elimination. Col: %d\n", col)) ; - /* nothing left but the pivot row in this column */ - KILL_PRINCIPAL_COL (col) ; - pivot_row_degree -= Col [col].shared1.thickness ; - ASSERT (pivot_row_degree >= 0) ; - /* order it */ - Col [col].shared2.order = k ; - /* increment order count by column thickness */ - k += Col [col].shared1.thickness ; - } - else - { - /* === Prepare for supercolumn detection ==================== */ - - DEBUG4 (("Preparing supercol detection for Col: %d.\n", col)) ; - - /* save score so far */ - Col [col].shared2.score = cur_score ; - - /* add column to hash table, for supercolumn detection */ - hash %= n_col + 1 ; - - DEBUG4 ((" Hash = %d, n_col = %d.\n", hash, n_col)) ; - ASSERT (hash <= n_col) ; - - head_column = head [hash] ; - if (head_column > EMPTY) - { - /* degree list "hash" is non-empty, use prev (shared3) of */ - /* first column in degree list as head of hash bucket */ - first_col = Col [head_column].shared3.headhash ; - Col [head_column].shared3.headhash = col ; - } - else - { - /* degree list "hash" is empty, use head as hash bucket */ - first_col = - (head_column + 2) ; - head [hash] = - (col + 2) ; - } - Col [col].shared4.hash_next = first_col ; - - /* save hash function in Col [col].shared3.hash */ - Col [col].shared3.hash = (int) hash ; - ASSERT (COL_IS_ALIVE (col)) ; - } - } - - /* The approximate external column degree is now computed. */ - - /* === Supercolumn detection ======================================== */ - - DEBUG3 (("** Supercolumn detection phase. **\n")) ; - - detect_super_cols ( - -#ifndef NDEBUG - n_col, Row, -#endif /* NDEBUG */ - - Col, A, head, pivot_row_start, pivot_row_length) ; - - /* === Kill the pivotal column ====================================== */ - - KILL_PRINCIPAL_COL (pivot_col) ; - - /* === Clear mark =================================================== */ - - tag_mark += (max_deg + 1) ; - if (tag_mark >= max_mark) - { - DEBUG2 (("clearing tag_mark\n")) ; - tag_mark = clear_mark (n_row, Row) ; - } - -#ifndef NDEBUG - DEBUG3 (("check3\n")) ; - debug_mark (n_row, Row, tag_mark, max_mark) ; -#endif /* NDEBUG */ - - /* === Finalize the new pivot row, and column scores ================ */ - - DEBUG3 (("** Finalize scores phase. **\n")) ; - - /* for each column in pivot row */ - rp = &A [pivot_row_start] ; - /* compact the pivot row */ - new_rp = rp ; - rp_end = rp + pivot_row_length ; - while (rp < rp_end) - { - col = *rp++ ; - /* skip dead columns */ - if (COL_IS_DEAD (col)) - { - continue ; - } - *new_rp++ = col ; - /* add new pivot row to column */ - A [Col [col].start + (Col [col].length++)] = pivot_row ; - - /* retrieve score so far and add on pivot row's degree. */ - /* (we wait until here for this in case the pivot */ - /* row's degree was reduced due to mass elimination). */ - cur_score = Col [col].shared2.score + pivot_row_degree ; - - /* calculate the max possible score as the number of */ - /* external columns minus the 'k' value minus the */ - /* columns thickness */ - max_score = n_col - k - Col [col].shared1.thickness ; - - /* make the score the external degree of the union-of-rows */ - cur_score -= Col [col].shared1.thickness ; - - /* make sure score is less or equal than the max score */ - cur_score = MIN (cur_score, max_score) ; - ASSERT (cur_score >= 0) ; - - /* store updated score */ - Col [col].shared2.score = cur_score ; - - /* === Place column back in degree list ========================= */ - - ASSERT (min_score >= 0) ; - ASSERT (min_score <= n_col) ; - ASSERT (cur_score >= 0) ; - ASSERT (cur_score <= n_col) ; - ASSERT (head [cur_score] >= EMPTY) ; - next_col = head [cur_score] ; - Col [col].shared4.degree_next = next_col ; - Col [col].shared3.prev = EMPTY ; - if (next_col != EMPTY) - { - Col [next_col].shared3.prev = col ; - } - head [cur_score] = col ; - - /* see if this score is less than current min */ - min_score = MIN (min_score, cur_score) ; - - } - -#ifndef NDEBUG - debug_deg_lists (n_row, n_col, Row, Col, head, - min_score, n_col2-k, max_deg) ; -#endif /* NDEBUG */ - - /* === Resurrect the new pivot row ================================== */ - - if (pivot_row_degree > 0) - { - /* update pivot row length to reflect any cols that were killed */ - /* during super-col detection and mass elimination */ - Row [pivot_row].start = pivot_row_start ; - Row [pivot_row].length = (int) (new_rp - &A[pivot_row_start]) ; - Row [pivot_row].shared1.degree = pivot_row_degree ; - Row [pivot_row].shared2.mark = 0 ; - /* pivot row is no longer dead */ - } - } - - /* === All principal columns have now been ordered ====================== */ - - return (ngarbage) ; -} - - -/* ========================================================================== */ -/* === order_children ======================================================= */ -/* ========================================================================== */ - -/* - The find_ordering routine has ordered all of the principal columns (the - representatives of the supercolumns). The non-principal columns have not - yet been ordered. This routine orders those columns by walking up the - parent tree (a column is a child of the column which absorbed it). The - final permutation vector is then placed in p [0 ... n_col-1], with p [0] - being the first column, and p [n_col-1] being the last. It doesn't look - like it at first glance, but be assured that this routine takes time linear - in the number of columns. Although not immediately obvious, the time - taken by this routine is O (n_col), that is, linear in the number of - columns. Not user-callable. -*/ - -PRIVATE void order_children -( - /* === Parameters ======================================================= */ - - int n_col, /* number of columns of A */ - Colamd_Col Col [], /* of size n_col+1 */ - int p [] /* p [0 ... n_col-1] is the column permutation*/ -) -{ - /* === Local variables ================================================== */ - - int i ; /* loop counter for all columns */ - int c ; /* column index */ - int parent ; /* index of column's parent */ - int order ; /* column's order */ - - /* === Order each non-principal column ================================== */ - - for (i = 0 ; i < n_col ; i++) - { - /* find an un-ordered non-principal column */ - ASSERT (COL_IS_DEAD (i)) ; - if (!COL_IS_DEAD_PRINCIPAL (i) && Col [i].shared2.order == EMPTY) - { - parent = i ; - /* once found, find its principal parent */ - do - { - parent = Col [parent].shared1.parent ; - } while (!COL_IS_DEAD_PRINCIPAL (parent)) ; - - /* now, order all un-ordered non-principal columns along path */ - /* to this parent. collapse tree at the same time */ - c = i ; - /* get order of parent */ - order = Col [parent].shared2.order ; - - do - { - ASSERT (Col [c].shared2.order == EMPTY) ; - - /* order this column */ - Col [c].shared2.order = order++ ; - /* collaps tree */ - Col [c].shared1.parent = parent ; - - /* get immediate parent of this column */ - c = Col [c].shared1.parent ; - - /* continue until we hit an ordered column. There are */ - /* guarranteed not to be anymore unordered columns */ - /* above an ordered column */ - } while (Col [c].shared2.order == EMPTY) ; - - /* re-order the super_col parent to largest order for this group */ - Col [parent].shared2.order = order ; - } - } - - /* === Generate the permutation ========================================= */ - - for (c = 0 ; c < n_col ; c++) - { - p [Col [c].shared2.order] = c ; - } -} - - -/* ========================================================================== */ -/* === detect_super_cols ==================================================== */ -/* ========================================================================== */ - -/* - Detects supercolumns by finding matches between columns in the hash buckets. - Check amongst columns in the set A [row_start ... row_start + row_length-1]. - The columns under consideration are currently *not* in the degree lists, - and have already been placed in the hash buckets. - - The hash bucket for columns whose hash function is equal to h is stored - as follows: - - if head [h] is >= 0, then head [h] contains a degree list, so: - - head [h] is the first column in degree bucket h. - Col [head [h]].headhash gives the first column in hash bucket h. - - otherwise, the degree list is empty, and: - - -(head [h] + 2) is the first column in hash bucket h. - - For a column c in a hash bucket, Col [c].shared3.prev is NOT a "previous - column" pointer. Col [c].shared3.hash is used instead as the hash number - for that column. The value of Col [c].shared4.hash_next is the next column - in the same hash bucket. - - Assuming no, or "few" hash collisions, the time taken by this routine is - linear in the sum of the sizes (lengths) of each column whose score has - just been computed in the approximate degree computation. - Not user-callable. -*/ - -PRIVATE void detect_super_cols -( - /* === Parameters ======================================================= */ - -#ifndef NDEBUG - /* these two parameters are only needed when debugging is enabled: */ - int n_col, /* number of columns of A */ - Colamd_Row Row [], /* of size n_row+1 */ -#endif /* NDEBUG */ - - Colamd_Col Col [], /* of size n_col+1 */ - int A [], /* row indices of A */ - int head [], /* head of degree lists and hash buckets */ - int row_start, /* pointer to set of columns to check */ - int row_length /* number of columns to check */ -) -{ - /* === Local variables ================================================== */ - - int hash ; /* hash value for a column */ - int *rp ; /* pointer to a row */ - int c ; /* a column index */ - int super_c ; /* column index of the column to absorb into */ - int *cp1 ; /* column pointer for column super_c */ - int *cp2 ; /* column pointer for column c */ - int length ; /* length of column super_c */ - int prev_c ; /* column preceding c in hash bucket */ - int i ; /* loop counter */ - int *rp_end ; /* pointer to the end of the row */ - int col ; /* a column index in the row to check */ - int head_column ; /* first column in hash bucket or degree list */ - int first_col ; /* first column in hash bucket */ - - /* === Consider each column in the row ================================== */ - - rp = &A [row_start] ; - rp_end = rp + row_length ; - while (rp < rp_end) - { - col = *rp++ ; - if (COL_IS_DEAD (col)) - { - continue ; - } - - /* get hash number for this column */ - hash = Col [col].shared3.hash ; - ASSERT (hash <= n_col) ; - - /* === Get the first column in this hash bucket ===================== */ - - head_column = head [hash] ; - if (head_column > EMPTY) - { - first_col = Col [head_column].shared3.headhash ; - } - else - { - first_col = - (head_column + 2) ; - } - - /* === Consider each column in the hash bucket ====================== */ - - for (super_c = first_col ; super_c != EMPTY ; - super_c = Col [super_c].shared4.hash_next) - { - ASSERT (COL_IS_ALIVE (super_c)) ; - ASSERT (Col [super_c].shared3.hash == hash) ; - length = Col [super_c].length ; - - /* prev_c is the column preceding column c in the hash bucket */ - prev_c = super_c ; - - /* === Compare super_c with all columns after it ================ */ - - for (c = Col [super_c].shared4.hash_next ; - c != EMPTY ; c = Col [c].shared4.hash_next) - { - ASSERT (c != super_c) ; - ASSERT (COL_IS_ALIVE (c)) ; - ASSERT (Col [c].shared3.hash == hash) ; - - /* not identical if lengths or scores are different */ - if (Col [c].length != length || - Col [c].shared2.score != Col [super_c].shared2.score) - { - prev_c = c ; - continue ; - } - - /* compare the two columns */ - cp1 = &A [Col [super_c].start] ; - cp2 = &A [Col [c].start] ; - - for (i = 0 ; i < length ; i++) - { - /* the columns are "clean" (no dead rows) */ - ASSERT (ROW_IS_ALIVE (*cp1)) ; - ASSERT (ROW_IS_ALIVE (*cp2)) ; - /* row indices will same order for both supercols, */ - /* no gather scatter nessasary */ - if (*cp1++ != *cp2++) - { - break ; - } - } - - /* the two columns are different if the for-loop "broke" */ - if (i != length) - { - prev_c = c ; - continue ; - } - - /* === Got it! two columns are identical =================== */ - - ASSERT (Col [c].shared2.score == Col [super_c].shared2.score) ; - - Col [super_c].shared1.thickness += Col [c].shared1.thickness ; - Col [c].shared1.parent = super_c ; - KILL_NON_PRINCIPAL_COL (c) ; - /* order c later, in order_children() */ - Col [c].shared2.order = EMPTY ; - /* remove c from hash bucket */ - Col [prev_c].shared4.hash_next = Col [c].shared4.hash_next ; - } - } - - /* === Empty this hash bucket ======================================= */ - - if (head_column > EMPTY) - { - /* corresponding degree list "hash" is not empty */ - Col [head_column].shared3.headhash = EMPTY ; - } - else - { - /* corresponding degree list "hash" is empty */ - head [hash] = EMPTY ; - } - } -} - - -/* ========================================================================== */ -/* === garbage_collection =================================================== */ -/* ========================================================================== */ - -/* - Defragments and compacts columns and rows in the workspace A. Used when - all avaliable memory has been used while performing row merging. Returns - the index of the first free position in A, after garbage collection. The - time taken by this routine is linear is the size of the array A, which is - itself linear in the number of nonzeros in the input matrix. - Not user-callable. -*/ - -PRIVATE int garbage_collection /* returns the new value of pfree */ -( - /* === Parameters ======================================================= */ - - int n_row, /* number of rows */ - int n_col, /* number of columns */ - Colamd_Row Row [], /* row info */ - Colamd_Col Col [], /* column info */ - int A [], /* A [0 ... Alen-1] holds the matrix */ - int *pfree /* &A [0] ... pfree is in use */ -) -{ - /* === Local variables ================================================== */ - - int *psrc ; /* source pointer */ - int *pdest ; /* destination pointer */ - int j ; /* counter */ - int r ; /* a row index */ - int c ; /* a column index */ - int length ; /* length of a row or column */ - -#ifndef NDEBUG - int debug_rows ; - DEBUG2 (("Defrag..\n")) ; - for (psrc = &A[0] ; psrc < pfree ; psrc++) ASSERT (*psrc >= 0) ; - debug_rows = 0 ; -#endif /* NDEBUG */ - - /* === Defragment the columns =========================================== */ - - pdest = &A[0] ; - for (c = 0 ; c < n_col ; c++) - { - if (COL_IS_ALIVE (c)) - { - psrc = &A [Col [c].start] ; - - /* move and compact the column */ - ASSERT (pdest <= psrc) ; - Col [c].start = (int) (pdest - &A [0]) ; - length = Col [c].length ; - for (j = 0 ; j < length ; j++) - { - r = *psrc++ ; - if (ROW_IS_ALIVE (r)) - { - *pdest++ = r ; - } - } - Col [c].length = (int) (pdest - &A [Col [c].start]) ; - } - } - - /* === Prepare to defragment the rows =================================== */ - - for (r = 0 ; r < n_row ; r++) - { - if (ROW_IS_ALIVE (r)) - { - if (Row [r].length == 0) - { - /* this row is of zero length. cannot compact it, so kill it */ - DEBUG3 (("Defrag row kill\n")) ; - KILL_ROW (r) ; - } - else - { - /* save first column index in Row [r].shared2.first_column */ - psrc = &A [Row [r].start] ; - Row [r].shared2.first_column = *psrc ; - ASSERT (ROW_IS_ALIVE (r)) ; - /* flag the start of the row with the one's complement of row */ - *psrc = ONES_COMPLEMENT (r) ; - -#ifndef NDEBUG - debug_rows++ ; -#endif /* NDEBUG */ - - } - } - } - - /* === Defragment the rows ============================================== */ - - psrc = pdest ; - while (psrc < pfree) - { - /* find a negative number ... the start of a row */ - if (*psrc++ < 0) - { - psrc-- ; - /* get the row index */ - r = ONES_COMPLEMENT (*psrc) ; - ASSERT (r >= 0 && r < n_row) ; - /* restore first column index */ - *psrc = Row [r].shared2.first_column ; - ASSERT (ROW_IS_ALIVE (r)) ; - - /* move and compact the row */ - ASSERT (pdest <= psrc) ; - Row [r].start = (int) (pdest - &A [0]) ; - length = Row [r].length ; - for (j = 0 ; j < length ; j++) - { - c = *psrc++ ; - if (COL_IS_ALIVE (c)) - { - *pdest++ = c ; - } - } - Row [r].length = (int) (pdest - &A [Row [r].start]) ; - -#ifndef NDEBUG - debug_rows-- ; -#endif /* NDEBUG */ - - } - } - /* ensure we found all the rows */ - ASSERT (debug_rows == 0) ; - - /* === Return the new value of pfree ==================================== */ - - return ((int) (pdest - &A [0])) ; -} - - -/* ========================================================================== */ -/* === clear_mark =========================================================== */ -/* ========================================================================== */ - -/* - Clears the Row [].shared2.mark array, and returns the new tag_mark. - Return value is the new tag_mark. Not user-callable. -*/ - -PRIVATE int clear_mark /* return the new value for tag_mark */ -( - /* === Parameters ======================================================= */ - - int n_row, /* number of rows in A */ - Colamd_Row Row [] /* Row [0 ... n_row-1].shared2.mark is set to zero */ -) -{ - /* === Local variables ================================================== */ - - int r ; - - for (r = 0 ; r < n_row ; r++) - { - if (ROW_IS_ALIVE (r)) - { - Row [r].shared2.mark = 0 ; - } - } - return (1) ; -} - - -/* ========================================================================== */ -/* === print_report ========================================================= */ -/* ========================================================================== */ - -PRIVATE void print_report -( - char *method, - int stats [COLAMD_STATS] -) -{ - - int i1, i2, i3 ; - - if (!stats) - { - PRINTF ("%s: No statistics available.\n", method) ; - return ; - } - - i1 = stats [COLAMD_INFO1] ; - i2 = stats [COLAMD_INFO2] ; - i3 = stats [COLAMD_INFO3] ; - - if (stats [COLAMD_STATUS] >= 0) - { - PRINTF ("%s: OK. ", method) ; - } - else - { - PRINTF ("%s: ERROR. ", method) ; - } - - switch (stats [COLAMD_STATUS]) - { - - case COLAMD_OK_BUT_JUMBLED: - - PRINTF ("Matrix has unsorted or duplicate row indices.\n") ; - - PRINTF ("%s: number of duplicate or out-of-order row indices: %d\n", - method, i3) ; - - PRINTF ("%s: last seen duplicate or out-of-order row index: %d\n", - method, INDEX (i2)) ; - - PRINTF ("%s: last seen in column: %d", - method, INDEX (i1)) ; - - /* no break - fall through to next case instead */ - - case COLAMD_OK: - - PRINTF ("\n") ; - - PRINTF ("%s: number of dense or empty rows ignored: %d\n", - method, stats [COLAMD_DENSE_ROW]) ; - - PRINTF ("%s: number of dense or empty columns ignored: %d\n", - method, stats [COLAMD_DENSE_COL]) ; - - PRINTF ("%s: number of garbage collections performed: %d\n", - method, stats [COLAMD_DEFRAG_COUNT]) ; - break ; - - case COLAMD_ERROR_A_not_present: - - PRINTF ("Array A (row indices of matrix) not present.\n") ; - break ; - - case COLAMD_ERROR_p_not_present: - - PRINTF ("Array p (column pointers for matrix) not present.\n") ; - break ; - - case COLAMD_ERROR_nrow_negative: - - PRINTF ("Invalid number of rows (%d).\n", i1) ; - break ; - - case COLAMD_ERROR_ncol_negative: - - PRINTF ("Invalid number of columns (%d).\n", i1) ; - break ; - - case COLAMD_ERROR_nnz_negative: - - PRINTF ("Invalid number of nonzero entries (%d).\n", i1) ; - break ; - - case COLAMD_ERROR_p0_nonzero: - - PRINTF ("Invalid column pointer, p [0] = %d, must be zero.\n", i1) ; - break ; - - case COLAMD_ERROR_A_too_small: - - PRINTF ("Array A too small.\n") ; - PRINTF (" Need Alen >= %d, but given only Alen = %d.\n", - i1, i2) ; - break ; - - case COLAMD_ERROR_col_length_negative: - - PRINTF - ("Column %d has a negative number of nonzero entries (%d).\n", - INDEX (i1), i2) ; - break ; - - case COLAMD_ERROR_row_index_out_of_bounds: - - PRINTF - ("Row index (row %d) out of bounds (%d to %d) in column %d.\n", - INDEX (i2), INDEX (0), INDEX (i3-1), INDEX (i1)) ; - break ; - - case COLAMD_ERROR_out_of_memory: - - PRINTF ("Out of memory.\n") ; - break ; - - case COLAMD_ERROR_internal_error: - - /* if this happens, there is a bug in the code */ - PRINTF - ("Internal error! Please contact authors (davis@cise.ufl.edu).\n") ; - break ; - } -} - - - - -/* ========================================================================== */ -/* === colamd debugging routines ============================================ */ -/* ========================================================================== */ - -/* When debugging is disabled, the remainder of this file is ignored. */ - -#ifndef NDEBUG - - -/* ========================================================================== */ -/* === debug_structures ===================================================== */ -/* ========================================================================== */ - -/* - At this point, all empty rows and columns are dead. All live columns - are "clean" (containing no dead rows) and simplicial (no supercolumns - yet). Rows may contain dead columns, but all live rows contain at - least one live column. -*/ - -PRIVATE void debug_structures -( - /* === Parameters ======================================================= */ - - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [], - int n_col2 -) -{ - /* === Local variables ================================================== */ - - int i ; - int c ; - int *cp ; - int *cp_end ; - int len ; - int score ; - int r ; - int *rp ; - int *rp_end ; - int deg ; - - /* === Check A, Row, and Col ============================================ */ - - for (c = 0 ; c < n_col ; c++) - { - if (COL_IS_ALIVE (c)) - { - len = Col [c].length ; - score = Col [c].shared2.score ; - DEBUG4 (("initial live col %5d %5d %5d\n", c, len, score)) ; - ASSERT (len > 0) ; - ASSERT (score >= 0) ; - ASSERT (Col [c].shared1.thickness == 1) ; - cp = &A [Col [c].start] ; - cp_end = cp + len ; - while (cp < cp_end) - { - r = *cp++ ; - ASSERT (ROW_IS_ALIVE (r)) ; - } - } - else - { - i = Col [c].shared2.order ; - ASSERT (i >= n_col2 && i < n_col) ; - } - } - - for (r = 0 ; r < n_row ; r++) - { - if (ROW_IS_ALIVE (r)) - { - i = 0 ; - len = Row [r].length ; - deg = Row [r].shared1.degree ; - ASSERT (len > 0) ; - ASSERT (deg > 0) ; - rp = &A [Row [r].start] ; - rp_end = rp + len ; - while (rp < rp_end) - { - c = *rp++ ; - if (COL_IS_ALIVE (c)) - { - i++ ; - } - } - ASSERT (i > 0) ; - } - } -} - - -/* ========================================================================== */ -/* === debug_deg_lists ====================================================== */ -/* ========================================================================== */ - -/* - Prints the contents of the degree lists. Counts the number of columns - in the degree list and compares it to the total it should have. Also - checks the row degrees. -*/ - -PRIVATE void debug_deg_lists -( - /* === Parameters ======================================================= */ - - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int head [], - int min_score, - int should, - int max_deg -) -{ - /* === Local variables ================================================== */ - - int deg ; - int col ; - int have ; - int row ; - - /* === Check the degree lists =========================================== */ - - if (n_col > 10000 && colamd_debug <= 0) - { - return ; - } - have = 0 ; - DEBUG4 (("Degree lists: %d\n", min_score)) ; - for (deg = 0 ; deg <= n_col ; deg++) - { - col = head [deg] ; - if (col == EMPTY) - { - continue ; - } - DEBUG4 (("%d:", deg)) ; - while (col != EMPTY) - { - DEBUG4 ((" %d", col)) ; - have += Col [col].shared1.thickness ; - ASSERT (COL_IS_ALIVE (col)) ; - col = Col [col].shared4.degree_next ; - } - DEBUG4 (("\n")) ; - } - DEBUG4 (("should %d have %d\n", should, have)) ; - ASSERT (should == have) ; - - /* === Check the row degrees ============================================ */ - - if (n_row > 10000 && colamd_debug <= 0) - { - return ; - } - for (row = 0 ; row < n_row ; row++) - { - if (ROW_IS_ALIVE (row)) - { - ASSERT (Row [row].shared1.degree <= max_deg) ; - } - } -} - - -/* ========================================================================== */ -/* === debug_mark =========================================================== */ -/* ========================================================================== */ - -/* - Ensures that the tag_mark is less that the maximum and also ensures that - each entry in the mark array is less than the tag mark. -*/ - -PRIVATE void debug_mark -( - /* === Parameters ======================================================= */ - - int n_row, - Colamd_Row Row [], - int tag_mark, - int max_mark -) -{ - /* === Local variables ================================================== */ - - int r ; - - /* === Check the Row marks ============================================== */ - - ASSERT (tag_mark > 0 && tag_mark <= max_mark) ; - if (n_row > 10000 && colamd_debug <= 0) - { - return ; - } - for (r = 0 ; r < n_row ; r++) - { - ASSERT (Row [r].shared2.mark < tag_mark) ; - } -} - - -/* ========================================================================== */ -/* === debug_matrix ========================================================= */ -/* ========================================================================== */ - -/* - Prints out the contents of the columns and the rows. -*/ - -PRIVATE void debug_matrix -( - /* === Parameters ======================================================= */ - - int n_row, - int n_col, - Colamd_Row Row [], - Colamd_Col Col [], - int A [] -) -{ - /* === Local variables ================================================== */ - - int r ; - int c ; - int *rp ; - int *rp_end ; - int *cp ; - int *cp_end ; - - /* === Dump the rows and columns of the matrix ========================== */ - - if (colamd_debug < 3) - { - return ; - } - DEBUG3 (("DUMP MATRIX:\n")) ; - for (r = 0 ; r < n_row ; r++) - { - DEBUG3 (("Row %d alive? %d\n", r, ROW_IS_ALIVE (r))) ; - if (ROW_IS_DEAD (r)) - { - continue ; - } - DEBUG3 (("start %d length %d degree %d\n", - Row [r].start, Row [r].length, Row [r].shared1.degree)) ; - rp = &A [Row [r].start] ; - rp_end = rp + Row [r].length ; - while (rp < rp_end) - { - c = *rp++ ; - DEBUG4 ((" %d col %d\n", COL_IS_ALIVE (c), c)) ; - } - } - - for (c = 0 ; c < n_col ; c++) - { - DEBUG3 (("Col %d alive? %d\n", c, COL_IS_ALIVE (c))) ; - if (COL_IS_DEAD (c)) - { - continue ; - } - DEBUG3 (("start %d length %d shared1 %d shared2 %d\n", - Col [c].start, Col [c].length, - Col [c].shared1.thickness, Col [c].shared2.score)) ; - cp = &A [Col [c].start] ; - cp_end = cp + Col [c].length ; - while (cp < cp_end) - { - r = *cp++ ; - DEBUG4 ((" %d row %d\n", ROW_IS_ALIVE (r), r)) ; - } - } -} - -PRIVATE void colamd_get_debug -( - char *method -) -{ - colamd_debug = 0 ; /* no debug printing */ - - /* get "D" environment variable, which gives the debug printing level */ - if (getenv ("D")) - { - colamd_debug = atoi (getenv ("D")) ; - } - - DEBUG0 (("%s: debug version, D = %d (THIS WILL BE SLOW!)\n", - method, colamd_debug)) ; -} - -#endif /* NDEBUG */ - diff --git a/code/3rd_lpsolve/colamd/colamd.h b/code/3rd_lpsolve/colamd/colamd.h deleted file mode 100644 index e8539922..00000000 --- a/code/3rd_lpsolve/colamd/colamd.h +++ /dev/null @@ -1,286 +0,0 @@ -/* ========================================================================== */ -/* === colamd/symamd prototypes and definitions ============================= */ -/* ========================================================================== */ - -/* - You must include this file (colamd.h) in any routine that uses colamd, - symamd, or the related macros and definitions. - - Authors: - - The authors of the code itself are Stefan I. Larimore and Timothy A. - Davis (davis@cise.ufl.edu), University of Florida. The algorithm was - developed in collaboration with John Gilbert, Xerox PARC, and Esmond - Ng, Oak Ridge National Laboratory. - - Date: - - May 4, 2001. Version 2.1. - - Acknowledgements: - - This work was supported by the National Science Foundation, under - grants DMS-9504974 and DMS-9803599. - - Notice: - - Copyright (c) 1998-2001 by the University of Florida. - All Rights Reserved. - - THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY - EXPRESSED OR IMPLIED. ANY USE IS AT YOUR OWN RISK. - - Permission is hereby granted to use or copy this program for any - purpose, provided the above notices are retained on all copies. - User documentation of any code that uses this code must cite the - Authors, the Copyright, and "Used by permission." If this code is - accessible from within Matlab, then typing "help colamd" and "help - symamd" must cite the Authors. Permission to modify the code and to - distribute modified code is granted, provided the above notices are - retained, and a notice that the code was modified is included with the - above copyright notice. You must also retain the Availability - information below, of the original version. - - This software is provided free of charge. - - Availability: - - The colamd/symamd library is available at - - http://www.cise.ufl.edu/research/sparse/colamd - - This is the http://www.cise.ufl.edu/research/sparse/colamd/colamd.h - file. It is required by the colamd.c, colamdmex.c, and symamdmex.c - files, and by any C code that calls the routines whose prototypes are - listed below, or that uses the colamd/symamd definitions listed below. - -*/ - -#ifndef COLAMD_H -#define COLAMD_H - -/* ========================================================================== */ -/* === Include files ======================================================== */ -/* ========================================================================== */ - -#include - -/* ========================================================================== */ -/* === Knob and statistics definitions ====================================== */ -/* ========================================================================== */ - -/* size of the knobs [ ] array. Only knobs [0..1] are currently used. */ -#define COLAMD_KNOBS 20 - -/* number of output statistics. Only stats [0..6] are currently used. */ -#define COLAMD_STATS 20 - -/* knobs [0] and stats [0]: dense row knob and output statistic. */ -#define COLAMD_DENSE_ROW 0 - -/* knobs [1] and stats [1]: dense column knob and output statistic. */ -#define COLAMD_DENSE_COL 1 - -/* stats [2]: memory defragmentation count output statistic */ -#define COLAMD_DEFRAG_COUNT 2 - -/* stats [3]: colamd status: zero OK, > 0 warning or notice, < 0 error */ -#define COLAMD_STATUS 3 - -/* stats [4..6]: error info, or info on jumbled columns */ -#define COLAMD_INFO1 4 -#define COLAMD_INFO2 5 -#define COLAMD_INFO3 6 - -/* error codes returned in stats [3]: */ -#define COLAMD_OK (0) -#define COLAMD_OK_BUT_JUMBLED (1) -#define COLAMD_ERROR_A_not_present (-1) -#define COLAMD_ERROR_p_not_present (-2) -#define COLAMD_ERROR_nrow_negative (-3) -#define COLAMD_ERROR_ncol_negative (-4) -#define COLAMD_ERROR_nnz_negative (-5) -#define COLAMD_ERROR_p0_nonzero (-6) -#define COLAMD_ERROR_A_too_small (-7) -#define COLAMD_ERROR_col_length_negative (-8) -#define COLAMD_ERROR_row_index_out_of_bounds (-9) -#define COLAMD_ERROR_out_of_memory (-10) -#define COLAMD_ERROR_internal_error (-999) - - - -/* ========================================================================== */ -/* === Row and Column structures ============================================ */ -/* ========================================================================== */ - -/* User code that makes use of the colamd/symamd routines need not directly */ -/* reference these structures. They are used only for the COLAMD_RECOMMENDED */ -/* macro. */ - -typedef struct Colamd_Col_struct -{ - int start ; /* index for A of first row in this column, or DEAD */ - /* if column is dead */ - int length ; /* number of rows in this column */ - union - { - int thickness ; /* number of original columns represented by this */ - /* col, if the column is alive */ - int parent ; /* parent in parent tree super-column structure, if */ - /* the column is dead */ - } shared1 ; - union - { - int score ; /* the score used to maintain heap, if col is alive */ - int order ; /* pivot ordering of this column, if col is dead */ - } shared2 ; - union - { - int headhash ; /* head of a hash bucket, if col is at the head of */ - /* a degree list */ - int hash ; /* hash value, if col is not in a degree list */ - int prev ; /* previous column in degree list, if col is in a */ - /* degree list (but not at the head of a degree list) */ - } shared3 ; - union - { - int degree_next ; /* next column, if col is in a degree list */ - int hash_next ; /* next column, if col is in a hash list */ - } shared4 ; - -} Colamd_Col ; - -typedef struct Colamd_Row_struct -{ - int start ; /* index for A of first col in this row */ - int length ; /* number of principal columns in this row */ - union - { - int degree ; /* number of principal & non-principal columns in row */ - int p ; /* used as a row pointer in init_rows_cols () */ - } shared1 ; - union - { - int mark ; /* for computing set differences and marking dead rows*/ - int first_column ;/* first column in row (used in garbage collection) */ - } shared2 ; - -} Colamd_Row ; - -/* ========================================================================== */ -/* === Colamd recommended memory size ======================================= */ -/* ========================================================================== */ - -/* - The recommended length Alen of the array A passed to colamd is given by - the COLAMD_RECOMMENDED (nnz, n_row, n_col) macro. It returns -1 if any - argument is negative. 2*nnz space is required for the row and column - indices of the matrix. COLAMD_C (n_col) + COLAMD_R (n_row) space is - required for the Col and Row arrays, respectively, which are internal to - colamd. An additional n_col space is the minimal amount of "elbow room", - and nnz/5 more space is recommended for run time efficiency. - - This macro is not needed when using symamd. -*/ - -#define COLAMD_C(n_col) (((n_col) + 1) * sizeof (Colamd_Col) / sizeof (int)) -#define COLAMD_R(n_row) (((n_row) + 1) * sizeof (Colamd_Row) / sizeof (int)) - -#define COLAMD_RECOMMENDED(nnz, n_row, n_col) \ -( \ -((nnz) < 0 || (n_row) < 0 || (n_col) < 0) \ -? \ - (-1) \ -: \ - (int) (2 * (nnz) + COLAMD_C (n_col) + COLAMD_R (n_row) + (n_col) + ((nnz) / 5)) \ -) - -/* ========================================================================== */ -/* === Prototypes of user-callable routines ================================= */ -/* ========================================================================== */ - -/* -#ifdef __cplusplus - #define __EXTERN_C extern "C" -#else - #define __EXTERN_C -#endif -*/ - -#ifndef __BORLANDC__ - - #ifdef __cplusplus - #define __EXTERN_C extern "C" - #else - #define __EXTERN_C - #endif - -#else /* Otherwise set up for the Borland compiler */ - - #define __EXTERN_C extern "C" - -#endif - -#ifdef __cplusplus -__EXTERN_C { -#endif - - - -int colamd_recommended /* returns recommended value of Alen, */ - /* or (-1) if input arguments are erroneous */ -( - int nnz, /* nonzeros in A */ - int n_row, /* number of rows in A */ - int n_col /* number of columns in A */ -) ; - -void colamd_set_defaults /* sets default parameters */ -( /* knobs argument is modified on output */ - double knobs [COLAMD_KNOBS] /* parameter settings for colamd */ -) ; - -int colamd /* returns (1) if successful, (0) otherwise*/ -( /* A and p arguments are modified on output */ - int n_row, /* number of rows in A */ - int n_col, /* number of columns in A */ - int Alen, /* size of the array A */ - int A [], /* row indices of A, of size Alen */ - int p [], /* column pointers of A, of size n_col+1 */ - double knobs [COLAMD_KNOBS],/* parameter settings for colamd */ - int stats [COLAMD_STATS] /* colamd output statistics and error codes */ -) ; - -int symamd /* return (1) if OK, (0) otherwise */ -( - int n, /* number of rows and columns of A */ - int A [], /* row indices of A */ - int p [], /* column pointers of A */ - int perm [], /* output permutation, size n_col+1 */ - double knobs [COLAMD_KNOBS], /* parameters (uses defaults if NULL) */ - int stats [COLAMD_STATS], /* output statistics and error codes */ - void * (*allocate) (size_t, size_t), - /* pointer to calloc (ANSI C) or */ - /* mxCalloc (for Matlab mexFunction) */ - void (*release) (void *) - /* pointer to free (ANSI C) or */ - /* mxFree (for Matlab mexFunction) */ -) ; - -void colamd_report -( - int stats [COLAMD_STATS] -) ; - -void symamd_report -( - int stats [COLAMD_STATS] -) ; - -#endif /* COLAMD_H */ - - -#ifdef __cplusplus -} -#endif - diff --git a/code/3rd_lpsolve/fortify.c b/code/3rd_lpsolve/fortify.c deleted file mode 100644 index 72370ced..00000000 --- a/code/3rd_lpsolve/fortify.c +++ /dev/null @@ -1,2352 +0,0 @@ -/* - * FILE: - * fortify.c - * - * DESCRIPTION: - * A fortified shell for malloc, realloc, calloc, strdup, getcwd, tempnam - * and free. - * To use Fortify, each source file will need to #include "fortify.h". To - * enable Fortify, define the symbol FORTIFY. If FORTIFY is not defined, it - * will compile away to nothing. If you do not have stderr available, you may - * wish to set an alternate output function. See _Fortify_SetOutputFunc(), - * below. - * You will also need to link in fortify.o - * - * None of the functions in this file should really be called - * directly; they really should be called through the macros - * defined in fortify.h - * - */ - -#if defined FORTIFY - -#include -#include -#include -#include - -#if defined MSDOS || defined __BORLANDC__ || defined WIN32 || defined __HIGHC__ -# include -#endif - -extern int EndOfPgr(int); - -#define __FORTIFY_C__ /* So fortify.h knows to not define the fortify macros */ -#include "fortify.h" - -#include "ufortify.h" /* the user's options */ - -char *_Fortify_file=NULL; -int _Fortify_line=0; - -#ifndef FORTIFY_TRANSPARENT - -#include -#include -#include -#include - -#if defined MSDOS || defined __BORLANDC__ || defined WIN32 || defined __HIGHC__ -# if !defined WIN32 -# undef MSDOS -# define MSDOS -# endif -# include -#else -# include -# include -# define getch() getchar() -#endif - -#if defined _WINDOWS -# include "windows.h" -# if !defined WIN32 -# include "toolhelp.h" -# endif -#endif - -#if defined LONGNAME -# include "longname.h" -#endif - -#if !defined MIN -# define MIN(a, b) ((a) < (b) ? (a) : (b)) -#endif - -struct Header -{ - char *File; /* The sourcefile of the caller */ - unsigned short Line; /* The sourceline of the caller */ - size_t Size; /* The size of the malloc'd block */ - struct Header *Prev, /* List pointers */ - *Next; - int Checksum; /* For validating the Header structure; see ChecksumHeader() */ - unsigned char Scope; -}; - -#if defined AViiON || defined __GNUC__ || defined _MSC_VER -# define _static static -#else -# define _static -#endif - -_static char *address __OF((void *addr)); -_static int TimeToCheck __OF((void)); -_static int CheckBlock __OF((struct Header *h, char *file, unsigned long line)); -_static int CheckPointer __OF((unsigned char *ptr, unsigned long size, char *file, unsigned long line)); -_static int CheckFortification __OF((unsigned char *ptr, unsigned char value, size_t size)); -_static void SetFortification __OF((unsigned char *ptr, unsigned char value, size_t size)); -_static void OutputFortification __OF((unsigned char *ptr, unsigned char value, size_t size)); -_static int IsHeaderValid __OF((struct Header *h)); -_static void MakeHeaderValid __OF((struct Header *h)); -_static int ChecksumHeader __OF((struct Header *h)); -_static int IsOnList __OF((struct Header *h)); -_static void OutputHeader __OF((struct Header *h)); -_static void OutputMemory __OF((struct Header *h)); -_static void st_DefaultOutput __OF((char *String)); -_static void WaitIfstdOutput __OF((void)); - -static char stdOutput = 0; /* If true, did some stderr output */ -static OutputFuncPtr st_Output = st_DefaultOutput; /* Output function for errors */ - -#if !defined MSDOS && !defined WIN32 -static int strnicmp(s1,s2,maxlen) - char *s1,*s2; - size_t maxlen; - { - while ((maxlen) && (*s1) && (*s2) && (toupper(*s1)==toupper(*s2))) { - maxlen--; - s1++; - s2++; - } - return((maxlen) ? toupper(*s1)-toupper(*s2) : 0); - } - -static int stricmp(s1,s2) - char *s1,*s2; - { - return(strnicmp(s1,s2,strlen(s1)+1)); - } -#endif - -static char *address(void *addr) -{ - static char str[80]; - -#if defined KNOWS_POINTER_TYPE - sprintf(str,"%p",addr); -#else - sprintf(str,"%lx",(unsigned long) addr); -#endif - return(str); -} - -#ifdef FORTIFY_CheckInterval -int TimeToCheck() -{ - static time_t lastcheck=0L; - time_t t; - int ret = 0; - - time(&t); - if ((lastcheck==0L) || (t-lastcheck>=FORTIFY_CheckInterval)) - { - lastcheck = t; - ret = 1; - } - return(ret); -} -#endif - -static FILE *gfile=NULL; -static int Nchars=0,Nlines=0; -static char flag=0; - -static void _Fortify_NoOutput() - { - } - -static void st_DefaultOutput(char *String) -{ - static FILE *file; - static char first=1; - - if (first) { - file=stderr; - first=0; - } - - if (stdOutput==0) { - Nchars=Nlines=0; - if (gfile!=NULL) rewind(gfile); - } - - if (flag==0) - { - char *ptr; - - file=stderr; - flag = 1; - if ((ptr=getenv("FORTIFY_OUTPUT"))!=NULL) - { - if ((stricmp(ptr,"null")==0) || (stricmp(ptr,"nul")==0)) - file=NULL; - else if (stricmp(ptr,"stderr")==0) - file=stderr; - else if (stricmp(ptr,"stdout")==0) - file=stdout; -#if defined MSDOS && !defined _WINDOWS && !defined WIN32 - else if (stricmp(ptr,"stdprn")==0) - file=stdprn; -#endif - else if ((file=fopen(ptr,"w"))==NULL) - { -#if !defined _WINDOWS - fprintf(stderr,"\r\nFortify: Unable to create logfile %s\r\n",ptr); - EndOfPgr(4); -#else - { - char str[255]; - - sprintf(str,"Unable to create logfile\n \"%s\"",ptr); - MessageBox((HWND) NULL,(LPCSTR) str,(LPCSTR) "Fortify",(UINT) MB_ICONSTOP); -#if 0 -#if defined WIN32 - /* TerminateProcess(GetCurrentProcess(),65535); */ - ExitProcess(65535); -#else - TerminateApp((HTASK) NULL,(WORD) NO_UAE_BOX); -#endif -#else - EndOfPgr(1); -#endif - } -#endif - } - } - if ((file!=NULL) && (file!=stderr) && (file!=stdout)) - { - time_t t; - - time(&t); - fprintf(file,"Generated on: %s%s\n", - ctime(&t), - (file==stdout) || (file==stderr) ? "\r" : "" - ); - } - } - if (file!=NULL) - { -#if defined _WINDOWS - if ((file==stdout) || (file==stderr)) { -#if defined LINE_BY_LINE - if (MessageBox((HWND) NULL,(LPCSTR) String,(LPCSTR) "Fortify",(UINT) MB_OKCANCEL /* |MB_ICONINFORMATION */)==IDCANCEL) -#if 0 -#if defined WIN32 - /* TerminateProcess(GetCurrentProcess(),65535); */ - ExitProcess(65535); -#else - TerminateApp((HTASK) NULL,(WORD) NO_UAE_BOX); -#endif -#else - EndOfPgr(1); -#endif -#else - { - char *ptr; - - ptr="fortify.tmp"; - if ((ptr==NULL) || ((file=gfile=fopen(ptr,"w+"))==NULL)) - { - char str[255]; - - sprintf(str,"Unable to create temporary file\n \"%s\"",(ptr==NULL) ? "(NULL)" : ptr); - MessageBox((HWND) NULL,(LPCSTR) str,(LPCSTR) "Fortify",(UINT) MB_ICONSTOP); -#if 0 -#if defined WIN32 - /* TerminateProcess(GetCurrentProcess(),65535); */ - ExitProcess(65535); -#else - TerminateApp((HTASK) NULL,(WORD) NO_UAE_BOX); -#endif -#else - EndOfPgr(1); -#endif - } - } -#endif - } - if ((file!=stdout) && (file!=stderr)) -#endif - { - int i,ch=-1; - - for (i=0;(String[i]) && (Nlines<30);i++) - if (String[i]=='\n') Nlines++; - if ((String[i]) && (String[i+1])) { - ch=String[i+1]; - String[i+1]=0; - } - if ((file==stdout) || (file==stderr)) { - char *ptr=String; - int i; - - do { - for (i=0;(ptr[i]) && (ptr[i]!='\r') && (ptr[i]!='\n');i++); - Nchars+=fprintf(file,"%-*.*s%s", - i,i, - ptr, - (ptr[i]) ? "\r\n" : "" - ); - ptr+=i; - if (ptr[0]=='\r') ptr++; - if (ptr[0]=='\n') ptr++; - } while (*ptr); - } - else Nchars+=fprintf(file,String); - if (ch>=0) String[i+1]=(char)ch; - if (Nlines>=30) { - WaitIfstdOutput(); - Nchars=Nlines=0; - stdOutput = 0; - if ((String[i]) && (String[i+1])) { - if ((file==stderr) || (file==stdout) || ((gfile!=NULL) && (Nchars))) - stdOutput = 1; - st_DefaultOutput(String+i); - } - } - } - if ((file==stderr) || (file==stdout) || ((gfile!=NULL) && (Nchars))) - stdOutput = 1; - } -} - -static void WaitIfstdOutput() -{ -#if !defined _WINDOWS - if((stdOutput) && (st_Output != (OutputFuncPtr) _Fortify_NoOutput)) - { -#ifdef FORTIFY_WAIT_FOR_KEY - static signed char wait_on_key=-1; - - if(wait_on_key<0) - { - char *ptr; - - if (((ptr=getenv("FORTIFY_WAIT_FOR_KEY"))!=NULL) && - (tolower(*ptr)=='n')) wait_on_key = 0; - else wait_on_key = 1; - - } - if(wait_on_key) - { - char c; - -#if !defined MSDOS && !defined WIN32 - struct termio tio,tiobak; - char flag; - - if ((flag=ioctl(0,TCGETA,&tio))==0) /* handle 0 is stdin */ - { - tiobak=tio; - tio.c_lflag&=~ICANON; - tio.c_lflag&=~ECHO; - tio.c_cc[VMIN]=1; - ioctl(0,TCSETA,&tio); - } -#endif /* !MSDOS */ - c = (char)getch(); - -#if !defined MSDOS && !defined WIN32 - if (flag==0) - ioctl(0,TCSETA,&tiobak); -#endif /* !MSDOS */ - - if ((c == 3) || (c == 0x1b)) EndOfPgr(3); - } -#endif /* FORTIFY_WAIT_FOR_KEY */ - - } -#else -# if !defined LINE_BY_LINE - if ((stdOutput) && (gfile!=NULL) && (Nchars)) - { - char *ptr; - - ptr=(char *) malloc(Nchars+1); - if (ptr!=NULL) - { - int n=0,l=0; - - rewind(gfile); - while ((n 0) - { - if(rand() % 100 < st_MallocFailRate) - { -#ifdef WARN_ON_FALSE_FAIL - sprintf(st_Buffer, - "\nFortify: %s.%ld\n malloc(%ld) \"false\" failed\n", - file, line, (unsigned long)size); - st_Output(st_Buffer); -#endif - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - } - - /* - * malloc the memory, including the space for the header and fortification - * buffers - */ -#ifdef WARN_ON_SIZE_T_OVERFLOW - { - size_t private_size = sizeof(struct Header) - + FORTIFY_BEFORE_SIZE + size + FORTIFY_AFTER_SIZE; - - if(private_size < size) /* Check to see if the added baggage is larger than size_t */ - { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n malloc(%ld) has overflowed size_t.\n", - file, line, (unsigned long)size); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - } -#endif - - ptr = (unsigned char *) malloc(sizeof(struct Header) + - FORTIFY_BEFORE_SIZE + size + FORTIFY_AFTER_SIZE); - if(!ptr) - { -#ifdef WARN_ON_MALLOC_FAIL - sprintf(st_Buffer, "\nFortify: %s.%ld\n malloc(%ld) failed\n", - file, line, (unsigned long)size); - st_Output(st_Buffer); -#endif - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - /* - * Initialize and validate the header - */ - h = (struct Header *)ptr; - - h->Size = size; - - h->File = file; - h->Line = (unsigned short) line; - - h->Next = st_Head; - h->Prev = 0; - - h->Scope = st_Scope; - - if(st_Head) - { - st_Head->Prev = h; - MakeHeaderValid(st_Head); - } - - st_Head = h; - - MakeHeaderValid(h); - - - /* - * Initialize the fortifications - */ - SetFortification(ptr + sizeof(struct Header), - FORTIFY_BEFORE_VALUE, FORTIFY_BEFORE_SIZE); - SetFortification(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE + size, - FORTIFY_AFTER_VALUE, FORTIFY_AFTER_SIZE); - -#ifdef FILL_ON_MALLOC - /* - * Fill the actual user memory - */ - SetFortification(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE, - FILL_ON_MALLOC_VALUE, size); -#endif - - /* - * We return the address of the user's memory, not the start of the block, - * which points to our magic cookies - */ - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return((void *) (ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE)); -} - -/* - * _Fortify_free() - This free must be used for all memory allocated with - * _Fortify_malloc(). - * - * Features: - * + Pointers are validated before attempting a free - the pointer - * must point to a valid malloc'd bit of memory. - * + Detects attempts at freeing the same block of memory twice - * + Can clear out memory as it is free'd, to prevent code from using - * the memory after it's been freed. - * + Checks the sentinals of the memory being freed. - * + Can check the sentinals of all memory. - */ - -void FORTIFY_STORAGE -_Fortify_free(void *uptr,char *file,unsigned long line) -{ - unsigned char *ptr; - struct Header *h; - - if(uptr == NULL) - return; - - ptr = (unsigned char *)uptr - sizeof(struct Header) - FORTIFY_BEFORE_SIZE; - h = (struct Header *)ptr; - - stdOutput = 0; - - FORTIFY_LOCK(); - - if(st_Disabled) - { - free(uptr); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return; - } - -#ifdef CHECK_ALL_MEMORY_ON_FREE -#ifdef FORTIFY_CheckInterval - if (TimeToCheck()) -#endif - _Fortify_CheckAllMemory(file, line); -#endif - -#ifdef PARANOID_FREE - if(!IsOnList(h)) - { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n Invalid pointer, corrupted header, or possible free twice\n", - file, line); - st_Output(st_Buffer); - OutputLastVerifiedPoint(); - goto fail; - } -#endif - - if(!CheckBlock(h, file, line)) - goto fail; - - /* - * Remove the block from the list - */ - if(h->Prev) - { - if(!CheckBlock(h->Prev, file, line)) - goto fail; - - h->Prev->Next = h->Next; - MakeHeaderValid(h->Prev); - } - else - st_Head = h->Next; - - if(h->Next) - { - if(!CheckBlock(h->Next, file, line)) - goto fail; - - h->Next->Prev = h->Prev; - MakeHeaderValid(h->Next); - } - -#ifdef FILL_ON_FREE - /* - * Nuke out all memory that is about to be freed - */ - SetFortification(ptr, FILL_ON_FREE_VALUE, - sizeof(struct Header) + FORTIFY_BEFORE_SIZE + h->Size + FORTIFY_AFTER_SIZE); -#endif - - /* - * And do the actual free - */ - free(ptr); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return; - -fail: - sprintf(st_Buffer, " free(%s) failed\n", address(uptr)); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); -} - -/* - * _Fortify_realloc() - Uses _Fortify_malloc() and _Fortify_free() to implement - * realloc(). - * - * Features: - * + The realloc'd block is ALWAYS moved. - * + The pointer passed to realloc() is verified in the same way that - * _Fortify_free() verifies pointers before it frees them. - * + All the _Fortify_malloc() and _Fortify_free() protection - */ -void *FORTIFY_STORAGE -_Fortify_realloc(void *ptr,size_t new_size,char *file,unsigned long line) -{ - void *new_ptr; - struct Header *h; - - if(new_size == 0) - { - _Fortify_free(ptr,file,line); - return(NULL); - } - - h = (struct Header *)((unsigned char *)ptr - sizeof(struct Header) - FORTIFY_BEFORE_SIZE); - - stdOutput = 0; - - if(st_Disabled) - { - FORTIFY_LOCK(); - new_ptr = (void *) realloc(ptr, new_size); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(new_ptr); - } - - if(!ptr) - { - void *FORTIFY_STORAGE ret = _Fortify_malloc(new_size, file, line); - - WaitIfstdOutput(); - return(ret); - } - - FORTIFY_LOCK(); - - if(!IsOnList(h)) - { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n Invalid pointer or corrupted header passed to realloc\n", - file, line); - st_Output(st_Buffer); - goto fail; - } - - if(!CheckBlock(h, file, line)) - goto fail; - - new_ptr = _Fortify_malloc(new_size, file, line); - if(!new_ptr) - { - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - if(h->Size < new_size) - memcpy(new_ptr, ptr, h->Size); - else - memcpy(new_ptr, ptr, new_size); - - _Fortify_free(ptr, file, line); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(new_ptr); - -fail: - sprintf(st_Buffer, " realloc(%s, %ld) failed\n", address(ptr), (unsigned long)new_size); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); -} - - -/* - * __Fortify_CheckPointer() - Returns true if the uptr points to a valid - * piece of _Fortify_malloc()'d memory. The memory must be on the malloc'd - * list, and it's sentinals must be in tact. - * If anything is wrong, an error message is issued. - * - * (Note - if fortify is disabled, this function always returns true). - */ -static int FORTIFY_STORAGE -__Fortify_CheckPointer(void *uptr,char OnlyStart,unsigned long size,char *file,unsigned long line) -{ - unsigned char *ptr = (unsigned char *)uptr - sizeof(struct Header) - FORTIFY_BEFORE_SIZE; - int r = 1, StartPointer; - - stdOutput = 0; - - if(st_Disabled) - { - WaitIfstdOutput(); - return(1); - } - - FORTIFY_LOCK(); - - StartPointer = IsOnList((struct Header *)ptr); - if((OnlyStart) && (!StartPointer)) - { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n Invalid pointer or corrupted header detected (%s)\n", - file, line, address(uptr)); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - if((OnlyStart) || (StartPointer)) - r = CheckBlock((struct Header *)ptr, file, line); - if(!OnlyStart) - r = CheckPointer((unsigned char *)uptr, size, file, line); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(r); -} - - -int FORTIFY_STORAGE -_Fortify_CheckPointer(void *uptr,char *file,unsigned long line) -{ - return(__Fortify_CheckPointer(uptr,1,0,file,line)); -} - -/* - * Fortify_SetOutputFunc(OutputFuncPtr Output) - Sets the function used to - * output all error and diagnostic messages by fortify. The output function - * takes a single unsigned char * argument, and must be able to handle newlines. - * The function returns the old pointer. - */ -Fortify_OutputFuncPtr FORTIFY_STORAGE -_Fortify_SetOutputFunc(Fortify_OutputFuncPtr Output) -{ - OutputFuncPtr Old = st_Output; - - st_Output = (OutputFuncPtr) Output; - - return((Fortify_OutputFuncPtr FORTIFY_STORAGE) Old); -} - -/* - * _Fortify_SetMallocFailRate(int Percent) - _Fortify_malloc() will make the - * malloc attempt fail this Percent of the time, even if the memory is - * available. Useful to "stress-test" an application. Returns the old - * value. The fail rate defaults to 0. - */ -int FORTIFY_STORAGE -_Fortify_SetMallocFailRate(int Percent) -{ - int Old = st_MallocFailRate; - - st_MallocFailRate = Percent; - - return(Old); -} - - -/* - * _Fortify_CheckAllMemory() - Checks the sentinals of all malloc'd memory. - * Returns the number of blocks that failed. - * - * (If Fortify is disabled, this function always returns 0). - */ -int FORTIFY_STORAGE -_Fortify_CheckAllMemory(char *file,unsigned long line) -{ - struct Header *curr = st_Head; - int count = 0; - - stdOutput = 0; - - if(st_Disabled) - { - WaitIfstdOutput(); - return(0); - } - - FORTIFY_LOCK(); - - while(curr) - { - if(!CheckBlock(curr, file, line)) - count++; - - curr = curr->Next; - } - - if(file) - { - st_LastVerifiedFile = file; - st_LastVerifiedLine = (short) line; - } - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(count); -} - -/* _Fortify_EnterScope - enters a new Fortify scope level. - * returns the new scope level. - */ -int FORTIFY_STORAGE -_Fortify_EnterScope(char *file,unsigned long line) -{ - return((int) ++st_Scope); -} - -/* _Fortify_LeaveScope - leaves a Fortify scope level, - * also prints a memory dump of all non-freed memory that was allocated - * during the scope being exited. - */ -int FORTIFY_STORAGE -_Fortify_LeaveScope(char *file,unsigned long line) -{ - struct Header *curr = st_Head; - int count = 0; - size_t size = 0; - - stdOutput = 0; - - if(st_Disabled) - { - WaitIfstdOutput(); - return(0); - } - - FORTIFY_LOCK(); - - st_Scope--; - while(curr) - { - if(curr->Scope > st_Scope) - { - if(count == 0) - { - sprintf(st_Buffer, "\nFortify: Memory Dump at %s.%ld\n", file, line); - st_Output(st_Buffer); - OutputLastVerifiedPoint(); - sprintf(st_Buffer, "%11s %8s %s\n", "Address", "Size", "Allocator"); - st_Output(st_Buffer); - } - - OutputHeader(curr); - count++; - size += curr->Size; - } - - curr = curr->Next; - } - - if(count) - { - sprintf(st_Buffer, "%11s %8ld bytes overhead\n", "and", - (unsigned long)(count * (sizeof(struct Header) + FORTIFY_BEFORE_SIZE + FORTIFY_AFTER_SIZE))); - st_Output(st_Buffer); - - sprintf(st_Buffer,"%11s %8ld bytes in %d blocks\n", "total", size, count); - st_Output(st_Buffer); - } - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(count); -} - -/* - * _Fortify_OutputAllMemory() - Outputs the entire list of currently - * malloc'd memory. For each malloc'd block is output it's Address, - * Size, and the SourceFile and Line that allocated it. - * - * If there is no memory on the list, this function outputs nothing. - * - * It returns the number of blocks on the list, unless fortify has been - * disabled, in which case it always returns 0. - */ -int FORTIFY_STORAGE -_Fortify_OutputAllMemory(char *file,unsigned long line) -{ - struct Header *curr = st_Head; - int count = 0; - size_t size = 0; - - stdOutput = 0; - - if(st_Disabled) - { - WaitIfstdOutput(); - return(0); - } - - FORTIFY_LOCK(); - - if(curr) - { - sprintf(st_Buffer, "\nFortify: Memory Dump at %s.%ld\n", file, line); - st_Output(st_Buffer); - OutputLastVerifiedPoint(); - sprintf(st_Buffer, "%11s %8s %s\n", "Address", "Size", "Allocator"); - st_Output(st_Buffer); - - while(curr) - { - OutputHeader(curr); - count++; - size += curr->Size; - curr = curr->Next; - } - - sprintf(st_Buffer, "%11s %8ld bytes overhead\n", "and", - (unsigned long)(count * (sizeof(struct Header) + FORTIFY_BEFORE_SIZE + FORTIFY_AFTER_SIZE))); - st_Output(st_Buffer); - - sprintf(st_Buffer,"%11s %8ld bytes in %d blocks\n", "total", size, count); - st_Output(st_Buffer); - } - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(count); -} - -/* _Fortify_DumpAllMemory(Scope) - Outputs the entire list of currently - * new'd memory within the specified scope. For each new'd block is output - * it's Address, Size, the SourceFile and Line that allocated it, a hex dump - * of the contents of the memory and an ascii dump of printable characters. - * - * If there is no memory on the list, this function outputs nothing. - * - * It returns the number of blocks on the list, unless Fortify has been - * disabled, in which case it always returns 0. - */ -int FORTIFY_STORAGE -_Fortify_DumpAllMemory(int scope,char *file,unsigned long line) -{ - struct Header *curr = st_Head; - int count = 0; - size_t size = 0; - - stdOutput = 0; - - if(st_Disabled) - { - WaitIfstdOutput(); - return(0); - } - - FORTIFY_LOCK(); - - while(curr) - { - if(curr->Scope >= scope) - { - if(count == 0) - { - sprintf(st_Buffer, "\nFortify: Memory Dump at %s.%ld\n", file, line); - st_Output(st_Buffer); - OutputLastVerifiedPoint(); - sprintf(st_Buffer, "%11s %8s %s\n", "Address", "Size", "Allocator"); - st_Output(st_Buffer); - } - - OutputHeader(curr); - OutputMemory(curr); - st_Output("\n"); - count++; - size += curr->Size; - } - - curr = curr->Next; - } - - if(count) - { - sprintf(st_Buffer, "%11s %8ld bytes overhead\n", "and", - (unsigned long)(count * (sizeof(struct Header) + FORTIFY_BEFORE_SIZE + FORTIFY_AFTER_SIZE))); - st_Output(st_Buffer); - - sprintf(st_Buffer,"%11s %8ld bytes in %d blocks\n", "total", size, count); - st_Output(st_Buffer); - } - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(count); -} - -/* - * _Fortify_Disable() - This function provides a mechanism to disable Fortify - * without recompiling all the sourcecode. - * If 'how' is zero then it can only be called when there is no memory on the - * Fortify malloc'd list. (Ideally, at the start of the program before any - * memory has been allocated). If you call this function when there IS - * memory on the Fortify malloc'd list, it will issue an error, and fortify - * will not be disabled. - * If 'how' is nonzero then output will only be disabled. This can always be - * done. - */ - -int FORTIFY_STORAGE -_Fortify_Disable(char *file,unsigned long line,int how) -{ - int result; - - if (how <= 0) - { - stdOutput = 0; - - FORTIFY_LOCK(); - - if((st_Head) && (how == 0)) - { - sprintf(st_Buffer, "Fortify: %s.%d\n", file, line); - st_Output(st_Buffer); - st_Output(" Fortify_Disable failed\n"); - st_Output(" (because there is memory on the Fortify memory list)\n"); - - _Fortify_OutputAllMemory(file, line); - result = 0; - } - else - { - st_Disabled = (how >= -1 ? 1 : 0); - result = 1; - } - - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - } - else - { - _Fortify_SetOutputFunc((Fortify_OutputFuncPtr) _Fortify_NoOutput); - result = 1; - } - return(result); -} - -/* - * Check a block's header and fortifications. - */ -static int CheckBlock(struct Header *h,char *file,unsigned long line) -{ - unsigned char *ptr = (unsigned char *)h; - int result = 1; - - stdOutput = 0; - - if(!IsHeaderValid(h)) - { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n Invalid pointer or corrupted header detected (%s)\n", - file, line, address(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE)); - st_Output(st_Buffer); - OutputLastVerifiedPoint(); - WaitIfstdOutput(); - return(0); - } - - if(!CheckFortification(ptr + sizeof(struct Header), - FORTIFY_BEFORE_VALUE, FORTIFY_BEFORE_SIZE)) - { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n Memory overrun detected before block\n", - file, line); - st_Output(st_Buffer); - - sprintf(st_Buffer," (%s,%ld,%s.%u)\n", - address(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE), - (unsigned long)h->Size, h->File, h->Line); - st_Output(st_Buffer); - - OutputFortification(ptr + sizeof(struct Header), - FORTIFY_BEFORE_VALUE, FORTIFY_BEFORE_SIZE); - OutputLastVerifiedPoint(); - result = 0; - } - - if(!CheckFortification(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE + h->Size, - FORTIFY_AFTER_VALUE, FORTIFY_AFTER_SIZE)) - { - sprintf(st_Buffer, "\nFortify: %s.%ld\n Memory overrun detected after block\n", - file, line); - st_Output(st_Buffer); - - sprintf(st_Buffer," (%s,%ld,%s.%u)\n", - address(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE), - (unsigned long)h->Size, h->File, h->Line); - st_Output(st_Buffer); - - OutputFortification(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE + h->Size, - FORTIFY_AFTER_VALUE, FORTIFY_AFTER_SIZE); - OutputLastVerifiedPoint(); - result = 0; - } - - WaitIfstdOutput(); - return(result); -} - -static int CheckPointer(unsigned char *ptr,unsigned long size,char *file,unsigned long line) -{ - struct Header *curr; - unsigned char *ptr1; - - curr = st_Head; - while(curr) - { - ptr1 = (unsigned char *)curr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE; - if(ptr + size <= (unsigned char *)curr) - ; - else if(ptr >= ptr1) - ; - else - { - sprintf(st_Buffer, "\nFortify: %s.%ld\n Memory access detected before block\n", - file, line); - st_Output(st_Buffer); - - sprintf(st_Buffer," (%s,%ld,%s.%u)\n", - address(ptr1), - (unsigned long)curr->Size, curr->File, curr->Line); - st_Output(st_Buffer); - - WaitIfstdOutput(); - return(0); - } - - if(ptr + size <= ptr1 + curr->Size) - ; - else if(ptr >= ptr1 + curr->Size + FORTIFY_AFTER_SIZE) - ; - else - { - sprintf(st_Buffer, "\nFortify: %s.%ld\n Memory access detected after block\n", - file, line); - st_Output(st_Buffer); - - sprintf(st_Buffer," (%s,%ld,%s.%u)\n", - address(ptr1), - (unsigned long)curr->Size, curr->File, curr->Line); - st_Output(st_Buffer); - - WaitIfstdOutput(); - return(0); - } - - if((ptr >= ptr1) && (ptr < ptr1 + curr->Size) && (ptr + size > ptr1 + curr->Size)) - { - sprintf(st_Buffer, "\nFortify: %s.%ld\n Memory access detected after block\n", - file, line); - st_Output(st_Buffer); - - sprintf(st_Buffer," (%s,%ld,%s.%u)\n", - address(ptr + sizeof(struct Header) + FORTIFY_BEFORE_SIZE), - (unsigned long)curr->Size, curr->File, curr->Line); - st_Output(st_Buffer); - WaitIfstdOutput(); - return(0); - } - - curr = curr->Next; - } - return(1); -} - -/* - * Checks if the _size_ bytes from _ptr_ are all set to _value_ - */ -static int CheckFortification(unsigned char *ptr,unsigned char value,size_t size) -{ - while(size--) - if(*ptr++ != value) - return(0); - - return(1); -} - -/* - * Set the _size_ bytes from _ptr_ to _value_. - */ -static void SetFortification(unsigned char *ptr,unsigned char value,size_t size) -{ - memset(ptr, value, size); -} - -/* - * Output the corrupted section of the fortification - */ -/* Output the corrupted section of the fortification */ -static void -OutputFortification(unsigned char *ptr,unsigned char value,size_t size) -{ - unsigned long offset, column; - char ascii[17]; - - st_Output("Address Offset Data"); - - offset = 0; - column = 0; - - while(offset < size) - { - if(column == 0) - { - sprintf(st_Buffer, "\n%8s %8d ", address(ptr), offset); - st_Output(st_Buffer); - } - - sprintf(st_Buffer, "%02x ", *ptr); - st_Output(st_Buffer); - - ascii[ (int) column ] = isprint( *ptr ) ? (char)(*ptr) : (char)(' '); - ascii[ (int) (column + 1) ] = '\0'; - - ptr++; - offset++; - column++; - - if(column == 16) - { - st_Output( " \"" ); - st_Output( ascii ); - st_Output( "\"" ); - column = 0; - } - } - - if ( column != 0 ) - { - while ( column ++ < 16 ) - { - st_Output( " " ); - } - st_Output( " \"" ); - st_Output( ascii ); - st_Output( "\"" ); - } - - st_Output("\n"); -} - -/* - * Returns true if the supplied pointer does indeed point to a real Header - */ -static int IsHeaderValid(struct Header *h) -{ - return(!ChecksumHeader(h)); -} - -/* - * Updates the checksum to make the header valid - */ -static void MakeHeaderValid(struct Header *h) -{ - h->Checksum = 0; - h->Checksum = -ChecksumHeader(h); -} - -/* - * Calculate (and return) the checksum of the header. (Including the Checksum - * variable itself. If all is well, the checksum returned by this function should - * be 0. - */ -static int ChecksumHeader(struct Header *h) -{ - register int c, checksum, *p; - - for(c = 0, checksum = 0, p = (int *)h; c < sizeof(struct Header)/sizeof(int); c++) - checksum += *p++; - - return(checksum); -} - -/* - * Examines the malloc'd list to see if the given header is on it. - */ -static int IsOnList(struct Header *h) -{ - struct Header *curr; - - curr = st_Head; - while(curr) - { - if(curr == h) - return(1); - curr = curr->Next; - } - - return(0); -} - - -/* - * Hex and ascii dump the memory - */ -static void -OutputMemory(struct Header *h) -{ - OutputFortification((unsigned char*)h + sizeof(struct Header) + FORTIFY_BEFORE_SIZE, - 0, h->Size); -} - - -/* - * Output the header... - */ -static void OutputHeader(struct Header *h) -{ - sprintf(st_Buffer, "%11s %8ld %s.%u (%d)\n", - address((unsigned char*)h + sizeof(struct Header) + FORTIFY_BEFORE_SIZE), - (unsigned long)h->Size, - h->File, h->Line, (int) h->Scope); - st_Output(st_Buffer); -} - -static void OutputLastVerifiedPoint() -{ - sprintf(st_Buffer, "\nLast Verified point: %s.%u\n", - st_LastVerifiedFile, - st_LastVerifiedLine); - st_Output(st_Buffer); -} - -#else /* FORTIFY_TRANSPARENT */ - -void *FORTIFY_STORAGE -_Fortify_malloc(size,file,line) - size_t size; - char *file; - unsigned long line; -{ - return(malloc(size)); -} - -void FORTIFY_STORAGE -_Fortify_free(uptr,file,line) - void *uptr; - char *file; - unsigned long line; -{ - free(uptr); -} - -void *FORTIFY_STORAGE -_Fortify_realloc(ptr,new_size,file,line) - void *ptr; - size_t new_size; - char *file; - unsigned long line; -{ - return(realloc(ptr, new_size)); -} - -int FORTIFY_STORAGE -_Fortify_CheckPointer(uptr,file,line) - void *uptr; - char *file; - unsigned long line; -{ - return(1); -} - -Fortify_OutputFuncPtr FORTIFY_STORAGE -_Fortify_SetOutputFunc(Output) - Fortify_OutputFuncPtr Output; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_SetMallocFailRate(Percent) - int Percent; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_CheckAllMemory(file,line) - char *file; - unsigned long line; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_EnterScope(file,line) - char *file; - unsigned long line; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_LeaveScope(file,line) - char *file; - unsigned long line; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_OutputAllMemory(file,line) - char *file; - unsigned long line; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_DumpAllMemory(scope,file,line) - int scope; - char *file; - unsigned long line; -{ - return(0); -} - -int FORTIFY_STORAGE -_Fortify_Disable(file,line) - char *file; - unsigned long line; -{ - return(1); -} - -#endif /* !FORTIFY_TRANSPARENT */ - -/* function that use _Fortify_malloc(), _Fortify_realloc(), _Fortify_free() */ - -/* - * Fortifty_calloc() - Uses _Fortify_malloc() to implement calloc(). Much - * the same protection as _Fortify_malloc(). - */ -void *FORTIFY_STORAGE -_Fortify_calloc(size_t nitems,size_t size,char *file,unsigned long line) -{ - void *ptr; - - ptr = _Fortify_malloc(nitems * size, file, line); - - if(ptr) - memset(ptr, 0, nitems * size); - - return(ptr); -} - -/* - * Fortifty_strdup() - Uses _Fortify_malloc() to implement strdup(). Much - * the same protection as _Fortify_malloc(). - * The library function is not used because it is not certain that getpwd - * uses the library malloc function (if linked with an alternate library) - * and if the memory is freed then strange things can happen - */ -char *FORTIFY_STORAGE -_Fortify_strdup(char *str,char *file,unsigned long line) -{ - char *ptr; - unsigned long l; - - if(str == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strdup pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - l = (int) strlen(str) + 1; - __Fortify_CheckPointer(str,0,l,file,line); - - ptr = (char *) _Fortify_malloc(l, file, line); - - if(ptr) - strcpy(ptr, str); - - return(ptr); -} - -/* - * Fortifty_getpwd() - Uses _Fortify_malloc() to implement getpwd(). Much - * the same protection as _Fortify_malloc(). - * Memory is not allocated bu getcwd but by our routine for the same reason - * as for strdup - */ -char *FORTIFY_STORAGE -_Fortify_getcwd(char *buf,int size,char *file,unsigned long line) -{ - char *ptr; - - if(buf!=NULL) - ptr = buf; - else - ptr = (char *) _Fortify_malloc(size + 1, file, line); - - if(ptr) - ptr = getcwd(ptr, size); - - return(ptr); -} - -/* - * Fortifty_tempnam() - Uses _Fortify_strdup() to implement tempnam(). Much - * the same protection as _Fortify_malloc(). - */ -char *FORTIFY_STORAGE -_Fortify_tempnam(char *dir,char *pfx,char *file,unsigned long line) -{ - char *ptr1, *ptr2; - - ptr1 = tempnam(dir,pfx); - - if(ptr1) - { - ptr2=_Fortify_strdup(ptr1,file,line); - free(ptr1); - ptr1=ptr2; - } - - return(ptr1); -} - -/* - * Fortify_memcpy() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -void *FORTIFY_STORAGE -_Fortify_memcpy(void *to,void *from,size_t size,char *file,unsigned long line) -{ - if (size) { - if((from == NULL) || (to == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(from == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "memcpy from pointer is NULL", file, line); - if(to == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (from == NULL) ? "" : " and ", "memcpy to pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(to,0,size,file,line); - __Fortify_CheckPointer(from,0,size,file,line); - } - return(memcpy(to,from,size)); -} - -/* - * Fortify_memmove() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -void *FORTIFY_STORAGE -_Fortify_memmove(void *to,void *from,size_t size,char *file,unsigned long line) -{ - if((from == NULL) || (to == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(from == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "memmove from pointer is NULL", file, line); - if(to == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (from == NULL) ? "" : " and ", "memmove to pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(to,0,size,file,line); - __Fortify_CheckPointer(from,0,size,file,line); - return(memmove(to,from,size)); -} - -/* - * Fortify_memccpy() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -void *FORTIFY_STORAGE -_Fortify_memccpy(void *to,void *from,int c,size_t size,char *file,unsigned long line) -{ - if((from == NULL) || (to == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(from == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "memccpy from pointer is NULL", file, line); - if(to == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (from == NULL) ? "" : " and ", "memccpy to pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(to,0,size,file,line); - __Fortify_CheckPointer(from,0,size,file,line); - return(memccpy(to,from,c,size)); -} - -/* - * Fortify_memset() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -void *FORTIFY_STORAGE -_Fortify_memset(void *buffer,int c,size_t size,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n memset pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,size,file,line); - return(memset(buffer,c,size)); -} - -/* - * Fortify_memchr() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -void *FORTIFY_STORAGE -_Fortify_memchr(void *buffer,int c,size_t size,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n memchr pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,size,file,line); - return(memchr(buffer,c,size)); -} - -/* - * Fortify_memcmp() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_memcmp(void *buffer1,void *buffer2,size_t size,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "memcmp first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "memcmp second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,size,file,line); - __Fortify_CheckPointer(buffer2,0,size,file,line); - return(memcmp(buffer1,buffer2,size)); -} - -/* - * Fortify_memicmp() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_memicmp(void *buffer1,void *buffer2,size_t size,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "memicmp first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "memicmp second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,size,file,line); - __Fortify_CheckPointer(buffer2,0,size,file,line); - return(memicmp(buffer1,buffer2,size)); -} - -/* - * Fortify_strcoll() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_strcoll(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strcoll first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strcoll second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - __Fortify_CheckPointer(buffer2,0,strlen(buffer2)+1,file,line); - return(strcoll(buffer1,buffer2)); -} - -/* - * Fortify_strcspn() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -size_t FORTIFY_STORAGE -_Fortify_strcspn(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strcspn first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strcspn second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - __Fortify_CheckPointer(buffer2,0,strlen(buffer2)+1,file,line); - return(strcspn(buffer1,buffer2)); -} - -/* - * Fortify_strcmp() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_strcmp(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strcmp first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strcmp second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - __Fortify_CheckPointer(buffer2,0,strlen(buffer2)+1,file,line); - return(strcmp(buffer1,buffer2)); -} - -/* - * Fortify_strcmpi() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_strcmpi(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strcmpi first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strcmpi second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - __Fortify_CheckPointer(buffer2,0,strlen(buffer2)+1,file,line); - return(strcmpi(buffer1,buffer2)); -} - -/* - * Fortify_strncmp() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_strncmp(char *buffer1,char *buffer2,size_t size,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strncmp first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strncmp second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,MIN(strlen(buffer1)+1,size),file,line); - __Fortify_CheckPointer(buffer2,0,MIN(strlen(buffer2)+1,size),file,line); - return(strncmp(buffer1,buffer2,size)); -} - -/* - * Fortify_strnicmp() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_strnicmp(char *buffer1,char *buffer2,size_t size,char *file,unsigned long line) -{ - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strnicmp first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strnicmp second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,MIN(strlen(buffer1)+1,size),file,line); - __Fortify_CheckPointer(buffer2,0,MIN(strlen(buffer2)+1,size),file,line); - return(strnicmp(buffer1,buffer2,size)); -} - -/* - * Fortify_strchr() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strchr(char *buffer,int c,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strchr pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strchr(buffer,c)); -} - -/* - * Fortify_strrchr() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strrchr(char *buffer,int c,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strchr pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strrchr(buffer,c)); -} - -/* - * Fortify_strlwr() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strlwr(char *buffer,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strlwr pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strlwr(buffer)); -} - -/* - * Fortify_strlwr() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strupr(char *buffer,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strupr pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strupr(buffer)); -} - -/* - * Fortify_strrev() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strrev(char *buffer,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strrev pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strrev(buffer)); -} - -/* - * Fortify_strlen() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -size_t FORTIFY_STORAGE -_Fortify_strlen(char *buffer,char *file,unsigned long line) -{ - unsigned long l; - - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strlen pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - l = strlen(buffer); - __Fortify_CheckPointer(buffer,0,l+1,file,line); - return(l); -} - -/* - * Fortify_strcat() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strcat(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - unsigned long l; - - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strcat first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strcat second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - l = strlen(buffer2)+1; - __Fortify_CheckPointer(buffer1,0,l,file,line); - __Fortify_CheckPointer(buffer2,0,l,file,line); - return(strcat(buffer1,buffer2)); -} - -/* - * Fortify_strpbrk() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strpbrk(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - unsigned long l; - - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strpbrk first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strpbrk second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - l = strlen(buffer2)+1; - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - __Fortify_CheckPointer(buffer2,0,l,file,line); - return(strpbrk(buffer1,buffer2)); -} - -/* - * Fortify_strstr() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strstr(char *buffer1,char *buffer2,char *file,unsigned long line) -{ - unsigned long l; - - if((buffer1 == NULL) || (buffer2 == NULL)) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - if(buffer1 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strstr first pointer is NULL", file, line); - if(buffer2 == NULL) - sprintf(st_Buffer + strlen(st_Buffer), "%s%s", (buffer2 == NULL) ? "" : " and ", "strstr second pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - l = strlen(buffer2)+1; - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - __Fortify_CheckPointer(buffer2,0,l,file,line); - return(strstr(buffer1,buffer2)); -} - -/* - * Fortify_strtol() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -long FORTIFY_STORAGE -_Fortify_strtol(char *buffer1,char **buffer2,int n,char *file,unsigned long line) -{ - if(buffer1 == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strtol first pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - return(strtol(buffer1,buffer2,n)); -} - -/* - * Fortify_atoi() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -int FORTIFY_STORAGE -_Fortify_atoi(char *buffer1,char *file,unsigned long line) -{ - if(buffer1 == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "atoi first pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - return(atoi(buffer1)); -} - -/* - * Fortify_atol() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -long FORTIFY_STORAGE -_Fortify_atol(char *buffer1,char *file,unsigned long line) -{ - if(buffer1 == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "atol first pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - return(atol(buffer1)); -} - -/* - * Fortify_atod() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -double FORTIFY_STORAGE -_Fortify_atof(char *buffer1,char *file,unsigned long line) -{ - if(buffer1 == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "atod first pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - return(atof(buffer1)); -} - -/* - * Fortify_strtoul() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -unsigned long FORTIFY_STORAGE -_Fortify_strtoul(char *buffer1,char **buffer2,int n,char *file,unsigned long line) -{ - if(buffer1 == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strtoul first pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - return(strtoul(buffer1,buffer2,n)); -} - -/* - * Fortify_strtod() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -double FORTIFY_STORAGE -_Fortify_strtod(char *buffer1,char **buffer2,char *file,unsigned long line) -{ - if(buffer1 == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n ", file, line); - sprintf(st_Buffer + strlen(st_Buffer), "%s", "strtod first pointer is NULL", file, line); - strcat(st_Buffer, "\n"); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(0); - } - - __Fortify_CheckPointer(buffer1,0,strlen(buffer1)+1,file,line); - return(strtod(buffer1,buffer2)); -} - -/* - * Fortify_strset() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strset(char *buffer,int c,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strset pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strset(buffer,c)); -} - -/* - * Fortify_strnset() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strnset(char *buffer,int c,size_t size,char *file,unsigned long line) -{ - if(buffer == NULL) { - sprintf(st_Buffer, - "\nFortify: %s.%ld\n strnset pointer is NULL\n", file, line); - st_Output(st_Buffer); - FORTIFY_UNLOCK(); - WaitIfstdOutput(); - return(NULL); - } - - __Fortify_CheckPointer(buffer,0,strlen(buffer)+1,file,line); - return(strnset(buffer,c,size)); -} - -/* - * Fortify_strncpy() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -static char *FORTIFY_STORAGE -__Fortify_strncpy(char *to,char *from,size_t size,int usesize,char *file,unsigned long line) -{ - size_t size1; - - size1 = strlen(from) + 1; - if(usesize) - { - if(size < size1) - size1 = size; - } - - return((char *) _Fortify_memcpy(to,from,size1,file,line)); -} - -/* - * Fortify_strncpy() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strncpy(char *to,char *from,size_t size,char *file,unsigned long line) -{ - return(__Fortify_strncpy(to,from,size,1,file,line)); -} - -/* - * Fortify_strncpy() - check if from/to is in allocated space and if so, then check if start/end is not outside allocated space. - */ -char *FORTIFY_STORAGE -_Fortify_strcpy(char *to,char *from,char *file,unsigned long line) -{ - return(__Fortify_strncpy(to,from,0,0,file,line)); -} - -#endif /* FORTIFY */ diff --git a/code/3rd_lpsolve/fortify.h b/code/3rd_lpsolve/fortify.h deleted file mode 100644 index 60d93cb8..00000000 --- a/code/3rd_lpsolve/fortify.h +++ /dev/null @@ -1,293 +0,0 @@ -#ifndef __FORTIFY_H__ -#define __FORTIFY_H__ -/* - * FILE: - * fortify.h - * - * DESCRIPTION: - * Header file for fortify.c - A fortified shell for malloc, realloc, - * calloc, strdup, getcwd, tempnam & free - * - * WRITTEN: - * spb 29/4/94 - * - * VERSION: - * 1.0 29/4/94 - */ -#include - -#include "declare.h" - -#if defined HP9000 || defined AViiON || defined ALPHA || defined SIGNED_UNKNOWN -# define signed -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -#ifdef FORTIFY - -typedef void (*OutputFuncPtr) __OF((char *)); - -extern char *_Fortify_file; -extern int _Fortify_line; - -#define Fortify_FILE(file) _Fortify_file=file -#define Fortify_LINE(line) _Fortify_line=line - -#define _Fortify_FILE (_Fortify_file==(char *) 0 ? __FILE__ : _Fortify_file) -#define _Fortify_LINE (_Fortify_line==0 ? __LINE__ : _Fortify_line) - -void _Fortify_Init __OF((char *file, unsigned long line)); -void *_Fortify_malloc __OF((size_t size, char *file, unsigned long line)); -void *_Fortify_realloc __OF((void *ptr, size_t new_size, char *file, unsigned long line)); -void *_Fortify_calloc __OF((size_t nitems, size_t size, char *file, unsigned long line)); -char *_Fortify_strdup __OF((char *str, char *file, unsigned long line)); -void *_Fortify_memcpy __OF((void *to, void *from, size_t size, char *file, unsigned long line)); -void *_Fortify_memmove __OF((void *to, void *from, size_t size, char *file, unsigned long line)); -void *_Fortify_memccpy __OF((void *to, void *from, int c, size_t size, char *file, unsigned long line)); -void *_Fortify_memset __OF((void *buffer, int c, size_t size, char *file, unsigned long line)); -void *_Fortify_memchr __OF((void *buffer, int c, size_t size, char *file, unsigned long line)); -int _Fortify_memcmp __OF((void *buffer1, void *buffer2, size_t size, char *file, unsigned long line)); -int _Fortify_memicmp __OF((void *buffer1, void *buffer2, size_t size, char *file, unsigned long line)); -char *_Fortify_strchr __OF((char *buffer, int c, char *file, unsigned long line)); -char *_Fortify_strrchr __OF((char *buffer, int c, char *file, unsigned long line)); -char *_Fortify_strlwr __OF((char *buffer, char *file, unsigned long line)); -char *_Fortify_strupr __OF((char *buffer, char *file, unsigned long line)); -char *_Fortify_strrev __OF((char *buffer, char *file, unsigned long line)); -char *_Fortify_strset __OF((char *buffer, int c, char *file, unsigned long line)); -char *_Fortify_strnset __OF((char *buffer, int c, size_t size, char *file, unsigned long line)); -char *_Fortify_strstr __OF((char *to, char *from, char *file, unsigned long line)); -char *_Fortify_strcpy __OF((char *to, char *from, char *file, unsigned long line)); -char *_Fortify_strncpy __OF((char *to, char *from, size_t size, char *file, unsigned long line)); -int _Fortify_strcmp __OF((char *buffer1, char *buffer2, char *file, unsigned long line)); -double _Fortify_strtod __OF((char *buffer1, char **buffer2, char *file, unsigned long line)); -long _Fortify_strtol __OF((char *buffer1, char **buffer2, int n, char *file, unsigned long line)); -int _Fortify_atoi __OF((char *buffer1, char *file, unsigned long line)); -long _Fortify_atol __OF((char *buffer1, char *file, unsigned long line)); -double _Fortify_atof __OF((char *buffer1, char *file, unsigned long line)); -unsigned long _Fortify_strtoul __OF((char *buffer1, char **buffer2, int n, char *file, unsigned long line)); -size_t _Fortify_strcspn __OF((char *buffer1, char *buffer2, char *file, unsigned long line)); -int _Fortify_strcoll __OF((char *buffer1, char *buffer2, char *file, unsigned long line)); -int _Fortify_strcmpi __OF((char *buffer1, char *buffer2, char *file, unsigned long line)); -int _Fortify_strncmp __OF((char *buffer1, char *buffer2, size_t size, char *file, unsigned long line)); -int _Fortify_strnicmp __OF((char *buffer1, char *buffer2, size_t size, char *file, unsigned long line)); -char *_Fortify_strcat __OF((char *buffer1, char *buffer2, char *file, unsigned long line)); -char *_Fortify_strpbrk __OF((char *buffer1, char *buffer2, char *file, unsigned long line)); -size_t _Fortify_strlen __OF((char *buffer, char *file, unsigned long line)); -char *_Fortify_getcwd __OF((char *buf, int size, char *file, unsigned long line)); -char *_Fortify_tempnam __OF((char *dir, char *pfx, char *file, unsigned long line)); -void _Fortify_free __OF((void *uptr, char *file, unsigned long line)); - -int _Fortify_OutputAllMemory __OF((char *file, unsigned long line)); -int _Fortify_CheckAllMemory __OF((char *file, unsigned long line)); -int _Fortify_CheckPointer __OF((void *uptr, char *file, unsigned long line)); -int _Fortify_Disable __OF((char *file, unsigned long line, int how)); -int _Fortify_SetMallocFailRate __OF((int Percent)); -int _Fortify_EnterScope __OF((char *file, unsigned long line)); -int _Fortify_LeaveScope __OF((char *file, unsigned long line)); -int _Fortify_DumpAllMemory __OF((int scope, char *file, unsigned long line)); - -typedef void (*Fortify_OutputFuncPtr) __OF((/* const */ char *)); -Fortify_OutputFuncPtr _Fortify_SetOutputFunc __OF((Fortify_OutputFuncPtr Output)); - -#endif /* FORTIFY */ - -#ifdef __cplusplus -} -#endif - -#ifndef __FORTIFY_C__ /* Only define the macros if we're NOT in fortify.c */ - -#ifdef FORTIFY /* Add file and line information to the fortify calls */ - -#if defined malloc -# undef malloc -#endif -#if defined realloc -# undef realloc -#endif -#if defined calloc -# undef calloc -#endif -#if defined strdup -# undef strdup -#endif -#if defined memcpy -# undef memcpy -#endif -#if defined memmove -# undef memmove -#endif -#if defined memccpy -# undef memccpy -#endif -#if defined memset -# undef memset -#endif -#if defined memchr -# undef memchr -#endif -#if defined memcmp -# undef memcmp -#endif -#if defined memicmp -# undef memicmp -#endif -#if defined strcoll -# undef strcoll -#endif -#if defined strcspn -# undef strcspn -#endif -#if defined strcmp -# undef strcmp -#endif -#if defined strcmpi -# undef strcmpi -#endif -#if defined stricmp -# undef stricmp -#endif -#if defined strncmp -# undef strncmp -#endif -#if defined strnicmp -# undef strnicmp -#endif -#if defined strlwr -# undef strlwr -#endif -#if defined strupr -# undef strupr -#endif -#if defined strrev -# undef strrev -#endif -#if defined strchr -# undef strchr -#endif -#if defined strrchr -# undef strrchr -#endif -#if defined strcat -# undef strcat -#endif -#if defined strpbrk -# undef strpbrk -#endif -#if defined strcpy -# undef strcpy -#endif -#if defined atoi -# undef atoi -#endif -#if defined atol -# undef atol -#endif -#if defined atof -# undef atof -#endif -#if defined strtol -# undef strtol -#endif -#if defined strtoul -# undef strtoul -#endif -#if defined strtod -# undef strtod -#endif -#if defined strstr -# undef strstr -#endif -#if defined strncpy -# undef strncpy -#endif -#if defined strset -# undef strset -#endif -#if defined strnset -# undef strnset -#endif -#if defined strlen -# undef strlen -#endif -#if defined getcwd -# undef getcwd -#endif -#if defined tempnam -# undef tempnam -#endif -#if defined free -# undef free -#endif - -#define malloc(size) _Fortify_malloc(size, _Fortify_FILE, _Fortify_LINE) -#define realloc(ptr,new_size) _Fortify_realloc(ptr, new_size, _Fortify_FILE, _Fortify_LINE) -#define calloc(num,size) _Fortify_calloc(num, size, _Fortify_FILE, _Fortify_LINE) -#define strdup(str) _Fortify_strdup(str, _Fortify_FILE, _Fortify_LINE) -#define memcpy(to,from,size) _Fortify_memcpy((void *)(to),(void *)(from),size, _Fortify_FILE, _Fortify_LINE) -#define memmove(to,from,size) _Fortify_memmove((void *)(to),(void *)(from),size, _Fortify_FILE, _Fortify_LINE) -#define memccpy(to,from,c,size) _Fortify_memccpy((void *)(to),(void *)(from),c,size, _Fortify_FILE, _Fortify_LINE) -#define memset(buffer,c,size) _Fortify_memset(buffer,c,size, _Fortify_FILE, _Fortify_LINE) -#define memchr(buffer,c,size) _Fortify_memchr(buffer,c,size, _Fortify_FILE, _Fortify_LINE) -#define memcmp(buffer1,buffer2,size) _Fortify_memcmp((void *)buffer1,(void *)buffer2,size, _Fortify_FILE, _Fortify_LINE) -#define memicmp(buffer1,buffer2,size) _Fortify_memicmp((void *)buffer1,(void *)buffer2,size, _Fortify_FILE, _Fortify_LINE) -#define strlwr(buffer) _Fortify_strlwr(buffer, _Fortify_FILE, _Fortify_LINE) -#define strupr(buffer) _Fortify_strupr(buffer, _Fortify_FILE, _Fortify_LINE) -#define strrev(buffer) _Fortify_strrev(buffer, _Fortify_FILE, _Fortify_LINE) -#define strchr(buffer,c) _Fortify_strchr(buffer,c, _Fortify_FILE, _Fortify_LINE) -#define strrchr(buffer,c) _Fortify_strrchr(buffer,c, _Fortify_FILE, _Fortify_LINE) -#define strset(buffer,c) _Fortify_strset(buffer,c, _Fortify_FILE, _Fortify_LINE) -#define strnset(buffer,c) _Fortify_strnset(buffer,c, _Fortify_FILE, _Fortify_LINE) -#define strstr(buffer1,buffer2) _Fortify_strstr(buffer1,buffer2, _Fortify_FILE, _Fortify_LINE) -#define atoi(buffer) _Fortify_atoi(buffer, _Fortify_FILE, _Fortify_LINE) -#define atol(buffer) _Fortify_atol(buffer, _Fortify_FILE, _Fortify_LINE) -#define atof(buffer) _Fortify_atof(buffer, _Fortify_FILE, _Fortify_LINE) -#define strtol(buffer1,buffer2,n) _Fortify_strtol(buffer1,buffer2,n, _Fortify_FILE, _Fortify_LINE) -#define strtoul(buffer1,buffer2,n) _Fortify_strtoul(buffer1,buffer2,n, _Fortify_FILE, _Fortify_LINE) -#define strtod(buffer1,buffer2) _Fortify_strtod(buffer1,buffer2, _Fortify_FILE, _Fortify_LINE) -#define strcpy(to,from) _Fortify_strcpy((char *)(to),(char *)(from), _Fortify_FILE, _Fortify_LINE) -#define strncpy(to,from,size) _Fortify_strncpy((char *)(to),(char *)(from),size, _Fortify_FILE, _Fortify_LINE) -#define strcoll(buffer1,buffer2) _Fortify_strcoll((char *)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define strcspn(buffer1,buffer2) _Fortify_strcspn((char*)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define strcmp(buffer1,buffer2) _Fortify_strcmp((char *)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define strcmpi(buffer1,buffer2) _Fortify_strcmpi((char *)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define stricmp(buffer1,buffer2) _Fortify_strcmpi((char *)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define strncmp(buffer1,buffer2,size) _Fortify_strncmp((char *)(buffer1),(char *)(buffer2),size, _Fortify_FILE, _Fortify_LINE) -#define strnicmp(buffer1,buffer2,size) _Fortify_strnicmp((char *)(buffer1),(char *)(buffer2),size, _Fortify_FILE, _Fortify_LINE) -#define strcat(buffer1,buffer2) _Fortify_strcat((char *)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define strpbrk(buffer1,buffer2) _Fortify_strpbrk((char *)(buffer1),(char *)(buffer2), _Fortify_FILE, _Fortify_LINE) -#define strlen(buffer) _Fortify_strlen((char*)(buffer), _Fortify_FILE, _Fortify_LINE) -#define getcwd(buf,size) _Fortify_getcwd(buf, size, _Fortify_FILE, _Fortify_LINE) -#define tempnam(dir,pfx) _Fortify_tempnam(dir, pfx, _Fortify_FILE, _Fortify_LINE) -#define free(ptr) _Fortify_free(ptr, _Fortify_FILE, _Fortify_LINE) - -#define Fortify_Init() _Fortify_Init(_Fortify_FILE, _Fortify_LINE) -#define Fortify_OutputAllMemory() _Fortify_OutputAllMemory(_Fortify_FILE, _Fortify_LINE) -#define Fortify_CheckAllMemory() _Fortify_CheckAllMemory(_Fortify_FILE, _Fortify_LINE) -#define Fortify_CheckPointer(ptr) _Fortify_CheckPointer(ptr, _Fortify_FILE, _Fortify_LINE) -#define Fortify_Disable(how) _Fortify_Disable(_Fortify_FILE, _Fortify_LINE,how) -#define Fortify_EnterScope() _Fortify_EnterScope(_Fortify_FILE, _Fortify_LINE) -#define Fortify_LeaveScope() _Fortify_LeaveScope(_Fortify_FILE, _Fortify_LINE) -#define Fortify_DumpAllMemory(s) _Fortify_DumpAllMemory(s,_Fortify_FILE, _Fortify_LINE) - -#else /* FORTIFY Define the special fortify functions away to nothing */ - -#define Fortify_FILE(file) -#define Fortify_LINE(line) -#define Fortify_Init() -#define Fortify_OutputAllMemory() 0 -#define Fortify_CheckAllMemory() 0 -#define Fortify_CheckPointer(ptr) 1 -#define Fortify_Disable(how) 1 -#define Fortify_SetOutputFunc() 0 -#define Fortify_SetMallocFailRate(p) 0 -#define Fortify_EnterScope() 0 -#define Fortify_LeaveScope() 0 -#define Fortify_DumpAllMemory(s) 0 - -#endif /* FORTIFY */ -#endif /* __FORTIFY_C__ */ -#endif /* __FORTIFY_H__ */ diff --git a/code/3rd_lpsolve/ini.c b/code/3rd_lpsolve/ini.c deleted file mode 100644 index df263f1e..00000000 --- a/code/3rd_lpsolve/ini.c +++ /dev/null @@ -1,76 +0,0 @@ -#include -#include -#include - -#include "lp_lib.h" - -#include "ini.h" - -FILE *ini_create(char *filename) -{ - FILE *fp; - - fp = fopen(filename, "w"); - - return(fp); -} - -FILE *ini_open(char *filename) -{ - FILE *fp; - - fp = fopen(filename, "r"); - - return(fp); -} - -void ini_writecomment(FILE *fp, char *comment) -{ - fprintf(fp, "; %s\n", comment); -} - -void ini_writeheader(FILE *fp, char *header, int addnewline) -{ - if((addnewline) && (ftell(fp) > 0)) - fputs("\n", fp); - fprintf(fp, "[%s]\n", header); -} - -void ini_writedata(FILE *fp, char *name, char *data) -{ - if(name != NULL) - fprintf(fp, "%s=%s\n", name, data); - else - fprintf(fp, "%s\n", data); -} - -int ini_readdata(FILE *fp, char *data, int szdata, int withcomment) -{ - int l; - char *ptr; - - if(fgets(data, szdata, fp) == NULL) - return(0); - - if(!withcomment) { - ptr = strchr(data, ';'); - if(ptr != NULL) - *ptr = 0; - } - - l = (int) strlen(data); - while((l > 0) && (isspace(data[l - 1]))) - l--; - data[l] = 0; - if((l >= 2) && (data[0] == '[') && (data[l - 1] == ']')) { - memcpy(data, data + 1, l - 2); - data[l - 2] = 0; - return(1); - } - return(2); -} - -void ini_close(FILE *fp) -{ - fclose(fp); -} diff --git a/code/3rd_lpsolve/ini.h b/code/3rd_lpsolve/ini.h deleted file mode 100644 index 3bef0d7b..00000000 --- a/code/3rd_lpsolve/ini.h +++ /dev/null @@ -1,17 +0,0 @@ -#include - -#ifdef __cplusplus -__EXTERN_C { -#endif - -extern FILE *ini_create(char *filename); -extern FILE *ini_open(char *filename); -extern void ini_writecomment(FILE *fp, char *comment); -extern void ini_writeheader(FILE *fp, char *header, int addnewline); -extern void ini_writedata(FILE *fp, char *name, char *data); -extern int ini_readdata(FILE *fp, char *data, int szdata, int withcomment); -extern void ini_close(FILE *fp); - -#ifdef __cplusplus -} -#endif diff --git a/code/3rd_lpsolve/lp_Hash.c b/code/3rd_lpsolve/lp_Hash.c deleted file mode 100644 index 5a5276ac..00000000 --- a/code/3rd_lpsolve/lp_Hash.c +++ /dev/null @@ -1,238 +0,0 @@ - -#include -#include -#include "lp_lib.h" -#include "lp_utils.h" -#include "lp_report.h" -#include "lp_Hash.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -#define HASH_START_SIZE 5000 /* Hash table size for row and column name storage */ -#define NUMHASHPRIMES 45 - -STATIC hashtable *create_hash_table(int size, int base) -{ - int i; - int HashPrimes[ ] = { - 29, 229, 883, 1671, 2791, 4801, 8629, 10007, - 15289, 25303, 34843, 65269, 99709, 129403, 147673, 166669, - 201403, 222163, 242729, 261431, 303491, 320237, 402761, 501131, - 602309, 701507, 800999, 900551, 1000619, 1100837, 1200359, 1300021, - 1400017, 1500007, 1750009, 2000003, 2500009, 3000017, 4000037, 5000011, - 6000011, 7000003, 8000009, 9000011, 9999991}; - hashtable *ht; - - /* Find a good size for the hash table */ - if(size < HASH_START_SIZE) - size = HASH_START_SIZE; - for(i = 0; i < NUMHASHPRIMES-1; i++) - if(HashPrimes[i] > size) - break; - size = HashPrimes[i]; - - /* Then allocate and initialize memory */ - ht = (hashtable *) calloc(1 , sizeof(*ht)); - ht->table = (hashelem **) calloc(size, sizeof(*(ht->table))); - ht->size = size; - ht->base = base; - ht->count = base-1; - - return(ht); -} - -STATIC void free_hash_item(hashelem **hp) -{ - free((*hp)->name); - free(*hp); - *hp = NULL; -} - -STATIC void free_hash_table(hashtable *ht) -{ - hashelem *hp, *thp; - - hp = ht->first; - while(hp != NULL) { - thp = hp; - hp = hp->nextelem; - free_hash_item(&thp); - } - free(ht->table); - free(ht); -} - - -/* make a good hash function for any int size */ -/* inspired by Aho, Sethi and Ullman, Compilers ..., p436 */ -#define HASH_1 sizeof(unsigned int) -#define HASH_2 (sizeof(unsigned int) * 6) -#define HASH_3 (((unsigned int)0xF0) << ((sizeof(unsigned int) - 1) * CHAR_BIT)) - -STATIC int hashval(const char *string, int size) -{ - unsigned int result = 0, tmp; - - for(; *string; string++) { - result = (result << HASH_1) + *string; - if((tmp = result & HASH_3) != 0) { - /* if any of the most significant bits is on */ - result ^= tmp >> HASH_2; /* xor them in in a less significant part */ - result ^= tmp; /* and reset the most significant bits to 0 */ - } - } - return(result % size); -} /* hashval */ - - -STATIC hashelem *findhash(const char *name, hashtable *ht) -{ - hashelem *h_tab_p; - for(h_tab_p = ht->table[hashval(name, ht->size)]; - h_tab_p != NULL; - h_tab_p = h_tab_p->next) - if(strcmp(name, h_tab_p->name) == 0) /* got it! */ - break; - return(h_tab_p); -} /* findhash */ - - -STATIC hashelem *puthash(const char *name, int index, hashelem **list, hashtable *ht) -{ - hashelem *hp = NULL; - int hashindex; - - if(list != NULL) { - hp = list[index]; - if(hp != NULL) - list[index] = NULL; - } - - if((hp = findhash(name, ht)) == NULL) { - - hashindex = hashval(name, ht->size); - hp = (hashelem *) calloc(1, sizeof(*hp)); - allocCHAR(NULL, &hp->name, (int) (strlen(name) + 1), FALSE); - strcpy(hp->name, name); - hp->index = index; - ht->count++; - if(list != NULL) - list[index] = hp; - - hp->next = ht->table[hashindex]; - ht->table[hashindex] = hp; - if(ht->first == NULL) - ht->first = hp; - if(ht->last != NULL) - ht->last->nextelem = hp; - ht->last = hp; - - } - return(hp); -} - -STATIC void drophash(const char *name, hashelem **list, hashtable *ht) { - hashelem *hp, *hp1, *hp2; - int hashindex; - - if((hp = findhash(name, ht)) != NULL) { - hashindex = hashval(name, ht->size); - if((hp1 = ht->table[hashindex]) != NULL) { - hp2 = NULL; - while((hp1 != NULL) && (hp1 != hp)) { - hp2 = hp1; - hp1 = hp1->next; - } - if(hp1 == hp) { - if(hp2 != NULL) - hp2->next = hp->next; - else - ht->table[hashindex] = hp->next; - } - - hp1 = ht->first; - hp2 = NULL; - while((hp1 != NULL) && (hp1 != hp)) { - hp2 = hp1; - hp1 = hp1->nextelem; - } - if(hp1 == hp) { - if(hp2 != NULL) - hp2->nextelem = hp->nextelem; - else { - ht->first = hp->nextelem; - if (ht->first == NULL) - ht->last = NULL; - } - } - if(list != NULL) - list[hp->index] = NULL; - free_hash_item(&hp); - ht->count--; - } - } -} - -STATIC hashtable *copy_hash_table(hashtable *ht, hashelem **list, int newsize) -{ - hashtable *copy; - hashelem *elem, *new_elem; - - if(newsize < ht->size) - newsize = ht->size; - - copy = create_hash_table(newsize, ht->base); - if (copy != NULL) { - elem = ht->first; - while (elem != NULL) { - if((new_elem = puthash(elem->name, elem->index, list, copy)) == NULL) { - free_hash_table(copy); - return(NULL); - } - elem = elem ->nextelem; - } - } - - return(copy); -} - -STATIC int find_row(lprec *lp, char *name, MYBOOL Unconstrained_rows_found) -{ - hashelem *hp; - - if (lp->rowname_hashtab != NULL) - hp = findhash(name, lp->rowname_hashtab); - else - hp = NULL; - - if (hp == NULL) { - if(Unconstrained_rows_found) { /* just ignore them in this case */ - return(-1); - } - else { - return(-1); - } - } - return(hp->index); -} - -STATIC int find_var(lprec *lp, char *name, MYBOOL verbose) -{ - hashelem *hp; - - if (lp->colname_hashtab != NULL) - hp = findhash(name, lp->colname_hashtab); - else - hp = NULL; - - if (hp == NULL) { - if(verbose) - report(lp, SEVERE, "find_var: Unknown variable name '%s'\n", name); - return(-1); - } - return(hp->index); -} - diff --git a/code/3rd_lpsolve/lp_Hash.h b/code/3rd_lpsolve/lp_Hash.h deleted file mode 100644 index b286960c..00000000 --- a/code/3rd_lpsolve/lp_Hash.h +++ /dev/null @@ -1,43 +0,0 @@ -#ifndef HEADER_lp_hash -#define HEADER_lp_hash - -/* For row and column name hash tables */ - -typedef struct _hashelem -{ - char *name; - int index; - struct _hashelem *next; - struct _hashelem *nextelem; -} hashelem; - -typedef struct /* _hashtable */ -{ - hashelem **table; - int size; - int base; - int count; - struct _hashelem *first; - struct _hashelem *last; -} hashtable; - -#ifdef __cplusplus -extern "C" { -#endif - -STATIC hashtable *create_hash_table(int size, int base); -STATIC void free_hash_table(hashtable *ht); -STATIC hashelem *findhash(const char *name, hashtable *ht); -STATIC hashelem *puthash(const char *name, int index, hashelem **list, hashtable *ht); -STATIC void drophash(const char *name, hashelem **list, hashtable *ht); -STATIC void free_hash_item(hashelem **hp); -STATIC hashtable *copy_hash_table(hashtable *ht, hashelem **list, int newsize); -STATIC int find_var(lprec *lp, char *name, MYBOOL verbose); -STATIC int find_row(lprec *lp, char *name, MYBOOL Unconstrained_rows_found); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_hash */ - diff --git a/code/3rd_lpsolve/lp_MDO.c b/code/3rd_lpsolve/lp_MDO.c deleted file mode 100644 index 12172298..00000000 --- a/code/3rd_lpsolve/lp_MDO.c +++ /dev/null @@ -1,241 +0,0 @@ -/* - Minimum matrix inverse fill-in modules - interface for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: string.h, colamd.h, lp_lib.h - - Release notes: - v1.0 1 September 2003 Preprocessing routines for minimum fill-in column - ordering for inverse factorization using the open - source COLAMD library. Suitable for the dense parts - of both the product form and LU factorization inverse - methods. - v1.1 1 July 2004 Renamed from lp_colamdMDO to lp_MDO. - - ---------------------------------------------------------------------------------- -*/ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "colamd.h" -#include "lp_MDO.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -STATIC MYBOOL includeMDO(MYBOOL *usedpos, int item) -{ -/* Legend: TRUE => A basic slack variable already in the basis - FALSE => A column free for being pivoted in - AUTOMATIC+TRUE => A row-singleton user column pivoted into the basis - AUTOMATIC+FALSE => A column-singleton user column pivoted into the basis */ - - /* Handle case where we are processing all columns */ - if(usedpos == NULL) - return( TRUE ); - - else { - /* Otherwise do the selective case */ - MYBOOL test = usedpos[item]; -#if 1 - return( test != TRUE ); -#else - test = test & TRUE; - return( test == FALSE ); -#endif - } -} - -STATIC int prepareMDO(lprec *lp, MYBOOL *usedpos, int *colorder, int *data, int *rowmap) -/* This routine prepares data structures for colamd(). It is called twice, the first - time to count applicable non-zero elements by column, and the second time to fill in - the row indexes of the non-zero values from the first call. Note that the colamd() - row index base is 0 (which suits lp_solve fine). */ -{ - int i, ii, j, k, kk; - int nrows = lp->rows+1, ncols = colorder[0]; - int offset = 0, Bnz = 0, Tnz; - MYBOOL dotally = (MYBOOL) (rowmap == NULL); - MATrec *mat = lp->matA; - REAL hold; - REAL *value; - int *rownr; - - if(dotally) - data[0] = 0; - - Tnz = nrows - ncols; - for(j = 1; j <= ncols; j++) { - kk = colorder[j]; - - /* Process slacks */ - if(kk <= lp->rows) { - if(includeMDO(usedpos, kk)) { - if(!dotally) - data[Bnz] = rowmap[kk]+offset; - Bnz++; - } - Tnz++; - } - /* Process user columns */ - else { - k = kk - lp->rows; - i = mat->col_end[k-1]; - ii= mat->col_end[k]; - Tnz += ii-i; -#ifdef Paranoia - if(i >= ii) - lp->report(lp, SEVERE, "prepareMDO: Encountered empty basic column %d\n", k); -#endif - - /* Detect if we need to do phase 1 adjustments of zero-valued OF variable */ - rownr = &COL_MAT_ROWNR(i); - value = &COL_MAT_VALUE(i); - hold = 0; - if((*rownr > 0) && includeMDO(usedpos, 0) && modifyOF1(lp, kk, &hold, 1.0)) { - if(!dotally) - data[Bnz] = offset; - Bnz++; - } - /* Loop over all NZ-variables */ - for(; i < ii; - i++, value += matValueStep, rownr += matRowColStep) { - if(!includeMDO(usedpos, *rownr)) - continue; - /* See if we need to change phase 1 OF value */ - if(*rownr == 0) { - hold = *value; - if(!modifyOF1(lp, kk, &hold, 1.0)) - continue; - } - /* Tally uneliminated constraint row values */ - if(!dotally) - data[Bnz] = rowmap[*rownr]+offset; - Bnz++; - } - } - if(dotally) - data[j] = Bnz; - } - return( Tnz ); -} - -STATIC MYBOOL verifyMDO(lprec *lp, int *col_end, int *row_nr, int rowmax, int colmax) -{ - int i, j, n, err = 0; - - for(i = 1; i <= colmax; i++) { - n = 0; - for(j = col_end[i-1]; (j < col_end[i]) && (err == 0); j++, n++) { - if(row_nr[j] < 0 || row_nr[j] > rowmax) - err = 1; - if(n > 0 && row_nr[j] <= row_nr[j-1]) - err = 2; - n++; - } - } - if(err != 0) - lp->report(lp, SEVERE, "verifyMDO: Invalid MDO input structure generated (error %d)\n", err); - return( (MYBOOL) (err == 0) ); -} - -void *mdo_calloc(size_t size, size_t count) -{ - return ( calloc(size, count) ); -} -void mdo_free(void *mem) -{ - free( mem ); -} - - -int __WINAPI getMDO(lprec *lp, MYBOOL *usedpos, int *colorder, int *size, MYBOOL symmetric) -{ - int error = FALSE; - int nrows = lp->rows+1, ncols = colorder[0]; - int i, j, kk, n; - int *col_end, *row_map = NULL; - int Bnz, Blen, *Brows = NULL; - int stats[COLAMD_STATS]; - double knobs[COLAMD_KNOBS]; - - /* Tally the non-zero counts of the unused columns/rows of the - basis matrix and store corresponding "net" starting positions */ - allocINT(lp, &col_end, ncols+1, FALSE); - n = prepareMDO(lp, usedpos, colorder, col_end, NULL); - Bnz = col_end[ncols]; - - /* Check that we have unused basic columns, otherwise skip analysis */ - if(ncols == 0 || Bnz == 0) - goto Transfer; - - /* Get net number of rows and fill mapper */ - allocINT(lp, &row_map, nrows, FALSE); - nrows = 0; - for(i = 0; i <= lp->rows; i++) { - row_map[i] = i-nrows; - /* Increment eliminated row counter if necessary */ - if(!includeMDO(usedpos, i)) - nrows++; - } - nrows = lp->rows+1 - nrows; - - /* Store row indeces of non-zero values in the basic columns */ - Blen = colamd_recommended(Bnz, nrows, ncols); - allocINT(lp, &Brows, Blen, FALSE); - prepareMDO(lp, usedpos, colorder, Brows, row_map); -#ifdef Paranoia - verifyMDO(lp, col_end, Brows, nrows, ncols); -#endif - - /* Compute the MDO */ -#if 1 - colamd_set_defaults(knobs); - knobs [COLAMD_DENSE_ROW] = 0.2+0.2 ; /* default changed for UMFPACK */ - knobs [COLAMD_DENSE_COL] = knobs [COLAMD_DENSE_ROW]; - if(symmetric && (nrows == ncols)) { - MEMCOPY(colorder, Brows, ncols + 1); - error = !symamd(nrows, colorder, col_end, Brows, knobs, stats, mdo_calloc, mdo_free); - } - else - error = !colamd(nrows, ncols, Blen, Brows, col_end, knobs, stats); -#else - if(symmetric && (nrows == ncols)) { - MEMCOPY(colorder, Brows, ncols + 1); - error = !symamd(nrows, colorder, col_end, Brows, knobs, stats, mdo_calloc, mdo_free); - } - else - error = !colamd(nrows, ncols, Blen, Brows, col_end, (double *) NULL, stats); -#endif - - /* Transfer the estimated optimal ordering, adjusting for index offsets */ -Transfer: - if(error) - error = stats[COLAMD_STATUS]; - else { - MEMCOPY(Brows, colorder, ncols + 1); - for(j = 0; j < ncols; j++) { - kk = col_end[j]; - n = Brows[kk+1]; - colorder[j+1] = n; - } - } - - /* Free temporary vectors */ - FREE(col_end); - if(row_map != NULL) - FREE(row_map); - if(Brows != NULL) - FREE(Brows); - - if(size != NULL) - *size = ncols; - return( error ); -} - - diff --git a/code/3rd_lpsolve/lp_MDO.h b/code/3rd_lpsolve/lp_MDO.h deleted file mode 100644 index 84753638..00000000 --- a/code/3rd_lpsolve/lp_MDO.h +++ /dev/null @@ -1,18 +0,0 @@ -#ifndef HEADER_MDO -#define HEADER_MDO - -#include "lp_types.h" - - -#ifdef __cplusplus -extern "C" { -#endif - -int __WINAPI getMDO(lprec *lp, MYBOOL *usedpos, int *colorder, int *size, MYBOOL symmetric); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_MDO */ - diff --git a/code/3rd_lpsolve/lp_MPS.c b/code/3rd_lpsolve/lp_MPS.c deleted file mode 100644 index 40f61399..00000000 --- a/code/3rd_lpsolve/lp_MPS.c +++ /dev/null @@ -1,1854 +0,0 @@ - -#include -#include -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_scale.h" -#include "lp_report.h" -#include "lp_MPS.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* Define buffer-size controled function mapping */ -# if defined _MSC_VER -# define vsnprintf _vsnprintf -# endif - -/* MPS file input and output routines for lp_solve */ -/* ------------------------------------------------------------------------- */ - -/* -A: MPS format was named after an early IBM LP product and has emerged -as a de facto standard ASCII medium among most of the commercial LP -codes. Essentially all commercial LP codes accept this format, but if -you are using public domain software and have MPS files, you may need -to write your own reader routine for this. It's not too hard. The -main things to know about MPS format are that it is column oriented (as -opposed to entering the model as equations), and everything (variables, -rows, etc.) gets a name. MPS format is described in more detail in -Murtagh's book, referenced in another section. Also, - -ftp://softlib.cs.rice.edu/pub/miplib/mps_format - -is a nice short introduction. exports - -MPS is an old format, so it is set up as though you were using punch -cards, and is not free format. Fields start in column 1, 5, 15, 25, 40 -and 50. Sections of an MPS file are marked by so-called header cards, -which are distinguished by their starting in column 1. Although it is -typical to use upper-case throughout the file (like I said, MPS has -long historical roots), many MPS-readers will accept mixed-case for -anything except the header cards, and some allow mixed-case anywhere. -The names that you choose for the individual entities (constraints or -variables) are not important to the solver; you should pick names that -are meaningful to you, or will be easy for a post-processing code to -read. - -Here is a little sample model written in MPS format (explained in more -detail below): - -NAME TESTPROB -ROWS - N COST - L LIM1 - G LIM2 - E MYEQN -COLUMNS - XONE COST 1 LIM1 1 - XONE LIM2 1 - YTWO COST 4 LIM1 1 - YTWO MYEQN -1 - ZTHREE COST 9 LIM2 1 - ZTHREE MYEQN 1 -RHS - RHS1 LIM1 5 LIM2 10 - RHS1 MYEQN 7 -BOUNDS - UP BND1 XONE 4 - LO BND1 YTWO -1 - UP BND1 YTWO 1 -ENDATA - -means: - -Optimize - COST: XONE + 4 YTWO + 9 ZTHREE -Subject To - LIM1: XONE + YTWO <= 5 - LIM2: XONE + ZTHREE >= 10 - MYEQN: - YTWO + ZTHREE = 7 -Bounds - 0 <= XONE <= 4 --1 <= YTWO <= 1 -End - -*/ - -/* copy a MPS name, only trailing spaces are removed. In MPS, names can have - embedded spaces! */ -STATIC void namecpy(char *into, char *from) -{ - int i; - - /* copy at most 8 characters of from, stop at end of string or newline */ - for(i = 0; (from[i] != '\0') && (from[i] != '\n') && (from[i] != '\r') && (i < 8); i++) - into[i] = from[i]; - - /* end with end of string */ - into[i] = '\0'; - - /* remove trailing spaces, if any */ - for(i--; (i >= 0) && (into[i] == ' '); i--) - into[i] = '\0'; -} - -/* scan an MPS line, and pick up the information in the fields that are - present */ - -/* scan_line for fixed MPS format */ -STATIC int scan_lineFIXED(lprec *lp, int section, char* line, char *field1, char *field2, char *field3, - double *field4, char *field5, double *field6) -{ - int items = 0, line_len; - char buf[16], *ptr1, *ptr2; - - line_len = (int) strlen(line); - while ((line_len) && ((line[line_len-1] == '\n') || (line[line_len-1] == '\r') || (line[line_len-1] == ' '))) - line_len--; - - if(line_len >= 1) { /* spaces or N/L/G/E or UP/LO */ - strncpy(buf, line, 4); - buf[4] = '\0'; - sscanf(buf, "%s", field1); - items++; - } - else - field1[0] = '\0'; - - line += 4; - - if(line_len >= 5) { /* name */ - if (line[-1] != ' ') { - report(lp, IMPORTANT, "MPS_readfile: invalid data card; column 4 must be blank\n"); - return(-1); - } - namecpy(field2, line); - items++; - } - else - field2[0] = '\0'; - - line += 10; - - if(line_len >= 14) { /* name */ - if (line[-1] != ' ' || line[-2] != ' ') { - report(lp, IMPORTANT, "MPS_readfile: invalid data card; columns 13-14 must be blank\n"); - return(-1); - } - namecpy(field3, line); - items++; - } - else - field3[0] = '\0'; - - line += 10; - - if(line_len >= 25) { /* number */ - if (line[-1] != ' ' || line[-2] != ' ') { - report(lp, IMPORTANT, "MPS_readfile: invalid data card; columns 23-24 must be blank\n"); - return(-1); - } - strncpy(buf, line, 15); - buf[15] = '\0'; - for(ptr1 = ptr2 = buf; ; ptr1++) - if(!isspace((unsigned char) *ptr1)) - if((*(ptr2++) = *ptr1) == 0) - break; - /* *field4 = atof(buf); */ - *field4 = strtod(buf, &ptr1); - if(*ptr1) { - report(lp, IMPORTANT, "MPS_readfile: invalid number in columns 25-36 \n"); - return(-1); - } - items++; - } - else - *field4 = 0; - - line += 15; - - if(line_len >= 40) { /* name */ - if (line[-1] != ' ' || line[-2] != ' ' || line[-3] != ' ') { - report(lp, IMPORTANT, "MPS_readfile: invalid data card; columns 37-39 must be blank\n"); - return(-1); - } - namecpy(field5, line); - items++; - } - else - field5[0] = '\0'; - line += 10; - - if(line_len >= 50) { /* number */ - if (line[-1] != ' ' || line[-2] != ' ') { - report(lp, IMPORTANT, "MPS_readfile: invalid data card; columns 48-49 must be blank\n"); - return(-1); - } - strncpy(buf, line, 15); - buf[15] = '\0'; - for(ptr1 = ptr2 = buf; ; ptr1++) - if(!isspace((unsigned char) *ptr1)) - if((*(ptr2++) = *ptr1) == 0) - break; - /* *field6 = atof(buf); */ - *field6 = strtod(buf, &ptr1); - if(*ptr1) { - report(lp, IMPORTANT, "MPS_readfile: invalid number in columns 50-61 \n"); - return(-1); - } - items++; - } - else - *field6 = 0; - - return(items); -} - -STATIC int spaces(char *line, int line_len) -{ - int l; - char *line1 = line; - - while (*line1 == ' ') - line1++; - l = (int) (line1 - line); - if (line_len < l) - l = line_len; - return(l); -} - -STATIC int lenfield(char *line, int line_len) -{ - int l; - char *line1 = line; - - while ((*line1) && (*line1 != ' ')) - line1++; - l = (int) (line1 - line); - if (line_len < l) - l = line_len; - return(l); -} - -/* scan_line for fixed MPS format */ -STATIC int scan_lineFREE(lprec *lp, int section, char* line, char *field1, char *field2, char *field3, - double *field4, char *field5, double *field6) -{ - int items = 0, line_len, len; - char buf[256], *ptr1 = NULL, *ptr2; - - line_len = (int) strlen(line); - while ((line_len) && ((line[line_len-1] == '\n') || (line[line_len-1] == '\r') || (line[line_len-1] == ' '))) - line_len--; - - len = spaces(line, line_len); - line += len; - line_len -= len; - - if ((section == MPSCOLUMNS) || (section == MPSRHS) || (section == MPSRANGES)) { - field1[0] = '\0'; - items++; - } - else { - len = lenfield(line, line_len); - if(line_len >= 1) { /* spaces or N/L/G/E or UP/LO */ - strncpy(buf, line, len); - buf[len] = '\0'; - sscanf(buf, "%s", field1); - if(section == MPSBOUNDS) { - for(ptr1 = field1; *ptr1; ptr1++) - *ptr1=(char)toupper(*ptr1); - } - items++; - } - else - field1[0] = '\0'; - - line += len; - line_len -= len; - - len = spaces(line, line_len); - line += len; - line_len -= len; - } - - len = lenfield(line, line_len); - if(line_len >= 1) { /* name */ - strncpy(field2, line, len); - field2[len] = '\0'; - items++; - } - else - field2[0] = '\0'; - - line += len; - line_len -= len; - - len = spaces(line, line_len); - line += len; - line_len -= len; - - len = lenfield(line, line_len); - if(line_len >= 1) { /* name */ - strncpy(field3, line, len); - field3[len] = '\0'; - items++; - } - else - field3[0] = '\0'; - - line += len; - line_len -= len; - - len = spaces(line, line_len); - line += len; - line_len -= len; - - if (*field3) { - if((section == MPSCOLUMNS) && (strcmp(field3, "'MARKER'") == 0)) { - *field4 = 0; - items++; - ptr1 = field3; - } - else if((section == MPSBOUNDS) && - ((strcmp(field1, "FR") == 0) || (strcmp(field1, "MI") == 0) || (strcmp(field1, "PL") == 0) || (strcmp(field1, "BV") == 0))) - /* field3 *is* the variable name */; - else { - /* Some free MPS formats allow that field 2 is not provided after the first time. - The fieldname is then the same as the the defined field the line before. - In that case field2 shifts to field3, field1 shifts to field 2. - This situation is tested by checking if field3 is numerical AND there are an even number of fields after. - */ - char *line1 = line; - int line_len1 = line_len; - int items1 = 0; - - while (line_len1 > 0) { - len = lenfield(line1, line_len1); - if (len > 0) { - line1 += len; - line_len1 -= len; - items1++; - } - len = spaces(line1, line_len1); - line1 += len; - line_len1 -= len; - } - if ((items1 % 2) == 0) { - *field4 = strtod(field3, &ptr1); - if(*ptr1 == 0) { - strcpy(field3, field2); - if ((section == MPSROWS) || (section == MPSBOUNDS) /* || (section == MPSSOS) */) - *field2 = 0; - else { - strcpy(field2, field1); - *field1 = 0; - } - items++; - } - else - ptr1 = NULL; - } - else - ptr1 = NULL; - } - } - else { - ptr1 = NULL; - if((section == MPSBOUNDS) && - ((strcmp(field1, "FR") == 0) || (strcmp(field1, "MI") == 0) || (strcmp(field1, "PL") == 0) || (strcmp(field1, "BV") == 0))) { - strcpy(field3, field2); - *field2 = 0; - items++; - } - } - - if(ptr1 == NULL) { - len = lenfield(line, line_len); - if(line_len >= 1) { /* number */ - strncpy(buf, line, len); - buf[len] = '\0'; - for(ptr1 = ptr2 = buf; ; ptr1++) - if(!isspace((unsigned char) *ptr1)) - if((*(ptr2++) = *ptr1) == 0) - break; - /* *field4 = atof(buf); */ - *field4 = strtod(buf, &ptr1); - if(*ptr1) - return(-1); - items++; - } - else - *field4 = 0; - - line += len; - line_len -= len; - - len = spaces(line, line_len); - line += len; - line_len -= len; - } - - len = lenfield(line, line_len); - if(line_len >= 1) { /* name */ - strncpy(field5, line, len); - field5[len] = '\0'; - items++; - } - else - field5[0] = '\0'; - line += len; - line_len -= len; - - len = spaces(line, line_len); - line += len; - line_len -= len; - - len = lenfield(line, line_len); - if(line_len >= 1) { /* number */ - strncpy(buf, line, len); - buf[len] = '\0'; - for(ptr1 = ptr2 = buf; ; ptr1++) - if(!isspace((unsigned char) *ptr1)) - if((*(ptr2++) = *ptr1) == 0) - break; - /* *field6 = atof(buf); */ - *field6 = strtod(buf, &ptr1); - if(*ptr1) - return(-1); - items++; - } - else - *field6 = 0; - - if((section == MPSSOS) && (items == 2)) { - strcpy(field3, field2); - strcpy(field2, field1); - *field1 = 0; - } - - if((section != MPSOBJNAME) && (section != MPSBOUNDS)) { - for(ptr1 = field1; *ptr1; ptr1++) - *ptr1=(char)toupper(*ptr1); - } - - return(items); -} - -STATIC int addmpscolumn(lprec *lp, MYBOOL Int_section, int typeMPS, MYBOOL *Column_ready, - int *count, REAL *Last_column, int *Last_columnno, char *Last_col_name) -{ - int ok = TRUE; - - if (*Column_ready) { - ok = add_columnex(lp, *count, Last_column, Last_columnno); - if (ok) { - ok = set_col_name(lp, lp->columns, Last_col_name); - } - if (ok) { - set_int(lp, lp->columns, Int_section); - if ((Int_section) && (typeMPS & MPSIBM)) - set_bounds(lp, lp->columns, 10.0 / DEF_INFINITE, DEF_INFINITE / 10.0); - } - } - *Column_ready = FALSE; - *count = 0; - return(ok); -} - -#if 0 -STATIC MYBOOL appendmpsitem(int *count, int rowIndex[], REAL rowValue[]) -{ - int i = *count; - - if(rowValue[i] == 0) - return( FALSE ); - - while((i > 0) && (rowIndex[i] < rowIndex[i-1])) { - swapINT (rowIndex+i, rowIndex+i-1); - swapREAL(rowValue+i, rowValue+i-1); - i--; - } - (*count)++; - return( TRUE ); -} -#endif - -STATIC MYBOOL appendmpsitem(int *count, int rowIndex[], REAL rowValue[]) -{ - int i = *count; - - /* Check for non-negativity of the index */ - if(rowIndex[i] < 0) - return( FALSE ); - - /* Move the element so that the index list is sorted ascending */ - while((i > 0) && (rowIndex[i] < rowIndex[i-1])) { - swapINT (rowIndex+i, rowIndex+i-1); - swapREAL(rowValue+i, rowValue+i-1); - i--; - } - - /* Add same-indexed items (which is rarely encountered), and shorten the list */ - if((i < *count) && (rowIndex[i] == rowIndex[i+1])) { - int ii = i + 1; - rowValue[i] += rowValue[ii]; - (*count)--; - while(ii < *count) { - rowIndex[ii] = rowIndex[ii+1]; - rowValue[ii] = rowValue[ii+1]; - ii++; - } - } - - /* Update the count and return */ - (*count)++; - return( TRUE ); -} - -MYBOOL MPS_readfile(lprec **newlp, char *filename, int typeMPS, int verbose) -{ - MYBOOL status = FALSE; - FILE *fpin; - - fpin = fopen(filename, "r"); - if(fpin != NULL) { - status = MPS_readhandle(newlp, fpin, typeMPS, verbose); - fclose(fpin); - } - return( status ); -} - -static int __WINAPI MPS_input(void *fpin, char *buf, int max_size) -{ - return(fgets(buf, max_size, (FILE *) fpin) != NULL); -} - -MYBOOL __WINAPI MPS_readhandle(lprec **newlp, FILE *filehandle, int typeMPS, int verbose) -{ - return(MPS_readex(newlp, (void *) filehandle, MPS_input, typeMPS, verbose)); -} - -MYBOOL __WINAPI MPS_readex(lprec **newlp, void *userhandle, read_modeldata_func read_modeldata, int typeMPS, int verbose) -{ - char field1[BUFSIZ], field2[BUFSIZ], field3[BUFSIZ], field5[BUFSIZ], line[BUFSIZ], tmp[BUFSIZ], - Last_col_name[BUFSIZ], probname[BUFSIZ], OBJNAME[BUFSIZ], *ptr; - int items, row, Lineno, var, - section = MPSUNDEF, variant = 0, NZ = 0, SOS = 0; - MYBOOL Int_section, Column_ready, Column_ready1, - Unconstrained_rows_found = FALSE, OF_found = FALSE, CompleteStatus = FALSE; - double field4, field6; - REAL *Last_column = NULL; - int count = 0, *Last_columnno = NULL; - int OBJSENSE = ROWTYPE_EMPTY; - lprec *lp; - int (*scan_line)(lprec *lp, int section, char* line, char *field1, char *field2, char *field3, - double *field4, char *field5, double *field6); - - if(newlp == NULL) - return( CompleteStatus ); - else if(*newlp == NULL) - lp = make_lp(0, 0); - else - lp = *newlp; - - if((typeMPS & MPSFIXED) == MPSFIXED) - scan_line = scan_lineFIXED; - else if((typeMPS & MPSFREE) == MPSFREE) - scan_line = scan_lineFREE; - else { - report(lp, IMPORTANT, "MPS_readfile: Unrecognized MPS line type.\n"); - if (*newlp == NULL) - delete_lp(lp); - return( CompleteStatus ); - } - - if (lp != NULL) { - lp->source_is_file = TRUE; - lp->verbose = verbose; - strcpy(Last_col_name, ""); - strcpy(OBJNAME, ""); - Int_section = FALSE; - Column_ready = FALSE; - Lineno = 0; - - /* let's initialize line to all zero's */ - MEMCLEAR(line, BUFSIZ); - - while(read_modeldata(userhandle, line, BUFSIZ - 1)) { - Lineno++; - - for(ptr = line; (*ptr) && (isspace((unsigned char) *ptr)); ptr++); - - /* skip lines which start with "*", they are comment */ - if((line[0] == '*') || (*ptr == 0) || (*ptr == '\n') || (*ptr == '\r')) { - report(lp, FULL, "Comment on line %d: %s", Lineno, line); - continue; - } - - report(lp, FULL, "Line %6d: %s", Lineno, line); - - /* first check for "special" lines: NAME, ROWS, BOUNDS .... */ - /* this must start in the first position of line */ - if(line[0] != ' ') { - sscanf(line, "%s", tmp); - if(strcmp(tmp, "NAME") == 0) { - section = MPSNAME; - *probname = 0; - sscanf(line, "NAME %s", probname); - if (!set_lp_name(lp, probname)) - break; - } - else if(((typeMPS & MPSFREE) == MPSFREE) && (strcmp(tmp, "OBJSENSE") == 0)) { - section = MPSOBJSENSE; - report(lp, FULL, "Switching to OBJSENSE section\n"); - } - else if(((typeMPS & MPSFREE) == MPSFREE) && (strcmp(tmp, "OBJNAME") == 0)) { - section = MPSOBJNAME; - report(lp, FULL, "Switching to OBJNAME section\n"); - } - else if(strcmp(tmp, "ROWS") == 0) { - section = MPSROWS; - report(lp, FULL, "Switching to ROWS section\n"); - } - else if(strcmp(tmp, "COLUMNS") == 0) { - allocREAL(lp, &Last_column, lp->rows + 1, TRUE); - allocINT(lp, &Last_columnno, lp->rows + 1, TRUE); - count = 0; - if ((Last_column == NULL) || (Last_columnno == NULL)) - break; - section = MPSCOLUMNS; - report(lp, FULL, "Switching to COLUMNS section\n"); - } - else if(strcmp(tmp, "RHS") == 0) { - if (!addmpscolumn(lp, Int_section, typeMPS, &Column_ready, &count, Last_column, Last_columnno, Last_col_name)) - break; - section = MPSRHS; - report(lp, FULL, "Switching to RHS section\n"); - } - else if(strcmp(tmp, "BOUNDS") == 0) { - section = MPSBOUNDS; - report(lp, FULL, "Switching to BOUNDS section\n"); - } - else if(strcmp(tmp, "RANGES") == 0) { - section = MPSRANGES; - report(lp, FULL, "Switching to RANGES section\n"); - } - else if((strcmp(tmp, "SOS") == 0) || (strcmp(tmp, "SETS") == 0)) { - section = MPSSOS; - if(strcmp(tmp, "SOS") == 0) - variant = 0; - else - variant = 1; - report(lp, FULL, "Switching to %s section\n", tmp); - } - else if(strcmp(tmp, "ENDATA") == 0) { - report(lp, FULL, "Finished reading MPS file\n"); - CompleteStatus = TRUE; - break; - } - else { /* line does not start with space and does not match above */ - report(lp, IMPORTANT, "Unrecognized MPS line %d: %s\n", Lineno, line); - break; - } - } - else { /* normal line, process */ - items = scan_line(lp, section, line, field1, field2, field3, &field4, field5, &field6); - if(items < 0){ - report(lp, IMPORTANT, "Syntax error on line %d: %s\n", Lineno, line); - break; - } - - switch(section) { - - case MPSNAME: - report(lp, IMPORTANT, "Error, extra line under NAME line\n"); - break; - - case MPSOBJSENSE: - if(OBJSENSE != ROWTYPE_EMPTY) { - report(lp, IMPORTANT, "Error, extra line under OBJSENSE line\n"); - break; - } - if((strcmp(field1, "MAXIMIZE") == 0) || (strcmp(field1, "MAX") == 0)) { - OBJSENSE = ROWTYPE_OFMAX; - set_maxim(lp); - } - else if((strcmp(field1, "MINIMIZE") == 0) || (strcmp(field1, "MIN") == 0)) { - OBJSENSE = ROWTYPE_OFMIN; - set_minim(lp); - } - else { - report(lp, SEVERE, "Unknown OBJSENSE direction '%s' on line %d\n", field1, Lineno); - break; - } - continue; - - case MPSOBJNAME: - if(*OBJNAME) { - report(lp, IMPORTANT, "Error, extra line under OBJNAME line\n"); - break; - } - strcpy(OBJNAME, field1); - continue; - - /* Process entries in the ROWS section */ - case MPSROWS: - /* field1: rel. operator; field2: name of constraint */ - - report(lp, FULL, "Row %5d: %s %s\n", lp->rows + 1, field1, field2); - - if(strcmp(field1, "N") == 0) { - if((*OBJNAME) && (strcmp(field2, OBJNAME))) - /* Ignore this objective name since it is not equal to the OBJNAME name */; - else if(!OF_found) { /* take the first N row as OF, ignore others */ - if (!set_row_name(lp, 0, field2)) - break; - OF_found = TRUE; - } - else if(!Unconstrained_rows_found) { - report(lp, IMPORTANT, "Unconstrained row %s ignored\n", field2); - report(lp, IMPORTANT, "Further messages of this kind will be suppressed\n"); - Unconstrained_rows_found = TRUE; - } - } - else if(strcmp(field1, "L") == 0) { - if ((!str_add_constraint(lp, "" ,LE ,0)) || (!set_row_name(lp, lp->rows, field2))) - break; - } - else if(strcmp(field1, "G") == 0) { - if ((!str_add_constraint(lp, "" ,GE ,0)) || (!set_row_name(lp, lp->rows, field2))) - break; - } - else if(strcmp(field1, "E") == 0) { - if ((!str_add_constraint(lp, "",EQ ,0)) || (!set_row_name(lp, lp->rows, field2))) - break; - } - else { - report(lp, SEVERE, "Unknown relation code '%s' on line %d\n", field1, Lineno); - break; - } - - continue; - - /* Process entries in the COLUMNS section */ - case MPSCOLUMNS: - /* field2: variable; field3: constraint; field4: coef */ - /* optional: field5: constraint; field6: coef */ - - report(lp, FULL, "Column %4d: %s %s %g %s %g\n", - lp->columns + 1, field2, field3, field4, field5, field6); - - if((items == 4) || (items == 5) || (items == 6)) { - if (NZ == 0) - strcpy(Last_col_name, field2); - else if(*field2) { - Column_ready1 = (MYBOOL) (strcmp(field2, Last_col_name) != 0); - if(Column_ready1) { - if (find_var(lp, field2, FALSE) >= 0) { - report(lp, SEVERE, "Variable name (%s) is already used!\n", field2); - break; - } - - if(Column_ready) { /* Added ability to handle non-standard "same as above" column name */ - if (addmpscolumn(lp, Int_section, typeMPS, &Column_ready, &count, Last_column, Last_columnno, Last_col_name)) { - strcpy(Last_col_name, field2); - NZ = 0; - } - else - break; - } - } - } - if(items == 5) { /* there might be an INTEND or INTORG marker */ - /* look for " 'MARKER' 'INTORG'" - or " 'MARKER' 'INTEND'" */ - if(strcmp(field3, "'MARKER'") != 0) - break; - if(strcmp(field5, "'INTORG'") == 0) { - Int_section = TRUE; - report(lp, FULL, "Switching to integer section\n"); - } - else if(strcmp(field5, "'INTEND'") == 0) { - Int_section = FALSE; - report(lp, FULL, "Switching to non-integer section\n"); - } - else - report(lp, IMPORTANT, "Unknown marker (ignored) at line %d: %s\n", - Lineno, field5); - } - else if((row = find_row(lp, field3, Unconstrained_rows_found)) >= 0) { - if(row > lp->rows) - report(lp, CRITICAL, "Invalid row %s encountered in the MPS file\n", field3); - Last_columnno[count] = row; - Last_column[count] = (REAL)field4; - if(appendmpsitem(&count, Last_columnno, Last_column)) { - NZ++; - Column_ready = TRUE; - } - } - } - if(items == 6) { - if((row = find_row(lp, field5, Unconstrained_rows_found)) >= 0) { - if(row > lp->rows) - report(lp, CRITICAL, "Invalid row %s encountered in the MPS file\n", field5); - Last_columnno[count] = row; - Last_column[count] = (REAL)field6; - if(appendmpsitem(&count, Last_columnno, Last_column)) { - NZ++; - Column_ready = TRUE; - } - } - } - - if((items < 4) || (items > 6)) { /* Wrong! */ - report(lp, CRITICAL, "Wrong number of items (%d) in COLUMNS section (line %d)\n", - items, Lineno); - break; - } - - continue; - - /* Process entries in the RHS section */ - /* field2: uninteresting name; field3: constraint name */ - /* field4: value */ - /* optional: field5: constraint name; field6: value */ - case MPSRHS: - - report(lp, FULL, "RHS line: %s %s %g %s %g\n", - field2, field3, field4, field5, field6); - - if((items != 4) && (items != 6)) { - report(lp, CRITICAL, "Wrong number of items (%d) in RHS section line %d\n", - items, Lineno); - break; - } - - if((row = find_row(lp, field3, Unconstrained_rows_found)) >= 0) { - if ((row == 0) && ((typeMPS & MPSNEGOBJCONST) == MPSNEGOBJCONST)) - field4 = -field4; - set_rh(lp, row, (REAL)field4); - } - - if(items == 6) { - if((row = find_row(lp, field5, Unconstrained_rows_found)) >= 0) { - if ((row == 0) && ((typeMPS & MPSNEGOBJCONST) == MPSNEGOBJCONST)) - field6 = -field6; - set_rh(lp, row, (REAL)field6); - } - } - - continue; - - /* Process entries in the BOUNDS section */ - /* field1: bound type; field2: uninteresting name; */ - /* field3: variable name; field4: value */ - case MPSBOUNDS: - - report(lp, FULL, "BOUNDS line: %s %s %s %g\n", - field1, field2, field3, field4); - - var = find_var(lp, field3, FALSE); - if(var < 0){ /* bound on undefined var in COLUMNS section ... */ - Column_ready = TRUE; - if (!addmpscolumn(lp, FALSE, typeMPS, &Column_ready, &count, Last_column, Last_columnno, field3)) - break; - Column_ready = TRUE; - var = find_var(lp, field3, TRUE); - } - if(var < 0) /* undefined var and could add ... */; - else if(strcmp(field1, "UP") == 0) { - /* upper bound */ - /* if(!set_bounds(lp, var, get_lowbo(lp, var), field4)) */ - if(!set_upbo(lp, var, field4)) - break; - } - else if(strcmp(field1, "SC") == 0) { - /* upper bound */ - if(field4 == 0) - field4 = lp->infinite; - /* if(!set_bounds(lp, var, get_lowbo(lp, var), field4)) */ - if(!set_upbo(lp, var, field4)) - break; - set_semicont(lp, var, TRUE); - } - else if(strcmp(field1, "SI") == 0) { - /* upper bound */ - if(field4 == 0) - field4 = lp->infinite; - /* if(!set_bounds(lp, var, get_lowbo(lp, var), field4)) */ - if(!set_upbo(lp, var, field4)) - break; - set_int(lp, var, TRUE); - set_semicont(lp, var, TRUE); - } - else if(strcmp(field1, "LO") == 0) { - /* lower bound */ - /* if(!set_bounds(lp, var, field4, get_upbo(lp, var))) */ - if(!set_lowbo(lp, var, field4)) - break; - } - else if(strcmp(field1, "PL") == 0) { /* plus-ranged variable */ - /* if(!set_bounds(lp, var, get_lowbo(lp, var), lp->infinite)) */ - if(!set_upbo(lp, var, lp->infinite)) - break; - } - else if(strcmp(field1, "MI") == 0) { /* minus-ranged variable */ - /* if(!set_bounds(lp, var, -lp->infinite, get_upbo(lp, var))) */ - if(!set_lowbo(lp, var, -lp->infinite)) - break; - } - else if(strcmp(field1, "FR") == 0) { /* free variable */ - set_unbounded(lp, var); - } - else if(strcmp(field1, "FX") == 0) { - /* fixed, upper _and_ lower */ - if(!set_bounds(lp, var, field4, field4)) - break; - } - else if(strcmp(field1, "BV") == 0) { /* binary variable */ - set_binary(lp, var, TRUE); - } - /* AMPL bounds type UI and LI added by E.Imamura (CRIEPI) */ - else if(strcmp(field1, "UI") == 0) { /* upper bound for integer variable */ - /* if(!set_bounds(lp, var, get_lowbo(lp, var), field4)) */ - if(!set_upbo(lp, var, field4)) - break; - set_int(lp, var, TRUE); - } - else if(strcmp(field1, "LI") == 0) { /* lower bound for integer variable - corrected by KE */ - /* if(!set_bounds(lp, var, field4, get_upbo(lp, var))) */ - if(!set_lowbo(lp, var, field4)) - break; - set_int(lp, var, TRUE); - } - else { - report(lp, CRITICAL, "BOUND type %s on line %d is not supported", - field1, Lineno); - break; - } - - continue; - - /* Process entries in the BOUNDS section */ - - /* We have to implement the following semantics: - - D. The RANGES section is for constraints of the form: h <= - constraint <= u . The range of the constraint is r = u - h . The - value of r is specified in the RANGES section, and the value of u or - h is specified in the RHS section. If b is the value entered in the - RHS section, and r is the value entered in the RANGES section, then - u and h are thus defined: - - row type sign of r h u - ---------------------------------------------- - G + or - b b + |r| - L + or - b - |r| b - E + b b + |r| - E - b - |r| b */ - - /* field2: uninteresting name; field3: constraint name */ - /* field4: value */ - /* optional: field5: constraint name; field6: value */ - - case MPSRANGES: - - report(lp, FULL, "RANGES line: %s %s %g %s %g", - field2, field3, field4, field5, field6); - - if((items != 4) && (items != 6)) { - report(lp, CRITICAL, "Wrong number of items (%d) in RANGES section line %d", - items, Lineno); - break; - } - - if((row = find_row(lp, field3, Unconstrained_rows_found)) >= 0) { - /* Determine constraint type */ - - if(fabs(field4) >= lp->infinite) { - report(lp, IMPORTANT, - "Warning, Range for row %s >= infinity (value %g) on line %d, ignored", - field3, field4, Lineno); - } - else if(field4 == 0) { - /* Change of a GE or LE to EQ */ - if(lp->orig_upbo[row] != 0) - set_constr_type(lp, row, EQ); - } - else if(is_chsign(lp, row)) { - /* GE */ - lp->orig_upbo[row] = fabs(field4); - } - else if((lp->orig_upbo[row] == 0) && (field4 >= 0)) { - /* EQ with positive sign of r value */ - set_constr_type(lp, row, GE); - lp->orig_upbo[row] = field4; - } - else if(lp->orig_upbo[row] == lp->infinite) { - /* LE */ - lp->orig_upbo[row] = fabs(field4); - } - else if((lp->orig_upbo[row] == 0) && (field4 < 0)) { - /* EQ with negative sign of r value */ - set_constr_type(lp, row, LE); - lp->orig_upbo[row] = my_flipsign(field4); - } - else { /* let's be paranoid */ - report(lp, IMPORTANT, - "Cannot figure out row type, row = %d, is_chsign = %d, upbo = %g on line %d", - row, is_chsign(lp, row), (double)lp->orig_upbo[row], Lineno); - } - } - - if(items == 6) { - if((row = find_row(lp, field5, Unconstrained_rows_found)) >= 0) { - /* Determine constraint type */ - - if(fabs(field6) >= lp->infinite) { - report(lp, IMPORTANT, - "Warning, Range for row %s >= infinity (value %g) on line %d, ignored", - field5, field6, Lineno); - } - else if(field6 == 0) { - /* Change of a GE or LE to EQ */ - if(lp->orig_upbo[row] != 0) - set_constr_type(lp, row, EQ); - } - else if(is_chsign(lp, row)) { - /* GE */ - lp->orig_upbo[row] = fabs(field6); - } - else if(lp->orig_upbo[row] == 0 && field6 >= 0) { - /* EQ with positive sign of r value */ - set_constr_type(lp, row, GE); - lp->orig_upbo[row] = field6; - } - else if(lp->orig_upbo[row] == lp->infinite) { - /* LE */ - lp->orig_upbo[row] = fabs(field6); - } - else if((lp->orig_upbo[row] == 0) && (field6 < 0)) { - /* EQ with negative sign of r value */ - set_constr_type(lp, row, LE); - lp->orig_upbo[row] = my_flipsign(field6); - } - else { /* let's be paranoid */ - report(lp, IMPORTANT, - "Cannot figure out row type, row = %d, is_chsign = %d, upbo = %g on line %d", - row, is_chsign(lp,row), (double) lp->orig_upbo[row], Lineno); - } - } - } - - continue; - - /* Process entries in the SOS section */ - - /* We have to implement the following semantics: - - E. The SOS section is for ordered variable sets of the form: - x1, x2, x3 ... xn where only a given number of consequtive variables - may be non-zero. Each set definition is prefaced by type, name - and priority data. Each set member has an optional weight that - determines its order. There are two forms supported; a full format - and a reduced CPLEX-like format. */ - - case MPSSOS: - report(lp, FULL, "SOS line: %s %s %g %s %g", - field2, field3, field4, field5, field6); - - if((items == 0) || (items > 4)) { - report(lp, IMPORTANT, - "Invalid number of items (%d) in SOS section line %d\n", - items, Lineno); - break; - } - - if(strlen(field1) == 0) items--; /* fix scanline anomoly! */ - - /* Check if this is the start of a new SOS */ - if(items == 1 || items == 4) { - row = (int) (field1[1] - '0'); - if((row <= 0) || (row > 9)) { - report(lp, IMPORTANT, - "Error: Invalid SOS type %s line %d\n", field1, Lineno); - break; - } - field1[0] = '\0'; /* fix scanline anomoly! */ - - /* lp_solve needs a name for the SOS */ - if(variant == 0) { - if(strlen(field3) == 0) /* CPLEX format does not provide a SOS name; create one */ - sprintf(field3, "SOS_%d", SOS_count(lp) + 1); - } - else { /* Remap XPRESS format name */ - strcpy(field3, field1); - } - /* Obtain the SOS priority */ - if(items == 4) - SOS = (int) field4; - else - SOS = 1; - - /* Define a new SOS instance */ - - SOS = add_SOS(lp, field3, (int) row, SOS, 0, NULL, NULL); - } - /* Otherwise, add set members to the active SOS */ - else { - char *field = (items == 3) ? field3 /* Native lp_solve and XPRESS formats */ : field2 /* CPLEX format */; - - var = find_var(lp, field, FALSE); /* Native lp_solve and XPRESS formats */ - if(var < 0){ /* SOS on undefined var in COLUMNS section ... */ - Column_ready = TRUE; - if (!addmpscolumn(lp, FALSE, typeMPS, &Column_ready, &count, Last_column, Last_columnno, field)) - break; - Column_ready = TRUE; - var = find_var(lp, field, TRUE); - } - if((var < 0) || (SOS < 1)) /* undefined var and could add ... */; - else append_SOSrec(lp->SOS->sos_list[SOS-1], 1, &var, &field4); - } - - continue; - } - - /* If we got here there was an error "upstream" */ - report(lp, IMPORTANT, - "Error: Cannot handle line %d\n", Lineno); - break; - } - } - - if((*OBJNAME) && (!OF_found)) { - report(lp, IMPORTANT, - "Error: Objective function specified by OBJNAME card not found\n"); - CompleteStatus = FALSE; - } - - if(CompleteStatus == FALSE) { - if (*newlp == NULL) - delete_lp(lp); - } - else { - if (typeMPS & MPSIBM) { - REAL lower, upper; - - for (var = 1; var <= lp->columns; var++) - if (is_int(lp, var)) { - lower = get_lowbo(lp, var); - upper = get_upbo(lp, var); - if ((lower == 10.0 / DEF_INFINITE) && (upper == DEF_INFINITE / 10.0)) - upper = 1.0; - if (lower == 10.0 / DEF_INFINITE) - lower = 0.0; - if (upper == DEF_INFINITE / 10.0) - upper = lp->infinite; - set_bounds(lp, var, lower, upper); - } - } - *newlp = lp; - } - if(Last_column != NULL) - FREE(Last_column); - if(Last_columnno != NULL) - FREE(Last_columnno); - } - - return( CompleteStatus ); -} - -static void number(char *str,REAL value) - { - char __str[80], *_str; - int i; - - /* sprintf(_str,"%12.6G",value); */ - _str=__str+2; - if (value>=0.0) - if ((value!=0.0) && ((value>0.99999999e12) || (value<0.0001))) { - int n=15; - - do { - n--; - i=sprintf(_str,"%*.*E",n,n-6,(double) value); - if (i>12) { - char *ptr=strchr(_str,'E'); - - if (ptr!=NULL) { - if (*(++ptr)=='-') ptr++; - while ((i>12) && ((*ptr=='+') || (*ptr=='0'))) { - strcpy(ptr,ptr+1); - i--; - } - } - } - } while (i>12); - } - else if (value>=1.0e10) { - int n=13; - - do { - i=sprintf(_str,"%*.0f",--n,(double) value); - } while (i>12); - } - else { - if (((i=sprintf(_str,"%12.10f",(double) value))>12) && (_str[12]>='5')) { - for (i=11;i>=0;i--) - if (_str[i]!='.') { - if (++_str[i]>'9') _str[i]='0'; - else break; - } - if (i<0) { - *(--_str)='1'; - *(--_str)=' '; - } - } - } - else - if ((value<-0.99999999e11) || (value>-0.0001)) { - int n=15; - - do { - n--; - i=sprintf(_str,"%*.*E",n,n-7,(double) value); - if (i>12) { - char *ptr=strchr(_str,'E'); - - if (ptr!=NULL) { - if (*(++ptr)=='-') ptr++; - while ((i>12) && ((*ptr=='+') || (*ptr=='0'))) { - strcpy(ptr,ptr+1); - i--; - } - } - } - } while (i>12); - } - else if (value<=-1.0e9) { - int n=13; - - do { - i=sprintf(_str,"%*.0f",--n,(double) value); - } while (i>12); - } - else - if (((i=sprintf(_str,"%12.9f",(double) value))>12) && (_str[12]>='5')) { - for (i=11;i>=1;i--) - if (_str[i]!='.') { - if (++_str[i]>'9') _str[i]='0'; - else break; - } - if (i<1) { - *_str='1'; - *(--_str)='-'; - *(--_str)=' '; - } - } - strncpy(str,_str,12); - } - -static char *formatnumber12(char *numberbuffer, double a) -{ -#if 0 - return(sprintf(numberbuffer, "%12g", a)); -#else - number(numberbuffer, a); - return(numberbuffer); -#endif -} - -STATIC char *MPSnameFIXED(char *name0, char *name) -{ - sprintf(name0, "%-8.8s", name); - return(name0); -} - -STATIC char *MPSnameFREE(char *name0, char *name) -{ - if(strlen(name) < 8) - return(MPSnameFIXED(name0, name)); - else - return(name); -} - -static void write_data(void *userhandle, write_modeldata_func write_modeldata, char *format, ...) -{ - char buff[DEF_STRBUFSIZE+1]; - va_list ap; - - va_start(ap, format); - vsnprintf(buff, DEF_STRBUFSIZE, format, ap); - va_end(ap); - write_modeldata(userhandle, buff); -} - -MYBOOL __WINAPI MPS_writefileex(lprec *lp, int typeMPS, void *userhandle, write_modeldata_func write_modeldata) -{ - int i, j, jj, je, k, marker, putheader, ChangeSignObj = FALSE, *idx, *idx1; - MYBOOL ok = TRUE, names_used; - REAL a, *val, *val1; - char * (*MPSname)(char *name0, char *name); - char numberbuffer[15]; - char name0[9]; - - if((typeMPS & MPSFIXED) == MPSFIXED) { - MPSname = MPSnameFIXED; - ChangeSignObj = is_maxim(lp); - } - else if((typeMPS & MPSFREE) == MPSFREE) { - MPSname = MPSnameFREE; - } - else { - report(lp, IMPORTANT, "MPS_writefile: unrecognized MPS name type.\n"); - return(FALSE); - } - - names_used = lp->names_used; - - if((typeMPS & MPSFIXED) == MPSFIXED) { - /* Check if there is no variable name where the first 8 charachters are equal to the first 8 characters of anothe variable */ - if(names_used) - for(i = 1; (i <= lp->columns) && (ok); i++) - if((lp->col_name[i] != NULL) && (lp->col_name[i]->name != NULL) && (!is_splitvar(lp, i)) && (strlen(lp->col_name[i]->name) > 8)) - for(j = 1; (j < i) && (ok); j++) - if((lp->col_name[j] != NULL) && (lp->col_name[j]->name != NULL) && (!is_splitvar(lp, j))) - if(strncmp(lp->col_name[i]->name, lp->col_name[j]->name, 8) == 0) - ok = FALSE; - } - - if(!ok) { - lp->names_used = FALSE; - ok = TRUE; - } - - memset(numberbuffer, 0, sizeof(numberbuffer)); - - marker = 0; - - /* First write metadata in structured comment form (lp_solve style) */ - write_data(userhandle, write_modeldata, "*\n", - (int) MAJORVERSION, (int) MINORVERSION); - write_data(userhandle, write_modeldata, "*\n", lp->rows); - write_data(userhandle, write_modeldata, "*\n", lp->columns); - write_data(userhandle, write_modeldata, "*\n", lp->equalities); - if(SOS_count(lp) > 0) - write_data(userhandle, write_modeldata, "*\n", SOS_count(lp)); - write_data(userhandle, write_modeldata, "*\n", lp->int_vars); - if(lp->sc_vars > 0) - write_data(userhandle, write_modeldata, "*\n", lp->sc_vars); - write_data(userhandle, write_modeldata, "*\n", (is_maxim(lp) ? "MAX" : "MIN")); - write_data(userhandle, write_modeldata, "*\n"); - - /* Write the MPS content */ - write_data(userhandle, write_modeldata, "NAME %s\n", MPSname(name0, get_lp_name(lp))); - if(((typeMPS & MPSFREE) == MPSFREE) && (is_maxim(lp))) - write_data(userhandle, write_modeldata, "OBJSENSE\n MAX\n"); - write_data(userhandle, write_modeldata, "ROWS\n"); - for(i = 0; i <= lp->rows; i++) { - if(i == 0) - write_data(userhandle, write_modeldata, " N "); - else if(lp->orig_upbo[i] != 0) { - if(is_chsign(lp,i)) - write_data(userhandle, write_modeldata, " G "); - else - write_data(userhandle, write_modeldata, " L "); - } - else - write_data(userhandle, write_modeldata, " E "); - write_data(userhandle, write_modeldata, "%s\n", MPSname(name0, get_row_name(lp, i))); - } - - allocREAL(lp, &val, 1 + lp->rows, TRUE); - allocINT(lp, &idx, 1 + lp->rows, TRUE); - write_data(userhandle, write_modeldata, "COLUMNS\n"); - for(i = 1; i <= lp->columns; i++) { - if(!is_splitvar(lp, i)) { - if(is_int(lp,i) && (marker % 2) == 0) { - write_data(userhandle, write_modeldata, " MARK%04d 'MARKER' 'INTORG'\n", - marker); - marker++; - } - if(!is_int(lp,i) && (marker % 2) == 1) { - write_data(userhandle, write_modeldata, " MARK%04d 'MARKER' 'INTEND'\n", - marker); - marker++; - } - - /* Loop over non-zero column entries */ - je = get_columnex(lp, i, val, idx); - for(k = 1, val1 = val, idx1 = idx, jj = 0; jj < je; jj++) { - k = 1 - k; - j = *(idx1++); - a = *(val1++); - if (k == 0) { - write_data(userhandle, write_modeldata, " %s", - MPSname(name0, get_col_name(lp, i))); - write_data(userhandle, write_modeldata, " %s %s", - MPSname(name0, get_row_name(lp, j)), -/* formatnumber12(numberbuffer, (double) a)); */ - formatnumber12(numberbuffer, (double) (a * (j == 0 && ChangeSignObj ? -1 : 1)))); - } - else - write_data(userhandle, write_modeldata, " %s %s\n", - MPSname(name0, get_row_name(lp, j)), - formatnumber12(numberbuffer, (double) (a * (j == 0 && ChangeSignObj ? -1 : 1)))); -/* formatnumber12(numberbuffer, (double) a)); */ - } - if(k == 0) - write_data(userhandle, write_modeldata, "\n"); - } - } - if((marker % 2) == 1) { - write_data(userhandle, write_modeldata, " MARK%04d 'MARKER' 'INTEND'\n", - marker); - /* marker++; */ /* marker not used after this */ - } - FREE(idx); - FREE(val); - - write_data(userhandle, write_modeldata, "RHS\n"); - for(k = 1, i = 0; i <= lp->rows; i++) { - a = lp->orig_rhs[i]; - if(a) { - a = unscaled_value(lp, a, i); - if ((i == 0) && ((typeMPS & MPSNEGOBJCONST) == MPSNEGOBJCONST)) - a = -a; - if((i == 0) || is_chsign(lp, i)) - a = my_flipsign(a); - k = 1 - k; - if(k == 0) - write_data(userhandle, write_modeldata, " RHS %s %s", - MPSname(name0, get_row_name(lp, i)), - formatnumber12(numberbuffer, (double)a)); - else - write_data(userhandle, write_modeldata, " %s %s\n", - MPSname(name0, get_row_name(lp, i)), - formatnumber12(numberbuffer, (double)a)); - } - } - if(k == 0) - write_data(userhandle, write_modeldata, "\n"); - - putheader = TRUE; - for(k = 1, i = 1; i <= lp->rows; i++){ - a = 0; - if((lp->orig_upbo[i] < lp->infinite) && (lp->orig_upbo[i] != 0.0)) - a = lp->orig_upbo[i]; - if(a) { - if(putheader) { - write_data(userhandle, write_modeldata, "RANGES\n"); - putheader = FALSE; - } - a = unscaled_value(lp, a, i); - k = 1 - k; - if(k == 0) - write_data(userhandle, write_modeldata, " RGS %s %s", - MPSname(name0, get_row_name(lp, i)), - formatnumber12(numberbuffer, (double)a)); - else - write_data(userhandle, write_modeldata, " %s %s\n", - MPSname(name0, get_row_name(lp, i)), - formatnumber12(numberbuffer, (double)a)); - } - } - if(k == 0) - write_data(userhandle, write_modeldata, "\n"); - - putheader = TRUE; - for(i = lp->rows + 1; i <= lp->sum; i++) - if(!is_splitvar(lp, i - lp->rows)) { - j = i - lp->rows; - if((lp->orig_lowbo[i] != 0) && (lp->orig_upbo[i] < lp->infinite) && - (lp->orig_lowbo[i] == lp->orig_upbo[i])) { - a = lp->orig_upbo[i]; - a = unscaled_value(lp, a, i); - if(putheader) { - write_data(userhandle, write_modeldata, "BOUNDS\n"); - putheader = FALSE; - } - write_data(userhandle, write_modeldata, " FX BND %s %s\n", - MPSname(name0, get_col_name(lp, j)), - formatnumber12(numberbuffer, (double)a)); - } - else if(is_binary(lp, j)) { - if(putheader) { - write_data(userhandle, write_modeldata, "BOUNDS\n"); - putheader = FALSE; - } - write_data(userhandle, write_modeldata, " BV BND %s\n", - MPSname(name0, get_col_name(lp, j))); - } - else if(is_unbounded(lp, j)) { - if(putheader) { - write_data(userhandle, write_modeldata, "BOUNDS\n"); - putheader = FALSE; - } - write_data(userhandle, write_modeldata, " FR BND %s\n", - MPSname(name0, get_col_name(lp, j))); - } - else { - if((lp->orig_lowbo[i] != 0) || (is_int(lp, j))) { /* Some solvers like CPLEX need to have a bound on a variable if it is integer, but not binary else it is interpreted as binary which is not ment */ - a = lp->orig_lowbo[i]; - a = unscaled_value(lp, a, i); - if(putheader) { - write_data(userhandle, write_modeldata, "BOUNDS\n"); - putheader = FALSE; - } - if(lp->orig_lowbo[i] != -lp->infinite) - write_data(userhandle, write_modeldata, " LO BND %s %s\n", - MPSname(name0, get_col_name(lp, j)), - formatnumber12(numberbuffer, (double)a)); - else - write_data(userhandle, write_modeldata, " MI BND %s\n", - MPSname(name0, get_col_name(lp, j))); - } - - if((lp->orig_upbo[i] < lp->infinite) || (is_semicont(lp, j))) { - a = lp->orig_upbo[i]; - if(a < lp->infinite) - a = unscaled_value(lp, a, i); - if(putheader) { - write_data(userhandle, write_modeldata, "BOUNDS\n"); - putheader = FALSE; - } - if(is_semicont(lp, j)) { - if(is_int(lp, j)) - write_data(userhandle, write_modeldata, " SI BND %s %s\n", - MPSname(name0, get_col_name(lp, j)), - (a < lp->infinite) ? formatnumber12(numberbuffer, (double)a) : " "); - else - write_data(userhandle, write_modeldata, " SC BND %s %s\n", - MPSname(name0, get_col_name(lp, j)), - (a < lp->infinite) ? formatnumber12(numberbuffer, (double)a) : " "); - } - else - write_data(userhandle, write_modeldata, " UP BND %s %s\n", - MPSname(name0, get_col_name(lp, j)), - formatnumber12(numberbuffer, (double)a)); - } - } - } - - /* Write optional SOS section */ - putheader = TRUE; - for(i = 0; i < SOS_count(lp); i++) { - SOSgroup *SOS = lp->SOS; - - if(putheader) { - write_data(userhandle, write_modeldata, "SOS\n"); - putheader = FALSE; - } - write_data(userhandle, write_modeldata, " S%1d SOS %s %s\n", - SOS->sos_list[i]->type, - MPSname(name0, SOS->sos_list[i]->name), - formatnumber12(numberbuffer, (double) SOS->sos_list[i]->priority)); - for(j = 1; j <= SOS->sos_list[i]->size; j++) { - write_data(userhandle, write_modeldata, " SOS %s %s\n", - MPSname(name0, get_col_name(lp, SOS->sos_list[i]->members[j])), - formatnumber12(numberbuffer, (double) SOS->sos_list[i]->weights[j])); - } - } - - write_data(userhandle, write_modeldata, "ENDATA\n"); - - lp->names_used = names_used; - - return(ok); -} - -static int __WINAPI write_lpdata(void *userhandle, char *buf) -{ - fputs(buf, (FILE *) userhandle); - return(TRUE); -} - -MYBOOL MPS_writefile(lprec *lp, int typeMPS, char *filename) -{ - FILE *output = stdout; - MYBOOL ok; - - if (filename != NULL) { - ok = ((output = fopen(filename, "w")) != NULL); - if(!ok) - return(ok); - } - else - output = lp->outstream; - - ok = MPS_writefileex(lp, typeMPS, (void *) output, write_lpdata); - - if (filename != NULL) - fclose(output); - - return(ok); -} - -MYBOOL MPS_writehandle(lprec *lp, int typeMPS, FILE *output) -{ - MYBOOL ok; - - if (output != NULL) - set_outputstream(lp, output); - - output = lp->outstream; - - ok = MPS_writefileex(lp, typeMPS, (void *) output, write_lpdata); - - return(ok); -} - - -/* Read and write BAS files */ -/* #define OldNameMatch */ -#ifdef OldNameMatch -static int MPS_getnameidx(lprec *lp, char *varname, MYBOOL isrow) -{ - int in = -1; - - in = get_nameindex(lp, varname, isrow); - if((in < 0) && (strncmp(varname, (isrow ? ROWNAMEMASK : COLNAMEMASK), 1) == 0)) { - if(sscanf(varname + 1, "%d", &in) != 1) - in = -1; - } - return( in ); -} -#else -static int MPS_getnameidx(lprec *lp, char *varname, MYBOOL tryrowfirst) -{ - int in = -1; - - /* Have we defined our own variable names? */ - if(lp->names_used) { - /* First check the primary name list */ - in = get_nameindex(lp, varname, tryrowfirst); - if((in > 0) && !tryrowfirst) - in += lp->rows; - /* If we were unsuccessful, try the secondary name list */ - else if(in < 0) { - in = get_nameindex(lp, varname, (MYBOOL) !tryrowfirst); - if((in > 0) && tryrowfirst) - in += lp->rows; - } - } - /* If not, see if we can match the standard name mask */ - - if(in == -1) { - if(strncmp(varname, (tryrowfirst ? ROWNAMEMASK : COLNAMEMASK), 1) == 0) { - /* Fail if we did not successfully scan as a valid integer */ - if((sscanf(varname + 1, "%d", &in) != 1) || - (in < (tryrowfirst ? 0 : 1)) || (in > (tryrowfirst ? lp->rows : lp->columns))) - in = -1; - } - else if(strncmp(varname, (!tryrowfirst ? ROWNAMEMASK : COLNAMEMASK), 1) == 0) { - /* Fail if we did not successfully scan as a valid integer */ - if((sscanf(varname + 1, "%d", &in) != 1) || - (in < (tryrowfirst ? 0 : 1)) || (in > (tryrowfirst ? lp->rows : lp->columns))) - in = -1; - } - } - return( in ); -} -#endif - -MYBOOL MPS_readBAS(lprec *lp, int typeMPS, char *filename, char *info) -{ - char field1[BUFSIZ], field2[BUFSIZ], field3[BUFSIZ], field5[BUFSIZ], - line[BUFSIZ], tmp[BUFSIZ], *ptr; - double field4, field6; - int ib, in, items, Lineno = 0; - MYBOOL ok; - FILE *input = stdin; - int (*scan_line)(lprec *lp, int section, char* line, char *field1, char *field2, char *field3, - double *field4, char *field5, double *field6); - - if((typeMPS & MPSFIXED) == MPSFIXED) - scan_line = scan_lineFIXED; - else if((typeMPS & MPSFREE) == MPSFREE) - scan_line = scan_lineFREE; - else { - report(lp, IMPORTANT, "MPS_readBAS: unrecognized MPS line type.\n"); - return(FALSE); - } - - ok = (MYBOOL) ((filename != NULL) && ((input = fopen(filename,"r")) != NULL)); - if(!ok) - return(ok); - default_basis(lp); - - /* Let's initialize line to all zero's */ - MEMCLEAR(line, BUFSIZ); - ok = FALSE; - while(fgets(line, BUFSIZ - 1, input)) { - Lineno++; - - for(ptr = line; (*ptr) && (isspace((unsigned char) *ptr)); ptr++); - - /* skip lines which start with "*", they are comment */ - if((line[0] == '*') || (*ptr == 0) || (*ptr == '\n') || (*ptr == '\r')) { - report(lp, FULL, "Comment on line %d: %s", Lineno, line); - continue; - } - - report(lp, FULL, "Line %6d: %s", Lineno, line); - - /* first check for "special" lines: in our case only NAME and ENDATA, - ...this must start in the first position of line */ - if(line[0] != ' ') { - sscanf(line, "%s", tmp); - if(strcmp(tmp, "NAME") == 0) { - if(info != NULL) { - *info = 0; - for(ptr = line + 4; (*ptr) && (isspace((unsigned char) *ptr)); ptr++); - in = (int) strlen(ptr); - while ((in > 0) && ((ptr[in - 1] == '\r') || (ptr[in - 1] == '\n') || isspace(ptr[in - 1]))) - in--; - ptr[in] = 0; - strcpy(info, ptr); - } - } - else if(strcmp(tmp, "ENDATA") == 0) { - report(lp, FULL, "Finished reading BAS file\n"); - ok = TRUE; - break; - } - else { /* line does not start with space and does not match above */ - report(lp, IMPORTANT, "Unrecognized BAS line %d: %s\n", Lineno, line); - break; - } - } - else { /* normal line, process */ - items = scan_line(lp, /* MPSRHS */ MPSBOUNDS, line, field1, field2, field3, &field4, field5, &field6); - if(items < 0){ - report(lp, IMPORTANT, "Syntax error on line %d: %s\n", Lineno, line); - break; - } - /* find first variable index value */ - in = MPS_getnameidx(lp, field2, FALSE); -#ifdef OldNameMatch - if(in < 0) - in = MPS_getnameidx(lp, field2, TRUE); - else - in += lp->rows; -#endif - if(in < 0) - break; - - /* check if we have the basic/non-basic variable format */ - if(field1[0] == 'X') { - /* find second variable index value */ - ib = in; - in = MPS_getnameidx(lp, field3, FALSE); -#ifdef OldNameMatch - if(in < 0) - in = MPS_getnameidx(lp, field3, TRUE); - else - in += lp->rows; -#endif - if(in < 0) - break; - - lp->is_lower[in] = (MYBOOL) (field1[1] == 'L'); - lp->is_basic[ib] = TRUE; - } - else - lp->is_lower[in] = (MYBOOL) (field1[0] == 'L'); - - lp->is_basic[in] = FALSE; - - } - } - /* Update the basis index-to-variable array */ - ib = 0; - items = lp->sum; - for(in = 1; in <= items; in++) - if(lp->is_basic[in]) { - ib++; - lp->var_basic[ib] = in; - } - - fclose(input); - return( ok ); -} - -MYBOOL MPS_writeBAS(lprec *lp, int typeMPS, char *filename) -{ - int ib, in; - MYBOOL ok; - char name1[100], name2[100]; - FILE *output = stdout; - char * (*MPSname)(char *name0, char *name); - char name0[9]; - - /* Set name formatter */ - if((typeMPS & MPSFIXED) == MPSFIXED) - MPSname = MPSnameFIXED; - else if((typeMPS & MPSFREE) == MPSFREE) - MPSname = MPSnameFREE; - else { - report(lp, IMPORTANT, "MPS_writeBAS: unrecognized MPS name type.\n"); - return(FALSE); - } - - /* Open the file for writing */ - ok = (MYBOOL) ((filename == NULL) || ((output = fopen(filename,"w")) != NULL)); - if(!ok) - return(ok); - if(filename == NULL && lp->outstream != NULL) - output = lp->outstream; - - fprintf(output, "NAME %s Rows %d Cols %d Iters %.0f\n", - get_lp_name(lp), lp->rows, lp->columns, (double) get_total_iter(lp)); - - ib = lp->rows; - in = 0; - while ((ib < lp->sum) || (in < lp->sum)) { - - /* Find next basic variable (skip slacks) */ - ib++; - while((ib <= lp->sum) && !lp->is_basic[ib]) - ib++; - - /* Find next non-basic variable (skip lower-bounded structural variables) */ - in++; - while((in <= lp->sum) && (lp->is_basic[in] || - ((in > lp->rows) && lp->is_lower[in]))) - in++; - - /* Check if we have a basic/non-basic variable pair */ - if((ib <= lp->sum) && (in <= lp->sum)) { - strcpy(name1, MPSname(name0, (ib <= lp->rows ? get_row_name(lp, ib) : - get_col_name(lp, ib-lp->rows)))); - strcpy(name2, MPSname(name0, (in <= lp->rows ? get_row_name(lp, in) : - get_col_name(lp, in-lp->rows)))); - fprintf(output, " %2s %s %s\n", (lp->is_lower[in] ? "XL" : "XU"), name1, name2); - } - - /* Otherwise just write the bound state of the non-basic variable */ - else if(in <= lp->sum) { - strcpy(name1, MPSname(name0, (in <= lp->rows ? get_row_name(lp, in) : - get_col_name(lp, in-lp->rows)))); - fprintf(output, " %2s %s\n", (lp->is_lower[in] ? "LL" : "UL"), name1); - } - - } - fprintf(output, "ENDATA\n"); - - if(filename != NULL) - fclose(output); - return( ok ); -} diff --git a/code/3rd_lpsolve/lp_MPS.h b/code/3rd_lpsolve/lp_MPS.h deleted file mode 100644 index 0d970269..00000000 --- a/code/3rd_lpsolve/lp_MPS.h +++ /dev/null @@ -1,33 +0,0 @@ -#ifndef HEADER_lp_MPS -#define HEADER_lp_MPS - -#include "lp_types.h" - -/* For MPS file reading and writing */ -#define ROWNAMEMASK "R%d" -#define ROWNAMEMASK2 "r%d" -#define COLNAMEMASK "C%d" -#define COLNAMEMASK2 "c%d" - -#ifdef __cplusplus -extern "C" { -#endif - -/* Read an MPS file */ -MYBOOL MPS_readfile(lprec **newlp, char *filename, int typeMPS, int verbose); -MYBOOL __WINAPI MPS_readhandle(lprec **newlp, FILE *filehandle, int typeMPS, int verbose); - -/* Write a MPS file to output */ -MYBOOL MPS_writefile(lprec *lp, int typeMPS, char *filename); -MYBOOL MPS_writehandle(lprec *lp, int typeMPS, FILE *output); - -/* Read and write BAS files */ -MYBOOL MPS_readBAS(lprec *lp, int typeMPS, char *filename, char *info); -MYBOOL MPS_writeBAS(lprec *lp, int typeMPS, char *filename); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_MPS */ - diff --git a/code/3rd_lpsolve/lp_SOS.c b/code/3rd_lpsolve/lp_SOS.c deleted file mode 100644 index 6155b51f..00000000 --- a/code/3rd_lpsolve/lp_SOS.c +++ /dev/null @@ -1,1575 +0,0 @@ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_report.h" -#include "lp_SOS.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* - Specially Ordered Set (SOS) routines - w/interface for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h - - Release notes: - v1.0 1 September 2003 Complete package for SOS creation and use in a LP - setting. Notable feature of this implementation - compared to those in other commercial systems is - the generalization to SOS'es of "unlimited" order. - v1.1 8 December 2003 Added variable (index) deletion method. - v1.2 17 December 2004 Added bound change tracking functionality. - v1.3 18 September 2005 Added sparse SOS handling to speed up processing - of large number of SOS'es. - - ---------------------------------------------------------------------------------- -*/ - -/* SOS group functions */ -STATIC SOSgroup *create_SOSgroup(lprec *lp) -{ - SOSgroup *group; - - group = (SOSgroup *) calloc(1, sizeof(*group)); - group->lp = lp; - group->sos_alloc = SOS_START_SIZE; - group->sos_list = (SOSrec **) malloc((group->sos_alloc) * sizeof(*group->sos_list)); - return(group); -} - -STATIC void resize_SOSgroup(SOSgroup *group) -{ - if(group->sos_count == group->sos_alloc) { - group->sos_alloc = (int)((double) group->sos_alloc*RESIZEFACTOR); - group->sos_list = (SOSrec **) realloc(group->sos_list, - (group->sos_alloc) * sizeof(*group->sos_list)); - } -} - -STATIC int append_SOSgroup(SOSgroup *group, SOSrec *SOS) -{ - int i, k; - SOSrec *SOSHold; - - /* Check if we should resize */ - resize_SOSgroup(group); - - /* First append to the end of the list */ - group->sos_list[group->sos_count] = SOS; - group->sos_count++; - i = abs(SOS->type); - SETMAX(group->maxorder, i); - if(i == 1) - group->sos1_count++; - k = group->sos_count; - SOS->tagorder = k; - - /* Sort the SOS list by given priority */ - for(i = group->sos_count-1; i > 0; i--) { - if(group->sos_list[i]->priority < group->sos_list[i-1]->priority) { - SOSHold = group->sos_list[i]; - group->sos_list[i] = group->sos_list[i-1]; - group->sos_list[i-1] = SOSHold; - if(SOSHold == SOS) - k = i; /* This is the index in the [1..> range */ - } - else - break; - } - /* Return the list index of the new SOS */ - return( k ); -} - - -STATIC int clean_SOSgroup(SOSgroup *group, MYBOOL forceupdatemap) -{ - int i, n, k; - SOSrec *SOS; - - if(group == NULL) - return( 0 ); - - /* Delete any SOS without members or trivial member count */ - n = 0; - if(group->sos_alloc > 0) { - group->maxorder = 0; - for(i = group->sos_count; i > 0; i--) { - SOS = group->sos_list[i-1]; - k = SOS->members[0]; - if((k == 0) || /* Empty */ - ((k == abs(SOS->type)) && (k <= 2))) { /* Trivial */ - delete_SOSrec(group, i); - n++; - } - else { - SETMAX(group->maxorder, abs(SOS->type)); - } - } - if((n > 0) || forceupdatemap) - SOS_member_updatemap(group); - } - return( n ); -} - - -STATIC void free_SOSgroup(SOSgroup **group) -{ - int i; - - if((group == NULL) || (*group == NULL)) - return; - if((*group)->sos_alloc > 0) { - for(i = 0; i < (*group)->sos_count; i++) - free_SOSrec((*group)->sos_list[i]); - FREE((*group)->sos_list); - FREE((*group)->membership); - FREE((*group)->memberpos); - } - FREE(*group); -} - -/* SOS record functions */ -STATIC SOSrec *create_SOSrec(SOSgroup *group, char *name, int type, int priority, int size, int *variables, REAL *weights) -{ - SOSrec *SOS; - - SOS = (SOSrec *) calloc(1 , sizeof(*SOS)); - SOS->parent = group; - SOS->type = type; - if(name == NULL) - SOS->name = NULL; - else - { - allocCHAR(group->lp, &SOS->name, (int) (strlen(name)+1), FALSE); - strcpy(SOS->name, name); - } - if(type < 0) - type = abs(type); - SOS->tagorder = 0; - SOS->size = 0; - SOS->priority = priority; - SOS->members = NULL; - SOS->weights = NULL; - SOS->membersSorted = NULL; - SOS->membersMapped = NULL; - - if(size > 0) - size = append_SOSrec(SOS, size, variables, weights); - - return(SOS); -} - - -STATIC int append_SOSrec(SOSrec *SOS, int size, int *variables, REAL *weights) -{ - int i, oldsize, newsize, nn; - lprec *lp = SOS->parent->lp; - - oldsize = SOS->size; - newsize = oldsize + size; - nn = abs(SOS->type); - - /* Shift existing active data right (normally zero) */ - if(SOS->members == NULL) - allocINT(lp, &SOS->members, 1+newsize+1+nn, TRUE); - else { - allocINT(lp, &SOS->members, 1+newsize+1+nn, AUTOMATIC); - for(i = newsize+1+nn; i > newsize+1; i--) - SOS->members[i] = SOS->members[i-size]; - } - SOS->members[0] = newsize; - SOS->members[newsize+1] = nn; - - /* Copy the new data into the arrays */ - if(SOS->weights == NULL) - allocREAL(lp, &SOS->weights, 1+newsize, TRUE); - else - allocREAL(lp, &SOS->weights, 1+newsize, AUTOMATIC); - for(i = oldsize+1; i <= newsize; i++) { - SOS->members[i] = variables[i-oldsize-1]; - if((SOS->members[i] < 1) || (SOS->members[i] > lp->columns)) - report(lp, IMPORTANT, "append_SOS_rec: Invalid SOS variable definition for index %d\n", SOS->members[i]); - else { - if(SOS->isGUB) - lp->var_type[SOS->members[i]] |= ISGUB; - else - lp->var_type[SOS->members[i]] |= ISSOS; - } - if(weights == NULL) - SOS->weights[i] = i; /* Follow standard, which is sorted ascending */ - else - SOS->weights[i] = weights[i-oldsize-1]; - SOS->weights[0] += SOS->weights[i]; - } - - /* Sort the new paired lists ascending by weight (simple bubble sort) */ - i = sortByREAL(SOS->members, SOS->weights, newsize, 1, TRUE); - if(i > 0) - report(lp, DETAILED, "append_SOS_rec: Non-unique SOS variable weight for index %d\n", i); - - /* Define mapping arrays to search large SOS's faster */ - allocINT(lp, &SOS->membersSorted, newsize, AUTOMATIC); - allocINT(lp, &SOS->membersMapped, newsize, AUTOMATIC); - for(i = oldsize+1; i <= newsize; i++) { - SOS->membersSorted[i - 1] = SOS->members[i]; - SOS->membersMapped[i - 1] = i; - } - sortByINT(SOS->membersMapped, SOS->membersSorted, newsize, 0, TRUE); - - /* Confirm the new size */ - SOS->size = newsize; - - return(newsize); - -} - -STATIC int make_SOSchain(lprec *lp, MYBOOL forceresort) -{ - int i, j, k, n; - MYBOOL *hold = NULL; - REAL *order, sum, weight; - SOSgroup *group = lp->SOS; - - /* PART A: Resort individual SOS member lists, if specified */ - if(forceresort) - SOS_member_sortlist(group, 0); - - /* PART B: Tally SOS variables and create master SOS variable list */ - n = 0; - for(i = 0; i < group->sos_count; i++) - n += group->sos_list[i]->size; - lp->sos_vars = n; - if(lp->sos_vars > 0) /* Prevent memory loss in case of multiple solves */ - FREE(lp->sos_priority); - allocINT(lp, &lp->sos_priority, n, FALSE); - allocREAL(lp, &order, n, FALSE); - - /* Move variable data to the master SOS list and sort by ascending weight */ - n = 0; - sum = 0; - for(i = 0; i < group->sos_count; i++) { - for(j = 1; j <= group->sos_list[i]->size; j++) { - lp->sos_priority[n] = group->sos_list[i]->members[j]; - weight = group->sos_list[i]->weights[j]; - sum += weight; - order[n] = sum; - n++; - } - } - hpsortex(order, n, 0, sizeof(*order), FALSE, compareREAL, lp->sos_priority); - FREE(order); - - /* Remove duplicate SOS variables */ - allocMYBOOL(lp, &hold, lp->columns+1, TRUE); - k = 0; - for(i = 0; i < n; i++) { - j = lp->sos_priority[i]; - if(!hold[j]) { - hold[j] = TRUE; - if(k < i) - lp->sos_priority[k] = j; - k++; - } - } - FREE(hold); - - /* Adjust the size of the master variable list, if necessary */ - if(k < lp->sos_vars) { - allocINT(lp, &lp->sos_priority, k, AUTOMATIC); - lp->sos_vars = k; - } - - return( k ); - -} - - -STATIC MYBOOL delete_SOSrec(SOSgroup *group, int sosindex) -{ -#ifdef Paranoia - if((sosindex <= 0) || (sosindex > group->sos_count)) { - report(group->lp, IMPORTANT, "delete_SOSrec: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - /* Delete and free the SOS record */ - if(abs(SOS_get_type(group, sosindex)) == 1) - group->sos1_count--; - free_SOSrec(group->sos_list[sosindex-1]); - while(sosindex < group->sos_count) { - group->sos_list[sosindex-1] = group->sos_list[sosindex]; - sosindex++; - } - group->sos_count--; - - /* Update maxorder */ - group->maxorder = 0; - for(sosindex = 0; sosindex < group->sos_count; sosindex++) { - SETMAX(group->maxorder, abs(group->sos_list[sosindex]->type)); - } - - return(TRUE); -} - - -STATIC void free_SOSrec(SOSrec *SOS) -{ - if(SOS->name != NULL) - FREE(SOS->name); - if(SOS->size > 0) { - FREE(SOS->members); - FREE(SOS->weights); - FREE(SOS->membersSorted); - FREE(SOS->membersMapped); - } - FREE(SOS); -} - - -STATIC MYBOOL SOS_member_sortlist(SOSgroup *group, int sosindex) -/* Routine to (re-)sort SOS member arrays for faster access to large SOSes */ -{ - int i, n; - int *list; - lprec *lp = group->lp; - SOSrec *SOS; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_member_sortlist: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if((sosindex == 0) && (group->sos_count == 1)) - sosindex = 1; - - if(sosindex == 0) { - for(i = 1; i <= group->sos_count; i++) { - if(!SOS_member_sortlist(group, i)) - return(FALSE); - } - } - else { - SOS = group->sos_list[sosindex-1]; - list = SOS->members; - n = list[0]; - /* Make sure that the arrays are properly allocated and sized */ - if(n != group->sos_list[sosindex-1]->size) { - allocINT(lp, &SOS->membersSorted, n, AUTOMATIC); - allocINT(lp, &SOS->membersMapped, n, AUTOMATIC); - group->sos_list[sosindex-1]->size = n; - } - /* Reload the arrays and do the sorting */ - for(i = 1; i <= n; i++) { - SOS->membersSorted[i - 1] = list[i]; - SOS->membersMapped[i - 1] = i; - } - sortByINT(SOS->membersMapped, SOS->membersSorted, n, 0, TRUE); - } - return( TRUE ); -} - -STATIC int SOS_member_updatemap(SOSgroup *group) -{ - int i, j, k, n, nvars = 0, - *list, *tally = NULL; - SOSrec *rec; - lprec *lp = group->lp; - - /* (Re)-initialize usage arrays */ - allocINT(lp, &group->memberpos, lp->columns+1, AUTOMATIC); - allocINT(lp, &tally, lp->columns+1, TRUE); - - /* Get each variable's SOS membership count */ - for(i = 0; i < group->sos_count; i++) { - rec = group->sos_list[i]; - n = rec->size; - list = rec->members; - for(j = 1; j <= n; j++) { - k = list[j]; -#ifdef Paranoia - if((k < 1) || (k > lp->columns)) - report(lp, SEVERE, "SOS_member_updatemap: Member %j of SOS number %d is out of column range (%d)\n", - j, i+1, k); -#endif - tally[k]++; - } - - } - - /* Compute pointer into column-sorted array */ - group->memberpos[0] = 0; - for(i = 1; i <= lp->columns; i++) { - n = tally[i]; - if(n > 0) - nvars++; - group->memberpos[i] = group->memberpos[i-1] + n; - } - n = group->memberpos[lp->columns]; - MEMCOPY(tally+1, group->memberpos, lp->columns); - - /* Load the column-sorted SOS indeces / pointers */ - allocINT(lp, &group->membership, n+1, AUTOMATIC); - for(i = 0; i < group->sos_count; i++) { - rec = group->sos_list[i]; - n = rec->size; - list = rec->members; - for(j = 1; j <= n; j++) { - k = tally[list[j]]++; -#ifdef Paranoia - if(k > group->memberpos[lp->columns]) - report(lp, SEVERE, "SOS_member_updatemap: Member mapping for variable %j of SOS number %d is invalid\n", - list[j], i+1); -#endif - group->membership[k] = i+1; - } - } - FREE(tally); - - return( nvars ); -} - - -STATIC MYBOOL SOS_shift_col(SOSgroup *group, int sosindex, int column, int delta, LLrec *usedmap, MYBOOL forceresort) -/* Routine to adjust SOS indeces for variable insertions or deletions; - Note: SOS_shift_col must be called before make_SOSchain! */ -{ - int i, ii, n, nn, nr; - int changed; - int *list; - REAL *weights; - -#ifdef Paranoia - lprec *lp = group->lp; - - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_shift_col: Invalid SOS index %d\n", sosindex); - return(FALSE); - } - else if((column < 1) || (delta == 0)) { - report(lp, IMPORTANT, "SOS_shift_col: Invalid column %d specified with delta %d\n", - column, delta); - return(FALSE); - } -#endif - - if((sosindex == 0) && (group->sos_count == 1)) - sosindex = 1; - - if(sosindex == 0) { - for(i = 1; i <= group->sos_count; i++) { - if(!SOS_shift_col(group, i, column, delta, usedmap, forceresort)) - return(FALSE); - } - } - else { - list = group->sos_list[sosindex-1]->members; - weights = group->sos_list[sosindex-1]->weights; - n = list[0]; - nn = list[n+1]; - - /* Case where variable indeces are to be incremented */ - if(delta > 0) { - for(i = 1; i <= n; i++) { - if(list[i] >= column) - list[i] += delta; - } - } - /* Case where variables are to be deleted/indeces decremented */ - else { - changed = 0; - if(usedmap != NULL) { - int *newidx = NULL; - /* Defer creation of index mapper until we are sure that a - member of this SOS is actually targeted for deletion */ - if(newidx == NULL) { - allocINT(group->lp, &newidx, group->lp->columns+1, TRUE); - for(i = firstActiveLink(usedmap), ii = 1; i != 0; - i = nextActiveLink(usedmap, i), ii++) - newidx[i] = ii; - } - for(i = 1, ii = 0; i <= n; i++) { - nr = list[i]; - /* Check if this SOS variable should be deleted */ - if(!isActiveLink(usedmap, nr)) - continue; - - /* If the index is "high" then make adjustment and shift */ - changed++; - ii++; - list[ii] = newidx[nr]; - weights[ii] = weights[i]; - } - FREE(newidx); - } - else - for(i = 1, ii = 0; i <= n; i++) { - nr = list[i]; - /* Check if this SOS variable should be deleted */ - if((nr >= column) && (nr < column-delta)) - continue; - /* If the index is "high" then decrement */ - if(nr > column) { - changed++; - nr += delta; - } - ii++; - list[ii] = nr; - weights[ii] = weights[i]; - } - /* Update the SOS length / type indicators */ - if(ii < n) { - list[0] = ii; - list[ii+1] = nn; - } - - /* Update mapping arrays to search large SOS's faster */ - if(forceresort && ((ii < n) || (changed > 0))) - SOS_member_sortlist(group, sosindex); - } - - } - return(TRUE); - -} - -int SOS_member_count(SOSgroup *group, int sosindex) -{ - SOSrec *SOS; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(group->lp, IMPORTANT, "SOS_member_count: Invalid SOS index %d\n", sosindex); - return( -1 ); - } -#endif - SOS = group->sos_list[sosindex-1]; - return( SOS->members[0] ); -} - -int SOS_member_delete(SOSgroup *group, int sosindex, int member) -{ - int *list, i, i2, k, n, nn = 0; - SOSrec *SOS; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(group->lp, IMPORTANT, "SOS_member_delete: Invalid SOS index %d\n", sosindex); - return( -1 ); - } -#endif - - if(sosindex == 0) { - for(i = group->memberpos[member-1]; i < group->memberpos[member]; i++) { - k = group->membership[i]; - n = SOS_member_delete(group, k, member); - if(n >= 0) - nn += n; - else - return( n ); - } - /* We must update the mapper */ - k = group->memberpos[member]; - i = group->memberpos[member-1]; - n = group->memberpos[lp->columns] - k; - if(n > 0) - MEMCOPY(group->membership + i, group->membership + k, n); - for(i = member; i <= lp->columns; i++) - group->memberpos[i] = group->memberpos[i-1]; - } - else { - SOS = group->sos_list[sosindex-1]; - list = SOS->members; - n = list[0]; - - /* Find the offset of the member */ - i = 1; - while((i <= n) && (abs(list[i]) != member)) - i++; - if(i > n) - return( -1 ); - nn++; - - /* Shift remaining members *and* the active count one position left */ - while(i <= n) { - list[i] = list[i+1]; - i++; - } - list[0]--; - SOS->size--; - - /* Do the same with the active list one position left */ - i = n + 1; - i2 = i + list[n]; - k = i + 1; - while(i < i2) { - if(abs(list[k]) == member) - k++; - list[i] = list[k]; - i++; - k++; - } - } - - return( nn ); -} - -int SOS_get_type(SOSgroup *group, int sosindex) -{ -#ifdef Paranoia - if((sosindex < 1) || (sosindex > group->sos_count)) { - report(group->lp, IMPORTANT, "SOS_get_type: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - return(group->sos_list[sosindex-1]->type); -} - - -int SOS_infeasible(SOSgroup *group, int sosindex) -{ - int i, n, nn, varnr, failindex, *list; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_infeasible: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(sosindex == 0 && group->sos_count == 1) - sosindex = 1; - - failindex = 0; - if(sosindex == 0) { - for(i = 1; i <= group->sos_count; i++) { - failindex = SOS_infeasible(group, i); - if(failindex > 0) break; - } - } - else { - list = group->sos_list[sosindex-1]->members; - n = list[0]; - nn = list[n+1]; - /* Find index of next lower-bounded variable */ - for(i = 1; i <= n; i++) { - varnr = abs(list[i]); - if((lp->orig_lowbo[lp->rows + varnr] > 0) && - !((lp->sc_vars > 0) && is_semicont(lp, varnr))) - break; - } - - /* Find if there is another lower-bounded variable beyond the type window */ - i = i+nn; - while(i <= n) { - varnr = abs(list[i]); - if((lp->orig_lowbo[lp->rows + varnr] > 0) && - !((lp->sc_vars > 0) && is_semicont(lp, varnr))) - break; - i++; - } - if(i <= n) - failindex = abs(list[i]); - } - return(failindex); -} - - -int SOS_member_index(SOSgroup *group, int sosindex, int member) -{ - int n; - SOSrec *SOS; - - SOS = group->sos_list[sosindex-1]; - n = SOS->members[0]; - - n = searchFor(member, SOS->membersSorted, n, 0, FALSE); - if(n >= 0) - n = SOS->membersMapped[n]; - - return(n); -} - - -int SOS_memberships(SOSgroup *group, int varnr) -{ - int i, n = 0; - lprec *lp; - - /* Check if there is anything to do */ - if((group == NULL) || (SOS_count(lp = group->lp) == 0)) - return( n ); - -#ifdef Paranoia - if((varnr < 0) || (varnr > lp->columns)) { - report(lp, IMPORTANT, "SOS_memberships: Invalid variable index %d given\n", varnr); - return( n ); - } -#endif - - if(varnr == 0) { - for(i = 1; i <= lp->columns; i++) - if(group->memberpos[i] > group->memberpos[i-1]) - n++; - } - else - n = group->memberpos[varnr] - group->memberpos[varnr-1]; - - return( n ); -} - - -int SOS_is_member(SOSgroup *group, int sosindex, int column) -{ - int i, n = FALSE, *list; - lprec *lp; - - if(group == NULL) - return( FALSE ); - lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_is_member: Invalid SOS index %d\n", sosindex); - return(n); - } -#endif - - if(sosindex == 0) { - if(lp->var_type[column] & (ISSOS | ISGUB)) - n = (MYBOOL) (SOS_memberships(group, column) > 0); - } - else if(lp->var_type[column] & (ISSOS | ISGUB)) { - - /* Search for the variable */ - i = SOS_member_index(group, sosindex, column); - - /* Signal active status if found, otherwise return FALSE */ - if(i > 0) { - list = group->sos_list[sosindex-1]->members; - if(list[i] < 0) - n = -TRUE; - else - n = TRUE; - } - } - return(n); -} - - -MYBOOL SOS_is_member_of_type(SOSgroup *group, int column, int sostype) -{ - int i, k, n; - - if(group != NULL) - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - k = group->membership[i]; - n = SOS_get_type(group, k); - if(((n == sostype) || - ((sostype == SOSn) && (n > 2))) && SOS_is_member(group, k, column)) - return(TRUE); - } - return(FALSE); -} - - -MYBOOL SOS_set_GUB(SOSgroup *group, int sosindex, MYBOOL state) -{ - int i; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(group->lp, IMPORTANT, "SOS_set_GUB: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - if((sosindex == 0) && (group->sos_count == 1)) - sosindex = 1; - - if(sosindex == 0) { - for(i = 1; i <= group->sos_count; i++) - SOS_set_GUB(group, i, state); - } - else - group->sos_list[sosindex-1]->isGUB = state; - return(TRUE); -} - - -MYBOOL SOS_is_GUB(SOSgroup *group, int sosindex) -{ - int i; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(group->lp, IMPORTANT, "SOS_is_GUB: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if((sosindex == 0) && (group->sos_count == 1)) - sosindex = 1; - - if(sosindex == 0) { - for(i = 1; i <= group->sos_count; i++) { - if(SOS_is_GUB(group, i)) - return(TRUE); - } - return(FALSE); - } - else - return( group->sos_list[sosindex-1]->isGUB ); -} - - -MYBOOL SOS_is_marked(SOSgroup *group, int sosindex, int column) -{ - int i, k, n, *list; - lprec *lp; - - if(group == NULL) - return( FALSE ); - lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_is_marked: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(!(lp->var_type[column] & (ISSOS | ISGUB))) - return(FALSE); - - if(sosindex == 0) { - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - k = group->membership[i]; - n = SOS_is_marked(group, k, column); - if(n) - return(TRUE); - } - } - else { - list = group->sos_list[sosindex-1]->members; - n = list[0]; - - /* Search for the variable (normally always faster to do linear search here) */ - column = -column; - for(i = 1; i <= n; i++) - if(list[i] == column) - return(TRUE); - } - return(FALSE); -} - - -MYBOOL SOS_is_active(SOSgroup *group, int sosindex, int column) -{ - int i, n, nn, *list; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_is_active: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(!(lp->var_type[column] & (ISSOS | ISGUB))) - return(FALSE); - - if(sosindex == 0) { - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - nn = group->membership[i]; - n = SOS_is_active(group, nn, column); - if(n) - return(TRUE); - } - } - else { - - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - - /* Scan the active (non-zero) SOS index list */ - for(i = 1; (i <= nn) && (list[n+i] != 0); i++) - if(list[n+i] == column) - return(TRUE); - } - return(FALSE); -} - - -MYBOOL SOS_is_full(SOSgroup *group, int sosindex, int column, MYBOOL activeonly) -{ - int i, nn, n, *list; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_is_full: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(!(lp->var_type[column] & (ISSOS | ISGUB))) - return(FALSE); - - if(sosindex == 0) { - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - nn = group->membership[i]; - if(SOS_is_full(group, nn, column, activeonly)) - return(TRUE); - } - } - else if(SOS_is_member(group, sosindex, column)) { - - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - - /* Info: Last item in the active list is non-zero if the current SOS is full */ - if(list[n+nn] != 0) - return(TRUE); - - if(!activeonly) { - /* Spool to last active variable */ - for(i = nn-1; (i > 0) && (list[n+i] == 0); i--); - /* Having found it, check if subsequent variables are set (via bounds) as inactive */ - if(i > 0) { - nn -= i; /* Compute unused active slots */ - i = SOS_member_index(group, sosindex, list[n+i]); - for(; (nn > 0) && (list[i] < 0); i++, nn--); - if(nn == 0) - return(TRUE); - } - } - } - - return(FALSE); -} - - -MYBOOL SOS_can_activate(SOSgroup *group, int sosindex, int column) -{ - int i, n, nn, nz, *list; - lprec *lp; - - if(group == NULL) - return( FALSE ); - lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_can_activate: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(!(lp->var_type[column] & (ISSOS | ISGUB))) - return(FALSE); - - if(sosindex == 0) { - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - nn = group->membership[i]; - n = SOS_can_activate(group, nn, column); - if(n == FALSE) - return(FALSE); - } - } - else if(SOS_is_member(group, sosindex, column)) { - - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - -#if 0 - /* Accept if the SOS is empty */ - if(list[n+1] == 0) - return(TRUE); -#endif - - /* Cannot activate a variable if the SOS is full */ - if(list[n+nn] != 0) - return(FALSE); - - /* Check if there are variables quasi-active via non-zero lower bounds */ - nz = 0; - for(i = 1; i < n; i++) - if(lp->bb_bounds->lowbo[lp->rows+abs(list[i])] > 0) { - nz++; - /* Reject outright if selected column has a non-zero lower bound */ - if(list[i] == column) - return(FALSE); - } -#ifdef Paranoia - if(nz > nn) - report(lp, SEVERE, "SOS_can_activate: Found too many non-zero member variables for SOS index %d\n", sosindex); -#endif - for(i = 1; i <= nn; i++) { - if(list[n+i] == 0) - break; - if(lp->bb_bounds->lowbo[lp->rows+list[n+i]] == 0) - nz++; - } - if(nz == nn) - return(FALSE); - - /* Accept if the SOS is empty */ - if(list[n+1] == 0) - return(TRUE); - - /* Check if we can set variable active in SOS2..SOSn - (must check left and right neighbours if one variable is already active) */ - if(nn > 1) { - - /* Find the variable that was last activated; - Also check that the candidate variable is not already active */ - for(i = 1; i <= nn; i++) { - if(list[n+i] == 0) - break; - if(list[n+i] == column) - return(FALSE); - } - i--; - nn = list[n+i]; - - /* SOS accepts an additional variable; confirm neighbourness of candidate; - Search for the SOS set index of the last activated variable */ - n = list[0]; - for(i = 1; i <= n; i++) - if(abs(list[i]) == nn) - break; - if(i > n) { - report(lp, CRITICAL, "SOS_can_activate: Internal index error at SOS %d\n", sosindex); - return(FALSE); - } - - /* SOS accepts an additional variable; confirm neighbourness of candidate */ - - /* Check left neighbour */ - if((i > 1) && (list[i-1] == column)) - return(TRUE); - /* Check right neighbour */ - if((i < n) && (list[i+1] == column)) - return(TRUE); - - /* It is not the right neighbour; return false */ - return(FALSE); - } - } - return(TRUE); -} - - -MYBOOL SOS_set_marked(SOSgroup *group, int sosindex, int column, MYBOOL asactive) -{ - int i, n, nn, *list; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_set_marked: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(!(lp->var_type[column] & (ISSOS | ISGUB))) - return(FALSE); - - if(sosindex == 0) { - - /* Define an IBM-"SOS3" member variable temporarily as integer, if it is - not already a permanent integer; is reset in SOS_unmark */ - if(asactive && !is_int(lp, column) && SOS_is_member_of_type(group, column, SOS3)) { - lp->var_type[column] |= ISSOSTEMPINT; - set_int(lp, column, TRUE); - } - - nn = 0; - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - n = group->membership[i]; - if(SOS_set_marked(group, n, column, asactive)) - nn++; - } - return((MYBOOL) (nn == group->sos_count)); - } - else { - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - - /* Search for the variable */ - i = SOS_member_index(group, sosindex, column); - - /* First mark active in the set member list as used */ - if((i > 0) && (list[i] > 0)) - list[i] *= -1; - else - return(TRUE); - - /* Then move the variable to the live list */ - if(asactive) { - for(i = 1; i <= nn; i++) { - if(list[n+i] == column) - return(FALSE); - else if(list[n+i] == 0) { - list[n+i] = column; - return(FALSE); - } - } - } - return(TRUE); - } -} - - -MYBOOL SOS_unmark(SOSgroup *group, int sosindex, int column) -{ - int i, n, nn, *list; - MYBOOL isactive; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_unmark: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - if(!(lp->var_type[column] & (ISSOS | ISGUB))) - return(FALSE); - - - if(sosindex == 0) { - - /* Undefine a SOS3 member variable that has temporarily been set as integer */ - if(lp->var_type[column] & ISSOSTEMPINT) { - lp->var_type[column] &= !ISSOSTEMPINT; - set_int(lp, column, FALSE); - } - - nn = 0; - for(i = group->memberpos[column-1]; i < group->memberpos[column]; i++) { - n = group->membership[i]; - if(SOS_unmark(group, n, column)) - nn++; - } - return((MYBOOL) (nn == group->sos_count)); - } - else { - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - - /* Search for the variable */ - i = SOS_member_index(group, sosindex, column); - - /* Restore sign in main list */ - if((i > 0) && (list[i] < 0)) - list[i] *= -1; - else - return(TRUE); - - /* Find the variable in the active list... */ - isactive = SOS_is_active(group, sosindex, column); - if(isactive) { - for(i = 1; i <= nn; i++) - if(list[n+i] == column) - break; - /* ...shrink the list if found, otherwise return error */ - if(i <= nn) { - for(; ilp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_fix_unmarked: Invalid SOS index %d\n", sosindex); - return(FALSE); - } -#endif - - count = 0; - if(sosindex == 0) { - for(i = group->memberpos[variable-1]; i < group->memberpos[variable]; i++) { - n = group->membership[i]; - count += SOS_fix_unmarked(group, n, variable, bound, value, isupper, diffcount, changelog); - } - } - else { - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - - /* Count the number of active and free SOS variables */ - nn = list[n]; - for(i = 1; i <= nn; i++) { - if(list[n+i] == 0) - break; - } - i--; - i = nn - i; /* Establish the number of unused slots */ - - /* Determine the free SOS variable window */ - if(i == nn) { - nLeft = 0; - nRight = SOS_member_index(group, sosindex, variable); - } - else { - nLeft = SOS_member_index(group, sosindex, list[n+1]); - if(variable == list[n+1]) - nRight = nLeft; - else - nRight = SOS_member_index(group, sosindex, variable); - } - - nRight += i; /* Loop (nRight+1)..n */ - - /* Fix variables outside of the free SOS variable window */ - for(i = 1; i < n; i++) { - /* Skip the SOS variable window */ - if((i >= nLeft) && (i <= nRight)) - continue; - /* Otherwise proceed to set bound */ - ii = list[i]; - if(ii > 0) { - ii += lp->rows; - if(bound[ii] != value) { - /* Verify that we don't violate original bounds */ - if(isupper && (value < lp->orig_lowbo[ii])) - return(-ii); - else if(!isupper && (value > lp->orig_upbo[ii])) - return(-ii); - /* OK, set the new bound */ - count++; - if(changelog == NULL) - bound[ii] = value; - else - modifyUndoLadder(changelog, ii, bound, value); - - } - if((diffcount != NULL) && (lp->solution[ii] != value)) - (*diffcount)++; - } - } - } - return(count); -} - -int *SOS_get_candidates(SOSgroup *group, int sosindex, int column, MYBOOL excludetarget, - REAL *upbound, REAL *lobound) -{ - int i, ii, j, n, nn = 0, *list, *candidates = NULL; - lprec *lp = group->lp; - - if(group == NULL) - return( candidates ); - -#ifdef Paranoia - if(sosindex > group->sos_count) { - report(lp, IMPORTANT, "SOS_get_candidates: Invalid index %d\n", sosindex); - return( candidates ); - } -#endif - - /* Determine SOS target(s); note that if "sosindex" is negative, only - the first non-empty SOS where "column" is a member is processed */ - if(sosindex <= 0) { - i = 0; - ii = group->sos_count; - } - else { - i = sosindex - 1; - ii = sosindex; - } - - /* Tally candidate usage */ - allocINT(lp, &candidates, lp->columns+1, TRUE); - for(; i < ii; i++) { - if(!SOS_is_member(group, i+1, column)) - continue; - list = group->sos_list[i]->members; - n = list[0]; - while(n > 0) { - j = list[n]; - if((j > 0) && (upbound[lp->rows+j] > 0)) { - if(lobound[lp->rows+j] > 0) { - report(lp, IMPORTANT, "SOS_get_candidates: Invalid non-zero lower bound setting\n"); - n = 0; - goto Finish; - } - if(candidates[j] == 0) - nn++; - candidates[j]++; - } - n--; - } - if((sosindex < 0) && (nn > 1)) - break; - } - - /* Condense the list into indeces */ - n = 0; - for(i = 1; i <= lp->columns; i++) { - if((candidates[i] > 0) && (!excludetarget || (i != column))) { - n++; - candidates[n] = i; - } - } - - /* Finalize */ -Finish: - candidates[0] = n; - if(n == 0) - FREE(candidates); - - return( candidates); - -} - -int SOS_fix_list(SOSgroup *group, int sosindex, int variable, REAL *bound, - int *varlist, MYBOOL isleft, DeltaVrec *changelog) -{ - int i, ii, jj, count = 0; - REAL value = 0; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_fix_list: Invalid index %d\n", sosindex); - return(FALSE); - } -#endif - - if(sosindex == 0) { - for(i = group->memberpos[variable-1]; i < group->memberpos[variable]; i++) { - ii = group->membership[i]; - count += SOS_fix_list(group, ii, variable, bound, varlist, isleft, changelog); - } - } - else { - - /* Establish the number of unmarked variables in the left window - (note that "variable" should have been marked previously) */ - ii = varlist[0] / 2; - if(isleft) { - i = 1; - if(isleft == AUTOMATIC) - ii = varlist[0]; - } - else { - i = ii + 1; - ii = varlist[0]; - } - - /* Loop over members to fix values at the new bound (zero) */ - while(i <= ii) { - if(SOS_is_member(group, sosindex, varlist[i])) { - jj = lp->rows + varlist[i]; - - /* Verify that we don't violate original bounds */ - if(value < lp->orig_lowbo[jj]) - return( -jj ); - /* OK, set the new bound */ - count++; - if(changelog == NULL) - bound[jj] = value; - else - modifyUndoLadder(changelog, jj, bound, value); - } - i++; - } - - } - return( count ); -} - -int SOS_is_satisfied(SOSgroup *group, int sosindex, REAL *solution) -/* Determine if the SOS is satisfied for the current solution vector; - The return code is in the range [-2..+2], depending on the type of - satisfaction. Positive return value means too many non-zero values, - negative value means set incomplete: - - -2: Set member count not full (SOS3) - -1: Set member count not full - 0: Set is full (also returned if the SOS index is invalid) - 1: Too many non-zero sequential variables - 2: Set consistency error - -*/ -{ - int i, n, nn, count, *list; - int type, status = 0; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_is_satisfied: Invalid index %d\n", sosindex); - return( SOS_COMPLETE ); - } -#endif - - if((sosindex == 0) && (group->sos_count == 1)) - sosindex = 1; - - if(sosindex == 0) { - for(i = 1; i <= group->sos_count; i++) { - status = SOS_is_satisfied(group, i, solution); - if((status != SOS_COMPLETE) && (status != SOS_INCOMPLETE)) - break; - } - } - else { - type = SOS_get_type(group, sosindex); - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - - /* Count the number of active SOS variables */ - for(i = 1; i <= nn; i++) { - if(list[n+i] == 0) - break; - } - count = i-1; - if(count == nn) - status = SOS_COMPLETE; /* Set is full */ - else - status = SOS_INCOMPLETE; /* Set is partial */ - - /* Find index of the first active variable; fail if some are non-zero */ - if(count > 0) { - nn = list[n+1]; - for(i = 1; i < n; i++) { - if((abs(list[i]) == nn) || (solution[lp->rows + abs(list[i])] != 0)) - break; - } - if(abs(list[i]) != nn) - status = SOS_INTERNALERROR; /* Set consistency error (leading set variables are non-zero) */ - else { - /* Scan active SOS variables until we find a non-zero value */ - while(count > 0) { - if(solution[lp->rows + abs(list[i])] != 0) - break; - i++; - count--; - } - /* Scan active non-zero SOS variables; break at first non-zero (rest required to be zero) */ - while(count > 0) { - if(solution[lp->rows + abs(list[i])] == 0) - break; - i++; - count--; - } - if(count > 0) - status = SOS_INTERNALERROR; /* Set consistency error (active set variables are zero) */ - } - } - else { - i = 1; - /* There are no active variables; see if we have happened to find a valid header */ - while((i < n) && (solution[lp->rows + abs(list[i])] == 0)) - i++; - count = 0; - while((i < n) && (count <= nn) && (solution[lp->rows + abs(list[i])] != 0)) { - count++; - i++; - } - if(count > nn) - status = SOS_INFEASIBLE; /* Too-many sequential non-zero variables */ - } - - /* Scan the trailing set of SOS variables; fail if some are non-zero */ - if(status <= 0) { - n--; - while(i <= n) { - if(solution[lp->rows + abs(list[i])] != 0) - break; - i++; - } - if(i <= n) - status = SOS_INFEASIBLE; /* Too-many sequential non-zero variables */ - - /* Code member deficiency for SOS3 separately */ - else if((status == -1) && (type <= SOS3)) - status = SOS3_INCOMPLETE; - } - - } - return( status ); -} - -MYBOOL SOS_is_feasible(SOSgroup *group, int sosindex, REAL *solution) -/* Determine if the SOS is feasible up to the current SOS variable */ -{ - int i, n, nn, *list; - MYBOOL status = TRUE; - lprec *lp = group->lp; - -#ifdef Paranoia - if((sosindex < 0) || (sosindex > group->sos_count)) { - report(lp, IMPORTANT, "SOS_is_feasible: Invalid SOS index %d\n", sosindex); - return( 0 ); - } -#endif - - if((sosindex == 0) && (group->sos_count == 1)) - sosindex = 1; - - if(sosindex == 0) { - for(i = 1; status && (i <= group->sos_count); i++) { - status = SOS_is_feasible(group, i, solution); - } - } - else { - list = group->sos_list[sosindex-1]->members; - n = list[0]+1; - nn = list[n]; - if(nn <= 2) - return(status); - - /* Find if we have a gap in the non-zero solution values */ - i = 1; - sosindex = 0; - while((i <= nn) && (list[n+i] != 0)) { - while((i <= nn) && (list[n+i] != 0) && (solution[lp->rows+list[n+i]] == 0)) - i++; - if((i <= nn) && (list[n+i] != 0)) { - i++; /* Step to next */ - while((i <= nn) && (list[n+i] != 0) && (solution[lp->rows+list[n+i]] != 0)) - i++; - sosindex++; - } - i++; /* Step to next */ - } - status = (MYBOOL) (sosindex <= 1); - } - return(status); -} diff --git a/code/3rd_lpsolve/lp_SOS.h b/code/3rd_lpsolve/lp_SOS.h deleted file mode 100644 index 27df12ce..00000000 --- a/code/3rd_lpsolve/lp_SOS.h +++ /dev/null @@ -1,108 +0,0 @@ -#ifndef HEADER_lp_SOS -#define HEADER_lp_SOS - -/* Specially Ordered Sets (SOS) prototypes and settings */ -/* ------------------------------------------------------------------------- */ - -#include "lp_types.h" -#include "lp_utils.h" -#include "lp_matrix.h" - - -/* SOS constraint defines */ -/* ------------------------------------------------------------------------- */ -#define SOS1 1 -#define SOS2 2 -#define SOS3 -1 -#define SOSn MAXINT32 -#define SOS_START_SIZE 10 /* Start size of SOS_list array; realloced if needed */ - -/* Define SOS_is_feasible() return values */ -/* ------------------------------------------------------------------------- */ -#define SOS3_INCOMPLETE -2 -#define SOS_INCOMPLETE -1 -#define SOS_COMPLETE 0 -#define SOS_INFEASIBLE 1 -#define SOS_INTERNALERROR 2 - - -typedef struct _SOSgroup SOSgroup; - -typedef struct _SOSrec -{ - SOSgroup *parent; - int tagorder; - char *name; - int type; - MYBOOL isGUB; - int size; - int priority; - int *members; - REAL *weights; - int *membersSorted; - int *membersMapped; -} SOSrec; - -/* typedef */ struct _SOSgroup -{ - lprec *lp; /* Pointer to owner */ - SOSrec **sos_list; /* Array of pointers to SOS lists */ - int sos_alloc; /* Size allocated to specially ordered sets (SOS1, SOS2...) */ - int sos_count; /* Number of specially ordered sets (SOS1, SOS2...) */ - int maxorder; /* The highest-order SOS in the group */ - int sos1_count; /* Number of the lowest order SOS in the group */ - int *membership; /* Array of variable-sorted indeces to SOSes that the variable is member of */ - int *memberpos; /* Starting positions of the each column's membership list */ -} /* SOSgroup */; - - -#ifdef __cplusplus -extern "C" { -#endif - -/* SOS storage structure */ -STATIC SOSgroup *create_SOSgroup(lprec *lp); -STATIC void resize_SOSgroup(SOSgroup *group); -STATIC int append_SOSgroup(SOSgroup *group, SOSrec *SOS); -STATIC int clean_SOSgroup(SOSgroup *group, MYBOOL forceupdatemap); -STATIC void free_SOSgroup(SOSgroup **group); - -STATIC SOSrec *create_SOSrec(SOSgroup *group, char *name, int type, int priority, int size, int *variables, REAL *weights); -STATIC MYBOOL delete_SOSrec(SOSgroup *group, int sosindex); -STATIC int append_SOSrec(SOSrec *SOS, int size, int *variables, REAL *weights); -STATIC void free_SOSrec(SOSrec *SOS); - -/* SOS utilities */ -STATIC int make_SOSchain(lprec *lp, MYBOOL forceresort); -STATIC int SOS_member_updatemap(SOSgroup *group); -STATIC MYBOOL SOS_member_sortlist(SOSgroup *group, int sosindex); -STATIC MYBOOL SOS_shift_col(SOSgroup *group, int sosindex, int column, int delta, LLrec *usedmap, MYBOOL forceresort); -int SOS_member_delete(SOSgroup *group, int sosindex, int member); -int SOS_get_type(SOSgroup *group, int sosindex); -int SOS_infeasible(SOSgroup *group, int sosindex); -int SOS_member_index(SOSgroup *group, int sosindex, int member); -int SOS_member_count(SOSgroup *group, int sosindex); -int SOS_memberships(SOSgroup *group, int column); -int *SOS_get_candidates(SOSgroup *group, int sosindex, int column, MYBOOL excludetarget, REAL *upbound, REAL *lobound); -int SOS_is_member(SOSgroup *group, int sosindex, int column); -MYBOOL SOS_is_member_of_type(SOSgroup *group, int column, int sostype); -MYBOOL SOS_set_GUB(SOSgroup *group, int sosindex, MYBOOL state); -MYBOOL SOS_is_GUB(SOSgroup *group, int sosindex); -MYBOOL SOS_is_marked(SOSgroup *group, int sosindex, int column); -MYBOOL SOS_is_active(SOSgroup *group, int sosindex, int column); -MYBOOL SOS_is_full(SOSgroup *group, int sosindex, int column, MYBOOL activeonly); -MYBOOL SOS_can_activate(SOSgroup *group, int sosindex, int column); -MYBOOL SOS_set_marked(SOSgroup *group, int sosindex, int column, MYBOOL asactive); -MYBOOL SOS_unmark(SOSgroup *group, int sosindex, int column); -int SOS_fix_unmarked(SOSgroup *group, int sosindex, int variable, REAL *bound, REAL value, - MYBOOL isupper, int *diffcount, DeltaVrec *changelog); -int SOS_fix_list(SOSgroup *group, int sosindex, int variable, REAL *bound, - int *varlist, MYBOOL isleft, DeltaVrec *changelog); -int SOS_is_satisfied(SOSgroup *group, int sosindex, REAL *solution); -MYBOOL SOS_is_feasible(SOSgroup *group, int sosindex, REAL *solution); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_SOS */ diff --git a/code/3rd_lpsolve/lp_bit.h b/code/3rd_lpsolve/lp_bit.h deleted file mode 100644 index 587726f8..00000000 --- a/code/3rd_lpsolve/lp_bit.h +++ /dev/null @@ -1,22 +0,0 @@ -#include "lp_types.h" - -#if defined INLINE -# define MYINLINE INLINE -#else -# define MYINLINE static -#endif - -MYINLINE void set_biton(MYBOOL *bitarray, int item) -{ - bitarray[item / 8] |= (1 << (item % 8)); -} - -MYINLINE void set_bitoff(MYBOOL *bitarray, int item) -{ - bitarray[item / 8] &= ~(1 << (item % 8)); -} - -MYINLINE MYBOOL is_biton(MYBOOL *bitarray, int item) -{ - return( (MYBOOL) ((bitarray[item / 8] & (1 << (item % 8))) != 0) ); -} diff --git a/code/3rd_lpsolve/lp_crash.c b/code/3rd_lpsolve/lp_crash.c deleted file mode 100644 index df56dff3..00000000 --- a/code/3rd_lpsolve/lp_crash.c +++ /dev/null @@ -1,863 +0,0 @@ - -/* - ---------------------------------------------------------------------------------- - Crash management routines in lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h, lp_utils.h, lp_matrix.h - - Release notes: - v1.0.0 1 April 2004 First version. - v1.1.0 20 July 2004 Reworked with flexible matrix storage model. - - ---------------------------------------------------------------------------------- -*/ - -#include - -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_scale.h" -#include "lp_utils.h" -#include "lp_report.h" -#include "lp_matrix.h" -#include "lp_crash.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -MYBOOL crash_basis(lprec *lp) -{ - int i; - MATrec *mat = lp->matA; - MYBOOL ok = TRUE; - - /* Initialize basis indicators */ - if(lp->basis_valid) - lp->var_basic[0] = FALSE; - else - default_basis(lp); - - /* Set initial partial pricing blocks */ - if(lp->rowblocks != NULL) - lp->rowblocks->blocknow = 1; - if(lp->colblocks != NULL) - lp->colblocks->blocknow = ((lp->crashmode == CRASH_NONE) || (lp->colblocks->blockcount == 1) ? 1 : 2); - - /* Construct a basis that is in some measure the "most feasible" */ - if((lp->crashmode == CRASH_MOSTFEASIBLE) && mat_validate(mat)) { - /* The logic here follows Maros */ - LLrec *rowLL = NULL, *colLL = NULL; - int ii, rx, cx, ix, nz; - REAL wx, tx, *rowMAX = NULL, *colMAX = NULL; - int *rowNZ = NULL, *colNZ = NULL, *rowWT = NULL, *colWT = NULL; - REAL *value; - int *rownr, *colnr; - - report(lp, NORMAL, "crash_basis: 'Most feasible' basis crashing selected\n"); - - /* Tally row and column non-zero counts */ - ok = allocINT(lp, &rowNZ, lp->rows+1, TRUE) && - allocINT(lp, &colNZ, lp->columns+1, TRUE) && - allocREAL(lp, &rowMAX, lp->rows+1, FALSE) && - allocREAL(lp, &colMAX, lp->columns+1, FALSE); - if(!ok) - goto Finish; - - nz = mat_nonzeros(mat); - rownr = &COL_MAT_ROWNR(0); - colnr = &COL_MAT_COLNR(0); - value = &COL_MAT_VALUE(0); - for(i = 0; i < nz; - i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) { - rx = *rownr; - cx = *colnr; - wx = fabs(*value); - rowNZ[rx]++; - colNZ[cx]++; - if(i == 0) { - rowMAX[rx] = wx; - colMAX[cx] = wx; - colMAX[0] = wx; - } - else { - SETMAX(rowMAX[rx], wx); - SETMAX(colMAX[cx], wx); - SETMAX(colMAX[0], wx); - } - } - /* Reduce counts for small magnitude to preserve stability */ - rownr = &COL_MAT_ROWNR(0); - colnr = &COL_MAT_COLNR(0); - value = &COL_MAT_VALUE(0); - for(i = 0; i < nz; - i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) { - rx = *rownr; - cx = *colnr; - wx = fabs(*value); -#ifdef CRASH_SIMPLESCALE - if(wx < CRASH_THRESHOLD * colMAX[0]) { - rowNZ[rx]--; - colNZ[cx]--; - } -#else - if(wx < CRASH_THRESHOLD * rowMAX[rx]) - rowNZ[rx]--; - if(wx < CRASH_THRESHOLD * colMAX[cx]) - colNZ[cx]--; -#endif - } - - /* Set up priority tables */ - ok = allocINT(lp, &rowWT, lp->rows+1, TRUE); - createLink(lp->rows, &rowLL, NULL); - ok &= (rowLL != NULL); - if(!ok) - goto Finish; - for(i = 1; i <= lp->rows; i++) { - if(get_constr_type(lp, i)==EQ) - ii = 3; - else if(lp->upbo[i] < lp->infinite) - ii = 2; - else if(fabs(lp->rhs[i]) < lp->infinite) - ii = 1; - else - ii = 0; - rowWT[i] = ii; - if(ii > 0) - appendLink(rowLL, i); - } - ok = allocINT(lp, &colWT, lp->columns+1, TRUE); - createLink(lp->columns, &colLL, NULL); - ok &= (colLL != NULL); - if(!ok) - goto Finish; - for(i = 1; i <= lp->columns; i++) { - ix = lp->rows+i; - if(is_unbounded(lp, i)) - ii = 3; - else if(lp->upbo[ix] >= lp->infinite) - ii = 2; - else if(fabs(lp->upbo[ix]-lp->lowbo[ix]) > lp->epsmachine) - ii = 1; - else - ii = 0; - colWT[i] = ii; - if(ii > 0) - appendLink(colLL, i); - } - - /* Loop over all basis variables */ - for(i = 1; i <= lp->rows; i++) { - - /* Select row */ - rx = 0; - wx = -lp->infinite; - for(ii = firstActiveLink(rowLL); ii > 0; ii = nextActiveLink(rowLL, ii)) { - tx = rowWT[ii] - CRASH_SPACER*rowNZ[ii]; - if(tx > wx) { - rx = ii; - wx = tx; - } - } - if(rx == 0) - break; - removeLink(rowLL, rx); - - /* Select column */ - cx = 0; - wx = -lp->infinite; - for(ii = mat->row_end[rx-1]; ii < mat->row_end[rx]; ii++) { - - /* Update NZ column counts for row selected above */ - tx = fabs(ROW_MAT_VALUE(ii)); - ix = ROW_MAT_COLNR(ii); -#ifdef CRASH_SIMPLESCALE - if(tx >= CRASH_THRESHOLD * colMAX[0]) -#else - if(tx >= CRASH_THRESHOLD * colMAX[ix]) -#endif - colNZ[ix]--; - if(!isActiveLink(colLL, ix) || (tx < CRASH_THRESHOLD * rowMAX[rx])) - continue; - - /* Now do the test for best pivot */ - tx = my_sign(lp->orig_obj[ix]) - my_sign(ROW_MAT_VALUE(ii)); - tx = colWT[ix] + CRASH_WEIGHT*tx - CRASH_SPACER*colNZ[ix]; - if(tx > wx) { - cx = ix; - wx = tx; - } - } - if(cx == 0) - break; - removeLink(colLL, cx); - - /* Update row NZ counts */ - ii = mat->col_end[cx-1]; - rownr = &COL_MAT_ROWNR(ii); - value = &COL_MAT_VALUE(ii); - for(; ii < mat->col_end[cx]; - ii++, rownr += matRowColStep, value += matValueStep) { - wx = fabs(*value); - ix = *rownr; -#ifdef CRASH_SIMPLESCALE - if(wx >= CRASH_THRESHOLD * colMAX[0]) -#else - if(wx >= CRASH_THRESHOLD * rowMAX[ix]) -#endif - rowNZ[ix]--; - } - - /* Set new basis variable */ - set_basisvar(lp, rx, lp->rows+cx); - } - - /* Clean up */ -Finish: - FREE(rowNZ); - FREE(colNZ); - FREE(rowMAX); - FREE(colMAX); - FREE(rowWT); - FREE(colWT); - freeLink(&rowLL); - freeLink(&colLL); - } - - /* Construct a basis that is in some measure the "least degenerate" */ - else if((lp->crashmode == CRASH_LEASTDEGENERATE) && mat_validate(mat)) { - /* The logic here follows Maros */ - LLrec *rowLL = NULL, *colLL = NULL; - int ii, rx, cx, ix, nz, *merit = NULL; - REAL *value, wx, hold, *rhs = NULL, *eta = NULL; - int *rownr, *colnr; - - report(lp, NORMAL, "crash_basis: 'Least degenerate' basis crashing selected\n"); - - /* Create temporary arrays */ - ok = allocINT(lp, &merit, lp->columns + 1, FALSE) && - allocREAL(lp, &eta, lp->rows + 1, FALSE) && - allocREAL(lp, &rhs, lp->rows + 1, FALSE); - createLink(lp->columns, &colLL, NULL); - createLink(lp->rows, &rowLL, NULL); - ok &= (colLL != NULL) && (rowLL != NULL); - if(!ok) - goto FinishLD; - MEMCOPY(rhs, lp->orig_rhs, lp->rows + 1); - for(i = 1; i <= lp->columns; i++) - appendLink(colLL, i); - for(i = 1; i <= lp->rows; i++) - appendLink(rowLL, i); - - /* Loop until we have found enough new bases */ - while(colLL->count > 0) { - - /* Tally non-zeros matching in RHS and each active column */ - nz = mat_nonzeros(mat); - rownr = &COL_MAT_ROWNR(0); - colnr = &COL_MAT_COLNR(0); - ii = 0; - MEMCLEAR(merit, lp->columns + 1); - for(i = 0; i < nz; - i++, rownr += matRowColStep, colnr += matRowColStep) { - rx = *rownr; - cx = *colnr; - if(isActiveLink(colLL, cx) && (rhs[rx] != 0)) { - merit[cx]++; - ii++; - } - } - if(ii == 0) - break; - - /* Find maximal match; break ties with column length */ - i = firstActiveLink(colLL); - cx = i; - for(i = nextActiveLink(colLL, i); i != 0; i = nextActiveLink(colLL, i)) { - if(merit[i] >= merit[cx]) { - if((merit[i] > merit[cx]) || (mat_collength(mat, i) > mat_collength(mat, cx))) - cx = i; - } - } - - /* Determine the best pivot row */ - i = mat->col_end[cx-1]; - nz = mat->col_end[cx]; - rownr = &COL_MAT_ROWNR(i); - value = &COL_MAT_VALUE(i); - rx = 0; - wx = 0; - MEMCLEAR(eta, lp->rows + 1); - for(; i < nz; - i++, rownr += matRowColStep, value += matValueStep) { - ix = *rownr; - hold = *value; - eta[ix] = rhs[ix] / hold; - hold = fabs(hold); - if(isActiveLink(rowLL, ix) && (hold > wx)) { - wx = hold; - rx = ix; - } - } - - /* Set new basis variable */ - if(rx > 0) { - - /* We have to update the rhs vector for the implied transformation - in order to be able to find the new RHS non-zero pattern */ - for(i = 1; i <= lp->rows; i++) - rhs[i] -= wx * eta[i]; - rhs[rx] = wx; - - /* Do the exchange */ - set_basisvar(lp, rx, lp->rows+cx); - removeLink(rowLL, rx); - } - removeLink(colLL, cx); - - } - - /* Clean up */ -FinishLD: - FREE(merit); - FREE(rhs); - freeLink(&rowLL); - freeLink(&colLL); - - } - return( ok ); -} - -#if 0 -MYBOOL __WINAPI guess_basis(lprec *lp, REAL *guessvector, int *basisvector) -{ - MYBOOL status = FALSE; - REAL *values = NULL, *violation = NULL, - *value, error, upB, loB, sortorder = 1.0; - int i, n, *rownr, *colnr; - MATrec *mat = lp->matA; - - if(!mat_validate(lp->matA)) - return( status ); - - /* Create helper arrays */ - if(!allocREAL(lp, &values, lp->sum+1, TRUE) || - !allocREAL(lp, &violation, lp->sum+1, TRUE)) - goto Finish; - - /* Compute values of slack variables for given guess vector */ - i = 0; - n = get_nonzeros(lp); - rownr = &COL_MAT_ROWNR(i); - colnr = &COL_MAT_COLNR(i); - value = &COL_MAT_VALUE(i); - for(; i < n; i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) - values[*rownr] += unscaled_mat(lp, my_chsign(is_chsign(lp, *rownr), *value), *rownr, *colnr) * - guessvector[*colnr]; - MEMMOVE(values+lp->rows+1, guessvector+1, lp->columns); - - /* Initialize constraint bound violation measures */ - for(i = 1; i <= lp->rows; i++) { - upB = get_rh_upper(lp, i); - loB = get_rh_lower(lp, i); - error = values[i] - upB; - if(error > lp->epsprimal) - violation[i] = sortorder*error; - else { - error = loB - values[i]; - if(error > lp->epsprimal) - violation[i] = sortorder*error; - else if(is_infinite(lp, loB) && is_infinite(lp, upB)) - ; - else if(is_infinite(lp, upB)) - violation[i] = sortorder*(loB - values[i]); - else if(is_infinite(lp, loB)) - violation[i] = sortorder*(values[i] - upB); - else - violation[i] = - sortorder*MAX(upB - values[i], values[i] - loB); - } - basisvector[i] = i; - } - - /* Initialize user variable bound violation measures */ - for(i = 1; i <= lp->columns; i++) { - n = lp->rows+i; - upB = get_upbo(lp, i); - loB = get_lowbo(lp, i); - error = guessvector[i] - upB; - if(error > lp->epsprimal) - violation[n] = sortorder*error; - else { - error = loB - values[n]; - if(error > lp->epsprimal) - violation[n] = sortorder*error; - else if(is_infinite(lp, loB) && is_infinite(lp, upB)) - ; - else if(is_infinite(lp, upB)) - violation[n] = sortorder*(loB - values[n]); - else if(is_infinite(lp, loB)) - violation[n] = sortorder*(values[n] - upB); - else - violation[n] = - sortorder*MAX(upB - values[n], values[n] - loB); - } - basisvector[n] = n; - } - - /* Sort decending by violation; this means that variables with - the largest violations will be designated as basic */ - sortByREAL(basisvector, violation, lp->sum, 1, FALSE); - - /* Adjust the non-basic indeces for the (proximal) bound state */ - error = lp->epsprimal; - for(i = lp->rows+1, rownr = basisvector+i; i <= lp->sum; i++, rownr++) { - if(*rownr <= lp->rows) { - if(values[*rownr] <= get_rh_lower(lp, *rownr)+error) - *rownr = -(*rownr); - } - else - if(values[i] <= get_lowbo(lp, (*rownr)-lp->rows)+error) - *rownr = -(*rownr); - } - - /* Clean up and return status */ - status = (MYBOOL) (violation[1] == 0); -Finish: - FREE(values); - FREE(violation); - - - return( status ); -} -#endif - -#if 0 -MYBOOL __WINAPI guess_basis(lprec *lp, REAL *guessvector, int *basisvector) -{ - MYBOOL *isnz, status = FALSE; - REAL *values = NULL, *violation = NULL, - eps = lp->epsprimal, - *value, error, upB, loB, sortorder = 1.0; - int i, j, n, *rownr, *colnr, *slkpos, - nrows = lp->rows, ncols = lp->columns; - MATrec *mat = lp->matA; - - if(!mat_validate(mat)) - return( status ); - - /* Create helper arrays */ - if(!allocREAL(lp, &values, lp->sum+1, TRUE) || - !allocREAL(lp, &violation, lp->sum+1, TRUE)) - goto Finish; - - /* Compute values of slack variables for given guess vector */ - i = 0; - n = get_nonzeros(lp); - rownr = &COL_MAT_ROWNR(i); - colnr = &COL_MAT_COLNR(i); - value = &COL_MAT_VALUE(i); - for(; i < n; i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) - values[*rownr] += unscaled_mat(lp, my_chsign(is_chsign(lp, *rownr), *value), *rownr, *colnr) * - guessvector[*colnr]; - MEMMOVE(values+nrows+1, guessvector+1, ncols); - - /* Initialize constraint bound violation measures (expressed as positive values) */ - for(i = 1; i <= nrows; i++) { - upB = get_rh_upper(lp, i); - loB = get_rh_lower(lp, i); - error = values[i] - upB; - if(error > eps) - violation[i] = sortorder*error; - else { - error = loB - values[i]; - if(error > eps) - violation[i] = sortorder*error; - else if(my_infinite(lp, loB) && my_infinite(lp, upB)) - ; - else if(my_infinite(lp, upB)) - violation[i] = sortorder*(loB - values[i]); - else if(my_infinite(lp, loB)) - violation[i] = sortorder*(values[i] - upB); - else - violation[i] = -sortorder*MAX(upB - values[i], values[i] - loB); - } - basisvector[i] = i; - } - - /* Initialize user variable bound violation measures (expressed as positive values) */ - for(i = 1; i <= ncols; i++) { - n = nrows+i; - upB = get_upbo(lp, i); - loB = get_lowbo(lp, i); - error = guessvector[i] - upB; - if(error > eps) - violation[n] = sortorder*error; - else { - error = loB - values[n]; - if(error > eps) - violation[n] = sortorder*error; - else if(my_infinite(lp, loB) && my_infinite(lp, upB)) - ; - else if(my_infinite(lp, upB)) - violation[n] = sortorder*(loB - values[n]); - else if(my_infinite(lp, loB)) - violation[n] = sortorder*(values[n] - upB); - else - violation[n] = -sortorder*MAX(upB - values[n], values[n] - loB); - } - basisvector[n] = n; - } - - /* Sort decending by violation; this means that variables with - the largest violations will be designated as basic */ - sortByREAL(basisvector, violation, lp->sum, 1, FALSE); - error = violation[1]; - - /* Adjust the non-basic indeces for the (proximal) bound state */ - for(i = nrows+1, rownr = basisvector+i; i <= lp->sum; i++, rownr++) { - if(*rownr <= nrows) { - if(values[*rownr] <= get_rh_lower(lp, *rownr)+eps) - *rownr = -(*rownr); - } - else - if(values[i] <= get_lowbo(lp, (*rownr)-nrows)+eps) - *rownr = -(*rownr); - } - -#if 1 - /* Let us check for obvious row singularities and try to fix these; - First assemble necessary basis statistics... */ - isnz = (MYBOOL *) values; - MEMCLEAR(isnz, nrows+1); - slkpos = (int *) violation; - MEMCLEAR(slkpos, nrows+1); - for(i = 1; i <= nrows; i++) { - j = abs(basisvector[i]); - if(j <= nrows) { - isnz[j] = TRUE; - slkpos[j] = i; - } - else { - j-= nrows; - j = mat->col_end[j-1]; - isnz[COL_MAT_ROWNR(j)] = TRUE; - /*isnz[COL_MAT_ROWNR(j+1)] = TRUE;*/ - } - } - for(; i <= lp->sum; i++) { - j = abs(basisvector[i]); - if(j <= nrows) - slkpos[j] = i; - } - - /* ...then set the corresponding slacks basic for row rank deficient positions */ - for(j = 1; j <= nrows; j++) { -#ifdef Paranoia - if(slkpos[j] == 0) - report(lp, SEVERE, "guess_basis: Internal error"); -#endif - if(!isnz[j]) { - isnz[j] = TRUE; - i = slkpos[j]; - swapINT(&basisvector[i], &basisvector[j]); - basisvector[j] = abs(basisvector[j]); - } - } -#endif - - /* Clean up and return status */ - status = (MYBOOL) (error <= eps); -Finish: - FREE(values); - FREE(violation); - - return( status ); -} -#endif - -#if 0 -MYBOOL __WINAPI guess_basis(lprec *lp, REAL *guessvector, int *basisvector) -{ - MYBOOL *isnz, status = FALSE; - REAL *values = NULL, *violation = NULL, - eps = lp->epsprimal, - *value, error, upB, loB, sortorder = 1.0; - int i, j, jj, n, *rownr, *colnr, *slkpos, - nrows = lp->rows, ncols = lp->columns; - MATrec *mat = lp->matA; - - if(!mat_validate(mat)) - return( status ); - - /* Create helper arrays */ - if(!allocREAL(lp, &values, lp->sum+1, TRUE) || - !allocREAL(lp, &violation, lp->sum+1, TRUE)) - goto Finish; - - /* Compute values of slack variables for given guess vector */ - i = 0; - n = get_nonzeros(lp); - rownr = &COL_MAT_ROWNR(i); - colnr = &COL_MAT_COLNR(i); - value = &COL_MAT_VALUE(i); - for(; i < n; i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) - values[*rownr] += unscaled_mat(lp, my_chsign(is_chsign(lp, *rownr), *value), *rownr, *colnr) * - guessvector[*colnr]; - MEMMOVE(values+nrows+1, guessvector+1, ncols); - - /* Initialize constraint bound violation measures (expressed as positive values) */ - for(i = 1; i <= nrows; i++) { - upB = get_rh_upper(lp, i); - loB = get_rh_lower(lp, i); - error = values[i] - upB; - if(error > -eps) - violation[i] = sortorder*MAX(0,error); - else { - error = loB - values[i]; - if(error > -eps) - violation[i] = sortorder*MAX(0,error); - else if(my_infinite(lp, loB) && my_infinite(lp, upB)) - ; - else if(my_infinite(lp, upB)) - violation[i] = sortorder*(loB - values[i]); - else if(my_infinite(lp, loB)) - violation[i] = sortorder*(values[i] - upB); - else - violation[i] = -sortorder*MAX(upB - values[i], values[i] - loB); - } - basisvector[i] = i; - } - - /* Initialize user variable bound violation measures (expressed as positive values) */ - for(i = 1; i <= ncols; i++) { - n = nrows+i; - upB = get_upbo(lp, i); - loB = get_lowbo(lp, i); - error = guessvector[i] - upB; - if(error > -eps) - violation[n] = sortorder*MAX(0,error); - else { - error = loB - values[n]; - if(error > -eps) - violation[n] = sortorder*MAX(0,error); - else if(my_infinite(lp, loB) && my_infinite(lp, upB)) - ; - else if(my_infinite(lp, upB)) - violation[n] = sortorder*(loB - values[n]); - else if(my_infinite(lp, loB)) - violation[n] = sortorder*(values[n] - upB); - else - violation[n] = -sortorder*MAX(upB - values[n], values[n] - loB); - } - basisvector[n] = n; - } - - /* Sort decending by violation; this means that variables with - the largest violations will be designated as basic */ - sortByREAL(basisvector, violation, lp->sum, 1, FALSE); - error = violation[1]; - - /* Adjust the non-basic indeces for the (proximal) bound state */ - for(i = nrows+1, rownr = basisvector+i; i <= lp->sum; i++, rownr++) { - if(*rownr <= nrows) { - values[*rownr] -= lp->orig_rhs[*rownr]; - if(values[*rownr] <= eps) - *rownr = -(*rownr); - } - else - if(values[i] <= get_lowbo(lp, (*rownr)-nrows)+eps) - *rownr = -(*rownr); - } - - /* Let us check for obvious row singularities and try to fix these; - First assemble necessary basis statistics... */ - isnz = (MYBOOL *) values; - MEMCLEAR(isnz, nrows+1); - slkpos = (int *) violation; - MEMCLEAR(slkpos, nrows+1); - for(i = 1; i <= nrows; i++) { - j = abs(basisvector[i]); - if(j <= nrows) { - isnz[j] = TRUE; - slkpos[j] = i; - } - else { - j-= nrows; - jj = mat->col_end[j-1]; - isnz[COL_MAT_ROWNR(jj)] = TRUE; -/* if(++jj < mat->col_end[j]) - isnz[COL_MAT_ROWNR(jj)] = TRUE; */ - } - } - for(; i <= lp->sum; i++) { - j = abs(basisvector[i]); - if(j <= nrows) - slkpos[j] = i; - } - - /* ...then set the corresponding slacks basic for row rank deficient positions */ - for(j = 1; j <= nrows; j++) { -#ifdef Paranoia - if(slkpos[j] == 0) - report(lp, SEVERE, "guess_basis: Internal error"); -#endif - if(!isnz[j]) { - isnz[j] = TRUE; - i = slkpos[j]; - swapINT(&basisvector[i], &basisvector[j]); - basisvector[j] = abs(basisvector[j]); - } - } - - /* Lastly normalize all basic variables to be coded as lower-bounded */ - for(i = 1; i <= nrows; i++) - basisvector[i] = -abs(basisvector[i]); - - /* Clean up and return status */ - status = (MYBOOL) (error <= eps); -Finish: - FREE(values); - FREE(violation); - - return( status ); -} -#endif - -MYBOOL __WINAPI guess_basis(lprec *lp, REAL *guessvector, int *basisvector) -{ - MYBOOL *isnz = NULL, status = FALSE; - REAL *values = NULL, *violation = NULL, - eps = lp->epsprimal, - *value, error, upB, loB, sortorder = -1.0; - int i, j, jj, n, *rownr, *colnr, *slkpos = NULL, - nrows = lp->rows, ncols = lp->columns, nsum = lp->sum; - int *basisnr; - MATrec *mat = lp->matA; - - if(!mat_validate(mat)) - return( status ); - - /* Create helper arrays, providing for multiple use of the violation array */ - if(!allocREAL(lp, &values, nsum+1, TRUE) || - !allocREAL(lp, &violation, nsum+1, TRUE)) - goto Finish; - - /* Compute the values of the constraints for the given guess vector */ - i = 0; - n = get_nonzeros(lp); - rownr = &COL_MAT_ROWNR(i); - colnr = &COL_MAT_COLNR(i); - value = &COL_MAT_VALUE(i); - for(; i < n; i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) - values[*rownr] += unscaled_mat(lp, my_chsign(is_chsign(lp, *rownr), *value), *rownr, *colnr) * - guessvector[*colnr]; - MEMMOVE(values+nrows+1, guessvector+1, ncols); - - /* Initialize bound "violation" or primal non-degeneracy measures, expressed - as the absolute value of the differences from the closest bound. */ - for(i = 1; i <= nsum; i++) { - if(i <= nrows) { - loB = get_rh_lower(lp, i); - upB = get_rh_upper(lp, i); - } - else { - loB = get_lowbo(lp, i-nrows); - upB = get_upbo(lp, i-nrows); - } - - /* Free constraints/variables */ - if(my_infinite(lp, loB) && my_infinite(lp, upB)) - error = 0; - /* Violated constraints/variable bounds */ - else if(values[i]+eps < loB) - error = loB-values[i]; - else if(values[i]-eps > upB) - error = values[i]-upB; - /* Non-violated constraints/variables bounds */ - else if(my_infinite(lp, upB)) - error = MAX(0, values[i]-loB); - else if(my_infinite(lp, loB)) - error = MAX(0, upB-values[i]); - else - error = MIN(upB-values[i], values[i]-loB); /* MAX(upB-values[i], values[i]-loB); */ - if(error != 0) - violation[i] = sortorder*error; - basisvector[i] = i; - } - - /* Sort decending , meaning that variables with the largest - "violations" will be designated basic. Effectively, we are performing a - greedy type algorithm, but start at the "least interesting" end. */ - sortByREAL(basisvector, violation, nsum, 1, FALSE); - error = violation[1]; /* Used for setting the return value */ - - /* Let us check for obvious row singularities and try to fix these. - Note that we reuse the memory allocated to the violation array. - First assemble necessary basis statistics... */ - slkpos = (int *) violation; - n = nrows+1; - MEMCLEAR(slkpos, n); - isnz = (MYBOOL *) (slkpos+n+1); - MEMCLEAR(isnz, n); - for(i = 1; i <= nrows; i++) { - j = abs(basisvector[i]); - if(j <= nrows) { - isnz[j] = TRUE; - slkpos[j] = i; - } - else { - j-= nrows; - jj = mat->col_end[j-1]; - jj = COL_MAT_ROWNR(jj); - isnz[jj] = TRUE; - } - } - for(; i <= nsum; i++) { - j = abs(basisvector[i]); - if(j <= nrows) - slkpos[j] = i; - } - - /* ...then set the corresponding slacks basic for row rank deficient positions */ - for(j = 1; j <= nrows; j++) { - if(slkpos[j] == 0) - report(lp, SEVERE, "guess_basis: Internal error"); - if(!isnz[j]) { - isnz[j] = TRUE; - i = slkpos[j]; - swapINT(&basisvector[i], &basisvector[j]); - basisvector[j] = abs(basisvector[j]); - } - } - - /* Adjust the non-basic indeces for the (proximal) bound state */ - for(i = nrows+1, basisnr = basisvector+i; i <= nsum; i++, basisnr++) { - n = *basisnr; - if(n <= nrows) { - values[n] -= get_rh_lower(lp, n); - if(values[n] <= eps) - *basisnr = -(*basisnr); - } - else - if(values[n]-eps <= get_lowbo(lp, n-nrows)) - *basisnr = -(*basisnr); - } - -/* Lastly normalize all basic variables to be coded as lower-bounded, - or effectively zero-based in the case of free variables. */ - for(i = 1; i <= nrows; i++) - basisvector[i] = -abs(basisvector[i]); - - /* Clean up and return status */ - status = (MYBOOL) (error <= eps); -Finish: - FREE(values); - FREE(violation); - - return( status ); -} diff --git a/code/3rd_lpsolve/lp_crash.h b/code/3rd_lpsolve/lp_crash.h deleted file mode 100644 index eb1f84de..00000000 --- a/code/3rd_lpsolve/lp_crash.h +++ /dev/null @@ -1,27 +0,0 @@ - -#ifndef HEADER_lp_crash -#define HEADER_lp_crash - - -#include "lp_types.h" - -#define CRASH_SIMPLESCALE /* Specify if we should use a simple absolute scaling threshold */ - -#define CRASH_THRESHOLD 0.167 -#define CRASH_SPACER 10 -#define CRASH_WEIGHT 0.500 - - - -#ifdef __cplusplus -__EXTERN_C { -#endif - -STATIC MYBOOL crash_basis(lprec *lp); - -#ifdef __cplusplus -} -#endif - -#endif /* HEADER_lp_crash */ - diff --git a/code/3rd_lpsolve/lp_explicit.h b/code/3rd_lpsolve/lp_explicit.h deleted file mode 100644 index 35b7c24d..00000000 --- a/code/3rd_lpsolve/lp_explicit.h +++ /dev/null @@ -1,1062 +0,0 @@ -#define HEADER_lp_report - -#include "lp_lib.h" - -#if defined LPSOLVEAPIFROMLPRECDEF -# define LPSOLVEAPIFROMLPREC -# define LPSOLVEAPIDEF -#endif - -#if defined LPSOLVEAPIFROMLIBDEF -# define LPSOLVEAPIFROMLIB -# define LPSOLVEAPIDEF -#endif - -#if !defined LPSOLVEAPIDEF -# define LPSOLVEAPIDEF extern -#endif - -/* entries for lp structure */ -LPSOLVEAPIDEF add_column_func *_add_column; -LPSOLVEAPIDEF add_columnex_func *_add_columnex; -LPSOLVEAPIDEF add_constraint_func *_add_constraint; -LPSOLVEAPIDEF add_constraintex_func *_add_constraintex; -LPSOLVEAPIDEF add_lag_con_func *_add_lag_con; -LPSOLVEAPIDEF add_SOS_func *_add_SOS; -LPSOLVEAPIDEF column_in_lp_func *_column_in_lp; -LPSOLVEAPIDEF copy_lp_func *_copy_lp; -LPSOLVEAPIDEF default_basis_func *_default_basis; -LPSOLVEAPIDEF del_column_func *_del_column; -LPSOLVEAPIDEF del_constraint_func *_del_constraint; -LPSOLVEAPIDEF delete_lp_func *_delete_lp; -LPSOLVEAPIDEF dualize_lp_func *_dualize_lp; -LPSOLVEAPIDEF free_lp_func *_free_lp; -/*LPSOLVEAPIDEF get_accuracy_func *_get_accuracy;*/ -LPSOLVEAPIDEF get_anti_degen_func *_get_anti_degen; -LPSOLVEAPIDEF get_basis_func *_get_basis; -LPSOLVEAPIDEF get_basiscrash_func *_get_basiscrash; -LPSOLVEAPIDEF get_bb_depthlimit_func *_get_bb_depthlimit; -LPSOLVEAPIDEF get_bb_floorfirst_func *_get_bb_floorfirst; -LPSOLVEAPIDEF get_bb_rule_func *_get_bb_rule; -LPSOLVEAPIDEF get_bounds_tighter_func *_get_bounds_tighter; -LPSOLVEAPIDEF get_break_at_value_func *_get_break_at_value; -/*LPSOLVEAPIDEF get_break_numeric_accuracy_func *_get_break_numeric_accuracy;*/ -LPSOLVEAPIDEF get_col_name_func *_get_col_name; -LPSOLVEAPIDEF get_column_func *_get_column; -LPSOLVEAPIDEF get_columnex_func *_get_columnex; -LPSOLVEAPIDEF get_constr_type_func *_get_constr_type; -LPSOLVEAPIDEF get_constr_value_func *_get_constr_value; -LPSOLVEAPIDEF get_constraints_func *_get_constraints; -LPSOLVEAPIDEF get_dual_solution_func *_get_dual_solution; -LPSOLVEAPIDEF get_epsb_func *_get_epsb; -LPSOLVEAPIDEF get_epsd_func *_get_epsd; -LPSOLVEAPIDEF get_epsel_func *_get_epsel; -LPSOLVEAPIDEF get_epsint_func *_get_epsint; -LPSOLVEAPIDEF get_epsperturb_func *_get_epsperturb; -LPSOLVEAPIDEF get_epspivot_func *_get_epspivot; -LPSOLVEAPIDEF get_improve_func *_get_improve; -LPSOLVEAPIDEF get_infinite_func *_get_infinite; -LPSOLVEAPIDEF get_lambda_func *_get_lambda; -LPSOLVEAPIDEF get_lowbo_func *_get_lowbo; -LPSOLVEAPIDEF get_lp_index_func *_get_lp_index; -LPSOLVEAPIDEF get_lp_name_func *_get_lp_name; -LPSOLVEAPIDEF get_Lrows_func *_get_Lrows; -LPSOLVEAPIDEF get_mat_func *_get_mat; -LPSOLVEAPIDEF get_mat_byindex_func *_get_mat_byindex; -LPSOLVEAPIDEF get_max_level_func *_get_max_level; -LPSOLVEAPIDEF get_maxpivot_func *_get_maxpivot; -LPSOLVEAPIDEF get_mip_gap_func *_get_mip_gap; -LPSOLVEAPIDEF get_multiprice_func *_get_multiprice; -LPSOLVEAPIDEF get_nameindex_func *_get_nameindex; -LPSOLVEAPIDEF get_Ncolumns_func *_get_Ncolumns; -LPSOLVEAPIDEF get_negrange_func *_get_negrange; -LPSOLVEAPIDEF get_nz_func *_get_nonzeros; -LPSOLVEAPIDEF get_Norig_columns_func *_get_Norig_columns; -LPSOLVEAPIDEF get_Norig_rows_func *_get_Norig_rows; -LPSOLVEAPIDEF get_Nrows_func *_get_Nrows; -LPSOLVEAPIDEF get_obj_bound_func *_get_obj_bound; -LPSOLVEAPIDEF get_objective_func *_get_objective; -LPSOLVEAPIDEF get_orig_index_func *_get_orig_index; -LPSOLVEAPIDEF get_origcol_name_func *_get_origcol_name; -LPSOLVEAPIDEF get_origrow_name_func *_get_origrow_name; -LPSOLVEAPIDEF get_partialprice_func *_get_partialprice; -LPSOLVEAPIDEF get_pivoting_func *_get_pivoting; -LPSOLVEAPIDEF get_presolve_func *_get_presolve; -LPSOLVEAPIDEF get_presolveloops_func *_get_presolveloops; -LPSOLVEAPIDEF get_primal_solution_func *_get_primal_solution; -LPSOLVEAPIDEF get_print_sol_func *_get_print_sol; -LPSOLVEAPIDEF get_pseudocosts_func *_get_pseudocosts; -LPSOLVEAPIDEF get_ptr_constraints_func *_get_ptr_constraints; -LPSOLVEAPIDEF get_ptr_dual_solution_func *_get_ptr_dual_solution; -LPSOLVEAPIDEF get_ptr_lambda_func *_get_ptr_lambda; -LPSOLVEAPIDEF get_ptr_primal_solution_func *_get_ptr_primal_solution; -LPSOLVEAPIDEF get_ptr_sensitivity_obj_func *_get_ptr_sensitivity_obj; -LPSOLVEAPIDEF get_ptr_sensitivity_objex_func *_get_ptr_sensitivity_objex; -LPSOLVEAPIDEF get_ptr_sensitivity_rhs_func *_get_ptr_sensitivity_rhs; -LPSOLVEAPIDEF get_ptr_variables_func *_get_ptr_variables; -LPSOLVEAPIDEF get_rh_func *_get_rh; -LPSOLVEAPIDEF get_rh_range_func *_get_rh_range; -LPSOLVEAPIDEF get_row_func *_get_row; -LPSOLVEAPIDEF get_rowex_func *_get_rowex; -LPSOLVEAPIDEF get_row_name_func *_get_row_name; -LPSOLVEAPIDEF get_scalelimit_func *_get_scalelimit; -LPSOLVEAPIDEF get_scaling_func *_get_scaling; -LPSOLVEAPIDEF get_sensitivity_obj_func *_get_sensitivity_obj; -LPSOLVEAPIDEF get_sensitivity_objex_func *_get_sensitivity_objex; -LPSOLVEAPIDEF get_sensitivity_rhs_func *_get_sensitivity_rhs; -LPSOLVEAPIDEF get_simplextype_func *_get_simplextype; -LPSOLVEAPIDEF get_solutioncount_func *_get_solutioncount; -LPSOLVEAPIDEF get_solutionlimit_func *_get_solutionlimit; -LPSOLVEAPIDEF get_status_func *_get_status; -LPSOLVEAPIDEF get_statustext_func *_get_statustext; -LPSOLVEAPIDEF get_timeout_func *_get_timeout; -LPSOLVEAPIDEF get_total_iter_func *_get_total_iter; -LPSOLVEAPIDEF get_total_nodes_func *_get_total_nodes; -LPSOLVEAPIDEF get_upbo_func *_get_upbo; -LPSOLVEAPIDEF get_var_branch_func *_get_var_branch; -LPSOLVEAPIDEF get_var_dualresult_func *_get_var_dualresult; -LPSOLVEAPIDEF get_var_primalresult_func *_get_var_primalresult; -LPSOLVEAPIDEF get_var_priority_func *_get_var_priority; -LPSOLVEAPIDEF get_variables_func *_get_variables; -LPSOLVEAPIDEF get_verbose_func *_get_verbose; -LPSOLVEAPIDEF get_working_objective_func *_get_working_objective; -LPSOLVEAPIDEF guess_basis_func *_guess_basis; -LPSOLVEAPIDEF has_BFP_func *_has_BFP; -LPSOLVEAPIDEF has_XLI_func *_has_XLI; -LPSOLVEAPIDEF is_add_rowmode_func *_is_add_rowmode; -LPSOLVEAPIDEF is_anti_degen_func *_is_anti_degen; -LPSOLVEAPIDEF is_binary_func *_is_binary; -LPSOLVEAPIDEF is_break_at_first_func *_is_break_at_first; -LPSOLVEAPIDEF is_constr_type_func *_is_constr_type; -LPSOLVEAPIDEF is_debug_func *_is_debug; -LPSOLVEAPIDEF is_feasible_func *_is_feasible; -LPSOLVEAPIDEF is_unbounded_func *_is_unbounded; -LPSOLVEAPIDEF is_infinite_func *_is_infinite; -LPSOLVEAPIDEF is_int_func *_is_int; -LPSOLVEAPIDEF is_integerscaling_func *_is_integerscaling; -LPSOLVEAPIDEF is_lag_trace_func *_is_lag_trace; -LPSOLVEAPIDEF is_maxim_func *_is_maxim; -LPSOLVEAPIDEF is_nativeBFP_func *_is_nativeBFP; -LPSOLVEAPIDEF is_nativeXLI_func *_is_nativeXLI; -LPSOLVEAPIDEF is_negative_func *_is_negative; -LPSOLVEAPIDEF is_piv_mode_func *_is_piv_mode; -LPSOLVEAPIDEF is_piv_rule_func *_is_piv_rule; -LPSOLVEAPIDEF is_presolve_func *_is_presolve; -LPSOLVEAPIDEF is_scalemode_func *_is_scalemode; -LPSOLVEAPIDEF is_scaletype_func *_is_scaletype; -LPSOLVEAPIDEF is_semicont_func *_is_semicont; -LPSOLVEAPIDEF is_SOS_var_func *_is_SOS_var; -LPSOLVEAPIDEF is_trace_func *_is_trace; -LPSOLVEAPIDEF is_use_names_func *_is_use_names; -LPSOLVEAPIDEF lp_solve_version_func *_lp_solve_version; -LPSOLVEAPIDEF make_lp_func *_make_lp; -LPSOLVEAPIDEF print_constraints_func *_print_constraints; -LPSOLVEAPIDEF print_debugdump_func *_print_debugdump; -LPSOLVEAPIDEF print_duals_func *_print_duals; -LPSOLVEAPIDEF print_lp_func *_print_lp; -LPSOLVEAPIDEF print_objective_func *_print_objective; -LPSOLVEAPIDEF print_scales_func *_print_scales; -LPSOLVEAPIDEF print_solution_func *_print_solution; -LPSOLVEAPIDEF print_str_func *_print_str; -LPSOLVEAPIDEF print_tableau_func *_print_tableau; -LPSOLVEAPIDEF put_abortfunc_func *_put_abortfunc; -LPSOLVEAPIDEF put_bb_nodefunc_func *_put_bb_nodefunc; -LPSOLVEAPIDEF put_bb_branchfunc_func *_put_bb_branchfunc; -LPSOLVEAPIDEF put_logfunc_func *_put_logfunc; -LPSOLVEAPIDEF put_msgfunc_func *_put_msgfunc; -LPSOLVEAPIDEF read_LP_func *_read_LP; -LPSOLVEAPIDEF read_MPS_func *_read_MPS; -LPSOLVEAPIDEF read_XLI_func *_read_XLI; -LPSOLVEAPIDEF read_params_func *_read_params; -LPSOLVEAPIDEF read_basis_func *_read_basis; -LPSOLVEAPIDEF reset_basis_func *_reset_basis; -LPSOLVEAPIDEF reset_params_func *_reset_params; -LPSOLVEAPIDEF reportfunc *_report; -LPSOLVEAPIDEF resize_lp_func *_resize_lp; -LPSOLVEAPIDEF set_add_rowmode_func *_set_add_rowmode; -LPSOLVEAPIDEF set_anti_degen_func *_set_anti_degen; -LPSOLVEAPIDEF set_basisvar_func *_set_basisvar; -LPSOLVEAPIDEF set_basis_func *_set_basis; -LPSOLVEAPIDEF set_basiscrash_func *_set_basiscrash; -LPSOLVEAPIDEF set_bb_depthlimit_func *_set_bb_depthlimit; -LPSOLVEAPIDEF set_bb_floorfirst_func *_set_bb_floorfirst; -LPSOLVEAPIDEF set_bb_rule_func *_set_bb_rule; -LPSOLVEAPIDEF set_BFP_func *_set_BFP; -LPSOLVEAPIDEF set_binary_func *_set_binary; -LPSOLVEAPIDEF set_bounds_func *_set_bounds; -LPSOLVEAPIDEF set_bounds_tighter_func *_set_bounds_tighter; -LPSOLVEAPIDEF set_break_at_first_func *_set_break_at_first; -LPSOLVEAPIDEF set_break_at_value_func *_set_break_at_value; -/*LPSOLVEAPIDEF set_break_numeric_accuracy_func *_set_break_numeric_accuracy;*/ -LPSOLVEAPIDEF set_column_func *_set_column; -LPSOLVEAPIDEF set_columnex_func *_set_columnex; -LPSOLVEAPIDEF set_col_name_func *_set_col_name; -LPSOLVEAPIDEF set_constr_type_func *_set_constr_type; -LPSOLVEAPIDEF set_debug_func *_set_debug; -LPSOLVEAPIDEF set_epsb_func *_set_epsb; -LPSOLVEAPIDEF set_epsd_func *_set_epsd; -LPSOLVEAPIDEF set_epsel_func *_set_epsel; -LPSOLVEAPIDEF set_epsint_func *_set_epsint; -LPSOLVEAPIDEF set_epslevel_func *_set_epslevel; -LPSOLVEAPIDEF set_epsperturb_func *_set_epsperturb; -LPSOLVEAPIDEF set_epspivot_func *_set_epspivot; -LPSOLVEAPIDEF set_unbounded_func *_set_unbounded; -LPSOLVEAPIDEF set_improve_func *_set_improve; -LPSOLVEAPIDEF set_infinite_func *_set_infinite; -LPSOLVEAPIDEF set_int_func *_set_int; -LPSOLVEAPIDEF set_lag_trace_func *_set_lag_trace; -LPSOLVEAPIDEF set_lowbo_func *_set_lowbo; -LPSOLVEAPIDEF set_lp_name_func *_set_lp_name; -LPSOLVEAPIDEF set_mat_func *_set_mat; -LPSOLVEAPIDEF set_maxim_func *_set_maxim; -LPSOLVEAPIDEF set_maxpivot_func *_set_maxpivot; -LPSOLVEAPIDEF set_minim_func *_set_minim; -LPSOLVEAPIDEF set_mip_gap_func *_set_mip_gap; -LPSOLVEAPIDEF set_multiprice_func *_set_multiprice; -LPSOLVEAPIDEF set_negrange_func *_set_negrange; -LPSOLVEAPIDEF set_obj_bound_func *_set_obj_bound; -LPSOLVEAPIDEF set_obj_fn_func *_set_obj_fn; -LPSOLVEAPIDEF set_obj_fnex_func *_set_obj_fnex; -LPSOLVEAPIDEF set_obj_func *_set_obj; -LPSOLVEAPIDEF set_outputfile_func *_set_outputfile; -LPSOLVEAPIDEF set_outputstream_func *_set_outputstream; -LPSOLVEAPIDEF set_partialprice_func *_set_partialprice; -LPSOLVEAPIDEF set_pivoting_func *_set_pivoting; -LPSOLVEAPIDEF set_preferdual_func *_set_preferdual; -LPSOLVEAPIDEF set_presolve_func *_set_presolve; -LPSOLVEAPIDEF set_print_sol_func *_set_print_sol; -LPSOLVEAPIDEF set_pseudocosts_func *_set_pseudocosts; -LPSOLVEAPIDEF set_rh_func *_set_rh; -LPSOLVEAPIDEF set_rh_range_func *_set_rh_range; -LPSOLVEAPIDEF set_rh_vec_func *_set_rh_vec; -LPSOLVEAPIDEF set_row_func *_set_row; -LPSOLVEAPIDEF set_rowex_func *_set_rowex; -LPSOLVEAPIDEF set_row_name_func *_set_row_name; -LPSOLVEAPIDEF set_scalelimit_func *_set_scalelimit; -LPSOLVEAPIDEF set_scaling_func *_set_scaling; -LPSOLVEAPIDEF set_semicont_func *_set_semicont; -LPSOLVEAPIDEF set_sense_func *_set_sense; -LPSOLVEAPIDEF set_simplextype_func *_set_simplextype; -LPSOLVEAPIDEF set_solutionlimit_func *_set_solutionlimit; -LPSOLVEAPIDEF set_timeout_func *_set_timeout; -LPSOLVEAPIDEF set_trace_func *_set_trace; -LPSOLVEAPIDEF set_upbo_func *_set_upbo; -LPSOLVEAPIDEF set_var_branch_func *_set_var_branch; -LPSOLVEAPIDEF set_var_weights_func *_set_var_weights; -LPSOLVEAPIDEF set_verbose_func *_set_verbose; -LPSOLVEAPIDEF set_XLI_func *_set_XLI; -LPSOLVEAPIDEF solve_func *_solve; -LPSOLVEAPIDEF str_add_column_func *_str_add_column; -LPSOLVEAPIDEF str_add_constraint_func *_str_add_constraint; -LPSOLVEAPIDEF str_add_lag_con_func *_str_add_lag_con; -LPSOLVEAPIDEF str_set_obj_fn_func *_str_set_obj_fn; -LPSOLVEAPIDEF str_set_rh_vec_func *_str_set_rh_vec; -LPSOLVEAPIDEF time_elapsed_func *_time_elapsed; -LPSOLVEAPIDEF unscale_func *_unscale; -LPSOLVEAPIDEF write_lp_func *_write_lp; -LPSOLVEAPIDEF write_LP_func *_write_LP; -LPSOLVEAPIDEF write_mps_func *_write_mps; -LPSOLVEAPIDEF write_MPS_func *_write_MPS; -LPSOLVEAPIDEF write_freemps_func *_write_freemps; -LPSOLVEAPIDEF write_freeMPS_func *_write_freeMPS; -LPSOLVEAPIDEF write_XLI_func *_write_XLI; -LPSOLVEAPIDEF write_basis_func *_write_basis; -LPSOLVEAPIDEF write_params_func *_write_params; - -#if defined LPSOLVEAPIFROMLPREC - -static int init_lpsolve(lprec *lp) -{ - _add_column = lp->add_column; - _add_columnex = lp->add_columnex; - _add_constraint = lp->add_constraint; - _add_constraintex = lp->add_constraintex; - _add_lag_con = lp->add_lag_con; - _add_SOS = lp->add_SOS; - _column_in_lp = lp->column_in_lp; - _copy_lp = lp->copy_lp; - _default_basis = lp->default_basis; - _del_column = lp->del_column; - _del_constraint = lp->del_constraint; - _delete_lp = lp->delete_lp; - _dualize_lp = lp->dualize_lp; - _free_lp = lp->free_lp; - /*_get_accuracy = lp->get_accuracy;*/ - _get_anti_degen = lp->get_anti_degen; - _get_basis = lp->get_basis; - _get_basiscrash = lp->get_basiscrash; - _get_bb_depthlimit = lp->get_bb_depthlimit; - _get_bb_floorfirst = lp->get_bb_floorfirst; - _get_bb_rule = lp->get_bb_rule; - _get_bounds_tighter = lp->get_bounds_tighter; - _get_break_at_value = lp->get_break_at_value; -/* _get_break_numeric_accuracy = lp->get_break_numeric_accuracy;*/ - _get_col_name = lp->get_col_name; - _get_columnex = lp->get_columnex; - _get_constr_type = lp->get_constr_type; - _get_constr_value = lp->get_constr_value; - _get_constraints = lp->get_constraints; - _get_dual_solution = lp->get_dual_solution; - _get_epsb = lp->get_epsb; - _get_epsd = lp->get_epsd; - _get_epsel = lp->get_epsel; - _get_epsint = lp->get_epsint; - _get_epsperturb = lp->get_epsperturb; - _get_epspivot = lp->get_epspivot; - _get_improve = lp->get_improve; - _get_infinite = lp->get_infinite; - _get_lambda = lp->get_lambda; - _get_lowbo = lp->get_lowbo; - _get_lp_index = lp->get_lp_index; - _get_lp_name = lp->get_lp_name; - _get_Lrows = lp->get_Lrows; - _get_mat = lp->get_mat; - _get_mat_byindex = lp->get_mat_byindex; - _get_max_level = lp->get_max_level; - _get_maxpivot = lp->get_maxpivot; - _get_mip_gap = lp->get_mip_gap; - _get_multiprice = lp->get_multiprice; - _get_nameindex = lp->get_nameindex; - _get_Ncolumns = lp->get_Ncolumns; - _get_negrange = lp->get_negrange; - _get_nonzeros = lp->get_nonzeros; - _get_Norig_columns = lp->get_Norig_columns; - _get_Norig_rows = lp->get_Norig_rows; - _get_Nrows = lp->get_Nrows; - _get_obj_bound = lp->get_obj_bound; - _get_objective = lp->get_objective; - _get_orig_index = lp->get_orig_index; - _get_origcol_name = lp->get_origcol_name; - _get_origrow_name = lp->get_origrow_name; - _get_partialprice = lp->get_partialprice; - _get_pivoting = lp->get_pivoting; - _get_presolve = lp->get_presolve; - _get_presolveloops = lp->get_presolveloops; - _get_primal_solution = lp->get_primal_solution; - _get_print_sol = lp->get_print_sol; - _get_pseudocosts = lp->get_pseudocosts; - _get_ptr_constraints = lp->get_ptr_constraints; - _get_ptr_dual_solution = lp->get_ptr_dual_solution; - _get_ptr_lambda = lp->get_ptr_lambda; - _get_ptr_primal_solution = lp->get_ptr_primal_solution; - _get_ptr_sensitivity_obj = lp->get_ptr_sensitivity_obj; - _get_ptr_sensitivity_objex = lp->get_ptr_sensitivity_objex; - _get_ptr_sensitivity_rhs = lp->get_ptr_sensitivity_rhs; - _get_ptr_variables = lp->get_ptr_variables; - _get_rh = lp->get_rh; - _get_rh_range = lp->get_rh_range; - _get_row = lp->get_row; - _get_rowex = lp->get_rowex; - _get_row_name = lp->get_row_name; - _get_scalelimit = lp->get_scalelimit; - _get_scaling = lp->get_scaling; - _get_sensitivity_obj = lp->get_sensitivity_obj; - _get_sensitivity_objex = lp->get_sensitivity_objex; - _get_sensitivity_rhs = lp->get_sensitivity_rhs; - _get_simplextype = lp->get_simplextype; - _get_solutioncount = lp->get_solutioncount; - _get_solutionlimit = lp->get_solutionlimit; - _get_status = lp->get_status; - _get_statustext = lp->get_statustext; - _get_timeout = lp->get_timeout; - _get_total_iter = lp->get_total_iter; - _get_total_nodes = lp->get_total_nodes; - _get_upbo = lp->get_upbo; - _get_var_branch = lp->get_var_branch; - _get_var_dualresult = lp->get_var_dualresult; - _get_var_primalresult = lp->get_var_primalresult; - _get_var_priority = lp->get_var_priority; - _get_variables = lp->get_variables; - _get_verbose = lp->get_verbose; - _get_working_objective = lp->get_working_objective; - _has_BFP = lp->has_BFP; - _has_XLI = lp->has_XLI; - _is_add_rowmode = lp->is_add_rowmode; - _is_anti_degen = lp->is_anti_degen; - _is_binary = lp->is_binary; - _is_break_at_first = lp->is_break_at_first; - _is_constr_type = lp->is_constr_type; - _is_debug = lp->is_debug; - _is_feasible = lp->is_feasible; - _is_unbounded = lp->is_unbounded; - _is_infinite = lp->is_infinite; - _is_int = lp->is_int; - _is_integerscaling = lp->is_integerscaling; - _is_lag_trace = lp->is_lag_trace; - _is_maxim = lp->is_maxim; - _is_nativeBFP = lp->is_nativeBFP; - _is_nativeXLI = lp->is_nativeXLI; - _is_negative = lp->is_negative; - _is_piv_mode = lp->is_piv_mode; - _is_piv_rule = lp->is_piv_rule; - _is_presolve = lp->is_presolve; - _is_scalemode = lp->is_scalemode; - _is_scaletype = lp->is_scaletype; - _is_semicont = lp->is_semicont; - _is_SOS_var = lp->is_SOS_var; - _is_trace = lp->is_trace; - _is_use_names = lp->is_use_names; - _lp_solve_version = lp->lp_solve_version; - _make_lp = lp->make_lp; - _print_constraints = lp->print_constraints; - _print_debugdump = lp->print_debugdump; - _print_duals = lp->print_duals; - _print_lp = lp->print_lp; - _print_objective = lp->print_objective; - _print_scales = lp->print_scales; - _print_solution = lp->print_solution; - _print_str = lp->print_str; - _print_tableau = lp->print_tableau; - _put_abortfunc = lp->put_abortfunc; - _put_bb_nodefunc = lp->put_bb_nodefunc; - _put_bb_branchfunc = lp->put_bb_branchfunc; - _put_logfunc = lp->put_logfunc; - _put_msgfunc = lp->put_msgfunc; - _read_LP = lp->read_LP; - _read_MPS = lp->read_MPS; - _read_XLI = lp->read_XLI; - _read_params = lp->read_params; - _read_basis = lp->read_basis; - _reset_basis = lp->reset_basis; - _reset_params = lp->reset_params; - _report = lp->report; - _resize_lp = lp->resize_lp; - _set_add_rowmode = lp->set_add_rowmode; - _set_anti_degen = lp->set_anti_degen; - _set_basisvar = lp->set_basisvar; - _set_basis = lp->set_basis; - _set_basiscrash = lp->set_basiscrash; - _set_bb_depthlimit = lp->set_bb_depthlimit; - _set_bb_floorfirst = lp->set_bb_floorfirst; - _set_bb_rule = lp->set_bb_rule; - _set_BFP = lp->set_BFP; - _set_binary = lp->set_binary; - _set_bounds = lp->set_bounds; - _set_bounds_tighter = lp->set_bounds_tighter; - _set_break_at_first = lp->set_break_at_first; - _set_break_at_value = lp->set_break_at_value; -/* _set_break_numeric_accuracy = lp->set_break_numeric_accuracy;*/ - _set_column = lp->set_column; - _set_columnex = lp->set_columnex; - _set_col_name = lp->set_col_name; - _set_constr_type = lp->set_constr_type; - _set_debug = lp->set_debug; - _set_epsb = lp->set_epsb; - _set_epsd = lp->set_epsd; - _set_epsel = lp->set_epsel; - _set_epsint = lp->set_epsint; - _set_epslevel = lp->set_epslevel; - _set_epsperturb = lp->set_epsperturb; - _set_epspivot = lp->set_epspivot; - _set_unbounded = lp->set_unbounded; - _set_improve = lp->set_improve; - _set_infinite = lp->set_infinite; - _set_int = lp->set_int; - _set_lag_trace = lp->set_lag_trace; - _set_lowbo = lp->set_lowbo; - _set_lp_name = lp->set_lp_name; - _set_mat = lp->set_mat; - _set_maxim = lp->set_maxim; - _set_maxpivot = lp->set_maxpivot; - _set_minim = lp->set_minim; - _set_mip_gap = lp->set_mip_gap; - _set_multiprice = lp->set_multiprice; - _set_negrange = lp->set_negrange; - _set_obj_bound = lp->set_obj_bound; - _set_obj_fn = lp->set_obj_fn; - _set_obj_fnex = lp->set_obj_fnex; - _set_obj = lp->set_obj; - _set_outputfile = lp->set_outputfile; - _set_outputstream = lp->set_outputstream; - _set_partialprice = lp->set_partialprice; - _set_pivoting = lp->set_pivoting; - _set_preferdual = lp->set_preferdual; - _set_presolve = lp->set_presolve; - _set_print_sol = lp->set_print_sol; - _set_pseudocosts = lp->set_pseudocosts; - _set_rh = lp->set_rh; - _set_rh_range = lp->set_rh_range; - _set_rh_vec = lp->set_rh_vec; - _set_row = lp->set_row; - _set_rowex = lp->set_rowex; - _set_row_name = lp->set_row_name; - _set_scalelimit = lp->set_scalelimit; - _set_scaling = lp->set_scaling; - _set_semicont = lp->set_semicont; - _set_sense = lp->set_sense; - _set_simplextype = lp->set_simplextype; - _set_solutionlimit = lp->set_solutionlimit; - _set_timeout = lp->set_timeout; - _set_trace = lp->set_trace; - _set_upbo = lp->set_upbo; - _set_var_branch = lp->set_var_branch; - _set_var_weights = lp->set_var_weights; - _set_verbose = lp->set_verbose; - _set_XLI = lp->set_XLI; - _solve = lp->solve; - _str_add_column = lp->str_add_column; - _str_add_constraint = lp->str_add_constraint; - _str_add_lag_con = lp->str_add_lag_con; - _str_set_obj_fn = lp->str_set_obj_fn; - _str_set_rh_vec = lp->str_set_rh_vec; - _time_elapsed = lp->time_elapsed; - _unscale = lp->unscale; - _write_lp = lp->write_lp; - _write_LP = lp->write_LP; - _write_mps = lp->write_mps; - _write_MPS = lp->write_MPS; - _write_freemps = lp->write_freemps; - _write_freeMPS = lp->write_freeMPS; - _write_XLI = lp->write_XLI; - _write_basis = lp->write_basis; - _write_params = lp->write_params; - - return(TRUE); -} - -#elif defined LPSOLVEAPIFROMLIB - -#ifdef WIN32 -# include -#else -# include -#endif - -#if defined WIN32 -# define hlpsolve HINSTANCE -#else -# define hlpsolve void * -#endif - -static hlpsolve open_lpsolve_lib(char *filename) -{ - hlpsolve lpsolve; - -# if defined WIN32 - /* Get a handle to the Windows DLL module. */ - lpsolve = LoadLibrary("lpsolve55.dll"); -# else - lpsolve = dlopen("liblpsolve55.so", RTLD_LAZY);; -# endif - return(lpsolve); -} - -static int close_lpsolve_lib(hlpsolve lpsolve) -{ -#ifdef WIN32 - FreeLibrary(lpsolve); -#else - dlclose(lpsolve); -#endif - - return(TRUE); -} - -static int init_lpsolve(hlpsolve lpsolve) -{ -# if defined WIN32 -# define AddressOf GetProcAddress -# else -# define AddressOf dlsym -# endif - - /* assign API functions to lp structure */ - _add_column = (add_column_func *) AddressOf(lpsolve, "add_column"); - _add_columnex = (add_columnex_func *) AddressOf(lpsolve, "add_columnex"); - _add_constraint = (add_constraint_func *) AddressOf(lpsolve, "add_constraint"); - _add_constraintex = (add_constraintex_func *) AddressOf(lpsolve, "add_constraintex"); - _add_lag_con = (add_lag_con_func *) AddressOf(lpsolve, "add_lag_con"); - _add_SOS = (add_SOS_func *) AddressOf(lpsolve, "add_SOS"); - _column_in_lp = (column_in_lp_func *) AddressOf(lpsolve, "column_in_lp"); - _copy_lp = (copy_lp_func *) AddressOf(lpsolve, "copy_lp"); - _default_basis = (default_basis_func *) AddressOf(lpsolve, "default_basis"); - _del_column = (del_column_func *) AddressOf(lpsolve, "del_column"); - _del_constraint = (del_constraint_func *) AddressOf(lpsolve, "del_constraint"); - _delete_lp = (delete_lp_func *) AddressOf(lpsolve, "delete_lp"); - _dualize_lp = (dualize_lp_func *) AddressOf(lpsolve, "dualize_lp"); - _free_lp = (free_lp_func *) AddressOf(lpsolve, "free_lp"); - /*_get_accuracy = (get_accuracy_func *) AddressOf(lpsolve, "get_accuracy");*/ - _get_anti_degen = (get_anti_degen_func *) AddressOf(lpsolve, "get_anti_degen"); - _get_basis = (get_basis_func *) AddressOf(lpsolve, "get_basis"); - _get_basiscrash = (get_basiscrash_func *) AddressOf(lpsolve, "get_basiscrash"); - _get_bb_depthlimit = (get_bb_depthlimit_func *) AddressOf(lpsolve, "get_bb_depthlimit"); - _get_bb_floorfirst = (get_bb_floorfirst_func *) AddressOf(lpsolve, "get_bb_floorfirst"); - _get_bb_rule = (get_bb_rule_func *) AddressOf(lpsolve, "get_bb_rule"); - _get_bounds_tighter = (get_bounds_tighter_func *) AddressOf(lpsolve, "get_bounds_tighter"); - _get_break_at_value = (get_break_at_value_func *) AddressOf(lpsolve, "get_break_at_value"); -/* _get_break_numeric_accuracy = (get_break_numeric_accuracy_func *) AddressOf(lpsolve, "get_break_numeric_accuracy");*/ - _get_col_name = (get_col_name_func *) AddressOf(lpsolve, "get_col_name"); - _get_column = (get_column_func *) AddressOf(lpsolve, "get_column"); - _get_columnex = (get_columnex_func *) AddressOf(lpsolve, "get_columnex"); - _get_constr_type = (get_constr_type_func *) AddressOf(lpsolve, "get_constr_type"); - _get_constr_value = (get_constr_value_func *) AddressOf(lpsolve, "get_constr_value"); - _get_constraints = (get_constraints_func *) AddressOf(lpsolve, "get_constraints"); - _get_dual_solution = (get_dual_solution_func *) AddressOf(lpsolve, "get_dual_solution"); - _get_epsb = (get_epsb_func *) AddressOf(lpsolve, "get_epsb"); - _get_epsd = (get_epsd_func *) AddressOf(lpsolve, "get_epsd"); - _get_epsel = (get_epsel_func *) AddressOf(lpsolve, "get_epsel"); - _get_epsint = (get_epsint_func *) AddressOf(lpsolve, "get_epsint"); - _get_epsperturb = (get_epsperturb_func *) AddressOf(lpsolve, "get_epsperturb"); - _get_epspivot = (get_epspivot_func *) AddressOf(lpsolve, "get_epspivot"); - _get_improve = (get_improve_func *) AddressOf(lpsolve, "get_improve"); - _get_infinite = (get_infinite_func *) AddressOf(lpsolve, "get_infinite"); - _get_lambda = (get_lambda_func *) AddressOf(lpsolve, "get_lambda"); - _get_lowbo = (get_lowbo_func *) AddressOf(lpsolve, "get_lowbo"); - _get_lp_index = (get_lp_index_func *) AddressOf(lpsolve, "get_lp_index"); - _get_lp_name = (get_lp_name_func *) AddressOf(lpsolve, "get_lp_name"); - _get_Lrows = (get_Lrows_func *) AddressOf(lpsolve, "get_Lrows"); - _get_mat = (get_mat_func *) AddressOf(lpsolve, "get_mat"); - _get_mat_byindex = (get_mat_byindex_func *) AddressOf(lpsolve, "get_mat_byindex"); - _get_max_level = (get_max_level_func *) AddressOf(lpsolve, "get_max_level"); - _get_maxpivot = (get_maxpivot_func *) AddressOf(lpsolve, "get_maxpivot"); - _get_mip_gap = (get_mip_gap_func *) AddressOf(lpsolve, "get_mip_gap"); - _get_multiprice = (get_multiprice_func *) AddressOf(lpsolve, "get_multiprice"); - _get_nameindex = (get_nameindex_func *) AddressOf(lpsolve, "get_nameindex"); - _get_Ncolumns = (get_Ncolumns_func *) AddressOf(lpsolve, "get_Ncolumns"); - _get_negrange = (get_negrange_func *) AddressOf(lpsolve, "get_negrange"); - _get_nonzeros = (get_nz_func *) AddressOf(lpsolve, "get_nonzeros"); - _get_Norig_columns = (get_Norig_columns_func *) AddressOf(lpsolve, "get_Norig_columns"); - _get_Norig_rows = (get_Norig_rows_func *) AddressOf(lpsolve, "get_Norig_rows"); - _get_Nrows = (get_Nrows_func *) AddressOf(lpsolve, "get_Nrows"); - _get_obj_bound = (get_obj_bound_func *) AddressOf(lpsolve, "get_obj_bound"); - _get_objective = (get_objective_func *) AddressOf(lpsolve, "get_objective"); - _get_orig_index = (get_orig_index_func *) AddressOf(lpsolve, "get_orig_index"); - _get_origcol_name = (get_origcol_name_func *) AddressOf(lpsolve, "get_origcol_name"); - _get_origrow_name = (get_origrow_name_func *) AddressOf(lpsolve, "get_origrow_name"); - _get_partialprice = (get_partialprice_func *) AddressOf(lpsolve, "get_partialprice"); - _get_pivoting = (get_pivoting_func *) AddressOf(lpsolve, "get_pivoting"); - _get_presolve = (get_presolve_func *) AddressOf(lpsolve, "get_presolve"); - _get_presolveloops = (get_presolveloops_func *) AddressOf(lpsolve, "get_presolveloops"); - _get_primal_solution = (get_primal_solution_func *) AddressOf(lpsolve, "get_primal_solution"); - _get_print_sol = (get_print_sol_func *) AddressOf(lpsolve, "get_print_sol"); - _get_pseudocosts = (get_pseudocosts_func *) AddressOf(lpsolve, "get_pseudocosts"); - _get_ptr_constraints = (get_ptr_constraints_func *) AddressOf(lpsolve, "get_ptr_constraints"); - _get_ptr_dual_solution = (get_ptr_dual_solution_func *) AddressOf(lpsolve, "get_ptr_dual_solution"); - _get_ptr_lambda = (get_ptr_lambda_func *) AddressOf(lpsolve, "get_ptr_lambda"); - _get_ptr_primal_solution = (get_ptr_primal_solution_func *) AddressOf(lpsolve, "get_ptr_primal_solution"); - _get_ptr_sensitivity_obj = (get_ptr_sensitivity_obj_func *) AddressOf(lpsolve, "get_ptr_sensitivity_obj"); - _get_ptr_sensitivity_objex = (get_ptr_sensitivity_objex_func *) AddressOf(lpsolve, "get_ptr_sensitivity_objex"); - _get_ptr_sensitivity_rhs = (get_ptr_sensitivity_rhs_func *) AddressOf(lpsolve, "get_ptr_sensitivity_rhs"); - _get_ptr_variables = (get_ptr_variables_func *) AddressOf(lpsolve, "get_ptr_variables"); - _get_rh = (get_rh_func *) AddressOf(lpsolve, "get_rh"); - _get_rh_range = (get_rh_range_func *) AddressOf(lpsolve, "get_rh_range"); - _get_row = (get_row_func *) AddressOf(lpsolve, "get_row"); - _get_rowex = (get_rowex_func *) AddressOf(lpsolve, "get_rowex"); - _get_row_name = (get_row_name_func *) AddressOf(lpsolve, "get_row_name"); - _get_scalelimit = (get_scalelimit_func *) AddressOf(lpsolve, "get_scalelimit"); - _get_scaling = (get_scaling_func *) AddressOf(lpsolve, "get_scaling"); - _get_sensitivity_obj = (get_sensitivity_obj_func *) AddressOf(lpsolve, "get_sensitivity_obj"); - _get_sensitivity_objex = (get_sensitivity_objex_func *) AddressOf(lpsolve, "get_sensitivity_objex"); - _get_sensitivity_rhs = (get_sensitivity_rhs_func *) AddressOf(lpsolve, "get_sensitivity_rhs"); - _get_simplextype = (get_simplextype_func *) AddressOf(lpsolve, "get_simplextype"); - _get_solutioncount = (get_solutioncount_func *) AddressOf(lpsolve, "get_solutioncount"); - _get_solutionlimit = (get_solutionlimit_func *) AddressOf(lpsolve, "get_solutionlimit"); - _get_status = (get_status_func *) AddressOf(lpsolve, "get_status"); - _get_statustext = (get_statustext_func *) AddressOf(lpsolve, "get_statustext"); - _get_timeout = (get_timeout_func *) AddressOf(lpsolve, "get_timeout"); - _get_total_iter = (get_total_iter_func *) AddressOf(lpsolve, "get_total_iter"); - _get_total_nodes = (get_total_nodes_func *) AddressOf(lpsolve, "get_total_nodes"); - _get_upbo = (get_upbo_func *) AddressOf(lpsolve, "get_upbo"); - _get_var_branch = (get_var_branch_func *) AddressOf(lpsolve, "get_var_branch"); - _get_var_dualresult = (get_var_dualresult_func *) AddressOf(lpsolve, "get_var_dualresult"); - _get_var_primalresult = (get_var_primalresult_func *) AddressOf(lpsolve, "get_var_primalresult"); - _get_var_priority = (get_var_priority_func *) AddressOf(lpsolve, "get_var_priority"); - _get_variables = (get_variables_func *) AddressOf(lpsolve, "get_variables"); - _get_verbose = (get_verbose_func *) AddressOf(lpsolve, "get_verbose"); - _get_working_objective = (get_working_objective_func *) AddressOf(lpsolve, "get_working_objective"); - _guess_basis = (guess_basis_func *) AddressOf(lpsolve, "guess_basis"); - _has_BFP = (has_BFP_func *) AddressOf(lpsolve, "has_BFP"); - _has_XLI = (has_XLI_func *) AddressOf(lpsolve, "has_XLI"); - _is_add_rowmode = (is_add_rowmode_func *) AddressOf(lpsolve, "is_add_rowmode"); - _is_anti_degen = (is_anti_degen_func *) AddressOf(lpsolve, "is_anti_degen"); - _is_binary = (is_binary_func *) AddressOf(lpsolve, "is_binary"); - _is_break_at_first = (is_break_at_first_func *) AddressOf(lpsolve, "is_break_at_first"); - _is_constr_type = (is_constr_type_func *) AddressOf(lpsolve, "is_constr_type"); - _is_debug = (is_debug_func *) AddressOf(lpsolve, "is_debug"); - _is_feasible = (is_feasible_func *) AddressOf(lpsolve, "is_feasible"); - _is_unbounded = (is_unbounded_func *) AddressOf(lpsolve, "is_unbounded"); - _is_infinite = (is_infinite_func *) AddressOf(lpsolve, "is_infinite"); - _is_int = (is_int_func *) AddressOf(lpsolve, "is_int"); - _is_integerscaling = (is_integerscaling_func *) AddressOf(lpsolve, "is_integerscaling"); - _is_lag_trace = (is_lag_trace_func *) AddressOf(lpsolve, "is_lag_trace"); - _is_maxim = (is_maxim_func *) AddressOf(lpsolve, "is_maxim"); - _is_nativeBFP = (is_nativeBFP_func *) AddressOf(lpsolve, "is_nativeBFP"); - _is_nativeXLI = (is_nativeXLI_func *) AddressOf(lpsolve, "is_nativeXLI"); - _is_negative = (is_negative_func *) AddressOf(lpsolve, "is_negative"); - _is_piv_mode = (is_piv_mode_func *) AddressOf(lpsolve, "is_piv_mode"); - _is_piv_rule = (is_piv_rule_func *) AddressOf(lpsolve, "is_piv_rule"); - _is_presolve = (is_presolve_func *) AddressOf(lpsolve, "is_presolve"); - _is_scalemode = (is_scalemode_func *) AddressOf(lpsolve, "is_scalemode"); - _is_scaletype = (is_scaletype_func *) AddressOf(lpsolve, "is_scaletype"); - _is_semicont = (is_semicont_func *) AddressOf(lpsolve, "is_semicont"); - _is_SOS_var = (is_SOS_var_func *) AddressOf(lpsolve, "is_SOS_var"); - _is_trace = (is_trace_func *) AddressOf(lpsolve, "is_trace"); - _is_use_names = (is_use_names_func *) AddressOf(lpsolve, "is_use_names"); - _lp_solve_version = (lp_solve_version_func *) AddressOf(lpsolve, "lp_solve_version"); - _make_lp = (make_lp_func *) AddressOf(lpsolve, "make_lp"); - _print_constraints = (print_constraints_func *) AddressOf(lpsolve, "print_constraints"); - _print_debugdump = (print_debugdump_func *) AddressOf(lpsolve, "print_debugdump"); - _print_duals = (print_duals_func *) AddressOf(lpsolve, "print_duals"); - _print_lp = (print_lp_func *) AddressOf(lpsolve, "print_lp"); - _print_objective = (print_objective_func *) AddressOf(lpsolve, "print_objective"); - _print_scales = (print_scales_func *) AddressOf(lpsolve, "print_scales"); - _print_solution = (print_solution_func *) AddressOf(lpsolve, "print_solution"); - _print_str = (print_str_func *) AddressOf(lpsolve, "print_str"); - _print_tableau = (print_tableau_func *) AddressOf(lpsolve, "print_tableau"); - _put_abortfunc = (put_abortfunc_func *) AddressOf(lpsolve, "put_abortfunc"); - _put_bb_nodefunc = (put_bb_nodefunc_func *) AddressOf(lpsolve, "put_bb_nodefunc"); - _put_bb_branchfunc = (put_bb_branchfunc_func *) AddressOf(lpsolve, "put_bb_branchfunc"); - _put_logfunc = (put_logfunc_func *) AddressOf(lpsolve, "put_logfunc"); - _put_msgfunc = (put_msgfunc_func *) AddressOf(lpsolve, "put_msgfunc"); - _read_LP = (read_LP_func *) AddressOf(lpsolve, "read_LP"); - _read_MPS = (read_MPS_func *) AddressOf(lpsolve, "read_MPS"); - _read_XLI = (read_XLI_func *) AddressOf(lpsolve, "read_XLI"); - _read_params = (read_params_func *) AddressOf(lpsolve, "read_params"); - _read_basis = (read_basis_func *) AddressOf(lpsolve, "read_basis"); - _reset_basis = (reset_basis_func *) AddressOf(lpsolve, "reset_basis"); - _reset_params = (reset_params_func *) AddressOf(lpsolve, "reset_params"); - _report = (reportfunc *) AddressOf(lpsolve, "report"); - _resize_lp = (resize_lp_func *) AddressOf(lpsolve, "resize_lp"); - _set_add_rowmode = (set_add_rowmode_func *) AddressOf(lpsolve, "set_add_rowmode"); - _set_anti_degen = (set_anti_degen_func *) AddressOf(lpsolve, "set_anti_degen"); - _set_basisvar = (set_basisvar_func *) AddressOf(lpsolve, "set_basisvar"); - _set_basis = (set_basis_func *) AddressOf(lpsolve, "set_basis"); - _set_basiscrash = (set_basiscrash_func *) AddressOf(lpsolve, "set_basiscrash"); - _set_bb_depthlimit = (set_bb_depthlimit_func *) AddressOf(lpsolve, "set_bb_depthlimit"); - _set_bb_floorfirst = (set_bb_floorfirst_func *) AddressOf(lpsolve, "set_bb_floorfirst"); - _set_bb_rule = (set_bb_rule_func *) AddressOf(lpsolve, "set_bb_rule"); - _set_BFP = (set_BFP_func *) AddressOf(lpsolve, "set_BFP"); - _set_binary = (set_binary_func *) AddressOf(lpsolve, "set_binary"); - _set_bounds = (set_bounds_func *) AddressOf(lpsolve, "set_bounds"); - _set_bounds_tighter = (set_bounds_tighter_func *) AddressOf(lpsolve, "set_bounds_tighter"); - _set_break_at_first = (set_break_at_first_func *) AddressOf(lpsolve, "set_break_at_first"); - _set_break_at_value = (set_break_at_value_func *) AddressOf(lpsolve, "set_break_at_value"); -/* _set_break_numeric_accuracy = (set_break_numeric_accuracy_func *) AddressOf(lpsolve, "set_break_numeric_accuracy");*/ - _set_column = (set_column_func *) AddressOf(lpsolve, "set_column"); - _set_columnex = (set_columnex_func *) AddressOf(lpsolve, "set_columnex"); - _set_col_name = (set_col_name_func *) AddressOf(lpsolve, "set_col_name"); - _set_constr_type = (set_constr_type_func *) AddressOf(lpsolve, "set_constr_type"); - _set_debug = (set_debug_func *) AddressOf(lpsolve, "set_debug"); - _set_epsb = (set_epsb_func *) AddressOf(lpsolve, "set_epsb"); - _set_epsd = (set_epsd_func *) AddressOf(lpsolve, "set_epsd"); - _set_epsel = (set_epsel_func *) AddressOf(lpsolve, "set_epsel"); - _set_epsint = (set_epsint_func *) AddressOf(lpsolve, "set_epsint"); - _set_epslevel = (set_epslevel_func *) AddressOf(lpsolve, "set_epslevel"); - _set_epsperturb = (set_epsperturb_func *) AddressOf(lpsolve, "set_epsperturb"); - _set_epspivot = (set_epspivot_func *) AddressOf(lpsolve, "set_epspivot"); - _set_unbounded = (set_unbounded_func *) AddressOf(lpsolve, "set_unbounded"); - _set_improve = (set_improve_func *) AddressOf(lpsolve, "set_improve"); - _set_infinite = (set_infinite_func *) AddressOf(lpsolve, "set_infinite"); - _set_int = (set_int_func *) AddressOf(lpsolve, "set_int"); - _set_lag_trace = (set_lag_trace_func *) AddressOf(lpsolve, "set_lag_trace"); - _set_lowbo = (set_lowbo_func *) AddressOf(lpsolve, "set_lowbo"); - _set_lp_name = (set_lp_name_func *) AddressOf(lpsolve, "set_lp_name"); - _set_mat = (set_mat_func *) AddressOf(lpsolve, "set_mat"); - _set_maxim = (set_maxim_func *) AddressOf(lpsolve, "set_maxim"); - _set_maxpivot = (set_maxpivot_func *) AddressOf(lpsolve, "set_maxpivot"); - _set_minim = (set_minim_func *) AddressOf(lpsolve, "set_minim"); - _set_mip_gap = (set_mip_gap_func *) AddressOf(lpsolve, "set_mip_gap"); - _set_multiprice = (set_multiprice_func *) AddressOf(lpsolve, "set_multiprice"); - _set_negrange = (set_negrange_func *) AddressOf(lpsolve, "set_negrange"); - _set_obj_bound = (set_obj_bound_func *) AddressOf(lpsolve, "set_obj_bound"); - _set_obj_fn = (set_obj_fn_func *) AddressOf(lpsolve, "set_obj_fn"); - _set_obj_fnex = (set_obj_fnex_func *) AddressOf(lpsolve, "set_obj_fnex"); - _set_obj = (set_obj_func *) AddressOf(lpsolve, "set_obj"); - _set_outputfile = (set_outputfile_func *) AddressOf(lpsolve, "set_outputfile"); - _set_outputstream = (set_outputstream_func *) AddressOf(lpsolve, "set_outputstream"); - _set_partialprice = (set_partialprice_func *) AddressOf(lpsolve, "set_partialprice"); - _set_pivoting = (set_pivoting_func *) AddressOf(lpsolve, "set_pivoting"); - _set_preferdual = (set_preferdual_func *) AddressOf(lpsolve, "set_preferdual"); - _set_presolve = (set_presolve_func *) AddressOf(lpsolve, "set_presolve"); - _set_print_sol = (set_print_sol_func *) AddressOf(lpsolve, "set_print_sol"); - _set_pseudocosts = (set_pseudocosts_func *) AddressOf(lpsolve, "set_pseudocosts"); - _set_rh = (set_rh_func *) AddressOf(lpsolve, "set_rh"); - _set_rh_range = (set_rh_range_func *) AddressOf(lpsolve, "set_rh_range"); - _set_rh_vec = (set_rh_vec_func *) AddressOf(lpsolve, "set_rh_vec"); - _set_row = (set_row_func *) AddressOf(lpsolve, "set_row"); - _set_rowex = (set_rowex_func *) AddressOf(lpsolve, "set_rowex"); - _set_row_name = (set_row_name_func *) AddressOf(lpsolve, "set_row_name"); - _set_scalelimit = (set_scalelimit_func *) AddressOf(lpsolve, "set_scalelimit"); - _set_scaling = (set_scaling_func *) AddressOf(lpsolve, "set_scaling"); - _set_semicont = (set_semicont_func *) AddressOf(lpsolve, "set_semicont"); - _set_sense = (set_sense_func *) AddressOf(lpsolve, "set_sense"); - _set_simplextype = (set_simplextype_func *) AddressOf(lpsolve, "set_simplextype"); - _set_solutionlimit = (set_solutionlimit_func *) AddressOf(lpsolve, "set_solutionlimit"); - _set_timeout = (set_timeout_func *) AddressOf(lpsolve, "set_timeout"); - _set_trace = (set_trace_func *) AddressOf(lpsolve, "set_trace"); - _set_upbo = (set_upbo_func *) AddressOf(lpsolve, "set_upbo"); - _set_var_branch = (set_var_branch_func *) AddressOf(lpsolve, "set_var_branch"); - _set_var_weights = (set_var_weights_func *) AddressOf(lpsolve, "set_var_weights"); - _set_verbose = (set_verbose_func *) AddressOf(lpsolve, "set_verbose"); - _set_XLI = (set_XLI_func *) AddressOf(lpsolve, "set_XLI"); - _solve = (solve_func *) AddressOf(lpsolve, "solve"); - _str_add_column = (str_add_column_func *) AddressOf(lpsolve, "str_add_column"); - _str_add_constraint = (str_add_constraint_func *) AddressOf(lpsolve, "str_add_constraint"); - _str_add_lag_con = (str_add_lag_con_func *) AddressOf(lpsolve, "str_add_lag_con"); - _str_set_obj_fn = (str_set_obj_fn_func *) AddressOf(lpsolve, "str_set_obj_fn"); - _str_set_rh_vec = (str_set_rh_vec_func *) AddressOf(lpsolve, "str_set_rh_vec"); - _time_elapsed = (time_elapsed_func *) AddressOf(lpsolve, "time_elapsed"); - _unscale = (unscale_func *) AddressOf(lpsolve, "unscale"); - _write_lp = (write_lp_func *) AddressOf(lpsolve, "write_lp"); - _write_LP = (write_LP_func *) AddressOf(lpsolve, "write_LP"); - _write_mps = (write_mps_func *) AddressOf(lpsolve, "write_mps"); - _write_MPS = (write_MPS_func *) AddressOf(lpsolve, "write_MPS"); - _write_freemps = (write_freemps_func *) AddressOf(lpsolve, "write_freemps"); - _write_freeMPS = (write_freeMPS_func *) AddressOf(lpsolve, "write_freeMPS"); - _write_XLI = (write_XLI_func *) AddressOf(lpsolve, "write_XLI"); - _write_basis = (write_basis_func *) AddressOf(lpsolve, "write_basis"); - _write_params = (write_params_func *) AddressOf(lpsolve, "write_params"); - - return(TRUE); -# undef AddressOf -} - -#else -# error Either LPSOLVEAPIFROMLPREC or LPSOLVEAPIFROMLIB must be defined -#endif - -#define add_column _add_column -#define add_columnex _add_columnex -#define add_constraint _add_constraint -#define add_constraintex _add_constraintex -#define add_lag_con _add_lag_con -#define add_SOS _add_SOS -#define column_in_lp _column_in_lp -#define copy_lp _copy_lp -#define default_basis _default_basis -#define del_column _del_column -#define del_constraint _del_constraint -#define delete_lp _delete_lp -#define dualize_lp _dualize_lp -#define free_lp _free_lp -/*#define get_accuracy _get_accuracy*/ -#define get_anti_degen _get_anti_degen -#define get_basis _get_basis -#define get_basiscrash _get_basiscrash -#define get_bb_depthlimit _get_bb_depthlimit -#define get_bb_floorfirst _get_bb_floorfirst -#define get_bb_rule _get_bb_rule -#define get_bounds_tighter _get_bounds_tighter -#define get_break_at_value _get_break_at_value -/*#define get_break_numeric_accuracy _get_break_numeric_accuracy*/ -#define get_col_name _get_col_name -#define get_column _get_column -#define get_columnex _get_columnex -#define get_constr_type _get_constr_type -#define get_constr_value _get_constr_value -#define get_constraints _get_constraints -#define get_dual_solution _get_dual_solution -#define get_epsb _get_epsb -#define get_epsd _get_epsd -#define get_epsel _get_epsel -#define get_epsint _get_epsint -#define get_epsperturb _get_epsperturb -#define get_epspivot _get_epspivot -#define get_improve _get_improve -#define get_infinite _get_infinite -#define get_lambda _get_lambda -#define get_lowbo _get_lowbo -#define get_lp_index _get_lp_index -#define get_lp_name _get_lp_name -#define get_Lrows _get_Lrows -#define get_mat _get_mat -#define get_mat_byindex _get_mat_byindex -#define get_max_level _get_max_level -#define get_maxpivot _get_maxpivot -#define get_mip_gap _get_mip_gap -#define get_multiprice _get_multiprice -#define get_nameindex _get_nameindex -#define get_Ncolumns _get_Ncolumns -#define get_negrange _get_negrange -#define get_nonzeros _get_nonzeros -#define get_Norig_columns _get_Norig_columns -#define get_Norig_rows _get_Norig_rows -#define get_Nrows _get_Nrows -#define get_obj_bound _get_obj_bound -#define get_objective _get_objective -#define get_orig_index _get_orig_index -#define get_origcol_name _get_origcol_name -#define get_origrow_name _get_origrow_name -#define get_partialprice _get_partialprice -#define get_pivoting _get_pivoting -#define get_presolve _get_presolve -#define get_presolveloops _get_presolveloops -#define get_primal_solution _get_primal_solution -#define get_print_sol _get_print_sol -#define get_pseudocosts _get_pseudocosts -#define get_ptr_constraints _get_ptr_constraints -#define get_ptr_dual_solution _get_ptr_dual_solution -#define get_ptr_lambda _get_ptr_lambda -#define get_ptr_primal_solution _get_ptr_primal_solution -#define get_ptr_sensitivity_obj _get_ptr_sensitivity_obj -#define get_ptr_sensitivity_objex _get_ptr_sensitivity_objex -#define get_ptr_sensitivity_rhs _get_ptr_sensitivity_rhs -#define get_ptr_variables _get_ptr_variables -#define get_rh _get_rh -#define get_rh_range _get_rh_range -#define get_row _get_row -#define get_rowex _get_rowex -#define get_row_name _get_row_name -#define get_scalelimit _get_scalelimit -#define get_scaling _get_scaling -#define get_sensitivity_obj _get_sensitivity_obj -#define get_sensitivity_objex _get_sensitivity_objex -#define get_sensitivity_rhs _get_sensitivity_rhs -#define get_simplextype _get_simplextype -#define get_solutioncount _get_solutioncount -#define get_solutionlimit _get_solutionlimit -#define get_status _get_status -#define get_statustext _get_statustext -#define get_timeout _get_timeout -#define get_total_iter _get_total_iter -#define get_total_nodes _get_total_nodes -#define get_upbo _get_upbo -#define get_var_branch _get_var_branch -#define get_var_dualresult _get_var_dualresult -#define get_var_primalresult _get_var_primalresult -#define get_var_priority _get_var_priority -#define get_variables _get_variables -#define get_verbose _get_verbose -#define get_working_objective _get_working_objective -#define guess_basis _guess_basis -#define has_BFP _has_BFP -#define has_XLI _has_XLI -#define is_add_rowmode _is_add_rowmode -#define is_anti_degen _is_anti_degen -#define is_binary _is_binary -#define is_break_at_first _is_break_at_first -#define is_constr_type _is_constr_type -#define is_debug _is_debug -#define is_feasible _is_feasible -#define is_unbounded _is_unbounded -#define is_infinite _is_infinite -#define is_int _is_int -#define is_integerscaling _is_integerscaling -#define is_lag_trace _is_lag_trace -#define is_maxim _is_maxim -#define is_nativeBFP _is_nativeBFP -#define is_nativeXLI _is_nativeXLI -#define is_negative _is_negative -#define is_piv_mode _is_piv_mode -#define is_piv_rule _is_piv_rule -#define is_presolve _is_presolve -#define is_scalemode _is_scalemode -#define is_scaletype _is_scaletype -#define is_semicont _is_semicont -#define is_SOS_var _is_SOS_var -#define is_trace _is_trace -#define is_use_names _is_use_names -#define lp_solve_version _lp_solve_version -#define make_lp _make_lp -#define print_constraints _print_constraints -#define print_debugdump _print_debugdump -#define print_duals _print_duals -#define print_lp _print_lp -#define print_objective _print_objective -#define print_scales _print_scales -#define print_solution _print_solution -#define print_str _print_str -#define print_tableau _print_tableau -#define put_abortfunc _put_abortfunc -#define put_bb_nodefunc _put_bb_nodefunc -#define put_bb_branchfunc _put_bb_branchfunc -#define put_logfunc _put_logfunc -#define put_msgfunc _put_msgfunc -#define read_LP _read_LP -#define read_MPS _read_MPS -#define read_XLI _read_XLI -#define read_params _read_params -#define read_basis _read_basis -#define reset_basis _reset_basis -#define reset_params _reset_params -#define report _report -#define resize_lp _resize_lp -#define set_add_rowmode _set_add_rowmode -#define set_anti_degen _set_anti_degen -#define set_basisvar _set_basisvar -#define set_basis _set_basis -#define set_basiscrash _set_basiscrash -#define set_bb_depthlimit _set_bb_depthlimit -#define set_bb_floorfirst _set_bb_floorfirst -#define set_bb_rule _set_bb_rule -#define set_BFP _set_BFP -#define set_binary _set_binary -#define set_bounds _set_bounds -#define set_bounds_tighter _set_bounds_tighter -#define set_break_at_first _set_break_at_first -#define set_break_at_value _set_break_at_value -/*#define set_break_numeric_accuracy _set_break_numeric_accuracy*/ -#define set_column _set_column -#define set_columnex _set_columnex -#define set_col_name _set_col_name -#define set_constr_type _set_constr_type -#define set_debug _set_debug -#define set_epsb _set_epsb -#define set_epsd _set_epsd -#define set_epsel _set_epsel -#define set_epsint _set_epsint -#define set_epslevel _set_epslevel -#define set_epsperturb _set_epsperturb -#define set_epspivot _set_epspivot -#define set_unbounded _set_unbounded -#define set_improve _set_improve -#define set_infinite _set_infinite -#define set_int _set_int -#define set_lag_trace _set_lag_trace -#define set_lowbo _set_lowbo -#define set_lp_name _set_lp_name -#define set_mat _set_mat -#define set_maxim _set_maxim -#define set_maxpivot _set_maxpivot -#define set_minim _set_minim -#define set_mip_gap _set_mip_gap -#define set_multiprice _set_multiprice -#define set_negrange _set_negrange -#define set_obj_bound _set_obj_bound -#define set_obj_fn _set_obj_fn -#define set_obj_fnex _set_obj_fnex -#define set_obj _set_obj -#define set_outputfile _set_outputfile -#define set_outputstream _set_outputstream -#define set_partialprice _set_partialprice -#define set_pivoting _set_pivoting -#define set_preferdual _set_preferdual -#define set_presolve _set_presolve -#define set_print_sol _set_print_sol -#define set_pseudocosts _set_pseudocosts -#define set_rh _set_rh -#define set_rh_range _set_rh_range -#define set_rh_vec _set_rh_vec -#define set_row _set_row -#define set_rowex _set_rowex -#define set_row_name _set_row_name -#define set_scalelimit _set_scalelimit -#define set_scaling _set_scaling -#define set_semicont _set_semicont -#define set_sense _set_sense -#define set_simplextype _set_simplextype -#define set_solutionlimit _set_solutionlimit -#define set_timeout _set_timeout -#define set_trace _set_trace -#define set_upbo _set_upbo -#define set_var_branch _set_var_branch -#define set_var_weights _set_var_weights -#define set_verbose _set_verbose -#define set_XLI _set_XLI -#define solve _solve -#define str_add_column _str_add_column -#define str_add_constraint _str_add_constraint -#define str_add_lag_con _str_add_lag_con -#define str_set_obj_fn _str_set_obj_fn -#define str_set_rh_vec _str_set_rh_vec -#define time_elapsed _time_elapsed -#define unscale _unscale -#define write_lp _write_lp -#define write_LP _write_LP -#define write_mps _write_mps -#define write_MPS _write_MPS -#define write_freemps _write_freemps -#define write_freeMPS _write_freeMPS -#define write_XLI _write_XLI -#define write_basis _write_basis -#define write_params _write_params diff --git a/code/3rd_lpsolve/lp_lib.c b/code/3rd_lpsolve/lp_lib.c deleted file mode 100644 index 988ab67d..00000000 --- a/code/3rd_lpsolve/lp_lib.c +++ /dev/null @@ -1,10323 +0,0 @@ - -/* ---------------------------------------------------------------------------------- - Main library of routines for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Michel Berkelaar (to v3.2) - Kjell Eikland (v4.0 and forward) - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: (see below) - - Release notes: - v5.0.0 1 January 2004 First integrated and repackaged version. - v5.0.1 8 May 2004 Cumulative update since initial release; - overall functionality scope maintained. - v5.1.0 20 July 2004 Reworked lp_solve throughout to fit new - flexible matrix storage model. - - ---------------------------------------------------------------------------------- */ - -/* ---------------------------------------------------------------------------------- */ -/* Main library of routines for lp_solve */ -/*----------------------------------------------------------------------------------- */ -#include -#include -#include -#include - -#if LoadInverseLib == TRUE - #ifdef WIN32 - #include - #else - #include - #endif -#endif - - -/* ---------------------------------------------------------------------------------- */ -/* Include core and support modules via headers */ -/* ---------------------------------------------------------------------------------- */ -#include "lp_lib.h" -#include "commonlib.h" -#include "lp_utils.h" -#include "lp_matrix.h" -#include "lp_SOS.h" -#include "lp_Hash.h" -#include "lp_MPS.h" -#include "lp_wlp.h" -#include "lp_presolve.h" -#include "lp_scale.h" -#include "lp_simplex.h" -#include "lp_mipbb.h" -#include "lp_report.h" -#include "lp_MDO.h" -#include "lp_bit.h" - -#if INVERSE_ACTIVE==INVERSE_LUMOD - #include "lp_LUMOD.h" -#elif INVERSE_ACTIVE==INVERSE_LUSOL - #include "lp_LUSOL.h" -#elif INVERSE_ACTIVE==INVERSE_GLPKLU - #include "lp_glpkLU.h" -#elif INVERSE_ACTIVE==INVERSE_ETAPFI - #include "lp_etaPFI.h" -#elif INVERSE_ACTIVE==INVERSE_LEGACY - #include "lp_etaPFI.h" -#endif - -#if libBLAS > 0 - #include "myblas.h" -#endif - -#ifdef __BORLANDC__ - #pragma hdrstop - #pragma package(smart_init) -#endif - -/* ---------------------------------------------------------------------------------- */ -/* Include selected basis inverse routines and price norm scalars */ -/* ---------------------------------------------------------------------------------- */ - -#include "lp_price.h" -#include "lp_pricePSE.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -#define sensrejvar TRUE - -/* Return lp_solve version information */ -void __WINAPI lp_solve_version(int *majorversion, int *minorversion, int *release, int *build) -{ - if(majorversion != NULL) - (*majorversion) = MAJORVERSION; - if(minorversion != NULL) - (*minorversion) = MINORVERSION; - if(release != NULL) - (*release) = RELEASE; - if(build != NULL) - (*build) = BUILD; -} - - -/* ---------------------------------------------------------------------------------- */ -/* Various interaction elements */ -/* ---------------------------------------------------------------------------------- */ - -MYBOOL __WINAPI userabort(lprec *lp, int message) -{ - MYBOOL abort; - int spx_save; - - spx_save = lp->spx_status; - lp->spx_status = RUNNING; - if(yieldformessages(lp) != 0) { - lp->spx_status = USERABORT; - if(lp->bb_level > 0) - lp->bb_break = TRUE; - } - if((message > 0) && (lp->usermessage != NULL) && (lp->msgmask & message)) - lp->usermessage(lp, lp->msghandle, message); - abort = (MYBOOL) (lp->spx_status != RUNNING); - if(!abort) - lp->spx_status = spx_save; - return( abort ); -} - -STATIC int yieldformessages(lprec *lp) -{ - if((lp->sectimeout > 0) && - ((timeNow()-lp->timestart)-(REAL)lp->sectimeout>0)) - lp->spx_status = TIMEOUT; - - if(lp->ctrlc != NULL) { - int retcode = lp->ctrlc(lp, lp->ctrlchandle); - /* Check for command to restart the B&B */ - if((retcode == ACTION_RESTART) && (lp->bb_level > 1)) { - lp->bb_break = AUTOMATIC; - retcode = 0; - } - return(retcode); - } - else - return(0); -} - -void __WINAPI set_outputstream(lprec *lp, FILE *stream) -{ - if((lp->outstream != NULL) && (lp->outstream != stdout)) { - if(lp->streamowned) - fclose(lp->outstream); - else - fflush(lp->outstream); - } - if(stream == NULL) - lp->outstream = stdout; - else - lp->outstream = stream; - lp->streamowned = FALSE; -} - -MYBOOL __WINAPI set_outputfile(lprec *lp, char *filename) -{ - MYBOOL ok; - FILE *output = stdout; - - ok = (MYBOOL) ((filename == NULL) || (*filename == 0) || ((output = fopen(filename,"w")) != NULL)); - if(ok) { - set_outputstream(lp, output); - lp->streamowned = (MYBOOL) ((filename != NULL) && (*filename != 0)); -#if 1 - if((filename != NULL) && (*filename == 0)) - lp->outstream = NULL; -#endif - } - return(ok); -} - -REAL __WINAPI time_elapsed(lprec *lp) -{ - if(lp->timeend > 0) - return(lp->timeend - lp->timestart); - else - return(timeNow() - lp->timestart); -} - -void __WINAPI put_bb_nodefunc(lprec *lp, lphandleint_intfunc newnode, void *bbnodehandle) -{ - lp->bb_usenode = newnode; - lp->bb_nodehandle = bbnodehandle; /* User-specified "owner process ID" */ -} -void __WINAPI put_bb_branchfunc(lprec *lp, lphandleint_intfunc newbranch, void *bbbranchhandle) -{ - lp->bb_usebranch = newbranch; - lp->bb_branchhandle = bbbranchhandle; /* User-specified "owner process ID" */ -} -void __WINAPI put_abortfunc(lprec *lp, lphandle_intfunc newctrlc, void *ctrlchandle) -{ - lp->ctrlc = newctrlc; - lp->ctrlchandle = ctrlchandle; /* User-specified "owner process ID" */ -} -void __WINAPI put_logfunc(lprec *lp, lphandlestr_func newlog, void *loghandle) -{ - lp->writelog = newlog; - lp->loghandle = loghandle; /* User-specified "owner process ID" */ -} -void __WINAPI put_msgfunc(lprec *lp, lphandleint_func newmsg, void *msghandle, int mask) -{ - lp->usermessage = newmsg; - lp->msghandle = msghandle; /* User-specified "owner process ID" */ - lp->msgmask = mask; -} - - -/* ---------------------------------------------------------------------------------- */ -/* DLL exported function */ -/* ---------------------------------------------------------------------------------- */ -lprec * __WINAPI read_MPS(char *filename, int options) -{ - lprec *lp = NULL; - int typeMPS; - - typeMPS = (options & ~0x07) >> 2; - if ((typeMPS & (MPSFIXED|MPSFREE)) == 0) - typeMPS |= MPSFIXED; - if(MPS_readfile(&lp, filename, typeMPS, options & 0x07)) - return( lp ); - else - return( NULL ); -} -lprec * __WINAPI read_mps(FILE *filename, int options) -{ - lprec *lp = NULL; - int typeMPS; - - typeMPS = (options & ~0x07) >> 2; - if ((typeMPS & (MPSFIXED|MPSFREE)) == 0) - typeMPS |= MPSFIXED; - if(MPS_readhandle(&lp, filename, typeMPS, options & 0x07)) - return( lp ); - else - return( NULL ); -} -/* #if defined develop */ -lprec * __WINAPI read_mpsex(void *userhandle, read_modeldata_func read_modeldata, int options) -{ - lprec *lp = NULL; - int typeMPS; - - typeMPS = (options & ~0x07) >> 2; - if ((typeMPS & (MPSFIXED|MPSFREE)) == 0) - typeMPS |= MPSFIXED; - if(MPS_readex(&lp, userhandle, read_modeldata, typeMPS, options & 0x07)) - return( lp ); - else - return( NULL ); -} -/* #endif */ -lprec * __WINAPI read_freeMPS(char *filename, int options) -{ - lprec *lp = NULL; - int typeMPS; - - typeMPS = (options & ~0x07) >> 2; - typeMPS &= ~MPSFIXED; - typeMPS |= MPSFREE; - if(MPS_readfile(&lp, filename, typeMPS, options & 0x07)) - return( lp ); - else - return( NULL ); -} -lprec * __WINAPI read_freemps(FILE *filename, int options) -{ - lprec *lp = NULL; - int typeMPS; - - typeMPS = (options & ~0x07) >> 2; - typeMPS &= ~MPSFIXED; - typeMPS |= MPSFREE; - if(MPS_readhandle(&lp, filename, typeMPS, options & 0x07)) - return( lp ); - else - return( NULL ); -} -/* #if defined develop */ -lprec * __WINAPI read_freempsex(void *userhandle, read_modeldata_func read_modeldata, int options) -{ - lprec *lp = NULL; - int typeMPS; - - typeMPS = (options & ~0x07) >> 2; - typeMPS &= ~MPSFIXED; - typeMPS |= MPSFREE; - if(MPS_readex(&lp, userhandle, read_modeldata, typeMPS, options & 0x07)) - return( lp ); - else - return( NULL ); -} -/* #endif */ -MYBOOL __WINAPI write_mps(lprec *lp, char *filename) -{ - return(MPS_writefile(lp, MPSFIXED, filename)); -} -MYBOOL __WINAPI write_MPS(lprec *lp, FILE *output) -{ - return(MPS_writehandle(lp, MPSFIXED, output)); -} - -MYBOOL __WINAPI write_freemps(lprec *lp, char *filename) -{ - return(MPS_writefile(lp, MPSFREE, filename)); -} -MYBOOL __WINAPI write_freeMPS(lprec *lp, FILE *output) -{ - return(MPS_writehandle(lp, MPSFREE, output)); -} - -MYBOOL __WINAPI write_lp(lprec *lp, char *filename) -{ - return(LP_writefile(lp, filename)); -} -MYBOOL __WINAPI write_LP(lprec *lp, FILE *output) -{ - return(LP_writehandle(lp, output)); -} -#ifndef PARSER_LP -MYBOOL __WINAPI LP_readhandle(lprec **lp, FILE *filename, int verbose, char *lp_name) -{ - return(FALSE); -} -lprec * __WINAPI read_lp(FILE *filename, int verbose, char *lp_name) -{ - return(NULL); -} -lprec * __WINAPI read_LP(char *filename, int verbose, char *lp_name) -{ - return(NULL); -} -#endif - -MYBOOL __WINAPI write_basis(lprec *lp, char *filename) -{ - int typeMPS = MPSFIXED; - return( MPS_writeBAS(lp, typeMPS, filename) ); -} -MYBOOL __WINAPI read_basis(lprec *lp, char *filename, char *info) -{ - int typeMPS = MPSFIXED; - - typeMPS = MPS_readBAS(lp, typeMPS, filename, info); - - /* Code basis */ - if(typeMPS) { - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - lp->basis_valid = TRUE; /* Do not re-initialize basis on entering Solve */ - lp->var_basic[0] = FALSE; /* Set to signal that this is a non-default basis */ - } - return( (MYBOOL) typeMPS ); -} - -/* Write and read lp_solve parameters (placeholders) - see lp_params.c */ -void __WINAPI reset_params(lprec *lp) -{ - int mode; - - lp->epsmachine = DEF_EPSMACHINE; - lp->epsperturb = DEF_PERTURB; - /* lp->lag_accept = DEF_LAGACCEPT; */ - set_epslevel(lp, EPS_DEFAULT); - - lp->tighten_on_set = FALSE; - lp->negrange = DEF_NEGRANGE; - -#if 0 - lp->do_presolve = PRESOLVE_ROWS | PRESOLVE_COLS | PRESOLVE_MERGEROWS | - PRESOLVE_REDUCEGCD | - PRESOLVE_ROWDOMINATE; -#else - lp->do_presolve = PRESOLVE_NONE; -#endif - lp->presolveloops = DEF_MAXPRESOLVELOOPS; - - lp->scalelimit = DEF_SCALINGLIMIT; - lp->scalemode = SCALE_INTEGERS | -#if 0 - SCALE_POWER2 | - SCALE_LOGARITHMIC | SCALE_MEAN; -#else - SCALE_LINEAR | SCALE_GEOMETRIC | - SCALE_EQUILIBRATE; -#endif - - lp->crashmode = CRASH_NONE; - - lp->max_pivots = 0; - lp->simplex_strategy = SIMPLEX_DUAL_PRIMAL; -#define PricerDefaultOpt 1 -#if PricerDefaultOpt == 1 - mode = PRICER_DEVEX; -#elif PricerDefaultOpt == 2 - mode = PRICER_STEEPESTEDGE; - mode |= PRICE_TRUENORMINIT; -#else - mode = PRICER_STEEPESTEDGE | PRICE_PRIMALFALLBACK; -#endif - mode |= PRICE_ADAPTIVE; -#ifdef EnableRandomizedPricing - mode |= PRICE_RANDOMIZE; -#endif - set_pivoting(lp, mode); - - lp->improve = IMPROVE_DEFAULT; - lp->anti_degen = ANTIDEGEN_DEFAULT; - - lp->bb_floorfirst = BRANCH_AUTOMATIC; - lp->bb_rule = NODE_DYNAMICMODE | NODE_GREEDYMODE | NODE_GAPSELECT | -#if 1 - NODE_PSEUDOCOSTSELECT | -#else - NODE_PSEUDOFEASSELECT | -#endif - NODE_RCOSTFIXING; - lp->bb_limitlevel = DEF_BB_LIMITLEVEL; - lp->bb_PseudoUpdates = DEF_PSEUDOCOSTUPDATES; - - lp->bb_heuristicOF = my_chsign(is_maxim(lp), MAX(DEF_INFINITE, lp->infinite)); - lp->bb_breakOF = -lp->bb_heuristicOF; - - lp->sectimeout = 0; - lp->solutionlimit = 1; - - set_outputstream(lp, NULL); /* Set to default output stream */ - lp->verbose = NORMAL; - lp->print_sol = FALSE; /* Can be FALSE, TRUE, AUTOMATIC (only non-zeros printed) */ - lp->spx_trace = FALSE; - lp->lag_trace = FALSE; - lp->bb_trace = FALSE; -} - -void __WINAPI unscale(lprec *lp) -{ - undoscale(lp); -} -int __WINAPI solve(lprec *lp) -{ -#if defined FPUexception - catchFPU(_EM_INVALID | _EM_ZERODIVIDE | _EM_OVERFLOW | _EM_UNDERFLOW); -#endif - - if(has_BFP(lp)) { - lp->solvecount++; - if(is_add_rowmode(lp)) - set_add_rowmode(lp, FALSE); - return(lin_solve(lp)); - } - else - return( NOBFP ); -} -void __WINAPI print_lp(lprec *lp) -{ - REPORT_lp(lp); -} -void __WINAPI print_tableau(lprec *lp) -{ - REPORT_tableau(lp); -} -void __WINAPI print_objective(lprec *lp) -{ - REPORT_objective(lp); -} -void __WINAPI print_solution(lprec *lp, int columns) -{ - REPORT_solution(lp, columns); -} -void __WINAPI print_constraints(lprec *lp, int columns) -{ - REPORT_constraints(lp, columns); -} -void __WINAPI print_duals(lprec *lp) -{ - REPORT_duals(lp); -} -void __WINAPI print_scales(lprec *lp) -{ - REPORT_scales(lp); -} -MYBOOL __WINAPI print_debugdump(lprec *lp, char *filename) -{ - return(REPORT_debugdump(lp, filename, (MYBOOL) (get_total_iter(lp) > 0))); -} -void __WINAPI print_str(lprec *lp, char *str) -{ - report(lp, lp->verbose, "%s", str); -} - - - -/* ---------------------------------------------------------------------------------- */ -/* Parameter setting and retrieval functions */ -/* ---------------------------------------------------------------------------------- */ - -void __WINAPI set_timeout(lprec *lp, long sectimeout) -{ - lp->sectimeout = sectimeout; -} - -long __WINAPI get_timeout(lprec *lp) -{ - return(lp->sectimeout); -} - -void __WINAPI set_verbose(lprec *lp, int verbose) -{ - lp->verbose = verbose; -} - -int __WINAPI get_verbose(lprec *lp) -{ - return(lp->verbose); -} - -void __WINAPI set_print_sol(lprec *lp, int print_sol) -{ - lp->print_sol = print_sol; -} - -int __WINAPI get_print_sol(lprec *lp) -{ - return(lp->print_sol); -} - -void __WINAPI set_debug(lprec *lp, MYBOOL debug) -{ - lp->bb_trace = debug; -} - -MYBOOL __WINAPI is_debug(lprec *lp) -{ - return(lp->bb_trace); -} - -void __WINAPI set_trace(lprec *lp, MYBOOL trace) -{ - lp->spx_trace = trace; -} - -MYBOOL __WINAPI is_trace(lprec *lp) -{ - return(lp->spx_trace); -} - -void __WINAPI set_anti_degen(lprec *lp, int anti_degen) -{ - lp->anti_degen = anti_degen; -} - -int __WINAPI get_anti_degen(lprec *lp) -{ - return(lp->anti_degen); -} - -MYBOOL __WINAPI is_anti_degen(lprec *lp, int testmask) -{ - return((MYBOOL) ((lp->anti_degen == testmask) || ((lp->anti_degen & testmask) != 0))); -} - -void __WINAPI set_presolve(lprec *lp, int presolvemode, int maxloops) -{ - presolvemode &= ~PRESOLVE_REDUCEMIP; /* disable PRESOLVE_REDUCEMIP since it is very rare that this is effective, and also that it adds code complications and delayed presolve effects that are not captured properly. */ - lp->do_presolve = presolvemode; - lp->presolveloops = maxloops; -} - -int __WINAPI get_presolve(lprec *lp) -{ - return(lp->do_presolve); -} - -int __WINAPI get_presolveloops(lprec *lp) -{ - if(lp->presolveloops < 0) - return(DEF_MAXPRESOLVELOOPS); - else if(lp->presolveloops == 0) - return(MAXINT32); - else - return(lp->presolveloops); -} - -MYBOOL __WINAPI is_presolve(lprec *lp, int testmask) -{ - return((MYBOOL) ((lp->do_presolve == testmask) || ((lp->do_presolve & testmask) != 0))); -} - -void __WINAPI set_maxpivot(lprec *lp, int maxpivot) -{ - lp->max_pivots = maxpivot; -} - -int __WINAPI get_maxpivot(lprec *lp) -{ - return( lp->bfp_pivotmax(lp) ); -} - -void __WINAPI set_bb_rule(lprec *lp, int bb_rule) -{ - lp->bb_rule = bb_rule; -} - -int __WINAPI get_bb_rule(lprec *lp) -{ - return(lp->bb_rule); -} - -/* INLINE */ MYBOOL is_bb_rule(lprec *lp, int bb_rule) -{ - return( (MYBOOL) ((lp->bb_rule & NODE_STRATEGYMASK) == bb_rule) ); -} - -/* INLINE */ MYBOOL is_bb_mode(lprec *lp, int bb_mask) -{ - return( (MYBOOL) ((lp->bb_rule & bb_mask) > 0) ); -} - -void __WINAPI set_action(int *actionvar, int actionmask) -{ - *actionvar |= actionmask; -} - -void __WINAPI clear_action(int *actionvar, int actionmask) -{ - *actionvar &= ~actionmask; -} - -MYBOOL __WINAPI is_action(int actionvar, int testmask) -{ - return( (MYBOOL) ((actionvar & testmask) != 0) ); -} - -void __WINAPI set_bb_depthlimit(lprec *lp, int bb_maxlevel) -{ - lp->bb_limitlevel = bb_maxlevel; -} - -int __WINAPI get_bb_depthlimit(lprec *lp) -{ - return(lp->bb_limitlevel); -} - -void __WINAPI set_obj_bound(lprec *lp, REAL bb_heuristicOF) -{ - lp->bb_heuristicOF = bb_heuristicOF; -} - -REAL __WINAPI get_obj_bound(lprec *lp) -{ - return(lp->bb_heuristicOF); -} - -void __WINAPI set_mip_gap(lprec *lp, MYBOOL absolute, REAL mip_gap) -{ - if(absolute) - lp->mip_absgap = mip_gap; - else - lp->mip_relgap = mip_gap; -} - -REAL __WINAPI get_mip_gap(lprec *lp, MYBOOL absolute) -{ - if(absolute) - return(lp->mip_absgap); - else - return(lp->mip_relgap); -} - -MYBOOL __WINAPI set_var_branch(lprec *lp, int colnr, int branch_mode) -{ - if(colnr > lp->columns || colnr < 1) { - report(lp, IMPORTANT, "set_var_branch: Column %d out of range\n", colnr); - return( FALSE ); - } - - if(lp->bb_varbranch == NULL) { - int i; - if(branch_mode == BRANCH_DEFAULT) - return( TRUE ); - allocMYBOOL(lp, &lp->bb_varbranch, lp->columns_alloc, FALSE); - for(i = 0; i < lp->columns; i++) - lp->bb_varbranch[i] = BRANCH_DEFAULT; - } - lp->bb_varbranch[colnr - 1] = (MYBOOL) branch_mode; - return( TRUE ); -} - -int __WINAPI get_var_branch(lprec *lp, int colnr) -{ - if(colnr > lp->columns || colnr < 1) { - report(lp, IMPORTANT, "get_var_branch: Column %d out of range\n", colnr); - return(lp->bb_floorfirst); - } - - if(lp->bb_varbranch == NULL) - return(lp->bb_floorfirst); - if(lp->bb_varbranch[colnr - 1] == BRANCH_DEFAULT) - return(lp->bb_floorfirst); - else - return(lp->bb_varbranch[colnr - 1]); -} - -static void set_infiniteex(lprec *lp, REAL infinite, MYBOOL init) -{ - int i; - - infinite = fabs(infinite); - if((init) || is_infinite(lp, lp->bb_heuristicOF)) - lp->bb_heuristicOF = my_chsign(is_maxim(lp), infinite); - if((init) || is_infinite(lp, lp->bb_breakOF)) - lp->bb_breakOF = my_chsign(is_maxim(lp), -infinite); - for(i = 0; i <= lp->sum; i++) { - if((!init) && is_infinite(lp, lp->orig_lowbo[i])) - lp->orig_lowbo[i] = -infinite; - if((init) || is_infinite(lp, lp->orig_upbo[i])) - lp->orig_upbo[i] = infinite; - } - lp->infinite = infinite; -} - - -MYBOOL __WINAPI is_infinite(lprec *lp, REAL value) -{ -#if 1 - return( (MYBOOL) (fabs(value) >= lp->infinite) ); -#else - if(fabs(value) >= lp->infinite) - return( TRUE ); - else - return( FALSE ); -#endif -} - -void __WINAPI set_infinite(lprec *lp, REAL infinite) -{ - set_infiniteex(lp, infinite, FALSE); -} - -REAL __WINAPI get_infinite(lprec *lp) -{ - return(lp->infinite); -} - -void __WINAPI set_epsperturb(lprec *lp, REAL epsperturb) -{ - lp->epsperturb = epsperturb; -} - -REAL __WINAPI get_epsperturb(lprec *lp) -{ - return(lp->epsperturb); -} - -void __WINAPI set_epspivot(lprec *lp, REAL epspivot) -{ - lp->epspivot = epspivot; -} - -REAL __WINAPI get_epspivot(lprec *lp) -{ - return(lp->epspivot); -} - -void __WINAPI set_epsint(lprec *lp, REAL epsint) -{ - lp->epsint = epsint; -} - -REAL __WINAPI get_epsint(lprec *lp) -{ - return(lp->epsint); -} - -void __WINAPI set_epsb(lprec *lp, REAL epsb) -{ - lp->epsprimal = MAX(epsb, lp->epsmachine); -} - -REAL __WINAPI get_epsb(lprec *lp) -{ - return(lp->epsprimal); -} - -void __WINAPI set_epsd(lprec *lp, REAL epsd) -{ - lp->epsdual = MAX(epsd, lp->epsmachine); /* Mainly used as tolerance for reduced cost */ -} - -REAL __WINAPI get_epsd(lprec *lp) -{ - return(lp->epsdual); -} - -void __WINAPI set_epsel(lprec *lp, REAL epsel) -{ - lp->epsvalue = MAX(epsel, lp->epsmachine); -} - -REAL __WINAPI get_epsel(lprec *lp) -{ - return(lp->epsvalue); -} - -MYBOOL __WINAPI set_epslevel(lprec *lp, int epslevel) -{ - REAL SPX_RELAX, MIP_RELAX; - - switch(epslevel) { - case EPS_TIGHT: SPX_RELAX = 1; - MIP_RELAX = 1; - break; - case EPS_MEDIUM: SPX_RELAX = 10; - MIP_RELAX = 1; - break; - case EPS_LOOSE: SPX_RELAX = 100; - MIP_RELAX = 10; - break; - case EPS_BAGGY: SPX_RELAX = 1000; - MIP_RELAX = 100; - break; - default: return( FALSE ); - } - lp->epsvalue = SPX_RELAX*DEF_EPSVALUE; - lp->epsprimal = SPX_RELAX*DEF_EPSPRIMAL; - lp->epsdual = SPX_RELAX*DEF_EPSDUAL; - lp->epspivot = SPX_RELAX*DEF_EPSPIVOT; - lp->epssolution= MIP_RELAX*DEF_EPSSOLUTION; - lp->epsint = MIP_RELAX*DEF_EPSINT; - lp->mip_absgap = MIP_RELAX*DEF_MIP_GAP; - lp->mip_relgap = MIP_RELAX*DEF_MIP_GAP; - - return( TRUE ); -} - -void __WINAPI set_scaling(lprec *lp, int scalemode) -{ - lp->scalemode = scalemode; -} - -int __WINAPI get_scaling(lprec *lp) -{ - return(lp->scalemode); -} - -MYBOOL __WINAPI is_scalemode(lprec *lp, int testmask) -{ - return((MYBOOL) ((lp->scalemode & testmask) != 0)); -} - -MYBOOL __WINAPI is_scaletype(lprec *lp, int scaletype) -{ - int testtype; - - testtype = lp->scalemode & SCALE_MAXTYPE; - return((MYBOOL) (scaletype == testtype)); -} - -void __WINAPI set_scalelimit(lprec *lp, REAL scalelimit) -/* Set the relative scaling convergence criterion for the active scaling mode; - the integer part specifies the maximum number of iterations (default = 5). */ -{ - lp->scalelimit = fabs(scalelimit); -} - -REAL __WINAPI get_scalelimit(lprec *lp) -{ - return(lp->scalelimit); -} - -MYBOOL __WINAPI is_integerscaling(lprec *lp) -{ - return(is_scalemode(lp, SCALE_INTEGERS)); -} - -void __WINAPI set_improve(lprec *lp, int improve) -{ - lp->improve = improve; -} - -int __WINAPI get_improve(lprec *lp) -{ - return(lp->improve); -} - -void __WINAPI set_lag_trace(lprec *lp, MYBOOL lag_trace) -{ - lp->lag_trace = lag_trace; -} - -MYBOOL __WINAPI is_lag_trace(lprec *lp) -{ - return(lp->lag_trace); -} - -void __WINAPI set_pivoting(lprec *lp, int pivoting) -{ - /* Set new pivoting strategy */ - lp->piv_strategy = pivoting; - report(lp, DETAILED, "set_pivoting: Pricing strategy set to '%s'\n", - get_str_piv_rule(get_piv_rule(lp))); -} - -int __WINAPI get_pivoting(lprec *lp) -{ - return( lp->piv_strategy ); -} - -/* INLINE */ int get_piv_rule(lprec *lp) -{ - return( (lp->piv_strategy | PRICE_STRATEGYMASK) ^ PRICE_STRATEGYMASK ); -} - -STATIC char *get_str_piv_rule(int rule) -{ - static char *pivotText[PRICER_LASTOPTION+1] = - {"Bland first index", "Dantzig", "Devex", "Steepest Edge"}; - - return( pivotText[rule] ); -} - -MYBOOL __WINAPI is_piv_rule(lprec *lp, int rule) -{ - return( (MYBOOL) (get_piv_rule(lp) == rule) ); -} - -MYBOOL __WINAPI is_piv_mode(lprec *lp, int testmask) -{ - return((MYBOOL) (((testmask & PRICE_STRATEGYMASK) != 0) && - ((lp->piv_strategy & testmask) != 0))); -} - -void __WINAPI set_break_at_first(lprec *lp, MYBOOL break_at_first) -{ - lp->bb_breakfirst = break_at_first; -} - -MYBOOL __WINAPI is_break_at_first(lprec *lp) -{ - return(lp->bb_breakfirst); -} - -void __WINAPI set_bb_floorfirst(lprec *lp, int bb_floorfirst) -{ - lp->bb_floorfirst = (MYBOOL) bb_floorfirst; -} - -int __WINAPI get_bb_floorfirst(lprec *lp) -{ - return(lp->bb_floorfirst); -} - -void __WINAPI set_break_at_value(lprec *lp, REAL break_at_value) -{ - lp->bb_breakOF = break_at_value; -} - -REAL __WINAPI get_break_at_value(lprec *lp) -{ - return(lp->bb_breakOF); -} - -void __WINAPI set_negrange(lprec *lp, REAL negrange) -{ - if(negrange <= 0) - lp->negrange = negrange; - else - lp->negrange = 0.0; -} - -REAL __WINAPI get_negrange(lprec *lp) -{ - return(lp->negrange); -} - -int __WINAPI get_max_level(lprec *lp) -{ - return(lp->bb_maxlevel); -} - -COUNTER __WINAPI get_total_nodes(lprec *lp) -{ - return(lp->bb_totalnodes); -} - -COUNTER __WINAPI get_total_iter(lprec *lp) -{ - return(lp->total_iter + lp->current_iter); -} - -REAL __WINAPI get_objective(lprec *lp) -{ - if(lp->spx_status == OPTIMAL) - ; - else if(!lp->basis_valid) { - report(lp, CRITICAL, "get_objective: Not a valid basis\n"); - return(0.0); - } - - return( lp->best_solution[0] ); -} - -int __WINAPI get_nonzeros(lprec *lp) -{ - return( mat_nonzeros(lp->matA) ); -} - -MYBOOL __WINAPI set_mat(lprec *lp, int rownr, int colnr, REAL value) -{ - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "set_mat: Row %d out of range\n", rownr); - return( FALSE ); - } - if((colnr < 1) || (colnr > lp->columns)) { - report(lp, IMPORTANT, "set_mat: Column %d out of range\n", colnr); - return( FALSE ); - } - -#ifdef DoMatrixRounding - if(rownr == 0) - value = roundToPrecision(value, lp->matA->epsvalue); -#endif - value = scaled_mat(lp, value, rownr, colnr); - if(rownr == 0) { - lp->orig_obj[colnr] = my_chsign(is_chsign(lp, rownr), value); - return( TRUE ); - } - else - return( mat_setvalue(lp->matA, rownr, colnr, value, FALSE) ); -} - -REAL __WINAPI get_working_objective(lprec *lp) -{ - REAL value = 0.0; - - if(!lp->basis_valid) - report(lp, CRITICAL, "get_working_objective: Not a valid basis\n"); - else if((lp->spx_status == RUNNING) && (lp->solutioncount == 0)) - value = my_chsign(!is_maxim(lp), lp->rhs[0]); - else - value = lp->solution[0]; - - return(value); -} - -REAL __WINAPI get_var_primalresult(lprec *lp, int index) -{ - if((index < 0) || (index > lp->presolve_undo->orig_sum)) { - report(lp, IMPORTANT, "get_var_primalresult: Index %d out of range\n", index); - return( 0.0 ); - } - if((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE) - return( lp->full_solution[index] ); - else - return( lp->best_solution[index] ); -} - -REAL __WINAPI get_var_dualresult(lprec *lp, int index) -{ - REAL *duals; - - if((index < 0) || (index > lp->presolve_undo->orig_sum)) { - report(lp, IMPORTANT, "get_var_dualresult: Index %d out of range\n", index); - return( 0.0 ); - } - - if(index == 0) - return( lp->best_solution[0] ); - - /* Make sure we actually have dual information available */ - if(!get_ptr_sensitivity_rhs(lp, &duals, NULL, NULL)) - return( 0.0 ); - else - duals = ((lp->full_duals == NULL) ? lp->duals : lp->full_duals); - return( duals[index] ); -} - -MYBOOL __WINAPI get_variables(lprec *lp, REAL *var) -{ - if(lp->spx_status == OPTIMAL) - ; - else if(!lp->basis_valid) { - report(lp, CRITICAL, "get_variables: Not a valid basis\n"); - return(FALSE); - } - - MEMCOPY(var, lp->best_solution + (1 + lp->rows), lp->columns); - return(TRUE); -} - -MYBOOL __WINAPI get_ptr_variables(lprec *lp, REAL **var) -{ - if(lp->spx_status == OPTIMAL) - ; - else if(!lp->basis_valid) { - report(lp, CRITICAL, "get_ptr_variables: Not a valid basis\n"); - return(FALSE); - } - - if(var != NULL) - *var = lp->best_solution + (1 + lp->rows); - return(TRUE); -} - -MYBOOL __WINAPI get_constraints(lprec *lp, REAL *constr) -{ - if(lp->spx_status == OPTIMAL) - ; - else if(!lp->basis_valid) { - report(lp, CRITICAL, "get_constraints: Not a valid basis\n"); - return(FALSE); - } - - MEMCOPY(constr, lp->best_solution + 1, lp->rows); - return(TRUE); -} - -MYBOOL __WINAPI get_ptr_constraints(lprec *lp, REAL **constr) -{ - if(lp->spx_status == OPTIMAL) - ; - else if(!lp->basis_valid) { - report(lp, CRITICAL, "get_ptr_constraints: Not a valid basis\n"); - return(FALSE); - } - - if(constr != NULL) - *constr = lp->best_solution + 1; - return(TRUE); -} - -MYBOOL __WINAPI get_sensitivity_rhs(lprec *lp, REAL *duals, REAL *dualsfrom, REAL *dualstill) -{ - REAL *duals0, *dualsfrom0, *dualstill0; - - if(!lp->basis_valid) { - report(lp, CRITICAL, "get_sensitivity_rhs: Not a valid basis\n"); - return(FALSE); - } - - if(!get_ptr_sensitivity_rhs(lp, - (duals != NULL) ? &duals0 : NULL, - (dualsfrom != NULL) ? &dualsfrom0 : NULL, - (dualstill != NULL) ? &dualstill0 : NULL)) - return(FALSE); - - if(duals != NULL) - MEMCOPY(duals, duals0, lp->sum); - if(dualsfrom != NULL) - MEMCOPY(dualsfrom, dualsfrom0, lp->sum); - if(dualstill != NULL) - MEMCOPY(dualstill, dualstill0, lp->sum); - return(TRUE); -} - -MYBOOL __WINAPI get_ptr_sensitivity_rhs(lprec *lp, REAL **duals, REAL **dualsfrom, REAL **dualstill) -{ - if(!lp->basis_valid) { - report(lp, CRITICAL, "get_ptr_sensitivity_rhs: Not a valid basis\n"); - return(FALSE); - } - - if(duals != NULL) { - if(lp->duals == NULL) { - if((MIP_count(lp) > 0) && (lp->bb_totalnodes > 0)) { - report(lp, CRITICAL, "get_ptr_sensitivity_rhs: Sensitivity unknown\n"); - return(FALSE); - } - if(!construct_duals(lp)) - return(FALSE); - } - *duals = lp->duals + 1; - } - - if((dualsfrom != NULL) || (dualstill != NULL)) { - if((lp->dualsfrom == NULL) || (lp->dualstill == NULL)) { - if((MIP_count(lp) > 0) && (lp->bb_totalnodes > 0)) { - report(lp, CRITICAL, "get_ptr_sensitivity_rhs: Sensitivity unknown\n"); - return(FALSE); - } - construct_sensitivity_duals(lp); - if((lp->dualsfrom == NULL) || (lp->dualstill == NULL)) - return(FALSE); - } - if(dualsfrom != NULL) - *dualsfrom = lp->dualsfrom + 1; - if(dualstill != NULL) - *dualstill = lp->dualstill + 1; - } - return(TRUE); -} - -MYBOOL __WINAPI get_sensitivity_objex(lprec *lp, REAL *objfrom, REAL *objtill, REAL *objfromvalue, REAL *objtillvalue) -{ - REAL *objfrom0, *objtill0, *objfromvalue0, *objtillvalue0; - - if(!lp->basis_valid) { - report(lp, CRITICAL, "get_sensitivity_objex: Not a valid basis\n"); - return(FALSE); - } - - if(!get_ptr_sensitivity_objex(lp, (objfrom != NULL) ? &objfrom0 : NULL, - (objtill != NULL) ? &objtill0 : NULL, - (objfromvalue != NULL) ? &objfromvalue0 : NULL, - (objtillvalue != NULL) ? &objtillvalue0 : NULL)) - return(FALSE); - - if((objfrom != NULL) && (objfrom0 != NULL)) - MEMCOPY(objfrom, objfrom0, lp->columns); - if((objtill != NULL) && (objtill0 != NULL)) - MEMCOPY(objtill, objtill0, lp->columns); - if((objfromvalue != NULL) && (objfromvalue0 != NULL)) - MEMCOPY(objfromvalue, objfromvalue0, lp->columns); - if((objtillvalue != NULL) && (objtillvalue0 != NULL)) - MEMCOPY(objtillvalue, objtillvalue0, lp->columns); - return(TRUE); -} - -MYBOOL __WINAPI get_sensitivity_obj(lprec *lp, REAL *objfrom, REAL *objtill) -{ - return(get_sensitivity_objex(lp, objfrom, objtill, NULL, NULL)); -} - -MYBOOL __WINAPI get_ptr_sensitivity_objex(lprec *lp, REAL **objfrom, REAL **objtill, REAL **objfromvalue, REAL **objtillvalue) -{ - if(!lp->basis_valid) { - report(lp, CRITICAL, "get_ptr_sensitivity_objex: Not a valid basis\n"); - return(FALSE); - } - - if((objfrom != NULL) || (objtill != NULL)) { - if((lp->objfrom == NULL) || (lp->objtill == NULL)) { - if((MIP_count(lp) > 0) && (lp->bb_totalnodes > 0)) { - report(lp, CRITICAL, "get_ptr_sensitivity_objex: Sensitivity unknown\n"); - return(FALSE); - } - construct_sensitivity_obj(lp); - if((lp->objfrom == NULL) || (lp->objtill == NULL)) - return(FALSE); - } - if(objfrom != NULL) - *objfrom = lp->objfrom + 1; - if(objtill != NULL) - *objtill = lp->objtill + 1; - } - - if((objfromvalue != NULL) /* || (objtillvalue != NULL) */) { - if((lp->objfromvalue == NULL) /* || (lp->objtillvalue == NULL) */) { - if((MIP_count(lp) > 0) && (lp->bb_totalnodes > 0)) { - report(lp, CRITICAL, "get_ptr_sensitivity_objex: Sensitivity unknown\n"); - return(FALSE); - } - construct_sensitivity_duals(lp); - if((lp->objfromvalue == NULL) /* || (lp->objtillvalue == NULL) */) - return(FALSE); - } - } - - if(objfromvalue != NULL) - *objfromvalue = lp->objfromvalue + 1; - - if(objtillvalue != NULL) - *objtillvalue = NULL /* lp->objtillvalue + 1 */; - - return(TRUE); -} - -MYBOOL __WINAPI get_ptr_sensitivity_obj(lprec *lp, REAL **objfrom, REAL **objtill) -{ - return(get_ptr_sensitivity_objex(lp, objfrom, objtill, NULL, NULL)); -} - -void __WINAPI set_solutionlimit(lprec *lp, int limit) -{ - lp->solutionlimit = limit; -} -int __WINAPI get_solutionlimit(lprec *lp) -{ - return(lp->solutionlimit); -} -int __WINAPI get_solutioncount(lprec *lp) -{ - return(lp->solutioncount); -} - -int __WINAPI get_Nrows(lprec *lp) -{ - return(lp->rows); -} - -int __WINAPI get_Norig_rows(lprec *lp) -{ - if(lp->varmap_locked) - return(lp->presolve_undo->orig_rows); - else - return(lp->rows); -} - -int __WINAPI get_Lrows(lprec *lp) -{ - if(lp->matL == NULL) - return( 0 ); - else - return( lp->matL->rows ); -} - -int __WINAPI get_Ncolumns(lprec *lp) -{ - return(lp->columns); -} - -int __WINAPI get_Norig_columns(lprec *lp) -{ - if(lp->varmap_locked) - return(lp->presolve_undo->orig_columns); - else - return(lp->columns); -} - - -/* ---------------------------------------------------------------------------------- */ -/* Core routines for lp_solve */ -/* ---------------------------------------------------------------------------------- */ -int __WINAPI get_status(lprec *lp) -{ - return(lp->spx_status); -} - -char * __WINAPI get_statustext(lprec *lp, int statuscode) -{ - if (statuscode == NOBFP) return("No basis factorization package"); - else if (statuscode == DATAIGNORED) return("Invalid input data provided"); - else if (statuscode == NOMEMORY) return("Not enough memory available"); - else if (statuscode == NOTRUN) return("Model has not been optimized"); - else if (statuscode == OPTIMAL) return("OPTIMAL solution"); - else if (statuscode == SUBOPTIMAL) return("SUB-OPTIMAL solution"); - else if (statuscode == INFEASIBLE) return("Model is primal INFEASIBLE"); - else if (statuscode == UNBOUNDED) return("Model is primal UNBOUNDED"); - else if (statuscode == RUNNING) return("lp_solve is currently running"); - else if (statuscode == NUMFAILURE) return("NUMERIC FAILURE encountered"); - else if (statuscode == DEGENERATE) return("DEGENERATE situation"); - else if (statuscode == USERABORT) return("User-requested termination"); - else if (statuscode == TIMEOUT) return("Termination due to timeout"); - else if (statuscode == PRESOLVED) return("Model solved by presolve"); - else if (statuscode == PROCFAIL) return("B&B routine failed"); - else if (statuscode == PROCBREAK) return("B&B routine terminated"); - else if (statuscode == FEASFOUND) return("Feasible B&B solution found"); - else if (statuscode == NOFEASFOUND) return("No feasible B&B solution found"); - else if (statuscode == FATHOMED) return("Fathomed/pruned branch"); - else return("Undefined internal error"); -} - -MYBOOL __WINAPI is_obj_in_basis(lprec *lp) -{ - return( lp->obj_in_basis ); -} - -void __WINAPI set_obj_in_basis(lprec *lp, MYBOOL obj_in_basis) -{ - lp->obj_in_basis = (MYBOOL) (obj_in_basis == TRUE); -} - -lprec * __WINAPI make_lp(int rows, int columns) -{ - lprec *lp; - -# if defined FORTIFY - /* Fortify_EnterScope(); */ -# endif - - if(rows < 0 || columns < 0) - return(NULL); - - lp = (lprec*) calloc(1, sizeof(*lp)); - if(!lp) - return(NULL); - - set_lp_name(lp, NULL); - lp->names_used = FALSE; - lp->use_row_names = TRUE; - lp->use_col_names = TRUE; - lp->rowcol_name = NULL; - - /* Do standard initializations ------------------------------------------------------------ */ -#if 1 - lp->obj_in_basis = DEF_OBJINBASIS; -#else - lp->obj_in_basis = FALSE; -#endif - lp->verbose = NORMAL; - set_callbacks(lp); - set_BFP(lp, NULL); - set_XLI(lp, NULL); -#if libBLAS > 0 - init_BLAS(); -#if libBLAS > 1 - if(is_nativeBLAS() && !load_BLAS(libnameBLAS)) - /*report(lp, "make_lp: Could not load external BLAS library '%s'.\n", libnameBLAS)*/; -#endif -#endif - - /* Define the defaults for key user-settable values --------------------------------------- */ - reset_params(lp); - - /* Do other initializations --------------------------------------------------------------- */ - lp->source_is_file = FALSE; - lp->model_is_pure = TRUE; - lp->model_is_valid = FALSE; - lp->spx_status = NOTRUN; - lp->lag_status = NOTRUN; - - lp->workarrays = mempool_create(lp); - lp->wasPreprocessed = FALSE; - lp->wasPresolved = FALSE; - presolve_createUndo(lp); - - lp->bb_varactive = NULL; - lp->bb_varbranch = NULL; - lp->var_priority = NULL; - - lp->rhsmax = 0.0; - lp->bigM = 0.0; - lp->bb_deltaOF = 0.0; - - lp->equalities = 0; - lp->fixedvars = 0; - lp->int_vars = 0; - lp->sc_vars = 0; - - lp->sos_ints = 0; - lp->sos_vars = 0; - lp->sos_priority = NULL; - - lp->rows_alloc = 0; - lp->columns_alloc = 0; - lp->sum_alloc = 0; - - lp->rows = rows; - lp->columns = columns; - lp->sum = rows + columns; - varmap_clear(lp); - - lp->matA = mat_create(lp, rows, columns, lp->epsvalue); - lp->matL = NULL; - lp->invB = NULL; - lp->duals = NULL; - lp->dualsfrom = NULL; - lp->dualstill = NULL; - lp->objfromvalue = NULL; - lp->objfrom = NULL; - lp->objtill = NULL; - - inc_col_space(lp, columns + 1); - inc_row_space(lp, rows + 1); - - /* Avoid bound-checker uninitialized variable error */ - lp->orig_lowbo[0] = 0; - - lp->rootbounds = NULL; - lp->bb_bounds = NULL; - lp->bb_basis = NULL; - - lp->basis_valid = FALSE; - lp->simplex_mode = SIMPLEX_DYNAMIC; - lp->scaling_used = FALSE; - lp->columns_scaled = FALSE; - lp->P1extraDim = 0; - lp->P1extraVal = 0.0; - lp->bb_strongbranches = 0; - lp->current_iter = 0; - lp->total_iter = 0; - lp->current_bswap = 0; - lp->total_bswap = 0; - lp->solutioncount = 0; - lp->solvecount = 0; - - allocINT(lp, &lp->rejectpivot, DEF_MAXPIVOTRETRY + 1, TRUE); - - set_minim(lp); - set_infiniteex(lp, DEF_INFINITE, TRUE); - /* set_break_numeric_accuracy(lp, DEF_INFINITE); */ - /* set_break_numeric_accuracy(lp, 1e-5); */ - /* set_break_numeric_accuracy(lp, 5e-5); */ - /* set_break_numeric_accuracy(lp, 1e-6); */ - /* set_break_numeric_accuracy(lp, 5e-6); */ - set_break_numeric_accuracy(lp, 5e-7); - - initPricer(lp); - - /* Call-back routines by KE */ - lp->ctrlc = NULL; - lp->ctrlchandle = NULL; - lp->writelog = NULL; - lp->loghandle = NULL; - lp->debuginfo = NULL; - lp->usermessage = NULL; - lp->msgmask = MSG_NONE; - lp->msghandle = NULL; - - lp->timecreate = timeNow(); - - return(lp); -} - -MYBOOL __WINAPI resize_lp(lprec *lp, int rows, int columns) -{ - MYBOOL status = TRUE; - - if(columns > lp->columns) - status = inc_col_space(lp, columns - lp->columns); - else - while(status && (lp->columns > columns)) { - status = del_column(lp, lp->columns); - } - if(status && (rows > lp->rows)) - status = inc_row_space(lp, rows - lp->rows); - else - while(status && (lp->rows > rows)) { - status = del_constraint(lp, lp->rows); - } - return( status ); -} - -void __WINAPI free_lp(lprec **plp) -{ - if(plp != NULL) { - lprec *lp = *plp; - if(lp != NULL) - delete_lp(lp); - *plp = NULL; - } -} - -void __WINAPI delete_lp(lprec *lp) -{ - if(lp == NULL) - return; - - FREE(lp->rowcol_name); - FREE(lp->lp_name); - FREE(lp->ex_status); - if(lp->names_used) { - FREE(lp->row_name); - FREE(lp->col_name); - free_hash_table(lp->rowname_hashtab); - free_hash_table(lp->colname_hashtab); - } - - mat_free(&lp->matA); - lp->bfp_free(lp); -#if LoadInverseLib == TRUE - if(lp->hBFP != NULL) - set_BFP(lp, NULL); -#endif -#if LoadLanguageLib == TRUE - if(lp->hXLI != NULL) - set_XLI(lp, NULL); -#endif - - unset_OF_p1extra(lp); - FREE(lp->orig_obj); - FREE(lp->orig_rhs); - FREE(lp->rhs); - FREE(lp->var_type); - set_var_weights(lp, NULL); - FREE(lp->bb_varbranch); - FREE(lp->sc_lobound); - FREE(lp->var_is_free); - FREE(lp->orig_upbo); - FREE(lp->orig_lowbo); - FREE(lp->upbo); - FREE(lp->lowbo); - FREE(lp->var_basic); - FREE(lp->is_basic); - FREE(lp->is_lower); - if(lp->bb_PseudoCost != NULL) { -/* report(lp, SEVERE, "delete_lp: The B&B pseudo-cost array was not cleared on delete\n"); */ - free_pseudocost(lp); - } - if(lp->bb_bounds != NULL) { - report(lp, SEVERE, "delete_lp: The stack of B&B levels was not empty (failed at %.0f nodes)\n", - (double) lp->bb_totalnodes); - unload_BB(lp); - } - if(lp->bb_basis != NULL) { -/* report(lp, SEVERE, "delete_lp: The stack of saved bases was not empty on delete\n"); */ - unload_basis(lp, FALSE); - } - - FREE(lp->rejectpivot); - partial_freeBlocks(&(lp->rowblocks)); - partial_freeBlocks(&(lp->colblocks)); - multi_free(&(lp->multivars)); - multi_free(&(lp->longsteps)); - - FREE(lp->solution); - FREE(lp->best_solution); - FREE(lp->full_solution); - - presolve_freeUndo(lp); - mempool_free(&(lp->workarrays)); - - freePricer(lp); - - FREE(lp->drow); - FREE(lp->nzdrow); - - FREE(lp->duals); - FREE(lp->full_duals); - FREE(lp->dualsfrom); - FREE(lp->dualstill); - FREE(lp->objfromvalue); - FREE(lp->objfrom); - FREE(lp->objtill); - FREE(lp->row_type); - - if(lp->sos_vars > 0) - FREE(lp->sos_priority); - free_SOSgroup(&(lp->SOS)); - free_SOSgroup(&(lp->GUB)); - freecuts_BB(lp); - - if(lp->scaling_used) - FREE(lp->scalars); - if(lp->matL != NULL) { - FREE(lp->lag_rhs); - FREE(lp->lambda); - FREE(lp->lag_con_type); - mat_free(&lp->matL); - } - if(lp->streamowned) - set_outputstream(lp, NULL); - -#if libBLAS > 0 - if(!is_nativeBLAS()) - unload_BLAS(); -#endif - - FREE(lp); - -# if defined FORTIFY - /* Fortify_LeaveScope(); */ -# endif -} - -static MYBOOL get_SOS(lprec *lp, int index, char *name, int *sostype, int *priority, int *count, int *sosvars, REAL *weights) -{ - SOSrec *SOS; - - if((index < 1) || (index > SOS_count(lp))) - return( FALSE ); - SOS = lp->SOS->sos_list[index-1]; - if(name != NULL) - strcpy(name, SOS->name); - if(sostype != NULL) - *sostype = SOS->type; - if(priority != NULL) - *priority = SOS->priority; - if(count != NULL) { - *count = SOS->size; - if(sosvars != NULL) { - int i; - for(i = 1; i <= *count; i++) { - sosvars[i-1] = SOS->members[i]; - if(weights != NULL) - weights[i-1] = SOS->weights[i]; - } - } - } - return( TRUE ); -} - -/* Make a copy of the existing model using (mostly) high-level - construction routines to simplify future maintainance. */ -lprec* __WINAPI copy_lp(lprec *lp) -{ - int i, n, *idx = NULL; - REAL hold, *val = NULL, infinite; - lprec *newlp = NULL; - char buf[256], ok = FALSE; - int sostype, priority, count, *sosvars, rows, columns; - REAL *weights = NULL; - -#if 0 - if(lp->wasPresolved) - return( newlp ); -#endif - - rows = get_Nrows(lp); - columns = get_Ncolumns(lp); - - if(!allocINT(lp, &idx, rows+1, FALSE) || - !allocREAL(lp, &val, rows+1, FALSE)) - goto Finish; - - /* Create the new object */ - newlp = make_lp(rows, 0); - if(newlp == NULL) - goto Finish; - if(!resize_lp(newlp, rows, columns)) - goto Finish; - set_sense(newlp, is_maxim(lp)); - set_use_names(newlp, FALSE, is_use_names(lp, FALSE)); - set_use_names(newlp, TRUE, is_use_names(lp, TRUE)); - if(!set_lp_name(newlp, get_lp_name(lp))) - goto Finish; - /* set_algopt(newlp, get_algopt(lp)); */ /* v6 */ - set_verbose(newlp, get_verbose(lp)); - - /* Transfer standard simplex parameters */ - set_epspivot(newlp, get_epspivot(lp)); - set_epsel(newlp, get_epsel(lp)); - set_epsb(newlp, get_epsb(lp)); - set_epsd(newlp, get_epsd(lp)); - set_pivoting(newlp, get_pivoting(lp)); - set_negrange(newlp, lp->negrange); - set_infinite(newlp, get_infinite(lp)); - set_presolve(newlp, get_presolve(lp), get_presolveloops(lp)); - set_scaling(newlp, get_scaling(lp)); - set_scalelimit(newlp, get_scalelimit(lp)); - set_simplextype(newlp, get_simplextype(lp)); - set_epsperturb(newlp, get_epsperturb(lp)); - set_anti_degen(newlp, get_anti_degen(lp)); - set_improve(newlp, get_improve(lp)); - set_basiscrash(newlp, get_basiscrash(lp)); - set_maxpivot(newlp, get_maxpivot(lp)); - set_timeout(newlp, get_timeout(lp)); - - /* Transfer MILP parameters */ - set_epsint(newlp, get_epsint(lp)); - set_bb_rule(newlp, get_bb_rule(lp)); - set_bb_depthlimit(newlp, get_bb_depthlimit(lp)); - set_bb_floorfirst(newlp, get_bb_floorfirst(lp)); - set_mip_gap(newlp, TRUE, get_mip_gap(lp, TRUE)); - set_mip_gap(newlp, FALSE, get_mip_gap(lp, FALSE)); - set_break_at_first(newlp, is_break_at_first(lp)); - set_break_at_value(newlp, get_break_at_value(lp)); - - /* Set RHS and range */ - infinite = get_infinite(lp); - for(i = 0; i <= rows; i++) { - if(i > 0) - if(!set_constr_type(newlp, i, get_constr_type(lp, i))) - goto Finish; - if(!set_rh(newlp, i, get_rh(lp, i))) - goto Finish; - if((i > 0) && ((hold = get_rh_range(lp, i)) < infinite)) - if(!set_rh_range(newlp, i, hold)) - goto Finish; - if(lp->names_used && lp->use_row_names && (lp->row_name[i] != NULL) && (lp->row_name[i]->name != NULL)) - if(!set_row_name(newlp, i, get_row_name(lp, i))) - goto Finish; - } - - /* Load the constraint matrix and variable definitions */ - for(i = 1; i <= columns; i++) { - n = get_columnex(lp, i, val, idx); - if ((n < 0) || (!add_columnex(newlp, n, val, idx))) - goto Finish; - if(is_binary(lp, i)) { - if (!set_binary(newlp, i, TRUE)) - goto Finish; - } - else { - if(is_int(lp, i)) - if(!set_int(newlp, i, TRUE)) - goto Finish; - if((hold = get_lowbo(lp, i)) != 0) - if(!set_lowbo(newlp, i, hold)) - goto Finish; - if((hold = get_upbo(lp, i)) < infinite) - if(!set_upbo(newlp, i, hold)) - goto Finish; - } - if(is_semicont(lp, i)) - if(!set_semicont(newlp, i, TRUE)) - goto Finish; - if(lp->names_used && lp->use_col_names && (lp->col_name[i] != NULL) && (lp->col_name[i]->name != NULL)) - if(!set_col_name(newlp, i, get_col_name(lp, i))) - goto Finish; - } - - /* copy SOS data */ - for(i = 1; get_SOS(lp, i, buf, &sostype, &priority, &count, NULL, NULL); i++) - if (count) { - if(!allocINT(lp, &sosvars, count, FALSE) || - !allocREAL(lp, &weights, count, FALSE)) - n = 0; - else { - get_SOS(lp, i, buf, &sostype, &priority, &count, sosvars, weights); - n = add_SOS(newlp, buf, sostype, priority, count, sosvars, weights); - } - FREE(weights); - FREE(sosvars); - if(n == 0) - goto Finish; - } - -#if 0 - /* Other parameters set if the source model was previously solved */ - if(lp->solvecount > 0) { - MEMCOPY(newlp->scalars, lp->scalars, lp->sum+1); - MEMCOPY(newlp->var_basic, lp->var_basic, rows+1); - MEMCOPY(newlp->is_basic, lp->is_basic, lp->sum+1); - MEMCOPY(newlp->is_lower, lp->is_lower, lp->sum+1); - MEMCOPY(newlp->solution, lp->solution, lp->sum+1); - if(lp->duals != NULL) { - allocREAL(newlp, &newlp->duals, newlp->sum_alloc+1, FALSE); - MEMCOPY(newlp->duals, lp->duals, lp->sum+1); - } - newlp->solutioncount = lp->solutioncount; - newlp->solvecount = lp->solvecount; - } -#endif - - ok = TRUE; - - /* Clean up before returning */ -Finish: - if(!ok) - free_lp(&newlp); - FREE(val); - FREE(idx); - - return( newlp ); -} -MYBOOL __WINAPI dualize_lp(lprec *lp) -{ - int i, n; - MATrec *mat = lp->matA; - REAL *item; - - /* Are we allowed to perform the operation? */ - if((MIP_count(lp) > 0) || (lp->solvecount > 0)) - return( FALSE ); - - /* Modify sense */ - set_sense(lp, (MYBOOL) !is_maxim(lp)); - - /* Transpose matrix and reverse signs */ - n = mat_nonzeros(mat); - mat_transpose(mat); - item = &COL_MAT_VALUE(0); - for(i = 0; i < n; i++, item += matValueStep) - *item *= -1; - - /* Row-column swap other vectors */ - swapINT(&lp->rows, &lp->columns); - swapINT(&lp->rows_alloc, &lp->columns_alloc); - swapREAL(lp->orig_rhs, lp->orig_obj); - if ((lp->rhs != NULL) && (lp->obj != NULL)) - swapREAL(lp->rhs, lp->obj); - - /* Reallocate storage */ -/* -var_type -sc_bound -solution -best_solution -full_solution -duals -*/ - - /* Shift variable bounds */ -/* -is_basic -orig_upbo -orig_lowbo -scalars -*/ - - return( TRUE ); -} - -/* Optimize memory usage */ -STATIC MYBOOL memopt_lp(lprec *lp, int rowextra, int colextra, int nzextra) -{ - MYBOOL status = FALSE; - - if(lp == NULL) - return( status ); - - status = mat_memopt(lp->matA, rowextra, colextra, nzextra) && - (++rowextra > 0) && (++colextra > 0) && (++nzextra > 0); - -#if 0 /* inc_ routines not well-tested for reduction in size allocation */ - if(status) { - int colalloc = lp->columns_alloc - MIN(lp->columns_alloc, lp->columns + colextra), - rowalloc = lp->rows_alloc - MIN(lp->rows_alloc, lp->rows + rowextra); - - status = inc_lag_space(lp, rowalloc, FALSE) && - inc_row_space(lp, rowalloc) && - inc_col_space(lp, colalloc); - } -#endif - - return( status ); -} - - -/* Utility routine group for constraint and column deletion/insertion - mapping in relation to the original set of constraints and columns */ -STATIC void varmap_lock(lprec *lp) -{ - presolve_fillUndo(lp, lp->rows, lp->columns, TRUE); - lp->varmap_locked = TRUE; -} -STATIC void varmap_clear(lprec *lp) -{ - presolve_setOrig(lp, 0, 0); - lp->varmap_locked = FALSE; -} -STATIC MYBOOL varmap_canunlock(lprec *lp) -{ - /* Don't do anything if variables aren't locked yet */ - if(lp->varmap_locked) { - int i; - presolveundorec *psundo = lp->presolve_undo; - - /* Check for the obvious */ - if(/*lp->names_used || - (psundo->orig_columns != lp->columns) || (psundo->orig_rows != lp->rows)) */ - (psundo->orig_columns > lp->columns) || (psundo->orig_rows > lp->rows)) - return( FALSE ); - - /* Check for deletions */ - for(i = psundo->orig_rows + psundo->orig_columns; i > 0; i--) - if(psundo->orig_to_var[i] == 0) - return( FALSE ); - - /* Check for insertions */ - for(i = lp->sum; i > 0; i--) - if(psundo->var_to_orig[i] == 0) - return( FALSE ); - } - return( TRUE ); -} -STATIC void varmap_add(lprec *lp, int base, int delta) -{ - int i, ii; - presolveundorec *psundo = lp->presolve_undo; - - /* Don't do anything if variables aren't locked yet */ - if(!lp->varmap_locked) - return; - - /* Set new constraints/columns to have an "undefined" mapping to original - constraints/columns (assumes that counters have NOT yet been updated) */ - for(i = lp->sum; i >= base; i--) { - ii = i + delta; - psundo->var_to_orig[ii] = psundo->var_to_orig[i]; - } - - /* Initialize map of added rows/columns */ - for(i = 0; i < delta; i++) { - ii = base + i; - psundo->var_to_orig[ii] = 0; - } -} - -STATIC void varmap_delete(lprec *lp, int base, int delta, LLrec *varmap) -{ - int i, ii, j; - MYBOOL preparecompact = (MYBOOL) (varmap != NULL); - presolveundorec *psundo = lp->presolve_undo; - - /* Set the model "dirty" if we are deleting row of constraint */ - lp->model_is_pure &= (MYBOOL) ((lp->solutioncount == 0) && !preparecompact); - - /* Don't do anything if - 1) variables aren't locked yet, or - 2) the constraint was added after the variables were locked */ - if(!lp->varmap_locked) { -#if 0 - if(lp->names_used) - varmap_lock(lp); - else - return; -#else - if(!lp->model_is_pure && lp->names_used) - varmap_lock(lp); -#endif - } - - /* Do mass deletion via a linked list */ - preparecompact = (MYBOOL) (varmap != NULL); - if(preparecompact) { - preparecompact = (MYBOOL) (base > lp->rows); /* Set TRUE for columns */ - for(j = firstInactiveLink(varmap); j != 0; j = nextInactiveLink(varmap, j)) { - i = j; - if(preparecompact) { -#ifdef Paranoia - if(SOS_is_member(lp->SOS, 0, j)) - report(lp, SEVERE, "varmap_delete: Deleting variable %d, which is in a SOS!\n", j); -#endif - i += lp->rows; - } - ii = psundo->var_to_orig[i]; - if(ii > 0) /* It was an original variable; reverse sign of index to flag deletion */ - psundo->var_to_orig[i] = -ii; - else /* It was a non-original variable; add special code for deletion */ - psundo->var_to_orig[i] = -(psundo->orig_rows+psundo->orig_columns+i); - } - return; - } - - /* Do legacy simplified version if we are doing batch delete operations */ - preparecompact = (MYBOOL) (base < 0); - if(preparecompact) { - base = -base; - if(base > lp->rows) - base += (psundo->orig_rows - lp->rows); - for(i = base; i < base-delta; i++) { - ii = psundo->var_to_orig[i]; - if(ii > 0) /* It was an original variable; reverse sign of index to flag deletion */ - psundo->var_to_orig[i] = -ii; - else /* It was a non-original variable; add special code for deletion */ - psundo->var_to_orig[i] = -(psundo->orig_rows+psundo->orig_columns+i); - } - return; - } - - /* We are deleting an original constraint/column; - 1) clear mapping of original to deleted - 2) shift the deleted variable to original mappings left - 3) decrement all subsequent original-to-current pointers - */ - if(varmap_canunlock(lp)) lp->varmap_locked = FALSE; - for(i = base; i < base-delta; i++) { - ii = psundo->var_to_orig[i]; - if(ii > 0) - psundo->orig_to_var[ii] = 0; - } - for(i = base; i <= lp->sum+delta; i++) { - ii = i - delta; - psundo->var_to_orig[i] = psundo->var_to_orig[ii]; - } - - i = 1; - j = psundo->orig_rows; - if(base > lp->rows) { - i += j; - j += psundo->orig_columns; - } - ii = base-delta; - for(; i <= j; i++) { - if(psundo->orig_to_var[i] >= ii) - psundo->orig_to_var[i] += delta; - } - -} - -STATIC MYBOOL varmap_validate(lprec *lp, int varno) -{ - MYBOOL success = TRUE; - int i, ii, ix, ie, - n_rows = lp->rows, - orig_sum = lp->presolve_undo->orig_sum, - orig_rows = lp->presolve_undo->orig_rows; - - if(varno <= 0) { - varno = 1; - ie = orig_sum; - } - else - ie = varno; - for(i = varno; success && (i <= ie); i++) { - ix = lp->presolve_undo->orig_to_var[i]; - if((ix > 0) && (i > orig_rows)) - ix += n_rows; - - /* Check for index out of range due to presolve */ - success = (MYBOOL) (ix <= orig_sum); - if(!success) - report(lp, SEVERE, "varmap_validate: Invalid new mapping found for variable %d\n", - i); - else if(ix != 0) { - ii = lp->presolve_undo->var_to_orig[ix]; - if(ix > n_rows) - ii += orig_rows; - success = (MYBOOL) (ii == i); - if(!success) - report(lp, SEVERE, "varmap_validate: Invalid old mapping found for variable %d (%d)\n", - i, ii); - } - } - return( success ); -} - -STATIC void varmap_compact(lprec *lp, int prev_rows, int prev_cols) -{ - presolveundorec *psundo = lp->presolve_undo; - int i, ii, n_sum, n_rows, - orig_rows = psundo->orig_rows, - prev_sum = prev_rows + prev_cols; - - /* Nothing to do if the model is not "dirty" or the variable map is not locked */ - if(lp->model_is_pure || !lp->varmap_locked) - return; - - /* We are deleting an original constraint/column; - 1) clear mapping of original to deleted - 2) shift the deleted variable to original mappings left - 3) decrement all subsequent original-to-current pointers - */ - n_sum = 0; - n_rows = 0; - for(i = 1; i <= prev_sum; i++) { - ii = psundo->var_to_orig[i]; - - /* Process variable if it was deleted in the previous round */ - if(ii < 0) { - ii = -ii; - /* Update map back if we have an original variable, otherwise just skip */ - if(i <= prev_rows) - psundo->orig_to_var[ii] = 0; - else - psundo->orig_to_var[orig_rows+ii] = 0; - } - /* Otherwise shift and update map back */ - else { - n_sum++; - /* Shift only if necessary */ - if(n_sum < i) - psundo->var_to_orig[n_sum] = ii; - /* Update map back if we have an original variable */ - if(ii > 0) { - if(i <= prev_rows) { - psundo->orig_to_var[ii] = n_sum; - n_rows = n_sum; - } - else - psundo->orig_to_var[orig_rows+ii] = n_sum-n_rows; - } - } - } -#ifdef xxParanoia - if(!varmap_validate(lp, 0)) - report(lp, SEVERE, "varmap_compact: Internal presolve mapping error at exit\n"); -#endif - -} - -/* Utility group for shifting row and column data */ -STATIC MYBOOL shift_rowcoldata(lprec *lp, int base, int delta, LLrec *usedmap, MYBOOL isrow) -/* Note: Assumes that "lp->sum" and "lp->rows" HAVE NOT been updated to the new counts */ -{ - int i, ii; - REAL lodefault; - - /* Shift data right/down (insert), and set default values in positive delta-gap */ - if(delta > 0) { - - /* Determine if we can take the easy way out */ - MYBOOL easyout = (MYBOOL) ((lp->solvecount == 0) && (base > lp->rows)); - - /* Shift the row/column data */ - - MEMMOVE(lp->orig_upbo + base + delta, lp->orig_upbo + base, lp->sum - base + 1); - MEMMOVE(lp->orig_lowbo + base + delta, lp->orig_lowbo + base, lp->sum - base + 1); - - if(!easyout) { - MEMMOVE(lp->upbo + base + delta, lp->upbo + base, lp->sum - base + 1); - MEMMOVE(lp->lowbo + base + delta, lp->lowbo + base, lp->sum - base + 1); - if(lp->model_is_valid) { - MEMMOVE(lp->solution + base + delta, lp->solution + base, lp->sum - base + 1); - MEMMOVE(lp->best_solution + base + delta, lp->best_solution + base, lp->sum - base + 1); - } - MEMMOVE(lp->is_lower + base + delta, lp->is_lower + base, lp->sum - base + 1); - } - - /* Deal with scalars; the vector can be NULL */ - if(lp->scalars != NULL) { - if(!easyout) - for(ii = lp->sum; ii >= base; ii--) { - i = ii + delta; - lp->scalars[i] = lp->scalars[ii]; - } - for(ii = base; ii < base + delta; ii++) - lp->scalars[ii] = 1; - } - - /* Set defaults */ -#ifdef SlackInitMinusInf - if(isrow) - lodefault = -lp->infinite; - else -#endif - lodefault = 0; - - for(i = 0; i < delta; i++) { - ii = base + i; - lp->orig_upbo[ii] = lp->infinite; - lp->orig_lowbo[ii] = lodefault; - if(!easyout) { - lp->upbo[ii] = lp->orig_upbo[ii]; - lp->lowbo[ii] = lp->orig_lowbo[ii]; - lp->is_lower[ii] = TRUE; - } - } - } - - /* Shift data left/up (delete) */ - else if(usedmap != NULL) { - int k, offset = 0; - if(!isrow) - offset += lp->rows; - i = offset + 1; - for(k = firstActiveLink(usedmap); k != 0; - i++, k = nextActiveLink(usedmap, k)) { - ii = k + offset; - if(ii == i) - continue; - lp->upbo[i] = lp->upbo[ii]; - lp->orig_upbo[i] = lp->orig_upbo[ii]; - lp->lowbo[i] = lp->lowbo[ii]; - lp->orig_lowbo[i] = lp->orig_lowbo[ii]; - lp->solution[i] = lp->solution[ii]; - lp->best_solution[i] = lp->best_solution[ii]; - lp->is_lower[i] = lp->is_lower[ii]; - if(lp->scalars != NULL) - lp->scalars[i] = lp->scalars[ii]; - } - if(isrow) { - base = lp->rows + 1; - MEMMOVE(lp->upbo + i, lp->upbo + base, lp->columns); - MEMMOVE(lp->orig_upbo + i, lp->orig_upbo + base, lp->columns); - MEMMOVE(lp->lowbo + i, lp->lowbo + base, lp->columns); - MEMMOVE(lp->orig_lowbo + i, lp->orig_lowbo + base, lp->columns); - if(lp->model_is_valid) { - MEMMOVE(lp->solution + i, lp->solution + base, lp->columns); - MEMMOVE(lp->best_solution + i, lp->best_solution + base, lp->columns); - } - MEMMOVE(lp->is_lower + i, lp->is_lower + base, lp->columns); - if(lp->scalars != NULL) - MEMMOVE(lp->scalars + i, lp->scalars + base, lp->columns); - } - } - - else if(delta < 0) { - - /* First make sure we don't cross the sum count border */ - if(base-delta-1 > lp->sum) - delta = base - lp->sum - 1; - - /* Shift the data*/ - for(i = base; i <= lp->sum + delta; i++) { - ii = i - delta; - lp->upbo[i] = lp->upbo[ii]; - lp->orig_upbo[i] = lp->orig_upbo[ii]; - lp->lowbo[i] = lp->lowbo[ii]; - lp->orig_lowbo[i] = lp->orig_lowbo[ii]; - lp->solution[i] = lp->solution[ii]; - lp->best_solution[i] = lp->best_solution[ii]; - lp->is_lower[i] = lp->is_lower[ii]; - if(lp->scalars != NULL) - lp->scalars[i] = lp->scalars[ii]; - } - - } - - lp->sum += delta; - - lp->matA->row_end_valid = FALSE; - - return(TRUE); -} - -STATIC MYBOOL shift_basis(lprec *lp, int base, int delta, LLrec *usedmap, MYBOOL isrow) -/* Note: Assumes that "lp->sum" and "lp->rows" HAVE NOT been updated to the new counts */ -{ - int i, ii; - MYBOOL Ok = TRUE; - - /* Don't bother to shift the basis if it is not yet ready */ - if(!is_BasisReady(lp)) - return( Ok ); - - /* Basis adjustments due to insertions (after actual row/column insertions) */ - if(delta > 0) { - - /* Determine if the basis becomes invalidated */ - if(isrow) - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT); - - /* Shift and fix invalid basis references (increment higher order basic variable index) */ - if(base <= lp->sum) - MEMMOVE(lp->is_basic + base + delta, lp->is_basic + base, lp->sum - base + 1); - - /* Prevent CPU-expensive basis updating if this is the initial model creation */ - if(!lp->model_is_pure || (lp->solvecount > 0)) - for(i = 1; i <= lp->rows; i++) { - ii = lp->var_basic[i]; - if(ii >= base) - lp->var_basic[i] += delta; - } - - /* Update the basis (shift and extend) */ - for(i = 0; i < delta; i++) { - ii = base + i; - lp->is_basic[ii] = isrow; - if(isrow) - lp->var_basic[lp->rows+1+i] = ii; - } - - } - /* Basis adjustments due to deletions (after actual row/column deletions) */ - else { - int j,k; - - /* Fix invalid basis references (decrement high basic slack variable indexes), - but reset the entire basis if a deleted variable is found in the basis */ - k = 0; - for(i = 1; i <= lp->rows; i++) { - ii = lp->var_basic[i]; - lp->is_basic[ii] = FALSE; - if(ii >= base) { - /* Skip to next basis variable if this one is to be deleted */ - if(ii < base-delta) { - set_action(&lp->spx_action, ACTION_REBASE); - continue; - } - /* Otherwise, update the index of the basic variable for deleted variables */ - ii += delta; - } - k++; - lp->var_basic[k] = ii; - } - - /* Set the new basis indicators */ - i = k; - if(isrow) - i = MIN(k, lp->rows+delta); - for(; i > 0; i--) { - j = lp->var_basic[i]; - lp->is_basic[j] = TRUE; - } - - /* If a column was deleted from the basis then simply add back a non-basic - slack variable; do two scans, if necessary to avoid adding equality slacks */ - if(!isrow && (k < lp->rows)) { - for(j = 0; j <= 1; j++) - for(i = 1; (i <= lp->rows) && (k < lp->rows); i++) - if(!lp->is_basic[i]) { - if(!is_constr_type(lp, i, EQ) || (j == 1)) { - k++; - lp->var_basic[k] = i; - lp->is_basic[i] = TRUE; - } - } - k = 0; - } - - /* We are left with "k" indexes; if no basis variable was deleted, k=rows and the - inverse is still valid, if k+delta < 0 we do not have a valid - basis and must create one (in most usage modes this should not happen, - unless there is a bug) */ - if(k+delta < 0) - Ok = FALSE; - if(isrow || (k != lp->rows)) - set_action(&lp->spx_action, ACTION_REINVERT); - - } - return(Ok); - -} - -STATIC MYBOOL shift_rowdata(lprec *lp, int base, int delta, LLrec *usedmap) -/* Note: Assumes that "lp->rows" HAS NOT been updated to the new count */ -{ - int i, ii; - - /* Shift sparse matrix row data */ - if(lp->matA->is_roworder) - mat_shiftcols(lp->matA, &base, delta, usedmap); - else - mat_shiftrows(lp->matA, &base, delta, usedmap); - - /* Shift data down (insert row), and set default values in positive delta-gap */ - if(delta > 0) { - - /* Shift row data */ - for(ii = lp->rows; ii >= base; ii--) { - i = ii + delta; - lp->orig_rhs[i] = lp->orig_rhs[ii]; - lp->rhs[i] = lp->rhs[ii]; - lp->row_type[i] = lp->row_type[ii]; - } - - /* Set defaults (actual basis set in separate procedure) */ - for(i = 0; i < delta; i++) { - ii = base + i; - lp->orig_rhs[ii] = 0; - lp->rhs[ii] = 0; - lp->row_type[ii] = ROWTYPE_EMPTY; - } - } - - /* Shift data up (delete row) */ - else if(usedmap != NULL) { - for(i = 1, ii = firstActiveLink(usedmap); ii != 0; - i++, ii = nextActiveLink(usedmap, ii)) { - if(i == ii) - continue; - lp->orig_rhs[i] = lp->orig_rhs[ii]; - lp->rhs[i] = lp->rhs[ii]; - lp->row_type[i] = lp->row_type[ii]; - } - delta = i - lp->rows - 1; - } - else if(delta < 0) { - - /* First make sure we don't cross the row count border */ - if(base-delta-1 > lp->rows) - delta = base - lp->rows - 1; - - /* Shift row data (don't shift basis indexes here; done in next step) */ - for(i = base; i <= lp->rows + delta; i++) { - ii = i - delta; - lp->orig_rhs[i] = lp->orig_rhs[ii]; - lp->rhs[i] = lp->rhs[ii]; - lp->row_type[i] = lp->row_type[ii]; - } - } - - shift_basis(lp, base, delta, usedmap, TRUE); - shift_rowcoldata(lp, base, delta, usedmap, TRUE); - inc_rows(lp, delta); - - return(TRUE); -} - -STATIC MYBOOL shift_coldata(lprec *lp, int base, int delta, LLrec *usedmap) -/* Note: Assumes that "lp->columns" has NOT been updated to the new count */ -{ - int i, ii; - - if(lp->bb_totalnodes == 0) - free_duals(lp); - - /* Shift A matrix data */ - if(lp->matA->is_roworder) - mat_shiftrows(lp->matA, &base, delta, usedmap); - else - mat_shiftcols(lp->matA, &base, delta, usedmap); - - /* Shift data right (insert), and set default values in positive delta-gap */ - if(delta > 0) { - - /* Fix variable priority data */ - if((lp->var_priority != NULL) && (base <= lp->columns)) { - for(i = 0; i < lp->columns; i++) - if(lp->var_priority[i] >= base) - lp->var_priority[i] += delta; - } - if((lp->sos_priority != NULL) && (base <= lp->columns)) { - for(i = 0; i < lp->sos_vars; i++) - if(lp->sos_priority[i] >= base) - lp->sos_priority[i] += delta; - } - - /* Fix invalid split variable data */ - if((lp->var_is_free != NULL) && (base <= lp->columns)) { - for(i = 1; i <= lp->columns; i++) - if(abs(lp->var_is_free[i]) >= base) - lp->var_is_free[i] += my_chsign(lp->var_is_free[i] < 0, delta); - } - - /* Shift column data right */ - for(ii = lp->columns; ii >= base; ii--) { - i = ii + delta; - lp->var_type[i] = lp->var_type[ii]; - lp->sc_lobound[i] = lp->sc_lobound[ii]; - lp->orig_obj[i] = lp->orig_obj[ii]; - if(lp->obj != NULL) - lp->obj[i] = lp->obj[ii]; -/* - if(lp->objfromvalue != NULL) - lp->objfromvalue[i] = lp->objfromvalue[ii]; - if(lp->objfrom != NULL) - lp->objfrom[i] = lp->objfrom[ii]; - if(lp->objtill != NULL) - lp->objtill[i] = lp->objtill[ii]; -*/ - if(lp->var_priority != NULL) - lp->var_priority[i-1] = lp->var_priority[ii-1]; - if(lp->bb_varbranch != NULL) - lp->bb_varbranch[i-1] = lp->bb_varbranch[ii-1]; - if(lp->var_is_free != NULL) - lp->var_is_free[i] = lp->var_is_free[ii]; - if(lp->best_solution != NULL) - lp->best_solution[lp->rows + i] = lp->best_solution[lp->rows + ii]; - } - - /* Set defaults */ - for(i = 0; i < delta; i++) { - ii = base + i; - lp->var_type[ii] = ISREAL; - lp->sc_lobound[ii] = 0; - lp->orig_obj[ii] = 0; - if(lp->obj != NULL) - lp->obj[ii] = 0; -/* - if(lp->objfromvalue != NULL) - lp->objfromvalue[ii] = 0; - if(lp->objfrom != NULL) - lp->objfrom[ii] = 0; - if(lp->objtill != NULL) - lp->objtill[ii] = 0; -*/ - if(lp->var_priority != NULL) - lp->var_priority[ii-1] = ii; - if(lp->bb_varbranch != NULL) - lp->bb_varbranch[ii-1] = BRANCH_DEFAULT; - if(lp->var_is_free != NULL) - lp->var_is_free[ii] = 0; - if(lp->best_solution != NULL) - lp->best_solution[lp->rows + ii] = 0; - } - } - - /* Shift data left (delete) */ - else if(usedmap != NULL) { - /* Assume there is no need to handle split columns, since we are doing - this only from presolve, which comes before splitting of columns. */ - - /* First update counts */ - if(lp->int_vars + lp->sc_vars > 0) - for(ii = firstInactiveLink(usedmap); ii != 0; ii = nextInactiveLink(usedmap, ii)) { - if(is_int(lp, ii)) { - lp->int_vars--; - if(SOS_is_member(lp->SOS, 0, ii)) - lp->sos_ints--; - } - if(is_semicont(lp, ii)) - lp->sc_vars--; - } - /* Shift array members */ - for(i = 1, ii = firstActiveLink(usedmap); ii != 0; - i++, ii = nextActiveLink(usedmap, ii)) { - if(i == ii) - continue; - lp->var_type[i] = lp->var_type[ii]; - lp->sc_lobound[i] = lp->sc_lobound[ii]; - lp->orig_obj[i] = lp->orig_obj[ii]; - if(lp->obj != NULL) - lp->obj[i] = lp->obj[ii]; -/* - if(lp->objfromvalue != NULL) - lp->objfromvalue[i] = lp->objfromvalue[ii]; - if(lp->objfrom != NULL) - lp->objfrom[i] = lp->objfrom[ii]; - if(lp->objtill != NULL) - lp->objtill[i] = lp->objtill[ii]; -*/ - if(lp->bb_varbranch != NULL) - lp->bb_varbranch[i-1] = lp->bb_varbranch[ii-1]; - if(lp->var_is_free != NULL) - lp->var_is_free[i] = lp->var_is_free[ii]; - if(lp->best_solution != NULL) - lp->best_solution[lp->rows + i] = lp->best_solution[lp->rows + ii]; - } - /* Shift variable priority data */ - if((lp->var_priority != NULL) || (lp->sos_priority != NULL)) { - int *colmap = NULL, k; - allocINT(lp, &colmap, lp->columns + 1, TRUE); - for(i = 1, ii = 0; i <= lp->columns; i++) { - if(isActiveLink(usedmap, i)) { - ii++; - colmap[i] = ii; - } - } - if(lp->var_priority != NULL) { - for(i = 0, ii = 0; i < lp->columns; i++) { - k = colmap[lp->var_priority[i]]; - if(k > 0) { - lp->var_priority[ii] = k; - ii++; - } - } - } - if(lp->sos_priority != NULL) { - for(i = 0, ii = 0; i < lp->sos_vars; i++) { - k = colmap[lp->sos_priority[i]]; - if(k > 0) { - lp->sos_priority[ii] = k; - ii++; - } - } - lp->sos_vars = ii; - } - FREE(colmap); - } - - delta = i - lp->columns - 1; - } - else if(delta < 0) { - - /* Fix invalid split variable data */ - if(lp->var_is_free != NULL) { - for(i = 1; i <= lp->columns; i++) - if(abs(lp->var_is_free[i]) >= base) - lp->var_is_free[i] -= my_chsign(lp->var_is_free[i] < 0, delta); - } - - /* Shift column data (excluding the basis) */ - for(i = base; i < base-delta; i++) { - if(is_int(lp, i)) { - lp->int_vars--; - if(SOS_is_member(lp->SOS, 0, i)) - lp->sos_ints--; - } - if(is_semicont(lp, i)) - lp->sc_vars--; - } - for(i = base; i <= lp->columns + delta; i++) { - ii = i - delta; - lp->var_type[i] = lp->var_type[ii]; - lp->sc_lobound[i] = lp->sc_lobound[ii]; - lp->orig_obj[i] = lp->orig_obj[ii]; - if(lp->obj != NULL) - lp->obj[i] = lp->obj[ii]; -/* - if(lp->objfromvalue != NULL) - lp->objfromvalue[i] = lp->objfromvalue[ii]; - if(lp->objfrom != NULL) - lp->objfrom[i] = lp->objfrom[ii]; - if(lp->objtill != NULL) - lp->objtill[i] = lp->objtill[ii]; -*/ - if(lp->var_priority != NULL) - lp->var_priority[i-1] = lp->var_priority[ii-1]; - if(lp->bb_varbranch != NULL) - lp->bb_varbranch[i-1] = lp->bb_varbranch[ii-1]; - if(lp->var_is_free != NULL) - lp->var_is_free[i] = lp->var_is_free[ii]; - if(lp->best_solution != NULL) - lp->best_solution[lp->rows + i] = lp->best_solution[lp->rows + ii]; - } - - /* Fix invalid variable priority data */ - if(lp->var_priority != NULL) { - for(i = 0, ii = 0; i < lp->columns; i++) - if(lp->var_priority[i] > base - delta) - lp->var_priority[ii++] = lp->var_priority[i] + delta; - else if(lp->var_priority[i] < base) - lp->var_priority[ii++] = lp->var_priority[i]; - } - if(lp->sos_priority != NULL) { - for(i = 0, ii = 0; i < lp->sos_vars; i++) { - if(lp->sos_priority[i] > base - delta) - lp->sos_priority[ii++] = lp->sos_priority[i] + delta; - else if(lp->sos_priority[i] < base) - lp->sos_priority[ii++] = lp->sos_priority[i]; - } - lp->sos_vars = ii; - } - - } - - shift_basis(lp, lp->rows+base, delta, usedmap, FALSE); - if(SOS_count(lp) > 0) - SOS_shift_col(lp->SOS, 0, base, delta, usedmap, FALSE); - shift_rowcoldata(lp, lp->rows+base, delta, usedmap, FALSE); - inc_columns(lp, delta); - - return( TRUE ); -} - -/* Utility group for incrementing row and column vector storage space */ -STATIC void inc_rows(lprec *lp, int delta) -{ - int i; - - if(lp->names_used && (lp->row_name != NULL)) - for(i = lp->rows + delta; i > lp->rows; i--) - lp->row_name[i] = NULL; - - lp->rows += delta; - if(lp->matA->is_roworder) - lp->matA->columns += delta; - else - lp->matA->rows += delta; -} - -STATIC void inc_columns(lprec *lp, int delta) -{ - int i; - - if(lp->names_used && (lp->col_name != NULL)) - for(i = lp->columns + delta; i > lp->columns; i--) - lp->col_name[i] = NULL; - - lp->columns += delta; - if(lp->matA->is_roworder) - lp->matA->rows += delta; - else - lp->matA->columns += delta; - if(get_Lrows(lp) > 0) - lp->matL->columns += delta; -} - -STATIC MYBOOL inc_rowcol_space(lprec *lp, int delta, MYBOOL isrows) -{ - int i, oldrowcolalloc, rowcolsum; - - /* Get rid of dual arrays */ - if(lp->solvecount > 0) - free_duals(lp); - - /* Set constants */ - oldrowcolalloc = lp->sum_alloc; - lp->sum_alloc += delta; - rowcolsum = lp->sum_alloc + 1; - - /* Reallocate lp memory */ - if(!allocREAL(lp, &lp->upbo, rowcolsum, AUTOMATIC) || - !allocREAL(lp, &lp->orig_upbo, rowcolsum, AUTOMATIC) || - !allocREAL(lp, &lp->lowbo, rowcolsum, AUTOMATIC) || - !allocREAL(lp, &lp->orig_lowbo, rowcolsum, AUTOMATIC) || - !allocREAL(lp, &lp->solution, rowcolsum, AUTOMATIC) || - !allocREAL(lp, &lp->best_solution, rowcolsum, AUTOMATIC) || - !allocMYBOOL(lp, &lp->is_basic, rowcolsum, AUTOMATIC) || - !allocMYBOOL(lp, &lp->is_lower, rowcolsum, AUTOMATIC) || - ((lp->scalars != NULL) && !allocREAL(lp, &lp->scalars, rowcolsum, AUTOMATIC))) - return( FALSE ); - - /* Fill in default values, where appropriate */ - for(i = oldrowcolalloc+1; i < rowcolsum; i++) { - lp->upbo[i] = lp->infinite; - lp->orig_upbo[i] = lp->upbo[i]; - lp->lowbo[i] = 0; - lp->orig_lowbo[i] = lp->lowbo[i]; - lp->is_basic[i] = FALSE; - lp->is_lower[i] = TRUE; - } - - /* Deal with scalars; the vector can be NULL and also contains Lagrangean information */ - if(lp->scalars != NULL) { - for(i = oldrowcolalloc+1; i < rowcolsum; i++) - lp->scalars[i] = 1; - if(oldrowcolalloc == 0) - lp->scalars[0] = 1; - } - - return( inc_presolve_space(lp, delta, isrows) && - resizePricer(lp) ); -} - -STATIC MYBOOL inc_lag_space(lprec *lp, int deltarows, MYBOOL ignoreMAT) -{ - int newsize; - - if(deltarows > 0) { - - newsize = get_Lrows(lp) + deltarows; - - /* Reallocate arrays */ - if(!allocREAL(lp, &lp->lag_rhs, newsize+1, AUTOMATIC) || - !allocREAL(lp, &lp->lambda, newsize+1, AUTOMATIC) || - !allocINT(lp, &lp->lag_con_type, newsize+1, AUTOMATIC)) - return( FALSE ); - - /* Reallocate the matrix (note that the row scalars are stored at index 0) */ - if(!ignoreMAT) { - if(lp->matL == NULL) - lp->matL = mat_create(lp, newsize, lp->columns, lp->epsvalue); - else - inc_matrow_space(lp->matL, deltarows); - } - lp->matL->rows += deltarows; - - } - /* Handle column count expansion as special case */ - else if(!ignoreMAT) { - inc_matcol_space(lp->matL, lp->columns_alloc-lp->matL->columns_alloc+1); - } - - - return( TRUE ); -} - -STATIC MYBOOL inc_row_space(lprec *lp, int deltarows) -{ - int i, rowsum, oldrowsalloc; - MYBOOL ok = TRUE; - - /* Adjust lp row structures */ - i = lp->rows_alloc+deltarows; - if(lp->matA->is_roworder) { - i -= lp->matA->columns_alloc; - SETMIN(i, deltarows); - if(i > 0) - inc_matcol_space(lp->matA, i); - rowsum = lp->matA->columns_alloc; - } - else { -#if 0 - if((lp->rows_alloc > 0) && (lp->rows + deltarows > lp->rows_alloc)) - i = deltarows; /* peno 25/12/06 */ - else -#endif - i -= lp->matA->rows_alloc; - SETMIN(i, deltarows); - if(i > 0) - inc_matrow_space(lp->matA, i); - rowsum = lp->matA->rows_alloc; - } - if(lp->rows+deltarows > lp->rows_alloc) { - - rowsum++; - oldrowsalloc = lp->rows_alloc; - lp->rows_alloc = rowsum; - deltarows = rowsum - oldrowsalloc; - rowsum++; - - if(!allocREAL(lp, &lp->orig_rhs, rowsum, AUTOMATIC) || - !allocLREAL(lp, &lp->rhs, rowsum, AUTOMATIC) || - !allocINT(lp, &lp->row_type, rowsum, AUTOMATIC) || - !allocINT(lp, &lp->var_basic, rowsum, AUTOMATIC)) - return( FALSE ); - - if(oldrowsalloc == 0) { - lp->var_basic[0] = AUTOMATIC; /*TRUE;*/ /* Indicates default basis */ - lp->orig_rhs[0] = 0; - lp->row_type[0] = ROWTYPE_OFMIN; - } - for(i = oldrowsalloc+1; i < rowsum; i++) { - lp->orig_rhs[i] = 0; - lp->rhs[i] = 0; - lp->row_type[i] = ROWTYPE_EMPTY; - lp->var_basic[i] = i; - } - - /* Adjust hash name structures */ - if(lp->names_used && (lp->row_name != NULL)) { - - /* First check the hash table */ - if(lp->rowname_hashtab->size < lp->rows_alloc) { - hashtable *ht; - - ht = copy_hash_table(lp->rowname_hashtab, lp->row_name, lp->rows_alloc + 1); - if(ht == NULL) { - lp->spx_status = NOMEMORY; - return( FALSE ); - } - free_hash_table(lp->rowname_hashtab); - lp->rowname_hashtab = ht; - } - - /* Then the string storage (i.e. pointer to the item's hash structure) */ - lp->row_name = (hashelem **) realloc(lp->row_name, (rowsum) * sizeof(*lp->row_name)); - if(lp->row_name == NULL) { - lp->spx_status = NOMEMORY; - return( FALSE ); - } - for(i = oldrowsalloc + 1; i < rowsum; i++) - lp->row_name[i] = NULL; - } - - ok = inc_rowcol_space(lp, deltarows, TRUE); - - } - return(ok); -} - -STATIC MYBOOL inc_col_space(lprec *lp, int deltacols) -{ - int i,colsum, oldcolsalloc; - - i = lp->columns_alloc+deltacols; - if(lp->matA->is_roworder) { - i -= lp->matA->rows_alloc; - SETMIN(i, deltacols); - if(i > 0) - inc_matrow_space(lp->matA, i); - colsum = lp->matA->rows_alloc; - } - else { - i -= lp->matA->columns_alloc; - SETMIN(i, deltacols); - if(i > 0) - inc_matcol_space(lp->matA, i); - colsum = lp->matA->columns_alloc; - } - - if(lp->columns+deltacols >= lp->columns_alloc) { - - colsum++; - oldcolsalloc = lp->columns_alloc; - lp->columns_alloc = colsum; - deltacols = colsum - oldcolsalloc; - colsum++; - - /* Adjust hash name structures */ - if(lp->names_used && (lp->col_name != NULL)) { - - /* First check the hash table */ - if(lp->colname_hashtab->size < lp->columns_alloc) { - hashtable *ht; - - ht = copy_hash_table(lp->colname_hashtab, lp->col_name, lp->columns_alloc + 1); - if(ht != NULL) { - free_hash_table(lp->colname_hashtab); - lp->colname_hashtab = ht; - } - } - - /* Then the string storage (i.e. pointer to the item's hash structure) */ - lp->col_name = (hashelem **) realloc(lp->col_name, (colsum) * sizeof(*lp->col_name)); - for(i = oldcolsalloc+1; i < colsum; i++) - lp->col_name[i] = NULL; - } - - if(!allocREAL(lp, &lp->orig_obj, colsum, AUTOMATIC) || - !allocMYBOOL(lp, &lp->var_type, colsum, AUTOMATIC) || - !allocREAL(lp, &lp->sc_lobound, colsum, AUTOMATIC) || - ((lp->obj != NULL) && !allocREAL(lp, &lp->obj, colsum, AUTOMATIC)) || - ((lp->var_priority != NULL) && !allocINT(lp, &lp->var_priority, colsum-1, AUTOMATIC)) || - ((lp->var_is_free != NULL) && !allocINT(lp, &lp->var_is_free, colsum, AUTOMATIC)) || - ((lp->bb_varbranch != NULL) && !allocMYBOOL(lp, &lp->bb_varbranch, colsum-1, AUTOMATIC))) - return( FALSE ); - - /* Make sure that Lagrangean constraints have the same number of columns */ - if(get_Lrows(lp) > 0) - inc_lag_space(lp, 0, FALSE); - - /* Update column pointers */ - for(i = MIN(oldcolsalloc, lp->columns) + 1; i < colsum; i++) { - lp->orig_obj[i] = 0; - if(lp->obj != NULL) - lp->obj[i] = 0; - lp->var_type[i] = ISREAL; - lp->sc_lobound[i] = 0; - if(lp->var_priority != NULL) - lp->var_priority[i-1] = i; - } - - if(lp->var_is_free != NULL) { - for(i = oldcolsalloc+1; i < colsum; i++) - lp->var_is_free[i] = 0; - } - - if(lp->bb_varbranch != NULL) { - for(i = oldcolsalloc; i < colsum-1; i++) - lp->bb_varbranch[i] = BRANCH_DEFAULT; - } - - inc_rowcol_space(lp, deltacols, FALSE); - - } - return(TRUE); -} - -/* Problem manipulation routines */ - -MYBOOL __WINAPI set_obj(lprec *lp, int colnr, REAL value) -{ - if(colnr <= 0) - colnr = set_rh(lp, 0, value); - else - colnr = set_mat(lp, 0, colnr, value); - return((MYBOOL) colnr); -} - -MYBOOL __WINAPI set_obj_fnex(lprec *lp, int count, REAL *row, int *colno) -{ - MYBOOL chsgn = is_maxim(lp); - int i, ix; - REAL value; - - if(row == NULL) - return( FALSE ); - - else if(colno == NULL) { - if(count <= 0) - count = lp->columns; - for(i = 1; i <= count; i++) { - value = row[i]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, lp->matA->epsvalue); -#endif - lp->orig_obj[i] = my_chsign(chsgn, scaled_mat(lp, value, 0, i)); - } - } - else { - MEMCLEAR(lp->orig_obj, lp->columns+1); - for(i = 0; i < count; i++) { - ix = colno[i]; - value = row[i]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, lp->matA->epsvalue); -#endif - lp->orig_obj[ix] = my_chsign(chsgn, scaled_mat(lp, value, 0, ix)); - } - } - - return(TRUE); -} - -MYBOOL __WINAPI set_obj_fn(lprec *lp, REAL *row) -{ - return( set_obj_fnex(lp, 0, row, NULL) ); -} - -MYBOOL __WINAPI str_set_obj_fn(lprec *lp, char *row_string) -{ - int i; - MYBOOL ret = TRUE; - REAL *arow; - char *p, *newp; - - allocREAL(lp, &arow, lp->columns + 1, FALSE); - p = row_string; - for(i = 1; i <= lp->columns; i++) { - arow[i] = (REAL) strtod(p, &newp); - if(p == newp) { - report(lp, IMPORTANT, "str_set_obj_fn: Bad string %s\n", p); - lp->spx_status = DATAIGNORED; - ret = FALSE; - break; - } - else - p = newp; - } - if(lp->spx_status != DATAIGNORED) - ret = set_obj_fn(lp, arow); - FREE(arow); - return( ret ); -} - -STATIC MYBOOL append_columns(lprec *lp, int deltacolumns) -{ - if(!inc_col_space(lp, deltacolumns)) - return( FALSE ); - varmap_add(lp, lp->sum+1, deltacolumns); - shift_coldata(lp, lp->columns+1, deltacolumns, NULL); - return( TRUE ); -} - -STATIC MYBOOL append_rows(lprec *lp, int deltarows) -{ - if(!inc_row_space(lp, deltarows)) - return( FALSE ); - varmap_add(lp, lp->rows+1, deltarows); - shift_rowdata(lp, lp->rows+1, deltarows, NULL); - - return( TRUE ); -} - -MYBOOL __WINAPI set_add_rowmode(lprec *lp, MYBOOL turnon) -{ - if((lp->solvecount == 0) && (turnon ^ lp->matA->is_roworder)) - return( mat_transpose(lp->matA) ); - else - return( FALSE ); -} - -MYBOOL __WINAPI is_add_rowmode(lprec *lp) -{ - return(lp->matA->is_roworder); -} - -MYBOOL __WINAPI set_row(lprec *lp, int rownr, REAL *row) -{ - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "set_row: Row %d out of range\n", rownr); - return( FALSE ); - } - if(rownr == 0) - return( set_obj_fn(lp, row) ); - else - return( mat_setrow(lp->matA, rownr, lp->columns, row, NULL, TRUE, TRUE) ); -} - -MYBOOL __WINAPI set_rowex(lprec *lp, int rownr, int count, REAL *row, int *colno) -{ - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "set_rowex: Row %d out of range\n", rownr); - return( FALSE ); - } - if(rownr == 0) - return( set_obj_fnex(lp, count, row, colno) ); - else - return( mat_setrow(lp->matA, rownr, count, row, colno, TRUE, TRUE) ); -} - -MYBOOL __WINAPI add_constraintex(lprec *lp, int count, REAL *row, int *colno, int constr_type, REAL rh) -{ - int n; - MYBOOL status = FALSE; - - if(!(constr_type == LE || constr_type == GE || constr_type == EQ)) { - report(lp, IMPORTANT, "add_constraintex: Invalid %d constraint type\n", constr_type); - return( status ); - } - - /* Prepare for a new row */ - if(!append_rows(lp, 1)) - return( status ); - - /* Set constraint parameters, fix the slack */ - if((constr_type & ROWTYPE_CONSTRAINT) == EQ) { - lp->equalities++; - lp->orig_upbo[lp->rows] = 0; - lp->upbo[lp->rows] = 0; - } - lp->row_type[lp->rows] = constr_type; - - if(is_chsign(lp, lp->rows) && (rh != 0)) - lp->orig_rhs[lp->rows] = -rh; - else - lp->orig_rhs[lp->rows] = rh; - - /* Insert the non-zero constraint values */ - if(colno == NULL && row != NULL) - n = lp->columns; - else - n = count; - mat_appendrow(lp->matA, n, row, colno, my_chsign(is_chsign(lp, lp->rows), 1.0), TRUE); - if(!lp->varmap_locked) - presolve_setOrig(lp, lp->rows, lp->columns); - -#ifdef Paranoia - if(lp->matA->is_roworder) - n = lp->matA->columns; - else - n = lp->matA->rows; - if(lp->rows != n) { - report(lp, SEVERE, "add_constraintex: Row count mismatch %d vs %d\n", - lp->rows, n); - } - else if(is_BasisReady(lp) && !verify_basis(lp)) - report(lp, SEVERE, "add_constraintex: Invalid basis detected for row %d\n", lp->rows); - else -#endif - status = TRUE; - - return( status ); -} - -MYBOOL __WINAPI add_constraint(lprec *lp, REAL *row, int constr_type, REAL rh) -{ - return( add_constraintex(lp, 0, row, NULL, constr_type, rh) ); -} - -MYBOOL __WINAPI str_add_constraint(lprec *lp, char *row_string, int constr_type, REAL rh) -{ - int i; - char *p, *newp; - REAL *aRow; - MYBOOL status = FALSE; - - allocREAL(lp, &aRow, lp->columns + 1, FALSE); - p = row_string; - - for(i = 1; i <= lp->columns; i++) { - aRow[i] = (REAL) strtod(p, &newp); - if(p == newp) { - report(lp, IMPORTANT, "str_add_constraint: Bad string '%s'\n", p); - lp->spx_status = DATAIGNORED; - break; - } - else - p = newp; - } - if(lp->spx_status != DATAIGNORED) - status = add_constraint(lp, aRow, constr_type, rh); - FREE(aRow); - - return(status); -} - -STATIC MYBOOL del_constraintex(lprec *lp, LLrec *rowmap) -{ - int i; - - if(lp->equalities > 0) - for(i = firstInactiveLink(rowmap); i != 0; i = nextInactiveLink(rowmap, i)) { - if(is_constr_type(lp, i, EQ)) { -#ifdef Paranoia - if(lp->equalities == 0) - report(lp, SEVERE, "del_constraintex: Invalid count of equality constraints\n"); -#endif - lp->equalities--; - } - } - - varmap_delete(lp, 1, -1, rowmap); - shift_rowdata(lp, 1, -1, rowmap); - if(!lp->varmap_locked) { - presolve_setOrig(lp, lp->rows, lp->columns); - if(lp->names_used) - del_varnameex(lp, lp->row_name, lp->rows, lp->rowname_hashtab, 0, rowmap); - } - -#ifdef Paranoia - if(is_BasisReady(lp) && !verify_basis(lp)) - report(lp, SEVERE, "del_constraintex: Invalid basis detected\n"); -#endif - - return(TRUE); -} -MYBOOL __WINAPI del_constraint(lprec *lp, int rownr) -{ - MYBOOL preparecompact = (MYBOOL) (rownr < 0); - - if(preparecompact) - rownr = -rownr; - if((rownr < 1) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "del_constraint: Attempt to delete non-existing constraint %d\n", rownr); - return(FALSE); - } - /* - if(lp->matA->is_roworder) { - report(lp, IMPORTANT, "del_constraint: Cannot delete constraint while in row entry mode.\n"); - return(FALSE); - } - */ - - if(is_constr_type(lp, rownr, EQ) && (lp->equalities > 0)) - lp->equalities--; - - varmap_delete(lp, my_chsign(preparecompact, rownr), -1, NULL); - shift_rowdata(lp, my_chsign(preparecompact, rownr), -1, NULL); - -/* - peno 04.10.07 - Fixes a problem with del_constraint. - Constraints names were not shifted and reported variable result was incorrect. - See UnitTest1, UnitTest2 - - min: -2 x3; - - c1: +x2 -x1 <= 10; - c: 0 x3 <= 0; - c2: +x3 +x2 +x1 <= 20; - - 2 <= x3 <= 3; - x1 <= 30; - - // del_constraint(lp, 2); - - // See write_LP and print_solution result - - // To fix, commented if(!lp->varmap_locked) - -*/ - if(!lp->varmap_locked) - { - presolve_setOrig(lp, lp->rows, lp->columns); - if(lp->names_used) - del_varnameex(lp, lp->row_name, lp->rows, lp->rowname_hashtab, rownr, NULL); - } - -#ifdef Paranoia - if(is_BasisReady(lp) && !verify_basis(lp)) - report(lp, SEVERE, "del_constraint: Invalid basis detected at row %d\n", rownr); -#endif - - return(TRUE); -} - -MYBOOL __WINAPI add_lag_con(lprec *lp, REAL *row, int con_type, REAL rhs) -{ - int k; - REAL sign; - - if(con_type == LE || con_type == EQ) - sign = 1; - else if(con_type == GE) - sign = -1; - else { - report(lp, IMPORTANT, "add_lag_con: Constraint type %d not implemented\n", con_type); - return(FALSE); - } - - inc_lag_space(lp, 1, FALSE); - - k = get_Lrows(lp); - lp->lag_rhs[k] = rhs * sign; - mat_appendrow(lp->matL, lp->columns, row, NULL, sign, TRUE); - lp->lambda[k] = 0; - lp->lag_con_type[k] = con_type; - - return(TRUE); -} - -MYBOOL __WINAPI str_add_lag_con(lprec *lp, char *row_string, int con_type, REAL rhs) -{ - int i; - MYBOOL ret = TRUE; - REAL *a_row; - char *p, *new_p; - - allocREAL(lp, &a_row, lp->columns + 1, FALSE); - p = row_string; - - for(i = 1; i <= lp->columns; i++) { - a_row[i] = (REAL) strtod(p, &new_p); - if(p == new_p) { - report(lp, IMPORTANT, "str_add_lag_con: Bad string '%s'\n", p); - lp->spx_status = DATAIGNORED; - ret = FALSE; - break; - } - else - p = new_p; - } - if(lp->spx_status != DATAIGNORED) - ret = add_lag_con(lp, a_row, con_type, rhs); - FREE(a_row); - return( ret ); -} - -/* INLINE */ MYBOOL is_splitvar(lprec *lp, int colnr) -/* Two cases handled by var_is_free: - - 1) LB:-Inf / UB:var_is_free != NULL) && - (lp->var_is_free[colnr] < 0) && (-lp->var_is_free[colnr] != colnr))); -} - -void del_splitvars(lprec *lp) -{ - int j, jj, i; - - if(lp->var_is_free != NULL) { - for(j = lp->columns; j >= 1; j--) - if(is_splitvar(lp, j)) { - /* Check if we need to modify the basis */ - jj = lp->rows+abs(lp->var_is_free[j]); - i = lp->rows+j; - if(lp->is_basic[i] && !lp->is_basic[jj]) { - i = findBasisPos(lp, i, NULL); - set_basisvar(lp, i, jj); - } - /* Delete the helper column */ - del_column(lp, j); - } - FREE(lp->var_is_free); - } -} - -MYBOOL __WINAPI set_column(lprec *lp, int colnr, REAL *column) -{ - return( mat_setcol(lp->matA, colnr, lp->rows, column, NULL, TRUE, TRUE) ); -} - -MYBOOL __WINAPI set_columnex(lprec *lp, int colnr, int count, REAL *column, int *rowno) -{ - return( mat_setcol(lp->matA, colnr, count, column, rowno, TRUE, TRUE) ); -} - -MYBOOL __WINAPI add_columnex(lprec *lp, int count, REAL *column, int *rowno) -/* This function adds a data column to the current model; three cases handled: - - 1: Prepare for column data by setting column = NULL - 2: Dense vector indicated by (rowno == NULL) over 0..count+get_Lrows() elements - 3: Sparse vector set over row vectors rowno, over 0..count-1 elements. - - NB! If the column has only one entry, this should be handled as - a bound, but this currently is not the case */ -{ - MYBOOL status = FALSE; - - /* Prepare and shift column vectors */ - if(!append_columns(lp, 1)) - return( status ); - - /* Append sparse regular constraint values */ - if(mat_appendcol(lp->matA, count, column, rowno, 1.0, TRUE) < 0) - report(lp, SEVERE, "add_columnex: Data column %d supplied in non-ascending row index order.\n", - lp->columns); - else -#ifdef Paranoia - if(lp->columns != (lp->matA->is_roworder ? lp->matA->rows : lp->matA->columns)) { - report(lp, SEVERE, "add_columnex: Column count mismatch %d vs %d\n", - lp->columns, (lp->matA->is_roworder ? lp->matA->rows : lp->matA->columns)); - } - else if(is_BasisReady(lp) && (lp->P1extraDim == 0) && !verify_basis(lp)) - report(lp, SEVERE, "add_columnex: Invalid basis detected for column %d\n", - lp->columns); - else -#endif - status = TRUE; - - if(!lp->varmap_locked) - presolve_setOrig(lp, lp->rows, lp->columns); - - return( status ); -} - -MYBOOL __WINAPI add_column(lprec *lp, REAL *column) -{ - del_splitvars(lp); - return(add_columnex(lp, lp->rows, column, NULL)); -} - -MYBOOL __WINAPI str_add_column(lprec *lp, char *col_string) -{ - int i; - MYBOOL ret = TRUE; - REAL *aCol; - char *p, *newp; - - allocREAL(lp, &aCol, lp->rows + 1, FALSE); - p = col_string; - - for(i = 0; i <= lp->rows; i++) { - aCol[i] = (REAL) strtod(p, &newp); - if(p == newp) { - report(lp, IMPORTANT, "str_add_column: Bad string '%s'\n", p); - lp->spx_status = DATAIGNORED; - ret = FALSE; - break; - } - else - p = newp; - } - if(lp->spx_status != DATAIGNORED) - ret = add_column(lp, aCol); - FREE(aCol); - return( ret ); -} - -STATIC MYBOOL del_varnameex(lprec *lp, hashelem **namelist, int items, hashtable *ht, int varnr, LLrec *varmap) -{ - int i, n; - - /* First drop hash table entries of the deleted variables */ - if(varmap != NULL) - i = firstInactiveLink(varmap); - else - i = varnr; - while(i > 0) { - if(namelist[i] != NULL) { - if(namelist[i]->name != NULL) - drophash(namelist[i]->name, namelist, ht); - } - if(varmap != NULL) - i = nextInactiveLink(varmap, i); - else - i = 0; - } - - /* Then compress the name list */ - if(varmap != NULL) { - i = firstInactiveLink(varmap); - n = nextActiveLink(varmap, i); - varnr = i; - } - else { - i = varnr; - n = i + 1; - } - while(n != 0) { - namelist[i] = namelist[n]; - if((namelist[i] != NULL) && (namelist[i]->index > varnr)) - namelist[i]->index -= n - i; - i++; - if(varmap != NULL) - n = nextActiveLink(varmap, i); - else if(n <= items) /* items has been updated for the new count */ - n++; - else - n = 0; - } - - return( TRUE ); -} -STATIC MYBOOL del_columnex(lprec *lp, LLrec *colmap) -{ - varmap_delete(lp, lp->rows+1, -1, colmap); - shift_coldata(lp, 1, -1, colmap); - if(!lp->varmap_locked) { - presolve_setOrig(lp, lp->rows, lp->columns); - if(lp->names_used) - del_varnameex(lp, lp->col_name, lp->columns, lp->colname_hashtab, 0, colmap); - } -#ifdef Paranoia - if(is_BasisReady(lp) && (lp->P1extraDim == 0) && !verify_basis(lp)) - report(lp, SEVERE, "del_columnex: Invalid basis detected\n"); -#endif - - return(TRUE); -} -MYBOOL __WINAPI del_column(lprec *lp, int colnr) -{ - MYBOOL preparecompact = (MYBOOL) (colnr < 0); - - if(preparecompact) - colnr = -colnr; - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "del_column: Column %d out of range\n", colnr); - return(FALSE); - } - /* - if(lp->matA->is_roworder) { - report(lp, IMPORTANT, "del_column: Cannot delete column while in row entry mode.\n"); - return(FALSE); - } - */ - - if((lp->var_is_free != NULL) && (lp->var_is_free[colnr] > 0)) - del_column(lp, lp->var_is_free[colnr]); /* delete corresponding split column (is always after this column) */ - - varmap_delete(lp, my_chsign(preparecompact, lp->rows+colnr), -1, NULL); - shift_coldata(lp, my_chsign(preparecompact, colnr), -1, NULL); - if(!lp->varmap_locked) { - presolve_setOrig(lp, lp->rows, lp->columns); - if(lp->names_used) - del_varnameex(lp, lp->col_name, lp->columns, lp->colname_hashtab, colnr, NULL); - } -#ifdef Paranoia - if(is_BasisReady(lp) && (lp->P1extraDim == 0) && !verify_basis(lp)) - report(lp, SEVERE, "del_column: Invalid basis detected at column %d (%d)\n", colnr, lp->columns); -#endif - - return(TRUE); -} - -void __WINAPI set_simplextype(lprec *lp, int simplextype) -{ - lp->simplex_strategy = simplextype; -} - -int __WINAPI get_simplextype(lprec *lp) -{ - return(lp->simplex_strategy); -} - -void __WINAPI set_preferdual(lprec *lp, MYBOOL dodual) -{ - if(dodual & TRUE) - lp->simplex_strategy = SIMPLEX_DUAL_DUAL; - else - lp->simplex_strategy = SIMPLEX_PRIMAL_PRIMAL; -} - -void __WINAPI set_bounds_tighter(lprec *lp, MYBOOL tighten) -{ - lp->tighten_on_set = tighten; -} -MYBOOL __WINAPI get_bounds_tighter(lprec *lp) -{ - return(lp->tighten_on_set); -} - -MYBOOL __WINAPI set_upbo(lprec *lp, int colnr, REAL value) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_upbo: Column %d out of range\n", colnr); - return(FALSE); - } - -#ifdef DoBorderRounding - if(fabs(value) < lp->infinite) - value = my_avoidtiny(value, lp->matA->epsvalue); -#endif - value = scaled_value(lp, value, lp->rows + colnr); - if(lp->tighten_on_set) { - if(value < lp->orig_lowbo[lp->rows + colnr]) { - report(lp, IMPORTANT, "set_upbo: Upperbound must be >= lowerbound\n"); - return(FALSE); - } - if(value < lp->orig_upbo[lp->rows + colnr]) { - set_action(&lp->spx_action, ACTION_REBASE); - lp->orig_upbo[lp->rows + colnr] = value; - } - } - else - { - set_action(&lp->spx_action, ACTION_REBASE); - if(value > lp->infinite) - value = lp->infinite; - if (value < lp->infinite && lp->orig_lowbo[lp->rows + colnr] > -lp->infinite && value != lp->orig_lowbo[lp->rows + colnr] && fabs(value - lp->orig_lowbo[lp->rows + colnr]) < lp->epsvalue) - value = lp->orig_lowbo[lp->rows + colnr]; - lp->orig_upbo[lp->rows + colnr] = value; - } - return(TRUE); -} - -REAL __WINAPI get_upbo(lprec *lp, int colnr) -{ - REAL value; - - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "get_upbo: Column %d out of range\n", colnr); - return(0); - } - - value = lp->orig_upbo[lp->rows + colnr]; - value = unscaled_value(lp, value, lp->rows + colnr); - return(value); -} - -MYBOOL __WINAPI set_lowbo(lprec *lp, int colnr, REAL value) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_lowbo: Column %d out of range\n", colnr); - return(FALSE); - } - -#ifdef DoBorderRounding - if(fabs(value) < lp->infinite) - value = my_avoidtiny(value, lp->matA->epsvalue); -#endif - value = scaled_value(lp, value, lp->rows + colnr); - if(lp->tighten_on_set) { - if(value > lp->orig_upbo[lp->rows + colnr]) { - report(lp, IMPORTANT, "set_lowbo: Upper bound must be >= lower bound\n"); - return(FALSE); - } - if((value < 0) || (value > lp->orig_lowbo[lp->rows + colnr])) { - set_action(&lp->spx_action, ACTION_REBASE); - lp->orig_lowbo[lp->rows + colnr] = value; - } - } - else - { - set_action(&lp->spx_action, ACTION_REBASE); - if(value < -lp->infinite) - value = -lp->infinite; - if (value > -lp->infinite && lp->orig_upbo[lp->rows + colnr] < lp->infinite && value != lp->orig_upbo[lp->rows + colnr] && fabs(value - lp->orig_upbo[lp->rows + colnr]) < lp->epsvalue) - value = lp->orig_upbo[lp->rows + colnr]; - lp->orig_lowbo[lp->rows + colnr] = value; - } - return(TRUE); -} - -REAL __WINAPI get_lowbo(lprec *lp, int colnr) -{ - REAL value; - - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "get_lowbo: Column %d out of range\n", colnr); - return(0); - } - - value = lp->orig_lowbo[lp->rows + colnr]; - value = unscaled_value(lp, value, lp->rows + colnr); - return(value); -} - -MYBOOL __WINAPI set_bounds(lprec *lp, int colnr, REAL lower, REAL upper) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_bounds: Column %d out of range\n", colnr); - return(FALSE); - } - if(fabs(upper - lower) < lp->epsvalue) { - if(lower < 0) - lower = upper; - else - upper = lower; - } - else if(lower > upper) { - report(lp, IMPORTANT, "set_bounds: Column %d upper bound must be >= lower bound\n", - colnr); - return( FALSE ); - } - - colnr += lp->rows; - - if(lower < -lp->infinite) - lower = -lp->infinite; - else if(lp->scaling_used) { - lower = scaled_value(lp, lower, colnr); -#ifdef DoBorderRounding - lower = my_avoidtiny(lower, lp->matA->epsvalue); -#endif - } - - if(upper > lp->infinite) - upper = lp->infinite; - else if(lp->scaling_used) { - upper = scaled_value(lp, upper, colnr); -#ifdef DoBorderRounding - upper = my_avoidtiny(upper, lp->matA->epsvalue); -#endif - } - - lp->orig_lowbo[colnr] = lower; - lp->orig_upbo[colnr] = upper; - set_action(&lp->spx_action, ACTION_REBASE); - - return(TRUE); -} - -MYBOOL get_bounds(lprec *lp, int column, REAL *lower, REAL *upper) -{ - if((column > lp->columns) || (column < 1)) { - report(lp, IMPORTANT, "get_bounds: Column %d out of range", column); - return(FALSE); - } - - if(lower != NULL) - *lower = get_lowbo(lp, column); - if(upper != NULL) - *upper = get_upbo(lp, column); - - return(TRUE); -} - -MYBOOL __WINAPI set_int(lprec *lp, int colnr, MYBOOL var_type) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_int: Column %d out of range\n", colnr); - return(FALSE); - } - - if((lp->var_type[colnr] & ISINTEGER) != 0) { - lp->int_vars--; - lp->var_type[colnr] &= ~ISINTEGER; - } - if(var_type) { - lp->var_type[colnr] |= ISINTEGER; - lp->int_vars++; - if(lp->columns_scaled && !is_integerscaling(lp)) - unscale_columns(lp); - } - return(TRUE); -} - -MYBOOL __WINAPI is_int(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "is_int: Column %d out of range\n", colnr); - return(FALSE); - } - - return((lp->var_type[colnr] & ISINTEGER) != 0); -} - -MYBOOL __WINAPI is_SOS_var(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "is_SOS_var: Column %d out of range\n", colnr); - return(FALSE); - } - - return((lp->var_type[colnr] & ISSOS) != 0); -} - -int __WINAPI add_SOS(lprec *lp, char *name, int sostype, int priority, int count, int *sosvars, REAL *weights) -{ - SOSrec *SOS; - int k; - - if((sostype < 1) || (count < 0)) { - report(lp, IMPORTANT, "add_SOS: Invalid SOS type definition %d\n", sostype); - return( 0 ); - } - - /* Make sure SOSes of order 3 and higher are properly defined */ - if(sostype > 2) { - int j; - for(k = 0; k < count; k++) { - j = sosvars[k]; - if(!is_int(lp, j) || !is_semicont(lp, j)) { - report(lp, IMPORTANT, "add_SOS: SOS3+ members all have to be integer or semi-continuous.\n"); - return( 0 ); - } - } - } - - /* Make size in the list to handle another SOS record */ - if(lp->SOS == NULL) - lp->SOS = create_SOSgroup(lp); - - /* Create and append SOS to list */ - SOS = create_SOSrec(lp->SOS, name, sostype, priority, count, sosvars, weights); - k = append_SOSgroup(lp->SOS, SOS); - - return(k); -} - -STATIC int add_GUB(lprec *lp, char *name, int priority, int count, int *gubvars) -{ - SOSrec *GUB; - int k; - -#ifdef Paranoia - if(count < 0) { - report(lp, IMPORTANT, "add_GUB: Invalid GUB member count %d\n", count); - return(FALSE); - } -#endif - - /* Make size in the list to handle another GUB record */ - if(lp->GUB == NULL) - lp->GUB = create_SOSgroup(lp); - - /* Create and append GUB to list */ - GUB = create_SOSrec(lp->GUB, name, 1, priority, count, gubvars, NULL); - GUB->isGUB = TRUE; - k = append_SOSgroup(lp->GUB, GUB); - - return(k); -} - -MYBOOL __WINAPI set_binary(lprec *lp, int colnr, MYBOOL must_be_bin) -{ - MYBOOL status = FALSE; - - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_binary: Column %d out of range\n", colnr); - return( status ); - } - - status = set_int(lp, colnr, must_be_bin); - if(status && must_be_bin) - status = set_bounds(lp, colnr, 0, 1); - return( status ); -} - -MYBOOL __WINAPI is_binary(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "is_binary: Column %d out of range\n", colnr); - return(FALSE); - } - - return((MYBOOL) (((lp->var_type[colnr] & ISINTEGER) != 0) && - (get_lowbo(lp, colnr) == 0) && - (fabs(get_upbo(lp, colnr) - 1) < lp->epsprimal))); -} - -MYBOOL __WINAPI set_unbounded(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_unbounded: Column %d out of range\n", colnr); - return( FALSE ); - } - - return( set_bounds(lp, colnr, -lp->infinite, lp->infinite) ); -} - -MYBOOL __WINAPI is_unbounded(lprec *lp, int colnr) -{ - MYBOOL test; - - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "is_unbounded: Column %d out of range\n", colnr); - return(FALSE); - } - - test = is_splitvar(lp, colnr); - if(!test) { - colnr += lp->rows; - test = (MYBOOL) ((lp->orig_lowbo[colnr] <= -lp->infinite) && - (lp->orig_upbo[colnr] >= lp->infinite)); - } - return( test ); -} - -MYBOOL __WINAPI is_negative(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "is_negative: Column %d out of range\n", colnr); - return( FALSE ); - } - - colnr += lp->rows; - return( (MYBOOL) ((lp->orig_upbo[colnr] <= 0) && - (lp->orig_lowbo[colnr] < 0)) ); -} - -MYBOOL __WINAPI set_var_weights(lprec *lp, REAL *weights) -{ - if(lp->var_priority != NULL) { - FREE(lp->var_priority); - } - if(weights != NULL) { - int n; - allocINT(lp, &lp->var_priority, lp->columns_alloc, FALSE); - for(n = 0; n < lp->columns; n++) { - lp->var_priority[n] = n+1; - } - n = sortByREAL(lp->var_priority, weights, lp->columns, 0, FALSE); - } - return(TRUE); -} - -MYBOOL __WINAPI set_var_priority(lprec *lp) -/* Experimental automatic variable ordering/priority setting */ -{ - MYBOOL status = FALSE; - - if(is_bb_mode(lp, NODE_AUTOORDER) && - (lp->var_priority == NULL) && - (SOS_count(lp) == 0)) { - - REAL *rcost = NULL; - int i, j, *colorder = NULL; - - allocINT(lp, &colorder, lp->columns+1, FALSE); - - /* Create an "optimal" B&B variable ordering; this MDO-based routine - returns column indeces in an increasing order of co-dependency. - It can be argued that arranging the columns in right-to-left - MDO order should tend to minimize the consequences of choosing the - wrong variable by reducing the average B&B depth. */ - colorder[0] = lp->columns; - for(j = 1; j <= lp->columns; j++) - colorder[j] = lp->rows+j; - i = getMDO(lp, NULL, colorder, NULL, FALSE); - - /* Map to variable weight */ - allocREAL(lp, &rcost, lp->columns+1, FALSE); - for(j = lp->columns; j > 0; j--) { - i = colorder[j]-lp->rows; - rcost[i] = -j; - } - - /* Establish the MIP variable priorities */ - set_var_weights(lp, rcost+1); - - FREE(rcost); - FREE(colorder); - status = TRUE; - } - - return( status ); -} - -int __WINAPI get_var_priority(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "get_var_priority: Column %d out of range\n", colnr); - return(FALSE); - } - - if(lp->var_priority == NULL) - return(colnr); - else - return(lp->var_priority[colnr - 1]); -} - -MYBOOL __WINAPI set_semicont(lprec *lp, int colnr, MYBOOL must_be_sc) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "set_semicont: Column %d out of range\n", colnr); - return(FALSE); - } - - if(lp->sc_lobound[colnr] != 0) { - lp->sc_vars--; - lp->var_type[colnr] &= ~ISSEMI; - } - lp->sc_lobound[colnr] = must_be_sc; - if(must_be_sc) { - lp->var_type[colnr] |= ISSEMI; - lp->sc_vars++; - } - return(TRUE); -} - -MYBOOL __WINAPI is_semicont(lprec *lp, int colnr) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "is_semicont: Column %d out of range\n", colnr); - return(FALSE); - } - - return((lp->var_type[colnr] & ISSEMI) != 0); -} - -MYBOOL __WINAPI set_rh(lprec *lp, int rownr, REAL value) -{ - if((rownr > lp->rows) || (rownr < 0)) { - report(lp, IMPORTANT, "set_rh: Row %d out of range\n", rownr); - return(FALSE); - } - - if(((rownr == 0) && (!is_maxim(lp))) || - ((rownr > 0) && is_chsign(lp, rownr))) /* setting of RHS of OF IS meaningful */ - value = my_flipsign(value); - if(fabs(value) > lp->infinite) { - if(value < 0) - value = -lp->infinite; - else - value = lp->infinite; - } -#ifdef DoBorderRounding - else - value = my_avoidtiny(value, lp->matA->epsvalue); -#endif - value = scaled_value(lp, value, rownr); - lp->orig_rhs[rownr] = value; - set_action(&lp->spx_action, ACTION_RECOMPUTE); - return(TRUE); -} - -REAL __WINAPI get_rh(lprec *lp, int rownr) -{ - REAL value; - - if((rownr > lp->rows) || (rownr < 0)) { - report(lp, IMPORTANT, "get_rh: Row %d out of range", rownr); - return( 0.0 ); - } - - value = lp->orig_rhs[rownr]; - if (((rownr == 0) && !is_maxim(lp)) || - ((rownr > 0) && is_chsign(lp, rownr))) /* setting of RHS of OF IS meaningful */ - value = my_flipsign(value); - value = unscaled_value(lp, value, rownr); - return(value); -} - -REAL get_rh_upper(lprec *lp, int rownr) -{ - REAL value, valueR; - - value = lp->orig_rhs[rownr]; - if(is_chsign(lp, rownr)) { - valueR = lp->orig_upbo[rownr]; - if(is_infinite(lp, valueR)) - return(lp->infinite); - value = my_flipsign(value); - value += valueR; - } - value = unscaled_value(lp, value, rownr); - return(value); -} - -REAL get_rh_lower(lprec *lp, int rownr) -{ - REAL value, valueR; - - value = lp->orig_rhs[rownr]; - if(is_chsign(lp, rownr)) - value = my_flipsign(value); - else { - valueR = lp->orig_upbo[rownr]; - if(is_infinite(lp, valueR)) - return(-lp->infinite); - value -= valueR; - } - value = unscaled_value(lp, value, rownr); - return(value); -} - -MYBOOL set_rh_upper(lprec *lp, int rownr, REAL value) -{ - if(rownr > lp->rows || rownr < 1) { - report(lp, IMPORTANT, "set_rh_upper: Row %d out of range", rownr); - return(FALSE); - } - - /* First scale the value */ - value = scaled_value(lp, value, rownr); - - /* orig_rhs stores the upper bound assuming a < constraint; - If we have a > constraint, we must adjust the range instead */ - if(is_chsign(lp, rownr)) { - if(is_infinite(lp, value)) - lp->orig_upbo[rownr] = lp->infinite; - else { -#ifdef Paranoia - if(value + lp->orig_rhs[rownr] < 0) { - report(lp, SEVERE, "set_rh_upper: Invalid negative range in row %d\n", - rownr); - return(FALSE); - } -#endif -#ifdef DoBorderRounding - lp->orig_upbo[rownr] = my_avoidtiny(value + lp->orig_rhs[rownr], lp->epsvalue); -#else - lp->orig_upbo[rownr] = value + lp->orig_rhs[rownr]; -#endif - } - } - else { - /* If there is a constraint range, then this has to be adjusted also */ - if(!is_infinite(lp, lp->orig_upbo[rownr])) { - lp->orig_upbo[rownr] -= lp->orig_rhs[rownr] - value; - my_roundzero(lp->orig_upbo[rownr], lp->epsvalue); - if(lp->orig_upbo[rownr] < 0) { - report(lp, IMPORTANT, "set_rh_upper: Negative bound set for constraint %d made 0\n", rownr); - lp->orig_upbo[rownr] = 0; - } - } - lp->orig_rhs[rownr] = value; - } - return(TRUE); -} - -MYBOOL set_rh_lower(lprec *lp, int rownr, REAL value) -{ - if(rownr > lp->rows || rownr < 1) { - report(lp, IMPORTANT, "set_rh_lower: Row %d out of range", rownr); - return(FALSE); - } - - /* First scale the value */ - value = scaled_value(lp, value, rownr); - - /* orig_rhs stores the upper bound assuming a < constraint; - If we have a < constraint, we must adjust the range instead */ - if(!is_chsign(lp, rownr)) { - if(is_infinite(lp, value)) - lp->orig_upbo[rownr] = lp->infinite; - else { -#ifdef Paranoia - if(lp->orig_rhs[rownr] - value < 0) { - report(lp, SEVERE, "set_rh_lower: Invalid negative range in row %d\n", - rownr); - return(FALSE); - } -#endif -#ifdef DoBorderRounding - lp->orig_upbo[rownr] = my_avoidtiny(lp->orig_rhs[rownr] - value, lp->epsvalue); -#else - lp->orig_upbo[rownr] = lp->orig_rhs[rownr] - value; -#endif - } - } - else { - value = my_flipsign(value); - /* If there is a constraint range, then this has to be adjusted also */ - if(!is_infinite(lp, lp->orig_upbo[rownr])) { - lp->orig_upbo[rownr] -= lp->orig_rhs[rownr] - value; - my_roundzero(lp->orig_upbo[rownr], lp->epsvalue); - if(lp->orig_upbo[rownr] < 0) { - report(lp, IMPORTANT, "set_rh_lower: Negative bound set for constraint %d made 0\n", rownr); - lp->orig_upbo[rownr] = 0; - } - } - lp->orig_rhs[rownr] = value; - } - return(TRUE); -} - -MYBOOL __WINAPI set_rh_range(lprec *lp, int rownr, REAL deltavalue) -{ - if((rownr > lp->rows) || (rownr < 1)) { - report(lp, IMPORTANT, "set_rh_range: Row %d out of range", rownr); - return(FALSE); - } - - deltavalue = scaled_value(lp, deltavalue, rownr); - if(deltavalue > lp->infinite) - deltavalue = lp->infinite; - else if(deltavalue < -lp->infinite) - deltavalue = -lp->infinite; -#ifdef DoBorderRounding - else - deltavalue = my_avoidtiny(deltavalue, lp->matA->epsvalue); -#endif - - if(fabs(deltavalue) < lp->epsprimal) { - /* Conversion to EQ */ - set_constr_type(lp, rownr, EQ); - } - else if(is_constr_type(lp, rownr, EQ)) { - /* EQ with a non-zero range */ - if(deltavalue > 0) - set_constr_type(lp, rownr, GE); - else - set_constr_type(lp, rownr, LE); - lp->orig_upbo[rownr] = fabs(deltavalue); - } - else { - /* Modify GE/LE ranges */ - lp->orig_upbo[rownr] = fabs(deltavalue); - } - - return(TRUE); -} - -REAL __WINAPI get_rh_range(lprec *lp, int rownr) -{ - if((rownr > lp->rows) || (rownr < 0)) { - report(lp, IMPORTANT, "get_rh_range: row %d out of range\n", rownr); - return(FALSE); - } - - if(lp->orig_upbo[rownr] >= lp->infinite) - return(lp->orig_upbo[rownr]); - else - return(unscaled_value(lp, lp->orig_upbo[rownr], rownr)); -} - -void __WINAPI set_rh_vec(lprec *lp, REAL *rh) -{ - int i; - REAL rhi; - - for(i = 1; i <= lp->rows; i++) { - rhi = rh[i]; -#ifdef DoBorderRounding - rhi = my_avoidtiny(rhi, lp->matA->epsvalue); -#endif - lp->orig_rhs[i] = my_chsign(is_chsign(lp, i), scaled_value(lp, rhi, i)); - } - set_action(&lp->spx_action, ACTION_RECOMPUTE); -} - -MYBOOL __WINAPI str_set_rh_vec(lprec *lp, char *rh_string) -{ - int i; - MYBOOL ret = TRUE; - REAL *newrh; - char *p, *newp; - - allocREAL(lp, &newrh, lp->rows + 1, TRUE); - p = rh_string; - - for(i = 1; i <= lp->rows; i++) { - newrh[i] = (REAL) strtod(p, &newp); - if(p == newp) { - report(lp, IMPORTANT, "str_set_rh_vec: Bad string %s\n", p); - lp->spx_status = DATAIGNORED; - ret = FALSE; - break; - } - else - p = newp; - } - if(!(lp->spx_status == DATAIGNORED)) - set_rh_vec(lp, newrh); - FREE(newrh); - return( ret ); -} - -void __WINAPI set_sense(lprec *lp, MYBOOL maximize) -{ - maximize = (MYBOOL) (maximize != FALSE); - if(is_maxim(lp) != maximize) { - int i; - if(is_infinite(lp, lp->bb_heuristicOF)) - lp->bb_heuristicOF = my_chsign(maximize, lp->infinite); - if(is_infinite(lp, lp->bb_breakOF)) - lp->bb_breakOF = my_chsign(maximize, -lp->infinite); - lp->orig_rhs[0] = my_flipsign(lp->orig_rhs[0]); - for(i = 1; i <= lp->columns; i++) - lp->orig_obj[i] = my_flipsign(lp->orig_obj[i]); - set_action(&lp->spx_action, ACTION_REINVERT | ACTION_RECOMPUTE); - } - if(maximize) - lp->row_type[0] = ROWTYPE_OFMAX; - else - lp->row_type[0] = ROWTYPE_OFMIN; -} - -void __WINAPI set_maxim(lprec *lp) -{ - set_sense(lp, TRUE); -} - -void __WINAPI set_minim(lprec *lp) -{ - set_sense(lp, FALSE); -} - -MYBOOL __WINAPI is_maxim(lprec *lp) -{ - return( (MYBOOL) ((lp->row_type != NULL) && - ((lp->row_type[0] & ROWTYPE_CHSIGN) == ROWTYPE_GE)) ); -} - -MYBOOL __WINAPI set_constr_type(lprec *lp, int rownr, int con_type) -{ - MYBOOL oldchsign; - - if(rownr > lp->rows+1 || rownr < 1) { - report(lp, IMPORTANT, "set_constr_type: Row %d out of range\n", rownr); - return( FALSE ); - } - - /* Prepare for a new row */ - if((rownr > lp->rows) && !append_rows(lp, rownr-lp->rows)) - return( FALSE ); - - /* Update the constraint type data */ - if(is_constr_type(lp, rownr, EQ)) - lp->equalities--; - - if((con_type & ROWTYPE_CONSTRAINT) == EQ) { - lp->equalities++; - lp->orig_upbo[rownr] = 0; - } - else if(((con_type & LE) > 0) || ((con_type & GE) > 0) || (con_type == FR)) - lp->orig_upbo[rownr] = lp->infinite; - else { - report(lp, IMPORTANT, "set_constr_type: Constraint type %d not implemented (row %d)\n", - con_type, rownr); - return( FALSE ); - } - - /* Change the signs of the row, if necessary */ - oldchsign = is_chsign(lp, rownr); - if(con_type == FR) - lp->row_type[rownr] = LE; - else - lp->row_type[rownr] = con_type; - if(oldchsign != is_chsign(lp, rownr)) { - MATrec *mat = lp->matA; - - if(mat->is_roworder) - mat_multcol(mat, rownr, -1, FALSE); - else - mat_multrow(mat, rownr, -1); - if(lp->orig_rhs[rownr] != 0) - lp->orig_rhs[rownr] *= -1; - set_action(&lp->spx_action, ACTION_RECOMPUTE); - } - if(con_type == FR) - lp->orig_rhs[rownr] = lp->infinite; - - set_action(&lp->spx_action, ACTION_REINVERT); - lp->basis_valid = FALSE; - - return( TRUE ); -} - -/* INLINE */ MYBOOL is_chsign(lprec *lp, int rownr) -{ - return( (MYBOOL) ((lp->row_type[rownr] & ROWTYPE_CONSTRAINT) == ROWTYPE_CHSIGN) ); -} - -MYBOOL __WINAPI is_constr_type(lprec *lp, int rownr, int mask) -{ - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "is_constr_type: Row %d out of range\n", rownr); - return( FALSE ); - } - return( (MYBOOL) ((lp->row_type[rownr] & ROWTYPE_CONSTRAINT) == mask)); -} - -int __WINAPI get_constr_type(lprec *lp, int rownr) -{ - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "get_constr_type: Row %d out of range\n", rownr); - return(-1); - } - return( lp->row_type[rownr] ); -} -REAL __WINAPI get_constr_value(lprec *lp, int rownr, int count, REAL *primsolution, int *nzindex) -{ - int i; - REAL value = 0.0; - MATrec *mat = lp->matA; - - if((rownr < 0) || (rownr > get_Nrows(lp))) - return( value ); - - /* First do validation and initialization of applicable primal solution */ - if(!mat_validate(mat) || ((primsolution == NULL) && (lp->solvecount == 0))) - return( value ); - i = get_Ncolumns(lp); - if((primsolution != NULL) && (nzindex == NULL) && - ((count <= 0) || (count > i))) - count = i; - if(primsolution == NULL) { - get_ptr_variables(lp, &primsolution); - primsolution--; - nzindex = NULL; - count = i; - } - - /* Do objective or constraint, as specified */ - if(rownr == 0) { - value += get_rh(lp, 0); - if(nzindex != NULL) - for(i = 0; i < count; i++) - value += get_mat(lp, 0, nzindex[i]) * primsolution[i]; - else - for(i = 1; i <= count; i++) - value += get_mat(lp, 0, i) * primsolution[i]; - } - else { - if(nzindex != NULL) { - for(i = 0; i < count; i++) - value += get_mat(lp, rownr, nzindex[i]) * primsolution[i]; - } - else { - int j; - - for(i = mat->row_end[rownr-1]; i < mat->row_end[rownr]; i++) { - j = ROW_MAT_COLNR(i); - value += unscaled_mat(lp, ROW_MAT_VALUE(i), rownr, j) * primsolution[j]; - } - value = my_chsign(is_chsign(lp, rownr), value); - } - } - return( value ); -} - -STATIC char *get_str_constr_class(lprec *lp, int con_class) -{ - switch(con_class) { - case ROWCLASS_Unknown: return("Unknown"); - case ROWCLASS_Objective: return("Objective"); - case ROWCLASS_GeneralREAL: return("General REAL"); - case ROWCLASS_GeneralMIP: return("General MIP"); - case ROWCLASS_GeneralINT: return("General INT"); - case ROWCLASS_GeneralBIN: return("General BIN"); - case ROWCLASS_KnapsackINT: return("Knapsack INT"); - case ROWCLASS_KnapsackBIN: return("Knapsack BIN"); - case ROWCLASS_SetPacking: return("Set packing"); - case ROWCLASS_SetCover: return("Set cover"); - case ROWCLASS_GUB: return("GUB"); - default: return("Error"); - } -} - -STATIC char *get_str_constr_type(lprec *lp, int con_type) -{ - switch(con_type) { - case FR: return("FR"); - case LE: return("LE"); - case GE: return("GE"); - case EQ: return("EQ"); - default: return("Error"); - } -} - -STATIC int get_constr_class(lprec *lp, int rownr) -{ - int aBIN = 0, aINT = 0, aREAL = 0, - xBIN = 0, xINT = 0, xREAL = 0; - int j, elmnr, elmend, nelm; - MYBOOL chsign; - REAL a; - MATrec *mat = lp->matA; - - if((rownr < 1) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "get_constr_class: Row %d out of range\n", rownr); - return( ROWCLASS_Unknown ); - } - mat_validate(mat); - - /* Tally counts of constraint variable types and coefficients */ - if(rownr == 0) { - elmnr = 1; - elmend = lp->columns; - nelm = 0; - } - else { - elmnr = mat->row_end[rownr - 1]; - elmend = mat->row_end[rownr]; - nelm = elmend - elmnr; - } - chsign = is_chsign(lp, rownr); - for(; elmnr < elmend; elmnr++) { - if(rownr == 0) { - a = lp->orig_obj[elmnr]; - if(a == 0) - continue; - j = elmnr; - } - else { - j = ROW_MAT_COLNR(elmnr); - a = ROW_MAT_VALUE(elmnr); - } - a = unscaled_mat(lp, my_chsign(chsign, a), rownr, j); - if(is_binary(lp, j)) - xBIN++; - else if((get_lowbo(lp, j) >= 0) && is_int(lp, j)) - xINT++; - else - xREAL++; /* Includes integer variables with negative lower bound */ - - if(fabs(a-1.0) < lp->epsvalue) - aBIN++; - else if((a > 0) && (fabs(floor(a+lp->epsvalue)-a) < lp->epsvalue)) - aINT++; - else - aREAL++; /* Includes negative integer-valued coefficients */ - } - - /* Get the constraint type and the RHS */ - if(rownr == 0) - return( ROWCLASS_Objective ); - j = get_constr_type(lp, rownr); - a = get_rh(lp, rownr); - - /* Determine the constraint class */ - if((aBIN == nelm) && (xBIN == nelm) && (a >= 1)) { - if(a > 1) - j = ROWCLASS_KnapsackBIN; - else if(j == EQ) - j = ROWCLASS_GUB; - else if(j == LE) - j = ROWCLASS_SetCover; - else - j = ROWCLASS_SetPacking; - } - else if((aINT == nelm) && (xINT == nelm) && (a >= 1)) - j = ROWCLASS_KnapsackINT; - else if(xBIN == nelm) - j = ROWCLASS_GeneralBIN; - else if(xINT == nelm) - j = ROWCLASS_GeneralINT; - else if((xREAL > 0) && (xINT+xBIN > 0)) - j = ROWCLASS_GeneralMIP; - else - j = ROWCLASS_GeneralREAL; - - return( j ); -} - -REAL __WINAPI get_mat(lprec *lp, int rownr, int colnr) -{ - REAL value; - int elmnr; - int colnr1 = colnr, rownr1 = rownr; - - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "get_mat: Row %d out of range", rownr); - return(0); - } - if((colnr < 1) || (colnr > lp->columns)) { - report(lp, IMPORTANT, "get_mat: Column %d out of range", colnr); - return(0); - } - if(rownr == 0) { - value = lp->orig_obj[colnr]; - value = my_chsign(is_chsign(lp, rownr), value); - value = unscaled_mat(lp, value, rownr, colnr); - } - else { - if(lp->matA->is_roworder) - swapINT(&colnr1, &rownr1); - elmnr = mat_findelm(lp->matA, rownr1, colnr1); - if(elmnr >= 0) { - MATrec *mat = lp->matA; - value = my_chsign(is_chsign(lp, rownr), COL_MAT_VALUE(elmnr)); - value = unscaled_mat(lp, value, rownr, colnr); - } - else - value = 0; - } - return(value); -} - -REAL __WINAPI get_mat_byindex(lprec *lp, int matindex, MYBOOL isrow, MYBOOL adjustsign) -/* Note that this function does not adjust for sign-changed GT constraints! */ -{ - int *rownr, *colnr; - REAL *value, result; - - mat_get_data(lp, matindex, isrow, &rownr, &colnr, &value); - if(adjustsign) - result = (*value) * (is_chsign(lp, *rownr) ? -1 : 1); - else - result = *value; - if(lp->scaling_used) - return( unscaled_mat(lp, result, *rownr, *colnr) ); - else - return( result ); -} - -static int mat_getrow(lprec *lp, int rownr, REAL *row, int *colno) -{ - MYBOOL isnz; - int j, countnz = 0; - REAL a; - - if((rownr == 0) || !mat_validate(lp->matA)) { - for(j = 1; j <= lp->columns; j++) { - a = get_mat(lp,rownr,j); - isnz = (a != 0); - if(colno == NULL) - row[j] = a; - else if(isnz) { - row[countnz] = a; - colno[countnz] = j; - } - if(isnz) - countnz++; - } - } - else { - MYBOOL chsign = FALSE; - int ie, i; - MATrec *mat = lp->matA; - - if(colno == NULL) - MEMCLEAR(row, lp->columns+1); - if(mat->is_roworder) { - /* Add the objective function */ - a = get_mat(lp, 0, rownr); - if(colno == NULL) { - row[countnz] = a; - if(a != 0) - countnz++; - } - else if(a != 0) { - row[countnz] = a; - colno[countnz] = 0; - countnz++; - } - } - i = mat->row_end[rownr-1]; - ie = mat->row_end[rownr]; - if(!lp->matA->is_roworder) - chsign = is_chsign(lp, rownr); - for(; i < ie; i++) { - j = ROW_MAT_COLNR(i); - a = get_mat_byindex(lp, i, TRUE, FALSE); - if(lp->matA->is_roworder) - chsign = is_chsign(lp, j); - a = my_chsign(chsign, a); - if(colno == NULL) - row[j] = a; - else { - row[countnz] = a; - colno[countnz] = j; - } - countnz++; - } - } - return( countnz ); -} - -static int mat_getcolumn(lprec *lp, int colnr, REAL *column, int *nzrow) -{ - int n = 0, i, ii, ie, *rownr; - REAL hold, *value; - MATrec *mat = lp->matA; - - if(nzrow == NULL) - MEMCLEAR(column, lp->rows + 1); - if(!mat->is_roworder) { - /* Add the objective function */ - hold = get_mat(lp, 0, colnr); - if(nzrow == NULL) { - column[n] = hold; - if(hold != 0) - n++; - } - else if(hold != 0) { - column[n] = hold; - nzrow[n] = 0; - n++; - } - } - - i = lp->matA->col_end[colnr - 1]; - ie = lp->matA->col_end[colnr]; - if(nzrow == NULL) - n += ie - i; - rownr = &COL_MAT_ROWNR(i); - value = &COL_MAT_VALUE(i); - for(; i < ie; - i++, rownr += matRowColStep, value += matValueStep) { - ii = *rownr; - - hold = my_chsign(is_chsign(lp, (mat->is_roworder) ? colnr : ii), *value); - hold = unscaled_mat(lp, hold, ii, colnr); - if(nzrow == NULL) - column[ii] = hold; - else if(hold != 0) { - column[n] = hold; - nzrow[n] = ii; - n++; - } - } - return( n ); -} - -int __WINAPI get_columnex(lprec *lp, int colnr, REAL *column, int *nzrow) -{ - if((colnr > lp->columns) || (colnr < 1)) { - report(lp, IMPORTANT, "get_columnex: Column %d out of range\n", colnr); - return( -1 ); - } - - if(lp->matA->is_roworder) - return(mat_getrow(lp, colnr, column, nzrow)); - else - return(mat_getcolumn(lp, colnr, column, nzrow)); -} - -MYBOOL __WINAPI get_column(lprec *lp, int colnr, REAL *column) -{ - return( (MYBOOL) (get_columnex(lp, colnr, column, NULL) >= 0) ); -} - -int __WINAPI get_rowex(lprec *lp, int rownr, REAL *row, int *colno) -{ - if((rownr < 0) || (rownr > lp->rows)) { - report(lp, IMPORTANT, "get_rowex: Row %d out of range\n", rownr); - return( -1 ); - } - - if(rownr != 0 && lp->matA->is_roworder) - return(mat_getcolumn(lp, rownr, row, colno)); - else - return(mat_getrow(lp, rownr, row, colno)); -} - -MYBOOL __WINAPI get_row(lprec *lp, int rownr, REAL *row) -{ - return((MYBOOL) (get_rowex(lp, rownr, row, NULL) >= 0) ); -} - -STATIC void set_OF_override(lprec *lp, REAL *ofVector) -/* The purpose of this function is to set, or clear if NULL, the - ofVector[0..columns] as the active objective function instead of - the one stored in the A-matrix. See also lag_solve().*/ -{ - lp->obj = ofVector; -} - -MYBOOL modifyOF1(lprec *lp, int index, REAL *ofValue, REAL mult) -/* Adjust objective function values for primal/dual phase 1, if appropriate */ -{ - MYBOOL accept = TRUE; - - /* Primal simplex: Set user variables to zero or BigM-scaled */ - if(((lp->simplex_mode & SIMPLEX_Phase1_PRIMAL) != 0) && (abs(lp->P1extraDim) > 0)) { -#ifndef Phase1EliminateRedundant - if(lp->P1extraDim < 0) { - if(index > lp->sum + lp->P1extraDim) - accept = FALSE; - } - else -#endif - if((index <= lp->sum - lp->P1extraDim) || (mult == 0)) { - if((mult == 0) || (lp->bigM == 0)) - accept = FALSE; - else - (*ofValue) /= lp->bigM; - } - } - - /* Dual simplex: Subtract P1extraVal from objective function values */ - else if(((lp->simplex_mode & SIMPLEX_Phase1_DUAL) != 0) && (index > lp->rows)) { -#if 1 /* This may help increase sparsity of the (extended) basis matrix; - Can it introduce degeneracy in some cases? */ - if((lp->P1extraVal != 0) && (lp->orig_obj[index - lp->rows] > 0)) - *ofValue = 0; - else -#endif - { - *ofValue -= lp->P1extraVal; -#if 0 - if(is_action(lp->anti_degen, ANTIDEGEN_RHSPERTURB)) - *ofValue -= rand_uniform(lp, lp->epsperturb); -#endif - } - } - - /* Do scaling and test for zero */ - if(accept) { - (*ofValue) *= mult; - if(fabs(*ofValue) < lp->epsmachine) { - (*ofValue) = 0; - accept = FALSE; - } - } - else - (*ofValue) = 0; - - return( accept ); -} - -STATIC void set_OF_p1extra(lprec *lp, REAL p1extra) -{ - int i; - REAL *value; - - if(lp->spx_trace) - report(lp, DETAILED, "set_OF_p1extra: Set dual objective offset to %g at iter %.0f.\n", - p1extra, (double) get_total_iter(lp)); - lp->P1extraVal = p1extra; - if(lp->obj == NULL) - allocREAL(lp, &lp->obj, lp->columns_alloc+1, TRUE); - for(i = 1, value = lp->obj+1; i <= lp->columns; i++, value++) { - *value = lp->orig_obj[i]; - modifyOF1(lp, lp->rows + i, value, 1.0); - } -} - -STATIC void unset_OF_p1extra(lprec *lp) -{ - lp->P1extraVal = 0; - FREE(lp->obj); -} - -REAL __WINAPI get_OF_active(lprec *lp, int varnr, REAL mult) -{ - int colnr = varnr - lp->rows; - REAL holdOF = 0; - -#ifdef Paranoia - if((colnr <= 0) || (colnr > lp->columns)) { - report(lp, SEVERE, "get_OF_active: Invalid column index %d supplied\n", colnr); - } - else -#endif - if(lp->obj == NULL) { - if(colnr > 0) - holdOF = lp->orig_obj[colnr]; - modifyOF1(lp, varnr, &holdOF, mult); - } - else if(colnr > 0) - holdOF = lp->obj[colnr] * mult; - - return( holdOF ); -} - -STATIC MYBOOL is_OF_nz(lprec *lp, int colnr) -{ - return( (MYBOOL) (lp->orig_obj[colnr] != 0) ); -} - -STATIC int singleton_column(lprec *lp, int row_nr, REAL *column, int *nzlist, REAL value, int *maxabs) -{ - int nz = 1; - - if(nzlist == NULL) { - MEMCLEAR(column, lp->rows + 1); - column[row_nr] = value; - } - else { - column[nz] = value; - nzlist[nz] = row_nr; - } - - if(maxabs != NULL) - *maxabs = row_nr; - return( nz ); -} - -STATIC int expand_column(lprec *lp, int col_nr, REAL *column, int *nzlist, REAL mult, int *maxabs) -{ - int i, ie, j, maxidx, nzcount; - REAL value, maxval; - MATrec *mat = lp->matA; - REAL *matValue; - int *matRownr; - - /* Retrieve a column from the user data matrix A */ - maxval = 0; - maxidx = -1; - if(nzlist == NULL) { - MEMCLEAR(column, lp->rows + 1); - i = mat->col_end[col_nr - 1]; - ie = mat->col_end[col_nr]; - matRownr = &COL_MAT_ROWNR(i); - matValue = &COL_MAT_VALUE(i); - nzcount = i; - for(; i < ie; - i++, matRownr += matRowColStep, matValue += matValueStep) { - j = *matRownr; - value = *matValue; - if(j > 0) { - value *= mult; - if(fabs(value) > maxval) { - maxval = fabs(value); - maxidx = j; - } - } - column[j] = value; - } - nzcount = i - nzcount; - - /* Get the objective as row 0, optionally adjusting the objective for phase 1 */ - if(lp->obj_in_basis) { - column[0] = get_OF_active(lp, lp->rows+col_nr, mult); - if(column[0] != 0) - nzcount++; - } - } - else { - nzcount = 0; - - /* Get the objective as row 0, optionally adjusting the objective for phase 1 */ - if(lp->obj_in_basis) { - value = get_OF_active(lp, lp->rows+col_nr, mult); - if(value != 0) { - nzcount++; - nzlist[nzcount] = 0; - column[nzcount] = value; - } - } - - /* Loop over the non-zero column entries */ - i = mat->col_end[col_nr - 1]; - ie = mat->col_end[col_nr]; - matRownr = &COL_MAT_ROWNR(i); - matValue = &COL_MAT_VALUE(i); - for(; i < ie; - i++, matRownr += matRowColStep, matValue += matValueStep) { - j = *matRownr; - value = (*matValue) * mult; - nzcount++; - nzlist[nzcount] = j; - column[nzcount] = value; - if(fabs(value) > maxval) { - maxval = fabs(value); - maxidx = nzcount; - } - } - } - - if(maxabs != NULL) - *maxabs = maxidx; - return( nzcount ); -} - - -/* Retrieve a column vector from the data matrix [1..rows, rows+1..rows+columns]; - needs __WINAPI call model since it may be called from BFPs */ -int __WINAPI obtain_column(lprec *lp, int varin, REAL *pcol, int *nzlist, int *maxabs) -{ - REAL value = my_chsign(lp->is_lower[varin], -1); - if(varin > lp->rows) { - varin -= lp->rows; - varin = expand_column(lp, varin, pcol, nzlist, value, maxabs); - } - else if(lp->obj_in_basis || (varin > 0)) - varin = singleton_column(lp, varin, pcol, nzlist, value, maxabs); - else - varin = get_basisOF(lp, NULL, pcol, nzlist); - - return(varin); -} - -/* GENERAL INVARIANT CALLBACK FUNCTIONS */ -MYBOOL set_callbacks(lprec *lp) -{ - /* Assign API functions to lp structure (mainly for XLIs) */ - lp->add_column = add_column; - lp->add_columnex = add_columnex; - lp->add_constraint = add_constraint; - lp->add_constraintex = add_constraintex; - lp->add_lag_con = add_lag_con; - lp->add_SOS = add_SOS; - lp->column_in_lp = column_in_lp; - lp->copy_lp = copy_lp; - lp->default_basis = default_basis; - lp->del_column = del_column; - lp->del_constraint = del_constraint; - lp->delete_lp = delete_lp; - lp->dualize_lp = dualize_lp; - lp->free_lp = free_lp; - lp->get_anti_degen = get_anti_degen; - lp->get_basis = get_basis; - lp->get_basiscrash = get_basiscrash; - lp->get_bb_depthlimit = get_bb_depthlimit; - lp->get_bb_floorfirst = get_bb_floorfirst; - lp->get_bb_rule = get_bb_rule; - lp->get_bounds_tighter = get_bounds_tighter; - lp->get_break_at_value = get_break_at_value; - lp->get_col_name = get_col_name; - lp->get_columnex = get_columnex; - lp->get_constr_type = get_constr_type; - lp->get_constr_value = get_constr_value; - lp->get_constraints = get_constraints; - lp->get_dual_solution = get_dual_solution; - lp->get_epsb = get_epsb; - lp->get_epsd = get_epsd; - lp->get_epsel = get_epsel; - lp->get_epsint = get_epsint; - lp->get_epsperturb = get_epsperturb; - lp->get_epspivot = get_epspivot; - lp->get_improve = get_improve; - lp->get_infinite = get_infinite; - lp->get_lambda = get_lambda; - lp->get_lowbo = get_lowbo; - lp->get_lp_index = get_lp_index; - lp->get_lp_name = get_lp_name; - lp->get_Lrows = get_Lrows; - lp->get_mat = get_mat; - lp->get_mat_byindex = get_mat_byindex; - lp->get_max_level = get_max_level; - lp->get_maxpivot = get_maxpivot; - lp->get_mip_gap = get_mip_gap; - lp->get_multiprice = get_multiprice; - lp->get_nameindex = get_nameindex; - lp->get_Ncolumns = get_Ncolumns; - lp->get_negrange = get_negrange; - lp->get_nonzeros = get_nonzeros; - lp->get_Norig_columns = get_Norig_columns; - lp->get_Norig_rows = get_Norig_rows; - lp->get_Nrows = get_Nrows; - lp->get_obj_bound = get_obj_bound; - lp->get_objective = get_objective; - lp->get_orig_index = get_orig_index; - lp->get_origcol_name = get_origcol_name; - lp->get_origrow_name = get_origrow_name; - lp->get_partialprice = get_partialprice; - lp->get_pivoting = get_pivoting; - lp->get_presolve = get_presolve; - lp->get_presolveloops = get_presolveloops; - lp->get_primal_solution = get_primal_solution; - lp->get_print_sol = get_print_sol; - lp->get_pseudocosts = get_pseudocosts; - lp->get_ptr_constraints = get_ptr_constraints; - lp->get_ptr_dual_solution = get_ptr_dual_solution; - lp->get_ptr_lambda = get_ptr_lambda; - lp->get_ptr_primal_solution = get_ptr_primal_solution; - lp->get_ptr_sensitivity_obj = get_ptr_sensitivity_obj; - lp->get_ptr_sensitivity_objex = get_ptr_sensitivity_objex; - lp->get_ptr_sensitivity_rhs = get_ptr_sensitivity_rhs; - lp->get_ptr_variables = get_ptr_variables; - lp->get_rh = get_rh; - lp->get_rh_range = get_rh_range; - lp->get_row = get_row; - lp->get_rowex = get_rowex; - lp->get_row_name = get_row_name; - lp->get_scalelimit = get_scalelimit; - lp->get_scaling = get_scaling; - lp->get_sensitivity_obj = get_sensitivity_obj; - lp->get_sensitivity_objex = get_sensitivity_objex; - lp->get_sensitivity_rhs = get_sensitivity_rhs; - lp->get_simplextype = get_simplextype; - lp->get_solutioncount = get_solutioncount; - lp->get_solutionlimit = get_solutionlimit; - lp->get_status = get_status; - lp->get_statustext = get_statustext; - lp->get_timeout = get_timeout; - lp->get_total_iter = get_total_iter; - lp->get_total_nodes = get_total_nodes; - lp->get_upbo = get_upbo; - lp->get_var_branch = get_var_branch; - lp->get_var_dualresult = get_var_dualresult; - lp->get_var_primalresult = get_var_primalresult; - lp->get_var_priority = get_var_priority; - lp->get_variables = get_variables; - lp->get_verbose = get_verbose; - lp->get_working_objective = get_working_objective; - lp->has_BFP = has_BFP; - lp->has_XLI = has_XLI; - lp->is_add_rowmode = is_add_rowmode; - lp->is_anti_degen = is_anti_degen; - lp->is_binary = is_binary; - lp->is_break_at_first = is_break_at_first; - lp->is_constr_type = is_constr_type; - lp->is_debug = is_debug; - lp->is_feasible = is_feasible; - lp->is_unbounded = is_unbounded; - lp->is_infinite = is_infinite; - lp->is_int = is_int; - lp->is_integerscaling = is_integerscaling; - lp->is_lag_trace = is_lag_trace; - lp->is_maxim = is_maxim; - lp->is_nativeBFP = is_nativeBFP; - lp->is_nativeXLI = is_nativeXLI; - lp->is_negative = is_negative; - lp->is_obj_in_basis = is_obj_in_basis; - lp->is_piv_mode = is_piv_mode; - lp->is_piv_rule = is_piv_rule; - lp->is_presolve = is_presolve; - lp->is_scalemode = is_scalemode; - lp->is_scaletype = is_scaletype; - lp->is_semicont = is_semicont; - lp->is_SOS_var = is_SOS_var; - lp->is_trace = is_trace; - lp->lp_solve_version = lp_solve_version; - lp->make_lp = make_lp; - lp->print_constraints = print_constraints; - lp->print_debugdump = print_debugdump; - lp->print_duals = print_duals; - lp->print_lp = print_lp; - lp->print_objective = print_objective; - lp->print_scales = print_scales; - lp->print_solution = print_solution; - lp->print_str = print_str; - lp->print_tableau = print_tableau; - lp->put_abortfunc = put_abortfunc; - lp->put_bb_nodefunc = put_bb_nodefunc; - lp->put_bb_branchfunc = put_bb_branchfunc; - lp->put_logfunc = put_logfunc; - lp->put_msgfunc = put_msgfunc; - lp->read_LP = read_LP; - lp->read_MPS = read_MPS; - lp->read_XLI = read_XLI; - lp->read_basis = read_basis; - lp->reset_basis = reset_basis; - lp->read_params = read_params; - lp->reset_params = reset_params; - lp->resize_lp = resize_lp; - lp->set_action = set_action; - lp->set_add_rowmode = set_add_rowmode; - lp->set_anti_degen = set_anti_degen; - lp->set_basisvar = set_basisvar; - lp->set_basis = set_basis; - lp->set_basiscrash = set_basiscrash; - lp->set_bb_depthlimit = set_bb_depthlimit; - lp->set_bb_floorfirst = set_bb_floorfirst; - lp->set_bb_rule = set_bb_rule; - lp->set_BFP = set_BFP; - lp->set_binary = set_binary; - lp->set_bounds = set_bounds; - lp->set_bounds_tighter = set_bounds_tighter; - lp->set_break_at_first = set_break_at_first; - lp->set_break_at_value = set_break_at_value; - lp->set_col_name = set_col_name; - lp->set_constr_type = set_constr_type; - lp->set_debug = set_debug; - lp->set_epsb = set_epsb; - lp->set_epsd = set_epsd; - lp->set_epsel = set_epsel; - lp->set_epsint = set_epsint; - lp->set_epslevel = set_epslevel; - lp->set_epsperturb = set_epsperturb; - lp->set_epspivot = set_epspivot; - lp->set_unbounded = set_unbounded; - lp->set_improve = set_improve; - lp->set_infinite = set_infinite; - lp->set_int = set_int; - lp->set_lag_trace = set_lag_trace; - lp->set_lowbo = set_lowbo; - lp->set_lp_name = set_lp_name; - lp->set_mat = set_mat; - lp->set_maxim = set_maxim; - lp->set_maxpivot = set_maxpivot; - lp->set_minim = set_minim; - lp->set_mip_gap = set_mip_gap; - lp->set_multiprice = set_multiprice; - lp->set_negrange = set_negrange; - lp->set_obj = set_obj; - lp->set_obj_bound = set_obj_bound; - lp->set_obj_fn = set_obj_fn; - lp->set_obj_fnex = set_obj_fnex; - lp->set_obj_in_basis = set_obj_in_basis; - lp->set_outputfile = set_outputfile; - lp->set_outputstream = set_outputstream; - lp->set_partialprice = set_partialprice; - lp->set_pivoting = set_pivoting; - lp->set_preferdual = set_preferdual; - lp->set_presolve = set_presolve; - lp->set_print_sol = set_print_sol; - lp->set_pseudocosts = set_pseudocosts; - lp->set_rh = set_rh; - lp->set_rh_range = set_rh_range; - lp->set_rh_vec = set_rh_vec; - lp->set_row = set_row; - lp->set_rowex = set_rowex; - lp->set_row_name = set_row_name; - lp->set_scalelimit = set_scalelimit; - lp->set_scaling = set_scaling; - lp->set_semicont = set_semicont; - lp->set_sense = set_sense; - lp->set_simplextype = set_simplextype; - lp->set_solutionlimit = set_solutionlimit; - lp->set_timeout = set_timeout; - lp->set_trace = set_trace; - lp->set_upbo = set_upbo; - lp->set_var_branch = set_var_branch; - lp->set_var_weights = set_var_weights; - lp->set_verbose = set_verbose; - lp->set_XLI = set_XLI; - lp->solve = solve; - lp->str_add_column = str_add_column; - lp->str_add_constraint = str_add_constraint; - lp->str_add_lag_con = str_add_lag_con; - lp->str_set_obj_fn = str_set_obj_fn; - lp->str_set_rh_vec = str_set_rh_vec; - lp->time_elapsed = time_elapsed; - lp->unscale = unscale; - lp->write_lp = write_lp; - lp->write_LP = write_LP; - lp->write_mps = write_mps; - lp->write_freemps = write_freemps; - lp->write_MPS = write_MPS; - lp->write_freeMPS = write_freeMPS; - lp->write_XLI = write_XLI; - lp->write_basis = write_basis; - lp->write_params = write_params; - - /* Utility functions (mainly for BFPs) */ - lp->userabort = userabort; - lp->report = report; - lp->explain = explain; - lp->set_basisvar = set_basisvar; - lp->get_lpcolumn = obtain_column; - lp->get_basiscolumn = get_basiscolumn; - lp->get_OF_active = get_OF_active; - lp->getMDO = getMDO; - lp->invert = invert; - lp->set_action = set_action; - lp->clear_action = clear_action; - lp->is_action = is_action; - - return( TRUE ); -} - -/* SUPPORT FUNCTION FOR BASIS FACTORIZATION PACKAGES */ -MYBOOL __WINAPI has_BFP(lprec *lp) -{ - return( is_nativeBFP(lp) -#if LoadInverseLib == TRUE - || (MYBOOL) (lp->hBFP != NULL) -#endif - ); -} - -MYBOOL __WINAPI is_nativeBFP(lprec *lp) -{ -#ifdef ExcludeNativeInverse - return( FALSE ); -#elif LoadInverseLib == TRUE - return( (MYBOOL) (lp->hBFP == NULL) ); -#else - return( TRUE ); -#endif -} - -MYBOOL __WINAPI set_BFP(lprec *lp, char *filename) -/* (Re)mapping of basis factorization variant methods is done here */ -{ - int result = LIB_LOADED; - - /* Release the BFP and basis if we are active */ - if(lp->invB != NULL) - bfp_free(lp); - -#if LoadInverseLib == TRUE - if(lp->hBFP != NULL) { - #ifdef WIN32 - FreeLibrary(lp->hBFP); - #else - dlclose(lp->hBFP); - #endif - lp->hBFP = NULL; - } -#endif - - if(filename == NULL) { - if(!is_nativeBFP(lp)) - return( FALSE ); -#ifndef ExcludeNativeInverse - lp->bfp_name = bfp_name; - lp->bfp_compatible = bfp_compatible; - lp->bfp_free = bfp_free; - lp->bfp_resize = bfp_resize; - lp->bfp_nonzeros = bfp_nonzeros; - lp->bfp_memallocated = bfp_memallocated; - lp->bfp_restart = bfp_restart; - lp->bfp_mustrefactorize = bfp_mustrefactorize; - lp->bfp_preparefactorization = bfp_preparefactorization; - lp->bfp_factorize = bfp_factorize; - lp->bfp_finishupdate = bfp_finishupdate; - lp->bfp_ftran_normal = bfp_ftran_normal; - lp->bfp_ftran_prepare = bfp_ftran_prepare; - lp->bfp_btran_normal = bfp_btran_normal; - lp->bfp_status = bfp_status; - lp->bfp_implicitslack = bfp_implicitslack; - lp->bfp_indexbase = bfp_indexbase; - lp->bfp_rowoffset = bfp_rowoffset; - lp->bfp_pivotmax = bfp_pivotmax; - lp->bfp_init = bfp_init; - lp->bfp_pivotalloc = bfp_pivotalloc; - lp->bfp_colcount = bfp_colcount; - lp->bfp_canresetbasis = bfp_canresetbasis; - lp->bfp_finishfactorization = bfp_finishfactorization; - lp->bfp_updaterefactstats = bfp_updaterefactstats; - lp->bfp_prepareupdate = bfp_prepareupdate; - lp->bfp_pivotRHS = bfp_pivotRHS; - lp->bfp_btran_double = bfp_btran_double; - lp->bfp_efficiency = bfp_efficiency; - lp->bfp_pivotvector = bfp_pivotvector; - lp->bfp_pivotcount = bfp_pivotcount; - lp->bfp_refactcount = bfp_refactcount; - lp->bfp_isSetI = bfp_isSetI; - lp->bfp_findredundant = bfp_findredundant; -#endif - } - else { -#if LoadInverseLib == TRUE - #ifdef WIN32 - /* Get a handle to the Windows DLL module. */ - lp->hBFP = LoadLibrary(filename); - - /* If the handle is valid, try to get the function addresses. */ - if(lp->hBFP != NULL) { - lp->bfp_compatible = (BFPbool_lpintintint *) - GetProcAddress(lp->hBFP, "bfp_compatible"); - if(lp->bfp_compatible == NULL) - result = LIB_NOINFO; - else if(lp->bfp_compatible(lp, BFPVERSION, MAJORVERSION, sizeof(REAL))) { - - lp->bfp_name = (BFPchar *) - GetProcAddress(lp->hBFP, "bfp_name"); - lp->bfp_free = (BFP_lp *) - GetProcAddress(lp->hBFP, "bfp_free"); - lp->bfp_resize = (BFPbool_lpint *) - GetProcAddress(lp->hBFP, "bfp_resize"); - lp->bfp_nonzeros = (BFPint_lpbool *) - GetProcAddress(lp->hBFP, "bfp_nonzeros"); - lp->bfp_memallocated = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_memallocated"); - lp->bfp_restart = (BFPbool_lp *) - GetProcAddress(lp->hBFP, "bfp_restart"); - lp->bfp_mustrefactorize = (BFPbool_lp *) - GetProcAddress(lp->hBFP, "bfp_mustrefactorize"); - lp->bfp_preparefactorization = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_preparefactorization"); - lp->bfp_factorize = (BFPint_lpintintboolbool *) - GetProcAddress(lp->hBFP, "bfp_factorize"); - lp->bfp_finishupdate = (BFPbool_lpbool *) - GetProcAddress(lp->hBFP, "bfp_finishupdate"); - lp->bfp_ftran_normal = (BFP_lprealint *) - GetProcAddress(lp->hBFP, "bfp_ftran_normal"); - lp->bfp_ftran_prepare = (BFP_lprealint *) - GetProcAddress(lp->hBFP, "bfp_ftran_prepare"); - lp->bfp_btran_normal = (BFP_lprealint *) - GetProcAddress(lp->hBFP, "bfp_btran_normal"); - lp->bfp_status = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_status"); - lp->bfp_implicitslack = (BFPbool_lp *) - GetProcAddress(lp->hBFP, "bfp_implicitslack"); - lp->bfp_indexbase = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_indexbase"); - lp->bfp_rowoffset = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_rowoffset"); - lp->bfp_pivotmax = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_pivotmax"); - lp->bfp_init = (BFPbool_lpintintchar *) - GetProcAddress(lp->hBFP, "bfp_init"); - lp->bfp_pivotalloc = (BFPbool_lpint *) - GetProcAddress(lp->hBFP, "bfp_pivotalloc"); - lp->bfp_colcount = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_colcount"); - lp->bfp_canresetbasis = (BFPbool_lp *) - GetProcAddress(lp->hBFP, "bfp_canresetbasis"); - lp->bfp_finishfactorization = (BFP_lp *) - GetProcAddress(lp->hBFP, "bfp_finishfactorization"); - lp->bfp_updaterefactstats = (BFP_lp *) - GetProcAddress(lp->hBFP, "bfp_updaterefactstats"); - lp->bfp_prepareupdate = (BFPlreal_lpintintreal *) - GetProcAddress(lp->hBFP, "bfp_prepareupdate"); - lp->bfp_pivotRHS = (BFPreal_lplrealreal *) - GetProcAddress(lp->hBFP, "bfp_pivotRHS"); - lp->bfp_btran_double = (BFP_lprealintrealint *) - GetProcAddress(lp->hBFP, "bfp_btran_double"); - lp->bfp_efficiency = (BFPreal_lp *) - GetProcAddress(lp->hBFP, "bfp_efficiency"); - lp->bfp_pivotvector = (BFPrealp_lp *) - GetProcAddress(lp->hBFP, "bfp_pivotvector"); - lp->bfp_pivotcount = (BFPint_lp *) - GetProcAddress(lp->hBFP, "bfp_pivotcount"); - lp->bfp_refactcount = (BFPint_lpint *) - GetProcAddress(lp->hBFP, "bfp_refactcount"); - lp->bfp_isSetI = (BFPbool_lp *) - GetProcAddress(lp->hBFP, "bfp_isSetI"); - lp->bfp_findredundant = (BFPint_lpintrealcbintint *) - GetProcAddress(lp->hBFP, "bfp_findredundant"); - } - else - result = LIB_VERINVALID; - } - #else - /* First standardize UNIX .SO library name format. */ - char bfpname[260], *ptr; - - strcpy(bfpname, filename); - if((ptr = strrchr(filename, '/')) == NULL) - ptr = filename; - else - ptr++; - bfpname[(int) (ptr - filename)] = 0; - if(strncmp(ptr, "lib", 3)) - strcat(bfpname, "lib"); - strcat(bfpname, ptr); - if(strcmp(bfpname + strlen(bfpname) - 3, ".so")) - strcat(bfpname, ".so"); - - /* Get a handle to the module. */ - lp->hBFP = dlopen(bfpname, RTLD_LAZY); - - /* If the handle is valid, try to get the function addresses. */ - if(lp->hBFP != NULL) { - lp->bfp_compatible = (BFPbool_lpintintint *) - dlsym(lp->hBFP, "bfp_compatible"); - if(lp->bfp_compatible == NULL) - result = LIB_NOINFO; - else if(lp->bfp_compatible(lp, BFPVERSION, MAJORVERSION, sizeof(REAL))) { - - lp->bfp_name = (BFPchar *) - dlsym(lp->hBFP, "bfp_name"); - lp->bfp_free = (BFP_lp *) - dlsym(lp->hBFP, "bfp_free"); - lp->bfp_resize = (BFPbool_lpint *) - dlsym(lp->hBFP, "bfp_resize"); - lp->bfp_nonzeros = (BFPint_lpbool *) - dlsym(lp->hBFP, "bfp_nonzeros"); - lp->bfp_memallocated = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_memallocated"); - lp->bfp_restart = (BFPbool_lp *) - dlsym(lp->hBFP, "bfp_restart"); - lp->bfp_mustrefactorize = (BFPbool_lp *) - dlsym(lp->hBFP, "bfp_mustrefactorize"); - lp->bfp_preparefactorization = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_preparefactorization"); - lp->bfp_factorize = (BFPint_lpintintboolbool *) - dlsym(lp->hBFP, "bfp_factorize"); - lp->bfp_finishupdate = (BFPbool_lpbool *) - dlsym(lp->hBFP, "bfp_finishupdate"); - lp->bfp_ftran_normal = (BFP_lprealint *) - dlsym(lp->hBFP, "bfp_ftran_normal"); - lp->bfp_ftran_prepare = (BFP_lprealint *) - dlsym(lp->hBFP, "bfp_ftran_prepare"); - lp->bfp_btran_normal = (BFP_lprealint *) - dlsym(lp->hBFP, "bfp_btran_normal"); - lp->bfp_status = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_status"); - lp->bfp_implicitslack = (BFPbool_lp *) - dlsym(lp->hBFP, "bfp_implicitslack"); - lp->bfp_indexbase = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_indexbase"); - lp->bfp_rowoffset = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_rowoffset"); - lp->bfp_pivotmax = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_pivotmax"); - lp->bfp_init = (BFPbool_lpintintchar *) - dlsym(lp->hBFP, "bfp_init"); - lp->bfp_pivotalloc = (BFPbool_lpint *) - dlsym(lp->hBFP, "bfp_pivotalloc"); - lp->bfp_colcount = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_colcount"); - lp->bfp_canresetbasis = (BFPbool_lp *) - dlsym(lp->hBFP, "bfp_canresetbasis"); - lp->bfp_finishfactorization = (BFP_lp *) - dlsym(lp->hBFP, "bfp_finishfactorization"); - lp->bfp_updaterefactstats = (BFP_lp *) - dlsym(lp->hBFP, "bfp_updaterefactstats"); - lp->bfp_prepareupdate = (BFPlreal_lpintintreal *) - dlsym(lp->hBFP, "bfp_prepareupdate"); - lp->bfp_pivotRHS = (BFPreal_lplrealreal *) - dlsym(lp->hBFP, "bfp_pivotRHS"); - lp->bfp_btran_double = (BFP_lprealintrealint *) - dlsym(lp->hBFP, "bfp_btran_double"); - lp->bfp_efficiency = (BFPreal_lp *) - dlsym(lp->hBFP, "bfp_efficiency"); - lp->bfp_pivotvector = (BFPrealp_lp *) - dlsym(lp->hBFP, "bfp_pivotvector"); - lp->bfp_pivotcount = (BFPint_lp *) - dlsym(lp->hBFP, "bfp_pivotcount"); - lp->bfp_refactcount = (BFPint_lpint *) - dlsym(lp->hBFP, "bfp_refactcount"); - lp->bfp_isSetI = (BFPbool_lp *) - dlsym(lp->hBFP, "bfp_isSetI"); - lp->bfp_findredundant = (BFPint_lpintrealcbintint *) - dlsym(lp->hBFP, "bfp_findredundant"); - } - else - result = LIB_VERINVALID; - } - #endif - else - result = LIB_NOTFOUND; -#endif - /* Do validation */ - if((result != LIB_LOADED) || - ((lp->bfp_name == NULL) || - (lp->bfp_compatible == NULL) || - (lp->bfp_free == NULL) || - (lp->bfp_resize == NULL) || - (lp->bfp_nonzeros == NULL) || - (lp->bfp_memallocated == NULL) || - (lp->bfp_restart == NULL) || - (lp->bfp_mustrefactorize == NULL) || - (lp->bfp_preparefactorization == NULL) || - (lp->bfp_factorize == NULL) || - (lp->bfp_finishupdate == NULL) || - (lp->bfp_ftran_normal == NULL) || - (lp->bfp_ftran_prepare == NULL) || - (lp->bfp_btran_normal == NULL) || - (lp->bfp_status == NULL) || - (lp->bfp_implicitslack == NULL) || - (lp->bfp_indexbase == NULL) || - (lp->bfp_rowoffset == NULL) || - (lp->bfp_pivotmax == NULL) || - (lp->bfp_init == NULL) || - (lp->bfp_pivotalloc == NULL) || - (lp->bfp_colcount == NULL) || - (lp->bfp_canresetbasis == NULL) || - (lp->bfp_finishfactorization == NULL) || - (lp->bfp_updaterefactstats == NULL) || - (lp->bfp_prepareupdate == NULL) || - (lp->bfp_pivotRHS == NULL) || - (lp->bfp_btran_double == NULL) || - (lp->bfp_efficiency == NULL) || - (lp->bfp_pivotvector == NULL) || - (lp->bfp_pivotcount == NULL) || - (lp->bfp_refactcount == NULL) || - (lp->bfp_isSetI == NULL) || - (lp->bfp_findredundant == NULL) - )) { - set_BFP(lp, NULL); - if(result == LIB_LOADED) - result = LIB_NOFUNCTION; - } - } - if(filename != NULL) { - char info[LIB_STR_MAXLEN+1]; - switch(result) { - case LIB_NOTFOUND: strcpy(info, LIB_STR_NOTFOUND); - break; - case LIB_NOINFO: strcpy(info, LIB_STR_NOINFO); - break; - case LIB_NOFUNCTION: strcpy(info, LIB_STR_NOFUNCTION); - break; - case LIB_VERINVALID: strcpy(info, LIB_STR_VERINVALID); - break; - default: strcpy(info, LIB_STR_LOADED); - } - report(lp, IMPORTANT, "set_BFP: %s '%s'\n", - info, filename); - } - return( (MYBOOL) (result == LIB_LOADED)); -} - - -/* External language interface routines */ -/* DON'T MODIFY */ -lprec * __WINAPI read_XLI(char *xliname, char *modelname, char *dataname, char *options, int verbose) -{ - lprec *lp; - - lp = make_lp(0, 0); - if(lp != NULL) { - lp->source_is_file = TRUE; - lp->verbose = verbose; - if(!set_XLI(lp, xliname)) { - free_lp(&lp); - printf("read_XLI: No valid XLI package selected or available.\n"); - } - else { - if(!lp->xli_readmodel(lp, modelname, (dataname != NULL) && (*dataname != 0) ? dataname : NULL, options, verbose)) - free_lp(&lp); - } - } - return( lp ); -} - -MYBOOL __WINAPI write_XLI(lprec *lp, char *filename, char *options, MYBOOL results) -{ - return( has_XLI(lp) && mat_validate(lp->matA) && lp->xli_writemodel(lp, filename, options, results) ); -} - -MYBOOL __WINAPI has_XLI(lprec *lp) -{ - return( is_nativeXLI(lp) -#if LoadLanguageLib == TRUE - || (MYBOOL) (lp->hXLI != NULL) -#endif - ); -} - -MYBOOL __WINAPI is_nativeXLI(lprec *lp) -{ -#ifdef ExcludeNativeLanguage - return( FALSE ); -#elif LoadLanguageLib == TRUE - return( (MYBOOL) (lp->hXLI == NULL) ); -#else - return( TRUE ); -#endif -} - -MYBOOL __WINAPI set_XLI(lprec *lp, char *filename) -/* (Re)mapping of external language interface variant methods is done here */ -{ - int result = LIB_LOADED; - -#if LoadLanguageLib == TRUE - if(lp->hXLI != NULL) { - #ifdef WIN32 - FreeLibrary(lp->hXLI); - #else - dlclose(lp->hXLI); - #endif - lp->hXLI = NULL; - } -#endif - - if(filename == NULL) { - if(!is_nativeXLI(lp)) - return( FALSE ); -#ifndef ExcludeNativeLanguage - lp->xli_name = xli_name; - lp->xli_compatible = xli_compatible; - lp->xli_readmodel = xli_readmodel; - lp->xli_writemodel = xli_writemodel; -#endif - } - else { -#if LoadLanguageLib == TRUE - #ifdef WIN32 - /* Get a handle to the Windows DLL module. */ - lp->hXLI = LoadLibrary(filename); - - /* If the handle is valid, try to get the function addresses. */ - if(lp->hXLI != NULL) { - lp->xli_compatible = (XLIbool_lpintintint *) - GetProcAddress(lp->hXLI, "xli_compatible"); - if(lp->xli_compatible == NULL) - result = LIB_NOINFO; - else if(lp->xli_compatible(lp, XLIVERSION, MAJORVERSION, sizeof(REAL))) { - - lp->xli_name = (XLIchar *) - GetProcAddress(lp->hXLI, "xli_name"); - lp->xli_readmodel = (XLIbool_lpcharcharcharint *) - GetProcAddress(lp->hXLI, "xli_readmodel"); - lp->xli_writemodel = (XLIbool_lpcharcharbool *) - GetProcAddress(lp->hXLI, "xli_writemodel"); - } - else - result = LIB_VERINVALID; - } - #else - /* First standardize UNIX .SO library name format. */ - char xliname[260], *ptr; - - strcpy(xliname, filename); - if((ptr = strrchr(filename, '/')) == NULL) - ptr = filename; - else - ptr++; - xliname[(int) (ptr - filename)] = 0; - if(strncmp(ptr, "lib", 3)) - strcat(xliname, "lib"); - strcat(xliname, ptr); - if(strcmp(xliname + strlen(xliname) - 3, ".so")) - strcat(xliname, ".so"); - - /* Get a handle to the module. */ - lp->hXLI = dlopen(xliname, RTLD_LAZY); - - /* If the handle is valid, try to get the function addresses. */ - if(lp->hXLI != NULL) { - lp->xli_compatible = (XLIbool_lpintintint *) - dlsym(lp->hXLI, "xli_compatible"); - if(lp->xli_compatible == NULL) - result = LIB_NOINFO; - else if(lp->xli_compatible(lp, XLIVERSION, MAJORVERSION, sizeof(REAL))) { - - lp->xli_name = (XLIchar *) - dlsym(lp->hXLI, "xli_name"); - lp->xli_readmodel = (XLIbool_lpcharcharcharint *) - dlsym(lp->hXLI, "xli_readmodel"); - lp->xli_writemodel = (XLIbool_lpcharcharbool *) - dlsym(lp->hXLI, "xli_writemodel"); - } - else - result = LIB_VERINVALID; - } - #endif - else - result = LIB_NOTFOUND; -#endif - /* Do validation */ - if((result != LIB_LOADED) || - ((lp->xli_name == NULL) || - (lp->xli_compatible == NULL) || - (lp->xli_readmodel == NULL) || - (lp->xli_writemodel == NULL) - )) { - set_XLI(lp, NULL); - if(result == LIB_LOADED) - result = LIB_NOFUNCTION; - } - } - if(filename != NULL) { - char info[LIB_STR_MAXLEN+1]; - switch(result) { - case LIB_NOTFOUND: strcpy(info, LIB_STR_NOTFOUND); - break; - case LIB_NOINFO: strcpy(info, LIB_STR_NOINFO); - break; - case LIB_NOFUNCTION: strcpy(info, LIB_STR_NOFUNCTION); - break; - case LIB_VERINVALID: strcpy(info, LIB_STR_VERINVALID); - break; - default: strcpy(info, LIB_STR_LOADED); - } - report(lp, IMPORTANT, "set_XLI: %s '%s'\n", - info, filename); - } - return( (MYBOOL) (result == LIB_LOADED)); -} - - -STATIC int get_basisOF(lprec *lp, int coltarget[], REAL crow[], int colno[]) -/* Fill vector of basic OF values or subtract incoming values from these. - This function is called twice during reduced cost updates when the basis - does not contain the basic OF vector as the top row. The colno[] array - is filled with the count of non-zero values and the index to those. */ -{ - int i, n = lp->rows, nz = 0; - REAL *obj = lp->obj; - register REAL epsvalue = lp->epsvalue; - - /* Compute offset over the specified objective indeces (step 2) */ - if(coltarget != NULL) { - register int ix, m = coltarget[0]; - register REAL value; - - for(i = 1, coltarget++; i <= m; i++, coltarget++) { - ix = *coltarget; - /* Finalize the computation of the reduced costs, based on the format that - duals are computed as negatives, ref description for step 1 above */ - value = crow[ix]; - if(ix > n) - value += obj[ix - n]; -/* if(value != 0) { */ - if(fabs(value) > epsvalue) { - nz++; - if(colno != NULL) - colno[nz] = ix; - } - else - value = 0.0; - crow[ix] = value; - } - } - - /* Get the basic objective function values (step 1) */ - else { - register int *basvar = lp->var_basic; - - for(i = 1, crow++, basvar++; i <= n; - i++, crow++, basvar++) { - /* Load the objective value of the active basic variable; note that we - change the sign of the value to maintain computational compatibility with - the calculation of duals using in-basis storage of the basic OF values */ - if(*basvar <= n) - *crow = 0; - else - *crow = -obj[(*basvar) - n]; - if((*crow) != 0) { -/* if(fabs(*crow) > epsvalue) { */ - nz++; - if(colno != NULL) - colno[nz] = i; - } - } - } - if(colno != NULL) - colno[0] = nz; - return( nz ); -} - -int __WINAPI get_basiscolumn(lprec *lp, int j, int rn[], double bj[]) -/* This routine returns sparse vectors for all basis - columns, including the OF dummy (index 0) and slack columns. - NOTE that the index usage is nonstandard for lp_solve, since - the array offset is 1, not 0. */ -{ - int k = lp->bfp_rowoffset(lp), - matbase = lp->bfp_indexbase(lp); - - /* Do target index adjustment (etaPFI with matbase==0 is special case) */ - if(matbase > 0) - matbase += k - 1; - - /* Convert index of slack and user columns */ - j -= k; - if((j > 0) && !lp->bfp_isSetI(lp)) - j = lp->var_basic[j]; - - /* Process OF dummy and slack columns (always at lower bound) */ - if(j <= lp->rows) { - rn[1] = j + matbase; - bj[1] = 1.0; - k = 1; - } - /* Process user columns (negated if at lower bound) */ - else { - k = obtain_column(lp, j, bj, rn, NULL); - if(matbase != 0) - for(j = 1; j <= k; j++) - rn[j] += matbase; - } - - return( k ); -} - -MYBOOL __WINAPI get_primal_solution(lprec *lp, REAL *pv) -{ - if(lp->spx_status == OPTIMAL) - ; - else if(!lp->basis_valid) { - report(lp, CRITICAL, "get_primal_solution: Not a valid basis"); - return(FALSE); - } - - MEMCOPY(pv, lp->best_solution, lp->sum + 1); - return(TRUE); -} - -MYBOOL __WINAPI get_ptr_primal_solution(lprec *lp, REAL **pv) -{ - *pv = lp->best_solution; - return(TRUE); -} - -MYBOOL __WINAPI get_dual_solution(lprec *lp, REAL *rc) -{ - REAL *duals; - MYBOOL ret; - - if(!lp->basis_valid) { - report(lp, CRITICAL, "get_dual_solution: Not a valid basis"); - return(FALSE); - } - - ret = get_ptr_sensitivity_rhs(lp, &duals, NULL, NULL); - - if(ret) - MEMCOPY(rc, duals - 1, lp->sum + 1); - return(ret); -} - -MYBOOL __WINAPI get_ptr_dual_solution(lprec *lp, REAL **rc) -{ - MYBOOL ret = lp->basis_valid; - - /* Just return availability of dual information if rc is NULL */ - if(rc == NULL) - return( ret && ((MIP_count(lp) == 0) || (lp->bb_totalnodes > 0)) ); - - if(!ret) { - report(lp, CRITICAL, "get_ptr_dual_solution: Not a valid basis"); - return(ret); - } - - /* Otherwise, get the pointer to the dual information (and optionally produce it) */ - ret = get_ptr_sensitivity_rhs(lp, rc, NULL, NULL); - if(ret) - (*rc)--; - - return(ret); -} - -MYBOOL __WINAPI get_lambda(lprec *lp, REAL *lambda) -{ - if(!lp->basis_valid || (get_Lrows(lp) == 0)) { - report(lp, CRITICAL, "get_lambda: Not a valid basis"); - return(FALSE); - } - - MEMCOPY(lambda, lp->lambda+1, get_Lrows(lp)); - return(TRUE); -} - -MYBOOL __WINAPI get_ptr_lambda(lprec *lp, REAL **lambda) -{ - *lambda = lp->lambda; - return(TRUE); -} - -int __WINAPI get_orig_index(lprec *lp, int lp_index) -{ - if(lp->varmap_locked) - return(lp->presolve_undo->var_to_orig[lp_index]); - else if(lp_index <= lp->presolve_undo->orig_rows) - return(lp_index); - else - return(lp_index-lp->presolve_undo->orig_rows); -} -int __WINAPI get_lp_index(lprec *lp, int orig_index) -{ - if(lp->varmap_locked) - return(lp->presolve_undo->orig_to_var[orig_index]); - else if(orig_index <= lp->presolve_undo->orig_rows) - return(orig_index); - else - return(orig_index-lp->presolve_undo->orig_rows); -} - -MYBOOL __WINAPI is_feasible(lprec *lp, REAL *values, REAL threshold) -/* Recommend to use threshold = lp->epspivot */ -{ - int i, j, elmnr, ie; - REAL *this_rhs, dist; - REAL *value; - int *rownr; - MATrec *mat = lp->matA; - - for(i = lp->rows + 1; i <= lp->sum; i++) { - if(values[i - lp->rows] < unscaled_value(lp, lp->orig_lowbo[i], i) - || values[i - lp->rows] > unscaled_value(lp, lp->orig_upbo[i], i)) { - if(!((lp->sc_lobound[i - lp->rows]>0) && (values[i - lp->rows]==0))) - return(FALSE); - } - } - - this_rhs = (REAL *) mempool_obtainVector(lp->workarrays, lp->rows+1, sizeof(*this_rhs)); -/* allocREAL(lp, &this_rhs, lp->rows + 1, TRUE); */ - for(j = 1; j <= lp->columns; j++) { - elmnr = mat->col_end[j - 1]; - ie = mat->col_end[j]; - rownr = &COL_MAT_ROWNR(elmnr); - value = &COL_MAT_VALUE(elmnr); - for(; elmnr < ie; elmnr++, rownr += matRowColStep, value += matValueStep) { - this_rhs[*rownr] += unscaled_mat(lp, *value, *rownr, j); - } - } - for(i = 1; i <= lp->rows; i++) { - dist = lp->orig_rhs[i] - this_rhs[i]; - my_roundzero(dist, threshold); - if((lp->orig_upbo[i] == 0 && dist != 0) ||( dist < 0)) { - FREE(this_rhs); - return(FALSE); - } - } - mempool_releaseVector(lp->workarrays, (char *) this_rhs, FALSE); -/* FREE(this_rhs); */ - return(TRUE); -} - -int __WINAPI column_in_lp(lprec *lp, REAL *testcolumn) -{ - int i, j, je, colnr = 0; - int nz, ident = 1; - MATrec *mat = lp->matA; - int *matRownr; - REAL value, *matValue; - - for(nz = 0, i = 1; i <= lp->rows; i++) - if(fabs(testcolumn[i]) > lp->epsvalue) nz++; - - for(i = 1; (i <= lp->columns) && (ident); i++) { - ident = nz; - value = fabs(get_mat(lp, 0, i)-testcolumn[0]); - if(value > lp->epsvalue) - continue; - j = mat->col_end[i - 1]; - je = mat->col_end[i]; - matRownr = &COL_MAT_ROWNR(j); - matValue = &COL_MAT_VALUE(j); - for(; (j < je) && (ident >= 0); - j++, ident--, matRownr += matRowColStep, matValue += matValueStep) { - value = *matValue; - if(is_chsign(lp, *matRownr)) - value = my_flipsign(value); - value = unscaled_mat(lp, value, *matRownr, i); - value -= testcolumn[*matRownr]; - if(fabs(value) > lp->epsvalue) - break; - } - if(ident == 0) - colnr = i; - } - return( colnr ); -} - -MYBOOL __WINAPI set_lp_name(lprec *lp, char *name) -{ - if (name == NULL) { - FREE(lp->lp_name); - lp->lp_name = NULL; - } - else { - allocCHAR(lp, &lp->lp_name, (int) (strlen(name) + 1), AUTOMATIC); - strcpy(lp->lp_name, name); - } - return(TRUE); -} - -char * __WINAPI get_lp_name(lprec *lp) -{ - return((lp->lp_name != NULL) ? lp->lp_name : (char *) ""); -} - -STATIC MYBOOL init_rowcol_names(lprec *lp) -{ - if(!lp->names_used) { - lp->row_name = (hashelem **) calloc(lp->rows_alloc + 1, sizeof(*lp->row_name)); - lp->col_name = (hashelem **) calloc(lp->columns_alloc + 1, sizeof(*lp->col_name)); - lp->rowname_hashtab = create_hash_table(lp->rows_alloc + 1, 0); - lp->colname_hashtab = create_hash_table(lp->columns_alloc + 1, 1); - lp->names_used = TRUE; - } - return(TRUE); -} - -MYBOOL rename_var(lprec *lp, int varindex, char *new_name, hashelem **list, hashtable **ht) -{ - hashelem *hp; - MYBOOL newitem; - - hp = list[varindex]; - newitem = (MYBOOL) (hp == NULL); - if(newitem) - hp = puthash(new_name, varindex, list, *ht); - else if((strlen(hp->name) != strlen(new_name)) || - (strcmp(hp->name, new_name) != 0)) { - hashtable *newht, *oldht; - - allocCHAR(lp, &hp->name, (int) (strlen(new_name) + 1), AUTOMATIC); - strcpy(hp->name, new_name); - oldht = *ht; - newht = copy_hash_table(oldht, list, oldht->size); - *ht = newht; - free_hash_table(oldht); - } - return(newitem); -} - -MYBOOL __WINAPI is_use_names(lprec *lp, MYBOOL isrow) -{ - if(isrow) - return( lp->use_row_names ); - else - return( lp->use_col_names ); -} - -void __WINAPI set_use_names(lprec *lp, MYBOOL isrow, MYBOOL use_names) -{ - if(isrow) - lp->use_row_names = use_names; - else - lp->use_col_names = use_names; -} - -int __WINAPI get_nameindex(lprec *lp, char *varname, MYBOOL isrow) -{ - if(isrow) - return( find_row(lp, varname, FALSE) ); - else - return( find_var(lp, varname, FALSE) ); -} - -MYBOOL __WINAPI set_row_name(lprec *lp, int rownr, char *new_name) -{ - if((rownr < 0) || (rownr > lp->rows+1)) { - report(lp, IMPORTANT, "set_row_name: Row %d out of range", rownr); - return(FALSE); - } - - /* Prepare for a new row */ - if((rownr > lp->rows) && !append_rows(lp, rownr-lp->rows)) - return( FALSE ); - if(!lp->names_used) { - if(!init_rowcol_names(lp)) - return(FALSE); - } - rename_var(lp, rownr, new_name, lp->row_name, &lp->rowname_hashtab); - - return(TRUE); -} - -char * __WINAPI get_row_name(lprec *lp, int rownr) -{ - if((rownr < 0) || (rownr > lp->rows+1)) { - report(lp, IMPORTANT, "get_row_name: Row %d out of range", rownr); - return(NULL); - } - - if((lp->presolve_undo->var_to_orig != NULL) && lp->wasPresolved) { - if(lp->presolve_undo->var_to_orig[rownr] == 0) - rownr = -rownr; - else - rownr = lp->presolve_undo->var_to_orig[rownr]; - } - return( get_origrow_name(lp, rownr) ); -} - -char * __WINAPI get_origrow_name(lprec *lp, int rownr) -{ - MYBOOL newrow; - char *ptr; - - newrow = (MYBOOL) (rownr < 0); - rownr = abs(rownr); -#ifdef Paranoia - if(((lp->presolve_undo->var_to_orig == NULL) && newrow) || - (rownr > MAX(lp->rows, lp->presolve_undo->orig_rows))) { - report(lp, IMPORTANT, "get_origrow_name: Row %d out of range", rownr); - return(NULL); - } -#endif - - if(lp->names_used && lp->use_row_names && (lp->row_name[rownr] != NULL) && - (lp->row_name[rownr]->name != NULL)) { -#ifdef Paranoia - if(lp->row_name[rownr]->index != rownr) - report(lp, SEVERE, "get_origrow_name: Inconsistent row ordinal %d vs %d\n", - rownr, lp->row_name[rownr]->index); -#endif - ptr = lp->row_name[rownr]->name; - } - else { - if(lp->rowcol_name == NULL) - if (!allocCHAR(lp, &lp->rowcol_name, 20, FALSE)) - return(NULL); - ptr = lp->rowcol_name; - if(newrow) - sprintf(ptr, ROWNAMEMASK2, rownr); - else - sprintf(ptr, ROWNAMEMASK, rownr); - } - return(ptr); -} - -MYBOOL __WINAPI set_col_name(lprec *lp, int colnr, char *new_name) -{ - if((colnr > lp->columns+1) || (colnr < 1)) { - report(lp, IMPORTANT, "set_col_name: Column %d out of range", colnr); - } - - if((colnr > lp->columns) && !append_columns(lp, colnr-lp->columns)) - return(FALSE); - - if(!lp->names_used) - init_rowcol_names(lp); - rename_var(lp, colnr, new_name, lp->col_name, &lp->colname_hashtab); - - return(TRUE); -} - -char * __WINAPI get_col_name(lprec *lp, int colnr) -{ - if((colnr > lp->columns+1) || (colnr < 1)) { - report(lp, IMPORTANT, "get_col_name: Column %d out of range", colnr); - return(NULL); - } - - if((lp->presolve_undo->var_to_orig != NULL) && lp->wasPresolved) { - if(lp->presolve_undo->var_to_orig[lp->rows + colnr] == 0) - colnr = -colnr; - else - colnr = lp->presolve_undo->var_to_orig[lp->rows + colnr]; - } - return( get_origcol_name(lp, colnr) ); -} - -char * __WINAPI get_origcol_name(lprec *lp, int colnr) -{ - MYBOOL newcol; - char *ptr; - - newcol = (MYBOOL) (colnr < 0); - colnr = abs(colnr); -#ifdef Paranoia - if(((lp->presolve_undo->var_to_orig == NULL) && newcol) || - (colnr > MAX(lp->columns, lp->presolve_undo->orig_columns))) { - report(lp, IMPORTANT, "get_origcol_name: Column %d out of range", colnr); - return(NULL); - } -#endif - - if(lp->names_used && lp->use_col_names && (lp->col_name[colnr] != NULL) && (lp->col_name[colnr]->name != NULL)) { -#ifdef Paranoia - if(lp->col_name[colnr]->index != colnr) - report(lp, SEVERE, "get_origcol_name: Inconsistent column ordinal %d vs %d\n", - colnr, lp->col_name[colnr]->index); -#endif - ptr = lp->col_name[colnr]->name; - } - else { - if(lp->rowcol_name == NULL) - if (!allocCHAR(lp, &lp->rowcol_name, 20, FALSE)) - return(NULL); - ptr = lp->rowcol_name; - if(newcol) - sprintf(ptr, COLNAMEMASK2, colnr); - else - sprintf(ptr, COLNAMEMASK, colnr); - } - return(ptr); -} - -STATIC int MIP_count(lprec *lp) -{ - return( lp->int_vars+lp->sc_vars+SOS_count(lp) ); -} -STATIC int bin_count(lprec *lp, MYBOOL working) -{ - int i, n = 0; - if(working) { - for(i = lp->rows+1; i <= lp->sum; i++) - if(fabs(unscaled_value(lp, lp->upbo[i], i) - 1) < lp->epsvalue) - n++; - } - else { - for(i = 1; i <= lp->columns; i++) - if((fabs(get_upbo(lp, i) - 1) < lp->epsvalue) && - (fabs(get_lowbo(lp, i) - 0) < lp->epsvalue)) - n++; - } - return( n ); -} -STATIC int SOS_count(lprec *lp) -{ - if(lp->SOS == NULL) - return( 0 ); - else - return( lp->SOS->sos_count ); -} -STATIC int GUB_count(lprec *lp) -{ - if(lp->GUB == NULL) - return( 0 ); - else - return( lp->GUB->sos_count ); -} - -STATIC REAL compute_violation(lprec *lp, int row_nr) -/* Returns the bound violation of a given basic variable; the return - value is negative if it is below is lower bound, it is positive - if it is greater than the upper bound, and zero otherwise. */ -{ - REAL value, test; - - value = lp->rhs[row_nr]; - row_nr = lp->var_basic[row_nr]; - test = value - my_lowbound(lp->lowbo[row_nr]); - my_roundzero(test, lp->epsprimal); - if(test > 0) { - test = value - lp->upbo[row_nr]; - my_roundzero(test, lp->epsprimal); - if(test < 0) - test = 0; - } - return( test ); -} - -STATIC REAL feasibilityOffset(lprec *lp, MYBOOL isdual) -{ - int i, j; - REAL f, Extra; - - Extra = 0; - if(isdual) { - /* This section computes a OF offset to ensure that the dual phase 1 is - feasible. It is used to compute a primal feasible base that can be - passed to the primal simplex in phase 2. */ -#if 0 - - /* This is the legacy (v3.2-) P1extraVal logic that sets Extra to be the - smallest negative reduced cost. Note that the reduced costs are the - values of the dual slacks, which are [0..Inf> for feasibility. - If we have negative reduced costs for bounded non-basic variables, we - can simply switch the bound to obtain feasibility and possibly avoid - having to set Extra. */ - if(!isDualFeasible(lp, lp->epsprimal, NULL, NULL, &f) - Extra = f; - -#else - /* Find the most negative of the objective coefficients. We will subtract this - value from every element of the objective row, making it non-negative and - the problem therefore dual feasible. */ - for(i = 1; i <= lp->columns; i++) { - f = lp->orig_obj[i]; - if(f < Extra) - Extra = f; - } -#endif - } - - else { - /* Set Extra to be the index of the most negative of the net RHS coefficients; - this approach can be used in the primal phase 1 followed by the dual phase 2 - and when there are no ranged constraints. When there are ranged constraints, - additional artificial variables must be introduced. */ - Extra = 0; - j = 0; - Extra = lp->infinite; - for(i = 1; i <= lp->rows; i++) { - f = lp->rhs[i]; - if(f < Extra) { - Extra = f; - j = i; - } - } - Extra = j; - } - - return(Extra); - -} - -STATIC REAL compute_dualslacks(lprec *lp, int target, REAL **dvalues, int **nzdvalues, MYBOOL dosum) -/* Note that this function is similar to the compute_reducedcosts function in lp_price.c */ -{ - int i, varnr, - *coltarget, **nzduals, *nzvtemp = NULL; - REAL d, g = 0, **duals, *vtemp = NULL; - MYBOOL localREAL = (MYBOOL) (dvalues == NULL), - localINT = (MYBOOL) (nzdvalues == NULL); - - if(is_action(lp->spx_action, ACTION_REBASE) || - is_action(lp->spx_action, ACTION_REINVERT) || !lp->basis_valid) - return( g ); - - /* Initialize */ - if(!localREAL) { - duals = dvalues; - nzduals = nzdvalues; - } - else { - duals = &vtemp; - nzduals = &nzvtemp; - } - if(localINT || (*nzduals == NULL)) - allocINT(lp, nzduals, lp->columns + 1, AUTOMATIC); - if(localREAL || (*duals == NULL)) - allocREAL(lp, duals, lp->sum + 1, AUTOMATIC); - if(target == 0) - target = SCAN_ALLVARS+ USE_NONBASICVARS; - - /* Define variable target list and compute the reduced costs */ - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->columns+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, target, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return(FALSE); - } - bsolve(lp, 0, *duals, NULL, lp->epsmachine*DOUBLEROUND, 1.0); - prod_xA(lp, coltarget, *duals, NULL, lp->epsmachine, 1.0, - *duals, *nzduals, MAT_ROUNDDEFAULT | MAT_ROUNDRC); - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - - /* Compute sum or maximum infeasibility as specified */ - for(i = 1; i <= (*nzduals)[0]; i++) { - varnr = (*nzduals)[i]; - d = my_chsign(!lp->is_lower[varnr], (*duals)[varnr]); - if(d < 0) { - if(dosum) - g += -d; /* Compute sum as a positive number */ - else { - SETMIN(g, d); /* Compute gap as a negative number */ - } - } - } - - /* Clean up */ - if(localREAL) - FREE(*duals); - if(localINT) - FREE(*nzduals); - - return( g ); -} - -STATIC REAL compute_feasibilitygap(lprec *lp, MYBOOL isdual, MYBOOL dosum) -{ - REAL f = 0; - - /* This computes the primal feasibility gap (for use with the dual simplex phase 1) */ - if(isdual) { - int i; - REAL g; - - for(i = 1; i <= lp->rows; i++) { - if(lp->rhs[i] < 0) - g = lp->rhs[i]; - else if(lp->rhs[i] > lp->upbo[lp->var_basic[i]]) - g = lp->rhs[i] - lp->upbo[lp->var_basic[i]]; - else - g = 0; - if(dosum) - f += g; - else { - SETMAX(f, g); - } - } - } - /* This computes the dual feasibility gap (for use with the primal simplex phase 1) */ - else - f = compute_dualslacks(lp, SCAN_USERVARS+USE_ALLVARS, NULL, NULL, dosum); - - return( f ); -} - -/* Find the smallest fractional value in a given row of the OF/constraint matrix */ -STATIC int row_decimals(lprec *lp, int rownr, MYBOOL intsonly, REAL *intscalar) -{ - int basi, i, j, ncols = lp->columns; - REAL f, /* g, */ epsvalue = lp->epsprimal; - - basi = 0; - for(j = 1; j <= ncols; j++) { - if(intsonly && !is_int(lp, j)) { - if(intsonly == TRUE) - break; - else - continue; - } - f = fabs(get_mat(lp, rownr, j)); - /* f = fmod(f, 1); */ - f -= floor (f + epsvalue); -/* - if(f <= epsvalue) - continue; - g = f; -*/ - for(i = 0; (i <= MAX_FRACSCALE) && (/* g */ f > epsvalue); i++) { - f *= 10; - /* g = fmod(f, 1); */ - f -= floor (f + epsvalue); - } - if(i > MAX_FRACSCALE) - /* i = MAX_FRACSCALE */ break; - SETMAX(basi, i); - } - if(j > ncols) - *intscalar = pow(10.0, basi); - else { - basi = -1; - *intscalar = 1; - } - return( basi ); -} - -STATIC int row_intstats(lprec *lp, int rownr, int pivcolnr, int *maxndec, - int *plucount, int *intcount, int *intval, REAL *valGCD, REAL *pivcolval) -{ - int jb, je, jj, nn = 0, multA, multB, intGCD = 0; - REAL rowval, inthold, intfrac; - MATrec *mat = lp->matA; - - /* Do we have a valid matrix? */ - if(mat_validate(mat)) { - - /* Get smallest fractional row value */ - *maxndec = row_decimals(lp, rownr, AUTOMATIC, &intfrac); - - /* Get OF row starting and ending positions, as well as the first column index */ - if(rownr == 0) { - jb = 1; - je = lp->columns+1; - } - else { - jb = mat->row_end[rownr-1]; - je = mat->row_end[rownr]; - } - nn = je - jb; - *pivcolval = 1.0; - *plucount = 0; - *intcount = 0; - *intval = 0; - for(; jb < je; jb++) { - - if(rownr == 0) { - if(lp->orig_obj[jb] == 0) { - nn--; - continue; - } - jj = jb; - } - else - jj = ROW_MAT_COLNR(jb); - - /* Pick up the value of the pivot column and continue */ - if(jj == pivcolnr) { - if(rownr == 0) - *pivcolval = unscaled_mat(lp, lp->orig_obj[jb], 0, jb); - else - *pivcolval = get_mat_byindex(lp, jb, TRUE, FALSE); - continue; - } - if(!is_int(lp, jj)) - continue; - - /* Update the count of integer columns */ - (*intcount)++; - - /* Update the count of positive parameter values */ - if(rownr == 0) - rowval = unscaled_mat(lp, lp->orig_obj[jb], 0, jb); - else - rowval = get_mat_byindex(lp, jb, TRUE, FALSE); - if(rowval > 0) - (*plucount)++; - - /* Check if the parameter value is integer and update the row's GCD */ - rowval = fabs(rowval) * intfrac; - rowval += rowval*lp->epsmachine; - rowval = modf(rowval, &inthold); - if(rowval < lp->epsprimal) { - (*intval)++; - if(*intval == 1) - intGCD = (int) inthold; - else - intGCD = gcd(intGCD, (LLONG) inthold, &multA, &multB); - } - } - *valGCD = intGCD; - *valGCD /= intfrac; - } - - return(nn); -} - -#if 0 -REAL MIP_stepOF(lprec *lp) -/* This function tries to find a non-zero minimum improvement - if the OF contains all integer variables (logic only applies if we are - looking for a single solution, not possibly several equal-valued ones). -*/ -{ - MYBOOL OFgcd; - int colnr, rownr, n, ib, ie, maxndec, - pluscount, intcount, intval; - REAL value, valOF, divOF, valGCD; - MATrec *mat = lp->matA; - - value = 0; - if((lp->int_vars > 0) && (lp->solutionlimit == 1) && mat_validate(mat)) { - - /* Get statistics for integer OF variables and compute base stepsize */ - n = row_intstats(lp, 0, -1, &maxndec, &pluscount, &intcount, &intval, &valGCD, &divOF); - if((n == 0) || (maxndec < 0)) - return( value ); - OFgcd = (MYBOOL) (intval > 0); - if(OFgcd) - value = valGCD; - - /* Check non-ints in the OF to see if we can get more info */ - if(n - intcount > 0) { - int nrv = 0; - - /* See if we have equality constraints */ - ie = lp->rows; - for(ib = 1; ib <= ie; ib++) { - if(is_constr_type(lp, ib, EQ)) - break; - } - - /* If so, there may be a chance to find an improved stepsize */ - if(ib < ie) - for(colnr = 1; colnr <= lp->columns; colnr++) { - - /* Go directly to the next variable if this is an integer or - there is no row candidate to explore for hidden bounds for - real-valued variables (limit scan to one row!) */ - if(is_int(lp, colnr)) - continue; - nrv++; - /* Scan equality constraints */ - ib = mat->col_end[colnr-1]; - ie = mat->col_end[colnr]; - while(ib < ie) { - if(is_constr_type(lp, (rownr = COL_MAT_ROWNR(ib)), EQ)) { - - /* Get "child" row statistics, but break out if we don't - find enough information, i.e. no integers with coefficients of proper type */ - n = row_intstats(lp, rownr, colnr, &maxndec, &pluscount, &intcount, &intval, &valGCD, &divOF); - if((intval < n - 1) || (maxndec < 0)) { - value = 0; - break; - } - - /* We can update */ - valOF = unscaled_mat(lp, lp->orig_obj[colnr], 0, colnr); - valOF = fabs( valOF * (valGCD / divOF) ); - if(OFgcd) { - SETMIN(value, valOF); - } - else { - OFgcd = TRUE; - value = valOF; - } - } - ib++; - } - - /* No point in continuing scan if we failed in current column */ - if(value == 0) - break; - } - - /* Check if we found information for any real-valued variable; - if not, then we must set the iprovement delta to 0 */ - if(nrv == 0) - value = 0; - } - } - return( value ); -} -#elif 0 -/* - original v5.5 implementation giving problems with some models - - ex: - - min: +r1 +r2; - - R1: +r1 +r2 >= 1; - R2: +r1 -5.345 b1 = 0; - R3: +r2 -4.456 b2 = 0; - - b1 <= 1; - b2 <= 1; - - //b2>0.1; - - int b1,b2; - -*/ -REAL MIP_stepOF(lprec *lp) -/* This function tries to find a non-zero minimum improvement - if the OF contains all integer variables (logic only applies if we are - looking for a single solution, not possibly several equal-valued ones). */ -{ - MYBOOL OFgcd; - int colnr, rownr, n, ib, ie, - pluscount, intcount; - int intval, maxndec; - REAL value = 0, valOF, divOF, valGCD; - MATrec *mat = lp->matA; - - if((lp->int_vars > 0) && (lp->solutionlimit == 1) && mat_validate(mat)) { - - /* Get statistics for integer OF variables and compute base stepsize */ - n = row_intstats(lp, 0, 0, &maxndec, &pluscount, &intcount, &intval, &valGCD, &divOF); - if((n == 0) || (maxndec < 0)) - return( value ); - OFgcd = (MYBOOL) (intval > 0); - if(OFgcd) - value = valGCD; - - /* Check non-ints in the OF to see if we can get more info */ - if(n - intcount > 0) { - int nrv = n - intcount; /* Number of real variables in the objective */ - int niv = 0; /* Number of real variables identified as integer */ - int nrows = lp->rows; - - /* See if we have equality constraints */ - for(ib = 1; ib <= nrows; ib++) { - if(is_constr_type(lp, ib, EQ)) - break; - } - - /* If so, there may be a chance to find an improved stepsize */ - if(ib <= nrows) - for(colnr = 1; colnr <= lp->columns; colnr++) { - - /* Go directly to the next variable if this is an integer or - there is no row candidate to explore for hidden bounds for - real-valued variables (limit scan to one row/no recursion) */ - if((lp->orig_obj[colnr] == 0) || is_int(lp, colnr)) - continue; - - /* Scan equality constraints */ - ib = mat->col_end[colnr-1]; - ie = mat->col_end[colnr]; - while(ib < ie) { - if(is_constr_type(lp, (rownr = COL_MAT_ROWNR(ib)), EQ)) { - - /* Get "child" row statistics, but break out if we don't - find enough information, i.e. no integers with coefficients of proper type */ - n = row_intstats(lp, rownr, colnr, &maxndec, &pluscount, &intcount, &intval, &valGCD, &divOF); - if((intval < n - 1) || (maxndec < 0)) { - value = 0; - break; - } - niv++; - - /* We can update */ - valOF = unscaled_mat(lp, lp->orig_obj[colnr], 0, colnr); - valOF = fabs( valOF * (valGCD / divOF) ); - if(OFgcd) { - SETMIN(value, valOF); - } - else { - OFgcd = TRUE; - value = valOF; - } - } - ib++; - } - - /* No point in continuing scan if we failed in current column */ - if(value == 0) - break; - } - - /* Check if we found information for any real-valued variable; - if not, then we must set the improvement delta to 0 */ - if(nrv > niv) - value = 0; - } - } - return( value ); -} -#else - -STATIC REAL row_plusdelta(lprec *lp, int rownr, int excludecol, int *intcount, int *realcount) -{ - MATrec *mat = lp->matA; - int j, jb, je, jj, bincount, - n = 0, nrows = lp->rows; - REAL rowval, deltaOF = 0, - *obj_orig = lp->orig_obj, *obj_sort = NULL; - - *realcount = 0; - *intcount = 0; - bincount = 0; - - /* Get OF row starting and ending positions, as well as the first column index */ - if(rownr == 0) { - jb = 1; - je = lp->columns+1; - } - else { - jb = mat->row_end[rownr-1]; - je = mat->row_end[rownr]; - } - - /* Fill the array */ - for(j = jb; j < je; j++) { - - if(rownr == 0) { - if(obj_orig[j] == 0) - continue; - jj = j; - } - else - jj = ROW_MAT_COLNR(j); - - /* Check for exclusion column */ - if(jj == excludecol) - continue; - - /* Check that the variable is integer */ - if(is_int(lp, jj)) { - rowval = lp->orig_upbo[nrows + jj]; - if((rowval < lp->infinite) && (fabs(unscaled_value(lp, rowval - lp->orig_lowbo[nrows + jj], nrows + jj) - 1) < lp->epsint)) // difference between upper and lower = 1 is ok - bincount++; - if(rownr == 0) - rowval = unscaled_mat(lp, obj_orig[jj], 0, jj); - else - rowval = get_mat_byindex(lp, j, TRUE, FALSE); - - /* Allocate array of coefficients to be sorted */ - if(n == 0) - allocREAL(lp, &obj_sort, je-jb, FALSE); - - obj_sort[n++] = rowval; - } - else - (*realcount)++; - - } - (*intcount) = n; - - if(*realcount == 0) { - if (n == 0 || bincount < n) - deltaOF = 0; - else if(n == 1) - deltaOF = obj_sort[0]; - else { - - REAL newval; - MYBOOL loops = 0; - - while(n > 0) { - - /* Sort the coefficients in ascending order */ - qsortex(obj_sort, n, 0, sizeof(*obj_sort), FALSE, compareREAL, NULL, 0); - - /* Eliminate array duplicates (could consider applying an eps) */ - j = 0; jb = 1; - do { - rowval = obj_sort[j]; - while((jb < n) && (obj_sort[jb] == rowval)) jb++; - if((jb < n) && (++j < jb)) - obj_sort[j] = obj_sort[jb]; - } while(++jb < n); - n = j+1; - - /* Get the reference minimum stepsize on the first iteration */ - if(loops == 0) { - /* Spool to the coefficient closest to zero, which is the reference OF stepsize */ - for(j = 0; (j < n) && (obj_sort[j] < 0); j++); - - /* Case 1: All negative coefficients */ - if(j >= n) - deltaOF = -obj_sort[n-1]; - /* Case2: All positive coefficients */ - else if(j == 0) - deltaOF = obj_sort[j]; - /* Case 3: Both negative and positive coefficients */ - else - deltaOF = MIN(-obj_sort[j-1], obj_sort[j]); - } - - /* Adjust the reference minimum stepsize on next iterations */ - loops++; - - /* Loop over non-zero coefficient differences to - obtain minimum change, i.e. if one increases */ - newval = lp->infinite; - for(j = 1; j < n; j++) { - rowval = obj_sort[j]-obj_sort[j-1]; - SETMIN(newval, rowval); - obj_sort[j-1] = rowval; - } - n--; - SETMIN(deltaOF, newval); - } - } - } - - /* Dispose of the work array */ - FREE(obj_sort); - - return( deltaOF ); -} - -/* v6.0 implementation converted to v5.5 */ - -STATIC REAL MIP_stepOF(lprec *lp) -/* This function tries to find a non-zero minimum improvement - if the OF contains all integer variables (logic only applies if we are - looking for a single solution, not possibly several equal-valued ones). */ -{ - REAL OFdelta = 0; - MATrec *mat = lp->matA; - - if((lp->int_vars > 0) && (lp->solutionlimit == 1) && mat_validate(mat)) { - - int colnr, ib, ie, - intcount, realcount; - - /* Get statistics for integer OF variables and compute base stepsize */ - OFdelta = row_plusdelta(lp, 0, 0, &intcount, &realcount); -/* return( value ); */ - - /* Check non-ints in the OF to see if we can get more info */ - if(realcount > 0) { - int niv = 0; /* Number of real variables identified as integer */ - int nrows = lp->rows; - REAL rowdelta; - - OFdelta = lp->infinite; - for(colnr = 1; (colnr <= lp->columns) && (niv < realcount); colnr++) { - - /* Go directly to the next variable if this is an integer or - there is no row candidate to explore for hidden bounds for - real-valued variables (limit scan to one row/no recursion) */ - if((lp->orig_obj[colnr] == 0) || is_int(lp, colnr)) - continue; - - /* Scan equality constraints */ - ib = mat->col_end[colnr-1]; - ie = mat->col_end[colnr]; - while(ib < ie) { - - /* Get "child" row statistics, but break out if we don't find enough - information, i.e. no integers with coefficients of proper type. */ - rowdelta = row_plusdelta(lp, COL_MAT_ROWNR(ib), colnr, &intcount, &realcount); - if(realcount > 0) { - OFdelta = 0; - break; - } - - /* We can update */ - SETMIN(OFdelta, rowdelta); - ib++; - } - - /* No point in continuing scan if we failed in the current column */ - if(OFdelta == 0) - break; - - /* We found an implied integer, update count */ - niv++; - } - - /* Check if we found information for any real-valued variable; - if not, then we must set the improvement delta to 0 */ - if(realcount > niv) - OFdelta = 0; - } - } - return( OFdelta ); -} - -#endif - -STATIC MYBOOL isPrimalSimplex(lprec *lp) -{ - return((MYBOOL) (((lp->simplex_mode & SIMPLEX_Phase1_PRIMAL) != 0) || - ((lp->simplex_mode & SIMPLEX_Phase2_PRIMAL) != 0))); -} - -STATIC MYBOOL isPhase1(lprec *lp) -{ - return((MYBOOL) (((lp->simplex_mode & SIMPLEX_Phase1_PRIMAL) != 0) || - ((lp->simplex_mode & SIMPLEX_Phase1_DUAL) != 0))); -} - -STATIC MYBOOL isP1extra(lprec *lp) -{ - return((MYBOOL) ((lp->P1extraDim > 0) || (lp->P1extraVal != 0))); -} - -STATIC MYBOOL feasiblePhase1(lprec *lp, REAL epsvalue) -{ - REAL gap; - MYBOOL test; - - gap = fabs(lp->rhs[0] - lp->orig_rhs[0]); - test = (MYBOOL) (gap < epsvalue); - return( test) ; -} - -STATIC MYBOOL isDegenerateBasis(lprec *lp, int basisvar) -{ - int varindex; - - varindex = lp->var_basic[basisvar]; - if((fabs(lp->rhs[basisvar]) < lp->epsprimal) || - (fabs(lp->upbo[varindex]-lp->rhs[basisvar]) < lp->epsprimal)) - return( TRUE ); - else - return( FALSE ); -} - -STATIC int findBasicFixedvar(lprec *lp, int afternr, MYBOOL slacksonly) -{ - int varnr, delta = 1; - - if(afternr < 0) { - delta = -1; - afternr = -afternr; - } - afternr += delta; - if((afternr < 1) || (afternr > lp->rows)) - return( 0 ); - - for(; (afternr > 0) && (afternr <= lp->rows); afternr += delta) { - varnr = lp->var_basic[afternr]; - if(((varnr <= lp->rows) && is_constr_type(lp, varnr, EQ)) || - (!slacksonly && (varnr > lp->rows) && is_fixedvar(lp, varnr))) - break; - } - - if(afternr > lp->rows) - afternr = 0; - - return( afternr ); -} - -STATIC MYBOOL isBasisVarFeasible(lprec *lp, REAL tol, int basis_row) -{ - int col; - REAL x; - MYBOOL Ok = TRUE; - MYBOOL doSC = FALSE; - - col = lp->var_basic[basis_row]; - x = lp->rhs[basis_row]; /* The current solution of basic variables stored here! */ - if((x < -tol) || (x > lp->upbo[col]+tol)) - Ok = FALSE; - else if(doSC && (col > lp->rows) && (fabs(lp->sc_lobound[col - lp->rows]) > 0)) { - if((x > tol) && (x < fabs(lp->sc_lobound[col - lp->rows])-tol)) - Ok = FALSE; - } - return( Ok ); -} -STATIC MYBOOL isPrimalFeasible(lprec *lp, REAL tol, int infeasibles[], REAL *feasibilitygap) -{ - int i; - MYBOOL feasible = TRUE; - - /* This is a short-hand call to rowdual() to check for primal infeasibility */ - -#if 0 - /* Traditional indexing style */ - for(i = 1; i <= lp->rows; i++) { - feasible = isBasisVarFeasible(lp, tol, i); -#else - /* Fast array pointer style */ - LREAL *rhsptr; - int *idxptr; - - if(infeasibles != NULL) - infeasibles[0] = 0; - for(i = 1, rhsptr = lp->rhs+1, idxptr = lp->var_basic+1; - (i <= lp->rows); i++, rhsptr++, idxptr++) { - feasible = TRUE; -/* if(((*rhsptr) < lp->lowbo[*idxptr]-tol) || ((*rhsptr) > lp->upbo[*idxptr]+tol)) */ - if(((*rhsptr) < -tol) || ((*rhsptr) > lp->upbo[*idxptr]+tol)) - feasible = FALSE; -#endif - if(!feasible) { - if(infeasibles == NULL) - break; - infeasibles[0]++; - infeasibles[infeasibles[0]] = i; - } - } - - /* Compute feasibility gap (could actually do this calculation above) */ - if(feasibilitygap != NULL) { - if(feasible) - *feasibilitygap = 0.0; - else - *feasibilitygap = feasibilityOffset(lp, FALSE); - } - - return(feasible); -} - -STATIC MYBOOL isDualFeasible(lprec *lp, REAL tol, int *boundflipcount, int infeasibles[], REAL *feasibilitygap) -{ - int i, varnr, - n = 0, /* Number of infeasible duals corrected with bound-swaps */ - m = 0, - target = SCAN_ALLVARS+USE_NONBASICVARS; - REAL f = 0; - MYBOOL feasible, islower; - - - /* The reduced costs are the values of the dual slacks, which - are [0..Inf> for feasibility. If we have negative reduced costs - for bounded non-basic variables, we can simply switch the bound - of bounded variables to obtain dual feasibility and possibly avoid - having to use dual simplex phase 1. */ - if((infeasibles != NULL) || (boundflipcount != NULL)) { - int *nzdcol = NULL; - REAL d, *dcol = NULL; - - f = compute_dualslacks(lp, target, &dcol, &nzdcol, FALSE); - if(nzdcol != NULL) - for(i = 1; i <= nzdcol[0]; i++) { - varnr = nzdcol[i]; - islower = lp->is_lower[varnr]; - d = my_chsign(!islower, dcol[varnr]); - - /* Don't bother with uninteresting non-basic variables */ - if((d > -tol) || /* Positive reduced costs with a tolerance */ - my_unbounded(lp, varnr) || /* Free variables cannot change bound */ - is_fixedvar(lp, varnr)) /* Equality slack or a fixed variable ("type 3") */ - continue; - - /* Check if we have non-flippable bounds, i.e. an unbounded variable - (types 2+4), or bounded variables (type 3), and if the counter is NULL. */ - if( (boundflipcount == NULL) || - ((lp->bb_level <= 1) && (my_rangebo(lp, varnr) > fabs(lp->negrange))) || - (islower && my_infinite(lp, lp->upbo[varnr])) || - (!islower && my_infinite(lp, my_lowbo(lp, varnr))) ) { - m++; - if(infeasibles != NULL) - infeasibles[m] = varnr; - } - /* Only do bound flips if the user-provided counter is non-NULL */ - else { - lp->is_lower[varnr] = !islower; - n++; - } - } - if(infeasibles != NULL) - infeasibles[0] = m; - FREE(dcol); - FREE(nzdcol); - if(n > 0) { - set_action(&lp->spx_action, ACTION_RECOMPUTE); - if(m == 0) - f = 0; - } - } - else - f = compute_dualslacks(lp, target, NULL, NULL, FALSE); -/* f = feasibilityOffset(lp, TRUE); */ /* Safe legacy mode */ - - /* Do an extra scan to see if there are bounded variables in the OF not present in any constraint; - Most typically, presolve fixes such cases, so this is rarely encountered. */ - - varnr = lp->rows + 1; - for(i = 1; i <= lp->columns; i++, varnr++) { - if (mat_collength(lp->matA, i) == 0) { - islower = lp->is_lower[varnr]; - if((my_chsign(islower, lp->orig_obj[i]) > 0) && !SOS_is_member(lp->SOS, 0, i)) { - lp->is_lower[varnr] = !islower; - if((islower && my_infinite(lp, lp->upbo[varnr] /* lp->orig_upbo[varnr] */)) || - (!islower && my_infinite(lp, my_lowbo(lp, varnr) /* lp->orig_lowbo[varnr] */))) { - lp->spx_status = UNBOUNDED; - break; - } - /* lp->is_lower[varnr] = !islower; */ - n++; - } - } - } - - /* Return status */ - - if(boundflipcount != NULL) - *boundflipcount = n; - if(feasibilitygap != NULL) { - my_roundzero(f, tol); - *feasibilitygap = f; - } - feasible = (MYBOOL) ((f == 0) && (m == 0)); - - return(feasible); -} - -void __WINAPI default_basis(lprec *lp) -{ - int i; - - /* Set the slack variables to be basic; note that the is_basic[] array - is a helper array filled in presolve() to match var_basic[]. */ - for(i = 1; i <= lp->rows; i++) { - lp->var_basic[i] = i; - lp->is_basic[i] = TRUE; - lp->is_lower[i] = TRUE; - } - lp->var_basic[0] = TRUE; /* Set to signal that this is the default basis */ - - /* Set user variables at their lower bound, including the - dummy slack for the objective "constraint" */ - for(; i <= lp->sum; i++) { - lp->is_basic[i] = FALSE; - lp->is_lower[i] = TRUE; - } - lp->is_lower[0] = TRUE; - - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - lp->basis_valid = TRUE; /* Do not re-initialize basis on entering Solve */ -} - -int __WINAPI get_basiscrash(lprec *lp) -{ - return(lp->crashmode); -} - -void __WINAPI set_basiscrash(lprec *lp, int mode) -{ - lp->crashmode = mode; -} - -MYBOOL __WINAPI set_basis(lprec *lp, int *bascolumn, MYBOOL nonbasic) /* Added by KE */ -{ - int i,s,k,n; - - /* Make sure we are consistent */ - if(lp->wasPresolved && ((lp->rows != lp->presolve_undo->orig_rows) || - (lp->columns != lp->presolve_undo->orig_columns))) - return( FALSE ); - - /* Initialize (lp->is_basic is set in preprocess); Note that as of v5 and before - it is an lp_solve convention that basic variables are at their lower bounds! - This routine provides for the a possible future case that basic variables - can be upper-bounded. */ - lp->is_lower[0] = TRUE; - for(i = 1; i <= lp->sum; i++) { - lp->is_lower[i] = TRUE; - lp->is_basic[i] = FALSE; - } - for(i = 1; i <= lp->rows; i++) - lp->var_basic[i] = FALSE; - - /* Set basic and optionally non-basic variables; - negative index means at lower bound, positive at upper bound */ - if(nonbasic) - n = lp->sum; - else - n = lp->rows; - for(i = 1; i <= n; i++) { - s = bascolumn[i]; - k = abs(s); - if(k <= 0 || k > lp->sum) - return( FALSE ); - if(i <= lp->rows) { - lp->var_basic[i] = k; - lp->is_basic[k] = TRUE; - } - else /* Remove this test if basic variables can be upper-bounded */ - if(s > 0) - lp->is_lower[k] = FALSE; - } - if(!verify_basis(lp)) - return( FALSE ); - - /* Invalidate basis */ - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - lp->basis_valid = TRUE; /* Do not re-initialize basis on entering Solve */ - lp->var_basic[0] = FALSE; /* Set to signal that this is a non-default basis */ - - return( TRUE ); -} - -void __WINAPI reset_basis(lprec *lp) -{ - lp->basis_valid = FALSE; /* Causes reinversion at next opportunity */ -} - -MYBOOL __WINAPI get_basis(lprec *lp, int *bascolumn, MYBOOL nonbasic) -{ - int k, i; - - if(!lp->basis_valid || - (lp->rows != lp->presolve_undo->orig_rows) || - (lp->columns != lp->presolve_undo->orig_columns)) - return( FALSE ); - - *bascolumn = 0; - - /* First save basic variable indexes */ - for(i = 1; i <= lp->rows; i++) { - k = lp->var_basic[i]; - bascolumn[i] = my_chsign(lp->is_lower[k], k); - } - - /* Then optionally save non-basic variable indeces */ - if(nonbasic) { - for(k = 1; (k <= lp->sum) && (i <= lp->sum); k++) { - if(lp->is_basic[k]) - continue; - bascolumn[i] = my_chsign(lp->is_lower[k], k); - i++; - } - } - return( TRUE ); -} - -STATIC MYBOOL is_BasisReady(lprec *lp) -{ - return( (MYBOOL) (lp->var_basic[0] != AUTOMATIC) ); -} - -STATIC MYBOOL is_slackbasis(lprec *lp) -{ - int n = 0, err = 0; - if(lp->basis_valid) { - int i, k; - MYBOOL *used = NULL; - - allocMYBOOL(lp, &used, lp->rows+1, TRUE); - for(i = 1; i <= lp->rows; i++) { - k = lp->var_basic[i]; - if(k <= lp->rows) { - if(used[k]) - err++; - else - used[k] = TRUE; - n++; - } - } - FREE(used); - if(err > 0) - report(lp, SEVERE, "is_slackbasis: %d inconsistencies found in slack basis\n", err); - } - return( (MYBOOL) (n == lp->rows) ); -} - -STATIC MYBOOL verify_basis(lprec *lp) -{ - int i, ii, k = 0; - MYBOOL result = FALSE; - - for(i = 1; i <= lp->rows; i++) { - ii = lp->var_basic[i]; - if((ii < 1) || (ii > lp->sum) || !lp->is_basic[ii]) { - k = i; - ii = 0; - goto Done; - } - } - - ii = lp->rows; - for(i = 1; i <= lp->sum; i++) { - if(lp->is_basic[i]) - ii--; - } - result = (MYBOOL) (ii == 0); - -Done: -#if 0 /* For testing */ - if(!result) - ii = 0; -#endif - return(result); -} - -int __WINAPI set_basisvar(lprec *lp, int basisPos, int enteringCol) -{ - int leavingCol; - - leavingCol = lp->var_basic[basisPos]; - -#ifdef Paranoia - if((basisPos < 1) || (basisPos > lp->rows)) - report(lp, SEVERE, "set_basisvar: Invalid leaving basis position %d specified at iter %.0f\n", - basisPos, (double) get_total_iter(lp)); - if((leavingCol < 1) || (leavingCol > lp->sum)) - report(lp, SEVERE, "set_basisvar: Invalid leaving column %d referenced at iter %.0f\n", - leavingCol, (double) get_total_iter(lp)); - if((enteringCol < 1) || (enteringCol > lp->sum)) - report(lp, SEVERE, "set_basisvar: Invalid entering column %d specified at iter %.0f\n", - enteringCol, (double) get_total_iter(lp)); -#endif - -#ifdef ParanoiaXY - if(!lp->is_basic[leavingCol]) - report(lp, IMPORTANT, "set_basisvar: Leaving variable %d is not basic at iter %.0f\n", - leavingCol, (double) get_total_iter(lp)); - if(enteringCol > lp->rows && lp->is_basic[enteringCol]) - report(lp, IMPORTANT, "set_basisvar: Entering variable %d is already basic at iter %.0f\n", - enteringCol, (double) get_total_iter(lp)); -#endif - - lp->var_basic[0] = FALSE; /* Set to signal that this is a non-default basis */ - lp->var_basic[basisPos] = enteringCol; - lp->is_basic[leavingCol] = FALSE; - lp->is_basic[enteringCol] = TRUE; - if(lp->bb_basis != NULL) - lp->bb_basis->pivots++; - - return(leavingCol); -} - -/* Bounds updating and unloading routines; requires that the - current values for upbo and lowbo are in the original base. */ -STATIC int perturb_bounds(lprec *lp, BBrec *perturbed, MYBOOL doRows, MYBOOL doCols, MYBOOL includeFIXED) -{ - int i, ii, n = 0; - REAL new_lb, new_ub, *upbo, *lowbo; - - if(perturbed == NULL) - return( n ); - - /* Map reference bounds to previous state, i.e. cumulate - perturbations in case of persistent problems */ - upbo = perturbed->upbo; - lowbo = perturbed->lowbo; - - /* Set appropriate target variable range */ - i = 1; - ii = lp->rows; - if(!doRows) - i += ii; - if(!doCols) - ii = lp->sum; - - /* Perturb (expand) finite variable bounds randomly */ - for(; i <= ii; i++) { - - /* Don't perturb regular slack variables */ - if((i <= lp->rows) && (lowbo[i] == 0) && (upbo[i] >= lp->infinite)) - continue; - - new_lb = lowbo[i]; - new_ub = upbo[i]; - - /* Don't perturb fixed variables if not specified */ - if(!includeFIXED && (new_ub == new_lb)) - continue; - - /* Lower bound for variables (consider implementing RHS here w/contentmode== AUTOMATIC) */ - if((i > lp->rows) && (new_lb < lp->infinite)) { - new_lb = rand_uniform(lp, RANDSCALE) + 1; - new_lb *= lp->epsperturb; - lowbo[i] -= new_lb; - n++; - } - - /* Upper bound */ - if(new_ub < lp->infinite) { - new_ub = rand_uniform(lp, RANDSCALE) + 1; - new_ub *= lp->epsperturb; - upbo[i] += new_ub; - n++; - } - } - - /* Make sure we start from scratch */ - set_action(&lp->spx_action, ACTION_REBASE); - - return( n ); -} - -STATIC MYBOOL impose_bounds(lprec *lp, REAL *upbo, REAL *lowbo) -/* Explicitly set working bounds to given vectors without pushing or popping */ -{ - MYBOOL ok; - - ok = (MYBOOL) ((upbo != NULL) || (lowbo != NULL)); - if(ok) { - if((upbo != NULL) && (upbo != lp->upbo)) - MEMCOPY(lp->upbo, upbo, lp->sum + 1); - if((lowbo != NULL) && (lowbo != lp->lowbo)) - MEMCOPY(lp->lowbo, lowbo, lp->sum + 1); - if(lp->bb_bounds != NULL) - lp->bb_bounds->UBzerobased = FALSE; - set_action(&lp->spx_action, ACTION_REBASE); - } - set_action(&lp->spx_action, ACTION_RECOMPUTE); - return( ok ); -} - -STATIC MYBOOL validate_bounds(lprec *lp, REAL *upbo, REAL *lowbo) -/* Check if all bounds are Explicitly set working bounds to given vectors without pushing or popping */ -{ - MYBOOL ok; - int i; - - ok = (MYBOOL) ((upbo != NULL) || (lowbo != NULL)); - if(ok) { - for(i = 1; i <= lp->sum; i++) - if((lowbo[i] > upbo[i]) || (lowbo[i] < lp->orig_lowbo[i]) || (upbo[i] > lp->orig_upbo[i])) - break; - ok = (MYBOOL) (i > lp->sum); - } - return( ok ); -} - -STATIC int unload_BB(lprec *lp) -{ - int levelsunloaded = 0; - - if(lp->bb_bounds != NULL) - while(pop_BB(lp->bb_bounds)) - levelsunloaded++; - return( levelsunloaded ); -} - - -#define LowerStorageModel 1 -#define BasisStorageModel 1 -STATIC basisrec *push_basis(lprec *lp, int *basisvar, MYBOOL *isbasic, MYBOOL *islower) -/* Save the ingoing basis and push it onto the stack */ -{ - int sum = lp->sum + 1; - basisrec *newbasis = NULL; - - newbasis = (basisrec *) calloc(sizeof(*newbasis), 1); - if((newbasis != NULL) && -#if LowerStorageModel == 0 - allocMYBOOL(lp, &newbasis->is_lower, sum, FALSE) && -#else - allocMYBOOL(lp, &newbasis->is_lower, (sum + 8) / 8, TRUE) && -#endif -#if BasisStorageModel == 0 - allocMYBOOL(lp, &newbasis->is_basic, sum, FALSE) && -#endif - allocINT(lp, &newbasis->var_basic, lp->rows + 1, FALSE)) { - - if(islower == NULL) - islower = lp->is_lower; - if(isbasic == NULL) - isbasic = lp->is_basic; - if(basisvar == NULL) - basisvar = lp->var_basic; - -#if LowerStorageModel == 0 - MEMCOPY(newbasis->is_lower, islower, sum); -#else - for(sum = 1; sum <= lp->sum; sum++) - if(islower[sum]) - set_biton(newbasis->is_lower, sum); -#endif -#if BasisStorageModel == 0 - MEMCOPY(newbasis->is_basic, isbasic, lp->sum + 1); -#endif - MEMCOPY(newbasis->var_basic, basisvar, lp->rows + 1); - - newbasis->previous = lp->bb_basis; - if(lp->bb_basis == NULL) - newbasis->level = 0; - else - newbasis->level = lp->bb_basis->level + 1; - newbasis->pivots = 0; - - lp->bb_basis = newbasis; - } - return( newbasis ); -} - -STATIC MYBOOL compare_basis(lprec *lp) -/* Compares the last pushed basis with the currently active basis */ -{ - int i, j; - MYBOOL same_basis = TRUE; - - if(lp->bb_basis == NULL) - return( FALSE ); - - /* Loop over basis variables until a mismatch (order can be different) */ - i = 1; - while(same_basis && (i <= lp->rows)) { - j = 1; - while(same_basis && (j <= lp->rows)) { - same_basis = (MYBOOL) (lp->bb_basis->var_basic[i] != lp->var_basic[j]); - j++; - } - same_basis = !same_basis; - i++; - } - /* Loop over bound status indicators until a mismatch */ - i = 1; - while(same_basis && (i <= lp->sum)) { - same_basis = (lp->bb_basis->is_lower[i] && lp->is_lower[i]); - i++; - } - - return( same_basis ); -} - -STATIC MYBOOL restore_basis(lprec *lp) -/* Restore values from the previously pushed / saved basis without popping it */ -{ - MYBOOL ok; - int i; - - ok = (MYBOOL) (lp->bb_basis != NULL); - if(ok) { - MEMCOPY(lp->var_basic, lp->bb_basis->var_basic, lp->rows + 1); -#if BasisStorageModel == 0 - MEMCOPY(lp->is_basic, lp->bb_basis->is_basic, lp->sum + 1); -#else - MEMCLEAR(lp->is_basic, lp->sum + 1); - for(i = 1; i <= lp->rows; i++) - lp->is_basic[lp->var_basic[i]] = TRUE; -#endif -#if LowerStorageModel == 0 - MEMCOPY(lp->is_lower, lp->bb_basis->is_lower, lp->sum + 1); -#else - for(i = 1; i <= lp->sum; i++) - lp->is_lower[i] = is_biton(lp->bb_basis->is_lower, i); -#endif - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT); - } - return( ok ); -} - -STATIC MYBOOL pop_basis(lprec *lp, MYBOOL restore) -/* Pop / free, and optionally restore the previously "pushed" / saved basis */ -{ - MYBOOL ok; - basisrec *oldbasis; - - ok = (MYBOOL) (lp->bb_basis != NULL); - if(ok) { - oldbasis = lp->bb_basis; - if(oldbasis != NULL) { - lp->bb_basis = oldbasis->previous; - FREE(oldbasis->var_basic); -#if BasisStorageModel == 0 - FREE(oldbasis->is_basic); -#endif - FREE(oldbasis->is_lower); - FREE(oldbasis); - } - if(restore && (lp->bb_basis != NULL)) - restore_basis(lp); - } - return( ok ); -} - -STATIC int unload_basis(lprec *lp, MYBOOL restorelast) -{ - int levelsunloaded = 0; - - if(lp->bb_basis != NULL) - while(pop_basis(lp, restorelast)) - levelsunloaded++; - return( levelsunloaded ); -} - - -STATIC REAL scaled_floor(lprec *lp, int colnr, REAL value, REAL epsscale) -{ - value = floor(value); - if(value != 0) - if(lp->columns_scaled && is_integerscaling(lp)) { - value = scaled_value(lp, value, colnr); - if(epsscale != 0) - value += epsscale*lp->epsmachine; -/* value += epsscale*lp->epsprimal; */ -/* value = restoreINT(value, lp->epsint); */ - } - return(value); -} - -STATIC REAL scaled_ceil(lprec *lp, int colnr, REAL value, REAL epsscale) -{ - value = ceil(value); - if(value != 0) - if(lp->columns_scaled && is_integerscaling(lp)) { - value = scaled_value(lp, value, colnr); - if(epsscale != 0) - value -= epsscale*lp->epsmachine; -/* value -= epsscale*lp->epsprimal; */ -/* value = restoreINT(value, lp->epsint); */ - } - return(value); -} - -/* Branch and bound variable selection functions */ - -STATIC MYBOOL is_sc_violated(lprec *lp, int column) -{ - int varno; - REAL tmpreal; - REAL eps = lp->epsvalue; /* ß adding eps here*/ - - varno = lp->rows+column; - tmpreal = unscaled_value(lp, lp->sc_lobound[column], varno); - return( (MYBOOL) ((tmpreal > 0) && /* it is an (inactive) SC variable... */ - (lp->solution[varno] < tmpreal - eps) && /* ...and the NZ lower bound is violated */ - (lp->solution[varno] > eps)) ); /* ...and the Z lowerbound is violated */ -} - -STATIC int find_sc_bbvar(lprec *lp, int *count) -{ - int i, ii, n, bestvar; - int firstsc, lastsc; - REAL hold, holdINT, bestval, OFval, randval, scval; - MYBOOL reversemode, greedymode, randomizemode, - pseudocostmode, pseudocostsel; - - bestvar = 0; - if((lp->sc_vars == 0) || (*count > 0)) - return(bestvar); - - reversemode = is_bb_mode(lp, NODE_WEIGHTREVERSEMODE); - greedymode = is_bb_mode(lp, NODE_GREEDYMODE); - randomizemode = is_bb_mode(lp, NODE_RANDOMIZEMODE); - pseudocostmode = is_bb_mode(lp, NODE_PSEUDOCOSTMODE); - pseudocostsel = is_bb_rule(lp, NODE_PSEUDOCOSTSELECT) || - is_bb_rule(lp, NODE_PSEUDONONINTSELECT) || - is_bb_rule(lp, NODE_PSEUDORATIOSELECT); - - bestvar = 0; - bestval = -lp->infinite; - hold = 0; - randval = 1; - firstsc = 0; - lastsc = lp->columns; - - for(n = 1; n <= lp->columns; n++) { - ii = get_var_priority(lp, n); - i = lp->rows + ii; - if(!lp->bb_varactive[ii] && is_sc_violated(lp, ii) && !SOS_is_marked(lp->SOS, 0, ii)) { - - /* Do tallies */ - (*count)++; - lastsc = i; - if(firstsc <= 0) - firstsc = i; - scval = get_pseudorange(lp->bb_PseudoCost, ii, BB_SC); - - /* Select default pricing/weighting mode */ - if(pseudocostmode) - OFval = get_pseudonodecost(lp->bb_PseudoCost, ii, BB_SC, lp->solution[i]); - else - OFval = my_chsign(is_maxim(lp), get_mat(lp, 0, ii)); - - if(randomizemode) - randval = exp(rand_uniform(lp, 1.0)); - - /* Find the maximum pseudo-cost of a variable (don't apply pseudocostmode here) */ - if(pseudocostsel) { - if(pseudocostmode) - hold = OFval; - else - hold = get_pseudonodecost(lp->bb_PseudoCost, ii, BB_SC, lp->solution[i]); - hold *= randval; - if(greedymode) { - if(pseudocostmode) /* Override! */ - OFval = my_chsign(is_maxim(lp), get_mat(lp, 0, ii)); - hold *= OFval; - } - hold = my_chsign(reversemode, hold); - } - else - /* Find the variable with the largest sc gap (closest to the sc mean) */ - if(is_bb_rule(lp, NODE_FRACTIONSELECT)) { - hold = modf(lp->solution[i]/scval, &holdINT); - holdINT = hold-1; - if(fabs(holdINT) > hold) - hold = holdINT; - if(greedymode) - hold *= OFval; - hold = my_chsign(reversemode, hold)*scval*randval; - } - else - /* Do first or last violated sc index selection (default) */ - /* if(is_bb_rule(lp, NODE_FIRSTSELECT)) */ - { - if(reversemode) - continue; - else { - bestvar = i; - break; - } - } - - /* Select better, check for ties, and split by proximity to 0.5*sc_lobound */ - if(hold > bestval) { - if( (bestvar == 0) || - (hold > bestval+lp->epsprimal) || - (fabs(modf(lp->solution[i]/scval, &holdINT) - 0.5) < - fabs(modf(lp->solution[bestvar]/ - get_pseudorange(lp->bb_PseudoCost, bestvar-lp->rows, BB_SC), &holdINT) - 0.5)) ) { - bestval = hold; - bestvar = i; - } - } - } - } - - if(is_bb_rule(lp, NODE_FIRSTSELECT) && reversemode) - bestvar = lastsc; - - return(bestvar); -} - -STATIC int find_sos_bbvar(lprec *lp, int *count, MYBOOL intsos) -{ - int k, i, j, var; - - var = 0; - if((lp->SOS == NULL) || (*count > 0)) - return(var); - - /* Check if the SOS'es happen to already be satisified */ - i = SOS_is_satisfied(lp->SOS, 0, lp->solution); - if((i == SOS_COMPLETE) || (i == SOS_INCOMPLETE)) - return(-1); - - /* Otherwise identify a SOS variable to enter B&B */ - for(k = 0; k < lp->sos_vars; k++) { - i = lp->sos_priority[k]; -#ifdef Paranoia - if((i < 1) || (i > lp->columns)) - report(lp, SEVERE, "find_sos_bbvar: Invalid SOS variable map %d at %d\n", - i, k); -#endif - j = lp->rows + i; - if(!SOS_is_marked(lp->SOS, 0, i) && !SOS_is_full(lp->SOS, 0, i, FALSE)) { -/* if(!SOS_is_marked(lp->SOS, 0, i) && !SOS_is_full(lp->SOS, 0, i, TRUE)) { */ - if(!intsos || is_int(lp, i)) { - (*count)++; - if(var == 0) { - var = j; - break; - } - } - } - } -#ifdef Paranoia - if((var > 0) && !SOS_is_member(lp->SOS, 0, var-lp->rows)) - report(lp, SEVERE, "find_sos_bbvar: Found variable %d, which is not a SOS!\n", var); -#endif - return(var); -} - -STATIC int find_int_bbvar(lprec *lp, int *count, BBrec *BB, MYBOOL *isfeasible) -{ - int i, ii, n, k, bestvar, depthmax, *nonint = NULL; - REAL hold, holdINT, bestval, OFval, randval, - *lowbo = BB->lowbo, *upbo = BB->upbo; - MYBOOL reversemode, greedymode, depthfirstmode, breadthfirstmode, - randomizemode, rcostmode, - pseudocostmode, pseudocostsel, pseudostrong, isINT, valINT; - - if((lp->int_vars == 0) || (*count > 0)) - return( 0 ); - if(lp->bb_usenode != NULL) { - i = lp->bb_usenode(lp, lp->bb_nodehandle, BB_INT); - if(i >= 0) { - if(i > 0) - (*count)++; - return( i ); - } - } - - reversemode = is_bb_mode(lp, NODE_WEIGHTREVERSEMODE); - greedymode = is_bb_mode(lp, NODE_GREEDYMODE); - randomizemode = is_bb_mode(lp, NODE_RANDOMIZEMODE); - depthfirstmode = is_bb_mode(lp, NODE_DEPTHFIRSTMODE); - breadthfirstmode = is_bb_mode(lp, NODE_BREADTHFIRSTMODE) && - (MYBOOL) (lp->bb_level <= lp->int_vars); - rcostmode = (MYBOOL) /* FALSE */ (BB->lp->solutioncount > 0) && is_bb_mode(lp, NODE_RCOSTFIXING) ; /* 5/2/08 peno disabled NODE_RCOSTFIXING because it results in non-optimal solutions with some models */ /* 15/2/8 peno enabled NODE_RCOSTFIXING again because a fix is found. See lp_simplex.c NODE__RCOSTFIXING fix */ - pseudocostmode = is_bb_mode(lp, NODE_PSEUDOCOSTMODE); - pseudocostsel = is_bb_rule(lp, NODE_PSEUDOCOSTSELECT) || - is_bb_rule(lp, NODE_PSEUDONONINTSELECT) || - is_bb_rule(lp, NODE_PSEUDORATIOSELECT); - pseudostrong = FALSE && - pseudocostsel && !rcostmode && is_bb_mode(lp, NODE_STRONGINIT); - - /* Fill list of non-ints */ - allocINT(lp, &nonint, lp->columns + 1, FALSE); - n = 0; - depthmax = -1; - if(isfeasible != NULL) - *isfeasible = TRUE; - BB->lastrcf = 0; - for(k = 1; (k <= lp->columns); k++) { - ii = get_var_priority(lp, k); - isINT = is_int(lp,ii); - i = lp->rows + ii; - - /* Tally reduced cost fixing opportunities for ranged non-basic nonINTs */ - if(!isINT) { -#ifdef UseMilpExpandedRCF - if(rcostmode) { - bestvar = rcfbound_BB(BB, i, isINT, NULL, isfeasible); - if(bestvar != FR) - BB->lastrcf++; - } -#endif - } - else { - - valINT = solution_is_int(lp, i, FALSE); - - /* Skip already fixed variables */ - if(lowbo[i] == upbo[i]) { - - /* Check for validity */ -#ifdef Paranoia - if(!valINT) { - report(lp, IMPORTANT, - "find_int_bbvar: INT var %d was fixed at %d, but computed as %g at node %.0f\n", - ii, (int) lowbo[i], lp->solution[i], (double) lp->bb_totalnodes); - lp->bb_break = TRUE; - lp->spx_status = UNKNOWNERROR; - bestvar = 0; - goto Done; - } -#endif - } - - /* The variable has not yet been fixed */ - else { - - /* Tally reduced cost fixing opportunities (also when the - variables are integer-valued at the current relaxation) */ - if(rcostmode) { - bestvar = rcfbound_BB(BB, i, isINT, NULL, isfeasible); - if(bestvar != FR) - BB->lastrcf++; - } - else - bestvar = FR; - - /* Only qualify variable as branching node if it is non-integer and - it will not be subsequently fixed via reduced cost fixing logic */ - if(!valINT && (bestvar >= FR)) { - - n++; - nonint[n] = ii; - SETMAX(depthmax, lp->bb_varactive[ii]); - } - } - - } - } - -#ifdef UseMilpSlacksRCF - /* Optionally also tally slacks */ - if(rcostmode) { - for(i = 1; (i <= lp->rows) && (BB->lastrcf == 0); i++) { - /* Skip already fixed slacks (equalities) */ - if(lowbo[i] < upbo[i]) { - bestvar = rcfbound_BB(BB, i, FALSE, NULL, isfeasible); - if(bestvar != FR) - BB->lastrcf++; - } - } - } -#endif - nonint[0] = n; - *count = n; - bestvar = 0; - if(n == 0) /* No non-integers found */ - goto Done; - - bestval = -lp->infinite; - hold = 0; - randval = 1; - - /* Sort non-ints by depth in case we have breadthfirst or depthfirst modes */ - if((lp->bb_level > 1) && (depthmax > 0) && (depthfirstmode || breadthfirstmode)) { - int *depths = NULL; - - /* Fill attribute array and make sure ordinal order breaks ties during sort */ - allocINT(lp, &depths, n + 1, FALSE); - for(i = 1; i <= n; i++) - depths[i] = (depthfirstmode ? n+1-i : i) + (n+1)*lp->bb_varactive[nonint[i]]; - hpsortex(depths, n, 1, sizeof(*nonint), depthfirstmode, compareINT, nonint); - FREE(depths); - } - - /* Do simple firstselect handling */ - if(is_bb_rule(lp, NODE_FIRSTSELECT)) { - if(reversemode) - bestvar = lp->rows + nonint[nonint[0]]; - else - bestvar = lp->rows + nonint[1]; - } - - else for(n = 1; n <= nonint[0]; n++) { - ii = nonint[n]; - i = lp->rows + ii; - - /* Do the naive detection */ - if(n == 1) - bestvar = i; - - /* Should we do a "strong" pseudo-cost initialization or an incremental update? */ - if(pseudostrong && - (MAX(lp->bb_PseudoCost->LOcost[ii].rownr, - lp->bb_PseudoCost->UPcost[ii].rownr) < lp->bb_PseudoCost->updatelimit) && - (MAX(lp->bb_PseudoCost->LOcost[ii].colnr, - lp->bb_PseudoCost->UPcost[ii].colnr) < 5*lp->bb_PseudoCost->updatelimit)) { - strongbranch_BB(lp, BB, ii, BB_INT, nonint[0]); - } - - /* Select default pricing/weighting mode */ - if(pseudocostmode) - OFval = get_pseudonodecost(lp->bb_PseudoCost, ii, BB_INT, lp->solution[i]); - else - OFval = my_chsign(is_maxim(lp), get_mat(lp, 0, ii)); - - if(randomizemode) - randval = exp(rand_uniform(lp, 1.0)); - - /* Find the maximum pseudo-cost of a variable (don't apply pseudocostmode here) */ - if(pseudocostsel) { - if(pseudocostmode) - hold = OFval; - else - hold = get_pseudonodecost(lp->bb_PseudoCost, ii, BB_INT, lp->solution[i]); - hold *= randval; - if(greedymode) { - if(pseudocostmode) /* Override! */ - OFval = my_chsign(is_maxim(lp), get_mat(lp, 0, ii)); - hold *= OFval; - } - hold = my_chsign(reversemode, hold); - } - else - /* Find the variable with the largest gap to its bounds (distance from being fixed) */ - if(is_bb_rule(lp, NODE_GAPSELECT)) { - hold = lp->solution[i]; - holdINT = hold-unscaled_value(lp, upbo[i], i); - hold -= unscaled_value(lp, lowbo[i], i); - if(fabs(holdINT) > hold) - hold = holdINT; - if(greedymode) - hold *= OFval; - hold = my_chsign(reversemode, hold)*randval; - } - else - /* Find the variable with the largest integer gap (closest to 0.5) */ - if(is_bb_rule(lp, NODE_FRACTIONSELECT)) { - hold = modf(lp->solution[i], &holdINT); - holdINT = hold-1; - if(fabs(holdINT) > hold) - hold = holdINT; - if(greedymode) - hold *= OFval; - hold = my_chsign(reversemode, hold)*randval; - } - else - /* Find the "range", most flexible variable */ - if(is_bb_rule(lp, NODE_RANGESELECT)) { - hold = unscaled_value(lp, upbo[i]-lowbo[i], i); - if(greedymode) - hold *= OFval; - hold = my_chsign(reversemode, hold)*randval; - } - - /* Select better, check for ties, and split by proximity to 0.5 */ - if(hold > bestval) { - if( (hold > bestval+lp->epsprimal) || - (fabs(modf(lp->solution[i], &holdINT) - 0.5) < - fabs(modf(lp->solution[bestvar], &holdINT) - 0.5)) ) { - bestval = hold; - bestvar = i; - } - } - } - -Done: - FREE(nonint); - return(bestvar); -} - -STATIC BBPSrec *init_pseudocost(lprec *lp, int pseudotype) -{ - int i; - REAL PSinitUP, PSinitLO; - BBPSrec *newitem; - MYBOOL isPSCount; - - /* Allocate memory */ - newitem = (BBPSrec*) malloc(sizeof(*newitem)); - newitem->lp = lp; - newitem->LOcost = (MATitem*) malloc((lp->columns+1) * sizeof(*newitem->LOcost)); - newitem->UPcost = (MATitem*) malloc((lp->columns+1) * sizeof(*newitem->UPcost)); - newitem->secondary = NULL; - - /* Initialize with OF values */ - newitem->pseodotype = (pseudotype & NODE_STRATEGYMASK); - isPSCount = ((pseudotype & NODE_PSEUDONONINTSELECT) != 0); - for(i = 1; i <= lp->columns; i++) { - newitem->LOcost[i].rownr = 1; /* Actual updates */ - newitem->LOcost[i].colnr = 1; /* Attempted updates */ - newitem->UPcost[i].rownr = 1; - newitem->UPcost[i].colnr = 1; - - /* Initialize with the plain OF value as conventional usage suggests, or - override in case of pseudo-nonint count strategy */ - PSinitUP = my_chsign(is_maxim(lp), get_mat(lp, 0, i)); - PSinitLO = -PSinitUP; - if(isPSCount) { - /* Set default assumed reduction in the number of non-ints by choosing this variable; - KE changed from 0 on 30 June 2004 and made two-sided selectable. Note that the - typical value range is <0..1>, with a positive bias for an "a priori" assumed - fast-converging (low "MIP-complexity") model. Very hard models may require - negative initialized values for one or both. */ - PSinitUP = 0.1*0; -#if 0 - PSinitUP = my_chsign(PSinitUP < 0, PSinitUP); - PSinitLO = -PSinitUP; -#else - PSinitLO = PSinitUP; -#endif - } - newitem->UPcost[i].value = PSinitUP; - newitem->LOcost[i].value = PSinitLO; - } - newitem->updatelimit = lp->bb_PseudoUpdates; - newitem->updatesfinished = 0; - newitem->restartlimit = DEF_PSEUDOCOSTRESTART; - - /* Let the user get an opportunity to initialize pseudocosts */ - if(userabort(lp, MSG_INITPSEUDOCOST)) - lp->spx_status = USERABORT; - - return( newitem ); -} - -STATIC MYBOOL free_pseudoclass(BBPSrec **PseudoClass) -{ - BBPSrec *target = *PseudoClass; - - FREE(target->LOcost); - FREE(target->UPcost); - target = target->secondary; - FREE(*PseudoClass); - *PseudoClass = target; - - return( (MYBOOL) (target != NULL) ); -} - -STATIC void free_pseudocost(lprec *lp) -{ - if((lp != NULL) && (lp->bb_PseudoCost != NULL)) { - while(free_pseudoclass(&(lp->bb_PseudoCost)) ); - } -} - -MYBOOL __WINAPI set_pseudocosts(lprec *lp, REAL *clower, REAL *cupper, int *updatelimit) -{ - int i; - - if((lp->bb_PseudoCost == NULL) || ((clower == NULL) && (cupper == NULL))) - return(FALSE); - for(i = 1; i <= lp->columns; i++) { - if(clower != NULL) - lp->bb_PseudoCost->LOcost[i].value = clower[i]; - if(cupper != NULL) - lp->bb_PseudoCost->UPcost[i].value = cupper[i]; - } - if(updatelimit != NULL) - lp->bb_PseudoCost->updatelimit = *updatelimit; - return(TRUE); -} - -MYBOOL __WINAPI get_pseudocosts(lprec *lp, REAL *clower, REAL *cupper, int *updatelimit) -{ - int i; - - if((lp->bb_PseudoCost == NULL) || ((clower == NULL) && (cupper == NULL))) - return(FALSE); - for(i = 1; i <= lp->columns; i++) { - if(clower != NULL) - clower[i] = lp->bb_PseudoCost->LOcost[i].value; - if(cupper != NULL) - cupper[i] = lp->bb_PseudoCost->UPcost[i].value; - } - if(updatelimit != NULL) - *updatelimit = lp->bb_PseudoCost->updatelimit; - return(TRUE); -} - -STATIC REAL get_pseudorange(BBPSrec *pc, int mipvar, int varcode) -{ - if(varcode == BB_SC) - return( unscaled_value(pc->lp, pc->lp->sc_lobound[mipvar], pc->lp->rows+mipvar) ); - else - return( 1.0 ); -} - -STATIC void update_pseudocost(BBPSrec *pc, int mipvar, int varcode, MYBOOL capupper, REAL varsol) -{ - REAL OFsol, uplim; - MATitem *PS; - MYBOOL nonIntSelect = is_bb_rule(pc->lp, NODE_PSEUDONONINTSELECT); - - /* Establish input values; - Note: The pseudocosts are normalized to the 0-1 range! */ - uplim = get_pseudorange(pc, mipvar, varcode); - varsol = modf(varsol/uplim, &OFsol); - - /* Set reference value according to pseudocost mode */ - if(nonIntSelect) - OFsol = pc->lp->bb_bounds->lastvarcus; /* The count of MIP infeasibilities */ - else - OFsol = pc->lp->solution[0]; /* The problem's objective function value */ - - if(isnan(varsol)) { - pc->lp->bb_parentOF = OFsol; - return; - } - - /* Point to the applicable (lower or upper) bound and increment attempted update count */ - if(capupper) { - PS = &pc->LOcost[mipvar]; - } - else { - PS = &pc->UPcost[mipvar]; - varsol = 1-varsol; - } - PS->colnr++; - - /* Make adjustment to divisor if we are using the ratio pseudo-cost approach */ - if(is_bb_rule(pc->lp, NODE_PSEUDORATIOSELECT)) - varsol *= capupper; - - /* Compute the update (consider weighting in favor of most recent) */ - mipvar = pc->updatelimit; - if(((mipvar <= 0) || (PS->rownr < mipvar)) && - (fabs(varsol) > pc->lp->epspivot)) { - /* We are interested in the change in the MIP measure (contribution to increase - or decrease, as the case may be) and not its last value alone. */ - PS->value = PS->value*PS->rownr + (pc->lp->bb_parentOF-OFsol) / (varsol*uplim); - PS->rownr++; - PS->value /= PS->rownr; - /* Check if we have enough information to restart */ - if(PS->rownr == mipvar) { - pc->updatesfinished++; - if(is_bb_mode(pc->lp, NODE_RESTARTMODE) && - (pc->updatesfinished/(2.0*pc->lp->int_vars) > - pc->restartlimit)) { - pc->lp->bb_break = AUTOMATIC; - pc->restartlimit *= 2.681; /* KE: Who can figure this one out? */ - if(pc->restartlimit > 1) - pc->lp->bb_rule -= NODE_RESTARTMODE; - report(pc->lp, NORMAL, "update_pseudocost: Restarting with updated pseudocosts\n"); - } - } - } - pc->lp->bb_parentOF = OFsol; -} - -STATIC REAL get_pseudobranchcost(BBPSrec *pc, int mipvar, MYBOOL dofloor) -{ - if(dofloor) - return( pc->LOcost[mipvar].value ); - else - return( pc->UPcost[mipvar].value ); -} - -STATIC REAL get_pseudonodecost(BBPSrec *pc, int mipvar, int vartype, REAL varsol) -{ - REAL hold, uplim; - - uplim = get_pseudorange(pc, mipvar, vartype); - varsol = modf(varsol/uplim, &hold); - if(isnan(varsol)) - varsol = 0; - - hold = pc->LOcost[mipvar].value*varsol + - pc->UPcost[mipvar].value*(1-varsol); - - return( hold*uplim ); -} - -STATIC int compute_theta(lprec *lp, int rownr, LREAL *theta, int isupbound, REAL HarrisScalar, MYBOOL primal) -/* The purpose of this routine is to compute the non-basic bound state / value of - the leaving variable. Note that the incoming theta is "d" in Chvatal-terminology */ -{ - int colnr = lp->var_basic[rownr]; - register LREAL x = lp->rhs[rownr]; - REAL lb = 0, /* Put lower bound here when the fully bounded version is implemented */ - ub = lp->upbo[colnr], - eps = lp->epsprimal; /* Primal feasibility tolerance */ - - /* Compute theta for the primal simplex */ - HarrisScalar *= eps; - if(primal) { - - if(*theta > 0) - x -= lb - HarrisScalar; /* A positive number */ - else if(ub < lp->infinite) - x -= ub + HarrisScalar; /* A negative number */ - else { - *theta = -lp->infinite; - return( colnr ); - } - } - /* Compute theta for the dual simplex */ - else { - - if(isupbound) - *theta = -(*theta); - - /* Current value is below or equal to its lower bound */ - if(x < lb+eps) - x -= lb - HarrisScalar; - - /* Current value is above or equal to its upper bound */ - else if(x > ub-eps) { - if(ub >= lp->infinite) { - *theta = lp->infinite * my_sign(*theta); - return( colnr ); - } - else - x -= ub + HarrisScalar; - } - } - my_roundzero(x, lp->epsmachine); - *theta = x / *theta; - -#ifdef EnforcePositiveTheta - /* Check if we have negative theta due to rounding or an internal error */ - if(*theta < 0) { - if(primal && (ub == lb)) - lp->rhs[rownr] = lb; - else -#ifdef Paranoia - if(*theta < -eps) { - report(lp, DETAILED, "compute_theta: Negative theta (%g) not allowed in base-0 version of lp_solve\n", - *theta); - } -#endif - *theta = 0; - } -#endif - - return( colnr ); -} - -STATIC MYBOOL check_degeneracy(lprec *lp, REAL *pcol, int *degencount) -/* Check if the entering column Pi=Inv(B)*a is likely to produce improvement; - (cfr. Istvan Maros: CTOTSM p. 233) */ -{ - int i, ndegen; - REAL *rhs, sdegen, epsmargin = lp->epsprimal; - - sdegen = 0; - ndegen = 0; - rhs = lp->rhs; - for(i = 1; i <= lp->rows; i++) { - rhs++; - pcol++; - if(fabs(*rhs) < epsmargin) { - sdegen += *pcol; - ndegen++; - } - else if(fabs((*rhs)-lp->upbo[lp->var_basic[i]]) < epsmargin) { - sdegen -= *pcol; - ndegen++; - } - } - if(degencount != NULL) - *degencount = ndegen; -/* sdegen += epsmargin*ndegen; */ - return( (MYBOOL) (sdegen <= 0) ); -} - -STATIC MYBOOL performiteration(lprec *lp, int rownr, int varin, LREAL theta, MYBOOL primal, MYBOOL allowminit, - REAL *prow, int *nzprow, REAL *pcol, int *nzpcol, int *boundswaps) -{ - int varout; - REAL pivot, epsmargin, leavingValue, leavingUB, enteringUB; - MYBOOL leavingToUB = FALSE, enteringFromUB, enteringIsFixed, leavingIsFixed; - MYBOOL *islower = &(lp->is_lower[varin]); - MYBOOL minitNow = FALSE, minitStatus = ITERATE_MAJORMAJOR; - LREAL deltatheta = theta; - - if(userabort(lp, MSG_ITERATION)) - return( minitNow ); - -#ifdef Paranoia - if(rownr > lp->rows) { - if (lp->spx_trace) - report(lp, IMPORTANT, "performiteration: Numeric instability encountered!\n"); - lp->spx_status = NUMFAILURE; - return( FALSE ); - } -#endif - varout = lp->var_basic[rownr]; -#ifdef Paranoia - if(!lp->is_lower[varout]) - report(lp, SEVERE, "performiteration: Leaving variable %d was at its upper bound at iter %.0f\n", - varout, (double) get_total_iter(lp)); -#endif - - /* Theta is the largest change possible (strictest constraint) for the entering - variable (Theta is Chvatal's "t", ref. Linear Programming, pages 124 and 156) */ - lp->current_iter++; - - /* Test if it is possible to do a cheap "minor iteration"; i.e. set entering - variable to its opposite bound, without entering the basis - which is - obviously not possible for fixed variables! */ - epsmargin = lp->epsprimal; - enteringFromUB = !(*islower); - enteringUB = lp->upbo[varin]; - leavingUB = lp->upbo[varout]; - enteringIsFixed = (MYBOOL) (fabs(enteringUB) < epsmargin); - leavingIsFixed = (MYBOOL) (fabs(leavingUB) < epsmargin); -#if defined _PRICE_NOBOUNDFLIP - allowminit &= !ISMASKSET(lp->piv_strategy, PRICE_NOBOUNDFLIP); -#endif -#ifdef Paranoia - if(enteringUB < 0) - report(lp, SEVERE, "performiteration: Negative range for entering variable %d at iter %.0f\n", - varin, (double) get_total_iter(lp)); - if(leavingUB < 0) - report(lp, SEVERE, "performiteration: Negative range for leaving variable %d at iter %.0f\n", - varout, (double) get_total_iter(lp)); -#endif - - /* Handle batch bound swaps with the dual long-step algorithm; - Loop over specified bound swaps; update RHS and Theta for bound swaps */ - if((boundswaps != NULL) && (boundswaps[0] > 0)) { - - int i, boundvar; - REAL *hold; - - /* Allocate and initialize accumulation array */ - allocREAL(lp, &hold, lp->rows + 1, TRUE); - - /* Accumulate effective bound swaps and update flag */ - for(i = 1; i <= boundswaps[0]; i++) { - boundvar = boundswaps[i]; - deltatheta = my_chsign(!lp->is_lower[boundvar], lp->upbo[boundvar]); - mat_multadd(lp->matA, hold, boundvar, deltatheta); - lp->is_lower[boundvar] = !lp->is_lower[boundvar]; - } - lp->current_bswap += boundswaps[0]; - lp->current_iter += boundswaps[0]; - - /* Solve for bound flip update vector (note that this does not - overwrite the stored update vector for the entering variable) */ - ftran(lp, hold, NULL, lp->epsmachine); - if(!lp->obj_in_basis) - hold[0] = 0; /* The correct reduced cost goes here (adjusted for bound state) ****** */ - - /* Update the RHS / basic variable values and set revised thetas */ - pivot = lp->bfp_pivotRHS(lp, 1, hold); - deltatheta = multi_enteringtheta(lp->longsteps); - theta = deltatheta; - - FREE(hold); - } - - /* Otherwise to traditional check for single bound swap */ - else if(allowminit && - !enteringIsFixed) { - -/* pivot = epsmargin; */ - pivot = lp->epsdual; -/* #define v51mode */ /* Enable this for v5.1 operation mode */ -#ifdef v51mode - if(((lp->simplex_mode & SIMPLEX_Phase1_DUAL) == 0) || - !is_constr_type(lp, rownr, EQ)) /* *** DEBUG CODE KE */ -#endif - if(enteringUB - theta < -pivot) { - -#ifndef v51mode - if(fabs(enteringUB - theta) < pivot) - minitStatus = ITERATE_MINORMAJOR; - else -#endif - minitStatus = ITERATE_MINORRETRY; - minitNow = (MYBOOL) (minitStatus != ITERATE_MAJORMAJOR); - } - } - - /* Process for traditional style single minor iteration */ - if(minitNow) { - - /* Set the new values (note that theta is set to always be positive) */ - theta = MIN(fabs(theta), enteringUB); - - /* Update the RHS / variable values and do bound-swap */ - pivot = lp->bfp_pivotRHS(lp, theta, NULL); - *islower = !(*islower); - - lp->current_bswap++; - - } - - /* Process for major iteration */ - else { - - /* Update the active pricer for the current pivot */ - updatePricer(lp, rownr, varin, lp->bfp_pivotvector(lp), prow, nzprow); - - /* Update the current basic variable values */ - pivot = lp->bfp_pivotRHS(lp, theta, NULL); - - /* See if the leaving variable goes directly to its upper bound. */ - leavingValue = lp->rhs[rownr]; - leavingToUB = (MYBOOL) (leavingValue > 0.5*leavingUB); - lp->is_lower[varout] = leavingIsFixed || !leavingToUB; - - /* Set the value of the entering varible (theta always set to be positive) */ - if(enteringFromUB) { - lp->rhs[rownr] = enteringUB - deltatheta; - *islower = TRUE; - } - else - lp->rhs[rownr] = deltatheta; - my_roundzero(lp->rhs[rownr], epsmargin); - - /* Update basis indeces */ - varout = set_basisvar(lp, rownr, varin); - - /* Finalize the update in preparation for next major iteration */ - lp->bfp_finishupdate(lp, enteringFromUB); - - } - - /* Show pivot tracking information, if specified */ - if((lp->verbose > NORMAL) && (MIP_count(lp) == 0) && - ((lp->current_iter % MAX(2, lp->rows / 10)) == 0)) - report(lp, NORMAL, "Objective value " RESULTVALUEMASK " at iter %10.0f.\n", - lp->rhs[0], (double) get_total_iter(lp)); - -#if 0 - if(verify_solution(lp, FALSE, my_if(minitNow, "MINOR", "MAJOR")) >= 0) { - if(minitNow) - pivot = get_obj_active(lp, varin); - else - pivot = get_obj_active(lp, varout); - } -#endif -#if 0 - if((lp->longsteps != NULL) && (boundswaps[0] > 0) && lp->longsteps->objcheck && - ((pivot = fabs(my_reldiff(lp->rhs[0], lp->longsteps->obj_last))) > lp->epssolution)) { - report(lp, IMPORTANT, "performiteration: Objective value gap %8.6f found at iter %6.0f (%d bound flips, %d)\n", - pivot, (double) get_total_iter(lp), boundswaps[0], enteringFromUB); - } -#endif - - if(lp->spx_trace) { - if(minitNow) - report(lp, NORMAL, "I:%5.0f - minor - %5d ignored, %5d flips from %s with THETA=%g and OBJ=%g\n", - (double) get_total_iter(lp), varout, varin, (enteringFromUB ? "UPPER" : "LOWER"), theta, lp->rhs[0]); - else - report(lp, NORMAL, "I:%5.0f - MAJOR - %5d leaves to %s, %5d enters from %s with THETA=%g and OBJ=%g\n", - (double) get_total_iter(lp), varout, (leavingToUB ? "UPPER" : "LOWER"), - varin, (enteringFromUB ? "UPPER" : "LOWER"), theta, lp->rhs[0]); - if(minitNow) { - if(!lp->is_lower[varin]) - report(lp, DETAILED, - "performiteration: Variable %d changed to its lower bound at iter %.0f (from %g)\n", - varin, (double) get_total_iter(lp), enteringUB); - else - report(lp, DETAILED, - "performiteration: Variable %d changed to its upper bound at iter %.0f (to %g)\n", - varin, (double) get_total_iter(lp), enteringUB); - } - else - report(lp, NORMAL, - "performiteration: Variable %d entered basis at iter %.0f at " RESULTVALUEMASK "\n", - varin, (double) get_total_iter(lp), lp->rhs[rownr]); - if(!primal) { - pivot = compute_feasibilitygap(lp, (MYBOOL)!primal, TRUE); - report(lp, NORMAL, "performiteration: Feasibility gap at iter %.0f is " RESULTVALUEMASK "\n", - (double) get_total_iter(lp), pivot); - } - else - report(lp, NORMAL, - "performiteration: Current objective function value at iter %.0f is " RESULTVALUEMASK "\n", - (double) get_total_iter(lp), lp->rhs[0]); - } - - return( minitStatus ); - -} /* performiteration */ - -STATIC REAL get_refactfrequency(lprec *lp, MYBOOL final) -{ - COUNTER iters; - int refacts; - - /* Get numerator and divisor information */ - iters = (lp->total_iter+lp->current_iter) - (lp->total_bswap+lp->current_bswap); - refacts = lp->bfp_refactcount(lp, BFP_STAT_REFACT_TOTAL); - - /* Return frequency for different cases: - 1) Actual frequency in case final statistic is desired - 2) Dummy if we are in a B&B process - 3) Frequency with added initialization offsets which - are diluted in course of the solution process */ - if(final) - return( (REAL) (iters) / MAX(1,refacts) ); - else if(lp->bb_totalnodes > 0) - return( (REAL) lp->bfp_pivotmax(lp) ); - else - return( (REAL) (lp->bfp_pivotmax(lp)+iters) / (1+refacts) ); -} - -#if 0 -/* INLINE */ MYBOOL is_fixedvar(lprec *lp, int variable) -{ - if((lp->bb_bounds != NULL && lp->bb_bounds->UBzerobased) || (variable <= lp->rows)) - return( (MYBOOL) (lp->upbo[variable] < lp->epsprimal) ); - else - return( (MYBOOL) (lp->upbo[variable]-lp->lowbo[variable] < lp->epsprimal) ); -} /* is_fixedvar */ -#else -MYBOOL is_fixedvar(lprec *lp, int varnr) -{ - if(lp->bb_bounds == NULL) { - if(varnr <= lp->rows) - return( (MYBOOL) (lp->orig_upbo[varnr] < lp->epsmachine) ); - else - return( (MYBOOL) (lp->orig_upbo[varnr]-lp->orig_lowbo[varnr] < lp->epsmachine) ); - } - else if((varnr <= lp->rows) || (lp->bb_bounds->UBzerobased == TRUE)) - return( (MYBOOL) (lp->upbo[varnr] < lp->epsvalue) ); - else - return( (MYBOOL) (lp->upbo[varnr]-lp->lowbo[varnr] < lp->epsvalue) ); -} -#endif - -STATIC MYBOOL solution_is_int(lprec *lp, int index, MYBOOL checkfixed) -{ -#if 1 - return( (MYBOOL) (isINT(lp, lp->solution[index]) && (!checkfixed || is_fixedvar(lp, index))) ); -#else - if(isINT(lp, lp->solution[index])) { - if(checkfixed) - return(is_fixedvar(lp, index)); - else - return(TRUE); - } - return(FALSE); -#endif -} /* solution_is_int */ - - -MYBOOL __WINAPI set_multiprice(lprec *lp, int multiblockdiv) -{ - /* See if we are resetting multiply priced column structures */ - if(multiblockdiv != lp->multiblockdiv) { - if(multiblockdiv < 1) - multiblockdiv = 1; - lp->multiblockdiv = multiblockdiv; - multi_free(&(lp->multivars)); - } - return( TRUE ); -} - -int __WINAPI get_multiprice(lprec *lp, MYBOOL getabssize) -{ - if((lp->multivars == NULL) || (lp->multivars->used == 0)) - return( 0 ); - if(getabssize) - return( lp->multivars->size ); - else - return( lp->multiblockdiv ); -} - -MYBOOL __WINAPI set_partialprice(lprec *lp, int blockcount, int *blockstart, MYBOOL isrow) -{ - int ne, i, items; - partialrec **blockdata; - - /* Determine partial target (rows or columns) */ - if(isrow) - blockdata = &(lp->rowblocks); - else - blockdata = &(lp->colblocks); - - /* See if we are resetting partial blocks */ - ne = 0; - items = IF(isrow, lp->rows, lp->columns); - if(blockcount == 1) - partial_freeBlocks(blockdata); - - /* Set a default block count if this was not specified */ - else if(blockcount <= 0) { - blockstart = NULL; - if(items < DEF_PARTIALBLOCKS*DEF_PARTIALBLOCKS) - blockcount = items / DEF_PARTIALBLOCKS + 1; - else - blockcount = DEF_PARTIALBLOCKS; - ne = items / blockcount; - if(ne * blockcount < items) - ne++; - } - - /* Fill partial block arrays; - Note: These will be modified during preprocess to reflect - presolved columns and the handling of slack variables. */ - if(blockcount > 1) { - MYBOOL isNew = (MYBOOL) (*blockdata == NULL); - - /* Provide for extra block with slack variables in the column mode */ - i = 0; - if(!isrow) - i++; - - /* (Re)-allocate memory */ - if(isNew) - *blockdata = partial_createBlocks(lp, isrow); - allocINT(lp, &((*blockdata)->blockend), blockcount+i+1, AUTOMATIC); - allocINT(lp, &((*blockdata)->blockpos), blockcount+i+1, AUTOMATIC); - - /* Copy the user-provided block start positions */ - if(blockstart != NULL) { - MEMCOPY((*blockdata)->blockend+i, blockstart, blockcount+i+1); - if(!isrow) { - blockcount++; - (*blockdata)->blockend[0] = 1; - for(i = 1; i < blockcount; i++) - (*blockdata)->blockend[i] += lp->rows; - } - } - - /* Fill the block ending positions if they were not specified */ - else { - (*blockdata)->blockend[0] = 1; - (*blockdata)->blockpos[0] = 1; - if(ne == 0) { - ne = items / blockcount; - /* Increase the block size if we have a fractional value */ - while(ne * blockcount < items) - ne++; - } - i = 1; - if(!isrow) { - (*blockdata)->blockend[i] = (*blockdata)->blockend[i-1]+lp->rows; - blockcount++; - i++; - items += lp->rows; - } - for(; i < blockcount; i++) - (*blockdata)->blockend[i] = (*blockdata)->blockend[i-1]+ne; - - /* Let the last block handle the "residual" */ - (*blockdata)->blockend[blockcount] = items+1; - } - - /* Fill starting positions (used in multiple partial pricing) */ - for(i = 1; i <= blockcount; i++) - (*blockdata)->blockpos[i] = (*blockdata)->blockend[i-1]; - - } - - /* Update block count */ - (*blockdata)->blockcount = blockcount; - - - return( TRUE ); -} /* set_partialprice */ - -void __WINAPI get_partialprice(lprec *lp, int *blockcount, int *blockstart, MYBOOL isrow) -{ - partialrec *blockdata; - - /* Determine partial target (rows or columns) */ - if(isrow) - blockdata = lp->rowblocks; - else - blockdata = lp->colblocks; - - *blockcount = partial_countBlocks(lp, isrow); - if((blockdata != NULL) && (blockstart != NULL)) { - int i = 0, k = *blockcount; - if(!isrow) - i++; - MEMCOPY(blockstart, blockdata->blockend + i, k - i); - if(!isrow) { - k -= i; - for(i = 0; i < k; i++) - blockstart[i] -= lp->rows; - } - } -} - - -/* Solution-related functions */ -STATIC MYBOOL bb_better(lprec *lp, int target, int mode) -/* Must handle four modes (logic assumes Min!): - -----|--.--|-----> - 1 ++++++----------- LHS exclusive test point is better - 2 +++++++++-------- LHS inclusive - 3 ++++++-----++++++ LHS+RHS exclusive - 4 --------+++++++++ RHS inclusive - 5 -----------++++++ RHS exclusive -*/ -{ - REAL epsvalue, offset = lp->epsprimal, - refvalue = lp->infinite, testvalue = lp->solution[0]; - MYBOOL ismax = is_maxim(lp), - relgap = is_action(mode, OF_TEST_RELGAP), - fcast = is_action(target, OF_PROJECTED), - delta = is_action(target, OF_DELTA); - - if(relgap) { - epsvalue = lp->mip_relgap; - clear_action(&mode, OF_TEST_RELGAP); - } - else - epsvalue = lp->mip_absgap; - - if(delta) - clear_action(&target, OF_DELTA); - if(fcast) - clear_action(&target, OF_PROJECTED); -#ifdef Paranoia - if((mode < OF_TEST_BT) || (mode > OF_TEST_WT)) - report(lp, SEVERE, "bb_better: Passed invalid mode '%d'\n", mode); -#endif - - switch(target) { - case OF_RELAXED: refvalue = lp->real_solution; - break; - case OF_INCUMBENT: refvalue = lp->best_solution[0]; - break; - case OF_WORKING: refvalue = my_chsign(!ismax, lp->bb_workOF /* unscaled_value(lp, lp->bb_workOF, 0) */ ); - if(fcast) - testvalue = my_chsign(!ismax, lp->longsteps->obj_last) - epsvalue; - else - testvalue = my_chsign(!ismax, lp->rhs[0] /* unscaled_value(lp, lp->rhs[0], 0) */); - break; - case OF_USERBREAK: refvalue = lp->bb_breakOF; - break; - case OF_HEURISTIC: refvalue = lp->bb_heuristicOF; - break; - case OF_DUALLIMIT: refvalue = lp->bb_limitOF; - break; - default : report(lp, SEVERE, "bb_better: Passed invalid test target '%d'\n", target); - return( FALSE ); - } - - /* Adjust the test value for the desired acceptability window */ - if(delta) { - SETMAX(epsvalue, lp->bb_deltaOF - epsvalue); - } - else - epsvalue = my_chsign(target >= OF_USERBREAK, epsvalue); /* *** This seems Ok, but should be verified */ - testvalue += my_chsign(ismax, epsvalue); - - /* Compute the raw test value */ - if(relgap) - testvalue = my_reldiff(testvalue, refvalue); - else - testvalue -= refvalue; - - /* Make test value adjustment based on the selected option */ - if(mode == OF_TEST_NE) - relgap = (MYBOOL) (fabs(testvalue) >= offset); - else { - testvalue = my_chsign(mode > OF_TEST_NE, testvalue); - testvalue = my_chsign(ismax, testvalue); - relgap = (MYBOOL) (testvalue < offset); - } - return( relgap ); -} - -STATIC void construct_solution(lprec *lp, REAL *target) -{ - int i, j, basi; - REAL f, epsvalue = lp->epsprimal; - REAL *solution; - REAL *value; - int *rownr; - MATrec *mat = lp->matA; - - if(target == NULL) - solution = lp->solution; - else - solution = target; - - /* Initialize OF and slack variables. */ - for(i = 0; i <= lp->rows; i++) { -#ifdef LegacySlackDefinition - if(i == 0) - f = unscaled_value(lp, -lp->orig_rhs[i], i); - else { - j = lp->presolve_undo->var_to_orig[i]; - if(j > 0) { - f = lp->presolve_undo->fixed_rhs[j]; - f = unscaled_value(lp, f, i); - } - else - f = 0; - } -#else - f = lp->orig_rhs[i]; - if((i > 0) && !lp->is_basic[i] && !lp->is_lower[i]) -#ifdef SlackInitMinusInf - f -= my_chsign(is_chsign(lp, i), fabs(lp->upbo[i])); -#else - f -= my_chsign(is_chsign(lp, i), fabs(lp->lowbo[i] + lp->upbo[i])); -#endif - f = unscaled_value(lp, -f, i); -#endif - solution[i] = f; - } - - /* Initialize user variables to their lower bounds. */ - for(i = lp->rows+1; i <= lp->sum; i++) - solution[i] = lp->lowbo[i]; - - /* Add values of user basic variables. */ - for(i = 1; i <= lp->rows; i++) { - basi = lp->var_basic[i]; - if(basi > lp->rows) { - solution[basi] += lp->rhs[i]; - } - } - - /* 1. Adjust non-basic variables at their upper bounds, - 2. Unscale all user variables, - 3. Optionally do precision management. */ - for(i = lp->rows + 1; i <= lp->sum; i++) { - if(!lp->is_basic[i] && !lp->is_lower[i]) - solution[i] += lp->upbo[i]; - solution[i] = unscaled_value(lp, solution[i], i); -#ifdef xImproveSolutionPrecision - if(is_int(lp, i-lp->rows)) - solution[i] = restoreINT(solution[i], lp->epsint); - else - solution[i] = restoreINT(solution[i], lp->epsprimal); -#endif - } - - /* Compute the OF and slack values "in extentio" */ - for(j = 1; j <= lp->columns; j++) { - f = solution[lp->rows + j]; - if(f != 0) { - solution[0] += f * unscaled_mat(lp, lp->orig_obj[j], 0, j); - i = mat->col_end[j-1]; - basi = mat->col_end[j]; - rownr = &COL_MAT_ROWNR(i); - value = &COL_MAT_VALUE(i); - for(; i < basi; - i++, rownr += matRowColStep, value += matValueStep) - solution[*rownr] += f * unscaled_mat(lp, *value, *rownr, j); - } - } - - /* Do slack precision management and sign reversal if necessary */ - for(i = 0; i <= lp->rows; i++) { -#ifdef ImproveSolutionPrecision - my_roundzero(solution[i], epsvalue); -#endif - if(is_chsign(lp, i)) - solution[i] = my_flipsign(solution[i]); - } - - /* Record the best real-valued solution and compute a simple MIP solution limit */ - if(target == NULL) { - if(is_infinite(lp, lp->real_solution)) { - lp->bb_workOF = lp->rhs[0]; - lp->real_solution = solution[0]; - if(is_infinite(lp, lp->bb_limitOF)) - lp->bb_limitOF = lp->real_solution; - else { - if(is_maxim(lp)) { - SETMIN(lp->bb_limitOF, lp->real_solution); - } - else { - SETMAX(lp->bb_limitOF, lp->real_solution); - } - } - - /* Do MIP-related tests and computations */ - if((lp->int_vars > 0) && mat_validate(lp->matA) /* && !lp->wasPresolved */) { /* && !lp->wasPresolved uncommented by findings of William H. Patton. The code was never executed when the test was there. The code has effect in an integer model with all integer objective coeff. to cut-off optimization and thus make it faster */ - REAL fixedOF = unscaled_value(lp, lp->orig_rhs[0], 0); - - /* Check if we have an all-integer OF */ - basi = lp->columns; - for(j = 1; j <= basi; j++) { - f = fabs(get_mat(lp, 0, j)) + lp->epsint / 2; - if(f > lp->epsint) { /* If coefficient is 0 then it doesn't influence OF, even it variable is not integer */ - if(!is_int(lp, j) || (fmod(f, 1) > lp->epsint)) - break; - } - } - - /* If so, we can round up the fractional OF */ - if(j > basi) { - f = my_chsign(is_maxim(lp), lp->real_solution) + fixedOF; - f = floor(f+(1-epsvalue)); - f = my_chsign(is_maxim(lp), f - fixedOF); - if(is_infinite(lp, lp->bb_limitOF)) - lp->bb_limitOF = f; - else if(is_maxim(lp)) { - SETMIN(lp->bb_limitOF, f); - } - else { - SETMAX(lp->bb_limitOF, f); - } - } - } - - /* Check that a user limit on the OF is feasible */ - if((lp->int_vars > 0) && - (my_chsign(is_maxim(lp), my_reldiff(lp->best_solution[0],lp->bb_limitOF)) < -epsvalue)) { - lp->spx_status = INFEASIBLE; - lp->bb_break = TRUE; - } - } - } - -} /* construct_solution */ - -STATIC int check_solution(lprec *lp, int lastcolumn, REAL *solution, - REAL *upbo, REAL *lowbo, REAL tolerance) -{ -/*#define UseMaxValueInCheck*/ -#define RelativeAccuracyCheck - MYBOOL isSC; - REAL test, value, diff, maxdiff = 0.0, maxerr = 0.0, *matValue; -#ifndef RelativeAccuracyCheck - REAL hold; -#endif -#ifdef UseMaxValueInCheck - REAL *maxvalue = NULL; -#elif !defined RelativeAccuracyCheck - REAL *plusum = NULL, *negsum = NULL; -#endif - int i,j,n, errlevel = IMPORTANT, errlimit = 10, *matRownr, *matColnr; - MATrec *mat = lp->matA; - int solveStatus = OPTIMAL; - - report(lp, NORMAL, " \n"); - if(MIP_count(lp) > 0) - report(lp, NORMAL, "%s solution " RESULTVALUEMASK " after %10.0f iter, %9.0f nodes (gap %.1f%%).\n", - my_if(lp->bb_break && !bb_better(lp, OF_DUALLIMIT, OF_TEST_BE) && bb_better(lp, OF_RELAXED, OF_TEST_NE), "Subopt.", "Optimal"), - solution[0], (double) lp->total_iter, (double) lp->bb_totalnodes, - 100.0*fabs(my_reldiff(solution[0], lp->bb_limitOF))); - else - report(lp, NORMAL, "Optimal solution " RESULTVALUEMASK " after %10.0f iter.\n", - solution[0], (double) lp->total_iter); - - /* Find the signed sums and the largest absolute product in the matrix (exclude the OF for speed) */ -#ifdef UseMaxValueInCheck - allocREAL(lp, &maxvalue, lp->rows + 1, FALSE); - for(i = 0; i <= lp->rows; i++) - maxvalue[i] = fabs(get_rh(lp, i)); -#elif !defined RelativeAccuracyCheck - allocREAL(lp, &plusum, lp->rows + 1, TRUE); - allocREAL(lp, &negsum, lp->rows + 1, TRUE); -#endif - -#if defined UseMaxValueInCheck || !defined RelativeAccuracyCheck - n = get_nonzeros(lp); - matRownr = &COL_MAT_ROWNR(0); - matColnr = &COL_MAT_COLNR(0); - matValue = &COL_MAT_VALUE(0); - for(i = 0; i < n; i++, matRownr += matRowColStep, - matColnr += matRowColStep, - matValue += matValueStep) { - test = unscaled_mat(lp, *matValue, *matRownr, *matColnr); - test *= solution[lp->rows + (*matColnr)]; -#ifdef UseMaxValueInCheck - test = fabs(test); - if(test > maxvalue[*matRownr]) - maxvalue[*matRownr] = test; -#elif !defined RelativeAccuracyCheck - if(test > 0) - plusum[*matRownr] += test; - else - negsum[*matRownr] += test; -#endif - } -#endif - - /* Check if solution values are within the bounds; allowing a margin for numeric errors */ - n = 0; - for(i = lp->rows + 1; i <= lp->rows+lastcolumn; i++) { - - value = solution[i]; - - /* Check for case where we are testing an intermediate solution - (variables shifted to the origin) */ - if(lowbo == NULL) - test = 0; - else - test = unscaled_value(lp, lowbo[i], i); - - isSC = is_semicont(lp, i - lp->rows); - diff = -my_reldiff(value, test); -#ifdef RelativeAccuracyCheck - if (isSC && diff > 0 && my_reldiff(fabs(value), 0.0) < diff) - diff = my_reldiff(fabs(value), 0.0); -#else - if(isSC && diff > 0 && (value < test/2)) - test = 0; -#endif - if(diff > 0) { - SETMAX(maxdiff, diff); -#ifdef RelativeAccuracyCheck - maxerr = maxdiff; -#else - SETMAX(maxerr, test - value); -#endif - } - if((diff > tolerance) && !isSC) { - if(n < errlimit) - report(lp, errlevel, - "check_solution: Variable %s = " RESULTVALUEMASK " is below its lower bound " RESULTVALUEMASK "\n", - get_col_name(lp, i-lp->rows), value, test); - n++; - } - - test = unscaled_value(lp, upbo[i], i); - diff = my_reldiff(value, test); - if(diff > 0) { - SETMAX(maxdiff, diff); -#ifdef RelativeAccuracyCheck - maxerr = maxdiff; -#else - SETMAX(maxerr, value - test); -#endif - } - if(diff > tolerance) { - if(n < errlimit) - report(lp, errlevel, - "check_solution: Variable %s = " RESULTVALUEMASK " is above its upper bound " RESULTVALUEMASK "\n", - get_col_name(lp, i-lp->rows), value, test); - n++; - } - } - - /* Check if constraint values are within the bounds; allowing a margin for numeric errors */ - for(i = 1; i <= lp->rows; i++) { - test = lp->orig_rhs[i]; - - if(is_infinite(lp, test)) - continue; - -#ifdef LegacySlackDefinition - j = lp->presolve_undo->var_to_orig[i]; - if(j != 0) { - if(is_infinite(lp, lp->presolve_undo->fixed_rhs[j])) - continue; - test += lp->presolve_undo->fixed_rhs[j]; - } -#endif - - if(is_chsign(lp, i)) { - test = my_flipsign(test); - test += fabs(upbo[i]); - } - value = solution[i]; - test = unscaled_value(lp, test, i); -#ifndef LegacySlackDefinition - value += test; -#endif - diff = my_reldiff(value, test); -#ifndef RelativeAccuracyCheck -#ifdef UseMaxValueInCheck - hold = maxvalue[i]; -#else - hold = plusum[i] - negsum[i]; -#endif - if(hold < lp->epsvalue) - hold = 1; - - diff = my_reldiff((value+1)/hold, (test+1)/hold); -#endif - if(diff > 0) { - SETMAX(maxdiff, diff); -#ifdef RelativeAccuracyCheck - maxerr = maxdiff; -#else - SETMAX(maxerr, value-test); -#endif - } - if(diff > tolerance) { - if(n < errlimit) - report(lp, errlevel, - "check_solution: Constraint %s = " RESULTVALUEMASK " is above its %s " RESULTVALUEMASK "\n", - get_row_name(lp, i), value, - (is_constr_type(lp, i, EQ) ? "equality of" : "upper bound"), test); - n++; - } - - test = lp->orig_rhs[i]; -#ifdef LegacySlackDefinition - j = lp->presolve_undo->var_to_orig[i]; - if(j != 0) { - if(is_infinite(lp, lp->presolve_undo->fixed_rhs[j])) - continue; - test += lp->presolve_undo->fixed_rhs[j]; - } -#endif - - value = solution[i]; - if(is_chsign(lp, i)) - test = my_flipsign(test); - else { - if(is_infinite(lp, upbo[i])) - continue; - test -= fabs(upbo[i]); -#ifndef LegacySlackDefinition - value = fabs(upbo[i]) - value; -#endif - } - test = unscaled_value(lp, test, i); -#ifndef LegacySlackDefinition - value += test; -#endif - diff = -my_reldiff(value, test); -#ifndef RelativeAccuracyCheck -#ifdef UseMaxValueInCheck - hold = maxvalue[i]; -#else - hold = plusum[i] - negsum[i]; -#endif - if(hold < lp->epsvalue) - hold = 1; - diff = -my_reldiff((value+1)/hold, (test+1)/hold); -#endif - if(diff > 0) { - SETMAX(maxdiff, diff); -#ifdef RelativeAccuracyCheck - maxerr = maxdiff; -#else - SETMAX(maxerr, test-value); -#endif - } - if(diff > tolerance) { - if(n < errlimit) - report(lp, errlevel, - "check_solution: Constraint %s = " RESULTVALUEMASK " is below its %s " RESULTVALUEMASK "\n", - get_row_name(lp, i), value, - (is_constr_type(lp, i, EQ) ? "equality of" : "lower bound"), test); - n++; - } - } - -#ifdef UseMaxValueInCheck - FREE(maxvalue); -#elif !defined RelativeAccuracyCheck - FREE(plusum); - FREE(negsum); -#endif - -#ifdef RelativeAccuracyCheck - report(lp, NORMAL, "\nRelative numeric accuracy ||*|| = %g\n", maxdiff); - if (maxdiff > lp->accuracy_error) - { - report(lp, IMPORTANT, "\nUnacceptable accuracy found (worse than required %g)\n", lp->accuracy_error); - solveStatus = ACCURACYERROR; - } -#else - if(n > 0) { - report(lp, IMPORTANT, "\nSeriously low accuracy found ||*|| = %g (rel. error %g)\n", - maxerr, maxdiff); - solveStatus = NUMFAILURE; - } - else { - if(maxerr > 1.0e-7) - report(lp, NORMAL, "\nMarginal numeric accuracy ||*|| = %g (rel. error %g)\n", - maxerr, maxdiff); - else if(maxerr > 1.0e-9) - report(lp, NORMAL, "\nReasonable numeric accuracy ||*|| = %g (rel. error %g)\n", - maxerr, maxdiff); - else if(maxerr > 1.0e11) - report(lp, NORMAL, "\nVery good numeric accuracy ||*|| = %g\n", maxerr); - else - report(lp, NORMAL, "\nExcellent numeric accuracy ||*|| = %g\n", maxerr); - } -#endif - - lp->accuracy = maxerr; - - return(solveStatus); -} /* check_solution */ - -REAL __WINAPI get_accuracy(lprec *lp) -{ - return(lp->accuracy); -} - -void __WINAPI set_break_numeric_accuracy(lprec *lp, REAL accuracy) -{ - lp->accuracy_error = accuracy; -} - -REAL __WINAPI get_break_numeric_accuracy(lprec *lp) -{ - return(lp->accuracy_error); -} - -STATIC void transfer_solution_var(lprec *lp, int uservar) -{ - if(lp->varmap_locked && (MYBOOL) ((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE)) { - uservar += lp->rows; - lp->full_solution[lp->presolve_undo->orig_rows + - lp->presolve_undo->var_to_orig[uservar]] = lp->best_solution[uservar]; - } -} -STATIC void transfer_solution(lprec *lp, MYBOOL dofinal) -{ - int i, ii; - - MEMCOPY(lp->best_solution, lp->solution, lp->sum + 1); - - /* Round integer solution values to actual integers */ - if(is_integerscaling(lp) && (lp->int_vars > 0)) - for(i = 1; i <= lp->columns; i++) { - if(is_int(lp, i)) { - ii = lp->rows + i; - lp->best_solution[ii] = floor(lp->best_solution[ii] + 0.5); - } - } - - /* Transfer to full solution vector in the case of presolved eliminations */ - if(dofinal && lp->varmap_locked && - (MYBOOL) ((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE)) { - presolveundorec *psundo = lp->presolve_undo; - - lp->full_solution[0] = lp->best_solution[0]; - for(i = 1; i <= lp->rows; i++) { - ii = psundo->var_to_orig[i]; -#ifdef Paranoia - if((ii < 0) || (ii > lp->presolve_undo->orig_rows)) - report(lp, SEVERE, "transfer_solution: Invalid mapping of row index %d to original index '%d'\n", - i, ii); -#endif - lp->full_solution[ii] = lp->best_solution[i]; - } - for(i = 1; i <= lp->columns; i++) { - ii = psundo->var_to_orig[lp->rows+i]; -#ifdef Paranoia - if((ii < 0) || (ii > lp->presolve_undo->orig_columns)) - report(lp, SEVERE, "transfer_solution: Invalid mapping of column index %d to original index '%d'\n", - i, ii); -#endif - lp->full_solution[psundo->orig_rows+ii] = lp->best_solution[lp->rows+i]; - } - } - -} - -STATIC MYBOOL construct_duals(lprec *lp) -{ - int i, n, *coltarget; - REAL scale0, value, dualOF; - - if(lp->duals != NULL) - free_duals(lp); - - if(is_action(lp->spx_action, ACTION_REBASE) || - is_action(lp->spx_action, ACTION_REINVERT) || (!lp->basis_valid) || - !allocREAL(lp, &(lp->duals), lp->sum + 1, AUTOMATIC)) - return(FALSE); - - /* Initialize */ - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->columns+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, SCAN_USERVARS+USE_NONBASICVARS, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return(FALSE); - } - bsolve(lp, 0, lp->duals, NULL, lp->epsmachine*DOUBLEROUND, 1.0); - prod_xA(lp, coltarget, lp->duals, NULL, lp->epsmachine, 1.0, - lp->duals, NULL, MAT_ROUNDDEFAULT | MAT_ROUNDRC); - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - - - /* The (Lagrangean) dual values are the reduced costs of the primal slacks; - when the slack is at its upper bound, change the sign. */ - n = lp->rows; - for(i = 1; i <= n; i++) { - if(lp->is_basic[i]) - lp->duals[i] = 0; - /* Added a test if variable is different from 0 because sometime you get -0 and this - is different from 0 on for example INTEL processors (ie 0 != -0 on INTEL !) PN */ - else if((is_chsign(lp, 0) == is_chsign(lp, i)) && lp->duals[i]) - lp->duals[i] = my_flipsign(lp->duals[i]); - } - if(is_maxim(lp)) { - n = lp->sum; - for(i = lp->rows + 1; i <= n; i++) - lp->duals[i] = my_flipsign(lp->duals[i]); - } - - /* If we presolved, then reconstruct the duals */ - n = lp->presolve_undo->orig_sum; - if(((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE) && - allocREAL(lp, &(lp->full_duals), n + 1, TRUE)) { - int ix, ii = lp->presolve_undo->orig_rows; - - n = lp->sum; - for(ix = 1; ix <= n; ix++) { - i = lp->presolve_undo->var_to_orig[ix]; - if(ix > lp->rows) - i += ii; -#ifdef Paranoia - /* Check for index out of range due to presolve */ - if(i > lp->presolve_undo->orig_sum) - report(lp, SEVERE, "construct_duals: Invalid presolve variable mapping found\n"); -#endif - lp->full_duals[i] = lp->duals[ix]; - } - presolve_rebuildUndo(lp, FALSE); - } - - /* Calculate the dual OF and do scaling adjustments to the duals */ - if(lp->scaling_used) - scale0 = lp->scalars[0]; - else - scale0 = 1; - dualOF = my_chsign(is_maxim(lp), lp->orig_rhs[0]) / scale0; - for(i = 1; i <= lp->sum; i++) { - value = scaled_value(lp, lp->duals[i] / scale0, i); - my_roundzero(value, lp->epsprimal); - lp->duals[i] = value; - if(i <= lp->rows) - dualOF += value * lp->solution[i]; - } - -#if 0 - /* See if we can make use of the dual OF; - note that we do not currently adjust properly for presolve */ - if(lp->rows == lp->presolve_undo->orig_rows) - if(MIP_count(lp) > 0) { - if(is_maxim(lp)) { - SETMIN(lp->bb_limitOF, dualOF); - } - else { - SETMAX(lp->bb_limitOF, dualOF); - } - } - else if(fabs(my_reldiff(dualOF, lp->solution[0])) > lp->epssolution) - report(lp, IMPORTANT, "calculate_duals: Check for possible suboptimal solution!\n"); -#endif - - return(TRUE); -} /* construct_duals */ - -/* Calculate sensitivity duals */ -STATIC MYBOOL construct_sensitivity_duals(lprec *lp) -{ - int k,varnr, ok = TRUE; - int *workINT = NULL; - REAL *pcol,a,infinite,epsvalue,from,till,objfromvalue; - - /* one column of the matrix */ - FREE(lp->objfromvalue); - FREE(lp->dualsfrom); - FREE(lp->dualstill); - if(!allocREAL(lp, &pcol, lp->rows + 1, TRUE) || - !allocREAL(lp, &lp->objfromvalue, lp->columns + 1, AUTOMATIC) || - !allocREAL(lp, &lp->dualsfrom, lp->sum + 1, AUTOMATIC) || - !allocREAL(lp, &lp->dualstill, lp->sum + 1, AUTOMATIC)) { - FREE(pcol); - FREE(lp->objfromvalue); - FREE(lp->dualsfrom); - FREE(lp->dualstill); - ok = FALSE; - } - else { - infinite=lp->infinite; - epsvalue=lp->epsmachine; - for(varnr=1; varnr<=lp->sum; varnr++) { - from=infinite; - till=infinite; - objfromvalue=infinite; - if (!lp->is_basic[varnr]) { - if (!fsolve(lp, varnr, pcol, workINT, epsvalue, 1.0, FALSE)) { /* construct one column of the tableau */ - ok = FALSE; - break; - } - /* Search for the rows(s) which first result in further iterations */ - for (k=1; k<=lp->rows; k++) { - if (fabs(pcol[k])>epsvalue) { - a = lp->rhs[k]/pcol[k]; - if((varnr > lp->rows) && (fabs(lp->solution[varnr]) <= epsvalue) && (a < objfromvalue) && (a >= lp->lowbo[varnr])) - objfromvalue = a; - if ((a<=0.0) && (pcol[k]<0.0) && (-a=0.0) && (pcol[k]>0.0) && ( aupbo[lp->var_basic[k]] < infinite) { - a = (REAL) ((lp->rhs[k]-lp->upbo[lp->var_basic[k]])/pcol[k]); - if((varnr > lp->rows) && (fabs(lp->solution[varnr]) <= epsvalue) && (a < objfromvalue) && (a >= lp->lowbo[varnr])) - objfromvalue = a; - if ((a<=0.0) && (pcol[k]>0.0) && (-a=0.0) && (pcol[k]<0.0) && ( ais_lower[varnr]) { - a=from; - from=till; - till=a; - } - if ((varnr<=lp->rows) && (!is_chsign(lp, varnr))) { - a=from; - from=till; - till=a; - } - } - - if (from!=infinite) - lp->dualsfrom[varnr]=lp->solution[varnr]-unscaled_value(lp, from, varnr); - else - lp->dualsfrom[varnr]=-infinite; - if (till!=infinite) - lp->dualstill[varnr]=lp->solution[varnr]+unscaled_value(lp, till, varnr); - else - lp->dualstill[varnr]=infinite; - - if (varnr > lp->rows) { - if (objfromvalue != infinite) { - if ((!sensrejvar) || (lp->upbo[varnr] != 0.0)) { - if (!lp->is_lower[varnr]) - objfromvalue = lp->upbo[varnr] - objfromvalue; - if ((lp->upbo[varnr] < infinite) && (objfromvalue > lp->upbo[varnr])) - objfromvalue = lp->upbo[varnr]; - } - objfromvalue += lp->lowbo[varnr]; - objfromvalue = unscaled_value(lp, objfromvalue, varnr); - } - else - objfromvalue = -infinite; - lp->objfromvalue[varnr - lp->rows] = objfromvalue; - } - - } - FREE(pcol); - } - return((MYBOOL) ok); -} /* construct_sensitivity_duals */ - -/* Calculate sensitivity objective function */ -STATIC MYBOOL construct_sensitivity_obj(lprec *lp) -{ - int i, l, varnr, row_nr, ok = TRUE; - REAL *OrigObj = NULL, *drow = NULL, *prow = NULL, - sign, a, min1, min2, infinite, epsvalue, from, till; - - /* objective function */ - FREE(lp->objfrom); - FREE(lp->objtill); - if(!allocREAL(lp, &drow, lp->sum + 1, TRUE) || - !allocREAL(lp, &OrigObj, lp->columns + 1, FALSE) || - !allocREAL(lp, &prow, lp->sum + 1, TRUE) || - !allocREAL(lp, &lp->objfrom, lp->columns + 1, AUTOMATIC) || - !allocREAL(lp, &lp->objtill, lp->columns + 1, AUTOMATIC)) { -Abandon: - FREE(drow); - FREE(OrigObj); - FREE(prow); - FREE(lp->objfrom); - FREE(lp->objtill); - ok = FALSE; - } - else { - int *coltarget; - - infinite=lp->infinite; - epsvalue=lp->epsmachine; - - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->columns+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, SCAN_USERVARS+USE_NONBASICVARS, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - goto Abandon; - } - bsolve(lp, 0, drow, NULL, epsvalue*DOUBLEROUND, 1.0); - prod_xA(lp, coltarget, drow, NULL, epsvalue, 1.0, - drow, NULL, MAT_ROUNDDEFAULT | MAT_ROUNDRC); - - /* original (unscaled) objective function */ - get_row(lp, 0, OrigObj); - for(i = 1; i <= lp->columns; i++) { - from=-infinite; - till= infinite; - varnr = lp->rows + i; - if(!lp->is_basic[varnr]) { - /* only the coeff of the objective function of column i changes. */ - a = unscaled_mat(lp, drow[varnr], 0, i); - if(is_maxim(lp)) - a = -a; - if ((!sensrejvar) && (lp->upbo[varnr] == 0.0)) - /* ignore, because this case doesn't results in further iterations */ ; - else if(((lp->is_lower[varnr] != 0) == (is_maxim(lp) == FALSE)) && (a > -epsvalue)) - from = OrigObj[i] - a; /* less than this value gives further iterations */ - else - till = OrigObj[i] - a; /* bigger than this value gives further iterations */ - } - else { - /* all the coeff of the objective function change. Search the minimal change needed for further iterations */ - for(row_nr=1; - (row_nr<=lp->rows) && (lp->var_basic[row_nr]!=varnr); row_nr++) - /* Search on which row the variable exists in the basis */ ; - if(row_nr<=lp->rows) { /* safety test; should always be found ... */ - /* Construct one row of the tableau */ - bsolve(lp, row_nr, prow, NULL, epsvalue*DOUBLEROUND, 1.0); - prod_xA(lp, coltarget, prow, NULL, epsvalue, 1.0, - prow, NULL, MAT_ROUNDDEFAULT); - /* sign = my_chsign(is_chsign(lp, row_nr), -1); */ - sign = my_chsign(lp->is_lower[row_nr], -1); - min1=infinite; - min2=infinite; - for(l=1; l<=lp->sum; l++) /* search for the column(s) which first results in further iterations */ - if ((!lp->is_basic[l]) && (lp->upbo[l]>0.0) && - (fabs(prow[l])>epsvalue) && (drow[l]*(lp->is_lower[l] ? -1 : 1)is_lower[l] ? 1 : -1) < 0.0) { - if(a < min1) - min1 = a; - } - else { - if(a < min2) - min2 = a; - } - } - if ((lp->is_lower[varnr] == 0) == (is_maxim(lp) == FALSE)) { - a = min1; - min1 = min2; - min2 = a; - } - if (min1solution[varnr]; - if (is_maxim(lp)) { - if (a - lp->lowbo[varnr] < epsvalue) - from = -infinite; /* if variable is at lower bound then decrementing objective coefficient will not result in extra iterations because it would only extra decrease the value, but since it is at its lower bound ... */ - else if (((!sensrejvar) || (lp->upbo[varnr] != 0.0)) && (lp->lowbo[varnr] + lp->upbo[varnr] - a < epsvalue)) - till = infinite; /* if variable is at upper bound then incrementing objective coefficient will not result in extra iterations because it would only extra increase the value, but since it is at its upper bound ... */ - } - else { - if (a - lp->lowbo[varnr] < epsvalue) - till = infinite; /* if variable is at lower bound then incrementing objective coefficient will not result in extra iterations because it would only extra decrease the value, but since it is at its lower bound ... */ - else if (((!sensrejvar) || (lp->upbo[varnr] != 0.0)) && (lp->lowbo[varnr] + lp->upbo[varnr] - a < epsvalue)) - from = -infinite; /* if variable is at upper bound then decrementing objective coefficient will not result in extra iterations because it would only extra increase the value, but since it is at its upper bound ... */ - } - } - } - lp->objfrom[i]=from; - lp->objtill[i]=till; - } - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - } - FREE(prow); - FREE(OrigObj); - FREE(drow); - - return((MYBOOL) ok); -} /* construct_sensitivity_obj */ - -STATIC MYBOOL refactRecent(lprec *lp) -{ - int pivcount = lp->bfp_pivotcount(lp); - if(pivcount == 0) - return( AUTOMATIC ); - else if (pivcount < 2*DEF_MAXPIVOTRETRY) - return( TRUE ); - else - return( FALSE ); -} - -STATIC MYBOOL check_if_less(lprec *lp, REAL x, REAL y, int variable) -{ - if(y < x-scaled_value(lp, lp->epsint, variable)) { - if(lp->bb_trace) - report(lp, NORMAL, "check_if_less: Invalid new bound %g should be < %g for %s\n", - x, y, get_col_name(lp, variable)); - return(FALSE); - } - else - return(TRUE); -} - -/* Various basis utility routines */ - -STATIC int findNonBasicSlack(lprec *lp, MYBOOL *is_basic) -{ - int i; - - for(i = lp->rows; i > 0; i--) - if(!is_basic[i]) - break; - return( i ); -} - -STATIC int findBasisPos(lprec *lp, int notint, int *var_basic) -{ - int i; - - if(var_basic == NULL) - var_basic = lp->var_basic; - for(i = lp->rows; i > 0; i--) - if(var_basic[i] == notint) - break; - return( i ); -} - -STATIC void replaceBasisVar(lprec *lp, int rownr, int var, int *var_basic, MYBOOL *is_basic) -{ - int out; - - out = var_basic[rownr]; - var_basic[rownr] = var; - is_basic[out] = FALSE; - is_basic[var] = TRUE; -} - -STATIC void free_duals(lprec *lp) -{ - FREE(lp->duals); - FREE(lp->full_duals); - FREE(lp->dualsfrom); - FREE(lp->dualstill); - FREE(lp->objfromvalue); - FREE(lp->objfrom); - FREE(lp->objtill); -} - -/* Transform RHS by adjusting for the bound state of variables; - optionally rebase upper bound, and account for this in later calls */ -STATIC void initialize_solution(lprec *lp, MYBOOL shiftbounds) -{ - int i, k1, k2, *matRownr, colnr; - LREAL theta; - REAL value, *matValue, loB, upB; - MATrec *mat = lp->matA; - - /* Set bounding status indicators */ - if(lp->bb_bounds != NULL) { - if(shiftbounds == INITSOL_SHIFTZERO) { - if(lp->bb_bounds->UBzerobased) - report(lp, SEVERE, "initialize_solution: The upper bounds are already zero-based at refactorization %d\n", - lp->bfp_refactcount(lp, BFP_STAT_REFACT_TOTAL)); - lp->bb_bounds->UBzerobased = TRUE; - } - else if(!lp->bb_bounds->UBzerobased) - report(lp, SEVERE, "initialize_solution: The upper bounds are not zero-based at refactorization %d\n", - lp->bfp_refactcount(lp, BFP_STAT_REFACT_TOTAL)); - } - - /* Initialize the working RHS/basic variable solution vector */ - i = is_action(lp->anti_degen, ANTIDEGEN_RHSPERTURB) && (lp->monitor != NULL) && lp->monitor->active; - if(sizeof(*lp->rhs) == sizeof(*lp->orig_rhs) && !i) { - MEMCOPY(lp->rhs, lp->orig_rhs, lp->rows+1); - } - else if(i) { - lp->rhs[0] = lp->orig_rhs[0]; - for(i = 1; i <= lp->rows; i++) { - if(is_constr_type(lp, i, EQ)) - theta = rand_uniform(lp, lp->epsvalue); - else { - theta = rand_uniform(lp, lp->epsperturb); -/* if(lp->orig_upbo[i] < lp->infinite) - lp->orig_upbo[i] += theta; */ - } - lp->rhs[i] = lp->orig_rhs[i] + theta; - } - } - else - for(i = 0; i <= lp->rows; i++) - lp->rhs[i] = lp->orig_rhs[i]; - -/* Adjust active RHS for variables at their active upper/lower bounds */ - for(i = 1; i <= lp->sum; i++) { - - upB = lp->upbo[i]; - loB = lp->lowbo[i]; - - /* Shift to "ranged" upper bound, tantamount to defining zero-based variables */ - if(shiftbounds == INITSOL_SHIFTZERO) { - if((loB > -lp->infinite) && (upB < lp->infinite)) - lp->upbo[i] -= loB; - if(lp->upbo[i] < 0) - report(lp, SEVERE, "initialize_solution: Invalid rebounding; variable %d at refact %d, iter %.0f\n", - i, lp->bfp_refactcount(lp, BFP_STAT_REFACT_TOTAL), (double) get_total_iter(lp)); - } - - /* Use "ranged" upper bounds */ - else if(shiftbounds == INITSOL_USEZERO) { - if((loB > -lp->infinite) && (upB < lp->infinite)) - upB += loB; - } - - /* Shift upper bound back to original value */ - else if(shiftbounds == INITSOL_ORIGINAL) { - if((loB > -lp->infinite) && (upB < lp->infinite)) { - lp->upbo[i] += loB; - upB += loB; - } - continue; - } - else - report(lp, SEVERE, "initialize_solution: Invalid option value '%d'\n", - shiftbounds); - - /* Set the applicable adjustment */ - if(lp->is_lower[i]) - theta = loB; - else - theta = upB; - - - /* Check if we need to pass through the matrix; - remember that basis variables are always lower-bounded */ - if(theta == 0) - continue; - - /* Do user and artificial variables */ - if(i > lp->rows) { - - /* Get starting and ending indeces in the NZ vector */ - colnr = i - lp->rows; - k1 = mat->col_end[colnr - 1]; - k2 = mat->col_end[colnr]; - matRownr = &COL_MAT_ROWNR(k1); - matValue = &COL_MAT_VALUE(k1); - - /* Get the objective as row 0, optionally adjusting the objective for phase 1 */ - value = get_OF_active(lp, i, theta); - lp->rhs[0] -= value; - - /* Do the normal case */ - for(; k1 < k2; - k1++, matRownr += matRowColStep, matValue += matValueStep) { - lp->rhs[*matRownr] -= theta * (*matValue); - } - } - - /* Do slack variables (constraint "bounds")*/ - else { - lp->rhs[i] -= theta; - } - - } - - /* Do final pass to get the maximum value */ - i = idamax(lp->rows /* +1 */, lp->rhs, 1); - lp->rhsmax = fabs(lp->rhs[i]); - - if(shiftbounds == INITSOL_SHIFTZERO) - clear_action(&lp->spx_action, ACTION_REBASE); - -} - -/* This routine recomputes the basic variables using the full inverse */ -STATIC void recompute_solution(lprec *lp, MYBOOL shiftbounds) -{ - /* Compute RHS = b - A(n)*x(n) */ - initialize_solution(lp, shiftbounds); - - /* Compute x(b) = Inv(B)*RHS (Ref. lp_solve inverse logic and Chvatal p. 121) */ - lp->bfp_ftran_normal(lp, lp->rhs, NULL); - if(!lp->obj_in_basis) { - int i, ib, n = lp->rows; - for(i = 1; i <= n; i++) { - ib = lp->var_basic[i]; - if(ib > n) - lp->rhs[0] -= get_OF_active(lp, ib, lp->rhs[i]); - } - } - - /* Round the values (should not be greater than the factor used in bfp_pivotRHS) */ - roundVector(lp->rhs, lp->rows, lp->epsvalue); - - clear_action(&lp->spx_action, ACTION_RECOMPUTE); -} - -/* This routine compares an existing basic solution to a recomputed one; - Note that the routine must provide for the possibility that the order of the - basis variables can be changed by the inversion engine. */ -STATIC int verify_solution(lprec *lp, MYBOOL reinvert, char *info) -{ - int i, ii, n, *oldmap, *newmap, *refmap = NULL; - REAL *oldrhs, err, errmax; - - allocINT(lp, &oldmap, lp->rows+1, FALSE); - allocINT(lp, &newmap, lp->rows+1, FALSE); - allocREAL(lp, &oldrhs, lp->rows+1, FALSE); - - /* Get sorted mapping of the old basis */ - for(i = 0; i <= lp->rows; i++) - oldmap[i] = i; - if(reinvert) { - allocINT(lp, &refmap, lp->rows+1, FALSE); - MEMCOPY(refmap, lp->var_basic, lp->rows+1); - sortByINT(oldmap, refmap, lp->rows, 1, TRUE); - } - - /* Save old and calculate the new RHS vector */ - MEMCOPY(oldrhs, lp->rhs, lp->rows+1); - if(reinvert) - invert(lp, INITSOL_USEZERO, FALSE); - else - recompute_solution(lp, INITSOL_USEZERO); - - /* Get sorted mapping of the new basis */ - for(i = 0; i <= lp->rows; i++) - newmap[i] = i; - if(reinvert) { - MEMCOPY(refmap, lp->var_basic, lp->rows+1); - sortByINT(newmap, refmap, lp->rows, 1, TRUE); - } - - /* Identify any gap */ - errmax = 0; - ii = -1; - n = 0; - for(i = lp->rows; i > 0; i--) { - err = fabs(my_reldiff(oldrhs[oldmap[i]], lp->rhs[newmap[i]])); - if(err > lp->epsprimal) { - n++; - if(err > errmax) { - ii = i; - errmax = err; - } - } - } - err = fabs(my_reldiff(oldrhs[i], lp->rhs[i])); - if(err < lp->epspivot) { - i--; - err = 0; - } - else { - n++; - if(ii < 0) { - ii = 0; - errmax = err; - } - } - if(n > 0) { - report(lp, IMPORTANT, "verify_solution: Iter %.0f %s - %d errors; OF %g, Max @row %d %g\n", - (double) get_total_iter(lp), my_if(info == NULL, "", info), n, err, newmap[ii], errmax); - } - /* Copy old results back (not possible for inversion) */ - if(!reinvert) - MEMCOPY(lp->rhs, oldrhs, lp->rows+1); - - FREE(oldmap); - FREE(newmap); - FREE(oldrhs); - if(reinvert) - FREE(refmap); - - return( ii ); - -} - -/* Preprocessing and postprocessing functions */ -STATIC int identify_GUB(lprec *lp, MYBOOL mark) -{ - int i, j, jb, je, k, knint, srh; - REAL rh, mv, tv, bv; - MATrec *mat = lp->matA; - - if((lp->equalities == 0) || !mat_validate(mat)) - return( 0 ); - - k = 0; - for(i = 1; i <= lp->rows; i++) { - - /* Check if it is an equality constraint */ - if(!is_constr_type(lp, i, EQ)) - continue; - - rh = get_rh(lp, i); - srh = my_sign(rh); - knint = 0; - je = mat->row_end[i]; - for(jb = mat->row_end[i-1]; jb < je; jb++) { - j = ROW_MAT_COLNR(jb); - - /* Check for validity of the equation elements */ - if(!is_int(lp, j)) - knint++; - if(knint > 1) - break; - - mv = get_mat_byindex(lp, jb, TRUE, FALSE); - if(fabs(my_reldiff(mv, rh)) > lp->epsprimal) - break; - - tv = mv*get_upbo(lp, j); - bv = get_lowbo(lp, j); -#if 0 /* Requires 1 as upper bound */ - if((fabs(my_reldiff(tv, rh)) > lp->epsprimal) || (bv != 0)) -#else /* Can handle any upper bound >= 1 */ - if((srh*(tv-rh) < -lp->epsprimal) || (bv != 0)) -#endif - break; - } - - /* Update GUB count and optionally mark the GUB */ - if(jb == je) { - k++; - if(mark == TRUE) - lp->row_type[i] |= ROWTYPE_GUB; - else if(mark == AUTOMATIC) - break; - } - - } - return( k ); -} - -STATIC int prepare_GUB(lprec *lp) -{ - int i, j, jb, je, k, *members = NULL; - REAL rh; - char GUBname[16]; - MATrec *mat = lp->matA; - - if((lp->equalities == 0) || - !allocINT(lp, &members, lp->columns+1, TRUE) || - !mat_validate(mat)) - return( 0 ); - - for(i = 1; i <= lp->rows; i++) { - - /* Check if it has been marked as a GUB */ - if(!(lp->row_type[i] & ROWTYPE_GUB)) - continue; - - /* Pick up the GUB column indeces */ - k = 0; - je = mat->row_end[i]; - for(jb = mat->row_end[i-1], k = 0; jb < je; jb++) { - members[k] = ROW_MAT_COLNR(jb); - k++; - } - - /* Add the GUB */ - j = GUB_count(lp) + 1; - sprintf(GUBname, "GUB_%d", i); - add_GUB(lp, GUBname, j, k, members); - - /* Unmark the GUBs */ - clear_action(&(lp->row_type[i]), ROWTYPE_GUB); - - /* Standardize coefficients to 1 if necessary */ - rh = get_rh(lp, i); - if(fabs(my_reldiff(rh, 1)) > lp->epsprimal) { - set_rh(lp, i, 1); - for(jb = mat->row_end[i-1]; jb < je; jb++) { - j = ROW_MAT_COLNR(jb); - set_mat(lp, i,j, 1); - } - } - - } - FREE(members); - return(GUB_count(lp)); -} - -/* Pre- and post processing functions, i.a. splitting free variables */ -STATIC MYBOOL pre_MIPOBJ(lprec *lp) -{ -#ifdef MIPboundWithOF - if(MIP_count(lp) > 0) { - int i = 1; - while((i <= lp->rows) && !mat_equalRows(lp->matA, 0, i) && !is_constr_type(lp, i, EQ)) - i++; - if(i <= lp->rows) - lp->constraintOF = i; - } -#endif - lp->bb_deltaOF = MIP_stepOF(lp); - if(lp->bb_deltaOF < MAX(lp->epsvalue, lp->mip_absgap)) - lp->bb_deltaOF = 0; - - return( TRUE ); -} -STATIC MYBOOL post_MIPOBJ(lprec *lp) -{ -#ifdef MIPboundWithOF -/* - if(lp->constraintOF) { - del_constraint(lp, lp->rows); - if(is_BasisReady(lp) && !verify_basis(lp)) - return( FALSE ); - } -*/ -#endif - return( TRUE ); -} - -int preprocess(lprec *lp) -{ - int i, j, k, ok = TRUE, *new_index = NULL; - REAL hold, *new_column = NULL; - MYBOOL scaled, primal1, primal2; - - /* do not process if already preprocessed */ - if(lp->wasPreprocessed) - return( ok ); - - /* Write model statistics and optionally initialize partial pricing structures */ - if(lp->lag_status != RUNNING) { - MYBOOL doPP; - - /* Extract the user-specified simplex strategy choices */ - primal1 = (MYBOOL) (lp->simplex_strategy & SIMPLEX_Phase1_PRIMAL); - primal2 = (MYBOOL) (lp->simplex_strategy & SIMPLEX_Phase2_PRIMAL); - - /* Initialize partial pricing structures */ - doPP = is_piv_mode(lp, PRICE_PARTIAL | PRICE_AUTOPARTIAL); -/* doPP &= (MYBOOL) (lp->columns / 2 > lp->rows); */ - if(doPP) { - i = partial_findBlocks(lp, FALSE, FALSE); - if(i < 4) - i = (int) (5 * log((REAL) lp->columns / lp->rows)); - report(lp, NORMAL, "The model is %s to have %d column blocks/stages.\n", - (i > 1 ? "estimated" : "set"), i); - set_partialprice(lp, i, NULL, FALSE); - } -/* doPP &= (MYBOOL) (lp->rows / 4 > lp->columns); */ - if(doPP) { - i = partial_findBlocks(lp, FALSE, TRUE); - if(i < 4) - i = (int) (5 * log((REAL) lp->rows / lp->columns)); - report(lp, NORMAL, "The model is %s to have %d row blocks/stages.\n", - (i > 1 ? "estimated" : "set"), i); - set_partialprice(lp, i, NULL, TRUE); - } - - /* Check for presence of valid pricing blocks if partial pricing - is defined, but not autopartial is not set */ - if(!doPP && is_piv_mode(lp, PRICE_PARTIAL)) { - if((lp->rowblocks == NULL) || (lp->colblocks == NULL)) { - report(lp, IMPORTANT, "Ignoring partial pricing, since block structures are not defined.\n"); - clear_action(&lp->piv_strategy, PRICE_PARTIAL); - } - } - - /* Initialize multiple pricing block divisor */ -#if 0 - if(primal1 || primal2) - lp->piv_strategy |= PRICE_MULTIPLE | PRICE_AUTOMULTIPLE; -#endif - if(is_piv_mode(lp, PRICE_MULTIPLE) && (primal1 || primal2)) { - doPP = is_piv_mode(lp, PRICE_AUTOMULTIPLE); - if(doPP) { - i = (int) (2.5*log((REAL) lp->sum)); - SETMAX( i, 1); - set_multiprice(lp, i); - } - if(lp->multiblockdiv > 1) - report(lp, NORMAL, "Using %d-candidate primal simplex multiple pricing block.\n", - lp->columns / lp->multiblockdiv); - } - else - set_multiprice(lp, 1); - - report(lp, NORMAL, "Using %s simplex for phase 1 and %s simplex for phase 2.\n", - my_if(primal1, "PRIMAL", "DUAL"), my_if(primal2, "PRIMAL", "DUAL")); - i = get_piv_rule(lp); - if((i == PRICER_STEEPESTEDGE) && is_piv_mode(lp, PRICE_PRIMALFALLBACK)) - report(lp, NORMAL, "The pricing strategy is set to '%s' for the dual and '%s' for the primal.\n", - get_str_piv_rule(i), get_str_piv_rule(i-1)); - else - report(lp, NORMAL, "The primal and dual simplex pricing strategy set to '%s'.\n", - get_str_piv_rule(i)); - - report(lp, NORMAL, " \n"); - } - - /* Compute a minimum step improvement step requirement */ - pre_MIPOBJ(lp); - - /* First create extra columns for FR variables or flip MI variables */ - for (j = 1; j <= lp->columns; j++) { - -#ifdef Paranoia - if((lp->rows != lp->matA->rows) || (lp->columns != lp->matA->columns)) - report(lp, SEVERE, "preprocess: Inconsistent variable counts found\n"); -#endif - - /* First handle sign-flipping of variables: - 1) ... with a finite upper bound and a negative Inf-bound (since basis variables are lower-bounded) - 2) ... with bound assymetry within negrange limits (for stability reasons) */ - i = lp->rows + j; - hold = lp->orig_upbo[i]; -/* - if((hold <= 0) || (!is_infinite(lp, lp->negrange) && - (hold < -lp->negrange) && - (lp->orig_lowbo[i] <= lp->negrange)) ) { -*/ -#define fullybounded FALSE - if( ((hold < lp->infinite) && my_infinite(lp, lp->orig_lowbo[i])) || - (!fullybounded && !my_infinite(lp, lp->negrange) && - (hold < -lp->negrange) && (lp->orig_lowbo[i] <= lp->negrange)) ) { - /* Delete split sibling variable if one existed from before */ - if((lp->var_is_free != NULL) && (lp->var_is_free[j] > 0)) - del_column(lp, lp->var_is_free[j]); - /* Negate the column / flip to the positive range */ - mat_multcol(lp->matA, j, -1, TRUE); - if(lp->var_is_free == NULL) { - if(!allocINT(lp, &lp->var_is_free, MAX(lp->columns, lp->columns_alloc) + 1, TRUE)) - return(FALSE); - } - lp->var_is_free[j] = -j; /* Indicator UB and LB are switched, with no helper variable added */ - lp->orig_upbo[i] = my_flipsign(lp->orig_lowbo[i]); - lp->orig_lowbo[i] = my_flipsign(hold); - /* Check for presence of negative ranged SC variable */ - if(lp->sc_lobound[j] > 0) { - lp->sc_lobound[j] = lp->orig_lowbo[i]; - lp->orig_lowbo[i] = 0; - } - } - /* Then deal with -+, full-range/FREE variables by creating a helper variable */ - else if((lp->orig_lowbo[i] <= lp->negrange) && (hold >= -lp->negrange)) { - if(lp->var_is_free == NULL) { - if(!allocINT(lp, &lp->var_is_free, MAX(lp->columns,lp->columns_alloc) + 1, TRUE)) - return(FALSE); - } - if(lp->var_is_free[j] <= 0) { /* If this variable wasn't split yet ... */ - if(SOS_is_member(lp->SOS, 0, i - lp->rows)) { /* Added */ - report(lp, IMPORTANT, "preprocess: Converted negative bound for SOS variable %d to zero", - i - lp->rows); - lp->orig_lowbo[i] = 0; - continue; - } - if(new_column == NULL) { - if(!allocREAL(lp, &new_column, lp->rows + 1, FALSE) || - !allocINT(lp, &new_index, lp->rows + 1, FALSE)) { - ok = FALSE; - break; - } - } - /* Avoid precision loss by turning off unscaling and rescaling */ - /* in get_column and add_column operations; also make sure that */ - /* full scaling information is preserved */ - scaled = lp->scaling_used; - lp->scaling_used = FALSE; - k = get_columnex(lp, j, new_column, new_index); - if(!add_columnex(lp, k, new_column, new_index)) { - ok = FALSE; - break; - } - mat_multcol(lp->matA, lp->columns, -1, TRUE); - if(scaled) - lp->scalars[lp->rows+lp->columns] = lp->scalars[i]; - lp->scaling_used = (MYBOOL) scaled; - /* Only create name if we are not clearing a pre-used item, since this - variable could have been deleted by presolve but the name is required - for solution reconstruction. */ - if(lp->names_used && (lp->col_name[j] == NULL)) { - char fieldn[50]; - - sprintf(fieldn, "__AntiBodyOf(%d)__", j); - if(!set_col_name(lp, lp->columns, fieldn)) { -/* if (!set_col_name(lp, lp->columns, get_col_name(lp, j))) { */ - ok = FALSE; - break; - } - } - /* Set (positive) index to the original column's split / helper and back */ - lp->var_is_free[j] = lp->columns; - } - lp->orig_upbo[lp->rows + lp->var_is_free[j]] = my_flipsign(lp->orig_lowbo[i]); - lp->orig_lowbo[i] = 0; - - /* Negative index indicates x is split var and -var_is_free[x] is index of orig var */ - lp->var_is_free[lp->var_is_free[j]] = -j; - lp->var_type[lp->var_is_free[j]] = lp->var_type[j]; - } - /* Check for positive ranged SC variables */ - else if(lp->sc_lobound[j] > 0) { - lp->sc_lobound[j] = lp->orig_lowbo[i]; - lp->orig_lowbo[i] = 0; - } - - /* Tally integer variables in SOS'es */ - if(SOS_is_member(lp->SOS, 0, j) && is_int(lp, j)) - lp->sos_ints++; - } - - FREE(new_column); - FREE(new_index); - - /* Fill lists of GUB constraints, if appropriate */ - if((MIP_count(lp) > 0) && is_bb_mode(lp, NODE_GUBMODE) && (identify_GUB(lp, AUTOMATIC) > 0)) - prepare_GUB(lp); - - /* (Re)allocate reduced cost arrays */ - ok = allocREAL(lp, &(lp->drow), lp->sum+1, AUTOMATIC) && - allocINT(lp, &(lp->nzdrow), lp->sum+1, AUTOMATIC); - if(ok) - lp->nzdrow[0] = 0; - - /* Minimize memory usage */ - memopt_lp(lp, 0, 0, 0); - - lp->wasPreprocessed = TRUE; - - return(ok); -} - -void postprocess(lprec *lp) -{ - int i,ii,j; - REAL hold; - - /* Check if the problem actually was preprocessed */ - if(!lp->wasPreprocessed) - return; - - /* Must compute duals here in case we have free variables; note that in - this case sensitivity analysis is not possible unless done here */ - if((lp->bb_totalnodes == 0) && (lp->var_is_free == NULL)) { - if(is_presolve(lp, PRESOLVE_DUALS)) - construct_duals(lp); - if(is_presolve(lp, PRESOLVE_SENSDUALS)) - if(!construct_sensitivity_duals(lp) || !construct_sensitivity_obj(lp)) - report(lp, IMPORTANT, "postprocess: Unable to allocate working memory for duals.\n"); - } - - /* Loop over all columns */ - for (j = 1; j <= lp->columns; j++) { - i = lp->rows + j; - /* Reconstruct strictly negative values */ - if((lp->var_is_free != NULL) && (lp->var_is_free[j] < 0)) { - /* Check if we have the simple case where the UP and LB are negated and switched */ - if(-lp->var_is_free[j] == j) { - mat_multcol(lp->matA, j, -1, TRUE); - hold = lp->orig_upbo[i]; - lp->orig_upbo[i] = my_flipsign(lp->orig_lowbo[i]); - lp->orig_lowbo[i] = my_flipsign(hold); - lp->best_solution[i] = my_flipsign(lp->best_solution[i]); - transfer_solution_var(lp, j); - - /* hold = lp->objfrom[j]; - lp->objfrom[j] = my_flipsign(lp->objtill[j]); - lp->objtill[j] = my_flipsign(hold); */ /* under investigation */ - - /* lp->duals[i] = my_flipsign(lp->duals[i]); - hold = lp->dualsfrom[i]; - lp->dualsfrom[i] = my_flipsign(lp->dualstill[i]); - lp->dualstill[i] = my_flipsign(hold); */ /* under investigation */ - /* Bound switch undone, so clear the status */ - lp->var_is_free[j] = 0; - /* Adjust negative ranged SC */ - if(lp->sc_lobound[j] > 0) - lp->orig_lowbo[lp->rows + j] = -lp->sc_lobound[j]; - } - /* Ignore the split / helper columns (will be deleted later) */ - } - /* Condense values of extra columns of quasi-free variables split in two */ - else if((lp->var_is_free != NULL) && (lp->var_is_free[j] > 0)) { - ii = lp->var_is_free[j]; /* Index of the split helper var */ - /* if(lp->objfrom[j] == -lp->infinite) - lp->objfrom[j] = -lp->objtill[ii]; - lp->objtill[ii] = lp->infinite; - if(lp->objtill[j] == lp->infinite) - lp->objtill[j] = my_flipsign(lp->objfrom[ii]); - lp->objfrom[ii] = -lp->infinite; */ /* under investigation */ - - ii += lp->rows; - lp->best_solution[i] -= lp->best_solution[ii]; /* join the solution again */ - transfer_solution_var(lp, j); - lp->best_solution[ii] = 0; - - /* if(lp->duals[i] == 0) - lp->duals[i] = my_flipsign(lp->duals[ii]); - lp->duals[ii] = 0; - if(lp->dualsfrom[i] == -lp->infinite) - lp->dualsfrom[i] = my_flipsign(lp->dualstill[ii]); - lp->dualstill[ii] = lp->infinite; - if(lp->dualstill[i] == lp->infinite) - lp->dualstill[i] = my_flipsign(lp->dualsfrom[ii]); - lp->dualsfrom[ii] = -lp->infinite; */ /* under investigation */ - - /* Reset to original bound */ - lp->orig_lowbo[i] = my_flipsign(lp->orig_upbo[ii]); - } - /* Adjust for semi-continuous variables */ - else if(lp->sc_lobound[j] > 0) { - lp->orig_lowbo[i] = lp->sc_lobound[j]; - } - } - - /* Remove any split column helper variables */ - del_splitvars(lp); - post_MIPOBJ(lp); - - /* Do extended reporting, if specified */ - if(lp->verbose > NORMAL) { - REPORT_extended(lp); - - } - - lp->wasPreprocessed = FALSE; -} - diff --git a/code/3rd_lpsolve/lp_lib.h b/code/3rd_lpsolve/lp_lib.h deleted file mode 100644 index 13f96e10..00000000 --- a/code/3rd_lpsolve/lp_lib.h +++ /dev/null @@ -1,2303 +0,0 @@ - -#ifndef HEADER_lp_lib -#define HEADER_lp_lib - -/* -------------------------------------------------------------------------- - - This is the main library header file for the lp_solve v5.0 release - - Starting at version 3.0, LP_Solve is released under the LGPL license. - For full information, see the enclosed file LGPL.txt. - - Original developer: Michel Berkelaar - michel@ics.ele.tue.nl - Most changes 1.5-2.0: Jeroen Dirks - jeroend@tor.numetrix.com - Changes 3.2-4.0: Kjell Eikland - kjell.eikland@broadpark.no - (Simplex code, SOS, SC, code optimization) - Peter Notebaert - lpsolve@peno.be - (Sensitivity analysis, documentation) - Changes 5.0+: Kjell Eikland - kjell.eikland@broadpark.no - (BFP, XLI, simplex, B&B, code modularization) - Peter Notebaert - lpsolve@peno.be - (Sensitivity analysis, New lp parser, LINDO (XLI) - parser, VB/.NET interface, documentation) - - Release notes: - - Version 4.0 enhances version 3.2 in terms of internal program/simplex - architecture, call level interfaces, data layout, features and contains - several bug fixes. There is now complete support for semi-continuous - variables and SOS constructions. In the process, a complete API - was added. The MPS parser has been amended to support this. - Sensitivity analysis and variouse bug fixes was provided by Peter - Notebaert in 4.0 sub-releases. Peter also wrote a complete - documentation of the API and contributed a VB interface, both of which - significantly enhanced the accessibility of lp_solve. - - Version 5.0 is a major rewrite and code cleanup. The main additions that - drove forward this cleanup were the modular inversion logic with optimal - column ordering, addition of primal phase 1 and dual phase 2 logic for - full flexibility in the selection of primal and dual simplex modes, - DEVEX and steepest edge pivot selection, along with dynamic cycling - detection and prevention. This cleanup made it possible to harmonize the - internal rounding principles, contributing to increased numerical stability. - - Version 5.1 rearranges the matrix storage model by enabling both legacy - element record-based storage and split vector storage. In addition the - lprec structure is optimized and additional routines are added, mainly for - sparse vector additions and enhanced XLI functionality. Support for XML- - based models was added on the basis of the LPFML schema via xli_LPFML. - - Version 5.2 removes the objective function from the constraint matrix, - adds a number of presolve options and speed them up. Degeneracy handling - is significantly improved. Support for XLI_ZIMPL was added. - Multiple and partial pricing has been enhanced and activated. - - -------------------------------------------------------------------------- */ -/* Define user program feature option switches */ -/* ------------------------------------------------------------------------- */ - -# if defined _WIN32 && !defined __GNUC__ -# define isnan _isnan -# endif -#if defined NOISNAN -# define isnan(x) FALSE -#endif - -#define SETMASK(variable, mask) variable |= mask -#define CLEARMASK(variable, mask) variable &= ~(mask) -#define TOGGLEMASK(variable, mask) variable ^= mask -#define ISMASKSET(variable, mask) (MYBOOL) (((variable) & (mask)) != 0) - -/* Utility/system settings */ -/* ------------------------------------------------------------------------- */ -/*#define INTEGERTIME */ /* Set use of lower-resolution timer */ - - -/* New v5.0+ simplex/optimization features and settings */ -/* ------------------------------------------------------------------------- */ -/*#define NoRowScaleOF */ /* Optionally skip row-scaling of the OF */ -#define DoMatrixRounding /* Round A matrix elements to precision */ -#define DoBorderRounding /* Round RHS, bounds and ranges to precision */ -#define Phase1EliminateRedundant /* Remove rows of redundant artificials */ -#define FixViolatedOptimal -#define ImproveSolutionPrecision /* Round optimal solution values */ -/*#define IncreasePivotOnReducedAccuracy */ /* Increase epspivot on instability */ -/*#define FixInaccurateDualMinit */ /* Reinvert on inaccuracy in dual minits */ -/*#define EnforcePositiveTheta */ /* Ensure that the theta range is valid */ -#define ResetMinitOnReinvert -/*#define UsePrimalReducedCostUpdate */ /* Not tested */ -/*#define UseDualReducedCostUpdate */ /* Seems Ok, but slower than expected */ -/*#ifdef UseLegacyExtrad */ /* Use v3.2- style Extrad method */ -#define UseMilpExpandedRCF /* Non-ints in reduced cost bound tightening */ -/*#define UseMilpSlacksRCF */ /* Slacks in reduced cost bound tightening (degen - prone); requires !SlackInitMinusInf */ -#define LegacySlackDefinition /* Slack as the "value of the constraint" */ - - -/* Development features (change at own risk) */ -/* ------------------------------------------------------------------------- */ -/*#define MIPboundWithOF */ /* Enable to detect OF constraint for use during B&B */ -/*#define SlackInitMinusInf */ /* Slacks have 0 LB if this is not defined */ -#define FULLYBOUNDEDSIMPLEX FALSE /* WARNING: Activate at your own risk! */ - - -/* Specify use of the basic linear algebra subroutine library */ -/* ------------------------------------------------------------------------- */ -#define libBLAS 2 /* 0: No, 1: Internal, 2: External */ -#define libnameBLAS "myBLAS" - - -/* Active inverse logic (default is optimized original etaPFI) */ -/* ------------------------------------------------------------------------- */ -#if !defined LoadInverseLib -# define LoadInverseLib TRUE /* Enable alternate inverse libraries */ -#endif -/*#define ExcludeNativeInverse */ /* Disable INVERSE_ACTIVE inverse engine */ - -#define DEF_OBJINBASIS TRUE /* Additional rows inserted at the top (1 => OF) */ - -#define INVERSE_NONE -1 -#define INVERSE_LEGACY 0 -#define INVERSE_ETAPFI 1 -#define INVERSE_LUMOD 2 -#define INVERSE_LUSOL 3 -#define INVERSE_GLPKLU 4 - -#ifndef RoleIsExternalInvEngine /* Defined in inverse DLL drivers */ - #ifdef ExcludeNativeInverse - #define INVERSE_ACTIVE INVERSE_NONE /* Disable native engine */ - #else - #define INVERSE_ACTIVE INVERSE_LEGACY /* User or DLL-selected */ - #endif -#endif - - -/* Active external language interface logic (default is none) */ -/* ------------------------------------------------------------------------- */ -#if !defined LoadLanguageLib -# define LoadLanguageLib TRUE /* Enable alternate language libraries */ -#endif -#define ExcludeNativeLanguage /* Disable LANGUAGE_ACTIVE XLI */ - -#define LANGUAGE_NONE -1 -#define LANGUAGE_LEGACYLP 0 -#define LANGUAGE_CPLEXLP 1 -#define LANGUAGE_MPSX 2 -#define LANGUAGE_LPFML 3 -#define LANGUAGE_MATHPROG 4 -#define LANGUAGE_AMPL 5 -#define LANGUAGE_GAMS 6 -#define LANGUAGE_ZIMPL 7 -#define LANGUAGE_S 8 -#define LANGUAGE_R 9 -#define LANGUAGE_MATLAB 10 -#define LANGUAGE_OMATRIX 11 -#define LANGUAGE_SCILAB 12 -#define LANGUAGE_OCTAVE 13 -#define LANGUAGE_EMPS 14 - -#ifndef RoleIsExternalLanguageEngine /* Defined in XLI driver libraries */ - #ifdef ExcludeNativeLanguage - #define LANGUAGE_ACTIVE LANGUAGE_NONE /* Disable native engine */ - #else - #define LANGUAGE_ACTIVE LANGUAGE_CPLEXLP /* User or DLL-selected */ - #endif -#endif - - -/* Default parameters and tolerances */ -/* ------------------------------------------------------------------------- */ -#define OriginalPARAM 0 -#define ProductionPARAM 1 -#define ChvatalPARAM 2 -#define LoosePARAM 3 -#if 1 - #define ActivePARAM ProductionPARAM -#else - #define ActivePARAM LoosePARAM -#endif - - -/* Miscellaneous settings */ -/* ------------------------------------------------------------------------- */ -#ifndef Paranoia - #ifdef _DEBUG - #define Paranoia - #endif -#endif - - -/* Program version data */ -/* ------------------------------------------------------------------------- */ -#define MAJORVERSION 5 -#define MINORVERSION 5 -#define RELEASE 2 -#define BUILD 5 -#define BFPVERSION 12 /* Checked against bfp_compatible() */ -#define XLIVERSION 12 /* Checked against xli_compatible() */ -/* Note that both BFPVERSION and XLIVERSION typically have to be incremented - in the case that the lprec structure changes. */ - - -/* Include/header files */ -/* ------------------------------------------------------------------------- */ -#include -#include -#include -#include -#include - -#include "lp_types.h" -#include "lp_utils.h" - -#if (LoadInverseLib == TRUE) || (LoadLanguageLib == TRUE) - #ifdef WIN32 - #include - #else - #include - #endif -#endif - -#ifndef BFP_CALLMODEL - #ifdef WIN32 - #define BFP_CALLMODEL __stdcall /* "Standard" call model */ - #else - #define BFP_CALLMODEL - #endif -#endif -#ifndef XLI_CALLMODEL - #define XLI_CALLMODEL BFP_CALLMODEL -#endif - -#define REGISTER register /* Speed up certain operations */ - - -/* Definition of program constrants */ -/* ------------------------------------------------------------------------- */ -#define SIMPLEX_UNDEFINED 0 -#define SIMPLEX_Phase1_PRIMAL 1 -#define SIMPLEX_Phase1_DUAL 2 -#define SIMPLEX_Phase2_PRIMAL 4 -#define SIMPLEX_Phase2_DUAL 8 -#define SIMPLEX_DYNAMIC 16 -#define SIMPLEX_AUTODUALIZE 32 - -#define SIMPLEX_PRIMAL_PRIMAL (SIMPLEX_Phase1_PRIMAL + SIMPLEX_Phase2_PRIMAL) -#define SIMPLEX_DUAL_PRIMAL (SIMPLEX_Phase1_DUAL + SIMPLEX_Phase2_PRIMAL) -#define SIMPLEX_PRIMAL_DUAL (SIMPLEX_Phase1_PRIMAL + SIMPLEX_Phase2_DUAL) -#define SIMPLEX_DUAL_DUAL (SIMPLEX_Phase1_DUAL + SIMPLEX_Phase2_DUAL) -#define SIMPLEX_DEFAULT (SIMPLEX_DUAL_PRIMAL) - -/* Variable codes (internal) */ -#define ISREAL 0 -#define ISINTEGER 1 -#define ISSEMI 2 -#define ISSOS 4 -#define ISSOSTEMPINT 8 -#define ISGUB 16 - -/* Presolve defines */ -#define PRESOLVE_NONE 0 -#define PRESOLVE_ROWS 1 -#define PRESOLVE_COLS 2 -#define PRESOLVE_LINDEP 4 -#define PRESOLVE_AGGREGATE 8 /* Not implemented */ -#define PRESOLVE_SPARSER 16 /* Not implemented */ -#define PRESOLVE_SOS 32 -#define PRESOLVE_REDUCEMIP 64 -#define PRESOLVE_KNAPSACK 128 /* Implementation not tested completely */ -#define PRESOLVE_ELIMEQ2 256 -#define PRESOLVE_IMPLIEDFREE 512 -#define PRESOLVE_REDUCEGCD 1024 -#define PRESOLVE_PROBEFIX 2048 -#define PRESOLVE_PROBEREDUCE 4096 -#define PRESOLVE_ROWDOMINATE 8192 -#define PRESOLVE_COLDOMINATE 16384 /* Reduced functionality, should be expanded */ -#define PRESOLVE_MERGEROWS 32768 -#define PRESOLVE_IMPLIEDSLK 65536 -#define PRESOLVE_COLFIXDUAL 131072 -#define PRESOLVE_BOUNDS 262144 -#define PRESOLVE_LASTMASKMODE (PRESOLVE_DUALS - 1) -#define PRESOLVE_DUALS 524288 -#define PRESOLVE_SENSDUALS 1048576 - -/* Basis crash options */ -#define CRASH_NONE 0 -#define CRASH_NONBASICBOUNDS 1 -#define CRASH_MOSTFEASIBLE 2 -#define CRASH_LEASTDEGENERATE 3 - -/* Solution recomputation options (internal) */ -#define INITSOL_SHIFTZERO 0 -#define INITSOL_USEZERO 1 -#define INITSOL_ORIGINAL 2 - -/* Strategy codes to avoid or recover from degenerate pivots, - infeasibility or numeric errors via randomized bound relaxation */ -#define ANTIDEGEN_NONE 0 -#define ANTIDEGEN_FIXEDVARS 1 -#define ANTIDEGEN_COLUMNCHECK 2 -#define ANTIDEGEN_STALLING 4 -#define ANTIDEGEN_NUMFAILURE 8 -#define ANTIDEGEN_LOSTFEAS 16 -#define ANTIDEGEN_INFEASIBLE 32 -#define ANTIDEGEN_DYNAMIC 64 -#define ANTIDEGEN_DURINGBB 128 -#define ANTIDEGEN_RHSPERTURB 256 -#define ANTIDEGEN_BOUNDFLIP 512 -#define ANTIDEGEN_DEFAULT (ANTIDEGEN_FIXEDVARS | ANTIDEGEN_STALLING /* | ANTIDEGEN_INFEASIBLE */) - -/* REPORT defines */ -#define NEUTRAL 0 -#define CRITICAL 1 -#define SEVERE 2 -#define IMPORTANT 3 -#define NORMAL 4 -#define DETAILED 5 -#define FULL 6 - -/* MESSAGE defines */ -#define MSG_NONE 0 -#define MSG_PRESOLVE 1 -#define MSG_ITERATION 2 -#define MSG_INVERT 4 -#define MSG_LPFEASIBLE 8 -#define MSG_LPOPTIMAL 16 -#define MSG_LPEQUAL 32 -#define MSG_LPBETTER 64 -#define MSG_MILPFEASIBLE 128 -#define MSG_MILPEQUAL 256 -#define MSG_MILPBETTER 512 -#define MSG_MILPSTRATEGY 1024 -#define MSG_MILPOPTIMAL 2048 -#define MSG_PERFORMANCE 4096 -#define MSG_INITPSEUDOCOST 8192 - -/* MPS file types */ -#define MPSFIXED 1 -#define MPSFREE 2 -#define MPSIBM 4 -#define MPSNEGOBJCONST 8 - -#define MPS_FREE (MPSFREE<<2) -#define MPS_IBM (MPSIBM<<2) -#define MPS_NEGOBJCONST (MPSNEGOBJCONST<<2) - -/* MPS defines (internal) */ -#define MPSUNDEF -4 -#define MPSNAME -3 -#define MPSOBJSENSE -2 -#define MPSOBJNAME -1 -#define MPSROWS 0 -#define MPSCOLUMNS 1 -#define MPSRHS 2 -#define MPSBOUNDS 3 -#define MPSRANGES 4 -#define MPSSOS 5 - -#define MPSVARMASK "%-8s" -#define MPSVALUEMASK "%12g" - -/* Constraint type codes (internal) */ -#define ROWTYPE_EMPTY 0 -#define ROWTYPE_LE 1 -#define ROWTYPE_GE 2 -#define ROWTYPE_EQ 3 -#define ROWTYPE_CONSTRAINT ROWTYPE_EQ /* This is the mask for modes */ -#define ROWTYPE_OF 4 -#define ROWTYPE_INACTIVE 8 -#define ROWTYPE_RELAX 16 -#define ROWTYPE_GUB 32 -#define ROWTYPE_OFMAX (ROWTYPE_OF + ROWTYPE_GE) -#define ROWTYPE_OFMIN (ROWTYPE_OF + ROWTYPE_LE) -#define ROWTYPE_CHSIGN ROWTYPE_GE - -/* Public constraint codes */ -#define FR ROWTYPE_EMPTY -#define LE ROWTYPE_LE -#define GE ROWTYPE_GE -#define EQ ROWTYPE_EQ -#define OF ROWTYPE_OF - -/* MIP constraint classes */ -#define ROWCLASS_Unknown 0 /* Undefined/unknown */ -#define ROWCLASS_Objective 1 /* The objective function */ -#define ROWCLASS_GeneralREAL 2 /* General real-values constraint */ -#define ROWCLASS_GeneralMIP 3 /* General mixed integer/binary and real valued constraint */ -#define ROWCLASS_GeneralINT 4 /* General integer-only constraint */ -#define ROWCLASS_GeneralBIN 5 /* General binary-only constraint */ -#define ROWCLASS_KnapsackINT 6 /* Sum of positive integer times integer variables <= positive integer */ -#define ROWCLASS_KnapsackBIN 7 /* Sum of positive integer times binary variables <= positive integer */ -#define ROWCLASS_SetPacking 8 /* Sum of binary variables >= 1 */ -#define ROWCLASS_SetCover 9 /* Sum of binary variables <= 1 */ -#define ROWCLASS_GUB 10 /* Sum of binary variables = 1 */ -#define ROWCLASS_MAX ROWCLASS_GUB - -/* Column subsets (internal) */ -#define SCAN_USERVARS 1 -#define SCAN_SLACKVARS 2 -#define SCAN_ARTIFICIALVARS 4 -#define SCAN_PARTIALBLOCK 8 -#define USE_BASICVARS 16 -#define USE_NONBASICVARS 32 -#define SCAN_NORMALVARS (SCAN_USERVARS + SCAN_ARTIFICIALVARS) -#define SCAN_ALLVARS (SCAN_SLACKVARS + SCAN_USERVARS + SCAN_ARTIFICIALVARS) -#define USE_ALLVARS (USE_BASICVARS + USE_NONBASICVARS) -#define OMIT_FIXED 64 -#define OMIT_NONFIXED 128 - -/* Improvement defines */ -#define IMPROVE_NONE 0 -#define IMPROVE_SOLUTION 1 -#define IMPROVE_DUALFEAS 2 -#define IMPROVE_THETAGAP 4 -#define IMPROVE_BBSIMPLEX 8 -#define IMPROVE_DEFAULT (IMPROVE_DUALFEAS + IMPROVE_THETAGAP) -#define IMPROVE_INVERSE (IMPROVE_SOLUTION + IMPROVE_THETAGAP) - -/* Scaling types */ -#define SCALE_NONE 0 -#define SCALE_EXTREME 1 -#define SCALE_RANGE 2 -#define SCALE_MEAN 3 -#define SCALE_GEOMETRIC 4 -#define SCALE_FUTURE1 5 -#define SCALE_FUTURE2 6 -#define SCALE_CURTISREID 7 /* Override to Curtis-Reid "optimal" scaling */ - -/* Alternative scaling weights */ -#define SCALE_LINEAR 0 -#define SCALE_QUADRATIC 8 -#define SCALE_LOGARITHMIC 16 -#define SCALE_USERWEIGHT 31 -#define SCALE_MAXTYPE (SCALE_QUADRATIC-1) - -/* Scaling modes */ -#define SCALE_POWER2 32 /* As is or rounded to power of 2 */ -#define SCALE_EQUILIBRATE 64 /* Make sure that no scaled number is above 1 */ -#define SCALE_INTEGERS 128 /* Apply to integer columns/variables */ -#define SCALE_DYNUPDATE 256 /* Apply incrementally every solve() */ -#define SCALE_ROWSONLY 512 /* Override any scaling to only scale the rows */ -#define SCALE_COLSONLY 1024 /* Override any scaling to only scale the rows */ - -/* Standard defines for typical scaling models (no Lagrangeans) */ -#define SCALEMODEL_EQUILIBRATED (SCALE_LINEAR+SCALE_EXTREME+SCALE_INTEGERS) -#define SCALEMODEL_GEOMETRIC (SCALE_LINEAR+SCALE_GEOMETRIC+SCALE_INTEGERS) -#define SCALEMODEL_ARITHMETIC (SCALE_LINEAR+SCALE_MEAN+SCALE_INTEGERS) -#define SCALEMODEL_DYNAMIC (SCALEMODEL_GEOMETRIC+SCALE_EQUILIBRATE) -#define SCALEMODEL_CURTISREID (SCALE_CURTISREID+SCALE_INTEGERS+SCALE_POWER2) - -/* Iteration status and strategies (internal) */ -#define ITERATE_MAJORMAJOR 0 -#define ITERATE_MINORMAJOR 1 -#define ITERATE_MINORRETRY 2 - -/* Pricing methods */ -#define PRICER_FIRSTINDEX 0 -#define PRICER_DANTZIG 1 -#define PRICER_DEVEX 2 -#define PRICER_STEEPESTEDGE 3 -#define PRICER_LASTOPTION PRICER_STEEPESTEDGE - -/* Additional settings for pricers (internal) */ -#define PRICER_RANDFACT 0.1 -#define DEVEX_RESTARTLIMIT 1.0e+09 /* Reset the norms if any value exceeds this limit */ -#define DEVEX_MINVALUE 0.000 /* Minimum weight [0..1] for entering variable, consider 0.01 */ - -/* Pricing strategies */ -#define PRICE_PRIMALFALLBACK 4 /* In case of Steepest Edge, fall back to DEVEX in primal */ -#define PRICE_MULTIPLE 8 /* Enable multiple pricing (primal simplex) */ -#define PRICE_PARTIAL 16 /* Enable partial pricing */ -#define PRICE_ADAPTIVE 32 /* Temporarily use alternative strategy if cycling is detected */ -#define PRICE_HYBRID 64 /* NOT IMPLEMENTED */ -#define PRICE_RANDOMIZE 128 /* Adds a small randomization effect to the selected pricer */ -#define PRICE_AUTOPARTIAL 256 /* Detect and use data on the block structure of the model (primal) */ -#define PRICE_AUTOMULTIPLE 512 /* Automatically select multiple pricing (primal simplex) */ -#define PRICE_LOOPLEFT 1024 /* Scan entering/leaving columns left rather than right */ -#define PRICE_LOOPALTERNATE 2048 /* Scan entering/leaving columns alternatingly left/right */ -#define PRICE_HARRISTWOPASS 4096 /* Use Harris' primal pivot logic rather than the default */ -#define PRICE_FORCEFULL 8192 /* Non-user option to force full pricing */ -#define PRICE_TRUENORMINIT 16384 /* Use true norms for Devex and Steepest Edge initializations */ - -/*#define _PRICE_NOBOUNDFLIP*/ -#if defined _PRICE_NOBOUNDFLIP -#define PRICE_NOBOUNDFLIP 65536 /* Disallow automatic bound-flip during pivot */ -#endif - -#define PRICE_STRATEGYMASK (PRICE_PRIMALFALLBACK + \ - PRICE_MULTIPLE + PRICE_PARTIAL + \ - PRICE_ADAPTIVE + PRICE_HYBRID + \ - PRICE_RANDOMIZE + PRICE_AUTOPARTIAL + PRICE_AUTOMULTIPLE + \ - PRICE_LOOPLEFT + PRICE_LOOPALTERNATE + \ - PRICE_HARRISTWOPASS + \ - PRICE_FORCEFULL + PRICE_TRUENORMINIT) - -/* B&B active variable codes (internal) */ -#define BB_REAL 0 -#define BB_INT 1 -#define BB_SC 2 -#define BB_SOS 3 -#define BB_GUB 4 - -/* B&B strategies */ -#define NODE_FIRSTSELECT 0 -#define NODE_GAPSELECT 1 -#define NODE_RANGESELECT 2 -#define NODE_FRACTIONSELECT 3 -#define NODE_PSEUDOCOSTSELECT 4 -#define NODE_PSEUDONONINTSELECT 5 /* Kjell Eikland #1 - Minimize B&B depth */ -#define NODE_PSEUDOFEASSELECT (NODE_PSEUDONONINTSELECT+NODE_WEIGHTREVERSEMODE) -#define NODE_PSEUDORATIOSELECT 6 /* Kjell Eikland #2 - Minimize a "cost/benefit" ratio */ -#define NODE_USERSELECT 7 -#define NODE_STRATEGYMASK (NODE_WEIGHTREVERSEMODE-1) /* Mask for B&B strategies */ -#define NODE_WEIGHTREVERSEMODE 8 -#define NODE_BRANCHREVERSEMODE 16 -#define NODE_GREEDYMODE 32 -#define NODE_PSEUDOCOSTMODE 64 -#define NODE_DEPTHFIRSTMODE 128 -#define NODE_RANDOMIZEMODE 256 -#define NODE_GUBMODE 512 -#define NODE_DYNAMICMODE 1024 -#define NODE_RESTARTMODE 2048 -#define NODE_BREADTHFIRSTMODE 4096 -#define NODE_AUTOORDER 8192 -#define NODE_RCOSTFIXING 16384 -#define NODE_STRONGINIT 32768 - -#define BRANCH_CEILING 0 -#define BRANCH_FLOOR 1 -#define BRANCH_AUTOMATIC 2 -#define BRANCH_DEFAULT 3 - -/* Action constants for simplex and B&B (internal) */ -#define ACTION_NONE 0 -#define ACTION_ACTIVE 1 -#define ACTION_REBASE 2 -#define ACTION_RECOMPUTE 4 -#define ACTION_REPRICE 8 -#define ACTION_REINVERT 16 -#define ACTION_TIMEDREINVERT 32 -#define ACTION_ITERATE 64 -#define ACTION_RESTART 255 - -/* Solver status values */ -#define UNKNOWNERROR -5 -#define DATAIGNORED -4 -#define NOBFP -3 -#define NOMEMORY -2 -#define NOTRUN -1 -#define OPTIMAL 0 -#define SUBOPTIMAL 1 -#define INFEASIBLE 2 -#define UNBOUNDED 3 -#define DEGENERATE 4 -#define NUMFAILURE 5 -#define USERABORT 6 -#define TIMEOUT 7 -#define RUNNING 8 -#define PRESOLVED 9 -#define ACCURACYERROR 25 - -/* Branch & Bound and Lagrangean extra status values (internal) */ -#define PROCFAIL 10 -#define PROCBREAK 11 -#define FEASFOUND 12 -#define NOFEASFOUND 13 -#define FATHOMED 14 - -/* Status values internal to the solver (internal) */ -#define SWITCH_TO_PRIMAL 20 -#define SWITCH_TO_DUAL 21 -#define SINGULAR_BASIS 22 -#define LOSTFEAS 23 -#define MATRIXERROR 24 - -/* Objective testing options for "bb_better" (internal) */ -#define OF_RELAXED 0 -#define OF_INCUMBENT 1 -#define OF_WORKING 2 -#define OF_USERBREAK 3 -#define OF_HEURISTIC 4 -#define OF_DUALLIMIT 5 -#define OF_DELTA 8 /* Mode */ -#define OF_PROJECTED 16 /* Mode - future, not active */ - -#define OF_TEST_BT 1 -#define OF_TEST_BE 2 -#define OF_TEST_NE 3 -#define OF_TEST_WE 4 -#define OF_TEST_WT 5 -#define OF_TEST_RELGAP 8 /* Mode */ - - -/* Name list and sparse matrix storage parameters (internal) */ -#define MAT_START_SIZE 10000 -#define DELTACOLALLOC 100 -#define DELTAROWALLOC 100 -#define RESIZEFACTOR 4 /* Fractional increase in selected memory allocations */ - -/* Default solver parameters and tolerances (internal) */ -#define DEF_PARTIALBLOCKS 10 /* The default number of blocks for partial pricing */ -#define DEF_MAXRELAX 7 /* Maximum number of non-BB relaxations in MILP */ -#define DEF_MAXPIVOTRETRY 10 /* Maximum number of times to retry a div-0 situation */ -#define DEF_MAXSINGULARITIES 10 /* Maximum number of singularities in refactorization */ -#define MAX_MINITUPDATES 60 /* Maximum number of bound swaps between refactorizations - without recomputing the whole vector - contain errors */ -#define MIN_REFACTFREQUENCY 5 /* Refactorization frequency indicating an inherent - numerical instability of the basis */ -#define LAG_SINGULARLIMIT 5 /* Number of times the objective does not change - before it is assumed that the Lagrangean constraints - are non-binding, and therefore impossible to converge; - upper iteration limit is divided by this threshold */ -#define MIN_TIMEPIVOT 5.0e-02 /* Minimum time per pivot for reinversion optimization - purposes; use active monitoring only if a pivot - takes more than MINTIMEPIVOT seconds. 5.0e-2 is - roughly suitable for a 1GHz system. */ -#define MAX_STALLCOUNT 12 /* The absolute upper limit to the number of stalling or - cycling iterations before switching rule */ -#define MAX_RULESWITCH 5 /* The maximum number of times to try an alternate pricing rule - to recover from stalling; set negative for no limit. */ -#define DEF_TIMEDREFACT AUTOMATIC /* Default for timed refactorization in BFPs; - can be FALSE, TRUE or AUTOMATIC (dynamic) */ - -#define DEF_SCALINGLIMIT 5 /* The default maximum number of scaling iterations */ - -#define DEF_NEGRANGE -1.0e+06 /* Downward limit for expanded variable range before the - variable is split into positive and negative components */ -#define DEF_BB_LIMITLEVEL -50 /* Relative B&B limit to protect against very deep, - memory-consuming trees */ - -#define MAX_FRACSCALE 6 /* The maximum decimal scan range for simulated integers */ -#define RANDSCALE 100 /* Randomization scaling range */ -#define DOUBLEROUND 0.0e-02 /* Extra rounding scalar used in btran/ftran calculations; the - rationale for 0.0 is that prod_xA() uses rounding as well */ -#define DEF_EPSMACHINE 2.22e-16 /* Machine relative precision (doubles) */ -#define MIN_STABLEPIVOT 5.0 /* Minimum pivot magnitude assumed to be numerically stable */ - - -/* Precision macros */ -/* -------------------------------------------------------------------------------------- */ -#define PREC_REDUCEDCOST lp->epsvalue -#define PREC_IMPROVEGAP lp->epsdual -#define PREC_SUBSTFEASGAP lp->epsprimal -#if 1 - #define PREC_BASICSOLUTION lp->epsvalue /* Zero-rounding of RHS/basic solution vector */ -#else - #define PREC_BASICSOLUTION lp->epsmachine /* Zero-rounding of RHS/basic solution vector */ -#endif -#define LIMIT_ABS_REL 10.0 /* Limit for testing using relative metric */ - - -/* Parameters constants for short-cut setting of tolerances */ -/* -------------------------------------------------------------------------------------- */ -#define EPS_TIGHT 0 -#define EPS_MEDIUM 1 -#define EPS_LOOSE 2 -#define EPS_BAGGY 3 -#define EPS_DEFAULT EPS_TIGHT - - -#if ActivePARAM==ProductionPARAM /* PARAMETER SET FOR PRODUCTION */ -/* -------------------------------------------------------------------------------------- */ -#define DEF_INFINITE 1.0e+30 /* Limit for dynamic range */ -#define DEF_EPSVALUE 1.0e-12 /* High accuracy and feasibility preserving tolerance */ -#define DEF_EPSPRIMAL 1.0e-10 /* For rounding primal/RHS values to 0 */ -#define DEF_EPSDUAL 1.0e-09 /* For rounding reduced costs to 0 */ -#define DEF_EPSPIVOT 2.0e-07 /* Pivot reject threshold */ -#define DEF_PERTURB 1.0e-05 /* Perturbation scalar for degenerate problems; - must at least be RANDSCALE greater than EPSPRIMAL */ -#define DEF_EPSSOLUTION 1.0e-05 /* Margin of error for solution bounds */ -#define DEF_EPSINT 1.0e-07 /* Accuracy for considering a float value as integer */ - -#elif ActivePARAM==OriginalPARAM /* PARAMETER SET FOR LEGACY VERSIONS */ -/* -------------------------------------------------------------------------------------- */ -#define DEF_INFINITE 1.0e+24 /* Limit for dynamic range */ -#define DEF_EPSVALUE 1.0e-08 /* High accuracy and feasibility preserving tolerance */ -#define DEF_EPSPRIMAL 5.01e-07 /* For rounding primal/RHS values to 0, infeasibility */ -#define DEF_EPSDUAL 1.0e-06 /* For rounding reduced costs to 0 */ -#define DEF_EPSPIVOT 1.0e-04 /* Pivot reject threshold */ -#define DEF_PERTURB 1.0e-05 /* Perturbation scalar for degenerate problems; - must at least be RANDSCALE greater than EPSPRIMAL */ -#define DEF_EPSSOLUTION 1.0e-02 /* Margin of error for solution bounds */ -#define DEF_EPSINT 1.0e-03 /* Accuracy for considering a float value as integer */ - -#elif ActivePARAM==ChvatalPARAM /* PARAMETER SET EXAMPLES FROM Vacek Chvatal */ -/* -------------------------------------------------------------------------------------- */ -#define DEF_INFINITE 1.0e+30 /* Limit for dynamic range */ -#define DEF_EPSVALUE 1.0e-10 /* High accuracy and feasibility preserving tolerance */ -#define DEF_EPSPRIMAL 10e-07 /* For rounding primal/RHS values to 0 */ -#define DEF_EPSDUAL 10e-05 /* For rounding reduced costs to 0 */ -#define DEF_EPSPIVOT 10e-05 /* Pivot reject threshold */ -#define DEF_PERTURB 10e-03 /* Perturbation scalar for degenerate problems; - must at least be RANDSCALE greater than EPSPRIMAL */ -#define DEF_EPSSOLUTION 1.0e-05 /* Margin of error for solution bounds */ -#define DEF_EPSINT 5.0e-03 /* Accuracy for considering a float value as integer */ - -#elif ActivePARAM==LoosePARAM /* PARAMETER SET FOR LOOSE TOLERANCES */ -/* -------------------------------------------------------------------------------------- */ -#define DEF_INFINITE 1.0e+30 /* Limit for dynamic range */ -#define DEF_EPSVALUE 1.0e-10 /* High accuracy and feasibility preserving tolerance */ -#define DEF_EPSPRIMAL 5.01e-08 /* For rounding primal/RHS values to 0 */ -#define DEF_EPSDUAL 1.0e-07 /* For rounding reduced costs to 0 */ -#define DEF_EPSPIVOT 1.0e-05 /* Pivot reject threshold */ -#define DEF_PERTURB 1.0e-05 /* Perturbation scalar for degenerate problems; - must at least be RANDSCALE greater than EPSPRIMAL */ -#define DEF_EPSSOLUTION 1.0e-05 /* Margin of error for solution bounds */ -#define DEF_EPSINT 1.0e-04 /* Accuracy for considering a float value as integer */ - -#endif - - -#define DEF_MIP_GAP 1.0e-11 /* The default absolute and relative MIP gap */ -#define SCALEDINTFIXRANGE 1.6 /* Epsilon range multiplier < 2 for collapsing bounds to fix */ - -#define MIN_SCALAR 1.0e-10 /* Smallest allowed scaling adjustment */ -#define MAX_SCALAR 1.0e+10 /* Largest allowed scaling adjustment */ -#define DEF_SCALINGEPS 1.0e-02 /* Relative scaling convergence criterion for auto_scale */ - -#define DEF_LAGACCEPT 1.0e-03 /* Default Lagrangean convergence acceptance criterion */ -#define DEF_LAGCONTRACT 0.90 /* The contraction parameter for Lagrangean iterations */ -#define DEF_LAGMAXITERATIONS 100 /* The maximum number of Lagrangean iterations */ - -#define DEF_PSEUDOCOSTUPDATES 7 /* The default number of times pseudo-costs are recalculated; - experiments indicate that costs tend to stabilize */ -#define DEF_PSEUDOCOSTRESTART 0.15 /* The fraction of price updates required for B&B restart - when the mode is NODE_RESTARTMODE */ -#define DEF_MAXPRESOLVELOOPS 0 /* Upper limit to the number of loops during presolve, - <= 0 for no limit. */ - - -/* Hashing prototypes and function headers */ -/* ------------------------------------------------------------------------- */ -#include "lp_Hash.h" - - -/* Sparse matrix prototypes */ -/* ------------------------------------------------------------------------- */ -#include "lp_matrix.h" - - -/* Basis storage (mainly for B&B) */ -typedef struct _basisrec -{ - int level; - int *var_basic; - MYBOOL *is_basic; - MYBOOL *is_lower; - int pivots; - struct _basisrec *previous; -} basisrec; - -/* Presolve undo data storage */ -typedef struct _presolveundorec -{ - lprec *lp; - int orig_rows; - int orig_columns; - int orig_sum; - int *var_to_orig; /* sum_alloc+1 : Mapping of variables from solution to - best_solution to account for removed variables and - rows during presolve; a non-positive value indicates - that the constraint or variable was removed */ - int *orig_to_var; /* sum_alloc+1 : Mapping from original variable index to - current / working index number */ - REAL *fixed_rhs; /* rows_alloc+1 : Storage of values of presolved fixed colums */ - REAL *fixed_obj; /* columns_alloc+1: Storage of values of presolved fixed rows */ - DeltaVrec *deletedA; /* A matrix of eliminated data from matA */ - DeltaVrec *primalundo; /* Affine translation vectors for eliminated primal variables */ - DeltaVrec *dualundo; /* Affine translation vectors for eliminated dual variables */ - MYBOOL OFcolsdeleted; -} presolveundorec; - -/* Pseudo-cost arrays used during B&B */ -typedef struct _BBPSrec -{ - lprec *lp; - int pseodotype; - int updatelimit; - int updatesfinished; - REAL restartlimit; - MATitem *UPcost; - MATitem *LOcost; - struct _BBPSrec *secondary; -} BBPSrec; - -#include "lp_mipbb.h" - - -/* Partial pricing block data */ -typedef struct _partialrec { - lprec *lp; - int blockcount; /* ## The number of logical blocks or stages in the model */ - int blocknow; /* The currently active block */ - int *blockend; /* Array of column indeces giving the start of each block */ - int *blockpos; /* Array of column indeces giving the start scan position */ - MYBOOL isrow; -} partialrec; - - -/* Specially Ordered Sets (SOS) prototypes and settings */ -/* ------------------------------------------------------------------------- */ -/* SOS storage structure (LINEARSEARCH is typically in the 0-10 range) */ -#ifndef LINEARSEARCH -#define LINEARSEARCH 0 -#endif - -#include "lp_SOS.h" - - -/* Prototypes for user call-back functions */ -/* ------------------------------------------------------------------------- */ -typedef int (__WINAPI lphandle_intfunc)(lprec *lp, void *userhandle); -typedef void (__WINAPI lphandlestr_func)(lprec *lp, void *userhandle, char *buf); -typedef void (__WINAPI lphandleint_func)(lprec *lp, void *userhandle, int message); -typedef int (__WINAPI lphandleint_intfunc)(lprec *lp, void *userhandle, int message); - - -/* API typedef definitions */ -/* ------------------------------------------------------------------------- */ -typedef MYBOOL (__WINAPI add_column_func)(lprec *lp, REAL *column); -typedef MYBOOL (__WINAPI add_columnex_func)(lprec *lp, int count, REAL *column, int *rowno); -typedef MYBOOL (__WINAPI add_constraint_func)(lprec *lp, REAL *row, int constr_type, REAL rh); -typedef MYBOOL (__WINAPI add_constraintex_func)(lprec *lp, int count, REAL *row, int *colno, int constr_type, REAL rh); -typedef MYBOOL (__WINAPI add_lag_con_func)(lprec *lp, REAL *row, int con_type, REAL rhs); -typedef int (__WINAPI add_SOS_func)(lprec *lp, char *name, int sostype, int priority, int count, int *sosvars, REAL *weights); -typedef int (__WINAPI column_in_lp_func)(lprec *lp, REAL *column); -typedef lprec * (__WINAPI copy_lp_func)(lprec *lp); -typedef void (__WINAPI default_basis_func)(lprec *lp); -typedef MYBOOL (__WINAPI del_column_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI del_constraint_func)(lprec *lp, int rownr); -typedef void (__WINAPI delete_lp_func)(lprec *lp); -typedef MYBOOL (__WINAPI dualize_lp_func)(lprec *lp); -typedef void (__WINAPI free_lp_func)(lprec **plp); -typedef int (__WINAPI get_anti_degen_func)(lprec *lp); -typedef MYBOOL (__WINAPI get_basis_func)(lprec *lp, int *bascolumn, MYBOOL nonbasic); -typedef int (__WINAPI get_basiscrash_func)(lprec *lp); -typedef int (__WINAPI get_bb_depthlimit_func)(lprec *lp); -typedef int (__WINAPI get_bb_floorfirst_func)(lprec *lp); -typedef int (__WINAPI get_bb_rule_func)(lprec *lp); -typedef MYBOOL (__WINAPI get_bounds_tighter_func)(lprec *lp); -typedef REAL (__WINAPI get_break_at_value_func)(lprec *lp); -typedef char * (__WINAPI get_col_name_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI get_column_func)(lprec *lp, int colnr, REAL *column); -typedef int (__WINAPI get_columnex_func)(lprec *lp, int colnr, REAL *column, int *nzrow); -typedef int (__WINAPI get_constr_type_func)(lprec *lp, int rownr); -typedef REAL (__WINAPI get_constr_value_func)(lprec *lp, int rownr, int count, REAL *primsolution, int *nzindex); -typedef MYBOOL (__WINAPI get_constraints_func)(lprec *lp, REAL *constr); -typedef MYBOOL (__WINAPI get_dual_solution_func)(lprec *lp, REAL *rc); -typedef REAL (__WINAPI get_epsb_func)(lprec *lp); -typedef REAL (__WINAPI get_epsd_func)(lprec *lp); -typedef REAL (__WINAPI get_epsel_func)(lprec *lp); -typedef REAL (__WINAPI get_epsint_func)(lprec *lp); -typedef REAL (__WINAPI get_epsperturb_func)(lprec *lp); -typedef REAL (__WINAPI get_epspivot_func)(lprec *lp); -typedef int (__WINAPI get_improve_func)(lprec *lp); -typedef REAL (__WINAPI get_infinite_func)(lprec *lp); -typedef MYBOOL (__WINAPI get_lambda_func)(lprec *lp, REAL *lambda); -typedef REAL (__WINAPI get_lowbo_func)(lprec *lp, int colnr); -typedef int (__WINAPI get_lp_index_func)(lprec *lp, int orig_index); -typedef char * (__WINAPI get_lp_name_func)(lprec *lp); -typedef int (__WINAPI get_Lrows_func)(lprec *lp); -typedef REAL (__WINAPI get_mat_func)(lprec *lp, int rownr, int colnr); -typedef REAL (__WINAPI get_mat_byindex_func)(lprec *lp, int matindex, MYBOOL isrow, MYBOOL adjustsign); -typedef int (__WINAPI get_max_level_func)(lprec *lp); -typedef int (__WINAPI get_maxpivot_func)(lprec *lp); -typedef REAL (__WINAPI get_mip_gap_func)(lprec *lp, MYBOOL absolute); -typedef int (__WINAPI get_multiprice_func)(lprec *lp, MYBOOL getabssize); -typedef MYBOOL (__WINAPI is_use_names_func)(lprec *lp, MYBOOL isrow); -typedef void (__WINAPI set_use_names_func)(lprec *lp, MYBOOL isrow, MYBOOL use_names); -typedef int (__WINAPI get_nameindex_func)(lprec *lp, char *varname, MYBOOL isrow); -typedef int (__WINAPI get_Ncolumns_func)(lprec *lp); -typedef REAL (__WINAPI get_negrange_func)(lprec *lp); -typedef int (__WINAPI get_nz_func)(lprec *lp); -typedef int (__WINAPI get_Norig_columns_func)(lprec *lp); -typedef int (__WINAPI get_Norig_rows_func)(lprec *lp); -typedef int (__WINAPI get_Nrows_func)(lprec *lp); -typedef REAL (__WINAPI get_obj_bound_func)(lprec *lp); -typedef REAL (__WINAPI get_objective_func)(lprec *lp); -typedef int (__WINAPI get_orig_index_func)(lprec *lp, int lp_index); -typedef char * (__WINAPI get_origcol_name_func)(lprec *lp, int colnr); -typedef char * (__WINAPI get_origrow_name_func)(lprec *lp, int rownr); -typedef void (__WINAPI get_partialprice_func)(lprec *lp, int *blockcount, int *blockstart, MYBOOL isrow); -typedef int (__WINAPI get_pivoting_func)(lprec *lp); -typedef int (__WINAPI get_presolve_func)(lprec *lp); -typedef int (__WINAPI get_presolveloops_func)(lprec *lp); -typedef MYBOOL (__WINAPI get_primal_solution_func)(lprec *lp, REAL *pv); -typedef int (__WINAPI get_print_sol_func)(lprec *lp); -typedef MYBOOL (__WINAPI get_pseudocosts_func)(lprec *lp, REAL *clower, REAL *cupper, int *updatelimit); -typedef MYBOOL (__WINAPI get_ptr_constraints_func)(lprec *lp, REAL **constr); -typedef MYBOOL (__WINAPI get_ptr_dual_solution_func)(lprec *lp, REAL **rc); -typedef MYBOOL (__WINAPI get_ptr_lambda_func)(lprec *lp, REAL **lambda); -typedef MYBOOL (__WINAPI get_ptr_primal_solution_func)(lprec *lp, REAL **pv); -typedef MYBOOL (__WINAPI get_ptr_sensitivity_obj_func)(lprec *lp, REAL **objfrom, REAL **objtill); -typedef MYBOOL (__WINAPI get_ptr_sensitivity_objex_func)(lprec *lp, REAL **objfrom, REAL **objtill, REAL **objfromvalue, REAL **objtillvalue); -typedef MYBOOL (__WINAPI get_ptr_sensitivity_rhs_func)(lprec *lp, REAL **duals, REAL **dualsfrom, REAL **dualstill); -typedef MYBOOL (__WINAPI get_ptr_variables_func)(lprec *lp, REAL **var); -typedef REAL (__WINAPI get_rh_func)(lprec *lp, int rownr); -typedef REAL (__WINAPI get_rh_range_func)(lprec *lp, int rownr); -typedef int (__WINAPI get_rowex_func)(lprec *lp, int rownr, REAL *row, int *colno); -typedef MYBOOL (__WINAPI get_row_func)(lprec *lp, int rownr, REAL *row); -typedef char * (__WINAPI get_row_name_func)(lprec *lp, int rownr); -typedef REAL (__WINAPI get_scalelimit_func)(lprec *lp); -typedef int (__WINAPI get_scaling_func)(lprec *lp); -typedef MYBOOL (__WINAPI get_sensitivity_obj_func)(lprec *lp, REAL *objfrom, REAL *objtill); -typedef MYBOOL (__WINAPI get_sensitivity_objex_func)(lprec *lp, REAL *objfrom, REAL *objtill, REAL *objfromvalue, REAL *objtillvalue); -typedef MYBOOL (__WINAPI get_sensitivity_rhs_func)(lprec *lp, REAL *duals, REAL *dualsfrom, REAL *dualstill); -typedef int (__WINAPI get_simplextype_func)(lprec *lp); -typedef int (__WINAPI get_solutioncount_func)(lprec *lp); -typedef int (__WINAPI get_solutionlimit_func)(lprec *lp); -typedef int (__WINAPI get_status_func)(lprec *lp); -typedef char * (__WINAPI get_statustext_func)(lprec *lp, int statuscode); -typedef long (__WINAPI get_timeout_func)(lprec *lp); -typedef COUNTER (__WINAPI get_total_iter_func)(lprec *lp); -typedef COUNTER (__WINAPI get_total_nodes_func)(lprec *lp); -typedef REAL (__WINAPI get_upbo_func)(lprec *lp, int colnr); -typedef int (__WINAPI get_var_branch_func)(lprec *lp, int colnr); -typedef REAL (__WINAPI get_var_dualresult_func)(lprec *lp, int index); -typedef REAL (__WINAPI get_var_primalresult_func)(lprec *lp, int index); -typedef int (__WINAPI get_var_priority_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI get_variables_func)(lprec *lp, REAL *var); -typedef int (__WINAPI get_verbose_func)(lprec *lp); -typedef MYBOOL (__WINAPI guess_basis_func)(lprec *lp, REAL *guessvector, int *basisvector); -typedef REAL (__WINAPI get_working_objective_func)(lprec *lp); -typedef MYBOOL (__WINAPI has_BFP_func)(lprec *lp); -typedef MYBOOL (__WINAPI has_XLI_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_add_rowmode_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_anti_degen_func)(lprec *lp, int testmask); -typedef MYBOOL (__WINAPI is_binary_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI is_break_at_first_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_constr_type_func)(lprec *lp, int rownr, int mask); -typedef MYBOOL (__WINAPI is_debug_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_feasible_func)(lprec *lp, REAL *values, REAL threshold); -typedef MYBOOL (__WINAPI is_unbounded_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI is_infinite_func)(lprec *lp, REAL value); -typedef MYBOOL (__WINAPI is_int_func)(lprec *lp, int column); -typedef MYBOOL (__WINAPI is_integerscaling_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_lag_trace_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_maxim_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_nativeBFP_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_nativeXLI_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_negative_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI is_obj_in_basis_func)(lprec *lp); -typedef MYBOOL (__WINAPI is_piv_mode_func)(lprec *lp, int testmask); -typedef MYBOOL (__WINAPI is_piv_rule_func)(lprec *lp, int rule); -typedef MYBOOL (__WINAPI is_presolve_func)(lprec *lp, int testmask); -typedef MYBOOL (__WINAPI is_scalemode_func)(lprec *lp, int testmask); -typedef MYBOOL (__WINAPI is_scaletype_func)(lprec *lp, int scaletype); -typedef MYBOOL (__WINAPI is_semicont_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI is_SOS_var_func)(lprec *lp, int colnr); -typedef MYBOOL (__WINAPI is_trace_func)(lprec *lp); -typedef void (__WINAPI lp_solve_version_func)(int *majorversion, int *minorversion, int *release, int *build); -typedef lprec * (__WINAPI make_lp_func)(int rows, int columns); -typedef void (__WINAPI print_constraints_func)(lprec *lp, int columns); -typedef MYBOOL (__WINAPI print_debugdump_func)(lprec *lp, char *filename); -typedef void (__WINAPI print_duals_func)(lprec *lp); -typedef void (__WINAPI print_lp_func)(lprec *lp); -typedef void (__WINAPI print_objective_func)(lprec *lp); -typedef void (__WINAPI print_scales_func)(lprec *lp); -typedef void (__WINAPI print_solution_func)(lprec *lp, int columns); -typedef void (__WINAPI print_str_func)(lprec *lp, char *str); -typedef void (__WINAPI print_tableau_func)(lprec *lp); -typedef void (__WINAPI put_abortfunc_func)(lprec *lp, lphandle_intfunc newctrlc, void *ctrlchandle); -typedef void (__WINAPI put_bb_nodefunc_func)(lprec *lp, lphandleint_intfunc newnode, void *bbnodehandle); -typedef void (__WINAPI put_bb_branchfunc_func)(lprec *lp, lphandleint_intfunc newbranch, void *bbbranchhandle); -typedef void (__WINAPI put_logfunc_func)(lprec *lp, lphandlestr_func newlog, void *loghandle); -typedef void (__WINAPI put_msgfunc_func)(lprec *lp, lphandleint_func newmsg, void *msghandle, int mask); -typedef lprec * (__WINAPI read_LP_func)(char *filename, int verbose, char *lp_name); -typedef lprec * (__WINAPI read_MPS_func)(char *filename, int options); -typedef lprec * (__WINAPI read_XLI_func)(char *xliname, char *modelname, char *dataname, char *options, int verbose); -typedef MYBOOL (__WINAPI read_basis_func)(lprec *lp, char *filename, char *info); -typedef void (__WINAPI reset_basis_func)(lprec *lp); -typedef MYBOOL (__WINAPI read_params_func)(lprec *lp, char *filename, char *options); -typedef void (__WINAPI reset_params_func)(lprec *lp); -typedef MYBOOL (__WINAPI resize_lp_func)(lprec *lp, int rows, int columns); -typedef MYBOOL (__WINAPI set_add_rowmode_func)(lprec *lp, MYBOOL turnon); -typedef void (__WINAPI set_anti_degen_func)(lprec *lp, int anti_degen); -typedef int (__WINAPI set_basisvar_func)(lprec *lp, int basisPos, int enteringCol); -typedef MYBOOL (__WINAPI set_basis_func)(lprec *lp, int *bascolumn, MYBOOL nonbasic); -typedef void (__WINAPI set_basiscrash_func)(lprec *lp, int mode); -typedef void (__WINAPI set_bb_depthlimit_func)(lprec *lp, int bb_maxlevel); -typedef void (__WINAPI set_bb_floorfirst_func)(lprec *lp, int bb_floorfirst); -typedef void (__WINAPI set_bb_rule_func)(lprec *lp, int bb_rule); -typedef MYBOOL (__WINAPI set_BFP_func)(lprec *lp, char *filename); -typedef MYBOOL (__WINAPI set_binary_func)(lprec *lp, int colnr, MYBOOL must_be_bin); -typedef MYBOOL (__WINAPI set_bounds_func)(lprec *lp, int colnr, REAL lower, REAL upper); -typedef void (__WINAPI set_bounds_tighter_func)(lprec *lp, MYBOOL tighten); -typedef void (__WINAPI set_break_at_first_func)(lprec *lp, MYBOOL break_at_first); -typedef void (__WINAPI set_break_at_value_func)(lprec *lp, REAL break_at_value); -typedef MYBOOL (__WINAPI set_column_func)(lprec *lp, int colnr, REAL *column); -typedef MYBOOL (__WINAPI set_columnex_func)(lprec *lp, int colnr, int count, REAL *column, int *rowno); -typedef MYBOOL (__WINAPI set_col_name_func)(lprec *lp, int colnr, char *new_name); -typedef MYBOOL (__WINAPI set_constr_type_func)(lprec *lp, int rownr, int con_type); -typedef void (__WINAPI set_debug_func)(lprec *lp, MYBOOL debug); -typedef void (__WINAPI set_epsb_func)(lprec *lp, REAL epsb); -typedef void (__WINAPI set_epsd_func)(lprec *lp, REAL epsd); -typedef void (__WINAPI set_epsel_func)(lprec *lp, REAL epsel); -typedef void (__WINAPI set_epsint_func)(lprec *lp, REAL epsint); -typedef MYBOOL (__WINAPI set_epslevel_func)(lprec *lp, int epslevel); -typedef void (__WINAPI set_epsperturb_func)(lprec *lp, REAL epsperturb); -typedef void (__WINAPI set_epspivot_func)(lprec *lp, REAL epspivot); -typedef MYBOOL (__WINAPI set_unbounded_func)(lprec *lp, int colnr); -typedef void (__WINAPI set_improve_func)(lprec *lp, int improve); -typedef void (__WINAPI set_infinite_func)(lprec *lp, REAL infinite); -typedef MYBOOL (__WINAPI set_int_func)(lprec *lp, int colnr, MYBOOL must_be_int); -typedef void (__WINAPI set_lag_trace_func)(lprec *lp, MYBOOL lag_trace); -typedef MYBOOL (__WINAPI set_lowbo_func)(lprec *lp, int colnr, REAL value); -typedef MYBOOL (__WINAPI set_lp_name_func)(lprec *lp, char *lpname); -typedef MYBOOL (__WINAPI set_mat_func)(lprec *lp, int row, int column, REAL value); -typedef void (__WINAPI set_maxim_func)(lprec *lp); -typedef void (__WINAPI set_maxpivot_func)(lprec *lp, int max_num_inv); -typedef void (__WINAPI set_minim_func)(lprec *lp); -typedef void (__WINAPI set_mip_gap_func)(lprec *lp, MYBOOL absolute, REAL mip_gap); -typedef MYBOOL (__WINAPI set_multiprice_func)(lprec *lp, int multiblockdiv); -typedef void (__WINAPI set_negrange_func)(lprec *lp, REAL negrange); -typedef MYBOOL (__WINAPI set_obj_func)(lprec *lp, int colnr, REAL value); -typedef void (__WINAPI set_obj_bound_func)(lprec *lp, REAL obj_bound); -typedef MYBOOL (__WINAPI set_obj_fn_func)(lprec *lp, REAL *row); -typedef MYBOOL (__WINAPI set_obj_fnex_func)(lprec *lp, int count, REAL *row, int *colno); -typedef void (__WINAPI set_obj_in_basis_func)(lprec *lp, MYBOOL obj_in_basis); -typedef MYBOOL (__WINAPI set_outputfile_func)(lprec *lp, char *filename); -typedef void (__WINAPI set_outputstream_func)(lprec *lp, FILE *stream); -typedef MYBOOL (__WINAPI set_partialprice_func)(lprec *lp, int blockcount, int *blockstart, MYBOOL isrow); -typedef void (__WINAPI set_pivoting_func)(lprec *lp, int piv_rule); -typedef void (__WINAPI set_preferdual_func)(lprec *lp, MYBOOL dodual); -typedef void (__WINAPI set_presolve_func)(lprec *lp, int presolvemode, int maxloops); -typedef void (__WINAPI set_print_sol_func)(lprec *lp, int print_sol); -typedef MYBOOL (__WINAPI set_pseudocosts_func)(lprec *lp, REAL *clower, REAL *cupper, int *updatelimit); -typedef MYBOOL (__WINAPI set_rh_func)(lprec *lp, int rownr, REAL value); -typedef MYBOOL (__WINAPI set_rh_range_func)(lprec *lp, int rownr, REAL deltavalue); -typedef void (__WINAPI set_rh_vec_func)(lprec *lp, REAL *rh); -typedef MYBOOL (__WINAPI set_row_func)(lprec *lp, int rownr, REAL *row); -typedef MYBOOL (__WINAPI set_rowex_func)(lprec *lp, int rownr, int count, REAL *row, int *colno); -typedef MYBOOL (__WINAPI set_row_name_func)(lprec *lp, int rownr, char *new_name); -typedef void (__WINAPI set_scalelimit_func)(lprec *lp, REAL scalelimit); -typedef void (__WINAPI set_scaling_func)(lprec *lp, int scalemode); -typedef MYBOOL (__WINAPI set_semicont_func)(lprec *lp, int colnr, MYBOOL must_be_sc); -typedef void (__WINAPI set_sense_func)(lprec *lp, MYBOOL maximize); -typedef void (__WINAPI set_simplextype_func)(lprec *lp, int simplextype); -typedef void (__WINAPI set_solutionlimit_func)(lprec *lp, int limit); -typedef void (__WINAPI set_timeout_func)(lprec *lp, long sectimeout); -typedef void (__WINAPI set_trace_func)(lprec *lp, MYBOOL trace); -typedef MYBOOL (__WINAPI set_upbo_func)(lprec *lp, int colnr, REAL value); -typedef MYBOOL (__WINAPI set_var_branch_func)(lprec *lp, int colnr, int branch_mode); -typedef MYBOOL (__WINAPI set_var_weights_func)(lprec *lp, REAL *weights); -typedef void (__WINAPI set_verbose_func)(lprec *lp, int verbose); -typedef MYBOOL (__WINAPI set_XLI_func)(lprec *lp, char *filename); -typedef int (__WINAPI solve_func)(lprec *lp); -typedef MYBOOL (__WINAPI str_add_column_func)(lprec *lp, char *col_string); -typedef MYBOOL (__WINAPI str_add_constraint_func)(lprec *lp, char *row_string ,int constr_type, REAL rh); -typedef MYBOOL (__WINAPI str_add_lag_con_func)(lprec *lp, char *row_string, int con_type, REAL rhs); -typedef MYBOOL (__WINAPI str_set_obj_fn_func)(lprec *lp, char *row_string); -typedef MYBOOL (__WINAPI str_set_rh_vec_func)(lprec *lp, char *rh_string); -typedef REAL (__WINAPI time_elapsed_func)(lprec *lp); -typedef void (__WINAPI unscale_func)(lprec *lp); -typedef MYBOOL (__WINAPI write_lp_func)(lprec *lp, char *filename); -typedef MYBOOL (__WINAPI write_LP_func)(lprec *lp, FILE *output); -typedef MYBOOL (__WINAPI write_mps_func)(lprec *lp, char *filename); -typedef MYBOOL (__WINAPI write_MPS_func)(lprec *lp, FILE *output); -typedef MYBOOL (__WINAPI write_freemps_func)(lprec *lp, char *filename); -typedef MYBOOL (__WINAPI write_freeMPS_func)(lprec *lp, FILE *output); -typedef MYBOOL (__WINAPI write_XLI_func)(lprec *lp, char *filename, char *options, MYBOOL results); -typedef MYBOOL (__WINAPI write_basis_func)(lprec *lp, char *filename); -typedef MYBOOL (__WINAPI write_params_func)(lprec *lp, char *filename, char *options); - - -/* Prototypes for callbacks from basis inverse/factorization libraries */ -/* ------------------------------------------------------------------------- */ -typedef MYBOOL (__WINAPI userabortfunc)(lprec *lp, int level); -typedef void (__VACALL reportfunc)(lprec *lp, int level, char *format, ...); -typedef char * (__VACALL explainfunc)(lprec *lp, char *format, ...); -typedef int (__WINAPI getvectorfunc)(lprec *lp, int varin, REAL *pcol, int *nzlist, int *maxabs); -typedef int (__WINAPI getpackedfunc)(lprec *lp, int j, int rn[], double bj[]); -typedef REAL (__WINAPI get_OF_activefunc)(lprec *lp, int varnr, REAL mult); -typedef int (__WINAPI getMDOfunc)(lprec *lp, MYBOOL *usedpos, int *colorder, int *size, MYBOOL symmetric); -typedef MYBOOL (__WINAPI invertfunc)(lprec *lp, MYBOOL shiftbounds, MYBOOL final); -typedef void (__WINAPI set_actionfunc)(int *actionvar, int actionmask); -typedef MYBOOL (__WINAPI is_actionfunc)(int actionvar, int testmask); -typedef void (__WINAPI clear_actionfunc)(int *actionvar, int actionmask); - - -/* Prototypes for basis inverse/factorization libraries */ -/* ------------------------------------------------------------------------- */ -typedef char *(BFP_CALLMODEL BFPchar)(void); -typedef void (BFP_CALLMODEL BFP_lp)(lprec *lp); -typedef void (BFP_CALLMODEL BFP_lpint)(lprec *lp, int newsize); -typedef int (BFP_CALLMODEL BFPint_lp)(lprec *lp); -typedef int (BFP_CALLMODEL BFPint_lpint)(lprec *lp, int kind); -typedef REAL (BFP_CALLMODEL BFPreal_lp)(lprec *lp); -typedef REAL *(BFP_CALLMODEL BFPrealp_lp)(lprec *lp); -typedef void (BFP_CALLMODEL BFP_lpbool)(lprec *lp, MYBOOL maximum); -typedef int (BFP_CALLMODEL BFPint_lpbool)(lprec *lp, MYBOOL maximum); -typedef int (BFP_CALLMODEL BFPint_lpintintboolbool)(lprec *lp, int uservars, int Bsize, MYBOOL *usedpos, MYBOOL final); -typedef void (BFP_CALLMODEL BFP_lprealint)(lprec *lp, REAL *pcol, int *nzidx); -typedef void (BFP_CALLMODEL BFP_lprealintrealint)(lprec *lp, REAL *prow, int *pnzidx, REAL *drow, int *dnzidx); -typedef MYBOOL (BFP_CALLMODEL BFPbool_lp)(lprec *lp); -typedef MYBOOL (BFP_CALLMODEL BFPbool_lpbool)(lprec *lp, MYBOOL changesign); -typedef MYBOOL (BFP_CALLMODEL BFPbool_lpint)(lprec *lp, int size); -typedef MYBOOL (BFP_CALLMODEL BFPbool_lpintintchar)(lprec *lp, int size, int deltasize, char *options); -typedef MYBOOL (BFP_CALLMODEL BFPbool_lpintintint)(lprec *lp, int size, int deltasize, int sizeofvar); -typedef LREAL (BFP_CALLMODEL BFPlreal_lpintintreal)(lprec *lp, int row_nr, int col_nr, REAL *pcol); -typedef REAL (BFP_CALLMODEL BFPreal_lplrealreal)(lprec *lp, LREAL theta, REAL *pcol); - -typedef int (BFP_CALLMODEL getcolumnex_func)(lprec *lp, int colnr, REAL *nzvalues, int *nzrows, int *mapin); -typedef int (BFP_CALLMODEL BFPint_lpintrealcbintint)(lprec *lp, int items, getcolumnex_func cb, int *maprow, int*mapcol); - -/* Prototypes for external language libraries */ -/* ------------------------------------------------------------------------- */ -typedef char *(XLI_CALLMODEL XLIchar)(void); -typedef MYBOOL (XLI_CALLMODEL XLIbool_lpintintint)(lprec* lp, int size, int deltasize, int sizevar); -typedef MYBOOL (XLI_CALLMODEL XLIbool_lpcharcharcharint)(lprec *lp, char *modelname, char *dataname, char *options, int verbose); -typedef MYBOOL (XLI_CALLMODEL XLIbool_lpcharcharbool)(lprec *lp, char *filename, char *options, MYBOOL results); - - -/* Main lp_solve prototypes and function definitions */ -/* ------------------------------------------------------------------------- */ -struct _lprec -{ - /* Full list of exported functions made available in a quasi object-oriented fashion */ - add_column_func *add_column; - add_columnex_func *add_columnex; - add_constraint_func *add_constraint; - add_constraintex_func *add_constraintex; - add_lag_con_func *add_lag_con; - add_SOS_func *add_SOS; - column_in_lp_func *column_in_lp; - copy_lp_func *copy_lp; - default_basis_func *default_basis; - del_column_func *del_column; - del_constraint_func *del_constraint; - delete_lp_func *delete_lp; - dualize_lp_func *dualize_lp; - free_lp_func *free_lp; - get_anti_degen_func *get_anti_degen; - get_basis_func *get_basis; - get_basiscrash_func *get_basiscrash; - get_bb_depthlimit_func *get_bb_depthlimit; - get_bb_floorfirst_func *get_bb_floorfirst; - get_bb_rule_func *get_bb_rule; - get_bounds_tighter_func *get_bounds_tighter; - get_break_at_value_func *get_break_at_value; - get_col_name_func *get_col_name; - get_columnex_func *get_columnex; - get_constr_type_func *get_constr_type; - get_constr_value_func *get_constr_value; - get_constraints_func *get_constraints; - get_dual_solution_func *get_dual_solution; - get_epsb_func *get_epsb; - get_epsd_func *get_epsd; - get_epsel_func *get_epsel; - get_epsint_func *get_epsint; - get_epsperturb_func *get_epsperturb; - get_epspivot_func *get_epspivot; - get_improve_func *get_improve; - get_infinite_func *get_infinite; - get_lambda_func *get_lambda; - get_lowbo_func *get_lowbo; - get_lp_index_func *get_lp_index; - get_lp_name_func *get_lp_name; - get_Lrows_func *get_Lrows; - get_mat_func *get_mat; - get_mat_byindex_func *get_mat_byindex; - get_max_level_func *get_max_level; - get_maxpivot_func *get_maxpivot; - get_mip_gap_func *get_mip_gap; - get_multiprice_func *get_multiprice; - get_nameindex_func *get_nameindex; - get_Ncolumns_func *get_Ncolumns; - get_negrange_func *get_negrange; - get_nz_func *get_nonzeros; - get_Norig_columns_func *get_Norig_columns; - get_Norig_rows_func *get_Norig_rows; - get_Nrows_func *get_Nrows; - get_obj_bound_func *get_obj_bound; - get_objective_func *get_objective; - get_orig_index_func *get_orig_index; - get_origcol_name_func *get_origcol_name; - get_origrow_name_func *get_origrow_name; - get_partialprice_func *get_partialprice; - get_pivoting_func *get_pivoting; - get_presolve_func *get_presolve; - get_presolveloops_func *get_presolveloops; - get_primal_solution_func *get_primal_solution; - get_print_sol_func *get_print_sol; - get_pseudocosts_func *get_pseudocosts; - get_ptr_constraints_func *get_ptr_constraints; - get_ptr_dual_solution_func *get_ptr_dual_solution; - get_ptr_lambda_func *get_ptr_lambda; - get_ptr_primal_solution_func *get_ptr_primal_solution; - get_ptr_sensitivity_obj_func *get_ptr_sensitivity_obj; - get_ptr_sensitivity_objex_func *get_ptr_sensitivity_objex; - get_ptr_sensitivity_rhs_func *get_ptr_sensitivity_rhs; - get_ptr_variables_func *get_ptr_variables; - get_rh_func *get_rh; - get_rh_range_func *get_rh_range; - get_row_func *get_row; - get_rowex_func *get_rowex; - get_row_name_func *get_row_name; - get_scalelimit_func *get_scalelimit; - get_scaling_func *get_scaling; - get_sensitivity_obj_func *get_sensitivity_obj; - get_sensitivity_objex_func *get_sensitivity_objex; - get_sensitivity_rhs_func *get_sensitivity_rhs; - get_simplextype_func *get_simplextype; - get_solutioncount_func *get_solutioncount; - get_solutionlimit_func *get_solutionlimit; - get_status_func *get_status; - get_statustext_func *get_statustext; - get_timeout_func *get_timeout; - get_total_iter_func *get_total_iter; - get_total_nodes_func *get_total_nodes; - get_upbo_func *get_upbo; - get_var_branch_func *get_var_branch; - get_var_dualresult_func *get_var_dualresult; - get_var_primalresult_func *get_var_primalresult; - get_var_priority_func *get_var_priority; - get_variables_func *get_variables; - get_verbose_func *get_verbose; - get_working_objective_func *get_working_objective; - has_BFP_func *has_BFP; - has_XLI_func *has_XLI; - is_add_rowmode_func *is_add_rowmode; - is_anti_degen_func *is_anti_degen; - is_binary_func *is_binary; - is_break_at_first_func *is_break_at_first; - is_constr_type_func *is_constr_type; - is_debug_func *is_debug; - is_feasible_func *is_feasible; - is_infinite_func *is_infinite; - is_int_func *is_int; - is_integerscaling_func *is_integerscaling; - is_lag_trace_func *is_lag_trace; - is_maxim_func *is_maxim; - is_nativeBFP_func *is_nativeBFP; - is_nativeXLI_func *is_nativeXLI; - is_negative_func *is_negative; - is_obj_in_basis_func *is_obj_in_basis; - is_piv_mode_func *is_piv_mode; - is_piv_rule_func *is_piv_rule; - is_presolve_func *is_presolve; - is_scalemode_func *is_scalemode; - is_scaletype_func *is_scaletype; - is_semicont_func *is_semicont; - is_SOS_var_func *is_SOS_var; - is_trace_func *is_trace; - is_unbounded_func *is_unbounded; - is_use_names_func *is_use_names; - lp_solve_version_func *lp_solve_version; - make_lp_func *make_lp; - print_constraints_func *print_constraints; - print_debugdump_func *print_debugdump; - print_duals_func *print_duals; - print_lp_func *print_lp; - print_objective_func *print_objective; - print_scales_func *print_scales; - print_solution_func *print_solution; - print_str_func *print_str; - print_tableau_func *print_tableau; - put_abortfunc_func *put_abortfunc; - put_bb_nodefunc_func *put_bb_nodefunc; - put_bb_branchfunc_func *put_bb_branchfunc; - put_logfunc_func *put_logfunc; - put_msgfunc_func *put_msgfunc; - read_LP_func *read_LP; - read_MPS_func *read_MPS; - read_XLI_func *read_XLI; - read_params_func *read_params; - read_basis_func *read_basis; - reset_basis_func *reset_basis; - reset_params_func *reset_params; - resize_lp_func *resize_lp; - set_add_rowmode_func *set_add_rowmode; - set_anti_degen_func *set_anti_degen; - set_basisvar_func *set_basisvar; - set_basis_func *set_basis; - set_basiscrash_func *set_basiscrash; - set_bb_depthlimit_func *set_bb_depthlimit; - set_bb_floorfirst_func *set_bb_floorfirst; - set_bb_rule_func *set_bb_rule; - set_BFP_func *set_BFP; - set_binary_func *set_binary; - set_bounds_func *set_bounds; - set_bounds_tighter_func *set_bounds_tighter; - set_break_at_first_func *set_break_at_first; - set_break_at_value_func *set_break_at_value; - set_column_func *set_column; - set_columnex_func *set_columnex; - set_col_name_func *set_col_name; - set_constr_type_func *set_constr_type; - set_debug_func *set_debug; - set_epsb_func *set_epsb; - set_epsd_func *set_epsd; - set_epsel_func *set_epsel; - set_epsint_func *set_epsint; - set_epslevel_func *set_epslevel; - set_epsperturb_func *set_epsperturb; - set_epspivot_func *set_epspivot; - set_unbounded_func *set_unbounded; - set_improve_func *set_improve; - set_infinite_func *set_infinite; - set_int_func *set_int; - set_lag_trace_func *set_lag_trace; - set_lowbo_func *set_lowbo; - set_lp_name_func *set_lp_name; - set_mat_func *set_mat; - set_maxim_func *set_maxim; - set_maxpivot_func *set_maxpivot; - set_minim_func *set_minim; - set_mip_gap_func *set_mip_gap; - set_multiprice_func *set_multiprice; - set_negrange_func *set_negrange; - set_obj_bound_func *set_obj_bound; - set_obj_fn_func *set_obj_fn; - set_obj_fnex_func *set_obj_fnex; - set_obj_func *set_obj; - set_obj_in_basis_func *set_obj_in_basis; - set_outputfile_func *set_outputfile; - set_outputstream_func *set_outputstream; - set_partialprice_func *set_partialprice; - set_pivoting_func *set_pivoting; - set_preferdual_func *set_preferdual; - set_presolve_func *set_presolve; - set_print_sol_func *set_print_sol; - set_pseudocosts_func *set_pseudocosts; - set_rh_func *set_rh; - set_rh_range_func *set_rh_range; - set_rh_vec_func *set_rh_vec; - set_row_func *set_row; - set_rowex_func *set_rowex; - set_row_name_func *set_row_name; - set_scalelimit_func *set_scalelimit; - set_scaling_func *set_scaling; - set_semicont_func *set_semicont; - set_sense_func *set_sense; - set_simplextype_func *set_simplextype; - set_solutionlimit_func *set_solutionlimit; - set_timeout_func *set_timeout; - set_trace_func *set_trace; - set_upbo_func *set_upbo; - set_use_names_func *set_use_names; - set_var_branch_func *set_var_branch; - set_var_weights_func *set_var_weights; - set_verbose_func *set_verbose; - set_XLI_func *set_XLI; - solve_func *solve; - str_add_column_func *str_add_column; - str_add_constraint_func *str_add_constraint; - str_add_lag_con_func *str_add_lag_con; - str_set_obj_fn_func *str_set_obj_fn; - str_set_rh_vec_func *str_set_rh_vec; - time_elapsed_func *time_elapsed; - unscale_func *unscale; - write_lp_func *write_lp; - write_LP_func *write_LP; - write_mps_func *write_mps; - write_MPS_func *write_MPS; - write_freemps_func *write_freemps; - write_freeMPS_func *write_freeMPS; - write_XLI_func *write_XLI; - write_basis_func *write_basis; - write_params_func *write_params; - - /* Spacer */ - int *alignmentspacer; - - /* Problem description */ - char *lp_name; /* The name of the model */ - - /* Problem sizes */ - int sum; /* The total number of variables, including slacks */ - int rows; - int columns; - int equalities; /* No of non-Lagrangean equality constraints in the problem */ - int boundedvars; /* Count of bounded variables */ - int INTfuture1; - - /* Memory allocation sizes */ - int sum_alloc; /* The allocated memory for row+column-sized data */ - int rows_alloc; /* The allocated memory for row-sized data */ - int columns_alloc; /* The allocated memory for column-sized data */ - - /* Model status and solver result variables */ - MYBOOL source_is_file; /* The base model was read from a file */ - MYBOOL model_is_pure; /* The model has been built entirely from row and column additions */ - MYBOOL model_is_valid; /* Has this lp pased the 'test' */ - MYBOOL tighten_on_set; /* Specify if bounds will be tightened or overriden at bound setting */ - MYBOOL names_used; /* Flag to indicate if names for rows and columns are used */ - MYBOOL use_row_names; /* Flag to indicate if names for rows are used */ - MYBOOL use_col_names; /* Flag to indicate if names for columns are used */ - - MYBOOL lag_trace; /* Print information on Lagrange progression */ - MYBOOL spx_trace; /* Print information on simplex progression */ - MYBOOL bb_trace; /* TRUE to print extra debug information */ - MYBOOL streamowned; /* TRUE if the handle should be closed at delete_lp() */ - MYBOOL obj_in_basis; /* TRUE if the objective function is in the basis matrix */ - - int spx_status; /* Simplex solver feasibility/mode code */ - int lag_status; /* Extra status variable for lag_solve */ - int solutioncount; /* number of equal-valued solutions found (up to solutionlimit) */ - int solutionlimit; /* upper number of equal-valued solutions kept track of */ - - REAL real_solution; /* Optimal non-MIP solution base */ - REAL *solution; /* sum_alloc+1 : Solution array of the next to optimal LP, - Index 0 : Objective function value, - Indeces 1..rows : Slack variable values, - Indeced rows+1..sum : Variable values */ - REAL *best_solution; /* sum_alloc+1 : Solution array of optimal 'Integer' LP, - structured as the solution array above */ - REAL *full_solution; /* sum_alloc+1 : Final solution array expanded for deleted variables */ - REAL *edgeVector; /* Array of reduced cost scaling norms (DEVEX and Steepest Edge) */ - - REAL *drow; /* sum+1: Reduced costs of the last simplex */ - int *nzdrow; /* sum+1: Indeces of non-zero reduced costs of the last simplex */ - REAL *duals; /* rows_alloc+1 : The dual variables of the last LP */ - REAL *full_duals; /* sum_alloc+1: Final duals array expanded for deleted variables */ - REAL *dualsfrom; /* sum_alloc+1 :The sensitivity on dual variables/reduced costs - of the last LP */ - REAL *dualstill; /* sum_alloc+1 :The sensitivity on dual variables/reduced costs - of the last LP */ - REAL *objfrom; /* columns_alloc+1 :The sensitivity on objective function - of the last LP */ - REAL *objtill; /* columns_alloc+1 :The sensitivity on objective function - of the last LP */ - REAL *objfromvalue; /* columns_alloc+1 :The value of the variables when objective value - is at its from value of the last LP */ - REAL *orig_obj; /* Unused pointer - Placeholder for OF not part of B */ - REAL *obj; /* Special vector used to temporarily change the OF vector */ - - COUNTER current_iter; /* Number of iterations in the current/last simplex */ - COUNTER total_iter; /* Number of iterations over all B&B steps */ - COUNTER current_bswap; /* Number of bound swaps in the current/last simplex */ - COUNTER total_bswap; /* Number of bount swaps over all B&B steps */ - int solvecount; /* The number of solve() performed in this model */ - int max_pivots; /* Number of pivots between refactorizations of the basis */ - - /* Various execution parameters */ - int simplex_strategy; /* Set desired combination of primal and dual simplex algorithms */ - int simplex_mode; /* Specifies the current simplex mode during solve; see simplex_strategy */ - int verbose; /* Set amount of run-time messages and results */ - int print_sol; /* TRUE to print optimal solution; AUTOMATIC skips zeros */ - FILE *outstream; /* Output stream, initialized to STDOUT */ - - /* Main Branch and Bound settings */ - MYBOOL *bb_varbranch; /* Determines branching strategy at the individual variable level; - the setting here overrides the bb_floorfirst setting */ - int piv_strategy; /* Strategy for selecting row and column entering/leaving */ - int _piv_rule_; /* Internal working rule-part of piv_strategy above */ - int bb_rule; /* Rule for selecting B&B variables */ - MYBOOL bb_floorfirst; /* Set BRANCH_FLOOR for B&B to set variables to floor bound first; - conversely with BRANCH_CEILING, the ceiling value is set first */ - MYBOOL bb_breakfirst; /* TRUE to stop at first feasible solution */ - MYBOOL _piv_left_; /* Internal variable indicating active pricing loop order */ - MYBOOL BOOLfuture1; - - REAL scalelimit; /* Relative convergence criterion for iterated scaling */ - int scalemode; /* OR-ed codes for data scaling */ - int improve; /* Set to non-zero for iterative improvement */ - int anti_degen; /* Anti-degen strategy (or none) TRUE to avoid cycling */ - int do_presolve; /* PRESOLVE_ parameters for LP presolving */ - int presolveloops; /* Maximum number of presolve loops */ - - int perturb_count; /* The number of bound relaxation retries performed */ - - /* Row and column names storage variables */ - hashelem **row_name; /* rows_alloc+1 */ - hashelem **col_name; /* columns_alloc+1 */ - hashtable *rowname_hashtab; /* hash table to store row names */ - hashtable *colname_hashtab; /* hash table to store column names */ - - /* Optionally specify continuous rows/column blocks for partial pricing */ - partialrec *rowblocks; - partialrec *colblocks; - - /* Row and column type codes */ - MYBOOL *var_type; /* sum_alloc+1 : TRUE if variable must be integer */ - - /* Data for multiple pricing */ - multirec *multivars; - int multiblockdiv; /* The divisor used to set or augment pricing block */ - - /* Variable (column) parameters */ - int fixedvars; /* The current number of basic fixed variables in the model */ - int int_vars; /* Number of variables required to be integer */ - - int sc_vars; /* Number of semi-continuous variables */ - REAL *sc_lobound; /* sum_columns+1 : TRUE if variable is semi-continuous; - value replaced by conventional lower bound during solve */ - int *var_is_free; /* columns+1: Index of twin variable if variable is free */ - int *var_priority; /* columns: Priority-mapping of variables */ - - SOSgroup *GUB; /* Pointer to record containing GUBs */ - - int sos_vars; /* Number of variables in the sos_priority list */ - int sos_ints; /* Number of integers in SOS'es above */ - SOSgroup *SOS; /* Pointer to record containing all SOS'es */ - int *sos_priority; /* Priority-sorted list of variables (no duplicates) */ - - /* Optionally specify list of active rows/columns used in multiple pricing */ - REAL *bsolveVal; /* rows+1: bsolved solution vector for reduced costs */ - int *bsolveIdx; /* rows+1: Non-zero indeces of bsolveVal */ - - /* RHS storage */ - REAL *orig_rhs; /* rows_alloc+1 : The RHS after scaling and sign - changing, but before 'Bound transformation' */ - LREAL *rhs; /* rows_alloc+1 : The RHS of the current simplex tableau */ - - /* Row (constraint) parameters */ - int *row_type; /* rows_alloc+1 : Row/constraint type coding */ - - /* Optionally specify data for dual long-step */ - multirec *longsteps; - - /* Original and working row and variable bounds */ - REAL *orig_upbo; /* sum_alloc+1 : Bound before transformations */ - REAL *upbo; /* " " : Upper bound after transformation and B&B work */ - REAL *orig_lowbo; /* " " */ - REAL *lowbo; /* " " : Lower bound after transformation and B&B work */ - - /* User data and basis factorization matrices (ETA or LU, product form) */ - MATrec *matA; - INVrec *invB; - - /* Basis and bounds */ - BBrec *bb_bounds; /* The linked list of B&B bounds */ - BBrec *rootbounds; /* The bounds at the lowest B&B level */ - basisrec *bb_basis; /* The linked list of B&B bases */ - basisrec *rootbasis; - OBJmonrec *monitor; /* Objective monitoring record for stalling/degeneracy handling */ - - /* Scaling parameters */ - REAL *scalars; /* sum_alloc+1:0..Rows the scaling of the rows, - Rows+1..Sum the scaling of the columns */ - MYBOOL scaling_used; /* TRUE if scaling is used */ - MYBOOL columns_scaled; /* TRUE if the columns are scaled too */ - MYBOOL varmap_locked; /* Determines whether the var_to_orig and orig_to_var are fixed */ - - /* Variable state information */ - MYBOOL basis_valid; /* TRUE is the basis is still valid */ - int crashmode; /* Basis crashing mode (or none) */ - int *var_basic; /* rows_alloc+1: The list of columns in the basis */ - REAL *val_nonbasic; /* Array to store current values of non-basic variables */ - MYBOOL *is_basic; /* sum_alloc+1: TRUE if the column is in the basis */ - MYBOOL *is_lower; /* " " : TRUE if the variable is at its - lower bound (or in the basis), FALSE otherwise */ - - /* Simplex basis indicators */ - int *rejectpivot; /* List of unacceptable pivot choices due to division-by-zero */ - BBPSrec *bb_PseudoCost; /* Data structure for costing of node branchings */ - int bb_PseudoUpdates; /* Maximum number of updates for pseudo-costs */ - int bb_strongbranches; /* The number of strong B&B branches performed */ - int is_strongbranch; /* Are we currently in a strong branch mode? */ - int bb_improvements; /* The number of discrete B&B objective improvement steps */ - - /* Solver working variables */ - REAL rhsmax; /* The maximum |value| of the rhs vector at any iteration */ - REAL suminfeas; /* The working sum of primal and dual infeasibilities */ - REAL bigM; /* Original objective weighting in primal phase 1 */ - REAL P1extraVal; /* Phase 1 OF/RHS offset for feasibility */ - int P1extraDim; /* Phase 1 additional columns/rows for feasibility */ - int spx_action; /* ACTION_ variables for the simplex routine */ - MYBOOL spx_perturbed; /* The variable bounds were relaxed/perturbed into this simplex */ - MYBOOL bb_break; /* Solver working variable; signals break of the B&B */ - MYBOOL wasPreprocessed; /* The solve preprocessing was performed */ - MYBOOL wasPresolved; /* The solve presolver was invoked */ - int INTfuture2; - - /* Lagragean solver storage and parameters */ - MATrec *matL; - REAL *lag_rhs; /* Array of Lagrangean rhs vector */ - int *lag_con_type; /* Array of GT, LT or EQ */ - REAL *lambda; /* Lambda values (Lagrangean multipliers) */ - /* REAL lag_bound; */ /* The Lagrangian lower OF bound */ - /* REAL lag_accept; */ /* The Lagrangian convergence criterion */ - - REAL accuracy; - REAL accuracy_error; - - /* Solver thresholds */ - REAL infinite; /* Limit for dynamic range */ - REAL negrange; /* Limit for negative variable range */ - REAL epsmachine; /* Default machine accuracy */ - REAL epsvalue; /* Input data precision / rounding of data values to 0 */ - REAL epsprimal; /* For rounding RHS values to 0/infeasibility */ - REAL epsdual; /* For rounding reduced costs to zero */ - REAL epspivot; /* Pivot reject tolerance */ - REAL epsperturb; /* Perturbation scalar */ - REAL epssolution; /* The solution tolerance for final validation */ - - /* Branch & Bound working parameters */ - int bb_status; /* Indicator that the last solvelp() gave an improved B&B solution */ - int bb_level; /* Solver B&B working variable (recursion depth) */ - int bb_maxlevel; /* The deepest B&B level of the last solution */ - int bb_limitlevel; /* The maximum B&B level allowed */ - COUNTER bb_totalnodes; /* Total number of nodes processed in B&B */ - int bb_solutionlevel; /* The B&B level of the last / best solution */ - int bb_cutpoolsize; /* Size of the B&B cut pool */ - int bb_cutpoolused; /* Currently used cut pool */ - int bb_constraintOF; /* General purpose B&B parameter (typically for testing) */ - int *bb_cuttype; /* The type of the currently used cuts */ - int *bb_varactive; /* The B&B state of the variable; 0 means inactive */ - DeltaVrec *bb_upperchange; /* Changes to upper bounds during the B&B phase */ - DeltaVrec *bb_lowerchange; /* Changes to lower bounds during the B&B phase */ - - REAL bb_deltaOF; /* Minimum OF step value; computed at beginning of solve() */ - - REAL bb_breakOF; /* User-settable value for the objective function deemed - to be sufficiently good in an integer problem */ - REAL bb_limitOF; /* "Dual" bound / limit to final optimal MIP solution */ - REAL bb_heuristicOF; /* Set initial "at least better than" guess for objective function - (can significantly speed up B&B iterations) */ - REAL bb_parentOF; /* The OF value of the previous BB simplex */ - REAL bb_workOF; /* The unadjusted OF value for the current best solution */ - - /* Internal work arrays allocated as required */ - presolveundorec *presolve_undo; - workarraysrec *workarrays; - - /* MIP parameters */ - REAL epsint; /* Margin of error in determining if a float value is integer */ - REAL mip_absgap; /* Absolute MIP gap */ - REAL mip_relgap; /* Relative MIP gap */ - - /* Time/timer variables and extended status text */ - double timecreate; - double timestart; - double timeheuristic; - double timepresolved; - double timeend; - long sectimeout; - - /* Extended status message text set via explain() */ - char *ex_status; - - /* Refactorization engine interface routines (for dynamic DLL/SO BFPs) */ -#if LoadInverseLib == TRUE - #ifdef WIN32 - HINSTANCE hBFP; - #else - void *hBFP; - #endif -#endif - BFPchar *bfp_name; - BFPbool_lpintintint *bfp_compatible; - BFPbool_lpintintchar *bfp_init; - BFP_lp *bfp_free; - BFPbool_lpint *bfp_resize; - BFPint_lp *bfp_memallocated; - BFPbool_lp *bfp_restart; - BFPbool_lp *bfp_mustrefactorize; - BFPint_lp *bfp_preparefactorization; - BFPint_lpintintboolbool *bfp_factorize; - BFP_lp *bfp_finishfactorization; - BFP_lp *bfp_updaterefactstats; - BFPlreal_lpintintreal *bfp_prepareupdate; - BFPreal_lplrealreal *bfp_pivotRHS; - BFPbool_lpbool *bfp_finishupdate; - BFP_lprealint *bfp_ftran_prepare; - BFP_lprealint *bfp_ftran_normal; - BFP_lprealint *bfp_btran_normal; - BFP_lprealintrealint *bfp_btran_double; - BFPint_lp *bfp_status; - BFPint_lpbool *bfp_nonzeros; - BFPbool_lp *bfp_implicitslack; - BFPint_lp *bfp_indexbase; - BFPint_lp *bfp_rowoffset; - BFPint_lp *bfp_pivotmax; - BFPbool_lpint *bfp_pivotalloc; - BFPint_lp *bfp_colcount; - BFPbool_lp *bfp_canresetbasis; - BFPreal_lp *bfp_efficiency; - BFPrealp_lp *bfp_pivotvector; - BFPint_lp *bfp_pivotcount; - BFPint_lpint *bfp_refactcount; - BFPbool_lp *bfp_isSetI; - BFPint_lpintrealcbintint *bfp_findredundant; - - /* External language interface routines (for dynamic DLL/SO XLIs) */ -#if LoadLanguageLib == TRUE - #ifdef WIN32 - HINSTANCE hXLI; - #else - void *hXLI; - #endif -#endif - XLIchar *xli_name; - XLIbool_lpintintint *xli_compatible; - XLIbool_lpcharcharcharint *xli_readmodel; - XLIbool_lpcharcharbool *xli_writemodel; - - /* Miscellaneous internal functions made available externally */ - userabortfunc *userabort; - reportfunc *report; - explainfunc *explain; - getvectorfunc *get_lpcolumn; - getpackedfunc *get_basiscolumn; - get_OF_activefunc *get_OF_active; - getMDOfunc *getMDO; - invertfunc *invert; - set_actionfunc *set_action; - is_actionfunc *is_action; - clear_actionfunc *clear_action; - - /* User program interface callbacks */ - lphandle_intfunc *ctrlc; - void *ctrlchandle; /* User-specified "owner process ID" */ - lphandlestr_func *writelog; - void *loghandle; /* User-specified "owner process ID" */ - lphandlestr_func *debuginfo; - lphandleint_func *usermessage; - int msgmask; - void *msghandle; /* User-specified "owner process ID" */ - lphandleint_intfunc *bb_usenode; - void *bb_nodehandle; /* User-specified "owner process ID" */ - lphandleint_intfunc *bb_usebranch; - void *bb_branchhandle; /* User-specified "owner process ID" */ - - /* replacement of static variables */ - char *rowcol_name; /* The name of a row/column */ -}; - - -#ifdef __cplusplus -__EXTERN_C { -#endif - - -/* User and system function interfaces */ -/* ------------------------------------------------------------------------- */ - -void __EXPORT_TYPE __WINAPI lp_solve_version(int *majorversion, int *minorversion, int *release, int *build); - -lprec __EXPORT_TYPE * __WINAPI make_lp(int rows, int columns); -MYBOOL __EXPORT_TYPE __WINAPI resize_lp(lprec *lp, int rows, int columns); -int __EXPORT_TYPE __WINAPI get_status(lprec *lp); -char __EXPORT_TYPE * __WINAPI get_statustext(lprec *lp, int statuscode); -MYBOOL __EXPORT_TYPE __WINAPI is_obj_in_basis(lprec *lp); -void __EXPORT_TYPE __WINAPI set_obj_in_basis(lprec *lp, MYBOOL obj_in_basis); -/* Create and initialise a lprec structure defaults */ - -lprec __EXPORT_TYPE * __WINAPI copy_lp(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI dualize_lp(lprec *lp); -STATIC MYBOOL memopt_lp(lprec *lp, int rowextra, int colextra, int nzextra); -/* Copy or dualize the lp */ - -void __EXPORT_TYPE __WINAPI delete_lp(lprec *lp); -void __EXPORT_TYPE __WINAPI free_lp(lprec **plp); -/* Remove problem from memory */ - -MYBOOL __EXPORT_TYPE __WINAPI set_lp_name(lprec *lp, char *lpname); -char __EXPORT_TYPE * __WINAPI get_lp_name(lprec *lp); -/* Set and get the problem name */ - -MYBOOL __EXPORT_TYPE __WINAPI has_BFP(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI is_nativeBFP(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI set_BFP(lprec *lp, char *filename); -/* Set basis factorization engine */ - -lprec __EXPORT_TYPE * __WINAPI read_XLI(char *xliname, char *modelname, char *dataname, char *options, int verbose); -MYBOOL __EXPORT_TYPE __WINAPI write_XLI(lprec *lp, char *filename, char *options, MYBOOL results); -MYBOOL __EXPORT_TYPE __WINAPI has_XLI(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI is_nativeXLI(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI set_XLI(lprec *lp, char *filename); -/* Set external language interface */ - -MYBOOL __EXPORT_TYPE __WINAPI set_obj(lprec *lp, int colnr, REAL value); -MYBOOL __EXPORT_TYPE __WINAPI set_obj_fn(lprec *lp, REAL *row); -MYBOOL __EXPORT_TYPE __WINAPI set_obj_fnex(lprec *lp, int count, REAL *row, int *colno); -/* set the objective function (Row 0) of the matrix */ -MYBOOL __EXPORT_TYPE __WINAPI str_set_obj_fn(lprec *lp, char *row_string); -/* The same, but with string input */ -void __EXPORT_TYPE __WINAPI set_sense(lprec *lp, MYBOOL maximize); -void __EXPORT_TYPE __WINAPI set_maxim(lprec *lp); -void __EXPORT_TYPE __WINAPI set_minim(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI is_maxim(lprec *lp); -/* Set optimization direction for the objective function */ - -MYBOOL __EXPORT_TYPE __WINAPI add_constraint(lprec *lp, REAL *row, int constr_type, REAL rh); -MYBOOL __EXPORT_TYPE __WINAPI add_constraintex(lprec *lp, int count, REAL *row, int *colno, int constr_type, REAL rh); -MYBOOL __EXPORT_TYPE __WINAPI set_add_rowmode(lprec *lp, MYBOOL turnon); -MYBOOL __EXPORT_TYPE __WINAPI is_add_rowmode(lprec *lp); -/* Add a constraint to the problem, row is the constraint row, rh is the right hand side, - constr_type is the type of constraint (LE (<=), GE(>=), EQ(=)) */ -MYBOOL __EXPORT_TYPE __WINAPI str_add_constraint(lprec *lp, char *row_string, int constr_type, REAL rh); -/* The same, but with string input */ - -MYBOOL __EXPORT_TYPE __WINAPI set_row(lprec *lp, int rownr, REAL *row); -MYBOOL __EXPORT_TYPE __WINAPI set_rowex(lprec *lp, int rownr, int count, REAL *row, int *colno); -MYBOOL __EXPORT_TYPE __WINAPI get_row(lprec *lp, int rownr, REAL *row); -int __EXPORT_TYPE __WINAPI get_rowex(lprec *lp, int rownr, REAL *row, int *colno); -/* Fill row with the row row_nr from the problem */ - -MYBOOL __EXPORT_TYPE __WINAPI del_constraint(lprec *lp, int rownr); -STATIC MYBOOL del_constraintex(lprec *lp, LLrec *rowmap); -/* Remove constrain nr del_row from the problem */ - -MYBOOL __EXPORT_TYPE __WINAPI add_lag_con(lprec *lp, REAL *row, int con_type, REAL rhs); -/* add a Lagrangian constraint of form Row' x contype Rhs */ -MYBOOL __EXPORT_TYPE __WINAPI str_add_lag_con(lprec *lp, char *row_string, int con_type, REAL rhs); -/* The same, but with string input */ -void __EXPORT_TYPE __WINAPI set_lag_trace(lprec *lp, MYBOOL lag_trace); -MYBOOL __EXPORT_TYPE __WINAPI is_lag_trace(lprec *lp); -/* Set debugging/tracing mode of the Lagrangean solver */ - -MYBOOL __EXPORT_TYPE __WINAPI set_constr_type(lprec *lp, int rownr, int con_type); -int __EXPORT_TYPE __WINAPI get_constr_type(lprec *lp, int rownr); -REAL __EXPORT_TYPE __WINAPI get_constr_value(lprec *lp, int rownr, int count, REAL *primsolution, int *nzindex); -MYBOOL __EXPORT_TYPE __WINAPI is_constr_type(lprec *lp, int rownr, int mask); -STATIC char *get_str_constr_type(lprec *lp, int con_type); -STATIC int get_constr_class(lprec *lp, int rownr); -STATIC char *get_str_constr_class(lprec *lp, int con_class); -/* Set the type of constraint in row Row (LE, GE, EQ) */ - -MYBOOL __EXPORT_TYPE __WINAPI set_rh(lprec *lp, int rownr, REAL value); -REAL __EXPORT_TYPE __WINAPI get_rh(lprec *lp, int rownr); -/* Set and get the right hand side of a constraint row */ -MYBOOL __EXPORT_TYPE __WINAPI set_rh_range(lprec *lp, int rownr, REAL deltavalue); -REAL __EXPORT_TYPE __WINAPI get_rh_range(lprec *lp, int rownr); -/* Set the RHS range; i.e. the lower and upper bounds of a constraint row */ -void __EXPORT_TYPE __WINAPI set_rh_vec(lprec *lp, REAL *rh); -/* Set the right hand side vector */ -MYBOOL __EXPORT_TYPE __WINAPI str_set_rh_vec(lprec *lp, char *rh_string); -/* The same, but with string input */ - -MYBOOL __EXPORT_TYPE __WINAPI add_column(lprec *lp, REAL *column); -MYBOOL __EXPORT_TYPE __WINAPI add_columnex(lprec *lp, int count, REAL *column, int *rowno); -MYBOOL __EXPORT_TYPE __WINAPI str_add_column(lprec *lp, char *col_string); -/* Add a column to the problem */ - -MYBOOL __EXPORT_TYPE __WINAPI set_column(lprec *lp, int colnr, REAL *column); -MYBOOL __EXPORT_TYPE __WINAPI set_columnex(lprec *lp, int colnr, int count, REAL *column, int *rowno); -/* Overwrite existing column data */ - -int __EXPORT_TYPE __WINAPI column_in_lp(lprec *lp, REAL *column); -/* Returns the column index if column is already present in lp, otherwise 0. - (Does not look at bounds and types, only looks at matrix values */ - -int __EXPORT_TYPE __WINAPI get_columnex(lprec *lp, int colnr, REAL *column, int *nzrow); -MYBOOL __EXPORT_TYPE __WINAPI get_column(lprec *lp, int colnr, REAL *column); -/* Fill column with the column col_nr from the problem */ - -MYBOOL __EXPORT_TYPE __WINAPI del_column(lprec *lp, int colnr); -STATIC MYBOOL del_columnex(lprec *lp, LLrec *colmap); -/* Delete a column */ - -MYBOOL __EXPORT_TYPE __WINAPI set_mat(lprec *lp, int rownr, int colnr, REAL value); -/* Fill in element (Row,Column) of the matrix - Row in [0..Rows] and Column in [1..Columns] */ -REAL __EXPORT_TYPE __WINAPI get_mat(lprec *lp, int rownr, int colnr); -REAL __EXPORT_TYPE __WINAPI get_mat_byindex(lprec *lp, int matindex, MYBOOL isrow, MYBOOL adjustsign); -int __EXPORT_TYPE __WINAPI get_nonzeros(lprec *lp); -/* get a single element from the matrix */ /* Name changed from "mat_elm" by KE */ - -void __EXPORT_TYPE __WINAPI set_bounds_tighter(lprec *lp, MYBOOL tighten); -MYBOOL get_bounds(lprec *lp, int column, REAL *lower, REAL *upper); -MYBOOL __EXPORT_TYPE __WINAPI get_bounds_tighter(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI set_upbo(lprec *lp, int colnr, REAL value); -REAL __EXPORT_TYPE __WINAPI get_upbo(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI set_lowbo(lprec *lp, int colnr, REAL value); -REAL __EXPORT_TYPE __WINAPI get_lowbo(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI set_bounds(lprec *lp, int colnr, REAL lower, REAL upper); -MYBOOL __EXPORT_TYPE __WINAPI set_unbounded(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI is_unbounded(lprec *lp, int colnr); -/* Set the upper and lower bounds of a variable */ - -MYBOOL __EXPORT_TYPE __WINAPI set_int(lprec *lp, int colnr, MYBOOL must_be_int); -MYBOOL __EXPORT_TYPE __WINAPI is_int(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI set_binary(lprec *lp, int colnr, MYBOOL must_be_bin); -MYBOOL __EXPORT_TYPE __WINAPI is_binary(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI set_semicont(lprec *lp, int colnr, MYBOOL must_be_sc); -MYBOOL __EXPORT_TYPE __WINAPI is_semicont(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI is_negative(lprec *lp, int colnr); -MYBOOL __EXPORT_TYPE __WINAPI set_var_weights(lprec *lp, REAL *weights); -int __EXPORT_TYPE __WINAPI get_var_priority(lprec *lp, int colnr); -/* Set the type of variable */ - -MYBOOL __EXPORT_TYPE __WINAPI set_pseudocosts(lprec *lp, REAL *clower, REAL *cupper, int *updatelimit); -MYBOOL __EXPORT_TYPE __WINAPI get_pseudocosts(lprec *lp, REAL *clower, REAL *cupper, int *updatelimit); -/* Set initial values for, or get computed pseudocost vectors; - note that setting of pseudocosts can only happen in response to a - call-back function optionally requesting this */ - -int __EXPORT_TYPE __WINAPI add_SOS(lprec *lp, char *name, int sostype, int priority, int count, int *sosvars, REAL *weights); -MYBOOL __EXPORT_TYPE __WINAPI is_SOS_var(lprec *lp, int colnr); -/* Add SOS constraints */ - -MYBOOL __EXPORT_TYPE __WINAPI set_row_name(lprec *lp, int rownr, char *new_name); -char __EXPORT_TYPE * __WINAPI get_row_name(lprec *lp, int rownr); -char __EXPORT_TYPE * __WINAPI get_origrow_name(lprec *lp, int rownr); -/* Set/Get the name of a constraint row */ /* Get added by KE */ - -MYBOOL __EXPORT_TYPE __WINAPI set_col_name(lprec *lp, int colnr, char *new_name); -char __EXPORT_TYPE * __WINAPI get_col_name(lprec *lp, int colnr); -char __EXPORT_TYPE * __WINAPI get_origcol_name(lprec *lp, int colnr); -/* Set/Get the name of a variable column */ /* Get added by KE */ - -void __EXPORT_TYPE __WINAPI unscale(lprec *lp); -/* Undo previous scaling of the problem */ - -void __EXPORT_TYPE __WINAPI set_preferdual(lprec *lp, MYBOOL dodual); -void __EXPORT_TYPE __WINAPI set_simplextype(lprec *lp, int simplextype); -int __EXPORT_TYPE __WINAPI get_simplextype(lprec *lp); -/* Set/Get if lp_solve should prefer the dual simplex over the primal -- added by KE */ - -void __EXPORT_TYPE __WINAPI default_basis(lprec *lp); -void __EXPORT_TYPE __WINAPI set_basiscrash(lprec *lp, int mode); -int __EXPORT_TYPE __WINAPI get_basiscrash(lprec *lp); -int __EXPORT_TYPE __WINAPI set_basisvar(lprec *lp, int basisPos, int enteringCol); -MYBOOL __EXPORT_TYPE __WINAPI set_basis(lprec *lp, int *bascolumn, MYBOOL nonbasic); -MYBOOL __EXPORT_TYPE __WINAPI get_basis(lprec *lp, int *bascolumn, MYBOOL nonbasic); -void __EXPORT_TYPE __WINAPI reset_basis(lprec *lp); -/* Set/Get basis for a re-solved system */ /* Added by KE */ -MYBOOL __EXPORT_TYPE __WINAPI guess_basis(lprec *lp, REAL *guessvector, int *basisvector); - -MYBOOL __EXPORT_TYPE __WINAPI is_feasible(lprec *lp, REAL *values, REAL threshold); -/* returns TRUE if the vector in values is a feasible solution to the lp */ - -int __EXPORT_TYPE __WINAPI solve(lprec *lp); -/* Solve the problem */ - -REAL __EXPORT_TYPE __WINAPI get_accuracy(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_break_numeric_accuracy(lprec *lp, REAL accuracy); -REAL __EXPORT_TYPE __WINAPI get_break_numeric_accuracy(lprec *lp); - -REAL __EXPORT_TYPE __WINAPI time_elapsed(lprec *lp); -/* Return the number of seconds since start of solution process */ - -void __EXPORT_TYPE __WINAPI put_bb_nodefunc(lprec *lp, lphandleint_intfunc newnode, void *bbnodehandle); -void __EXPORT_TYPE __WINAPI put_bb_branchfunc(lprec *lp, lphandleint_intfunc newbranch, void *bbbranchhandle); -/* Allow the user to override B&B node and branching decisions */ - -void __EXPORT_TYPE __WINAPI put_abortfunc(lprec *lp, lphandle_intfunc newctrlc, void *ctrlchandle); -/* Allow the user to define an interruption callback function */ - -void __EXPORT_TYPE __WINAPI put_logfunc(lprec *lp, lphandlestr_func newlog, void *loghandle); -/* Allow the user to define a logging function */ - -void __EXPORT_TYPE __WINAPI put_msgfunc(lprec *lp, lphandleint_func newmsg, void *msghandle, int mask); -/* Allow the user to define an event-driven message/reporting */ - -MYBOOL __EXPORT_TYPE __WINAPI get_primal_solution(lprec *lp, REAL *pv); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_primal_solution(lprec *lp, REAL **pv); -MYBOOL __EXPORT_TYPE __WINAPI get_dual_solution(lprec *lp, REAL *rc); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_dual_solution(lprec *lp, REAL **rc); -MYBOOL __EXPORT_TYPE __WINAPI get_lambda(lprec *lp, REAL *lambda); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_lambda(lprec *lp, REAL **lambda); -/* Get the primal, dual/reduced costs and Lambda vectors */ - -/* Read an MPS file */ -lprec __EXPORT_TYPE * __WINAPI read_MPS(char *filename, int options); -lprec __EXPORT_TYPE * __WINAPI read_mps(FILE *filename, int options); -lprec __EXPORT_TYPE * __WINAPI read_freeMPS(char *filename, int options); -lprec __EXPORT_TYPE * __WINAPI read_freemps(FILE *filename, int options); - -/* Write a MPS file to output */ -MYBOOL __EXPORT_TYPE __WINAPI write_mps(lprec *lp, char *filename); -MYBOOL __EXPORT_TYPE __WINAPI write_MPS(lprec *lp, FILE *output); -MYBOOL __EXPORT_TYPE __WINAPI write_freemps(lprec *lp, char *filename); -MYBOOL __EXPORT_TYPE __WINAPI write_freeMPS(lprec *lp, FILE *output); - -MYBOOL __EXPORT_TYPE __WINAPI write_lp(lprec *lp, char *filename); -MYBOOL __EXPORT_TYPE __WINAPI write_LP(lprec *lp, FILE *output); - /* Write a LP file to output */ - -MYBOOL __WINAPI LP_readhandle(lprec **lp, FILE *filename, int verbose, char *lp_name); -lprec __EXPORT_TYPE * __WINAPI read_lp(FILE *filename, int verbose, char *lp_name); -lprec __EXPORT_TYPE * __WINAPI read_LP(char *filename, int verbose, char *lp_name); -/* Old-style lp format file parser */ - -MYBOOL __EXPORT_TYPE __WINAPI write_basis(lprec *lp, char *filename); -MYBOOL __EXPORT_TYPE __WINAPI read_basis(lprec *lp, char *filename, char *info); -/* Read and write basis from/to file in CPLEX BAS format */ - -MYBOOL __EXPORT_TYPE __WINAPI write_params(lprec *lp, char *filename, char *options); -MYBOOL __EXPORT_TYPE __WINAPI read_params(lprec *lp, char *filename, char *options); -void __EXPORT_TYPE __WINAPI reset_params(lprec *lp); -/* Read and write parameter file */ - -void __EXPORT_TYPE __WINAPI print_lp(lprec *lp); -void __EXPORT_TYPE __WINAPI print_tableau(lprec *lp); -/* Print the current problem, only useful in very small (test) problems */ - -void __EXPORT_TYPE __WINAPI print_objective(lprec *lp); -void __EXPORT_TYPE __WINAPI print_solution(lprec *lp, int columns); -void __EXPORT_TYPE __WINAPI print_constraints(lprec *lp, int columns); -/* Print the solution to stdout */ - -void __EXPORT_TYPE __WINAPI print_duals(lprec *lp); -/* Print the dual variables of the solution */ - -void __EXPORT_TYPE __WINAPI print_scales(lprec *lp); -/* If scaling is used, print the scaling factors */ - -void __EXPORT_TYPE __WINAPI print_str(lprec *lp, char *str); - -void __EXPORT_TYPE __WINAPI set_outputstream(lprec *lp, FILE *stream); -MYBOOL __EXPORT_TYPE __WINAPI set_outputfile(lprec *lp, char *filename); - -void __EXPORT_TYPE __WINAPI set_verbose(lprec *lp, int verbose); -int __EXPORT_TYPE __WINAPI get_verbose(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_timeout(lprec *lp, long sectimeout); -long __EXPORT_TYPE __WINAPI get_timeout(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_print_sol(lprec *lp, int print_sol); -int __EXPORT_TYPE __WINAPI get_print_sol(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_debug(lprec *lp, MYBOOL debug); -MYBOOL __EXPORT_TYPE __WINAPI is_debug(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_trace(lprec *lp, MYBOOL trace); -MYBOOL __EXPORT_TYPE __WINAPI is_trace(lprec *lp); - -MYBOOL __EXPORT_TYPE __WINAPI print_debugdump(lprec *lp, char *filename); - -void __EXPORT_TYPE __WINAPI set_anti_degen(lprec *lp, int anti_degen); -int __EXPORT_TYPE __WINAPI get_anti_degen(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI is_anti_degen(lprec *lp, int testmask); - -void __EXPORT_TYPE __WINAPI set_presolve(lprec *lp, int presolvemode, int maxloops); -int __EXPORT_TYPE __WINAPI get_presolve(lprec *lp); -int __EXPORT_TYPE __WINAPI get_presolveloops(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI is_presolve(lprec *lp, int testmask); - -int __EXPORT_TYPE __WINAPI get_orig_index(lprec *lp, int lp_index); -int __EXPORT_TYPE __WINAPI get_lp_index(lprec *lp, int orig_index); - -void __EXPORT_TYPE __WINAPI set_maxpivot(lprec *lp, int max_num_inv); -int __EXPORT_TYPE __WINAPI get_maxpivot(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_obj_bound(lprec *lp, REAL obj_bound); -REAL __EXPORT_TYPE __WINAPI get_obj_bound(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_mip_gap(lprec *lp, MYBOOL absolute, REAL mip_gap); -REAL __EXPORT_TYPE __WINAPI get_mip_gap(lprec *lp, MYBOOL absolute); - -void __EXPORT_TYPE __WINAPI set_bb_rule(lprec *lp, int bb_rule); -int __EXPORT_TYPE __WINAPI get_bb_rule(lprec *lp); - -MYBOOL __EXPORT_TYPE __WINAPI set_var_branch(lprec *lp, int colnr, int branch_mode); -int __EXPORT_TYPE __WINAPI get_var_branch(lprec *lp, int colnr); - -MYBOOL __EXPORT_TYPE __WINAPI is_infinite(lprec *lp, REAL value); -void __EXPORT_TYPE __WINAPI set_infinite(lprec *lp, REAL infinite); -REAL __EXPORT_TYPE __WINAPI get_infinite(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_epsint(lprec *lp, REAL epsint); -REAL __EXPORT_TYPE __WINAPI get_epsint(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_epsb(lprec *lp, REAL epsb); -REAL __EXPORT_TYPE __WINAPI get_epsb(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_epsd(lprec *lp, REAL epsd); -REAL __EXPORT_TYPE __WINAPI get_epsd(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_epsel(lprec *lp, REAL epsel); -REAL __EXPORT_TYPE __WINAPI get_epsel(lprec *lp); - -MYBOOL __EXPORT_TYPE __WINAPI set_epslevel(lprec *lp, int epslevel); - -void __EXPORT_TYPE __WINAPI set_scaling(lprec *lp, int scalemode); -int __EXPORT_TYPE __WINAPI get_scaling(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI is_scalemode(lprec *lp, int testmask); -MYBOOL __EXPORT_TYPE __WINAPI is_scaletype(lprec *lp, int scaletype); -MYBOOL __EXPORT_TYPE __WINAPI is_integerscaling(lprec *lp); -void __EXPORT_TYPE __WINAPI set_scalelimit(lprec *lp, REAL scalelimit); -REAL __EXPORT_TYPE __WINAPI get_scalelimit(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_improve(lprec *lp, int improve); -int __EXPORT_TYPE __WINAPI get_improve(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_pivoting(lprec *lp, int piv_rule); -int __EXPORT_TYPE __WINAPI get_pivoting(lprec *lp); -MYBOOL __EXPORT_TYPE __WINAPI set_partialprice(lprec *lp, int blockcount, int *blockstart, MYBOOL isrow); -void __EXPORT_TYPE __WINAPI get_partialprice(lprec *lp, int *blockcount, int *blockstart, MYBOOL isrow); - -MYBOOL __EXPORT_TYPE __WINAPI set_multiprice(lprec *lp, int multiblockdiv); -int __EXPORT_TYPE __WINAPI get_multiprice(lprec *lp, MYBOOL getabssize); - -MYBOOL __EXPORT_TYPE __WINAPI is_use_names(lprec *lp, MYBOOL isrow); -void __EXPORT_TYPE __WINAPI set_use_names(lprec *lp, MYBOOL isrow, MYBOOL use_names); - -int __EXPORT_TYPE __WINAPI get_nameindex(lprec *lp, char *varname, MYBOOL isrow); - -MYBOOL __EXPORT_TYPE __WINAPI is_piv_mode(lprec *lp, int testmask); -MYBOOL __EXPORT_TYPE __WINAPI is_piv_rule(lprec *lp, int rule); - -void __EXPORT_TYPE __WINAPI set_break_at_first(lprec *lp, MYBOOL break_at_first); -MYBOOL __EXPORT_TYPE __WINAPI is_break_at_first(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_bb_floorfirst(lprec *lp, int bb_floorfirst); -int __EXPORT_TYPE __WINAPI get_bb_floorfirst(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_bb_depthlimit(lprec *lp, int bb_maxlevel); -int __EXPORT_TYPE __WINAPI get_bb_depthlimit(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_break_at_value(lprec *lp, REAL break_at_value); -REAL __EXPORT_TYPE __WINAPI get_break_at_value(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_negrange(lprec *lp, REAL negrange); -REAL __EXPORT_TYPE __WINAPI get_negrange(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_epsperturb(lprec *lp, REAL epsperturb); -REAL __EXPORT_TYPE __WINAPI get_epsperturb(lprec *lp); - -void __EXPORT_TYPE __WINAPI set_epspivot(lprec *lp, REAL epspivot); -REAL __EXPORT_TYPE __WINAPI get_epspivot(lprec *lp); - -int __EXPORT_TYPE __WINAPI get_max_level(lprec *lp); -COUNTER __EXPORT_TYPE __WINAPI get_total_nodes(lprec *lp); -COUNTER __EXPORT_TYPE __WINAPI get_total_iter(lprec *lp); - -REAL __EXPORT_TYPE __WINAPI get_objective(lprec *lp); -REAL __EXPORT_TYPE __WINAPI get_working_objective(lprec *lp); - -REAL __EXPORT_TYPE __WINAPI get_var_primalresult(lprec *lp, int index); -REAL __EXPORT_TYPE __WINAPI get_var_dualresult(lprec *lp, int index); - -MYBOOL __EXPORT_TYPE __WINAPI get_variables(lprec *lp, REAL *var); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_variables(lprec *lp, REAL **var); - -MYBOOL __EXPORT_TYPE __WINAPI get_constraints(lprec *lp, REAL *constr); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_constraints(lprec *lp, REAL **constr); - -MYBOOL __EXPORT_TYPE __WINAPI get_sensitivity_rhs(lprec *lp, REAL *duals, REAL *dualsfrom, REAL *dualstill); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_sensitivity_rhs(lprec *lp, REAL **duals, REAL **dualsfrom, REAL **dualstill); - -MYBOOL __EXPORT_TYPE __WINAPI get_sensitivity_obj(lprec *lp, REAL *objfrom, REAL *objtill); -MYBOOL __EXPORT_TYPE __WINAPI get_sensitivity_objex(lprec *lp, REAL *objfrom, REAL *objtill, REAL *objfromvalue, REAL *objtillvalue); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_sensitivity_obj(lprec *lp, REAL **objfrom, REAL **objtill); -MYBOOL __EXPORT_TYPE __WINAPI get_ptr_sensitivity_objex(lprec *lp, REAL **objfrom, REAL **objtill, REAL **objfromvalue, REAL **objtillvalue); - -void __EXPORT_TYPE __WINAPI set_solutionlimit(lprec *lp, int limit); -int __EXPORT_TYPE __WINAPI get_solutionlimit(lprec *lp); -int __EXPORT_TYPE __WINAPI get_solutioncount(lprec *lp); - -int __EXPORT_TYPE __WINAPI get_Norig_rows(lprec *lp); -int __EXPORT_TYPE __WINAPI get_Nrows(lprec *lp); -int __EXPORT_TYPE __WINAPI get_Lrows(lprec *lp); - -int __EXPORT_TYPE __WINAPI get_Norig_columns(lprec *lp); -int __EXPORT_TYPE __WINAPI get_Ncolumns(lprec *lp); - -typedef int (__WINAPI read_modeldata_func)(void *userhandle, char *buf, int max_size); -typedef int (__WINAPI write_modeldata_func)(void *userhandle, char *buf); -MYBOOL __WINAPI MPS_readex(lprec **newlp, void *userhandle, read_modeldata_func read_modeldata, int typeMPS, int options); - -/* #if defined develop */ -lprec __EXPORT_TYPE * __WINAPI read_lpex(void *userhandle, read_modeldata_func read_modeldata, int verbose, char *lp_name); -MYBOOL __EXPORT_TYPE __WINAPI write_lpex(lprec *lp, void *userhandle, write_modeldata_func write_modeldata); - -lprec __EXPORT_TYPE * __WINAPI read_mpsex(void *userhandle, read_modeldata_func read_modeldata, int options); -lprec __EXPORT_TYPE * __WINAPI read_freempsex(void *userhandle, read_modeldata_func read_modeldata, int options); - -MYBOOL __EXPORT_TYPE __WINAPI MPS_writefileex(lprec *lp, int typeMPS, void *userhandle, write_modeldata_func write_modeldata); -/* #endif */ - -#ifdef __cplusplus -} -#endif - - -/* Forward definitions of functions used internaly by the lp toolkit */ -MYBOOL set_callbacks(lprec *lp); -STATIC int yieldformessages(lprec *lp); -MYBOOL __WINAPI userabort(lprec *lp, int message); -/*char * __VACALL explain(lprec *lp, char *format, ...); -void __VACALL report(lprec *lp, int level, char *format, ...);*/ - -/* Memory management routines */ -STATIC MYBOOL append_rows(lprec *lp, int deltarows); -STATIC MYBOOL append_columns(lprec *lp, int deltacolumns); -STATIC void inc_rows(lprec *lp, int delta); -STATIC void inc_columns(lprec *lp, int delta); -STATIC MYBOOL init_rowcol_names(lprec *lp); -STATIC MYBOOL inc_row_space(lprec *lp, int deltarows); -STATIC MYBOOL inc_col_space(lprec *lp, int deltacols); -STATIC MYBOOL shift_rowcoldata(lprec *lp, int base, int delta, LLrec *usedmap, MYBOOL isrow); -STATIC MYBOOL shift_basis(lprec *lp, int base, int delta, LLrec *usedmap, MYBOOL isrow); -STATIC MYBOOL shift_rowdata(lprec *lp, int base, int delta, LLrec *usedmap); -STATIC MYBOOL shift_coldata(lprec *lp, int base, int delta, LLrec *usedmap); - -/* INLINE */ MYBOOL is_chsign(lprec *lp, int rownr); - -STATIC MYBOOL inc_lag_space(lprec *lp, int deltarows, MYBOOL ignoreMAT); -lprec *make_lag(lprec *server); - -REAL get_rh_upper(lprec *lp, int rownr); -REAL get_rh_lower(lprec *lp, int rownr); -MYBOOL set_rh_upper(lprec *lp, int rownr, REAL value); -MYBOOL set_rh_lower(lprec *lp, int rownr, REAL value); -STATIC int bin_count(lprec *lp, MYBOOL working); -STATIC int MIP_count(lprec *lp); -STATIC int SOS_count(lprec *lp); -STATIC int GUB_count(lprec *lp); -STATIC int identify_GUB(lprec *lp, MYBOOL mark); -STATIC int prepare_GUB(lprec *lp); - -STATIC MYBOOL refactRecent(lprec *lp); -STATIC MYBOOL check_if_less(lprec *lp, REAL x, REAL y, int variable); -STATIC MYBOOL feasiblePhase1(lprec *lp, REAL epsvalue); -STATIC void free_duals(lprec *lp); -STATIC void initialize_solution(lprec *lp, MYBOOL shiftbounds); -STATIC void recompute_solution(lprec *lp, MYBOOL shiftbounds); -STATIC int verify_solution(lprec *lp, MYBOOL reinvert, char *info); -STATIC int check_solution(lprec *lp, int lastcolumn, REAL *solution, - REAL *upbo, REAL *lowbo, REAL tolerance); -/* INLINE */ MYBOOL is_fixedvar(lprec *lp, int variable); -/* INLINE */ MYBOOL is_splitvar(lprec *lp, int colnr); - -void __WINAPI set_action(int *actionvar, int actionmask); -void __WINAPI clear_action(int *actionvar, int actionmask); -MYBOOL __WINAPI is_action(int actionvar, int testmask); - -/* INLINE */ MYBOOL is_bb_rule(lprec *lp, int bb_rule); -/* INLINE */ MYBOOL is_bb_mode(lprec *lp, int bb_mask); -/* INLINE */ int get_piv_rule(lprec *lp); -STATIC char *get_str_piv_rule(int rule); -STATIC MYBOOL __WINAPI set_var_priority(lprec *lp); -STATIC int find_sc_bbvar(lprec *lp, int *count); -STATIC int find_sos_bbvar(lprec *lp, int *count, MYBOOL intsos); -STATIC int find_int_bbvar(lprec *lp, int *count, BBrec *BB, MYBOOL *isfeasible); - -/* Solution-related functions */ -STATIC REAL compute_dualslacks(lprec *lp, int target, REAL **dvalues, int **nzdvalues, MYBOOL dosum); -STATIC MYBOOL solution_is_int(lprec *lp, int index, MYBOOL checkfixed); -STATIC MYBOOL bb_better(lprec *lp, int target, int mode); -STATIC void construct_solution(lprec *lp, REAL *target); -STATIC void transfer_solution_var(lprec *lp, int uservar); -STATIC MYBOOL construct_duals(lprec *lp); -STATIC MYBOOL construct_sensitivity_duals(lprec *lp); -STATIC MYBOOL construct_sensitivity_obj(lprec *lp); - -STATIC int add_GUB(lprec *lp, char *name, int priority, int count, int *sosvars); -STATIC basisrec *push_basis(lprec *lp, int *basisvar, MYBOOL *isbasic, MYBOOL *islower); -STATIC MYBOOL compare_basis(lprec *lp); -STATIC MYBOOL restore_basis(lprec *lp); -STATIC MYBOOL pop_basis(lprec *lp, MYBOOL restore); -STATIC MYBOOL is_BasisReady(lprec *lp); -STATIC MYBOOL is_slackbasis(lprec *lp); -STATIC MYBOOL verify_basis(lprec *lp); -STATIC int unload_basis(lprec *lp, MYBOOL restorelast); - -STATIC int perturb_bounds(lprec *lp, BBrec *perturbed, MYBOOL doRows, MYBOOL doCols, MYBOOL includeFIXED); -STATIC MYBOOL validate_bounds(lprec *lp, REAL *upbo, REAL *lowbo); -STATIC MYBOOL impose_bounds(lprec *lp, REAL * upbo, REAL *lowbo); -STATIC int unload_BB(lprec *lp); - -STATIC REAL feasibilityOffset(lprec *lp, MYBOOL isdual); -STATIC MYBOOL isP1extra(lprec *lp); -STATIC REAL get_refactfrequency(lprec *lp, MYBOOL final); -STATIC int findBasicFixedvar(lprec *lp, int afternr, MYBOOL slacksonly); -STATIC MYBOOL isBasisVarFeasible(lprec *lp, REAL tol, int basis_row); -STATIC MYBOOL isPrimalFeasible(lprec *lp, REAL tol, int infeasibles[], REAL *feasibilitygap); -STATIC MYBOOL isDualFeasible(lprec *lp, REAL tol, int *boundflips, int infeasibles[], REAL *feasibilitygap); - -/* Main simplex driver routines */ -STATIC int preprocess(lprec *lp); -STATIC void postprocess(lprec *lp); -STATIC MYBOOL performiteration(lprec *lp, int rownr, int varin, LREAL theta, MYBOOL primal, MYBOOL allowminit, REAL *prow, int *nzprow, REAL *pcol, int *nzpcol, int *boundswaps); -STATIC void transfer_solution_var(lprec *lp, int uservar); -STATIC void transfer_solution(lprec *lp, MYBOOL dofinal); - -/* Scaling utilities */ -STATIC REAL scaled_floor(lprec *lp, int colnr, REAL value, REAL epsscale); -STATIC REAL scaled_ceil(lprec *lp, int colnr, REAL value, REAL epsscale); - -/* Variable mapping utility routines */ -STATIC void varmap_lock(lprec *lp); -STATIC void varmap_clear(lprec *lp); -STATIC MYBOOL varmap_canunlock(lprec *lp); -STATIC void varmap_addconstraint(lprec *lp); -STATIC void varmap_addcolumn(lprec *lp); -STATIC void varmap_delete(lprec *lp, int base, int delta, LLrec *varmap); -STATIC void varmap_compact(lprec *lp, int prev_rows, int prev_cols); -STATIC MYBOOL varmap_validate(lprec *lp, int varno); -/* STATIC MYBOOL del_varnameex(lprec *lp, hashelem **namelist, hashtable *ht, int varnr, LLrec *varmap); */ - STATIC MYBOOL del_varnameex(lprec *lp, hashelem **namelist, int items, hashtable *ht, int varnr, LLrec *varmap); - -/* Pseudo-cost routines (internal) */ -STATIC BBPSrec *init_pseudocost(lprec *lp, int pseudotype); -STATIC void free_pseudocost(lprec *lp); -STATIC REAL get_pseudorange(BBPSrec *pc, int mipvar, int varcode); -STATIC void update_pseudocost(BBPSrec *pc, int mipvar, int varcode, MYBOOL capupper, REAL varsol); -STATIC REAL get_pseudobranchcost(BBPSrec *pc, int mipvar, MYBOOL dofloor); -STATIC REAL get_pseudonodecost(BBPSrec *pc, int mipvar, int vartype, REAL varsol); - -/* Matrix access and equation solving routines */ -STATIC void set_OF_override(lprec *lp, REAL *ofVector); -STATIC void set_OF_p1extra(lprec *lp, REAL p1extra); -STATIC void unset_OF_p1extra(lprec *lp); -MYBOOL modifyOF1(lprec *lp, int index, REAL *ofValue, REAL mult); -REAL __WINAPI get_OF_active(lprec *lp, int varnr, REAL mult); -STATIC MYBOOL is_OF_nz(lprec *lp, int colnr); - -STATIC int get_basisOF(lprec *lp, int coltarget[], REAL crow[], int colno[]); -int __WINAPI get_basiscolumn(lprec *lp, int j, int rn[], double bj[]); -int __WINAPI obtain_column(lprec *lp, int varin, REAL *pcol, int *nzlist, int *maxabs); -STATIC int compute_theta(lprec *lp, int rownr, LREAL *theta, int isupbound, REAL HarrisScalar, MYBOOL primal); - -/* Pivot utility routines */ -STATIC int findBasisPos(lprec *lp, int notint, int *var_basic); -STATIC MYBOOL check_degeneracy(lprec *lp, REAL *pcol, int *degencount); - -#endif /* HEADER_lp_lib */ diff --git a/code/3rd_lpsolve/lp_matrix.c b/code/3rd_lpsolve/lp_matrix.c deleted file mode 100644 index 9c8d7a47..00000000 --- a/code/3rd_lpsolve/lp_matrix.c +++ /dev/null @@ -1,3820 +0,0 @@ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_scale.h" -#include "lp_report.h" -#include "lp_price.h" -#include "lp_pricePSE.h" -#include "lp_matrix.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* ------------------------------------------------------------------------- - Basic matrix routines in lp_solve v5.0+ - ------------------------------------------------------------------------- - Author: Michel Berkelaar (to lp_solve v3.2), - Kjell Eikland (v4.0 and forward) - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h, lp_pricerPSE.h, lp_matrix.h - - Release notes: - v5.0.0 1 January 2004 First integrated and repackaged version. - v5.0.1 7 May 2004 Added matrix transpose function. - v5.1.0 20 July 2004 Reworked with flexible matrix storage model. - v5.2.0 10 January 2005 Added fast deletion methods. - Added data extraction to matrix method. - Changed to explicit OF storage mode. - - ------------------------------------------------------------------------- */ - -STATIC MATrec *mat_create(lprec *lp, int rows, int columns, REAL epsvalue) -{ - MATrec *newmat; - - newmat = (MATrec *) calloc(1, sizeof(*newmat)); - newmat->lp = lp; - - newmat->rows_alloc = 0; - newmat->columns_alloc = 0; - newmat->mat_alloc = 0; - - inc_matrow_space(newmat, rows); - newmat->rows = rows; - inc_matcol_space(newmat, columns); - newmat->columns = columns; - inc_mat_space(newmat, 0); - - newmat->epsvalue = epsvalue; - - return( newmat ); -} - -STATIC void mat_free(MATrec **matrix) -{ - if((matrix == NULL) || (*matrix == NULL)) - return; - -#if MatrixColAccess==CAM_Record - FREE((*matrix)->col_mat); -#else /*if MatrixColAccess==CAM_Vector*/ - FREE((*matrix)->col_mat_colnr); - FREE((*matrix)->col_mat_rownr); - FREE((*matrix)->col_mat_value); -#endif - FREE((*matrix)->col_end); - FREE((*matrix)->col_tag); - -#if MatrixRowAccess==RAM_Index - FREE((*matrix)->row_mat); -#elif MatrixColAccess==CAM_Record - FREE((*matrix)->row_mat); -#else /*if MatrixRowAccess==COL_Vector*/ - FREE((*matrix)->row_mat_colnr); - FREE((*matrix)->row_mat_rownr); - FREE((*matrix)->row_mat_value); -#endif - FREE((*matrix)->row_end); - FREE((*matrix)->row_tag); - - FREE((*matrix)->colmax); - FREE((*matrix)->rowmax); - - FREE(*matrix); -} - -STATIC MYBOOL mat_memopt(MATrec *mat, int rowextra, int colextra, int nzextra) -{ - MYBOOL status = TRUE; - int matalloc, colalloc, rowalloc; - - if((mat == NULL) || -#if 0 - (++rowextra < 1) || (++colextra < 1) || (++nzextra < 1)) -#else - (rowextra < 0) || (colextra < 0) || (nzextra < 0)) -#endif - return( FALSE ); - - mat->rows_alloc = MIN(mat->rows_alloc, mat->rows + rowextra); - mat->columns_alloc = MIN(mat->columns_alloc, mat->columns + colextra); - mat->mat_alloc = MIN(mat->mat_alloc, mat->col_end[mat->columns] + nzextra); -#if 0 - rowalloc = mat->rows_alloc; - colalloc = mat->columns_alloc; - matalloc = mat->mat_alloc; -#else - rowalloc = mat->rows_alloc + 1; - colalloc = mat->columns_alloc + 1; - matalloc = mat->mat_alloc + 1; -#endif - -#if MatrixColAccess==CAM_Record - mat->col_mat = (MATitem *) realloc(mat->col_mat, matalloc * sizeof(*(mat->col_mat))); - status &= (mat->col_mat != NULL); -#else /*if MatrixColAccess==CAM_Vector*/ - status &= allocINT(mat->lp, &(mat->col_mat_colnr), matalloc, AUTOMATIC) && - allocINT(mat->lp, &(mat->col_mat_rownr), matalloc, AUTOMATIC) && - allocREAL(mat->lp, &(mat->col_mat_value), matalloc, AUTOMATIC); -#endif - status &= allocINT(mat->lp, &mat->col_end, colalloc, AUTOMATIC); - if(mat->col_tag != NULL) - status &= allocINT(mat->lp, &mat->col_tag, colalloc, AUTOMATIC); - -#if MatrixRowAccess==RAM_Index - status &= allocINT(mat->lp, &(mat->row_mat), matalloc, AUTOMATIC); -#elif MatrixColAccess==CAM_Record - mat->row_mat = (MATitem *) realloc(mat->row_mat, matalloc * sizeof(*(mat->row_mat))); - status &= (mat->row_mat != NULL); -#else /*if MatrixRowAccess==COL_Vector*/ - status &= allocINT(mat->lp, &(mat->row_mat_colnr), matalloc, AUTOMATIC) && - allocINT(mat->lp, &(mat->row_mat_rownr), matalloc, AUTOMATIC) && - allocREAL(mat->lp, &(mat->row_mat_value), matalloc, AUTOMATIC); -#endif - status &= allocINT(mat->lp, &mat->row_end, rowalloc, AUTOMATIC); - if(mat->row_tag != NULL) - status &= allocINT(mat->lp, &mat->row_tag, rowalloc, AUTOMATIC); - - if(mat->colmax != NULL) - status &= allocREAL(mat->lp, &(mat->colmax), colalloc, AUTOMATIC); - if(mat->rowmax != NULL) - status &= allocREAL(mat->lp, &(mat->rowmax), rowalloc, AUTOMATIC); - - return( status ); -} - -STATIC MYBOOL inc_mat_space(MATrec *mat, int mindelta) -{ - int spaceneeded, nz = mat_nonzeros(mat); - - if(mindelta <= 0) - mindelta = MAX(mat->rows, mat->columns) + 1; - spaceneeded = DELTA_SIZE(mindelta, nz); - SETMAX(mindelta, spaceneeded); - - if(mat->mat_alloc == 0) - spaceneeded = mindelta; - else - spaceneeded = nz + mindelta; - - if(spaceneeded >= mat->mat_alloc) { - /* Let's allocate at least MAT_START_SIZE entries */ - if(mat->mat_alloc < MAT_START_SIZE) - mat->mat_alloc = MAT_START_SIZE; - - /* Increase the size by RESIZEFACTOR each time it becomes too small */ - while(spaceneeded >= mat->mat_alloc) - mat->mat_alloc += mat->mat_alloc / RESIZEFACTOR; - -#if MatrixColAccess==CAM_Record - mat->col_mat = (MATitem *) realloc(mat->col_mat, (mat->mat_alloc) * sizeof(*(mat->col_mat))); -#else /*if MatrixColAccess==CAM_Vector*/ - allocINT(mat->lp, &(mat->col_mat_colnr), mat->mat_alloc, AUTOMATIC); - allocINT(mat->lp, &(mat->col_mat_rownr), mat->mat_alloc, AUTOMATIC); - allocREAL(mat->lp, &(mat->col_mat_value), mat->mat_alloc, AUTOMATIC); -#endif - -#if MatrixRowAccess==RAM_Index - allocINT(mat->lp, &(mat->row_mat), mat->mat_alloc, AUTOMATIC); -#elif MatrixColAccess==CAM_Record - mat->row_mat = (MATitem *) realloc(mat->row_mat, (mat->mat_alloc) * sizeof(*(mat->row_mat))); -#else /*if MatrixColAccess==CAM_Vector*/ - allocINT(mat->lp, &(mat->row_mat_colnr), mat->mat_alloc, AUTOMATIC); - allocINT(mat->lp, &(mat->row_mat_rownr), mat->mat_alloc, AUTOMATIC); - allocREAL(mat->lp, &(mat->row_mat_value), mat->mat_alloc, AUTOMATIC); -#endif - } - return(TRUE); -} - -STATIC MYBOOL inc_matrow_space(MATrec *mat, int deltarows) -{ - int rowsum, oldrowsalloc; - MYBOOL status = TRUE; - - /* Adjust lp row structures */ - if(mat->rows+deltarows >= mat->rows_alloc) { - - /* Update memory allocation and sizes */ - oldrowsalloc = mat->rows_alloc; - deltarows = DELTA_SIZE(deltarows, mat->rows); - SETMAX(deltarows, DELTAROWALLOC); - mat->rows_alloc += deltarows; - rowsum = mat->rows_alloc + 1; - - /* Update row pointers */ - status = allocINT(mat->lp, &mat->row_end, rowsum, AUTOMATIC); - mat->row_end_valid = FALSE; - } - return( status ); -} - -STATIC MYBOOL inc_matcol_space(MATrec *mat, int deltacols) -{ - int i, colsum, oldcolsalloc; - MYBOOL status = TRUE; - - /* Adjust lp column structures */ - if(mat->columns+deltacols >= mat->columns_alloc) { - - /* Update memory allocation and sizes */ - oldcolsalloc = mat->columns_alloc; - deltacols = DELTA_SIZE(deltacols, mat->columns); - SETMAX(deltacols, DELTACOLALLOC); - mat->columns_alloc += deltacols; - colsum = mat->columns_alloc + 1; - status = allocINT(mat->lp, &mat->col_end, colsum, AUTOMATIC); - - /* Update column pointers */ - if(oldcolsalloc == 0) - mat->col_end[0] = 0; - for(i = MIN(oldcolsalloc, mat->columns) + 1; i < colsum; i++) - mat->col_end[i] = mat->col_end[i-1]; - mat->row_end_valid = FALSE; - } - return( status ); -} - -STATIC int mat_collength(MATrec *mat, int colnr) -{ - return( mat->col_end[colnr] - mat->col_end[colnr-1] ); -} - -STATIC int mat_rowlength(MATrec *mat, int rownr) -{ - if(mat_validate(mat)) { - if(rownr <= 0) - return( mat->row_end[0] ); - else - return( mat->row_end[rownr] - mat->row_end[rownr-1] ); - } - else - return( 0 ); -} - -STATIC int mat_nonzeros(MATrec *mat) -{ - return( mat->col_end[mat->columns] ); -} - -STATIC MYBOOL mat_indexrange(MATrec *mat, int index, MYBOOL isrow, int *startpos, int *endpos) -{ -#ifdef Paranoia - if(isrow && ((index < 0) || (index > mat->rows))) - return( FALSE ); - else if(!isrow && ((index < 1) || (index > mat->columns))) - return( FALSE ); -#endif - - if(isrow && mat_validate(mat)) { - if(index == 0) - *startpos = 0; - else - *startpos = mat->row_end[index-1]; - *endpos = mat->row_end[index]; - } - else { - *startpos = mat->col_end[index-1]; - *endpos = mat->col_end[index]; - } - return( TRUE ); -} - -STATIC int mat_shiftrows(MATrec *mat, int *bbase, int delta, LLrec *varmap) -{ - int j, k, i, ii, thisrow, *colend, base; - MYBOOL preparecompact = FALSE; - int *rownr; - - if(delta == 0) - return( 0 ); - base = abs(*bbase); - - if(delta > 0) { - - /* Insert row by simply incrementing existing row indeces */ - if(base <= mat->rows) { - k = mat_nonzeros(mat); - rownr = &COL_MAT_ROWNR(0); - for(ii = 0; ii < k; ii++, rownr += matRowColStep) { - if(*rownr >= base) - *rownr += delta; - } - } - - /* Set defaults (actual basis set in separate procedure) */ - for(i = 0; i < delta; i++) { - ii = base + i; - mat->row_end[ii] = 0; - } - } - else if(base <= mat->rows) { - - /* Check for preparation of mass-deletion of rows */ - preparecompact = (MYBOOL) (varmap != NULL); - if(preparecompact) { - /* Create the offset array */ - int *newrowidx = NULL; - allocINT(mat->lp, &newrowidx, mat->rows+1, FALSE); - newrowidx[0] = 0; - delta = 0; - for(j = 1; j <= mat->rows; j++) { - if(isActiveLink(varmap, j)) { - delta++; - newrowidx[j] = delta; - } - else - newrowidx[j] = -1; - } - k = 0; - delta = 0; - base = mat_nonzeros(mat); - rownr = &COL_MAT_ROWNR(0); - for(i = 0; i < base; i++, rownr += matRowColStep) { - thisrow = newrowidx[*rownr]; - if(thisrow < 0) { - *rownr = -1; - delta++; - } - else - *rownr = thisrow; - } - FREE(newrowidx); - return(delta); - } - - /* Check if we should prepare for compacting later - (this is in order to speed up multiple row deletions) */ - preparecompact = (MYBOOL) (*bbase < 0); - if(preparecompact) - *bbase = my_flipsign((*bbase)); - - /* First make sure we don't cross the row count border */ - if(base-delta-1 > mat->rows) - delta = base - mat->rows - 1; - - /* Then scan over all entries shifting and updating rows indeces */ - if(preparecompact) { - k = 0; - for(j = 1, colend = mat->col_end + 1; - j <= mat->columns; j++, colend++) { - i = k; - k = *colend; - rownr = &COL_MAT_ROWNR(i); - for(; i < k; i++, rownr += matRowColStep) { - thisrow = *rownr; - if(thisrow < base) - continue; - else if(thisrow >= base-delta) - *rownr += delta; - else - *rownr = -1; - } - } - } - else { - k = 0; - ii = 0; - for(j = 1, colend = mat->col_end + 1; - j <= mat->columns; j++, colend++) { - i = k; - k = *colend; - rownr = &COL_MAT_ROWNR(i); - for(; i < k; i++, rownr += matRowColStep) { - thisrow = *rownr; - if(thisrow >= base) { - if(thisrow >= base-delta) - *rownr += delta; - else - continue; - } - if(ii != i) { - COL_MAT_COPY(ii, i); - } - ii++; - } - *colend = ii; - } - } - } - return( 0 ); -} - -/* Map-based compacting+insertion of matrix elements without changing row and column indeces. - When mat2 is NULL, a simple compacting of non-deleted rows and columns is done. */ -STATIC int mat_mapreplace(MATrec *mat, LLrec *rowmap, LLrec *colmap, MATrec *mat2) -{ - lprec *lp = mat->lp; - int i, ib, ie, ii, j, jj, jb, je, nz, *colend, *rownr, *rownr2, *indirect = NULL; - REAL *value, *value2; - - /* Check if there is something to insert */ - if((mat2 != NULL) && ((mat2->col_tag == NULL) || (mat2->col_tag[0] <= 0) || (mat_nonzeros(mat2) == 0))) - return( 0 ); - - /* Create map and sort by increasing index in "mat" */ - if(mat2 != NULL) { - jj = mat2->col_tag[0]; - allocINT(lp, &indirect, jj+1, FALSE); - indirect[0] = jj; - for(i = 1; i <= jj; i++) - indirect[i] = i; - hpsortex(mat2->col_tag, jj, 1, sizeof(*indirect), FALSE, compareINT, indirect); - } - - /* Do the compacting */ - mat->row_end_valid = FALSE; - nz = mat->col_end[mat->columns]; - ie = 0; - ii = 0; - if((mat2 == NULL) || (indirect[0] == 0)) { - je = mat->columns + 1; - jj = 1; - jb = 0; - } - else { - je = indirect[0]; - jj = 0; - do { - jj++; - jb = mat2->col_tag[jj]; - } while(jb <= 0); - - } - for(j = 1, colend = mat->col_end + 1; - j <= mat->columns; j++, colend++) { - ib = ie; - ie = *colend; - - /* Always skip (condense) replacement columns */ - if(j == jb) { - jj++; - if(jj <= je) - jb = mat2->col_tag[jj]; - else - jb = mat->columns + 1; - } - - /* Only include active columns */ - else if(isActiveLink(colmap, j)) { - rownr = &COL_MAT_ROWNR(ib); - for(; ib < ie; ib++, rownr += matRowColStep) { - - /* Also make sure the row is active */ - if(isActiveLink(rowmap, *rownr)) { - if(ii != ib) { - COL_MAT_COPY(ii, ib); - } - ii++; - } - } - } - *colend = ii; - } - if(mat2 == NULL) - goto Finish; - - /* Tally non-zero insertions */ - i = 0; - for(j = 1; j <= mat2->col_tag[0]; j++) { - jj = mat2->col_tag[j]; - if((jj > 0) && isActiveLink(colmap, jj)) { - jj = indirect[j]; - je = mat2->col_end[jj]; - jb = mat2->col_end[jj-1]; - rownr2 = &COL_MAT2_ROWNR(jb); - for(; jb < je; jb++, rownr2 += matRowColStep) { - if((*rownr2 > 0) && isActiveLink(rowmap, *rownr2)) - i++; - } - } - } - - /* Make sure we have enough matrix space */ - ii = mat->col_end[mat->columns] + i; - if(mat->mat_alloc <= ii) - inc_mat_space(mat, i); - - /* Do shifting and insertion - loop from the end going forward */ - jj = indirect[0]; - jj = mat2->col_tag[jj]; - for(j = mat->columns, colend = mat->col_end + mat->columns, ib = *colend; - j > 0; j--) { - - /* Update indeces for this loop */ - ie = ib; - *colend = ii; - colend--; - ib = *colend; - - /* Insert new values */ - if(j == jj) { - /* Only include an active column */ - if(isActiveLink(colmap, j)) { - jj = indirect[0]; - jj = indirect[jj]; - rownr = &COL_MAT_ROWNR(ii-1); - value = &COL_MAT_VALUE(ii-1); - jb = mat2->col_end[jj-1]; - je = mat2->col_end[jj] - 1; - rownr2 = &COL_MAT2_ROWNR(je); - value2 = &COL_MAT2_VALUE(je); - - /* Process constraint coefficients */ - for(; je >= jb; je--, rownr2 -= matRowColStep, value2 -= matValueStep) { - i = *rownr2; - if(i == 0) { - i = -1; - break; - } - else if(isActiveLink(rowmap, i)) { - ii--; - *rownr = i; - rownr -= matRowColStep; - *value = my_chsign(is_chsign(lp, i), *value2); - value -= matValueStep; - } - } - - /* Then handle the objective */ - if(i == -1) { - lp->orig_obj[j] = my_chsign(is_maxim(lp), *value2); - rownr2 -= matRowColStep; - value2 -= matValueStep; - } - else - lp->orig_obj[j] = 0; - - } - /* Update replacement column index or break if no more candidates */ - jj = --indirect[0]; - if(jj == 0) - break; - jj = mat2->col_tag[jj]; - if(jj <= 0) - break; - } - /* Shift existing values down */ - else { - if(isActiveLink(colmap, j)) - while(ie > ib) { - ii--; - ie--; - if(ie != ii) { - COL_MAT_COPY(ii, ie); - } - } - } - } - - /* Return the delta number of non-zero elements */ -Finish: - nz -= mat->col_end[mat->columns]; - FREE(indirect); - - return( nz ); -} - -/* Routines to compact rows in matrix based on precoded entries */ -STATIC int mat_zerocompact(MATrec *mat) -{ - return( mat_rowcompact(mat, TRUE) ); -} -STATIC int mat_rowcompact(MATrec *mat, MYBOOL dozeros) -{ - int i, ie, ii, j, nn, *colend, *rownr; - REAL *value; - - nn = 0; - ie = 0; - ii = 0; - for(j = 1, colend = mat->col_end + 1; - j <= mat->columns; j++, colend++) { - i = ie; - ie = *colend; - rownr = &COL_MAT_ROWNR(i); - value = &COL_MAT_VALUE(i); - for(; i < ie; - i++, rownr += matRowColStep, value += matValueStep) { - if((*rownr < 0) || (dozeros && (fabs(*value) < mat->epsvalue))) { - nn++; - continue; - } - if(ii != i) { - COL_MAT_COPY(ii, i); - } - ii++; - } - *colend = ii; - } - return( nn ); -} - -/* Routines to compact columns and their indeces based on precoded entries */ -STATIC int mat_colcompact(MATrec *mat, int prev_rows, int prev_cols) -{ - int i, ii, j, k, n_del, n_sum, *colend, *newcolend, *colnr, newcolnr; - MYBOOL deleted; - lprec *lp = mat->lp; - presolveundorec *lpundo = lp->presolve_undo; - - - n_sum = 0; - k = 0; - ii = 0; - newcolnr = 1; - for(j = 1, colend = newcolend = mat->col_end + 1; - j <= prev_cols; j++, colend++) { - n_del = 0; - i = k; - k = *colend; - for(colnr = &COL_MAT_COLNR(i); i < k; - i++, colnr += matRowColStep) { - if(*colnr < 0) { - n_del++; - n_sum++; - continue; - } - if(ii < i) { - COL_MAT_COPY(ii, i); - } - if(newcolnr < j) { - COL_MAT_COLNR(ii) = newcolnr; - } - ii++; - } - *newcolend = ii; - - deleted = (MYBOOL) (n_del > 0); -#if 1 - /* Do hoops in case there was an empty column */ - deleted |= (MYBOOL) (!lp->wasPresolved && (lpundo->var_to_orig[prev_rows+j] < 0)); - -#endif - /* Increment column variables if current column was not deleted */ - if(!deleted) { - newcolend++; - newcolnr++; - } - } - return(n_sum); -} - -STATIC int mat_shiftcols(MATrec *mat, int *bbase, int delta, LLrec *varmap) -{ - int i, ii, k, n, base; - - - k = 0; - if(delta == 0) - return( k ); - base = abs(*bbase); - - if(delta > 0) { - /* Shift pointers right */ - for(ii = mat->columns; ii > base; ii--) { - i = ii + delta; - mat->col_end[i] = mat->col_end[ii]; - } - /* Set defaults */ - for(i = 0; i < delta; i++) { - ii = base + i; - mat->col_end[ii] = mat->col_end[ii-1]; - } - } - else { - - /* Check for preparation of mass-deletion of columns */ - MYBOOL preparecompact = (MYBOOL) (varmap != NULL); - if(preparecompact) { - /* Create the offset array */ - int j, *colnr, *colend; - n = 0; - k = 0; - base = 0; - for(j = 1, colend = mat->col_end + 1; - j <= mat->columns; j++, colend++) { - i = k; - k = *colend; - if(isActiveLink(varmap, j)) { - base++; - ii = base; - } - else - ii = -1; - if(ii < 0) - n += k - i; - colnr = &COL_MAT_COLNR(i); - for(; i < k; i++, colnr += matRowColStep) - *colnr = ii; - } - return(n); - } - - /* Check if we should prepare for compacting later - (this is in order to speed up multiple column deletions) */ - preparecompact = (MYBOOL) (*bbase < 0); - if(preparecompact) - *bbase = my_flipsign((*bbase)); - - /* First make sure we don't cross the column count border */ - if(base-delta-1 > mat->columns) - delta = base - mat->columns - 1; - - /* Then scan over all entries shifting and updating column indeces */ - if(preparecompact) { - int *colnr; - n = 0; - i = mat->col_end[base-1]; - k = mat->col_end[base-delta-1]; - for(colnr = &COL_MAT_COLNR(i); i < k; - i++, colnr += matRowColStep) { - n++; - *colnr = -1; - } - k = n; - } - else { - /* Delete sparse matrix data, if required */ - if(base <= mat->columns) { - - i = mat->col_end[base-1]; /* Beginning of data to be deleted */ - ii = mat->col_end[base-delta-1]; /* Beginning of data to be shifted left */ - n = mat_nonzeros(mat); /* Total number of non-zeros */ - k = ii-i; /* Number of entries to be deleted */ - if((k > 0) && (n > i)) { - n -= ii; - COL_MAT_MOVE(i, ii, n); - } - - /* Update indexes */ - for(i = base; i <= mat->columns + delta; i++) { - ii = i - delta; - mat->col_end[i] = mat->col_end[ii] - k; - } - } - } - } - return( k ); -} - -STATIC MATrec *mat_extractmat(MATrec *mat, LLrec *rowmap, LLrec *colmap, MYBOOL negated) -{ - int *rownr, *colnr, xa, na; - REAL *value; - MATrec *newmat = mat_create(mat->lp, mat->rows, mat->columns, mat->epsvalue); - - /* Initialize */ - na = mat_nonzeros(mat); - rownr = &COL_MAT_ROWNR(0); - colnr = &COL_MAT_COLNR(0); - value = &COL_MAT_VALUE(0); - - /* Loop over the indeces, picking out values in qualifying rows and colums - (note that the loop could be speeded up for dense matrices by making an - outer loop for columns and inner loop for rows) */ - for(xa = 0; xa < na; xa++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) { - if((isActiveLink(colmap, *colnr) ^ negated) && - (isActiveLink(rowmap, *rownr) ^ negated)) - mat_setvalue(newmat, *rownr, *colnr, *value, FALSE); - } - - /* Return the populated new matrix */ - return( newmat ); -} - -STATIC MYBOOL mat_setcol(MATrec *mat, int colno, int count, REAL *column, int *rowno, MYBOOL doscale, MYBOOL checkrowmode) -{ - int i, jj = 0, elmnr, orignr, newnr, firstrow; - MYBOOL *addto = NULL, isA, isNZ; - REAL value, saved = 0; - lprec *lp = mat->lp; - - /* Check if we are in row order mode and should add as row instead; - the matrix will be transposed at a later stage */ - if(checkrowmode && mat->is_roworder) - return( mat_setrow(mat, colno, count, column, rowno, doscale, FALSE) ); - - /* Initialize and validate */ - isA = (MYBOOL) (mat == mat->lp->matA); - isNZ = (MYBOOL) (rowno != NULL); - if(!isNZ) - count = mat->lp->rows; - else if((count < 0) || (count > mat->rows+((mat->is_roworder) ? 0 : 1))) - return( FALSE ); - if(isNZ && (count > 0)) { - if(count > 1) - sortREALByINT(column, rowno, count, 0, TRUE); - if((rowno[0] < 0) || (rowno[count-1] > mat->rows)) - return( FALSE ); - } - - /* Capture OF definition in column mode */ - if(isA && !mat->is_roworder) { - if(isNZ && (count > 0) && (rowno[0] == 0)) { - value = column[0]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - if(doscale) - value = scaled_mat(lp, value, 0, colno); - value = my_chsign(is_maxim(lp), value); - lp->orig_obj[colno] = value; - count--; - column++; - rowno++; - } - else if(!isNZ && (column[0] != 0)) { - value = saved = column[0]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - if(doscale) - value = scaled_mat(lp, value, 0, colno); - value = my_chsign(is_maxim(lp), value); - lp->orig_obj[colno] = value; - column[0] = 0; - } - else - lp->orig_obj[colno] = 0; - } - - /* Optionally tally and map the new non-zero values */ - firstrow = mat->rows + 1; - if(isNZ) { - newnr = count; - if(newnr) { - firstrow = rowno[0]; - jj = rowno[newnr - 1]; - } - } - else { - newnr = 0; - if(!allocMYBOOL(lp, &addto, mat->rows + 1, TRUE)) { - return( FALSE ); - } - for(i = mat->rows; i >= 0; i--) { - if(fabs(column[i]) > mat->epsvalue) { - addto[i] = TRUE; - firstrow = i; - newnr++; - } - } - } - - /* Make sure we have enough matrix space */ - if(!inc_mat_space(mat, newnr)) { - newnr = 0; - goto Done; - } - - /* Shift existing column data and adjust position indeces */ - orignr = mat_collength(mat, colno); - elmnr = newnr - orignr; - i = mat_nonzeros(mat) - mat->col_end[colno]; - if((elmnr != 0) && (i > 0)) { - COL_MAT_MOVE(mat->col_end[colno] + elmnr, mat->col_end[colno], i); - } - if(elmnr != 0) - for(i = colno; i <= mat->columns; i++) - mat->col_end[i] += elmnr; - - /* We are now ready to copy the new data */ - jj = mat->col_end[colno-1]; - if(isNZ) { - for(i = 0; i < count; jj++, i++) { - value = column[i]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - if(mat->is_roworder) { /* Fix following Ingmar Stein bug report 12.10.2006 */ - if(isA && doscale) - value = scaled_mat(lp, value, colno, rowno[i]); - if(isA) - value = my_chsign(is_chsign(lp, colno), value); - } - else { - if(isA && doscale) - value = scaled_mat(lp, value, rowno[i], colno); - if(isA) - value = my_chsign(is_chsign(lp, rowno[i]), value); - } - SET_MAT_ijA(jj, rowno[i], colno, value); - } - } - else { - for(i = firstrow; i <= mat->rows; i++) { - if(!addto[i]) - continue; - value = column[i]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - if(mat->is_roworder) { /* Fix following Ingmar Stein bug report 12.10.2006 */ - if(isA && doscale) - value = scaled_mat(lp, value, colno, i); - if(isA) - value = my_chsign(is_chsign(lp, colno), value); - } - else { - if(isA && doscale) - value = scaled_mat(lp, value, i, colno); - if(isA) - value = my_chsign(is_chsign(lp, i), value); - } - SET_MAT_ijA(jj, i, colno, value); - jj++; - } - } - mat->row_end_valid = FALSE; - - /* Finish and return */ -Done: - if(saved != 0) - column[0] = saved; - FREE(addto); - return( TRUE ); - -} - -STATIC MYBOOL mat_mergemat(MATrec *target, MATrec *source, MYBOOL usecolmap) -{ - lprec *lp = target->lp; - int i, ix, iy, n, *colmap = NULL; - REAL *colvalue = NULL; - - if((target->rows < source->rows) || !allocREAL(lp, &colvalue, target->rows+1, FALSE)) - return( FALSE ); - - if(usecolmap) { - n = source->col_tag[0]; - allocINT(lp, &colmap, n+1, FALSE); - for(i = 1; i <= n; i++) - colmap[i] = i; - hpsortex(source->col_tag, n, 1, sizeof(*colmap), FALSE, compareINT, colmap); - } - else - n = source->columns; - for(i = 1; i <= n; i++) { - if(!usecolmap && (mat_collength(source, i) == 0)) - continue; - if(usecolmap) { - ix = colmap[i]; - if(ix <= 0) - continue; - iy = source->col_tag[i]; - if(iy <= 0) - continue; - } - else - ix = iy = i; - mat_expandcolumn(source, ix, colvalue, NULL, FALSE); - mat_setcol(target, iy, 0, colvalue, NULL, FALSE, FALSE); - } - - FREE( colvalue ); - FREE( colmap ); - - return( TRUE ); -} - -STATIC int mat_nz_unused(MATrec *mat) -{ - return( mat->mat_alloc - mat->col_end[mat->columns] ); -} - -#if 0 -STATIC MYBOOL mat_setrow(MATrec *mat, int rowno, int count, REAL *row, int *colno, MYBOOL doscale, MYBOOL checkrowmode) -{ - lprec *lp = mat->lp; - int delta; - int k, kk, i, ii, j, jj = 0, jj_j, elmnr, orignr, newnr, firstcol, rownr, colnr, matz = 0; - MYBOOL *addto = NULL, isA, isNZ; - REAL value = 0.0, saved = 0; - - /* Check if we are in row order mode and should add as column instead; - the matrix will be transposed at a later stage */ - if(checkrowmode && mat->is_roworder) - return( mat_setcol(mat, rowno, count, row, colno, doscale, FALSE) ); - - /* Do initialization and validation */ - if(!mat_validate(mat)) - return( FALSE ); - isA = (MYBOOL) (mat == lp->matA); - isNZ = (MYBOOL) (colno != NULL); - if(!isNZ) - count = mat->columns; - else if((count < 0) || (count > mat->columns)) - return( FALSE ); - if(isNZ && (count > 0)) { - if(count > 1) - sortREALByINT(row, (int *) colno, count, 0, TRUE); - if((colno[0] < 1) || (colno[count-1] > mat->columns)) - return( FALSE ); - } - - /* Capture OF definition in row mode */ - if(isA && mat->is_roworder) { - lp->orig_obj[rowno] = 0; - if(isNZ && (count > 0) && (colno[0] == 0)) { - value = row[0]; - if(doscale) - value = scaled_mat(lp, value, 0, rowno); - value = my_chsign(is_maxim(lp), value); -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - lp->orig_obj[rowno] = value; - count--; - row++; - colno++; - } - else if(!isNZ && (row[0] != 0)) { - value = saved = row[0]; - if(doscale) - value = scaled_mat(lp, value, 0, rowno); - value = my_chsign(is_maxim(lp), value); -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - lp->orig_obj[rowno] = value; - row[0] = 0; - } - else { - lp->orig_obj[rowno] = 0; - value = 0; - } - } - - /* Optionally tally and map the new non-zero values */ - i = mat->row_end[rowno-1]; - ii = mat->row_end[rowno]; /* ****** KE 20070106 - was "-1" */ - firstcol = mat->columns + 1; - if(isNZ) { - /* See if we can do fast in-place replacements of leading items */ - colnr = 1; /* initialise in case of an empty row */ - while((i < ii) /* && (count > 0) */ && ((colnr = ROW_MAT_COLNR(i)) == *colno) && (count > 0)) { - value = *row; /* ****** KE 20080111 - Added line */ - if(mat->is_roworder) { - if(isA && doscale) - value = scaled_mat(lp, value, colnr, rowno); - if(isA) - value = my_chsign(is_chsign(lp, colnr), value); - } - else { - if(isA && doscale) - value = scaled_mat(lp, value, rowno, colnr); - if(isA) - value = my_chsign(is_chsign(lp, rowno), value); - } -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); - if(value == 0) - matz++; -#endif - ROW_MAT_VALUE(i) = value; - i++; - count--; - row++; - colno++; - } - if(i >= ii) - colnr = 0; - /* Proceed with remaining entries */ - newnr = count; - if(newnr > 0) - firstcol = colno[0]; - } - else { - newnr = 0; - kk = mat->columns; - if(i < ii) - colnr = ROW_MAT_COLNR(i); - else - colnr = 0; - for(k = 1; k <= kk; k++) { - value = row[k]; /* ****** KE 20080111 - Added line */ - if(fabs(value) > mat->epsvalue) { - /* See if we can do fast in-place replacements of leading items */ - if((addto == NULL) && (i < ii) && (colnr == k)) { - if(mat->is_roworder) { - if(isA && doscale) - value = scaled_mat(lp, value, colnr, rowno); - if(isA) - value = my_chsign(is_chsign(lp, colnr), value); - } - else { - if(isA && doscale) - value = scaled_mat(lp, value, rowno, colnr); - if(isA) - value = my_chsign(is_chsign(lp, rowno), value); - } -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); - if(value == 0) - matz++; -#endif - ROW_MAT_VALUE(i) = value; - i++; - if(i < ii) - colnr = ROW_MAT_COLNR(i); - else - colnr = 0; - } - /* Otherwise update addto-list */ - else { - if(addto == NULL) { - if(!allocMYBOOL(lp, &addto, mat->columns + 1, TRUE)) - return( FALSE ); - firstcol = k; - } - addto[k] = TRUE; - newnr++; - } - } - } - } - if(newnr == 0) - if (FALSE) - return( TRUE ); - - /* Make sure we have enough matrix space */ - if((newnr > 0) && (mat_nz_unused(mat) <= newnr) && !inc_mat_space(mat, newnr)) { - newnr = 0; - goto Done; - } - - /* Pack initial entries if existing row data has a lower column - start index than the first index of the new vector */ - orignr = mat_nonzeros(mat); - /* delta = newnr - mat_rowlength(mat, rowno);*/ - kk = 0; - if(rowno == 0) - ii = 0; - else - ii = mat->row_end[rowno-1]; - - if((orignr == 0) || (ii >= orignr)) - j = firstcol; - else if(isNZ||TRUE) - j = colnr; - else - j = ROW_MAT_COLNR(ii); /* first column with a value on that row */ - - jj = mat->col_end[firstcol-1]; /* Set the index of the insertion point for the first new value */ - if(jj >= orignr) - colnr = firstcol; - else - colnr = COL_MAT_COLNR(jj); /* first column with a value starting from firstcol */ - - if((j > 0) && (j < colnr)) { - jj = elmnr = mat->col_end[j-1]; - for( ; j < colnr; j++) { - /* Shift entries in current column */ - k = mat->col_end[j]; - for( ; jj < k; jj++) { - if(COL_MAT_ROWNR(jj) != rowno) { - COL_MAT_COPY(elmnr, jj); - elmnr++; - } - } - /* Update next column start index */ - mat->col_end[j] = elmnr; - } - delta = elmnr - jj; /* The shrinkage count */ - } - else { - delta = 0; - /* Adjust for case where we simply append values - jj is initially the first column item */ - if((mat->col_end[firstcol] == orignr) && 0) - jj = orignr; - } - - /* Make sure we have sufficient space for any additional entries and move existing data down; - this ensures that we only have to relocate matrix elements up in the next stage */ - jj_j = MAX(0, newnr + delta); - if(jj_j > 0) { - if(!inc_mat_space(mat, jj_j)) { - FREE(addto); - return( FALSE ); - } - if(orignr-jj > 0) { - COL_MAT_MOVE(jj+jj_j, jj, orignr-jj); - } - jj += jj_j; - } - - /* Handle case where the matrix was empty before (or we can simply append) */ - if((delta >= 0) && (mat->col_end[firstcol] == orignr) && 0) { - if(isNZ) - elmnr = count; - else - elmnr = mat->columns; - jj_j = mat->col_end[firstcol]; - for(newnr = 0; newnr < elmnr; newnr++) { - if(isNZ) - colnr = colno[newnr]; - else - colnr = newnr + 1; - /* Update column start position if we have crossed a column */ - while(colnr > firstcol) { - mat->col_end[firstcol] = jj_j; - firstcol++; - } - if(isNZ || ((addto != NULL) && addto[colnr])) { - if(isNZ) - value = row[newnr]; - else - value = row[colnr]; - if(isA && doscale) - value = scaled_mat(lp, value, rowno, colnr); - if(isA) - value = my_chsign(is_chsign(lp, rowno), value); -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); - if(value == 0) - matz++; -#endif - SET_MAT_ijA(jj_j, rowno, colnr, value); - jj_j++; - /* Update last column start position */ - mat->col_end[firstcol] = jj_j; - firstcol++; - } - } - - /* Make sure we update tail empty column offsets */ - while(firstcol <= mat->columns) { - mat->col_end[firstcol] = jj_j; - firstcol++; - } - jj_j = 0; - } - - /* Start from the top of the first non-zero column of the new row */ - elmnr = orignr + jj_j; - if(jj <= elmnr) { - if(isNZ) - newnr = 0; - else - newnr = firstcol - 1; - j = jj - mat->col_end[firstcol-1]; - colnr = firstcol; - while((jj < elmnr) || (newnr < count)) { - - /* Update column start position if we have crossed a column */ - while(colnr > firstcol) { - mat->col_end[firstcol] = kk; - firstcol++; - } - - /* See if we have a row equal to or greater than the target row */ - jj_j = jj - j; - if(jj < elmnr) { - rownr = COL_MAT_ROWNR(jj); - colnr = COL_MAT_COLNR(jj); - } - else { - rownr = rowno; - if(!isNZ) /* KE added this conditional on 13.9.2006 */ - colnr = firstcol + 1; - else - colnr = mat->columns + 1; - } - - if(isNZ) { - if(newnr < count) - kk = colno[newnr]; - else - kk = mat->columns + 1; - } - else - kk = newnr + 1; - - /* Test if there is an available new item ... */ - if((isNZ && (kk > colnr)) || /* If this is not the case */ - (!isNZ && ((kk > colnr) || (!addto[kk])))) { - /* DELETE if there is an existing value */ - if(!isNZ && (kk <= colnr)) - newnr++; - if(rownr == rowno) { - kk = jj_j; - j++; - jj++; - continue; - } - /* KEEP otherwise and move entry up */ - if(!isNZ && (colnr > kk)) { - colnr = kk; - kk = jj_j; - continue; - } - } - else if((colnr > kk) || /* Existing column index > new => INSERT */ - ((colnr == kk) && (rownr >= rowno)) ) { /* Same column index, existing row >= target row => INSERT/REPLACE */ - - if(isNZ) - value = row[newnr]; - else - value = row[newnr+1]; - newnr++; - if(isA && doscale) - value = scaled_mat(lp, value, rowno, kk); - if(isA) - value = my_chsign(is_chsign(lp, rowno), value); -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); - if(value == 0) - matz++; -#endif - SET_MAT_ijA(jj_j, rowno, kk, value); - - /* Adjust if we have inserted an element */ - if((colnr > kk) || (rownr > rowno)) { - j--; - jj--; - } - colnr = kk; - kk = jj_j; - jj++; - continue; - } - - /* Shift the matrix element up by the active difference */ - if(jj_j != jj) { - COL_MAT_COPY(jj_j, jj); - } - kk = jj_j; - jj++; - - } - - /* Update pending / incomplete column start position */ - while(colnr > firstcol) { - mat->col_end[firstcol] = kk; - firstcol++; - } - - /* Make sure we update tail column offsets */ - jj_j = jj - j; - while(firstcol <= mat->columns) { - mat->col_end[firstcol] = jj_j; - firstcol++; - } - } - - /* Compact in the case that we added zeros and set flag for row index update */ - if(matz > 0) - mat_zerocompact(mat); - mat->row_end_valid = FALSE; - -Done: - if(saved != 0) - row[0] = saved; - FREE(addto); - return( (MYBOOL) (newnr > 0) ); - -} - -#else - -STATIC MYBOOL mat_setrow(MATrec *mat, int rowno, int count, REAL *row, int *colno, MYBOOL doscale, MYBOOL checkrowmode) -{ - lprec *lp = mat->lp; - int delta, delta1; - int k, i, ii, j, jj_j, lendense, - origidx = 0, newidx, orignz, newnz, - rownr, colnr, colnr1; - MYBOOL isA, isNZ; - REAL value = 0.0; - - /* Check if we are in row order mode and should add as column instead; - the matrix will be transposed at a later stage */ - if(checkrowmode && mat->is_roworder) - return( mat_setcol(mat, rowno, count, row, colno, doscale, FALSE) ); - - /* Do initialization and validation */ - if(!mat_validate(mat)) - return( FALSE ); - isA = (MYBOOL) (mat == lp->matA); - if(doscale && isA && !lp->scaling_used) - doscale = FALSE; - isNZ = (MYBOOL) (colno != NULL); - lendense = (mat->is_roworder ? lp->rows : lp->columns); - if((count < 0) || (count > lendense)) - return( FALSE ); - colnr1 = lendense + 1; - - /* Capture OF definition in row mode */ - if(isA && mat->is_roworder) { - lp->orig_obj[rowno] = 0; - if((count > 0) && (colno[0] == 0)) { - value = row[0]; - if(doscale) - value = scaled_mat(lp, value, 0, rowno); - value = my_chsign(is_maxim(lp), value); -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - lp->orig_obj[rowno] = value; - if(isNZ) { - colno++; - row++; - count--; - } - } - else { - lp->orig_obj[rowno] = 0; - value = 0; - } - } - - /* Make local working data copies */ - if(!isNZ) { - REAL *tmprow = NULL; - if(!allocINT(lp, &colno, lendense+1, FALSE)) - return( FALSE ); - newnz = 0; - for(i = 1; i <= lendense; i++) - if((value = row[i]) != 0) { - if((tmprow == NULL) && !allocREAL(lp, &tmprow, lendense-i+1, FALSE)) { - FREE(colno); - return( FALSE ); - } - tmprow[newnz] = value; - colno[newnz++] = i; - } - count = newnz; - row = tmprow; - } - else { - int *tmpcolno = NULL; - if(!allocINT(lp, &tmpcolno, lendense, FALSE)) - return( FALSE ); - newnz = count; - MEMCOPY(tmpcolno, colno, newnz); - colno = tmpcolno; - if(newnz > 1) - sortREALByINT(row, (int *) colno, newnz, 0, TRUE); - if((newnz > 0) && ((colno[0] < 0) || (colno[newnz-1] > lendense))) { - FREE(colno); - newnz = 0; - return( FALSE ); - } - } - - /* Make sure we have enough matrix space */ - i = mat->row_end[rowno-1]; - ii = mat->row_end[rowno]; - delta1 = delta = count - (ii-i); - colnr1 = (newnz > 0 ? colno[0] : lendense+1); - - /* Pack initial entries if existing row data has a lower column - start index than the first index of the new vector */ - orignz = mat_nonzeros(mat); - j = (i >= orignz ? colnr1 : ROW_MAT_COLNR(i)); - - /* Index of the column-top insertion point for the first new value */ - origidx = mat->col_end[colnr1-1]; - colnr = (origidx >= orignz ? colnr1 : COL_MAT_COLNR(origidx)); - - if(j < colnr) { - origidx = newidx = mat->col_end[j-1]; - for( ; j < colnr; j++) { - /* Shift entries in current column */ - jj_j = mat->col_end[j]; - for( ; origidx < jj_j; origidx++) { - if(COL_MAT_ROWNR(origidx) != rowno) { - if(newidx != origidx) { - COL_MAT_COPY(newidx, origidx); - } - newidx++; - } - } - /* Update next column start index */ - mat->col_end[j] = newidx; - } - delta = newidx - origidx; /* The first stage element shrinkage count */ - } - else { - delta = 0; - newidx = origidx; - } - - /* Make sure we have sufficient space for any additional entries and move existing data down; - this ensures that we only have to relocate matrix elements up in the next stage */ - jj_j = MAX(0, (int) newnz + delta); - - j = !((orignz == lendense) && (newnz == orignz) && (delta1 == 0)) && (jj_j > 0) && (orignz > origidx); - - if ((j) && (jj_j > delta1)) - delta1 = jj_j; - - if((delta1 > 0) && (mat_nz_unused(mat) <= delta1) && !inc_mat_space(mat, delta1)) { - newnz = 0; - goto Done; - } - - if(j) { - COL_MAT_MOVE(origidx+jj_j, origidx, orignz-origidx); - origidx += jj_j; - orignz += jj_j; - } - - /* Start from the top of the first non-zero column of the new row */ - newnz = 0; - j = origidx - mat->col_end[colnr1-1]; - k = colnr1; /* Last column for which col_end is valid/updated */ - while((colnr1 <= lendense) || (origidx < orignz)) { - - /* Get the column index of the active update item */ - if(newnz < count) - colnr1 = colno[newnz]; - else - colnr1 = lendense + 1; - - /* Get coordinate of active existing matrix entries */ - if(origidx < orignz) { - rownr = COL_MAT_ROWNR(origidx); - colnr = COL_MAT_COLNR(origidx); - } - else { - if(colnr1 > lendense) - break; - rownr = rowno; - colnr = lendense + 1; - } - - /* Update column start position if we just crossed into a column */ - jj_j = origidx - j; - i = MIN(colnr, colnr1); - for(; k < i; k++) - mat->col_end[k] = jj_j; - - /* Test if there is an available new item ... */ - if(colnr1 > colnr) { /* If this is not the case */ - /* DELETE if there is an existing value */ - if(rownr == rowno) { -ForceDelete: - j++; - delta--; - origidx++; - continue; - } - } - else if((colnr > colnr1) || /* Existing column index > new => INSERT */ - ((colnr == colnr1) && (rownr >= rowno)) ) { /* Same column index, existing row >= target row => INSERT/REPLACE */ - - value = row[newnz]; - newnz++; - if(isA && doscale) - value = scaled_mat(lp, value, rowno, colnr1); - if(isA) - value = my_chsign(is_chsign(lp, rowno), value); -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); - if(value == 0) { - if((colnr > colnr1) || (rownr > rowno)) - ; - else - goto ForceDelete; - } -#endif - SET_MAT_ijA(jj_j, rowno, colnr1, value); - - /* Adjust if we have inserted an element */ - if((colnr > colnr1) || (rownr > rowno)) { - j--; - origidx--; - jj_j++; - delta++; - } - origidx++; - continue; - } - - /* Shift the matrix element up by the active difference */ - if(jj_j != origidx) { - COL_MAT_COPY(jj_j, origidx); - } - origidx++; - } - - /* Update pending / incomplete column start position */ - jj_j = origidx - j; - for(; k <= lendense; k++) - mat->col_end[k] = jj_j; - mat->row_end_valid = FALSE; - -Done: - if(!isNZ) - FREE(row); - FREE(colno); - return( (MYBOOL) (newnz > 0) ); - -} /* mat_setrow */ -#endif - -STATIC int mat_appendrow(MATrec *mat, int count, REAL *row, int *colno, REAL mult, MYBOOL checkrowmode) -{ - int i, j, jj = 0, stcol, elmnr, orignr, newnr, firstcol; - MYBOOL *addto = NULL, isA, isNZ; - REAL value, saved = 0; - lprec *lp = mat->lp; - - /* Check if we are in row order mode and should add as column instead; - the matrix will be transposed at a later stage */ - if(checkrowmode && mat->is_roworder) - return( mat_appendcol(mat, count, row, colno, mult, FALSE) ); - - /* Do initialization and validation */ - isA = (MYBOOL) (mat == lp->matA); - isNZ = (MYBOOL) (colno != NULL); - if(isNZ && (count > 0)) { - if(count > 1) - sortREALByINT(row, colno, count, 0, TRUE); - if((colno[0] < 1) || (colno[count-1] > mat->columns)) - return( 0 ); - } - /* else if((row != NULL) && !mat->is_roworder) */ - else if(!isNZ && (row != NULL) && !mat->is_roworder) - row[0] = 0; - - /* Capture OF definition in row mode */ - if(isA && mat->is_roworder) { - if(isNZ && (colno[0] == 0)) { - value = row[0]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - value = scaled_mat(lp, value, 0, lp->columns); - value = my_chsign(is_maxim(lp), value); - lp->orig_obj[lp->columns] = value; - count--; - row++; - colno++; - } - else if(!isNZ && (row != NULL) && (row[0] != 0)) { - value = saved = row[0]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - value = scaled_mat(lp, value, 0, lp->columns); - value = my_chsign(is_maxim(lp), value); - lp->orig_obj[lp->columns] = value; - row[0] = 0; - } - else - lp->orig_obj[lp->columns] = 0; - } - - /* Optionally tally and map the new non-zero values */ - firstcol = mat->columns + 1; - if(isNZ) { - newnr = count; - if(newnr) { - firstcol = colno[0]; - jj = colno[newnr - 1]; - } - } - else { - newnr = 0; - if(row != NULL) { - if(!allocMYBOOL(lp, &addto, mat->columns + 1, TRUE)) { - return( newnr ); - } - for(i = mat->columns; i >= 1; i--) { - if(fabs(row[i]) > mat->epsvalue) { - addto[i] = TRUE; - firstcol = i; - newnr++; - } - } - } - } - - /* Make sure we have sufficient space */ - if(!inc_mat_space(mat, newnr)) { - newnr = 0; - goto Done; - } - - /* Insert the non-zero constraint values */ - orignr = mat_nonzeros(mat) - 1; - elmnr = orignr + newnr; - - for(j = mat->columns; j >= firstcol; j--) { - stcol = mat->col_end[j] - 1; - mat->col_end[j] = elmnr + 1; - - /* Add a new non-zero entry */ - if(((isNZ) && (j == jj)) || ((addto != NULL) && (addto[j]))) { - newnr--; - if(isNZ) { - value = row[newnr]; - if(newnr) - jj = colno[newnr - 1]; - else - jj = 0; - } - else - value = row[j]; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - value *= mult; - if(isA) { - if(mat->is_roworder) - value = my_chsign(is_chsign(lp, j), value); - value = scaled_mat(lp, value, mat->rows, j); - } - SET_MAT_ijA(elmnr, mat->rows, j, value); - elmnr--; - } - - /* Shift previous column entries down */ - i = stcol - mat->col_end[j-1] + 1; - if(i > 0) { - orignr -= i; - elmnr -= i; - COL_MAT_MOVE(elmnr+1, orignr+1, i); - } - } - -Done: - if(saved != 0) - row[0] = saved; - FREE(addto); - - return( newnr ); - -} - -STATIC int mat_appendcol(MATrec *mat, int count, REAL *column, int *rowno, REAL mult, MYBOOL checkrowmode) -{ - int i, row, elmnr, lastnr; - REAL value; - MYBOOL isA, isNZ; - lprec *lp = mat->lp; - - /* Check if we are in row order mode and should add as row instead; - the matrix will be transposed at a later stage */ - if(checkrowmode && mat->is_roworder) - return( mat_appendrow(mat, count, column, rowno, mult, FALSE) ); - - /* Make sure we have enough space */ -/* - if(!inc_mat_space(mat, mat->rows+1)) - return( 0 ); -*/ - if(column == NULL) - i = 0; - else if(rowno != NULL) - i = count; - else { - int nrows = mat->rows; - - elmnr = 0; - for(i = 1; i <= nrows; i++) - if(column[i] != 0) - elmnr++; - i = elmnr; - } - if((mat_nz_unused(mat) <= i) && !inc_mat_space(mat, i)) - return( 0 ); - - /* Do initialization and validation */ - isA = (MYBOOL) (mat == lp->matA); - isNZ = (MYBOOL) (column == NULL || rowno != NULL); - if(isNZ && (count > 0)) { - if(count > 1) - sortREALByINT(column, rowno, count, 0, TRUE); - if((rowno[0] < 0)) - return( 0 ); - } - if(rowno != NULL) - count--; - - /* Append sparse regular constraint values */ - elmnr = mat->col_end[mat->columns - 1]; - if(column != NULL) { - row = -1; - for(i = ((isNZ || !mat->is_roworder) ? 0 : 1); i <= count ; i++) { - value = column[i]; - if(fabs(value) > mat->epsvalue) { - if(isNZ) { - lastnr = row; - row = rowno[i]; - /* Check if we have come to the Lagrangean constraints */ - if(row > mat->rows) - break; - if(row <= lastnr) - return( -1 ); - } - else - row = i; -#ifdef DoMatrixRounding - value = roundToPrecision(value, mat->epsvalue); -#endif - if(mat->is_roworder) - value *= mult; - else if(isA) { - value = my_chsign(is_chsign(lp, row), value); - value = scaled_mat(lp, value, row, mat->columns); - if(!mat->is_roworder && (row == 0)) { - lp->orig_obj[mat->columns] = value; - continue; - } - } - - /* Store the item and update counters */ - SET_MAT_ijA(elmnr, row, mat->columns, value); - elmnr++; - } - } - - /* Fill dense Lagrangean constraints */ - if(get_Lrows(lp) > 0) - mat_appendcol(lp->matL, get_Lrows(lp), column+mat->rows, NULL, mult, checkrowmode); - - } - - /* Set end of data */ - mat->col_end[mat->columns] = elmnr; - - return( mat->col_end[mat->columns] - mat->col_end[mat->columns-1] ); -} - -STATIC int mat_checkcounts(MATrec *mat, int *rownum, int *colnum, MYBOOL freeonexit) -{ - int i, j, n; - int *rownr; - - if(rownum == NULL) - allocINT(mat->lp, &rownum, mat->rows + 1, TRUE); - if(colnum == NULL) - allocINT(mat->lp, &colnum, mat->columns + 1, TRUE); - - for(i = 1 ; i <= mat->columns; i++) { - j = mat->col_end[i - 1]; - n = mat->col_end[i]; - rownr = &COL_MAT_ROWNR(j); - for(; j < n; - j++, rownr += matRowColStep) { - colnum[i]++; - rownum[*rownr]++; - } - } - - n = 0; - if((mat->lp->do_presolve != PRESOLVE_NONE) && - (mat->lp->spx_trace || (mat->lp->verbose > NORMAL))) { - for(j = 1; j <= mat->columns; j++) - if(colnum[j] == 0) { - n++; - report(mat->lp, FULL, "mat_checkcounts: Variable %s is not used in any constraints\n", - get_col_name(mat->lp, j)); - } - for(i = 0; i <= mat->rows; i++) - if(rownum[i] == 0) { - n++; - report(mat->lp, FULL, "mat_checkcounts: Constraint %s empty\n", - get_row_name(mat->lp, i)); - } - } - - if(freeonexit) { - FREE(rownum); - FREE(colnum); - } - - return( n ); - -} - -STATIC MYBOOL mat_validate(MATrec *mat) -/* Routine to make sure that row mapping arrays are valid */ -{ - int i, j, je, *rownum; - int *rownr, *colnr; - - if(!mat->row_end_valid) { - - MEMCLEAR(mat->row_end, mat->rows + 1); - allocINT(mat->lp, &rownum, mat->rows + 1, TRUE); - - /* First tally row counts and then cumulate them */ - j = mat_nonzeros(mat); - rownr = &COL_MAT_ROWNR(0); - for(i = 0; i < j; i++, rownr += matRowColStep) - mat->row_end[*rownr]++; - for(i = 1; i <= mat->rows; i++) - mat->row_end[i] += mat->row_end[i - 1]; - - /* Calculate the column index for every non-zero */ - for(i = 1; i <= mat->columns; i++) { - j = mat->col_end[i - 1]; - je = mat->col_end[i]; - rownr = &COL_MAT_ROWNR(j); - colnr = &COL_MAT_COLNR(j); - for(; j < je; j++, rownr += matRowColStep, colnr += matRowColStep) { -#ifdef Paranoia - if(/*(*colnr < 0) || (*colnr > mat->columns) || (Normally violated in primal phase 1) */ - (*rownr < 0) || (*rownr > mat->rows)) { - report(mat->lp, SEVERE, "mat_validate: Matrix value storage error row %d [0..%d], column %d [1..%d]\n", - *rownr, mat->rows, *colnr, mat->columns); - mat->lp->spx_status = UNKNOWNERROR; - return(FALSE); - } -#endif - *colnr = i; - if(*rownr == 0) - mat_set_rowmap(mat, rownum[*rownr], - *rownr, i, j); - else - mat_set_rowmap(mat, mat->row_end[*rownr - 1] + rownum[*rownr], - *rownr, i, j); - rownum[*rownr]++; - } - } - - FREE(rownum); - mat->row_end_valid = TRUE; - } - - if(mat == mat->lp->matA) - mat->lp->model_is_valid = TRUE; - return( TRUE ); -} - -MYBOOL mat_get_data(lprec *lp, int matindex, MYBOOL isrow, int **rownr, int **colnr, REAL **value) -{ - MATrec *mat = lp->matA; - -#if MatrixRowAccess == RAM_Index - if(isrow) - matindex = mat->row_mat[matindex]; - if(rownr != NULL) - *rownr = &COL_MAT_ROWNR(matindex); - if(colnr != NULL) - *colnr = &COL_MAT_COLNR(matindex); - if(value != NULL) - *value = &COL_MAT_VALUE(matindex); - -#else - if(isrow) { - if(rownr != NULL) - *rownr = &ROW_MAT_ROWNR(matindex); - if(colnr != NULL) - *colnr = &ROW_MAT_COLNR(matindex); - if(value != NULL) - *value = &ROW_MAT_VALUE(matindex); - } - else { - if(rownr != NULL) - *rownr = &COL_MAT_ROWNR(matindex); - if(colnr != NULL) - *colnr = &COL_MAT_COLNR(matindex); - if(value != NULL) - *value = &COL_MAT_VALUE(matindex); - } - -#endif - - return( TRUE ); -} - - -MYBOOL mat_set_rowmap(MATrec *mat, int row_mat_index, int rownr, int colnr, int col_mat_index) -{ -#if MatrixRowAccess == RAM_Index - mat->row_mat[row_mat_index] = col_mat_index; - -#elif MatrixColAccess==CAM_Record - mat->row_mat[row_mat_index].rownr = rownr; - mat->row_mat[row_mat_index].colnr = colnr; - mat->row_mat[row_mat_index].value = COL_MAT_VALUE(col_mat_index); - -#else /* if MatrixColAccess==CAM_Vector */ - mat->row_mat_rownr[row_mat_index] = rownr; - mat->row_mat_colnr[row_mat_index] = colnr; - mat->row_mat_value[row_mat_index] = COL_MAT_VALUE(col_mat_index); - -#endif - - return( TRUE ); -} - -/* Implement combined binary/linear sub-search for matrix look-up */ -int mat_findelm(MATrec *mat, int row, int column) -{ - int low, high, mid, item; - -#if 0 - if(mat->row_end_valid && (row > 0) && - (ROW_MAT_COLNR(mat->row_mat[(low = mat->row_end[row-1])]) == column)) - return(low); -#endif - - if((column < 1) || (column > mat->columns)) { - report(mat->lp, IMPORTANT, "mat_findelm: Column %d out of range\n", column); - return( -1 ); - } - if((row < 0) || (row > mat->rows)) { - report(mat->lp, IMPORTANT, "mat_findelm: Row %d out of range\n", row); - return( -1 ); - } - - low = mat->col_end[column - 1]; - high = mat->col_end[column] - 1; - if(low > high) - return( -2 ); - - /* Do binary search logic */ - mid = (low+high) / 2; - item = COL_MAT_ROWNR(mid); - while(high - low > LINEARSEARCH) { - if(item < row) { - low = mid + 1; - mid = (low+high) / 2; - item = COL_MAT_ROWNR(mid); - } - else if(item > row) { - high = mid - 1; - mid = (low+high) / 2; - item = COL_MAT_ROWNR(mid); - } - else { - low = mid; - high = mid; - } - } - - /* Do linear scan search logic */ - if((high > low) && (high - low <= LINEARSEARCH)) { - item = COL_MAT_ROWNR(low); - while((low < high) && (item < row)) { - low++; - item = COL_MAT_ROWNR(low); - } - if(item == row) - high = low; - } - - if((low == high) && (row == item)) - return( low ); - else - return( -2 ); -} - -int mat_findins(MATrec *mat, int row, int column, int *insertpos, MYBOOL validate) -{ - int low, high, mid, item, exitvalue, insvalue; - -#if 0 - if(mat->row_end_valid && (row > 0) && - (ROW_MAT_COLNR(mat->row_mat[(low = mat->row_end[row-1])]) == column)) { - insvalue = low; - exitvalue = low; - goto Done; - } -#endif - - insvalue = -1; - - if((column < 1) || (column > mat->columns)) { - if((column > 0) && !validate) { - insvalue = mat->col_end[mat->columns]; - exitvalue = -2; - goto Done; - } - report(mat->lp, IMPORTANT, "mat_findins: Column %d out of range\n", column); - exitvalue = -1; - goto Done; - } - if((row < 0) || (row > mat->rows)) { - if((row >= 0) && !validate) { - insvalue = mat->col_end[column]; - exitvalue = -2; - goto Done; - } - report(mat->lp, IMPORTANT, "mat_findins: Row %d out of range\n", row); - exitvalue = -1; - goto Done; - } - - low = mat->col_end[column - 1]; - insvalue = low; - high = mat->col_end[column] - 1; - if(low > high) { - exitvalue = -2; - goto Done; - } - - /* Do binary search logic */ - mid = (low+high) / 2; - item = COL_MAT_ROWNR(mid); - while(high - low > LINEARSEARCH) { - if(item < row) { - low = mid + 1; - mid = (low+high) / 2; - item = COL_MAT_ROWNR(mid); - } - else if(item > row) { - high = mid - 1; - mid = (low+high) / 2; - item = COL_MAT_ROWNR(mid); - } - else { - low = mid; - high = mid; - } - } - - /* Do linear scan search logic */ - if((high > low) && (high - low <= LINEARSEARCH)) { - item = COL_MAT_ROWNR(low); - while((low < high) && (item < row)) { - low++; - item = COL_MAT_ROWNR(low); - } - if(item == row) - high = low; - } - - insvalue = low; - if((low == high) && (row == item)) - exitvalue = low; - else { - if((low < mat->col_end[column]) && (COL_MAT_ROWNR(low) < row)) - insvalue++; - exitvalue = -2; - } - -Done: - if(insertpos != NULL) - (*insertpos) = insvalue; - return( exitvalue ); -} - -STATIC REAL mat_getitem(MATrec *mat, int row, int column) -{ - int elmnr; - -#ifdef DirectOverrideOF - if((row == 0) && (mat == mat->lp->matA) && (mat->lp->OF_override != NULL)) - return( mat->lp->OF_override[column] ); - else -#endif - { - elmnr = mat_findelm(mat, row, column); - if(elmnr >= 0) - return( COL_MAT_VALUE(elmnr) ); - else - return( 0 ); - } -} - -STATIC MYBOOL mat_additem(MATrec *mat, int row, int column, REAL delta) -{ - int elmnr; - -#ifdef DirectOverrideOF - if((row == 0) && (mat == mat->lp->matA) && (mat->lp->OF_override != NULL)) - return( mat->lp->OF_override[column] ); - else -#endif - { - elmnr = mat_findelm(mat, row, column); - if(elmnr >= 0) { - COL_MAT_VALUE(elmnr) += delta; - return( TRUE ); - } - else { - mat_setitem(mat, row, column, delta); - return( FALSE ); - } - } -} - -STATIC MYBOOL mat_setitem(MATrec *mat, int row, int column, REAL value) -{ - return( mat_setvalue(mat, row, column, value, FALSE) ); -} - -STATIC void mat_multrow(MATrec *mat, int row_nr, REAL mult) -{ - int i, k1, k2; - -#if 0 - if(row_nr == 0) { - k2 = mat->col_end[0]; - for(i = 1; i <= mat->columns; i++) { - k1 = k2; - k2 = mat->col_end[i]; - if((k1 < k2) && (COL_MAT_ROWNR(k1) == row_nr)) - COL_MAT_VALUE(k1) *= mult; - } - } - else if(mat_validate(mat)) { - if(row_nr == 0) - k1 = 0; - else -#else - if(mat_validate(mat)) { - if(row_nr == 0) - k1 = 0; - else -#endif - k1 = mat->row_end[row_nr-1]; - k2 = mat->row_end[row_nr]; - for(i = k1; i < k2; i++) - ROW_MAT_VALUE(i) *= mult; - } -} - -STATIC void mat_multcol(MATrec *mat, int col_nr, REAL mult, MYBOOL DoObj) -{ - int i, ie; - MYBOOL isA; - -#ifdef Paranoia - if((col_nr < 1) || (col_nr > mat->columns)) { - report(mat->lp, IMPORTANT, "mult_column: Column %d out of range\n", col_nr); - return; - } -#endif - if(mult == 1.0) - return; - - isA = (MYBOOL) (mat == mat->lp->matA); - - ie = mat->col_end[col_nr]; - for(i = mat->col_end[col_nr - 1]; i < ie; i++) - COL_MAT_VALUE(i) *= mult; - if(isA) { - if(DoObj) - mat->lp->orig_obj[col_nr] *= mult; - if(get_Lrows(mat->lp) > 0) - mat_multcol(mat->lp->matL, col_nr, mult, DoObj); - } -} - -STATIC void mat_multadd(MATrec *mat, REAL *lhsvector, int varnr, REAL mult) -{ - int colnr; - register int ib, ie, *matRownr; - register REAL *matValue; - - /* Handle case of a slack variable */ - if(varnr <= mat->lp->rows) { - lhsvector[varnr] += mult; - return; - } - - /* Do operation on the objective */ - if(mat->lp->matA == mat) - lhsvector[0] += get_OF_active(mat->lp, varnr, mult); - - /* Scan the constraint matrix target columns */ - colnr = varnr - mat->lp->rows; - ib = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - if(ib < ie) { - - /* Initialize pointers */ - matRownr = &COL_MAT_ROWNR(ib); - matValue = &COL_MAT_VALUE(ib); - - /* Then loop over all regular rows */ - for(; ib < ie; - ib++, matValue += matValueStep, matRownr += matRowColStep) { - lhsvector[*matRownr] += mult * (*matValue); - } - } - -} - -STATIC MYBOOL mat_setvalue(MATrec *mat, int Row, int Column, REAL Value, MYBOOL doscale) -{ - int elmnr, lastelm, i, RowA = Row, ColumnA = Column; - MYBOOL isA; - - /* This function is inefficient if used to add new matrix entries in - other places than at the end of the matrix. OK for replacing existing - a non-zero value with another non-zero value */ - isA = (MYBOOL) (mat == mat->lp->matA); - if(mat->is_roworder) - swapINT(&Row, &Column); - - /* Set small numbers to zero */ - if(fabs(Value) < mat->epsvalue) - Value = 0; -#ifdef DoMatrixRounding - else - Value = roundToPrecision(Value, mat->epsvalue); -#endif - - /* Check if we need to update column space */ - if(Column > mat->columns) { - if(isA) - inc_col_space(mat->lp, ColumnA - mat->columns); - else - inc_matcol_space(mat, Column - mat->columns); - } - - /* Find out if we already have such an entry, or return insertion point */ - i = mat_findins(mat, Row, Column, &elmnr, FALSE); - if(i == -1) - return(FALSE); - - if(isA) - set_action(&mat->lp->spx_action, ACTION_REBASE | ACTION_RECOMPUTE | ACTION_REINVERT); - - if(i >= 0) { - /* there is an existing entry */ - if(fabs(Value) > mat->epsvalue) { /* we replace it by something non-zero */ - if(isA) { - Value = my_chsign(is_chsign(mat->lp, RowA), Value); - if(doscale && mat->lp->scaling_used) - Value = scaled_mat(mat->lp, Value, RowA, ColumnA); - } - COL_MAT_VALUE(elmnr) = Value; - } - else { /* setting existing non-zero entry to zero. Remove the entry */ - /* This might remove an entire column, or leave just a bound. No - nice solution for that yet */ - - /* Shift up tail end of the matrix */ - lastelm = mat_nonzeros(mat); -#if 0 - for(i = elmnr; i < lastelm ; i++) { - COL_MAT_COPY(i, i + 1); - } -#else - lastelm -= elmnr; - COL_MAT_MOVE(elmnr, elmnr + 1, lastelm); -#endif - for(i = Column; i <= mat->columns; i++) - mat->col_end[i]--; - - mat->row_end_valid = FALSE; - } - } - else if(fabs(Value) > mat->epsvalue) { - /* no existing entry. make new one only if not nearly zero */ - /* check if more space is needed for matrix */ - if(!inc_mat_space(mat, 1)) - return(FALSE); - - if(Column > mat->columns) { - i = mat->columns + 1; - if(isA) - shift_coldata(mat->lp, i, ColumnA - mat->columns, NULL); - else - mat_shiftcols(mat, &i, Column - mat->columns, NULL); - } - - /* Shift down tail end of the matrix by one */ - lastelm = mat_nonzeros(mat); -#if 1 /* Does compiler optimization work better here? */ - for(i = lastelm; i > elmnr ; i--) { - COL_MAT_COPY(i, i - 1); - } -#else - lastelm -= elmnr - 1; - COL_MAT_MOVE(elmnr + 1, elmnr, lastelm); -#endif - - /* Set new element */ - if(isA) { - Value = my_chsign(is_chsign(mat->lp, RowA), Value); - if(doscale) - Value = scaled_mat(mat->lp, Value, RowA, ColumnA); - } - SET_MAT_ijA(elmnr, Row, Column, Value); - - /* Update column indexes */ - for(i = Column; i <= mat->columns; i++) - mat->col_end[i]++; - - mat->row_end_valid = FALSE; - } - - if(isA && (mat->lp->var_is_free != NULL) && (mat->lp->var_is_free[ColumnA] > 0)) - return( mat_setvalue(mat, RowA, mat->lp->var_is_free[ColumnA], -Value, doscale) ); - return(TRUE); -} - -STATIC MYBOOL mat_appendvalue(MATrec *mat, int Row, REAL Value) -{ - int *elmnr, Column = mat->columns; - - /* Set small numbers to zero */ - if(fabs(Value) < mat->epsvalue) - Value = 0; -#ifdef DoMatrixRounding - else - Value = roundToPrecision(Value, mat->epsvalue); -#endif - - /* Check if more space is needed for matrix */ - if(!inc_mat_space(mat, 1)) - return(FALSE); - -#ifdef Paranoia - /* Check valid indeces */ - if((Row < 0) || (Row > mat->rows)) { - report(mat->lp, SEVERE, "mat_appendvalue: Invalid row index %d specified\n", Row); - return(FALSE); - } -#endif - - /* Get insertion point and set value */ - elmnr = mat->col_end + Column; - SET_MAT_ijA((*elmnr), Row, Column, Value); - - /* Update column count */ - (*elmnr)++; - mat->row_end_valid = FALSE; - - return(TRUE); -} - -STATIC MYBOOL mat_equalRows(MATrec *mat, int baserow, int comprow) -{ - MYBOOL status = FALSE; - - if(mat_validate(mat)) { - int bj1 = 0, ej1, bj2 = 0, ej2; - - /* Get starting and ending positions */ - if(baserow >= 0) - bj1 = mat->row_end[baserow-1]; - ej1 = mat->row_end[baserow]; - if(comprow >= 0) - bj2 = mat->row_end[comprow-1]; - ej2 = mat->row_end[comprow]; - /* Fail if row lengths are unequal */ - if((ej1-bj1) != (ej2-bj2)) - return( status ); - - /* Compare column index and value, element by element */ - for(; bj1 < ej1; bj1++, bj2++) { - if(COL_MAT_COLNR(bj1) != COL_MAT_COLNR(bj2)) - break; -#if 1 - if(fabs(get_mat_byindex(mat->lp, bj1, TRUE, FALSE)-get_mat_byindex(mat->lp, bj2, TRUE, FALSE)) > mat->lp->epsprimal) -#else - if(fabs(COL_MAT_VALUE(bj1)-COL_MAT_VALUE(bj2)) > mat->lp->epsprimal) -#endif - break; - } - status = (MYBOOL) (bj1 == ej1); - } - return( status ); -} - -STATIC int mat_findcolumn(MATrec *mat, int matindex) -{ - int j; - - for(j = 1; j <= mat->columns; j++) { - if(matindex < mat->col_end[j]) - break; - } - return(j); -} - -STATIC int mat_expandcolumn(MATrec *mat, int colnr, REAL *column, int *nzlist, MYBOOL signedA) -{ - MYBOOL isA = (MYBOOL) (mat->lp->matA == mat); - int i, ie, j, nzcount = 0; - REAL *matValue; - int *matRownr; - - signedA &= isA; - - /* Retrieve a column from the user data matrix A */ - MEMCLEAR(column, mat->rows + 1); - if(isA) { - column[0] = mat->lp->orig_obj[colnr]; - if(signedA && is_chsign(mat->lp, 0)) - column[0] = -column[0]; - } - - i = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - matRownr = &COL_MAT_ROWNR(i); - matValue = &COL_MAT_VALUE(i); - for(; i < ie; - i++, matRownr += matRowColStep, matValue += matValueStep) { - j = *matRownr; - column[j] = *matValue; - if(signedA && is_chsign(mat->lp, j)) - column[j] = -column[j]; - nzcount++; - if(nzlist != NULL) - nzlist[nzcount] = j; - } - if(nzlist != NULL) - nzlist[0] = nzcount; - return( nzcount ); -} - -STATIC MYBOOL mat_computemax(MATrec *mat) -{ - int *rownr = &COL_MAT_ROWNR(0), - *colnr = &COL_MAT_COLNR(0), - i = 0, ie = mat->col_end[mat->columns], ez = 0; - REAL *value = &COL_MAT_VALUE(0), epsmachine = mat->lp->epsmachine, absvalue; - - /* Prepare arrays */ - if(!allocREAL(mat->lp, &mat->colmax, mat->columns_alloc+1, AUTOMATIC) || - !allocREAL(mat->lp, &mat->rowmax, mat->rows_alloc+1, AUTOMATIC)) - return( FALSE ); - MEMCLEAR(mat->colmax, mat->columns+1); - MEMCLEAR(mat->rowmax, mat->rows+1); - - /* Obtain the row and column maxima in one sweep */ - mat->dynrange = mat->lp->infinite; - for(; i < ie; - i++, rownr += matRowColStep, colnr += matRowColStep, value += matValueStep) { - absvalue = fabs(*value); - SETMAX(mat->colmax[*colnr], absvalue); - SETMAX(mat->rowmax[*rownr], absvalue); - SETMIN(mat->dynrange, absvalue); - if(absvalue < epsmachine) - ez++; - } - - /* Lastly, compute the global maximum and get the dynamic range */ - for(i = 1; i <= mat->rows; i++) - SETMAX(mat->rowmax[0], mat->rowmax[i]); - mat->infnorm = mat->colmax[0] = mat->rowmax[0]; - if(mat->dynrange == 0) { - report(mat->lp, SEVERE, "%d matrix contains zero-valued coefficients.\n", ez); - mat->dynrange = mat->lp->infinite; - } - else { - mat->dynrange = mat->infnorm / mat->dynrange; - if(ez > 0) - report(mat->lp, IMPORTANT, "%d matrix coefficients below machine precision were found.\n", ez); - } - - return( TRUE ); -} - -STATIC MYBOOL mat_transpose(MATrec *mat) -{ - int i, j, nz, k; - MYBOOL status; - - status = mat_validate(mat); - if(status) { - - /* Create a column-ordered sparse element list; "column" index must be shifted */ - nz = mat_nonzeros(mat); - if(nz > 0) { -#if MatrixColAccess==CAM_Record - MATitem *newmat; - newmat = (MATitem *) malloc((mat->mat_alloc) * sizeof(*(mat->col_mat))); - j = mat->row_end[0]; - for(i = nz-1; i >= j ; i--) { - k = i-j; - newmat[k] = mat->col_mat[mat->row_mat[i]]; - newmat[k].row_nr = newmat[k].col_nr; - } - for(i = j-1; i >= 0 ; i--) { - k = nz-j+i; - newmat[k] = mat->col_mat[mat->row_mat[i]]; - newmat[k].row_nr = newmat[k].col_nr; - } - swapPTR((void **) &mat->col_mat, (void **) &newmat); - FREE(newmat); -#else /*if MatrixColAccess==CAM_Vector*/ - REAL *newValue = NULL; - int *newRownr = NULL; - allocREAL(mat->lp, &newValue, mat->mat_alloc, FALSE); - allocINT(mat->lp, &newRownr, mat->mat_alloc, FALSE); - - j = mat->row_end[0]; - for(i = nz-1; i >= j ; i--) { - k = i-j; - newValue[k] = ROW_MAT_VALUE(i); - newRownr[k] = ROW_MAT_COLNR(i); - } - for(i = j-1; i >= 0 ; i--) { - k = nz-j+i; - newValue[k] = ROW_MAT_VALUE(i); - newRownr[k] = ROW_MAT_COLNR(i); - } - - swapPTR((void **) &mat->col_mat_rownr, (void **) &newRownr); - swapPTR((void **) &mat->col_mat_value, (void **) &newValue); - FREE(newValue); - FREE(newRownr); -#endif - } - - /* Transfer row start to column start position; must adjust for different offsets */ - if(mat->rows == mat->rows_alloc) - inc_matcol_space(mat, 1); - j = mat->row_end[0]; - for(i = mat->rows; i >= 1; i--) - mat->row_end[i] -= j; - mat->row_end[mat->rows] = nz; - swapPTR((void **) &mat->row_end, (void **) &mat->col_end); - - /* Swap arrays of maximum values */ - swapPTR((void **) &mat->rowmax, (void **) &mat->colmax); - - /* Swap array sizes */ - swapINT(&mat->rows, &mat->columns); - swapINT(&mat->rows_alloc, &mat->columns_alloc); - - /* Finally set current storage mode */ - mat->is_roworder = (MYBOOL) !mat->is_roworder; - mat->row_end_valid = FALSE; - } - return(status); -} - - -/* ---------------------------------------------------------------------------------- */ -/* Change-tracking routines */ -/* ---------------------------------------------------------------------------------- */ -STATIC DeltaVrec *createUndoLadder(lprec *lp, int levelitems, int maxlevels) -{ - DeltaVrec *hold; - - hold = (DeltaVrec *) malloc(sizeof(*hold)); - hold->lp = lp; - hold->activelevel = 0; - hold->tracker = mat_create(lp, levelitems, 0, 0.0); - inc_matcol_space(hold->tracker, maxlevels); - return( hold ); -} -STATIC int incrementUndoLadder(DeltaVrec *DV) -{ - DV->activelevel++; - inc_matcol_space(DV->tracker, 1); - mat_shiftcols(DV->tracker, &(DV->activelevel), 1, NULL); - DV->tracker->columns++; - return(DV->activelevel); -} -STATIC MYBOOL modifyUndoLadder(DeltaVrec *DV, int itemno, REAL target[], REAL newvalue) -{ - MYBOOL status; - int varindex = itemno; - REAL oldvalue = target[itemno]; - -#ifndef UseMilpSlacksRCF /* Check if we should include ranged constraints */ - varindex -= DV->lp->rows; -#endif - status = mat_appendvalue(DV->tracker, varindex, oldvalue); - target[itemno] = newvalue; - return(status); -} -STATIC int countsUndoLadder(DeltaVrec *DV) -{ - if(DV->activelevel > 0) - return( mat_collength(DV->tracker, DV->activelevel) ); - else - return( 0 ); -} -STATIC int restoreUndoLadder(DeltaVrec *DV, REAL target[]) -{ - int iD = 0; - - if(DV->activelevel > 0) { - MATrec *mat = DV->tracker; - int iB = mat->col_end[DV->activelevel-1], - iE = mat->col_end[DV->activelevel], - *matRownr = &COL_MAT_ROWNR(iB); - REAL *matValue = &COL_MAT_VALUE(iB), - oldvalue; - - /* Restore the values */ - iD = iE-iB; - for(; iB < iE; iB++, matValue += matValueStep, matRownr += matRowColStep) { - oldvalue = *matValue; -#ifdef UseMilpSlacksRCF /* Check if we should include ranged constraints */ - target[(*matRownr)] = oldvalue; -#else - target[DV->lp->rows+(*matRownr)] = oldvalue; -#endif - } - - /* Get rid of the changes */ - mat_shiftcols(DV->tracker, &(DV->activelevel), -1, NULL); - } - - return(iD); -} -STATIC int decrementUndoLadder(DeltaVrec *DV) -{ - int deleted = 0; - - if(DV->activelevel > 0) { - deleted = mat_shiftcols(DV->tracker, &(DV->activelevel), -1, NULL); - DV->activelevel--; - DV->tracker->columns--; - } - return(deleted); -} -STATIC MYBOOL freeUndoLadder(DeltaVrec **DV) -{ - if((DV == NULL) || (*DV == NULL)) - return(FALSE); - - mat_free(&((*DV)->tracker)); - FREE(*DV); - return(TRUE); -} - -STATIC MYBOOL appendUndoPresolve(lprec *lp, MYBOOL isprimal, REAL beta, int colnrDep) -{ - MATrec *mat; - - /* Point to correct undo structure */ - if(isprimal) - mat = lp->presolve_undo->primalundo->tracker; - else - mat = lp->presolve_undo->dualundo->tracker; - - /* Append the data */ - if((colnrDep > 0) && (beta != 0) && - (mat != NULL) && (mat->col_tag[0] > 0)) { - int ix = mat->col_tag[0]; -#if 0 - report(lp, NORMAL, "appendUndoPresolve: %s %g * x%d\n", - ( beta < 0 ? "-" : "+"), fabs(beta), colnrDep); -#endif - - /* Do normal user variable case */ - if(colnrDep <= lp->columns) - mat_setvalue(mat, colnrDep, ix, beta, FALSE); - - /* Handle case where a slack variable is referenced */ - else { - int ipos, jx = mat->col_tag[ix]; - mat_setvalue(mat, jx, ix, beta, FALSE); - jx = mat_findins(mat, jx, ix, &ipos, FALSE); - COL_MAT_ROWNR(ipos) = colnrDep; - } - return( TRUE ); - } - else - return( FALSE ); -} -STATIC MYBOOL addUndoPresolve(lprec *lp, MYBOOL isprimal, int colnrElim, REAL alpha, REAL beta, int colnrDep) -{ - int ix; - DeltaVrec **DV; - MATrec *mat; - presolveundorec *psdata = lp->presolve_undo; - - /* Point to and initialize undo structure at first call */ - if(isprimal) { - DV = &(psdata->primalundo); - if(*DV == NULL) { - *DV = createUndoLadder(lp, lp->columns+1, lp->columns); - mat = (*DV)->tracker; - mat->epsvalue = lp->matA->epsvalue; - allocINT(lp, &(mat->col_tag), lp->columns+1, FALSE); - mat->col_tag[0] = 0; - } - } - else { - DV = &(psdata->dualundo); - if(*DV == NULL) { - *DV = createUndoLadder(lp, lp->rows+1, lp->rows); - mat = (*DV)->tracker; - mat->epsvalue = lp->matA->epsvalue; - allocINT(lp, &(mat->col_tag), lp->rows+1, FALSE); - mat->col_tag[0] = 0; - } - } - mat = (*DV)->tracker; -#if 0 - report(lp, NORMAL, "addUndoPresolve: x%d = %g %s %g * x%d\n", - colnrElim, alpha, ( beta < 0 ? "-" : "+"), fabs(beta), colnrDep); -#endif - /* Add the data */ - ix = mat->col_tag[0] = incrementUndoLadder(*DV); - mat->col_tag[ix] = colnrElim; - if(alpha != 0) - mat_setvalue(mat, 0, ix, alpha, FALSE); -/* mat_appendvalue(*mat, 0, alpha);*/ - if((colnrDep > 0) && (beta != 0)) { - if(colnrDep > lp->columns) - return( appendUndoPresolve(lp, isprimal, beta, colnrDep) ); - else - mat_setvalue(mat, colnrDep, ix, beta, FALSE); - } - - return( TRUE ); -} - - - -/* ---------------------------------------------------------------------------------- */ -/* High level matrix inverse and product routines in lp_solve */ -/* ---------------------------------------------------------------------------------- */ - -/* ---------------------------------------------------------------------------------- */ -/* A brief description of the basis inverse and factorization logic in lp_solve */ -/* ---------------------------------------------------------------------------------- */ -/* - - In order to better understand the legacy code for operating with the - basis and its factorization in lp_solve I (KE) will briefly explain - the conventions and associated matrix algebra. Note that with lp_solve - version 5.5, it is also possible to direct lp_solve to use the traditional - (textbook) format by setting the obj_in_B parameter to FALSE. - - The matrix description of a linear program (as represented by lp_solve) goes - like this: - - maximize c'x - subject to r <= Ax <= b - where l <= x <= u - - The matrix A is partitioned into two column sets [B|N], where B is - a square matrix of "basis" variables containing non-fixed - variables of the linear program at any given stage and N is the - submatrix of corresponding non-basic, fixed variables. The - variables (columns) in N may be fixed at their lower or upper levels. - - Similarly, the c vector is partitioned into the basic and non-basic - parts [z|n]. - - While lp_solve stores the objective vector c in a dense format, and - the constraint matrix A in a (fairly standard) sparse format, the - column vectors passed to the factorization routine include the - objective coefficient at row index 0. (In versions of lp_solve - before v5.2, c was actually explicitly stored as the 0-th row of A). - The expanded matrix may be called the "A~" form and looks like this: - - A~ = [ c ] - [ A ] - - Linear programming involves solving linear equations based on the - square basis matrix B, which includes is a subset of columns from A~. - The implications of the common storage of c and A (e.g. A~) vs. the - inverse / factorization of B for the operations and updates performed - by the simplex routine therefore needs to be understood. As a consquence - of A~, in lp_solve B is stored in an expanded, bordered format using the - following (non-singular) representation: - - B~ = [ 1 z ] - [ 0 B ] - - Any basis inversion / factorization engine used by lp_solve must therefore - explicitly represent and handle the implications of this structure for - associated matrix operations. - - The standard matrix formula for computing the inverse of a bordered - matrix shows what the inversion of B~ actually produces: - - Inv(B~) = [ 1 -z*Inv(B) ] - [ 0 Inv(B) ] - - The A~ and B~ representations require awareness by the developer of the side - effects of the presence of the top row when doing product operations such as - b'N, btran and ftran. Note in particular z*Inv(B) in the top row of Inv(B~), - which is actually the dual solution vector of the given basis. This fact - makes a very common update in the simplex algorithm (reduced costs) returnable - as a vector simply by setting 1 at the top of a vector being pre-multiplied - with Inv(B~). - - However, if the objective vector (c) is changed, the expanded representation - requires that B / B~ be refactorized. Also, when doing FTRAN, BTRAN - and x'A-type operations, you will patently get the incorrect result - if you simply copy the operations given in textbooks. First I'll show the - results of an FTRAN operation: - - Bx = a ==> x = FTRAN(a) - - In lp_solve, this operation solves: - - [ 1 z ] [y] = [d] - [ 0 B ] [x] [a] - - Using the Inv(B~) expression earlier, the FTRAN result is therefore: - - [y] = [ 1 -z*Inv(B) ] [d] = [ d - z*Inv(B)*a ] - [x] [ 0 Inv(B) ] [a] [ Inv(B)*a ] - - As an example, the value of the dual objective can be returned at the - 0-th index by passing the active RHS vector with 0 at the 0-th position. - - Similarily, doing the left solve - performing the BTRAN calculation: - - [x y] [ 1 z ] = [d a'] - [ 0 B ] - - ... will produce the following result in lp_solve: - - [x y] = [d a'] [ 1 -z*Inv(B) ] = [ d | -d*z*Inv(B) + a'*Inv(B) ] - [ 0 Inv(B) ] - - So, if you thought you were simply computing "a'*Inv(B)", look again. - In order to produce the desired result, you have to set d to 0 before - the BTRAN operation. On the other hand, if you set d to 1 and a to 0, - then you are very conveniently on your way to obtain the reduced costs - (needs a further matrix premultiplication with non-basic variables). - - Incidentally, the BTRAN with [1 0] that yields [ 1 | -z*Inv(B) ] can - also be used as a fast way of checking the accuracy of the current - factorization. - - Equipped with this understanding, I hope that you see that - the approach in lp_solve is actually pretty convenient. It also - becomes easier to extend functionality in lp_solve by drawing on - formulas and expressions from LP literature that otherwise assume - the non-bordered syntax and representation. - - Kjell Eikland -- November 2003 - KE update -- April 2005 - KE update -- June 2005 - -*/ - -STATIC MYBOOL __WINAPI invert(lprec *lp, MYBOOL shiftbounds, MYBOOL final) -{ - MYBOOL *usedpos, resetbasis; - REAL test; - int k, i, j; - int singularities, usercolB; - - /* Make sure the tags are correct */ - if(!mat_validate(lp->matA)) { - lp->spx_status = INFEASIBLE; - return(FALSE); - } - - /* Create the inverse management object at the first call to invert() */ - if(lp->invB == NULL) - lp->bfp_init(lp, lp->rows, 0, NULL); - else - lp->bfp_preparefactorization(lp); - singularities = 0; - - /* Must save spx_status since it is used to carry information about - the presence and handling of singular columns in the matrix */ - if(userabort(lp, MSG_INVERT)) - return(FALSE); - -#ifdef Paranoia - if(lp->spx_trace) - report(lp, DETAILED, "invert: Iter %10g, fact-length %7d, OF " RESULTVALUEMASK ".\n", - (double) get_total_iter(lp), lp->bfp_colcount(lp), (double) -lp->rhs[0]); -#endif - - /* Store state of pre-existing basis, and at the same time check if - the basis is I; in this case take the easy way out */ - if(!allocMYBOOL(lp, &usedpos, lp->sum + 1, TRUE)) { - lp->bb_break = TRUE; - return(FALSE); - } - usedpos[0] = TRUE; - usercolB = 0; - for(i = 1; i <= lp->rows; i++) { - k = lp->var_basic[i]; - if(k > lp->rows) - usercolB++; - usedpos[k] = TRUE; - } -#ifdef Paranoia - if(!verify_basis(lp)) - report(lp, SEVERE, "invert: Invalid basis detected (iter %g).\n", - (double) get_total_iter(lp)); -#endif - - /* Tally matrix nz-counts and check if we should reset basis - indicators to all slacks */ - resetbasis = (MYBOOL) ((usercolB > 0) && lp->bfp_canresetbasis(lp)); - k = 0; - for(i = 1; i <= lp->rows; i++) { - if(lp->var_basic[i] > lp->rows) - k += mat_collength(lp->matA, lp->var_basic[i] - lp->rows) + (is_OF_nz(lp,lp->var_basic[i] - lp->rows) ? 1 : 0); - if(resetbasis) { - j = lp->var_basic[i]; - if(j > lp->rows) - lp->is_basic[j] = FALSE; - lp->var_basic[i] = i; - lp->is_basic[i] = TRUE; - } - } - - /* Now do the refactorization */ - singularities = lp->bfp_factorize(lp, usercolB, k, usedpos, final); - - /* Do user reporting */ - if(userabort(lp, MSG_INVERT)) - goto Cleanup; - - /* Finalize factorization/inversion */ - lp->bfp_finishfactorization(lp); - - /* Recompute the RHS ( Ref. lp_solve inverse logic and Chvatal p. 121 ) */ -#ifdef DebugInv - blockWriteLREAL(stdout, "RHS-values pre invert", lp->rhs, 0, lp->rows); -#endif - recompute_solution(lp, shiftbounds); - restartPricer(lp, AUTOMATIC); -#ifdef DebugInv - blockWriteLREAL(stdout, "RHS-values post invert", lp->rhs, 0, lp->rows); -#endif - -Cleanup: - /* Check for numerical instability indicated by frequent refactorizations */ - test = get_refactfrequency(lp, FALSE); - if(test < MIN_REFACTFREQUENCY) { - test = get_refactfrequency(lp, TRUE); - report(lp, NORMAL, "invert: Refactorization frequency %.1g indicates numeric instability.\n", - test); - lp->spx_status = NUMFAILURE; - } - - FREE(usedpos); - return((MYBOOL) (singularities <= 0)); -} /* invert */ - - -STATIC MYBOOL fimprove(lprec *lp, REAL *pcol, int *nzidx, REAL roundzero) -{ - REAL *errors, sdp; - int j; - MYBOOL Ok = TRUE; - - allocREAL(lp, &errors, lp->rows + 1, FALSE); - if(errors == NULL) { - Ok = FALSE; - return(Ok); - } - MEMCOPY(errors, pcol, lp->rows + 1); - lp->bfp_ftran_normal(lp, pcol, nzidx); - prod_Ax(lp, NULL, pcol, NULL, 0.0, -1, - errors, NULL, MAT_ROUNDDEFAULT); - lp->bfp_ftran_normal(lp, errors, NULL); - - sdp = 0; - for(j = 1; j <= lp->rows; j++) - if(fabs(errors[j])>sdp) - sdp = fabs(errors[j]); - if(sdp > lp->epsmachine) { - report(lp, DETAILED, "Iterative FTRAN correction metric %g", sdp); - for(j = 1; j <= lp->rows; j++) { - pcol[j] += errors[j]; - my_roundzero(pcol[j], roundzero); - } - } - FREE(errors); - return(Ok); -} - -STATIC MYBOOL bimprove(lprec *lp, REAL *rhsvector, int *nzidx, REAL roundzero) -{ - int j; - REAL *errors, err, maxerr; - MYBOOL Ok = TRUE; - - allocREAL(lp, &errors, lp->sum + 1, FALSE); - if(errors == NULL) { - Ok = FALSE; - return(Ok); - } - MEMCOPY(errors, rhsvector, lp->sum + 1); - - /* Solve Ax=b for x, compute b back */ - lp->bfp_btran_normal(lp, errors, nzidx); - prod_xA(lp, NULL, errors, NULL, 0.0, 1.0, - errors, NULL, - MAT_ROUNDDEFAULT); - - /* Take difference with ingoing values, while shifting the column values - to the rows section and zeroing the columns again */ - for(j = 1; j <= lp->rows; j++) - errors[j] = errors[lp->rows+lp->var_basic[j]] - rhsvector[j]; - for(j = lp->rows; j <= lp->sum; j++) - errors[j] = 0; - - /* Solve the b errors for the iterative x adjustment */ - lp->bfp_btran_normal(lp, errors, NULL); - - /* Generate the adjustments and compute statistic */ - maxerr = 0; - for(j = 1; j <= lp->rows; j++) { - if(lp->var_basic[j]<=lp->rows) continue; - err = errors[lp->rows+lp->var_basic[j]]; - if(fabs(err)>maxerr) - maxerr = fabs(err); - } - if(maxerr > lp->epsmachine) { - report(lp, DETAILED, "Iterative BTRAN correction metric %g", maxerr); - for(j = 1; j <= lp->rows; j++) { - if(lp->var_basic[j]<=lp->rows) continue; - rhsvector[j] += errors[lp->rows+lp->var_basic[j]]; - my_roundzero(rhsvector[j], roundzero); - } - } - FREE(errors); - return(Ok); -} - -STATIC void ftran(lprec *lp, REAL *rhsvector, int *nzidx, REAL roundzero) -{ -#if 0 - if(is_action(lp->improve, IMPROVE_SOLUTION) && lp->bfp_pivotcount(lp)) - fimprove(lp, rhsvector, nzidx, roundzero); - else -#endif - lp->bfp_ftran_normal(lp, rhsvector, nzidx); -} - -STATIC void btran(lprec *lp, REAL *rhsvector, int *nzidx, REAL roundzero) -{ -#if 0 - if(is_action(lp->improve, IMPROVE_SOLUTION) && lp->bfp_pivotcount(lp)) - bimprove(lp, rhsvector, nzidx, roundzero); - else -#endif - lp->bfp_btran_normal(lp, rhsvector, nzidx); -} - -STATIC MYBOOL fsolve(lprec *lp, int varin, REAL *pcol, int *nzidx, REAL roundzero, REAL ofscalar, MYBOOL prepareupdate) -/* Was setpivcol in versions earlier than 4.0.1.8 - KE */ -{ - MYBOOL ok = TRUE; - - if(varin > 0) - obtain_column(lp, varin, pcol, nzidx, NULL); - - /* Solve, adjusted for objective function scalar */ - pcol[0] *= ofscalar; - if(prepareupdate) - lp->bfp_ftran_prepare(lp, pcol, nzidx); - else - ftran(lp, pcol, nzidx, roundzero); - - return(ok); - -} /* fsolve */ - - -STATIC MYBOOL bsolve(lprec *lp, int row_nr, REAL *rhsvector, int *nzidx, REAL roundzero, REAL ofscalar) -{ - MYBOOL ok = TRUE; - - if(row_nr >= 0) /* Note that row_nr == 0 returns the [1, 0...0 ] vector */ - row_nr = obtain_column(lp, row_nr, rhsvector, nzidx, NULL); - - /* Solve, adjusted for objective function scalar */ - rhsvector[0] *= ofscalar; - btran(lp, rhsvector, nzidx, roundzero); - - return(ok); - -} /* bsolve */ - - -/* Vector compression and expansion routines */ -STATIC MYBOOL vec_compress(REAL *densevector, int startpos, int endpos, REAL epsilon, - REAL *nzvector, int *nzindex) -{ - int n; - - if((densevector == NULL) || (nzindex == NULL) || (startpos > endpos)) - return( FALSE ); - - n = 0; - densevector += startpos; - while(startpos <= endpos) { - if(fabs(*densevector) > epsilon) { /* Apply zero-threshold */ - if(nzvector != NULL) /* Only produce index if no nzvector is given */ - nzvector[n] = *densevector; - n++; - nzindex[n] = startpos; - } - startpos++; - densevector++; - } - nzindex[0] = n; - return( TRUE ); -} - -STATIC MYBOOL vec_expand(REAL *nzvector, int *nzindex, REAL *densevector, int startpos, int endpos) -{ - int i, n; - - n = nzindex[0]; - i = nzindex[n]; - densevector += endpos; - while(endpos >= startpos) { /* Loop from behind to allow densevector == nzvector */ - if(endpos == i) { - n--; - *densevector = nzvector[n]; - i = nzindex[n]; - } - else - *densevector = 0; - endpos--; - densevector--; - } - return( TRUE ); -} - - -/* ----------------------------------------------------------------------- */ -/* Sparse matrix product routines and utility */ -/* ----------------------------------------------------------------------- */ - -STATIC MYBOOL get_colIndexA(lprec *lp, int varset, int *colindex, MYBOOL append) -{ - int i, varnr, P1extraDim, vb, ve, n, nrows = lp->rows, nsum = lp->sum; - MYBOOL omitfixed, omitnonfixed; - REAL v; - - /* Find what variable range to scan - default is {SCAN_USERVARS} */ - /* First determine the starting position; add from the top, going down */ - P1extraDim = abs(lp->P1extraDim); - vb = nrows + 1; - if(varset & SCAN_ARTIFICIALVARS) - vb = nsum - P1extraDim + 1; - if(varset & SCAN_USERVARS) - vb = nrows + 1; - if(varset & SCAN_SLACKVARS) - vb = 1; - - /* Then determine the ending position, add from the bottom, going up */ - ve = nsum; - if(varset & SCAN_SLACKVARS) - ve = nrows; - if(varset & SCAN_USERVARS) - ve = nsum - P1extraDim; - if(varset & SCAN_ARTIFICIALVARS) - ve = nsum; - - /* Adjust for partial pricing */ - if(varset & SCAN_PARTIALBLOCK) { - SETMAX(vb, partial_blockStart(lp, FALSE)); - SETMIN(ve, partial_blockEnd(lp, FALSE)); - } - - /* Determine exclusion columns */ - omitfixed = (MYBOOL) ((varset & OMIT_FIXED) != 0); - omitnonfixed = (MYBOOL) ((varset & OMIT_NONFIXED) != 0); - if(omitfixed && omitnonfixed) - return(FALSE); - - /* Scan the target colums */ - if(append) - n = colindex[0]; - else - n = 0; - for(varnr = vb; varnr <= ve; varnr++) { - - /* Skip gap in the specified column scan range (possibly user variables) */ - if(varnr > nrows) { - if((varnr <= nsum-P1extraDim) && !(varset & SCAN_USERVARS)) - continue; -#if 1 - /* Skip empty columns */ - if(/*(lp->P1extraVal == 0) &&*/ - (mat_collength(lp->matA, varnr-nrows) == 0)) - continue; -#endif - } - - /* Find if the variable is in the scope - default is {Ø} */ - i = lp->is_basic[varnr]; - if((varset & USE_BASICVARS) > 0 && (i)) - ; - else if((varset & USE_NONBASICVARS) > 0 && (!i)) - ; - else - continue; - - v = lp->upbo[varnr]; - if((omitfixed && (v == 0)) || - (omitnonfixed && (v != 0))) - continue; - - /* Append to list */ - n++; - colindex[n] = varnr; - } - colindex[0] = n; - - return(TRUE); -} - -STATIC int prod_Ax(lprec *lp, int *coltarget, REAL *input, int *nzinput, - REAL roundzero, REAL ofscalar, - REAL *output, int *nzoutput, int roundmode) -/* prod_Ax is only used in fimprove; note that it is NOT VALIDATED/verified as of 20030801 - KE */ -{ - int j, colnr, ib, ie, vb, ve; - MYBOOL localset, localnz = FALSE, isRC; - MATrec *mat = lp->matA; - REAL sdp; - REAL *value; - int *rownr; - - /* Find what variable range to scan - default is {SCAN_USERVARS} */ - /* Define default column target if none was provided */ - isRC = (MYBOOL) ((roundmode & MAT_ROUNDRC) != 0); - localset = (MYBOOL) (coltarget == NULL); - if(localset) { - int varset = SCAN_SLACKVARS | SCAN_USERVARS | - USE_BASICVARS | OMIT_FIXED; - if(isRC && is_piv_mode(lp, PRICE_PARTIAL) && !is_piv_mode(lp, PRICE_FORCEFULL)) - varset |= SCAN_PARTIALBLOCK; - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->sum+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, varset, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return(FALSE); - } - } - localnz = (MYBOOL) (nzinput == NULL); - if(localnz) { - nzinput = (int *) mempool_obtainVector(lp->workarrays, lp->rows+1, sizeof(*nzinput)); - vec_compress(input, 0, lp->rows, lp->matA->epsvalue, NULL, nzinput); - } - - /* Scan the columns */ - vb = 1; - ve = coltarget[0]; - for(vb = 1; vb <= coltarget[0]; vb++) { - colnr = coltarget[vb]; - j = lp->is_basic[colnr]; - - /* Perform the multiplication */ - sdp = ofscalar*input[j]; - if(colnr <= lp->rows) /* A slack variable is in the basis */ - output[colnr] += sdp; - else { /* A normal variable is in the basis */ - colnr -= lp->rows; - ib = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - rownr = &COL_MAT_ROWNR(ib); - value = &COL_MAT_VALUE(ib); - for(; ib < ie; - ib++, rownr += matRowColStep, value += matValueStep) { - output[*rownr] += (*value)*sdp; - } - } - } - roundVector(output+1, lp->rows-1, roundzero); - - /* Clean up and return */ - if(localset) - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - if(localnz) - mempool_releaseVector(lp->workarrays, (char *) nzinput, FALSE); - - return(TRUE); -} - -STATIC int prod_xA(lprec *lp, int *coltarget, - REAL *input, int *nzinput, REAL roundzero, REAL ofscalar, - REAL *output, int *nzoutput, int roundmode) -/* Note that the dot product xa is stored at the active column index of A, i.e. of a. - This means that if the basis only contains non-slack variables, output may point to - the same vector as input, without overwriting the [0..rows] elements. */ -{ - int colnr, rownr, varnr, ib, ie, vb, ve, nrows = lp->rows; - MYBOOL localset, localnz = FALSE, includeOF, isRC; - REALXP vmax; - register REALXP v; - int inz, *rowin, countNZ = 0; - MATrec *mat = lp->matA; - register REAL *matValue; - register int *matRownr; - - /* Clean output area (only necessary if we are returning the full vector) */ - isRC = (MYBOOL) ((roundmode & MAT_ROUNDRC) != 0); - if(nzoutput == NULL) { - if(input == output) - MEMCLEAR(output+nrows+1, lp->columns); - else - MEMCLEAR(output, lp->sum+1); - } - - /* Find what variable range to scan - default is {SCAN_USERVARS} */ - /* Define default column target if none was provided */ - localset = (MYBOOL) (coltarget == NULL); - if(localset) { - int varset = SCAN_SLACKVARS | SCAN_USERVARS | - USE_NONBASICVARS | OMIT_FIXED; - if(isRC && is_piv_mode(lp, PRICE_PARTIAL) && !is_piv_mode(lp, PRICE_FORCEFULL)) - varset |= SCAN_PARTIALBLOCK; - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->sum+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, varset, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return(FALSE); - } - } -/*#define UseLocalNZ*/ -#ifdef UseLocalNZ - localnz = (MYBOOL) (nzinput == NULL); - if(localnz) { - nzinput = (int *) mempool_obtainVector(lp->workarrays, nrows+1, sizeof(*nzinput)); - vec_compress(input, 0, nrows, lp->matA->epsvalue, NULL, nzinput); - } -#endif - includeOF = (MYBOOL) (((nzinput == NULL) || (nzinput[1] == 0)) && - (input[0] != 0) && lp->obj_in_basis); - - /* Scan the target colums */ - vmax = 0; - ve = coltarget[0]; - for(vb = 1; vb <= ve; vb++) { - - varnr = coltarget[vb]; - - if(varnr <= nrows) { - v = input[varnr]; - } - else { - colnr = varnr - nrows; - v = 0; - ib = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - if(ib < ie) { - - /* Do dense input vector version */ -#ifdef UseLocalNZ - if(localnz || (nzinput == NULL)) { -#else - if(nzinput == NULL) { -#endif - /* Do the OF */ - if(includeOF) -#ifdef DirectArrayOF - v += input[0] * lp->obj[colnr] * ofscalar; -#else - v += input[0] * get_OF_active(lp, varnr, ofscalar); -#endif - - /* Initialize pointers */ - matRownr = &COL_MAT_ROWNR(ib); - matValue = &COL_MAT_VALUE(ib); - - /* Do extra loop optimization based on target window overlaps */ -#ifdef UseLocalNZ - if((ib < ie) - && (colnr <= *nzinput) - && (COL_MAT_ROWNR(ie-1) >= nzinput[colnr]) - && (*matRownr <= nzinput[*nzinput]) - ) -#endif -#ifdef NoLoopUnroll - /* Then loop over all regular rows */ - for(; ib < ie; ib++) { - v += input[*matRownr] * (*matValue); - matValue += matValueStep; - matRownr += matRowColStep; - } -#else - /* Prepare for simple loop unrolling */ - if(((ie-ib) % 2) == 1) { - v += input[*matRownr] * (*matValue); - ib++; - matValue += matValueStep; - matRownr += matRowColStep; - } - - /* Then loop over remaining pairs of regular rows */ - while(ib < ie) { - v += input[*matRownr] * (*matValue); - v += input[*(matRownr+matRowColStep)] * (*(matValue+matValueStep)); - ib += 2; - matValue += 2*matValueStep; - matRownr += 2*matRowColStep; - } -#endif - } - /* Do sparse input vector version */ - else { - - /* Do the OF */ - if(includeOF) -#ifdef DirectArrayOF - v += input[0] * lp->obj[colnr] * ofscalar; -#else - v += input[0] * get_OF_active(lp, varnr, ofscalar); -#endif - - /* Initialize pointers */ - inz = 1; - rowin = nzinput+inz; - matRownr = &COL_MAT_ROWNR(ib); - matValue = &COL_MAT_VALUE(ib); - ie--; - - /* Then loop over all non-OF rows */ - while((inz <= *nzinput) && (ib <= ie)) { - - /* Try to synchronize at right */ - while((*rowin > *matRownr) && (ib < ie)) { - ib++; - matValue += matValueStep; - matRownr += matRowColStep; - } - /* Try to synchronize at left */ - while((*rowin < *matRownr) && (inz < *nzinput)) { - inz++; - rowin++; - } - /* Perform dot product operation if there was a match */ - if(*rowin == *matRownr) { - v += input[*rowin] * (*matValue); - /* Step forward at left */ - inz++; - rowin++; - } - } - } - } - if((roundmode & MAT_ROUNDABS) != 0) { - my_roundzero(v, roundzero); - } - } - - /* Special handling of small reduced cost values */ - if(!isRC || (my_chsign(lp->is_lower[varnr], v) < 0)) { - SETMAX(vmax, fabs((REAL) v)); - } - if(v != 0) { - countNZ++; - if(nzoutput != NULL) - nzoutput[countNZ] = varnr; - } - output[varnr] = (REAL) v; - } - - /* Compute reduced cost if this option is active */ - if(isRC && !lp->obj_in_basis) - countNZ = get_basisOF(lp, coltarget, output, nzoutput); - - /* Check if we should do relative rounding */ - if((roundmode & MAT_ROUNDREL) != 0) { - if((roundzero > 0) && (nzoutput != NULL)) { - ie = 0; - if(isRC) { - SETMAX(vmax, MAT_ROUNDRCMIN); /* Make sure we don't use very small values */ - } - vmax *= roundzero; - for(ib = 1; ib <= countNZ; ib++) { - rownr = nzoutput[ib]; - if(fabs(output[rownr]) < vmax) - output[rownr] = 0; - else { - ie++; - nzoutput[ie] = rownr; - } - } - countNZ = ie; - } - } - - /* Clean up and return */ - if(localset) - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - if(localnz) - mempool_releaseVector(lp->workarrays, (char *) nzinput, FALSE); - - if(nzoutput != NULL) - *nzoutput = countNZ; - return(countNZ); -} - -STATIC MYBOOL prod_xA2(lprec *lp, int *coltarget, - REAL *prow, REAL proundzero, int *nzprow, - REAL *drow, REAL droundzero, int *nzdrow, - REAL ofscalar, int roundmode) -{ - int varnr, colnr, ib, ie, vb, ve, nrows = lp->rows; - MYBOOL includeOF, isRC; - REALXP dmax, pmax; - register REALXP d, p; - MATrec *mat = lp->matA; - REAL value; - register REAL *matValue; - register int *matRownr; - MYBOOL localset; - - /* Find what variable range to scan - default is {SCAN_USERVARS} */ - /* First determine the starting position; add from the top, going down */ - localset = (MYBOOL) (coltarget == NULL); - if(localset) { - int varset = SCAN_SLACKVARS + SCAN_USERVARS + /*SCAN_ALLVARS +*/ - /*SCAN_PARTIALBLOCK+*/ - USE_NONBASICVARS+OMIT_FIXED; - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->sum+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, varset, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return(FALSE); - } - } - - /* Initialize variables */ - isRC = (MYBOOL) ((roundmode & MAT_ROUNDRC) != 0); - pmax = 0; - dmax = 0; - if(nzprow != NULL) - *nzprow = 0; - if(nzdrow != NULL) - *nzdrow = 0; - includeOF = (MYBOOL) (((prow[0] != 0) || (drow[0] != 0)) && - lp->obj_in_basis); - - /* Scan the target colums */ - ve = coltarget[0]; - for(vb = 1; vb <= ve; vb++) { - - varnr = coltarget[vb]; - - if(varnr <= nrows) { - p = prow[varnr]; - d = drow[varnr]; - } - else { - - colnr = varnr - nrows; - - p = 0; - d = 0; - ib = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - - if(ib < ie) { - - /* Do the OF */ - if(includeOF) { -#ifdef DirectArrayOF - value = lp->obj[colnr] * ofscalar; -#else - value = get_OF_active(lp, varnr, ofscalar); -#endif - p += prow[0] * value; - d += drow[0] * value; - } - - /* Then loop over all regular rows */ - matRownr = &COL_MAT_ROWNR(ib); - matValue = &COL_MAT_VALUE(ib); -#ifdef NoLoopUnroll - for( ; ib < ie; ib++) { - p += prow[*matRownr] * (*matValue); - d += drow[*matRownr] * (*matValue); - matValue += matValueStep; - matRownr += matRowColStep; - } -#else - /* Prepare for simple loop unrolling */ - if(((ie-ib) % 2) == 1) { - p += prow[*matRownr] * (*matValue); - d += drow[*matRownr] * (*matValue); - ib++; - matValue += matValueStep; - matRownr += matRowColStep; - } - - /* Then loop over remaining pairs of regular rows */ - while(ib < ie) { - p += prow[*matRownr] * (*matValue); - p += prow[*(matRownr+matRowColStep)] * (*(matValue+matValueStep)); - d += drow[*matRownr] * (*matValue); - d += drow[*(matRownr+matRowColStep)] * (*(matValue+matValueStep)); - ib += 2; - matValue += 2*matValueStep; - matRownr += 2*matRowColStep; - } -#endif - - } - if((roundmode & MAT_ROUNDABS) != 0) { - my_roundzero(p, proundzero); - my_roundzero(d, droundzero); - } - } - - SETMAX(pmax, fabs((REAL) p)); - prow[varnr] = (REAL) p; - if((nzprow != NULL) && (p != 0)) { - (*nzprow)++; - nzprow[*nzprow] = varnr; - } - - /* Special handling of reduced cost rounding */ - if(!isRC || (my_chsign(lp->is_lower[varnr], d) < 0)) { - SETMAX(dmax, fabs((REAL) d)); - } - drow[varnr] = (REAL) d; - if((nzdrow != NULL) && (d != 0)) { - (*nzdrow)++; - nzdrow[*nzdrow] = varnr; - } - } - - /* Compute reduced cost here if this option is active */ - if((drow != 0) && !lp->obj_in_basis) - get_basisOF(lp, coltarget, drow, nzdrow); - - /* Check if we should do relative rounding */ - if((roundmode & MAT_ROUNDREL) != 0) { - if((proundzero > 0) && (nzprow != NULL)) { - ie = 0; - pmax *= proundzero; - for(ib = 1; ib <= *nzprow; ib++) { - varnr = nzprow[ib]; - if(fabs(prow[varnr]) < pmax) - prow[varnr] = 0; - else { - ie++; - nzprow[ie] = varnr; - } - } - *nzprow = ie; - } - if((droundzero > 0) && (nzdrow != NULL)) { - ie = 0; - if(isRC) { - SETMAX(dmax, MAT_ROUNDRCMIN); /* Make sure we don't use very small values */ - } - dmax *= droundzero; - for(ib = 1; ib <= *nzdrow; ib++) { - varnr = nzdrow[ib]; - if(fabs(drow[varnr]) < dmax) - drow[varnr] = 0; - else { - ie++; - nzdrow[ie] = varnr; - } - } - *nzdrow = ie; - } - } - - /* Clean up and return */ - if(localset) - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return( TRUE ); -} - -STATIC void bsolve_xA2(lprec *lp, int* coltarget, - int row_nr1, REAL *vector1, REAL roundzero1, int *nzvector1, - int row_nr2, REAL *vector2, REAL roundzero2, int *nzvector2, int roundmode) -{ - REAL ofscalar = 1.0; - - /* Clear and initialize first vector */ - if(nzvector1 == NULL) - MEMCLEAR(vector1, lp->sum + 1); - else - MEMCLEAR(vector1, lp->rows + 1); - vector1[row_nr1] = 1; -/* workINT[0] = 1; - workINT[1] = row_nr1; */ - - if(vector2 == NULL) { - lp->bfp_btran_normal(lp, vector1, NULL); - prod_xA(lp, coltarget, vector1, NULL, roundzero1, ofscalar*0, - vector1, nzvector1, roundmode); - } - else { - - /* Clear and initialize second vector */ - if(nzvector2 == NULL) - MEMCLEAR(vector2, lp->sum + 1); - else - MEMCLEAR(vector2, lp->rows + 1); - if(lp->obj_in_basis || (row_nr2 > 0)) { - vector2[row_nr2] = 1; -/* workINT[2] = 1; - workINT[3] = row_nr2; */ - } - else - get_basisOF(lp, NULL, vector2, nzvector2); - - /* A double BTRAN equation solver process is implemented "in-line" below in - order to save time and to implement different rounding for the two */ - lp->bfp_btran_double(lp, vector1, NULL, vector2, NULL); - - /* Multiply solution vectors with matrix values */ - prod_xA2(lp, coltarget, vector1, roundzero1, nzvector1, - vector2, roundzero2, nzvector2, - ofscalar, roundmode); - } -} - diff --git a/code/3rd_lpsolve/lp_matrix.h b/code/3rd_lpsolve/lp_matrix.h deleted file mode 100644 index 3a1044f9..00000000 --- a/code/3rd_lpsolve/lp_matrix.h +++ /dev/null @@ -1,257 +0,0 @@ -#ifndef HEADER_lp_matrix -#define HEADER_lp_matrix - -#include "lp_types.h" -#include "lp_utils.h" - - -/* Sparse matrix element (ordered columnwise) */ -typedef struct _MATitem -{ - int rownr; - int colnr; - REAL value; -} MATitem; - -/* Constants for matrix product rounding options */ -#define MAT_ROUNDNONE 0 -#define MAT_ROUNDABS 1 -#define MAT_ROUNDREL 2 -#define MAT_ROUNDABSREL (MAT_ROUNDABS + MAT_ROUNDREL) -#define MAT_ROUNDRC 4 -#define MAT_ROUNDRCMIN 1.0 /* lp->epspivot */ -#if 1 - #define MAT_ROUNDDEFAULT MAT_ROUNDREL /* Typically increases performance */ -#else - #define MAT_ROUNDDEFAULT MAT_ROUNDABS /* Probably gives more precision */ -#endif - -/* Compiler option development features */ -/*#define DebugInv*/ /* Report array values at factorization/inversion */ -#define NoLoopUnroll /* Do not do loop unrolling */ -#define DirectArrayOF /* Reference lp->obj[] array instead of function call */ - - -/* Matrix column access macros to be able to easily change storage model */ -#define CAM_Record 0 -#define CAM_Vector 1 -#if 0 - #define MatrixColAccess CAM_Record -#else - #define MatrixColAccess CAM_Vector -#endif - -#if MatrixColAccess==CAM_Record -#define SET_MAT_ijA(item,i,j,A) mat->col_mat[item].rownr = i; \ - mat->col_mat[item].colnr = j; \ - mat->col_mat[item].value = A -#define COL_MAT_COLNR(item) (mat->col_mat[item].colnr) -#define COL_MAT_ROWNR(item) (mat->col_mat[item].rownr) -#define COL_MAT_VALUE(item) (mat->col_mat[item].value) -#define COL_MAT_COPY(left,right) mat->col_mat[left] = mat->col_mat[right] -#define COL_MAT_MOVE(to,from,rec) MEMMOVE(&(mat->col_mat[to]),&(mat->col_mat[from]),rec) -#define COL_MAT2_COLNR(item) (mat2->col_mat[item].colnr) -#define COL_MAT2_ROWNR(item) (mat2->col_mat[item].rownr) -#define COL_MAT2_VALUE(item) (mat2->col_mat[item].value) -#define matRowColStep (sizeof(MATitem)/sizeof(int)) -#define matValueStep (sizeof(MATitem)/sizeof(REAL)) - -#else /* if MatrixColAccess==CAM_Vector */ -#define SET_MAT_ijA(item,i,j,A) mat->col_mat_rownr[item] = i; \ - mat->col_mat_colnr[item] = j; \ - mat->col_mat_value[item] = A -#define COL_MAT_COLNR(item) (mat->col_mat_colnr[item]) -#define COL_MAT_ROWNR(item) (mat->col_mat_rownr[item]) -#define COL_MAT_VALUE(item) (mat->col_mat_value[item]) -#define COL_MAT_COPY(left,right) COL_MAT_COLNR(left) = COL_MAT_COLNR(right); \ - COL_MAT_ROWNR(left) = COL_MAT_ROWNR(right); \ - COL_MAT_VALUE(left) = COL_MAT_VALUE(right) -#define COL_MAT_MOVE(to,from,rec) MEMMOVE(&COL_MAT_COLNR(to),&COL_MAT_COLNR(from),rec); \ - MEMMOVE(&COL_MAT_ROWNR(to),&COL_MAT_ROWNR(from),rec); \ - MEMMOVE(&COL_MAT_VALUE(to),&COL_MAT_VALUE(from),rec) -#define COL_MAT2_COLNR(item) (mat2->col_mat_colnr[item]) -#define COL_MAT2_ROWNR(item) (mat2->col_mat_rownr[item]) -#define COL_MAT2_VALUE(item) (mat2->col_mat_value[item]) -#define matRowColStep 1 -#define matValueStep 1 - -#endif - - -/* Matrix row access macros to be able to easily change storage model */ -#define RAM_Index 0 -#define RAM_FullCopy 1 -#define MatrixRowAccess RAM_Index - -#if MatrixRowAccess==RAM_Index -#define ROW_MAT_COLNR(item) COL_MAT_COLNR(mat->row_mat[item]) -#define ROW_MAT_ROWNR(item) COL_MAT_ROWNR(mat->row_mat[item]) -#define ROW_MAT_VALUE(item) COL_MAT_VALUE(mat->row_mat[item]) - -#elif MatrixColAccess==CAM_Record -#define ROW_MAT_COLNR(item) (mat->row_mat[item].colnr) -#define ROW_MAT_ROWNR(item) (mat->row_mat[item].rownr) -#define ROW_MAT_VALUE(item) (mat->row_mat[item].value) - -#else /* if MatrixColAccess==CAM_Vector */ -#define ROW_MAT_COLNR(item) (mat->row_mat_colnr[item]) -#define ROW_MAT_ROWNR(item) (mat->row_mat_rownr[item]) -#define ROW_MAT_VALUE(item) (mat->row_mat_value[item]) - -#endif - - -typedef struct _MATrec -{ - /* Owner reference */ - lprec *lp; - - /* Active dimensions */ - int rows; - int columns; - - /* Allocated memory */ - int rows_alloc; - int columns_alloc; - int mat_alloc; /* The allocated size for matrix sized structures */ - - /* Sparse problem matrix storage */ -#if MatrixColAccess==CAM_Record - MATitem *col_mat; /* mat_alloc : The sparse data storage */ -#else /*MatrixColAccess==CAM_Vector*/ - int *col_mat_colnr; - int *col_mat_rownr; - REAL *col_mat_value; -#endif - int *col_end; /* columns_alloc+1 : col_end[i] is the index of the - first element after column i; column[i] is stored - in elements col_end[i-1] to col_end[i]-1 */ - int *col_tag; /* user-definable tag associated with each column */ - -#if MatrixRowAccess==RAM_Index - int *row_mat; /* mat_alloc : From index 0, row_mat contains the - row-ordered index of the elements of col_mat */ -#elif MatrixColAccess==CAM_Record - MATitem *row_mat; /* mat_alloc : From index 0, row_mat contains the - row-ordered copy of the elements in col_mat */ -#else /*if MatrixColAccess==CAM_Vector*/ - int *row_mat_colnr; - int *row_mat_rownr; - REAL *row_mat_value; -#endif - int *row_end; /* rows_alloc+1 : row_end[i] is the index of the - first element in row_mat after row i */ - int *row_tag; /* user-definable tag associated with each row */ - - REAL *colmax; /* Array of maximum values of each column */ - REAL *rowmax; /* Array of maximum values of each row */ - - REAL epsvalue; /* Zero element rejection threshold */ - REAL infnorm; /* The largest absolute value in the matrix */ - REAL dynrange; - MYBOOL row_end_valid; /* TRUE if row_end & row_mat are valid */ - MYBOOL is_roworder; /* TRUE if the current (temporary) matrix order is row-wise */ - -} MATrec; - -typedef struct _DeltaVrec -{ - lprec *lp; - int activelevel; - MATrec *tracker; -} DeltaVrec; - - -#ifdef __cplusplus -__EXTERN_C { -#endif - -/* Sparse matrix routines */ -STATIC MATrec *mat_create(lprec *lp, int rows, int columns, REAL epsvalue); -STATIC MYBOOL mat_memopt(MATrec *mat, int rowextra, int colextra, int nzextra); -STATIC void mat_free(MATrec **matrix); -STATIC MYBOOL inc_matrow_space(MATrec *mat, int deltarows); -STATIC int mat_mapreplace(MATrec *mat, LLrec *rowmap, LLrec *colmap, MATrec *insmat); -STATIC int mat_matinsert(MATrec *mat, MATrec *insmat); -STATIC int mat_zerocompact(MATrec *mat); -STATIC int mat_rowcompact(MATrec *mat, MYBOOL dozeros); -STATIC int mat_colcompact(MATrec *mat, int prev_rows, int prev_cols); -STATIC MYBOOL inc_matcol_space(MATrec *mat, int deltacols); -STATIC MYBOOL inc_mat_space(MATrec *mat, int mindelta); -STATIC int mat_shiftrows(MATrec *mat, int *bbase, int delta, LLrec *varmap); -STATIC int mat_shiftcols(MATrec *mat, int *bbase, int delta, LLrec *varmap); -STATIC MATrec *mat_extractmat(MATrec *mat, LLrec *rowmap, LLrec *colmap, MYBOOL negated); -STATIC int mat_appendrow(MATrec *mat, int count, REAL *row, int *colno, REAL mult, MYBOOL checkrowmode); -STATIC int mat_appendcol(MATrec *mat, int count, REAL *column, int *rowno, REAL mult, MYBOOL checkrowmode); -MYBOOL mat_get_data(lprec *lp, int matindex, MYBOOL isrow, int **rownr, int **colnr, REAL **value); -MYBOOL mat_set_rowmap(MATrec *mat, int row_mat_index, int rownr, int colnr, int col_mat_index); -STATIC MYBOOL mat_indexrange(MATrec *mat, int index, MYBOOL isrow, int *startpos, int *endpos); -STATIC MYBOOL mat_validate(MATrec *mat); -STATIC MYBOOL mat_equalRows(MATrec *mat, int baserow, int comprow); -STATIC int mat_findelm(MATrec *mat, int row, int column); -STATIC int mat_findins(MATrec *mat, int row, int column, int *insertpos, MYBOOL validate); -STATIC void mat_multcol(MATrec *mat, int col_nr, REAL mult, MYBOOL DoObj); -STATIC REAL mat_getitem(MATrec *mat, int row, int column); -STATIC MYBOOL mat_setitem(MATrec *mat, int row, int column, REAL value); -STATIC MYBOOL mat_additem(MATrec *mat, int row, int column, REAL delta); -STATIC MYBOOL mat_setvalue(MATrec *mat, int Row, int Column, REAL Value, MYBOOL doscale); -STATIC int mat_nonzeros(MATrec *mat); -STATIC int mat_collength(MATrec *mat, int colnr); -STATIC int mat_rowlength(MATrec *mat, int rownr); -STATIC void mat_multrow(MATrec *mat, int row_nr, REAL mult); -STATIC void mat_multadd(MATrec *mat, REAL *lhsvector, int varnr, REAL mult); -STATIC MYBOOL mat_setrow(MATrec *mat, int rowno, int count, REAL *row, int *colno, MYBOOL doscale, MYBOOL checkrowmode); -STATIC MYBOOL mat_setcol(MATrec *mat, int colno, int count, REAL *column, int *rowno, MYBOOL doscale, MYBOOL checkrowmode); -STATIC MYBOOL mat_mergemat(MATrec *target, MATrec *source, MYBOOL usecolmap); -STATIC int mat_checkcounts(MATrec *mat, int *rownum, int *colnum, MYBOOL freeonexit); -STATIC int mat_expandcolumn(MATrec *mat, int colnr, REAL *column, int *nzlist, MYBOOL signedA); -STATIC MYBOOL mat_computemax(MATrec *mat); -STATIC MYBOOL mat_transpose(MATrec *mat); - -/* Refactorization and recomputation routine */ -MYBOOL __WINAPI invert(lprec *lp, MYBOOL shiftbounds, MYBOOL final); - -/* Vector compression and expansion routines */ -STATIC MYBOOL vec_compress(REAL *densevector, int startpos, int endpos, REAL epsilon, REAL *nzvector, int *nzindex); -STATIC MYBOOL vec_expand(REAL *nzvector, int *nzindex, REAL *densevector, int startpos, int endpos); - -/* Sparse matrix products */ -STATIC MYBOOL get_colIndexA(lprec *lp, int varset, int *colindex, MYBOOL append); -STATIC int prod_Ax(lprec *lp, int *coltarget, REAL *input, int *nzinput, REAL roundzero, REAL ofscalar, REAL *output, int *nzoutput, int roundmode); -STATIC int prod_xA(lprec *lp, int *coltarget, REAL *input, int *nzinput, REAL roundzero, REAL ofscalar, REAL *output, int *nzoutput, int roundmode); -STATIC MYBOOL prod_xA2(lprec *lp, int *coltarget, REAL *prow, REAL proundzero, int *pnzprow, - REAL *drow, REAL droundzero, int *dnzdrow, REAL ofscalar, int roundmode); - -/* Equation solution */ -STATIC MYBOOL fimprove(lprec *lp, REAL *pcol, int *nzidx, REAL roundzero); -STATIC void ftran(lprec *lp, REAL *rhsvector, int *nzidx, REAL roundzero); -STATIC MYBOOL bimprove(lprec *lp, REAL *rhsvector, int *nzidx, REAL roundzero); -STATIC void btran(lprec *lp, REAL *rhsvector, int *nzidx, REAL roundzero); - -/* Combined equation solution and matrix product for simplex operations */ -STATIC MYBOOL fsolve(lprec *lp, int varin, REAL *pcol, int *nzidx, REAL roundzero, REAL ofscalar, MYBOOL prepareupdate); -STATIC MYBOOL bsolve(lprec *lp, int row_nr, REAL *rhsvector, int *nzidx, REAL roundzero, REAL ofscalar); -STATIC void bsolve_xA2(lprec *lp, int* coltarget, - int row_nr1, REAL *vector1, REAL roundzero1, int *nzvector1, - int row_nr2, REAL *vector2, REAL roundzero2, int *nzvector2, int roundmode); - -/* Change-tracking routines (primarily for B&B and presolve) */ -STATIC DeltaVrec *createUndoLadder(lprec *lp, int levelitems, int maxlevels); -STATIC int incrementUndoLadder(DeltaVrec *DV); -STATIC MYBOOL modifyUndoLadder(DeltaVrec *DV, int itemno, REAL target[], REAL newvalue); -STATIC int countsUndoLadder(DeltaVrec *DV); -STATIC int restoreUndoLadder(DeltaVrec *DV, REAL target[]); -STATIC int decrementUndoLadder(DeltaVrec *DV); -STATIC MYBOOL freeUndoLadder(DeltaVrec **DV); - -/* Specialized presolve undo functions */ -STATIC MYBOOL appendUndoPresolve(lprec *lp, MYBOOL isprimal, REAL beta, int colnrDep); -STATIC MYBOOL addUndoPresolve(lprec *lp, MYBOOL isprimal, int colnrElim, REAL alpha, REAL beta, int colnrDep); - - -#ifdef __cplusplus -} -#endif - -#endif /* HEADER_lp_matrix */ - diff --git a/code/3rd_lpsolve/lp_mipbb.c b/code/3rd_lpsolve/lp_mipbb.c deleted file mode 100644 index 54794734..00000000 --- a/code/3rd_lpsolve/lp_mipbb.c +++ /dev/null @@ -1,1441 +0,0 @@ - -/* - Mixed integer programming optimization drivers for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Michel Berkelaar (to lp_solve v3.2) - Kjell Eikland (v4.0 and forward) - Contact: - License terms: LGPL. - - Requires: string.h, float.h, commonlib.h, lp_lib.h, lp_report.h, - lp_simplex.h - - Release notes: - v5.0.0 31 January 2004 New unit isolating B&B routines. - v5.0.1 01 February 2004 Complete rewrite into non-recursive version. - v5.0.2 05 April 2004 Expanded pseudocosting with options for MIP fraction - counts and "cost/benefit" ratio (KE special!). - Added GUB functionality based on SOS structures. - v5.0.3 1 May 2004 Changed routine names to be more intuitive. - v5.0.4 15 May 2004 Added functinality to pack bounds in order to - conserve memory in B&B-processing large MIP models. - v5.1.0 25 July 2004 Added functions for dynamic cut generation. - v5.2.0 15 December 2004 Added functions for reduced cost variable fixing - and converted to delta-model of B&B bound storage. - ---------------------------------------------------------------------------------- -*/ - -#include -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_scale.h" -#include "lp_report.h" -#include "lp_simplex.h" -#include "lp_mipbb.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* Allocation routine for the BB record structure */ -STATIC BBrec *create_BB(lprec *lp, BBrec *parentBB, MYBOOL dofullcopy) -{ - BBrec *newBB; - - newBB = (BBrec *) calloc(1, sizeof(*newBB)); - if(newBB != NULL) { - - if(parentBB == NULL) { - allocREAL(lp, &newBB->upbo, lp->sum + 1, FALSE); - allocREAL(lp, &newBB->lowbo, lp->sum + 1, FALSE); - MEMCOPY(newBB->upbo, lp->orig_upbo, lp->sum + 1); - MEMCOPY(newBB->lowbo, lp->orig_lowbo, lp->sum + 1); - } - else if(dofullcopy) { - allocREAL(lp, &newBB->upbo, lp->sum + 1, FALSE); - allocREAL(lp, &newBB->lowbo, lp->sum + 1, FALSE); - MEMCOPY(newBB->upbo, parentBB->upbo, lp->sum + 1); - MEMCOPY(newBB->lowbo, parentBB->lowbo, lp->sum + 1); - } - else { - newBB->upbo = parentBB->upbo; - newBB->lowbo = parentBB->lowbo; - } - newBB->contentmode = dofullcopy; - - newBB->lp = lp; - - /* Set parent by default, but not child */ - newBB->parent = parentBB; - - } - return( newBB ); -} - - -/* Pushing and popping routines for the B&B structure */ - -STATIC BBrec *push_BB(lprec *lp, BBrec *parentBB, int varno, int vartype, int varcus) -/* Push ingoing bounds and B&B data onto the stack */ -{ - BBrec *newBB; - - /* Do initialization and updates */ - if(parentBB == NULL) - parentBB = lp->bb_bounds; - newBB = create_BB(lp, parentBB, FALSE); - if(newBB != NULL) { - - newBB->varno = varno; - newBB->vartype = vartype; - newBB->lastvarcus = varcus; - incrementUndoLadder(lp->bb_lowerchange); - newBB->LBtrack++; - incrementUndoLadder(lp->bb_upperchange); - newBB->UBtrack++; - - /* Adjust variable fixing/bound tightening based on the last reduced cost */ - if((parentBB != NULL) && (parentBB->lastrcf > 0)) { - MYBOOL isINT; - int k, ii, nfixed = 0, ntighten = 0; - REAL deltaUL; - - for(k = 1; k <= lp->nzdrow[0]; k++) { - ii = lp->nzdrow[k]; -#ifdef UseMilpSlacksRCF /* Check if we should include ranged constraints */ - isINT = FALSE; -#else - if(ii <= lp->rows) - continue; - isINT = is_int(lp, ii-lp->rows); -#endif -#ifndef UseMilpExpandedRCF /* Don't include non-integers if it is not defined */ - if(!isINT) - continue; -#endif - switch(abs(rcfbound_BB(newBB, ii, isINT, &deltaUL, NULL))) { - case LE: SETMIN(deltaUL, newBB->upbo[ii]); - SETMAX(deltaUL, newBB->lowbo[ii]); - modifyUndoLadder(lp->bb_upperchange, ii, newBB->upbo, deltaUL); - break; - case GE: SETMAX(deltaUL, newBB->lowbo[ii]); - SETMIN(deltaUL, newBB->upbo[ii]); - modifyUndoLadder(lp->bb_lowerchange, ii, newBB->lowbo, deltaUL); - break; - default: continue; - } - if(newBB->upbo[ii] == newBB->lowbo[ii]) - nfixed++; - else - ntighten++; - } - if(lp->bb_trace) { - report(lp, DETAILED, - "push_BB: Used reduced cost to fix %d variables and tighten %d bounds\n", - nfixed, ntighten); - } - } - - /* Handle case where we are pushing at the end */ - if(parentBB == lp->bb_bounds) - lp->bb_bounds = newBB; - /* Handle case where we are pushing in the middle */ - else - newBB->child = parentBB->child; - if(parentBB != NULL) - parentBB->child = newBB; - - lp->bb_level++; - if(lp->bb_level > lp->bb_maxlevel) - lp->bb_maxlevel = lp->bb_level; - - if(!initbranches_BB(newBB)) - newBB = pop_BB(newBB); - else if(MIP_count(lp) > 0) { - if( (lp->bb_level <= 1) && (lp->bb_varactive == NULL) && - (!allocINT(lp, &lp->bb_varactive, lp->columns+1, TRUE) || - !initcuts_BB(lp)) ) - newBB = pop_BB(newBB); - if(varno > 0) { - lp->bb_varactive[varno-lp->rows]++; - } - } - } - return( newBB ); -} - -STATIC MYBOOL free_BB(BBrec **BB) -{ - MYBOOL parentreturned = FALSE; - - if((BB != NULL) && (*BB != NULL)) { - BBrec *parent = (*BB)->parent; - - if((parent == NULL) || (*BB)->contentmode) { - FREE((*BB)->upbo); - FREE((*BB)->lowbo); - } - FREE((*BB)->varmanaged); - FREE(*BB); - - parentreturned = (MYBOOL) (parent != NULL); - if(parentreturned) - *BB = parent; - - } - return( parentreturned ); -} - -STATIC BBrec *pop_BB(BBrec *BB) -/* Pop / free the previously "pushed" / saved bounds */ -{ - int k; - BBrec *parentBB; - lprec *lp = BB->lp; - - if(BB == NULL) - return( BB ); - - /* Handle case where we are popping the end of the chain */ - parentBB = BB->parent; - if(BB == lp->bb_bounds) { - lp->bb_bounds = parentBB; - if(parentBB != NULL) - parentBB->child = NULL; - } - /* Handle case where we are popping inside or at the beginning of the chain */ - else { - if(parentBB != NULL) - parentBB->child = BB->child; - if(BB->child != NULL) - BB->child->parent = parentBB; - } - - /* Unwind other variables */ - if(lp->bb_upperchange != NULL) { - restoreUndoLadder(lp->bb_upperchange, BB->upbo); - for(; BB->UBtrack > 0; BB->UBtrack--) { - decrementUndoLadder(lp->bb_upperchange); - restoreUndoLadder(lp->bb_upperchange, BB->upbo); - } - } - if(lp->bb_lowerchange != NULL) { - restoreUndoLadder(lp->bb_lowerchange, BB->lowbo); - for(; BB->LBtrack > 0; BB->LBtrack--) { - decrementUndoLadder(lp->bb_lowerchange); - restoreUndoLadder(lp->bb_lowerchange, BB->lowbo); - } - } - lp->bb_level--; - k = BB->varno - lp->rows; - if(lp->bb_level == 0) { - if(lp->bb_varactive != NULL) { - FREE(lp->bb_varactive); - freecuts_BB(lp); - } - if(lp->int_vars+lp->sc_vars > 0) - free_pseudocost(lp); - pop_basis(lp, FALSE); - lp->rootbounds = NULL; - } - else - lp->bb_varactive[k]--; - - /* Undo SOS/GUB markers */ - if(BB->isSOS && (BB->vartype != BB_INT)) - SOS_unmark(lp->SOS, 0, k); - else if(BB->isGUB) - SOS_unmark(lp->GUB, 0, k); - - /* Undo the SC marker */ - if(BB->sc_canset) - lp->sc_lobound[k] *= -1; - - /* Pop the associated basis */ -#if 1 - /* Original version that does not restore previous basis */ - pop_basis(lp, FALSE); -#else - /* Experimental version that restores previous basis */ - pop_basis(lp, BB->isSOS); -#endif - - /* Finally free the B&B object */ - free_BB(&BB); - - /* Return the parent BB */ - return( parentBB ); -} - -/* Here are heuristic routines to see if we need bother with branching further - - 1. A probing routine to see of the best OF can be better than incumbent - 2. A presolve routine to fix other variables and detect infeasibility - - THIS IS INACTIVE CODE, PLACEHOLDERS FOR FUTURE DEVELOPMENT!!! */ -STATIC REAL probe_BB(BBrec *BB) -{ - int i, ii; - REAL coefOF, sum = 0; - lprec *lp = BB->lp; - - /* Loop over all ints to see if the best possible solution - stands any chance of being better than the incumbent solution */ - if(lp->solutioncount == 0) - return( lp->infinite ); - for(i = 1; i <= lp->columns; i++) { - if(!is_int(lp, i)) - continue; - ii = lp->rows + i; - coefOF = lp->obj[i]; - if(coefOF < 0) { - if(is_infinite(lp, BB->lowbo[ii])) - return( lp->infinite ); - sum += coefOF * (lp->solution[ii]-BB->lowbo[ii]); - } - else { - if(is_infinite(lp, BB->upbo[ii])) - return( lp->infinite ); - sum += coefOF * (BB->upbo[ii] - lp->solution[ii]); - } - } - return( sum ); -} - -STATIC REAL presolve_BB(BBrec *BB) -{ - return( 0 ); -} - -/* Node and branch management routines */ -STATIC MYBOOL initbranches_BB(BBrec *BB) -{ - REAL new_bound, temp; - int k; - lprec *lp = BB->lp; - - /* Create and initialize local bounds and basis */ - BB->nodestatus = NOTRUN; - BB->noderesult = lp->infinite; - push_basis(lp, NULL, NULL, NULL); - - /* Set default number of branches at the current B&B branch */ - if(BB->vartype == BB_REAL) - BB->nodesleft = 1; - - else { - /* The default is a binary up-low branching */ - BB->nodesleft = 2; - - /* Initialize the MIP status code pair and set reference values */ - k = BB->varno - lp->rows; - BB->lastsolution = lp->solution[BB->varno]; - - /* Determine if we must process in the B&B SOS mode */ - BB->isSOS = (MYBOOL) ((BB->vartype == BB_SOS) || SOS_is_member(lp->SOS, 0, k)); -#ifdef Paranoia - if((BB->vartype == BB_SOS) && !SOS_is_member(lp->SOS, 0, k)) - report(lp, SEVERE, "initbranches_BB: Inconsistent identification of SOS variable %s (%d)\n", - get_col_name(lp, k), k); -#endif - - /* Check if we have a GUB-member variable that needs a triple-branch */ - BB->isGUB = (MYBOOL) ((BB->vartype == BB_INT) && SOS_can_activate(lp->GUB, 0, k)); - if(BB->isGUB) { - /* Obtain variable index list from applicable GUB - now the first GUB is used, - but we could also consider selecting the longest */ - BB->varmanaged = SOS_get_candidates(lp->GUB, -1, k, TRUE, BB->upbo, BB->lowbo); - BB->nodesleft++; - } - - - /* Set local pruning info, automatic, or user-defined strategy */ - if(BB->vartype == BB_SOS) { - if(!SOS_can_activate(lp->SOS, 0, k)) { - BB->nodesleft--; - BB->isfloor = TRUE; - } - else - BB->isfloor = (MYBOOL) (BB->lastsolution == 0); - } - - /* First check if the user wishes to select the branching direction */ - else if(lp->bb_usebranch != NULL) - BB->isfloor = (MYBOOL) lp->bb_usebranch(lp, lp->bb_branchhandle, k); - - /* Otherwise check if we should do automatic branching */ - else if(get_var_branch(lp, k) == BRANCH_AUTOMATIC) { - new_bound = modf(BB->lastsolution/get_pseudorange(lp->bb_PseudoCost, k, BB->vartype), &temp); - if(isnan(new_bound)) - new_bound = 0; - else if(new_bound < 0) - new_bound += 1.0; - BB->isfloor = (MYBOOL) (new_bound <= 0.5); - - /* Set direction by OF value; note that a zero-value in - the OF gives priority to floor_first = TRUE */ - if(is_bb_mode(lp, NODE_GREEDYMODE)) { - if(is_bb_mode(lp, NODE_PSEUDOCOSTMODE)) - BB->sc_bound = get_pseudonodecost(lp->bb_PseudoCost, k, BB->vartype, BB->lastsolution); - else - BB->sc_bound = mat_getitem(lp->matA, 0, k); - new_bound -= 0.5; - BB->sc_bound *= new_bound; - BB->isfloor = (MYBOOL) (BB->sc_bound > 0); - } - /* Set direction by pseudocost (normally used in tandem with NODE_PSEUDOxxxSELECT) */ - else if(is_bb_mode(lp, NODE_PSEUDOCOSTMODE)) { - BB->isfloor = (MYBOOL) (get_pseudobranchcost(lp->bb_PseudoCost, k, TRUE) > - get_pseudobranchcost(lp->bb_PseudoCost, k, FALSE)); - if(is_maxim(lp)) - BB->isfloor = !BB->isfloor; - } - - /* Check for reversal */ - if(is_bb_mode(lp, NODE_BRANCHREVERSEMODE)) - BB->isfloor = !BB->isfloor; - } - else - BB->isfloor = (MYBOOL) (get_var_branch(lp, k) == BRANCH_FLOOR); - - /* SC logic: If the current SC variable value is in the [0..NZLOBOUND> range, then - - UP: Set lower bound to NZLOBOUND, upper bound is the original - LO: Fix the variable by setting upper/lower bound to zero - - ... indicate that the variable is B&B-active by reversing sign of sc_lobound[]. */ - new_bound = fabs(lp->sc_lobound[k]); - BB->sc_bound = new_bound; - BB->sc_canset = (MYBOOL) (new_bound != 0); - - /* Must make sure that we handle fractional lower bounds properly; - also to ensure that we do a full binary tree search */ - new_bound = unscaled_value(lp, new_bound, BB->varno); - if(is_int(lp, k) && ((new_bound > 0) && - (BB->lastsolution > floor(new_bound)))) { - if(BB->lastsolution < ceil(new_bound)) - BB->lastsolution += 1; - modifyUndoLadder(lp->bb_lowerchange, BB->varno, BB->lowbo, - scaled_floor(lp, BB->varno, BB->lastsolution, 1)); - } - } - - /* Now initialize the brances and set to first */ - return( fillbranches_BB(BB) ); -} - -STATIC MYBOOL fillbranches_BB(BBrec *BB) -{ - int K, k; - REAL ult_upbo, ult_lowbo; - REAL new_bound, SC_bound, intmargin = BB->lp->epsprimal; - lprec *lp = BB->lp; - MYBOOL OKstatus = FALSE; - - if(lp->bb_break || userabort(lp, MSG_MILPSTRATEGY)) - return( OKstatus ); - - K = BB->varno; - if(K > 0) { - - /* Shortcut variables */ - k = BB->varno - lp->rows; - ult_upbo = lp->orig_upbo[K]; - ult_lowbo = lp->orig_lowbo[K]; - SC_bound = unscaled_value(lp, BB->sc_bound, K); - - /* First, establish the upper bound to be applied (when isfloor == TRUE) - --------------------------------------------------------------------- */ -/*SetUB:*/ - BB->UPbound = lp->infinite; - - /* Handle SC-variables for the [0-LoBound> range */ - if((SC_bound > 0) && (fabs(BB->lastsolution) < SC_bound-intmargin)) { - new_bound = 0; - } - /* Handle pure integers (non-SOS, non-SC) */ - else if(BB->vartype == BB_INT) { -#if 1 - if(((ult_lowbo >= 0) && - ((floor(BB->lastsolution) < /* Skip cases where the lower bound becomes violated */ - unscaled_value(lp, MAX(ult_lowbo, fabs(lp->sc_lobound[k])), K)-intmargin))) || - ((ult_upbo <= 0) && /* Was ((ult_lowbo < 0) && */ - ((floor(BB->lastsolution) > /* Skip cases where the upper bound becomes violated */ - unscaled_value(lp, MIN(ult_upbo, -fabs(lp->sc_lobound[k])), K)-intmargin)))) { -#else - if((floor(BB->lastsolution) < /* Skip cases where the lower bound becomes violated */ - unscaled_value(lp, MAX(ult_lowbo, fabs(lp->sc_lobound[k])), K)-intmargin)) { -#endif - BB->nodesleft--; - goto SetLB; - } - new_bound = scaled_floor(lp, K, BB->lastsolution, 1); - } - else if(BB->isSOS) { /* Handle all SOS variants */ - new_bound = ult_lowbo; - if(is_int(lp, k)) - new_bound = scaled_ceil(lp, K, unscaled_value(lp, new_bound, K), -1); - } - else /* Handle all other variable incarnations */ - new_bound = BB->sc_bound; - - /* Check if the new bound might conflict and possibly make adjustments */ - if(new_bound < BB->lowbo[K]) - new_bound = BB->lowbo[K] - my_avoidtiny(new_bound-BB->lowbo[K], intmargin); - if(new_bound < BB->lowbo[K]) { -#ifdef Paranoia - debug_print(lp, - "fillbranches_BB: New upper bound value %g conflicts with old lower bound %g\n", - new_bound, BB->lowbo[K]); -#endif - BB->nodesleft--; - goto SetLB; - } -#ifdef Paranoia - /* Do additional consistency checking */ - else if(!check_if_less(lp, new_bound, BB->upbo[K], K)) { - BB->nodesleft--; - goto SetLB; - } -#endif - /* Bound (at least near) feasible */ - else { - /* Makes a difference with models like QUEEN - (note consistent use of epsint for scaled integer variables) */ - if(fabs(new_bound - BB->lowbo[K]) < intmargin*SCALEDINTFIXRANGE) - new_bound = BB->lowbo[K]; - } - - BB->UPbound = new_bound; - - - /* Next, establish the lower bound to be applied (when isfloor == FALSE) - --------------------------------------------------------------------- */ -SetLB: - BB->LObound = -lp->infinite; - - /* Handle SC-variables for the [0-LoBound> range */ - if((SC_bound > 0) && (fabs(BB->lastsolution) < SC_bound)) { - if(is_int(lp, k)) - new_bound = scaled_ceil(lp, K, SC_bound, 1); - else - new_bound = BB->sc_bound; - } - /* Handle pure integers (non-SOS, non-SC, but Ok for GUB!) */ - else if((BB->vartype == BB_INT)) { - if(((ceil(BB->lastsolution) == BB->lastsolution)) || /* Skip branch 0 if the current solution is integer */ - (ceil(BB->lastsolution) > /* Skip cases where the upper bound becomes violated */ - unscaled_value(lp, ult_upbo, K)+intmargin) || - (BB->isSOS && (BB->lastsolution == 0))) { /* Don't branch 0 since this is handled in SOS logic */ - BB->nodesleft--; - goto Finish; - } - new_bound = scaled_ceil(lp, K, BB->lastsolution, 1); - } - else if(BB->isSOS) { /* Handle all SOS variants */ - if(SOS_is_member_of_type(lp->SOS, k, SOS3)) - new_bound = scaled_floor(lp, K, 1, 1); - else { - new_bound = ult_lowbo; - if(is_int(lp, k)) - new_bound = scaled_floor(lp, K, unscaled_value(lp, new_bound, K), 1); - /* If we have a high-order SOS (SOS3+) and this variable is "intermediate" - between members previously lower-bounded at a non-zero level, then we should - set this and similar neighbouring variables at non-zero lowbo-values (remember - that SOS3+ members are all either integers or semi-continuous). Flag this - situation and prune tree, since we cannot lower-bound. */ - if((lp->SOS->maxorder > 2) && (BB->lastsolution == 0) && - SOS_is_member_of_type(lp->SOS, k, SOSn)) { - BB->isSOS = AUTOMATIC; - } - } - } - else /* Handle all other variable incarnations */ - new_bound = BB->sc_bound; - - /* Check if the new bound might conflict and possibly make adjustments */ - if(new_bound > BB->upbo[K]) - new_bound = BB->upbo[K] + my_avoidtiny(new_bound-BB->upbo[K], intmargin); - if(new_bound > BB->upbo[K]) { -#ifdef Paranoia - debug_print(lp, - "fillbranches_BB: New lower bound value %g conflicts with old upper bound %g\n", - new_bound, BB->upbo[K]); -#endif - BB->nodesleft--; - goto Finish; - } -#ifdef Paranoia - /* Do additional consistency checking */ - else if(!check_if_less(lp, BB->lowbo[K], new_bound, K)) { - BB->nodesleft--; - goto Finish; - } -#endif - /* Bound (at least near-)feasible */ - else { - /* Makes a difference with models like QUEEN - (note consistent use of lp->epsprimal for scaled integer variables) */ - if(fabs(BB->upbo[K]-new_bound) < intmargin*SCALEDINTFIXRANGE) - new_bound = BB->upbo[K]; - } - - BB->LObound = new_bound; - - /* Prepare for the first branch by making sure we are pointing correctly */ -Finish: - if(BB->nodesleft > 0) { - - /* Make sure the change tracker levels are "clean" for the B&B */ - if(countsUndoLadder(lp->bb_upperchange) > 0) { - incrementUndoLadder(lp->bb_upperchange); - BB->UBtrack++; - } - if(countsUndoLadder(lp->bb_lowerchange) > 0) { - incrementUndoLadder(lp->bb_lowerchange); - BB->LBtrack++; - } - - /* Do adjustments */ - if((BB->vartype != BB_SOS) && (fabs(BB->LObound-BB->UPbound) < intmargin)) { - BB->nodesleft--; - if(fabs(BB->lowbo[K]-BB->LObound) < intmargin) - BB->isfloor = FALSE; - else if(fabs(BB->upbo[K]-BB->UPbound) < intmargin) - BB->isfloor = TRUE; - else { - BB->isfloor = TRUE; - report(BB->lp, IMPORTANT, "fillbranches_BB: Inconsistent equal-valued bounds for %s\n", - get_col_name(BB->lp, k)); - } - } - if((BB->nodesleft == 1) && - ((BB->isfloor && (BB->UPbound >= lp->infinite)) || - (!BB->isfloor && (BB->LObound <= -lp->infinite)))) - BB->isfloor = !BB->isfloor; - /* Header initialization */ - BB->isfloor = !BB->isfloor; - while(!OKstatus && /* !userabort(lp, -1) */ lp->spx_status != TIMEOUT && !lp->bb_break && (BB->nodesleft > 0)) - OKstatus = nextbranch_BB( BB ); - } - - /* Set an SC variable active, if necessary */ - if(BB->sc_canset) - lp->sc_lobound[k] *= -1; - - } - else { - BB->nodesleft--; - OKstatus = TRUE; - } - - return( OKstatus ); -} - -STATIC MYBOOL nextbranch_BB(BBrec *BB) -{ - int k; - lprec *lp = BB->lp; - MYBOOL OKstatus = FALSE; - - /* Undo the most recently imposed B&B bounds using the data - in the last level of change tracker; this code handles changes - to both upper and lower bounds */ - if(BB->nodessolved > 0) { - restoreUndoLadder(lp->bb_upperchange, BB->upbo); - restoreUndoLadder(lp->bb_lowerchange, BB->lowbo); - } - - if(lp->bb_break || userabort(lp, MSG_MILPSTRATEGY)) { - /* Handle the special case of B&B restart; - (typically used with the restart after pseudocost initialization) */ - if((lp->bb_level == 1) && (lp->bb_break == AUTOMATIC)) { - lp->bb_break = FALSE; - OKstatus = TRUE; - } - return( OKstatus ); - } - - if(BB->nodesleft > 0) { - - /* Step and update remaining branch count */ - k = BB->varno - lp->rows; - BB->isfloor = !BB->isfloor; - BB->nodesleft--; - - /* Special SOS handling: - 1) Undo and set new marker for k, - 2) In case that previous branch was ceiling restore upper bounds of the - non-k variables outside of the SOS window set to 0 */ - if(BB->isSOS && (BB->vartype != BB_INT)) { - - /* First undo previous marker */ - if((BB->nodessolved > 0) || ((BB->nodessolved == 0) && (BB->nodesleft == 0))) { - if(BB->isfloor) { - if((BB->nodesleft == 0) && (lp->orig_lowbo[BB->varno] != 0)) - return( OKstatus ); - } - SOS_unmark(lp->SOS, 0, k); - } - - /* Set new SOS marker */ - if(BB->isfloor) { - SOS_set_marked(lp->SOS, 0, k, (MYBOOL) (BB->UPbound != 0)); - /* Do case of high-order SOS where intervening variables need to be set */ - if(BB->isSOS == AUTOMATIC) { - -/* SOS_fix_list(lp->SOS, 0, k, BB->lowbo, NULL, AUTOMATIC, lp->bb_lowerchange); */ - } - } - else { - SOS_set_marked(lp->SOS, 0, k, TRUE); - if(SOS_fix_unmarked(lp->SOS, 0, k, BB->upbo, 0, TRUE, - NULL, lp->bb_upperchange) < 0) - return( OKstatus ); - } - } - - /* Special GUB handling (three branches): - 1) Undo and set new marker for k, - 2) Restore upper bounds of the left/right/all non-k variables - set to 0 in the previous branch - 3) Set new upper bounds for the non-k variables (k is set later) */ - else if(BB->isGUB) { - - /* First undo previous marker */ - if(BB->nodessolved > 0) - SOS_unmark(lp->GUB, 0, k); - - /* Make sure we take floor bound twice */ - if((BB->nodesleft == 0) && !BB->isfloor) - BB->isfloor = !BB->isfloor; - - /* Handle two floor instances; - (selected variable and left/right halves of non-selected variables at 0) */ - SOS_set_marked(lp->GUB, 0, k, (MYBOOL) !BB->isfloor); - if(BB->isfloor) { - if(SOS_fix_list(lp->GUB, 0, k, BB->upbo, - BB->varmanaged, (MYBOOL) (BB->nodesleft > 0), lp->bb_upperchange) < 0) - return( OKstatus ); - } - /* Handle one ceil instance; - (selected variable at 1, all other at 0) */ - else { - if(SOS_fix_unmarked(lp->GUB, 0, k, BB->upbo, 0, TRUE, - NULL, lp->bb_upperchange) < 0) - return( OKstatus ); - } - } - - OKstatus = TRUE; - - } - /* Initialize simplex status variables */ - if(OKstatus) { - lp->bb_totalnodes++; - BB->nodestatus = NOTRUN; - BB->noderesult = lp->infinite; - } - return( OKstatus ); -} - - -/* Cut generation and management routines */ -STATIC MYBOOL initcuts_BB(lprec *lp) -{ - return( TRUE ); -} - -STATIC int updatecuts_BB(lprec *lp) -{ - return( 0 ); -} - -STATIC MYBOOL freecuts_BB(lprec *lp) -{ - if(lp->bb_cuttype != NULL) - FREE(lp->bb_cuttype); - return( TRUE ); -} - -/* B&B solver routines */ -STATIC int solve_LP(lprec *lp, BBrec *BB) -{ - int tilted, restored, status; - REAL testOF, *upbo = BB->upbo, *lowbo = BB->lowbo; - BBrec *perturbed = NULL; - - if(lp->bb_break) - return(PROCBREAK); - -#ifdef Paranoia - debug_print(lp, "solve_LP: Starting solve for iter %.0f, B&B node level %d.\n", - (double) lp->total_iter, lp->bb_level); - if(lp->bb_trace && - !validate_bounds(lp, upbo, lowbo)) - report(lp, SEVERE, "solve_LP: Inconsistent bounds at iter %.0f, B&B node level %d.\n", - (double) lp->total_iter, lp->bb_level); -#endif - - /* Copy user-specified entering bounds into lp_solve working bounds */ - impose_bounds(lp, upbo, lowbo); - - /* Restore previously pushed / saved basis for this level if we are in - the B&B mode and it is not the first call of the binary tree */ - if(BB->nodessolved > 1) - restore_basis(lp); - - /* Solve and possibly handle degeneracy cases via bound relaxations */ - status = RUNNING; - tilted = 0; - restored = 0; - - while(status == RUNNING) { - - /* Copy user-specified entering bounds into lp_solve working bounds and run */ - status = spx_run(lp, (MYBOOL) (tilted+restored > 0)); - lp->bb_status = status; - lp->spx_perturbed = FALSE; - - if(tilted < 0) - break; - - else if((status == OPTIMAL) && (tilted > 0)) { - if(lp->spx_trace) - report(lp, DETAILED, "solve_LP: Restoring relaxed bounds at level %d.\n", - tilted); - - /* Restore original pre-perturbed problem bounds, and solve again using the basis - found for the perturbed problem; also make sure we rebase and recompute. */ - free_BB(&perturbed); - if((perturbed == NULL) || (perturbed == BB)) { - perturbed = NULL; - impose_bounds(lp, upbo, lowbo); - } - else - impose_bounds(lp, perturbed->upbo, perturbed->lowbo); - set_action(&lp->spx_action, ACTION_REBASE | ACTION_RECOMPUTE); - BB->UBzerobased = FALSE; - if(lp->bb_totalnodes == 0) - lp->real_solution = lp->infinite; - status = RUNNING; - tilted--; - restored++; - lp->spx_perturbed = TRUE; - } - - else if(((lp->bb_level <= 1) || is_anti_degen(lp, ANTIDEGEN_DURINGBB)) && - (((status == LOSTFEAS) && is_anti_degen(lp, ANTIDEGEN_LOSTFEAS)) || - ((status == INFEASIBLE) && is_anti_degen(lp, ANTIDEGEN_INFEASIBLE)) || - ((status == NUMFAILURE) && is_anti_degen(lp, ANTIDEGEN_NUMFAILURE)) || - ((status == DEGENERATE) && is_anti_degen(lp, ANTIDEGEN_STALLING)))) { - /* Allow up to .. consecutive relaxations for non-B&B phases */ - if((tilted <= DEF_MAXRELAX) && /* Conventional recovery case,... */ - !((tilted == 0) && (restored > DEF_MAXRELAX))) { /* but not iterating infeasibility */ - - /* Create working copy of ingoing bounds if this is the first perturbation */ - if(tilted == 0) - perturbed = BB; - perturbed = create_BB(lp, perturbed, TRUE); - - /* Perturb/shift variable bounds; also make sure we rebase and recompute - (no refactorization is necessary, since the basis is unchanged) */ -#if 1 - perturb_bounds(lp, perturbed, TRUE, TRUE, TRUE); -#else - perturb_bounds(lp, perturbed, TRUE, TRUE, FALSE); -#endif - impose_bounds(lp, perturbed->upbo, perturbed->lowbo); - set_action(&lp->spx_action, ACTION_REBASE | ACTION_RECOMPUTE); - BB->UBzerobased = FALSE; - status = RUNNING; - tilted++; - lp->perturb_count++; - lp->spx_perturbed = TRUE; - if(lp->spx_trace) - report(lp, DETAILED, "solve_LP: Starting bound relaxation #%d ('%s')\n", - tilted, get_statustext(lp, status)); - } - else { - if(lp->spx_trace) - report(lp, DETAILED, "solve_LP: Relaxation limit exceeded in resolving infeasibility\n"); - while((perturbed != NULL) && (perturbed != BB)) - free_BB(&perturbed); - perturbed = NULL; - } - } - } - - /* Handle the different simplex outcomes */ - if(status != OPTIMAL) { - if(lp->bb_level <= 1) - lp->bb_parentOF = lp->infinite; - if((status == USERABORT) || (status == TIMEOUT)) { - /* Construct the last feasible solution, if available */ - if((lp->solutioncount == 0) && - /* - 30/01/08 added MIP_count test because in following situation thing were wrong: - - The model contains integers - - A break at first is set - - A timeout is set - - The timeout occurs before a first integer solution is found - - When the timeout occurs, the simplex algorithm is in phase 2 and has a feasible (but non-integer) solution, but not optimal yet. - If above situation occurs then a (sub-optimal) solution was returned while no integer - solution isn't found yet at this time - */ - (MIP_count(lp) == 0) && - ((lp->simplex_mode & (SIMPLEX_Phase2_PRIMAL | SIMPLEX_Phase2_DUAL)) > 0)) { - lp->solutioncount++; - construct_solution(lp, NULL); - transfer_solution(lp, TRUE); - } - /* Return messages */ - report(lp, NORMAL, "\nlp_solve optimization was stopped %s.\n", - ((status == USERABORT) ? "by the user" : "due to time-out")); - } - else if(BB->varno == 0) - report(lp, NORMAL, "The model %s\n", - (status == UNBOUNDED) ? "is UNBOUNDED" : - ((status == INFEASIBLE) ? "is INFEASIBLE" : "FAILED")); - else { -#ifdef Paranoia - if((status != FATHOMED) && (status != INFEASIBLE)) - report(lp, SEVERE, "spx_solve: Invalid return code %d during B&B\n", status); -#endif - /* If we fathomed a node due to an inferior OF having been detected, return infeasible */ - if(status == FATHOMED) - lp->spx_status = INFEASIBLE; - } - } - - else { /* ... there is a good solution */ - construct_solution(lp, NULL); - if((lp->bb_level <= 1) && (restored > 0)) - report(lp, DETAILED, "%s numerics encountered; validate accuracy\n", - (restored == 1) ? "Difficult" : "Severe"); - /* Handle case where a user bound on the OF was found to - have been set too aggressively, giving an infeasible model */ - if(lp->spx_status != OPTIMAL) - status = lp->spx_status; - - else if((lp->bb_totalnodes == 0) && (MIP_count(lp) > 0)) { - if(lp->lag_status != RUNNING) { - report(lp, NORMAL, "\nRelaxed solution " RESULTVALUEMASK " after %10.0f iter is B&B base.\n", - lp->solution[0], (double) lp->total_iter); - report(lp, NORMAL, " \n"); - } - if((lp->usermessage != NULL) && (lp->msgmask & MSG_LPOPTIMAL)) { - REAL *best_solution = lp->best_solution; - - /* transfer_solution(lp, TRUE); */ - lp->best_solution = lp->solution; - lp->usermessage(lp, lp->msghandle, MSG_LPOPTIMAL); - lp->best_solution = best_solution; - } - set_var_priority(lp); - } - - /* Check if we have a numeric problem (an earlier version of this code used the - absolute difference, but it is not robust for large-valued OFs) */ - testOF = my_chsign(is_maxim(lp), my_reldiff(lp->solution[0], lp->real_solution)); - if(testOF < -lp->epsprimal) { - report(lp, DETAILED, "solve_LP: A MIP subproblem returned a value better than the base.\n"); - status = INFEASIBLE; - lp->spx_status = status; - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - } - else if(testOF < 0) /* Avoid problems later (could undo integer roundings, but usually Ok) */ - lp->solution[0] = lp->real_solution; - - } - - /* status can have the following values: - OPTIMAL, SUBOPTIMAL, TIMEOUT, USERABORT, PROCFAIL, UNBOUNDED and INFEASIBLE. */ - - return( status ); -} /* solve_LP */ - -STATIC BBrec *findself_BB(BBrec *BB) -{ - int varno = BB->varno, vartype = BB->vartype; - - BB = BB->parent; - while((BB != NULL) && (BB->vartype != vartype) && (BB->varno != varno)) - BB = BB->parent; - return( BB ); -} - -/* Function to determine the opportunity for variable fixing and bound - tightening based on a previous best MILP solution and a variable's - reduced cost at the current relaxation - inspired by Wolsley */ -STATIC int rcfbound_BB(BBrec *BB, int varno, MYBOOL isINT, REAL *newbound, MYBOOL *isfeasible) -{ - int i = FR; - lprec *lp = BB->lp; - REAL deltaRC, rangeLU, deltaOF, lowbo, upbo; - - /* Make sure we only accept non-basic variables */ - if(lp->is_basic[varno]) - return( i ); - - /* Make sure we only accept non-fixed variables */ - lowbo = BB->lowbo[varno]; - upbo = BB->upbo[varno]; - rangeLU = upbo - lowbo; - - if(rangeLU > lp->epsprimal) { -#if 1 /* v5.5 problematic - Gap between current node and the current best bound */ - deltaOF = lp->rhs[0] - lp->bb_workOF; -#elif 0 /* v6 less aggressive - Gap between current best bound and the relaxed problem */ - deltaOF = my_chsign(is_maxim(lp), lp->real_solution) - lp->bb_workOF; -#else /* v6 more aggressive - Gap between current node and the relaxed problem */ - deltaOF = my_chsign(is_maxim(lp), lp->real_solution) - lp->rhs[0]; -#endif - - deltaRC = my_chsign(!lp->is_lower[varno], lp->drow[varno]); - /* Protect against divisions with tiny numbers and stray sign - reversals of the reduced cost */ - if(deltaRC < lp->epspivot) - return( i ); - deltaRC = deltaOF / deltaRC; /* Should always be a positive number! */ -#ifdef Paranoia - if(deltaRC <= 0) - report(lp, SEVERE, "rcfbound_BB: A negative bound fixing level was encountered after node %.0f\n", - (double) lp->bb_totalnodes); -#endif - - /* Check if bound implied by the reduced cost is less than existing range */ - if(deltaRC < rangeLU + lp->epsint) { - if(lp->is_lower[varno]) { - if(isINT) - deltaRC = scaled_floor(lp, varno, unscaled_value(lp, deltaRC, varno)+lp->epsprimal, 1); - upbo = lowbo + deltaRC; - deltaRC = upbo; - i = LE; /* Sets the upper bound */ - } - else { - if(isINT) - deltaRC = scaled_ceil(lp, varno, unscaled_value(lp, deltaRC, varno)+lp->epsprimal, 1); - lowbo = upbo - deltaRC; - deltaRC = lowbo; - i = GE; /* Sets the lower bound */ - } - - /* Check and set feasibility status */ - if((isfeasible != NULL) && (upbo - lowbo < -lp->epsprimal)) - *isfeasible = FALSE; - - /* Flag that we can fix the variable by returning the relation code negated */ - else if(fabs(upbo - lowbo) < lp->epsprimal) - i = -i; - if(newbound != NULL) { - my_roundzero(deltaRC, lp->epsprimal); - *newbound = deltaRC; - } - } - - } - return( i ); -} - - -STATIC MYBOOL findnode_BB(BBrec *BB, int *varno, int *vartype, int *varcus) -{ - int countsossc, countnint, k, reasonmsg = MSG_NONE; - REAL varsol; - MYBOOL is_better = FALSE, is_equal = FALSE, is_feasible = TRUE; - lprec *lp = BB->lp; - - /* Initialize result and return variables */ - *varno = 0; - *vartype = BB_REAL; - *varcus = 0; - countnint = 0; - BB->nodestatus = lp->spx_status; - BB->noderesult = lp->solution[0]; - - /* If this solution is worse than the best so far, this branch dies. - If we can only have integer OF values, and we only need the first solution - then the OF must be at least (unscaled) 1 better than the best so far */ - if((lp->bb_limitlevel != 1) && (MIP_count(lp) > 0)) { - - /* Check that we don't have a limit on the recursion level; two versions supported: - 1) Absolute B&B level (bb_limitlevel > 0), and - 2) B&B level relative to the "B&B order" (bb_limitlevel < 0). */ - countsossc = lp->sos_vars + lp->sc_vars; - if((lp->bb_limitlevel > 0) && (lp->bb_level > lp->bb_limitlevel+countsossc)) - return( FALSE ); - else if((lp->bb_limitlevel < 0) && - (lp->bb_level > 2*(lp->int_vars+countsossc)*abs(lp->bb_limitlevel))) { - if(lp->bb_limitlevel == DEF_BB_LIMITLEVEL) - report(lp, IMPORTANT, "findnode_BB: Default B&B limit reached at %d; optionally change strategy or limit.\n\n", - lp->bb_level); - return( FALSE ); - } - - /* First initialize or update pseudo-costs from previous optimal solution */ - if(BB->varno == 0) { - varsol = lp->infinite; - if((lp->int_vars+lp->sc_vars > 0) && (lp->bb_PseudoCost == NULL)) - lp->bb_PseudoCost = init_pseudocost(lp, get_bb_rule(lp)); - } - else { - varsol = lp->solution[BB->varno]; - if( ((lp->int_vars > 0) && (BB->vartype == BB_INT)) || - ((lp->sc_vars > 0) && (BB->vartype == BB_SC) && !is_int(lp, BB->varno-lp->rows)) ) - update_pseudocost(lp->bb_PseudoCost, BB->varno-lp->rows, BB->vartype, BB->isfloor, varsol); - } - - /* Make sure we don't have numeric problems (typically due to integer scaling) */ - if((lp->bb_totalnodes > 0) && !bb_better(lp, OF_RELAXED, OF_TEST_WE)) { - if(lp->bb_trace) - report(lp, IMPORTANT, "findnode_BB: Simplex failure due to loss of numeric accuracy\n"); - lp->spx_status = NUMFAILURE; - return( FALSE ); - } - - /* Abandon this branch if the solution is "worse" than a heuristically - determined limit or the previous best MIP solution */ - if(((lp->solutioncount == 0) && !bb_better(lp, OF_HEURISTIC, OF_TEST_BE)) || - ((lp->solutioncount > 0) && - (!bb_better(lp, OF_INCUMBENT | OF_DELTA, OF_TEST_BE | OF_TEST_RELGAP) || - !bb_better(lp, OF_INCUMBENT | OF_DELTA, OF_TEST_BE)))) { - return( FALSE ); - } - - /* Collect violated SC variables (since they can also be real-valued); the - approach is to get them out of the way, since a 0-value is assumed to be "cheap" */ - if(lp->sc_vars > 0) { - *varno = find_sc_bbvar(lp, &countnint); - if(*varno > 0) - *vartype = BB_SC; - } - - /* Look among SOS variables if no SC candidate was found */ - if((SOS_count(lp) > 0) && (*varno == 0)) { - *varno = find_sos_bbvar(lp, &countnint, FALSE); - if(*varno < 0) - *varno = 0; - else if(*varno > 0) - *vartype = BB_SOS; - } - - /* Then collect INTS that are not integer valued, and verify bounds */ - if((lp->int_vars > 0) && (*varno == 0)) { - *varno = find_int_bbvar(lp, &countnint, BB, &is_feasible); - if(*varno > 0) { - *vartype = BB_INT; - if((countnint == 1) && !is_feasible) { - BB->lastrcf = 0; - return( FALSE ); - } - } - } - -#if 1 /* peno */ - /* Check if we have reached the depth limit for any individual variable - (protects against infinite recursions of mainly integer variables) */ - k = *varno-lp->rows; - if((*varno > 0) && (lp->bb_limitlevel != 0) && (lp->bb_varactive[k] >= abs(lp->bb_limitlevel) /* abs(DEF_BB_LIMITLEVEL) */)) { - /* if(!is_action(lp->nomessage, NOMSG_BBLIMIT)) {*/ -/* - report(lp, IMPORTANT, "findnode_BB: Reached B&B depth limit %d for variable %d; will not dive further.\n\n", - lp->bb_varactive[k], k); -*/ - /* set_action(&lp->nomessage, NOMSG_BBLIMIT); */ - /* } */ - return( FALSE ); - } -#endif - - /* Check if the current MIP solution is optimal; equal or better */ - if(*varno == 0) { - is_better = (MYBOOL) (lp->solutioncount == 0) || bb_better(lp, OF_INCUMBENT | OF_DELTA, OF_TEST_BT); -#if 1 - is_better &= bb_better(lp, OF_INCUMBENT | OF_DELTA, OF_TEST_BT | OF_TEST_RELGAP); -#else - /* Check if we can determine clear improvement */ - is_better = (MYBOOL) (lp->solutioncount == 0) || - (MYBOOL) ((lp->bb_deltaOF > 0) && - (my_chsign(is_maxim(lp), lp->solution[0]-lp->best_solution[0]) < 0)); - - /* Apply gap-based improvement testing if the current solution is not clearly better */ - - if(!is_better) { - is_better = bb_better(lp, OF_INCUMBENT | OF_DELTA, OF_TEST_BT); - is_better |= bb_better(lp, OF_INCUMBENT | OF_DELTA, OF_TEST_BT | OF_TEST_RELGAP); - } -#endif - is_equal = !is_better; - - if(is_equal) { - if((lp->solutionlimit <= 0) || (lp->solutioncount < lp->solutionlimit)) { - lp->solutioncount++; - SETMIN(lp->bb_solutionlevel, lp->bb_level); - reasonmsg = MSG_MILPEQUAL; - } - } - - /* Current solution is better */ - else if(is_better) { - - /* Update grand total solution count and check if we should go from - depth-first to best-first variable selection mode */ - if(lp->bb_varactive != NULL) { - lp->bb_varactive[0]++; - if((lp->bb_varactive[0] == 1) && - is_bb_mode(lp, NODE_DEPTHFIRSTMODE) && is_bb_mode(lp, NODE_DYNAMICMODE)) - lp->bb_rule &= !NODE_DEPTHFIRSTMODE; - } - - if(lp->bb_trace || - ((lp->verbose >= NORMAL) && (lp->print_sol == FALSE) && (lp->lag_status != RUNNING))) { - report(lp, IMPORTANT, - "%s solution " RESULTVALUEMASK " after %10.0f iter, %9.0f nodes (gap %.1f%%)\n", - (lp->bb_improvements == 0) ? "Feasible" : "Improved", - lp->solution[0], (double) lp->total_iter, (double) lp->bb_totalnodes, - 100.0*fabs(my_reldiff(lp->solution[0], lp->bb_limitOF))); - } - if(MIP_count(lp) > 0) { - if(lp->bb_improvements == 0) - reasonmsg = MSG_MILPFEASIBLE; - else - reasonmsg = MSG_MILPBETTER; - } - - lp->bb_status = FEASFOUND; - lp->bb_solutionlevel = lp->bb_level; - lp->solutioncount = 1; - lp->bb_improvements++; - lp->bb_workOF = lp->rhs[0]; - - if(lp->bb_breakfirst || - (!is_infinite(lp, lp->bb_breakOF) && bb_better(lp, OF_USERBREAK, OF_TEST_BE))) - lp->bb_break = TRUE; - } - } - } - else { - is_better = TRUE; - lp->solutioncount = 1; - } - - /* Transfer the successful solution vector */ - if(is_better || is_equal) { -#ifdef ParanoiaMIP - if((lp->bb_level > 0) && - (check_solution(lp, lp->columns, lp->solution, - lp->orig_upbo, lp->orig_lowbo, lp->epssolution) != OPTIMAL)) { - lp->solutioncount = 0; - lp->spx_status = NUMFAILURE; - lp->bb_status = lp->spx_status; - lp->bb_break = TRUE; - return( FALSE ); - } -#endif - transfer_solution(lp, (MYBOOL) ((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE)); - if((MIP_count(lp) > 0) && (lp->bb_totalnodes > 0)) { - if ((!construct_duals(lp)) || - (is_presolve(lp, PRESOLVE_SENSDUALS) && - (!construct_sensitivity_duals(lp) || !construct_sensitivity_obj(lp)) - ) - ) { - } - } - if((reasonmsg != MSG_NONE) && (lp->msgmask & reasonmsg) && (lp->usermessage != NULL)) - lp->usermessage(lp, lp->msghandle, reasonmsg); - - if(lp->print_sol != FALSE) { - print_objective(lp); - print_solution(lp, 1); - } - } - - /* Do tracing and determine if we have arrived at the estimated lower MIP limit */ - *varcus = countnint; - if(MIP_count(lp) > 0) { - if((countnint == 0) && (lp->solutioncount == 1) && (lp->solutionlimit == 1) && - (bb_better(lp, OF_DUALLIMIT, OF_TEST_BE) || bb_better(lp, OF_USERBREAK, OF_TEST_BE | OF_TEST_RELGAP))) { - lp->bb_break = (MYBOOL) (countnint == 0); - return( FALSE ); - } - else if(lp->bb_level > 0) { -#ifdef MIPboundWithOF - if((lp->constraintOF > 0) && (countnint == 0)) - set_rh(lp, lp->constraintOF, lp->solution[0] + my_chsign(!is_maxim(lp), lp->bb_deltaOF)); -#endif - if(lp->spx_trace) - report(lp, DETAILED, "B&B level %5d OPT %16s value " RESULTVALUEMASK "\n", - lp->bb_level, (*varno) ? " " : "INT", lp->solution[0]); - } - return( (MYBOOL) (*varno > 0)); - } - else - return( FALSE ); - -} - -STATIC int solve_BB(BBrec *BB) -{ - int K, status; - lprec *lp = BB->lp; - - /* Protect against infinite recursions do to integer rounding effects */ - status = PROCFAIL; - - /* Shortcut variables, set default bounds */ - K = BB->varno; - - /* Load simple MIP bounds */ - if(K > 0) { - - /* Update cuts, if specified */ - updatecuts_BB(lp); - - /* BRANCH_FLOOR: Force the variable to be smaller than the B&B upper bound */ - if(BB->isfloor) - modifyUndoLadder(lp->bb_upperchange, K, BB->upbo, BB->UPbound); - - /* BRANCH_CEILING: Force the variable to be greater than the B&B lower bound */ - else - modifyUndoLadder(lp->bb_lowerchange, K, BB->lowbo, BB->LObound); - - /* Update MIP node count */ - BB->nodessolved++; - - } - - /* Solve! */ - status = solve_LP(lp, BB); - - /* Do special feasibility assessment of high order SOS'es */ -#if 1 - if((status == OPTIMAL) && (BB->vartype == BB_SOS) && !SOS_is_feasible(lp->SOS, 0, lp->solution)) - status = INFEASIBLE; -#endif - - return( status ); -} - -/* Routine to compute a "strong" pseudo-cost update for a node */ -STATIC MYBOOL strongbranch_BB(lprec *lp, BBrec *BB, int varno, int vartype, int varcus) -{ - MYBOOL success = FALSE; - int i; - BBrec *strongBB; - - /* Create new B&B level and solve each of the branches */ - lp->is_strongbranch = TRUE; - push_basis(lp, lp->var_basic, lp->is_basic, lp->is_lower); - strongBB = push_BB(lp, BB, lp->rows+varno, vartype, varcus); - if(strongBB == BB) - return( success ); - - do { - - /* Solve incremental problem to local optimality */ - lp->bb_strongbranches++; -/* set_action(&lp->spx_action, ACTION_REBASE | ACTION_RECOMPUTE); */ - if(solve_BB(strongBB) == OPTIMAL) { - - /* Update result indicator*/ - success |= 1 << strongBB->isfloor; - - /* Compute new count of non-ints */ - strongBB->lastvarcus = 0; - for(i = 1; i <= lp->columns; i++) { - if(is_int(lp, i) && !solution_is_int(lp, lp->rows+i, FALSE)) - strongBB->lastvarcus++; - } - - /* Perform the pseudo-cost update */ - update_pseudocost(lp->bb_PseudoCost, varno, strongBB->vartype, strongBB->isfloor, - lp->solution[strongBB->varno]); - } - } - while(nextbranch_BB(strongBB)); - - strongBB = pop_BB(strongBB); - if(strongBB != BB) - report(lp, SEVERE, "strongbranch_BB: Invalid bound settings restored for variable %d\n", - varno); - pop_basis(lp, TRUE); - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - - lp->is_strongbranch = FALSE; - - return( success ); -} - -/* Future functions */ -STATIC MYBOOL pre_BB(lprec *lp) -{ - return( TRUE ); -} -STATIC MYBOOL post_BB(lprec *lp) -{ - return( TRUE ); -} - -/* This is the non-recursive B&B driver routine - beautifully simple, yet so subtle! */ -STATIC int run_BB(lprec *lp) -{ - BBrec *currentBB; - int varno, vartype, varcus, prevsolutions; - int status = NOTRUN; - - /* Initialize */ - pre_BB(lp); - prevsolutions = lp->solutioncount; -#ifdef UseMilpSlacksRCF /* Check if we should include ranged constraints */ - varno = lp->sum; -#else - varno = lp->columns; -#endif - lp->bb_upperchange = createUndoLadder(lp, varno, 2*MIP_count(lp)); - lp->bb_lowerchange = createUndoLadder(lp, varno, 2*MIP_count(lp)); - lp->rootbounds = currentBB = push_BB(lp, NULL, 0, BB_REAL, 0); - - /* Perform the branch & bound loop */ - while(lp->bb_level > 0) { - status = solve_BB(currentBB); - -#if 0 - if((lp->bb_level == 1) && (MIP_count(lp) > 0)) { - if(status == RUNNING) - ; - - /* Check if there was an integer solution of an aborted model */ - else if((status == SUBOPTIMAL) && (lp->solutioncount == 1) && - findnode_BB(currentBB, &varno, &vartype, &varcus)) - status = USERABORT; - } -#endif - - if((status == OPTIMAL) && findnode_BB(currentBB, &varno, &vartype, &varcus)) - currentBB = push_BB(lp, currentBB, varno, vartype, varcus); - - else while((lp->bb_level > 0) && !nextbranch_BB(currentBB)) - currentBB = pop_BB(currentBB); - - } - - /* Finalize */ - freeUndoLadder(&(lp->bb_upperchange)); - freeUndoLadder(&(lp->bb_lowerchange)); - - /* Check if we should adjust status */ - if(lp->solutioncount > prevsolutions) { - if((status == PROCBREAK) || (status == USERABORT) || (status == TIMEOUT) || userabort(lp, -1)) - status = SUBOPTIMAL; - else - status = OPTIMAL; - if(lp->bb_totalnodes > 0) - lp->spx_status = OPTIMAL; - } - post_BB(lp); - return( status ); -} - diff --git a/code/3rd_lpsolve/lp_mipbb.h b/code/3rd_lpsolve/lp_mipbb.h deleted file mode 100644 index db1947c2..00000000 --- a/code/3rd_lpsolve/lp_mipbb.h +++ /dev/null @@ -1,64 +0,0 @@ -#ifndef HEADER_lp_mipbb -#define HEADER_lp_mipbb - -#include "lp_types.h" -#include "lp_utils.h" - - -/* Bounds storage for B&B routines */ -typedef struct _BBrec -{ - struct _BBrec *parent; - struct _BBrec *child; - lprec *lp; - int varno; - int vartype; - int lastvarcus; /* Count of non-int variables of the previous branch */ - int lastrcf; - int nodesleft; - int nodessolved; - int nodestatus; - REAL noderesult; - REAL lastsolution; /* Optimal solution of the previous branch */ - REAL sc_bound; - REAL *upbo, *lowbo; - REAL UPbound, LObound; - int UBtrack, LBtrack; /* Signals that incoming bounds were changed */ - MYBOOL contentmode; /* Flag indicating if we "own" the bound vectors */ - MYBOOL sc_canset; - MYBOOL isSOS; - MYBOOL isGUB; - int *varmanaged; /* Extended list of variables managed by this B&B level */ - MYBOOL isfloor; /* State variable indicating the active B&B bound */ - MYBOOL UBzerobased; /* State variable indicating if bounds have been rebased */ -} BBrec; - -#ifdef __cplusplus -extern "C" { -#endif - -STATIC BBrec *create_BB(lprec *lp, BBrec *parentBB, MYBOOL dofullcopy); -STATIC BBrec *push_BB(lprec *lp, BBrec *parentBB, int varno, int vartype, int varcus); -STATIC MYBOOL initbranches_BB(BBrec *BB); -STATIC MYBOOL fillbranches_BB(BBrec *BB); -STATIC MYBOOL nextbranch_BB(BBrec *BB); -STATIC MYBOOL strongbranch_BB(lprec *lp, BBrec *BB, int varno, int vartype, int varcus); -STATIC MYBOOL initcuts_BB(lprec *lp); -STATIC int updatecuts_BB(lprec *lp); -STATIC MYBOOL freecuts_BB(lprec *lp); -STATIC BBrec *findself_BB(BBrec *BB); -STATIC int solve_LP(lprec *lp, BBrec *BB); -STATIC int rcfbound_BB(BBrec *BB, int varno, MYBOOL isINT, REAL *newbound, MYBOOL *isfeasible); -STATIC MYBOOL findnode_BB(BBrec *BB, int *varno, int *vartype, int *varcus); -STATIC int solve_BB(BBrec *BB); -STATIC MYBOOL free_BB(BBrec **BB); -STATIC BBrec *pop_BB(BBrec *BB); - -STATIC int run_BB(lprec *lp); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_mipbb */ - diff --git a/code/3rd_lpsolve/lp_params.c b/code/3rd_lpsolve/lp_params.c deleted file mode 100644 index af83d5c4..00000000 --- a/code/3rd_lpsolve/lp_params.c +++ /dev/null @@ -1,692 +0,0 @@ -#include -#include -#include -#include - -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_report.h" -#include "ini.h" - -typedef int (__WINAPI fn_int_get_function)(lprec *lp); -typedef long (__WINAPI fn_long_get_function)(lprec *lp); -typedef MYBOOL (__WINAPI fn_MYBOOL_get_function)(lprec *lp); -typedef REAL (__WINAPI fn_REAL_get_function)(lprec *lp); -typedef void (__WINAPI fn_int_set_function)(lprec *lp, int value); -typedef void (__WINAPI fn_long_set_function)(lprec *lp, long value); -typedef void (__WINAPI fn_MYBOOL_set_function)(lprec *lp, MYBOOL value); -typedef void (__WINAPI fn_REAL_set_function)(lprec *lp, REAL value); - -#define intfunction 1 -#define longfunction 2 -#define MYBOOLfunction 3 -#define REALfunction 4 - -#define setvalues(values, basemask) values, sizeof(values) / sizeof(*values), basemask -#define setNULLvalues NULL, 0, 0 -#define setvalue(value) value, #value -#define setintfunction(get_function, set_function) { get_function }, { set_function }, intfunction -#define setlongfunction(get_function, set_function) { (fn_int_get_function *) get_function }, {(fn_int_set_function *) set_function }, longfunction -#define setMYBOOLfunction(get_function, set_function) { (fn_int_get_function *) get_function }, { (fn_int_set_function *) set_function }, MYBOOLfunction -#define setREALfunction(get_function, set_function) {(fn_int_get_function *) get_function }, { (fn_int_set_function *) set_function }, REALfunction - -#define WRITE_COMMENTED 0 -#define WRITE_ACTIVE 1 - -struct _values { - int value; - char *svalue; -}; - -struct _functions { - char *par; /* name of parameter in ini file */ - union { - fn_int_get_function *int_get_function; /* set via setintfunction */ - fn_long_get_function *long_get_function; /* set via setlongfunction */ - fn_MYBOOL_get_function *MYBOOL_get_function; /* set via setMYBOOLfunction */ - fn_REAL_get_function *REAL_get_function; /* set via setREALfunction */ - } get_function; - union { - fn_int_set_function *int_set_function; /* set via setintfunction */ - fn_long_set_function *long_set_function; /* set via setlongfunction */ - fn_MYBOOL_set_function *MYBOOL_set_function; /* set via setMYBOOLfunction */ - fn_REAL_set_function *REAL_set_function; /* set via setREALfunction */ - } set_function; - int type; /* set via set*function */ - struct _values *values; /* set via setvalues to a structure of _values */ - int elements; /* or via setNULLvalues if the value is shown as is */ - unsigned int basemask; - int mask; /* WRITE_ACTIVE or WRITE_COMMENTED */ -}; - -static struct _values anti_degen[] = -{ - { setvalue(ANTIDEGEN_NONE) }, - { setvalue(ANTIDEGEN_FIXEDVARS) }, - { setvalue(ANTIDEGEN_COLUMNCHECK) }, - { setvalue(ANTIDEGEN_STALLING) }, - { setvalue(ANTIDEGEN_NUMFAILURE) }, - { setvalue(ANTIDEGEN_LOSTFEAS) }, - { setvalue(ANTIDEGEN_INFEASIBLE) }, - { setvalue(ANTIDEGEN_DYNAMIC) }, - { setvalue(ANTIDEGEN_DURINGBB) }, - { setvalue(ANTIDEGEN_RHSPERTURB) }, - { setvalue(ANTIDEGEN_BOUNDFLIP) }, -}; - -static struct _values basiscrash[] = -{ - { setvalue(CRASH_NONE) }, - /* { setvalue(CRASH_NONBASICBOUNDS) }, */ /* not yet implemented */ - { setvalue(CRASH_MOSTFEASIBLE) }, - { setvalue(CRASH_LEASTDEGENERATE) }, -}; - -static struct _values bb_floorfirst[] = -{ - { setvalue(BRANCH_CEILING) }, - { setvalue(BRANCH_FLOOR) }, - { setvalue(BRANCH_AUTOMATIC) }, -}; - -static struct _values bb_rule[] = -{ - { setvalue(NODE_FIRSTSELECT) }, - { setvalue(NODE_GAPSELECT) }, - { setvalue(NODE_RANGESELECT) }, - { setvalue(NODE_FRACTIONSELECT) }, - { setvalue(NODE_PSEUDOCOSTSELECT) }, - { setvalue(NODE_PSEUDONONINTSELECT) }, - { setvalue(NODE_PSEUDORATIOSELECT) }, - { setvalue(NODE_USERSELECT) }, - { setvalue(NODE_WEIGHTREVERSEMODE) }, - { setvalue(NODE_BRANCHREVERSEMODE) }, - { setvalue(NODE_GREEDYMODE) }, - { setvalue(NODE_PSEUDOCOSTMODE) }, - { setvalue(NODE_DEPTHFIRSTMODE) }, - { setvalue(NODE_RANDOMIZEMODE) }, - { setvalue(NODE_GUBMODE) }, - { setvalue(NODE_DYNAMICMODE) }, - { setvalue(NODE_RESTARTMODE) }, - { setvalue(NODE_BREADTHFIRSTMODE) }, - { setvalue(NODE_AUTOORDER) }, - { setvalue(NODE_RCOSTFIXING) }, - { setvalue(NODE_STRONGINIT) }, -}; - -static struct _values improve[] = -{ - { setvalue(IMPROVE_NONE) }, - { setvalue(IMPROVE_SOLUTION) }, - { setvalue(IMPROVE_DUALFEAS) }, - { setvalue(IMPROVE_THETAGAP) }, - { setvalue(IMPROVE_BBSIMPLEX) }, -}; - -static REAL __WINAPI get_mip_gap_abs(lprec *lp) -{ - return(get_mip_gap(lp, TRUE)); -} - -static REAL __WINAPI get_mip_gap_rel(lprec *lp) -{ - return(get_mip_gap(lp, FALSE)); -} - -static void __WINAPI set_mip_gap_abs(lprec *lp, REAL mip_gap) -{ - set_mip_gap(lp, TRUE, mip_gap); -} - -static void __WINAPI set_mip_gap_rel(lprec *lp, REAL mip_gap) -{ - set_mip_gap(lp, FALSE, mip_gap); -} - -static struct _values pivoting[] = -{ - { setvalue(PRICER_FIRSTINDEX) }, - { setvalue(PRICER_DANTZIG) }, - { setvalue(PRICER_DEVEX) }, - { setvalue(PRICER_STEEPESTEDGE) }, - { setvalue(PRICE_PRIMALFALLBACK) }, - { setvalue(PRICE_MULTIPLE) }, - { setvalue(PRICE_PARTIAL) }, - { setvalue(PRICE_ADAPTIVE) }, - { setvalue(PRICE_RANDOMIZE) }, - { setvalue(PRICE_AUTOPARTIAL) }, - { setvalue(PRICE_LOOPLEFT) }, - { setvalue(PRICE_LOOPALTERNATE) }, - { setvalue(PRICE_HARRISTWOPASS) }, - { setvalue(PRICE_TRUENORMINIT) }, -}; - -static struct _values presolving[] = -{ - { setvalue(PRESOLVE_NONE) }, - { setvalue(PRESOLVE_ROWS) }, - { setvalue(PRESOLVE_COLS) }, - { setvalue(PRESOLVE_LINDEP) }, - { setvalue(PRESOLVE_AGGREGATE) }, - { setvalue(PRESOLVE_SPARSER) }, - { setvalue(PRESOLVE_SOS) }, - { setvalue(PRESOLVE_REDUCEMIP) }, - { setvalue(PRESOLVE_KNAPSACK) }, - { setvalue(PRESOLVE_ELIMEQ2) }, - { setvalue(PRESOLVE_IMPLIEDFREE) }, - { setvalue(PRESOLVE_REDUCEGCD) }, - { setvalue(PRESOLVE_PROBEFIX) }, - { setvalue(PRESOLVE_PROBEREDUCE) }, - { setvalue(PRESOLVE_ROWDOMINATE) }, - { setvalue(PRESOLVE_COLDOMINATE) }, - { setvalue(PRESOLVE_MERGEROWS) }, - { setvalue(PRESOLVE_IMPLIEDSLK) }, - { setvalue(PRESOLVE_COLFIXDUAL) }, - { setvalue(PRESOLVE_BOUNDS) }, - { setvalue(PRESOLVE_DUALS) }, - { setvalue(PRESOLVE_SENSDUALS) }, -}; - -static char *STRLWR(char *str) -{ - char *ptr; - - for(ptr = str; *ptr; ptr++) - *ptr = (char) tolower((unsigned char) *ptr); - - return(str); -} - -static char *STRUPR(char *str) -{ - char *ptr; - - for(ptr = str; *ptr; ptr++) - *ptr = (char) toupper((unsigned char) *ptr); - - return(str); -} - -static void __WINAPI set_presolve1(lprec *lp, int do_presolve) -{ - set_presolve(lp, do_presolve, get_presolveloops(lp)); -} - -static void __WINAPI set_presolve2(lprec *lp, int maxloops) -{ - set_presolve(lp, get_presolve(lp), maxloops); -} - -static struct _values print_sol[] = -{ - { FALSE, "0" }, - { TRUE, "1" }, - { setvalue(AUTOMATIC) }, -}; - -static struct _values scaling[] = -{ - { setvalue(SCALE_NONE) }, - { setvalue(SCALE_EXTREME) }, - { setvalue(SCALE_RANGE) }, - { setvalue(SCALE_MEAN) }, - { setvalue(SCALE_GEOMETRIC) }, - { setvalue(SCALE_CURTISREID) }, - { setvalue(SCALE_QUADRATIC) }, - { setvalue(SCALE_LOGARITHMIC) }, - { setvalue(SCALE_USERWEIGHT) }, - { setvalue(SCALE_POWER2) }, - { setvalue(SCALE_EQUILIBRATE) }, - { setvalue(SCALE_INTEGERS) }, - { setvalue(SCALE_DYNUPDATE) }, - { setvalue(SCALE_ROWSONLY) }, - { setvalue(SCALE_COLSONLY) }, -}; - -static struct _values simplextype[] = -{ - { setvalue(SIMPLEX_PRIMAL_PRIMAL) }, - { setvalue(SIMPLEX_DUAL_PRIMAL) }, - { setvalue(SIMPLEX_PRIMAL_DUAL) }, - { setvalue(SIMPLEX_DUAL_DUAL) }, -}; - -static struct _values verbose[] = -{ - { setvalue(NEUTRAL) }, - { setvalue(CRITICAL) }, - { setvalue(SEVERE) }, - { setvalue(IMPORTANT) }, - { setvalue(NORMAL) }, - { setvalue(DETAILED) }, - { setvalue(FULL) }, -}; - -static struct _functions functions[] = -{ - /* solve options */ - { "ANTI_DEGEN", setintfunction(get_anti_degen, set_anti_degen), setvalues(anti_degen, ~0), WRITE_ACTIVE }, - { "BASISCRASH", setintfunction(get_basiscrash, set_basiscrash), setvalues(basiscrash, ~0), WRITE_ACTIVE }, - { "IMPROVE", setintfunction(get_improve, set_improve), setvalues(improve, ~0), WRITE_ACTIVE }, - { "MAXPIVOT", setintfunction(get_maxpivot, set_maxpivot), setNULLvalues, WRITE_ACTIVE }, - { "NEGRANGE", setREALfunction(get_negrange, set_negrange), setNULLvalues, WRITE_ACTIVE }, - { "PIVOTING", setintfunction(get_pivoting, set_pivoting), setvalues(pivoting, PRICER_LASTOPTION), WRITE_ACTIVE }, - { "PRESOLVE", setintfunction(get_presolve, set_presolve1), setvalues(presolving, ~0), WRITE_ACTIVE }, - { "PRESOLVELOOPS", setintfunction(get_presolveloops, set_presolve2), setNULLvalues, WRITE_ACTIVE }, - { "SCALELIMIT", setREALfunction(get_scalelimit, set_scalelimit), setNULLvalues, WRITE_ACTIVE }, - { "SCALING", setintfunction(get_scaling, set_scaling), setvalues(scaling, SCALE_CURTISREID), WRITE_ACTIVE }, - { "SIMPLEXTYPE", setintfunction(get_simplextype, set_simplextype), setvalues(simplextype, ~0), WRITE_ACTIVE }, - { "OBJ_IN_BASIS", setMYBOOLfunction(is_obj_in_basis, set_obj_in_basis), setNULLvalues, WRITE_COMMENTED }, - - /* B&B options */ - { "BB_DEPTHLIMIT", setintfunction(get_bb_depthlimit, set_bb_depthlimit), setNULLvalues, WRITE_ACTIVE }, - { "BB_FLOORFIRST", setintfunction(get_bb_floorfirst, set_bb_floorfirst), setvalues(bb_floorfirst, ~0), WRITE_ACTIVE }, - { "BB_RULE", setintfunction(get_bb_rule, set_bb_rule), setvalues(bb_rule, NODE_STRATEGYMASK), WRITE_ACTIVE }, - { "BREAK_AT_FIRST", setMYBOOLfunction(is_break_at_first, set_break_at_first), setNULLvalues, WRITE_COMMENTED }, - { "BREAK_AT_VALUE", setREALfunction(get_break_at_value, set_break_at_value), setNULLvalues, WRITE_COMMENTED }, - { "MIP_GAP_ABS", setREALfunction(get_mip_gap_abs, set_mip_gap_abs), setNULLvalues, WRITE_ACTIVE }, - { "MIP_GAP_REL", setREALfunction(get_mip_gap_rel, set_mip_gap_rel), setNULLvalues, WRITE_ACTIVE }, - { "EPSINT", setREALfunction(get_epsint, set_epsint), setNULLvalues, WRITE_ACTIVE }, - - /* tolerances, values */ - { "EPSB", setREALfunction(get_epsb, set_epsb), setNULLvalues, WRITE_ACTIVE }, - { "EPSD", setREALfunction(get_epsd, set_epsd), setNULLvalues, WRITE_ACTIVE }, - { "EPSEL", setREALfunction(get_epsel, set_epsel), setNULLvalues, WRITE_ACTIVE }, - { "EPSPERTURB", setREALfunction(get_epsperturb, set_epsperturb), setNULLvalues, WRITE_ACTIVE }, - { "EPSPIVOT", setREALfunction(get_epspivot, set_epspivot), setNULLvalues, WRITE_ACTIVE }, - { "INFINITE", setREALfunction(get_infinite, set_infinite), setNULLvalues, WRITE_ACTIVE }, - { "BREAK_NUMERIC_ACCURACY", setREALfunction(get_break_numeric_accuracy, set_break_numeric_accuracy), setNULLvalues, WRITE_ACTIVE }, - - /* read-only options */ - { "DEBUG", setMYBOOLfunction(is_debug, set_debug), setNULLvalues, WRITE_COMMENTED }, - { "OBJ_BOUND", setREALfunction(get_obj_bound, set_obj_bound), setNULLvalues, WRITE_COMMENTED }, - { "PRINT_SOL", setintfunction(get_print_sol, set_print_sol), setvalues(print_sol, ~0), WRITE_COMMENTED }, - { "TIMEOUT", setlongfunction(get_timeout, set_timeout), setNULLvalues, WRITE_COMMENTED }, - { "TRACE", setMYBOOLfunction(is_trace, set_trace), setNULLvalues, WRITE_COMMENTED }, - { "VERBOSE", setintfunction(get_verbose, set_verbose), setvalues(verbose, ~0), WRITE_COMMENTED }, -}; - -static void write_params1(lprec *lp, FILE *fp, char *header, int newline) -{ - int ret = 0, ret2, i, j, k, value, value2, elements, majorversion, minorversion, release, build; - unsigned int basemask; - REAL a = 0; - char buf[4096], par[30]; - - ini_writeheader(fp, header, newline); - lp_solve_version(&majorversion, &minorversion, &release, &build); - sprintf(buf, "lp_solve version %d.%d settings\n", majorversion, minorversion); - ini_writecomment(fp, buf); - for(i = 0; i < sizeof(functions) / sizeof(*functions); i++) { - switch(functions[i].type) { - case intfunction: - if(functions[i].get_function.int_get_function == NULL) - continue; - ret = functions[i].get_function.int_get_function(lp); - break; - case longfunction: - if(functions[i].get_function.long_get_function == NULL) - continue; - ret = functions[i].get_function.long_get_function(lp); - break; - case MYBOOLfunction: - if(functions[i].get_function.MYBOOL_get_function == NULL) - continue; - ret = (int) functions[i].get_function.MYBOOL_get_function(lp); - break; - case REALfunction: - if(functions[i].get_function.REAL_get_function == NULL) - continue; - a = functions[i].get_function.REAL_get_function(lp); - break; - } - buf[0] = 0; - if(functions[i].values == NULL) { - switch(functions[i].type) { - case intfunction: - case longfunction: - case MYBOOLfunction: - sprintf(buf, "%d", ret); - break; - case REALfunction: - sprintf(buf, "%g", a); - break; - } - } - else { - elements = functions[i].elements; - basemask = functions[i].basemask; - for(j = 0; j < elements; j++) { - value = functions[i].values[j].value; - ret2 = ret; - if(((unsigned int) value) < basemask) - ret2 &= basemask; - if(value == 0) { - if(ret2 == 0) { - if(*buf) - strcat(buf, " + "); - strcat(buf, functions[i].values[j].svalue); - } - } - else if((ret2 & value) == value) { - for(k = 0; k < elements; k++) { - value2 = functions[i].values[k].value; - if((k != j) && (value2 > value) && ((value2 & value) == value) && ((ret2 & value2) == value2)) - break; - } - if(k == elements) { - if(*buf) - strcat(buf, " + "); - strcat(buf, functions[i].values[j].svalue); - } - } - } - } - if(functions[i].mask & WRITE_ACTIVE) - par[0] = 0; - else - strcpy(par, ";"); - strcat(par, functions[i].par); - ini_writedata(fp, STRLWR(par), buf); - } -} - -static void readoptions(char *options, char **header) -{ - char *ptr1, *ptr2; - - if(options != NULL) { - ptr1 = options; - while(*ptr1) { - ptr2 = strchr(ptr1, '-'); - if(ptr2 == NULL) - break; - ptr2++; - if(tolower((unsigned char) *ptr2) == 'h') { - for(++ptr2; (*ptr2) && (isspace(*ptr2)); ptr2++); - for(ptr1 = ptr2; (*ptr1) && (!isspace(*ptr1)); ptr1++); - *header = (char *) calloc(1 + (int) (ptr1 - ptr2), 1); - memcpy(*header, ptr2, (int) (ptr1 - ptr2)); - } - } - } - - if(*header == NULL) - *header = strdup("Default"); -} - -MYBOOL __WINAPI write_params(lprec *lp, char *filename, char *options) -{ - int k, ret, params_written; - FILE *fp, *fp0; - int state = 0, looping, newline; - char buf[4096], *filename0, *ptr1, *ptr2, *header = NULL; - - readoptions(options, &header); - - k = (int) strlen(filename); - filename0 = (char *) malloc(k + 1 + 1); - strcpy(filename0, filename); - ptr1 = strrchr(filename0, '.'); - ptr2 = strrchr(filename0, '\\'); - if((ptr1 == NULL) || ((ptr2 != NULL) && (ptr1 < ptr2))) - ptr1 = filename0 + k; - memmove(ptr1 + 1, ptr1, k + 1 - (int) (ptr1 - filename0)); - ptr1[0] = '_'; - if(rename(filename, filename0)) { - switch(errno) { - case ENOENT: /* File or path specified by oldname not found */ - FREE(filename0); - filename0 = NULL; - break; - case EACCES: /* File or directory specified by newname already exists or could not be created (invalid path); or oldname is a directory and newname specifies a different path. */ - FREE(filename0); - FREE(header); - return(FALSE); - break; - } - } - - if((fp = ini_create(filename)) == NULL) - ret = FALSE; - else { - params_written = FALSE; - newline = TRUE; - if(filename0 != NULL) { - fp0 = ini_open(filename0); - if(fp0 == NULL) { - rename(filename0, filename); - FREE(filename0); - FREE(header); - return(FALSE); - } - looping = TRUE; - while(looping) { - switch(ini_readdata(fp0, buf, sizeof(buf), TRUE)) { - case 0: /* End of file */ - looping = FALSE; - break; - case 1: /* header */ - ptr1 = strdup(buf); - STRUPR(buf); - ptr2 = strdup(header); - STRUPR(ptr2); - if(strcmp(buf, ptr2) == 0) { - write_params1(lp, fp, ptr1, newline); - params_written = TRUE; - newline = TRUE; - state = 1; - } - else { - state = 0; - ini_writeheader(fp, ptr1, newline); - newline = TRUE; - } - FREE(ptr2); - FREE(ptr1); - break; - case 2: /* data */ - if(state == 0) { - ini_writedata(fp, NULL, buf); - newline = (*buf != 0); - } - break; - } - } - ini_close(fp0); - } - - if(!params_written) - write_params1(lp, fp, header, newline); - - ini_close(fp); - ret = TRUE; - } - - if(filename0 != NULL) { - remove(filename0); - FREE(filename0); - } - - FREE(header); - - return( (MYBOOL) ret ); -} - - -MYBOOL __WINAPI read_params(lprec *lp, char *filename, char *options) -{ - int ret, looping, line; - FILE *fp; - hashtable *hashfunctions, *hashparameters; - hashelem *hp; - int i, j, elements, n, intvalue, state = 0; - REAL REALvalue; - char buf[4096], *header = NULL, *ptr, *ptr1, *ptr2; - - if((fp = ini_open(filename)) == NULL) - ret = FALSE; - else { - /* create hashtable of all callable commands to find them quickly */ - hashfunctions = create_hash_table(sizeof(functions) / sizeof(*functions), 0); - for (n = 0, i = 0; i < (int) (sizeof(functions)/sizeof(*functions)); i++) { - puthash(functions[i].par, i, NULL, hashfunctions); - if(functions[i].values != NULL) - n += functions[i].elements; - } - /* create hashtable of all arguments to find them quickly */ - hashparameters = create_hash_table(n, 0); - for (n = 0, i = 0; i < (int) (sizeof(functions)/sizeof(*functions)); i++) { - if(functions[i].values != NULL) { - elements = functions[i].elements; - for(j = 0; j < elements; j++) - if((strcmp(functions[i].values[j].svalue, "0") != 0) && - (strcmp(functions[i].values[j].svalue, "1") != 0)) - puthash(functions[i].values[j].svalue, j, NULL, hashparameters); - } - } - - readoptions(options, &header); - - STRUPR(header); - ret = looping = TRUE; - line = 0; - while((ret) && (looping)) { - line++; - switch(ini_readdata(fp, buf, sizeof(buf), FALSE)) { - case 0: /* End of file */ - looping = FALSE; - break; - case 1: /* header */ - switch(state) { - case 0: - STRUPR(buf); - if(strcmp(buf, header) == 0) - state = 1; - break; - case 1: - looping = FALSE; - break; - } - break; - case 2: /* data */ - if(state == 1) { - for(ptr = buf; (*ptr) && (isspace(*ptr)); ptr++); - } - else - ptr = NULL; - if((ptr != NULL) && (*ptr)) { - STRUPR(buf); - ptr = strchr(buf, '='); - if(ptr == NULL) { - report(lp, IMPORTANT, "read_params: No equal sign on line %d\n", line); - ret = FALSE; - } - else { - *ptr = 0; - for(ptr1 = buf; isspace(*ptr1); ptr1++); - for(ptr2 = ptr - 1; (ptr2 >= ptr1) && (isspace(*ptr2)); ptr2--); - if(ptr2 <= ptr1) { - report(lp, IMPORTANT, "read_params: No parameter name before equal sign on line %d\n", line); - ret = FALSE; - } - else { - ptr2[1] = 0; - hp = findhash(ptr1, hashfunctions); - if(hp == NULL) { - report(lp, IMPORTANT, "read_params: Unknown parameter name (%s) before equal sign on line %d\n", ptr1, line); - ret = FALSE; - } - else { - i = hp->index; - ptr1 = ++ptr; - intvalue = 0; - REALvalue = 0; - if(functions[i].values == NULL) { - switch(functions[i].type) { - case intfunction: - case longfunction: - case MYBOOLfunction: - intvalue = strtol(ptr1, &ptr2, 10); - while((*ptr2) && (isspace(*ptr2))) - ptr2++; - if(*ptr2) { - report(lp, IMPORTANT, "read_params: Invalid integer value on line %d\n", line); - ret = FALSE; - } - break; - case REALfunction: - REALvalue = strtod(ptr1, &ptr2); - while((*ptr2) && (isspace(*ptr2))) - ptr2++; - if(*ptr2) { - report(lp, IMPORTANT, "read_params: Invalid real value on line %d\n", line); - ret = FALSE; - } - break; - } - } - else { - while(ret) { - ptr = strchr(ptr1, '+'); - if(ptr == NULL) - ptr = ptr1 + strlen(ptr1); - for(; isspace(*ptr1); ptr1++); - for(ptr2 = ptr - 1; (ptr2 >= ptr1) && (isspace(*ptr2)); ptr2--); - if(ptr2 <= ptr1) - break; - else { - ptr2[1] = 0; - hp = findhash(ptr1, hashparameters); - if (hp == NULL) { - report(lp, IMPORTANT, "read_params: Invalid parameter name (%s) on line %d\n", ptr1, line); - ret = FALSE; - } - else { - j = hp->index; - if((j >= functions[i].elements) || - (strcmp(functions[i].values[j].svalue, ptr1))) { - report(lp, IMPORTANT, "read_params: Inappropriate parameter name (%s) on line %d\n", ptr1, line); - ret = FALSE; - } - else { - intvalue += functions[i].values[j].value; - } - } - ptr1 = ptr + 1; - } - } - } - if(ret) { - switch(functions[i].type) { - case intfunction: - functions[i].set_function.int_set_function(lp, intvalue); - break; - case longfunction: - functions[i].set_function.long_set_function(lp, intvalue); - break; - case MYBOOLfunction: - functions[i].set_function.MYBOOL_set_function(lp, (MYBOOL) intvalue); - break; - case REALfunction: - functions[i].set_function.REAL_set_function(lp, REALvalue); - break; - } - } - } - } - } - } - break; - } - } - - FREE(header); - free_hash_table(hashfunctions); - free_hash_table(hashparameters); - - ini_close(fp); - } - - return( (MYBOOL) ret ); -} diff --git a/code/3rd_lpsolve/lp_presolve.c b/code/3rd_lpsolve/lp_presolve.c deleted file mode 100644 index b66674ae..00000000 --- a/code/3rd_lpsolve/lp_presolve.c +++ /dev/null @@ -1,5897 +0,0 @@ - -/* ------------------------------------------------------------------------- - Presolve routines for lp_solve v5.0+ - ------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h, lp_presolve, lp_crash.h, lp_scale.h, lp_report.h - - Release notes: - v1.0.0 1 January 2003 Initial crude version used with lp_solve v4. - v5.0.0 1 January 2004 Significantly expanded and repackaged - presolve routines for lp_solve v5 release. - v5.0.1 1 April 2004 Added reference to new crash module - v5.1.0 20 August 2004 Reworked infeasibility detection. - Added encapsulation of presolve undo logic. - v5.1.1 10 September 2004 Added variable bound tightening based on - full-constraint information, as well as - variable fixing by duality. - v5.2.0 1 January 2005 Fixes to bound fixing handling. - Added fast batch compression after presolve. - Restructured calls by adding presolve wrapper. - Major optimization of identification logic - along with bug fixes. - Enabled storage of eliminated matrix data. - Added function to report on constraint classes. - v5.5.0 1 June 2005 Added implied slack presolve, restructured - looping logic to be more modular, and made - active row/column selection logic faster. - v5.5.1 18 June 2005 Finished sparsity-enhancing logic and added - initial version of column aggregation code. - ------------------------------------------------------------------------- */ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_presolve.h" -#include "lp_crash.h" -#include "lp_scale.h" -#include "lp_report.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -#define presolve_setstatus(one, two) presolve_setstatusex(one, two, __LINE__, __FILE__) -STATIC int presolve_setstatusex(presolverec *psdata, int status, int lineno, char *filename) -{ - if((status == INFEASIBLE) || (status == UNBOUNDED)) { - report(psdata->lp, -#ifdef Paranoia - NORMAL, -#else - DETAILED, -#endif - "presolve_setstatus: Status set to '%s' on code line %d, file '%s'\n", - (status == INFEASIBLE ? "INFEASIBLE" : "UNBOUNDED"), lineno, (filename == NULL ? "Unknown" : filename)); - } - return( status ); -} - -STATIC MYBOOL presolve_statuscheck(presolverec *psdata, int *status) -{ - if(*status == RUNNING) { - lprec *lp = psdata->lp; - if(!mat_validate(lp->matA)) - *status = MATRIXERROR; - else if(userabort(lp, -1)) - *status = lp->spx_status; - } - return( (MYBOOL) (*status == RUNNING) ); -} - -STATIC MYBOOL presolve_createUndo(lprec *lp) -{ - if(lp->presolve_undo != NULL) - presolve_freeUndo(lp); - lp->presolve_undo = (presolveundorec *) calloc(1, sizeof(presolveundorec)); - lp->presolve_undo->lp = lp; - if(lp->presolve_undo == NULL) - return( FALSE ); - return( TRUE ); -} -STATIC MYBOOL inc_presolve_space(lprec *lp, int delta, MYBOOL isrows) -{ - int i, ii, - oldrowcolalloc, rowcolsum, oldrowalloc, oldcolalloc; - presolveundorec *psundo = lp->presolve_undo; - - if(psundo == NULL) { - presolve_createUndo(lp); - psundo = lp->presolve_undo; - } - - /* Set constants */ - oldrowalloc = lp->rows_alloc-delta; - oldcolalloc = lp->columns_alloc-delta; - oldrowcolalloc = lp->sum_alloc-delta; - rowcolsum = lp->sum_alloc + 1; - - /* Reallocate lp memory */ - if(isrows) - allocREAL(lp, &psundo->fixed_rhs, lp->rows_alloc+1, AUTOMATIC); - else - allocREAL(lp, &psundo->fixed_obj, lp->columns_alloc+1, AUTOMATIC); - allocINT(lp, &psundo->var_to_orig, rowcolsum, AUTOMATIC); - allocINT(lp, &psundo->orig_to_var, rowcolsum, AUTOMATIC); - - /* Fill in default values, where appropriate */ - if(isrows) - ii = oldrowalloc+1; - else - ii = oldcolalloc+1; - for(i = oldrowcolalloc+1; i < rowcolsum; i++, ii++) { - psundo->var_to_orig[i] = 0; - psundo->orig_to_var[i] = 0; - if(isrows) - psundo->fixed_rhs[ii] = 0; - else - psundo->fixed_obj[ii] = 0; - } - - return(TRUE); -} -STATIC MYBOOL presolve_setOrig(lprec *lp, int orig_rows, int orig_cols) -{ - presolveundorec *psundo = lp->presolve_undo; - - if(psundo == NULL) - return( FALSE ); - psundo->orig_rows = orig_rows; - psundo->orig_columns = orig_cols; - psundo->orig_sum = orig_rows + orig_cols; - if(lp->wasPresolved) - presolve_fillUndo(lp, orig_rows, orig_cols, FALSE); - return( TRUE ); -} -STATIC MYBOOL presolve_fillUndo(lprec *lp, int orig_rows, int orig_cols, MYBOOL setOrig) -{ - int i; - presolveundorec *psundo = lp->presolve_undo; - - for(i = 0; i <= orig_rows; i++) { - psundo->var_to_orig[i] = i; - psundo->orig_to_var[i] = i; - psundo->fixed_rhs[i] = 0; - } - for(i = 1; i <= orig_cols; i++) { - psundo->var_to_orig[orig_rows + i] = i; - psundo->orig_to_var[orig_rows + i] = i; - psundo->fixed_obj[i] = 0; - } - if(setOrig) - presolve_setOrig(lp, orig_rows, orig_cols); - - return( TRUE ); -} -STATIC MYBOOL presolve_rebuildUndo(lprec *lp, MYBOOL isprimal) -{ - int ik, ie, ix, j, k, *colnrDep; - REAL hold, *value, *solution, *slacks; - presolveundorec *psdata = lp->presolve_undo; - MATrec *mat = NULL; - - /* Point to and initialize undo structure at first call */ - if(isprimal) { - if(psdata->primalundo != NULL) - mat = psdata->primalundo->tracker; - solution = lp->full_solution + lp->presolve_undo->orig_rows; - slacks = lp->full_solution; - } - else { - if(psdata->dualundo != NULL) - mat = psdata->dualundo->tracker; - solution = lp->full_duals; - slacks = lp->full_duals + lp->presolve_undo->orig_rows; - } - if(mat == NULL) - return( FALSE ); - - /* Loop backward over the undo chain */ - for(j = mat->col_tag[0]; j > 0; j--) { - ix = mat->col_tag[j]; - ik = mat->col_end[j-1]; - ie = mat->col_end[j]; - colnrDep = &COL_MAT_ROWNR(ik); - value = &COL_MAT_VALUE(ik); - hold = 0; - k = 0; - for(; ik < ie; ik++, colnrDep += matRowColStep, value += matValueStep) { - - /* Constant term */ - if(*colnrDep == 0) - hold += *value; - - /* Special case with dependence on a slack variable */ - else if(isprimal && (*colnrDep > lp->presolve_undo->orig_columns)) { - k = (*colnrDep) - lp->presolve_undo->orig_columns; - hold -= (*value) * slacks[k]; - slacks[k] = 0; - } - else if(!isprimal && (*colnrDep > lp->presolve_undo->orig_rows)) { - k = (*colnrDep) - lp->presolve_undo->orig_rows; - hold -= (*value) * slacks[k]; - slacks[k] = 0; - } - - /* Dependence on other user variable */ - else - hold -= (*value) * solution[*colnrDep]; - - *value = 0; - } - if(fabs(hold) > lp->epsvalue) - solution[ix] = hold; - } - - return( TRUE ); -} -STATIC MYBOOL presolve_freeUndo(lprec *lp) -{ - presolveundorec *psundo = lp->presolve_undo; - - if(psundo == NULL) - return( FALSE ); - FREE(psundo->orig_to_var); - FREE(psundo->var_to_orig); - FREE(psundo->fixed_rhs); - FREE(psundo->fixed_obj); - if(psundo->deletedA != NULL) - freeUndoLadder(&(psundo->deletedA)); - if(psundo->primalundo != NULL) - freeUndoLadder(&(psundo->primalundo)); - if(psundo->dualundo != NULL) - freeUndoLadder(&(psundo->dualundo)); - FREE(lp->presolve_undo); - return( TRUE ); -} - -STATIC void presolve_storeDualUndo(presolverec *psdata, int rownr, int colnr) -{ - lprec *lp = psdata->lp; - MYBOOL firstdone = FALSE; - int ix, iix, item; - REAL Aij = get_mat(lp, rownr, colnr); - MATrec *mat = lp->matA; - - if(presolve_collength(psdata, colnr) == 0) - return; - - /* Add undo information for the dual of the deleted constraint */ - item = 0; - for(ix = presolve_nextrow(psdata, colnr, &item); ix >= 0; - ix = presolve_nextrow(psdata, colnr, &item)) { - iix = COL_MAT_ROWNR(ix); - if(iix == rownr) - continue; - if(!firstdone) - firstdone = addUndoPresolve(lp, FALSE, rownr, get_mat(lp, 0, colnr)/Aij, - get_mat_byindex(lp, ix, FALSE, TRUE)/Aij, iix); - else - appendUndoPresolve(lp, FALSE, get_mat_byindex(lp, ix, FALSE, TRUE)/Aij, iix); - } -} - -/* ----------------------------------------------------------------------------- */ -/* Presolve debugging routines */ -/* ----------------------------------------------------------------------------- */ -STATIC MYBOOL presolve_SOScheck(presolverec *psdata) -{ - MYBOOL status = TRUE; - lprec *lp = psdata->lp; - int *list, i, j, n, k, nk, colnr, nSOS = SOS_count(lp), nerr = 0; - SOSrec *SOS; - - if(nSOS == 0) - return( status ); - - /* For each SOS and each member check validity */ - for(i = 1; i<= nSOS; i++) { - SOS = lp->SOS->sos_list[i-1]; - list = SOS->members; - n = list[0]; - for(j = 1; j<= n; j++) { - colnr = list[j]; - /* Check valid range */ - if((colnr < 1) || (colnr > lp->columns)) { - nerr++; - report(lp, IMPORTANT, "presolve_SOScheck: A - Column index %d is outside of valid range\n", - colnr); - } - /* Check for deletion */ - if(!isActiveLink(psdata->cols->varmap, colnr)) { - nerr++; - report(lp, IMPORTANT, "presolve_SOScheck: B - Column index %d has been marked for deletion\n", - colnr); - } - /* Check if sorted member array is Ok */ - if(SOS_member_index(lp->SOS, i, colnr) != j) { - nerr++; - report(lp, IMPORTANT, "presolve_SOScheck: C - Column index %d not found in fast search array\n", - colnr); - } - /* Check for variable membership in this SOS record of the sparse storage */ - k = lp->SOS->memberpos[colnr-1]; - nk = lp->SOS->memberpos[colnr]; - while((k < nk) && (lp->SOS->membership[k] != i)) - k++; - if(k >= nk) { - nerr++; - report(lp, IMPORTANT, "presolve_SOScheck: D - Column index %d was not found in sparse array\n", - colnr); - } - } - } - - /* Check that all members in the sparse array can be validated as SOS members */ - for(colnr = 1; colnr <= lp->columns; colnr++) { - k = lp->SOS->memberpos[colnr-1]; - nk = lp->SOS->memberpos[colnr]; - for(; k < nk; k++) { - if(!SOS_is_member(lp->SOS, lp->SOS->membership[k], colnr)) { - nerr++; - report(lp, IMPORTANT, "presolve_SOScheck: E - Sparse array did not indicate column index %d as member of SOS %d\n", - colnr, lp->SOS->membership[k]); - } - } - } - status = (MYBOOL) (nerr == 0); - if(!status) - report(lp, IMPORTANT, "presolve_SOScheck: There were %d errors\n", - nerr); - - - return( status ); -} - -/* ----------------------------------------------------------------------------- */ -/* Presolve routines for tightening the model */ -/* ----------------------------------------------------------------------------- */ - -INLINE REAL presolve_roundrhs(lprec *lp, REAL value, MYBOOL isGE) -{ -#ifdef DoPresolveRounding - REAL eps = PRESOLVE_EPSVALUE*1000, - /* REAL eps = PRESOLVE_EPSVALUE*pow(10.0,MAX(0,log10(1+fabs(value)))), */ - testout = my_precision(value, eps); -#if 1 - if(my_chsign(isGE, value-testout) < 0) - value = testout; -#elif 0 - if(my_chsign(isGE, value-testout) < 0) - value = testout; - else if(value != testout) - value += my_chsign(isGE, (value-testout)/2); - /* value = testout + my_chsign(isGE, (value-testout)/2); */ -#else - if(testout != value) - value += my_chsign(isGE, eps*1000); /* BASE OPTION */ -#endif - -#endif - return( value ); -} - -INLINE REAL presolve_roundval(lprec *lp, REAL value) -{ -#ifdef DoPresolveRounding - /* value = my_precision(value, PRESOLVE_EPSVALUE*MAX(1,log10(1+fabs(value)))); */ - value = my_precision(value, PRESOLVE_EPSVALUE); /* BASE OPTION */ -#endif - return( value ); -} - -INLINE MYBOOL presolve_mustupdate(lprec *lp, int colnr) -{ -#if 0 - return( my_infinite(lp, get_lowbo(lp, colnr)) || - my_infinite(lp, get_upbo(lp, colnr)) ); -#else - return( my_infinite(lp, lp->orig_lowbo[lp->rows+colnr]) || - my_infinite(lp, lp->orig_upbo[lp->rows+colnr]) ); -#endif -} - -INLINE REAL presolve_sumplumin(lprec *lp, int item, psrec *ps, MYBOOL doUpper) -{ - REAL *plu = (doUpper ? ps->pluupper : ps->plulower), - *neg = (doUpper ? ps->negupper : ps->neglower); - - if(fabs(plu[item]) >= lp->infinite) - return( plu[item] ); - else if(fabs(neg[item]) >= lp->infinite) - return( neg[item] ); - else - return( plu[item]+neg[item] ); -} - -INLINE void presolve_range(lprec *lp, int rownr, psrec *ps, REAL *loValue, REAL *hiValue) -{ - *loValue = presolve_sumplumin(lp, rownr, ps, FALSE); - *hiValue = presolve_sumplumin(lp, rownr, ps, TRUE); -} - -STATIC void presolve_rangeorig(lprec *lp, int rownr, psrec *ps, REAL *loValue, REAL *hiValue, REAL delta) -{ - delta = my_chsign(is_chsign(lp, rownr), lp->presolve_undo->fixed_rhs[rownr] + delta); - *loValue = presolve_sumplumin(lp, rownr, ps, FALSE) + delta; - *hiValue = presolve_sumplumin(lp, rownr, ps, TRUE) + delta; -} - -STATIC MYBOOL presolve_rowfeasible(presolverec *psdata, int rownr, MYBOOL userowmap) -{ - lprec *lp = psdata->lp; - MYBOOL status = TRUE; - int contype, origrownr = rownr; - REAL LHS, RHS, value; - - /* Optionally loop across all active rows in the provided map (debugging) */ - if(userowmap) - rownr = firstActiveLink(psdata->rows->varmap); - - /* Now do once for ingoing rownr or loop across rowmap */ - while((status == TRUE) && (rownr != 0)) { - - /* Check the lower bound */ - value = presolve_sumplumin(lp, rownr, psdata->rows, TRUE); - LHS = get_rh_lower(lp, rownr); - if(value < LHS-lp->epssolution) { - contype = get_constr_type(lp, rownr); - report(lp, NORMAL, "presolve_rowfeasible: Lower bound infeasibility in %s row %s (%g << %g)\n", - get_str_constr_type(lp, contype), get_row_name(lp, rownr), value, LHS); - if(rownr != origrownr) - report(lp, NORMAL, " ... Input row base used for testing was %s\n", - get_row_name(lp, origrownr)); - status = FALSE; - } - - /* Check the upper bound */ - value = presolve_sumplumin(lp, rownr, psdata->rows, FALSE); - RHS = get_rh_upper(lp, rownr); - if(value > RHS+lp->epssolution) { - contype = get_constr_type(lp, rownr); - report(lp, NORMAL, "presolve_rowfeasible: Upper bound infeasibility in %s row %s (%g >> %g)\n", - get_str_constr_type(lp, contype), get_row_name(lp, rownr), value, RHS); - status = FALSE; - } - if(userowmap) - rownr = nextActiveLink(psdata->rows->varmap, rownr); - else - rownr = 0; - } - return( status ); -} - -STATIC MYBOOL presolve_debugmap(presolverec *psdata, char *caption) -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - int colnr, ix, ie, nx, jx, je, *cols, *rows, n; - int nz = mat->col_end[lp->columns] - 1; - MYBOOL status = FALSE; - - for(colnr = 1; colnr <= lp->columns; colnr++) { - rows = psdata->cols->next[colnr]; - if(!isActiveLink(psdata->cols->varmap, colnr)) { - if(rows != NULL) { - report(lp, SEVERE, "presolve_debugmap: Inactive column %d is non-empty\n", - colnr); - goto Done; - } - else - continue; - } - if(rows == NULL) - report(lp, SEVERE, "presolve_debugmap: Active column %d is empty\n", - colnr); - je = *rows; - rows++; - for(jx = 1; jx <= je; jx++, rows++) { - if((*rows < 0) || (*rows > nz)) { - report(lp, SEVERE, "presolve_debugmap: NZ index %d for column %d out of range (index %d<=%d)\n", - *rows, colnr, jx, je); - goto Done; - } - cols = psdata->rows->next[COL_MAT_ROWNR(*rows)]; - ie = cols[0]; - n = 0; - for(ix = 1; ix <= ie; ix++) { - nx = cols[ix]; - if((nx < 0) || (nx > nz)) { - report(lp, SEVERE, "presolve_debugmap: NZ index %d for column %d to row %d out of range\n", - nx, colnr, jx); - goto Done; - } - } - } - } - status = TRUE; -Done: - if(!status && (caption != NULL)) - report(lp, SEVERE, "...caller was '%s'\n", caption); - return( status ); -} - -STATIC MYBOOL presolve_validate(presolverec *psdata, MYBOOL forceupdate) -{ - int i, ie, j, je, k, rownr, *items; - REAL upbound, lobound, value; - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - MYBOOL status = mat->row_end_valid && !forceupdate; - - if(status) - return( status ); - else if(!mat->row_end_valid) - status = mat_validate(mat); - else - status = forceupdate; - if(status) { - - /* First update rows... */ - for(i = 1; i <= lp->rows; i++) { - - psdata->rows->plucount[i] = 0; - psdata->rows->negcount[i] = 0; - psdata->rows->pluneg[i] = 0; - - if(!isActiveLink(psdata->rows->varmap, i)) { - FREE(psdata->rows->next[i]); - } - else { - /* Create next column pointers by row */ - k = mat_rowlength(mat, i); - allocINT(lp, &(psdata->rows->next[i]), k+1, AUTOMATIC); - items = psdata->rows->next[i]; - je = mat->row_end[i]; - k = 0; - for(j = mat->row_end[i-1]; j < je; j++) - if(isActiveLink(psdata->cols->varmap, ROW_MAT_COLNR(j))) { - k++; - items[k] = j; - } - items[0] = k; - } - } - - /* ...then update columns */ - for(j = 1; j <= lp->columns; j++) { - - psdata->cols->plucount[j] = 0; - psdata->cols->negcount[j] = 0; - psdata->cols->pluneg[j] = 0; - - if(!isActiveLink(psdata->cols->varmap, j)) { - FREE(psdata->cols->next[j]); - } - else { - upbound = get_upbo(lp, j); - lobound = get_lowbo(lp, j); - if(is_semicont(lp, j) && (upbound > lobound)) { - if(lobound > 0) - lobound = 0; - else if(upbound < 0) - upbound = 0; - } - - /* Create next row pointers by column */ - k = mat_collength(mat, j); - allocINT(lp, &(psdata->cols->next[j]), k+1, AUTOMATIC); - items = psdata->cols->next[j]; - ie = mat->col_end[j]; - k = 0; - for(i = mat->col_end[j-1]; i < ie; i++) { - rownr = COL_MAT_ROWNR(i); - if(isActiveLink(psdata->rows->varmap, rownr)) { - k++; - items[k] = i; - - /* Cumulate counts */ - value = COL_MAT_VALUE(i); - if(my_chsign(is_chsign(lp, rownr), value) > 0) { - psdata->rows->plucount[rownr]++; - psdata->cols->plucount[j]++; - } - else { - psdata->rows->negcount[rownr]++; - psdata->cols->negcount[j]++; - } - if((lobound < 0) && (upbound >= 0)) { - psdata->rows->pluneg[rownr]++; - psdata->cols->pluneg[j]++; - } - } - } - items[0] = k; - } - } -#ifdef Paranoia - presolve_debugmap(psdata, "presolve_validate"); -#endif - } - return( status ); -} - -STATIC MYBOOL presolve_rowtallies(presolverec *psdata, int rownr, int *plu, int *neg, int *pluneg) -{ - REAL value; - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - int ix, jx, ib = 0; - MYBOOL chsign = is_chsign(lp, rownr); - - /* Initialize */ - *plu = 0; - *neg = 0; - *pluneg = 0; - - /* Loop over still active row members */ - for(ix = presolve_nextcol(psdata, rownr, &ib); ix >= 0; ix = presolve_nextcol(psdata, rownr, &ib)) { - - /* Get matrix column and value */ - jx = ROW_MAT_COLNR(ix); - value = ROW_MAT_VALUE(ix); - - /* Cumulate counts */ - if(my_chsign(chsign, value) > 0) - (*plu)++; - else - (*neg)++; - if((get_lowbo(lp, jx) < 0) && (get_upbo(lp, jx) >= 0)) - (*pluneg)++; - } - return( TRUE ); -} -STATIC MYBOOL presolve_debugrowtallies(presolverec *psdata) -{ - lprec *lp = psdata->lp; - int i, plu, neg, pluneg, nerr = 0; - - for(i = 1; i <= lp->rows; i++) - if(isActiveLink(psdata->rows->varmap, i) && - presolve_rowtallies(psdata, i, &plu, &neg, &pluneg)) { - if((psdata->rows->plucount[i] != plu) || - (psdata->rows->negcount[i] != neg) || - (psdata->rows->pluneg[i] != pluneg)) { - nerr++; - report(lp, SEVERE, "presolve_debugrowtallies: Detected inconsistent count for row %d\n", i); - } - } - return( (MYBOOL) (nerr == 0) ); -} - -STATIC int presolve_debugcheck(lprec *lp, LLrec *rowmap, LLrec *colmap) -{ - int i, j, errc = 0; - - /* Validate constraint bounds */ - for(i = 1; i < lp->rows; i++) { - if((rowmap != NULL) && !isActiveLink(rowmap, i)) - continue; - /* Check if we have a negative range */ - if(lp->orig_upbo[i] < 0) { - errc++; - report(lp, SEVERE, "presolve_debugcheck: Detected negative range %g for row %d\n", - lp->orig_upbo[i], i); - } - } - /* Validate variables */ - for(j = 1; j < lp->columns; j++) { - if((colmap != NULL) && !isActiveLink(colmap, j)) - continue; - i = lp->rows+j; - /* Check if we have infeasible bounds */ - if(lp->orig_lowbo[i] > lp->orig_upbo[i]) { - errc++; - report(lp, SEVERE, "presolve_debugcheck: Detected UB < LB for column %d\n", - j); - } - } - /* Return total number of errors */ - return( errc ); -} - -STATIC MYBOOL presolve_candeletevar(presolverec *psdata, int colnr) -{ - lprec *lp = psdata->lp; - int usecount = SOS_memberships(lp->SOS, colnr); - - return( (MYBOOL) ((lp->SOS == NULL) || (usecount == 0) || - (/*is_presolve(lp, PRESOLVE_SOS) &&*/ - (((lp->SOS->sos1_count == lp->SOS->sos_count)) || - (usecount == SOS_is_member_of_type(lp->SOS, colnr, SOS1))))) ); -} - -STATIC int presolve_rowlengthex(presolverec *psdata, int rownr) -{ - int j1 = psdata->rows->plucount[rownr] + psdata->rows->negcount[rownr]; -#ifdef Paranoia - int j2 = presolve_rowlength(psdata, rownr); - - if(j1 != j2) { - report(psdata->lp, SEVERE, "presolve_rowlengthex: Expected row length %d, but found %d in row %s\n", - j2, j1, get_row_name(psdata->lp, rownr)); - j1 = -j1; - } -#endif - - return( j1 ); -} -STATIC int presolve_rowlengthdebug(presolverec *psdata) -{ - int rownr, n = 0; - - for(rownr = firstActiveLink(psdata->rows->varmap); rownr != 0; - rownr = nextActiveLink(psdata->rows->varmap, rownr)) - n += presolve_rowlengthex(psdata, rownr); - return( n ); -} - -INLINE int presolve_nextrecord(psrec *ps, int recnr, int *previtem) -{ - int *nzlist = ps->next[recnr], nzcount = nzlist[0], status = -1; - - /* Check if we simply wish the last active column */ - if(previtem == NULL) { - if(nzlist != NULL) - status = nzlist[*nzlist]; - return( status ); - } - - /* Step to next */ -#ifdef Paranoia - else if((*previtem < 0) || (*previtem > nzcount)) - return( status ); -#endif - (*previtem)++; - - /* Set the return values */ - if(*previtem > nzcount) - (*previtem) = 0; - else - status = nzlist[*previtem]; - - return( status ); -} -INLINE int presolve_nextcol(presolverec *psdata, int rownr, int *previtem) -/* Find the first active (non-eliminated) nonzero column in rownr after prevcol */ -{ - return( presolve_nextrecord(psdata->rows, rownr, previtem) ); -} -INLINE int presolve_lastcol(presolverec *psdata, int rownr) -{ - return( presolve_nextrecord(psdata->rows, rownr, NULL) ); -} -INLINE int presolve_nextrow(presolverec *psdata, int colnr, int *previtem) -/* Find the first active (non-eliminated) nonzero row in colnr after prevrow */ -{ - return( presolve_nextrecord(psdata->cols, colnr, previtem) ); -} -INLINE int presolve_lastrow(presolverec *psdata, int colnr) -{ - return( presolve_nextrecord(psdata->cols, colnr, NULL) ); -} - -INLINE void presolve_adjustrhs(presolverec *psdata, int rownr, REAL fixdelta, REAL epsvalue) -{ - lprec *lp = psdata->lp; - - lp->orig_rhs[rownr] -= fixdelta; - if(epsvalue > 0) -#if 1 - my_roundzero(lp->orig_rhs[rownr], epsvalue); -#else - lp->orig_rhs[rownr] = presolve_roundrhs(lp, lp->orig_rhs[rownr], FALSE); -#endif - lp->presolve_undo->fixed_rhs[rownr] += fixdelta; -} - -STATIC int presolve_shrink(presolverec *psdata, int *nConRemove, int *nVarRemove) -{ - SOSgroup *SOS = psdata->lp->SOS; - int status = RUNNING, countR = 0, countC = 0, - i, ix, n, *list; - REAL fixValue; - - /* Remove empty rows */ - list = psdata->rows->empty; - if(list != NULL) { - n = list[0]; - for(i = 1; i <= n; i++) - if(isActiveLink(psdata->rows->varmap, list[i])) { - presolve_rowremove(psdata, list[i], FALSE); - countR++; - } - if(nConRemove != NULL) - (*nConRemove) += countR; - list[0] = 0; - } - - /* Fix and remove empty columns (unless they are in a SOS) */ - list = psdata->cols->empty; - if(list != NULL) { - n = list[0]; - for(i = 1; i <= n; i++) { - ix = list[i]; - if(isActiveLink(psdata->cols->varmap, ix)) { - if(presolve_colfixdual(psdata, ix, &fixValue, &status)) { - if(!presolve_colfix(psdata, ix, fixValue, TRUE, nVarRemove)) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - presolve_colremove(psdata, ix, FALSE); - countC++; - } - else if(SOS_is_member(SOS, 0, ix)) - report(psdata->lp, DETAILED, "presolve_shrink: Empty column %d is member of a SOS\n", ix); - } - } - list[0] = 0; - } - - return( status ); -} - -STATIC void presolve_rowremove(presolverec *psdata, int rownr, MYBOOL allowcoldelete) -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - int ix, ie, nx, jx, je, *cols, *rows, n, colnr; - -#ifdef Paranoia - if((rownr < 1) || (rownr > lp->rows)) - report(lp, SEVERE, "presolve_rowremove: Row %d out of range\n", rownr); -#endif - - /* Remove this row for each column that is active in the row */ - cols = psdata->rows->next[rownr]; - ie = *cols; - cols++; - for(ix = 1; ix <= ie; ix++, cols++) { - n = 0; - colnr = ROW_MAT_COLNR(*cols); - rows = psdata->cols->next[colnr]; - je = rows[0]; - /* See if we can narrow the search window */ - jx = je / 2; - if((jx > 5) && (rownr >= COL_MAT_ROWNR(rows[jx]))) - n = jx-1; - else - jx = 1; - /* Do the compression loop */ - for(; jx <= je; jx++) { - nx = rows[jx]; - if(COL_MAT_ROWNR(nx) != rownr) { - n++; - rows[n] = nx; - } - } - rows[0] = n; - - /* Make sure we delete columns that have become empty */ -#if 1 - if((n == 0) && allowcoldelete) { - int *list = psdata->cols->empty; - n = ++list[0]; - list[n] = colnr; - } -#endif - - } - FREE(psdata->rows->next[rownr]); - - removeLink(psdata->rows->varmap, rownr); - switch(get_constr_type(lp, rownr)) { - case LE: removeLink(psdata->LTmap, rownr); - break; - case EQ: removeLink(psdata->EQmap, rownr); - break; - } - if(isActiveLink(psdata->INTmap, rownr)) - removeLink(psdata->INTmap, rownr); -} - -STATIC int presolve_colremove(presolverec *psdata, int colnr, MYBOOL allowrowdelete) -{ - lprec *lp = psdata->lp; - -#ifdef Paranoia - if((colnr < 1) || (colnr > lp->columns)) - report(lp, SEVERE, "presolve_colremove: Column %d out of range\n", colnr); - if(!isActiveLink(psdata->cols->varmap, colnr) || !presolve_candeletevar(psdata, colnr)) - colnr = -1; - else -#endif - { - MATrec *mat = lp->matA; - int ix, ie, nx, jx, je, *cols, *rows, n, rownr; - - /* Remove this column for each row that is active in the column */ - rows = psdata->cols->next[colnr]; - je = *rows; - rows++; - for(jx = 1; jx <= je; jx++, rows++) { - n = 0; - rownr = COL_MAT_ROWNR(*rows); - cols = psdata->rows->next[rownr]; - ie = cols[0]; - /* See if we can narrow the search window */ - ix = ie / 2; - if((ix > 5) && (colnr >= ROW_MAT_COLNR(cols[ix]))) - n = ix-1; - else - ix = 1; - /* Do the compression loop */ - for(; ix <= ie; ix++) { - nx = cols[ix]; - if(ROW_MAT_COLNR(nx) != colnr) { - n++; - cols[n] = nx; - } - } - cols[0] = n; - - /* Make sure we delete rows that become empty */ -#if 1 - if((n == 0) && allowrowdelete) { - int *list = psdata->rows->empty; - n = ++list[0]; - list[n] = rownr; - } -#endif - - } - FREE(psdata->cols->next[colnr]); - - /* Update other counts */ - if(SOS_is_member(lp->SOS, 0, colnr)) { - if(lp->sos_priority != NULL) { - lp->sos_vars--; - if(is_int(lp, colnr)) - lp->sos_ints--; - } - SOS_member_delete(lp->SOS, 0, colnr); - clean_SOSgroup(lp->SOS, TRUE); - if(SOS_count(lp) == 0) - free_SOSgroup(&(lp->SOS)); - } - - /* Finally remove the column from the active column list */ - colnr = removeLink(psdata->cols->varmap, colnr); - } - return( colnr ); -} - -STATIC int presolve_redundantSOS(presolverec *psdata, int *nb, int *nSum) -{ - lprec *lp = psdata->lp; - int i, ii, k, kk, j, nrows = lp->rows, *fixed = NULL, - iBoundTighten = 0, status = RUNNING; - SOSrec *SOS; - - /* Is there anything to do? */ - i = ii = SOS_count(lp); - if(ii == 0) - return( status ); - - /* Allocate working member list */ - if(!allocINT(lp, &fixed, lp->columns+1, FALSE) ) - return( lp->spx_status ); - - /* Check if we have SOS'es that are already satisfied or fixable/satisfiable */ - while(i > 0) { - SOS = lp->SOS->sos_list[i-1]; - kk = SOS->members[0]; - fixed[0] = 0; - for(k = 1; k <= kk; k++) { - j = SOS->members[k]; - if((get_lowbo(lp, j) > 0) && !is_semicont(lp, j)) { - fixed[++fixed[0]] = k; - /* Abort if we have identified SOS infeasibility */ - if(fixed[0] > SOS->type) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Done; - } - } - } - /* If there were an exact number of non-zero SOS members, check their sequentiality */ - if(fixed[0] == SOS->type) { - /* Check sequentiality of members with non-zero lower bounds */ - for(k = 2; k <= fixed[0]; k++) { - if(fixed[k] != fixed[k-1]+1) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Done; - } - } - /* Fix other member variables to zero, if necessary */ - for(k = kk; k > 0; k--) { - j = SOS->members[k]; - if((get_lowbo(lp, j) > 0) && !is_semicont(lp, j)) - continue; - if(!presolve_colfix(psdata, j, 0.0, AUTOMATIC, &iBoundTighten)) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Done; - } - } - /* Remove the SOS */ - delete_SOSrec(lp->SOS, i /* , FALSE */); - } - /* Otherwise, try to fix variables outside the SOS type window */ - else if(fixed[0] > 0) { - for(k = kk; k > 0; k--) { - if((k > fixed[fixed[0]]-SOS->type) && /* After leading entries */ - (k < fixed[1]+SOS->type)) /* Before trailing entries */ - continue; - j = SOS->members[k]; - SOS_member_delete(lp->SOS, i, j); - /* if(get_upbo(lp, j) - get_lowbo(lp, j) < lp->epsprimal) */ - if(is_fixedvar(lp, nrows+j)) - continue; - if(!presolve_colfix(psdata, j, 0.0, AUTOMATIC, &iBoundTighten)) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Done; - } - } - } - i--; - } - - /* Update the sparse member map if there were SOS deletions; - Remember that delete_SOSrec() above specified deferred updating! */ - i = SOS_count(lp); - if((i < ii) || (iBoundTighten > 0)) { - SOS_member_updatemap(lp->SOS); - } - - /* Update tag orders */ - for(; i > 0; i--) - lp->SOS->sos_list[i-1]->tagorder = i; - -Done: - FREE(fixed); - (*nb) += iBoundTighten; - (*nSum) += iBoundTighten; - - return( status ); -} - -STATIC MYBOOL presolve_fixSOS1(presolverec *psdata, int colnr, REAL fixvalue, int *nr, int *nv) -{ - lprec *lp = psdata->lp; - int i, k, j; - SOSrec *SOS; - REAL newvalue; - MYBOOL *fixed = NULL, status = FALSE; - - /* Allocate working member list */ - if(!allocMYBOOL(lp, &fixed, lp->columns+1, TRUE) ) - return(FALSE); - - /* Fix variables in SOS's where colnr is a member */ - i = SOS_count(lp); - while(i > 0) { - /* Set next SOS target (note that colnr has been tested earlier as not being a member of a higher order SOS) */ - SOS = lp->SOS->sos_list[i-1]; - if(SOS_is_member(lp->SOS, i, colnr)) { - for(k = SOS->members[0]; k > 0; k--) { - j = SOS->members[k]; - if(fixed[j]) - continue; - if(j == colnr) { - fixed[j] = TRUE; - newvalue = fixvalue; - } - else { - fixed[j] = AUTOMATIC; - newvalue = 0.0; - } - /* If it is a member of a higher order SOS then just change bounds */ - if(!presolve_candeletevar(psdata, j)) { - set_bounds(lp, j, newvalue, newvalue); - fixed[j] = TRUE | AUTOMATIC; - psdata->forceupdate = TRUE; - } - /* Otherwise fix it in preparation for removal */ - else if(!presolve_colfix(psdata, j, newvalue, TRUE, nv)) - goto Done; - } - } - i--; - } - - /* Delete SOS'es or SOS member variables where we can */ - k = i = SOS_count(lp); - while(i > 0) { - /* Set next SOS target */ - SOS = lp->SOS->sos_list[i-1]; - if(SOS_is_member(lp->SOS, i, colnr)) { - /* Always delete SOS1's */ - if(SOS->type == SOS1) - delete_SOSrec(lp->SOS, i /* , FALSE */); - /* Only delete leading or trailing SOS members in higher-order SOS'es that are fixed at 0; - (note that this section of the code will never be called in the current setup) */ - else { - /* First the leading entries... */ - for(j = 1; j <= SOS->members[0]; j++) { - if(fixed[SOS->members[j]] == AUTOMATIC) - SOS_member_delete(lp->SOS, i, SOS->members[j]); - } - /* ...then trailing entries */ - for(j = SOS->members[0]; j > 0; j--) { - if(fixed[SOS->members[j]] == AUTOMATIC) - SOS_member_delete(lp->SOS, i, SOS->members[j]); - } - } - } - i--; - } - - /* Update the sparse member map if there were SOS deletions; delete_SOSrec() above - specified deferred updating */ - i = SOS_count(lp); - if(i < k) - SOS_member_updatemap(lp->SOS); - - /* Delete the variables that have been fixed */ - k = 0; - for(j = lp->columns; j > 0; j--) { - if((fixed[j] == TRUE) || (fixed[j] == AUTOMATIC)) { - presolve_colremove(psdata, j, TRUE); - k++; - } - } - - /* Update tag orders */ - i = SOS_count(lp); - for(; i > 0; i--) - lp->SOS->sos_list[i-1]->tagorder = i; - - status = TRUE; - -Done: - FREE(fixed); - return( status ); -} - -STATIC void presolve_setEQ(presolverec *psdata, int rownr) -{ - lprec *lp = psdata->lp; - - if(is_constr_type(lp, rownr, LE)) - removeLink(psdata->LTmap, rownr); - setLink(psdata->EQmap, rownr); - set_constr_type(lp, rownr, EQ); - psdata->dv_lobo[rownr] = -lp->infinite; - psdata->dv_upbo[rownr] = lp->infinite; -} - -STATIC MYBOOL presolve_singletonbounds(presolverec *psdata, int rownr, int colnr, REAL *lobound, REAL *upbound, REAL *aval) -{ - lprec *lp = psdata->lp; - REAL coeff_a, epsvalue = psdata->epsvalue; - MYBOOL isneg; - - /* Compute row singleton variable range */ - if(is_constr_type(lp, rownr, EQ) && (fabs(*lobound) < epsvalue)) - *lobound = *upbound = 0; - else { - if(aval == NULL) - coeff_a = get_mat(lp, rownr, colnr); - else - coeff_a = *aval; - isneg = (MYBOOL) (coeff_a < 0); - if(*lobound > -lp->infinite) - *lobound /= coeff_a; - else if(isneg) - *lobound = -(*lobound); - if(*upbound < lp->infinite) - *upbound /= coeff_a; - else if(isneg) - *upbound = -(*upbound); - if(isneg) - swapREAL(lobound, upbound); - } - - /* Check against bound - handle SC variables specially */ - if(is_semicont(lp, colnr)) { - coeff_a = get_lowbo(lp, colnr); - if(coeff_a > 0) { - SETMAX(*lobound, 0.0); - SETMIN(*upbound, get_upbo(lp, colnr)); - } - else { - coeff_a = get_upbo(lp, colnr); - if(coeff_a > 0) { - SETMAX(*lobound, get_lowbo(lp, colnr)); - SETMIN(*upbound, 0.0); - } - } - } - else { - SETMAX(*lobound, get_lowbo(lp, colnr)); - SETMIN(*upbound, get_upbo(lp, colnr)); - } - - /* Return with consistency status */ -#ifdef DoPresolveRelativeTest - isneg = (MYBOOL) (my_reldiff(*upbound, *lobound) >= - epsvalue); -#else - isneg = (MYBOOL) (*upbound >= *lobound - epsvalue); -#endif - if(!isneg) { - /* Attempt bound-related error correction */ - if(fabs(my_reldiff(*lobound, get_upbo(lp, colnr))) < PRESOLVE_BOUNDSLACK*epsvalue) - *lobound = get_upbo(lp, colnr); - else if(fabs(my_reldiff(*upbound, get_lowbo(lp, colnr))) < PRESOLVE_BOUNDSLACK*epsvalue) - *upbound = get_lowbo(lp, colnr); -#ifdef DoPresolveRelativeTest - isneg = (MYBOOL) (my_reldiff(*upbound, *lobound) >= - epsvalue); -#else - isneg = (MYBOOL) (*upbound >= *lobound - epsvalue); -#endif - if(!isneg) - report(lp, NORMAL, "presolve_singletonbounds: Singleton variable %s in row %s infeasibility (%g << %g)\n", - get_col_name(lp, colnr), get_row_name(lp, rownr), *lobound, *upbound); - } - return( isneg ); -} - -STATIC MYBOOL presolve_altsingletonvalid(presolverec *psdata, int rownr, int colnr, REAL reflotest, REAL refuptest) -{ - lprec *lp = psdata->lp; - REAL coeff_bl, coeff_bu, epsvalue = psdata->epsvalue; - - coeff_bl = get_rh_lower(lp, rownr); - coeff_bu = get_rh_upper(lp, rownr); - - /* Check base data validity */ -#ifdef DoPresolveRelativeTest - if((my_reldiff(refuptest, reflotest) < -epsvalue) || -#else - if((reflotest > refuptest + epsvalue) || -#endif - !presolve_singletonbounds(psdata, rownr, colnr, &coeff_bl, &coeff_bu, NULL)) - return( FALSE ); - - /* Base data is Ok, now check against against each other */ - epsvalue = MAX(reflotest-coeff_bu, coeff_bl-refuptest) / epsvalue; - if(epsvalue > PRESOLVE_BOUNDSLACK) { - report(lp, NORMAL, "presolve_altsingletonvalid: Singleton variable %s in row %s infeasible (%g)\n", - get_col_name(lp, colnr), get_row_name(lp, rownr), MAX(reflotest-coeff_bu, coeff_bl-refuptest)); - return( FALSE ); - } - else - return( TRUE ); -} - -STATIC MYBOOL presolve_multibounds(presolverec *psdata, int rownr, int colnr, - REAL *lobound, REAL *upbound, REAL *aval, MYBOOL *rowbinds) -{ - lprec *lp = psdata->lp; - MYBOOL rowbindsvar = FALSE, status = FALSE; - REAL coeff_a, LHS, RHS, netX, Xupper, Xlower, epsvalue = psdata->epsvalue; - - /* Get variable bounds for netting */ - LHS = *lobound; - RHS = *upbound; - Xlower = get_lowbo(lp, colnr); - Xupper = get_upbo(lp, colnr); - - /* Identify opportunity for bound tightening */ - if(aval == NULL) - coeff_a = get_mat(lp, rownr, colnr); - else - coeff_a = *aval; - - netX = presolve_sumplumin(lp, rownr, psdata->rows, TRUE); - if(!my_infinite(lp, LHS) && !my_infinite(lp, netX)) { - if(coeff_a > 0) { - LHS -= netX-coeff_a*Xupper; - LHS /= coeff_a; - if(LHS > Xlower + epsvalue) { - Xlower = presolve_roundrhs(lp, LHS, TRUE); - status = TRUE; - } - else if(LHS > Xlower - epsvalue) - rowbindsvar = TRUE; - } - else { - LHS -= netX-coeff_a*Xlower; - LHS /= coeff_a; - if(LHS < Xupper - epsvalue) { - Xupper = presolve_roundrhs(lp, LHS, FALSE); - status = AUTOMATIC; - } - else if(LHS < Xupper + epsvalue) - rowbindsvar = AUTOMATIC; - } - } - - netX = presolve_sumplumin(lp, rownr, psdata->rows, FALSE); - if(!my_infinite(lp, RHS) && !my_infinite(lp, netX)) { - if(coeff_a < 0) { - if(!my_infinite(lp, Xupper)) { - RHS -= netX-coeff_a*Xupper; - RHS /= coeff_a; - if(RHS > Xlower + epsvalue) { - Xlower = presolve_roundrhs(lp, RHS, TRUE); - status |= TRUE; - } - else if(RHS > Xlower - epsvalue) - rowbindsvar |= TRUE; - } - } - else if(!my_infinite(lp, Xlower)) { - RHS -= netX-coeff_a*Xlower; - RHS /= coeff_a; - if(RHS < Xupper - epsvalue) { - Xupper = presolve_roundrhs(lp, RHS, FALSE); - status |= AUTOMATIC; - } - else if(RHS < Xupper + epsvalue) - rowbindsvar |= AUTOMATIC; - } - } - - *lobound = Xlower; - *upbound = Xupper; - if(rowbinds != NULL) - *rowbinds = rowbindsvar; - - return(status); -} - -STATIC MYBOOL isnz_origobj(lprec *lp, int colnr) -{ - return( (MYBOOL) (lp->orig_obj[colnr] != 0) ); -} - -STATIC MYBOOL presolve_testrow(presolverec *psdata, int lastrow) -{ - if(psdata->forceupdate) { - presolve_updatesums(psdata); - psdata->forceupdate = FALSE; - } - if(!presolve_rowfeasible(psdata, 0, TRUE)) - return( FALSE ); - else - return( TRUE ); -} - -STATIC MYBOOL presolve_coltighten(presolverec *psdata, int colnr, REAL LOnew, REAL UPnew, int *count) -{ - lprec *lp = psdata->lp; - int elmnr, elmend, k, oldcount = 0, newcount = 0, deltainf; - REAL LOold, UPold, Value, margin = psdata->epsvalue; - MATrec *mat = lp->matA; - REAL *value; - int *rownr; - - /* Attempt correction of marginally equal, but inconsistent input values */ - Value = UPnew - LOnew; - if((Value <= -margin) && (Value > -lp->epsprimal)) { - if(fabs(fmod(UPnew, 1.0)) < margin) - LOnew = UPnew; - else - UPnew = LOnew; - } - - /* Check if there is anything to do */ - LOold = get_lowbo(lp, colnr); - UPold = get_upbo(lp, colnr); -#ifdef Paranoia - if(((LOold > LOnew) && !is_semicont(lp, colnr)) || (UPold < UPnew)) { - report(lp, SEVERE, "presolve_coltighten: Inconsistent new bounds requested for column %d\n", colnr); - return( FALSE ); - } -#endif - if(count != NULL) - newcount = *count; - oldcount = newcount; - - /* Modify inf-count */ - deltainf = 0; - if((UPold < lp->infinite) || (LOold > -lp->infinite)) - deltainf -= 1; - if((UPnew < lp->infinite) || (LOnew > -lp->infinite)) - deltainf += 1; - if(isnz_origobj(lp, colnr)) - psdata->rows->infcount[0] += deltainf; - elmnr = mat->col_end[colnr-1]; - elmend = mat->col_end[colnr]; - rownr = &COL_MAT_ROWNR(elmnr); - for(; elmnr < elmend; elmnr++, rownr += matRowColStep) { - k = *rownr; - if(isActiveLink(psdata->rows->varmap, k)) - psdata->rows->infcount[k] += deltainf; - } - - /* Look for opportunity to tighten upper variable bound */ - if((UPnew < lp->infinite) && (UPnew+margin < UPold)) { - if(is_int(lp, colnr)) - UPnew = floor(UPnew+margin); - if(UPold < lp->infinite) { - /* First do OF */ - k = 0; - Value = my_chsign(is_chsign(lp, k), lp->orig_obj[colnr]); - if((Value > 0) && (psdata->rows->pluupper[k] < lp->infinite)) - psdata->rows->pluupper[k] += (UPnew-UPold)*Value; - else if((Value < 0) && (psdata->rows->negupper[k] < lp->infinite)) - psdata->rows->negupper[k] += (LOnew-LOold)*Value; - psdata->rows->infcount[k] += deltainf; - - /* Then scan the constraint rows */ - elmnr = mat->col_end[colnr-1]; - elmend = mat->col_end[colnr]; - rownr = &COL_MAT_ROWNR(elmnr); - value = &COL_MAT_VALUE(elmnr); - for(; elmnr < elmend; - elmnr++, rownr += matRowColStep, value += matValueStep) { - k = *rownr; - if(!isActiveLink(psdata->rows->varmap, k)) - continue; - Value = my_chsign(is_chsign(lp, k), *value); - if((Value > 0) && (psdata->rows->pluupper[k] < lp->infinite)) - psdata->rows->pluupper[k] += (UPnew-UPold)*Value; - else if((Value < 0) && (psdata->rows->negupper[k] < lp->infinite)) - psdata->rows->negupper[k] += (LOnew-LOold)*Value; - } - } - else - psdata->forceupdate = TRUE; - if(UPnew < UPold) { - UPold = UPnew; - newcount++; - } - } - - /* Look for opportunity to tighten lower variable bound */ - if((LOnew > -lp->infinite) && (LOnew-margin > LOold)) { - if(is_int(lp, colnr)) - LOnew = ceil(LOnew-margin); - if(LOold > -lp->infinite) { - /* First do OF */ - k = 0; - Value = my_chsign(is_chsign(lp, k), lp->orig_obj[colnr]); - if((Value > 0) && (psdata->rows->plulower[k] > -lp->infinite)) - psdata->rows->plulower[k] += (LOnew-LOold)*Value; - else if((Value < 0) && (psdata->rows->neglower[k] > -lp->infinite)) - psdata->rows->neglower[k] += (UPnew-UPold)*Value; - - /* Then scan the constraint rows */ - elmnr = mat->col_end[colnr-1]; - elmend = mat->col_end[colnr]; - rownr = &COL_MAT_ROWNR(elmnr); - value = &COL_MAT_VALUE(elmnr); - for(; elmnr < elmend; - elmnr++, rownr += matRowColStep, value += matValueStep) { - k = *rownr; - if(!isActiveLink(psdata->rows->varmap, k)) - continue; - Value = my_chsign(is_chsign(lp, k), *value); - if((Value > 0) && (psdata->rows->plulower[k] > -lp->infinite)) - psdata->rows->plulower[k] += (LOnew-LOold)*Value; - else if((Value < 0) && (psdata->rows->neglower[k] > -lp->infinite)) - psdata->rows->neglower[k] += (UPnew-UPold)*Value; - } - } - else - psdata->forceupdate = TRUE; - if(LOnew > LOold) { - LOold = LOnew; - newcount++; - } - } - - /* Now set the new variable bounds, if they are tighter */ - if(newcount > oldcount) { - UPnew = presolve_roundval(lp, UPnew); - LOnew = presolve_roundval(lp, LOnew); - if(LOnew > UPnew) { - if(LOnew-UPnew < margin) { - LOnew = UPnew; - } - else { - report(lp, NORMAL, "presolve_coltighten: Found column %s with LB %g > UB %g\n", - get_col_name(lp, colnr), LOnew, UPnew); - return( FALSE ); - } - } - if(lp->spx_trace || (lp->verbose > DETAILED)) - report(lp, NORMAL, "presolve_coltighten: Replaced bounds on column %s to [%g ... %g]\n", - get_col_name(lp, colnr), LOnew, UPnew); - set_bounds(lp, colnr, LOnew, UPnew); - } - if(count != NULL) - *count = newcount; - - return( TRUE ); -} - -STATIC int presolve_rowtighten(presolverec *psdata, int rownr, int *tally, MYBOOL intsonly) -{ - lprec *lp = psdata->lp; - MYBOOL rowbinds; - int item = 0, jx, jjx, ix, idxn = 0, *idxbound = NULL, status = RUNNING; - REAL *newbound = NULL, RHlo = get_rh_lower(lp, rownr), RHup = get_rh_upper(lp, rownr), - VARlo, VARup, Aval; - MATrec *mat = lp->matA; - - jx = presolve_rowlength(psdata, rownr); - allocREAL(lp, &newbound, 2*jx, TRUE); - allocINT (lp, &idxbound, 2*jx, TRUE); - - /* Identify bound tightening for each active variable in the constraint */ - for(jx = presolve_nextcol(psdata, rownr, &item); jx >= 0; - jx = presolve_nextcol(psdata, rownr, &item)) { - jjx = ROW_MAT_COLNR(jx); - Aval = ROW_MAT_VALUE(jx); - Aval = my_chsign(rownr, Aval); - - VARlo = RHlo; - VARup = RHup; - presolve_multibounds(psdata, rownr,jjx, &VARlo, &VARup, &Aval, &rowbinds); - if(rowbinds & TRUE) { - idxbound[idxn] = -jjx; - newbound[idxn] = VARlo; - idxn++; - } - if(rowbinds & AUTOMATIC) { - idxbound[idxn] = jjx; - newbound[idxn] = VARup; - idxn++; - } - } - - /* Loop over the bounds identified for tightening and perform update */ - ix = 0; - while(ix < idxn) { - jjx = idxbound[ix]; - jx = abs(jjx); - - /* Skip free variables and non-ints, if specified */ - if(is_unbounded(lp, jx) || - (intsonly && !is_int(lp, jx))) - continue; - - VARlo = get_lowbo(lp, jx); - VARup = get_upbo(lp, jx); - /* while((ix < idxn) && (jx == abs(jjx))) { */ - while((ix < idxn) && (jx == abs((jjx = idxbound[ix])))) { - if(jjx < 0) - VARlo = newbound[ix]; - else - VARup = newbound[ix]; - ix++; - } - if(!presolve_coltighten(psdata, jx, VARlo, VARup, tally)) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - } - - FREE(newbound); - FREE(idxbound); - - return(status); -} - -STATIC void set_dv_bounds(presolverec *psdata, int rownr, REAL lowbo, REAL upbo) -{ - psdata->dv_lobo[rownr] = lowbo; - psdata->dv_upbo[rownr] = upbo; -} -STATIC REAL get_dv_lower(presolverec *psdata, int rownr) -{ - return( psdata->dv_lobo[rownr] ); -} - -STATIC REAL get_dv_upper(presolverec *psdata, int rownr) -{ - return( psdata->dv_upbo[rownr] ); -} - -STATIC MYBOOL presolve_rowfix(presolverec *psdata, int rownr, REAL newvalue, MYBOOL remove, int *tally) -{ - lprec *lp = psdata->lp; - int i, ix, ie; - MYBOOL isneg, lofinite, upfinite, doupdate = FALSE, chsign = is_chsign(lp, rownr); - REAL lobound, upbound, lovalue, upvalue, - Value, fixvalue, fixprod, mult; - MATrec *mat = lp->matA; - psrec *ps = psdata->cols; - - /* Set "fixed" value in case we are deleting a variable */ - upbound = get_dv_upper(psdata, rownr); - lobound = get_dv_lower(psdata, rownr); - if(remove) { - if(upbound-lobound < psdata->epsvalue) { - if((newvalue > lobound) && (newvalue < upbound)) - fixvalue = newvalue; - else - fixvalue = lobound; - } - else { - if(my_infinite(lp, newvalue) && (get_rh(lp, rownr) == 0)) - fixvalue = ((lobound <= 0) && (upbound >= 0) ? 0 : MIN(upbound, lobound)); - else - fixvalue = newvalue; - } - set_dv_bounds(psdata, rownr, fixvalue, fixvalue); - if(fixvalue != 0) - addUndoPresolve(lp, FALSE, rownr, fixvalue, 0, 0); - mult = -1; - } - else { - mult = 1; - fixvalue = 0; - } - - /* Loop over rows to update statistics */ - ix = mat->row_end[rownr - 1]; - ie = mat->row_end[rownr]; - for(; ix < ie; ix++) { - - /* Retrieve row data and adjust RHS if we are deleting a variable */ - i = ROW_MAT_COLNR(ix); - Value = ROW_MAT_VALUE(ix); - if(Value == 0) - continue; - - if(remove && (fixvalue != 0)) { - fixprod = Value*fixvalue; - lp->orig_obj[i] -= fixprod; - my_roundzero(lp->orig_obj[i], psdata->epsvalue); - lp->presolve_undo->fixed_obj[i] += fixprod; - } - - /* Prepare for further processing */ - Value = my_chsign(chsign, Value); - isneg = (MYBOOL) (Value < 0); - - /* Reduce row variable counts if we are removing the variable */ - if(!isActiveLink(ps->varmap, i)) - continue; - if(remove) { - if(isneg) { - ps->negcount[i]--; - } - else { - ps->plucount[i]--; - } - if((lobound < 0) && (upbound >= 0)) { - ps->pluneg[i]--; - } - } - - /* Compute associated constraint contribution values */ - upfinite = (MYBOOL) (upbound < lp->infinite); - lofinite = (MYBOOL) (lobound > -lp->infinite); - if(upfinite || lofinite) { - if(remove) - ps->infcount[i]--; - else - ps->infcount[i]++; - } - upvalue = my_if(upfinite, Value*upbound, my_chsign(isneg, lp->infinite)); - lovalue = my_if(lofinite, Value*lobound, my_chsign(isneg, -lp->infinite)); - - /* Cumulate effective upper column bound (only bother with non-finite bound) */ - if(isneg) { - if((ps->negupper[i] < lp->infinite) && lofinite) { - ps->negupper[i] += mult*lovalue; - ps->negupper[i] = presolve_roundrhs(lp, ps->negupper[i], FALSE); - } - else if(remove && !lofinite) - doupdate = TRUE; - else - ps->negupper[i] = lp->infinite; - } - else { - if((ps->pluupper[i] < lp->infinite) && upfinite) { - ps->pluupper[i] += mult*upvalue; - ps->pluupper[i] = presolve_roundrhs(lp, ps->pluupper[i], FALSE); - } - else if(remove && !upfinite) - doupdate = TRUE; - else - ps->pluupper[i] = lp->infinite; - } - - /* Cumulate effective lower column bound (only bother with non-finite bound) */ - if(isneg) { - if((ps->neglower[i] > -lp->infinite) && upfinite) { - ps->neglower[i] += mult*upvalue; - ps->neglower[i] = presolve_roundrhs(lp, ps->neglower[i], TRUE); - } - else if(remove && !upfinite) - doupdate = TRUE; - else - ps->neglower[i] = -lp->infinite; - } - else { - if((ps->plulower[i] > -lp->infinite) && lofinite) { - ps->plulower[i] += mult*lovalue; - ps->plulower[i] = presolve_roundrhs(lp, ps->plulower[i], TRUE); - } - else if(remove && !lofinite) - doupdate = TRUE; - else - ps->plulower[i] = -lp->infinite; - } - - /* Validate consistency of eliminated singleton */ - if(remove && ((i == 0) || (ps->next[i][0] == 1)) && !psdata->forceupdate) { - presolve_range(lp, i, ps, &lovalue, &upvalue); - Value = get_mat(lp, 0, i); - if((upvalue < Value) || - (lovalue > Value)) { - report(lp, IMPORTANT, "presolve: Row %s (%g << %g) infeasibility in column %s (OF=%g)\n", - get_row_name(lp, rownr), lovalue, upvalue, get_col_name(lp, i), Value); - return( FALSE ); - } - } - } - if(remove) { - psdata->forceupdate |= doupdate; - if(tally != NULL) - (*tally)++; - } - return( TRUE ); -} - - -STATIC int presolve_colsingleton(presolverec *psdata, int i, int j, int *count) -{ - lprec *lp = psdata->lp; - REAL RHlow, RHup, LObound, UPbound, Value; - -#ifdef Paranoia - if(!isActiveLink(psdata->cols->varmap, j)) - report(lp, SEVERE, "presolve_colsingleton: Nothing to do, column %d was eliminated earlier\n", - j); -#endif - - Value = get_mat(lp,i,j); - if(Value == 0) - return( RUNNING ); - - /* Initialize and identify semicontinuous variable */ - LObound = get_lowbo(lp, j); - UPbound = get_upbo(lp, j); - if(is_semicont(lp, j) && (UPbound > LObound)) { - if(LObound > 0) - LObound = 0; - else if(UPbound < 0) - UPbound = 0; - } - - /* Get singleton variable bounds */ - RHlow = get_rh_lower(lp, i); - RHup = get_rh_upper(lp, i); - if(!presolve_singletonbounds(psdata, i,j, &RHlow, &RHup, &Value)) - return( presolve_setstatus(psdata, INFEASIBLE) ); - - if(presolve_coltighten(psdata, j, RHlow, RHup, count)) - return( RUNNING ); - else - return( presolve_setstatus(psdata, INFEASIBLE) ); -} - -STATIC MYBOOL presolve_colfix(presolverec *psdata, int colnr, REAL newvalue, MYBOOL remove, int *tally) -{ - lprec *lp = psdata->lp; - int i, ix, ie; - MYBOOL isneg, lofinite, upfinite, doupdate = FALSE, doOF = TRUE; - REAL lobound, upbound, lovalue, upvalue, - Value, fixvalue, mult; - MATrec *mat = lp->matA; - psrec *ps = psdata->rows; - REAL *value; - int *rownr; - - /* Set "fixed" value in case we are deleting a variable */ - upbound = get_upbo(lp, colnr); - lobound = get_lowbo(lp, colnr); - if(remove) { - if(upbound-lobound < psdata->epsvalue) { - if((newvalue > lobound) && (newvalue < upbound)) - fixvalue = newvalue; - else - fixvalue = lobound; - } - else { - if(my_infinite(lp, newvalue) && (get_mat(lp, 0, colnr) == 0)) - fixvalue = ((lobound <= 0) && (upbound >= 0) ? 0 : MIN(upbound, lobound)); - else - fixvalue = newvalue; - } -#if 1 /* Fast normal version */ - set_bounds(lp, colnr, fixvalue, fixvalue); -#else /* Slower version that can be used for debugging/control purposes */ - presolve_coltighten(psdata, colnr, fixvalue, fixvalue, NULL); - lobound = fixvalue; - upbound = fixvalue; -#endif - if(fixvalue != 0) - addUndoPresolve(lp, TRUE, colnr, fixvalue, 0, 0); - mult = -1; - } - else { - mult = 1; - fixvalue = 0; - } - - /* Adjust semi-continuous variable bounds to zero-base */ - if(is_semicont(lp, colnr) && (upbound > lobound)) { - if(lobound > 0) - lobound = 0; - else if(upbound < 0) - upbound = 0; - } - - /* Loop over rows to update statistics */ - ix = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - rownr = &COL_MAT_ROWNR(ix); - value = &COL_MAT_VALUE(ix); - for(; doOF || (ix < ie); - ix++, rownr += matRowColStep, value += matValueStep) { - - /* Retrieve row data and adjust RHS if we are deleting a variable */ -Restart: - if(doOF) { - i = 0; - Value = lp->orig_obj[colnr]; - } - else { - i = *rownr; - Value = *value; - if(!isActiveLink(ps->varmap, i)) - continue; - } - if(Value == 0) - goto BlockEnd; - - if(remove && (fixvalue != 0)) - presolve_adjustrhs(psdata, i, Value*fixvalue, psdata->epsvalue); - - /* Prepare for further processing */ - Value = my_chsign(is_chsign(lp, i), Value); - isneg = (MYBOOL) (Value < 0); - - /* Reduce row variable counts if we are removing the variable */ - if(remove == TRUE) { - if(isneg) { - ps->negcount[i]--; - } - else { - ps->plucount[i]--; - } - if((lobound < 0) && (upbound >= 0)) { - ps->pluneg[i]--; - } - } - - /* Compute associated constraint contribution values */ - upfinite = (MYBOOL) (upbound < lp->infinite); - lofinite = (MYBOOL) (lobound > -lp->infinite); - if(upfinite || lofinite) { - if(remove) - ps->infcount[i]--; - else - ps->infcount[i]++; - } - upvalue = my_if(upfinite, Value*upbound, my_chsign(isneg, lp->infinite)); - lovalue = my_if(lofinite, Value*lobound, my_chsign(isneg, -lp->infinite)); - - /* Cumulate effective upper row bound (only bother with non-finite bound) */ - if(isneg) { - if((ps->negupper[i] < lp->infinite) && lofinite) { - ps->negupper[i] += mult*lovalue; - ps->negupper[i] = presolve_roundrhs(lp, ps->negupper[i], FALSE); - } - else if(remove && !lofinite) - doupdate = TRUE; - else - ps->negupper[i] = lp->infinite; - } - else { - if((ps->pluupper[i] < lp->infinite) && upfinite) { - ps->pluupper[i] += mult*upvalue; - ps->pluupper[i] = presolve_roundrhs(lp, ps->pluupper[i], FALSE); - } - else if(remove && !upfinite) - doupdate = TRUE; - else - ps->pluupper[i] = lp->infinite; - } - - /* Cumulate effective lower row bound (only bother with non-finite bound) */ - if(isneg) { - if((ps->neglower[i] > -lp->infinite) && upfinite) { - ps->neglower[i] += mult*upvalue; - ps->neglower[i] = presolve_roundrhs(lp, ps->neglower[i], TRUE); - } - else if(remove && !upfinite) - doupdate = TRUE; - else - ps->neglower[i] = -lp->infinite; - } - else { - if((ps->plulower[i] > -lp->infinite) && lofinite) { - ps->plulower[i] += mult*lovalue; - ps->plulower[i] = presolve_roundrhs(lp, ps->plulower[i], TRUE); - } - else if(remove && !lofinite) - doupdate = TRUE; - else - ps->plulower[i] = -lp->infinite; - } - - /* Validate consistency of eliminated singleton */ - if(remove && ((i == 0) || (ps->next[i][0] == 1)) && !psdata->forceupdate) { - if(i == 0) { - lovalue = get_rh_lower(lp, i); - upvalue = get_rh_upper(lp, i); - report(lp, DETAILED, "presolve_colfix: Objective determined by presolve as %18g\n", - (is_maxim(lp) ? upvalue : lovalue)); - } - else { - presolve_range(lp, i, ps, &lovalue, &upvalue); -#if 1 - Value = 0; -#else - Value = MAX(fabs(upvalue), fabs(lovalue)); - Value = psdata->epsvalue * MAX(1, Value); -#endif - if((upvalue < get_rh_lower(lp, i)-Value) || - (lovalue > get_rh_upper(lp, i)+Value)) { - report(lp, NORMAL, "presolve_colfix: Variable %s (%g << %g) infeasibility in row %s (%g << %g)\n", - get_col_name(lp, colnr), lovalue, upvalue, - get_row_name(lp, i), get_rh_lower(lp,i), get_rh_upper(lp, i)); - return( FALSE ); - } - } - } -BlockEnd: - if(doOF) { - doOF = FALSE; - if(ix < ie) - goto Restart; - } - - } - if(remove) { - psdata->forceupdate |= doupdate; - if(tally != NULL) - (*tally)++; - } - return( TRUE ); -} - -/* Delete the columns of the specified row, but make sure we don't delete SOS variables. - Note that we cannot use presolve_nextcol() here, since the variables are deleted. */ -STATIC int presolve_rowfixzero(presolverec *psdata, int rownr, int *nv) -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - int ix, jx, ib = mat->row_end[rownr-1]; - for(ix = mat->row_end[rownr]-1; ix >= ib; ix--) { - jx = ROW_MAT_COLNR(ix); - if(isActiveLink(psdata->cols->varmap, jx)) { - if(!presolve_colfix(psdata, jx, 0.0, TRUE, nv)) - return( presolve_setstatus(psdata, INFEASIBLE) ); - if(presolve_candeletevar(psdata, jx)) - presolve_colremove(psdata, jx, TRUE); - } - } -#ifdef xxParanoia - if(!presolve_debugrowtallies(psdata)) - return( INFEASIBLE ); -#endif - return( RUNNING ); -} - -/* Function to find if a variable can be fixed based on considering the dual */ -STATIC MYBOOL presolve_colfixdual(presolverec *psdata, int colnr, REAL *fixValue, int *status) -{ - lprec *lp = psdata->lp; - MYBOOL hasOF, isMI, isDualFREE = TRUE; - int i, ix, ie, *rownr, signOF; - REAL *value, loX, upX, eps = psdata->epsvalue; - MATrec *mat = lp->matA; - - /* First check basic variable range */ - loX = get_lowbo(lp, colnr); - upX = get_upbo(lp, colnr); - if(((loX < 0) && (upX > 0)) || - (fabs(upX-loX) < lp->epsvalue) || - SOS_is_member_of_type(lp->SOS, colnr, SOSn)) - return( FALSE ); - isMI = (MYBOOL) (upX <= 0); - - /* Retrieve OF (standard form assuming maximization) */ - ix = mat->col_end[colnr - 1]; - ie = mat->col_end[colnr]; - rownr = &COL_MAT_ROWNR(ix); - value = &COL_MAT_VALUE(ix); - hasOF = isnz_origobj(lp, colnr); - if(hasOF) - signOF = my_sign(lp->orig_obj[colnr]); - else - signOF = 0; - - /* Loop over all constraints involving active variable (standard form with LE constraints)*/ - for(; (ix < ie) && isDualFREE; - ix++, rownr += matRowColStep, value += matValueStep) { - i = *rownr; - if(!isActiveLink(psdata->rows->varmap, i)) - continue; - if(presolve_rowlength(psdata, i) == 1) { - REAL val = my_chsign(is_chsign(lp, i), *value), - loR = get_rh_lower(lp, i), - upR = get_rh_upper(lp, i); - if(!presolve_singletonbounds(psdata, i, colnr, &loR, &upR, &val)) { - *status = presolve_setstatus(psdata, INFEASIBLE); - return( FALSE ); - } - if(loR > loX + psdata->epsvalue) - loX = presolve_roundrhs(lp, loR, TRUE); - if(upR < upX - psdata->epsvalue) - upX = presolve_roundrhs(lp, upR, FALSE); - continue; - } - else - isDualFREE = my_infinite(lp, get_rh_range(lp, i)) || /* Explicitly free */ - ((presolve_sumplumin(lp, i, psdata->rows, TRUE)-eps <= get_rh_upper(lp, i)) && /* Implicitly free */ - (presolve_sumplumin(lp, i, psdata->rows, FALSE)+eps >= get_rh_lower(lp, i))); - if(isDualFREE) { - if(signOF == 0) /* Test on the basis of identical signs in the constraints */ - signOF = my_sign(*value); - else /* Test on the basis of constraint sign equal to OF sign */ - isDualFREE = (MYBOOL) (signOF == my_sign(*value)); - } - } - - /* Set fixing value if we were successful */ - if(isDualFREE) { - if(signOF == 0) { - SETMAX(loX, 0); - *fixValue = MIN(loX, upX); - } - else if(signOF > 0) { - if(my_infinite(lp, loX)) - isDualFREE = FALSE; - else { - if(is_int(lp, colnr)) - *fixValue = ceil(loX-PRESOLVE_EPSVALUE); - else - *fixValue = loX; - } - } - else { - if(my_infinite(lp, upX)) - isDualFREE = FALSE; - else { - if(is_int(lp, colnr) && (upX != 0)) - *fixValue = floor(upX+PRESOLVE_EPSVALUE); - else - *fixValue = upX; - } - } - if((*fixValue != 0) && SOS_is_member(lp->SOS, 0, colnr)) - return( FALSE ); - - } - - return( isDualFREE ); -} - -#if 0 -STATIC MYBOOL presolve_probefix01(presolverec *psdata, int colnr, REAL *fixvalue) -{ - lprec *lp = psdata->lp; - int i, ix, item; - REAL loLim, absvalue, epsvalue = psdata->epsvalue; - MATrec *mat = lp->matA; - MYBOOL chsign, canfix = FALSE; - - if(!is_binary(lp, colnr)) - return( canfix ); - - /* Loop over all active rows to search for fixing opportunity */ - item = 0; - for(ix = presolve_nextrow(psdata, colnr, &item); - (ix >= 0) && !canfix; - ix = presolve_nextrow(psdata, colnr, &item)) { - i = COL_MAT_ROWNR(ix); - *fixvalue = COL_MAT_VALUE(ix); - chsign = is_chsign(lp, i); - - /* First check the lower bound of the normalized constraint */ - loLim = presolve_sumplumin(lp, i, psdata->rows, chsign); - loLim = my_chsign(chsign, loLim); - absvalue = fabs(*fixvalue); - canfix = (MYBOOL) ((loLim + absvalue > lp->orig_rhs[i]+epsvalue*MAX(1, absvalue))); - - /* If we were unsuccessful in fixing above, try the upper bound - of the normalized constraint - if it is finite */ - if(!canfix && !my_infinite(lp, get_rh_range(lp, i))) { - loLim = presolve_sumplumin(lp, i, psdata->rows, (MYBOOL) !chsign); - loLim = my_chsign(!chsign, loLim); - *fixvalue = -(*fixvalue); - canfix = (MYBOOL) ((loLim + absvalue > get_rh_range(lp, i)-lp->orig_rhs[i]+epsvalue*MAX(1, absvalue))); - } - } - - /* Check if we were successful in identifying fixing opportunity */ - if(canfix) { - if(*fixvalue < 0) - *fixvalue = 1; - else - *fixvalue = 0; - } - return( canfix ); -} -#else -STATIC MYBOOL presolve_probefix01(presolverec *psdata, int colnr, REAL *fixvalue) -{ - lprec *lp = psdata->lp; - int i, ix, item; - REAL loLim, upLim, range, absvalue, epsvalue = psdata->epsvalue, tolgap; - MATrec *mat = lp->matA; - MYBOOL chsign, status = FALSE; - - if(!is_binary(lp, colnr)) - return( status ); - - /* Loop over all active rows to search for fixing opportunity. The logic is that if a - constraint gets violated by setting a variable at one of its bounds, then it can be - fixed at its opposite bound. */ - item = 0; - - for(ix = presolve_nextrow(psdata, colnr, &item); (ix >= 0); ix = presolve_nextrow(psdata, colnr, &item)) { - i = COL_MAT_ROWNR(ix); - *fixvalue = COL_MAT_VALUE(ix); - absvalue = fabs(*fixvalue); - SETMIN(absvalue, 100); - tolgap = epsvalue*MAX(1, absvalue); - chsign = is_chsign(lp, i); - - /* Get the constraint value limits based on variable bounds, normalized to LE constraint */ - loLim = presolve_sumplumin(lp, i, psdata->rows, FALSE); - upLim = presolve_sumplumin(lp, i, psdata->rows, TRUE); - if(chsign) { - loLim = my_chsign(chsign, loLim); - upLim = my_chsign(chsign, upLim); - swapREAL(&loLim, &upLim); - } - - /* Check the upper constraint bound for possible violation if the value were to be fixed at 1 */ - if(loLim + *fixvalue > lp->orig_rhs[i]+tolgap) { - if(*fixvalue < 0) - presolve_setstatus(psdata, INFEASIBLE); - *fixvalue = 0; - break; - } - - /* Check the lower constraint bound for possible violation if the value were to be fixed at 1 */ - range = get_rh_range(lp, i); - if(!my_infinite(lp, range) && - (upLim + *fixvalue < lp->orig_rhs[i]-range-tolgap)) { - if(*fixvalue > 0) - presolve_setstatus(psdata, INFEASIBLE); - *fixvalue = 0; - break; - } - - /* Check if we have to fix the value at 1 to avoid constraint infeasibility */ - if(psdata->rows->infcount[i] >= 1) - continue; - if(((*fixvalue < 0) && (upLim + *fixvalue >= loLim-tolgap) && (upLim > lp->orig_rhs[i]+tolgap)) || - ((*fixvalue > 0) && (loLim + *fixvalue <= upLim+tolgap) && (loLim < lp->orig_rhs[i]-range-tolgap) && !my_infinite(lp, range))) { - *fixvalue = 1; - break; - } - } - status = (MYBOOL) (ix >= 0); - - /* Returns TRUE if fixing opportunity was identified */ - return( status ); -} -#endif - -STATIC int presolve_probetighten01(presolverec *psdata, int colnr) -{ - lprec *lp = psdata->lp; - MYBOOL chsign; - int i, ix, item, n = 0; - REAL upLim, value, absvalue, epsvalue = psdata->epsvalue; - MATrec *mat = lp->matA; - -#if 0 /* Handled in calling routine */ - if(!is_binary(lp, colnr)) - return( n ); -#endif - - /* Loop over all active rows and do coefficient tightening for qualifying constraints */ - item = 0; - for(ix = presolve_nextrow(psdata, colnr, &item); ix >= 0; - ix = presolve_nextrow(psdata, colnr, &item)) { - i = COL_MAT_ROWNR(ix); - value = COL_MAT_VALUE(ix); - chsign = is_chsign(lp, i); - upLim = presolve_sumplumin(lp, i, psdata->rows, (MYBOOL) !chsign); - upLim = my_chsign(chsign, upLim); - - /* Does this constraint qualify for coefficient tightening? */ - absvalue = fabs(value); - if(upLim - absvalue < lp->orig_rhs[i]-epsvalue*MAX(1, absvalue)) { - REAL delta = lp->orig_rhs[i] - upLim; - lp->orig_rhs[i] = upLim; - upLim = value - my_chsign(value < 0, delta); - COL_MAT_VALUE(ix) = upLim; - if(my_sign(value) != my_sign(upLim)) { - if(chsign) { - psdata->rows->negcount[i]--; - psdata->rows->plucount[i]++; - } - else { - psdata->rows->negcount[i]++; - psdata->rows->plucount[i]--; - } - } - n++; - } - } - return( n ); -} - -STATIC int presolve_mergerows(presolverec *psdata, int *nRows, int *nSum) -{ - lprec *lp = psdata->lp; - MYBOOL candelete; - int status = RUNNING, item1, item2, - firstix, RT1, RT2, i, ix, iix, j, jjx, n = 0; - REAL Value1, Value2, bound; - MATrec *mat = lp->matA; - - for(i = lastActiveLink(psdata->rows->varmap); (i > 0) && (status == RUNNING); ) { - - /* First scan for rows with identical row lengths */ - ix = prevActiveLink(psdata->rows->varmap, i); - if(ix == 0) - break; - - /* Don't bother about empty rows or row singletons, since they are - handled by PRESOLVE_ROWS */ - j = presolve_rowlength(psdata, i); - if(j <= 1) { - i = ix; - continue; - } - -#if 0 - /* Enable this to scan all rows back */ - RT2 = lp->rows; - - /* Check abort since this section can be pretty "expensive" */ - if(!presolve_statuscheck(psdata, &status)) - return( status ); -#else - RT2 = 2+1; -#endif - firstix = ix; - for(RT1 = 0; (ix > 0) && (RT1 < RT2) && (status == RUNNING); - ix = prevActiveLink(psdata->rows->varmap, ix), RT1++) { - candelete = FALSE; - if(presolve_rowlength(psdata, ix) != j) - continue; - - /* Check if the beginning columns are identical; if not, continue */ - item1 = 0; - iix = presolve_nextcol(psdata, ix, &item1); - item2 = 0; - jjx = presolve_nextcol(psdata, i, &item2); - - if(ROW_MAT_COLNR(iix) != ROW_MAT_COLNR(jjx)) - continue; - - /* We have a candidate row; check if the entries have a fixed non-zero ratio */ - Value1 = get_mat_byindex(lp, iix, TRUE, FALSE); - Value2 = get_mat_byindex(lp, jjx, TRUE, FALSE); - bound = Value1 / Value2; - Value1 = bound; - - /* Loop over remaining entries */ - jjx = presolve_nextcol(psdata, i, &item2); - for(; (jjx >= 0) && (Value1 == bound); - jjx = presolve_nextcol(psdata, i, &item2)) { - iix = presolve_nextcol(psdata, ix, &item1); - if(ROW_MAT_COLNR(iix) != ROW_MAT_COLNR(jjx)) - break; - Value1 = get_mat_byindex(lp, iix, TRUE, FALSE); - Value2 = get_mat_byindex(lp, jjx, TRUE, FALSE); - - /* If the ratio is different from the reference value we have a mismatch */ - Value1 = Value1 / Value2; - if(bound == lp->infinite) - bound = Value1; - else if(fabs(Value1 - bound) > psdata->epsvalue) - break; - } - - /* Check if we found a match (we traversed all active columns without a break) */ - if(jjx < 0) { - - /* Get main reference values */ - Value1 = lp->orig_rhs[ix]; - Value2 = lp->orig_rhs[i] * bound; - - /* First check for inconsistent equalities */ - if((fabs(Value1 - Value2) > psdata->epsvalue) && - ((get_constr_type(lp, ix) == EQ) && (get_constr_type(lp, i) == EQ))) { - report(lp, NORMAL, "presolve_mergerows: Inconsistent equalities %d and %d found\n", - ix, i); - status = presolve_setstatus(psdata, INFEASIBLE); - } - - else { - - /* Update lower and upper bounds */ - if(is_chsign(lp, i) != is_chsign(lp, ix)) - bound = -bound; - - Value1 = get_rh_lower(lp, i); - if(Value1 <= -lp->infinite) - Value1 *= my_sign(bound); - else - Value1 *= bound; - my_roundzero(Value1, lp->epsdual); /* Extra rounding tolerance *** */ - - Value2 = get_rh_upper(lp, i); - if(Value2 >= lp->infinite) - Value2 *= my_sign(bound); - else - Value2 *= bound; - my_roundzero(Value2, lp->epsdual); /* Extra rounding tolerance *** */ - - if((bound < 0)) - swapREAL(&Value1, &Value2); - - bound = get_rh_lower(lp, ix); - if(Value1 > bound + psdata->epsvalue) - set_rh_lower(lp, ix, Value1); - else - Value1 = bound; - bound = get_rh_upper(lp, ix); - if(Value2 < bound - psdata->epsvalue) - set_rh_upper(lp, ix, Value2); - else - Value2 = bound; - - /* Check results and make equality if appropriate */ - if(fabs(Value2-Value1) < psdata->epsvalue) - presolve_setEQ(psdata, ix); - else if(Value2 < Value1) { - status = presolve_setstatus(psdata, INFEASIBLE); - } - - /* Verify if we can continue */ - candelete = (MYBOOL) (status == RUNNING); - if(!candelete) { - report(lp, NORMAL, "presolve: Range infeasibility found involving rows %s and %s\n", - get_row_name(lp, ix), get_row_name(lp, i)); - } - } - } - /* Perform i-row deletion if authorized */ - if(candelete) { - presolve_rowremove(psdata, i, TRUE); - n++; - break; - } - } - i = firstix; - } - (*nRows) += n; - (*nSum) += n; - - return( status ); -} - -STATIC MYBOOL presolve_reduceGCD(presolverec *psdata, int *nn, int *nb, int *nsum) -{ - lprec *lp = psdata->lp; - MYBOOL status = TRUE; - int i, jx, je, in = 0, ib = 0; - LLONG GCDvalue; - REAL *Avalue, Rvalue, epsvalue = psdata->epsvalue; - MATrec *mat = lp->matA; - - for(i = firstActiveLink(psdata->INTmap); i != 0; i = nextActiveLink(psdata->INTmap, i)) { - - /* Obtain the row GCD */ - jx = mat->row_end[i - 1]; - je = mat->row_end[i]; - Rvalue = ROW_MAT_VALUE(jx); - GCDvalue = abs((int) Rvalue); - jx++; - if(jx < je) - for(; (jx < je) && (GCDvalue > 1); jx++) { - Rvalue = fabs(ROW_MAT_VALUE(jx)); - GCDvalue = gcd((LLONG) Rvalue, GCDvalue, NULL, NULL); - } - - /* Reduce the coefficients, if possible */ - if(GCDvalue > 1) { - jx = mat->row_end[i - 1]; - je = mat->row_end[i]; - for(; jx < je; jx++) { - Avalue = &ROW_MAT_VALUE(jx); - *Avalue /= GCDvalue; - in++; - } - Rvalue = (lp->orig_rhs[i] / GCDvalue) + epsvalue; - lp->orig_rhs[i] = floor(Rvalue); - Rvalue = fabs(lp->orig_rhs[i]-Rvalue); - if(is_constr_type(lp, i, EQ) && (Rvalue > epsvalue)) { - report(lp, NORMAL, "presolve_reduceGCD: Infeasible equality constraint %d\n", i); - status = FALSE; - break; - } - if(!my_infinite(lp, lp->orig_upbo[i])) - lp->orig_upbo[i] = floor(lp->orig_upbo[i] / GCDvalue); - ib++; - } - } - if(status && (in > 0)) - report(lp, DETAILED, "presolve_reduceGCD: Did %d constraint coefficient reductions.\n", in); - - (*nn) += in; - (*nb) += ib; - (*nsum) += in + ib; - - return( status ); -} - -STATIC int presolve_knapsack(presolverec *psdata, int *nn) -{ - lprec *lp = psdata->lp; - int m, n, i, ix, j, jx, colnr, *rownr = NULL, - status = RUNNING; - REAL *colOF = lp->orig_obj, value, *ratio = NULL; - LLrec *map = psdata->EQmap; - MATrec *mat = lp->matA; - - /* Check if it is worth trying */ - m = mat->row_end[0]; - if((map->count == 0) || (m < 2)) - return( status ); - - /* Get the OF row */ - allocINT(lp, &rownr, map->count+1, FALSE); - allocREAL(lp, &ratio, map->count+1, FALSE); - - /* Loop over each row trying to find equal entries in the OF */ - rownr[0] = 0; - for(i = firstActiveLink(map); i != 0; i = nextActiveLink(map, i)) { - if(get_rh(lp, i) <= 0) - continue; - jx = mat->row_end[i]; - n = 0; - for(j = mat->row_end[i-1]; j < jx; j++, n++) { - colnr = ROW_MAT_COLNR(j); - value = ROW_MAT_VALUE(j); - if(colOF[colnr] == 0) - break; - if(n == 0) { - ratio[0] = colOF[colnr] / value; - } - else if(fabs(value * ratio[0] - colOF[colnr]) > psdata->epsvalue) { - n = -1; - break; - } - } - /* Register row if we were successful (and row long enough) */ - if(n >= 2) { - ix = ++rownr[0]; - rownr[ix] = i; - ratio[ix] = ratio[0]; - } - } - n = rownr[0]; - if(n == 0) - goto Finish; - - /* Process the identified rows, eliminating the OF value */ - for(ix = 1; ix <= n; ix++) { - i = rownr[ix]; - jx = mat->row_end[i]; - for(j = mat->row_end[i-1]; j < jx; j++) { - colnr = ROW_MAT_COLNR(j); - colOF[colnr] = 0; - } - } - - /* Update key mapper structures */ - j = lp->columns; - psdata->cols->varmap = cloneLink(psdata->cols->varmap, j+n, TRUE); - psdata->forceupdate = TRUE; - - /* Finally, add helper columns */ - for(ix = 1; ix <= n; ix++) { - i = rownr[ix]; - rownr[0] = 0; - colOF[0] = my_chsign(is_maxim(lp), ratio[ix]); - rownr[1] = i; - colOF[1] = -1; - value = get_rh(lp, i); -/* j = get_constr_type(lp, i); */ - add_columnex(lp, 2, colOF, rownr); - set_bounds(lp, lp->columns, value, value); -/* presolve_setEQ(psdata, i); */ - set_rh(lp, i, 0); - appendLink(psdata->cols->varmap, j+ix); - } - presolve_validate(psdata, TRUE); - - /* Clean up before returning */ -Finish: - FREE(rownr); - FREE(ratio); - (*nn) += n; - - return( status ); -} - -STATIC MYBOOL presolve_invalideq2(lprec *lp, presolverec *psdata) -{ - int jx, jjx, i = 0, item; - MATrec *mat = lp->matA; - MYBOOL error = FALSE; - - do { - - if(i == 0) - i = firstActiveLink(psdata->EQmap); - else - i = nextActiveLink(psdata->EQmap, i); - if(i == 0) - return( error ); - - /* Get the row index of the first 2-element equality */ - for(; i > 0; i = nextActiveLink(psdata->EQmap, i)) - if(presolve_rowlength(psdata, i) == 2) - break; - if(i == 0) - return( error ); - - /* Get the first column */ - item = 0; - jx = presolve_nextcol(psdata, i, &item); - if(jx < 0) - error = TRUE; - jx = ROW_MAT_COLNR(jx); - - /* Get the second column */ - jjx = presolve_nextcol(psdata, i, &item); - if(jjx < 0) - error = AUTOMATIC; - } while(!error); - - return( error ); -} - -/* Callback to obtain the non-zero rows of equality constraints */ -int BFP_CALLMODEL presolve_getcolumnEQ(lprec *lp, int colnr, REAL nzvalues[], int nzrows[], int mapin[]) -{ - int i, ib, ie, nn = 0; - MATrec *mat = lp->matA; - - ib = mat->col_end[colnr-1]; - ie = mat->col_end[colnr]; - for(; ib < ie; ib++) { - i = COL_MAT_ROWNR(ib); - if(!is_constr_type(lp, i, EQ) || /* It has to be an equality constraint */ - (mapin[i] == 0)) /* And it should not already have been deleted */ - continue; - if(nzvalues != NULL) { - nzrows[nn] = mapin[i]; - nzvalues[nn] = COL_MAT_VALUE(ib); - } - nn++; - } - return( nn ); -} -STATIC int presolve_singularities(presolverec *psdata, int *nn, int *nr, int *nv, int *nSum) -{ - lprec *lp = psdata->lp; - int i, j, n, *rmapin = NULL, *rmapout = NULL, *cmapout = NULL; - - if(lp->bfp_findredundant(lp, 0, NULL, NULL, NULL) == 0) - return( 0 ); - - /* Create condensed row map */ - allocINT(lp, &rmapin, lp->rows+1, TRUE); - allocINT(lp, &rmapout, psdata->EQmap->count+1, FALSE); - allocINT(lp, &cmapout, lp->columns+1, FALSE); - n = 0; - for(i = firstActiveLink(psdata->EQmap); i != 0; i = nextActiveLink(psdata->EQmap, i)) { - n++; - rmapout[n] = i; - rmapin[i] = n; - } - rmapout[0] = n; - n = 0; - for(i = firstActiveLink(psdata->cols->varmap); i != 0; i = nextActiveLink(psdata->cols->varmap, i)) { - n++; - cmapout[n] = i; - } - cmapout[0] = n; - - /* Do the rank-revealing factorization */ - n = lp->bfp_findredundant(lp, psdata->EQmap->count, presolve_getcolumnEQ, rmapin, cmapout); - - /* Delete the redundant rows */ - for(i = 1; i <= n; i++) { - j = rmapin[i]; - j = rmapout[j]; - presolve_rowremove(psdata, j, TRUE); - } - (*nn) += n; - (*nr) += n; - (*nSum) += n; - - /* Clean up */ - FREE(rmapout); - FREE(rmapin); - FREE(cmapout); - - return( n ); -} - -STATIC int presolve_elimeq2(presolverec *psdata, int *nn, int *nr, int *nc, int *nSum) -{ - lprec *lp = psdata->lp; - int n, i, jx, jjx, k, item, *plucount, *negcount, colplu, colneg, - iCoeffChanged = 0, iRowsRemoved = 0, iVarsFixed = 0, nrows = lp->rows, - status = RUNNING, *colindex = NULL; - MYBOOL freshupdate; - REAL Coeff1, Coeff2, Value1, Value2, lobound, upbound, bound, test, product, - *colvalue = NULL, *delvalue = NULL, *colitem; - MATrec *mat = lp->matA, *rev = NULL; - DeltaVrec *DV = NULL; - LLrec *EQ2 = NULL; - - /* See if there is anything to do */ - if(psdata->EQmap->count == 0) { - (*nSum) = 0; - return( status ); - } - - /* Tally counts */ - createLink(lp->rows, &EQ2, NULL); - if((EQ2 == NULL) || !allocREAL(lp, &colvalue, nrows+1, FALSE) || - !allocREAL(lp, &delvalue, nrows+1, FALSE)) - goto Finish; - for(i = firstActiveLink(psdata->EQmap); i > 0; i = nextActiveLink(psdata->EQmap, i)) { - if(presolve_rowlength(psdata, i) == 2) - appendLink(EQ2, i); - } - if(EQ2->count == 0) - goto Finish; - n = 0; - - /* Do the elimination loop for all identified 2-element equalities */ - for(i = firstActiveLink(EQ2); i > 0; i = nextActiveLink(EQ2, i)) { - - /* Check if the constraint has been modified by a previous elimination */ - if(presolve_rowlength(psdata, i) != 2) - continue; - - /* Get the column indeces of NZ-values of the "pivot" row */ - item = 0; - jx = presolve_nextcol(psdata, i, &item); /* Eliminated variable coefficient b */ -#ifdef Paranoia - if(jx < 0) - report(lp, SEVERE, "presolve_elimeq2: No qualifying %dst column was found in row %s (ostensible length %d)\n", - 1, get_row_name(lp, i), presolve_rowlength(psdata, i)); -#endif - Coeff2 = ROW_MAT_VALUE(jx); - jx = ROW_MAT_COLNR(jx); - jjx = presolve_nextcol(psdata, i, &item); /* Non-eliminated variable coefficient a */ -#ifdef Paranoia - if(jjx < 0) - report(lp, SEVERE, "presolve_elimeq2: No qualifying %dnd column was found in row %s (ostensible length %d)\n", - 2, get_row_name(lp, i), presolve_rowlength(psdata, i)); -#endif - Coeff1 = ROW_MAT_VALUE(jjx); - jjx = ROW_MAT_COLNR(jjx); - - /* Check if at least one of the coefficients is large enough to preserve stability; - use opposing maximum column values for stability testing. */ - if((fabs(Coeff1) < psdata->epspivot*mat->colmax[jx]) && - ((fabs(Coeff1) != 1) && (fabs(Coeff2) != 1)) && - (fabs(Coeff2) < psdata->epspivot*mat->colmax[jjx])) - continue; - - /* Cannot eliminate a variable if both are SOS members or SC variables */ - if((is_semicont(lp, jx) && is_semicont(lp, jjx)) || - (SOS_is_member(lp->SOS, 0, jx) && SOS_is_member(lp->SOS, 0, jjx))) - continue; - - /* First check if we are allowed to swap; set swap "blockers" */ - k = 0; - if(!is_int(lp, jx) && is_int(lp, jjx)) - k += 1; - else if(!is_semicont(lp, jx) && is_semicont(lp, jjx)) - k += 2; - else if(!SOS_is_member(lp->SOS, 0, jx) && SOS_is_member(lp->SOS, 0, jjx)) - k += 4; - - /* If there were no blockers, determine if we MUST swap the variable to be eliminated */ - if(k == 0) { - if(is_int(lp, jx) && !is_int(lp, jjx)) - k += 8; - else if(is_semicont(lp, jx) && !is_semicont(lp, jjx)) - k += 16; - else if(SOS_is_member(lp->SOS, 0, jx) && !SOS_is_member(lp->SOS, 0, jjx)) - k += 32; - - /* If we are not forced to swap, decide if it otherwise makes sense - high order */ - if(k == 0) { - if((fabs(Coeff2) < psdata->epspivot*mat->colmax[jjx]) && - (fabs(Coeff1) > psdata->epspivot*mat->colmax[jx])) - k += 64; - else if(presolve_collength(psdata, jx) > presolve_collength(psdata, jjx)) - k += 128; - } - - /* If we are not forced to swap, decide if it otherwise makes sense - low order */ - if(k == 0) { - Value2 = Coeff1/Coeff2; -#ifdef DualFeasibilityLogicEQ2 - if((Value2*lp->orig_obj[jx] < 0) && - (Value2*lp->orig_obj[jjx] > 0)) /* Seek increased dual feasibility */ - k += 256; -#endif -#ifdef DivisorIntegralityLogicEQ2 - if((fabs(modf(Coeff2, &Value2)) >= lp->epsvalue) && /* Seek integrality of result */ - (fabs(modf(Coeff1, &Value2)) < lp->epsvalue)) - k += 512; - else if((fabs(fabs(Coeff2)-1) >= lp->epsvalue) && /* Seek integrality of divisor */ - (fabs(fabs(Coeff1)-1) < lp->epsvalue)) - k += 1024; -#endif - } - - } - else - k = 0; - - /* Perform variable index swap if indicated */ - if(k != 0) { - swapINT(&jx, &jjx); - swapREAL(&Coeff1, &Coeff2); - } - - Value1 = lp->orig_rhs[i]/Coeff2; /* Delta constant term */ - Value2 = Coeff1/Coeff2; /* Delta variable term */ - upbound = lp->orig_upbo[lp->rows+jx]; - lobound = lp->orig_lowbo[lp->rows+jx]; - if(lp->spx_trace) { - report(lp, DETAILED, "Row %3d : Elim %g %s - %d\n", i, Coeff2, get_col_name(lp, jx), jx); - report(lp, DETAILED, " Keep %g %s - %d\n", Coeff1, get_col_name(lp, jjx), jjx); - } - - /* Get the coefficient vectors of the independent (jjx) and dependent (jx) columns; - the dependent column will be deleted and reconstructed during postsolve. */ - freshupdate = (MYBOOL) ((colindex == NULL) || (colindex[jjx] == 0)); - if(freshupdate) - mat_expandcolumn(mat, jjx, colvalue, NULL, TRUE); - else - mat_expandcolumn(rev, colindex[jjx], colvalue, NULL, FALSE); - if((colindex == NULL) || (colindex[jx] == 0)) - mat_expandcolumn(mat, jx, delvalue, NULL, TRUE); - else - mat_expandcolumn(rev, colindex[jx], delvalue, NULL, FALSE); - - /* Add variable reconstruction information */ - addUndoPresolve(lp, TRUE, jx, Value1, Value2, jjx); - - /* If possible, tighten the bounds of the uneliminated variable based - on the bounds of the eliminated variable. Also handle roundings - and attempt precision management. */ - bound = lobound; - k = nrows+jjx; - if(bound > -lp->infinite) { - bound = (lp->orig_rhs[i] - Coeff2*bound) / Coeff1; - if(Value2 > 0) { - test = lp->orig_upbo[k]; - if(bound < test - psdata->epsvalue) { - if(is_int(lp, jjx)) - lp->orig_upbo[k] = floor(bound + lp->epsint); - else - lp->orig_upbo[k] = presolve_roundrhs(lp, bound, FALSE); - } - } - else { - test = lp->orig_lowbo[k]; - if(bound > test + psdata->epsvalue) { - if(is_int(lp, jjx)) - lp->orig_lowbo[k] = ceil(bound - lp->epsint); - else - lp->orig_lowbo[k] = presolve_roundrhs(lp, bound, TRUE); - } - } - } - bound = upbound; - if(bound < lp->infinite) { - bound = (lp->orig_rhs[i] - Coeff2*bound) / Coeff1; - if(Value2 < 0) { - test = lp->orig_upbo[k]; - if(bound < test - psdata->epsvalue) { - if(is_int(lp, jjx)) - lp->orig_upbo[k] = floor(bound + lp->epsint); - else - lp->orig_upbo[k] = presolve_roundrhs(lp, bound, FALSE); - } - } - else { - test = lp->orig_lowbo[k]; - if(bound > test + psdata->epsvalue) { - if(is_int(lp, jjx)) - lp->orig_lowbo[k] = ceil(bound - lp->epsint); - else - lp->orig_lowbo[k] = presolve_roundrhs(lp, bound, TRUE); - } - } - } - -#ifdef Eq2Reldiff - test = 2*lp->epsvalue; -#else - test = psdata->epsvalue; -#endif - if(/*(lp->orig_upbo[k] < lp->orig_lowbo[k]) ||*/ -#ifdef Eq2Reldiff - (fabs(my_reldiff(lp->orig_upbo[k],lp->orig_lowbo[k])) < test)) { -#else - (fabs(lp->orig_upbo[k] - lp->orig_lowbo[k]) < test)) { -#endif - my_roundzero(lp->orig_lowbo[k], test); - lp->orig_upbo[k] = lp->orig_lowbo[k]; - } - else { - my_roundzero(lp->orig_upbo[k], test); - my_roundzero(lp->orig_lowbo[k], test); - } - - if(/*(upbound < lobound) ||*/ -#ifdef Eq2Reldiff - (fabs(my_reldiff(upbound, lobound)) < test)) { -#else - (fabs(upbound - lobound) < test)) { -#endif - my_roundzero(lobound, test); - lp->orig_upbo[nrows+jx] = lobound; - upbound = lobound; - } - - /* Loop over the non-zero rows of the column (jx) to be eliminated; - substitute jx-variable by updating rhs and jjx coefficients */ - colitem = colvalue; - plucount = psdata->rows->plucount; - negcount = psdata->rows->negcount; - colplu = 0; - colneg = 0; - /* Count of non-zeros in the independent column jjx */ - item = presolve_collength(psdata, jjx) - 1; - if(isnz_origobj(lp, jjx)) - item++; - for(k = 0; k <= nrows; k++, colitem++) { - - bound = delvalue[k]; - if((k == i) || (bound == 0) || - ((k > 0) && !isActiveLink(psdata->rows->varmap, k))) - continue; - - /* Do constraint and nz-count updates for the substituted variable */ - product = bound*Value1; - - /* "Raw"/unsigned data */ - presolve_adjustrhs(psdata, k, my_chsign(is_chsign(lp, k), product), test); - - /* Change back to signed part */ - if(*colitem != 0) { - if(*colitem > 0) { - colplu--; - plucount[k]--; - } - else { - colneg--; - negcount[k]--; - } - if((lobound < 0) && (upbound >= 0)) { - psdata->cols->pluneg[jjx]--; - psdata->rows->pluneg[k]--; - } - item--; - } - (*colitem) -= bound*Value2; - iCoeffChanged++; - - /* Update counts */ - if(fabs(*colitem) >= mat->epsvalue) { - if(*colitem > 0) { - colplu++; - plucount[k]++; - } - else { - colneg++; - negcount[k]++; - } - if((lobound < 0) && (upbound >= 0)) { - psdata->cols->pluneg[jjx]++; - psdata->rows->pluneg[k]++; - } - item++; - } - else { - *colitem = 0; - } - - /* Also reduce count if the row contains the deleted variable */ - if(bound > 0) - plucount[k]--; - else - negcount[k]--; - } - psdata->cols->plucount[jjx] += colplu; - psdata->cols->negcount[jjx] += colneg; - - /* Save the new column */ - if(rev == NULL) { - DV = createUndoLadder(lp, nrows, lp->columns / RESIZEFACTOR); - rev = DV->tracker; - rev->epsvalue = mat->epsvalue; - allocINT(lp, &(rev->col_tag), mat->columns_alloc+1, FALSE); - allocINT(lp, &colindex, lp->columns+1, TRUE); - rev->col_tag[0] = 0; - } - n = rev->col_tag[0] = incrementUndoLadder(DV); - mat_setcol(rev, n, 0, colvalue, NULL, FALSE, FALSE); - rev->col_tag[n] = jjx; - - /* Save index to updated vector, but specially handle case where we have - the same independent variable for multiple equations! */ - if(!freshupdate) - rev->col_tag[colindex[jjx]] *= -1; - colindex[jjx] = n; - - /* Delete the column dependent variable */ - jx = presolve_colremove(psdata, jx, FALSE); - iVarsFixed++; - - /* Check if we have been lucky enough to have eliminated the independent - variable via substitution of the dependent variable */ - if(item == 0) { -#ifdef Paranoia - report(lp, DETAILED, "presolve_elimeq2: Was able to remove variables %d and %d in row %s\n", - jx, jjx, get_row_name(lp, i)); -#endif - if(presolve_colfix(psdata, jjx, 0.0, TRUE, nc)) - jjx = presolve_colremove(psdata, jjx, FALSE); - } - - /* Delete the row */ - presolve_rowremove(psdata, i, FALSE); - iRowsRemoved++; - } - - /* Perform the column updates collected above */ - if(n > 0) { - mat_mapreplace(mat, psdata->rows->varmap, psdata->cols->varmap, rev); - presolve_validate(psdata, TRUE); -#ifdef PresolveForceUpdateMax - mat_computemax(mat /* , FALSE */); -#endif - psdata->forceupdate = TRUE; - } - - /* Free work arrays */ -Finish: - if(DV != NULL) - freeUndoLadder(&DV); - freeLink(&EQ2); - FREE(colvalue); - FREE(delvalue); - FREE(colindex); - - /* Update counters */ - (*nn) += iCoeffChanged; - (*nr) += iRowsRemoved; - (*nc) += iVarsFixed; - (*nSum) += iCoeffChanged + iRowsRemoved + iVarsFixed; - - return( status ); -} - -STATIC MYBOOL presolve_impliedfree(lprec *lp, presolverec *psdata, int colnr) -{ - int i, ix, ie; - REAL Tlower, Tupper; - MYBOOL status, rowbinds, isfree = FALSE; - MATrec *mat = lp->matA; - - if(my_infinite(lp, get_lowbo(lp, colnr)) && my_infinite(lp, get_upbo(lp, colnr))) - return( TRUE ); - - ie = mat->col_end[colnr]; - for(ix = mat->col_end[colnr-1]; (isfree != (TRUE | AUTOMATIC)) && (ix < ie); ix++) { - i = COL_MAT_ROWNR(ix); - if(!isActiveLink(psdata->rows->varmap, i)) - continue; - Tlower = get_rh_lower(lp, i); - Tupper = get_rh_upper(lp, i); - status = presolve_multibounds(psdata, i, colnr, &Tlower, &Tupper, NULL, &rowbinds); - isfree = isfree | status | rowbinds; - } - - return( (MYBOOL) (isfree == (TRUE | AUTOMATIC)) ); -} - -STATIC MYBOOL presolve_impliedcolfix(presolverec *psdata, int rownr, int colnr, MYBOOL isfree) -{ - lprec *lp = psdata->lp; - MYBOOL signflip, undoadded = FALSE; - MATrec *mat = lp->matA; - int jx, i, ib, ie = mat->row_end[rownr]; - REAL varLo = 0, varHi = 0, varRange, conRange = 0, matValue = 0, dual, RHS = lp->orig_rhs[rownr], - pivot, matAij = mat_getitem(mat, rownr, colnr), *vecOF = lp->orig_obj; - - /* We cannot have semi-continuous or non-qualifying integers */ - if(is_semicont(lp, colnr) || is_SOS_var(lp, colnr)) - return( FALSE ); - if(is_int(lp, colnr)) { - if(!isActiveLink(psdata->INTmap, rownr) || !is_presolve(lp, PRESOLVE_KNAPSACK)) - return( FALSE ); - /* colnr must have a coefficient equal to the smallest in the row */ - varRange = lp->infinite; - i = 0; - pivot = 0; - for(ib = presolve_nextcol(psdata, rownr, &i); i != 0; ib = presolve_nextcol(psdata, rownr, &i)) { - jx = ROW_MAT_COLNR(ib); - dual = fabs(ROW_MAT_VALUE(ib)); - /* Check if we have the target column and save the pivot value */ - if(jx == colnr) { - /* Always accept unit coefficient */ - if(fabs(dual - 1) < psdata->epsvalue) - break; - pivot = dual; - /* Otherwise continue scan */ - } - /* Cannot accept case where result can be fractional */ - else if((pivot > dual + psdata->epsvalue) || - ((pivot > 0) && (fabs(fmod(dual, pivot)) > psdata->epsvalue))) - return( FALSE ); - } - } - - /* Ascertain that the pivot value is large enough to preserve stability */ - pivot = matAij; - if(fabs(pivot) < psdata->epspivot*mat->colmax[colnr]) - return( FALSE ); - - /* Must ascertain that the row variables are not SOS'es; this is because - the eliminated variable will be a function of another variable. */ - if(SOS_count(lp) > 0) { - for(ib = mat->row_end[rownr-1]; ib < ie; ib++) - if(SOS_is_member(lp->SOS, 0, ROW_MAT_COLNR(ib))) - return( FALSE ); - } - - /* Calculate the dual value */ - dual = vecOF[colnr]/pivot; - - /* Here we have free variable in an equality constraint; this means we can - can adjust the OF for the deleted variable and also delete the constraint. */ - if(isfree && is_constr_type(lp, rownr, EQ)) { - matValue = RHS/pivot; - if(matValue != 0) - undoadded = addUndoPresolve(lp, TRUE, colnr, matValue, 0.0, 0); - } - - else { - - /* IMPLIEDFREE: For simplicity, ensure that we can keep the slack based at 0, - and not its upper bound. Effectively, we consider the constraint - an equality, using the information of the sign of the dual. - IMPLIEDSLK: Since we already have an equality constraint, we wish to make sure - that the ensuing inequality constraint will have an RHS that is - non-infinite. */ - if(isfree) { - SETMIN(RHS, presolve_sumplumin(lp, rownr, psdata->rows, TRUE)); - matValue = presolve_sumplumin(lp, rownr, psdata->rows, FALSE); - conRange = get_rh_lower(lp, rownr); - conRange = RHS - MAX(matValue, conRange); - signflip = (MYBOOL) ((dual > 0) && - !my_infinite(lp, conRange)); - } - else { - varLo = get_lowbo(lp, colnr); - varLo *= (my_infinite(lp, varLo) ? my_sign(pivot) : pivot); - varHi = get_upbo(lp, colnr); - varHi *= (my_infinite(lp, varHi) ? my_sign(pivot) : pivot); - if(pivot < 0) - swapREAL(&varHi, &varLo); - signflip = my_infinite(lp, varLo); - } - if(signflip) { - mat_multrow(mat, rownr, -1); - RHS -= conRange; - RHS = -RHS; - lp->orig_rhs[rownr] = RHS; - pivot = -pivot; - dual = -dual; - if(!isfree) { - varLo = -varLo; - varHi = -varHi; - swapREAL(&varHi, &varLo); - } - } - matValue = RHS/pivot; - - /* Prepare for deleting free or implied free variable in inequality constraint. - Different strategies need to be used: - - ACTUAL: Find the proper constraint bound and store undo information for - recovering the value of the implied free variable. The constraint - is then deleted. We have to adjust the objective function if the - OF coefficient for the implied free variable is non-zero. - IMPLIED: Convert the constraint to an inequality at the proper bound. - For given models, the new equality constraint can later provide - an implied slack, which means that a further variable is eliminated, - and the constraint again becomes an inequality constraint. - - Note that this version only implements the ACTUAL mode */ - if(isfree) { - /* Add undo information connecting the deleted variable to the RHS */ - if(matValue != 0) - undoadded = addUndoPresolve(lp, TRUE, colnr, matValue, 0.0, 0); - /* Add undo information for the dual of the deleted constraint */ - if(dual != 0) - addUndoPresolve(lp, FALSE, rownr, dual, 0.0, 0); - } - - /* Prepare for deleting implied slack variable. The following two cases are - handled: - - 1. Equality constraint: Convert the constraint to an inequality constraint - that is possibly ranged - 2. Other constraints: Expand existing slack variable / constraint - range, if required. */ - else { - if(my_infinite(lp, varHi)) - varRange = lp->infinite; -#ifdef Paranoia - else if(my_infinite(lp, varLo)) { - report(lp, SEVERE, "presolve_impliedcolfix: Negative infinite limit for variable %d\n", colnr); - varRange = lp->infinite; - } -#endif - else - varRange = my_precision(fabs(varHi - varLo) + lp->epsvalue, psdata->epsvalue); - presolve_adjustrhs(psdata, rownr, varLo, psdata->epsvalue); - - /* Handle case 1 of an equality constraint */ - if(is_constr_type(lp, rownr, EQ)) { - /* Make sure we actually have a ranged constraint */ - if(varRange > 0) { - set_constr_type(lp, rownr, LE); - if(!my_infinite(lp, varRange)) - lp->orig_upbo[rownr] = varRange; - setLink(psdata->LTmap, rownr); - removeLink(psdata->EQmap, rownr); - } - } - /* Handle case 2 of an inequality constraint (UNDER CONSTRUCTION!)*/ - else { - if(!my_infinite(lp, lp->orig_upbo[rownr])) { - if(my_infinite(lp, varRange)) - lp->orig_upbo[rownr] = lp->infinite; - else - lp->orig_upbo[rownr] += varHi - varLo; - } - } - /* Update counts */ - if(matAij > 0) - psdata->rows->plucount[rownr]--; - else - psdata->rows->negcount[rownr]--; - if(my_sign(varLo) != my_sign(varHi)) - psdata->rows->pluneg[rownr]--; - - /* Add undo information for the deleted variable; note that we cannot link the - deleted variable to the slack, since it may not be available during undo. - We really should have a mini LP to compute this allocation ex-post. */ - if(RHS != 0) - undoadded = addUndoPresolve(lp, TRUE, colnr, RHS/pivot, 0.0, 0); - } - } - - /* Update the OF constant */ - if(dual != 0) { - presolve_adjustrhs(psdata, 0, dual * RHS, 0); -/* lp->orig_rhs[0] -= dual * RHS; */ - vecOF[colnr] = 0; - } - - /* Do affine transformation with the constraint row */ - i = 0; - for(ib = presolve_nextcol(psdata, rownr, &i); ib >= 0; - ib = presolve_nextcol(psdata, rownr, &i)) { - - /* Get the constraint element */ - jx = ROW_MAT_COLNR(ib); - if(jx == colnr) - continue; - matValue = ROW_MAT_VALUE(ib); - - /* Adjust OF for the variable to be deleted */ - if(dual != 0) - vecOF[jx] -= dual * matValue; - - /* Add reconstruction/undo parameters for the deleted variable */ - if(!undoadded) - undoadded = addUndoPresolve(lp, TRUE, colnr, 0.0, matValue/pivot, jx); - else - appendUndoPresolve(lp, TRUE, matValue/pivot, jx); - } - - return( TRUE ); -} - -STATIC psrec *presolve_initpsrec(lprec *lp, int size) -{ - psrec *ps = (psrec *) calloc(1, sizeof(*ps)); - - createLink(size, &ps->varmap, NULL); - fillLink(ps->varmap); - - size++; - - allocINT(lp, &ps->empty, size, FALSE); - ps->empty[0] = 0; - - allocREAL(lp, &ps->pluupper, size, FALSE); - allocREAL(lp, &ps->negupper, size, FALSE); - allocREAL(lp, &ps->plulower, size, FALSE); - allocREAL(lp, &ps->neglower, size, FALSE); - allocINT(lp, &ps->infcount, size, FALSE); - - ps->next = (int **) calloc(size, sizeof(*(ps->next))); - - allocINT(lp, &ps->plucount, size, TRUE); - allocINT(lp, &ps->negcount, size, TRUE); - allocINT(lp, &ps->pluneg, size, TRUE); - - ps->allocsize = size; - - return( ps ); -} -STATIC void presolve_freepsrec(psrec **ps) -{ - FREE((*ps)->plucount); - FREE((*ps)->negcount); - FREE((*ps)->pluneg); - FREE((*ps)->infcount); - - if((*ps)->next != NULL) { - int i, n = (*ps)->allocsize; - for(i = 0; i < n; i++) - FREE((*ps)->next[i]); - FREE((*ps)->next); - } - - FREE((*ps)->plulower); - FREE((*ps)->neglower); - FREE((*ps)->pluupper); - FREE((*ps)->negupper); - - FREE((*ps)->empty); - - freeLink(&(*ps)->varmap); - - FREE(*ps); -} - -STATIC presolverec *presolve_init(lprec *lp) -{ - int k, i, ix, ixx, colnr, - ncols = lp->columns, - nrows = lp->rows; - REAL hold; - MATrec *mat = lp->matA; - presolverec *psdata = NULL; - - /* Optimize memory usage if we have a very large model; - this is to reduce the risk of out-of-memory situations. */ - ix = get_nonzeros(lp); - ixx = lp->matA->mat_alloc; - if((ixx - ix > MAT_START_SIZE) && ((ixx - ix) * 20 > ixx)) - mat_memopt(lp->matA, nrows / 20, ncols / 20, ix / 20); - - psdata = (presolverec *) calloc(1, sizeof(*psdata)); - - psdata->lp = lp; - psdata->rows = presolve_initpsrec(lp, nrows); - psdata->cols = presolve_initpsrec(lp, ncols); - - psdata->epsvalue = PRESOLVE_EPSVALUE; - psdata->epspivot = PRESOLVE_EPSPIVOT; - psdata->forceupdate = TRUE; - - /* Save incoming primal bounds */ - k = lp->sum + 1; - allocREAL(lp, &psdata->pv_lobo, k, FALSE); - MEMCOPY(psdata->pv_lobo, lp->orig_lowbo, k); - allocREAL(lp, &psdata->pv_upbo, k, FALSE); - MEMCOPY(psdata->pv_upbo, lp->orig_upbo, k); - - /* Create and initialize dual value (Langrangean and slack) limits */ - allocREAL(lp, &psdata->dv_lobo, k, FALSE); - allocREAL(lp, &psdata->dv_upbo, k, FALSE); - for(i = 0; i <= nrows; i++) { - psdata->dv_lobo[i] = (is_constr_type(lp, i, EQ) ? -lp->infinite : 0); - psdata->dv_upbo[i] = lp->infinite; - } - k--; - for(; i <= k; i++) { - psdata->dv_lobo[i] = 0; - psdata->dv_upbo[i] = lp->infinite; - } - - /* Create NZ count and sign arrays, and do general initialization of row bounds */ - createLink(nrows, &psdata->EQmap, NULL); - createLink(nrows, &psdata->LTmap, NULL); - createLink(nrows, &psdata->INTmap, NULL); - for(i = 1; i <= nrows; i++) { - switch (get_constr_type(lp, i)) { - case LE: appendLink(psdata->LTmap, i); - break; - case EQ: appendLink(psdata->EQmap, i); - break; - } - k = mat_rowlength(mat, i); - if((lp->int_vars > 0) && (k > 0)) - appendLink(psdata->INTmap, i); - } - - /* Seek to reduce set of sum(INT*INT) rows (mainly for GCD coefficient reductions) */ - if(psdata->INTmap->count > 0) - for(i = 1; i <= nrows; i++) { - if(!isActiveLink(psdata->INTmap, i)) - continue; - /* Disqualify if there is a non-int variable, otherwise find smallest absolute fractional row value */ - ix = mat->row_end[i - 1]; - ixx = mat->row_end[i]; - colnr = 0; - for(; ix < ixx; ix++) { - if(!is_int(lp, ROW_MAT_COLNR(ix))) { - removeLink(psdata->INTmap, i); - break; - } - hold = fabs(ROW_MAT_VALUE(ix)); - hold = fmod(hold, 1); - /* Adjust colnr to be a decimal scalar */ - for(k = 0; (k <= MAX_FRACSCALE) && (hold+psdata->epsvalue < 1); k++) - hold *= 10; - if(k > MAX_FRACSCALE) { - removeLink(psdata->INTmap, i); - break; - } - SETMAX(colnr, k); - } - if(!isActiveLink(psdata->INTmap, i)) - continue; - hold = pow(10.0, colnr); - /* Also disqualify if the RHS is fractional after scaling */ - if(fabs(fmod(lp->orig_rhs[i] * hold, 1)) > psdata->epsvalue) { - removeLink(psdata->INTmap, i); - continue; - } - /* We have an all-int constraint, see if we should scale it up */ - if(k > 0) { - ix = mat->row_end[i - 1]; - for(; ix < ixx; ix++) { - ROW_MAT_VALUE(ix) *= hold; - } - lp->orig_rhs[i] *= hold; - if(!my_infinite(lp, lp->orig_upbo[i])) - lp->orig_upbo[i] *= hold; /* KE: Fix due to Andy Loto - 20070619 */ - } - } - - /* Do the real tallying and ordering work */ - presolve_validate(psdata, TRUE); - - return( psdata ); -} - -STATIC void presolve_free(presolverec **psdata) -{ - presolve_freepsrec(&(*psdata)->rows); - presolve_freepsrec(&(*psdata)->cols); - FREE((*psdata)->dv_lobo); - FREE((*psdata)->dv_upbo); - FREE((*psdata)->pv_lobo); - FREE((*psdata)->pv_upbo); - freeLink(&(*psdata)->EQmap); - freeLink(&(*psdata)->LTmap); - freeLink(&(*psdata)->INTmap); - FREE(*psdata); -} - -STATIC int presolve_makefree(presolverec *psdata) -{ - lprec *lp = psdata->lp; - int i, ix, j, nn = 0; - REAL Xlower, Xupper, losum, upsum, lorhs, uprhs, freeinf = lp->infinite / 10; - MATrec *mat = lp->matA; - LLrec *colLL = NULL; - - /* First see if we can relax ranged constraints */ - for(i = firstActiveLink(psdata->rows->varmap); i != 0; i = nextActiveLink(psdata->rows->varmap, i)) { - if(is_constr_type(lp, i, EQ)) - continue; - presolve_range(lp, i, psdata->rows, &losum, &upsum); - lorhs = get_rh_lower(lp, i); - uprhs = get_rh_upper(lp, i); - - /* Look for opportunity to relax constraint bounds */ - if(presolve_rowlength(psdata, i) > 1) { - if((is_constr_type(lp, i, GE) && (upsum <= uprhs)) || - (is_constr_type(lp, i, LE) && (losum >= lorhs))) - set_rh_range(lp, i, lp->infinite); - } - } - - /* Collect columns available for bound relaxation (find implied free variables) - (consider sorting the list in decending order of column lengths or do call to - COLAMD to maximize impact) */ - createLink(lp->columns, &colLL, NULL); - for(j = firstActiveLink(psdata->cols->varmap); j != 0; j = nextActiveLink(psdata->cols->varmap, j)) - if(presolve_impliedfree(lp, psdata, j)) - appendLink(colLL, j); - - /* Find what columns to relax (ideally one per row) */ - if(colLL->count > 0) { - LLrec *rowLL = NULL; - MYBOOL canfree; - - /* Create row tracker */ - createLink(lp->rows, &rowLL, NULL); - fillLink(rowLL); - - /* Loop over all column candidates */ - for(j = firstActiveLink(colLL); (j > 0) && (rowLL->count > 0); j = nextActiveLink(colLL, j)) { - - /* Verify that the variable is applicable */ - canfree = TRUE; - for(ix = mat->col_end[j-1]; canfree && (ix < mat->col_end[j]); ix++) - canfree = isActiveLink(rowLL, COL_MAT_ROWNR(ix)); - - /* If so, then open the bounds and update the row availability mapper */ - if(canfree) { - nn++; - Xlower = get_lowbo(lp, j); - Xupper = get_upbo(lp, j); - if(Xlower >= 0) - set_bounds(lp, j, 0, freeinf); - else if(Xupper <= 0) - set_bounds(lp, j, -freeinf, 0); - else -/* set_bounds(lo, j, -freeinf, freeinf); */ - set_unbounded(lp, j); - for(ix = mat->col_end[j-1]; ix < mat->col_end[j]; ix++) - removeLink(rowLL, COL_MAT_ROWNR(ix)); - } - } - freeLink(&rowLL); - } - - /* Free list and return */ - freeLink(&colLL); - return( nn ); -} - -STATIC MYBOOL presolve_updatesums(presolverec *psdata) -{ - lprec *lp = psdata->lp; - int j; - - /* Initialize row accumulation arrays */ - MEMCLEAR(psdata->rows->pluupper, lp->rows + 1); - MEMCLEAR(psdata->rows->negupper, lp->rows + 1); - MEMCLEAR(psdata->rows->plulower, lp->rows + 1); - MEMCLEAR(psdata->rows->neglower, lp->rows + 1); - MEMCLEAR(psdata->rows->infcount, lp->rows + 1); - - /* Loop over active columns */ - for(j = firstActiveLink(psdata->cols->varmap); j != 0; - j = nextActiveLink(psdata->cols->varmap, j)) { - presolve_colfix(psdata, j, lp->infinite, FALSE, NULL); - } - -#ifdef UseDualPresolve - /* Initialize column accumulation arrays */ - MEMCLEAR(psdata->cols->pluupper, lp->columns + 1); - MEMCLEAR(psdata->cols->negupper, lp->columns + 1); - MEMCLEAR(psdata->cols->plulower, lp->columns + 1); - MEMCLEAR(psdata->cols->neglower, lp->columns + 1); - MEMCLEAR(psdata->cols->infcount, lp->columns + 1); - - /* Loop over active rows */ - for(j = firstActiveLink(psdata->rows->varmap); j != 0; - j = nextActiveLink(psdata->rows->varmap, j)) { - presolve_rowfix(psdata, j, lp->infinite, FALSE, NULL); - } -#endif - - return( TRUE ); -} - -STATIC MYBOOL presolve_finalize(presolverec *psdata) -{ - lprec *lp = psdata->lp; - MYBOOL compactvars = FALSE; - int ke, n; - - /* Save eliminated rows and columns for restoration purposes */ -#ifdef SavePresolveEliminated - psdata->deletedA = mat_extractmat(lp->matA, rowmap, colmap, TRUE); - if(!mat_validate(psdata->deletedA)) - report(lp, SEVERE, "presolve_finalize: Could not validate matrix with undo data\n"); -#endif - - /* Check if OF columns are to be deleted */ - lp->presolve_undo->OFcolsdeleted = FALSE; - for(n = firstInactiveLink(psdata->cols->varmap); (n != 0) && !lp->presolve_undo->OFcolsdeleted; - n = nextInactiveLink(psdata->cols->varmap, n)) - lp->presolve_undo->OFcolsdeleted = (MYBOOL) (lp->orig_obj[n] != 0); - - /* Delete eliminated columns */ - ke = lastInactiveLink(psdata->cols->varmap); - n = countInactiveLink(psdata->cols->varmap); - if((n > 0) && (ke > 0)) { - del_columnex(lp, psdata->cols->varmap); - mat_colcompact(lp->matA, lp->presolve_undo->orig_rows, - lp->presolve_undo->orig_columns); - compactvars = TRUE; - } - - /* Delete eliminated rows */ - ke = lastInactiveLink(psdata->rows->varmap); - n = countInactiveLink(psdata->rows->varmap); - if((n > 0) && (ke > 0)) { - del_constraintex(lp, psdata->rows->varmap); - mat_rowcompact(lp->matA, TRUE); - compactvars = TRUE; - } - else if(psdata->nzdeleted > 0) - mat_zerocompact(lp->matA); - - /* Do compacting and updating of variable maps */ - if(compactvars) - varmap_compact(lp, lp->presolve_undo->orig_rows, - lp->presolve_undo->orig_columns); - - /* Reduce memory usage of postsolve matrices */ - if(lp->presolve_undo->primalundo != NULL) - mat_memopt(lp->presolve_undo->primalundo->tracker, 0, 0, 0); - if(lp->presolve_undo->dualundo != NULL) - mat_memopt(lp->presolve_undo->dualundo->tracker, 0, 0, 0); - - /* Round near-zero objective function coefficients and RHS values */ - ke = lp->columns; - for(n = 1; n <= ke; n++) - my_roundzero(lp->orig_obj[n], lp->epsvalue); - ke = lp->rows; - for(n = 1; n <= ke; n++) - my_roundzero(lp->orig_rhs[n], lp->epsvalue); - - /* Update the SOS sparse mapping */ - if(SOS_count(lp) > 0) - SOS_member_updatemap(lp->SOS); - - /* Validate matrix and reconstruct row indexation */ - return(mat_validate(lp->matA)); -} - -STATIC MYBOOL presolve_debugdump(lprec *lp, presolverec *psdata, char *filename, MYBOOL doappend) -{ - FILE *output = stdout; - int size; - MYBOOL ok; - - ok = (MYBOOL) ((filename == NULL) || ((output = fopen(filename, my_if(doappend, "a", "w"))) != NULL)); - if(!ok) - return(ok); - if((filename == NULL) && (lp->outstream != NULL)) - output = lp->outstream; - - fprintf(output, "\nPRESOLVE - Status at loop %d:%d:%d\n", - psdata->outerloops, psdata->middleloops, psdata->innerloops); - fprintf(output, "Model size: %d rows (%d equalities, %d less than), %d columns\n", - psdata->rows->varmap->count, psdata->EQmap->count, psdata->LTmap->count, psdata->cols->varmap->count); - - fprintf(output, "\nMAPPERS\n-------\n\n"); - size = 1; - blockWriteINT(output, "colmap", psdata->cols->varmap->map, 0, size*psdata->cols->varmap->size); - blockWriteINT(output, "rowmap", psdata->rows->varmap->map, 0, size*psdata->rows->varmap->size); - blockWriteINT(output, "EQmap", psdata->EQmap->map, 0, size*psdata->EQmap->size); - blockWriteINT(output, "LTmap", psdata->LTmap->map, 0, size*psdata->LTmap->size); - - fprintf(output, "\nCOUNTS\n------\n\n"); - blockWriteINT(output, "plucount", psdata->rows->plucount, 0, lp->rows); - blockWriteINT(output, "negcount", psdata->rows->negcount, 0, lp->rows); - blockWriteINT(output, "pluneg", psdata->rows->pluneg, 0, lp->rows); - - fprintf(output, "\nSUMS\n----\n\n"); - blockWriteREAL(output, "pluupper", psdata->rows->pluupper, 0, lp->rows); - blockWriteREAL(output, "negupper", psdata->rows->negupper, 0, lp->rows); - blockWriteREAL(output, "plulower", psdata->rows->pluupper, 0, lp->rows); - blockWriteREAL(output, "neglower", psdata->rows->negupper, 0, lp->rows); - - if(filename != NULL) - fclose(output); - return(ok); -} - -int CMP_CALLMODEL compRedundant(const UNIONTYPE QSORTrec *current, const UNIONTYPE QSORTrec *candidate) -{ - int start1 = (int) (current->int4.intpar1), - start2 = (int) (candidate->int4.intpar1), - result = CMP_COMPARE(start1, start2); - - if(result == 0) { - start1 = (int) (current->int4.intpar2); - start2 = (int) (candidate->int4.intpar2); - result = -CMP_COMPARE(start1, start2); - } - return( result ); -} -int CMP_CALLMODEL compSparsity(const UNIONTYPE QSORTrec *current, const UNIONTYPE QSORTrec *candidate) -{ - int start1 = (int) (current->int4.intpar1), - start2 = (int) (candidate->int4.intpar1), - result = CMP_COMPARE(start1, start2); - - if(result == 0) { - start1 = (int) (current->int4.intpar2); - start2 = (int) (candidate->int4.intpar2); - result = -CMP_COMPARE(start1, start2); - } - - if(result == 0) { - start1 = (int) (current->int4.intval); - start2 = (int) (candidate->int4.intval); - result = CMP_COMPARE(start1, start2); - } - return( result ); -} -int CMP_CALLMODEL compAggregate(const UNIONTYPE QSORTrec *current, const UNIONTYPE QSORTrec *candidate) -{ - int index1 = (int) (current->pvoidint2.intval), - index2 = (int) (candidate->pvoidint2.intval); - lprec *lp = (lprec *) current->pvoidint2.ptr; - REAL value1 = lp->orig_obj[index1], - value2 = lp->orig_obj[index2]; - - /* Smallest objective coefficient (largest contribution to OF) */ - int result = CMP_COMPARE(value1, value2); - - /* Smallest lower variable bound */ - if(result == 0) { - index1 += lp->rows; - index2 += lp->rows; - value1 = lp->orig_lowbo[index1]; - value2 = lp->orig_lowbo[index2]; - result = CMP_COMPARE(value1, value2); - } - - /* Largest upper variable bound */ - if(result == 0) { - value1 = lp->orig_upbo[index1]; - value2 = lp->orig_upbo[index2]; - result = -CMP_COMPARE(value1, value2); - } - return( result ); -} - -STATIC int presolve_rowdominance(presolverec *psdata, int *nCoeffChanged, int *nRowsRemoved, int *nVarsFixed, int *nSum) -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - int i, ii, ib, ie, n, jb, je, jx, *coldel = NULL, status = RUNNING, item, - iCoeffChanged = 0, iRowRemoved = 0, iVarFixed = 0; - REAL ratio, *rowvalues = NULL; - UNIONTYPE QSORTrec *QS = (UNIONTYPE QSORTrec *) calloc(lp->rows+1, sizeof(*QS)); - - /* Check if we were able to obtain working memory */ - if(QS == NULL) - return( status); - - /* A dominating row of variables always satisfy the following criteria: - 1) The starting column position is never lower, but could be the same - 2) The non-zero row count is always lower */ - n = 0; - for(i = firstActiveLink(psdata->EQmap); i != 0; i = nextActiveLink(psdata->EQmap, i)) { - /* Make sure we have no SOS or semi-continuous variables */ - jb = je = 0; - if((SOS_count(lp) > 0) || (lp->sc_vars > 0)) { - item = 0; - for(jb = presolve_nextcol(psdata, i, &item); jb >= 0; - jb = presolve_nextcol(psdata, i, &item)) { - jx = ROW_MAT_COLNR(jb); - if(SOS_is_member(lp->SOS, 0, jx) || is_semicont(lp, jx)) - break; - } - } - - /* Add to list if we are Ok */ - if(jb < 0) { - QS[n].int4.intval = i; - item = 0; - ii = presolve_nextcol(psdata, i, &item); - QS[n].int4.intpar1 = ROW_MAT_COLNR(ii); - QS[n].int4.intpar2 = presolve_rowlength(psdata, i); - n++; - } - } - if(n <= 1) - goto Finish; - QS_execute(QS, n, (findCompare_func *) compRedundant, NULL); - - /* Let us start from the top of the list, going forward and looking - for the longest possible dominating row */ - if(!allocREAL(lp, &rowvalues, lp->columns + 1, TRUE) || - !allocINT(lp, &coldel, lp->columns + 1, FALSE)) - goto Finish; - - for(ib = 0; ib < n; ib++) { - - /* Get row and check if it was previously eliminated */ - i = QS[ib].int4.intval; - if(i < 0) - continue; - - /* Load the non-zero row values */ - item = 0; - for(jb = presolve_nextcol(psdata, i, &item); jb >= 0; - jb = presolve_nextcol(psdata, i, &item)) { - jx = ROW_MAT_COLNR(jb); - rowvalues[jx] = ROW_MAT_VALUE(jb); - } - - for(ie = ib+1; ie < n; ie++) { - - /* Get row and check if it was previously eliminated */ - ii = QS[ie].int4.intval; - if(ii < 0) - continue; - -#ifdef Paranoia - if((QS[ib].int4.intpar1 > QS[ie].int4.intpar1) || - ((QS[ib].int4.intpar1 == QS[ie].int4.intpar1) && (QS[ib].int4.intpar2 < QS[ie].int4.intpar2))) - report(lp, SEVERE, "presolve_rowdominance: Invalid sorted row order\n"); -#endif - - /* Loop over every row member to confirm that the candidate - actually dominates in every position */ - if((lp->orig_rhs[i] == 0) && (lp->orig_rhs[ii] == 0)) - ratio = 0; - else if((lp->orig_rhs[i] != 0) && (lp->orig_rhs[ii] != 0)) - ratio = lp->orig_rhs[i] / lp->orig_rhs[ii]; - else - continue; - item = 0; - for(jb = presolve_nextcol(psdata, ii, &item); jb >= 0; - jb = presolve_nextcol(psdata, ii, &item)) { - jx = ROW_MAT_COLNR(jb); - if(rowvalues[jx] == 0) - break; - if(ratio == 0) - ratio = rowvalues[jx] / ROW_MAT_VALUE(jb); - else if(fabs(rowvalues[jx] - ratio*ROW_MAT_VALUE(jb)) > psdata->epsvalue) - break; - } - - /* "We have contact" */ - if(jb < 0) { - int sign_1 = 0, sign_j = 0; - - /* Need to fix any superset columns, but require that they have equal signs */ - coldel[0] = 0; - item = 0; - for(jb = presolve_nextcol(psdata, i, &item); jb >= 0; - jb = presolve_nextcol(psdata, i, &item)) { - jx = ROW_MAT_COLNR(jb); - if(mat_findelm(mat, ii, jx) <= 0) { - - /* Cancel if we detect a free or "quasi-free" variable */ - if((lp->orig_lowbo[lp->rows + jx] < 0) && - (lp->orig_upbo[lp->rows + jx] > 0)) { - coldel[0] = -1; - break; - } - - /* Ensure that we are feasible */ - else if((lp->orig_lowbo[lp->rows + jx] > 0) || - (lp->orig_upbo[lp->rows + jx] < 0)) { - report(lp, DETAILED, "presolve_rowdominate: Column %s is infeasible due to conflict in rows %s and %s\n", - get_col_name(lp, jx), get_row_name(lp, i), get_row_name(lp, ii)); - coldel[0] = -1; - break; - } - - /* Check consistency / uniformity of signs */ - sign_j = my_sign(ROW_MAT_VALUE(jb)); - sign_j = my_chsign(is_negative(lp, jx), sign_j); - if(coldel[0] == 0) { - sign_1 = sign_j; - coldel[++coldel[0]] = jx; - } - else if(sign_j == sign_1) { - coldel[++coldel[0]] = jx; - } - else { - coldel[0] = -1; - break; - } - } - } - - /* Force break / continuation if the superset columns were incompatible */ - if(coldel[0] < 0) - continue; - - /* Do the column fixing and deletion (check for infeasibility in the process) */ - for(jb = 1; jb <= coldel[0]; jb++) { - jx = coldel[jb]; - if(!presolve_colfix(psdata, jx, 0, TRUE, &iVarFixed)) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Finish; - } - presolve_colremove(psdata, jx, TRUE); - rowvalues[jx] = 0; - } - - /* Then delete the row */ - presolve_rowremove(psdata, ii, TRUE); - iRowRemoved++; - QS[ie].int4.intval = -ii; - } - } - - /* Clear the non-zero row values ahead of the next row candidate */ - ie = mat->row_end[i-1]; - ii = mat->row_end[i]; - for(; ie < ii; ie++) - rowvalues[ROW_MAT_COLNR(ie)] = 0; - - } -Finish: - FREE(QS); - FREE(rowvalues); - FREE(coldel); - - (*nCoeffChanged) += iCoeffChanged; - (*nRowsRemoved) += iRowRemoved; - (*nVarsFixed) += iVarFixed; - (*nSum) += iCoeffChanged + iRowRemoved + iVarFixed; - - return( status ); -} - -#if 0 -STATIC int presolve_coldominance01(presolverec *psdata, int *nConRemoved, int *nVarsFixed, int *nSum) -/* The current version of this routine eliminates binary variables - that are dominated via set coverage or unit knapsack constraints */ -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - MYBOOL first; - int i, ii, ib, ie, n, jb, je, jx, jj, item, item2, - *coldel = NULL, status = RUNNING, iVarFixed = 0; - REAL scale, rhsval, *colvalues = NULL; - UNIONTYPE QSORTrec *QS = (UNIONTYPE QSORTrec *) calloc(lp->columns+1, sizeof(*QS)); - - /* Check if we were able to obtain working memory */ - if(QS == NULL) - return( status); - if(lp->int_vars == 0) - goto Finish; - - /* A column dominates another binary variable column with the following criteria: - 1) The relative matrix non-zero entries are identical - 2) The relative objective coefficient is worse than the other; - if the OF coefficients are identical, we can delete an arbitrary variable */ - n = 0; - for(i = firstActiveLink(psdata->cols->varmap); i != 0; i = nextActiveLink(psdata->cols->varmap, i)) - if(is_binary(lp, i) && !SOS_is_member(lp->SOS, 0, i)) { - /* Make sure we have an all-binary, unit-coefficient row */ - je = mat->col_end[i]; - item = 0; - for(jb = presolve_nextrow(psdata, i, &item); jb >= 0; - jb = presolve_nextrow(psdata, i, &item)) { - jx = COL_MAT_ROWNR(jb); - if(COL_MAT_VALUE(jb) != 1) - break; - } - - /* Add to list if we are Ok */ - if(jb < 0) { - QS[n].int4.intval = i; - item = 0; - ii = presolve_nextrow(psdata, i, &item); - QS[n].int4.intpar1 = COL_MAT_ROWNR(ii); - ii = presolve_collength(psdata, i); - QS[n].int4.intpar2 = ii; - n++; - } - } - if(n <= 1) { - FREE(QS); - return( status ); - } - QS_execute(QS, n, (findCompare_func *) compRedundant, NULL); - - /* Let us start from the top of the list, going forward and looking - for the longest possible dominated column */ - if(!allocREAL(lp, &colvalues, lp->rows + 1, TRUE) || - !allocINT(lp, &coldel, lp->columns + 1, FALSE)) - goto Finish; - - for(ib = 0; ib < n; ib++) { - - /* Get column and check if it was previously eliminated */ - i = QS[ib].int4.intval; - if(i < 0) - continue; - - /* Load the non-zero column values */ - item = 0; - for(jb = presolve_nextrow(psdata, i, &item); jb >= 0; - jb = presolve_nextrow(psdata, i, &item)) { - jx = COL_MAT_ROWNR(jb); - colvalues[jx] = COL_MAT_VALUE(jb); - } - - coldel[0] = 0; - for(ie = ib+1; ie < n; ie++) { - - /* Insist on identical column lengths (sort is decending in column lengths) */ - ii = QS[ib].int4.intpar2 - QS[ie].int4.intpar2; - if(ii != 0) - break; - - /* Also insist on identical starting positions */ - ii = QS[ib].int4.intpar1 - QS[ie].int4.intpar1; - if(ii != 0) - break; - - /* Get column and check if it was previously eliminated */ - ii = QS[ie].int4.intval; - if(ii < 0) - continue; - - /* Also make sure that the variables have "compatible" bounds */ -#if 1 - if((fabs(my_reldiff(lp->orig_lowbo[lp->rows + i], lp->orig_lowbo[lp->rows + ii])) > psdata->epsvalue) || - (fabs(my_reldiff(lp->orig_upbo[lp->rows + i], lp->orig_upbo[lp->rows + ii] )) > psdata->epsvalue)) - continue; -#endif - -#ifdef Paranoia - if((QS[ib].int4.intpar1 > QS[ie].int4.intpar1) || - ((QS[ib].int4.intpar1 == QS[ie].int4.intpar1) && (QS[ib].int4.intpar2 < QS[ie].int4.intpar2))) - report(lp, SEVERE, "presolve_coldominance01: Invalid sorted column order\n"); -#endif - - /* Loop over every column member to confirm that the candidate is - relatively identical in every position */ - first = TRUE; - item = 0; - item2 = 0; - scale = 1; - for(jb = presolve_nextrow(psdata, ii, &item), - jj = presolve_nextrow(psdata, i, &item2); jb >= 0; - jb = presolve_nextrow(psdata, ii, &item), - jj = presolve_nextrow(psdata, i, &item2)) { - jx = COL_MAT_ROWNR(jb); - if(jx != COL_MAT_ROWNR(jj)) - break; - if(first) { - first = !first; - scale = colvalues[jx] / COL_MAT_VALUE(jb); - } - else { - if(fabs(colvalues[jx] - scale * COL_MAT_VALUE(jb)) > psdata->epsvalue) - break; - } - /* Also make sure we have a compatible RHS (since this version of the - dominance logic only applies to "sets") */ - rhsval = scale*lp->orig_rhs[jx] - 1.0; - /* if((rhsval < 0) || (rhsval > 1 + psdata->epsvalue)) */ - if(fabs(rhsval) > psdata->epsvalue) - break; - } - - /* "We have contact" */ - if(jb < 0) { - coldel[++coldel[0]] = ii; - QS[ie].int4.intval = -ii; - } - } - - /* Find the dominant column and delete / fix the others; - if there is a tie, simply delete the second candidate */ - ii = i; - for(jb = 1; jb <= coldel[0]; jb++) { - jx = coldel[jb]; - if(lp->orig_obj[jx] < lp->orig_obj[ii]) - swapINT(&ii, &coldel[jb]); - } - for(jb = 1; jb <= coldel[0]; jb++) { - jx = coldel[jb]; - if(!presolve_colfix(psdata, jx, lp->orig_lowbo[lp->rows+jx], TRUE, &iVarFixed)) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Finish; - } - presolve_colremove(psdata, jx, TRUE); - } - - /* Clear the non-zero row values ahead of the next row candidate */ - if(ib + 1 < n) { - ie = mat->col_end[i-1]; - ii = mat->col_end[i]; - for(; ie < ii; ie++) - colvalues[COL_MAT_ROWNR(ie)] = 0; - } - } -Finish: - FREE(QS); - FREE(colvalues); - FREE(coldel); - - (*nVarsFixed) += iVarFixed; - (*nSum) += iVarFixed; - - return( status ); -} -#else - -/* DEVELOPMENT/TEST CODE FOR POSSIBLE REPLACEMENT OF SIMILAR FUNCTION IN lp_presolve.c */ - -#define NATURAL int - -STATIC int presolve_coldominance01(presolverec *psdata, NATURAL *nConRemoved, NATURAL *nVarsFixed, NATURAL *nSum) -/* The current version of this routine eliminates binary variables - that are dominated via set coverage or unit knapsack constraints */ -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - NATURAL i, ib, ie, jx, item, item2, - n = lp->int_vars, iVarFixed = 0, nrows = lp->rows, - *coldel = NULL; - int jb, jj, ii, - status = RUNNING; - REAL rhsval = 0.0, - *colvalues = NULL, *colobj = NULL; - LLrec *sets = NULL; - UNIONTYPE QSORTrec *QS = (UNIONTYPE QSORTrec *) calloc(n+1, sizeof(*QS)); - - /* Check if we were able to obtain working memory */ - if(QS == NULL) - return( status); - if(n == 0) - goto Finish; - - /* Create list of set coverage and knapsack constraints */ - createLink(nrows, &sets, NULL); - for(i = firstActiveLink(psdata->rows->varmap); i != 0; i = nextActiveLink(psdata->rows->varmap, i)) { - if((lp->orig_rhs[i] < 0) || (psdata->rows->negcount[i] > 0)) - continue; - item = 0; - for(jb = presolve_nextcol(psdata, i, &item); jb >= 0; - jb = presolve_nextcol(psdata, i, &item)) { - jx = ROW_MAT_COLNR(jb); - if(!is_binary(lp, jx)) - break; - rhsval = ROW_MAT_VALUE(jb) - 1; - if(fabs(rhsval) > lp->epsvalue) - break; - } - if(jb < 0) - setLink(sets, i); - } - if(countActiveLink(sets) == 0) - goto Finish; - - /* A column dominates another binary variable column with the following criteria: - 1) The relative matrix non-zero entries are identical - 2) The relative objective coefficient is worse than the other; - if the OF coefficients are identical, we can delete an arbitrary variable */ - n = 0; - for(i = firstActiveLink(psdata->cols->varmap); i != 0; i = nextActiveLink(psdata->cols->varmap, i)) - if(is_binary(lp, i) && !SOS_is_member(lp->SOS, 0, i)) { - /* Make sure the column is member of at least one set */ - item = 0; - for(jb = presolve_nextrow(psdata, i, &item); jb >= 0; - jb = presolve_nextrow(psdata, i, &item)) { - jx = COL_MAT_ROWNR(jb); - if(isActiveLink(sets, jx)) - break; - } - - /* Add to list if set membership test is Ok */ - if(jb >= 0) { - QS[n].int4.intval = i; - item = 0; - ii = presolve_nextrow(psdata, i, &item); - QS[n].int4.intpar1 = COL_MAT_ROWNR(ii); - ii = presolve_collength(psdata, i); - QS[n].int4.intpar2 = ii; - n++; - } - } - if(n <= 1) { - FREE(QS); - return( status ); - } - QS_execute(QS, n, (findCompare_func *) compRedundant, NULL); - - /* Let us start from the top of the list, going forward and looking - for the longest possible dominated column */ - if(!allocREAL(lp, &colvalues, nrows + 1, TRUE) || - !allocREAL(lp, &colobj, n + 1, FALSE) || - !allocINT(lp, &coldel, n + 1, FALSE)) - goto Finish; - - for(ib = 0; ib < n; ib++) { - - /* Get column and check if it was previously eliminated */ - i = QS[ib].int4.intval; - if(!isActiveLink(psdata->cols->varmap, i)) - continue; - - /* Load the non-zero column values */ - item = 0; - for(jb = presolve_nextrow(psdata, i, &item); jb >= 0; - jb = presolve_nextrow(psdata, i, &item)) { - jx = COL_MAT_ROWNR(jb); - colvalues[jx] = COL_MAT_VALUE(jb); - } - - /* Store data for current column */ - coldel[0] = 1; - coldel[1] = i; - colobj[1] = lp->orig_obj[i]; - - /* Loop over all other columns to see if they have equal constraint coefficients */ - for(ie = ib+1; ie < n; ie++) { - - /* Check if this column was previously eliminated */ - ii = QS[ie].int4.intval; - if(!isActiveLink(psdata->cols->varmap, ii)) - continue; - - /* Insist on identical column lengths (sort is decending in column lengths) */ - ii = QS[ib].int4.intpar2 - QS[ie].int4.intpar2; - if(ii != 0) - break; - - /* Also insist on identical starting positions */ - ii = QS[ib].int4.intpar1 - QS[ie].int4.intpar1; - if(ii != 0) - break; - - /* Get column and check if it was previously eliminated */ - ii = QS[ie].int4.intval; - -#ifdef Paranoia - if((QS[ib].int4.intpar1 > QS[ie].int4.intpar1) || - ((QS[ib].int4.intpar1 == QS[ie].int4.intpar1) && (QS[ib].int4.intpar2 < QS[ie].int4.intpar2))) - report(lp, SEVERE, "presolve_coldominance01: Invalid sorted column order\n"); -#endif - - /* Loop over every column member to confirm that the candidate is identical in every row; - we also compute the minimal set order */ - rhsval = lp->infinite; - item = 0; - item2 = 0; - for(jb = presolve_nextrow(psdata, ii, &item), - jj = presolve_nextrow(psdata, i, &item2); jb >= 0; - jb = presolve_nextrow(psdata, ii, &item), - jj = presolve_nextrow(psdata, i, &item2)) { - jx = COL_MAT_ROWNR(jb); - if(jx != COL_MAT_ROWNR(jj)) - break; - if(isActiveLink(sets, jx)) - SETMIN(rhsval, lp->orig_rhs[jx]); - } - - /* "We have contact" */ - if(jb < 0) { - coldel[++coldel[0]] = ii; - colobj[coldel[0]] = lp->orig_obj[ii]; - } - } - - /* Find the dominant columns, fix and delete the others */ - if(coldel[0] > 1) { - qsortex(colobj+1, coldel[0], 0, sizeof(*colobj), FALSE, compareREAL, coldel+1, sizeof(*coldel)); - /* if(rhsval+lp->epsvalue < lp->infinite) { */ - jb = (NATURAL) (rhsval+lp->epsvalue); - /* printf("%f / %d\n", rhsval, jb); */ - for(jb++; jb <= coldel[0]; jb++) { - jx = coldel[jb]; - if(!presolve_colfix(psdata, jx, lp->orig_lowbo[nrows+jx], TRUE, &iVarFixed)) { - status = presolve_setstatus(psdata, INFEASIBLE); - goto Finish; - } - presolve_colremove(psdata, jx, TRUE); - } - /*} */ - } - - /* Clear the non-zero row values ahead of the next row candidate */ - if(ib + 1 < n) { - ie = mat->col_end[i-1]; - ii = mat->col_end[i]; - for(; ie < ii; ie++) - colvalues[COL_MAT_ROWNR(ie)] = 0; - } - } -Finish: - freeLink(&sets); - FREE(QS); - FREE(colvalues); - FREE(coldel); - FREE(colobj); - - (*nVarsFixed) += iVarFixed; - (*nSum) += iVarFixed; - - return( status ); -} - -#endif - -STATIC int presolve_aggregate(presolverec *psdata, int *nConRemoved, int *nVarsFixed, int *nSum) -/* This routine combines compatible or identical columns */ -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - MYBOOL first; - int i, ii, ib, ie, ix, n, jb, je, jx, jj, item, item2, - *coldel = NULL, status = RUNNING, iVarFixed = 0; - REAL scale, *colvalues = NULL; - UNIONTYPE QSORTrec *QScand = (UNIONTYPE QSORTrec *) calloc(lp->columns+1, sizeof(*QScand)); - - /* Check if we were able to obtain working memory */ - if(QScand == NULL) - return( status); - - /* Obtain the list of qualifying columns to be sorted */ - n = 0; - for(i = firstActiveLink(psdata->cols->varmap); i != 0; i = nextActiveLink(psdata->cols->varmap, i)) - if(!is_semicont(lp, i) && !SOS_is_member(lp->SOS, 0, i)) { - QScand[n].int4.intval = i; - item = 0; - ii = presolve_nextrow(psdata, i, &item); - QScand[n].int4.intpar1 = COL_MAT_ROWNR(ii); - ii = presolve_collength(psdata, i); - QScand[n].int4.intpar2 = ii; - n++; - } - if(n <= 1) { - FREE(QScand); - return( status ); - } - QS_execute(QScand, n, (findCompare_func *) compRedundant, NULL); - - /* Let us start from the top of the list, going forward and looking - for the longest possible identical column */ - if(!allocREAL(lp, &colvalues, lp->rows + 1, TRUE) || - !allocINT(lp, &coldel, lp->columns + 1, FALSE)) - goto Finish; - - for(ib = 0; ib < n; ib++) { - - /* Get column and check if it was previously eliminated */ - i = QScand[ib].int4.intval; - if(i < 0) - continue; - - /* Load the non-zero column values of this active/reference column */ - item = 0; - for(jb = presolve_nextrow(psdata, i, &item); jb >= 0; - jb = presolve_nextrow(psdata, i, &item)) { - jx = COL_MAT_ROWNR(jb); - colvalues[jx] = COL_MAT_VALUE(jb); - } - - coldel[0] = 0; - for(ie = ib+1; ie < n; ie++) { - - /* Insist on identical column lengths (sort is decending in column lengths) */ - ii = QScand[ib].int4.intpar2 - QScand[ie].int4.intpar2; - if(ii != 0) - break; - - /* Also insist on identical starting positions */ - ii = QScand[ib].int4.intpar1 - QScand[ie].int4.intpar1; - if(ii != 0) - break; - - /* Get column and check if it was previously eliminated */ - ii = QScand[ie].int4.intval; - if(ii < 0) - continue; - - /* Loop over every column member to confirm that the candidate is - relatively identical in every position */ - first = TRUE; - item = 0; - item2 = 0; - scale = 1; - for(jb = presolve_nextrow(psdata, ii, &item), - jj = presolve_nextrow(psdata, i, &item2); jb >= 0; - jb = presolve_nextrow(psdata, ii, &item), - jj = presolve_nextrow(psdata, i, &item2)) { - jx = COL_MAT_ROWNR(jb); - if(jx != COL_MAT_ROWNR(jj)) - break; - if(first) { - first = !first; - scale = colvalues[jx] / COL_MAT_VALUE(jb); - } - else { - if(fabs(colvalues[jx] - scale * COL_MAT_VALUE(jb)) > psdata->epsvalue) - break; - } - } - - /* "We have contact", store the column in the aggregation list */ - if(jb < 0) { - coldel[++coldel[0]] = ii; - QScand[ie].int4.intval = -ii; - } - } - - /* Sort the aggregation list if we have aggregation candidates */ - if(coldel[0] > 1) { - REAL of, ofelim, fixvalue; - MYBOOL isint; - UNIONTYPE QSORTrec *QSagg = (UNIONTYPE QSORTrec *) calloc(coldel[0], sizeof(*QSagg)); - - for(jb = 1; jb <= coldel[0]; jb++) { - ii = jb - 1; - QSagg[ii].pvoidint2.intval = coldel[jb]; - QSagg[ii].pvoidint2.ptr = (void *) lp; - } - QS_execute(QSagg, coldel[0], (findCompare_func *) compAggregate, NULL); - - /* Process columns with identical OF coefficients */ - jb = 0; - while((status == RUNNING) && (jb < coldel[0])) { - ii = QSagg[jb].pvoidint2.intval; - of = lp->orig_obj[ii]; - isint = is_int(lp, ii); - je = jb + 1; - while((status == RUNNING) && (je < coldel[0]) && - (fabs(lp->orig_obj[ix = QSagg[je].pvoidint2.intval] - of) < psdata->epsvalue)) { - /* We now have two columns with equal OFs; the following cases are possible: - - 1) The first column has Inf upper bound, which means that it can - "absorb" compatible columns, which are then fixed at the appropriate - bounds (or zero in case of free variables). - 2) The first column has a -Inf lower bound, and further columns are - Inf upper bounds, which means steps towards forming a free variable - can be made. - 3) The first column is a non-Inf upper bound, in which case the bounds - are summed into a helper variable and the variable simply deleted. - The deleted variables' value are allocated/distributed via a simple - linear programming routine at postsolve. - - In the current version of this code, we only handle case 1. */ - if(is_int(lp, ix) == isint) { - ofelim = lp->orig_obj[ix]; - if(of == 0) - scale = 1; - else - scale = ofelim / of; - - if(my_infinite(lp, lp->orig_upbo[lp->rows+ii])) { /* Case 1 (recipe.mps) */ - if(is_unbounded(lp, ix)) - fixvalue = 0; - else if(ofelim < 0) - fixvalue = lp->orig_upbo[lp->rows+ix]; - else - fixvalue = lp->orig_lowbo[lp->rows+ix]; - if(my_infinite(lp, fixvalue)) - status = presolve_setstatus(psdata, UNBOUNDED); - else if(!presolve_colfix(psdata, ix, fixvalue, TRUE, &iVarFixed)) - status = presolve_setstatus(psdata, INFEASIBLE); - else - presolve_colremove(psdata, ix, TRUE); - } - - else if(my_infinite(lp, lp->orig_lowbo[lp->rows+ii])) { /* Case 2 */ - /* Do nothing */ - } - - else { /* Case 3 */ -#if 0 - /* Do nothing */ -#else - if(ofelim >= 0) { - fixvalue = lp->orig_lowbo[lp->rows+ix]; - lp->orig_upbo[lp->rows+ii] += scale * (lp->orig_upbo[lp->rows+ix] - fixvalue); - } - else { - fixvalue = lp->orig_upbo[lp->rows+ix]; - lp->orig_upbo[lp->rows+ii] -= scale * (fixvalue - lp->orig_lowbo[lp->rows+ix]); - } - if(my_infinite(lp, fixvalue)) - status = presolve_setstatus(psdata, UNBOUNDED); - else if(!presolve_colfix(psdata, ix, fixvalue, TRUE, &iVarFixed)) - status = presolve_setstatus(psdata, INFEASIBLE); - else - presolve_colremove(psdata, ix, TRUE); -#ifdef xxParanoia - if(presolve_rowlengthdebug(psdata) > 0) - report(lp, SEVERE, "presolve_aggregate: Invalid row count\n"); -#endif - psdata->forceupdate = TRUE; -#endif - } - } - je++; - } - jb = je; - } - FREE(QSagg); - } - - /* Clear the non-zero row values ahead of the next row candidate */ - if(ib + 1 < n) { - ie = mat->col_end[i-1]; - ii = mat->col_end[i]; - for(; ie < ii; ie++) - colvalues[COL_MAT_ROWNR(ie)] = 0; - } - } -Finish: - FREE(QScand); - FREE(colvalues); - FREE(coldel); - - (*nVarsFixed) += iVarFixed; - (*nSum) += iVarFixed; - - return( status ); -} - -STATIC int presolve_makesparser(presolverec *psdata, int *nCoeffChanged, int *nConRemove, int *nVarFixed, int *nSum) -{ - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - MYBOOL chsign; - int i, ii, ib, ix, k, n, jb, je, jl, jjb, jje, jjl, jx, jjx, item, itemEQ, - *nzidx = NULL, status = RUNNING, iObjChanged = 0, iCoeffChanged = 0, iConRemove = 0; - REAL test, ratio, value, valueEQ, *valptr; - LLrec *EQlist = NULL; - UNIONTYPE QSORTrec *QS = (UNIONTYPE QSORTrec *) calloc(lp->rows, sizeof(*QS)); - - /* Check if we were able to obtain working memory */ - if((QS == NULL) || (psdata->rows->varmap->count == 0) || (psdata->EQmap->count == 0)) - return( status); - - /* Sort rows in 1) increasing order of start index, 2) decreasing length, and - 3) non-equalities (i.e. equalities last) */ - n = 0; - for(i = firstActiveLink(psdata->rows->varmap); i != 0; i = nextActiveLink(psdata->rows->varmap, i)) { - k = presolve_rowlength(psdata, i); - if(k >= 2) { - item = 0; - ii = presolve_nextcol(psdata, i, &item); -#ifdef Paranoia - if((ii < 0) || (item == 0)) { - report(lp, SEVERE, "presolve_makesparser: Unexpected zero-length row %d\n", i); - continue; - } -#endif - QS[n].int4.intval = my_chsign(is_constr_type(lp, i, EQ), i); - QS[n].int4.intpar1 = ROW_MAT_COLNR(ii); - QS[n].int4.intpar2 = k; - n++; - } - } - if(n <= 1) { - FREE(QS); - return( status ); - } - QS_execute(QS, n, (findCompare_func *) compSparsity, NULL); - - /* Create associated sorted map of indeces to equality constraints; - note that we need to have a unit offset for compatibility. */ - allocINT(lp, &nzidx, lp->columns + 1, FALSE); - createLink(lp->rows, &EQlist, NULL); - for(ib = 0; ib < n; ib++) { - i = QS[ib].int4.intval; - if(i < 0) - appendLink(EQlist, ib + 1); - } - - /* Loop over all equality masks */ - for(ix = firstActiveLink(EQlist); ix != 0; ) { - - /* Get row starting and ending positions of the mask */ - ii = abs(QS[ix-1].int4.intval); - jjb = QS[ix-1].int4.intpar1; - jje = presolve_lastcol(psdata, ii); - jje = ROW_MAT_COLNR(jje); - jjl = QS[ix-1].int4.intpar2; - - /* Scan the OF */ - i = 0; - chsign = is_chsign(lp, i); - test = ratio = 0.0; - itemEQ = 0; - nzidx[0] = 0; - while(((jjx = presolve_nextcol(psdata, ii, &itemEQ)) >= 0) && /*(itemEQ > 0) && */ - (fabs(test-ratio) < psdata->epsvalue)) { - valueEQ = ROW_MAT_VALUE(jjx); - if(valueEQ == 0) - continue; - k = ROW_MAT_COLNR(jjx); - value = lp->orig_obj[k]; - if(fabs(value) < psdata->epsvalue) - break; - if(ratio == 0.0) { - test = ratio = value / valueEQ; - } - else - test = value / valueEQ; - /* Store nz index */ - nzidx[++nzidx[0]] = k; - } - - /* We were successful if the equality was completely traversed; we will - then zero-out the OF coefficients and update the constant term. */ - if((itemEQ == 0) && (nzidx[0] > 0) && (fabs(test-ratio) < psdata->epsvalue)) { - for(k = 1; k <= nzidx[0]; k++) { - /* We should add recovery data for the zero'ed coefficient here */ - jx = nzidx[k]; - value = lp->orig_obj[jx]; - lp->orig_obj[jx] = 0.0; - /* Update counts */ - value = my_chsign(chsign, value); - if(value < 0) { - psdata->rows->negcount[i]--; - psdata->cols->negcount[jx]--; - } - else { - psdata->rows->plucount[i]--; - psdata->cols->plucount[jx]--; - } - iObjChanged++; - } - value = ratio * lp->orig_rhs[ii]; - presolve_adjustrhs(psdata, i, value, psdata->epsvalue); - } - - /* Scan for compatible constraints that can be masked for sparsity elimination */ - for(ib = 1; ib < ix; ib++) { - - /* Get row starting and ending positions of the target constraint */ - i = abs(QS[ib-1].int4.intval); - jb = QS[ib-1].int4.intpar1; - je = presolve_lastcol(psdata, i); - je = ROW_MAT_COLNR(je); - jl = QS[ib-1].int4.intpar2; - - /* Check if there is a window mismatch */ - if((jjb < jb) || (jje > je) || (jjl > jl)) - goto NextEQ; - - /* We have a window match; now check if there is a (scalar) member-by-member - match as well. We approach this in the following manner: - 1) Get first (or next) member of active equality - 2) Loop to matching member in the target constraint, but abandon if no match - 3) Set ratio if this is the first match, otherwise compare ratio and abandon - on mismatch - 4) Go to 1) of there are more elements in the active equality - 5) Proceed to do sparsity elimination if we were successful. */ - chsign = is_chsign(lp, i); - test = ratio = 0.0; - itemEQ = 0; - item = 0; - nzidx[0] = 0; - while(((jjx = presolve_nextcol(psdata, ii, &itemEQ)) >= 0) && /*(itemEQ > 0) &&*/ - (fabs(test-ratio) < psdata->epsvalue)) { - valueEQ = ROW_MAT_VALUE(jjx); - if(valueEQ == 0) - continue; - jx = 0; - jjx = ROW_MAT_COLNR(jjx); - for(k = presolve_nextcol(psdata, i, &item); - (jx < jjx) && (item > 0); - k = presolve_nextcol(psdata, i, &item)) { - jx = ROW_MAT_COLNR(k); - /* Do we have a column index match? */ - if(jx == jjx) { - value = ROW_MAT_VALUE(k); - /* Abandon if we have a zero value */ - if(value == 0) - goto NextEQ; - if(ratio == 0.0) { - test = ratio = value / valueEQ; - } - else - test = value / valueEQ; - /* Store nz index */ - nzidx[++nzidx[0]] = k; - break; - } - /* Give up matching if there is overshooting */ - else if(jx > jjx) - goto NextEQ; - } - } - - /* We were successful if the equality was completely traversed */ - if((itemEQ == 0) && (nzidx[0] > 0) && (fabs(test-ratio) < psdata->epsvalue)) { - - /* Check if we have found parametrically indentical constraints */ - if(presolve_rowlength(psdata, i) == presolve_rowlength(psdata,ii)) { - - value = lp->orig_rhs[i]; - valueEQ = lp->orig_rhs[ii]; - - /* Are they both equalities? */ - if(is_constr_type(lp, i, EQ)) { - /* Determine applicable ratio for the RHS */ - if(fabs(valueEQ) < psdata->epsvalue) { - if(fabs(value) < psdata->epsvalue) - test = ratio; - else - test = lp->infinite; - } - else - test = value / valueEQ; - /* Check for infeasibility */ - if(fabs(test-ratio) > psdata->epsvalue) { - report(lp, NORMAL, "presolve_sparser: Infeasibility of relatively equal constraints %d and %d\n", - i, ii); - status = presolve_setstatus(psdata, INFEASIBLE); - goto Finish; - } - /* Otherwise we can delete a redundant constraint */ - else { - removeLink(EQlist, i); - presolve_rowremove(psdata, i, TRUE); - MEMCOPY(&QS[ib-1], &QS[ib], n-ib); - n--; - iConRemove++; - } - } - /* ... if not, then delete the inequality, since the equality dominates */ - else { - /* First verify feasibility of the RHS */ - if((value+psdata->epsvalue < valueEQ) || - (value-get_rh_range(lp, i)-psdata->epsvalue > valueEQ)) { - report(lp, NORMAL, "presolve_sparser: Infeasibility of relatively equal RHS values for %d and %d\n", - i, ii); - status = presolve_setstatus(psdata, INFEASIBLE); - goto Finish; - } - presolve_rowremove(psdata, i, TRUE); - MEMCOPY(&QS[ib-1], &QS[ib], n-ib); - n--; - iConRemove++; - } - } - - /* Otherwise zero-out the target constraint coefficients and update the RHS */ - else { - for(k = 1; k <= nzidx[0]; k++) { - /* We should add recovery data for the zero'ed coefficient here */ - jjx = nzidx[k]; - jx = ROW_MAT_COLNR(jjx); - valptr = &ROW_MAT_VALUE(jjx); - value = *valptr; - *valptr = 0.0; - /* Update counts */ - value = my_chsign(chsign, value); - if(value < 0) { - psdata->rows->negcount[i]--; - psdata->cols->negcount[jx]--; - } - else { - psdata->rows->plucount[i]--; - psdata->cols->plucount[jx]--; - } - iCoeffChanged++; - } - value = ratio * lp->orig_rhs[ii]; - presolve_adjustrhs(psdata, i, value, psdata->epsvalue); - } - } - - } - /* Get next equality index */ -NextEQ: - ix = nextActiveLink(EQlist, ix); - } - -Finish: - FREE(QS); - freeLink(&EQlist); - FREE(nzidx); - - /* Let us condense the matrix if we modified the constraint matrix */ - if(iCoeffChanged > 0) { - mat->row_end_valid = FALSE; - mat_zerocompact(mat); - presolve_validate(psdata, TRUE); -#ifdef PresolveForceUpdateMax - mat_computemax(mat /* , FALSE */); -#endif - psdata->forceupdate = TRUE; - } - - (*nConRemove) += iConRemove; - (*nCoeffChanged) += iCoeffChanged + iObjChanged; - (*nSum) += iCoeffChanged + iObjChanged + iConRemove; - - return( status ); -} - -STATIC int presolve_SOS1(presolverec *psdata, int *nCoeffChanged, int *nConRemove, int *nVarFixed, int *nSOS, int *nSum) -{ - lprec *lp = psdata->lp; - MYBOOL candelete, SOS_GUBactive = FALSE; - int iCoeffChanged = 0, iConRemove = 0, iSOS = 0, - i,ix,iix, j,jx,jjx, status = RUNNING; - REAL Value1; - MATrec *mat = lp->matA; - - for(i = lastActiveLink(psdata->rows->varmap); i > 0; ) { - candelete = FALSE; - Value1 = get_rh(lp, i); - jx = get_constr_type(lp, i); - if((Value1 == 1) && (presolve_rowlength(psdata, i) >= MIN_SOS1LENGTH) && - ((SOS_GUBactive && (jx != GE)) || (!SOS_GUBactive && (jx == LE)))) { - jjx = mat->row_end[i-1]; - iix = mat->row_end[i]; - for(; jjx < iix; jjx++) { - j = ROW_MAT_COLNR(jjx); - if(!isActiveLink(psdata->cols->varmap, j)) - continue; - if(!is_binary(lp, j) || (ROW_MAT_VALUE(jjx) != 1)) - break; - } - if(jjx >= iix) { - char SOSname[16]; - - /* Define a new SOS instance */ - ix = SOS_count(lp) + 1; - sprintf(SOSname, "SOS_%d", ix); - ix = add_SOS(lp, SOSname, 1, ix, 0, NULL, NULL); - if(jx == EQ) - SOS_set_GUB(lp->SOS, ix, TRUE); - Value1 = 0; - jjx = mat->row_end[i-1]; - for(; jjx < iix; jjx++) { - j = ROW_MAT_COLNR(jjx); - if(!isActiveLink(psdata->cols->varmap, j)) - continue; - Value1 += 1; - append_SOSrec(lp->SOS->sos_list[ix-1], 1, &j, &Value1); - } - candelete = TRUE; - iSOS++; - } - } - - /* Get next row and do the deletion of the previous, if indicated */ - ix = i; - i = prevActiveLink(psdata->rows->varmap, i); - if(candelete) { - presolve_rowremove(psdata, ix, TRUE); - iConRemove++; - } - } - if(iSOS) - report(lp, DETAILED, "presolve_SOS1: Converted %5d constraints to SOS1.\n", iSOS); - clean_SOSgroup(lp->SOS, (MYBOOL) (iSOS > 0)); - - (*nCoeffChanged) += iCoeffChanged; - (*nConRemove) += iConRemove; - (*nSOS) += iSOS; - (*nSum) += iCoeffChanged+iConRemove+iSOS; - - return( status ); -} - -STATIC int presolve_boundconflict(presolverec *psdata, int baserowno, int colno) -{ - REAL Value1, Value2; - lprec *lp = psdata->lp; - MATrec *mat = lp->matA; - int ix, item = 0, - status = RUNNING; - - if(baserowno <= 0) do { - ix = presolve_nextrow(psdata, colno, &item); - if(ix < 0) - return( status ); - baserowno = COL_MAT_ROWNR(ix); - } while(presolve_rowlength(psdata, baserowno) != 1); - Value1 = get_rh_upper(lp, baserowno), - Value2 = get_rh_lower(lp, baserowno); - - if(presolve_singletonbounds(psdata, baserowno, colno, &Value2, &Value1, NULL)) { - int iix; - item = 0; - for(ix = presolve_nextrow(psdata, colno, &item); - ix >= 0; ix = presolve_nextrow(psdata, colno, &item)) { - iix = COL_MAT_ROWNR(ix); - if((iix != baserowno) && - (presolve_rowlength(psdata, iix) == 1) && - !presolve_altsingletonvalid(psdata, iix, colno, Value2, Value1)) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - } - } - else - status = presolve_setstatus(psdata, INFEASIBLE); - return( status ); -} - -STATIC int presolve_columns(presolverec *psdata, int *nCoeffChanged, int *nConRemove, int *nVarFixed, int *nBoundTighten, int *nSum) -{ - lprec *lp = psdata->lp; - MYBOOL candelete, isOFNZ, unbounded, - probefix = is_presolve(lp, PRESOLVE_PROBEFIX), -#if 0 - probereduce = is_presolve(lp, PRESOLVE_PROBEREDUCE), -#endif - colfixdual = is_presolve(lp, PRESOLVE_COLFIXDUAL); - int iCoeffChanged = 0, iConRemove = 0, iVarFixed = 0, iBoundTighten = 0, - status = RUNNING, ix, j, countNZ, item; - REAL Value1; - - for(j = firstActiveLink(psdata->cols->varmap); (j != 0) && (status == RUNNING); ) { - - /* Don't presolve members SOS'es */ - if(SOS_is_member(lp->SOS, 0, j)) { - j = nextActiveLink(psdata->cols->varmap, j); - continue; - } - - /* Initialize */ - countNZ = presolve_collength(psdata, j); - isOFNZ = isnz_origobj(lp, j); - Value1 = get_lowbo(lp, j); - unbounded = is_unbounded(lp, j); - - /* Clear unnecessary semicont-definitions */ - if((lp->sc_vars > 0) && (Value1 == 0) && is_semicont(lp, j)) - set_semicont(lp, j, FALSE); - - candelete = FALSE; - item = 0; - ix = lp->rows + j; - - /* Check if the variable is unused */ - if((countNZ == 0) && !isOFNZ) { - if(Value1 != 0) - report(lp, DETAILED, "presolve_columns: Eliminated unused variable %s\n", - get_col_name(lp,j)); - candelete = TRUE; - } - - /* Check if the variable has a cost, but is not limited by constraints */ - else if((countNZ == 0) && isOFNZ) { - if(lp->orig_obj[j] < 0) - Value1 = get_upbo(lp, j); - if(fabs(Value1) >= lp->infinite) { - report(lp, DETAILED, "presolve_columns: Unbounded variable %s\n", - get_col_name(lp,j)); - status = presolve_setstatus(psdata, UNBOUNDED); - } - else { - /* Fix the value at its best bound */ - report(lp, DETAILED, "presolve_columns: Eliminated trivial variable %s fixed at %g\n", - get_col_name(lp,j), Value1); - candelete = TRUE; - } - } - - /* Check if the variable can be eliminated because it is fixed */ - else if(isOrigFixed(lp, ix)) { - if(countNZ > 0) { - status = presolve_boundconflict(psdata, -1, j); - if(status != RUNNING) - break; - } - report(lp, DETAILED, "presolve_columns: Eliminated variable %s fixed at %g\n", - get_col_name(lp,j), Value1); - candelete = TRUE; - } - -#if 0 - /* Merge OF-constraint column doubleton in equality constraint (if it has - not been captured by the singleton free variable rule above) */ - else if((countNZ == 1) && isOFNZ && - ((i = presolve_nextrow(psdata, j, &item)) >= 0) && - is_constr_type(lp, i = COL_MAT_ROWNR(i), EQ)) { - MATrec *mat = lp->matA; - - /* Merge the constraint into the OF */ - Value1 = lp->orig_obj[j] / get_mat(lp, i, j); - for(jx = mat->row_end[i-1]; jx < mat->row_end[i]; jx++) { - jjx = ROW_MAT_COLNR(jx); - lp->orig_obj[jjx] -= Value1 * ROW_MAT_VALUE(jx); - } - Value2 = lp->orig_rhs[i]; - presolve_adjustrhs(psdata, 0, Value1 * Value2, 0.0); - - /* Verify feasibility */ - Value2 /= get_mat(lp, i, j); - if((Value2 < get_lowbo(lp, j)) || (Value2 > get_upbo(lp, j))) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - - /* Do column (and flag row) deletion */ - presolve_rowremove(psdata, i, TRUE); - psdata->forceupdate = TRUE; - iConRemove++; - candelete = TRUE; - } -#endif - /* Look for opportunity to fix column based on the dual */ - else if(colfixdual && presolve_colfixdual(psdata, j, &Value1, &status)) { - if(my_infinite(lp, Value1)) { - report(lp, DETAILED, "presolve_columns: Unbounded variable %s\n", - get_col_name(lp,j)); - status = presolve_setstatus(psdata, UNBOUNDED); - } - else { - /* Fix the value at its best bound */ - report(lp, DETAILED, "presolve_columns: Eliminated dual-zero variable %s fixed at %g\n", - get_col_name(lp,j), Value1); - candelete = TRUE; - } - } - - /* Do probing of binary variables to see if we can fix them */ - else if(probefix && is_binary(lp, j) && - presolve_probefix01(psdata, j, &Value1)) { - report(lp, DETAILED, "presolve_columns: Fixed binary variable %s at %g\n", - get_col_name(lp,j), Value1); - candelete = TRUE; - } -#if 0 - /* Do probing of binary variables to see if we can tighten their coefficients */ - else if(probereduce && is_binary(lp, j) && - (ix = presolve_probetighten01(psdata, j) > 0)) { - report(lp, DETAILED, "presolve_columns: Tightened coefficients for binary variable %s in %d rows\n", - get_col_name(lp,j), ix); - iCoeffChanged += ix; - psdata->forceupdate = TRUE; - } -#endif - - /* Perform fixing and deletion, if indicated */ - if(candelete) { - - /* If we have a SOS1 member variable fixed at a non-zero value, then we - must fix the other member variables at zero and delete the SOS(es) */ - if((Value1 != 0) && SOS_is_member(lp->SOS, 0, j)) { - ix = iVarFixed; - if(!presolve_fixSOS1(psdata, j, Value1, &iConRemove, &iVarFixed)) - status = presolve_setstatus(psdata, INFEASIBLE); - if(iVarFixed > ix) - psdata->forceupdate = TRUE; - break; - } - else { - if(!presolve_colfix(psdata, j, Value1, TRUE, &iVarFixed)) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - j = presolve_colremove(psdata, j, TRUE); - } - } - else - j = nextActiveLink(psdata->cols->varmap, j); - } - - /* Remove any "hanging" empty row and columns */ - if(status == RUNNING) - status = presolve_shrink(psdata, &iConRemove, &iVarFixed); - - (*nCoeffChanged) += iCoeffChanged; - (*nConRemove) += iConRemove; - (*nVarFixed) += iVarFixed; - (*nBoundTighten) += iBoundTighten; - (*nSum) += iCoeffChanged+iConRemove+iVarFixed+iBoundTighten; - - return( status ); -} - -STATIC int presolve_freeandslacks(presolverec *psdata, int *nCoeffChanged, int *nConRemove, int *nVarFixed, int *nSum) -{ - lprec *lp = psdata->lp; - MYBOOL isOFNZ, unbounded, - impliedfree = is_presolve(lp, PRESOLVE_IMPLIEDFREE), - impliedslack = is_presolve(lp, PRESOLVE_IMPLIEDSLK); - int iCoeffChanged = 0, iConRemove = 0, iVarFixed = 0, - status = RUNNING, i, ix, j, countNZ; - REAL coeff_bl, coeff_bu; - MATrec *mat = lp->matA; - - if(impliedfree || impliedslack) - for(j = firstActiveLink(psdata->cols->varmap); j != 0; ) { - - /* Check and initialize */ - if((presolve_collength(psdata, j) != 1) || - is_int(lp, j) || is_semicont(lp, j) || - !presolve_candeletevar(psdata, j)) { - j = nextActiveLink(psdata->cols->varmap, j); - continue; - } - ix = 0; - i = COL_MAT_ROWNR(presolve_nextrow(psdata, j, &ix)); - isOFNZ = isnz_origobj(lp, j); - countNZ = presolve_rowlength(psdata, i); - coeff_bu = get_upbo(lp, j); - coeff_bl = get_lowbo(lp, j); - unbounded = my_infinite(lp, coeff_bl) && my_infinite(lp, coeff_bu); - ix = lp->rows + j; - - /* Eliminate singleton free variable and its associated constraint */ - if(impliedfree && unbounded && - presolve_impliedcolfix(psdata, i, j, TRUE)) { - report(lp, DETAILED, "presolve_freeandslacks: Eliminated free variable %s and row %s\n", - get_col_name(lp, j), get_row_name(lp, i)); - presolve_rowremove(psdata, i, TRUE); - iConRemove++; - j = presolve_colremove(psdata, j, TRUE); - iVarFixed++; - } - - /* Check for implied slack variable in equality constraint */ - else if(impliedslack && - (countNZ > 1) && - is_constr_type(lp, i, EQ) && - presolve_impliedcolfix(psdata, i, j, FALSE)) { - report(lp, DETAILED, "presolve_freeandslacks: Eliminated implied slack variable %s via row %s\n", - get_col_name(lp, j), get_row_name(lp, i)); - psdata->forceupdate = TRUE; - j = presolve_colremove(psdata, j, TRUE); - iVarFixed++; - } - - /* Check for implied (generalized) slack variable in inequality constraint */ - else if(impliedslack && !isOFNZ && - my_infinite(lp, coeff_bu) && /* Consider removing this test */ -#if 0 /* Force zero-bounded implicit slack */ - (coeff_bl == 0)) && -#else - !my_infinite(lp, coeff_bl) && -#endif - (countNZ > 1) && - !is_constr_type(lp, i, EQ)) { - REAL *target, - ValueA = COL_MAT_VALUE(presolve_lastrow(psdata, j)); -#if 0 - coeff_bu = get_rh_upper(lp, i); - coeff_bl = get_rh_lower(lp, i); - if(!presolve_singletonbounds(psdata, i, j, &coeff_bl, &coeff_bu, &ValueA)) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } -#endif - if((coeff_bl != 0) && !my_infinite(lp, coeff_bl) && !my_infinite(lp, coeff_bu)) - coeff_bu -= coeff_bl; - - /* If the coefficient is negative, reduce the lower bound / increase range */ - if(ValueA > 0) { - target = &lp->orig_upbo[i]; - if(!my_infinite(lp, *target)) { - if(my_infinite(lp, coeff_bu)) { - *target = lp->infinite; - psdata->forceupdate = TRUE; - } - else { - *target += ValueA * coeff_bu; - *target = presolve_roundrhs(lp, *target, FALSE); - } - } - } - /* Otherwise see if the upper bound should be changed */ - else { - target = &lp->orig_rhs[i]; - if(my_infinite(lp, coeff_bu) || my_infinite(lp, *target)) { - /* Do we suddenly find that the constraint becomes redundant? (e226.mps) */ - if(my_infinite(lp, lp->orig_upbo[i])) { - presolve_rowremove(psdata, i, TRUE); - iConRemove++; - } - /* Or does the upper bound of a ranged constraint become Inf? */ - else { - *target -= lp->orig_upbo[i]; - *target = -(*target); - mat_multrow(mat, i, -1); - lp->orig_upbo[i] = lp->infinite; - psdata->forceupdate = TRUE; - } - } - else { - *target -= ValueA * coeff_bu; - *target = presolve_roundrhs(lp, *target, FALSE); - } - } - presolve_colfix(psdata, j, coeff_bl, TRUE, &iVarFixed); - report(lp, DETAILED, "presolve_freeandslacks: Eliminated duplicate slack variable %s via row %s\n", - get_col_name(lp, j), get_row_name(lp, i)); - j = presolve_colremove(psdata, j, TRUE); - } - - /* Go to next column */ - else - j = nextActiveLink(psdata->cols->varmap, j); - } - - (*nCoeffChanged) += iCoeffChanged; - (*nConRemove) += iConRemove; - (*nVarFixed) += iVarFixed; - (*nSum) += iCoeffChanged+iConRemove+iVarFixed; - - return( status ); -} - -STATIC int presolve_preparerows(presolverec *psdata, int *nBoundTighten, int *nSum) -{ - lprec *lp = psdata->lp; - MYBOOL impliedfree = is_presolve(lp, PRESOLVE_IMPLIEDFREE), - tightenbounds = is_presolve(lp, PRESOLVE_BOUNDS); - int iRangeTighten = 0, iBoundTighten = 0, status = RUNNING, i, j; - REAL losum, upsum, lorhs, uprhs, epsvalue = psdata->epsvalue; - MATrec *mat = lp->matA; - - for(i = lastActiveLink(psdata->rows->varmap); i > 0; i = prevActiveLink(psdata->rows->varmap, i)) { - - /* First identify any full row infeasibilities */ - j = presolve_rowlengthex(psdata, i); -#ifdef Paranoia - if(!presolve_testrow(psdata, nextActiveLink(psdata->rows->varmap, i))) { -#else - if((j > 1) && !psdata->forceupdate && !presolve_rowfeasible(psdata, i, FALSE)) { -#endif - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - - /* Do bound (LHS) or constraint range (RHS) tightening if we will later identify - implied free variables (tends to produce degeneracy otherwise) */ - if(impliedfree && (j > 1) && mat_validate(mat)){ - - /* Look for opportunity to tighten constraint bounds (and check for feasibility again) */ - presolve_range(lp, i, psdata->rows, &losum, &upsum); - lorhs = get_rh_lower(lp, i); - uprhs = get_rh_upper(lp, i); - if((losum > MIN(upsum, uprhs)+epsvalue) || - (upsum < MAX(losum, lorhs)-epsvalue)) { - report(lp, NORMAL, "presolve_preparerows: Variable bound / constraint value infeasibility in row %s.\n", - get_row_name(lp, i)); - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - - if(losum > lorhs+epsvalue) { - set_rh_lower(lp, i, presolve_roundrhs(lp, losum, TRUE)); - iRangeTighten++; - } - if(upsum < uprhs-epsvalue) { - set_rh_upper(lp, i, presolve_roundrhs(lp, upsum, FALSE)); - iRangeTighten++; - } - } - - /* Seek to tighten bounds on individual variables */ - if(tightenbounds && mat_validate(mat)) { -#if 1 - if(j > 1) - status = presolve_rowtighten(psdata, i, &iBoundTighten, FALSE); -#else - if((MIP_count(lp) > 0) && (j > 1)) - status = presolve_rowtighten(psdata, i, &iBoundTighten, TRUE); -#endif - } - - /* Look for opportunity to convert ranged constraint to equality-type */ - if(!is_constr_type(lp, i, EQ) && (get_rh_range(lp, i) < epsvalue)) { - presolve_setEQ(psdata, i); - iRangeTighten++; - } - } - - psdata->forceupdate |= (MYBOOL) (iBoundTighten > 0); - (*nBoundTighten) += iBoundTighten+iRangeTighten; - (*nSum) += iBoundTighten+iRangeTighten; - - return( status ); -} - -STATIC int presolve_rows(presolverec *psdata, int *nCoeffChanged, int *nConRemove, int *nVarFixed, int *nBoundTighten, int *nSum) -{ - lprec *lp = psdata->lp; - MYBOOL candelete; - int iCoeffChanged = 0, iConRemove = 0, iVarFixed = 0, iBoundTighten = 0, - status = RUNNING, i,ix, j,jx, item; - REAL Value1, Value2, losum, upsum, lorhs, uprhs, epsvalue = psdata->epsvalue; - MATrec *mat = lp->matA; - - for(i = lastActiveLink(psdata->rows->varmap); (i > 0) && (status == RUNNING); ) { - - candelete = FALSE; - - /* First identify any full row infeasibilities - Note: Handle singletons below to ensure that conflicting multiple singleton - rows with this variable do not provoke notice of infeasibility */ - j = presolve_rowlengthex(psdata, i); - if((j > 1) && - !psdata->forceupdate && !presolve_rowfeasible(psdata, i, FALSE)) { - status = presolve_setstatus(psdata, INFEASIBLE); - break; - } - presolve_range(lp, i, psdata->rows, &losum, &upsum); - lorhs = get_rh_lower(lp, i); - uprhs = get_rh_upper(lp, i); -#ifdef Paranoia - if((losum>uprhs+epsvalue) || (upsum= -epsvalue)) { -#else /* Version that deletes bound-fixed columns here */ - if((j == 1) && (uprhs-lorhs >= -epsvalue)) { -#endif - item = 0; - jx = presolve_nextcol(psdata, i, &item); - j = ROW_MAT_COLNR(jx); - - /* Make sure we don't have conflicting other singleton rows with this variable */ - Value1 = lp->infinite; - Value2 = -Value1; - if(presolve_collength(psdata, j) > 1) - status = presolve_boundconflict(psdata, i, j); - else if(is_constr_type(lp, i, EQ)) { - Value2 = ROW_MAT_VALUE(jx); - Value1 = lp->orig_rhs[i] / Value2; - if(Value2 < 0) - swapREAL(&losum, &upsum); - if((Value1 < losum / my_if(my_infinite(lp, losum), my_sign(Value2), Value2) - epsvalue) || - (Value1 > upsum / my_if(my_infinite(lp, upsum), my_sign(Value2), Value2) + epsvalue)) - status = presolve_setstatus(psdata, INFEASIBLE); - Value2 = Value1; - } - - /* Proceed to fix and remove variable (if it is not a SOS member) */ - if(status == RUNNING) { - if((fabs(Value2-Value1) < epsvalue) && (fabs(Value2) > epsvalue)) { - MYBOOL isSOS = (MYBOOL) (SOS_is_member(lp->SOS, 0, j) != FALSE), - deleteSOS = isSOS && presolve_candeletevar(psdata, j); - if((Value1 != 0) && deleteSOS) { - if(!presolve_fixSOS1(psdata, j, Value1, &iConRemove, &iVarFixed)) - status = presolve_setstatus(psdata, INFEASIBLE); - psdata->forceupdate = TRUE; - } - else { - if(!presolve_colfix(psdata, j, Value1, (MYBOOL) !isSOS, NULL)) - status = presolve_setstatus(psdata, INFEASIBLE); - else if(isSOS && !deleteSOS) - iBoundTighten++; - else { - presolve_colremove(psdata, j, TRUE); - iVarFixed++; - } - } - } - else - status = presolve_colsingleton(psdata, i, j, &iBoundTighten); - } - if(status == INFEASIBLE) { - break; - } - if(psdata->forceupdate != AUTOMATIC) { - /* Store dual recovery information and code for deletion */ - presolve_storeDualUndo(psdata, i, j); - candelete = TRUE; - } - } - - /* Delete non-empty rows and variables that are completely determined at zero */ - else if((j > 0) /* Only examine non-empty rows, */ - && (fabs(lp->orig_rhs[i]) < epsvalue) /* .. and the current RHS is zero, */ - && ((psdata->rows->plucount[i] == 0) || - (psdata->rows->negcount[i] == 0)) /* .. and the parameter signs are all equal, */ - && (psdata->rows->pluneg[i] == 0) /* .. and no (quasi) free variables, */ - && (is_constr_type(lp, i, EQ) -#ifdef FindImpliedEqualities - || (fabs(lorhs-upsum) < epsvalue) /* Convert to equalities */ - || (fabs(uprhs-losum) < epsvalue) /* Convert to equalities */ -#endif - ) - ) { - - /* Delete the columns we can delete */ - status = presolve_rowfixzero(psdata, i, &iVarFixed); - - /* Then delete the row, which is redundant */ - if(status == RUNNING) - candelete = TRUE; - } - - - /* Check if we have a constraint made redundant through bounds on individual - variables; such constraints are often referred to as "forcing constraints" */ - else if((losum >= lorhs-epsvalue) && - (upsum <= uprhs+epsvalue)) { - - /* Check if we can also fix all the variables */ - if(fabs(losum-upsum) < epsvalue) { - item = 0; - jx = presolve_nextcol(psdata, i, &item); - while((status == RUNNING) && (jx >= 0)) { - j = ROW_MAT_COLNR(jx); - Value1 = get_lowbo(lp, j); - if(presolve_colfix(psdata, j, Value1, TRUE, &iVarFixed)) { - presolve_colremove(psdata, j, TRUE); - iVarFixed++; - jx = presolve_nextcol(psdata, i, &item); - } - else - status = presolve_setstatus(psdata, INFEASIBLE); - } - } - candelete = TRUE; - } - - /* Get next row and do the deletion of the previous, if indicated */ - ix = i; - i = prevActiveLink(psdata->rows->varmap, i); - if(candelete) { - presolve_rowremove(psdata, ix, TRUE); - iConRemove++; - } - } - - /* Remove any "hanging" empty row and columns */ - if(status == RUNNING) - status = presolve_shrink(psdata, &iConRemove, &iVarFixed); - - (*nCoeffChanged) += iCoeffChanged; - (*nConRemove) += iConRemove; - (*nVarFixed) += iVarFixed; - (*nBoundTighten) += iBoundTighten; - (*nSum) += iCoeffChanged+iConRemove+iVarFixed+iBoundTighten; - - return( status ); -} - -/* Top level presolve routine */ -STATIC int presolve(lprec *lp) -{ - int status = RUNNING, - i, j = 0, jx = 0, jjx = 0, k, oSum, - iCoeffChanged = 0, iConRemove = 0, iVarFixed = 0, iBoundTighten = 0, iSOS = 0, iSum = 0, - nCoeffChanged = 0, nConRemove = 0, nVarFixed = 0, nBoundTighten = 0, nSOS = 0, nSum = 0; - REAL Value1, Value2, initrhs0 = lp->orig_rhs[0]; - presolverec *psdata = NULL; - MATrec *mat = lp->matA; - -#if 0 - lp->do_presolve = PRESOLVE_ROWS; - report(lp, IMPORTANT, "presolve: Debug override of presolve setting to %d\n", lp->do_presolve); -#endif - - /* Lock the variable mapping arrays and counts ahead of any row/column - deletion or creation in the course of presolve, solvelp or postsolve */ - if(!lp->varmap_locked) - varmap_lock(lp); - - /* Check if we have already done presolve */ - mat_validate(mat); - if(lp->wasPresolved) { - if(SOS_count(lp) > 0) { - SOS_member_updatemap(lp->SOS); - make_SOSchain(lp, (MYBOOL) ((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE)); - } - if((lp->solvecount > 1) && (lp->bb_level < 1) && - ((lp->scalemode & SCALE_DYNUPDATE) != 0)) - auto_scale(lp); - if(!lp->basis_valid) { - crash_basis(lp); - report(lp, DETAILED, "presolve: Had to repair broken basis.\n"); - } - lp->timepresolved = timeNow(); - return(status); - } - - /* Produce original model statistics (do hoops to produce correct stats if we have SOS'es) */ - i = SOS_count(lp); - if(i > 0) { - SOS_member_updatemap(lp->SOS); - lp->sos_vars = SOS_memberships(lp->SOS, 0); - } - REPORT_modelinfo(lp, TRUE, "SUBMITTED"); - report(lp, NORMAL, " \n"); - if(i > 0) - lp->sos_vars = 0; - - /* Finalize basis indicators; if no basis was created earlier via - set_basis or crash_basis then simply set the default basis. */ - if(!lp->basis_valid) - lp->var_basic[0] = AUTOMATIC; /* Flag that we are presolving */ - -#if 0 -write_lp(lp, "test_in.lp"); /* Write to lp-formatted file for debugging */ -/*write_mps(lp, "test_in.mps");*/ /* Write to lp-formatted file for debugging */ -#endif - - /* Update inf norms and check for potential factorization trouble */ - mat_computemax(mat /*, FALSE */); -#if 0 - Value1 = fabs(lp->negrange); - if(is_obj_in_basis(lp) && (mat->dynrange < Value1) && vec_computeext(lp->orig_obj, 1, lp->columns, TRUE, &i, &j)) { - - /* Compute relative scale metric */ - Value2 = fabs(lp->orig_obj[j]/lp->orig_obj[i]) / mat->dynrange; - if(Value2 < 1.0) - Value2 = 1.0 / Value2; - - /* Determine if we should alert modeler and possibly move the OF out of the coefficient matrix */ - if((Value2 > Value1) /* Case with extreme scale difference */ -#if 1 - || (mat->dynrange == 1.0) /* Case where we have an all-unit coefficient matrix, possibly totally unimodular */ -#endif - ) - if((lp->simplex_strategy & SIMPLEX_DYNAMIC) > 0) { - clear_action(&lp->algopt, ALGOPT_OBJINBASIS); - report(lp, NORMAL, "Moved objective function out of the basis matrix to enhance factorization accuracy.\n"); - } - else if(mat->dynrange > 1.0) - report(lp, IMPORTANT, "Warning: Objective/matrix coefficient magnitude differences will cause inaccuracy!\n"); - } -#endif - - /* Do traditional simple presolve */ - yieldformessages(lp); - if((lp->do_presolve & PRESOLVE_LASTMASKMODE) == PRESOLVE_NONE) { - mat_checkcounts(mat, NULL, NULL, TRUE); - i = 0; - } - else { - - if(lp->full_solution == NULL) - allocREAL(lp, &lp->full_solution, lp->sum_alloc+1, TRUE); - - /* Identify infeasible SOS'es prior to any pruning */ - j = 0; - for(i = 1; i <= SOS_count(lp); i++) { - k = SOS_infeasible(lp->SOS, i); - if(k > 0) { - presolverec psdata; - - psdata.lp = lp; - report(lp, NORMAL, "presolve: Found SOS %d (type %d) to be range-infeasible on variable %d\n", - i, SOS_get_type(lp->SOS, i), k); - status = presolve_setstatus(&psdata, INFEASIBLE); - j++; - } - } - if(j > 0) - goto Finish; - - /* Create and initialize the presolve data structures */ - psdata = presolve_init(lp); - - /* Reentry point for the outermost, computationally expensive presolve loop */ - psdata->outerloops = 0; - do { - psdata->outerloops++; - iCoeffChanged = 0; - iConRemove = 0; - iVarFixed = 0; - iBoundTighten = 0; - iSOS = 0; - oSum = nSum; - - /* Do the middle elimination loop */ - do { - psdata->middleloops++; - nSum += iSum; - iSum = 0; - - /* Accumulate constraint bounds based on bounds on individual variables. */ - j = 0; - while(presolve_statuscheck(psdata, &status) && psdata->forceupdate) { - psdata->forceupdate = FALSE; - /* Update sums, but limit iteration count to avoid possible - "endless" loops with only marginal bound improvements */ - if(presolve_updatesums(psdata) && (j < MAX_PSBOUNDTIGHTENLOOPS)) { - /* Do row preparation useful for subsequent column and row presolve operations */ - if((psdata->outerloops == 1) && (psdata->middleloops == 1)) - status = presolve_preparerows(psdata, &iBoundTighten, &iSum); - nBoundTighten += iBoundTighten; - iBoundTighten = 0; - nSum += iSum; - iSum = 0; - j++; - if(status != RUNNING) - report(lp, NORMAL, "presolve: Break after bound tightening iteration %d.\n", j); - } - } - if(status != RUNNING) - break; - - /* Do the relatively cheap innermost elimination loop */ - do { - - psdata->innerloops++; - nSum += iSum; - iSum = 0; - - /* Eliminate empty rows, convert row singletons to bounds, - tighten bounds, and remove always satisfied rows */ - if(presolve_statuscheck(psdata, &status) && - is_presolve(lp, PRESOLVE_ROWS)) - status = presolve_rows(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iBoundTighten, &iSum); - - /* Eliminate empty or fixed columns (including trivial OF column singletons) */ - if(presolve_statuscheck(psdata, &status) && - is_presolve(lp, PRESOLVE_COLS)) - status = presolve_columns(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iBoundTighten, &iSum); - - /* Presolve SOS'es if possible (always do this) */ - if(presolve_statuscheck(psdata, &status)) - status = presolve_redundantSOS(psdata, &iBoundTighten, &iSum); - - } while((status == RUNNING) && (iSum > 0)); - if(status != RUNNING) - break; - - /* Merge compatible similar rows; loop backwards over every row */ - if(presolve_statuscheck(psdata, &status) && - (psdata->outerloops == 1) && (psdata->middleloops <= MAX_PSMERGELOOPS) && - is_presolve(lp, PRESOLVE_MERGEROWS)) - status = presolve_mergerows(psdata, &iConRemove, &iSum); - - /* Eliminate dominated rows */ - if(presolve_statuscheck(psdata, &status) && - is_presolve(lp, PRESOLVE_ROWDOMINATE)) - presolve_rowdominance(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSum); - - /* See if we can convert some constraints to SOSes (only SOS1 handled) */ - if(presolve_statuscheck(psdata, &status) && (MIP_count(lp) > 0) && - is_presolve(lp, PRESOLVE_SOS)) - status = presolve_SOS1(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSOS, &iSum); - - /* Eliminate dominated columns in set coverage models */ - if(presolve_statuscheck(psdata, &status) && (lp->int_vars > 1) && - is_presolve(lp, PRESOLVE_COLDOMINATE)) - presolve_coldominance01(psdata, &iConRemove, &iVarFixed, &iSum); - - /* Aggregate compatible columns */ - if(presolve_statuscheck(psdata, &status) && /*TRUE ||*/ - is_presolve(lp, PRESOLVE_AGGREGATE)) - presolve_aggregate(psdata, &iConRemove, &iVarFixed, &iSum); - - /* Eliminate free variables and implied slacks */ - if(presolve_statuscheck(psdata, &status) && -/* !is_presolve(lp, PRESOLVE_ELIMEQ2) && */ - is_presolve(lp, PRESOLVE_IMPLIEDSLK | PRESOLVE_IMPLIEDFREE)) - status = presolve_freeandslacks(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSum); - - } while((status == RUNNING) && (iSum > 0)); - if(status != RUNNING) - break; - - /* Check if we can do elimination of rank-deficient equality constraints */ - if(presolve_statuscheck(psdata, &status) && (psdata->EQmap->count > 1) && - is_presolve(lp, PRESOLVE_LINDEP)) { -#if 0 - REPORT_mat_mmsave(lp, "A.mtx", NULL, FALSE, "Constraint matrix A"); -#endif - presolve_singularities(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSum); - } - - /* Eliminate variable and tighten bounds using 2-element EQs; - note that this involves modifying the coefficients of A and - can therefore be a slow operation. */ - if(presolve_statuscheck(psdata, &status) && - is_presolve(lp, PRESOLVE_ELIMEQ2)) { - jjx = 0; - do { - jjx += iSum; - status = presolve_elimeq2(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSum); - } while((status == RUNNING) && (iSum > jjx)); - iSum = jjx; - -#if 0 - /* Eliminate free variables and implied slacks */ - if(presolve_statuscheck(psdata, &status) && - is_presolve(lp, PRESOLVE_IMPLIEDSLK | PRESOLVE_IMPLIEDFREE)) - status = presolve_freeandslacks(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSum); -#endif - } - - /* Increase A matrix sparsity by discovering common subsets using EQs */ - if(presolve_statuscheck(psdata, &status) && (psdata->EQmap->count > 0) && - is_presolve(lp, PRESOLVE_SPARSER)) - status = presolve_makesparser(psdata, &iCoeffChanged, &iConRemove, &iVarFixed, &iSum); - - /* Do GCD-based coefficient reductions (also does row scaling, - even if no rhs INT truncations are possible) */ - if(presolve_statuscheck(psdata, &status) && (psdata->INTmap->count > 0) && - is_presolve(lp, PRESOLVE_REDUCEGCD)) - if(!presolve_reduceGCD(psdata, &iCoeffChanged, &iBoundTighten, &iSum)) - status = presolve_setstatus(psdata, INFEASIBLE); - - /* Simplify knapsack or set coverage models where OF coefficients are - duplicated in the constraints. At the cost of adding helper columns, this - increases sparsity and facilitates identification of lower and upper bounds. */ - if(presolve_statuscheck(psdata, &status) && - is_presolve(lp, PRESOLVE_KNAPSACK)) { - i = iCoeffChanged; - status = presolve_knapsack(psdata, &iCoeffChanged); - } - - /* Remove any "hanging" empty row and columns */ - if(status == RUNNING) - status = presolve_shrink(psdata, &iConRemove, &iVarFixed); - - nCoeffChanged += iCoeffChanged; - nConRemove += iConRemove; - nVarFixed += iVarFixed; - nBoundTighten += iBoundTighten; - nSOS += iSOS; - nSum += iSum; - - iSum = iConRemove + iVarFixed + iBoundTighten + iCoeffChanged; - if(iSum > 0) - report(lp, NORMAL, "Presolve O:%d -> Reduced rows:%5d, cols:%5d --- changed bnds:%5d, Ab:%5d.\n", - psdata->outerloops, iConRemove, iVarFixed, iBoundTighten, iCoeffChanged); - - /* Do the outermost loop again if we were successful in this presolve sequences */ - } while(presolve_statuscheck(psdata, &status) && - (psdata->forceupdate || (oSum < nSum)) && - (psdata->outerloops < get_presolveloops(lp)) && - (psdata->rows->varmap->count+psdata->cols->varmap->count > 0)); - - /* Finalize presolve */ -#ifdef Paranoia - i = presolve_debugcheck(lp, psdata->rows->varmap, psdata->cols->varmap); - if(i > 0) - report(lp, SEVERE, "presolve: %d internal consistency failure%s\n", i, my_plural_std(i)); - if((SOS_count(lp) > 0) && !presolve_SOScheck(psdata)) - report(lp, SEVERE, "presolve: SOS sparse member mapping problem - part 1\n"); -#endif - /* Perform bound relaxation to reduce chance of degeneracy. */ - if((status == RUNNING) && !is_presolve(lp, PRESOLVE_IMPLIEDFREE)) - jjx = presolve_makefree(psdata); - else - jjx = 0; - - - /* Finalize the presolve */ - if(!presolve_finalize(psdata)) - report(lp, SEVERE, "presolve: Unable to construct internal data representation\n"); - - /* Report summary information */ - i = NORMAL; - iVarFixed = lp->presolve_undo->orig_columns - psdata->cols->varmap->count; - iConRemove = lp->presolve_undo->orig_rows - psdata->rows->varmap->count; - if(nSum > 0) - report(lp, i, "PRESOLVE Elimination loops performed.......... O%d:M%d:I%d\n", - psdata->outerloops, psdata->middleloops, psdata->innerloops); - if(nVarFixed) - report(lp, i, " %8d empty or fixed variables............. %s.\n", nVarFixed, "REMOVED"); - if(nConRemove) - report(lp, i, " %8d empty or redundant constraints....... %s.\n", nConRemove, "REMOVED"); - if(nBoundTighten) - report(lp, i, " %8d bounds............................... %s.\n", nBoundTighten, "TIGHTENED"); - if(nCoeffChanged) - report(lp, i, " %8d matrix coefficients.................. %s.\n", nCoeffChanged, "CHANGED"); - if(jjx > 0) - report(lp, i, " %8d variables' final bounds.............. %s.\n", jjx, "RELAXED"); - if(nSOS) - report(lp, i, " %8d constraints detected as SOS1......... %s.\n", nSOS, "CONVERTED"); - - /* Report optimality or infeasibility */ - if(status == UNBOUNDED) - report(lp, NORMAL, "%20s Solution status detected............. %s.\n", "", "UNBOUNDED"); - else if(status == INFEASIBLE) - report(lp, NORMAL, "%20s Solution status detected............. %s.\n", "", "INFEASIBLE"); - else { - if(psdata->cols->varmap->count == 0) - Value1 = Value2 = lp->presolve_undo->fixed_rhs[0] -initrhs0; - else - presolve_rangeorig(lp, 0, psdata->rows, &Value1, &Value2, -initrhs0); - if((fabs(Value1 - Value2) < psdata->epsvalue) || (fabs(my_reldiff(Value1, Value2)) < psdata->epsvalue)) { - if((lp->rows == 0) && (lp->columns == 0)) { - status = PRESOLVED; - Value1 = my_chsign(is_maxim(lp), Value1); - lp->solution[0] = Value1; - lp->best_solution[0] = Value1; - lp->full_solution[0] = Value1; - } - report(lp, NORMAL, "%20s OPTIMAL solution found............... %-g", "", Value1); - } - else if((status == RUNNING) && (i >= NORMAL)) { - char lonum[20], upnum[20]; - if(my_infinite(lp, Value1)) - sprintf(lonum, "%13s", "-Inf"); - else - sprintf(lonum, "%+12g", Value1); - if(my_infinite(lp, Value2)) - sprintf(upnum, "%-13s", "Inf"); - else - sprintf(upnum, "%+-12g", Value2); - report(lp, i, "%20s [ %s < Z < %s ]\n", "", lonum, upnum); - } - - /* Update values for dual limit and best heuristic values */ - if((MIP_count(lp) > 0) || (get_Lrows(lp) > 0)) { - if(is_maxim(lp)) { - SETMAX(lp->bb_heuristicOF, Value1); - SETMIN(lp->bb_limitOF, Value2); - } - else { - SETMIN(lp->bb_heuristicOF, Value2); - SETMAX(lp->bb_limitOF, Value1); - } - } - } - report(lp, NORMAL, " \n"); - - /* Clean up (but save counts of constraint types for display later) */ - j = psdata->LTmap->count; - jx = psdata->EQmap->count; - jjx = lp->rows - j - jx; - presolve_free(&psdata); - - } - - /* Signal that we are done presolving */ - if((lp->usermessage != NULL) && - ((lp->do_presolve & PRESOLVE_LASTMASKMODE) != 0) && (lp->msgmask & MSG_PRESOLVE)) - lp->usermessage(lp, lp->msghandle, MSG_PRESOLVE); - - /* Create master SOS variable list */ - if(SOS_count(lp) > 0) { - /*SOS_member_updatemap(lp->SOS); */ - make_SOSchain(lp, (MYBOOL) ((lp->do_presolve & PRESOLVE_LASTMASKMODE) != PRESOLVE_NONE)); - } - - /* Finalize model not identified as infeasible or unbounded */ - if(status == RUNNING) { - - /* Resolve GUBs */ - if(is_bb_mode(lp, NODE_GUBMODE)) - identify_GUB(lp, TRUE); - -#if 0 - /* Mark rows containing hidden identity matrices so that supporting factorization - engines can use this structural information to boost efficiency */ - if(is_algopt(lp, ALGOPT_COMPACTBPF)) - lp->bfpoptimize = (MYBOOL) (assist_factorization(lp, ROWTYPE_LOGICAL, - &lp->rowset1, &lp->rowno1) > 0); -#endif - - /* Scale the model based on current settings */ - auto_scale(lp); - - /* Crash the basis, if specified */ - crash_basis(lp); - - /* Produce presolved model statistics */ - if(nConRemove+nVarFixed+nBoundTighten+nVarFixed+nCoeffChanged > 0) { - REPORT_modelinfo(lp, FALSE, "REDUCED"); - if(nSum > 0) { - report(lp, NORMAL, "Row-types: %7d LE, %7d GE, %7d EQ.\n", - j, jjx, jx); - report(lp, NORMAL, " \n"); - } - } - } - - /* Optionally produce data on constraint classes */ - if(lp->verbose > NORMAL) { - report(lp, NORMAL, " \n"); - REPORT_constraintinfo(lp, "CONSTRAINT CLASSES"); - report(lp, NORMAL, " \n"); - } - -Finish: - lp->wasPresolved = TRUE; - lp->timepresolved = timeNow(); - -#if 0 -/* write_mps(lp, "test_out.mps"); */ /* Must put here due to variable name mapping */ - write_lp(lp, "test_out.lp"); /* Must put here due to variable name mapping */ -#endif -#if 0 - REPORT_debugdump(lp, "testint2.txt", FALSE); -#endif - - return( status ); - -} - -STATIC MYBOOL postsolve(lprec *lp, int status) -{ - /* Verify solution */ - if(lp->lag_status != RUNNING) { - int itemp; - - if(status == PRESOLVED) - status = OPTIMAL; - - if((status == OPTIMAL) || (status == SUBOPTIMAL)) { - itemp = check_solution(lp, lp->columns, lp->best_solution, - lp->orig_upbo, lp->orig_lowbo, lp->epssolution); - if((itemp != OPTIMAL) && (lp->spx_status == OPTIMAL)) - lp->spx_status = itemp; - else if((itemp == OPTIMAL) && ((status == SUBOPTIMAL) || (lp->spx_status == PRESOLVED))) - lp->spx_status = status; - } - else if(status != PRESOLVED) { - report(lp, NORMAL, "lp_solve unsuccessful after %.0f iter and a last best value of %g\n", - (double) get_total_iter(lp), lp->best_solution[0]); - if(lp->bb_totalnodes > 0) - report(lp, NORMAL, "lp_solve explored %.0f nodes before termination\n", - (double) get_total_nodes(lp)); - } - else - lp->spx_status = OPTIMAL; - - /* Only rebuild primal solution here, since the dual is only computed on request */ - presolve_rebuildUndo(lp, TRUE); - } - - /* Check if we can clear the variable map */ - if(varmap_canunlock(lp)) - lp->varmap_locked = FALSE; -#if 0 - REPORT_mat_mmsave(lp, "basis.mtx", NULL, FALSE); /* Write the current basis matrix (no OF) */ -#endif - - return( TRUE ); -} diff --git a/code/3rd_lpsolve/lp_presolve.h b/code/3rd_lpsolve/lp_presolve.h deleted file mode 100644 index f59406ba..00000000 --- a/code/3rd_lpsolve/lp_presolve.h +++ /dev/null @@ -1,126 +0,0 @@ -#ifndef HEADER_lp_presolve -#define HEADER_lp_presolve - -#include "lp_types.h" -#include "lp_matrix.h" - -/* -------------------------------------------------------------------------------------------- */ -/* Defines for various presolve options */ -/* -------------------------------------------------------------------------------------------- */ - -#define MAX_PSMERGELOOPS 2 /* Max loops to merge compatible constraints */ -#define MAX_PSLINDEPLOOPS 1 /* Max loops to detect linearly dependendent constraints */ -#define MAX_PSBOUNDTIGHTENLOOPS 5 /* Maximumn number of loops to allow bound tightenings */ -#define MIN_SOS1LENGTH 4 /* Minimum length of a constraint for conversion to SOS1 */ -#if 1 - #define PRESOLVE_EPSVALUE (0.1*lp->epsprimal) -#else - #define PRESOLVE_EPSVALUE lp->epsvalue -#endif -#define PRESOLVE_EPSPIVOT 1.0e-3 /* Looses robustness at values smaller than ~1.0e-3 */ -#define PRESOLVE_BOUNDSLACK 10 /* Extra error recovery/tolerance margin */ - -#define DoPresolveRounding /* Use absolute and directed rounding (disable at own risk) */ -/*#define DoPresolveRelativeTest*/ - -/*#define PresolveForceUpdateMax*/ - -/*#define DualFeasibilityLogicEQ2*/ /* Add low-order feasibility/accuracy logic to elimEQ2 */ -#define DivisorIntegralityLogicEQ2 /* Always prefer integer divisors */ -#define FindImpliedEqualities /* Detect equalities (default is enabled) */ -#define Eq2Reldiff - -/*#define SavePresolveEliminated */ /* Enable to activate storage of eliminated matrix data */ -/*#define UseDualPresolve */ /* Enable to use full dual information for presolve */ - - -typedef struct _psrec -{ - LLrec *varmap; - int **next; - int *empty; - int *plucount; - int *negcount; - int *pluneg; - int *infcount; - REAL *plulower; - REAL *neglower; - REAL *pluupper; - REAL *negupper; - int allocsize; -} psrec; - -typedef struct _presolverec -{ - psrec *rows; - psrec *cols; - LLrec *EQmap; - LLrec *LTmap; - LLrec *INTmap; - REAL *pv_upbo; - REAL *pv_lobo; - REAL *dv_upbo; - REAL *dv_lobo; - lprec *lp; - REAL epsvalue; - REAL epspivot; - int innerloops; - int middleloops; - int outerloops; - int nzdeleted; - MYBOOL forceupdate; -} presolverec; - -#ifdef __cplusplus -extern "C" { -#endif - -/* Put function headers here */ - -STATIC MYBOOL presolve_createUndo(lprec *lp); -STATIC MYBOOL presolve_rebuildUndo(lprec *lp, MYBOOL isprimal); -STATIC MYBOOL inc_presolve_space(lprec *lp, int delta, MYBOOL isrows); -STATIC MYBOOL presolve_setOrig(lprec *lp, int orig_rows, int orig_cols); -STATIC MYBOOL presolve_colfix(presolverec *psdata, int colnr, REAL newvalue, MYBOOL remove, int *tally); -STATIC MYBOOL presolve_fillUndo(lprec *lp, int orig_rows, int orig_cols, MYBOOL setOrig); -STATIC MYBOOL presolve_freeUndo(lprec *lp); - -STATIC MYBOOL presolve_updatesums(presolverec *psdata); - -INLINE int presolve_nextrow(presolverec *psdata, int colnr, int *previtem); -INLINE int presolve_nextcol(presolverec *psdata, int rownr, int *previtem); - -STATIC presolverec *presolve_init(lprec *lp); -STATIC void presolve_free(presolverec **psdata); -STATIC int presolve_shrink(presolverec *psdata, int *nConRemove, int *nVarRemove); -STATIC void presolve_rowremove(presolverec *psdata, int rownr, MYBOOL allowcoldelete); -STATIC int presolve_colremove(presolverec *psdata, int colnr, MYBOOL allowrowdelete); - -STATIC MYBOOL presolve_colfixdual(presolverec *psdata, int colnr, REAL *fixValue, int *status); - -INLINE int presolve_rowlength(presolverec *psdata, int rownr) -{ - int *items = psdata->rows->next[rownr]; - - if(items == NULL) - return( 0 ); - else - return( items[0] ); -} -INLINE int presolve_collength(presolverec *psdata, int colnr) -{ - int *items = psdata->cols->next[colnr]; - if(items == NULL) - return( 0 ); - else - return( items[0] ); -} - -STATIC int presolve(lprec *lp); -STATIC MYBOOL postsolve(lprec *lp, int status); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_presolve */ diff --git a/code/3rd_lpsolve/lp_price.c b/code/3rd_lpsolve/lp_price.c deleted file mode 100644 index dd777a34..00000000 --- a/code/3rd_lpsolve/lp_price.c +++ /dev/null @@ -1,2107 +0,0 @@ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_report.h" -#include "lp_pricePSE.h" -#include "lp_price.h" - -#if libBLAS > 0 - #include "myblas.h" -#endif - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* Simplex pricing utility module - w/interface for lp_solve v5.0+ - ------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h, commonlib.h - - Release notes: - v1.0.0 1 July 2004 Routines extracted from lp_lib. - v1.0.1 10 July 2004 Added comparison operators for determination - of entering and leaving variables. - Added routines for multiple and partial - pricing and made corresponding changes to - colprim and rowdual. - v1.0.2 20 August 2004 Implemented relative pivot size control in - rowprim and rowdual. - v1.1.0 15 October 2004 Added dual long step logic. - v1.1.1 22 October 2004 Added bound sort order to variable selections. - v1.2.0 24 March 2005 Completed multiple pricing logic. - ------------------------------------------------------------------------- */ - - -/* Comparison operators for entering and leaving variables for both the primal and - dual simplexes. The functions compare a candidate variable with an incumbent. */ -int CMP_CALLMODEL compareImprovementVar(const pricerec *current, const pricerec *candidate) -{ - register int result = COMP_PREFERNONE; - register lprec *lp = current->lp; - register REAL testvalue, margin = PREC_IMPROVEGAP; - int currentcolno, currentvarno = current->varno, - candidatecolno, candidatevarno = candidate->varno; - MYBOOL isdual = candidate->isdual; - - if(isdual) { - candidatevarno = lp->var_basic[candidatevarno]; - currentvarno = lp->var_basic[currentvarno]; - } - candidatecolno = candidatevarno - lp->rows; - currentcolno = currentvarno - lp->rows; - - /* Do pivot-based selection unless Bland's (first index) rule is active */ - if(lp->_piv_rule_ != PRICER_FIRSTINDEX) { - - MYBOOL candbetter; - - /* Find the largest value - normalize in case of the dual, since - constraint violation is expressed as a negative number. */ - /* Use absolute test for "small numbers", relative otherwise */ - testvalue = candidate->pivot; - if(fabs(testvalue) < LIMIT_ABS_REL) - testvalue -= current->pivot; - else - testvalue = my_reldiff(testvalue, current->pivot); - if(isdual) - testvalue = -testvalue; - - candbetter = (MYBOOL) (testvalue > 0); - if(candbetter) { - if(testvalue > margin) - result = COMP_PREFERCANDIDATE; - } -#if 0 /* Give more opportunity to optimize on non-primary criteria */ - else if (testvalue < -margin) -#else /* Give reduced opportunity to optimize on non-primary criteria */ - else if (testvalue < -lp->epsvalue) -#endif - result = COMP_PREFERINCUMBENT; - -#ifdef UseSortOnBound - /* Extra selection criterion based on the variable's range; - variable with - DUAL: small bound out; PRIMAL: large bound in */ - if(result == COMP_PREFERNONE) { - testvalue = lp->upbo[candidatevarno] - lp->upbo[currentvarno]; - if(testvalue < -margin) - result = COMP_PREFERINCUMBENT; - else if(testvalue > margin) - result = COMP_PREFERCANDIDATE; - result = my_chsign(isdual, result); - } -#endif - -#ifdef UseSortOnColumnLength - /* Prevent long columns from entering the basis */ - if(result == COMP_PREFERNONE) { - if(candidatecolno > 0) - testvalue = mat_collength(lp->matA, candidatecolno) + - (is_obj_in_basis(lp) && (lp->obj[candidatecolno] != 0) ? 1 : 0); - else - testvalue = 1; - if(currentcolno > 0) - testvalue -= mat_collength(lp->matA, currentcolno) + - (is_obj_in_basis(lp) && (lp->obj[currentcolno] != 0) ? 1 : 0); - else - testvalue -= 1; - if(testvalue > 0) - result = COMP_PREFERINCUMBENT; - else if(testvalue < 0) - result = COMP_PREFERCANDIDATE; - result = my_chsign(isdual, result); - } -#endif - - /* Select absolute best if the non-primary criteria failed to separate */ - if((result == COMP_PREFERNONE) && candbetter) { - result = COMP_PREFERCANDIDATE; - goto Finish; - } - } - - /* Final tie-breakers */ - if(result == COMP_PREFERNONE) { - - /* Add randomization tie-braker */ - if(lp->piv_strategy & PRICE_RANDOMIZE) { - result = my_sign(PRICER_RANDFACT - rand_uniform(lp, 1.0)); - if(candidatevarno < currentvarno) - result = -result; - } - - /* Resolve ties via index ordinal */ - if(result == COMP_PREFERNONE) { - if(candidatevarno < currentvarno) - result = COMP_PREFERCANDIDATE; - else /* if(candidatevarno > currentvarno) */ - result = COMP_PREFERINCUMBENT; - if(lp->_piv_left_) - result = -result; - } - } - -Finish: - return( result ); - -} - -int CMP_CALLMODEL compareSubstitutionVar(const pricerec *current, const pricerec *candidate) -{ - register int result = COMP_PREFERNONE; - register lprec *lp = current->lp; - register REAL testvalue = candidate->theta, - margin = current->theta; - MYBOOL isdual = candidate->isdual, candbetter; - int currentcolno, currentvarno = current->varno, - candidatecolno, candidatevarno = candidate->varno; - - if(!isdual) { - candidatevarno = lp->var_basic[candidatevarno]; - currentvarno = lp->var_basic[currentvarno]; - } - candidatecolno = candidatevarno - lp->rows; - currentcolno = currentvarno - lp->rows; - - /* Compute the ranking test metric. */ - if(isdual) { - testvalue = fabs(testvalue); - margin = fabs(margin); - } - - /* Use absolute test for "small numbers", relative otherwise */ - if(fabs(testvalue) < LIMIT_ABS_REL) - testvalue -= margin; - else - testvalue = my_reldiff(testvalue, margin); - - /* Find if the new Theta is smaller or near equal (i.e. testvalue <= eps) - compared to the previous best; ties will be broken by pivot size or index - NB! The margin below is essential in maintaining primal/dual feasibility - during the primal/dual simplex, respectively. Sometimes a small - value prevents the selection of a suitable pivot, thereby weakening - the numerical stability of some models */ - margin = PREC_SUBSTFEASGAP; - candbetter = (MYBOOL) (testvalue < 0); - if(candbetter) { - if(testvalue < -margin) - result = COMP_PREFERCANDIDATE; - } - else if(testvalue > margin) - result = COMP_PREFERINCUMBENT; - - /* Resolve a tie */ - if(result == COMP_PREFERNONE) { - REAL currentpivot = fabs(current->pivot), - candidatepivot = fabs(candidate->pivot); - - /* Handle first index / Bland's rule specially */ - if(lp->_piv_rule_ == PRICER_FIRSTINDEX) { -#if 1 - /* Special secondary selection by pivot size (limited stability protection) */ - margin = candidate->epspivot; - if((candidatepivot >= margin) && (currentpivot < margin)) - result = COMP_PREFERCANDIDATE; -#endif - } - - else { - - /* General secondary selection based on pivot size */ -#if 0 - if(candidatepivot > MIN_STABLEPIVOT) - testvalue = my_reldiff(testvalue, currentpivot); - else -#endif - testvalue = candidatepivot - currentpivot; - if(testvalue > margin) - result = COMP_PREFERCANDIDATE; - else if(testvalue < -margin) - result = COMP_PREFERINCUMBENT; - -#ifdef UseSortOnBound - /* Extra selection criterion based on the variable's range; - variable with - PRIMAL: small bound out; DUAL: large bound in */ - if(result == COMP_PREFERNONE) { - testvalue = lp->upbo[candidatevarno] - lp->upbo[currentvarno]; - if(testvalue < -margin) - result = COMP_PREFERCANDIDATE; - else if(testvalue > margin) - result = COMP_PREFERINCUMBENT; - result = my_chsign(isdual, result); - } -#endif - -#ifdef UseSortOnColumnLength - /* Prevent long columns from entering the basis */ - if(result == COMP_PREFERNONE) { - if(candidatecolno > 0) - testvalue = mat_collength(lp->matA, candidatecolno) + - (is_obj_in_basis(lp) && (lp->obj[candidatecolno] != 0) ? 1 : 0); - else - testvalue = 1; - if(currentcolno > 0) - testvalue -= mat_collength(lp->matA, currentcolno) + - (is_obj_in_basis(lp) && (lp->obj[currentcolno] != 0) ? 1 : 0); - else - testvalue -= 1; - if(testvalue > 0) - result = COMP_PREFERCANDIDATE; - else if(testvalue < 0) - result = COMP_PREFERINCUMBENT; - result = my_chsign(isdual, result); - } -#endif - - } - } - - /* Select absolute best if the non-primary criteria failed to separate */ - if((result == COMP_PREFERNONE) && candbetter) { - result = COMP_PREFERCANDIDATE; - goto Finish; - } - - /* Final tie-breakers */ - if(result == COMP_PREFERNONE) { - - /* Add randomization tie-braker */ - if(lp->piv_strategy & PRICE_RANDOMIZE) { - result = my_sign(PRICER_RANDFACT - rand_uniform(lp, 1.0)); - if(candidatevarno < currentvarno) - result = -result; - } - - /* Resolve ties via index ordinal (also prefers slacks over user variables) */ - if(result == COMP_PREFERNONE) { - if(candidatevarno < currentvarno) - result = COMP_PREFERCANDIDATE; - else /* if(candidatevarno > currentvarno) */ - result = COMP_PREFERINCUMBENT; - if(lp->_piv_left_) - result = -result; - } - } - -Finish: - return( result ); -} -int CMP_CALLMODEL compareBoundFlipVar(const pricerec *current, const pricerec *candidate) -{ - register REAL testvalue, margin; - register int result = COMP_PREFERNONE; - register lprec *lp = current->lp; - MYBOOL candbetter; - int currentvarno = current->varno, - candidatevarno = candidate->varno; - - if(!current->isdual) { - candidatevarno = lp->var_basic[candidatevarno]; - currentvarno = lp->var_basic[currentvarno]; - } - - /* Compute the ranking test metric. */ - testvalue = candidate->theta; - margin = current->theta; - if(candidate->isdual) { - testvalue = fabs(testvalue); - margin = fabs(margin); - } - if(fabs(margin) < LIMIT_ABS_REL) - testvalue -= margin; - else - testvalue = my_reldiff(testvalue, margin); - - /* Find if the new Theta is smaller or near equal (i.e. testvalue <= eps) - compared to the previous best; ties will be broken by pivot size or index */ - margin = PREC_SUBSTFEASGAP; - candbetter = (MYBOOL) (testvalue < 0); - if(candbetter) { - if(testvalue < -margin) - result = COMP_PREFERCANDIDATE; - } - else if(testvalue > margin) - result = COMP_PREFERINCUMBENT; - - /* Resolve a tie */ - if(result == COMP_PREFERNONE) { - - /* Tertiary selection based on priority for large pivot sizes */ - if(result == COMP_PREFERNONE) { - REAL currentpivot = fabs(current->pivot), - candidatepivot = fabs(candidate->pivot); - if(candidatepivot > currentpivot+margin) - result = COMP_PREFERCANDIDATE; - else if(candidatepivot < currentpivot-margin) - result = COMP_PREFERINCUMBENT; - } - - /* Secondary selection based on priority for narrow-bounded variables */ - if(result == COMP_PREFERNONE) - result = compareREAL(&(lp->upbo[currentvarno]), - &(lp->upbo[candidatevarno])); - - } - - /* Select absolute best if the non-primary criteria failed to separate */ - if((result == COMP_PREFERNONE) && candbetter) { - result = COMP_PREFERCANDIDATE; - goto Finish; - } - - /* Quaternary selection by index value */ - if(result == COMP_PREFERNONE) { - if(candidatevarno < currentvarno) - result = COMP_PREFERCANDIDATE; - else - result = COMP_PREFERINCUMBENT; - if(lp->_piv_left_) - result = -result; - } - -Finish: - return( result ); -} - -/* Validity operators for entering and leaving columns for both the primal and dual - simplex. All candidates must satisfy these tests to qualify to be allowed to be - a subject for the comparison functions/operators. */ -STATIC MYBOOL validImprovementVar(pricerec *candidate) -{ - register REAL candidatepivot = fabs(candidate->pivot); - -#ifdef Paranoia - return( (MYBOOL) ((candidate->varno > 0) && (candidatepivot > candidate->lp->epsvalue)) ); -#else - return( (MYBOOL) (candidatepivot > candidate->lp->epsvalue) ); -#endif -} - -STATIC MYBOOL validSubstitutionVar(pricerec *candidate) -{ - register lprec *lp = candidate->lp; - register REAL theta = (candidate->isdual ? fabs(candidate->theta) : candidate->theta); - -#ifdef Paranoia - if(candidate->varno <= 0) - return( FALSE ); - else -#endif - if(fabs(candidate->pivot) >= lp->infinite) - return( (MYBOOL) (theta < lp->infinite) ); - else - return( (MYBOOL) ((theta < lp->infinite) && - (fabs(candidate->pivot) >= candidate->epspivot)) ); -} - -int CMP_CALLMODEL compareImprovementQS(const UNIONTYPE QSORTrec *current, const UNIONTYPE QSORTrec *candidate) -{ - return( compareImprovementVar((pricerec *) current->pvoidint2.ptr, (pricerec *) candidate->pvoidint2.ptr) ); -} -int CMP_CALLMODEL compareSubstitutionQS(const UNIONTYPE QSORTrec *current, const UNIONTYPE QSORTrec *candidate) -{ - return( compareBoundFlipVar((pricerec *) current->pvoidint2.ptr, (pricerec *) candidate->pvoidint2.ptr) ); -/* return( compareSubstitutionVar((pricerec *) current->self, (pricerec *) candidate->self) ); */ -} - -/* Function to add a valid pivot candidate into the specified list */ -STATIC int addCandidateVar(pricerec *candidate, multirec *multi, findCompare_func findCompare, MYBOOL allowSortedExpand) -{ - int insertpos, delta = 1; - pricerec *targetrec; - - /* Find the insertion point (if any) */ - if((multi->freeList[0] == 0) || - (multi->sorted && allowSortedExpand) || - (candidate->isdual && (multi->used == 1) && ((multi->step_last >= multi->epszero) || - multi_truncatingvar(multi, ((pricerec *) (multi->sortedList[0].pvoidreal.ptr))->varno))) - ) { - UNIONTYPE QSORTrec searchTarget; - - /* Make sure that the list is sorted before the search for an insertion point */ - if((multi->freeList[0] == 0) && !multi->sorted) { - multi->sorted = QS_execute(multi->sortedList, multi->used, findCompare, &insertpos); - multi->dirty = (MYBOOL) (insertpos > 0); - } - - /* Perform the search */ - searchTarget.pvoidint2.ptr = (void *) candidate; - insertpos = sizeof(searchTarget); - insertpos = findIndexEx(&searchTarget, multi->sortedList-delta, multi->used, delta, insertpos, findCompare, TRUE); - if(insertpos > 0) - return( -1 ); - insertpos = -insertpos - delta; - - /* Check if the candidate is worse than the worst of the list */ - if(((insertpos >= multi->size) && (multi->freeList[0] == 0)) || - ((insertpos == multi->used) && (!allowSortedExpand || - (multi->step_last >= multi->epszero)))) - return( -1 ); - -#ifdef Paranoia - /* Do validation */ - if((insertpos < 0) || (insertpos > multi->used)) - return( -1 ); -#endif - - /* Define the target for storing the candidate; - Case 1: List is full and we must discard the previously worst candidate - Case 2: List is not full and we simply use the next free position */ - if(multi->freeList[0] == 0) - targetrec = (pricerec *) multi->sortedList[multi->used-1].pvoidreal.ptr; - else { - delta = multi->freeList[0]--; - delta = multi->freeList[delta]; - targetrec = &(multi->items[delta]); - } - } - else { - delta = multi->freeList[0]--; - delta = multi->freeList[delta]; - targetrec = &(multi->items[delta]); - insertpos = multi->used; - } - - /* Insert the new candidate record in the data store */ - MEMCOPY(targetrec, candidate, 1); - - /* Store the pointer data and handle tree cases: - Case 1: The list is unsorted and not full; simply append pointer to list, - Case 2: The list is sorted and full; insert the pointer by discarding previous last, - Case 3: The list is sorted and not full; shift the inferior items down, and increment count */ - if((multi->used < multi->size) && (insertpos >= multi->used)) { - QS_append(multi->sortedList, insertpos, targetrec); - multi->used++; - } - else { - if(multi->used == multi->size) - QS_insert(multi->sortedList, insertpos, targetrec, multi->size-1); /* Discard previous last */ - else { - QS_insert(multi->sortedList, insertpos, targetrec, multi->used); /* Keep previous last */ - multi->used++; - } - } - multi->active = insertpos; - -#ifdef Paranoia - if((insertpos >= multi->size) || (insertpos >= multi->used)) - report(multi->lp, SEVERE, "addCandidateVar: Insertion point beyond limit!\n"); -#endif - - return( insertpos ); -} - -STATIC MYBOOL findImprovementVar(pricerec *current, pricerec *candidate, MYBOOL collectMP, int *candidatecount) -/* PRIMAL: Find a variable to enter the basis - DUAL: Find a variable to leave the basis - - Allowed variable set: Any pivot PRIMAL:larger or DUAL:smaller than threshold value of 0 */ -{ - MYBOOL Action = FALSE, -#ifdef ExtractedValidityTest - Accept = TRUE; -#else /* Check for validity and compare result with previous best */ - Accept = validImprovementVar(candidate); -#endif - if(Accept) { - if(candidatecount != NULL) - (*candidatecount)++; - if(collectMP) { - if(addCandidateVar(candidate, current->lp->multivars, (findCompare_func *) compareImprovementQS, FALSE) < 0) - return(Action); - } - if(current->varno > 0) - Accept = (MYBOOL) (compareImprovementVar(current, candidate) > 0); - } - - /* Apply candidate if accepted */ - if(Accept) { - (*current) = *candidate; - - /* Force immediate acceptance for Bland's rule using the primal simplex */ - if(!candidate->isdual) - Action = (MYBOOL) (candidate->lp->_piv_rule_ == PRICER_FIRSTINDEX); - } - return(Action); -} - -/* Bound flip variable accumulation routine */ -STATIC MYBOOL collectMinorVar(pricerec *candidate, multirec *longsteps, MYBOOL isphase2, MYBOOL isbatch) -{ - int inspos; - - /* 1. Check for ratio and pivot validity (to have the extra flexibility that all - bound-flip candidates are also possible as basis-entering variables */ - if(!validSubstitutionVar(candidate)) - return( FALSE ); - - /* 2. If the free-list is empty we need to see if we have a better candidate, - and for this the candidate list has to be sorted by merit */ - if(!isbatch && - !longsteps->sorted && (longsteps->used > 1) && - ((longsteps->freeList[0] == 0) || - multi_truncatingvar(longsteps, candidate->varno) || - (longsteps->step_last >= longsteps->epszero) )) { - longsteps->sorted = QS_execute(longsteps->sortedList, longsteps->used, - (findCompare_func *) compareSubstitutionQS, &inspos); - longsteps->dirty = (MYBOOL) (inspos > 0); - if(longsteps->dirty) - multi_recompute(longsteps, 0, isphase2, TRUE); - } - - /* 3. Now handle three cases... - - Add to the list when the list is not full and there is opportunity for improvement, - - Check if we should replace an incumbent when the list is full, - - Check if we should replace an incumbent when the list is not full, there is no room - for improvement, but the current candidate is better than an incumbent. */ - inspos = addCandidateVar(candidate, longsteps, (findCompare_func *) compareSubstitutionQS, TRUE); - - /* 4. Recompute steps and objective, and (if relevant) determine if we - may be suboptimal in relation to an incumbent MILP solution. */ - return( (MYBOOL) (inspos >= 0) && - ((isbatch == TRUE) || multi_recompute(longsteps, inspos, isphase2, TRUE)) ); -} - -STATIC MYBOOL findSubstitutionVar(pricerec *current, pricerec *candidate, int *candidatecount) -/* PRIMAL: Find a variable to leave the basis - DUAL: Find a variable to enter the basis - - Allowed variable set: "Equal-valued" smallest thetas! */ -{ - MYBOOL Action = FALSE, -#ifdef ExtractedValidityTest - Accept = TRUE; -#else /* Check for validity and comparison result with previous best */ - Accept = validSubstitutionVar(candidate); -#endif - if(Accept) { - if(candidatecount != NULL) - (*candidatecount)++; - if(current->varno != 0) - Accept = (MYBOOL) (compareSubstitutionVar(current, candidate) > 0); - } - - /* Apply candidate if accepted */ - if(Accept) { - (*current) = *candidate; - - /* Force immediate acceptance for Bland's rule using the dual simplex */ -#ifdef ForceEarlyBlandRule - if(candidate->isdual) - Action = (MYBOOL) (candidate->lp->_piv_rule_ == PRICER_FIRSTINDEX); -#endif - } - return(Action); -} - -/* Partial pricing management routines */ -STATIC partialrec *partial_createBlocks(lprec *lp, MYBOOL isrow) -{ - partialrec *blockdata; - - blockdata = (partialrec *) calloc(1, sizeof(*blockdata)); - blockdata->lp = lp; - blockdata->blockcount = 1; - blockdata->blocknow = 1; - blockdata->isrow = isrow; - - return(blockdata); -} -STATIC int partial_countBlocks(lprec *lp, MYBOOL isrow) -{ - partialrec *blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - - if(blockdata == NULL) - return( 1 ); - else - return( blockdata->blockcount ); -} -STATIC int partial_activeBlocks(lprec *lp, MYBOOL isrow) -{ - partialrec *blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - - if(blockdata == NULL) - return( 1 ); - else - return( blockdata->blocknow ); -} -STATIC void partial_freeBlocks(partialrec **blockdata) -{ - if((blockdata == NULL) || (*blockdata == NULL)) - return; - FREE((*blockdata)->blockend); - FREE((*blockdata)->blockpos); - FREE(*blockdata); -} - - -/* Function to provide for left-right or right-left scanning of entering/leaving - variables; note that *end must have been initialized by the calling routine! */ -STATIC void makePriceLoop(lprec *lp, int *start, int *end, int *delta) -{ - int offset = is_piv_mode(lp, PRICE_LOOPLEFT); - - if((offset) || - (((lp->total_iter+offset) % 2 == 0) && is_piv_mode(lp, PRICE_LOOPALTERNATE))) { - *delta = -1; /* Step backwards - "left" */ - swapINT(start, end); - lp->_piv_left_ = TRUE; - } - else { - *delta = 1; /* Step forwards - "right" */ - lp->_piv_left_ = FALSE; - } -} - -/* Routine to verify accuracy of the current basis factorization */ -STATIC MYBOOL serious_facterror(lprec *lp, REAL *bvector, int maxcols, REAL tolerance) -{ - int i, j, ib, ie, nz, nc; - REAL sum, tsum = 0, err = 0; - MATrec *mat = lp->matA; - - if(bvector == 0) - bvector = lp->bsolveVal; - nc =0; - nz = 0; - for(i = 1; (i <= lp->rows) && (nc <= maxcols); i++) { - - /* Do we have a non-slack variable? (we choose to skip slacks, - since they have "natural" good accuracy properties) */ - j = lp->var_basic[i] - lp->rows; - if(j <= 0) - continue; - nc++; - - /* Compute cross product for basic, non-slack column */ - ib = mat->col_end[j-1]; - ie = mat->col_end[j]; - nz += ie - ib; - sum = get_OF_active(lp, j+lp->rows, bvector[0]); - for(; ib < ie; ib++) - sum += COL_MAT_VALUE(ib)*bvector[COL_MAT_ROWNR(ib)]; - - /* Catch high precision early, so we don't to uneccessary work */ - tsum += sum; - SETMAX(err, fabs(sum)); - if((tsum / nc > tolerance / 100) && (err < tolerance / 100)) - break; - } - err /= mat->infnorm; - return( (MYBOOL) (err >= tolerance) ); -} - -/* Computation of reduced costs */ -STATIC void update_reducedcosts(lprec *lp, MYBOOL isdual, int leave_nr, int enter_nr, REAL *prow, REAL *drow) -{ - /* "Fast" update of the dual reduced cost vector; note that it must be called - after the pivot operation and only applies to a major "true" iteration */ - int i; - REAL hold; - - if(isdual) { - hold = -drow[enter_nr]/prow[enter_nr]; - for(i=1; i <= lp->sum; i++) - if(!lp->is_basic[i]) { - if(i == leave_nr) - drow[i] = hold; - else { - drow[i] += hold*prow[i]; - my_roundzero(drow[i], lp->epsmachine); - } - } - } - else - report(lp, SEVERE, "update_reducedcosts: Cannot update primal reduced costs!\n"); -} - - -STATIC void compute_reducedcosts(lprec *lp, MYBOOL isdual, int row_nr, int *coltarget, MYBOOL dosolve, - REAL *prow, int *nzprow, - REAL *drow, int *nzdrow, - int roundmode) -{ - REAL epsvalue = lp->epsvalue; /* Any larger value can produce a suboptimal result */ - roundmode |= MAT_ROUNDRC; - - if(isdual) { - bsolve_xA2(lp, coltarget, - row_nr, prow, epsvalue, nzprow, /* Calculate net sensitivity given a leaving variable */ - 0, drow, epsvalue, nzdrow, /* Calculate the net objective function values */ - roundmode); - } - else { - REAL *bVector; - -#if 1 /* Legacy mode, that is possibly a little faster */ - if((lp->multivars == NULL) && (lp->P1extraDim == 0)) - bVector = drow; - else -#endif - bVector = lp->bsolveVal; - if(dosolve) { - bsolve(lp, 0, bVector, lp->bsolveIdx, epsvalue*DOUBLEROUND, 1.0); - if(!isdual && (row_nr == 0) && (lp->improve & IMPROVE_SOLUTION) && !refactRecent(lp) && - serious_facterror(lp, bVector, lp->rows, lp->epsvalue)) - set_action(&lp->spx_action, ACTION_REINVERT); - } - prod_xA(lp, coltarget, - bVector, lp->bsolveIdx, epsvalue, 1.0, - drow, nzdrow, roundmode); - } -} - - -/* Primal: Prevent acceptance of an entering variable when the magnitude of - other candidates is also very small. - Dual: Prevent acceptance of a leaving variable when the magnitude of - other candidates is also very small. - - Both of these cases are associated with numerical stalling, which we could - argue should be detected and handled by the stalling monitor routine. */ -STATIC MYBOOL verify_stability(lprec *lp, MYBOOL isprimal, REAL xfeas, REAL sfeas, int nfeas) -{ - MYBOOL testOK = TRUE; - return( testOK ); - -#if 1 - /* Try to make dual feasibility as tight as possible */ - if(!isprimal) -/* if(lp->P1extraVal == 0) */ - { - xfeas /= (1+lp->rhsmax); - sfeas /= (1+lp->rhsmax); - } -#endif - xfeas = fabs(xfeas); /* Maximum (positive) infeasibility */ -/* if(xfeas < lp->epspivot) { */ - if(xfeas < lp->epssolution) { - REAL f; - sfeas = fabs(sfeas); /* Make sum of infeasibilities positive */ - xfeas = (sfeas-xfeas)/nfeas; /* Average "residual" feasibility */ - f = 1 + log10((REAL) nfeas); /* Some numerical complexity scalar */ - /* Numerical errors can interact to cause non-convergence, and the - idea is to relax the tolerance to account for this and only - marginally weakening the (user-specified) tolerance. */ - if((sfeas-xfeas) < f*lp->epsprimal) - testOK = FALSE; - } - return( testOK ); -} - - -/* Find an entering column for the case that the specified basic variable - is fixed or zero - typically used for artificial variable elimination */ -STATIC int find_rowReplacement(lprec *lp, int rownr, REAL *prow, int *nzprow) -/* The logic in this section generally follows Chvatal: Linear Programming, p. 130 - Basically, the function is a specialized coldual(). */ -{ - int i, bestindex; - REAL bestvalue; - - /* Solve for "local reduced cost" */ - set_action(&lp->piv_strategy, PRICE_FORCEFULL); - compute_reducedcosts(lp, TRUE, rownr, NULL, TRUE, - prow, nzprow, NULL, NULL, MAT_ROUNDDEFAULT); - clear_action(&lp->piv_strategy, PRICE_FORCEFULL); - - /* Find a suitably non-singular variable to enter ("most orthogonal") */ - bestindex = 0; - bestvalue = 0; - for(i = 1; i <= lp->sum-abs(lp->P1extraDim); i++) { - if(!lp->is_basic[i] && !is_fixedvar(lp, i) && - (fabs(prow[i]) > bestvalue)) { - bestindex = i; - bestvalue = fabs(prow[i]); - } - } - - /* Prepare to update inverse and pivot/iterate (compute Bw=a) */ - if(i > lp->sum-abs(lp->P1extraDim)) - bestindex = 0; - else - fsolve(lp, bestindex, prow, nzprow, lp->epsmachine, 1.0, TRUE); - - return( bestindex ); -} - -/* Find the primal simplex entering non-basic column variable */ -STATIC int colprim(lprec *lp, REAL *drow, int *nzdrow, MYBOOL skipupdate, int partialloop, int *candidatecount, MYBOOL updateinfeas, REAL *xviol) -{ - int i, ix, iy, iz, ninfeas, nloop = 0; - REAL f, sinfeas, xinfeas, epsvalue = lp->epsdual; - pricerec current, candidate; - MYBOOL collectMP = FALSE; - int *coltarget = NULL; - - /* Identify pivot column according to pricing strategy; set - entering variable initial threshold reduced cost value to "0" */ - current.pivot = lp->epsprimal; /* Minimum acceptable improvement */ - current.varno = 0; - current.lp = lp; - current.isdual = FALSE; - candidate.lp = lp; - candidate.isdual = FALSE; - *candidatecount = 0; - - /* Update local value of pivot setting and determine active multiple pricing set */ - lp->_piv_rule_ = get_piv_rule(lp); -doLoop: - nloop++; - if((lp->multivars != NULL) && ((lp->simplex_mode & SIMPLEX_PRIMAL_PRIMAL) != 0)) { - collectMP = multi_mustupdate(lp->multivars); - if(collectMP) { - multi_restart(lp->multivars); - coltarget = NULL; - } - else - coltarget = multi_indexSet(lp->multivars, FALSE); - } - - /* Compute reduced costs c - c*Inv(B), if necessary - (i.e. the previous iteration was not a "minor" iteration/bound flip) */ - if(!skipupdate) { -#ifdef UsePrimalReducedCostUpdate - /* Recompute from scratch only at the beginning, otherwise update */ - if((lp->current_iter > 0) && (refactRecent(lp) == AUTOMATIC)) -#endif - compute_reducedcosts(lp, FALSE, 0, coltarget, (MYBOOL) ((nloop <= 1) || (partialloop > 1)), - NULL, NULL, - drow, nzdrow, - MAT_ROUNDDEFAULT); - } - - /* Loop over active partial column set; we presume that reduced costs - have only been updated for columns in the active partial range. */ - ix = 1; - iy = nzdrow[0]; - ninfeas = 0; - xinfeas = 0; - sinfeas = 0; - makePriceLoop(lp, &ix, &iy, &iz); - iy *= iz; - for(; ix*iz <= iy; ix += iz) { - i = nzdrow[ix]; -#if 0 /* Not necessary since we masked them out in compute_reducedcosts() */ - if(i > lp->sum-abs(lp->P1extraDim)) - continue; -#endif - - /* Check if the pivot candidate is on the block-list */ - if(lp->rejectpivot[0] > 0) { - int kk; - for(kk = 1; (kk <= lp->rejectpivot[0]) && (i != lp->rejectpivot[kk]); kk++); - if(kk <= lp->rejectpivot[0]) - continue; - } - - /* Retrieve the applicable reduced cost - threshold should not be smaller than 0 */ - f = my_chsign(lp->is_lower[i], drow[i]); - if(f <= epsvalue) - continue; - - /* Find entering variable according to strategy (largest positive f) */ - ninfeas++; - SETMAX(xinfeas, f); - sinfeas += f; - candidate.pivot = normalizeEdge(lp, i, f, FALSE); - candidate.varno = i; - if(findImprovementVar(¤t, &candidate, collectMP, candidatecount)) - break; - } - - /* Check if we should loop again after a multiple pricing update */ - if(lp->multivars != NULL) { - if(collectMP) { - if(!lp->multivars->sorted) - lp->multivars->sorted = QS_execute(lp->multivars->sortedList, lp->multivars->used, - (findCompare_func *) compareImprovementQS, NULL); - coltarget = multi_indexSet(lp->multivars, TRUE); - } - else if((current.varno == 0) && (lp->multivars->retries == 0)) { - ix = partial_blockStart(lp, FALSE); - iy = partial_blockEnd(lp, FALSE); - lp->multivars->used = 0; - lp->multivars->retries++; - goto doLoop; - } - /* Shrink the candidate list */ - lp->multivars->retries = 0; - if(current.varno != 0) - multi_removevar(lp->multivars, current.varno); - } - - /* Check for optimality */ - if(xviol != NULL) - *xviol = xinfeas; - if(updateinfeas) - lp->suminfeas = fabs(sinfeas); - if((lp->multivars == NULL) && (current.varno > 0) && - !verify_stability(lp, TRUE, xinfeas, sinfeas, ninfeas)) - current.varno = 0; - - /* Produce statistics */ - if(lp->spx_trace) { - if(current.varno > 0) - report(lp, DETAILED, "colprim: Column %d reduced cost = " RESULTVALUEMASK "\n", - current.varno, current.pivot); - else - report(lp, DETAILED, "colprim: No positive reduced costs found, optimality!\n"); - } - - return( current.varno ); -} /* colprim */ - -/* Find the primal simplex leaving basic column variable */ -STATIC int rowprim(lprec *lp, int colnr, LREAL *theta, REAL *pcol, int *nzpcol, MYBOOL forceoutEQ, REAL *xviol) -{ - int i, ii, iy, iz, Hpass, k, *nzlist; - LREAL f, savef; - REAL Heps, Htheta, Hlimit, epsvalue, epspivot, p = 0.0; - pricerec current, candidate; - MYBOOL isupper = !lp->is_lower[colnr], HarrisTwoPass = FALSE; - - /* Update local value of pivot setting */ - lp->_piv_rule_ = get_piv_rule(lp); - if(nzpcol == NULL) - nzlist = (int *) mempool_obtainVector(lp->workarrays, lp->rows+1, sizeof(*nzlist)); - else - nzlist = nzpcol; - - /* Find unconditional non-zeros and optionally compute relative size of epspivot */ - epspivot = lp->epspivot; - epsvalue = lp->epsvalue; - Hlimit = 0; - Htheta = 0; - k = 0; - for(i = 1; i <= lp->rows; i++) { - p = fabs(pcol[i]); - if(p > Hlimit) - Hlimit = p; - if(p > epsvalue) { - k++; - nzlist[k] = i; - SETMAX(Htheta, p); - } -#ifdef Paranoia - else { - if(lp->spx_trace) - report(lp, FULL, "rowprim: Row %d with pivot " RESULTVALUEMASK " rejected as too small\n", - i, p); - } -#endif - } - if(xviol != NULL) - *xviol = Htheta; - Htheta = 0; - - /* Update non-zero list based on the new pivot threshold */ -#ifdef UseRelativePivot_Primal -/* epspivot *= sqrt(lp->matA->dynrange) / lp->matA->infnorm; */ - epspivot /= MAX(1, sqrt(lp->matA->colmax[colnr])); - iy = k; - k = 0; - p = 0; - for(ii = 1; ii <= iy; ii++) { - i = nzlist[ii]; - p = fabs(pcol[i]); - - /* Compress the list of valid alternatives, if appropriate */ - if(p > epspivot) { - k++; - nzlist[k] = i; - } -#ifdef Paranoia - else { - if(lp->spx_trace) - report(lp, FULL, "rowprim: Row %d with pivot " RESULTVALUEMASK " rejected as too small\n", - i, p); - } -#endif - } -#endif - - /* Initialize counters */ - nzlist[0] = k; - k = 0; - -Retry: - k++; - HarrisTwoPass = is_piv_mode(lp, PRICE_HARRISTWOPASS); - if(HarrisTwoPass) - Hpass = 1; - else - Hpass = 2; - current.theta = lp->infinite; - current.pivot = 0; - current.varno = 0; - current.isdual = FALSE; - current.epspivot = epspivot; - current.lp = lp; - candidate.epspivot = epspivot; - candidate.isdual = FALSE; - candidate.lp = lp; - savef = 0; - for(; Hpass <= 2; Hpass++) { - Htheta = lp->infinite; - if(Hpass == 1) { - Hlimit = lp->infinite; /* Don't apply any limit in the first pass */ - Heps = epspivot/lp->epsprimal; /* Scaled to lp->epsprimal used in compute_theta() */ - } - else { - Hlimit = current.theta; /* This is the smallest Theta of the first pass */ - Heps = 0.0; - } - current.theta = lp->infinite; - current.pivot = 0; - current.varno = 0; - savef = 0; - - ii = 1; - iy = nzlist[0]; - makePriceLoop(lp, &ii, &iy, &iz); - iy *= iz; - for(; ii*iz <= iy; ii += iz) { - i = nzlist[ii]; - f = pcol[i]; - candidate.theta = f; - candidate.pivot = f; - candidate.varno = i; - - /*i =*/ compute_theta(lp, i, &candidate.theta, isupper, - my_if(lp->upbo[lp->var_basic[i]] < lp->epsprimal, Heps/10, Heps), TRUE); - - if(fabs(candidate.theta) >= lp->infinite) { - savef = f; - candidate.theta = 2*lp->infinite; - continue; - } - - /* Find the candidate leaving variable according to strategy (smallest theta) */ - if((Hpass == 2) && (candidate.theta > Hlimit)) - continue; - - /* Give a slight preference to fixed variables (mainly equality slacks) */ - if(forceoutEQ) { - p = candidate.pivot; - if(lp->upbo[lp->var_basic[i]] < lp->epsprimal) { - /* Give an extra early boost to equality slack elimination, if specified */ - if(forceoutEQ == AUTOMATIC) - candidate.pivot *= 1.0+lp->epspivot; - else - candidate.pivot *= 10.0; - - } - } - if(HarrisTwoPass) { - f = candidate.theta; - if(Hpass == 2) - candidate.theta = 1; - if(findSubstitutionVar(¤t, &candidate, NULL)) - break; - if((Hpass == 2) && (current.varno == candidate.varno)) - Htheta = f; - } - else - if(findSubstitutionVar(¤t, &candidate, NULL)) - break; - /* Restore temporarily modified pivot */ - if(forceoutEQ && (current.varno == candidate.varno)) - current.pivot = p; - } - } - if(HarrisTwoPass) - current.theta = Htheta; - - /* Handle case of no available leaving variable */ - if(current.varno == 0) { - if(lp->upbo[colnr] >= lp->infinite) { - /* Optionally try again with reduced pivot threshold level */ - if(k < 2) { - epspivot = epspivot / 10; - goto Retry; - } - } - else { -#if 1 - i = 1; - while((pcol[i] >= 0) && (i <= lp->rows)) - i++; - if(i > lp->rows) { /* Empty column with upper bound! */ - lp->is_lower[colnr] = !lp->is_lower[colnr]; -/* lp->is_lower[colnr] = FALSE; */ - lp->rhs[0] += lp->upbo[colnr]*pcol[0]; - } - else /* if(pcol[i]<0) */ - { - current.varno = i; - } -#endif - } - } - else if(current.theta >= lp->infinite) { - report(lp, IMPORTANT, "rowprim: Numeric instability pcol[%d] = %g, rhs[%d] = %g, upbo = %g\n", - current.varno, savef, current.varno, lp->rhs[current.varno], - lp->upbo[lp->var_basic[current.varno]]); - } - - /* Return working array to pool */ - if(nzpcol == NULL) - mempool_releaseVector(lp->workarrays, (char *) nzlist, FALSE); - - if(lp->spx_trace) - report(lp, DETAILED, "row_prim: %d, pivot size = " RESULTVALUEMASK "\n", - current.varno, current.pivot); - -/* *theta = current.theta; */ - *theta = fabs(current.theta); - - return(current.varno); -} /* rowprim */ - - -/* Find the dual simplex leaving basic variable */ -STATIC int rowdual(lprec *lp, REAL *rhvec, MYBOOL forceoutEQ, MYBOOL updateinfeas, REAL *xviol) -{ - int k, i, iy, iz, ii, ninfeas; - register REAL rh; - REAL up, lo = 0, - epsvalue, sinfeas, xinfeas; - pricerec current, candidate; - MYBOOL collectMP = FALSE; - - /* Initialize */ - if(rhvec == NULL) - rhvec = lp->rhs; - epsvalue = lp->epsdual; - current.pivot = -epsvalue; /* Initialize leaving variable threshold; "less than 0" */ - current.theta = 0; - current.varno = 0; - current.isdual = TRUE; - current.lp = lp; - candidate.isdual = TRUE; - candidate.lp = lp; - - /* Loop over active partial row set */ - if(is_action(lp->piv_strategy, PRICE_FORCEFULL)) { - k = 1; - iy = lp->rows; - } - else { - k = partial_blockStart(lp, TRUE); - iy = partial_blockEnd(lp, TRUE); - } - ninfeas = 0; - xinfeas = 0; - sinfeas = 0; - makePriceLoop(lp, &k, &iy, &iz); - iy *= iz; - for(; k*iz <= iy; k += iz) { - - /* Map loop variable to target */ - i = k; - - /* Check if the pivot candidate is on the block-list */ - if(lp->rejectpivot[0] > 0) { - int kk; - for(kk = 1; (kk <= lp->rejectpivot[0]) && (i != lp->rejectpivot[kk]); kk++); - if(kk <= lp->rejectpivot[0]) - continue; - } - - /* Set local variables - express violation as a negative number */ - ii = lp->var_basic[i]; - up = lp->upbo[ii]; - lo = 0; - rh = rhvec[i]; - if(rh > up) - rh = up - rh; - else - rh -= lo; - up -= lo; - - /* Analyze relevant constraints ... - KE version skips uninteresting alternatives and gives a noticeable speedup */ -/* if((rh < -epsvalue*sqrt(lp->matA->rowmax[i])) || */ - if((rh < -epsvalue) || - ((forceoutEQ == TRUE) && (up < epsvalue))) { /* It causes instability to remove the "TRUE" test */ - - /* Accumulate stats */ - ninfeas++; - SETMIN(xinfeas, rh); - sinfeas += rh; - - /* Give a slight preference to fixed variables (mainly equality slacks) */ - if(up < epsvalue) { - /* Break out immediately if we are directed to force slacks out of the basis */ - if(forceoutEQ == TRUE) { - current.varno = i; - current.pivot = -1; - break; - } - /* Give an extra early boost to equality slack elimination, if specified */ - if(forceoutEQ == AUTOMATIC) - rh *= 10.0; - else /* .. or just the normal. marginal boost */ - rh *= 1.0+lp->epspivot; - } - - /* Select leaving variable according to strategy (the most negative/largest violation) */ - candidate.pivot = normalizeEdge(lp, i, rh, TRUE); - candidate.varno = i; - if(findImprovementVar(¤t, &candidate, collectMP, NULL)) - break; - } - } - - /* Verify infeasibility */ - if(updateinfeas) - lp->suminfeas = fabs(sinfeas); - if((ninfeas > 1) && - !verify_stability(lp, FALSE, xinfeas, sinfeas, ninfeas)) { - report(lp, IMPORTANT, "rowdual: Check for reduced accuracy and tolerance settings.\n"); - current.varno = 0; - } - - /* Produce statistics */ - if(lp->spx_trace) { - report(lp, NORMAL, "rowdual: Infeasibility sum " RESULTVALUEMASK " in %7d constraints.\n", - sinfeas, ninfeas); - if(current.varno > 0) { - report(lp, DETAILED, "rowdual: rhs[%d] = " RESULTVALUEMASK "\n", - current.varno, lp->rhs[current.varno]); - } - else - report(lp, FULL, "rowdual: Optimality - No primal infeasibilities found\n"); - } - if(xviol != NULL) - *xviol = fabs(xinfeas); - - return(current.varno); -} /* rowdual */ - - -STATIC void longdual_testset(lprec *lp, int which, int rownr, REAL *prow, int *nzprow, - REAL *drow, int *nzdrow) -{ - int i,j; - REAL F = lp->infinite; - if(which == 0) { /* Maros Example-1 - raw data */ - j = 1; i = lp->rows+j; lp->upbo[i] = 0; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 2; drow[i] = -1; - j = 2; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -2; drow[i] = 2; - j = 3; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 1; drow[i] = 5; - j = 4; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = 3; drow[i] = -6; - j = 5; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = -4; drow[i] = -2; - j = 6; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -1; drow[i] = 0; - j = 7; i = lp->rows+j; lp->upbo[i] = 2; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = 1; drow[i] = 0; - j = 8; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = -2; drow[i] = 0; - j = 9; i = lp->rows+j; lp->upbo[i] = 5; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -1; drow[i] = 4; - j = 10; i = lp->rows+j; lp->upbo[i] = F; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -2; drow[i] = 10; - nzprow[0] = i-lp->rows; - lp->rhs[rownr] = -11; - lp->upbo[lp->var_basic[rownr]] = F; - lp->rhs[0] = 1; - } - else if(which == 1) { /* Maros Example-1 - presorted in correct order */ - j = 1; i = lp->rows+j; lp->upbo[i] = 0; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 2; drow[i] = -1; - j = 2; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 1; drow[i] = 5; - j = 3; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = -4; drow[i] = -2; - j = 4; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = -2; drow[i] = 0; - - j = 5; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -1; drow[i] = 0; - j = 6; i = lp->rows+j; lp->upbo[i] = 2; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = 1; drow[i] = 0; - j = 7; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -2; drow[i] = 2; - j = 8; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = 3; drow[i] = -6; - j = 9; i = lp->rows+j; lp->upbo[i] = 5; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -1; drow[i] = 4; - j = 10; i = lp->rows+j; lp->upbo[i] = F; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -2; drow[i] = 10; - nzprow[0] = i-lp->rows; - lp->rhs[rownr] = -11; - lp->upbo[lp->var_basic[rownr]] = F; - lp->rhs[0] = 1; - } - - else if(which == 10) { /* Maros Example-2 - raw data */ - j = 1; i = lp->rows+j; lp->upbo[i] = 5; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = -2; drow[i] = 2; - j = 2; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 3; drow[i] = 3; - j = 3; i = lp->rows+j; lp->upbo[i] = 1; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = -2; drow[i] = 0; - j = 4; i = lp->rows+j; lp->upbo[i] = 2; lp->is_lower[i] = FALSE; nzprow[j] = i; prow[i] = -1; drow[i] = -2; - j = 5; i = lp->rows+j; lp->upbo[i] = 2; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 1; drow[i] = 0; - j = 6; i = lp->rows+j; lp->upbo[i] = F; lp->is_lower[i] = TRUE; nzprow[j] = i; prow[i] = 3; drow[i] = 9; - nzprow[0] = i-lp->rows; - lp->rhs[rownr] = 14; - lp->upbo[lp->var_basic[rownr]] = 2; - lp->rhs[0] = 6; - } -} - - -/* Find the dual simplex entering non-basic variable */ -STATIC int coldual(lprec *lp, int row_nr, REAL *prow, int *nzprow, - REAL *drow, int *nzdrow, - MYBOOL dualphase1, MYBOOL skipupdate, - int *candidatecount, REAL *xviol) -{ - int i, iy, iz, ix, k, nbound; - LREAL w, g, quot; - REAL viol, p, epspivot = lp->epspivot; -#ifdef MachinePrecRoundRHS - REAL epsvalue = lp->epsmachine; -#else - REAL epsvalue = lp->epsvalue; -#endif - pricerec current, candidate; - MYBOOL isbatch = FALSE, /* Requires that lp->longsteps->size > lp->sum */ - dolongsteps = (MYBOOL) (lp->longsteps != NULL); - - /* Initialize */ - if(xviol != NULL) - *xviol = lp->infinite; - if(dolongsteps && !dualphase1) - dolongsteps = AUTOMATIC; /* Sets Phase1 = TRUE, Phase2 = AUTOMATIC */ - current.theta = lp->infinite; - current.pivot = 0; - current.varno = 0; - current.epspivot = epspivot; - current.isdual = TRUE; - current.lp = lp; - candidate.epspivot = epspivot; - candidate.isdual = TRUE; - candidate.lp = lp; - *candidatecount = 0; - - /* Compute reduced costs */ - if(!skipupdate) { -#ifdef UseDualReducedCostUpdate - /* Recompute from scratch only at the beginning, otherwise update */ - if((lp->current_iter > 0) && (refactRecent(lp) < AUTOMATIC)) - compute_reducedcosts(lp, TRUE, row_nr, NULL, TRUE, - prow, nzprow, - NULL, NULL, - MAT_ROUNDDEFAULT); - else -#endif - compute_reducedcosts(lp, TRUE, row_nr, NULL, TRUE, - prow, nzprow, - drow, nzdrow, - MAT_ROUNDDEFAULT); - } - -#if 0 - /* Override all above to do in-line testing with fixed test set */ - if(lp->rows > 1 && lp->columns > 10) - longdual_testset(lp, 10, row_nr, prow, nzprow, drow, nzdrow); -#endif - - /* Compute the current violation of the bounds of the outgoing variable, - negative for violation of lower bound, positive for upper bound violation. - (Basic variables are always lower-bounded, by lp_solve convention) */ - g = 1; - viol = lp->rhs[row_nr]; - if(viol > 0) { /* Check if the leaving variable is >= its upper bound */ - p = lp->upbo[lp->var_basic[row_nr]]; - if(p < lp->infinite) { - viol -= p; - my_roundzero(viol, epsvalue); - if(viol > 0) - g = -1; - } - /* Do validation of numerics */ - if(g == 1) { - if(viol >= lp->infinite) { - report(lp, IMPORTANT, "coldual: Large basic solution value %g at iter %.0f indicates numerical instability\n", - lp->rhs[row_nr], (double) get_total_iter(lp)); - lp->spx_status = NUMFAILURE; - return( 0 ); - - } - if(skipupdate) - report(lp, DETAILED, "coldual: Inaccurate bound-flip accuracy at iter %.0f\n", - (double) get_total_iter(lp)); - else - report(lp, SEVERE, "coldual: Leaving variable %d does not violate bounds at iter %.0f\n", - row_nr, (double) get_total_iter(lp)); - return( -1 ); - } - } - - /* Update local value of pivot setting */ - lp->_piv_rule_ = get_piv_rule(lp); - - /* Condense list of relevant targets */ - p = 0; - k = 0; - nbound = 0; - ix = 1; - iy = nzprow[0]; - for(ix = 1; ix <= iy; ix++) { - i = nzprow[ix]; - w = prow[i] * g; /* Change sign if upper bound of the leaving variable is violated */ - /* Change sign if the non-basic variable is currently upper-bounded */ - /* w *= 2*lp->is_lower[i] - 1; */ /* fails on AIX!!! */ - w = my_chsign(!lp->is_lower[i], w); - - /* Check if the candidate is worth using for anything */ - if(w < -epsvalue) { - /* Tally bounded variables */ - if(lp->upbo[i] < lp->infinite) - nbound++; - - /* Update the nz-index */ - k++; - nzprow[k] = i; - SETMAX(p, -w); - } -#ifdef Paranoia - else { - if(lp->spx_trace) { - report(lp, FULL, "coldual: Candidate variable prow[%d] rejected with %g too small\n", - i, w); - } - } -#endif - - } - nzprow[0] = k; - if(xviol != NULL) - *xviol = p; - -#ifdef UseRelativePivot_Dual -/* epspivot *= sqrt(lp->matA->dynrange) / lp->matA->infnorm; */ - epspivot /= MAX(1, sqrt(lp->matA->rowmax[row_nr])); -#endif - current.epspivot = epspivot; - candidate.epspivot = epspivot; - - /* Initialize the long-step structures if indicated */ - if(dolongsteps) { - if((nzprow[0] <= 1) || (nbound == 0)) { /* Don't bother */ - dolongsteps = FALSE; - lp->longsteps->indexSet[0] = 0; - } - else { - multi_restart(lp->longsteps); - multi_valueInit(lp->longsteps, g*viol, lp->rhs[0]); - } - } - - /* Loop over all entering column candidates */ - ix = 1; - iy = nzprow[0]; - makePriceLoop(lp, &ix, &iy, &iz); - iy *= iz; - for(; ix*iz <= iy; ix += iz) { - i = nzprow[ix]; - - /* Compute the dual ratio (prow = w and drow = cbar in Chvatal's "nomenclatura") */ - w = prow[i] * g; /* Change sign if upper bound of the leaving variable is violated */ - quot = -drow[i] / w; /* Remember this sign-reversal in multi_recompute! */ - - /* Apply the selected pivot strategy (smallest theta) */ - candidate.theta = quot; /* Note that abs() is applied in findSubstitutionVar */ - candidate.pivot = w; - candidate.varno = i; - - /* Collect candidates for minor iterations/bound flips */ - if(dolongsteps) { - if(isbatch && (ix == iy)) - isbatch = AUTOMATIC; - if(collectMinorVar(&candidate, lp->longsteps, (MYBOOL) (dolongsteps == AUTOMATIC), isbatch) && - lp->spx_trace) - report(lp, DETAILED, "coldual: Long-dual break point with %d bound-flip variables\n", - lp->longsteps->used); - if(lp->spx_status == FATHOMED) - return( 0 ); - } - - /* We have a candidate for entering the basis; check if it is better than the incumbent */ - else if(findSubstitutionVar(¤t, &candidate, candidatecount)) - break; - } - - /* Set entering variable and long-step bound swap variables */ - if(dolongsteps) { - *candidatecount = lp->longsteps->used; - i = multi_enteringvar(lp->longsteps, NULL, 3); - } - else - i = current.varno; - - if(lp->spx_trace) - report(lp, NORMAL, "coldual: Entering column %d, reduced cost %g, pivot value %g, bound swaps %d\n", - i, drow[i], prow[i], multi_used(lp->longsteps)); - - return( i ); -} /* coldual */ - - -INLINE REAL normalizeEdge(lprec *lp, int item, REAL edge, MYBOOL isdual) -{ -#if 1 - /* Don't use the pricer "close to home", since this can possibly - worsen the final feasibility picture (mainly a Devex issue?) */ - if(fabs(edge) > lp->epssolution) -#endif - edge /= getPricer(lp, item, isdual); - if((lp->piv_strategy & PRICE_RANDOMIZE) != 0) - edge *= (1.0-PRICER_RANDFACT) + PRICER_RANDFACT*rand_uniform(lp, 1.0); - return( edge ); - -} - -/* Support routines for block detection and partial pricing */ -STATIC int partial_findBlocks(lprec *lp, MYBOOL autodefine, MYBOOL isrow) -{ - int i, jj, n, nb, ne, items; - REAL hold, biggest, *sum = NULL; - MATrec *mat = lp->matA; - partialrec *blockdata; - - if(!mat_validate(mat)) - return( 1 ); - - blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - items = IF(isrow, lp->rows, lp->columns); - allocREAL(lp, &sum, items+1, FALSE); - - /* Loop over items and compute the average column index for each */ - sum[0] = 0; - for(i = 1; i <= items; i++) { - n = 0; - if(isrow) { - nb = mat->row_end[i-1]; - ne = mat->row_end[i]; - } - else { - nb = mat->col_end[i-1]; - ne = mat->col_end[i]; - } - n = ne-nb; - sum[i] = 0; - if(n > 0) { - if(isrow) - for(jj = nb; jj < ne; jj++) - sum[i] += ROW_MAT_COLNR(jj); - else - for(jj = nb; jj < ne; jj++) - sum[i] += COL_MAT_ROWNR(jj); - sum[i] /= n; - } - else - sum[i] = sum[i-1]; - } - - /* Loop over items again, find largest difference and make monotone */ - hold = 0; - biggest = 0; - for(i = 2; i <= items; i++) { - hold = sum[i] - sum[i-1]; - if(hold > 0) { - if(hold > biggest) - biggest = hold; - } - else - hold = 0; - sum[i-1] = hold; - } - - /* Loop over items again and find differences exceeding threshold; - the discriminatory power of this routine depends strongly on the - magnitude of the scaling factor - from empirical evidence > 0.9 */ - biggest = MAX(1, 0.9*biggest); - n = 0; - nb = 0; - ne = 0; - for(i = 1; i < items; i++) - if(sum[i] > biggest) { - ne += i-nb; /* Compute sum of index gaps between maxima */ - nb = i; - n++; /* Increment count */ - } - - /* Clean up */ - FREE(sum); - - /* Require that the maxima are spread "nicely" across the columns, - otherwise return that there is only one monolithic block. - (This is probably an area for improvement in the logic!) */ - if(n > 0) { - ne /= n; /* Average index gap between maxima */ - i = IF(isrow, lp->columns, lp->rows); - nb = i / ne; /* Another estimated block count */ - if(abs(nb - n) > 2) /* Probably Ok to require equality (nb==n)*/ - n = 1; - else if(autodefine) /* Generate row/column break-indeces for partial pricing */ - set_partialprice(lp, nb, NULL, isrow); - } - else - n = 1; - - return( n ); -} -STATIC int partial_blockStart(lprec *lp, MYBOOL isrow) -{ - partialrec *blockdata; - - blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - if(blockdata == NULL) - return( 1 ); - else { - if((blockdata->blocknow < 1) || (blockdata->blocknow > blockdata->blockcount)) - blockdata->blocknow = 1; - return( blockdata->blockend[blockdata->blocknow-1] ); - } -} -STATIC int partial_blockEnd(lprec *lp, MYBOOL isrow) -{ - partialrec *blockdata; - - blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - if(blockdata == NULL) - return( IF(isrow, lp->rows, lp->sum) ); - else { - if((blockdata->blocknow < 1) || (blockdata->blocknow > blockdata->blockcount)) - blockdata->blocknow = 1; - return( blockdata->blockend[blockdata->blocknow]-1 ); - } -} -STATIC int partial_blockNextPos(lprec *lp, int block, MYBOOL isrow) -{ - partialrec *blockdata; - - blockdata = IF(isrow, lp->rowblocks, lp->colblocks); -#ifdef Paranoia - if((blockdata == NULL) || (block <= 1) || (block > blockdata->blockcount)) { - report(lp, SEVERE, "partial_blockNextPos: Invalid block %d specified.\n", - block); - return( -1 ); - } -#endif - block--; - if(blockdata->blockpos[block] == blockdata->blockend[block+1]) - blockdata->blockpos[block] = blockdata->blockend[block]; - else - blockdata->blockpos[block]++; - return( blockdata->blockpos[block] ); -} -STATIC MYBOOL partial_blockStep(lprec *lp, MYBOOL isrow) -{ - partialrec *blockdata; - - blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - if(blockdata == NULL) - return( FALSE ); - else if(blockdata->blocknow < blockdata->blockcount) { - blockdata->blocknow++; - return( TRUE); - } - else { - blockdata->blocknow = 1; - return( TRUE ); - } -} -STATIC MYBOOL partial_isVarActive(lprec *lp, int varno, MYBOOL isrow) -{ - partialrec *blockdata; - - blockdata = IF(isrow, lp->rowblocks, lp->colblocks); - if(blockdata == NULL) - return( TRUE ); - else { - return( (MYBOOL) ((varno >= blockdata->blockend[blockdata->blocknow-1]) && - (varno < blockdata->blockend[blockdata->blocknow])) ); - } -} - - -/* Multiple pricing routines */ -STATIC multirec *multi_create(lprec *lp, MYBOOL truncinf) -{ - multirec *multi; - - multi = (multirec *) calloc(1, sizeof(*multi)); - if(multi != NULL) { - multi->active = 1; - multi->lp = lp; - multi->epszero = lp->epsprimal; - multi->truncinf = truncinf; - } - - return(multi); -} -STATIC void multi_free(multirec **multi) -{ - if((multi == NULL) || (*multi == NULL)) - return; - FREE((*multi)->items); - FREE((*multi)->valueList); - FREE((*multi)->indexSet); - FREE((*multi)->freeList); - FREE((*multi)->sortedList); - FREE(*multi); -} -STATIC MYBOOL multi_mustupdate(multirec *multi) -{ - return( (MYBOOL) ((multi != NULL) && - (multi->used < multi->limit)) ); -} -STATIC MYBOOL multi_resize(multirec *multi, int blocksize, int blockdiv, MYBOOL doVlist, MYBOOL doIset) -{ - MYBOOL ok = TRUE; - - if((blocksize > 1) && (blockdiv > 0)) { - int oldsize = multi->size; - - multi->size = blocksize; - if(blockdiv > 1) - multi->limit += (multi->size-oldsize) / blockdiv; - - multi->items = (pricerec *) realloc(multi->items, (multi->size+1)*sizeof(*(multi->items))); - multi->sortedList = (UNIONTYPE QSORTrec *) realloc(multi->sortedList, (multi->size+1)*sizeof(*(multi->sortedList))); - ok = (multi->items != NULL) && (multi->sortedList != NULL) && - allocINT(multi->lp, &(multi->freeList), multi->size+1, AUTOMATIC); - if(ok) { - int i, n; - - if(oldsize == 0) - i = 0; - else - i = multi->freeList[0]; - multi->freeList[0] = i + (multi->size-oldsize); - for(n = multi->size - 1, i++; i <= multi->freeList[0]; i++, n--) - multi->freeList[i] = n; - } - if(doVlist) - ok &= allocREAL(multi->lp, &(multi->valueList), multi->size+1, AUTOMATIC); - if(doIset) { - ok &= allocINT(multi->lp, &(multi->indexSet), multi->size+1, AUTOMATIC); - if(ok && (oldsize == 0)) - multi->indexSet[0] = 0; - } - if(!ok) - goto Undo; - - } - else { -Undo: - multi->size = 0; - FREE(multi->items); - FREE(multi->valueList); - FREE(multi->indexSet); - FREE(multi->freeList); - FREE(multi->sortedList); - } - multi->active = 1; - - return( ok ); -} - -STATIC int multi_size(multirec *multi) -{ - if(multi == NULL) - return( 0 ); - else - return( multi->size ); -} - -STATIC int multi_used(multirec *multi) -{ - if(multi == NULL) - return( 0 ); - else - return( multi->used ); -} - -STATIC int multi_restart(multirec *multi) -{ - int i, n = multi->used; - - multi->used = 0; - multi->sorted = FALSE; - multi->dirty = FALSE; - if(multi->freeList != NULL) { - for(i = 1; i <= multi->size; i++) - multi->freeList[i] = multi->size - i; - multi->freeList[0] = multi->size; - } -#if 0 - if(multi->indexSet != NULL) - multi->indexSet[0] = 0; -#endif - return( n ); -} - -STATIC void multi_valueInit(multirec *multi, REAL step_base, REAL obj_base) -{ - multi->step_base = multi->step_last = step_base; - multi->obj_base = multi->obj_last = obj_base; -#ifdef Paranoia - if(step_base > 0) - report(multi->lp, SEVERE, "multi_valueInit: Positive constraint violation %g provided at iteration %6.0f\n", - step_base, (double) get_total_iter(multi->lp)); -#endif -} - -STATIC REAL *multi_valueList(multirec *multi) -{ - return(multi->valueList); -} - -STATIC int *multi_indexSet(multirec *multi, MYBOOL regenerate) -{ - if(regenerate) - multi_populateSet(multi, NULL, -1); - return(multi->indexSet); -} - -STATIC int multi_getvar(multirec *multi, int item) -{ -#ifdef Paranoia - if((item < 1) || (item >= multi->size)) - return(-1); -#endif - return( ((pricerec *) &(multi->sortedList[item].pvoidreal.ptr))->varno ); -} - -STATIC MYBOOL multi_recompute(multirec *multi, int index, MYBOOL isphase2, MYBOOL fullupdate) -{ - int i, n; - REAL lB, uB, Alpha, this_theta, prev_theta; - lprec *lp = multi->lp; - pricerec *thisprice; - - /* Define target update window */ - if(multi->dirty) { - index = 0; - n = multi->used - 1; - } - else if(fullupdate) - n = multi->used - 1; - else - n = index; - - /* Initialize accumulators from the specified update index */ - if(index == 0) { - multi->maxpivot = 0; - multi->maxbound = 0; - multi->step_last = multi->step_base; - multi->obj_last = multi->obj_base; - thisprice = NULL; - this_theta = 0; - } - else { - multi->obj_last = multi->valueList[index-1]; - multi->step_last = multi->sortedList[index-1].pvoidreal.realval; - thisprice = (pricerec *) (multi->sortedList[index-1].pvoidreal.ptr); - this_theta = thisprice->theta; - } - - /* Update step lengths and objective values */ - while((index <= n) && (multi->step_last < multi->epszero)) { - - /* Update parameters for this loop */ - prev_theta = this_theta; - thisprice = (pricerec *) (multi->sortedList[index].pvoidreal.ptr); - this_theta = thisprice->theta; - Alpha = fabs(thisprice->pivot); - uB = lp->upbo[thisprice->varno]; - lB = 0; - SETMAX(multi->maxpivot, Alpha); - SETMAX(multi->maxbound, uB); - - /* Do the value updates */ - if(isphase2) { - multi->obj_last += (this_theta - prev_theta) * multi->step_last; /* Sign-readjusted from coldual()/Maros */ - if(uB >= lp->infinite) - multi->step_last = lp->infinite; - else - multi->step_last += Alpha*(uB-lB); - } - else { - multi->obj_last += (this_theta - prev_theta) * multi->step_last; /* Sign-readjusted from coldual()/Maros */ - multi->step_last += Alpha; - } - - /* Store updated values at the indexed locations */ - multi->sortedList[index].pvoidreal.realval = multi->step_last; - multi->valueList[index] = multi->obj_last; -#ifdef Paranoia - if(lp->spx_trace && - (multi->step_last > lp->infinite)) - report(lp, SEVERE, "multi_recompute: A very large step-size %g was generated at iteration %6.0f\n", - multi->step_last, (double) get_total_iter(lp)); -#endif - index++; - } - - /* Discard candidates entered earlier that now make the OF worsen, and - make sure that the released positions are added to the free list. */ - n = index; - while(n < multi->used) { - i = ++multi->freeList[0]; - multi->freeList[i] = (int) (((pricerec *) multi->sortedList[n].pvoidreal.ptr) - multi->items); - n++; - } - multi->used = index; - if(multi->sorted && (index == 1)) - multi->sorted = FALSE; - multi->dirty = FALSE; - - /* Return TRUE if the step is now positive */ - return( (MYBOOL) (multi->step_last >= multi->epszero) ); -} - -STATIC MYBOOL multi_truncatingvar(multirec *multi, int varnr) -{ - return( multi->truncinf && is_infinite(multi->lp, multi->lp->upbo[varnr]) ); -} - -STATIC MYBOOL multi_removevar(multirec *multi, int varnr) -{ - int i = 1; - int *coltarget = multi->indexSet; - - if(coltarget == NULL) - return( FALSE ); - - while((i <= multi->used) && (coltarget[i] != varnr)) - i++; - if(i > multi->used) - return( FALSE ); - - for(; i < multi->used; i++) - coltarget[i] = coltarget[i+1]; - coltarget[0]--; - multi->used--; - multi->dirty = TRUE; - return( TRUE ); -} - -STATIC int multi_enteringvar(multirec *multi, pricerec *current, int priority) -{ - lprec *lp = multi->lp; - int i = 0, bestindex, colnr; - REAL bound, score, bestscore = -lp->infinite; - REAL b1, b2, b3; - pricerec *candidate, *bestcand; - - /* Check that we have a candidate */ - multi->active = bestindex = 0; - if((multi == NULL) || (multi->used == 0)) - return( bestindex ); - - /* Check for pruning possibility of the B&B tree */ - if(multi->objcheck && (lp->solutioncount > 0) && - bb_better(lp, OF_WORKING | OF_PROJECTED, OF_TEST_WE)) { - lp->spx_status = FATHOMED; - return( bestindex ); - } - - /* Check the trivial case */ - if(multi->used == 1) { - bestcand = (pricerec *) (multi->sortedList[bestindex].pvoidreal.ptr); - goto Finish; - } - - /* Set priority weights */ -Redo: - switch(priority) { - case 0: b1 = 0.0, b2 = 0.0, b3 = 1.0; /* Only OF */ - bestindex = multi->used - 2; break; - case 1: b1 = 0.2, b2 = 0.3, b3 = 0.5; break; /* Emphasize OF */ - case 2: b1 = 0.3, b2 = 0.5, b3 = 0.2; break; /* Emphasize bound */ - case 3: b1 = 0.6, b2 = 0.2, b3 = 0.2; break; /* Emphasize pivot */ - case 4: b1 = 1.0, b2 = 0.0, b3 = 0.0; break; /* Only pivot */ - default: b1 = 0.4, b2 = 0.2, b3 = 0.4; /* Balanced default */ - } - bestcand = (pricerec *) (multi->sortedList[bestindex].pvoidreal.ptr); - - /* Loop over all candidates to get the best entering candidate; - start at the end to try to maximize the chain length */ - for(i = multi->used - 1; i >= 0; i--) { - candidate = (pricerec *) (multi->sortedList[i].pvoidreal.ptr); - colnr = candidate->varno; - bound = lp->upbo[colnr]; - score = fabs(candidate->pivot) / multi->maxpivot; - score = pow(1.0 + score , b1) * - pow(1.0 + log(bound / multi->maxbound + 1), b2) * - pow(1.0 + (REAL) i / multi->used , b3); - if(score > bestscore) { - bestscore = score; - bestindex = i; - bestcand = candidate; - } - } - - /* Do pivot protection */ - if((priority < 4) && (fabs(bestcand->pivot) < lp->epssolution)) { - bestindex = 0; - priority++; - goto Redo; - } - -Finish: - /* Make sure we shrink the list and update */ - multi->active = colnr = bestcand->varno; - if(bestindex < multi->used - 1) { -#if 0 -/* if(lp->upbo[colnr] >= lp->infinite) */ - QS_swap(multi->sortedList, bestindex, multi->used-1); - multi_recompute(multi, bestindex, (bestcand->isdual == AUTOMATIC), TRUE); -#else - multi->used = i + 1; -#endif - } - multi_populateSet(multi, NULL, multi->active); - - /* Compute the entering theta and update parameters */ - score = (multi->used == 1 ? multi->step_base : multi->sortedList[multi->used-2].pvoidreal.realval); - score /= bestcand->pivot; - score = my_chsign(!lp->is_lower[multi->active], score); - - if(lp->spx_trace && - (fabs(score) > 1/lp->epsprimal)) - report(lp, IMPORTANT, "multi_enteringvar: A very large Theta %g was generated (pivot %g)\n", - score, bestcand->pivot); - multi->step_base = score; - if(current != NULL) - *current = *bestcand; - - return( multi->active ); -} - -STATIC REAL multi_enteringtheta(multirec *multi) -{ - return( multi->step_base ); -} - -STATIC int multi_populateSet(multirec *multi, int **list, int excludenr) -{ - int n = 0; - if(list == NULL) - list = &(multi->indexSet); - if((multi->used > 0) && - ((*list != NULL) || allocINT(multi->lp, list, multi->size+1, FALSE))) { - int i, colnr; - - for(i = 0; i < multi->used; i++) { - colnr = ((pricerec *) (multi->sortedList[i].pvoidreal.ptr))->varno; - if((colnr != excludenr) && - /* Prevent an unbounded variable from "bound-flip"; this could - actually indicate that we should let the entering variable be - bound-swapped (in the case that it is bounded), but we - disregard this possibility here, since it brings with it - issues of pivot size, etc. */ - ((excludenr > 0) && (multi->lp->upbo[colnr] < multi->lp->infinite))) { - n++; - (*list)[n] = colnr; - } - } - (*list)[0] = n; - } - return( n ); -} - diff --git a/code/3rd_lpsolve/lp_price.h b/code/3rd_lpsolve/lp_price.h deleted file mode 100644 index 9e8d8bcd..00000000 --- a/code/3rd_lpsolve/lp_price.h +++ /dev/null @@ -1,99 +0,0 @@ -#ifndef HEADER_lp_price -#define HEADER_lp_price - -/* Local defines */ -/* ------------------------------------------------------------------------- */ -#define UseSortOnBound_Improve -/*#define UseSortOnBound_Substitute*/ - -#if 0 /* Stricter feasibility-preserving tolerance; use w/ *_UseRejectionList */ - #define UseRelativeFeasibility /* Use machine-precision and A-scale data */ -#endif -#if 0 /* Stricter pivot-selection criteria; use w/ *UseRejectionList */ - #define UseRelativePivot_Primal /* In rowprim based on A-scale data */ - #define UseRelativePivot_Dual /* In coldual based on A-scale data */ -#endif - - -/* Include required library headers */ -/* ------------------------------------------------------------------------- */ -#include "lp_types.h" - - -#ifdef __cplusplus -extern "C" { -#endif - -/* Comparison and validity routines */ -int CMP_CALLMODEL compareImprovementVar(const pricerec *current, const pricerec *candidate); -int CMP_CALLMODEL compareSubstitutionVar(const pricerec *current, const pricerec *candidate); -int CMP_CALLMODEL compareBoundFlipVar(const pricerec *current, const pricerec *candidate); -STATIC int addCandidateVar(pricerec *candidate, multirec *multi, findCompare_func findCompare, MYBOOL allowSortedExpand); -STATIC MYBOOL collectMinorVar(pricerec *candidate, multirec *longsteps, MYBOOL isphase2, MYBOOL isbatch); -STATIC MYBOOL validImprovementVar(pricerec *candidate); -STATIC MYBOOL validSubstitutionVar(pricerec *candidate); - -/* Row+column selection routines */ -STATIC MYBOOL findImprovementVar(pricerec *current, pricerec *candidate, MYBOOL collectMP, int *candidatecount); -STATIC MYBOOL findSubstitutionVar(pricerec *current, pricerec *candidate, int *candidatecount); -INLINE REAL normalizeEdge(lprec *lp, int item, REAL edge, MYBOOL isdual); -STATIC void makePriceLoop(lprec *lp, int *start, int *end, int *delta); - -/* Computation of reduced costs */ -STATIC void update_reducedcosts(lprec *lp, MYBOOL isdual, int leave_nr, int enter_nr, REAL *prow, REAL *drow); -STATIC void compute_reducedcosts(lprec *lp, MYBOOL isdual, int row_nr, int *coltarget, MYBOOL dosolve, - REAL *prow, int *nzprow, - REAL *drow, int *nzdrow, - int roundmode); - -/* Leaving variable selection and entering column pricing loops */ -STATIC int find_rowReplacement(lprec *lp, int rownr, REAL *prow, int *nzprow); -STATIC int colprim(lprec *lp, REAL *drow, int *nzdrow, - MYBOOL skipupdate, int partialloop, int *candidatecount, MYBOOL updateinfeas, REAL *xviol); -STATIC int rowprim(lprec *lp, int colnr, LREAL *theta, REAL *pcol, int *nzpcol, MYBOOL forceoutEQ, REAL *xviol); -STATIC int rowdual(lprec *lp, REAL *rhvec, MYBOOL forceoutEQ, MYBOOL updateinfeas, REAL *xviol); -STATIC int coldual(lprec *lp, int row_nr, - REAL *prow, int *nzprow, REAL *drow, int *nzdrow, - MYBOOL dualphase1, MYBOOL skipupdate, - int *candidatecount, REAL *xviol); - -/* Partial pricing management routines */ -STATIC partialrec *partial_createBlocks(lprec *lp, MYBOOL isrow); -STATIC int partial_countBlocks(lprec *lp, MYBOOL isrow); -STATIC int partial_activeBlocks(lprec *lp, MYBOOL isrow); -STATIC void partial_freeBlocks(partialrec **blockdata); - -/* Partial pricing utility routines */ -STATIC int partial_findBlocks(lprec *lp, MYBOOL autodefine, MYBOOL isrow); -STATIC int partial_blockStart(lprec *lp, MYBOOL isrow); -STATIC int partial_blockEnd(lprec *lp, MYBOOL isrow); -STATIC int partial_blockNextPos(lprec *lp, int block, MYBOOL isrow); - -STATIC MYBOOL partial_blockStep(lprec *lp, MYBOOL isrow); -STATIC MYBOOL partial_isVarActive(lprec *lp, int varno, MYBOOL isrow); - -/* Multiple pricing / dual long step management routines */ -STATIC multirec *multi_create(lprec *lp, MYBOOL truncinf); -STATIC MYBOOL multi_resize(multirec *multi, int blocksize, int blockdiv, MYBOOL doVlist, MYBOOL doIset); -STATIC int multi_restart(multirec *multi); -STATIC int multi_size(multirec *multi); -STATIC int multi_used(multirec *multi); -STATIC MYBOOL multi_truncatingvar(multirec *multi, int varnr); -STATIC MYBOOL multi_mustupdate(multirec *multi); -STATIC void multi_valueInit(multirec *multi, REAL step_base, REAL obj_base); -STATIC REAL *multi_valueList(multirec *multi); -STATIC int *multi_indexSet(multirec *multi, MYBOOL regenerate); -STATIC int multi_getvar(multirec *multi, int item); -STATIC MYBOOL multi_recompute(multirec *multi, int index, MYBOOL isphase2, MYBOOL fullupdate); -STATIC MYBOOL multi_removevar(multirec *multi, int varnr); -STATIC int multi_enteringvar(multirec *multi, pricerec *current, int priority); -STATIC REAL multi_enteringtheta(multirec *multi); -STATIC void multi_free(multirec **multi); -STATIC int multi_populateSet(multirec *multi, int **list, int excludenr); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_price */ - diff --git a/code/3rd_lpsolve/lp_pricePSE.c b/code/3rd_lpsolve/lp_pricePSE.c deleted file mode 100644 index f77fdbe6..00000000 --- a/code/3rd_lpsolve/lp_pricePSE.c +++ /dev/null @@ -1,536 +0,0 @@ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_report.h" -#include "lp_pricePSE.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* - Advanced simplex price scaling modules - w/interface for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h - - Release notes: - v1.0.0 1 September 2003 Implementation of DEVEX and STEEPEST EDGE - routines for the primal and dual simplex. - v1.0.1 1 January 2004 Made initial value of weight of ingoing - variable for the standard mode of DEVEX - consistent with the initialization at restart; - original version could at worst contribute - to cycling. - v1.0.2 23 March 2004 Added floors to Steepest Edge updates and - moved tests for tiny update higher. Previous - logic can be simulated by disabling the compiler - define ApplySteepestEdgeMinimum. - v1.1.0 1 July 2004 Renamed from lp_pricerPSE to lp_pricePSE in - conjuction with the creation of a separate - price library. - v1.2.0 1 March 2005 Changed memory allocation routines to use - standard lp_solve functions, improve error handling - and return boolean status values. - - ---------------------------------------------------------------------------------- -*/ - -INLINE MYBOOL applyPricer(lprec *lp) -{ - int rule = get_piv_rule(lp); - return( (MYBOOL) ((rule == PRICER_DEVEX) || (rule == PRICER_STEEPESTEDGE)) ); -} - - -STATIC void simplexPricer(lprec *lp, MYBOOL isdual) -{ - if(lp->edgeVector != NULL) - lp->edgeVector[0] = (REAL) isdual; -} - - -STATIC void freePricer(lprec *lp) -{ - FREE(lp->edgeVector); -} - - -STATIC MYBOOL resizePricer(lprec *lp) -{ - if(!applyPricer(lp)) - return( TRUE ); - - /* Reallocate vector for new size */ - if(!allocREAL(lp, &(lp->edgeVector), lp->sum_alloc+1, AUTOMATIC)) - return( FALSE ); - - /* Signal that we have not yet initialized the price vector */ - MEMCLEAR(lp->edgeVector, lp->sum_alloc+1); - lp->edgeVector[0] = -1; - return( TRUE ); -} - - -STATIC MYBOOL initPricer(lprec *lp) -{ - if(!applyPricer(lp)) - return( FALSE ); - - /* Free any pre-existing pricer */ - freePricer(lp); - - /* Allocate vector to fit current problem size */ - return( resizePricer(lp) ); -} - - -STATIC REAL getPricer(lprec *lp, int item, MYBOOL isdual) -{ - REAL value = 1.0; - - if(!applyPricer(lp)) - return( value ); - - value = *lp->edgeVector; - - /* Make sure we have a price vector to use */ - if(value < 0) { -#ifdef Paranoia - report(lp, SEVERE, "getPricer: Called without having being initialized!\n"); -#endif - return( 1.0 ); - } - /* We may be calling the primal from the dual (and vice-versa) for validation - of feasibility; ignore calling origin and simply return 1 */ - else if(isdual != value) { - return( 1.0 ); - } - /* Do the normal norm retrieval */ - else { - - if(isdual) - item = lp->var_basic[item]; - - value = lp->edgeVector[item]; - - if(value == 0) { - value = 1.0; - report(lp, SEVERE, "getPricer: Detected a zero-valued price at index %d\n", item); - } -#ifdef Paranoia - else if(value < 0) - report(lp, SEVERE, "getPricer: Invalid %s reduced cost norm %g at index %d\n", - my_if(isdual, "dual", "primal"), value, item); -#endif - - /* Return the norm */ - return( sqrt(value) ); - } -} - -STATIC MYBOOL restartPricer(lprec *lp, MYBOOL isdual) -{ - REAL *sEdge = NULL, seNorm, hold; - int i, j, m; - MYBOOL isDEVEX, ok = applyPricer(lp); - - if(!ok) - return( ok ); - - /* Store the active/current pricing type */ - if(isdual == AUTOMATIC) - isdual = (MYBOOL) lp->edgeVector[0]; - else - lp->edgeVector[0] = isdual; - - m = lp->rows; - - /* Determine strategy and check if we have strategy fallback for the primal */ - isDEVEX = is_piv_rule(lp, PRICER_DEVEX); - if(!isDEVEX && !isdual) - isDEVEX = is_piv_mode(lp, PRICE_PRIMALFALLBACK); - - /* Check if we only need to do the simple DEVEX initialization */ - if(!is_piv_mode(lp, PRICE_TRUENORMINIT)) { - if(isdual) { - for(i = 1; i <= m; i++) - lp->edgeVector[lp->var_basic[i]] = 1.0; - } - else { - for(i = 1; i <= lp->sum; i++) - if(!lp->is_basic[i]) - lp->edgeVector[i] = 1.0; - } - return( ok ); - } - - /* Otherwise do the full Steepest Edge norm initialization */ - ok = allocREAL(lp, &sEdge, m+1, FALSE); - if(!ok) - return( ok ); - - if(isdual) { - - /* Extract the rows of the basis inverse and compute their squared norms */ - - for(i = 1; i <= m; i++) { - - bsolve(lp, i, sEdge, NULL, 0, 0.0); - - /* Compute the edge norm */ - seNorm = 0; - for(j = 1; j <= m; j++) { - hold = sEdge[j]; - seNorm += hold*hold; - } - - j = lp->var_basic[i]; - lp->edgeVector[j] = seNorm; - } - - } - else { - - /* Solve a=Bb for b over all non-basic variables and compute their squared norms */ - - for(i = 1; i <= lp->sum; i++) { - if(lp->is_basic[i]) - continue; - - fsolve(lp, i, sEdge, NULL, 0, 0.0, FALSE); - - /* Compute the edge norm */ - seNorm = 1; - for(j = 1; j <= m; j++) { - hold = sEdge[j]; - seNorm += hold*hold; - } - - lp->edgeVector[i] = seNorm; - } - - } - - FREE(sEdge); - - return( ok ); - -} - - -STATIC MYBOOL formWeights(lprec *lp, int colnr, REAL *pcol, REAL **w) -/* This computes Bw = a, where B is the basis and a is a column of A */ -{ - MYBOOL ok = allocREAL(lp, w, lp->rows+1, FALSE); - - if(ok) { - if(pcol == NULL) - fsolve(lp, colnr, *w, NULL, 0.0, 0.0, FALSE); - else { - MEMCOPY(*w, pcol, lp->rows+1); -/* *w[0] = 0; */ /* Test */ - } - } -/* - if(pcol != NULL) { - REAL cEdge, hold; - int i; - - cEdge = 0; - for(i = 1; i <= m; i++) { - hold = *w[i]-pcol[i]; - cEdge += hold*hold; - } - cEdge /= m; - cEdge = sqrt(cEdge); - if(cEdge > lp->epspivot) - report(lp, SEVERE, "updatePricer: MRS error is %g\n", cEdge); - } -*/ - return(ok); -} -STATIC void freeWeights(REAL *w) -{ - FREE(w); -} - - -STATIC MYBOOL updatePricer(lprec *lp, int rownr, int colnr, REAL *pcol, REAL *prow, int *nzprow) -{ - REAL *vEdge = NULL, cEdge, hold, *newEdge, *w = NULL; - int i, m, n, exitcol, errlevel = DETAILED; - MYBOOL forceRefresh = FALSE, isDual, isDEVEX, ok = FALSE; - - if(!applyPricer(lp)) - return(ok); - - /* Make sure we have something to update */ - hold = lp->edgeVector[0]; - if(hold < 0) - return(ok); - isDual = (MYBOOL) (hold > 0); - - /* Do common initializations and computations */ - m = lp->rows; - n = lp->sum; - isDEVEX = is_piv_rule(lp, PRICER_DEVEX); - exitcol = lp->var_basic[rownr]; - - /* Solve/copy Bw = a */ -#if 0 - ok = formWeights(lp, colnr, NULL, &w); /* Compute from scratch - Experimental */ -#else - ok = formWeights(lp, colnr, pcol, &w); /* Use previously computed values */ -#endif - if(!ok) - return( ok ); - - /* Price norms for the dual simplex - the basic columns */ - if(isDual) { - REAL rw; - int targetcol; - - /* Don't need to compute cross-products with DEVEX */ - if(!isDEVEX) { - ok = allocREAL(lp, &vEdge, m+1, FALSE); - if(!ok) - return( ok ); - - /* Extract the row of the inverse containing the leaving variable - and then form the dot products against the other variables, i.e. "Tau" */ -#if 0 /* Extract row explicitly */ - bsolve(lp, rownr, vEdge, 0, 0.0); -#else /* Reuse previously extracted row data */ - MEMCOPY(vEdge, prow, m+1); - vEdge[0] = 0; -#endif - lp->bfp_ftran_normal(lp, vEdge, NULL); - } - - /* Update the squared steepest edge norms; first store some constants */ - cEdge = lp->edgeVector[exitcol]; - rw = w[rownr]; - if(fabs(rw) < lp->epspivot) { - forceRefresh = TRUE; - goto Finish2; - } - - /* Deal with the variable entering the basis to become a new leaving candidate */ - hold = 1 / rw; - lp->edgeVector[colnr] = (hold*hold) * cEdge; - -#ifdef Paranoia - if(lp->edgeVector[colnr] <= lp->epsmachine) - report(lp, errlevel, "updatePricer: Invalid dual norm %g at entering index %d - iteration %.0f\n", - lp->edgeVector[colnr], rownr, (double) (lp->total_iter+lp->current_iter)); -#endif - - /* Then loop over all basic variables, but skip the leaving row */ - for(i = 1; i <= m; i++) { - if(i == rownr) - continue; - targetcol = lp->var_basic[i]; - hold = w[i]; - if(hold == 0) - continue; - hold /= rw; - if(fabs(hold) < lp->epsmachine) - continue; - - newEdge = &(lp->edgeVector[targetcol]); - *newEdge += (hold*hold) * cEdge; - if(isDEVEX) { - if((*newEdge) > DEVEX_RESTARTLIMIT) { - forceRefresh = TRUE; - break; - } - } - else { - *newEdge -= 2*hold*vEdge[i]; -#ifdef xxApplySteepestEdgeMinimum - SETMAX(*newEdge, hold*hold+1); /* Kludge; use the primal lower bound */ -#else - if(*newEdge <= 0) { - report(lp, errlevel, "updatePricer: Invalid dual norm %g at index %d - iteration %.0f\n", - *newEdge, i, (double) (lp->total_iter+lp->current_iter)); - forceRefresh = TRUE; - break; - } -#endif - } - } - - - } - /* Price norms for the primal simplex - the non-basic columns */ - else { - - REAL *vTemp = NULL, *vAlpha = NULL, cAlpha; - int *coltarget; - - ok = allocREAL(lp, &vTemp, m+1, TRUE) && - allocREAL(lp, &vAlpha, n+1, TRUE); - if(!ok) - return( ok ); - - /* Check if we have strategy fallback for the primal */ - if(!isDEVEX) - isDEVEX = is_piv_mode(lp, PRICE_PRIMALFALLBACK); - - /* Initialize column target array */ - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->sum+1, sizeof(*coltarget)); - ok = get_colIndexA(lp, SCAN_SLACKVARS+SCAN_USERVARS+USE_NONBASICVARS, coltarget, FALSE); - if(!ok) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return( ok ); - } - - /* Don't need to compute cross-products with DEVEX */ - if(!isDEVEX) { - ok = allocREAL(lp, &vEdge, n+1, TRUE); - if(!ok) - return( ok ); - - /* Compute v and then N'v */ - MEMCOPY(vTemp, w, m+1); - bsolve(lp, -1, vTemp, NULL, lp->epsmachine*DOUBLEROUND, 0.0); - vTemp[0] = 0; - prod_xA(lp, coltarget, vTemp, NULL, lp->epsmachine, 0.0, - vEdge, NULL, MAT_ROUNDDEFAULT); - } - - /* Compute Sigma and then Alpha */ - bsolve(lp, rownr, vTemp, NULL, 0*DOUBLEROUND, 0.0); - vTemp[0] = 0; - prod_xA(lp, coltarget, vTemp, NULL, lp->epsmachine, 0.0, - vAlpha, NULL, MAT_ROUNDDEFAULT); - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - - /* Update the squared steepest edge norms; first store some constants */ - cEdge = lp->edgeVector[colnr]; - cAlpha = vAlpha[colnr]; - if(fabs(cAlpha) < lp->epspivot) { - forceRefresh = TRUE; - goto Finish1; - } - - /* Deal with the variable leaving the basis to become a new entry candidate */ - hold = 1 / cAlpha; - lp->edgeVector[exitcol] = (hold*hold) * cEdge; - -#ifdef Paranoia - if(lp->edgeVector[exitcol] <= lp->epsmachine) - report(lp, errlevel, "updatePricer: Invalid primal norm %g at leaving index %d - iteration %.0f\n", - lp->edgeVector[exitcol], exitcol, (double) (lp->total_iter+lp->current_iter)); -#endif - - /* Then loop over all non-basic variables, but skip the entering column */ - for(i = 1; i <= lp->sum; i++) { - if(lp->is_basic[i] || (i == colnr)) - continue; - hold = vAlpha[i]; - if(hold == 0) - continue; - hold /= cAlpha; - if(fabs(hold) < lp->epsmachine) - continue; - - newEdge = &(lp->edgeVector[i]); - *newEdge += (hold*hold) * cEdge; - if(isDEVEX) { - if((*newEdge) > DEVEX_RESTARTLIMIT) { - forceRefresh = TRUE; - break; - } - } - else { - *newEdge -= 2*hold*vEdge[i]; -#ifdef ApplySteepestEdgeMinimum - SETMAX(*newEdge, hold*hold+1); -#else - if(*newEdge < 0) { - report(lp, errlevel, "updatePricer: Invalid primal norm %g at index %d - iteration %.0f\n", - *newEdge, i, (double) (lp->total_iter+lp->current_iter)); - if(lp->spx_trace) - report(lp, errlevel, "Error detail: (RelAlpha=%g, vEdge=%g, cEdge=%g)\n", hold, vEdge[i], cEdge); - forceRefresh = TRUE; - break; - } -#endif - } - } - -Finish1: - FREE(vAlpha); - FREE(vTemp); - - } - -Finish2: - FREE(vEdge); - freeWeights(w); - - if(forceRefresh) - ok = restartPricer(lp, AUTOMATIC); - else - ok = TRUE; - - return( ok ); - -} - - -STATIC MYBOOL verifyPricer(lprec *lp) -{ - REAL value; - int i, n; - MYBOOL ok = applyPricer(lp); - - if(!ok) - return( ok ); - ok = FALSE; - - /* Verify */ - if(lp->edgeVector == NULL) - return( ok ); - value = *lp->edgeVector; - if(value < 0) - return( ok ); - - /* Check the primal */ - n = 1; - if(value == 0) { - - for(n = lp->sum; n > 0; n--) { - if(lp->is_basic[n]) - continue; - value = lp->edgeVector[n]; - if(value <= 0) - break; - } - } - /* Check the dual */ - else { - for(i = lp->rows; i > 0; i--) { - n = lp->var_basic[i]; - value = lp->edgeVector[n]; - if(value <= 0) - break; - } - } - - ok = (MYBOOL) (n == 0); -#ifdef Paranoia - if(!ok) - report(lp, SEVERE, "verifyPricer: Invalid norm %g at index %d\n", - value, n); -#endif - return( ok ); -} - diff --git a/code/3rd_lpsolve/lp_pricePSE.h b/code/3rd_lpsolve/lp_pricePSE.h deleted file mode 100644 index 1c8c2f8f..00000000 --- a/code/3rd_lpsolve/lp_pricePSE.h +++ /dev/null @@ -1,28 +0,0 @@ -#ifndef HEADER_lp_pricePSE -#define HEADER_lp_pricePSE - -#include "lp_types.h" - -#define ApplySteepestEdgeMinimum - -#ifdef __cplusplus -extern "C" { -#endif - -/* Price norm management routines */ -STATIC MYBOOL initPricer(lprec *lp); -INLINE MYBOOL applyPricer(lprec *lp); -STATIC void simplexPricer(lprec *lp, MYBOOL isdual); -STATIC void freePricer(lprec *lp); -STATIC MYBOOL resizePricer(lprec *lp); -STATIC REAL getPricer(lprec *lp, int item, MYBOOL isdual); -STATIC MYBOOL restartPricer(lprec *lp, MYBOOL isdual); -STATIC MYBOOL updatePricer(lprec *lp, int rownr, int colnr, REAL *pcol, REAL *prow, int *nzprow); -STATIC MYBOOL verifyPricer(lprec *lp); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_pricePSE */ - diff --git a/code/3rd_lpsolve/lp_report.c b/code/3rd_lpsolve/lp_report.c deleted file mode 100644 index a46ee8ef..00000000 --- a/code/3rd_lpsolve/lp_report.c +++ /dev/null @@ -1,790 +0,0 @@ - -/* - Mixed integer programming optimization drivers for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Michel Berkelaar (to lp_solve v3.2), - Kjell Eikland - Contact: - License terms: LGPL. - - Requires: stdarg.h, lp_lib.h - - Release notes: - v5.0.0 3 1 January 2004 New unit isolating reporting routines. - v5.2.0.0 1 December 2005 Addition of Matrix Market writing function. - - ---------------------------------------------------------------------------------- -*/ - -#include -#include -#include - -#include "lp_lib.h" -#include "lp_scale.h" -#include "commonlib.h" -#include "lp_report.h" - -#include "mmio.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* Define buffer-size controled function mapping */ -# if defined _MSC_VER -# define vsnprintf _vsnprintf -# endif - -/* Various reporting functions for lp_solve */ -/* ------------------------------------------------------------------------- */ - -/* First define general utilties for reporting and output */ -char * __VACALL explain(lprec *lp, char *format, ...) -{ - char buff[DEF_STRBUFSIZE+1]; - va_list ap; - - va_start(ap, format); - vsnprintf(buff, DEF_STRBUFSIZE, format, ap); - va_end(ap); - allocCHAR(lp, &(lp->ex_status), (int) strlen(buff), AUTOMATIC); - strcpy(lp->ex_status, buff); - return( lp->ex_status ); -} -void __VACALL report(lprec *lp, int level, char *format, ...) -{ - char buff[DEF_STRBUFSIZE+1]; - va_list ap; - - if(lp == NULL) { - va_start(ap, format); - vfprintf(stderr, format, ap); - va_end(ap); - } - else if(level <= lp->verbose) { - if(lp->writelog != NULL) { - va_start(ap, format); - vsnprintf(buff, DEF_STRBUFSIZE, format, ap); - va_end(ap); - lp->writelog(lp, lp->loghandle, buff); - } - if(lp->outstream != NULL) { - va_start(ap, format); - vfprintf(lp->outstream, format, ap); - va_end(ap); - if(lp->outstream != stdout) - fflush(lp->outstream); - } - } -#ifdef xParanoia - if(level == CRITICAL) - raise(SIGSEGV); -#endif -} - -STATIC void print_indent(lprec *lp) -{ - int i; - - report(lp, NEUTRAL, "%2d", lp->bb_level); - if(lp->bb_level < 50) /* useless otherwise */ - for(i = lp->bb_level; i > 0; i--) - report(lp, NEUTRAL, "--"); - else - report(lp, NEUTRAL, " *** too deep ***"); - report(lp, NEUTRAL, "> "); -} /* print_indent */ - -STATIC void debug_print(lprec *lp, char *format, ...) -{ - va_list ap; - - if(lp->bb_trace) { - print_indent(lp); - if (lp == NULL) - { - va_start(ap, format); - vfprintf(stderr, format, ap); - va_end(ap); - fputc('\n', stderr); - } - else if(lp->debuginfo != NULL) - { - char buff[DEF_STRBUFSIZE+1]; - va_start(ap, format); - vsnprintf(buff, DEF_STRBUFSIZE, format, ap); - va_end(ap); - lp->debuginfo(lp, lp->loghandle, buff); - } - } -} /* debug_print */ - -STATIC void debug_print_solution(lprec *lp) -{ - int i; - - if(lp->bb_trace) - for (i = lp->rows + 1; i <= lp->sum; i++) { - print_indent(lp); - report(lp, NEUTRAL, "%s " RESULTVALUEMASK "\n", - get_col_name(lp, i - lp->rows), - (double)lp->solution[i]); - } -} /* debug_print_solution */ - -STATIC void debug_print_bounds(lprec *lp, REAL *upbo, REAL *lowbo) -{ - int i; - - if(lp->bb_trace) - for(i = lp->rows + 1; i <= lp->sum; i++) { - if(lowbo[i] == upbo[i]) { - print_indent(lp); - report(lp, NEUTRAL, "%s = " RESULTVALUEMASK "\n", get_col_name(lp, i - lp->rows), - (double)lowbo[i]); - } - else { - if(lowbo[i] != 0) { - print_indent(lp); - report(lp, NEUTRAL, "%s > " RESULTVALUEMASK "\n", get_col_name(lp, i - lp->rows), - (double)lowbo[i]); - } - if(upbo[i] != lp->infinite) { - print_indent(lp); - report(lp, NEUTRAL, "%s < " RESULTVALUEMASK "\n", get_col_name(lp, i - lp->rows), - (double)upbo[i]); - } - } - } -} /* debug_print_bounds */ - -/* List a vector of LREAL values for the given index range */ -void blockWriteLREAL(FILE *output, char *label, LREAL *vector, int first, int last) -{ - int i, k = 0; - - fprintf(output, "%s", label); - fprintf(output, "\n"); - for(i = first; i <= last; i++) { - fprintf(output, " %18g", vector[i]); - k++; - if(my_mod(k, 4) == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(my_mod(k, 4) != 0) - fprintf(output, "\n"); -} - -/* List the current user data matrix columns over the selected row range */ -void blockWriteAMAT(FILE *output, const char *label, lprec* lp, int first, int last) -{ - int i, j, k = 0; - int nzb, nze, jb; - double hold; - MATrec *mat = lp->matA; - - if(!mat_validate(mat)) - return; - if(first < 0) - first = 0; - if(last < 0) - last = lp->rows; - - fprintf(output, "%s", label); - fprintf(output, "\n"); - - if(first == 0) { - for(j = 1; j <= lp->columns; j++) { - hold = get_mat(lp, 0, j); - fprintf(output, " %18g", hold); - k++; - if(my_mod(k, 4) == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(my_mod(k, 4) != 0) { - fprintf(output, "\n"); - k = 0; - } - first++; - } - nze = mat->row_end[first-1]; - for(i = first; i <= last; i++) { - nzb = nze; - nze = mat->row_end[i]; - if(nzb >= nze) - jb = lp->columns+1; - else - jb = ROW_MAT_COLNR(nzb); - for(j = 1; j <= lp->columns; j++) { - if(j < jb) - hold = 0; - else { - hold = get_mat(lp, i, j); - nzb++; - if(nzb < nze) - jb = ROW_MAT_COLNR(nzb); - else - jb = lp->columns+1; - } - fprintf(output, " %18g", hold); - k++; - if(my_mod(k, 4) == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(my_mod(k, 4) != 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(my_mod(k, 4) != 0) - fprintf(output, "\n"); -} - -/* List the current basis matrix columns over the selected row range */ -void blockWriteBMAT(FILE *output, const char *label, lprec* lp, int first, int last) -{ - int i, j, jb, k = 0; - double hold; - - if(first < 0) - first = 0; - if(last < 0) - last = lp->rows; - - fprintf(output, "%s", label); - fprintf(output, "\n"); - - for(i = first; i <= last; i++) { - for(j = 1; j <= lp->rows; j++) { - jb = lp->var_basic[j]; - if(jb <= lp->rows) { - if(jb == i) - hold = 1; - else - hold = 0; - } - else - hold = get_mat(lp, i, j); - if(i == 0) - modifyOF1(lp, jb, &hold, 1); - hold = unscaled_mat(lp, hold, i, jb); - fprintf(output, " %18g", hold); - k++; - if(my_mod(k, 4) == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(my_mod(k, 4) != 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(my_mod(k, 4) != 0) - fprintf(output, "\n"); -} - -/* Do a generic readable data dump of key lp_solve model variables; - principally for run difference and debugging purposes */ -MYBOOL REPORT_debugdump(lprec *lp, char *filename, MYBOOL livedata) -{ - FILE *output = stdout; - MYBOOL ok; - - ok = (MYBOOL) ((filename == NULL) || ((output = fopen(filename,"w")) != NULL)); - if(!ok) - return(ok); - if((filename == NULL) && (lp->outstream != NULL)) - output = lp->outstream; - - fprintf(output, "\nGENERAL INFORMATION\n-------------------\n\n"); - fprintf(output, "Model size: %d rows (%d equalities, %d Lagrangean), %d columns (%d integers, %d SC, %d SOS, %d GUB)\n", - lp->rows, lp->equalities, get_Lrows(lp), lp->columns, - lp->int_vars, lp->sc_vars, SOS_count(lp), GUB_count(lp)); - fprintf(output, "Data size: %d model non-zeros, %d invB non-zeros (engine is %s)\n", - get_nonzeros(lp), my_if(lp->invB == NULL, 0, lp->bfp_nonzeros(lp, FALSE)), lp->bfp_name()); - fprintf(output, "Internal sizes: %d rows allocated, %d columns allocated, %d columns used, %d eta length\n", - lp->rows_alloc, lp->columns_alloc, lp->columns, my_if(lp->invB == NULL, 0, lp->bfp_colcount(lp))); - fprintf(output, "Memory use: %d sparse matrix, %d eta\n", - lp->matA->mat_alloc, my_if(lp->invB == NULL, 0, lp->bfp_memallocated(lp))); - fprintf(output, "Parameters: Maximize=%d, Names used=%d, Scalingmode=%d, Presolve=%d, SimplexPivot=%d\n", - is_maxim(lp), lp->names_used, lp->scalemode, lp->do_presolve, lp->piv_strategy); - fprintf(output, "Precision: EpsValue=%g, EpsPrimal=%g, EpsDual=%g, EpsPivot=%g, EpsPerturb=%g\n", - lp->epsvalue, lp->epsprimal, lp->epsdual, lp->epspivot, lp->epsperturb); - fprintf(output, "Stability: AntiDegen=%d, Improvement=%d, Split variables at=%g\n", - lp->improve, lp->anti_degen, lp->negrange); - fprintf(output, "B&B settings: BB pivot rule=%d, BB branching=%s, BB strategy=%d, Integer precision=%g, MIP gaps=%g,%g\n", - lp->bb_rule, my_boolstr(lp->bb_varbranch), lp->bb_floorfirst, lp->epsint, lp->mip_absgap, lp->mip_relgap); - - fprintf(output, "\nCORE DATA\n---------\n\n"); - blockWriteINT(output, "Column starts", lp->matA->col_end, 0, lp->columns); - blockWriteINT(output, "row_type", lp->row_type, 0, lp->rows); - blockWriteREAL(output, "orig_rhs", lp->orig_rhs, 0, lp->rows); - blockWriteREAL(output, "orig_lowbo", lp->orig_lowbo, 0, lp->sum); - blockWriteREAL(output, "orig_upbo", lp->orig_upbo, 0, lp->sum); - blockWriteINT(output, "row_type", lp->row_type, 0, lp->rows); - blockWriteBOOL(output, "var_type", lp->var_type, 0, lp->columns, TRUE); - blockWriteAMAT(output, "A", lp, 0, lp->rows); - - if(livedata) { - fprintf(output, "\nPROCESS DATA\n------------\n\n"); - blockWriteREAL(output, "Active rhs", lp->rhs, 0, lp->rows); - blockWriteINT(output, "Basic variables", lp->var_basic, 0, lp->rows); - blockWriteBOOL(output, "is_basic", lp->is_basic, 0, lp->sum, TRUE); - blockWriteREAL(output, "lowbo", lp->lowbo, 0, lp->sum); - blockWriteREAL(output, "upbo", lp->upbo, 0, lp->sum); - if(lp->scalars != NULL) - blockWriteREAL(output, "scalars", lp->scalars, 0, lp->sum); - } - - if(filename != NULL) - fclose(output); - return(ok); -} - - -/* High level reports for model results */ - -void REPORT_objective(lprec *lp) -{ - if(lp->outstream == NULL) - return; - if(fabs(lp->best_solution[0]) < 1e-5) - fprintf(lp->outstream, "\nValue of objective function: %g\n", (double)lp->best_solution[0]); - else - fprintf(lp->outstream, "\nValue of objective function: %.8f\n", (double)lp->best_solution[0]); - fflush(lp->outstream); -} - -void REPORT_solution(lprec *lp, int columns) -{ - int i, j, n; - REAL value; - presolveundorec *psundo = lp->presolve_undo; - MYBOOL NZonly = (MYBOOL) ((lp->print_sol & AUTOMATIC) > 0); - - if(lp->outstream == NULL) - return; - - fprintf(lp->outstream, "\nActual values of the variables:\n"); - if(columns <= 0) - columns = 2; - n = 0; - for(i = 1; i <= psundo->orig_columns; i++) { - j = psundo->orig_rows + i; - value = get_var_primalresult(lp, j); - if(NZonly && (fabs(value) < lp->epsprimal)) - continue; - n = (n+1) % columns; - fprintf(lp->outstream, "%-20s %12g", get_origcol_name(lp, i), (double) value); - if(n == 0) - fprintf(lp->outstream, "\n"); - else - fprintf(lp->outstream, " "); - } - - fflush(lp->outstream); -} /* REPORT_solution */ - -void REPORT_constraints(lprec *lp, int columns) -{ - int i, n; - REAL value; - MYBOOL NZonly = (MYBOOL) ((lp->print_sol & AUTOMATIC) > 0); - - if(lp->outstream == NULL) - return; - - if(columns <= 0) - columns = 2; - - fprintf(lp->outstream, "\nActual values of the constraints:\n"); - n = 0; - for(i = 1; i <= lp->rows; i++) { - value = (double)lp->best_solution[i]; - if(NZonly && (fabs(value) < lp->epsprimal)) - continue; - n = (n+1) % columns; - fprintf(lp->outstream, "%-20s %12g", get_row_name(lp, i), value); - if(n == 0) - fprintf(lp->outstream, "\n"); - else - fprintf(lp->outstream, " "); - } - - fflush(lp->outstream); -} - -void REPORT_duals(lprec *lp) -{ - int i; - REAL *duals, *dualsfrom, *dualstill, *objfrom, *objtill, *objfromvalue; - MYBOOL ret; - - if(lp->outstream == NULL) - return; - - ret = get_ptr_sensitivity_objex(lp, &objfrom, &objtill, &objfromvalue, NULL); - if(ret) { - fprintf(lp->outstream, "\nObjective function limits:\n"); - fprintf(lp->outstream, " From Till FromValue\n"); - for(i = 1; i <= lp->columns; i++) - if(!is_splitvar(lp, i)) - fprintf(lp->outstream, "%-20s %15.7g %15.7g %15.7g\n", get_col_name(lp, i), - (double)objfrom[i - 1], (double)objtill[i - 1], (double)objfromvalue[i - 1]); - } - - ret = get_ptr_sensitivity_rhs(lp, &duals, &dualsfrom, &dualstill); - if(ret) { - fprintf(lp->outstream, "\nDual values with from - till limits:\n"); - fprintf(lp->outstream, " Dual value From Till\n"); - for(i = 1; i <= lp->sum; i++) - fprintf(lp->outstream, "%-20s %15.7g %15.7g %15.7g\n", - (i <= lp->rows) ? get_row_name(lp, i) : get_col_name(lp, i - lp->rows), - (double)duals[i - 1], (double)dualsfrom[i - 1], (double)dualstill[i - 1]); - fflush(lp->outstream); - } -} - -/* Printing of sensitivity analysis reports */ -void REPORT_extended(lprec *lp) -{ - int i, j; - REAL hold; - REAL *duals, *dualsfrom, *dualstill, *objfrom, *objtill; - MYBOOL ret; - - ret = get_ptr_sensitivity_obj(lp, &objfrom, &objtill); - report(lp, NORMAL, " \n"); - report(lp, NORMAL, "Primal objective:\n"); - report(lp, NORMAL, " \n"); - report(lp, NORMAL, " Column name Value Objective Min Max\n"); - report(lp, NORMAL, " --------------------------------------------------------------------------\n"); - for(j = 1; j <= lp->columns; j++) { - hold = get_mat(lp,0,j); - report(lp, NORMAL, " %-25s " MPSVALUEMASK MPSVALUEMASK MPSVALUEMASK MPSVALUEMASK "\n", - get_col_name(lp,j), - my_precision(hold,lp->epsprimal), - my_precision(hold*lp->best_solution[lp->rows+j],lp->epsprimal), - my_precision((ret) ? objfrom[j - 1] : 0.0,lp->epsprimal), - my_precision((ret) ? objtill[j - 1] : 0.0,lp->epsprimal)); - } - report(lp, NORMAL, " \n"); - - ret = get_ptr_sensitivity_rhs(lp, &duals, &dualsfrom, &dualstill); - report(lp, NORMAL, "Primal variables:\n"); - report(lp, NORMAL, " \n"); - report(lp, NORMAL, " Column name Value Slack Min Max\n"); - report(lp, NORMAL, " --------------------------------------------------------------------------\n"); - for(j = 1; j <= lp->columns; j++) - report(lp, NORMAL, " %-25s " MPSVALUEMASK MPSVALUEMASK MPSVALUEMASK MPSVALUEMASK "\n", - get_col_name(lp,j), - my_precision(lp->best_solution[lp->rows+j],lp->epsprimal), - my_precision(my_inflimit(lp, (ret) ? duals[lp->rows+j-1] : 0.0),lp->epsprimal), - my_precision((ret) ? dualsfrom[lp->rows+j-1] : 0.0,lp->epsprimal), - my_precision((ret) ? dualstill[lp->rows+j-1] : 0.0,lp->epsprimal)); - - report(lp, NORMAL, " \n"); - report(lp, NORMAL, "Dual variables:\n"); - report(lp, NORMAL, " \n"); - report(lp, NORMAL, " Row name Value Slack Min Max\n"); - report(lp, NORMAL, " --------------------------------------------------------------------------\n"); - for(i = 1; i <= lp->rows; i++) - report(lp, NORMAL, " %-25s " MPSVALUEMASK MPSVALUEMASK MPSVALUEMASK MPSVALUEMASK "\n", - get_row_name(lp,i), - my_precision((ret) ? duals[i - 1] : 0.0, lp->epsprimal), - my_precision(lp->best_solution[i], lp->epsprimal), - my_precision((ret) ? dualsfrom[i - 1] : 0.0,lp->epsprimal), - my_precision((ret) ? dualstill[i - 1] : 0.0,lp->epsprimal)); - - report(lp, NORMAL, " \n"); -} - -/* A more readable lp-format report of the model; antiquated and not updated */ -void REPORT_lp(lprec *lp) -{ - int i, j; - - if(lp->outstream == NULL) - return; - - fprintf(lp->outstream, "Model name: %s\n", get_lp_name(lp)); - fprintf(lp->outstream, " "); - - for(j = 1; j <= lp->columns; j++) - fprintf(lp->outstream, "%8s ", get_col_name(lp,j)); - - fprintf(lp->outstream, "\n%simize ", (is_maxim(lp) ? "Max" : "Min")); - for(j = 1; j <= lp->columns; j++) - fprintf(lp->outstream, "%8g ", get_mat(lp, 0, j)); - fprintf(lp->outstream, "\n"); - - for(i = 1; i <= lp->rows; i++) { - fprintf(lp->outstream, "%-9s ", get_row_name(lp, i)); - for(j = 1; j <= lp->columns; j++) - fprintf(lp->outstream, "%8g ", get_mat(lp, i, j)); - if(is_constr_type(lp, i, GE)) - fprintf(lp->outstream, ">= "); - else if(is_constr_type(lp, i, LE)) - fprintf(lp->outstream, "<= "); - else - fprintf(lp->outstream, " = "); - fprintf(lp->outstream, "%8g", get_rh(lp, i)); - - if(is_constr_type(lp, i, GE)) { - if(get_rh_upper(lp, i) < lp->infinite) - fprintf(lp->outstream, " %s = %8g", "upbo", get_rh_upper(lp, i)); - } - else if(is_constr_type(lp, i, LE)) { - if(get_rh_lower(lp, i) > -lp->infinite) - fprintf(lp->outstream, " %s = %8g", "lowbo", get_rh_lower(lp, i)); - } - fprintf(lp->outstream, "\n"); - } - - fprintf(lp->outstream, "Type "); - for(i = 1; i <= lp->columns; i++) { - if(is_int(lp,i)) - fprintf(lp->outstream, " Int "); - else - fprintf(lp->outstream, " Real "); - } - - fprintf(lp->outstream, "\nupbo "); - for(i = 1; i <= lp->columns; i++) - if(get_upbo(lp, i) >= lp->infinite) - fprintf(lp->outstream, " Inf "); - else - fprintf(lp->outstream, "%8g ", get_upbo(lp, i)); - fprintf(lp->outstream, "\nlowbo "); - for(i = 1; i <= lp->columns; i++) - if(get_lowbo(lp, i) <= -lp->infinite) - fprintf(lp->outstream, " -Inf "); - else - fprintf(lp->outstream, "%8g ", get_lowbo(lp, i)); - fprintf(lp->outstream, "\n"); - - fflush(lp->outstream); -} - -/* Report the scaling factors used; extremely rarely used */ -void REPORT_scales(lprec *lp) -{ - int i, colMax; - - colMax = lp->columns; - - if(lp->outstream == NULL) - return; - - if(lp->scaling_used) { - fprintf(lp->outstream, "\nScale factors:\n"); - for(i = 0; i <= lp->rows + colMax; i++) - fprintf(lp->outstream, "%-20s scaled at %g\n", - (i <= lp->rows) ? get_row_name(lp, i) : get_col_name(lp, i - lp->rows), - (double)lp->scalars[i]); - } - fflush(lp->outstream); -} - -/* Report the traditional tableau corresponding to the current basis */ -MYBOOL REPORT_tableau(lprec *lp) -{ - int j, row_nr, *coltarget; - REAL *prow = NULL; - FILE *stream = lp->outstream; - - if(lp->outstream == NULL) - return(FALSE); - - if(!lp->model_is_valid || !has_BFP(lp) || - (get_total_iter(lp) == 0) || (lp->spx_status == NOTRUN)) { - lp->spx_status = NOTRUN; - return(FALSE); - } - if(!allocREAL(lp, &prow,lp->sum + 1, TRUE)) { - lp->spx_status = NOMEMORY; - return(FALSE); - } - - fprintf(stream, "\n"); - fprintf(stream, "Tableau at iter %.0f:\n", (double) get_total_iter(lp)); - - for(j = 1; j <= lp->sum; j++) - if (!lp->is_basic[j]) - fprintf(stream, "%15d", (j <= lp->rows ? - (j + lp->columns) * ((lp->orig_upbo[j] == 0) || - (is_chsign(lp, j)) ? 1 : -1) : j - lp->rows) * - (lp->is_lower[j] ? 1 : -1)); - fprintf(stream, "\n"); - - coltarget = (int *) mempool_obtainVector(lp->workarrays, lp->columns+1, sizeof(*coltarget)); - if(!get_colIndexA(lp, SCAN_USERVARS+USE_NONBASICVARS, coltarget, FALSE)) { - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - return(FALSE); - } - for(row_nr = 1; (row_nr <= lp->rows + 1); row_nr++) { - if (row_nr <= lp->rows) - fprintf(stream, "%3d", (lp->var_basic[row_nr] <= lp->rows ? - (lp->var_basic[row_nr] + lp->columns) * ((lp->orig_upbo[lp->var_basic [row_nr]] == 0) || - (is_chsign(lp, lp->var_basic[row_nr])) ? 1 : -1) : lp->var_basic[row_nr] - lp->rows) * - (lp->is_lower[lp->var_basic [row_nr]] ? 1 : -1)); - else - fprintf(stream, " "); - bsolve(lp, row_nr <= lp->rows ? row_nr : 0, prow, NULL, lp->epsmachine*DOUBLEROUND, 1.0); - prod_xA(lp, coltarget, prow, NULL, lp->epsmachine, 1.0, - prow, NULL, MAT_ROUNDDEFAULT); - - for(j = 1; j <= lp->rows + lp->columns; j++) - if (!lp->is_basic[j]) - fprintf(stream, "%15.7f", prow[j] * (lp->is_lower[j] ? 1 : -1) * - (row_nr <= lp->rows ? 1 : -1)); - fprintf(stream, "%15.7f", lp->rhs[row_nr <= lp->rows ? row_nr : 0] * - (double) ((row_nr <= lp->rows) || (is_maxim(lp)) ? 1 : -1)); - fprintf(stream, "\n"); - } - fflush(stream); - - mempool_releaseVector(lp->workarrays, (char *) coltarget, FALSE); - FREE(prow); - return(TRUE); -} - -void REPORT_constraintinfo(lprec *lp, char *datainfo) -{ - int i, tally[ROWCLASS_MAX+1]; - - MEMCLEAR(tally, ROWCLASS_MAX+1); - for(i = 1; i <= lp->rows; i++) - tally[get_constr_class(lp, i)]++; - - if(datainfo != NULL) - report(lp, NORMAL, "%s\n", datainfo); - - for(i = 0; i <= ROWCLASS_MAX; i++) - if(tally[i] > 0) - report(lp, NORMAL, "%-15s %4d\n", get_str_constr_class(lp, i), tally[i]); -} - -void REPORT_modelinfo(lprec *lp, MYBOOL doName, char *datainfo) -{ - if(doName) { - report(lp, NORMAL, "\nModel name: '%s' - run #%-5d\n", - get_lp_name(lp), lp->solvecount); - report(lp, NORMAL, "Objective: %simize(%s)\n", - my_if(is_maxim(lp), "Max", "Min"), get_row_name(lp, 0)); - report(lp, NORMAL, " \n"); - } - if(datainfo != NULL) - report(lp, NORMAL, "%s\n", datainfo); - - report(lp, NORMAL, "Model size: %7d constraints, %7d variables, %12d non-zeros.\n", - lp->rows, lp->columns, get_nonzeros(lp)); - if(GUB_count(lp)+SOS_count(lp) > 0) - report(lp, NORMAL, "Var-types: %7d integer, %7d semi-cont., %7d SOS.\n", - lp->int_vars, lp->sc_vars, lp->sos_vars); - report(lp, NORMAL, "Sets: %7d GUB, %7d SOS.\n", - GUB_count(lp), SOS_count(lp)); -} - -/* Save a matrix column subset to a MatrixMarket formatted file, - say to export the basis matrix for further numerical analysis. - If colndx is NULL, then the full constraint matrix is assumed. */ -MYBOOL REPORT_mat_mmsave(lprec *lp, char *filename, int *colndx, MYBOOL includeOF, char *infotext) -{ - int n, m, nz, i, j, k, kk; - MATrec *mat = lp->matA; - MM_typecode matcode; - FILE *output = stdout; - MYBOOL ok; - REAL *acol = NULL; - int *nzlist = NULL; - - /* Open file */ - ok = (MYBOOL) ((filename == NULL) || ((output = fopen(filename,"w")) != NULL)); - if(!ok) - return(ok); - if((filename == NULL) && (lp->outstream != NULL)) - output = lp->outstream; - - /* Compute column and non-zero counts */ - if(colndx == lp->var_basic) { - if(!lp->basis_valid) - return( FALSE ); - m = lp->rows; - } - else if(colndx != NULL) - m = colndx[0]; - else - m = lp->columns; - n = lp->rows; - nz = 0; - - for(j = 1; j <= m; j++) { - k = (colndx == NULL ? n + j : colndx[j]); - if(k > n) { - k -= lp->rows; - nz += mat_collength(mat, k); - if(includeOF && is_OF_nz(lp, k)) - nz++; - } - else - nz++; - } - kk = 0; - if(includeOF) { - n++; /* Row count */ - kk++; /* Row index offset */ - } - - /* Initialize */ - mm_initialize_typecode(&matcode); - mm_set_matrix(&matcode); - mm_set_coordinate(&matcode); - mm_set_real(&matcode); - - mm_write_banner(output, matcode); - mm_write_mtx_crd_size(output, n+kk, m, nz+(colndx == lp->var_basic ? 1 : 0)); - - /* Allocate working arrays for sparse column storage */ - allocREAL(lp, &acol, n+2, FALSE); - allocINT(lp, &nzlist, n+2, FALSE); - - /* Write the matrix non-zero values column-by-column. - NOTE: matrixMarket files use 1-based indeces, - i.e. first row of a vector has index 1, not 0. */ - if(infotext != NULL) { - fprintf(output, "%%\n"); - fprintf(output, "%% %s\n", infotext); - fprintf(output, "%%\n"); - } - if(includeOF && (colndx == lp->var_basic)) - fprintf(output, "%d %d %g\n", 1, 1, 1.0); - for(j = 1; j <= m; j++) { - k = (colndx == NULL ? lp->rows + j : colndx[j]); - if(k == 0) - continue; - nz = obtain_column(lp, k, acol, nzlist, NULL); - for(i = 1; i <= nz; i++) { - if(!includeOF && (nzlist[i] == 0)) - continue; - fprintf(output, "%d %d %g\n", nzlist[i]+kk, j+kk, acol[i]); - } - } - fprintf(output, "%% End of MatrixMarket file\n"); - - /* Finish */ - FREE(acol); - FREE(nzlist); - fclose(output); - - return(ok); -} - diff --git a/code/3rd_lpsolve/lp_report.h b/code/3rd_lpsolve/lp_report.h deleted file mode 100644 index a8567fed..00000000 --- a/code/3rd_lpsolve/lp_report.h +++ /dev/null @@ -1,42 +0,0 @@ -#ifndef HEADER_lp_report -#define HEADER_lp_report - -#ifdef __cplusplus -extern "C" { -#endif - -/* General information functions */ -char * __VACALL explain(lprec *lp, char *format, ...); -void __VACALL report(lprec *lp, int level, char *format, ...); - -/* Prototypes for debugging and general data dumps */ -void debug_print(lprec *lp, char *format, ...); -void debug_print_solution(lprec *lp); -void debug_print_bounds(lprec *lp, REAL *upbo, REAL *lowbo); -void blockWriteLREAL(FILE *output, char *label, LREAL *vector, int first, int last); -void blockWriteAMAT(FILE *output, const char *label, lprec* lp, int first, int last); -void blockWriteBMAT(FILE *output, const char *label, lprec* lp, int first, int last); - - -/* Model reporting headers */ -void REPORT_objective(lprec *lp); -void REPORT_solution(lprec *lp, int columns); -void REPORT_constraints(lprec *lp, int columns); -void REPORT_duals(lprec *lp); -void REPORT_extended(lprec *lp); - -/* Other rarely used, but sometimes extremely useful reports */ -void REPORT_constraintinfo(lprec *lp, char *datainfo); -void REPORT_modelinfo(lprec *lp, MYBOOL doName, char *datainfo); -void REPORT_lp(lprec *lp); -MYBOOL REPORT_tableau(lprec *lp); -void REPORT_scales(lprec *lp); -MYBOOL REPORT_debugdump(lprec *lp, char *filename, MYBOOL livedata); -MYBOOL REPORT_mat_mmsave(lprec *lp, char *filename, int *colndx, MYBOOL includeOF, char *infotext); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_report */ - diff --git a/code/3rd_lpsolve/lp_rlp.c b/code/3rd_lpsolve/lp_rlp.c deleted file mode 100644 index d3ba441f..00000000 --- a/code/3rd_lpsolve/lp_rlp.c +++ /dev/null @@ -1,2484 +0,0 @@ -/* A Bison parser, made by GNU Bison 2.3. */ - -/* Skeleton implementation for Bison's Yacc-like parsers in C - - Copyright (C) 1984, 1989, 1990, 2000, 2001, 2002, 2003, 2004, 2005, 2006 - Free Software Foundation, Inc. - - This program is free software; you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation; either version 2, or (at your option) - any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program; if not, write to the Free Software - Foundation, Inc., 51 Franklin Street, Fifth Floor, - Boston, MA 02110-1301, USA. */ - -/* As a special exception, you may create a larger work that contains - part or all of the Bison parser skeleton and distribute that work - under terms of your choice, so long as that work isn't itself a - parser generator using the skeleton or a modified version thereof - as a parser skeleton. Alternatively, if you modify or redistribute - the parser skeleton itself, you may (at your option) remove this - special exception, which will cause the skeleton and the resulting - Bison output files to be licensed under the GNU General Public - License without this special exception. - - This special exception was added by the Free Software Foundation in - version 2.2 of Bison. */ - -/* C LALR(1) parser skeleton written by Richard Stallman, by - simplifying the original so-called "semantic" parser. */ - -/* All symbols defined below should begin with lp_yy or YY, to avoid - infringing on user name space. This should be done even for local - variables, as they might otherwise be expanded by user macros. - There are some unavoidable exceptions within include files to - define necessary library symbols; they are noted "INFRINGES ON - USER NAME SPACE" below. */ - -/* Identify Bison output. */ -#define YYBISON 1 - -/* Bison version. */ -#define YYBISON_VERSION "2.3" - -/* Skeleton name. */ -#define YYSKELETON_NAME "yacc.c" - -/* Pure parsers. */ -#define YYPURE 1 - -/* Using locations. */ -#define YYLSP_NEEDED 0 - - - -/* Tokens. */ -#ifndef YYTOKENTYPE -# define YYTOKENTYPE - /* Put the tokens into the symbol table, so that GDB and other debuggers - know about them. */ - enum lp_yytokentype { - VAR = 258, - CONS = 259, - INTCONS = 260, - VARIABLECOLON = 261, - INF = 262, - SEC_INT = 263, - SEC_BIN = 264, - SEC_SEC = 265, - SEC_SOS = 266, - SOSDESCR = 267, - SEC_FREE = 268, - TOK_SIGN = 269, - AR_M_OP = 270, - RE_OPEQ = 271, - RE_OPLE = 272, - RE_OPGE = 273, - END_C = 274, - COMMA = 275, - COLON = 276, - MINIMISE = 277, - MAXIMISE = 278, - UNDEFINED = 279 - }; -#endif -/* Tokens. */ -#define VAR 258 -#define CONS 259 -#define INTCONS 260 -#define VARIABLECOLON 261 -#define INF 262 -#define SEC_INT 263 -#define SEC_BIN 264 -#define SEC_SEC 265 -#define SEC_SOS 266 -#define SOSDESCR 267 -#define SEC_FREE 268 -#define TOK_SIGN 269 -#define AR_M_OP 270 -#define RE_OPEQ 271 -#define RE_OPLE 272 -#define RE_OPGE 273 -#define END_C 274 -#define COMMA 275 -#define COLON 276 -#define MINIMISE 277 -#define MAXIMISE 278 -#define UNDEFINED 279 - - - - -/* Copy the first part of user declarations. */ - - -#include -#include -#include - -#define scanner lp_yyscanner -#define PARM lp_yyget_extra(lp_yyscanner) -#define YYSTYPE int -#define YY_EXTRA_TYPE parse_parm * -#define YY_FATAL_ERROR(msg) lex_fatal_error(PARM, lp_yyscanner, msg) -#undef YY_INPUT -#define YY_INPUT(buf,result,max_size) result = lp_input((void *) PARM, buf, max_size); -#define lp_yyerror read_error - -#include "lpkit.h" -#include "yacc_read.h" - -typedef struct parse_vars_s -{ - read_modeldata_func *lp_input; - void *userhandle; - char HadVar, HadVar0, HadVar1, HadVar2, HasAR_M_OP, HadConstraint, Had_lineair_sum, Had_lineair_sum0, do_add_row, HadSign, OP, Sign, isign, isign0, make_neg; - char state, state0; - char Within_int_decl; /* TRUE when we are within an char declaration */ - char Within_bin_decl; /* TRUE when we are within an bin declaration */ - char Within_sec_decl; /* TRUE when we are within a sec declaration */ - char Within_sos_decl; /* TRUE when we are within a sos declaration */ - char Within_sos_decl1; - char Within_free_decl; /* TRUE when we are within a free declaration */ - short SOStype, SOStype0; /* SOS type */ - int SOSNr; - int SOSweight; /* SOS weight */ - char *Last_var, *Last_var0; - REAL f, f0, f1; -} parse_vars; - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* let's please C++ users */ -#ifdef __cplusplus -extern "C" { -#endif - -#if defined MSDOS || defined __MSDOS__ || defined WINDOWS || defined _WINDOWS || defined WIN32 || defined _WIN32 -#define YY_NO_UNISTD_H - -static int isatty(int f) -{ - return(FALSE); -} - -#if !defined _STDLIB_H -# define _STDLIB_H -#endif -#endif - -static int __WINAPI lp_input_lp_yyin(void *fpin, char *buf, int max_size) -{ - int result; - - result = fread( (char*)buf, sizeof(char), max_size, (FILE *)fpin); - - return(result); -} - -static int __WINAPI lp_input(void *vpp, char *buf, int max_size) -{ - parse_parm *pp = (parse_parm *) vpp; - parse_vars *pv = (parse_vars *) pp->parse_vars; - int result; - - result = pv->lp_input(pv->userhandle, buf, max_size); - if (result < 0) - lex_fatal_error(pp, pp->scanner, "read() in flex scanner failed"); - return(result); -} - -#ifdef __cplusplus -}; -#endif - -#include "lp_rlp.h" - -#undef lp_yylval - - - -/* Enabling traces. */ -#ifndef YYDEBUG -# define YYDEBUG 0 -#endif - -/* Enabling verbose error messages. */ -#ifdef YYERROR_VERBOSE -# undef YYERROR_VERBOSE -# define YYERROR_VERBOSE 1 -#else -# define YYERROR_VERBOSE 0 -#endif - -/* Enabling the token table. */ -#ifndef YYTOKEN_TABLE -# define YYTOKEN_TABLE 0 -#endif - -#if ! defined YYSTYPE && ! defined YYSTYPE_IS_DECLARED -typedef int YYSTYPE; -# define lp_yystype YYSTYPE /* obsolescent; will be withdrawn */ -# define YYSTYPE_IS_DECLARED 1 -# define YYSTYPE_IS_TRIVIAL 1 -#endif - - - -/* Copy the second part of user declarations. */ - - -/* Line 216 of yacc.c. */ - - -#ifdef short -# undef short -#endif - -#ifdef YYTYPE_UINT8 -typedef YYTYPE_UINT8 lp_yytype_uint8; -#else -typedef unsigned char lp_yytype_uint8; -#endif - -#ifdef YYTYPE_INT8 -typedef YYTYPE_INT8 lp_yytype_int8; -#elif (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -typedef signed char lp_yytype_int8; -#else -typedef short int lp_yytype_int8; -#endif - -#ifdef YYTYPE_UINT16 -typedef YYTYPE_UINT16 lp_yytype_uint16; -#else -typedef unsigned short int lp_yytype_uint16; -#endif - -#ifdef YYTYPE_INT16 -typedef YYTYPE_INT16 lp_yytype_int16; -#else -typedef short int lp_yytype_int16; -#endif - -#ifndef YYSIZE_T -# ifdef __SIZE_TYPE__ -# define YYSIZE_T __SIZE_TYPE__ -# elif defined size_t -# define YYSIZE_T size_t -# elif ! defined YYSIZE_T && (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -# include /* INFRINGES ON USER NAME SPACE */ -# define YYSIZE_T size_t -# else -# define YYSIZE_T unsigned int -# endif -#endif - -#define YYSIZE_MAXIMUM ((YYSIZE_T) -1) - -#ifndef YY_ -# if YYENABLE_NLS -# if ENABLE_NLS -# include /* INFRINGES ON USER NAME SPACE */ -# define YY_(msgid) dgettext ("bison-runtime", msgid) -# endif -# endif -# ifndef YY_ -# define YY_(msgid) msgid -# endif -#endif - -/* Suppress unused-variable warnings by "using" E. */ -#if ! defined lint || defined __GNUC__ -# define YYUSE(e) ((void) (e)) -#else -# define YYUSE(e) /* empty */ -#endif - -/* Identity function, used to suppress warnings about constant conditions. */ -#ifndef lint -# define YYID(n) (n) -#else -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static int -YYID (int i) -#else -static int -YYID (i) - int i; -#endif -{ - return i; -} -#endif - -#if ! defined lp_yyoverflow || YYERROR_VERBOSE - -/* The parser invokes alloca or malloc; define the necessary symbols. */ - -# ifdef YYSTACK_USE_ALLOCA -# if YYSTACK_USE_ALLOCA -# ifdef __GNUC__ -# define YYSTACK_ALLOC __builtin_alloca -# elif defined __BUILTIN_VA_ARG_INCR -# include /* INFRINGES ON USER NAME SPACE */ -# elif defined _AIX -# define YYSTACK_ALLOC __alloca -# elif defined _MSC_VER -# include /* INFRINGES ON USER NAME SPACE */ -# define alloca _alloca -# else -# define YYSTACK_ALLOC alloca -# if ! defined _ALLOCA_H && ! defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -# include /* INFRINGES ON USER NAME SPACE */ -# ifndef _STDLIB_H -# define _STDLIB_H 1 -# endif -# endif -# endif -# endif -# endif - -# ifdef YYSTACK_ALLOC - /* Pacify GCC's `empty if-body' warning. */ -# define YYSTACK_FREE(Ptr) do { /* empty */; } while (YYID (0)) -# ifndef YYSTACK_ALLOC_MAXIMUM - /* The OS might guarantee only one guard page at the bottom of the stack, - and a page size can be as small as 4096 bytes. So we cannot safely - invoke alloca (N) if N exceeds 4096. Use a slightly smaller number - to allow for a few compiler-allocated temporary stack slots. */ -# define YYSTACK_ALLOC_MAXIMUM 4032 /* reasonable circa 2006 */ -# endif -# else -# define YYSTACK_ALLOC YYMALLOC -# define YYSTACK_FREE YYFREE -# ifndef YYSTACK_ALLOC_MAXIMUM -# define YYSTACK_ALLOC_MAXIMUM YYSIZE_MAXIMUM -# endif -# if (defined __cplusplus && ! defined _STDLIB_H \ - && ! ((defined YYMALLOC || defined malloc) \ - && (defined YYFREE || defined free))) -# include /* INFRINGES ON USER NAME SPACE */ -# ifndef _STDLIB_H -# define _STDLIB_H 1 -# endif -# endif -# ifndef YYMALLOC -# define YYMALLOC malloc -# if ! defined malloc && ! defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -void *malloc (YYSIZE_T); /* INFRINGES ON USER NAME SPACE */ -# endif -# endif -# ifndef YYFREE -# define YYFREE free -# if ! defined free && ! defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -void free (void *); /* INFRINGES ON USER NAME SPACE */ -# endif -# endif -# endif -#endif /* ! defined lp_yyoverflow || YYERROR_VERBOSE */ - - -#if (! defined lp_yyoverflow \ - && (! defined __cplusplus \ - || (defined YYSTYPE_IS_TRIVIAL && YYSTYPE_IS_TRIVIAL))) - -/* A type that is properly aligned for any stack member. */ -union lp_yyalloc -{ - lp_yytype_int16 lp_yyss; - YYSTYPE lp_yyvs; - }; - -/* The size of the maximum gap between one aligned stack and the next. */ -# define YYSTACK_GAP_MAXIMUM (sizeof (union lp_yyalloc) - 1) - -/* The size of an array large to enough to hold all stacks, each with - N elements. */ -# define YYSTACK_BYTES(N) \ - ((N) * (sizeof (lp_yytype_int16) + sizeof (YYSTYPE)) \ - + YYSTACK_GAP_MAXIMUM) - -/* Copy COUNT objects from FROM to TO. The source and destination do - not overlap. */ -# ifndef YYCOPY -# if defined __GNUC__ && 1 < __GNUC__ -# define YYCOPY(To, From, Count) \ - __builtin_memcpy (To, From, (Count) * sizeof (*(From))) -# else -# define YYCOPY(To, From, Count) \ - do \ - { \ - YYSIZE_T lp_yyi; \ - for (lp_yyi = 0; lp_yyi < (Count); lp_yyi++) \ - (To)[lp_yyi] = (From)[lp_yyi]; \ - } \ - while (YYID (0)) -# endif -# endif - -/* Relocate STACK from its old location to the new one. The - local variables YYSIZE and YYSTACKSIZE give the old and new number of - elements in the stack, and YYPTR gives the new location of the - stack. Advance YYPTR to a properly aligned location for the next - stack. */ -# define YYSTACK_RELOCATE(Stack) \ - do \ - { \ - YYSIZE_T lp_yynewbytes; \ - YYCOPY (&lp_yyptr->Stack, Stack, lp_yysize); \ - Stack = &lp_yyptr->Stack; \ - lp_yynewbytes = lp_yystacksize * sizeof (*Stack) + YYSTACK_GAP_MAXIMUM; \ - lp_yyptr += lp_yynewbytes / sizeof (*lp_yyptr); \ - } \ - while (YYID (0)) - -#endif - -/* YYFINAL -- State number of the termination state. */ -#define YYFINAL 3 -/* YYLAST -- Last index in YYTABLE. */ -#define YYLAST 115 - -/* YYNTOKENS -- Number of terminals. */ -#define YYNTOKENS 25 -/* YYNNTS -- Number of nonterminals. */ -#define YYNNTS 56 -/* YYNRULES -- Number of rules. */ -#define YYNRULES 89 -/* YYNRULES -- Number of states. */ -#define YYNSTATES 123 - -/* YYTRANSLATE(YYLEX) -- Bison symbol number corresponding to YYLEX. */ -#define YYUNDEFTOK 2 -#define YYMAXUTOK 279 - -#define YYTRANSLATE(YYX) \ - ((unsigned int) (YYX) <= YYMAXUTOK ? lp_yytranslate[YYX] : YYUNDEFTOK) - -/* YYTRANSLATE[YYLEX] -- Bison symbol number corresponding to YYLEX. */ -static const lp_yytype_uint8 lp_yytranslate[] = -{ - 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, - 2, 2, 2, 2, 2, 2, 1, 2, 3, 4, - 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, - 15, 16, 17, 18, 19, 20, 21, 22, 23, 24 -}; - -#if YYDEBUG -/* YYPRHS[YYN] -- Index of the first RHS symbol of rule number YYN in - YYRHS. */ -static const lp_yytype_uint8 lp_yyprhs[] = -{ - 0, 0, 3, 4, 5, 10, 13, 16, 18, 21, - 23, 25, 27, 29, 31, 34, 36, 37, 41, 42, - 43, 44, 53, 55, 56, 57, 63, 65, 67, 69, - 70, 74, 75, 78, 80, 83, 86, 88, 89, 93, - 95, 97, 99, 102, 104, 106, 108, 110, 112, 114, - 116, 118, 120, 122, 124, 127, 129, 131, 133, 135, - 137, 138, 142, 143, 149, 151, 154, 156, 157, 161, - 163, 164, 169, 171, 174, 176, 178, 180, 184, 186, - 188, 190, 191, 193, 195, 198, 202, 205, 208, 211 -}; - -/* YYRHS -- A `-1'-separated list of the rules' RHS. */ -static const lp_yytype_int8 lp_yyrhs[] = -{ - 27, 0, -1, -1, -1, 28, 29, 32, 58, -1, - 23, 30, -1, 22, 30, -1, 30, -1, 31, 19, - -1, 26, -1, 46, -1, 26, -1, 33, -1, 34, - -1, 33, 34, -1, 36, -1, -1, 6, 35, 36, - -1, -1, -1, -1, 43, 37, 52, 38, 44, 39, - 40, 19, -1, 26, -1, -1, -1, 52, 41, 53, - 42, 57, -1, 26, -1, 44, -1, 46, -1, -1, - 7, 45, 57, -1, -1, 47, 48, -1, 49, -1, - 48, 49, -1, 55, 50, -1, 54, -1, -1, 56, - 51, 3, -1, 16, -1, 17, -1, 18, -1, 55, - 54, -1, 7, -1, 5, -1, 4, -1, 26, -1, - 14, -1, 26, -1, 15, -1, 26, -1, 26, -1, - 59, -1, 61, -1, 59, 61, -1, 8, -1, 9, - -1, 10, -1, 11, -1, 13, -1, -1, 60, 62, - 65, -1, -1, 64, 66, 71, 68, 19, -1, 63, - -1, 65, 63, -1, 26, -1, -1, 12, 67, 77, - -1, 26, -1, -1, 17, 5, 69, 70, -1, 26, - -1, 21, 5, -1, 26, -1, 72, -1, 78, -1, - 72, 73, 78, -1, 26, -1, 20, -1, 26, -1, - -1, 26, -1, 26, -1, 3, 74, -1, 6, 75, - 79, -1, 54, 76, -1, 77, 80, -1, 3, 74, - -1, 6, 75, 54, 76, -1 -}; - -/* YYRLINE[YYN] -- source line where rule number YYN was defined. */ -static const lp_yytype_uint16 lp_yyrline[] = -{ - 0, 116, 116, 120, 120, 145, 149, 153, 156, 170, - 171, 201, 202, 205, 206, 210, 212, 211, 224, 232, - 242, 223, 288, 298, 311, 297, 342, 355, 364, 366, - 365, 376, 376, 400, 401, 405, 444, 452, 451, 470, - 470, 470, 473, 475, 489, 489, 492, 500, 510, 517, - 526, 547, 548, 551, 552, 555, 555, 555, 555, 555, - 560, 559, 570, 570, 598, 599, 602, 604, 603, 614, - 626, 624, 643, 650, 660, 661, 664, 665, 670, 671, - 674, 703, 724, 749, 770, 772, 777, 779, 784, 786 -}; -#endif - -#if YYDEBUG || YYERROR_VERBOSE || YYTOKEN_TABLE -/* YYTNAME[SYMBOL-NUM] -- String name of the symbol SYMBOL-NUM. - First, the terminals, then, starting at YYNTOKENS, nonterminals. */ -static const char *const lp_yytname[] = -{ - "$end", "error", "$undefined", "VAR", "CONS", "INTCONS", - "VARIABLECOLON", "INF", "SEC_INT", "SEC_BIN", "SEC_SEC", "SEC_SOS", - "SOSDESCR", "SEC_FREE", "TOK_SIGN", "AR_M_OP", "RE_OPEQ", "RE_OPLE", - "RE_OPGE", "END_C", "COMMA", "COLON", "MINIMISE", "MAXIMISE", - "UNDEFINED", "$accept", "EMPTY", "inputfile", "@1", "objective_function", - "real_of", "lineair_sum", "constraints", "x_constraints", "constraint", - "@2", "real_constraint", "@3", "@4", "@5", "optionalrange", "@6", "@7", - "x_lineair_sum2", "x_lineair_sum3", "@8", "x_lineair_sum", "@9", - "x_lineair_sum1", "x_lineair_term", "x_lineair_term1", "@10", "RE_OP", - "cons_term", "REALCONS", "x_SIGN", "optional_AR_M_OP", "RHS_STORE", - "int_bin_sec_sos_free_declarations", "real_int_bin_sec_sos_free_decls", - "SEC_INT_BIN_SEC_SOS_FREE", "int_bin_sec_sos_free_declaration", "@11", - "xx_int_bin_sec_sos_free_declaration", "@12", - "x_int_bin_sec_sos_free_declaration", "optionalsos", "@13", - "optionalsostype", "@14", "optionalSOSweight", "vars", "x_vars", - "optionalcomma", "variable", "variablecolon", "sosweight", "sosdescr", - "onevarwithoptionalweight", "INTCONSorVARIABLE", - "x_onevarwithoptionalweight", 0 -}; -#endif - -# ifdef YYPRINT -/* YYTOKNUM[YYLEX-NUM] -- Internal token number corresponding to - token YYLEX-NUM. */ -static const lp_yytype_uint16 lp_yytoknum[] = -{ - 0, 256, 257, 258, 259, 260, 261, 262, 263, 264, - 265, 266, 267, 268, 269, 270, 271, 272, 273, 274, - 275, 276, 277, 278, 279 -}; -# endif - -/* YYR1[YYN] -- Symbol number of symbol that rule YYN derives. */ -static const lp_yytype_uint8 lp_yyr1[] = -{ - 0, 25, 26, 28, 27, 29, 29, 29, 30, 31, - 31, 32, 32, 33, 33, 34, 35, 34, 37, 38, - 39, 36, 40, 41, 42, 40, 43, 43, 44, 45, - 44, 47, 46, 48, 48, 49, 50, 51, 50, 52, - 52, 52, 53, 53, 54, 54, 55, 55, 56, 56, - 57, 58, 58, 59, 59, 60, 60, 60, 60, 60, - 62, 61, 64, 63, 65, 65, 66, 67, 66, 68, - 69, 68, 70, 70, 71, 71, 72, 72, 73, 73, - 74, 75, 76, 77, 78, 78, 79, 79, 80, 80 -}; - -/* YYR2[YYN] -- Number of symbols composing right hand side of rule YYN. */ -static const lp_yytype_uint8 lp_yyr2[] = -{ - 0, 2, 0, 0, 4, 2, 2, 1, 2, 1, - 1, 1, 1, 1, 2, 1, 0, 3, 0, 0, - 0, 8, 1, 0, 0, 5, 1, 1, 1, 0, - 3, 0, 2, 1, 2, 2, 1, 0, 3, 1, - 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, - 0, 3, 0, 5, 1, 2, 1, 0, 3, 1, - 0, 4, 1, 2, 1, 1, 1, 3, 1, 1, - 1, 0, 1, 1, 2, 3, 2, 2, 2, 4 -}; - -/* YYDEFACT[STATE-NAME] -- Default rule to reduce with in state - STATE-NUM when YYTABLE doesn't specify something else to do. Zero - means the default is an error. */ -static const lp_yytype_uint8 lp_yydefact[] = -{ - 3, 0, 31, 1, 31, 31, 9, 2, 7, 0, - 10, 2, 6, 5, 16, 29, 11, 2, 12, 13, - 15, 18, 27, 28, 8, 47, 46, 2, 33, 2, - 31, 2, 55, 56, 57, 58, 59, 51, 4, 52, - 60, 53, 26, 14, 0, 34, 45, 44, 49, 48, - 35, 36, 37, 17, 50, 30, 54, 62, 39, 40, - 41, 19, 0, 64, 2, 61, 31, 38, 67, 66, - 2, 65, 20, 2, 2, 81, 74, 2, 2, 76, - 2, 83, 68, 80, 84, 2, 0, 69, 0, 79, - 78, 0, 22, 0, 23, 2, 0, 85, 70, 63, - 77, 21, 2, 82, 86, 2, 81, 87, 2, 43, - 24, 0, 88, 0, 0, 72, 71, 2, 42, 2, - 73, 25, 89 -}; - -/* YYDEFGOTO[NTERM-NUM]. */ -static const lp_yytype_int8 lp_yydefgoto[] = -{ - -1, 6, 1, 2, 7, 8, 9, 17, 18, 19, - 30, 20, 44, 66, 80, 93, 102, 117, 21, 22, - 31, 23, 11, 27, 28, 50, 62, 61, 110, 51, - 29, 52, 55, 38, 39, 40, 41, 57, 63, 64, - 65, 70, 73, 88, 108, 116, 77, 78, 91, 84, - 85, 104, 82, 79, 97, 107 -}; - -/* YYPACT[STATE-NUM] -- Index in YYTABLE of the portion describing - STATE-NUM. */ -#define YYPACT_NINF -85 -static const lp_yytype_int8 lp_yypact[] = -{ - -85, 13, 46, -85, 2, 2, -85, 32, -85, 7, - -85, 26, -85, -85, -85, -85, 64, 40, 27, -85, - -85, -85, -85, -85, -85, -85, -85, 42, -85, 3, - -2, -85, -85, -85, -85, -85, -85, -85, -85, 40, - -85, -85, -85, -85, 67, -85, -85, -85, -85, -85, - -85, -85, -85, -85, -85, -85, -85, -85, -85, -85, - -85, -85, 59, -85, 52, 0, 84, -85, -85, -85, - 22, -85, -85, -85, -85, -85, -85, 60, 35, -85, - 67, -85, -85, -85, -85, 82, 87, -85, 74, -85, - -85, 22, -85, 75, -85, -85, 73, -85, -85, -85, - -85, -85, -5, -85, -85, -85, -85, -85, 76, -85, - -85, 82, -85, 82, 91, -85, -85, -85, -85, -85, - -85, -85, -85 -}; - -/* YYPGOTO[NTERM-NUM]. */ -static const lp_yytype_int8 lp_yypgoto[] = -{ - -85, -7, -85, -85, -85, 85, -85, -85, -85, 81, - -85, 72, -85, -85, -85, -85, -85, -85, -85, 34, - -85, 70, -85, -85, 77, -85, -85, 23, -85, -84, - 4, -85, -12, -85, -85, -85, 68, -85, 43, -85, - -85, -85, -85, -85, -85, -85, -85, -85, -85, 6, - 8, -10, 28, 24, -85, -85 -}; - -/* YYTABLE[YYPACT[STATE-NUM]]. What to do in state STATE-NUM. If - positive, shift that token. If negative, reduce the rule which - number is the opposite. If zero, do what YYDEFACT says. - If YYTABLE_NINF, syntax error. */ -#define YYTABLE_NINF -76 -static const lp_yytype_int8 lp_yytable[] = -{ - 16, 95, 109, -62, 26, 15, -62, 46, 47, 25, - 37, 42, -62, 3, -2, -2, -2, -62, 48, -62, - 26, -2, 49, 42, 54, 74, 24, 118, 75, 119, - -31, -31, -31, 14, 15, -31, -31, -31, 14, 15, - 25, -31, -31, -2, -2, -2, -31, -31, 32, 33, - 34, 35, -75, 36, -75, 89, 25, 69, -32, -32, - -32, -32, 67, 76, 68, -2, 81, 83, 4, 5, - 87, 90, 10, 92, 10, 10, 105, 86, 81, 106, - -26, -26, -26, 58, 59, 60, 46, 47, 103, 12, - 13, 15, 98, 99, 101, 26, 120, 114, 83, 43, - 72, 115, 53, 94, 45, 121, 111, 56, 71, 122, - 54, 112, 103, 96, 113, 100 -}; - -static const lp_yytype_uint8 lp_yycheck[] = -{ - 7, 85, 7, 3, 11, 7, 6, 4, 5, 14, - 17, 18, 12, 0, 16, 17, 18, 17, 15, 19, - 27, 19, 29, 30, 31, 3, 19, 111, 6, 113, - 3, 4, 5, 6, 7, 3, 4, 5, 6, 7, - 14, 14, 15, 16, 17, 18, 14, 15, 8, 9, - 10, 11, 17, 13, 19, 20, 14, 64, 16, 17, - 18, 19, 3, 70, 12, 19, 73, 74, 22, 23, - 77, 78, 2, 80, 4, 5, 3, 17, 85, 6, - 16, 17, 18, 16, 17, 18, 4, 5, 95, 4, - 5, 7, 5, 19, 19, 102, 5, 21, 105, 18, - 66, 108, 30, 80, 27, 117, 102, 39, 65, 119, - 117, 105, 119, 85, 106, 91 -}; - -/* YYSTOS[STATE-NUM] -- The (internal number of the) accessing - symbol of state STATE-NUM. */ -static const lp_yytype_uint8 lp_yystos[] = -{ - 0, 27, 28, 0, 22, 23, 26, 29, 30, 31, - 46, 47, 30, 30, 6, 7, 26, 32, 33, 34, - 36, 43, 44, 46, 19, 14, 26, 48, 49, 55, - 35, 45, 8, 9, 10, 11, 13, 26, 58, 59, - 60, 61, 26, 34, 37, 49, 4, 5, 15, 26, - 50, 54, 56, 36, 26, 57, 61, 62, 16, 17, - 18, 52, 51, 63, 64, 65, 38, 3, 12, 26, - 66, 63, 44, 67, 3, 6, 26, 71, 72, 78, - 39, 26, 77, 26, 74, 75, 17, 26, 68, 20, - 26, 73, 26, 40, 52, 54, 77, 79, 5, 19, - 78, 19, 41, 26, 76, 3, 6, 80, 69, 7, - 53, 55, 74, 75, 21, 26, 70, 42, 54, 54, - 5, 57, 76 -}; - -#define lp_yyerrok (lp_yyerrstatus = 0) -#define lp_yyclearin (lp_yychar = YYEMPTY) -#define YYEMPTY (-2) -#define YYEOF 0 - -#define YYACCEPT goto lp_yyacceptlab -#define YYABORT goto lp_yyabortlab -#define YYERROR goto lp_yyerrorlab - - -/* Like YYERROR except do call lp_yyerror. This remains here temporarily - to ease the transition to the new meaning of YYERROR, for GCC. - Once GCC version 2 has supplanted version 1, this can go. */ - -#define YYFAIL goto lp_yyerrlab - -#define YYRECOVERING() (!!lp_yyerrstatus) - -#define YYBACKUP(Token, Value) \ -do \ - if (lp_yychar == YYEMPTY && lp_yylen == 1) \ - { \ - lp_yychar = (Token); \ - lp_yylval = (Value); \ - lp_yytoken = YYTRANSLATE (lp_yychar); \ - YYPOPSTACK (1); \ - goto lp_yybackup; \ - } \ - else \ - { \ - lp_yyerror (parm, scanner, YY_("syntax error: cannot back up")); \ - YYERROR; \ - } \ -while (YYID (0)) - - -#define YYTERROR 1 -#define YYERRCODE 256 - - -/* YYLLOC_DEFAULT -- Set CURRENT to span from RHS[1] to RHS[N]. - If N is 0, then set CURRENT to the empty location which ends - the previous symbol: RHS[0] (always defined). */ - -#define YYRHSLOC(Rhs, K) ((Rhs)[K]) -#ifndef YYLLOC_DEFAULT -# define YYLLOC_DEFAULT(Current, Rhs, N) \ - do \ - if (YYID (N)) \ - { \ - (Current).first_line = YYRHSLOC (Rhs, 1).first_line; \ - (Current).first_column = YYRHSLOC (Rhs, 1).first_column; \ - (Current).last_line = YYRHSLOC (Rhs, N).last_line; \ - (Current).last_column = YYRHSLOC (Rhs, N).last_column; \ - } \ - else \ - { \ - (Current).first_line = (Current).last_line = \ - YYRHSLOC (Rhs, 0).last_line; \ - (Current).first_column = (Current).last_column = \ - YYRHSLOC (Rhs, 0).last_column; \ - } \ - while (YYID (0)) -#endif - - -/* YY_LOCATION_PRINT -- Print the location on the stream. - This macro was not mandated originally: define only if we know - we won't break user code: when these are the locations we know. */ - -#ifndef YY_LOCATION_PRINT -# if YYLTYPE_IS_TRIVIAL -# define YY_LOCATION_PRINT(File, Loc) \ - fprintf (File, "%d.%d-%d.%d", \ - (Loc).first_line, (Loc).first_column, \ - (Loc).last_line, (Loc).last_column) -# else -# define YY_LOCATION_PRINT(File, Loc) ((void) 0) -# endif -#endif - - -/* YYLEX -- calling `lp_yylex' with the right arguments. */ - -#ifdef YYLEX_PARAM -# define YYLEX lp_yylex (&lp_yylval, YYLEX_PARAM) -#else -# define YYLEX lp_yylex (&lp_yylval, scanner) -#endif - -/* Enable debugging if requested. */ -#if YYDEBUG - -# ifndef YYFPRINTF -# include /* INFRINGES ON USER NAME SPACE */ -# define YYFPRINTF fprintf -# endif - -# define YYDPRINTF(Args) \ -do { \ - if (lp_yydebug) \ - YYFPRINTF Args; \ -} while (YYID (0)) - -# define YY_SYMBOL_PRINT(Title, Type, Value, Location) \ -do { \ - if (lp_yydebug) \ - { \ - YYFPRINTF (stderr, "%s ", Title); \ - lp_yy_symbol_print (stderr, \ - Type, Value, parm, scanner); \ - YYFPRINTF (stderr, "\n"); \ - } \ -} while (YYID (0)) - - -/*--------------------------------. -| Print this symbol on YYOUTPUT. | -`--------------------------------*/ - -/*ARGSUSED*/ -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static void -lp_yy_symbol_value_print (FILE *lp_yyoutput, int lp_yytype, YYSTYPE const * const lp_yyvaluep, parse_parm *parm, void *scanner) -#else -static void -lp_yy_symbol_value_print (lp_yyoutput, lp_yytype, lp_yyvaluep, parm, scanner) - FILE *lp_yyoutput; - int lp_yytype; - YYSTYPE const * const lp_yyvaluep; - parse_parm *parm; - void *scanner; -#endif -{ - if (!lp_yyvaluep) - return; - YYUSE (parm); - YYUSE (scanner); -# ifdef YYPRINT - if (lp_yytype < YYNTOKENS) - YYPRINT (lp_yyoutput, lp_yytoknum[lp_yytype], *lp_yyvaluep); -# else - YYUSE (lp_yyoutput); -# endif - switch (lp_yytype) - { - default: - break; - } -} - - -/*--------------------------------. -| Print this symbol on YYOUTPUT. | -`--------------------------------*/ - -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static void -lp_yy_symbol_print (FILE *lp_yyoutput, int lp_yytype, YYSTYPE const * const lp_yyvaluep, parse_parm *parm, void *scanner) -#else -static void -lp_yy_symbol_print (lp_yyoutput, lp_yytype, lp_yyvaluep, parm, scanner) - FILE *lp_yyoutput; - int lp_yytype; - YYSTYPE const * const lp_yyvaluep; - parse_parm *parm; - void *scanner; -#endif -{ - if (lp_yytype < YYNTOKENS) - YYFPRINTF (lp_yyoutput, "token %s (", lp_yytname[lp_yytype]); - else - YYFPRINTF (lp_yyoutput, "nterm %s (", lp_yytname[lp_yytype]); - - lp_yy_symbol_value_print (lp_yyoutput, lp_yytype, lp_yyvaluep, parm, scanner); - YYFPRINTF (lp_yyoutput, ")"); -} - -/*------------------------------------------------------------------. -| lp_yy_stack_print -- Print the state stack from its BOTTOM up to its | -| TOP (included). | -`------------------------------------------------------------------*/ - -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static void -lp_yy_stack_print (lp_yytype_int16 *bottom, lp_yytype_int16 *top) -#else -static void -lp_yy_stack_print (bottom, top) - lp_yytype_int16 *bottom; - lp_yytype_int16 *top; -#endif -{ - YYFPRINTF (stderr, "Stack now"); - for (; bottom <= top; ++bottom) - YYFPRINTF (stderr, " %d", *bottom); - YYFPRINTF (stderr, "\n"); -} - -# define YY_STACK_PRINT(Bottom, Top) \ -do { \ - if (lp_yydebug) \ - lp_yy_stack_print ((Bottom), (Top)); \ -} while (YYID (0)) - - -/*------------------------------------------------. -| Report that the YYRULE is going to be reduced. | -`------------------------------------------------*/ - -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static void -lp_yy_reduce_print (YYSTYPE *lp_yyvsp, int lp_yyrule, parse_parm *parm, void *scanner) -#else -static void -lp_yy_reduce_print (lp_yyvsp, lp_yyrule, parm, scanner) - YYSTYPE *lp_yyvsp; - int lp_yyrule; - parse_parm *parm; - void *scanner; -#endif -{ - int lp_yynrhs = lp_yyr2[lp_yyrule]; - int lp_yyi; - unsigned long int lp_yylno = lp_yyrline[lp_yyrule]; - YYFPRINTF (stderr, "Reducing stack by rule %d (line %lu):\n", - lp_yyrule - 1, lp_yylno); - /* The symbols being reduced. */ - for (lp_yyi = 0; lp_yyi < lp_yynrhs; lp_yyi++) - { - fprintf (stderr, " $%d = ", lp_yyi + 1); - lp_yy_symbol_print (stderr, lp_yyrhs[lp_yyprhs[lp_yyrule] + lp_yyi], - &(lp_yyvsp[(lp_yyi + 1) - (lp_yynrhs)]) - , parm, scanner); - fprintf (stderr, "\n"); - } -} - -# define YY_REDUCE_PRINT(Rule) \ -do { \ - if (lp_yydebug) \ - lp_yy_reduce_print (lp_yyvsp, Rule, parm, scanner); \ -} while (YYID (0)) - -/* Nonzero means print parse trace. It is left uninitialized so that - multiple parsers can coexist. */ -int lp_yydebug; -#else /* !YYDEBUG */ -# define YYDPRINTF(Args) -# define YY_SYMBOL_PRINT(Title, Type, Value, Location) -# define YY_STACK_PRINT(Bottom, Top) -# define YY_REDUCE_PRINT(Rule) -#endif /* !YYDEBUG */ - - -/* YYINITDEPTH -- initial size of the parser's stacks. */ -#ifndef YYINITDEPTH -# define YYINITDEPTH 200 -#endif - -/* YYMAXDEPTH -- maximum size the stacks can grow to (effective only - if the built-in stack extension method is used). - - Do not make this value too large; the results are undefined if - YYSTACK_ALLOC_MAXIMUM < YYSTACK_BYTES (YYMAXDEPTH) - evaluated with infinite-precision integer arithmetic. */ - -#ifndef YYMAXDEPTH -# define YYMAXDEPTH 10000 -#endif - - - -#if YYERROR_VERBOSE - -# ifndef lp_yystrlen -# if defined __GLIBC__ && defined _STRING_H -# define lp_yystrlen strlen -# else -/* Return the length of YYSTR. */ -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static YYSIZE_T -lp_yystrlen (const char *lp_yystr) -#else -static YYSIZE_T -lp_yystrlen (lp_yystr) - const char *lp_yystr; -#endif -{ - YYSIZE_T lp_yylen; - for (lp_yylen = 0; lp_yystr[lp_yylen]; lp_yylen++) - continue; - return lp_yylen; -} -# endif -# endif - -# ifndef lp_yystpcpy -# if defined __GLIBC__ && defined _STRING_H && defined _GNU_SOURCE -# define lp_yystpcpy stpcpy -# else -/* Copy YYSRC to YYDEST, returning the address of the terminating '\0' in - YYDEST. */ -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static char * -lp_yystpcpy (char *lp_yydest, const char *lp_yysrc) -#else -static char * -lp_yystpcpy (lp_yydest, lp_yysrc) - char *lp_yydest; - const char *lp_yysrc; -#endif -{ - char *lp_yyd = lp_yydest; - const char *lp_yys = lp_yysrc; - - while ((*lp_yyd++ = *lp_yys++) != '\0') - continue; - - return lp_yyd - 1; -} -# endif -# endif - -# ifndef lp_yytnamerr -/* Copy to YYRES the contents of YYSTR after stripping away unnecessary - quotes and backslashes, so that it's suitable for lp_yyerror. The - heuristic is that double-quoting is unnecessary unless the string - contains an apostrophe, a comma, or backslash (other than - backslash-backslash). YYSTR is taken from lp_yytname. If YYRES is - null, do not copy; instead, return the length of what the result - would have been. */ -static YYSIZE_T -lp_yytnamerr (char *lp_yyres, const char *lp_yystr) -{ - if (*lp_yystr == '"') - { - YYSIZE_T lp_yyn = 0; - char const *lp_yyp = lp_yystr; - - for (;;) - switch (*++lp_yyp) - { - case '\'': - case ',': - goto do_not_strip_quotes; - - case '\\': - if (*++lp_yyp != '\\') - goto do_not_strip_quotes; - /* Fall through. */ - default: - if (lp_yyres) - lp_yyres[lp_yyn] = *lp_yyp; - lp_yyn++; - break; - - case '"': - if (lp_yyres) - lp_yyres[lp_yyn] = '\0'; - return lp_yyn; - } - do_not_strip_quotes: ; - } - - if (! lp_yyres) - return lp_yystrlen (lp_yystr); - - return lp_yystpcpy (lp_yyres, lp_yystr) - lp_yyres; -} -# endif - -/* Copy into YYRESULT an error message about the unexpected token - YYCHAR while in state YYSTATE. Return the number of bytes copied, - including the terminating null byte. If YYRESULT is null, do not - copy anything; just return the number of bytes that would be - copied. As a special case, return 0 if an ordinary "syntax error" - message will do. Return YYSIZE_MAXIMUM if overflow occurs during - size calculation. */ -static YYSIZE_T -lp_yysyntax_error (char *lp_yyresult, int lp_yystate, int lp_yychar) -{ - int lp_yyn = lp_yypact[lp_yystate]; - - if (! (YYPACT_NINF < lp_yyn && lp_yyn <= YYLAST)) - return 0; - else - { - int lp_yytype = YYTRANSLATE (lp_yychar); - YYSIZE_T lp_yysize0 = lp_yytnamerr (0, lp_yytname[lp_yytype]); - YYSIZE_T lp_yysize = lp_yysize0; - YYSIZE_T lp_yysize1; - int lp_yysize_overflow = 0; - enum { YYERROR_VERBOSE_ARGS_MAXIMUM = 5 }; - char const *lp_yyarg[YYERROR_VERBOSE_ARGS_MAXIMUM]; - int lp_yyx; - -# if 0 - /* This is so xgettext sees the translatable formats that are - constructed on the fly. */ - YY_("syntax error, unexpected %s"); - YY_("syntax error, unexpected %s, expecting %s"); - YY_("syntax error, unexpected %s, expecting %s or %s"); - YY_("syntax error, unexpected %s, expecting %s or %s or %s"); - YY_("syntax error, unexpected %s, expecting %s or %s or %s or %s"); -# endif - char *lp_yyfmt; - char const *lp_yyf; - static char const lp_yyunexpected[] = "syntax error, unexpected %s"; - static char const lp_yyexpecting[] = ", expecting %s"; - static char const lp_yyor[] = " or %s"; - char lp_yyformat[sizeof lp_yyunexpected - + sizeof lp_yyexpecting - 1 - + ((YYERROR_VERBOSE_ARGS_MAXIMUM - 2) - * (sizeof lp_yyor - 1))]; - char const *lp_yyprefix = lp_yyexpecting; - - /* Start YYX at -YYN if negative to avoid negative indexes in - YYCHECK. */ - int lp_yyxbegin = lp_yyn < 0 ? -lp_yyn : 0; - - /* Stay within bounds of both lp_yycheck and lp_yytname. */ - int lp_yychecklim = YYLAST - lp_yyn + 1; - int lp_yyxend = lp_yychecklim < YYNTOKENS ? lp_yychecklim : YYNTOKENS; - int lp_yycount = 1; - - lp_yyarg[0] = lp_yytname[lp_yytype]; - lp_yyfmt = lp_yystpcpy (lp_yyformat, lp_yyunexpected); - - for (lp_yyx = lp_yyxbegin; lp_yyx < lp_yyxend; ++lp_yyx) - if (lp_yycheck[lp_yyx + lp_yyn] == lp_yyx && lp_yyx != YYTERROR) - { - if (lp_yycount == YYERROR_VERBOSE_ARGS_MAXIMUM) - { - lp_yycount = 1; - lp_yysize = lp_yysize0; - lp_yyformat[sizeof lp_yyunexpected - 1] = '\0'; - break; - } - lp_yyarg[lp_yycount++] = lp_yytname[lp_yyx]; - lp_yysize1 = lp_yysize + lp_yytnamerr (0, lp_yytname[lp_yyx]); - lp_yysize_overflow |= (lp_yysize1 < lp_yysize); - lp_yysize = lp_yysize1; - lp_yyfmt = lp_yystpcpy (lp_yyfmt, lp_yyprefix); - lp_yyprefix = lp_yyor; - } - - lp_yyf = YY_(lp_yyformat); - lp_yysize1 = lp_yysize + lp_yystrlen (lp_yyf); - lp_yysize_overflow |= (lp_yysize1 < lp_yysize); - lp_yysize = lp_yysize1; - - if (lp_yysize_overflow) - return YYSIZE_MAXIMUM; - - if (lp_yyresult) - { - /* Avoid sprintf, as that infringes on the user's name space. - Don't have undefined behavior even if the translation - produced a string with the wrong number of "%s"s. */ - char *lp_yyp = lp_yyresult; - int lp_yyi = 0; - while ((*lp_yyp = *lp_yyf) != '\0') - { - if (*lp_yyp == '%' && lp_yyf[1] == 's' && lp_yyi < lp_yycount) - { - lp_yyp += lp_yytnamerr (lp_yyp, lp_yyarg[lp_yyi++]); - lp_yyf += 2; - } - else - { - lp_yyp++; - lp_yyf++; - } - } - } - return lp_yysize; - } -} -#endif /* YYERROR_VERBOSE */ - - -/*-----------------------------------------------. -| Release the memory associated to this symbol. | -`-----------------------------------------------*/ - -/*ARGSUSED*/ -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -static void -lp_yydestruct (const char *lp_yymsg, int lp_yytype, YYSTYPE *lp_yyvaluep, parse_parm *parm, void *scanner) -#else -static void -lp_yydestruct (lp_yymsg, lp_yytype, lp_yyvaluep, parm, scanner) - const char *lp_yymsg; - int lp_yytype; - YYSTYPE *lp_yyvaluep; - parse_parm *parm; - void *scanner; -#endif -{ - YYUSE (lp_yyvaluep); - YYUSE (parm); - YYUSE (scanner); - - if (!lp_yymsg) - lp_yymsg = "Deleting"; - YY_SYMBOL_PRINT (lp_yymsg, lp_yytype, lp_yyvaluep, lp_yylocationp); - - switch (lp_yytype) - { - - default: - break; - } -} - - -/* Prevent warnings from -Wmissing-prototypes. */ - -#ifdef YYPARSE_PARAM -#if defined __STDC__ || defined __cplusplus -int lp_yyparse (void *YYPARSE_PARAM); -#else -int lp_yyparse (); -#endif -#else /* ! YYPARSE_PARAM */ -#if defined __STDC__ || defined __cplusplus -int lp_yyparse (parse_parm *parm, void *scanner); -#else -int lp_yyparse (); -#endif -#endif /* ! YYPARSE_PARAM */ - - - - - - -/*----------. -| lp_yyparse. | -`----------*/ - -#ifdef YYPARSE_PARAM -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -int -lp_yyparse (void *YYPARSE_PARAM) -#else -int -lp_yyparse (YYPARSE_PARAM) - void *YYPARSE_PARAM; -#endif -#else /* ! YYPARSE_PARAM */ -#if (defined __STDC__ || defined __C99__FUNC__ \ - || defined __cplusplus || defined _MSC_VER) -int -lp_yyparse (parse_parm *parm, void *scanner) -#else -int -lp_yyparse (parm, scanner) - parse_parm *parm; - void *scanner; -#endif -#endif -{ - /* The look-ahead symbol. */ -int lp_yychar; - -/* The semantic value of the look-ahead symbol. */ -YYSTYPE lp_yylval; - -/* Number of syntax errors so far. */ -int lp_yynerrs; - - int lp_yystate; - int lp_yyn; - int lp_yyresult; - /* Number of tokens to shift before error messages enabled. */ - int lp_yyerrstatus; - /* Look-ahead token as an internal (translated) token number. */ - int lp_yytoken = 0; -#if YYERROR_VERBOSE - /* Buffer for error messages, and its allocated size. */ - char lp_yymsgbuf[128]; - char *lp_yymsg = lp_yymsgbuf; - YYSIZE_T lp_yymsg_alloc = sizeof lp_yymsgbuf; -#endif - - /* Three stacks and their tools: - `lp_yyss': related to states, - `lp_yyvs': related to semantic values, - `lp_yyls': related to locations. - - Refer to the stacks thru separate pointers, to allow lp_yyoverflow - to reallocate them elsewhere. */ - - /* The state stack. */ - lp_yytype_int16 lp_yyssa[YYINITDEPTH]; - lp_yytype_int16 *lp_yyss = lp_yyssa; - lp_yytype_int16 *lp_yyssp; - - /* The semantic value stack. */ - YYSTYPE lp_yyvsa[YYINITDEPTH]; - YYSTYPE *lp_yyvs = lp_yyvsa; - YYSTYPE *lp_yyvsp; - - - -#define YYPOPSTACK(N) (lp_yyvsp -= (N), lp_yyssp -= (N)) - - YYSIZE_T lp_yystacksize = YYINITDEPTH; - - /* The variables used to return semantic value and location from the - action routines. */ - YYSTYPE lp_yyval; - - - /* The number of symbols on the RHS of the reduced rule. - Keep to zero when no symbol should be popped. */ - int lp_yylen = 0; - - YYDPRINTF ((stderr, "Starting parse\n")); - - lp_yystate = 0; - lp_yyerrstatus = 0; - lp_yynerrs = 0; - lp_yychar = YYEMPTY; /* Cause a token to be read. */ - - /* Initialize stack pointers. - Waste one element of value and location stack - so that they stay on the same level as the state stack. - The wasted elements are never initialized. */ - - lp_yyssp = lp_yyss; - lp_yyvsp = lp_yyvs; - - goto lp_yysetstate; - -/*------------------------------------------------------------. -| lp_yynewstate -- Push a new state, which is found in lp_yystate. | -`------------------------------------------------------------*/ - lp_yynewstate: - /* In all cases, when you get here, the value and location stacks - have just been pushed. So pushing a state here evens the stacks. */ - lp_yyssp++; - - lp_yysetstate: - *lp_yyssp = lp_yystate; - - if (lp_yyss + lp_yystacksize - 1 <= lp_yyssp) - { - /* Get the current used size of the three stacks, in elements. */ - YYSIZE_T lp_yysize = lp_yyssp - lp_yyss + 1; - -#ifdef lp_yyoverflow - { - /* Give user a chance to reallocate the stack. Use copies of - these so that the &'s don't force the real ones into - memory. */ - YYSTYPE *lp_yyvs1 = lp_yyvs; - lp_yytype_int16 *lp_yyss1 = lp_yyss; - - - /* Each stack pointer address is followed by the size of the - data in use in that stack, in bytes. This used to be a - conditional around just the two extra args, but that might - be undefined if lp_yyoverflow is a macro. */ - lp_yyoverflow (YY_("memory exhausted"), - &lp_yyss1, lp_yysize * sizeof (*lp_yyssp), - &lp_yyvs1, lp_yysize * sizeof (*lp_yyvsp), - - &lp_yystacksize); - - lp_yyss = lp_yyss1; - lp_yyvs = lp_yyvs1; - } -#else /* no lp_yyoverflow */ -# ifndef YYSTACK_RELOCATE - goto lp_yyexhaustedlab; -# else - /* Extend the stack our own way. */ - if (YYMAXDEPTH <= lp_yystacksize) - goto lp_yyexhaustedlab; - lp_yystacksize *= 2; - if (YYMAXDEPTH < lp_yystacksize) - lp_yystacksize = YYMAXDEPTH; - - { - lp_yytype_int16 *lp_yyss1 = lp_yyss; - union lp_yyalloc *lp_yyptr = - (union lp_yyalloc *) YYSTACK_ALLOC (YYSTACK_BYTES (lp_yystacksize)); - if (! lp_yyptr) - goto lp_yyexhaustedlab; - YYSTACK_RELOCATE (lp_yyss); - YYSTACK_RELOCATE (lp_yyvs); - -# undef YYSTACK_RELOCATE - if (lp_yyss1 != lp_yyssa) - YYSTACK_FREE (lp_yyss1); - } -# endif -#endif /* no lp_yyoverflow */ - - lp_yyssp = lp_yyss + lp_yysize - 1; - lp_yyvsp = lp_yyvs + lp_yysize - 1; - - - YYDPRINTF ((stderr, "Stack size increased to %lu\n", - (unsigned long int) lp_yystacksize)); - - if (lp_yyss + lp_yystacksize - 1 <= lp_yyssp) - YYABORT; - } - - YYDPRINTF ((stderr, "Entering state %d\n", lp_yystate)); - - goto lp_yybackup; - -/*-----------. -| lp_yybackup. | -`-----------*/ -lp_yybackup: - - /* Do appropriate processing given the current state. Read a - look-ahead token if we need one and don't already have one. */ - - /* First try to decide what to do without reference to look-ahead token. */ - lp_yyn = lp_yypact[lp_yystate]; - if (lp_yyn == YYPACT_NINF) - goto lp_yydefault; - - /* Not known => get a look-ahead token if don't already have one. */ - - /* YYCHAR is either YYEMPTY or YYEOF or a valid look-ahead symbol. */ - if (lp_yychar == YYEMPTY) - { - YYDPRINTF ((stderr, "Reading a token: ")); - lp_yychar = YYLEX; - } - - if (lp_yychar <= YYEOF) - { - lp_yychar = lp_yytoken = YYEOF; - YYDPRINTF ((stderr, "Now at end of input.\n")); - } - else - { - lp_yytoken = YYTRANSLATE (lp_yychar); - YY_SYMBOL_PRINT ("Next token is", lp_yytoken, &lp_yylval, &lp_yylloc); - } - - /* If the proper action on seeing token YYTOKEN is to reduce or to - detect an error, take that action. */ - lp_yyn += lp_yytoken; - if (lp_yyn < 0 || YYLAST < lp_yyn || lp_yycheck[lp_yyn] != lp_yytoken) - goto lp_yydefault; - lp_yyn = lp_yytable[lp_yyn]; - if (lp_yyn <= 0) - { - if (lp_yyn == 0 || lp_yyn == YYTABLE_NINF) - goto lp_yyerrlab; - lp_yyn = -lp_yyn; - goto lp_yyreduce; - } - - if (lp_yyn == YYFINAL) - YYACCEPT; - - /* Count tokens shifted since error; after three, turn off error - status. */ - if (lp_yyerrstatus) - lp_yyerrstatus--; - - /* Shift the look-ahead token. */ - YY_SYMBOL_PRINT ("Shifting", lp_yytoken, &lp_yylval, &lp_yylloc); - - /* Discard the shifted token unless it is eof. */ - if (lp_yychar != YYEOF) - lp_yychar = YYEMPTY; - - lp_yystate = lp_yyn; - *++lp_yyvsp = lp_yylval; - - goto lp_yynewstate; - - -/*-----------------------------------------------------------. -| lp_yydefault -- do the default action for the current state. | -`-----------------------------------------------------------*/ -lp_yydefault: - lp_yyn = lp_yydefact[lp_yystate]; - if (lp_yyn == 0) - goto lp_yyerrlab; - goto lp_yyreduce; - - -/*-----------------------------. -| lp_yyreduce -- Do a reduction. | -`-----------------------------*/ -lp_yyreduce: - /* lp_yyn is the number of a rule to reduce with. */ - lp_yylen = lp_yyr2[lp_yyn]; - - /* If YYLEN is nonzero, implement the default value of the action: - `$$ = $1'. - - Otherwise, the following line sets YYVAL to garbage. - This behavior is undocumented and Bison - users should not rely upon it. Assigning to YYVAL - unconditionally makes the parser a bit smaller, and it avoids a - GCC warning that YYVAL may be used uninitialized. */ - lp_yyval = lp_yyvsp[1-lp_yylen]; - - - YY_REDUCE_PRINT (lp_yyn); - switch (lp_yyn) - { - case 3: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->isign = 0; - pv->make_neg = 0; - pv->Sign = 0; - pv->HadConstraint = FALSE; - pv->HadVar = pv->HadVar0 = FALSE; -} - break; - - case 5: - - { - set_obj_dir(PARM, TRUE); -} - break; - - case 6: - - { - set_obj_dir(PARM, FALSE); -} - break; - - case 8: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - add_row(pp); - pv->HadConstraint = FALSE; - pv->HadVar = pv->HadVar0 = FALSE; - pv->isign = 0; - pv->make_neg = 0; -} - break; - - case 16: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(!add_constraint_name(pp, pv->Last_var)) - YYABORT; - pv->HadConstraint = TRUE; -} - break; - - case 18: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->HadVar1 = pv->HadVar0; - pv->HadVar0 = FALSE; -} - break; - - case 19: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(!store_re_op(pp, pv->OP, (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum)) - YYABORT; - pv->make_neg = 1; - pv->f1 = 0; -} - break; - - case 20: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->Had_lineair_sum0 = pv->Had_lineair_sum; - pv->Had_lineair_sum = TRUE; - pv->HadVar2 = pv->HadVar0; - pv->HadVar0 = FALSE; - pv->do_add_row = FALSE; - if(pv->HadConstraint && !pv->HadVar ) { - /* it is a range */ - /* already handled */ - } - else if(!pv->HadConstraint && pv->HadVar) { - /* it is a bound */ - - if(!store_bounds(pp, TRUE)) - YYABORT; - } - else { - /* it is a row restriction */ - if(pv->HadConstraint && pv->HadVar) - store_re_op(pp, '\0', (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum); /* makes sure that data stored in temporary buffers is treated correctly */ - pv->do_add_row = TRUE; - } -} - break; - - case 21: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if((!pv->HadVar) && (!pv->HadConstraint)) { - lp_yyerror(pp, pp->scanner, "parse error"); - YYABORT; - } - if(pv->do_add_row) - add_row(pp); - pv->HadConstraint = FALSE; - pv->HadVar = pv->HadVar0 = FALSE; - pv->isign = 0; - pv->make_neg = 0; - null_tmp_store(pp, TRUE); -} - break; - - case 22: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if((!pv->HadVar1) && (pv->Had_lineair_sum0)) - if(!negate_constraint(pp)) - YYABORT; -} - break; - - case 23: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->make_neg = 0; - pv->isign = 0; - if(pv->HadConstraint) - pv->HadVar = pv->Had_lineair_sum = FALSE; - pv->HadVar0 = FALSE; - if(!store_re_op(pp, (char) ((pv->OP == '<') ? '>' : (pv->OP == '>') ? '<' : pv->OP), (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum)) - YYABORT; -} - break; - - case 24: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->f -= pv->f1; -} - break; - - case 25: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if((pv->HadVar1) || (!pv->HadVar2) || (pv->HadVar0)) { - lp_yyerror(pp, pp->scanner, "parse error"); - YYABORT; - } - - if(pv->HadConstraint && !pv->HadVar ) { - /* it is a range */ - /* already handled */ - if(!negate_constraint(pp)) - YYABORT; - } - else if(!pv->HadConstraint && pv->HadVar) { - /* it is a bound */ - - if(!store_bounds(pp, TRUE)) - YYABORT; - } -} - break; - - case 26: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - /* to allow a range */ - /* constraint: < max */ - if(!pv->HadConstraint) { - lp_yyerror(pp, pp->scanner, "parse error"); - YYABORT; - } - pv->Had_lineair_sum = FALSE; -} - break; - - case 27: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->Had_lineair_sum = TRUE; -} - break; - - case 29: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->isign = pv->Sign; -} - break; - - case 31: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->state = pv->state0 = 0; -} - break; - - case 32: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if (pv->state == 1) { - /* RHS_STORE */ - if ( (pv->isign0 || !pv->make_neg) - && !(pv->isign0 && !pv->make_neg)) /* but not both! */ - pv->f0 = -pv->f0; - if(pv->make_neg) - pv->f1 += pv->f0; - if(!rhs_store(pp, pv->f0, (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum)) - YYABORT; - } -} - break; - - case 35: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if ((pv->HadSign || pv->state == 1) && (pv->state0 == 1)) { - /* RHS_STORE */ - if ( (pv->isign0 || !pv->make_neg) - && !(pv->isign0 && !pv->make_neg)) /* but not both! */ - pv->f0 = -pv->f0; - if(pv->make_neg) - pv->f1 += pv->f0; - if(!rhs_store(pp, pv->f0, (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum)) - YYABORT; - } - if (pv->state == 1) { - pv->f0 = pv->f; - pv->isign0 = pv->isign; - } - if (pv->state == 2) { - if((pv->HadSign) || (pv->state0 != 1)) { - pv->isign0 = pv->isign; - pv->f0 = 1.0; - } - if ( (pv->isign0 || pv->make_neg) - && !(pv->isign0 && pv->make_neg)) /* but not both! */ - pv->f0 = -pv->f0; - if(!var_store(pp, pv->Last_var, pv->f0, (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum)) { - lp_yyerror(pp, pp->scanner, "var_store failed"); - YYABORT; - } - pv->HadConstraint |= pv->HadVar; - pv->HadVar = pv->HadVar0 = TRUE; - } - pv->state0 = pv->state; -} - break; - - case 36: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->state = 1; -} - break; - - case 37: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if ((pv->HasAR_M_OP) && (pv->state != 1)) { - lp_yyerror(pp, pp->scanner, "parse error"); - YYABORT; - } -} - break; - - case 38: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->state = 2; -} - break; - - case 43: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->isign = pv->Sign; -} - break; - - case 46: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->isign = 0; - pv->HadSign = FALSE; -} - break; - - case 47: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->isign = pv->Sign; - pv->HadSign = TRUE; -} - break; - - case 48: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->HasAR_M_OP = FALSE; -} - break; - - case 49: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->HasAR_M_OP = TRUE; -} - break; - - case 50: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if ( (pv->isign || !pv->make_neg) - && !(pv->isign && !pv->make_neg)) /* but not both! */ - pv->f = -pv->f; - if(!rhs_store(pp, pv->f, (int) pv->HadConstraint, (int) pv->HadVar, (int) pv->Had_lineair_sum)) - YYABORT; - pv->isign = 0; -} - break; - - case 60: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pv->Within_sos_decl1 = pv->Within_sos_decl; -} - break; - - case 62: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if((!pv->Within_int_decl) && (!pv->Within_sec_decl) && (!pv->Within_sos_decl1) && (!pv->Within_free_decl)) { - lp_yyerror(pp, pp->scanner, "parse error"); - YYABORT; - } - pv->SOStype = pv->SOStype0; - check_int_sec_sos_free_decl(pp, (int) pv->Within_int_decl, (int) pv->Within_sec_decl, (int) (pv->Within_sos_decl1 = (pv->Within_sos_decl1 ? 1 : 0)), (int) pv->Within_free_decl); -} - break; - - case 63: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if((pv->Within_sos_decl1) && (pv->SOStype == 0)) - { - lp_yyerror(pp, pp->scanner, "Unsupported SOS type (0)"); - YYABORT; - } -} - break; - - case 67: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - FREE(pv->Last_var0); - pv->Last_var0 = strdup(pv->Last_var); -} - break; - - case 69: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(pv->Within_sos_decl1) { - set_sos_type(pp, pv->SOStype); - set_sos_weight(pp, (double) pv->SOSweight, 1); - } -} - break; - - case 70: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if((pv->Within_sos_decl1) && (!pv->SOStype)) - { - set_sos_type(pp, pv->SOStype = (short) (pv->f + .1)); - } - else - { - lp_yyerror(pp, pp->scanner, "SOS type not expected"); - YYABORT; - } -} - break; - - case 72: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - set_sos_weight(pp, (double) pv->SOSweight, 1); -} - break; - - case 73: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - set_sos_weight(pp, pv->f, 1); -} - break; - - case 80: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(pv->Within_sos_decl1 == 1) - { - char buf[16]; - - pv->SOSweight++; - sprintf(buf, "SOS%d", pv->SOSweight); - storevarandweight(pp, buf); - - check_int_sec_sos_free_decl(pp, (int) pv->Within_int_decl, (int) pv->Within_sec_decl, 2, (int) pv->Within_free_decl); - pv->Within_sos_decl1 = 2; - pv->SOSNr = 0; - } - - storevarandweight(pp, pv->Last_var); - - if(pv->Within_sos_decl1 == 2) - { - pv->SOSNr++; - set_sos_weight(pp, (double) pv->SOSNr, 2); - } -} - break; - - case 81: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(!pv->Within_sos_decl1) { - lp_yyerror(pp, pp->scanner, "parse error"); - YYABORT; - } - if(pv->Within_sos_decl1 == 1) { - FREE(pv->Last_var0); - pv->Last_var0 = strdup(pv->Last_var); - } - if(pv->Within_sos_decl1 == 2) - { - storevarandweight(pp, pv->Last_var); - pv->SOSNr++; - set_sos_weight(pp, (double) pv->SOSNr, 2); - } -} - break; - - case 82: - - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(pv->Within_sos_decl1 == 1) - { - char buf[16]; - - pv->SOSweight++; - sprintf(buf, "SOS%d", pv->SOSweight); - storevarandweight(pp, buf); - - check_int_sec_sos_free_decl(pp, (int) pv->Within_int_decl, (int) pv->Within_sec_decl, 2, (int) pv->Within_free_decl); - pv->Within_sos_decl1 = 2; - pv->SOSNr = 0; - - storevarandweight(pp, pv->Last_var0); - pv->SOSNr++; - } - - set_sos_weight(pp, pv->f, 2); -} - break; - - case 83: - - { /* SOS name */ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - if(pv->Within_sos_decl1 == 1) - { - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - storevarandweight(pp, pv->Last_var0); - set_sos_type(pp, pv->SOStype); - check_int_sec_sos_free_decl(pp, (int) pv->Within_int_decl, (int) pv->Within_sec_decl, 2, (int) pv->Within_free_decl); - pv->Within_sos_decl1 = 2; - pv->SOSNr = 0; - pv->SOSweight++; - } -} - break; - - -/* Line 1267 of yacc.c. */ - - default: break; - } - YY_SYMBOL_PRINT ("-> $$ =", lp_yyr1[lp_yyn], &lp_yyval, &lp_yyloc); - - YYPOPSTACK (lp_yylen); - lp_yylen = 0; - YY_STACK_PRINT (lp_yyss, lp_yyssp); - - *++lp_yyvsp = lp_yyval; - - - /* Now `shift' the result of the reduction. Determine what state - that goes to, based on the state we popped back to and the rule - number reduced by. */ - - lp_yyn = lp_yyr1[lp_yyn]; - - lp_yystate = lp_yypgoto[lp_yyn - YYNTOKENS] + *lp_yyssp; - if (0 <= lp_yystate && lp_yystate <= YYLAST && lp_yycheck[lp_yystate] == *lp_yyssp) - lp_yystate = lp_yytable[lp_yystate]; - else - lp_yystate = lp_yydefgoto[lp_yyn - YYNTOKENS]; - - goto lp_yynewstate; - - -/*------------------------------------. -| lp_yyerrlab -- here on detecting error | -`------------------------------------*/ -lp_yyerrlab: - /* If not already recovering from an error, report this error. */ - if (!lp_yyerrstatus) - { - ++lp_yynerrs; -#if ! YYERROR_VERBOSE - lp_yyerror (parm, scanner, YY_("syntax error")); -#else - { - YYSIZE_T lp_yysize = lp_yysyntax_error (0, lp_yystate, lp_yychar); - if (lp_yymsg_alloc < lp_yysize && lp_yymsg_alloc < YYSTACK_ALLOC_MAXIMUM) - { - YYSIZE_T lp_yyalloc = 2 * lp_yysize; - if (! (lp_yysize <= lp_yyalloc && lp_yyalloc <= YYSTACK_ALLOC_MAXIMUM)) - lp_yyalloc = YYSTACK_ALLOC_MAXIMUM; - if (lp_yymsg != lp_yymsgbuf) - YYSTACK_FREE (lp_yymsg); - lp_yymsg = (char *) YYSTACK_ALLOC (lp_yyalloc); - if (lp_yymsg) - lp_yymsg_alloc = lp_yyalloc; - else - { - lp_yymsg = lp_yymsgbuf; - lp_yymsg_alloc = sizeof lp_yymsgbuf; - } - } - - if (0 < lp_yysize && lp_yysize <= lp_yymsg_alloc) - { - (void) lp_yysyntax_error (lp_yymsg, lp_yystate, lp_yychar); - lp_yyerror (parm, scanner, lp_yymsg); - } - else - { - lp_yyerror (parm, scanner, YY_("syntax error")); - if (lp_yysize != 0) - goto lp_yyexhaustedlab; - } - } -#endif - } - - - - if (lp_yyerrstatus == 3) - { - /* If just tried and failed to reuse look-ahead token after an - error, discard it. */ - - if (lp_yychar <= YYEOF) - { - /* Return failure if at end of input. */ - if (lp_yychar == YYEOF) - YYABORT; - } - else - { - lp_yydestruct ("Error: discarding", - lp_yytoken, &lp_yylval, parm, scanner); - lp_yychar = YYEMPTY; - } - } - - /* Else will try to reuse look-ahead token after shifting the error - token. */ - goto lp_yyerrlab1; - - -/*---------------------------------------------------. -| lp_yyerrorlab -- error raised explicitly by YYERROR. | -`---------------------------------------------------*/ -lp_yyerrorlab: - - /* Pacify compilers like GCC when the user code never invokes - YYERROR and the label lp_yyerrorlab therefore never appears in user - code. */ - if (/*CONSTCOND*/ 0) - goto lp_yyerrorlab; - - /* Do not reclaim the symbols of the rule which action triggered - this YYERROR. */ - YYPOPSTACK (lp_yylen); - lp_yylen = 0; - YY_STACK_PRINT (lp_yyss, lp_yyssp); - lp_yystate = *lp_yyssp; - goto lp_yyerrlab1; - - -/*-------------------------------------------------------------. -| lp_yyerrlab1 -- common code for both syntax error and YYERROR. | -`-------------------------------------------------------------*/ -lp_yyerrlab1: - lp_yyerrstatus = 3; /* Each real token shifted decrements this. */ - - for (;;) - { - lp_yyn = lp_yypact[lp_yystate]; - if (lp_yyn != YYPACT_NINF) - { - lp_yyn += YYTERROR; - if (0 <= lp_yyn && lp_yyn <= YYLAST && lp_yycheck[lp_yyn] == YYTERROR) - { - lp_yyn = lp_yytable[lp_yyn]; - if (0 < lp_yyn) - break; - } - } - - /* Pop the current state because it cannot handle the error token. */ - if (lp_yyssp == lp_yyss) - YYABORT; - - - lp_yydestruct ("Error: popping", - lp_yystos[lp_yystate], lp_yyvsp, parm, scanner); - YYPOPSTACK (1); - lp_yystate = *lp_yyssp; - YY_STACK_PRINT (lp_yyss, lp_yyssp); - } - - if (lp_yyn == YYFINAL) - YYACCEPT; - - *++lp_yyvsp = lp_yylval; - - - /* Shift the error token. */ - YY_SYMBOL_PRINT ("Shifting", lp_yystos[lp_yyn], lp_yyvsp, lp_yylsp); - - lp_yystate = lp_yyn; - goto lp_yynewstate; - - -/*-------------------------------------. -| lp_yyacceptlab -- YYACCEPT comes here. | -`-------------------------------------*/ -lp_yyacceptlab: - lp_yyresult = 0; - goto lp_yyreturn; - -/*-----------------------------------. -| lp_yyabortlab -- YYABORT comes here. | -`-----------------------------------*/ -lp_yyabortlab: - lp_yyresult = 1; - goto lp_yyreturn; - -#ifndef lp_yyoverflow -/*-------------------------------------------------. -| lp_yyexhaustedlab -- memory exhaustion comes here. | -`-------------------------------------------------*/ -lp_yyexhaustedlab: - lp_yyerror (parm, scanner, YY_("memory exhausted")); - lp_yyresult = 2; - /* Fall through. */ -#endif - -lp_yyreturn: - if (lp_yychar != YYEOF && lp_yychar != YYEMPTY) - lp_yydestruct ("Cleanup: discarding lookahead", - lp_yytoken, &lp_yylval, parm, scanner); - /* Do not reclaim the symbols of the rule which action triggered - this YYABORT or YYACCEPT. */ - YYPOPSTACK (lp_yylen); - YY_STACK_PRINT (lp_yyss, lp_yyssp); - while (lp_yyssp != lp_yyss) - { - lp_yydestruct ("Cleanup: popping", - lp_yystos[*lp_yyssp], lp_yyvsp, parm, scanner); - YYPOPSTACK (1); - } -#ifndef lp_yyoverflow - if (lp_yyss != lp_yyssa) - YYSTACK_FREE (lp_yyss); -#endif -#if YYERROR_VERBOSE - if (lp_yymsg != lp_yymsgbuf) - YYSTACK_FREE (lp_yymsg); -#endif - /* Make sure YYID is used. */ - return YYID (lp_yyresult); -} - - - - - -static void lp_yy_delete_allocated_memory(parse_parm *pp) -{ - parse_vars *pv = (parse_vars *) pp->parse_vars; - /* free memory allocated by flex. Otherwise some memory is not freed. - This is a bit tricky. There is not much documentation about this, but a lot of - reports of memory that keeps allocated */ - - /* If you get errors on this function call, just comment it. This will only result - in some memory that is not being freed. */ - -# if defined YY_CURRENT_BUFFER - /* flex defines the macro YY_CURRENT_BUFFER, so you should only get here if lp_rlp.h is - generated by flex */ - /* lex doesn't define this macro and thus should not come here, but lex doesn't has - this memory leak also ...*/ - -# if 0 - /* older versions of flex */ - lp_yy_delete_buffer(YY_CURRENT_BUFFER); /* comment this line if you have problems with it */ - lp_yy_init = 1; /* make sure that the next time memory is allocated again */ - lp_yy_start = 0; -# else - /* As of version 2.5.9 Flex */ - lp_yylex_destroy(pp->scanner); /* comment this line if you have problems with it */ -# endif -# endif - - FREE(pv->Last_var); - FREE(pv->Last_var0); -} - -static int parse(parse_parm *pp) -{ - return(lp_yyparse(pp, pp->scanner)); -} - -lprec *read_lp1(lprec *lp, void *userhandle, read_modeldata_func read_modeldata, int verbose, char *lp_name) -{ - parse_vars *pv; - lprec *lp1 = NULL; - - CALLOC(pv, 1, parse_vars); - if (pv != NULL) { - parse_parm pp; - - memset(&pp, 0, sizeof(pp)); - pp.parse_vars = (void *) pv; - - lp_yylex_init(&pp.scanner); - lp_yyset_extra(&pp, pp.scanner); - - lp_yyset_in((FILE *) userhandle, pp.scanner); - lp_yyset_out(NULL, pp.scanner); - pv->lp_input = read_modeldata; - pv->userhandle = userhandle; - lp1 = yacc_read(lp, verbose, lp_name, parse, &pp, lp_yy_delete_allocated_memory); - FREE(pv); - } - return(lp1); -} - -lprec * __WINAPI read_lp(FILE *filename, int verbose, char *lp_name) -{ - return(read_lp1(NULL, filename, lp_input_lp_yyin, verbose, lp_name)); -} - -lprec * __WINAPI read_lpex(void *userhandle, read_modeldata_func read_modeldata, int verbose, char *lp_name) -{ - return(read_lp1(NULL, userhandle, read_modeldata, verbose, lp_name)); -} - -lprec *read_LP1(lprec *lp, char *filename, int verbose, char *lp_name) -{ - FILE *fpin; - - if((fpin = fopen(filename, "r")) != NULL) { - lp = read_lp1(lp, fpin, lp_input_lp_yyin, verbose, lp_name); - fclose(fpin); - } - else - lp = NULL; - return(lp); -} - -lprec * __WINAPI read_LP(char *filename, int verbose, char *lp_name) -{ - return(read_LP1(NULL, filename, verbose, lp_name)); -} - -MYBOOL __WINAPI LP_readhandle(lprec **lp, FILE *filename, int verbose, char *lp_name) -{ - if(lp != NULL) - *lp = read_lp1(*lp, filename, lp_input_lp_yyin, verbose, lp_name); - - return((lp != NULL) && (*lp != NULL)); -} - diff --git a/code/3rd_lpsolve/lp_rlp.h b/code/3rd_lpsolve/lp_rlp.h deleted file mode 100644 index 76c74658..00000000 --- a/code/3rd_lpsolve/lp_rlp.h +++ /dev/null @@ -1,2453 +0,0 @@ - - - -#define YY_INT_ALIGNED short int - -/* A lexical scanner generated by flex */ - -#define FLEX_SCANNER -#define YY_FLEX_MAJOR_VERSION 2 -#define YY_FLEX_MINOR_VERSION 5 -#define YY_FLEX_SUBMINOR_VERSION 35 -#if YY_FLEX_SUBMINOR_VERSION > 0 -#define FLEX_BETA -#endif - -/* First, we deal with platform-specific or compiler-specific issues. */ - -/* begin standard C headers. */ -#include -#include -#include -#include - -/* end standard C headers. */ - -/* flex integer type definitions */ - -#ifndef FLEXINT_H -#define FLEXINT_H - -/* C99 systems have . Non-C99 systems may or may not. */ - -#if defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L - -/* C99 says to define __STDC_LIMIT_MACROS before including stdint.h, - * if you want the limit (max/min) macros for int types. - */ -#ifndef __STDC_LIMIT_MACROS -#define __STDC_LIMIT_MACROS 1 -#endif - -#include -typedef int8_t flex_int8_t; -typedef uint8_t flex_uint8_t; -typedef int16_t flex_int16_t; -typedef uint16_t flex_uint16_t; -typedef int32_t flex_int32_t; -typedef uint32_t flex_uint32_t; -#else -typedef signed char flex_int8_t; -typedef short int flex_int16_t; -typedef int flex_int32_t; -typedef unsigned char flex_uint8_t; -typedef unsigned short int flex_uint16_t; -typedef unsigned int flex_uint32_t; -#endif /* ! C99 */ - -/* Limits of integral types. */ -#ifndef INT8_MIN -#define INT8_MIN (-128) -#endif -#ifndef INT16_MIN -#define INT16_MIN (-32767-1) -#endif -#ifndef INT32_MIN -#define INT32_MIN (-2147483647-1) -#endif -#ifndef INT8_MAX -#define INT8_MAX (127) -#endif -#ifndef INT16_MAX -#define INT16_MAX (32767) -#endif -#ifndef INT32_MAX -#define INT32_MAX (2147483647) -#endif -#ifndef UINT8_MAX -#define UINT8_MAX (255U) -#endif -#ifndef UINT16_MAX -#define UINT16_MAX (65535U) -#endif -#ifndef UINT32_MAX -#define UINT32_MAX (4294967295U) -#endif - -#endif /* ! FLEXINT_H */ - -#ifdef __cplusplus - -/* The "const" storage-class-modifier is valid. */ -#define YY_USE_CONST - -#else /* ! __cplusplus */ - -/* C99 requires __STDC__ to be defined as 1. */ -#if defined (__STDC__) - -#define YY_USE_CONST - -#endif /* defined (__STDC__) */ -#endif /* ! __cplusplus */ - -#ifdef YY_USE_CONST -#define lp_yyconst const -#else -#define lp_yyconst -#endif - -/* Returned upon end-of-file. */ -#define YY_NULL 0 - -/* Promotes a possibly negative, possibly signed char to an unsigned - * integer for use as an array index. If the signed char is negative, - * we want to instead treat it as an 8-bit unsigned char, hence the - * double cast. - */ -#define YY_SC_TO_UI(c) ((unsigned int) (unsigned char) c) - -/* An opaque pointer. */ -#ifndef YY_TYPEDEF_YY_SCANNER_T -#define YY_TYPEDEF_YY_SCANNER_T -typedef void* lp_yyscan_t; -#endif - -/* For convenience, these vars (plus the bison vars far below) - are macros in the reentrant scanner. */ -#define lp_yyin lp_yyg->lp_yyin_r -#define lp_yyout lp_yyg->lp_yyout_r -#define lp_yyextra lp_yyg->lp_yyextra_r -#define lp_yyleng lp_yyg->lp_yyleng_r -#define lp_yytext lp_yyg->lp_yytext_r -#define lp_yylineno (YY_CURRENT_BUFFER_LVALUE->lp_yy_bs_lineno) -#define lp_yycolumn (YY_CURRENT_BUFFER_LVALUE->lp_yy_bs_column) -#define lp_yy_flex_debug lp_yyg->lp_yy_flex_debug_r - -/* Enter a start condition. This macro really ought to take a parameter, - * but we do it the disgusting crufty way forced on us by the ()-less - * definition of BEGIN. - */ -#define BEGIN lp_yyg->lp_yy_start = 1 + 2 * - -/* Translate the current start state into a value that can be later handed - * to BEGIN to return to the state. The YYSTATE alias is for lex - * compatibility. - */ -#define YY_START ((lp_yyg->lp_yy_start - 1) / 2) -#define YYSTATE YY_START - -/* Action number for EOF rule of a given start state. */ -#define YY_STATE_EOF(state) (YY_END_OF_BUFFER + state + 1) - -/* Special action meaning "start processing a new file". */ -#define YY_NEW_FILE lp_yyrestart(lp_yyin ,lp_yyscanner ) - -#define YY_END_OF_BUFFER_CHAR 0 - -/* Size of default input buffer. */ -#ifndef YY_BUF_SIZE -#define YY_BUF_SIZE 16384 -#endif - -/* The state buf must be large enough to hold one state per character in the main buffer. - */ -#define YY_STATE_BUF_SIZE ((YY_BUF_SIZE + 2) * sizeof(lp_yy_state_type)) - -#ifndef YY_TYPEDEF_YY_BUFFER_STATE -#define YY_TYPEDEF_YY_BUFFER_STATE -typedef struct lp_yy_buffer_state *YY_BUFFER_STATE; -#endif - -#define EOB_ACT_CONTINUE_SCAN 0 -#define EOB_ACT_END_OF_FILE 1 -#define EOB_ACT_LAST_MATCH 2 - - /* Note: We specifically omit the test for lp_yy_rule_can_match_eol because it requires - * access to the local variable lp_yy_act. Since lp_yyless() is a macro, it would break - * existing scanners that call lp_yyless() from OUTSIDE lp_yylex. - * One obvious solution it to make lp_yy_act a global. I tried that, and saw - * a 5% performance hit in a non-lp_yylineno scanner, because lp_yy_act is - * normally declared as a register variable-- so it is not worth it. - */ - #define YY_LESS_LINENO(n) \ - do { \ - int lp_yyl;\ - for ( lp_yyl = n; lp_yyl < lp_yyleng; ++lp_yyl )\ - if ( lp_yytext[lp_yyl] == '\n' )\ - --lp_yylineno;\ - }while(0) - -/* Return all but the first "n" matched characters back to the input stream. */ -#define lp_yyless(n) \ - do \ - { \ - /* Undo effects of setting up lp_yytext. */ \ - int lp_yyless_macro_arg = (n); \ - YY_LESS_LINENO(lp_yyless_macro_arg);\ - *lp_yy_cp = lp_yyg->lp_yy_hold_char; \ - YY_RESTORE_YY_MORE_OFFSET \ - lp_yyg->lp_yy_c_buf_p = lp_yy_cp = lp_yy_bp + lp_yyless_macro_arg - YY_MORE_ADJ; \ - YY_DO_BEFORE_ACTION; /* set up lp_yytext again */ \ - } \ - while ( 0 ) - -#define unput(c) lp_yyunput( c, lp_yyg->lp_yytext_ptr , lp_yyscanner ) - -#ifndef YY_TYPEDEF_YY_SIZE_T -#define YY_TYPEDEF_YY_SIZE_T -typedef size_t lp_yy_size_t; -#endif - -#ifndef YY_STRUCT_YY_BUFFER_STATE -#define YY_STRUCT_YY_BUFFER_STATE -struct lp_yy_buffer_state - { - FILE *lp_yy_input_file; - - char *lp_yy_ch_buf; /* input buffer */ - char *lp_yy_buf_pos; /* current position in input buffer */ - - /* Size of input buffer in bytes, not including room for EOB - * characters. - */ - lp_yy_size_t lp_yy_buf_size; - - /* Number of characters read into lp_yy_ch_buf, not including EOB - * characters. - */ - int lp_yy_n_chars; - - /* Whether we "own" the buffer - i.e., we know we created it, - * and can realloc() it to grow it, and should free() it to - * delete it. - */ - int lp_yy_is_our_buffer; - - /* Whether this is an "interactive" input source; if so, and - * if we're using stdio for input, then we want to use getc() - * instead of fread(), to make sure we stop fetching input after - * each newline. - */ - int lp_yy_is_interactive; - - /* Whether we're considered to be at the beginning of a line. - * If so, '^' rules will be active on the next match, otherwise - * not. - */ - int lp_yy_at_bol; - - int lp_yy_bs_lineno; /**< The line count. */ - int lp_yy_bs_column; /**< The column count. */ - - /* Whether to try to fill the input buffer when we reach the - * end of it. - */ - int lp_yy_fill_buffer; - - int lp_yy_buffer_status; - -#define YY_BUFFER_NEW 0 -#define YY_BUFFER_NORMAL 1 - /* When an EOF's been seen but there's still some text to process - * then we mark the buffer as YY_EOF_PENDING, to indicate that we - * shouldn't try reading from the input source any more. We might - * still have a bunch of tokens to match, though, because of - * possible backing-up. - * - * When we actually see the EOF, we change the status to "new" - * (via lp_yyrestart()), so that the user can continue scanning by - * just pointing lp_yyin at a new input file. - */ -#define YY_BUFFER_EOF_PENDING 2 - - }; -#endif /* !YY_STRUCT_YY_BUFFER_STATE */ - -/* We provide macros for accessing buffer states in case in the - * future we want to put the buffer states in a more general - * "scanner state". - * - * Returns the top of the stack, or NULL. - */ -#define YY_CURRENT_BUFFER ( lp_yyg->lp_yy_buffer_stack \ - ? lp_yyg->lp_yy_buffer_stack[lp_yyg->lp_yy_buffer_stack_top] \ - : NULL) - -/* Same as previous macro, but useful when we know that the buffer stack is not - * NULL or when we need an lvalue. For internal use only. - */ -#define YY_CURRENT_BUFFER_LVALUE lp_yyg->lp_yy_buffer_stack[lp_yyg->lp_yy_buffer_stack_top] - -void lp_yyrestart (FILE *input_file ,lp_yyscan_t lp_yyscanner ); -void lp_yy_switch_to_buffer (YY_BUFFER_STATE new_buffer ,lp_yyscan_t lp_yyscanner ); -YY_BUFFER_STATE lp_yy_create_buffer (FILE *file,int size ,lp_yyscan_t lp_yyscanner ); -void lp_yy_delete_buffer (YY_BUFFER_STATE b ,lp_yyscan_t lp_yyscanner ); -void lp_yy_flush_buffer (YY_BUFFER_STATE b ,lp_yyscan_t lp_yyscanner ); -void lp_yypush_buffer_state (YY_BUFFER_STATE new_buffer ,lp_yyscan_t lp_yyscanner ); -void lp_yypop_buffer_state (lp_yyscan_t lp_yyscanner ); - -static void lp_yyensure_buffer_stack (lp_yyscan_t lp_yyscanner ); -static void lp_yy_load_buffer_state (lp_yyscan_t lp_yyscanner ); -static void lp_yy_init_buffer (YY_BUFFER_STATE b,FILE *file ,lp_yyscan_t lp_yyscanner ); - -#define YY_FLUSH_BUFFER lp_yy_flush_buffer(YY_CURRENT_BUFFER ,lp_yyscanner) - -YY_BUFFER_STATE lp_yy_scan_buffer (char *base,lp_yy_size_t size ,lp_yyscan_t lp_yyscanner ); -YY_BUFFER_STATE lp_yy_scan_string (lp_yyconst char *lp_yy_str ,lp_yyscan_t lp_yyscanner ); -YY_BUFFER_STATE lp_yy_scan_bytes (lp_yyconst char *bytes,int len ,lp_yyscan_t lp_yyscanner ); - -void *lp_yyalloc (lp_yy_size_t ,lp_yyscan_t lp_yyscanner ); -void *lp_yyrealloc (void *,lp_yy_size_t ,lp_yyscan_t lp_yyscanner ); -void lp_yyfree (void * ,lp_yyscan_t lp_yyscanner ); - -#define lp_yy_new_buffer lp_yy_create_buffer - -#define lp_yy_set_interactive(is_interactive) \ - { \ - if ( ! YY_CURRENT_BUFFER ){ \ - lp_yyensure_buffer_stack (lp_yyscanner); \ - YY_CURRENT_BUFFER_LVALUE = \ - lp_yy_create_buffer(lp_yyin,YY_BUF_SIZE ,lp_yyscanner); \ - } \ - YY_CURRENT_BUFFER_LVALUE->lp_yy_is_interactive = is_interactive; \ - } - -#define lp_yy_set_bol(at_bol) \ - { \ - if ( ! YY_CURRENT_BUFFER ){\ - lp_yyensure_buffer_stack (lp_yyscanner); \ - YY_CURRENT_BUFFER_LVALUE = \ - lp_yy_create_buffer(lp_yyin,YY_BUF_SIZE ,lp_yyscanner); \ - } \ - YY_CURRENT_BUFFER_LVALUE->lp_yy_at_bol = at_bol; \ - } - -#define YY_AT_BOL() (YY_CURRENT_BUFFER_LVALUE->lp_yy_at_bol) - -/* Begin user sect3 */ - -#define lp_yywrap(n) 1 -#define YY_SKIP_YYWRAP - -typedef unsigned char YY_CHAR; - -typedef int lp_yy_state_type; - -#define lp_yytext_ptr lp_yytext_r - -static lp_yy_state_type lp_yy_get_previous_state (lp_yyscan_t lp_yyscanner ); -static lp_yy_state_type lp_yy_try_NUL_trans (lp_yy_state_type current_state ,lp_yyscan_t lp_yyscanner); -static int lp_yy_get_next_buffer (lp_yyscan_t lp_yyscanner ); -static void lp_yy_fatal_error (lp_yyconst char msg[] ,lp_yyscan_t lp_yyscanner ); - -/* Done after the current pattern has been matched and before the - * corresponding action - sets up lp_yytext. - */ -#define YY_DO_BEFORE_ACTION \ - lp_yyg->lp_yytext_ptr = lp_yy_bp; \ - lp_yyleng = (size_t) (lp_yy_cp - lp_yy_bp); \ - lp_yyg->lp_yy_hold_char = *lp_yy_cp; \ - *lp_yy_cp = '\0'; \ - lp_yyg->lp_yy_c_buf_p = lp_yy_cp; - -#define YY_NUM_RULES 33 -#define YY_END_OF_BUFFER 34 -/* This struct is not used in this scanner, - but its presence is necessary. */ -struct lp_yy_trans_info - { - flex_int32_t lp_yy_verify; - flex_int32_t lp_yy_nxt; - }; -static lp_yyconst flex_int16_t lp_yy_accept[144] = - { 0, - 0, 0, 0, 0, 0, 0, 34, 32, 10, 10, - 27, 17, 11, 32, 32, 14, 26, 31, 29, 28, - 30, 25, 25, 10, 25, 25, 25, 25, 3, 4, - 3, 3, 9, 7, 8, 10, 17, 17, 0, 15, - 1, 6, 15, 14, 0, 29, 30, 0, 25, 24, - 0, 25, 25, 10, 0, 0, 0, 0, 25, 25, - 25, 25, 25, 2, 0, 15, 0, 15, 22, 0, - 25, 25, 0, 0, 0, 0, 0, 19, 25, 18, - 20, 25, 25, 21, 0, 25, 0, 13, 25, 0, - 12, 25, 19, 0, 18, 20, 21, 25, 23, 25, - - 20, 21, 21, 16, 16, 0, 25, 25, 0, 23, - 0, 21, 25, 25, 0, 0, 25, 25, 0, 0, - 19, 25, 0, 0, 25, 25, 19, 0, 18, 0, - 0, 25, 25, 18, 0, 0, 0, 0, 0, 0, - 0, 0, 0 - } ; - -static lp_yyconst flex_int32_t lp_yy_ec[256] = - { 0, - 1, 1, 1, 1, 1, 1, 1, 1, 2, 3, - 1, 1, 4, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 2, 1, 1, 5, 6, 5, 5, 5, 1, - 1, 7, 8, 9, 10, 11, 12, 13, 14, 14, - 13, 13, 13, 13, 13, 13, 13, 15, 16, 17, - 18, 19, 1, 5, 20, 21, 22, 23, 24, 25, - 26, 23, 27, 23, 23, 23, 28, 29, 30, 23, - 23, 31, 32, 33, 34, 23, 23, 35, 36, 37, - 5, 1, 5, 5, 5, 1, 20, 21, 22, 23, - - 24, 25, 26, 23, 27, 23, 23, 23, 28, 29, - 30, 23, 23, 31, 32, 33, 34, 23, 23, 35, - 36, 37, 5, 1, 5, 5, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1 - } ; - -static lp_yyconst flex_int32_t lp_yy_meta[38] = - { 0, - 1, 2, 3, 3, 4, 5, 6, 3, 6, 3, - 5, 5, 5, 5, 7, 6, 7, 6, 6, 4, - 4, 4, 4, 4, 4, 4, 8, 4, 4, 4, - 4, 4, 4, 4, 4, 4, 4 - } ; - -static lp_yyconst flex_int16_t lp_yy_base[150] = - { 0, - 0, 36, 36, 38, 43, 45, 366, 388, 48, 62, - 388, 338, 388, 40, 48, 60, 388, 388, 346, 388, - 326, 60, 65, 91, 81, 74, 85, 102, 388, 388, - 388, 330, 388, 388, 388, 125, 313, 134, 308, 96, - 388, 388, 117, 132, 139, 388, 388, 88, 146, 320, - 0, 149, 152, 0, 307, 301, 294, 83, 153, 156, - 157, 189, 160, 388, 286, 126, 65, 108, 388, 289, - 181, 185, 272, 273, 250, 249, 220, 199, 203, 208, - 192, 211, 219, 227, 243, 109, 163, 225, 202, 174, - 224, 215, 213, 207, 191, 388, 189, 227, 228, 231, - - 244, 240, 253, 276, 388, 170, 260, 262, 166, 388, - 169, 179, 263, 241, 166, 159, 270, 272, 149, 155, - 284, 288, 130, 124, 296, 303, 388, 103, 300, 96, - 45, 324, 328, 388, 82, 311, 79, 68, 54, 56, - 25, 12, 388, 345, 353, 360, 367, 372, 379 - } ; - -static lp_yyconst flex_int16_t lp_yy_def[150] = - { 0, - 143, 1, 144, 144, 145, 145, 143, 143, 143, 143, - 143, 146, 143, 143, 143, 143, 143, 143, 143, 143, - 143, 147, 147, 143, 147, 147, 147, 147, 143, 143, - 143, 143, 143, 143, 143, 143, 146, 143, 143, 143, - 143, 143, 143, 143, 143, 143, 143, 143, 147, 143, - 148, 147, 147, 24, 143, 143, 143, 143, 147, 147, - 147, 147, 147, 143, 143, 143, 143, 143, 143, 148, - 147, 147, 143, 143, 143, 143, 143, 147, 147, 147, - 147, 147, 147, 147, 149, 143, 143, 143, 62, 143, - 143, 62, 143, 143, 143, 143, 143, 62, 62, 62, - - 62, 62, 62, 143, 143, 143, 62, 62, 143, 143, - 143, 143, 62, 62, 143, 143, 62, 62, 143, 143, - 62, 62, 143, 143, 62, 62, 143, 143, 62, 143, - 143, 147, 147, 143, 143, 149, 143, 143, 143, 143, - 143, 143, 0, 143, 143, 143, 143, 143, 143 - } ; - -static lp_yyconst flex_int16_t lp_yy_nxt[426] = - { 0, - 8, 9, 10, 9, 8, 8, 11, 12, 13, 12, - 14, 15, 16, 16, 17, 18, 19, 20, 21, 22, - 22, 22, 22, 22, 22, 22, 22, 23, 22, 22, - 22, 22, 22, 22, 22, 22, 22, 24, 30, 31, - 30, 31, 32, 96, 32, 34, 35, 34, 35, 36, - 36, 36, 40, 40, 41, 37, 25, 37, 142, 42, - 26, 48, 27, 36, 36, 36, 48, 28, 136, 37, - 43, 37, 44, 44, 50, 48, 51, 68, 68, 50, - 136, 51, 48, 45, 52, 141, 48, 140, 50, 48, - 51, 53, 54, 36, 36, 50, 139, 51, 37, 50, - - 37, 51, 50, 48, 60, 138, 76, 59, 40, 40, - 48, 55, 77, 61, 137, 56, 50, 57, 51, 45, - 68, 68, 58, 50, 135, 62, 36, 36, 36, 66, - 66, 63, 37, 134, 37, 38, 38, 38, 66, 66, - 45, 38, 43, 38, 44, 44, 67, 48, 67, 45, - 48, 68, 68, 48, 48, 45, 131, 48, 48, 130, - 50, 48, 51, 50, 87, 51, 50, 50, 51, 51, - 50, 50, 51, 51, 50, 90, 51, 88, 128, 79, - 72, 78, 87, 71, 127, 124, 90, 123, 91, 80, - 48, 84, 112, 48, 120, 88, 119, 51, 116, 91, - - 48, 51, 112, 50, 48, 51, 50, 89, 51, 48, - 81, 92, 48, 50, 111, 51, 82, 50, 98, 51, - 48, 83, 50, 49, 51, 50, 99, 51, 48, 107, - 110, 100, 109, 50, 49, 51, 49, 101, 69, 69, - 103, 50, 108, 51, 104, 104, 104, 49, 49, 49, - 102, 97, 49, 115, 49, 49, 114, 113, 49, 49, - 49, 49, 49, 49, 122, 49, 103, 49, 49, 106, - 96, 49, 49, 49, 49, 81, 49, 104, 104, 104, - 49, 49, 95, 49, 49, 49, 117, 49, 118, 49, - 49, 49, 49, 49, 49, 49, 94, 49, 121, 49, - - 93, 125, 49, 126, 49, 49, 125, 86, 126, 49, - 85, 49, 104, 104, 104, 49, 49, 49, 129, 132, - 49, 49, 75, 49, 49, 87, 133, 49, 49, 90, - 49, 74, 49, 73, 69, 49, 65, 143, 88, 39, - 51, 64, 91, 47, 51, 29, 29, 29, 29, 29, - 29, 29, 29, 33, 33, 33, 33, 33, 33, 33, - 33, 38, 38, 46, 39, 143, 143, 38, 49, 143, - 49, 49, 143, 49, 49, 70, 70, 143, 143, 70, - 105, 105, 143, 105, 105, 105, 105, 7, 143, 143, - 143, 143, 143, 143, 143, 143, 143, 143, 143, 143, - - 143, 143, 143, 143, 143, 143, 143, 143, 143, 143, - 143, 143, 143, 143, 143, 143, 143, 143, 143, 143, - 143, 143, 143, 143, 143 - } ; - -static lp_yyconst flex_int16_t lp_yy_chk[426] = - { 0, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 2, 3, 3, - 4, 4, 3, 142, 4, 5, 5, 6, 6, 9, - 9, 9, 14, 14, 15, 9, 2, 9, 141, 15, - 2, 22, 2, 10, 10, 10, 23, 2, 131, 10, - 16, 10, 16, 16, 22, 26, 22, 67, 67, 23, - 131, 23, 25, 16, 23, 140, 27, 139, 26, 48, - 26, 23, 24, 24, 24, 25, 138, 25, 24, 27, - - 24, 27, 48, 28, 26, 137, 58, 25, 40, 40, - 86, 24, 58, 27, 135, 24, 28, 24, 28, 40, - 68, 68, 24, 86, 130, 28, 36, 36, 36, 43, - 43, 28, 36, 128, 36, 38, 38, 38, 66, 66, - 43, 38, 44, 38, 44, 44, 45, 49, 45, 66, - 52, 45, 45, 53, 59, 44, 124, 60, 61, 123, - 49, 63, 49, 52, 87, 52, 53, 59, 53, 59, - 60, 61, 60, 61, 63, 90, 63, 87, 120, 60, - 53, 59, 71, 52, 119, 116, 72, 115, 90, 61, - 62, 63, 112, 81, 111, 71, 109, 71, 106, 72, - - 78, 72, 97, 62, 79, 62, 81, 71, 81, 80, - 62, 72, 82, 78, 95, 78, 62, 79, 78, 79, - 83, 62, 80, 89, 80, 82, 79, 82, 84, 89, - 94, 80, 93, 83, 89, 83, 92, 82, 91, 88, - 84, 84, 92, 84, 85, 85, 85, 92, 98, 99, - 83, 77, 100, 101, 98, 99, 100, 98, 100, 98, - 99, 102, 114, 100, 114, 101, 103, 102, 114, 85, - 76, 101, 102, 114, 103, 101, 101, 104, 104, 104, - 103, 107, 75, 108, 113, 103, 107, 107, 108, 108, - 113, 117, 107, 118, 108, 113, 74, 117, 113, 118, - - 73, 117, 117, 118, 118, 121, 117, 70, 118, 122, - 65, 121, 136, 136, 136, 122, 121, 125, 122, 125, - 122, 129, 57, 125, 126, 132, 126, 129, 125, 133, - 126, 56, 129, 55, 50, 126, 39, 136, 132, 37, - 132, 32, 133, 21, 133, 144, 144, 144, 144, 144, - 144, 144, 144, 145, 145, 145, 145, 145, 145, 145, - 145, 146, 146, 19, 12, 7, 0, 146, 147, 0, - 147, 147, 0, 147, 147, 148, 148, 0, 0, 148, - 149, 149, 0, 149, 149, 149, 149, 143, 143, 143, - 143, 143, 143, 143, 143, 143, 143, 143, 143, 143, - - 143, 143, 143, 143, 143, 143, 143, 143, 143, 143, - 143, 143, 143, 143, 143, 143, 143, 143, 143, 143, - 143, 143, 143, 143, 143 - } ; - -/* Table of booleans, true if rule could match eol. */ -static lp_yyconst flex_int32_t lp_yy_rule_can_match_eol[34] = - { 0, -0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 1, 1, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, }; - -/* The intent behind this definition is that it'll catch - * any uses of REJECT which flex missed. - */ -#define REJECT reject_used_but_not_detected -#define lp_yymore() lp_yymore_used_but_not_detected -#define YY_MORE_ADJ 0 -#define YY_RESTORE_YY_MORE_OFFSET -/* - made reentrant with help of - http://www.usualcoding.eu/post/2007/09/03/Building-a-reentrant-parser-in-C-with-Flex/Bison -*/ -/* - Note that a minimum version of flex is needed to be able to compile this. - Older version don't know the reentrant code. - Version 2.5.4 is not enough. Probably at least v2.5.31 is needed. Tested with v2.5.35 -*/ -/* -** We want the scanner to be reentrant, therefore generate no global variables. -** That what the 'reentrant' option is for. -** 'bison-bridge' is used to create a bison compatible scanner and share lp_yylval -*/ - -#define INITIAL 0 -#define COMMENT 1 -#define LINECOMMENT 2 - -#ifndef YY_NO_UNISTD_H -/* Special case for "unistd.h", since it is non-ANSI. We include it way - * down here because we want the user's section 1 to have been scanned first. - * The user has a chance to override it with an option. - */ -#include -#endif - -#ifndef YY_EXTRA_TYPE -#define YY_EXTRA_TYPE void * -#endif - -/* Holds the entire state of the reentrant scanner. */ -struct lp_yyguts_t - { - - /* User-defined. Not touched by flex. */ - YY_EXTRA_TYPE lp_yyextra_r; - - /* The rest are the same as the globals declared in the non-reentrant scanner. */ - FILE *lp_yyin_r, *lp_yyout_r; - size_t lp_yy_buffer_stack_top; /**< index of top of stack. */ - size_t lp_yy_buffer_stack_max; /**< capacity of stack. */ - YY_BUFFER_STATE * lp_yy_buffer_stack; /**< Stack as an array. */ - char lp_yy_hold_char; - int lp_yy_n_chars; - int lp_yyleng_r; - char *lp_yy_c_buf_p; - int lp_yy_init; - int lp_yy_start; - int lp_yy_did_buffer_switch_on_eof; - int lp_yy_start_stack_ptr; - int lp_yy_start_stack_depth; - int *lp_yy_start_stack; - lp_yy_state_type lp_yy_last_accepting_state; - char* lp_yy_last_accepting_cpos; - - int lp_yylineno_r; - int lp_yy_flex_debug_r; - - char *lp_yytext_r; - int lp_yy_more_flag; - int lp_yy_more_len; - - YYSTYPE * lp_yylval_r; - - }; /* end struct lp_yyguts_t */ - -static int lp_yy_init_globals (lp_yyscan_t lp_yyscanner ); - - /* This must go here because YYSTYPE and YYLTYPE are included - * from bison output in section 1.*/ - # define lp_yylval lp_yyg->lp_yylval_r - -int lp_yylex_init (lp_yyscan_t* scanner); - -int lp_yylex_init_extra (YY_EXTRA_TYPE user_defined,lp_yyscan_t* scanner); - -/* Accessor methods to globals. - These are made visible to non-reentrant scanners for convenience. */ - -int lp_yylex_destroy (lp_yyscan_t lp_yyscanner ); - -int lp_yyget_debug (lp_yyscan_t lp_yyscanner ); - -void lp_yyset_debug (int debug_flag ,lp_yyscan_t lp_yyscanner ); - -YY_EXTRA_TYPE lp_yyget_extra (lp_yyscan_t lp_yyscanner ); - -void lp_yyset_extra (YY_EXTRA_TYPE user_defined ,lp_yyscan_t lp_yyscanner ); - -FILE *lp_yyget_in (lp_yyscan_t lp_yyscanner ); - -void lp_yyset_in (FILE * in_str ,lp_yyscan_t lp_yyscanner ); - -FILE *lp_yyget_out (lp_yyscan_t lp_yyscanner ); - -void lp_yyset_out (FILE * out_str ,lp_yyscan_t lp_yyscanner ); - -int lp_yyget_leng (lp_yyscan_t lp_yyscanner ); - -char *lp_yyget_text (lp_yyscan_t lp_yyscanner ); - -int lp_yyget_lineno (lp_yyscan_t lp_yyscanner ); - -void lp_yyset_lineno (int line_number ,lp_yyscan_t lp_yyscanner ); - -YYSTYPE * lp_yyget_lval (lp_yyscan_t lp_yyscanner ); - -void lp_yyset_lval (YYSTYPE * lp_yylval_param ,lp_yyscan_t lp_yyscanner ); - -/* Macros after this point can all be overridden by user definitions in - * section 1. - */ - -#ifndef YY_SKIP_YYWRAP -#ifdef __cplusplus -extern "C" int lp_yywrap (lp_yyscan_t lp_yyscanner ); -#else -extern int lp_yywrap (lp_yyscan_t lp_yyscanner ); -#endif -#endif - - static void lp_yyunput (int c,char *buf_ptr ,lp_yyscan_t lp_yyscanner); - -#ifndef lp_yytext_ptr -static void lp_yy_flex_strncpy (char *,lp_yyconst char *,int ,lp_yyscan_t lp_yyscanner); -#endif - -#ifdef YY_NEED_STRLEN -static int lp_yy_flex_strlen (lp_yyconst char * ,lp_yyscan_t lp_yyscanner); -#endif - -#ifndef YY_NO_INPUT - -#ifdef __cplusplus -static int lp_yyinput (lp_yyscan_t lp_yyscanner ); -#else -static int input (lp_yyscan_t lp_yyscanner ); -#endif - -#endif - -/* Amount of stuff to slurp up with each read. */ -#ifndef YY_READ_BUF_SIZE -#define YY_READ_BUF_SIZE 8192 -#endif - -/* Copy whatever the last rule matched to the standard output. */ -#ifndef ECHO -/* This used to be an fputs(), but since the string might contain NUL's, - * we now use fwrite(). - */ -#define ECHO fwrite( lp_yytext, lp_yyleng, 1, lp_yyout ) -#endif - -/* Gets input and stuffs it into "buf". number of characters read, or YY_NULL, - * is returned in "result". - */ -#ifndef YY_INPUT -#define YY_INPUT(buf,result,max_size) \ - if ( YY_CURRENT_BUFFER_LVALUE->lp_yy_is_interactive ) \ - { \ - int c = '*'; \ - int n; \ - for ( n = 0; n < max_size && \ - (c = getc( lp_yyin )) != EOF && c != '\n'; ++n ) \ - buf[n] = (char) c; \ - if ( c == '\n' ) \ - buf[n++] = (char) c; \ - if ( c == EOF && ferror( lp_yyin ) ) \ - YY_FATAL_ERROR( "input in flex scanner failed" ); \ - result = n; \ - } \ - else \ - { \ - errno=0; \ - while ( (result = fread(buf, 1, max_size, lp_yyin))==0 && ferror(lp_yyin)) \ - { \ - if( errno != EINTR) \ - { \ - YY_FATAL_ERROR( "input in flex scanner failed" ); \ - break; \ - } \ - errno=0; \ - clearerr(lp_yyin); \ - } \ - }\ -\ - -#endif - -/* No semi-colon after return; correct usage is to write "lp_yyterminate();" - - * we don't want an extra ';' after the "return" because that will cause - * some compilers to complain about unreachable statements. - */ -#ifndef lp_yyterminate -#define lp_yyterminate() return YY_NULL -#endif - -/* Number of entries by which start-condition stack grows. */ -#ifndef YY_START_STACK_INCR -#define YY_START_STACK_INCR 25 -#endif - -/* Report a fatal error. */ -#ifndef YY_FATAL_ERROR -#define YY_FATAL_ERROR(msg) lp_yy_fatal_error( msg , lp_yyscanner) -#endif - -/* end tables serialization structures and prototypes */ - -/* Default declaration of generated scanner - a define so the user can - * easily add parameters. - */ -#ifndef YY_DECL -#define YY_DECL_IS_OURS 1 - -extern int lp_yylex \ - (YYSTYPE * lp_yylval_param ,lp_yyscan_t lp_yyscanner); - -#define YY_DECL int lp_yylex \ - (YYSTYPE * lp_yylval_param , lp_yyscan_t lp_yyscanner) -#endif /* !YY_DECL */ - -/* Code executed at the beginning of each rule, after lp_yytext and lp_yyleng - * have been set up. - */ -#ifndef YY_USER_ACTION -#define YY_USER_ACTION -#endif - -/* Code executed at the end of each rule. */ -#ifndef YY_BREAK -#define YY_BREAK break; -#endif - -#define YY_RULE_SETUP \ - if ( lp_yyleng > 0 ) \ - YY_CURRENT_BUFFER_LVALUE->lp_yy_at_bol = \ - (lp_yytext[lp_yyleng - 1] == '\n'); \ - YY_USER_ACTION - -/** The main scanner function which does all the work. - */ -YY_DECL -{ - register lp_yy_state_type lp_yy_current_state; - register char *lp_yy_cp, *lp_yy_bp; - register int lp_yy_act; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - lp_yylval = lp_yylval_param; - - if ( !lp_yyg->lp_yy_init ) - { - lp_yyg->lp_yy_init = 1; - -#ifdef YY_USER_INIT - YY_USER_INIT; -#endif - - if ( ! lp_yyg->lp_yy_start ) - lp_yyg->lp_yy_start = 1; /* first start state */ - - if ( ! lp_yyin ) - lp_yyin = stdin; - - if ( ! lp_yyout ) - lp_yyout = stdout; - - if ( ! YY_CURRENT_BUFFER ) { - lp_yyensure_buffer_stack (lp_yyscanner); - YY_CURRENT_BUFFER_LVALUE = - lp_yy_create_buffer(lp_yyin,YY_BUF_SIZE ,lp_yyscanner); - } - - lp_yy_load_buffer_state(lp_yyscanner ); - } - - while ( 1 ) /* loops until end-of-file is reached */ - { - lp_yy_cp = lp_yyg->lp_yy_c_buf_p; - - /* Support of lp_yytext. */ - *lp_yy_cp = lp_yyg->lp_yy_hold_char; - - /* lp_yy_bp points to the position in lp_yy_ch_buf of the start of - * the current run. - */ - lp_yy_bp = lp_yy_cp; - - lp_yy_current_state = lp_yyg->lp_yy_start; - lp_yy_current_state += YY_AT_BOL(); -lp_yy_match: - do - { - register YY_CHAR lp_yy_c = lp_yy_ec[YY_SC_TO_UI(*lp_yy_cp)]; - if ( lp_yy_accept[lp_yy_current_state] ) - { - lp_yyg->lp_yy_last_accepting_state = lp_yy_current_state; - lp_yyg->lp_yy_last_accepting_cpos = lp_yy_cp; - } - while ( lp_yy_chk[lp_yy_base[lp_yy_current_state] + lp_yy_c] != lp_yy_current_state ) - { - lp_yy_current_state = (int) lp_yy_def[lp_yy_current_state]; - if ( lp_yy_current_state >= 144 ) - lp_yy_c = lp_yy_meta[(unsigned int) lp_yy_c]; - } - lp_yy_current_state = lp_yy_nxt[lp_yy_base[lp_yy_current_state] + (unsigned int) lp_yy_c]; - ++lp_yy_cp; - } - while ( lp_yy_base[lp_yy_current_state] != 388 ); - -lp_yy_find_action: - lp_yy_act = lp_yy_accept[lp_yy_current_state]; - if ( lp_yy_act == 0 ) - { /* have to back up */ - lp_yy_cp = lp_yyg->lp_yy_last_accepting_cpos; - lp_yy_current_state = lp_yyg->lp_yy_last_accepting_state; - lp_yy_act = lp_yy_accept[lp_yy_current_state]; - } - - YY_DO_BEFORE_ACTION; - - if ( lp_yy_act != YY_END_OF_BUFFER && lp_yy_rule_can_match_eol[lp_yy_act] ) - { - int lp_yyl; - for ( lp_yyl = 0; lp_yyl < lp_yyleng; ++lp_yyl ) - if ( lp_yytext[lp_yyl] == '\n' ) - - do{ lp_yylineno++; - lp_yycolumn=0; - }while(0) -; - } - -do_action: /* This label is used only to access EOF actions. */ - - switch ( lp_yy_act ) - { /* beginning of action switch */ - case 0: /* must back up */ - /* undo the effects of YY_DO_BEFORE_ACTION */ - *lp_yy_cp = lp_yyg->lp_yy_hold_char; - lp_yy_cp = lp_yyg->lp_yy_last_accepting_cpos; - lp_yy_current_state = lp_yyg->lp_yy_last_accepting_state; - goto lp_yy_find_action; - -case 1: -YY_RULE_SETUP -{ - BEGIN COMMENT; -} /* begin skip comment */ - YY_BREAK -case 2: -YY_RULE_SETUP -{ - BEGIN INITIAL; -} /* end skip comment */ - YY_BREAK -case 3: -YY_RULE_SETUP -{ -} - YY_BREAK -case 4: -/* rule 4 can match eol */ -YY_RULE_SETUP -{ -} - YY_BREAK -case 5: -YY_RULE_SETUP -{ -} - YY_BREAK -case 6: -YY_RULE_SETUP -{ - BEGIN LINECOMMENT; -} /* begin skip LINECOMMENT */ - YY_BREAK -case 7: -/* rule 7 can match eol */ -YY_RULE_SETUP -{ - BEGIN INITIAL; -} /* end skip LINECOMMENT */ - YY_BREAK -case 8: -YY_RULE_SETUP -{ - BEGIN INITIAL; -} /* end skip LINECOMMENT */ - YY_BREAK -case 9: -YY_RULE_SETUP -{ -} - YY_BREAK -case 10: -/* rule 10 can match eol */ -YY_RULE_SETUP -{ -} - YY_BREAK -case 11: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - - pp->lineno = lp_yylineno; - return(COMMA); -} - YY_BREAK -case 12: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - - pp->lineno = lp_yylineno; - return(MINIMISE); -} - YY_BREAK -case 13: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - - pp->lineno = lp_yylineno; - return(MAXIMISE); -} - YY_BREAK -case 14: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - pv->f = atof((char *)lp_yytext); - return(INTCONS); -} /* f contains the last float */ - YY_BREAK -case 15: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - pv->f = atof((char *)lp_yytext); - return(CONS); -} /* f contains the last float */ - YY_BREAK -case 16: -/* rule 16 can match eol */ -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - char *ptr, c; - - pp->lineno = lp_yylineno; - pv->f = DEF_INFINITE; - pv->Sign = 0; - ptr = (char *)lp_yytext; - while (isspace(*ptr)) ptr++; - if(*ptr == '-') - pv->Sign = 1; - if(lp_yyleng > 0) { - c = lp_yytext[lp_yyleng - 1]; - if(!isalnum(c)) - unput(c); - } - return(INF); -} /* f contains the last float */ - YY_BREAK -case 17: -/* rule 17 can match eol */ -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - int x; - - pp->lineno = lp_yylineno; - pv->Sign = 0; - for(x = 0; x < lp_yyleng; x++) - if(lp_yytext[x] == '-' || lp_yytext[x] == '+') - pv->Sign = (pv->Sign == (lp_yytext[x] == '+')); - return (TOK_SIGN); - /* Sign is TRUE if the sign-string - represents a '-'. Otherwise Sign - is FALSE */ -} - YY_BREAK -case 18: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - if((!pv->Within_int_decl) && (!pv->Within_sec_decl) && (!pv->Within_sos_decl) && (!pv->Within_free_decl)) { - pv->Within_int_decl = 1; - pv->Within_sos_decl1 = FALSE; - } - return(SEC_INT); -} - YY_BREAK -case 19: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - if((!pv->Within_int_decl) && (!pv->Within_sec_decl) && (!pv->Within_sos_decl) && (!pv->Within_free_decl)) { - pv->Within_int_decl = 2; - pv->Within_sos_decl1 = FALSE; - } - return(SEC_BIN); -} - YY_BREAK -case 20: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - if((!pv->Within_int_decl) && (!pv->Within_sec_decl) && (!pv->Within_sos_decl) && (!pv->Within_free_decl)) { - pv->Within_sec_decl = TRUE; - pv->Within_sos_decl1 = FALSE; - } - return(SEC_SEC); -} - YY_BREAK -case 21: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - if(!pv->Within_sos_decl) - pv->SOStype0 = (short)atoi(((char *)lp_yytext) + 3); - if((!pv->Within_int_decl) && (!pv->Within_sec_decl) && (!pv->Within_sos_decl) && (!pv->Within_free_decl)) - pv->Within_sos_decl = TRUE; - return(SEC_SOS); -} - YY_BREAK -case 22: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - FREE(pv->Last_var); - pv->Last_var = strdup((char *)lp_yytext); - pv->Last_var[strlen(pv->Last_var) - 2] = 0; - return(SOSDESCR); -} - YY_BREAK -case 23: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - if((!pv->Within_int_decl) && (!pv->Within_sec_decl) && (!pv->Within_sos_decl) && (!pv->Within_free_decl)) { - pv->Within_free_decl = TRUE; - pv->Within_sos_decl1 = FALSE; - } - return(SEC_FREE); -} - YY_BREAK -case 24: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - char *ptr; - - pp->lineno = lp_yylineno; - FREE(pv->Last_var); - pv->Last_var = strdup((char *)lp_yytext); - ptr = pv->Last_var + strlen(pv->Last_var); - ptr[-1] = ' '; - while ((--ptr >= pv->Last_var) && (isspace(*ptr))) - *ptr = 0; - return(VARIABLECOLON); -} - YY_BREAK -case 25: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - FREE(pv->Last_var); - pv->Last_var = strdup((char *)lp_yytext); - return(VAR); -} - YY_BREAK -case 26: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - - pp->lineno = lp_yylineno; - return (COLON); -} - YY_BREAK -case 27: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - - pp->lineno = lp_yylineno; - return(AR_M_OP); -} - YY_BREAK -case 28: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - pv->OP = *lp_yytext; - return(RE_OPEQ); -} - YY_BREAK -case 29: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - pv->OP = *lp_yytext; - return(RE_OPLE); -} - YY_BREAK -case 30: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - pv->OP = *lp_yytext; - return(RE_OPGE); -} - YY_BREAK -case 31: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - parse_vars *pv = (parse_vars *) pp->parse_vars; - - pp->lineno = lp_yylineno; - pv->Within_int_decl = pv->Within_sec_decl = pv->Within_sos_decl = pv->Within_free_decl = FALSE; - check_int_sec_sos_free_decl(pp, (int) pv->Within_int_decl, (int) pv->Within_sec_decl, (int) pv->Within_sos_decl, (int) pv->Within_free_decl); - return(END_C); -} - YY_BREAK -case 32: -YY_RULE_SETUP -{ - parse_parm *pp = PARM; - - pp->lineno = lp_yylineno; - report(NULL, CRITICAL, "LEX ERROR : %s lineno %d\n", lp_yytext, lp_yylineno); - return(UNDEFINED); -} - YY_BREAK -case 33: -YY_RULE_SETUP -ECHO; - YY_BREAK -case YY_STATE_EOF(INITIAL): -case YY_STATE_EOF(COMMENT): -case YY_STATE_EOF(LINECOMMENT): - lp_yyterminate(); - - case YY_END_OF_BUFFER: - { - /* Amount of text matched not including the EOB char. */ - int lp_yy_amount_of_matched_text = (int) (lp_yy_cp - lp_yyg->lp_yytext_ptr) - 1; - - /* Undo the effects of YY_DO_BEFORE_ACTION. */ - *lp_yy_cp = lp_yyg->lp_yy_hold_char; - YY_RESTORE_YY_MORE_OFFSET - - if ( YY_CURRENT_BUFFER_LVALUE->lp_yy_buffer_status == YY_BUFFER_NEW ) - { - /* We're scanning a new file or input source. It's - * possible that this happened because the user - * just pointed lp_yyin at a new source and called - * lp_yylex(). If so, then we have to assure - * consistency between YY_CURRENT_BUFFER and our - * globals. Here is the right place to do so, because - * this is the first action (other than possibly a - * back-up) that will match for the new input source. - */ - lp_yyg->lp_yy_n_chars = YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars; - YY_CURRENT_BUFFER_LVALUE->lp_yy_input_file = lp_yyin; - YY_CURRENT_BUFFER_LVALUE->lp_yy_buffer_status = YY_BUFFER_NORMAL; - } - - /* Note that here we test for lp_yy_c_buf_p "<=" to the position - * of the first EOB in the buffer, since lp_yy_c_buf_p will - * already have been incremented past the NUL character - * (since all states make transitions on EOB to the - * end-of-buffer state). Contrast this with the test - * in input(). - */ - if ( lp_yyg->lp_yy_c_buf_p <= &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[lp_yyg->lp_yy_n_chars] ) - { /* This was really a NUL. */ - lp_yy_state_type lp_yy_next_state; - - lp_yyg->lp_yy_c_buf_p = lp_yyg->lp_yytext_ptr + lp_yy_amount_of_matched_text; - - lp_yy_current_state = lp_yy_get_previous_state( lp_yyscanner ); - - /* Okay, we're now positioned to make the NUL - * transition. We couldn't have - * lp_yy_get_previous_state() go ahead and do it - * for us because it doesn't know how to deal - * with the possibility of jamming (and we don't - * want to build jamming into it because then it - * will run more slowly). - */ - - lp_yy_next_state = lp_yy_try_NUL_trans( lp_yy_current_state , lp_yyscanner); - - lp_yy_bp = lp_yyg->lp_yytext_ptr + YY_MORE_ADJ; - - if ( lp_yy_next_state ) - { - /* Consume the NUL. */ - lp_yy_cp = ++lp_yyg->lp_yy_c_buf_p; - lp_yy_current_state = lp_yy_next_state; - goto lp_yy_match; - } - - else - { - lp_yy_cp = lp_yyg->lp_yy_c_buf_p; - goto lp_yy_find_action; - } - } - - else switch ( lp_yy_get_next_buffer( lp_yyscanner ) ) - { - case EOB_ACT_END_OF_FILE: - { - lp_yyg->lp_yy_did_buffer_switch_on_eof = 0; - - if ( lp_yywrap(lp_yyscanner ) ) - { - /* Note: because we've taken care in - * lp_yy_get_next_buffer() to have set up - * lp_yytext, we can now set up - * lp_yy_c_buf_p so that if some total - * hoser (like flex itself) wants to - * call the scanner after we return the - * YY_NULL, it'll still work - another - * YY_NULL will get returned. - */ - lp_yyg->lp_yy_c_buf_p = lp_yyg->lp_yytext_ptr + YY_MORE_ADJ; - - lp_yy_act = YY_STATE_EOF(YY_START); - goto do_action; - } - - else - { - if ( ! lp_yyg->lp_yy_did_buffer_switch_on_eof ) - YY_NEW_FILE; - } - break; - } - - case EOB_ACT_CONTINUE_SCAN: - lp_yyg->lp_yy_c_buf_p = - lp_yyg->lp_yytext_ptr + lp_yy_amount_of_matched_text; - - lp_yy_current_state = lp_yy_get_previous_state( lp_yyscanner ); - - lp_yy_cp = lp_yyg->lp_yy_c_buf_p; - lp_yy_bp = lp_yyg->lp_yytext_ptr + YY_MORE_ADJ; - goto lp_yy_match; - - case EOB_ACT_LAST_MATCH: - lp_yyg->lp_yy_c_buf_p = - &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[lp_yyg->lp_yy_n_chars]; - - lp_yy_current_state = lp_yy_get_previous_state( lp_yyscanner ); - - lp_yy_cp = lp_yyg->lp_yy_c_buf_p; - lp_yy_bp = lp_yyg->lp_yytext_ptr + YY_MORE_ADJ; - goto lp_yy_find_action; - } - break; - } - - default: - YY_FATAL_ERROR( - "fatal flex scanner internal error--no action found" ); - } /* end of action switch */ - } /* end of scanning one token */ -} /* end of lp_yylex */ - -/* lp_yy_get_next_buffer - try to read in a new buffer - * - * Returns a code representing an action: - * EOB_ACT_LAST_MATCH - - * EOB_ACT_CONTINUE_SCAN - continue scanning from current position - * EOB_ACT_END_OF_FILE - end of file - */ -static int lp_yy_get_next_buffer (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - register char *dest = YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf; - register char *source = lp_yyg->lp_yytext_ptr; - register int number_to_move, i; - int ret_val; - - if ( lp_yyg->lp_yy_c_buf_p > &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[lp_yyg->lp_yy_n_chars + 1] ) - YY_FATAL_ERROR( - "fatal flex scanner internal error--end of buffer missed" ); - - if ( YY_CURRENT_BUFFER_LVALUE->lp_yy_fill_buffer == 0 ) - { /* Don't try to fill the buffer, so this is an EOF. */ - if ( lp_yyg->lp_yy_c_buf_p - lp_yyg->lp_yytext_ptr - YY_MORE_ADJ == 1 ) - { - /* We matched a single character, the EOB, so - * treat this as a final EOF. - */ - return EOB_ACT_END_OF_FILE; - } - - else - { - /* We matched some text prior to the EOB, first - * process it. - */ - return EOB_ACT_LAST_MATCH; - } - } - - /* Try to read more data. */ - - /* First move last chars to start of buffer. */ - number_to_move = (int) (lp_yyg->lp_yy_c_buf_p - lp_yyg->lp_yytext_ptr) - 1; - - for ( i = 0; i < number_to_move; ++i ) - *(dest++) = *(source++); - - if ( YY_CURRENT_BUFFER_LVALUE->lp_yy_buffer_status == YY_BUFFER_EOF_PENDING ) - /* don't do the read, it's not guaranteed to return an EOF, - * just force an EOF - */ - YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars = lp_yyg->lp_yy_n_chars = 0; - - else - { - int num_to_read = - YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_size - number_to_move - 1; - - while ( num_to_read <= 0 ) - { /* Not enough room in the buffer - grow it. */ - - /* just a shorter name for the current buffer */ - YY_BUFFER_STATE b = YY_CURRENT_BUFFER; - - int lp_yy_c_buf_p_offset = - (int) (lp_yyg->lp_yy_c_buf_p - b->lp_yy_ch_buf); - - if ( b->lp_yy_is_our_buffer ) - { - int new_size = b->lp_yy_buf_size * 2; - - if ( new_size <= 0 ) - b->lp_yy_buf_size += b->lp_yy_buf_size / 8; - else - b->lp_yy_buf_size *= 2; - - b->lp_yy_ch_buf = (char *) - /* Include room in for 2 EOB chars. */ - lp_yyrealloc((void *) b->lp_yy_ch_buf,b->lp_yy_buf_size + 2 ,lp_yyscanner ); - } - else - /* Can't grow it, we don't own it. */ - b->lp_yy_ch_buf = 0; - - if ( ! b->lp_yy_ch_buf ) - YY_FATAL_ERROR( - "fatal error - scanner input buffer overflow" ); - - lp_yyg->lp_yy_c_buf_p = &b->lp_yy_ch_buf[lp_yy_c_buf_p_offset]; - - num_to_read = YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_size - - number_to_move - 1; - - } - - if ( num_to_read > YY_READ_BUF_SIZE ) - num_to_read = YY_READ_BUF_SIZE; - - /* Read in more data. */ - YY_INPUT( (&YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[number_to_move]), - lp_yyg->lp_yy_n_chars, (size_t) num_to_read ); - - YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars = lp_yyg->lp_yy_n_chars; - } - - if ( lp_yyg->lp_yy_n_chars == 0 ) - { - if ( number_to_move == YY_MORE_ADJ ) - { - ret_val = EOB_ACT_END_OF_FILE; - lp_yyrestart(lp_yyin ,lp_yyscanner); - } - - else - { - ret_val = EOB_ACT_LAST_MATCH; - YY_CURRENT_BUFFER_LVALUE->lp_yy_buffer_status = - YY_BUFFER_EOF_PENDING; - } - } - - else - ret_val = EOB_ACT_CONTINUE_SCAN; - - if ((lp_yy_size_t) (lp_yyg->lp_yy_n_chars + number_to_move) > YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_size) { - /* Extend the array by 50%, plus the number we really need. */ - lp_yy_size_t new_size = lp_yyg->lp_yy_n_chars + number_to_move + (lp_yyg->lp_yy_n_chars >> 1); - YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf = (char *) lp_yyrealloc((void *) YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf,new_size ,lp_yyscanner ); - if ( ! YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yy_get_next_buffer()" ); - } - - lp_yyg->lp_yy_n_chars += number_to_move; - YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[lp_yyg->lp_yy_n_chars] = YY_END_OF_BUFFER_CHAR; - YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[lp_yyg->lp_yy_n_chars + 1] = YY_END_OF_BUFFER_CHAR; - - lp_yyg->lp_yytext_ptr = &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[0]; - - return ret_val; -} - -/* lp_yy_get_previous_state - get the state just before the EOB char was reached */ - - static lp_yy_state_type lp_yy_get_previous_state (lp_yyscan_t lp_yyscanner) -{ - register lp_yy_state_type lp_yy_current_state; - register char *lp_yy_cp; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - lp_yy_current_state = lp_yyg->lp_yy_start; - lp_yy_current_state += YY_AT_BOL(); - - for ( lp_yy_cp = lp_yyg->lp_yytext_ptr + YY_MORE_ADJ; lp_yy_cp < lp_yyg->lp_yy_c_buf_p; ++lp_yy_cp ) - { - register YY_CHAR lp_yy_c = (*lp_yy_cp ? lp_yy_ec[YY_SC_TO_UI(*lp_yy_cp)] : 1); - if ( lp_yy_accept[lp_yy_current_state] ) - { - lp_yyg->lp_yy_last_accepting_state = lp_yy_current_state; - lp_yyg->lp_yy_last_accepting_cpos = lp_yy_cp; - } - while ( lp_yy_chk[lp_yy_base[lp_yy_current_state] + lp_yy_c] != lp_yy_current_state ) - { - lp_yy_current_state = (int) lp_yy_def[lp_yy_current_state]; - if ( lp_yy_current_state >= 144 ) - lp_yy_c = lp_yy_meta[(unsigned int) lp_yy_c]; - } - lp_yy_current_state = lp_yy_nxt[lp_yy_base[lp_yy_current_state] + (unsigned int) lp_yy_c]; - } - - return lp_yy_current_state; -} - -/* lp_yy_try_NUL_trans - try to make a transition on the NUL character - * - * synopsis - * next_state = lp_yy_try_NUL_trans( current_state ); - */ - static lp_yy_state_type lp_yy_try_NUL_trans (lp_yy_state_type lp_yy_current_state , lp_yyscan_t lp_yyscanner) -{ - register int lp_yy_is_jam; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; /* This var may be unused depending upon options. */ - register char *lp_yy_cp = lp_yyg->lp_yy_c_buf_p; - - register YY_CHAR lp_yy_c = 1; - if ( lp_yy_accept[lp_yy_current_state] ) - { - lp_yyg->lp_yy_last_accepting_state = lp_yy_current_state; - lp_yyg->lp_yy_last_accepting_cpos = lp_yy_cp; - } - while ( lp_yy_chk[lp_yy_base[lp_yy_current_state] + lp_yy_c] != lp_yy_current_state ) - { - lp_yy_current_state = (int) lp_yy_def[lp_yy_current_state]; - if ( lp_yy_current_state >= 144 ) - lp_yy_c = lp_yy_meta[(unsigned int) lp_yy_c]; - } - lp_yy_current_state = lp_yy_nxt[lp_yy_base[lp_yy_current_state] + (unsigned int) lp_yy_c]; - lp_yy_is_jam = (lp_yy_current_state == 143); - - return lp_yy_is_jam ? 0 : lp_yy_current_state; -} - - static void lp_yyunput (int c, register char * lp_yy_bp , lp_yyscan_t lp_yyscanner) -{ - register char *lp_yy_cp; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - lp_yy_cp = lp_yyg->lp_yy_c_buf_p; - - /* undo effects of setting up lp_yytext */ - *lp_yy_cp = lp_yyg->lp_yy_hold_char; - - if ( lp_yy_cp < YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf + 2 ) - { /* need to shift things up to make room */ - /* +2 for EOB chars. */ - register int number_to_move = lp_yyg->lp_yy_n_chars + 2; - register char *dest = &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[ - YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_size + 2]; - register char *source = - &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[number_to_move]; - - while ( source > YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf ) - *--dest = *--source; - - lp_yy_cp += (int) (dest - source); - lp_yy_bp += (int) (dest - source); - YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars = - lp_yyg->lp_yy_n_chars = YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_size; - - if ( lp_yy_cp < YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf + 2 ) - YY_FATAL_ERROR( "flex scanner push-back overflow" ); - } - - *--lp_yy_cp = (char) c; - - if ( c == '\n' ){ - --lp_yylineno; - } - - lp_yyg->lp_yytext_ptr = lp_yy_bp; - lp_yyg->lp_yy_hold_char = *lp_yy_cp; - lp_yyg->lp_yy_c_buf_p = lp_yy_cp; -} - -#ifndef YY_NO_INPUT -#ifdef __cplusplus - static int lp_yyinput (lp_yyscan_t lp_yyscanner) -#else - static int input (lp_yyscan_t lp_yyscanner) -#endif - -{ - int c; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - *lp_yyg->lp_yy_c_buf_p = lp_yyg->lp_yy_hold_char; - - if ( *lp_yyg->lp_yy_c_buf_p == YY_END_OF_BUFFER_CHAR ) - { - /* lp_yy_c_buf_p now points to the character we want to return. - * If this occurs *before* the EOB characters, then it's a - * valid NUL; if not, then we've hit the end of the buffer. - */ - if ( lp_yyg->lp_yy_c_buf_p < &YY_CURRENT_BUFFER_LVALUE->lp_yy_ch_buf[lp_yyg->lp_yy_n_chars] ) - /* This was really a NUL. */ - *lp_yyg->lp_yy_c_buf_p = '\0'; - - else - { /* need more input */ - int offset = lp_yyg->lp_yy_c_buf_p - lp_yyg->lp_yytext_ptr; - ++lp_yyg->lp_yy_c_buf_p; - - switch ( lp_yy_get_next_buffer( lp_yyscanner ) ) - { - case EOB_ACT_LAST_MATCH: - /* This happens because lp_yy_g_n_b() - * sees that we've accumulated a - * token and flags that we need to - * try matching the token before - * proceeding. But for input(), - * there's no matching to consider. - * So convert the EOB_ACT_LAST_MATCH - * to EOB_ACT_END_OF_FILE. - */ - - /* Reset buffer status. */ - lp_yyrestart(lp_yyin ,lp_yyscanner); - - /*FALLTHROUGH*/ - - case EOB_ACT_END_OF_FILE: - { - if ( lp_yywrap(lp_yyscanner ) ) - return EOF; - - if ( ! lp_yyg->lp_yy_did_buffer_switch_on_eof ) - YY_NEW_FILE; -#ifdef __cplusplus - return lp_yyinput(lp_yyscanner); -#else - return input(lp_yyscanner); -#endif - } - - case EOB_ACT_CONTINUE_SCAN: - lp_yyg->lp_yy_c_buf_p = lp_yyg->lp_yytext_ptr + offset; - break; - } - } - } - - c = *(unsigned char *) lp_yyg->lp_yy_c_buf_p; /* cast for 8-bit char's */ - *lp_yyg->lp_yy_c_buf_p = '\0'; /* preserve lp_yytext */ - lp_yyg->lp_yy_hold_char = *++lp_yyg->lp_yy_c_buf_p; - - YY_CURRENT_BUFFER_LVALUE->lp_yy_at_bol = (c == '\n'); - if ( YY_CURRENT_BUFFER_LVALUE->lp_yy_at_bol ) - - do{ lp_yylineno++; - lp_yycolumn=0; - }while(0) -; - - return c; -} -#endif /* ifndef YY_NO_INPUT */ - -/** Immediately switch to a different input stream. - * @param input_file A readable stream. - * @param lp_yyscanner The scanner object. - * @note This function does not reset the start condition to @c INITIAL . - */ - void lp_yyrestart (FILE * input_file , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - if ( ! YY_CURRENT_BUFFER ){ - lp_yyensure_buffer_stack (lp_yyscanner); - YY_CURRENT_BUFFER_LVALUE = - lp_yy_create_buffer(lp_yyin,YY_BUF_SIZE ,lp_yyscanner); - } - - lp_yy_init_buffer(YY_CURRENT_BUFFER,input_file ,lp_yyscanner); - lp_yy_load_buffer_state(lp_yyscanner ); -} - -/** Switch to a different input buffer. - * @param new_buffer The new input buffer. - * @param lp_yyscanner The scanner object. - */ - void lp_yy_switch_to_buffer (YY_BUFFER_STATE new_buffer , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - /* TODO. We should be able to replace this entire function body - * with - * lp_yypop_buffer_state(); - * lp_yypush_buffer_state(new_buffer); - */ - lp_yyensure_buffer_stack (lp_yyscanner); - if ( YY_CURRENT_BUFFER == new_buffer ) - return; - - if ( YY_CURRENT_BUFFER ) - { - /* Flush out information for old buffer. */ - *lp_yyg->lp_yy_c_buf_p = lp_yyg->lp_yy_hold_char; - YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_pos = lp_yyg->lp_yy_c_buf_p; - YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars = lp_yyg->lp_yy_n_chars; - } - - YY_CURRENT_BUFFER_LVALUE = new_buffer; - lp_yy_load_buffer_state(lp_yyscanner ); - - /* We don't actually know whether we did this switch during - * EOF (lp_yywrap()) processing, but the only time this flag - * is looked at is after lp_yywrap() is called, so it's safe - * to go ahead and always set it. - */ - lp_yyg->lp_yy_did_buffer_switch_on_eof = 1; -} - -static void lp_yy_load_buffer_state (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - lp_yyg->lp_yy_n_chars = YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars; - lp_yyg->lp_yytext_ptr = lp_yyg->lp_yy_c_buf_p = YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_pos; - lp_yyin = YY_CURRENT_BUFFER_LVALUE->lp_yy_input_file; - lp_yyg->lp_yy_hold_char = *lp_yyg->lp_yy_c_buf_p; -} - -/** Allocate and initialize an input buffer state. - * @param file A readable stream. - * @param size The character buffer size in bytes. When in doubt, use @c YY_BUF_SIZE. - * @param lp_yyscanner The scanner object. - * @return the allocated buffer state. - */ - YY_BUFFER_STATE lp_yy_create_buffer (FILE * file, int size , lp_yyscan_t lp_yyscanner) -{ - YY_BUFFER_STATE b; - - b = (YY_BUFFER_STATE) lp_yyalloc(sizeof( struct lp_yy_buffer_state ) ,lp_yyscanner ); - if ( ! b ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yy_create_buffer()" ); - - b->lp_yy_buf_size = size; - - /* lp_yy_ch_buf has to be 2 characters longer than the size given because - * we need to put in 2 end-of-buffer characters. - */ - b->lp_yy_ch_buf = (char *) lp_yyalloc(b->lp_yy_buf_size + 2 ,lp_yyscanner ); - if ( ! b->lp_yy_ch_buf ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yy_create_buffer()" ); - - b->lp_yy_is_our_buffer = 1; - - lp_yy_init_buffer(b,file ,lp_yyscanner); - - return b; -} - -/** Destroy the buffer. - * @param b a buffer created with lp_yy_create_buffer() - * @param lp_yyscanner The scanner object. - */ - void lp_yy_delete_buffer (YY_BUFFER_STATE b , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - if ( ! b ) - return; - - if ( b == YY_CURRENT_BUFFER ) /* Not sure if we should pop here. */ - YY_CURRENT_BUFFER_LVALUE = (YY_BUFFER_STATE) 0; - - if ( b->lp_yy_is_our_buffer ) - lp_yyfree((void *) b->lp_yy_ch_buf ,lp_yyscanner ); - - lp_yyfree((void *) b ,lp_yyscanner ); -} - -#ifndef __cplusplus -extern int isatty (int ); -#endif /* __cplusplus */ - -/* Initializes or reinitializes a buffer. - * This function is sometimes called more than once on the same buffer, - * such as during a lp_yyrestart() or at EOF. - */ - static void lp_yy_init_buffer (YY_BUFFER_STATE b, FILE * file , lp_yyscan_t lp_yyscanner) - -{ - int oerrno = errno; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - lp_yy_flush_buffer(b ,lp_yyscanner); - - b->lp_yy_input_file = file; - b->lp_yy_fill_buffer = 1; - - /* If b is the current buffer, then lp_yy_init_buffer was _probably_ - * called from lp_yyrestart() or through lp_yy_get_next_buffer. - * In that case, we don't want to reset the lineno or column. - */ - if (b != YY_CURRENT_BUFFER){ - b->lp_yy_bs_lineno = 1; - b->lp_yy_bs_column = 0; - } - - b->lp_yy_is_interactive = file ? (isatty( fileno(file) ) > 0) : 0; - - errno = oerrno; -} - -/** Discard all buffered characters. On the next scan, YY_INPUT will be called. - * @param b the buffer state to be flushed, usually @c YY_CURRENT_BUFFER. - * @param lp_yyscanner The scanner object. - */ - void lp_yy_flush_buffer (YY_BUFFER_STATE b , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - if ( ! b ) - return; - - b->lp_yy_n_chars = 0; - - /* We always need two end-of-buffer characters. The first causes - * a transition to the end-of-buffer state. The second causes - * a jam in that state. - */ - b->lp_yy_ch_buf[0] = YY_END_OF_BUFFER_CHAR; - b->lp_yy_ch_buf[1] = YY_END_OF_BUFFER_CHAR; - - b->lp_yy_buf_pos = &b->lp_yy_ch_buf[0]; - - b->lp_yy_at_bol = 1; - b->lp_yy_buffer_status = YY_BUFFER_NEW; - - if ( b == YY_CURRENT_BUFFER ) - lp_yy_load_buffer_state(lp_yyscanner ); -} - -/** Pushes the new state onto the stack. The new state becomes - * the current state. This function will allocate the stack - * if necessary. - * @param new_buffer The new state. - * @param lp_yyscanner The scanner object. - */ -void lp_yypush_buffer_state (YY_BUFFER_STATE new_buffer , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - if (new_buffer == NULL) - return; - - lp_yyensure_buffer_stack(lp_yyscanner); - - /* This block is copied from lp_yy_switch_to_buffer. */ - if ( YY_CURRENT_BUFFER ) - { - /* Flush out information for old buffer. */ - *lp_yyg->lp_yy_c_buf_p = lp_yyg->lp_yy_hold_char; - YY_CURRENT_BUFFER_LVALUE->lp_yy_buf_pos = lp_yyg->lp_yy_c_buf_p; - YY_CURRENT_BUFFER_LVALUE->lp_yy_n_chars = lp_yyg->lp_yy_n_chars; - } - - /* Only push if top exists. Otherwise, replace top. */ - if (YY_CURRENT_BUFFER) - lp_yyg->lp_yy_buffer_stack_top++; - YY_CURRENT_BUFFER_LVALUE = new_buffer; - - /* copied from lp_yy_switch_to_buffer. */ - lp_yy_load_buffer_state(lp_yyscanner ); - lp_yyg->lp_yy_did_buffer_switch_on_eof = 1; -} - -/** Removes and deletes the top of the stack, if present. - * The next element becomes the new top. - * @param lp_yyscanner The scanner object. - */ -void lp_yypop_buffer_state (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - if (!YY_CURRENT_BUFFER) - return; - - lp_yy_delete_buffer(YY_CURRENT_BUFFER ,lp_yyscanner); - YY_CURRENT_BUFFER_LVALUE = NULL; - if (lp_yyg->lp_yy_buffer_stack_top > 0) - --lp_yyg->lp_yy_buffer_stack_top; - - if (YY_CURRENT_BUFFER) { - lp_yy_load_buffer_state(lp_yyscanner ); - lp_yyg->lp_yy_did_buffer_switch_on_eof = 1; - } -} - -/* Allocates the stack if it does not exist. - * Guarantees space for at least one push. - */ -static void lp_yyensure_buffer_stack (lp_yyscan_t lp_yyscanner) -{ - int num_to_alloc; - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - if (!lp_yyg->lp_yy_buffer_stack) { - - /* First allocation is just for 2 elements, since we don't know if this - * scanner will even need a stack. We use 2 instead of 1 to avoid an - * immediate realloc on the next call. - */ - num_to_alloc = 1; - lp_yyg->lp_yy_buffer_stack = (struct lp_yy_buffer_state**)lp_yyalloc - (num_to_alloc * sizeof(struct lp_yy_buffer_state*) - , lp_yyscanner); - if ( ! lp_yyg->lp_yy_buffer_stack ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yyensure_buffer_stack()" ); - - memset(lp_yyg->lp_yy_buffer_stack, 0, num_to_alloc * sizeof(struct lp_yy_buffer_state*)); - - lp_yyg->lp_yy_buffer_stack_max = num_to_alloc; - lp_yyg->lp_yy_buffer_stack_top = 0; - return; - } - - if (lp_yyg->lp_yy_buffer_stack_top >= (lp_yyg->lp_yy_buffer_stack_max) - 1){ - - /* Increase the buffer to prepare for a possible push. */ - int grow_size = 8 /* arbitrary grow size */; - - num_to_alloc = lp_yyg->lp_yy_buffer_stack_max + grow_size; - lp_yyg->lp_yy_buffer_stack = (struct lp_yy_buffer_state**)lp_yyrealloc - (lp_yyg->lp_yy_buffer_stack, - num_to_alloc * sizeof(struct lp_yy_buffer_state*) - , lp_yyscanner); - if ( ! lp_yyg->lp_yy_buffer_stack ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yyensure_buffer_stack()" ); - - /* zero only the new slots.*/ - memset(lp_yyg->lp_yy_buffer_stack + lp_yyg->lp_yy_buffer_stack_max, 0, grow_size * sizeof(struct lp_yy_buffer_state*)); - lp_yyg->lp_yy_buffer_stack_max = num_to_alloc; - } -} - -/** Setup the input buffer state to scan directly from a user-specified character buffer. - * @param base the character buffer - * @param size the size in bytes of the character buffer - * @param lp_yyscanner The scanner object. - * @return the newly allocated buffer state object. - */ -YY_BUFFER_STATE lp_yy_scan_buffer (char * base, lp_yy_size_t size , lp_yyscan_t lp_yyscanner) -{ - YY_BUFFER_STATE b; - - if ( size < 2 || - base[size-2] != YY_END_OF_BUFFER_CHAR || - base[size-1] != YY_END_OF_BUFFER_CHAR ) - /* They forgot to leave room for the EOB's. */ - return 0; - - b = (YY_BUFFER_STATE) lp_yyalloc(sizeof( struct lp_yy_buffer_state ) ,lp_yyscanner ); - if ( ! b ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yy_scan_buffer()" ); - - b->lp_yy_buf_size = size - 2; /* "- 2" to take care of EOB's */ - b->lp_yy_buf_pos = b->lp_yy_ch_buf = base; - b->lp_yy_is_our_buffer = 0; - b->lp_yy_input_file = 0; - b->lp_yy_n_chars = b->lp_yy_buf_size; - b->lp_yy_is_interactive = 0; - b->lp_yy_at_bol = 1; - b->lp_yy_fill_buffer = 0; - b->lp_yy_buffer_status = YY_BUFFER_NEW; - - lp_yy_switch_to_buffer(b ,lp_yyscanner ); - - return b; -} - -/** Setup the input buffer state to scan a string. The next call to lp_yylex() will - * scan from a @e copy of @a str. - * @param lp_yystr a NUL-terminated string to scan - * @param lp_yyscanner The scanner object. - * @return the newly allocated buffer state object. - * @note If you want to scan bytes that may contain NUL values, then use - * lp_yy_scan_bytes() instead. - */ -YY_BUFFER_STATE lp_yy_scan_string (lp_yyconst char * lp_yystr , lp_yyscan_t lp_yyscanner) -{ - - return lp_yy_scan_bytes(lp_yystr,strlen(lp_yystr) ,lp_yyscanner); -} - -/** Setup the input buffer state to scan the given bytes. The next call to lp_yylex() will - * scan from a @e copy of @a bytes. - * @param bytes the byte buffer to scan - * @param len the number of bytes in the buffer pointed to by @a bytes. - * @param lp_yyscanner The scanner object. - * @return the newly allocated buffer state object. - */ -YY_BUFFER_STATE lp_yy_scan_bytes (lp_yyconst char * lp_yybytes, int _lp_yybytes_len , lp_yyscan_t lp_yyscanner) -{ - YY_BUFFER_STATE b; - char *buf; - lp_yy_size_t n; - int i; - - /* Get memory for full buffer, including space for trailing EOB's. */ - n = _lp_yybytes_len + 2; - buf = (char *) lp_yyalloc(n ,lp_yyscanner ); - if ( ! buf ) - YY_FATAL_ERROR( "out of dynamic memory in lp_yy_scan_bytes()" ); - - for ( i = 0; i < _lp_yybytes_len; ++i ) - buf[i] = lp_yybytes[i]; - - buf[_lp_yybytes_len] = buf[_lp_yybytes_len+1] = YY_END_OF_BUFFER_CHAR; - - b = lp_yy_scan_buffer(buf,n ,lp_yyscanner); - if ( ! b ) - YY_FATAL_ERROR( "bad buffer in lp_yy_scan_bytes()" ); - - /* It's okay to grow etc. this buffer, and we should throw it - * away when we're done. - */ - b->lp_yy_is_our_buffer = 1; - - return b; -} - -#ifndef YY_EXIT_FAILURE -#define YY_EXIT_FAILURE 2 -#endif - -static void lp_yy_fatal_error (lp_yyconst char* msg , lp_yyscan_t lp_yyscanner) -{ - (void) fprintf( stderr, "%s\n", msg ); - exit( YY_EXIT_FAILURE ); -} - -/* Redefine lp_yyless() so it works in section 3 code. */ - -#undef lp_yyless -#define lp_yyless(n) \ - do \ - { \ - /* Undo effects of setting up lp_yytext. */ \ - int lp_yyless_macro_arg = (n); \ - YY_LESS_LINENO(lp_yyless_macro_arg);\ - lp_yytext[lp_yyleng] = lp_yyg->lp_yy_hold_char; \ - lp_yyg->lp_yy_c_buf_p = lp_yytext + lp_yyless_macro_arg; \ - lp_yyg->lp_yy_hold_char = *lp_yyg->lp_yy_c_buf_p; \ - *lp_yyg->lp_yy_c_buf_p = '\0'; \ - lp_yyleng = lp_yyless_macro_arg; \ - } \ - while ( 0 ) - -/* Accessor methods (get/set functions) to struct members. */ - -/** Get the user-defined data for this scanner. - * @param lp_yyscanner The scanner object. - */ -YY_EXTRA_TYPE lp_yyget_extra (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yyextra; -} - -/** Get the current line number. - * @param lp_yyscanner The scanner object. - */ -int lp_yyget_lineno (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - if (! YY_CURRENT_BUFFER) - return 0; - - return lp_yylineno; -} - -/** Get the current column number. - * @param lp_yyscanner The scanner object. - */ -int lp_yyget_column (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - if (! YY_CURRENT_BUFFER) - return 0; - - return lp_yycolumn; -} - -/** Get the input stream. - * @param lp_yyscanner The scanner object. - */ -FILE *lp_yyget_in (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yyin; -} - -/** Get the output stream. - * @param lp_yyscanner The scanner object. - */ -FILE *lp_yyget_out (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yyout; -} - -/** Get the length of the current token. - * @param lp_yyscanner The scanner object. - */ -int lp_yyget_leng (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yyleng; -} - -/** Get the current token. - * @param lp_yyscanner The scanner object. - */ - -char *lp_yyget_text (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yytext; -} - -/** Set the user-defined data. This data is never touched by the scanner. - * @param user_defined The data to be associated with this scanner. - * @param lp_yyscanner The scanner object. - */ -void lp_yyset_extra (YY_EXTRA_TYPE user_defined , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - lp_yyextra = user_defined ; -} - -/** Set the current line number. - * @param line_number - * @param lp_yyscanner The scanner object. - */ -void lp_yyset_lineno (int line_number , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - /* lineno is only valid if an input buffer exists. */ - if (! YY_CURRENT_BUFFER ) - lp_yy_fatal_error( "lp_yyset_lineno called with no buffer" , lp_yyscanner); - - lp_yylineno = line_number; -} - -/** Set the current column. - * @param line_number - * @param lp_yyscanner The scanner object. - */ -void lp_yyset_column (int column_no , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - /* column is only valid if an input buffer exists. */ - if (! YY_CURRENT_BUFFER ) - lp_yy_fatal_error( "lp_yyset_column called with no buffer" , lp_yyscanner); - - lp_yycolumn = column_no; -} - -/** Set the input stream. This does not discard the current - * input buffer. - * @param in_str A readable stream. - * @param lp_yyscanner The scanner object. - * @see lp_yy_switch_to_buffer - */ -void lp_yyset_in (FILE * in_str , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - lp_yyin = in_str ; -} - -void lp_yyset_out (FILE * out_str , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - lp_yyout = out_str ; -} - -int lp_yyget_debug (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yy_flex_debug; -} - -void lp_yyset_debug (int bdebug , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - lp_yy_flex_debug = bdebug ; -} - -/* Accessor methods for lp_yylval and lp_yylloc */ - -YYSTYPE * lp_yyget_lval (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - return lp_yylval; -} - -void lp_yyset_lval (YYSTYPE * lp_yylval_param , lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - lp_yylval = lp_yylval_param; -} - -/* User-visible API */ - -/* lp_yylex_init is special because it creates the scanner itself, so it is - * the ONLY reentrant function that doesn't take the scanner as the last argument. - * That's why we explicitly handle the declaration, instead of using our macros. - */ - -int lp_yylex_init(lp_yyscan_t* ptr_lp_yy_globals) - -{ - if (ptr_lp_yy_globals == NULL){ - errno = EINVAL; - return 1; - } - - *ptr_lp_yy_globals = (lp_yyscan_t) lp_yyalloc ( sizeof( struct lp_yyguts_t ), NULL ); - - if (*ptr_lp_yy_globals == NULL){ - errno = ENOMEM; - return 1; - } - - /* By setting to 0xAA, we expose bugs in lp_yy_init_globals. Leave at 0x00 for releases. */ - memset(*ptr_lp_yy_globals,0x00,sizeof(struct lp_yyguts_t)); - - return lp_yy_init_globals ( *ptr_lp_yy_globals ); -} - -/* lp_yylex_init_extra has the same functionality as lp_yylex_init, but follows the - * convention of taking the scanner as the last argument. Note however, that - * this is a *pointer* to a scanner, as it will be allocated by this call (and - * is the reason, too, why this function also must handle its own declaration). - * The user defined value in the first argument will be available to lp_yyalloc in - * the lp_yyextra field. - */ - -int lp_yylex_init_extra(YY_EXTRA_TYPE lp_yy_user_defined,lp_yyscan_t* ptr_lp_yy_globals ) - -{ - struct lp_yyguts_t dummy_lp_yyguts; - - lp_yyset_extra (lp_yy_user_defined, &dummy_lp_yyguts); - - if (ptr_lp_yy_globals == NULL){ - errno = EINVAL; - return 1; - } - - *ptr_lp_yy_globals = (lp_yyscan_t) lp_yyalloc ( sizeof( struct lp_yyguts_t ), &dummy_lp_yyguts ); - - if (*ptr_lp_yy_globals == NULL){ - errno = ENOMEM; - return 1; - } - - /* By setting to 0xAA, we expose bugs in - lp_yy_init_globals. Leave at 0x00 for releases. */ - memset(*ptr_lp_yy_globals,0x00,sizeof(struct lp_yyguts_t)); - - lp_yyset_extra (lp_yy_user_defined, *ptr_lp_yy_globals); - - return lp_yy_init_globals ( *ptr_lp_yy_globals ); -} - -static int lp_yy_init_globals (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - /* Initialization is the same as for the non-reentrant scanner. - * This function is called from lp_yylex_destroy(), so don't allocate here. - */ - - lp_yyg->lp_yy_buffer_stack = 0; - lp_yyg->lp_yy_buffer_stack_top = 0; - lp_yyg->lp_yy_buffer_stack_max = 0; - lp_yyg->lp_yy_c_buf_p = (char *) 0; - lp_yyg->lp_yy_init = 0; - lp_yyg->lp_yy_start = 0; - - lp_yyg->lp_yy_start_stack_ptr = 0; - lp_yyg->lp_yy_start_stack_depth = 0; - lp_yyg->lp_yy_start_stack = NULL; - -/* Defined in main.c */ -#ifdef YY_STDINIT - lp_yyin = stdin; - lp_yyout = stdout; -#else - lp_yyin = (FILE *) 0; - lp_yyout = (FILE *) 0; -#endif - - /* For future reference: Set errno on error, since we are called by - * lp_yylex_init() - */ - return 0; -} - -/* lp_yylex_destroy is for both reentrant and non-reentrant scanners. */ -int lp_yylex_destroy (lp_yyscan_t lp_yyscanner) -{ - struct lp_yyguts_t * lp_yyg = (struct lp_yyguts_t*)lp_yyscanner; - - /* Pop the buffer stack, destroying each element. */ - while(YY_CURRENT_BUFFER){ - lp_yy_delete_buffer(YY_CURRENT_BUFFER ,lp_yyscanner ); - YY_CURRENT_BUFFER_LVALUE = NULL; - lp_yypop_buffer_state(lp_yyscanner); - } - - /* Destroy the stack itself. */ - lp_yyfree(lp_yyg->lp_yy_buffer_stack ,lp_yyscanner); - lp_yyg->lp_yy_buffer_stack = NULL; - - /* Destroy the start condition stack. */ - lp_yyfree(lp_yyg->lp_yy_start_stack ,lp_yyscanner ); - lp_yyg->lp_yy_start_stack = NULL; - - /* Reset the globals. This is important in a non-reentrant scanner so the next time - * lp_yylex() is called, initialization will occur. */ - lp_yy_init_globals( lp_yyscanner); - - /* Destroy the main struct (reentrant only). */ - lp_yyfree ( lp_yyscanner , lp_yyscanner ); - lp_yyscanner = NULL; - return 0; -} - -/* - * Internal utility routines. - */ - -#ifndef lp_yytext_ptr -static void lp_yy_flex_strncpy (char* s1, lp_yyconst char * s2, int n , lp_yyscan_t lp_yyscanner) -{ - register int i; - for ( i = 0; i < n; ++i ) - s1[i] = s2[i]; -} -#endif - -#ifdef YY_NEED_STRLEN -static int lp_yy_flex_strlen (lp_yyconst char * s , lp_yyscan_t lp_yyscanner) -{ - register int n; - for ( n = 0; s[n]; ++n ) - ; - - return n; -} -#endif - -void *lp_yyalloc (lp_yy_size_t size , lp_yyscan_t lp_yyscanner) -{ - return (void *) malloc( size ); -} - -void *lp_yyrealloc (void * ptr, lp_yy_size_t size , lp_yyscan_t lp_yyscanner) -{ - /* The cast to (char *) in the following accommodates both - * implementations that use char* generic pointers, and those - * that use void* generic pointers. It works with the latter - * because both ANSI C and C++ allow castless assignment from - * any pointer type to void*, and deal with argument conversions - * as though doing an assignment. - */ - return (void *) realloc( (char *) ptr, size ); -} - -void lp_yyfree (void * ptr , lp_yyscan_t lp_yyscanner) -{ - free( (char *) ptr ); /* see lp_yyrealloc() for (char *) cast */ -} - -#define YYTABLES_NAME "lp_yytables" - diff --git a/code/3rd_lpsolve/lp_scale.c b/code/3rd_lpsolve/lp_scale.c deleted file mode 100644 index 826ec592..00000000 --- a/code/3rd_lpsolve/lp_scale.c +++ /dev/null @@ -1,1075 +0,0 @@ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_report.h" -#include "lp_scale.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* - Scaling routines for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: LGPL. - - Requires: lp_lib.h, lp_scale.h - - Release notes: - v5.0.0 1 January 2004 Significantly expanded and repackaged scaling - routines. - v5.0.1 20 February 2004 Modified rounding behaviour in several areas. - v5.1.0 20 July 2004 Reworked with flexible matrix storage model. - v5.2.0 20 February 2005 Converted to matrix storage model without the OF. - - ---------------------------------------------------------------------------------- -*/ - -/* First define scaling and unscaling primitives */ - -REAL scaled_value(lprec *lp, REAL value, int index) -{ - if(fabs(value) < lp->infinite) { - if(lp->scaling_used) { - if(index > lp->rows) - value /= lp->scalars[index]; - else - value *= lp->scalars[index]; - } - } - else - value = my_sign(value)*lp->infinite; - return(value); -} - -REAL unscaled_value(lprec *lp, REAL value, int index) -{ - if(fabs(value) < lp->infinite) { - if(lp->scaling_used) { - if(index > lp->rows) - value *= lp->scalars[index]; - else - value /= lp->scalars[index]; - } - } - else - value = my_sign(value)*lp->infinite; - return(value); -} - -STATIC REAL scaled_mat(lprec *lp, REAL value, int rownr, int colnr) -{ - if(lp->scaling_used) - value *= lp->scalars[rownr] * lp->scalars[lp->rows + colnr]; - return( value ); -} - -STATIC REAL unscaled_mat(lprec *lp, REAL value, int rownr, int colnr) -{ - if(lp->scaling_used) - value /= lp->scalars[rownr] * lp->scalars[lp->rows + colnr]; - return( value ); -} - -/* Compute the scale factor by the formulae: - FALSE: SUM (log |Aij|) ^ 2 - TRUE: SUM (log |Aij| - RowScale[i] - ColScale[j]) ^ 2 */ -REAL CurtisReidMeasure(lprec *lp, MYBOOL _Advanced, REAL *FRowScale, REAL *FColScale) -{ - int i, nz; - REAL absvalue, logvalue; - register REAL result; - MATrec *mat = lp->matA; - REAL *value; - int *rownr, *colnr; - - /* Do OF part */ - result = 0; - for(i = 1; i <= lp->columns; i++) { - absvalue = fabs(lp->orig_obj[i]); - if(absvalue > 0) { - logvalue = log(absvalue); - if(_Advanced) - logvalue -= FRowScale[0] + FColScale[i]; - result += logvalue*logvalue; - } - } - - /* Do constraint matrix part */ - mat_validate(mat); - value = &(COL_MAT_VALUE(0)); - rownr = &(COL_MAT_ROWNR(0)); - colnr = &(COL_MAT_COLNR(0)); - nz = get_nonzeros(lp); - for(i = 0; i < nz; - i++, value += matValueStep, rownr += matRowColStep, colnr += matRowColStep) { - absvalue = fabs(*value); - if(absvalue > 0) { - logvalue = log(absvalue); - if(_Advanced) - logvalue -= FRowScale[*rownr] + FColScale[*colnr]; - result += logvalue*logvalue; - } - } - return( result ); -} - -/* Implementation of the Curtis-Reid scaling based on the paper - "On the Automatic Scaling of Matrices for Gaussian - Elimination," Journal of the Institute of Mathematics and - Its Applications (1972) 10, 118-124. - - Solve the system | M E | (r) (sigma) - | | ( ) = ( ) - | E^T N | (c) ( tau ) - - by the conjugate gradient method (clever recurrences). - - E is the matrix A with all elements = 1 - - M is diagonal matrix of row counts (RowCount) - N is diagonal matrix of column counts (ColCount) - - sigma is the vector of row logarithm sums (RowSum) - tau is the vector of column logarithm sums (ColSum) - - r, c are resulting row and column scalings (RowScale, ColScale) */ - -int CurtisReidScales(lprec *lp, MYBOOL _Advanced, REAL *FRowScale, REAL *FColScale) -{ - int i, row, col, ent, nz; - REAL *RowScalem2, *ColScalem2, - *RowSum, *ColSum, - *residual_even, *residual_odd; - REAL sk, qk, ek, - skm1, qkm1, ekm1, - qkm2, qkqkm1, ekm2, ekekm1, - absvalue, logvalue, - StopTolerance; - int *RowCount, *ColCount, colMax; - int Result; - MATrec *mat = lp->matA; - REAL *value; - int *rownr, *colnr; - - if(CurtisReidMeasure(lp, _Advanced, FRowScale, FColScale)<0.1*get_nonzeros(lp)) - return(0); - - /* Allocate temporary memory and find RowSum and ColSum measures */ - nz = get_nonzeros(lp); - colMax = lp->columns; - - allocREAL(lp, &RowSum, lp->rows+1, TRUE); - allocINT(lp, &RowCount, lp->rows+1, TRUE); - allocREAL(lp, &residual_odd, lp->rows+1, TRUE); - - allocREAL(lp, &ColSum, colMax+1, TRUE); - allocINT(lp, &ColCount, colMax+1, TRUE); - allocREAL(lp, &residual_even, colMax+1, TRUE); - - allocREAL(lp, &RowScalem2, lp->rows+1, FALSE); - allocREAL(lp, &ColScalem2, colMax+1, FALSE); - - /* Set origin for row scaling */ - for(i = 1; i <= colMax; i++) { - absvalue=fabs(lp->orig_obj[i]); - if(absvalue>0) { - logvalue = log(absvalue); - ColSum[i] += logvalue; - RowSum[0] += logvalue; - ColCount[i]++; - RowCount[0]++; - } - } - - value = &(COL_MAT_VALUE(0)); - rownr = &(COL_MAT_ROWNR(0)); - colnr = &(COL_MAT_COLNR(0)); - for(i = 0; i < nz; - i++, value += matValueStep, rownr += matRowColStep, colnr += matRowColStep) { - absvalue=fabs(*value); - if(absvalue>0) { - logvalue = log(absvalue); - ColSum[*colnr] += logvalue; - RowSum[*rownr] += logvalue; - ColCount[*colnr]++; - RowCount[*rownr]++; - } - } - - /* Make sure we dont't have division by zero errors */ - for(row = 0; row <= lp->rows; row++) - if(RowCount[row] == 0) - RowCount[row] = 1; - for(col = 1; col <= colMax; col++) - if(ColCount[col] == 0) - ColCount[col] = 1; - - /* Initialize to RowScale = RowCount-1 RowSum - ColScale = 0.0 - residual = ColSum - ET RowCount-1 RowSum */ - - StopTolerance= MAX(lp->scalelimit-floor(lp->scalelimit), DEF_SCALINGEPS); - StopTolerance *= (REAL) nz; - for(row = 0; row <= lp->rows; row++) { - FRowScale[row] = RowSum[row] / (REAL) RowCount[row]; - RowScalem2[row] = FRowScale[row]; - } - - /* Compute initial residual */ - for(col = 1; col <= colMax; col++) { - FColScale[col] = 0; - ColScalem2[col] = 0; - residual_even[col] = ColSum[col]; - - if(lp->orig_obj[col] != 0) - residual_even[col] -= RowSum[0] / (REAL) RowCount[0]; - - i = mat->col_end[col-1]; - rownr = &(COL_MAT_ROWNR(i)); - ent = mat->col_end[col]; - for(; i < ent; - i++, rownr += matRowColStep) { - residual_even[col] -= RowSum[*rownr] / (REAL) RowCount[*rownr]; - } - } - - /* Compute sk */ - sk = 0; - skm1 = 0; - for(col = 1; col <= colMax; col++) - sk += (residual_even[col]*residual_even[col]) / (REAL) ColCount[col]; - - Result = 0; - qk=1; qkm1=0; qkm2=0; - ek=0; ekm1=0; ekm2=0; - - while(sk>StopTolerance) { - /* Given the values of residual and sk, construct - ColScale (when pass is even) - RowScale (when pass is odd) */ - - qkqkm1 = qk * qkm1; - ekekm1 = ek * ekm1; - if((Result % 2) == 0) { /* pass is even; construct RowScale[pass+1] */ - if(Result != 0) { - for(row = 0; row <= lp->rows; row++) - RowScalem2[row] = FRowScale[row]; - if(qkqkm1 != 0) { - for(row = 0; row <= lp->rows; row++) - FRowScale[row]*=(1 + ekekm1 / qkqkm1); - for(row = 0; row<=lp->rows; row++) - FRowScale[row]+=(residual_odd[row] / (qkqkm1 * (REAL) RowCount[row]) - - RowScalem2[row] * ekekm1 / qkqkm1); - } - } - } - else { /* pass is odd; construct ColScale[pass+1] */ - for(col = 1; col <= colMax; col++) - ColScalem2[col] = FColScale[col]; - if(qkqkm1 != 0) { - for(col = 1; col <= colMax; col++) - FColScale[col] *= (1 + ekekm1 / qkqkm1); - for(col = 1; col <= colMax; col++) - FColScale[col] += (residual_even[col] / ((REAL) ColCount[col] * qkqkm1) - - ColScalem2[col] * ekekm1 / qkqkm1); - } - } - - /* update residual and sk (pass + 1) */ - if((Result % 2) == 0) { /* even */ - /* residual */ - for(row = 0; row <= lp->rows; row++) - residual_odd[row] *= ek; - - for(i = 1; i <= colMax; i++) - if(lp->orig_obj[i] != 0) - residual_odd[0] += (residual_even[i] / (REAL) ColCount[i]); - - rownr = &(COL_MAT_ROWNR(0)); - colnr = &(COL_MAT_COLNR(0)); - for(i = 0; i < nz; - i++, rownr += matRowColStep, colnr += matRowColStep) { - residual_odd[*rownr] += (residual_even[*colnr] / (REAL) ColCount[*colnr]); - } - for(row = 0; row <= lp->rows; row++) - residual_odd[row] *= (-1 / qk); - - /* sk */ - skm1 = sk; - sk = 0; - for(row = 0; row <= lp->rows; row++) - sk += (residual_odd[row]*residual_odd[row]) / (REAL) RowCount[row]; - } - else { /* odd */ - /* residual */ - for(col = 1; col <= colMax; col++) - residual_even[col] *= ek; - - for(i = 1; i <= colMax; i++) - if(lp->orig_obj[i] != 0) - residual_even[i] += (residual_odd[0] / (REAL) RowCount[0]); - - rownr = &(COL_MAT_ROWNR(0)); - colnr = &(COL_MAT_COLNR(0)); - for(i = 0; i < nz; - i++, rownr += matRowColStep, colnr += matRowColStep) { - residual_even[*colnr] += (residual_odd[*rownr] / (REAL) RowCount[*rownr]); - } - for(col = 1; col <= colMax; col++) - residual_even[col] *= (-1 / qk); - - /* sk */ - skm1 = sk; - sk = 0; - for(col = 1; col <= colMax; col++) - sk += (residual_even[col]*residual_even[col]) / (REAL) ColCount[col]; - } - - /* Compute ek and qk */ - ekm2=ekm1; - ekm1=ek; - ek=qk * sk / skm1; - - qkm2=qkm1; - qkm1=qk; - qk=1-ek; - - Result++; - } - - /* Synchronize the RowScale and ColScale vectors */ - ekekm1 = ek * ekm1; - if(qkm1 != 0) { - if((Result % 2) == 0) { /* pass is even, compute RowScale */ - for(row = 0; row<=lp->rows; row++) - FRowScale[row]*=(1.0 + ekekm1 / qkm1); - for(row = 0; row<=lp->rows; row++) - FRowScale[row]+=(residual_odd[row] / (qkm1 * (REAL) RowCount[row]) - - RowScalem2[row] * ekekm1 / qkm1); - } - else { /* pass is odd, compute ColScale */ - for(col=1; col<=colMax; col++) - FColScale[col]*=(1 + ekekm1 / qkm1); - for(col=1; col<=colMax; col++) - FColScale[col]+=(residual_even[col] / ((REAL) ColCount[col] * qkm1) - - ColScalem2[col] * ekekm1 / qkm1); - } - } - - /* Do validation, if indicated */ - if(FALSE && mat_validate(mat)){ - double check, error; - - /* CHECK: M RowScale + E ColScale = RowSum */ - error = 0; - for(row = 0; row <= lp->rows; row++) { - check = (REAL) RowCount[row] * FRowScale[row]; - if(row == 0) { - for(i = 1; i <= colMax; i++) { - if(lp->orig_obj[i] != 0) - check += FColScale[i]; - } - } - else { - i = mat->row_end[row-1]; - ent = mat->row_end[row]; - for(; i < ent; i++) { - col = ROW_MAT_COLNR(i); - check += FColScale[col]; - } - } - check -= RowSum[row]; - error += check*check; - } - - /* CHECK: E^T RowScale + N ColScale = ColSum */ - error = 0; - for(col = 1; col <= colMax; col++) { - check = (REAL) ColCount[col] * FColScale[col]; - - if(lp->orig_obj[col] != 0) - check += FRowScale[0]; - - i = mat->col_end[col-1]; - ent = mat->col_end[col]; - rownr = &(COL_MAT_ROWNR(i)); - for(; i < ent; - i++, rownr += matRowColStep) { - check += FRowScale[*rownr]; - } - check -= ColSum[col]; - error += check*check; - } - } - - /* Convert to scaling factors (rounding to nearest power - of 2 can optionally be done as a separate step later) */ - for(col = 1; col <= colMax; col++) { - absvalue = exp(-FColScale[col]); - if(absvalue < MIN_SCALAR) absvalue = MIN_SCALAR; - if(absvalue > MAX_SCALAR) absvalue = MAX_SCALAR; - if(!is_int(lp,col) || is_integerscaling(lp)) - FColScale[col] = absvalue; - else - FColScale[col] = 1; - } - for(row = 0; row <= lp->rows; row++) { - absvalue = exp(-FRowScale[row]); - if(absvalue < MIN_SCALAR) absvalue = MIN_SCALAR; - if(absvalue > MAX_SCALAR) absvalue = MAX_SCALAR; - FRowScale[row] = absvalue; - } - - /* free temporary memory */ - FREE(RowSum); - FREE(ColSum); - FREE(RowCount); - FREE(ColCount); - FREE(residual_even); - FREE(residual_odd); - FREE(RowScalem2); - FREE(ColScalem2); - - return(Result); - -} - -STATIC MYBOOL scaleCR(lprec *lp, REAL *scaledelta) -{ - REAL *scalechange = NULL; - int Result; - - if(!lp->scaling_used) { - allocREAL(lp, &lp->scalars, lp->sum_alloc + 1, FALSE); - for(Result = 0; Result <= lp->sum; Result++) - lp->scalars[Result] = 1; - lp->scaling_used = TRUE; - } - - if(scaledelta == NULL) - allocREAL(lp, &scalechange, lp->sum + 1, FALSE); - else - scalechange = scaledelta; - - Result=CurtisReidScales(lp, FALSE, scalechange, &scalechange[lp->rows]); - if(Result>0) { - - /* Do the scaling*/ - if(scale_updaterows(lp, scalechange, TRUE) || - scale_updatecolumns(lp, &scalechange[lp->rows], TRUE)) - lp->scalemode |= SCALE_CURTISREID; - - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - } - - if(scaledelta == NULL) - FREE(scalechange); - - return((MYBOOL) (Result > 0)); -} - -STATIC MYBOOL transform_for_scale(lprec *lp, REAL *value) -{ - MYBOOL Accept = TRUE; - *value = fabs(*value); -#ifdef Paranoia - if(*value < lp->epsmachine) { - Accept = FALSE; - report(lp, SEVERE, "transform_for_scale: A zero-valued entry was passed\n"); - } - else -#endif - if(is_scalemode(lp, SCALE_LOGARITHMIC)) - *value = log(*value); - else if(is_scalemode(lp, SCALE_QUADRATIC)) - (*value) *= (*value); - return( Accept ); -} - -STATIC void accumulate_for_scale(lprec *lp, REAL *min, REAL *max, REAL value) -{ - if(transform_for_scale(lp, &value)) { - if(is_scaletype(lp, SCALE_MEAN)) { - *max += value; - *min += 1; - } - else { - SETMAX(*max, value); - SETMIN(*min, value); - } - } -} - -STATIC REAL minmax_to_scale(lprec *lp, REAL min, REAL max, int itemcount) -{ - REAL scale; - - /* Initialize according to transformation / weighting model */ - if(is_scalemode(lp, SCALE_LOGARITHMIC)) - scale = 0; - else - scale = 1; - if(itemcount <= 0) - return(scale); - - /* Compute base scalar according to chosen scaling type */ - if(is_scaletype(lp, SCALE_MEAN)) { - if(min > 0) - scale = max / min; - } - else if(is_scaletype(lp, SCALE_RANGE)) - scale = (max + min) / 2; - else if(is_scaletype(lp, SCALE_GEOMETRIC)) - scale = sqrt(min*max); - else if(is_scaletype(lp, SCALE_EXTREME)) - scale = max; - - /* Compute final scalar according to transformation / weighting model */ - if(is_scalemode(lp, SCALE_LOGARITHMIC)) - scale = exp(-scale); - else if(is_scalemode(lp, SCALE_QUADRATIC)) { - if(scale == 0) - scale = 1; - else - scale = 1 / sqrt(scale); - } - else { - if(scale == 0) - scale = 1; - else - scale = 1 / scale; - } - - /* Make sure we are within acceptable scaling ranges */ - SETMAX(scale, MIN_SCALAR); - SETMIN(scale, MAX_SCALAR); - - return(scale); -} - -STATIC REAL roundPower2(REAL scale) -/* Purpose is to round a number to it nearest power of 2; in a system - with binary number representation, this avoids rounding errors when - scale is used to normalize another value */ -{ - long int power2; - MYBOOL isSmall = FALSE; - - if(scale == 1) - return( scale ); - - /* Obtain the fractional power of 2 */ - if(scale < 2) { - scale = 2 / scale; - isSmall = TRUE; - } - else - scale /= 2; - scale = log(scale)/log(2.0); - - /* Find the desired nearest power of two and compute the associated scalar */ - power2 = (long) ceil(scale-0.5); - scale = 1 << power2; - if(isSmall) - scale = 1.0 / scale; - - return( scale ); - -} - -STATIC MYBOOL scale_updatecolumns(lprec *lp, REAL *scalechange, MYBOOL updateonly) -{ - int i, j; - - /* Verify that the scale change is significant (different from the unit) */ - for(i = lp->columns; i > 0; i--) - if(fabs(scalechange[i]-1) > lp->epsprimal) - break; - if(i <= 0) - return( FALSE ); - - /* Update the pre-existing column scalar */ - if(updateonly) - for(i = 1, j = lp->rows + 1; j <= lp->sum; i++, j++) - lp->scalars[j] *= scalechange[i]; - else - for(i = 1, j = lp->rows + 1; j <= lp->sum; i++, j++) - lp->scalars[j] = scalechange[i]; - - return( TRUE ); -} - -STATIC MYBOOL scale_updaterows(lprec *lp, REAL *scalechange, MYBOOL updateonly) -{ - int i; - - /* Verify that the scale change is significant (different from the unit) */ - for(i = lp->rows; i >= 0; i--) { - if(fabs(scalechange[i]-1) > lp->epsprimal) - break; - } - if(i < 0) - return( FALSE ); - - /* Update the pre-existing row scalar */ - if(updateonly) - for(i = 0; i <= lp->rows; i++) - lp->scalars[i] *= scalechange[i]; - else - for(i = 0; i <= lp->rows; i++) - lp->scalars[i] = scalechange[i]; - - return( TRUE ); -} - -STATIC MYBOOL scale_columns(lprec *lp, REAL *scaledelta) -{ - int i,j, colMax, nz; - REAL *scalechange; - REAL *value; - int *colnr; - MATrec *mat = lp->matA; - - /* Check that columns are in fact targeted */ - if((lp->scalemode & SCALE_ROWSONLY) != 0) - return( TRUE ); - - if(scaledelta == NULL) - scalechange = &lp->scalars[lp->rows]; - else - scalechange = &scaledelta[lp->rows]; - - colMax = lp->columns; - - /* Scale matrix entries (including any Lagrangean constraints) */ - for(i = 1; i <= lp->columns; i++) { - lp->orig_obj[i] *= scalechange[i]; - } - - mat_validate(lp->matA); - nz = get_nonzeros(lp); - value = &(COL_MAT_VALUE(0)); - colnr = &(COL_MAT_COLNR(0)); - for(i = 0; i < nz; - i++, value += matValueStep, colnr += matRowColStep) { - (*value) *= scalechange[*colnr]; - } - - /* Scale variable bounds as well */ - for(i = 1, j = lp->rows + 1; j <= lp->sum; i++, j++) { - if(lp->orig_lowbo[j] > -lp->infinite) - lp->orig_lowbo[j] /= scalechange[i]; - if(lp->orig_upbo[j] < lp->infinite) - lp->orig_upbo[j] /= scalechange[i]; - if(lp->sc_lobound[i] != 0) - lp->sc_lobound[i] /= scalechange[i]; - } - - lp->columns_scaled = TRUE; - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - - return( TRUE ); -} - -STATIC MYBOOL scale_rows(lprec *lp, REAL *scaledelta) -{ - int i, j, nz, colMax; - REAL *scalechange; - REAL *value; - int *rownr; - MATrec *mat = lp->matA; - - - /* Check that rows are in fact targeted */ - if((lp->scalemode & SCALE_COLSONLY) != 0) - return( TRUE ); - - if(scaledelta == NULL) - scalechange = lp->scalars; - else - scalechange = scaledelta; - - colMax = lp->columns; - - /* First row-scale the matrix (including the objective function) */ - for(i = 1; i <= colMax; i++) { - lp->orig_obj[i] *= scalechange[0]; - } - - nz = get_nonzeros(lp); - value = &(COL_MAT_VALUE(0)); - rownr = &(COL_MAT_ROWNR(0)); - for(i = 0; i < nz; - i++, value += matValueStep, rownr += matRowColStep) { - (*value) *= scalechange[*rownr]; - } - - /* ...and scale the rhs and the row bounds (RANGES in MPS!!) */ - for(i = 0; i <= lp->rows; i++) { - if(fabs(lp->orig_rhs[i]) < lp->infinite) - lp->orig_rhs[i] *= scalechange[i]; - - j = lp->presolve_undo->var_to_orig[i]; - if(j != 0) - lp->presolve_undo->fixed_rhs[j] *= scalechange[i]; - - if(lp->orig_upbo[i] < lp->infinite) /* This is the range */ - lp->orig_upbo[i] *= scalechange[i]; - - if((lp->orig_lowbo[i] != 0) && (fabs(lp->orig_lowbo[i]) < lp->infinite)) - lp->orig_lowbo[i] *= scalechange[i]; - } - - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - - return( TRUE ); -} - -STATIC REAL scale(lprec *lp, REAL *scaledelta) -{ - int i, j, nz, row_count, nzOF = 0; - REAL *row_max, *row_min, *scalechange = NULL, absval; - REAL col_max, col_min; - MYBOOL rowscaled, colscaled; - MATrec *mat = lp->matA; - REAL *value; - int *rownr; - - if(is_scaletype(lp, SCALE_NONE)) - return(0.0); - - if(!lp->scaling_used) { - allocREAL(lp, &lp->scalars, lp->sum_alloc + 1, FALSE); - for(i = 0; i <= lp->sum; i++) { - lp->scalars[i] = 1; - } - lp->scaling_used = TRUE; - } -#ifdef Paranoia - else - for(i = 0; i <= lp->sum; i++) { - if(lp->scalars[i] == 0) - report(lp, SEVERE, "scale: Zero-valued scalar found at index %d\n", i); - } -#endif - if(scaledelta == NULL) - allocREAL(lp, &scalechange, lp->sum + 1, FALSE); - else - scalechange = scaledelta; - - /* Must initialize due to computation of scaling statistic - KE */ - for(i = 0; i <= lp->sum; i++) - scalechange[i] = 1; - - row_count = lp->rows; - allocREAL(lp, &row_max, row_count + 1, TRUE); - allocREAL(lp, &row_min, row_count + 1, FALSE); - - /* Initialise min and max values of rows */ - for(i = 0; i <= row_count; i++) { - if(is_scaletype(lp, SCALE_MEAN)) - row_min[i] = 0; /* Carries the count of elements */ - else - row_min[i] = lp->infinite; /* Carries the minimum element */ - } - - /* Calculate row scaling data */ - for(j = 1; j <= lp->columns; j++) { - - absval = lp->orig_obj[j]; - if(absval != 0) { - absval = scaled_mat(lp, absval, 0, j); - accumulate_for_scale(lp, &row_min[0], &row_max[0], absval); - nzOF++; - } - - i = mat->col_end[j - 1]; - value = &(COL_MAT_VALUE(i)); - rownr = &(COL_MAT_ROWNR(i)); - nz = mat->col_end[j]; - for(; i < nz; - i++, value += matValueStep, rownr += matRowColStep) { - absval = scaled_mat(lp, *value, *rownr, j); - accumulate_for_scale(lp, &row_min[*rownr], &row_max[*rownr], absval); - } - } - - /* Calculate scale factors for rows */ - i = 0; - for(; i <= lp->rows; i++) { - if(i == 0) - nz = nzOF; - else - nz = mat_rowlength(lp->matA, i); - absval = minmax_to_scale(lp, row_min[i], row_max[i], nz); /* nz instead of nzOF KJEI 20/05/2010 */ - if(absval == 0) - absval = 1; - scalechange[i] = absval; - } - - FREE(row_max); - FREE(row_min); - - /* Row-scale the matrix (including the objective function and Lagrangean constraints) */ - rowscaled = scale_updaterows(lp, scalechange, TRUE); - - /* Calculate column scales */ - i = 1; - for(j = 1; j <= lp->columns; j++) { - if(is_int(lp,j) && !is_integerscaling(lp)) { /* do not scale integer columns */ - scalechange[lp->rows + j] = 1; - } - else { - col_max = 0; - if(is_scaletype(lp, SCALE_MEAN)) - col_min = 0; - else - col_min = lp->infinite; - - absval = lp->orig_obj[j]; - if(absval != 0) { - absval = scaled_mat(lp, absval, 0, j); - accumulate_for_scale(lp, &col_min, &col_max, absval); - } - - i = mat->col_end[j - 1]; - value = &(COL_MAT_VALUE(i)); - rownr = &(COL_MAT_ROWNR(i)); - nz = mat->col_end[j]; - for(; i < nz; - i++, value += matValueStep, rownr += matRowColStep) { - absval = scaled_mat(lp, *value, *rownr, j); - accumulate_for_scale(lp, &col_min, &col_max, absval); - } - nz = mat_collength(lp->matA, j); - if(fabs(lp->orig_obj[j]) > 0) - nz++; - scalechange[lp->rows + j] = minmax_to_scale(lp, col_min, col_max, nz); - } - } - - /* ... and then column-scale the already row-scaled matrix */ - colscaled = scale_updatecolumns(lp, &scalechange[lp->rows], TRUE); - - /* Create a geometric mean-type measure of the extent of scaling performed; */ - /* ideally, upon successive calls to scale() the value should converge to 0 */ - if(rowscaled || colscaled) { - col_max = 0; - for(j = 1; j <= lp->columns; j++) - col_max += log(scalechange[lp->rows + j]); - col_max = exp(col_max/lp->columns); - - i = 0; - col_min = 0; - for(; i <= lp->rows; i++) - col_min += log(scalechange[i]); - col_min = exp(col_min/row_count); - } - else { - col_max = 1; - col_min = 1; - } - - if(scaledelta == NULL) - FREE(scalechange); - - return(1 - sqrt(col_max*col_min)); -} - -STATIC MYBOOL finalize_scaling(lprec *lp, REAL *scaledelta) -{ - int i; - - /* Check if we should equilibrate */ - if(is_scalemode(lp, SCALE_EQUILIBRATE) && !is_scaletype(lp, SCALE_CURTISREID)) { - int oldmode; - - oldmode = lp->scalemode; - lp->scalemode = SCALE_LINEAR + SCALE_EXTREME; - scale(lp, scaledelta); - lp->scalemode = oldmode; - } - - /* Check if we should prevent rounding errors */ - if(is_scalemode(lp, SCALE_POWER2)) { - REAL *scalars; - if(scaledelta == NULL) - scalars = lp->scalars; - else - scalars = scaledelta; - - for(i = 0; i <= lp->sum; i++) - scalars[i] = roundPower2(scalars[i]); - } - - /* Then transfer the scalars to the model's data */ - return( scale_rows(lp, scaledelta) && scale_columns(lp, scaledelta) ); - -} - -STATIC REAL auto_scale(lprec *lp) -{ - int n = 1; - REAL scalingmetric = 0, *scalenew = NULL; - - if(lp->scaling_used && - ((((lp->scalemode & SCALE_DYNUPDATE) == 0)) || (lp->bb_level > 0))) - return( scalingmetric); - - if(lp->scalemode != SCALE_NONE) { - - /* Allocate array for incremental scaling if appropriate */ - if((lp->solvecount > 1) && (lp->bb_level < 1) && - ((lp->scalemode & SCALE_DYNUPDATE) != 0)) - allocREAL(lp, &scalenew, lp->sum + 1, FALSE); - - if(is_scaletype(lp, SCALE_CURTISREID)) { - scalingmetric = scaleCR(lp, scalenew); - } - else { - REAL scalinglimit, scalingdelta; - int count; - - /* Integer value of scalelimit holds the maximum number of iterations; default to 1 */ - count = (int) floor(lp->scalelimit); - scalinglimit = lp->scalelimit; - if((count == 0) || (scalinglimit == 0)) { - if(scalinglimit > 0) - count = DEF_SCALINGLIMIT; /* A non-zero convergence has been given, default to max 5 iterations */ - else - count = 1; - } - else - scalinglimit -= count; - - /* Scale to desired relative convergence or iteration limit */ - n = 0; - scalingdelta = 1.0; - scalingmetric = 1.0; - while((n < count) && (fabs(scalingdelta) > scalinglimit)) { - n++; - scalingdelta = scale(lp, scalenew); - scalingmetric = scalingmetric*(1+scalingdelta); - } - scalingmetric -= 1; - } - } - - /* Update the inf norm of the elements of the matrix (excluding the OF) */ - mat_computemax(lp->matA); - - /* Check if we really have to do scaling */ - if(lp->scaling_used && (fabs(scalingmetric) >= lp->epsprimal)) - /* Ok, do it */ - finalize_scaling(lp, scalenew); - - else { - - /* Otherwise reset scaling variables */ - if(lp->scalars != NULL) { - FREE(lp->scalars); - } - lp->scaling_used = FALSE; - lp->columns_scaled = FALSE; - } - if(scalenew != NULL) - FREE(scalenew); - - return(scalingmetric); -} - -STATIC void unscale_columns(lprec *lp) -{ - int i, j, nz; - MATrec *mat = lp->matA; - REAL *value; - int *rownr, *colnr; - - if(!lp->columns_scaled) - return; - - /* Unscale OF */ - for(j = 1; j <= lp->columns; j++) { - lp->orig_obj[j] = unscaled_mat(lp, lp->orig_obj[j], 0, j); - } - - /* Unscale mat */ - mat_validate(mat); - nz = get_nonzeros(lp); - value = &(COL_MAT_VALUE(0)); - rownr = &(COL_MAT_ROWNR(0)); - colnr = &(COL_MAT_COLNR(0)); - for(j = 0; j < nz; - j++, value += matValueStep, rownr += matRowColStep, colnr += matRowColStep) { - *value = unscaled_mat(lp, *value, *rownr, *colnr); - } - - /* Unscale bounds as well */ - for(i = lp->rows + 1, j = 1; i <= lp->sum; i++, j++) { - lp->orig_lowbo[i] = unscaled_value(lp, lp->orig_lowbo[i], i); - lp->orig_upbo[i] = unscaled_value(lp, lp->orig_upbo[i], i); - lp->sc_lobound[j] = unscaled_value(lp, lp->sc_lobound[j], i); - } - - for(i = lp->rows + 1; i<= lp->sum; i++) - lp->scalars[i] = 1; - - lp->columns_scaled = FALSE; - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); -} - -void undoscale(lprec *lp) -{ - int i, j, nz; - MATrec *mat = lp->matA; - REAL *value; - int *rownr, *colnr; - - if(lp->scaling_used) { - - /* Unscale the OF */ - for(j = 1; j <= lp->columns; j++) { - lp->orig_obj[j] = unscaled_mat(lp, lp->orig_obj[j], 0, j); - } - - /* Unscale the matrix */ - mat_validate(mat); - nz = get_nonzeros(lp); - value = &(COL_MAT_VALUE(0)); - rownr = &(COL_MAT_ROWNR(0)); - colnr = &(COL_MAT_COLNR(0)); - for(j = 0; j < nz; - j++, value += matValueStep, rownr += matRowColStep, colnr += matRowColStep) { - *value = unscaled_mat(lp, *value, *rownr, *colnr); - } - - /* Unscale variable bounds */ - for(i = lp->rows + 1, j = 1; i <= lp->sum; i++, j++) { - lp->orig_lowbo[i] = unscaled_value(lp, lp->orig_lowbo[i], i); - lp->orig_upbo[i] = unscaled_value(lp, lp->orig_upbo[i], i); - lp->sc_lobound[j] = unscaled_value(lp, lp->sc_lobound[j], i); - } - - /* Unscale the rhs, upper and lower bounds... */ - for(i = 0; i <= lp->rows; i++) { - lp->orig_rhs[i] = unscaled_value(lp, lp->orig_rhs[i], i); - j = lp->presolve_undo->var_to_orig[i]; - if(j != 0) - lp->presolve_undo->fixed_rhs[j] = unscaled_value(lp, lp->presolve_undo->fixed_rhs[j], i); - lp->orig_lowbo[i] = unscaled_value(lp, lp->orig_lowbo[i], i); - lp->orig_upbo[i] = unscaled_value(lp, lp->orig_upbo[i], i); - } - - FREE(lp->scalars); - lp->scaling_used = FALSE; - lp->columns_scaled = FALSE; - - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT | ACTION_RECOMPUTE); - } -} - diff --git a/code/3rd_lpsolve/lp_scale.h b/code/3rd_lpsolve/lp_scale.h deleted file mode 100644 index 2aa8cc63..00000000 --- a/code/3rd_lpsolve/lp_scale.h +++ /dev/null @@ -1,31 +0,0 @@ -#ifndef HEADER_lp_scale -#define HEADER_lp_scale - -#include "lp_types.h" - -#ifdef __cplusplus -extern "C" { -#endif - -/* Put function headers here */ -STATIC MYBOOL scale_updatecolumns(lprec *lp, REAL *scalechange, MYBOOL updateonly); -STATIC MYBOOL scale_updaterows(lprec *lp, REAL *scalechange, MYBOOL updateonly); -STATIC MYBOOL scale_rows(lprec *lp, REAL *scaledelta); -STATIC MYBOOL scale_columns(lprec *lp, REAL *scaledelta); -STATIC void unscale_columns(lprec *lp); -STATIC REAL scale(lprec *lp, REAL *scaledelta); -STATIC REAL scaled_mat(lprec *lp, REAL value, int rownr, int colnr); -STATIC REAL unscaled_mat(lprec *lp, REAL value, int rownr, int colnr); -STATIC REAL scaled_value(lprec *lp, REAL value, int index); -STATIC REAL unscaled_value(lprec *lp, REAL value, int index); -STATIC MYBOOL scaleCR(lprec *lp, REAL *scaledelta); -STATIC MYBOOL finalize_scaling(lprec *lp, REAL *scaledelta); -STATIC REAL auto_scale(lprec *lp); -void undoscale(lprec *lp); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_scale */ - diff --git a/code/3rd_lpsolve/lp_simplex.c b/code/3rd_lpsolve/lp_simplex.c deleted file mode 100644 index cf0ed18e..00000000 --- a/code/3rd_lpsolve/lp_simplex.c +++ /dev/null @@ -1,2206 +0,0 @@ - -/* - Core optimization drivers for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Michel Berkelaar (to lp_solve v3.2), - Kjell Eikland (v4.0 and forward) - Contact: - License terms: LGPL. - - Requires: lp_lib.h, lp_simplex.h, lp_presolve.h, lp_pricerPSE.h - - Release notes: - v5.0.0 1 January 2004 New unit applying stacked basis and bounds storage. - v5.0.1 31 January 2004 Moved B&B routines to separate file and implemented - a new runsolver() general purpose call method. - v5.0.2 1 May 2004 Changed routine names to be more intuitive. - v5.1.0 10 January 2005 Created modular stalling/cycling functions. - Rewrote dualloop() to optimize long dual and - also streamlined primloop() correspondingly. - v5.2.0 20 March 2005 Reimplemented primal phase 1 logic. - Made multiple pricing finally work (primal simplex). - - ---------------------------------------------------------------------------------- -*/ - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_BFP.h" -#include "lp_simplex.h" -#include "lp_crash.h" -#include "lp_presolve.h" -#include "lp_price.h" -#include "lp_pricePSE.h" -#include "lp_report.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -STATIC void stallMonitor_update(lprec *lp, REAL newOF) -{ - int newpos; - OBJmonrec *monitor = lp->monitor; - - if(monitor->countstep < OBJ_STEPS) - monitor->countstep++; - else - monitor->startstep = mod(monitor->startstep + 1, OBJ_STEPS); - newpos = mod(monitor->startstep + monitor->countstep - 1, OBJ_STEPS); - monitor->objstep[newpos] = newOF; - monitor->idxstep[newpos] = monitor->Icount; - monitor->currentstep = newpos; -} - -STATIC MYBOOL stallMonitor_creepingObj(lprec *lp) -{ - OBJmonrec *monitor = lp->monitor; - - if(monitor->countstep > 1) { - REAL deltaOF = (monitor->objstep[monitor->currentstep] - - monitor->objstep[monitor->startstep]) / monitor->countstep; - deltaOF /= MAX(1, (monitor->idxstep[monitor->currentstep] - - monitor->idxstep[monitor->startstep])); - deltaOF = my_chsign(monitor->isdual, deltaOF); - return( (MYBOOL) (deltaOF < monitor->epsvalue) ); - } - else - return( FALSE ); -} - -STATIC MYBOOL stallMonitor_shortSteps(lprec *lp) -{ - OBJmonrec *monitor = lp->monitor; - - if(monitor->countstep == OBJ_STEPS) { - REAL deltaOF = MAX(1, (monitor->idxstep[monitor->currentstep] - - monitor->idxstep[monitor->startstep])) / monitor->countstep; - deltaOF = pow(deltaOF*OBJ_STEPS, 0.66); - return( (MYBOOL) (deltaOF > monitor->limitstall[TRUE]) ); - } - else - return( FALSE ); -} - -STATIC void stallMonitor_reset(lprec *lp) -{ - OBJmonrec *monitor = lp->monitor; - - monitor->ruleswitches = 0; - monitor->Ncycle = 0; - monitor->Mcycle = 0; - monitor->Icount = 0; - monitor->startstep = 0; - monitor->objstep[monitor->startstep] = lp->infinite; - monitor->idxstep[monitor->startstep] = monitor->Icount; - monitor->prevobj = 0; - monitor->countstep = 1; -} - -STATIC MYBOOL stallMonitor_create(lprec *lp, MYBOOL isdual, char *funcname) -{ - OBJmonrec *monitor = NULL; - if(lp->monitor != NULL) - return( FALSE ); - - monitor = (OBJmonrec *) calloc(sizeof(*monitor), 1); - if(monitor == NULL) - return( FALSE ); - - monitor->lp = lp; - strcpy(monitor->spxfunc, funcname); - monitor->isdual = isdual; - monitor->pivdynamic = is_piv_mode(lp, PRICE_ADAPTIVE); - monitor->oldpivstrategy = lp->piv_strategy; - monitor->oldpivrule = get_piv_rule(lp); - if(MAX_STALLCOUNT <= 1) - monitor->limitstall[FALSE] = 0; - else - monitor->limitstall[FALSE] = MAX(MAX_STALLCOUNT, - (int) pow((REAL) (lp->rows+lp->columns)/2, 0.667)); -#if 1 - monitor->limitstall[FALSE] *= 2+2; /* Expand degeneracy/stalling tolerance range */ -#endif - monitor->limitstall[TRUE] = monitor->limitstall[FALSE]; - if(monitor->oldpivrule == PRICER_DEVEX) /* Increase tolerance since primal Steepest Edge is expensive */ - monitor->limitstall[TRUE] *= 2; - if(MAX_RULESWITCH <= 0) - monitor->limitruleswitches = MAXINT32; - else - monitor->limitruleswitches = MAX(MAX_RULESWITCH, - lp->rows/MAX_RULESWITCH); - monitor->epsvalue = lp->epsprimal; /* lp->epsvalue; */ - lp->monitor = monitor; - stallMonitor_reset(lp); - lp->suminfeas = lp->infinite; - return( TRUE ); -} - -STATIC MYBOOL stallMonitor_check(lprec *lp, int rownr, int colnr, int lastnr, - MYBOOL minit, MYBOOL approved, MYBOOL *forceoutEQ) -{ - OBJmonrec *monitor = lp->monitor; - MYBOOL isStalled, isCreeping, acceptance = TRUE; - int altrule, -#ifdef Paranoia - msglevel = NORMAL; -#else - msglevel = DETAILED; -#endif - REAL deltaobj = lp->suminfeas; - - /* Accept unconditionally if this is the first or second call */ - monitor->active = FALSE; - if(monitor->Icount <= 1) { - if(monitor->Icount == 1) { - monitor->prevobj = lp->rhs[0]; - monitor->previnfeas = deltaobj; - } - monitor->Icount++; - return( acceptance ); - } - - /* Define progress as primal objective less sum of (primal/dual) infeasibilities */ - monitor->thisobj = lp->rhs[0]; - monitor->thisinfeas = deltaobj; - if(lp->spx_trace && - (lastnr > 0)) - report(lp, NORMAL, "%s: Objective at iter %10.0f is " RESULTVALUEMASK " (%4d: %4d %s- %4d)\n", - monitor->spxfunc, - (double) get_total_iter(lp), monitor->thisobj, rownr, lastnr, - my_if(minit == ITERATE_MAJORMAJOR, "<","|"), colnr); - monitor->pivrule = get_piv_rule(lp); - - /* Check if we have a stationary solution at selected tolerance level; - allow some difference in case we just refactorized the basis. */ - deltaobj = my_reldiff(monitor->thisobj, monitor->prevobj); - deltaobj = fabs(deltaobj); /* Pre v5.2 version */ - isStalled = (MYBOOL) (deltaobj < monitor->epsvalue); - - /* Also require that we have a measure of infeasibility-stalling */ - if(isStalled) { - REAL testvalue, refvalue = monitor->epsvalue; -#if 1 - if(monitor->isdual) - refvalue *= 1000*log10(9.0+lp->rows); - else - refvalue *= 1000*log10(9.0+lp->columns); -#else - refvalue *= 1000*log10(9.0+lp->sum); -#endif - testvalue = my_reldiff(monitor->thisinfeas, monitor->previnfeas); - isStalled &= (fabs(testvalue) < refvalue); - - /* Check if we should force "major" pivoting, i.e. no bound flips; - this is activated when we see the feasibility deteriorate */ -/* if(!isStalled && (testvalue > 0) && (TRUE || is_action(lp->anti_degen, ANTIDEGEN_BOUNDFLIP))) */ -#if !defined _PRICE_NOBOUNDFLIP - if(!isStalled && (testvalue > 0) && is_action(lp->anti_degen, ANTIDEGEN_BOUNDFLIP)) - acceptance = AUTOMATIC; - } -#else - if(!isStalled && (testvalue > 0) && !ISMASKSET(lp->piv_strategy, PRICE_NOBOUNDFLIP)) { - SETMASK(lp->piv_strategy, PRICE_NOBOUNDFLIP); - acceptance = AUTOMATIC; - } - } - else - CLEARMASK(lp->piv_strategy, PRICE_NOBOUNDFLIP); -#endif - -#if 1 - isCreeping = FALSE; -#else - isCreeping |= stallMonitor_creepingObj(lp); -/* isCreeping |= stallMonitor_shortSteps(lp); */ -#endif - if(isStalled || isCreeping) { - - /* Update counters along with specific tolerance for bound flips */ -#if 1 - if(minit != ITERATE_MAJORMAJOR) { - if(++monitor->Mcycle > 2) { - monitor->Mcycle = 0; - monitor->Ncycle++; - } - } - else -#endif - monitor->Ncycle++; - - /* Start to monitor for variable cycling if this is the initial stationarity */ - if(monitor->Ncycle <= 1) { - monitor->Ccycle = colnr; - monitor->Rcycle = rownr; - } - - /* Check if we should change pivoting strategy */ - else if(isCreeping || /* We have OF creep */ - (monitor->Ncycle > monitor->limitstall[monitor->isdual]) || /* KE empirical value */ - ((monitor->Ccycle == rownr) && (monitor->Rcycle == colnr))) { /* Obvious cycling */ - - monitor->active = TRUE; - - /* Try to force out equality slacks to combat degeneracy */ - if((lp->fixedvars > 0) && (*forceoutEQ != TRUE)) { - *forceoutEQ = TRUE; - goto Proceed; - } - - /* Our options are now to select an alternative rule or to do bound perturbation; - check if these options are available to us or if we must signal failure and break out. */ - approved &= monitor->pivdynamic && (monitor->ruleswitches < monitor->limitruleswitches); - if(!approved && !is_anti_degen(lp, ANTIDEGEN_STALLING)) { - lp->spx_status = DEGENERATE; - report(lp, msglevel, "%s: Stalling at iter %10.0f; no alternative strategy left.\n", - monitor->spxfunc, (double) get_total_iter(lp)); - acceptance = FALSE; - return( acceptance ); - } - - /* See if we can do the appropriate alternative rule. */ - switch (monitor->oldpivrule) { - case PRICER_FIRSTINDEX: altrule = PRICER_DEVEX; - break; - case PRICER_DANTZIG: altrule = PRICER_DEVEX; - break; - case PRICER_DEVEX: altrule = PRICER_STEEPESTEDGE; - break; - case PRICER_STEEPESTEDGE: altrule = PRICER_DEVEX; - break; - default: altrule = PRICER_FIRSTINDEX; - } - if(approved && - (monitor->pivrule != altrule) && (monitor->pivrule == monitor->oldpivrule)) { - - /* Switch rule to combat degeneracy. */ - monitor->ruleswitches++; - lp->piv_strategy = altrule; - monitor->Ccycle = 0; - monitor->Rcycle = 0; - monitor->Ncycle = 0; - monitor->Mcycle = 0; - report(lp, msglevel, "%s: Stalling at iter %10.0f; changed to '%s' rule.\n", - monitor->spxfunc, (double) get_total_iter(lp), - get_str_piv_rule(get_piv_rule(lp))); - if((altrule == PRICER_DEVEX) || (altrule == PRICER_STEEPESTEDGE)) - restartPricer(lp, AUTOMATIC); - } - - /* If not, code for bound relaxation/perturbation */ - else { - report(lp, msglevel, "%s: Stalling at iter %10.0f; proceed to bound relaxation.\n", - monitor->spxfunc, (double) get_total_iter(lp)); - acceptance = FALSE; - lp->spx_status = DEGENERATE; - return( acceptance ); - } - } - } - - /* Otherwise change back to original selection strategy as soon as possible */ - else { - if(monitor->pivrule != monitor->oldpivrule) { - lp->piv_strategy = monitor->oldpivstrategy; - altrule = monitor->oldpivrule; - if((altrule == PRICER_DEVEX) || (altrule == PRICER_STEEPESTEDGE)) - restartPricer(lp, AUTOMATIC); - report(lp, msglevel, "...returned to original pivot selection rule at iter %.0f.\n", - (double) get_total_iter(lp)); - } - stallMonitor_update(lp, monitor->thisobj); - monitor->Ccycle = 0; - monitor->Rcycle = 0; - monitor->Ncycle = 0; - monitor->Mcycle = 0; - } - - /* Update objective progress tracker */ -Proceed: - monitor->Icount++; - if(deltaobj >= monitor->epsvalue) - monitor->prevobj = monitor->thisobj; - monitor->previnfeas = monitor->thisinfeas; - - return( acceptance ); -} - -STATIC void stallMonitor_finish(lprec *lp) -{ - OBJmonrec *monitor = lp->monitor; - if(monitor == NULL) - return; - if(lp->piv_strategy != monitor->oldpivstrategy) - lp->piv_strategy = monitor->oldpivstrategy; - FREE(monitor); - lp->monitor = NULL; -} - - -STATIC MYBOOL add_artificial(lprec *lp, int forrownr, REAL *nzarray, int *idxarray) -/* This routine is called for each constraint at the start of - primloop and the primal problem is infeasible. Its - purpose is to add artificial variables and associated - objective function values to populate primal phase 1. */ -{ - MYBOOL add; - - /* Make sure we don't add unnecessary artificials, i.e. avoid - cases where the slack variable is enough */ - add = !isBasisVarFeasible(lp, lp->epspivot, forrownr); - - if(add) { - int *rownr = NULL, i, bvar, ii; - REAL *avalue = NULL, rhscoef, acoef; - MATrec *mat = lp->matA; - - /* Check the simple case where a slack is basic */ - for(i = 1; i <= lp->rows; i++) { - if(lp->var_basic[i] == forrownr) - break; - } - acoef = 1; - - /* If not, look for any basic user variable that has a - non-zero coefficient in the current constraint row */ - if(i > lp->rows) { - for(i = 1; i <= lp->rows; i++) { - ii = lp->var_basic[i] - lp->rows; - if((ii <= 0) || (ii > (lp->columns-lp->P1extraDim))) - continue; - ii = mat_findelm(mat, forrownr, ii); - if(ii >= 0) { - acoef = COL_MAT_VALUE(ii); - break; - } - } - } - - /* If no candidate was found above, gamble on using the densest column available */ -#if 0 - if(i > lp->rows) { - int len = 0; - bvar = 0; - for(i = 1; i <= lp->rows; i++) { - ii = lp->var_basic[i] - lp->rows; - if((ii <= 0) || (ii > (lp->columns-lp->P1extraDim))) - continue; - if(mat_collength(mat, ii) > len) { - len = mat_collength(mat, ii); - bvar = i; - } - } - i = bvar; - acoef = 1; - } -#endif - - bvar = i; - - add = (MYBOOL) (bvar <= lp->rows); - if(add) { - rhscoef = lp->rhs[forrownr]; - - /* Create temporary sparse array storage */ - if(nzarray == NULL) - allocREAL(lp, &avalue, 2, FALSE); - else - avalue = nzarray; - if(idxarray == NULL) - allocINT(lp, &rownr, 2, FALSE); - else - rownr = idxarray; - - /* Set the objective coefficient */ - rownr[0] = 0; - avalue[0] = my_chsign(is_chsign(lp, 0), 1); - - /* Set the constraint row coefficient */ - rownr[1] = forrownr; - avalue[1] = my_chsign(is_chsign(lp, forrownr), my_sign(rhscoef/acoef)); - - /* Add the column of artificial variable data to the user data matrix */ - add_columnex(lp, 2, avalue, rownr); - - /* Free the temporary sparse array storage */ - if(idxarray == NULL) - FREE(rownr); - if(nzarray == NULL) - FREE(avalue); - - /* Now set the artificial variable to be basic */ - set_basisvar(lp, bvar, lp->sum); - lp->P1extraDim++; - } - else { - report(lp, CRITICAL, "add_artificial: Could not find replacement basis variable for row %d\n", - forrownr); - lp->basis_valid = FALSE; - } - - } - - return(add); - -} - -STATIC int get_artificialRow(lprec *lp, int colnr) -{ - MATrec *mat = lp->matA; - -#ifdef Paranoia - if((colnr <= lp->columns-abs(lp->P1extraDim)) || (colnr > lp->columns)) - report(lp, SEVERE, "get_artificialRow: Invalid column index %d\n", colnr); - if(mat->col_end[colnr] - mat->col_end[colnr-1] != 1) - report(lp, SEVERE, "get_artificialRow: Invalid column non-zero count\n"); -#endif - - /* Return the row index of the singleton */ - colnr = mat->col_end[colnr-1]; - colnr = COL_MAT_ROWNR(colnr); - return( colnr ); -} - -STATIC int findAnti_artificial(lprec *lp, int colnr) -/* Primal simplex: Find a basic artificial variable to swap - against the non-basic slack variable, if possible */ -{ - int i, k, rownr = 0, P1extraDim = abs(lp->P1extraDim); - - if((P1extraDim == 0) || (colnr > lp->rows) || !lp->is_basic[colnr]) - return( rownr ); - - for(i = 1; i <= lp->rows; i++) { - k = lp->var_basic[i]; - if((k > lp->sum-P1extraDim) && (lp->rhs[i] == 0)) { - rownr = get_artificialRow(lp, k-lp->rows); - - /* Should we find the artificial's slack direct "antibody"? */ - if(rownr == colnr) - break; - rownr = 0; - } - } - return( rownr ); -} - -STATIC int findBasicArtificial(lprec *lp, int before) -{ - int i = 0, P1extraDim = abs(lp->P1extraDim); - - if(P1extraDim > 0) { - if(before > lp->rows || before <= 1) - i = lp->rows; - else - i = before; - - while((i > 0) && (lp->var_basic[i] <= lp->sum-P1extraDim)) - i--; - } - - return(i); -} - -STATIC void eliminate_artificials(lprec *lp, REAL *prow) -{ - int i, j, colnr, rownr, P1extraDim = abs(lp->P1extraDim); - - for(i = 1; (i <= lp->rows) && (P1extraDim > 0); i++) { - j = lp->var_basic[i]; - if(j <= lp->sum-P1extraDim) - continue; - j -= lp->rows; - rownr = get_artificialRow(lp, j); - colnr = find_rowReplacement(lp, rownr, prow, NULL); -#if 0 - performiteration(lp, rownr, colnr, 0.0, TRUE, FALSE, prow, NULL, - NULL, NULL, NULL); -#else - set_basisvar(lp, rownr, colnr); -#endif - del_column(lp, j); - P1extraDim--; - } - lp->P1extraDim = 0; -} - -STATIC void clear_artificials(lprec *lp) -{ - int i, j, n, P1extraDim; - - /* Substitute any basic artificial variable for its slack counterpart */ - n = 0; - P1extraDim = abs(lp->P1extraDim); - for(i = 1; (i <= lp->rows) && (n < P1extraDim); i++) { - j = lp->var_basic[i]; - if(j <= lp->sum-P1extraDim) - continue; - j = get_artificialRow(lp, j-lp->rows); - set_basisvar(lp, i, j); - n++; - } -#ifdef Paranoia - if(n != lp->P1extraDim) - report(lp, SEVERE, "clear_artificials: Unable to clear all basic artificial variables\n"); -#endif - - /* Delete any remaining non-basic artificial variables */ - while(P1extraDim > 0) { - i = lp->sum-lp->rows; - del_column(lp, i); - P1extraDim--; - } - lp->P1extraDim = 0; - if(n > 0) { - set_action(&lp->spx_action, ACTION_REINVERT); - lp->basis_valid = TRUE; - } -} - - -STATIC int primloop(lprec *lp, MYBOOL primalfeasible, REAL primaloffset) -{ - MYBOOL primal = TRUE, bfpfinal = FALSE, changedphase = FALSE, forceoutEQ = AUTOMATIC, - primalphase1, pricerCanChange, minit, stallaccept, pendingunbounded; - int i, j, k, colnr = 0, rownr = 0, lastnr = 0, - candidatecount = 0, minitcount = 0, ok = TRUE; - LREAL theta = 0.0; - REAL epsvalue, xviolated = 0.0, cviolated = 0.0, - *prow = NULL, *pcol = NULL, - *drow = lp->drow; - int *workINT = NULL, - *nzdrow = lp->nzdrow; - - if(lp->spx_trace) - report(lp, DETAILED, "Entered primal simplex algorithm with feasibility %s\n", - my_boolstr(primalfeasible)); - - /* Add sufficent number of artificial variables to make the problem feasible - through the first phase; delete when primal feasibility has been achieved */ - lp->P1extraDim = 0; - if(!primalfeasible) { - lp->simplex_mode = SIMPLEX_Phase1_PRIMAL; -#ifdef Paranoia - if(!verify_basis(lp)) - report(lp, SEVERE, "primloop: No valid basis for artificial variables\n"); -#endif -#if 0 - /* First check if we can get away with a single artificial variable */ - if(lp->equalities == 0) { - i = (int) feasibilityOffset(lp, !primal); - add_artificial(lp, i, prow, (int *) pcol); - } - else -#endif - /* Otherwise add as many artificial variables as is necessary - to force primal feasibility. */ - for(i = 1; i <= lp->rows; i++) { - add_artificial(lp, i, NULL, NULL); - } - - /* Make sure we update the working objective */ - if(lp->P1extraDim > 0) { -#if 1 /* v5.1 code: Not really necessary since we do not price the artificial - variables (stored at the end of the column list, they are initially - basic and are never allowed to enter the basis, once they exit) */ - ok = allocREAL(lp, &(lp->drow), lp->sum+1, AUTOMATIC) && - allocINT(lp, &(lp->nzdrow), lp->sum+1, AUTOMATIC); - if(!ok) - goto Finish; - lp->nzdrow[0] = 0; - drow = lp->drow; - nzdrow = lp->nzdrow; -#endif - mat_validate(lp->matA); - set_OF_p1extra(lp, 0.0); - } - if(lp->spx_trace) - report(lp, DETAILED, "P1extraDim count = %d\n", lp->P1extraDim); - - simplexPricer(lp, (MYBOOL)!primal); - invert(lp, INITSOL_USEZERO, TRUE); - } - else { - lp->simplex_mode = SIMPLEX_Phase2_PRIMAL; - restartPricer(lp, (MYBOOL)!primal); - } - - /* Create work arrays and optionally the multiple pricing structure */ - ok = allocREAL(lp, &(lp->bsolveVal), lp->rows + 1, FALSE) && - allocREAL(lp, &prow, lp->sum + 1, TRUE) && - allocREAL(lp, &pcol, lp->rows + 1, TRUE); - if(is_piv_mode(lp, PRICE_MULTIPLE) && (lp->multiblockdiv > 1)) { - lp->multivars = multi_create(lp, FALSE); - ok &= (lp->multivars != NULL) && - multi_resize(lp->multivars, lp->sum / lp->multiblockdiv, 2, FALSE, TRUE); - } - if(!ok) - goto Finish; - - /* Initialize regular primal simplex algorithm variables */ - lp->spx_status = RUNNING; - minit = ITERATE_MAJORMAJOR; - epsvalue = lp->epspivot; - pendingunbounded = FALSE; - - ok = stallMonitor_create(lp, FALSE, "primloop"); - if(!ok) - goto Finish; - - lp->rejectpivot[0] = 0; - - /* Iterate while we are successful; exit when the model is infeasible/unbounded, - or we must terminate due to numeric instability or user-determined reasons */ - while((lp->spx_status == RUNNING) && !userabort(lp, -1)) { - - primalphase1 = (MYBOOL) (lp->P1extraDim > 0); - clear_action(&lp->spx_action, ACTION_REINVERT | ACTION_ITERATE); - - /* Check if we have stalling (from numerics or degenerate cycling) */ - pricerCanChange = !primalphase1; - stallaccept = stallMonitor_check(lp, rownr, colnr, lastnr, minit, pricerCanChange, &forceoutEQ); - if(!stallaccept) - break; - - /* Find best column to enter the basis */ -RetryCol: -#if 0 - if(verify_solution(lp, FALSE, "spx_loop") > 0) - i = 1; /* This is just a debug trap */ -#endif - if(!changedphase) { - i = 0; - do { - i++; - colnr = colprim(lp, drow, nzdrow, (MYBOOL) (minit == ITERATE_MINORRETRY), i, &candidatecount, TRUE, &xviolated); - } while ((colnr == 0) && (i < partial_countBlocks(lp, (MYBOOL) !primal)) && - partial_blockStep(lp, (MYBOOL) !primal)); - - /* Handle direct outcomes */ - if(colnr == 0) - lp->spx_status = OPTIMAL; - if(lp->rejectpivot[0] > 0) - minit = ITERATE_MAJORMAJOR; - - /* See if accuracy check during compute_reducedcosts flagged refactorization */ - if(is_action(lp->spx_action, ACTION_REINVERT)) - bfpfinal = TRUE; - - } - - /* Make sure that we do not erroneously conclude that an unbounded model is optimal */ -#ifdef primal_UseRejectionList - if((colnr == 0) && (lp->rejectpivot[0] > 0)) { - lp->spx_status = UNBOUNDED; - if((lp->spx_trace && (lp->bb_totalnodes == 0)) || - (lp->bb_trace && (lp->bb_totalnodes > 0))) - report(lp, DETAILED, "The model is primal unbounded.\n"); - colnr = lp->rejectpivot[1]; - rownr = 0; - lp->rejectpivot[0] = 0; - ok = FALSE; - break; - } -#endif - - /* Check if we found an entering variable (indicating that we are still dual infeasible) */ - if(colnr > 0) { - changedphase = FALSE; - fsolve(lp, colnr, pcol, NULL, lp->epsmachine, 1.0, TRUE); /* Solve entering column for Pi */ - - /* Do special anti-degeneracy column selection, if specified */ - if(is_anti_degen(lp, ANTIDEGEN_COLUMNCHECK) && !check_degeneracy(lp, pcol, NULL)) { - if(lp->rejectpivot[0] < DEF_MAXPIVOTRETRY/3) { - i = ++lp->rejectpivot[0]; - lp->rejectpivot[i] = colnr; - report(lp, DETAILED, "Entering column %d found to be non-improving due to degeneracy.\n", - colnr); - minit = ITERATE_MINORRETRY; - goto RetryCol; - } - else { - lp->rejectpivot[0] = 0; - report(lp, DETAILED, "Gave up trying to find a strictly improving entering column.\n"); - } - } - - /* Find the leaving variable that gives the most stringent bound on the entering variable */ - theta = drow[colnr]; - rownr = rowprim(lp, colnr, &theta, pcol, workINT, forceoutEQ, &cviolated); - -#ifdef AcceptMarginalAccuracy - /* Check for marginal accuracy */ - if((rownr > 0) && (xviolated+cviolated < lp->epspivot)) { - if(lp->bb_trace || (lp->bb_totalnodes == 0)) - report(lp, DETAILED, "primloop: Assuming convergence with reduced accuracy %g.\n", - MAX(xviolated, cviolated)); - rownr = 0; - colnr = 0; - goto Optimality; - } - else -#endif - - /* See if we can do a straight artificial<->slack replacement (when "colnr" is a slack) */ - if((lp->P1extraDim != 0) && (rownr == 0) && (colnr <= lp->rows)) - rownr = findAnti_artificial(lp, colnr); - - if(rownr > 0) { - pendingunbounded = FALSE; - lp->rejectpivot[0] = 0; - set_action(&lp->spx_action, ACTION_ITERATE); - if(!lp->obj_in_basis) /* We must manually copy the reduced cost for RHS update */ - pcol[0] = my_chsign(!lp->is_lower[colnr], drow[colnr]); - lp->bfp_prepareupdate(lp, rownr, colnr, pcol); - } - - /* We may be unbounded... */ - else { - /* First make sure that we are not suffering from precision loss */ -#ifdef primal_UseRejectionList - if(lp->rejectpivot[0] < DEF_MAXPIVOTRETRY) { - lp->spx_status = RUNNING; - lp->rejectpivot[0]++; - lp->rejectpivot[lp->rejectpivot[0]] = colnr; - report(lp, DETAILED, "...trying to recover via another pivot column.\n"); - minit = ITERATE_MINORRETRY; - goto RetryCol; - } - else -#endif - /* Check that we are not having numerical problems */ - if(!refactRecent(lp) && !pendingunbounded) { - bfpfinal = TRUE; - pendingunbounded = TRUE; - set_action(&lp->spx_action, ACTION_REINVERT); - } - - /* Conclude that the model is unbounded */ - else { - lp->spx_status = UNBOUNDED; - report(lp, DETAILED, "The model is primal unbounded.\n"); - break; - } - } - } - - /* We handle optimality and phase 1 infeasibility ... */ - else { - -Optimality: - /* Handle possible transition from phase 1 to phase 2 */ - if(!primalfeasible || isP1extra(lp)) { - - if(feasiblePhase1(lp, epsvalue)) { - lp->spx_status = RUNNING; - if(lp->bb_totalnodes == 0) { - report(lp, NORMAL, "Found feasibility by primal simplex after %10.0f iter.\n", - (double) get_total_iter(lp)); - if((lp->usermessage != NULL) && (lp->msgmask & MSG_LPFEASIBLE)) - lp->usermessage(lp, lp->msghandle, MSG_LPFEASIBLE); - } - changedphase = FALSE; - primalfeasible = TRUE; - lp->simplex_mode = SIMPLEX_Phase2_PRIMAL; - set_OF_p1extra(lp, 0.0); - - /* We can do two things now; - 1) delete the rows belonging to those variables, since they are redundant, OR - 2) drive out the existing artificial variables via pivoting. */ - if(lp->P1extraDim > 0) { - -#ifdef Phase1EliminateRedundant - /* If it is not a MIP model we can try to delete redundant rows */ - if((lp->bb_totalnodes == 0) && (MIP_count(lp) == 0)) { - while(lp->P1extraDim > 0) { - i = lp->rows; - while((i > 0) && (lp->var_basic[i] <= lp->sum-lp->P1extraDim)) - i--; -#ifdef Paranoia - if(i <= 0) { - report(lp, SEVERE, "primloop: Could not find redundant artificial.\n"); - break; - } -#endif - /* Obtain column and row indeces */ - j = lp->var_basic[i]-lp->rows; - k = get_artificialRow(lp, j); - - /* Delete row before column due to basis "compensation logic" */ - if(lp->is_basic[k]) { - lp->is_basic[lp->rows+j] = FALSE; - del_constraint(lp, k); - } - else - set_basisvar(lp, i, k); - del_column(lp, j); - lp->P1extraDim--; - } - lp->basis_valid = TRUE; - } - /* Otherwise we drive out the artificials by elimination pivoting */ - else - eliminate_artificials(lp, prow); - -#else - /* Indicate phase 2 with artificial variables by negating P1extraDim */ - lp->P1extraDim = my_flipsign(lp->P1extraDim); -#endif - } - - /* We must refactorize since the OF changes from phase 1 to phase 2 */ - set_action(&lp->spx_action, ACTION_REINVERT); - bfpfinal = TRUE; - } - - /* We are infeasible in phase 1 */ - else { - lp->spx_status = INFEASIBLE; - minit = ITERATE_MAJORMAJOR; - if(lp->spx_trace) - report(lp, NORMAL, "Model infeasible by primal simplex at iter %10.0f.\n", - (double) get_total_iter(lp)); - } - } - - /* Handle phase 1 optimality */ - else { - /* (Do nothing special) */ - } - - /* Check if we are still primal feasible; the default assumes that this check - is not necessary after the relaxed problem has been solved satisfactorily. */ - if((lp->bb_level <= 1) || (lp->improve & IMPROVE_BBSIMPLEX) /* || (lp->bb_rule & NODE_RCOSTFIXING) */) { /* NODE_RCOSTFIXING fix */ - set_action(&lp->piv_strategy, PRICE_FORCEFULL); - i = rowdual(lp, lp->rhs, FALSE, FALSE, NULL); - clear_action(&lp->piv_strategy, PRICE_FORCEFULL); - if(i > 0) { - lp->spx_status = LOSTFEAS; - if(lp->total_iter == 0) - report(lp, DETAILED, "primloop: Lost primal feasibility at iter %10.0f: will try to recover.\n", - (double) get_total_iter(lp)); - } - } - } - - /* Pivot row/col and update the inverse */ - if(is_action(lp->spx_action, ACTION_ITERATE)) { - lastnr = lp->var_basic[rownr]; - - if(refactRecent(lp) == AUTOMATIC) - minitcount = 0; - else if(minitcount > MAX_MINITUPDATES) { - recompute_solution(lp, INITSOL_USEZERO); - minitcount = 0; - } - minit = performiteration(lp, rownr, colnr, theta, primal, - (MYBOOL) (/*(candidatecount > 1) && */ - (stallaccept != AUTOMATIC)), - NULL, NULL, - pcol, NULL, NULL); - if(minit != ITERATE_MAJORMAJOR) - minitcount++; - - if((lp->spx_status == USERABORT) || (lp->spx_status == TIMEOUT)) - break; - else if(minit == ITERATE_MINORMAJOR) - continue; -#ifdef UsePrimalReducedCostUpdate - /* Do a fast update of the reduced costs in preparation for the next iteration */ - if(minit == ITERATE_MAJORMAJOR) - update_reducedcosts(lp, primal, lastnr, colnr, pcol, drow); -#endif - - /* Detect if an auxiliary variable has left the basis and delete it; if - the non-basic variable only changed bound (a "minor iteration"), the - basic artificial variable did not leave and there is nothing to do */ - if((minit == ITERATE_MAJORMAJOR) && (lastnr > lp->sum - abs(lp->P1extraDim))) { -#ifdef Paranoia - if(lp->is_basic[lastnr] || !lp->is_basic[colnr]) - report(lp, SEVERE, "primloop: Invalid basis indicator for variable %d at iter %10.0f.\n", - lastnr, (double) get_total_iter(lp)); -#endif - del_column(lp, lastnr-lp->rows); - if(lp->P1extraDim > 0) - lp->P1extraDim--; - else - lp->P1extraDim++; - if(lp->P1extraDim == 0) { - colnr = 0; - changedphase = TRUE; - stallMonitor_reset(lp); - } - } - } - - if(lp->spx_status == SWITCH_TO_DUAL) - ; - else if(!changedphase && lp->bfp_mustrefactorize(lp)) { -#ifdef ResetMinitOnReinvert - minit = ITERATE_MAJORMAJOR; -#endif - if(!invert(lp, INITSOL_USEZERO, bfpfinal)) - lp->spx_status = SINGULAR_BASIS; - bfpfinal = FALSE; - } - } - - /* Remove any remaining artificial variables (feasible or infeasible model) */ - lp->P1extraDim = abs(lp->P1extraDim); -/* if((lp->P1extraDim > 0) && (lp->spx_status != DEGENERATE)) { */ - if(lp->P1extraDim > 0) { - clear_artificials(lp); - if(lp->spx_status != OPTIMAL) - restore_basis(lp); - i = invert(lp, INITSOL_USEZERO, TRUE); - } -#ifdef Paranoia - if(!verify_basis(lp)) - report(lp, SEVERE, "primloop: Invalid basis detected due to internal error\n"); -#endif - - /* Switch to dual phase 1 simplex for MIP models during - B&B phases, since this is typically far more efficient */ -#ifdef ForceDualSimplexInBB - if((lp->bb_totalnodes == 0) && (MIP_count(lp) > 0) && - ((lp->simplex_strategy & SIMPLEX_Phase1_DUAL) == 0)) { - lp->simplex_strategy &= ~SIMPLEX_Phase1_PRIMAL; - lp->simplex_strategy += SIMPLEX_Phase1_DUAL; - } -#endif - -Finish: - stallMonitor_finish(lp); - multi_free(&(lp->multivars)); - FREE(prow); - FREE(pcol); - FREE(lp->bsolveVal); - - return(ok); -} /* primloop */ - -STATIC int dualloop(lprec *lp, MYBOOL dualfeasible, int dualinfeasibles[], REAL dualoffset) -{ - MYBOOL primal = FALSE, inP1extra, dualphase1 = FALSE, changedphase = TRUE, - pricerCanChange, minit, stallaccept, longsteps, - forceoutEQ = FALSE, bfpfinal = FALSE; - int i, colnr = 0, rownr = 0, lastnr = 0, - candidatecount = 0, minitcount = 0, -#ifdef FixInaccurateDualMinit - minitcolnr = 0, -#endif - ok = TRUE; - int *boundswaps = NULL; - LREAL theta = 0.0; - REAL epsvalue, xviolated, cviolated, - *prow = NULL, *pcol = NULL, - *drow = lp->drow; - int *nzprow = NULL, *workINT = NULL, - *nzdrow = lp->nzdrow; - - if(lp->spx_trace) - report(lp, DETAILED, "Entered dual simplex algorithm with feasibility %s.\n", - my_boolstr(dualfeasible)); - - /* Allocate work arrays */ - ok = allocREAL(lp, &prow, lp->sum + 1, TRUE) && - allocINT (lp, &nzprow, lp->sum + 1, FALSE) && - allocREAL(lp, &pcol, lp->rows + 1, TRUE); - if(!ok) - goto Finish; - - /* Set non-zero P1extraVal value to force dual feasibility when the dual - simplex is used as a phase 1 algorithm for the primal simplex. - The value will be reset when primal feasibility has been achieved, or - a dual non-feasibility has been encountered (no candidate for a first - leaving variable) */ - inP1extra = (MYBOOL) (dualoffset != 0); - if(inP1extra) { - set_OF_p1extra(lp, dualoffset); - simplexPricer(lp, (MYBOOL)!primal); - invert(lp, INITSOL_USEZERO, TRUE); - } - else - restartPricer(lp, (MYBOOL)!primal); - - /* Prepare dual long-step structures */ -#if 0 - longsteps = TRUE; -#elif 0 - longsteps = (MYBOOL) ((MIP_count(lp) > 0) && (lp->bb_level > 1)); -#elif 0 - longsteps = (MYBOOL) ((MIP_count(lp) > 0) && (lp->solutioncount >= 1)); -#else - longsteps = FALSE; -#endif -#ifdef UseLongStepDualPhase1 - longsteps = !dualfeasible && (MYBOOL) (dualinfeasibles != NULL); -#endif - - if(longsteps) { - lp->longsteps = multi_create(lp, TRUE); - ok = (lp->longsteps != NULL) && - multi_resize(lp->longsteps, MIN(lp->boundedvars+2, 11), 1, TRUE, TRUE); - if(!ok) - goto Finish; -#ifdef UseLongStepPruning - lp->longsteps->objcheck = TRUE; -#endif - boundswaps = multi_indexSet(lp->longsteps, FALSE); - } - - /* Do regular dual simplex variable initializations */ - lp->spx_status = RUNNING; - minit = ITERATE_MAJORMAJOR; - epsvalue = lp->epspivot; - - ok = stallMonitor_create(lp, TRUE, "dualloop"); - if(!ok) - goto Finish; - - lp->rejectpivot[0] = 0; - if(dualfeasible) - lp->simplex_mode = SIMPLEX_Phase2_DUAL; - else - lp->simplex_mode = SIMPLEX_Phase1_DUAL; - - /* Check if we have equality slacks in the basis and we should try to - drive them out in order to reduce chance of degeneracy in Phase 1. - forceoutEQ = FALSE : Only eliminate assured "good" violated - equality constraint slacks - AUTOMATIC: Seek more elimination of equality constraint - slacks (but not as aggressive as the rule - used in lp_solve v4.0 and earlier) - TRUE: Force remaining equality slacks out of the - basis */ - if(dualphase1 || inP1extra || - ((lp->fixedvars > 0) && is_anti_degen(lp, ANTIDEGEN_FIXEDVARS))) { - forceoutEQ = AUTOMATIC; - } -#if 1 - if(is_anti_degen(lp, ANTIDEGEN_DYNAMIC) && (bin_count(lp, TRUE)*2 > lp->columns)) { - switch (forceoutEQ) { - case FALSE: forceoutEQ = AUTOMATIC; - break; - /* case AUTOMATIC: forceoutEQ = TRUE; - break; - default: forceoutEQ = TRUE; */ - } - } -#endif - - while((lp->spx_status == RUNNING) && !userabort(lp, -1)) { - - /* Check if we have stalling (from numerics or degenerate cycling) */ - pricerCanChange = !dualphase1 && !inP1extra; - stallaccept = stallMonitor_check(lp, rownr, colnr, lastnr, minit, pricerCanChange, &forceoutEQ); - if(!stallaccept) - break; - - /* Store current LP index for reference at next iteration */ - changedphase = FALSE; - - /* Compute (pure) dual phase1 offsets / reduced costs if appropriate */ - dualphase1 &= (MYBOOL) (lp->simplex_mode == SIMPLEX_Phase1_DUAL); - if(longsteps && dualphase1 && !inP1extra) { - obtain_column(lp, dualinfeasibles[1], pcol, NULL, NULL); - i = 2; - for(i = 2; i <= dualinfeasibles[0]; i++) - mat_multadd(lp->matA, pcol, dualinfeasibles[i], 1.0); - /* Solve (note that solved pcol will be used instead of lp->rhs) */ - ftran(lp, pcol, NULL, lp->epsmachine); - } - - /* Do minor iterations (non-basic variable bound flips) for as - long as possible since this is a cheap way of iterating */ -#if (defined dual_Phase1PriceEqualities) || (defined dual_UseRejectionList) -RetryRow: -#endif - if(minit != ITERATE_MINORRETRY) { - i = 0; - do { - i++; - rownr = rowdual(lp, my_if(dualphase1, pcol, NULL), forceoutEQ, TRUE, &xviolated); - } while ((rownr == 0) && (i < partial_countBlocks(lp, (MYBOOL) !primal)) && - partial_blockStep(lp, (MYBOOL) !primal)); - } - - /* Make sure that we do not erroneously conclude that an infeasible model is optimal */ -#ifdef dual_UseRejectionList - if((rownr == 0) && (lp->rejectpivot[0] > 0)) { - lp->spx_status = INFEASIBLE; - if((lp->spx_trace && (lp->bb_totalnodes == 0)) || - (lp->bb_trace && (lp->bb_totalnodes > 0))) - report(lp, DETAILED, "The model is primal infeasible.\n"); - rownr = lp->rejectpivot[1]; - colnr = 0; - lp->rejectpivot[0] = 0; - ok = FALSE; - break; - } -#endif - - /* If we found a leaving variable, find a matching entering one */ - clear_action(&lp->spx_action, ACTION_ITERATE); - if(rownr > 0) { - colnr = coldual(lp, rownr, prow, nzprow, drow, nzdrow, - (MYBOOL) (dualphase1 && !inP1extra), - (MYBOOL) (minit == ITERATE_MINORRETRY), &candidatecount, &cviolated); - if(colnr < 0) { - minit = ITERATE_MAJORMAJOR; - continue; - } -#ifdef AcceptMarginalAccuracy - else if(xviolated+cviolated < lp->epspivot) { - if(lp->bb_trace || (lp->bb_totalnodes == 0)) - report(lp, DETAILED, "dualloop: Assuming convergence with reduced accuracy %g.\n", - MAX(xviolated, cviolated)); - rownr = 0; - colnr = 0; - } -#endif - /* Check if the long-dual found reason to prune the B&B tree */ - if(lp->spx_status == FATHOMED) - break; - } - else - colnr = 0; - - /* Process primal-infeasible row */ - if(rownr > 0) { - - if(colnr > 0) { -#ifdef Paranoia - if((rownr > lp->rows) || (colnr > lp->sum)) { - report(lp, SEVERE, "dualloop: Invalid row %d(%d) and column %d(%d) pair selected at iteration %.0f\n", - rownr, lp->rows, colnr-lp->columns, lp->columns, (double) get_total_iter(lp)); - lp->spx_status = UNKNOWNERROR; - break; - } -#endif - fsolve(lp, colnr, pcol, workINT, lp->epsmachine, 1.0, TRUE); - -#ifdef FixInaccurateDualMinit - /* Prevent bound flip-flops during minor iterations; used to detect - infeasibility after triggering of minor iteration accuracy management */ - if(colnr != minitcolnr) - minitcolnr = 0; -#endif - - /* Getting division by zero here; catch it and try to recover */ - if(pcol[rownr] == 0) { - if(lp->spx_trace) - report(lp, DETAILED, "dualloop: Attempt to divide by zero (pcol[%d])\n", rownr); - if(!refactRecent(lp)) { - report(lp, DETAILED, "...trying to recover by refactorizing basis.\n"); - set_action(&lp->spx_action, ACTION_REINVERT); - bfpfinal = FALSE; - } - else { - if(lp->bb_totalnodes == 0) - report(lp, DETAILED, "...cannot recover by refactorizing basis.\n"); - lp->spx_status = NUMFAILURE; - ok = FALSE; - } - } - else { - set_action(&lp->spx_action, ACTION_ITERATE); - lp->rejectpivot[0] = 0; - if(!lp->obj_in_basis) /* We must manually copy the reduced cost for RHS update */ - pcol[0] = my_chsign(!lp->is_lower[colnr], drow[colnr]); - theta = lp->bfp_prepareupdate(lp, rownr, colnr, pcol); - - /* Verify numeric accuracy of the basis factorization and change to - the "theoretically" correct version of the theta */ - if((lp->improve & IMPROVE_THETAGAP) && !refactRecent(lp) && - (my_reldiff(fabs(theta), fabs(prow[colnr])) > - lp->epspivot*10.0*log(2.0+50.0*lp->rows))) { /* This is my kludge - KE */ - set_action(&lp->spx_action, ACTION_REINVERT); - bfpfinal = TRUE; -#ifdef IncreasePivotOnReducedAccuracy - lp->epspivot = MIN(1.0e-4, lp->epspivot*2.0); -#endif - report(lp, DETAILED, "dualloop: Refactorizing at iter %.0f due to loss of accuracy.\n", - (double) get_total_iter(lp)); - } - theta = prow[colnr]; - compute_theta(lp, rownr, &theta, !lp->is_lower[colnr], 0, primal); - } - } - -#ifdef FixInaccurateDualMinit - /* Force reinvertion and try another row if we did not find a bound-violated leaving column */ - else if(!refactRecent(lp) && (minit != ITERATE_MAJORMAJOR) && (colnr != minitcolnr)) { - minitcolnr = colnr; - i = invert(lp, INITSOL_USEZERO, TRUE); - if((lp->spx_status == USERABORT) || (lp->spx_status == TIMEOUT)) - break; - else if(!i) { - lp->spx_status = SINGULAR_BASIS; - break; - } - minit = ITERATE_MAJORMAJOR; - continue; - } -#endif - - /* We may be infeasible, have lost dual feasibility, or simply have no valid entering - variable for the selected row. The strategy is to refactorize if we suspect numerical - problems and loss of dual feasibility; this is done if it has been a while since - refactorization. If not, first try to select a different row/leaving variable to - see if a valid entering variable can be found. Otherwise, determine this model - as infeasible. */ - else { - - /* As a first option, try to recover from any numerical trouble by refactorizing */ - if(!refactRecent(lp)) { - set_action(&lp->spx_action, ACTION_REINVERT); - bfpfinal = TRUE; - } - -#ifdef dual_UseRejectionList - /* Check for pivot size issues */ - else if(lp->rejectpivot[0] < DEF_MAXPIVOTRETRY) { - lp->spx_status = RUNNING; - lp->rejectpivot[0]++; - lp->rejectpivot[lp->rejectpivot[0]] = rownr; - if(lp->bb_totalnodes == 0) - report(lp, DETAILED, "...trying to find another pivot row!\n"); - goto RetryRow; - } -#endif - /* Check if we may have lost dual feasibility if we also did phase 1 here */ - else if(dualphase1 && (dualoffset != 0)) { - lp->spx_status = LOSTFEAS; - if((lp->spx_trace && (lp->bb_totalnodes == 0)) || - (lp->bb_trace && (lp->bb_totalnodes > 0))) - report(lp, DETAILED, "dualloop: Model lost dual feasibility.\n"); - ok = FALSE; - break; - } - - /* Otherwise just determine that we are infeasible */ - else { - if(lp->spx_status == RUNNING) { -#if 1 - if(xviolated < lp->epspivot) { - if(lp->bb_trace || (lp->bb_totalnodes == 0)) - report(lp, NORMAL, "The model is primal optimal, but marginally infeasible.\n"); - lp->spx_status = OPTIMAL; - break; - } -#endif - lp->spx_status = INFEASIBLE; - if((lp->spx_trace && (lp->bb_totalnodes == 0)) || - (lp->bb_trace && (lp->bb_totalnodes > 0))) - report(lp, DETAILED, "The model is primal infeasible.\n"); - } - ok = FALSE; - break; - } - } - } - - /* Make sure that we enter the primal simplex with a high quality solution */ - else if(inP1extra && !refactRecent(lp) && is_action(lp->improve, IMPROVE_INVERSE)) { - set_action(&lp->spx_action, ACTION_REINVERT); - bfpfinal = TRUE; - } - - /* High quality solution with no leaving candidates available ... */ - else { - - bfpfinal = TRUE; - -#ifdef dual_RemoveBasicFixedVars - /* See if we should try to eliminate basic fixed variables; - can be time-consuming for some models */ - if(inP1extra && (colnr == 0) && (lp->fixedvars > 0) && is_anti_degen(lp, ANTIDEGEN_FIXEDVARS)) { - report(lp, DETAILED, "dualloop: Trying to pivot out %d fixed basic variables at iter %.0f\n", - lp->fixedvars, (double) get_total_iter(lp)); - rownr = 0; - while(lp->fixedvars > 0) { - rownr = findBasicFixedvar(lp, rownr, TRUE); - if(rownr == 0) { - colnr = 0; - break; - } - colnr = find_rowReplacement(lp, rownr, prow, nzprow); - if(colnr > 0) { - theta = 0; - performiteration(lp, rownr, colnr, theta, TRUE, FALSE, prow, NULL, - NULL, NULL, NULL); - lp->fixedvars--; - } - } - } -#endif - - /* Check if we are INFEASIBLE for the case that the dual is used - as phase 1 before the primal simplex phase 2 */ - if(inP1extra && (colnr < 0) && !isPrimalFeasible(lp, lp->epsprimal, NULL, NULL)) { - if(lp->bb_totalnodes == 0) { - if(dualfeasible) - report(lp, DETAILED, "The model is primal infeasible and dual feasible.\n"); - else - report(lp, DETAILED, "The model is primal infeasible and dual unbounded.\n"); - } - set_OF_p1extra(lp, 0); - inP1extra = FALSE; - set_action(&lp->spx_action, ACTION_REINVERT); - lp->spx_status = INFEASIBLE; - lp->simplex_mode = SIMPLEX_UNDEFINED; - ok = FALSE; - } - - /* Check if we are FEASIBLE (and possibly also optimal) for the case that the - dual is used as phase 1 before the primal simplex phase 2 */ - else if(inP1extra) { - - /* Set default action; force an update of the rhs vector, adjusted for - the new P1extraVal=0 (set here so that usermessage() behaves properly) */ - if(lp->bb_totalnodes == 0) { - report(lp, NORMAL, "Found feasibility by dual simplex after %10.0f iter.\n", - (double) get_total_iter(lp)); - if((lp->usermessage != NULL) && (lp->msgmask & MSG_LPFEASIBLE)) - lp->usermessage(lp, lp->msghandle, MSG_LPFEASIBLE); - } - set_OF_p1extra(lp, 0); - inP1extra = FALSE; - set_action(&lp->spx_action, ACTION_REINVERT); - -#if 1 - /* Optionally try another dual loop, if so selected by the user */ - if((lp->simplex_strategy & SIMPLEX_DUAL_PRIMAL) && (lp->fixedvars == 0)) - lp->spx_status = SWITCH_TO_PRIMAL; -#endif - changedphase = TRUE; - - } - - /* We are primal feasible and also optimal if we were in phase 2 */ - else { - - lp->simplex_mode = SIMPLEX_Phase2_DUAL; - - /* Check if we still have equality slacks stuck in the basis; drive them out? */ - if((lp->fixedvars > 0) && (lp->bb_totalnodes == 0)) { -#ifdef dual_Phase1PriceEqualities - if(forceoutEQ != TRUE) { - forceoutEQ = TRUE; - goto RetryRow; - } -#endif -#ifdef Paranoia - report(lp, NORMAL, -#else - report(lp, DETAILED, -#endif - "Found dual solution with %d fixed slack variables left basic.\n", - lp->fixedvars); - } - /* Check if we are still dual feasible; the default assumes that this check - is not necessary after the relaxed problem has been solved satisfactorily. */ - colnr = 0; - if((dualoffset != 0) || (lp->bb_level <= 1) || (lp->improve & IMPROVE_BBSIMPLEX) || (lp->bb_rule & NODE_RCOSTFIXING)) { /* NODE_RCOSTFIXING fix */ - set_action(&lp->piv_strategy, PRICE_FORCEFULL); - colnr = colprim(lp, drow, nzdrow, FALSE, 1, &candidatecount, FALSE, NULL); - clear_action(&lp->piv_strategy, PRICE_FORCEFULL); - if((dualoffset == 0) && (colnr > 0)) { - lp->spx_status = LOSTFEAS; - if(lp->total_iter == 0) - report(lp, DETAILED, "Recovering lost dual feasibility at iter %10.0f.\n", - (double) get_total_iter(lp)); - break; - } - } - - if(colnr == 0) - lp->spx_status = OPTIMAL; - else { - lp->spx_status = SWITCH_TO_PRIMAL; - if(lp->total_iter == 0) - report(lp, DETAILED, "Use primal simplex for finalization at iter %10.0f.\n", - (double) get_total_iter(lp)); - } - if((lp->total_iter == 0) && (lp->spx_status == OPTIMAL)) - report(lp, DETAILED, "Optimal solution with dual simplex at iter %10.0f.\n", - (double) get_total_iter(lp)); - } - - /* Determine if we are ready to break out of the loop */ - if(!changedphase) - break; - } - - /* Check if we are allowed to iterate on the chosen column and row */ - if(is_action(lp->spx_action, ACTION_ITERATE)) { - - lastnr = lp->var_basic[rownr]; - if(refactRecent(lp) == AUTOMATIC) - minitcount = 0; - else if(minitcount > MAX_MINITUPDATES) { - recompute_solution(lp, INITSOL_USEZERO); - minitcount = 0; - } - minit = performiteration(lp, rownr, colnr, theta, primal, - (MYBOOL) (/*(candidatecount > 1) && */ - (stallaccept != AUTOMATIC)), - prow, nzprow, - pcol, NULL, boundswaps); - - /* Check if we should abandon iterations on finding that there is no - hope that this branch can improve on the incumbent B&B solution */ - if(!lp->is_strongbranch && (lp->solutioncount >= 1) && !lp->spx_perturbed && !inP1extra && - bb_better(lp, OF_WORKING, OF_TEST_WE)) { - lp->spx_status = FATHOMED; - ok = FALSE; - break; - } - - if(minit != ITERATE_MAJORMAJOR) - minitcount++; - - /* Update reduced costs for (pure) dual long-step phase 1 */ - if(longsteps && dualphase1 && !inP1extra) { - dualfeasible = isDualFeasible(lp, lp->epsprimal, NULL, dualinfeasibles, NULL); - if(dualfeasible) { - dualphase1 = FALSE; - changedphase = TRUE; - lp->simplex_mode = SIMPLEX_Phase2_DUAL; - } - } -#ifdef UseDualReducedCostUpdate - /* Do a fast update of reduced costs in preparation for the next iteration */ - else if(minit == ITERATE_MAJORMAJOR) - update_reducedcosts(lp, primal, lastnr, colnr, prow, drow); -#endif - if((minit == ITERATE_MAJORMAJOR) && (lastnr <= lp->rows) && is_fixedvar(lp, lastnr)) - lp->fixedvars--; - } - - /* Refactorize if required to */ - if(lp->bfp_mustrefactorize(lp)) { - if(invert(lp, INITSOL_USEZERO, bfpfinal)) { - -#if 0 - /* Verify dual feasibility in case we are attempting the extra dual loop */ - if(changedphase && (dualoffset != 0) && !inP1extra && (lp->spx_status != SWITCH_TO_PRIMAL)) { -#if 1 - if(!isDualFeasible(lp, lp->epsdual, &colnr, NULL, NULL)) { -#else - set_action(&lp->piv_strategy, PRICE_FORCEFULL); - colnr = colprim(lp, drow, nzdrow, FALSE, 1, &candidatecount, FALSE, NULL); - clear_action(&lp->piv_strategy, PRICE_FORCEFULL); - if(colnr > 0) { -#endif - lp->spx_status = SWITCH_TO_PRIMAL; - colnr = 0; - } - } -#endif - - bfpfinal = FALSE; -#ifdef ResetMinitOnReinvert - minit = ITERATE_MAJORMAJOR; -#endif - } - else - lp->spx_status = SINGULAR_BASIS; - } - } - -Finish: - stallMonitor_finish(lp); - multi_free(&(lp->longsteps)); - FREE(prow); - FREE(nzprow); - FREE(pcol); - - return(ok); -} - -STATIC int spx_run(lprec *lp, MYBOOL validInvB) -{ - int i, j, singular_count, lost_feas_count, *infeasibles = NULL, *boundflip_count; - MYBOOL primalfeasible, dualfeasible, lost_feas_state, isbb; - REAL primaloffset = 0, dualoffset = 0; - - lp->current_iter = 0; - lp->current_bswap = 0; - lp->spx_status = RUNNING; - lp->bb_status = lp->spx_status; - lp->P1extraDim = 0; - set_OF_p1extra(lp, 0); - singular_count = 0; - lost_feas_count = 0; - lost_feas_state = FALSE; - lp->simplex_mode = SIMPLEX_DYNAMIC; - - /* Compute the number of fixed basic and bounded variables (used in long duals) */ - lp->fixedvars = 0; - lp->boundedvars = 0; - for(i = 1; i <= lp->rows; i++) { - j = lp->var_basic[i]; - if((j <= lp->rows) && is_fixedvar(lp, j)) - lp->fixedvars++; - if((lp->upbo[i] < lp->infinite) && (lp->upbo[i] > lp->epsprimal)) - lp->boundedvars++; - } - for(; i <= lp->sum; i++){ - if((lp->upbo[i] < lp->infinite) && (lp->upbo[i] > lp->epsprimal)) - lp->boundedvars++; - } -#ifdef UseLongStepDualPhase1 - allocINT(lp, &infeasibles, lp->columns + 1, FALSE); - infeasibles[0] = 0; -#endif - - /* Reinvert for initialization, if necessary */ - isbb = (MYBOOL) ((MIP_count(lp) > 0) && (lp->bb_level > 1)); - if(is_action(lp->spx_action, ACTION_REINVERT)) { - if(isbb && (lp->bb_bounds->nodessolved == 0)) -/* if(isbb && (lp->bb_basis->pivots == 0)) */ - recompute_solution(lp, INITSOL_SHIFTZERO); - else { - i = my_if(is_action(lp->spx_action, ACTION_REBASE), INITSOL_SHIFTZERO, INITSOL_USEZERO); - invert(lp, (MYBOOL) i, TRUE); - } - } - else if(is_action(lp->spx_action, ACTION_REBASE)) - recompute_solution(lp, INITSOL_SHIFTZERO); - - /* Optionally try to do bound flips to obtain dual feasibility */ - if(is_action(lp->improve, IMPROVE_DUALFEAS) || (lp->rows == 0)) - boundflip_count = &i; - else - boundflip_count = NULL; - - /* Loop for as long as is needed */ - while(lp->spx_status == RUNNING) { - - /* Check for dual and primal feasibility */ - dualfeasible = isbb || - isDualFeasible(lp, lp->epsprimal, boundflip_count, infeasibles, &dualoffset); - - /* Recompute if the dual feasibility check included bound flips */ - if(is_action(lp->spx_action, ACTION_RECOMPUTE)) - recompute_solution(lp, INITSOL_USEZERO); - primalfeasible = isPrimalFeasible(lp, lp->epsprimal, NULL, &primaloffset); - - if(userabort(lp, -1)) - break; - - if(lp->spx_trace) { - if(primalfeasible) - report(lp, NORMAL, "Start at primal feasible basis\n"); - else if(dualfeasible) - report(lp, NORMAL, "Start at dual feasible basis\n"); - else if(lost_feas_count > 0) - report(lp, NORMAL, "Continuing at infeasible basis\n"); - else - report(lp, NORMAL, "Start at infeasible basis\n"); - } - - /* Now do the simplex magic */ - if(((lp->simplex_strategy & SIMPLEX_Phase1_DUAL) == 0) || - ((MIP_count(lp) > 0) && (lp->total_iter == 0) && - is_presolve(lp, PRESOLVE_REDUCEMIP))) { - if(!lost_feas_state && primalfeasible && ((lp->simplex_strategy & SIMPLEX_Phase2_DUAL) > 0)) - lp->spx_status = SWITCH_TO_DUAL; - else - primloop(lp, primalfeasible, 0.0); - if(lp->spx_status == SWITCH_TO_DUAL) - dualloop(lp, TRUE, NULL, 0.0); - } - else { - if(!lost_feas_state && primalfeasible && ((lp->simplex_strategy & SIMPLEX_Phase2_PRIMAL) > 0)) - lp->spx_status = SWITCH_TO_PRIMAL; - else - dualloop(lp, dualfeasible, infeasibles, dualoffset); - if(lp->spx_status == SWITCH_TO_PRIMAL) - primloop(lp, TRUE, 0.0); - } - - /* Check for simplex outcomes that always involve breaking out of the loop; - this includes optimality, unboundedness, pure infeasibility (i.e. not - loss of feasibility), numerical failure and perturbation-based degeneracy - handling */ - i = lp->spx_status; - primalfeasible = (MYBOOL) (i == OPTIMAL); - if(primalfeasible || (i == UNBOUNDED)) - break; - else if(((i == INFEASIBLE) && is_anti_degen(lp, ANTIDEGEN_INFEASIBLE)) || - ((i == LOSTFEAS) && is_anti_degen(lp, ANTIDEGEN_LOSTFEAS)) || - ((i == NUMFAILURE) && is_anti_degen(lp, ANTIDEGEN_NUMFAILURE)) || - ((i == DEGENERATE) && is_anti_degen(lp, ANTIDEGEN_STALLING))) { - /* Check if we should not loop here, but do perturbations */ - if((lp->bb_level <= 1) || is_anti_degen(lp, ANTIDEGEN_DURINGBB)) - break; - - /* Assume that accuracy during B&B is high and that infeasibility is "real" */ -#ifdef AssumeHighAccuracyInBB - if((lp->bb_level > 1) && (i == INFEASIBLE)) - break; -#endif - } - - /* Check for outcomes that may involve trying another simplex loop */ - if(lp->spx_status == SINGULAR_BASIS) { - lost_feas_state = FALSE; - singular_count++; - if(singular_count >= DEF_MAXSINGULARITIES) { - report(lp, IMPORTANT, "spx_run: Failure due to too many singular bases.\n"); - lp->spx_status = NUMFAILURE; - break; - } - if(lp->spx_trace || (lp->verbose > DETAILED)) - report(lp, NORMAL, "spx_run: Singular basis; attempting to recover.\n"); - lp->spx_status = RUNNING; - /* Singular pivots are simply skipped by the inversion, leaving a row's - slack variable in the basis instead of the singular user variable. */ - } - else { - lost_feas_state = (MYBOOL) (lp->spx_status == LOSTFEAS); -#if 0 - /* Optionally handle loss of numerical accuracy as loss of feasibility, - but only attempt a single loop to try to recover from this. */ - lost_feas_state |= (MYBOOL) ((lp->spx_status == NUMFAILURE) && (lost_feas_count < 1)); -#endif - if(lost_feas_state) { - lost_feas_count++; - if(lost_feas_count < DEF_MAXSINGULARITIES) { - report(lp, DETAILED, "spx_run: Recover lost feasibility at iter %10.0f.\n", - (double) get_total_iter(lp)); - lp->spx_status = RUNNING; - } - else { - report(lp, IMPORTANT, "spx_run: Lost feasibility %d times - iter %10.0f and %9.0f nodes.\n", - lost_feas_count, (double) get_total_iter(lp), (double) lp->bb_totalnodes); - lp->spx_status = NUMFAILURE; - } - } - } - } - - /* Update iteration tallies before returning */ - lp->total_iter += lp->current_iter; - lp->current_iter = 0; - lp->total_bswap += lp->current_bswap; - lp->current_bswap = 0; - FREE(infeasibles); - - return(lp->spx_status); -} /* spx_run */ - -lprec *make_lag(lprec *lpserver) -{ - int i; - lprec *hlp; - MYBOOL ret; - REAL *duals; - - /* Create a Lagrangean solver instance */ - hlp = make_lp(0, lpserver->columns); - - if(hlp != NULL) { - - /* First create and core variable data */ - set_sense(hlp, is_maxim(lpserver)); - /*hlp->lag_bound = lpserver->bb_limitOF;*/ - for(i = 1; i <= lpserver->columns; i++) { - set_mat(hlp, 0, i, get_mat(lpserver, 0, i)); - if(is_binary(lpserver, i)) - set_binary(hlp, i, TRUE); - else { - set_int(hlp, i, is_int(lpserver, i)); - set_bounds(hlp, i, get_lowbo(lpserver, i), get_upbo(lpserver, i)); - } - } - /* Then fill data for the Lagrangean constraints */ - hlp->matL = lpserver->matA; - inc_lag_space(hlp, lpserver->rows, TRUE); - ret = get_ptr_sensitivity_rhs(hlp, &duals, NULL, NULL); - for(i = 1; i <= lpserver->rows; i++) { - hlp->lag_con_type[i] = get_constr_type(lpserver, i); - hlp->lag_rhs[i] = lpserver->orig_rhs[i]; - hlp->lambda[i] = (ret) ? duals[i - 1] : 0.0; - } - } - - return(hlp); -} - -STATIC int heuristics(lprec *lp, int mode) -/* Initialize / bound a MIP problem */ -{ - lprec *hlp; - int status = PROCFAIL; - - if(lp->bb_level > 1) - return( status ); - - status = RUNNING; - lp->bb_limitOF = my_chsign(is_maxim(lp), -lp->infinite); - if(FALSE && (lp->int_vars > 0)) { - - /* 1. Copy the problem into a new relaxed instance, extracting Lagrangean constraints */ - hlp = make_lag(lp); - - /* 2. Run the Lagrangean relaxation */ - status = solve(hlp); - - /* 3. Copy the key results (bound) into the original problem */ - lp->bb_heuristicOF = hlp->best_solution[0]; - - /* 4. Delete the helper heuristic */ - hlp->matL = NULL; - delete_lp(hlp); - } - - lp->timeheuristic = timeNow(); - return( status ); -} - -STATIC int lag_solve(lprec *lp, REAL start_bound, int num_iter) -{ - int i, j, citer, nochange, oldpresolve; - MYBOOL LagFeas, AnyFeas, Converged, same_basis; - REAL *OrigObj, *ModObj, *SubGrad, *BestFeasSol; - REAL Zub, Zlb, Znow, Zprev, Zbest, rhsmod, hold; - REAL Phi, StepSize = 0.0, SqrsumSubGrad; - - /* Make sure we have something to work with */ - if(lp->spx_status != OPTIMAL) { - lp->lag_status = NOTRUN; - return( lp->lag_status ); - } - - /* Allocate iteration arrays */ - if(!allocREAL(lp, &OrigObj, lp->columns + 1, FALSE) || - !allocREAL(lp, &ModObj, lp->columns + 1, TRUE) || - !allocREAL(lp, &SubGrad, get_Lrows(lp) + 1, TRUE) || - !allocREAL(lp, &BestFeasSol, lp->sum + 1, TRUE)) { - lp->lag_status = NOMEMORY; - return( lp->lag_status ); - } - lp->lag_status = RUNNING; - - /* Prepare for Lagrangean iterations using results from relaxed problem */ - oldpresolve = lp->do_presolve; - lp->do_presolve = PRESOLVE_NONE; - push_basis(lp, NULL, NULL, NULL); - - /* Initialize variables (assume minimization problem in overall structure) */ - Zlb = lp->best_solution[0]; - Zub = start_bound; - Zbest = Zub; - Znow = Zlb; - Zprev = lp->infinite; - rhsmod = 0; - - Phi = DEF_LAGCONTRACT; /* In the range 0-2.0 to guarantee convergence */ -/* Phi = 0.15; */ - LagFeas = FALSE; - Converged= FALSE; - AnyFeas = FALSE; - citer = 0; - nochange = 0; - - /* Initialize reference and solution vectors; don't bother about the - original OF offset since we are maintaining an offset locally. */ - -/* #define DirectOverrideOF */ - - get_row(lp, 0, OrigObj); -#ifdef DirectOverrideOF - set_OF_override(lp, ModObj); -#endif - OrigObj[0] = get_rh(lp, 0); - for(i = 1 ; i <= get_Lrows(lp); i++) - lp->lambda[i] = 0; - - /* Iterate to convergence, failure or user-specified termination */ - while((lp->lag_status == RUNNING) && (citer < num_iter)) { - - citer++; - - /* Compute constraint feasibility gaps and associated sum of squares, - and determine feasibility over the Lagrangean constraints; - SubGrad is the subgradient, which here is identical to the slack. */ - LagFeas = TRUE; - Converged = TRUE; - SqrsumSubGrad = 0; - for(i = 1; i <= get_Lrows(lp); i++) { - hold = lp->lag_rhs[i]; - for(j = 1; j <= lp->columns; j++) - hold -= mat_getitem(lp->matL, i, j) * lp->best_solution[lp->rows + j]; - if(LagFeas) { - if(lp->lag_con_type[i] == EQ) { - if(fabs(hold) > lp->epsprimal) - LagFeas = FALSE; - } - else if(hold < -lp->epsprimal) - LagFeas = FALSE; - } - /* Test for convergence and update */ - if(Converged && (fabs(my_reldiff(hold , SubGrad[i])) > /* lp->lag_accept */ DEF_LAGACCEPT)) - Converged = FALSE; - SubGrad[i] = hold; - SqrsumSubGrad += hold * hold; - } - SqrsumSubGrad = sqrt(SqrsumSubGrad); -#if 1 - Converged &= LagFeas; -#endif - if(Converged) - break; - - /* Modify step parameters and initialize ahead of next iteration */ - Znow = lp->best_solution[0] - rhsmod; - if(Znow > Zub) { - /* Handle exceptional case where we overshoot */ - Phi *= DEF_LAGCONTRACT; - StepSize *= (Zub-Zlb) / (Znow-Zlb); - } - else -#define LagBasisContract -#ifdef LagBasisContract -/* StepSize = Phi * (Zub - Znow) / SqrsumSubGrad; */ - StepSize = Phi * (2-DEF_LAGCONTRACT) * (Zub - Znow) / SqrsumSubGrad; -#else - StepSize = Phi * (Zub - Znow) / SqrsumSubGrad; -#endif - - /* Compute the new dual price vector (Lagrangean multipliers, lambda) */ - for(i = 1; i <= get_Lrows(lp); i++) { - lp->lambda[i] += StepSize * SubGrad[i]; - if((lp->lag_con_type[i] != EQ) && (lp->lambda[i] > 0)) { - /* Handle case where we overshoot and need to correct (see above) */ - if(Znow < Zub) - lp->lambda[i] = 0; - } - } -/* normalizeVector(lp->lambda, get_Lrows(lp)); */ - - /* Save the current vector if it is better */ - if(LagFeas && (Znow < Zbest)) { - - /* Recompute the objective function value in terms of the original values */ - MEMCOPY(BestFeasSol, lp->best_solution, lp->sum+1); - hold = OrigObj[0]; - for(i = 1; i <= lp->columns; i++) - hold += lp->best_solution[lp->rows + i] * OrigObj[i]; - BestFeasSol[0] = hold; - if(lp->lag_trace) - report(lp, NORMAL, "lag_solve: Improved feasible solution at iteration %d of %g\n", - citer, hold); - - /* Reset variables */ - Zbest = Znow; - AnyFeas = TRUE; - nochange = 0; - } - else if(Znow == Zprev) { - nochange++; - if(nochange > LAG_SINGULARLIMIT) { - Phi *= 0.5; - nochange = 0; - } - } - Zprev = Znow; - - /* Recompute the objective function values for the next iteration */ - for(j = 1; j <= lp->columns; j++) { - hold = 0; - for(i = 1; i <= get_Lrows(lp); i++) - hold += lp->lambda[i] * mat_getitem(lp->matL, i, j); - ModObj[j] = OrigObj[j] - my_chsign(is_maxim(lp), hold); -#ifndef DirectOverrideOF - set_mat(lp, 0, j, ModObj[j]); -#endif - } - - /* Recompute the fixed part of the new objective function */ - rhsmod = my_chsign(is_maxim(lp), get_rh(lp, 0)); - for(i = 1; i <= get_Lrows(lp); i++) - rhsmod += lp->lambda[i] * lp->lag_rhs[i]; - - /* Print trace/debugging information, if specified */ - if(lp->lag_trace) { - report(lp, IMPORTANT, "Zub: %10g Zlb: %10g Stepsize: %10g Phi: %10g Feas %d\n", - (double) Zub, (double) Zlb, (double) StepSize, (double) Phi, LagFeas); - for(i = 1; i <= get_Lrows(lp); i++) - report(lp, IMPORTANT, "%3d SubGrad %10g lambda %10g\n", - i, (double) SubGrad[i], (double) lp->lambda[i]); - if(lp->sum < 20) - print_lp(lp); - } - - /* Solve the Lagrangean relaxation, handle failures and compute - the Lagrangean objective value, if successful */ - i = spx_solve(lp); - if(lp->spx_status == UNBOUNDED) { - if(lp->lag_trace) { - report(lp, NORMAL, "lag_solve: Unbounded solution encountered with this OF:\n"); - for(i = 1; i <= lp->columns; i++) - report(lp, NORMAL, RESULTVALUEMASK " ", (double) ModObj[i]); - } - goto Leave; - } - else if((lp->spx_status == NUMFAILURE) || (lp->spx_status == PROCFAIL) || - (lp->spx_status == USERABORT) || (lp->spx_status == TIMEOUT) || - (lp->spx_status == INFEASIBLE)) { - lp->lag_status = lp->spx_status; - } - - /* Compare optimal bases and contract if we have basis stationarity */ -#ifdef LagBasisContract - same_basis = compare_basis(lp); - if(LagFeas && - !same_basis) { - pop_basis(lp, FALSE); - push_basis(lp, NULL, NULL, NULL); - Phi *= DEF_LAGCONTRACT; - } - if(lp->lag_trace) { - report(lp, DETAILED, "lag_solve: Simplex status code %d, same basis %s\n", - lp->spx_status, my_boolstr(same_basis)); - print_solution(lp, 1); - } -#endif - } - - /* Transfer solution values */ - if(AnyFeas) { - /*lp->lag_bound = my_chsign(is_maxim(lp), Zbest);*/ - for(i = 0; i <= lp->sum; i++) - lp->solution[i] = BestFeasSol[i]; - transfer_solution(lp, TRUE); - if(!is_maxim(lp)) - for(i = 1; i <= get_Lrows(lp); i++) - lp->lambda[i] = my_flipsign(lp->lambda[i]); - } - - /* Do standard postprocessing */ -Leave: - - /* Set status variables and report */ - if(citer >= num_iter) { - if(AnyFeas) - lp->lag_status = FEASFOUND; - else - lp->lag_status = NOFEASFOUND; - } - else - lp->lag_status = lp->spx_status; - if(lp->lag_status == OPTIMAL) { - report(lp, NORMAL, "\nLagrangean convergence achieved in %d iterations\n", citer); - i = check_solution(lp, lp->columns, - lp->best_solution, lp->orig_upbo, lp->orig_lowbo, lp->epssolution); - } - else { - report(lp, NORMAL, "\nUnsatisfactory convergence achieved over %d Lagrangean iterations.\n", - citer); - if(AnyFeas) - report(lp, NORMAL, "The best feasible Lagrangean objective function value was %g\n", - lp->best_solution[0]); - } - - /* Restore the original objective function */ -#ifdef DirectOverrideOF - set_OF_override(lp, NULL); -#else - for(i = 1; i <= lp->columns; i++) - set_mat(lp, 0, i, OrigObj[i]); -#endif - - /* ... and then free memory */ - FREE(BestFeasSol); - FREE(SubGrad); - FREE(OrigObj); - FREE(ModObj); - pop_basis(lp, FALSE); - - lp->do_presolve = oldpresolve; - - return( lp->lag_status ); -} - -STATIC int spx_solve(lprec *lp) -{ - int status; - MYBOOL iprocessed; - - lp->total_iter = 0; - lp->total_bswap = 0; - lp->perturb_count = 0; - lp->bb_maxlevel = 1; - lp->bb_totalnodes = 0; - lp->bb_improvements = 0; - lp->bb_strongbranches= 0; - lp->is_strongbranch = FALSE; - lp->bb_level = 0; - lp->bb_solutionlevel = 0; - lp->best_solution[0] = my_chsign(is_maxim(lp), lp->infinite); - if(lp->invB != NULL) - lp->bfp_restart(lp); - - lp->spx_status = presolve(lp); - if(lp->spx_status == PRESOLVED) { - status = lp->spx_status; - goto Reconstruct; - } - else if(lp->spx_status != RUNNING) - goto Leave; - - iprocessed = !lp->wasPreprocessed; - if(!preprocess(lp) || userabort(lp, -1)) - goto Leave; - - if(mat_validate(lp->matA)) { - - /* Do standard initializations */ - lp->solutioncount = 0; - lp->real_solution = lp->infinite; - set_action(&lp->spx_action, ACTION_REBASE | ACTION_REINVERT); - lp->bb_break = FALSE; - - /* Do the call to the real underlying solver (note that - run_BB is replaceable with any compatible MIP solver) */ - status = run_BB(lp); - - /* Restore modified problem */ - if(iprocessed) - postprocess(lp); - - /* Restore data related to presolve (mainly a placeholder as of v5.1) */ -Reconstruct: - if(!postsolve(lp, status)) - report(lp, SEVERE, "spx_solve: Failure during postsolve.\n"); - - goto Leave; - } - - /* If we get here, mat_validate(lp) failed. */ - if(lp->bb_trace || lp->spx_trace) - report(lp, CRITICAL, "spx_solve: The current LP seems to be invalid\n"); - lp->spx_status = NUMFAILURE; - -Leave: - lp->timeend = timeNow(); - - if((lp->lag_status != RUNNING) && (lp->invB != NULL)) { - int itemp; - REAL test; - - itemp = lp->bfp_nonzeros(lp, TRUE); - test = 100; - if(lp->total_iter > 0) - test *= (REAL) lp->total_bswap/lp->total_iter; - report(lp, NORMAL, "\n "); - report(lp, NORMAL, "MEMO: lp_solve version %d.%d.%d.%d for %d bit OS, with %d bit REAL variables.\n", - MAJORVERSION, MINORVERSION, RELEASE, BUILD, 8*sizeof(void *), 8*sizeof(REAL)); - report(lp, NORMAL, " In the total iteration count %.0f, %.0f (%.1f%%) were bound flips.\n", - (double) lp->total_iter, (double) lp->total_bswap, test); - report(lp, NORMAL, " There were %d refactorizations, %d triggered by time and %d by density.\n", - lp->bfp_refactcount(lp, BFP_STAT_REFACT_TOTAL), - lp->bfp_refactcount(lp, BFP_STAT_REFACT_TIMED), - lp->bfp_refactcount(lp, BFP_STAT_REFACT_DENSE)); - report(lp, NORMAL, " ... on average %.1f major pivots per refactorization.\n", - get_refactfrequency(lp, TRUE)); - report(lp, NORMAL, " The largest [%s] fact(B) had %d NZ entries, %.1fx largest basis.\n", - lp->bfp_name(), itemp, lp->bfp_efficiency(lp)); - if(lp->perturb_count > 0) - report(lp, NORMAL, " The bounds were relaxed via perturbations %d times.\n", - lp->perturb_count); - if(MIP_count(lp) > 0) { - if(lp->bb_solutionlevel > 0) - report(lp, NORMAL, " The maximum B&B level was %d, %.1fx MIP order, %d at the optimal solution.\n", - lp->bb_maxlevel, (double) lp->bb_maxlevel / (MIP_count(lp)+lp->int_vars), lp->bb_solutionlevel); - else - report(lp, NORMAL, " The maximum B&B level was %d, %.1fx MIP order, with %.0f nodes explored.\n", - lp->bb_maxlevel, (double) lp->bb_maxlevel / (MIP_count(lp)+lp->int_vars), (double) get_total_nodes(lp)); - if(GUB_count(lp) > 0) - report(lp, NORMAL, " %d general upper-bounded (GUB) structures were employed during B&B.\n", - GUB_count(lp)); - } - report(lp, NORMAL, " The constraint matrix inf-norm is %g, with a dynamic range of %g.\n", - lp->matA->infnorm, lp->matA->dynrange); - report(lp, NORMAL, " Time to load data was %.3f seconds, presolve used %.3f seconds,\n", - lp->timestart-lp->timecreate, lp->timepresolved-lp->timestart); - report(lp, NORMAL, " ... %.3f seconds in simplex solver, in total %.3f seconds.\n", - lp->timeend-lp->timepresolved, lp->timeend-lp->timecreate); - } - return( lp->spx_status ); - -} /* spx_solve */ - -int lin_solve(lprec *lp) -{ - int status = NOTRUN; - - /* Don't do anything in case of an empty model */ - lp->lag_status = NOTRUN; - /* if(get_nonzeros(lp) == 0) { */ - if(lp->columns == 0) { - default_basis(lp); - lp->spx_status = NOTRUN; - return( /* OPTIMAL */ lp->spx_status); - } - - /* Otherwise reset selected arrays before solving */ - unset_OF_p1extra(lp); - free_duals(lp); - FREE(lp->drow); - FREE(lp->nzdrow); - if(lp->bb_cuttype != NULL) - freecuts_BB(lp); - - /* Reset/initialize timers */ - lp->timestart = timeNow(); - lp->timeheuristic = 0; - lp->timepresolved = 0; - lp->timeend = 0; - - /* Do heuristics ahead of solving the model */ - if(heuristics(lp, AUTOMATIC) != RUNNING) - return( INFEASIBLE ); - - /* Solve the full, prepared model */ - status = spx_solve(lp); - if((get_Lrows(lp) > 0) && (lp->lag_status == NOTRUN)) { - if(status == OPTIMAL) - status = lag_solve(lp, lp->bb_heuristicOF, DEF_LAGMAXITERATIONS); - else - report(lp, IMPORTANT, "\nCannot do Lagrangean optimization since root model was not solved.\n"); - } - - /* Reset heuristic in preparation for next run (if any) */ - lp->bb_heuristicOF = my_chsign(is_maxim(lp), lp->infinite); - - /* Check that correct status code is returned */ -/* - peno 26.12.07 - status was not set to SUBOPTIMAL, only lp->spx_status - Bug occured by a change in 5.5.0.10 when && (lp->bb_totalnodes > 0) was added - added status = - See UnitTest3 -*/ -/* - peno 12.01.08 - If an integer solution is found with the same objective value as the relaxed solution then - searching is stopped. This by setting lp->bb_break. However this resulted in a report of SUBOPTIMAL - solution. For this, && !bb_better(lp, OF_DUALLIMIT, OF_TEST_BE) is added in the test. - See UnitTest20 -*/ - if((lp->spx_status == OPTIMAL) && (lp->bb_totalnodes > 0)) { - if((lp->bb_break && !bb_better(lp, OF_DUALLIMIT, OF_TEST_BE)) /* || - ISMASKSET(lp->trace, TRACE_NOBBLIMIT) */) - status = lp->spx_status = SUBOPTIMAL; - } - - return( status ); -} diff --git a/code/3rd_lpsolve/lp_simplex.h b/code/3rd_lpsolve/lp_simplex.h deleted file mode 100644 index d973f72b..00000000 --- a/code/3rd_lpsolve/lp_simplex.h +++ /dev/null @@ -1,34 +0,0 @@ -#ifndef HEADER_lp_simplex -#define HEADER_lp_simplex - -#include "lp_types.h" - -#define ForceDualSimplexInBB /* Force use/switch of dual simplex in B&B */ -#define AssumeHighAccuracyInBB /* No iteration of simplex solves at infeasibility */ -/*#define UseLongStepPruning*/ -/*#define UseLongStepDualPhase1*/ -#define primal_UseRejectionList -#define dual_UseRejectionList -#define dual_RemoveBasicFixedVars -/*#define dual_Phase1PriceEqualities */ /* Force elimination of equality slacks */ -#define AcceptMarginalAccuracy - -#ifdef __cplusplus -extern "C" { -#endif - -/* Put function headers here */ -STATIC int primloop(lprec *lp, MYBOOL primalfeasible, REAL primaloffset); -STATIC int dualloop(lprec *lp, MYBOOL dualfeasible, int dualinfeasibles[], REAL dualoffset); -STATIC int spx_run(lprec *lp, MYBOOL validInvB); -STATIC int spx_solve(lprec *lp); -STATIC int lag_solve(lprec *lp, REAL start_bound, int num_iter); -STATIC int heuristics(lprec *lp, int mode); -STATIC int lin_solve(lprec *lp); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_simplex */ - diff --git a/code/3rd_lpsolve/lp_types.h b/code/3rd_lpsolve/lp_types.h deleted file mode 100644 index 889beed7..00000000 --- a/code/3rd_lpsolve/lp_types.h +++ /dev/null @@ -1,330 +0,0 @@ -#ifndef HEADER_lp_types -#define HEADER_lp_types - -#ifdef WIN32 - #include -#endif - -/* Define data types */ -/* ------------------------------------------------------------------------- */ -#ifndef LLONG - #if defined __BORLANDC__ - #define LLONG __int64 - #elif !defined _MSC_VER || _MSC_VER >= 1310 - #define LLONG long long - #else - #define LLONG __int64 - #endif -#endif - -#ifndef COUNTER - #define COUNTER LLONG -#endif - -#ifndef REAL - #define REAL double -#endif - -#ifndef REALXP - #if 1 - #define REALXP long double /* Set local accumulation variable as long double */ - #else - #define REALXP REAL /* Set local accumulation as default precision */ - #endif -#endif - -#ifndef LREAL - #if 0 - #define LREAL long double /* Set global solution update variable as long double */ - #else - #define LREAL REAL /* Set global solution update variable as default precision */ - #endif -#endif - -#define RESULTVALUEMASK "%18.12g" /* Set fixed-format real-valued output precision; - suggested width: ABS(exponent of DEF_EPSVALUE)+6. */ -#define INDEXVALUEMASK "%8d" /* Set fixed-format integer-valued output width */ - -#ifndef DEF_STRBUFSIZE - #define DEF_STRBUFSIZE 512 -#endif -#ifndef MAXINT32 - #define MAXINT32 2147483647 -#endif -#ifndef MAXUINT32 - #define MAXUINT32 4294967295 -#endif - -#ifndef MAXINT64 - #if defined _LONGLONG || defined __LONG_LONG_MAX__ || defined LLONG_MAX - #define MAXINT64 9223372036854775807ll - #else - #define MAXINT64 9223372036854775807l - #endif -#endif -#ifndef MAXUINT64 - #if defined _LONGLONG || defined __LONG_LONG_MAX__ || defined LLONG_MAX - #define MAXUINT64 18446744073709551615ll - #else - #define MAXUINT64 18446744073709551615l - #endif -#endif - -#ifndef CHAR_BIT - #define CHAR_BIT 8 -#endif -#ifndef MYBOOL - #define MYBOOL unsigned char /* Conserve memory, could be unsigned int */ -#endif - - -/* Constants */ -/* ------------------------------------------------------------------------- */ -#ifndef NULL - #define NULL 0 -#endif - -/* Byte-sized Booleans and extended options */ -#define FALSE 0 -#define TRUE 1 -#define AUTOMATIC 2 -#define DYNAMIC 4 - -/* Sorting and comparison constants */ -#define COMP_PREFERCANDIDATE 1 -#define COMP_PREFERNONE 0 -#define COMP_PREFERINCUMBENT -1 - -/* Library load status values */ -#define LIB_LOADED 0 -#define LIB_NOTFOUND 1 -#define LIB_NOINFO 2 -#define LIB_NOFUNCTION 3 -#define LIB_VERINVALID 4 -#define LIB_STR_LOADED "Successfully loaded" -#define LIB_STR_NOTFOUND "File not found" -#define LIB_STR_NOINFO "No version data" -#define LIB_STR_NOFUNCTION "Missing function header" -#define LIB_STR_VERINVALID "Incompatible version" -#define LIB_STR_MAXLEN 23 - - -/* Compiler/target settings */ -/* ------------------------------------------------------------------------- */ -#if (defined _WIN32) || (defined WIN32) || (defined _WIN64) || (defined WIN64) -# define __WINAPI WINAPI -#else -# define __WINAPI -#endif - -#if (defined _WIN32) || (defined WIN32) || (defined _WIN64) || (defined WIN64) -# define __VACALL __cdecl -#else -# define __VACALL -#endif - -#ifndef __BORLANDC__ - - #ifdef _USRDLL - - #if 1 - #define __EXPORT_TYPE __declspec(dllexport) - #else - /* Set up for the Microsoft compiler */ - #ifdef LP_SOLVE_EXPORTS - #define __EXPORT_TYPE __declspec(dllexport) - #else - #define __EXPORT_TYPE __declspec(dllimport) - #endif - #endif - - #else - - #define __EXPORT_TYPE - - #endif - - #ifdef __cplusplus - #define __EXTERN_C extern "C" - #else - #define __EXTERN_C - #endif - -#else /* Otherwise set up for the Borland compiler */ - - #ifdef __DLL__ - - #define _USRDLL - #define __EXTERN_C extern "C" - - #ifdef __READING_THE_DLL - #define __EXPORT_TYPE __import - #else - #define __EXPORT_TYPE __export - #endif - - #else - - #define __EXPORT_TYPE - #define __EXTERN_C extern "C" - - #endif - -#endif - - -#if 0 - #define STATIC static -#else - #define STATIC -#endif - -#if !defined INLINE - #if defined __cplusplus - #define INLINE inline - #elif (defined _WIN32) || (defined WIN32) || (defined _WIN64) || (defined WIN64) - #define INLINE __inline - #else - #define INLINE static - #endif -#endif - -/* Function macros */ -/* ------------------------------------------------------------------------- */ -#define my_limitrange(x, lo, hi) ((x) < (lo) ? (lo) : ((x) > (hi) ? (hi) : (x))) -#ifndef my_mod - #define my_mod(n, m) ((n) % (m)) -#endif -#define my_if(t, x, y) ((t) ? (x) : (y)) -#define my_sign(x) ((x) < 0 ? -1 : 1) -#if 1 - #define my_chsign(t, x) ( ((t) && ((x) != 0)) ? -(x) : (x)) -#else - #define my_chsign(t, x) ( (2*((t) == 0) - 1) * (x) ) /* "Pipelined", but problem with "negative zero" and possible problems on AIX */ -#endif -#define my_flipsign(x) ( fabs((REAL) (x)) == 0 ? 0 : -(x) ) -#define my_roundzero(val, eps) if (fabs((REAL) (val)) < eps) val = 0 -#define my_avoidtiny(val, eps) (fabs((REAL) (val)) < eps ? 0 : val) - -#if 1 - #define my_infinite(lp, val) ( (MYBOOL) (fabs(val) >= lp->infinite) ) -#else - #define my_infinite(lp, val) is_infinite(lp, val) -#endif -#define my_inflimit(lp, val) ( my_infinite(lp, val) ? lp->infinite * my_sign(val) : (val) ) -#if 0 - #define my_precision(val, eps) ((fabs((REAL) (val))) < (eps) ? 0 : (val)) -#else - #define my_precision(val, eps) restoreINT(val, eps) -#endif -#define my_reldiff(x, y) (((x) - (y)) / (1.0 + fabs((REAL) (y)))) -#define my_boundstr(x) (fabs(x) < lp->infinite ? sprintf("%g",x) : ((x) < 0 ? "-Inf" : "Inf") ) -#ifndef my_boolstr - #define my_boolstr(x) (!(x) ? "FALSE" : "TRUE") -#endif -#define my_basisstr(isbasic) ((isbasic) ? "BASIC" : "NON-BASIC") -#define my_simplexstr(isdual) ((isdual) ? "DUAL" : "PRIMAL") -#define my_plural_std(count) (count == 1 ? "" : "s") -#define my_plural_y(count) (count == 1 ? "y" : "ies") -#define my_lowbound(x) ((FULLYBOUNDEDSIMPLEX) ? (x) : 0) - - -/* Bound macros usable for both the standard and fully bounded simplex */ -/* ------------------------------------------------------------------------- */ -/* -#define my_lowbo(lp, varnr) ( lp->isfullybounded ? lp->lowbo[varnr] : 0.0 ) -#define my_upbo(lp, varnr) ( lp->isfullybounded ? lp->upbo[varnr] : lp->lowbo[varnr] + lp->upbo[varnr] ) -#define my_rangebo(lp, varnr) ( lp->isfullybounded ? lp->upbo[varnr] - lp->lowbo[varnr] : lp->upbo[varnr] ) -*/ -#define my_lowbo(lp, varnr) ( 0.0 ) -#define my_upbo(lp, varnr) ( lp->lowbo[varnr] + lp->upbo[varnr] ) -#define my_rangebo(lp, varnr) ( lp->upbo[varnr] ) - -#define my_unbounded(lp, varnr) ((lp->upbo[varnr] >= lp->infinite) && (lp->lowbo[varnr] <= -lp->infinite)) -#define my_bounded(lp, varnr) ((lp->upbo[varnr] < lp->infinite) && (lp->lowbo[varnr] > -lp->infinite)) - -/* Forward declarations */ -/* ------------------------------------------------------------------------- */ -typedef struct _lprec lprec; -typedef struct _INVrec INVrec; -union QSORTrec; - -#ifndef UNIONTYPE - #ifdef __cplusplus - #define UNIONTYPE - #else - #define UNIONTYPE union - #endif -#endif - -/* B4 factorization optimization data */ -typedef struct _B4rec -{ - int *B4_var; /* Position of basic columns in the B4 basis */ - int *var_B4; /* Variable in the B4 basis */ - int *B4_row; /* B4 position of the i'th row */ - int *row_B4; /* Original position of the i'th row */ - REAL *wcol; - int *nzwcol; -} B4rec; - -#define OBJ_STEPS 5 -typedef struct _OBJmonrec { - lprec *lp; - int oldpivstrategy, - oldpivrule, pivrule, ruleswitches, - limitstall[2], limitruleswitches, - idxstep[OBJ_STEPS], countstep, startstep, currentstep, - Rcycle, Ccycle, Ncycle, Mcycle, Icount; - REAL thisobj, prevobj, - objstep[OBJ_STEPS], - thisinfeas, previnfeas, - epsvalue; - char spxfunc[10]; - MYBOOL pivdynamic; - MYBOOL isdual; - MYBOOL active; -} OBJmonrec; - -typedef struct _edgerec -{ - REAL *edgeVector; -} edgerec; - -typedef struct _pricerec -{ - REAL theta; - REAL pivot; - REAL epspivot; - int varno; - lprec *lp; - MYBOOL isdual; -} pricerec; -typedef struct _multirec -{ - lprec *lp; - int size; /* The maximum number of multiply priced rows/columns */ - int used; /* The current / active number of multiply priced rows/columns */ - int limit; /* The active/used count at which a full update is triggered */ - pricerec *items; /* Array of best multiply priced rows/columns */ - int *freeList; /* The indeces of available positions in "items" */ - UNIONTYPE QSORTrec *sortedList; /* List of pointers to "pricerec" items in sorted order */ - REAL *stepList; /* Working array (values in sortedList order) */ - REAL *valueList; /* Working array (values in sortedList order) */ - int *indexSet; /* The final exported index list of pivot variables */ - int active; /* Index of currently active multiply priced row/column */ - int retries; - REAL step_base; - REAL step_last; - REAL obj_base; - REAL obj_last; - REAL epszero; - REAL maxpivot; - REAL maxbound; - MYBOOL sorted; - MYBOOL truncinf; - MYBOOL objcheck; - MYBOOL dirty; -} multirec; - -#endif /* HEADER_lp_types */ diff --git a/code/3rd_lpsolve/lp_utils.c b/code/3rd_lpsolve/lp_utils.c deleted file mode 100644 index 418f7c86..00000000 --- a/code/3rd_lpsolve/lp_utils.c +++ /dev/null @@ -1,1058 +0,0 @@ -#define CODE_lp_utils - -#include -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_utils.h" -#include -#include -#include "lp_bit.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - - -/* - Miscellaneous utilities as implemented for lp_solve v5.0+ - ---------------------------------------------------------------------------------- - Author: Kjell Eikland - Contact: kjell.eikland@broadpark.no - License terms: GLPL. - - Requires: lp_utils.h, lp_lib.h - - Release notes: - v1.0.0 1 January 2003 Memory allocation, sorting, searching, time and - doubly linked list functions. - v1.1.0 15 May 2004 Added vector packing functionality - v1.2.0 10 January 2005 Added vector pushing/popping functionality - Modified return values and fixed problem in - linked list functions. - - ---------------------------------------------------------------------------------- -*/ - -STATIC MYBOOL allocCHAR(lprec *lp, char **ptr, int size, MYBOOL clear) -{ - if(clear == TRUE) - *ptr = (char *) calloc(size, sizeof(**ptr)); - else if(clear & AUTOMATIC) { - *ptr = (char *) realloc(*ptr, size * sizeof(**ptr)); - if(clear & TRUE) - MEMCLEAR(*ptr, size); - } - else - *ptr = (char *) malloc(size * sizeof(**ptr)); - if(((*ptr) == NULL) && (size > 0)) { - lp->report(lp, CRITICAL, "alloc of %d 'char' failed\n", size); - lp->spx_status = NOMEMORY; - return( FALSE ); - } - else - return( TRUE ); -} -STATIC MYBOOL allocMYBOOL(lprec *lp, MYBOOL **ptr, int size, MYBOOL clear) -{ - if(clear == TRUE) - *ptr = (MYBOOL *) calloc(size, sizeof(**ptr)); - else if(clear & AUTOMATIC) { - *ptr = (MYBOOL *) realloc(*ptr, size * sizeof(**ptr)); - if(clear & TRUE) - MEMCLEAR(*ptr, size); - } - else - *ptr = (MYBOOL *) malloc(size * sizeof(**ptr)); - if(((*ptr) == NULL) && (size > 0)) { - lp->report(lp, CRITICAL, "alloc of %d 'MYBOOL' failed\n", size); - lp->spx_status = NOMEMORY; - return( FALSE ); - } - else - return( TRUE ); -} -STATIC MYBOOL allocINT(lprec *lp, int **ptr, int size, MYBOOL clear) -{ - if(clear == TRUE) - *ptr = (int *) calloc(size, sizeof(**ptr)); - else if(clear & AUTOMATIC) { - *ptr = (int *) realloc(*ptr, size * sizeof(**ptr)); - if(clear & TRUE) - MEMCLEAR(*ptr, size); - } - else - *ptr = (int *) malloc(size * sizeof(**ptr)); - if(((*ptr) == NULL) && (size > 0)) { - lp->report(lp, CRITICAL, "alloc of %d 'INT' failed\n", size); - lp->spx_status = NOMEMORY; - return( FALSE ); - } - else - return( TRUE ); -} -STATIC MYBOOL allocREAL(lprec *lp, REAL **ptr, int size, MYBOOL clear) -{ - if(clear == TRUE) - *ptr = (REAL *) calloc(size, sizeof(**ptr)); - else if(clear & AUTOMATIC) { - *ptr = (REAL *) realloc(*ptr, size * sizeof(**ptr)); - if(clear & TRUE) - MEMCLEAR(*ptr, size); - } - else - *ptr = (REAL *) malloc(size * sizeof(**ptr)); - if(((*ptr) == NULL) && (size > 0)) { - lp->report(lp, CRITICAL, "alloc of %d 'REAL' failed\n", size); - lp->spx_status = NOMEMORY; - return( FALSE ); - } - else - return( TRUE ); -} -STATIC MYBOOL allocLREAL(lprec *lp, LREAL **ptr, int size, MYBOOL clear) -{ - if(clear == TRUE) - *ptr = (LREAL *) calloc(size, sizeof(**ptr)); - else if(clear & AUTOMATIC) { - *ptr = (LREAL *) realloc(*ptr, size * sizeof(**ptr)); - if(clear & TRUE) - MEMCLEAR(*ptr, size); - } - else - *ptr = (LREAL *) malloc(size * sizeof(**ptr)); - if(((*ptr) == NULL) && (size > 0)) { - lp->report(lp, CRITICAL, "alloc of %d 'LREAL' failed\n", size); - lp->spx_status = NOMEMORY; - return( FALSE ); - } - else - return( TRUE ); -} - -STATIC MYBOOL allocFREE(lprec *lp, void **ptr) -{ - MYBOOL status = TRUE; - - if(*ptr != NULL) { - free(*ptr); - *ptr = NULL; - } - else { - status = FALSE; - lp->report(lp, CRITICAL, "free() failed on line %d of file %s\n", - __LINE__, __FILE__); - } - return(status); -} - -/* Do hoops to provide debugging info with FORTIFY */ -#undef CODE_lp_utils -#include "lp_utils.h" -/* alloc-routines should always be before this line! */ - -int comp_bits(MYBOOL *bitarray1, MYBOOL *bitarray2, int items) -{ - int i, items4, left = 0, right = 0; - MYBOOL comp1; - unsigned long comp4; - - /* Convert items count to 8-bit representation, if necessary */ - if(items > 0) { - i = items % 8; - items /= 8; - if(i) - items++; - } - else - items = -items; - - /* Do the wide unsigned integer part for speed */ - items4 = items / sizeof(unsigned long); - i = 0; - while(i < items4) { - comp4 = ((unsigned long *) bitarray1)[i] & ~((unsigned long *) bitarray2)[i]; - if(comp4) - left++; - comp4 = ((unsigned long *) bitarray2)[i] & ~((unsigned long *) bitarray1)[i]; - if(comp4) - right++; - i++; - } - - /* Do the trailing slow narrow unsigned integer part */ - i *= sizeof(unsigned long); - i++; - while(i < items) { - comp1 = bitarray1[i] & ~bitarray2[i]; - if(comp1) - left++; - comp1 = bitarray2[i] & ~bitarray1[i]; - if(comp1) - right++; - i++; - } - - /* Determine set comparison outcomes */ - if((left > 0) && (right == 0)) /* array1 is a superset of array2 */ - i = 1; - else if((left == 0) && (right > 0)) /* array2 is a superset of array1 */ - i = -1; - else if((left == 0) && (right == 0)) /* array1 and array2 are identical */ - i = 0; - else - i = -2; /* indicate all other outcomes */ - return( i ); -} - - -STATIC workarraysrec *mempool_create(lprec *lp) -{ - workarraysrec *temp; - temp = (workarraysrec *) calloc(1, sizeof(workarraysrec)); - temp->lp = lp; - return( temp ); -} -STATIC char *mempool_obtainVector(workarraysrec *mempool, int count, int unitsize) -{ - char *newmem = NULL; - MYBOOL *bnewmem = NULL; - int *inewmem = NULL, size, i, ib, ie, memMargin = 0; - REAL *rnewmem = NULL; - - /* First find the iso-sized window (binary search) */ - size = count*unitsize; - memMargin += size; - ib = 0; - ie = mempool->count-1; - while(ie >= ib) { - i = (ib+ie) / 2; - if(abs(mempool->vectorsize[i]) > memMargin) - ie = i-1; - else if(abs(mempool->vectorsize[i]) < size) - ib = i+1; - else { - /* Find the beginning of the exact-sized array group */ - do { - ib = i; - i--; - } while((i >= 0) && (abs(mempool->vectorsize[i]) >= size)); - break; - } - } - - /* Check if we have a preallocated unused array of sufficient size */ - ie = mempool->count-1; - for(i = ib; i <= ie; i++) - if(mempool->vectorsize[i] < 0) - break; - - /* Obtain and activate existing, unused vector if we are permitted */ - if(i <= ie) { -#ifdef Paranoia - if((mempool->vectorsize[i] > 0) || (abs(mempool->vectorsize[i]) < size)) { - lprec *lp = mempool->lp; - lp->report(lp, SEVERE, "mempool_obtainVector: Invalid %s existing vector selected\n", - (ie < 0 ? "too small" : "occupied")); - lp->spx_status = NOMEMORY; - lp->bb_break = TRUE; - return( newmem ); - } -#endif - newmem = mempool->vectorarray[i]; - mempool->vectorsize[i] *= -1; - } - - /* Otherwise allocate a new vector */ - else if(unitsize == sizeof(MYBOOL)) { - allocMYBOOL(mempool->lp, &bnewmem, count, TRUE); - newmem = (char *) bnewmem; - } - else if(unitsize == sizeof(int)) { - allocINT(mempool->lp, &inewmem, count, TRUE); - newmem = (char *) inewmem; - } - else if(unitsize == sizeof(REAL)) { - allocREAL(mempool->lp, &rnewmem, count, TRUE); - newmem = (char *) rnewmem; - } - - /* Insert into master array if necessary (maintain sort by ascending size) */ - if((i > ie) && (newmem != NULL)) { - mempool->count++; - if(mempool->count >= mempool->size) { - mempool->size += 10; - mempool->vectorarray = (char **) realloc(mempool->vectorarray, - sizeof(*(mempool->vectorarray))*mempool->size); - mempool->vectorsize = (int *) realloc(mempool->vectorsize, - sizeof(*(mempool->vectorsize))*mempool->size); - } - ie++; - i = ie + 1; - if(i < mempool->count) { - MEMMOVE(mempool->vectorarray+i, mempool->vectorarray+ie, 1); - MEMMOVE(mempool->vectorsize+i, mempool->vectorsize+ie, 1); - } - mempool->vectorarray[ie] = newmem; - mempool->vectorsize[ie] = size; - } - - return( newmem ); -} -STATIC MYBOOL mempool_releaseVector(workarraysrec *mempool, char *memvector, MYBOOL forcefree) -{ - int i; - -#if 0 - forcefree = TRUE; -#endif - - for(i = mempool->count-1; i >= 0; i--) - if(mempool->vectorarray[i] == memvector) - break; - - if((i < 0) || (mempool->vectorsize[i] < 0)) - return( FALSE ); - - if(forcefree) { - FREE(mempool->vectorarray[i]); - mempool->count--; - for(; i < mempool->count; i++) - mempool->vectorarray[i] = mempool->vectorarray[i+1]; - } - else - mempool->vectorsize[i] *= -1; - - return( TRUE ); -} -STATIC MYBOOL mempool_free(workarraysrec **mempool) -{ - int i = (*mempool)->count; - - while(i > 0) { - i--; - if((*mempool)->vectorsize[i] < 0) /* Handle unused vectors */ - (*mempool)->vectorsize[i] *= -1; - mempool_releaseVector(*mempool, (*mempool)->vectorarray[i], TRUE); - } - FREE((*mempool)->vectorarray); - FREE((*mempool)->vectorsize); - FREE(*mempool); - return( TRUE ); -} - -REAL *cloneREAL(lprec *lp, REAL *origlist, int size) -{ - REAL *newlist; - - size += 1; - if(allocREAL(lp, &newlist, size, FALSE)) - MEMCOPY(newlist, origlist, size); - return(newlist); -} -MYBOOL *cloneMYBOOL(lprec *lp, MYBOOL *origlist, int size) -{ - MYBOOL *newlist; - - size += 1; - if(allocMYBOOL(lp, &newlist, size, FALSE)) - MEMCOPY(newlist, origlist, size); - return(newlist); -} -int *cloneINT(lprec *lp, int *origlist, int size) -{ - int *newlist; - - size += 1; - if(allocINT(lp, &newlist, size, FALSE)) - MEMCOPY(newlist, origlist, size); - return(newlist); -} - -STATIC void roundVector(LREAL *myvector, int endpos, LREAL roundzero) -{ - if(roundzero > 0) - for(; endpos >= 0; myvector++, endpos--) - if(fabs(*myvector) < roundzero) - *myvector = 0; -} - -STATIC REAL normalizeVector(REAL *myvector, int endpos) -/* Scale the ingoing vector so that its norm is unit, and return the original length */ -{ - int i; - REAL SSQ; - - /* Cumulate squares */ - SSQ = 0; - for(i = 0; i <= endpos; myvector++, i++) - SSQ += (*myvector) * (*myvector); - - /* Normalize */ - SSQ = sqrt(SSQ); - if(SSQ > 0) - for(myvector--; i > 0; myvector--, i--) - (*myvector) /= SSQ; - - return( SSQ ); -} - -/* ---------------------------------------------------------------------------------- */ -/* Other general utilities */ -/* ---------------------------------------------------------------------------------- */ - -STATIC void swapINT(int *item1, int *item2) -{ - int hold = *item1; - *item1 = *item2; - *item2 = hold; -} - -STATIC void swapREAL(REAL *item1, REAL *item2) -{ - REAL hold = *item1; - *item1 = *item2; - *item2 = hold; -} - -STATIC void swapPTR(void **item1, void **item2) -{ - void *hold; - hold = *item1; - *item1 = *item2; - *item2 = hold; -} - - -STATIC REAL restoreINT(REAL valREAL, REAL epsilon) -{ - REAL valINT, fracREAL, fracABS; - - fracREAL = modf(valREAL, &valINT); - fracABS = fabs(fracREAL); - if(fracABS < epsilon) - return(valINT); - else if(fracABS > 1-epsilon) { - if(fracREAL < 0) - return(valINT-1); - else - return(valINT+1); - } - return(valREAL); -} - -STATIC REAL roundToPrecision(REAL value, REAL precision) -{ -#if 1 - REAL vmod; - int vexp2, vexp10; - LLONG sign; - - if(precision == 0) - return(value); - - sign = my_sign(value); - value = fabs(value); - - /* Round to integer if possible */ - if(value < precision) - return( 0 ); - else if(value == floor(value)) - return( value*sign ); - else if((value < (REAL) MAXINT64) && - (modf((REAL) (value+precision), &vmod) < precision)) { - /* sign *= (LLONG) (value+precision); */ - sign *= (LLONG) (value+0.5); - return( (REAL) sign ); - } - - /* Optionally round with base 2 representation for additional precision */ -#define roundPrecisionBase2 -#ifdef roundPrecisionBase2 - value = frexp(value, &vexp2); -#else - vexp2 = 0; -#endif - - /* Convert to desired precision */ - vexp10 = (int) log10(value); - precision *= pow(10.0, vexp10); - modf(value/precision+0.5, &value); - value *= sign*precision; - - /* Restore base 10 representation if base 2 was active */ - if(vexp2 != 0) - value = ldexp(value, vexp2); -#endif - - return( value ); -} - - -/* ---------------------------------------------------------------------------------- */ -/* Searching function specialized for lp_solve */ -/* ---------------------------------------------------------------------------------- */ -STATIC int searchFor(int target, int *attributes, int size, int offset, MYBOOL absolute) -{ - int beginPos, endPos; - int newPos, match; - - /* Set starting and ending index offsets */ - beginPos = offset; - endPos = beginPos + size - 1; - - /* Do binary search logic based on a sorted attribute vector */ - newPos = (beginPos + endPos) / 2; - match = attributes[newPos]; - if(absolute) - match = abs(match); - while(endPos - beginPos > LINEARSEARCH) { - if(match < target) { - beginPos = newPos + 1; - newPos = (beginPos + endPos) / 2; - match = attributes[newPos]; - if(absolute) - match = abs(match); - } - else if(match > target) { - endPos = newPos - 1; - newPos = (beginPos + endPos) / 2; - match = attributes[newPos]; - if(absolute) - match = abs(match); - } - else { - beginPos = newPos; - endPos = newPos; - } - } - - /* Do linear (unsorted) search logic */ - if(endPos - beginPos <= LINEARSEARCH) { - match = attributes[beginPos]; - if(absolute) - match = abs(match); - while((beginPos < endPos) && (match != target)) { - beginPos++; - match = attributes[beginPos]; - if(absolute) - match = abs(match); - } - if(match == target) - endPos = beginPos; - } - - /* Return the index if a match was found, or signal failure with a -1 */ - if((beginPos == endPos) && (match == target)) - return(beginPos); - else - return(-1); - -} - - -/* ---------------------------------------------------------------------------------- */ -/* Other supporting math routines */ -/* ---------------------------------------------------------------------------------- */ - -STATIC MYBOOL isINT(lprec *lp, REAL value) -{ -#if 0 - return( (MYBOOL) (modf(fabs(value)+lp->epsint, &value) < 2*lp->epsint) ); -#elif 1 - value = fabs(value)+lp->epsint; - return( (MYBOOL) (my_reldiff(value, floor(value)) < 2*lp->epsint) ); -#elif 0 - REAL hold; - value = fabs(value); - hold = pow(10, MIN(-2, log10(value+1)+log10(lp->epsint))); - return( (MYBOOL) (modf(value+lp->epsint, &value) < 2*hold) ); -#elif 0 - value -= (REAL)floor(value); - return( (MYBOOL) ((value < lp->epsint) || (value > (1 - lp->epsint)) ); -#else - value += lp->epsint; - return( (MYBOOL) (fabs(value-floor(value)) < 2*lp->epsint) ); -#endif -} - -STATIC MYBOOL isOrigFixed(lprec *lp, int varno) -{ - return( (MYBOOL) (lp->orig_upbo[varno] - lp->orig_lowbo[varno] <= lp->epsmachine) ); -} - -STATIC void chsign_bounds(REAL *lobound, REAL *upbound) -{ - REAL temp; - temp = *upbound; - if(fabs(*lobound) > 0) - *upbound = -(*lobound); - else - *upbound = 0; - if(fabs(temp) > 0) - *lobound = -temp; - else - *lobound = 0; -} - - -/* ---------------------------------------------------------------------------------- */ -/* Define randomization routine */ -/* ---------------------------------------------------------------------------------- */ -STATIC REAL rand_uniform(lprec *lp, REAL range) -{ - static MYBOOL randomized = FALSE; /* static ok here for reentrancy/multithreading */ - - if(!randomized) { - randomized = TRUE; - srand((unsigned) time( NULL )); - } - range *= (REAL) rand() / (REAL) RAND_MAX; - return( range ); -} - - -/* ---------------------------------------------------------------------------------- */ -/* Define routines for doubly linked lists of integers */ -/* ---------------------------------------------------------------------------------- */ - -STATIC int createLink(int size, LLrec **linkmap, MYBOOL *usedpos) -{ - int i, j; - MYBOOL reverse; - - *linkmap = (LLrec *) calloc(1, sizeof(**linkmap)); - if(*linkmap == NULL) - return( -1 ); - - reverse = (MYBOOL) (size < 0); - if(reverse) - size = -size; - (*linkmap)->map = (int *) calloc(2*(size + 1), sizeof(int)); - if((*linkmap)->map == NULL) - return( -1 ); - - (*linkmap)->size = size; - j = 0; - if(usedpos == NULL) - (*linkmap)->map[0] = 0; - else { - for(i = 1; i <= size; i++) - if(!usedpos[i] ^ reverse) { - /* Set the forward link */ - (*linkmap)->map[j] = i; - /* Set the backward link */ - (*linkmap)->map[size+i] = j; - j = i; - if((*linkmap)->count == 0) - (*linkmap)->firstitem = i; - (*linkmap)->lastitem = i; - (*linkmap)->count++; - } - } - (*linkmap)->map[2*size+1] = j; - - return( (*linkmap)->count ); -} - -STATIC MYBOOL freeLink(LLrec **linkmap) -{ - MYBOOL status = TRUE; - - if((linkmap == NULL) || (*linkmap == NULL)) - status = FALSE; - else { - if((*linkmap)->map != NULL) - free((*linkmap)->map); - free(*linkmap); - *linkmap = NULL; - } - return( status ); -} - -STATIC int sizeLink(LLrec *linkmap) -{ - return(linkmap->size); -} - -STATIC MYBOOL isActiveLink(LLrec *linkmap, int itemnr) -{ - if((linkmap->map[itemnr] != 0) || - (linkmap->map[linkmap->size+itemnr] != 0) || - (linkmap->map[0] == itemnr)) - return( TRUE ); - else - return( FALSE ); -} - -STATIC int countActiveLink(LLrec *linkmap) -{ - return(linkmap->count); -} - -STATIC int countInactiveLink(LLrec *linkmap) -{ - return(linkmap->size-linkmap->count); -} - -STATIC int firstActiveLink(LLrec *linkmap) -{ - return(linkmap->map[0]); -} - -STATIC int lastActiveLink(LLrec *linkmap) -{ - return(linkmap->map[2*linkmap->size+1]); -} - -STATIC MYBOOL appendLink(LLrec *linkmap, int newitem) -{ - int k, size; - size = linkmap->size; - - if(linkmap->map[newitem] != 0) - return( FALSE ); - - /* Link forward */ - k = linkmap->map[2*size+1]; - linkmap->map[k] = newitem; - - /* Link backward */ - linkmap->map[size+newitem] = k; - linkmap->map[2*size+1] = newitem; - - /* Update count and return */ - if(linkmap->count == 0) - linkmap->firstitem = newitem; - linkmap->lastitem = newitem; - linkmap->count++; - - return( TRUE ); -} - -STATIC MYBOOL insertLink(LLrec *linkmap, int afteritem, int newitem) -{ - int k, size; - - size = linkmap->size; - - if(linkmap->map[newitem] != 0) - return( FALSE ); - - if(afteritem == linkmap->map[2*size+1]) - appendLink(linkmap, newitem); - else { - /* Link forward */ - k = linkmap->map[afteritem]; - linkmap->map[afteritem] = newitem; - linkmap->map[newitem] = k; - - /* Link backward */ - linkmap->map[size+k] = newitem; - linkmap->map[size+newitem] = afteritem; - - /* Update count */ - SETMIN(linkmap->firstitem, newitem); - SETMAX(linkmap->lastitem, newitem); - linkmap->count++; - } - - return( TRUE ); -} - -STATIC MYBOOL setLink(LLrec *linkmap, int newitem) -{ - if(isActiveLink(linkmap, newitem)) - return( FALSE ); - else - return( insertLink(linkmap, prevActiveLink(linkmap, newitem), newitem) ); -} - -STATIC MYBOOL fillLink(LLrec *linkmap) -{ - int k, size; - size = linkmap->size; - - k = firstActiveLink(linkmap); - if(k != 0) - return( FALSE ); - for(k = 1; k <= size; k++) - appendLink(linkmap, k); - return( TRUE ); -} - -STATIC int nextActiveLink(LLrec *linkmap, int backitemnr) -{ - if((backitemnr < 0) || (backitemnr > linkmap->size)) - return( -1 ); - else { - if(backitemnr < linkmap->lastitem) - while((backitemnr > linkmap->firstitem) && (linkmap->map[backitemnr] == 0)) - backitemnr--; - return(linkmap->map[backitemnr]); - } -} - -STATIC int prevActiveLink(LLrec *linkmap, int forwitemnr) -{ - if((forwitemnr <= 0) || (forwitemnr > linkmap->size+1)) - return( -1 ); - else { - if(forwitemnr > linkmap->lastitem) - return( linkmap->lastitem); - if(forwitemnr > linkmap->firstitem) { - forwitemnr += linkmap->size; - while((forwitemnr < linkmap->size + linkmap->lastitem) && (linkmap->map[forwitemnr] == 0)) - forwitemnr++; - } - else - forwitemnr += linkmap->size; - return(linkmap->map[forwitemnr]); - } -} - -STATIC int firstInactiveLink(LLrec *linkmap) -{ - int i, n; - - if(countInactiveLink(linkmap) == 0) - return( 0 ); - n = 1; - i = firstActiveLink(linkmap); - while(i == n) { - n++; - i = nextActiveLink(linkmap, i); - } - return( n ); -} - -STATIC int lastInactiveLink(LLrec *linkmap) -{ - int i, n; - - if(countInactiveLink(linkmap) == 0) - return( 0 ); - n = linkmap->size; - i = lastActiveLink(linkmap); - while(i == n) { - n--; - i = prevActiveLink(linkmap, i); - } - return( n ); -} - -STATIC int nextInactiveLink(LLrec *linkmap, int backitemnr) -{ - do { - backitemnr++; - } while((backitemnr <= linkmap->size) && isActiveLink(linkmap, backitemnr)); - if(backitemnr <= linkmap->size) - return( backitemnr ); - else - return( 0 ); -} - -STATIC int prevInactiveLink(LLrec *linkmap, int forwitemnr) -{ - return( 0 ); -} - -STATIC int removeLink(LLrec *linkmap, int itemnr) -{ - int size, prevnr, nextnr = -1; - - size = linkmap->size; - if((itemnr <= 0) || (itemnr > size)) - return( nextnr ); -#ifdef Paranoia - if(!isActiveLink(linkmap, itemnr)) - return( nextnr ); -#endif - - /* Get link data at the specified position */ - nextnr = linkmap->map[itemnr]; - prevnr = linkmap->map[size+itemnr]; - if(itemnr == linkmap->firstitem) - linkmap->firstitem = nextnr; - if(itemnr == linkmap->lastitem) - linkmap->lastitem = prevnr; - - /* Update forward link */ - linkmap->map[prevnr] = linkmap->map[itemnr]; - linkmap->map[itemnr] = 0; - - /* Update backward link */ - if(nextnr == 0) - linkmap->map[2*size+1] = prevnr; - else - linkmap->map[size+nextnr] = linkmap->map[size+itemnr]; - linkmap->map[size+itemnr] = 0; - - /* Decrement the count */ - linkmap->count--; - - /* Return the next active item */ - return( nextnr ); -} - -STATIC LLrec *cloneLink(LLrec *sourcemap, int newsize, MYBOOL freesource) -{ - LLrec *testmap = NULL; - - if((newsize == sourcemap->size) || (newsize <= 0)) { - createLink(sourcemap->size, &testmap, NULL); - MEMCOPY(testmap->map, sourcemap->map, 2*(sourcemap->size+1)); - testmap->firstitem = sourcemap->firstitem; - testmap->lastitem = sourcemap->lastitem; - testmap->size = sourcemap->size; - testmap->count = sourcemap->count; - } - else { - int j; - - createLink(newsize, &testmap, NULL); - for(j = firstActiveLink(sourcemap); (j != 0) && (j <= newsize); j = nextActiveLink(sourcemap, j)) - appendLink(testmap, j); - } - if(freesource) - freeLink(&sourcemap); - - return(testmap); -} - -STATIC int compareLink(LLrec *linkmap1, LLrec *linkmap2) -{ - int test; - - test = memcmp(&linkmap1->size, &linkmap2->size, sizeof(int)); - if(test == 0) - test = memcmp(&linkmap1->count, &linkmap2->count, sizeof(int)); - if(test == 0) - test = memcmp(linkmap1->map, linkmap2->map, sizeof(int)*(2*linkmap1->size+1)); - - return( test ); -} - -STATIC MYBOOL verifyLink(LLrec *linkmap, int itemnr, MYBOOL doappend) -{ - LLrec *testmap; - - testmap = cloneLink(linkmap, -1, FALSE); - if(doappend) { - appendLink(testmap, itemnr); - removeLink(testmap, itemnr); - } - else { - int previtem = prevActiveLink(testmap, itemnr); - removeLink(testmap, itemnr); - insertLink(testmap, previtem, itemnr); - } - itemnr = compareLink(linkmap, testmap); - freeLink(&testmap); - return((MYBOOL) (itemnr == 0)); -} - -/* Packed vector routines */ -STATIC PVrec *createPackedVector(int size, REAL *values, int *workvector) -{ - int i, k; - REGISTER REAL ref; - PVrec *newPV = NULL; - MYBOOL localWV = (MYBOOL) (workvector == NULL); - - if(localWV) - workvector = (int *) malloc((size+1)*sizeof(*workvector)); - - /* Tally equal-valued vector entries - also check if it is worth compressing */ - k = 0; - workvector[k] = 1; - ref = values[1]; - for(i = 2; i <= size; i++) { - if(fabs(ref - values[i]) > DEF_EPSMACHINE) { - k++; - workvector[k] = i; - ref = values[i]; - } - } - if(k > size / 2) { - if(localWV) - FREE(workvector); - return( newPV ); - } - - /* Create the packing object, adjust the position vector and allocate value vector */ - newPV = (PVrec *) malloc(sizeof(*newPV)); - k++; /* Adjust from index to to count */ - newPV->count = k; - if(localWV) - newPV->startpos = (int *) realloc(workvector, (k + 1)*sizeof(*(newPV->startpos))); - else { - newPV->startpos = (int *) malloc((k + 1)*sizeof(*(newPV->startpos))); - MEMCOPY(newPV->startpos, workvector, k); - } - newPV->startpos[k] = size + 1; /* Store terminal index + 1 for searching purposes */ - newPV->value = (REAL *) malloc(k*sizeof(*(newPV->value))); - - /* Fill the values vector before returning */ - for(i = 0; i < k; i++) - newPV->value[i] = values[newPV->startpos[i]]; - - return( newPV ); -} - -STATIC MYBOOL unpackPackedVector(PVrec *PV, REAL **target) -{ - int i, ii, k; - REGISTER REAL ref; - - /* Test for validity of the target and create it if necessary */ - if(target == NULL) - return( FALSE ); - if(*target == NULL) - allocREAL(NULL, target, PV->startpos[PV->count], FALSE); - - /* Expand the packed vector into the target */ - i = PV->startpos[0]; - for(k = 0; k < PV->count; k++) { - ii = PV->startpos[k+1]; - ref = PV->value[k]; - while (i < ii) { - (*target)[i] = ref; - i++; - } - } - return( TRUE ); -} - -STATIC REAL getvaluePackedVector(PVrec *PV, int index) -{ - index = searchFor(index, PV->startpos, PV->count, 0, FALSE); - index = abs(index)-1; - if(index >= 0) - return( PV->value[index] ); - else - return( 0 ); -} - -STATIC MYBOOL freePackedVector(PVrec **PV) -{ - if((PV == NULL) || (*PV == NULL)) - return( FALSE ); - - FREE((*PV)->value); - FREE((*PV)->startpos); - FREE(*PV); - return( TRUE ); -} - -STATIC void pushPackedVector(PVrec *PV, PVrec *parent) -{ - PV->parent = parent; -} - -STATIC PVrec *popPackedVector(PVrec *PV) -{ - PVrec *parent = PV->parent; - freePackedVector(&PV); - return( parent ); -} - diff --git a/code/3rd_lpsolve/lp_utils.h b/code/3rd_lpsolve/lp_utils.h deleted file mode 100644 index 47f3ec69..00000000 --- a/code/3rd_lpsolve/lp_utils.h +++ /dev/null @@ -1,146 +0,0 @@ -#ifndef HEADER_lp_utils -#define HEADER_lp_utils - -#ifdef FORTIFY - -#include "lp_fortify.h" - -#define allocCHAR allocCHAR_FORTIFY -#define allocMYBOOL allocMYBOOL_FORTIFY -#define allocINT allocINT_FORTIFY -#define allocREAL allocREAL_FORTIFY -#define allocLREAL allocLREAL_FORTIFY - -#endif - -#include "lp_types.h" - -/* Temporary data storage arrays */ -typedef struct _workarraysrec -{ - lprec *lp; - int size; - int count; - char **vectorarray; - int *vectorsize; -} workarraysrec; - -typedef struct _LLrec -{ - int size; /* The allocated list size */ - int count; /* The current entry count */ - int firstitem; - int lastitem; - int *map; /* The list of forward and backward-mapped entries */ -} LLrec; - -typedef struct _PVrec -{ - int count; /* The allocated list item count */ - int *startpos; /* Starting index of the current value */ - REAL *value; /* The list of forward and backward-mapped entries */ - struct _PVrec *parent; /* The parent record in a pushed chain */ -} PVrec; - - -#ifdef __cplusplus -extern "C" { -#endif - -/* Put function headers here */ -STATIC MYBOOL allocCHAR(lprec *lp, char **ptr, int size, MYBOOL clear); -STATIC MYBOOL allocMYBOOL(lprec *lp, MYBOOL **ptr, int size, MYBOOL clear); -STATIC MYBOOL allocINT(lprec *lp, int **ptr, int size, MYBOOL clear); -STATIC MYBOOL allocREAL(lprec *lp, REAL **ptr, int size, MYBOOL clear); -STATIC MYBOOL allocLREAL(lprec *lp, LREAL **ptr, int size, MYBOOL clear); -STATIC MYBOOL allocFREE(lprec *lp, void **ptr); -REAL *cloneREAL(lprec *lp, REAL *origlist, int size); -MYBOOL *cloneMYBOOL(lprec *lp, MYBOOL *origlist, int size); -int *cloneINT(lprec *lp, int *origlist, int size); - -int comp_bits(MYBOOL *bitarray1, MYBOOL *bitarray2, int items); - -STATIC workarraysrec *mempool_create(lprec *lp); -STATIC char *mempool_obtainVector(workarraysrec *mempool, int count, int unitsize); -STATIC MYBOOL mempool_releaseVector(workarraysrec *mempool, char *memvector, MYBOOL forcefree); -STATIC MYBOOL mempool_free(workarraysrec **mempool); - -STATIC void roundVector(LREAL *myvector, int endpos, LREAL roundzero); -STATIC REAL normalizeVector(REAL *myvector, int endpos); - -STATIC void swapINT(int *item1, int *item2); -STATIC void swapREAL(REAL *item1, REAL *item2); -STATIC void swapPTR(void **item1, void **item2); -STATIC REAL restoreINT(REAL valREAL, REAL epsilon); -STATIC REAL roundToPrecision(REAL value, REAL precision); - -STATIC int searchFor(int target, int *attributes, int size, int offset, MYBOOL absolute); - -STATIC MYBOOL isINT(lprec *lp, REAL value); -STATIC MYBOOL isOrigFixed(lprec *lp, int varno); -STATIC void chsign_bounds(REAL *lobound, REAL *upbound); -STATIC REAL rand_uniform(lprec *lp, REAL range); - -/* Doubly linked list routines */ -STATIC int createLink(int size, LLrec **linkmap, MYBOOL *usedpos); -STATIC MYBOOL freeLink(LLrec **linkmap); -STATIC int sizeLink(LLrec *linkmap); -STATIC MYBOOL isActiveLink(LLrec *linkmap, int itemnr); -STATIC int countActiveLink(LLrec *linkmap); -STATIC int countInactiveLink(LLrec *linkmap); -STATIC int firstActiveLink(LLrec *linkmap); -STATIC int lastActiveLink(LLrec *linkmap); -STATIC MYBOOL appendLink(LLrec *linkmap, int newitem); -STATIC MYBOOL insertLink(LLrec *linkmap, int afteritem, int newitem); -STATIC MYBOOL setLink(LLrec *linkmap, int newitem); -STATIC MYBOOL fillLink(LLrec *linkmap); -STATIC int nextActiveLink(LLrec *linkmap, int backitemnr); -STATIC int prevActiveLink(LLrec *linkmap, int forwitemnr); -STATIC int firstInactiveLink(LLrec *linkmap); -STATIC int lastInactiveLink(LLrec *linkmap); -STATIC int nextInactiveLink(LLrec *linkmap, int backitemnr); -STATIC int prevInactiveLink(LLrec *linkmap, int forwitemnr); -STATIC int removeLink(LLrec *linkmap, int itemnr); -STATIC LLrec *cloneLink(LLrec *sourcemap, int newsize, MYBOOL freesource); -STATIC int compareLink(LLrec *linkmap1, LLrec *linkmap2); -STATIC MYBOOL verifyLink(LLrec *linkmap, int itemnr, MYBOOL doappend); - -/* Packed vector routines */ -STATIC PVrec *createPackedVector(int size, REAL *values, int *workvector); -STATIC void pushPackedVector(PVrec *PV, PVrec *parent); -STATIC MYBOOL unpackPackedVector(PVrec *PV, REAL **target); -STATIC REAL getvaluePackedVector(PVrec *PV, int index); -STATIC PVrec *popPackedVector(PVrec *PV); -STATIC MYBOOL freePackedVector(PVrec **PV); - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_utils */ - -#ifdef FORTIFY - -#if defined CODE_lp_utils && !defined CODE_lp_utils_ -int _Fortify_ret; -#else -extern int _Fortify_ret; -#endif - -#ifdef CODE_lp_utils -#define CODE_lp_utils_ -#else -# undef allocCHAR -# undef allocMYBOOL -# undef allocINT -# undef allocREAL -# undef allocLREAL -# define allocCHAR(lp, ptr, size, clear) (Fortify_LINE(__LINE__), Fortify_FILE(__FILE__), _Fortify_ret = allocCHAR_FORTIFY(lp, ptr, size, clear), Fortify_LINE(0), Fortify_FILE(NULL), _Fortify_ret) -# define allocMYBOOL(lp, ptr, size, clear) (Fortify_LINE(__LINE__), Fortify_FILE(__FILE__), _Fortify_ret = allocMYBOOL_FORTIFY(lp, ptr, size, clear), Fortify_LINE(0), Fortify_FILE(NULL), _Fortify_ret) -# define allocINT(lp, ptr, size, clear) (Fortify_LINE(__LINE__), Fortify_FILE(__FILE__), _Fortify_ret = allocINT_FORTIFY(lp, ptr, size, clear), Fortify_LINE(0), Fortify_FILE(NULL), _Fortify_ret) -# define allocREAL(lp, ptr, size, clear) (Fortify_LINE(__LINE__), Fortify_FILE(__FILE__), _Fortify_ret = allocREAL_FORTIFY(lp, ptr, size, clear), Fortify_LINE(0), Fortify_FILE(NULL), _Fortify_ret) -# define allocLREAL(lp, ptr, size, clear) (Fortify_LINE(__LINE__), Fortify_FILE(__FILE__), _Fortify_ret = allocLREAL_FORTIFY(lp, ptr, size, clear), Fortify_LINE(0), Fortify_FILE(NULL), _Fortify_ret) -#endif - -#endif - diff --git a/code/3rd_lpsolve/lp_wlp.c b/code/3rd_lpsolve/lp_wlp.c deleted file mode 100644 index afd7f22a..00000000 --- a/code/3rd_lpsolve/lp_wlp.c +++ /dev/null @@ -1,362 +0,0 @@ - -#include -#include -#include - -#include "commonlib.h" -#include "lp_lib.h" -#include "lp_scale.h" -#include "lp_utils.h" -#include "lp_report.h" -#include "lp_wlp.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* Define buffer-size controled function mapping */ -# if defined _MSC_VER -# define vsnprintf _vsnprintf -# endif - -/* ------------------------------------------------------------------------- */ -/* Input and output of lp format model files for lp_solve */ -/* ------------------------------------------------------------------------- */ - -static int write_data(void *userhandle, write_modeldata_func write_modeldata, char *format, ...) -{ - char buff[DEF_STRBUFSIZE+1]; - va_list ap; - int n; - - va_start(ap, format); - vsnprintf(buff, DEF_STRBUFSIZE, format, ap); - va_end(ap); - n = write_modeldata(userhandle, buff); - return(n); -} - -STATIC void write_lpcomment(void *userhandle, write_modeldata_func write_modeldata, char *string, MYBOOL newlinebefore) -{ - write_data(userhandle, write_modeldata, "%s/* %s */\n", (newlinebefore) ? "\n" : "", string); -} - -STATIC int write_lprow(lprec *lp, int rowno, void *userhandle, write_modeldata_func write_modeldata, int maxlen, int *idx, REAL *val) -{ - int i, j, nchars, elements; - REAL a; - MYBOOL first = TRUE; - char buf[50]; - - elements = get_rowex(lp, rowno, val, idx); - if(write_modeldata != NULL) { - nchars = 0; - for(i = 0; i < elements; i++) { - j = idx[i]; - if(is_splitvar(lp, j)) - continue; - a = val[i]; - if(!first) - nchars += write_data(userhandle, write_modeldata, " "); - else - first = FALSE; - sprintf(buf, "%+.12g", (double)a); - if(strcmp(buf, "-1") == 0) - nchars += write_data(userhandle, write_modeldata, "-"); - else if(strcmp(buf, "+1") == 0) - nchars += write_data(userhandle, write_modeldata, "+"); - else - nchars += write_data(userhandle, write_modeldata, "%s ", buf); - nchars += write_data(userhandle, write_modeldata, "%s", get_col_name(lp, j)); - /* Check if we should add a linefeed */ - if((maxlen > 0) && (nchars >= maxlen) && (i < elements-1)) { - write_data(userhandle, write_modeldata, "%s", "\n"); - nchars = 0; - } - } - } - return(elements); -} - -#if !defined LP_MAXLINELEN -# define LP_MAXLINELEN 100 -#endif - -MYBOOL __WINAPI write_lpex(lprec *lp, void *userhandle, write_modeldata_func write_modeldata) -{ - int i, j, b, - nrows = lp->rows, - ncols = lp->columns, - nchars, maxlen = LP_MAXLINELEN, - *idx; - MYBOOL ok; - REAL a, *val; - char *ptr; - - if(!mat_validate(lp->matA)) { - report(lp, IMPORTANT, "LP_writefile: Could not validate the data matrix.\n"); - return(FALSE); - } - - /* Write name of model */ - ptr = get_lp_name(lp); - if(ptr != NULL) { - if(*ptr) - write_lpcomment(userhandle, write_modeldata, ptr, FALSE); - else - ptr = NULL; - } - - /* Write the objective function */ - write_lpcomment(userhandle, write_modeldata, "Objective function", (MYBOOL) (ptr != NULL)); - if(is_maxim(lp)) - write_data(userhandle, write_modeldata, "max: "); - else - write_data(userhandle, write_modeldata, "min: "); - - allocREAL(lp, &val, 1 + lp->columns, TRUE); - allocINT(lp, &idx, 1 + lp->columns, TRUE); - - write_lprow(lp, 0, userhandle, write_modeldata, maxlen, idx, val); - a = get_rh(lp, 0); - if(a != 0) - write_data(userhandle, write_modeldata, " %+.12g", a); - write_data(userhandle, write_modeldata, ";\n"); - - /* Write constraints */ - if(nrows > 0) - write_lpcomment(userhandle, write_modeldata, "Constraints", TRUE); - for(j = 1; j <= nrows; j++) { - if(((lp->names_used) && (lp->row_name[j] != NULL)) || (write_lprow(lp, j, userhandle, NULL, maxlen, idx, val) == 1)) - ptr = get_row_name(lp, j); - else - ptr = NULL; - if((ptr != NULL) && (*ptr)) - write_data(userhandle, write_modeldata, "%s: ", ptr); - -#ifndef SingleBoundedRowInLP - /* Write the ranged part of the constraint, if specified */ - if ((lp->orig_upbo[j]) && (lp->orig_upbo[j] < lp->infinite)) { - if(my_chsign(is_chsign(lp, j), lp->orig_rhs[j]) == -lp->infinite) - write_data(userhandle, write_modeldata, "-Inf %s ", (is_chsign(lp, j)) ? ">=" : "<="); - else if(my_chsign(is_chsign(lp, j), lp->orig_rhs[j]) == lp->infinite) - write_data(userhandle, write_modeldata, "+Inf %s ", (is_chsign(lp, j)) ? ">=" : "<="); - else - write_data(userhandle, write_modeldata, "%+.12g %s ", - (lp->orig_upbo[j]-lp->orig_rhs[j]) * (is_chsign(lp, j) ? 1.0 : -1.0) / (lp->scaling_used ? lp->scalars[j] : 1.0), - (is_chsign(lp, j)) ? ">=" : "<="); - } -#endif - - if((!write_lprow(lp, j, userhandle, write_modeldata, maxlen, idx, val)) && (ncols >= 1)) - write_data(userhandle, write_modeldata, "0 %s", get_col_name(lp, 1)); - - if(lp->orig_upbo[j] == 0) - write_data(userhandle, write_modeldata, " ="); - else if(is_chsign(lp, j)) - write_data(userhandle, write_modeldata, " >="); - else - write_data(userhandle, write_modeldata, " <="); - if(fabs(get_rh(lp, j) + lp->infinite) < 1) - write_data(userhandle, write_modeldata, " -Inf;\n"); - else if(fabs(get_rh(lp, j) - lp->infinite) < 1) - write_data(userhandle, write_modeldata, " +Inf;\n"); - else - write_data(userhandle, write_modeldata, " %.12g;\n", get_rh(lp, j)); - -#ifdef SingleBoundedRowInLP - /* Write the ranged part of the constraint, if specified */ - if ((lp->orig_upbo[j]) && (lp->orig_upbo[j] < lp->infinite)) { - if(((lp->names_used) && (lp->row_name[j] != NULL)) || (write_lprow(lp, j, userhandle, NULL, maxlen, idx, val) == 1)) - ptr = get_row_name(lp, j); - else - ptr = NULL; - if((ptr != NULL) && (*ptr)) - write_data(userhandle, write_modeldata, "%s: ", ptr); - if((!write_lprow(lp, j, userhandle, write_modeldata, maxlen, idx, val)) && (get_Ncolumns(lp) >= 1)) - write_data(userhandle, write_modeldata, "0 %s", get_col_name(lp, 1)); - write_data(userhandle, write_modeldata, " %s %g;\n", - (is_chsign(lp, j)) ? "<=" : ">=", - (lp->orig_upbo[j]-lp->orig_rhs[j]) * (is_chsign(lp, j) ? 1.0 : -1.0) / (lp->scaling_used ? lp->scalars[j] : 1.0)); - } -#endif - } - - /* Write bounds on variables */ - ok = FALSE; - for(i = nrows + 1; i <= lp->sum; i++) - if(!is_splitvar(lp, i - nrows)) { - if(lp->orig_lowbo[i] == lp->orig_upbo[i]) { - if(!ok) { - write_lpcomment(userhandle, write_modeldata, "Variable bounds", TRUE); - ok = TRUE; - } - write_data(userhandle, write_modeldata, "%s = %.12g;\n", get_col_name(lp, i - nrows), get_upbo(lp, i - nrows)); - } - else { -#ifndef SingleBoundedRowInLP - if((lp->orig_lowbo[i] != 0) && (lp->orig_upbo[i] < lp->infinite)) { - if(!ok) { - write_lpcomment(userhandle, write_modeldata, "Variable bounds", TRUE); - ok = TRUE; - } - if(lp->orig_lowbo[i] == -lp->infinite) - write_data(userhandle, write_modeldata, "-Inf"); - else - write_data(userhandle, write_modeldata, "%.12g", get_lowbo(lp, i - nrows)); - write_data(userhandle, write_modeldata, " <= %s <= ", get_col_name(lp, i - nrows)); - if(lp->orig_lowbo[i] == lp->infinite) - write_data(userhandle, write_modeldata, "+Inf"); - else - write_data(userhandle, write_modeldata, "%.12g", get_upbo(lp, i - nrows)); - write_data(userhandle, write_modeldata, ";\n"); - } - else -#endif - { - if(lp->orig_lowbo[i] != 0) { - if(!ok) { - write_lpcomment(userhandle, write_modeldata, "Variable bounds", TRUE); - ok = TRUE; - } - if(lp->orig_lowbo[i] == -lp->infinite) - write_data(userhandle, write_modeldata, "%s >= -Inf;\n", get_col_name(lp, i - nrows)); - else if(lp->orig_lowbo[i] == lp->infinite) - write_data(userhandle, write_modeldata, "%s >= +Inf;\n", get_col_name(lp, i - nrows)); - else - write_data(userhandle, write_modeldata, "%s >= %.12g;\n", - get_col_name(lp, i - nrows), get_lowbo(lp, i - nrows)); - } - if(lp->orig_upbo[i] != lp->infinite) { - if(!ok) { - write_lpcomment(userhandle, write_modeldata, "Variable bounds", TRUE); - ok = TRUE; - } - write_data(userhandle, write_modeldata, "%s <= %.12g;\n", - get_col_name(lp, i - nrows), get_upbo(lp, i - nrows)); - } - } - } - } - - /* Write optional integer section */ - if(lp->int_vars > 0) { - write_lpcomment(userhandle, write_modeldata, "Integer definitions", TRUE); - i = 1; - while((i <= ncols) && !is_int(lp, i)) - i++; - if(i <= ncols) { - nchars = write_data(userhandle, write_modeldata, "int %s", get_col_name(lp, i)); - i++; - for(; i <= ncols; i++) - if((!is_splitvar(lp, i)) && (is_int(lp, i))) { - if((maxlen!= 0) && (nchars > maxlen)) { - write_data(userhandle, write_modeldata, "%s", "\n"); - nchars = 0; - } - write_data(userhandle, write_modeldata, ",%s", get_col_name(lp, i)); - } - write_data(userhandle, write_modeldata, ";\n"); - } - } - - /* Write optional SEC section */ - if(lp->sc_vars > 0) { - write_lpcomment(userhandle, write_modeldata, "Semi-continuous variables", TRUE); - i = 1; - while((i <= ncols) && !is_semicont(lp, i)) - i++; - if(i <= ncols) { - nchars = write_data(userhandle, write_modeldata, "sec %s", get_col_name(lp, i)); - i++; - for(; i <= ncols; i++) - if((!is_splitvar(lp, i)) && (is_semicont(lp, i))) { - if((maxlen != 0) && (nchars > maxlen)) { - write_data(userhandle, write_modeldata, "%s", "\n"); - nchars = 0; - } - nchars += write_data(userhandle, write_modeldata, ",%s", get_col_name(lp, i)); - } - write_data(userhandle, write_modeldata, ";\n"); - } - } - - /* Write optional SOS section */ - if(SOS_count(lp) > 0) { - SOSgroup *SOS = lp->SOS; - write_lpcomment(userhandle, write_modeldata, "SOS definitions", TRUE); - write_data(userhandle, write_modeldata, "SOS\n"); - for(b = 0, i = 0; i < SOS->sos_count; b = SOS->sos_list[i]->priority, i++) { - nchars = write_data(userhandle, write_modeldata, "%s: ", - (SOS->sos_list[i]->name == NULL) || - (*SOS->sos_list[i]->name==0) ? "SOS" : SOS->sos_list[i]->name); /* formatnumber12((double) lp->sos_list[i]->priority) */ - - for(a = 0.0, j = 1; j <= SOS->sos_list[i]->size; a = SOS->sos_list[i]->weights[j], j++) { - if((maxlen != 0) && (nchars > maxlen)) { - write_data(userhandle, write_modeldata, "%s", "\n"); - nchars = 0; - } - if(SOS->sos_list[i]->weights[j] == ++a) - nchars += write_data(userhandle, write_modeldata, "%s%s", - (j > 1) ? "," : "", - get_col_name(lp, SOS->sos_list[i]->members[j])); - else - nchars += write_data(userhandle, write_modeldata, "%s%s:%.12g", - (j > 1) ? "," : "", - get_col_name(lp, SOS->sos_list[i]->members[j]), - SOS->sos_list[i]->weights[j]); - } - if(SOS->sos_list[i]->priority == ++b) - nchars += write_data(userhandle, write_modeldata, " <= %d;\n", SOS->sos_list[i]->type); - else - nchars += write_data(userhandle, write_modeldata, " <= %d:%d;\n", SOS->sos_list[i]->type, SOS->sos_list[i]->priority); - } - } - - FREE(val); - FREE(idx); - - ok = TRUE; - - return(ok); -} - -static int __WINAPI write_lpdata(void *userhandle, char *buf) -{ - return(fprintf((FILE *) userhandle, "%s", buf)); -} - -MYBOOL LP_writefile(lprec *lp, char *filename) -{ - FILE *output = stdout; - MYBOOL ok; - - if (filename != NULL) { - ok = (MYBOOL) ((output = fopen(filename, "w")) != NULL); - if(!ok) - return(ok); - } - else - output = lp->outstream; - - ok = write_lpex(lp, (void *) output, write_lpdata); - - if (filename != NULL) - fclose(output); - - return(ok); -} - -MYBOOL LP_writehandle(lprec *lp, FILE *output) -{ - MYBOOL ok; - - if (output != NULL) - set_outputstream(lp, output); - - output = lp->outstream; - - ok = write_lpex(lp, (void *) output, write_lpdata); - - return(ok); -} diff --git a/code/3rd_lpsolve/lp_wlp.h b/code/3rd_lpsolve/lp_wlp.h deleted file mode 100644 index 010858ba..00000000 --- a/code/3rd_lpsolve/lp_wlp.h +++ /dev/null @@ -1,21 +0,0 @@ -#ifndef HEADER_lp_lp -#define HEADER_lp_lp - -#include "lp_types.h" - - -#ifdef __cplusplus -extern "C" { -#endif - -/* Put function headers here */ -MYBOOL LP_writefile(lprec *lp, char *filename); -MYBOOL LP_writehandle(lprec *lp, FILE *output); - - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_lp_lp */ - diff --git a/code/3rd_lpsolve/lpkit.h b/code/3rd_lpsolve/lpkit.h deleted file mode 100644 index 87913a41..00000000 --- a/code/3rd_lpsolve/lpkit.h +++ /dev/null @@ -1,32 +0,0 @@ -#include "lp_lib.h" -#include "lp_report.h" - -#define MALLOC(ptr, nr, type)\ - ((((nr) == 0) || ((ptr = (type *) malloc((size_t)((nr) * sizeof(*ptr)))) == NULL)) ? \ - report(NULL, CRITICAL, "malloc of %d bytes failed on line %d of file %s\n",\ - (nr) * sizeof(*ptr), __LINE__, __FILE__), (ptr = NULL /* (void *) 0 */) : \ - ptr\ - ) - -#define CALLOC(ptr, nr, type)\ - ((((nr) == 0) || ((ptr = (type *) calloc((size_t)(nr), sizeof(*ptr))) == NULL)) ? \ - report(NULL, CRITICAL, "calloc of %d bytes failed on line %d of file %s\n",\ - (nr) * sizeof(*ptr), __LINE__, __FILE__), (ptr = NULL /* (void *) 0 */) : \ - ptr\ - ) - -#define REALLOC(ptr, nr, type)\ - ((((nr) == 0) || ((ptr = (type *) realloc(ptr, (size_t)((nr) * sizeof(*ptr)))) == NULL)) ? \ - report(NULL, CRITICAL, "realloc of %d bytes failed on line %d of file %s\n",\ - (nr) * sizeof(*ptr), __LINE__, __FILE__), (ptr = NULL /* (void *) 0 */) : \ - ptr\ - ) - -#if defined FREE -# undef FREE -#endif - -#define FREE(ptr) if (ptr != NULL) {free(ptr), ptr = NULL;} else - -#define MALLOCCPY(nptr, optr, nr, type)\ - (MALLOC(nptr, nr, type), (nptr != NULL) ? memcpy(nptr, optr, (size_t)((nr) * sizeof(*optr))) : 0, nptr) diff --git a/code/3rd_lpsolve/shared/commonlib.c b/code/3rd_lpsolve/shared/commonlib.c deleted file mode 100644 index 5ab4f12c..00000000 --- a/code/3rd_lpsolve/shared/commonlib.c +++ /dev/null @@ -1,992 +0,0 @@ - -#include - -#if defined INTEGERTIME || defined CLOCKTIME || defined PosixTime -# include -#elif defined EnhTime -# include -#else -# include -#endif - -#include -#include -#ifdef WIN32 -# include /* Used in file search functions */ -#endif -#include -#include -#include -#include -#include "commonlib.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -#if defined FPUexception -/* FPU exception masks */ -unsigned int clearFPU() -{ - return( _clearfp() ); -} -unsigned int resetFPU(unsigned int mask) -{ - _clearfp(); - mask = _controlfp( mask, 0xfffff); - return( mask ); -} -unsigned int catchFPU(unsigned int mask) -{ - /* Always call _clearfp before enabling/unmasking a FPU exception */ - unsigned int u = _clearfp(); - - /* Set the new mask by not-and'ing it with the previous settings */ - u = _controlfp(0, 0); - mask = u & ~mask; - mask = _controlfp(mask, _MCW_EM); - - /* Return the previous mask */ - return( u ); -} -#endif - -/* Math operator equivalence function */ -int intpow(int base, int exponent) -{ - int result = 1; - while(exponent > 0) { - result *= base; - exponent--; - } - while(exponent < 0) { - result /= base; - exponent++; - } - return( result ); -} -int mod(int n, int d) -{ - return(n % d); -} - -/* Some string functions */ -void strtoup(char *s) -{ - if(s != NULL) - while (*s) { - *s = toupper(*s); - s++; - } -} -void strtolo(char *s) -{ - if(s != NULL) - while (*s) { - *s = tolower(*s); - s++; - } -} -void strcpyup(char *t, char *s) -{ - if((s != NULL) && (t != NULL)) { - while (*s) { - *t = toupper(*s); - t++; - s++; - } - *t = '\0'; - } -} -void strcpylo(char *t, char *s) -{ - if((s != NULL) && (t != NULL)) { - while (*s) { - *t = tolower(*s); - t++; - s++; - } - *t = '\0'; - } -} - -/* Unix library naming utility function */ -MYBOOL so_stdname(char *stdname, char *descname, int buflen) -{ - char *ptr; - - if((descname == NULL) || (stdname == NULL) || (((int) strlen(descname)) >= buflen - 6)) - return( FALSE ); - - strcpy(stdname, descname); - if((ptr = strrchr(descname, '/')) == NULL) - ptr = descname; - else - ptr++; - stdname[(int) (ptr - descname)] = 0; - if(strncmp(ptr, "lib", 3)) - strcat(stdname, "lib"); - strcat(stdname, ptr); - if(strcmp(stdname + strlen(stdname) - 3, ".so")) - strcat(stdname, ".so"); - return( TRUE ); -} - -/* Return the greatest common divisor of a and b, or -1 if it is - not defined. Return through the pointer arguments the integers - such that gcd(a,b) = c*a + b*d. */ -int gcd(LLONG a, LLONG b, int *c, int *d) -{ - LLONG q,r,t; - int cret,dret,C,D,rval, sgn_a = 1,sgn_b = 1, swap = 0; - - if((a == 0) || (b == 0)) - return( -1 ); - - /* Use local multiplier instances, if necessary */ - if(c == NULL) - c = &cret; - if(d == NULL) - d = &dret; - - /* Normalize so that 0 < a <= b */ - if(a < 0){ - a = -a; - sgn_a = -1; - } - if(b < 0){ - b = -b; - sgn_b = -1; - } - if(b < a){ - t = b; - b = a; - a = t; - swap = 1; - } - - /* Now a <= b and both >= 1. */ - q = b/a; - r = b - a*q; - if(r == 0) { - if(swap){ - *d = 1; - *c = 0; - } - else { - *c = 1; - *d = 0; - } - *c = sgn_a*(*c); - *d = sgn_b*(*d); - return( (int) a ); - } - - rval = gcd(a,r,&C,&D); - if(swap){ - *d = (int) (C-D*q); - *c = D; - } - else { - *d = D; - *c = (int) (C-D*q); - } - *c = sgn_a*(*c); - *d = sgn_b*(*d); - return( rval ); -} - -/* Array search functions */ -int findIndex(int target, int *attributes, int count, int offset) -{ - int focusPos, beginPos, endPos; - int focusAttrib, beginAttrib, endAttrib; - - /* Set starting and ending index offsets */ - beginPos = offset; - endPos = beginPos + count - 1; - if(endPos < beginPos) - return(-1); - - /* Do binary search logic based on a sorted (decending) attribute vector */ - focusPos = (beginPos + endPos) / 2; - beginAttrib = attributes[beginPos]; - focusAttrib = attributes[focusPos]; - endAttrib = attributes[endPos]; - - while(endPos - beginPos > LINEARSEARCH) { - if(beginAttrib == target) { - focusAttrib = beginAttrib; - endPos = beginPos; - } - else if(endAttrib == target) { - focusAttrib = endAttrib; - beginPos = endPos; - } - else if(focusAttrib < target) { - beginPos = focusPos + 1; - beginAttrib = attributes[beginPos]; - focusPos = (beginPos + endPos) / 2; - focusAttrib = attributes[focusPos]; - } - else if(focusAttrib > target) { - endPos = focusPos - 1; - endAttrib = attributes[endPos]; - focusPos = (beginPos + endPos) / 2; - focusAttrib = attributes[focusPos]; - } - else { - beginPos = focusPos; - endPos = focusPos; - } - } - - /* Do linear (unsorted) search logic */ - if(endPos - beginPos <= LINEARSEARCH) { - - /* CPU intensive loop; provide alternative evaluation models */ -#if defined DOFASTMATH - /* Do fast pointer arithmetic */ - int *attptr = attributes + beginPos; - while((beginPos < endPos) && ((*attptr) < target)) { - beginPos++; - attptr++; - } - focusAttrib = (*attptr); -#else - /* Do traditional indexed access */ - focusAttrib = attributes[beginPos]; - while((beginPos < endPos) && (focusAttrib < target)) { - beginPos++; - focusAttrib = attributes[beginPos]; - } -#endif - } - - /* Return the index if a match was found, or signal failure with a -1 */ - if(focusAttrib == target) /* Found; return retrieval index */ - return(beginPos); - else if(focusAttrib > target) /* Not found; last item */ - return(-beginPos); - else if(beginPos > offset+count-1) - return(-(endPos+1)); /* Not found; end of list */ - else - return(-(beginPos+1)); /* Not found; intermediate point */ - -} -int findIndexEx(void *target, void *attributes, int count, int offset, int recsize, findCompare_func findCompare, MYBOOL ascending) -{ - int focusPos, beginPos, endPos, compare, order; - void *focusAttrib, *beginAttrib, *endAttrib; - - /* Set starting and ending index offsets */ - beginPos = offset; - endPos = beginPos + count - 1; - if(endPos < beginPos) - return(-1); - order = (ascending ? -1 : 1); - - /* Do binary search logic based on a sorted attribute vector */ - focusPos = (beginPos + endPos) / 2; - beginAttrib = CMP_ATTRIBUTES(beginPos); - focusAttrib = CMP_ATTRIBUTES(focusPos); - endAttrib = CMP_ATTRIBUTES(endPos); - - compare = 0; - while(endPos - beginPos > LINEARSEARCH) { - if(findCompare(target, beginAttrib) == 0) { - focusAttrib = beginAttrib; - endPos = beginPos; - } - else if(findCompare(target, endAttrib) == 0) { - focusAttrib = endAttrib; - beginPos = endPos; - } - else { - compare = findCompare(target, focusAttrib)*order; - if(compare < 0) { - beginPos = focusPos + 1; - beginAttrib = CMP_ATTRIBUTES(beginPos); - focusPos = (beginPos + endPos) / 2; - focusAttrib = CMP_ATTRIBUTES(focusPos); - } - else if(compare > 0) { - endPos = focusPos - 1; - endAttrib = CMP_ATTRIBUTES(endPos); - focusPos = (beginPos + endPos) / 2; - focusAttrib = CMP_ATTRIBUTES(focusPos); - } - else { - beginPos = focusPos; - endPos = focusPos; - } - } - } - - /* Do linear (unsorted) search logic */ - if(endPos - beginPos <= LINEARSEARCH) { - - /* Do traditional indexed access */ - focusAttrib = CMP_ATTRIBUTES(beginPos); - if(beginPos == endPos) - compare = findCompare(target, focusAttrib)*order; - else - while((beginPos < endPos) && - ((compare = findCompare(target, focusAttrib)*order) < 0)) { - beginPos++; - focusAttrib = CMP_ATTRIBUTES(beginPos); - } - } - - /* Return the index if a match was found, or signal failure with a -1 */ - if(compare == 0) /* Found; return retrieval index */ - return(beginPos); - else if(compare > 0) /* Not found; last item */ - return(-beginPos); - else if(beginPos > offset+count-1) - return(-(endPos+1)); /* Not found; end of list */ - else - return(-(beginPos+1)); /* Not found; intermediate point */ - -} - -/* Simple sorting and searching comparison "operators" */ -int CMP_CALLMODEL compareCHAR(const void *current, const void *candidate) -{ - return( CMP_COMPARE( *(char *) current, *(char *) candidate ) ); -} -int CMP_CALLMODEL compareINT(const void *current, const void *candidate) -{ - return( CMP_COMPARE( *(int *) current, *(int *) candidate ) ); -} -int CMP_CALLMODEL compareREAL(const void *current, const void *candidate) -{ - return( CMP_COMPARE( *(REAL *) current, *(REAL *) candidate ) ); -} - -/* Heap sort function (procedurally based on the Numerical Recipes version, - but expanded and generalized to hande any object with the use of - qsort-style comparison operator). An expanded version is also implemented, - where interchanges are reflected in a caller-initialized integer "tags" list. */ -void hpsort(void *attributes, int count, int offset, int recsize, MYBOOL descending, findCompare_func findCompare) -{ - register int i, j, k, ir, order; - register char *hold, *base; - char *save; - - if(count < 2) - return; - offset -= 1; - attributes = CMP_ATTRIBUTES(offset); - base = CMP_ATTRIBUTES(1); - save = (char *) malloc(recsize); - if(descending) - order = -1; - else - order = 1; - - k = (count >> 1) + 1; - ir = count; - - for(;;) { - if(k > 1) { - MEMCOPY(save, CMP_ATTRIBUTES(--k), recsize); - } - else { - hold = CMP_ATTRIBUTES(ir); - MEMCOPY(save, hold, recsize); - MEMCOPY(hold, base, recsize); - if(--ir == 1) { - MEMCOPY(base, save, recsize); - break; - } - } - - i = k; - j = k << 1; - while(j <= ir) { - hold = CMP_ATTRIBUTES(j); - if( (j < ir) && (findCompare(hold, CMP_ATTRIBUTES(j+1))*order < 0) ) { - hold += recsize; - j++; - } - if(findCompare(save, hold)*order < 0) { - MEMCOPY(CMP_ATTRIBUTES(i), hold, recsize); - i = j; - j <<= 1; - } - else - break; - } - MEMCOPY(CMP_ATTRIBUTES(i), save, recsize); - } - - FREE(save); -} -void hpsortex(void *attributes, int count, int offset, int recsize, MYBOOL descending, findCompare_func findCompare, int *tags) -{ - if(count < 2) - return; - if(tags == NULL) { - hpsort(attributes, count, offset, recsize, descending, findCompare); - return; - } - else { - register int i, j, k, ir, order; - register char *hold, *base; - char *save; - int savetag; - - offset -= 1; - attributes = CMP_ATTRIBUTES(offset); - tags += offset; - base = CMP_ATTRIBUTES(1); - save = (char *) malloc(recsize); - if(descending) - order = -1; - else - order = 1; - - k = (count >> 1) + 1; - ir = count; - - for(;;) { - if(k > 1) { - MEMCOPY(save, CMP_ATTRIBUTES(--k), recsize); - savetag = tags[k]; - } - else { - hold = CMP_ATTRIBUTES(ir); - MEMCOPY(save, hold, recsize); - MEMCOPY(hold, base, recsize); - savetag = tags[ir]; - tags[ir] = tags[1]; - if(--ir == 1) { - MEMCOPY(base, save, recsize); - tags[1] = savetag; - break; - } - } - - i = k; - j = k << 1; - while(j <= ir) { - hold = CMP_ATTRIBUTES(j); - if( (j < ir) && (findCompare(hold, CMP_ATTRIBUTES(j+1))*order < 0) ) { - hold += recsize; - j++; - } - if(findCompare(save, hold)*order < 0) { - MEMCOPY(CMP_ATTRIBUTES(i), hold, recsize); - tags[i] = tags[j]; - i = j; - j <<= 1; - } - else - break; - } - MEMCOPY(CMP_ATTRIBUTES(i), save, recsize); - tags[i] = savetag; - } - - FREE(save); - } -} - -/* This is a "specialized generic" version of C.A.R Hoare's Quick Sort algorithm. - It will handle arrays that are already sorted, and arrays with duplicate keys. - There are two versions here; one extended conventional with optional tag data - for each sortable value, and a version for the QSORTrec format. The QSORTrec - format i.a. includes the ability for to do linked list sorting. If the passed - comparison operator is NULL, the comparison is assumed to be for integers. */ -#define QS_IS_switch LINEARSEARCH /* Threshold for switching to insertion sort */ - -void qsortex_swap(void *attributes, int l, int r, int recsize, - void *tags, int tagsize, char *save, char *savetag) -{ - MEMCOPY(save, CMP_ATTRIBUTES(l), recsize); - MEMCOPY(CMP_ATTRIBUTES(l), CMP_ATTRIBUTES(r), recsize); - MEMCOPY(CMP_ATTRIBUTES(r), save, recsize); - if(tags != NULL) { - MEMCOPY(savetag, CMP_TAGS(l), tagsize); - MEMCOPY(CMP_TAGS(l), CMP_TAGS(r), tagsize); - MEMCOPY(CMP_TAGS(r), savetag, tagsize); - } -} - -int qsortex_sort(void *attributes, int l, int r, int recsize, int sortorder, findCompare_func findCompare, - void *tags, int tagsize, char *save, char *savetag) -{ - register int i, j, nmove = 0; - char *v; - - /* Perform the a fast QuickSort */ - if((r-l) > QS_IS_switch) { - i = (r+l)/2; - - /* Tri-Median Method */ - if(sortorder*findCompare(CMP_ATTRIBUTES(l), CMP_ATTRIBUTES(i)) > 0) - { nmove++; qsortex_swap(attributes, l,i, recsize, tags, tagsize, save, savetag); } - if(sortorder*findCompare(CMP_ATTRIBUTES(l), CMP_ATTRIBUTES(r)) > 0) - { nmove++; qsortex_swap(attributes, l,r, recsize, tags, tagsize, save, savetag); } - if(sortorder*findCompare(CMP_ATTRIBUTES(i), CMP_ATTRIBUTES(r)) > 0) - { nmove++; qsortex_swap(attributes, i,r, recsize, tags, tagsize, save, savetag); } - - j = r-1; - qsortex_swap(attributes, i,j, recsize, tags, tagsize, save, savetag); - i = l; - v = CMP_ATTRIBUTES(j); - for(;;) { - while(sortorder*findCompare(CMP_ATTRIBUTES(++i), v) < 0); - while(sortorder*findCompare(CMP_ATTRIBUTES(--j), v) > 0); - if(j < i) break; - nmove++; qsortex_swap(attributes, i,j, recsize, tags, tagsize, save, savetag); - } - nmove++; qsortex_swap(attributes, i,r-1, recsize, tags, tagsize, save, savetag); - nmove += qsortex_sort(attributes, l,j, recsize, sortorder, findCompare, tags, tagsize, save, savetag); - nmove += qsortex_sort(attributes, i+1,r, recsize, sortorder, findCompare, tags, tagsize, save, savetag); - } - return( nmove ); -} - -int qsortex_finish(void *attributes, int lo0, int hi0, int recsize, int sortorder, findCompare_func findCompare, - void *tags, int tagsize, char *save, char *savetag) -{ - int i, j, nmove = 0; - - /* This is actually InsertionSort, which is faster for local sorts */ - for(i = lo0+1; i <= hi0; i++) { - - /* Save bottom-most item */ - MEMCOPY(save, CMP_ATTRIBUTES(i), recsize); - if(tags != NULL) - MEMCOPY(savetag, CMP_TAGS(i), tagsize); - - /* Shift down! */ - j = i; - while ((j > lo0) && (sortorder*findCompare(CMP_ATTRIBUTES(j-1), save) > 0)) { - MEMCOPY(CMP_ATTRIBUTES(j), CMP_ATTRIBUTES(j-1), recsize); - if(tags != NULL) - MEMCOPY(CMP_TAGS(j), CMP_TAGS(j-1), tagsize); - j--; - nmove++; - } - - /* Store bottom-most item at the top */ - MEMCOPY(CMP_ATTRIBUTES(j), save, recsize); - if(tags != NULL) - MEMCOPY(CMP_TAGS(j), savetag, tagsize); - } - return( nmove ); -} - -int qsortex(void *attributes, int count, int offset, int recsize, MYBOOL descending, findCompare_func findCompare, void *tags, int tagsize) -{ - int iswaps = 0, sortorder = (descending ? -1 : 1); - char *save = NULL, *savetag = NULL; - - /* Check and initialize to zero-based arrays */ - if(count <= 1) - goto Finish; - attributes = (void *) CMP_ATTRIBUTES(offset); - save = (char *) malloc(recsize); - if((tagsize <= 0) && (tags != NULL)) - tags = NULL; - else if(tags != NULL) { - tags = (void *) CMP_TAGS(offset); - savetag = (char *) malloc(tagsize); - } - count--; - - /* Perform sort */ - iswaps = qsortex_sort(attributes, 0, count, recsize, sortorder, findCompare, tags, tagsize, save, savetag); -#if QS_IS_switch > 0 - iswaps += qsortex_finish(attributes, 0, count, recsize, sortorder, findCompare, tags, tagsize, save, savetag); -#endif - -Finish: - FREE(save); - FREE(savetag); - return( iswaps ); -} - -#undef QS_IS_switch - -/* This is a "specialized generic" version of C.A.R Hoare's Quick Sort algorithm. - It will handle arrays that are already sorted, and arrays with duplicate keys. - The implementation here requires the user to pass a comparison operator and - assumes that the array passed has the QSORTrec format, which i.a. includes - the ability for to do linked list sorting. If the passed comparison operator - is NULL, the comparison is assumed to be for integers. */ -#define QS_IS_switch 4 /* Threshold for switching to insertion sort */ - -void QS_swap(UNIONTYPE QSORTrec a[], int i, int j) -{ - UNIONTYPE QSORTrec T = a[i]; - a[i] = a[j]; - a[j] = T; -} -int QS_addfirst(UNIONTYPE QSORTrec a[], void *mydata) -{ - a[0].pvoid2.ptr = mydata; - return( 0 ); -} -int QS_append(UNIONTYPE QSORTrec a[], int ipos, void *mydata) -{ - if(ipos <= 0) - ipos = QS_addfirst(a, mydata); - else - a[ipos].pvoid2.ptr = mydata; - return( ipos ); -} -void QS_replace(UNIONTYPE QSORTrec a[], int ipos, void *mydata) -{ - a[ipos].pvoid2.ptr = mydata; -} -void QS_insert(UNIONTYPE QSORTrec a[], int ipos, void *mydata, int epos) -{ - for(; epos > ipos; epos--) - a[epos] = a[epos-1]; - a[ipos].pvoid2.ptr = mydata; -} -void QS_delete(UNIONTYPE QSORTrec a[], int ipos, int epos) -{ - for(; epos > ipos; epos--) - a[epos] = a[epos-1]; -} -int QS_sort(UNIONTYPE QSORTrec a[], int l, int r, findCompare_func findCompare) -{ - register int i, j, nmove = 0; - UNIONTYPE QSORTrec v; - - /* Perform the a fast QuickSort */ - if((r-l) > QS_IS_switch) { - i = (r+l)/2; - - /* Tri-Median Method */ - if(findCompare((char *) &a[l], (char *) &a[i]) > 0) - { nmove++; QS_swap(a,l,i); } - if(findCompare((char *) &a[l], (char *) &a[r]) > 0) - { nmove++; QS_swap(a,l,r); } - if(findCompare((char *) &a[i], (char *) &a[r]) > 0) - { nmove++; QS_swap(a,i,r); } - - j = r-1; - QS_swap(a,i,j); - i = l; - v = a[j]; - for(;;) { - while(findCompare((char *) &a[++i], (char *) &v) < 0); - while(findCompare((char *) &a[--j], (char *) &v) > 0); - if(j < i) break; - nmove++; QS_swap (a,i,j); - } - nmove++; QS_swap(a,i,r-1); - nmove += QS_sort(a,l,j,findCompare); - nmove += QS_sort(a,i+1,r,findCompare); - } - return( nmove ); -} -int QS_finish(UNIONTYPE QSORTrec a[], int lo0, int hi0, findCompare_func findCompare) -{ - int i, j, nmove = 0; - UNIONTYPE QSORTrec v; - - /* This is actually InsertionSort, which is faster for local sorts */ - for(i = lo0+1; i <= hi0; i++) { - - /* Save bottom-most item */ - v = a[i]; - - /* Shift down! */ - j = i; - while ((j > lo0) && (findCompare((char *) &a[j-1], (char *) &v) > 0)) { - a[j] = a[j-1]; - j--; - nmove++; - } - - /* Store bottom-most item at the top */ - a[j] = v; - } - return( nmove ); -} -MYBOOL QS_execute(UNIONTYPE QSORTrec a[], int count, findCompare_func findCompare, int *nswaps) -{ - int iswaps = 0; - - /* Check and initialize */ - if(count <= 1) - goto Finish; - count--; - - /* Perform sort */ - iswaps = QS_sort(a, 0, count, findCompare); -#if QS_IS_switch > 0 - iswaps += QS_finish(a, 0, count, findCompare); -#endif - -Finish: - if(nswaps != NULL) - *nswaps = iswaps; - return( TRUE ); -} - - - -/* Simple specialized bubble/insertion sort functions */ -int sortByREAL(int *item, REAL *weight, int size, int offset, MYBOOL unique) -{ - int i, ii, saveI; - REAL saveW; - - for(i = 1; i < size; i++) { - ii = i+offset-1; - while ((ii >= offset) && (weight[ii] >= weight[ii+1])) { - if(weight[ii] == weight[ii+1]) { - if(unique) - return(item[ii]); - } - else { - saveI = item[ii]; - saveW = weight[ii]; - item[ii] = item[ii+1]; - weight[ii] = weight[ii+1]; - item[ii+1] = saveI; - weight[ii+1] = saveW; - } - ii--; - } - } - return(0); -} -int sortByINT(int *item, int *weight, int size, int offset, MYBOOL unique) -{ - int i, ii, saveI; - int saveW; - - for(i = 1; i < size; i++) { - ii = i+offset-1; - while ((ii >= offset) && (weight[ii] >= weight[ii+1])) { - if(weight[ii] == weight[ii+1]) { - if(unique) - return(item[ii]); - } - else { - saveI = item[ii]; - saveW = weight[ii]; - item[ii] = item[ii+1]; - weight[ii] = weight[ii+1]; - item[ii+1] = saveI; - weight[ii+1] = saveW; - } - ii--; - } - } - return(0); -} -REAL sortREALByINT(REAL *item, int *weight, int size, int offset, MYBOOL unique) -{ - int i, ii, saveW; - REAL saveI; - - for(i = 1; i < size; i++) { - ii = i+offset-1; - while ((ii >= offset) && (weight[ii] >= weight[ii+1])) { - if(weight[ii] == weight[ii+1]) { - if(unique) - return(item[ii]); - } - else { - saveI = item[ii]; - saveW = weight[ii]; - item[ii] = item[ii+1]; - weight[ii] = weight[ii+1]; - item[ii+1] = saveI; - weight[ii+1] = saveW; - } - ii--; - } - } - return(0); -} - - -/* Time and message functions */ -double timeNow(void) -{ -#ifdef INTEGERTIME - return((double)time(NULL)); -#elif defined CLOCKTIME - return((double)clock()/CLOCKS_PER_SEC /* CLK_TCK */); -#elif defined PosixTime - struct timespec t; -# if 0 - clock_gettime(CLOCK_REALTIME, &t); - return( (double) t.tv_sec + (double) t.tv_nsec/1.0e9 ); -# else - static double timeBase; - - clock_gettime(CLOCK_MONOTONIC, &t); - if(timeBase == 0) - timeBase = clockNow() - ((double) t.tv_sec + (double) t.tv_nsec/1.0e9); - return( timeBase + (double) t.tv_sec + (double) t.tv_nsec/1.0e9 ); -# endif -#elif defined EnhTime - static LARGE_INTEGER freq; - static double timeBase; - LARGE_INTEGER now; - - QueryPerformanceCounter(&now); - if(timeBase == 0) { - QueryPerformanceFrequency(&freq); - timeBase = clockNow() - (double) now.QuadPart/(double) freq.QuadPart; - } - return( timeBase + (double) now.QuadPart/(double) freq.QuadPart ); -#else - struct timeb buf; - - ftime(&buf); - return((double)buf.time+((double) buf.millitm)/1000.0); -#endif -} - - -/* Miscellaneous reporting functions */ - -/* List a vector of INT values for the given index range */ -void blockWriteINT(FILE *output, char *label, int *myvector, int first, int last) -{ - int i, k = 0; - - fprintf(output, "%s", label); - fprintf(output, "\n"); - for(i = first; i <= last; i++) { - fprintf(output, " %5d", myvector[i]); - k++; - if(k % 12 == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(k % 12 != 0) - fprintf(output, "\n"); -} - -/* List a vector of MYBOOL values for the given index range */ -void blockWriteBOOL(FILE *output, char *label, MYBOOL *myvector, int first, int last, MYBOOL asRaw) -{ - int i, k = 0; - - fprintf(output, "%s", label); - fprintf(output, "\n"); - for(i = first; i <= last; i++) { - if(asRaw) - fprintf(output, " %1d", myvector[i]); - else - fprintf(output, " %5s", my_boolstr(myvector[i])); - k++; - if(k % 36 == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(k % 36 != 0) - fprintf(output, "\n"); -} - -/* List a vector of REAL values for the given index range */ -void blockWriteREAL(FILE *output, char *label, REAL *myvector, int first, int last) -{ - int i, k = 0; - - fprintf(output, "%s", label); - fprintf(output, "\n"); - for(i = first; i <= last; i++) { - fprintf(output, " %18g", myvector[i]); - k++; - if(k % 4 == 0) { - fprintf(output, "\n"); - k = 0; - } - } - if(k % 4 != 0) - fprintf(output, "\n"); -} - - -/* CONSOLE vector and matrix printing routines */ -void printvec( int n, REAL *x, int modulo ) -{ - int i; - - if (modulo <= 0) modulo = 5; - for (i = 1; i<=n; i++) { - if(mod(i, modulo) == 1) - printf("\n%2d:%12g", i, x[i]); - else - printf(" %2d:%12g", i, x[i]); - } - if(i % modulo != 0) printf("\n"); -} - - -void printmatUT( int size, int n, REAL *U, int modulo) -{ - int i, ll; - ll = 0; - for(i = 1; i<=n; i++) { - printvec(n-i+1, &U[ll], modulo); - ll += size-i+1; - } -} - - -void printmatSQ( int size, int n, REAL *X, int modulo) -{ - int i, ll; - ll = 0; - for(i = 1; i<=n; i++) { - printvec(n, &X[ll], modulo); - ll += size; - } -} - -/* Miscellaneous file functions */ -#if defined _MSC_VER -/* Check MS versions before 7 */ -#if _MSC_VER < 1300 -# define intptr_t long -#endif - -int fileCount( char *filemask ) -{ - struct _finddata_t c_file; - intptr_t hFile; - int count = 0; - - /* Find first .c file in current directory */ - if( (hFile = _findfirst( filemask, &c_file )) == -1L ) - ; - /* Iterate over all matching names */ - else { - while( _findnext( hFile, &c_file ) == 0 ) - count++; - _findclose( hFile ); - } - return( count ); -} -MYBOOL fileSearchPath( char *envvar, char *searchfile, char *foundpath ) -{ - char pathbuffer[_MAX_PATH]; - - _searchenv( searchfile, envvar, pathbuffer ); - if(pathbuffer[0] == '\0') - return( FALSE ); - else { - if(foundpath != NULL) - strcpy(foundpath, pathbuffer); - return( TRUE ); - } -} -#endif diff --git a/code/3rd_lpsolve/shared/commonlib.h b/code/3rd_lpsolve/shared/commonlib.h deleted file mode 100644 index e1c78892..00000000 --- a/code/3rd_lpsolve/shared/commonlib.h +++ /dev/null @@ -1,334 +0,0 @@ -#ifndef HEADER_commonlib -#define HEADER_commonlib - -#include -#include -#ifdef WIN32 - #include -#endif - -/* static char SpaceChars[3] = {" " "\7"}; */ -/* static char NumChars[14] = {"0123456789-+."}; */ - -#define BIGNUMBER 1.0e+30 -#define TINYNUMBER 1.0e-04 -#define MACHINEPREC 2.22e-16 -#define MATHPREC 1.0e-16 -#define ERRLIMIT 1.0e-06 - -#ifndef LINEARSEARCH - #define LINEARSEARCH 5 -#endif - -#if 0 - #define INTEGERTIME -#endif - -/* ************************************************************************ */ -/* Define loadable library function headers */ -/* ************************************************************************ */ -#if (defined WIN32) || (defined WIN64) - #define my_LoadLibrary(name) LoadLibrary(name) - #define my_GetProcAddress(handle, name) GetProcAddress(handle, name) - #define my_FreeLibrary(handle) FreeLibrary(handle); \ - handle = NULL -#else - #define my_LoadLibrary(name) dlopen(name, RTLD_LAZY) - #define my_GetProcAddress(handle, name) dlsym(handle, name) - #define my_FreeLibrary(handle) dlclose(handle); \ - handle = NULL -#endif - - -/* ************************************************************************ */ -/* Define sizes of standard number types */ -/* ************************************************************************ */ -#ifndef LLONG - #if defined __BORLANDC__ - #define LLONG __int64 - #elif !defined _MSC_VER || _MSC_VER >= 1310 - #define LLONG long long - #else - #define LLONG __int64 - #endif -#endif - -#ifndef MYBOOL - #if 0 - #define MYBOOL unsigned int - #else - #define MYBOOL unsigned char - #endif -#endif - -#ifndef REAL - #define REAL double -#endif -#ifndef BLAS_prec - #define BLAS_prec "d" /* The BLAS precision prefix must correspond to the REAL type */ -#endif - -#ifndef REALXP - #if 1 - #define REALXP long double /* Set local accumulation variable as long double */ - #else - #define REALXP REAL /* Set local accumulation as default precision */ - #endif -#endif - -#ifndef my_boolstr - #define my_boolstr(x) (!(x) ? "FALSE" : "TRUE") -#endif - -#ifndef NULL - #define NULL 0 -#endif - -#ifndef FALSE - #define FALSE 0 - #define TRUE 1 -#endif - -#ifndef DEF_STRBUFSIZE - #define DEF_STRBUFSIZE 512 -#endif -#ifndef MAXINT32 - #define MAXINT32 2147483647 -#endif -#ifndef MAXUINT32 - #define MAXUINT32 4294967295 -#endif - -#ifndef MAXINT64 - #if defined _LONGLONG || defined __LONG_LONG_MAX__ || defined LLONG_MAX - #define MAXINT64 9223372036854775807ll - #else - #define MAXINT64 9223372036854775807l - #endif -#endif -#ifndef MAXUINT64 - #if defined _LONGLONG || defined __LONG_LONG_MAX__ || defined LLONG_MAX - #define MAXUINT64 18446744073709551615ll - #else - #define MAXUINT64 18446744073709551615l - #endif -#endif - -#ifndef DOFASTMATH - #define DOFASTMATH -#endif - - -#ifndef CALLOC -#define CALLOC(ptr, nr)\ - if(!((void *) ptr = calloc((size_t)(nr), sizeof(*ptr))) && nr) {\ - printf("calloc of %d bytes failed on line %d of file %s\n",\ - (size_t) nr * sizeof(*ptr), __LINE__, __FILE__);\ - } -#endif - -#ifndef MALLOC -#define MALLOC(ptr, nr)\ - if(!((void *) ptr = malloc((size_t)((size_t) (nr) * sizeof(*ptr)))) && nr) {\ - printf("malloc of %d bytes failed on line %d of file %s\n",\ - (size_t) nr * sizeof(*ptr), __LINE__, __FILE__);\ - } -#endif - -#ifndef REALLOC -#define REALLOC(ptr, nr)\ - if(!((void *) ptr = realloc(ptr, (size_t)((size_t) (nr) * sizeof(*ptr)))) && nr) {\ - printf("realloc of %d bytes failed on line %d of file %s\n",\ - (size_t) nr * sizeof(*ptr), __LINE__, __FILE__);\ - } -#endif - -#ifndef FREE -#define FREE(ptr)\ - if((void *) ptr != NULL) {\ - free(ptr);\ - ptr = NULL; \ - } -#endif - -#ifndef MEMCOPY -#define MEMCOPY(nptr, optr, nr)\ - memcpy((nptr), (optr), (size_t)((size_t)(nr) * sizeof(*(optr)))) -#endif - -#ifndef MEMMOVE -#define MEMMOVE(nptr, optr, nr)\ - memmove((nptr), (optr), (size_t)((size_t)(nr) * sizeof(*(optr)))) -#endif - -#ifndef MEMALLOCCOPY -#define MEMALLOCCOPY(nptr, optr, nr)\ - {MALLOC(nptr, (size_t)(nr));\ - MEMCOPY(nptr, optr, (size_t)(nr));} -#endif - -#ifndef STRALLOCCOPY -#define STRALLOCCOPY(nstr, ostr)\ - {nstr = (char *) malloc((size_t) (strlen(ostr) + 1));\ - strcpy(nstr, ostr);} -#endif - -#ifndef MEMCLEAR -/*#define useMMX*/ -#ifdef useMMX - #define MEMCLEAR(ptr, nr)\ - mem_set((ptr), '\0', (size_t)((size_t)(nr) * sizeof(*(ptr)))) -#else - #define MEMCLEAR(ptr, nr)\ - memset((ptr), '\0', (size_t)((size_t)(nr) * sizeof(*(ptr)))) -#endif -#endif - - -#define MIN(x, y) ((x) < (y) ? (x) : (y)) -#define MAX(x, y) ((x) > (y) ? (x) : (y)) -#define SETMIN(x, y) if((x) > (y)) x = y -#define SETMAX(x, y) if((x) < (y)) x = y -#define LIMIT(lo, x, hi) ((x < (lo) ? lo : ((x) > hi ? hi : x))) -#define BETWEEN(x, a, b) (MYBOOL) (((x)-(a)) * ((x)-(b)) <= 0) -#define IF(t, x, y) ((t) ? (x) : (y)) -#define SIGN(x) ((x) < 0 ? -1 : 1) - -#define DELTA_SIZE(newSize, oldSize) ((int) ((newSize) * MIN(1.33, pow(1.5, fabs((double)newSize)/((oldSize+newSize)+1))))) - -#ifndef CMP_CALLMODEL -#if (defined WIN32) || (defined WIN64) - #define CMP_CALLMODEL _cdecl -#else - #define CMP_CALLMODEL -#endif -#endif - -typedef int (CMP_CALLMODEL findCompare_func)(const void *current, const void *candidate); -#define CMP_COMPARE(current, candidate) ( current < candidate ? -1 : (current > candidate ? 1 : 0) ) -#define CMP_ATTRIBUTES(item) (((char *) attributes)+(item)*recsize) -#define CMP_TAGS(item) (((char *) tags)+(item)*tagsize) - -#ifndef UNIONTYPE - #ifdef __cplusplus - #define UNIONTYPE - #else - #define UNIONTYPE union - #endif -#endif - -/* This defines a 16 byte sort record (in both 32 and 64 bit OS-es) */ -typedef struct _QSORTrec1 -{ - void *ptr; - void *ptr2; -} QSORTrec1; -typedef struct _QSORTrec2 -{ - void *ptr; - double realval; -} QSORTrec2; -typedef struct _QSORTrec3 -{ - void *ptr; - int intval; - int intpar1; -} QSORTrec3; -typedef struct _QSORTrec4 -{ - REAL realval; - int intval; - int intpar1; -} QSORTrec4; -typedef struct _QSORTrec5 -{ - double realval; - long int longval; -} QSORTrec5; -typedef struct _QSORTrec6 -{ - double realval; - double realpar1; -} QSORTrec6; -typedef struct _QSORTrec7 -{ - int intval; - int intpar1; - int intpar2; - int intpar3; -} QSORTrec7; -union QSORTrec -{ - QSORTrec1 pvoid2; - QSORTrec2 pvoidreal; - QSORTrec3 pvoidint2; - QSORTrec4 realint2; - QSORTrec5 reallong; - QSORTrec6 real2; - QSORTrec7 int4; -}; - - -#ifdef __cplusplus - extern "C" { -#endif - -int intpow(int base, int exponent); -int mod(int n, int d); - -void strtoup(char *s); -void strtolo(char *s); -void strcpyup(char *t, char *s); -void strcpylo(char *t, char *s); - -MYBOOL so_stdname(char *stdname, char *descname, int buflen); -int gcd(LLONG a, LLONG b, int *c, int *d); - -int findIndex(int target, int *attributes, int count, int offset); -int findIndexEx(void *target, void *attributes, int count, int offset, int recsize, findCompare_func findCompare, MYBOOL ascending); - -void qsortex_swap(void *attributes, int l, int r, int recsize, - void *tags, int tagsize, char *save, char *savetag); - -int qsortex(void *attributes, int count, int offset, int recsize, MYBOOL descending, findCompare_func findCompare, void *tags, int tagsize); - -int CMP_CALLMODEL compareCHAR(const void *current, const void *candidate); -int CMP_CALLMODEL compareINT(const void *current, const void *candidate); -int CMP_CALLMODEL compareREAL(const void *current, const void *candidate); -void hpsort(void *attributes, int count, int offset, int recsize, MYBOOL descending, findCompare_func findCompare); -void hpsortex(void *attributes, int count, int offset, int recsize, MYBOOL descending, findCompare_func findCompare, int *tags); - -void QS_swap(UNIONTYPE QSORTrec a[], int i, int j); -int QS_addfirst(UNIONTYPE QSORTrec a[], void *mydata); -int QS_append(UNIONTYPE QSORTrec a[], int ipos, void *mydata); -void QS_replace(UNIONTYPE QSORTrec a[], int ipos, void *mydata); -void QS_insert(UNIONTYPE QSORTrec a[], int ipos, void *mydata, int epos); -void QS_delete(UNIONTYPE QSORTrec a[], int ipos, int epos); -MYBOOL QS_execute(UNIONTYPE QSORTrec a[], int count, findCompare_func findCompare, int *nswaps); - -int sortByREAL(int *item, REAL *weight, int size, int offset, MYBOOL unique); -int sortByINT(int *item, int *weight, int size, int offset, MYBOOL unique); -REAL sortREALByINT(REAL *item, int *weight, int size, int offset, MYBOOL unique); - -double timeNow(void); - -void blockWriteBOOL(FILE *output, char *label, MYBOOL *myvector, int first, int last, MYBOOL asRaw); -void blockWriteINT(FILE *output, char *label, int *myvector, int first, int last); -void blockWriteREAL(FILE *output, char *label, REAL *myvector, int first, int last); - -void printvec( int n, REAL *x, int modulo ); -void printmatSQ( int size, int n, REAL *X, int modulo ); -void printmatUT( int size, int n, REAL *U, int modulo ); - -unsigned int catchFPU(unsigned int mask); - -#if defined _MSC_VER -int fileCount( char *filemask ); -MYBOOL fileSearchPath( char *envvar, char *searchfile, char *foundpath ); -#endif - -#ifdef __cplusplus - } -#endif - -#endif /* HEADER_commonlib */ diff --git a/code/3rd_lpsolve/shared/mmio.c b/code/3rd_lpsolve/shared/mmio.c deleted file mode 100644 index 185ef51b..00000000 --- a/code/3rd_lpsolve/shared/mmio.c +++ /dev/null @@ -1,496 +0,0 @@ -/* -* Matrix Market I/O library for ANSI C -* -* See http://math.nist.gov/MatrixMarket for details. -* -* (Version 1.01, 5/2003) -*/ - - -#include -#include -#include -#include - -#include "mmio.h" - -int mm_read_unsymmetric_sparse(const char *fname, int *M_, int *N_, int *nz_, - double **val_, int **I_, int **J_) -{ - FILE *f; - MM_typecode matcode; - int M, N, nz; - int i; - double *val; - int *I, *J; - int x; - - if ((f = fopen(fname, "r")) == NULL) - return -1; - - - if (mm_read_banner(f, &matcode) != 0) - { - printf("mm_read_unsymetric: Could not process Matrix Market banner "); - printf(" in file [%s]\n", fname); - return -1; - } - - - - if ( !(mm_is_real(matcode) && mm_is_matrix(matcode) && - mm_is_sparse(matcode))) - { - fprintf(stderr, "Sorry, this application does not support "); - fprintf(stderr, "Market Market type: [%s]\n", - mm_typecode_to_str(matcode)); - return -1; - } - - /* find out size of sparse matrix: M, N, nz .... */ - - if (mm_read_mtx_crd_size(f, &M, &N, &nz) !=0) - { - fprintf(stderr, "read_unsymmetric_sparse(): could not parse matrix size.\n"); - return -1; - } - - *M_ = M; - *N_ = N; - *nz_ = nz; - - /* reseve memory for matrices */ - - I = (int *) malloc(nz * sizeof(int)); - J = (int *) malloc(nz * sizeof(int)); - val = (double *) malloc(nz * sizeof(double)); - - *val_ = val; - *I_ = I; - *J_ = J; - - /* NOTE: when reading in doubles, ANSI C requires the use of the "l" */ - /* specifier as in "%lg", "%lf", "%le", otherwise errors will occur */ - /* (ANSI C X3.159-1989, Sec. 4.9.6.2, p. 136 lines 13-15) */ - - for (i=0; i= 2) - return 0; - - else - do { - num_items_read = fscanf(f, "%d %d %d", M, N, nz); - if (num_items_read == EOF) return MM_PREMATURE_EOF; - } while (num_items_read < 2); - - return 0; -} - - -int mm_read_mtx_array_size(FILE *f, int *M, int *N) -{ - char line[MM_MAX_LINE_LENGTH]; - int num_items_read; - /* set return null parameter values, in case we exit with errors */ - *M = *N = 0; - - /* now continue scanning until you reach the end-of-comments */ - do - { - if (fgets(line,MM_MAX_LINE_LENGTH,f) == NULL) - return MM_PREMATURE_EOF; - }while (line[0] == '%'); - - /* line[] is either blank or has M,N, nz */ - if (sscanf(line, "%d %d", M, N) == 2) - return 0; - - else /* we have a blank line */ - do - { - num_items_read = fscanf(f, "%d %d", M, N); - if (num_items_read == EOF) return MM_PREMATURE_EOF; - } - while (num_items_read != 2); - - return 0; -} - -int mm_write_mtx_array_size(FILE *f, int M, int N) -{ - if (fprintf(f, "%d %d\n", M, N) < 0) - return MM_COULD_NOT_WRITE_FILE; - else - return 0; -} - - - -/*-------------------------------------------------------------------------*/ - -/******************************************************************/ -/* use when I[], J[], and val[]J, and val[] are already allocated */ -/******************************************************************/ - -int mm_read_mtx_crd_data(FILE *f, int M, int N, int nz, int I[], int J[], - double val[], MM_typecode matcode) -{ - int i; - if (mm_is_complex(matcode)) - { - for (i=0; i -#include -/*#include */ -#include -#include -#include "myblas.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -/* ************************************************************************ */ -/* Initialize BLAS interfacing routines */ -/* ************************************************************************ */ -MYBOOL mustinitBLAS = TRUE; -#ifdef WIN32 - HINSTANCE hBLAS = NULL; -#else - void *hBLAS = NULL; -#endif - - -/* ************************************************************************ */ -/* Function pointers for external BLAS library (C base 0) */ -/* ************************************************************************ */ -BLAS_dscal_func *BLAS_dscal; -BLAS_dcopy_func *BLAS_dcopy; -BLAS_daxpy_func *BLAS_daxpy; -BLAS_dswap_func *BLAS_dswap; -BLAS_ddot_func *BLAS_ddot; -BLAS_idamax_func *BLAS_idamax; -BLAS_dload_func *BLAS_dload; -BLAS_dnormi_func *BLAS_dnormi; - - -/* ************************************************************************ */ -/* Define the BLAS interfacing routines */ -/* ************************************************************************ */ - -void init_BLAS(void) -{ - if(mustinitBLAS) { - load_BLAS(NULL); - mustinitBLAS = FALSE; - } -} - -MYBOOL is_nativeBLAS(void) -{ -#ifdef LoadableBlasLib - return( (MYBOOL) (hBLAS == NULL) ); -#else - return( TRUE ); -#endif -} - -MYBOOL load_BLAS(char *libname) -{ - MYBOOL result = TRUE; - -#ifdef LoadableBlasLib - if(hBLAS != NULL) { - #ifdef WIN32 - FreeLibrary(hBLAS); - #else - dlclose(hBLAS); - #endif - hBLAS = NULL; - } -#endif - - if(libname == NULL) { - if(!mustinitBLAS && is_nativeBLAS()) - return( FALSE ); - BLAS_dscal = my_dscal; - BLAS_dcopy = my_dcopy; - BLAS_daxpy = my_daxpy; - BLAS_dswap = my_dswap; - BLAS_ddot = my_ddot; - BLAS_idamax = my_idamax; - BLAS_dload = my_dload; - BLAS_dnormi = my_dnormi; - if(mustinitBLAS) - mustinitBLAS = FALSE; - } - else { -#ifdef LoadableBlasLib - #ifdef WIN32 - /* Get a handle to the Windows DLL module. */ - hBLAS = LoadLibrary(libname); - - /* If the handle is valid, try to get the function addresses. */ - result = (MYBOOL) (hBLAS != NULL); - if(result) { - BLAS_dscal = (BLAS_dscal_func *) GetProcAddress(hBLAS, BLAS_prec "scal"); - BLAS_dcopy = (BLAS_dcopy_func *) GetProcAddress(hBLAS, BLAS_prec "copy"); - BLAS_daxpy = (BLAS_daxpy_func *) GetProcAddress(hBLAS, BLAS_prec "axpy"); - BLAS_dswap = (BLAS_dswap_func *) GetProcAddress(hBLAS, BLAS_prec "swap"); - BLAS_ddot = (BLAS_ddot_func *) GetProcAddress(hBLAS, BLAS_prec "dot"); - BLAS_idamax = (BLAS_idamax_func *) GetProcAddress(hBLAS, "i" BLAS_prec "amax"); -#if 0 - BLAS_dload = (BLAS_dload_func *) GetProcAddress(hBLAS, BLAS_prec "load"); - BLAS_dnormi = (BLAS_dnormi_func *) GetProcAddress(hBLAS, BLAS_prec "normi"); -#endif - } - #else - /* First standardize UNIX .SO library name format. */ - char blasname[260], *ptr; - - strcpy(blasname, libname); - if((ptr = strrchr(libname, '/')) == NULL) - ptr = libname; - else - ptr++; - blasname[(int) (ptr - libname)] = 0; - if(strncmp(ptr, "lib", 3)) - strcat(blasname, "lib"); - strcat(blasname, ptr); - if(strcmp(blasname + strlen(blasname) - 3, ".so")) - strcat(blasname, ".so"); - - /* Get a handle to the module. */ - hBLAS = dlopen(blasname, RTLD_LAZY); - - /* If the handle is valid, try to get the function addresses. */ - result = (MYBOOL) (hBLAS != NULL); - if(result) { - BLAS_dscal = (BLAS_dscal_func *) dlsym(hBLAS, BLAS_prec "scal"); - BLAS_dcopy = (BLAS_dcopy_func *) dlsym(hBLAS, BLAS_prec "copy"); - BLAS_daxpy = (BLAS_daxpy_func *) dlsym(hBLAS, BLAS_prec "axpy"); - BLAS_dswap = (BLAS_dswap_func *) dlsym(hBLAS, BLAS_prec "swap"); - BLAS_ddot = (BLAS_ddot_func *) dlsym(hBLAS, BLAS_prec "dot"); - BLAS_idamax = (BLAS_idamax_func *) dlsym(hBLAS, "i" BLAS_prec "amax"); -#if 0 - BLAS_dload = (BLAS_dload_func *) dlsym(hBLAS, BLAS_prec "load"); - BLAS_dnormi = (BLAS_dnormi_func *) dlsym(hBLAS, BLAS_prec "normi"); -#endif - } - #endif -#endif - /* Do validation */ - if(!result || - ((BLAS_dscal == NULL) || - (BLAS_dcopy == NULL) || - (BLAS_daxpy == NULL) || - (BLAS_dswap == NULL) || - (BLAS_ddot == NULL) || - (BLAS_idamax == NULL) || - (BLAS_dload == NULL) || - (BLAS_dnormi == NULL)) - ) { - load_BLAS(NULL); - result = FALSE; - } - } - return( result ); -} -MYBOOL unload_BLAS(void) -{ - return( load_BLAS(NULL) ); -} - - -/* ************************************************************************ */ -/* Now define the unoptimized local BLAS functions */ -/* ************************************************************************ */ -void daxpy( int n, REAL da, REAL *dx, int incx, REAL *dy, int incy) -{ - dx++; - dy++; - BLAS_daxpy( &n, &da, dx, &incx, dy, &incy); -} -void BLAS_CALLMODEL my_daxpy( int *_n, REAL *_da, REAL *dx, int *_incx, REAL *dy, int *_incy) -{ - -/* constant times a vector plus a vector. - uses unrolled loops for increments equal to one. - jack dongarra, linpack, 3/11/78. - modified 12/3/93, array[1] declarations changed to array[*] */ - - int i, ix, iy; -#if !defined DOFASTMATH - int m, mp1; -#endif - register REAL rda; - REAL da = *_da; - int n = *_n, incx = *_incx, incy = *_incy; - - if (n <= 0) return; - if (da == 0.0) return; - - dx--; - dy--; - ix = 1; - iy = 1; - if (incx < 0) - ix = (-n+1)*incx + 1; - if (incy < 0) - iy = (-n+1)*incy + 1; - rda = da; - -/* CPU intensive loop; option to do pointer arithmetic */ -#if defined DOFASTMATH - { - REAL *xptr, *yptr; - for (i = 1, xptr = dx + ix, yptr = dy + iy; - i <= n; i++, xptr += incx, yptr += incy) - (*yptr) += rda*(*xptr); - return; - } -#else - - if (incx==1 && incy==1) goto x20; - -/* code for unequal increments or equal increments not equal to 1 */ - for (i = 1; i<=n; i++) { - dy[iy]+= rda*dx[ix]; - ix+= incx; - iy+= incy; - } - return; - -/* code for both increments equal to 1 */ - -/* clean-up loop */ -x20: - m = n % 4; - if (m == 0) goto x40; - for (i = 1; i<=m; i++) - dy[i]+= rda*dx[i]; - if(n < 4) return; -x40: - mp1 = m + 1; - for (i = mp1; i<=n; i=i+4) { - dy[i]+= rda*dx[i]; - dy[i + 1]+= rda*dx[i + 1]; - dy[i + 2]+= rda*dx[i + 2]; - dy[i + 3]+= rda*dx[i + 3]; - } -#endif -} - - -/* ************************************************************************ */ -void dcopy( int n, REAL *dx, int incx, REAL *dy, int incy) -{ - dx++; - dy++; - BLAS_dcopy( &n, dx, &incx, dy, &incy); -} - -void BLAS_CALLMODEL my_dcopy (int *_n, REAL *dx, int *_incx, REAL *dy, int *_incy) -{ - -/* copies a vector, x, to a vector, y. - uses unrolled loops for increments equal to one. - jack dongarra, linpack, 3/11/78. - modified 12/3/93, array[1] declarations changed to array[*] */ - - int i, ix, iy; -#if !defined DOFASTMATH - int m, mp1; -#endif - int n = *_n, incx = *_incx, incy = *_incy; - - if (n<=0) return; - - dx--; - dy--; - ix = 1; - iy = 1; - if (incx<0) - ix = (-n+1)*incx + 1; - if (incy<0) - iy = (-n+1)*incy + 1; - - -/* CPU intensive loop; option to do pointer arithmetic */ -#if defined DOFASTMATH - { - REAL *xptr, *yptr; - for (i = 1, xptr = dx + ix, yptr = dy + iy; - i <= n; i++, xptr += incx, yptr += incy) - (*yptr) = (*xptr); - return; - } -#else - - if (incx==1 && incy==1) goto x20; - -/* code for unequal increments or equal increments not equal to 1 */ - - for (i = 1; i<=n; i++) { - dy[iy] = dx[ix]; - ix+= incx; - iy+= incy; - } - return; - -/* code for both increments equal to 1 */ - -/* version with fast machine copy logic (requires memory.h or string.h) */ -x20: -#if defined DOFASTMATH - MEMCOPY(&dy[1], &dx[1], n); - return; -#else - - m = n % 7; - if (m == 0) goto x40; - for (i = 1; i<=m; i++) - dy[i] = dx[i]; - if (n < 7) return; - -x40: - mp1 = m + 1; - for (i = mp1; i<=n; i=i+7) { - dy[i] = dx[i]; - dy[i + 1] = dx[i + 1]; - dy[i + 2] = dx[i + 2]; - dy[i + 3] = dx[i + 3]; - dy[i + 4] = dx[i + 4]; - dy[i + 5] = dx[i + 5]; - dy[i + 6] = dx[i + 6]; - } -#endif -#endif -} - - -/* ************************************************************************ */ - -void dscal (int n, REAL da, REAL *dx, int incx) -{ - dx++; - BLAS_dscal (&n, &da, dx, &incx); -} - -void BLAS_CALLMODEL my_dscal (int *_n, REAL *_da, REAL *dx, int *_incx) -{ - -/* Multiply a vector by a constant. - - --Input-- - N number of elements in input vector(s) - DA double precision scale factor - DX double precision vector with N elements - INCX storage spacing between elements of DX - - --Output-- - DX double precision result (unchanged if N.LE.0) - - Replace double precision DX by double precision DA*DX. - For I = 0 to N-1, replace DX(IX+I*INCX) with DA * DX(IX+I*INCX), - where IX = 1 if INCX .GE. 0, else IX = 1+(1-N)*INCX. */ - - int i; -#if !defined DOFASTMATH - int ix, m, mp1; -#endif - register REAL rda; - REAL da = *_da; - int n = *_n, incx = *_incx; - - if (n <= 0) - return; - rda = da; - - dx--; - -/* Optionally do fast pointer arithmetic */ -#if defined DOFASTMATH - { - REAL *xptr; - for (i = 1, xptr = dx + 1; i <= n; i++, xptr += incx) - (*xptr) *= rda; - return; - } -#else - - if (incx == 1) - goto x20; - -/* Code for increment not equal to 1 */ - ix = 1; - if (incx < 0) - ix = (-n+1)*incx + 1; - for(i = 1; i <= n; i++, ix += incx) - dx[ix] *= rda; - return; - -/* Code for increment equal to 1. */ -/* Clean-up loop so remaining vector length is a multiple of 5. */ -x20: - m = n % 5; - if (m == 0) goto x40; - for( i = 1; i <= m; i++) - dx[i] *= rda; - if (n < 5) - return; -x40: - mp1 = m + 1; - for(i = mp1; i <= n; i += 5) { - dx[i] *= rda; - dx[i+1] *= rda; - dx[i+2] *= rda; - dx[i+3] *= rda; - dx[i+4] *= rda; - } -#endif -} - - -/* ************************************************************************ */ - -REAL ddot(int n, REAL *dx, int incx, REAL *dy, int incy) -{ - dx++; - dy++; - return( BLAS_ddot (&n, dx, &incx, dy, &incy) ); -} - -REAL BLAS_CALLMODEL my_ddot(int *_n, REAL *dx, int *_incx, REAL *dy, int *_incy) -{ - -/* forms the dot product of two vectors. - uses unrolled loops for increments equal to one. - jack dongarra, linpack, 3/11/78. - modified 12/3/93, array[1] declarations changed to array[*] */ - - register REAL dtemp; - int i, ix, iy; -#if !defined DOFASTMATH - int m, mp1; -#endif - int n = *_n, incx = *_incx, incy = *_incy; - - dtemp = 0.0; - if (n<=0) - return( (REAL) dtemp); - - dx--; - dy--; - ix = 1; - iy = 1; - if (incx<0) - ix = (-n+1)*incx + 1; - if (incy<0) - iy = (-n+1)*incy + 1; - -/* CPU intensive loop; option to do pointer arithmetic */ - -#if defined DOFASTMATH - { - REAL *xptr, *yptr; - for (i = 1, xptr = dx + ix, yptr = dy + iy; - i <= n; i++, xptr += incx, yptr += incy) - dtemp+= (*yptr)*(*xptr); - return(dtemp); - } -#else - - if (incx==1 && incy==1) goto x20; - -/* code for unequal increments or equal increments not equal to 1 */ - - for (i = 1; i<=n; i++) { - dtemp+= dx[ix]*dy[iy]; - ix+= incx; - iy+= incy; - } - return(dtemp); - -/* code for both increments equal to 1 */ - -/* clean-up loop */ - -x20: - m = n % 5; - if (m == 0) goto x40; - for (i = 1; i<=m; i++) - dtemp+= dx[i]*dy[i]; - if (n < 5) goto x60; - -x40: - mp1 = m + 1; - for (i = mp1; i<=n; i=i+5) - dtemp+= dx[i]*dy[i] + dx[i + 1]*dy[i + 1] + - dx[i + 2]*dy[i + 2] + dx[i + 3]*dy[i + 3] + dx[i + 4]*dy[i + 4]; - -x60: - return(dtemp); -#endif -} - - -/* ************************************************************************ */ - -void dswap( int n, REAL *dx, int incx, REAL *dy, int incy ) -{ - dx++; - dy++; - BLAS_dswap( &n, dx, &incx, dy, &incy ); -} - -void BLAS_CALLMODEL my_dswap( int *_n, REAL *dx, int *_incx, REAL *dy, int *_incy ) -{ - int i, ix, iy; -#if !defined DOFASTMATH - int m, mp1, ns; - REAL dtemp2, dtemp3; -#endif - REAL dtemp1; - int n = *_n, incx = *_incx, incy = *_incy; - - if (n <= 0) return; - - dx--; - dy--; - ix = 1; - iy = 1; - if (incx < 0) - ix = (-n+1)*incx + 1; - if (incy < 0) - iy = (-n+1)*incy + 1; - -/* CPU intensive loop; option to do pointer arithmetic */ -#if defined DOFASTMATH - { - REAL *xptr, *yptr; - for (i = 1, xptr = dx + ix, yptr = dy + iy; - i <= n; i++, xptr += incx, yptr += incy) { - dtemp1 = (*xptr); - (*xptr) = (*yptr); - (*yptr) = dtemp1; - } - return; - } -#else - - if (incx == incy) { - if (incx <= 0) goto x5; - if (incx == 1) goto x20; - goto x60; - } - -/* code for unequal or nonpositive increments. */ -x5: - for (i = 1; i<=n; i++) { - dtemp1 = dx[ix]; - dx[ix] = dy[iy]; - dy[iy] = dtemp1; - ix+= incx; - iy+= incy; - } - return; - -/* code for both increments equal to 1. - clean-up loop so remaining vector length is a multiple of 3. */ -x20: - m = n % 3; - if (m == 0) goto x40; - for (i = 1; i<=m; i++) { - dtemp1 = dx[i]; - dx[i] = dy[i]; - dy[i] = dtemp1; - } - if (n < 3) return; - -x40: - mp1 = m + 1; - for (i = mp1; i<=n; i=i+3) { - dtemp1 = dx[i]; - dtemp2 = dx[i+1]; - dtemp3 = dx[i+2]; - dx[i] = dy[i]; - dx[i+1] = dy[i+1]; - dx[i+2] = dy[i+2]; - dy[i] = dtemp1; - dy[i+1] = dtemp2; - dy[i+2] = dtemp3; - } - return; - -/* code for equal, positive, non-unit increments. */ -x60: - ns = n*incx; - for (i = 1; i<=ns; i=i+incx) { - dtemp1 = dx[i]; - dx[i] = dy[i]; - dy[i] = dtemp1; - } -#endif -} - - -/* ************************************************************************ */ - -void dload(int n, REAL da, REAL *dx, int incx) -{ - dx++; - BLAS_dload (&n, &da, dx, &incx); -} - -void BLAS_CALLMODEL my_dload (int *_n, REAL *_da, REAL *dx, int *_incx) -{ -/* copies a scalar, a, to a vector, x. - uses unrolled loops when incx equals one. - - To change the precision of this program, run the change - program on dload.f - Alternatively, to make a single precision version append a - comment character to the start of all lines between sequential - precision > double - and - end precision > double - comments and delete the comment character at the start of all - lines between sequential - precision > single - and - end precision > single - comments. To make a double precision version interchange - the append and delete operations in these instructions. */ - - int i, ix, m, mp1; - REAL da = *_da; - int n = *_n, incx = *_incx; - - if (n<=0) return; - dx--; - if (incx==1) goto x20; - -/* code for incx not equal to 1 */ - - ix = 1; - if (incx<0) - ix = (-n+1)*incx + 1; - for (i = 1; i<=n; i++) { - dx[ix] = da; - ix+= incx; - } - return; - -/* code for incx equal to 1 and clean-up loop */ - -x20: - m = n % 7; - if (m == 0) goto x40; - for (i = 1; i<=m; i++) - dx[i] = da; - if (n < 7) return; - -x40: - mp1 = m + 1; - for (i = mp1; i<=n; i=i+7) { - dx[i] = da; - dx[i + 1] = da; - dx[i + 2] = da; - dx[i + 3] = da; - dx[i + 4] = da; - dx[i + 5] = da; - dx[i + 6] = da; - } -} - -/* ************************************************************************ */ -int idamax( int n, REAL *x, int is ) -{ - x++; - return ( BLAS_idamax( &n, x, &is ) ); -} - -int BLAS_CALLMODEL my_idamax( int *_n, REAL *x, int *_is ) -{ - register REAL xmax, xtest; - int i, imax = 0; -#if !defined DOFASTMATH - int ii; -#endif - int n = *_n, is = *_is; - - if((n < 1) || (is <= 0)) - return(imax); - imax = 1; - if(n == 1) - return(imax); - -#if defined DOFASTMATH - xmax = fabs(*x); - for (i = 2, x += is; i <= n; i++, x += is) { - xtest = fabs(*x); - if(xtest > xmax) { - xmax = xtest; - imax = i; - } - } -#else - x--; - ii = 1; - xmax = fabs(x[ii]); - for(i = 2, ii+ = is; i <= n; i++, ii+ = is) { - xtest = fabs(x[ii]); - if(xtest > xmax) { - xmax = xtest; - imax = i; - } - } -#endif - return(imax); -} - - -/* ************************************************************************ */ -REAL dnormi( int n, REAL *x ) -{ - x++; - return( BLAS_dnormi( &n, x ) ); -} - -REAL BLAS_CALLMODEL my_dnormi( int *_n, REAL *x ) -{ -/* =============================================================== - dnormi returns the infinity-norm of the vector x. - =============================================================== */ - int j; - register REAL hold, absval; - int n = *_n; - - x--; - hold = 0.0; -/* for(j = 1; j <= n; j++) */ - for(j = n; j > 0; j--) { - absval = fabs(x[j]); - hold = MAX( hold, absval ); - } - - return( hold ); -} - - -/* ************************************************************************ */ -/* Subvector and submatrix access routines (Fortran compatibility) */ -/* ************************************************************************ */ - -#ifndef UseMacroVector -int subvec( int item) -{ - return( item-1 ); -} -#endif - -int submat( int nrowb, int row, int col) -{ - return( nrowb*(col-1) + subvec(row) ); -} - -int posmat( int nrowb, int row, int col) -{ - return( submat(nrowb, row, col)+BLAS_BASE ); -} - -/* ************************************************************************ */ -/* Randomization functions */ -/* ************************************************************************ */ - -void randomseed(int seeds[]) -/* Simply create some default seed values */ -{ - seeds[1] = 123456; - seeds[2] = 234567; - seeds[3] = 345678; -} - -void randomdens( int n, REAL *x, REAL r1, REAL r2, REAL densty, int *seeds ) -{ -/* ------------------------------------------------------------------ - random generates a vector x[*] of random numbers - in the range (r1, r2) with (approximate) specified density. - seeds[*] must be initialized before the first call. - ------------------------------------------------------------------ */ - - int i; - REAL *y; - - y = (REAL *) malloc(sizeof(*y) * (n+1)); - ddrand( n, x, 1, seeds ); - ddrand( n, y, 1, seeds ); - - for (i = 1; i<=n; i++) { - if (y[i] < densty) - x[i] = r1 + (r2 - r1) * x[i]; - else - x[i] = 0.0; - } - free(y); -} - - -/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */ - -void ddrand( int n, REAL *x, int incx, int *seeds ) -{ - -/* ------------------------------------------------------------------ - ddrand fills a vector x with uniformly distributed random numbers - in the interval (0, 1) using a method due to Wichman and Hill. - - seeds[1..3] should be set to integer values - between 1 and 30000 before the first entry. - - Integer arithmetic up to 30323 is required. - - Blatantly copied from Wichman and Hill 19-January-1987. - 14-Feb-94. Original version. - 30 Jun 1999. seeds stored in an array. - 30 Jun 1999. This version of ddrand. - ------------------------------------------------------------------ */ - - int ix, xix; - - if (n < 1) return; - - for (ix = 1; ix<=1+(n-1)*incx; ix=ix+incx) { - seeds[1] = 171*(seeds[1] % 177) - 2*(seeds[1]/177); - seeds[2] = 172*(seeds[2] % 176) - 35*(seeds[2]/176); - seeds[3] = 170*(seeds[3] % 178) - 63*(seeds[3]/178); - - if (seeds[1] < 0) seeds[1] = seeds[1] + 30269; - if (seeds[2] < 0) seeds[2] = seeds[2] + 30307; - if (seeds[3] < 0) seeds[3] = seeds[3] + 30323; - - x[ix] = ((REAL) seeds[1])/30269.0 + - ((REAL) seeds[2])/30307.0 + - ((REAL) seeds[3])/30323.0; - xix = (int) x[ix]; - x[ix] = fabs(x[ix] - xix); - } - -} - diff --git a/code/3rd_lpsolve/shared/myblas.h b/code/3rd_lpsolve/shared/myblas.h deleted file mode 100644 index ea23df85..00000000 --- a/code/3rd_lpsolve/shared/myblas.h +++ /dev/null @@ -1,132 +0,0 @@ -#ifndef HEADER_myblas -#define HEADER_myblas - -/* ************************************************************************ */ -/* BLAS function interface with local and external loadable versions */ -/* Author: Kjell Eikland */ -/* Version: Initial version spring 2004 */ -/* Licence: LGPL */ -/* ************************************************************************ */ -/* Changes: 19 September 2004 Moved function pointer variable */ -/* declarations from myblas.h to myblas.c */ -/* to avoid linker problems with the Mac. */ -/* 20 April 2005 Modified all double types to REAL to self- */ -/* adjust to global settings. Note that BLAS */ -/* as of now does not have double double. */ -/* ************************************************************************ */ - -#define BLAS_BASE 1 -#define UseMacroVector -#if defined LoadableBlasLib -# if LoadableBlasLib == 0 -# undef LoadableBlasLib -# endif -#else -# define LoadableBlasLib -#endif - - -/* ************************************************************************ */ -/* Include necessary libraries */ -/* ************************************************************************ */ -#include "commonlib.h" -#ifdef LoadableBlasLib - #ifdef WIN32 - #include - #else - #include - #endif -#endif - - -#ifdef __cplusplus -extern "C" { -#endif - - -/* ************************************************************************ */ -/* BLAS functions */ -/* ************************************************************************ */ - -#ifndef BLAS_CALLMODEL -#ifdef WIN32 -# define BLAS_CALLMODEL _cdecl -#else -# define BLAS_CALLMODEL -#endif -#endif - -typedef void (BLAS_CALLMODEL BLAS_dscal_func) (int *n, REAL *da, REAL *dx, int *incx); -typedef void (BLAS_CALLMODEL BLAS_dcopy_func) (int *n, REAL *dx, int *incx, REAL *dy, int *incy); -typedef void (BLAS_CALLMODEL BLAS_daxpy_func) (int *n, REAL *da, REAL *dx, int *incx, REAL *dy, int *incy); -typedef void (BLAS_CALLMODEL BLAS_dswap_func) (int *n, REAL *dx, int *incx, REAL *dy, int *incy); -typedef double (BLAS_CALLMODEL BLAS_ddot_func) (int *n, REAL *dx, int *incx, REAL *dy, int *incy); -typedef int (BLAS_CALLMODEL BLAS_idamax_func)(int *n, REAL *x, int *is); -typedef void (BLAS_CALLMODEL BLAS_dload_func) (int *n, REAL *da, REAL *dx, int *incx); -typedef double (BLAS_CALLMODEL BLAS_dnormi_func)(int *n, REAL *x); - -#ifndef __WINAPI -# ifdef WIN32 -# define __WINAPI WINAPI -# else -# define __WINAPI -# endif -#endif - -void init_BLAS(void); -MYBOOL is_nativeBLAS(void); -MYBOOL load_BLAS(char *libname); -MYBOOL unload_BLAS(void); - -/* ************************************************************************ */ -/* User-callable BLAS definitions (C base 1) */ -/* ************************************************************************ */ -void dscal ( int n, REAL da, REAL *dx, int incx ); -void dcopy ( int n, REAL *dx, int incx, REAL *dy, int incy ); -void daxpy ( int n, REAL da, REAL *dx, int incx, REAL *dy, int incy ); -void dswap ( int n, REAL *dx, int incx, REAL *dy, int incy ); -REAL ddot ( int n, REAL *dx, int incx, REAL *dy, int incy ); -int idamax( int n, REAL *x, int is ); -void dload ( int n, REAL da, REAL *dx, int incx ); -REAL dnormi( int n, REAL *x ); - - -/* ************************************************************************ */ -/* Locally implemented BLAS functions (C base 0) */ -/* ************************************************************************ */ -void BLAS_CALLMODEL my_dscal ( int *n, REAL *da, REAL *dx, int *incx ); -void BLAS_CALLMODEL my_dcopy ( int *n, REAL *dx, int *incx, REAL *dy, int *incy ); -void BLAS_CALLMODEL my_daxpy ( int *n, REAL *da, REAL *dx, int *incx, REAL *dy, int *incy ); -void BLAS_CALLMODEL my_dswap ( int *n, REAL *dx, int *incx, REAL *dy, int *incy ); -REAL BLAS_CALLMODEL my_ddot ( int *n, REAL *dx, int *incx, REAL *dy, int *incy ); -int BLAS_CALLMODEL my_idamax( int *n, REAL *x, int *is ); -void BLAS_CALLMODEL my_dload ( int *n, REAL *da, REAL *dx, int *incx ); -REAL BLAS_CALLMODEL my_dnormi( int *n, REAL *x ); - - -/* ************************************************************************ */ -/* Subvector and submatrix access routines (Fortran compatibility) */ -/* ************************************************************************ */ -#ifdef UseMacroVector - #define subvec(item) (item - 1) -#else - int subvec( int item ); -#endif - -int submat( int nrowb, int row, int col ); -int posmat( int nrowb, int row, int col ); - - -/* ************************************************************************ */ -/* Randomization functions */ -/* ************************************************************************ */ -void randomseed(int *seeds); -void randomdens( int n, REAL *x, REAL r1, REAL r2, REAL densty, int *seeds); -void ddrand( int n, REAL *x, int incx, int *seeds ); - - -#ifdef __cplusplus -} -#endif - -#endif diff --git a/code/3rd_lpsolve/yacc_read.c b/code/3rd_lpsolve/yacc_read.c deleted file mode 100644 index e415a705..00000000 --- a/code/3rd_lpsolve/yacc_read.c +++ /dev/null @@ -1,1300 +0,0 @@ -/* - ============================================================================ - NAME : yacc_read.c - - PURPOSE : translation of lp-problem and storage in sparse matrix - - SHORT : Subroutines for yacc program to store the input in an intermediate - data-structure. The yacc and lex programs translate the input. First the - problemsize is determined and the date is read into an intermediate - structure, then readinput fills the sparse matrix. - - USAGE : call yyparse(); to start reading the input. call readinput(); to - fill the sparse matrix. - ============================================================================ - Rows : contains the amount of rows + 1. Rows-1 is the amount of constraints - (no bounds) Rows also contains the rownr 0 which is the objective function - - Columns : contains the amount of columns (different variable names found in - the constraints) - - Nonnuls : contains the amount of nonnuls = sum of different entries of all - columns in the constraints and in the objectfunction - - Hash_tab : contains all columnnames on the first level of the structure the - row information is kept under each column structure in a linked list (also - the objective funtion is in this structure) Bound information is also - stored under under the column name - - First_rside : points to a linked list containing all relational operators - and the righthandside values of the constraints the linked list is in - reversed order with respect to the rownumbers - ============================================================================ */ -#include -#include -#include -#include "lpkit.h" -#include "yacc_read.h" - -#ifdef FORTIFY -# include "lp_fortify.h" -#endif - -#define tol 1.0e-10 -#define coldatastep 100 - -#define HASHSIZE 10007 /* A prime number! */ - -struct structSOSvars { - char *name; - int col; - REAL weight; - struct structSOSvars *next; -}; - -struct structSOS { - char *name; - short type; - int Nvars; - int weight; - struct structSOSvars *SOSvars, *LastSOSvars; - struct structSOS *next; -}; - -struct SOSrow { - int col; - REAL value; - struct SOSrow *next; -}; - -struct SOSrowdata { - short type; - char *name; - struct SOSrow *SOSrow; -}; - -struct rside /* contains relational operator and rhs value */ -{ - int row; - REAL value; - REAL range_value; - struct rside *next; - short relat; - short range_relat; - char negate; - short SOStype; -}; - -struct column -{ - int row; - REAL value; - struct column *next; - struct column *prev; -}; - -struct structcoldata { - int must_be_int; - int must_be_sec; - int must_be_free; - REAL upbo; - REAL lowbo; - struct column *firstcol; - struct column *col; -}; - -static void error(parse_parm *pp, int verbose, char *string) -{ - if(pp == NULL) - report(NULL, CRITICAL, string); - else if(pp->Verbose >= verbose) - report(NULL, verbose, "%s on line %d\n", string, pp->lineno); -} - -/* - * error handling routine for yyparse() - */ -void read_error(parse_parm *pp, void *scanner, char *string) -{ - error(pp, CRITICAL, string); -} - -/* called when lex gets a fatal error */ -void lex_fatal_error(parse_parm *pp, void *scanner, char *msg) -{ - read_error(pp, scanner, msg); - longjmp(pp->jump_buf, 1); -} - -void add_row(parse_parm *pp) -{ - pp->Rows++; - pp->rs = NULL; - pp->Lin_term_count = 0; -} - -void add_sos_row(parse_parm *pp, short SOStype) -{ - if (pp->rs != NULL) - pp->rs->SOStype = SOStype; - pp->Rows++; - pp->rs = NULL; - pp->Lin_term_count = 0; -} - -void check_int_sec_sos_free_decl(parse_parm *pp, int within_int_decl, int within_sec_decl, int sos_decl0, int within_free_decl) -{ - pp->Ignore_int_decl = TRUE; - pp->Ignore_sec_decl = TRUE; - pp->Ignore_free_decl = TRUE; - pp->sos_decl = 0; - if(within_int_decl) { - pp->Ignore_int_decl = FALSE; - pp->int_decl = (char) within_int_decl; - if(within_sec_decl) - pp->Ignore_sec_decl = FALSE; - } - else if(within_sec_decl) { - pp->Ignore_sec_decl = FALSE; - } - else if(sos_decl0) { - pp->sos_decl = (char) sos_decl0; - } - else if(within_free_decl) { - pp->Ignore_free_decl = FALSE; - } -} - -static void add_int_var(parse_parm *pp, char *name, short int_decl) -{ - hashelem *hp; - - if((hp = findhash(name, pp->Hash_tab)) == NULL) { - char buf[256]; - - sprintf(buf, "Unknown variable %s declared integer, ignored", name); - error(pp, IMPORTANT, buf); - } - else if(pp->coldata[hp->index].must_be_int) { - char buf[256]; - - sprintf(buf, "Variable %s declared integer more than once, ignored", name); - error(pp, IMPORTANT, buf); - } - else { - pp->coldata[hp->index].must_be_int = TRUE; - if(int_decl == 2) { - if(pp->coldata[hp->index].lowbo != -DEF_INFINITE * (REAL) 10.0) { - char buf[256]; - - sprintf(buf, "Variable %s: lower bound on variable redefined", name); - error(pp, IMPORTANT, buf); - } - pp->coldata[hp->index].lowbo = 0; - if(pp->coldata[hp->index].upbo < DEF_INFINITE) { - char buf[256]; - - sprintf(buf, "Variable %s: upper bound on variable redefined", name); - error(pp, IMPORTANT, buf); - } - pp->coldata[hp->index].upbo = 1; - } - else if(int_decl == 3) { - if(pp->coldata[hp->index].upbo == DEF_INFINITE * (REAL) 10.0) - pp->coldata[hp->index].upbo = 1.0; - } - } -} - -static void add_sec_var(parse_parm *pp, char *name) -{ - hashelem *hp; - - if((hp = findhash(name, pp->Hash_tab)) == NULL) { - char buf[256]; - - sprintf(buf, "Unknown variable %s declared semi-continuous, ignored", name); - error(pp, IMPORTANT, buf); - } - else if(pp->coldata[hp->index].must_be_sec) { - char buf[256]; - - sprintf(buf, "Variable %s declared semi-continuous more than once, ignored", name); - error(pp, IMPORTANT, buf); - } - else - pp->coldata[hp->index].must_be_sec = TRUE; -} - -int set_sec_threshold(parse_parm *pp, char *name, REAL threshold) -{ - hashelem *hp; - - if((hp = findhash(name, pp->Hash_tab)) == NULL) { - char buf[256]; - - sprintf(buf, "Unknown variable %s declared semi-continuous, ignored", name); - error(pp, IMPORTANT, buf); - return(FALSE); - } - - if ((pp->coldata[hp->index].lowbo > 0.0) && (threshold > 0.0)) { - char buf[256]; - - pp->coldata[hp->index].must_be_sec = FALSE; - sprintf(buf, "Variable %s declared semi-continuous, but it has a non-negative lower bound (%f), ignored", name, pp->coldata[hp->index].lowbo); - error(pp, IMPORTANT, buf); - } - if (threshold > pp->coldata[hp->index].lowbo) - pp->coldata[hp->index].lowbo = threshold; - - return(pp->coldata[hp->index].must_be_sec); -} - -static void add_free_var(parse_parm *pp, char *name) -{ - hashelem *hp; - - if((hp = findhash(name, pp->Hash_tab)) == NULL) { - char buf[256]; - - sprintf(buf, "Unknown variable %s declared free, ignored", name); - error(pp, IMPORTANT, buf); - } - else if(pp->coldata[hp->index].must_be_free) { - char buf[256]; - - sprintf(buf, "Variable %s declared free more than once, ignored", name); - error(pp, IMPORTANT, buf); - } - else - { - if(pp->coldata[hp->index].lowbo != -DEF_INFINITE * (REAL) 10.0) { - char buf[256]; - - sprintf(buf, "Variable %s: lower bound on variable redefined", name); - error(pp, IMPORTANT, buf); - } - - if(pp->coldata[hp->index].upbo < DEF_INFINITE) { - char buf[256]; - - sprintf(buf, "Variable %s: upper bound on variable redefined", name); - error(pp, IMPORTANT, buf); - } - - pp->coldata[hp->index].must_be_free = TRUE; - } -} - -static int add_sos_name(parse_parm *pp, char *name) -{ - struct structSOS *SOS; - - if(CALLOC(SOS, 1, struct structSOS) == NULL) - return(FALSE); - - if(MALLOC(SOS->name, strlen(name) + 1, char) == NULL) - { - FREE(SOS); - return(FALSE); - } - strcpy(SOS->name, name); - SOS->type = 0; - - if(pp->FirstSOS == NULL) - pp->FirstSOS = SOS; - else - pp->LastSOS->next = SOS; - pp->LastSOS = SOS; - - return(TRUE); -} - -static int add_sos_var(parse_parm *pp, char *name) -{ - struct structSOSvars *SOSvar; - - if(name != NULL) { - if(CALLOC(SOSvar, 1, struct structSOSvars) == NULL) - return(FALSE); - - if(MALLOC(SOSvar->name, strlen(name) + 1, char) == NULL) - { - FREE(SOSvar); - return(FALSE); - } - strcpy(SOSvar->name, name); - - if(pp->LastSOS->SOSvars == NULL) - pp->LastSOS->SOSvars = SOSvar; - else - pp->LastSOS->LastSOSvars->next = SOSvar; - pp->LastSOS->LastSOSvars = SOSvar; - pp->LastSOS->Nvars = pp->LastSOS->Nvars + 1; - } - pp->LastSOS->LastSOSvars->weight = 0; - - return(TRUE); -} - -void storevarandweight(parse_parm *pp, char *name) -{ - if(!pp->Ignore_int_decl) { - add_int_var(pp, name, pp->int_decl); - if(!pp->Ignore_sec_decl) - add_sec_var(pp, name); - } - else if(!pp->Ignore_sec_decl) - add_sec_var(pp, name); - else if(pp->sos_decl==1) - add_sos_name(pp, name); - else if(pp->sos_decl==2) - add_sos_var(pp, name); - else if(!pp->Ignore_free_decl) - add_free_var(pp, name); -} - -int set_sos_type(parse_parm *pp, int SOStype) -{ - if(pp->LastSOS != NULL) - pp->LastSOS->type = (short) SOStype; - return(TRUE); -} - -int set_sos_weight(parse_parm *pp, double weight, int sos_decl) -{ - if(pp->LastSOS != NULL) { - if(sos_decl==1) - pp->LastSOS->weight = (int) (weight+.1); - else - pp->LastSOS->LastSOSvars->weight = weight; - } - return(TRUE); -} - -static int inccoldata(parse_parm *pp) -{ - long Columns = pp->Columns; - - if(Columns == 0) - CALLOC(pp->coldata, coldatastep, struct structcoldata); - else if((Columns%coldatastep) == 0) - REALLOC(pp->coldata, Columns + coldatastep, struct structcoldata); - - if(pp->coldata != NULL) { - pp->coldata[Columns].upbo = (REAL) DEF_INFINITE * (REAL) 10.0; - pp->coldata[Columns].lowbo = (REAL) -DEF_INFINITE * (REAL) 10.0; /* temporary. If still this value then 0 will be taken */ - pp->coldata[Columns].col = NULL; - pp->coldata[Columns].firstcol = NULL; - pp->coldata[Columns].must_be_int = FALSE; - pp->coldata[Columns].must_be_sec = FALSE; - pp->coldata[Columns].must_be_free = FALSE; - } - - return(pp->coldata != NULL); -} - -/* - * initialisation of hashstruct and globals. - */ -static int init_read(parse_parm *pp, int verbose) -{ - int ok = FALSE; - - pp->Verbose = verbose; - set_obj_dir(pp, TRUE); - pp->Rows = 0; - pp->Non_zeros = 0; - pp->Columns = 0; - pp->FirstSOS = pp->LastSOS = NULL; - pp->Lin_term_count = 0; - if (CALLOC(pp->First_rside, 1, struct rside) != NULL) { - pp->rs = pp->First_rside; - pp->rs->value = pp->rs->range_value = 0; - /* first row (nr 0) is always the objective function */ - pp->rs->relat = OF; - pp->rs->range_relat = -1; - pp->rs->SOStype = 0; - pp->Hash_tab = NULL; - pp->Hash_constraints = NULL; - if (((pp->Hash_tab = create_hash_table(HASHSIZE, 0)) == NULL) || - ((pp->Hash_constraints = create_hash_table(HASHSIZE, 0)) == NULL)){ - FREE(pp->First_rside); - FREE(pp->Hash_tab); - FREE(pp->Hash_constraints); - } - else - ok = TRUE; - } - return(ok); -} /* init */ - -/* - * clears the tmp_store variable after all information has been copied - */ -void null_tmp_store(parse_parm *pp, int init_Lin_term_count) -{ - pp->tmp_store.value = 0; - pp->tmp_store.rhs_value = 0; - FREE(pp->tmp_store.name); - if(init_Lin_term_count) - pp->Lin_term_count = 0; -} - -/* - * variable : pointer to text array with name of variable - * row : the rownumber of the constraint - * value : value of matrixelement - * A(row, variable). - * Sign : (global) determines the sign of value. - * store() : stores value in matrix - * A(row, variable). If A(row, variable) already contains data, - * value is added to the existing value. - */ -static int store(parse_parm *pp, char *variable, - int row, - REAL value) -{ - hashelem *h_tab_p; - struct column *col_p; - - if(value == 0) { - char buf[256]; - - sprintf(buf, "(store) Warning, variable %s has an effective coefficient of 0, Ignored", variable); - error(pp, NORMAL, buf); - /* return(TRUE); */ - } - - if((h_tab_p = findhash(variable, pp->Hash_tab)) == NULL) { - if (((h_tab_p = puthash(variable, pp->Columns, NULL, pp->Hash_tab)) == NULL) - ) return(FALSE); - inccoldata(pp); - pp->Columns++; /* counter for calloc of final array */ - if(value) { - if (CALLOC(col_p, 1, struct column) == NULL) - return(FALSE); - pp->Non_zeros++; /* for calloc of final arrays */ - col_p->row = row; - col_p->value = value; - pp->coldata[h_tab_p->index].firstcol = pp->coldata[h_tab_p->index].col = col_p; - } - } - else if((pp->coldata[h_tab_p->index].col == NULL) || (pp->coldata[h_tab_p->index].col->row != row)) { - if(value) { - if (CALLOC(col_p, 1, struct column) == NULL) - return(FALSE); - pp->Non_zeros++; /* for calloc of final arrays */ - if(pp->coldata[h_tab_p->index].col != NULL) - pp->coldata[h_tab_p->index].col->prev = col_p; - else - pp->coldata[h_tab_p->index].firstcol = col_p; - col_p->value = value; - col_p->row = row; - col_p->next = pp->coldata[h_tab_p->index].col; - pp->coldata[h_tab_p->index].col = col_p; - } - } - else if(value) { - pp->coldata[h_tab_p->index].col->value += value; - if(fabs(pp->coldata[h_tab_p->index].col->value) < tol) /* eliminitate rounding errors */ - pp->coldata[h_tab_p->index].col->value = 0; - } - return(TRUE); -} /* store */ - -static int storefirst(parse_parm *pp) -{ - struct rside *rp; - - if ((pp->rs != NULL) && (pp->rs->row == pp->tmp_store.row)) - return(TRUE); - - /* make space for the rhs information */ - if (CALLOC(rp, 1, struct rside) == NULL) - return(FALSE); - rp->next = pp->First_rside; - pp->First_rside = pp->rs = rp; - pp->rs->row = /* row */ pp->tmp_store.row; - pp->rs->value = pp->tmp_store.rhs_value; - pp->rs->relat = pp->tmp_store.relat; - pp->rs->range_relat = -1; - pp->rs->SOStype = 0; - - if(pp->tmp_store.name != NULL) { - if(pp->tmp_store.value != 0) { - if (!store(pp, pp->tmp_store.name, pp->tmp_store.row, pp->tmp_store.value)) - return(FALSE); - } - else { - char buf[256]; - - sprintf(buf, "Warning, variable %s has an effective coefficient of 0, ignored", pp->tmp_store.name); - error(pp, NORMAL, buf); - } - } - null_tmp_store(pp, FALSE); - return(TRUE); -} - -/* - * store relational operator given in yylex[0] in the rightside list. - * Also checks if it constraint was a bound and if so stores it in the - * boundslist - */ -int store_re_op(parse_parm *pp, char OP, int HadConstraint, int HadVar, int Had_lineair_sum) -{ - short tmp_relat; - - switch(OP) { - - case '=': - tmp_relat = EQ; - break; - - case '>': - tmp_relat = GE; - break; - - case '<': - tmp_relat = LE; - break; - - case 0: - if(pp->rs != NULL) - tmp_relat = pp->rs->relat; - else - tmp_relat = pp->tmp_store.relat; - break; - - default: - { - char buf[256]; - - sprintf(buf, "Error: unknown relational operator %c", OP); - error(pp, CRITICAL, buf); - } - return(FALSE); - break; - } - - if(/* pp->Lin_term_count > 1 */ HadConstraint && HadVar) {/* it is not a bound */ - if(pp->Lin_term_count <= 1) - if(!storefirst(pp)) - return(FALSE); - pp->rs->relat = tmp_relat; - } - else if(/* pp->Lin_term_count == 0 */ HadConstraint && !Had_lineair_sum /* HadVar */ /* && (pp->rs != NULL) */) { /* it is a range */ - if(pp->Lin_term_count == 1) - if(!storefirst(pp)) - return(FALSE); - if(pp->rs == NULL) { /* range before row, already reported */ - error(pp, CRITICAL, "Error: range for undefined row"); - return(FALSE); - } - - if(pp->rs->negate) - switch (tmp_relat) { - case LE: - tmp_relat = GE; - break; - case GE: - tmp_relat = LE; - break; - } - - if(pp->rs->range_relat != -1) { - error(pp, CRITICAL, "Error: There was already a range for this row"); - return(FALSE); - } - else if(tmp_relat == pp->rs->relat) { - error(pp, CRITICAL, "Error: relational operator for range is the same as relation operator for equation"); - return(FALSE); - } - else - pp->rs->range_relat = tmp_relat; - } - else /* could be a bound */ - pp->tmp_store.relat = tmp_relat; - - return(TRUE); -} /* store_re_op */ - -int negate_constraint(parse_parm *pp) -{ - if(pp->rs != NULL) - pp->rs->negate = TRUE; - - return(TRUE); -} - -/* - * store RHS value in the rightside structure - * if type = true then - */ -int rhs_store(parse_parm *pp, REAL value, int HadConstraint, int HadVar, int Had_lineair_sum) -{ - if(/* pp->Lin_term_count > 1 */ (HadConstraint && HadVar) || (pp->Rows == 0)){ /* not a bound */ - if (pp->Rows == 0) - value = -value; - /* if(pp->Lin_term_count < 2) */ - if(pp->rs == NULL) - pp->tmp_store.rhs_value += value; - else - - if(pp->rs == NULL) { - error(pp, CRITICAL, "Error: No variable specified"); - return(FALSE); - } - else - pp->rs->value += value; - } - else if(/* pp->Lin_term_count == 0 */ HadConstraint && !HadVar) { /* a range */ - if(pp->rs == NULL) /* if range before row, already reported */ - pp->tmp_store.rhs_value += value; - else if(pp->rs->range_relat < 0) /* was a bad range; ignore */; - else { - if(pp->rs->negate) - value = -value; - if(((pp->rs->relat == LE) && (pp->rs->range_relat == GE) && - (pp->rs->value < value)) || - ((pp->rs->relat == GE) && (pp->rs->range_relat == LE) && - (pp->rs->value > value)) || - ((pp->rs->relat == EQ) || (pp->rs->range_relat == EQ))) { - pp->rs->range_relat = -2; - error(pp, CRITICAL, "Error: range restriction conflicts"); - return(FALSE); - } - else - pp->rs->range_value += value; - } - } - else /* a bound */ - pp->tmp_store.rhs_value += value; - return(TRUE); -} /* RHS_store */ - -/* - * store all data in the right place - * count the amount of lineair terms in a constraint - * only store in data-structure if the constraint is not a bound - */ -int var_store(parse_parm *pp, char *var, REAL value, int HadConstraint, int HadVar, int Had_lineair_sum) -{ - int row; - - row = pp->Rows; - - /* also in a bound the same var name can occur more than once. Check for - this. Don't increment Lin_term_count */ - - if(pp->Lin_term_count != 1 || pp->tmp_store.name == NULL || strcmp(pp->tmp_store.name, var) != 0) - pp->Lin_term_count++; - - /* always store objective function with rownr == 0. */ - if(row == 0) - return(store(pp, var, row, value)); - - if(pp->Lin_term_count == 1) { /* don't store yet. could be a bound */ - if(MALLOC(pp->tmp_store.name, strlen(var) + 1, char) != NULL) - strcpy(pp->tmp_store.name, var); - pp->tmp_store.row = row; - pp->tmp_store.value += value; - return(TRUE); - } - - if(pp->Lin_term_count == 2) { /* now you can also store the first variable */ - if(!storefirst(pp)) - return(FALSE); - /* null_tmp_store(pp, FALSE); */ - } - - return(store(pp, var, row, value)); -} /* var_store */ - - - -/* - * store the information in tmp_store because it is a bound - */ -int store_bounds(parse_parm *pp, int warn) -{ - if(pp->tmp_store.value != 0) { - hashelem *h_tab_p; - REAL boundvalue; - - if((h_tab_p = findhash(pp->tmp_store.name, pp->Hash_tab)) == NULL) { - /* a new columnname is found, create an entry in the hashlist */ - if ((h_tab_p = puthash(pp->tmp_store.name, pp->Columns, NULL, pp->Hash_tab)) == NULL) { - error(pp, CRITICAL, "Not enough memory"); - return(FALSE); - } - inccoldata(pp); - pp->Columns++; /* counter for calloc of final array */ - } - - if(pp->tmp_store.value < 0) { /* divide by negative number, */ - /* relational operator may change */ - if(pp->tmp_store.relat == GE) - pp->tmp_store.relat = LE; - else if(pp->tmp_store.relat == LE) - pp->tmp_store.relat = GE; - } - - boundvalue = pp->tmp_store.rhs_value / pp->tmp_store.value; - -#if FALSE - /* Check sanity of bound; all variables should be positive */ - if( ((pp->tmp_store.relat == EQ) && (boundvalue < 0)) - || ((pp->tmp_store.relat == LE) && (boundvalue < 0))) { /* Error */ - error(pp, CRITICAL, "Error: variables must always be non-negative"); - return(FALSE); - } -#endif - -#if FALSE - if((pp->tmp_store.relat == GE) && (boundvalue <= 0)) /* Warning */ - error(pp, NORMAL, "Warning: useless bound; variables are always >= 0"); -#endif - - /* bound seems to be sane, add it */ - if((pp->tmp_store.relat == GE) || (pp->tmp_store.relat == EQ)) { - if(boundvalue > pp->coldata[h_tab_p->index].lowbo - tol) - pp->coldata[h_tab_p->index].lowbo = boundvalue; - else if(warn) - error(pp, NORMAL, "Ineffective lower bound, ignored"); - } - if((pp->tmp_store.relat == LE) || (pp->tmp_store.relat == EQ)) { - if(boundvalue < pp->coldata[h_tab_p->index].upbo + tol) - pp->coldata[h_tab_p->index].upbo = boundvalue; - else if (warn) - error(pp, NORMAL, "Ineffective upper bound, ignored"); - } - - /* check for empty range */ - if((warn) && (pp->coldata[h_tab_p->index].upbo + tol < pp->coldata[h_tab_p->index].lowbo)) { - error(pp, CRITICAL, "Error: bound contradicts earlier bounds"); - return(FALSE); - } - } - else /* pp->tmp_store.value = 0 ! */ { - char buf[256]; - - if((pp->tmp_store.rhs_value == 0) || - ((pp->tmp_store.rhs_value > 0) && (pp->tmp_store.relat == LE)) || - ((pp->tmp_store.rhs_value < 0) && (pp->tmp_store.relat == GE))) { - sprintf(buf, "Variable %s has an effective coefficient of 0 in bound, ignored", - pp->tmp_store.name); - if(warn) - error(pp, NORMAL, buf); - } - else { - sprintf(buf, "Error, variable %s has an effective coefficient of 0 in bound", - pp->tmp_store.name); - error(pp, CRITICAL, buf); - return(FALSE); - } - } - - /* null_tmp_store(pp, FALSE); */ - pp->tmp_store.rhs_value = 0; - - return(TRUE); -} /* store_bounds */ - -int set_title(parse_parm *pp, char *name) -{ - pp->title = strdup(name); - return(TRUE); -} - -int add_constraint_name(parse_parm *pp, char *name) -{ - int row; - hashelem *hp; - - if((hp = findhash(name, pp->Hash_constraints)) != NULL) { - row = hp->index; - pp->rs = pp->First_rside; - while ((pp->rs != NULL) && (pp->rs->row != row)) - pp->rs = pp->rs->next; - } - else { - row = pp->Rows; - if (((hp = puthash(name, row, NULL, pp->Hash_constraints)) == NULL) - ) return(FALSE); - if(row) - pp->rs = NULL; - } - - return(TRUE); -} - -/* - * transport the data from the intermediate structure to the sparse matrix - * and free the intermediate structure - */ -static int readinput(parse_parm *pp, lprec *lp) -{ - int i, i1, count, index, col; - struct column *cp, *tcp; - hashelem *hp; - struct rside *rp; - signed char *negateAndSOS = NULL; - REAL *row = NULL, a; - int *rowno = NULL; - MYBOOL SOSinMatrix = FALSE; - struct SOSrowdata *SOSrowdata = NULL; - struct SOSrow *SOSrow, *SOSrow1; - - if(lp != NULL) { - if (CALLOC(negateAndSOS, 1 + pp->Rows, signed char) == NULL) - return(FALSE); - - rp = pp->First_rside; - for(i = pp->Rows; (i >= 0) && (rp != NULL); i--) { - if(rp->SOStype == 0) - negateAndSOS[i] = (rp->negate ? -1 : 0); - else - negateAndSOS[i] = (signed char) rp->SOStype; - - rp = rp->next; - } - - /* fill names with the rownames */ - hp = pp->Hash_constraints->first; - while(hp != NULL) { - if (/* (negateAndSOS[hp->index] <= 0) && */ (!set_row_name(lp, hp->index, hp->name))) - return(FALSE); - hp = hp->nextelem; - } - } - - for(i = pp->Rows; i >= 0; i--) { - rp = pp->First_rside; - if((lp != NULL) && (rp != NULL)) { - if(rp->SOStype == 0) { - if (rp->negate) { - switch (rp->relat) { - case LE: - rp->relat = GE; - break; - case GE: - rp->relat = LE; - break; - } - switch (rp->range_relat) { - case LE: - rp->range_relat = GE; - break; - case GE: - rp->range_relat = LE; - break; - } - rp->range_value = -rp->range_value; - rp->value = -rp->value; - } - - if((rp->range_relat >= 0) && (rp->value == lp->infinite)) { - rp->value = rp->range_value; - rp->relat = rp->range_relat; - rp->range_relat = -1; - } - else if((rp->range_relat >= 0) && (rp->value == -lp->infinite)) { - rp->value = rp->range_value; - rp->relat = rp->range_relat; - rp->range_relat = -1; - } - if ((rp->range_relat >= 0) && (rp->range_value == rp->value)) { - rp->relat = EQ; - rp->range_relat = EQ; - } - if(i) { - set_constr_type(lp, i, rp->relat); - pp->relat[i] = rp->relat; - } - set_rh(lp, i, rp->value); - if (rp->range_relat >= 0) - set_rh_range(lp, i, rp->range_value - rp->value); - } - else { - SOSinMatrix = TRUE; - if(i) - pp->relat[i] = rp->relat; - } - } - if(rp != NULL) { - pp->First_rside = rp->next; - free(rp); /* free memory when data has been read */ - } - else - pp->First_rside = NULL; - } - - while(pp->First_rside != NULL) { - rp = pp->First_rside; - pp->First_rside = rp->next; - free(rp); /* free memory when data has been read */ - } - - /* start reading the Hash_list structure */ - index = 0; - - if((SOSinMatrix) && (CALLOC(SOSrowdata, 1 + pp->Rows, struct SOSrowdata) == NULL)) { - FREE(negateAndSOS); - FREE(row); - FREE(rowno); - return(FALSE); - } - - if((lp != NULL) && - ((MALLOC(row, 1 + pp->Rows, REAL) == NULL) || (MALLOC(rowno, 1 + pp->Rows, int) == NULL))) { - FREE(SOSrowdata); - FREE(negateAndSOS); - FREE(row); - FREE(rowno); - return(FALSE); - } - - /* for(i = 0; i < pp->Hash_tab->size; i++) { - hp = pp->Hash_tab->table[i]; */ - hp = pp->Hash_tab->first; - while(hp != NULL) { - count = 0; - index++; - cp = pp->coldata[hp->index].firstcol; - col = hp->index + 1; - while(cp != NULL) { - if(lp != NULL) { - if (negateAndSOS[cp->row] <= 0) { - rowno[count] = cp->row; - a = cp->value; - if (negateAndSOS[cp->row]) - a = -a; - row[count++] = a; - } - else { - if (MALLOC(SOSrow, 1, struct SOSrow) == NULL) { - FREE(SOSrowdata); - FREE(negateAndSOS); - FREE(row); - FREE(rowno); - return(FALSE); - } - if(SOSrowdata[cp->row].SOSrow == NULL) - SOSrowdata[cp->row].name = strdup(get_row_name(lp, cp->row)); - SOSrow->next = SOSrowdata[cp->row].SOSrow; - SOSrowdata[cp->row].SOSrow = SOSrow; - SOSrowdata[cp->row].type = negateAndSOS[cp->row]; - SOSrow->col = col; - SOSrow->value = cp->value; - } - } - tcp = cp; - /* cp = cp->next; */ - cp = cp->prev; - free(tcp); /* free memory when data has been read */ - } - - if(lp != NULL) { - add_columnex(lp, count, row, rowno); - /* check for bound */ - if(pp->coldata[hp->index].lowbo == -DEF_INFINITE * 10.0) - /* lp->orig_lowbo[pp->Rows+index] = 0.0; */ - set_lowbo(lp, index, 0); - else - /* lp->orig_lowbo[pp->Rows+index] = pp->coldata[hp->index].lowbo; */ - set_lowbo(lp, index, pp->coldata[hp->index].lowbo); - /* lp->orig_upbo[pp->Rows+index] = pp->coldata[hp->index].upbo; */ - if(pp->coldata[hp->index].upbo >= DEF_INFINITE) - set_upbo(lp, index, DEF_INFINITE); - else - set_upbo(lp, index, pp->coldata[hp->index].upbo); - - /* check if it must be an integer variable */ - if(pp->coldata[hp->index].must_be_int) { - /* lp->must_be_int[pp->Rows + index]=TRUE; */ - set_int(lp, index, TRUE); - } - if(pp->coldata[hp->index].must_be_sec) { - set_semicont(lp, index, TRUE); - } - if(pp->coldata[hp->index].must_be_free) { - set_unbounded(lp, index); - } - - /* copy name of column variable */ - if (!set_col_name(lp, index, hp->name)) { - FREE(SOSrowdata); - FREE(negateAndSOS); - FREE(row); - FREE(rowno); - return(FALSE); - } - - /* put matrix values in intermediate row */ - /* cp = hp->col; */ - /* cp = hp->firstcol; */ - } - - /* thp = hp; */ - /* hp = hp->next; */ - /* free(thp->name); */ - /* free(thp); */ /* free memory when data has been read */ - - hp = hp->nextelem; - - } - /* pp->Hash_tab->table[i] = NULL; */ - - FREE(pp->coldata); - - if(SOSrowdata != NULL) { - struct structSOS *structSOS; - struct structSOSvars *SOSvars, *SOSvars1; - int SOSweight = 0; - - for(i = 1; i <= pp->Rows; i++) { - SOSrow = SOSrowdata[i].SOSrow; - if(SOSrow != NULL) { - if(MALLOC(structSOS, 1, struct structSOS) == NULL) { - FREE(SOSrowdata); - FREE(negateAndSOS); - FREE(row); - FREE(rowno); - return(FALSE); - } - structSOS->Nvars = 0; - structSOS->type = SOSrowdata[i].type; - structSOS->weight = ++SOSweight; - structSOS->name = strdup(SOSrowdata[i].name); - structSOS->LastSOSvars = NULL; - structSOS->next = pp->FirstSOS; - pp->FirstSOS = structSOS; - SOSvars = NULL; - while(SOSrow != NULL) { - SOSvars1 = SOSvars; - MALLOC(SOSvars, 1, struct structSOSvars); - SOSvars->next = SOSvars1; - SOSvars->col = SOSrow->col; - SOSvars->weight = SOSrow->value; - SOSvars->name = NULL; - structSOS->Nvars++; - SOSrow1 = SOSrow->next; - FREE(SOSrow); - SOSrow = SOSrow1; - } - structSOS->SOSvars = SOSvars; - } - } - FREE(SOSrowdata); - } - - while(pp->FirstSOS != NULL) - { - struct structSOSvars *SOSvars, *SOSvars1; - int *sosvars, n, col; - REAL *weights; - hashelem *hp; - - pp->LastSOS = pp->FirstSOS; - pp->FirstSOS = pp->FirstSOS->next; - SOSvars = pp->LastSOS->SOSvars; - if(lp != NULL) { - MALLOC(sosvars, pp->LastSOS->Nvars, int); - MALLOC(weights, pp->LastSOS->Nvars, double); - } - else { - sosvars = NULL; - weights = NULL; - } - n = 0; - while(SOSvars != NULL) - { - SOSvars1 = SOSvars; - SOSvars = SOSvars->next; - if(lp != NULL) { - col = SOSvars1->col; - if(col == 0) - if((hp = findhash(SOSvars1->name, lp->colname_hashtab)) != NULL) - col = hp->index; - if (col) { - sosvars[n] = col; - weights[n++] = SOSvars1->weight; - } - } - FREE(SOSvars1->name); - FREE(SOSvars1); - } - if(lp != NULL) { - add_SOS(lp, pp->LastSOS->name, pp->LastSOS->type, pp->LastSOS->weight, n, sosvars, weights); - FREE(weights); - FREE(sosvars); - } - FREE(pp->LastSOS->name); - FREE(pp->LastSOS); - } - - if(negateAndSOS != NULL) { - for(i1 = 0, i = 1; i <= pp->Rows; i++) - if(negateAndSOS[i] <= 0) - pp->relat[++i1] = pp->relat[i]; - -#if 01 - for(i = pp->Rows; i > 0; i--) - if(negateAndSOS[i] > 0) { - del_constraint(lp, i); - pp->Rows--; - } -#endif - } - - /* the following should be replaced by a call to the MPS print routine MB */ - -#if 0 - if(pp->Verbose) { - int j; - - printf("\n"); - printf("**********Data read**********\n"); - printf("Rows : %d\n", pp->Rows); - printf("Columns : %d\n", pp->Columns); - printf("Nonnuls : %d\n", pp->Non_zeros); - printf("NAME LPPROB\n"); - printf("ROWS\n"); - for(i = 0; i <= pp->Rows; i++) { - if(pp->relat[i] == LE) - printf(" L "); - else if(pp->relat[i] == EQ) - printf(" E "); - else if(pp->relat[i] == GE) - printf(" G "); - else if(pp->relat[i] == OF) - printf(" N "); - printf("%s\n", get_row_name(lp, i)); - } - - printf("COLUMNS\n"); - j = 0; - for(i = 0; i < pp->Non_zeros; i++) { - if(i == lp->col_end[j]) - j++; - printf(" %-8s %-8s %g\n", get_col_name(lp, j), - get_row_name(lp, lp->mat[i].row_nr), (double)lp->mat[i].value); - } - - printf("RHS\n"); - for(i = 0; i <= pp->Rows; i++) { - printf(" RHS %-8s %g\n", get_row_name(lp, i), - (double)lp->orig_rhs[i]); - } - - printf("RANGES\n"); - for(i = 1; i <= pp->Rows; i++) - if((lp->orig_upbo[i] != lp->infinite) && (lp->orig_upbo[i] != 0)) { - printf(" RGS %-8s %g\n", get_row_name(lp, i), - (double)lp->orig_upbo[i]); - } - else if((lp->orig_lowbo[i] != 0)) { - printf(" RGS %-8s %g\n", get_row_name(lp, i), - (double)-lp->orig_lowbo[i]); - } - - printf("BOUNDS\n"); - for(i = pp->Rows + 1; i <= pp->Rows + pp->Columns; i++) { - if((lp->orig_lowbo[i] != 0) && (lp->orig_upbo[i] < lp->infinite) && - (lp->orig_lowbo[i] == lp->orig_upbo[i])) { - printf(" FX BND %-8s %g\n", get_col_name(lp, i - pp->Rows), - (double)lp->orig_upbo[i]); - } - else { - if(lp->orig_upbo[i] < lp->infinite) - printf(" UP BND %-8s %g\n", get_col_name(lp, i - pp->Rows), - (double)lp->orig_upbo[i]); - if(lp->orig_lowbo[i] > 0) - printf(" LO BND %-8s %g\n", get_col_name(lp, i - pp->Rows), - (double)lp->orig_lowbo[i]); - } - } - - printf("ENDATA\n"); - } -#endif - - FREE(row); - FREE(rowno); - FREE(negateAndSOS); - return(TRUE); -} /* readinput */ - -lprec *yacc_read(lprec *lp, int verbose, char *lp_name, int (*parse) (parse_parm *pp), parse_parm *pp, void (*delete_allocated_memory) (parse_parm *pp)) -{ - REAL *orig_upbo; - int stat = -1; - lprec *lp0 = lp; - - pp->title = lp_name; - - if(!init_read(pp, verbose)) - error(pp, CRITICAL, "init_read failed"); - else if (setjmp(pp->jump_buf) == 0) - stat = parse(pp); - - delete_allocated_memory(pp); - - pp->Rows--; - - pp->relat = NULL; - if((stat != 0) || (CALLOC(pp->relat, pp->Rows + 1, short) != NULL)) { - if(stat == 0) { - if(lp == NULL) { - lp = make_lp(pp->Rows, 0); - } - else { - int NRows; - - for(NRows = get_Nrows(lp); NRows < pp->Rows; NRows++) - add_constraintex(lp, 0, NULL, NULL, LE, 0); - } - } - else - lp = NULL; - if ((stat != 0) || (lp != NULL)) { - if(lp != NULL) { - set_verbose(lp, pp->Verbose); - } - - if (!readinput(pp, lp)) { - if((lp != NULL) && (lp0 == NULL)) - delete_lp(lp); - lp = NULL; - } - - if(lp != NULL) { - set_lp_name(lp, pp->title); - if(pp->Maximise) - set_maxim(lp); - - if(pp->Rows) { - int row; - - MALLOCCPY(orig_upbo, lp->orig_upbo, 1 + pp->Rows, REAL); - for(row = 1; row <= pp->Rows; row++) - set_constr_type(lp, row, pp->relat[row]); - - memcpy(lp->orig_upbo, orig_upbo, (1 + pp->Rows) * sizeof(*orig_upbo)); /* restore upper bounds (range) */ - FREE(orig_upbo); - } - } - if((pp->title != NULL) && (pp->title != lp_name)) - free(pp->title); - - free_hash_table(pp->Hash_tab); - free_hash_table(pp->Hash_constraints); - } - FREE(pp->relat); - } - null_tmp_store(pp, FALSE); - return(lp); -} diff --git a/code/3rd_lpsolve/yacc_read.h b/code/3rd_lpsolve/yacc_read.h deleted file mode 100644 index 5298e873..00000000 --- a/code/3rd_lpsolve/yacc_read.h +++ /dev/null @@ -1,57 +0,0 @@ -/* prototypes of functions used in the parser */ - -#include - -#ifndef __READ_H__ -#define __READ_H__ - -struct _tmp_store_struct -{ - char *name; - int row; - REAL value; - REAL rhs_value; - short relat; -}; - -typedef struct parse_parm_s -{ - void *scanner; - long lineno; - int Verbose; - jmp_buf jump_buf; - long Rows, Columns, Non_zeros, Lin_term_count; - struct rside *First_rside, *rs; - short SOStype; /* SOS type */ - char Ignore_int_decl, int_decl, Ignore_sec_decl, Ignore_free_decl, sos_decl, Maximise; - hashtable *Hash_tab, *Hash_constraints; - struct structcoldata *coldata; - struct structSOS *FirstSOS, *LastSOS; - struct _tmp_store_struct tmp_store; - char *title; - short *relat; - void *parse_vars; -} parse_parm; - -void lex_fatal_error(parse_parm *, void *, char *); -int set_title(parse_parm *pp, char *name); -int add_constraint_name(parse_parm *pp, char *name); -int store_re_op(parse_parm *pp, char OP, int HadConstraint, int HadVar, int Had_lineair_sum); -void null_tmp_store(parse_parm *pp, int init_Lin_term_count); -int store_bounds(parse_parm *pp, int warn); -void storevarandweight(parse_parm *pp, char *name); -int set_sos_type(parse_parm *pp, int SOStype); -int set_sos_weight(parse_parm *pp, double weight, int sos_decl); -int set_sec_threshold(parse_parm *pp, char *name, REAL threshold); -int rhs_store(parse_parm *pp, REAL value, int HadConstraint, int HadVar, int Had_lineair_sum); -int var_store(parse_parm *pp, char *var, REAL value, int HadConstraint, int HadVar, int Had_lineair_sum); -int negate_constraint(parse_parm *pp); -void add_row(parse_parm *pp); -void add_sos_row(parse_parm *pp, short SOStype); - -void read_error(parse_parm *, void *, char *); -void check_int_sec_sos_free_decl(parse_parm *, int, int, int, int); -lprec *yacc_read(lprec *lp, int verbose, char *lp_name, int (*parse) (parse_parm *pp), parse_parm *pp, void (*delete_allocated_memory) (parse_parm *pp)); - -#define set_obj_dir(pp, maximise) pp->Maximise = maximise -#endif diff --git a/code/CMakeLists.txt b/code/CMakeLists.txt index 964ffa56..c0ebced0 100644 --- a/code/CMakeLists.txt +++ b/code/CMakeLists.txt @@ -40,8 +40,6 @@ set(POLYFIT_ROOT ${CMAKE_CURRENT_LIST_DIR}) set(POLYFIT_INCLUDE_DIR ${POLYFIT_ROOT} ${CMAKE_CURRENT_BINARY_DIR}) set(POLYFIT_SOURCE_DIR ${POLYFIT_ROOT}) -set(POLYFIT_glpk_DIR ${POLYFIT_ROOT}/3rd_glpk) -set(POLYFIT_lpsolve_DIR ${POLYFIT_ROOT}/3rd_lpsolve) set(POLYFIT_qglviewer_DIR ${POLYFIT_ROOT}/3rd_QGLViewer) set(POLYFIT_scip_DIR ${POLYFIT_ROOT}/3rd_scip) set(POLYFIT_soplex_DIR ${POLYFIT_ROOT}/3rd_soplex) @@ -66,8 +64,6 @@ SET(CMAKE_RUNTIME_OUTPUT_DIRECTORY ################################################################################ -add_subdirectory(3rd_glpk) -add_subdirectory(3rd_lpsolve) add_subdirectory(3rd_QGLViewer) add_subdirectory(3rd_scip) add_subdirectory(3rd_soplex) diff --git a/code/Example/main.cpp b/code/Example/main.cpp index 73ecaadf..d9a1630f 100644 --- a/code/Example/main.cpp +++ b/code/Example/main.cpp @@ -73,7 +73,13 @@ int main(int argc, char **argv) std::cout << "optimization..." << std::endl; const auto& adjacency = hypothesis.extract_adjacency(mesh); FaceSelection selector(pset, mesh); + +#ifdef HAS_GUROBI + selector.optimize(adjacency, LinearProgramSolver::GUROBI); +#else selector.optimize(adjacency, LinearProgramSolver::SCIP); +#endif + if (mesh->size_of_facets() == 0) { std::cerr << "optimization failed: model has no face" << std::endl; return EXIT_FAILURE; diff --git a/code/PolyFit/main_window.cpp b/code/PolyFit/main_window.cpp index ec1d48c7..6806382d 100644 --- a/code/PolyFit/main_window.cpp +++ b/code/PolyFit/main_window.cpp @@ -598,14 +598,15 @@ void MainWindow::snapshotScreen() { LinearProgramSolver::SolverName MainWindow::active_solver() const { const QString& solverString = solverBox_->currentText(); - if (solverString == "GLPK") - return LinearProgramSolver::GLPK; + //if (solverString == "GLPK") + // return LinearProgramSolver::GLPK; #ifdef HAS_GUROBI - else if (solverString == "GUROBI") + if (solverString == "GUROBI") return LinearProgramSolver::GUROBI; #endif - else if (solverString == "LPSOLVE") - return LinearProgramSolver::LPSOLVE; - else // default to SCIP - return LinearProgramSolver::SCIP; + //else if (solverString == "LPSOLVE") + // return LinearProgramSolver::LPSOLVE; + + // default to SCIP + return LinearProgramSolver::SCIP; } diff --git a/code/math/CMakeLists.txt b/code/math/CMakeLists.txt index e266b074..55c6a3ff 100644 --- a/code/math/CMakeLists.txt +++ b/code/math/CMakeLists.txt @@ -27,8 +27,8 @@ set(math_SOURCES linear_program.cpp linear_program_io.cpp linear_program_solver.cpp - linear_program_solver_GLPK.cpp - linear_program_solver_LPSOLVE.cpp +# linear_program_solver_GLPK.cpp +# linear_program_solver_LPSOLVE.cpp linear_program_solver_SCIP.cpp linear_program_solver_GUROBI.cpp ) @@ -67,4 +67,11 @@ if (GUROBI_FOUND) endif () -target_link_libraries(${PROJECT_NAME} PRIVATE basic 3rd_scip 3rd_lpsolve 3rd_glpk 3rd_soplex ${CMAKE_DL_LIBS}) +target_link_libraries(${PROJECT_NAME} PRIVATE + basic + 3rd_scip +# 3rd_lpsolve +# 3rd_glpk + 3rd_soplex + ${CMAKE_DL_LIBS} + ) diff --git a/code/math/linear_program_solver.cpp b/code/math/linear_program_solver.cpp index dee81877..bb2d374d 100644 --- a/code/math/linear_program_solver.cpp +++ b/code/math/linear_program_solver.cpp @@ -61,10 +61,10 @@ bool LinearProgramSolver::solve(const LinearProgram* program, SolverName solver) case GUROBI: return _solve_GUROBI(program); #endif - case GLPK: - return _solve_GLPK(program); - case LPSOLVE: - return _solve_LPSOLVE(program); +// case GLPK: +// return _solve_GLPK(program); +// case LPSOLVE: +// return _solve_LPSOLVE(program); case SCIP: return _solve_SCIP(program); } diff --git a/code/math/linear_program_solver.h b/code/math/linear_program_solver.h index 5004cff6..d8d0f917 100644 --- a/code/math/linear_program_solver.h +++ b/code/math/linear_program_solver.h @@ -34,8 +34,8 @@ class MATH_API LinearProgramSolver GUROBI, #endif SCIP, // Recommended default value. - GLPK, - LPSOLVE, +// GLPK, +// LPSOLVE, }; public: @@ -69,8 +69,8 @@ class MATH_API LinearProgramSolver bool _solve_GUROBI(const LinearProgram* program); #endif bool _solve_SCIP(const LinearProgram* program); - bool _solve_GLPK(const LinearProgram* program); - bool _solve_LPSOLVE(const LinearProgram* program); +// bool _solve_GLPK(const LinearProgram* program); +// bool _solve_LPSOLVE(const LinearProgram* program); private: std::vector result_; diff --git a/code/renderer/opengl_info.cpp b/code/renderer/opengl_info.cpp index 404f7dea..8df7dff6 100644 --- a/code/renderer/opengl_info.cpp +++ b/code/renderer/opengl_info.cpp @@ -24,6 +24,9 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. #if defined(__APPLE__) && defined(__MACH__) # include #else +#ifdef WIN32 +# include +#endif # include #endif diff --git a/code/renderer/surface_render.cpp b/code/renderer/surface_render.cpp index 23a658cc..78a91ec8 100644 --- a/code/renderer/surface_render.cpp +++ b/code/renderer/surface_render.cpp @@ -29,6 +29,9 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. #if defined(__APPLE__) && defined(__MACH__) # include #else +#ifdef WIN32 +# include +#endif # include #endif