run_ci.sh 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. #!/bin/bash
  2. # Copyright (c) 2017 Linaro Limited
  3. # Copyright (c) 2018 Intel Corporation
  4. #
  5. # SPDX-License-Identifier: Apache-2.0
  6. #
  7. # Author: Anas Nashif
  8. #
  9. # This script is run in CI and support both pull request and commit modes. So
  10. # it can be used also on commits to the tree or on tags.
  11. # The following options are supports:
  12. # -p start the script for pull requests
  13. # -m matrix node number, for example 3 on a 5 node matrix
  14. # -M total number of nodes in the matrix
  15. # -b base branch
  16. # -r the remote to rebase on
  17. #
  18. # The script can be run locally using for example:
  19. # ./scripts/ci/run_ci.sh -b main -r origin -l -R <commit range>
  20. set -xe
  21. twister_options=" --inline-logs -M -N -v --integration"
  22. west_commands_results_file="./pytest_out/west_commands.xml"
  23. matrix_builds=1
  24. matrix=1
  25. function handle_coverage() {
  26. # Upload to codecov.io only on merged builds or if CODECOV_IO variable
  27. # is set.
  28. if [ -n "${CODECOV_IO}" -o -z "${pull_request_nr}" ]; then
  29. # Capture data
  30. echo "Running lcov --capture ..."
  31. lcov --capture \
  32. --directory twister-out/native_posix/ \
  33. --directory twister-out/nrf52_bsim/ \
  34. --directory twister-out/unit_testing/ \
  35. --output-file lcov.pre.info -q --rc lcov_branch_coverage=1
  36. # Remove noise
  37. echo "Exclude data from coverage report..."
  38. lcov -q \
  39. --remove lcov.pre.info mylib.c \
  40. --remove lcov.pre.info tests/\* \
  41. --remove lcov.pre.info samples/\* \
  42. --remove lcov.pre.info ext/\* \
  43. --remove lcov.pre.info *generated* \
  44. -o lcov.info --rc lcov_branch_coverage=1
  45. # Cleanup
  46. rm lcov.pre.info
  47. rm -rf twister-out out-2nd-pass
  48. # Upload to codecov.io
  49. echo "Upload coverage reports to codecov.io"
  50. bash <(curl -s https://codecov.io/bash) -f "lcov.info" -X coveragepy -X fixes
  51. rm -f lcov.info
  52. fi
  53. rm -rf twister-out out-2nd-pass
  54. }
  55. function handle_compiler_cache() {
  56. # Give more details in case we fail because of compiler cache
  57. if [ -f "$HOME/.cache/zephyr/ToolchainCapabilityDatabase.cmake" ]; then
  58. echo "Dumping the capability database in case we are affected by #9992"
  59. cat $HOME/.cache/zephyr/ToolchainCapabilityDatabase.cmake
  60. fi
  61. }
  62. function on_complete() {
  63. source zephyr-env.sh
  64. if [ "$1" == "failure" ]; then
  65. handle_compiler_cache
  66. fi
  67. rm -rf ccache $HOME/.cache/zephyr
  68. if [ "$matrix" = "1" ]; then
  69. echo "Skip handling coverage data..."
  70. #handle_coverage
  71. else
  72. rm -rf twister-out out-2nd-pass
  73. fi
  74. }
  75. function build_test_file() {
  76. # cleanup
  77. rm -f test_file_boards.txt test_file_tests.txt test_file_archs.txt test_file_full.txt
  78. touch test_file_boards.txt test_file_tests.txt test_file_archs.txt test_file_full.txt
  79. twister_exclude_tag_opt=""
  80. # In a pull-request see if we have changed any tests or board definitions
  81. if [ -n "${pull_request_nr}" -o -n "${local_run}" ]; then
  82. ./scripts/zephyr_module.py --twister-out module_tests.args
  83. ./scripts/ci/get_twister_opt.py --commits ${commit_range}
  84. if [ -s modified_tags.args ]; then
  85. twister_exclude_tag_opt="+modified_tags.args"
  86. fi
  87. if [ -s modified_boards.args ]; then
  88. ${twister} ${twister_options} ${twister_exclude_tag_opt} \
  89. +modified_boards.args \
  90. --save-tests test_file_boards.txt || exit 1
  91. fi
  92. if [ -s modified_tests.args ]; then
  93. ${twister} ${twister_options} ${twister_exclude_tag_opt} \
  94. +modified_tests.args \
  95. --save-tests test_file_tests.txt || exit 1
  96. fi
  97. if [ -s modified_archs.args ]; then
  98. ${twister} ${twister_options} ${twister_exclude_tag_opt} \
  99. +modified_archs.args \
  100. --save-tests test_file_archs.txt || exit 1
  101. fi
  102. rm -f modified_tests.args modified_boards.args modified_archs.args
  103. fi
  104. if [ "$SC" == "full" ]; then
  105. # Save list of tests to be run
  106. ${twister} ${twister_options} ${twister_exclude_tag_opt} \
  107. --save-tests test_file_full.txt || exit 1
  108. fi
  109. rm -f modified_tags.args
  110. # Remove headers from all files. We insert it into test_file.txt explicitly
  111. # so we treat all test_file*.txt files the same.
  112. tail -n +2 test_file_full.txt > test_file_full_in.txt
  113. tail -n +2 test_file_archs.txt > test_file_archs_in.txt
  114. tail -n +2 test_file_tests.txt > test_file_tests_in.txt
  115. tail -n +2 test_file_boards.txt > test_file_boards_in.txt
  116. echo "test,arch,platform,status,extra_args,handler,handler_time,ram_size,rom_size" \
  117. > test_file.txt
  118. cat test_file_full_in.txt test_file_archs_in.txt test_file_tests_in.txt \
  119. test_file_boards_in.txt >> test_file.txt
  120. }
  121. function west_setup() {
  122. # West handling
  123. git_dir=$(basename $PWD)
  124. pushd ..
  125. if [ ! -d .west ]; then
  126. west init -l ${git_dir}
  127. west update 1> west.update.log || west update 1> west.update-2.log
  128. west forall -c 'git reset --hard HEAD'
  129. fi
  130. popd
  131. }
  132. while getopts ":p:m:b:r:M:cfslR:" opt; do
  133. case $opt in
  134. c)
  135. echo "Execute CI" >&2
  136. main_ci=1
  137. ;;
  138. l)
  139. echo "Executing script locally" >&2
  140. local_run=1
  141. main_ci=1
  142. ;;
  143. s)
  144. echo "Success" >&2
  145. success=1
  146. ;;
  147. f)
  148. echo "Failure" >&2
  149. failure=1
  150. ;;
  151. p)
  152. echo "Testing a Pull Request: $OPTARG." >&2
  153. pull_request_nr=$OPTARG
  154. ;;
  155. m)
  156. echo "Running on Matrix $OPTARG" >&2
  157. matrix=$OPTARG
  158. ;;
  159. M)
  160. echo "Running a matrix of $OPTARG slaves" >&2
  161. matrix_builds=$OPTARG
  162. ;;
  163. b)
  164. echo "Base Branch: $OPTARG" >&2
  165. branch=$OPTARG
  166. ;;
  167. r)
  168. echo "Remote: $OPTARG" >&2
  169. remote=$OPTARG
  170. ;;
  171. R)
  172. echo "Range: $OPTARG" >&2
  173. range=$OPTARG
  174. ;;
  175. \?)
  176. echo "Invalid option: -$OPTARG" >&2
  177. ;;
  178. esac
  179. done
  180. if [ -n "$main_ci" ]; then
  181. west_setup
  182. if [ -z "$branch" ]; then
  183. echo "No base branch given"
  184. exit 1
  185. else
  186. commit_range=$remote/${branch}..HEAD
  187. echo "Commit range:" ${commit_range}
  188. fi
  189. if [ -n "$range" ]; then
  190. commit_range=$range
  191. fi
  192. source zephyr-env.sh
  193. twister="${ZEPHYR_BASE}/scripts/twister"
  194. # Possibly the only record of what exact version is being tested:
  195. short_git_log='git log -n 5 --oneline --decorate --abbrev=12 '
  196. # check what files have changed for PRs or local runs. If we are
  197. # building for a commit than we always do a "Full Run".
  198. if [ -n "${pull_request_nr}" -o -n "${local_run}" ]; then
  199. SC=`./scripts/ci/what_changed.py --commits ${commit_range}`
  200. else
  201. echo "Full Run"
  202. SC="full"
  203. fi
  204. if [ -n "$pull_request_nr" ]; then
  205. $short_git_log $remote/${branch}
  206. # Now let's pray this script is being run from a
  207. # different location
  208. # https://stackoverflow.com/questions/3398258/edit-shell-script-while-its-running
  209. git rebase $remote/${branch}
  210. fi
  211. $short_git_log
  212. build_test_file
  213. echo "+++ run twister"
  214. # Run a subset of tests based on matrix size
  215. ${twister} ${twister_options} --load-tests test_file.txt \
  216. --subset ${matrix}/${matrix_builds} --retry-failed 3
  217. # Run module tests on matrix #1
  218. if [ "$matrix" = "1" -a "$SC" == "full" ]; then
  219. if [ -s module_tests.args ]; then
  220. ${twister} ${twister_options} \
  221. +module_tests.args --outdir module_tests
  222. fi
  223. fi
  224. # cleanup
  225. rm -f test_file*
  226. elif [ -n "$failure" ]; then
  227. on_complete failure
  228. elif [ -n "$success" ]; then
  229. on_complete
  230. else
  231. echo "Nothing to do"
  232. fi