1# Copyright 2013 The Chromium Authors. All rights reserved. 2# Use of this source code is governed by a BSD-style license that can be 3# found in the LICENSE file. 4 5"""Code shared by the various language-specific code generators.""" 6 7from functools import partial 8import os.path 9import re 10 11import module as mojom 12import mojom.fileutil as fileutil 13import pack 14 15def ExpectedArraySize(kind): 16 if mojom.IsArrayKind(kind): 17 return kind.length 18 return None 19 20def StudlyCapsToCamel(studly): 21 return studly[0].lower() + studly[1:] 22 23def UnderToCamel(under): 24 """Converts underscore_separated strings to CamelCase strings.""" 25 return ''.join(word.capitalize() for word in under.split('_')) 26 27def WriteFile(contents, full_path): 28 # Make sure the containing directory exists. 29 full_dir = os.path.dirname(full_path) 30 fileutil.EnsureDirectoryExists(full_dir) 31 32 # Dump the data to disk. 33 with open(full_path, "w+") as f: 34 f.write(contents) 35 36class Generator(object): 37 # Pass |output_dir| to emit files to disk. Omit |output_dir| to echo all 38 # files to stdout. 39 def __init__(self, module, output_dir=None, typemap=None, variant=None, 40 bytecode_path=None, for_blink=False, 41 use_new_wrapper_types=False): 42 self.module = module 43 self.output_dir = output_dir 44 self.typemap = typemap or {} 45 self.variant = variant 46 self.bytecode_path = bytecode_path 47 self.for_blink = for_blink 48 self.use_new_wrapper_types = use_new_wrapper_types 49 50 def GetStructsFromMethods(self): 51 result = [] 52 for interface in self.module.interfaces: 53 for method in interface.methods: 54 result.append(self._GetStructFromMethod(method)) 55 if method.response_parameters != None: 56 result.append(self._GetResponseStructFromMethod(method)) 57 return result 58 59 def GetStructs(self): 60 return map(partial(self._AddStructComputedData, True), self.module.structs) 61 62 def GetUnions(self): 63 return map(self._AddUnionComputedData, self.module.unions) 64 65 def GetInterfaces(self): 66 return map(self._AddInterfaceComputedData, self.module.interfaces) 67 68 # Prepend the filename with a directory that matches the directory of the 69 # original .mojom file, relative to the import root. 70 def MatchMojomFilePath(self, filename): 71 return os.path.join(os.path.dirname(self.module.path), filename) 72 73 def Write(self, contents, filename): 74 if self.output_dir is None: 75 print contents 76 return 77 full_path = os.path.join(self.output_dir, filename) 78 WriteFile(contents, full_path) 79 80 def GenerateFiles(self, args): 81 raise NotImplementedError("Subclasses must override/implement this method") 82 83 def GetJinjaParameters(self): 84 """Returns default constructor parameters for the jinja environment.""" 85 return {} 86 87 def GetGlobals(self): 88 """Returns global mappings for the template generation.""" 89 return {} 90 91 def _AddStructComputedData(self, exported, struct): 92 """Adds computed data to the given struct. The data is computed once and 93 used repeatedly in the generation process.""" 94 struct.packed = pack.PackedStruct(struct) 95 struct.bytes = pack.GetByteLayout(struct.packed) 96 struct.versions = pack.GetVersionInfo(struct.packed) 97 struct.exported = exported 98 return struct 99 100 def _AddUnionComputedData(self, union): 101 """Adds computed data to the given union. The data is computed once and 102 used repeatedly in the generation process.""" 103 ordinal = 0 104 for field in union.fields: 105 if field.ordinal is not None: 106 ordinal = field.ordinal 107 field.ordinal = ordinal 108 ordinal += 1 109 return union 110 111 def _AddInterfaceComputedData(self, interface): 112 """Adds computed data to the given interface. The data is computed once and 113 used repeatedly in the generation process.""" 114 interface.version = 0 115 for method in interface.methods: 116 if method.min_version is not None: 117 interface.version = max(interface.version, method.min_version) 118 119 method.param_struct = self._GetStructFromMethod(method) 120 interface.version = max(interface.version, 121 method.param_struct.versions[-1].version) 122 123 if method.response_parameters is not None: 124 method.response_param_struct = self._GetResponseStructFromMethod(method) 125 interface.version = max( 126 interface.version, 127 method.response_param_struct.versions[-1].version) 128 else: 129 method.response_param_struct = None 130 return interface 131 132 def _GetStructFromMethod(self, method): 133 """Converts a method's parameters into the fields of a struct.""" 134 params_class = "%s_%s_Params" % (method.interface.name, method.name) 135 struct = mojom.Struct(params_class, module=method.interface.module) 136 for param in method.parameters: 137 struct.AddField(param.name, param.kind, param.ordinal, 138 attributes=param.attributes) 139 return self._AddStructComputedData(False, struct) 140 141 def _GetResponseStructFromMethod(self, method): 142 """Converts a method's response_parameters into the fields of a struct.""" 143 params_class = "%s_%s_ResponseParams" % (method.interface.name, method.name) 144 struct = mojom.Struct(params_class, module=method.interface.module) 145 for param in method.response_parameters: 146 struct.AddField(param.name, param.kind, param.ordinal, 147 attributes=param.attributes) 148 return self._AddStructComputedData(False, struct) 149