Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1203)

Unified Diff: bindings/scripts/generate_global_constructors.py

Issue 540533002: Roll IDL to Dartium37 (r181268) (Closed) Base URL: https://dart.googlecode.com/svn/third_party/WebCore
Patch Set: Created 6 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « bindings/scripts/generate_event_interfaces.py ('k') | bindings/scripts/idl_compiler.py » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: bindings/scripts/generate_global_constructors.py
diff --git a/bindings/scripts/generate_global_constructors.py b/bindings/scripts/generate_global_constructors.py
new file mode 100755
index 0000000000000000000000000000000000000000..81838fe1865d5f819eaa9e7bdd41e2bd458584bc
--- /dev/null
+++ b/bindings/scripts/generate_global_constructors.py
@@ -0,0 +1,184 @@
+#!/usr/bin/python
+#
+# Copyright 2014 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+"""Generates interface properties on global objects.
+
+Concretely these are implemented as "constructor attributes", meaning
+"attributes whose name ends with Constructor" (special-cased by code generator),
+hence "global constructors" for short.
+
+For reference on global objects, see:
+http://heycam.github.io/webidl/#Global
+http://heycam.github.io/webidl/#Exposed
+
+Design document: http://www.chromium.org/developers/design-documents/idl-build
+"""
+
+import itertools
+import optparse
+import os
+import cPickle as pickle
+import re
+import sys
+
+from collections import defaultdict
+from utilities import get_file_contents, idl_filename_to_interface_name, read_file_to_list, write_file, get_interface_extended_attributes_from_idl, is_callback_interface_from_idl
+
+interface_name_to_global_names = {}
+global_name_to_constructors = defaultdict(list)
+
+
+HEADER_FORMAT = """// Stub header file for {{idl_basename}}
+// Required because the IDL compiler assumes that a corresponding header file
+// exists for each IDL file.
+"""
+
+def parse_options():
+ parser = optparse.OptionParser()
+ parser.add_option('--idl-files-list', help='file listing IDL files')
+ parser.add_option('--global-objects-file', help='pickle file of global objects')
+ parser.add_option('--write-file-only-if-changed', type='int', help='if true, do not write an output file if it would be identical to the existing one, which avoids unnecessary rebuilds in ninja')
+
+ options, args = parser.parse_args()
+
+ if options.idl_files_list is None:
+ parser.error('Must specify a file listing IDL files using --idl-files-list.')
+ if options.global_objects_file is None:
+ parser.error('Must specify a pickle file of global objects using --global-objects-file.')
+ if options.write_file_only_if_changed is None:
+ parser.error('Must specify whether output files are only written if changed using --write-file-only-if-changed.')
+ options.write_file_only_if_changed = bool(options.write_file_only_if_changed)
+
+ return options, args
+
+
+def flatten_list(iterable):
+ return list(itertools.chain.from_iterable(iterable))
+
+
+def interface_name_to_constructors(interface_name):
+ """Returns constructors for an interface."""
+ global_names = interface_name_to_global_names[interface_name]
+ return flatten_list(global_name_to_constructors[global_name]
+ for global_name in global_names)
+
+
+def record_global_constructors(idl_filename):
+ interface_name = idl_filename_to_interface_name(idl_filename)
+ full_path = os.path.realpath(idl_filename)
+ idl_file_contents = get_file_contents(full_path)
+ extended_attributes = get_interface_extended_attributes_from_idl(idl_file_contents)
+
+ # An interface property is produced for every non-callback interface
+ # that does not have [NoInterfaceObject].
+ # Callback interfaces with constants also have interface properties,
+ # but there are none of these in Blink.
+ # http://heycam.github.io/webidl/#es-interfaces
+ if (is_callback_interface_from_idl(idl_file_contents) or
+ 'NoInterfaceObject' in extended_attributes):
+ return
+
+ # The [Exposed] extended attribute MUST take an identifier list. Each
+ # identifier in the list MUST be a global name. An interface or interface
+ # member the extended attribute applies to will be exposed only on objects
+ # associated with ECMAScript global environments whose global object
+ # implements an interface that has a matching global name.
+ # FIXME: In spec names are comma-separated, but that makes parsing very
+ # difficult (https://www.w3.org/Bugs/Public/show_bug.cgi?id=24959).
+ exposed_global_names = extended_attributes.get('Exposed', 'Window').split('&')
+ new_constructors_list = generate_global_constructors_list(interface_name, extended_attributes)
+ for exposed_global_name in exposed_global_names:
+ global_name_to_constructors[exposed_global_name].extend(new_constructors_list)
+
+
+def generate_global_constructors_list(interface_name, extended_attributes):
+ extended_attributes_list = [
+ name + '=' + extended_attributes[name]
+ for name in 'Conditional', 'PerContextEnabled', 'RuntimeEnabled'
+ if name in extended_attributes]
+ if extended_attributes_list:
+ extended_string = '[%s] ' % ', '.join(extended_attributes_list)
+ else:
+ extended_string = ''
+
+ attribute_string = 'attribute {interface_name}Constructor {interface_name}'.format(interface_name=interface_name)
+ attributes_list = [extended_string + attribute_string]
+
+ # In addition to the usual interface property, for every [NamedConstructor]
+ # extended attribute on an interface, a corresponding property MUST exist
+ # on the ECMAScript global object.
+ # http://heycam.github.io/webidl/#NamedConstructor
+ if 'NamedConstructor' in extended_attributes:
+ named_constructor = extended_attributes['NamedConstructor']
+ # Extract function name, namely everything before opening '('
+ constructor_name = re.sub(r'\(.*', '', named_constructor)
+ # Note the reduplicated 'ConstructorConstructor'
+ # FIXME: rename to NamedConstructor
+ attribute_string = 'attribute %sConstructorConstructor %s' % (interface_name, constructor_name)
+ attributes_list.append(extended_string + attribute_string)
+
+ return attributes_list
+
+
+def write_global_constructors_partial_interface(interface_name, idl_filename, constructor_attributes_list, only_if_changed):
+ # FIXME: replace this with a simple Jinja template
+ lines = (['partial interface %s {\n' % interface_name] +
+ [' %s;\n' % constructor_attribute
+ # FIXME: sort by interface name (not first by extended attributes)
+ for constructor_attribute in sorted(constructor_attributes_list)] +
+ ['};\n'])
+ write_file(''.join(lines), idl_filename, only_if_changed)
+ header_filename = os.path.splitext(idl_filename)[0] + '.h'
+ idl_basename = os.path.basename(idl_filename)
+ write_file(HEADER_FORMAT.format(idl_basename=idl_basename),
+ header_filename, only_if_changed)
+
+
+################################################################################
+
+def main():
+ options, args = parse_options()
+
+ # Input IDL files are passed in a file, due to OS command line length
+ # limits. This is generated at GYP time, which is ok b/c files are static.
+ idl_files = read_file_to_list(options.idl_files_list)
+
+ # Output IDL files (to generate) are passed at the command line, since
+ # these are in the build directory, which is determined at build time, not
+ # GYP time.
+ # These are passed as pairs of GlobalObjectName, GlobalObject.idl
+ interface_name_idl_filename = [(args[i], args[i + 1])
+ for i in range(0, len(args), 2)]
+
+ with open(options.global_objects_file) as global_objects_file:
+ interface_name_to_global_names.update(pickle.load(global_objects_file))
+
+ for idl_filename in idl_files:
+ record_global_constructors(idl_filename)
+
+ # Check for [Exposed] / [Global] mismatch.
+ known_global_names = frozenset(itertools.chain.from_iterable(interface_name_to_global_names.values()))
+ exposed_global_names = frozenset(global_name_to_constructors)
+ if not exposed_global_names.issubset(known_global_names):
+ unknown_global_names = exposed_global_names.difference(known_global_names)
+ raise ValueError('The following global names were used in '
+ '[Exposed=xxx] but do not match any [Global] / '
+ '[PrimaryGlobal] interface: %s'
+ % list(unknown_global_names))
+
+ # Write partial interfaces containing constructor attributes for each
+ # global interface.
+ for interface_name, idl_filename in interface_name_idl_filename:
+ constructors = interface_name_to_constructors(interface_name)
+ write_global_constructors_partial_interface(
+ interface_name,
+ idl_filename,
+ constructors,
+ options.write_file_only_if_changed)
+
+
+if __name__ == '__main__':
+ sys.exit(main())
« no previous file with comments | « bindings/scripts/generate_event_interfaces.py ('k') | bindings/scripts/idl_compiler.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698