This is lpkit.c in view mode; [Download] [Up]
#include "lpkit.h"
#include "lpglob.h"
#include <stdlib.h>
#include <stdarg.h>
#include <stdio.h>
#include <string.h>
/* Globals */
lprec *Lp=NULL; /* pointer to active problem */
int Rows;
int Columns;
int Sum;
int Non_zeros;
int Level;
matrec *Mat;
int *Col_no;
int *Col_end;
int *Row_end;
REAL *Orig_rh;
REAL *Rh;
REAL *Rhs;
short *Must_be_int;
REAL *Orig_upbo;
REAL *Orig_lowbo;
REAL *Upbo;
REAL *Lowbo;
int *Bas;
short *Basis;
short *Lower;
int Eta_alloc;
int Eta_size;
REAL *Eta_value;
int *Eta_row_nr;
int *Eta_col_end;
int Num_inv;
REAL *Solution;
REAL *Best_solution;
REAL Infinite;
REAL Epsilon;
REAL Epsb;
REAL Epsd;
REAL Epsel;
REAL TREJ;
REAL TINV;
short Maximise;
short Floor_first;
REAL Extrad;
int Warn_count; /* used in CHECK version of rounding macro */
void error(char *format, ...)
{
va_list ap;
va_start(ap, format);
vfprintf(stderr, format, ap);
fputc('\n', stderr);
va_end(ap);
exit(FAIL);
}
lprec *make_lp(int rows, int columns)
{
lprec *newlp;
int i, sum;
sum=rows+columns;
if(rows < 0 || columns < 0)
error("rows < 0 or columns < 0");
CALLOC(newlp, 1, lprec);
strcpy(newlp->lp_name, "unnamed");
newlp->active=FALSE;
newlp->verbose=FALSE;
newlp->print_duals=FALSE;
newlp->print_sol=FALSE;
newlp->debug=FALSE;
newlp->print_at_invert=FALSE;
newlp->trace=FALSE;
newlp->rows=rows;
newlp->columns=columns;
newlp->sum=sum;
newlp->rows_alloc=rows;
newlp->columns_alloc=columns;
newlp->sum_alloc=sum;
newlp->names_used=FALSE;
newlp->obj_bound=DEF_INFINITE;
newlp->infinite=DEF_INFINITE;
newlp->epsilon=DEF_EPSILON;
newlp->epsb=DEF_EPSB;
newlp->epsd=DEF_EPSD;
newlp->epsel=DEF_EPSEL;
newlp->non_zeros=0;
newlp->mat_alloc=1;
CALLOC(newlp->mat, newlp->mat_alloc, matrec);
CALLOC(newlp->col_no, newlp->mat_alloc, int);
CALLOC(newlp->col_end, columns + 1, int);
CALLOC(newlp->row_end, rows + 1, int);
newlp->row_end_valid=FALSE;
CALLOC(newlp->orig_rh, rows + 1, REAL);
CALLOC(newlp->rh, rows + 1, REAL);
CALLOC(newlp->rhs, rows + 1, REAL);
CALLOC(newlp->must_be_int, sum + 1, short);
for(i = 0; i <= sum; i++)
newlp->must_be_int[i]=FALSE;
CALLOC(newlp->orig_upbo, sum + 1, REAL);
for(i = 0; i <= sum; i++)
newlp->orig_upbo[i]=newlp->infinite;
CALLOC(newlp->upbo, sum + 1, REAL);
CALLOC(newlp->orig_lowbo, sum + 1, REAL);
CALLOC(newlp->lowbo, sum + 1, REAL);
newlp->basis_valid=TRUE;
CALLOC(newlp->bas, rows+1, int);
CALLOC(newlp->basis, sum + 1, short);
CALLOC(newlp->lower, sum + 1, short);
for(i = 0; i <= rows; i++)
{
newlp->bas[i]=i;
newlp->basis[i]=TRUE;
}
for(i = rows + 1; i <= sum; i++)
newlp->basis[i]=FALSE;
for(i = 0 ; i <= sum; i++)
newlp->lower[i]=TRUE;
newlp->eta_valid=TRUE;
newlp->eta_size=0;
newlp->eta_alloc=10000;
newlp->max_num_inv=DEFNUMINV;
newlp->nr_lagrange=0;
CALLOC(newlp->eta_value, newlp->eta_alloc, REAL);
CALLOC(newlp->eta_row_nr, newlp->eta_alloc, int);
CALLOC(newlp->eta_col_end, newlp->rows_alloc + newlp->max_num_inv, int);
newlp->bb_rule=FIRST_NI;
newlp->break_at_int=FALSE;
newlp->break_value=0;
newlp->iter=0;
newlp->total_iter=0;
CALLOC(newlp->solution, sum + 1, REAL);
CALLOC(newlp->best_solution, sum + 1, REAL);
CALLOC(newlp->duals, rows + 1, REAL);
newlp->maximise = FALSE;
newlp->floor_first = TRUE;
newlp->scaling_used = FALSE;
newlp->columns_scaled = FALSE;
CALLOC(newlp->ch_sign, rows + 1, short);
for(i = 0; i <= rows; i++)
newlp->ch_sign[i] = FALSE;
newlp->valid = FALSE;
return(newlp);
}
void delete_lp(lprec *lp)
{
int i;
if(lp->active)
Lp=NULL;
if(lp->names_used)
{
free(lp->row_name);
free(lp->col_name);
}
free(lp->mat);
free(lp->col_no);
free(lp->col_end);
free(lp->row_end);
free(lp->orig_rh);
free(lp->rh);
free(lp->rhs);
free(lp->must_be_int);
free(lp->orig_upbo);
free(lp->orig_lowbo);
free(lp->upbo);
free(lp->lowbo);
free(lp->bas);
free(lp->basis);
free(lp->lower);
free(lp->eta_value);
free(lp->eta_row_nr);
free(lp->eta_col_end);
free(lp->solution);
free(lp->best_solution);
free(lp->duals);
free(lp->ch_sign);
if(lp->scaling_used)
free(lp->scale);
if(lp->nr_lagrange>0)
{
free(lp->lag_rhs);
free(lp->lambda);
free(lp->lag_con_type);
for(i=0; i < lp->nr_lagrange; i++)
free(lp->lag_row[i]);
free(lp->lag_row);
}
free(lp);
lp=NULL;
}
lprec *copy_lp(lprec *lp)
{
lprec *newlp;
int i, rowsplus, colsplus, sumplus;
rowsplus=lp->rows_alloc+1;
colsplus=lp->columns_alloc+1;
sumplus=lp->sum_alloc+1;
MALLOCCPY(newlp, lp, 1, lprec); /* copy all non pointers */
newlp->active=FALSE;
if(newlp->names_used)
{
MALLOCCPY(newlp->col_name, lp->col_name, colsplus, nstring);
MALLOCCPY(newlp->row_name, lp->row_name, rowsplus, nstring);
}
MALLOCCPY(newlp->mat, lp->mat, newlp->mat_alloc, matrec);
MALLOCCPY(newlp->col_end, lp->col_end, colsplus, int);
MALLOCCPY(newlp->col_no, lp->col_no, newlp->mat_alloc, int);
MALLOCCPY(newlp->row_end, lp->row_end, rowsplus, int);
MALLOCCPY(newlp->orig_rh, lp->orig_rh, rowsplus, REAL);
MALLOCCPY(newlp->rh, lp->rh, rowsplus, REAL);
MALLOCCPY(newlp->rhs, lp->rhs, rowsplus, REAL);
MALLOCCPY(newlp->must_be_int, lp->must_be_int, sumplus, short);
MALLOCCPY(newlp->orig_upbo, lp->orig_upbo, sumplus, REAL);
MALLOCCPY(newlp->orig_lowbo, lp->orig_lowbo, sumplus, REAL);
MALLOCCPY(newlp->upbo, lp->upbo, sumplus, REAL);
MALLOCCPY(newlp->lowbo, lp->lowbo, sumplus, REAL);
MALLOCCPY(newlp->bas, lp->bas, rowsplus, int);
MALLOCCPY(newlp->basis, lp->basis, sumplus, short);
MALLOCCPY(newlp->lower, lp->lower, sumplus, short);
MALLOCCPY(newlp->eta_value, lp->eta_value, lp->eta_alloc, REAL);
MALLOCCPY(newlp->eta_row_nr, lp->eta_row_nr, lp->eta_alloc, int);
MALLOCCPY(newlp->eta_col_end, lp->eta_col_end,
lp->rows_alloc + lp->max_num_inv, int);
MALLOCCPY(newlp->solution, lp->solution, sumplus, REAL);
MALLOCCPY(newlp->best_solution, lp->best_solution, sumplus, REAL);
MALLOCCPY(newlp->duals, lp->duals, rowsplus, REAL);
MALLOCCPY(newlp->ch_sign, lp->ch_sign, rowsplus, short);
if(newlp->scaling_used)
MALLOCCPY(newlp->scale, lp->scale, sumplus, REAL);
if(newlp->nr_lagrange > 0)
{
MALLOCCPY(newlp->lag_rhs, lp->lag_rhs, newlp->nr_lagrange, REAL);
MALLOCCPY(newlp->lambda, lp->lambda, newlp->nr_lagrange, REAL);
MALLOCCPY(newlp->lag_con_type, lp->lag_con_type, newlp->nr_lagrange,
short);
MALLOC(newlp->lag_row, newlp->nr_lagrange, REAL*);
for(i = 0; i < newlp->nr_lagrange; i++)
MALLOCCPY(newlp->lag_row[i], lp->lag_row[i], colsplus, REAL);
}
return(newlp);
}
void inc_mat_space(lprec *lp, int maxextra)
{
if(lp->non_zeros + maxextra >= lp->mat_alloc)
{
lp->mat_alloc = lp->non_zeros + maxextra;
REALLOC(lp->mat, lp->mat_alloc, matrec);
REALLOC(lp->col_no, lp->mat_alloc, int);
if (lp->active)
{
Mat=lp->mat;
Col_no=lp->col_no;
}
}
}
void inc_row_space(lprec *lp)
{
if(lp->rows > lp->rows_alloc)
{
lp->rows_alloc=lp->rows+10;
lp->sum_alloc=lp->rows_alloc+lp->columns_alloc;
REALLOC(lp->orig_rh, lp->rows_alloc + 1, REAL);
REALLOC(lp->rh, lp->rows_alloc + 1, REAL);
REALLOC(lp->rhs, lp->rows_alloc + 1, REAL);
REALLOC(lp->orig_upbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->upbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->orig_lowbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->lowbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->solution, lp->sum_alloc + 1, REAL);
REALLOC(lp->best_solution, lp->sum_alloc + 1, REAL);
REALLOC(lp->row_end, lp->rows_alloc + 1, int);
REALLOC(lp->basis, lp->sum_alloc + 1, short);
REALLOC(lp->lower, lp->sum_alloc + 1, short);
REALLOC(lp->must_be_int, lp->sum_alloc + 1, short);
REALLOC(lp->bas, lp->rows_alloc + 1, int);
REALLOC(lp->duals, lp->rows_alloc + 1, REAL);
REALLOC(lp->ch_sign, lp->rows_alloc + 1, short);
REALLOC(lp->eta_col_end, lp->rows_alloc + lp->max_num_inv, int);
if(lp->names_used)
REALLOC(lp->row_name, lp->rows_alloc + 1, nstring);
if(lp->scaling_used)
REALLOC(lp->scale, lp->sum_alloc + 1, REAL);
if(lp->active)
set_globals(lp);
}
}
void inc_col_space(lprec *lp)
{
if(lp->columns >= lp->columns_alloc)
{
lp->columns_alloc=lp->columns+10;
lp->sum_alloc=lp->rows_alloc+lp->columns_alloc;
REALLOC(lp->must_be_int, lp->sum_alloc + 1, short);
REALLOC(lp->orig_upbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->upbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->orig_lowbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->lowbo, lp->sum_alloc + 1, REAL);
REALLOC(lp->solution, lp->sum_alloc + 1, REAL);
REALLOC(lp->best_solution, lp->sum_alloc + 1, REAL);
REALLOC(lp->basis, lp->sum_alloc + 1, short);
REALLOC(lp->lower, lp->sum_alloc + 1, short);
if(lp->names_used)
REALLOC(lp->col_name, lp->columns_alloc + 1, nstring);
if(lp->scaling_used)
REALLOC(lp->scale, lp->sum_alloc + 1, REAL);
REALLOC(lp->col_end, lp->columns_alloc + 1, int);
if(lp->active)
set_globals(lp);
}
}
void set_mat(lprec *lp, int Row, int Column, REAL Value)
{
int elmnr, lastelm, i;
if(Row > lp->rows || Row < 0)
error("Row out of range");
if(Column > lp->columns || Column < 1)
error("Column out of range");
if(lp->scaling_used)
Value *= lp->scale[Row] * lp->scale[lp->rows + Column];
if(TRUE /*abs(Value) > lp->epsilon*/)
{
if (lp->basis[Column] == TRUE && Row > 0)
lp->basis_valid = FALSE;
lp->eta_valid = FALSE;
elmnr = lp->col_end[Column-1];
while((elmnr < lp->col_end[Column]) ?
(lp->mat[elmnr].row_nr != Row) : FALSE)
elmnr++;
if((elmnr != lp->col_end[Column]) ?
(lp->mat[elmnr].row_nr == Row) : FALSE )
if (lp->scaling_used)
{
if(lp->ch_sign[Row])
lp->mat[elmnr].value =
-Value * lp->scale[Row] * lp->scale[Column];
else
lp->mat[elmnr].value =
Value * lp->scale[Row] * lp->scale[Column];
}
else
{
if(lp->ch_sign[Row])
lp->mat[elmnr].value = -Value;
else
lp->mat[elmnr].value = Value;
}
else
{
/* check if more space is needed for matrix */
inc_mat_space(lp,1);
/* Shift the matrix */
lastelm=lp->non_zeros;
for(i = lastelm; i > elmnr ; i--)
lp->mat[i]=lp->mat[i-1];
for(i = Column; i <= lp->columns; i++)
lp->col_end[i]++;
/* Set new element */
lp->mat[elmnr].row_nr=Row;
if (lp->scaling_used)
{
if(lp->ch_sign[Row])
lp->mat[elmnr].value=-Value*lp->scale[Row]*lp->scale[Column];
else
lp->mat[elmnr].value=Value*lp->scale[Row]*lp->scale[Column];
}
else
{
if(lp->ch_sign[Row])
lp->mat[elmnr].value=-Value;
else
lp->mat[elmnr].value=Value;
}
lp->row_end_valid=FALSE;
lp->non_zeros++;
if (lp->active)
Non_zeros=lp->non_zeros;
}
}
}
void set_obj_fn(lprec *lp, REAL *row)
{
int i;
for(i = 1; i <= lp->columns; i++)
set_mat(lp, 0, i, row[i]);
}
void str_set_obj_fn(lprec *lp, char *row)
{
int i;
REAL *arow;
char *p, *newp;
CALLOC(arow, lp->columns + 1, REAL);
p = row;
for(i = 1; i <= lp->columns; i++)
{
arow[i] = (REAL) strtod(p, &newp);
if(p==newp)
error("Bad string in str_set_obj_fn");
else
p=newp;
}
set_obj_fn(lp, arow);
free(arow);
}
void add_constraint(lprec *lp, REAL *row, short constr_type, REAL rh)
{
matrec *newmat;
int i, j;
int elmnr;
int stcol;
int *addtoo;
MALLOC(addtoo, lp->columns + 1, int)
for(i = 1; i <= lp->columns; i++)
if(row[i]!=0)
{
addtoo[i]=TRUE;
lp->non_zeros++;
}
else
addtoo[i]=FALSE;
MALLOC(newmat, lp->non_zeros, matrec);
inc_mat_space(lp, 0);
lp->rows++;
lp->sum++;
inc_row_space(lp);
if(lp->scaling_used)
{
/* shift scale */
for(i=lp->sum; i > lp->rows; i--)
lp->scale[i]=lp->scale[i-1];
lp->scale[lp->rows]=1;
}
if(lp->names_used)
sprintf(lp->row_name[lp->rows], "r_%d", lp->rows);
if(lp->scaling_used && lp->columns_scaled)
for(i = 1; i <= lp->columns; i++)
row[i] *= lp->scale[lp->rows+i];
if(constr_type==GE)
lp->ch_sign[lp->rows] = TRUE;
else
lp->ch_sign[lp->rows] = FALSE;
elmnr = 0;
stcol = 0;
for(i = 1; i <= lp->columns; i++)
{
for(j = stcol; j < lp->col_end[i]; j++)
{
newmat[elmnr].row_nr=lp->mat[j].row_nr;
newmat[elmnr].value=lp->mat[j].value;
elmnr++;
}
if(addtoo[i])
{
if(lp->ch_sign[lp->rows])
newmat[elmnr].value = -row[i];
else
newmat[elmnr].value = row[i];
newmat[elmnr].row_nr = lp->rows;
elmnr++;
}
stcol=lp->col_end[i];
lp->col_end[i]=elmnr;
}
memcpy(lp->mat, newmat, lp->non_zeros*sizeof(matrec));
free(newmat);
free(addtoo);
for(i=lp->sum ; i > lp->rows; i--)
{
lp->orig_upbo[i]=lp->orig_upbo[i-1];
lp->orig_lowbo[i]=lp->orig_lowbo[i-1];
lp->basis[i]=lp->basis[i-1];
lp->lower[i]=lp->lower[i-1];
lp->must_be_int[i]=lp->must_be_int[i-1];
}
for(i= 1 ; i <= lp->rows; i++)
if(lp->bas[i] >= lp->rows)
lp->bas[i]++;
if(constr_type==LE || constr_type==GE)
{
lp->orig_upbo[lp->rows]=lp->infinite;
}
else if(constr_type==EQ)
{
lp->orig_upbo[lp->rows]=0;
}
else
{
fprintf(stderr, "Wrong constraint type\n");
exit(FAIL);
}
lp->orig_lowbo[lp->rows]=0;
if(constr_type==GE && rh != 0)
lp->orig_rh[lp->rows]=-rh;
else
lp->orig_rh[lp->rows]=rh;
lp->row_end_valid=FALSE;
lp->bas[lp->rows]=lp->rows;
lp->basis[lp->rows]=TRUE;
lp->lower[lp->rows]=TRUE;
if(lp->active)
set_globals(lp);
lp->eta_valid=FALSE;
}
void str_add_constraint(lprec *lp,
char *row_string,
short constr_type,
REAL rh)
{
int i;
REAL *aRow;
char *p, *newp;
CALLOC(aRow, lp->columns + 1, REAL);
p = row_string;
for(i = 1; i <= lp->columns; i++)
{
aRow[i] = (REAL) strtod(p, &newp);
if(p==newp)
error("Bad string in str_add_constr");
else
p=newp;
}
add_constraint(lp, aRow, constr_type, rh);
free(aRow);
}
void del_constraint(lprec *lp, int del_row)
{
int i, j;
unsigned elmnr;
int startcol;
if(del_row<1 || del_row>lp->rows)
{
fprintf(stderr, "There is no constraint nr. %d\n", del_row);
exit(FAIL);
}
elmnr=0;
startcol=0;
for(i = 1; i <= lp->columns; i++)
{
for(j=startcol; j < lp->col_end[i]; j++)
{
if(lp->mat[j].row_nr!=del_row)
{
lp->mat[elmnr]=lp->mat[j];
if(lp->mat[elmnr].row_nr > del_row)
lp->mat[elmnr].row_nr--;
elmnr++;
}
else
lp->non_zeros--;
}
startcol=lp->col_end[i];
lp->col_end[i]=elmnr;
}
for(i = del_row; i < lp->rows; i++)
{
lp->orig_rh[i] = lp->orig_rh[i+1];
lp->ch_sign[i] = lp->ch_sign[i+1];
lp->bas[i] = lp->bas[i+1];
if(lp->names_used)
strcpy(lp->row_name[i], lp->row_name[i+1]);
}
for(i = 1; i < lp->rows; i++)
if(lp->bas[i] > del_row)
lp->bas[i]--;
for(i=del_row; i < lp->sum; i++)
{
lp->lower[i]=lp->lower[i+1];
lp->basis[i]=lp->basis[i+1];
lp->orig_upbo[i]=lp->orig_upbo[i+1];
lp->orig_lowbo[i]=lp->orig_lowbo[i+1];
lp->must_be_int[i]=lp->must_be_int[i+1];
if(lp->scaling_used)
lp->scale[i]=lp->scale[i+1];
}
lp->rows--;
lp->sum--;
lp->row_end_valid=FALSE;
if(lp->active)
set_globals(lp);
lp->eta_valid=FALSE;
lp->basis_valid=FALSE;
}
void add_lag_con(lprec *lp, REAL *row, short con_type, REAL rhs)
{
int i;
REAL sign;
if(con_type == LE || con_type == EQ)
sign = 1;
else if(con_type == GE)
sign = -1;
else
error("con_type not implemented\n");
lp->nr_lagrange++;
if(lp->nr_lagrange==1)
{
CALLOC(lp->lag_row, lp->nr_lagrange, REAL*);
CALLOC(lp->lag_rhs, lp->nr_lagrange, REAL);
CALLOC(lp->lambda, lp->nr_lagrange, REAL);
CALLOC(lp->lag_con_type, lp->nr_lagrange, short);
}
else
{
REALLOC(lp->lag_row, lp->nr_lagrange, REAL*);
REALLOC(lp->lag_rhs, lp->nr_lagrange, REAL);
REALLOC(lp->lambda, lp->nr_lagrange, REAL);
REALLOC(lp->lag_con_type, lp->nr_lagrange, short);
}
CALLOC(lp->lag_row[lp->nr_lagrange-1], lp->columns+1, REAL);
lp->lag_rhs[lp->nr_lagrange-1]=rhs * sign;
for( i=1; i <= lp->columns; i++)
lp->lag_row[lp->nr_lagrange-1][i]=row[i] * sign;
lp->lambda[lp->nr_lagrange-1]=0;
lp->lag_con_type[lp->nr_lagrange-1]=(con_type == EQ);
}
void str_add_lag_con(lprec *lp, char *row, short con_type, REAL rhs)
{
int i;
REAL *a_row;
char *p, *new_p;
CALLOC(a_row, lp->columns + 1, REAL);
p = row;
for(i = 1; i <= lp->columns; i++)
{
a_row[i] = (REAL) strtod(p, &new_p);
if(p==new_p)
error("Bad string in str_add_lag_con");
else
p=new_p;
}
add_lag_con(lp, a_row, con_type, rhs);
free(a_row);
}
void add_column(lprec *lp, REAL *column)
{
int i, elmnr;
lp->columns++;
lp->sum++;
inc_col_space(lp);
inc_mat_space(lp, lp->rows+1);
if(lp->scaling_used)
{
for(i = 0; i <= lp->rows; i++)
column[i]*=lp->scale[i];
lp->scale[lp->sum]=1;
}
elmnr=lp->col_end[lp->columns-1];
for(i = 0 ; i <= lp->rows ; i++)
if(column[i] != 0)
{
lp->mat[elmnr].row_nr=i;
if(lp->ch_sign[i])
lp->mat[elmnr].value=-column[i];
else
lp->mat[elmnr].value=column[i];
lp->non_zeros++;
elmnr++;
}
lp->col_end[lp->columns]=elmnr;
lp->orig_lowbo[lp->sum]=0;
lp->orig_upbo[lp->sum]=lp->infinite;
lp->lower[lp->sum]=TRUE;
lp->basis[lp->sum]=FALSE;
lp->must_be_int[lp->sum]=FALSE;
if(lp->names_used)
sprintf(lp->col_name[lp->columns], "var_%d", lp->columns);
lp->row_end_valid=FALSE;
if(lp->active)
{
Sum=lp->sum;
Columns=lp->columns;
Non_zeros=lp->non_zeros;
}
}
void str_add_column(lprec *lp, char *col_string)
{
int i;
REAL *aCol;
char *p, *newp;
CALLOC(aCol, lp->rows + 1, REAL);
p = col_string;
for(i = 0; i <= lp->rows; i++)
{
aCol[i] = (REAL) strtod(p, &newp);
if(p==newp)
error("Bad string in str_add_column");
else
p=newp;
}
add_column(lp, aCol);
free(aCol);
}
void del_column(lprec *lp, int column)
{
int i, j, from_elm, to_elm, elm_in_col;
if(column > lp->columns || column < 1)
error("Column out of range in del_column");
for(i = 1; i <= lp->rows; i++)
{
if(lp->bas[i]==lp->rows+column)
lp->basis_valid=FALSE;
else if(lp->bas[i] > lp->rows+column)
lp->bas[i]--;
}
for(i = lp->rows+column; i < lp->sum; i++)
{
if(lp->names_used)
strcpy(lp->col_name[i-lp->rows], lp->col_name[i-lp->rows+1]);
lp->must_be_int[i]=lp->must_be_int[i+1];
lp->orig_upbo[i]=lp->orig_upbo[i+1];
lp->orig_lowbo[i]=lp->orig_lowbo[i+1];
lp->upbo[i]=lp->upbo[i+1];
lp->lowbo[i]=lp->lowbo[i+1];
lp->basis[i]=lp->basis[i+1];
lp->lower[i]=lp->lower[i+1];
if(lp->scaling_used)
lp->scale[i]=lp->scale[i+1];
}
for(i = 0; i < lp->nr_lagrange; i++)
for(j = column; j <= lp->columns; j++)
lp->lag_row[i][j]=lp->lag_row[i][j+1];
to_elm=lp->col_end[column-1];
from_elm=lp->col_end[column];
elm_in_col=from_elm-to_elm;
for(i = from_elm; i < lp->non_zeros; i++)
{
lp->mat[to_elm]=lp->mat[i];
to_elm++;
}
for(i = column; i < lp->columns; i++)
lp->col_end[i]=lp->col_end[i+1]-elm_in_col;
lp->non_zeros -= elm_in_col;
lp->row_end_valid=FALSE;
lp->eta_valid=FALSE;
lp->sum--;
lp->columns--;
if(lp->active)
set_globals(lp);
}
void set_upbo(lprec *lp, int column, REAL value)
{
if(column > lp->columns || column < 1)
error("Column out of range");
if(lp->scaling_used)
value /= lp->scale[lp->rows + column];
if(value < lp->orig_lowbo[lp->rows + column])
error("Upperbound must be >= lowerbound");
lp->eta_valid = FALSE;
lp->orig_upbo[lp->rows+column] = value;
}
void set_lowbo(lprec *lp, int column, REAL value)
{
if(column > lp->columns || column < 1)
error("Column out of range");
if(lp->scaling_used)
value /= lp->scale[lp->rows + column];
if(value > lp->orig_upbo[lp->rows + column])
error("Upperbound must be >= lowerbound");
lp->eta_valid = FALSE;
lp->orig_lowbo[lp->rows+column] = value;
}
void set_int(lprec *lp, int column, short must_be_int)
{
if(column > lp->columns || column < 1)
error("Column out of range");
lp->must_be_int[lp->rows+column]=must_be_int;
if(must_be_int==TRUE)
if(lp->columns_scaled)
unscale_columns(lp);
}
void set_rh(lprec *lp, int row, REAL value)
{
if(row > lp->rows || row < 0)
error("Row out of Range");
if(row == 0) /* setting of RHS of OF not meaningful */
{
fprintf(stderr,
"Warning: attempt to set RHS of objective function, ignored\n");
return;
}
if(lp->scaling_used)
if(lp->ch_sign[row])
lp->orig_rh[row] = -value * lp->scale[row];
else
lp->orig_rh[row] = value * lp->scale[row];
else
if(lp->ch_sign[row])
lp->orig_rh[row] = -value;
else
lp->orig_rh[row] = value;
lp->eta_valid = FALSE;
}
void set_rh_vec(lprec *lp, REAL *rh)
{
int i;
if(lp->scaling_used)
for(i = 1; i <= lp->rows; i++)
if(lp->ch_sign[i])
lp->orig_rh[i]=-rh[i]*lp->scale[i];
else
lp->orig_rh[i]=rh[i]*lp->scale[i];
else
for(i=1; i <= lp->rows; i++)
if(lp->ch_sign[i])
lp->orig_rh[i]=-rh[i];
else
lp->orig_rh[i]=rh[i];
lp->eta_valid=FALSE;
}
void str_set_rh_vec(lprec *lp, char *rh_string)
{
int i;
REAL *newrh;
char *p, *newp;
CALLOC(newrh, lp->rows + 1, REAL);
p = rh_string;
for(i = 1; i <= lp->rows; i++)
{
newrh[i] = (REAL) strtod(p, &newp);
if(p==newp)
error("Bad string in str_set_rh_vec");
else
p=newp;
}
set_rh_vec(lp, newrh);
free(newrh);
}
void set_maxim(lprec *lp)
{
int i;
if(lp->maximise==FALSE)
{
for(i = 0; i < lp->non_zeros; i++)
if(lp->mat[i].row_nr==0)
lp->mat[i].value*=-1;
lp->eta_valid=FALSE;
}
lp->maximise=TRUE;
lp->ch_sign[0]=TRUE;
if(lp->active)
Maximise=TRUE;
}
void set_minim(lprec *lp)
{
int i;
if(lp->maximise==TRUE)
{
for(i = 0; i < lp->non_zeros; i++)
if(lp->mat[i].row_nr==0)
lp->mat[i].value = -lp->mat[i].value;
lp->eta_valid=FALSE;
}
lp->maximise=FALSE;
lp->ch_sign[0]=FALSE;
if(lp->active)
Maximise=FALSE;
}
void set_constr_type(lprec *lp, int row, short con_type)
{
int i;
if(row > lp->rows || row < 1)
error("Row out of Range");
if(con_type==EQ)
{
lp->orig_upbo[row]=0;
lp->basis_valid=FALSE;
if(lp->ch_sign[row])
{
for(i = 0; i < lp->non_zeros; i++)
if(lp->mat[i].row_nr==row)
lp->mat[i].value*=-1;
lp->eta_valid=FALSE;
lp->ch_sign[row]=FALSE;
if(lp->orig_rh[row]!=0)
lp->orig_rh[row]*=-1;
}
}
else if(con_type==LE)
{
lp->orig_upbo[row]=lp->infinite;
lp->basis_valid=FALSE;
if(lp->ch_sign[row])
{
for(i = 0; i < lp->non_zeros; i++)
if(lp->mat[i].row_nr==row)
lp->mat[i].value*=-1;
lp->eta_valid=FALSE;
lp->ch_sign[row]=FALSE;
if(lp->orig_rh[row]!=0)
lp->orig_rh[row]*=-1;
}
}
else if(con_type==GE)
{
lp->orig_upbo[row]=lp->infinite;
lp->basis_valid=FALSE;
if(!lp->ch_sign[row])
{
for(i = 0; i < lp->non_zeros; i++)
if(lp->mat[i].row_nr==row)
lp->mat[i].value*=-1;
lp->eta_valid=FALSE;
lp->ch_sign[row]=TRUE;
if(lp->orig_rh[row]!=0)
lp->orig_rh[row]*=-1;
}
}
else
error("Constraint type not (yet) implemented");
}
REAL mat_elm(lprec *lp, int row, int column)
{
REAL value;
int elmnr;
if(row < 0 || row > lp->rows)
error("Row out of range in mat_elm");
if(column < 1 || column > lp->columns)
error("Column out of range in mat_elm");
value=0;
elmnr=lp->col_end[column-1];
while(lp->mat[elmnr].row_nr != row && elmnr < lp->col_end[column])
elmnr++;
if(elmnr != lp->col_end[column])
{
value = lp->mat[elmnr].value;
if(lp->ch_sign[row])
value = -value;
if(lp->scaling_used)
value /= lp->scale[row] * lp->scale[lp->rows + column];
}
return(value);
}
void get_row(lprec *lp, int row_nr, REAL *row)
{
int i, j;
if(row_nr <0 || row_nr > lp->rows)
error("Row nr. out of range in get_row");
for(i = 1; i <= lp->columns; i++)
{
row[i]=0;
for(j=lp->col_end[i-1]; j < lp->col_end[i]; j++)
if(lp->mat[j].row_nr==row_nr)
row[i]=lp->mat[j].value;
if(lp->scaling_used)
row[i] /= lp->scale[lp->rows+i] * lp->scale[row_nr];
}
if(lp->ch_sign[row_nr])
for(i=0; i <= lp->columns; i++)
if(row[i]!=0)
row[i] = -row[i];
}
void get_column(lprec *lp, int col_nr, REAL *column)
{
int i;
if(col_nr < 1 || col_nr > lp->columns)
error("Col. nr. out of range in get_column");
for(i = 0; i <= lp->rows; i++)
column[i]=0;
for(i = lp->col_end[col_nr-1]; i < lp->col_end[col_nr]; i++)
column[lp->mat[i].row_nr]=lp->mat[i].value;
for(i = 0; i <= lp->rows; i++)
if(column[i] !=0)
{
if(lp->ch_sign[i])
column[i]*=-1;
if(lp->scaling_used)
column[i]/=(lp->scale[i] * lp->scale[lp->rows+col_nr]);
}
}
void get_reduced_costs(lprec *lp, REAL *rc)
{
int varnr, i, j;
REAL f;
if(!lp->basis_valid)
error("Not a valid basis in get_reduced_costs");
set_globals(lp);
if(!lp->eta_valid)
invert();
for(i = 1; i <= lp->sum; i++)
rc[i] = 0;
rc[0] = 1;
btran(rc);
for(i = 1; i <= lp->columns; i++)
{
varnr = lp->rows + i;
if(!Basis[varnr])
if(Upbo[varnr] > 0)
{
f = 0;
for(j = Col_end[i - 1]; j < Col_end[i]; j++)
f += rc[Mat[j].row_nr] * Mat[j].value;
rc[varnr] = f;
}
}
for(i = 1; i <= Sum; i++)
my_round(rc[i], Epsd);
}
short is_feasible(lprec *lp, REAL *values)
{
int i, elmnr;
REAL *this_rhs;
REAL dist;
if(lp->scaling_used)
{
for(i = lp->rows+1; i <= lp->sum; i++)
if( values[i - lp->rows] < lp->orig_lowbo[i]*lp->scale[i]
|| values[i - lp->rows] > lp->orig_upbo[i]*lp->scale[i])
return(FALSE);
}
else
{
for(i = lp->rows+1; i <= lp->sum; i++)
if( values[i - lp->rows] < lp->orig_lowbo[i]
|| values[i - lp->rows] > lp->orig_upbo[i])
return(FALSE);
}
CALLOC(this_rhs, lp->rows+1, REAL)
for(i = 1; i <= lp->columns; i++)
for(elmnr = lp->col_end[i - 1]; elmnr < lp->col_end[i]; elmnr++)
this_rhs[lp->mat[elmnr].row_nr] += lp->mat[elmnr].value * values[i];
for(i = 1; i <= lp->rows; i++)
{
dist = lp->orig_rh[i] - this_rhs[i];
my_round(dist, 0.001)
if((lp->orig_upbo[i] == 0 && dist != 0) || dist < 0)
{
free(this_rhs);
return(FALSE);
}
}
free(this_rhs);
return(TRUE);
}
short column_in_lp(lprec *lp, REAL *testcolumn)
{
int i, j;
short ident;
REAL value;
if(lp->scaling_used)
for(i = 1; i <= lp->columns; i++)
{
ident = TRUE;
j = lp->col_end[i-1];
while(ident && (j < lp->col_end[i]))
{
value = lp->mat[j].value;
if(lp->ch_sign[lp->mat[j].row_nr])
value = -value;
value /= lp->scale[lp->rows+i];
value /= lp->scale[lp->mat[j].row_nr];
value -= testcolumn[lp->mat[j].row_nr];
if(my_abs(value) > 0.001) /* should be some epsilon? MB */
ident=FALSE;
j++;
}
if(ident)
return(TRUE);
}
else
for(i = 1; i <= lp->columns; i++)
{
ident = TRUE;
j = lp->col_end[i-1];
while(ident && j < lp->col_end[i])
{
value = lp->mat[j].value;
if(lp->ch_sign[lp->mat[j].row_nr])
value *= -1;
value -= testcolumn[lp->mat[j].row_nr];
if( my_abs(value) > 0.001 )
ident=FALSE;
j++;
}
if(ident)
return(TRUE);
}
return(FALSE);
}
void print_lp(lprec *lp)
{
int i, j;
REAL *fatmat;
CALLOC(fatmat, (lp->rows + 1) * lp->columns, REAL);
for(i = 1; i <= lp->columns; i++)
for(j = lp->col_end[i-1]; j < lp->col_end[i]; j++)
fatmat[(i - 1) * (lp->rows + 1) + lp->mat[j].row_nr]=lp->mat[j].value;
printf("problem name: %s\n", lp->lp_name);
printf(" ");
for(j = 1; j <= lp->columns; j++)
if(lp->names_used)
printf("%8s ", lp->col_name[j]);
else
printf("Var[%3d] ", j);
if(lp->maximise)
{
printf("\nMaximise ");
for(j = 0; j < lp->columns; j++)
printf("% 8.2f ",-fatmat[j*(lp->rows+1)]);
}
else
{
printf("\nMinimize ");
for(j = 0; j < lp->columns; j++)
printf("% 8.2f ", fatmat[j*(lp->rows+1)]);
}
printf("\n");
for(i = 1; i <= lp->rows; i++)
{
if(lp->names_used)
printf("%9s ", lp->row_name[i]);
else
printf("Row[%3d] ", i);
for(j = 0; j < lp->columns; j++)
if(lp->ch_sign[i] && fatmat[j*(lp->rows+1)+i] != 0)
printf("% 8.2f ",-fatmat[j*(lp->rows+1)+i]);
else
printf("% 8.2f ", fatmat[j*(lp->rows+1)+i]);
if(lp->orig_upbo[i]==lp->infinite)
if(lp->ch_sign[i])
printf(">= ");
else
printf("<= ");
else
printf(" = ");
if(lp->ch_sign[i])
printf("% 8.2f\n",-lp->orig_rh[i]);
else
printf("% 8.2f\n", lp->orig_rh[i]);
}
printf("Type ");
for(i = 1; i <= lp->columns; i++)
if(lp->must_be_int[lp->rows+i]==TRUE)
printf(" Int ");
else
printf(" Real ");
printf("\nupbo ");
for(i = 1; i <= lp->columns; i++)
if(lp->orig_upbo[lp->rows+i]==lp->infinite)
printf(" Inf ");
else
printf("% 8.2f ", lp->orig_upbo[lp->rows+i]);
printf("\nlowbo ");
for(i = 1; i <= lp->columns; i++)
printf("% 8.2f ", lp->orig_lowbo[lp->rows+i]);
printf("\n");
for(i = 0; i < lp->nr_lagrange; i++)
{
printf("lag[%3d] ", i);
for(j = 1; j <= lp->columns; j++)
printf("% 8.2f ", lp->lag_row[i][j]);
if(lp->orig_upbo[i]==lp->infinite)
if(lp->lag_con_type[i] == GE)
printf(">= ");
else if(lp->lag_con_type[i] == LE)
printf("<= ");
else if(lp->lag_con_type[i] == EQ)
printf(" = ");
printf("% 8.2f\n", lp->lag_rhs[i]);
}
free(fatmat);
}
void set_row_name(lprec *lp, int row, nstring new_name)
{
int i;
if(!lp->names_used)
{
CALLOC(lp->row_name, lp->rows_alloc + 1, nstring);
CALLOC(lp->col_name, lp->columns_alloc + 1, nstring);
lp->names_used=TRUE;
for(i = 0; i <= lp->rows; i++)
sprintf(lp->row_name[i], "r_%d", i);
for(i = 1; i <= lp->columns; i++)
sprintf(lp->col_name[i], "var_%d", i);
}
strcpy(lp->row_name[row], new_name);
}
void set_col_name(lprec *lp, int column, nstring new_name)
{
int i;
if(!lp->names_used)
{
CALLOC(lp->row_name, lp->rows_alloc + 1, nstring);
CALLOC(lp->col_name, lp->columns_alloc + 1, nstring);
lp->names_used=TRUE;
for(i = 0; i <= lp->rows; i++)
sprintf(lp->row_name[i], "r_%d", i);
for(i = 1; i <= lp->columns; i++)
sprintf(lp->col_name[i], "var_%d", i);
}
strcpy(lp->col_name[column], new_name);
}
static REAL minmax_to_scale(REAL min, REAL max)
{
REAL scale;
/* should do something sensible when min or max is 0, MB */
if((min == 0) || (max == 0))
return((REAL)1);
scale = 1 / pow(10, (log10(min) + log10(max)) / 2);
return(scale);
}
void unscale_columns(lprec *lp)
{
int i, j;
/* unscale mat */
for(j = 1; j <= lp->columns; j++)
for(i = lp->col_end[j - 1]; i < lp->col_end[j]; i++)
lp->mat[i].value /= lp->scale[lp->rows + j];
/* unscale bounds as well */
for(i = lp->rows + 1; i < lp->sum; i++)
{
if(lp->orig_lowbo[i] != 0)
lp->orig_lowbo[i] *= lp->scale[i];
if(lp->orig_upbo[i] != lp->infinite)
lp->orig_upbo[i] *= lp->scale[i];
}
for(i=lp->rows+1; i<= lp->sum; i++)
lp->scale[i]=1;
lp->columns_scaled=FALSE;
lp->eta_valid=FALSE;
}
void unscale(lprec *lp)
{
int i, j;
if(lp->scaling_used)
{
/* unscale mat */
for(j = 1; j <= lp->columns; j++)
for(i = lp->col_end[j - 1]; i < lp->col_end[j]; i++)
lp->mat[i].value /= lp->scale[lp->rows + j];
/* unscale bounds */
for(i = lp->rows + 1; i < lp->sum; i++)
{
if(lp->orig_lowbo[i] != 0)
lp->orig_lowbo[i] *= lp->scale[i];
if(lp->orig_upbo[i] != lp->infinite)
lp->orig_upbo[i] *= lp->scale[i];
}
/* unscale the matrix */
for(j = 1; j <= lp->columns; j++)
for(i = lp->col_end[j-1]; i < lp->col_end[j]; i++)
lp->mat[i].value /= lp->scale[lp->mat[i].row_nr];
/* unscale the rhs! */
for(i = 0; i <= lp->rows; i++)
lp->orig_rh[i] /= lp->scale[i];
free(lp->scale);
lp->scaling_used=FALSE;
lp->eta_valid=FALSE;
}
}
void auto_scale(lprec *lp)
{
int i, j, row_nr, IntUsed;
REAL *row_max, *row_min, *scalechange, absval;
if(!lp->scaling_used)
{
MALLOC(lp->scale, lp->sum_alloc + 1, REAL);
for(i=0; i <= lp->sum; i++)
lp->scale[i]=1;
}
MALLOC(row_max, lp->rows + 1, REAL);
MALLOC(row_min, lp->rows + 1, REAL);
MALLOC(scalechange, lp->sum + 1, REAL);
/* initialise min and max values */
for(i = 0; i <= lp->rows; i++)
{
row_max[i]=0;
row_min[i]=lp->infinite;
}
/* calculate min and max absolute values of rows */
for(j = 1; j <= lp->columns; j++)
for(i = lp->col_end[j - 1]; i < lp->col_end[j]; i++)
{
row_nr = lp->mat[i].row_nr;
absval = my_abs(lp->mat[i].value);
if(absval!=0)
{
row_max[row_nr] = my_max(row_max[row_nr], absval);
row_min[row_nr] = my_min(row_min[row_nr], absval);
}
}
/* calculate scale factors for rows */
for(i = 0; i <= lp->rows; i++)
{
scalechange[i] = minmax_to_scale(row_min[i], row_max[i]);
lp->scale[i] *= scalechange[i];
}
/* now actually scale the matrix */
for(j = 1; j <= lp->columns; j++)
for(i = lp->col_end[j - 1]; i < lp->col_end[j]; i++)
lp->mat[i].value *= scalechange[lp->mat[i].row_nr];
/* and scale the rhs! */
for(i = 0; i <= lp->rows; i++)
lp->orig_rh[i] *= scalechange[i];
free(row_max);
free(row_min);
IntUsed=FALSE;
i=lp->rows+1;
while(!IntUsed && i <= lp->sum)
{
IntUsed=lp->must_be_int[i];
i++;
}
if(!IntUsed)
{
REAL col_max, col_min;
/* calculate scales */
for(j = 1; j <= lp->columns; j++)
{
col_max = 0;
col_min = lp->infinite;
for(i = lp->col_end[j - 1]; i < lp->col_end[j]; i++)
{
if(lp->mat[i].value!=0)
{
col_max = my_max(col_max, my_abs(lp->mat[i].value));
col_min = my_min(col_min, my_abs(lp->mat[i].value));
}
}
scalechange[lp->rows + j] = minmax_to_scale(col_min, col_max);
lp->scale[lp->rows + j] *= scalechange[lp->rows + j];
}
/* scale mat */
for(j = 1; j <= lp->columns; j++)
for(i = lp->col_end[j - 1]; i < lp->col_end[j]; i++)
lp->mat[i].value *= scalechange[lp->rows + j];
/* scale bounds as well */
for(i = lp->rows + 1; i < lp->sum; i++)
{
if(lp->orig_lowbo[i] != 0)
lp->orig_lowbo[i] /= scalechange[i];
if(lp->orig_upbo[i] != lp->infinite)
lp->orig_upbo[i] /= scalechange[i];
}
lp->columns_scaled=TRUE;
}
free(scalechange);
lp->scaling_used=TRUE;
lp->eta_valid=FALSE;
}
void reset_basis(lprec *lp)
{
lp->basis_valid=FALSE;
}
void print_solution(lprec *lp)
{
int i;
fprintf(stdout, "Value of objective function: %16.10g\n",
(double)lp->best_solution[0]);
/* print normal variables */
for(i = 1; i <= lp->columns; i++)
if(lp->names_used)
fprintf(stdout, "%-10s%16.5g\n", lp->col_name[i],
(double)lp->best_solution[lp->rows+i]);
else
fprintf(stdout, "Var [%4d] %16.5g\n", i,
(double)lp->best_solution[lp->rows+i]);
/* print achieved constraint values */
if(lp->verbose)
{
fprintf(stdout, "\nActual values of the constraints:\n");
for(i = 1; i <= lp->rows; i++)
if(lp->names_used)
fprintf(stdout, "%-10s%16.5g\n", lp->row_name[i],
(double)lp->best_solution[i]);
else
fprintf(stdout, "Row [%4d] %16.5g\n", i,
(double)lp->best_solution[i]);
}
if((lp->verbose || lp->print_duals))
{
if(lp->max_level != 1)
fprintf(stdout,
"These are the duals from the node that gave the optimal solution.\n");
else
fprintf(stdout, "\nDual values:\n");
for(i = 1; i <= lp->rows; i++)
if(lp->names_used)
fprintf(stdout, "%-10s%16.5g\n", lp->row_name[i],
(double)lp->duals[i]);
else
fprintf(stdout, "Row [%4d] %16.5g\n", i, (double)lp->duals[i]);
}
} /* Printsolution */
void write_LP(lprec *lp, FILE *output)
{
int i, j;
REAL *row;
MALLOC(row, lp->columns+1, REAL);
if(lp->maximise)
fprintf(output, "max:");
else
fprintf(output, "min:");
get_row(lp, 0, row);
for(i = 1; i <= lp->columns; i++)
if(row[i] != 0)
{
if(row[i] == -1)
fprintf(output, " -");
else if(row[i] == 1)
fprintf(output, " +");
else
fprintf(output, " %+g ", row[i]);
if(lp->names_used)
fprintf(output, "%s", lp->col_name[i]);
else
fprintf(output, "x%d", i);
}
fprintf(output, ";\n");
for(j = 1; j <= lp->rows; j++)
{
if(lp->names_used)
fprintf(output, "%s:", lp->row_name[j]);
get_row(lp, j, row);
for(i = 1; i <= lp->columns; i++)
if(row[i] != 0)
{
if(row[i] == -1)
fprintf(output, " -");
else if(row[i] == 1)
fprintf(output, " +");
else
fprintf(output, " %+g ", row[i]);
if(lp->names_used)
fprintf(output, "%s", lp->col_name[i]);
else
fprintf(output, "x%d", i);
}
if(lp->orig_upbo[j] == 0)
fprintf(output, " =");
else if(lp->ch_sign[j])
fprintf(output, " >");
else
fprintf(output, " <");
if(lp->ch_sign[j])
fprintf(output, " %g;\n",-lp->orig_rh[j]);
else
fprintf(output, " %g;\n", lp->orig_rh[j]);
}
for(i = lp->rows+1; i <= lp->sum; i++)
{
if(lp->orig_lowbo[i] != 0)
{
if(lp->names_used)
fprintf(output, "%s > %g;\n", lp->col_name[i - lp->rows],
lp->orig_lowbo[i]);
else
fprintf(output, "x%d > %g;\n", i - lp->rows,
lp->orig_lowbo[i]);
}
if(lp->orig_upbo[i] != lp->infinite)
{
if(lp->names_used)
fprintf(output, "%s < %g;\n", lp->col_name[i - lp->rows],
lp->orig_upbo[i]);
else
fprintf(output, "x%d < %g;\n", i - lp->rows, lp->orig_upbo[i]);
}
}
i=1;
while(!lp->must_be_int[lp->rows+i] && i <= lp->columns)
i++;
if(i <= lp->columns)
{
if(lp->names_used)
fprintf(output, "\nint %s", lp->col_name[i]);
else
fprintf(output, "\nint x%d", i);
i++;
for(; i <= lp->columns; i++)
if(lp->must_be_int[lp->rows+i])
if(lp->names_used)
fprintf(output, ",%s", lp->col_name[i]);
else
fprintf(output, ", x%d", i);
fprintf(output, ";\n");
}
free(row);
}
void write_MPS(lprec *lp, FILE *output)
{
int i, j, marker;
REAL *column;
MALLOC(column, lp->rows+1, REAL);
marker=0;
fprintf(output, "NAME %s\n", lp->lp_name);
fprintf(output, "ROWS\n");
for(i = 0; i <= lp->rows; i++)
{
if(i==0)
fprintf(output, " N ");
else
if(lp->orig_upbo[i]==lp->infinite)
if(lp->ch_sign[i])
fprintf(output, " G ");
else
fprintf(output, " L ");
else
fprintf(output, " E ");
if(lp->names_used)
fprintf(output, "%s\n", lp->row_name[i]);
else
fprintf(output, "r_%d\n", i);
}
fprintf(output, "COLUMNS\n");
j = 0;
for(i = 1; i <= lp->columns; i++)
{
if((lp->must_be_int[i+lp->rows]) && (marker % 2)==0)
{
fprintf(output,
" MARK%04d 'MARKER' 'INTORG'\n",
marker);
marker++;
}
if((!lp->must_be_int[i+lp->rows]) && (marker % 2)==1)
{
fprintf(output,
" MARK%04d 'MARKER' 'INTEND'\n",
marker);
marker++;
}
get_column(lp, i, column);
j=0;
if(lp->maximise)
{
if(column[j] != 0)
{
if(lp->names_used)
fprintf(output, " %-8s %-8s %g\n", lp->col_name[i],
lp->row_name[j], -column[j]);
else
fprintf(output, " var_%-4d r_%-6d %g\n", i, j,
-column[j]);
}
}
else
{
if(column[j] != 0)
{
if(lp->names_used)
fprintf(output, " %-8s %-8s %g\n", lp->col_name[i],
lp->row_name[j], column[j]);
else
fprintf(output, " var_%-4d r_%-6d %g\n", i, j,
column[j]);
}
}
for(j=1; j <= lp->rows; j++)
if(column[j] != 0)
{
if(lp->names_used)
fprintf(output, " %-8s %-8s %g\n", lp->col_name[i],
lp->row_name[j], column[j]);
else
fprintf(output, " var_%-4d r_%-6d %g\n", i, j,
column[j]);
}
}
if((marker % 2) ==1)
{
fprintf(output, " MARK%04d 'MARKER' 'INTEND'\n",
marker);
marker++;
}
fprintf(output, "RHS\n");
for(i = 1; i <= lp->rows; i++)
{
if(lp->ch_sign[i])
{
if(lp->names_used)
fprintf(output, " RHS %-8s %g\n", lp->row_name[i],
(double)-lp->orig_rh[i]);
else
fprintf(output, " RHS r_%-6d %g\n", i,
(double)-lp->orig_rh[i]);
}
else
{
if(lp->names_used)
fprintf(output, " RHS %-8s %g\n", lp->row_name[i],
(double)lp->orig_rh[i]);
else
fprintf(output, " RHS r_%-6d %g\n", i,
(double)lp->orig_rh[i]);
}
}
fprintf(output, "BOUNDS\n");
if(lp->names_used)
for(i = lp->rows + 1; i <= lp->sum; i++)
{
if(lp->orig_upbo[i] < lp->infinite)
fprintf(output, " UP BND %-8s %g\n",
lp->col_name[i- lp->rows], (double)lp->orig_upbo[i]);
if(lp->orig_lowbo[i] != 0)
fprintf(output, " LO BND %-8s %g\n",
lp->col_name[i- lp->rows], (double)lp->orig_lowbo[i]);
}
else
for(i = lp->rows + 1; i <= lp->sum; i++)
{
if(lp->orig_upbo[i] < lp->infinite)
fprintf(output, " UP BND var_%-4d %g\n",
i - lp->rows, (double)lp->orig_upbo[i]);
if(lp->orig_lowbo[i] != 0)
fprintf(output, " LO BND var_%-4d %g\n", i - lp->rows,
(double)lp->orig_lowbo[i]);
}
fprintf(output, "ENDATA\n");
free(column);
}
void print_duals(lprec *lp)
{
int i;
for(i = 1; i <= lp->rows; i++)
if(lp->names_used)
fprintf(stdout, "%10s [%3d] % 10.4f\n", lp->row_name[i], i,
lp->duals[i]);
else
fprintf(stdout, "Dual [%3d] % 10.4f\n", i, lp->duals[i]);
}
void print_scales(lprec *lp)
{
int i;
if(lp->scaling_used)
{
for(i = 0; i <= lp->rows; i++)
fprintf(stdout, "Row[%3d] scaled at % 10.6f\n", i, lp->scale[i]);
for(i = 1; i <= lp->columns; i++)
fprintf(stdout, "Column[%3d] scaled at % 10.6f\n", i,
lp->scale[lp->rows + i]);
}
}
These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.