fbxcel 0.9.0

Excellent FBX library
Documentation
//! FBX 7.4 footer.

use std::io::{self, Read};

use log::debug;

use crate::{
    low::FbxVersion,
    pull_parser::{
        error::DataError,
        v7400::{FromParser, Parser},
        Error as ParserError, SyntacticPosition, Warning,
    },
};

/// FBX 7.4 footer.
///
/// Data contained in a FBX 7.4 footer is not useful for normal usage.
/// Most of users can safely ignore the footer.
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct FbxFooter {
    /// Unknown (semirandom) 16-bytes data.
    ///
    /// This field is expected to have prescribed upper 4 bits, i.e. the field
    /// is `fx bx ax 0x dx cx dx 6x bx 7x fx 8x 1x fx 2x 7x` if the FBX data is
    /// exported from official SDK.
    ///
    /// Note that third party exporter will use completely random data.
    pub unknown1: [u8; 16],
    /// Padding length.
    ///
    /// Padding is `padding_len` `0`s.
    /// `padding_len >= 0 && padding <= 15` should hold.
    ///
    /// Note that third party exporter will not use correct padding length.
    pub padding_len: u8,
    /// Unknown 4-bytes data.
    ///
    /// This is expected to be `[0u8; 4]`.
    pub unknown2: [u8; 4],
    /// FBX version.
    ///
    /// This is expected to be same as the version in header.
    pub fbx_version: FbxVersion,
    /// Unknown 16-bytes data.
    ///
    /// This is expected to be `[0xf8, 0x5a, 0x8c, 0x6a, 0xde, 0xf5, 0xd9, 0x7e,
    /// 0xec, 0xe9, 0x0c, 0xe3, 0x75, 0x8f, 0x29, 0x0b]`.
    pub unknown3: [u8; 16],
}

impl FromParser for FbxFooter {
    fn read_from_parser<R>(parser: &mut Parser<R>) -> Result<Self, ParserError>
    where
        R: io::Read,
    {
        let start_pos = parser.reader().position();

        // Read unknown field 1.
        let unknown1 = {
            /// Expected upper 4-bits of the unknown field 1.
            const EXPECTED: [u8; 16] = [
                0xf0, 0xb0, 0xa0, 0x00, 0xd0, 0xc0, 0xd0, 0x60, 0xb0, 0x70, 0xf0, 0x80, 0x10, 0xf0,
                0x20, 0x70,
            ];
            let mut buf = [0u8; 16];
            parser.reader().read_exact(&mut buf)?;

            for (byte, expected) in buf.iter().zip(&EXPECTED) {
                if (byte & 0xf0) != *expected {
                    let pos = SyntacticPosition {
                        byte_pos: parser.reader().position() - 16,
                        component_byte_pos: start_pos,
                        node_path: Vec::new(),
                        attribute_index: None,
                    };
                    parser.warn(Warning::UnexpectedFooterFieldValue, pos)?;
                    break;
                }
            }

            buf
        };

        // Read padding, following 144-bytes zeroes, unknown field 2, FBX
        // version, and unknown field 3.
        let (padding_len, unknown2, version, unknown3) = {
            let buf_start_pos = parser.reader().position();

            // Expected padding length.
            let expected_padding_len = (buf_start_pos.wrapping_neg() & 0x0f) as usize;
            debug!(
                "Current position = {}, expected padding length = {}",
                buf_start_pos, expected_padding_len
            );

            /// Buffer length to load footer partially.
            // Padding (min 0) + unknown2 (4) + version (4) + zeroes (120)
            // + unknown3 (16) = 144.
            const BUF_LEN: usize = 144;
            let mut buf = [0u8; BUF_LEN];
            parser.reader().read_exact(&mut buf)?;

            // First, get the beginning position of unknown field 3,
            // because it is expected to be starting with a non-zero byte.
            let unknown3_pos = {
                /// Start offset of search of unknown field 3.
                const SEARCH_OFFSET: usize = BUF_LEN - 16;
                let pos = &buf[SEARCH_OFFSET..]
                    .iter()
                    .position(|&v| v != 0)
                    .ok_or(DataError::BrokenFbxFooter)?;
                SEARCH_OFFSET + pos
            };

            let padding_len = unknown3_pos & 0x0f;
            assert!(padding_len < 16);
            assert_eq!(unknown3_pos, padding_len + 128);
            let padding = &buf[..padding_len];
            let mut unknown2 = [0u8; 4];
            unknown2.copy_from_slice(&buf[padding_len..(padding_len + 4)]);
            let version = u32::from_le_bytes(
                buf[(padding_len + 4)..(padding_len + 8)]
                    .try_into()
                    .expect("the slice must be 4 bytes"),
            );
            let zeroes_120 = &buf[(padding_len + 8)..(padding_len + 128)];
            let unknown3_part = &buf[(padding_len + 128)..];

            // Check that the padding has only zeroes.
            if !padding.iter().all(|&v| v == 0) {
                return Err(DataError::BrokenFbxFooter.into());
            }

            // Check that the unknown field 2 has only zeroes.
            if unknown2 != [0u8; 4] {
                return Err(DataError::BrokenFbxFooter.into());
            }

            // Check that the FBX version is same as the FBX header.
            let version = FbxVersion::new(version);
            if version != parser.fbx_version() {
                // Version mismatch.
                return Err(DataError::BrokenFbxFooter.into());
            }

            // Check that there are 120-bytes zeroes.
            if !zeroes_120.iter().all(|&v| v == 0) {
                return Err(DataError::BrokenFbxFooter.into());
            }

            // Check that the unknown field 3 has expected pattern.
            /// Expected value of unknown field 3.
            const UNKNOWN3_EXPECTED: [u8; 16] = [
                0xf8, 0x5a, 0x8c, 0x6a, 0xde, 0xf5, 0xd9, 0x7e, 0xec, 0xe9, 0x0c, 0xe3, 0x75, 0x8f,
                0x29, 0x0b,
            ];
            let mut unknown3 = [0u8; 16];
            unknown3[0..unknown3_part.len()].copy_from_slice(unknown3_part);
            parser
                .reader()
                .read_exact(&mut unknown3[unknown3_part.len()..])?;
            if unknown3 != UNKNOWN3_EXPECTED {
                return Err(DataError::BrokenFbxFooter.into());
            }

            // If the execution comes here, footer may have no error.
            // Emit warning if necessary.

            // Check if the padding has correct length.
            if padding_len != expected_padding_len {
                let pos = SyntacticPosition {
                    byte_pos: buf_start_pos,
                    component_byte_pos: start_pos,
                    node_path: Vec::new(),
                    attribute_index: None,
                };
                parser.warn(
                    Warning::InvalidFooterPaddingLength(expected_padding_len, padding_len),
                    pos,
                )?;
            }

            (padding_len, unknown2, version, unknown3)
        };

        Ok(Self {
            unknown1,
            padding_len: padding_len as u8,
            unknown2,
            fbx_version: version,
            unknown3,
        })
    }
}