valence/crates/valence_protocol/benches/benches.rs
Ryan Johnson 0960ad7ead
Refactor valence_protocol (#253)
## Description

- Remove duplicate packet definitions by using `Cow`.
- Rename packets to match yarn mappings.
- Remove some top-level re-exports.
- Move every packet into its own module for consistency.
- Move packet-specific types from the `types` module into the
appropriate packet module.
- Remove internal use of `EncodePacket`/`DecodePacket` derives and move
packet identification to `packet_group`. This can be done because there
are no duplicate packets anymore.
- Simplify some events.

In a future PR I plan to clean things up further by properly bounding
packet data (to prevent DoS exploits) and fixing any remaining
inconsistencies with the game's packet definitions.

## Test Plan

Behavior of `valence_protocol` should be the same.

Steps:
1. Use the packet inspector against the vanilla server to ensure packet
behavior has not changed.
2. Run the examples.
3. Run `valence_stresser`.
2023-02-23 22:16:22 -08:00

404 lines
11 KiB
Rust

use std::borrow::Cow;
use std::time::Duration;
use criterion::{black_box, criterion_group, criterion_main, Criterion};
use rand::Rng;
use valence_nbt::{compound, List};
use valence_protocol::array::LengthPrefixedArray;
use valence_protocol::block::{BlockKind, BlockState, PropName, PropValue};
use valence_protocol::byte_angle::ByteAngle;
use valence_protocol::codec::{
encode_packet, encode_packet_compressed, PacketDecoder, PacketEncoder,
};
use valence_protocol::item::ItemKind;
use valence_protocol::packet::s2c::play::{ChunkDataS2c, EntitySpawnS2c, PlayerListHeaderS2c};
use valence_protocol::text::{Color, TextFormat};
use valence_protocol::var_int::VarInt;
use valence_protocol::var_long::VarLong;
use valence_protocol::{Decode, Encode};
criterion_group! {
name = benches;
config = Criterion::default()
.measurement_time(Duration::from_secs(5)).confidence_level(0.99);
targets = blocks, packets, var_int, var_long, decode_array
}
criterion_main!(benches);
fn blocks(c: &mut Criterion) {
let states = BlockKind::ALL.map(BlockKind::to_state);
c.bench_function("BlockState::from_kind", |b| {
b.iter(|| {
for kind in black_box(BlockKind::ALL) {
black_box(BlockState::from_kind(kind));
}
});
});
c.bench_function("BlockState::to_kind", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.to_kind());
}
});
});
c.bench_function("BlockState::get", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.get(PropName::Note));
}
});
});
c.bench_function("BlockState::set", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.set(PropName::Note, PropValue::Didgeridoo));
}
});
});
c.bench_function("BlockState::is_liquid", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.is_liquid());
}
});
});
c.bench_function("BlockState::is_opaque", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.is_opaque());
}
})
});
c.bench_function("BlockState::is_replaceable", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.is_replaceable());
}
})
});
c.bench_function("BlockState::luminance", |b| {
b.iter(|| {
for state in black_box(states) {
black_box(state.luminance());
}
})
});
c.bench_function("BlockKind::to_item_kind", |b| {
b.iter(|| {
for kind in black_box(BlockKind::ALL) {
black_box(kind.to_item_kind());
}
});
});
c.bench_function("BlockKind::from_item_kind", |b| {
b.iter(|| {
for kind in black_box(ItemKind::ALL) {
black_box(BlockKind::from_item_kind(kind));
}
});
});
}
fn packets(c: &mut Criterion) {
let mut encoder = PacketEncoder::new();
const BLOCKS_AND_BIOMES: [u8; 2000] = [0x80; 2000];
const SKY_LIGHT_ARRAYS: [LengthPrefixedArray<u8, 2048>; 26] =
[LengthPrefixedArray([0xff; 2048]); 26];
let chunk_data_packet = ChunkDataS2c {
chunk_x: 123,
chunk_z: 456,
heightmaps: Cow::Owned(compound! {
"MOTION_BLOCKING" => List::Long(vec![123; 256]),
}),
blocks_and_biomes: BLOCKS_AND_BIOMES.as_slice(),
block_entities: Cow::Borrowed(&[]),
trust_edges: false,
sky_light_mask: Cow::Borrowed(&[]),
block_light_mask: Cow::Borrowed(&[]),
empty_sky_light_mask: Cow::Borrowed(&[]),
empty_block_light_mask: Cow::Borrowed(&[]),
sky_light_arrays: Cow::Borrowed(SKY_LIGHT_ARRAYS.as_slice()),
block_light_arrays: Cow::Borrowed(&[]),
};
let player_list_header_packet = PlayerListHeaderS2c {
header: ("this".italic() + " is the " + "header".bold().color(Color::RED)).into(),
footer: ("this".italic()
+ " is the "
+ "footer".bold().color(Color::BLUE)
+ ". I am appending some extra text so that the packet goes over the compression \
threshold.")
.into(),
};
let spawn_entity_packet = EntitySpawnS2c {
entity_id: VarInt(1234),
object_uuid: Default::default(),
kind: VarInt(5),
position: [123.0, 456.0, 789.0],
pitch: ByteAngle(200),
yaw: ByteAngle(100),
head_yaw: ByteAngle(50),
data: VarInt(i32::MIN),
velocity: [12, 34, 56],
};
c.bench_function("encode_chunk_data", |b| {
b.iter(|| {
let encoder = black_box(&mut encoder);
encoder.clear();
encoder.append_packet(&chunk_data_packet).unwrap();
black_box(encoder);
});
});
c.bench_function("encode_player_list_header", |b| {
b.iter(|| {
let encoder = black_box(&mut encoder);
encoder.clear();
encoder.append_packet(&player_list_header_packet).unwrap();
black_box(encoder);
});
});
c.bench_function("encode_spawn_entity", |b| {
b.iter(|| {
let encoder = black_box(&mut encoder);
encoder.clear();
encoder.append_packet(&spawn_entity_packet).unwrap();
black_box(encoder);
});
});
encoder.set_compression(Some(256));
c.bench_function("encode_chunk_data_compressed", |b| {
b.iter(|| {
let encoder = black_box(&mut encoder);
encoder.clear();
encoder.append_packet(&chunk_data_packet).unwrap();
black_box(encoder);
});
});
c.bench_function("encode_player_list_header_compressed", |b| {
b.iter(|| {
let encoder = black_box(&mut encoder);
encoder.clear();
encoder.append_packet(&player_list_header_packet).unwrap();
black_box(encoder);
});
});
c.bench_function("encode_spawn_entity_compressed", |b| {
b.iter(|| {
let encoder = black_box(&mut encoder);
encoder.clear();
encoder.append_packet(&spawn_entity_packet).unwrap();
black_box(encoder);
});
});
let mut decoder = PacketDecoder::new();
let mut packet_buf = vec![];
encode_packet(&mut packet_buf, &chunk_data_packet).unwrap();
c.bench_function("decode_chunk_data", |b| {
b.iter(|| {
let decoder = black_box(&mut decoder);
decoder.queue_slice(&packet_buf);
decoder.try_next_packet::<ChunkDataS2c>().unwrap();
black_box(decoder);
});
});
packet_buf.clear();
encode_packet(&mut packet_buf, &player_list_header_packet).unwrap();
c.bench_function("decode_player_list_header", |b| {
b.iter(|| {
let decoder = black_box(&mut decoder);
decoder.queue_slice(&packet_buf);
decoder.try_next_packet::<PlayerListHeaderS2c>().unwrap();
black_box(decoder);
});
});
packet_buf.clear();
encode_packet(&mut packet_buf, &spawn_entity_packet).unwrap();
c.bench_function("decode_entity_spawn", |b| {
b.iter(|| {
let decoder = black_box(&mut decoder);
decoder.queue_slice(&packet_buf);
decoder.try_next_packet::<EntitySpawnS2c>().unwrap();
black_box(decoder);
});
});
decoder.set_compression(true);
let mut scratch = vec![];
packet_buf.clear();
encode_packet_compressed(&mut packet_buf, &chunk_data_packet, 256, &mut scratch).unwrap();
c.bench_function("decode_chunk_data_compressed", |b| {
b.iter(|| {
let decoder = black_box(&mut decoder);
decoder.queue_slice(&packet_buf);
decoder.try_next_packet::<ChunkDataS2c>().unwrap();
black_box(decoder);
});
});
packet_buf.clear();
encode_packet_compressed(
&mut packet_buf,
&player_list_header_packet,
256,
&mut scratch,
)
.unwrap();
c.bench_function("decode_player_list_header_compressed", |b| {
b.iter(|| {
let decoder = black_box(&mut decoder);
decoder.queue_slice(&packet_buf);
decoder.try_next_packet::<PlayerListHeaderS2c>().unwrap();
black_box(decoder);
});
});
packet_buf.clear();
encode_packet_compressed(&mut packet_buf, &spawn_entity_packet, 256, &mut scratch).unwrap();
c.bench_function("decode_spawn_entity_compressed", |b| {
b.iter(|| {
let decoder = black_box(&mut decoder);
decoder.queue_slice(&packet_buf);
decoder.try_next_packet::<EntitySpawnS2c>().unwrap();
black_box(decoder);
});
});
}
fn var_int(c: &mut Criterion) {
let mut rng = rand::thread_rng();
c.bench_function("VarInt::encode", |b| {
b.iter_with_setup(
|| rng.gen(),
|i| {
let i: i32 = black_box(i);
let mut buf = [0; VarInt::MAX_SIZE];
let _ = black_box(VarInt(i).encode(buf.as_mut_slice()));
},
);
});
c.bench_function("VarInt::decode", |b| {
b.iter_with_setup(
|| {
let mut buf = [0; VarInt::MAX_SIZE];
VarInt(rng.gen()).encode(buf.as_mut_slice()).unwrap();
buf
},
|buf| {
let mut r = black_box(buf.as_slice());
let _ = black_box(VarInt::decode(&mut r));
},
)
});
}
fn var_long(c: &mut Criterion) {
let mut rng = rand::thread_rng();
c.bench_function("VarLong::encode", |b| {
b.iter_with_setup(
|| rng.gen(),
|i| {
let i: i64 = black_box(i);
let mut buf = [0; VarLong::MAX_SIZE];
let _ = black_box(VarLong(i).encode(buf.as_mut_slice()));
},
);
});
c.bench_function("VarLong::decode", |b| {
b.iter_with_setup(
|| {
let mut buf = [0; VarLong::MAX_SIZE];
VarLong(rng.gen()).encode(buf.as_mut_slice()).unwrap();
buf
},
|buf| {
let mut r = black_box(buf.as_slice());
let _ = black_box(VarLong::decode(&mut r));
},
)
});
}
fn decode_array(c: &mut Criterion) {
let floats = [123.0, 456.0, 789.0];
let mut buf = [0u8; 24];
floats.encode(buf.as_mut_slice()).unwrap();
c.bench_function("<[f64; 3]>::decode", |b| {
b.iter(|| {
let mut r = black_box(buf.as_slice());
let _ = black_box(<[f64; 3]>::decode(&mut r));
});
});
let bytes = [42; 4096];
c.bench_function("<[u8; 4096]>::decode", |b| {
b.iter(|| {
let mut r = black_box(bytes.as_slice());
let _ = black_box(<[u8; 4096]>::decode(&mut r));
})
});
}