-- It corresponds to model MachRep2 described in document -- Introduction to Discrete-Event Simulation and the SimPy Language -- [http://heather.cs.ucdavis.edu/~matloff/156/PLN/DESimIntro.pdf]. -- SimPy is available on [http://simpy.sourceforge.net/]. -- -- The model description is as follows. -- -- Two machines, but sometimes break down. Up time is exponentially -- distributed with mean 1.0, and repair time is exponentially distributed -- with mean 0.5. In this example, there is only one repairperson, so -- the two machines cannot be repaired simultaneously if they are down -- at the same time. -- -- In addition to finding the long-run proportion of up time as in -- model MachRep1, let’s also find the long-run proportion of the time -- that a given machine does not have immediate access to the repairperson -- when the machine breaks down. Output values should be about 0.6 and 0.67. import System.Random import Control.Monad import Control.Monad.Trans import Simulation.Aivika.Specs import Simulation.Aivika.Simulation import Simulation.Aivika.Dynamics import Simulation.Aivika.Event import Simulation.Aivika.Ref import Simulation.Aivika.QueueStrategy import Simulation.Aivika.Resource import Simulation.Aivika.Process upRate = 1.0 / 1.0 -- reciprocal of mean up time repairRate = 1.0 / 0.5 -- reciprocal of mean repair time specs = Specs { spcStartTime = 0.0, spcStopTime = 1000.0, spcDT = 1.0, spcMethod = RungeKutta4 } exprnd :: Double -> IO Double exprnd lambda = do x <- getStdRandom random return (- log x / lambda) model :: Simulation (Double, Double) model = do -- number of times the machines have broken down nRep <- newRef 0 -- number of breakdowns in which the machine -- started repair service right away nImmedRep <- newRef 0 -- total up time for all machines totalUpTime <- newRef 0.0 repairPerson <- newResource FCFS 1 pid1 <- newProcessId pid2 <- newProcessId let machine :: Process () machine = do startUpTime <- liftDynamics time upTime <- liftIO $ exprnd upRate holdProcess upTime finishUpTime <- liftDynamics time liftEvent $ modifyRef totalUpTime (+ (finishUpTime - startUpTime)) -- check the resource availability liftEvent $ do modifyRef nRep (+ 1) n <- resourceCount repairPerson when (n == 1) $ modifyRef nImmedRep (+ 1) requestResource repairPerson repairTime <- liftIO $ exprnd repairRate holdProcess repairTime releaseResource repairPerson machine runProcessInStartTime IncludingCurrentEvents pid1 machine runProcessInStartTime IncludingCurrentEvents pid2 machine runEventInStopTime IncludingCurrentEvents $ do x <- readRef totalUpTime y <- liftDynamics stoptime n <- readRef nRep nImmed <- readRef nImmedRep return (x / (2 * y), fromIntegral nImmed / fromIntegral n) main = runSimulation model specs >>= print