1*e82f7db8SAndroid Build Coastguard Worker /*
2*e82f7db8SAndroid Build Coastguard Worker * Copyright (c) 1998, 2005, Oracle and/or its affiliates. All rights reserved.
3*e82f7db8SAndroid Build Coastguard Worker * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*e82f7db8SAndroid Build Coastguard Worker *
5*e82f7db8SAndroid Build Coastguard Worker * This code is free software; you can redistribute it and/or modify it
6*e82f7db8SAndroid Build Coastguard Worker * under the terms of the GNU General Public License version 2 only, as
7*e82f7db8SAndroid Build Coastguard Worker * published by the Free Software Foundation. Oracle designates this
8*e82f7db8SAndroid Build Coastguard Worker * particular file as subject to the "Classpath" exception as provided
9*e82f7db8SAndroid Build Coastguard Worker * by Oracle in the LICENSE file that accompanied this code.
10*e82f7db8SAndroid Build Coastguard Worker *
11*e82f7db8SAndroid Build Coastguard Worker * This code is distributed in the hope that it will be useful, but WITHOUT
12*e82f7db8SAndroid Build Coastguard Worker * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13*e82f7db8SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14*e82f7db8SAndroid Build Coastguard Worker * version 2 for more details (a copy is included in the LICENSE file that
15*e82f7db8SAndroid Build Coastguard Worker * accompanied this code).
16*e82f7db8SAndroid Build Coastguard Worker *
17*e82f7db8SAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License version
18*e82f7db8SAndroid Build Coastguard Worker * 2 along with this work; if not, write to the Free Software Foundation,
19*e82f7db8SAndroid Build Coastguard Worker * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20*e82f7db8SAndroid Build Coastguard Worker *
21*e82f7db8SAndroid Build Coastguard Worker * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22*e82f7db8SAndroid Build Coastguard Worker * or visit www.oracle.com if you need additional information or have any
23*e82f7db8SAndroid Build Coastguard Worker * questions.
24*e82f7db8SAndroid Build Coastguard Worker */
25*e82f7db8SAndroid Build Coastguard Worker
26*e82f7db8SAndroid Build Coastguard Worker #include "util.h"
27*e82f7db8SAndroid Build Coastguard Worker #include "MethodImpl.h"
28*e82f7db8SAndroid Build Coastguard Worker #include "inStream.h"
29*e82f7db8SAndroid Build Coastguard Worker #include "outStream.h"
30*e82f7db8SAndroid Build Coastguard Worker
31*e82f7db8SAndroid Build Coastguard Worker static jboolean
lineTable(PacketInputStream * in,PacketOutputStream * out)32*e82f7db8SAndroid Build Coastguard Worker lineTable(PacketInputStream *in, PacketOutputStream *out)
33*e82f7db8SAndroid Build Coastguard Worker {
34*e82f7db8SAndroid Build Coastguard Worker jvmtiError error;
35*e82f7db8SAndroid Build Coastguard Worker jint count = 0;
36*e82f7db8SAndroid Build Coastguard Worker jvmtiLineNumberEntry *table = NULL;
37*e82f7db8SAndroid Build Coastguard Worker jmethodID method;
38*e82f7db8SAndroid Build Coastguard Worker jlocation firstCodeIndex;
39*e82f7db8SAndroid Build Coastguard Worker jlocation lastCodeIndex;
40*e82f7db8SAndroid Build Coastguard Worker jboolean isNative;
41*e82f7db8SAndroid Build Coastguard Worker
42*e82f7db8SAndroid Build Coastguard Worker /* JVMDI needed the class, but JVMTI does not so we ignore it */
43*e82f7db8SAndroid Build Coastguard Worker (void)inStream_readClassRef(getEnv(), in);
44*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
45*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
46*e82f7db8SAndroid Build Coastguard Worker }
47*e82f7db8SAndroid Build Coastguard Worker method = inStream_readMethodID(in);
48*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
49*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
50*e82f7db8SAndroid Build Coastguard Worker }
51*e82f7db8SAndroid Build Coastguard Worker
52*e82f7db8SAndroid Build Coastguard Worker /*
53*e82f7db8SAndroid Build Coastguard Worker * JVMTI behavior for the calls below is unspecified for native
54*e82f7db8SAndroid Build Coastguard Worker * methods, so we must check explicitly.
55*e82f7db8SAndroid Build Coastguard Worker */
56*e82f7db8SAndroid Build Coastguard Worker isNative = isMethodNative(method);
57*e82f7db8SAndroid Build Coastguard Worker if (isNative) {
58*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, JDWP_ERROR(NATIVE_METHOD));
59*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
60*e82f7db8SAndroid Build Coastguard Worker }
61*e82f7db8SAndroid Build Coastguard Worker
62*e82f7db8SAndroid Build Coastguard Worker error = methodLocation(method, &firstCodeIndex, &lastCodeIndex);
63*e82f7db8SAndroid Build Coastguard Worker if (error != JVMTI_ERROR_NONE) {
64*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, map2jdwpError(error));
65*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
66*e82f7db8SAndroid Build Coastguard Worker }
67*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeLocation(out, firstCodeIndex);
68*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeLocation(out, lastCodeIndex);
69*e82f7db8SAndroid Build Coastguard Worker
70*e82f7db8SAndroid Build Coastguard Worker error = JVMTI_FUNC_PTR(gdata->jvmti,GetLineNumberTable)
71*e82f7db8SAndroid Build Coastguard Worker (gdata->jvmti, method, &count, &table);
72*e82f7db8SAndroid Build Coastguard Worker if (error == JVMTI_ERROR_ABSENT_INFORMATION) {
73*e82f7db8SAndroid Build Coastguard Worker /*
74*e82f7db8SAndroid Build Coastguard Worker * Indicate no line info with an empty table. The code indices
75*e82f7db8SAndroid Build Coastguard Worker * are still useful, so we don't want to return an error
76*e82f7db8SAndroid Build Coastguard Worker */
77*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, 0);
78*e82f7db8SAndroid Build Coastguard Worker } else if (error == JVMTI_ERROR_NONE) {
79*e82f7db8SAndroid Build Coastguard Worker jint i;
80*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, count);
81*e82f7db8SAndroid Build Coastguard Worker for (i = 0; (i < count) && !outStream_error(out); i++) {
82*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeLocation(out, table[i].start_location);
83*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, table[i].line_number);
84*e82f7db8SAndroid Build Coastguard Worker }
85*e82f7db8SAndroid Build Coastguard Worker jvmtiDeallocate(table);
86*e82f7db8SAndroid Build Coastguard Worker } else {
87*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, map2jdwpError(error));
88*e82f7db8SAndroid Build Coastguard Worker }
89*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
90*e82f7db8SAndroid Build Coastguard Worker }
91*e82f7db8SAndroid Build Coastguard Worker
92*e82f7db8SAndroid Build Coastguard Worker
93*e82f7db8SAndroid Build Coastguard Worker static jboolean
doVariableTable(PacketInputStream * in,PacketOutputStream * out,int outputGenerics)94*e82f7db8SAndroid Build Coastguard Worker doVariableTable(PacketInputStream *in, PacketOutputStream *out,
95*e82f7db8SAndroid Build Coastguard Worker int outputGenerics)
96*e82f7db8SAndroid Build Coastguard Worker {
97*e82f7db8SAndroid Build Coastguard Worker jvmtiError error;
98*e82f7db8SAndroid Build Coastguard Worker jint count;
99*e82f7db8SAndroid Build Coastguard Worker jvmtiLocalVariableEntry *table;
100*e82f7db8SAndroid Build Coastguard Worker jmethodID method;
101*e82f7db8SAndroid Build Coastguard Worker jint argsSize;
102*e82f7db8SAndroid Build Coastguard Worker jboolean isNative;
103*e82f7db8SAndroid Build Coastguard Worker
104*e82f7db8SAndroid Build Coastguard Worker /* JVMDI needed the class, but JVMTI does not so we ignore it */
105*e82f7db8SAndroid Build Coastguard Worker (void)inStream_readClassRef(getEnv(), in);
106*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
107*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
108*e82f7db8SAndroid Build Coastguard Worker }
109*e82f7db8SAndroid Build Coastguard Worker method = inStream_readMethodID(in);
110*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
111*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
112*e82f7db8SAndroid Build Coastguard Worker }
113*e82f7db8SAndroid Build Coastguard Worker
114*e82f7db8SAndroid Build Coastguard Worker /*
115*e82f7db8SAndroid Build Coastguard Worker * JVMTI behavior for the calls below is unspecified for native
116*e82f7db8SAndroid Build Coastguard Worker * methods, so we must check explicitly.
117*e82f7db8SAndroid Build Coastguard Worker */
118*e82f7db8SAndroid Build Coastguard Worker isNative = isMethodNative(method);
119*e82f7db8SAndroid Build Coastguard Worker if (isNative) {
120*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, JDWP_ERROR(NATIVE_METHOD));
121*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
122*e82f7db8SAndroid Build Coastguard Worker }
123*e82f7db8SAndroid Build Coastguard Worker
124*e82f7db8SAndroid Build Coastguard Worker error = JVMTI_FUNC_PTR(gdata->jvmti,GetArgumentsSize)
125*e82f7db8SAndroid Build Coastguard Worker (gdata->jvmti, method, &argsSize);
126*e82f7db8SAndroid Build Coastguard Worker if (error != JVMTI_ERROR_NONE) {
127*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, map2jdwpError(error));
128*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
129*e82f7db8SAndroid Build Coastguard Worker }
130*e82f7db8SAndroid Build Coastguard Worker
131*e82f7db8SAndroid Build Coastguard Worker error = JVMTI_FUNC_PTR(gdata->jvmti,GetLocalVariableTable)
132*e82f7db8SAndroid Build Coastguard Worker (gdata->jvmti, method, &count, &table);
133*e82f7db8SAndroid Build Coastguard Worker if (error == JVMTI_ERROR_NONE) {
134*e82f7db8SAndroid Build Coastguard Worker jint i;
135*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, argsSize);
136*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, count);
137*e82f7db8SAndroid Build Coastguard Worker for (i = 0; (i < count) && !outStream_error(out); i++) {
138*e82f7db8SAndroid Build Coastguard Worker jvmtiLocalVariableEntry *entry = &table[i];
139*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeLocation(out, entry->start_location);
140*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeString(out, entry->name);
141*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeString(out, entry->signature);
142*e82f7db8SAndroid Build Coastguard Worker if (outputGenerics == 1) {
143*e82f7db8SAndroid Build Coastguard Worker writeGenericSignature(out, entry->generic_signature);
144*e82f7db8SAndroid Build Coastguard Worker }
145*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, entry->length);
146*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeInt(out, entry->slot);
147*e82f7db8SAndroid Build Coastguard Worker
148*e82f7db8SAndroid Build Coastguard Worker jvmtiDeallocate(entry->name);
149*e82f7db8SAndroid Build Coastguard Worker jvmtiDeallocate(entry->signature);
150*e82f7db8SAndroid Build Coastguard Worker if (entry->generic_signature != NULL) {
151*e82f7db8SAndroid Build Coastguard Worker jvmtiDeallocate(entry->generic_signature);
152*e82f7db8SAndroid Build Coastguard Worker }
153*e82f7db8SAndroid Build Coastguard Worker }
154*e82f7db8SAndroid Build Coastguard Worker
155*e82f7db8SAndroid Build Coastguard Worker jvmtiDeallocate(table);
156*e82f7db8SAndroid Build Coastguard Worker } else {
157*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, map2jdwpError(error));
158*e82f7db8SAndroid Build Coastguard Worker }
159*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
160*e82f7db8SAndroid Build Coastguard Worker }
161*e82f7db8SAndroid Build Coastguard Worker
162*e82f7db8SAndroid Build Coastguard Worker
163*e82f7db8SAndroid Build Coastguard Worker static jboolean
variableTable(PacketInputStream * in,PacketOutputStream * out)164*e82f7db8SAndroid Build Coastguard Worker variableTable(PacketInputStream *in, PacketOutputStream *out) {
165*e82f7db8SAndroid Build Coastguard Worker return doVariableTable(in, out, 0);
166*e82f7db8SAndroid Build Coastguard Worker }
167*e82f7db8SAndroid Build Coastguard Worker
168*e82f7db8SAndroid Build Coastguard Worker static jboolean
variableTableWithGenerics(PacketInputStream * in,PacketOutputStream * out)169*e82f7db8SAndroid Build Coastguard Worker variableTableWithGenerics(PacketInputStream *in, PacketOutputStream *out) {
170*e82f7db8SAndroid Build Coastguard Worker return doVariableTable(in, out, 1);
171*e82f7db8SAndroid Build Coastguard Worker }
172*e82f7db8SAndroid Build Coastguard Worker
173*e82f7db8SAndroid Build Coastguard Worker
174*e82f7db8SAndroid Build Coastguard Worker static jboolean
bytecodes(PacketInputStream * in,PacketOutputStream * out)175*e82f7db8SAndroid Build Coastguard Worker bytecodes(PacketInputStream *in, PacketOutputStream *out)
176*e82f7db8SAndroid Build Coastguard Worker {
177*e82f7db8SAndroid Build Coastguard Worker jvmtiError error;
178*e82f7db8SAndroid Build Coastguard Worker unsigned char * bcp;
179*e82f7db8SAndroid Build Coastguard Worker jint bytecodeCount;
180*e82f7db8SAndroid Build Coastguard Worker jmethodID method;
181*e82f7db8SAndroid Build Coastguard Worker
182*e82f7db8SAndroid Build Coastguard Worker /* JVMDI needed the class, but JVMTI does not so we ignore it */
183*e82f7db8SAndroid Build Coastguard Worker (void)inStream_readClassRef(getEnv(), in);
184*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
185*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
186*e82f7db8SAndroid Build Coastguard Worker }
187*e82f7db8SAndroid Build Coastguard Worker method = inStream_readMethodID(in);
188*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
189*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
190*e82f7db8SAndroid Build Coastguard Worker }
191*e82f7db8SAndroid Build Coastguard Worker
192*e82f7db8SAndroid Build Coastguard Worker /* Initialize assuming no bytecodes and no error */
193*e82f7db8SAndroid Build Coastguard Worker error = JVMTI_ERROR_NONE;
194*e82f7db8SAndroid Build Coastguard Worker bytecodeCount = 0;
195*e82f7db8SAndroid Build Coastguard Worker bcp = NULL;
196*e82f7db8SAndroid Build Coastguard Worker
197*e82f7db8SAndroid Build Coastguard Worker /* Only non-native methods have bytecodes, don't even ask if native. */
198*e82f7db8SAndroid Build Coastguard Worker if ( !isMethodNative(method) ) {
199*e82f7db8SAndroid Build Coastguard Worker error = JVMTI_FUNC_PTR(gdata->jvmti,GetBytecodes)
200*e82f7db8SAndroid Build Coastguard Worker (gdata->jvmti, method, &bytecodeCount, &bcp);
201*e82f7db8SAndroid Build Coastguard Worker }
202*e82f7db8SAndroid Build Coastguard Worker if (error != JVMTI_ERROR_NONE) {
203*e82f7db8SAndroid Build Coastguard Worker outStream_setError(out, map2jdwpError(error));
204*e82f7db8SAndroid Build Coastguard Worker } else {
205*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeByteArray(out, bytecodeCount, (jbyte *)bcp);
206*e82f7db8SAndroid Build Coastguard Worker jvmtiDeallocate(bcp);
207*e82f7db8SAndroid Build Coastguard Worker }
208*e82f7db8SAndroid Build Coastguard Worker
209*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
210*e82f7db8SAndroid Build Coastguard Worker }
211*e82f7db8SAndroid Build Coastguard Worker
212*e82f7db8SAndroid Build Coastguard Worker static jboolean
isObsolete(PacketInputStream * in,PacketOutputStream * out)213*e82f7db8SAndroid Build Coastguard Worker isObsolete(PacketInputStream *in, PacketOutputStream *out)
214*e82f7db8SAndroid Build Coastguard Worker {
215*e82f7db8SAndroid Build Coastguard Worker jboolean isObsolete;
216*e82f7db8SAndroid Build Coastguard Worker jmethodID method;
217*e82f7db8SAndroid Build Coastguard Worker
218*e82f7db8SAndroid Build Coastguard Worker /* JVMDI needed the class, but JVMTI does not so we ignore it */
219*e82f7db8SAndroid Build Coastguard Worker (void)inStream_readClassRef(getEnv(), in);
220*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
221*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
222*e82f7db8SAndroid Build Coastguard Worker }
223*e82f7db8SAndroid Build Coastguard Worker method = inStream_readMethodID(in);
224*e82f7db8SAndroid Build Coastguard Worker if (inStream_error(in)) {
225*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
226*e82f7db8SAndroid Build Coastguard Worker }
227*e82f7db8SAndroid Build Coastguard Worker
228*e82f7db8SAndroid Build Coastguard Worker isObsolete = isMethodObsolete(method);
229*e82f7db8SAndroid Build Coastguard Worker (void)outStream_writeBoolean(out, isObsolete);
230*e82f7db8SAndroid Build Coastguard Worker
231*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
232*e82f7db8SAndroid Build Coastguard Worker }
233*e82f7db8SAndroid Build Coastguard Worker
234*e82f7db8SAndroid Build Coastguard Worker void *Method_Cmds[] = { (void *)0x5
235*e82f7db8SAndroid Build Coastguard Worker ,(void *)lineTable
236*e82f7db8SAndroid Build Coastguard Worker ,(void *)variableTable
237*e82f7db8SAndroid Build Coastguard Worker ,(void *)bytecodes
238*e82f7db8SAndroid Build Coastguard Worker ,(void *)isObsolete
239*e82f7db8SAndroid Build Coastguard Worker ,(void *)variableTableWithGenerics
240*e82f7db8SAndroid Build Coastguard Worker };
241