summaryrefslogtreecommitdiffstats
path: root/libopencm3/include/libopencm3/cm3/assert.h
blob: f1aabc3ca635c83ff561a3f4be3c20f08eb5a84d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
/** @defgroup debugging Debugging

@brief Macros and functions to aid in debugging

@version 1.0.0

@date 25 September 2012

Two preprocessor defines control the behavior of assertion check macros in
this module. They allow the choice between generated code size and ease of
debugging.

If NDEBUG is defined, all assertion checks are disabled and macros do not
generate any code.

If CM3_ASSERT_VERBOSE is defined, information regarding the position of
assertion checks will be stored in the binary, allowing for more
informative error messages, but also significantly increased code size. As
default assertion checks do not use this information it is only useful if
the application linked with libopencm3 defines its own
cm3_assert_failed_verbose() implementation.

LGPL License Terms @ref lgpl_license
*/

/*
 * This file is part of the libopencm3 project.
 *
 * Copyright (C) 2012 Tomaz Solc <tomaz.solc@tablix.org>
 *
 * This library is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this library.  If not, see <http://www.gnu.org/licenses/>.
 */

/**@{*/

#ifndef LIBOPENCM3_CM3_ASSERT_H
#define LIBOPENCM3_CM3_ASSERT_H

#include <libopencm3/cm3/common.h>

#define CM3_LIKELY(expr) (__builtin_expect(!!(expr), 1))

#ifdef NDEBUG
# define cm3_assert(expr) (void)0
# define cm3_assert_not_reached() do { } while (1)
#else
# ifdef CM3_ASSERT_VERBOSE
#  define cm3_assert(expr) do { \
				if (CM3_LIKELY(expr)) { \
					(void)0; \
				} else { \
					cm3_assert_failed_verbose( \
						__FILE__, __LINE__, \
						__func__, #expr); \
				} \
			   } while (0)
#  define cm3_assert_not_reached() \
	cm3_assert_failed_verbose( \
			__FILE__, __LINE__, \
			__func__, 0)
# else
/** @brief Check if assertion is true.
 *
 * If NDEBUG macro is defined, this macro generates no code. Otherwise
 * cm3_assert_failed() or cm3_assert_failed_verbose() is called if assertion
 * is false.
 *
 * The purpose of this macro is to aid in debugging libopencm3 and
 * applications using it. It can be used for example to check if function
 * arguments are within expected ranges and stop execution in case an
 * unexpected state is reached.
 *
 * @param expr expression to check */
#  define cm3_assert(expr) do { \
				if (CM3_LIKELY(expr)) { \
					(void)0; \
				} else { \
					cm3_assert_failed(); \
				} \
			} while (0)
/** @brief Check if unreachable code is reached.
 *
 * If NDEBUG macro is defined, this macro generates code for an infinite loop.
 * Otherwise cm3_assert_failed() or cm3_assert_failed_verbose() is called if
 * the macro is ever reached.
 *
 * The purpose of this macro is to aid in debugging libopencm3 and
 * applications using it. It can be used for example to stop execution if an
 * unreachable portion of code is reached. */
#  define cm3_assert_not_reached() cm3_assert_failed()
# endif
#endif

BEGIN_DECLS

/** @brief Called on a failed assertion.
 *
 * Halts execution in an infinite loop. This function never returns.
 *
 * Defined as a weak symbol, so applications can define their own
 * implementation. Usually, a custom implementation of this function should
 * report an error in some way (print a message to a debug console, display,
 * LED, ...) and halt execution or reboot the device. */
void cm3_assert_failed(void) __attribute__((__noreturn__));

/** @brief Called on a failed assertion with verbose messages enabled.
 *
 * Halts execution in an infinite loop. This function never returns.
 *
 * Defined as a weak symbol, so applications can define their own
 * implementation. Usually, a custom implementation of this function should
 * report an error in some way (print a message to a debug console, display,
 * LED, ...) and halt execution or reboot the device.
 *
 * @param file File name where the failed assertion occurred
 * @param line Line number where the failed assertion occurred
 * @param func Name of the function where the failed assertion occurred
 * @param assert_expr Expression that evaluated to false (can be NULL) */
void cm3_assert_failed_verbose(const char *file, int line, const char *func,
		const char *assert_expr) __attribute__((__noreturn__));

END_DECLS

#endif

/**@}*/