use core::fmt;
use crate::bytes::{Cursor, OutOfBounds};
use crate::Rgb6;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub struct Voxel {
pub col: u32,
pub z: u16,
pub vis: u8,
pub dir: u8,
}
#[derive(Debug, Clone)]
pub struct Kv6 {
pub xsiz: u32,
pub ysiz: u32,
pub zsiz: u32,
pub xpiv: f32,
pub ypiv: f32,
pub zpiv: f32,
pub voxels: Vec<Voxel>,
pub xlen: Vec<u32>,
pub ylen: Vec<Vec<u16>>,
pub palette: Option<[Rgb6; 256]>,
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum ParseError {
TooSmall { got: usize },
BadMagic { got: [u8; 4] },
Truncated { at: usize, need: usize },
}
impl fmt::Display for ParseError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match *self {
Self::TooSmall { got } => write!(
f,
"kv6 file too small ({got} bytes; need at least 32 byte header)"
),
Self::BadMagic { got } => write!(
f,
"kv6 bad magic: got [{:#04x},{:#04x},{:#04x},{:#04x}], expected b\"Kvxl\"",
got[0], got[1], got[2], got[3]
),
Self::Truncated { at, need } => {
write!(f, "kv6 truncated: need {need} bytes at offset {at}")
}
}
}
}
impl std::error::Error for ParseError {}
impl From<OutOfBounds> for ParseError {
fn from(e: OutOfBounds) -> Self {
Self::Truncated {
at: e.at,
need: e.need,
}
}
}
const HEADER_LEN: usize = 32;
const MAGIC: &[u8; 4] = b"Kvxl";
const PALETTE_MAGIC: &[u8; 4] = b"SPal";
const PALETTE_LEN: usize = 768;
pub fn parse(bytes: &[u8]) -> Result<Kv6, ParseError> {
if bytes.len() < HEADER_LEN {
return Err(ParseError::TooSmall { got: bytes.len() });
}
let mut cur = Cursor::new(bytes);
let magic = cur.read_bytes(4)?;
if magic != MAGIC {
return Err(ParseError::BadMagic {
got: [magic[0], magic[1], magic[2], magic[3]],
});
}
let xsiz = cur.read_u32()?;
let ysiz = cur.read_u32()?;
let zsiz = cur.read_u32()?;
let xpiv = cur.read_f32()?;
let ypiv = cur.read_f32()?;
let zpiv = cur.read_f32()?;
let numvoxs = cur.read_u32()?;
let mut voxels = Vec::with_capacity(numvoxs as usize);
for _ in 0..numvoxs {
let col = cur.read_u32()?;
let z = cur.read_u16()?;
let vis = cur.read_u8()?;
let dir = cur.read_u8()?;
voxels.push(Voxel { col, z, vis, dir });
}
let mut xlen = Vec::with_capacity(xsiz as usize);
for _ in 0..xsiz {
xlen.push(cur.read_u32()?);
}
let mut ylen = Vec::with_capacity(xsiz as usize);
for _ in 0..xsiz {
let mut row = Vec::with_capacity(ysiz as usize);
for _ in 0..ysiz {
row.push(cur.read_u16()?);
}
ylen.push(row);
}
let palette =
if cur.remaining() >= 4 + PALETTE_LEN && cur.peek(4) == Some(PALETTE_MAGIC.as_slice()) {
cur.read_bytes(4)?;
let mut pal = [Rgb6::default(); 256];
for entry in &mut pal {
entry.r = cur.read_u8()?;
entry.g = cur.read_u8()?;
entry.b = cur.read_u8()?;
}
Some(pal)
} else {
None
};
Ok(Kv6 {
xsiz,
ysiz,
zsiz,
xpiv,
ypiv,
zpiv,
voxels,
xlen,
ylen,
palette,
})
}
#[must_use]
pub fn serialize(kv6: &Kv6) -> Vec<u8> {
let pal_bytes = if kv6.palette.is_some() {
4 + PALETTE_LEN
} else {
0
};
let body_bytes = kv6.voxels.len() * 8
+ kv6.xlen.len() * 4
+ kv6.ylen.iter().map(|row| row.len() * 2).sum::<usize>();
let mut out = Vec::with_capacity(HEADER_LEN + body_bytes + pal_bytes);
out.extend_from_slice(MAGIC);
out.extend_from_slice(&kv6.xsiz.to_le_bytes());
out.extend_from_slice(&kv6.ysiz.to_le_bytes());
out.extend_from_slice(&kv6.zsiz.to_le_bytes());
out.extend_from_slice(&kv6.xpiv.to_le_bytes());
out.extend_from_slice(&kv6.ypiv.to_le_bytes());
out.extend_from_slice(&kv6.zpiv.to_le_bytes());
let numvoxs =
u32::try_from(kv6.voxels.len()).expect("kv6 numvoxs must fit in u32 (file format limit)");
out.extend_from_slice(&numvoxs.to_le_bytes());
for v in &kv6.voxels {
out.extend_from_slice(&v.col.to_le_bytes());
out.extend_from_slice(&v.z.to_le_bytes());
out.push(v.vis);
out.push(v.dir);
}
for v in &kv6.xlen {
out.extend_from_slice(&v.to_le_bytes());
}
for row in &kv6.ylen {
for v in row {
out.extend_from_slice(&v.to_le_bytes());
}
}
if let Some(pal) = &kv6.palette {
out.extend_from_slice(PALETTE_MAGIC);
for e in pal {
out.push(e.r);
out.push(e.g);
out.push(e.b);
}
}
out
}
#[cfg(test)]
mod tests {
use super::*;
const COCO_KV6: &[u8] = include_bytes!("../../../assets/coco.kv6");
#[test]
fn parse_coco_header() {
let kv6 = parse(COCO_KV6).expect("parse coco.kv6");
assert_eq!(kv6.xsiz, 9);
assert_eq!(kv6.ysiz, 11);
assert_eq!(kv6.zsiz, 9);
assert!((kv6.xpiv - 2.0).abs() < f32::EPSILON);
assert!((kv6.ypiv - 3.0).abs() < f32::EPSILON);
assert!((kv6.zpiv - 9.0).abs() < f32::EPSILON);
assert_eq!(kv6.voxels.len(), 148);
}
#[test]
fn coco_voxel_counts_consistent() {
let kv6 = parse(COCO_KV6).expect("parse coco.kv6");
assert_eq!(kv6.xlen.len(), kv6.xsiz as usize);
assert_eq!(kv6.ylen.len(), kv6.xsiz as usize);
for row in &kv6.ylen {
assert_eq!(row.len(), kv6.ysiz as usize);
}
let xlen_sum: u64 = kv6.xlen.iter().map(|&n| u64::from(n)).sum();
let ylen_sum: u64 = kv6
.ylen
.iter()
.flat_map(|row| row.iter().map(|&n| u64::from(n)))
.sum();
let nv = kv6.voxels.len() as u64;
assert_eq!(xlen_sum, nv);
assert_eq!(ylen_sum, nv);
}
#[test]
fn coco_palette_present_and_matches_kvx() {
let kv6 = parse(COCO_KV6).expect("parse coco.kv6");
let pal = kv6.palette.as_ref().expect("SPal trailer present");
assert_eq!((pal[0].r, pal[0].g, pal[0].b), (0x3f, 0x19, 0x19));
}
#[test]
fn coco_first_voxel_packed_colour() {
let kv6 = parse(COCO_KV6).expect("parse coco.kv6");
let v0 = kv6.voxels[0];
assert_eq!(v0.col, 0x80fc_a460);
assert_eq!(v0.col & 0x8000_0000, 0x8000_0000);
}
#[test]
fn coco_roundtrips_byte_equal() {
let kv6 = parse(COCO_KV6).expect("parse coco.kv6");
let out = serialize(&kv6);
assert_eq!(out.len(), COCO_KV6.len(), "length differs");
assert_eq!(out.as_slice(), COCO_KV6, "byte content differs");
}
#[test]
fn parse_truncated_header_fails() {
let r = parse(&[0u8; 16]);
assert!(matches!(r, Err(ParseError::TooSmall { .. })));
}
#[test]
fn parse_bad_magic_fails() {
let mut bad = COCO_KV6.to_vec();
bad[0] = b'X';
let r = parse(&bad);
assert!(matches!(r, Err(ParseError::BadMagic { .. })));
}
}