mirror of
https://github.com/SquidDev-CC/CC-Tweaked
synced 2025-10-24 10:27:38 +00:00
Don't store terminal snapshot as a ByteBuf
There's too many issues here around object pooling and reference counting, that it's just not practical to correctly handle. We now just use a plain old byte[] rather than a ByteBuf. This does mean the array now lives on the Java heap rather than the direct heap, but I *think* that's fine. It's something that is hard to measure. Fixes #2059
This commit is contained in:
@@ -118,7 +118,7 @@ public class ServerComputer implements InputHandler, ComputerEnvironment {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public TerminalState getTerminalState() {
|
public TerminalState getTerminalState() {
|
||||||
return new TerminalState(terminal);
|
return TerminalState.create(terminal);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void keepAlive() {
|
public void keepAlive() {
|
||||||
|
@@ -8,7 +8,6 @@ import dan200.computercraft.core.terminal.Palette;
|
|||||||
import dan200.computercraft.core.terminal.Terminal;
|
import dan200.computercraft.core.terminal.Terminal;
|
||||||
import dan200.computercraft.core.util.Colour;
|
import dan200.computercraft.core.util.Colour;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
import net.minecraft.network.FriendlyByteBuf;
|
|
||||||
|
|
||||||
public class NetworkedTerminal extends Terminal {
|
public class NetworkedTerminal extends Terminal {
|
||||||
public NetworkedTerminal(int width, int height, boolean colour) {
|
public NetworkedTerminal(int width, int height, boolean colour) {
|
||||||
@@ -19,59 +18,61 @@ public class NetworkedTerminal extends Terminal {
|
|||||||
super(width, height, colour, changedCallback);
|
super(width, height, colour, changedCallback);
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void write(FriendlyByteBuf buffer) {
|
synchronized TerminalState write() {
|
||||||
buffer.writeInt(cursorX);
|
var contents = new byte[width * height * 2 + Palette.PALETTE_SIZE * 3];
|
||||||
buffer.writeInt(cursorY);
|
var idx = 0;
|
||||||
buffer.writeBoolean(cursorBlink);
|
|
||||||
buffer.writeByte(cursorBackgroundColour << 4 | cursorColour);
|
|
||||||
|
|
||||||
for (var y = 0; y < height; y++) {
|
for (var y = 0; y < height; y++) {
|
||||||
var text = this.text[y];
|
var text = this.text[y];
|
||||||
var textColour = this.textColour[y];
|
var textColour = this.textColour[y];
|
||||||
var backColour = backgroundColour[y];
|
var backColour = backgroundColour[y];
|
||||||
|
|
||||||
for (var x = 0; x < width; x++) buffer.writeByte(text.charAt(x) & 0xFF);
|
for (var x = 0; x < width; x++) contents[idx++] = (byte) (text.charAt(x) & 0xFF);
|
||||||
for (var x = 0; x < width; x++) {
|
for (var x = 0; x < width; x++) {
|
||||||
buffer.writeByte(getColour(
|
contents[idx++] = (byte) (getColour(backColour.charAt(x), Colour.BLACK) << 4 | getColour(textColour.charAt(x), Colour.WHITE));
|
||||||
backColour.charAt(x), Colour.BLACK) << 4 |
|
|
||||||
getColour(textColour.charAt(x), Colour.WHITE)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var i = 0; i < Palette.PALETTE_SIZE; i++) {
|
for (var i = 0; i < Palette.PALETTE_SIZE; i++) {
|
||||||
for (var channel : palette.getColour(i)) buffer.writeByte((int) (channel * 0xFF) & 0xFF);
|
for (var channel : palette.getColour(i)) contents[idx++] = (byte) ((int) (channel * 0xFF) & 0xFF);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
assert idx == contents.length;
|
||||||
|
return new TerminalState(colour, width, height, cursorX, cursorY, cursorBlink, cursorColour, cursorBackgroundColour, contents);
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void read(FriendlyByteBuf buffer) {
|
synchronized void read(TerminalState state) {
|
||||||
cursorX = buffer.readInt();
|
resize(state.width, state.height);
|
||||||
cursorY = buffer.readInt();
|
cursorX = state.cursorX;
|
||||||
cursorBlink = buffer.readBoolean();
|
cursorY = state.cursorY;
|
||||||
|
cursorBlink = state.cursorBlink;
|
||||||
|
|
||||||
var cursorColour = buffer.readByte();
|
cursorBackgroundColour = state.cursorBgColour;
|
||||||
cursorBackgroundColour = (cursorColour >> 4) & 0xF;
|
this.cursorColour = state.cursorFgColour;
|
||||||
this.cursorColour = cursorColour & 0xF;
|
|
||||||
|
|
||||||
|
var contents = state.contents;
|
||||||
|
var idx = 0;
|
||||||
for (var y = 0; y < height; y++) {
|
for (var y = 0; y < height; y++) {
|
||||||
var text = this.text[y];
|
var text = this.text[y];
|
||||||
var textColour = this.textColour[y];
|
var textColour = this.textColour[y];
|
||||||
var backColour = backgroundColour[y];
|
var backColour = backgroundColour[y];
|
||||||
|
|
||||||
for (var x = 0; x < width; x++) text.setChar(x, (char) (buffer.readByte() & 0xFF));
|
for (var x = 0; x < width; x++) text.setChar(x, (char) (contents[idx++] & 0xFF));
|
||||||
for (var x = 0; x < width; x++) {
|
for (var x = 0; x < width; x++) {
|
||||||
var colour = buffer.readByte();
|
var colour = contents[idx++];
|
||||||
backColour.setChar(x, BASE_16.charAt((colour >> 4) & 0xF));
|
backColour.setChar(x, BASE_16.charAt((colour >> 4) & 0xF));
|
||||||
textColour.setChar(x, BASE_16.charAt(colour & 0xF));
|
textColour.setChar(x, BASE_16.charAt(colour & 0xF));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var i = 0; i < Palette.PALETTE_SIZE; i++) {
|
for (var i = 0; i < Palette.PALETTE_SIZE; i++) {
|
||||||
var r = (buffer.readByte() & 0xFF) / 255.0;
|
var r = (contents[idx++] & 0xFF) / 255.0;
|
||||||
var g = (buffer.readByte() & 0xFF) / 255.0;
|
var g = (contents[idx++] & 0xFF) / 255.0;
|
||||||
var b = (buffer.readByte() & 0xFF) / 255.0;
|
var b = (contents[idx++] & 0xFF) / 255.0;
|
||||||
palette.setColour(i, r, g, b);
|
palette.setColour(i, r, g, b);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
assert idx == contents.length;
|
||||||
setChanged();
|
setChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -4,8 +4,6 @@
|
|||||||
|
|
||||||
package dan200.computercraft.shared.computer.terminal;
|
package dan200.computercraft.shared.computer.terminal;
|
||||||
|
|
||||||
import io.netty.buffer.ByteBuf;
|
|
||||||
import io.netty.buffer.Unpooled;
|
|
||||||
import net.minecraft.network.FriendlyByteBuf;
|
import net.minecraft.network.FriendlyByteBuf;
|
||||||
import org.jetbrains.annotations.Contract;
|
import org.jetbrains.annotations.Contract;
|
||||||
|
|
||||||
@@ -20,53 +18,72 @@ import javax.annotation.Nullable;
|
|||||||
*/
|
*/
|
||||||
public class TerminalState {
|
public class TerminalState {
|
||||||
private final boolean colour;
|
private final boolean colour;
|
||||||
private final int width;
|
final int width;
|
||||||
private final int height;
|
final int height;
|
||||||
private final ByteBuf buffer;
|
final int cursorX;
|
||||||
|
final int cursorY;
|
||||||
|
final boolean cursorBlink;
|
||||||
|
final int cursorBgColour;
|
||||||
|
final int cursorFgColour;
|
||||||
|
final byte[] contents;
|
||||||
|
|
||||||
public TerminalState(NetworkedTerminal terminal) {
|
TerminalState(
|
||||||
colour = terminal.isColour();
|
boolean colour, int width, int height, int cursorX, int cursorY, boolean cursorBlink, int cursorFgColour, int cursorBgColour, byte[] contents
|
||||||
width = terminal.getWidth();
|
) {
|
||||||
height = terminal.getHeight();
|
this.colour = colour;
|
||||||
|
this.width = width;
|
||||||
var buf = buffer = Unpooled.buffer();
|
this.height = height;
|
||||||
terminal.write(new FriendlyByteBuf(buf));
|
this.cursorX = cursorX;
|
||||||
|
this.cursorY = cursorY;
|
||||||
|
this.cursorBlink = cursorBlink;
|
||||||
|
this.cursorFgColour = cursorFgColour;
|
||||||
|
this.cursorBgColour = cursorBgColour;
|
||||||
|
this.contents = contents;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Contract("null -> null; !null -> !null")
|
@Contract("null -> null; !null -> !null")
|
||||||
public static @Nullable TerminalState create(@Nullable NetworkedTerminal terminal) {
|
public static @Nullable TerminalState create(@Nullable NetworkedTerminal terminal) {
|
||||||
return terminal == null ? null : new TerminalState(terminal);
|
return terminal == null ? null : terminal.write();
|
||||||
}
|
}
|
||||||
|
|
||||||
public TerminalState(FriendlyByteBuf buf) {
|
public TerminalState(FriendlyByteBuf buf) {
|
||||||
colour = buf.readBoolean();
|
colour = buf.readBoolean();
|
||||||
width = buf.readVarInt();
|
width = buf.readVarInt();
|
||||||
height = buf.readVarInt();
|
height = buf.readVarInt();
|
||||||
|
cursorX = buf.readVarInt();
|
||||||
|
cursorY = buf.readVarInt();
|
||||||
|
cursorBlink = buf.readBoolean();
|
||||||
|
|
||||||
var length = buf.readVarInt();
|
var cursorColour = buf.readByte();
|
||||||
buffer = buf.readBytes(length);
|
this.cursorBgColour = (cursorColour >> 4) & 0xF;
|
||||||
|
this.cursorFgColour = cursorColour & 0xF;
|
||||||
|
|
||||||
|
contents = buf.readByteArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void write(FriendlyByteBuf buf) {
|
public void write(FriendlyByteBuf buf) {
|
||||||
buf.writeBoolean(colour);
|
buf.writeBoolean(colour);
|
||||||
buf.writeVarInt(width);
|
buf.writeVarInt(width);
|
||||||
buf.writeVarInt(height);
|
buf.writeVarInt(height);
|
||||||
buf.writeVarInt(buffer.readableBytes());
|
buf.writeVarInt(cursorX);
|
||||||
buf.writeBytes(buffer, buffer.readerIndex(), buffer.readableBytes());
|
buf.writeVarInt(cursorY);
|
||||||
|
buf.writeBoolean(cursorBlink);
|
||||||
|
buf.writeByte(cursorBgColour << 4 | cursorFgColour);
|
||||||
|
|
||||||
|
buf.writeByteArray(contents);
|
||||||
}
|
}
|
||||||
|
|
||||||
public int size() {
|
public int size() {
|
||||||
return buffer.readableBytes();
|
return contents.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void apply(NetworkedTerminal terminal) {
|
public void apply(NetworkedTerminal terminal) {
|
||||||
terminal.resize(width, height);
|
terminal.read(this);
|
||||||
terminal.read(new FriendlyByteBuf(buffer));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public NetworkedTerminal create() {
|
public NetworkedTerminal create() {
|
||||||
var terminal = new NetworkedTerminal(width, height, colour);
|
var terminal = new NetworkedTerminal(width, height, colour);
|
||||||
terminal.read(new FriendlyByteBuf(buffer));
|
terminal.read(this);
|
||||||
return terminal;
|
return terminal;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -7,9 +7,7 @@ package dan200.computercraft.shared.computer.terminal;
|
|||||||
import dan200.computercraft.api.lua.LuaValues;
|
import dan200.computercraft.api.lua.LuaValues;
|
||||||
import dan200.computercraft.core.terminal.Terminal;
|
import dan200.computercraft.core.terminal.Terminal;
|
||||||
import dan200.computercraft.test.core.CallCounter;
|
import dan200.computercraft.test.core.CallCounter;
|
||||||
import io.netty.buffer.Unpooled;
|
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
import net.minecraft.network.FriendlyByteBuf;
|
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
import static dan200.computercraft.test.core.terminal.TerminalMatchers.*;
|
import static dan200.computercraft.test.core.terminal.TerminalMatchers.*;
|
||||||
@@ -18,36 +16,6 @@ import static org.hamcrest.Matchers.allOf;
|
|||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
|
||||||
class NetworkedTerminalTest {
|
class NetworkedTerminalTest {
|
||||||
@Test
|
|
||||||
void testPacketBufferRoundtrip() {
|
|
||||||
var writeTerminal = new NetworkedTerminal(2, 1, true);
|
|
||||||
|
|
||||||
blit(writeTerminal, "hi", "11", "ee");
|
|
||||||
writeTerminal.setCursorPos(2, 5);
|
|
||||||
writeTerminal.setTextColour(3);
|
|
||||||
writeTerminal.setBackgroundColour(5);
|
|
||||||
|
|
||||||
var packetBuffer = new FriendlyByteBuf(Unpooled.buffer());
|
|
||||||
writeTerminal.write(packetBuffer);
|
|
||||||
|
|
||||||
var callCounter = new CallCounter();
|
|
||||||
var readTerminal = new NetworkedTerminal(2, 1, true, callCounter);
|
|
||||||
packetBuffer.writeBytes(packetBuffer);
|
|
||||||
readTerminal.read(packetBuffer);
|
|
||||||
|
|
||||||
assertThat(readTerminal, allOf(
|
|
||||||
textMatches(new String[]{ "hi", }),
|
|
||||||
textColourMatches(new String[]{ "11", }),
|
|
||||||
backgroundColourMatches(new String[]{ "ee", })
|
|
||||||
));
|
|
||||||
|
|
||||||
assertEquals(2, readTerminal.getCursorX());
|
|
||||||
assertEquals(5, readTerminal.getCursorY());
|
|
||||||
assertEquals(3, readTerminal.getTextColour());
|
|
||||||
assertEquals(5, readTerminal.getBackgroundColour());
|
|
||||||
callCounter.assertCalledTimes(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void testNbtRoundtrip() {
|
void testNbtRoundtrip() {
|
||||||
var writeTerminal = new NetworkedTerminal(10, 5, true);
|
var writeTerminal = new NetworkedTerminal(10, 5, true);
|
||||||
|
@@ -23,7 +23,7 @@ public class TerminalStateTest {
|
|||||||
var terminal = randomTerminal();
|
var terminal = randomTerminal();
|
||||||
|
|
||||||
var buffer = new FriendlyByteBuf(Unpooled.directBuffer());
|
var buffer = new FriendlyByteBuf(Unpooled.directBuffer());
|
||||||
new TerminalState(terminal).write(buffer);
|
TerminalState.create(terminal).write(buffer);
|
||||||
|
|
||||||
checkEqual(terminal, read(buffer));
|
checkEqual(terminal, read(buffer));
|
||||||
assertEquals(0, buffer.readableBytes());
|
assertEquals(0, buffer.readableBytes());
|
||||||
@@ -37,6 +37,10 @@ public class TerminalStateTest {
|
|||||||
for (var x = 0; x < buffer.length(); x++) buffer.setChar(x, (char) (random.nextInt(26) + 65));
|
for (var x = 0; x < buffer.length(); x++) buffer.setChar(x, (char) (random.nextInt(26) + 65));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
terminal.setTextColour(random.nextInt(16));
|
||||||
|
terminal.setBackgroundColour(random.nextInt(16));
|
||||||
|
terminal.setCursorPos(random.nextInt(terminal.getWidth()), random.nextInt(terminal.getHeight()));
|
||||||
|
|
||||||
return terminal;
|
return terminal;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -46,6 +50,10 @@ public class TerminalStateTest {
|
|||||||
assertEquals(expected.isColour(), actual.isColour(), "isColour must match");
|
assertEquals(expected.isColour(), actual.isColour(), "isColour must match");
|
||||||
assertEquals(expected.getHeight(), actual.getHeight(), "Heights must match");
|
assertEquals(expected.getHeight(), actual.getHeight(), "Heights must match");
|
||||||
assertEquals(expected.getWidth(), actual.getWidth(), "Widths must match");
|
assertEquals(expected.getWidth(), actual.getWidth(), "Widths must match");
|
||||||
|
assertEquals(expected.getTextColour(), actual.getTextColour(), "Text colours must match");
|
||||||
|
assertEquals(expected.getBackgroundColour(), actual.getBackgroundColour(), "Background colours must match");
|
||||||
|
assertEquals(expected.getCursorX(), actual.getCursorX(), "Cursor x must match");
|
||||||
|
assertEquals(expected.getCursorY(), actual.getCursorY(), "Cursor y must match");
|
||||||
|
|
||||||
for (var y = 0; y < expected.getHeight(); y++) {
|
for (var y = 0; y < expected.getHeight(); y++) {
|
||||||
assertEquals(expected.getLine(y).toString(), actual.getLine(y).toString());
|
assertEquals(expected.getLine(y).toString(), actual.getLine(y).toString());
|
||||||
|
Reference in New Issue
Block a user