1
0
mirror of https://github.com/SquidDev-CC/CC-Tweaked synced 2025-01-18 21:22:56 +00:00

Update Cobalt to 0.7

- Timeouts are now driven by an interrupt system, rather than polling.
   While we do not impose memory limits, this should close #1333.

 - Update the table library to largely match Lua 5.4:
    - Add table.move
    - Table methods (with the exception of foreach/foreachi) now use
      metamethods (closes #1088).
   There's still some remaining quirks (for instance, table.insert
   accepts values out-of-bounds), but I think that's fine.

 - Cobalt's threaded-coroutine system is gone (load now supports
   yielding), so we no longer track coroutine metrics.

 - Type errors now use __name. See #1355, though this does not apply to
   CC methods (either on the Java or CraftOS side), so is not enough to
   resolve it.

See https://github.com/SquidDev/Cobalt/compare/v0.6.0...v0.7.0 for the
full delta.
This commit is contained in:
Jonathan Coates 2023-03-26 19:42:55 +01:00
parent 0046b095b1
commit 5bb2e8e8cd
No known key found for this signature in database
GPG Key ID: B9E431FF07C98D06
9 changed files with 107 additions and 227 deletions

View File

@ -21,7 +21,7 @@ CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However,
| Hex literal fractional/exponent parts | ✔ | | | Hex literal fractional/exponent parts | ✔ | |
| Empty statements | ❌ | | | Empty statements | ❌ | |
| `__len` metamethod | ✔ | | | `__len` metamethod | ✔ | |
| `__ipairs` metamethod | ❌ | | | `__ipairs` metamethod | ❌ | Deprecated in Lua 5.3. `ipairs` uses `__len`/`__index` instead. |
| `__pairs` metamethod | ✔ | | | `__pairs` metamethod | ✔ | |
| `bit32` library | ✔ | | | `bit32` library | ✔ | |
| `collectgarbage` isrunning, generational, incremental options | ❌ | `collectgarbage` does not exist in CC:T. | | `collectgarbage` isrunning, generational, incremental options | ❌ | `collectgarbage` does not exist in CC:T. |
@ -32,8 +32,8 @@ CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However,
| `rawlen` function | ✔ | | | `rawlen` function | ✔ | |
| Negative index to `select` | ✔ | | | Negative index to `select` | ✔ | |
| Removed `unpack` | 🔶 | Only if `disable_lua51_features` is enabled in the configuration. | | Removed `unpack` | 🔶 | Only if `disable_lua51_features` is enabled in the configuration. |
| Arguments to `xpcall` | | | | Arguments to `xpcall` | | |
| Second return value from `coroutine.running` | | | | Second return value from `coroutine.running` | | |
| Removed `module` | ✔ | | | Removed `module` | ✔ | |
| `package.loaders` -> `package.searchers` | ❌ | | | `package.loaders` -> `package.searchers` | ❌ | |
| Second argument to loader functions | ✔ | | | Second argument to loader functions | ✔ | |
@ -41,7 +41,7 @@ CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However,
| `package.searchpath` | ✔ | | | `package.searchpath` | ✔ | |
| Removed `package.seeall` | ✔ | | | Removed `package.seeall` | ✔ | |
| `string.dump` on functions with upvalues (blanks them out) | ✔ | | | `string.dump` on functions with upvalues (blanks them out) | ✔ | |
| `string.rep` separator | | | | `string.rep` separator | | |
| `%g` match group | ❌ | | | `%g` match group | ❌ | |
| Removal of `%z` match group | ❌ | | | Removal of `%z` match group | ❌ | |
| Removed `table.maxn` | 🔶 | Only if `disable_lua51_features` is enabled in the configuration. | | Removed `table.maxn` | 🔶 | Only if `disable_lua51_features` is enabled in the configuration. |
@ -64,7 +64,7 @@ CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However,
| Removal of ambiguity error | ❌ | | | Removal of ambiguity error | ❌ | |
| Identifiers may no longer use locale-dependent letters | ✔ | | | Identifiers may no longer use locale-dependent letters | ✔ | |
| Ephemeron tables | ❌ | | | Ephemeron tables | ❌ | |
| Identical functions may be reused | ❌ | | | Identical functions may be reused | ❌ | Removed in Lua 5.4 |
| Generational garbage collector | ❌ | Cobalt uses the built-in Java garbage collector. | | Generational garbage collector | ❌ | Cobalt uses the built-in Java garbage collector. |
## Lua 5.3 ## Lua 5.3
@ -75,10 +75,10 @@ CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However,
| `\u{XXX}` escape sequence | ✔ | | | `\u{XXX}` escape sequence | ✔ | |
| `utf8` library | ✔ | | | `utf8` library | ✔ | |
| removed `__ipairs` metamethod | ✔ | | | removed `__ipairs` metamethod | ✔ | |
| `coroutine.isyieldable` | | | | `coroutine.isyieldable` | | |
| `string.dump` strip argument | ✔ | | | `string.dump` strip argument | ✔ | |
| `string.pack`/`string.unpack`/`string.packsize` | ✔ | | | `string.pack`/`string.unpack`/`string.packsize` | ✔ | |
| `table.move` | | | | `table.move` | | |
| `math.atan2` -> `math.atan` | ❌ | | | `math.atan2` -> `math.atan` | ❌ | |
| Removed `math.frexp`, `math.ldexp`, `math.pow`, `math.cosh`, `math.sinh`, `math.tanh` | ❌ | | | Removed `math.frexp`, `math.ldexp`, `math.pow`, `math.cosh`, `math.sinh`, `math.tanh` | ❌ | |
| `math.maxinteger`/`math.mininteger` | ❌ | | | `math.maxinteger`/`math.mininteger` | ❌ | |
@ -87,7 +87,7 @@ CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However,
| `math.ult` | ❌ | | | `math.ult` | ❌ | |
| Removed `bit32` library | ❌ | | | Removed `bit32` library | ❌ | |
| Remove `*` from `file:read` modes | ✔ | | | Remove `*` from `file:read` modes | ✔ | |
| Metamethods respected in `table.*`, `ipairs` | 🔶 | Only `__lt` is respected. | | Metamethods respected in `table.*`, `ipairs` | ✔ | |
## Lua 5.0 ## Lua 5.0
| Feature | Supported? | Notes | | Feature | Supported? | Notes |

