Chromium Code Reviews| Index: Source/bindings/scripts/compute_dependencies.py |
| diff --git a/Source/bindings/scripts/compute_dependencies.py b/Source/bindings/scripts/compute_dependencies.py |
| index 816b5b237dab754b87f9915967abad33fb62d5bd..2240c38e0e74b193d68d6193b511c3f5a2b340bc 100755 |
| --- a/Source/bindings/scripts/compute_dependencies.py |
| +++ b/Source/bindings/scripts/compute_dependencies.py |
| @@ -48,8 +48,10 @@ class IdlInterfaceFileNotFoundError(Exception): |
| def parse_options(): |
| parser = optparse.OptionParser() |
| parser.add_option('--event-names-file', help='output file') |
| - parser.add_option('--idl-files-list', help='file listing all IDLs') |
| + parser.add_option('--main-idl-files-list', help='file listing main (compiled to Blink) IDL files') |
| + parser.add_option('--support-idl-files-list', help='file listing support IDL files (not compiled to Blink, e.g. testing)') |
| parser.add_option('--interface-dependencies-file', help='output file') |
| + parser.add_option('--bindings-derived-sources-file', help='output file') |
| parser.add_option('--window-constructors-file', help='output file') |
| parser.add_option('--workerglobalscope-constructors-file', help='output file') |
| parser.add_option('--sharedworkerglobalscope-constructors-file', help='output file') |
| @@ -60,6 +62,8 @@ def parse_options(): |
| parser.error('Must specify an output file using --event-names-file.') |
| if options.interface_dependencies_file is None: |
| parser.error('Must specify an output file using --interface-dependencies-file.') |
| + if options.bindings_derived_sources_file is None: |
| + parser.error('Must specify an output file using --bindings-derived-sources-file.') |
| if options.window_constructors_file is None: |
| parser.error('Must specify an output file using --window-constructors-file.') |
| if options.workerglobalscope_constructors_file is None: |
| @@ -68,8 +72,10 @@ def parse_options(): |
| parser.error('Must specify an output file using --sharedworkerglobalscope-constructors-file.') |
| if options.workerglobalscope_constructors_file is None: |
| parser.error('Must specify an output file using --dedicatedworkerglobalscope-constructors-file.') |
| - if options.idl_files_list is None: |
| - parser.error('Must specify the file listing all IDLs using --idl-files-list.') |
| + if options.main_idl_files_list is None: |
| + parser.error('Must specify a file listing main IDL files using --main-idl-files-list.') |
| + if options.support_idl_files_list is None: |
| + parser.error('Must specify a file listing support IDL files using --support-idl-files-list.') |
| if options.write_file_only_if_changed is None: |
| parser.error('Must specify whether file is only written if changed using --write-file-only-if-changed.') |
| options.write_file_only_if_changed = bool(options.write_file_only_if_changed) |
| @@ -200,12 +206,68 @@ def generate_global_constructors_partial_interface(interface_name, destination_f |
| write_file(lines, destination_filename, only_if_changed) |
| -def parse_idl_files(idl_files, global_constructors_filenames): |
| +def generate_dependencies(idl_file_name, interfaces, dependencies, partial_interface_files, implements_interfaces, implemented_somewhere): |
| + interface_name, _ = os.path.splitext(os.path.basename(idl_file_name)) |
|
eseidel
2013/10/10 06:08:37
I miss our FileSystem abstraction from webkitpy. :
|
| + full_path = os.path.realpath(idl_file_name) |
| + idl_file_contents = get_file_contents(full_path) |
| + |
| + # Handle partial interfaces |
| + partial_interface_name = get_partial_interface_name_from_idl(idl_file_contents) |
|
eseidel
2013/10/10 06:08:37
I take it get_ is chromium python style?
Nils Barth (inactive)
2013/10/10 08:51:02
No particular style AFAICT, and I don't think it's
|
| + if partial_interface_name: |
| + partial_interface_files[partial_interface_name].append(full_path) |
| + return partial_interface_name |
| + |
| + interfaces.add(interface_name) |
| + # Non-partial interfaces default to having bindings generated |
| + dependencies[full_path] = [] |
| + |
| + # Parse 'identifier-A implements identifier-B;' statements |
| + implemented_interfaces = get_implemented_interfaces_from_idl(idl_file_contents, interface_name) |
| + implements_interfaces[interface_name] = implemented_interfaces |
| + implemented_somewhere |= set(implemented_interfaces) |
| + |
| + return partial_interface_name |
| + |
| + |
| +def remove_implementation_redundancy(dependencies, interface_name_to_idl_file, implemented_somewhere): |
|
haraken
2013/10/10 05:57:51
remove_implementation_redundancy => remove_interfa
kihong
2013/10/12 05:43:12
Done.
|
| + # Interfaces that are implemented by another interface do not have |
| + # their own bindings generated, as this would be redundant with the |
| + # actual implementation. |
| + for implemented_interface in implemented_somewhere: |
| + full_path = interface_name_to_idl_file[implemented_interface] |
| + del dependencies[full_path] |
| + |
| + |
| +def record_global_constructors_and_extended_attribute(idl_file_name, global_constructors, interface_extended_attribute, parent_interface): |
| + interface_name, _ = os.path.splitext(os.path.basename(idl_file_name)) |
| + full_path = os.path.realpath(idl_file_name) |
| + idl_file_contents = get_file_contents(full_path) |
| + extended_attributes = get_interface_extended_attributes_from_idl(idl_file_contents) |
| + |
| + # Record global constructors |
| + if not is_callback_interface_from_idl(idl_file_contents) and 'NoInterfaceObject' not in extended_attributes: |
| + global_contexts = extended_attributes.get('GlobalContext', 'Window').split('&') |
| + new_constructor_list = generate_constructor_attribute_list(interface_name, extended_attributes) |
| + for global_object in global_contexts: |
| + global_constructors[global_object].extend(new_constructor_list) |
| + |
| + # Record parents and extended attributes for generating event names |
| + if interface_name == 'Event': |
| + interface_extended_attribute[interface_name] = extended_attributes |
| + parent = get_parent_interface(idl_file_contents) |
| + if parent: |
| + parent_interface[interface_name] = parent |
| + interface_extended_attribute[interface_name] = extended_attributes |
| + |
| + |
| +def parse_idl_files(main_idl_files, support_idl_files, global_constructors_filenames): |
| """Return dependencies between IDL files, constructors on global objects, and events. |
| Returns: |
| interfaces: |
| set of all interfaces |
| + bindings_derived_sources: |
| + main IDL filename except test support IDLs |
|
haraken
2013/10/10 05:57:51
list of main IDL file names (except support IDL fi
kihong
2013/10/12 05:43:12
Done.
|
| dependencies: |
| dict of main IDL filename (for a given interface) -> list of partial IDL filenames (for that interface) |
| The keys (main IDL files) are the files for which bindings are |
| @@ -232,59 +294,29 @@ def parse_idl_files(idl_files, global_constructors_filenames): |
| interface_extended_attribute = {} |
| interface_name_to_idl_file = {} |
| - for idl_file_name in idl_files: |
| + for idl_file_name in main_idl_files + support_idl_files: |
| full_path = os.path.realpath(idl_file_name) |
| interface_name, _ = os.path.splitext(os.path.basename(idl_file_name)) |
| interface_name_to_idl_file[interface_name] = full_path |
| partial_interface_files[interface_name] = [] |
| - for idl_file_name in idl_files: |
| - interface_name, _ = os.path.splitext(os.path.basename(idl_file_name)) |
| - full_path = interface_name_to_idl_file[interface_name] |
| - idl_file_contents = get_file_contents(full_path) |
| - |
| - # Handle partial interfaces |
| - partial_interface_name = get_partial_interface_name_from_idl(idl_file_contents) |
| - if partial_interface_name: |
| - partial_interface_files[partial_interface_name].append(full_path) |
| - continue |
| + # Generate dependencies, global_constructors and interface_extended_attributes for main IDL files |
| + for idl_file_name in main_idl_files: |
| + if not generate_dependencies(idl_file_name, interfaces, dependencies, partial_interface_files, implements_interfaces, implemented_somewhere): |
| + record_global_constructors_and_extended_attribute(idl_file_name, global_constructors, interface_extended_attribute, parent_interface) |
| - interfaces.add(interface_name) |
| - # Non-partial interfaces default to having bindings generated |
| - dependencies[full_path] = [] |
| - extended_attributes = get_interface_extended_attributes_from_idl(idl_file_contents) |
| - |
| - # Parse 'identifier-A implements identifier-B;' statements |
| - implemented_interfaces = get_implemented_interfaces_from_idl(idl_file_contents, interface_name) |
| - implements_interfaces[interface_name] = implemented_interfaces |
| - implemented_somewhere |= set(implemented_interfaces) |
| - |
| - # Record global constructors |
| - if not is_callback_interface_from_idl(idl_file_contents) and 'NoInterfaceObject' not in extended_attributes: |
| - global_contexts = extended_attributes.get('GlobalContext', 'Window').split('&') |
| - new_constructor_list = generate_constructor_attribute_list(interface_name, extended_attributes) |
| - for global_object in global_contexts: |
| - global_constructors[global_object].extend(new_constructor_list) |
| - |
| - # Record parents and extended attributes for generating event names |
| - if interface_name == 'Event': |
| - interface_extended_attribute[interface_name] = extended_attributes |
| - parent = get_parent_interface(idl_file_contents) |
| - if parent: |
| - parent_interface[interface_name] = parent |
| - interface_extended_attribute[interface_name] = extended_attributes |
| + bindings_derived_sources = dependencies.copy() |
| + remove_implementation_redundancy(bindings_derived_sources, interface_name_to_idl_file, implemented_somewhere) |
| # Add constructors on global objects to partial interfaces |
| for global_object, filename in global_constructors_filenames.iteritems(): |
| if global_object in interfaces: |
| partial_interface_files[global_object].append(filename) |
| - # Interfaces that are implemented by another interface do not have |
| - # their own bindings generated, as this would be redundant with the |
| - # actual implementation. |
| - for implemented_interface in implemented_somewhere: |
| - full_path = interface_name_to_idl_file[implemented_interface] |
| - del dependencies[full_path] |
| + # Add support IDL files to the dependencies for supporting partial interface |
| + for idl_file_name in support_idl_files: |
| + generate_dependencies(idl_file_name, interfaces, dependencies, partial_interface_files, implements_interfaces, implemented_somewhere) |
| + remove_implementation_redundancy(dependencies, interface_name_to_idl_file, implemented_somewhere) |
| # An IDL file's dependencies are partial interface files that extend it, |
| # and files for other interfaces that this interfaces implements. |
| @@ -308,7 +340,7 @@ def parse_idl_files(idl_files, global_constructors_filenames): |
| if parent == 'Event': |
| event_names[interface_name_to_idl_file[interface]] = interface_extended_attribute[interface] |
| - return interfaces, dependencies, global_constructors, event_names |
| + return interfaces, dependencies, bindings_derived_sources, global_constructors, event_names |
| def write_dependency_file(filename, dependencies, only_if_changed): |
| @@ -336,10 +368,14 @@ def write_dependency_file(filename, dependencies, only_if_changed): |
| def main(): |
| options = parse_options() |
| - idl_files = [] |
| - with open(options.idl_files_list) as idl_files_list_file: |
| + main_idl_files = [] |
| + with open(options.main_idl_files_list) as idl_files_list_file: |
| + for line in idl_files_list_file: |
| + main_idl_files.append(string.rstrip(line, '\n')) |
|
eseidel
2013/10/10 06:08:37
This could be a list comprehension.
with open() a
Nils Barth (inactive)
2013/10/10 08:51:02
Good point, also for support_idl_files.
kihong
2013/10/12 05:43:12
Done.
|
| + support_idl_files = [] |
| + with open(options.support_idl_files_list) as idl_files_list_file: |
| for line in idl_files_list_file: |
| - idl_files.append(string.rstrip(line, '\n')) |
| + support_idl_files.append(string.rstrip(line, '\n')) |
| only_if_changed = options.write_file_only_if_changed |
| global_constructors_filenames = { |
| 'Window': options.window_constructors_file, |
| @@ -348,9 +384,10 @@ def main(): |
| 'DedicatedWorkerGlobalScope': options.dedicatedworkerglobalscope_constructors_file, |
| } |
| - interfaces, dependencies, global_constructors, event_names = parse_idl_files(idl_files, global_constructors_filenames) |
| + interfaces, dependencies, bindings_derived_sources, global_constructors, event_names = parse_idl_files(main_idl_files, support_idl_files, global_constructors_filenames) |
| write_dependency_file(options.interface_dependencies_file, dependencies, only_if_changed) |
| + write_dependency_file(options.bindings_derived_sources_file, bindings_derived_sources, only_if_changed) |
| for interface_name, filename in global_constructors_filenames.iteritems(): |
| if interface_name in interfaces: |
| generate_global_constructors_partial_interface(interface_name, filename, global_constructors[interface_name], only_if_changed) |