blob: 533c0cf8affaefb2d8ad6687452c3a3b9dab77cc [file] [log] [blame]
#!/usr/bin/env python
# Copyright 2016 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 math
import sys
import json5_generator
import template_expander
import make_style_builder
from name_utilities import (
enum_for_css_keyword, enum_type_name, enum_value_name, class_member_name, method_name,
join_name
)
from collections import OrderedDict
class Field(object):
"""
The generated ComputedStyle object is made up of a series of Fields.
Each Field has a name, size, type, etc, and a bunch of attributes to
determine which methods it will be used in.
A Field also has enough information to use any storage type in C++, such as
regular member variables, or more complex storage like vectors or hashmaps.
Almost all properties will have at least one Field, often more than one.
Most attributes in this class correspond to parameters in CSSProperties.json5.
See that file for a more detailed explanation of each attribute.
Attributes:
field_role: The semantic role of the field. Can be:
- 'property': for fields that store CSS properties
- 'inherited_flag': for single-bit flags that store whether a property is
inherited by this style or set explicitly
- 'nonproperty': for fields that are not CSS properties
name_for_methods: String used to form the names of getters and setters.
Should be in upper camel case.
property_name: Name of the property that the field is part of.
type_name: Name of the C++ type exposed by the generated interface (e.g. EClear, int).
field_template: Determines the interface generated for the field. Can be one of:
keyword, flag, or monotonic_flag.
size: Number of bits needed for storage.
default_value: Default value for this field when it is first initialized.
"""
def __init__(self, field_role, name_for_methods, property_name, type_name,
field_template, size, default_value, getter_method_name, setter_method_name,
initial_method_name, **kwargs):
"""Creates a new field."""
self.name = class_member_name(name_for_methods)
self.property_name = property_name
self.type_name = type_name
self.field_template = field_template
self.size = size
self.default_value = default_value
# Field role: one of these must be true
self.is_property = field_role == 'property'
self.is_inherited_flag = field_role == 'inherited_flag'
self.is_nonproperty = field_role == 'nonproperty'
assert (self.is_property, self.is_inherited_flag, self.is_nonproperty).count(True) == 1, \
'Field role has to be exactly one of: property, inherited_flag, nonproperty'
if not self.is_inherited_flag:
self.is_inherited = kwargs.pop('inherited')
self.is_independent = kwargs.pop('independent')
assert self.is_inherited or not self.is_independent, 'Only inherited fields can be independent'
self.is_inherited_method_name = method_name(join_name(name_for_methods, 'is inherited'))
# Method names
# TODO(nainar): Method name generation is inconsistent. Fix.
self.getter_method_name = getter_method_name
self.setter_method_name = setter_method_name
self.internal_getter_method_name = method_name(join_name(getter_method_name, 'Internal'))
self.internal_mutable_method_name = method_name(join_name('Mutable', name_for_methods, 'Internal'))
self.internal_setter_method_name = method_name(join_name(setter_method_name, 'Internal'))
self.initial_method_name = initial_method_name
self.resetter_method_name = method_name(join_name('Reset', name_for_methods))
# If the size of the field is not None, it means it is a bit field
self.is_bit_field = self.size is not None
assert len(kwargs) == 0, 'Unexpected arguments provided to Field: ' + str(kwargs)
def _get_include_paths(properties):
"""
Get a list of paths that need to be included for ComputedStyleBase.
"""
include_paths = set()
for property_ in properties:
if property_['field_type_path'] is not None:
include_paths.add(property_['field_type_path'] + '.h')
return list(sorted(include_paths))
def _create_enums(properties):
"""
Returns an OrderedDict of enums to be generated, enum name -> [list of enum values]
"""
enums = {}
for property_ in properties:
# Only generate enums for keyword properties that use the default field_type_path.
if property_['field_template'] == 'keyword' and property_['field_type_path'] is None:
enum_name = property_['type_name']
enum_values = [enum_value_name(k) for k in property_['keywords']]
if enum_name in enums:
# There's an enum with the same name, check if the enum values are the same
assert set(enums[enum_name]) == set(enum_values), \
("'" + property_['name'] + "' can't have type_name '" + enum_name + "' "
"because it was used by a previous property, but with a different set of keywords. "
"Either give it a different name or ensure the keywords are the same.")
enums[enum_name] = enum_values
# Return the enums sorted by key (enum name)
return OrderedDict(sorted(enums.items(), key=lambda t: t[0]))
def _create_field(field_role, property_):
"""
Create a property or nonproperty field.
"""
assert field_role in ('property', 'nonproperty')
name_for_methods = property_['name_for_methods']
assert property_['default_value'] is not None, \
('MakeComputedStyleBase requires an default value for all fields, none specified '
'for property ' + property_['name'])
if property_['field_template'] == 'keyword':
type_name = property_['type_name']
default_value = type_name + '::' + enum_value_name(property_['default_value'])
size = int(math.ceil(math.log(len(property_['keywords']), 2)))
elif property_['field_template'] == 'storage_only':
# 'storage_only' fields need to specify a size, type_name and default_value
type_name = property_['type_name']
default_value = property_['default_value']
size = property_['field_size']
elif property_['field_template'] == 'external':
type_name = property_['type_name']
default_value = property_['default_value']
size = None
elif property_['field_template'] == 'primitive':
type_name = property_['type_name']
default_value = property_['default_value']
size = 1 if type_name == 'bool' else None # pack bools with 1 bit.
else:
assert property_['field_template'] in ('monotonic_flag',)
type_name = 'bool'
default_value = 'false'
size = 1
return Field(
field_role,
name_for_methods,
property_name=property_['name'],
inherited=property_['inherited'],
independent=property_['independent'],
type_name=type_name,
field_template=property_['field_template'],
size=size,
default_value=default_value,
getter_method_name=property_['getter'],
setter_method_name=property_['setter'],
initial_method_name=property_['initial'],
)
def _create_inherited_flag_field(property_):
"""
Create the field used for an inheritance fast path from an independent CSS property,
and return the Field object.
"""
name_for_methods = join_name(property_['name_for_methods'], 'is inherited')
return Field(
'inherited_flag',
name_for_methods,
property_name=property_['name'],
type_name='bool',
field_template='primitive',
size=1,
default_value='true',
getter_method_name=method_name(name_for_methods),
setter_method_name=method_name(join_name('set', name_for_methods)),
initial_method_name=method_name(join_name('initial', name_for_methods)),
)
def _create_fields(properties):
"""
Create ComputedStyle fields from properties or nonproperties and return a list of Field objects.
"""
fields = []
for property_ in properties:
# Only generate properties that have a field template
if property_['field_template'] is not None:
# If the property is independent, add the single-bit sized isInherited flag
# to the list of Fields as well.
if property_['independent']:
fields.append(_create_inherited_flag_field(property_))
# TODO(shend): Get rid of the property/nonproperty field roles.
field_role = 'nonproperty' if property_['has_custom_compare_and_copy'] else 'property'
fields.append(_create_field(field_role, property_))
return fields
def _pack_fields(fields):
"""
Group a list of fields into buckets to minimise padding.
Returns a list of buckets, where each bucket is a list of Field objects.
"""
# Since fields cannot cross word boundaries, in order to minimize
# padding, group fields into buckets so that as many buckets as possible
# are exactly 32 bits. Although this greedy approach may not always
# produce the optimal solution, we add a static_assert to the code to
# ensure ComputedStyleBase results in the expected size. If that
# static_assert fails, this code is falling into the small number of
# cases that are suboptimal, and may need to be rethought.
# For more details on packing bit fields to reduce padding, see:
# http://www.catb.org/esr/structure-packing/#_bitfields
field_buckets = []
# Consider fields in descending order of size to reduce fragmentation
# when they are selected. Ties broken in alphabetical order by name.
for field in sorted(fields, key=lambda f: (-f.size, f.name)):
added_to_bucket = False
# Go through each bucket and add this field if it will not increase
# the bucket's size to larger than 32 bits. Otherwise, make a new
# bucket containing only this field.
for bucket in field_buckets:
if sum(f.size for f in bucket) + field.size <= 32:
bucket.append(field)
added_to_bucket = True
break
if not added_to_bucket:
field_buckets.append([field])
return field_buckets
class ComputedStyleBaseWriter(make_style_builder.StyleBuilderWriter):
def __init__(self, json5_file_paths):
# Read CSS properties
super(ComputedStyleBaseWriter, self).__init__([json5_file_paths[0]])
# Ignore shorthand properties
for property_ in self._properties.values():
if property_['field_template'] is not None:
assert not property_['longhands'], \
"Shorthand '{}' cannot have a field_template.".format(property_['name'])
css_properties = [value for value in self._properties.values() if not value['longhands']]
for property_ in css_properties:
# All CSS properties that are generated do not have custom comparison and copy logic.
property_['has_custom_compare_and_copy'] = False
# CSS properties are not allowed to explicitly specify their field_size.
property_['field_size'] = None
# Read extra fields using the parameter specification from the CSS properties file.
extra_fields = json5_generator.Json5File.load_from_files(
[json5_file_paths[1]],
default_parameters=self.json5_file.parameters
).name_dictionaries
for property_ in extra_fields:
make_style_builder.apply_property_naming_defaults(property_)
all_properties = css_properties + extra_fields
# Override the type name when field_type_path is specified
for property_ in all_properties:
if property_['field_type_path']:
property_['type_name'] = property_['field_type_path'].split('/')[-1]
self._generated_enums = _create_enums(all_properties)
all_fields = _create_fields(all_properties)
# Separate the normal fields from the bit fields
bit_fields = [field for field in all_fields if field.is_bit_field]
normal_fields = [field for field in all_fields if not field.is_bit_field]
# Pack bit fields into buckets
field_buckets = _pack_fields(bit_fields)
# The expected size of ComputedStyleBase is equivalent to as many words
# as the total number of buckets.
self._expected_bit_field_bytes = len(field_buckets)
# The most optimal size of ComputedStyleBase is the total sum of all the
# field sizes, rounded up to the nearest word. If this produces the
# incorrect value, either the packing algorithm is not optimal or there
# is no way to pack the fields such that excess padding space is not
# added.
# If this fails, increase extra_padding_bytes by 1, but be aware that
# this also increases ComputedStyleBase by 1 word.
# We should be able to bring extra_padding_bytes back to 0 from time to
# time.
extra_padding_bytes = 0
optimal_bit_field_bytes = int(math.ceil(sum(f.size for f in bit_fields) / 32.0))
real_bit_field_bytes = optimal_bit_field_bytes + extra_padding_bytes
assert self._expected_bit_field_bytes == real_bit_field_bytes, \
('The field packing algorithm produced %s bytes, optimal is %s bytes' %
(self._expected_bit_field_bytes, real_bit_field_bytes))
# Normal fields go first, then the bit fields.
self._fields = list(normal_fields)
# Order the fields so fields in each bucket are adjacent.
for bucket in field_buckets:
for field in bucket:
self._fields.append(field)
self._include_paths = _get_include_paths(all_properties)
self._outputs = {
'ComputedStyleBase.h': self.generate_base_computed_style_h,
'ComputedStyleBase.cpp': self.generate_base_computed_style_cpp,
'ComputedStyleBaseConstants.h': self.generate_base_computed_style_constants,
}
@template_expander.use_jinja('ComputedStyleBase.h.tmpl')
def generate_base_computed_style_h(self):
return {
'properties': self._properties,
'enums': self._generated_enums,
'include_paths': self._include_paths,
'fields': self._fields,
}
@template_expander.use_jinja('ComputedStyleBase.cpp.tmpl')
def generate_base_computed_style_cpp(self):
return {
'properties': self._properties,
'enums': self._generated_enums,
'fields': self._fields,
'expected_bit_field_bytes': self._expected_bit_field_bytes,
}
@template_expander.use_jinja('ComputedStyleBaseConstants.h.tmpl')
def generate_base_computed_style_constants(self):
return {
'properties': self._properties,
'enums': self._generated_enums,
'fields': self._fields,
}
if __name__ == '__main__':
json5_generator.Maker(ComputedStyleBaseWriter).main()