View File

@ -19,7 +19,7 @@ parchmentMc = "1.19.3"
asm = "9.3" asm = "9.3"
autoService = "1.0.1" autoService = "1.0.1"
checkerFramework = "3.32.0" checkerFramework = "3.32.0"
cobalt = "0.6.0" cobalt = "0.7.0"
fastutil = "8.5.9" fastutil = "8.5.9"
guava = "31.1-jre" guava = "31.1-jre"
jetbrainsAnnotations = "24.0.1" jetbrainsAnnotations = "24.0.1"

View File

@ -178,8 +178,6 @@ public final class LanguageProvider implements DataProvider {
add(Metrics.HTTP_DOWNLOAD, "HTTP download"); add(Metrics.HTTP_DOWNLOAD, "HTTP download");
add(Metrics.WEBSOCKET_INCOMING, "Websocket incoming"); add(Metrics.WEBSOCKET_INCOMING, "Websocket incoming");
add(Metrics.WEBSOCKET_OUTGOING, "Websocket outgoing"); add(Metrics.WEBSOCKET_OUTGOING, "Websocket outgoing");
add(Metrics.COROUTINES_CREATED, "Coroutines created");
add(Metrics.COROUTINES_DISPOSED, "Coroutines disposed");
add(Metrics.TURTLE_OPS, "Turtle operations"); add(Metrics.TURTLE_OPS, "Turtle operations");
add(AggregatedMetric.TRANSLATION_PREFIX + Aggregate.MAX.id(), "%s (max)"); add(AggregatedMetric.TRANSLATION_PREFIX + Aggregate.MAX.id(), "%s (max)");

View File

@ -4,9 +4,13 @@
package dan200.computercraft.core.computer; package dan200.computercraft.core.computer;
import com.google.errorprone.annotations.concurrent.GuardedBy;
import dan200.computercraft.core.lua.ILuaMachine; import dan200.computercraft.core.lua.ILuaMachine;
import dan200.computercraft.core.lua.MachineResult; import dan200.computercraft.core.lua.MachineResult;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
/** /**
@ -48,6 +52,8 @@ public final class TimeoutState {
public static final String ABORT_MESSAGE = "Too long without yielding"; public static final String ABORT_MESSAGE = "Too long without yielding";
private final ComputerThread scheduler; private final ComputerThread scheduler;
@GuardedBy("this")
private final List<Runnable> listeners = new ArrayList<>(0);
private boolean paused; private boolean paused;
private boolean softAbort; private boolean softAbort;
@ -92,8 +98,15 @@ public final class TimeoutState {
// Important: The weird arithmetic here is important, as nanoTime may return negative values, and so we // Important: The weird arithmetic here is important, as nanoTime may return negative values, and so we
// need to handle overflow. // need to handle overflow.
var now = System.nanoTime(); var now = System.nanoTime();
if (!paused) paused = currentDeadline - now <= 0 && scheduler.hasPendingWork(); // now >= currentDeadline var changed = false;
if (!softAbort) softAbort = now - cumulativeStart - TIMEOUT >= 0; // now - cumulativeStart >= TIMEOUT if (!paused && (paused = currentDeadline - now <= 0 && scheduler.hasPendingWork())) { // now >= currentDeadline
changed = true;
}
if (!softAbort && (softAbort = now - cumulativeStart - TIMEOUT >= 0)) { // now - cumulativeStart >= TIMEOUT
changed = true;
}
if (changed) updateListeners();
} }
/** /**
@ -131,6 +144,9 @@ public final class TimeoutState {
*/ */
void hardAbort() { void hardAbort() {
softAbort = hardAbort = true; softAbort = hardAbort = true;
synchronized (this) {
updateListeners();
}
} }
/** /**
@ -153,6 +169,7 @@ public final class TimeoutState {
// We set the cumulative time to difference between current time and "nominal start time". // We set the cumulative time to difference between current time and "nominal start time".
cumulativeElapsed = System.nanoTime() - cumulativeStart; cumulativeElapsed = System.nanoTime() - cumulativeStart;
paused = false; paused = false;
updateListeners();
} }
/** /**
@ -161,5 +178,22 @@ public final class TimeoutState {
synchronized void stopTimer() { synchronized void stopTimer() {
cumulativeElapsed = 0; cumulativeElapsed = 0;
paused = softAbort = hardAbort = false; paused = softAbort = hardAbort = false;
updateListeners();
}
@GuardedBy("this")
private void updateListeners() {
for (var listener : listeners) listener.run();
}
public synchronized void addListener(Runnable listener) {
Objects.requireNonNull(listener, "listener cannot be null");
listeners.add(listener);
listener.run();
}
public synchronized void removeListener(Runnable listener) {
Objects.requireNonNull(listener, "listener cannot be null");
listeners.remove(listener);
} }
} }

View File

@ -13,19 +13,15 @@ import dan200.computercraft.core.Logging;
import dan200.computercraft.core.asm.LuaMethod; import dan200.computercraft.core.asm.LuaMethod;
import dan200.computercraft.core.asm.ObjectSource; import dan200.computercraft.core.asm.ObjectSource;
import dan200.computercraft.core.computer.TimeoutState; import dan200.computercraft.core.computer.TimeoutState;
import dan200.computercraft.core.metrics.Metrics;
import dan200.computercraft.core.util.Nullability; import dan200.computercraft.core.util.Nullability;
import dan200.computercraft.core.util.ThreadUtils;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.squiddev.cobalt.*; import org.squiddev.cobalt.*;
import org.squiddev.cobalt.compiler.CompileException; import org.squiddev.cobalt.compiler.CompileException;
import org.squiddev.cobalt.compiler.LoadState; import org.squiddev.cobalt.compiler.LoadState;
import org.squiddev.cobalt.debug.DebugFrame; import org.squiddev.cobalt.interrupt.InterruptAction;
import org.squiddev.cobalt.debug.DebugHandler; import org.squiddev.cobalt.lib.Bit32Lib;
import org.squiddev.cobalt.debug.DebugState; import org.squiddev.cobalt.lib.CoreLibraries;
import org.squiddev.cobalt.lib.*;
import org.squiddev.cobalt.lib.platform.VoidResourceManipulator;
import javax.annotation.Nullable; import javax.annotation.Nullable;
import java.io.IOException; import java.io.IOException;
@ -33,56 +29,41 @@ import java.io.InputStream;
import java.io.Serial; import java.io.Serial;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.util.*; import java.util.*;
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import static org.squiddev.cobalt.ValueFactory.valueOf; import static org.squiddev.cobalt.ValueFactory.valueOf;
import static org.squiddev.cobalt.ValueFactory.varargsOf; import static org.squiddev.cobalt.ValueFactory.varargsOf;
import static org.squiddev.cobalt.debug.DebugFrame.FLAG_HOOKED;
import static org.squiddev.cobalt.debug.DebugFrame.FLAG_HOOKYIELD;
public class CobaltLuaMachine implements ILuaMachine { public class CobaltLuaMachine implements ILuaMachine {
private static final Logger LOG = LoggerFactory.getLogger(CobaltLuaMachine.class); private static final Logger LOG = LoggerFactory.getLogger(CobaltLuaMachine.class);
private static final ThreadPoolExecutor COROUTINES = new ThreadPoolExecutor(
0, Integer.MAX_VALUE,
5L, TimeUnit.MINUTES,
new SynchronousQueue<>(),
ThreadUtils.factory("Coroutine")
);
private static final LuaMethod FUNCTION_METHOD = (target, context, args) -> ((ILuaFunction) target).call(args); private static final LuaMethod FUNCTION_METHOD = (target, context, args) -> ((ILuaFunction) target).call(args);
private final TimeoutState timeout; private final TimeoutState timeout;
private final TimeoutDebugHandler debug; private final Runnable timeoutListener = this::updateTimeout;
private final ILuaContext context; private final ILuaContext context;
private @Nullable LuaState state; private final LuaState state;
private @Nullable LuaTable globals; private final LuaThread mainRoutine;
private volatile boolean isDisposed = false;
private boolean thrownSoftAbort;
private @Nullable LuaThread mainRoutine = null;
private @Nullable String eventFilter = null; private @Nullable String eventFilter = null;
public CobaltLuaMachine(MachineEnvironment environment, InputStream bios) throws MachineException, IOException { public CobaltLuaMachine(MachineEnvironment environment, InputStream bios) throws MachineException, IOException {
timeout = environment.timeout(); timeout = environment.timeout();
context = environment.context(); context = environment.context();
debug = new TimeoutDebugHandler();
// Create an environment to run in // Create an environment to run in
var metrics = environment.metrics();
var state = this.state = LuaState.builder() var state = this.state = LuaState.builder()
.resourceManipulator(new VoidResourceManipulator()) .interruptHandler(() -> {
.debug(debug) if (timeout.isHardAborted() || isDisposed) throw new HardAbortError();
.coroutineExecutor(command -> { if (timeout.isSoftAborted() && !thrownSoftAbort) {
metrics.observe(Metrics.COROUTINES_CREATED); thrownSoftAbort = true;
COROUTINES.execute(() -> { throw new LuaError(TimeoutState.ABORT_MESSAGE);
try {
command.run();
} finally {
metrics.observe(Metrics.COROUTINES_DISPOSED);
} }
});
return timeout.isPaused() ? InterruptAction.SUSPEND : InterruptAction.CONTINUE;
}) })
.errorReporter((e, msg) -> { .errorReporter((e, msg) -> {
if (LOG.isErrorEnabled(Logging.VM_ERROR)) { if (LOG.isErrorEnabled(Logging.VM_ERROR)) {
@ -91,35 +72,16 @@ public class CobaltLuaMachine implements ILuaMachine {
}) })
.build(); .build();
globals = new LuaTable(); // Set up our global table.
state.setupThread(globals); var globals = state.getMainThread().getfenv();
CoreLibraries.debugGlobals(state);
// Add basic libraries Bit32Lib.add(state, globals);
globals.load(state, new BaseLib());
globals.load(state, new TableLib());
globals.load(state, new StringLib());
globals.load(state, new MathLib());
globals.load(state, new CoroutineLib());
globals.load(state, new Bit32Lib());
globals.load(state, new Utf8Lib());
globals.load(state, new DebugLib());
// Remove globals we don't want to expose
globals.rawset("collectgarbage", Constants.NIL);
globals.rawset("dofile", Constants.NIL);
globals.rawset("loadfile", Constants.NIL);
globals.rawset("print", Constants.NIL);
// Add version globals
globals.rawset("_VERSION", valueOf("Lua 5.1"));
globals.rawset("_HOST", valueOf(environment.hostString())); globals.rawset("_HOST", valueOf(environment.hostString()));
globals.rawset("_CC_DEFAULT_SETTINGS", valueOf(CoreConfig.defaultComputerSettings)); globals.rawset("_CC_DEFAULT_SETTINGS", valueOf(CoreConfig.defaultComputerSettings));
if (CoreConfig.disableLua51Features) { if (CoreConfig.disableLua51Features) globals.rawset("_CC_DISABLE_LUA51_FEATURES", Constants.TRUE);
globals.rawset("_CC_DISABLE_LUA51_FEATURES", Constants.TRUE);
}
// Add default APIs // Add default APIs
for (var api : environment.apis()) addAPI(api); for (var api : environment.apis()) addAPI(globals, api);
// And load the BIOS // And load the BIOS
try { try {
@ -128,11 +90,11 @@ public class CobaltLuaMachine implements ILuaMachine {
} catch (CompileException e) { } catch (CompileException e) {
throw new MachineException(Nullability.assertNonNull(e.getMessage())); throw new MachineException(Nullability.assertNonNull(e.getMessage()));
} }
timeout.addListener(timeoutListener);
} }
private void addAPI(ILuaAPI api) { private void addAPI(LuaTable globals, ILuaAPI api) {
if (globals == null) throw new IllegalStateException("Machine has been closed");
// Add the methods of an API to the global table // Add the methods of an API to the global table
var table = wrapLuaObject(api); var table = wrapLuaObject(api);
if (table == null) { if (table == null) {
@ -144,42 +106,41 @@ public class CobaltLuaMachine implements ILuaMachine {
for (var name : names) globals.rawset(name, table); for (var name : names) globals.rawset(name, table);
} }
private void updateTimeout() {
if (isDisposed) return;
if (!timeout.isSoftAborted()) thrownSoftAbort = false;
if (timeout.isSoftAborted() || timeout.isPaused()) state.interrupt();
}
@Override @Override
public MachineResult handleEvent(@Nullable String eventName, @Nullable Object[] arguments) { public MachineResult handleEvent(@Nullable String eventName, @Nullable Object[] arguments) {
if (mainRoutine == null || state == null) throw new IllegalStateException("Machine has been closed"); if (isDisposed) throw new IllegalStateException("Machine has been closed");
if (eventFilter != null && eventName != null && !eventName.equals(eventFilter) && !eventName.equals("terminate")) { if (eventFilter != null && eventName != null && !eventName.equals(eventFilter) && !eventName.equals("terminate")) {
return MachineResult.OK; return MachineResult.OK;
} }
// If the soft abort has been cleared then we can reset our flag.
timeout.refresh();
if (!timeout.isSoftAborted()) debug.thrownSoftAbort = false;
try { try {
Varargs resumeArgs = Constants.NONE; var resumeArgs = eventName == null ? Constants.NONE : varargsOf(valueOf(eventName), toValues(arguments));
if (eventName != null) {
resumeArgs = varargsOf(valueOf(eventName), toValues(arguments));
}
// Resume the current thread, or the main one when first starting off. // Resume the current thread, or the main one when first starting off.
var thread = state.getCurrentThread(); var thread = state.getCurrentThread();
if (thread == null || thread == state.getMainThread()) thread = mainRoutine; if (thread == null || thread == state.getMainThread()) thread = mainRoutine;
var results = LuaThread.run(thread, resumeArgs); var results = LuaThread.run(thread, resumeArgs);
if (timeout.isHardAborted()) throw HardAbortError.INSTANCE; if (timeout.isHardAborted()) throw new HardAbortError();
if (results == null) return MachineResult.PAUSE; if (results == null) return MachineResult.PAUSE;
var filter = results.first(); var filter = results.first();
eventFilter = filter.isString() ? filter.toString() : null; eventFilter = filter.isString() ? filter.toString() : null;
if (mainRoutine.getStatus().equals("dead")) { if (!mainRoutine.isAlive()) {
close(); close();
return MachineResult.GENERIC_ERROR; return MachineResult.GENERIC_ERROR;
} else { } else {
return MachineResult.OK; return MachineResult.OK;
} }
} catch (HardAbortError | InterruptedException e) { } catch (HardAbortError e) {
close(); close();
return MachineResult.TIMEOUT; return MachineResult.TIMEOUT;
} catch (LuaError e) { } catch (LuaError e) {
@ -191,23 +152,13 @@ public class CobaltLuaMachine implements ILuaMachine {
@Override @Override
public void printExecutionState(StringBuilder out) { public void printExecutionState(StringBuilder out) {
var state = this.state;
if (state == null) {
out.append("CobaltLuaMachine is terminated\n");
} else {
state.printExecutionState(out);
}
} }
@Override @Override
public void close() { public void close() {
var state = this.state; isDisposed = true;
if (state == null) return; state.interrupt();
timeout.removeListener(timeoutListener);
state.abandon();
mainRoutine = null;
this.state = null;
globals = null;
} }
@Nullable @Nullable
@ -228,7 +179,7 @@ public class CobaltLuaMachine implements ILuaMachine {
: new ResultInterpreterFunction(this, method.getMethod(), instance, context, method.getName()))); : new ResultInterpreterFunction(this, method.getMethod(), instance, context, method.getName())));
try { try {
if (table.keyCount() == 0) return null; if (table.next(Constants.NIL).first().isNil()) return null;
} catch (LuaError ignored) { } catch (LuaError ignored) {
// next should never throw on nil. // next should never throw on nil.
} }
@ -241,9 +192,7 @@ public class CobaltLuaMachine implements ILuaMachine {
if (object instanceof Number num) return valueOf(num.doubleValue()); if (object instanceof Number num) return valueOf(num.doubleValue());
if (object instanceof Boolean bool) return valueOf(bool); if (object instanceof Boolean bool) return valueOf(bool);
if (object instanceof String str) return valueOf(str); if (object instanceof String str) return valueOf(str);
if (object instanceof byte[] b) { if (object instanceof byte[] b) return valueOf(Arrays.copyOf(b, b.length));
return valueOf(Arrays.copyOf(b, b.length));
}
if (object instanceof ByteBuffer b) { if (object instanceof ByteBuffer b) {
var bytes = new byte[b.remaining()]; var bytes = new byte[b.remaining()];
b.get(bytes); b.get(bytes);
@ -317,25 +266,19 @@ public class CobaltLuaMachine implements ILuaMachine {
@Nullable @Nullable
static Object toObject(LuaValue value, @Nullable IdentityHashMap<LuaValue, Object> objects) { static Object toObject(LuaValue value, @Nullable IdentityHashMap<LuaValue, Object> objects) {
switch (value.type()) { return switch (value.type()) {
case Constants.TNIL: case Constants.TNIL -> null;
case Constants.TNONE: case Constants.TINT, Constants.TNUMBER -> value.toDouble();
return null; case Constants.TBOOLEAN -> value.toBoolean();
case Constants.TINT: case Constants.TSTRING -> value.toString();
case Constants.TNUMBER: case Constants.TTABLE -> {
return value.toDouble();
case Constants.TBOOLEAN:
return value.toBoolean();
case Constants.TSTRING:
return value.toString();
case Constants.TTABLE: {
// Table: // Table:
// Start remembering stuff // Start remembering stuff
if (objects == null) { if (objects == null) {
objects = new IdentityHashMap<>(1); objects = new IdentityHashMap<>(1);
} else { } else {
var existing = objects.get(value); var existing = objects.get(value);
if (existing != null) return existing; if (existing != null) yield existing;
} }
Map<Object, Object> table = new HashMap<>(); Map<Object, Object> table = new HashMap<>();
objects.put(value, table); objects.put(value, table);
@ -361,11 +304,10 @@ public class CobaltLuaMachine implements ILuaMachine {
table.put(keyObject, valueObject); table.put(keyObject, valueObject);
} }
} }
return table; yield table;
}
default:
return null;
} }
default -> null;
};
} }
static Object[] toObjects(Varargs values) { static Object[] toObjects(Varargs values) {
@ -375,83 +317,12 @@ public class CobaltLuaMachine implements ILuaMachine {
return objects; return objects;
} }
/**
* A {@link DebugHandler} which observes the {@link TimeoutState} and responds accordingly.
*/
private class TimeoutDebugHandler extends DebugHandler {
private final TimeoutState timeout;
private int count = 0;
boolean thrownSoftAbort;
private boolean isPaused;
private int oldFlags;
private boolean oldInHook;
TimeoutDebugHandler() {
timeout = CobaltLuaMachine.this.timeout;
}
@Override
public void onInstruction(DebugState ds, DebugFrame di, int pc) throws LuaError, UnwindThrowable {
di.pc = pc;
if (isPaused) resetPaused(ds, di);
// We check our current pause/abort state every 128 instructions.
if ((count = (count + 1) & 127) == 0) {
if (timeout.isHardAborted() || state == null) throw HardAbortError.INSTANCE;
if (timeout.isPaused()) handlePause(ds, di);
if (timeout.isSoftAborted()) handleSoftAbort();
}
super.onInstruction(ds, di, pc);
}
@Override
public void poll() throws LuaError {
var state = CobaltLuaMachine.this.state;
if (timeout.isHardAborted() || state == null) throw HardAbortError.INSTANCE;
if (timeout.isPaused()) LuaThread.suspendBlocking(state);
if (timeout.isSoftAborted()) handleSoftAbort();
}
private void resetPaused(DebugState ds, DebugFrame di) {
// Restore the previous paused state
isPaused = false;
ds.inhook = oldInHook;
di.flags = oldFlags;
}
private void handleSoftAbort() throws LuaError {
// If we already thrown our soft abort error then don't do it again.
if (thrownSoftAbort) return;
thrownSoftAbort = true;
throw new LuaError(TimeoutState.ABORT_MESSAGE);
}
private void handlePause(DebugState ds, DebugFrame di) throws LuaError, UnwindThrowable {
// Preserve the current state
isPaused = true;
oldInHook = ds.inhook;
oldFlags = di.flags;
// Suspend the state. This will probably throw, but we need to handle the case where it won't.
di.flags |= FLAG_HOOKYIELD | FLAG_HOOKED;
LuaThread.suspend(ds.getLuaState());
resetPaused(ds, di);
}
}
private static final class HardAbortError extends Error { private static final class HardAbortError extends Error {
@Serial @Serial
private static final long serialVersionUID = 7954092008586367501L; private static final long serialVersionUID = 7954092008586367501L;
@SuppressWarnings("StaticAssignmentOfThrowable")
static final HardAbortError INSTANCE = new HardAbortError();
private HardAbortError() { private HardAbortError() {
super("Hard Abort", null, true, false); super("Hard Abort");
} }
} }
} }

