コード例 #1
0
ファイル: vm.cpp プロジェクト: alpine9000/fs-uae
static void *try_alloc_32bit(uae_u32 size, int native_flags, int native_protect,
							uae_u8 *p, uae_u8 *p_end)
{
	if (p_end <= p) {
		return NULL;
	}
	if ((uintptr_t) p % uae_vm_page_size() != 0) {
		/* Round up to the nearest page size */
		p += uae_vm_page_size() - (uintptr_t) p % uae_vm_page_size();
	}
	void *address = NULL;
	int step = uae_vm_page_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, native_protect, native_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, native_flags, native_protect);
#else
		address = mmap(p, size, native_protect, native_flags, -1, 0);
#ifdef LOG_ALLOCATIONS
		write_log("VM: trying %p step is 0x%x = %p\n", p, step, address);
#endif
		if (address == MAP_FAILED) {
			address = NULL;
		} else if (((uintptr_t) address) + size > (uintptr_t) 0xffffffff) {
			munmap(address, size);
			address = NULL;
		}
#endif
		p += step;
	}
	return address;
}
コード例 #2
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;
}
コード例 #3
0
ファイル: vm.cpp プロジェクト: jens-maus/fs-uae
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;
}