2012-11-01 15:19:01 +00:00
|
|
|
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
|
|
|
|
// 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
|
2012-11-04 22:01:49 +00:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// 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 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
|
|
|
#include "Globals.h"
|
2013-09-29 23:56:11 +00:00
|
|
|
#include "Core/CoreTiming.h"
|
2013-09-21 16:52:30 +00:00
|
|
|
#include "Core/MemMap.h"
|
2013-06-08 11:37:40 +00:00
|
|
|
#include "Core/Reporting.h"
|
2013-06-08 10:50:12 +00:00
|
|
|
#include "Core/HLE/HLE.h"
|
|
|
|
#include "GPU/GPUInterface.h"
|
|
|
|
#include "GPU/GPUState.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
u64 dmacMemcpyDeadline;
|
|
|
|
|
|
|
|
void __DmacInit() {
|
|
|
|
dmacMemcpyDeadline = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void __DmacDoState(PointerWrap &p) {
|
|
|
|
auto s = p.Section("sceDmac", 0, 1);
|
|
|
|
if (s == 0) {
|
|
|
|
dmacMemcpyDeadline = 0;
|
|
|
|
return;
|
2013-06-08 10:50:12 +00:00
|
|
|
}
|
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
p.Do(dmacMemcpyDeadline);
|
|
|
|
}
|
2013-06-08 10:50:12 +00:00
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
int __DmacMemcpy(u32 dst, u32 src, u32 size) {
|
2012-11-27 09:18:36 +00:00
|
|
|
Memory::Memcpy(dst, Memory::GetPointer(src), size);
|
2013-06-08 10:50:12 +00:00
|
|
|
|
2013-06-08 11:37:40 +00:00
|
|
|
src &= ~0x40000000;
|
2013-06-09 08:21:50 +00:00
|
|
|
dst &= ~0x40000000;
|
2013-09-21 16:52:30 +00:00
|
|
|
if (Memory::IsVRAMAddress(src) || Memory::IsVRAMAddress(dst)) {
|
2013-06-09 08:21:50 +00:00
|
|
|
gpu->UpdateMemory(dst, src, size);
|
2013-09-21 16:52:30 +00:00
|
|
|
}
|
2013-09-29 23:51:49 +00:00
|
|
|
|
|
|
|
// This number seems strangely reproducible.
|
|
|
|
if (size >= 272) {
|
|
|
|
// Approx. 225 MiB/s or 235929600 B/s, so let's go with 236 B/us.
|
2013-09-29 23:56:11 +00:00
|
|
|
int delayUs = size / 236;
|
|
|
|
dmacMemcpyDeadline = CoreTiming::GetTicks() + usToCycles(delayUs);
|
|
|
|
return hleDelayResult(0, "dmac copy", delayUs);
|
2013-09-29 23:51:49 +00:00
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
u32 sceDmacMemcpy(u32 dst, u32 src, u32 size) {
|
2013-09-29 23:51:49 +00:00
|
|
|
if (size == 0) {
|
2013-09-29 23:56:11 +00:00
|
|
|
ERROR_LOG(HLE, "sceDmacMemcpy(dest=%08x, src=%08x, size=%i): invalid size", dst, src, size);
|
2013-09-29 23:51:49 +00:00
|
|
|
return SCE_KERNEL_ERROR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
if (!Memory::IsValidAddress(dst) || !Memory::IsValidAddress(src)) {
|
2013-09-29 23:56:11 +00:00
|
|
|
ERROR_LOG(HLE, "sceDmacMemcpy(dest=%08x, src=%08x, size=%i): invalid address", dst, src, size);
|
2013-09-29 23:51:49 +00:00
|
|
|
return SCE_KERNEL_ERROR_INVALID_POINTER;
|
|
|
|
}
|
2013-10-22 12:57:07 +00:00
|
|
|
if (dst + size >= 0x80000000 || src + size >= 0x80000000) {
|
|
|
|
ERROR_LOG(HLE, "sceDmacMemcpy(dest=%08x, src=%08x, size=%i): illegal size", dst, src, size);
|
|
|
|
return 0x80000023;
|
|
|
|
}
|
2013-10-22 09:11:43 +00:00
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
if (dmacMemcpyDeadline > CoreTiming::GetTicks()) {
|
|
|
|
WARN_LOG_REPORT(HLE, "sceDmacMemcpy(dest=%08x, src=%08x, size=%i): overlapping read", dst, src, size);
|
|
|
|
// TODO: Should block, seems like copy doesn't start until previous finishes.
|
|
|
|
// Might matter for overlapping copies.
|
|
|
|
} else {
|
|
|
|
DEBUG_LOG(HLE, "sceDmacMemcpy(dest=%08x, src=%08x, size=%i)", dst, src, size);
|
|
|
|
}
|
2013-09-29 23:51:49 +00:00
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
return __DmacMemcpy(dst, src, size);
|
|
|
|
}
|
2013-09-29 23:51:49 +00:00
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
u32 sceDmacTryMemcpy(u32 dst, u32 src, u32 size) {
|
|
|
|
if (size == 0) {
|
|
|
|
ERROR_LOG(HLE, "sceDmacTryMemcpy(dest=%08x, src=%08x, size=%i): invalid size", dst, src, size);
|
|
|
|
return SCE_KERNEL_ERROR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
if (!Memory::IsValidAddress(dst) || !Memory::IsValidAddress(src)) {
|
|
|
|
ERROR_LOG(HLE, "sceDmacTryMemcpy(dest=%08x, src=%08x, size=%i): invalid address", dst, src, size);
|
|
|
|
return SCE_KERNEL_ERROR_INVALID_POINTER;
|
2013-09-29 23:51:49 +00:00
|
|
|
}
|
2013-10-22 12:57:07 +00:00
|
|
|
if (dst + size >= 0x80000000 || src + size >= 0x80000000) {
|
|
|
|
ERROR_LOG(HLE, "sceDmacTryMemcpy(dest=%08x, src=%08x, size=%i): illegal size", dst, src, size);
|
|
|
|
return 0x80000023;
|
|
|
|
}
|
2013-09-29 23:51:49 +00:00
|
|
|
|
2013-09-29 23:56:11 +00:00
|
|
|
if (dmacMemcpyDeadline > CoreTiming::GetTicks()) {
|
|
|
|
DEBUG_LOG(HLE, "sceDmacTryMemcpy(dest=%08x, src=%08x, size=%i): busy", dst, src, size);
|
|
|
|
return SCE_KERNEL_ERROR_BUSY;
|
|
|
|
} else {
|
|
|
|
DEBUG_LOG(HLE, "sceDmacTryMemcpy(dest=%08x, src=%08x, size=%i)", dst, src, size);
|
2013-09-29 23:51:49 +00:00
|
|
|
}
|
2013-09-29 23:56:11 +00:00
|
|
|
|
|
|
|
return __DmacMemcpy(dst, src, size);
|
2013-09-26 09:25:56 +00:00
|
|
|
}
|
|
|
|
|
2013-09-21 16:52:30 +00:00
|
|
|
const HLEFunction sceDmac[] = {
|
2012-11-05 09:05:09 +00:00
|
|
|
{0x617f3fe6, &WrapU_UUU<sceDmacMemcpy>, "sceDmacMemcpy"},
|
2013-09-26 09:38:45 +00:00
|
|
|
{0xd97f94d8, &WrapU_UUU<sceDmacTryMemcpy>, "sceDmacTryMemcpy"},
|
2012-11-01 15:19:01 +00:00
|
|
|
};
|
|
|
|
|
2013-09-21 16:52:30 +00:00
|
|
|
void Register_sceDmac() {
|
2012-11-01 15:19:01 +00:00
|
|
|
RegisterModule("sceDmac", ARRAY_SIZE(sceDmac), sceDmac);
|
|
|
|
}
|