#!/usr/bin/env python # Copyright 2014 the V8 project authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. # for py2/py3 compatibility from __future__ import print_function import json import optparse import os import random import shutil import subprocess import sys BLACKLIST = [ # Skip special d8 functions. "load", "os", "print", "read", "readline", "quit" ] def GetRandomObject(): return random.choice([ "0", "1", "2.5", "0x1000", "\"string\"", "{foo: \"bar\"}", "[1, 2, 3]", "function() { return 0; }" ]) g_var_index = 0 def GetVars(result, num, first = []): global g_var_index variables = [] for i in range(num): variables.append("__v_%d" % g_var_index) g_var_index += 1 for var in variables: result.append("var %s = %s;" % (var, GetRandomObject())) return ", ".join(first + variables) # Wraps |string| in try..catch. def TryCatch(result, string, exception_behavior = ""): result.append("try { %s } catch(e) { %s }" % (string, exception_behavior)) def BuildTests(function, full_name, options): assert function["type"] == "function" global g_var_index g_var_index = 0 result = ["// AUTO-GENERATED BY tools/generate-builtins-tests.py.\n"] result.append("// Function call test:") length = function["length"] TryCatch(result, "%s(%s);" % (full_name, GetVars(result, length))) if "prototype" in function: proto = function["prototype"] result.append("\n// Constructor test:") TryCatch(result, "var recv = new %s(%s);" % (full_name, GetVars(result, length)), "var recv = new Object();") getters = [] methods = [] for prop in proto: proto_property = proto[prop] proto_property_type = proto_property["type"] if proto_property_type == "getter": getters.append(proto_property) result.append("recv.__defineGetter__(\"%s\", " "function() { return %s; });" % (proto_property["name"], GetVars(result, 1))) if proto_property_type == "number": result.append("recv.__defineGetter__(\"%s\", " "function() { return %s; });" % (proto_property["name"], GetVars(result, 1))) if proto_property_type == "function": methods.append(proto_property) if getters: result.append("\n// Getter tests:") for getter in getters: result.append("print(recv.%s);" % getter["name"]) if methods: result.append("\n// Method tests:") for method in methods: args = GetVars(result, method["length"], ["recv"]) call = "%s.prototype.%s.call(%s)" % (full_name, method["name"], args) TryCatch(result, call) filename = os.path.join(options.outdir, "%s.js" % (full_name)) with open(filename, "w") as f: f.write("\n".join(result)) f.write("\n") def VisitObject(obj, path, options): obj_type = obj["type"] obj_name = "%s%s" % (path, obj["name"]) if obj_type == "function": BuildTests(obj, obj_name, options) if "properties" in obj: for prop_name in obj["properties"]: prop = obj["properties"][prop_name] VisitObject(prop, "%s." % (obj_name), options) def ClearGeneratedFiles(options): if os.path.exists(options.outdir): shutil.rmtree(options.outdir) def GenerateTests(options): ClearGeneratedFiles(options) # Re-generate everything. output = subprocess.check_output( "%s %s" % (options.d8, options.script), shell=True).strip() objects = json.loads(output) os.makedirs(options.outdir) for obj_name in objects: if obj_name in BLACKLIST: continue obj = objects[obj_name] VisitObject(obj, "", options) def BuildOptions(): result = optparse.OptionParser() result.add_option("--d8", help="d8 binary to use", default="out/ia32.release/d8") result.add_option("--outdir", help="directory where to place generated tests", default="test/mjsunit/builtins-gen") result.add_option("--script", help="builtins detector script to run in d8", default="tools/detect-builtins.js") return result def Main(): parser = BuildOptions() (options, args) = parser.parse_args() if len(args) != 1 or args[0] == "help": parser.print_help() return 1 action = args[0] if action == "generate": GenerateTests(options) return 0 if action == "clear": ClearGeneratedFiles(options) return 0 print("Unknown action: %s" % action) parser.print_help() return 1 if __name__ == "__main__": sys.exit(Main())