blob: 99a8638782fda23f5959461e61a0fb44a9a8390a [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
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070043 void Wake();
44
45 void Wait(int);
46
Christopher Ferrise4846072014-05-23 14:46:36 -070047 void CopyUcontextFromSigcontext(void*);
48
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070049 inline void Lock() {
50 pthread_mutex_lock(&mutex_);
Christopher Ferris3cdbfdc2014-11-08 15:57:11 -080051
52 // Always reset the wait value since this could be the first or nth
53 // time this entry is locked.
54 wait_value_ = 0;
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070055 }
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_;
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070071 int ref_count_;
72 pthread_mutex_t mutex_;
Christopher Ferris3cdbfdc2014-11-08 15:57:11 -080073 pthread_mutex_t wait_mutex_;
74 pthread_cond_t wait_cond_;
75 int wait_value_;
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070076 ThreadEntry* next_;
77 ThreadEntry* prev_;
78 ucontext_t ucontext_;
Christopher Ferris17e91d42013-10-21 13:30:52 -070079
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070080 static ThreadEntry* list_;
81 static pthread_mutex_t list_mutex_;
Christopher Ferris17e91d42013-10-21 13:30:52 -070082};
83
84class BacktraceThread : public BacktraceCurrent {
85public:
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070086 BacktraceThread(BacktraceImpl* impl, pid_t tid, BacktraceMap* map);
Christopher Ferris17e91d42013-10-21 13:30:52 -070087 virtual ~BacktraceThread();
88
Christopher Ferrisa2efd3a2014-05-06 15:23:59 -070089 virtual bool Unwind(size_t num_ignore_frames, ucontext_t* ucontext);
Christopher Ferris17e91d42013-10-21 13:30:52 -070090};
91
92#endif // _LIBBACKTRACE_BACKTRACE_THREAD_H