1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
// SPDX-FileCopyrightText: 2023 Joshua Goins <josh@redstrate.com>
// SPDX-License-Identifier: GPL-3.0-or-later
use std::io::{Cursor, Seek, SeekFrom};
use crate::ByteSpan;
use binrw::BinRead;
use binrw::binrw;
/// A set of scaling parameters for a race.
#[binrw]
#[br(little)]
#[repr(C)]
#[derive(Debug, Clone, Copy)]
pub struct RacialScalingParameters {
/// Male minimum height.
pub male_min_size: f32,
/// Male maximum height.
pub male_max_size: f32,
/// Male minimum tail size.
pub male_min_tail: f32,
/// Male maximum tail size.
pub male_max_tail: f32,
/// Female minimum height.
pub female_min_size: f32,
/// Female maximum height.
pub female_max_size: f32,
/// Female minimum tail size.
pub female_min_tail: f32,
/// Female maximum tail size.
pub female_max_tail: f32,
/// Minimum bust size on the X-axis.
pub bust_min_x: f32,
/// Minimum bust size on the Y-axis.
pub bust_min_y: f32,
/// Minimum bust size on the Z-axis.
pub bust_min_z: f32,
/// Maximum bust size on the X-axis.
pub bust_max_x: f32,
/// Maximum bust size on the Y-axis.
pub bust_max_y: f32,
/// Maximum bust size on the Z-axis.
pub bust_max_z: f32,
}
/// Character multiplier make file, usually with the `.cmp` file extension.
///
/// This is used to determine various scaling limits for height, and so on.
#[derive(Debug)]
pub struct CMP {
/// The racial scaling parameters
pub parameters: Vec<RacialScalingParameters>,
}
impl CMP {
/// Read an existing file.
pub fn from_existing(buffer: ByteSpan) -> Option<CMP> {
let mut cursor = Cursor::new(buffer);
cursor.seek(SeekFrom::Start(0x2a800)).ok()?;
let rem = buffer.len() - cursor.position() as usize;
let entries = rem / std::mem::size_of::<RacialScalingParameters>();
let mut parameters = vec![];
for _ in 0..entries {
parameters.push(RacialScalingParameters::read(&mut cursor).ok()?);
}
Some(CMP { parameters })
}
}
#[cfg(test)]
mod tests {
use std::fs::read;
use std::path::PathBuf;
use super::*;
#[test]
fn test_invalid() {
let mut d = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
d.push("resources/tests");
d.push("random");
// Feeding it invalid data should not panic
CMP::from_existing(&read(d).unwrap());
}
}