View File

@ -27,11 +27,7 @@ class TableImpl implements dan200.computercraft.api.lua.LuaTable<Object, Object>
@Override @Override
public int size() { public int size() {
checkValid(); checkValid();
try { return table.size();
return table.keyCount();
} catch (LuaError e) {
throw new IllegalStateException(e);
}
} }
@Override @Override

View File

@ -73,20 +73,16 @@ final class VarargArguments implements IArguments {
@Override @Override
public ByteBuffer getBytes(int index) throws LuaException { public ByteBuffer getBytes(int index) throws LuaException {
var value = varargs.arg(index + 1); var value = varargs.arg(index + 1);
if (!(value instanceof LuaBaseString)) throw LuaValues.badArgument(index, "string", value.typeName()); if (!(value instanceof LuaString str)) throw LuaValues.badArgument(index, "string", value.typeName());
return str.toBuffer();
var str = ((LuaBaseString) value).strvalue();
return ByteBuffer.wrap(str.bytes, str.offset, str.length).asReadOnlyBuffer();
} }
@Override @Override
public Optional<ByteBuffer> optBytes(int index) throws LuaException { public Optional<ByteBuffer> optBytes(int index) throws LuaException {
var value = varargs.arg(index + 1); var value = varargs.arg(index + 1);
if (value.isNil()) return Optional.empty(); if (value.isNil()) return Optional.empty();
if (!(value instanceof LuaBaseString)) throw LuaValues.badArgument(index, "string", value.typeName()); if (!(value instanceof LuaString str)) throw LuaValues.badArgument(index, "string", value.typeName());
return Optional.of(str.toBuffer());
var str = ((LuaBaseString) value).strvalue();
return Optional.of(ByteBuffer.wrap(str.bytes, str.offset, str.length).asReadOnlyBuffer());
} }
@Override @Override

View File

@ -24,9 +24,6 @@ public final class Metrics {
public static final Metric.Event WEBSOCKET_INCOMING = new Metric.Event("websocket_incoming", "bytes", Metric::formatBytes); public static final Metric.Event WEBSOCKET_INCOMING = new Metric.Event("websocket_incoming", "bytes", Metric::formatBytes);
public static final Metric.Event WEBSOCKET_OUTGOING = new Metric.Event("websocket_outgoing", "bytes", Metric::formatBytes); public static final Metric.Event WEBSOCKET_OUTGOING = new Metric.Event("websocket_outgoing", "bytes", Metric::formatBytes);
public static final Metric.Counter COROUTINES_CREATED = new Metric.Counter("coroutines_created");
public static final Metric.Counter COROUTINES_DISPOSED = new Metric.Counter("coroutines_dead");
public static final Metric.Counter TURTLE_OPS = new Metric.Counter("turtle_ops"); public static final Metric.Counter TURTLE_OPS = new Metric.Counter("turtle_ops");
/** /**

View File

@ -27,11 +27,12 @@ import org.opentest4j.AssertionFailedError;
import org.opentest4j.TestAbortedException; import org.opentest4j.TestAbortedException;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.squiddev.cobalt.*; import org.squiddev.cobalt.LuaState;
import org.squiddev.cobalt.LuaString;
import org.squiddev.cobalt.LuaThread;
import org.squiddev.cobalt.debug.DebugFrame; import org.squiddev.cobalt.debug.DebugFrame;
import org.squiddev.cobalt.debug.DebugHook; import org.squiddev.cobalt.debug.DebugHook;
import org.squiddev.cobalt.debug.DebugState; import org.squiddev.cobalt.debug.DebugState;
import org.squiddev.cobalt.function.OneArgFunction;
import javax.annotation.Nullable; import javax.annotation.Nullable;
import java.io.File; import java.io.File;
@ -480,13 +481,8 @@ public class ComputerTestDelegate {
CoverageLuaMachine(MachineEnvironment environment, InputStream bios) throws MachineException, IOException { CoverageLuaMachine(MachineEnvironment environment, InputStream bios) throws MachineException, IOException {
super(environment, bios); super(environment, bios);
LuaTable globals;
LuaThread mainRoutine; LuaThread mainRoutine;
try { try {
var globalField = CobaltLuaMachine.class.getDeclaredField("globals");
globalField.setAccessible(true);
globals = (LuaTable) globalField.get(this);
var threadField = CobaltLuaMachine.class.getDeclaredField("mainRoutine"); var threadField = CobaltLuaMachine.class.getDeclaredField("mainRoutine");
threadField.setAccessible(true); threadField.setAccessible(true);
mainRoutine = (LuaThread) threadField.get(this); mainRoutine = (LuaThread) threadField.get(this);
@ -513,21 +509,13 @@ public class ComputerTestDelegate {
if (frame.closure == null) return; if (frame.closure == null) return;
var proto = frame.closure.getPrototype(); var proto = frame.closure.getPrototype();
if (!proto.source.startsWith('@')) return; if (!proto.source.startsWith((byte) '@')) return;
var map = coverage.computeIfAbsent(proto.source, x -> new Int2IntArrayMap()); var map = coverage.computeIfAbsent(proto.source, x -> new Int2IntArrayMap());
map.put(newLine, map.get(newLine) + 1); map.put(newLine, map.get(newLine) + 1);
} }
}; };
((LuaTable) globals.rawget("coroutine")).rawset("create", new OneArgFunction() {
@Override
public LuaValue call(LuaState state, LuaValue arg) throws LuaError {
var thread = new LuaThread(state, arg.checkFunction(), state.getCurrentThread().getfenv());
thread.getDebugState().setHook(hook, false, true, false, 0);
return thread;
}
});
mainRoutine.getDebugState().setHook(hook, false, true, false, 0); mainRoutine.getDebugState().setHook(hook, false, true, false, 0);
} }
} }