src/share/vm/interpreter/bytecode.hpp

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) 1997, 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 // Base class for different kinds of abstractions working
duke@435 26 // relative to an objects 'this' pointer.
duke@435 27
duke@435 28 class ThisRelativeObj VALUE_OBJ_CLASS_SPEC {
duke@435 29 public:
duke@435 30 // Address computation
duke@435 31 address addr_at (int offset) const { return (address)this + offset; }
jrose@1920 32 int byte_at (int offset) const { return *(addr_at(offset)); }
duke@435 33 address aligned_addr_at (int offset) const { return (address)round_to((intptr_t)addr_at(offset), jintSize); }
duke@435 34 int aligned_offset (int offset) const { return aligned_addr_at(offset) - addr_at(0); }
duke@435 35
jrose@1920 36 // Word access:
jrose@1920 37 int get_Java_u2_at (int offset) const { return Bytes::get_Java_u2(addr_at(offset)); }
jrose@1920 38 int get_Java_u4_at (int offset) const { return Bytes::get_Java_u4(addr_at(offset)); }
jrose@1920 39 int get_native_u2_at (int offset) const { return Bytes::get_native_u2(addr_at(offset)); }
jrose@1920 40 int get_native_u4_at (int offset) const { return Bytes::get_native_u4(addr_at(offset)); }
duke@435 41 };
duke@435 42
duke@435 43
duke@435 44 // The base class for different kinds of bytecode abstractions.
duke@435 45 // Provides the primitive operations to manipulate code relative
duke@435 46 // to an objects 'this' pointer.
jrose@1920 47 // FIXME: Make this a ResourceObj, include the enclosing methodOop, and cache the opcode.
duke@435 48
duke@435 49 class Bytecode: public ThisRelativeObj {
duke@435 50 protected:
duke@435 51 u_char byte_at(int offset) const { return *addr_at(offset); }
jrose@1920 52 bool check_must_rewrite(Bytecodes::Code bc) const;
duke@435 53
duke@435 54 public:
duke@435 55 // Attributes
duke@435 56 address bcp() const { return addr_at(0); }
jrose@1161 57 int instruction_size() const { return Bytecodes::length_at(bcp()); }
duke@435 58
jrose@1920 59 // Warning: Use code() with caution on live bytecode streams. 4926272
duke@435 60 Bytecodes::Code code() const { return Bytecodes::code_at(addr_at(0)); }
duke@435 61 Bytecodes::Code java_code() const { return Bytecodes::java_code(code()); }
jrose@1920 62 bool must_rewrite(Bytecodes::Code code) const { return Bytecodes::can_rewrite(code) && check_must_rewrite(code); }
duke@435 63
duke@435 64 // Creation
duke@435 65 inline friend Bytecode* Bytecode_at(address bcp);
jrose@1161 66
jrose@1920 67 // Static functions for parsing bytecodes in place.
jrose@1920 68 int get_index_u1(Bytecodes::Code bc) const {
jrose@1920 69 assert_same_format_as(bc); assert_index_size(1, bc);
jrose@1920 70 return *(jubyte*)addr_at(1);
jrose@1920 71 }
jrose@1920 72 int get_index_u2(Bytecodes::Code bc, bool is_wide = false) const {
jrose@1920 73 assert_same_format_as(bc, is_wide); assert_index_size(2, bc, is_wide);
jrose@1920 74 address p = addr_at(is_wide ? 2 : 1);
jrose@1920 75 if (can_use_native_byte_order(bc, is_wide))
jrose@1920 76 return Bytes::get_native_u2(p);
jrose@1920 77 else return Bytes::get_Java_u2(p);
jrose@1920 78 }
jrose@1957 79 int get_index_u1_cpcache(Bytecodes::Code bc) const {
jrose@1957 80 assert_same_format_as(bc); assert_index_size(1, bc);
jrose@1957 81 return *(jubyte*)addr_at(1) + constantPoolOopDesc::CPCACHE_INDEX_TAG;
jrose@1957 82 }
jrose@1920 83 int get_index_u2_cpcache(Bytecodes::Code bc) const {
jrose@1920 84 assert_same_format_as(bc); assert_index_size(2, bc); assert_native_index(bc);
jrose@1957 85 return Bytes::get_native_u2(addr_at(1)) + constantPoolOopDesc::CPCACHE_INDEX_TAG;
jrose@1920 86 }
jrose@1920 87 int get_index_u4(Bytecodes::Code bc) const {
jrose@1920 88 assert_same_format_as(bc); assert_index_size(4, bc);
jrose@1920 89 assert(can_use_native_byte_order(bc), "");
jrose@1920 90 return Bytes::get_native_u4(addr_at(1));
jrose@1920 91 }
jrose@1920 92 bool has_index_u4(Bytecodes::Code bc) const {
jrose@1920 93 return bc == Bytecodes::_invokedynamic;
jrose@1920 94 }
jrose@1920 95
jrose@1920 96 int get_offset_s2(Bytecodes::Code bc) const {
jrose@1920 97 assert_same_format_as(bc); assert_offset_size(2, bc);
jrose@1920 98 return (jshort) Bytes::get_Java_u2(addr_at(1));
jrose@1920 99 }
jrose@1920 100 int get_offset_s4(Bytecodes::Code bc) const {
jrose@1920 101 assert_same_format_as(bc); assert_offset_size(4, bc);
jrose@1920 102 return (jint) Bytes::get_Java_u4(addr_at(1));
jrose@1920 103 }
jrose@1920 104
jrose@1920 105 int get_constant_u1(int offset, Bytecodes::Code bc) const {
jrose@1920 106 assert_same_format_as(bc); assert_constant_size(1, offset, bc);
jrose@1920 107 return *(jbyte*)addr_at(offset);
jrose@1920 108 }
jrose@1920 109 int get_constant_u2(int offset, Bytecodes::Code bc, bool is_wide = false) const {
jrose@1920 110 assert_same_format_as(bc, is_wide); assert_constant_size(2, offset, bc, is_wide);
jrose@1920 111 return (jshort) Bytes::get_Java_u2(addr_at(offset));
jrose@1920 112 }
jrose@1920 113
jrose@1920 114 // These are used locally and also from bytecode streams.
jrose@1920 115 void assert_same_format_as(Bytecodes::Code testbc, bool is_wide = false) const NOT_DEBUG_RETURN;
jrose@1920 116 static void assert_index_size(int required_size, Bytecodes::Code bc, bool is_wide = false) NOT_DEBUG_RETURN;
jrose@1920 117 static void assert_offset_size(int required_size, Bytecodes::Code bc, bool is_wide = false) NOT_DEBUG_RETURN;
jrose@1920 118 static void assert_constant_size(int required_size, int where, Bytecodes::Code bc, bool is_wide = false) NOT_DEBUG_RETURN;
jrose@1920 119 static void assert_native_index(Bytecodes::Code bc, bool is_wide = false) NOT_DEBUG_RETURN;
jrose@1920 120 static bool can_use_native_byte_order(Bytecodes::Code bc, bool is_wide = false) {
jrose@1920 121 return (!Bytes::is_Java_byte_ordering_different() || Bytecodes::native_byte_order(bc /*, is_wide*/));
jrose@1161 122 }
duke@435 123 };
duke@435 124
duke@435 125 inline Bytecode* Bytecode_at(address bcp) {
jrose@1920 126 // Warning: Use with caution on live bytecode streams. 4926272
duke@435 127 return (Bytecode*)bcp;
duke@435 128 }
duke@435 129
duke@435 130
duke@435 131 // Abstractions for lookupswitch bytecode
duke@435 132
duke@435 133 class LookupswitchPair: ThisRelativeObj {
duke@435 134 private:
duke@435 135 int _match;
duke@435 136 int _offset;
duke@435 137
duke@435 138 public:
jrose@1920 139 int match() const { return get_Java_u4_at(0 * jintSize); }
jrose@1920 140 int offset() const { return get_Java_u4_at(1 * jintSize); }
duke@435 141 };
duke@435 142
duke@435 143
duke@435 144 class Bytecode_lookupswitch: public Bytecode {
duke@435 145 public:
duke@435 146 void verify() const PRODUCT_RETURN;
duke@435 147
duke@435 148 // Attributes
jrose@1920 149 int default_offset() const { return get_Java_u4_at(aligned_offset(1 + 0*jintSize)); }
jrose@1920 150 int number_of_pairs() const { return get_Java_u4_at(aligned_offset(1 + 1*jintSize)); }
duke@435 151 LookupswitchPair* pair_at(int i) const { assert(0 <= i && i < number_of_pairs(), "pair index out of bounds");
duke@435 152 return (LookupswitchPair*)aligned_addr_at(1 + (1 + i)*2*jintSize); }
duke@435 153 // Creation
duke@435 154 inline friend Bytecode_lookupswitch* Bytecode_lookupswitch_at(address bcp);
duke@435 155 };
duke@435 156
duke@435 157 inline Bytecode_lookupswitch* Bytecode_lookupswitch_at(address bcp) {
duke@435 158 Bytecode_lookupswitch* b = (Bytecode_lookupswitch*)bcp;
jrose@1957 159 DEBUG_ONLY(b->verify());
duke@435 160 return b;
duke@435 161 }
duke@435 162
duke@435 163
duke@435 164 class Bytecode_tableswitch: public Bytecode {
duke@435 165 public:
duke@435 166 void verify() const PRODUCT_RETURN;
duke@435 167
duke@435 168 // Attributes
jrose@1920 169 int default_offset() const { return get_Java_u4_at(aligned_offset(1 + 0*jintSize)); }
jrose@1920 170 int low_key() const { return get_Java_u4_at(aligned_offset(1 + 1*jintSize)); }
jrose@1920 171 int high_key() const { return get_Java_u4_at(aligned_offset(1 + 2*jintSize)); }
duke@435 172 int dest_offset_at(int i) const;
duke@435 173 int length() { return high_key()-low_key()+1; }
duke@435 174
duke@435 175 // Creation
duke@435 176 inline friend Bytecode_tableswitch* Bytecode_tableswitch_at(address bcp);
duke@435 177 };
duke@435 178
duke@435 179 inline Bytecode_tableswitch* Bytecode_tableswitch_at(address bcp) {
duke@435 180 Bytecode_tableswitch* b = (Bytecode_tableswitch*)bcp;
jrose@1957 181 DEBUG_ONLY(b->verify());
duke@435 182 return b;
duke@435 183 }
duke@435 184
duke@435 185
jrose@1957 186 // Common code for decoding invokes and field references.
duke@435 187
jrose@1957 188 class Bytecode_member_ref: public ResourceObj {
duke@435 189 protected:
duke@435 190 methodHandle _method; // method containing the bytecode
duke@435 191 int _bci; // position of the bytecode
duke@435 192
jrose@1957 193 Bytecode_member_ref(methodHandle method, int bci) : _method(method), _bci(bci) {}
jrose@1957 194
jrose@1957 195 public:
jrose@1957 196 // Attributes
jrose@1957 197 methodHandle method() const { return _method; }
jrose@1957 198 int bci() const { return _bci; }
jrose@1957 199 address bcp() const { return _method->bcp_from(bci()); }
jrose@1957 200 Bytecode* bytecode() const { return Bytecode_at(bcp()); }
jrose@1957 201
jrose@1957 202 int index() const; // cache index (loaded from instruction)
jrose@1957 203 int pool_index() const; // constant pool index
jrose@1957 204 symbolOop name() const; // returns the name of the method or field
jrose@1957 205 symbolOop signature() const; // returns the signature of the method or field
jrose@1957 206
jrose@1957 207 BasicType result_type(Thread* thread) const; // returns the result type of the getfield or invoke
jrose@1957 208
jrose@1957 209 Bytecodes::Code code() const { return Bytecodes::code_at(bcp(), _method()); }
jrose@1957 210 Bytecodes::Code java_code() const { return Bytecodes::java_code(code()); }
jrose@1957 211 };
jrose@1957 212
jrose@1957 213 // Abstraction for invoke_{virtual, static, interface, special}
jrose@1957 214
jrose@1957 215 class Bytecode_invoke: public Bytecode_member_ref {
jrose@1957 216 protected:
jrose@1957 217 Bytecode_invoke(methodHandle method, int bci) : Bytecode_member_ref(method, bci) {}
duke@435 218
duke@435 219 public:
duke@435 220 void verify() const;
duke@435 221
duke@435 222 // Attributes
duke@435 223 methodHandle static_target(TRAPS); // "specified" method (from constant pool)
duke@435 224
duke@435 225 // Testers
jrose@1957 226 bool is_invokeinterface() const { return java_code() == Bytecodes::_invokeinterface; }
jrose@1957 227 bool is_invokevirtual() const { return java_code() == Bytecodes::_invokevirtual; }
jrose@1957 228 bool is_invokestatic() const { return java_code() == Bytecodes::_invokestatic; }
jrose@1957 229 bool is_invokespecial() const { return java_code() == Bytecodes::_invokespecial; }
jrose@1957 230 bool is_invokedynamic() const { return java_code() == Bytecodes::_invokedynamic; }
jrose@1161 231
twisti@1573 232 bool has_receiver() const { return !is_invokestatic() && !is_invokedynamic(); }
duke@435 233
duke@435 234 bool is_valid() const { return is_invokeinterface() ||
duke@435 235 is_invokevirtual() ||
duke@435 236 is_invokestatic() ||
twisti@1570 237 is_invokespecial() ||
twisti@1570 238 is_invokedynamic(); }
duke@435 239
duke@435 240 // Creation
duke@435 241 inline friend Bytecode_invoke* Bytecode_invoke_at(methodHandle method, int bci);
duke@435 242
duke@435 243 // Like Bytecode_invoke_at. Instead it returns NULL if the bci is not at an invoke.
duke@435 244 inline friend Bytecode_invoke* Bytecode_invoke_at_check(methodHandle method, int bci);
duke@435 245 };
duke@435 246
duke@435 247 inline Bytecode_invoke* Bytecode_invoke_at(methodHandle method, int bci) {
duke@435 248 Bytecode_invoke* b = new Bytecode_invoke(method, bci);
jrose@1957 249 DEBUG_ONLY(b->verify());
duke@435 250 return b;
duke@435 251 }
duke@435 252
duke@435 253 inline Bytecode_invoke* Bytecode_invoke_at_check(methodHandle method, int bci) {
duke@435 254 Bytecode_invoke* b = new Bytecode_invoke(method, bci);
duke@435 255 return b->is_valid() ? b : NULL;
duke@435 256 }
duke@435 257
duke@435 258
jrose@1957 259 // Abstraction for all field accesses (put/get field/static)
jrose@1957 260 class Bytecode_field: public Bytecode_member_ref {
jrose@1957 261 protected:
jrose@1957 262 Bytecode_field(methodHandle method, int bci) : Bytecode_member_ref(method, bci) {}
jrose@1957 263
jrose@1957 264 public:
jrose@1957 265 // Testers
jrose@1957 266 bool is_getfield() const { return java_code() == Bytecodes::_getfield; }
jrose@1957 267 bool is_putfield() const { return java_code() == Bytecodes::_putfield; }
jrose@1957 268 bool is_getstatic() const { return java_code() == Bytecodes::_getstatic; }
jrose@1957 269 bool is_putstatic() const { return java_code() == Bytecodes::_putstatic; }
jrose@1957 270
jrose@1957 271 bool is_getter() const { return is_getfield() || is_getstatic(); }
jrose@1957 272 bool is_static() const { return is_getstatic() || is_putstatic(); }
jrose@1957 273
jrose@1957 274 bool is_valid() const { return is_getfield() ||
jrose@1957 275 is_putfield() ||
jrose@1957 276 is_getstatic() ||
jrose@1957 277 is_putstatic(); }
duke@435 278 void verify() const;
duke@435 279
duke@435 280 // Creation
jrose@1957 281 inline friend Bytecode_field* Bytecode_field_at(methodHandle method, int bci);
duke@435 282 };
duke@435 283
jrose@1957 284 inline Bytecode_field* Bytecode_field_at(methodHandle method, int bci) {
jrose@1957 285 Bytecode_field* b = new Bytecode_field(method, bci);
jrose@1957 286 DEBUG_ONLY(b->verify());
duke@435 287 return b;
duke@435 288 }
duke@435 289
duke@435 290
duke@435 291 // Abstraction for checkcast
duke@435 292
duke@435 293 class Bytecode_checkcast: public Bytecode {
duke@435 294 public:
duke@435 295 void verify() const { assert(Bytecodes::java_code(code()) == Bytecodes::_checkcast, "check checkcast"); }
duke@435 296
duke@435 297 // Returns index
jrose@1920 298 long index() const { return get_index_u2(Bytecodes::_checkcast); };
duke@435 299
duke@435 300 // Creation
duke@435 301 inline friend Bytecode_checkcast* Bytecode_checkcast_at(address bcp);
duke@435 302 };
duke@435 303
duke@435 304 inline Bytecode_checkcast* Bytecode_checkcast_at(address bcp) {
duke@435 305 Bytecode_checkcast* b = (Bytecode_checkcast*)bcp;
jrose@1957 306 DEBUG_ONLY(b->verify());
duke@435 307 return b;
duke@435 308 }
duke@435 309
duke@435 310
duke@435 311 // Abstraction for instanceof
duke@435 312
duke@435 313 class Bytecode_instanceof: public Bytecode {
duke@435 314 public:
duke@435 315 void verify() const { assert(code() == Bytecodes::_instanceof, "check instanceof"); }
duke@435 316
duke@435 317 // Returns index
jrose@1920 318 long index() const { return get_index_u2(Bytecodes::_instanceof); };
duke@435 319
duke@435 320 // Creation
duke@435 321 inline friend Bytecode_instanceof* Bytecode_instanceof_at(address bcp);
duke@435 322 };
duke@435 323
duke@435 324 inline Bytecode_instanceof* Bytecode_instanceof_at(address bcp) {
duke@435 325 Bytecode_instanceof* b = (Bytecode_instanceof*)bcp;
jrose@1957 326 DEBUG_ONLY(b->verify());
duke@435 327 return b;
duke@435 328 }
duke@435 329
duke@435 330
duke@435 331 class Bytecode_new: public Bytecode {
duke@435 332 public:
duke@435 333 void verify() const { assert(java_code() == Bytecodes::_new, "check new"); }
duke@435 334
duke@435 335 // Returns index
jrose@1920 336 long index() const { return get_index_u2(Bytecodes::_new); };
duke@435 337
duke@435 338 // Creation
duke@435 339 inline friend Bytecode_new* Bytecode_new_at(address bcp);
duke@435 340 };
duke@435 341
duke@435 342 inline Bytecode_new* Bytecode_new_at(address bcp) {
duke@435 343 Bytecode_new* b = (Bytecode_new*)bcp;
jrose@1957 344 DEBUG_ONLY(b->verify());
duke@435 345 return b;
duke@435 346 }
duke@435 347
duke@435 348
duke@435 349 class Bytecode_multianewarray: public Bytecode {
duke@435 350 public:
duke@435 351 void verify() const { assert(java_code() == Bytecodes::_multianewarray, "check new"); }
duke@435 352
duke@435 353 // Returns index
jrose@1920 354 long index() const { return get_index_u2(Bytecodes::_multianewarray); };
duke@435 355
duke@435 356 // Creation
duke@435 357 inline friend Bytecode_multianewarray* Bytecode_multianewarray_at(address bcp);
duke@435 358 };
duke@435 359
duke@435 360 inline Bytecode_multianewarray* Bytecode_multianewarray_at(address bcp) {
duke@435 361 Bytecode_multianewarray* b = (Bytecode_multianewarray*)bcp;
jrose@1957 362 DEBUG_ONLY(b->verify());
duke@435 363 return b;
duke@435 364 }
duke@435 365
duke@435 366
duke@435 367 class Bytecode_anewarray: public Bytecode {
duke@435 368 public:
duke@435 369 void verify() const { assert(java_code() == Bytecodes::_anewarray, "check anewarray"); }
duke@435 370
duke@435 371 // Returns index
jrose@1920 372 long index() const { return get_index_u2(Bytecodes::_anewarray); };
duke@435 373
duke@435 374 // Creation
duke@435 375 inline friend Bytecode_anewarray* Bytecode_anewarray_at(address bcp);
duke@435 376 };
duke@435 377
duke@435 378 inline Bytecode_anewarray* Bytecode_anewarray_at(address bcp) {
duke@435 379 Bytecode_anewarray* b = (Bytecode_anewarray*)bcp;
jrose@1957 380 DEBUG_ONLY(b->verify());
duke@435 381 return b;
duke@435 382 }
duke@435 383
duke@435 384
duke@435 385 // Abstraction for ldc, ldc_w and ldc2_w
duke@435 386
jrose@1957 387 class Bytecode_loadconstant: public ResourceObj {
jrose@1957 388 private:
jrose@1957 389 int _bci;
jrose@1957 390 methodHandle _method;
jrose@1957 391
jrose@1957 392 Bytecodes::Code code() const { return bytecode()->code(); }
jrose@1957 393
jrose@1957 394 int raw_index() const;
jrose@1957 395
jrose@1957 396 Bytecode_loadconstant(methodHandle method, int bci) : _method(method), _bci(bci) {}
jrose@1957 397
duke@435 398 public:
jrose@1957 399 // Attributes
jrose@1957 400 methodHandle method() const { return _method; }
jrose@1957 401 int bci() const { return _bci; }
jrose@1957 402 address bcp() const { return _method->bcp_from(bci()); }
jrose@1957 403 Bytecode* bytecode() const { return Bytecode_at(bcp()); }
jrose@1957 404
duke@435 405 void verify() const {
jrose@1957 406 assert(_method.not_null(), "must supply method");
duke@435 407 Bytecodes::Code stdc = Bytecodes::java_code(code());
duke@435 408 assert(stdc == Bytecodes::_ldc ||
duke@435 409 stdc == Bytecodes::_ldc_w ||
duke@435 410 stdc == Bytecodes::_ldc2_w, "load constant");
duke@435 411 }
duke@435 412
jrose@1957 413 // Only non-standard bytecodes (fast_aldc) have CP cache indexes.
jrose@1957 414 bool has_cache_index() const { return code() >= Bytecodes::number_of_java_codes; }
duke@435 415
jrose@1957 416 int pool_index() const; // index into constant pool
jrose@1957 417 int cache_index() const { // index into CP cache (or -1 if none)
jrose@1957 418 return has_cache_index() ? raw_index() : -1;
jrose@1957 419 }
jrose@1957 420
jrose@1957 421 BasicType result_type() const; // returns the result type of the ldc
jrose@1957 422
jrose@1957 423 oop resolve_constant(TRAPS) const;
jrose@1957 424
jrose@1957 425 // Creation
jrose@1957 426 inline friend Bytecode_loadconstant* Bytecode_loadconstant_at(methodHandle method, int bci);
duke@435 427 };
duke@435 428
jrose@1957 429 inline Bytecode_loadconstant* Bytecode_loadconstant_at(methodHandle method, int bci) {
jrose@1957 430 Bytecode_loadconstant* b = new Bytecode_loadconstant(method, bci);
jrose@1957 431 DEBUG_ONLY(b->verify());
duke@435 432 return b;
duke@435 433 }

mercurial