Example #1
0
static void *try_reserve(uintptr_t try_addr, uae_u32 size, int flags)
{
	void *address = NULL;
	if (try_addr) {
		uae_log("VM: Reserve  0x%-8x bytes, try address 0x%llx\n",
				size, (uae_u64) try_addr);
	} else {
		uae_log("VM: Reserve  0x%-8x bytes\n", size);
	}
#ifdef _WIN32
	int va_type = MEM_RESERVE;
	if (flags & UAE_VM_WRITE_WATCH) {
		va_type |= MEM_WRITE_WATCH;
	}
	int va_protect = protect_to_native(UAE_VM_NO_ACCESS);
	address = VirtualAlloc((void *) try_addr, size, va_type, va_protect);
	if (address == NULL) {
		return NULL;
	}
#else
	int mmap_flags = MAP_PRIVATE | MAP_ANON;
	address = mmap((void *) try_addr, size, PROT_NONE, mmap_flags, -1, 0);
	if (address == MAP_FAILED) {
		return NULL;
	}
#endif
#ifdef CPU_64_BIT
	if (flags & UAE_VM_32BIT) {
		uintptr_t end = (uintptr_t) address + size;
		if (address && end > (uintptr_t) 0x100000000ULL) {
			uae_log("VM: Reserve  0x%-8x bytes, got address 0x%llx (> 32-bit)\n",
					size, (uae_u64) (uintptr_t) address);
#ifdef _WIN32
			VirtualFree(address, 0, MEM_RELEASE);
#else
			munmap(address, size);
#endif
			return NULL;
		}
	}
#endif
	return address;
}
Example #2
0
bool uae_vm_decommit(void *address, uae_u32 size)
{
	uae_log("VM: Decommit 0x%-8x bytes at %p\n", size, address);
#ifdef _WIN32
	return VirtualFree (address, size, MEM_DECOMMIT) != 0;
#else
    /* Re-map the memory so we get fresh unused pages (and the old ones can be
     * released and physical memory reclaimed). We also assume that the new
     * pages will be zero-initialized (tested on Linux and OS X). */
    void *result = mmap(address, size, PROT_NONE,
                        MAP_ANON | MAP_PRIVATE | MAP_FIXED, -1, 0);
    if (result == MAP_FAILED) {
        uae_log("VM: Warning - could not re-map with MAP_FIXED at %p\n",
                address);
        do_protect(address, size, UAE_VM_NO_ACCESS);
    }
    return result != MAP_FAILED;
#endif
}
Example #3
0
void amiga_set_cpu_idle(int idle)
{
    idle = CLAMP(idle, 0, 10);
    if (idle != 0) {
        idle = (12 - idle) * 15;
    }
    uae_log("setting cpu_idle = %d\n", idle);
    changed_prefs.cpu_idle = idle;
    currprefs.cpu_idle = idle;
}
Example #4
0
static void add_allocation(void *address, uae_u32 size)
{
	uae_log("VM: add_allocation %p (%d)\n", address, size);
	for (int i = 0; i < MAX_ALLOCATIONS; i++) {
		if (alloc_sizes[i].address == NULL) {
			alloc_sizes[i].address = address;
			alloc_sizes[i].size = size;
			return;
		}
	}
	abort();
}
Example #5
0
static int protect_to_native(int protect)
{
#ifdef _WIN32
	if (protect == UAE_VM_NO_ACCESS) return PAGE_NOACCESS;
	if (protect == UAE_VM_READ) return PAGE_READONLY;
	if (protect == UAE_VM_READ_WRITE) return PAGE_READWRITE;
	if (protect == UAE_VM_READ_EXECUTE) return PAGE_EXECUTE_READ;
	if (protect == UAE_VM_READ_WRITE_EXECUTE) return PAGE_EXECUTE_READWRITE;
	uae_log("VM: Invalid protect value %d\n", protect);
	return PAGE_NOACCESS;
#else
	if (protect == UAE_VM_NO_ACCESS) return PROT_NONE;
	if (protect == UAE_VM_READ) return PROT_READ;
	if (protect == UAE_VM_READ_WRITE) return PROT_READ | PROT_WRITE;
	if (protect == UAE_VM_READ_EXECUTE) return PROT_READ | PROT_EXEC;
	if (protect == UAE_VM_READ_WRITE_EXECUTE) {
		return PROT_READ | PROT_WRITE | PROT_EXEC;
	}
	uae_log("VM: Invalid protect value %d\n", protect);
	return PROT_NONE;
#endif
}
Example #6
0
static bool do_protect(void *address, int size, int protect)
{
#ifdef TRACK_ALLOCATIONS
	uae_u32 allocated_size = find_allocation(address);
	assert(allocated_size == size);
#endif
#ifdef _WIN32
	DWORD old;
	if (VirtualProtect(address, size, protect_to_native(protect), &old) == 0) {
		uae_log("VM: uae_vm_protect(%p, %d, %d) VirtualProtect failed (%d)\n",
				address, size, protect, GetLastError());
		return false;
	}
#else
	if (mprotect(address, size, protect_to_native(protect)) != 0) {
		uae_log("VM: uae_vm_protect(%p, %d, %d) mprotect failed (%d)\n",
				address, size, protect, errno);
		return false;
	}
#endif
	return true;
}
Example #7
0
int hdf_resize_target(struct hardfiledata *hfd, uae_u64 newsize)
{
    if (newsize < hfd->physsize) {
        uae_log("hdf_resize_target: truncation not implemented\n");
        return 0;
    }
    if (newsize == hfd->physsize) {
        return 1;
    }
    /* Now, newsize must be larger than hfd->physsize, we seek to newsize - 1
     * and write a single 0 byte to make the file exactly newsize bytes big. */
    if (uae_fseeko64(hfd->handle->h, newsize - 1, SEEK_SET) != 0) {
        uae_log("hdf_resize_target: fseek failed errno %d\n", errno);
        return 0;
    }
    if (fwrite("", 1, 1, hfd->handle->h) != 1) {
        uae_log("hdf_resize_target: failed to write byte at position "
                "%lld errno %d\n", newsize - 1, errno);
        return 0;
    }
    uae_log("hdf_resize_target: %lld -> %lld\n", hfd->physsize, newsize);
    hfd->physsize = newsize;
    return 1;
}
Example #8
0
void *uae_vm_commit(void *address, uae_u32 size, int protect)
{
	uae_log("VM: Commit   0x%-8x bytes at %p (%s)\n",
			size, address, protect_description(protect));
#ifdef _WIN32
	int va_type = MEM_COMMIT ;
	int va_protect = protect_to_native(protect);
	address = VirtualAlloc(address, size, va_type, va_protect);
#else
#ifdef CLEAR_MEMORY_ON_COMMIT
	do_protect(address, size, UAE_VM_READ_WRITE);
	memset(address, 0, size);
#endif
	do_protect(address, size, protect);
#endif
	return address;
}
Example #9
0
static bool do_free(void *address, int size)
{
#ifdef TRACK_ALLOCATIONS
	uae_u32 allocated_size = remove_allocation(address);
	assert(allocated_size == size);
#endif
#ifdef _WIN32
	return VirtualFree(address, 0, MEM_RELEASE) != 0;
#else
	if (munmap(address, size) != 0) {
		uae_log("VM: uae_vm_free(%p, %d) munmap failed (%d)\n",
				address, size, errno);
		return false;
	}
#endif
	return true;
}
Example #10
0
bool uae_vm_decommit(void *address, uae_u32 size)
{
	uae_log("VM: Decommit 0x%-8x bytes at %p\n", size, address);
#ifdef _WIN32
	return VirtualFree (address, size, MEM_DECOMMIT) != 0;
#else
#if 0
	/* FIXME: perhaps we can unmmap and mmap the memory region again to
	 * allow the operating system to throw away the (unused) pages. Of
	 * course, we have problem if re-mmaping it fails. If we do this,
	 * we may be able to remove the memory clear operation in
	 * uae_vm_commit. We might also be able to use mmap with MAP_FIXED
	 * to more "safely" overwrite the old mapping. */
	munmap(address, size);
	mmap(...);
#endif
	return do_protect(address, size, UAE_VM_NO_ACCESS);
#endif
}
Example #11
0
void *uae_vm_reserve(uae_u32 size, int flags)
{
	void *address = NULL;
#ifdef _WIN32
	address = try_reserve(0x80000000, size, flags);
	if (address == NULL && (flags & UAE_VM_32BIT)) {
		if (size <= 768 * 1024 * 1024) {
			address = try_reserve(0x78000000 - size, size, flags);
		}
	}
	if (address == NULL && (flags & UAE_VM_32BIT) == 0) {
		address = try_reserve(0, size, flags);
	}
#else
#ifdef CPU_64_BIT
	if (flags & UAE_VM_32BIT) {
#else
	if (true) {
#endif
		uintptr_t try_addr = 0x80000000;
		while (address == NULL) {
			address = try_reserve(try_addr, size, flags);
			if (address == NULL) {
				try_addr -= 0x4000000;
				if (try_addr < 0x20000000) {
					break;
				}
				continue;
			}
		}
	}
	if (address == NULL && (flags & UAE_VM_32BIT) == 0) {
		address = try_reserve(0, size, flags);
	}
#endif
	if (address) {
		uae_log("VM: Reserve  0x%-8x bytes, got address 0x%llx\n",
				size, (uae_u64) (uintptr_t) address);
	} else {
		uae_log("VM: Reserve  0x%-8x bytes failed!\n", size);
	}
	return address;
}

void *uae_vm_reserve_fixed(void *want_addr, uae_u32 size, int flags)
{
	void *address = NULL;
	uae_log("VM: Reserve  0x%-8x bytes at %p (fixed)\n", size, want_addr);
	address = try_reserve((uintptr_t) want_addr, size, flags);
	if (address == NULL) {
		uae_log("VM: Reserve  0x%-8x bytes at %p failed!\n", size, want_addr);
		return NULL;
	}
	if (address != want_addr) {
		do_free(address, size);
		return NULL;
	}
	uae_log("VM: Reserve  0x%-8x bytes, got address 0x%llx\n",
			size, (uae_u64) (uintptr_t) address);
	return address;
}
Example #12
0
bool uae_vm_free(void *address, int size)
{
	uae_log("VM: Free     0x%-8x bytes at %p\n", size, address);
	return do_free(address, size);
}
Example #13
0
static void *uae_vm_alloc_with_flags(uae_u32 size, int flags, int protect)
{
	void *address = NULL;
	static bool first_allocation = true;
	if (first_allocation) {
		/* FIXME: log contents of /proc/self/maps on Linux */
		/* FIXME: use VirtualQuery function on Windows? */
		first_allocation = false;
	}
#ifdef LOG_ALLOCATIONS
	uae_log("VM: Allocate 0x%-8x bytes [%d] (%s)\n",
			size, flags, protect_description(protect));
#endif

#ifdef _WIN32
	int va_type = MEM_COMMIT | MEM_RESERVE;
	if (flags & UAE_VM_WRITE_WATCH) {
		va_type |= MEM_WRITE_WATCH;
	}
	int va_protect = protect_to_native(protect);
#else
	int mmap_flags = MAP_PRIVATE | MAP_ANON;
	int mmap_prot = protect_to_native(protect);
#endif

#ifndef CPU_64_BIT
	flags &= ~UAE_VM_32BIT;
#endif
	if (flags & UAE_VM_32BIT) {
		/* Stupid algorithm to find available space, but should
		 * work well enough when there is not a lot of allocations. */
		int step = uae_vm_page_size();
		uae_u8 *p = (uae_u8 *) 0x40000000;
		uae_u8 *p_end = natmem_reserved - size;
		if (size > 1024 * 1024) {
			/* Reserve some space for smaller allocations */
			p += 32 * 1024 * 1024;
			step = 1024 * 1024;
		}
#ifdef HAVE_MAP_32BIT
		address = mmap(0, size, mmap_prot, mmap_flags | MAP_32BIT, -1, 0);
		if (address == MAP_FAILED) {
			address = NULL;
		}
#endif
		while (address == NULL) {
			if (p > p_end) {
				break;
			}
#ifdef _WIN32
			address = VirtualAlloc(p, size, va_type, va_protect);
#else
			address = mmap(p, size, mmap_prot, mmap_flags, -1, 0);
			// write_log("VM: trying %p step is 0x%x = %p\n", p, step, address);
			if (address == MAP_FAILED) {
				address = NULL;
			} else if (((uintptr_t) address) + size > (uintptr_t) 0xffffffff) {
				munmap(address, size);
				address = NULL;
			}
#endif
			p += step;
		}
	} else {
#ifdef _WIN32
		address = VirtualAlloc(NULL, size, va_type, va_protect);
#else
		address = mmap(0, size, mmap_prot, mmap_flags, -1, 0);
		if (address == MAP_FAILED) {
			address = NULL;
		}
#endif
	}

	if (address == NULL) {
		uae_log("VM: uae_vm_alloc(%u, %d, %d) mmap failed (%d)\n",
		    size, flags, protect, errno);
	    return NULL;
	}
#ifdef TRACK_ALLOCATIONS
	add_allocation(address, size);
#endif
#ifdef LOG_ALLOCATIONS
	uae_log("VM: %p\n", address);
#endif
	return address;
}
Example #14
0
static void *uae_vm_alloc_with_flags(uae_u32 size, int flags, int protect)
{
	void *address = NULL;
	uae_log("VM: Allocate 0x%-8x bytes [%d] (%s)\n",
			size, flags, protect_description(protect));
#ifdef _WIN32
	int va_type = MEM_COMMIT | MEM_RESERVE;
	if (flags & UAE_VM_WRITE_WATCH) {
		va_type |= MEM_WRITE_WATCH;
	}
	int va_protect = protect_to_native(protect);
#ifdef CPU_64_BIT
	if (flags & UAE_VM_32BIT) {
		/* Stupid algorithm to find available space, but should
		 * work well enough when there is not a lot of allocations. */
		uae_u8 *p = (uae_u8 *) 0x50000000;
		while (address == NULL) {
			if (p >= (void*) 0x60000000) {
				break;
			}
			address = VirtualAlloc(p, size, va_type, va_protect);
			p += uae_vm_page_size();
		}
	}
#endif
	if (!address) {
		address = VirtualAlloc(NULL, size, va_type, va_protect);
	}
#else
	//size = size < uae_vm_page_size() ? uae_vm_page_size() : size;
	int mmap_flags = MAP_PRIVATE | MAP_ANON;
	int mmap_prot = protect_to_native(protect);
#ifdef CPU_64_BIT
	if (flags & UAE_VM_32BIT) {
#ifdef HAVE_MAP_32BIT
		mmap_flags |= MAP_32BIT;
#else
		/* Stupid algorithm to find available space, but should
		 * work well enough when there is not a lot of allocations. */
		uae_u8 *p = natmem_offset - 0x10000000;
		uae_u8 *p_end = p + 0x10000000;
		while (address == NULL) {
			if (p >= p_end) {
				break;
			}
			address = mmap(p, size, mmap_prot, mmap_flags, -1, 0);
			/* FIXME: check 32-bit result */
			if (address == MAP_FAILED) {
				address = NULL;
			}
			p += uae_vm_page_size();
		}
#endif
	}
#endif
	if (address == NULL) {
		address = mmap(0, size, mmap_prot, mmap_flags, -1, 0);
		if (address == MAP_FAILED) {
			address = NULL;
		}
	}
#endif
	if (address == NULL) {
		uae_log("VM: uae_vm_alloc(%u, %d, %d) mmap failed (%d)\n",
		    size, flags, protect, errno);
	    return NULL;
	}
#ifdef TRACK_ALLOCATIONS
	add_allocation(address, size);
#endif
	uae_log("VM: %p\n", address);
	return address;
}
Example #15
0
static void *uae_vm_alloc_with_flags(uae_u32 size, int flags, int protect)
{
	void *address = NULL;
	static bool first_allocation = true;
	if (first_allocation) {
		/* FIXME: log contents of /proc/self/maps on Linux */
		/* FIXME: use VirtualQuery function on Windows? */
		first_allocation = false;
	}
#ifdef LOG_ALLOCATIONS
	uae_log("VM: Allocate 0x%-8x bytes [%d] (%s)\n",
			size, flags, protect_description(protect));
#endif

#ifdef _WIN32
	int native_flags = MEM_COMMIT | MEM_RESERVE;
	if (flags & UAE_VM_WRITE_WATCH) {
		native_flags |= MEM_WRITE_WATCH;
	}
	int native_protect = protect_to_native(protect);
#else
	int native_flags = MAP_PRIVATE | MAP_ANON;
	int native_protect = protect_to_native(protect);
#endif

#ifndef CPU_64_BIT
	flags &= ~UAE_VM_32BIT;
#endif
	if (flags & UAE_VM_32BIT) {
		/* Stupid algorithm to find available space, but should
		 * work well enough when there is not a lot of allocations. */
		/* FIXME: Consider allocating a bigger chunk of memory, and manually
		 * keep track of allocations. */
#if 1
		if (!address) {
			address = try_alloc_32bit(
						size, native_flags, native_protect,
						(uae_u8 *) 0x40000000, natmem_reserved - size);
		}
#endif
		if (!address && natmem_reserved < (uae_u8 *) 0x60000000) {
			address = try_alloc_32bit(
						size, native_flags, native_protect,
						(uae_u8 *) natmem_reserved + natmem_reserved_size,
						(uae_u8 *) 0xffffffff - size + 1);
		}
		if (!address) {
			address = try_alloc_32bit(
						size, native_flags, native_protect,
						(uae_u8 *) 0x20000000,
						min((uae_u8 *) 0x40000000, natmem_reserved - size));
		}
	} else {
#ifdef _WIN32
		address = VirtualAlloc(NULL, size, native_flags, native_protect);
#else
		address = mmap(0, size, native_protect, native_flags, -1, 0);
		if (address == MAP_FAILED) {
			address = NULL;
		}
#endif
	}

	if (address == NULL) {
		uae_log("VM: uae_vm_alloc(%u, %d, %d) mmap failed (%d)\n",
		    size, flags, protect, errno);
	    return NULL;
	}
#ifdef TRACK_ALLOCATIONS
	add_allocation(address, size);
#endif
#ifdef LOG_ALLOCATIONS
	uae_log("VM: %p\n", address);
#endif
	return address;
}