blob: a75a807be4f180933c684457ee66087c4e64395d [file] [log] [blame]
Christopher Ferris17e91d42013-10-21 13:30:52 -07001/*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef _LIBBACKTRACE_BACKTRACE_THREAD_H
18#define _LIBBACKTRACE_BACKTRACE_THREAD_H
19
20#include <inttypes.h>
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070021#include <pthread.h>
Christopher Ferrisaa63d9f2014-04-29 09:35:30 -070022#include <signal.h>
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070023#include <string.h>
Christopher Ferris17e91d42013-10-21 13:30:52 -070024#include <sys/types.h>
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070025#include <ucontext.h>
Christopher Ferris17e91d42013-10-21 13:30:52 -070026
Christopher Ferrisdf290612014-01-22 19:21:07 -080027#include "BacktraceImpl.h"
Christopher Ferris17e91d42013-10-21 13:30:52 -070028
Christopher Ferrisaa63d9f2014-04-29 09:35:30 -070029// The signal used to cause a thread to dump the stack.
30#if defined(__GLIBC__)
31// GLIBC reserves __SIGRTMIN signals, so use SIGRTMIN to avoid errors.
32#define THREAD_SIGNAL SIGRTMIN
33#else
34#define THREAD_SIGNAL (__SIGRTMIN+1)
35#endif
36
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070037class ThreadEntry {
38public:
39 static ThreadEntry* Get(pid_t pid, pid_t tid, bool create = true);
Christopher Ferris17e91d42013-10-21 13:30:52 -070040
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070041 static void Remove(ThreadEntry* entry);
42
43 inline void CopyUcontext(ucontext_t* ucontext) {
44 memcpy(&ucontext_, ucontext, sizeof(ucontext_));
45 }
46
47 void Wake();
48
49 void Wait(int);
50
51 inline void Lock() {
52 pthread_mutex_lock(&mutex_);
53 // Reset the futex value in case of multiple unwinds of the same thread.
54 futex_ = 0;
55 }
56
57 inline void Unlock() {
58 pthread_mutex_unlock(&mutex_);
59 }
60
61 inline ucontext_t* GetUcontext() { return &ucontext_; }
62
63private:
64 ThreadEntry(pid_t pid, pid_t tid);
Christopher Ferris17e91d42013-10-21 13:30:52 -070065 ~ThreadEntry();
66
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070067 bool Match(pid_t chk_pid, pid_t chk_tid) { return (chk_pid == pid_ && chk_tid == tid_); }
Christopher Ferris17e91d42013-10-21 13:30:52 -070068
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070069 pid_t pid_;
70 pid_t tid_;
71 int futex_;
72 int ref_count_;
73 pthread_mutex_t mutex_;
74 ThreadEntry* next_;
75 ThreadEntry* prev_;
76 ucontext_t ucontext_;
Christopher Ferris17e91d42013-10-21 13:30:52 -070077
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070078 static ThreadEntry* list_;
79 static pthread_mutex_t list_mutex_;
Christopher Ferris17e91d42013-10-21 13:30:52 -070080};
81
82class BacktraceThread : public BacktraceCurrent {
83public:
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070084 BacktraceThread(BacktraceImpl* impl, pid_t tid, BacktraceMap* map);
Christopher Ferris17e91d42013-10-21 13:30:52 -070085 virtual ~BacktraceThread();
86
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070087 virtual bool Unwind(size_t num_ignore_frames, ucontext_t* ucontext);
Christopher Ferris17e91d42013-10-21 13:30:52 -070088};
89
90#endif // _LIBBACKTRACE_BACKTRACE_THREAD_H