src/share/vm/ci/ciStreams.cpp

Wed, 09 Jun 2010 18:50:45 -0700

author
jrose
date
Wed, 09 Jun 2010 18:50:45 -0700
changeset 1957
136b78722a08
parent 1934
e9ff18c4ace7
child 2314
f95d63e2154a
permissions
-rw-r--r--

6939203: JSR 292 needs method handle constants
Summary: Add new CP types CONSTANT_MethodHandle, CONSTANT_MethodType; extend 'ldc' bytecode.
Reviewed-by: twisti, never

duke@435 1 /*
jrose@1934 2 * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved.
duke@435 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
duke@435 4 *
duke@435 5 * This code is free software; you can redistribute it and/or modify it
duke@435 6 * under the terms of the GNU General Public License version 2 only, as
duke@435 7 * published by the Free Software Foundation.
duke@435 8 *
duke@435 9 * This code is distributed in the hope that it will be useful, but WITHOUT
duke@435 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
duke@435 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
duke@435 12 * version 2 for more details (a copy is included in the LICENSE file that
duke@435 13 * accompanied this code).
duke@435 14 *
duke@435 15 * You should have received a copy of the GNU General Public License version
duke@435 16 * 2 along with this work; if not, write to the Free Software Foundation,
duke@435 17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
duke@435 18 *
trims@1907 19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
trims@1907 20 * or visit www.oracle.com if you need additional information or have any
trims@1907 21 * questions.
duke@435 22 *
duke@435 23 */
duke@435 24
duke@435 25 #include "incls/_precompiled.incl"
duke@435 26 #include "incls/_ciStreams.cpp.incl"
duke@435 27
duke@435 28 // ciExceptionHandlerStream
duke@435 29 //
duke@435 30 // Walk over some selected set of a methods exception handlers.
duke@435 31
duke@435 32 // ------------------------------------------------------------------
duke@435 33 // ciExceptionHandlerStream::count
duke@435 34 //
duke@435 35 // How many exception handlers are there in this stream?
duke@435 36 //
duke@435 37 // Implementation note: Compiler2 needs this functionality, so I had
duke@435 38 int ciExceptionHandlerStream::count() {
duke@435 39 int save_pos = _pos;
duke@435 40 int save_end = _end;
duke@435 41
duke@435 42 int count = 0;
duke@435 43
duke@435 44 _pos = -1;
duke@435 45 _end = _method->_handler_count;
duke@435 46
duke@435 47
duke@435 48 next();
duke@435 49 while (!is_done()) {
duke@435 50 count++;
duke@435 51 next();
duke@435 52 }
duke@435 53
duke@435 54 _pos = save_pos;
duke@435 55 _end = save_end;
duke@435 56
duke@435 57 return count;
duke@435 58 }
duke@435 59
duke@435 60 int ciExceptionHandlerStream::count_remaining() {
duke@435 61 int save_pos = _pos;
duke@435 62 int save_end = _end;
duke@435 63
duke@435 64 int count = 0;
duke@435 65
duke@435 66 while (!is_done()) {
duke@435 67 count++;
duke@435 68 next();
duke@435 69 }
duke@435 70
duke@435 71 _pos = save_pos;
duke@435 72 _end = save_end;
duke@435 73
duke@435 74 return count;
duke@435 75 }
duke@435 76
duke@435 77 // ciBytecodeStream
duke@435 78 //
duke@435 79 // The class is used to iterate over the bytecodes of a method.
duke@435 80 // It hides the details of constant pool structure/access by
duke@435 81 // providing accessors for constant pool items.
duke@435 82
duke@435 83 // ------------------------------------------------------------------
jrose@1920 84 // ciBytecodeStream::next_wide_or_table
duke@435 85 //
duke@435 86 // Special handling for switch ops
jrose@1920 87 Bytecodes::Code ciBytecodeStream::next_wide_or_table(Bytecodes::Code bc) {
jrose@1920 88 switch (bc) { // Check for special bytecode handling
jrose@1920 89 case Bytecodes::_wide:
jrose@1920 90 // Special handling for the wide bytcode
jrose@1920 91 // Get following bytecode; do not return wide
jrose@1920 92 assert(Bytecodes::Code(_pc[0]) == Bytecodes::_wide, "");
jrose@1920 93 bc = Bytecodes::java_code(_raw_bc = (Bytecodes::Code)_pc[1]);
jrose@1920 94 assert(Bytecodes::wide_length_for(bc) > 2, "must make progress");
jrose@1920 95 _pc += Bytecodes::wide_length_for(bc);
jrose@1920 96 _was_wide = _pc; // Flag last wide bytecode found
jrose@1920 97 assert(is_wide(), "accessor works right");
jrose@1920 98 break;
duke@435 99
duke@435 100 case Bytecodes::_lookupswitch:
duke@435 101 _pc++; // Skip wide bytecode
duke@435 102 _pc += (_start-_pc)&3; // Word align
duke@435 103 _table_base = (jint*)_pc; // Capture for later usage
duke@435 104 // table_base[0] is default far_dest
duke@435 105 // Table has 2 lead elements (default, length), then pairs of u4 values.
duke@435 106 // So load table length, and compute address at end of table
duke@435 107 _pc = (address)&_table_base[2+ 2*Bytes::get_Java_u4((address)&_table_base[1])];
duke@435 108 break;
duke@435 109
duke@435 110 case Bytecodes::_tableswitch: {
duke@435 111 _pc++; // Skip wide bytecode
duke@435 112 _pc += (_start-_pc)&3; // Word align
duke@435 113 _table_base = (jint*)_pc; // Capture for later usage
duke@435 114 // table_base[0] is default far_dest
duke@435 115 int lo = Bytes::get_Java_u4((address)&_table_base[1]);// Low bound
duke@435 116 int hi = Bytes::get_Java_u4((address)&_table_base[2]);// High bound
duke@435 117 int len = hi - lo + 1; // Dense table size
duke@435 118 _pc = (address)&_table_base[3+len]; // Skip past table
duke@435 119 break;
duke@435 120 }
duke@435 121
duke@435 122 default:
duke@435 123 fatal("unhandled bytecode");
duke@435 124 }
duke@435 125 return bc;
duke@435 126 }
duke@435 127
duke@435 128 // ------------------------------------------------------------------
duke@435 129 // ciBytecodeStream::reset_to_bci
duke@435 130 void ciBytecodeStream::reset_to_bci( int bci ) {
duke@435 131 _bc_start=_was_wide=0;
duke@435 132 _pc = _start+bci;
duke@435 133 }
duke@435 134
duke@435 135 // ------------------------------------------------------------------
duke@435 136 // ciBytecodeStream::force_bci
duke@435 137 void ciBytecodeStream::force_bci(int bci) {
duke@435 138 if (bci < 0) {
duke@435 139 reset_to_bci(0);
duke@435 140 _bc_start = _start + bci;
duke@435 141 _bc = EOBC();
duke@435 142 } else {
duke@435 143 reset_to_bci(bci);
duke@435 144 next();
duke@435 145 }
duke@435 146 }
duke@435 147
duke@435 148
duke@435 149 // ------------------------------------------------------------------
duke@435 150 // Constant pool access
duke@435 151 // ------------------------------------------------------------------
duke@435 152
duke@435 153 // ------------------------------------------------------------------
duke@435 154 // ciBytecodeStream::get_klass_index
duke@435 155 //
duke@435 156 // If this bytecodes references a klass, return the index of the
duke@435 157 // referenced klass.
duke@435 158 int ciBytecodeStream::get_klass_index() const {
duke@435 159 switch(cur_bc()) {
duke@435 160 case Bytecodes::_ldc:
jrose@1920 161 return get_index_u1();
duke@435 162 case Bytecodes::_ldc_w:
duke@435 163 case Bytecodes::_ldc2_w:
duke@435 164 case Bytecodes::_checkcast:
duke@435 165 case Bytecodes::_instanceof:
duke@435 166 case Bytecodes::_anewarray:
duke@435 167 case Bytecodes::_multianewarray:
duke@435 168 case Bytecodes::_new:
duke@435 169 case Bytecodes::_newarray:
jrose@1920 170 return get_index_u2();
duke@435 171 default:
duke@435 172 ShouldNotReachHere();
duke@435 173 return 0;
duke@435 174 }
duke@435 175 }
duke@435 176
duke@435 177 // ------------------------------------------------------------------
duke@435 178 // ciBytecodeStream::get_klass
duke@435 179 //
duke@435 180 // If this bytecode is a new, newarray, multianewarray, instanceof,
duke@435 181 // or checkcast, get the referenced klass.
duke@435 182 ciKlass* ciBytecodeStream::get_klass(bool& will_link) {
twisti@1573 183 VM_ENTRY_MARK;
twisti@1573 184 constantPoolHandle cpool(_method->get_methodOop()->constants());
twisti@1573 185 return CURRENT_ENV->get_klass_by_index(cpool, get_klass_index(), will_link, _holder);
duke@435 186 }
duke@435 187
duke@435 188 // ------------------------------------------------------------------
jrose@1957 189 // ciBytecodeStream::get_constant_raw_index
duke@435 190 //
duke@435 191 // If this bytecode is one of the ldc variants, get the index of the
duke@435 192 // referenced constant.
jrose@1957 193 int ciBytecodeStream::get_constant_raw_index() const {
jrose@1957 194 // work-alike for Bytecode_loadconstant::raw_index()
jrose@1957 195 switch (cur_bc()) {
duke@435 196 case Bytecodes::_ldc:
jrose@1920 197 return get_index_u1();
duke@435 198 case Bytecodes::_ldc_w:
duke@435 199 case Bytecodes::_ldc2_w:
jrose@1920 200 return get_index_u2();
duke@435 201 default:
duke@435 202 ShouldNotReachHere();
duke@435 203 return 0;
duke@435 204 }
duke@435 205 }
jrose@1957 206
jrose@1957 207 // ------------------------------------------------------------------
jrose@1957 208 // ciBytecodeStream::get_constant_pool_index
jrose@1957 209 // Decode any CP cache index into a regular pool index.
jrose@1957 210 int ciBytecodeStream::get_constant_pool_index() const {
jrose@1957 211 // work-alike for Bytecode_loadconstant::pool_index()
jrose@1957 212 int index = get_constant_raw_index();
jrose@1957 213 if (has_cache_index()) {
jrose@1957 214 return get_cpcache()->get_pool_index(index);
jrose@1957 215 }
jrose@1957 216 return index;
jrose@1957 217 }
jrose@1957 218
jrose@1957 219 // ------------------------------------------------------------------
jrose@1957 220 // ciBytecodeStream::get_constant_cache_index
jrose@1957 221 // Return the CP cache index, or -1 if there isn't any.
jrose@1957 222 int ciBytecodeStream::get_constant_cache_index() const {
jrose@1957 223 // work-alike for Bytecode_loadconstant::cache_index()
jrose@1957 224 return has_cache_index() ? get_constant_raw_index() : -1;
jrose@1957 225 }
jrose@1957 226
duke@435 227 // ------------------------------------------------------------------
duke@435 228 // ciBytecodeStream::get_constant
duke@435 229 //
duke@435 230 // If this bytecode is one of the ldc variants, get the referenced
duke@435 231 // constant.
duke@435 232 ciConstant ciBytecodeStream::get_constant() {
jrose@1957 233 int pool_index = get_constant_raw_index();
jrose@1957 234 int cache_index = -1;
jrose@1957 235 if (has_cache_index()) {
jrose@1957 236 cache_index = pool_index;
jrose@1957 237 pool_index = -1;
jrose@1957 238 }
twisti@1573 239 VM_ENTRY_MARK;
twisti@1573 240 constantPoolHandle cpool(_method->get_methodOop()->constants());
jrose@1957 241 return CURRENT_ENV->get_constant_by_index(cpool, pool_index, cache_index, _holder);
duke@435 242 }
duke@435 243
duke@435 244 // ------------------------------------------------------------------
jrose@1957 245 // ciBytecodeStream::get_constant_pool_tag
jrose@1957 246 //
jrose@1957 247 // If this bytecode is one of the ldc variants, get the referenced
jrose@1957 248 // constant.
jrose@1957 249 constantTag ciBytecodeStream::get_constant_pool_tag(int index) const {
jrose@1957 250 VM_ENTRY_MARK;
jrose@1957 251 return _method->get_methodOop()->constants()->tag_at(index);
duke@435 252 }
duke@435 253
duke@435 254 // ------------------------------------------------------------------
duke@435 255 // ciBytecodeStream::get_field_index
duke@435 256 //
duke@435 257 // If this is a field access bytecode, get the constant pool
duke@435 258 // index of the referenced field.
duke@435 259 int ciBytecodeStream::get_field_index() {
duke@435 260 assert(cur_bc() == Bytecodes::_getfield ||
duke@435 261 cur_bc() == Bytecodes::_putfield ||
duke@435 262 cur_bc() == Bytecodes::_getstatic ||
duke@435 263 cur_bc() == Bytecodes::_putstatic, "wrong bc");
jrose@1920 264 return get_index_u2_cpcache();
duke@435 265 }
duke@435 266
duke@435 267
duke@435 268 // ------------------------------------------------------------------
duke@435 269 // ciBytecodeStream::get_field
duke@435 270 //
duke@435 271 // If this bytecode is one of get_field, get_static, put_field,
duke@435 272 // or put_static, get the referenced field.
duke@435 273 ciField* ciBytecodeStream::get_field(bool& will_link) {
duke@435 274 ciField* f = CURRENT_ENV->get_field_by_index(_holder, get_field_index());
duke@435 275 will_link = f->will_link(_holder, _bc);
duke@435 276 return f;
duke@435 277 }
duke@435 278
duke@435 279
duke@435 280 // ------------------------------------------------------------------
duke@435 281 // ciBytecodeStream::get_declared_field_holder
duke@435 282 //
duke@435 283 // Get the declared holder of the currently referenced field.
duke@435 284 //
duke@435 285 // Usage note: the holder() of a ciField class returns the canonical
duke@435 286 // holder of the field, rather than the holder declared in the
duke@435 287 // bytecodes.
duke@435 288 //
duke@435 289 // There is no "will_link" result passed back. The user is responsible
duke@435 290 // for checking linkability when retrieving the associated field.
duke@435 291 ciInstanceKlass* ciBytecodeStream::get_declared_field_holder() {
twisti@1573 292 VM_ENTRY_MARK;
twisti@1573 293 constantPoolHandle cpool(_method->get_methodOop()->constants());
duke@435 294 int holder_index = get_field_holder_index();
duke@435 295 bool ignore;
twisti@1573 296 return CURRENT_ENV->get_klass_by_index(cpool, holder_index, ignore, _holder)
duke@435 297 ->as_instance_klass();
duke@435 298 }
duke@435 299
duke@435 300 // ------------------------------------------------------------------
duke@435 301 // ciBytecodeStream::get_field_holder_index
duke@435 302 //
duke@435 303 // Get the constant pool index of the declared holder of the field
duke@435 304 // referenced by the current bytecode. Used for generating
duke@435 305 // deoptimization information.
duke@435 306 int ciBytecodeStream::get_field_holder_index() {
twisti@1573 307 GUARDED_VM_ENTRY(
twisti@1573 308 constantPoolOop cpool = _holder->get_instanceKlass()->constants();
twisti@1573 309 return cpool->klass_ref_index_at(get_field_index());
twisti@1573 310 )
duke@435 311 }
duke@435 312
duke@435 313 // ------------------------------------------------------------------
duke@435 314 // ciBytecodeStream::get_field_signature_index
duke@435 315 //
duke@435 316 // Get the constant pool index of the signature of the field
duke@435 317 // referenced by the current bytecode. Used for generating
duke@435 318 // deoptimization information.
duke@435 319 int ciBytecodeStream::get_field_signature_index() {
duke@435 320 VM_ENTRY_MARK;
duke@435 321 constantPoolOop cpool = _holder->get_instanceKlass()->constants();
duke@435 322 int nt_index = cpool->name_and_type_ref_index_at(get_field_index());
duke@435 323 return cpool->signature_ref_index_at(nt_index);
duke@435 324 }
duke@435 325
duke@435 326 // ------------------------------------------------------------------
duke@435 327 // ciBytecodeStream::get_method_index
duke@435 328 //
duke@435 329 // If this is a method invocation bytecode, get the constant pool
duke@435 330 // index of the invoked method.
duke@435 331 int ciBytecodeStream::get_method_index() {
jrose@1161 332 #ifdef ASSERT
duke@435 333 switch (cur_bc()) {
duke@435 334 case Bytecodes::_invokeinterface:
duke@435 335 case Bytecodes::_invokevirtual:
duke@435 336 case Bytecodes::_invokespecial:
duke@435 337 case Bytecodes::_invokestatic:
jrose@1161 338 case Bytecodes::_invokedynamic:
jrose@1161 339 break;
duke@435 340 default:
duke@435 341 ShouldNotReachHere();
duke@435 342 }
jrose@1161 343 #endif
jrose@1920 344 if (has_index_u4())
jrose@1920 345 return get_index_u4(); // invokedynamic
jrose@1920 346 return get_index_u2_cpcache();
duke@435 347 }
duke@435 348
duke@435 349 // ------------------------------------------------------------------
duke@435 350 // ciBytecodeStream::get_method
duke@435 351 //
duke@435 352 // If this is a method invocation bytecode, get the invoked method.
duke@435 353 ciMethod* ciBytecodeStream::get_method(bool& will_link) {
twisti@1573 354 VM_ENTRY_MARK;
twisti@1573 355 constantPoolHandle cpool(_method->get_methodOop()->constants());
twisti@1573 356 ciMethod* m = CURRENT_ENV->get_method_by_index(cpool, get_method_index(), cur_bc(), _holder);
duke@435 357 will_link = m->is_loaded();
duke@435 358 return m;
duke@435 359 }
duke@435 360
duke@435 361 // ------------------------------------------------------------------
duke@435 362 // ciBytecodeStream::get_declared_method_holder
duke@435 363 //
duke@435 364 // Get the declared holder of the currently referenced method.
duke@435 365 //
duke@435 366 // Usage note: the holder() of a ciMethod class returns the canonical
duke@435 367 // holder of the method, rather than the holder declared in the
duke@435 368 // bytecodes.
duke@435 369 //
duke@435 370 // There is no "will_link" result passed back. The user is responsible
duke@435 371 // for checking linkability when retrieving the associated method.
duke@435 372 ciKlass* ciBytecodeStream::get_declared_method_holder() {
twisti@1573 373 VM_ENTRY_MARK;
twisti@1573 374 constantPoolHandle cpool(_method->get_methodOop()->constants());
duke@435 375 bool ignore;
twisti@1570 376 // report as InvokeDynamic for invokedynamic, which is syntactically classless
jrose@1161 377 if (cur_bc() == Bytecodes::_invokedynamic)
twisti@1570 378 return CURRENT_ENV->get_klass_by_name(_holder, ciSymbol::java_dyn_InvokeDynamic(), false);
twisti@1573 379 return CURRENT_ENV->get_klass_by_index(cpool, get_method_holder_index(), ignore, _holder);
duke@435 380 }
duke@435 381
duke@435 382 // ------------------------------------------------------------------
duke@435 383 // ciBytecodeStream::get_method_holder_index
duke@435 384 //
duke@435 385 // Get the constant pool index of the declared holder of the method
duke@435 386 // referenced by the current bytecode. Used for generating
duke@435 387 // deoptimization information.
duke@435 388 int ciBytecodeStream::get_method_holder_index() {
twisti@1573 389 constantPoolOop cpool = _method->get_methodOop()->constants();
duke@435 390 return cpool->klass_ref_index_at(get_method_index());
duke@435 391 }
duke@435 392
duke@435 393 // ------------------------------------------------------------------
duke@435 394 // ciBytecodeStream::get_method_signature_index
duke@435 395 //
duke@435 396 // Get the constant pool index of the signature of the method
duke@435 397 // referenced by the current bytecode. Used for generating
duke@435 398 // deoptimization information.
duke@435 399 int ciBytecodeStream::get_method_signature_index() {
duke@435 400 VM_ENTRY_MARK;
duke@435 401 constantPoolOop cpool = _holder->get_instanceKlass()->constants();
duke@435 402 int method_index = get_method_index();
duke@435 403 int name_and_type_index = cpool->name_and_type_ref_index_at(method_index);
duke@435 404 return cpool->signature_ref_index_at(name_and_type_index);
duke@435 405 }
twisti@1572 406
twisti@1572 407 // ------------------------------------------------------------------
twisti@1572 408 // ciBytecodeStream::get_cpcache
jrose@1957 409 ciCPCache* ciBytecodeStream::get_cpcache() const {
jrose@1957 410 if (_cpcache == NULL) {
jrose@1957 411 VM_ENTRY_MARK;
jrose@1957 412 // Get the constant pool.
jrose@1957 413 constantPoolOop cpool = _holder->get_instanceKlass()->constants();
jrose@1957 414 constantPoolCacheOop cpcache = cpool->cache();
twisti@1572 415
jrose@1957 416 *(ciCPCache**)&_cpcache = CURRENT_ENV->get_object(cpcache)->as_cpcache();
jrose@1957 417 }
jrose@1957 418 return _cpcache;
twisti@1572 419 }
twisti@1573 420
twisti@1573 421 // ------------------------------------------------------------------
twisti@1573 422 // ciBytecodeStream::get_call_site
twisti@1573 423 ciCallSite* ciBytecodeStream::get_call_site() {
twisti@1573 424 VM_ENTRY_MARK;
twisti@1573 425 // Get the constant pool.
twisti@1573 426 constantPoolOop cpool = _holder->get_instanceKlass()->constants();
twisti@1573 427 constantPoolCacheOop cpcache = cpool->cache();
twisti@1573 428
twisti@1573 429 // Get the CallSite from the constant pool cache.
twisti@1573 430 int method_index = get_method_index();
twisti@1573 431 ConstantPoolCacheEntry* cpcache_entry = cpcache->secondary_entry_at(method_index);
twisti@1573 432 oop call_site_oop = cpcache_entry->f1();
twisti@1573 433
twisti@1573 434 // Create a CallSite object and return it.
twisti@1573 435 return CURRENT_ENV->get_object(call_site_oop)->as_call_site();
twisti@1573 436 }

mercurial