2010-12-27 21:49:19 +01: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 19:52:41 +01: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 15:52:36 +02:00
|
|
|
#include "catch_platform.h"
|
2011-01-18 10:20:06 +01:00
|
|
|
|
2011-03-21 13:36:58 +01:00
|
|
|
#ifdef CATCH_PLATFORM_MAC
|
2010-12-27 21:49:19 +01:00
|
|
|
|
2017-11-07 16:55:09 +01:00
|
|
|
# include <assert.h>
|
|
|
|
# include <stdbool.h>
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <unistd.h>
|
|
|
|
# include <sys/sysctl.h>
|
|
|
|
# include <cstddef>
|
|
|
|
# include <ostream>
|
2010-12-27 21:49:19 +01:00
|
|
|
|
2017-11-07 16:55:09 +01:00
|
|
|
namespace Catch {
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
// The following function is taken directly from the following technical note:
|
|
|
|
// http://developer.apple.com/library/mac/#qa/qa2004/qa1361.html
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01: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 19:52:41 +01:00
|
|
|
bool isDebuggerActive(){
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
int mib[4];
|
|
|
|
struct kinfo_proc info;
|
2017-09-18 18:13:17 +02:00
|
|
|
std::size_t size;
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
// Initialize the flags so that, if sysctl fails for some bizarre
|
|
|
|
// reason, we get a predictable result.
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
info.kp_proc.p_flag = 0;
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01: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 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
mib[0] = CTL_KERN;
|
|
|
|
mib[1] = KERN_PROC;
|
|
|
|
mib[2] = KERN_PROC_PID;
|
|
|
|
mib[3] = getpid();
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
// Call sysctl.
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01: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 20:08:19 +02:00
|
|
|
Catch::cerr() << "\n** Call to sysctl failed - unable to determine if debugger is active **\n" << std::endl;
|
2014-01-08 18:17:31 +01:00
|
|
|
return false;
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
// We're being debugged if the P_TRACED flag is set.
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2010-12-27 21:49:19 +01:00
|
|
|
return ( (info.kp_proc.p_flag & P_TRACED) != 0 );
|
2013-07-03 20:14:59 +02:00
|
|
|
}
|
2013-12-03 19:52:41 +01:00
|
|
|
} // namespace Catch
|
2010-12-27 21:49:19 +01: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 10:09:03 +02:00
|
|
|
#elif defined(_MSC_VER)
|
2011-03-21 13:36:58 +01:00
|
|
|
extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent();
|
2013-12-03 19:52:41 +01:00
|
|
|
namespace Catch {
|
|
|
|
bool isDebuggerActive() {
|
|
|
|
return IsDebuggerPresent() != 0;
|
|
|
|
}
|
2011-03-24 20:23:28 +01:00
|
|
|
}
|
2012-05-10 22:29:50 +02:00
|
|
|
#elif defined(__MINGW32__)
|
|
|
|
extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent();
|
2013-12-03 19:52:41 +01:00
|
|
|
namespace Catch {
|
|
|
|
bool isDebuggerActive() {
|
|
|
|
return IsDebuggerPresent() != 0;
|
|
|
|
}
|
2012-05-10 22:29:50 +02:00
|
|
|
}
|
2011-02-16 19:58:15 +01:00
|
|
|
#else
|
2013-12-03 19:52:41 +01:00
|
|
|
namespace Catch {
|
2017-07-12 16:16:55 +02:00
|
|
|
bool isDebuggerActive() { return false; }
|
2013-12-03 19:52:41 +01:00
|
|
|
}
|
|
|
|
#endif // Platform
|