Merge changes I0f73b41a,I0cd2f80c
* changes: Add a real parser for the version scripts. Add tag for "versioned=API".
This commit is contained in:
@@ -16,6 +16,7 @@
|
|||||||
#
|
#
|
||||||
"""Generates source for stub shared libraries for the NDK."""
|
"""Generates source for stub shared libraries for the NDK."""
|
||||||
import argparse
|
import argparse
|
||||||
|
import logging
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
|
||||||
@@ -30,45 +31,28 @@ ALL_ARCHITECTURES = (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class Scope(object):
|
def logger():
|
||||||
"""Enum for version script scope.
|
"""Return the main logger for this module."""
|
||||||
|
return logging.getLogger(__name__)
|
||||||
Top: Top level of the file.
|
|
||||||
Global: In a version and visibility section where symbols should be visible
|
|
||||||
to the NDK.
|
|
||||||
Local: In a visibility section of a public version where symbols should be
|
|
||||||
hidden to the NDK.
|
|
||||||
Private: In a version where symbols should not be visible to the NDK.
|
|
||||||
"""
|
|
||||||
Top = 1
|
|
||||||
Global = 2
|
|
||||||
Local = 3
|
|
||||||
Private = 4
|
|
||||||
|
|
||||||
|
|
||||||
class Stack(object):
|
|
||||||
"""Basic stack implementation."""
|
|
||||||
def __init__(self):
|
|
||||||
self.stack = []
|
|
||||||
|
|
||||||
def push(self, obj):
|
|
||||||
"""Push an item on to the stack."""
|
|
||||||
self.stack.append(obj)
|
|
||||||
|
|
||||||
def pop(self):
|
|
||||||
"""Remove and return the item on the top of the stack."""
|
|
||||||
return self.stack.pop()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def top(self):
|
|
||||||
"""Return the top of the stack."""
|
|
||||||
return self.stack[-1]
|
|
||||||
|
|
||||||
|
|
||||||
def get_tags(line):
|
def get_tags(line):
|
||||||
"""Returns a list of all tags on this line."""
|
"""Returns a list of all tags on this line."""
|
||||||
_, _, all_tags = line.strip().partition('#')
|
_, _, all_tags = line.strip().partition('#')
|
||||||
return re.split(r'\s+', all_tags)
|
return [e for e in re.split(r'\s+', all_tags) if e.strip()]
|
||||||
|
|
||||||
|
|
||||||
|
def get_tag_value(tag):
|
||||||
|
"""Returns the value of a key/value tag.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: Tag is not a key/value type tag.
|
||||||
|
|
||||||
|
Returns: Value part of tag as a string.
|
||||||
|
"""
|
||||||
|
if '=' not in tag:
|
||||||
|
raise ValueError('Not a key/value tag: ' + tag)
|
||||||
|
return tag.partition('=')[2]
|
||||||
|
|
||||||
|
|
||||||
def version_is_private(version):
|
def version_is_private(version):
|
||||||
@@ -87,82 +71,11 @@ def should_omit_version(name, tags, arch, api):
|
|||||||
return True
|
return True
|
||||||
if not symbol_in_arch(tags, arch):
|
if not symbol_in_arch(tags, arch):
|
||||||
return True
|
return True
|
||||||
if not symbol_in_version(tags, arch, api):
|
if not symbol_in_api(tags, arch, api):
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
def enter_version(scope, line, version_file, arch, api):
|
|
||||||
"""Enters a new version block scope."""
|
|
||||||
if scope.top != Scope.Top:
|
|
||||||
raise RuntimeError('Encountered nested version block.')
|
|
||||||
|
|
||||||
# Entering a new version block. By convention symbols with versions ending
|
|
||||||
# with "_PRIVATE" or "_PLATFORM" are not included in the NDK.
|
|
||||||
version_name = line.split('{')[0].strip()
|
|
||||||
tags = get_tags(line)
|
|
||||||
if should_omit_version(version_name, tags, arch, api):
|
|
||||||
scope.push(Scope.Private)
|
|
||||||
else:
|
|
||||||
scope.push(Scope.Global) # By default symbols are visible.
|
|
||||||
version_file.write(line)
|
|
||||||
|
|
||||||
|
|
||||||
def leave_version(scope, line, version_file):
|
|
||||||
"""Leave a version block scope."""
|
|
||||||
# There is no close to a visibility section, just the end of the version or
|
|
||||||
# a new visiblity section.
|
|
||||||
assert scope.top in (Scope.Global, Scope.Local, Scope.Private)
|
|
||||||
if scope.top != Scope.Private:
|
|
||||||
version_file.write(line)
|
|
||||||
scope.pop()
|
|
||||||
assert scope.top == Scope.Top
|
|
||||||
|
|
||||||
|
|
||||||
def enter_visibility(scope, line, version_file):
|
|
||||||
"""Enters a new visibility block scope."""
|
|
||||||
leave_visibility(scope)
|
|
||||||
version_file.write(line)
|
|
||||||
visibility = line.split(':')[0].strip()
|
|
||||||
if visibility == 'local':
|
|
||||||
scope.push(Scope.Local)
|
|
||||||
elif visibility == 'global':
|
|
||||||
scope.push(Scope.Global)
|
|
||||||
else:
|
|
||||||
raise RuntimeError('Unknown visiblity label: ' + visibility)
|
|
||||||
|
|
||||||
|
|
||||||
def leave_visibility(scope):
|
|
||||||
"""Leaves a visibility block scope."""
|
|
||||||
assert scope.top in (Scope.Global, Scope.Local)
|
|
||||||
scope.pop()
|
|
||||||
assert scope.top == Scope.Top
|
|
||||||
|
|
||||||
|
|
||||||
def handle_top_scope(scope, line, version_file, arch, api):
|
|
||||||
"""Processes a line in the top level scope."""
|
|
||||||
if '{' in line:
|
|
||||||
enter_version(scope, line, version_file, arch, api)
|
|
||||||
else:
|
|
||||||
raise RuntimeError('Unexpected contents at top level: ' + line)
|
|
||||||
|
|
||||||
|
|
||||||
def handle_private_scope(scope, line, version_file):
|
|
||||||
"""Eats all input."""
|
|
||||||
if '}' in line:
|
|
||||||
leave_version(scope, line, version_file)
|
|
||||||
|
|
||||||
|
|
||||||
def handle_local_scope(scope, line, version_file):
|
|
||||||
"""Passes through input."""
|
|
||||||
if ':' in line:
|
|
||||||
enter_visibility(scope, line, version_file)
|
|
||||||
elif '}' in line:
|
|
||||||
leave_version(scope, line, version_file)
|
|
||||||
else:
|
|
||||||
version_file.write(line)
|
|
||||||
|
|
||||||
|
|
||||||
def symbol_in_arch(tags, arch):
|
def symbol_in_arch(tags, arch):
|
||||||
"""Returns true if the symbol is present for the given architecture."""
|
"""Returns true if the symbol is present for the given architecture."""
|
||||||
has_arch_tags = False
|
has_arch_tags = False
|
||||||
@@ -178,8 +91,8 @@ def symbol_in_arch(tags, arch):
|
|||||||
return not has_arch_tags
|
return not has_arch_tags
|
||||||
|
|
||||||
|
|
||||||
def symbol_in_version(tags, arch, version):
|
def symbol_in_api(tags, arch, api):
|
||||||
"""Returns true if the symbol is present for the given version."""
|
"""Returns true if the symbol is present for the given API level."""
|
||||||
introduced_tag = None
|
introduced_tag = None
|
||||||
arch_specific = False
|
arch_specific = False
|
||||||
for tag in tags:
|
for tag in tags:
|
||||||
@@ -191,7 +104,7 @@ def symbol_in_version(tags, arch, version):
|
|||||||
arch_specific = True
|
arch_specific = True
|
||||||
elif tag == 'future':
|
elif tag == 'future':
|
||||||
# This symbol is not in any released API level.
|
# This symbol is not in any released API level.
|
||||||
# TODO(danalbert): These need to be emitted for version == current.
|
# TODO(danalbert): These need to be emitted for api == current.
|
||||||
# That's not a construct we have yet, so just skip it for now.
|
# That's not a construct we have yet, so just skip it for now.
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@@ -200,66 +113,204 @@ def symbol_in_version(tags, arch, version):
|
|||||||
# available.
|
# available.
|
||||||
return True
|
return True
|
||||||
|
|
||||||
# The tag is a key=value pair, and we only care about the value now.
|
return api >= int(get_tag_value(introduced_tag))
|
||||||
_, _, version_str = introduced_tag.partition('=')
|
|
||||||
return version >= int(version_str)
|
|
||||||
|
|
||||||
|
|
||||||
def handle_global_scope(scope, line, src_file, version_file, arch, api):
|
def symbol_versioned_in_api(tags, api):
|
||||||
"""Emits present symbols to the version file and stub source file."""
|
"""Returns true if the symbol should be versioned for the given API.
|
||||||
if ':' in line:
|
|
||||||
enter_visibility(scope, line, version_file)
|
|
||||||
return
|
|
||||||
if '}' in line:
|
|
||||||
leave_version(scope, line, version_file)
|
|
||||||
return
|
|
||||||
|
|
||||||
if ';' not in line:
|
This models the `versioned=API` tag. This should be a very uncommonly
|
||||||
raise RuntimeError('Expected ; to terminate symbol: ' + line)
|
needed tag, and is really only needed to fix versioning mistakes that are
|
||||||
if '*' in line:
|
already out in the wild.
|
||||||
raise RuntimeError('Wildcard global symbols are not permitted.')
|
|
||||||
|
|
||||||
# Line is now in the format "<symbol-name>; # tags"
|
For example, some of libc's __aeabi_* functions were originally placed in
|
||||||
# Tags are whitespace separated.
|
the private version, but that was incorrect. They are now in LIBC_N, but
|
||||||
symbol_name, _, rest = line.strip().partition(';')
|
when building against any version prior to N we need the symbol to be
|
||||||
tags = get_tags(line)
|
unversioned (otherwise it won't resolve on M where it is private).
|
||||||
|
"""
|
||||||
if not symbol_in_arch(tags, arch):
|
for tag in tags:
|
||||||
return
|
if tag.startswith('versioned='):
|
||||||
if not symbol_in_version(tags, arch, api):
|
return api >= int(get_tag_value(tag))
|
||||||
return
|
# If there is no "versioned" tag, the tag has been versioned for as long as
|
||||||
|
# it was introduced.
|
||||||
if 'var' in tags:
|
return True
|
||||||
src_file.write('int {} = 0;\n'.format(symbol_name))
|
|
||||||
else:
|
|
||||||
src_file.write('void {}() {{}}\n'.format(symbol_name))
|
|
||||||
version_file.write(line)
|
|
||||||
|
|
||||||
|
|
||||||
def generate(symbol_file, src_file, version_file, arch, api):
|
class ParseError(RuntimeError):
|
||||||
"""Generates the stub source file and version script."""
|
"""An error that occurred while parsing a symbol file."""
|
||||||
scope = Stack()
|
pass
|
||||||
scope.push(Scope.Top)
|
|
||||||
for line in symbol_file:
|
|
||||||
if line.strip() == '' or line.strip().startswith('#'):
|
class Version(object):
|
||||||
version_file.write(line)
|
"""A version block of a symbol file."""
|
||||||
elif scope.top == Scope.Top:
|
def __init__(self, name, base, tags, symbols):
|
||||||
handle_top_scope(scope, line, version_file, arch, api)
|
self.name = name
|
||||||
elif scope.top == Scope.Private:
|
self.base = base
|
||||||
handle_private_scope(scope, line, version_file)
|
self.tags = tags
|
||||||
elif scope.top == Scope.Local:
|
self.symbols = symbols
|
||||||
handle_local_scope(scope, line, version_file)
|
|
||||||
elif scope.top == Scope.Global:
|
def __eq__(self, other):
|
||||||
handle_global_scope(scope, line, src_file, version_file, arch, api)
|
if self.name != other.name:
|
||||||
|
return False
|
||||||
|
if self.base != other.base:
|
||||||
|
return False
|
||||||
|
if self.tags != other.tags:
|
||||||
|
return False
|
||||||
|
if self.symbols != other.symbols:
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class Symbol(object):
|
||||||
|
"""A symbol definition from a symbol file."""
|
||||||
|
def __init__(self, name, tags):
|
||||||
|
self.name = name
|
||||||
|
self.tags = tags
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return self.name == other.name and set(self.tags) == set(other.tags)
|
||||||
|
|
||||||
|
|
||||||
|
class SymbolFileParser(object):
|
||||||
|
"""Parses NDK symbol files."""
|
||||||
|
def __init__(self, input_file):
|
||||||
|
self.input_file = input_file
|
||||||
|
self.current_line = None
|
||||||
|
|
||||||
|
def parse(self):
|
||||||
|
"""Parses the symbol file and returns a list of Version objects."""
|
||||||
|
versions = []
|
||||||
|
while self.next_line() != '':
|
||||||
|
if '{' in self.current_line:
|
||||||
|
versions.append(self.parse_version())
|
||||||
|
else:
|
||||||
|
raise ParseError(
|
||||||
|
'Unexpected contents at top level: ' + self.current_line)
|
||||||
|
return versions
|
||||||
|
|
||||||
|
def parse_version(self):
|
||||||
|
"""Parses a single version section and returns a Version object."""
|
||||||
|
name = self.current_line.split('{')[0].strip()
|
||||||
|
tags = get_tags(self.current_line)
|
||||||
|
symbols = []
|
||||||
|
global_scope = True
|
||||||
|
while self.next_line() != '':
|
||||||
|
if '}' in self.current_line:
|
||||||
|
# Line is something like '} BASE; # tags'. Both base and tags
|
||||||
|
# are optional here.
|
||||||
|
base = self.current_line.partition('}')[2]
|
||||||
|
base = base.partition('#')[0].strip()
|
||||||
|
if not base.endswith(';'):
|
||||||
|
raise ParseError(
|
||||||
|
'Unterminated version block (expected ;).')
|
||||||
|
base = base.rstrip(';').rstrip()
|
||||||
|
if base == '':
|
||||||
|
base = None
|
||||||
|
return Version(name, base, tags, symbols)
|
||||||
|
elif ':' in self.current_line:
|
||||||
|
visibility = self.current_line.split(':')[0].strip()
|
||||||
|
if visibility == 'local':
|
||||||
|
global_scope = False
|
||||||
|
elif visibility == 'global':
|
||||||
|
global_scope = True
|
||||||
|
else:
|
||||||
|
raise ParseError('Unknown visiblity label: ' + visibility)
|
||||||
|
elif global_scope:
|
||||||
|
symbols.append(self.parse_symbol())
|
||||||
|
else:
|
||||||
|
# We're in a hidden scope. Ignore everything.
|
||||||
|
pass
|
||||||
|
raise ParseError('Unexpected EOF in version block.')
|
||||||
|
|
||||||
|
def parse_symbol(self):
|
||||||
|
"""Parses a single symbol line and returns a Symbol object."""
|
||||||
|
if ';' not in self.current_line:
|
||||||
|
raise ParseError(
|
||||||
|
'Expected ; to terminate symbol: ' + self.current_line)
|
||||||
|
if '*' in self.current_line:
|
||||||
|
raise ParseError(
|
||||||
|
'Wildcard global symbols are not permitted.')
|
||||||
|
# Line is now in the format "<symbol-name>; # tags"
|
||||||
|
name, _, _ = self.current_line.strip().partition(';')
|
||||||
|
tags = get_tags(self.current_line)
|
||||||
|
return Symbol(name, tags)
|
||||||
|
|
||||||
|
def next_line(self):
|
||||||
|
"""Returns the next non-empty non-comment line.
|
||||||
|
|
||||||
|
A return value of '' indicates EOF.
|
||||||
|
"""
|
||||||
|
line = self.input_file.readline()
|
||||||
|
while line.strip() == '' or line.strip().startswith('#'):
|
||||||
|
line = self.input_file.readline()
|
||||||
|
|
||||||
|
# We want to skip empty lines, but '' indicates EOF.
|
||||||
|
if line == '':
|
||||||
|
break
|
||||||
|
self.current_line = line
|
||||||
|
return self.current_line
|
||||||
|
|
||||||
|
|
||||||
|
class Generator(object):
|
||||||
|
"""Output generator that writes stub source files and version scripts."""
|
||||||
|
def __init__(self, src_file, version_script, arch, api):
|
||||||
|
self.src_file = src_file
|
||||||
|
self.version_script = version_script
|
||||||
|
self.arch = arch
|
||||||
|
self.api = api
|
||||||
|
|
||||||
|
def write(self, versions):
|
||||||
|
"""Writes all symbol data to the output files."""
|
||||||
|
for version in versions:
|
||||||
|
self.write_version(version)
|
||||||
|
|
||||||
|
def write_version(self, version):
|
||||||
|
"""Writes a single version block's data to the output files."""
|
||||||
|
name = version.name
|
||||||
|
tags = version.tags
|
||||||
|
if should_omit_version(name, tags, self.arch, self.api):
|
||||||
|
return
|
||||||
|
|
||||||
|
version_empty = True
|
||||||
|
pruned_symbols = []
|
||||||
|
for symbol in version.symbols:
|
||||||
|
if not symbol_in_arch(symbol.tags, self.arch):
|
||||||
|
continue
|
||||||
|
if not symbol_in_api(symbol.tags, self.arch, self.api):
|
||||||
|
continue
|
||||||
|
|
||||||
|
if symbol_versioned_in_api(symbol.tags, self.api):
|
||||||
|
version_empty = False
|
||||||
|
pruned_symbols.append(symbol)
|
||||||
|
|
||||||
|
if len(pruned_symbols) > 0:
|
||||||
|
if not version_empty:
|
||||||
|
self.version_script.write(version.name + ' {\n')
|
||||||
|
self.version_script.write(' global:\n')
|
||||||
|
for symbol in pruned_symbols:
|
||||||
|
if symbol_versioned_in_api(symbol.tags, self.api):
|
||||||
|
self.version_script.write(' ' + symbol.name + ';\n')
|
||||||
|
|
||||||
|
if 'var' in symbol.tags:
|
||||||
|
self.src_file.write('int {} = 0;\n'.format(symbol.name))
|
||||||
|
else:
|
||||||
|
self.src_file.write('void {}() {{}}\n'.format(symbol.name))
|
||||||
|
|
||||||
|
if not version_empty:
|
||||||
|
base = '' if version.base is None else ' ' + version.base
|
||||||
|
self.version_script.write('}' + base + ';\n')
|
||||||
|
|
||||||
|
|
||||||
def parse_args():
|
def parse_args():
|
||||||
"""Parses and returns command line arguments."""
|
"""Parses and returns command line arguments."""
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
|
|
||||||
parser.add_argument('--api', type=int, help='API level being targeted.')
|
parser.add_argument('-v', '--verbose', action='count', default=0)
|
||||||
|
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
'--arch', choices=ALL_ARCHITECTURES,
|
'--api', type=int, required=True, help='API level being targeted.')
|
||||||
|
parser.add_argument(
|
||||||
|
'--arch', choices=ALL_ARCHITECTURES, required=True,
|
||||||
help='Architecture being targeted.')
|
help='Architecture being targeted.')
|
||||||
|
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
@@ -278,11 +329,19 @@ def main():
|
|||||||
"""Program entry point."""
|
"""Program entry point."""
|
||||||
args = parse_args()
|
args = parse_args()
|
||||||
|
|
||||||
|
verbose_map = (logging.WARNING, logging.INFO, logging.DEBUG)
|
||||||
|
verbosity = args.verbose
|
||||||
|
if verbosity > 2:
|
||||||
|
verbosity = 2
|
||||||
|
logging.basicConfig(level=verbose_map[verbosity])
|
||||||
|
|
||||||
with open(args.symbol_file) as symbol_file:
|
with open(args.symbol_file) as symbol_file:
|
||||||
with open(args.stub_src, 'w') as src_file:
|
versions = SymbolFileParser(symbol_file).parse()
|
||||||
with open(args.version_script, 'w') as version_file:
|
|
||||||
generate(symbol_file, src_file, version_file, args.arch,
|
with open(args.stub_src, 'w') as src_file:
|
||||||
args.api)
|
with open(args.version_script, 'w') as version_file:
|
||||||
|
generator = Generator(src_file, version_file, args.arch, args.api)
|
||||||
|
generator.write(versions)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
440
cc/test_gen_stub_libs.py
Executable file
440
cc/test_gen_stub_libs.py
Executable file
@@ -0,0 +1,440 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Copyright (C) 2016 The Android Open Source Project
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
#
|
||||||
|
"""Tests for gen_stub_libs.py."""
|
||||||
|
import cStringIO
|
||||||
|
import textwrap
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
import gen_stub_libs as gsl
|
||||||
|
|
||||||
|
|
||||||
|
# pylint: disable=missing-docstring
|
||||||
|
|
||||||
|
|
||||||
|
class TagsTest(unittest.TestCase):
|
||||||
|
def test_get_tags_no_tags(self):
|
||||||
|
self.assertEqual([], gsl.get_tags(''))
|
||||||
|
self.assertEqual([], gsl.get_tags('foo bar baz'))
|
||||||
|
|
||||||
|
def test_get_tags(self):
|
||||||
|
self.assertEqual(['foo', 'bar'], gsl.get_tags('# foo bar'))
|
||||||
|
self.assertEqual(['bar', 'baz'], gsl.get_tags('foo # bar baz'))
|
||||||
|
|
||||||
|
def test_get_tag_value(self):
|
||||||
|
self.assertEqual('bar', gsl.get_tag_value('foo=bar'))
|
||||||
|
self.assertEqual('bar=baz', gsl.get_tag_value('foo=bar=baz'))
|
||||||
|
with self.assertRaises(ValueError):
|
||||||
|
gsl.get_tag_value('foo')
|
||||||
|
|
||||||
|
|
||||||
|
class PrivateVersionTest(unittest.TestCase):
|
||||||
|
def test_version_is_private(self):
|
||||||
|
self.assertFalse(gsl.version_is_private('foo'))
|
||||||
|
self.assertFalse(gsl.version_is_private('PRIVATE'))
|
||||||
|
self.assertFalse(gsl.version_is_private('PLATFORM'))
|
||||||
|
self.assertFalse(gsl.version_is_private('foo_private'))
|
||||||
|
self.assertFalse(gsl.version_is_private('foo_platform'))
|
||||||
|
self.assertFalse(gsl.version_is_private('foo_PRIVATE_'))
|
||||||
|
self.assertFalse(gsl.version_is_private('foo_PLATFORM_'))
|
||||||
|
|
||||||
|
self.assertTrue(gsl.version_is_private('foo_PRIVATE'))
|
||||||
|
self.assertTrue(gsl.version_is_private('foo_PLATFORM'))
|
||||||
|
|
||||||
|
|
||||||
|
class SymbolPresenceTest(unittest.TestCase):
|
||||||
|
def test_symbol_in_arch(self):
|
||||||
|
self.assertTrue(gsl.symbol_in_arch([], 'arm'))
|
||||||
|
self.assertTrue(gsl.symbol_in_arch(['arm'], 'arm'))
|
||||||
|
|
||||||
|
self.assertFalse(gsl.symbol_in_arch(['x86'], 'arm'))
|
||||||
|
|
||||||
|
def test_symbol_in_api(self):
|
||||||
|
self.assertTrue(gsl.symbol_in_api([], 'arm', 9))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(['introduced=9'], 'arm', 9))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(['introduced=9'], 'arm', 14))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(['introduced-arm=9'], 'arm', 14))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(['introduced-arm=9'], 'arm', 14))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(['introduced-x86=14'], 'arm', 9))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(
|
||||||
|
['introduced-arm=9', 'introduced-x86=21'], 'arm', 14))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(
|
||||||
|
['introduced=9', 'introduced-x86=21'], 'arm', 14))
|
||||||
|
self.assertTrue(gsl.symbol_in_api(
|
||||||
|
['introduced=21', 'introduced-arm=9'], 'arm', 14))
|
||||||
|
|
||||||
|
self.assertFalse(gsl.symbol_in_api(['introduced=14'], 'arm', 9))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(['introduced-arm=14'], 'arm', 9))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(['future'], 'arm', 9))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(
|
||||||
|
['introduced=9', 'future'], 'arm', 14))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(
|
||||||
|
['introduced-arm=9', 'future'], 'arm', 14))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(
|
||||||
|
['introduced-arm=21', 'introduced-x86=9'], 'arm', 14))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(
|
||||||
|
['introduced=9', 'introduced-arm=21'], 'arm', 14))
|
||||||
|
self.assertFalse(gsl.symbol_in_api(
|
||||||
|
['introduced=21', 'introduced-x86=9'], 'arm', 14))
|
||||||
|
|
||||||
|
# Interesting edge case: this symbol should be omitted from the
|
||||||
|
# library, but this call should still return true because none of the
|
||||||
|
# tags indiciate that it's not present in this API level.
|
||||||
|
self.assertTrue(gsl.symbol_in_api(['x86'], 'arm', 9))
|
||||||
|
|
||||||
|
def test_verioned_in_api(self):
|
||||||
|
self.assertTrue(gsl.symbol_versioned_in_api([], 9))
|
||||||
|
self.assertTrue(gsl.symbol_versioned_in_api(['versioned=9'], 9))
|
||||||
|
self.assertTrue(gsl.symbol_versioned_in_api(['versioned=9'], 14))
|
||||||
|
|
||||||
|
self.assertFalse(gsl.symbol_versioned_in_api(['versioned=14'], 9))
|
||||||
|
|
||||||
|
|
||||||
|
class OmitVersionTest(unittest.TestCase):
|
||||||
|
def test_omit_private(self):
|
||||||
|
self.assertFalse(gsl.should_omit_version('foo', [], 'arm', 9))
|
||||||
|
|
||||||
|
self.assertTrue(gsl.should_omit_version('foo_PRIVATE', [], 'arm', 9))
|
||||||
|
self.assertTrue(gsl.should_omit_version('foo_PLATFORM', [], 'arm', 9))
|
||||||
|
|
||||||
|
def test_omit_arch(self):
|
||||||
|
self.assertFalse(gsl.should_omit_version('foo', [], 'arm', 9))
|
||||||
|
self.assertFalse(gsl.should_omit_version('foo', ['arm'], 'arm', 9))
|
||||||
|
|
||||||
|
self.assertTrue(gsl.should_omit_version('foo', ['x86'], 'arm', 9))
|
||||||
|
|
||||||
|
def test_omit_api(self):
|
||||||
|
self.assertFalse(gsl.should_omit_version('foo', [], 'arm', 9))
|
||||||
|
self.assertFalse(
|
||||||
|
gsl.should_omit_version('foo', ['introduced=9'], 'arm', 9))
|
||||||
|
|
||||||
|
self.assertTrue(
|
||||||
|
gsl.should_omit_version('foo', ['introduced=14'], 'arm', 9))
|
||||||
|
|
||||||
|
|
||||||
|
class SymbolFileParseTest(unittest.TestCase):
|
||||||
|
def test_next_line(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
foo
|
||||||
|
|
||||||
|
bar
|
||||||
|
# baz
|
||||||
|
qux
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
self.assertIsNone(parser.current_line)
|
||||||
|
|
||||||
|
self.assertEqual('foo', parser.next_line().strip())
|
||||||
|
self.assertEqual('foo', parser.current_line.strip())
|
||||||
|
|
||||||
|
self.assertEqual('bar', parser.next_line().strip())
|
||||||
|
self.assertEqual('bar', parser.current_line.strip())
|
||||||
|
|
||||||
|
self.assertEqual('qux', parser.next_line().strip())
|
||||||
|
self.assertEqual('qux', parser.current_line.strip())
|
||||||
|
|
||||||
|
self.assertEqual('', parser.next_line())
|
||||||
|
self.assertEqual('', parser.current_line)
|
||||||
|
|
||||||
|
def test_parse_version(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 { # foo bar
|
||||||
|
baz;
|
||||||
|
qux; # woodly doodly
|
||||||
|
};
|
||||||
|
|
||||||
|
VERSION_2 {
|
||||||
|
} VERSION_1; # asdf
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
|
||||||
|
parser.next_line()
|
||||||
|
version = parser.parse_version()
|
||||||
|
self.assertEqual('VERSION_1', version.name)
|
||||||
|
self.assertIsNone(version.base)
|
||||||
|
self.assertEqual(['foo', 'bar'], version.tags)
|
||||||
|
|
||||||
|
expected_symbols = [
|
||||||
|
gsl.Symbol('baz', []),
|
||||||
|
gsl.Symbol('qux', ['woodly', 'doodly']),
|
||||||
|
]
|
||||||
|
self.assertEqual(expected_symbols, version.symbols)
|
||||||
|
|
||||||
|
parser.next_line()
|
||||||
|
version = parser.parse_version()
|
||||||
|
self.assertEqual('VERSION_2', version.name)
|
||||||
|
self.assertEqual('VERSION_1', version.base)
|
||||||
|
self.assertEqual([], version.tags)
|
||||||
|
|
||||||
|
def test_parse_version_eof(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
parser.next_line()
|
||||||
|
with self.assertRaises(gsl.ParseError):
|
||||||
|
parser.parse_version()
|
||||||
|
|
||||||
|
def test_unknown_scope_label(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
foo:
|
||||||
|
}
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
parser.next_line()
|
||||||
|
with self.assertRaises(gsl.ParseError):
|
||||||
|
parser.parse_version()
|
||||||
|
|
||||||
|
def test_parse_symbol(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
foo;
|
||||||
|
bar; # baz qux
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
|
||||||
|
parser.next_line()
|
||||||
|
symbol = parser.parse_symbol()
|
||||||
|
self.assertEqual('foo', symbol.name)
|
||||||
|
self.assertEqual([], symbol.tags)
|
||||||
|
|
||||||
|
parser.next_line()
|
||||||
|
symbol = parser.parse_symbol()
|
||||||
|
self.assertEqual('bar', symbol.name)
|
||||||
|
self.assertEqual(['baz', 'qux'], symbol.tags)
|
||||||
|
|
||||||
|
def test_wildcard_symbol_global(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
*;
|
||||||
|
};
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
parser.next_line()
|
||||||
|
with self.assertRaises(gsl.ParseError):
|
||||||
|
parser.parse_version()
|
||||||
|
|
||||||
|
def test_wildcard_symbol_local(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
local:
|
||||||
|
*;
|
||||||
|
};
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
parser.next_line()
|
||||||
|
version = parser.parse_version()
|
||||||
|
self.assertEqual([], version.symbols)
|
||||||
|
|
||||||
|
def test_missing_semicolon(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
foo
|
||||||
|
};
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
parser.next_line()
|
||||||
|
with self.assertRaises(gsl.ParseError):
|
||||||
|
parser.parse_version()
|
||||||
|
|
||||||
|
def test_parse_fails_invalid_input(self):
|
||||||
|
with self.assertRaises(gsl.ParseError):
|
||||||
|
input_file = cStringIO.StringIO('foo')
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
parser.parse()
|
||||||
|
|
||||||
|
def test_parse(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
local:
|
||||||
|
hidden1;
|
||||||
|
global:
|
||||||
|
foo;
|
||||||
|
bar; # baz
|
||||||
|
};
|
||||||
|
|
||||||
|
VERSION_2 { # wasd
|
||||||
|
# Implicit global scope.
|
||||||
|
woodly;
|
||||||
|
doodly; # asdf
|
||||||
|
local:
|
||||||
|
qwerty;
|
||||||
|
} VERSION_1;
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
versions = parser.parse()
|
||||||
|
|
||||||
|
expected = [
|
||||||
|
gsl.Version('VERSION_1', None, [], [
|
||||||
|
gsl.Symbol('foo', []),
|
||||||
|
gsl.Symbol('bar', ['baz']),
|
||||||
|
]),
|
||||||
|
gsl.Version('VERSION_2', 'VERSION_1', ['wasd'], [
|
||||||
|
gsl.Symbol('woodly', []),
|
||||||
|
gsl.Symbol('doodly', ['asdf']),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
|
||||||
|
self.assertEqual(expected, versions)
|
||||||
|
|
||||||
|
|
||||||
|
class GeneratorTest(unittest.TestCase):
|
||||||
|
def test_omit_version(self):
|
||||||
|
# Thorough testing of the cases involved here is handled by
|
||||||
|
# OmitVersionTest, PrivateVersionTest, and SymbolPresenceTest.
|
||||||
|
src_file = cStringIO.StringIO()
|
||||||
|
version_file = cStringIO.StringIO()
|
||||||
|
generator = gsl.Generator(src_file, version_file, 'arm', 9)
|
||||||
|
|
||||||
|
version = gsl.Version('VERSION_PRIVATE', None, [], [
|
||||||
|
gsl.Symbol('foo', []),
|
||||||
|
])
|
||||||
|
generator.write_version(version)
|
||||||
|
self.assertEqual('', src_file.getvalue())
|
||||||
|
self.assertEqual('', version_file.getvalue())
|
||||||
|
|
||||||
|
version = gsl.Version('VERSION', None, ['x86'], [
|
||||||
|
gsl.Symbol('foo', []),
|
||||||
|
])
|
||||||
|
generator.write_version(version)
|
||||||
|
self.assertEqual('', src_file.getvalue())
|
||||||
|
self.assertEqual('', version_file.getvalue())
|
||||||
|
|
||||||
|
version = gsl.Version('VERSION', None, ['introduced=14'], [
|
||||||
|
gsl.Symbol('foo', []),
|
||||||
|
])
|
||||||
|
generator.write_version(version)
|
||||||
|
self.assertEqual('', src_file.getvalue())
|
||||||
|
self.assertEqual('', version_file.getvalue())
|
||||||
|
|
||||||
|
def test_omit_symbol(self):
|
||||||
|
# Thorough testing of the cases involved here is handled by
|
||||||
|
# SymbolPresenceTest.
|
||||||
|
src_file = cStringIO.StringIO()
|
||||||
|
version_file = cStringIO.StringIO()
|
||||||
|
generator = gsl.Generator(src_file, version_file, 'arm', 9)
|
||||||
|
|
||||||
|
version = gsl.Version('VERSION_1', None, [], [
|
||||||
|
gsl.Symbol('foo', ['x86']),
|
||||||
|
])
|
||||||
|
generator.write_version(version)
|
||||||
|
self.assertEqual('', src_file.getvalue())
|
||||||
|
self.assertEqual('', version_file.getvalue())
|
||||||
|
|
||||||
|
version = gsl.Version('VERSION_1', None, [], [
|
||||||
|
gsl.Symbol('foo', ['introduced=14']),
|
||||||
|
])
|
||||||
|
generator.write_version(version)
|
||||||
|
self.assertEqual('', src_file.getvalue())
|
||||||
|
self.assertEqual('', version_file.getvalue())
|
||||||
|
|
||||||
|
def test_write(self):
|
||||||
|
src_file = cStringIO.StringIO()
|
||||||
|
version_file = cStringIO.StringIO()
|
||||||
|
generator = gsl.Generator(src_file, version_file, 'arm', 9)
|
||||||
|
|
||||||
|
versions = [
|
||||||
|
gsl.Version('VERSION_1', None, [], [
|
||||||
|
gsl.Symbol('foo', []),
|
||||||
|
gsl.Symbol('bar', ['var']),
|
||||||
|
]),
|
||||||
|
gsl.Version('VERSION_2', 'VERSION_1', [], [
|
||||||
|
gsl.Symbol('baz', []),
|
||||||
|
]),
|
||||||
|
gsl.Version('VERSION_3', 'VERSION_1', [], [
|
||||||
|
gsl.Symbol('qux', ['versioned=14']),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
|
||||||
|
generator.write(versions)
|
||||||
|
expected_src = textwrap.dedent("""\
|
||||||
|
void foo() {}
|
||||||
|
int bar = 0;
|
||||||
|
void baz() {}
|
||||||
|
void qux() {}
|
||||||
|
""")
|
||||||
|
self.assertEqual(expected_src, src_file.getvalue())
|
||||||
|
|
||||||
|
expected_version = textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
global:
|
||||||
|
foo;
|
||||||
|
bar;
|
||||||
|
};
|
||||||
|
VERSION_2 {
|
||||||
|
global:
|
||||||
|
baz;
|
||||||
|
} VERSION_1;
|
||||||
|
""")
|
||||||
|
self.assertEqual(expected_version, version_file.getvalue())
|
||||||
|
|
||||||
|
|
||||||
|
class IntegrationTest(unittest.TestCase):
|
||||||
|
def test_integration(self):
|
||||||
|
input_file = cStringIO.StringIO(textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
global:
|
||||||
|
foo; # var
|
||||||
|
bar; # x86
|
||||||
|
local:
|
||||||
|
*;
|
||||||
|
};
|
||||||
|
|
||||||
|
VERSION_2 { # arm
|
||||||
|
baz; # introduced=9
|
||||||
|
qux; # versioned=14
|
||||||
|
} VERSION_1;
|
||||||
|
|
||||||
|
VERSION_3 { # introduced=14
|
||||||
|
woodly;
|
||||||
|
doodly; # var
|
||||||
|
} VERSION_2;
|
||||||
|
"""))
|
||||||
|
parser = gsl.SymbolFileParser(input_file)
|
||||||
|
versions = parser.parse()
|
||||||
|
|
||||||
|
src_file = cStringIO.StringIO()
|
||||||
|
version_file = cStringIO.StringIO()
|
||||||
|
generator = gsl.Generator(src_file, version_file, 'arm', 9)
|
||||||
|
generator.write(versions)
|
||||||
|
|
||||||
|
expected_src = textwrap.dedent("""\
|
||||||
|
int foo = 0;
|
||||||
|
void baz() {}
|
||||||
|
void qux() {}
|
||||||
|
""")
|
||||||
|
self.assertEqual(expected_src, src_file.getvalue())
|
||||||
|
|
||||||
|
expected_version = textwrap.dedent("""\
|
||||||
|
VERSION_1 {
|
||||||
|
global:
|
||||||
|
foo;
|
||||||
|
};
|
||||||
|
VERSION_2 {
|
||||||
|
global:
|
||||||
|
baz;
|
||||||
|
} VERSION_1;
|
||||||
|
""")
|
||||||
|
self.assertEqual(expected_version, version_file.getvalue())
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
suite = unittest.TestLoader().loadTestsFromName(__name__)
|
||||||
|
unittest.TextTestRunner(verbosity=3).run(suite)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
Reference in New Issue
Block a user