mirror of
https://github.com/libretro/ppsspp.git
synced 2025-01-06 00:18:21 +00:00
104 lines
2.2 KiB
C++
104 lines
2.2 KiB
C++
#ifdef _WIN32
|
|
#include "stdafx.h"
|
|
#endif
|
|
#include "MIPSAsm.h"
|
|
#include <cstdarg>
|
|
#include <cstring>
|
|
#include "util/text/utf8.h"
|
|
#include "Core/MemMap.h"
|
|
#include "Core/MIPS/JitCommon/NativeJit.h"
|
|
#include "Core/Debugger/SymbolMap.h"
|
|
|
|
#if defined(_WIN32) || defined(ANDROID)
|
|
#include "ext/armips/Core/Assembler.h"
|
|
#endif
|
|
|
|
namespace MIPSAsm
|
|
{
|
|
static std::wstring errorText;
|
|
|
|
std::wstring GetAssembleError()
|
|
{
|
|
return errorText;
|
|
}
|
|
|
|
#if defined(_WIN32) || defined(ANDROID)
|
|
class PspAssemblerFile: public AssemblerFile
|
|
{
|
|
public:
|
|
PspAssemblerFile()
|
|
{
|
|
address = 0;
|
|
}
|
|
|
|
virtual bool open(bool onlyCheck) { return true; };
|
|
virtual void close() { };
|
|
virtual bool isOpen() { return true; };
|
|
virtual bool write(void* data, size_t length)
|
|
{
|
|
if (!Memory::IsValidAddress((u32)(address+length-1)))
|
|
return false;
|
|
|
|
Memory::Memcpy((u32)address,data,(u32)length);
|
|
|
|
// In case this is a delay slot or combined instruction, clear cache above it too.
|
|
if (MIPSComp::jit)
|
|
MIPSComp::jit->InvalidateCacheAt((u32)(address - 4),(int)length+4);
|
|
|
|
address += length;
|
|
return true;
|
|
}
|
|
virtual u64 getVirtualAddress() { return address; };
|
|
virtual u64 getPhysicalAddress() { return getVirtualAddress(); };
|
|
virtual bool seekVirtual(u64 virtualAddress)
|
|
{
|
|
if (!Memory::IsValidAddress(virtualAddress))
|
|
return false;
|
|
address = virtualAddress;
|
|
return true;
|
|
}
|
|
virtual bool seekPhysical(u64 physicalAddress) { return seekVirtual(physicalAddress); }
|
|
private:
|
|
u64 address;
|
|
};
|
|
#endif
|
|
|
|
bool MipsAssembleOpcode(const char* line, DebugInterface* cpu, u32 address)
|
|
{
|
|
#if defined(_WIN32) || defined(ANDROID)
|
|
PspAssemblerFile file;
|
|
StringList errors;
|
|
|
|
wchar_t str[64];
|
|
swprintf(str,64,L".psp\n.org 0x%08X\n",address);
|
|
|
|
ArmipsArguments args;
|
|
args.mode = ArmipsMode::MEMORY;
|
|
args.content = str + ConvertUTF8ToWString(line);
|
|
args.silent = true;
|
|
args.memoryFile = &file;
|
|
args.errorsResult = &errors;
|
|
|
|
symbolMap.GetLabels(args.labels);
|
|
|
|
errorText = L"";
|
|
if (!runArmips(args))
|
|
{
|
|
for (size_t i = 0; i < errors.size(); i++)
|
|
{
|
|
errorText += errors[i];
|
|
if (i != errors.size()-1)
|
|
errorText += L"\n";
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
#else
|
|
errorText = L"Unsupported platform";
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
} |