Skip to content

Commit 60b64fb

Browse files
committed
Added macros to support noreturn functions
Added TEST_PROTECT_NORETURN(), TEST_DO_NOT_RETURN(), and TEST_NOT_RETURNING(call) to test for non-returning functions. Mock functions that are declared noreturn shall use TEST_DO_NOT_RETURN() instead of return.
1 parent 6eaf693 commit 60b64fb

File tree

2 files changed

+67
-0
lines changed

2 files changed

+67
-0
lines changed

docs/UnityAssertionsReference.md

+44
Original file line numberDiff line numberDiff line change
@@ -222,6 +222,50 @@ This can be useful for outputting `INFO` messages into the Unity output stream
222222
without actually ending the test. Like pass and fail messages, it will be output
223223
with the filename and line number.
224224
225+
#### `TEST_PROTECT_NORETURN()`
226+
227+
This macro prepares an environment to test a function that is declared `noreturn`.
228+
229+
```
230+
if(TEST_PROTECT_NORETURN()){
231+
}
232+
```
233+
234+
The function `mock_go_elsewhere()` would use `TEST_DO_NOT_RETURN()`
235+
instead of returning.
236+
237+
#### `TEST_NOT_RETURNING(call)`
238+
239+
Calls a function, and the test fails if the function returns.
240+
Instead, the called function should use `TEST_DO_NOT_RETURN()`.
241+
242+
```
243+
/* prepare the test here */
244+
TEST_NOT_RETURNING(go_elsewhere(42));
245+
```
246+
247+
Note: this can be used to call a mock function (that uses
248+
`TEST_NOT_RETURNING()` instead of returning), or an actual function
249+
that effectively does not return. In the later case, the control
250+
won't resume to the caller test (unless there's a failure), so the
251+
test should expect that and have set up things accordingly.
252+
253+
This cannot be used if `UNITY_EXCLUDE_SETJMP_H` is defined.
254+
255+
#### `TEST_DO_NOT_RETURN()`
256+
257+
Aborts the execution, and resume after the current
258+
`TEST_NOT_RETURNING()`, successfully.
259+
260+
This should be used in mock functions declared `noreturn` instead of
261+
returning (explicitely or implicitely).
262+
263+
Note: actual functions that don't return will go wherever they shoud
264+
go and it's assumed this is handled in the test. These macros are
265+
intended to be used by mock functions.
266+
267+
This cannot be used if `UNITY_EXCLUDE_SETJMP_H` is defined.
268+
225269
### Boolean
226270
227271
#### `TEST_ASSERT (condition)`

src/unity_internals.h

+23
Original file line numberDiff line numberDiff line change
@@ -518,6 +518,7 @@ struct UNITY_STORAGE_T
518518
#endif
519519
#ifndef UNITY_EXCLUDE_SETJMP_H
520520
jmp_buf AbortFrame;
521+
jmp_buf NoReturnFrame;
521522
#endif
522523
};
523524

@@ -788,6 +789,28 @@ extern const char UnityStrErrShorthand[];
788789
#endif
789790
#endif
790791

792+
#ifndef UNITY_EXCLUDE_SETJMP_H
793+
#define TEST_PROTECT_NORETURN() (setjmp(Unity.NoReturnFrame) == 0)
794+
#define TEST_DO_NOT_RETURN() longjmp(Unity.NoReturnFrame, 1)
795+
796+
/*
797+
TEST_NOT_RETURNING(call)
798+
This macro uses TEST_PROTECT_NORETURN to protect the call.
799+
If the call returns, then the test fails.
800+
*/
801+
#define TEST_NOT_RETURNING(call) \
802+
if (TEST_PROTECT_NORETURN()) \
803+
{ \
804+
call; \
805+
UnityFail("Call Was Expected Not To Return: " #call, __LINE__); \
806+
}
807+
808+
#else
809+
#define TEST_PROTECT_NORETURN() _Static_assert(false, "TEST_PROTECT_NORETURN() cannot be used when UNITY_EXCLUDE_SETJMP_H is defined"
810+
#define TEST_DO_NOT_RETURN() _Static_assert(false, "TEST_DO_NOT_RETURN() cannot be used when UNITY_EXCLUDE_SETJMP_H is defined"
811+
#define TEST_NOT_RETURNING(call) _Static_assert(false, "TEST_NOT_RETURNING(call) cannot be used when UNITY_EXCLUDE_SETJMP_H is defined"
812+
#endif
813+
791814
/* Automatically enable variadic macros support, if it not enabled before */
792815
#ifndef UNITY_SUPPORT_VARIADIC_MACROS
793816
#ifdef __STDC_VERSION__

0 commit comments

Comments
 (0)