
The methodology used to generate this CL is documented in https://crbug.com/1098010#c95. No-Try: true No-Presubmit: true Bug: 1098010 Change-Id: I3a8a7b150e7bd64690534727150646081df50439 Reviewed-on: https://chromium-review.googlesource.com/c/chromium/src/+/3900697 Reviewed-by: Mark Mentovai <mark@chromium.org> Auto-Submit: Avi Drissman <avi@chromium.org> Owners-Override: Avi Drissman <avi@chromium.org> Commit-Queue: Avi Drissman <avi@chromium.org> Cr-Commit-Position: refs/heads/main@{#1047644}
159 lines
6.3 KiB
Python
159 lines
6.3 KiB
Python
# Copyright 2012 The Chromium Authors
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
# found in the LICENSE file.
|
|
|
|
import json
|
|
import struct_generator
|
|
|
|
def _JSONToCString16(json_string_literal):
|
|
"""Converts a JSON string literal to a C++ UTF-16 string literal. This is
|
|
done by converting \\u#### to \\x####.
|
|
"""
|
|
c_string_literal = json_string_literal
|
|
escape_index = c_string_literal.find('\\')
|
|
while escape_index > 0:
|
|
if c_string_literal[escape_index + 1] == 'u':
|
|
# We close the C string literal after the 4 hex digits and reopen it right
|
|
# after, otherwise the Windows compiler will sometimes try to get more
|
|
# than 4 characters in the hex string.
|
|
c_string_literal = (c_string_literal[0:escape_index + 1] + 'x' +
|
|
c_string_literal[escape_index + 2:escape_index + 6] + '" L"' +
|
|
c_string_literal[escape_index + 6:])
|
|
escape_index = c_string_literal.find('\\', escape_index + 6)
|
|
return c_string_literal
|
|
|
|
def _GenerateString(content, lines, indent=' '):
|
|
"""Generates an UTF-8 string to be included in a static structure initializer.
|
|
If content is not specified, uses NULL.
|
|
"""
|
|
if content is None:
|
|
lines.append(indent + 'NULL,')
|
|
else:
|
|
# json.dumps quotes the string and escape characters as required.
|
|
lines.append(indent + '%s,' % json.dumps(content))
|
|
|
|
def _GenerateString16(content, lines, indent=' '):
|
|
"""Generates an UTF-16 string to be included in a static structure
|
|
initializer. If content is not specified, uses NULL.
|
|
"""
|
|
if content is None:
|
|
lines.append(indent + 'NULL,')
|
|
else:
|
|
# json.dumps quotes the string and escape characters as required.
|
|
lines.append(indent + 'L%s,' % _JSONToCString16(json.dumps(content)))
|
|
|
|
def _GenerateArrayVariableName(element_name, field_name, field_name_count):
|
|
# Generates a unique variable name for an array variable.
|
|
var = 'array_%s_%s' % (element_name, field_name)
|
|
if var not in field_name_count:
|
|
field_name_count[var] = 0
|
|
return var
|
|
new_var = '%s_%d' % (var, field_name_count[var])
|
|
field_name_count[var] += 1
|
|
return new_var
|
|
|
|
def _GenerateArray(element_name, field_info, content, lines, indent,
|
|
field_name_count):
|
|
"""Generates an array to be included in a static structure initializer. If
|
|
content is not specified, uses NULL. The array is assigned to a temporary
|
|
variable which is initialized before the structure.
|
|
"""
|
|
if content is None:
|
|
lines.append(indent + 'NULL,')
|
|
lines.append(indent + '0,') # Size of the array.
|
|
return
|
|
|
|
# Create a new array variable and use it in the structure initializer.
|
|
# This prohibits nested arrays. Add a clash detection and renaming mechanism
|
|
# to solve the problem.
|
|
var = _GenerateArrayVariableName(element_name, field_info['field'],
|
|
field_name_count)
|
|
lines.append(indent + '%s,' % var)
|
|
lines.append(indent + '%s,' % len(content)) # Size of the array.
|
|
# Generate the array content.
|
|
array_lines = []
|
|
field_info['contents']['field'] = var;
|
|
array_lines.append(struct_generator.GenerateField(
|
|
field_info['contents']) + '[] = {')
|
|
for subcontent in content:
|
|
GenerateFieldContent(element_name, field_info['contents'], subcontent,
|
|
array_lines, indent, field_name_count)
|
|
array_lines.append('};')
|
|
# Prepend the generated array so it is initialized before the structure.
|
|
lines.reverse()
|
|
array_lines.reverse()
|
|
lines.extend(array_lines)
|
|
lines.reverse()
|
|
|
|
def _GenerateStruct(element_name, field_info, content, lines, indent,
|
|
field_name_count):
|
|
"""Generates a struct to be included in a static structure initializer. If
|
|
content is not specified, uses {0}.
|
|
"""
|
|
if content is None:
|
|
lines.append(indent + '{0},')
|
|
return
|
|
|
|
fields = field_info['fields']
|
|
lines.append(indent + '{')
|
|
for field in fields:
|
|
subcontent = content.get(field['field'])
|
|
GenerateFieldContent(element_name, field, subcontent, lines, ' ' + indent,
|
|
field_name_count)
|
|
lines.append(indent + '},')
|
|
|
|
def GenerateFieldContent(element_name, field_info, content, lines, indent,
|
|
field_name_count):
|
|
"""Generate the content of a field to be included in the static structure
|
|
initializer. If the field's content is not specified, uses the default value
|
|
if one exists.
|
|
"""
|
|
if content is None:
|
|
content = field_info.get('default', None)
|
|
type = field_info['type']
|
|
if type in ('int', 'enum', 'class'):
|
|
lines.append('%s%s,' % (indent, content))
|
|
elif type == 'string':
|
|
_GenerateString(content, lines, indent)
|
|
elif type == 'string16':
|
|
_GenerateString16(content, lines, indent)
|
|
elif type == 'array':
|
|
_GenerateArray(element_name, field_info, content, lines, indent,
|
|
field_name_count)
|
|
elif type == 'struct':
|
|
_GenerateStruct(element_name, field_info, content, lines, indent,
|
|
field_name_count)
|
|
else:
|
|
raise RuntimeError('Unknown field type "%s"' % type)
|
|
|
|
def GenerateElement(type_name, schema, element_name, element, field_name_count):
|
|
"""Generate the static structure initializer for one element.
|
|
"""
|
|
lines = [];
|
|
lines.append('const %s %s = {' % (type_name, element_name));
|
|
for field_info in schema:
|
|
content = element.get(field_info['field'], None)
|
|
if (content == None and not field_info.get('optional', False)):
|
|
raise RuntimeError('Mandatory field "%s" omitted in element "%s".' %
|
|
(field_info['field'], element_name))
|
|
GenerateFieldContent(element_name, field_info, content, lines, ' ',
|
|
field_name_count)
|
|
lines.append('};')
|
|
return '\n'.join(lines)
|
|
|
|
def GenerateElements(type_name, schema, description, field_name_count={}):
|
|
"""Generate the static structure initializer for all the elements in the
|
|
description['elements'] dictionary, as well as for any variables in
|
|
description['int_variables'].
|
|
"""
|
|
result = [];
|
|
for var_name, value in description.get('int_variables', {}).items():
|
|
result.append('const int %s = %s;' % (var_name, value))
|
|
result.append('')
|
|
|
|
for element_name, element in description.get('elements', {}).items():
|
|
result.append(GenerateElement(type_name, schema, element_name, element,
|
|
field_name_count))
|
|
result.append('')
|
|
return '\n'.join(result)
|