check_non_test_cc_deps.py 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. #!/usr/bin/env python3
  2. """Check that non-test C++ rules only depend on Carbon and LLVM.
  3. Carbon works to ensure its user-visible libraries and binaries only depend on
  4. their code and LLVM. Among other benefits, this provides a single, simple
  5. license used for the whole project.
  6. However, we frequently use third-party projects and libraries where useful in
  7. our test code. Here, we verify that the dependencies of non-test C++ rules only
  8. include Carbon and LLVM code.
  9. """
  10. __copyright__ = """
  11. Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  12. Exceptions. See /LICENSE for license information.
  13. SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  14. """
  15. import os
  16. import sys
  17. from pathlib import Path
  18. runfiles = Path(os.environ["TEST_SRCDIR"])
  19. deps_path = runfiles / "_main" / "bazel" / "check_deps" / "non_test_cc_deps.txt"
  20. try:
  21. with deps_path.open() as deps_file:
  22. deps = deps_file.read().splitlines()
  23. except FileNotFoundError:
  24. sys.exit("ERROR: unable to find deps file: %s" % deps_path)
  25. # bazel_dep repos can have a `name~version` format, whereas use_repo and
  26. # use_repo_rule have a `@@_main~rule_name~repo_name` format. We process the
  27. # latter case first because we only have a couple; once done, we can assume
  28. # anything after the ~ is a version.
  29. for dep in deps:
  30. print("Checking dependency: " + dep)
  31. repo, _, rule = dep.partition("//")
  32. if repo == "@@_main~llvm_project~llvm-project":
  33. package, _, rule = rule.partition(":")
  34. # Other packages in the LLVM project shouldn't be accidentally used
  35. # in Carbon. We can expand the above list if use cases emerge.
  36. if package not in (
  37. "llvm",
  38. "lld",
  39. "clang",
  40. "clang-tools-extra/clangd",
  41. "libunwind",
  42. ):
  43. sys.exit(
  44. "ERROR: unexpected dependency into the LLVM project: %s" % dep
  45. )
  46. # Check for accidentally using the copy of GoogleTest in LLVM.
  47. if rule in ("gmock", "gtest", "gtest_main"):
  48. sys.exit(
  49. "ERROR: dependency on LLVM's GoogleTest from non-test code: %s"
  50. % dep
  51. )
  52. # The rest of LLVM, LLD, and Clang themselves are safe to depend on.
  53. continue
  54. # Ignore the version, just use the repo name.
  55. repo_base = repo.split("~")[0]
  56. # Carbon code is always allowed.
  57. if repo_base == "" and not rule.startswith("third_party"):
  58. continue
  59. # An empty stub library added by rules_cc:
  60. # https://github.com/bazelbuild/rules_cc/blob/main/BUILD
  61. if repo_base == "@@rules_cc" and rule == ":link_extra_lib":
  62. continue
  63. # An utility library provided by Bazel that is under a compatible license.
  64. if repo_base == "@@bazel_tools" and rule == "tools/cpp/runfiles:runfiles":
  65. continue
  66. # These are stubs wrapping system libraries for LLVM. They aren't
  67. # distributed and so should be fine.
  68. if repo_base in (
  69. "@@zlib",
  70. "@@zstd",
  71. ):
  72. continue
  73. # This should never be reached from non-test code, but these targets do
  74. # exist. Specially diagnose them to try to provide a more helpful
  75. # message.
  76. if repo_base in (
  77. "@google_benchmark",
  78. "@protobuf",
  79. "@abseil-cpp",
  80. "@googletest",
  81. ):
  82. sys.exit("ERROR: dependency only allowed in test code: %s" % dep)
  83. # Conservatively fail if a dependency isn't explicitly allowed above.
  84. sys.exit("ERROR: unknown dependency: %s" % dep)