diff options
Diffstat (limited to 'tools/testing/selftests/kvm/lib/kvm_util.c')
| -rw-r--r-- | tools/testing/selftests/kvm/lib/kvm_util.c | 84 | 
1 files changed, 53 insertions, 31 deletions
| diff --git a/tools/testing/selftests/kvm/lib/kvm_util.c b/tools/testing/selftests/kvm/lib/kvm_util.c index 1d26a2160178..e9607eb089be 100644 --- a/tools/testing/selftests/kvm/lib/kvm_util.c +++ b/tools/testing/selftests/kvm/lib/kvm_util.c @@ -186,13 +186,10 @@ const struct vm_guest_mode_params vm_guest_mode_params[] = {  _Static_assert(sizeof(vm_guest_mode_params)/sizeof(struct vm_guest_mode_params) == NUM_VM_MODES,  	       "Missing new mode params?"); -struct kvm_vm *____vm_create(enum vm_guest_mode mode, uint64_t nr_pages) +struct kvm_vm *____vm_create(enum vm_guest_mode mode)  {  	struct kvm_vm *vm; -	pr_debug("%s: mode='%s' pages='%ld'\n", __func__, -		 vm_guest_mode_string(mode), nr_pages); -  	vm = calloc(1, sizeof(*vm));  	TEST_ASSERT(vm != NULL, "Insufficient Memory"); @@ -288,9 +285,6 @@ struct kvm_vm *____vm_create(enum vm_guest_mode mode, uint64_t nr_pages)  	/* Allocate and setup memory for guest. */  	vm->vpages_mapped = sparsebit_alloc(); -	if (nr_pages != 0) -		vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, -					    0, 0, nr_pages, 0);  	return vm;  } @@ -337,8 +331,16 @@ struct kvm_vm *__vm_create(enum vm_guest_mode mode, uint32_t nr_runnable_vcpus,  						 nr_extra_pages);  	struct userspace_mem_region *slot0;  	struct kvm_vm *vm; +	int i; + +	pr_debug("%s: mode='%s' pages='%ld'\n", __func__, +		 vm_guest_mode_string(mode), nr_pages); -	vm = ____vm_create(mode, nr_pages); +	vm = ____vm_create(mode); + +	vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, 0, 0, nr_pages, 0); +	for (i = 0; i < NR_MEM_REGIONS; i++) +		vm->memslots[i] = 0;  	kvm_vm_elf_load(vm, program_invocation_name); @@ -649,6 +651,12 @@ static void __vm_mem_region_delete(struct kvm_vm *vm,  	sparsebit_free(®ion->unused_phy_pages);  	ret = munmap(region->mmap_start, region->mmap_size);  	TEST_ASSERT(!ret, __KVM_SYSCALL_ERROR("munmap()", ret)); +	if (region->fd >= 0) { +		/* There's an extra map when using shared memory. */ +		ret = munmap(region->mmap_alias, region->mmap_size); +		TEST_ASSERT(!ret, __KVM_SYSCALL_ERROR("munmap()", ret)); +		close(region->fd); +	}  	free(region);  } @@ -986,6 +994,7 @@ void vm_userspace_mem_region_add(struct kvm_vm *vm,  			    vm_mem_backing_src_alias(src_type)->name);  	} +	region->backing_src_type = src_type;  	region->unused_phy_pages = sparsebit_alloc();  	sparsebit_set_num(region->unused_phy_pages,  		guest_paddr >> vm->page_shift, npages); @@ -1280,32 +1289,15 @@ va_found:  	return pgidx_start * vm->page_size;  } -/* - * VM Virtual Address Allocate - * - * Input Args: - *   vm - Virtual Machine - *   sz - Size in bytes - *   vaddr_min - Minimum starting virtual address - * - * Output Args: None - * - * Return: - *   Starting guest virtual address - * - * Allocates at least sz bytes within the virtual address space of the vm - * given by vm.  The allocated bytes are mapped to a virtual address >= - * the address given by vaddr_min.  Note that each allocation uses a - * a unique set of pages, with the minimum real allocation being at least - * a page. - */ -vm_vaddr_t vm_vaddr_alloc(struct kvm_vm *vm, size_t sz, vm_vaddr_t vaddr_min) +vm_vaddr_t __vm_vaddr_alloc(struct kvm_vm *vm, size_t sz, vm_vaddr_t vaddr_min, +			    enum kvm_mem_region_type type)  {  	uint64_t pages = (sz >> vm->page_shift) + ((sz % vm->page_size) != 0);  	virt_pgd_alloc(vm);  	vm_paddr_t paddr = vm_phy_pages_alloc(vm, pages, -					      KVM_UTIL_MIN_PFN * vm->page_size, 0); +					      KVM_UTIL_MIN_PFN * vm->page_size, +					      vm->memslots[type]);  	/*  	 * Find an unused range of virtual page addresses of at least @@ -1326,6 +1318,30 @@ vm_vaddr_t vm_vaddr_alloc(struct kvm_vm *vm, size_t sz, vm_vaddr_t vaddr_min)  }  /* + * VM Virtual Address Allocate + * + * Input Args: + *   vm - Virtual Machine + *   sz - Size in bytes + *   vaddr_min - Minimum starting virtual address + * + * Output Args: None + * + * Return: + *   Starting guest virtual address + * + * Allocates at least sz bytes within the virtual address space of the vm + * given by vm.  The allocated bytes are mapped to a virtual address >= + * the address given by vaddr_min.  Note that each allocation uses a + * a unique set of pages, with the minimum real allocation being at least + * a page. The allocated physical space comes from the TEST_DATA memory region. + */ +vm_vaddr_t vm_vaddr_alloc(struct kvm_vm *vm, size_t sz, vm_vaddr_t vaddr_min) +{ +	return __vm_vaddr_alloc(vm, sz, vaddr_min, MEM_REGION_TEST_DATA); +} + +/*   * VM Virtual Address Allocate Pages   *   * Input Args: @@ -1344,6 +1360,11 @@ vm_vaddr_t vm_vaddr_alloc_pages(struct kvm_vm *vm, int nr_pages)  	return vm_vaddr_alloc(vm, nr_pages * getpagesize(), KVM_UTIL_MIN_VADDR);  } +vm_vaddr_t __vm_vaddr_alloc_page(struct kvm_vm *vm, enum kvm_mem_region_type type) +{ +	return __vm_vaddr_alloc(vm, getpagesize(), KVM_UTIL_MIN_VADDR, type); +} +  /*   * VM Virtual Address Allocate Page   * @@ -1570,7 +1591,7 @@ struct kvm_reg_list *vcpu_get_reg_list(struct kvm_vcpu *vcpu)  void *vcpu_map_dirty_ring(struct kvm_vcpu *vcpu)  { -	uint32_t page_size = vcpu->vm->page_size; +	uint32_t page_size = getpagesize();  	uint32_t size = vcpu->vm->dirty_ring_size;  	TEST_ASSERT(size > 0, "Should enable dirty ring first"); @@ -1911,7 +1932,8 @@ vm_paddr_t vm_phy_page_alloc(struct kvm_vm *vm, vm_paddr_t paddr_min,  vm_paddr_t vm_alloc_page_table(struct kvm_vm *vm)  { -	return vm_phy_page_alloc(vm, KVM_GUEST_PAGE_TABLE_MIN_PADDR, 0); +	return vm_phy_page_alloc(vm, KVM_GUEST_PAGE_TABLE_MIN_PADDR, +				 vm->memslots[MEM_REGION_PT]);  }  /* |