mirror of
https://github.com/libretro/ppsspp.git
synced 2025-01-08 17:40:45 +00:00
118 lines
3.1 KiB
C++
118 lines
3.1 KiB
C++
// Copyright (C) 2003 Dolphin 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
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
// 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 SVN repository and contact information can be found at
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
#include "ppsspp_config.h"
|
|
|
|
#if defined(__APPLE__)
|
|
|
|
#include <string>
|
|
#include <inttypes.h>
|
|
|
|
#include "FileUtil.h"
|
|
#include "MemoryUtil.h"
|
|
#include "MemArena.h"
|
|
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <cerrno>
|
|
#include <cstring>
|
|
|
|
#include <mach/mach.h>
|
|
#include <mach/vm_map.h>
|
|
|
|
size_t MemArena::roundup(size_t x) {
|
|
return x;
|
|
}
|
|
|
|
void MemArena::GrabLowMemSpace(size_t size) {
|
|
vm_size = size;
|
|
kern_return_t retval = vm_allocate(mach_task_self(), &vm_mem, size, VM_FLAGS_ANYWHERE);
|
|
if (retval != KERN_SUCCESS) {
|
|
ERROR_LOG(MEMMAP, "Failed to grab a block of virtual memory");
|
|
} else {
|
|
INFO_LOG(MEMMAP, "Successfully allocated %d bytes at %p", (int)size, (void *)vm_mem);
|
|
}
|
|
}
|
|
|
|
void MemArena::ReleaseSpace() {
|
|
vm_deallocate(mach_task_self(), vm_mem, vm_size);
|
|
vm_size = 0;
|
|
vm_mem = 0;
|
|
}
|
|
|
|
void *MemArena::CreateView(s64 offset, size_t size, void *base) {
|
|
mach_port_t self = mach_task_self();
|
|
vm_address_t target = (vm_address_t)base;
|
|
uint64_t mask = 0;
|
|
bool anywhere = false;
|
|
vm_address_t source = vm_mem + offset;
|
|
vm_prot_t cur_protection = 0;
|
|
vm_prot_t max_protection = 0;
|
|
kern_return_t retval =
|
|
vm_remap(self, &target, size, mask, anywhere,
|
|
self, source, false,
|
|
&cur_protection, &max_protection, VM_INHERIT_DEFAULT);
|
|
if (retval != KERN_SUCCESS) {
|
|
// 1 == KERN_INVALID_ADDRESS
|
|
// 3 == KERN_NO_SPACE (race?)
|
|
// 4 == KERN_INVALID_ARGUMENT
|
|
ERROR_LOG(MEMMAP, "vm_remap failed (%d) - could not remap from %llx (offset %llx) of size %llx to %p",
|
|
(int)retval, (uint64_t)source, (uint64_t)offset, (uint64_t)size, base);
|
|
return nullptr;
|
|
}
|
|
return (void *)target;
|
|
}
|
|
|
|
void MemArena::ReleaseView(void* view, size_t size) {
|
|
vm_address_t addr = (vm_address_t)view;
|
|
vm_deallocate(mach_task_self(), addr, size);
|
|
}
|
|
|
|
bool MemArena::NeedsProbing() {
|
|
#if defined(IOS) && PPSSPP_ARCH(64BIT)
|
|
return true;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
u8* MemArena::Find4GBBase() {
|
|
#if defined(IOS) && PPSSPP_ARCH(64BIT)
|
|
// The caller will need to do probing, like on 32-bit Windows.
|
|
return nullptr;
|
|
#else
|
|
size_t size;
|
|
#if PPSSPP_ARCH(64BIT)
|
|
size = 0xE1000000;
|
|
#else
|
|
size = 0x10000000;
|
|
#endif
|
|
|
|
vm_address_t addr = 0;
|
|
kern_return_t retval = vm_allocate(mach_task_self(), &addr, size, VM_FLAGS_ANYWHERE);
|
|
if (retval == KERN_SUCCESS) {
|
|
// Don't need the memory now, was just probing.
|
|
vm_deallocate(mach_task_self(), addr, size);
|
|
return (u8 *)addr;
|
|
}
|
|
#endif
|
|
return nullptr;
|
|
}
|
|
|
|
#endif // __APPLE__
|