1
Fork 0
mirror of https://github.com/redstrate/Physis.git synced 2025-04-26 06:07:45 +00:00
physis/src/model_vertex_declarations.rs
Joshua Goins bb7c74fec8 Redesign how IndexFile parsing works, again
Now it reads much more of the index file format, and tries to handle
index1/index2 differences more opaquely. I'm not yet happy with the API, it
needs a bit more work.
2025-03-06 18:52:26 -05:00

160 lines
4.1 KiB
Rust

// SPDX-FileCopyrightText: 2023 Joshua Goins <josh@redstrate.com>
// SPDX-License-Identifier: GPL-3.0-or-later
use crate::model::NUM_VERTICES;
use binrw::{BinRead, BinResult, BinWrite, binrw};
use std::io::SeekFrom;
/// Marker for end of stream (0xFF)
const END_OF_STREAM: u8 = 0xFF;
/// The format of the vertex stream.
#[binrw]
#[brw(repr = u8)]
#[repr(u8)]
#[derive(Copy, Clone, Debug, PartialEq)]
pub enum VertexType {
/// 1 32-bit float
Single1 = 0,
/// 2 32-bit floats
Single2 = 1,
/// 3 32-bit floats
Single3 = 2,
/// 4 32-bit floats
Single4 = 3,
/// 4 bytes
Byte4 = 5,
/// 2 16-bit signed integers
Short2 = 6,
/// 4 16-bit signed integers
Short4 = 7,
/// 4 8-bit floats
ByteFloat4 = 8,
/// Duplicate of Short2?
Short2n = 9,
/// Duplicate of Short4?
Short4n = 10,
/// 2 16-bit floats
Half2 = 13,
/// 4 16-bit floats
Half4 = 14,
/// 2 16-bit unsigned integers
UnsignedShort2 = 16,
/// 4 16-bit unsigned integers
UnsignedShort4 = 17,
}
/// In bytes
pub fn get_vertex_type_size(vertex_type: VertexType) -> usize {
// TODO: Most of these are wrong
match vertex_type {
VertexType::Single1 => 4,
VertexType::Single2 => 8,
VertexType::Single3 => 12,
VertexType::Single4 => 16,
VertexType::Byte4 => 4,
VertexType::Short2 => 4,
VertexType::Short4 => 8,
VertexType::ByteFloat4 => 4,
VertexType::Short2n => 4,
VertexType::Short4n => 4,
VertexType::Half2 => 4,
VertexType::Half4 => 8,
VertexType::UnsignedShort2 => 4,
VertexType::UnsignedShort4 => 8,
}
}
/// What the vertex stream is used for.
#[binrw]
#[brw(repr = u8)]
#[repr(u8)]
#[derive(Copy, Clone, Debug, PartialEq)]
pub enum VertexUsage {
Position = 0,
BlendWeights = 1,
BlendIndices = 2,
Normal = 3,
UV = 4,
Tangent = 5,
BiTangent = 6,
Color = 7,
}
/// Represents an element within a bigger vertex stream.
#[binrw]
#[derive(Copy, Clone, Debug, PartialEq)]
#[allow(dead_code)]
#[repr(C)]
#[brw(little)]
pub struct VertexElement {
pub stream: u8,
pub offset: u8,
pub vertex_type: VertexType,
pub vertex_usage: VertexUsage,
#[brw(pad_after = 3)]
pub usage_index: u8,
}
/// Represents the true size of VertexElement. Always use this value instead of std::mem::size_of.
// 3 extra bytes to account for the padding that doesn't appear in the struct itself
pub const VERTEX_ELEMENT_SIZE: usize = std::mem::size_of::<VertexElement>() + 3;
#[derive(Clone, Debug, PartialEq)]
pub struct VertexDeclaration {
pub elements: Vec<VertexElement>,
}
#[binrw::parser(reader, endian)]
pub(crate) fn vertex_element_parser(count: u16) -> BinResult<Vec<VertexDeclaration>> {
let mut vertex_declarations: Vec<VertexDeclaration> =
vec![VertexDeclaration { elements: vec![] }; count.into()];
for declaration in &mut vertex_declarations {
let mut element = VertexElement::read_options(reader, endian, ())?;
loop {
declaration.elements.push(element);
element = VertexElement::read_options(reader, endian, ())?;
if element.stream == END_OF_STREAM {
break;
}
}
let to_seek = NUM_VERTICES as usize * 8 - (declaration.elements.len() + 1) * 8;
reader.seek(SeekFrom::Current(to_seek as i64))?;
}
Ok(vertex_declarations)
}
#[binrw::writer(writer, endian)]
pub(crate) fn vertex_element_writer(declarations: &Vec<VertexDeclaration>) -> BinResult<()> {
// write vertex declarations
for declaration in declarations {
for element in &declaration.elements {
element.write_options(writer, endian, ())?;
}
VertexElement {
stream: END_OF_STREAM,
offset: 0,
vertex_type: VertexType::Single1,
vertex_usage: VertexUsage::Position,
usage_index: 0,
}
.write_options(writer, endian, ())?;
let to_seek = (NUM_VERTICES as usize - 1 - declaration.elements.len()) * 8;
writer.seek(SeekFrom::Current(to_seek as i64))?;
}
Ok(())
}