/* =========================================================================== Doom 3 BFG Edition GPL Source Code Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company. This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code"). Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Doom 3 BFG Edition Source Code. If not, see . In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below. If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA. =========================================================================== */ #ifndef __SWAP_H__ #define __SWAP_H__ /* ================================================================================================ Contains the Swap class, for CrossPlatform endian conversion. works ================================================================================================ */ /* ======================== IsPointer ======================== */ template< typename type > bool IsPointer( type ) { return false; } /* ======================== IsPointer ======================== */ template< typename type > bool IsPointer( type * ) { return true; } /* ================================================ The *Swap* static template class, idSwap, is used by the SwapClass template class for performing EndianSwapping. ================================================ */ class idSwap { public: //#define SwapBytes( x, y ) (x) ^= (y) ^= (x) ^= (y) #define SwapBytes( x, y ) { byte t = (x); (x) = (y); (y) = t; } template static void Little( type &c ) { // byte swapping pointers is pointless because we should never store pointers on disk assert( !IsPointer( c ) ); } template static void Big( type &c ) { // byte swapping pointers is pointless because we should never store pointers on disk assert( !IsPointer( c ) ); if ( sizeof( type ) == 1 ) { } else if ( sizeof( type ) == 2 ) { byte *b = (byte *)&c; SwapBytes( b[0], b[1] ); } else if ( sizeof( type ) == 4 ) { byte *b = (byte *)&c; SwapBytes( b[0], b[3] ); SwapBytes( b[1], b[2] ); } else if ( sizeof( type ) == 8 ) { byte * b = (byte *)&c; SwapBytes( b[0], b[7] ); SwapBytes( b[1], b[6]); SwapBytes( b[2], b[5] ); SwapBytes( b[3], b[4] ); } else { assert( false ); } } template static void LittleArray( type *c, int count ) { } template static void BigArray( type *c, int count ) { for ( int i = 0; i < count; i++ ) { Big( c[i] ); } } static void SixtetsForInt( byte *out, int src ) { byte *b = (byte *)&src; out[0] = ( b[0] & 0xfc ) >> 2; out[1] = ( ( b[0] & 0x3 ) << 4 ) + ( ( b[1] & 0xf0 ) >> 4 ); out[2] = ( ( b[1] & 0xf ) << 2 ) + ( ( b[2] & 0xc0 ) >> 6 ); out[3] = b[2] & 0x3f; } static int IntForSixtets( byte *in ) { int ret = 0; byte *b = (byte *)&ret; b[0] |= in[0] << 2; b[0] |= ( in[1] & 0x30 ) >> 4; b[1] |= ( in[1] & 0xf ) << 4; b[1] |= ( in[2] & 0x3c ) >> 2; b[2] |= ( in[2] & 0x3 ) << 6; b[2] |= in[3]; return ret; } public: // specializations #ifndef ID_SWAP_LITE // avoid dependency avalanche for SPU code #define SWAP_VECTOR( x ) \ static void Little( x &c ) { LittleArray( c.ToFloatPtr(), c.GetDimension() ); } \ static void Big( x &c ) { BigArray( c.ToFloatPtr(), c.GetDimension() ); } SWAP_VECTOR( idVec2 ); SWAP_VECTOR( idVec3 ); SWAP_VECTOR( idVec4 ); SWAP_VECTOR( idVec5 ); SWAP_VECTOR( idVec6 ); SWAP_VECTOR( idMat2 ); SWAP_VECTOR( idMat3 ); SWAP_VECTOR( idMat4 ); SWAP_VECTOR( idMat5 ); SWAP_VECTOR( idMat6 ); SWAP_VECTOR( idPlane ); SWAP_VECTOR( idQuat ); SWAP_VECTOR( idCQuat ); SWAP_VECTOR( idAngles ); SWAP_VECTOR( idBounds ); static void Little( idDrawVert &v ) { Little( v.xyz ); LittleArray( v.st, 2 ); LittleArray( v.normal, 4 ); LittleArray( v.tangent, 4 ); LittleArray( v.color, 4 ); } static void Big( idDrawVert &v ) { Big( v.xyz ); BigArray( v.st, 2 ); BigArray( v.normal, 4 ); BigArray( v.tangent, 4 ); BigArray( v.color, 4 ); } #endif }; /* ================================================ idSwapClass is a template class for performing EndianSwapping. ================================================ */ template class idSwapClass { public: idSwapClass() { #ifdef _DEBUG size = 0; #endif } ~idSwapClass() { #ifdef _DEBUG assert( size == sizeof( classType ) ); #endif } template void Little( type &c ) { idSwap::Little( c ); #ifdef _DEBUG size += sizeof( type ); #endif } template void Big( type &c ) { idSwap::Big( c ); #ifdef _DEBUG size += sizeof( type ); #endif } template void LittleArray( type *c, int count ) { idSwap::LittleArray( c, count ); #ifdef _DEBUG size += count * sizeof( type ); #endif } template void BigArray( type *c, int count ) { idSwap::BigArray( c, count ); #ifdef _DEBUG size += count * sizeof( type ); #endif } #ifdef _DEBUG private: int size; #endif }; #define BIG32(v) ((((uint32)(v)) >> 24) | (((uint32)(v) & 0x00FF0000) >> 8) | (((uint32)(v) & 0x0000FF00) << 8) | ((uint32)(v) << 24)) #define BIG16(v) ((((uint16)(v)) >> 8) | ((uint16)(v) << 8)) #endif // !__SWAP_H__