1 /* Glue to interface gcj with bytecode verifier.
2 Copyright (C) 2003, 2004, 2005 Free Software Foundation, Inc.
4 This file is part of GCC.
6 GCC is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2, or (at your option)
11 GCC is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with GCC; see the file COPYING. If not, write to
18 the Free Software Foundation, 59 Temple Place - Suite 330,
19 Boston, MA 02111-1307, USA.
21 Java and all Java-based marks are trademarks or registered trademarks
22 of Sun Microsystems, Inc. in the United States and other countries.
23 The Free Software Foundation is independent of Sun Microsystems, Inc. */
25 /* Written by Tom Tromey <tromey@redhat.com>. */
30 #include "coretypes.h"
37 #include "java-tree.h"
38 #include "java-except.h"
41 vfy_alloc (size_t bytes
)
43 return xmalloc (bytes
);
53 vfy_strings_equal (vfy_string one
, vfy_string two
)
59 vfy_string_bytes (vfy_string str
)
61 return IDENTIFIER_POINTER (str
);
65 vfy_string_length (vfy_string str
)
67 return IDENTIFIER_LENGTH (str
);
73 return init_identifier_node
;
77 vfy_clinit_name (void)
79 return clinit_identifier_node
;
83 skip_one_type (const char* ptr
)
94 do { ch
= *ptr
++; } while (ch
!= ';');
101 vfy_count_arguments (vfy_string signature
)
103 const char *ptr
= IDENTIFIER_POINTER (signature
);
112 ptr
= skip_one_type (ptr
);
120 vfy_get_string (const char *s
, int len
)
122 return get_identifier_with_length (s
, len
);
126 vfy_get_signature (vfy_method
*method
)
128 return method
->signature
;
132 vfy_get_method_name (vfy_method
*method
)
138 vfy_is_static (vfy_method
*method
)
140 return METHOD_STATIC (method
->method
);
143 const unsigned char *
144 vfy_get_bytecode (vfy_method
*method
)
146 return method
->bytes
;
150 vfy_get_exceptions (vfy_method
*method
)
152 return method
->exceptions
;
156 vfy_get_exception (vfy_exception
*exceptions
, int index
, int *handler
,
157 int *start
, int *end
, int *handler_type
)
159 *handler
= exceptions
[index
].handler
;
160 *start
= exceptions
[index
].start
;
161 *end
= exceptions
[index
].end
;
162 *handler_type
= exceptions
[index
].type
;
166 vfy_tag (vfy_constants
*pool
, int index
)
168 int result
= JPOOL_TAG (pool
, index
);
169 /* gcj will resolve constant pool entries other than string and
170 class references. The verifier doesn't care about the values, so
171 we just strip off the resolved flag. */
172 if ((result
& CONSTANT_ResolvedFlag
) != 0
173 && result
!= CONSTANT_ResolvedString
174 && result
!= CONSTANT_ResolvedClass
)
175 result
&= ~ CONSTANT_ResolvedFlag
;
180 vfy_load_indexes (vfy_constants
*pool
, int index
,
181 vfy_uint_16
*index0
, vfy_uint_16
*index1
)
183 *index0
= JPOOL_USHORT1 (pool
, index
);
184 *index1
= JPOOL_USHORT2 (pool
, index
);
188 vfy_get_constants (vfy_jclass klass
)
190 return TYPE_JCF (klass
);
194 vfy_get_constants_size (vfy_jclass klass
)
196 return JPOOL_SIZE (TYPE_JCF (klass
));
200 vfy_get_pool_string (vfy_constants
*pool
, int index
)
202 return get_name_constant (pool
, index
);
206 vfy_get_pool_class (vfy_constants
*pool
, int index
)
209 k
= get_class_constant (pool
, index
);
214 vfy_make_string (const char *s
, int len
)
217 char *s2
= (char *) s
;
220 result
= get_identifier (s2
);
226 vfy_get_class_name (vfy_jclass klass
)
228 return DECL_NAME (TYPE_NAME (klass
));
232 vfy_is_assignable_from (vfy_jclass target
, vfy_jclass source
)
234 /* At compile time, for the BC-ABI we assume that reference types are always
235 compatible. However, a type assertion table entry is emitted so that the
236 runtime can detect binary-incompatible changes. */
238 /* FIXME: implement real test for old ABI. */
240 /* Any class is always assignable to itself, or java.lang.Object. */
241 if (source
== target
|| target
== object_type_node
)
244 /* Otherwise, a type assertion is required. */
245 add_type_assertion (current_class
, JV_ASSERT_TYPES_COMPATIBLE
, source
,
251 vfy_get_primitive_char (vfy_jclass klass
)
254 if (! vfy_is_primitive (klass
))
256 sig
= build_java_signature (klass
);
257 return (IDENTIFIER_POINTER (sig
))[0];
261 vfy_get_interface_count (vfy_jclass klass ATTRIBUTE_UNUSED
)
263 /* FIXME: Need to merge from mainline to get this. */
265 return BINFO_N_BASE_BINFOS (klass
);
271 vfy_get_interface (vfy_jclass klass ATTRIBUTE_UNUSED
, int index ATTRIBUTE_UNUSED
)
273 /* FIXME: Need to merge from mainline to get this. */
276 k
= BINFO_BASE_BINFO (klass
, index
);
283 vfy_is_array (vfy_jclass klass
)
285 return TYPE_ARRAY_P (klass
);
289 vfy_is_interface (vfy_jclass klass
)
291 return CLASS_INTERFACE (TYPE_NAME (klass
));
295 vfy_is_primitive (vfy_jclass klass
)
297 return JPRIMITIVE_TYPE_P (klass
);
301 vfy_get_superclass (vfy_jclass klass
)
304 k
= CLASSTYPE_SUPER (klass
);
309 vfy_get_array_class (vfy_jclass klass
)
312 k
= build_java_array_type (klass
, -1);
317 vfy_get_component_type (vfy_jclass klass
)
320 if (! vfy_is_array (klass
))
322 k
= TYPE_ARRAY_ELEMENT (klass
);
323 if (TREE_CODE (k
) == POINTER_TYPE
)
329 vfy_is_abstract (vfy_jclass klass
)
331 return CLASS_ABSTRACT (TYPE_NAME (klass
));
335 vfy_find_class (vfy_jclass ignore ATTRIBUTE_UNUSED
, vfy_string name
)
339 k
= get_type_from_signature (name
);
340 if (TREE_CODE (k
) == POINTER_TYPE
)
347 vfy_object_type (void)
350 k
= object_type_node
;
355 vfy_string_type (void)
358 k
= string_type_node
;
363 vfy_throwable_type (void)
366 k
= throwable_type_node
;
371 vfy_unsuitable_type (void)
377 vfy_return_address_type (void)
379 return TYPE_RETURN_ADDR
;
389 vfy_class_has_field (vfy_jclass klass
, vfy_string name
,
390 vfy_string signature
)
392 tree field
= TYPE_FIELDS (klass
);
393 while (field
!= NULL_TREE
)
395 if (DECL_NAME (field
) == name
396 && build_java_signature (TREE_TYPE (field
)) == signature
)
398 field
= TREE_CHAIN (field
);
404 vfy_fail (const char *message
, int pc
, vfy_jclass ignore1 ATTRIBUTE_UNUSED
,
405 vfy_method
*ignore2 ATTRIBUTE_UNUSED
)
408 error ("verification failed: %s", message
);
410 error ("verification failed at PC=%d: %s", pc
, message
);
411 /* We have to return a value for the verifier to throw. */
416 vfy_get_primitive_type (int type
)
419 k
= decode_newarray_type (type
);
424 vfy_note_stack_depth (vfy_method
*method
, int pc
, int depth
)
426 tree label
= lookup_label (pc
);
427 LABEL_TYPE_STATE (label
) = make_tree_vec (method
->max_locals
+ depth
);
431 vfy_note_stack_type (vfy_method
*method
, int pc
, int slot
, vfy_jclass type
)
435 slot
+= method
->max_locals
;
437 if (type
== object_type_node
)
438 type
= object_ptr_type_node
;
440 label
= lookup_label (pc
);
441 vec
= LABEL_TYPE_STATE (label
);
442 TREE_VEC_ELT (vec
, slot
) = type
;
446 vfy_note_local_type (vfy_method
*method ATTRIBUTE_UNUSED
, int pc
, int slot
,
451 if (type
== object_type_node
)
452 type
= object_ptr_type_node
;
454 label
= lookup_label (pc
);
455 vec
= LABEL_TYPE_STATE (label
);
456 TREE_VEC_ELT (vec
, slot
) = type
;
460 vfy_note_instruction_seen (int pc
)
462 instruction_bits
[pc
] |= BCODE_VERIFIED
;
465 /* Verify the bytecodes of the current method.
466 Return 1 on success, 0 on failure. */
468 verify_jvm_instructions_new (JCF
*jcf
, const unsigned char *byte_ops
,
472 int i
, result
, eh_count
;
473 vfy_exception
*exceptions
;
475 method_init_exceptions ();
477 JCF_SEEK (jcf
, DECL_CODE_OFFSET (current_function_decl
) + length
);
478 eh_count
= JCF_readu2 (jcf
);
480 exceptions
= (vfy_exception
*) xmalloc (eh_count
* sizeof (vfy_exception
));
481 for (i
= 0; i
< eh_count
; ++i
)
483 int start_pc
, end_pc
, handler_pc
, catch_type
;
484 unsigned char *p
= jcf
->read_ptr
+ 8 * i
;
485 start_pc
= GET_u2 (p
);
486 end_pc
= GET_u2 (p
+2);
487 handler_pc
= GET_u2 (p
+4);
488 catch_type
= GET_u2 (p
+6);
490 if (start_pc
< 0 || start_pc
>= length
491 || end_pc
< 0 || end_pc
> length
|| start_pc
>= end_pc
492 || handler_pc
< 0 || handler_pc
>= length
)
494 error ("bad pc in exception_table");
499 exceptions
[i
].handler
= handler_pc
;
500 exceptions
[i
].start
= start_pc
;
501 exceptions
[i
].end
= end_pc
;
502 exceptions
[i
].type
= catch_type
;
504 add_handler (start_pc
, end_pc
,
505 lookup_label (handler_pc
),
506 catch_type
== 0 ? NULL_TREE
507 : get_class_constant (jcf
, catch_type
));
508 instruction_bits
[handler_pc
] |= BCODE_EXCEPTION_TARGET
;
511 handle_nested_ranges ();
513 method
.method
= current_function_decl
;
514 method
.signature
= build_java_signature (TREE_TYPE (current_function_decl
));
515 method
.name
= DECL_NAME (current_function_decl
);
516 method
.bytes
= byte_ops
;
517 method
.exceptions
= exceptions
;
518 method
.defining_class
= DECL_CONTEXT (current_function_decl
);
519 method
.max_stack
= DECL_MAX_STACK (current_function_decl
);
520 method
.max_locals
= DECL_MAX_LOCALS (current_function_decl
);
521 method
.code_length
= length
;
522 method
.exc_count
= eh_count
;
524 result
= verify_method (&method
);