lkdtm: Add a test for STACKLEAK
authorAlexander Popov <alex.popov@linux.com>
Thu, 16 Aug 2018 22:17:00 +0000 (01:17 +0300)
committerKees Cook <keescook@chromium.org>
Tue, 4 Sep 2018 17:35:47 +0000 (10:35 -0700)
Introduce an lkdtm test for the STACKLEAK feature: check that the
current task stack is properly erased (filled with STACKLEAK_POISON).

Signed-off-by: Alexander Popov <alex.popov@linux.com>
Signed-off-by: Tycho Andersen <tycho@tycho.ws>
Tested-by: Laura Abbott <labbott@redhat.com>
Signed-off-by: Kees Cook <keescook@chromium.org>
drivers/misc/lkdtm/Makefile
drivers/misc/lkdtm/core.c
drivers/misc/lkdtm/lkdtm.h
drivers/misc/lkdtm/stackleak.c [new file with mode: 0644]

index 3370a4138e942621a008573b1fa1ad6912b1c75c..951c984de61ae9e85af22eb05e76199c5752baa6 100644 (file)
@@ -8,7 +8,9 @@ lkdtm-$(CONFIG_LKDTM)           += perms.o
 lkdtm-$(CONFIG_LKDTM)          += refcount.o
 lkdtm-$(CONFIG_LKDTM)          += rodata_objcopy.o
 lkdtm-$(CONFIG_LKDTM)          += usercopy.o
+lkdtm-$(CONFIG_LKDTM)          += stackleak.o
 
+KASAN_SANITIZE_stackleak.o     := n
 KCOV_INSTRUMENT_rodata.o       := n
 
 OBJCOPYFLAGS :=
index 2154d1bfd18b610b60f0fc8409cf7bf8919f2aad..aca26d81e9b8d3345eb1543406817a31c5f1668b 100644 (file)
@@ -183,6 +183,7 @@ static const struct crashtype crashtypes[] = {
        CRASHTYPE(USERCOPY_STACK_FRAME_FROM),
        CRASHTYPE(USERCOPY_STACK_BEYOND),
        CRASHTYPE(USERCOPY_KERNEL),
+       CRASHTYPE(STACKLEAK_ERASING),
 };
 
 
index 9e513dcfd8093613dd767765829158aa3addc810..b611b157c84fae3bb659ed290d1ea35d2f53cf6f 100644 (file)
@@ -83,4 +83,7 @@ void lkdtm_USERCOPY_STACK_FRAME_FROM(void);
 void lkdtm_USERCOPY_STACK_BEYOND(void);
 void lkdtm_USERCOPY_KERNEL(void);
 
+/* lkdtm_stackleak.c */
+void lkdtm_STACKLEAK_ERASING(void);
+
 #endif
diff --git a/drivers/misc/lkdtm/stackleak.c b/drivers/misc/lkdtm/stackleak.c
new file mode 100644 (file)
index 0000000..d5a0844
--- /dev/null
@@ -0,0 +1,73 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * This code tests that the current task stack is properly erased (filled
+ * with STACKLEAK_POISON).
+ *
+ * Authors:
+ *   Alexander Popov <alex.popov@linux.com>
+ *   Tycho Andersen <tycho@tycho.ws>
+ */
+
+#include "lkdtm.h"
+#include <linux/stackleak.h>
+
+void lkdtm_STACKLEAK_ERASING(void)
+{
+       unsigned long *sp, left, found, i;
+       const unsigned long check_depth =
+                       STACKLEAK_SEARCH_DEPTH / sizeof(unsigned long);
+
+       /*
+        * For the details about the alignment of the poison values, see
+        * the comment in stackleak_track_stack().
+        */
+       sp = PTR_ALIGN(&i, sizeof(unsigned long));
+
+       left = ((unsigned long)sp & (THREAD_SIZE - 1)) / sizeof(unsigned long);
+       sp--;
+
+       /*
+        * One 'long int' at the bottom of the thread stack is reserved
+        * and not poisoned.
+        */
+       if (left > 1) {
+               left--;
+       } else {
+               pr_err("FAIL: not enough stack space for the test\n");
+               return;
+       }
+
+       pr_info("checking unused part of the thread stack (%lu bytes)...\n",
+                                       left * sizeof(unsigned long));
+
+       /*
+        * Search for 'check_depth' poison values in a row (just like
+        * stackleak_erase() does).
+        */
+       for (i = 0, found = 0; i < left && found <= check_depth; i++) {
+               if (*(sp - i) == STACKLEAK_POISON)
+                       found++;
+               else
+                       found = 0;
+       }
+
+       if (found <= check_depth) {
+               pr_err("FAIL: thread stack is not erased (checked %lu bytes)\n",
+                                               i * sizeof(unsigned long));
+               return;
+       }
+
+       pr_info("first %lu bytes are unpoisoned\n",
+                               (i - found) * sizeof(unsigned long));
+
+       /* The rest of thread stack should be erased */
+       for (; i < left; i++) {
+               if (*(sp - i) != STACKLEAK_POISON) {
+                       pr_err("FAIL: thread stack is NOT properly erased\n");
+                       return;
+               }
+       }
+
+       pr_info("OK: the rest of the thread stack is properly erased\n");
+       return;
+}