1 /*
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 * DDM-related heap functions
18 */
19 #include <sys/time.h>
20 #include <time.h>
21
22 #include "Dalvik.h"
23 #include "alloc/Heap.h"
24 #include "alloc/HeapInternal.h"
25 #include "alloc/DdmHeap.h"
26 #include "alloc/HeapSource.h"
27
28 #define DEFAULT_HEAP_ID 1
29
30 enum HpifWhen {
31 HPIF_WHEN_NEVER = 0,
32 HPIF_WHEN_NOW = 1,
33 HPIF_WHEN_NEXT_GC = 2,
34 HPIF_WHEN_EVERY_GC = 3
35 };
36
37 /*
38 * Chunk HPIF (client --> server)
39 *
40 * Heap Info. General information about the heap,
41 * suitable for a summary display.
42 *
43 * [u4]: number of heaps
44 *
45 * For each heap:
46 * [u4]: heap ID
47 * [u8]: timestamp in ms since Unix epoch
48 * [u1]: capture reason (same as 'when' value from server)
49 * [u4]: max heap size in bytes (-Xmx)
50 * [u4]: current heap size in bytes
51 * [u4]: current number of bytes allocated
52 * [u4]: current number of objects allocated
53 */
54 #define HPIF_SIZE(numHeaps) \
55 (sizeof(u4) + (numHeaps) * (5 * sizeof(u4) + sizeof(u1) + sizeof(u8)))
dvmDdmSendHeapInfo(int reason,bool shouldLock)56 void dvmDdmSendHeapInfo(int reason, bool shouldLock)
57 {
58 struct timeval now;
59 u8 nowMs;
60 u1 *buf, *b;
61
62 buf = (u1 *)malloc(HPIF_SIZE(1));
63 if (buf == NULL) {
64 return;
65 }
66 b = buf;
67
68 /* If there's a one-shot 'when', reset it.
69 */
70 if (reason == gDvm.gcHeap->ddmHpifWhen) {
71 if (shouldLock && ! dvmLockHeap()) {
72 ALOGW("%s(): can't lock heap to clear when", __func__);
73 goto skip_when;
74 }
75 if (reason == gDvm.gcHeap->ddmHpifWhen) {
76 if (gDvm.gcHeap->ddmHpifWhen == HPIF_WHEN_NEXT_GC) {
77 gDvm.gcHeap->ddmHpifWhen = HPIF_WHEN_NEVER;
78 }
79 }
80 if (shouldLock) {
81 dvmUnlockHeap();
82 }
83 }
84 skip_when:
85
86 /* The current time, in milliseconds since 0:00 GMT, 1/1/70.
87 */
88 if (gettimeofday(&now, NULL) < 0) {
89 nowMs = 0;
90 } else {
91 nowMs = (u8)now.tv_sec * 1000 + now.tv_usec / 1000;
92 }
93
94 /* number of heaps */
95 set4BE(b, 1); b += 4;
96
97 /* For each heap (of which there is one) */
98 {
99 /* heap ID */
100 set4BE(b, DEFAULT_HEAP_ID); b += 4;
101
102 /* timestamp */
103 set8BE(b, nowMs); b += 8;
104
105 /* 'when' value */
106 *b++ = (u1)reason;
107
108 /* max allowed heap size in bytes */
109 set4BE(b, dvmHeapSourceGetMaximumSize()); b += 4;
110
111 /* current heap size in bytes */
112 set4BE(b, dvmHeapSourceGetValue(HS_FOOTPRINT, NULL, 0)); b += 4;
113
114 /* number of bytes allocated */
115 set4BE(b, dvmHeapSourceGetValue(HS_BYTES_ALLOCATED, NULL, 0)); b += 4;
116
117 /* number of objects allocated */
118 set4BE(b, dvmHeapSourceGetValue(HS_OBJECTS_ALLOCATED, NULL, 0)); b += 4;
119 }
120 assert((intptr_t)b == (intptr_t)buf + (intptr_t)HPIF_SIZE(1));
121
122 dvmDbgDdmSendChunk(CHUNK_TYPE("HPIF"), b - buf, buf);
123 }
124
dvmDdmHandleHpifChunk(int when)125 bool dvmDdmHandleHpifChunk(int when)
126 {
127 switch (when) {
128 case HPIF_WHEN_NOW:
129 dvmDdmSendHeapInfo(when, true);
130 break;
131 case HPIF_WHEN_NEVER:
132 case HPIF_WHEN_NEXT_GC:
133 case HPIF_WHEN_EVERY_GC:
134 if (dvmLockHeap()) {
135 gDvm.gcHeap->ddmHpifWhen = when;
136 dvmUnlockHeap();
137 } else {
138 ALOGI("%s(): can't lock heap to set when", __func__);
139 return false;
140 }
141 break;
142 default:
143 ALOGI("%s(): bad when value 0x%08x", __func__, when);
144 return false;
145 }
146
147 return true;
148 }
149
150 enum HpsgSolidity {
151 SOLIDITY_FREE = 0,
152 SOLIDITY_HARD = 1,
153 SOLIDITY_SOFT = 2,
154 SOLIDITY_WEAK = 3,
155 SOLIDITY_PHANTOM = 4,
156 SOLIDITY_FINALIZABLE = 5,
157 SOLIDITY_SWEEP = 6,
158 };
159
160 enum HpsgKind {
161 KIND_OBJECT = 0,
162 KIND_CLASS_OBJECT = 1,
163 KIND_ARRAY_1 = 2,
164 KIND_ARRAY_2 = 3,
165 KIND_ARRAY_4 = 4,
166 KIND_ARRAY_8 = 5,
167 KIND_UNKNOWN = 6,
168 KIND_NATIVE = 7,
169 };
170
171 #define HPSG_PARTIAL (1<<7)
172 #define HPSG_STATE(solidity, kind) \
173 ((u1)((((kind) & 0x7) << 3) | ((solidity) & 0x7)))
174
175 struct HeapChunkContext {
176 u1 *buf;
177 u1 *p;
178 u1 *pieceLenField;
179 size_t bufLen;
180 size_t totalAllocationUnits;
181 int type;
182 bool merge;
183 bool needHeader;
184 };
185
186 #define ALLOCATION_UNIT_SIZE 8
187
flush_hpsg_chunk(HeapChunkContext * ctx)188 static void flush_hpsg_chunk(HeapChunkContext *ctx)
189 {
190 /* Patch the "length of piece" field.
191 */
192 assert(ctx->buf <= ctx->pieceLenField &&
193 ctx->pieceLenField <= ctx->p);
194 set4BE(ctx->pieceLenField, ctx->totalAllocationUnits);
195
196 /* Send the chunk.
197 */
198 dvmDbgDdmSendChunk(ctx->type, ctx->p - ctx->buf, ctx->buf);
199
200 /* Reset the context.
201 */
202 ctx->p = ctx->buf;
203 ctx->totalAllocationUnits = 0;
204 ctx->needHeader = true;
205 ctx->pieceLenField = NULL;
206 }
207
heap_chunk_callback(const void * chunkptr,size_t chunklen,const void * userptr,size_t userlen,void * arg)208 static void heap_chunk_callback(const void *chunkptr, size_t chunklen,
209 const void *userptr, size_t userlen, void *arg)
210 {
211 HeapChunkContext *ctx = (HeapChunkContext *)arg;
212 u1 state;
213
214 UNUSED_PARAMETER(userlen);
215
216 assert((chunklen & (ALLOCATION_UNIT_SIZE-1)) == 0);
217
218 /* Make sure there's enough room left in the buffer.
219 * We need to use two bytes for every fractional 256
220 * allocation units used by the chunk.
221 */
222 {
223 size_t needed = (((chunklen/ALLOCATION_UNIT_SIZE + 255) / 256) * 2);
224 size_t bytesLeft = ctx->bufLen - (size_t)(ctx->p - ctx->buf);
225 if (bytesLeft < needed) {
226 flush_hpsg_chunk(ctx);
227 }
228
229 bytesLeft = ctx->bufLen - (size_t)(ctx->p - ctx->buf);
230 if (bytesLeft < needed) {
231 ALOGW("chunk is too big to transmit (chunklen=%zd, %zd bytes)",
232 chunklen, needed);
233 return;
234 }
235 }
236
237 //TODO: notice when there's a gap and start a new heap, or at least a new range.
238 if (ctx->needHeader) {
239 /*
240 * Start a new HPSx chunk.
241 */
242
243 /* [u4]: heap ID */
244 set4BE(ctx->p, DEFAULT_HEAP_ID); ctx->p += 4;
245
246 /* [u1]: size of allocation unit, in bytes */
247 *ctx->p++ = 8;
248
249 /* [u4]: virtual address of segment start */
250 set4BE(ctx->p, (uintptr_t)chunkptr); ctx->p += 4;
251
252 /* [u4]: offset of this piece (relative to the virtual address) */
253 set4BE(ctx->p, 0); ctx->p += 4;
254
255 /* [u4]: length of piece, in allocation units
256 * We won't know this until we're done, so save the offset
257 * and stuff in a dummy value.
258 */
259 ctx->pieceLenField = ctx->p;
260 set4BE(ctx->p, 0x55555555); ctx->p += 4;
261
262 ctx->needHeader = false;
263 }
264
265 /* Determine the type of this chunk.
266 */
267 if (userptr == NULL) {
268 /* It's a free chunk.
269 */
270 state = HPSG_STATE(SOLIDITY_FREE, 0);
271 } else {
272 const Object *obj = (const Object *)userptr;
273 /* If we're looking at the native heap, we'll just return
274 * (SOLIDITY_HARD, KIND_NATIVE) for all allocated chunks
275 */
276 bool native = ctx->type == CHUNK_TYPE("NHSG");
277
278 /* It's an allocated chunk. Figure out what it is.
279 */
280 //TODO: if ctx.merge, see if this chunk is different from the last chunk.
281 // If it's the same, we should combine them.
282 if (!native && dvmIsValidObject(obj)) {
283 ClassObject *clazz = obj->clazz;
284 if (clazz == NULL) {
285 /* The object was probably just created
286 * but hasn't been initialized yet.
287 */
288 state = HPSG_STATE(SOLIDITY_HARD, KIND_OBJECT);
289 } else if (dvmIsTheClassClass(clazz)) {
290 state = HPSG_STATE(SOLIDITY_HARD, KIND_CLASS_OBJECT);
291 } else if (IS_CLASS_FLAG_SET(clazz, CLASS_ISARRAY)) {
292 if (IS_CLASS_FLAG_SET(clazz, CLASS_ISOBJECTARRAY)) {
293 state = HPSG_STATE(SOLIDITY_HARD, KIND_ARRAY_4);
294 } else {
295 switch (clazz->elementClass->primitiveType) {
296 case PRIM_BOOLEAN:
297 case PRIM_BYTE:
298 state = HPSG_STATE(SOLIDITY_HARD, KIND_ARRAY_1);
299 break;
300 case PRIM_CHAR:
301 case PRIM_SHORT:
302 state = HPSG_STATE(SOLIDITY_HARD, KIND_ARRAY_2);
303 break;
304 case PRIM_INT:
305 case PRIM_FLOAT:
306 state = HPSG_STATE(SOLIDITY_HARD, KIND_ARRAY_4);
307 break;
308 case PRIM_DOUBLE:
309 case PRIM_LONG:
310 state = HPSG_STATE(SOLIDITY_HARD, KIND_ARRAY_8);
311 break;
312 default:
313 assert(!"Unknown GC heap object type");
314 state = HPSG_STATE(SOLIDITY_HARD, KIND_UNKNOWN);
315 break;
316 }
317 }
318 } else {
319 state = HPSG_STATE(SOLIDITY_HARD, KIND_OBJECT);
320 }
321 } else {
322 obj = NULL; // it's not actually an object
323 state = HPSG_STATE(SOLIDITY_HARD, KIND_NATIVE);
324 }
325 }
326
327 /* Write out the chunk description.
328 */
329 chunklen /= ALLOCATION_UNIT_SIZE; // convert to allocation units
330 ctx->totalAllocationUnits += chunklen;
331 while (chunklen > 256) {
332 *ctx->p++ = state | HPSG_PARTIAL;
333 *ctx->p++ = 255; // length - 1
334 chunklen -= 256;
335 }
336 *ctx->p++ = state;
337 *ctx->p++ = chunklen - 1;
338 }
339
340 enum HpsgWhen {
341 HPSG_WHEN_NEVER = 0,
342 HPSG_WHEN_EVERY_GC = 1,
343 };
344 enum HpsgWhat {
345 HPSG_WHAT_MERGED_OBJECTS = 0,
346 HPSG_WHAT_DISTINCT_OBJECTS = 1,
347 };
348
349 /*
350 * Maximum chunk size. Obtain this from the formula:
351 *
352 * (((maximum_heap_size / ALLOCATION_UNIT_SIZE) + 255) / 256) * 2
353 */
354 #define HPSx_CHUNK_SIZE (16384 - 16)
355
356 extern "C" void dlmalloc_walk_heap(void(*)(const void*, size_t, const void*, size_t, void*),void*);
357
walkHeap(bool merge,bool native)358 static void walkHeap(bool merge, bool native)
359 {
360 HeapChunkContext ctx;
361
362 memset(&ctx, 0, sizeof(ctx));
363 ctx.bufLen = HPSx_CHUNK_SIZE;
364 ctx.buf = (u1 *)malloc(ctx.bufLen);
365 if (ctx.buf == NULL) {
366 return;
367 }
368
369 ctx.merge = merge;
370 if (native) {
371 ctx.type = CHUNK_TYPE("NHSG");
372 } else {
373 if (ctx.merge) {
374 ctx.type = CHUNK_TYPE("HPSG");
375 } else {
376 ctx.type = CHUNK_TYPE("HPSO");
377 }
378 }
379
380 ctx.p = ctx.buf;
381 ctx.needHeader = true;
382 if (native) {
383 dlmalloc_walk_heap(heap_chunk_callback, (void *)&ctx);
384 } else {
385 dvmHeapSourceWalk(heap_chunk_callback, (void *)&ctx);
386 }
387 if (ctx.p > ctx.buf) {
388 flush_hpsg_chunk(&ctx);
389 }
390
391 free(ctx.buf);
392 }
393
dvmDdmSendHeapSegments(bool shouldLock,bool native)394 void dvmDdmSendHeapSegments(bool shouldLock, bool native)
395 {
396 u1 heapId[sizeof(u4)];
397 GcHeap *gcHeap = gDvm.gcHeap;
398 int when, what;
399 bool merge;
400
401 /* Don't even grab the lock if there's nothing to do when we're called.
402 */
403 if (!native) {
404 when = gcHeap->ddmHpsgWhen;
405 what = gcHeap->ddmHpsgWhat;
406 if (when == HPSG_WHEN_NEVER) {
407 return;
408 }
409 } else {
410 when = gcHeap->ddmNhsgWhen;
411 what = gcHeap->ddmNhsgWhat;
412 if (when == HPSG_WHEN_NEVER) {
413 return;
414 }
415 }
416 if (shouldLock && !dvmLockHeap()) {
417 ALOGW("Can't lock heap for DDM HPSx dump");
418 return;
419 }
420
421 /* Figure out what kind of chunks we'll be sending.
422 */
423 if (what == HPSG_WHAT_MERGED_OBJECTS) {
424 merge = true;
425 } else if (what == HPSG_WHAT_DISTINCT_OBJECTS) {
426 merge = false;
427 } else {
428 assert(!"bad HPSG.what value");
429 return;
430 }
431
432 /* First, send a heap start chunk.
433 */
434 set4BE(heapId, DEFAULT_HEAP_ID);
435 dvmDbgDdmSendChunk(native ? CHUNK_TYPE("NHST") : CHUNK_TYPE("HPST"),
436 sizeof(u4), heapId);
437
438 /* Send a series of heap segment chunks.
439 */
440 walkHeap(merge, native);
441
442 /* Finally, send a heap end chunk.
443 */
444 dvmDbgDdmSendChunk(native ? CHUNK_TYPE("NHEN") : CHUNK_TYPE("HPEN"),
445 sizeof(u4), heapId);
446
447 if (shouldLock) {
448 dvmUnlockHeap();
449 }
450 }
451
dvmDdmHandleHpsgNhsgChunk(int when,int what,bool native)452 bool dvmDdmHandleHpsgNhsgChunk(int when, int what, bool native)
453 {
454 ALOGI("dvmDdmHandleHpsgChunk(when %d, what %d, heap %d)", when, what,
455 native);
456 switch (when) {
457 case HPSG_WHEN_NEVER:
458 case HPSG_WHEN_EVERY_GC:
459 break;
460 default:
461 ALOGI("%s(): bad when value 0x%08x", __func__, when);
462 return false;
463 }
464
465 switch (what) {
466 case HPSG_WHAT_MERGED_OBJECTS:
467 case HPSG_WHAT_DISTINCT_OBJECTS:
468 break;
469 default:
470 ALOGI("%s(): bad what value 0x%08x", __func__, what);
471 return false;
472 }
473
474 if (dvmLockHeap()) {
475 if (!native) {
476 gDvm.gcHeap->ddmHpsgWhen = when;
477 gDvm.gcHeap->ddmHpsgWhat = what;
478 } else {
479 gDvm.gcHeap->ddmNhsgWhen = when;
480 gDvm.gcHeap->ddmNhsgWhat = what;
481 }
482 //TODO: if what says we should dump immediately, signal (or do) it from here
483 dvmUnlockHeap();
484 } else {
485 ALOGI("%s(): can't lock heap to set when/what", __func__);
486 return false;
487 }
488
489 return true;
490 }
491