// power manager example
mdp

const int QMAX; // max queue size

// to model the pm making a choice and then a move being made we need
// two clock ticks for each transition
// first the pm decides tick1 and then the system moves tick2

module timer

	c : [0..1];

	[tick1] c=0 -> (c'=1);
	[tick2] c=1 -> (c'=0);
	
endmodule

//-------------------------------------------------------------------------

// POWER MANAGER
module PM

	pm : [0..4] init 4;
	// 0 - go to active 
	// 1 - go to idle 
	// 2 - go to idlelp 
	// 3 - go to stby  
	// 4 - go to sleep 

	[tick1] true -> (pm'=0);
	[tick1] true -> (pm'=1);
	[tick1] true -> (pm'=2);
	[tick1] true -> (pm'=3);
	[tick1] true -> (pm'=4);
	
endmodule


//-------------------------------------------------------------------------

// SERVICE REQUESTER
module SR

	sr : [0..1] init 0;
	// 0 idle
	// 1 1req
	
	[tick2] sr=0 -> 0.898: (sr'=0) + 0.102: (sr'=1);
	[tick2] sr=1 -> 0.454: (sr'=0) + 0.546: (sr'=1);

endmodule

//-------------------------------------------------------------------------

// SERVICE PROVIDER

module SP 

	sp : [0..10] init 9;
	// 0 active
	// 1 idle
	// 2 active_idlelp
	// 3 idlelp
	// 4 idlelp_active
	// 5 active_stby
	// 6 stby
	// 7 stby_active
    // 8 active_sleep
	// 9 sleep
	// 10 sleep_active
	
	// states where PM has no control (transient states)
	[tick2] sp=2  ->   0.75   : (sp'=2) + 0.25   : (sp'=3);   // active_idlelp
	[tick2] sp=4  ->   0.25   : (sp'=0) + 0.75   : (sp'=4);  // idlelp_active
	[tick2] sp=5  ->   0.995  : (sp'=5) + 0.005  : (sp'=6);  // active_stby
	[tick2] sp=7  ->   0.005  : (sp'=0) + 0.995  : (sp'=7);  // stby_active
	[tick2] sp=8  ->   0.9983 : (sp'=8) + 0.0017 : (sp'=9);  // active_sleep
	[tick2] sp=10 ->   0.0017 : (sp'=0) + 0.9983 : (sp'=10); // sleep_active

	// states where PM has control
	// goto_active
	[tick2] sp=0 & pm=0 -> (sp'=0); // active
	[tick2] sp=1 & pm=0 -> (sp'=0); // idle
	[tick2] sp=3 & pm=0 -> (sp'=4); // idlelp
	[tick2] sp=6 & pm=0 -> (sp'=7); // stby
	[tick2] sp=9 & pm=0 -> (sp'=10); // sleep
	// goto_idle
	[tick2] sp=0 & pm=1 -> (sp'=1); // active
	[tick2] sp=1 & pm=1 -> (sp'=1); // idle
	[tick2] sp=3 & pm=1 -> (sp'=3); // idlelp
	[tick2] sp=6 & pm=1 -> (sp'=6); // stby
	[tick2] sp=9 & pm=1 -> (sp'=9); // sleep
	// goto_idlelp
	[tick2] sp=0 & pm=2 -> (sp'=2); // active
	[tick2] sp=1 & pm=2 -> (sp'=2); // idle
	[tick2] sp=3 & pm=2 -> (sp'=3); // idlelp
	[tick2] sp=6 & pm=2 -> (sp'=6); // stby
	[tick2] sp=9 & pm=2 -> (sp'=9); // sleep
	// goto_stby
	[tick2] sp=0 & pm=3 -> (sp'=5); // active
	[tick2] sp=1 & pm=3 -> (sp'=5); // idle
	[tick2] sp=3 & pm=3 -> (sp'=5); // idlelp
	[tick2] sp=6 & pm=3 -> (sp'=6); // stby
	[tick2] sp=9 & pm=3 -> (sp'=9); // sleep
	// goto_sleep
	[tick2] sp=0 & pm=4 -> (sp'=8); // active
	[tick2] sp=1 & pm=4 -> (sp'=8); // idle
	[tick2] sp=3 & pm=4 -> (sp'=8); // idlelp
	[tick2] sp=6 & pm=4 -> (sp'=8); // stby
	[tick2] sp=9 & pm=4 -> (sp'=9); // sleep
	  
endmodule


//-------------------------------------------------------------------------

// SQ
module SQ

	q : [0..QMAX] init 0;
	
	// serve if busy
	[tick2] sr=0 & sp=0 -> (q'=max(q-1,0));
	[tick2] sr=1 & sp=0 -> (q'=q);
	
	// otherwise do nothing
	[tick2] sr=0 & sp>0 -> (q'=q);
	[tick2] sr=1 & sp>0 -> (q'=min(q+1,QMAX));

endmodule

//-------------------------------------------------------------------------
//rewards "time"
//	[tick2] bat=1 : 1;
//endrewards

rewards "power"
	[tick2] sp=0  & c=1  : 2.5;
	[tick2] sp=1  & c=1  : 1.5;
	[tick2] sp=2  & c=1  : 2.5;
	[tick2] sp=3  & c=1  : 0.8;
	[tick2] sp=4  & c=1  : 2.5;
	[tick2] sp=5  & c=1  : 2.5;
	[tick2] sp=6  & c=1  : 0.3;
	[tick2] sp=7  & c=1  : 2.5;
	[tick2] sp=8  & c=1  : 2.5;
	[tick2] sp=9  & c=1  : 0.1;
	[tick2] sp=10 & c=1  : 2.5;
endrewards

// is an instantaneous property but I suppose we can look at average size
// i.e. divide by the expected number of time steps
rewards "queue"
	[tick2] c=1  : q;
endrewards

rewards "lost"
	[tick2] sr=1 & sp>0 & q=2  : 1;
endrewards