- Notifications
You must be signed in to change notification settings - Fork 3.1k
/
Copy pathcatch_debugger.cpp
120 lines (102 loc) · 4.31 KB
/
catch_debugger.cpp
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
// Copyright Catch2 Authors
// Distributed under the Boost Software License, Version 1.0.
// (See accompanying file LICENSE.txt or copy at
// https://www.boost.org/LICENSE_1_0.txt)
// SPDX-License-Identifier: BSL-1.0
#include<catch2/internal/catch_debugger.hpp>
#include<catch2/internal/catch_errno_guard.hpp>
#include<catch2/internal/catch_platform.hpp>
#include<catch2/internal/catch_stdstreams.hpp>
#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
namespaceCatch {
#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).
boolisDebuggerActive(){
int mib[4];
structkinfo_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\n" << std::flush;
returnfalse;
}
// We're being debugged if the P_TRACED flag is set.
return ( (info.kp_proc.p_flag & P_TRACED) != 0 );
}
#else
boolisDebuggerActive() {
// We need to find another way to determine this for non-appleclang compilers on macOS
returnfalse;
}
#endif
} // namespace Catch
#elif defined(CATCH_PLATFORM_LINUX)
#include<fstream>
#include<string>
namespaceCatch{
// 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.
boolisDebuggerActive(){
// 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); ) {
staticconstint 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';
}
}
returnfalse;
}
} // namespace Catch
#elif defined(_MSC_VER)
extern"C" __declspec(dllimport) int __stdcall IsDebuggerPresent();
namespaceCatch {
boolisDebuggerActive() {
returnIsDebuggerPresent() != 0;
}
}
#elif defined(__MINGW32__)
extern"C" __declspec(dllimport) int __stdcall IsDebuggerPresent();
namespaceCatch {
boolisDebuggerActive() {
returnIsDebuggerPresent() != 0;
}
}
#else
namespaceCatch {
boolisDebuggerActive() { returnfalse; }
}
#endif// Platform