#include "base/profiler/stack_base_address_posix.h"
#include "base/check_op.h"
#include "base/compiler_specific.h"
#include "base/logging.h"
#include "base/process/process_handle.h"
#include "build/build_config.h"
#if BUILDFLAG(IS_ANDROID)
#include <inttypes.h>
#include <stdio.h>
#include "base/files/file_util.h"
#include "base/files/scoped_file.h"
#endif
#if BUILDFLAG(IS_CHROMEOS)
extern "C" void* __libc_stack_end;
#endif
namespace base {
namespace {
#if BUILDFLAG(IS_ANDROID)
std::optional<uintptr_t> GetAndroidMainThreadStackBaseAddressImpl() {
char line[1024];
base::ScopedFILE fp(base::OpenFile(base::FilePath("/proc/self/maps"), "r"));
uintptr_t stack_addr = reinterpret_cast<uintptr_t>(line);
if (!fp) {
return std::nullopt;
}
while (UNSAFE_TODO(fgets(line, sizeof(line), fp.get())) != nullptr) {
uintptr_t start, end;
if (UNSAFE_TODO(sscanf(line, "%" SCNxPTR "-%" SCNxPTR, &start, &end)) ==
2) {
if (start <= stack_addr && stack_addr < end) {
return end;
}
}
}
return std::nullopt;
}
#endif
#if !BUILDFLAG(IS_LINUX)
uintptr_t GetThreadStackBaseAddressImpl(pthread_t pthread_id) {
pthread_attr_t attr;
int result = pthread_getattr_np(pthread_id, &attr);
CHECK_EQ(result, 0) << "pthread_getattr_np returned "
<< logging::SystemErrorCodeToString(result);
void* address;
size_t size;
result = pthread_attr_getstack(&attr, &address, &size);
CHECK_EQ(result, 0) << "pthread_attr_getstack returned "
<< logging::SystemErrorCodeToString(result);
pthread_attr_destroy(&attr);
const uintptr_t base_address = reinterpret_cast<uintptr_t>(address) + size;
return base_address;
}
#endif
}
std::optional<uintptr_t> GetThreadStackBaseAddress(PlatformThreadId id,
pthread_t pthread_id) {
#if BUILDFLAG(IS_LINUX)
return std::nullopt;
#else
const bool is_main_thread = id.raw() == GetCurrentProcId();
if (is_main_thread) {
#if BUILDFLAG(IS_ANDROID)
static const std::optional<uintptr_t> main_thread_base_address =
GetAndroidMainThreadStackBaseAddressImpl();
return main_thread_base_address;
#elif BUILDFLAG(IS_CHROMEOS)
return reinterpret_cast<uintptr_t>(__libc_stack_end);
#endif
}
return GetThreadStackBaseAddressImpl(pthread_id);
#endif
}
}