| Index: mojo/public/tools/bindings/generators/mojom_go_generator.py
|
| diff --git a/mojo/public/tools/bindings/generators/mojom_go_generator.py b/mojo/public/tools/bindings/generators/mojom_go_generator.py
|
| deleted file mode 100644
|
| index 88d83ac303fd8c66d5bc80fc74d4cfb48ea1e482..0000000000000000000000000000000000000000
|
| --- a/mojo/public/tools/bindings/generators/mojom_go_generator.py
|
| +++ /dev/null
|
| @@ -1,455 +0,0 @@
|
| -# Copyright 2015 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 Go source files from a mojom.Module.'''
|
| -
|
| -from itertools import chain
|
| -import os
|
| -import re
|
| -
|
| -from mojom.generate.template_expander import UseJinja
|
| -
|
| -import mojom.generate.generator as generator
|
| -import mojom.generate.module as mojom
|
| -import mojom.generate.pack as pack
|
| -
|
| -class KindInfo(object):
|
| - def __init__(self, go_type, encode_suffix, decode_suffix, bit_size):
|
| - self.go_type = go_type
|
| - self.encode_suffix = encode_suffix
|
| - self.decode_suffix = decode_suffix
|
| - self.bit_size = bit_size
|
| -
|
| -_kind_infos = {
|
| - mojom.BOOL: KindInfo('bool', 'Bool', 'Bool', 1),
|
| - mojom.INT8: KindInfo('int8', 'Int8', 'Int8', 8),
|
| - mojom.UINT8: KindInfo('uint8', 'Uint8', 'Uint8', 8),
|
| - mojom.INT16: KindInfo('int16', 'Int16', 'Int16', 16),
|
| - mojom.UINT16: KindInfo('uint16', 'Uint16', 'Uint16', 16),
|
| - mojom.INT32: KindInfo('int32', 'Int32', 'Int32', 32),
|
| - mojom.UINT32: KindInfo('uint32', 'Uint32', 'Uint32', 32),
|
| - mojom.FLOAT: KindInfo('float32', 'Float32', 'Float32', 32),
|
| - mojom.HANDLE: KindInfo(
|
| - 'system.Handle', 'Handle', 'Handle', 32),
|
| - mojom.DCPIPE: KindInfo(
|
| - 'system.ConsumerHandle', 'Handle', 'ConsumerHandle', 32),
|
| - mojom.DPPIPE: KindInfo(
|
| - 'system.ProducerHandle', 'Handle', 'ProducerHandle', 32),
|
| - mojom.MSGPIPE: KindInfo(
|
| - 'system.MessagePipeHandle', 'Handle', 'MessagePipeHandle', 32),
|
| - mojom.SHAREDBUFFER: KindInfo(
|
| - 'system.SharedBufferHandle', 'Handle', 'SharedBufferHandle', 32),
|
| - mojom.NULLABLE_HANDLE: KindInfo(
|
| - 'system.Handle', 'Handle', 'Handle', 32),
|
| - mojom.NULLABLE_DCPIPE: KindInfo(
|
| - 'system.ConsumerHandle', 'Handle', 'ConsumerHandle', 32),
|
| - mojom.NULLABLE_DPPIPE: KindInfo(
|
| - 'system.ProducerHandle', 'Handle', 'ProducerHandle', 32),
|
| - mojom.NULLABLE_MSGPIPE: KindInfo(
|
| - 'system.MessagePipeHandle', 'Handle', 'MessagePipeHandle', 32),
|
| - mojom.NULLABLE_SHAREDBUFFER: KindInfo(
|
| - 'system.SharedBufferHandle', 'Handle', 'SharedBufferHandle', 32),
|
| - mojom.INT64: KindInfo('int64', 'Int64', 'Int64', 64),
|
| - mojom.UINT64: KindInfo('uint64', 'Uint64', 'Uint64', 64),
|
| - mojom.DOUBLE: KindInfo('float64', 'Float64', 'Float64', 64),
|
| - mojom.STRING: KindInfo('string', 'String', 'String', 64),
|
| - mojom.NULLABLE_STRING: KindInfo('string', 'String', 'String', 64),
|
| -}
|
| -
|
| -
|
| -# The mojom_types.mojom and service_describer.mojom files are special because
|
| -# they are used to generate mojom Type's and ServiceDescription implementations.
|
| -_service_describer_pkg_short = "service_describer"
|
| -_service_describer_pkg = "mojo/public/interfaces/bindings/%s" % \
|
| - _service_describer_pkg_short
|
| -_mojom_types_pkg_short = "mojom_types"
|
| -_mojom_types_pkg = "mojo/public/interfaces/bindings/%s" % _mojom_types_pkg_short
|
| -
|
| -def GetBitSize(kind):
|
| - if isinstance(kind, (mojom.Union)):
|
| - return 128
|
| - if isinstance(kind, (mojom.Array, mojom.Map, mojom.Struct, mojom.Interface)):
|
| - return 64
|
| - if mojom.IsUnionKind(kind):
|
| - return 2*64
|
| - if isinstance(kind, (mojom.InterfaceRequest)):
|
| - kind = mojom.MSGPIPE
|
| - if isinstance(kind, mojom.Enum):
|
| - kind = mojom.INT32
|
| - return _kind_infos[kind].bit_size
|
| -
|
| -# Returns go type corresponding to provided kind. If |nullable| is true
|
| -# and kind is nullable adds an '*' to type (example: ?string -> *string).
|
| -def GetGoType(kind, nullable = True):
|
| - if nullable and mojom.IsNullableKind(kind) and not mojom.IsUnionKind(kind):
|
| - return '*%s' % GetNonNullableGoType(kind)
|
| - return GetNonNullableGoType(kind)
|
| -
|
| -# Returns go type corresponding to provided kind. Ignores nullability of
|
| -# top-level kind.
|
| -def GetNonNullableGoType(kind):
|
| - if mojom.IsStructKind(kind) or mojom.IsUnionKind(kind):
|
| - return '%s' % GetFullName(kind)
|
| - if mojom.IsArrayKind(kind):
|
| - if kind.length:
|
| - return '[%s]%s' % (kind.length, GetGoType(kind.kind))
|
| - return '[]%s' % GetGoType(kind.kind)
|
| - if mojom.IsMapKind(kind):
|
| - return 'map[%s]%s' % (GetGoType(kind.key_kind), GetGoType(kind.value_kind))
|
| - if mojom.IsInterfaceKind(kind):
|
| - return '%s_Pointer' % GetFullName(kind)
|
| - if mojom.IsInterfaceRequestKind(kind):
|
| - return '%s_Request' % GetFullName(kind.kind)
|
| - if mojom.IsEnumKind(kind):
|
| - return GetNameForNestedElement(kind)
|
| - return _kind_infos[kind].go_type
|
| -
|
| -def IsPointer(kind):
|
| - return mojom.IsObjectKind(kind) and not mojom.IsUnionKind(kind)
|
| -
|
| -# Splits name to lower-cased parts used for camel-casing
|
| -# (example: HTTPEntry2FooBar -> ['http', 'entry2', 'foo', 'bar']).
|
| -def NameToComponent(name):
|
| - # insert '_' between anything and a Title name (e.g, HTTPEntry2FooBar ->
|
| - # HTTP_Entry2_FooBar)
|
| - name = re.sub('([^_])([A-Z][^A-Z0-9_]+)', r'\1_\2', name)
|
| - # insert '_' between non upper and start of upper blocks (e.g.,
|
| - # HTTP_Entry2_FooBar -> HTTP_Entry2_Foo_Bar)
|
| - name = re.sub('([^A-Z_])([A-Z])', r'\1_\2', name)
|
| - return [x.lower() for x in name.split('_')]
|
| -
|
| -def UpperCamelCase(name):
|
| - return ''.join([x.capitalize() for x in NameToComponent(name)])
|
| -
|
| -# Formats a name. If |exported| is true makes name camel-cased with first
|
| -# letter capital, otherwise does no camel-casing and makes first letter
|
| -# lower-cased (which is used for making internal names more readable).
|
| -def FormatName(name, exported=True):
|
| - if exported:
|
| - return UpperCamelCase(name)
|
| - # Leave '_' symbols for unexported names.
|
| - return name[0].lower() + name[1:]
|
| -
|
| -# Returns full name of an imported element.
|
| -# If the |element| is not imported returns formatted name of it.
|
| -# |element| should have attr 'name'. |exported| argument is used to make
|
| -# |FormatName()| calls only.
|
| -def GetFullName(element, exported=True):
|
| - return GetQualifiedName(
|
| - element.name, GetPackageNameForElement(element), exported)
|
| -
|
| -def GetUnqualifiedNameForElement(element, exported=True):
|
| - return FormatName(element.name, exported)
|
| -
|
| -# Returns a name for nested elements like enum field or constant.
|
| -# The returned name consists of camel-cased parts separated by '_'.
|
| -def GetNameForNestedElement(element):
|
| - if element.parent_kind:
|
| - return "%s_%s" % (GetNameForElement(element.parent_kind),
|
| - FormatName(element.name))
|
| - return GetFullName(element)
|
| -
|
| -def GetNameForElement(element, exported=True):
|
| - if (mojom.IsInterfaceKind(element) or mojom.IsStructKind(element)
|
| - or mojom.IsUnionKind(element)):
|
| - return GetFullName(element, exported)
|
| - if isinstance(element, (mojom.EnumField,
|
| - mojom.Field,
|
| - mojom.Method,
|
| - mojom.Parameter)):
|
| - element_name = (element.go_name if hasattr(element, "go_name")
|
| - else element.name)
|
| - return FormatName(element_name, exported)
|
| - if isinstance(element, (mojom.Enum,
|
| - mojom.Constant,
|
| - mojom.ConstantValue)):
|
| - return GetNameForNestedElement(element)
|
| - raise Exception('Unexpected element: %s' % element)
|
| -
|
| -def ExpressionToText(token):
|
| - if isinstance(token, mojom.EnumValue):
|
| - return "%s_%s" % (GetNameForNestedElement(token.enum),
|
| - FormatName(token.name, True))
|
| - if isinstance(token, mojom.ConstantValue):
|
| - return GetNameForNestedElement(token)
|
| - if isinstance(token, mojom.Constant):
|
| - return ExpressionToText(token.value)
|
| - return token
|
| -
|
| -def DecodeSuffix(kind):
|
| - if mojom.IsEnumKind(kind):
|
| - return DecodeSuffix(mojom.INT32)
|
| - if mojom.IsInterfaceKind(kind):
|
| - return 'Interface'
|
| - if mojom.IsInterfaceRequestKind(kind):
|
| - return DecodeSuffix(mojom.MSGPIPE)
|
| - return _kind_infos[kind].decode_suffix
|
| -
|
| -def EncodeSuffix(kind):
|
| - if mojom.IsEnumKind(kind):
|
| - return EncodeSuffix(mojom.INT32)
|
| - if mojom.IsInterfaceKind(kind):
|
| - return 'Interface'
|
| - if mojom.IsInterfaceRequestKind(kind):
|
| - return EncodeSuffix(mojom.MSGPIPE)
|
| - return _kind_infos[kind].encode_suffix
|
| -
|
| -def GetPackageName(module):
|
| - return module.name.split('.')[0]
|
| -
|
| -def GetPackageNameForElement(element):
|
| - if not hasattr(element, 'imported_from') or not element.imported_from:
|
| - return ''
|
| - return element.imported_from.get('go_name', '')
|
| -
|
| -def GetTypeKeyForElement(element):
|
| - if not hasattr(element, 'type_key') or not element.type_key:
|
| - return ''
|
| - return element.type_key
|
| -
|
| -def GetQualifiedName(name, package=None, exported=True):
|
| - if not package:
|
| - return FormatName(name, exported)
|
| - return '%s.%s' % (package, FormatName(name, exported))
|
| -
|
| -def GetPackagePath(module):
|
| - name = module.name.split('.')[0]
|
| - return '/'.join(module.path.split('/')[:-1] + [name])
|
| -
|
| -def GetAllConstants(module):
|
| - data = [module] + module.structs + module.interfaces
|
| - constants = [x.constants for x in data]
|
| - return [i for i in chain.from_iterable(constants)]
|
| -
|
| -def GetAllEnums(module):
|
| - data = [module] + module.structs + module.interfaces
|
| - enums = [x.enums for x in data]
|
| - return [i for i in chain.from_iterable(enums)]
|
| -
|
| -def AddImport(imports, mojom_imports, module, element):
|
| - """Adds an import required to use the provided element.
|
| -
|
| - The required import is stored in the imports parameter.
|
| - The corresponding mojom import is stored in the mojom_imports parameter.
|
| - Each import is also updated to include a 'go_name' entry. The 'go_name' entry
|
| - is the name by which the imported module will be referred to in the generated
|
| - code. Because the import dictionary is accessible from the element's
|
| - imported_from field this allows us to generate the qualified name for the
|
| - element.
|
| -
|
| - Args:
|
| - imports: {dict<str, str>} The key is the path to the import and the value
|
| - is the go name.
|
| - mojom_imports: {dict<str, str>} The key is the path to the import and the
|
| - value is the go name.
|
| - module: {module.Module} the module being processed.
|
| - element: {module.Kind} the element whose import is to be tracked.
|
| - """
|
| - if not isinstance(element, mojom.Kind):
|
| - return
|
| -
|
| - if mojom.IsArrayKind(element) or mojom.IsInterfaceRequestKind(element):
|
| - AddImport(imports, mojom_imports, module, element.kind)
|
| - return
|
| - if mojom.IsMapKind(element):
|
| - AddImport(imports, mojom_imports, module, element.key_kind)
|
| - AddImport(imports, mojom_imports, module, element.value_kind)
|
| - return
|
| - if mojom.IsAnyHandleKind(element):
|
| - imports['mojo/public/go/system'] = 'system'
|
| - return
|
| -
|
| - if not hasattr(element, 'imported_from') or not element.imported_from:
|
| - return
|
| - imported = element.imported_from
|
| - if GetPackagePath(imported['module']) == GetPackagePath(module):
|
| - return
|
| - path = GetPackagePath(imported['module'])
|
| - if path in imports:
|
| - return
|
| - name = GetPackageName(imported['module'])
|
| - while name in imports.values(): # This avoids repeated names.
|
| - name += '_'
|
| - imported['go_name'] = name
|
| - imports[path] = name
|
| - mojom_imports[path] = name
|
| -
|
| -class Generator(generator.Generator):
|
| - go_filters = {
|
| - 'array': lambda kind: mojom.Array(kind),
|
| - 'bit_size': GetBitSize,
|
| - 'decode_suffix': DecodeSuffix,
|
| - 'encode_suffix': EncodeSuffix,
|
| - 'go_type': GetGoType,
|
| - 'expression_to_text': ExpressionToText,
|
| - 'has_response': lambda method: method.response_parameters is not None,
|
| - 'is_array': mojom.IsArrayKind,
|
| - 'is_enum': mojom.IsEnumKind,
|
| - 'is_handle': mojom.IsAnyHandleKind,
|
| - 'is_interface': mojom.IsInterfaceKind,
|
| - 'is_interface_request': mojom.IsInterfaceRequestKind,
|
| - 'is_map': mojom.IsMapKind,
|
| - 'is_none_or_empty': lambda array: array is None or len(array) == 0,
|
| - 'is_nullable': mojom.IsNullableKind,
|
| - 'is_pointer': IsPointer,
|
| - 'is_object': mojom.IsObjectKind,
|
| - 'is_struct': mojom.IsStructKind,
|
| - 'is_union': mojom.IsUnionKind,
|
| - 'qualified': GetQualifiedName,
|
| - 'mojom_type_key' : GetTypeKeyForElement,
|
| - 'name': GetNameForElement,
|
| - 'unqualified_name': GetUnqualifiedNameForElement,
|
| - 'package': GetPackageNameForElement,
|
| - 'tab_indent': lambda s, size = 1: ('\n' + '\t' * size).join(s.splitlines())
|
| - }
|
| -
|
| - # If set to True, then mojom type information will be generated.
|
| - should_gen_mojom_types = False
|
| -
|
| - def GetParameters(self):
|
| - package = GetPackageName(self.module)
|
| - imports, mojom_imports = self.GetImports()
|
| - return {
|
| - 'enums': GetAllEnums(self.module),
|
| - 'imports': imports,
|
| - 'interfaces': self.GetInterfaces(),
|
| - 'mojom_imports': mojom_imports,
|
| - 'package': package,
|
| - 'structs': self.GetStructs(),
|
| - 'descpkg': '%s.' % _service_describer_pkg_short \
|
| - if package != _service_describer_pkg_short else '',
|
| - 'typepkg': '%s.' % _mojom_types_pkg_short \
|
| - if package != _mojom_types_pkg_short else '',
|
| - 'unions': self.GetUnions(),
|
| - }
|
| -
|
| - @UseJinja('go_templates/source.tmpl', filters=go_filters)
|
| - def GenerateSource(self):
|
| - return self.GetParameters()
|
| -
|
| - def GenerateFiles(self, args):
|
| - self.should_gen_mojom_types = "--generate_type_info" in args
|
| -
|
| - self.Write(self.GenerateSource(), os.path.join("go", "src",
|
| - GetPackagePath(self.module), "%s.go" % self.module.name))
|
| -
|
| - def GetJinjaParameters(self):
|
| - return {
|
| - 'lstrip_blocks': True,
|
| - 'trim_blocks': True,
|
| - }
|
| -
|
| - def GetGlobals(self):
|
| - return {
|
| - 'namespace': self.module.namespace,
|
| - 'module': self.module,
|
| - 'should_gen_mojom_types': self.should_gen_mojom_types,
|
| - }
|
| -
|
| - def GetImports(self):
|
| - """Gets the current module's imports.
|
| -
|
| - Returns:
|
| - tuple(dict<str, str>, dict<str, str>)
|
| - The first element of the tuple is a dictionary mapping import paths to
|
| - import names.
|
| - The second element is a dictionary mapping import paths to import names
|
| - only for imported mojom files.
|
| - """
|
| - imports = {}
|
| - mojom_imports = {}
|
| - # Imports are referred to by the imported_from field of imported kinds.
|
| - # Imported kinds can only be referred to in structs, constants, enums,
|
| - # unions and interfaces.
|
| - all_structs = list(self.module.structs)
|
| - for i in self.module.interfaces:
|
| - for method in i.methods:
|
| - all_structs.append(self._GetStructFromMethod(method))
|
| - if method.response_parameters:
|
| - all_structs.append(self._GetResponseStructFromMethod(method))
|
| -
|
| - if (len(all_structs) > 0 or len(self.module.interfaces) > 0
|
| - or len(self.module.unions) > 0):
|
| - imports['fmt'] = 'fmt'
|
| - imports['mojo/public/go/bindings'] = 'bindings'
|
| - if len(self.module.interfaces) > 0:
|
| - imports['mojo/public/go/system'] = 'system'
|
| - if len(all_structs) > 0:
|
| - imports['sort'] = 'sort'
|
| -
|
| - for union in self.module.unions:
|
| - for field in union.fields:
|
| - AddImport(imports, mojom_imports, self.module, field.kind)
|
| -
|
| - for struct in all_structs:
|
| - for field in struct.fields:
|
| - AddImport(imports, mojom_imports, self.module, field.kind)
|
| -# TODO(rogulenko): add these after generating constants and struct defaults.
|
| -# if field.default:
|
| -# AddImport(imports, mojom_imports, self.module, field.default)
|
| -
|
| - for enum in GetAllEnums(self.module):
|
| - for field in enum.fields:
|
| - if field.value:
|
| - AddImport(imports, mojom_imports, self.module, field.value)
|
| -
|
| - # Mojom Type generation requires additional imports.
|
| - defInterface = len(self.module.interfaces) > 0
|
| - defOtherType = len(self.module.unions) + len(all_structs) + \
|
| - len(GetAllEnums(self.module)) > 0
|
| -
|
| - if self.should_gen_mojom_types:
|
| - imports['bytes'] = 'bytes'
|
| - imports['compress/gzip'] = 'gzip'
|
| - imports['encoding/base64'] = 'base64'
|
| - imports['fmt'] = 'fmt'
|
| - imports['io/ioutil'] = 'ioutil'
|
| - imports['mojo/public/go/bindings'] = 'bindings'
|
| -
|
| - if GetPackageName(self.module) != _mojom_types_pkg_short:
|
| - if defInterface:
|
| - # Each Interface has a service description that uses this.
|
| - imports[_mojom_types_pkg] = _mojom_types_pkg_short
|
| - if defOtherType and self.should_gen_mojom_types:
|
| - # This import is needed only if generating mojom type definitions.
|
| - imports[_mojom_types_pkg] = _mojom_types_pkg_short
|
| -
|
| - if GetPackageName(self.module) != _service_describer_pkg_short and \
|
| - defInterface:
|
| - # Each Interface has a service description that uses this.
|
| - imports[_service_describer_pkg] = _service_describer_pkg_short
|
| -
|
| -# TODO(rogulenko): add these after generating constants and struct defaults.
|
| -# for constant in GetAllConstants(self.module):
|
| -# AddImport(imports, mojom_imports, self.module, constant.value)
|
| -
|
| - return imports, mojom_imports
|
| -
|
| - # Overrides the implementation from the base class in order to customize the
|
| - # struct and field names. Since the Python objects representing the struct
|
| - # and fields are shared by all language generators we don't want to actually
|
| - # modify the |name| property. Instead we add a |go_name| property.
|
| - def _GetStructFromMethod(self, method):
|
| - self._AddStructComputedData(False, method.param_struct)
|
| - # Only generate the go_names if they have not already been generated.
|
| - if not hasattr(method.param_struct, "go_name"):
|
| - method.param_struct.go_name = "%s_%s_Params" % (
|
| - GetNameForElement(method.interface), GetNameForElement(method))
|
| - for field in method.param_struct.fields:
|
| - field.go_name = "in%s" % GetNameForElement(field)
|
| - return method.param_struct
|
| -
|
| - # Overrides the implementation from the base class in order to customize the
|
| - # struct and field names. Since the Python objects representing the struct
|
| - # and fields are shared by all language generators we don't want to actually
|
| - # modify the |name| property. Instead we add a |go_name| property.
|
| - def _GetResponseStructFromMethod(self, method):
|
| - self._AddStructComputedData(False, method.response_param_struct)
|
| - if not hasattr(method.response_param_struct, "go_name"):
|
| - # Only generate the go_names if they have not already been generated.
|
| - method.response_param_struct.go_name = "%s_%s_ResponseParams" % (
|
| - GetNameForElement(method.interface), GetNameForElement(method))
|
| - for field in method.response_param_struct.fields:
|
| - field.go_name = "out%s" % GetNameForElement(field)
|
| - return method.response_param_struct
|
|
|