/* Apache 2.0 - Copyright 2007-2018 - pancake and dso class.c rewrite: Adam Pridgen */ #include #include #include #include #include #include #include #include #include #include "class.h" #include "dsojson.h" static PrintfCallback Eprintf =(PrintfCallback) printf; #ifdef IFDBG #undef IFDBG #endif #define DO_THE_DBG 0 #define IFDBG if (DO_THE_DBG) #define IFINT if (0) #define MAX_CPITEMS 8192 R_API char *U(r_bin_java_unmangle_method)(const char *flags, const char *name, const char *params, const char *r_value); R_API int r_bin_java_is_fm_type_private(RBinJavaField *fm_type); R_API int r_bin_java_is_fm_type_protected(RBinJavaField *fm_type); R_API ut32 U(r_bin_java_swap_uint)(ut32 x); // R_API const char * r_bin_java_get_this_class_name(RBinJavaObj *bin); R_API void U(add_cp_objs_to_sdb)(RBinJavaObj * bin); R_API void U(add_field_infos_to_sdb)(RBinJavaObj * bin); R_API void U(add_method_infos_to_sdb)(RBinJavaObj * bin); R_API RList *retrieve_all_access_string_and_value(RBinJavaAccessFlags *access_flags); R_API char *retrieve_access_string(ut16 flags, RBinJavaAccessFlags *access_flags); R_API ut16 calculate_access_value(const char *access_flags_str, RBinJavaAccessFlags *access_flags); R_API int r_bin_java_new_bin(RBinJavaObj *bin, ut64 loadaddr, Sdb *kv, const ut8 *buf, ut64 len); R_API int extract_type_value(const char *arg_str, char **output); R_API int r_bin_java_check_reset_cp_obj(RBinJavaCPTypeObj *cp_obj, ut8 tag); R_API ut8 *r_bin_java_cp_get_4bytes(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len); R_API ut8 *r_bin_java_cp_get_8bytes(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len); R_API ut8 *r_bin_java_cp_get_utf8(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len); R_API RBinJavaCPTypeObj *r_bin_java_get_item_from_bin_cp_list(RBinJavaObj *bin, ut64 idx); R_API RBinJavaCPTypeObj *r_bin_java_get_item_from_cp_item_list(RList *cp_list, ut64 idx); // Allocs for objects R_API RBinJavaCPTypeObj *r_bin_java_class_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_fieldref_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_methodref_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_interfacemethodref_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_name_and_type_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_string_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_integer_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_float_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_long_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_double_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_utf8_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 offset); R_API RBinJavaCPTypeObj *r_bin_java_do_nothing_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz); R_API RBinJavaCPTypeObj *r_bin_java_clone_cp_item(RBinJavaCPTypeObj *obj); R_API RBinJavaCPTypeObj *r_bin_java_clone_cp_idx(RBinJavaObj *bin, ut32 idx); R_API RBinJavaCPTypeObj *r_bin_java_methodhandle_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz); R_API RBinJavaCPTypeObj *r_bin_java_methodtype_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz); R_API RBinJavaCPTypeObj *r_bin_java_invokedynamic_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz); // Deallocs for type objects R_API void r_bin_java_default_free(void /*RBinJavaCPTypeObj*/ *obj); R_API void r_bin_java_obj_free(void /*RBinJavaCPTypeObj*/ *obj); R_API void r_bin_java_utf8_info_free(void /*RBinJavaCPTypeObj*/ *obj); R_API void r_bin_java_do_nothing_free(void /*RBinJavaCPTypeObj*/ *obj); R_API void r_bin_java_fmtype_free(void /*RBinJavaField*/ *fm_type); // handle freeing the lists // handle the reading of the various field R_API RBinJavaAttrInfo *r_bin_java_read_next_attr(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len); R_API RBinJavaCPTypeObj *r_bin_java_read_next_constant_pool_item(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, ut64 len); R_API RBinJavaAttrMetas *r_bin_java_get_attr_type_by_name(const char *name); R_API RBinJavaCPTypeObj *r_bin_java_get_java_null_cp(void); R_API ut64 r_bin_java_read_class_file2(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, ut64 len); R_API RBinJavaAttrInfo *r_bin_java_get_attr_from_field(RBinJavaField *field, R_BIN_JAVA_ATTR_TYPE attr_type, ut32 pos); R_API RBinJavaField *r_bin_java_read_next_field(RBinJavaObj *bin, const ut64 offset, const ut8 *buffer, const ut64 len); R_API RBinJavaField *r_bin_java_read_next_method(RBinJavaObj *bin, const ut64 offset, const ut8 *buffer, const ut64 len); R_API void r_bin_java_print_utf8_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_name_and_type_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_double_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_long_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_float_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_integer_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_string_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_classref_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_fieldref_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_methodref_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_interfacemethodref_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_unknown_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_null_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_unknown_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_methodhandle_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_methodtype_cp_summary(RBinJavaCPTypeObj *obj); R_API void r_bin_java_print_invokedynamic_cp_summary(RBinJavaCPTypeObj *obj); R_API RBinJavaCPTypeObj *r_bin_java_unknown_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz); R_API RBinJavaInterfaceInfo *r_bin_java_interface_new(RBinJavaObj *bin, const ut8 *buf, ut64 sz); R_API RBinJavaInterfaceInfo *r_bin_java_read_next_interface_item(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, ut64 len); R_API void r_bin_java_interface_free(void /*RBinJavaInterfaceInfo*/ *obj); R_API void r_bin_java_stack_frame_free(void /*RBinJavaStackMapFrame*/ *obj); R_API void r_bin_java_stack_map_table_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_verification_info_free(void /*RBinJavaVerificationObj*/ *obj); R_API void r_bin_java_print_stack_map_table_attr_summary(RBinJavaAttrInfo *obj); R_API void r_bin_java_print_stack_map_frame_summary(RBinJavaStackMapFrame *obj); R_API void r_bin_java_print_verification_info_summary(RBinJavaVerificationObj *obj); R_API RBinJavaStackMapFrame *r_bin_java_build_stack_frame_from_local_variable_table(RBinJavaObj *bin, RBinJavaAttrInfo *attr); R_API void U(r_bin_java_print_stack_map_append_frame_summary)(RBinJavaStackMapFrame * obj); R_API void U(r_bin_java_stack_frame_default_free)(void /*RBinJavaStackMapFrame*/ *stack_frame); R_API void U(r_bin_java_stack_frame_do_nothing_free)(void /*RBinJavaStackMapFrame*/ *stack_frame); R_API void U(r_bin_java_stack_frame_do_nothing_new)(RBinJavaObj * bin, RBinJavaStackMapFrame * stack_frame, ut64 offset); R_API RBinJavaStackMapFrame *r_bin_java_stack_map_frame_new(ut8 *buffer, ut64 sz, RBinJavaStackMapFrame *p_frame, ut64 buf_offset); // R_API RBinJavaStackMapFrame* r_bin_java_stack_map_frame_new (ut8* buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaElementValue *r_bin_java_element_value_new(ut8 *buffer, ut64 sz, ut64 buf_offset); // R_API RBinJavaVerificationObj* r_bin_java_read_next_verification_info_new(ut8* buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaAnnotation *r_bin_java_annotation_new(ut8 *buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaElementValuePair *r_bin_java_element_pair_new(ut8 *buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaElementValue *r_bin_java_element_value_new(ut8 *buffer, ut64 sz, ut64 buf_offset); // R_API RBinJavaBootStrapArgument* r_bin_java_bootstrap_method_argument_new(ut8* buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaBootStrapMethod *r_bin_java_bootstrap_method_new(ut8 *buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaAnnotationsArray *r_bin_java_annotation_array_new(ut8 *buffer, ut64 sz, ut64 buf_offset); R_API RBinJavaElementValueMetas *r_bin_java_get_ev_meta_from_tag(ut8 tag); R_API RBinJavaCPTypeMetas *U(r_bin_java_get_cp_meta_from_tag)(ut8 tag); R_API void r_bin_java_inner_classes_attr_entry_free(void /*RBinJavaClassesAttribute*/ *attr); R_API void r_bin_java_annotation_default_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_enclosing_methods_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_local_variable_type_table_attr_entry_free(void /*RBinJavaLocalVariableTypeAttribute*/ *lvattr); R_API void r_bin_java_local_variable_type_table_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_signature_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_source_debug_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_element_value_free(void /*RBinJavaElementValue*/ *element_value); R_API void r_bin_java_element_pair_free(void /*RBinJavaElementValuePair*/ *evp); R_API void r_bin_java_annotation_free(void /*RBinJavaAnnotation*/ *annotation); R_API void r_bin_java_rtv_annotations_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_rti_annotations_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_annotation_array_free(void /*RBinJavaAnnotationsArray*/ *annotation_array); R_API void r_bin_java_bootstrap_methods_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_bootstrap_method_free(void /*RBinJavaBootStrapMethod*/ *bsm); R_API void r_bin_java_bootstrap_method_argument_free(void /*RBinJavaBootStrapArgument*/ *bsm_arg); R_API void r_bin_java_rtvp_annotations_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_rtip_annotations_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_unknown_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_code_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_constant_value_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_deprecated_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_exceptions_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_inner_classes_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_line_number_table_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_local_variable_table_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_source_code_file_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_synthetic_attr_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_print_annotation_default_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_enclosing_methods_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_local_variable_type_attr_summary(RBinJavaLocalVariableTypeAttribute *lvattr); R_API void r_bin_java_print_local_variable_type_table_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_signature_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_source_debug_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_element_value_summary(RBinJavaElementValue *element_value); R_API void r_bin_java_print_annotation_summary(RBinJavaAnnotation *annotation); R_API void r_bin_java_print_element_pair_summary(RBinJavaElementValuePair *evp); R_API void r_bin_java_print_bootstrap_methods_attr_summary(RBinJavaAttrInfo *attr); // R_API void r_bin_java_bootstrap_method_summary(RBinJavaBootStrapMethod *bsm); // R_API void r_bin_java_bootstrap_method_argument_summary(RBinJavaBootStrapArgument *bsm_arg); R_API void r_bin_java_print_rtv_annotations_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_rti_annotations_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_annotation_array_summary(RBinJavaAnnotationsArray *annotation_array); R_API void r_bin_java_print_rtvp_annotations_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_rtip_annotations_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_attribute_free(void /*RBinJavaAttrInfo*/ *attr); R_API void r_bin_java_constant_pool(void /*RBinJavaCPTypeObj*/ *obj); R_API void r_bin_java_print_field_summary(RBinJavaField *field); // R_API void r_bin_java_print_interface_summary(RBinJavaField *field); R_API void r_bin_java_print_method_summary(RBinJavaField *field); R_API void r_bin_java_print_code_exceptions_attr_summary(RBinJavaExceptionEntry *exc_entry); R_API void r_bin_java_print_code_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_constant_value_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_deprecated_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_exceptions_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_classes_attr_summary(RBinJavaClassesAttribute *icattr); R_API void r_bin_java_print_inner_classes_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_line_number_table_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_local_variable_attr_summary(RBinJavaLocalVariableAttribute *lvattr); R_API void r_bin_java_print_local_variable_table_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_source_code_file_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_synthetic_attr_summary(RBinJavaAttrInfo *attr); R_API void r_bin_java_print_attr_summary(RBinJavaAttrInfo *attr); R_API RBinJavaAttrInfo *r_bin_java_read_next_attr_from_buffer(RBinJavaObj *bin, ut8 *buffer, st64 sz, st64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_unknown_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_annotation_default_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_enclosing_methods_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_local_variable_type_table_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_signature_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_source_debug_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_bootstrap_methods_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_rtv_annotations_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_rti_annotations_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_rtvp_annotations_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_rtip_annotations_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_code_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_constant_value_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_deprecated_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_exceptions_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_inner_classes_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_line_number_table_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_local_variable_table_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_source_code_file_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_stack_map_table_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API RBinJavaAttrInfo *r_bin_java_synthetic_attr_new(RBinJavaObj *bin, ut8 *buf, ut64 sz, ut64 buf_offset); R_API ut64 r_bin_java_unknown_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_annotation_default_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_enclosing_methods_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_local_variable_type_table_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_signature_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_source_debug_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_bootstrap_methods_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_rtv_annotations_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_rti_annotations_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_rtvp_annotations_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_rtip_annotations_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_code_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_constant_value_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_deprecated_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_exceptions_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_inner_classes_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_line_number_table_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_local_variable_table_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_source_code_file_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_stack_map_table_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_synthetic_attr_calc_size(RBinJavaAttrInfo *attr); R_API ut64 r_bin_java_bootstrap_method_calc_size(RBinJavaBootStrapMethod *bsm); R_API ut64 r_bin_java_element_pair_calc_size(RBinJavaElementValuePair *evp); R_API ut64 r_bin_java_element_value_calc_size(RBinJavaElementValue *element_value); R_API ut64 r_bin_java_unknown_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_class_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_fieldref_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_methodref_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_interfacemethodref_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_name_and_type_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_string_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_integer_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_float_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_long_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_double_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_utf8_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_do_nothing_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_methodhandle_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_methodtype_cp_calc_size(RBinJavaCPTypeObj *obj); R_API ut64 r_bin_java_invokedynamic_cp_calc_size(RBinJavaCPTypeObj *obj); R_API RBinJavaStackMapFrame *r_bin_java_default_stack_frame(void); R_API RList *r_bin_java_find_cp_const_by_val_float(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len); R_API RList *r_bin_java_find_cp_const_by_val_double(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len); R_API RList *r_bin_java_find_cp_const_by_val_int(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len); R_API RList *r_bin_java_find_cp_const_by_val_long(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len); R_API RList *r_bin_java_find_cp_const_by_val_utf8(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len); R_API ut8 *r_bin_java_cp_append_classref_and_name(RBinJavaObj *bin, ut32 *out_sz, const char *classname, const ut32 classname_len); R_API ut8 *U(r_bin_java_cp_append_ref_cname_fname_ftype)(RBinJavaObj * bin, ut32 * out_sz, ut8 tag, const char *cname, const ut32 c_len, const char *fname, const ut32 f_len, const char *tname, const ut32 t_len); R_API ut8 *r_bin_java_cp_get_classref(RBinJavaObj *bin, ut32 *out_sz, const char *classname, const ut32 classname_len, const ut16 name_idx); R_API ut8 *U(r_bin_java_cp_get_method_ref)(RBinJavaObj * bin, ut32 * out_sz, ut16 class_idx, ut16 name_and_type_idx); R_API ut8 *U(r_bin_java_cp_get_field_ref)(RBinJavaObj * bin, ut32 * out_sz, ut16 class_idx, ut16 name_and_type_idx); R_API ut8 *r_bin_java_cp_get_fm_ref(RBinJavaObj *bin, ut32 *out_sz, ut8 tag, ut16 class_idx, ut16 name_and_type_idx); R_API ut8 *r_bin_java_cp_get_2_ut16(RBinJavaObj *bin, ut32 *out_sz, ut8 tag, ut16 ut16_one, ut16 ut16_two); R_API ut8 *r_bin_java_cp_get_name_type(RBinJavaObj *bin, ut32 *out_sz, ut16 name_idx, ut16 type_idx); static char *convert_string(const char *bytes, ut32 len) { ut32 idx = 0, pos = 0; ut32 str_sz = 32 * len + 1; char *cpy_buffer = len > 0 ? malloc (str_sz) : NULL; if (!cpy_buffer) { return cpy_buffer; } // 4x is the increase from byte to \xHH where HH represents hexed byte memset (cpy_buffer, 0, str_sz); while (idx < len && pos < len) { if (dso_json_char_needs_hexing (bytes[idx])) { if (pos + 2 < len) { free (cpy_buffer); return NULL; } sprintf (cpy_buffer + pos, "\\x%02x", bytes[idx]); pos += 4; } else { cpy_buffer[pos] = bytes[idx]; pos++; } idx++; } return cpy_buffer; } // taken from LLVM Code Byte Swap // TODO: move into r_util R_API ut32 U(r_bin_java_swap_uint)(ut32 x) { const ut32 Byte0 = x & 0x000000FF; const ut32 Byte1 = x & 0x0000FF00; const ut32 Byte2 = x & 0x00FF0000; const ut32 Byte3 = x & 0xFF000000; return (Byte0 << 24) | (Byte1 << 8) | (Byte2 >> 8) | (Byte3 >> 24); } static bool R_BIN_JAVA_NULL_TYPE_INITTED = false; // XXX - this is a global variable used while parsing the class file // this variable should DIE. static RBinJavaObj *R_BIN_JAVA_GLOBAL_BIN = NULL; static RBinJavaAccessFlags FIELD_ACCESS_FLAGS[] = { { "public", R_BIN_JAVA_FIELD_ACC_PUBLIC, 6 }, { "private", R_BIN_JAVA_FIELD_ACC_PRIVATE, 7 }, { "protected", R_BIN_JAVA_FIELD_ACC_PROTECTED, 9 }, { "static", R_BIN_JAVA_FIELD_ACC_STATIC, 6 }, { "final", R_BIN_JAVA_FIELD_ACC_FINAL, 5 }, { "undefined.0x0020", 0x0020, 16 }, { "volatile", R_BIN_JAVA_FIELD_ACC_VOLATILE, 8 }, { "transient", R_BIN_JAVA_FIELD_ACC_TRANSIENT, 9 }, { "undefined.0x0100", 0x0100, 16 }, { "undefined.0x0200", 0x0200, 16 }, { "undefined.0x0400", 0x0400, 16 }, { "undefined.0x0800", 0x0800, 16 }, { "synthetic", R_BIN_JAVA_FIELD_ACC_SYNTHETIC, 9 }, { "undefined.0x2000", 0x2000, 16 }, { "enum", R_BIN_JAVA_FIELD_ACC_ENUM, 16 }, { "undefined.0x8000", 0x8000, 16 }, { NULL, 0, 0 } }; static RBinJavaAccessFlags METHOD_ACCESS_FLAGS[] = { { "public", R_BIN_JAVA_METHOD_ACC_PUBLIC, 6 }, { "private", R_BIN_JAVA_METHOD_ACC_PRIVATE, 7 }, { "protected", R_BIN_JAVA_METHOD_ACC_PROTECTED, 9 }, { "static", R_BIN_JAVA_METHOD_ACC_STATIC, 6 }, { "final", R_BIN_JAVA_METHOD_ACC_FINAL, 5 }, { "synchronized", R_BIN_JAVA_METHOD_ACC_SYNCHRONIZED, 12 }, { "bridge", R_BIN_JAVA_METHOD_ACC_BRIDGE, 6 }, { "varargs", R_BIN_JAVA_METHOD_ACC_VARARGS, 7 }, { "native", R_BIN_JAVA_METHOD_ACC_NATIVE, 6 }, { "interface", R_BIN_JAVA_METHOD_ACC_INTERFACE, 9 }, { "abstract", R_BIN_JAVA_METHOD_ACC_ABSTRACT, 8 }, { "strict", R_BIN_JAVA_METHOD_ACC_STRICT, 6 }, { "synthetic", R_BIN_JAVA_METHOD_ACC_SYNTHETIC, 9 }, { "annotation", R_BIN_JAVA_METHOD_ACC_ANNOTATION, 10 }, { "enum", R_BIN_JAVA_METHOD_ACC_ENUM, 4 }, { "undefined.0x8000", 0x8000, 16 }, { NULL, 0, 0 } }; // XXX - Fix these there are some incorrect ongs static RBinJavaAccessFlags CLASS_ACCESS_FLAGS[] = { { "public", R_BIN_JAVA_CLASS_ACC_PUBLIC, 6 }, { "undefined.0x0002", 0x0002, 16 }, { "undefined.0x0004", 0x0004, 16 }, { "undefined.0x0008", 0x0008, 16 }, { "final", R_BIN_JAVA_CLASS_ACC_FINAL, 5 }, { "super", R_BIN_JAVA_CLASS_ACC_SUPER, 5 }, { "undefined.0x0040", 0x0040, 16 }, { "undefined.0x0080", 0x0080, 16 }, { "undefined.0x0100", 0x0100, 16 }, { "interface", R_BIN_JAVA_CLASS_ACC_INTERFACE, 9 }, { "abstract", R_BIN_JAVA_CLASS_ACC_ABSTRACT, 8 }, { "undefined.0x0800", 0x0800, 16 }, { "synthetic", R_BIN_JAVA_CLASS_ACC_SYNTHETIC, 9 }, { "annotation", R_BIN_JAVA_CLASS_ACC_ANNOTATION, 10 }, { "enum", R_BIN_JAVA_CLASS_ACC_ENUM, 4 }, { "undefined.0x8000", 0x8000, 16 }, { NULL, 0, 0 } }; static RBinJavaRefMetas R_BIN_JAVA_REF_METAS[] = { { "Unknown", R_BIN_JAVA_REF_UNKNOWN }, { "GetField", R_BIN_JAVA_REF_GETFIELD }, { "GetStatic", R_BIN_JAVA_REF_GETSTATIC }, { "PutField", R_BIN_JAVA_REF_PUTFIELD }, { "PutStatic", R_BIN_JAVA_REF_PUTSTATIC }, { "InvokeVirtual", R_BIN_JAVA_REF_INVOKEVIRTUAL }, { "InvokeStatic", R_BIN_JAVA_REF_INVOKESTATIC }, { "InvokeSpecial", R_BIN_JAVA_REF_INVOKESPECIAL }, { "NewInvokeSpecial", R_BIN_JAVA_REF_NEWINVOKESPECIAL }, { "InvokeInterface", R_BIN_JAVA_REF_INVOKEINTERFACE } }; static ut16 R_BIN_JAVA_ELEMENT_VALUE_METAS_SZ = 14; static RBinJavaElementValueMetas R_BIN_JAVA_ELEMENT_VALUE_METAS[] = { { "Byte", R_BIN_JAVA_EV_TAG_BYTE, NULL }, { "Char", R_BIN_JAVA_EV_TAG_CHAR, NULL }, { "Double", R_BIN_JAVA_EV_TAG_DOUBLE, NULL }, { "Float", R_BIN_JAVA_EV_TAG_FLOAT, NULL }, { "Integer", R_BIN_JAVA_EV_TAG_INT, NULL }, { "Long", R_BIN_JAVA_EV_TAG_LONG, NULL }, { "Short", R_BIN_JAVA_EV_TAG_SHORT, NULL }, { "Boolean", R_BIN_JAVA_EV_TAG_BOOLEAN, NULL }, { "Array of ", R_BIN_JAVA_EV_TAG_ARRAY, NULL }, { "String", R_BIN_JAVA_EV_TAG_STRING, NULL }, { "Enum", R_BIN_JAVA_EV_TAG_ENUM, NULL }, { "Class", R_BIN_JAVA_EV_TAG_CLASS, NULL }, { "Annotation", R_BIN_JAVA_EV_TAG_ANNOTATION, NULL }, { "Unknown", R_BIN_JAVA_EV_TAG_UNKNOWN, NULL }, }; static RBinJavaVerificationMetas R_BIN_JAVA_VERIFICATION_METAS[] = { { "Top", R_BIN_JAVA_STACKMAP_TOP }, { "Integer", R_BIN_JAVA_STACKMAP_INTEGER }, { "Float", R_BIN_JAVA_STACKMAP_FLOAT }, { "Double", R_BIN_JAVA_STACKMAP_DOUBLE }, { "Long", R_BIN_JAVA_STACKMAP_LONG }, { "NULL", R_BIN_JAVA_STACKMAP_NULL }, { "This", R_BIN_JAVA_STACKMAP_THIS }, { "Object", R_BIN_JAVA_STACKMAP_OBJECT }, { "Uninitialized", R_BIN_JAVA_STACKMAP_UNINIT }, { "Unknown", R_BIN_JAVA_STACKMAP_UNKNOWN } }; static RBinJavaStackMapFrameMetas R_BIN_JAVA_STACK_MAP_FRAME_METAS[] = { { "ImplicitStackFrame", R_BIN_JAVA_STACK_FRAME_IMPLICIT, NULL }, { "Same", R_BIN_JAVA_STACK_FRAME_SAME, NULL }, { "SameLocals1StackItem", R_BIN_JAVA_STACK_FRAME_SAME_LOCALS_1, NULL }, { "Chop", R_BIN_JAVA_STACK_FRAME_CHOP, NULL }, { "SameFrameExtended", R_BIN_JAVA_STACK_FRAME_SAME_FRAME_EXTENDED, NULL }, { "Append", R_BIN_JAVA_STACK_FRAME_APPEND, NULL }, { "FullFrame", R_BIN_JAVA_STACK_FRAME_FULL_FRAME, NULL }, { "Reserved", R_BIN_JAVA_STACK_FRAME_RESERVED, NULL } }; static RBinJavaCPTypeObjectAllocs R_BIN_ALLOCS_CONSTANTS[] = { { r_bin_java_do_nothing_new, r_bin_java_do_nothing_free, r_bin_java_print_null_cp_summary, r_bin_java_do_nothing_calc_size, r_bin_java_print_null_cp_stringify }, { r_bin_java_utf8_cp_new, r_bin_java_utf8_info_free, r_bin_java_print_utf8_cp_summary, r_bin_java_utf8_cp_calc_size, r_bin_java_print_utf8_cp_stringify }, { r_bin_java_unknown_cp_new, r_bin_java_default_free, r_bin_java_print_unknown_cp_summary, r_bin_java_unknown_cp_calc_size, r_bin_java_print_unknown_cp_stringify }, { r_bin_java_integer_cp_new, r_bin_java_default_free, r_bin_java_print_integer_cp_summary, r_bin_java_integer_cp_calc_size, r_bin_java_print_integer_cp_stringify }, { r_bin_java_float_cp_new, r_bin_java_default_free, r_bin_java_print_float_cp_summary, r_bin_java_float_cp_calc_size, r_bin_java_print_float_cp_stringify }, { r_bin_java_long_cp_new, r_bin_java_default_free, r_bin_java_print_long_cp_summary, r_bin_java_long_cp_calc_size, r_bin_java_print_long_cp_stringify }, { r_bin_java_double_cp_new, r_bin_java_default_free, r_bin_java_print_double_cp_summary, r_bin_java_double_cp_calc_size, r_bin_java_print_double_cp_stringify }, { r_bin_java_class_cp_new, r_bin_java_default_free, r_bin_java_print_classref_cp_summary, r_bin_java_class_cp_calc_size, r_bin_java_print_classref_cp_stringify }, { r_bin_java_string_cp_new, r_bin_java_default_free, r_bin_java_print_string_cp_summary, r_bin_java_string_cp_calc_size, r_bin_java_print_string_cp_stringify }, { r_bin_java_fieldref_cp_new, r_bin_java_default_free, r_bin_java_print_fieldref_cp_summary, r_bin_java_fieldref_cp_calc_size, r_bin_java_print_fieldref_cp_stringify }, { r_bin_java_methodref_cp_new, r_bin_java_default_free, r_bin_java_print_methodref_cp_summary, r_bin_java_methodref_cp_calc_size, r_bin_java_print_methodref_cp_stringify }, { r_bin_java_interfacemethodref_cp_new, r_bin_java_default_free, r_bin_java_print_interfacemethodref_cp_summary, r_bin_java_interfacemethodref_cp_calc_size, r_bin_java_print_interfacemethodref_cp_stringify }, { r_bin_java_name_and_type_cp_new, r_bin_java_default_free, r_bin_java_print_name_and_type_cp_summary, r_bin_java_name_and_type_cp_calc_size, r_bin_java_print_name_and_type_cp_stringify }, { NULL, NULL, NULL, NULL, NULL }, { NULL, NULL, NULL, NULL, NULL }, { r_bin_java_methodhandle_cp_new, r_bin_java_default_free, r_bin_java_print_methodhandle_cp_summary, r_bin_java_methodhandle_cp_calc_size, r_bin_java_print_methodhandle_cp_stringify }, { r_bin_java_methodtype_cp_new, r_bin_java_default_free, r_bin_java_print_methodtype_cp_summary, r_bin_java_methodtype_cp_calc_size, r_bin_java_print_methodtype_cp_stringify }, { NULL, NULL, NULL, NULL, NULL }, { r_bin_java_invokedynamic_cp_new, r_bin_java_default_free, r_bin_java_print_invokedynamic_cp_summary, r_bin_java_invokedynamic_cp_calc_size, r_bin_java_print_invokedynamic_cp_stringify }, }; static RBinJavaCPTypeObj R_BIN_JAVA_NULL_TYPE; static ut8 R_BIN_JAVA_CP_METAS_SZ = 12; static RBinJavaCPTypeMetas R_BIN_JAVA_CP_METAS[] = { // Each field has a name pointer and a tag field { "NULL", R_BIN_JAVA_CP_NULL, 0, &R_BIN_ALLOCS_CONSTANTS[0] }, { "Utf8", R_BIN_JAVA_CP_UTF8, 3, &R_BIN_ALLOCS_CONSTANTS[1] }, // 2 bytes = length, N bytes string (containts a pointer in the field) { "Unknown", R_BIN_JAVA_CP_UNKNOWN, 0, &R_BIN_ALLOCS_CONSTANTS[2] }, { "Integer", R_BIN_JAVA_CP_INTEGER, 5, &R_BIN_ALLOCS_CONSTANTS[3] }, // 4 bytes { "Float", R_BIN_JAVA_CP_FLOAT, 5, &R_BIN_ALLOCS_CONSTANTS[4] }, // 4 bytes { "Long", R_BIN_JAVA_CP_LONG, 9, &R_BIN_ALLOCS_CONSTANTS[5] }, // 4 high 4 low { "Double", R_BIN_JAVA_CP_DOUBLE, 9, &R_BIN_ALLOCS_CONSTANTS[6] }, // 4 high 4 low { "Class", R_BIN_JAVA_CP_CLASS, 3, &R_BIN_ALLOCS_CONSTANTS[7] }, // 2 name_idx { "String", R_BIN_JAVA_CP_STRING, 3, &R_BIN_ALLOCS_CONSTANTS[8] }, // 2 string_idx { "FieldRef", R_BIN_JAVA_CP_FIELDREF, 5, &R_BIN_ALLOCS_CONSTANTS[9] }, // 2 class idx, 2 name/type_idx { "MethodRef", R_BIN_JAVA_CP_METHODREF, 5, &R_BIN_ALLOCS_CONSTANTS[10] }, // 2 class idx, 2 name/type_idx { "InterfaceMethodRef", R_BIN_JAVA_CP_INTERFACEMETHOD_REF, 5, &R_BIN_ALLOCS_CONSTANTS[11] }, // 2 class idx, 2 name/type_idx { "NameAndType", R_BIN_JAVA_CP_NAMEANDTYPE, 5, &R_BIN_ALLOCS_CONSTANTS[12] }, // 4 high 4 low { "Unknown", R_BIN_JAVA_CP_UNKNOWN, 0, &R_BIN_ALLOCS_CONSTANTS[2] }, { "Unknown", R_BIN_JAVA_CP_UNKNOWN, 0, &R_BIN_ALLOCS_CONSTANTS[2] }, { "MethodHandle", R_BIN_JAVA_CP_METHODHANDLE, 4, &R_BIN_ALLOCS_CONSTANTS[15] }, // 4 high 4 low { "MethodType", R_BIN_JAVA_CP_METHODTYPE, 3, &R_BIN_ALLOCS_CONSTANTS[16] }, // 4 high 4 low { "Unknown", R_BIN_JAVA_CP_UNKNOWN, 0, &R_BIN_ALLOCS_CONSTANTS[2] }, { "InvokeDynamic", R_BIN_JAVA_CP_INVOKEDYNAMIC, 5, &R_BIN_ALLOCS_CONSTANTS[18] }, // 4 high 4 low }; static RBinJavaAttrInfoObjectAllocs RBIN_JAVA_ATTRS_ALLOCS[] = { { r_bin_java_annotation_default_attr_new, r_bin_java_annotation_default_attr_free, r_bin_java_print_annotation_default_attr_summary, r_bin_java_annotation_default_attr_calc_size }, { r_bin_java_bootstrap_methods_attr_new, r_bin_java_bootstrap_methods_attr_free, r_bin_java_print_bootstrap_methods_attr_summary, r_bin_java_bootstrap_methods_attr_calc_size }, { r_bin_java_code_attr_new, r_bin_java_code_attr_free, r_bin_java_print_code_attr_summary, r_bin_java_code_attr_calc_size }, { r_bin_java_constant_value_attr_new, r_bin_java_constant_value_attr_free, r_bin_java_print_constant_value_attr_summary, r_bin_java_constant_value_attr_calc_size }, { r_bin_java_deprecated_attr_new, r_bin_java_deprecated_attr_free, r_bin_java_print_deprecated_attr_summary, r_bin_java_deprecated_attr_calc_size }, { r_bin_java_enclosing_methods_attr_new, r_bin_java_enclosing_methods_attr_free, r_bin_java_print_enclosing_methods_attr_summary, r_bin_java_enclosing_methods_attr_calc_size }, { r_bin_java_exceptions_attr_new, r_bin_java_exceptions_attr_free, r_bin_java_print_exceptions_attr_summary, r_bin_java_exceptions_attr_calc_size }, { r_bin_java_inner_classes_attr_new, r_bin_java_inner_classes_attr_free, r_bin_java_print_inner_classes_attr_summary, r_bin_java_inner_classes_attr_calc_size }, { r_bin_java_line_number_table_attr_new, r_bin_java_line_number_table_attr_free, r_bin_java_print_line_number_table_attr_summary, r_bin_java_line_number_table_attr_calc_size }, { r_bin_java_local_variable_table_attr_new, r_bin_java_local_variable_table_attr_free, r_bin_java_print_local_variable_table_attr_summary, r_bin_java_local_variable_table_attr_calc_size }, { r_bin_java_local_variable_type_table_attr_new, r_bin_java_local_variable_type_table_attr_free, r_bin_java_print_local_variable_type_table_attr_summary, r_bin_java_local_variable_type_table_attr_calc_size }, { r_bin_java_rti_annotations_attr_new, r_bin_java_rti_annotations_attr_free, r_bin_java_print_rti_annotations_attr_summary, r_bin_java_rti_annotations_attr_calc_size }, { r_bin_java_rtip_annotations_attr_new, r_bin_java_rtip_annotations_attr_free, r_bin_java_print_rtip_annotations_attr_summary, r_bin_java_rtip_annotations_attr_calc_size }, { r_bin_java_rtv_annotations_attr_new, r_bin_java_rtv_annotations_attr_free, r_bin_java_print_rtv_annotations_attr_summary, r_bin_java_rtv_annotations_attr_calc_size }, { r_bin_java_rtvp_annotations_attr_new, r_bin_java_rtvp_annotations_attr_free, r_bin_java_print_rtvp_annotations_attr_summary, r_bin_java_rtvp_annotations_attr_calc_size }, { r_bin_java_signature_attr_new, r_bin_java_signature_attr_free, r_bin_java_print_signature_attr_summary, r_bin_java_signature_attr_calc_size }, { r_bin_java_source_debug_attr_new, r_bin_java_source_debug_attr_free, r_bin_java_print_source_debug_attr_summary, r_bin_java_source_debug_attr_calc_size }, { r_bin_java_source_code_file_attr_new, r_bin_java_source_code_file_attr_free, r_bin_java_print_source_code_file_attr_summary, r_bin_java_source_code_file_attr_calc_size }, { r_bin_java_stack_map_table_attr_new, r_bin_java_stack_map_table_attr_free, r_bin_java_print_stack_map_table_attr_summary, r_bin_java_stack_map_table_attr_calc_size }, { r_bin_java_synthetic_attr_new, r_bin_java_synthetic_attr_free, r_bin_java_print_synthetic_attr_summary, r_bin_java_synthetic_attr_calc_size }, { r_bin_java_unknown_attr_new, r_bin_java_unknown_attr_free, r_bin_java_print_unknown_attr_summary, r_bin_java_unknown_attr_calc_size } }; // R_API ut32 RBIN_JAVA_ATTRS_METAS_SZ = 21; static ut32 RBIN_JAVA_ATTRS_METAS_SZ = 20; static RBinJavaAttrMetas RBIN_JAVA_ATTRS_METAS[] = { { "AnnotationDefault", R_BIN_JAVA_ATTR_TYPE_ANNOTATION_DEFAULT_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[0] }, { "BootstrapMethods", R_BIN_JAVA_ATTR_TYPE_BOOTSTRAP_METHODS_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[1] }, { "Code", R_BIN_JAVA_ATTR_TYPE_CODE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[2] }, { "ConstantValue", R_BIN_JAVA_ATTR_TYPE_CONST_VALUE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[3] }, { "Deperecated", R_BIN_JAVA_ATTR_TYPE_DEPRECATED_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[4] }, { "EnclosingMethod", R_BIN_JAVA_ATTR_TYPE_ENCLOSING_METHOD_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[5] }, { "Exceptions", R_BIN_JAVA_ATTR_TYPE_EXCEPTIONS_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[6] }, { "InnerClasses", R_BIN_JAVA_ATTR_TYPE_INNER_CLASSES_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[7] }, { "LineNumberTable", R_BIN_JAVA_ATTR_TYPE_LINE_NUMBER_TABLE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[8] }, { "LocalVariableTable", R_BIN_JAVA_ATTR_TYPE_LOCAL_VARIABLE_TABLE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[9] }, { "LocalVariableTypeTable", R_BIN_JAVA_ATTR_TYPE_LOCAL_VARIABLE_TYPE_TABLE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[10] }, { "RuntimeInvisibleAnnotations", R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_ANNOTATION_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[11] }, { "RuntimeInvisibleParameterAnnotations", R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_PARAMETER_ANNOTATION_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[12] }, { "RuntimeVisibleAnnotations", R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_ANNOTATION_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[13] }, { "RuntimeVisibleParameterAnnotations", R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_PARAMETER_ANNOTATION_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[14] }, { "Signature", R_BIN_JAVA_ATTR_TYPE_SIGNATURE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[15] }, { "SourceDebugExtension", R_BIN_JAVA_ATTR_TYPE_SOURCE_DEBUG_EXTENTSION_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[16] }, { "SourceFile", R_BIN_JAVA_ATTR_TYPE_SOURCE_FILE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[17] }, { "StackMapTable", R_BIN_JAVA_ATTR_TYPE_STACK_MAP_TABLE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[18] }, // { "StackMap", R_BIN_JAVA_ATTR_TYPE_STACK_MAP_TABLE_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[18]}, { "Synthetic", R_BIN_JAVA_ATTR_TYPE_SYNTHETIC_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[19] }, { "Unknown", R_BIN_JAVA_ATTR_TYPE_UNKNOWN_ATTR, &RBIN_JAVA_ATTRS_ALLOCS[20] } }; R_API bool r_bin_java_is_old_format(RBinJavaObj *bin) { return bin->cf.major[1] == 45 && bin->cf.minor[1] <= 2; } R_API void r_bin_java_reset_bin_info(RBinJavaObj *bin) { free (bin->cf2.flags_str); free (bin->cf2.this_class_name); r_list_free (bin->imports_list); r_list_free (bin->methods_list); r_list_free (bin->fields_list); r_list_free (bin->attrs_list); r_list_free (bin->cp_list); r_list_free (bin->interfaces_list); r_str_constpool_fini (&bin->constpool); memset (bin, 0, sizeof (RBinJavaObj)); r_str_constpool_init (&bin->constpool); bin->cf2.flags_str = strdup ("unknown"); bin->cf2.this_class_name = strdup ("unknown"); bin->imports_list = r_list_newf (free); bin->methods_list = r_list_newf (r_bin_java_fmtype_free); bin->fields_list = r_list_newf (r_bin_java_fmtype_free); bin->attrs_list = r_list_newf (r_bin_java_attribute_free); bin->cp_list = r_list_newf (r_bin_java_constant_pool); bin->interfaces_list = r_list_newf (r_bin_java_interface_free); } R_API char *r_bin_java_unmangle_method(const char *flags, const char *name, const char *params, const char *r_value) { RList *the_list = params ? r_bin_java_extract_type_values (params) : r_list_new (); RListIter *iter = NULL; // second case removes leading space if no flags are given const char *fmt = flags ? "%s %s %s (%s)" : "%s%s %s (%s)"; char *str = NULL, *f_val_str = NULL, *r_val_str = NULL, *prototype = NULL, *p_val_str = NULL; ut32 params_idx = 0, params_len = 0, prototype_len = 0; if (!extract_type_value (r_value, &r_val_str)) { r_list_free (the_list); return NULL; } if (!r_val_str) { r_val_str = strdup ("UNKNOWN"); } f_val_str = strdup (r_str_get (flags)); r_list_foreach (the_list, iter, str) { params_len += strlen (str); if (params_idx > 0) { params_len += 2; } params_idx++; } if (params_len > 0) { ut32 offset = 0; params_len += 1; p_val_str = malloc (params_len); r_list_foreach (the_list, iter, str) { if (offset != 0) { offset += snprintf (p_val_str + offset, params_len - offset, ", %s", str); } else { offset += snprintf (p_val_str + offset, params_len - offset, "%s", str); } } } else { p_val_str = strdup (""); } prototype_len += (flags ? strlen (flags) + 1 : 0); // space vs no space prototype_len += strlen (name) + 1; // name + space prototype_len += strlen (r_val_str) + 1; // r_value + space prototype_len += strlen (p_val_str) + 3; // space + l_paren + params + r_paren prototype_len += 1; // null prototype = malloc (prototype_len); /// TODO enable this function and start using it to demangle strings snprintf (prototype, prototype_len, fmt, f_val_str, r_val_str, name, p_val_str); free (f_val_str); free (r_val_str); free (p_val_str); r_list_free (the_list); return prototype; } R_API char *r_bin_java_unmangle(const char *flags, const char *name, const char *descriptor) { ut32 l_paren_pos = -1, r_paren_pos = -1; char *result = NULL; ut32 desc_len = descriptor && *descriptor ? strlen (descriptor) : 0, name_len = name && *name ? strlen (name) : 0, flags_len = flags && *flags ? strlen (flags) : 0, i = 0; if (desc_len == 0 || name == 0) { return NULL; } for (i = 0; i < desc_len; i++) { if (descriptor[i] == '(') { l_paren_pos = i; } else if (l_paren_pos != (ut32) - 1 && descriptor[i] == ')') { r_paren_pos = i; break; } } // handle field case; if (l_paren_pos == (ut32) - 1 && r_paren_pos == (ut32) - 1) { char *unmangle_field_desc = NULL; ut32 len = extract_type_value (descriptor, &unmangle_field_desc); if (len == 0) { eprintf ("Warning: attempting to unmangle invalid type descriptor.\n"); free (unmangle_field_desc); return result; } if (flags_len > 0) { len += (flags_len + name_len + 5); // space and null result = malloc (len); snprintf (result, len, "%s %s %s", flags, unmangle_field_desc, name); } else { len += (name_len + 5); // space and null result = malloc (len); snprintf (result, len, "%s %s", unmangle_field_desc, name); } free (unmangle_field_desc); } else if (l_paren_pos != (ut32) - 1 && r_paren_pos != (ut32) - 1 && l_paren_pos < r_paren_pos) { // params_len account for l_paren + 1 and null ut32 params_len = r_paren_pos - (l_paren_pos + 1) != 0 ? r_paren_pos - (l_paren_pos + 1) + 1 : 0; char *params = params_len ? malloc (params_len) : NULL; const char *rvalue = descriptor + r_paren_pos + 1; if (params) { snprintf (params, params_len, "%s", descriptor + l_paren_pos + 1); } result = r_bin_java_unmangle_method (flags, name, params, rvalue); free (params); } return result; } R_API DsoJsonObj *r_bin_java_get_bin_obj_json(RBinJavaObj *bin) { DsoJsonObj *imports_list = r_bin_java_get_import_json_definitions (bin); DsoJsonObj *fields_list = r_bin_java_get_field_json_definitions (bin); DsoJsonObj *methods_list = r_bin_java_get_method_json_definitions (bin); // interfaces_list = r_bin_java_get_interface_json_definitions (bin); DsoJsonObj *class_dict = r_bin_java_get_class_info_json (bin); char *res = dso_json_obj_to_str (methods_list); // eprintf ("Resulting methods json: \n%s\n", res); free (res); if (dso_json_dict_insert_str_key_obj (class_dict, "methods", methods_list)) { // dso_json_list_free (methods_list); dso_json_obj_del (methods_list); } res = dso_json_obj_to_str (fields_list); // eprintf ("Resulting fields json: \n%s\n", res); free (res); if (dso_json_dict_insert_str_key_obj (class_dict, "fields", fields_list)) { // dso_json_list_free (fields_list); dso_json_obj_del (fields_list); } res = dso_json_obj_to_str (imports_list); // eprintf ("Resulting imports json: \n%s\n", res); free (res); if (dso_json_dict_insert_str_key_obj (class_dict, "imports", imports_list)) { // dso_json_list_free (imports_list); dso_json_obj_del (imports_list); } // res = dso_json_obj_to_str (interfaces_list); // eprintf ("Resulting interfaces json: \n%s\n", res); // free (res); // dso_json_dict_insert_str_key_obj (class_dict, "interfaces", interfaces_list); res = dso_json_obj_to_str (class_dict); // eprintf ("Resulting class info json: \n%s\n", res); free (res); // dso_json_obj_del (class_dict); return class_dict; } R_API DsoJsonObj *r_bin_java_get_import_json_definitions(RBinJavaObj *bin) { RList *the_list; DsoJsonObj *json_list = dso_json_list_new (); RListIter *iter = NULL; char *new_str; if (!bin || !(the_list = r_bin_java_get_lib_names (bin))) { return json_list; } r_list_foreach (the_list, iter, new_str) { char *tmp = new_str; // eprintf ("Processing string: %s\n", new_str); while (*tmp) { if (*tmp == '/') { *tmp = '.'; } tmp++; } // eprintf ("adding string: %s\n", new_str); dso_json_list_append_str (json_list, new_str); } r_list_free (the_list); return json_list; } R_API DsoJsonObj *r_bin_java_get_class_info_json(RBinJavaObj *bin) { RList *classes = r_bin_java_get_classes (bin); DsoJsonObj *interfaces_list = dso_json_list_new (); DsoJsonObj *class_info_dict = dso_json_dict_new (); RBinClass *class_ = r_list_get_n (classes, 0); if (class_) { int dummy = 0; RListIter *iter; RBinClass *class_v = NULL; // add access flags like in methods bool is_public = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_PUBLIC) != 0); bool is_final = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_FINAL) != 0); bool is_super = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_SUPER) != 0); bool is_interface = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_INTERFACE) != 0); bool is_abstract = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_ABSTRACT) != 0); bool is_synthetic = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_SYNTHETIC) != 0); bool is_annotation = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_ANNOTATION) != 0); bool is_enum = ((class_->visibility & R_BIN_JAVA_CLASS_ACC_ENUM) != 0); dso_json_dict_insert_str_key_num (class_info_dict, "access_flags", class_->visibility); dso_json_dict_insert_str_key_num (class_info_dict, "is_public", is_public); dso_json_dict_insert_str_key_num (class_info_dict, "is_final", is_final); dso_json_dict_insert_str_key_num (class_info_dict, "is_super", is_super); dso_json_dict_insert_str_key_num (class_info_dict, "is_interface", is_interface); dso_json_dict_insert_str_key_num (class_info_dict, "is_abstract", is_abstract); dso_json_dict_insert_str_key_num (class_info_dict, "is_synthetic", is_synthetic); dso_json_dict_insert_str_key_num (class_info_dict, "is_annotation", is_annotation); dso_json_dict_insert_str_key_num (class_info_dict, "is_enum", is_enum); dso_json_dict_insert_str_key_str (class_info_dict, "name", class_->name); if (!class_->super) { DsoJsonObj *str = dso_json_str_new (); if (dso_json_dict_insert_str_key_obj (class_info_dict, "super", str)) { dso_json_str_free (str); } } else { dso_json_dict_insert_str_key_str (class_info_dict, "super", class_->super); } r_list_foreach (classes, iter, class_v) { if (!dummy) { dummy++; continue; } // enumerate all interface classes and append them to the interfaces if ((class_v->visibility & R_BIN_JAVA_CLASS_ACC_INTERFACE) != 0) { dso_json_list_append_str (interfaces_list, class_v->name); } } } if (dso_json_dict_insert_str_key_obj (class_info_dict, "interfaces", interfaces_list)) { // dso_json_list_free (interfaces_list); dso_json_obj_del (interfaces_list); } r_list_free (classes); return class_info_dict; } R_API DsoJsonObj *r_bin_java_get_interface_json_definitions(RBinJavaObj *bin) { RList *the_list; DsoJsonObj *json_list = dso_json_list_new (); RListIter *iter = NULL; char *new_str; if (!bin || !(the_list = r_bin_java_get_interface_names (bin))) { return json_list; } r_list_foreach (the_list, iter, new_str) { char *tmp = new_str; // eprintf ("Processing string: %s\n", new_str); while (*tmp) { if (*tmp == '/') { *tmp = '.'; } tmp++; } // eprintf ("adding string: %s\n", new_str); dso_json_list_append_str (json_list, new_str); } r_list_free (the_list); return json_list; } R_API DsoJsonObj *r_bin_java_get_method_json_definitions(RBinJavaObj *bin) { RBinJavaField *fm_type = NULL; RListIter *iter = NULL; DsoJsonObj *json_list = dso_json_list_new (); if (!bin) { return json_list; } r_list_foreach (bin->methods_list, iter, fm_type) { DsoJsonObj *method_proto = r_bin_java_get_method_json_definition (bin, fm_type); // eprintf ("Method json: %s\n", method_proto); dso_json_list_append (json_list, method_proto); } return json_list; } R_API DsoJsonObj *r_bin_java_get_field_json_definitions(RBinJavaObj *bin) { RBinJavaField *fm_type = NULL; RListIter *iter = NULL; DsoJsonObj *json_list = dso_json_list_new (); if (!bin) { return json_list; } r_list_foreach (bin->fields_list, iter, fm_type) { DsoJsonObj *field_proto = r_bin_java_get_field_json_definition (bin, fm_type); // eprintf ("Field json: %s\n", field_proto); dso_json_list_append (json_list, field_proto); } return json_list; } R_API char *r_bin_java_create_method_fq_str(const char *klass, const char *name, const char *signature) { if (!klass) { klass = "null_class"; } if (!name) { name = "null_name"; } if (!signature) { signature = "null_signature"; } return r_str_newf ("%s.%s.%s", klass, name, signature); } R_API char *r_bin_java_create_field_fq_str(const char *klass, const char *name, const char *signature) { if (!klass) { klass = "null_class"; } if (!name) { name = "null_name"; } if (!signature) { signature = "null_signature"; } return r_str_newf ("%s %s.%s", signature, klass, name); } R_API DsoJsonObj *r_bin_java_get_fm_type_definition_json(RBinJavaObj *bin, RBinJavaField *fm_type, int is_method) { ut64 addr = UT64_MAX; char *prototype = NULL, *fq_name = NULL; bool is_native = ((fm_type->flags & R_BIN_JAVA_METHOD_ACC_NATIVE) != 0); bool is_static = ((fm_type->flags & R_BIN_JAVA_METHOD_ACC_STATIC) != 0); bool is_synthetic = ((fm_type->flags & R_BIN_JAVA_METHOD_ACC_SYNTHETIC) != 0); bool is_private = ((fm_type->flags & R_BIN_JAVA_METHOD_ACC_PRIVATE) != 0); bool is_public = ((fm_type->flags & R_BIN_JAVA_METHOD_ACC_PUBLIC) != 0); bool is_protected = ((fm_type->flags & R_BIN_JAVA_METHOD_ACC_PROTECTED) != 0); bool is_super = ((fm_type->flags & R_BIN_JAVA_CLASS_ACC_SUPER) != 0); DsoJsonObj *fm_type_dict = dso_json_dict_new (); dso_json_dict_insert_str_key_num (fm_type_dict, "access_flags", fm_type->flags); dso_json_dict_insert_str_key_num (fm_type_dict, "is_method", is_method); dso_json_dict_insert_str_key_num (fm_type_dict, "is_native", is_native); dso_json_dict_insert_str_key_num (fm_type_dict, "is_synthetic", is_synthetic); dso_json_dict_insert_str_key_num (fm_type_dict, "is_private", is_private); dso_json_dict_insert_str_key_num (fm_type_dict, "is_public", is_public); dso_json_dict_insert_str_key_num (fm_type_dict, "is_static", is_static); dso_json_dict_insert_str_key_num (fm_type_dict, "is_protected", is_protected); dso_json_dict_insert_str_key_num (fm_type_dict, "is_super", is_super); addr = r_bin_java_get_method_code_offset (fm_type); if (addr == 0) { addr = fm_type->file_offset; } addr += bin->loadaddr; dso_json_dict_insert_str_key_num (fm_type_dict, "addr", addr); dso_json_dict_insert_str_key_num (fm_type_dict, "offset", fm_type->file_offset + bin->loadaddr); dso_json_dict_insert_str_key_str (fm_type_dict, "class_name", fm_type->class_name); dso_json_dict_insert_str_key_str (fm_type_dict, "signature", fm_type->descriptor); dso_json_dict_insert_str_key_str (fm_type_dict, "name", fm_type->name); if (is_method) { fq_name = r_bin_java_create_method_fq_str (fm_type->class_name, fm_type->name, fm_type->descriptor); } else { fq_name = r_bin_java_create_field_fq_str (fm_type->class_name, fm_type->name, fm_type->descriptor); } dso_json_dict_insert_str_key_str (fm_type_dict, "fq_name", fq_name); prototype = r_bin_java_unmangle (fm_type->flags_str, fm_type->name, fm_type->descriptor); dso_json_dict_insert_str_key_str (fm_type_dict, "prototype", prototype); free (prototype); free (fq_name); return fm_type_dict; } R_API char *r_bin_java_get_method_definition(RBinJavaField *fm_type) { return r_bin_java_unmangle (fm_type->flags_str, fm_type->name, fm_type->descriptor); } R_API char *r_bin_java_get_field_definition(RBinJavaField *fm_type) { return r_bin_java_unmangle (fm_type->flags_str, fm_type->name, fm_type->descriptor); } R_API DsoJsonObj *r_bin_java_get_method_json_definition(RBinJavaObj *bin, RBinJavaField *fm_type) { return r_bin_java_get_fm_type_definition_json (bin, fm_type, 1); } R_API DsoJsonObj *r_bin_java_get_field_json_definition(RBinJavaObj *bin, RBinJavaField *fm_type) { return r_bin_java_get_fm_type_definition_json (bin, fm_type, 0); } R_API int r_bin_java_extract_reference_name(const char *input_str, char **ref_str, ut8 array_cnt) { char *new_str = NULL; ut32 str_len = array_cnt ? (array_cnt + 1) * 2 : 0; const char *str_pos = input_str; int consumed = 0, len = 0; if (!str_pos || *str_pos != 'L' || !*str_pos) { return -1; } consumed++; str_pos++; while (*str_pos && *str_pos != ';') { str_pos++; len++; consumed++; } str_pos = input_str + 1; free (*ref_str); str_len += len; *ref_str = malloc (str_len + 1); new_str = *ref_str; memcpy (new_str, str_pos, str_len); new_str[str_len] = 0; while (*new_str) { if (*new_str == '/') { *new_str = '.'; } new_str++; } return len + 2; } R_API void UNUSED_FUNCTION(r_bin_java_print_prototypes)(RBinJavaObj * bin) { RList *the_list = r_bin_java_get_method_definitions (bin); RListIter *iter; char *str; r_list_foreach (the_list, iter, str) { eprintf ("%s;\n", str); } r_list_free (the_list); } R_API char *get_type_value_str(const char *arg_str, ut8 array_cnt) { ut32 str_len = array_cnt ? (array_cnt + 1) * 2 + strlen (arg_str) : strlen (arg_str); char *str = malloc (str_len + 1); ut32 bytes_written = snprintf (str, str_len + 1, "%s", arg_str); while (array_cnt > 0) { strcpy (str + bytes_written, "[]"); bytes_written += 2; array_cnt--; } return str; } R_API int extract_type_value(const char *arg_str, char **output) { ut8 found_one = 0, array_cnt = 0; ut32 len = 0, consumed = 0; char *str = NULL; if (!arg_str || !output) { return 0; } if (output && *output && *output != NULL) { R_FREE (*output); } while (arg_str && *arg_str && !found_one) { len = 1; // handle the end of an object switch (*arg_str) { case 'V': str = get_type_value_str ("void", array_cnt); break; case 'J': str = get_type_value_str ("long", array_cnt); array_cnt = 0; break; case 'I': str = get_type_value_str ("int", array_cnt); array_cnt = 0; break; case 'D': str = get_type_value_str ("double", array_cnt); array_cnt = 0; break; case 'F': str = get_type_value_str ("float", array_cnt); array_cnt = 0; break; case 'B': str = get_type_value_str ("byte", array_cnt); array_cnt = 0; break; case 'C': str = get_type_value_str ("char", array_cnt); array_cnt = 0; break; case 'Z': str = get_type_value_str ("boolean", array_cnt); array_cnt = 0; break; case 'S': str = get_type_value_str ("short", array_cnt); array_cnt = 0; break; case '[': array_cnt++; break; case 'L': len = r_bin_java_extract_reference_name (arg_str, &str, array_cnt); array_cnt = 0; break; case '(': str = strdup ("("); break; case ')': str = strdup (")"); break; default: return 0; } if (len < 1) { break; } consumed += len; arg_str += len; if (str) { *output = str; break; } } return consumed; } R_API RList *r_bin_java_extract_type_values(const char *arg_str) { RList *list_args = r_list_new (); if (!list_args) { return NULL; } char *str = NULL; const char *str_cur_pos = NULL; ut32 len = 0; if (!arg_str) { return list_args; } str_cur_pos = arg_str; list_args->free = free; while (str_cur_pos && *str_cur_pos) { // handle the end of an object len = extract_type_value (str_cur_pos, &str); if (len < 1) { r_list_free (list_args); return NULL; } str_cur_pos += len; r_list_append (list_args, str); str = NULL; } return list_args; } R_API int r_bin_java_is_fm_type_private(RBinJavaField *fm_type) { if (fm_type && fm_type->type == R_BIN_JAVA_FIELD_TYPE_METHOD) { return fm_type->flags & R_BIN_JAVA_METHOD_ACC_PRIVATE; } if (fm_type && fm_type->type == R_BIN_JAVA_FIELD_TYPE_FIELD) { return fm_type->flags & R_BIN_JAVA_FIELD_ACC_PRIVATE; } return 0; } R_API int r_bin_java_is_fm_type_protected(RBinJavaField *fm_type) { if (fm_type && fm_type->type == R_BIN_JAVA_FIELD_TYPE_METHOD) { return fm_type->flags & R_BIN_JAVA_METHOD_ACC_PROTECTED; } if (fm_type && fm_type->type == R_BIN_JAVA_FIELD_TYPE_FIELD) { return fm_type->flags & R_BIN_JAVA_FIELD_ACC_PROTECTED; } return 0; } R_API RList *r_bin_java_get_args(RBinJavaField *fm_type) { RList *the_list = r_bin_java_extract_type_values (fm_type->descriptor); RList *arg_list = r_list_new (); ut8 in_args = 0; RListIter *desc_iter; char *str; r_list_foreach (the_list, desc_iter, str) { if (str && *str == '(') { in_args = 1; continue; } if (str && *str == ')') { break; } if (in_args && str) { r_list_append (arg_list, strdup (str)); } } r_list_free (the_list); return arg_list; } R_API RList *r_bin_java_get_ret(RBinJavaField *fm_type) { RList *the_list = r_bin_java_extract_type_values (fm_type->descriptor); RList *ret_list = r_list_new (); ut8 in_ret = 0; RListIter *desc_iter; char *str; r_list_foreach (the_list, desc_iter, str) { if (str && *str != ')') { in_ret = 0; } if (in_ret) { r_list_append (ret_list, strdup (str)); } } r_list_free (the_list); return ret_list; } R_API char *r_bin_java_get_this_class_name(RBinJavaObj *bin) { return (bin->cf2.this_class_name ? strdup (bin->cf2.this_class_name) : strdup ("unknown")); } R_API ut16 calculate_access_value(const char *access_flags_str, RBinJavaAccessFlags *access_flags) { ut16 result = 0; ut16 size = strlen (access_flags_str) + 1; char *p_flags, *my_flags = malloc (size); RBinJavaAccessFlags *iter = NULL; if (size < 5 || !my_flags) { free (my_flags); return result; } memcpy (my_flags, access_flags_str, size); p_flags = strtok (my_flags, " "); while (p_flags && access_flags) { int idx = 0; do { iter = &access_flags[idx]; if (!iter || !iter->str) { continue; } if (iter->len > 0 && iter->len != 16) { if (!strncmp (iter->str, p_flags, iter->len)) { result |= iter->value; } } idx++; } while (access_flags[idx].str != NULL); p_flags = strtok (NULL, " "); } free (my_flags); return result; } R_API RList *retrieve_all_access_string_and_value(RBinJavaAccessFlags *access_flags) { const char *fmt = "%s = 0x%04x"; RList *result = r_list_new (); if (!result) { return NULL; } result->free = free; int i = 0; for (i = 0; access_flags[i].str != NULL; i++) { char *str = malloc (50); if (!str) { r_list_free (result); return NULL; } snprintf (str, 49, fmt, access_flags[i].str, access_flags[i].value); r_list_append (result, str); } return result; } R_API char *retrieve_access_string(ut16 flags, RBinJavaAccessFlags *access_flags) { char *outbuffer = NULL, *cur_pos = NULL; ut16 i; ut16 max_str_len = 0; for (i = 0; access_flags[i].str != NULL; i++) { if (flags & access_flags[i].value) { max_str_len += (strlen (access_flags[i].str) + 1); if (max_str_len < strlen (access_flags[i].str)) { return NULL; } } } max_str_len++; outbuffer = (char *) malloc (max_str_len); if (outbuffer) { memset (outbuffer, 0, max_str_len); cur_pos = outbuffer; for (i = 0; access_flags[i].str != NULL; i++) { if (flags & access_flags[i].value) { ut8 len = strlen (access_flags[i].str); const char *the_string = access_flags[i].str; memcpy (cur_pos, the_string, len); memcpy (cur_pos + len, " ", 1); cur_pos += len + 1; } } if (cur_pos != outbuffer) { *(cur_pos - 1) = 0; } } return outbuffer; } R_API char *retrieve_method_access_string(ut16 flags) { return retrieve_access_string (flags, METHOD_ACCESS_FLAGS); } R_API char *retrieve_field_access_string(ut16 flags) { return retrieve_access_string (flags, FIELD_ACCESS_FLAGS); } R_API char *retrieve_class_method_access_string(ut16 flags) { return retrieve_access_string (flags, CLASS_ACCESS_FLAGS); } R_API char *r_bin_java_build_obj_key(RBinJavaObj *bin) { char *jvcname = NULL; char *cname = r_bin_java_get_this_class_name (bin); ut32 class_name_len = cname ? strlen (cname) : strlen ("_unknown_"); jvcname = malloc (class_name_len + 8 + 30); if (cname) { snprintf (jvcname, class_name_len + 30, "%d.%s.class", bin->id, cname); free (cname); } else { snprintf (jvcname, class_name_len + 30, "%d._unknown_.class", bin->id); } return jvcname; } R_API bool sdb_iterate_build_list(void *user, const char *k, const char *v) { RList *bin_objs_list = (RList *) user; size_t value = (size_t) sdb_atoi (v); RBinJavaObj *bin_obj = NULL; IFDBG eprintf("Found %s == %"PFMT64x " bin_objs db\n", k, (ut64) value); if (value != 0 && value != (size_t) -1) { bin_obj = (RBinJavaObj *) value; r_list_append (bin_objs_list, bin_obj); } return true; } R_API RBinJavaCPTypeObj *r_bin_java_get_java_null_cp(void) { if (R_BIN_JAVA_NULL_TYPE_INITTED) { return &R_BIN_JAVA_NULL_TYPE; } memset (&R_BIN_JAVA_NULL_TYPE, 0, sizeof (R_BIN_JAVA_NULL_TYPE)); R_BIN_JAVA_NULL_TYPE.metas = R_NEW0 (RBinJavaMetaInfo); if (!R_BIN_JAVA_NULL_TYPE.metas) { return NULL; } memset (R_BIN_JAVA_NULL_TYPE.metas, 0, sizeof (RBinJavaMetaInfo)); R_BIN_JAVA_NULL_TYPE.metas->type_info = &R_BIN_JAVA_CP_METAS[0]; R_BIN_JAVA_NULL_TYPE.metas->ord = 0; R_BIN_JAVA_NULL_TYPE.file_offset = 0; R_BIN_JAVA_NULL_TYPE_INITTED = true; return &R_BIN_JAVA_NULL_TYPE; } R_API RBinJavaElementValueMetas *r_bin_java_get_ev_meta_from_tag(ut8 tag) { ut16 i = 0; RBinJavaElementValueMetas *res = &R_BIN_JAVA_ELEMENT_VALUE_METAS[13]; for (i = 0; i < R_BIN_JAVA_ELEMENT_VALUE_METAS_SZ; i++) { if (tag == R_BIN_JAVA_ELEMENT_VALUE_METAS[i].tag) { res = &R_BIN_JAVA_ELEMENT_VALUE_METAS[i]; break; } } return res; } R_API ut8 r_bin_java_quick_check(ut8 expected_tag, ut8 actual_tag, ut32 actual_len, const char *name) { ut8 res = 0; if (expected_tag > R_BIN_JAVA_CP_METAS_SZ) { eprintf ("Invalid tag '%d' expected 0x%02x for %s.\n", actual_tag, expected_tag, name); res = 1; } else if (expected_tag != actual_tag) { eprintf ("Invalid tag '%d' expected 0x%02x for %s.\n", actual_tag, expected_tag, name); res = 1; } else if (actual_len < R_BIN_JAVA_CP_METAS[expected_tag].len) { eprintf ("Unable to parse '%d' expected sz=0x%02x got 0x%02x for %s.\n", actual_tag, R_BIN_JAVA_CP_METAS[expected_tag].len, actual_len, name); res = 2; } return res; } R_API ut64 r_bin_java_raw_to_long(const ut8 *raw, ut64 offset) { return R_BIN_JAVA_LONG (raw, offset); } // yanked from careercup, because i am lazy: // 1) dont want to figure out how make radare use math library // 2) dont feel like figuring it out when google does it in O(1). R_API double my_pow(ut64 base, int exp) { ut8 flag = 0; ut64 res = 1; if (exp < 0) { flag = 1; exp *= -1; } while (exp) { if (exp & 1) { res *= base; } exp >>= 1; base *= base; IFDBG eprintf("Result: %"PFMT64d ", base: %"PFMT64d ", exp: %d\n", res, base, exp); } if (flag == 0) { return 1.0 * res; } return (1.0 / res); } R_API double r_bin_java_raw_to_double(const ut8 *raw, ut64 offset) { ut64 bits = R_BIN_JAVA_LONG (raw, offset); int s = ((bits >> 63) == 0) ? 1 : -1; int e = (int) ((bits >> 52) & 0x7ffL); long m = (e == 0) ? (bits & 0xfffffffffffffLL) << 1 : (bits & 0xfffffffffffffLL) | 0x10000000000000LL; double res = 0.0; IFDBG eprintf("Convert Long to Double: %08"PFMT64x "\n", bits); if (bits == 0x7ff0000000000000LL) { return INFINITY; } if (bits == 0xfff0000000000000LL) { return -INFINITY; } if (0x7ff0000000000001LL <= bits && bits <= 0x7fffffffffffffffLL) { return NAN; } if (0xfff0000000000001LL <= bits && bits <= 0xffffffffffffffffLL) { return NAN; } res = s * m * my_pow (2, e - 1075);// XXXX TODO Get double to work correctly here IFDBG eprintf(" High-bytes = %02x %02x %02x %02x\n", raw[0], raw[1], raw[2], raw[3]); IFDBG eprintf(" Low-bytes = %02x %02x %02x %02x\n", raw[4], raw[5], raw[6], raw[7]); IFDBG eprintf("Convert Long to Double s: %d, m: 0x%08lx, e: 0x%08x, res: %f\n", s, m, e, res); return res; } R_API RBinJavaField *r_bin_java_read_next_method(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { ut32 i, idx; const ut8 *f_buf = buf + offset; ut64 adv = 0; RBinJavaCPTypeObj *item = NULL; if (!bin || offset + 8 >= len) { return NULL; } RBinJavaField *method = (RBinJavaField *) R_NEW0 (RBinJavaField); if (!method) { eprintf ("Unable to allocate memory for method information\n"); return NULL; } method->metas = (RBinJavaMetaInfo *) R_NEW0 (RBinJavaMetaInfo); if (!method->metas) { eprintf ("Unable to allocate memory for meta information\n"); free (method); return NULL; } method->file_offset = offset; method->flags = R_BIN_JAVA_USHORT (f_buf, 0); method->flags_str = retrieve_method_access_string (method->flags); // need to subtract 1 for the idx method->name_idx = R_BIN_JAVA_USHORT (f_buf, 2); method->descriptor_idx = R_BIN_JAVA_USHORT (f_buf, 4); method->attr_count = R_BIN_JAVA_USHORT (f_buf, 6); method->attributes = r_list_newf (r_bin_java_attribute_free); method->type = R_BIN_JAVA_FIELD_TYPE_METHOD; method->metas->ord = bin->method_idx; adv += 8; idx = method->name_idx; item = r_bin_java_get_item_from_bin_cp_list (bin, idx); method->name = r_bin_java_get_utf8_from_bin_cp_list (bin, (ut32) (method->name_idx)); IFDBG eprintf("Method name_idx: %d, which is: ord: %d, name: %s, value: %s\n", idx, item->metas->ord, ((RBinJavaCPTypeMetas *)item->metas->type_info)->name, method->name); if (!method->name) { method->name = (char *) malloc (21); snprintf ((char *) method->name, 20, "sym.method_%08x", method->metas->ord); IFDBG eprintf("r_bin_java_read_next_method: Unable to find the name for 0x%02x index.\n", method->name_idx); } idx = method->descriptor_idx; item = r_bin_java_get_item_from_bin_cp_list (bin, idx); method->descriptor = r_bin_java_get_utf8_from_bin_cp_list (bin, (ut32) method->descriptor_idx); IFDBG eprintf("Method descriptor_idx: %d, which is: ord: %d, name: %s, value: %s\n", idx, item->metas->ord, ((RBinJavaCPTypeMetas *)item->metas->type_info)->name, method->descriptor); if (!method->descriptor) { method->descriptor = r_str_dup (NULL, "NULL"); IFDBG eprintf("r_bin_java_read_next_method: Unable to find the descriptor for 0x%02x index.\n", method->descriptor_idx); } IFDBG eprintf("Looking for a NameAndType CP with name_idx: %d descriptor_idx: %d\n", method->name_idx, method->descriptor_idx); method->field_ref_cp_obj = r_bin_java_find_cp_ref_info_from_name_and_type (bin, method->name_idx, method->descriptor_idx); if (method->field_ref_cp_obj) { IFDBG eprintf("Found the obj.\n"); item = r_bin_java_get_item_from_bin_cp_list (bin, method->field_ref_cp_obj->info.cp_method.class_idx); IFDBG eprintf("Method class reference value: %d, which is: ord: %d, name: %s\n", method->field_ref_cp_obj->info.cp_method.class_idx, item->metas->ord, ((RBinJavaCPTypeMetas *)item->metas->type_info)->name); method->class_name = r_bin_java_get_item_name_from_bin_cp_list (bin, item); IFDBG eprintf("Method requesting ref_cp_obj the following which is: ord: %d, name: %s\n", method->field_ref_cp_obj->metas->ord, ((RBinJavaCPTypeMetas *)method->field_ref_cp_obj->metas->type_info)->name); IFDBG eprintf("MethodRef class name resolves to: %s\n", method->class_name); if (!method->class_name) { method->class_name = r_str_dup (NULL, "NULL"); } } else { // XXX - default to this class? method->field_ref_cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, bin->cf2.this_class); method->class_name = r_bin_java_get_item_name_from_bin_cp_list (bin, method->field_ref_cp_obj); } IFDBG eprintf("Parsing %s(%s)\n", method->name, method->descriptor); if (method->attr_count > 0) { method->attr_offset = adv + offset; RBinJavaAttrInfo *attr = NULL; for (i = 0; i < method->attr_count; i++) { attr = r_bin_java_read_next_attr (bin, adv + offset, buf, len); if (!attr) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Method Attribute: %d.\n", i); break; } if ((r_bin_java_get_attr_type_by_name (attr->name))->type == R_BIN_JAVA_ATTR_TYPE_CODE_ATTR) { // This is necessary for determing the appropriate number of bytes when readin // uoffset, ustack, ulocalvar values bin->cur_method_code_length = attr->info.code_attr.code_length; bin->offset_sz = 2;// (attr->info.code_attr.code_length > 65535) ? 4 : 2; bin->ustack_sz = 2;// (attr->info.code_attr.max_stack > 65535) ? 4 : 2; bin->ulocalvar_sz = 2;// (attr->info.code_attr.max_locals > 65535) ? 4 : 2; } IFDBG eprintf("Parsing @ 0x%"PFMT64x " (%s) = 0x%"PFMT64x " bytes\n", attr->file_offset, attr->name, attr->size); r_list_append (method->attributes, attr); adv += attr->size; if (adv + offset >= len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Method Attribute: %d.\n", i); break; } } } method->size = adv; // reset after parsing the method attributes IFDBG eprintf("Parsing @ 0x%"PFMT64x " %s(%s) = 0x%"PFMT64x " bytes\n", method->file_offset, method->name, method->descriptor, method->size); return method; } R_API RBinJavaField *r_bin_java_read_next_field(RBinJavaObj *bin, const ut64 offset, const ut8 *buffer, const ut64 len) { RBinJavaAttrInfo *attr; ut32 i, idx; ut8 buf[8]; RBinJavaCPTypeObj *item = NULL; const ut8 *f_buf = buffer + offset; ut64 adv = 0; if (!bin || offset + 8 >= len) { return NULL; } RBinJavaField *field = (RBinJavaField *) R_NEW0 (RBinJavaField); if (!field) { eprintf ("Unable to allocate memory for field information\n"); return NULL; } field->metas = (RBinJavaMetaInfo *) R_NEW0 (RBinJavaMetaInfo); if (!field->metas) { eprintf ("Unable to allocate memory for meta information\n"); free (field); return NULL; } memcpy (buf, f_buf, 8); field->file_offset = offset; field->flags = R_BIN_JAVA_USHORT (buf, 0); field->flags_str = retrieve_field_access_string (field->flags); field->name_idx = R_BIN_JAVA_USHORT (buf, 2); field->descriptor_idx = R_BIN_JAVA_USHORT (buf, 4); field->attr_count = R_BIN_JAVA_USHORT (buf, 6); field->attributes = r_list_newf (r_bin_java_attribute_free); field->type = R_BIN_JAVA_FIELD_TYPE_FIELD; adv += 8; field->metas->ord = bin->field_idx; idx = field->name_idx; item = r_bin_java_get_item_from_bin_cp_list (bin, idx); field->name = r_bin_java_get_utf8_from_bin_cp_list (bin, (ut32) (field->name_idx)); IFDBG eprintf("Field name_idx: %d, which is: ord: %d, name: %s, value: %s\n", idx, item->metas->ord, ((RBinJavaCPTypeMetas *)item->metas->type_info)->name, field->name); if (!field->name) { field->name = (char *) malloc (21); snprintf ((char *) field->name, 20, "sym.field_%08x", field->metas->ord); IFDBG eprintf("r_bin_java_read_next_field: Unable to find the name for 0x%02x index.\n", field->name_idx); } idx = field->descriptor_idx; item = r_bin_java_get_item_from_bin_cp_list (bin, idx); field->descriptor = r_bin_java_get_utf8_from_bin_cp_list (bin, (ut32) field->descriptor_idx); IFDBG eprintf("Field descriptor_idx: %d, which is: ord: %d, name: %s, value: %s\n", idx, item->metas->ord, ((RBinJavaCPTypeMetas *)item->metas->type_info)->name, field->descriptor); if (!field->descriptor) { field->descriptor = r_str_dup (NULL, "NULL"); IFDBG eprintf("r_bin_java_read_next_field: Unable to find the descriptor for 0x%02x index.\n", field->descriptor_idx); } IFDBG eprintf("Looking for a NameAndType CP with name_idx: %d descriptor_idx: %d\n", field->name_idx, field->descriptor_idx); field->field_ref_cp_obj = r_bin_java_find_cp_ref_info_from_name_and_type (bin, field->name_idx, field->descriptor_idx); if (field->field_ref_cp_obj) { IFDBG eprintf("Found the obj.\n"); item = r_bin_java_get_item_from_bin_cp_list (bin, field->field_ref_cp_obj->info.cp_field.class_idx); IFDBG eprintf("Field class reference value: %d, which is: ord: %d, name: %s\n", field->field_ref_cp_obj->info.cp_field.class_idx, item->metas->ord, ((RBinJavaCPTypeMetas *)item->metas->type_info)->name); field->class_name = r_bin_java_get_item_name_from_bin_cp_list (bin, item); IFDBG eprintf("Field requesting ref_cp_obj the following which is: ord: %d, name: %s\n", field->field_ref_cp_obj->metas->ord, ((RBinJavaCPTypeMetas *)field->field_ref_cp_obj->metas->type_info)->name); IFDBG eprintf("FieldRef class name resolves to: %s\n", field->class_name); if (!field->class_name) { field->class_name = r_str_dup (NULL, "NULL"); } } else { // XXX - default to this class? field->field_ref_cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, bin->cf2.this_class); field->class_name = r_bin_java_get_item_name_from_bin_cp_list (bin, field->field_ref_cp_obj); } IFDBG eprintf("Parsing %s(%s)", field->name, field->descriptor); if (field->attr_count > 0) { field->attr_offset = adv + offset; for (i = 0; i < field->attr_count && offset + adv < len; i++) { attr = r_bin_java_read_next_attr (bin, offset + adv, buffer, len); if (!attr) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Field Attribute: %d.\n", i); free (field->metas); free (field); return NULL; } if ((r_bin_java_get_attr_type_by_name (attr->name))->type == R_BIN_JAVA_ATTR_TYPE_CODE_ATTR) { // This is necessary for determing the appropriate number of bytes when readin // uoffset, ustack, ulocalvar values bin->cur_method_code_length = attr->info.code_attr.code_length; bin->offset_sz = 2;// (attr->info.code_attr.code_length > 65535) ? 4 : 2; bin->ustack_sz = 2;// (attr->info.code_attr.max_stack > 65535) ? 4 : 2; bin->ulocalvar_sz = 2;// (attr->info.code_attr.max_locals > 65535) ? 4 : 2; } r_list_append (field->attributes, attr); adv += attr->size; if (adv + offset >= len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Field Attribute: %d.\n", i); r_bin_java_fmtype_free (field); return NULL; } } } field->size = adv; return field; } R_API RBinJavaCPTypeObj *r_bin_java_clone_cp_idx(RBinJavaObj *bin, ut32 idx) { RBinJavaCPTypeObj *obj = NULL; if (bin) { obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); } return r_bin_java_clone_cp_item (obj); } R_API RBinJavaCPTypeObj *r_bin_java_clone_cp_item(RBinJavaCPTypeObj *obj) { RBinJavaCPTypeObj *clone_obj = NULL; if (obj == NULL) { return clone_obj; } clone_obj = R_NEW0 (RBinJavaCPTypeObj); if (clone_obj) { memcpy (clone_obj, obj, sizeof (RBinJavaCPTypeObj)); clone_obj->metas = (RBinJavaMetaInfo *) R_NEW0 (RBinJavaMetaInfo); clone_obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[clone_obj->tag]; clone_obj->name = strdup (obj->name? obj->name: "unk"); if (obj->tag == R_BIN_JAVA_CP_UTF8) { clone_obj->info.cp_utf8.bytes = (ut8 *) malloc (obj->info.cp_utf8.length + 1); if (clone_obj->info.cp_utf8.bytes) { memcpy (clone_obj->info.cp_utf8.bytes, obj->info.cp_utf8.bytes, clone_obj->info.cp_utf8.length); } else { // TODO: eprintf allocation error } } } return clone_obj; } R_API RBinJavaCPTypeObj *r_bin_java_read_next_constant_pool_item(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, ut64 len) { RBinJavaCPTypeMetas *java_constant_info = NULL; ut8 tag = 0; ut64 buf_sz = 0; ut8 *cp_buf = NULL; ut32 str_len = 0; RBinJavaCPTypeObj *java_obj = NULL; tag = buf[offset]; if (tag > R_BIN_JAVA_CP_METAS_SZ) { eprintf ("Invalid tag '%d' at offset 0x%08"PFMT64x "\n", tag, (ut64) offset); return NULL; #if 0 java_obj = r_bin_java_unknown_cp_new (bin, &tag, 1); if (java_obj != NULL && java_obj->metas != NULL) { java_obj->file_offset = offset; java_obj->loadaddr = bin->loadaddr; } return NULL; // early error to avoid future overflows // return java_obj; #endif } java_constant_info = &R_BIN_JAVA_CP_METAS[tag]; if (java_constant_info->tag == 0 || java_constant_info->tag == 2) { return java_obj; } buf_sz += java_constant_info->len; if (java_constant_info->tag == 1) { if (offset + 32 < len) { str_len = R_BIN_JAVA_USHORT (buf, offset + 1); buf_sz += str_len; } else { return NULL; } } cp_buf = calloc (buf_sz, 1); if (!cp_buf) { return java_obj; } if (offset + buf_sz < len) { memcpy (cp_buf, (ut8 *) buf + offset, buf_sz); IFDBG eprintf("Parsed the tag '%d':%s and create object from offset 0x%08"PFMT64x ".\n", tag, R_BIN_JAVA_CP_METAS[tag].name, offset); java_obj = (*java_constant_info->allocs->new_obj)(bin, cp_buf, buf_sz); if (java_obj != NULL && java_obj->metas != NULL) { java_obj->file_offset = offset; // IFDBG eprintf ("java_obj->file_offset = 0x%08"PFMT64x".\n",java_obj->file_offset); } else if (!java_obj) { eprintf ("Unable to parse the tag '%d' and create valid object.\n", tag); } else if (!java_obj->metas) { eprintf ("Unable to parse the tag '%d' and create valid object.\n", tag); } else { eprintf ("Failed to set the java_obj->metas-file_offset for '%d' offset is(0x%08"PFMT64x ").\n", tag, offset); } } free (cp_buf); return java_obj; } R_API RBinJavaInterfaceInfo *r_bin_java_read_next_interface_item(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { ut8 idx[2] = { 0 }; RBinJavaInterfaceInfo *ifobj; const ut8 *if_buf = buf + offset; if (offset + 2 >= len) { return NULL; } memcpy (&idx, if_buf, 2); ifobj = r_bin_java_interface_new (bin, if_buf, len - offset); if (ifobj) { ifobj->file_offset = offset; } return ifobj; } // R_API void addrow (RBinJavaObj *bin, int addr, int line) { // int n = bin->lines.count++; //// XXX. possible memleak // bin->lines.addr = realloc (bin->lines.addr, sizeof (int)*n+1); // bin->lines.addr[n] = addr; // bin->lines.line = realloc (bin->lines.line, sizeof (int)*n+1); // bin->lines.line[n] = line; // } // R_API struct r_bin_java_cp_item_t* r_bin_java_get_item_from_cp_CP(RBinJavaObj *bin, int i) { // return (i<0||i>bin->cf.cp_count)? &cp_null_item: &bin->cp_items[i]; // } R_API char *r_bin_java_get_utf8_from_bin_cp_list(RBinJavaObj *bin, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: new char* for caller to free. */ if (bin == NULL) { return NULL; } return r_bin_java_get_utf8_from_cp_item_list (bin->cp_list, idx); } R_API ut32 r_bin_java_get_utf8_len_from_bin_cp_list(RBinJavaObj *bin, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: new char* for caller to free. */ if (bin == NULL) { return 0; } return r_bin_java_get_utf8_len_from_cp_item_list (bin->cp_list, idx); } R_API char *r_bin_java_get_name_from_bin_cp_list(RBinJavaObj *bin, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: new char* for caller to free. */ if (bin == NULL) { return NULL; } return r_bin_java_get_name_from_cp_item_list (bin->cp_list, idx); } R_API char *r_bin_java_get_desc_from_bin_cp_list(RBinJavaObj *bin, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: new char* for caller to free. */ if (bin == NULL) { return NULL; } return r_bin_java_get_desc_from_cp_item_list (bin->cp_list, idx); } R_API RBinJavaCPTypeObj *r_bin_java_get_item_from_bin_cp_list(RBinJavaObj *bin, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: RBinJavaObj* (user does NOT free). */ if (bin == NULL) { return NULL; } if (idx > bin->cp_count || idx == 0) { return r_bin_java_get_java_null_cp (); } return r_bin_java_get_item_from_cp_item_list (bin->cp_list, idx); } R_API char *r_bin_java_get_item_name_from_bin_cp_list(RBinJavaObj *bin, RBinJavaCPTypeObj *obj) { char *res = NULL; /* Given a constant poool object Class, FieldRef, MethodRef, or InterfaceMethodRef return the actual descriptor string. @param cp_list: RList of RBinJavaCPTypeObj * @param obj object to look up the name for @rvalue char* (user frees) or NULL */ if (bin && obj) { res = r_bin_java_get_item_name_from_cp_item_list ( bin->cp_list, obj, MAX_CPITEMS); } return res; } R_API char *r_bin_java_get_item_desc_from_bin_cp_list(RBinJavaObj *bin, RBinJavaCPTypeObj *obj) { /* Given a constant poool object Class, FieldRef, MethodRef, or InterfaceMethodRef return the actual descriptor string. @param cp_list: RList of RBinJavaCPTypeObj * @param obj object to look up the name for @rvalue char* (user frees) or NULL */ return bin? r_bin_java_get_item_desc_from_cp_item_list (bin->cp_list, obj, MAX_CPITEMS): NULL; } R_API char *r_bin_java_get_utf8_from_cp_item_list(RList *cp_list, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: new char* for caller to free. */ char *value = NULL; RListIter *iter; if (!cp_list) { return NULL; } RBinJavaCPTypeObj *item = (RBinJavaCPTypeObj *) r_list_get_n (cp_list, idx); if (item && item->tag == R_BIN_JAVA_CP_UTF8 && item->metas->ord == idx) { value = convert_string ((const char *) item->info.cp_utf8.bytes, item->info.cp_utf8.length); } if (!value) { r_list_foreach (cp_list, iter, item) { if (item && (item->tag == R_BIN_JAVA_CP_UTF8) && item->metas->ord == idx) { value = convert_string ((const char *) item->info.cp_utf8.bytes, item->info.cp_utf8.length); break; } } } return value; } R_API ut32 r_bin_java_get_utf8_len_from_cp_item_list(RList *cp_list, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. If the idx not found by directly going to the list index, the list will be walked and then the IDX will be checked. rvalue: new ut32 . */ ut32 value = -1; RListIter *iter; if (!cp_list) { return 0; } RBinJavaCPTypeObj *item = (RBinJavaCPTypeObj *) r_list_get_n (cp_list, idx); if (item && (item->tag == R_BIN_JAVA_CP_UTF8) && item->metas->ord == idx) { value = item->info.cp_utf8.length; } if (value == -1) { r_list_foreach (cp_list, iter, item) { if (item && (item->tag == R_BIN_JAVA_CP_UTF8) && item->metas->ord == idx) { value = item->info.cp_utf8.length; break; } } } return value; } R_API RBinJavaCPTypeObj *r_bin_java_get_item_from_cp_item_list(RList *cp_list, ut64 idx) { /* Search through the Constant Pool list for the given CP Index. rvalue: RBinJavaObj * */ RBinJavaCPTypeObj *item = NULL; if (cp_list == NULL) { return NULL; } item = (RBinJavaCPTypeObj *) r_list_get_n (cp_list, idx); return item; } R_API char *r_bin_java_get_item_name_from_cp_item_list(RList *cp_list, RBinJavaCPTypeObj *obj, int depth) { /* Given a constant poool object Class, FieldRef, MethodRef, or InterfaceMethodRef return the actual descriptor string. @param cp_list: RList of RBinJavaCPTypeObj * @param obj object to look up the name for @rvalue ut8* (user frees) or NULL */ if (obj == NULL || cp_list == NULL || depth < 0) { return NULL; } switch (obj->tag) { case R_BIN_JAVA_CP_NAMEANDTYPE: return r_bin_java_get_utf8_from_cp_item_list ( cp_list, obj->info.cp_name_and_type.name_idx); case R_BIN_JAVA_CP_CLASS: return r_bin_java_get_utf8_from_cp_item_list ( cp_list, obj->info.cp_class.name_idx); // XXX - Probably not good form, but they are the same memory structure case R_BIN_JAVA_CP_FIELDREF: case R_BIN_JAVA_CP_INTERFACEMETHOD_REF: case R_BIN_JAVA_CP_METHODREF: obj = r_bin_java_get_item_from_cp_item_list ( cp_list, obj->info.cp_method.name_and_type_idx); return r_bin_java_get_item_name_from_cp_item_list ( cp_list, obj, depth - 1); default: return NULL; case 0: IFDBG eprintf("Invalid 0 tag in the constant pool\n"); return NULL; } return NULL; } R_API char *r_bin_java_get_name_from_cp_item_list(RList *cp_list, ut64 idx) { /* Given a constant poool object Class, FieldRef, MethodRef, or InterfaceMethodRef return the actual descriptor string. @param cp_list: RList of RBinJavaCPTypeObj * @param obj object to look up the name for @rvalue ut8* (user frees) or NULL */ RBinJavaCPTypeObj *obj = r_bin_java_get_item_from_cp_item_list ( cp_list, idx); if (obj && cp_list) { return r_bin_java_get_item_name_from_cp_item_list ( cp_list, obj, MAX_CPITEMS); } return NULL; } R_API char *r_bin_java_get_item_desc_from_cp_item_list(RList *cp_list, RBinJavaCPTypeObj *obj, int depth) { /* Given a constant poool object FieldRef, MethodRef, or InterfaceMethodRef return the actual descriptor string. @rvalue ut8* (user frees) or NULL */ if (!obj || !cp_list || depth < 0) { return NULL; } switch (obj->tag) { case R_BIN_JAVA_CP_NAMEANDTYPE: return r_bin_java_get_utf8_from_cp_item_list (cp_list, obj->info.cp_name_and_type.descriptor_idx); // XXX - Probably not good form, but they are the same memory structure case R_BIN_JAVA_CP_FIELDREF: case R_BIN_JAVA_CP_INTERFACEMETHOD_REF: case R_BIN_JAVA_CP_METHODREF: obj = r_bin_java_get_item_from_cp_item_list (cp_list, obj->info.cp_method.name_and_type_idx); return r_bin_java_get_item_desc_from_cp_item_list ( cp_list, obj, depth - 1); default: return NULL; } return NULL; } R_API char *r_bin_java_get_desc_from_cp_item_list(RList *cp_list, ut64 idx) { /* Given a constant poool object FieldRef, MethodRef, or InterfaceMethodRef return the actual descriptor string. @rvalue ut8* (user frees) or NULL */ RBinJavaCPTypeObj *obj = r_bin_java_get_item_from_cp_item_list (cp_list, idx); if (cp_list == NULL) { return NULL; } return r_bin_java_get_item_desc_from_cp_item_list (cp_list, obj, MAX_CPITEMS); } R_API RBinJavaAttrInfo *r_bin_java_get_method_code_attribute(const RBinJavaField *method) { /* Search through a methods attributes and return the code attr. rvalue: RBinJavaAttrInfo* if found otherwise NULL. */ RBinJavaAttrInfo *res = NULL, *attr = NULL; RListIter *iter; if (method) { r_list_foreach (method->attributes, iter, attr) { if (attr && (attr->type == R_BIN_JAVA_ATTR_TYPE_CODE_ATTR)) { res = attr; break; } } } return res; } R_API RBinJavaAttrInfo *r_bin_java_get_attr_from_field(RBinJavaField *field, R_BIN_JAVA_ATTR_TYPE attr_type, ut32 pos) { /* Search through the Attribute list for the given type starting at position pos. rvalue: NULL or the first occurrence of attr_type after pos */ RBinJavaAttrInfo *attr = NULL, *item; RListIter *iter; ut32 i = 0; if (field) { r_list_foreach (field->attributes, iter, item) { // Note the increment happens after the comparison if ((i++) >= pos) { if (item && (item->type == attr_type)) { attr = item; break; } } } } return attr; } R_API ut8 *r_bin_java_get_attr_buf(RBinJavaObj *bin, ut64 sz, const ut64 offset, const ut8 *buf, const ut64 len) { ut8 *attr_buf = NULL; int pending = len - offset; const ut8 *a_buf = offset + buf; attr_buf = (ut8 *) calloc (pending + 1, 1); if (attr_buf == NULL) { eprintf ("Unable to allocate enough bytes (0x%04"PFMT64x ") to read in the attribute.\n", sz); return attr_buf; } memcpy (attr_buf, a_buf, pending); // sz+1); return attr_buf; } R_API RBinJavaAttrInfo *r_bin_java_default_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { // NOTE: this function receives the buffer offset in the original buffer, // but the buffer is already point to that particular offset. // XXX - all the code that relies on this function should probably be modified // so that the original buffer pointer is passed in and then the buffer+buf_offset // points to the correct location. RBinJavaAttrInfo *attr = R_NEW0 (RBinJavaAttrInfo); if (!attr) { return NULL; } RBinJavaAttrMetas *type_info = NULL; attr->metas = R_NEW0 (RBinJavaMetaInfo); if (attr->metas == NULL) { free (attr); return NULL; } attr->is_attr_in_old_format = r_bin_java_is_old_format(bin); attr->file_offset = buf_offset; attr->name_idx = R_BIN_JAVA_USHORT (buffer, 0); attr->length = R_BIN_JAVA_UINT (buffer, 2); attr->size = R_BIN_JAVA_UINT (buffer, 2) + 6; attr->name = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, attr->name_idx); if (attr->name == NULL) { // Something bad has happened attr->name = r_str_dup (NULL, "NULL"); eprintf ("r_bin_java_default_attr_new: Unable to find the name for %d index.\n", attr->name_idx); } type_info = r_bin_java_get_attr_type_by_name (attr->name); attr->metas->ord = (R_BIN_JAVA_GLOBAL_BIN->attr_idx++); attr->metas->type_info = (void *) type_info; // IFDBG eprintf (" Addrs for type_info [tag=%d]: 0x%08"PFMT64x"\n", type_val, &attr->metas->type_info); return attr; } R_API RBinJavaAttrMetas *r_bin_java_get_attr_type_by_name(const char *name) { // TODO: use sdb/hashtable here int i; for (i = 0; i < RBIN_JAVA_ATTRS_METAS_SZ; i++) { if (!strcmp ((const char *) name, RBIN_JAVA_ATTRS_METAS[i].name)) { return &RBIN_JAVA_ATTRS_METAS[i]; } } return &RBIN_JAVA_ATTRS_METAS[R_BIN_JAVA_ATTR_TYPE_UNKNOWN_ATTR]; } R_API RBinJavaAttrInfo *r_bin_java_read_next_attr(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 buf_len) { RBinJavaAttrInfo *attr = NULL; ut32 sz = 0; ut8 *buffer = NULL; const ut8 *a_buf = offset + buf; ut8 attr_idx_len = 6; if (offset + 6 > buf_len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile in Attribute offset " "(0x%"PFMT64x ") > len of remaining bytes (0x%"PFMT64x ").\n", offset, buf_len); return NULL; } // ut16 attr_idx, ut32 length of attr. sz = R_BIN_JAVA_UINT (a_buf, 2) + attr_idx_len; // r_bin_java_read_int (bin, buf_offset+2) + attr_idx_len; if (sz + offset > buf_len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile in Attribute len " "(0x%x) + offset (0x%"PFMT64x ") exceeds length of buffer (0x%"PFMT64x ").\n", sz, offset, buf_len); return NULL; } // when reading the attr bytes, need to also // include the initial 6 bytes, which // are not included in the attribute length // , // sz, buf_offset, buf_offset+sz); buffer = r_bin_java_get_attr_buf (bin, sz, offset, buf, buf_len); // printf ("%d %d %d\n", sz, buf_len, offset); if (offset < buf_len) { attr = r_bin_java_read_next_attr_from_buffer (bin, buffer, buf_len - offset, offset); free (buffer); if (!attr) { return NULL; } attr->size = sz; } else { free (buffer); eprintf ("IS OOB\n"); } return attr; } R_API RBinJavaAttrInfo *r_bin_java_read_next_attr_from_buffer(RBinJavaObj *bin, ut8 *buffer, st64 sz, st64 buf_offset) { RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; ut16 name_idx; st64 nsz; if (!buffer || ((int) sz) < 4 || buf_offset < 0) { eprintf ("r_bin_Java_read_next_attr_from_buffer: invalid buffer size %d\n", (int) sz); return NULL; } name_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; nsz = R_BIN_JAVA_UINT (buffer, offset); // DEAD INCREMENT offset += 4; char *name = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, name_idx); if (!name) { name = strdup ("unknown"); } IFDBG eprintf("r_bin_java_read_next_attr: name_idx = %d is %s\n", name_idx, name); RBinJavaAttrMetas *type_info = r_bin_java_get_attr_type_by_name (name); if (type_info) { IFDBG eprintf("Typeinfo: %s, was %s\n", type_info->name, name); // printf ("SZ %d %d %d\n", nsz, sz, buf_offset); if (nsz > sz) { free (name); return NULL; } if ((attr = type_info->allocs->new_obj (bin, buffer, nsz, buf_offset))) { attr->metas->ord = (R_BIN_JAVA_GLOBAL_BIN->attr_idx++); } } else { eprintf ("r_bin_java_read_next_attr_from_buffer: Cannot find type_info for %s\n", name); } free (name); return attr; } R_API ut64 r_bin_java_read_class_file2(RBinJavaObj *bin, const ut64 offset, const ut8 *obuf, ut64 len) { const ut8 *cf2_buf = obuf + offset; RBinJavaCPTypeObj *this_class_cp_obj = NULL; IFDBG eprintf("\n0x%"PFMT64x " Offset before reading the cf2 structure\n", offset); /* Reading the following fields: ut16 access_flags; ut16 this_class; ut16 super_class; */ if (cf2_buf + 6 > obuf + len) { return 0; } bin->cf2.cf2_size = 6; bin->cf2.access_flags = R_BIN_JAVA_USHORT (cf2_buf, 0); bin->cf2.this_class = R_BIN_JAVA_USHORT (cf2_buf, 2); bin->cf2.super_class = R_BIN_JAVA_USHORT (cf2_buf, 4); free (bin->cf2.flags_str); free (bin->cf2.this_class_name); bin->cf2.flags_str = retrieve_class_method_access_string (bin->cf2.access_flags); this_class_cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, bin->cf2.this_class); bin->cf2.this_class_name = r_bin_java_get_item_name_from_bin_cp_list (bin, this_class_cp_obj); IFDBG eprintf("This class flags are: %s\n", bin->cf2.flags_str); return bin->cf2.cf2_size; } R_API ut64 r_bin_java_parse_cp_pool(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { int ord = 0; ut64 adv = 0; RBinJavaCPTypeObj *obj = NULL; const ut8 *cp_buf = buf + offset; r_list_free (bin->cp_list); bin->cp_list = r_list_newf (r_bin_java_constant_pool); bin->cp_offset = offset; memcpy ((char *) &bin->cp_count, cp_buf, 2); bin->cp_count = R_BIN_JAVA_USHORT (cp_buf, 0) - 1; adv += 2; IFDBG eprintf("ConstantPoolCount %d\n", bin->cp_count); r_list_append (bin->cp_list, r_bin_java_get_java_null_cp ()); for (ord = 1, bin->cp_idx = 0; bin->cp_idx < bin->cp_count && adv < len; ord++, bin->cp_idx++) { obj = r_bin_java_read_next_constant_pool_item (bin, offset + adv, buf, len); if (obj) { // IFDBG eprintf ("SUCCESS Read ConstantPoolItem %d\n", i); obj->metas->ord = ord; obj->idx = ord; r_list_append (bin->cp_list, obj); if (obj->tag == R_BIN_JAVA_CP_LONG || obj->tag == R_BIN_JAVA_CP_DOUBLE) { // i++; ord++; bin->cp_idx++; r_list_append (bin->cp_list, &R_BIN_JAVA_NULL_TYPE); } IFDBG ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->print_summary (obj); adv += ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->calc_size (obj); if (offset + adv > len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Constant Pool Object: %d.\n", ord); break; } } else { IFDBG eprintf("Failed to read ConstantPoolItem %d\n", bin->cp_idx); break; } } // Update the imports r_bin_java_set_imports (bin); bin->cp_size = adv; return bin->cp_size; } R_API ut64 r_bin_java_parse_interfaces(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { int i = 0; ut64 adv = 0; RBinJavaInterfaceInfo *interfaces_obj; const ut8 *if_buf = buf + offset; bin->cp_offset = offset; bin->interfaces_offset = offset; r_list_free (bin->interfaces_list); bin->interfaces_list = r_list_newf (r_bin_java_interface_free); if (offset + 2 > len) { bin->interfaces_size = 0; return 0; } bin->interfaces_count = R_BIN_JAVA_USHORT (if_buf, 0); adv += 2; IFDBG eprintf("Interfaces count: %d\n", bin->interfaces_count); if (bin->interfaces_count > 0) { for (i = 0; i < bin->interfaces_count; i++) { interfaces_obj = r_bin_java_read_next_interface_item (bin, offset + adv, buf, len); if (interfaces_obj) { r_list_append (bin->interfaces_list, interfaces_obj); adv += interfaces_obj->size; if (offset + adv > len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Interface: %d.\n", i); break; } } else { break; } } } bin->interfaces_size = adv; return adv; } R_API ut64 r_bin_java_parse_fields(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { int i = 0; ut64 adv = 0; RBinJavaField *field; const ut8 *fm_buf = buf + offset; r_list_free (bin->fields_list); bin->fields_list = r_list_newf (r_bin_java_fmtype_free); bin->fields_offset = offset; if (offset + 2 >= len) { return UT64_MAX; } bin->fields_count = R_BIN_JAVA_USHORT (fm_buf, 0); adv += 2; IFDBG eprintf("Fields count: %d 0x%"PFMT64x "\n", bin->fields_count, bin->fields_offset); if (bin->fields_count > 0) { for (i = 0; i < bin->fields_count; i++, bin->field_idx++) { field = r_bin_java_read_next_field (bin, offset + adv, buf, len); if (field) { adv += field->size; r_list_append (bin->fields_list, field); IFDBG r_bin_java_print_field_summary(field); if (adv + offset > len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Field: %d.\n", i); break; } } else { IFDBG eprintf("Failed to read Field %d\n", i); break; } } } bin->fields_size = adv; return adv; } R_API ut64 r_bin_java_parse_attrs(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { int i = 0; ut64 adv = 0; const ut8 *a_buf = buf + offset; if (offset + 2 >= len) { // Check if we can read that USHORT return UT64_MAX; } r_list_free (bin->attrs_list); bin->attrs_list = r_list_newf (r_bin_java_attribute_free); bin->attrs_offset = offset; bin->attrs_count = R_BIN_JAVA_USHORT (a_buf, adv); adv += 2; if (bin->attrs_count > 0) { for (i = 0; i < bin->attrs_count; i++, bin->attr_idx++) { RBinJavaAttrInfo *attr = r_bin_java_read_next_attr (bin, offset + adv, buf, len); if (!attr) { // eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Attribute: %d.\n", i); break; } r_list_append (bin->attrs_list, attr); adv += attr->size; if (adv + offset >= len) { // eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Attribute: %d.\n", i); break; } } } bin->attrs_size = adv; return adv; } R_API ut64 r_bin_java_parse_methods(RBinJavaObj *bin, const ut64 offset, const ut8 *buf, const ut64 len) { int i = 0; ut64 adv = 0; RBinJavaField *method; const ut8 *fm_buf = buf + offset; r_list_free (bin->methods_list); bin->methods_list = r_list_newf (r_bin_java_fmtype_free); if (offset + 2 >= len) { return 0LL; } bin->methods_offset = offset; bin->methods_count = R_BIN_JAVA_USHORT (fm_buf, 0); adv += 2; IFDBG eprintf("Methods count: %d 0x%"PFMT64x "\n", bin->methods_count, bin->methods_offset); bin->main = NULL; bin->entrypoint = NULL; bin->main_code_attr = NULL; bin->entrypoint_code_attr = NULL; for (i = 0; i < bin->methods_count; i++, bin->method_idx++) { method = r_bin_java_read_next_method (bin, offset + adv, buf, len); if (method) { adv += method->size; r_list_append (bin->methods_list, method); } // Update Main, Init, or Class Init if (method && !strcmp ((const char *) method->name, "main")) { bin->main = method; // get main code attr bin->main_code_attr = r_bin_java_get_attr_from_field (method, R_BIN_JAVA_ATTR_TYPE_CODE_ATTR, 0); } else if (method && (!strcmp ((const char *) method->name, "") || !strcmp ((const char *) method->name, "init"))) { IFDBG eprintf("Found an init function.\n"); bin->entrypoint = method; bin->entrypoint_code_attr = r_bin_java_get_attr_from_field (method, R_BIN_JAVA_ATTR_TYPE_CODE_ATTR, 0); } else if (method && (!strcmp ((const char *) method->name, "") || !strcmp ((const char *) method->name, "cinit"))) { bin->cf2.this_class_entrypoint = method; bin->cf2.this_class_entrypoint_code_attr = r_bin_java_get_attr_from_field (method, R_BIN_JAVA_ATTR_TYPE_CODE_ATTR, 0); } if (adv + offset > len) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Method: %d.\n", i); break; } IFDBG r_bin_java_print_field_summary(method); } bin->methods_size = adv; return adv; } R_API int r_bin_java_new_bin(RBinJavaObj *bin, ut64 loadaddr, Sdb *kv, const ut8 *buf, ut64 len) { R_BIN_JAVA_GLOBAL_BIN = bin; if (!r_str_constpool_init (&bin->constpool)) { return false; } bin->lines.count = 0; bin->loadaddr = loadaddr; r_bin_java_get_java_null_cp (); bin->id = r_num_rand (UT32_MAX); bin->kv = kv ? kv : sdb_new (NULL, NULL, 0); bin->AllJavaBinObjs = NULL; return r_bin_java_load_bin (bin, buf, len); } R_API int r_bin_java_load_bin(RBinJavaObj *bin, const ut8 *buf, ut64 buf_sz) { ut64 adv = 0; R_BIN_JAVA_GLOBAL_BIN = bin; if (!bin) { return false; } r_bin_java_reset_bin_info (bin); memcpy ((ut8 *) &bin->cf, buf, 10); if (memcmp (bin->cf.cafebabe, "\xCA\xFE\xBA\xBE", 4)) { eprintf ("r_bin_java_new_bin: Invalid header (%02x %02x %02x %02x)\n", bin->cf.cafebabe[0], bin->cf.cafebabe[1], bin->cf.cafebabe[2], bin->cf.cafebabe[3]); return false; } if (bin->cf.major[0] == bin->cf.major[1] && bin->cf.major[0] == 0) { eprintf ("Java CLASS with MACH0 header?\n"); return false; } adv += 8; // -2 so that the cp_count will be parsed adv += r_bin_java_parse_cp_pool (bin, adv, buf, buf_sz); if (adv > buf_sz) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Constant Pool.\n"); return true; } adv += r_bin_java_read_class_file2 (bin, adv, buf, buf_sz); if (adv > buf_sz) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after class file info.\n"); return true; } IFDBG eprintf("This class: %d %s\n", bin->cf2.this_class, bin->cf2.this_class_name); IFDBG eprintf("0x%"PFMT64x " Access flags: 0x%04x\n", adv, bin->cf2.access_flags); adv += r_bin_java_parse_interfaces (bin, adv, buf, buf_sz); if (adv > buf_sz) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Interfaces.\n"); return true; } adv += r_bin_java_parse_fields (bin, adv, buf, buf_sz); if (adv > buf_sz) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Fields.\n"); return true; } adv += r_bin_java_parse_methods (bin, adv, buf, buf_sz); if (adv > buf_sz) { eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Methods.\n"); return true; } adv += r_bin_java_parse_attrs (bin, adv, buf, buf_sz); bin->calc_size = adv; // if (adv > buf_sz) { // eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Attributes.\n"); // return true; // } // add_cp_objs_to_sdb(bin); // add_method_infos_to_sdb(bin); // add_field_infos_to_sdb(bin); return true; } R_API char *r_bin_java_get_version(RBinJavaObj *bin) { return r_str_newf ("0x%02x%02x 0x%02x%02x", bin->cf.major[1], bin->cf.major[0], bin->cf.minor[1], bin->cf.minor[0]); } R_API RList *r_bin_java_get_entrypoints(RBinJavaObj *bin) { RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaField *fm_type; RList *ret = r_list_newf (free); if (!ret) { return NULL; } r_list_foreach_safe (bin->methods_list, iter, iter_tmp, fm_type) { if (!strcmp (fm_type->name, "main") || !strcmp (fm_type->name, "") || !strcmp (fm_type->name, "") || strstr (fm_type->flags_str, "static")) { RBinAddr *addr = R_NEW0 (RBinAddr); if (addr) { addr->vaddr = addr->paddr = \ r_bin_java_get_method_code_offset (fm_type) + bin->loadaddr; addr->hpaddr = fm_type->file_offset; r_list_append (ret, addr); } } } return ret; } R_API RBinJavaField *r_bin_java_get_method_code_attribute_with_addr(RBinJavaObj *bin, ut64 addr) { RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaField *fm_type, *res = NULL; if (bin == NULL && R_BIN_JAVA_GLOBAL_BIN) { bin = R_BIN_JAVA_GLOBAL_BIN; } else if (bin == NULL) { eprintf ("Attempting to analyse function when the R_BIN_JAVA_GLOBAL_BIN has not been set.\n"); return NULL; } r_list_foreach_safe (bin->methods_list, iter, iter_tmp, fm_type) { ut64 offset = r_bin_java_get_method_code_offset (fm_type) + bin->loadaddr, size = r_bin_java_get_method_code_size (fm_type); if (addr >= offset && addr <= size + offset) { res = fm_type; } } return res; } R_API RBinAddr *r_bin_java_get_entrypoint(RBinJavaObj *bin, int sym) { RBinAddr *ret = NULL; ret = R_NEW0 (RBinAddr); if (!ret) { return NULL; } ret->paddr = UT64_MAX; switch (sym) { case R_BIN_SYM_ENTRY: case R_BIN_SYM_INIT: ret->paddr = r_bin_java_find_method_offset (bin, ""); if (ret->paddr == UT64_MAX) { ret->paddr = r_bin_java_find_method_offset (bin, ""); } break; case R_BIN_SYM_FINI: ret->paddr = UT64_MAX; break; case R_BIN_SYM_MAIN: ret->paddr = r_bin_java_find_method_offset (bin, "main"); break; default: ret->paddr = -1; } if (ret->paddr != -1) { ret->paddr += bin->loadaddr; } return ret; } R_API ut64 r_bin_java_get_method_code_size(RBinJavaField *fm_type) { RListIter *attr_iter = NULL, *attr_iter_tmp = NULL; RBinJavaAttrInfo *attr = NULL; ut64 sz = 0; r_list_foreach_safe (fm_type->attributes, attr_iter, attr_iter_tmp, attr) { if (attr->type == R_BIN_JAVA_ATTR_TYPE_CODE_ATTR) { sz = attr->info.code_attr.code_length; break; } } return sz; } R_API ut64 r_bin_java_find_method_offset(RBinJavaObj *bin, const char *method_name) { RListIter *attr_iter = NULL, *attr_iter_tmp = NULL; RBinJavaField *method = NULL; ut64 offset = -1; r_list_foreach_safe (bin->methods_list, attr_iter, attr_iter_tmp, method) { if (method && !strcmp ((const char *) method->name, method_name)) { offset = r_bin_java_get_method_code_offset (method) + bin->loadaddr; break; } } return offset; } R_API ut64 r_bin_java_get_method_code_offset(RBinJavaField *fm_type) { RListIter *attr_iter = NULL, *attr_iter_tmp = NULL; RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; r_list_foreach_safe (fm_type->attributes, attr_iter, attr_iter_tmp, attr) { if (attr->type == R_BIN_JAVA_ATTR_TYPE_CODE_ATTR) { offset = attr->info.code_attr.code_offset; break; } } return offset; } R_API RBinField *r_bin_java_allocate_rbinfield(void) { RBinField *t = (RBinField *) malloc (sizeof (RBinField)); if (t) { memset (t, 0, sizeof (RBinField)); } return t; } R_API RBinField *r_bin_java_create_new_rbinfield_from_field(RBinJavaField *fm_type, ut64 baddr) { RBinField *field = r_bin_java_allocate_rbinfield (); if (field) { field->name = strdup (fm_type->name); field->paddr = fm_type->file_offset + baddr; field->visibility = fm_type->flags; } return field; } R_API RBinSymbol *r_bin_java_create_new_symbol_from_field(RBinJavaField *fm_type, ut64 baddr) { RBinSymbol *sym = R_NEW0 (RBinSymbol); if (fm_type == NULL || fm_type->field_ref_cp_obj == NULL || fm_type->field_ref_cp_obj == &R_BIN_JAVA_NULL_TYPE) { R_FREE (sym); } if (sym) { sym->name = strdup (fm_type->name); // strncpy (sym->type, fm_type->descriptor, R_BIN_SIZEOF_STRINGS); if (fm_type->type == R_BIN_JAVA_FIELD_TYPE_METHOD) { sym->type = R_BIN_TYPE_FUNC_STR; sym->paddr = r_bin_java_get_method_code_offset (fm_type); sym->vaddr = r_bin_java_get_method_code_offset (fm_type) + baddr; sym->size = r_bin_java_get_method_code_size (fm_type); } else { sym->type = "FIELD"; sym->paddr = fm_type->file_offset;// r_bin_java_get_method_code_offset (fm_type); sym->vaddr = fm_type->file_offset + baddr; sym->size = fm_type->size; } if (r_bin_java_is_fm_type_protected (fm_type)) { sym->bind = R_BIN_BIND_LOCAL_STR; } else if (r_bin_java_is_fm_type_private (fm_type)) { sym->bind = R_BIN_BIND_LOCAL_STR; } else if (r_bin_java_is_fm_type_protected (fm_type)) { sym->bind = R_BIN_BIND_GLOBAL_STR; } sym->forwarder = "NONE"; if (fm_type->class_name) { sym->classname = strdup (fm_type->class_name); } else { sym->classname = strdup ("UNKNOWN"); // dupped names? } sym->ordinal = fm_type->metas->ord; sym->visibility = fm_type->flags; if (fm_type->flags_str) { sym->visibility_str = strdup (fm_type->flags_str); } } return sym; } R_API RBinSymbol *r_bin_java_create_new_symbol_from_fm_type_meta(RBinJavaField *fm_type, ut64 baddr) { RBinSymbol *sym = R_NEW0 (RBinSymbol); if (!sym || !fm_type || !fm_type->field_ref_cp_obj || fm_type->field_ref_cp_obj == &R_BIN_JAVA_NULL_TYPE) { free (sym); return NULL; } // ut32 new_name_len = strlen (fm_type->name) + strlen ("_meta") + 1; // char *new_name = malloc (new_name_len); sym->name = r_str_newf ("meta_%s", fm_type->name); if (fm_type->type == R_BIN_JAVA_FIELD_TYPE_METHOD) { sym->type = "FUNC_META"; } else { sym->type = "FIELD_META"; } if (r_bin_java_is_fm_type_protected (fm_type)) { sym->bind = R_BIN_BIND_LOCAL_STR; } else if (r_bin_java_is_fm_type_private (fm_type)) { sym->bind = R_BIN_BIND_LOCAL_STR; } else if (r_bin_java_is_fm_type_protected (fm_type)) { sym->bind = R_BIN_BIND_GLOBAL_STR; } sym->forwarder = "NONE"; if (fm_type->class_name) { sym->classname = strdup (fm_type->class_name); } else { sym->classname = strdup ("UNKNOWN"); } sym->paddr = fm_type->file_offset;// r_bin_java_get_method_code_offset (fm_type); sym->vaddr = fm_type->file_offset + baddr; sym->ordinal = fm_type->metas->ord; sym->size = fm_type->size; sym->visibility = fm_type->flags; if (fm_type->flags_str) { sym->visibility_str = strdup (fm_type->flags_str); } return sym; } R_API RBinSymbol *r_bin_java_create_new_symbol_from_ref(RBinJavaObj *bin, RBinJavaCPTypeObj *obj, ut64 baddr) { RBinSymbol *sym = R_NEW0 (RBinSymbol); if (!sym) { return NULL; } char *class_name, *name, *type_name; if (obj == NULL || (obj->tag != R_BIN_JAVA_CP_METHODREF && obj->tag != R_BIN_JAVA_CP_INTERFACEMETHOD_REF && obj->tag != R_BIN_JAVA_CP_FIELDREF)) { R_FREE (sym); return sym; } if (sym) { class_name = r_bin_java_get_name_from_bin_cp_list (bin, obj->info.cp_method.class_idx); name = r_bin_java_get_name_from_bin_cp_list (bin, obj->info.cp_method.name_and_type_idx); type_name = r_bin_java_get_name_from_bin_cp_list (bin, obj->info.cp_method.name_and_type_idx); if (name) { sym->name = name; name = NULL; } if (type_name) { sym->type = r_str_constpool_get (&bin->constpool, type_name); R_FREE (type_name); } if (class_name) { sym->classname = strdup (class_name); } sym->paddr = obj->file_offset + baddr; sym->vaddr = obj->file_offset + baddr; sym->ordinal = obj->metas->ord; sym->size = 0; } return sym; } // TODO: vaddr+vsize break things if set R_API RList *r_bin_java_get_sections(RBinJavaObj *bin) { RBinSection *section = NULL; RList *sections = r_list_newf (free); ut64 baddr = bin->loadaddr; RBinJavaField *fm_type; RListIter *iter = NULL; if (bin->cp_count > 0) { section = R_NEW0 (RBinSection); if (section) { section->name = strdup ("constant_pool"); section->paddr = bin->cp_offset + baddr; section->size = bin->cp_size; #if 0 section->vsize = section->size; section->vaddr = 0x10; // XXX // bin->cp_offset; // + baddr; #endif section->vaddr = baddr; // section->vaddr = section->paddr; // section->vsize = section->size; section->perm = R_PERM_R; section->add = true; r_list_append (sections, section); } section = NULL; } if (bin->fields_count > 0) { section = R_NEW0 (RBinSection); if (section) { section->name = strdup ("fields"); section->size = bin->fields_size; section->paddr = bin->fields_offset + baddr; #if 0 section->vsize = section->size; section->vaddr = section->paddr; #endif section->perm = R_PERM_R; section->add = true; r_list_append (sections, section); section = NULL; r_list_foreach (bin->fields_list, iter, fm_type) { if (fm_type->attr_offset == 0) { continue; } section = R_NEW0 (RBinSection); if (section) { section->name = r_str_newf ("attrs.%s", fm_type->name); section->size = fm_type->size - (fm_type->file_offset - fm_type->attr_offset); #if 0 section->vsize = section->size; section->vaddr = section->paddr; #endif section->paddr = fm_type->attr_offset + baddr; section->perm = R_PERM_R; section->add = true; r_list_append (sections, section); } } } } if (bin->methods_count > 0) { section = R_NEW0 (RBinSection); if (section) { section->name = strdup ("methods"); section->paddr = bin->methods_offset + baddr; section->size = bin->methods_size; // section->vaddr = section->paddr; // section->vsize = section->size; section->perm = R_PERM_RX; section->add = true; r_list_append (sections, section); section = NULL; r_list_foreach (bin->methods_list, iter, fm_type) { if (fm_type->attr_offset == 0) { continue; } section = R_NEW0 (RBinSection); if (section) { section->name = r_str_newf ("attrs.%s", fm_type->name); section->size = fm_type->size - (fm_type->file_offset - fm_type->attr_offset); // section->vsize = section->size; // section->vaddr = section->paddr; section->paddr = fm_type->attr_offset + baddr; section->perm = R_PERM_R | R_PERM_X; section->add = true; r_list_append (sections, section); } } } } if (bin->interfaces_count > 0) { section = R_NEW0 (RBinSection); if (section) { section->name = strdup ("interfaces"); section->paddr = bin->interfaces_offset + baddr; section->size = bin->interfaces_size; // section->vaddr = section->paddr; // section->vsize = section->size; section->perm = R_PERM_R; section->add = true; r_list_append (sections, section); } section = NULL; } if (bin->attrs_count > 0) { section = R_NEW0 (RBinSection); if (section) { section->name = strdup ("attributes"); section->paddr = bin->attrs_offset + baddr; section->size = bin->attrs_size; // section->vaddr = section->paddr; // section->vsize = section->size; section->perm = R_PERM_R; section->perm = R_PERM_R; section->add = true; r_list_append (sections, section); } section = NULL; } return sections; } R_API RList *r_bin_java_enum_class_methods(RBinJavaObj *bin, ut16 class_idx) { RList *methods = r_list_newf (free); RListIter *iter; RBinJavaField *field; r_list_foreach (bin->methods_list, iter, field) { if (field->field_ref_cp_obj && 0) { if ((field && field->field_ref_cp_obj->metas->ord == class_idx)) { RBinSymbol *sym = r_bin_java_create_new_symbol_from_ref ( bin, field->field_ref_cp_obj, bin->loadaddr); if (sym) { r_list_append (methods, sym); } } } else { RBinSymbol *sym = R_NEW0 (RBinSymbol); sym->name = strdup (field->name); // func defintion // sym->paddr = field->file_offset + bin->loadaddr; // code implementation sym->paddr = r_bin_java_get_method_code_offset (field); sym->vaddr = sym->paddr; // + bin->loadaddr; r_list_append (methods, sym); } } return methods; } R_API RList *r_bin_java_enum_class_fields(RBinJavaObj *bin, ut16 class_idx) { RList *fields = r_list_newf (free); RListIter *iter; RBinJavaField *fm_type; RBinField *field = NULL; r_list_foreach (bin->fields_list, iter, fm_type) { if (fm_type) { if (fm_type && fm_type->field_ref_cp_obj && fm_type->field_ref_cp_obj->metas->ord == class_idx) { field = r_bin_java_create_new_rbinfield_from_field (fm_type, bin->loadaddr); if (field) { r_list_append (fields, field); } } } } return fields; } R_API int is_class_interface(RBinJavaObj *bin, RBinJavaCPTypeObj *cp_obj) { RBinJavaInterfaceInfo *ifobj; RListIter *iter; int res = false; r_list_foreach (bin->interfaces_list, iter, ifobj) { if (ifobj) { res = cp_obj == ifobj->cp_class; if (res) { break; } } } return res; } /* R_API RList * r_bin_java_get_interface_classes(RBinJavaObj * bin) { RList *interfaces_names = r_list_new (); RListIter *iter; RBinJavaInterfaceInfo *ifobj; r_list_foreach(bin->interfaces_list, iter, iinfo) { RBinClass *class_ = R_NEW0 (RBinClass); RBinJavaCPTypeObj *cp_obj = ; if (ifobj && ifobj->name) { ut8 * name = strdup(ifobj->name); r_list_append(interfaces_names, name); } } return interfaces_names; } */ R_API RList *r_bin_java_get_lib_names(RBinJavaObj *bin) { RList *lib_names = r_list_newf (free); RListIter *iter; RBinJavaCPTypeObj *cp_obj = NULL; if (!bin) { return lib_names; } r_list_foreach (bin->cp_list, iter, cp_obj) { if (cp_obj && cp_obj->tag == R_BIN_JAVA_CP_CLASS && (bin->cf2.this_class != cp_obj->info.cp_class.name_idx || !is_class_interface (bin, cp_obj))) { char *name = r_bin_java_get_item_name_from_bin_cp_list (bin, cp_obj); if (name) { r_list_append (lib_names, name); } } } return lib_names; } R_API void r_bin_java_classes_free(void /*RBinClass*/ *k) { RBinClass *klass = k; if (klass) { r_list_free (klass->methods); r_list_free (klass->fields); free (klass->name); free (klass->super); free (klass->visibility_str); free (klass); } } R_API RList *r_bin_java_get_classes(RBinJavaObj *bin) { RList *classes = r_list_newf (r_bin_java_classes_free); RListIter *iter; RBinJavaCPTypeObj *cp_obj = NULL; RBinJavaCPTypeObj *this_class_cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, bin->cf2.this_class); ut32 idx = 0; RBinClass *k = R_NEW0 (RBinClass); if (!k) { r_list_free (classes); return NULL; } k->visibility = bin->cf2.access_flags; if (bin->cf2.flags_str) { k->visibility_str = strdup (bin->cf2.flags_str); } k->methods = r_bin_java_enum_class_methods (bin, bin->cf2.this_class); k->fields = r_bin_java_enum_class_fields (bin, bin->cf2.this_class); k->name = r_bin_java_get_this_class_name (bin); k->super = r_bin_java_get_name_from_bin_cp_list (bin, bin->cf2.super_class); k->index = (idx++); r_list_append (classes, k); r_list_foreach (bin->cp_list, iter, cp_obj) { if (cp_obj && cp_obj->tag == R_BIN_JAVA_CP_CLASS && (this_class_cp_obj != cp_obj && is_class_interface (bin, cp_obj))) { k = R_NEW0 (RBinClass); if (!k) { break; } k->methods = r_bin_java_enum_class_methods (bin, cp_obj->info.cp_class.name_idx); k->fields = r_bin_java_enum_class_fields (bin, cp_obj->info.cp_class.name_idx); k->index = idx; k->name = r_bin_java_get_item_name_from_bin_cp_list (bin, cp_obj); r_list_append (classes, k); idx++; } } return classes; } R_API RBinSymbol *r_bin_java_create_new_symbol_from_invoke_dynamic(RBinJavaCPTypeObj *obj, ut64 baddr) { if (!obj || (obj->tag != R_BIN_JAVA_CP_INVOKEDYNAMIC)) { return NULL; } return r_bin_java_create_new_symbol_from_cp_idx (obj->info.cp_invoke_dynamic.name_and_type_index, baddr); } R_API RBinSymbol *r_bin_java_create_new_symbol_from_cp_idx(ut32 cp_idx, ut64 baddr) { RBinSymbol *sym = NULL; RBinJavaCPTypeObj *obj = r_bin_java_get_item_from_bin_cp_list ( R_BIN_JAVA_GLOBAL_BIN, cp_idx); if (obj) { switch (obj->tag) { case R_BIN_JAVA_CP_METHODREF: case R_BIN_JAVA_CP_FIELDREF: case R_BIN_JAVA_CP_INTERFACEMETHOD_REF: sym = r_bin_java_create_new_symbol_from_ref (R_BIN_JAVA_GLOBAL_BIN, obj, baddr); break; case R_BIN_JAVA_CP_INVOKEDYNAMIC: sym = r_bin_java_create_new_symbol_from_invoke_dynamic (obj, baddr); break; default: break; } } return sym; } R_API RList *U(r_bin_java_get_fields)(RBinJavaObj * bin) { RListIter *iter = NULL, *iter_tmp = NULL; RList *fields = r_list_new (); RBinJavaField *fm_type; RBinField *field; r_list_foreach_safe (bin->fields_list, iter, iter_tmp, fm_type) { field = r_bin_java_create_new_rbinfield_from_field (fm_type, bin->loadaddr); if (field) { r_list_append (fields, field); } } return fields; } R_API void r_bin_add_import(RBinJavaObj *bin, RBinJavaCPTypeObj *obj, const char *type) { RBinImport *imp = R_NEW0 (RBinImport); char *class_name = r_bin_java_get_name_from_bin_cp_list (bin, obj->info.cp_method.class_idx); char *name = r_bin_java_get_name_from_bin_cp_list (bin, obj->info.cp_method.name_and_type_idx); char *descriptor = r_bin_java_get_desc_from_bin_cp_list (bin, obj->info.cp_method.name_and_type_idx); class_name = class_name ? class_name : strdup ("INVALID CLASS NAME INDEX"); name = name ? name : strdup ("InvalidNameIndex"); descriptor = descriptor ? descriptor : strdup ("INVALID DESCRIPTOR INDEX"); imp->classname = class_name; imp->name = name; imp->bind = "NONE"; imp->type = r_str_constpool_get (&bin->constpool, type); imp->descriptor = descriptor; imp->ordinal = obj->idx; r_list_append (bin->imports_list, imp); } R_API void r_bin_java_set_imports(RBinJavaObj *bin) { RListIter *iter = NULL; RBinJavaCPTypeObj *obj = NULL; r_list_free (bin->imports_list); bin->imports_list = r_list_newf (free); r_list_foreach (bin->cp_list, iter, obj) { const char *type = NULL; switch (obj->tag) { case R_BIN_JAVA_CP_METHODREF: type = "METHOD"; break; case R_BIN_JAVA_CP_INTERFACEMETHOD_REF: type = "FIELD"; break; case R_BIN_JAVA_CP_FIELDREF: type = "INTERFACE_METHOD"; break; default: type = NULL; break; } if (type) { r_bin_add_import (bin, obj, type); } } } R_API RList *r_bin_java_get_imports(RBinJavaObj *bin) { RList *ret = r_list_newf (free); RBinImport *import = NULL; RListIter *iter; r_list_foreach (bin->imports_list, iter, import) { RBinImport *n_import = R_NEW0 (RBinImport); if (!n_import) { r_list_free (ret); return NULL; } memcpy (n_import, import, sizeof (RBinImport)); r_list_append (ret, n_import); } return ret; } R_API RList *r_bin_java_get_symbols(RBinJavaObj *bin) { RListIter *iter = NULL, *iter_tmp = NULL; RList *imports, *symbols = r_list_newf (free); RBinSymbol *sym = NULL; RBinImport *imp; RBinJavaField *fm_type; r_list_foreach_safe (bin->methods_list, iter, iter_tmp, fm_type) { sym = r_bin_java_create_new_symbol_from_field (fm_type, bin->loadaddr); if (sym) { r_list_append (symbols, (void *) sym); } sym = r_bin_java_create_new_symbol_from_fm_type_meta (fm_type, bin->loadaddr); if (sym) { r_list_append (symbols, (void *) sym); } } r_list_foreach_safe (bin->fields_list, iter, iter_tmp, fm_type) { sym = r_bin_java_create_new_symbol_from_field (fm_type, bin->loadaddr); if (sym) { r_list_append (symbols, (void *) sym); } sym = r_bin_java_create_new_symbol_from_fm_type_meta (fm_type, bin->loadaddr); if (sym) { r_list_append (symbols, (void *) sym); } } bin->lang = "java"; if (bin->cf.major[1] >= 46) { switch (bin->cf.major[1]) { static char lang[32]; int langid; case 46: case 47: case 48: langid = 2 + (bin->cf.major[1] - 46); snprintf (lang, sizeof (lang) - 1, "java 1.%d", langid); bin->lang = lang; break; default: langid = 5 + (bin->cf.major[1] - 49); snprintf (lang, sizeof (lang) - 1, "java %d", langid); bin->lang = lang; } } imports = r_bin_java_get_imports (bin); r_list_foreach (imports, iter, imp) { sym = R_NEW0 (RBinSymbol); if (!sym) { break; } if (imp->classname && !strncmp (imp->classname, "kotlin/jvm", 10)) { bin->lang = "kotlin"; } sym->name = strdup (imp->name); sym->is_imported = true; if (!sym->name) { free (sym); break; } sym->type = "import"; if (!sym->type) { free (sym); break; } sym->vaddr = sym->paddr = imp->ordinal; sym->ordinal = imp->ordinal; r_list_append (symbols, (void *) sym); } r_list_free (imports); return symbols; } R_API RList *r_bin_java_get_strings(RBinJavaObj *bin) { RList *strings = r_list_newf (free); RBinString *str = NULL; RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaCPTypeObj *cp_obj = NULL; r_list_foreach_safe (bin->cp_list, iter, iter_tmp, cp_obj) { if (cp_obj && cp_obj->tag == R_BIN_JAVA_CP_UTF8) { str = (RBinString *) R_NEW0 (RBinString); if (str) { str->paddr = cp_obj->file_offset + bin->loadaddr; str->ordinal = cp_obj->metas->ord; str->size = cp_obj->info.cp_utf8.length + 3; str->length = cp_obj->info.cp_utf8.length; if (str->size > 0) { str->string = r_str_ndup ((const char *) cp_obj->info.cp_utf8.bytes, R_BIN_JAVA_MAXSTR); } r_list_append (strings, (void *) str); } } } return strings; } R_API void *r_bin_java_free(RBinJavaObj *bin) { char *bin_obj_key = NULL; if (!bin) { return NULL; } // Delete the bin object from the data base. bin_obj_key = r_bin_java_build_obj_key (bin); // if (bin->AllJavaBinObjs && sdb_exists (bin->AllJavaBinObjs, bin_obj_key)) { // sdb_unset (bin->AllJavaBinObjs, bin_obj_key, 0); // } free (bin_obj_key); r_list_free (bin->imports_list); // XXX - Need to remove all keys belonging to this class from // the share meta information sdb. // TODO e.g. iterate over bin->kv and delete all obj, func, etc. keys // sdb_free (bin->kv); // free up the constant pool list r_list_free (bin->cp_list); // free up the fields list r_list_free (bin->fields_list); // free up methods list r_list_free (bin->methods_list); // free up interfaces list r_list_free (bin->interfaces_list); r_list_free (bin->attrs_list); // TODO: XXX if a class list of all inner classes // are formed then this will need to be updated free (bin->cf2.flags_str); free (bin->cf2.this_class_name); if (bin == R_BIN_JAVA_GLOBAL_BIN) { R_BIN_JAVA_GLOBAL_BIN = NULL; } free (bin->file); r_str_constpool_fini (&bin->constpool); free (bin); return NULL; } R_API RBinJavaObj *r_bin_java_new_buf(RBuffer *buf, ut64 loadaddr, Sdb *kv) { RBinJavaObj *bin = R_NEW0 (RBinJavaObj); if (!bin) { return NULL; } ut64 tmpsz; const ut8 *tmp = r_buf_data (buf, &tmpsz); if (!r_bin_java_new_bin (bin, loadaddr, kv, tmp, tmpsz)) { return r_bin_java_free (bin); } return bin; } R_API void r_bin_java_attribute_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { IFDBG eprintf("Deleting attr %s, %p\n", attr->name, attr); if (attr && attr->metas && attr->metas->type_info) { RBinJavaAttrMetas *a = attr->metas->type_info; if (a && a->allocs && a->allocs->delete_obj) { a->allocs->delete_obj (attr); } } // free (attr->metas); // free (attr); } } R_API void r_bin_java_constant_pool(void /*RBinJavaCPTypeObj*/ *o) { RBinJavaCPTypeObj *obj = o; if (obj != &R_BIN_JAVA_NULL_TYPE) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->delete_obj (obj); } } R_API void r_bin_java_fmtype_free(void /*RBinJavaField*/ *f) { RBinJavaField *fm_type = f; if (!fm_type) { return; } free (fm_type->descriptor); free (fm_type->name); free (fm_type->flags_str); free (fm_type->class_name); free (fm_type->metas); r_list_free (fm_type->attributes); free (fm_type); } // Start Free the various attribute types R_API void r_bin_java_unknown_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_local_variable_table_attr_entry_free(void /*RBinJavaLocalVariableAttribute*/ *a) { RBinJavaLocalVariableAttribute *lvattr = a; if (lvattr) { free (lvattr->descriptor); free (lvattr->name); free (lvattr); } } R_API void r_bin_java_local_variable_table_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); r_list_free (attr->info.local_variable_table_attr.local_variable_table); free (attr); } } R_API void r_bin_java_local_variable_type_table_attr_entry_free(void /*RBinJavaLocalVariableTypeAttribute*/ *a) { RBinJavaLocalVariableTypeAttribute *attr = a; if (attr) { free (attr->name); free (attr->signature); free (attr); } } R_API void r_bin_java_local_variable_type_table_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); r_list_free (attr->info.local_variable_type_table_attr.local_variable_table); free (attr); } } R_API void r_bin_java_deprecated_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_enclosing_methods_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr->info.enclosing_method_attr.class_name); free (attr->info.enclosing_method_attr.method_name); free (attr->info.enclosing_method_attr.method_descriptor); free (attr); } } R_API void r_bin_java_synthetic_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_constant_value_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_line_number_table_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); r_list_free (attr->info.line_number_table_attr.line_number_table); free (attr); } } R_API void r_bin_java_code_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { // XXX - Intentional memory leak here. When one of the // Code attributes is parsed, the code (the r_bin_java) // is not properly parsing the class file r_bin_java_stack_frame_free (attr->info.code_attr.implicit_frame); r_list_free (attr->info.code_attr.attributes); free (attr->info.code_attr.code); r_list_free (attr->info.code_attr.exception_table); free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_exceptions_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr->info.exceptions_attr.exception_idx_table); free (attr); } } R_API void r_bin_java_inner_classes_attr_entry_free(void /*RBinJavaClassesAttribute*/ *a) { RBinJavaClassesAttribute *attr = a; if (attr) { free (attr->name); free (attr->flags_str); free (attr); } } R_API void r_bin_java_inner_classes_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); r_list_free (attr->info.inner_classes_attr.classes); free (attr); } } R_API void r_bin_java_signature_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr->info.signature_attr.signature); free (attr); } } R_API void r_bin_java_source_debug_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr->info.debug_extensions.debug_extension); free (attr); } } R_API void r_bin_java_source_code_file_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_stack_map_table_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { free (attr->name); free (attr->metas); r_list_free (attr->info.stack_map_table_attr.stack_map_frame_entries); free (attr); } } R_API void r_bin_java_stack_frame_free(void /*RBinJavaStackMapFrame*/ *o) { RBinJavaStackMapFrame *obj = o; if (obj) { r_list_free (obj->local_items); r_list_free (obj->stack_items); free (obj->metas); free (obj); } } R_API void r_bin_java_verification_info_free(void /*RBinJavaVerificationObj*/ *o) { RBinJavaVerificationObj *obj = o; // eprintf ("Freeing verification object\n"); if (obj) { free (obj->name); free (obj); } } R_API void r_bin_java_interface_free(void /*RBinJavaInterfaceInfo*/ *o) { RBinJavaInterfaceInfo *obj = o; if (obj) { free (obj->name); free (obj); } } // End Free the various attribute types // Start the various attibute types new R_API ut64 r_bin_java_attr_calc_size(RBinJavaAttrInfo *attr) { return attr ? ((RBinJavaAttrMetas *) attr->metas->type_info)->allocs->calc_size (attr) : 0; } R_API ut64 r_bin_java_unknown_attr_calc_size(RBinJavaAttrInfo *attr) { return attr ? 6 : 0; } R_API RBinJavaAttrInfo *r_bin_java_unknown_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { return r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); } R_API ut64 r_bin_java_code_attr_calc_size(RBinJavaAttrInfo *attr) { RListIter *iter; // RListIter *iter_tmp; ut64 size = 0; bool is_attr_in_old_format = attr->is_attr_in_old_format; if (attr) { // attr = r_bin_java_default_attr_new (buffer, sz, buf_offset); size += is_attr_in_old_format ? 4 : 6; // attr->info.code_attr.max_stack = R_BIN_JAVA_USHORT (buffer, 0); size += is_attr_in_old_format ? 1 : 2; // attr->info.code_attr.max_locals = R_BIN_JAVA_USHORT (buffer, 2); size += is_attr_in_old_format ? 1 : 2; // attr->info.code_attr.code_length = R_BIN_JAVA_UINT (buffer, 4); size += is_attr_in_old_format ? 2 : 4; if (attr->info.code_attr.code) { size += attr->info.code_attr.code_length; } // attr->info.code_attr.exception_table_length = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // RBinJavaExceptionEntry *exc_entry; // r_list_foreach_safe (attr->info.code_attr.exception_table, iter, iter_tmp, exc_entry) { r_list_foreach_iter (attr->info.code_attr.exception_table, iter) { // exc_entry->start_pc = R_BIN_JAVA_USHORT (buffer,offset); size += 2; // exc_entry->end_pc = R_BIN_JAVA_USHORT (buffer,offset); size += 2; // exc_entry->handler_pc = R_BIN_JAVA_USHORT (buffer,offset); size += 2; // exc_entry->catch_type = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } // attr->info.code_attr.attributes_count = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // RBinJavaAttrInfo *_attr; if (attr->info.code_attr.attributes_count > 0) { // r_list_foreach_safe (attr->info.code_attr.attributes, iter, iter_tmp, _attr) { r_list_foreach_iter (attr->info.code_attr.attributes, iter) { size += r_bin_java_attr_calc_size (attr); } } } return size; } R_API RBinJavaAttrInfo *r_bin_java_code_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaAttrInfo *attr = NULL, *_attr = NULL; ut32 k = 0, curpos; ut64 offset = 0; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr) { return NULL; } if (sz < 16 || sz > buf_offset) {// sz > buf_offset) { free (attr); return NULL; } offset += 6; attr->type = R_BIN_JAVA_ATTR_TYPE_CODE_ATTR; attr->info.code_attr.max_stack = attr->is_attr_in_old_format ? buffer[offset] : R_BIN_JAVA_USHORT (buffer, offset); offset += attr->is_attr_in_old_format ? 1 : 2; attr->info.code_attr.max_locals = attr->is_attr_in_old_format ? buffer[offset] : R_BIN_JAVA_USHORT (buffer, offset); offset += attr->is_attr_in_old_format ? 1 : 2; attr->info.code_attr.code_length = attr->is_attr_in_old_format ? R_BIN_JAVA_USHORT(buffer, offset) : R_BIN_JAVA_UINT (buffer, offset); offset += attr->is_attr_in_old_format ? 2 : 4; // BUG: possible unsigned integer overflow here attr->info.code_attr.code_offset = buf_offset + offset; attr->info.code_attr.code = (ut8 *) malloc (attr->info.code_attr.code_length); if (!attr->info.code_attr.code) { eprintf ("Handling Code Attributes: Unable to allocate memory " "(%u bytes) for a code.\n", attr->info.code_attr.code_length); return attr; } R_BIN_JAVA_GLOBAL_BIN->current_code_attr = attr; { int len = attr->info.code_attr.code_length; memset (attr->info.code_attr.code, 0, len); if (offset + len >= sz) { return attr; } memcpy (attr->info.code_attr.code, buffer + offset, len); offset += len; } attr->info.code_attr.exception_table_length = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.code_attr.exception_table = r_list_newf (free); for (k = 0; k < attr->info.code_attr.exception_table_length; k++) { curpos = buf_offset + offset; if (curpos + 8 > sz) { return attr; } RBinJavaExceptionEntry *e = R_NEW0 (RBinJavaExceptionEntry); if (!e) { free (attr); return NULL; } e->file_offset = curpos; e->start_pc = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; e->end_pc = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; e->handler_pc = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; e->catch_type = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; r_list_append (attr->info.code_attr.exception_table, e); e->size = 8; } attr->info.code_attr.attributes_count = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // IFDBG eprintf (" code Attributes_count: %d\n", attr->info.code_attr.attributes_count); // XXX - attr->info.code_attr.attributes is not freed because one of the code attributes is improperly parsed. attr->info.code_attr.attributes = r_list_newf (r_bin_java_attribute_free); if (attr->info.code_attr.attributes_count > 0) { for (k = 0; k < attr->info.code_attr.attributes_count; k++) { int size = (offset < sz) ? sz - offset : 0; if (size > sz || size <= 0) { break; } _attr = r_bin_java_read_next_attr_from_buffer (bin, buffer + offset, size, buf_offset + offset); if (!_attr) { eprintf ("[X] r_bin_java_code_attr_new: Error unable to parse remainder of classfile after Method's Code Attribute: %d.\n", k); break; } IFDBG eprintf("Parsing @ 0x%"PFMT64x " (%s) = 0x%"PFMT64x " bytes, %p\n", _attr->file_offset, _attr->name, _attr->size, _attr); offset += _attr->size; r_list_append (attr->info.code_attr.attributes, _attr); if (_attr->type == R_BIN_JAVA_ATTR_TYPE_LOCAL_VARIABLE_TABLE_ATTR) { IFDBG eprintf("Parsed the LocalVariableTable, preparing the implicit mthod frame.\n"); // r_bin_java_print_attr_summary(_attr); attr->info.code_attr.implicit_frame = r_bin_java_build_stack_frame_from_local_variable_table (R_BIN_JAVA_GLOBAL_BIN, _attr); attr->info.code_attr.implicit_frame->file_offset = buf_offset; IFDBG r_bin_java_print_stack_map_frame_summary(attr->info.code_attr.implicit_frame); // r_list_append (attr->info.code_attr.attributes, attr->info.code_attr.implicit_frame); } // if (offset > sz) { // eprintf ("[X] r_bin_java: Error unable to parse remainder of classfile after Attribute: %d.\n", k); // break; // } } } if (attr->info.code_attr.implicit_frame == NULL) { // build a default implicit_frame attr->info.code_attr.implicit_frame = r_bin_java_default_stack_frame (); // r_list_append (attr->info.code_attr.attributes, attr->info.code_attr.implicit_frame); } attr->size = offset; return attr; } R_API RBinJavaAttrInfo *r_bin_java_constant_value_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 6; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_CONST_VALUE_ATTR; attr->info.constant_value_attr.constantvalue_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->size = offset; } // IFDBG r_bin_java_print_constant_value_attr_summary(attr); return attr; } R_API ut64 r_bin_java_constant_value_attr_calc_size(RBinJavaAttrInfo *attr) { return attr ? 8 : 0; } R_API RBinJavaAttrInfo *r_bin_java_deprecated_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_DEPRECATED_ATTR; attr->size = offset; } // IFDBG r_bin_java_print_deprecated_attr_summary(attr); return attr; } R_API ut64 r_bin_java_deprecated_attr_calc_size(RBinJavaAttrInfo *attr) { return attr ? 6 : 0; } R_API RBinJavaAttrInfo *r_bin_java_signature_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 6; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr) { return NULL; } attr->type = R_BIN_JAVA_ATTR_TYPE_SIGNATURE_ATTR; // attr->info.source_file_attr.sourcefile_idx = R_BIN_JAVA_USHORT (buffer, offset); // offset += 2; attr->info.signature_attr.signature_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.signature_attr.signature = r_bin_java_get_utf8_from_bin_cp_list ( R_BIN_JAVA_GLOBAL_BIN, attr->info.signature_attr.signature_idx); if (!attr->info.signature_attr.signature) { eprintf ("r_bin_java_signature_attr_new: Unable to resolve the " "Signature UTF8 String Index: 0x%02x\n", attr->info.signature_attr.signature_idx); } attr->size = offset; // IFDBG r_bin_java_print_source_code_file_attr_summary(attr); return attr; } R_API ut64 r_bin_java_signature_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; if (attr == NULL) { // TODO eprintf allocation fail return size; } size += 6; // attr->info.source_file_attr.sourcefile_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // attr->info.signature_attr.signature_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; return size; } R_API RBinJavaAttrInfo *r_bin_java_enclosing_methods_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 6; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr || sz < 10) { free (attr); return NULL; } attr->type = R_BIN_JAVA_ATTR_TYPE_ENCLOSING_METHOD_ATTR; attr->info.enclosing_method_attr.class_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.enclosing_method_attr.method_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.enclosing_method_attr.class_name = r_bin_java_get_name_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, attr->info.enclosing_method_attr.class_idx); if (attr->info.enclosing_method_attr.class_name == NULL) { eprintf ("Could not resolve enclosing class name for the enclosed method.\n"); } attr->info.enclosing_method_attr.method_name = r_bin_java_get_name_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, attr->info.enclosing_method_attr.method_idx); if (attr->info.enclosing_method_attr.class_name == NULL) { eprintf ("Could not resolve method descriptor for the enclosed method.\n"); } attr->info.enclosing_method_attr.method_descriptor = r_bin_java_get_desc_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, attr->info.enclosing_method_attr.method_idx); if (attr->info.enclosing_method_attr.method_name == NULL) { eprintf ("Could not resolve method name for the enclosed method.\n"); } attr->size = offset; return attr; } R_API ut64 r_bin_java_enclosing_methods_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; if (attr) { size += 6; // attr->info.enclosing_method_attr.class_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // attr->info.enclosing_method_attr.method_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } return size; } R_API RBinJavaAttrInfo *r_bin_java_exceptions_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0, offset = 0; ut64 size; RBinJavaAttrInfo *attr = NULL; if (sz < 8) { return NULL; } attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (!attr) { return attr; } attr->type = R_BIN_JAVA_ATTR_TYPE_LINE_NUMBER_TABLE_ATTR; attr->info.exceptions_attr.number_of_exceptions = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; size = sizeof (ut16) * attr->info.exceptions_attr.number_of_exceptions; if (size < attr->info.exceptions_attr.number_of_exceptions) { free (attr); return NULL; } attr->info.exceptions_attr.exception_idx_table = (ut16 *) malloc (size); if (!attr->info.exceptions_attr.exception_idx_table) { free (attr); return NULL; } for (i = 0; i < attr->info.exceptions_attr.number_of_exceptions; i++) { if (offset + 2 > sz) { break; } attr->info.exceptions_attr.exception_idx_table[i] = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; } attr->size = offset; // IFDBG r_bin_java_print_exceptions_attr_summary(attr); return attr; } R_API ut64 r_bin_java_exceptions_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0, i = 0; if (attr) { size += 6; for (i = 0; i < attr->info.exceptions_attr.number_of_exceptions; i++) { // attr->info.exceptions_attr.exception_idx_table[i] = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } } return size; } R_API RBinJavaAttrInfo *r_bin_java_inner_classes_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaClassesAttribute *icattr; RBinJavaAttrInfo *attr = NULL; RBinJavaCPTypeObj *obj; ut32 i = 0; ut64 offset = 0, curpos; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr == NULL) { // TODO eprintf return attr; } attr->type = R_BIN_JAVA_ATTR_TYPE_INNER_CLASSES_ATTR; attr->info.inner_classes_attr.number_of_classes = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.inner_classes_attr.classes = r_list_newf (r_bin_java_inner_classes_attr_entry_free); for (i = 0; i < attr->info.inner_classes_attr.number_of_classes; i++) { curpos = buf_offset + offset; if (offset + 8 > sz) { eprintf ("Invalid amount of inner classes\n"); break; } icattr = R_NEW0 (RBinJavaClassesAttribute); if (!icattr) { break; } icattr->inner_class_info_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; icattr->outer_class_info_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; icattr->inner_name_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; icattr->inner_class_access_flags = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; icattr->flags_str = retrieve_class_method_access_string (icattr->inner_class_access_flags); icattr->file_offset = curpos; icattr->size = 8; obj = r_bin_java_get_item_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, icattr->inner_name_idx); if (obj == NULL) { eprintf ("BINCPLIS IS HULL %d\n", icattr->inner_name_idx); } icattr->name = r_bin_java_get_item_name_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, obj); if (!icattr->name) { obj = r_bin_java_get_item_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, icattr->inner_class_info_idx); if (!obj) { eprintf ("BINCPLIST IS NULL %d\n", icattr->inner_class_info_idx); } icattr->name = r_bin_java_get_item_name_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, obj); if (!icattr->name) { icattr->name = r_str_dup (NULL, "NULL"); eprintf ("r_bin_java_inner_classes_attr: Unable to find the name for %d index.\n", icattr->inner_name_idx); free (icattr); break; } } IFDBG eprintf("r_bin_java_inner_classes_attr: Inner class name %d is %s.\n", icattr->inner_name_idx, icattr->name); r_list_append (attr->info.inner_classes_attr.classes, (void *) icattr); } attr->size = offset; // IFDBG r_bin_java_print_inner_classes_attr_summary(attr); return attr; } R_API ut64 r_bin_java_inner_class_attr_calc_size(RBinJavaClassesAttribute *icattr) { ut64 size = 0; if (icattr) { // icattr->inner_class_info_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // icattr->outer_class_info_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // icattr->inner_name_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // icattr->inner_class_access_flags = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } return size; } R_API ut64 r_bin_java_inner_classes_attr_calc_size(RBinJavaAttrInfo *attr) { RBinJavaClassesAttribute *icattr = NULL; RListIter *iter; ut64 size = 6; if (!attr) { return 0; } r_list_foreach (attr->info.inner_classes_attr.classes, iter, icattr) { size += r_bin_java_inner_class_attr_calc_size (icattr); } return size; } R_API RBinJavaAttrInfo *r_bin_java_line_number_table_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; ut64 curpos, offset = 0; RBinJavaLineNumberAttribute *lnattr; if (sz < 6) { return NULL; } RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr) { return NULL; } offset += 6; attr->type = R_BIN_JAVA_ATTR_TYPE_LINE_NUMBER_TABLE_ATTR; attr->info.line_number_table_attr.line_number_table_length = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.line_number_table_attr.line_number_table = r_list_newf (free); ut32 linenum_len = attr->info.line_number_table_attr.line_number_table_length; RList *linenum_list = attr->info.line_number_table_attr.line_number_table; for (i = 0; i < linenum_len; i++) { curpos = buf_offset + offset; // printf ("%llx %llx \n", curpos, sz); // XXX if (curpos + 8 >= sz) break; lnattr = R_NEW0 (RBinJavaLineNumberAttribute); if (!lnattr) { break; } // wtf it works if (offset - 2 > sz) { R_FREE (lnattr); break; } lnattr->start_pc = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lnattr->line_number = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lnattr->file_offset = curpos; lnattr->size = 4; r_list_append (linenum_list, lnattr); } attr->size = offset; return attr; } R_API ut64 r_bin_java_line_number_table_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 6; // RBinJavaLineNumberAttribute *lnattr; RListIter *iter; // RListIter *iter_tmp; if (!attr) { return 0LL; } // r_list_foreach_safe (attr->info.line_number_table_attr.line_number_table, iter, iter_tmp, lnattr) { r_list_foreach_iter (attr->info.line_number_table_attr.line_number_table, iter) { // lnattr->start_pc = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lnattr->line_number = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } return size; } R_API RBinJavaAttrInfo *r_bin_java_source_debug_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 6; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr) { return NULL; } attr->type = R_BIN_JAVA_ATTR_TYPE_SOURCE_DEBUG_EXTENTSION_ATTR; if (attr->length == 0) { eprintf ("r_bin_java_source_debug_attr_new: Attempting to allocate 0 bytes for debug_extension.\n"); attr->info.debug_extensions.debug_extension = NULL; return attr; } else if ((attr->length + offset) > sz) { eprintf ("r_bin_java_source_debug_attr_new: Expected %d byte(s) got %" PFMT64d " bytes for debug_extension.\n", attr->length, (offset + sz)); } attr->info.debug_extensions.debug_extension = (ut8 *) malloc (attr->length); if (attr->info.debug_extensions.debug_extension && (attr->length > (sz - offset))) { memcpy (attr->info.debug_extensions.debug_extension, buffer + offset, sz - offset); } else if (attr->info.debug_extensions.debug_extension) { memcpy (attr->info.debug_extensions.debug_extension, buffer + offset, attr->length); } else { eprintf ("r_bin_java_source_debug_attr_new: Unable to allocate the data for the debug_extension.\n"); } offset += attr->length; attr->size = offset; return attr; } R_API ut64 r_bin_java_source_debug_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 6; if (!attr) { return 0LL; } if (attr->info.debug_extensions.debug_extension) { size += attr->length; } return size; } R_API ut64 r_bin_java_local_variable_table_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; // ut64 offset = 0; RListIter *iter; // RBinJavaLocalVariableAttribute *lvattr; if (!attr) { return 0LL; } size += 6; // attr->info.local_variable_table_attr.table_length = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // r_list_foreach (attr->info.local_variable_table_attr.local_variable_table, iter, lvattr) { r_list_foreach_iter (attr->info.local_variable_table_attr.local_variable_table, iter) { // lvattr->start_pc = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->length = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->name_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->descriptor_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->index = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } return size; } R_API RBinJavaAttrInfo *r_bin_java_local_variable_table_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaLocalVariableAttribute *lvattr; ut64 curpos = 0, offset = 6; RBinJavaAttrInfo *attr; ut32 i = 0; if (!buffer || sz < 1) { return NULL; } attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr) { return NULL; } attr->type = R_BIN_JAVA_ATTR_TYPE_LOCAL_VARIABLE_TABLE_ATTR; attr->info.local_variable_table_attr.table_length = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.local_variable_table_attr.local_variable_table =\ r_list_newf (r_bin_java_local_variable_table_attr_entry_free); for (i = 0; i < attr->info.local_variable_table_attr.table_length; i++) { if (offset + 10 > sz) { break; } curpos = buf_offset + offset; lvattr = R_NEW0 (RBinJavaLocalVariableAttribute); lvattr->start_pc = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->length = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->name_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->descriptor_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->index = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->file_offset = curpos; lvattr->name = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, lvattr->name_idx); lvattr->size = 10; if (!lvattr->name) { lvattr->name = strdup ("NULL"); eprintf ("r_bin_java_local_variable_table_attr_new: Unable to find the name for %d index.\n", lvattr->name_idx); } lvattr->descriptor = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, lvattr->descriptor_idx); if (!lvattr->descriptor) { lvattr->descriptor = strdup ("NULL"); eprintf ("r_bin_java_local_variable_table_attr_new: Unable to find the descriptor for %d index.\n", lvattr->descriptor_idx); } r_list_append (attr->info.local_variable_table_attr.local_variable_table, lvattr); } attr->size = offset; // IFDBG r_bin_java_print_local_variable_table_attr_summary(attr); return attr; } R_API ut64 r_bin_java_local_variable_type_table_attr_calc_size(RBinJavaAttrInfo *attr) { // RBinJavaLocalVariableTypeAttribute *lvattr; RListIter *iter; ut64 size = 0; if (attr) { RList *list = attr->info.local_variable_type_table_attr.local_variable_table; size += 6; // attr->info.local_variable_type_table_attr.table_length = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // r_list_foreach (list, iter, lvattr) { r_list_foreach_iter (list, iter) { // lvattr->start_pc = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->length = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->name_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->signature_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // lvattr->index = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } } return size; } R_API RBinJavaAttrInfo *r_bin_java_local_variable_type_table_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaLocalVariableTypeAttribute *lvattr; ut64 offset = 6; ut32 i = 0; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, 0); if (!attr) { return NULL; } attr->type = R_BIN_JAVA_ATTR_TYPE_LOCAL_VARIABLE_TYPE_TABLE_ATTR; attr->info.local_variable_type_table_attr.table_length = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.local_variable_type_table_attr.local_variable_table = r_list_newf (r_bin_java_local_variable_type_table_attr_entry_free); for (i = 0; i < attr->info.local_variable_type_table_attr.table_length; i++) { ut64 curpos = buf_offset + offset; lvattr = R_NEW0 (RBinJavaLocalVariableTypeAttribute); if (!lvattr) { perror ("calloc"); break; } if (offset + 10 > sz) { eprintf ("oob"); free (lvattr); break; } lvattr->start_pc = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->length = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->name_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->signature_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->index = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; lvattr->file_offset = curpos; lvattr->name = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, lvattr->name_idx); lvattr->size = 10; if (!lvattr->name) { lvattr->name = strdup ("NULL"); eprintf ("r_bin_java_local_variable_type_table_attr_new: Unable to find the name for %d index.\n", lvattr->name_idx); } lvattr->signature = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, lvattr->signature_idx); if (!lvattr->signature) { lvattr->signature = strdup ("NULL"); eprintf ("r_bin_java_local_variable_type_table_attr_new: Unable to find the descriptor for %d index.\n", lvattr->signature_idx); } r_list_append (attr->info.local_variable_type_table_attr.local_variable_table, lvattr); } // IFDBG r_bin_java_print_local_variable_type_table_attr_summary(attr); attr->size = offset; return attr; } R_API RBinJavaAttrInfo *r_bin_java_source_code_file_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { if (!sz) { return NULL; } ut64 offset = 0; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (!attr) { return NULL; } attr->type = R_BIN_JAVA_ATTR_TYPE_SOURCE_FILE_ATTR; // if (buffer + offset > buffer + sz) return NULL; attr->info.source_file_attr.sourcefile_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->size = offset; // IFDBG r_bin_java_print_source_code_file_attr_summary(attr); return attr; } R_API ut64 r_bin_java_source_code_file_attr_calc_size(RBinJavaAttrInfo *attr) { return attr ? 8 : 0; } R_API RBinJavaAttrInfo *r_bin_java_synthetic_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 0; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); if (!attr) { return NULL; } offset += 6; attr->type = R_BIN_JAVA_ATTR_TYPE_SYNTHETIC_ATTR; attr->size = offset; return attr; } R_API ut64 r_bin_java_synthetic_attr_calc_size(RBinJavaAttrInfo *attr) { return attr ? 12 : 6; } R_API RBinJavaInterfaceInfo *r_bin_java_interface_new(RBinJavaObj *bin, const ut8 *buffer, ut64 sz) { RBinJavaInterfaceInfo *ifobj = NULL; ifobj = R_NEW0 (RBinJavaInterfaceInfo); IFDBG eprintf("Parsing RBinJavaInterfaceInfo\n"); if (ifobj) { if (buffer) { ifobj->class_info_idx = R_BIN_JAVA_USHORT (buffer, 0); ifobj->cp_class = r_bin_java_get_item_from_bin_cp_list (bin, ifobj->class_info_idx); if (ifobj->cp_class) { ifobj->name = r_bin_java_get_item_name_from_bin_cp_list (bin, ifobj->cp_class); } else { ifobj->name = r_str_dup (NULL, "NULL"); } ifobj->size = 2; } else { ifobj->class_info_idx = 0; ifobj->name = r_str_dup (NULL, "NULL"); } } return ifobj; } R_API RBinJavaVerificationObj *r_bin_java_verification_info_from_type(RBinJavaObj *bin, R_BIN_JAVA_STACKMAP_TYPE type, ut32 value) { RBinJavaVerificationObj *se = R_NEW0 (RBinJavaVerificationObj); if (!se) { return NULL; } se->tag = type; if (se->tag == R_BIN_JAVA_STACKMAP_OBJECT) { se->info.obj_val_cp_idx = (ut16) value; } else if (se->tag == R_BIN_JAVA_STACKMAP_UNINIT) { /*if (bin->offset_sz == 4) { se->info.uninit_offset = value; } else { se->info.uninit_offset = (ut16) value; }*/ se->info.uninit_offset = (ut16) value; } return se; } R_API RBinJavaVerificationObj *r_bin_java_read_from_buffer_verification_info_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 0; RBinJavaVerificationObj *se = R_NEW0 (RBinJavaVerificationObj); if (!se) { return NULL; } se->file_offset = buf_offset; se->tag = buffer[offset]; offset += 1; if (se->tag == R_BIN_JAVA_STACKMAP_OBJECT) { se->info.obj_val_cp_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; } else if (se->tag == R_BIN_JAVA_STACKMAP_UNINIT) { se->info.uninit_offset = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; } if (R_BIN_JAVA_STACKMAP_UNINIT < se->tag) { r_bin_java_verification_info_free (se); return NULL; } se->size = offset; return se; } R_API ut64 rbin_java_verification_info_calc_size(RBinJavaVerificationObj *se) { ut64 sz = 1; if (!se) { return 0; } // r_buf_read_at (bin->b, offset, (ut8*)(&se->tag), 1) switch (se->tag) { case R_BIN_JAVA_STACKMAP_OBJECT: // r_buf_read_at (bin->b, offset+1, (ut8*)buf, 2) sz += 2; break; case R_BIN_JAVA_STACKMAP_UNINIT: // r_buf_read_at (bin->b, offset+1, (ut8*)buf, 2) sz += 2; break; } return sz; } R_API RBinJavaStackMapFrameMetas *r_bin_java_determine_stack_frame_type(ut8 tag) { ut8 type_value = 0; if (tag < 64) { type_value = R_BIN_JAVA_STACK_FRAME_SAME; } else if (tag < 128) { type_value = R_BIN_JAVA_STACK_FRAME_SAME_LOCALS_1; } else if (247 < tag && tag < 251) { type_value = R_BIN_JAVA_STACK_FRAME_CHOP; } else if (tag == 251) { type_value = R_BIN_JAVA_STACK_FRAME_SAME_FRAME_EXTENDED; } else if (251 < tag && tag < 255) { type_value = R_BIN_JAVA_STACK_FRAME_APPEND; } else if (tag == 255) { type_value = R_BIN_JAVA_STACK_FRAME_FULL_FRAME; } else { type_value = R_BIN_JAVA_STACK_FRAME_RESERVED; } return &R_BIN_JAVA_STACK_MAP_FRAME_METAS[type_value]; } R_API ut64 r_bin_java_stack_map_frame_calc_size(RBinJavaStackMapFrame *sf) { ut64 size = 0; RListIter *iter, *iter_tmp; RBinJavaVerificationObj *se; if (sf) { // sf->tag = buffer[offset]; size += 1; switch (sf->type) { case R_BIN_JAVA_STACK_FRAME_SAME: // Nothing to read break; case R_BIN_JAVA_STACK_FRAME_SAME_LOCALS_1: r_list_foreach_safe (sf->stack_items, iter, iter_tmp, se) { size += rbin_java_verification_info_calc_size (se); } break; case R_BIN_JAVA_STACK_FRAME_CHOP: // sf->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); size += 2; break; case R_BIN_JAVA_STACK_FRAME_SAME_FRAME_EXTENDED: // sf->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (sf->stack_items, iter, iter_tmp, se) { size += rbin_java_verification_info_calc_size (se); } break; case R_BIN_JAVA_STACK_FRAME_APPEND: // sf->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (sf->stack_items, iter, iter_tmp, se) { size += rbin_java_verification_info_calc_size (se); } break; case R_BIN_JAVA_STACK_FRAME_FULL_FRAME: // sf->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // sf->number_of_locals = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (sf->local_items, iter, iter_tmp, se) { size += rbin_java_verification_info_calc_size (se); } // sf->number_of_stack_items = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (sf->stack_items, iter, iter_tmp, se) { size += rbin_java_verification_info_calc_size (se); } break; default: eprintf ("Unknown type\n"); break; } } return size; } R_API RBinJavaStackMapFrame *r_bin_java_stack_map_frame_new(ut8 *buffer, ut64 sz, RBinJavaStackMapFrame *p_frame, ut64 buf_offset) { RBinJavaStackMapFrame *stack_frame = r_bin_java_default_stack_frame (); RBinJavaVerificationObj *se = NULL; ut64 offset = 0; if (!stack_frame) { return NULL; } stack_frame->tag = buffer[offset]; offset += 1; stack_frame->metas->type_info = (void *) r_bin_java_determine_stack_frame_type (stack_frame->tag); stack_frame->type = ((RBinJavaStackMapFrameMetas *) stack_frame->metas->type_info)->type; stack_frame->file_offset = buf_offset; stack_frame->p_stack_frame = p_frame; switch (stack_frame->type) { case R_BIN_JAVA_STACK_FRAME_SAME: // Maybe? 1. Copy the previous frames locals and set the locals count. // copy_type_info_to_stack_frame_list_up_to_idx (p_frame->local_items, stack_frame->local_items, idx); if (p_frame) { stack_frame->number_of_locals = p_frame->number_of_locals; } else { IFINT eprintf("><\n"); IFDBG eprintf("Unable to set previous stackframe with the number of locals (current info.code_attr.implicit_frame was probably not set :/)"); } IFDBG eprintf("r_bin_java_stack_map_frame_new: TODO Stack Frame Same Locals Condition is untested, so there may be issues.\n"); break; case R_BIN_JAVA_STACK_FRAME_SAME_LOCALS_1: // 1. Read the stack type stack_frame->number_of_stack_items = 1; if (offset > sz) { r_bin_java_stack_frame_free (stack_frame); return NULL; } se = r_bin_java_read_from_buffer_verification_info_new (buffer + offset, sz - offset, buf_offset + offset); IFDBG eprintf("r_bin_java_stack_map_frame_new: Parsed R_BIN_JAVA_STACK_FRAME_SAME_LOCALS_1.\n"); if (se) { offset += se->size; } else { eprintf ("r_bin_java_stack_map_frame_new: Unable to parse the Stack Items for the stack frame.\n"); r_bin_java_stack_frame_free (stack_frame); return NULL; } r_list_append (stack_frame->stack_items, (void *) se); // Maybe? 3. Copy the previous frames locals and set the locals count. // copy_type_info_to_stack_frame_list_up_to_idx (p_frame->local_items, stack_frame->local_items, idx); if (p_frame) { stack_frame->number_of_locals = p_frame->number_of_locals; } else { IFDBG eprintf("Unable to set previous stackframe with the number of locals (current info.code_attr.implicit_frame was probably not set :/)"); } IFDBG eprintf("r_bin_java_stack_map_frame_new: TODO Stack Frame Same Locals 1 Stack Element Condition is untested, so there may be issues.\n"); break; case R_BIN_JAVA_STACK_FRAME_CHOP: // 1. Calculate the max index we want to copy from the list of the // previous frames locals IFDBG eprintf("r_bin_java_stack_map_frame_new: Parsing R_BIN_JAVA_STACK_FRAME_CHOP.\n"); // ut16 k = 251 - stack_frame->tag; /*, idx = p_frame->number_of_locals - k; */ // 2. read the uoffset value stack_frame->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // Maybe? 3. Copy the previous frames locals and set the locals count. // copy_type_info_to_stack_frame_list_up_to_idx (p_frame->local_items, stack_frame->local_items, idx); if (p_frame) { stack_frame->number_of_locals = p_frame->number_of_locals; } else { IFINT eprintf("><\n"); IFDBG eprintf("Unable to set previous stackframe with the number of locals (current info.code_attr.implicit_frame was probably not set :/)"); } IFDBG eprintf("r_bin_java_stack_map_frame_new: TODO Stack Frame Chop Condition is untested, so there may be issues.\n"); break; case R_BIN_JAVA_STACK_FRAME_SAME_FRAME_EXTENDED: IFDBG eprintf("r_bin_java_stack_map_frame_new: Parsing R_BIN_JAVA_STACK_FRAME_SAME_FRAME_EXTENDED.\n"); // 1. Read the uoffset stack_frame->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // 2. Read the stack element type stack_frame->number_of_stack_items = 1; se = r_bin_java_read_from_buffer_verification_info_new (buffer + offset, sz - offset, buf_offset + offset); if (se) { offset += se->size; } else { eprintf ("r_bin_java_stack_map_frame_new: Unable to parse the Stack Items for the stack frame.\n"); r_bin_java_stack_frame_free (stack_frame); return NULL; } r_list_append (stack_frame->stack_items, (void *) se); // Maybe? 3. Copy the previous frames locals to the current locals // copy_type_info_to_stack_frame_list_up_to_idx (p_frame->local_items, stack_frame->local_items, idx); if (p_frame) { stack_frame->number_of_locals = p_frame->number_of_locals; } else { IFINT eprintf("><\n"); IFDBG eprintf("Unable to set previous stackframe with the number of locals (current info.code_attr.implicit_frame was probably not set :/)"); } IFDBG eprintf("r_bin_java_stack_map_frame_new: TODO Stack Frame Same Locals Frame Stack 1 Extended Condition is untested, so there may be issues.\n"); break; case R_BIN_JAVA_STACK_FRAME_APPEND: IFDBG eprintf("r_bin_java_stack_map_frame_new: Parsing R_BIN_JAVA_STACK_FRAME_APPEND.\n"); // 1. Calculate the max index we want to copy from the list of the // previous frames locals ut16 k = stack_frame->tag - 251; ut32 i = 0; // 2. Read the uoffset stack_frame->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // Maybe? 3. Copy the previous frames locals to the current locals // copy_type_info_to_stack_frame_list_up_to_idx (p_frame->local_items, stack_frame->local_items, idx); // 4. Read off the rest of the appended locals types for (i = 0; i < k; i++) { if (offset >= sz) { break; } IFDBG eprintf("r_bin_java_stack_map_frame_new: Parsing verifying the k'th frame: %d of %d.\n", i, k); se = r_bin_java_read_from_buffer_verification_info_new (buffer + offset, sz - offset, buf_offset + offset); IFDBG eprintf("r_bin_java_stack_map_frame_new: Completed Parsing\n"); if (se) { offset += se->size; } else { eprintf ("r_bin_java_stack_map_frame_new: Unable to parse the locals for the stack frame.\n"); r_bin_java_stack_frame_free (stack_frame); return NULL; } r_list_append (stack_frame->local_items, (void *) se); } IFDBG eprintf("r_bin_java_stack_map_frame_new: Breaking out of loop"); IFDBG eprintf("p_frame: %p\n", p_frame); if (p_frame) { stack_frame->number_of_locals = p_frame->number_of_locals + k; } else { IFINT eprintf("><\n"); IFDBG eprintf("Unable to set previous stackframe with the number of locals (current info.code_attr.implicit_frame was probably not set :/)"); } IFDBG eprintf("r_bin_java_stack_map_frame_new: TODO Stack Frame Same Locals Frame Stack 1 Extended Condition is untested, so there may be issues.\n"); break; case R_BIN_JAVA_STACK_FRAME_FULL_FRAME: IFDBG eprintf("r_bin_java_stack_map_frame_new: Parsing R_BIN_JAVA_STACK_FRAME_FULL_FRAME.\n"); stack_frame->offset_delta = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // IFDBG eprintf ("r_bin_java_stack_map_frame_new: Code Size > 65535, read(%d byte(s)), offset = 0x%08x.\n", var_sz, stack_frame->offset_delta); // Read the number of variables based on the max # local variable stack_frame->number_of_locals = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // IFDBG eprintf ("r_bin_java_stack_map_frame_new: Max ulocalvar > 65535, read(%d byte(s)), number_of_locals = 0x%08x.\n", var_sz, stack_frame->number_of_locals); IFDBG r_bin_java_print_stack_map_frame_summary(stack_frame); // read the number of locals off the stack for (i = 0; i < stack_frame->number_of_locals; i++) { if (offset >= sz) { break; } se = r_bin_java_read_from_buffer_verification_info_new (buffer + offset, sz - offset, buf_offset + offset); if (se) { offset += se->size; // r_list_append (stack_frame->local_items, (void *) se); } else { eprintf ("r_bin_java_stack_map_frame_new: Unable to parse the locals for the stack frame.\n"); r_bin_java_stack_frame_free (stack_frame); return NULL; } r_list_append (stack_frame->local_items, (void *) se); } // Read the number of stack items based on the max size of stack stack_frame->number_of_stack_items = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // IFDBG eprintf ("r_bin_java_stack_map_frame_new: Max ustack items > 65535, read(%d byte(s)), number_of_locals = 0x%08x.\n", var_sz, stack_frame->number_of_stack_items); // read the stack items for (i = 0; i < stack_frame->number_of_stack_items; i++) { if (offset >= sz) { break; } se = r_bin_java_read_from_buffer_verification_info_new (buffer + offset, sz - offset, buf_offset + offset); if (se) { offset += se->size; // r_list_append (stack_frame->stack_items, (void *) se); } else { eprintf ("r_bin_java_stack_map_frame_new: Unable to parse the stack items for the stack frame.\n"); r_bin_java_stack_frame_free (stack_frame); return NULL; } r_list_append (stack_frame->local_items, (void *) se); } break; default: eprintf ("java: Unknown type\n"); break; } // IFDBG eprintf ("Created a stack frame at offset(0x%08"PFMT64x") of size: %d\n", buf_offset, stack_frame->size);//r_bin_java_print_stack_map_frame_summary(stack_frame); stack_frame->size = offset; // IFDBG r_bin_java_print_stack_map_frame_summary(stack_frame); return stack_frame; } R_API ut16 r_bin_java_find_cp_class_ref_from_name_idx(RBinJavaObj *bin, ut16 name_idx) { ut16 pos, len = (ut16) r_list_length (bin->cp_list); RBinJavaCPTypeObj *item; for (pos = 0; pos < len; pos++) { item = (RBinJavaCPTypeObj *) r_list_get_n (bin->cp_list, pos); if (item && item->tag == R_BIN_JAVA_CP_CLASS && item->info.cp_class.name_idx == name_idx) { break; } } return (pos != len) ? pos : 0; } R_API RBinJavaStackMapFrame *r_bin_java_default_stack_frame(void) { RBinJavaStackMapFrame *sf = R_NEW0 (RBinJavaStackMapFrame); if (!sf) { return NULL; } sf->metas = R_NEW0 (RBinJavaMetaInfo); if (!sf->metas) { free (sf); return NULL; } sf->metas->type_info = (void *) &R_BIN_JAVA_STACK_MAP_FRAME_METAS[R_BIN_JAVA_STACK_FRAME_IMPLICIT]; sf->type = ((RBinJavaStackMapFrameMetas *) sf->metas->type_info)->type; sf->local_items = r_list_newf (r_bin_java_verification_info_free); sf->stack_items = r_list_newf (r_bin_java_verification_info_free); sf->number_of_stack_items = 0; sf->number_of_locals = 0; return sf; } R_API RBinJavaStackMapFrame *r_bin_java_build_stack_frame_from_local_variable_table(RBinJavaObj *bin, RBinJavaAttrInfo *attr) { RBinJavaStackMapFrame *sf = r_bin_java_default_stack_frame (); RBinJavaLocalVariableAttribute *lvattr = NULL; RBinJavaVerificationObj *type_item; RListIter *iter = NULL; ut32 value_cnt = 0; ut8 value; if (!sf || !bin || !attr || attr->type != R_BIN_JAVA_ATTR_TYPE_LOCAL_VARIABLE_TABLE_ATTR) { eprintf ("Attempting to create a stack_map frame from a bad attribute.\n"); return sf; } sf->number_of_locals = attr->info.local_variable_table_attr.table_length; r_list_foreach (attr->info.local_variable_table_attr.local_variable_table, iter, lvattr) { ut32 pos = 0; // knock the array Types while (lvattr->descriptor[pos] == '[') { pos++; } value = lvattr->descriptor[pos]; // IFDBG eprintf ("Found the following type value: %c at pos %d in %s\n", value, pos, lvattr->descriptor); switch (value) { case 'I': case 'Z': case 'S': case 'B': case 'C': type_item = r_bin_java_verification_info_from_type (bin, R_BIN_JAVA_STACKMAP_INTEGER, 0); break; case 'F': type_item = r_bin_java_verification_info_from_type (bin, R_BIN_JAVA_STACKMAP_FLOAT, 0); break; case 'D': type_item = r_bin_java_verification_info_from_type (bin, R_BIN_JAVA_STACKMAP_DOUBLE, 0); break; case 'J': type_item = r_bin_java_verification_info_from_type (bin, R_BIN_JAVA_STACKMAP_LONG, 0); break; case 'L': // TODO: FIXME write something that will iterate over the CP Pool and find the // CONSTANT_Class_info referencing this { ut16 idx = r_bin_java_find_cp_class_ref_from_name_idx (bin, lvattr->name_idx); type_item = r_bin_java_verification_info_from_type (bin, R_BIN_JAVA_STACKMAP_OBJECT, idx); } break; default: eprintf ("r_bin_java_build_stack_frame_from_local_variable_table: " "not sure how to handle: name: %s, type: %s\n", lvattr->name, lvattr->descriptor); type_item = r_bin_java_verification_info_from_type (bin, R_BIN_JAVA_STACKMAP_NULL, 0); } if (type_item) { r_list_append (sf->local_items, (void *) type_item); } value_cnt++; } if (value_cnt != attr->info.local_variable_table_attr.table_length) { IFDBG eprintf("r_bin_java_build_stack_frame_from_local_variable_table: " "Number of locals not accurate. Expected %d but got %d", attr->info.local_variable_table_attr.table_length, value_cnt); } return sf; } R_API ut64 r_bin_java_stack_map_table_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; RListIter *iter, *iter_tmp; RBinJavaStackMapFrame *sf; if (attr) { // attr = r_bin_java_default_attr_new (buffer, sz, buf_offset); size += 6; // IFDBG r_bin_java_print_source_code_file_attr_summary(attr); // Current spec does not call for variable sizes. // attr->info.stack_map_table_attr.number_of_entries = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (attr->info.stack_map_table_attr.stack_map_frame_entries, iter, iter_tmp, sf) { size += r_bin_java_stack_map_frame_calc_size (sf); } } return size; } R_API RBinJavaAttrInfo *r_bin_java_stack_map_table_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; ut64 offset = 0; RBinJavaStackMapFrame *stack_frame = NULL, *new_stack_frame = NULL; if (sz < 10) { return NULL; } RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; IFDBG eprintf("r_bin_java_stack_map_table_attr_new: New stack map allocated.\n"); if (!attr) { return NULL; } attr->info.stack_map_table_attr.stack_map_frame_entries = r_list_newf (r_bin_java_stack_frame_free); // IFDBG r_bin_java_print_source_code_file_attr_summary(attr); // Current spec does not call for variable sizes. attr->info.stack_map_table_attr.number_of_entries = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; IFDBG eprintf("r_bin_java_stack_map_table_attr_new: Processing stack map, summary is:\n"); IFDBG r_bin_java_print_stack_map_table_attr_summary(attr); for (i = 0; i < attr->info.stack_map_table_attr.number_of_entries; i++) { // read next stack frame IFDBG eprintf("Reading StackMap Entry #%d @ 0x%08"PFMT64x "\n", i, buf_offset + offset); if (stack_frame == NULL && R_BIN_JAVA_GLOBAL_BIN && R_BIN_JAVA_GLOBAL_BIN->current_code_attr) { IFDBG eprintf("Setting an implicit frame at #%d @ 0x%08"PFMT64x "\n", i, buf_offset + offset); stack_frame = R_BIN_JAVA_GLOBAL_BIN->current_code_attr->info.code_attr.implicit_frame; } IFDBG eprintf("Reading StackMap Entry #%d @ 0x%08"PFMT64x ", current stack_frame: %p\n", i, buf_offset + offset, stack_frame); if (offset >= sz) { r_bin_java_stack_map_table_attr_free (attr); return NULL; } new_stack_frame = r_bin_java_stack_map_frame_new (buffer + offset, sz - offset, stack_frame, buf_offset + offset); if (new_stack_frame) { offset += new_stack_frame->size; // append stack frame to the list r_list_append (attr->info.stack_map_table_attr.stack_map_frame_entries, (void *) new_stack_frame); stack_frame = new_stack_frame; } else { eprintf ("r_bin_java_stack_map_table_attr_new: Unable to parse the stack frame for the stack map table.\n"); r_bin_java_stack_map_table_attr_free (attr); attr = NULL; break; } } if (attr) { attr->size = offset; } return attr; } // End attribute types new // Start new Constant Pool Types R_API RBinJavaCPTypeObj *r_bin_java_do_nothing_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { return (RBinJavaCPTypeObj *) NULL; } R_API ut64 r_bin_java_do_nothing_calc_size(RBinJavaCPTypeObj *obj) { return 0; } R_API void r_bin_java_do_nothing_free(void /*RBinJavaCPTypeObj*/ *obj) { return; } R_API RBinJavaCPTypeObj *r_bin_java_unknown_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; obj = (RBinJavaCPTypeObj *) malloc (sizeof (RBinJavaCPTypeObj)); if (obj) { memset (obj, 0, sizeof (RBinJavaCPTypeObj)); obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[R_BIN_JAVA_CP_UNKNOWN]; } return obj; } R_API ut64 r_bin_java_unknown_cp_calc_size(RBinJavaCPTypeObj *obj) { return 1LL; } R_API RBinJavaCPTypeObj *r_bin_java_class_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_CLASS, tag, sz, "Class"); if (quick_check > 0) { return NULL; } RBinJavaCPTypeObj *obj = R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->info.cp_class.name_idx = R_BIN_JAVA_USHORT (buffer, 1); } return obj; } R_API ut64 r_bin_java_class_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // ut8 tag = buffer[0]; size += 1; // obj->info.cp_class.name_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_fieldref_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_FIELDREF, tag, sz, "FieldRef"); if (quick_check > 0) { return obj; } obj = (RBinJavaCPTypeObj *) malloc (sizeof (RBinJavaCPTypeObj)); if (obj) { memset (obj, 0, sizeof (RBinJavaCPTypeObj)); obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->info.cp_field.class_idx = R_BIN_JAVA_USHORT (buffer, 1); obj->info.cp_field.name_and_type_idx = R_BIN_JAVA_USHORT (buffer, 3); } return (RBinJavaCPTypeObj *) obj; } R_API ut64 r_bin_java_fieldref_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_field.class_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 2; // obj->info.cp_field.name_and_type_idx = R_BIN_JAVA_USHORT (buffer, 3); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_methodref_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_METHODREF, tag, sz, "MethodRef"); if (quick_check > 0) { return obj; } obj = (RBinJavaCPTypeObj *) malloc (sizeof (RBinJavaCPTypeObj)); if (obj) { memset (obj, 0, sizeof (RBinJavaCPTypeObj)); obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->info.cp_method.class_idx = R_BIN_JAVA_USHORT (buffer, 1); obj->info.cp_method.name_and_type_idx = R_BIN_JAVA_USHORT (buffer, 3); } return obj; } R_API ut64 r_bin_java_methodref_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_method.class_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 2; // obj->info.cp_method.name_and_type_idx = R_BIN_JAVA_USHORT (buffer, 3); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_interfacemethodref_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_INTERFACEMETHOD_REF, tag, sz, "InterfaceMethodRef"); if (quick_check > 0) { return NULL; } RBinJavaCPTypeObj *obj = R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); obj->info.cp_interface.class_idx = R_BIN_JAVA_USHORT (buffer, 1); obj->info.cp_interface.name_and_type_idx = R_BIN_JAVA_USHORT (buffer, 3); } return obj; } R_API ut64 r_bin_java_interfacemethodref_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_interface.class_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 2; // obj->info.cp_interface.name_and_type_idx = R_BIN_JAVA_USHORT (buffer, 3); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_string_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_STRING, tag, sz, "String"); if (quick_check > 0) { return NULL; } RBinJavaCPTypeObj *obj = R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); obj->info.cp_string.string_idx = R_BIN_JAVA_USHORT (buffer, 1); } return obj; } R_API ut64 r_bin_java_string_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_string.string_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_integer_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_INTEGER, tag, sz, "Integer"); if (quick_check > 0) { return obj; } obj = (RBinJavaCPTypeObj *) R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); memset (&obj->info.cp_integer.bytes, 0, sizeof (obj->info.cp_integer.bytes)); memcpy (&obj->info.cp_integer.bytes.raw, buffer + 1, 4); } return obj; } R_API ut64 r_bin_java_integer_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_string.string_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 4; return size; } R_API RBinJavaCPTypeObj *r_bin_java_float_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_FLOAT, tag, sz, "Float"); if (quick_check > 0) { return obj; } obj = (RBinJavaCPTypeObj *) calloc (1, sizeof (RBinJavaCPTypeObj)); if (obj) { obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); memset (&obj->info.cp_float.bytes, 0, sizeof (obj->info.cp_float.bytes)); memcpy (&obj->info.cp_float.bytes.raw, buffer, 4); } return (RBinJavaCPTypeObj *) obj; } R_API ut64 r_bin_java_float_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_string.string_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 4; return size; } R_API RBinJavaCPTypeObj *r_bin_java_long_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_LONG, tag, sz, "Long"); if (quick_check > 0) { return obj; } obj = (RBinJavaCPTypeObj *) malloc (sizeof (RBinJavaCPTypeObj)); if (obj) { memset (obj, 0, sizeof (RBinJavaCPTypeObj)); obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); memset (&obj->info.cp_long.bytes, 0, sizeof (obj->info.cp_long.bytes)); memcpy (&(obj->info.cp_long.bytes), buffer + 1, 8); } return obj; } R_API ut64 r_bin_java_long_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_string.string_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 8; return size; } R_API RBinJavaCPTypeObj *r_bin_java_double_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_DOUBLE, tag, sz, "Double"); if (quick_check > 0) { return (RBinJavaCPTypeObj *) obj; } obj = (RBinJavaCPTypeObj *) malloc (sizeof (RBinJavaCPTypeObj)); if (obj) { memset (obj, 0, sizeof (RBinJavaCPTypeObj)); obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); memset (&obj->info.cp_double.bytes, 0, sizeof (obj->info.cp_double.bytes)); memcpy (&obj->info.cp_double.bytes, buffer + 1, 8); } return obj; } R_API ut64 r_bin_java_double_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; // tag size += 1; // obj->info.cp_string.string_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 8; return size; } R_API RBinJavaCPTypeObj *r_bin_java_utf8_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_UTF8, tag, sz, "Utf8"); if (quick_check > 0) { return NULL; } if ((obj = R_NEW0 (RBinJavaCPTypeObj))) { obj->tag = tag; obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name); obj->info.cp_utf8.length = R_BIN_JAVA_USHORT (buffer, 1); obj->info.cp_utf8.bytes = (ut8 *) malloc (obj->info.cp_utf8.length + 1); if (obj->info.cp_utf8.bytes) { memset (obj->info.cp_utf8.bytes, 0, obj->info.cp_utf8.length + 1); if (obj->info.cp_utf8.length < (sz - 3)) { memcpy (obj->info.cp_utf8.bytes, buffer + 3, (sz - 3)); obj->info.cp_utf8.length = sz - 3; } else { memcpy (obj->info.cp_utf8.bytes, buffer + 3, obj->info.cp_utf8.length); } obj->value = obj->info.cp_utf8.bytes; } else { r_bin_java_obj_free (obj); obj = NULL; } } return obj; } R_API ut64 r_bin_java_utf8_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; size += 1; if (obj && R_BIN_JAVA_CP_UTF8 == obj->tag) { size += 2; size += obj->info.cp_utf8.length; } return size; } R_API RBinJavaCPTypeObj *r_bin_java_name_and_type_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj = NULL; int quick_check = 0; quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_NAMEANDTYPE, tag, sz, "RBinJavaCPTypeNameAndType"); if (quick_check > 0) { return obj; } obj = R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name);; obj->tag = tag; obj->info.cp_name_and_type.name_idx = R_BIN_JAVA_USHORT (buffer, 1); obj->info.cp_name_and_type.descriptor_idx = R_BIN_JAVA_USHORT (buffer, 3); } return obj; } R_API ut64 r_bin_java_name_and_type_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; if (obj) { size += 1; // obj->info.cp_name_and_type.name_idx = R_BIN_JAVA_USHORT (buffer, 1); size += 2; // obj->info.cp_name_and_type.descriptor_idx = R_BIN_JAVA_USHORT (buffer, 3); size += 2; } return size; } R_API RBinJavaCPTypeObj *r_bin_java_methodtype_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_METHODTYPE, tag, sz, "RBinJavaCPTypeMethodType"); if (quick_check > 0) { return NULL; } RBinJavaCPTypeObj *obj = R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name);; obj->tag = tag; obj->info.cp_method_type.descriptor_index = R_BIN_JAVA_USHORT (buffer, 1); } return obj; } R_API ut64 r_bin_java_methodtype_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; size += 1; // obj->info.cp_method_type.descriptor_index = R_BIN_JAVA_USHORT (buffer, 1); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_methodhandle_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_METHODHANDLE, tag, sz, "RBinJavaCPTypeMethodHandle"); if (quick_check > 0) { return NULL; } RBinJavaCPTypeObj *obj = R_NEW0 (RBinJavaCPTypeObj); if (obj) { obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name);; obj->tag = tag; obj->info.cp_method_handle.reference_kind = buffer[1]; obj->info.cp_method_handle.reference_index = R_BIN_JAVA_USHORT (buffer, 2); } return obj; } R_API ut64 r_bin_java_methodhandle_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; size += 1; // obj->info.cp_method_handle.reference_index = R_BIN_JAVA_USHORT (buffer, 2); size += 2; return size; } R_API RBinJavaCPTypeObj *r_bin_java_invokedynamic_cp_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz) { ut8 tag = buffer[0]; RBinJavaCPTypeObj *obj; int quick_check = r_bin_java_quick_check (R_BIN_JAVA_CP_INVOKEDYNAMIC, tag, sz, "RBinJavaCPTypeMethodHandle"); if (quick_check > 0) { return NULL; } if ((obj = R_NEW0 (RBinJavaCPTypeObj))) { obj->metas = R_NEW0 (RBinJavaMetaInfo); obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; obj->name = r_str_dup (NULL, (const char *) R_BIN_JAVA_CP_METAS[tag].name);; obj->tag = tag; obj->info.cp_invoke_dynamic.bootstrap_method_attr_index = R_BIN_JAVA_USHORT (buffer, 1); obj->info.cp_invoke_dynamic.name_and_type_index = R_BIN_JAVA_USHORT (buffer, 3); } return obj; } R_API int r_bin_java_check_reset_cp_obj(RBinJavaCPTypeObj *cp_obj, ut8 tag) { bool res = false; if (tag < R_BIN_JAVA_CP_METAS_SZ) { if (tag != cp_obj->tag) { if (cp_obj->tag == R_BIN_JAVA_CP_UTF8) { R_FREE (cp_obj->info.cp_utf8.bytes); cp_obj->info.cp_utf8.length = 0; R_FREE (cp_obj->name); } cp_obj->tag = tag; cp_obj->metas->type_info = (void *) &R_BIN_JAVA_CP_METAS[tag]; cp_obj->name = strdup (R_BIN_JAVA_CP_METAS[tag].name); res = true; } else { eprintf ("Invalid tag\n"); } } else { eprintf ("Invalid tag '%d'.\n", tag); } return res; } R_API ut8 *r_bin_java_cp_get_4bytes(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len) { ut8 *buffer = malloc (5); if (!buffer) { return NULL; } ut32 val = 0; if (!buffer || len < 4) { if (out_sz) { *out_sz = 0; } free (buffer); return NULL; } buffer[0] = tag; val = R_BIN_JAVA_UINT (buf, 0); memcpy (buffer + 1, (const char *) &val, 4); *out_sz = 5; return buffer; } R_API ut8 *r_bin_java_cp_get_8bytes(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len) { ut8 *buffer = malloc (10); if (!buffer) { return NULL; } ut64 val = 0; if (len < 8) { *out_sz = 0; free (buffer); return NULL; } buffer[0] = tag; val = r_bin_java_raw_to_long (buf, 0); memcpy (buffer + 1, (const char *) &val, 8); *out_sz = 9; return buffer; } R_API ut8 *r_bin_java_cp_append_classref_and_name(RBinJavaObj *bin, ut32 *out_sz, const char *classname, const ut32 classname_len) { ut16 use_name_idx = bin->cp_idx + 1; ut8 *bytes = NULL, *name_bytes = NULL; name_bytes = r_bin_java_cp_get_utf8 (R_BIN_JAVA_CP_UTF8, out_sz, (const ut8 *) classname, classname_len); if (*out_sz > 0 && name_bytes) { ut8 *idx_addr = (ut8 *) &use_name_idx; bytes = malloc (*out_sz + 3); memcpy (bytes, name_bytes, *out_sz); bytes[*out_sz + 0] = R_BIN_JAVA_CP_CLASS; bytes[*out_sz + 1] = idx_addr[1]; bytes[*out_sz + 2] = idx_addr[0]; *out_sz += 3; } free (name_bytes); return bytes; } R_API ut8 *r_bin_java_cp_get_fref_bytes(RBinJavaObj *bin, ut32 *out_sz, ut8 tag, ut16 cn_idx, ut16 fn_idx, ut16 ft_idx) { ut8 *bytes = NULL, *fnt_bytes = NULL; RBinJavaCPTypeObj *ref_cp_obj = NULL; ut16 fnt_idx = 0, cref_idx = 0; ut32 fnt_len = 0; ut16 ref_cp_obj_idx = r_bin_java_find_cp_class_ref_from_name_idx (bin, cn_idx); if (!ref_cp_obj_idx) { return NULL; } ref_cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, ref_cp_obj_idx); if (ref_cp_obj) { cref_idx = ref_cp_obj->idx; } ref_cp_obj = r_bin_java_find_cp_name_and_type_info (bin, fn_idx, ft_idx); if (ref_cp_obj) { fnt_idx = ref_cp_obj->idx; } else { fnt_bytes = r_bin_java_cp_get_name_type (bin, &fnt_len, fn_idx, ft_idx); fnt_idx = bin->cp_idx + 1; } if (cref_idx && fnt_idx) { bytes = r_bin_java_cp_get_fm_ref (bin, out_sz, tag, cref_idx, fnt_idx); if (fnt_bytes) { ut8 *tbuf = malloc (fnt_len + *out_sz); if (!tbuf) { free (bytes); free (fnt_bytes); return NULL; } // copy the bytes to the new buffer memcpy (tbuf, fnt_bytes, fnt_len); memcpy (tbuf + fnt_len, bytes, *out_sz); // update the values free old buffer *out_sz += fnt_len; free (bytes); bytes = tbuf; } } free (fnt_bytes); return bytes; } R_API ut8 *r_bin_java_cp_get_classref(RBinJavaObj *bin, ut32 *out_sz, const char *classname, const ut32 classname_len, const ut16 name_idx) { ut16 use_name_idx = -1; ut8 *bytes = NULL; if (name_idx == (ut16) - 1 && classname && *classname && classname_len > 0) { // find class_name_idx by class name RList *results = r_bin_java_find_cp_const_by_val_utf8 (bin, (const ut8 *) classname, classname_len); if (r_list_length (results) == 1) { use_name_idx = (ut16) * ((ut32 *) r_list_get_n (results, 0)); } r_list_free (results); } else if (name_idx != (ut16) - 1 && name_idx != 0) { use_name_idx = name_idx; } if (use_name_idx == (ut16) - 1 && classname && *classname && classname_len > 0) { bytes = r_bin_java_cp_append_classref_and_name (bin, out_sz, classname, classname_len); } else if (use_name_idx != (ut16) - 1) { ut8 *idx_addr = (ut8 *) &use_name_idx; bytes = malloc (3); if (!bytes) { return NULL; } bytes[0] = R_BIN_JAVA_CP_CLASS; bytes[1] = idx_addr[1]; bytes[2] = idx_addr[0]; *out_sz += 3; } return bytes; } R_API ut8 *r_bin_java_cp_get_fm_ref(RBinJavaObj *bin, ut32 *out_sz, ut8 tag, ut16 class_idx, ut16 name_and_type_idx) { return r_bin_java_cp_get_2_ut16 (bin, out_sz, tag, class_idx, name_and_type_idx); } R_API ut8 *r_bin_java_cp_get_2_ut16(RBinJavaObj *bin, ut32 *out_sz, ut8 tag, ut16 ut16_one, ut16 ut16_two) { ut8 *bytes = malloc (7); if (!bytes) { return NULL; } ut8 *idx_addr = NULL; bytes[*out_sz] = tag; *out_sz += 1; idx_addr = (ut8 *) &ut16_one; bytes[*out_sz + 1] = idx_addr[1]; bytes[*out_sz + 2] = idx_addr[0]; *out_sz += 3; idx_addr = (ut8 *) &ut16_two; bytes[*out_sz + 1] = idx_addr[1]; bytes[*out_sz + 2] = idx_addr[0]; *out_sz += 3; return bytes; } R_API ut8 *r_bin_java_cp_get_name_type(RBinJavaObj *bin, ut32 *out_sz, ut16 name_idx, ut16 type_idx) { return r_bin_java_cp_get_2_ut16 (bin, out_sz, R_BIN_JAVA_CP_NAMEANDTYPE, name_idx, type_idx); } R_API ut8 *r_bin_java_cp_get_utf8(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len) { ut8 *buffer = NULL; ut16 sz = 0; ut16 t = (ut16) len; if (len > 0 && len > (ut16) - 1) { *out_sz = 0; return NULL; } sz = R_BIN_JAVA_USHORT (((ut8 *) (ut16 *) &t), 0); *out_sz = 3 + t; // tag + sz + bytes buffer = malloc (*out_sz + 3); if (!buffer) { return NULL; } // XXX - excess bytes are created to ensure null for string operations. memset (buffer, 0, *out_sz + 3); buffer[0] = tag; memcpy (buffer + 1, (const char *) &sz, 2); memcpy (buffer + 3, buf, *out_sz - 3); return buffer; } R_API ut64 r_bin_java_invokedynamic_cp_calc_size(RBinJavaCPTypeObj *obj) { ut64 size = 0; size += 1; // obj->info.cp_invoke_dynamic.bootstrap_method_attr_index = R_BIN_JAVA_USHORT (buffer, 1); size += 2; // obj->info.cp_invoke_dynamic.name_and_type_index = R_BIN_JAVA_USHORT (buffer, 3); size += 2; return size; } // End new Constant Pool types // Start free Constant Pool types R_API void r_bin_java_default_free(void /* RBinJavaCPTypeObj*/ *o) { RBinJavaCPTypeObj *obj = o; if (obj) { free (obj->metas); free (obj->name); free (obj->value); free (obj); } } R_API void r_bin_java_utf8_info_free(void /* RBinJavaCPTypeObj*/ *o) { RBinJavaCPTypeObj *obj = o; if (obj) { free (obj->name); free (obj->metas); free (obj->info.cp_utf8.bytes); free (obj); } } // Deallocs for type objects R_API void r_bin_java_obj_free(void /*RBinJavaCPTypeObj*/ *o) { RBinJavaCPTypeObj *obj = o; ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->delete_obj (obj); } R_API void r_bin_java_print_attr_summary(RBinJavaAttrInfo *attr) { if (attr == NULL) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *.\n"); return; } ((RBinJavaAttrMetas *) attr->metas->type_info)->allocs->print_summary (attr); } R_API void r_bin_java_print_source_debug_attr_summary(RBinJavaAttrInfo *attr) { ut32 i = 0; if (attr == NULL) { eprintf ("Attempting to print an invalid RBinJavaSourceDebugExtensionAttr *.\n"); return; } Eprintf ("Source Debug Extension Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Extension Length: %d\n", attr->length); Eprintf (" Source Debug Extension value: \n"); for (i = 0; i < attr->length; i++) { Eprintf ("%c", attr->info.debug_extensions.debug_extension[i]); } Eprintf ("\n Source Debug Extension End\n"); } R_API void r_bin_java_print_unknown_attr_summary(RBinJavaAttrInfo *attr) { if (attr == NULL) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *Unknown.\n"); return; } Eprintf ("Unknown Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); } R_API void r_bin_java_print_code_exceptions_attr_summary(RBinJavaExceptionEntry *exc_entry) { if (exc_entry == NULL) { eprintf ("Attempting to print an invalid RBinJavaExceptionEntry *.\n"); return; } Eprintf (" Exception Table Entry Information\n"); Eprintf (" offset: 0x%08"PFMT64x"\n", exc_entry->file_offset); Eprintf (" catch_type: %d\n", exc_entry->catch_type); Eprintf (" start_pc: 0x%04x\n", exc_entry->start_pc); Eprintf (" end_pc: 0x%04x\n", exc_entry->end_pc); Eprintf (" handler_pc: 0x%04x\n", exc_entry->handler_pc); } // End free Constant Pool types R_API void r_bin_java_print_code_attr_summary(RBinJavaAttrInfo *attr) { RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaExceptionEntry *exc_entry = NULL; RBinJavaAttrInfo *_attr = NULL; if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *Code.\n"); return; } Eprintf ("Code Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d, Attribute Count: %d\n", attr->length, attr->info.code_attr.attributes_count); Eprintf (" Max Stack: %d\n", attr->info.code_attr.max_stack); Eprintf (" Max Locals: %d\n", attr->info.code_attr.max_locals); Eprintf (" Code Length: %d\n", attr->info.code_attr.code_length); Eprintf (" Code At Offset: 0x%08"PFMT64x "\n", (ut64) attr->info.code_attr.code_offset); Eprintf ("Code Attribute Exception Table Information:\n"); Eprintf (" Exception Table Length: %d\n", attr->info.code_attr.exception_table_length); if (attr->info.code_attr.exception_table) { // Delete the attr entries r_list_foreach_safe (attr->info.code_attr.exception_table, iter, iter_tmp, exc_entry) { r_bin_java_print_code_exceptions_attr_summary (exc_entry); } } Eprintf (" Implicit Method Stack Frame:\n"); r_bin_java_print_stack_map_frame_summary (attr->info.code_attr.implicit_frame); Eprintf ("Code Attribute Attributes Information:\n"); if (attr->info.code_attr.attributes && attr->info.code_attr.attributes_count > 0) { Eprintf (" Code Attribute Attributes Count: %d\n", attr->info.code_attr.attributes_count); r_list_foreach_safe (attr->info.code_attr.attributes, iter, iter_tmp, _attr) { r_bin_java_print_attr_summary (_attr); } } } R_API void r_bin_java_print_constant_value_attr_summary(RBinJavaAttrInfo *attr) { if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *ConstantValue.\n"); return; } Eprintf ("Constant Value Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); Eprintf (" ConstantValue Index: %d\n", attr->info.constant_value_attr.constantvalue_idx); } R_API void r_bin_java_print_deprecated_attr_summary(RBinJavaAttrInfo *attr) { if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *Deperecated.\n"); return; } Eprintf ("Deperecated Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); } R_API void r_bin_java_print_enclosing_methods_attr_summary(RBinJavaAttrInfo *attr) { if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *Deperecated.\n"); return; } Eprintf ("Enclosing Method Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); Eprintf (" Class Info Index : 0x%02x\n", attr->info.enclosing_method_attr.class_idx); Eprintf (" Method Name and Type Index : 0x%02x\n", attr->info.enclosing_method_attr.method_idx); Eprintf (" Class Name : %s\n", attr->info.enclosing_method_attr.class_name); Eprintf (" Method Name and Desc : %s %s\n", attr->info.enclosing_method_attr.method_name, attr->info.enclosing_method_attr.method_descriptor); } R_API void r_bin_java_print_exceptions_attr_summary(RBinJavaAttrInfo *attr) { ut32 i = 0; if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *Exceptions.\n"); return; } Eprintf ("Exceptions Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); for (i = 0; i < attr->info.exceptions_attr.number_of_exceptions; i++) { Eprintf (" Exceptions Attribute Index[%d]: %d\n", i, attr->info.exceptions_attr.exception_idx_table[i]); } } R_API void r_bin_java_print_classes_attr_summary(RBinJavaClassesAttribute *icattr) { if (!icattr) { eprintf ("Attempting to print an invalid RBinJavaClassesAttribute* (InnerClasses element).\n"); return; } eprintf (" Inner Classes Class Attribute Offset: 0x%08"PFMT64x "\n", icattr->file_offset); eprintf (" Inner Classes Class Attribute Class Name (%d): %s\n", icattr->inner_name_idx, icattr->name); eprintf (" Inner Classes Class Attribute Class inner_class_info_idx: %d\n", icattr->inner_class_info_idx); eprintf (" Inner Classes Class Attribute Class inner_class_access_flags: 0x%02x %s\n", icattr->inner_class_access_flags, icattr->flags_str); eprintf (" Inner Classes Class Attribute Class outer_class_info_idx: %d\n", icattr->outer_class_info_idx); eprintf (" Inner Classes Class Field Information:\n"); r_bin_java_print_field_summary (icattr->clint_field); eprintf (" Inner Classes Class Field Information:\n"); r_bin_java_print_field_summary (icattr->clint_field); eprintf (" Inner Classes Class Attr Info Information:\n"); r_bin_java_print_attr_summary (icattr->clint_attr); } R_API void r_bin_java_print_inner_classes_attr_summary(RBinJavaAttrInfo *attr) { RBinJavaClassesAttribute *icattr; RListIter *iter, *iter_tmp; if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *InnerClasses.\n"); return; } Eprintf ("Inner Classes Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); r_list_foreach_safe (attr->info.inner_classes_attr.classes, iter, iter_tmp, icattr) { r_bin_java_print_classes_attr_summary (icattr); } } R_API void r_bin_java_print_line_number_attr_summary(RBinJavaLineNumberAttribute *lnattr) { if (!lnattr) { eprintf ("Attempting to print an invalid RBinJavaLineNumberAttribute *.\n"); return; } Eprintf (" Line Number Attribute Offset: 0x%08"PFMT64x "\n", lnattr->file_offset); Eprintf (" Line Number Attribute StartPC: %d\n", lnattr->start_pc); Eprintf (" Line Number Attribute LineNumber: %d\n", lnattr->line_number); } R_API void r_bin_java_print_line_number_table_attr_summary(RBinJavaAttrInfo *attr) { RBinJavaLineNumberAttribute *lnattr; RListIter *iter, *iter_tmp; if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *LineNumberTable.\n"); return; } Eprintf ("Line Number Table Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); r_list_foreach_safe (attr->info.line_number_table_attr.line_number_table, iter, iter_tmp, lnattr) { r_bin_java_print_line_number_attr_summary (lnattr); } } R_API void r_bin_java_print_local_variable_attr_summary(RBinJavaLocalVariableAttribute *lvattr) { if (!lvattr) { eprintf ("Attempting to print an invalid RBinJavaLocalVariableAttribute *.\n"); return; } Eprintf (" Local Variable Attribute offset: 0x%08"PFMT64x "\n", lvattr->file_offset); Eprintf (" Local Variable Attribute start_pc: %d\n", lvattr->start_pc); Eprintf (" Local Variable Attribute Length: %d\n", lvattr->length); Eprintf (" Local Variable Attribute name_idx: %d\n", lvattr->name_idx); Eprintf (" Local Variable Attribute name: %s\n", lvattr->name); Eprintf (" Local Variable Attribute descriptor_idx: %d\n", lvattr->descriptor_idx); Eprintf (" Local Variable Attribute descriptor: %s\n", lvattr->descriptor); Eprintf (" Local Variable Attribute index: %d\n", lvattr->index); } R_API void r_bin_java_print_local_variable_table_attr_summary(RBinJavaAttrInfo *attr) { RBinJavaLocalVariableAttribute *lvattr; RListIter *iter, *iter_tmp; if (attr == NULL) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *LocalVariableTable.\n"); return; } Eprintf ("Local Variable Table Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); r_list_foreach_safe (attr->info.local_variable_table_attr.local_variable_table, iter, iter_tmp, lvattr) { r_bin_java_print_local_variable_attr_summary (lvattr); } } R_API void r_bin_java_print_local_variable_type_attr_summary(RBinJavaLocalVariableTypeAttribute *lvattr) { if (!lvattr) { eprintf ("Attempting to print an invalid RBinJavaLocalVariableTypeAttribute *.\n"); return; } eprintf (" Local Variable Type Attribute offset: 0x%08"PFMT64x "\n", lvattr->file_offset); eprintf (" Local Variable Type Attribute start_pc: %d\n", lvattr->start_pc); eprintf (" Local Variable Type Attribute Length: %d\n", lvattr->length); eprintf (" Local Variable Type Attribute name_idx: %d\n", lvattr->name_idx); eprintf (" Local Variable Type Attribute name: %s\n", lvattr->name); eprintf (" Local Variable Type Attribute signature_idx: %d\n", lvattr->signature_idx); eprintf (" Local Variable Type Attribute signature: %s\n", lvattr->signature); eprintf (" Local Variable Type Attribute index: %d\n", lvattr->index); } R_API void r_bin_java_print_local_variable_type_table_attr_summary(RBinJavaAttrInfo *attr) { RBinJavaLocalVariableTypeAttribute *lvtattr; RListIter *iter, *iter_tmp; if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *LocalVariableTable.\n"); return; } eprintf ("Local Variable Type Table Attribute Information:\n"); eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); eprintf (" Attribute Length: %d\n", attr->length); r_list_foreach_safe (attr->info.local_variable_type_table_attr.local_variable_table, iter, iter_tmp, lvtattr) { r_bin_java_print_local_variable_type_attr_summary (lvtattr); } } R_API void r_bin_java_print_signature_attr_summary(RBinJavaAttrInfo *attr) { if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *SignatureAttr.\n"); return; } Eprintf ("Signature Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); Eprintf (" Signature UTF8 Index: %d\n", attr->info.signature_attr.signature_idx); Eprintf (" Signature string: %s\n", attr->info.signature_attr.signature); } R_API void r_bin_java_print_source_code_file_attr_summary(RBinJavaAttrInfo *attr) { if (!attr) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *SourceFile.\n"); return; } Eprintf ("Source File Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); Eprintf (" Source File Index: %d\n", attr->info.source_file_attr.sourcefile_idx); } R_API void r_bin_java_print_synthetic_attr_summary(RBinJavaAttrInfo *attr) { if (attr == NULL) { eprintf ("Attempting to print an invalid RBinJavaAttrInfo *Synthetic.\n"); return; } Eprintf ("Synthetic Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); Eprintf (" Attribute Index: %d\n", attr->info.source_file_attr.sourcefile_idx); } R_API void r_bin_java_print_stack_map_table_attr_summary(RBinJavaAttrInfo *attr) { RListIter *iter, *iter_tmp; RList *ptrList; RBinJavaStackMapFrame *frame; if (attr == NULL) { eprintf ("Attempting to print an invalid RBinJavaStackMapTableAttr* .\n"); return; } Eprintf ("StackMapTable Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); Eprintf (" StackMapTable Method Code Size: 0x%08x\n", attr->info.stack_map_table_attr.code_size); Eprintf (" StackMapTable Frame Entries: 0x%08x\n", attr->info.stack_map_table_attr.number_of_entries); Eprintf (" StackMapTable Frames:\n"); ptrList = attr->info.stack_map_table_attr.stack_map_frame_entries; if (ptrList) { r_list_foreach_safe (ptrList, iter, iter_tmp, frame) { r_bin_java_print_stack_map_frame_summary (frame); } } } R_API void r_bin_java_print_stack_map_frame_summary(RBinJavaStackMapFrame *obj) { RListIter *iter, *iter_tmp; RList *ptrList; RBinJavaVerificationObj *ver_obj; if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaStackMapFrame* .\n"); return; } Eprintf ("Stack Map Frame Information\n"); Eprintf (" Tag Value = 0x%02x Name: %s\n", obj->tag, ((RBinJavaStackMapFrameMetas *) obj->metas->type_info)->name); Eprintf (" Offset: 0x%08"PFMT64x "\n", obj->file_offset); Eprintf (" Local Variable Count = 0x%04x\n", obj->number_of_locals); Eprintf (" Stack Items Count = 0x%04x\n", obj->number_of_stack_items); Eprintf (" Local Variables:\n"); ptrList = obj->local_items; r_list_foreach_safe (ptrList, iter, iter_tmp, ver_obj) { r_bin_java_print_verification_info_summary (ver_obj); } Eprintf (" Stack Items:\n"); ptrList = obj->stack_items; r_list_foreach_safe (ptrList, iter, iter_tmp, ver_obj) { r_bin_java_print_verification_info_summary (ver_obj); } } R_API void r_bin_java_print_verification_info_summary(RBinJavaVerificationObj *obj) { ut8 tag_value = R_BIN_JAVA_STACKMAP_UNKNOWN; if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaVerificationObj* .\n"); return; } if (obj->tag < R_BIN_JAVA_STACKMAP_UNKNOWN) { tag_value = obj->tag; } Eprintf ("Verification Information\n"); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" Tag Value = 0x%02x\n", obj->tag); Eprintf (" Name = %s\n", R_BIN_JAVA_VERIFICATION_METAS[tag_value].name); if (obj->tag == R_BIN_JAVA_STACKMAP_OBJECT) { Eprintf (" Object Constant Pool Index = 0x%x\n", obj->info.obj_val_cp_idx); } else if (obj->tag == R_BIN_JAVA_STACKMAP_UNINIT) { Eprintf (" Uninitialized Object offset in code = 0x%x\n", obj->info.uninit_offset); } } R_API void r_bin_java_print_field_summary(RBinJavaField *field) { RBinJavaAttrInfo *attr; RListIter *iter, *iter_tmp; if (field) { if (field->type == R_BIN_JAVA_FIELD_TYPE_METHOD) { r_bin_java_print_method_summary (field); } else { #if 0 r_bin_java_print_interface_summary (field); return; } * / #endif Eprintf ("Field Summary Information:\n"); Eprintf (" File Offset: 0x%08"PFMT64x "\n", field->file_offset); Eprintf (" Name Index: %d (%s)\n", field->name_idx, field->name); Eprintf (" Descriptor Index: %d (%s)\n", field->descriptor_idx, field->descriptor); Eprintf (" Access Flags: 0x%02x (%s)\n", field->flags, field->flags_str); Eprintf (" Field Attributes Count: %d\n", field->attr_count); Eprintf (" Field Attributes:\n"); r_list_foreach_safe (field->attributes, iter, iter_tmp, attr) { r_bin_java_print_attr_summary (attr); } } } else { eprintf ("Attempting to print an invalid RBinJavaField* Field.\n"); } } R_API void r_bin_java_print_method_summary(RBinJavaField *field) { RBinJavaAttrInfo *attr; RListIter *iter, *iter_tmp; if (field == NULL) { eprintf ("Attempting to print an invalid RBinJavaField* Method.\n"); return; } Eprintf ("Method Summary Information:\n"); Eprintf (" File Offset: 0x%08"PFMT64x "\n", field->file_offset); Eprintf (" Name Index: %d (%s)\n", field->name_idx, field->name); Eprintf (" Descriptor Index: %d (%s)\n", field->descriptor_idx, field->descriptor); Eprintf (" Access Flags: 0x%02x (%s)\n", field->flags, field->flags_str); Eprintf (" Method Attributes Count: %d\n", field->attr_count); Eprintf (" Method Attributes:\n"); r_list_foreach_safe (field->attributes, iter, iter_tmp, attr) { r_bin_java_print_attr_summary (attr); } } /* R_API void r_bin_java_print_interface_summary(ut16 idx) {//RBinJavaField *field) { RBinJavaAttrInfo *attr; RBinJavaCPTypeObj *class_info; RListIter *iter, *iter_tmp; if (field == NULL) { eprintf ("Attempting to print an invalid RBinJavaField* Interface.\n"); return; } eprintf ("Interface Summary Information:\n"); eprintf (" File offset: 0x%08"PFMT64x"", field->file_offset); eprintf (" Access Flags: %d\n", field->flags); eprintf (" Name Index: %d (%s)\n", field->name_idx, field->name); eprintf (" Descriptor Index: %d (%s)\n", field->descriptor_idx, field->descriptor); eprintf (" Interface Attributes Count: %d\n", field->attr_count); eprintf (" Interface Attributes:\n"); r_list_foreach_safe (field->attributes, iter, iter_tmp, attr) { r_bin_java_print_attr_summary(attr); } } */ R_API void r_bin_java_print_interfacemethodref_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* InterfaceMethodRef.\n"); return; } eprintf ("InterfaceMethodRef ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" Class Index = %d\n", obj->info.cp_interface.class_idx); eprintf (" Name and type Index = %d\n", obj->info.cp_interface.name_and_type_idx); } R_API char *r_bin_java_print_interfacemethodref_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_interface.class_idx, obj->info.cp_interface.name_and_type_idx); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_interface.class_idx, obj->info.cp_interface.name_and_type_idx); } } } return value; } R_API void r_bin_java_print_methodhandle_cp_summary(RBinJavaCPTypeObj *obj) { ut8 ref_kind; if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* RBinJavaCPTypeMethodHandle.\n"); return; } ref_kind = obj->info.cp_method_handle.reference_kind; eprintf ("MethodHandle ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" Reference Kind = (0x%02x) %s\n", ref_kind, R_BIN_JAVA_REF_METAS[ref_kind].name); eprintf (" Reference Index = %d\n", obj->info.cp_method_handle.reference_index); } R_API char *r_bin_java_print_methodhandle_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); ut8 ref_kind = obj->info.cp_method_handle.reference_kind; if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, R_BIN_JAVA_REF_METAS[ref_kind].name, obj->info.cp_method_handle.reference_index); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, R_BIN_JAVA_REF_METAS[ref_kind].name, obj->info.cp_method_handle.reference_index); } } } return value; } R_API void r_bin_java_print_methodtype_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* RBinJavaCPTypeMethodType.\n"); return; } Eprintf ("MethodType ConstantPool Type (%d) ", obj->metas->ord); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" Descriptor Index = 0x%02x\n", obj->info.cp_method_type.descriptor_index); } R_API char *r_bin_java_print_methodtype_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_method_type.descriptor_index); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_method_type.descriptor_index); } } } return value; } R_API void r_bin_java_print_invokedynamic_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* RBinJavaCPTypeInvokeDynamic.\n"); return; } eprintf ("InvokeDynamic ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" Bootstrap Method Attr Index = (0x%02x)\n", obj->info.cp_invoke_dynamic.bootstrap_method_attr_index); eprintf (" Bootstrap Name and Type Index = (0x%02x)\n", obj->info.cp_invoke_dynamic.name_and_type_index); } R_API char *r_bin_java_print_invokedynamic_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_invoke_dynamic.bootstrap_method_attr_index, obj->info.cp_invoke_dynamic.name_and_type_index); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_invoke_dynamic.bootstrap_method_attr_index, obj->info.cp_invoke_dynamic.name_and_type_index); } } } return value; } R_API void r_bin_java_print_methodref_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* MethodRef.\n"); return; } eprintf ("MethodRef ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" Class Index = %d\n", obj->info.cp_method.class_idx); eprintf (" Name and type Index = %d\n", obj->info.cp_method.name_and_type_idx); } R_API char *r_bin_java_print_methodref_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_method.class_idx, obj->info.cp_method.name_and_type_idx); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_method.class_idx, obj->info.cp_method.name_and_type_idx); } } } return value; } R_API void r_bin_java_print_fieldref_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* FieldRef.\n"); return; } eprintf ("FieldRef ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" Class Index = %d\n", obj->info.cp_field.class_idx); eprintf (" Name and type Index = %d\n", obj->info.cp_field.name_and_type_idx); } R_API char *r_bin_java_print_fieldref_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_field.class_idx, obj->info.cp_field.name_and_type_idx); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_field.class_idx, obj->info.cp_field.name_and_type_idx); } } } return value; } R_API void r_bin_java_print_classref_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* ClassRef.\n"); return; } eprintf ("ClassRef ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" Name Index = %d\n", obj->info.cp_class.name_idx); } R_API char *r_bin_java_print_classref_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_class.name_idx); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_class.name_idx); } } } return value; } R_API void r_bin_java_print_string_cp_summary(RBinJavaCPTypeObj *obj) { if (!obj) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* String.\n"); return; } Eprintf ("String ConstantPool Type (%d) ", obj->metas->ord); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" String Index = %d\n", obj->info.cp_string.string_idx); } R_API char *r_bin_java_print_string_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_string.string_idx); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d", obj->metas->ord, obj->file_offset, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_string.string_idx); } } } return value; } R_API void r_bin_java_print_integer_cp_summary(RBinJavaCPTypeObj *obj) { ut8 *b = NULL; if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* Integer.\n"); return; } b = obj->info.cp_integer.bytes.raw; eprintf ("Integer ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" bytes = %02x %02x %02x %02x\n", b[0], b[1], b[2], b[3]); eprintf (" integer = %d\n", R_BIN_JAVA_UINT (obj->info.cp_integer.bytes.raw, 0)); } R_API char *r_bin_java_print_integer_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.0x%08x", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, R_BIN_JAVA_UINT (obj->info.cp_integer.bytes.raw, 0)); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.0x%08x", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, R_BIN_JAVA_UINT (obj->info.cp_integer.bytes.raw, 0)); } } } return value; } R_API void r_bin_java_print_float_cp_summary(RBinJavaCPTypeObj *obj) { ut8 *b = NULL; if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* Double.\n"); return; } b = obj->info.cp_float.bytes.raw; Eprintf ("Float ConstantPool Type (%d) ", obj->metas->ord); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" Bytes = %02x %02x %02x %02x\n", b[0], b[1], b[2], b[3]); Eprintf (" Float = %f\n", R_BIN_JAVA_FLOAT (obj->info.cp_float.bytes.raw, 0)); } R_API char *r_bin_java_print_float_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%f", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, R_BIN_JAVA_FLOAT (obj->info.cp_float.bytes.raw, 0)); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%f", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, R_BIN_JAVA_FLOAT (obj->info.cp_float.bytes.raw, 0)); } } } return value; } R_API void r_bin_java_print_long_cp_summary(RBinJavaCPTypeObj *obj) { ut8 *b = NULL; if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* Long.\n"); return; } b = obj->info.cp_long.bytes.raw; Eprintf ("Long ConstantPool Type (%d) ", obj->metas->ord); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" High-Bytes = %02x %02x %02x %02x\n", b[0], b[1], b[2], b[3]); Eprintf (" Low-Bytes = %02x %02x %02x %02x\n", b[4], b[5], b[6], b[7]); Eprintf (" Long = %08"PFMT64x "\n", r_bin_java_raw_to_long (obj->info.cp_long.bytes.raw, 0)); } R_API char *r_bin_java_print_long_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.0x%08"PFMT64x "", obj->metas->ord, obj->file_offset, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, r_bin_java_raw_to_long (obj->info.cp_long.bytes.raw, 0)); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.0x%08"PFMT64x "", obj->metas->ord, obj->file_offset, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, r_bin_java_raw_to_long (obj->info.cp_long.bytes.raw, 0)); } } } return value; } R_API void r_bin_java_print_double_cp_summary(RBinJavaCPTypeObj *obj) { ut8 *b = NULL; if (!obj) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* Double.\n"); return; } b = obj->info.cp_double.bytes.raw; Eprintf ("Double ConstantPool Type (%d) ", obj->metas->ord); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" High-Bytes = %02x %02x %02x %02x\n", b[0], b[1], b[2], b[3]); Eprintf (" Low-Bytes = %02x %02x %02x %02x\n", b[4], b[5], b[6], b[7]); Eprintf (" Double = %f\n", r_bin_java_raw_to_double (obj->info.cp_double.bytes.raw, 0)); } R_API char *r_bin_java_print_double_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%f", obj->metas->ord, obj->file_offset, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, r_bin_java_raw_to_double (obj->info.cp_double.bytes.raw, 0)); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%f", obj->metas->ord, obj->file_offset, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, r_bin_java_raw_to_double (obj->info.cp_double.bytes.raw, 0)); } } } return value; } R_API void r_bin_java_print_name_and_type_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* Name_And_Type.\n"); return; } Eprintf ("Name_And_Type ConstantPool Type (%d) ", obj->metas->ord); Eprintf (" Offset: 0x%08"PFMT64x "", obj->file_offset); Eprintf (" name_idx = (%d)\n", obj->info.cp_name_and_type.name_idx); Eprintf (" descriptor_idx = (%d)\n", obj->info.cp_name_and_type.descriptor_idx); } R_API char *r_bin_java_print_name_and_type_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_name_and_type.name_idx, obj->info.cp_name_and_type.descriptor_idx); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%d", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_name_and_type.name_idx, obj->info.cp_name_and_type.descriptor_idx); } } } return value; } R_API void r_bin_java_print_utf8_cp_summary(RBinJavaCPTypeObj *obj) { if (obj == NULL) { eprintf ("Attempting to print an invalid RBinJavaCPTypeObj* Utf8.\n"); return; } char *str = convert_string ((const char *) obj->info.cp_utf8.bytes, obj->info.cp_utf8.length); eprintf ("UTF8 ConstantPool Type (%d) ", obj->metas->ord); eprintf (" Offset: 0x%08"PFMT64x"", obj->file_offset); eprintf (" length = %d\n", obj->info.cp_utf8.length); eprintf (" utf8 = %s\n", str); free (str); } R_API char *r_bin_java_print_utf8_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *utf8_str = r_hex_bin2strdup (obj->info.cp_utf8.bytes, obj->info.cp_utf8.length); char *value = malloc (size + strlen (utf8_str)); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%s", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_utf8.length, utf8_str); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size + strlen (utf8_str)); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s.%d.%s", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name, obj->info.cp_utf8.length, utf8_str); } } } free (utf8_str); return value; } R_API void r_bin_java_print_null_cp_summary(RBinJavaCPTypeObj *obj) { eprintf ("Unknown ConstantPool Type Tag: 0x%04x .\n", obj->tag); } R_API char *r_bin_java_print_null_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255, consumed = 0; char *value = malloc (size); if (value) { memset (value, 0, size); consumed = snprintf (value, size, "%d.0x%04"PFMT64x ".%s", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name); if (consumed >= size - 1) { free (value); size += size >> 1; value = malloc (size); if (value) { memset (value, 0, size); (void)snprintf (value, size, "%d.0x%04"PFMT64x ".%s", obj->metas->ord, obj->file_offset, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name); } } } return value; } R_API void r_bin_java_print_unknown_cp_summary(RBinJavaCPTypeObj *obj) { eprintf ("NULL ConstantPool Type.\n"); } R_API char *r_bin_java_print_unknown_cp_stringify(RBinJavaCPTypeObj *obj) { ut32 size = 255; char *value = malloc (size); if (value) { memset (value, 0, size); snprintf (value, size, "%d.0x%04"PFMT64x ".%s", obj->metas->ord, obj->file_offset + obj->loadaddr, ((RBinJavaCPTypeMetas *) obj->metas->type_info)->name); } return value; } R_API RBinJavaElementValuePair *r_bin_java_element_pair_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { if (!buffer || sz < 4) { return NULL; } RBinJavaElementValuePair *evp = R_NEW0 (RBinJavaElementValuePair); if (!evp) { return NULL; } // TODO: What is the signifigance of evp element evp->element_name_idx = R_BIN_JAVA_USHORT (buffer, 0); ut64 offset = 2; evp->file_offset = buf_offset; evp->name = r_bin_java_get_utf8_from_bin_cp_list (R_BIN_JAVA_GLOBAL_BIN, evp->element_name_idx); if (!evp->name) { // TODO: eprintf unable to find the name for the given index eprintf ("ElementValue Name is invalid.\n"); evp->name = strdup ("UNKNOWN"); } if (offset >= sz) { free (evp); return NULL; } evp->value = r_bin_java_element_value_new (buffer + offset, sz - offset, buf_offset + offset); offset += evp->value->size; if (offset >= sz) { free (evp->value); free (evp); return NULL; } evp->size = offset; return evp; } R_API void r_bin_java_print_element_pair_summary(RBinJavaElementValuePair *evp) { if (!evp) { eprintf ("Attempting to print an invalid RBinJavaElementValuePair *pair.\n"); return; } Eprintf ("Element Value Pair information:\n"); Eprintf (" EV Pair File Offset: 0x%08"PFMT64x "\n", evp->file_offset); Eprintf (" EV Pair Element Name index: 0x%02x\n", evp->element_name_idx); Eprintf (" EV Pair Element Name: %s\n", evp->name); Eprintf (" EV Pair Element Value:\n"); r_bin_java_print_element_value_summary (evp->value); } R_API void r_bin_java_print_element_value_summary(RBinJavaElementValue *element_value) { RBinJavaCPTypeObj *obj; RBinJavaElementValue *ev_element = NULL; RListIter *iter = NULL, *iter_tmp = NULL; char *name; if (!element_value) { eprintf ("Attempting to print an invalid RBinJavaElementValuePair *pair.\n"); return; } name = ((RBinJavaElementValueMetas *) element_value->metas->type_info)->name; eprintf ("Element Value information:\n"); eprintf (" EV Pair File Offset: 0x%08"PFMT64x "\n", element_value->file_offset); eprintf (" EV Value Type (%d): %s\n", element_value->tag, name); switch (element_value->tag) { case R_BIN_JAVA_EV_TAG_BYTE: case R_BIN_JAVA_EV_TAG_CHAR: case R_BIN_JAVA_EV_TAG_DOUBLE: case R_BIN_JAVA_EV_TAG_FLOAT: case R_BIN_JAVA_EV_TAG_INT: case R_BIN_JAVA_EV_TAG_LONG: case R_BIN_JAVA_EV_TAG_SHORT: case R_BIN_JAVA_EV_TAG_BOOLEAN: case R_BIN_JAVA_EV_TAG_STRING: eprintf (" EV Value Constant Value index: 0x%02x\n", element_value->value.const_value.const_value_idx); eprintf (" EV Value Constant Value Information:\n"); obj = element_value->value.const_value.const_value_cp_obj; if (obj && obj->metas && obj->metas->type_info) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->print_summary (obj); } break; case R_BIN_JAVA_EV_TAG_ENUM: eprintf (" EV Value Enum Constant Value Const Name Index: 0x%02x\n", element_value->value.enum_const_value.const_name_idx); eprintf (" EV Value Enum Constant Value Type Name Index: 0x%02x\n", element_value->value.enum_const_value.type_name_idx); eprintf (" EV Value Enum Constant Value Const CP Information:\n"); obj = element_value->value.enum_const_value.const_name_cp_obj; if (obj && obj->metas && obj->metas->type_info) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->print_summary (obj); } eprintf (" EV Value Enum Constant Value Type CP Information:\n"); obj = element_value->value.enum_const_value.type_name_cp_obj; if (obj && obj->metas && obj->metas->type_info) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->print_summary (obj); } break; case R_BIN_JAVA_EV_TAG_CLASS: eprintf (" EV Value Class Info Index: 0x%02x\n", element_value->value.class_value.class_info_idx); eprintf (" EV Value Class Info CP Information:\n"); obj = element_value->value.class_value.class_info_cp_obj; if (obj && obj->metas && obj->metas->type_info) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->print_summary (obj); } break; case R_BIN_JAVA_EV_TAG_ARRAY: eprintf (" EV Value Array Value Number of Values: 0x%04x\n", element_value->value.array_value.num_values); eprintf (" EV Value Array Values\n"); r_list_foreach_safe (element_value->value.array_value.values, iter, iter_tmp, ev_element) { r_bin_java_print_element_value_summary (ev_element); } break; case R_BIN_JAVA_EV_TAG_ANNOTATION: eprintf (" EV Annotation Information:\n"); r_bin_java_print_annotation_summary (&element_value->value.annotation_value); break; default: // eprintf unable to handle tag break; } } R_API void r_bin_java_element_pair_free(void /*RBinJavaElementValuePair*/ *e) { RBinJavaElementValuePair *evp = e; if (evp) { free (evp->name); r_bin_java_element_value_free (evp->value); free (evp); } evp = NULL; } R_API void r_bin_java_element_value_free(void /*RBinJavaElementValue*/ *e) { RBinJavaElementValue *element_value = e; RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaCPTypeObj *obj = NULL; RBinJavaElementValue *ev_element = NULL; if (element_value) { R_FREE (element_value->metas); switch (element_value->tag) { case R_BIN_JAVA_EV_TAG_BYTE: case R_BIN_JAVA_EV_TAG_CHAR: case R_BIN_JAVA_EV_TAG_DOUBLE: case R_BIN_JAVA_EV_TAG_FLOAT: case R_BIN_JAVA_EV_TAG_INT: case R_BIN_JAVA_EV_TAG_LONG: case R_BIN_JAVA_EV_TAG_SHORT: case R_BIN_JAVA_EV_TAG_BOOLEAN: case R_BIN_JAVA_EV_TAG_STRING: // Delete the CP Type Object obj = element_value->value.const_value.const_value_cp_obj; if (obj && obj->metas) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->delete_obj (obj); } break; case R_BIN_JAVA_EV_TAG_ENUM: // Delete the CP Type Objects obj = element_value->value.enum_const_value.const_name_cp_obj; if (obj && obj->metas) { RBinJavaCPTypeMetas *ti = obj->metas->type_info; if (ti && ti->allocs && ti->allocs->delete_obj) { ti->allocs->delete_obj (obj); } } obj = element_value->value.enum_const_value.type_name_cp_obj; if (obj && obj->metas) { RBinJavaCPTypeMetas *tm = obj->metas->type_info; if (tm && tm->allocs && tm->allocs->delete_obj) { tm->allocs->delete_obj (obj); } } break; case R_BIN_JAVA_EV_TAG_CLASS: // Delete the CP Type Object obj = element_value->value.class_value.class_info_cp_obj; if (obj && obj->metas) { ((RBinJavaCPTypeMetas *) obj->metas->type_info)->allocs->delete_obj (obj); } break; case R_BIN_JAVA_EV_TAG_ARRAY: // Delete the Element Value array List r_list_foreach_safe (element_value->value.array_value.values, iter, iter_tmp, ev_element) { if (ev_element) { r_bin_java_element_value_free (ev_element); } else { // TODO eprintf evps value was NULL } // r_list_delete (element_value->value.array_value.values, iter); ev_element = NULL; } r_list_free (element_value->value.array_value.values); break; case R_BIN_JAVA_EV_TAG_ANNOTATION: // Delete the Annotations List r_list_free (element_value->value.annotation_value.element_value_pairs); break; default: // eprintf unable to free the tag break; } free (element_value); } } R_API ut64 r_bin_java_annotation_default_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; if (attr) { // attr = r_bin_java_default_attr_new (buffer, sz, buf_offset); size += 6; // attr->info.annotation_default_attr.default_value = r_bin_java_element_value_new (buffer+offset, sz-offset, buf_offset+offset); size += r_bin_java_element_value_calc_size (attr->info.annotation_default_attr.default_value); } return size; } R_API RBinJavaAttrInfo *r_bin_java_annotation_default_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 0; RBinJavaAttrInfo *attr = NULL; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr && sz >= offset) { attr->type = R_BIN_JAVA_ATTR_TYPE_ANNOTATION_DEFAULT_ATTR; attr->info.annotation_default_attr.default_value = r_bin_java_element_value_new (buffer + offset, sz - offset, buf_offset + offset); if (attr->info.annotation_default_attr.default_value) { offset += attr->info.annotation_default_attr.default_value->size; } } r_bin_java_print_annotation_default_attr_summary (attr); return attr; } static void delete_obj(RBinJavaCPTypeObj *obj) { if (obj && obj->metas && obj->metas->type_info) { RBinJavaCPTypeMetas *ti = obj->metas->type_info; if (ti && ti->allocs && ti->allocs->delete_obj) { ti->allocs->delete_obj (obj); } } } R_API void r_bin_java_annotation_default_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; RBinJavaElementValue *ev_element = NULL; RListIter *iter = NULL, *iter_tmp = NULL; if (!attr || attr->type != R_BIN_JAVA_ATTR_TYPE_ANNOTATION_DEFAULT_ATTR) { return; } RBinJavaElementValue *element_value = attr->info.annotation_default_attr.default_value; if (!element_value) { return; } switch (element_value->tag) { case R_BIN_JAVA_EV_TAG_BYTE: case R_BIN_JAVA_EV_TAG_CHAR: case R_BIN_JAVA_EV_TAG_DOUBLE: case R_BIN_JAVA_EV_TAG_FLOAT: case R_BIN_JAVA_EV_TAG_INT: case R_BIN_JAVA_EV_TAG_LONG: case R_BIN_JAVA_EV_TAG_SHORT: case R_BIN_JAVA_EV_TAG_BOOLEAN: case R_BIN_JAVA_EV_TAG_STRING: // Delete the CP Type Object delete_obj (element_value->value.const_value.const_value_cp_obj); break; case R_BIN_JAVA_EV_TAG_ENUM: // Delete the CP Type Objects delete_obj (element_value->value.enum_const_value.const_name_cp_obj); break; case R_BIN_JAVA_EV_TAG_CLASS: // Delete the CP Type Object delete_obj (element_value->value.class_value.class_info_cp_obj); break; case R_BIN_JAVA_EV_TAG_ARRAY: // Delete the Element Value array List r_list_foreach_safe (element_value->value.array_value.values, iter, iter_tmp, ev_element) { r_bin_java_element_value_free (ev_element); // r_list_delete (element_value->value.array_value.values, iter); ev_element = NULL; } r_list_free (element_value->value.array_value.values); break; case R_BIN_JAVA_EV_TAG_ANNOTATION: // Delete the Annotations List r_list_free (element_value->value.annotation_value.element_value_pairs); break; default: // eprintf unable to free the tag break; } if (attr) { free (attr->name); free (attr->metas); free (attr); } } R_API RBinJavaAnnotation *r_bin_java_annotation_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; RBinJavaAnnotation *annotation = NULL; RBinJavaElementValuePair *evps = NULL; ut64 offset = 0; annotation = R_NEW0 (RBinJavaAnnotation); if (!annotation) { return NULL; } // (ut16) read and set annotation_value.type_idx; annotation->type_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; // (ut16) read and set annotation_value.num_element_value_pairs; annotation->num_element_value_pairs = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; annotation->element_value_pairs = r_list_newf (r_bin_java_element_pair_free); // read annotation_value.num_element_value_pairs, and append to annotation_value.element_value_pairs for (i = 0; i < annotation->num_element_value_pairs; i++) { if (offset > sz) { break; } evps = r_bin_java_element_pair_new (buffer + offset, sz - offset, buf_offset + offset); if (evps) { offset += evps->size; r_list_append (annotation->element_value_pairs, (void *) evps); } } annotation->size = offset; return annotation; } R_API ut64 r_bin_java_annotation_calc_size(RBinJavaAnnotation *annotation) { ut64 sz = 0; RListIter *iter, *iter_tmp; RBinJavaElementValuePair *evps = NULL; if (!annotation) { // TODO eprintf allocation fail return sz; } // annotation->type_idx = R_BIN_JAVA_USHORT (buffer, offset); sz += 2; // annotation->num_element_value_pairs = R_BIN_JAVA_USHORT (buffer, offset); sz += 2; r_list_foreach_safe (annotation->element_value_pairs, iter, iter_tmp, evps) { if (evps) { sz += r_bin_java_element_pair_calc_size (evps); } } return sz; } R_API void r_bin_java_annotation_free(void /*RBinJavaAnnotation*/ *a) { RBinJavaAnnotation *annotation = a; if (annotation) { r_list_free (annotation->element_value_pairs); free (annotation); } } R_API void r_bin_java_print_annotation_summary(RBinJavaAnnotation *annotation) { RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaElementValuePair *evp = NULL; if (!annotation) { // TODO eprintf invalid annotation return; } Eprintf (" Annotation Type Index: 0x%02x\n", annotation->type_idx); Eprintf (" Annotation Number of EV Pairs: 0x%04x\n", annotation->num_element_value_pairs); Eprintf (" Annotation EV Pair Values:\n"); if (annotation->element_value_pairs) { r_list_foreach_safe (annotation->element_value_pairs, iter, iter_tmp, evp) { r_bin_java_print_element_pair_summary (evp); } } } R_API ut64 r_bin_java_element_pair_calc_size(RBinJavaElementValuePair *evp) { ut64 sz = 0; if (evp == NULL) { return sz; } // evp->element_name_idx = r_bin_java_read_short(bin, bin->b->cur); sz += 2; // evp->value = r_bin_java_element_value_new (bin, offset+2); if (evp->value) { sz += r_bin_java_element_value_calc_size (evp->value); } return sz; } R_API ut64 r_bin_java_element_value_calc_size(RBinJavaElementValue *element_value) { RListIter *iter, *iter_tmp; RBinJavaElementValue *ev_element; RBinJavaElementValuePair *evps; ut64 sz = 0; if (element_value == NULL) { return sz; } // tag sz += 1; switch (element_value->tag) { case R_BIN_JAVA_EV_TAG_BYTE: case R_BIN_JAVA_EV_TAG_CHAR: case R_BIN_JAVA_EV_TAG_DOUBLE: case R_BIN_JAVA_EV_TAG_FLOAT: case R_BIN_JAVA_EV_TAG_INT: case R_BIN_JAVA_EV_TAG_LONG: case R_BIN_JAVA_EV_TAG_SHORT: case R_BIN_JAVA_EV_TAG_BOOLEAN: case R_BIN_JAVA_EV_TAG_STRING: // look up value in bin->cp_list // (ut16) read and set const_value.const_value_idx // element_value->value.const_value.const_value_idx = r_bin_java_read_short(bin, bin->b->cur); sz += 2; break; case R_BIN_JAVA_EV_TAG_ENUM: // (ut16) read and set enum_const_value.type_name_idx // element_value->value.enum_const_value.type_name_idx = r_bin_java_read_short(bin, bin->b->cur); sz += 2; // (ut16) read and set enum_const_value.const_name_idx // element_value->value.enum_const_value.const_name_idx = r_bin_java_read_short(bin, bin->b->cur); sz += 2; break; case R_BIN_JAVA_EV_TAG_CLASS: // (ut16) read and set class_value.class_info_idx // element_value->value.class_value.class_info_idx = r_bin_java_read_short(bin, bin->b->cur); sz += 2; break; case R_BIN_JAVA_EV_TAG_ARRAY: // (ut16) read and set array_value.num_values // element_value->value.array_value.num_values = r_bin_java_read_short(bin, bin->b->cur); sz += 2; r_list_foreach_safe (element_value->value.array_value.values, iter, iter_tmp, ev_element) { if (ev_element) { sz += r_bin_java_element_value_calc_size (ev_element); } } break; case R_BIN_JAVA_EV_TAG_ANNOTATION: // annotation new is not used here. // (ut16) read and set annotation_value.type_idx; // element_value->value.annotation_value.type_idx = r_bin_java_read_short(bin, bin->b->cur); sz += 2; // (ut16) read and set annotation_value.num_element_value_pairs; // element_value->value.annotation_value.num_element_value_pairs = r_bin_java_read_short(bin, bin->b->cur); sz += 2; element_value->value.annotation_value.element_value_pairs = r_list_newf (r_bin_java_element_pair_free); r_list_foreach_safe (element_value->value.annotation_value.element_value_pairs, iter, iter_tmp, evps) { if (evps) { sz += r_bin_java_element_pair_calc_size (evps); } } break; default: // eprintf unable to handle tag break; } return sz; } R_API RBinJavaElementValue *r_bin_java_element_value_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; ut64 offset = 0; RBinJavaElementValue *element_value = R_NEW0 (RBinJavaElementValue); if (!element_value) { return NULL; } RBinJavaElementValuePair *evps = NULL; element_value->metas = R_NEW0 (RBinJavaMetaInfo); if (!element_value->metas) { R_FREE (element_value); return NULL; } element_value->file_offset = buf_offset; element_value->tag = buffer[offset]; element_value->size += 1; offset += 1; element_value->metas->type_info = (void *) r_bin_java_get_ev_meta_from_tag (element_value->tag); switch (element_value->tag) { case R_BIN_JAVA_EV_TAG_BYTE: case R_BIN_JAVA_EV_TAG_CHAR: case R_BIN_JAVA_EV_TAG_DOUBLE: case R_BIN_JAVA_EV_TAG_FLOAT: case R_BIN_JAVA_EV_TAG_INT: case R_BIN_JAVA_EV_TAG_LONG: case R_BIN_JAVA_EV_TAG_SHORT: case R_BIN_JAVA_EV_TAG_BOOLEAN: case R_BIN_JAVA_EV_TAG_STRING: // look up value in bin->cp_list // (ut16) read and set const_value.const_value_idx element_value->value.const_value.const_value_idx = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; // look-up, deep copy, and set const_value.const_value_cp_obj element_value->value.const_value.const_value_cp_obj = r_bin_java_clone_cp_idx (R_BIN_JAVA_GLOBAL_BIN, element_value->value.const_value.const_value_idx); break; case R_BIN_JAVA_EV_TAG_ENUM: // (ut16) read and set enum_const_value.type_name_idx element_value->value.enum_const_value.type_name_idx = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; offset += 2; // (ut16) read and set enum_const_value.const_name_idx element_value->value.enum_const_value.const_name_idx = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; offset += 2; // look up type_name_index in bin->cp_list // look-up, deep copy, and set enum_const_value.const_name_cp_obj element_value->value.enum_const_value.const_name_cp_obj = r_bin_java_clone_cp_idx (R_BIN_JAVA_GLOBAL_BIN, element_value->value.enum_const_value.const_name_idx); // look-up, deep copy, and set enum_const_value.type_name_cp_obj element_value->value.enum_const_value.type_name_cp_obj = r_bin_java_clone_cp_idx (R_BIN_JAVA_GLOBAL_BIN, element_value->value.enum_const_value.type_name_idx); break; case R_BIN_JAVA_EV_TAG_CLASS: // (ut16) read and set class_value.class_info_idx element_value->value.class_value.class_info_idx = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; offset += 2; // look up type_name_index in bin->cp_list // look-up, deep copy, and set class_value.class_info_cp_obj element_value->value.class_value.class_info_cp_obj = r_bin_java_clone_cp_idx (R_BIN_JAVA_GLOBAL_BIN, element_value->value.class_value.class_info_idx); break; case R_BIN_JAVA_EV_TAG_ARRAY: // (ut16) read and set array_value.num_values element_value->value.array_value.num_values = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; offset += 2; element_value->value.array_value.values = r_list_new (); for (i = 0; i < element_value->value.array_value.num_values; i++) { if (offset >= sz) { break; } RBinJavaElementValue *ev_element = r_bin_java_element_value_new (buffer + offset, sz - offset, buf_offset + offset); if (ev_element) { element_value->size += ev_element->size; offset += ev_element->size; // read array_value.num_values, and append to array_value.values r_list_append (element_value->value.array_value.values, (void *) ev_element); } } break; case R_BIN_JAVA_EV_TAG_ANNOTATION: // annotation new is not used here. // (ut16) read and set annotation_value.type_idx; if (offset + 8 < sz) { element_value->value.annotation_value.type_idx = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; offset += 2; // (ut16) read and set annotation_value.num_element_value_pairs; element_value->value.annotation_value.num_element_value_pairs = R_BIN_JAVA_USHORT (buffer, offset); element_value->size += 2; offset += 2; } element_value->value.annotation_value.element_value_pairs = r_list_newf (r_bin_java_element_pair_free); // read annotation_value.num_element_value_pairs, and append to annotation_value.element_value_pairs for (i = 0; i < element_value->value.annotation_value.num_element_value_pairs; i++) { if (offset > sz) { break; } evps = r_bin_java_element_pair_new (buffer + offset, sz - offset, buf_offset + offset); if (evps) { element_value->size += evps->size; offset += evps->size; } if (evps == NULL) { // TODO: eprintf error when reading element pair } r_list_append (element_value->value.annotation_value.element_value_pairs, (void *) evps); } break; default: // eprintf unable to handle tag break; } return element_value; } R_API void r_bin_java_bootstrap_method_argument_free(void /*RBinJavaBootStrapArgument*/ *b) { RBinJavaBootStrapArgument *bsm_arg = b; if (bsm_arg) { RBinJavaCPTypeMetas *tm = (RBinJavaCPTypeMetas*)bsm_arg->argument_info_cp_obj; if (tm) { if (tm && (size_t)(tm->allocs) > 1024 && tm->allocs->delete_obj) { tm->allocs->delete_obj (tm); } bsm_arg->argument_info_cp_obj = NULL; } free (bsm_arg); } } R_API void r_bin_java_print_bootstrap_method_argument_summary(RBinJavaBootStrapArgument *bsm_arg) { if (!bsm_arg) { eprintf ("Attempting to print an invalid RBinJavaBootStrapArgument *.\n"); return; } eprintf ("Bootstrap Method Argument Information:\n"); eprintf (" Offset: 0x%08"PFMT64x"", bsm_arg->file_offset); eprintf (" Name_And_Type Index = (0x%02x)\n", bsm_arg->argument_info_idx); if (bsm_arg->argument_info_cp_obj) { eprintf (" Bootstrap Method Argument Type and Name Info:\n"); ((RBinJavaCPTypeMetas *) bsm_arg->argument_info_cp_obj)->allocs->print_summary (bsm_arg->argument_info_cp_obj); } else { eprintf (" Bootstrap Method Argument Type and Name Info: INVALID\n"); } } R_API void r_bin_java_print_bootstrap_method_summary(RBinJavaBootStrapMethod *bsm) { RBinJavaBootStrapArgument *bsm_arg = NULL; RListIter *iter = NULL, *iter_tmp = NULL; if (!bsm) { eprintf ("Attempting to print an invalid RBinJavaBootStrapArgument *.\n"); return; } eprintf ("Bootstrap Method Information:\n"); eprintf (" Offset: 0x%08"PFMT64x"", bsm->file_offset); eprintf (" Method Reference Index = (0x%02x)\n", bsm->bootstrap_method_ref); eprintf (" Number of Method Arguments = (0x%02x)\n", bsm->num_bootstrap_arguments); if (bsm->bootstrap_arguments) { r_list_foreach_safe (bsm->bootstrap_arguments, iter, iter_tmp, bsm_arg) { if (bsm_arg) { r_bin_java_print_bootstrap_method_argument_summary (bsm_arg); } } } else { eprintf (" Bootstrap Method Argument: NONE \n"); } } R_API RBinJavaBootStrapArgument *r_bin_java_bootstrap_method_argument_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { ut64 offset = 0; RBinJavaBootStrapArgument *bsm_arg = (RBinJavaBootStrapArgument *) malloc (sizeof (RBinJavaBootStrapArgument)); if (!bsm_arg) { // TODO eprintf failed to allocate bytes for bootstrap_method. return bsm_arg; } memset (bsm_arg, 0, sizeof (RBinJavaBootStrapArgument)); bsm_arg->file_offset = buf_offset; bsm_arg->argument_info_idx = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; bsm_arg->argument_info_cp_obj = r_bin_java_clone_cp_idx (R_BIN_JAVA_GLOBAL_BIN, bsm_arg->argument_info_idx); bsm_arg->size = offset; return bsm_arg; } R_API void r_bin_java_bootstrap_method_free(void /*/RBinJavaBootStrapMethod*/ *b) { RBinJavaBootStrapMethod *bsm = b; RListIter *iter, *iter_tmp; RBinJavaBootStrapArgument *obj = NULL; if (bsm) { if (bsm->bootstrap_arguments) { r_list_foreach_safe (bsm->bootstrap_arguments, iter, iter_tmp, obj) { if (obj) { r_bin_java_bootstrap_method_argument_free (obj); } // r_list_delete (bsm->bootstrap_arguments, iter); } r_list_free (bsm->bootstrap_arguments); bsm->bootstrap_arguments = NULL; } free (bsm); } } R_API RBinJavaBootStrapMethod *r_bin_java_bootstrap_method_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaBootStrapArgument *bsm_arg = NULL; ut32 i = 0; ut64 offset = 0; RBinJavaBootStrapMethod *bsm = R_NEW0 (RBinJavaBootStrapMethod); if (!bsm) { // TODO eprintf failed to allocate bytes for bootstrap_method. return bsm; } memset (bsm, 0, sizeof (RBinJavaBootStrapMethod)); bsm->file_offset = buf_offset; bsm->bootstrap_method_ref = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; bsm->num_bootstrap_arguments = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; bsm->bootstrap_arguments = r_list_new (); for (i = 0; i < bsm->num_bootstrap_arguments; i++) { if (offset >= sz) { break; } // bsm_arg = r_bin_java_bootstrap_method_argument_new (bin, bin->b->cur); bsm_arg = r_bin_java_bootstrap_method_argument_new (buffer + offset, sz - offset, buf_offset + offset); if (bsm_arg) { offset += bsm_arg->size; r_list_append (bsm->bootstrap_arguments, (void *) bsm_arg); } else { // TODO eprintf Failed to read the %d boot strap method. } } bsm->size = offset; return bsm; } R_API void r_bin_java_print_bootstrap_methods_attr_summary(RBinJavaAttrInfo *attr) { RListIter *iter, *iter_tmp; RBinJavaBootStrapMethod *obj = NULL; if (!attr || attr->type == R_BIN_JAVA_ATTR_TYPE_BOOTSTRAP_METHODS_ATTR) { eprintf ("Unable to print attribue summary for RBinJavaAttrInfo *RBinJavaBootstrapMethodsAttr"); return; } eprintf ("Bootstrap Methods Attribute Information Information:\n"); eprintf (" Attribute Offset: 0x%08"PFMT64x"", attr->file_offset); eprintf (" Length: 0x%08x", attr->length); eprintf (" Number of Method Arguments = (0x%02x)\n", attr->info.bootstrap_methods_attr.num_bootstrap_methods); if (attr->info.bootstrap_methods_attr.bootstrap_methods) { r_list_foreach_safe (attr->info.bootstrap_methods_attr.bootstrap_methods, iter, iter_tmp, obj) { if (obj) { r_bin_java_print_bootstrap_method_summary (obj); } } } else { eprintf (" Bootstrap Methods: NONE \n"); } } R_API void r_bin_java_bootstrap_methods_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_BOOTSTRAP_METHODS_ATTR) { free (attr->name); free (attr->metas); r_list_free (attr->info.bootstrap_methods_attr.bootstrap_methods); free (attr); } } R_API ut64 r_bin_java_bootstrap_methods_attr_calc_size(RBinJavaAttrInfo *attr) { RListIter *iter, *iter_tmp; RBinJavaBootStrapMethod *bsm = NULL; ut64 size = 0; if (attr) { size += 6; // attr->info.bootstrap_methods_attr.num_bootstrap_methods = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (attr->info.bootstrap_methods_attr.bootstrap_methods, iter, iter_tmp, bsm) { if (bsm) { size += r_bin_java_bootstrap_method_calc_size (bsm); } else { // TODO eprintf Failed to read the %d boot strap method. } } } return size; } R_API ut64 r_bin_java_bootstrap_arg_calc_size(RBinJavaBootStrapArgument *bsm_arg) { ut64 size = 0; if (bsm_arg) { // bsm_arg->argument_info_idx = R_BIN_JAVA_USHORT (buffer, offset); size += 2; } return size; } R_API ut64 r_bin_java_bootstrap_method_calc_size(RBinJavaBootStrapMethod *bsm) { RListIter *iter, *iter_tmp; RBinJavaBootStrapArgument *bsm_arg = NULL; ut64 size = 0; if (bsm) { size += 6; // bsm->bootstrap_method_ref = R_BIN_JAVA_USHORT (buffer, offset); size += 2; // bsm->num_bootstrap_arguments = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (bsm->bootstrap_arguments, iter, iter_tmp, bsm_arg) { if (bsm_arg) { size += r_bin_java_bootstrap_arg_calc_size (bsm_arg); } else { // TODO eprintf Failed to read the %d boot strap method. } } } return size; } R_API RBinJavaAttrInfo *r_bin_java_bootstrap_methods_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; RBinJavaBootStrapMethod *bsm = NULL; ut64 offset = 0; RBinJavaAttrInfo *attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_BOOTSTRAP_METHODS_ATTR; attr->info.bootstrap_methods_attr.num_bootstrap_methods = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.bootstrap_methods_attr.bootstrap_methods = r_list_newf (r_bin_java_bootstrap_method_free); for (i = 0; i < attr->info.bootstrap_methods_attr.num_bootstrap_methods; i++) { // bsm = r_bin_java_bootstrap_method_new (bin, bin->b->cur); if (offset >= sz) { break; } bsm = r_bin_java_bootstrap_method_new (buffer + offset, sz - offset, buf_offset + offset); if (bsm) { offset += bsm->size; r_list_append (attr->info.bootstrap_methods_attr.bootstrap_methods, (void *) bsm); } else { // TODO eprintf Failed to read the %d boot strap method. } } attr->size = offset; } return attr; } R_API void r_bin_java_print_annotation_default_attr_summary(RBinJavaAttrInfo *attr) { if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_ANNOTATION_DEFAULT_ATTR) { eprintf ("Annotation Default Attribute Information:\n"); eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); eprintf (" Attribute Length: %d\n", attr->length); r_bin_java_print_element_value_summary ((attr->info.annotation_default_attr.default_value)); } else { // TODO: eprintf attr is invalid } } R_API void r_bin_java_annotation_array_free(void /*RBinJavaAnnotationsArray*/ *a) { RBinJavaAnnotationsArray *annotation_array = a; RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaAnnotation *annotation; if (!annotation_array->annotations) { // TODO eprintf return; } r_list_foreach_safe (annotation_array->annotations, iter, iter_tmp, annotation) { if (annotation) { r_bin_java_annotation_free (annotation); } // r_list_delete (annotation_array->annotations, iter); } r_list_free (annotation_array->annotations); free (annotation_array); } R_API void r_bin_java_print_annotation_array_summary(RBinJavaAnnotationsArray *annotation_array) { RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaAnnotation *annotation; if (!annotation_array->annotations) { // TODO eprintf return; } eprintf (" Annotation Array Information:\n"); eprintf (" Number of Annotation Array Elements: %d\n", annotation_array->num_annotations); r_list_foreach_safe (annotation_array->annotations, iter, iter_tmp, annotation) { r_bin_java_print_annotation_summary (annotation); } } R_API RBinJavaAnnotationsArray *r_bin_java_annotation_array_new(ut8 *buffer, ut64 sz, ut64 buf_offset) { RBinJavaAnnotation *annotation; RBinJavaAnnotationsArray *annotation_array; ut32 i; ut64 offset = 0; annotation_array = (RBinJavaAnnotationsArray *) malloc (sizeof (RBinJavaAnnotationsArray)); if (!annotation_array) { // TODO eprintf return NULL; } annotation_array->num_annotations = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; annotation_array->annotations = r_list_new (); for (i = 0; i < annotation_array->num_annotations; i++) { if (offset > sz) { break; } annotation = r_bin_java_annotation_new (buffer + offset, sz - offset, buf_offset + offset); if (annotation) { offset += annotation->size; r_list_append (annotation_array->annotations, (void *) annotation); } } annotation_array->size = offset; return annotation_array; } R_API RBinJavaAttrInfo *r_bin_java_rtv_annotations_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_ANNOTATION_ATTR; attr->info.annotation_array.num_annotations = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.annotation_array.annotations = r_list_newf (r_bin_java_annotation_free); for (i = 0; i < attr->info.annotation_array.num_annotations; i++) { if (offset >= sz) { break; } RBinJavaAnnotation *annotation = r_bin_java_annotation_new (buffer + offset, sz - offset, buf_offset + offset); if (annotation) { offset += annotation->size; r_list_append (attr->info.annotation_array.annotations, (void *) annotation); } } attr->size = offset; } return attr; } R_API ut64 r_bin_java_annotation_array_calc_size(RBinJavaAnnotationsArray *annotation_array) { ut64 size = 0; RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaAnnotation *annotation; if (!annotation_array->annotations) { // TODO eprintf return size; } // annotation_array->num_annotations = R_BIN_JAVA_USHORT (buffer, offset); size += 2; r_list_foreach_safe (annotation_array->annotations, iter, iter_tmp, annotation) { size += r_bin_java_annotation_calc_size (annotation); } return size; } R_API ut64 r_bin_java_rtv_annotations_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; if (!attr) { // TODO eprintf allocation fail return size; } size += (6 + r_bin_java_annotation_array_calc_size (&(attr->info.annotation_array))); return size; } R_API RBinJavaAttrInfo *r_bin_java_rti_annotations_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_ANNOTATION_ATTR; attr->info.annotation_array.num_annotations = R_BIN_JAVA_USHORT (buffer, offset); offset += 2; attr->info.annotation_array.annotations = r_list_newf (r_bin_java_annotation_free); for (i = 0; i < attr->info.rtv_annotations_attr.num_annotations; i++) { if (offset >= sz) { break; } RBinJavaAnnotation *annotation = r_bin_java_annotation_new (buffer + offset, sz - offset, buf_offset + offset); if (annotation) { offset += annotation->size; } r_list_append (attr->info.annotation_array.annotations, (void *) annotation); } attr->size = offset; } return attr; } R_API ut64 r_bin_java_rti_annotations_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; if (!attr) { // TODO eprintf allocation fail return size; } size += (6 + r_bin_java_annotation_array_calc_size (&(attr->info.annotation_array))); return size; } R_API void r_bin_java_rtv_annotations_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_ANNOTATION_ATTR) { r_list_free (attr->info.annotation_array.annotations); free (attr->metas); free (attr->name); free (attr); } } R_API void r_bin_java_rti_annotations_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_ANNOTATION_ATTR) { r_list_free (attr->info.annotation_array.annotations); free (attr->metas); free (attr->name); free (attr); } } R_API void r_bin_java_print_rtv_annotations_attr_summary(RBinJavaAttrInfo *attr) { if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_ANNOTATION_ATTR) { Eprintf ("Runtime Visible Annotations Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); r_bin_java_print_annotation_array_summary (&attr->info.annotation_array); } } R_API void r_bin_java_print_rti_annotations_attr_summary(RBinJavaAttrInfo *attr) { if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_ANNOTATION_ATTR) { Eprintf ("Runtime Invisible Annotations Attribute Information:\n"); Eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); Eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); Eprintf (" Attribute Length: %d\n", attr->length); r_bin_java_print_annotation_array_summary (&attr->info.annotation_array); } } R_API ut64 r_bin_java_rtip_annotations_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaAnnotationsArray *annotation_array; if (!attr) { // TODO eprintf allocation fail return size; } // attr->info.rtip_annotations_attr.num_parameters = buffer[offset]; size += (6 + 1); r_list_foreach_safe (attr->info.rtip_annotations_attr.parameter_annotations, iter, iter_tmp, annotation_array) { if (annotation_array) { size += r_bin_java_annotation_array_calc_size (annotation_array); } } return size; } R_API RBinJavaAttrInfo *r_bin_java_rtip_annotations_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_PARAMETER_ANNOTATION_ATTR; attr->info.rtip_annotations_attr.num_parameters = buffer[offset]; offset += 1; attr->info.rtip_annotations_attr.parameter_annotations = r_list_newf (r_bin_java_annotation_array_free); for (i = 0; i < attr->info.rtip_annotations_attr.num_parameters; i++) { if (offset >= sz) { break; } RBinJavaAnnotationsArray *annotation_array = r_bin_java_annotation_array_new ( buffer + offset, sz - offset, buf_offset + offset); if (annotation_array) { offset += annotation_array->size; r_list_append (attr->info.rtip_annotations_attr.parameter_annotations, (void *) annotation_array); } } attr->size = offset; } return attr; } R_API RBinJavaAttrInfo *r_bin_java_rtvp_annotations_attr_new(RBinJavaObj *bin, ut8 *buffer, ut64 sz, ut64 buf_offset) { ut32 i = 0; RBinJavaAttrInfo *attr = NULL; ut64 offset = 0; attr = r_bin_java_default_attr_new (bin, buffer, sz, buf_offset); offset += 6; RBinJavaAnnotationsArray *annotation_array; if (attr) { attr->type = R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_PARAMETER_ANNOTATION_ATTR; attr->info.rtvp_annotations_attr.num_parameters = buffer[offset]; offset += 1; attr->info.rtvp_annotations_attr.parameter_annotations = r_list_newf (r_bin_java_annotation_array_free); for (i = 0; i < attr->info.rtvp_annotations_attr.num_parameters; i++) { if (offset > sz) { break; } annotation_array = r_bin_java_annotation_array_new (buffer + offset, sz - offset, buf_offset + offset); if (annotation_array) { offset += annotation_array->size; } r_list_append (attr->info.rtvp_annotations_attr.parameter_annotations, (void *) annotation_array); } attr->size = offset; } return attr; } R_API ut64 r_bin_java_rtvp_annotations_attr_calc_size(RBinJavaAttrInfo *attr) { ut64 size = 0; RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaAnnotationsArray *annotation_array; if (!attr) { return size; } size += (6 + 1); r_list_foreach_safe (attr->info.rtvp_annotations_attr.parameter_annotations, iter, iter_tmp, annotation_array) { if (annotation_array) { size += r_bin_java_annotation_array_calc_size ( annotation_array); } } return size; } R_API void r_bin_java_rtvp_annotations_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { if (attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_PARAMETER_ANNOTATION_ATTR) { r_list_free (attr->info.rtvp_annotations_attr.parameter_annotations); } free (attr->name); free (attr->metas); free (attr); } } R_API void r_bin_java_rtip_annotations_attr_free(void /*RBinJavaAttrInfo*/ *a) { RBinJavaAttrInfo *attr = a; if (attr) { // && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_PARAMETER_ANNOTATION_ATTR) { r_list_free (attr->info.rtip_annotations_attr.parameter_annotations); free (attr->metas); free (attr->name); free (attr); } } R_API void r_bin_java_print_rtvp_annotations_attr_summary(RBinJavaAttrInfo *attr) { RBinJavaAnnotationsArray *annotation_array = NULL; RListIter *iter = NULL, *iter_tmp = NULL; if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_VISIBLE_PARAMETER_ANNOTATION_ATTR) { eprintf ("Runtime Visible Parameter Annotations Attribute Information:\n"); eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); eprintf (" Attribute Length: %d\n", attr->length); eprintf (" Number of Runtime Invisible Parameters: %d\n", attr->info.rtvp_annotations_attr.num_parameters); r_list_foreach_safe (attr->info.rtvp_annotations_attr.parameter_annotations, iter, iter_tmp, annotation_array) { r_bin_java_print_annotation_array_summary (annotation_array); } } } R_API void r_bin_java_print_rtip_annotations_attr_summary(RBinJavaAttrInfo *attr) { RBinJavaAnnotationsArray *annotation_array = NULL; RListIter *iter = NULL, *iter_tmp = NULL; if (attr && attr->type == R_BIN_JAVA_ATTR_TYPE_RUNTIME_INVISIBLE_PARAMETER_ANNOTATION_ATTR) { eprintf ("Runtime Invisible Parameter Annotations Attribute Information:\n"); eprintf (" Attribute Offset: 0x%08"PFMT64x "\n", attr->file_offset); eprintf (" Attribute Name Index: %d (%s)\n", attr->name_idx, attr->name); eprintf (" Attribute Length: %d\n", attr->length); eprintf (" Number of Runtime Invisible Parameters: %d\n", attr->info.rtip_annotations_attr.num_parameters); r_list_foreach_safe (attr->info.rtip_annotations_attr.parameter_annotations, iter, iter_tmp, annotation_array) { r_bin_java_print_annotation_array_summary (annotation_array); } } } R_API RBinJavaCPTypeObj *r_bin_java_find_cp_name_and_type_info(RBinJavaObj *bin, ut16 name_idx, ut16 descriptor_idx) { RListIter *iter, *iter_tmp; RBinJavaCPTypeObj *res = NULL, *obj = NULL; IFDBG eprintf("Looking for name_idx: %d and descriptor_idx: %d\n", name_idx, descriptor_idx); r_list_foreach_safe (bin->cp_list, iter, iter_tmp, obj) { if (obj && obj->tag == R_BIN_JAVA_CP_NAMEANDTYPE) { IFDBG eprintf("RBinJavaCPTypeNameAndType has name_idx: %d and descriptor_idx: %d\n", obj->info.cp_name_and_type.name_idx, obj->info.cp_name_and_type.descriptor_idx); if (obj->info.cp_name_and_type.name_idx == name_idx && obj->info.cp_name_and_type.descriptor_idx == descriptor_idx) { res = obj; break; } } } return res; } R_API char *r_bin_java_resolve_cp_idx_type(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaCPTypeObj *item = NULL; char *str = NULL; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { // r_bin_java_new_bin(BIN_OBJ); return NULL; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { str = strdup (((RBinJavaCPTypeMetas *) item->metas->type_info)->name); } else { str = strdup ("INVALID"); } return str; } R_API RBinJavaCPTypeObj *r_bin_java_find_cp_ref_info_from_name_and_type(RBinJavaObj *bin, ut16 name_idx, ut16 descriptor_idx) { RBinJavaCPTypeObj *obj = r_bin_java_find_cp_name_and_type_info (bin, name_idx, descriptor_idx); if (obj) { return r_bin_java_find_cp_ref_info (bin, obj->metas->ord); } return NULL; } R_API RBinJavaCPTypeObj *r_bin_java_find_cp_ref_info(RBinJavaObj *bin, ut16 name_and_type_idx) { RListIter *iter, *iter_tmp; RBinJavaCPTypeObj *res = NULL, *obj = NULL; r_list_foreach_safe (bin->cp_list, iter, iter_tmp, obj) { if (obj->tag == R_BIN_JAVA_CP_FIELDREF && obj->info.cp_field.name_and_type_idx == name_and_type_idx) { res = obj; break; } else if (obj->tag == R_BIN_JAVA_CP_METHODREF && obj->info.cp_method.name_and_type_idx == name_and_type_idx) { res = obj; break; } } return res; } R_API char *r_bin_java_resolve(RBinJavaObj *BIN_OBJ, int idx, ut8 space_bn_name_type) { // TODO XXX FIXME add a size parameter to the str when it is passed in RBinJavaCPTypeObj *item = NULL, *item2 = NULL; char *class_str = NULL, *name_str = NULL, *desc_str = NULL, *string_str = NULL, *empty = "", *cp_name = NULL, *str = NULL; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { // r_bin_java_new_bin(BIN_OBJ); return NULL; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { cp_name = ((RBinJavaCPTypeMetas *) item->metas->type_info)->name; IFDBG eprintf("java_resolve Resolved: (%d) %s\n", idx, cp_name); } else { str = malloc (512); if (str) { snprintf (str, 512, "(%d) INVALID CP_OBJ", idx); } return str; } if (strcmp (cp_name, "Class") == 0) { item2 = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); // str = r_bin_java_get_name_from_bin_cp_list (BIN_OBJ, idx-1); class_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!class_str) { class_str = empty; } name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item2); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item2); if (!desc_str) { desc_str = empty; } str = r_str_newf ("%s%s%s", name_str, space_bn_name_type ? " " : "", desc_str); if (class_str != empty) { free (class_str); } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else if (!strcmp (cp_name, "MethodRef") || !strcmp (cp_name, "FieldRef") || !strcmp (cp_name, "InterfaceMethodRef")) { /* * The MethodRef, FieldRef, and InterfaceMethodRef structures */ class_str = r_bin_java_get_name_from_bin_cp_list (BIN_OBJ, item->info.cp_method.class_idx); if (!class_str) { class_str = empty; } name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item); if (!desc_str) { desc_str = empty; } str = r_str_newf ("%s/%s%s%s", class_str, name_str, space_bn_name_type ? " " : "", desc_str); if (class_str != empty) { free (class_str); } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else if (!strcmp (cp_name, "String")) { string_str = r_bin_java_get_utf8_from_bin_cp_list (BIN_OBJ, item->info.cp_string.string_idx); str = NULL; IFDBG eprintf("java_resolve String got: (%d) %s\n", item->info.cp_string.string_idx, string_str); if (!string_str) { string_str = empty; } str = r_str_newf ("\"%s\"", string_str); IFDBG eprintf("java_resolve String return: %s\n", str); if (string_str != empty) { free (string_str); } } else if (!strcmp (cp_name, "Utf8")) { char *tmp_str = convert_string ((const char *) item->info.cp_utf8.bytes, item->info.cp_utf8.length); ut32 tmp_str_len = tmp_str ? strlen (tmp_str) + 4 : 0; if (tmp_str) { str = malloc (tmp_str_len + 4); snprintf (str, tmp_str_len + 4, "\"%s\"", tmp_str); } free (tmp_str); } else if (!strcmp (cp_name, "Long")) { str = r_str_newf ("0x%"PFMT64x, r_bin_java_raw_to_long (item->info.cp_long.bytes.raw, 0)); } else if (!strcmp (cp_name, "Double")) { str = r_str_newf ("%f", r_bin_java_raw_to_double (item->info.cp_double.bytes.raw, 0)); } else if (!strcmp (cp_name, "Integer")) { str = r_str_newf ("0x%08x", R_BIN_JAVA_UINT (item->info.cp_integer.bytes.raw, 0)); } else if (!strcmp (cp_name, "Float")) { str = r_str_newf ("%f", R_BIN_JAVA_FLOAT (item->info.cp_float.bytes.raw, 0)); } else if (!strcmp (cp_name, "NameAndType")) { name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item); if (!desc_str) { desc_str = empty; } str = r_str_newf ("%s%s%s", name_str, space_bn_name_type ? " " : "", desc_str); if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else { str = strdup ("(null)"); } return str; } R_API ut8 r_bin_java_does_cp_idx_ref_method(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaField *fm_type = NULL; RListIter *iter; ut8 res = 0; r_list_foreach (BIN_OBJ->methods_list, iter, fm_type) { if (fm_type->field_ref_cp_obj->metas->ord == idx) { res = 1; break; } } return res; } R_API ut8 r_bin_java_does_cp_idx_ref_field(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaField *fm_type = NULL; RListIter *iter; ut8 res = 0; r_list_foreach (BIN_OBJ->fields_list, iter, fm_type) { if (fm_type->field_ref_cp_obj->metas->ord == idx) { res = 1; break; } } return res; } R_API char *r_bin_java_get_method_name(RBinJavaObj *bin_obj, ut32 idx) { char *name = NULL; if (idx < r_list_length (bin_obj->methods_list)) { RBinJavaField *fm_type = r_list_get_n (bin_obj->methods_list, idx); name = strdup (fm_type->name); } return name; } R_API RList *r_bin_java_get_method_num_name(RBinJavaObj *bin_obj) { ut32 i = 0; RListIter *iter = NULL; RBinJavaField *fm_type; RList *res = r_list_newf (free); r_list_foreach (bin_obj->methods_list, iter, fm_type) { ut32 len = strlen (fm_type->name) + 30; char *str = malloc (len); snprintf (str, len, "%d %s", i, fm_type->name); ++i; r_list_append (res, str); } return res; } /* R_API int r_bin_java_does_cp_obj_ref_idx (RBinJavaObj *bin_obj, RBinJavaCPTypeObj *cp_obj, ut16 idx) { int res = false; RBinJavaCPTypeObj *t_obj = NULL; if (cp_obj) { switch (cp_obj->tag) { case R_BIN_JAVA_CP_NULL: break; case R_BIN_JAVA_CP_UTF8: break; case R_BIN_JAVA_CP_UNKNOWN: break; case R_BIN_JAVA_CP_INTEGER: break; case R_BIN_JAVA_CP_FLOAT: break; case R_BIN_JAVA_CP_LONG: break; case R_BIN_JAVA_CP_DOUBLE: break; case R_BIN_JAVA_CP_CLASS: res = idx == cp_obj->info.cp_class.name_idx ? true : false; break; case R_BIN_JAVA_CP_STRING: res = idx == cp_obj->info.cp_string.string_idx ? true : false; break; case R_BIN_JAVA_CP_METHODREF: break;// check if idx is referenced here case R_BIN_JAVA_CP_INTERFACEMETHOD_REF: break; // check if idx is referenced here case R_BIN_JAVA_CP_FIELDREF: t_obj = r_bin_java_get_item_from_cp (bin_obj, cp_obj->info.cp_method.class_idx); res = r_bin_java_does_cp_obj_ref_idx (bin_obj, t_obj, idx); if (res == true) break; t_obj = r_bin_java_get_item_from_cp (bin_obj, cp_obj->info.cp_method.name_and_type_idx); res = r_bin_java_does_cp_obj_ref_idx (bin_obj, t_obj, idx); break; case R_BIN_JAVA_CP_NAMEANDTYPE: break;// check if idx is referenced here obj->info.cp_name_and_type.name_idx case R_BIN_JAVA_CP_METHODHANDLE: break;// check if idx is referenced here case R_BIN_JAVA_CP_METHODTYPE: break;// check if idx is referenced here case R_BIN_JAVA_CP_INVOKEDYNAMIC: break;// check if idx is referenced here } } } */ R_API RList *r_bin_java_find_cp_const_by_val_long(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len) { RList *res = r_list_newf (free); ut32 *v = NULL; RListIter *iter; RBinJavaCPTypeObj *cp_obj; eprintf ("Looking for 0x%08x\n", R_BIN_JAVA_UINT (bytes, 0)); r_list_foreach (bin_obj->cp_list, iter, cp_obj) { if (cp_obj->tag == R_BIN_JAVA_CP_LONG) { if (len == 8 && r_bin_java_raw_to_long (cp_obj->info.cp_long.bytes.raw, 0) == r_bin_java_raw_to_long (bytes, 0)) { // TODO: we can safely store a ut32 inside the list without having to allocate it v = malloc (sizeof (ut32)); if (!v) { r_list_free (res); return NULL; } *v = cp_obj->idx; r_list_append (res, v); } } } return res; } R_API RList *r_bin_java_find_cp_const_by_val_double(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len) { RList *res = r_list_newf (free); ut32 *v = NULL; RListIter *iter; RBinJavaCPTypeObj *cp_obj; eprintf ("Looking for %f\n", r_bin_java_raw_to_double (bytes, 0)); r_list_foreach (bin_obj->cp_list, iter, cp_obj) { if (cp_obj->tag == R_BIN_JAVA_CP_DOUBLE) { if (len == 8 && r_bin_java_raw_to_double (cp_obj->info.cp_long.bytes.raw, 0) == r_bin_java_raw_to_double (bytes, 0)) { v = malloc (sizeof (ut32)); if (!v) { r_list_free (res); return NULL; } *v = cp_obj->idx; r_list_append (res, v); } } } return res; } R_API RList *r_bin_java_find_cp_const_by_val_float(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len) { RList *res = r_list_newf (free); ut32 *v = NULL; RListIter *iter; RBinJavaCPTypeObj *cp_obj; eprintf ("Looking for %f\n", R_BIN_JAVA_FLOAT (bytes, 0)); r_list_foreach (bin_obj->cp_list, iter, cp_obj) { if (cp_obj->tag == R_BIN_JAVA_CP_FLOAT) { if (len == 4 && R_BIN_JAVA_FLOAT (cp_obj->info.cp_long.bytes.raw, 0) == R_BIN_JAVA_FLOAT (bytes, 0)) { v = malloc (sizeof (ut32)); if (!v) { r_list_free (res); return NULL; } *v = cp_obj->idx; r_list_append (res, v); } } } return res; } R_API RList *r_bin_java_find_cp_const_by_val(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len, const char t) { switch (t) { case R_BIN_JAVA_CP_UTF8: return r_bin_java_find_cp_const_by_val_utf8 (bin_obj, bytes, len); case R_BIN_JAVA_CP_INTEGER: return r_bin_java_find_cp_const_by_val_int (bin_obj, bytes, len); case R_BIN_JAVA_CP_FLOAT: return r_bin_java_find_cp_const_by_val_float (bin_obj, bytes, len); case R_BIN_JAVA_CP_LONG: return r_bin_java_find_cp_const_by_val_long (bin_obj, bytes, len); case R_BIN_JAVA_CP_DOUBLE: return r_bin_java_find_cp_const_by_val_double (bin_obj, bytes, len); case R_BIN_JAVA_CP_UNKNOWN: default: eprintf ("Failed to perform the search for: %s\n", bytes); return r_list_new (); } } // #if 0 // Attempted to clean up these functions and remove them since they are "unused" but without // them there are some compile time warnings, because other projects actually depend on these // for some form of information. R_API void U(add_cp_objs_to_sdb)(RBinJavaObj * bin) { /* Add Constant Pool Serialized Object to an Array the key for this info is: Key: java..cp_obj Each Value varies by type: In general its: ...[type specific stuff] Example: UTF-8: .... Integer: ... Long: ...abs(long)> FieldRef/MethodRef: .... */ ut32 idx = 0, class_name_inheap = 1; RBinJavaCPTypeObj *cp_obj = NULL; char *key = NULL, *value = NULL; char str_cnt[40]; char *class_name = r_bin_java_get_this_class_name (bin); ut32 key_buf_size = 0; if (!class_name) { class_name = "unknown"; class_name_inheap = 0; } // 4 - format, 8 number, 1 null byte, 7 "unknown" key_buf_size = strlen (class_name) + 4 + 8 + 1; key = malloc (key_buf_size); if (key == NULL) { if (class_name_inheap) { free (class_name); } return; } snprintf (key, key_buf_size - 1, "%s.cp_count", class_name); key[key_buf_size - 1] = 0; snprintf (str_cnt, 39, "%d", bin->cp_count); str_cnt[39] = 0; sdb_set (bin->kv, key, value, 0); // sdb_alist(bin->kv, key); for (idx = 0; idx < bin->cp_count; idx++) { snprintf (key, key_buf_size - 1, "%s.cp.%d", class_name, idx); key[key_buf_size - 1] = 0; cp_obj = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (bin, idx); IFDBG eprintf("Adding %s to the sdb.\n", key); if (cp_obj) { value = ((RBinJavaCPTypeMetas *) cp_obj->metas->type_info)-> allocs->stringify_obj (cp_obj); sdb_set (bin->kv, key, value, 0); free (value); } } if (class_name_inheap) { free (class_name); } free (key); } R_API void U(add_field_infos_to_sdb)(RBinJavaObj * bin) { /* *** Experimental and May Change *** Add field information to an Array the key for this info variable depenedent on addr, method ordinal, etc. Key 1, mapping to method key: java. = Key 3, method description .info = [, , , ] key 4, method meta .meta = [, ?] */ RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaField *fm_type; ut32 key_size = 255, value_buffer_size = 1024, class_name_inheap = 1; char *field_key = NULL, *field_key_value = NULL, *value_buffer = NULL; char *class_name = r_bin_java_get_this_class_name (bin); if (!class_name) { class_name = "unknown"; class_name_inheap = 0; } key_size += strlen (class_name); value_buffer_size += strlen (class_name); field_key = malloc (key_size); value_buffer = malloc (value_buffer_size); field_key_value = malloc (key_size); snprintf (field_key, key_size, "%s.methods", class_name); field_key[key_size - 1] = 0; r_list_foreach_safe (bin->fields_list, iter, iter_tmp, fm_type) { char number_buffer[80]; ut64 file_offset = fm_type->file_offset + bin->loadaddr; snprintf (number_buffer, sizeof (number_buffer), "0x%04"PFMT64x, file_offset); IFDBG eprintf("Inserting: []%s = %s\n", field_key, number_buffer); sdb_array_push (bin->kv, field_key, number_buffer, 0); } r_list_foreach_safe (bin->fields_list, iter, iter_tmp, fm_type) { ut64 field_offset = fm_type->file_offset + bin->loadaddr; // generate method specific key & value snprintf (field_key, key_size, "%s.0x%04"PFMT64x, class_name, field_offset); field_key[key_size - 1] = 0; snprintf (field_key_value, key_size, "%s.0x%04"PFMT64x ".field", class_name, field_offset); field_key_value[key_size - 1] = 0; sdb_set (bin->kv, field_key, field_key_value, 0); IFDBG eprintf("Inserting: %s = %s\n", field_key, field_key_value); // generate info key, and place values in method info array snprintf (field_key, key_size, "%s.info", field_key_value); field_key[key_size - 1] = 0; snprintf (value_buffer, value_buffer_size, "%s", fm_type->flags_str); value_buffer[value_buffer_size - 1] = 0; sdb_array_push (bin->kv, field_key, value_buffer, 0); IFDBG eprintf("Inserting: []%s = %s\n", field_key, value_buffer); snprintf (value_buffer, value_buffer_size, "%s", fm_type->class_name); value_buffer[value_buffer_size - 1] = 0; sdb_array_push (bin->kv, field_key, value_buffer, 0); IFDBG eprintf("Inserting: []%s = %s\n", field_key, value_buffer); snprintf (value_buffer, value_buffer_size, "%s", fm_type->name); value_buffer[value_buffer_size - 1] = 0; sdb_array_push (bin->kv, field_key, value_buffer, 0); IFDBG eprintf("Inserting: []%s = %s\n", field_key, value_buffer); snprintf (value_buffer, value_buffer_size, "%s", fm_type->descriptor); value_buffer[value_buffer_size - 1] = 0; sdb_array_push (bin->kv, field_key, value_buffer, 0); IFDBG eprintf("Inserting: []%s = %s\n", field_key, value_buffer); } free (field_key); free (field_key_value); free (value_buffer); if (class_name_inheap) { free (class_name); } } R_API void U(add_method_infos_to_sdb)(RBinJavaObj * bin) { /* *** Experimental and May Change *** Add Mehtod information to an Array the key for this info variable depenedent on addr, method ordinal, etc. Key 1, mapping to method key: java. = Key 2, basic code information .code = [, ] Key 3, method description .info = [, , , ,] key 4, method meta .meta = [, ?] // TODO in key 3 add ? e.g. .. Note: method name not used because of collisions with operator overloading also take note that code offset and the method offset are not the same values. */ RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaField *fm_type; ut32 key_size = 255, value_buffer_size = 1024, class_name_inheap = 1; char *method_key = NULL, *method_key_value = NULL, *value_buffer = NULL; char *class_name = r_bin_java_get_this_class_name (bin); ut64 baddr = bin->loadaddr; if (!class_name) { class_name = "unknown"; class_name_inheap = 0; } key_size += strlen (class_name); value_buffer_size += strlen (class_name); method_key = malloc (key_size); value_buffer = malloc (value_buffer_size); method_key_value = malloc (key_size); snprintf (method_key, key_size, "%s.methods", class_name); method_key[key_size - 1] = 0; r_list_foreach_safe (bin->methods_list, iter, iter_tmp, fm_type) { char number_buffer[80]; ut64 file_offset = fm_type->file_offset + baddr; snprintf (number_buffer, sizeof (number_buffer), "0x%04"PFMT64x, file_offset); sdb_array_push (bin->kv, method_key, number_buffer, 0); } r_list_foreach_safe (bin->methods_list, iter, iter_tmp, fm_type) { ut64 code_offset = r_bin_java_get_method_code_offset (fm_type) + baddr, code_size = r_bin_java_get_method_code_size (fm_type), method_offset = fm_type->file_offset + baddr; // generate method specific key & value snprintf (method_key, key_size, "%s.0x%04"PFMT64x, class_name, code_offset); method_key[key_size - 1] = 0; snprintf (method_key_value, key_size, "%s.0x%04"PFMT64x ".method", class_name, method_offset); method_key_value[key_size - 1] = 0; IFDBG eprintf("Adding %s to sdb_array: %s\n", method_key_value, method_key); sdb_set (bin->kv, method_key, method_key_value, 0); // generate code key and values snprintf (method_key, key_size, "%s.code", method_key_value); method_key[key_size - 1] = 0; snprintf (value_buffer, value_buffer_size, "0x%04"PFMT64x, code_offset); value_buffer[value_buffer_size - 1] = 0; sdb_array_push (bin->kv, method_key, value_buffer, 0); snprintf (value_buffer, value_buffer_size, "0x%04"PFMT64x, code_size); value_buffer[value_buffer_size - 1] = 0; sdb_array_push (bin->kv, method_key, value_buffer, 0); // generate info key, and place values in method info array snprintf (method_key, key_size, "%s.info", method_key_value); method_key[key_size - 1] = 0; snprintf (value_buffer, value_buffer_size, "%s", fm_type->flags_str); value_buffer[value_buffer_size - 1] = 0; IFDBG eprintf("Adding %s to sdb_array: %s\n", value_buffer, method_key); sdb_array_push (bin->kv, method_key, value_buffer, 0); snprintf (value_buffer, value_buffer_size, "%s", fm_type->class_name); value_buffer[value_buffer_size - 1] = 0; IFDBG eprintf("Adding %s to sdb_array: %s\n", value_buffer, method_key); sdb_array_push (bin->kv, method_key, value_buffer, 0); snprintf (value_buffer, value_buffer_size, "%s", fm_type->name); value_buffer[value_buffer_size - 1] = 0; IFDBG eprintf("Adding %s to sdb_array: %s\n", value_buffer, method_key); sdb_array_push (bin->kv, method_key, value_buffer, 0); snprintf (value_buffer, value_buffer_size, "%s", fm_type->descriptor); value_buffer[value_buffer_size - 1] = 0; IFDBG eprintf("Adding %s to sdb_array: %s\n", value_buffer, method_key); sdb_array_push (bin->kv, method_key, value_buffer, 0); } free (method_key); free (method_key_value); free (value_buffer); if (class_name_inheap) { free (class_name); } } R_API RList *U(r_bin_java_get_args_from_bin)(RBinJavaObj * bin_obj, ut64 addr) { RBinJavaField *fm_type = r_bin_java_get_method_code_attribute_with_addr (bin_obj, addr); return fm_type ? r_bin_java_get_args (fm_type) : NULL; } R_API RList *U(r_bin_java_get_ret_from_bin)(RBinJavaObj * bin_obj, ut64 addr) { RBinJavaField *fm_type = r_bin_java_get_method_code_attribute_with_addr (bin_obj, addr); return fm_type ? r_bin_java_get_ret (fm_type) : NULL; } R_API char *U(r_bin_java_get_fcn_name_from_bin)(RBinJavaObj * bin_obj, ut64 addr) { RBinJavaField *fm_type = r_bin_java_get_method_code_attribute_with_addr (bin_obj, addr); return fm_type && fm_type->name ? strdup (fm_type->name) : NULL; } R_API int U(r_bin_java_is_method_static)(RBinJavaObj * bin_obj, ut64 addr) { RBinJavaField *fm_type = r_bin_java_get_method_code_attribute_with_addr (bin_obj, addr); return fm_type && fm_type->flags & R_BIN_JAVA_METHOD_ACC_STATIC; } R_API int U(r_bin_java_is_method_private)(RBinJavaObj * bin_obj, ut64 addr) { return r_bin_java_is_fm_type_private (r_bin_java_get_method_code_attribute_with_addr (bin_obj, addr)); } R_API int U(r_bin_java_is_method_protected)(RBinJavaObj * bin_obj, ut64 addr) { return r_bin_java_is_fm_type_protected ( r_bin_java_get_method_code_attribute_with_addr (bin_obj, addr)); } R_API int r_bin_java_print_method_idx_summary(RBinJavaObj *bin_obj, ut32 idx) { int res = false; if (idx < r_list_length (bin_obj->methods_list)) { RBinJavaField *fm_type = r_list_get_n (bin_obj->methods_list, idx); r_bin_java_print_method_summary (fm_type); res = true; } return res; } R_API ut32 r_bin_java_get_method_count(RBinJavaObj *bin_obj) { return r_list_length (bin_obj->methods_list); } R_API RList *r_bin_java_get_interface_names(RBinJavaObj *bin) { RList *interfaces_names = r_list_new (); RListIter *iter; RBinJavaInterfaceInfo *ifobj; r_list_foreach (bin->interfaces_list, iter, ifobj) { if (ifobj && ifobj->name) { r_list_append (interfaces_names, strdup (ifobj->name)); } } return interfaces_names; } R_API ut64 r_bin_java_get_main(RBinJavaObj *bin) { if (bin->main_code_attr) { return bin->main_code_attr->info.code_attr.code_offset + bin->loadaddr; } return 0; } R_API RBinJavaObj *r_bin_java_new(const char *file, ut64 loadaddr, Sdb *kv) { RBinJavaObj *bin = R_NEW0 (RBinJavaObj); if (!bin) { return NULL; } bin->file = strdup (file); size_t sz; ut8 *buf = (ut8 *)r_file_slurp (file, &sz); bin->size = sz; if (!buf) { return r_bin_java_free (bin); } if (!r_bin_java_new_bin (bin, loadaddr, kv, buf, bin->size)) { r_bin_java_free (bin); bin = NULL; } free (buf); return bin; } R_API ut64 r_bin_java_get_class_entrypoint(RBinJavaObj *bin) { if (bin->cf2.this_class_entrypoint_code_attr) { return bin->cf2.this_class_entrypoint_code_attr->info.code_attr.code_offset; } return 0; } R_API RList *r_bin_java_get_method_exception_table_with_addr(RBinJavaObj *bin, ut64 addr) { RListIter *iter = NULL, *iter_tmp = NULL; RBinJavaField *fm_type, *res = NULL; if (!bin && R_BIN_JAVA_GLOBAL_BIN) { bin = R_BIN_JAVA_GLOBAL_BIN; } if (!bin) { eprintf ("Attempting to analyse function when the R_BIN_JAVA_GLOBAL_BIN has not been set.\n"); return NULL; } r_list_foreach_safe (bin->methods_list, iter, iter_tmp, fm_type) { ut64 offset = r_bin_java_get_method_code_offset (fm_type) + bin->loadaddr, size = r_bin_java_get_method_code_size (fm_type); if (addr >= offset && addr <= size + offset) { res = fm_type; } } if (res) { RBinJavaAttrInfo *code_attr = r_bin_java_get_method_code_attribute (res); return code_attr->info.code_attr.exception_table; } return NULL; } R_API const RList *r_bin_java_get_methods_list(RBinJavaObj *bin) { if (bin) { return bin->methods_list; } if (R_BIN_JAVA_GLOBAL_BIN) { return R_BIN_JAVA_GLOBAL_BIN->methods_list; } return NULL; } R_API RList *r_bin_java_get_bin_obj_list_thru_obj(RBinJavaObj *bin_obj) { RList *the_list; Sdb *sdb; if (!bin_obj) { return NULL; } sdb = bin_obj->AllJavaBinObjs; if (!sdb) { return NULL; } the_list = r_list_new (); if (!the_list) { return NULL; } sdb_foreach (sdb, sdb_iterate_build_list, (void *) the_list); return the_list; } R_API RList *r_bin_java_extract_all_bin_type_values(RBinJavaObj *bin_obj) { RListIter *fm_type_iter; RList *all_types = r_list_new (); RBinJavaField *fm_type; // get all field types r_list_foreach (bin_obj->fields_list, fm_type_iter, fm_type) { char *desc = NULL; if (!extract_type_value (fm_type->descriptor, &desc)) { return NULL; } IFDBG eprintf("Adding field type: %s\n", desc); r_list_append (all_types, desc); } // get all method types r_list_foreach (bin_obj->methods_list, fm_type_iter, fm_type) { RList *the_list = r_bin_java_extract_type_values (fm_type->descriptor); RListIter *desc_iter; char *str; r_list_foreach (the_list, desc_iter, str) { if (str && *str != '(' && *str != ')') { r_list_append (all_types, strdup (str)); IFDBG eprintf("Adding method type: %s\n", str); } } r_list_free (the_list); } return all_types; } R_API RList *r_bin_java_get_method_definitions(RBinJavaObj *bin) { RBinJavaField *fm_type = NULL; RList *the_list = r_list_new (); if (!the_list) { return NULL; } RListIter *iter = NULL; if (!bin) { return the_list; } r_list_foreach (bin->methods_list, iter, fm_type) { char *method_proto = r_bin_java_get_method_definition (fm_type); // eprintf ("Method prototype: %s\n", method_proto); r_list_append (the_list, method_proto); } return the_list; } R_API RList *r_bin_java_get_field_definitions(RBinJavaObj *bin) { RBinJavaField *fm_type = NULL; RList *the_list = r_list_new (); if (!the_list) { return NULL; } RListIter *iter = NULL; if (!bin) { return the_list; } r_list_foreach (bin->fields_list, iter, fm_type) { char *field_def = r_bin_java_get_field_definition (fm_type); // eprintf ("Field def: %s, %s, %s, %s\n", fm_type->name, fm_type->descriptor, fm_type->flags_str, field_def); r_list_append (the_list, field_def); } return the_list; } R_API RList *r_bin_java_get_import_definitions(RBinJavaObj *bin) { RList *the_list = r_bin_java_get_lib_names (bin); RListIter *iter = NULL; char *new_str; if (!bin || !the_list) { return the_list; } r_list_foreach (the_list, iter, new_str) { while (*new_str) { if (*new_str == '/') { *new_str = '.'; } new_str++; } } return the_list; } R_API RList *r_bin_java_get_field_offsets(RBinJavaObj *bin) { RBinJavaField *fm_type = NULL; RList *the_list = r_list_new (); if (!the_list) { return NULL; } RListIter *iter = NULL; ut64 *paddr = NULL; if (!bin) { return the_list; } the_list->free = free; r_list_foreach (bin->fields_list, iter, fm_type) { paddr = malloc (sizeof(ut64)); if (!paddr) { r_list_free (the_list); return NULL; } *paddr = fm_type->file_offset + bin->loadaddr; // eprintf ("Field def: %s, %s, %s, %s\n", fm_type->name, fm_type->descriptor, fm_type->flags_str, field_def); r_list_append (the_list, paddr); } return the_list; } R_API RList *r_bin_java_get_method_offsets(RBinJavaObj *bin) { RBinJavaField *fm_type = NULL; RList *the_list = r_list_new (); RListIter *iter = NULL; ut64 *paddr = NULL; if (!bin) { return the_list; } the_list->free = free; r_list_foreach (bin->methods_list, iter, fm_type) { paddr = R_NEW0 (ut64); *paddr = fm_type->file_offset + bin->loadaddr; r_list_append (the_list, paddr); } return the_list; } R_API ut16 r_bin_java_calculate_field_access_value(const char *access_flags_str) { return calculate_access_value (access_flags_str, FIELD_ACCESS_FLAGS); } R_API ut16 r_bin_java_calculate_class_access_value(const char *access_flags_str) { return calculate_access_value (access_flags_str, CLASS_ACCESS_FLAGS); } R_API ut16 r_bin_java_calculate_method_access_value(const char *access_flags_str) { return calculate_access_value (access_flags_str, METHOD_ACCESS_FLAGS); } R_API RList *retrieve_all_method_access_string_and_value(void) { return retrieve_all_access_string_and_value (METHOD_ACCESS_FLAGS); } R_API RList *retrieve_all_field_access_string_and_value(void) { return retrieve_all_access_string_and_value (FIELD_ACCESS_FLAGS); } R_API RList *retrieve_all_class_access_string_and_value(void) { return retrieve_all_access_string_and_value (CLASS_ACCESS_FLAGS); } R_API char *r_bin_java_resolve_with_space(RBinJavaObj *obj, int idx) { return r_bin_java_resolve (obj, idx, 1); } R_API char *r_bin_java_resolve_without_space(RBinJavaObj *obj, int idx) { return r_bin_java_resolve (obj, idx, 0); } R_API char *r_bin_java_resolve_b64_encode(RBinJavaObj *BIN_OBJ, ut16 idx) { RBinJavaCPTypeObj *item = NULL, *item2 = NULL; char *class_str = NULL, *name_str = NULL, *desc_str = NULL, *string_str = NULL, *empty = "", *cp_name = NULL, *str = NULL, *out = NULL; int memory_alloc = 0; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { // r_bin_java_new_bin(BIN_OBJ); return NULL; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { cp_name = ((RBinJavaCPTypeMetas *) item->metas->type_info)->name; IFDBG eprintf("java_resolve Resolved: (%d) %s\n", idx, cp_name); } else { return NULL; } if (!strcmp (cp_name, "Class")) { item2 = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); // str = r_bin_java_get_name_from_bin_cp_list (BIN_OBJ, idx-1); class_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!class_str) { class_str = empty; } name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item2); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item2); if (!desc_str) { desc_str = empty; } memory_alloc = strlen (class_str) + strlen (name_str) + strlen (desc_str) + 3; if (memory_alloc) { str = malloc (memory_alloc); if (str) { snprintf (str, memory_alloc, "%s%s", name_str, desc_str); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } if (class_str != empty) { free (class_str); } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else if (strcmp (cp_name, "MethodRef") == 0 || strcmp (cp_name, "FieldRef") == 0 || strcmp (cp_name, "InterfaceMethodRef") == 0) { /* * The MethodRef, FieldRef, and InterfaceMethodRef structures */ class_str = r_bin_java_get_name_from_bin_cp_list (BIN_OBJ, item->info.cp_method.class_idx); if (!class_str) { class_str = empty; } name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item); if (!desc_str) { desc_str = empty; } memory_alloc = strlen (class_str) + strlen (name_str) + strlen (desc_str) + 3; if (memory_alloc) { str = malloc (memory_alloc); if (str) { snprintf (str, memory_alloc, "%s/%s%s", class_str, name_str, desc_str); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } if (class_str != empty) { free (class_str); } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else if (strcmp (cp_name, "String") == 0) { ut32 length = r_bin_java_get_utf8_len_from_bin_cp_list (BIN_OBJ, item->info.cp_string.string_idx); string_str = r_bin_java_get_utf8_from_bin_cp_list (BIN_OBJ, item->info.cp_string.string_idx); str = NULL; IFDBG eprintf("java_resolve String got: (%d) %s\n", item->info.cp_string.string_idx, string_str); if (!string_str) { string_str = empty; length = strlen (empty); } memory_alloc = length + 3; if (memory_alloc) { str = malloc (memory_alloc); if (str) { snprintf (str, memory_alloc, "\"%s\"", string_str); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } IFDBG eprintf("java_resolve String return: %s\n", str); if (string_str != empty) { free (string_str); } } else if (strcmp (cp_name, "Utf8") == 0) { ut64 sz = item->info.cp_utf8.length ? item->info.cp_utf8.length + 10 : 10; str = malloc (sz); memset (str, 0, sz); if (sz > 10) { r_base64_encode (str, item->info.cp_utf8.bytes, item->info.cp_utf8.length); } } else if (strcmp (cp_name, "Long") == 0) { str = malloc (34); if (str) { snprintf (str, 34, "0x%"PFMT64x, r_bin_java_raw_to_long (item->info.cp_long.bytes.raw, 0)); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } else if (strcmp (cp_name, "Double") == 0) { str = malloc (1000); if (str) { snprintf (str, 1000, "%f", r_bin_java_raw_to_double (item->info.cp_double.bytes.raw, 0)); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } else if (strcmp (cp_name, "Integer") == 0) { str = calloc (34, 1); if (str) { snprintf (str, 34, "0x%08x", R_BIN_JAVA_UINT (item->info.cp_integer.bytes.raw, 0)); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } else if (strcmp (cp_name, "Float") == 0) { str = malloc (34); if (str) { snprintf (str, 34, "%f", R_BIN_JAVA_FLOAT (item->info.cp_float.bytes.raw, 0)); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } else if (!strcmp (cp_name, "NameAndType")) { name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item); if (!desc_str) { desc_str = empty; } memory_alloc = strlen (name_str) + strlen (desc_str) + 3; if (memory_alloc) { str = malloc (memory_alloc); if (str) { snprintf (str, memory_alloc, "%s %s", name_str, desc_str); out = r_base64_encode_dyn ((const char *) str, strlen (str)); free (str); str = out; } } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else { str = r_base64_encode_dyn ((const char *) "(null)", 6); } return str; } R_API ut64 r_bin_java_resolve_cp_idx_address(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaCPTypeObj *item = NULL; ut64 addr = -1; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { return -1; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { addr = item->file_offset + item->loadaddr; } return addr; } R_API char *r_bin_java_resolve_cp_idx_to_string(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaCPTypeObj *item = NULL; char *value = NULL; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { return NULL; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { value = ((RBinJavaCPTypeMetas *) item->metas->type_info)-> allocs->stringify_obj (item); } return value; } R_API int r_bin_java_resolve_cp_idx_print_summary(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaCPTypeObj *item = NULL; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { return false; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { ((RBinJavaCPTypeMetas *) item->metas->type_info)-> allocs->print_summary (item); } else { eprintf ("Error: Invalid CP Object.\n"); } return item ? true : false; } R_API ConstJavaValue *U(r_bin_java_resolve_to_const_value)(RBinJavaObj * BIN_OBJ, int idx) { // TODO XXX FIXME add a size parameter to the str when it is passed in RBinJavaCPTypeObj *item = NULL, *item2 = NULL; ConstJavaValue *result = R_NEW0 (ConstJavaValue); if (!result) { return NULL; } char *class_str = NULL, *name_str = NULL, *desc_str = NULL, *string_str = NULL, *empty = "", *cp_name = NULL; result->type = "unknown"; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { // r_bin_java_new_bin(BIN_OBJ); return result; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (!item) { return result; } cp_name = ((RBinJavaCPTypeMetas *) item->metas->type_info)->name; IFDBG eprintf("java_resolve Resolved: (%d) %s\n", idx, cp_name); if (strcmp (cp_name, "Class") == 0) { item2 = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); // str = r_bin_java_get_name_from_bin_cp_list (BIN_OBJ, idx-1); class_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!class_str) { class_str = empty; } name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item2); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item2); if (!desc_str) { desc_str = empty; } result->value._ref = R_NEW0 (_JavaRef); result->type = "ref"; result->value._ref->class_name = strdup (class_str); result->value._ref->name = strdup (name_str); result->value._ref->desc = strdup (desc_str); if (class_str != empty) { free (class_str); } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else if (strcmp (cp_name, "MethodRef") == 0 || strcmp (cp_name, "FieldRef") == 0 || strcmp (cp_name, "InterfaceMethodRef") == 0) { /* * The MethodRef, FieldRef, and InterfaceMethodRef structures */ class_str = r_bin_java_get_name_from_bin_cp_list (BIN_OBJ, item->info.cp_method.class_idx); if (!class_str) { class_str = empty; } name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item); if (!desc_str) { desc_str = empty; } result->value._ref = R_NEW0 (_JavaRef); result->type = "ref"; result->value._ref->class_name = strdup (class_str); result->value._ref->name = strdup (name_str); result->value._ref->desc = strdup (desc_str); if (class_str != empty) { free (class_str); } if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } } else if (strcmp (cp_name, "String") == 0) { ut32 length = r_bin_java_get_utf8_len_from_bin_cp_list (BIN_OBJ, item->info.cp_string.string_idx); string_str = r_bin_java_get_utf8_from_bin_cp_list (BIN_OBJ, item->info.cp_string.string_idx); IFDBG eprintf("java_resolve String got: (%d) %s\n", item->info.cp_string.string_idx, string_str); if (!string_str) { string_str = empty; length = strlen (empty); } result->type = "str"; result->value._str = R_NEW0 (struct java_const_value_str_t); result->value._str->len = length; if (length > 0) { result->value._str->str = r_str_ndup (string_str, length); } else { result->value._str->str = strdup (""); } if (string_str != empty) { free (string_str); } } else if (strcmp (cp_name, "Utf8") == 0) { result->type = "str"; result->value._str = R_NEW0 (struct java_const_value_str_t); result->value._str->str = malloc (item->info.cp_utf8.length); result->value._str->len = item->info.cp_utf8.length; memcpy (result->value._str->str, item->info.cp_utf8.bytes, item->info.cp_utf8.length); } else if (strcmp (cp_name, "Long") == 0) { result->type = "long"; result->value._long = r_bin_java_raw_to_long (item->info.cp_long.bytes.raw, 0); } else if (strcmp (cp_name, "Double") == 0) { result->type = "double"; result->value._double = r_bin_java_raw_to_double (item->info.cp_double.bytes.raw, 0); } else if (strcmp (cp_name, "Integer") == 0) { result->type = "int"; result->value._int = R_BIN_JAVA_UINT (item->info.cp_integer.bytes.raw, 0); } else if (strcmp (cp_name, "Float") == 0) { result->type = "float"; result->value._float = R_BIN_JAVA_FLOAT (item->info.cp_float.bytes.raw, 0); } else if (strcmp (cp_name, "NameAndType") == 0) { result->value._ref = R_NEW0 (struct java_const_value_ref_t); result->type = "ref"; name_str = r_bin_java_get_item_name_from_bin_cp_list (BIN_OBJ, item); if (!name_str) { name_str = empty; } desc_str = r_bin_java_get_item_desc_from_bin_cp_list (BIN_OBJ, item); if (!desc_str) { desc_str = empty; } result->value._ref->class_name = strdup (empty); result->value._ref->name = strdup (name_str); result->value._ref->desc = strdup (desc_str); if (name_str != empty) { free (name_str); } if (desc_str != empty) { free (desc_str); } result->value._ref->is_method = r_bin_java_does_cp_idx_ref_method (BIN_OBJ, idx); result->value._ref->is_field = r_bin_java_does_cp_idx_ref_field (BIN_OBJ, idx); } return result; } R_API void U(r_bin_java_free_const_value)(ConstJavaValue * cp_value) { char first_char = cp_value && cp_value->type ? *cp_value->type : 0, second_char = cp_value && cp_value->type ? *(cp_value->type + 1) : 0; switch (first_char) { case 'r': if (cp_value && cp_value->value._ref) { free (cp_value->value._ref->class_name); free (cp_value->value._ref->name); free (cp_value->value._ref->desc); } break; case 's': if (second_char == 't' && cp_value->value._str) { free (cp_value->value._str->str); } break; } free (cp_value); } R_API char *r_bin_java_get_field_name(RBinJavaObj *bin_obj, ut32 idx) { char *name = NULL; if (idx < r_list_length (bin_obj->fields_list)) { RBinJavaField *fm_type = r_list_get_n (bin_obj->fields_list, idx); name = strdup (fm_type->name); } return name; } R_API int r_bin_java_print_field_idx_summary(RBinJavaObj *bin_obj, ut32 idx) { int res = false; if (idx < r_list_length (bin_obj->fields_list)) { RBinJavaField *fm_type = r_list_get_n (bin_obj->fields_list, idx); r_bin_java_print_field_summary (fm_type); res = true; } return res; } R_API ut32 r_bin_java_get_field_count(RBinJavaObj *bin_obj) { return r_list_length (bin_obj->fields_list); } R_API RList *r_bin_java_get_field_num_name(RBinJavaObj *bin_obj) { ut32 i = 0; RBinJavaField *fm_type; RListIter *iter = NULL; RList *res = r_list_newf (free); r_list_foreach (bin_obj->fields_list, iter, fm_type) { ut32 len = strlen (fm_type->name) + 30; char *str = malloc (len); if (!str) { r_list_free (res); return NULL; } snprintf (str, len, "%d %s", i, fm_type->name); ++i; r_list_append (res, str); } return res; } R_API RList *r_bin_java_find_cp_const_by_val_utf8(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len) { RList *res = r_list_newf (free); ut32 *v = NULL; RListIter *iter; RBinJavaCPTypeObj *cp_obj; IFDBG eprintf("In UTF-8 Looking for %s\n", bytes); r_list_foreach (bin_obj->cp_list, iter, cp_obj) { if (cp_obj->tag == R_BIN_JAVA_CP_UTF8) { IFDBG eprintf("In UTF-8 Looking @ %s\n", cp_obj->info.cp_utf8.bytes); IFDBG eprintf("UTF-8 len = %d and memcmp = %d\n", cp_obj->info.cp_utf8.length, memcmp (bytes, cp_obj->info.cp_utf8.bytes, len)); if (len == cp_obj->info.cp_utf8.length && !memcmp (bytes, cp_obj->info.cp_utf8.bytes, len)) { v = malloc (sizeof (ut32)); if (!v) { r_list_free (res); return NULL; } *v = cp_obj->metas->ord; IFDBG eprintf("Found a match adding idx: %d\n", *v); r_list_append (res, v); } } } return res; } R_API RList *r_bin_java_find_cp_const_by_val_int(RBinJavaObj *bin_obj, const ut8 *bytes, ut32 len) { RList *res = r_list_newf (free); ut32 *v = NULL; RListIter *iter; RBinJavaCPTypeObj *cp_obj; eprintf ("Looking for 0x%08x\n", (ut32) R_BIN_JAVA_UINT (bytes, 0)); r_list_foreach (bin_obj->cp_list, iter, cp_obj) { if (cp_obj->tag == R_BIN_JAVA_CP_INTEGER) { if (len == 4 && R_BIN_JAVA_UINT (bytes, 0) == R_BIN_JAVA_UINT (cp_obj->info.cp_integer.bytes.raw, 0)) { v = malloc (sizeof (ut32)); if (!v) { r_list_free (res); return NULL; } *v = cp_obj->idx; r_list_append (res, v); } } } return res; } R_API char r_bin_java_resolve_cp_idx_tag(RBinJavaObj *BIN_OBJ, int idx) { RBinJavaCPTypeObj *item = NULL; if (BIN_OBJ && BIN_OBJ->cp_count < 1) { // r_bin_java_new_bin(BIN_OBJ); return R_BIN_JAVA_CP_UNKNOWN; } item = (RBinJavaCPTypeObj *) r_bin_java_get_item_from_bin_cp_list (BIN_OBJ, idx); if (item) { return item->tag; } return R_BIN_JAVA_CP_UNKNOWN; } R_API int U(r_bin_java_integer_cp_set)(RBinJavaObj * bin, ut16 idx, ut32 val) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); if (!cp_obj) { return false; } ut8 bytes[4] = { 0 }; if (cp_obj->tag != R_BIN_JAVA_CP_INTEGER && cp_obj->tag != R_BIN_JAVA_CP_FLOAT) { eprintf ("Not supporting the overwrite of CP Objects with one of a different size.\n"); return false; } r_bin_java_check_reset_cp_obj (cp_obj, R_BIN_JAVA_CP_INTEGER); cp_obj->tag = R_BIN_JAVA_CP_INTEGER; memcpy (bytes, (const char *) &val, 4); val = R_BIN_JAVA_UINT (bytes, 0); memcpy (&cp_obj->info.cp_integer.bytes.raw, (const char *) &val, 4); return true; } R_API int U(r_bin_java_float_cp_set)(RBinJavaObj * bin, ut16 idx, float val) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); if (!cp_obj) { return false; } ut8 bytes[4] = { 0 }; if (cp_obj->tag != R_BIN_JAVA_CP_INTEGER && cp_obj->tag != R_BIN_JAVA_CP_FLOAT) { eprintf ("Not supporting the overwrite of CP Objects with one of a different size.\n"); return false; } r_bin_java_check_reset_cp_obj (cp_obj, R_BIN_JAVA_CP_FLOAT); cp_obj->tag = R_BIN_JAVA_CP_FLOAT; memcpy (bytes, (const char *) &val, 4); float *foo = (float*) bytes; val = *foo; //(float)R_BIN_JAVA_UINT (bytes, 0); memcpy (&cp_obj->info.cp_float.bytes.raw, (const char *) &val, 4); return true; } R_API int U(r_bin_java_long_cp_set)(RBinJavaObj * bin, ut16 idx, ut64 val) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); if (!cp_obj) { return false; } ut8 bytes[8] = { 0 }; if (cp_obj->tag != R_BIN_JAVA_CP_LONG && cp_obj->tag != R_BIN_JAVA_CP_DOUBLE) { eprintf ("Not supporting the overwrite of CP Objects with one of a different size.\n"); return false; } r_bin_java_check_reset_cp_obj (cp_obj, R_BIN_JAVA_CP_LONG); cp_obj->tag = R_BIN_JAVA_CP_LONG; memcpy (bytes, (const char *) &val, 8); val = r_bin_java_raw_to_long (bytes, 0); memcpy (&cp_obj->info.cp_long.bytes.raw, (const char *) &val, 8); return true; } R_API int U(r_bin_java_double_cp_set)(RBinJavaObj * bin, ut16 idx, ut32 val) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); if (!cp_obj) { return false; } ut8 bytes[8] = { 0 }; if (cp_obj->tag != R_BIN_JAVA_CP_LONG && cp_obj->tag != R_BIN_JAVA_CP_DOUBLE) { eprintf ("Not supporting the overwrite of CP Objects with one of a different size.\n"); return false; } r_bin_java_check_reset_cp_obj (cp_obj, R_BIN_JAVA_CP_DOUBLE); cp_obj->tag = R_BIN_JAVA_CP_DOUBLE; ut64 val64 = val; memcpy (bytes, (const char *) &val64, 8); val64 = r_bin_java_raw_to_long (bytes, 0); memcpy (&cp_obj->info.cp_double.bytes.raw, (const char *) &val64, 8); return true; } R_API int U(r_bin_java_utf8_cp_set)(RBinJavaObj * bin, ut16 idx, const ut8 * buffer, ut32 len) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); if (!cp_obj) { return false; } eprintf ("Writing %d byte(s) (%s)\n", len, buffer); // r_bin_java_check_reset_cp_obj(cp_obj, R_BIN_JAVA_CP_INTEGER); if (cp_obj->tag != R_BIN_JAVA_CP_UTF8) { eprintf ("Not supporting the overwrite of CP Objects with one of a different size.\n"); return false; } if (cp_obj->info.cp_utf8.length != len) { eprintf ("Not supporting the resize, rewriting utf8 string up to %d byte(s).\n", cp_obj->info.cp_utf8.length); if (cp_obj->info.cp_utf8.length > len) { eprintf ("Remaining %d byte(s) will be filled with \\x00.\n", cp_obj->info.cp_utf8.length - len); } } memcpy (cp_obj->info.cp_utf8.bytes, buffer, cp_obj->info.cp_utf8.length); if (cp_obj->info.cp_utf8.length > len) { memset (cp_obj->info.cp_utf8.bytes + len, 0, cp_obj->info.cp_utf8.length - len); } return true; } R_API ut8 *r_bin_java_cp_get_bytes(ut8 tag, ut32 *out_sz, const ut8 *buf, const ut64 len) { if (!out_sz) { return NULL; } if (out_sz) { *out_sz = 0; } switch (tag) { case R_BIN_JAVA_CP_INTEGER: case R_BIN_JAVA_CP_FLOAT: return r_bin_java_cp_get_4bytes (tag, out_sz, buf, len); case R_BIN_JAVA_CP_LONG: case R_BIN_JAVA_CP_DOUBLE: return r_bin_java_cp_get_8bytes (tag, out_sz, buf, len); case R_BIN_JAVA_CP_UTF8: return r_bin_java_cp_get_utf8 (tag, out_sz, buf, len); } return NULL; } R_API ut32 r_bin_java_cp_get_size(RBinJavaObj *bin, ut16 idx) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); switch (cp_obj->tag) { case R_BIN_JAVA_CP_INTEGER: case R_BIN_JAVA_CP_FLOAT: return 1 + 4; case R_BIN_JAVA_CP_LONG: case R_BIN_JAVA_CP_DOUBLE: return 1 + 8; case R_BIN_JAVA_CP_UTF8: return 1 + 2 + cp_obj->info.cp_utf8.length; } return 0; } R_API ut64 r_bin_java_get_method_start(RBinJavaObj *bin, RBinJavaField *fm_type) { return r_bin_java_get_method_code_offset (fm_type) + bin->loadaddr; } R_API ut64 r_bin_java_get_method_end(RBinJavaObj *bin, RBinJavaField *fm_type) { return r_bin_java_get_method_code_offset (fm_type) + bin->loadaddr + +r_bin_java_get_method_code_size (fm_type); } R_API ut8 *U(r_bin_java_cp_append_method_ref)(RBinJavaObj * bin, ut32 * out_sz, ut16 cn_idx, ut16 fn_idx, ut16 ft_idx) { return r_bin_java_cp_get_fref_bytes (bin, out_sz, R_BIN_JAVA_CP_METHODREF, cn_idx, fn_idx, ft_idx); } R_API ut8 *U(r_bin_java_cp_append_field_ref)(RBinJavaObj * bin, ut32 * out_sz, ut16 cn_idx, ut16 fn_idx, ut16 ft_idx) { return r_bin_java_cp_get_fref_bytes (bin, out_sz, R_BIN_JAVA_CP_FIELDREF, cn_idx, fn_idx, ft_idx); } R_API char *r_bin_java_unmangle_without_flags(const char *name, const char *descriptor) { return r_bin_java_unmangle (NULL, name, descriptor); } R_API void U(r_bin_java_print_stack_map_append_frame_summary)(RBinJavaStackMapFrame * obj) { RListIter *iter, *iter_tmp; RList *ptrList; RBinJavaVerificationObj *ver_obj; Eprintf ("Stack Map Frame Information\n"); Eprintf (" Tag Value = 0x%02x Name: %s\n", obj->tag, ((RBinJavaStackMapFrameMetas *) obj->metas->type_info)->name); Eprintf (" Offset: 0x%08"PFMT64x "\n", obj->file_offset); Eprintf (" Local Variable Count = 0x%04x\n", obj->number_of_locals); Eprintf (" Local Variables:\n"); ptrList = obj->local_items; r_list_foreach_safe (ptrList, iter, iter_tmp, ver_obj) { r_bin_java_print_verification_info_summary (ver_obj); } Eprintf (" Stack Items Count = 0x%04x\n", obj->number_of_stack_items); Eprintf (" Stack Items:\n"); ptrList = obj->stack_items; r_list_foreach_safe (ptrList, iter, iter_tmp, ver_obj) { r_bin_java_print_verification_info_summary (ver_obj); } } R_API void U(r_bin_java_stack_frame_default_free)(void *s) { RBinJavaStackMapFrame *stack_frame = s; if (stack_frame) { free (stack_frame->metas); free (stack_frame); } } R_API void U(r_bin_java_stack_frame_do_nothing_free)(void /*RBinJavaStackMapFrame*/ *stack_frame) {} R_API void U(r_bin_java_stack_frame_do_nothing_new)(RBinJavaObj * bin, RBinJavaStackMapFrame * stack_frame, ut64 offset) {} R_API RBinJavaCPTypeMetas *U(r_bin_java_get_cp_meta_from_tag)(ut8 tag) { ut16 i = 0; // set default to unknown. RBinJavaCPTypeMetas *res = &R_BIN_JAVA_CP_METAS[2]; for (i = 0; i < R_BIN_JAVA_CP_METAS_SZ; i++) { if (tag == R_BIN_JAVA_CP_METAS[i].tag) { res = &R_BIN_JAVA_CP_METAS[i]; break; } } return res; } R_API ut8 *U(r_bin_java_cp_append_ref_cname_fname_ftype)(RBinJavaObj * bin, ut32 * out_sz, ut8 tag, const char *cname, const ut32 c_len, const char *fname, const ut32 f_len, const char *tname, const ut32 t_len) { ut32 cn_len = 0, fn_len = 0, ft_len = 0, total_len; ut16 cn_idx = 0, fn_idx = 0, ft_idx = 0; ut8 *bytes = NULL, *cn_bytes = NULL, *fn_bytes = NULL, *ft_bytes = NULL, *cref_bytes = NULL, *fref_bytes = NULL, *fnt_bytes = NULL; *out_sz = 0; cn_bytes = r_bin_java_cp_get_utf8 (R_BIN_JAVA_CP_UTF8, &cn_len, (const ut8 *) cname, c_len); cn_idx = bin->cp_idx + 1; if (cn_bytes) { fn_bytes = r_bin_java_cp_get_utf8 (R_BIN_JAVA_CP_UTF8, &fn_len, (const ut8 *) fname, f_len); fn_idx = bin->cp_idx + 2; } if (fn_bytes) { ft_bytes = r_bin_java_cp_get_utf8 (R_BIN_JAVA_CP_UTF8, &ft_len, (const ut8 *) tname, t_len); ft_idx = bin->cp_idx + 3; } if (cn_bytes && fn_bytes && ft_bytes) { ut32 cref_len = 0, fnt_len = 0, fref_len = 0; ut32 cref_idx = 0, fnt_idx = 0; cref_bytes = r_bin_java_cp_get_classref (bin, &cref_len, NULL, 0, cn_idx); cref_idx = bin->cp_idx + 3; fnt_bytes = r_bin_java_cp_get_name_type (bin, &fnt_len, fn_idx, ft_idx); fnt_idx = bin->cp_idx + 4; fref_bytes = r_bin_java_cp_get_2_ut16 (bin, &fref_len, tag, cref_idx, fnt_idx); if (cref_bytes && fref_bytes && fnt_bytes) { total_len = cn_len + fn_len + ft_len + cref_len + fnt_len + fref_len + 2; if (total_len < cn_len) { goto beach; } bytes = calloc (1, total_len); // class name bytes if (*out_sz + cn_len >= total_len) { goto beach; } memcpy (bytes, cn_bytes + *out_sz, cn_len); *out_sz += cn_len; // field name bytes if (*out_sz + fn_len >= total_len) { goto beach; } memcpy (bytes, fn_bytes + *out_sz, fn_len); *out_sz += fn_len; // field type bytes if (*out_sz + ft_len >= total_len) { goto beach; } memcpy (bytes, ft_bytes + *out_sz, ft_len); *out_sz += ft_len; // class ref bytes if (*out_sz + cref_len >= total_len) { goto beach; } memcpy (bytes, cref_bytes + *out_sz, cref_len); *out_sz += fn_len; // field name and type bytes if (*out_sz + fnt_len >= total_len) { goto beach; } memcpy (bytes, fnt_bytes + *out_sz, fnt_len); *out_sz += fnt_len; // field ref bytes if (*out_sz + fref_len >= total_len) { goto beach; } memcpy (bytes, fref_bytes + *out_sz, fref_len); *out_sz += fref_len; } } beach: free (cn_bytes); free (ft_bytes); free (fn_bytes); free (fnt_bytes); free (fref_bytes); free (cref_bytes); return bytes; } R_API ut8 *U(r_bin_java_cp_get_method_ref)(RBinJavaObj * bin, ut32 * out_sz, ut16 class_idx, ut16 name_and_type_idx) { return r_bin_java_cp_get_fm_ref (bin, out_sz, R_BIN_JAVA_CP_METHODREF, class_idx, name_and_type_idx); } R_API ut8 *U(r_bin_java_cp_get_field_ref)(RBinJavaObj * bin, ut32 * out_sz, ut16 class_idx, ut16 name_and_type_idx) { return r_bin_java_cp_get_fm_ref (bin, out_sz, R_BIN_JAVA_CP_FIELDREF, class_idx, name_and_type_idx); } R_API void U(deinit_java_type_null)(void) { free (R_BIN_JAVA_NULL_TYPE.metas); } R_API RBinJavaCPTypeObj *r_bin_java_get_item_from_cp(RBinJavaObj *bin, int i) { if (i < 1 || i > bin->cf.cp_count) { return &R_BIN_JAVA_NULL_TYPE; } RBinJavaCPTypeObj *obj = (RBinJavaCPTypeObj *) r_list_get_n (bin->cp_list, i); return obj ? obj : &R_BIN_JAVA_NULL_TYPE; } R_API void U(copy_type_info_to_stack_frame_list)(RList * type_list, RList * sf_list) { RListIter *iter, *iter_tmp; RBinJavaVerificationObj *ver_obj, *new_ver_obj; if (!type_list || !sf_list) { return; } r_list_foreach_safe (type_list, iter, iter_tmp, ver_obj) { new_ver_obj = (RBinJavaVerificationObj *) malloc (sizeof (RBinJavaVerificationObj)); // FIXME: how to handle failed memory allocation? if (new_ver_obj && ver_obj) { memcpy (new_ver_obj, ver_obj, sizeof (RBinJavaVerificationObj)); if (!r_list_append (sf_list, (void *) new_ver_obj)) { R_FREE (new_ver_obj); } } else { R_FREE (new_ver_obj); } } } R_API void U(copy_type_info_to_stack_frame_list_up_to_idx)(RList * type_list, RList * sf_list, ut64 idx) { RListIter *iter, *iter_tmp; RBinJavaVerificationObj *ver_obj, *new_ver_obj; ut32 pos = 0; if (!type_list || !sf_list) { return; } r_list_foreach_safe (type_list, iter, iter_tmp, ver_obj) { new_ver_obj = (RBinJavaVerificationObj *) malloc (sizeof (RBinJavaVerificationObj)); // FIXME: how to handle failed memory allocation? if (new_ver_obj && ver_obj) { memcpy (new_ver_obj, ver_obj, sizeof (RBinJavaVerificationObj)); if (!r_list_append (sf_list, (void *) new_ver_obj)) { R_FREE (new_ver_obj); } } else { R_FREE (new_ver_obj); } pos++; if (pos == idx) { break; } } } R_API ut8 *r_bin_java_cp_get_idx_bytes(RBinJavaObj *bin, ut16 idx, ut32 *out_sz) { RBinJavaCPTypeObj *cp_obj = r_bin_java_get_item_from_bin_cp_list (bin, idx); if (!cp_obj || !out_sz) { return NULL; } if (out_sz) { *out_sz = 0; } switch (cp_obj->tag) { case R_BIN_JAVA_CP_INTEGER: case R_BIN_JAVA_CP_FLOAT: return r_bin_java_cp_get_4bytes (cp_obj->tag, out_sz, cp_obj->info.cp_integer.bytes.raw, 5); case R_BIN_JAVA_CP_LONG: case R_BIN_JAVA_CP_DOUBLE: return r_bin_java_cp_get_4bytes (cp_obj->tag, out_sz, cp_obj->info.cp_long.bytes.raw, 9); case R_BIN_JAVA_CP_UTF8: // eprintf ("Getting idx: %d = %p (3+0x%"PFMT64x")\n", idx, cp_obj, cp_obj->info.cp_utf8.length); if (cp_obj->info.cp_utf8.length > 0) { return r_bin_java_cp_get_utf8 (cp_obj->tag, out_sz, cp_obj->info.cp_utf8.bytes, cp_obj->info.cp_utf8.length); } } return NULL; } R_API int r_bin_java_valid_class(const ut8 *buf, ut64 buf_sz) { RBinJavaObj *bin = R_NEW0 (RBinJavaObj), *cur_bin = R_BIN_JAVA_GLOBAL_BIN; if (!bin) { return false; } int res = r_bin_java_load_bin (bin, buf, buf_sz); if (bin->calc_size == buf_sz) { res = true; } r_bin_java_free (bin); R_BIN_JAVA_GLOBAL_BIN = cur_bin; return res; } R_API ut64 r_bin_java_calc_class_size(ut8 *bytes, ut64 size) { RBinJavaObj *bin = R_NEW0 (RBinJavaObj); if (!bin) { return false; } RBinJavaObj *cur_bin = R_BIN_JAVA_GLOBAL_BIN; ut64 bin_size = UT64_MAX; if (bin) { if (r_bin_java_load_bin (bin, bytes, size)) { bin_size = bin->calc_size; } r_bin_java_free (bin); R_BIN_JAVA_GLOBAL_BIN = cur_bin; } return bin_size; } R_API int U(r_bin_java_get_cp_idx_with_name)(RBinJavaObj * bin_obj, const char *name, ut32 len) { RListIter *iter; RBinJavaCPTypeObj *obj; r_list_foreach (bin_obj->cp_list, iter, obj) { if (obj->tag == R_BIN_JAVA_CP_UTF8) { if (!strncmp (name, (const char *) obj->info.cp_utf8.bytes, len)) { return obj->metas->ord; } } } return 0; } R_API void r_bin_java_external_printf(PrintfCallback cb) { Eprintf = cb ? cb : (PrintfCallback) printf; }