// Copyright 2018 The SwiftShader Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "Debug.hpp" #include #include #include #include #if __ANDROID__ # include #endif #if defined(__unix__) # define PTRACE # include # include #elif defined(_WIN32) || defined(_WIN64) # include #elif defined(__APPLE__) || defined(__MACH__) # include # include #endif #ifdef ERROR # undef ERROR // b/127920555 #endif #ifndef REACTOR_LOGGING_LEVEL # define REACTOR_LOGGING_LEVEL Info #endif namespace { bool IsUnderDebugger() { #if defined(PTRACE) && !defined(__APPLE__) && !defined(__MACH__) static bool checked = false; static bool res = false; if(!checked) { // If a debugger is attached then we're already being ptraced and ptrace // will return a non-zero value. checked = true; if(ptrace(PTRACE_TRACEME, 0, 1, 0) != 0) { res = true; } else { ptrace(PTRACE_DETACH, 0, 1, 0); } } return res; #elif defined(_WIN32) || defined(_WIN64) return IsDebuggerPresent() != 0; #elif defined(__APPLE__) || defined(__MACH__) // Code comes from the Apple Technical Q&A QA1361 // Tell sysctl what info we're requestion. Specifically we're asking for // info about this our PID. int res = 0; int request[4] = { CTL_KERN, KERN_PROC, KERN_PROC_PID, getpid() }; struct kinfo_proc info; size_t size = sizeof(info); info.kp_proc.p_flag = 0; // Get the info we're requesting, if sysctl fails then info.kp_proc.p_flag will remain 0. res = sysctl(request, sizeof(request) / sizeof(*request), &info, &size, NULL, 0); ASSERT_MSG(res == 0, "syscl returned %d", res); // We're being debugged if the P_TRACED flag is set return ((info.kp_proc.p_flag & P_TRACED) != 0); #else return false; #endif } enum class Level { Debug, Info, Warn, Error, Fatal, }; #ifdef __ANDROID__ [[maybe_unused]] void logv_android(Level level, const char *msg) { switch(level) { case Level::Debug: __android_log_write(ANDROID_LOG_DEBUG, "SwiftShader", msg); break; case Level::Info: __android_log_write(ANDROID_LOG_INFO, "SwiftShader", msg); break; case Level::Warn: __android_log_write(ANDROID_LOG_WARN, "SwiftShader", msg); break; case Level::Error: __android_log_write(ANDROID_LOG_ERROR, "SwiftShader", msg); break; case Level::Fatal: __android_log_write(ANDROID_LOG_FATAL, "SwiftShader", msg); break; } } #else [[maybe_unused]] void logv_std(Level level, const char *msg) { switch(level) { case Level::Debug: case Level::Info: fprintf(stdout, "%s", msg); break; case Level::Warn: case Level::Error: case Level::Fatal: fprintf(stderr, "%s", msg); break; } } #endif void logv(Level level, const char *format, va_list args) { if(static_cast(level) < static_cast(Level::REACTOR_LOGGING_LEVEL)) { return; } #ifndef SWIFTSHADER_DISABLE_TRACE char buffer[2048]; vsnprintf(buffer, sizeof(buffer), format, args); # if defined(__ANDROID__) logv_android(level, buffer); # elif defined(_WIN32) logv_std(level, buffer); ::OutputDebugString(buffer); # else logv_std(level, buffer); # endif const bool traceToFile = false; if(traceToFile) { FILE *file = fopen(TRACE_OUTPUT_FILE, "a"); if(file) { vfprintf(file, format, args); fclose(file); } } #endif // SWIFTSHADER_DISABLE_TRACE } } // anonymous namespace namespace rr { void trace(const char *format, ...) { va_list vararg; va_start(vararg, format); logv(Level::Debug, format, vararg); va_end(vararg); } void warn(const char *format, ...) { va_list vararg; va_start(vararg, format); logv(Level::Warn, format, vararg); va_end(vararg); } void abort(const char *format, ...) { va_list vararg; va_start(vararg, format); logv(Level::Fatal, format, vararg); va_end(vararg); ::abort(); } void trace_assert(const char *format, ...) { static std::atomic asserted = { false }; if(IsUnderDebugger() && !asserted.exchange(true)) { // Abort after tracing and printing to stderr va_list vararg; va_start(vararg, format); logv(Level::Fatal, format, vararg); va_end(vararg); ::abort(); } else if(!asserted) { va_list vararg; va_start(vararg, format); logv(Level::Fatal, format, vararg); va_end(vararg); } } } // namespace rr