tests/ztest: Add ztest_test_pass()

ztest provides a ztest_test_fail() interface to fail the currently
running test, but does not provide an equivalent ztest_test_pass().
Normally a test passes just by returning without an assertion failure
or other call to ztest_test_fail().  However, if the correct behavior
for a test is to trigger a fatal fault (as with tests/kernel/fatal or
protection or MPU tests), then we need a way for the test to pass the
currently running test before aborting the current thread.
Otherwise, ztest hangs forever in run_test() on the
k_sem_take(&test_end_signal, K_FOREVER) call.  Add
a ztest_test_pass() interface and implement it for kernel and
userspace variants of ztest.  This interface will be used in the
protection tests.

Signed-off-by: Stephen Smalley <sds@tycho.nsa.gov>
diff --git a/tests/ztest/include/ztest_test.h b/tests/ztest/include/ztest_test.h
index b9e4f97..81f0cd7 100644
--- a/tests/ztest/include/ztest_test.h
+++ b/tests/ztest/include/ztest_test.h
@@ -41,6 +41,16 @@
 void ztest_test_fail(void);
 
 /**
+ * @brief Pass the currently running test.
+ *
+ * Normally a test passes just by returning without an assertion failure.
+ * However, if the success case for your test involves a fatal fault,
+ * you can call this function from _SysFatalErrorHandler to indicate that
+ * the test passed before aborting the thread.
+ */
+void ztest_test_pass(void);
+
+/**
  * @brief Do nothing, successfully.
  *
  * Unit test / setup function / teardown function that does
diff --git a/tests/ztest/src/ztest.c b/tests/ztest/src/ztest.c
index 488ecab..f3ddfd3 100644
--- a/tests/ztest/src/ztest.c
+++ b/tests/ztest/src/ztest.c
@@ -56,6 +56,7 @@
 #define FAIL_FAST 0
 
 static jmp_buf test_fail;
+static jmp_buf test_pass;
 static jmp_buf stack_fail;
 
 void ztest_test_fail(void)
@@ -63,6 +64,11 @@
 	raise(SIGABRT);
 }
 
+void ztest_test_pass(void)
+{
+	longjmp(test_pass, 1);
+}
+
 static void handle_signal(int sig)
 {
 	static const char *const phase_str[] = {
@@ -106,6 +112,11 @@
 		goto out;
 	}
 
+	if (setjmp(test_pass)) {
+		ret = TC_PASS;
+		goto out;
+	}
+
 	run_test_functions(test);
 out:
 	ret |= cleanup_test(test);
@@ -142,6 +153,13 @@
 	k_thread_abort(k_current_get());
 }
 
+void ztest_test_pass(void)
+{
+	test_result = 0;
+	k_sem_give(&test_end_signal);
+	k_thread_abort(k_current_get());
+}
+
 static void init_testing(void)
 {
 	k_sem_init(&test_end_signal, 0, 1);