blob: f40e62ab25e3c01946a7cda5412f52302eb554e2 [file] [log] [blame]
// Copyright 2024 The Pigweed Authors
//
// 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
//
// https://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 "pw_cpu_exception_cortex_m/crash.h"
#include <cstdarg>
#include <mutex>
#include "pw_cpu_exception_cortex_m/cpu_state.h"
#include "pw_cpu_exception_cortex_m_private/cortex_m_constants.h"
#include "pw_status/status.h"
#include "pw_string/string_builder.h"
#include "pw_sync/lock_annotations.h"
#include "pw_sync/mutex.h"
#include "pw_unit_test/framework.h"
namespace pw::cpu_exception::cortex_m {
namespace {
const char* kSampleThreadName = "BadThread";
class FakeCrashHandler : public testing::Test {
public:
void CaptureCrashAnalysis(const char* fmt, va_list args) {
StringBuilder message_builder(buffer_);
ASSERT_EQ(message_builder.FormatVaList(fmt, args).status(), OkStatus());
}
protected:
std::array<char, 124> buffer_;
};
pw::sync::Mutex crash_handler_lock;
FakeCrashHandler* crash_handler PW_GUARDED_BY(crash_handler_lock) = nullptr;
TEST_F(FakeCrashHandler, CaptureCrashInfoDivideByZero) {
std::lock_guard lock(crash_handler_lock);
crash_handler = this;
pw_cpu_exception_State cpu_state = {};
cpu_state.extended.cfsr = kCfsrDivbyzeroMask;
AnalyzeCpuStateAndCrash(cpu_state, kSampleThreadName);
std::string_view analysis{buffer_.data(), sizeof(buffer_)};
EXPECT_NE(analysis.find(kSampleThreadName), static_cast<size_t>(0));
EXPECT_NE(analysis.find("DIVBYZERO"), static_cast<size_t>(0));
crash_handler = nullptr;
}
TEST_F(FakeCrashHandler, CaptureCrashInfoNoThreadName) {
std::lock_guard lock(crash_handler_lock);
crash_handler = this;
pw_cpu_exception_State cpu_state = {};
cpu_state.extended.cfsr = kCfsrDivbyzeroMask;
AnalyzeCpuStateAndCrash(cpu_state, nullptr);
std::string_view analysis{buffer_.data(), sizeof(buffer_)};
EXPECT_NE(analysis.find(kSampleThreadName), static_cast<size_t>(0));
EXPECT_NE(analysis.find("Thread=?"), static_cast<size_t>(0));
crash_handler = nullptr;
}
} // namespace
void CaptureCrashAnalysisForTest(const pw_cpu_exception_State&,
const char* fmt,
...) {
ASSERT_NE(crash_handler, nullptr);
va_list args;
va_start(args, fmt);
crash_handler->CaptureCrashAnalysis(fmt, args);
va_end(args);
}
} // namespace pw::cpu_exception::cortex_m