225 lines
4.9 KiB
Java
225 lines
4.9 KiB
Java
package appeng.me.cache;
|
|
|
|
import java.util.HashMap;
|
|
import java.util.PriorityQueue;
|
|
|
|
import net.minecraft.crash.CrashReport;
|
|
import net.minecraft.crash.CrashReportCategory;
|
|
import net.minecraft.util.ReportedException;
|
|
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<IGridNode, TickTracker> alertable = new HashMap<IGridNode, TickTracker>();
|
|
|
|
final HashMap<IGridNode, TickTracker> sleeping = new HashMap<IGridNode, TickTracker>();
|
|
final HashMap<IGridNode, TickTracker> awake = new HashMap<IGridNode, TickTracker>();
|
|
|
|
final PriorityQueue<TickTracker> upcomingTicks = new PriorityQueue<TickTracker>();
|
|
|
|
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()
|
|
{
|
|
TickTracker tt = null;
|
|
try
|
|
{
|
|
currentTick++;
|
|
while (!upcomingTicks.isEmpty())
|
|
{
|
|
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!
|
|
}
|
|
}
|
|
catch( Throwable t )
|
|
{
|
|
CrashReport crashreport = CrashReport.makeCrashReport(t, "Ticking GridNode");
|
|
CrashReportCategory crashreportcategory = crashreport.makeCategory( tt.gt.getClass().getSimpleName() + " being ticked." );
|
|
tt.addEntityCrashInfo(crashreportcategory);
|
|
throw new ReportedException(crashreport);
|
|
}
|
|
}
|
|
|
|
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 alerted 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)
|
|
{
|
|
|
|
}
|
|
}
|