mirror of
https://github.com/libretro/pcsx2.git
synced 2024-12-21 09:18:18 +00:00
da691894c3
* Moved the x86 emitter to /common, so that plugins can link against it if they wish. * Created a new "utility" class in /common which houses string utils, fast memcpy, common exception classes, and other handy dandies. * Removed old-style linux automake files from the pcsx2 dir since they were hopelessly out of date (and their multi-file-per-line format makes svn merging impossible >_<) git-svn-id: http://pcsx2.googlecode.com/svn/branches/wxgui@1454 96395faa-99c1-11dd-bbfe-3dabce05a288
183 lines
4.0 KiB
C++
183 lines
4.0 KiB
C++
/* Pcsx2 - Pc Ps2 Emulator
|
|
* Copyright (C) 2002-2009 Pcsx2 Team
|
|
*
|
|
* This program 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 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#ifndef _LNX_MEMZERO_H_
|
|
#define _LNX_MEMZERO_H_
|
|
|
|
// This header contains non-optimized implementation of memzero_ptr and memset8_obj,
|
|
// memset16_obj, etc.
|
|
|
|
template< u32 data, typename T >
|
|
static __forceinline void memset32_obj( T& obj )
|
|
{
|
|
// this function works on 32-bit aligned lengths of data only.
|
|
// If the data length is not a factor of 32 bits, the C++ optimizing compiler will
|
|
// probably just generate mysteriously broken code in Release builds. ;)
|
|
|
|
jASSUME( (sizeof(T) & 0x3) == 0 );
|
|
|
|
u32* dest = (u32*)&obj;
|
|
for( int i=sizeof(T)>>2; i; --i, ++dest )
|
|
*dest = data;
|
|
}
|
|
|
|
template< uint size >
|
|
static __forceinline void memzero_ptr( void* dest )
|
|
{
|
|
memset( dest, 0, size );
|
|
}
|
|
|
|
template< typename T >
|
|
static __forceinline void memzero_obj( T& obj )
|
|
{
|
|
memset( &obj, 0, sizeof( T ) );
|
|
}
|
|
|
|
template< u8 data, typename T >
|
|
static __forceinline void memset8_obj( T& obj )
|
|
{
|
|
// Aligned sizes use the optimized 32 bit inline memset. Unaligned sizes use memset.
|
|
if( (sizeof(T) & 0x3) != 0 )
|
|
memset( &obj, data, sizeof( T ) );
|
|
else
|
|
memset32_obj<data + (data<<8) + (data<<16) + (data<<24)>( obj );
|
|
}
|
|
|
|
template< u16 data, typename T >
|
|
static __forceinline void memset16_obj( T& obj )
|
|
{
|
|
if( (sizeof(T) & 0x3) != 0 )
|
|
_memset_16_unaligned( &obj, data, sizeof( T ) );
|
|
else
|
|
memset32_obj<data + (data<<16)>( obj );
|
|
}
|
|
|
|
|
|
// An optimized memset for 8 bit destination data.
|
|
template< u8 data, size_t bytes >
|
|
static __forceinline void memset_8( void *dest )
|
|
{
|
|
if( bytes == 0 ) return;
|
|
|
|
if( (bytes & 0x3) != 0 )
|
|
{
|
|
// unaligned data length. No point in doing an optimized inline version (too complicated!)
|
|
// So fall back on the compiler implementation:
|
|
|
|
memset( dest, data, bytes );
|
|
return;
|
|
}
|
|
|
|
// This function only works on 32-bit alignments of data copied.
|
|
jASSUME( (bytes & 0x3) == 0 );
|
|
|
|
enum
|
|
{
|
|
remdat = bytes>>2,
|
|
data32 = data + (data<<8) + (data<<16) + (data<<24)
|
|
};
|
|
|
|
// macro to execute the x86/32 "stosd" copies.
|
|
switch( remdat )
|
|
{
|
|
case 1:
|
|
*(u32*)dest = data32;
|
|
return;
|
|
|
|
case 2:
|
|
((u32*)dest)[0] = data32;
|
|
((u32*)dest)[1] = data32;
|
|
return;
|
|
|
|
case 3:
|
|
__asm__
|
|
(
|
|
".intel_syntax noprefix\n"
|
|
"cld\n"
|
|
// "mov edi, %[dest]\n"
|
|
// "mov eax, %[data32]\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
".att_syntax\n"
|
|
:
|
|
: [dest]"D"(dest), [data32]"a"(data32)
|
|
// D - edi, a -- eax, c ecx
|
|
:
|
|
);
|
|
return;
|
|
|
|
case 4:
|
|
__asm__
|
|
(
|
|
".intel_syntax noprefix\n"
|
|
"cld\n"
|
|
// "mov edi, %[dest]\n"
|
|
// "mov eax, %[data32]\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
".att_syntax\n"
|
|
:
|
|
: [dest]"D"(dest), [data32]"a"(data32)
|
|
:
|
|
|
|
);
|
|
return;
|
|
|
|
case 5:
|
|
__asm__
|
|
(
|
|
".intel_syntax noprefix\n"
|
|
"cld\n"
|
|
// "mov edi, %[dest]\n"
|
|
// "mov eax, %[data32]\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
"stosd\n"
|
|
".att_syntax\n"
|
|
:
|
|
: [dest]"D"(dest), [data32]"a"(data32)
|
|
:
|
|
|
|
);
|
|
return;
|
|
|
|
default:
|
|
__asm__
|
|
(
|
|
".intel_syntax noprefix\n"
|
|
"cld\n"
|
|
// "mov ecx, %[remdat]\n"
|
|
// "mov edi, %[dest]\n"
|
|
// "mov eax, %\[data32]n"
|
|
"rep stosd\n"
|
|
".att_syntax\n"
|
|
:
|
|
: [remdat]"c"(remdat), [dest]"D"(dest), [data32]"a"(data32)
|
|
:
|
|
);
|
|
return;
|
|
}
|
|
}
|
|
|
|
#endif
|