| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  | // Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-18 02:06:48 -02:00
										 |  |  | #include <cstdlib>
 | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 02:52:31 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-18 02:06:48 -02:00
										 |  |  | // For asserts we'd like to keep all the junk executed when an assert happens away from the
 | 
					
						
							|  |  |  | // important code in the function. One way of doing this is to put all the relevant code inside a
 | 
					
						
							|  |  |  | // lambda and force the compiler to not inline it. Unfortunately, MSVC seems to have no syntax to
 | 
					
						
							|  |  |  | // specify __declspec on lambda functions, so what we do instead is define a noinline wrapper
 | 
					
						
							|  |  |  | // template that calls the lambda. This seems to generate an extra instruction at the call-site
 | 
					
						
							|  |  |  | // compared to the ideal implementation (which wouldn't support ASSERT_MSG parameters), but is good
 | 
					
						
							|  |  |  | // enough for our purposes.
 | 
					
						
							|  |  |  | template <typename Fn> | 
					
						
							|  |  |  | #if defined(_MSC_VER)
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | __declspec(noinline, noreturn) | 
					
						
							| 
									
										
										
										
											2015-02-18 02:06:48 -02:00
										 |  |  | #elif defined(__GNUC__)
 | 
					
						
							|  |  |  |     __attribute__((noinline, noreturn, cold)) | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     static void assert_noinline_call(const Fn& fn) { | 
					
						
							| 
									
										
										
										
											2015-02-18 02:06:48 -02:00
										 |  |  |     fn(); | 
					
						
							|  |  |  |     Crash(); | 
					
						
							|  |  |  |     exit(1); // Keeps GCC's mouth shut about this actually returning
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define ASSERT(_a_)                                                                                \
 | 
					
						
							|  |  |  |     do                                                                                             \ | 
					
						
							|  |  |  |         if (!(_a_)) {                                                                              \ | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |             assert_noinline_call([] { LOG_CRITICAL(Debug, "Assertion Failed!"); });                \ | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         }                                                                                          \ | 
					
						
							|  |  |  |     while (0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define ASSERT_MSG(_a_, ...)                                                                       \
 | 
					
						
							|  |  |  |     do                                                                                             \ | 
					
						
							|  |  |  |         if (!(_a_)) {                                                                              \ | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |             assert_noinline_call([&] { LOG_CRITICAL(Debug, "Assertion Failed!\n" __VA_ARGS__); }); \ | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         }                                                                                          \ | 
					
						
							|  |  |  |     while (0) | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define UNREACHABLE() ASSERT_MSG(false, "Unreachable code!")
 | 
					
						
							| 
									
										
										
										
											2016-04-24 23:40:14 +08:00
										 |  |  | #define UNREACHABLE_MSG(...) ASSERT_MSG(false, __VA_ARGS__)
 | 
					
						
							| 
									
										
										
										
											2015-01-20 17:16:47 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef _DEBUG
 | 
					
						
							|  |  |  | #define DEBUG_ASSERT(_a_) ASSERT(_a_)
 | 
					
						
							|  |  |  | #define DEBUG_ASSERT_MSG(_a_, ...) ASSERT_MSG(_a_, __VA_ARGS__)
 | 
					
						
							|  |  |  | #else // not debug
 | 
					
						
							|  |  |  | #define DEBUG_ASSERT(_a_)
 | 
					
						
							|  |  |  | #define DEBUG_ASSERT_MSG(_a_, _desc_, ...)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-20 17:58:57 -05:00
										 |  |  | #define UNIMPLEMENTED() ASSERT_MSG(false, "Unimplemented code!")
 | 
					
						
							| 
									
										
										
										
											2018-01-07 22:43:41 +00:00
										 |  |  | #define UNIMPLEMENTED_MSG(...) ASSERT_MSG(false, __VA_ARGS__)
 |