blob: 57d8f2afe82e5a4d739a948944cc89f5bd4bbf40 [file] [log] [blame]
Christopher Ferris63860cb2015-11-16 17:30:32 -08001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28
29#include <errno.h>
30#include <signal.h>
31#include <stdint.h>
32#include <stdlib.h>
33#include <sys/types.h>
34#include <unistd.h>
35
36#include <private/bionic_macros.h>
37
38#include "BacktraceData.h"
39#include "Config.h"
40#include "DebugData.h"
41#include "debug_log.h"
42#include "malloc_debug.h"
43
Christopher Ferris602b88c2017-08-04 13:04:04 -070044static void ToggleBacktraceEnable(int, siginfo_t*, void*) {
45 g_debug->backtrace->ToggleBacktraceEnabled();
46}
47
48static void EnableDump(int, siginfo_t*, void*) {
49 g_debug->backtrace->EnableDumping();
Christopher Ferris55a89a42016-04-07 17:14:53 -070050}
51
52BacktraceData::BacktraceData(DebugData* debug_data, const Config& config, size_t* offset)
53 : OptionData(debug_data) {
Christopher Ferris2b2b25b2017-04-05 19:13:03 -070054 size_t hdr_len = sizeof(BacktraceHeader) + sizeof(uintptr_t) * config.backtrace_frames();
Christopher Ferris63860cb2015-11-16 17:30:32 -080055 alloc_offset_ = *offset;
Dan Alberta613d0d2017-10-05 16:39:33 -070056 *offset += __BIONIC_ALIGN(hdr_len, MINIMUM_ALIGNMENT_BYTES);
Christopher Ferris63860cb2015-11-16 17:30:32 -080057}
58
Christopher Ferris63860cb2015-11-16 17:30:32 -080059bool BacktraceData::Initialize(const Config& config) {
Christopher Ferris2b2b25b2017-04-05 19:13:03 -070060 enabled_ = config.backtrace_enabled();
61 if (config.backtrace_enable_on_signal()) {
Elliott Hughes3e235912018-02-01 14:21:51 -080062 struct sigaction64 enable_act = {};
Christopher Ferris602b88c2017-08-04 13:04:04 -070063 enable_act.sa_sigaction = ToggleBacktraceEnable;
Christopher Ferris63860cb2015-11-16 17:30:32 -080064 enable_act.sa_flags = SA_RESTART | SA_SIGINFO | SA_ONSTACK;
Elliott Hughes3e235912018-02-01 14:21:51 -080065 if (sigaction64(config.backtrace_signal(), &enable_act, nullptr) != 0) {
Christopher Ferris63860cb2015-11-16 17:30:32 -080066 error_log("Unable to set up backtrace signal enable function: %s", strerror(errno));
67 return false;
68 }
69 info_log("%s: Run: 'kill -%d %d' to enable backtracing.", getprogname(),
Christopher Ferris2b2b25b2017-04-05 19:13:03 -070070 config.backtrace_signal(), getpid());
Christopher Ferris63860cb2015-11-16 17:30:32 -080071 }
Christopher Ferris602b88c2017-08-04 13:04:04 -070072
Elliott Hughes3e235912018-02-01 14:21:51 -080073 struct sigaction64 act = {};
Christopher Ferris602b88c2017-08-04 13:04:04 -070074 act.sa_sigaction = EnableDump;
75 act.sa_flags = SA_RESTART | SA_SIGINFO | SA_ONSTACK;
Elliott Hughes3e235912018-02-01 14:21:51 -080076 if (sigaction64(config.backtrace_dump_signal(), &act, nullptr) != 0) {
Christopher Ferris602b88c2017-08-04 13:04:04 -070077 error_log("Unable to set up backtrace dump signal function: %s", strerror(errno));
78 return false;
79 }
80 info_log("%s: Run: 'kill -%d %d' to dump the backtrace.", getprogname(),
81 config.backtrace_dump_signal(), getpid());
82
83 dump_ = false;
Christopher Ferris63860cb2015-11-16 17:30:32 -080084 return true;
85}