| #!/usr/bin/env python3 |
| # |
| # Copyright 2020 the V8 project authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| from gen_cmake import CMakeBuilder, V8GNTransformer, ParseGN, V8_TARGET_TYPES |
| import unittest |
| |
| |
| class CMakeMockBuilder(CMakeBuilder): |
| """ |
| Similar to CMakeBuilder but doesn't produce prologues/epilogues. |
| """ |
| def BuildPrologue(self): |
| pass |
| |
| def BuildEpilogue(self): |
| pass |
| |
| |
| class CMakeGenerationTest(unittest.TestCase): |
| TARGET = 'cppgc_base' |
| CMAKE_TARGET_SOURCES = TARGET.upper() + '_SOURCES' |
| |
| def test_source_assignment(self): |
| self._CompileAndCheck( |
| f'set({self.CMAKE_TARGET_SOURCES} "source1.h" "source1.cc")', |
| 'sources = [ "source1.h", "source1.cc", ]') |
| |
| def test_source_append(self): |
| self._CompileAndCheck( |
| f'list(APPEND {self.CMAKE_TARGET_SOURCES} "source1.h" "source1.cc")', |
| 'sources += [ "source1.h", "source1.cc", ]') |
| |
| def test_source_remove(self): |
| self._CompileAndCheck( |
| f'list(REMOVE_ITEM {self.CMAKE_TARGET_SOURCES} "source1.h" "source1.cc")', |
| 'sources -= [ "source1.h", "source1.cc", ]') |
| |
| def test_equal(self): |
| self._CompileExpressionAndCheck('"${CURRENT_CPU}" STREQUAL "x64"', |
| 'current_cpu == "x64"') |
| |
| def test_not_equal(self): |
| self._CompileExpressionAndCheck('NOT "${CURRENT_CPU}" STREQUAL "x86"', |
| 'current_cpu != "x86"') |
| |
| def test_comparison_ops(self): |
| OPS = { |
| '<': 'LESS', |
| '<=': 'LESS_EQUAL', |
| '>': 'GREATER', |
| '>=': 'GREATER_EQUAL', |
| } |
| for gn_op, cmake_op in OPS.items(): |
| self._CompileExpressionAndCheck( |
| f'"${{GCC_VERSION}}" {cmake_op} 40802', |
| f'gcc_version {gn_op} 40802') |
| |
| def test_parenthesized_expressions(self): |
| self._CompileExpressionAndCheck( |
| '(("${IS_POSIX}" AND NOT "${IS_ANDROID}") OR "${IS_FUCHSIA}") AND NOT "${USING_SANITIZER}"', |
| '((is_posix && !is_android) || is_fuchsia) && !using_sanitizer') |
| |
| def test_conditional_statements(self): |
| self._CompileAndCheck( |
| f""" |
| if("${{IS_POSIX}}") |
| list(APPEND {self.CMAKE_TARGET_SOURCES} "unistd.h") |
| else() |
| list(REMOVE_ITEM {self.CMAKE_TARGET_SOURCES} "unistd.h") |
| endif() |
| """, """ |
| if (is_posix) { |
| sources += ["unistd.h"] |
| } else { |
| sources -= ["unistd.h"] |
| } |
| """) |
| |
| def test_conditional_statements_elseif(self): |
| self._CompileAndCheck( |
| f""" |
| if("${{IS_POSIX}}") |
| list(APPEND {self.CMAKE_TARGET_SOURCES} "unistd.h") |
| elseif("${{IS_WIN}}") |
| list(REMOVE_ITEM {self.CMAKE_TARGET_SOURCES} "unistd.h") |
| endif() |
| """, """ |
| if (is_posix) { |
| sources += ["unistd.h"] |
| } else if (is_win) { |
| sources -= ["unistd.h"] |
| } |
| """) |
| |
| def _Compile(self, gn_string): |
| gn_code = f'v8_component({self.TARGET}) {{ {gn_string} }}' |
| tree = ParseGN(gn_code) |
| builder = CMakeMockBuilder() |
| V8GNTransformer(builder, [self.TARGET]).Traverse(tree) |
| return builder.GetResult() |
| |
| def _CompileAndCheck(self, expected_cmake, gn_string): |
| actual_cmake = self._Compile(gn_string) |
| self.assertIn(self._Canonicalize(expected_cmake), |
| self._Canonicalize(actual_cmake)) |
| pass |
| |
| def _CompileExpressionAndCheck(self, expected_cmake, gn_string): |
| gn_string = f'if ({gn_string}) {{ sources = [ "source.cc" ] }}' |
| expected_cmake = f'if({expected_cmake})' |
| actual_cmake = self._Compile(gn_string) |
| self.assertIn(self._Canonicalize(expected_cmake), |
| self._Canonicalize(actual_cmake)) |
| pass |
| |
| @staticmethod |
| def _Canonicalize(str): |
| return ' '.join(str.split()).strip() |
| |
| |
| if __name__ == '__main__': |
| unittest.main() |