| #!/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. |
| |
| 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()) |