|
SoPlex Doxygen Documentation
|
Go to the documentation of this file.
31 #define FREE_LHS_RHS 1
32 #define FREE_CONSTRAINT 1
33 #define EMPTY_CONSTRAINT 1
34 #define ROW_SINGLETON 1
35 #define FORCE_CONSTRAINT 1
38 #define EMPTY_COLUMN 1
39 #define FIX_VARIABLE 1
40 #define FREE_ZERO_OBJ_VARIABLE 1
41 #define ZERO_OBJ_COL_SINGLETON 1
42 #define DOUBLETON_EQUATION 1
43 #define FREE_COL_SINGLETON 1
45 #define DOMINATED_COLUMN 1
46 #define WEAKLY_DOMINATED_COLUMN 1
54 #define DUPLICATE_ROWS 1
55 #define DUPLICATE_COLS 1
59 #define CHECK_BASIC_DIM 1
68 for( int rs = 0; rs < nRows; ++rs)
74 for( int cs = 0; cs < nCols; ++cs)
79 return numBasis == nRows;
89 rStatus[m_old_i] = rStatus[m_i];
94 for ( int k = 0; k < m_row.size(); ++k)
95 slack += m_row.value(k) * x[m_row.index(k)];
106 if (!checkBasisDim(rStatus, cStatus))
120 rStatus[m_old_i] = rStatus[m_i];
132 if (!checkBasisDim(rStatus, cStatus))
146 rStatus[m_old_i] = rStatus[m_i];
148 Real aij = m_col[m_i];
151 s[m_i] = aij * x[m_j];
156 for( int k = 0; k < m_col.size(); ++k)
157 if (m_col.index(k) != m_i)
158 val -= m_col.value(k) * y[m_col.index(k)];
160 Real m_newLo = (aij > 0) ? m_lhs/aij : m_rhs/aij;
161 Real m_newUp = (aij > 0) ? m_rhs/aij : m_lhs/aij;
166 if(m_newLo <= m_oldLo && m_newUp >= m_oldUp)
172 else if( EQrel(m_newLo, m_newUp, eps()))
175 assert( EQrel(m_newLo, x[m_j], eps()));
177 if( EQrel(m_oldLo, m_oldUp, eps()))
183 else if(( EQrel(m_oldLo, x[m_j], eps()) && r[m_j] <= -eps())
184 || ( EQrel(m_oldUp, x[m_j], eps()) && r[m_j] >= eps())
185 || (! EQrel(m_oldLo, x[m_j], eps()) && !( EQrel(m_oldUp, x[m_j], eps()))))
196 assert( EQrel(m_oldLo, x[m_j], eps()) || EQrel(m_oldUp, x[m_j], eps()));
204 else if( EQrel(m_newLo, m_oldUp, eps()))
210 assert( EQrel(m_rhs, x[m_j]*aij, eps()) || EQrel(m_lhs, x[m_j]*aij, eps()));
219 assert( EQrel(m_oldUp, x[m_j], eps()));
227 else if( EQrel(m_newUp, m_oldLo, eps()))
233 assert( EQrel(m_rhs, x[m_j]*aij, eps()) || EQrel(m_lhs, x[m_j]*aij, eps()));
242 assert( EQrel(m_oldLo, x[m_j], eps()));
263 if( EQrel(m_oldLo, x[m_j], eps()))
271 assert( EQrel(m_rhs, x[m_j]*aij, eps()) || EQrel(m_lhs, x[m_j]*aij, eps()));
280 if( EQrel(m_oldUp, x[m_j], eps()))
288 assert( EQrel(m_rhs, x[m_j]*aij, eps()) || EQrel(m_lhs, x[m_j]*aij, eps()));
305 if (!checkBasisDim(rStatus, cStatus))
319 rStatus[m_old_i] = rStatus[m_i];
325 int cBasisCandidate = -1;
326 Real maxViolation = -1.0;
329 for( int k = 0; k < m_row.size(); ++k)
331 int cIdx = m_row.index(k);
332 Real aij = m_row.value(k);
333 Real oldLo = m_oldLowers[k];
334 Real oldUp = m_oldUppers[k];
336 switch(cStatus[cIdx])
341 assert( EQrel(oldLo, x[cIdx], eps()) || EQrel(oldUp, x[cIdx], eps()));
343 Real violation = fabs(r[cIdx]/aij);
347 if( violation > maxViolation && ( ( EQrel(oldLo, x[cIdx], eps()) && r[cIdx] < -eps()) || ( EQrel(oldUp, x[cIdx], eps()) && r[cIdx] > eps()) ) )
349 maxViolation = violation;
350 cBasisCandidate = cIdx;
365 if(cBasisCandidate >= 0)
367 assert( EQrel(m_lRhs, m_rhs, eps()) || EQrel(m_lRhs, m_lhs, eps()));
369 assert(cBasisCandidate == m_row.index(bas_k));
374 Real aij = m_row.value(bas_k);
375 Real multiplier = r[cBasisCandidate]/aij;
376 r[cBasisCandidate] = 0.0;
378 for( int k = 0; k < m_row.size(); ++k)
384 r[m_row.index(k)] -= m_row.value(k) * multiplier;
388 Real val = m_objs[bas_k];
391 for( int k = 0; k < basis_col. size(); ++k)
393 if (basis_col. index(k) != m_i)
394 val -= basis_col. value(k) * y[basis_col. index(k)];
406 if (!checkBasisDim(rStatus, cStatus))
422 cStatus[m_old_j] = cStatus[m_j];
428 for( int k = 0; k < m_col.size(); ++k)
429 s[m_col.index(k)] += m_col.value(k) * x[m_j];
434 for( int k = 0; k < m_col.size(); ++k)
435 val -= m_col.value(k) * y[m_col.index(k)];
440 if( EQrel(m_lower, m_upper))
442 assert( EQrel(m_lower, m_val));
448 assert( EQrel(m_val, m_lower) || EQrel(m_val, m_upper) || m_val == 0.0);
456 if (!checkBasisDim(rStatus, cStatus))
469 cStatus[m_j] = m_status;
479 cStatus[m_old_j] = cStatus[m_j];
481 int rIdx = m_old_i - m_col.size() + 1;
483 for( int k = 0; k < m_col.size(); ++k)
485 int rIdx_new = m_col.index(k);
486 s[ rIdx] = s[rIdx_new];
487 y[ rIdx] = y[rIdx_new];
488 rStatus[ rIdx] = rStatus[rIdx_new];
500 for( int k = 0; k < m_rows.size(); ++k)
503 const SVector& row = m_rows[k];
505 for( int l = 0; l < row. size(); ++l)
506 if (row. index(l) != m_j)
514 Real z = (m_lRhs[k] / scale) - (val / scale);
519 Real up = z * scale / row[m_j];
529 if (m_bnd < minRowUp)
541 for( int k = 0; k < m_rows.size(); ++k)
544 const SVector& row = m_rows[k];
546 for( int l = 0; l < row. size(); ++l)
547 if (row. index(l) != m_j)
555 Real z = (m_lRhs[k] / scale) - (val / scale);
560 Real lo = z * scale / row[m_j];
570 if (m_bnd > maxRowLo)
579 for( int k = 0; k < m_col.size(); ++k)
580 s[m_col.index(k)] = slack[k] + m_col.value(k) * x[m_j];
585 for( int k = 0; k < m_col.size(); ++k)
586 y[m_col.index(k)] = 0.0;
589 for( int k = 0; k < m_col.size(); ++k)
612 if (!checkBasisDim(rStatus, cStatus))
626 cStatus[m_old_j] = cStatus[m_j];
629 Real aij = m_row[m_j];
642 Real z1 = (m_lhs / scale1) - (s[m_i] / scale1);
643 Real z2 = (m_rhs / scale2) - (s[m_i] / scale2);
650 Real lo = (aij > 0) ? z1 * scale1 / aij : z2 * scale2 / aij;
651 Real up = (aij > 0) ? z2 * scale2 / aij : z1 * scale1 / aij;
658 assert( LErel(lo, up));
663 if ( EQrel(m_lower, m_upper))
683 if ( EQrel(m_lower, m_upper))
703 assert( EQrel(m_lower, m_upper));
740 s[m_i] += aij * x[m_j];
743 r[m_j] = -1.0 * aij * y[m_i];
748 if (!checkBasisDim(rStatus, cStatus))
763 rStatus[m_old_i] = rStatus[m_i];
768 cStatus[m_old_j] = cStatus[m_j];
772 Real aij = m_row[m_j];
774 for( int k = 0; k < m_row.size(); ++k)
775 if (m_row.index(k) != m_j)
776 val += m_row.value(k) * x[m_row.index(k)];
783 Real z = (m_lRhs / scale) - (val / scale);
788 x[m_j] = z * scale / aij;
792 y[m_i] = m_obj / aij;
806 if (!checkBasisDim(rStatus, cStatus))
822 (( m_maxSense && ((r[m_j] > 0 && m_strictUp) || (r[m_j] < 0 && m_strictLo))) ||
823 (!m_maxSense && ((r[m_j] > 0 && m_strictLo) || (r[m_j] < 0 && m_strictUp)))))))
826 Real aik = m_col[m_i];
828 for( int _k = 0; _k < m_col.size(); ++_k)
829 if (m_col.index(_k) != m_i)
830 val -= m_col.value(_k) * y[m_col.index(_k)];
835 r[m_j] = m_jObj - val * m_aij / aik;
872 if (!checkBasisDim(rStatus, cStatus))
886 for( int i = m_perm.size() - 1; i >= 0; --i)
890 int rIdx_new = m_perm[i];
892 s[ rIdx] = s[rIdx_new];
893 y[ rIdx] = y[rIdx_new];
894 rStatus[ rIdx] = rStatus[rIdx_new];
900 for( int k = 0; k < m_scale.size(); ++k)
901 if (m_scale.index(k) != m_i)
902 s[m_scale.index(k)] = s[m_i] / m_scale.value(k);
905 bool haveSetBasis = false;
907 for( int k = 0; k < m_scale.size(); ++k)
909 int i = m_scale.index(k);
922 if (rStatus[m_i] == SPxSolver::FIXED && (i == m_maxLhsIdx || i == m_minRhsIdx))
925 y[i] = y[m_i] * m_scale.value(k);
928 if(m_isLhsEqualRhs[k])
932 else if(i == m_maxLhsIdx)
938 assert(i == m_minRhsIdx);
949 y[i] = y[m_i] * m_scale.value(k);
960 y[i] = y[m_i] * m_scale.value(k);
977 if(m_isFirst && !checkBasisDim(rStatus, cStatus))
997 if (!checkBasisDim(rStatus, cStatus))
1009 for( int i = m_perm.size() - 1; i >= 0; --i)
1013 int cIdx_new = m_perm[i];
1015 x[ cIdx] = x[cIdx_new];
1016 r[ cIdx] = r[cIdx_new];
1017 cStatus[ cIdx] = cStatus[cIdx_new];
1065 assert(x[m_k] == 0.0);
1066 assert( LErel(m_loJ, 0.0));
1067 assert( GErel(m_upJ, 0.0));
1068 assert( LErel(m_loK, 0.0));
1069 assert( GErel(m_upK, 0.0));
1077 else if ( LErel(m_loK, 0.0) && GErel(m_upK, 0.0))
1089 else if ( LErel(m_loJ, 0.0) && GErel(m_upJ, 0.0))
1104 Real z1 = (x[m_k] / scale1) - (m_loK / scale1);
1105 Real z2 = (x[m_k] / scale2) - (m_upK / scale2);
1112 if( m_loJ <= -infinity && m_upJ >= infinity && m_loK <= -infinity && m_upK >= infinity )
1117 else if( m_scale > 0.0 )
1119 if( GErel(x[m_k], m_upK + m_scale * m_upJ) )
1124 x[m_k] -= m_scale * x[m_j];
1126 else if( GErel(x[m_k], m_loK + m_scale * m_upJ) && m_upJ < infinity )
1130 x[m_k] -= m_scale * x[m_j];
1132 else if( GErel(x[m_k], m_upK + m_scale * m_loJ) && m_upK < infinity )
1137 x[m_j] = z2 * scale2 / m_scale;
1139 else if( GErel(x[m_k], m_loK + m_scale * m_loJ) && m_loJ > - infinity )
1143 x[m_k] -= m_scale * x[m_j];
1145 else if( GErel(x[m_k], m_loK + m_scale * m_loJ) && m_loK > - infinity )
1150 x[m_j] = z1 * scale1 / m_scale;
1152 else if( LTrel(x[m_k], m_loK + m_scale * m_loJ) )
1157 x[m_k] -= m_scale * x[m_j];
1166 assert(m_scale < 0.0);
1168 if( GErel(x[m_k], m_upK + m_scale * m_loJ) )
1173 x[m_k] -= m_scale * x[m_j];
1175 else if( GErel(x[m_k], m_loK + m_scale * m_loJ) && m_loJ > - infinity )
1179 x[m_k] -= m_scale * x[m_j];
1181 else if( GErel(x[m_k], m_upK + m_scale * m_upJ) && m_upK < infinity )
1186 x[m_j] = z2 * scale2 / m_scale;
1188 else if( GErel(x[m_k], m_loK + m_scale * m_upJ) && m_upJ < infinity )
1192 x[m_k] -= m_scale * x[m_j];
1194 else if( GErel(x[m_k], m_loK + m_scale * m_upJ) && m_loK > - infinity )
1199 x[m_j] = z1 * scale1 / m_scale;
1201 else if( LTrel(x[m_k], m_loK + m_scale * m_upJ) )
1206 x[m_k] -= m_scale * x[m_j];
1216 r[m_j] = m_scale * r[m_k];
1221 METHOD( "SPxMainSM::handleExtremes" );
1235 for( int i = lp. nRows()-1; i >= 0; --i)
1245 else if (lhs > - infinity && lhs < -maxVal)
1250 else if (lhs < infinity && lhs > maxVal)
1264 else if (rhs > - infinity && rhs < -maxVal)
1269 else if (rhs < infinity && rhs > maxVal)
1286 for( int j = 0; j < lp. nCols(); ++j)
1296 else if (lo > - infinity && lo < -maxVal)
1301 else if (lo < infinity && lo > maxVal)
1315 else if (up > - infinity && up < -maxVal)
1320 else if (up < infinity && up > maxVal)
1332 Real absBnd = (lo > up) ? lo : up;
1341 while(i < col. size())
1354 << " removed, absBnd=" << absBnd
1376 else if (obj > - infinity && obj < -maxVal)
1381 else if (obj < infinity && obj > maxVal)
1388 if (remRows + remNzos + chgLRhs + chgBnds + objCnt > 0)
1396 << remRows << " rows, "
1397 << remNzos << " non-zeros, "
1398 << chgBnds << " col bounds, "
1399 << chgLRhs << " row bounds, "
1400 << objCnt << " objective coefficients" << std::endl; )
1407 METHOD( "SPxMainSM::removeEmpty" );
1414 for( int i = lp. nRows()-1; i >= 0; --i)
1418 if (row. size() == 0)
1426 << " rhs=" << lp. rhs(i) << std::endl; )
1440 for( int j = lp. nCols()-1; j >= 0; --j)
1444 if (col. size() == 0)
1447 << ": empty -> maxObj=" << lp. maxObj(j)
1448 << " lower=" << lp. lower(j)
1449 << " upper=" << lp. upper(j); )
1494 if (remRows + remCols > 0)
1499 MSG_INFO2( spxout << "IMAISM10 Main simplifier (empty rows/colums) removed "
1500 << remRows << " rows, "
1501 << remCols << " cols"
1510 assert(row. size() == 1);
1513 int j = row. index(0);
1518 << ": singleton -> val=" << aij
1519 << " lhs=" << lp. lhs(i)
1520 << " rhs=" << lp. rhs(i); )
1546 << " (" << lp. lower(j)
1548 << " (" << lp. upper(j)
1549 << ")" << std::endl; )
1551 bool stricterUp = false;
1552 bool stricterLo = false;
1581 METHOD( "SPxMainSM::simplifyRows" );
1599 for( int i = lp. nRows()-1; i >= 0; --i)
1609 for( int k = 0; k < row. size(); ++k)
1612 int j = row. index(k);
1621 lhsBnd += aij * lp. lower(j);
1626 rhsBnd += aij * lp. upper(j);
1633 rhsBnd += aij * lp. lower(j);
1638 lhsBnd += aij * lp. upper(j);
1644 if (rhsCnt <= 1 || lhsCnt <= 1)
1646 for( int k = 0; k < row. size(); ++k)
1649 int j = row. index(k);
1665 Real z = (lp. lhs(i) / scale) - (rhsBnd / scale);
1673 lo = lp. upper(j) + z * scale / aij;
1675 lo = z * scale / aij;
1683 << ": redundant lower bound on x" << j
1684 << " -> lower=" << lo
1685 << " (" << lp. lower(j)
1686 << ")" << std::endl; )
1689 lhsBnd -= aij * lp. lower(j);
1705 Real z = (lp. rhs(i) / scale) - (lhsBnd / scale);
1713 up = lp. lower(j) + z * scale / aij;
1715 up = z * scale / aij;
1723 << ": redundant upper bound on x" << j
1724 << " -> upper=" << up
1725 << " (" << lp. upper(j)
1726 << ")" << std::endl; )
1729 rhsBnd -= aij * lp. upper(j);
1748 Real z = (lp. lhs(i) / scale) - (rhsBnd / scale);
1756 up = lp. lower(j) + z * scale / aij;
1758 up = z * scale / aij;
1766 << ": redundant upper bound on x" << j
1767 << " -> upper=" << up
1768 << " (" << lp. upper(j)
1769 << ")" << std::endl; )
1772 lhsBnd -= aij * lp. upper(j);
1788 Real z = (lp. rhs(i) / scale) - (lhsBnd / scale);
1796 lo = lp. upper(j) + z * scale / aij;
1798 lo = z * scale / aij;
1806 << ": redundant lower bound on x" << j
1807 << " -> lower=" << lo
1808 << " (" << lp. lower(j)
1809 << ")" << std::endl; )
1812 rhsBnd -= aij * lp. lower(j);
1828 << ": redundant lhs -> lhsBnd=" << lhsBnd
1829 << " lhs=" << lp. lhs(i)
1838 << ": redundant rhs -> rhsBnd=" << rhsBnd
1839 << " rhs=" << lp. rhs(i)
1853 << ": infeasible -> lhs=" << lp. lhs(i)
1854 << " rhs=" << lp. rhs(i)
1855 << " lhsBnd=" << lhsBnd
1856 << " rhsBnd=" << rhsBnd
1866 << ": unconstrained -> removed" << std::endl; )
1871 remNzos += row. size();
1880 #if EMPTY_CONSTRAINT
1882 if (row. size() == 0)
1890 << " rhs=" << lp. rhs(i) << std::endl; )
1908 if (row. size() == 1)
1915 #if FORCE_CONSTRAINT
1921 << ": forcing constraint fix on lhs ->"
1922 << " lhs=" << lp. lhs(i)
1923 << " rhsBnd=" << rhsBnd
1930 for( int k = 0; k < row. size(); ++k)
1933 int j = row. index(k);
1937 lowers[k] = lp. lower(j);
1938 uppers[k] = lp. upper(j);
1951 remNzos += row. size();
1962 << ": forcing constraint fix on rhs ->"
1963 << " rhs=" << lp. rhs(i)
1964 << " lhsBnd=" << lhsBnd
1971 for( int k = 0; k < row. size(); ++k)
1974 int j = row. index(k);
1978 lowers[k] = lp. lower(j);
1979 uppers[k] = lp. upper(j);
1992 remNzos += row. size();
2002 assert(remRows > 0 || remNzos == 0);
2004 if (remRows + chgLRhs + chgBnds > 0)
2013 << remRows << " rows, "
2014 << remNzos << " non-zeros, "
2015 << chgBnds << " col bounds, "
2016 << chgLRhs << " row bounds"
2025 METHOD( "SPxMainSM::simplifyCols" );
2044 for( int j = lp. nCols()-1; j >= 0; --j)
2052 << ": infeasible bounds on x" << j
2053 << " -> lower=" << lp. lower(j)
2054 << " upper=" << lp. upper(j)
2060 if (col. size() == 0)
2064 << ": empty -> maxObj=" << lp. maxObj(j)
2065 << " lower=" << lp. lower(j)
2066 << " upper=" << lp. upper(j); )
2120 for( int k = 0; k < col. size(); ++k)
2122 if (!loFree && !upFree)
2125 int i = col. index(k);
2130 if (col. value(k) > 0.0)
2138 else if (col. value(k) < 0.0)
2156 << " unconstrained above ->"; )
2175 << " unconstrained below ->"; )
2191 #if FREE_ZERO_OBJ_VARIABLE
2192 bool unconstrained_below = loFree && lp. lower(j) <= - infinity;
2193 bool unconstrained_above = upFree && lp. upper(j) >= infinity;
2195 if (unconstrained_below || unconstrained_above)
2199 << " unconstrained "
2200 << (unconstrained_below ? "below" : "above")
2201 << " with zero objective (" << lp. maxObj(j)
2202 << ")" << std::endl; )
2208 sorter_qsort(col_idx_sorted.mem()+1, col_idx_sorted.size(), compare);
2214 remRows += col. size();
2215 for( int k = col_idx_sorted.size()-1; k >= 0; --k)
2222 for( int k = 0; k < col. size(); ++k)
2224 int l = col. index(k);
2243 << " fixed -> lower=" << lp. lower(j)
2244 << " upper=" << lp. upper(j) << std::endl; )
2249 remNzos += col. size();
2259 if (col. size() == 1)
2262 int i = col. index(0);
2267 #if ZERO_OBJ_COL_SINGLETON
2269 << ": singleton in row " << i
2270 << " with zero objective"; )
2278 lhs = lp. lhs(i) - aij * lp. upper(j);
2280 rhs = lp. rhs(i) - aij * lp. lower(j);
2285 lhs = lp. lhs(i) - aij * lp. lower(j);
2287 rhs = lp. rhs(i) - aij * lp. upper(j);
2301 << " (" << lp. lhs(i)
2303 << " (" << lp. rhs(i)
2304 << ")" << std::endl; )
2335 #if DOUBLETON_EQUATION
2337 << ": singleton in row " << i
2338 << " with doubleton equation ->"; )
2347 if (row. index(0) == j)
2352 else if (row. index(1) == j)
2374 Real z1 = (lhs / scale1) - (aij * lp. upper(j) / scale1);
2375 Real z2 = (lhs / scale2) - (aij * lp. lower(j) / scale2);
2402 << ": lower=" << lp. lower(k)
2404 << ") upper=" << lp. upper(k)
2406 << ")" << std::endl; )
2432 #if FREE_COL_SINGLETON
2439 << ": free singleton in inequality constraint" << std::endl; )
2484 << ": free singleton removed" << std::endl; )
2488 for ( int h = 0; h < row.size(); ++h)
2490 int k = row. index(h);
2494 Real new_obj = lp. obj(k) - (lp. obj(j) * row.value(h) / aij);
2501 remNzos += row.size();
2511 if (remCols + remRows > 0)
2520 << remRows << " rows, "
2521 << remCols << " cols, "
2522 << remNzos << " non-zeros, "
2523 << chgBnds << " col bounds"
2531 METHOD( "SPxMainSM::simplifyDual" );
2551 for( int i = lp. nRows()-1; i >= 0; --i)
2557 << ": unconstrained" << std::endl; )
2576 for( int j = 0; j < lp. nCols(); ++j)
2590 dualVarUp[i] = bound;
2592 dualVarLo[i] = bound;
2597 dualVarLo[i] = bound;
2599 dualVarUp[i] = bound;
2605 for( int j = 0; j < lp. nCols(); ++j)
2607 dualConsLo[j] = dualConsUp[j] = 0.0;
2611 for( int k = 0; k < col. size(); ++k)
2617 int i = col. index(k);
2626 dualConsLo[j] += aij * dualVarLo[i];
2631 dualConsUp[j] += aij * dualVarUp[i];
2638 dualConsUp[j] += aij * dualVarLo[i];
2643 dualConsLo[j] += aij * dualVarUp[i];
2648 for( int j = lp. nCols()-1; j >= 0; --j)
2654 if ( LTrel(dualConsUp[j], dualConsLo[j], opttol()))
2657 << ": dual infeasible -> dual lhs bound=" << dualConsLo[j]
2658 << " dual rhs bound=" << dualConsUp[j] << std::endl; )
2668 #if DOMINATED_COLUMN
2670 << ": dominated -> maxObj=" << obj
2671 << " dual rhs bound=" << dualConsUp[j] << std::endl; )
2689 #if DOMINATED_COLUMN
2691 << ": dominated -> maxObj=" << obj
2692 << " dual lhs bound=" << dualConsLo[j] << std::endl; )
2712 #if WEAKLY_DOMINATED_COLUMN
2714 << ": weakly dominated -> maxObj=" << obj
2715 << " dual rhs bound=" << dualConsUp[j] << std::endl; )
2725 #if WEAKLY_DOMINATED_COLUMN
2727 << ": weakly dominated -> maxObj=" << obj
2728 << " dual lhs bound=" << dualConsLo[j] << std::endl; )
2752 assert(remRows > 0 || remCols > 0 || remNzos == 0);
2754 if (remCols + remRows > 0)
2762 << remRows << " rows, "
2763 << remCols << " cols, "
2764 << remNzos << " non-zeros"
2772 METHOD( "SPxMainSM::duplicateRows" );
2790 for ( int i = 0; i < lp. nRows(); ++i)
2794 if (row. size() == 1)
2803 MSG_INFO2( spxout << "IMAISM79 Main simplifier duplicate rows (row singleton stage) removed "
2804 << rs_remRows << " rows, "
2805 << rs_remRows << " non-zeros"
2825 classSize[0] = lp. nRows();
2827 for( int i = 1; i < lp. nRows(); ++i)
2840 for( int j = 0; j < lp. nCols(); ++j)
2844 for( int k = 0; k < col. size(); ++k)
2847 int i = col. index(k);
2851 if (scale[i] == 0.0)
2854 classSet[pClass[i]].add(i, aij / scale[i]);
2855 if (--classSize[pClass[i]] == 0)
2856 idxSet.addIdx(pClass[i]);
2860 for( int m = 0; m < col. size(); ++m)
2862 int k = pClass[col. index(m)];
2864 if (classSet[k].size() > 0)
2869 if (classSet[k].size() > 1)
2870 sorter_qsort(classSet[k].mem()+1, classSet[k].size(), compare);
2873 int classIdx = idxSet.index(0);
2876 for( int l = 0; l < classSet[k].size(); ++l)
2878 if (l != 0 && NErel(classSet[k].value(l), oldVal, epsZero()))
2880 classIdx = idxSet.index(0);
2884 pClass[classSet[k].index(l)] = classIdx;
2885 ++classSize[classIdx];
2887 oldVal = classSet[k].value(l);
2890 classSet[k]. clear();
2895 catch(std::bad_alloc& x)
2911 for( int k = 0; k < dupRows.size(); ++k)
2914 dupRows[pClass[k]].add(k, 0.0);
2917 const int nRowsOld_tmp = lp. nRows();
2918 int* perm_tmp = new int[nRowsOld_tmp];
2920 for( int j = 0; j < nRowsOld_tmp; ++j)
2925 int idxFirstDupRows = -1;
2926 int idxLastDupRows = -1;
2929 for( int k = 0; k < dupRows.size(); ++k)
2931 if (dupRows[k].size() > 1 && !(lp. rowVector(dupRows[k].index(0)). size() == 1))
2935 if(idxFirstDupRows < 0)
2937 idxFirstDupRows = k;
2940 for( int l = 1; l < dupRows[k].size(); ++l)
2942 int i = dupRows[k].index(l);
2946 numDelRows += (dupRows[k].size()-1);
2951 int k_tmp, j_tmp = -1;
2952 for (k_tmp = j_tmp = 0; k_tmp < nRowsOld_tmp; ++k_tmp)
2954 if (perm_tmp[k_tmp] >= 0)
2955 perm_tmp[k_tmp] = j_tmp++;
2960 bool doChangeRanges = false;
2964 for( int k = 0; k < dupRows.size(); ++k)
2966 if (dupRows[k].size() > 1 && !(lp. rowVector(dupRows[k].index(0)). size() == 1))
2969 << " duplicate rows found" << std::endl; )
2983 for( int l = 0; l < dupRows[k].size(); ++l)
2985 int i = dupRows[k].index(l);
2993 maxLhs = lp. lhs(rowIdx);
2994 minRhs = lp. rhs(rowIdx);
2998 Real scaledLhs, scaledRhs;
2999 Real factor = scale[rowIdx] / scale[i];
3011 if (scaledLhs > maxLhs)
3016 if (scaledRhs < minRhs)
3028 Real newLhs = (maxLhs > lp. lhs(rowIdx)) ? maxLhs : lp. lhs(rowIdx);
3029 Real newRhs = (minRhs < lp. rhs(rowIdx)) ? minRhs : lp. rhs(rowIdx);
3031 if(k == idxLastDupRows)
3034 for( int j = 0; j < nRowsOld_tmp; ++j)
3036 da_perm[j] = perm_tmp[j];
3038 m_hist.append( new DuplicateRowsPS(lp, rowIdx, maxLhsIdx, minRhsIdx, dupRows[k], scale, da_perm, isLhsEqualRhs, true, EQrel(newLhs, newRhs), k==idxFirstDupRows));
3044 m_hist.append( new DuplicateRowsPS(lp, rowIdx, maxLhsIdx, minRhsIdx, dupRows[k], scale, da_perm_empty, isLhsEqualRhs, false, EQrel(newLhs, newRhs), k == idxFirstDupRows));
3047 if (maxLhs > lp. lhs(rowIdx) || minRhs < lp. rhs(rowIdx))
3050 doChangeRanges = true;
3054 MSG_INFO3( spxout << "IMAISM55 duplicate rows yield infeasible bounds:"
3055 << " lhs=" << newLhs
3056 << " rhs=" << newRhs << std::endl; )
3061 if( newRhs < newLhs )
3064 newLhsVec[rowIdx] = newLhs;
3065 newRhsVec[rowIdx] = newRhs;
3078 const int nRowsOld = lp. nRows();
3079 int* perm = new int[nRowsOld];
3081 for( int i = 0; i < nRowsOld; ++i)
3094 for( int i = 0; i < nRowsOld; ++i)
3097 assert(perm[i] == perm_tmp[i]);
3107 if (remRows + remNzos > 0)
3113 MSG_INFO2( spxout << "IMAISM56 Main simplifier (duplicate rows) removed "
3114 << remRows << " rows, "
3115 << remNzos << " non-zeros"
3123 METHOD( "SPxMainSM::duplicateCols" );
3151 classSize[0] = lp. nCols();
3153 for( int j = 1; j < lp. nCols(); ++j)
3167 for( int i = 0; i < lp. nRows(); ++i)
3171 for( int k = 0; k < row. size(); ++k)
3174 int j = row. index(k);
3178 if (scale[j] == 0.0)
3181 classSet[pClass[j]].add(j, aij / scale[j]);
3182 if (--classSize[pClass[j]] == 0)
3183 idxSet.addIdx(pClass[j]);
3187 for( int m = 0; m < row. size(); ++m)
3189 int k = pClass[row. index(m)];
3191 if (classSet[k].size() > 0)
3196 if (classSet[k].size() > 1)
3197 sorter_qsort(classSet[k].mem()+1, classSet[k].size(), compare);
3200 int classIdx = idxSet.index(0);
3203 for( int l = 0; l < classSet[k].size(); ++l)
3205 if (l != 0 && NErel(classSet[k].value(l), oldVal, epsZero()))
3208 classIdx = idxSet.index(0);
3212 pClass[classSet[k].index(l)] = classIdx;
3213 ++classSize[classIdx];
3215 oldVal = classSet[k].value(l);
3218 classSet[k]. clear();
3223 catch(std::bad_alloc& x)
3239 for( int k = 0; k < dupCols.size(); ++k)
3240 dupCols[pClass[k]].add(k, 0.0);
3245 for( int j = 0; j < lp. nCols(); ++j)
3248 fixAndRemCol[j] = false;
3251 bool hasDuplicateCol = false;
3254 for( int k = 0; k < dupCols.size(); ++k)
3256 if (dupCols[k].size() > 1 && !(lp. colVector(dupCols[k].index(0)). size() == 1))
3259 << " duplicate columns found" << std::endl; )
3261 if (!hasDuplicateCol)
3264 hasDuplicateCol = true;
3267 for( int l = 0; l < dupCols[k].size(); ++l)
3269 for( int m = 0; m < dupCols[k].size(); ++m)
3271 int j1 = dupCols[k].index(l);
3272 int j2 = dupCols[k].index(m);
3274 if (l != m && !remCol[j1] && !remCol[j2])
3280 Real factor = scale[j1] / scale[j2];
3281 Real objDif = cj1 - cj2 * scale[j1] / scale[j2];
3310 else if (factor < 0)
3325 << " replaced by one" << std::endl; )
3333 MSG_INFO3( spxout << "IMAISM80 not removing two duplicate columns " << j1
3335 << " because zero not contained in their bounds" << std::endl; )
3344 if (factor > 0 && objDif > 0)
3355 << " first one fixed at upper bound=" << lp. upper(j1) << std::endl; )
3360 else if (factor < 0 && objDif < 0)
3371 << " first one fixed at lower bound=" << lp. lower(j1) << std::endl; )
3380 if (factor < 0 && objDif > 0)
3391 << " first one fixed at upper bound=" << lp. upper(j1) << std::endl; )
3398 else if (factor > 0 && objDif < 0)
3409 << " first one fixed at lower bound=" << lp. lower(j1) << std::endl; )
3418 fixAndRemCol[j1] = true;
3429 for( int j = 0; j < lp. nCols(); ++j)
3441 const int nColsOld = lp. nCols();
3442 int* perm = new int[nColsOld];
3444 for( int j = 0; j < nColsOld; ++j)
3457 for( int j = 0; j < nColsOld; ++j)
3464 for( int j = 0; j < nColsOld; ++j)
3466 da_perm[j] = perm[j];
3469 if (hasDuplicateCol)
3476 assert(remCols > 0 || remNzos == 0);
3484 MSG_INFO2( spxout << "IMAISM65 Main simplifier (duplicate columns) removed "
3485 << remCols << " cols, "
3486 << remNzos << " non-zeros"
3494 METHOD( "SPxMainSM::fixColumn" );
3504 << ": lower=" << lp. lower(j)
3505 << " upper=" << lp. upper(j)
3510 for( int k = 0; k < col. size(); ++k)
3512 int i = col. index(k);
3522 Real rhs = (lp. rhs(i) / scale) - (y / scale);
3531 << " (" << lp. rhs(i)
3532 << ") aij=" << col. value(k)
3545 Real lhs = (lp. lhs(i) / scale) - (y / scale);
3554 << " (" << lp. lhs(i)
3555 << ") aij=" << col. value(k)
3560 assert(lp. lhs(i) <= lp. rhs(i));
3569 METHOD( "SPxMainSM::simplify()" );
3598 for( int k = 0; k < m_hist.size(); ++k)
3622 for( int i = 0; i < lp. nRows(); ++i)
3625 for( int j = 0; j < lp. nCols(); ++j)
3639 while(again && ret == OKAY)
3682 << lp. nRows() << " rows "
3683 << lp. nCols() << " columns "
3684 << lp. nNzos() << " nonzeros"
3689 MSG_INFO1( spxout << "IMAISM70 Main simplifier removed all rows and columns" << std::endl; )
3721 assert(x. dim() == r. dim());
3722 assert(y. dim() == s. dim());
3724 METHOD( "SPxMainSM::unsimplify()" );
3729 for( int j = 0; j < x. dim(); ++j)
3735 for( int i = 0; i < y. dim(); ++i)
3743 for( int k = m_hist.size()-1; k >= 0; --k)
|