Commit 2bb99540 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'gcc-plugins-v5.1-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/kees/linux

Pull gcc-plugins updates from Kees Cook:
 "This adds additional type coverage to the existing structleak plugin
  and adds a large set of selftests to help evaluate stack variable
  zero-initialization coverage.

  That can be used to test whatever instrumentation might be performing
  zero-initialization: either with the structleak plugin or with Clang's
  coming "-ftrivial-auto-var-init=zero" option.

  Summary:

   - Add scalar and array initialization coverage

   - Refactor Kconfig to make options more clear

   - Add self-test module for testing automatic initialization"

* tag 'gcc-plugins-v5.1-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/kees/linux:
  lib: Introduce test_stackinit module
  gcc-plugins: structleak: Generalize to all variable types
parents a39f009a 50ceaa95
...@@ -2016,6 +2016,16 @@ config TEST_OBJAGG ...@@ -2016,6 +2016,16 @@ config TEST_OBJAGG
(or module load). (or module load).
config TEST_STACKINIT
tristate "Test level of stack variable initialization"
help
Test if the kernel is zero-initializing stack variables and
padding. Coverage is controlled by compiler flags,
CONFIG_GCC_PLUGIN_STRUCTLEAK, CONFIG_GCC_PLUGIN_STRUCTLEAK_BYREF,
or CONFIG_GCC_PLUGIN_STRUCTLEAK_BYREF_ALL.
If unsure, say N.
endif # RUNTIME_TESTING_MENU endif # RUNTIME_TESTING_MENU
config MEMTEST config MEMTEST
......
...@@ -77,6 +77,7 @@ obj-$(CONFIG_TEST_KMOD) += test_kmod.o ...@@ -77,6 +77,7 @@ obj-$(CONFIG_TEST_KMOD) += test_kmod.o
obj-$(CONFIG_TEST_DEBUG_VIRTUAL) += test_debug_virtual.o obj-$(CONFIG_TEST_DEBUG_VIRTUAL) += test_debug_virtual.o
obj-$(CONFIG_TEST_MEMCAT_P) += test_memcat_p.o obj-$(CONFIG_TEST_MEMCAT_P) += test_memcat_p.o
obj-$(CONFIG_TEST_OBJAGG) += test_objagg.o obj-$(CONFIG_TEST_OBJAGG) += test_objagg.o
obj-$(CONFIG_TEST_STACKINIT) += test_stackinit.o
obj-$(CONFIG_TEST_LIVEPATCH) += livepatch/ obj-$(CONFIG_TEST_LIVEPATCH) += livepatch/
......
This diff is collapsed.
...@@ -15,6 +15,8 @@ gcc-plugin-$(CONFIG_GCC_PLUGIN_SANCOV) += sancov_plugin.so ...@@ -15,6 +15,8 @@ gcc-plugin-$(CONFIG_GCC_PLUGIN_SANCOV) += sancov_plugin.so
gcc-plugin-$(CONFIG_GCC_PLUGIN_STRUCTLEAK) += structleak_plugin.so gcc-plugin-$(CONFIG_GCC_PLUGIN_STRUCTLEAK) += structleak_plugin.so
gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK_VERBOSE) \ gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK_VERBOSE) \
+= -fplugin-arg-structleak_plugin-verbose += -fplugin-arg-structleak_plugin-verbose
gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK_BYREF) \
+= -fplugin-arg-structleak_plugin-byref
gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK_BYREF_ALL) \ gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK_BYREF_ALL) \
+= -fplugin-arg-structleak_plugin-byref-all += -fplugin-arg-structleak_plugin-byref-all
gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK) \ gcc-plugin-cflags-$(CONFIG_GCC_PLUGIN_STRUCTLEAK) \
......
...@@ -67,23 +67,59 @@ config GCC_PLUGIN_LATENT_ENTROPY ...@@ -67,23 +67,59 @@ config GCC_PLUGIN_LATENT_ENTROPY
* https://pax.grsecurity.net/ * https://pax.grsecurity.net/
config GCC_PLUGIN_STRUCTLEAK config GCC_PLUGIN_STRUCTLEAK
bool "Force initialization of variables containing userspace addresses" bool "Zero initialize stack variables"
help help
This plugin zero-initializes any structures containing a While the kernel is built with warnings enabled for any missed
__user attribute. This can prevent some classes of information stack variable initializations, this warning is silenced for
exposures. anything passed by reference to another function, under the
occasionally misguided assumption that the function will do
This plugin was ported from grsecurity/PaX. More information at: the initialization. As this regularly leads to exploitable
flaws, this plugin is available to identify and zero-initialize
such variables, depending on the chosen level of coverage.
This plugin was originally ported from grsecurity/PaX. More
information at:
* https://grsecurity.net/ * https://grsecurity.net/
* https://pax.grsecurity.net/ * https://pax.grsecurity.net/
config GCC_PLUGIN_STRUCTLEAK_BYREF_ALL choice
bool "Force initialize all struct type variables passed by reference" prompt "Coverage"
depends on GCC_PLUGIN_STRUCTLEAK depends on GCC_PLUGIN_STRUCTLEAK
depends on !COMPILE_TEST default GCC_PLUGIN_STRUCTLEAK_BYREF_ALL
help help
Zero initialize any struct type local variable that may be passed by This chooses the level of coverage over classes of potentially
reference without having been initialized. uninitialized variables. The selected class will be
zero-initialized before use.
config GCC_PLUGIN_STRUCTLEAK_USER
bool "structs marked for userspace"
help
Zero-initialize any structures on the stack containing
a __user attribute. This can prevent some classes of
uninitialized stack variable exploits and information
exposures, like CVE-2013-2141:
https://git.kernel.org/linus/b9e146d8eb3b9eca
config GCC_PLUGIN_STRUCTLEAK_BYREF
bool "structs passed by reference"
help
Zero-initialize any structures on the stack that may
be passed by reference and had not already been
explicitly initialized. This can prevent most classes
of uninitialized stack variable exploits and information
exposures, like CVE-2017-1000410:
https://git.kernel.org/linus/06e7e776ca4d3654
config GCC_PLUGIN_STRUCTLEAK_BYREF_ALL
bool "anything passed by reference"
help
Zero-initialize any stack variables that may be passed
by reference and had not already been explicitly
initialized. This is intended to eliminate all classes
of uninitialized stack variable exploits and information
exposures.
endchoice
config GCC_PLUGIN_STRUCTLEAK_VERBOSE config GCC_PLUGIN_STRUCTLEAK_VERBOSE
bool "Report forcefully initialized variables" bool "Report forcefully initialized variables"
......
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
* Options: * Options:
* -fplugin-arg-structleak_plugin-disable * -fplugin-arg-structleak_plugin-disable
* -fplugin-arg-structleak_plugin-verbose * -fplugin-arg-structleak_plugin-verbose
* -fplugin-arg-structleak_plugin-byref
* -fplugin-arg-structleak_plugin-byref-all * -fplugin-arg-structleak_plugin-byref-all
* *
* Usage: * Usage:
...@@ -26,7 +27,6 @@ ...@@ -26,7 +27,6 @@
* $ gcc -fplugin=./structleak_plugin.so test.c -O2 * $ gcc -fplugin=./structleak_plugin.so test.c -O2
* *
* TODO: eliminate redundant initializers * TODO: eliminate redundant initializers
* increase type coverage
*/ */
#include "gcc-common.h" #include "gcc-common.h"
...@@ -37,13 +37,18 @@ ...@@ -37,13 +37,18 @@
__visible int plugin_is_GPL_compatible; __visible int plugin_is_GPL_compatible;
static struct plugin_info structleak_plugin_info = { static struct plugin_info structleak_plugin_info = {
.version = "201607271510vanilla", .version = "20190125vanilla",
.help = "disable\tdo not activate plugin\n" .help = "disable\tdo not activate plugin\n"
"verbose\tprint all initialized variables\n", "byref\tinit structs passed by reference\n"
"byref-all\tinit anything passed by reference\n"
"verbose\tprint all initialized variables\n",
}; };
#define BYREF_STRUCT 1
#define BYREF_ALL 2
static bool verbose; static bool verbose;
static bool byref_all; static int byref;
static tree handle_user_attribute(tree *node, tree name, tree args, int flags, bool *no_add_attrs) static tree handle_user_attribute(tree *node, tree name, tree args, int flags, bool *no_add_attrs)
{ {
...@@ -118,6 +123,7 @@ static void initialize(tree var) ...@@ -118,6 +123,7 @@ static void initialize(tree var)
gimple_stmt_iterator gsi; gimple_stmt_iterator gsi;
tree initializer; tree initializer;
gimple init_stmt; gimple init_stmt;
tree type;
/* this is the original entry bb before the forced split */ /* this is the original entry bb before the forced split */
bb = single_succ(ENTRY_BLOCK_PTR_FOR_FN(cfun)); bb = single_succ(ENTRY_BLOCK_PTR_FOR_FN(cfun));
...@@ -148,11 +154,15 @@ static void initialize(tree var) ...@@ -148,11 +154,15 @@ static void initialize(tree var)
if (verbose) if (verbose)
inform(DECL_SOURCE_LOCATION(var), inform(DECL_SOURCE_LOCATION(var),
"%s variable will be forcibly initialized", "%s variable will be forcibly initialized",
(byref_all && TREE_ADDRESSABLE(var)) ? "byref" (byref && TREE_ADDRESSABLE(var)) ? "byref"
: "userspace"); : "userspace");
/* build the initializer expression */ /* build the initializer expression */
initializer = build_constructor(TREE_TYPE(var), NULL); type = TREE_TYPE(var);
if (AGGREGATE_TYPE_P(type))
initializer = build_constructor(type, NULL);
else
initializer = fold_convert(type, integer_zero_node);
/* build the initializer stmt */ /* build the initializer stmt */
init_stmt = gimple_build_assign(var, initializer); init_stmt = gimple_build_assign(var, initializer);
...@@ -184,13 +194,13 @@ static unsigned int structleak_execute(void) ...@@ -184,13 +194,13 @@ static unsigned int structleak_execute(void)
if (!auto_var_in_fn_p(var, current_function_decl)) if (!auto_var_in_fn_p(var, current_function_decl))
continue; continue;
/* only care about structure types */ /* only care about structure types unless byref-all */
if (TREE_CODE(type) != RECORD_TYPE && TREE_CODE(type) != UNION_TYPE) if (byref != BYREF_ALL && TREE_CODE(type) != RECORD_TYPE && TREE_CODE(type) != UNION_TYPE)
continue; continue;
/* if the type is of interest, examine the variable */ /* if the type is of interest, examine the variable */
if (TYPE_USERSPACE(type) || if (TYPE_USERSPACE(type) ||
(byref_all && TREE_ADDRESSABLE(var))) (byref && TREE_ADDRESSABLE(var)))
initialize(var); initialize(var);
} }
...@@ -232,8 +242,12 @@ __visible int plugin_init(struct plugin_name_args *plugin_info, struct plugin_gc ...@@ -232,8 +242,12 @@ __visible int plugin_init(struct plugin_name_args *plugin_info, struct plugin_gc
verbose = true; verbose = true;
continue; continue;
} }
if (!strcmp(argv[i].key, "byref")) {
byref = BYREF_STRUCT;
continue;
}
if (!strcmp(argv[i].key, "byref-all")) { if (!strcmp(argv[i].key, "byref-all")) {
byref_all = true; byref = BYREF_ALL;
continue; continue;
} }
error(G_("unknown option '-fplugin-arg-%s-%s'"), plugin_name, argv[i].key); error(G_("unknown option '-fplugin-arg-%s-%s'"), plugin_name, argv[i].key);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment