Received: by 2002:a05:6902:102b:0:0:0:0 with SMTP id x11csp3778703ybt; Tue, 23 Jun 2020 10:29:16 -0700 (PDT) X-Google-Smtp-Source: ABdhPJzSKxCs/FS7fB3SOGW2nJh/X5ZBJYHyRC7kADuyRokbfgyS5TMxvdTIUsLetfmWNX05lK3U X-Received: by 2002:a50:f297:: with SMTP id f23mr22499202edm.222.1592933356557; Tue, 23 Jun 2020 10:29:16 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1592933356; cv=none; d=google.com; s=arc-20160816; b=Z6Uvy0+qACWjd7mOktnd053bbIcii75qkKjJr2FXOXaRZjcFXGk2t20BDiD6Cb8civ Hvu5Uvwro4qNVImXpZJb/ywbll6hRpUDz9xfpmv/iYeIWcX0c3BV3BDWyVTnFJEVgQBy ogx4799MuYvt8s/S8nLtogXlcl8YbxVpZCRc82sxHlQtverSdzQuyOjZyoVKvOfYLCfL gGEJJB6R79ypX8db6f6dVyrJE2ckTVmzvmKGR4J+fcEQgF49FGrGW/pl+eoV5UqL6xBg T8TAxIU/s7cAh36KH7Vms+MMFuwrB7shsZmZNAXc59QwG+hFzgEwBu0ZQmFTipwfS6DY WanQ== 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 :references:in-reply-to:message-id:date:subject:cc:to:from :ironport-sdr:ironport-sdr; bh=PvZTOrIt6/dGWw2M6xZL24kqJ1Sbe3KOEC6L74i2Ro0=; b=Yvr1pz1td6pPv8UU1h2n2hDhNzqiJj0ovLANjAIqZ0ZGeHFPqIEx6dLnLyYZDVOFkc MgqHWidl0GCjrNOc3tNxq58RFs4p8RMXLxQC5VH4lBrpdetQwUZUpHx4nWyp6fSxyjI4 +rjlJAYvH4fRqgan5JkFOjw6LCCQzdn++fikNt3I6mrRXXaM5RveZ3ngHXK6P7hOeY42 TDn3aGbNHsAExDwQIbtYucEbQJAwFh3yHJ2G9vXoth91jNOFfgfdYas6P5YPCA4VVWik +Tq5TVQ5b9AhiFoFfSy1haPmM3TzV5Zc5SJMwh3Kb0861DoWs8jEl2bRZrWIIkmqGbcb vjZw== 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 v3si12023504eja.251.2020.06.23.10.28.53; Tue, 23 Jun 2020 10:29:16 -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 S1733184AbgFWRZC (ORCPT + 99 others); Tue, 23 Jun 2020 13:25:02 -0400 Received: from mga03.intel.com ([134.134.136.65]:17064 "EHLO mga03.intel.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1733173AbgFWRY7 (ORCPT ); Tue, 23 Jun 2020 13:24:59 -0400 IronPort-SDR: 9zsascDBMbVXqjMcLJU/8hOOH6gVX/BUwXh/ZKcSmq/FlNBvQ/a++3QXxesI9sCy+ihFBUPkFx aPSSf4IneX/Q== X-IronPort-AV: E=McAfee;i="6000,8403,9661"; a="144170852" X-IronPort-AV: E=Sophos;i="5.75,272,1589266800"; d="scan'208";a="144170852" X-Amp-Result: SKIPPED(no attachment in message) X-Amp-File-Uploaded: False Received: from orsmga004.jf.intel.com ([10.7.209.38]) by orsmga103.jf.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 23 Jun 2020 10:24:58 -0700 IronPort-SDR: 6FKo5AYJO89uy2TXBhYaEwaeeIctRMkHeZWGpt5FsEfjVkRWN8B3u+8RM+if4UejheJbDZWIyo a/5Q+bRKp53Q== X-ExtLoop1: 1 X-IronPort-AV: E=Sophos;i="5.75,272,1589266800"; d="scan'208";a="423080181" Received: from kcaccard-mobl.amr.corp.intel.com (HELO kcaccard-mobl1.jf.intel.com) ([10.213.182.184]) by orsmga004.jf.intel.com with ESMTP; 23 Jun 2020 10:24:54 -0700 From: Kristen Carlson Accardi To: keescook@chromium.org, tglx@linutronix.de, mingo@redhat.com, bp@alien8.de, x86@kernel.org, "H. Peter Anvin" , Jessica Yu Cc: arjan@linux.intel.com, linux-kernel@vger.kernel.org, kernel-hardening@lists.openwall.com, rick.p.edgecombe@intel.com, Kristen Carlson Accardi , Ard Biesheuvel , Tony Luck Subject: [PATCH v3 10/10] module: Reorder functions Date: Tue, 23 Jun 2020 10:23:27 -0700 Message-Id: <20200623172327.5701-11-kristen@linux.intel.com> X-Mailer: git-send-email 2.20.1 In-Reply-To: <20200623172327.5701-1-kristen@linux.intel.com> References: <20200623172327.5701-1-kristen@linux.intel.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Introduce a new config option to allow modules to be re-ordered by function. This option can be enabled independently of the kernel text KASLR or FG_KASLR settings so that it can be used by architectures that do not support either of these features. This option will be selected by default if CONFIG_FG_KASLR is selected. If a module has functions split out into separate text sections (i.e. compiled with the -ffunction-sections flag), reorder the functions to provide some code diversification to modules. Signed-off-by: Kristen Carlson Accardi Reviewed-by: Kees Cook Acked-by: Ard Biesheuvel Tested-by: Ard Biesheuvel Reviewed-by: Tony Luck Tested-by: Tony Luck --- arch/x86/Makefile | 5 +++ init/Kconfig | 12 +++++++ kernel/kallsyms.c | 2 +- kernel/module.c | 81 +++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 99 insertions(+), 1 deletion(-) diff --git a/arch/x86/Makefile b/arch/x86/Makefile index 00e378de8bc0..0f2dbc46eb5c 100644 --- a/arch/x86/Makefile +++ b/arch/x86/Makefile @@ -51,6 +51,11 @@ ifdef CONFIG_X86_NEED_RELOCS LDFLAGS_vmlinux := --emit-relocs --discard-none endif +ifndef CONFIG_FG_KASLR + ifdef CONFIG_MODULE_FG_KASLR + KBUILD_CFLAGS_MODULE += -ffunction-sections + endif +endif # # Prevent GCC from generating any FP code by mistake. # diff --git a/init/Kconfig b/init/Kconfig index e29c032e4d66..87706a08a2ca 100644 --- a/init/Kconfig +++ b/init/Kconfig @@ -1994,6 +1994,7 @@ config FG_KASLR bool "Function Granular Kernel Address Space Layout Randomization" depends on $(cc-option, -ffunction-sections) depends on ARCH_HAS_FG_KASLR + select MODULE_FG_KASLR default n help This option improves the randomness of the kernel text @@ -2278,6 +2279,17 @@ config UNUSED_KSYMS_WHITELIST one per line. The path can be absolute, or relative to the kernel source tree. +config MODULE_FG_KASLR + depends on $(cc-option, -ffunction-sections) + bool "Module Function Granular Layout Randomization" + help + This option randomizes the module text section by reordering the text + section by function at module load time. In order to use this + feature, the module must have been compiled with the + -ffunction-sections compiler flag. + + If unsure, say N. + endif # MODULES config MODULES_TREE_LOOKUP diff --git a/kernel/kallsyms.c b/kernel/kallsyms.c index df2b20e1b7f2..da65593bffc7 100644 --- a/kernel/kallsyms.c +++ b/kernel/kallsyms.c @@ -761,7 +761,7 @@ int get_all_symbol_name(void *data, const char *name, struct module *mod, return 0; } -#if defined(CONFIG_FG_KASLR) +#if defined(CONFIG_FG_KASLR) || defined(CONFIG_MODULE_FG_KASLR) /* * When fine grained kaslr is enabled, we need to * print out the symbols sorted by name rather than by diff --git a/kernel/module.c b/kernel/module.c index e8a198588f26..ff1e82b54127 100644 --- a/kernel/module.c +++ b/kernel/module.c @@ -56,6 +56,7 @@ #include #include #include +#include #include #include "module-internal.h" @@ -2388,6 +2389,83 @@ static long get_offset(struct module *mod, unsigned int *size, return ret; } +/* + * shuffle_text_list() + * Use a Fisher Yates algorithm to shuffle a list of text sections. + */ +static void shuffle_text_list(Elf_Shdr **list, int size) +{ + int i; + unsigned int j; + Elf_Shdr *temp; + + for (i = size - 1; i > 0; i--) { + /* + * pick a random index from 0 to i + */ + get_random_bytes(&j, sizeof(j)); + j = j % (i + 1); + + temp = list[i]; + list[i] = list[j]; + list[j] = temp; + } +} + +/* + * randomize_text() + * Look through the core section looking for executable code sections. + * Store sections in an array and then shuffle the sections + * to reorder the functions. + */ +static void randomize_text(struct module *mod, struct load_info *info) +{ + int i; + int num_text_sections = 0; + Elf_Shdr **text_list; + int size = 0; + int max_sections = info->hdr->e_shnum; + unsigned int sec = find_sec(info, ".text"); + + if (sec == 0) + return; + + text_list = kmalloc_array(max_sections, sizeof(*text_list), GFP_KERNEL); + if (!text_list) + return; + + for (i = 0; i < max_sections; i++) { + Elf_Shdr *shdr = &info->sechdrs[i]; + const char *sname = info->secstrings + shdr->sh_name; + + if (!(shdr->sh_flags & SHF_ALLOC) || + !(shdr->sh_flags & SHF_EXECINSTR) || + strstarts(sname, ".init")) + continue; + + text_list[num_text_sections] = shdr; + num_text_sections++; + } + + shuffle_text_list(text_list, num_text_sections); + + for (i = 0; i < num_text_sections; i++) { + Elf_Shdr *shdr = text_list[i]; + + /* + * get_offset has a section index for it's last + * argument, that is only used by arch_mod_section_prepend(), + * which is only defined by parisc. Since this this type + * of randomization isn't supported on parisc, we can + * safely pass in zero as the last argument, as it is + * ignored. + */ + shdr->sh_entsize = get_offset(mod, &size, shdr, 0); + } + + kfree(text_list); +} + /* Lay out the SHF_ALLOC sections in a way not dissimilar to how ld might -- code, read-only data, read-write data, small data. Tally sizes, and place the offsets into sh_entsize fields: high bit means it @@ -2478,6 +2556,9 @@ static void layout_sections(struct module *mod, struct load_info *info) break; } } + + if (IS_ENABLED(CONFIG_MODULE_FG_KASLR)) + randomize_text(mod, info); } static void set_license(struct module *mod, const char *license) -- 2.20.1