/*----------------------------------------------------------------------------*/ /* Copyright (c) 2008-2018 FIRST. All Rights Reserved. */ /* Open Source Software - may be modified and shared by FRC teams. The code */ /* must be accompanied by the FIRST BSD license file in the root directory of */ /* the project. */ /*----------------------------------------------------------------------------*/ #include "frc/Utility.h" #ifndef _WIN32 #include #include #endif #include #include #include #include #include #include #include #include #include "frc/ErrorBase.h" using namespace frc; bool wpi_assert_impl(bool conditionValue, const wpi::Twine& conditionText, const wpi::Twine& message, wpi::StringRef fileName, int lineNumber, wpi::StringRef funcName) { if (!conditionValue) { wpi::SmallString<128> locBuf; wpi::raw_svector_ostream locStream(locBuf); locStream << funcName << " [" << wpi::sys::path::filename(fileName) << ":" << lineNumber << "]"; wpi::SmallString<128> errorBuf; wpi::raw_svector_ostream errorStream(errorBuf); errorStream << "Assertion \"" << conditionText << "\" "; if (message.isTriviallyEmpty() || (message.isSingleStringRef() && message.getSingleStringRef().empty())) { errorStream << "failed.\n"; } else { errorStream << "failed: " << message << "\n"; } std::string stack = GetStackTrace(2); // Print the error and send it to the DriverStation HAL_SendError(1, 1, 0, errorBuf.c_str(), locBuf.c_str(), stack.c_str(), 1); } return conditionValue; } /** * Common error routines for wpi_assertEqual_impl and wpi_assertNotEqual_impl. * * This should not be called directly; it should only be used by * wpi_assertEqual_impl and wpi_assertNotEqual_impl. */ void wpi_assertEqual_common_impl(const wpi::Twine& valueA, const wpi::Twine& valueB, const wpi::Twine& equalityType, const wpi::Twine& message, wpi::StringRef fileName, int lineNumber, wpi::StringRef funcName) { wpi::SmallString<128> locBuf; wpi::raw_svector_ostream locStream(locBuf); locStream << funcName << " [" << wpi::sys::path::filename(fileName) << ":" << lineNumber << "]"; wpi::SmallString<128> errorBuf; wpi::raw_svector_ostream errorStream(errorBuf); errorStream << "Assertion \"" << valueA << " " << equalityType << " " << valueB << "\" "; if (message.isTriviallyEmpty() || (message.isSingleStringRef() && message.getSingleStringRef().empty())) { errorStream << "failed.\n"; } else { errorStream << "failed: " << message << "\n"; } std::string trace = GetStackTrace(3); // Print the error and send it to the DriverStation HAL_SendError(1, 1, 0, errorBuf.c_str(), locBuf.c_str(), trace.c_str(), 1); } bool wpi_assertEqual_impl(int valueA, int valueB, const wpi::Twine& valueAString, const wpi::Twine& valueBString, const wpi::Twine& message, wpi::StringRef fileName, int lineNumber, wpi::StringRef funcName) { if (!(valueA == valueB)) { wpi_assertEqual_common_impl(valueAString, valueBString, "==", message, fileName, lineNumber, funcName); } return valueA == valueB; } bool wpi_assertNotEqual_impl(int valueA, int valueB, const wpi::Twine& valueAString, const wpi::Twine& valueBString, const wpi::Twine& message, wpi::StringRef fileName, int lineNumber, wpi::StringRef funcName) { if (!(valueA != valueB)) { wpi_assertEqual_common_impl(valueAString, valueBString, "!=", message, fileName, lineNumber, funcName); } return valueA != valueB; } namespace frc { int GetFPGAVersion() { int32_t status = 0; int version = HAL_GetFPGAVersion(&status); wpi_setGlobalErrorWithContext(status, HAL_GetErrorMessage(status)); return version; } int64_t GetFPGARevision() { int32_t status = 0; int64_t revision = HAL_GetFPGARevision(&status); wpi_setGlobalErrorWithContext(status, HAL_GetErrorMessage(status)); return revision; } uint64_t GetFPGATime() { int32_t status = 0; uint64_t time = HAL_GetFPGATime(&status); wpi_setGlobalErrorWithContext(status, HAL_GetErrorMessage(status)); return time; } bool GetUserButton() { int32_t status = 0; bool value = HAL_GetFPGAButton(&status); wpi_setGlobalError(status); return value; } #ifndef _WIN32 /** * Demangle a C++ symbol, used for printing stack traces. */ static std::string demangle(char const* mangledSymbol) { char buffer[256]; size_t length; int32_t status; if (std::sscanf(mangledSymbol, "%*[^(]%*[(]%255[^)+]", buffer)) { char* symbol = abi::__cxa_demangle(buffer, nullptr, &length, &status); if (status == 0) { return symbol; } else { // If the symbol couldn't be demangled, it's probably a C function, // so just return it as-is. return buffer; } } // If everything else failed, just return the mangled symbol return mangledSymbol; } std::string GetStackTrace(int offset) { void* stackTrace[128]; int stackSize = backtrace(stackTrace, 128); char** mangledSymbols = backtrace_symbols(stackTrace, stackSize); wpi::SmallString<1024> buf; wpi::raw_svector_ostream trace(buf); for (int i = offset; i < stackSize; i++) { // Only print recursive functions once in a row. if (i == 0 || stackTrace[i] != stackTrace[i - 1]) { trace << "\tat " << demangle(mangledSymbols[i]) << "\n"; } } std::free(mangledSymbols); return trace.str(); } #else static std::string demangle(char const* mangledSymbol) { return "no demangling on windows"; } std::string GetStackTrace(int offset) { return "no stack trace on windows"; } #endif } // namespace frc