#!/usr/bin/python3 # Copyright (C) 2017 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # # Generates the src/art/Test988Intrinsics.java file. # Re-run this every time art/compiler/intrinics_list.h is modified. # # $> python3.4 gen_srcs.py > src/art/Test988Intrinsics.java # import argparse import os import re import collections import sys from string import Template # Relative path to art/runtime/intrinsics_list.h INTRINSICS_LIST_H = os.path.dirname(os.path.realpath(__file__)) + "/../../runtime/intrinsics_list.h" # Macro parameter index to V(). Negative means from the end. IDX_STATIC_OR_VIRTUAL = 1 IDX_SIGNATURE = -1 IDX_METHOD_NAME = -2 IDX_CLASS_NAME = -3 # Exclude all hidden API. KLASS_BLOCK_LIST = ['sun.misc.Unsafe', 'libcore.io.Memory', 'java.lang.StringFactory', 'java.lang.invoke.MethodHandle', # invokes are tested by 956-method-handles 'java.lang.invoke.VarHandle' ] # TODO(b/65872996): will tested separately METHOD_BLOCK_LIST = [('java.lang.ref.Reference', 'getReferent'), ('java.lang.String', 'getCharsNoCheck'), ('java.lang.System', 'arraycopy')] # arraycopy has a manual test. # When testing a virtual function, it needs to operate on an instance. # These instances will be created with the following values, # otherwise a default 'new T()' is used. KLASS_INSTANCE_INITIALIZERS = { 'java.lang.String' : '"some large string"', 'java.lang.StringBuffer' : 'new java.lang.StringBuffer("some large string buffer")', 'java.lang.StringBuilder' : 'new java.lang.StringBuilder("some large string builder")', 'java.lang.ref.Reference' : 'new java.lang.ref.WeakReference(new Object())' }; OUTPUT_TPL = Template(""" /* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // AUTO-GENENERATED by gen_srcs.py: DO NOT EDIT HERE DIRECTLY. // // $$> python3.4 gen_srcs.py > src/art/Test988Intrinsics.java // // RUN ABOVE COMMAND TO REGENERATE THIS FILE. package art; class Test988Intrinsics { // Pre-initialize *all* instance variables used so that their constructors are not in the trace. $static_fields static void initialize() { // Ensure all static variables are initialized. // In addition, pre-load classes here so that we don't see diverging class loading traces. $initialize_classes } static void test() { // Call each intrinsic from art/runtime/intrinsics_list.h to make sure they are traced. $test_body } } """) JNI_TYPES = { 'Z' : 'boolean', 'B' : 'byte', 'C' : 'char', 'S' : 'short', 'I' : 'int', 'J' : 'long', 'F' : 'float', 'D' : 'double', 'L' : 'object' }; debug_printing_enabled = False def debug_print(x): if debug_printing_enabled: print(x, file=sys.stderr) # Parse JNI sig into a list, e.g. "II" -> ['I', 'I'], '[[IJ' -> ['[[I', 'J'], etc. def sig_to_parameter_type_list(sig): sig = re.sub(r'[(](.*)[)].*', r'\1', sig) lst = [] obj = "" is_obj = False is_array = False for x in sig: if is_obj: obj = obj + x if x == ";": is_obj = False lst.append(obj) obj = "" elif is_array: obj = obj + x if x != "[": is_array = False lst.append(obj) obj = "" else: if x == "[": obj = "[" is_array = True elif x == "L": obj = "L" is_obj = True else: lst.append(x) return lst # Convert a single JNI descriptor into a pretty java name, e.g. "[I" -> "int[]", etc. def javafy_name(kls_name): if kls_name.startswith("L"): kls_name = kls_name.lstrip("L").rstrip(";") return kls_name.replace("/", ".") elif kls_name.startswith("["): array_count = kls_name.count("[") non_array = javafy_name(kls_name.lstrip("[")) return non_array + ("[]" * array_count) return JNI_TYPES.get(kls_name, kls_name) def extract_staticness(static_or_virtual): if static_or_virtual == "kStatic": return 'static' return 'virtual' # kVirtual, kDirect class MethodInfo: def __init__(self, staticness, pretty_params, method, kls): # 'virtual' or 'static' self.staticness = staticness # list of e.g. ['int', 'double', 'java.lang.String'] etc self.parameters = pretty_params # e.g. 'toString' self.method_name = method # e.g. 'java.lang.String' self.klass = kls def __str__(self): return "MethodInfo " + str(self.__dict__) def placeholder_parameters(self): placeholder_values = { 'boolean' : 'false', 'byte' : '(byte)0', 'char' : "'x'", 'short' : '(short)0', 'int' : '0', 'long' : '0L', 'float' : '0.0f', 'double' : '0.0' } def object_placeholder(name): if name == "java.lang.String": return '"hello"' else: return "(%s)null" %(name) return [ placeholder_values.get(param, object_placeholder(param)) for param in self.parameters ] def placeholder_instance_value(self): return KLASS_INSTANCE_INITIALIZERS.get(self.klass, 'new %s()' %(self.klass)) def is_blocklisted(self): for blk in KLASS_BLOCK_LIST: if self.klass.startswith(blk): return True return (self.klass, self.method_name) in METHOD_BLOCK_LIST # parse the V(...) \ list of items into a MethodInfo def parse_method_info(items): def get_item(idx): return items[idx].strip().strip("\"") staticness = get_item(IDX_STATIC_OR_VIRTUAL) sig = get_item(IDX_SIGNATURE) method = get_item(IDX_METHOD_NAME) kls = get_item(IDX_CLASS_NAME) debug_print ((sig, method, kls)) staticness = extract_staticness(staticness) kls = javafy_name(kls) param_types = sig_to_parameter_type_list(sig) pretty_params = param_types pretty_params = [javafy_name(i) for i in param_types] return MethodInfo(staticness, pretty_params, method, kls) # parse a line containing ' V(...)' into a MethodInfo def parse_line(line): line = line.strip() if not line.startswith("V("): return None line = re.sub(r'V[(](.*)[)]', r'\1', line) debug_print(line) items = line.split(",") method_info = parse_method_info(items) return method_info # Generate all the MethodInfo that we parse from intrinsics_list.h def parse_all_method_infos(): with open(INTRINSICS_LIST_H) as f: for line in f: s = parse_line(line) if s is not None: yield s # Format a receiver name. For statics, it's the class name, for receivers, it's an instance variable def format_receiver_name(method_info): receiver = method_info.klass if method_info.staticness == 'virtual': receiver = "instance_" + method_info.klass.replace(".", "_") return receiver # Format a placeholder call with placeholder method parameters to the requested method. def format_call_to(method_info): placeholder_args = ", ".join(method_info.placeholder_parameters()) receiver = format_receiver_name(method_info) return ("%s.%s(%s);" %(receiver, method_info.method_name, placeholder_args)) # Format a static variable with an instance that could be used as the receiver # (or None for non-static methods). def format_instance_variable(method_info): if method_info.staticness == 'static': return None return "static %s %s = %s;" %(method_info.klass, format_receiver_name(method_info), method_info.placeholder_instance_value()) def format_initialize_klass(method_info): return "%s.class.toString();" %(method_info.klass) def indent_list(lst, indent): return [' ' * indent + i for i in lst] def main(): global debug_printing_enabled parser = argparse.ArgumentParser(description='Generate art/test/988-method-trace/src/art/Test988Intrinsics.java') parser.add_argument('-d', '--debug', action='store_true', dest='debug', help='Print extra debugging information to stderr.') parser.add_argument('output_file', nargs='?', metavar='', default=sys.stdout, type=argparse.FileType('w'), help='Destination file to write to (default: stdout).') args = parser.parse_args() debug_printing_enabled = args.debug ##### call_str_list = [] instance_variable_dict = collections.OrderedDict() initialize_klass_dict = collections.OrderedDict() for i in parse_all_method_infos(): debug_print(i) if i.is_blocklisted(): debug_print("Blocklisted: " + str(i)) continue call_str = format_call_to(i) debug_print(call_str) call_str_list.append(call_str) instance_variable = format_instance_variable(i) if instance_variable is not None: debug_print(instance_variable) instance_variable_dict[i.klass] = instance_variable initialize_klass_dict[i.klass] = format_initialize_klass(i) static_fields = indent_list([ value for (key, value) in instance_variable_dict.items() ], 2) test_body = indent_list(call_str_list, 4) initialize_classes = indent_list([ value for (key, value) in initialize_klass_dict.items() ], 4) print(OUTPUT_TPL.substitute(static_fields="\n".join(static_fields), test_body="\n".join(test_body), initialize_classes="\n".join(initialize_classes)). strip("\n"), \ file=args.output_file) if __name__ == '__main__': main()