Received: by 2002:a05:6a10:a0d1:0:0:0:0 with SMTP id j17csp930367pxa; Sat, 1 Aug 2020 10:39:59 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyPnIHFjbeXC9dPMxAW5Zqlxd5dnoSFxvl8KsQlE9AUDHv74oO2O5G0cdp+xyHNY/Me1iYE X-Received: by 2002:a17:906:a05a:: with SMTP id bg26mr9884145ejb.501.1596303598856; Sat, 01 Aug 2020 10:39:58 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1596303598; cv=none; d=google.com; s=arc-20160816; b=fCnhhJSHfVi2k8r/KIw+dhNrgue3mcefW+rtd6WPV3fQ8D1tbfziMLXVNNjKIyE3D/ FJfn50FUHBcaueE5ClqAHrcp3bf6712dgJeYXpJWcRMhwIPFH8+XL/MAsKBEyEkbcShv /hASzDTASgQpN8HdB2x/s0AVi9bUAEhyB0zv8HipdY4A3tDXINxmQ2Iw50fCu31/YolB 0af4en49ZXWh6wZMeu2t19IWBT+27zIVGdM399nNIDa4U2NboJk1uec4mr9Ln05/jYbb uTk7FBEEo0lNZPms0zj4t9c0jdCC4lRIrm0M/V2UGl5A3F8011jCqPYG6N/X5BJYNPWF bSZg== 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:cc:to:from :subject:ironport-sdr:ironport-sdr; bh=ntBOpCAxc5MZ+vukKryKUIr65SEdRhPBqyup1hTZf/Y=; b=rxD/3qimwGRielf0AS3IOvcMA3QpKpzMbhpbDJVRPJ5iDieQdR6fQsO3IYgpnlMmmf /fuUvi6td4QqtvNoOGTZSSUDmKOBUt7vMiyojVh8mKINmhTLN4dGn3SlRkUGyq2jFSCJ RAPi/bC37xzPKg0sM2CNdeBqLogBrZGQQ/8hFwIPzq0c4EIqkHZdiX5Ljo0CUqw04v82 cxx7S/8nqE/EvkNqSuOWRBX1FqzweJkd1AKSb4x/yeAUGNmICPflTBvJXccp0hal/TC2 CBUPCQmgCOmY7F/MZnxEk51KOs4Jqbma1uiDdGKKUwiLzyz0jmxMJ9mLbwwJGem5bSmF VqtA== ARC-Authentication-Results: i=1; mx.google.com; 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=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id g8si7190701ejm.100.2020.08.01.10.39.37; Sat, 01 Aug 2020 10:39:58 -0700 (PDT) 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; 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=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726955AbgHARjH (ORCPT + 99 others); Sat, 1 Aug 2020 13:39:07 -0400 Received: from mga07.intel.com ([134.134.136.100]:54364 "EHLO mga07.intel.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726906AbgHARjH (ORCPT ); Sat, 1 Aug 2020 13:39:07 -0400 IronPort-SDR: xXH/GV+rI3U5l6mBDdNFdYD2Eq9gUUZn0tRos1DnG3QK+ivGvgvT2MXo9iYBvfCub57cinV5a7 m+yO4C7LQz5w== X-IronPort-AV: E=McAfee;i="6000,8403,9700"; a="216379187" X-IronPort-AV: E=Sophos;i="5.75,423,1589266800"; d="scan'208";a="216379187" X-Amp-Result: SKIPPED(no attachment in message) X-Amp-File-Uploaded: False Received: from orsmga004.jf.intel.com ([10.7.209.38]) by orsmga105.jf.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 01 Aug 2020 10:39:06 -0700 IronPort-SDR: L58xj0bzWq/qheUBGFf9msfEY0MSHcK0VH/SK/anKb5cKsUtcW3tp5XqVrqk0A66xq4HJQF+r0 1XfJNpYAjztA== X-IronPort-AV: E=Sophos;i="5.75,423,1589266800"; d="scan'208";a="435752695" Received: from dwillia2-desk3.jf.intel.com (HELO dwillia2-desk3.amr.corp.intel.com) ([10.54.39.16]) by orsmga004-auth.jf.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 01 Aug 2020 10:39:06 -0700 Subject: [PATCH v8 2/2] x86/copy_mc: Introduce copy_mc_generic() From: Dan Williams To: tglx@linutronix.de, mingo@redhat.com, vishal.l.verma@intel.com Cc: x86@kernel.org, stable@vger.kernel.org, Borislav Petkov , Vivek Goyal , "H. Peter Anvin" , Andy Lutomirski , Peter Zijlstra , Linus Torvalds , Tony Luck , Erwin Tsaur , Erwin Tsaur , linux-nvdimm@lists.01.org, linux-kernel@vger.kernel.org Date: Sat, 01 Aug 2020 10:22:48 -0700 Message-ID: <159630256804.3143511.8894023468833792004.stgit@dwillia2-desk3.amr.corp.intel.com> In-Reply-To: <159630255616.3143511.18110575960499749012.stgit@dwillia2-desk3.amr.corp.intel.com> References: <159630255616.3143511.18110575960499749012.stgit@dwillia2-desk3.amr.corp.intel.com> User-Agent: StGit/0.18-3-g996c MIME-Version: 1.0 Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org The original copy_mc_fragile() implementation had negative performance implications since it did not use the fast-string instruction sequence to perform copies. For this reason copy_mc_to_kernel() fell back to plain memcpy() to preserve performance on platform that did not indicate the capability to recover from machine check exceptions. However, that capability detection was not architectural and now that some platforms can recover from fast-string consumption of memory errors the memcpy() fallback now causes these more capable platforms to fail. Introduce copy_mc_generic() as the fast default implementation of copy_mc_to_kernel() and finalize the transition of copy_mc_fragile() to be a platform quirk to indicate 'fragility'. With this in place copy_mc_to_kernel() is fast and recovery-ready by default regardless of hardware capability. Thanks to Vivek for identifying that copy_user_generic() is not suitable as the copy_mc_to_user() backend since the #MC handler explicitly checks ex_has_fault_handler(). Cc: x86@kernel.org Cc: Cc: Ingo Molnar Cc: Borislav Petkov Cc: Vivek Goyal Cc: "H. Peter Anvin" Cc: Andy Lutomirski Cc: Thomas Gleixner Cc: Peter Zijlstra Cc: Linus Torvalds Reviewed-by: Tony Luck Reported-by: Erwin Tsaur Tested-by: Erwin Tsaur Fixes: 92b0729c34ca ("x86/mm, x86/mce: Add memcpy_mcsafe()") Signed-off-by: Dan Williams --- arch/x86/include/asm/uaccess.h | 3 +++ arch/x86/lib/copy_mc.c | 12 +++++------- arch/x86/lib/copy_mc_64.S | 40 ++++++++++++++++++++++++++++++++++++++++ tools/objtool/check.c | 1 + 4 files changed, 49 insertions(+), 7 deletions(-) diff --git a/arch/x86/include/asm/uaccess.h b/arch/x86/include/asm/uaccess.h index 4b2082b61e3e..b038eda58958 100644 --- a/arch/x86/include/asm/uaccess.h +++ b/arch/x86/include/asm/uaccess.h @@ -464,6 +464,9 @@ copy_mc_to_user(void *to, const void *from, unsigned len); unsigned long __must_check copy_mc_fragile(void *dst, const void *src, unsigned cnt); + +unsigned long __must_check +copy_mc_generic(void *dst, const void *src, unsigned cnt); #else static inline void enable_copy_mc_fragile(void) { diff --git a/arch/x86/lib/copy_mc.c b/arch/x86/lib/copy_mc.c index cdb8f5dc403d..9e6fac1ab72e 100644 --- a/arch/x86/lib/copy_mc.c +++ b/arch/x86/lib/copy_mc.c @@ -23,7 +23,7 @@ void enable_copy_mc_fragile(void) * * Call into the 'fragile' version on systems that have trouble * actually do machine check recovery. Everyone else can just - * use memcpy(). + * use copy_mc_generic(). * * Return 0 for success, or number of bytes not copied if there was an * exception. @@ -33,8 +33,7 @@ copy_mc_to_kernel(void *dst, const void *src, unsigned cnt) { if (static_branch_unlikely(©_mc_fragile_key)) return copy_mc_fragile(dst, src, cnt); - memcpy(dst, src, cnt); - return 0; + return copy_mc_generic(dst, src, cnt); } EXPORT_SYMBOL_GPL(copy_mc_to_kernel); @@ -56,11 +55,10 @@ copy_mc_to_user(void *to, const void *from, unsigned len) { unsigned long ret; - if (!static_branch_unlikely(©_mc_fragile_key)) - return copy_user_generic(to, from, len); - __uaccess_begin(); - ret = copy_mc_fragile(to, from, len); + if (static_branch_unlikely(©_mc_fragile_key)) + ret = copy_mc_fragile(to, from, len); + ret = copy_mc_generic(to, from, len); __uaccess_end(); return ret; } diff --git a/arch/x86/lib/copy_mc_64.S b/arch/x86/lib/copy_mc_64.S index 35a67c50890b..a08e7a4d9e28 100644 --- a/arch/x86/lib/copy_mc_64.S +++ b/arch/x86/lib/copy_mc_64.S @@ -2,7 +2,9 @@ /* Copyright(c) 2016-2020 Intel Corporation. All rights reserved. */ #include +#include #include +#include #include #include @@ -122,4 +124,42 @@ EXPORT_SYMBOL_GPL(copy_mc_fragile) _ASM_EXTABLE(.L_write_leading_bytes, .E_leading_bytes) _ASM_EXTABLE(.L_write_words, .E_write_words) _ASM_EXTABLE(.L_write_trailing_bytes, .E_trailing_bytes) + +/* + * copy_mc_generic - memory copy with exception handling + * + * Fast string copy + fault / exception handling. If the CPU does + * support machine check exception recovery, but does not support + * recovering from fast-string exceptions then this CPU needs to be + * added to the copy_mc_fragile_key set of quirks. Otherwise, absent any + * machine check recovery support this version should be no slower than + * standard memcpy. + */ +SYM_FUNC_START(copy_mc_generic) + ALTERNATIVE "jmp copy_mc_fragile", "", X86_FEATURE_ERMS + movq %rdi, %rax + movq %rdx, %rcx +.L_copy: + rep movsb + /* Copy successful. Return zero */ + xorl %eax, %eax + ret +SYM_FUNC_END(copy_mc_generic) +EXPORT_SYMBOL_GPL(copy_mc_generic) + + .section .fixup, "ax" +.E_copy: + /* + * On fault %rcx is updated such that the copy instruction could + * optionally be restarted at the fault position, i.e. it + * contains 'bytes remaining'. A non-zero return indicates error + * to copy_mc_generic() users, or indicate short transfers to + * user-copy routines. + */ + movq %rcx, %rax + ret + + .previous + + _ASM_EXTABLE_FAULT(.L_copy, .E_copy) #endif diff --git a/tools/objtool/check.c b/tools/objtool/check.c index 17cb0933bf42..d2e1f01df10b 100644 --- a/tools/objtool/check.c +++ b/tools/objtool/check.c @@ -548,6 +548,7 @@ static const char *uaccess_safe_builtin[] = { "__ubsan_handle_shift_out_of_bounds", /* misc */ "csum_partial_copy_generic", + "copy_mc_generic", "copy_mc_fragile", "copy_mc_fragile_handle_tail", "ftrace_likely_update", /* CONFIG_TRACE_BRANCH_PROFILING */