mirror of
https://github.com/catchorg/Catch2.git
synced 2025-01-04 00:43:30 +01:00
123 lines
4.3 KiB
C++
123 lines
4.3 KiB
C++
/*
|
|
* 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)
|
|
*
|
|
*/
|
|
|
|
#include "catch_debugger.h"
|
|
#include "catch_errno_guard.h"
|
|
#include "catch_stream.h"
|
|
#include "catch_platform.h"
|
|
|
|
#if defined(CATCH_PLATFORM_MAC) || defined(CATCH_PLATFORM_IPHONE)
|
|
|
|
# include <cassert>
|
|
# include <sys/types.h>
|
|
# include <unistd.h>
|
|
# include <cstddef>
|
|
# include <ostream>
|
|
|
|
#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
|
|
|
|
namespace Catch {
|
|
#ifdef __apple_build_version__
|
|
// The following function is taken directly from the following technical note:
|
|
// https://developer.apple.com/library/archive/qa/qa1361/_index.html
|
|
|
|
// Returns true if the current process is being debugged (either
|
|
// running under the debugger or has a debugger attached post facto).
|
|
bool isDebuggerActive(){
|
|
int mib[4];
|
|
struct kinfo_proc info;
|
|
std::size_t size;
|
|
|
|
// Initialize the flags so that, if sysctl fails for some bizarre
|
|
// reason, we get a predictable result.
|
|
|
|
info.kp_proc.p_flag = 0;
|
|
|
|
// Initialize mib, which tells sysctl the info we want, in this case
|
|
// we're looking for information about a specific process ID.
|
|
|
|
mib[0] = CTL_KERN;
|
|
mib[1] = KERN_PROC;
|
|
mib[2] = KERN_PROC_PID;
|
|
mib[3] = getpid();
|
|
|
|
// Call sysctl.
|
|
|
|
size = sizeof(info);
|
|
if( sysctl(mib, sizeof(mib) / sizeof(*mib), &info, &size, nullptr, 0) != 0 ) {
|
|
Catch::cerr() << "\n** Call to sysctl failed - unable to determine if debugger is active **\n" << std::endl;
|
|
return false;
|
|
}
|
|
|
|
// We're being debugged if the P_TRACED flag is set.
|
|
|
|
return ( (info.kp_proc.p_flag & P_TRACED) != 0 );
|
|
}
|
|
#else
|
|
bool isDebuggerActive() {
|
|
// We need to find another way to determine this for non-appleclang compilers on macOS
|
|
return false;
|
|
}
|
|
#endif
|
|
} // namespace Catch
|
|
|
|
#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(){
|
|
// Libstdc++ has a bug, where std::ifstream sets errno to 0
|
|
// This way our users can properly assert over errno values
|
|
ErrnoGuard guard;
|
|
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
|
|
#elif defined(_MSC_VER)
|
|
extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent();
|
|
namespace Catch {
|
|
bool isDebuggerActive() {
|
|
return IsDebuggerPresent() != 0;
|
|
}
|
|
}
|
|
#elif defined(__MINGW32__)
|
|
extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent();
|
|
namespace Catch {
|
|
bool isDebuggerActive() {
|
|
return IsDebuggerPresent() != 0;
|
|
}
|
|
}
|
|
#else
|
|
namespace Catch {
|
|
bool isDebuggerActive() { return false; }
|
|
}
|
|
#endif // Platform
|