| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  Created by Phil on 27/12/2010. | 
					
						
							|  |  |  |  *  Copyright 2010 Two Blue Cubes Ltd. All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Distributed under the Boost Software License, Version 1.0. (See accompanying | 
					
						
							|  |  |  |  *  file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  | #include "catch_debugger.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-06 22:28:42 +02:00
										 |  |  | #include "catch_errno_guard.h"
 | 
					
						
							|  |  |  | #include "catch_stream.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-12 14:52:36 +01:00
										 |  |  | #include "catch_platform.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-18 09:20:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-21 12:36:58 +00:00
										 |  |  | #ifdef CATCH_PLATFORM_MAC
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-07 15:55:09 +00:00
										 |  |  | #  include <assert.h>
 | 
					
						
							|  |  |  | #  include <stdbool.h>
 | 
					
						
							|  |  |  | #  include <sys/types.h>
 | 
					
						
							|  |  |  | #  include <unistd.h>
 | 
					
						
							|  |  |  | #  include <cstddef>
 | 
					
						
							|  |  |  | #  include <ostream>
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 11:39:13 +02:00
										 |  |  | #ifdef __apple_build_version__
 | 
					
						
							|  |  |  |     // These headers will only compile with AppleClang (XCode)
 | 
					
						
							|  |  |  |     // For other compilers (Clang, GCC, ... ) we need to exclude them
 | 
					
						
							|  |  |  | #  include <sys/sysctl.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 11:39:13 +02:00
										 |  |  |     namespace Catch { | 
					
						
							|  |  |  |         #ifdef __apple_build_version__
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |         // The following function is taken directly from the following technical note:
 | 
					
						
							| 
									
										
										
										
											2019-04-05 11:39:13 +02:00
										 |  |  |         // https://developer.apple.com/library/archive/qa/qa1361/_index.html
 | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |         // Returns true if the current process is being debugged (either
 | 
					
						
							|  |  |  |         // running under the debugger or has a debugger attached post facto).
 | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  |         bool isDebuggerActive(){ | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             int                 mib[4]; | 
					
						
							|  |  |  |             struct kinfo_proc   info; | 
					
						
							| 
									
										
										
										
											2017-09-18 17:13:17 +01:00
										 |  |  |             std::size_t         size; | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             // Initialize the flags so that, if sysctl fails for some bizarre
 | 
					
						
							|  |  |  |             // reason, we get a predictable result.
 | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             info.kp_proc.p_flag = 0; | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             // Initialize mib, which tells sysctl the info we want, in this case
 | 
					
						
							|  |  |  |             // we're looking for information about a specific process ID.
 | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             mib[0] = CTL_KERN; | 
					
						
							|  |  |  |             mib[1] = KERN_PROC; | 
					
						
							|  |  |  |             mib[2] = KERN_PROC_PID; | 
					
						
							|  |  |  |             mib[3] = getpid(); | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             // Call sysctl.
 | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             size = sizeof(info); | 
					
						
							| 
									
										
										
										
											2017-04-25 12:41:30 +02:00
										 |  |  |             if( sysctl(mib, sizeof(mib) / sizeof(*mib), &info, &size, nullptr, 0) != 0 ) { | 
					
						
							| 
									
										
										
										
											2014-10-02 19:08:19 +01:00
										 |  |  |                 Catch::cerr() << "\n** Call to sysctl failed - unable to determine if debugger is active **\n" << std::endl; | 
					
						
							| 
									
										
										
										
											2014-01-08 17:17:31 +00:00
										 |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             // We're being debugged if the P_TRACED flag is set.
 | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  |             return ( (info.kp_proc.p_flag & P_TRACED) != 0 ); | 
					
						
							| 
									
										
										
										
											2013-07-03 19:14:59 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-05 11:39:13 +02:00
										 |  |  |         #else
 | 
					
						
							|  |  |  |         bool isDebuggerActive() { | 
					
						
							|  |  |  |             // We need to find another way to determine this for non-appleclang compilers on macOS
 | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         #endif
 | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  |     } // namespace Catch
 | 
					
						
							| 
									
										
										
										
											2010-12-27 20:49:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-05 14:56:20 +01:00
										 |  |  | #elif defined(CATCH_PLATFORM_LINUX)
 | 
					
						
							|  |  |  |     #include <fstream>
 | 
					
						
							|  |  |  |     #include <string>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     namespace Catch{ | 
					
						
							|  |  |  |         // The standard POSIX way of detecting a debugger is to attempt to
 | 
					
						
							|  |  |  |         // ptrace() the process, but this needs to be done from a child and not
 | 
					
						
							|  |  |  |         // this process itself to still allow attaching to this process later
 | 
					
						
							|  |  |  |         // if wanted, so is rather heavy. Under Linux we have the PID of the
 | 
					
						
							|  |  |  |         // "debugger" (which doesn't need to be gdb, of course, it could also
 | 
					
						
							|  |  |  |         // be strace, for example) in /proc/$PID/status, so just get it from
 | 
					
						
							|  |  |  |         // there instead.
 | 
					
						
							|  |  |  |         bool isDebuggerActive(){ | 
					
						
							| 
									
										
										
										
											2017-03-06 21:51:22 +01:00
										 |  |  |             // Libstdc++ has a bug, where std::ifstream sets errno to 0
 | 
					
						
							|  |  |  |             // This way our users can properly assert over errno values
 | 
					
						
							|  |  |  |             ErrnoGuard guard; | 
					
						
							| 
									
										
										
										
											2016-02-05 14:56:20 +01:00
										 |  |  |             std::ifstream in("/proc/self/status"); | 
					
						
							|  |  |  |             for( std::string line; std::getline(in, line); ) { | 
					
						
							|  |  |  |                 static const int PREFIX_LEN = 11; | 
					
						
							|  |  |  |                 if( line.compare(0, PREFIX_LEN, "TracerPid:\t") == 0 ) { | 
					
						
							|  |  |  |                     // We're traced if the PID is not 0 and no other PID starts
 | 
					
						
							|  |  |  |                     // with 0 digit, so it's enough to check for just a single
 | 
					
						
							|  |  |  |                     // character.
 | 
					
						
							|  |  |  |                     return line.length() > PREFIX_LEN && line[PREFIX_LEN] != '0'; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } // namespace Catch
 | 
					
						
							| 
									
										
										
										
											2011-07-04 09:09:03 +01:00
										 |  |  | #elif defined(_MSC_VER)
 | 
					
						
							| 
									
										
										
										
											2011-03-21 12:36:58 +00:00
										 |  |  |     extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent(); | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  |     namespace Catch { | 
					
						
							|  |  |  |         bool isDebuggerActive() { | 
					
						
							|  |  |  |             return IsDebuggerPresent() != 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-03-24 19:23:28 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-05-10 21:29:50 +01:00
										 |  |  | #elif defined(__MINGW32__)
 | 
					
						
							|  |  |  |     extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent(); | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  |     namespace Catch { | 
					
						
							|  |  |  |         bool isDebuggerActive() { | 
					
						
							|  |  |  |             return IsDebuggerPresent() != 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-05-10 21:29:50 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-16 18:58:15 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  |     namespace Catch { | 
					
						
							| 
									
										
										
										
											2017-07-12 16:16:55 +02:00
										 |  |  |        bool isDebuggerActive() { return false; } | 
					
						
							| 
									
										
										
										
											2013-12-03 18:52:41 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | #endif // Platform
 |