Received: by 2002:ac0:a5a6:0:0:0:0:0 with SMTP id m35-v6csp1125867imm; Sun, 2 Sep 2018 10:36:04 -0700 (PDT) X-Google-Smtp-Source: ANB0VdYXSpNmraOJS3phOwmYWwzwZ0qMhulM2KE2m6uly0dl6tvuedhO8vIOj2WUWjJIGNwD1+yV X-Received: by 2002:a63:c60:: with SMTP id 32-v6mr21565388pgm.155.1535909764782; Sun, 02 Sep 2018 10:36:04 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1535909764; cv=none; d=google.com; s=arc-20160816; b=Mb214+pRtRxhqor5LULToQhadaQOrfqKbttfUpfG0np7b+EaeQEIEpKgmqb/WcVmDu l/KQQyReELTcxrdKK3vyytnnHCLt/AXCPJRrO9JcAZ5n7CMm9UeoA396b4Wdee8Z4oxZ 87ksZ1mD81Ct8H2lhiEI1QXeVi8JjJA9SkaKxnDF6Oo5EzuLY+6yveWz6VoEgAnYm4ig T8SE7ErMaCsnBKPiL9eqlpnLxvmcMQI0V0wK3XWw9EN/L3+81o6q97V0crn9gkmYbgIL 4gMHqCwEDmUJij1k6knNaHkDxAReEOzA10p8VjG6xX8Kr0Y64Q2ZK35h76uvjiz4yi/K KiBQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:mime-version:references:in-reply-to :message-id:date:subject:cc:to:from:arc-authentication-results; bh=BBqho/WbWtmmFByBwL99+phg8sk9P3eRWf1OOEtwjX0=; b=me61Vd13bgUgfbseR+lGpkiIZcul3zhWeX3R5W9g7VZjwihy1JOLaCJi8gFHXdrd6o ar3FEVOaasrB2CxKC1p2LMConApsmLnL6F1TctRnwFT7IfKSLd5LRVbjPIzIqL5jyXEo lNMZbazvapg+TW8A9NHbogWvDwmTulQ6B/zd7gfqO+MN0rZKQ+xMm2NJF9oZd5fxYVMh SVsqb+6Uc8NXqcWatMCaXZyCTMLVOLXiBQjSkriqDdXzy0a7ZurXcsCwXASOfc/xfQai 3K2gDn0nA8Z5SS8rJH6GSWBGo84vTfkpY148D9uox6Jw7pBibyg92w2o5F9vnHfK1lbs A7WQ== ARC-Authentication-Results: i=1; mx.google.com; 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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=vmware.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id m1-v6si1484585pfd.60.2018.09.02.10.35.35; Sun, 02 Sep 2018 10:36:04 -0700 (PDT) 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; 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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=vmware.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727327AbeIBVuS (ORCPT + 99 others); Sun, 2 Sep 2018 17:50:18 -0400 Received: from ex13-edg-ou-001.vmware.com ([208.91.0.189]:48413 "EHLO EX13-EDG-OU-001.vmware.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727077AbeIBVuR (ORCPT ); Sun, 2 Sep 2018 17:50:17 -0400 Received: from sc9-mailhost3.vmware.com (10.113.161.73) by EX13-EDG-OU-001.vmware.com (10.113.208.155) with Microsoft SMTP Server id 15.0.1156.6; Sun, 2 Sep 2018 10:33:41 -0700 Received: from sc2-haas01-esx0118.eng.vmware.com (sc2-haas01-esx0118.eng.vmware.com [10.172.44.118]) by sc9-mailhost3.vmware.com (Postfix) with ESMTP id D41B64070E; Sun, 2 Sep 2018 10:33:42 -0700 (PDT) From: Nadav Amit To: Thomas Gleixner CC: , Ingo Molnar , , Arnd Bergmann , , Dave Hansen , Nadav Amit , Nadav Amit , Kees Cook , Peter Zijlstra , Dave Hansen Subject: [PATCH v2 4/6] x86/alternatives: initializing temporary mm for patching Date: Sun, 2 Sep 2018 10:32:22 -0700 Message-ID: <20180902173224.30606-5-namit@vmware.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20180902173224.30606-1-namit@vmware.com> References: <20180902173224.30606-1-namit@vmware.com> MIME-Version: 1.0 Content-Type: text/plain Received-SPF: None (EX13-EDG-OU-001.vmware.com: namit@vmware.com does not designate permitted sender hosts) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org To prevent improper use of the PTEs that are used for text patching, we want to use a temporary mm struct. We initailize it by copying the init mm. The address that will be used for patching is taken from the lower area that is usually used for the task memory. Doing so prevents the need to frequently synchronize the temporary-mm (e.g., when BPF programs are installed), since different PGDs are used for the task memory. Finally, we randomize the address of the PTEs to harden against exploits that use these PTEs. Cc: Kees Cook Cc: Peter Zijlstra Cc: Dave Hansen Reviewed-by: Masami Hiramatsu Tested-by: Masami Hiramatsu Suggested-by: Andy Lutomirski Signed-off-by: Nadav Amit --- arch/x86/include/asm/pgtable.h | 3 +++ arch/x86/include/asm/text-patching.h | 2 ++ arch/x86/mm/init_64.c | 29 ++++++++++++++++++++++++++++ init/main.c | 3 +++ 4 files changed, 37 insertions(+) diff --git a/arch/x86/include/asm/pgtable.h b/arch/x86/include/asm/pgtable.h index e4ffa565a69f..3de9a1fb7a9a 100644 --- a/arch/x86/include/asm/pgtable.h +++ b/arch/x86/include/asm/pgtable.h @@ -1022,6 +1022,9 @@ static inline void __meminit init_trampoline_default(void) /* Default trampoline pgd value */ trampoline_pgd_entry = init_top_pgt[pgd_index(__PAGE_OFFSET)]; } + +void __init poking_init(void); + # ifdef CONFIG_RANDOMIZE_MEMORY void __meminit init_trampoline(void); # else diff --git a/arch/x86/include/asm/text-patching.h b/arch/x86/include/asm/text-patching.h index e85ff65c43c3..ffe7902cc326 100644 --- a/arch/x86/include/asm/text-patching.h +++ b/arch/x86/include/asm/text-patching.h @@ -38,5 +38,7 @@ extern void *text_poke(void *addr, const void *opcode, size_t len); extern int poke_int3_handler(struct pt_regs *regs); extern void *text_poke_bp(void *addr, const void *opcode, size_t len, void *handler); extern int after_bootmem; +extern __ro_after_init struct mm_struct *poking_mm; +extern __ro_after_init unsigned long poking_addr; #endif /* _ASM_X86_TEXT_PATCHING_H */ diff --git a/arch/x86/mm/init_64.c b/arch/x86/mm/init_64.c index dd519f372169..db33a724a054 100644 --- a/arch/x86/mm/init_64.c +++ b/arch/x86/mm/init_64.c @@ -54,6 +54,7 @@ #include #include #include +#include #include "mm_internal.h" @@ -1389,6 +1390,34 @@ unsigned long memory_block_size_bytes(void) return memory_block_size_probed; } +/* + * Initialize an mm_struct to be used during poking and a pointer to be used + * during patching. If anything fails during initialization, poking will be done + * using the fixmap, which is unsafe, so warn the user about it. + */ +void __init poking_init(void) +{ + unsigned long poking_addr; + + poking_mm = copy_init_mm(); + if (!poking_mm) { + pr_err("x86/mm: error setting a separate poking address space"); + return; + } + + /* + * Randomize the poking address, but make sure that the following page + * will be mapped at the same PMD. We need 2 pages, so find space for 3, + * and adjust the address if the PMD ends after the first one. + */ + poking_addr = TASK_UNMAPPED_BASE + + (kaslr_get_random_long("Poking") & PAGE_MASK) % + (TASK_SIZE - TASK_UNMAPPED_BASE - 3 * PAGE_SIZE); + + if (((poking_addr + PAGE_SIZE) & ~PMD_MASK) == 0) + poking_addr += PAGE_SIZE; +} + #ifdef CONFIG_SPARSEMEM_VMEMMAP /* * Initialise the sparsemem vmemmap using huge-pages at the PMD level. diff --git a/init/main.c b/init/main.c index 18f8f0140fa0..8c6dd8d88fca 100644 --- a/init/main.c +++ b/init/main.c @@ -496,6 +496,8 @@ void __init __weak thread_stack_cache_init(void) void __init __weak mem_encrypt_init(void) { } +void __init __weak poking_init(void) { } + bool initcall_debug; core_param(initcall_debug, initcall_debug, bool, 0644); @@ -725,6 +727,7 @@ asmlinkage __visible void __init start_kernel(void) taskstats_init_early(); delayacct_init(); + poking_init(); check_bugs(); acpi_subsystem_init(); -- 2.17.1