blob: 6610347ab8749f7c19fd9717120ea2f449e72cab [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)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100223 LOCKS_EXCLUDED(Locks::thread_list_lock_)
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800224 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100225 static JDWP::JdwpError GetContendedMonitor(JDWP::ObjectId thread_id,
226 JDWP::ObjectId& contended_monitor)
227 LOCKS_EXCLUDED(Locks::thread_list_lock_)
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800228 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
229
230 //
231 // Heap.
232 //
233 static JDWP::JdwpError GetInstanceCounts(const std::vector<JDWP::RefTypeId>& class_ids,
234 std::vector<uint64_t>& counts)
235 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes3b78c942013-01-15 17:35:41 -0800236 static JDWP::JdwpError GetInstances(JDWP::RefTypeId class_id, int32_t max_count,
237 std::vector<JDWP::ObjectId>& instances)
238 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes0cbaff52013-01-16 15:28:01 -0800239 static JDWP::JdwpError GetReferringObjects(JDWP::ObjectId object_id, int32_t max_count,
240 std::vector<JDWP::ObjectId>& referring_objects)
241 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes64f574f2013-02-20 14:57:12 -0800242 static JDWP::JdwpError DisableCollection(JDWP::ObjectId object_id)
243 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
244 static JDWP::JdwpError EnableCollection(JDWP::ObjectId object_id)
245 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
246 static JDWP::JdwpError IsCollected(JDWP::ObjectId object_id, bool& is_collected)
247 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
248 static void DisposeObject(JDWP::ObjectId object_id, uint32_t reference_count)
249 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800250
Elliott Hughes9777ba22013-01-17 09:04:19 -0800251 //
252 // Methods and fields.
253 //
Elliott Hughesa96836a2013-01-17 12:27:49 -0800254 static std::string GetMethodName(JDWP::MethodId method_id)
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 OutputDeclaredFields(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 OutputDeclaredMethods(JDWP::RefTypeId ref_type_id, bool with_generic,
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 JDWP::JdwpError OutputDeclaredInterfaces(JDWP::RefTypeId ref_type_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 OutputLineTable(JDWP::RefTypeId ref_type_id, JDWP::MethodId method_id,
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_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800268 static void OutputVariableTable(JDWP::RefTypeId ref_type_id, JDWP::MethodId id, bool with_generic,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700269 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700270 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Jeff Hao579b0242013-11-18 13:16:49 -0800271 static void OutputMethodReturnValue(JDWP::MethodId method_id, const JValue* return_value,
272 JDWP::ExpandBuf* pReply)
273 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes9777ba22013-01-17 09:04:19 -0800274 static JDWP::JdwpError GetBytecodes(JDWP::RefTypeId class_id, JDWP::MethodId method_id,
275 std::vector<uint8_t>& bytecodes)
276 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700277
Elliott Hughesa96836a2013-01-17 12:27:49 -0800278 static std::string GetFieldName(JDWP::FieldId field_id)
279 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800280 static JDWP::JdwpTag GetFieldBasicTag(JDWP::FieldId field_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700281 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800282 static JDWP::JdwpTag GetStaticFieldBasicTag(JDWP::FieldId field_id)
Brian Carlstromf69863b2013-07-17 21:53:13 -0700283 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800284 static JDWP::JdwpError GetFieldValue(JDWP::ObjectId object_id, JDWP::FieldId field_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700285 JDWP::ExpandBuf* pReply)
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 SetFieldValue(JDWP::ObjectId object_id, JDWP::FieldId field_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700288 uint64_t value, int width)
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 GetStaticFieldValue(JDWP::RefTypeId ref_type_id, JDWP::FieldId field_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700291 JDWP::ExpandBuf* pReply)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700292 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800293 static JDWP::JdwpError SetStaticFieldValue(JDWP::FieldId field_id, uint64_t value, int width)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700294 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700295
Elliott Hughes88d63092013-01-09 09:55:54 -0800296 static std::string StringToUtf8(JDWP::ObjectId string_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700297 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Jeff Hao579b0242013-11-18 13:16:49 -0800298 static void OutputJValue(JDWP::JdwpTag tag, const JValue* return_value, JDWP::ExpandBuf* pReply)
299 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700300
301 /*
302 * Thread, ThreadGroup, Frame
303 */
Elliott Hughes88d63092013-01-09 09:55:54 -0800304 static JDWP::JdwpError GetThreadName(JDWP::ObjectId thread_id, std::string& name)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700305 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_)
306 LOCKS_EXCLUDED(Locks::thread_list_lock_);
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100307 static JDWP::JdwpError GetThreadGroup(JDWP::ObjectId thread_id, JDWP::ExpandBuf* pReply)
308 LOCKS_EXCLUDED(Locks::thread_list_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800309 static std::string GetThreadGroupName(JDWP::ObjectId thread_group_id);
310 static JDWP::ObjectId GetThreadGroupParent(JDWP::ObjectId thread_group_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700311 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700312 static JDWP::ObjectId GetSystemThreadGroupId()
Ian Rogersb726dcb2012-09-05 08:57:23 -0700313 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700314 static JDWP::ObjectId GetMainThreadGroupId();
315
Jeff Hao920af3e2013-08-28 15:46:38 -0700316 static JDWP::JdwpThreadStatus ToJdwpThreadStatus(ThreadState state);
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100317 static JDWP::JdwpError GetThreadStatus(JDWP::ObjectId thread_id,
318 JDWP::JdwpThreadStatus* pThreadStatus,
319 JDWP::JdwpSuspendStatus* pSuspendStatus)
320 LOCKS_EXCLUDED(Locks::thread_list_lock_);
321 static JDWP::JdwpError GetThreadDebugSuspendCount(JDWP::ObjectId thread_id,
322 JDWP::ExpandBuf* pReply)
323 LOCKS_EXCLUDED(Locks::thread_list_lock_,
324 Locks::thread_suspend_count_lock_);
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700325 // static void WaitForSuspend(JDWP::ObjectId thread_id);
Elliott Hughescaf76542012-06-28 16:08:22 -0700326
Elliott Hughes026b1462012-06-28 20:43:49 -0700327 // Fills 'thread_ids' with the threads in the given thread group. If thread_group_id == 0,
Elliott Hughescaf76542012-06-28 16:08:22 -0700328 // returns all threads.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700329 static void GetThreads(JDWP::ObjectId thread_group_id, std::vector<JDWP::ObjectId>& thread_ids)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700330 LOCKS_EXCLUDED(Locks::thread_list_lock_)
331 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughescaf76542012-06-28 16:08:22 -0700332 static void GetChildThreadGroups(JDWP::ObjectId thread_group_id, std::vector<JDWP::ObjectId>& child_thread_group_ids);
333
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100334 static JDWP::JdwpError GetThreadFrameCount(JDWP::ObjectId thread_id, size_t& result)
335 LOCKS_EXCLUDED(Locks::thread_list_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700336 static JDWP::JdwpError GetThreadFrames(JDWP::ObjectId thread_id, size_t start_frame,
337 size_t frame_count, JDWP::ExpandBuf* buf)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100338 LOCKS_EXCLUDED(Locks::thread_list_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700339 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700340
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700341 static JDWP::ObjectId GetThreadSelfId()
Ian Rogersb726dcb2012-09-05 08:57:23 -0700342 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700343 static void SuspendVM()
Ian Rogersb726dcb2012-09-05 08:57:23 -0700344 LOCKS_EXCLUDED(Locks::thread_list_lock_,
345 Locks::thread_suspend_count_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700346 static void ResumeVM();
Elliott Hughes88d63092013-01-09 09:55:54 -0800347 static JDWP::JdwpError SuspendThread(JDWP::ObjectId thread_id, bool request_suspension = true)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700348 LOCKS_EXCLUDED(Locks::mutator_lock_,
349 Locks::thread_list_lock_,
350 Locks::thread_suspend_count_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700351
Elliott Hughes88d63092013-01-09 09:55:54 -0800352 static void ResumeThread(JDWP::ObjectId thread_id)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700353 LOCKS_EXCLUDED(Locks::thread_list_lock_,
354 Locks::thread_suspend_count_lock_)
355 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700356 static void SuspendSelf();
357
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700358 static JDWP::JdwpError GetThisObject(JDWP::ObjectId thread_id, JDWP::FrameId frame_id,
359 JDWP::ObjectId* result)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700360 LOCKS_EXCLUDED(Locks::thread_list_lock_,
361 Locks::thread_suspend_count_lock_)
362 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800363 static void GetLocalValue(JDWP::ObjectId thread_id, JDWP::FrameId frame_id, int slot,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700364 JDWP::JdwpTag tag, uint8_t* buf, size_t expectedLen)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100365 LOCKS_EXCLUDED(Locks::thread_list_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700366 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes88d63092013-01-09 09:55:54 -0800367 static void SetLocalValue(JDWP::ObjectId thread_id, JDWP::FrameId frame_id, int slot,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700368 JDWP::JdwpTag tag, uint64_t value, size_t width)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100369 LOCKS_EXCLUDED(Locks::thread_list_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700370 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700371
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100372 static JDWP::JdwpError Interrupt(JDWP::ObjectId thread_id)
373 LOCKS_EXCLUDED(Locks::thread_list_lock_);
Elliott Hughesf9501702013-01-11 11:22:27 -0800374
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700375 /*
376 * Debugger notification
377 */
378 enum {
Elliott Hughes86964332012-02-15 19:37:42 -0800379 kBreakpoint = 0x01,
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700380 kSingleStep = 0x02,
381 kMethodEntry = 0x04,
382 kMethodExit = 0x08,
383 };
Ian Rogersef7d42f2014-01-06 12:55:46 -0800384 static void PostLocationEvent(mirror::ArtMethod* method, int pcOffset,
Jeff Hao579b0242013-11-18 13:16:49 -0800385 mirror::Object* thisPtr, int eventFlags,
386 const JValue* return_value)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700387 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers62d6c772013-02-27 08:32:07 -0800388 static void PostException(Thread* thread, const ThrowLocation& throw_location,
Brian Carlstromea46f952013-07-30 01:26:50 -0700389 mirror::ArtMethod* catch_method,
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800390 uint32_t catch_dex_pc, mirror::Throwable* exception)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700391 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700392 static void PostThreadStart(Thread* t)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700393 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700394 static void PostThreadDeath(Thread* t)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700395 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800396 static void PostClassPrepare(mirror::Class* c)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700397 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700398
Ian Rogers62d6c772013-02-27 08:32:07 -0800399 static void UpdateDebugger(Thread* thread, mirror::Object* this_object,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800400 mirror::ArtMethod* method, uint32_t new_dex_pc)
jeffhao09bfc6a2012-12-11 18:11:43 -0800401 LOCKS_EXCLUDED(Locks::breakpoint_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700402 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes91bf6cd2012-02-14 17:27:48 -0800403
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100404 // Full Deoptimization control. Only used for method entry/exit and single-stepping.
405 static void EnableFullDeoptimization()
406 LOCKS_EXCLUDED(Locks::deoptimization_lock_)
407 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
408 static void DisableFullDeoptimization()
Ian Rogers719d1a32014-03-06 12:13:39 -0800409 LOCKS_EXCLUDED(Locks::deoptimization_lock_)
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100410 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
411
412 // Manage deoptimization after updating JDWP events list. This must be done while all mutator
413 // threads are suspended.
414 static void ManageDeoptimization()
415 LOCKS_EXCLUDED(Locks::deoptimization_lock_)
416 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
417
418 // Breakpoints.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700419 static void WatchLocation(const JDWP::JdwpLocation* pLoc)
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100420 LOCKS_EXCLUDED(Locks::breakpoint_lock_, Locks::deoptimization_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700421 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700422 static void UnwatchLocation(const JDWP::JdwpLocation* pLoc)
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100423 LOCKS_EXCLUDED(Locks::breakpoint_lock_, Locks::deoptimization_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700424 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz138dbfc2013-12-04 18:15:25 +0100425
426 // Single-stepping.
Elliott Hughes88d63092013-01-09 09:55:54 -0800427 static JDWP::JdwpError ConfigureStep(JDWP::ObjectId thread_id, JDWP::JdwpStepSize size,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700428 JDWP::JdwpStepDepth depth)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700429 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz61b7f1b2013-11-15 15:59:30 +0100430 static void UnconfigureStep(JDWP::ObjectId thread_id)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100431 LOCKS_EXCLUDED(Locks::thread_list_lock_)
Sebastien Hertz61b7f1b2013-11-15 15:59:30 +0100432 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700433
Elliott Hughes88d63092013-01-09 09:55:54 -0800434 static JDWP::JdwpError InvokeMethod(JDWP::ObjectId thread_id, JDWP::ObjectId object_id,
435 JDWP::RefTypeId class_id, JDWP::MethodId method_id,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700436 uint32_t arg_count, uint64_t* arg_values,
437 JDWP::JdwpTag* arg_types, uint32_t options,
438 JDWP::JdwpTag* pResultTag, uint64_t* pResultValue,
439 JDWP::ObjectId* pExceptObj)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700440 LOCKS_EXCLUDED(Locks::thread_list_lock_,
441 Locks::thread_suspend_count_lock_)
442 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700443 static void ExecuteMethod(DebugInvokeReq* pReq);
444
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700445 /*
446 * DDM support.
447 */
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700448 static void DdmSendThreadNotification(Thread* t, uint32_t type)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700449 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100450 static void DdmSetThreadNotification(bool enable)
451 LOCKS_EXCLUDED(Locks::thread_list_lock_);
Elliott Hughes4b9702c2013-02-20 18:13:24 -0800452 static bool DdmHandlePacket(JDWP::Request& request, uint8_t** pReplyBuf, int* pReplyLen);
Ian Rogersb726dcb2012-09-05 08:57:23 -0700453 static void DdmConnected() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
454 static void DdmDisconnected() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700455 static void DdmSendChunk(uint32_t type, const std::vector<uint8_t>& bytes)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700456 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700457 static void DdmSendChunk(uint32_t type, size_t len, const uint8_t* buf)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700458 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700459 static void DdmSendChunkV(uint32_t type, const iovec* iov, int iov_count)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700460 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes767a1472011-10-26 18:49:02 -0700461
Elliott Hughes545a0642011-11-08 19:10:03 -0800462 /*
463 * Recent allocation tracking support.
464 */
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800465 static void RecordAllocation(mirror::Class* type, size_t byte_count)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100466 LOCKS_EXCLUDED(alloc_tracker_lock_)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700467 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100468 static void SetAllocTrackingEnabled(bool enabled) LOCKS_EXCLUDED(alloc_tracker_lock_);
Ian Rogers719d1a32014-03-06 12:13:39 -0800469 static bool IsAllocTrackingEnabled() {
470 return recent_allocation_records_ != nullptr;
471 }
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100472 static jbyteArray GetRecentAllocations()
473 LOCKS_EXCLUDED(alloc_tracker_lock_)
474 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers719d1a32014-03-06 12:13:39 -0800475 static size_t HeadIndex() EXCLUSIVE_LOCKS_REQUIRED(alloc_tracker_lock_);
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100476 static void DumpRecentAllocations() LOCKS_EXCLUDED(alloc_tracker_lock_);
Elliott Hughes545a0642011-11-08 19:10:03 -0800477
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800478 // Updates the stored direct object pointers (called from SweepSystemWeaks).
Mathieu Chartier83c8ee02014-01-28 14:50:23 -0800479 static void UpdateObjectPointers(IsMarkedCallback* callback, void* arg)
Sebastien Hertz52d131d2014-03-13 16:17:40 +0100480 LOCKS_EXCLUDED(alloc_tracker_lock_)
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800481 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
482
Elliott Hughes767a1472011-10-26 18:49:02 -0700483 enum HpifWhen {
484 HPIF_WHEN_NEVER = 0,
485 HPIF_WHEN_NOW = 1,
486 HPIF_WHEN_NEXT_GC = 2,
487 HPIF_WHEN_EVERY_GC = 3
488 };
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700489 static int DdmHandleHpifChunk(HpifWhen when)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700490 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes767a1472011-10-26 18:49:02 -0700491
492 enum HpsgWhen {
493 HPSG_WHEN_NEVER = 0,
494 HPSG_WHEN_EVERY_GC = 1,
495 };
496 enum HpsgWhat {
497 HPSG_WHAT_MERGED_OBJECTS = 0,
498 HPSG_WHAT_DISTINCT_OBJECTS = 1,
499 };
500 static bool DdmHandleHpsgNhsgChunk(HpsgWhen when, HpsgWhat what, bool native);
501
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700502 static void DdmSendHeapInfo(HpifWhen reason)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700503 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700504 static void DdmSendHeapSegments(bool native)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700505 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughes545a0642011-11-08 19:10:03 -0800506
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800507 static void AllowNewObjectRegistryObjects() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
508 static void DisallowNewObjectRegistryObjects() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
509
Elliott Hughes545a0642011-11-08 19:10:03 -0800510 private:
Brian Carlstrom2d888622013-07-18 17:02:00 -0700511 static void DdmBroadcast(bool connect) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700512 static void PostThreadStartOrStop(Thread*, uint32_t)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700513 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
Elliott Hughesa2155262011-11-16 16:26:58 -0800514
Ian Rogers719d1a32014-03-06 12:13:39 -0800515 static Mutex* alloc_tracker_lock_ DEFAULT_MUTEX_ACQUIRED_AFTER;
516
517 static AllocRecord* recent_allocation_records_ PT_GUARDED_BY(alloc_tracker_lock_);
518 static size_t alloc_record_max_ GUARDED_BY(alloc_tracker_lock_);
519 static size_t alloc_record_head_ GUARDED_BY(alloc_tracker_lock_);
520 static size_t alloc_record_count_ GUARDED_BY(alloc_tracker_lock_);
521
522 DISALLOW_COPY_AND_ASSIGN(Dbg);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700523};
524
525#define CHUNK_TYPE(_name) \
Elliott Hughes82188472011-11-07 18:11:48 -0800526 static_cast<uint32_t>((_name)[0] << 24 | (_name)[1] << 16 | (_name)[2] << 8 | (_name)[3])
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700527
528} // namespace art
529
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700530#endif // ART_RUNTIME_DEBUGGER_H_