| /* |
| * Copyright (C) 2022 The Android Open Source Project |
| * |
| * 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 <unistd.h> |
| |
| #include <thread> |
| |
| #include <android-base/threads.h> |
| #include <gtest/gtest.h> |
| #include <utils/CallStack.h> |
| |
| __attribute__((__noinline__)) extern "C" void CurrentCaller(android::String8& backtrace) { |
| android::CallStack cs; |
| cs.update(); |
| backtrace = cs.toString(); |
| } |
| |
| TEST(CallStackTest, current_backtrace) { |
| android::String8 backtrace; |
| CurrentCaller(backtrace); |
| |
| ASSERT_NE(-1, backtrace.find("(CurrentCaller")) << "Full backtrace:\n" << backtrace; |
| } |
| |
| __attribute__((__noinline__)) extern "C" void ThreadBusyWait(std::atomic<pid_t>* tid, |
| volatile bool* done) { |
| *tid = android::base::GetThreadId(); |
| while (!*done) { |
| } |
| } |
| |
| TEST(CallStackTest, thread_backtrace) { |
| // Use a volatile to avoid any problems unwinding since sometimes |
| // accessing a std::atomic does not include unwind data at every |
| // instruction and leads to failed unwinds. |
| volatile bool done = false; |
| std::atomic<pid_t> tid = -1; |
| std::thread thread([&tid, &done]() { ThreadBusyWait(&tid, &done); }); |
| |
| while (tid == -1) { |
| } |
| |
| android::CallStack cs; |
| cs.update(0, tid); |
| |
| done = true; |
| thread.join(); |
| |
| ASSERT_NE(-1, cs.toString().find("(ThreadBusyWait")) << "Full backtrace:\n" << cs.toString(); |
| } |