package appeng.me.cache; import java.util.HashMap; import java.util.PriorityQueue; import appeng.api.networking.IGrid; import appeng.api.networking.IGridHost; import appeng.api.networking.IGridNode; import appeng.api.networking.IGridStorage; import appeng.api.networking.ticking.IGridTickable; import appeng.api.networking.ticking.ITickManager; import appeng.api.networking.ticking.TickRateModulation; import appeng.api.networking.ticking.TickingRequest; import appeng.me.cache.helpers.TickTracker; public class TickManagerCache implements ITickManager { private long currentTick = 0; final IGrid myGrid; public TickManagerCache(IGrid g) { myGrid = g; } final HashMap alertable = new HashMap(); final HashMap sleeping = new HashMap(); final HashMap awake = new HashMap(); final PriorityQueue upcomingTicks = new PriorityQueue(); public long getCurrentTick() { return currentTick; } public long getAvgNanoTime(IGridNode node) { TickTracker tt = awake.get( node ); if ( tt == null ) tt = sleeping.get( node ); if ( tt == null ) return -1; return tt.getAvgNanos(); } @Override public void onUpdateTick() { currentTick++; while (!upcomingTicks.isEmpty()) { TickTracker tt = upcomingTicks.peek(); int diff = (int) (currentTick - tt.lastTick); if ( diff >= tt.current_rate ) { // remove tt.. upcomingTicks.poll(); TickRateModulation mod = tt.gt.tickingRequest( tt.node, diff ); switch (mod) { case FASTER: tt.setRate( tt.current_rate - 2 ); break; case IDLE: tt.setRate( tt.request.maxTickRate ); break; case SAME: break; case SLEEP: sleepDevice( tt.node ); break; case SLOWER: tt.setRate( tt.current_rate + 1 ); break; case URGENT: tt.setRate( 0 ); break; default: break; } if ( awake.containsKey( tt.node ) ) addToQueue( tt ); } else return; // done! } } private void addToQueue(TickTracker tt) { tt.lastTick = currentTick; upcomingTicks.add( tt ); } @Override public boolean alertDevice(IGridNode node) { TickTracker tt = alertable.get( node ); if ( tt == null ) return false; // throw new RuntimeException( // "Invalid Alertted device, this node is not marked as alertable, or part of this grid." ); // set to awake, this is for sanity. sleeping.remove( node ); awake.put( node, tt ); // configure sort. tt.lastTick = tt.lastTick - tt.request.maxTickRate; tt.current_rate = tt.request.minTickRate; // prevent dupes and tick build up. upcomingTicks.remove( tt ); upcomingTicks.add( tt ); return true; } @Override public boolean sleepDevice(IGridNode node) { if ( awake.containsKey( node ) ) { TickTracker gt = awake.get( node ); awake.remove( node ); sleeping.put( node, gt ); return true; } return false; } @Override public boolean wakeDevice(IGridNode node) { if ( sleeping.containsKey( node ) ) { TickTracker gt = sleeping.get( node ); sleeping.remove( node ); awake.put( node, gt ); addToQueue( gt ); return true; } return false; } @Override public void removeNode(IGridNode gridNode, IGridHost machine) { if ( machine instanceof IGridTickable ) { alertable.remove( gridNode ); sleeping.remove( gridNode ); awake.remove( gridNode ); } } @Override public void addNode(IGridNode gridNode, IGridHost machine) { if ( machine instanceof IGridTickable ) { TickingRequest tr = ((IGridTickable) machine).getTickingRequest( gridNode ); if ( tr != null ) { TickTracker tt = new TickTracker( tr, gridNode, (IGridTickable) machine, currentTick, this ); if ( tr.canBeAlerted ) alertable.put( gridNode, tt ); if ( tr.isSleeping ) sleeping.put( gridNode, tt ); else { awake.put( gridNode, tt ); addToQueue( tt ); } } } } @Override public void onSplit(IGridStorage storageB) { } @Override public void onJoin(IGridStorage storageB) { } @Override public void populateGridStorage(IGridStorage storage) { } }