blob: 6c44bdea8f6bfa1c42b80fce8498188a3390581f [file] [log] [blame]
Elliott Hughes872d4ec2011-10-21 17:07:15 -07001/*
2 * Copyright (C) 2008 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/*
18 * Dalvik-specific side of debugger support. (The JDWP code is intended to
19 * be relatively generic.)
20 */
Brian Carlstromfc0e3212013-07-17 14:40:12 -070021#ifndef ART_RUNTIME_DEBUGGER_H_
22#define ART_RUNTIME_DEBUGGER_H_
Elliott Hughes872d4ec2011-10-21 17:07:15 -070023
24#include <pthread.h>
25
Sebastien Hertz61b7f1b2013-11-15 15:59:30 +010026#include <set>
Elliott Hughes3bb81562011-10-21 18:52:59 -070027#include <string>
28
Elliott Hughes872d4ec2011-10-21 17:07:15 -070029#include "jdwp/jdwp.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080030#include "jni.h"
31#include "jvalue.h"
Mathieu Chartier83c8ee02014-01-28 14:50:23 -080032#include "object_callbacks.h"
Jeff Hao920af3e2013-08-28 15:46:38 -070033#include "thread_state.h"
Elliott Hughes872d4ec2011-10-21 17:07:15 -070034
35namespace art {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080036namespace mirror {
Brian Carlstromea46f952013-07-30 01:26:50 -070037class ArtMethod;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080038class Class;
39class Object;
40class Throwable;
41} // namespace mirror
Elliott Hughes545a0642011-11-08 19:10:03 -080042struct AllocRecord;
Ian Rogers1b09b092012-08-20 15:35:52 -070043class Thread;
Ian Rogers62d6c772013-02-27 08:32:07 -080044class ThrowLocation;
Elliott Hughes872d4ec2011-10-21 17:07:15 -070045
46/*
47 * Invoke-during-breakpoint support.
48 */
49struct DebugInvokeReq {
Elliott Hughes7b3cdfc2011-12-08 21:28:17 -080050 DebugInvokeReq()
Sebastien Hertzd38667a2013-11-25 15:43:54 +010051 : ready(false), invoke_needed(false),
52 receiver(NULL), thread(NULL), klass(NULL), method(NULL),
53 arg_count(0), arg_values(NULL), options(0), error(JDWP::ERR_NONE),
Ian Rogers00f7d0e2012-07-19 15:28:27 -070054 result_tag(JDWP::JT_VOID), exception(0),
Sebastien Hertzd38667a2013-11-25 15:43:54 +010055 lock("a DebugInvokeReq lock", kBreakpointInvokeLock),
56 cond("a DebugInvokeReq condition variable", lock) {
Elliott Hughes475fc232011-10-25 15:00:35 -070057 }
58
Elliott Hughes872d4ec2011-10-21 17:07:15 -070059 /* boolean; only set when we're in the tail end of an event handler */
60 bool ready;
61
62 /* boolean; set if the JDWP thread wants this thread to do work */
Sebastien Hertzd38667a2013-11-25 15:43:54 +010063 bool invoke_needed;
Elliott Hughes872d4ec2011-10-21 17:07:15 -070064
65 /* request */
Sebastien Hertzd38667a2013-11-25 15:43:54 +010066 mirror::Object* receiver; /* not used for ClassType.InvokeMethod */
67 mirror::Object* thread;
68 mirror::Class* klass;
69 mirror::ArtMethod* method;
70 uint32_t arg_count;
71 uint64_t* arg_values; /* will be NULL if arg_count_ == 0 */
72 uint32_t options;
Elliott Hughes872d4ec2011-10-21 17:07:15 -070073
74 /* result */
Elliott Hughes475fc232011-10-25 15:00:35 -070075 JDWP::JdwpError error;
Elliott Hughesd07986f2011-12-06 18:27:45 -080076 JDWP::JdwpTag result_tag;
Elliott Hughes475fc232011-10-25 15:00:35 -070077 JValue result_value;
78 JDWP::ObjectId exception;
Elliott Hughes872d4ec2011-10-21 17:07:15 -070079
80 /* condition variable to wait on while the method executes */
Sebastien Hertzd38667a2013-11-25 15:43:54 +010081 Mutex lock DEFAULT_MUTEX_ACQUIRED_AFTER;
82 ConditionVariable cond GUARDED_BY(lock);
Sebastien Hertz61b7f1b2013-11-15 15:59:30 +010083
84 private:
85 DISALLOW_COPY_AND_ASSIGN(DebugInvokeReq);
86};
87
88// Thread local data-structure that holds fields for controlling single-stepping.
89struct SingleStepControl {
90 SingleStepControl()
91 : is_active(false), step_size(JDWP::SS_MIN), step_depth(JDWP::SD_INTO),
92 method(nullptr), stack_depth(0) {
93 }
94
95 // Are we single-stepping right now?
96 bool is_active;
97
98 // See JdwpStepSize and JdwpStepDepth for details.
99 JDWP::JdwpStepSize step_size;
100 JDWP::JdwpStepDepth step_depth;
101
102 // The location this single-step was initiated from.
103 // A single-step is initiated in a suspended thread. We save here the current method and the
104 // set of DEX pcs associated to the source line number where the suspension occurred.
105 // This is used to support SD_INTO and SD_OVER single-step depths so we detect when a single-step
106 // causes the execution of an instruction in a different method or at a different line number.
107 mirror::ArtMethod* method;
108 std::set<uint32_t> dex_pcs;
109
110 // The stack depth when this single-step was initiated. This is used to support SD_OVER and SD_OUT
111 // single-step depth.
112 int stack_depth;
113
114 private:
115 DISALLOW_COPY_AND_ASSIGN(SingleStepControl);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700116};
117
118class Dbg {
Elliott Hughesff17f1f2012-01-24 18:12:29 -0800119 public:
Elliott Hughes3bb81562011-10-21 18:52:59 -0700120 static bool ParseJdwpOptions(const std::string& options);
Elliott Hughes4ffd3132011-10-24 12:06:42 -0700121 static void SetJdwpAllowed(bool allowed);
122
Elliott Hughesd1cc8362011-10-24 16:58:50 -0700123 static void StartJdwp();
124 static void StopJdwp();
125
Elliott Hughes767a1472011-10-26 18:49:02 -0700126 // Invoked by the GC in case we need to keep DDMS informed.
Ian Rogersb726dcb2012-09-05 08:57:23 -0700127 static void GcDidFinish() LOCKS_EXCLUDED(Locks::mutator_lock_);
Elliott Hughes767a1472011-10-26 18:49:02 -0700128
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700129 // Return the DebugInvokeReq for the current thread.
130 static DebugInvokeReq* GetInvokeReq();
131
Elliott Hughes475fc232011-10-25 15:00:35 -0700132 static Thread* GetDebugThread();
133 static void ClearWaitForEventThread();
134
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700135 /*
136 * Enable/disable breakpoints and step modes. Used to provide a heads-up
137 * when the debugger attaches.
138 */
139 static void Connected();
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100140 static void GoActive()
141 LOCKS_EXCLUDED(Locks::breakpoint_lock_, Locks::deoptimization_lock_, Locks::mutator_lock_);
142 static void Disconnected() LOCKS_EXCLUDED(Locks::deoptimization_lock_, Locks::mutator_lock_);
Elliott Hughes86964332012-02-15 19:37:42 -0800143 static void Disposed();
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700144
Elliott Hughesc0f09332012-03-26 13:27:06 -0700145 // Returns true if we're actually debugging with a real debugger, false if it's
146 // just DDMS (or nothing at all).
147 static bool IsDebuggerActive();
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700148
Elliott Hughesc0f09332012-03-26 13:27:06 -0700149 // Returns true if we had -Xrunjdwp or -agentlib:jdwp= on the command line.
150 static bool IsJdwpConfigured();
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700151
Elliott Hughes86964332012-02-15 19:37:42 -0800152 static bool IsDisposed();
153
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700154 /*
155 * Time, in milliseconds, since the last debugger activity. Does not
156 * include DDMS activity. Returns -1 if there has been no activity.
157 * Returns 0 if we're in the middle of handling a debugger request.
158 */
159 static int64_t LastDebuggerActivity();
160
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700161 static void UndoDebuggerSuspensions();
162
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700163 /*
164 * Class, Object, Array
165 */
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700166 static std::string GetClassName(JDWP::RefTypeId id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700167 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800168 static JDWP::JdwpError GetClassObject(JDWP::RefTypeId id, JDWP::ObjectId& class_object_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700169 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800170 static JDWP::JdwpError GetSuperclass(JDWP::RefTypeId id, JDWP::RefTypeId& superclass_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700171 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700172 static JDWP::JdwpError GetClassLoader(JDWP::RefTypeId id, JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700173 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700174 static JDWP::JdwpError GetModifiers(JDWP::RefTypeId id, JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700175 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800176 static JDWP::JdwpError GetReflectedType(JDWP::RefTypeId class_id, JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700177 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700178 static void GetClassList(std::vector<JDWP::RefTypeId>& classes)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700179 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800180 static JDWP::JdwpError GetClassInfo(JDWP::RefTypeId class_id, JDWP::JdwpTypeTag* pTypeTag,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700181 uint32_t* pStatus, std::string* pDescriptor)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700182 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700183 static void FindLoadedClassBySignature(const char* descriptor, std::vector<JDWP::RefTypeId>& ids)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700184 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes64f574f2013-02-20 14:57:12 -0800185 static JDWP::JdwpError GetReferenceType(JDWP::ObjectId object_id, JDWP::ExpandBuf* pReply)
186 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogersfc0e94b2013-09-23 23:51:32 -0700187 static JDWP::JdwpError GetSignature(JDWP::RefTypeId ref_type_id, std::string* signature)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700188 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800189 static JDWP::JdwpError GetSourceFile(JDWP::RefTypeId ref_type_id, std::string& source_file)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700190 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800191 static JDWP::JdwpError GetObjectTag(JDWP::ObjectId object_id, uint8_t& tag)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700192 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughesaed4be92011-12-02 16:16:23 -0800193 static size_t GetTagWidth(JDWP::JdwpTag tag);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700194
Elliott Hughes88d63092013-01-09 09:55:54 -0800195 static JDWP::JdwpError GetArrayLength(JDWP::ObjectId array_id, int& length)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700196 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800197 static JDWP::JdwpError OutputArray(JDWP::ObjectId array_id, int offset, int count,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700198 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700199 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800200 static JDWP::JdwpError SetArrayElements(JDWP::ObjectId array_id, int offset, int count,
Elliott Hughes4b9702c2013-02-20 18:13:24 -0800201 JDWP::Request& request)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700202 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700203
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700204 static JDWP::ObjectId CreateString(const std::string& str)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700205 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800206 static JDWP::JdwpError CreateObject(JDWP::RefTypeId class_id, JDWP::ObjectId& new_object)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700207 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800208 static JDWP::JdwpError CreateArrayObject(JDWP::RefTypeId array_class_id, uint32_t length,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700209 JDWP::ObjectId& new_array)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700210 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700211
Elliott Hughes88d63092013-01-09 09:55:54 -0800212 static bool MatchType(JDWP::RefTypeId instance_class_id, JDWP::RefTypeId class_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700213 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700214
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800215 //
216 // Monitors.
217 //
218 static JDWP::JdwpError GetMonitorInfo(JDWP::ObjectId object_id, JDWP::ExpandBuf* reply)
219 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
220 static JDWP::JdwpError GetOwnedMonitors(JDWP::ObjectId thread_id,
221 std::vector<JDWP::ObjectId>& monitors,
222 std::vector<uint32_t>& stack_depths)
223 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
224 static JDWP::JdwpError GetContendedMonitor(JDWP::ObjectId thread_id, JDWP::ObjectId& contended_monitor)
225 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
226
227 //
228 // Heap.
229 //
230 static JDWP::JdwpError GetInstanceCounts(const std::vector<JDWP::RefTypeId>& class_ids,
231 std::vector<uint64_t>& counts)
232 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes3b78c942013-01-15 17:35:41 -0800233 static JDWP::JdwpError GetInstances(JDWP::RefTypeId class_id, int32_t max_count,
234 std::vector<JDWP::ObjectId>& instances)
235 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes0cbaff52013-01-16 15:28:01 -0800236 static JDWP::JdwpError GetReferringObjects(JDWP::ObjectId object_id, int32_t max_count,
237 std::vector<JDWP::ObjectId>& referring_objects)
238 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes64f574f2013-02-20 14:57:12 -0800239 static JDWP::JdwpError DisableCollection(JDWP::ObjectId object_id)
240 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
241 static JDWP::JdwpError EnableCollection(JDWP::ObjectId object_id)
242 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
243 static JDWP::JdwpError IsCollected(JDWP::ObjectId object_id, bool& is_collected)
244 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
245 static void DisposeObject(JDWP::ObjectId object_id, uint32_t reference_count)
246 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800247
Elliott Hughes9777ba22013-01-17 09:04:19 -0800248 //
249 // Methods and fields.
250 //
Elliott Hughesa96836a2013-01-17 12:27:49 -0800251 static std::string GetMethodName(JDWP::MethodId method_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700252 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800253 static JDWP::JdwpError OutputDeclaredFields(JDWP::RefTypeId ref_type_id, bool with_generic,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700254 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700255 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800256 static JDWP::JdwpError OutputDeclaredMethods(JDWP::RefTypeId ref_type_id, bool with_generic,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700257 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700258 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800259 static JDWP::JdwpError OutputDeclaredInterfaces(JDWP::RefTypeId ref_type_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700260 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700261 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800262 static void OutputLineTable(JDWP::RefTypeId ref_type_id, JDWP::MethodId method_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700263 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700264 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800265 static void OutputVariableTable(JDWP::RefTypeId ref_type_id, JDWP::MethodId id, bool with_generic,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700266 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700267 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Jeff Hao579b0242013-11-18 13:16:49 -0800268 static void OutputMethodReturnValue(JDWP::MethodId method_id, const JValue* return_value,
269 JDWP::ExpandBuf* pReply)
270 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes9777ba22013-01-17 09:04:19 -0800271 static JDWP::JdwpError GetBytecodes(JDWP::RefTypeId class_id, JDWP::MethodId method_id,
272 std::vector<uint8_t>& bytecodes)
273 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700274
Elliott Hughesa96836a2013-01-17 12:27:49 -0800275 static std::string GetFieldName(JDWP::FieldId field_id)
276 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800277 static JDWP::JdwpTag GetFieldBasicTag(JDWP::FieldId field_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700278 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800279 static JDWP::JdwpTag GetStaticFieldBasicTag(JDWP::FieldId field_id)
Brian Carlstromf69863b2013-07-17 21:53:13 -0700280 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800281 static JDWP::JdwpError GetFieldValue(JDWP::ObjectId object_id, JDWP::FieldId field_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700282 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700283 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800284 static JDWP::JdwpError SetFieldValue(JDWP::ObjectId object_id, JDWP::FieldId field_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700285 uint64_t value, int width)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700286 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800287 static JDWP::JdwpError GetStaticFieldValue(JDWP::RefTypeId ref_type_id, JDWP::FieldId field_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700288 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700289 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800290 static JDWP::JdwpError SetStaticFieldValue(JDWP::FieldId field_id, uint64_t value, int width)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700291 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700292
Elliott Hughes88d63092013-01-09 09:55:54 -0800293 static std::string StringToUtf8(JDWP::ObjectId string_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700294 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Jeff Hao579b0242013-11-18 13:16:49 -0800295 static void OutputJValue(JDWP::JdwpTag tag, const JValue* return_value, JDWP::ExpandBuf* pReply)
296 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700297
298 /*
299 * Thread, ThreadGroup, Frame
300 */
Elliott Hughes88d63092013-01-09 09:55:54 -0800301 static JDWP::JdwpError GetThreadName(JDWP::ObjectId thread_id, std::string& name)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700302 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_)
303 LOCKS_EXCLUDED(Locks::thread_list_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800304 static JDWP::JdwpError GetThreadGroup(JDWP::ObjectId thread_id, JDWP::ExpandBuf* pReply);
305 static std::string GetThreadGroupName(JDWP::ObjectId thread_group_id);
306 static JDWP::ObjectId GetThreadGroupParent(JDWP::ObjectId thread_group_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700307 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700308 static JDWP::ObjectId GetSystemThreadGroupId()
Ian Rogersb726dcb2012-09-05 08:57:23 -0700309 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700310 static JDWP::ObjectId GetMainThreadGroupId();
311
Jeff Hao920af3e2013-08-28 15:46:38 -0700312 static JDWP::JdwpThreadStatus ToJdwpThreadStatus(ThreadState state);
Elliott Hughes88d63092013-01-09 09:55:54 -0800313 static JDWP::JdwpError GetThreadStatus(JDWP::ObjectId thread_id, JDWP::JdwpThreadStatus* pThreadStatus, JDWP::JdwpSuspendStatus* pSuspendStatus);
314 static JDWP::JdwpError GetThreadDebugSuspendCount(JDWP::ObjectId thread_id, JDWP::ExpandBuf* pReply);
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700315 // static void WaitForSuspend(JDWP::ObjectId thread_id);
Elliott Hughescaf76542012-06-28 16:08:22 -0700316
Elliott Hughes026b1462012-06-28 20:43:49 -0700317 // Fills 'thread_ids' with the threads in the given thread group. If thread_group_id == 0,
Elliott Hughescaf76542012-06-28 16:08:22 -0700318 // returns all threads.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700319 static void GetThreads(JDWP::ObjectId thread_group_id, std::vector<JDWP::ObjectId>& thread_ids)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700320 LOCKS_EXCLUDED(Locks::thread_list_lock_)
321 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughescaf76542012-06-28 16:08:22 -0700322 static void GetChildThreadGroups(JDWP::ObjectId thread_group_id, std::vector<JDWP::ObjectId>& child_thread_group_ids);
323
Elliott Hughes88d63092013-01-09 09:55:54 -0800324 static JDWP::JdwpError GetThreadFrameCount(JDWP::ObjectId thread_id, size_t& result);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700325 static JDWP::JdwpError GetThreadFrames(JDWP::ObjectId thread_id, size_t start_frame,
326 size_t frame_count, JDWP::ExpandBuf* buf)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700327 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700328
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700329 static JDWP::ObjectId GetThreadSelfId()
Ian Rogersb726dcb2012-09-05 08:57:23 -0700330 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700331 static void SuspendVM()
Ian Rogersb726dcb2012-09-05 08:57:23 -0700332 LOCKS_EXCLUDED(Locks::thread_list_lock_,
333 Locks::thread_suspend_count_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700334 static void ResumeVM();
Elliott Hughes88d63092013-01-09 09:55:54 -0800335 static JDWP::JdwpError SuspendThread(JDWP::ObjectId thread_id, bool request_suspension = true)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700336 LOCKS_EXCLUDED(Locks::mutator_lock_,
337 Locks::thread_list_lock_,
338 Locks::thread_suspend_count_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700339
Elliott Hughes88d63092013-01-09 09:55:54 -0800340 static void ResumeThread(JDWP::ObjectId thread_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700341 LOCKS_EXCLUDED(Locks::thread_list_lock_,
342 Locks::thread_suspend_count_lock_)
343 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700344 static void SuspendSelf();
345
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700346 static JDWP::JdwpError GetThisObject(JDWP::ObjectId thread_id, JDWP::FrameId frame_id,
347 JDWP::ObjectId* result)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700348 LOCKS_EXCLUDED(Locks::thread_list_lock_,
349 Locks::thread_suspend_count_lock_)
350 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800351 static void GetLocalValue(JDWP::ObjectId thread_id, JDWP::FrameId frame_id, int slot,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700352 JDWP::JdwpTag tag, uint8_t* buf, size_t expectedLen)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700353 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800354 static void SetLocalValue(JDWP::ObjectId thread_id, JDWP::FrameId frame_id, int slot,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700355 JDWP::JdwpTag tag, uint64_t value, size_t width)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700356 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700357
Elliott Hughesf9501702013-01-11 11:22:27 -0800358 static JDWP::JdwpError Interrupt(JDWP::ObjectId thread_id);
359
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700360 /*
361 * Debugger notification
362 */
363 enum {
Elliott Hughes86964332012-02-15 19:37:42 -0800364 kBreakpoint = 0x01,
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700365 kSingleStep = 0x02,
366 kMethodEntry = 0x04,
367 kMethodExit = 0x08,
368 };
Ian Rogersef7d42f2014-01-06 12:55:46 -0800369 static void PostLocationEvent(mirror::ArtMethod* method, int pcOffset,
Jeff Hao579b0242013-11-18 13:16:49 -0800370 mirror::Object* thisPtr, int eventFlags,
371 const JValue* return_value)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700372 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers62d6c772013-02-27 08:32:07 -0800373 static void PostException(Thread* thread, const ThrowLocation& throw_location,
Brian Carlstromea46f952013-07-30 01:26:50 -0700374 mirror::ArtMethod* catch_method,
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800375 uint32_t catch_dex_pc, mirror::Throwable* exception)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700376 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700377 static void PostThreadStart(Thread* t)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700378 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700379 static void PostThreadDeath(Thread* t)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700380 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800381 static void PostClassPrepare(mirror::Class* c)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700382 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700383
Ian Rogers62d6c772013-02-27 08:32:07 -0800384 static void UpdateDebugger(Thread* thread, mirror::Object* this_object,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800385 mirror::ArtMethod* method, uint32_t new_dex_pc)
jeffhao09bfc6a2012-12-11 18:11:43 -0800386 LOCKS_EXCLUDED(Locks::breakpoint_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700387 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes91bf6cd2012-02-14 17:27:48 -0800388
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100389 // Full Deoptimization control. Only used for method entry/exit and single-stepping.
390 static void EnableFullDeoptimization()
391 LOCKS_EXCLUDED(Locks::deoptimization_lock_)
392 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
393 static void DisableFullDeoptimization()
Ian Rogers719d1a32014-03-06 12:13:39 -0800394 LOCKS_EXCLUDED(Locks::deoptimization_lock_)
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100395 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
396
397 // Manage deoptimization after updating JDWP events list. This must be done while all mutator
398 // threads are suspended.
399 static void ManageDeoptimization()
400 LOCKS_EXCLUDED(Locks::deoptimization_lock_)
401 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
402
403 // Breakpoints.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700404 static void WatchLocation(const JDWP::JdwpLocation* pLoc)
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100405 LOCKS_EXCLUDED(Locks::breakpoint_lock_, Locks::deoptimization_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700406 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700407 static void UnwatchLocation(const JDWP::JdwpLocation* pLoc)
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100408 LOCKS_EXCLUDED(Locks::breakpoint_lock_, Locks::deoptimization_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700409 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100410
411 // Single-stepping.
Elliott Hughes88d63092013-01-09 09:55:54 -0800412 static JDWP::JdwpError ConfigureStep(JDWP::ObjectId thread_id, JDWP::JdwpStepSize size,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700413 JDWP::JdwpStepDepth depth)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700414 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz61b7f1b2013-11-15 15:59:30 +0100415 static void UnconfigureStep(JDWP::ObjectId thread_id)
416 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700417
Elliott Hughes88d63092013-01-09 09:55:54 -0800418 static JDWP::JdwpError InvokeMethod(JDWP::ObjectId thread_id, JDWP::ObjectId object_id,
419 JDWP::RefTypeId class_id, JDWP::MethodId method_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700420 uint32_t arg_count, uint64_t* arg_values,
421 JDWP::JdwpTag* arg_types, uint32_t options,
422 JDWP::JdwpTag* pResultTag, uint64_t* pResultValue,
423 JDWP::ObjectId* pExceptObj)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700424 LOCKS_EXCLUDED(Locks::thread_list_lock_,
425 Locks::thread_suspend_count_lock_)
426 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700427 static void ExecuteMethod(DebugInvokeReq* pReq);
428
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700429 /*
430 * DDM support.
431 */
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700432 static void DdmSendThreadNotification(Thread* t, uint32_t type)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700433 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes47fce012011-10-25 18:37:19 -0700434 static void DdmSetThreadNotification(bool enable);
Elliott Hughes4b9702c2013-02-20 18:13:24 -0800435 static bool DdmHandlePacket(JDWP::Request& request, uint8_t** pReplyBuf, int* pReplyLen);
Ian Rogersb726dcb2012-09-05 08:57:23 -0700436 static void DdmConnected() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
437 static void DdmDisconnected() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700438 static void DdmSendChunk(uint32_t type, const std::vector<uint8_t>& bytes)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700439 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700440 static void DdmSendChunk(uint32_t type, size_t len, const uint8_t* buf)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700441 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700442 static void DdmSendChunkV(uint32_t type, const iovec* iov, int iov_count)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700443 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes767a1472011-10-26 18:49:02 -0700444
Elliott Hughes545a0642011-11-08 19:10:03 -0800445 /*
446 * Recent allocation tracking support.
447 */
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800448 static void RecordAllocation(mirror::Class* type, size_t byte_count)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700449 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes545a0642011-11-08 19:10:03 -0800450 static void SetAllocTrackingEnabled(bool enabled);
Ian Rogers719d1a32014-03-06 12:13:39 -0800451 static bool IsAllocTrackingEnabled() {
452 return recent_allocation_records_ != nullptr;
453 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800454 static jbyteArray GetRecentAllocations() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers719d1a32014-03-06 12:13:39 -0800455 static size_t HeadIndex() EXCLUSIVE_LOCKS_REQUIRED(alloc_tracker_lock_);
Elliott Hughes545a0642011-11-08 19:10:03 -0800456 static void DumpRecentAllocations();
457
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800458 // Updates the stored direct object pointers (called from SweepSystemWeaks).
Mathieu Chartier83c8ee02014-01-28 14:50:23 -0800459 static void UpdateObjectPointers(IsMarkedCallback* callback, void* arg)
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800460 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
461
Elliott Hughes767a1472011-10-26 18:49:02 -0700462 enum HpifWhen {
463 HPIF_WHEN_NEVER = 0,
464 HPIF_WHEN_NOW = 1,
465 HPIF_WHEN_NEXT_GC = 2,
466 HPIF_WHEN_EVERY_GC = 3
467 };
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700468 static int DdmHandleHpifChunk(HpifWhen when)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700469 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes767a1472011-10-26 18:49:02 -0700470
471 enum HpsgWhen {
472 HPSG_WHEN_NEVER = 0,
473 HPSG_WHEN_EVERY_GC = 1,
474 };
475 enum HpsgWhat {
476 HPSG_WHAT_MERGED_OBJECTS = 0,
477 HPSG_WHAT_DISTINCT_OBJECTS = 1,
478 };
479 static bool DdmHandleHpsgNhsgChunk(HpsgWhen when, HpsgWhat what, bool native);
480
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700481 static void DdmSendHeapInfo(HpifWhen reason)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700482 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700483 static void DdmSendHeapSegments(bool native)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700484 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes545a0642011-11-08 19:10:03 -0800485
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800486 static void AllowNewObjectRegistryObjects() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
487 static void DisallowNewObjectRegistryObjects() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
488
Elliott Hughes545a0642011-11-08 19:10:03 -0800489 private:
Brian Carlstrom2d888622013-07-18 17:02:00 -0700490 static void DdmBroadcast(bool connect) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700491 static void PostThreadStartOrStop(Thread*, uint32_t)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700492 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughesa2155262011-11-16 16:26:58 -0800493
Ian Rogers719d1a32014-03-06 12:13:39 -0800494 static Mutex* alloc_tracker_lock_ DEFAULT_MUTEX_ACQUIRED_AFTER;
495
496 static AllocRecord* recent_allocation_records_ PT_GUARDED_BY(alloc_tracker_lock_);
497 static size_t alloc_record_max_ GUARDED_BY(alloc_tracker_lock_);
498 static size_t alloc_record_head_ GUARDED_BY(alloc_tracker_lock_);
499 static size_t alloc_record_count_ GUARDED_BY(alloc_tracker_lock_);
500
501 DISALLOW_COPY_AND_ASSIGN(Dbg);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700502};
503
504#define CHUNK_TYPE(_name) \
Elliott Hughes82188472011-11-07 18:11:48 -0800505 static_cast<uint32_t>((_name)[0] << 24 | (_name)[1] << 16 | (_name)[2] << 8 | (_name)[3])
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700506
507} // namespace art
508
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700509#endif // ART_RUNTIME_DEBUGGER_H_