Index: bindings/scripts/v8_union.py |
diff --git a/bindings/scripts/v8_union.py b/bindings/scripts/v8_union.py |
new file mode 100644 |
index 0000000000000000000000000000000000000000..1b94261bc8e3a3d79a3a23879c14b89928f756d8 |
--- /dev/null |
+++ b/bindings/scripts/v8_union.py |
@@ -0,0 +1,153 @@ |
+# 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. |
+ |
+import v8_utilities |
+ |
+ |
+UNION_H_INCLUDES = frozenset([ |
+ 'bindings/core/v8/Dictionary.h', |
+ 'bindings/core/v8/ExceptionState.h', |
+ 'bindings/core/v8/V8Binding.h', |
+ 'platform/heap/Handle.h', |
+]) |
+ |
+UNION_CPP_INCLUDES_BLACKLIST = frozenset([ |
+ # This header defines static functions needed to implement event handler |
+ # attributes in interfaces that implement GlobalEventHandlers. They are not |
+ # needed or used by UnionTypes*.cpp, so including the header causes |
+ # compilation errors. |
+ # FIXME: We should solve this problem in a way that doesn't involve special- |
+ # casing a header like this. |
+ 'core/dom/GlobalEventHandlers.h', |
+]) |
+ |
+ |
+cpp_includes = set() |
+header_forward_decls = set() |
+ |
+ |
+def union_context(union_types, interfaces_info): |
+ cpp_includes.clear() |
+ header_forward_decls.clear() |
+ |
+ # For container classes we strip nullable wrappers. For example, |
+ # both (A or B)? and (A? or B) will become AOrB. This should be OK |
+ # because container classes can handle null and it seems that |
+ # distinguishing (A or B)? and (A? or B) doesn't make sense. |
+ container_cpp_types = set() |
+ union_types_for_containers = set() |
+ nullable_cpp_types = set() |
+ for union_type in union_types: |
+ cpp_type = union_type.cpp_type |
+ if cpp_type not in container_cpp_types: |
+ union_types_for_containers.add(union_type) |
+ container_cpp_types.add(cpp_type) |
+ if union_type.includes_nullable_type: |
+ nullable_cpp_types.add(cpp_type) |
+ |
+ union_types_for_containers = sorted(union_types_for_containers, |
+ key=lambda union_type: union_type.cpp_type) |
+ nullable_cpp_types = sorted(nullable_cpp_types) |
+ |
+ return { |
+ 'containers': [container_context(union_type, interfaces_info) |
+ for union_type in union_types_for_containers], |
+ 'cpp_includes': sorted(cpp_includes - UNION_CPP_INCLUDES_BLACKLIST), |
+ 'header_forward_decls': sorted(header_forward_decls), |
+ 'header_includes': sorted(UNION_H_INCLUDES), |
+ 'nullable_cpp_types': nullable_cpp_types, |
+ } |
+ |
+ |
+def container_context(union_type, interfaces_info): |
+ members = [] |
+ |
+ # These variables refer to member contexts if the given union type has |
+ # corresponding types. They are used for V8 -> impl conversion. |
+ array_buffer_type = None |
+ array_buffer_view_type = None |
+ array_or_sequence_type = None |
+ boolean_type = None |
+ dictionary_type = None |
+ interface_types = [] |
+ numeric_type = None |
+ string_type = None |
+ for member in union_type.member_types: |
+ context = member_context(member, interfaces_info) |
+ members.append(context) |
+ if member.base_type == 'ArrayBuffer': |
+ if array_buffer_type: |
+ raise Exception('%s is ambiguous.' % union_type.name) |
+ array_buffer_type = context |
+ elif member.base_type == 'ArrayBufferView': |
+ if array_buffer_view_type: |
+ raise Exception('%s is ambiguous.' % union_type.name) |
+ array_buffer_view_type = context |
+ # FIXME: Remove generic Dictionary special casing. |
+ elif member.is_dictionary or member.base_type == 'Dictionary': |
+ if dictionary_type: |
+ raise Exception('%s is ambiguous.' % union_type.name) |
+ dictionary_type = context |
+ elif member.is_array_or_sequence_type: |
+ if array_or_sequence_type: |
+ raise Exception('%s is ambiguous.' % union_type.name) |
+ array_or_sequence_type = context |
+ elif member.is_interface_type: |
+ interface_types.append(context) |
+ elif member is union_type.boolean_member_type: |
+ boolean_type = context |
+ elif member is union_type.numeric_member_type: |
+ numeric_type = context |
+ elif member is union_type.string_member_type: |
+ string_type = context |
+ else: |
+ raise Exception('%s is not supported as an union member.' % member.name) |
+ |
+ # Nullable restriction checks |
+ nullable_members = union_type.number_of_nullable_member_types |
+ if nullable_members > 1: |
+ raise Exception('%s contains more than one nullable members' % union_type.name) |
+ if dictionary_type and nullable_members == 1: |
+ raise Exception('%s has a dictionary and a nullable member' % union_type.name) |
+ |
+ return { |
+ 'array_buffer_type': array_buffer_type, |
+ 'array_buffer_view_type': array_buffer_view_type, |
+ 'array_or_sequence_type': array_or_sequence_type, |
+ 'boolean_type': boolean_type, |
+ 'cpp_class': union_type.cpp_type, |
+ 'dictionary_type': dictionary_type, |
+ 'includes_nullable_type': union_type.includes_nullable_type, |
+ 'interface_types': interface_types, |
+ 'members': members, |
+ 'numeric_type': numeric_type, |
+ 'string_type': string_type, |
+ 'type_string': str(union_type), |
+ } |
+ |
+ |
+def member_context(member, interfaces_info): |
+ cpp_includes.update(member.includes_for_type()) |
+ interface_info = interfaces_info.get(member.name, None) |
+ if interface_info: |
+ cpp_includes.update(interface_info.get('dependencies_include_paths', [])) |
+ header_forward_decls.add(member.implemented_as) |
+ if member.is_nullable: |
+ member = member.inner_type |
+ return { |
+ 'cpp_name': v8_utilities.uncapitalize(member.name), |
+ 'cpp_type': member.cpp_type_args(used_in_cpp_sequence=True), |
+ 'cpp_local_type': member.cpp_type, |
+ 'cpp_value_to_v8_value': member.cpp_value_to_v8_value( |
+ cpp_value='impl.getAs%s()' % member.name, isolate='isolate', |
+ creation_context='creationContext'), |
+ 'enum_values': member.enum_values, |
+ 'is_traceable': member.is_traceable, |
+ 'rvalue_cpp_type': member.cpp_type_args(used_as_rvalue_type=True), |
+ 'specific_type_enum': 'SpecificType' + member.name, |
+ 'type_name': member.name, |
+ 'v8_value_to_local_cpp_value': member.v8_value_to_local_cpp_value( |
+ {}, 'v8Value', 'cppValue', isolate='isolate', |
+ use_exception_state=True, restricted_float=True), |
+ } |