2021-07-24 21:11:47 -07:00

146 lines
5.7 KiB
C++

//===== Copyright c 1996-2008, Valve Corporation, All rights reserved. ======//
//
// Purpose: Describes our resource format. All resource files have a simplistic
// dictionary of resource "blocks", which can be looked up using a block type id.
// Each resource block type is expected to be associated with a well defined
// structure. The macro DEFINE_RESOURCE_BLOCK_TYPE is used to create this
// association.
//
// The current design choice is that we expect files using this resource format
// to be small. Large-scale files, like sound or texture bits, are expected to
// exist in a file parallel to the file containing these resource blocks owing
// to issues of alignment or streaming. We therefore expect users of files
// containing the data described in this header to load the entire file in
// so that all blocks are in memory.
//
// $NoKeywords: $
//===========================================================================//
#ifndef RESOURCEFILE_H
#define RESOURCEFILE_H
#pragma once
#include "resourcefile/schema.h"
#include "tier0/platform.h"
#include "resourcefile/resourcestream.h"
#include "datamap.h"
//-----------------------------------------------------------------------------
//
// On-disk structures related to the resource file format
//
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// Resource block types
//-----------------------------------------------------------------------------
typedef uint32 ResourceBlockId_t;
//-----------------------------------------------------------------------------
// A block of resource data
//-----------------------------------------------------------------------------
struct ResourceBlockEntry_t
{
DECLARE_BYTESWAP_DATADESC();
ResourceBlockId_t m_nBlockType;
CResourcePointer<void> m_pBlockData;
};
//-----------------------------------------------------------------------------
// Structure of resource file header
//-----------------------------------------------------------------------------
enum ResourceFileHeaderVersion_t
{
RESOURCE_FILE_HEADER_VERSION = 1,
};
struct ResourceFileHeader_t
{
DECLARE_BYTESWAP_DATADESC();
uint32 m_nVersion; // see ResourceFileHeaderVersion_t
uint32 m_nSizeInBytes; // Size in bytes of entire file
CResourceArray< ResourceBlockEntry_t > m_ResourceBlocks;
};
//-----------------------------------------------------------------------------
//
// Methods used to define resource blocks/read them from files
//
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// Resource block IDs
//-----------------------------------------------------------------------------
#define RSRC_BYTE_POS( byteVal, shft ) ResourceBlockId_t( uint32(uint8(byteVal)) << uint8(shft * 8) )
#if !defined( PLATFORM_X360 )
#define MK_RSRC_BLOCK_ID(a, b, c, d) ResourceBlockId_t( RSRC_BYTE_POS(a, 0) | RSRC_BYTE_POS(b, 1) | RSRC_BYTE_POS(c, 2) | RSRC_BYTE_POS(d, 3) )
#else
#define MK_RSRC_BLOCK_ID(a, b, c, d) ResourceBlockId_t( RSRC_BYTE_POS(a, 3) | RSRC_BYTE_POS(b, 2) | RSRC_BYTE_POS(c, 1) | RSRC_BYTE_POS(d, 0) )
#endif
#define RESOURCE_BLOCK_ID_INVALID 0xFFFFFFFF
//-----------------------------------------------------------------------------
// Helpers used to define resource block IDs + associated structs
//-----------------------------------------------------------------------------
template< class T >
struct ResourceBlockIdSelector_t
{
enum { RESOURCE_BLOCK_ID = RESOURCE_BLOCK_ID_INVALID };
};
#define DEFINE_RESOURCE_BLOCK_TYPE( _class, _ida, _idb, _idc, _idd ) \
template <> struct ResourceBlockIdSelector_t< _class > { enum { RESOURCE_BLOCK_ID = MK_RSRC_BLOCK_ID( _ida, _idb, _idc, _idd ) }; };
//-----------------------------------------------------------------------------
// Does this resource file contain a particular block?
//-----------------------------------------------------------------------------
bool Resource_IsBlockDefined( const ResourceFileHeader_t *pHeader, ResourceBlockId_t id );
//-----------------------------------------------------------------------------
// Gets the data associated with a particular data block
//-----------------------------------------------------------------------------
const void *Resource_GetBlock( const ResourceFileHeader_t *pHeader, ResourceBlockId_t id );
template< class T > inline const T* Resource_GetBlock( const ResourceFileHeader_t *pHeader )
{
return (const T*)Resource_GetBlock( pHeader, ResourceBlockIdSelector_t< T >::RESOURCE_BLOCK_ID );
}
//-----------------------------------------------------------------------------
// Helper methods to write block information
//-----------------------------------------------------------------------------
inline ResourceFileHeader_t *Resource_AllocateHeader( CResourceStream *pStream, int nBlockCount )
{
ResourceFileHeader_t *pHeader = pStream->Allocate< ResourceFileHeader_t >();
pHeader->m_nVersion = RESOURCE_FILE_HEADER_VERSION;
pHeader->m_ResourceBlocks = pStream->Allocate< ResourceBlockEntry_t >( nBlockCount );
return pHeader;
}
//-----------------------------------------------------------------------------
// Helper methods to write block information
//-----------------------------------------------------------------------------
template< class T > inline T* Resource_AllocateBlock( CResourceStream *pStream, ResourceFileHeader_t *pHeader, int nBlockIndex )
{
pHeader->m_ResourceBlocks[nBlockIndex].m_nBlockType = ResourceBlockIdSelector_t< T >::RESOURCE_BLOCK_ID;
T *pBlockData = pStream->Allocate< T >( 1 );
pHeader->m_ResourceBlocks[nBlockIndex].m_pBlockData = pBlockData;
return pBlockData;
}
#endif // RESOURCEFILE_H