Received: by 2002:ad5:474a:0:0:0:0:0 with SMTP id i10csp5150002imu; Tue, 15 Jan 2019 12:08:12 -0800 (PST) X-Google-Smtp-Source: ALg8bN7ZCmY6ZNY1dFfy70URHPkjiftzBLMFXNM/BajHyANef9gvvFOvrJm3olYOaDdp0yvw6VWZ X-Received: by 2002:a63:1258:: with SMTP id 24mr5285018pgs.114.1547582892107; Tue, 15 Jan 2019 12:08:12 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1547582892; cv=none; d=google.com; s=arc-20160816; b=W1HcfSNIqWvFqkJTR3YJ0A1zTu/XQFb4hMaqAoMVWEDKPAmB6zGHbTskvS2+2fG+m+ o4mDInWgcHBSpJNVMWfin/vsKnWcoh48zdoMU9oQ6STjfGviv5xeqCqBNl4bNeqapibE GWhyVHAHuPZfg2x4017+CTFIekD4s7EoHUt4cIId0ivwnOEcu8ELSDIdhZ2cW742Uo1w uus9T4fTTMyT+wSevP9joGBd9/KHvLyL8WB9MKhCilEZWf9XYqyrBZO5qy+01w0f7qn3 fS7Z6VvtnlSXBhF/8ekSgO6URFa1dOiBTIeRizQbuKhwgyOfIpKXXUIMFsYmc4nDJu7u yKbw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=AFBFJFcr+JnB8yLi1+gHOBKZXIe8AUBHSgdPqE9+PeY=; b=AwLD0eWCmWzgY5LyrTYY4uEyP44CmDAv6ehxfOCCZ4Gr7PaWTV8axJRucRMmRj1yWO yphDtuGCPn1/dAOlbtcWmKy/C5+nSnwXwCRkEPo0rtnTurQU44IJIxEXZKQtrs6x/afM Av2W3VaeJjmVUrzHa+wL2zWab6m+3chRzkL7cz3jGy/SWOtH8j4EwOGwMIq+MAeOwSmR Ca7OQtZkDoMAYcT0UGRD6ey10/W8TpAIYFRhzOCwu9OG1qTXbRXQq0qveZRKRQuZHS/4 +dfxMoFVf+htvapqXWDbVY5NFd4NNmI39hHtHx3hkJhdqYPsioVI4fCMFBbAMtIoQmXY xiRA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=Ort12DRd; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id z29si828720pfi.9.2019.01.15.12.07.53; Tue, 15 Jan 2019 12:08:12 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=Ort12DRd; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1732693AbfAOQky (ORCPT + 99 others); Tue, 15 Jan 2019 11:40:54 -0500 Received: from mail.kernel.org ([198.145.29.99]:57374 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1732676AbfAOQkw (ORCPT ); Tue, 15 Jan 2019 11:40:52 -0500 Received: from localhost (5356596B.cm-6-7b.dynamic.ziggo.nl [83.86.89.107]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 6C6B62054F; Tue, 15 Jan 2019 16:40:50 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1547570450; bh=wKX+v7Ayh9D9Ayomi9/v940GNfK//n5ZNnK2yrvjfQo=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=Ort12DRdBJ7mVk0kX6wmbp/MtBNaGKfR7snfOLz83pc8Q6NDHT9A5HSqy41o6awvM 8W/zxsJtxFvauJBOYP88dzgtv8Nsh2BnU/S3Tmr1GYG3BZs2ymiWZYltEDqcFJ5D2l vp40w6/iOVwvbeuTlnfX9huy4mZ4N23Ejn6XR+sY= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Julien Thierry , Christoffer Dall , Marc Zyngier Subject: [PATCH 4.14 27/27] KVM: arm/arm64: Fix VMID alloc race by reverting to lock-less Date: Tue, 15 Jan 2019 17:36:16 +0100 Message-Id: <20190115154903.313894096@linuxfoundation.org> X-Mailer: git-send-email 2.20.1 In-Reply-To: <20190115154901.189747728@linuxfoundation.org> References: <20190115154901.189747728@linuxfoundation.org> User-Agent: quilt/0.65 X-stable: review X-Patchwork-Hint: ignore MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org 4.14-stable review patch. If anyone has any objections, please let me know. ------------------ From: Christoffer Dall commit fb544d1ca65a89f7a3895f7531221ceeed74ada7 upstream. We recently addressed a VMID generation race by introducing a read/write lock around accesses and updates to the vmid generation values. However, kvm_arch_vcpu_ioctl_run() also calls need_new_vmid_gen() but does so without taking the read lock. As far as I can tell, this can lead to the same kind of race: VM 0, VCPU 0 VM 0, VCPU 1 ------------ ------------ update_vttbr (vmid 254) update_vttbr (vmid 1) // roll over read_lock(kvm_vmid_lock); force_vm_exit() local_irq_disable need_new_vmid_gen == false //because vmid gen matches enter_guest (vmid 254) kvm_arch.vttbr = : read_unlock(kvm_vmid_lock); enter_guest (vmid 1) Which results in running two VCPUs in the same VM with different VMIDs and (even worse) other VCPUs from other VMs could now allocate clashing VMID 254 from the new generation as long as VCPU 0 is not exiting. Attempt to solve this by making sure vttbr is updated before another CPU can observe the updated VMID generation. Cc: stable@vger.kernel.org Fixes: f0cf47d939d0 "KVM: arm/arm64: Close VMID generation race" Reviewed-by: Julien Thierry Signed-off-by: Christoffer Dall Signed-off-by: Marc Zyngier Signed-off-by: Greg Kroah-Hartman --- virt/kvm/arm/arm.c | 23 +++++++++++------------ 1 file changed, 11 insertions(+), 12 deletions(-) --- a/virt/kvm/arm/arm.c +++ b/virt/kvm/arm/arm.c @@ -61,7 +61,7 @@ static DEFINE_PER_CPU(struct kvm_vcpu *, static atomic64_t kvm_vmid_gen = ATOMIC64_INIT(1); static u32 kvm_next_vmid; static unsigned int kvm_vmid_bits __read_mostly; -static DEFINE_RWLOCK(kvm_vmid_lock); +static DEFINE_SPINLOCK(kvm_vmid_lock); static bool vgic_present; @@ -447,7 +447,9 @@ void force_vm_exit(const cpumask_t *mask */ static bool need_new_vmid_gen(struct kvm *kvm) { - return unlikely(kvm->arch.vmid_gen != atomic64_read(&kvm_vmid_gen)); + u64 current_vmid_gen = atomic64_read(&kvm_vmid_gen); + smp_rmb(); /* Orders read of kvm_vmid_gen and kvm->arch.vmid */ + return unlikely(READ_ONCE(kvm->arch.vmid_gen) != current_vmid_gen); } /** @@ -462,16 +464,11 @@ static void update_vttbr(struct kvm *kvm { phys_addr_t pgd_phys; u64 vmid; - bool new_gen; - read_lock(&kvm_vmid_lock); - new_gen = need_new_vmid_gen(kvm); - read_unlock(&kvm_vmid_lock); - - if (!new_gen) + if (!need_new_vmid_gen(kvm)) return; - write_lock(&kvm_vmid_lock); + spin_lock(&kvm_vmid_lock); /* * We need to re-check the vmid_gen here to ensure that if another vcpu @@ -479,7 +476,7 @@ static void update_vttbr(struct kvm *kvm * use the same vmid. */ if (!need_new_vmid_gen(kvm)) { - write_unlock(&kvm_vmid_lock); + spin_unlock(&kvm_vmid_lock); return; } @@ -502,7 +499,6 @@ static void update_vttbr(struct kvm *kvm kvm_call_hyp(__kvm_flush_vm_context); } - kvm->arch.vmid_gen = atomic64_read(&kvm_vmid_gen); kvm->arch.vmid = kvm_next_vmid; kvm_next_vmid++; kvm_next_vmid &= (1 << kvm_vmid_bits) - 1; @@ -513,7 +509,10 @@ static void update_vttbr(struct kvm *kvm vmid = ((u64)(kvm->arch.vmid) << VTTBR_VMID_SHIFT) & VTTBR_VMID_MASK(kvm_vmid_bits); kvm->arch.vttbr = pgd_phys | vmid; - write_unlock(&kvm_vmid_lock); + smp_wmb(); + WRITE_ONCE(kvm->arch.vmid_gen, atomic64_read(&kvm_vmid_gen)); + + spin_unlock(&kvm_vmid_lock); } static int kvm_vcpu_first_run_init(struct kvm_vcpu *vcpu)