guideline_check.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. #!/usr/bin/env python3
  2. # SPDX-License-Identifier: Apache-2.0
  3. # Copyright (c) 2021 Intel Corporation
  4. import os
  5. import sh
  6. import argparse
  7. import re
  8. from unidiff import PatchSet
  9. if "ZEPHYR_BASE" not in os.environ:
  10. exit("$ZEPHYR_BASE environment variable undefined.")
  11. repository_path = os.environ['ZEPHYR_BASE']
  12. sh_special_args = {
  13. '_tty_out': False,
  14. '_cwd': repository_path
  15. }
  16. coccinelle_scripts = ["/scripts/coccinelle/reserved_names.cocci",
  17. "/scripts/coccinelle/same_identifier.cocci",
  18. #"/scripts/coccinelle/identifier_length.cocci",
  19. ]
  20. def parse_coccinelle(contents: str, violations: dict):
  21. reg = re.compile("([a-zA-Z0-9_/]*\\.[ch]:[0-9]*)(:[0-9\\-]*: )(.*)")
  22. for line in contents.split("\n"):
  23. r = reg.match(line)
  24. if r:
  25. f = r.group(1)
  26. if f in violations:
  27. violations[f].append(r.group(3))
  28. else:
  29. violations[r.group(1)] = [r.group(3)]
  30. def parse_args():
  31. parser = argparse.ArgumentParser(
  32. description="Check if change requires full twister")
  33. parser.add_argument('-c', '--commits', default=None,
  34. help="Commit range in the form: a..b")
  35. parser.add_argument("-o", "--output", required=False,
  36. help="Print violation into a file")
  37. return parser.parse_args()
  38. def main():
  39. args = parse_args()
  40. if not args.commits:
  41. exit("missing commit range")
  42. # pylint does not like the 'sh' library
  43. # pylint: disable=too-many-function-args,unexpected-keyword-arg
  44. commit = sh.git("diff", args.commits, **sh_special_args)
  45. patch_set = PatchSet(commit)
  46. zephyr_base = os.getenv("ZEPHYR_BASE")
  47. violations = {}
  48. numViolations = 0
  49. for f in patch_set:
  50. if not f.path.endswith(".c") and not f.path.endswith(".h") or not os.path.exists(zephyr_base + "/" + f.path):
  51. continue
  52. for script in coccinelle_scripts:
  53. script_path = os.getenv("ZEPHYR_BASE") + "/" + script
  54. print(f"Running {script} on {f.path}")
  55. try:
  56. cocci = sh.coccicheck(
  57. "--mode=report",
  58. "--cocci=" +
  59. script_path,
  60. f.path,
  61. _timeout=10,
  62. **sh_special_args)
  63. parse_coccinelle(cocci, violations)
  64. except sh.TimeoutException:
  65. print("we timed out waiting, skipping...")
  66. for hunk in f:
  67. for line in hunk:
  68. if line.is_added:
  69. violation = "{}:{}".format(f.path, line.target_line_no)
  70. if violation in violations:
  71. numViolations += 1
  72. if args.output:
  73. with open(args.output, "a+") as fp:
  74. fp.write("{}:{}\n".format(
  75. violation, "\t\n".join(
  76. violations[violation])))
  77. else:
  78. print(
  79. "{}:{}".format(
  80. violation, "\t\n".join(
  81. violations[violation])))
  82. return numViolations
  83. if __name__ == "__main__":
  84. ret = main()
  85. exit(ret)