blob: e492d7eb2689ea96763eee8fe5149819a9943c8f [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 * Implementation of an expandable byte buffer. Designed for serializing
18 * primitive values, e.g. JDWP replies.
19 */
20
Elliott Hughes872d4ec2011-10-21 17:07:15 -070021#include "jdwp/jdwp_expand_buf.h"
Elliott Hughes872d4ec2011-10-21 17:07:15 -070022
23#include <stdlib.h>
24#include <string.h>
25
Elliott Hughes07ed66b2012-12-12 18:34:25 -080026#include "base/logging.h"
27#include "jdwp/jdwp.h"
28#include "jdwp/jdwp_bits.h"
29
Elliott Hughes872d4ec2011-10-21 17:07:15 -070030namespace art {
31
32namespace JDWP {
33
34/*
35 * Data structure used to track buffer use.
36 */
37struct ExpandBuf {
38 uint8_t* storage;
39 int curLen;
40 int maxLen;
41};
42
43#define kInitialStorage 64
44
45/*
46 * Allocate a JdwpBuf and some initial storage.
47 */
48ExpandBuf* expandBufAlloc() {
Elliott Hughesa21039c2012-06-21 12:09:25 -070049 ExpandBuf* newBuf = new ExpandBuf;
50 newBuf->storage = reinterpret_cast<uint8_t*>(malloc(kInitialStorage));
Elliott Hughes872d4ec2011-10-21 17:07:15 -070051 newBuf->curLen = 0;
52 newBuf->maxLen = kInitialStorage;
Elliott Hughes872d4ec2011-10-21 17:07:15 -070053 return newBuf;
54}
55
56/*
57 * Free a JdwpBuf and associated storage.
58 */
59void expandBufFree(ExpandBuf* pBuf) {
Sebastien Hertz7d955652014-10-22 10:57:10 +020060 if (pBuf == nullptr) {
Elliott Hughes872d4ec2011-10-21 17:07:15 -070061 return;
62 }
63
64 free(pBuf->storage);
Elliott Hughesa21039c2012-06-21 12:09:25 -070065 delete pBuf;
Elliott Hughes872d4ec2011-10-21 17:07:15 -070066}
67
68/*
69 * Get a pointer to the start of the buffer.
70 */
71uint8_t* expandBufGetBuffer(ExpandBuf* pBuf) {
72 return pBuf->storage;
73}
74
75/*
76 * Get the amount of data currently in the buffer.
77 */
78size_t expandBufGetLength(ExpandBuf* pBuf) {
79 return pBuf->curLen;
80}
81
Elliott Hughes872d4ec2011-10-21 17:07:15 -070082/*
83 * Ensure that the buffer has enough space to hold incoming data. If it
84 * doesn't, resize the buffer.
85 */
86static void ensureSpace(ExpandBuf* pBuf, int newCount) {
87 if (pBuf->curLen + newCount <= pBuf->maxLen) {
88 return;
89 }
90
91 while (pBuf->curLen + newCount > pBuf->maxLen) {
92 pBuf->maxLen *= 2;
93 }
94
Elliott Hughesa21039c2012-06-21 12:09:25 -070095 uint8_t* newPtr = reinterpret_cast<uint8_t*>(realloc(pBuf->storage, pBuf->maxLen));
Sebastien Hertz7d955652014-10-22 10:57:10 +020096 if (newPtr == nullptr) {
Elliott Hughes6c1c69e2012-04-23 16:12:51 -070097 LOG(FATAL) << "realloc(" << pBuf->maxLen << ") failed";
Elliott Hughes872d4ec2011-10-21 17:07:15 -070098 }
99
100 pBuf->storage = newPtr;
101}
102
103/*
104 * Allocate some space in the buffer.
105 */
106uint8_t* expandBufAddSpace(ExpandBuf* pBuf, int gapSize) {
107 uint8_t* gapStart;
108
109 ensureSpace(pBuf, gapSize);
110 gapStart = pBuf->storage + pBuf->curLen;
111 /* do we want to garbage-fill the gap for debugging? */
112 pBuf->curLen += gapSize;
113
114 return gapStart;
115}
116
117/*
118 * Append a byte.
119 */
120void expandBufAdd1(ExpandBuf* pBuf, uint8_t val) {
121 ensureSpace(pBuf, sizeof(val));
122 *(pBuf->storage + pBuf->curLen) = val;
123 pBuf->curLen++;
124}
125
126/*
127 * Append two big-endian bytes.
128 */
129void expandBufAdd2BE(ExpandBuf* pBuf, uint16_t val) {
130 ensureSpace(pBuf, sizeof(val));
Elliott Hughesf7c3b662011-10-27 12:04:56 -0700131 Set2BE(pBuf->storage + pBuf->curLen, val);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700132 pBuf->curLen += sizeof(val);
133}
134
135/*
136 * Append four big-endian bytes.
137 */
138void expandBufAdd4BE(ExpandBuf* pBuf, uint32_t val) {
139 ensureSpace(pBuf, sizeof(val));
Elliott Hughesf7c3b662011-10-27 12:04:56 -0700140 Set4BE(pBuf->storage + pBuf->curLen, val);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700141 pBuf->curLen += sizeof(val);
142}
143
144/*
145 * Append eight big-endian bytes.
146 */
147void expandBufAdd8BE(ExpandBuf* pBuf, uint64_t val) {
148 ensureSpace(pBuf, sizeof(val));
Elliott Hughesf7c3b662011-10-27 12:04:56 -0700149 Set8BE(pBuf->storage + pBuf->curLen, val);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700150 pBuf->curLen += sizeof(val);
151}
152
Elliott Hughesa2155262011-11-16 16:26:58 -0800153static void SetUtf8String(uint8_t* buf, const char* str, size_t strLen) {
Elliott Hughes21f32d72011-11-09 17:44:13 -0800154 Set4BE(buf, strLen);
155 memcpy(buf + sizeof(uint32_t), str, strLen);
156}
157
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700158/*
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700159 * Add a UTF8 string as a 4-byte length followed by a non-nullptr-terminated
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700160 * string.
161 *
162 * Because these strings are coming out of the VM, it's safe to assume that
163 * they can be null-terminated (either they don't have null bytes or they
164 * have stored null bytes in a multi-byte encoding).
165 */
Elliott Hughes4740cdf2011-12-07 14:07:12 -0800166void expandBufAddUtf8String(ExpandBuf* pBuf, const char* s) {
167 int strLen = strlen(s);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700168 ensureSpace(pBuf, sizeof(uint32_t) + strLen);
Elliott Hughes4740cdf2011-12-07 14:07:12 -0800169 SetUtf8String(pBuf->storage + pBuf->curLen, s, strLen);
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700170 pBuf->curLen += sizeof(uint32_t) + strLen;
171}
172
Elliott Hughes4740cdf2011-12-07 14:07:12 -0800173void expandBufAddUtf8String(ExpandBuf* pBuf, const std::string& s) {
174 ensureSpace(pBuf, sizeof(uint32_t) + s.size());
175 SetUtf8String(pBuf->storage + pBuf->curLen, s.data(), s.size());
176 pBuf->curLen += sizeof(uint32_t) + s.size();
177}
178
Elliott Hughes6e9d22c2012-06-22 15:02:37 -0700179void expandBufAddLocation(ExpandBuf* buf, const JdwpLocation& location) {
180 expandBufAdd1(buf, location.type_tag);
181 expandBufAddObjectId(buf, location.class_id);
182 expandBufAddMethodId(buf, location.method_id);
183 expandBufAdd8BE(buf, location.dex_pc);
184}
185
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700186} // namespace JDWP
187
188} // namespace art