chromium/media/ffmpeg/scripts/generate_gn_unittest.py

#!/usr/bin/env python3
#
# Copyright 2011 The Chromium Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Unittest for generate_gn.py.

It's tough to test the lower-level GetSourceFiles() and GetObjectFiles()
functions, so this focuses on the higher-level functions assuming those two
functions are working as intended (i.e., producing lists of files).
"""

import generate_gn as gg
from generate_gn import SourceSet, SourceListCondition
import unittest
from os import path


class ModuleUnittest(unittest.TestCase):

    def testGetObjectToSourceMapping(self):
        srcs = [
            'a.c',
            'b.asm',
            'c.cc',
        ]
        expected = {
            'a.o': 'a.c',
            'b.o': 'b.asm',
            'c.o': 'c.cc',
        }
        self.assertEqual(expected, gg.GetObjectToSourceMapping(srcs))

    def testGetSourceFileSet(self):
        objs_to_srcs = {
            'a.o': 'a.c',
            'b.o': 'b.asm',
            'c.o': 'c.cc',
        }
        objs = [
            'a.o',
            'c.o',
        ]
        expected = set(['a.c', 'c.cc'])
        self.assertEqual(expected, gg.GetSourceFileSet(objs_to_srcs, objs))

    def testGetSourceFileSet_NotFound(self):
        objs_to_srcs = {
            'a.o': 'a.c',
            'b.o': 'b.asm',
            'c.o': 'c.cc',
        }
        objs = [
            'd.o',
        ]
        self.assertRaises(KeyError, gg.GetSourceFileSet, objs_to_srcs, objs)


class SourceSetUnittest(unittest.TestCase):

    def testEquals(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        b = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        c = SourceSet(set(['c', 'd']),
                      set([SourceListCondition('1', '2', '3')]))
        d = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('0', '2', '3')]))
        e = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '0', '3')]))
        f = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '0')]))

        self.assertEqual(a, b)
        self.assertNotEqual(a, c)
        self.assertNotEqual(a, d)
        self.assertNotEqual(a, e)
        self.assertNotEqual(a, f)

    def testIntersect_Exact(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        b = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('3', '4', '6')]))

        c = a.Intersect(b)

        self.assertEqual(c.sources, set(['a', 'b']))
        self.assertEqual(
            c.conditions,
            set([
                SourceListCondition('1', '2', '3'),
                SourceListCondition('3', '4', '6')
            ]))
        self.assertFalse(c.IsEmpty())

    def testIntersect_Disjoint(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        b = SourceSet(set(['c', 'd']),
                      set([SourceListCondition('3', '4', '6')]))

        c = a.Intersect(b)

        self.assertEqual(c.sources, set())
        self.assertEqual(
            c.conditions,
            set([
                SourceListCondition('1', '2', '3'),
                SourceListCondition('3', '4', '6')
            ]))
        self.assertTrue(c.IsEmpty())

    def testIntersect_Overlap(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        b = SourceSet(set(['b', 'c']),
                      set([SourceListCondition('3', '4', '6')]))

        c = a.Intersect(b)

        self.assertEqual(c.sources, set(['b']))
        self.assertEqual(
            c.conditions,
            set([
                SourceListCondition('1', '2', '3'),
                SourceListCondition('3', '4', '6')
            ]))
        self.assertFalse(c.IsEmpty())

    def testDifference_Exact(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        b = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))

        c = a.Difference(b)

        self.assertEqual(c.sources, set())
        self.assertEqual(c.conditions,
                         set([SourceListCondition('1', '2', '3')]))
        self.assertTrue(c.IsEmpty())

    def testDifference_Disjoint(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '3')]))
        b = SourceSet(set(['c', 'd']),
                      set([SourceListCondition('3', '4', '6')]))

        c = a.Difference(b)

        self.assertEqual(c.sources, set(['a', 'b']))
        self.assertEqual(c.conditions, set())
        self.assertTrue(c.IsEmpty())

    def testDifference_Overlap(self):
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('1', '2', '5')]))
        b = SourceSet(
            set(['b', 'c', 'd']),
            set([
                SourceListCondition('1', '2', '5'),
                SourceListCondition('3', '4', '6')
            ]))

        c = a.Difference(b)

        self.assertEqual(c.sources, set(['a']))
        self.assertEqual(c.conditions,
                         set([SourceListCondition('1', '2', '5')]))
        self.assertFalse(c.IsEmpty())

    def testGenerateGnStanza(self):
        # ia32 should be x86.  Win should appear as an OS restriction.
        a = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('ia32', 'Chromium', 'win')]))
        a_stanza = a.GenerateGnStanza()
        a_stanza.index('current_cpu == "x86"')
        a_stanza.index('is_win')

        # x64 should just be x64.  Linux should appear as an OS restriction.
        b = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('x64', 'Chromium', 'linux')]))
        b_stanza = b.GenerateGnStanza()
        b_stanza.index('current_cpu == "x64"')
        b_stanza.index('use_linux_config')

        # arm should just be arm.
        c = SourceSet(set(['a', 'b']),
                      set([SourceListCondition('arm', 'Chromium', 'linux')]))
        c_stanza = c.GenerateGnStanza()
        c_stanza.index('current_cpu == "arm"')

        # arm-neon should be arm and flip the arm_neon switch.
        d = SourceSet(
            set(['a', 'b']),
            set([SourceListCondition('arm-neon', 'Chromium', 'linux')]))
        d_stanza = d.GenerateGnStanza()
        d_stanza.index('current_cpu == "arm" && arm_use_neon')

        # Multiple conditions
        e = SourceSet(
            set(['a', 'b']),
            set([
                SourceListCondition('arm', 'Chrome', 'win'),
                SourceListCondition('x64', 'Chromium', 'linux')
            ]))
        e_stanza = e.GenerateGnStanza()
        e_stanza.index(('is_win && current_cpu == "arm"'
                        ' && ffmpeg_branding == "Chrome"'))
        e_stanza.index(('use_linux_config && current_cpu == "x64"'
                        ' && ffmpeg_branding == "Chromium"'))

        # mac should imply is_apple.
        f = SourceSet(set(['a']),
                      set([SourceListCondition('arm64', 'Chromium', 'mac')]))
        f_stanza = f.GenerateGnStanza()
        f_stanza.index('is_apple')

    def testComplexSourceListConditions(self):
        # Create 2 sets with intersecting source 'a', but setup such that 'a'
        # is only valid for combinations (x86 && windows) || (x64 && linux). The
        # generated gn stanza should then not allow for inclusion of the 'a' file
        # for combinations like x86 && linux.
        a = SourceSet(set(['a']), set([SourceListCondition('x86', 'c',
                                                           'win')]))
        b = SourceSet(set(['a']),
                      set([SourceListCondition('x64', 'c', 'linux')]))
        disjoint_sets = gg.CreatePairwiseDisjointSets([a, b])

        # This condition is bad because x86 && linux would pass. Admittedly a very
        # fragile way to test this, but evaluating gn stanzas is hard, and it at
        # least serves to document the motivation for the associated changes to
        # our generate_gn.py
        bad_condition = ('(current_cpu == "x86" || current_cpu == "x64")'
                         ' && (ffmpeg_branding == "c")'
                         ' && (is_win || is_linux || is_chromeos)')

        # Expect only a single set since the two original sets have the same source
        # list.
        self.assertEqual(1, len(disjoint_sets))

        stanza = disjoint_sets[0].GenerateGnStanza()
        self.assertEqual(stanza.find(bad_condition), -1)

    def assertEqualSourceSets(self, expected, actual):
        assert all(isinstance(a, SourceSet) for a in expected)
        assert all(isinstance(a, SourceSet) for a in actual)

        def SourceSetToString(source_set):
            sources = [str(e) for e in source_set.sources]
            conditions = [str(e) for e in source_set.conditions]
            sources_str = ','.join(sources)
            conditions_str = '\n\t'.join(conditions)
            return '  sources:%s\n  cs:\t%s' % (sources_str, conditions_str)

        missing_elements = expected.difference(actual)
        extra_elements = actual.difference(expected)
        msg = ''
        if len(missing_elements):
            msg += 'Missing expected elements:\n'
            for e in missing_elements:
                msg += SourceSetToString(e) + '\n'
        if len(extra_elements):
            msg += 'Found extra elements:\n'
            for e in extra_elements:
                msg += SourceSetToString(e) + '\n'

        self.assertTrue(expected == actual, msg=msg)

    def testCreatePairwiseDisjointSets_Pair(self):
        a = SourceSet(set(['common', 'intel']),
                      set([SourceListCondition('ia32', 'Chromium', 'win')]))
        b = SourceSet(set(['common', 'intel', 'chrome']),
                      set([SourceListCondition('ia32', 'Chrome', 'win')]))

        expected = set()
        expected.add(
            SourceSet(
                set(['common', 'intel']),
                set([
                    SourceListCondition('ia32', 'Chromium', 'win'),
                    SourceListCondition('ia32', 'Chrome', 'win')
                ])))
        expected.add(
            SourceSet(set(['chrome']),
                      set([SourceListCondition('ia32', 'Chrome', 'win')])))

        source_sets = gg.CreatePairwiseDisjointSets([a, b])
        self.assertEqualSourceSets(expected, set(source_sets))

    def testCreatePairwiseDisjointSets_Triplet(self):
        a = SourceSet(set(['common', 'intel']),
                      set([SourceListCondition('ia32', 'Chromium', 'win')]))
        b = SourceSet(set(['common', 'intel', 'chrome']),
                      set([SourceListCondition('x64', 'Chrome', 'win')]))
        c = SourceSet(set(['common', 'arm']),
                      set([SourceListCondition('arm', 'Chromium', 'win')]))

        expected = set()
        expected.add(
            SourceSet(
                set(['common']),
                set([
                    SourceListCondition('ia32', 'Chromium', 'win'),
                    SourceListCondition('x64', 'Chrome', 'win'),
                    SourceListCondition('arm', 'Chromium', 'win')
                ])))
        expected.add(
            SourceSet(
                set(['intel']),
                set([
                    SourceListCondition('ia32', 'Chromium', 'win'),
                    SourceListCondition('x64', 'Chrome', 'win')
                ])))
        expected.add(
            SourceSet(set(['chrome']),
                      set([SourceListCondition('x64', 'Chrome', 'win')])))
        expected.add(
            SourceSet(set(['arm']),
                      set([SourceListCondition('arm', 'Chromium', 'win')])))

        source_sets = gg.CreatePairwiseDisjointSets([a, b, c])
        self.assertEqualSourceSets(expected, set(source_sets))

    def testCreatePairwiseDisjointSets_Multiple(self):
        a = SourceSet(set(['common', 'intel']),
                      set([SourceListCondition('ia32', 'Chromium', 'linux')]))
        b = SourceSet(set(['common', 'intel', 'chrome']),
                      set([SourceListCondition('ia32', 'Chrome', 'linux')]))
        c = SourceSet(set(['common', 'intel']),
                      set([SourceListCondition('x64', 'Chromium', 'linux')]))
        d = SourceSet(set(['common', 'intel', 'chrome']),
                      set([SourceListCondition('x64', 'Chrome', 'linux')]))
        e = SourceSet(set(['common', 'arm']),
                      set([SourceListCondition('arm', 'Chromium', 'linux')]))
        f = SourceSet(
            set(['common', 'arm-neon', 'chrome']),
            set([SourceListCondition('arm-neon', 'Chrome', 'linux')]))

        expected = set()
        expected.add(
            SourceSet(
                set(['common']),
                set([
                    SourceListCondition('ia32', 'Chromium', 'linux'),
                    SourceListCondition('ia32', 'Chrome', 'linux'),
                    SourceListCondition('x64', 'Chromium', 'linux'),
                    SourceListCondition('x64', 'Chrome', 'linux'),
                    SourceListCondition('arm', 'Chromium', 'linux'),
                    SourceListCondition('arm-neon', 'Chrome', 'linux')
                ])))
        expected.add(
            SourceSet(
                set(['intel']),
                set([
                    SourceListCondition('ia32', 'Chromium', 'linux'),
                    SourceListCondition('ia32', 'Chrome', 'linux'),
                    SourceListCondition('x64', 'Chromium', 'linux'),
                    SourceListCondition('x64', 'Chrome', 'linux')
                ])))
        expected.add(
            SourceSet(set(['arm']),
                      set([SourceListCondition('arm', 'Chromium', 'linux')])))
        expected.add(
            SourceSet(
                set(['chrome']),
                set([
                    SourceListCondition('ia32', 'Chrome', 'linux'),
                    SourceListCondition('x64', 'Chrome', 'linux'),
                    SourceListCondition('arm-neon', 'Chrome', 'linux')
                ])))
        expected.add(
            SourceSet(
                set(['arm-neon']),
                set([SourceListCondition('arm-neon', 'Chrome', 'linux')])))
        source_sets = gg.CreatePairwiseDisjointSets([a, b, c, d, e, f])
        self.assertEqualSourceSets(expected, set(source_sets))

    def testReduceConditions(self):
        # Set conditions span all of the supported architectures for linux.
        a = SourceSet(
            set(['foo.c']),
            set([
                SourceListCondition('ia32', 'Chromium', 'linux'),
                SourceListCondition('x64', 'Chromium', 'linux'),
                SourceListCondition('arm', 'Chromium', 'linux'),
                SourceListCondition('arm64', 'Chromium', 'linux'),
                SourceListCondition('arm-neon', 'Chromium', 'linux'),
            ]))
        gg.ReduceConditionalLogic(a)

        # Conditions should reduce to a single condition with wild-card for arch.
        expected = set([SourceListCondition('*', 'Chromium', 'linux')])
        self.assertEqual(expected, a.conditions)

        # Set conditions span all of the supported architectures for windows.
        b = SourceSet(
            set(['foo.c']),
            set([
                SourceListCondition('ia32', 'Chromium', 'win'),
                SourceListCondition('x64', 'Chromium', 'win'),
                SourceListCondition('arm64', 'Chromium', 'win'),
            ]))
        gg.ReduceConditionalLogic(b)

        # Conditions should reduce to a single condition with wild-card for
        expected = set([SourceListCondition('*', 'Chromium', 'win')])
        self.assertEqual(expected, b.conditions)

        # Set conditions span all supported architectures and brandings for windows.
        b = SourceSet(
            set(['foo.c']),
            set([
                SourceListCondition('ia32', 'Chromium', 'win'),
                SourceListCondition('x64', 'Chromium', 'win'),
                SourceListCondition('arm64', 'Chromium', 'win'),
                SourceListCondition('ia32', 'Chrome', 'win'),
                SourceListCondition('x64', 'Chrome', 'win'),
                SourceListCondition('arm64', 'Chrome', 'win'),
            ]))
        gg.ReduceConditionalLogic(b)
        expected = set([SourceListCondition('*', '*', 'win')])
        self.assertEqual(expected, b.conditions)

        # Set conditions span all supported platforms.
        c = SourceSet(
            set(['foo.c']),
            set([
                SourceListCondition('x64', 'Chromium', 'win'),
                SourceListCondition('x64', 'Chromium', 'mac'),
                SourceListCondition('x64', 'Chromium', 'linux'),
                SourceListCondition('x64', 'Chromium', 'android'),
            ]))
        gg.ReduceConditionalLogic(c)
        expected = set([SourceListCondition('x64', 'Chromium', '*')])
        self.assertEqual(expected, c.conditions)

        # Spans all architectures for Chromium, but also all targets for ia32 & win.
        d = SourceSet(
            set(['foo.c']),
            set([
                SourceListCondition('arm64', 'Chromium', 'win'),
                SourceListCondition('x64', 'Chromium', 'win'),
                SourceListCondition('ia32', 'Chromium', 'win'),
                SourceListCondition('ia32', 'Chrome', 'win'),
            ]))
        gg.ReduceConditionalLogic(d)
        expected = set([
            SourceListCondition('*', 'Chromium', 'win'),
            SourceListCondition('ia32', '*', 'win'),
        ])
        self.assertEqual(expected, d.conditions)

    def testReduceConditions_fullSpan(self):
        # Build SourceSet with conditions spanning every combination of attributes.
        ss = SourceSet(set(['foo.c']), set())
        for arch in gg.SUPPORT_MATRIX[gg.Attr.ARCHITECTURE]:
            for target in gg.SUPPORT_MATRIX[gg.Attr.TARGET]:
                for platform in gg.SUPPORT_MATRIX[gg.Attr.PLATFORM]:
                    ss.conditions.add(
                        SourceListCondition(arch, target, platform))

        gg.ReduceConditionalLogic(ss)
        expected = set([SourceListCondition('*', '*', '*')])
        self.assertEqual(expected, ss.conditions)

    def testGenerateStanzaWildCard(self):
        a = SourceSet(set(['foo.c']),
                      set([SourceListCondition('x64', 'Chromium', '*')]))
        stanza = a.GenerateGnStanza()
        stanza.index('== "x64"')
        stanza.index('ffmpeg_branding == "Chromium"')
        # OS is wild-card, so it should not be mentioned in the stanza.
        self.assertEqual(-1, stanza.find('OS =='))

    def testFixObjectBasenameCollisions(self):
        # Use callback to capture executed renames.
        observed_renames = set()

        def do_rename_cb(old_path, new_path, content):
            observed_renames.add((old_path, new_path))

        # Verify basic rename case - same basename in different directories.
        a = SourceSet(set(['foo.c']), set([SourceListCondition('*', '*',
                                                               '*')]))
        b = SourceSet(set([path.join('a', 'foo.c'),
                           path.join('b', 'foo.c')]),
                      set([SourceListCondition('*', '*', '*')]))
        expected_renames = set([
            (path.join('a', 'foo.c'), path.join('a', 'autorename_a_foo.c')),
            (path.join('b', 'foo.c'), path.join('b', 'autorename_b_foo.c'))
        ])
        gg.FixObjectBasenameCollisions([a, b], [],
                                       do_rename_cb,
                                       log_renames=False)
        self.assertEqual(expected_renames, observed_renames)

        # Verify renames file extensions in same and different directory.
        observed_renames = set()
        a = SourceSet(set(['foo.c']), set([SourceListCondition('*', '*',
                                                               '*')]))
        b = SourceSet(set(['foo.asm']),
                      set([SourceListCondition('*', '*', '*')]))
        c = SourceSet(
            set([path.join('a', 'foo.S'),
                 path.join('b', 'foo.asm')]),
            set([SourceListCondition('*', '*', '*')]))
        expected_renames = set([
            ('foo.asm', 'autorename_foo.asm'),
            (path.join('a', 'foo.S'), path.join('a', 'autorename_a_foo.S')),
            (path.join('b', 'foo.asm'), path.join('b', 'autorename_b_foo.asm'))
        ])
        gg.FixObjectBasenameCollisions([a, b, c], [],
                                       do_rename_cb,
                                       log_renames=False)
        self.assertEqual(expected_renames, observed_renames)


if __name__ == '__main__':
    unittest.main()