From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from bombadil.infradead.org (bombadil.infradead.org [198.137.202.133]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.lore.kernel.org (Postfix) with ESMTPS id C72D5CD5BB0 for ; Wed, 20 May 2026 15:27:54 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=lists.infradead.org; s=bombadil.20210309; h=Sender:List-Subscribe:List-Help :List-Post:List-Archive:List-Unsubscribe:List-Id:Content-Type:Cc:To:From: Subject:Message-ID:References:Mime-Version:In-Reply-To:Date:Reply-To: Content-Transfer-Encoding:Content-ID:Content-Description:Resent-Date: Resent-From:Resent-Sender:Resent-To:Resent-Cc:Resent-Message-ID:List-Owner; bh=tpWpj3dv5OUE+vpY373CnggGjNL8Kcfew3TEOiqcEjU=; b=t+Xs32NaYFPiaHnW1LCdFWW1HD jF3a5nOB1fR4XbXWV3OLl0scOcxRY5bENDskOxbBMw1RM1hZAli1cIfb9v/gz3Z2p4+yZqJ28wd25 KwYg68bGT2xRVQlJ1Ft3Z9QJvs2CG7ejBaA2JtdUqL53vRed9nILVeGWfQBR8lssyQ2YNqrMd3Nbk Mrn8gNT09xqiBbmUDjlgpU36g2xbI3gkOfqR0wqQQhzhyFO6/JzWXUxlRx/qImQ5P50CdpkUYpiUR 1nzNaezu2096Ag7WvmDx25Q11lgzKfLHCde0iHtQkb8Qvr7wmlTYig83U1vXJkBKHqSIVpCHYcria ldn+iBcQ==; Received: from localhost ([::1] helo=bombadil.infradead.org) by bombadil.infradead.org with esmtp (Exim 4.99.1 #2 (Red Hat Linux)) id 1wPipn-00000004zR8-3vuU; Wed, 20 May 2026 15:27:47 +0000 Received: from casper.infradead.org ([2001:8b0:10b:1236::1]) by bombadil.infradead.org with esmtps (Exim 4.99.1 #2 (Red Hat Linux)) id 1wPipc-00000004zDv-3QC5 for linux-arm-kernel@bombadil.infradead.org; Wed, 20 May 2026 15:27:38 +0000 DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=infradead.org; s=casper.20170209; h=Content-Type:Cc:To:From:Subject: Message-ID:References:Mime-Version:In-Reply-To:Date:Sender:Reply-To: Content-Transfer-Encoding:Content-ID:Content-Description; bh=tpWpj3dv5OUE+vpY373CnggGjNL8Kcfew3TEOiqcEjU=; b=KjSB2n/vUVCSIlzFa2+aBPUIsh So5y7dbJF6tyXJXiaILsDJDkmU3ybI4Qtk+DoxyBRRJJSC9xjWdHOnoq3ACrNWNAlB/NqZqpP3flb ph7tRIBnxRzH5cUZCuPa775CusuehWL2VK5JG67iciqu8IlzOUipue4piNPXizo8MdpRUroI/zL0g r9YwPbHPg7G+zhsl06acEn0yYxM3rhFUE1W0vQyUc8yP+63RGPUjU3fpoqscMIc2SJUuABrg4Pvfk vgdTKTHVN0uO7rRbG1nBERxQQPQfj8GCz1TYp/xVDd6AKDgKWGhBN2h+aUa4vYrcFbT/y+jAd9N3d DIGkxF3g==; Received: from mail-wr1-x44a.google.com ([2a00:1450:4864:20::44a]) by casper.infradead.org with esmtps (Exim 4.99.1 #2 (Red Hat Linux)) id 1wPipZ-00000007J3I-0e9Z for linux-arm-kernel@lists.infradead.org; Wed, 20 May 2026 15:27:35 +0000 Received: by mail-wr1-x44a.google.com with SMTP id ffacd0b85a97d-43d7b7bacddso3770081f8f.0 for ; Wed, 20 May 2026 08:27:32 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20251104; t=1779290851; x=1779895651; darn=lists.infradead.org; h=cc:to:from:subject:message-id:references:mime-version:in-reply-to :date:from:to:cc:subject:date:message-id:reply-to; bh=tpWpj3dv5OUE+vpY373CnggGjNL8Kcfew3TEOiqcEjU=; b=IlMR51Yyz3PmemZbAO8zxupqqj7tOmboVEF2Gh7fEZCdS1mZpGT9rLCaAnDAlD1Zst YQyfecVteRZtFEHomyL/eprHe4j8oYcQkkfHLRyV7n9KhngXM+Wvjjz3RdI+aMsGyRsT zCEJKubhYWd0OMnOZDKP3EMkpW+F6HrCOxlK2qt23RRJjhOlFaHOis7MHw2sZNZKkbsf 8BcD6Ii5phoT4PVtu5nyLg7dUHqUfCyC9ZqWd24UN2ycvoY62lUd03OpmYeDumiUJxIH B1xQ9pgDDlnDmo6QtQN/KZc3ZRF85ItcL9AsWkYaUnxwAdv+thgYLvpbVx8PGgbfTBeN 1eyQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20251104; t=1779290851; x=1779895651; h=cc:to:from:subject:message-id:references:mime-version:in-reply-to :date:x-gm-message-state:from:to:cc:subject:date:message-id:reply-to; bh=tpWpj3dv5OUE+vpY373CnggGjNL8Kcfew3TEOiqcEjU=; b=hgY96cCtYCHqRZ+S7lhKyStogILSkEgCedyttOt05qn1cquhBDOPIxO/uK9E+UDK8w dkM2N83BsGZsB7IdDv0m0U6bNUwg44lVwaheR5JenuGhKckC68JhRW2Rta6JSU3HPX8j vXQkz67e6oiSTF5bTpH0fkOgut0nUkGm1yWdUwSUl5yeL/ahFdGRVCPeeeMeeGmNup3a R1ETlmPhUB8HLXTx8lpz5s+Zx6xZW7arnv6nj86ctB7Jvf7DUlA8h2esdBQ8YMjHc7W8 kfahrd1TOj2j41i3oe3mdZ5EBCiRy2a6IZmZnpVwu31Fnk5BK0j5ext6avMGozQf7C87 5DWg== X-Gm-Message-State: AOJu0Yxv534igGcD5EjWvpvFL3ePgofN0Wk3furP2R+fUpvc8lI8Yyv9 zR+zfgL99kW2wsLWN0Y6hftXuzGQbst+hJAqBQGXFY7OP3THarsXBVYGqiMXN7f/Dn28df/KmSy 3SDPGOudqaQxFJMjBh/UoPQ== X-Received: from wrzq17.prod.google.com ([2002:a05:6000:1371:b0:44c:2531:86ce]) (user=vdonnefort job=prod-delivery.src-stubby-dispatcher) by 2002:a05:6000:400d:b0:449:af52:6167 with SMTP id ffacd0b85a97d-45e5c58fd33mr40232381f8f.8.1779290850878; Wed, 20 May 2026 08:27:30 -0700 (PDT) Date: Wed, 20 May 2026 16:26:48 +0100 In-Reply-To: <20260520152650.4107895-1-vdonnefort@google.com> Mime-Version: 1.0 References: <20260520152650.4107895-1-vdonnefort@google.com> X-Mailer: git-send-email 2.54.0.631.ge1b05301d1-goog Message-ID: <20260520152650.4107895-16-vdonnefort@google.com> Subject: [PATCH 15/17] KVM: arm64: Alloc pkvm_hyp_vm using pKVM heap allocator From: Vincent Donnefort To: maz@kernel.org, oliver.upton@linux.dev, joey.gouly@arm.com, suzuki.poulose@arm.com, yuzenghui@huawei.com, catalin.marinas@arm.com, will@kernel.org Cc: linux-arm-kernel@lists.infradead.org, kvmarm@lists.linux.dev, kernel-team@android.com, qperret@google.com, tabba@google.com, Vincent Donnefort Content-Type: text/plain; charset="UTF-8" X-CRM114-Version: 20100106-BlameMichelson ( TRE 0.9.0 (BSD) ) MR-646709E3 X-CRM114-CacheID: sfid-20260520_162733_284292_E25B0686 X-CRM114-Status: GOOD ( 19.18 ) X-BeenThere: linux-arm-kernel@lists.infradead.org X-Mailman-Version: 2.1.34 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Sender: "linux-arm-kernel" Errors-To: linux-arm-kernel-bounces+linux-arm-kernel=archiver.kernel.org@lists.infradead.org Transition the allocation of the hypervisor VM state structure (pkvm_hyp_vm) from the host to the hypervisor using the new pKVM heap allocator (hyp_alloc()). Previously, the host was responsible for calculating the size of, allocating, and donating memory for pkvm_hyp_vm during VM creation. With the heap allocator in place, the hypervisor now allocates this structure dynamically at EL2. Use the pkvm_call_hyp_req() wrapper in the host to invoke __pkvm_init_vm, which automatically handles any top-up requests if the hypervisor runs out of heap memory during allocation. Signed-off-by: Vincent Donnefort diff --git a/arch/arm64/kvm/hyp/hyp-constants.c b/arch/arm64/kvm/hyp/hyp-constants.c index b257a3b4bfc5..501ab35a3840 100644 --- a/arch/arm64/kvm/hyp/hyp-constants.c +++ b/arch/arm64/kvm/hyp/hyp-constants.c @@ -7,7 +7,6 @@ int main(void) { DEFINE(STRUCT_HYP_PAGE_SIZE, sizeof(struct hyp_page)); - DEFINE(PKVM_HYP_VM_SIZE, sizeof(struct pkvm_hyp_vm)); DEFINE(PKVM_HYP_VCPU_SIZE, sizeof(struct pkvm_hyp_vcpu)); return 0; } diff --git a/arch/arm64/kvm/hyp/include/nvhe/pkvm.h b/arch/arm64/kvm/hyp/include/nvhe/pkvm.h index 624367d0ef5b..8e930c8729af 100644 --- a/arch/arm64/kvm/hyp/include/nvhe/pkvm.h +++ b/arch/arm64/kvm/hyp/include/nvhe/pkvm.h @@ -82,8 +82,7 @@ void pkvm_hyp_vm_table_init(void *tbl); int __pkvm_reserve_vm(void); void __pkvm_unreserve_vm(pkvm_handle_t handle); -int __pkvm_init_vm(struct kvm *host_kvm, unsigned long vm_hva, - unsigned long pgd_hva); +int __pkvm_init_vm(struct kvm *host_kvm, void *pgd); int __pkvm_init_vcpu(pkvm_handle_t handle, struct kvm_vcpu *host_vcpu, unsigned long vcpu_hva); diff --git a/arch/arm64/kvm/hyp/nvhe/hyp-main.c b/arch/arm64/kvm/hyp/nvhe/hyp-main.c index 4e7db8b48614..ebd6b5c09928 100644 --- a/arch/arm64/kvm/hyp/nvhe/hyp-main.c +++ b/arch/arm64/kvm/hyp/nvhe/hyp-main.c @@ -556,14 +556,30 @@ static void handle___pkvm_unreserve_vm(struct kvm_cpu_context *host_ctxt) __pkvm_unreserve_vm(handle); } +static void errno_to_smccc(int ret, struct kvm_cpu_context *host_ctxt) +{ + struct pkvm_hyp_req req = { .type = PKVM_HYP_NO_REQ }; + + switch (ret) { + case -ENOMEM: + req.type = PKVM_HYP_REQ_HYP_ALLOC; + req.mem.nr_pages = hyp_alloc_topup_needed(); + break; + } + + cpu_reg(host_ctxt, 1) = ret; + pkvm_hyp_req_to_smccc(host_ctxt, &req); +} + static void handle___pkvm_init_vm(struct kvm_cpu_context *host_ctxt) { DECLARE_REG(struct kvm *, host_kvm, host_ctxt, 1); - DECLARE_REG(unsigned long, vm_hva, host_ctxt, 2); - DECLARE_REG(unsigned long, pgd_hva, host_ctxt, 3); + DECLARE_REG(unsigned long, pgd_hva, host_ctxt, 2); + void *pgd; host_kvm = kern_hyp_va(host_kvm); - cpu_reg(host_ctxt, 1) = __pkvm_init_vm(host_kvm, vm_hva, pgd_hva); + pgd = (void *)kern_hyp_va(pgd_hva); + errno_to_smccc(__pkvm_init_vm(host_kvm, pgd), host_ctxt); } static void handle___pkvm_init_vcpu(struct kvm_cpu_context *host_ctxt) diff --git a/arch/arm64/kvm/hyp/nvhe/pkvm.c b/arch/arm64/kvm/hyp/nvhe/pkvm.c index 3e7f7606a3da..7405626e103a 100644 --- a/arch/arm64/kvm/hyp/nvhe/pkvm.c +++ b/arch/arm64/kvm/hyp/nvhe/pkvm.c @@ -11,6 +11,7 @@ #include +#include #include #include #include @@ -783,24 +784,22 @@ void teardown_selftest_vm(void) * Unmap the donated memory from the host at stage 2. * * host_kvm: A pointer to the host's struct kvm. - * vm_hva: The host va of the area being donated for the VM state. - * Must be page aligned. - * pgd_hva: The host va of the area being donated for the stage-2 PGD for - * the VM. Must be page aligned. Its size is implied by the VM's - * VTCR. + * pgd: The va of the area being donated for the stage-2 PGD for the VM. Must + * be page aligned. Its size is implied by the VM's VTCR. * * Return 0 success, negative error code on failure. */ -int __pkvm_init_vm(struct kvm *host_kvm, unsigned long vm_hva, - unsigned long pgd_hva) +int __pkvm_init_vm(struct kvm *host_kvm, void *pgd) { struct pkvm_hyp_vm *hyp_vm = NULL; size_t vm_size, pgd_size; unsigned int nr_vcpus; pkvm_handle_t handle; - void *pgd = NULL; int ret; + if (!PAGE_ALIGNED(pgd)) + return -EINVAL; + ret = hyp_pin_shared_mem(host_kvm, host_kvm + 1); if (ret) return ret; @@ -820,15 +819,15 @@ int __pkvm_init_vm(struct kvm *host_kvm, unsigned long vm_hva, vm_size = pkvm_get_hyp_vm_size(nr_vcpus); pgd_size = kvm_pgtable_stage2_pgd_size(host_mmu.arch.mmu.vtcr); - ret = -ENOMEM; - - hyp_vm = map_donated_memory(vm_hva, vm_size); - if (!hyp_vm) - goto err_remove_mappings; + hyp_vm = hyp_alloc(vm_size); + if (!hyp_vm) { + ret = hyp_alloc_errno(); + goto err_unpin_kvm; + } - pgd = map_donated_memory_noclear(pgd_hva, pgd_size); - if (!pgd) - goto err_remove_mappings; + ret = __pkvm_host_donate_hyp(hyp_virt_to_pfn(pgd), PAGE_ALIGN(pgd_size) >> PAGE_SHIFT); + if (ret) + goto err_free_hyp_vm; init_pkvm_hyp_vm(host_kvm, hyp_vm, nr_vcpus, handle); @@ -844,8 +843,9 @@ int __pkvm_init_vm(struct kvm *host_kvm, unsigned long vm_hva, return 0; err_remove_mappings: - unmap_donated_memory(hyp_vm, vm_size); unmap_donated_memory_noclear(pgd, pgd_size); +err_free_hyp_vm: + hyp_free(hyp_vm); err_unpin_kvm: hyp_unpin_shared_mem(host_kvm, host_kvm + 1); return ret; @@ -981,7 +981,6 @@ int __pkvm_finalize_teardown_vm(pkvm_handle_t handle) struct pkvm_hyp_vm *hyp_vm; struct kvm *host_kvm; unsigned int idx; - size_t vm_size; int err; hyp_spin_lock(&vm_table_lock); @@ -1024,8 +1023,7 @@ int __pkvm_finalize_teardown_vm(pkvm_handle_t handle) teardown_donated_memory(mc, hyp_vcpu, sizeof(*hyp_vcpu)); } - vm_size = pkvm_get_hyp_vm_size(hyp_vm->kvm.created_vcpus); - teardown_donated_memory(mc, hyp_vm, vm_size); + hyp_free(hyp_vm); hyp_unpin_shared_mem(host_kvm, host_kvm + 1); return 0; diff --git a/arch/arm64/kvm/pkvm.c b/arch/arm64/kvm/pkvm.c index 15281ae1be39..8fc2e954d382 100644 --- a/arch/arm64/kvm/pkvm.c +++ b/arch/arm64/kvm/pkvm.c @@ -216,8 +216,8 @@ static int __pkvm_create_hyp_vcpu(struct kvm_vcpu *vcpu) */ static int __pkvm_create_hyp_vm(struct kvm *kvm) { - size_t pgd_sz, hyp_vm_sz; - void *pgd, *hyp_vm; + size_t pgd_sz; + void *pgd; int ret; if (kvm->created_vcpus < 1) @@ -234,28 +234,15 @@ static int __pkvm_create_hyp_vm(struct kvm *kvm) if (!pgd) return -ENOMEM; - /* Allocate memory to donate to hyp for vm and vcpu pointers. */ - hyp_vm_sz = PAGE_ALIGN(size_add(PKVM_HYP_VM_SIZE, - size_mul(sizeof(void *), - kvm->created_vcpus))); - hyp_vm = alloc_pages_exact(hyp_vm_sz, GFP_KERNEL_ACCOUNT); - if (!hyp_vm) { - ret = -ENOMEM; - goto free_pgd; - } - - /* Donate the VM memory to hyp and let hyp initialize it. */ - ret = kvm_call_hyp_nvhe(__pkvm_init_vm, kvm, hyp_vm, pgd); + ret = pkvm_call_hyp_req(__pkvm_init_vm, kvm, pgd); if (ret) - goto free_vm; + goto free_pgd; kvm->arch.pkvm.is_created = true; init_hyp_stage2_memcache(&kvm->arch.pkvm.stage2_teardown_mc); kvm_account_pgtable_pages(pgd, pgd_sz / PAGE_SIZE); return 0; -free_vm: - free_pages_exact(hyp_vm, hyp_vm_sz); free_pgd: free_pages_exact(pgd, pgd_sz); return ret; -- 2.54.0.631.ge1b05301d1-goog