Bochs/bochs/misc/bswap.h

192 lines
5.5 KiB
C

/////////////////////////////////////////////////////////////////////////
// $Id$
/////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 2012-2021 The Bochs Project
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later version.
//
// This library 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
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
/////////////////////////////////////////////////////////////////////////
#ifndef BX_BSWAP_H
#define BX_BSWAP_H
BX_CPP_INLINE Bit16u bx_bswap16(Bit16u val16)
{
return (val16<<8) | (val16>>8);
}
#if !defined(__MORPHOS__)
#if BX_HAVE___BUILTIN_BSWAP32
#define bx_bswap32 __builtin_bswap32
#else
BX_CPP_INLINE Bit32u bx_bswap32(Bit32u val32)
{
val32 = ((val32<<8) & 0xFF00FF00) | ((val32>>8) & 0x00FF00FF);
return (val32<<16) | (val32>>16);
}
#endif
#if BX_HAVE___BUILTIN_BSWAP64
#define bx_bswap64 __builtin_bswap64
#else
BX_CPP_INLINE Bit64u bx_bswap64(Bit64u val64)
{
Bit32u lo = bx_bswap32((Bit32u)(val64 >> 32));
Bit32u hi = bx_bswap32((Bit32u)(val64 & 0xFFFFFFFF));
return ((Bit64u)hi << 32) | (Bit64u)lo;
}
#endif
#endif // !MorphOS
// These are some convenience macros which abstract out accesses between
// a variable in native byte ordering to/from guest (x86) memory, which is
// always in little endian format. You must deal with alignment (if your
// system cares) and endian rearranging. Don't assume anything. You could
// put some platform specific asm() statements here, to make use of native
// instructions to help perform these operations more efficiently than C++.
#ifdef BX_LITTLE_ENDIAN
BX_CPP_INLINE void WriteHostWordToLittleEndian(Bit16u *hostPtr, Bit16u nativeVar16)
{
*(hostPtr) = nativeVar16;
}
BX_CPP_INLINE void WriteHostDWordToLittleEndian(Bit32u *hostPtr, Bit32u nativeVar32)
{
*(hostPtr) = nativeVar32;
}
BX_CPP_INLINE void WriteHostQWordToLittleEndian(Bit64u *hostPtr, Bit64u nativeVar64)
{
#ifdef ANDROID
// Resolve problems with unaligned access
((Bit8u *)(hostPtr))[0] = (Bit8u) (nativeVar64);
((Bit8u *)(hostPtr))[1] = (Bit8u) ((nativeVar64)>>8);
((Bit8u *)(hostPtr))[2] = (Bit8u) ((nativeVar64)>>16);
((Bit8u *)(hostPtr))[3] = (Bit8u) ((nativeVar64)>>24);
((Bit8u *)(hostPtr))[4] = (Bit8u) ((nativeVar64)>>32);
((Bit8u *)(hostPtr))[5] = (Bit8u) ((nativeVar64)>>40);
((Bit8u *)(hostPtr))[6] = (Bit8u) ((nativeVar64)>>48);
((Bit8u *)(hostPtr))[7] = (Bit8u) ((nativeVar64)>>56);
#else
*(hostPtr) = nativeVar64;
#endif
}
BX_CPP_INLINE Bit16u ReadHostWordFromLittleEndian(Bit16u *hostPtr)
{
return *(hostPtr);
}
BX_CPP_INLINE Bit32u ReadHostDWordFromLittleEndian(Bit32u *hostPtr)
{
return *(hostPtr);
}
BX_CPP_INLINE Bit64u ReadHostQWordFromLittleEndian(Bit64u *hostPtr)
{
#ifdef ANDROID
// Resolve problems with unaligned access
Bit64u nativeVar64 = ((Bit64u) ((Bit8u *)(hostPtr))[0]) |
(((Bit64u) ((Bit8u *)(hostPtr))[1])<<8) |
(((Bit64u) ((Bit8u *)(hostPtr))[2])<<16) |
(((Bit64u) ((Bit8u *)(hostPtr))[3])<<24) |
(((Bit64u) ((Bit8u *)(hostPtr))[4])<<32) |
(((Bit64u) ((Bit8u *)(hostPtr))[5])<<40) |
(((Bit64u) ((Bit8u *)(hostPtr))[6])<<48) |
(((Bit64u) ((Bit8u *)(hostPtr))[7])<<56);
return nativeVar64;
#else
return *(hostPtr);
#endif
}
#else // !BX_LITTLE_ENDIAN
#ifdef __MORPHOS__
#define bx_bswap16 bx_ppc_bswap16
#define bx_bswap32 bx_ppc_bswap32
#define bx_bswap64 bx_ppc_bswap64
BX_CPP_INLINE void WriteHostWordToLittleEndian(Bit16u *hostPtr, Bit16u nativeVar16)
{
bx_ppc_store_le16(hostPtr, nativeVar16);
}
BX_CPP_INLINE void WriteHostDWordToLittleEndian(Bit32u *hostPtr, Bit32u nativeVar32)
{
bx_ppc_store_le32(hostPtr, nativeVar32);
}
BX_CPP_INLINE void WriteHostQWordToLittleEndian(Bit64u *hostPtr, Bit64u nativeVar64)
{
bx_ppc_store_le64(hostPtr, nativeVar64);
}
BX_CPP_INLINE Bit16u ReadHostWordFromLittleEndian(Bit16u *hostPtr)
{
return bx_ppc_load_le16(hostPtr);
}
BX_CPP_INLINE Bit32u ReadHostDWordFromLittleEndian(Bit32u *hostPtr)
{
return bx_ppc_load_le32(hostPtr);
}
BX_CPP_INLINE Bit64u ReadHostQWordFromLittleEndian(Bit64u *hostPtr)
{
return bx_ppc_load_le64(hostPtr);
}
#else // !__MORPHOS__
BX_CPP_INLINE void WriteHostWordToLittleEndian(Bit16u *hostPtr, Bit16u nativeVar16)
{
*(hostPtr) = bx_bswap16(nativeVar16);
}
BX_CPP_INLINE void WriteHostDWordToLittleEndian(Bit32u *hostPtr, Bit32u nativeVar32)
{
*(hostPtr) = bx_bswap32(nativeVar32);
}
BX_CPP_INLINE void WriteHostQWordToLittleEndian(Bit64u *hostPtr, Bit64u nativeVar64)
{
*(hostPtr) = bx_bswap64(nativeVar64);
}
BX_CPP_INLINE Bit16u ReadHostWordFromLittleEndian(Bit16u *hostPtr)
{
return bx_bswap16(*hostPtr);
}
BX_CPP_INLINE Bit32u ReadHostDWordFromLittleEndian(Bit32u *hostPtr)
{
return bx_bswap32(*hostPtr);
}
BX_CPP_INLINE Bit64u ReadHostQWordFromLittleEndian(Bit64u *hostPtr)
{
return bx_bswap64(*hostPtr);
}
#endif // !__MORPHOS__
#endif // !BX_LITTLE_ENDIAN
#endif // BX_BSWAP_H