blob: 11d2228603a66c410edc7766e1e464cdb1bf983f [file] [log] [blame]
# Copyright (C) 2013 Google Inc. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following disclaimer
# in the documentation and/or other materials provided with the
# distribution.
# * Neither the name of Google Inc. nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
import os.path
import re
import in_generator
from in_generator import Maker
import license
HEADER_TEMPLATE = """%(license)s
#ifndef %(class_name)sHeaders_h
#define %(class_name)sHeaders_h
%(includes)s
#endif // %(class_name)sHeaders_h
"""
INTERFACES_HEADER_TEMPLATE = """%(license)s
#ifndef %(class_name)sInterfaces_h
#define %(class_name)sInterfaces_h
%(declare_conditional_macros)s
#define %(macro_style_name)s_INTERFACES_FOR_EACH(macro) \\
\\
%(unconditional_macros)s
\\
%(conditional_macros)s
#endif // %(class_name)sInterfaces_h
"""
def _to_macro_style(name):
s1 = re.sub('(.)([A-Z][a-z]+)', r'\1_\2', name)
return re.sub('([a-z0-9])([A-Z])', r'\1_\2', s1).upper()
class Writer(in_generator.Writer):
def __init__(self, in_file_path, enabled_conditions):
super(Writer, self).__init__(in_file_path, enabled_conditions)
self.class_name = self.in_file.parameters['namespace'].strip('"')
self._entries_by_conditional = {}
self._unconditional_entries = []
self._sort_entries_by_conditional()
self._outputs = {(self.class_name + "Headers.h"): self.generate_headers_header,
(self.class_name + "Interfaces.h"): self.generate_interfaces_header,
}
def _sort_entries_by_conditional(self):
unconditional_names = set()
for entry in self.in_file.name_dictionaries:
conditional = entry['conditional']
if not conditional:
name = self._class_name_for_entry(entry)
if name in unconditional_names:
continue
unconditional_names.add(name)
self._unconditional_entries.append(entry)
continue
for entry in self.in_file.name_dictionaries:
name = self._class_name_for_entry(entry)
if name in unconditional_names:
continue
conditional = entry['conditional']
if not conditional in self._entries_by_conditional:
self._entries_by_conditional[conditional] = []
self._entries_by_conditional[conditional].append(entry)
def _class_name_for_entry(self, entry):
if entry['implementedAs']:
return entry['implementedAs']
return os.path.basename(entry['name'])
def _headers_header_include_path(self, entry):
if entry['implementedAs']:
path = os.path.dirname(entry['name'])
if len(path):
path += '/'
path += entry['implementedAs']
else:
path = entry['name']
return path + '.h'
def _headers_header_includes(self, entries):
includes = dict()
for entry in entries:
class_name = self._class_name_for_entry(entry)
# Avoid duplicate includes.
if class_name in includes:
continue
include = '#include "%(path)s"\n#include "V8%(js_name)s.h"' % {
'path': self._headers_header_include_path(entry),
'js_name': os.path.basename(entry['name']),
}
includes[class_name] = self.wrap_with_condition(include, entry['conditional'])
return includes.values()
def generate_headers_header(self):
return HEADER_TEMPLATE % {
'license': license.license_for_generated_cpp(),
'class_name': self.class_name,
'includes': '\n'.join(self._headers_header_includes(self.in_file.name_dictionaries)),
}
def _declare_one_conditional_macro(self, conditional, entries):
macro_name = '%(macro_style_name)s_INTERFACES_FOR_EACH_%(conditional)s' % {
'macro_style_name': _to_macro_style(self.class_name),
'conditional': conditional,
}
return self.wrap_with_condition("""#define %(macro_name)s(macro) \\
%(declarations)s
#else
#define %(macro_name)s(macro)""" % {
'macro_name': macro_name,
'declarations': '\n'.join(sorted(set([
' macro(%(name)s) \\' % {'name': self._class_name_for_entry(entry)}
for entry in entries]))),
}, conditional)
def _declare_conditional_macros(self):
return '\n'.join([
self._declare_one_conditional_macro(conditional, entries)
for conditional, entries in self._entries_by_conditional.items()])
def _unconditional_macro(self, entry):
return ' macro(%(name)s) \\' % {'name': self._class_name_for_entry(entry)}
def _conditional_macros(self, conditional):
return ' %(macro_style_name)s_INTERFACES_FOR_EACH_%(conditional)s(macro) \\' % {
'macro_style_name': _to_macro_style(self.class_name),
'conditional': conditional,
}
def generate_interfaces_header(self):
return INTERFACES_HEADER_TEMPLATE % {
'license': license.license_for_generated_cpp(),
'class_name': self.class_name,
'macro_style_name': _to_macro_style(self.class_name),
'declare_conditional_macros': self._declare_conditional_macros(),
'unconditional_macros': '\n'.join(sorted(set(map(self._unconditional_macro, self._unconditional_entries)))),
'conditional_macros': '\n'.join(map(self._conditional_macros, self._entries_by_conditional.keys())),
}