Received: by 2002:a05:6a10:16a7:0:0:0:0 with SMTP id gp39csp1523919pxb; Wed, 4 Nov 2020 11:13:51 -0800 (PST) X-Google-Smtp-Source: ABdhPJyyUBKthaVZ0MTfdra2dkAwSxytF5wbcQwTTRMQ+fvQfYAzrB/iif6Fpn6HlcsrwCvdOT5t X-Received: by 2002:aa7:dd17:: with SMTP id i23mr8624585edv.276.1604517230955; Wed, 04 Nov 2020 11:13:50 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1604517230; cv=none; d=google.com; s=arc-20160816; b=tIqXC17HwGz8N3hhljvjRqIqtakMS9Nbp9TranwvV0pR9eJlkO9sqzMHm+F+MuESSJ h4NiJE4vYB18QzQdBpxgearn3ObAmgCLFol03VXOdUUTgOKiSyVsxCJBBuZWM+xG1mpg aEmv+WZtT2Nhzv3it3sfUhF8t8q20cZsOrW57LDDNldz7m7rSkY3Ig/u8VeB9Igv+LGb 0WNeMxl99FOp5gpycsQ+BVsWhLBX7ZFXj7iEQc+naKNLonm2twI7mv+lRj14y7vNAB0N LxeK+TkVEILBRYLXahecFXZsU0MCLwnQR7BEGy3OxTxvRm92CQZ4HVpd7KXe9IOABK3m Sz5A== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature; bh=4axi3BZMWI251/Z2i3b7o8wb6S4cFpifUi2vsGe+rz4=; b=X0lA7oFFPBDO0NA20JCCYn6BNVRZWf/N/PobRz/HYtqso3U+KzEhy8pITv439B+wjC rLLhDBL0EL5g+pCeMMH5japYBj2T7iz0zZ5FQ4EIvqN6aWQ31hFSVnvk2uQRym9vvnlg ArXOBpzsql00Npf8NobB3ThOFwERvkqUn3UNcJ/QLiG6urLJbLYQ0cqSbqDDOO17AUXc pYPcc2fYeS1BD0KeqZJ1nu70jirKgZFae3NBdAIydjzYrgPLf3lr4L5aAJzvbXv9BN3W 5z63VCgpQglAPYHZMU26VsGKiVVGBxfwB8y8gzhikP2p9IkirIuP/N7351t6kU3r9RTK mgwg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=PTFxrsxx; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id o6si2485827edi.562.2020.11.04.11.13.28; Wed, 04 Nov 2020 11:13:50 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=PTFxrsxx; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1732592AbgKDShb (ORCPT + 99 others); Wed, 4 Nov 2020 13:37:31 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:56582 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1731163AbgKDSh3 (ORCPT ); Wed, 4 Nov 2020 13:37:29 -0500 Received: from mail-wm1-x342.google.com (mail-wm1-x342.google.com [IPv6:2a00:1450:4864:20::342]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 29C73C0613D3 for ; Wed, 4 Nov 2020 10:37:29 -0800 (PST) Received: by mail-wm1-x342.google.com with SMTP id h22so3301892wmb.0 for ; Wed, 04 Nov 2020 10:37:29 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=from:to:cc:subject:date:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=4axi3BZMWI251/Z2i3b7o8wb6S4cFpifUi2vsGe+rz4=; b=PTFxrsxxpSERkmBRu9fPY5EKuAOc5OMAk+oP0QxNUjwk1I4qOAxjjuDjwlbiIQiOi4 ENPQ8dyCyKiCtZYIUoXP4qNzgO1iQmho4X26GDMojC0ezu7o+eSB9bOWq4XwW/fs9VsO GK+U7yFays8yxLyrBp6Ib20GJbzuKSQMyHfvfLWNZ2hziMNn+LNIYgMyU308/fzbV2UF OHxgxnoPZgOGtvfH4FD78QPYcZo6KN3V9aMoEobtBl8VI6DoiYPHBWqXUbTXgtrG9Xbp WcUVQDUTe6Bb2sVj3CdVNgiY1YkUaOtdTU0HThcbNldjcSi8LHR5ARyKviCnMhX0Jb7t IyDQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=4axi3BZMWI251/Z2i3b7o8wb6S4cFpifUi2vsGe+rz4=; b=Rk9KE6ZuYnvJID3LiD53EG5yUdNCkREXUouTePtRMujfg5mS7RdaRPHbKO1iP37q5Y b3KUzEFkGlDyvOZ9/vB2cqvzJg4IZNtCLN3b29gQy9gtHPxF+EW8VYq7ZtZ3rsu7rhPZ 1gHQufyE3eAjY6zxWecsFryvHx+8Vw6N6Ba90DDIWzMDhgwGhXFrq99Wjp/4C6vCON0b lkgwJjmfDU2tGjLMiG/1l6xZShuywS9quwI6O0R/lmrxyhMa8WAJ+XDB34MoXaR6KU4r FjQfEYa5tKWSZL/Z2WADU6K8b/VioXjvxYjxguYxecy+Cezay0gSD4MgBJ2y3YeIVsID bSTg== X-Gm-Message-State: AOAM531hqBlrtR5n4zVWxZyARpCDNfo1J8gMfal7bkjNXxqFVZKPTGSa rTfhJxPcWWSR8t5m8Wiaz4j5uw== X-Received: by 2002:a1c:5545:: with SMTP id j66mr6336527wmb.31.1604515047599; Wed, 04 Nov 2020 10:37:27 -0800 (PST) Received: from localhost ([2a01:4b00:8523:2d03:c8d2:30f2:53c6:bc2]) by smtp.gmail.com with ESMTPSA id s11sm3661835wrm.56.2020.11.04.10.37.25 (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Wed, 04 Nov 2020 10:37:26 -0800 (PST) From: David Brazdil To: kvmarm@lists.cs.columbia.edu Cc: linux-arm-kernel@lists.infradead.org, linux-kernel@vger.kernel.org, Marc Zyngier , James Morse , Julien Thierry , Suzuki K Poulose , Catalin Marinas , Will Deacon , Dennis Zhou , Tejun Heo , Christoph Lameter , Mark Rutland , Lorenzo Pieralisi , Quentin Perret , Andrew Scull , kernel-team@android.com, David Brazdil Subject: [RFC PATCH 23/26] kvm: arm64: Intercept host's CPU_SUSPEND PSCI SMCs Date: Wed, 4 Nov 2020 18:36:27 +0000 Message-Id: <20201104183630.27513-24-dbrazdil@google.com> X-Mailer: git-send-email 2.29.1 In-Reply-To: <20201104183630.27513-1-dbrazdil@google.com> References: <20201104183630.27513-1-dbrazdil@google.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Add a handler of CPU_SUSPEND host PSCI SMCs. When invoked, it determines whether the requested power state loses context, ie. whether it is indistinguishable from a WHI or whether it is a deeper sleep state that behaves like a CPU_OFF+CPU_ON. If it's the former, it forwards the call to EL3 and returns to the host after waking up. If it's the latter, it saves r0,pc of the host into and makes the same call to EL3 with the hyp CPU entry point. When the core wakes up, EL2 state is initialized before dropping back to EL1. Signed-off-by: David Brazdil --- arch/arm64/kvm/arm.c | 2 ++ arch/arm64/kvm/hyp/nvhe/psci.c | 49 +++++++++++++++++++++++++++++++++- drivers/firmware/psci/psci.c | 9 ------- include/uapi/linux/psci.h | 7 +++++ 4 files changed, 57 insertions(+), 10 deletions(-) diff --git a/arch/arm64/kvm/arm.c b/arch/arm64/kvm/arm.c index 166975999ead..6fbda652200b 100644 --- a/arch/arm64/kvm/arm.c +++ b/arch/arm64/kvm/arm.c @@ -1521,9 +1521,11 @@ static void init_psci(void) { extern u32 kvm_nvhe_sym(kvm_host_psci_version); extern u32 kvm_nvhe_sym(kvm_host_psci_function_id)[PSCI_FN_MAX]; + extern u32 kvm_nvhe_sym(kvm_host_psci_cpu_suspend_feature); int cpu; kvm_nvhe_sym(kvm_host_psci_version) = psci_driver_version; + kvm_nvhe_sym(kvm_host_psci_cpu_suspend_feature) = psci_cpu_suspend_feature; memcpy(kvm_nvhe_sym(kvm_host_psci_function_id), psci_function_id, sizeof(psci_function_id)); diff --git a/arch/arm64/kvm/hyp/nvhe/psci.c b/arch/arm64/kvm/hyp/nvhe/psci.c index 42ee5effa827..4899c8319bb4 100644 --- a/arch/arm64/kvm/hyp/nvhe/psci.c +++ b/arch/arm64/kvm/hyp/nvhe/psci.c @@ -21,6 +21,7 @@ /* Config options set by the host. */ u32 kvm_host_psci_version = PSCI_VERSION(0, 0); u32 kvm_host_psci_function_id[PSCI_FN_MAX]; +u32 kvm_host_psci_cpu_suspend_feature; s64 hyp_physvirt_offset; #define __hyp_pa(x) ((phys_addr_t)(x) + hyp_physvirt_offset) @@ -83,6 +84,20 @@ static __noreturn unsigned long psci_forward_noreturn(struct kvm_cpu_context *ho hyp_panic(); /* unreachable */ } +static bool psci_has_ext_power_state(void) +{ + return kvm_host_psci_cpu_suspend_feature & PSCI_1_0_FEATURES_CPU_SUSPEND_PF_MASK; +} + +static bool psci_power_state_loses_context(u32 state) +{ + const u32 mask = psci_has_ext_power_state() ? + PSCI_1_0_EXT_POWER_STATE_TYPE_MASK : + PSCI_0_2_POWER_STATE_TYPE_MASK; + + return state & mask; +} + static unsigned int find_cpu_id(u64 mpidr) { int i; @@ -106,6 +121,34 @@ static phys_addr_t cpu_entry_pa(void) return kern_va - kimage_voffset; } +static int psci_cpu_suspend(u64 func_id, struct kvm_cpu_context *host_ctxt) +{ + u64 power_state = host_ctxt->regs.regs[1]; + unsigned long pc = host_ctxt->regs.regs[2]; + unsigned long r0 = host_ctxt->regs.regs[3]; + hyp_spinlock_t *cpu_lock; + struct vcpu_reset_state *cpu_reset; + struct kvm_nvhe_init_params *cpu_params; + + if (!psci_power_state_loses_context(power_state)) { + /* This power state has the same semantics as WFI. */ + return psci_call(PSCI_0_2_FN64_CPU_SUSPEND, 0, 0, 0); + } + + cpu_lock = this_cpu_ptr(&psci_cpu_lock); + cpu_reset = this_cpu_ptr(&psci_cpu_reset); + cpu_params = this_cpu_ptr(&kvm_init_params); + + /* Resuming from this state has the same semantics as CPU_ON. */ + hyp_spin_lock(cpu_lock); + *cpu_reset = (struct vcpu_reset_state){ + .pc = pc, + .r0 = r0, + }; + hyp_spin_unlock(cpu_lock); + return psci_call(func_id, power_state, cpu_entry_pa(), __hyp_pa(cpu_params)); +} + static int psci_cpu_off(u64 func_id, struct kvm_cpu_context *host_ctxt) { hyp_spinlock_t *cpu_lock = this_cpu_ptr(&psci_cpu_lock); @@ -193,7 +236,9 @@ static int psci_cpu_on(u64 func_id, struct kvm_cpu_context *host_ctxt) static unsigned long psci_0_1_handler(u64 func_id, struct kvm_cpu_context *host_ctxt) { - if (func_id == kvm_host_psci_function_id[PSCI_FN_CPU_OFF]) + if (func_id == kvm_host_psci_function_id[PSCI_FN_CPU_SUSPEND]) + return psci_cpu_suspend(func_id, host_ctxt); + else if (func_id == kvm_host_psci_function_id[PSCI_FN_CPU_OFF]) return psci_cpu_off(func_id, host_ctxt); else if (func_id == kvm_host_psci_function_id[PSCI_FN_CPU_ON]) return psci_cpu_on(func_id, host_ctxt); @@ -216,6 +261,8 @@ static unsigned long psci_0_2_handler(u64 func_id, struct kvm_cpu_context *host_ case PSCI_0_2_FN_SYSTEM_RESET: psci_forward_noreturn(host_ctxt); unreachable(); + case PSCI_0_2_FN64_CPU_SUSPEND: + return psci_cpu_suspend(func_id, host_ctxt); case PSCI_0_2_FN_CPU_OFF: return psci_cpu_off(func_id, host_ctxt); case PSCI_0_2_FN64_CPU_ON: diff --git a/drivers/firmware/psci/psci.c b/drivers/firmware/psci/psci.c index b6ad237b1518..387e24409da7 100644 --- a/drivers/firmware/psci/psci.c +++ b/drivers/firmware/psci/psci.c @@ -62,15 +62,6 @@ static psci_fn *invoke_psci_fn; u32 psci_function_id[PSCI_FN_MAX]; -#define PSCI_0_2_POWER_STATE_MASK \ - (PSCI_0_2_POWER_STATE_ID_MASK | \ - PSCI_0_2_POWER_STATE_TYPE_MASK | \ - PSCI_0_2_POWER_STATE_AFFL_MASK) - -#define PSCI_1_0_EXT_POWER_STATE_MASK \ - (PSCI_1_0_EXT_POWER_STATE_ID_MASK | \ - PSCI_1_0_EXT_POWER_STATE_TYPE_MASK) - u32 psci_cpu_suspend_feature; static bool psci_system_reset2_supported; diff --git a/include/uapi/linux/psci.h b/include/uapi/linux/psci.h index 0d52b8dbe8c2..df3d85ce86f7 100644 --- a/include/uapi/linux/psci.h +++ b/include/uapi/linux/psci.h @@ -65,6 +65,10 @@ #define PSCI_0_2_POWER_STATE_AFFL_SHIFT 24 #define PSCI_0_2_POWER_STATE_AFFL_MASK \ (0x3 << PSCI_0_2_POWER_STATE_AFFL_SHIFT) +#define PSCI_0_2_POWER_STATE_MASK \ + (PSCI_0_2_POWER_STATE_ID_MASK | \ + PSCI_0_2_POWER_STATE_TYPE_MASK | \ + PSCI_0_2_POWER_STATE_AFFL_MASK) /* PSCI extended power state encoding for CPU_SUSPEND function */ #define PSCI_1_0_EXT_POWER_STATE_ID_MASK 0xfffffff @@ -72,6 +76,9 @@ #define PSCI_1_0_EXT_POWER_STATE_TYPE_SHIFT 30 #define PSCI_1_0_EXT_POWER_STATE_TYPE_MASK \ (0x1 << PSCI_1_0_EXT_POWER_STATE_TYPE_SHIFT) +#define PSCI_1_0_EXT_POWER_STATE_MASK \ + (PSCI_1_0_EXT_POWER_STATE_ID_MASK | \ + PSCI_1_0_EXT_POWER_STATE_TYPE_MASK) /* PSCI v0.2 affinity level state returned by AFFINITY_INFO */ #define PSCI_0_2_AFFINITY_LEVEL_ON 0 -- 2.29.1.341.ge80a0c044ae-goog