module Numeric.LinearProgramming(
simplex,
exact,
sparseOfGeneral,
Optimization(..),
Constraints(..),
Bounds,
Bound(..),
(#),
Solution(..)
) where
import Numeric.LinearAlgebra.HMatrix
import Numeric.LinearAlgebra.Devel hiding (Dense)
import Foreign(Ptr)
import System.IO.Unsafe(unsafePerformIO)
import Foreign.C.Types
import Data.List((\\),sortBy,nub)
import Data.Function(on)
import qualified Data.Map.Strict as Map
(#) :: Double -> Int -> (Double,Int)
infixl 5 #
(#) = (,)
data Bound x = x :<=: Double
| x :>=: Double
| x :&: (Double,Double)
| x :==: Double
| Free x
deriving Show
data Solution = Undefined
| Feasible (Double, [Double])
| Infeasible (Double, [Double])
| NoFeasible
| Optimal (Double, [Double])
| Unbounded
deriving Show
data Constraints = Dense [ Bound [Double] ]
| Sparse [ Bound [(Double,Int)] ]
| General [ Bound [(Double,Int)] ]
data Optimization = Maximize [Double]
| Minimize [Double]
type Bounds = [Bound Int]
sparseOfGeneral :: Constraints -> Constraints
sparseOfGeneral (General cs) =
Sparse $ map (\bl ->
let cl = obj bl in
let cl_unique = foldr (\(c,t) m -> Map.insertWith (+) t c m) Map.empty cl in
withObj bl (Map.foldrWithKey' (\t c l -> (c#t) : l) [] cl_unique)) cs
sparseOfGeneral _ = error "sparseOfGeneral: a general system of constraints was expected"
simplex :: Optimization -> Constraints -> Bounds -> Solution
simplex opt (Dense []) bnds = simplex opt (Sparse []) bnds
simplex opt (Sparse []) bnds = simplex opt (Sparse [Free [0#1]]) bnds
simplex opt (General []) bnds = simplex opt (Sparse [Free [0#1]]) bnds
simplex opt (Dense constr) bnds = extract sg sol where
sol = simplexSparse m n (mkConstrD sz objfun constr) (mkBounds sz constr bnds)
n = length objfun
m = length constr
(sz, sg, objfun) = adapt opt
simplex opt (Sparse constr) bnds = extract sg sol where
sol = simplexSparse m n (mkConstrS sz objfun constr) (mkBounds sz constr bnds)
n = length objfun
m = length constr
(sz, sg, objfun) = adapt opt
simplex opt constr@(General _) bnds = simplex opt (sparseOfGeneral constr) bnds
exact :: Optimization -> Constraints -> Bounds -> Solution
exact opt (Dense []) bnds = exact opt (Sparse []) bnds
exact opt (Sparse []) bnds = exact opt (Sparse [Free [0#1]]) bnds
exact opt (General []) bnds = exact opt (Sparse [Free [0#1]]) bnds
exact opt (Dense constr) bnds = extract sg sol where
sol = exactSparse m n (mkConstrD sz objfun constr) (mkBounds sz constr bnds)
n = length objfun
m = length constr
(sz, sg, objfun) = adapt opt
exact opt (Sparse constr) bnds = extract sg sol where
sol = exactSparse m n (mkConstrS sz objfun constr) (mkBounds sz constr bnds)
n = length objfun
m = length constr
(sz, sg, objfun) = adapt opt
exact opt constr@(General _) bnds = exact opt (sparseOfGeneral constr) bnds
adapt :: Optimization -> (Int, Double, [Double])
adapt opt = case opt of
Maximize x -> (sz x, 1 ,x)
Minimize x -> (sz x, 1, (map negate x))
where
sz x | null x = error "simplex: objective function with zero variables"
| otherwise = length x
extract :: Double -> Vector Double -> Solution
extract sg sol = r where
z = sg * (sol!1)
v = toList $ subVector 2 (size sol 2) sol
r = case round(sol!0)::Int of
1 -> Undefined
2 -> Feasible (z,v)
3 -> Infeasible (z,v)
4 -> NoFeasible
5 -> Optimal (z,v)
6 -> Unbounded
_ -> error "simplex: solution type unknown"
obj :: Bound t -> t
obj (x :<=: _) = x
obj (x :>=: _) = x
obj (x :&: _) = x
obj (x :==: _) = x
obj (Free x) = x
withObj :: Bound t -> t -> Bound t
withObj (_ :<=: b) x = (x :<=: b)
withObj (_ :>=: b) x = (x :>=: b)
withObj (_ :&: b) x = (x :&: b)
withObj (_ :==: b) x = (x :==: b)
withObj (Free _) x = Free x
tb :: Bound t -> Double
tb (_ :<=: _) = glpUP
tb (_ :>=: _) = glpLO
tb (_ :&: _) = glpDB
tb (_ :==: _) = glpFX
tb (Free _) = glpFR
lb :: Bound t -> Double
lb (_ :<=: _) = 0
lb (_ :>=: a) = a
lb (_ :&: (a,_)) = a
lb (_ :==: a) = a
lb (Free _) = 0
ub :: Bound t -> Double
ub (_ :<=: a) = a
ub (_ :>=: _) = 0
ub (_ :&: (_,a)) = a
ub (_ :==: a) = a
ub (Free _) = 0
mkBound1 :: Bound t -> [Double]
mkBound1 b = [tb b, lb b, ub b]
mkBound2 :: Bound t -> (t, [Double])
mkBound2 b = (obj b, mkBound1 b)
mkBounds :: Int -> [Bound [a]] -> [Bound Int] -> Matrix Double
mkBounds n b1 b2 = fromLists (cb++vb) where
gv' = map obj b2
gv | nub gv' == gv' = gv'
| otherwise = error $ "simplex: duplicate bounds for vars " ++ show (gv'\\nub gv')
rv | null gv || minimum gv >= 0 && maximum gv <= n = [1..n] \\ gv
| otherwise = error $ "simplex: bounds: variables "++show gv++" not in 1.."++show n
vb = map snd $ sortBy (compare `on` fst) $ map (mkBound2 . (:>=: 0)) rv ++ map mkBound2 b2
cb = map mkBound1 b1
mkConstrD :: Int -> [Double] -> [Bound [Double]] -> Matrix Double
mkConstrD n f b1 | ok = fromLists (ob ++ co)
| otherwise = error $ "simplex: dense constraints require "++show n
++" variables, given " ++ show ls
where
cs = map obj b1
ls = map length cs
ok = all (==n) ls
den = fromLists cs
ob = map (([0,0]++).return) f
co = [[fromIntegral i, fromIntegral j,den `atIndex` (i1,j1)]| i<-[1 ..rows den], j<-[1 .. cols den]]
mkConstrS :: Int -> [Double] -> [Bound [(Double, Int)]] -> Matrix Double
mkConstrS n objfun b1 = fromLists (ob ++ co) where
ob = map (([0,0]++).return) objfun
co = concat $ zipWith f [1::Int ..] cs
cs = map obj b1
f k = map (g k)
g k (c,v) | v >=1 && v<= n = [fromIntegral k, fromIntegral v,c]
| otherwise = error $ "simplex: sparse constraints: variable "++show v++" not in 1.."++show n
(##) :: TransArray c => TransRaw c b -> c -> b
infixl 1 ##
a ## b = applyRaw a b
foreign import ccall unsafe "c_simplex_sparse" c_simplex_sparse
:: CInt -> CInt
-> CInt -> CInt -> Ptr Double
-> CInt -> CInt -> Ptr Double
-> CInt -> Ptr Double
-> IO CInt
simplexSparse :: Int -> Int -> Matrix Double -> Matrix Double -> Vector Double
simplexSparse m n c b = unsafePerformIO $ do
s <- createVector (2+n)
c_simplex_sparse (fi m) (fi n) ## (cmat c) ## (cmat b) ## s #|"c_simplex_sparse"
return s
foreign import ccall unsafe "c_exact_sparse" c_exact_sparse
:: CInt -> CInt
-> CInt -> CInt -> Ptr Double
-> CInt -> CInt -> Ptr Double
-> CInt -> Ptr Double
-> IO CInt
exactSparse :: Int -> Int -> Matrix Double -> Matrix Double -> Vector Double
exactSparse m n c b = unsafePerformIO $ do
s <- createVector (2+n)
c_exact_sparse (fi m) (fi n) ## (cmat c) ## (cmat b) ## s #|"c_exact_sparse"
return s
glpFR, glpLO, glpUP, glpDB, glpFX :: Double
glpFR = 0
glpLO = 1
glpUP = 2
glpDB = 3
glpFX = 4