From c6d6fe031646cbca84e27268fcba1fb5e6c43829 Mon Sep 17 00:00:00 2001 From: Ivan Grokhotkov Date: Wed, 18 Dec 2019 16:11:11 +0100 Subject: [PATCH 1/4] ci: add job to check for C++ guards in header files --- tools/ci/check_cpp_guards.sh | 127 +++++++++++++++++++++++ tools/ci/check_cpp_guards_exceptions.txt | 118 +++++++++++++++++++++ tools/ci/config/pre_check.yml | 16 +++ tools/ci/executable-list.txt | 1 + 4 files changed, 262 insertions(+) create mode 100755 tools/ci/check_cpp_guards.sh create mode 100644 tools/ci/check_cpp_guards_exceptions.txt diff --git a/tools/ci/check_cpp_guards.sh b/tools/ci/check_cpp_guards.sh new file mode 100755 index 0000000000..cc9f5fad12 --- /dev/null +++ b/tools/ci/check_cpp_guards.sh @@ -0,0 +1,127 @@ +#!/bin/bash +# +# This script finds all files which do not contain "#ifdef __cplusplus", except the ones listed in check_cpp_guards_exceptions.txt + +set -euo pipefail + +if [ -z "${IDF_PATH:-}" ]; then + echo "Please run export.sh before invoking this script" + exit 1 +fi + +##################### helper functions ##################### + +# check_one_header +# +# Function used to implement the logic of checking a single header. +# Returns 0 on success (as a normal bash function would). +# Ideally shouldn't print anything. +function check_one_header +{ + include_file=$1 + grep -q '#ifdef __cplusplus' ${include_file} +} + +# get_include_dir_list +# +# Get the list of public include directories from compile_commands.json. +# Stores the result in INCLUDE_DIRS variable. +function get_include_dirs_list +{ + pushd $IDF_PATH/examples/get-started/blink + idf.py reconfigure + + # Get the compiler command line from compile_commands.json + # Poor man's replacement of `jq` in Python. + extract_build_command_py='import json; print(json.load(open("build/compile_commands.json", "r"))[0]["command"])' + build_command=$(python -c "${extract_build_command_py}") + + # build the list of include directories, only considering the directories in "components/" + include_dirs="" + for token in ${build_command}; do + if [[ ! "${token}" =~ ^-I.*/components/.* ]]; then + continue + fi + newline=$'\n' + include_dirs="${include_dirs}${token##-I}${newline}" + done + + export INCLUDE_DIRS=${include_dirs} +} + +# check_include_dirs +# +# Check all header files in directories listed in include_dirs variable. +# Places the list of header files with some issues into file. +# Calls check_one_header function to check a single header file. +function check_include_dirs +{ + local include_dirs="$1" + local failed_headers_list_file="$2" + + rm -f ${failed_headers_list_file} + touch ${failed_headers_list_file} + + for include_dir in ${include_dirs}; do + echo "Processing include directory: ${include_dir}" + include_files=$(find ${include_dir} -name '*.h') + for include_file in ${include_files}; do + printf " Processing ${include_file}" + if ! check_one_header ${include_file}; then + printf " FAIL!\n" + echo ${include_file##${IDF_PATH}/} >> ${failed_headers_list_file} + else + printf " OK\n" + fi + done + done +} + +# filter_exclusions +# +# Takes the list of files in file, and removes all files which +# are listed in file. +# Saves the final (filtered) list in file. +# The exclusions list file may contain comments and empty lines. +function filter_exclusions +{ + local input_list_file=$1 + local exclusions_list_file=$2 + local filtered_list_file=$3 + local temp_list=$(mktemp) + + cp ${input_list_file} ${filtered_list_file} + touch ${temp_list} + + while read -r line; do + if [[ "${line}" =~ ^\# || -z "${line}" ]]; then + continue + fi + echo "Excluding ${line}" + grep -vE "${line}" ${filtered_list_file} > ${temp_list} || true + mv ${temp_list} ${filtered_list_file} + done < ${exclusions_list_file} + + rm -f ${temp_list} +} + +##################### main logic starts here ##################### + +EXCLUSIONS_LIST=${IDF_PATH}/tools/ci/check_cpp_guards_exceptions.txt +MISSING_GUARDS_LIST=missing.txt +FILTERED_LIST=filtered.txt +LINE=--------------------------------------------------------------- + + +get_include_dirs_list +check_include_dirs "${INCLUDE_DIRS}" ${MISSING_GUARDS_LIST} +filter_exclusions ${MISSING_GUARDS_LIST} ${EXCLUSIONS_LIST} ${FILTERED_LIST} + +if [ -s ${FILTERED_LIST} ]; then + echo "The following files don't have C++ guards:" + echo "" + cat ${FILTERED_LIST} + exit 1 +fi + +echo "No errors found" diff --git a/tools/ci/check_cpp_guards_exceptions.txt b/tools/ci/check_cpp_guards_exceptions.txt new file mode 100644 index 0000000000..4985524812 --- /dev/null +++ b/tools/ci/check_cpp_guards_exceptions.txt @@ -0,0 +1,118 @@ +components/xtensa/include/xtensa/* +components/xtensa/include/* +components/xtensa/esp32/include/xtensa/config/* + +components/newlib/platform_include/* + +components/freertos/include/freertos/* + +components/log/include/esp_log_internal.h + +components/soc/esp32/include/hal/* +components/soc/esp32/include/soc/* +components/soc/include/hal/* +components/soc/include/soc/* + +components/esp_rom/include/esp32s2beta/rom/rsa_pss.h + +components/esp_common/include/esp_types.h +components/esp_common/include/esp_assert.h +components/esp_common/include/esp_bit_defs.h +components/esp_common/include/esp_task.h +components/esp_common/include/esp_private/* + +components/esp32/include/esp32/* +components/esp32/include/rom/* +components/esp32/include/esp_clk.h +components/esp32/include/esp_intr.h +components/esp32/include/esp_himem.h +components/esp32/include/esp_spiram.h +components/driver/include/driver/touch_pad.h + +components/driver/include/driver/sdmmc_defs.h +components/driver/include/driver/sdmmc_types.h +components/driver/include/driver/rtc_cntl.h +components/driver/include/driver/sigmadelta.h + +components/esp_event/include/esp_event_loop.h + +components/lwip/include/apps/dhcpserver/* +components/lwip/include/apps/esp_sntp.h +components/lwip/lwip/src/include/compat/* +components/lwip/lwip/src/include/lwip/* +components/lwip/lwip/src/include/netif/* +components/lwip/port/esp32/include/* + +components/vfs/include/sys/dirent.h + +components/esp_wifi/include/esp_private/* +components/esp_wifi/include/esp_wifi_netif.h +components/esp_wifi/include/esp_wifi_default.h +components/esp_wifi/esp32/include/phy_init_data.h + +components/tcpip_adapter/include/tcpip_adapter_compatible/tcpip_adapter_compat.h +components/tcpip_adapter/include/tcpip_adapter.h +components/tcpip_adapter/include/tcpip_adapter_types.h + +components/spi_flash/include/spi_flash_chip_issi.h +components/spi_flash/include/spi_flash_chip_gd.h +components/spi_flash/include/memspi_host_driver.h +components/spi_flash/include/spi_flash_chip_driver.h +components/spi_flash/include/spi_flash_chip_generic.h + +components/bootloader_support/include/esp_flash_data_types.h +components/bootloader_support/include/esp_app_format.h + +components/wpa_supplicant/include/* +components/wpa_supplicant/port/* + +components/mbedtls/port/include/* +components/mbedtls/mbedtls/include/mbedtls/* + +components/espcoredump/include/esp_core_dump.h + +components/perfmon/include/perfmon.h + +components/asio/* +components/coap/* +components/nghttp/* +components/cbor/* + +components/esp-tls/private_include/* + +components/esp_http_server/include/http_server.h + +components/protobuf-c/* + +components/mdns/include/mdns_console.h + +components/esp_serial_slave_link/include/esp_serial_slave_link/essl.h +components/esp_serial_slave_link/include/esp_serial_slave_link/essl_sdio.h + +components/expat/* + +components/fatfs/vfs/vfs_fat_internal.h +components/fatfs/src/ffconf.h + +components/freemodbus/common/include/esp_modbus_common.h +components/freemodbus/common/include/mbcontroller.h +components/freemodbus/common/include/esp_modbus_master.h +components/freemodbus/common/include/esp_modbus_slave.h + +components/idf_test/include/idf_performance.h + +components/json/cJSON/* + +components/libsodium/* + +components/mqtt/esp-mqtt/include/mqtt_supported_features.h +components/mqtt/esp-mqtt/include/mqtt_config.h + +components/openssl/include/platform/ssl_opt.h + +components/spiffs/include/spiffs_config.h + +components/unity/unity/src/unity_internals.h +components/unity/include/priv/setjmp.h + + diff --git a/tools/ci/config/pre_check.yml b/tools/ci/config/pre_check.yml index 881cc2c2ef..f057904728 100644 --- a/tools/ci/config/pre_check.yml +++ b/tools/ci/config/pre_check.yml @@ -151,3 +151,19 @@ build_template_app: - for dir in esp32 esp32s2beta; do test $(xtensa-esp32-elf-nm $dir/*.a | grep -w printf | wc -l) -eq 0; done; - for dir in esp32 esp32s2beta; do test $(xtensa-esp32-elf-nm $dir/*.a | grep -w ets_printf | wc -l) -eq 0; done; - popd + +check_header_files_cpp_guards: + stage: pre_check + image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG + tags: + - build + variables: + BATCH_BUILD: "1" + only: + variables: + - $BOT_TRIGGER_WITH_LABEL == null + - $BOT_LABEL_BUILD + - $BOT_LABEL_REGULAR_TEST + script: + - tools/ci/check_cpp_guards.sh + diff --git a/tools/ci/executable-list.txt b/tools/ci/executable-list.txt index bea16d10f5..443053f0e5 100644 --- a/tools/ci/executable-list.txt +++ b/tools/ci/executable-list.txt @@ -36,6 +36,7 @@ tools/ci/build_examples_cmake.sh tools/ci/check-executable.sh tools/ci/check-line-endings.sh tools/ci/check_build_warnings.py +tools/ci/check_cpp_guards.sh tools/ci/check_deprecated_kconfigs.py tools/ci/check_examples_cmake_make.sh tools/ci/check_examples_rom_header.sh From cf9cab4723cadb3f25c81c399daba1b197c1a37c Mon Sep 17 00:00:00 2001 From: David Cermak Date: Fri, 20 Dec 2019 16:42:39 +0100 Subject: [PATCH 2/4] ci: public header check: add test whether all headers compile script checks cpp guards preprocessing with C and CPP compiler and compares outputs. As a part of this check also a test whether all header files compile is performed. IDF-288 --- tools/ci/check_cpp_guards.sh | 197 ++++++++++++++++++----- tools/ci/check_cpp_guards_exceptions.txt | 57 +++---- tools/ci/config/pre_check.yml | 2 +- 3 files changed, 181 insertions(+), 75 deletions(-) diff --git a/tools/ci/check_cpp_guards.sh b/tools/ci/check_cpp_guards.sh index cc9f5fad12..558990d56e 100755 --- a/tools/ci/check_cpp_guards.sh +++ b/tools/ci/check_cpp_guards.sh @@ -1,6 +1,15 @@ #!/bin/bash # -# This script finds all files which do not contain "#ifdef __cplusplus", except the ones listed in check_cpp_guards_exceptions.txt +# This script finds all *.h files in public include dirs and checks for +# * compilation issues +# * c++ header guards +# +# The script fails if another a new header fails and not listed in check_cpp_guards_exceptions.txt +# +# Script arguments: +# --verbose: enables printing details if compilation fails (verbose is off by default) +# --process-all-files: all files are checked and then excluded, by default the excluded files are skipped and not checked +# --include-maxdepth=X: maximum directory depth for finding include files in include dirs (by default not specified=infinity) set -euo pipefail @@ -11,15 +20,94 @@ fi ##################### helper functions ##################### +# log_error +# +# Logs details on filename processing when the include fails +function log_error() +{ + if [ ! -z "$VERBOSE" ]; then + file=$1 + echo " verbose..." + echo "$LINE" + grep -A5 "error:" + echo "$LINE" + echo -n "$file" + fi +} + +# preproc_one_header +# +# Function to preprocess a header with CC and CXX and compares result +# Returns +# 0 -> Never returns zero (for easier return value expansion) +# 1 -> Cannot compile, but failed with "#error" directive (header seems OK) +# 2 -> Cannot compile with another issue, logged if verbose (header FAILs) +# 3 -> Both preprocessors produce zero out (header file is OK) +# 4 -> Both preprocessors produce the same, non-zero output (header file FAILs) +# 5 -> Both preprocessors produce different, non-zero output with extern "C" (header seems OK) +# 6 -> Both preprocessors produce different, non-zero output without extern "C" (header seems OK) +function preproc_one_header() +{ + file=$1 + if ! cpp_out=`xtensa-esp32-elf-g++ -w -P -E --std=c++17 ${file} ${FLAGS_INCLUDE_DIRS} 2>&1`; then + if [[ "$cpp_out" =~ "#error" ]]; then + return 1 + fi; + # Logging the compilation issue issue + echo "$cpp_out" | log_error $file + return 2 + fi; + if ! c99_out=`xtensa-esp32-elf-gcc -w -P -E --std=c99 ${file} ${FLAGS_INCLUDE_DIRS} 2>/dev/null`; then + # Should not happen, fail with compilation issue + return 2 + fi; + if diff_out=`diff <(echo "$cpp_out" ) <(echo "$c99_out")`; then + # checks zero output: contains spaces only if lines < 5 (for performace) + lines=`echo "$cpp_out" | wc -l` + if [ $lines -lt 5 ] && [ -z "${cpp_out// }" ]; then + return 3 + fi + return 4 + elif echo $diff_out | grep -q 'extern "C"'; then + return 5 + fi + return 6 +} + # check_one_header # # Function used to implement the logic of checking a single header. # Returns 0 on success (as a normal bash function would). -# Ideally shouldn't print anything. +# Ideally shouldn't print anything (if verbose, preproc_one_header outputs compilation errors) function check_one_header { include_file=$1 - grep -q '#ifdef __cplusplus' ${include_file} + # Procedure: + # 1) Preprocess the include file with C preprocessor and with CPP preprocessor + # - Pass the test if the preprocessor outputs are the same and whitespaces only (#define only header) + # - Fail the test if the preprocessor outputs are the same (but with some code) + # - If outputs different, continue with 2) + # 2) Strip out all include directives to generate "temp.h" + # 3) Preprocess the temp.h the same way in (1) + # - Pass the test if the preprocessor outputs are the same and whitespaces only (#include only header) + # - Fail the test if the preprocessor outputs are the same (but with some code) + # - If outputs different, pass the test + # + preproc_one_header ${include_file} || ret=$? + case $ret in + 1*) return 0 ;; # Pass if #error + 2*) return 2 ;; # Fail if other compile issue + 3*) return 0 ;; # Pass if zero out + 4*) return 1 ;; # Fail if the same non-zero out + [5-6]*) # Process further if different and non-zero + sed '/#include/d; /#error/d' ${include_file} > temp.h + case `preproc_one_header temp.h || echo $?` in + [1-3]*) return 0 ;; # Pass if compile issues (have stripped includes) + 4*) return 1 ;; # Fail if the same non-zero out + 5*) return 0 ;; # Pass if different and non-zero and with extern "C" + 6*) return 1 ;; # Fail if different and non-zero, but without extern "C" + esac + esac } # get_include_dir_list @@ -39,6 +127,9 @@ function get_include_dirs_list # build the list of include directories, only considering the directories in "components/" include_dirs="" for token in ${build_command}; do + if [[ "${token}" =~ ^-I.* ]]; then + flags_include_dirs+=" $token" + fi if [[ ! "${token}" =~ ^-I.*/components/.* ]]; then continue fi @@ -47,78 +138,112 @@ function get_include_dirs_list done export INCLUDE_DIRS=${include_dirs} + flags_include_dirs+=" -I $IDF_PATH/examples/get-started/blink/build/config" + export FLAGS_INCLUDE_DIRS=${flags_include_dirs} } -# check_include_dirs +# check_include_dirs # # Check all header files in directories listed in include_dirs variable. -# Places the list of header files with some issues into file. +# Places the list of header files where suspects missing C++ guard to file. +# Places the list of header files where compilation issues found to file. # Calls check_one_header function to check a single header file. function check_include_dirs { local include_dirs="$1" - local failed_headers_list_file="$2" + local missing_guard_list_file="$2" + local compile_error_list_file="$3" + local include_dir_maxdepth=${INCLUDE_DIR_MAXDEPTH:-} + local grep_expr=$(mktemp) - rm -f ${failed_headers_list_file} - touch ${failed_headers_list_file} + rm -f ${missing_guard_list_file} ${compile_error_list_file} + touch ${missing_guard_list_file} ${compile_error_list_file} + + if [ -z "$PROCESS_ALL_INCLUDES" ]; then + # generate grep expr: 1) strip comments, 2) generate '\|\|...' + exclusions=`cat $EXCLUSIONS_LIST | sed '/^#/d'` + echo $exclusions | sed 's/ /\\|/g' > ${grep_expr} + fi; for include_dir in ${include_dirs}; do echo "Processing include directory: ${include_dir}" - include_files=$(find ${include_dir} -name '*.h') + include_files=$(find ${include_dir} ${include_dir_maxdepth} -name '*.h') for include_file in ${include_files}; do - printf " Processing ${include_file}" - if ! check_one_header ${include_file}; then - printf " FAIL!\n" - echo ${include_file##${IDF_PATH}/} >> ${failed_headers_list_file} - else - printf " OK\n" + if [ -z "$PROCESS_ALL_INCLUDES" ] && echo "$include_file" | grep -q -f ${grep_expr}; then + printf " Excluding ${include_file}\n" + continue fi + printf " Processing ${include_file}" + check_one_header ${include_file} && ret=0 || ret=$? + case "$ret" in + 0*) printf " OK\n" ;; + 1*) + echo ${include_file##${IDF_PATH}/} >> ${missing_guard_list_file} + printf " FAIL (missing c++ guard)\n" ;; + 2*) + echo ${include_file##${IDF_PATH}/} >> ${compile_error_list_file} + printf " FAIL (cannot compile)\n" ;; + esac done done + rm -f ${grep_expr} } -# filter_exclusions +# filter_exclusions # -# Takes the list of files in file, and removes all files which +# Takes the list of files from stdin, and removes all files which # are listed in file. # Saves the final (filtered) list in file. # The exclusions list file may contain comments and empty lines. function filter_exclusions { - local input_list_file=$1 - local exclusions_list_file=$2 - local filtered_list_file=$3 - local temp_list=$(mktemp) + local exclusions_list_file=$1 + local filtered_list_file=$2 + local grep_expr=$(mktemp) - cp ${input_list_file} ${filtered_list_file} - touch ${temp_list} + # generate grep expr: 1) strip comments, 2) generate '\|\|...' + exclusions=`cat $exclusions_list_file | sed '/^#/d'` + echo $exclusions | sed 's/ /\\|/g' > ${grep_expr} - while read -r line; do - if [[ "${line}" =~ ^\# || -z "${line}" ]]; then - continue - fi - echo "Excluding ${line}" - grep -vE "${line}" ${filtered_list_file} > ${temp_list} || true - mv ${temp_list} ${filtered_list_file} - done < ${exclusions_list_file} - - rm -f ${temp_list} + # pass the expression as file (input goes from stdin) + grep -vf ${grep_expr} > ${filtered_list_file} || true + rm -f ${grep_expr} } ##################### main logic starts here ##################### EXCLUSIONS_LIST=${IDF_PATH}/tools/ci/check_cpp_guards_exceptions.txt MISSING_GUARDS_LIST=missing.txt +CANNOT_COMPILE_LIST=cannot_compile.txt FILTERED_LIST=filtered.txt LINE=--------------------------------------------------------------- +VERBOSE="" +PROCESS_ALL_INCLUDES="" +# Process command line arguments +for i in $@; do + if [[ "$i" =~ "--verbose" ]]; then + echo "$0: Enabling verbose output" + VERBOSE="1" + fi + if [[ "$i" =~ "--process-all-files" ]]; then + echo "$0: Processing all files" + PROCESS_ALL_INCLUDES="1" + fi + if [[ "$i" =~ --include-maxdepth=* ]]; then + INCLUDE_DIR_MAXDEPTH="-maxdepth ${i##--include-maxdepth=}" + echo "$0: Searching for includes with: $INCLUDE_DIR_MAXDEPTH" + fi +done +echo $LINE get_include_dirs_list -check_include_dirs "${INCLUDE_DIRS}" ${MISSING_GUARDS_LIST} -filter_exclusions ${MISSING_GUARDS_LIST} ${EXCLUSIONS_LIST} ${FILTERED_LIST} +check_include_dirs "${INCLUDE_DIRS}" ${MISSING_GUARDS_LIST} ${CANNOT_COMPILE_LIST} +cat ${MISSING_GUARDS_LIST} ${CANNOT_COMPILE_LIST} | filter_exclusions ${EXCLUSIONS_LIST} ${FILTERED_LIST} +echo $LINE if [ -s ${FILTERED_LIST} ]; then - echo "The following files don't have C++ guards:" + echo "The following files failed:" echo "" cat ${FILTERED_LIST} exit 1 diff --git a/tools/ci/check_cpp_guards_exceptions.txt b/tools/ci/check_cpp_guards_exceptions.txt index 4985524812..8fe96a4420 100644 --- a/tools/ci/check_cpp_guards_exceptions.txt +++ b/tools/ci/check_cpp_guards_exceptions.txt @@ -1,3 +1,5 @@ +### General ignore list +# components/xtensa/include/xtensa/* components/xtensa/include/* components/xtensa/esp32/include/xtensa/config/* @@ -15,52 +17,35 @@ components/soc/include/soc/* components/esp_rom/include/esp32s2beta/rom/rsa_pss.h -components/esp_common/include/esp_types.h -components/esp_common/include/esp_assert.h -components/esp_common/include/esp_bit_defs.h -components/esp_common/include/esp_task.h components/esp_common/include/esp_private/* -components/esp32/include/esp32/* -components/esp32/include/rom/* -components/esp32/include/esp_clk.h -components/esp32/include/esp_intr.h -components/esp32/include/esp_himem.h -components/esp32/include/esp_spiram.h -components/driver/include/driver/touch_pad.h +components/esp32/include/esp32/brownout.h +components/esp32/include/esp32/spiram.h +components/esp32/include/esp32/cache_err_int.h components/driver/include/driver/sdmmc_defs.h components/driver/include/driver/sdmmc_types.h -components/driver/include/driver/rtc_cntl.h -components/driver/include/driver/sigmadelta.h -components/esp_event/include/esp_event_loop.h +components/lwip/include/apps/dhcpserver/dhcpserver_options.h +components/lwip/include/apps/dhcpserver/dhcpserver.h -components/lwip/include/apps/dhcpserver/* -components/lwip/include/apps/esp_sntp.h -components/lwip/lwip/src/include/compat/* -components/lwip/lwip/src/include/lwip/* -components/lwip/lwip/src/include/netif/* -components/lwip/port/esp32/include/* +components/lwip/lwip/src/include/lwip/priv/memp_std.h +components/lwip/port/esp32/include/arch/cc.h +components/lwip/port/esp32/include/debug/lwip_debug.h +components/lwip/port/esp32/include/arch/cc.h components/vfs/include/sys/dirent.h -components/esp_wifi/include/esp_private/* components/esp_wifi/include/esp_wifi_netif.h components/esp_wifi/include/esp_wifi_default.h components/esp_wifi/esp32/include/phy_init_data.h -components/tcpip_adapter/include/tcpip_adapter_compatible/tcpip_adapter_compat.h -components/tcpip_adapter/include/tcpip_adapter.h -components/tcpip_adapter/include/tcpip_adapter_types.h - components/spi_flash/include/spi_flash_chip_issi.h components/spi_flash/include/spi_flash_chip_gd.h components/spi_flash/include/memspi_host_driver.h components/spi_flash/include/spi_flash_chip_driver.h components/spi_flash/include/spi_flash_chip_generic.h -components/bootloader_support/include/esp_flash_data_types.h components/bootloader_support/include/esp_app_format.h components/wpa_supplicant/include/* @@ -71,17 +56,12 @@ components/mbedtls/mbedtls/include/mbedtls/* components/espcoredump/include/esp_core_dump.h -components/perfmon/include/perfmon.h - -components/asio/* components/coap/* components/nghttp/* components/cbor/* components/esp-tls/private_include/* -components/esp_http_server/include/http_server.h - components/protobuf-c/* components/mdns/include/mdns_console.h @@ -95,7 +75,6 @@ components/fatfs/vfs/vfs_fat_internal.h components/fatfs/src/ffconf.h components/freemodbus/common/include/esp_modbus_common.h -components/freemodbus/common/include/mbcontroller.h components/freemodbus/common/include/esp_modbus_master.h components/freemodbus/common/include/esp_modbus_slave.h @@ -105,14 +84,16 @@ components/json/cJSON/* components/libsodium/* -components/mqtt/esp-mqtt/include/mqtt_supported_features.h -components/mqtt/esp-mqtt/include/mqtt_config.h - -components/openssl/include/platform/ssl_opt.h - components/spiffs/include/spiffs_config.h components/unity/unity/src/unity_internals.h -components/unity/include/priv/setjmp.h +components/unity/include/unity_config.h +components/unity/include/unity_test_runner.h +### Here are the files that do not compile for some reason +# +components/esp_rom/include/esp32s2beta/rom/spi_flash.h +components/app_trace/include/esp_sysview_trace.h +components/esp_gdbstub/include/esp_gdbstub.h +components/openssl/include/platform/ssl_pm.h diff --git a/tools/ci/config/pre_check.yml b/tools/ci/config/pre_check.yml index f057904728..34bd440b89 100644 --- a/tools/ci/config/pre_check.yml +++ b/tools/ci/config/pre_check.yml @@ -165,5 +165,5 @@ check_header_files_cpp_guards: - $BOT_LABEL_BUILD - $BOT_LABEL_REGULAR_TEST script: - - tools/ci/check_cpp_guards.sh + - tools/ci/check_cpp_guards.sh --verbose --process-all-files From ff0128dd02904138250fe50c5218e0be78f14af7 Mon Sep 17 00:00:00 2001 From: David Cermak Date: Mon, 23 Dec 2019 20:41:15 +0100 Subject: [PATCH 3/4] drivers: fixed c++ guards in driver public headers --- components/driver/include/driver/sigmadelta.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/components/driver/include/driver/sigmadelta.h b/components/driver/include/driver/sigmadelta.h index 783baa1421..28906d069c 100644 --- a/components/driver/include/driver/sigmadelta.h +++ b/components/driver/include/driver/sigmadelta.h @@ -19,7 +19,7 @@ #include "driver/gpio.h" #include "hal/sigmadelta_types.h" -#ifdef _cplusplus +#ifdef __cplusplus extern "C" { #endif From 2c0bd8878633c8e042a40c428f0271866f7439bb Mon Sep 17 00:00:00 2001 From: David Cermak Date: Wed, 25 Dec 2019 21:03:41 +0100 Subject: [PATCH 4/4] ci: public header check: add test for sdkconfig values if the CONFIG_XXX macros used then sdkconfig.h should be included. Also added support for multi-process execution --- tools/ci/check_cpp_guards.sh | 252 ----------- tools/ci/check_public_headers.sh | 391 ++++++++++++++++++ ...xt => check_public_headers_exceptions.txt} | 8 +- tools/ci/config/pre_check.yml | 4 +- tools/ci/executable-list.txt | 2 +- 5 files changed, 400 insertions(+), 257 deletions(-) delete mode 100755 tools/ci/check_cpp_guards.sh create mode 100755 tools/ci/check_public_headers.sh rename tools/ci/{check_cpp_guards_exceptions.txt => check_public_headers_exceptions.txt} (90%) diff --git a/tools/ci/check_cpp_guards.sh b/tools/ci/check_cpp_guards.sh deleted file mode 100755 index 558990d56e..0000000000 --- a/tools/ci/check_cpp_guards.sh +++ /dev/null @@ -1,252 +0,0 @@ -#!/bin/bash -# -# This script finds all *.h files in public include dirs and checks for -# * compilation issues -# * c++ header guards -# -# The script fails if another a new header fails and not listed in check_cpp_guards_exceptions.txt -# -# Script arguments: -# --verbose: enables printing details if compilation fails (verbose is off by default) -# --process-all-files: all files are checked and then excluded, by default the excluded files are skipped and not checked -# --include-maxdepth=X: maximum directory depth for finding include files in include dirs (by default not specified=infinity) - -set -euo pipefail - -if [ -z "${IDF_PATH:-}" ]; then - echo "Please run export.sh before invoking this script" - exit 1 -fi - -##################### helper functions ##################### - -# log_error -# -# Logs details on filename processing when the include fails -function log_error() -{ - if [ ! -z "$VERBOSE" ]; then - file=$1 - echo " verbose..." - echo "$LINE" - grep -A5 "error:" - echo "$LINE" - echo -n "$file" - fi -} - -# preproc_one_header -# -# Function to preprocess a header with CC and CXX and compares result -# Returns -# 0 -> Never returns zero (for easier return value expansion) -# 1 -> Cannot compile, but failed with "#error" directive (header seems OK) -# 2 -> Cannot compile with another issue, logged if verbose (header FAILs) -# 3 -> Both preprocessors produce zero out (header file is OK) -# 4 -> Both preprocessors produce the same, non-zero output (header file FAILs) -# 5 -> Both preprocessors produce different, non-zero output with extern "C" (header seems OK) -# 6 -> Both preprocessors produce different, non-zero output without extern "C" (header seems OK) -function preproc_one_header() -{ - file=$1 - if ! cpp_out=`xtensa-esp32-elf-g++ -w -P -E --std=c++17 ${file} ${FLAGS_INCLUDE_DIRS} 2>&1`; then - if [[ "$cpp_out" =~ "#error" ]]; then - return 1 - fi; - # Logging the compilation issue issue - echo "$cpp_out" | log_error $file - return 2 - fi; - if ! c99_out=`xtensa-esp32-elf-gcc -w -P -E --std=c99 ${file} ${FLAGS_INCLUDE_DIRS} 2>/dev/null`; then - # Should not happen, fail with compilation issue - return 2 - fi; - if diff_out=`diff <(echo "$cpp_out" ) <(echo "$c99_out")`; then - # checks zero output: contains spaces only if lines < 5 (for performace) - lines=`echo "$cpp_out" | wc -l` - if [ $lines -lt 5 ] && [ -z "${cpp_out// }" ]; then - return 3 - fi - return 4 - elif echo $diff_out | grep -q 'extern "C"'; then - return 5 - fi - return 6 -} - -# check_one_header -# -# Function used to implement the logic of checking a single header. -# Returns 0 on success (as a normal bash function would). -# Ideally shouldn't print anything (if verbose, preproc_one_header outputs compilation errors) -function check_one_header -{ - include_file=$1 - # Procedure: - # 1) Preprocess the include file with C preprocessor and with CPP preprocessor - # - Pass the test if the preprocessor outputs are the same and whitespaces only (#define only header) - # - Fail the test if the preprocessor outputs are the same (but with some code) - # - If outputs different, continue with 2) - # 2) Strip out all include directives to generate "temp.h" - # 3) Preprocess the temp.h the same way in (1) - # - Pass the test if the preprocessor outputs are the same and whitespaces only (#include only header) - # - Fail the test if the preprocessor outputs are the same (but with some code) - # - If outputs different, pass the test - # - preproc_one_header ${include_file} || ret=$? - case $ret in - 1*) return 0 ;; # Pass if #error - 2*) return 2 ;; # Fail if other compile issue - 3*) return 0 ;; # Pass if zero out - 4*) return 1 ;; # Fail if the same non-zero out - [5-6]*) # Process further if different and non-zero - sed '/#include/d; /#error/d' ${include_file} > temp.h - case `preproc_one_header temp.h || echo $?` in - [1-3]*) return 0 ;; # Pass if compile issues (have stripped includes) - 4*) return 1 ;; # Fail if the same non-zero out - 5*) return 0 ;; # Pass if different and non-zero and with extern "C" - 6*) return 1 ;; # Fail if different and non-zero, but without extern "C" - esac - esac -} - -# get_include_dir_list -# -# Get the list of public include directories from compile_commands.json. -# Stores the result in INCLUDE_DIRS variable. -function get_include_dirs_list -{ - pushd $IDF_PATH/examples/get-started/blink - idf.py reconfigure - - # Get the compiler command line from compile_commands.json - # Poor man's replacement of `jq` in Python. - extract_build_command_py='import json; print(json.load(open("build/compile_commands.json", "r"))[0]["command"])' - build_command=$(python -c "${extract_build_command_py}") - - # build the list of include directories, only considering the directories in "components/" - include_dirs="" - for token in ${build_command}; do - if [[ "${token}" =~ ^-I.* ]]; then - flags_include_dirs+=" $token" - fi - if [[ ! "${token}" =~ ^-I.*/components/.* ]]; then - continue - fi - newline=$'\n' - include_dirs="${include_dirs}${token##-I}${newline}" - done - - export INCLUDE_DIRS=${include_dirs} - flags_include_dirs+=" -I $IDF_PATH/examples/get-started/blink/build/config" - export FLAGS_INCLUDE_DIRS=${flags_include_dirs} -} - -# check_include_dirs -# -# Check all header files in directories listed in include_dirs variable. -# Places the list of header files where suspects missing C++ guard to file. -# Places the list of header files where compilation issues found to file. -# Calls check_one_header function to check a single header file. -function check_include_dirs -{ - local include_dirs="$1" - local missing_guard_list_file="$2" - local compile_error_list_file="$3" - local include_dir_maxdepth=${INCLUDE_DIR_MAXDEPTH:-} - local grep_expr=$(mktemp) - - rm -f ${missing_guard_list_file} ${compile_error_list_file} - touch ${missing_guard_list_file} ${compile_error_list_file} - - if [ -z "$PROCESS_ALL_INCLUDES" ]; then - # generate grep expr: 1) strip comments, 2) generate '\|\|...' - exclusions=`cat $EXCLUSIONS_LIST | sed '/^#/d'` - echo $exclusions | sed 's/ /\\|/g' > ${grep_expr} - fi; - - for include_dir in ${include_dirs}; do - echo "Processing include directory: ${include_dir}" - include_files=$(find ${include_dir} ${include_dir_maxdepth} -name '*.h') - for include_file in ${include_files}; do - if [ -z "$PROCESS_ALL_INCLUDES" ] && echo "$include_file" | grep -q -f ${grep_expr}; then - printf " Excluding ${include_file}\n" - continue - fi - printf " Processing ${include_file}" - check_one_header ${include_file} && ret=0 || ret=$? - case "$ret" in - 0*) printf " OK\n" ;; - 1*) - echo ${include_file##${IDF_PATH}/} >> ${missing_guard_list_file} - printf " FAIL (missing c++ guard)\n" ;; - 2*) - echo ${include_file##${IDF_PATH}/} >> ${compile_error_list_file} - printf " FAIL (cannot compile)\n" ;; - esac - done - done - rm -f ${grep_expr} -} - -# filter_exclusions -# -# Takes the list of files from stdin, and removes all files which -# are listed in file. -# Saves the final (filtered) list in file. -# The exclusions list file may contain comments and empty lines. -function filter_exclusions -{ - local exclusions_list_file=$1 - local filtered_list_file=$2 - local grep_expr=$(mktemp) - - # generate grep expr: 1) strip comments, 2) generate '\|\|...' - exclusions=`cat $exclusions_list_file | sed '/^#/d'` - echo $exclusions | sed 's/ /\\|/g' > ${grep_expr} - - # pass the expression as file (input goes from stdin) - grep -vf ${grep_expr} > ${filtered_list_file} || true - rm -f ${grep_expr} -} - -##################### main logic starts here ##################### - -EXCLUSIONS_LIST=${IDF_PATH}/tools/ci/check_cpp_guards_exceptions.txt -MISSING_GUARDS_LIST=missing.txt -CANNOT_COMPILE_LIST=cannot_compile.txt -FILTERED_LIST=filtered.txt -LINE=--------------------------------------------------------------- -VERBOSE="" -PROCESS_ALL_INCLUDES="" - -# Process command line arguments -for i in $@; do - if [[ "$i" =~ "--verbose" ]]; then - echo "$0: Enabling verbose output" - VERBOSE="1" - fi - if [[ "$i" =~ "--process-all-files" ]]; then - echo "$0: Processing all files" - PROCESS_ALL_INCLUDES="1" - fi - if [[ "$i" =~ --include-maxdepth=* ]]; then - INCLUDE_DIR_MAXDEPTH="-maxdepth ${i##--include-maxdepth=}" - echo "$0: Searching for includes with: $INCLUDE_DIR_MAXDEPTH" - fi -done - -echo $LINE -get_include_dirs_list -check_include_dirs "${INCLUDE_DIRS}" ${MISSING_GUARDS_LIST} ${CANNOT_COMPILE_LIST} -cat ${MISSING_GUARDS_LIST} ${CANNOT_COMPILE_LIST} | filter_exclusions ${EXCLUSIONS_LIST} ${FILTERED_LIST} -echo $LINE - -if [ -s ${FILTERED_LIST} ]; then - echo "The following files failed:" - echo "" - cat ${FILTERED_LIST} - exit 1 -fi - -echo "No errors found" diff --git a/tools/ci/check_public_headers.sh b/tools/ci/check_public_headers.sh new file mode 100755 index 0000000000..9d77d020f2 --- /dev/null +++ b/tools/ci/check_public_headers.sh @@ -0,0 +1,391 @@ +#!/bin/bash +# +# This script finds all *.h files in public include dirs and checks for +# * compilation issues +# * c++ header guards +# * need for sdkconfig.h inclusion (if CONFIG_ macros referenced) +# +# Script arguments: +# --verbose: enables printing details if compilation fails (verbose is off by default) +# --process-all-files: all files are checked and then excluded, by default the excluded files are skipped and not checked +# --include-maxdepth=X: maximum directory depth for finding include files in include dirs (by default not specified=infinity) +# --jobs=X: run the checker in X separate processes + +set -euo pipefail + +if [ -z "${IDF_PATH:-}" ]; then + echo "Please run export.sh before invoking this script" + exit 1 +fi + +##################### helper functions ##################### + +# log_error +# +# Logs details on filename processing when the include fails +function log_error() +{ + if [ ! -z "${VERBOSE:-}" ]; then + file=$1 + echo " verbose..." + echo "$LINE" + grep -A5 "error:" + echo "$LINE" + echo -n "$file" + fi +} + +# log_details +# +# Logs details as function argument to stdout +function log_details() +{ + if [ ! -z "${VERBOSE:-}" ]; then + printf "... $1 " + fi +} + + +# preproc_one_header +# +# Function to preprocess a header with CC and CXX and compares result +# Returns +# 0 -> Never returns zero (for easier return value expansion) +# 1 -> Cannot compile and failed with injected SDKCONFIG #error (header FAILs) +# 2 -> Cannot compile, but failed with "#error" directive (header seems OK) +# 3 -> Cannot compile with another issue, logged if verbose (header FAILs) +# 4 -> Both preprocessors produce zero out (header file is OK) +# 5 -> Both preprocessors produce the same, non-zero output (header file FAILs) +# 6 -> Both preprocessors produce different, non-zero output with extern "C" (header seems OK) +# 7 -> Both preprocessors produce different, non-zero output without extern "C" (header fails) +function preproc_one_header() +{ + local extra_flags="" + local file=$1 + # check if the header (after removing comments) references CONFIG_XXX macros + if ${CC} -fpreprocessed -dD -P -E ${file} ${FLAGS_INCLUDE_DIRS} 2>/dev/null | grep -q -wo -E 'CONFIG_[A-Z0-9_]+'; then + # enable sdkconfig.h inclusion checker + extra_flags="-DIDF_CHECK_SDKCONFIG_INCLUDED" + fi + if ! cpp_out=`${CXX} -w -P -E --std=c++17 -include ${file} ${extra_flags} ${TEST_MAIN_C} ${FLAGS_INCLUDE_DIRS} 2>&1`; then + if [[ "$cpp_out" =~ "#error" ]]; then + if [[ "$cpp_out" =~ "#error CONFIG_VARS_USED_WHILE_SDKCONFIG_NOT_INCLUDED" ]]; then + # failed for sdkconfig.h issue + return 1 + fi; + # header passed in general (error directive) + return 2 + fi; + # Logging the compilation issue + echo "$cpp_out" | log_error $file + return 3 + fi; + if ! c99_out=`${CC} -w -P -E --std=c99 -include ${file} ${extra_flags} ${TEST_MAIN_C} ${FLAGS_INCLUDE_DIRS} 2>/dev/null`; then + # Should not happen, fail with compilation issue + return 3 + fi; + if diff_out=`diff <(echo "$cpp_out" ) <(echo "$c99_out")`; then + # checks zero output: contains spaces only if lines < 5 (for performace) + lines=`echo "$cpp_out" | wc -l` + if [ $lines -lt 5 ] && [ -z "${cpp_out// }" ]; then + return 4 + fi + return 5 + elif echo $diff_out | grep -q 'extern "C"'; then + return 6 + fi + return 7 +} + +# check_one_header +# +# Function used to implement the logic of checking a single header. +# Returns 0 on success (as a normal bash function would). +# - returns 1 if suspects C++ guard missing +# - returns 2 if compile issue (other then error directive) +# - returns 3 if CONFIG_XXX variable used but sdkconfig.h not included +# Ideally shouldn't print anything (if verbose, preproc_one_header outputs compilation errors) +function check_one_header +{ + include_file=$1 + # Procedure: + # 1) Preprocess the include file with C preprocessor and with CPP preprocessor + # - Pass the test if the preprocessor outputs are the same and whitespaces only (#define only header) + # - Fail the test if the preprocessor outputs are the same (but with some code) + # - If outputs different, continue with 2) + # 2) Strip out all include directives to generate "temp.h" + # 3) Preprocess the temp.h the same way in (1) + # - Pass the test if the preprocessor outputs are the same and whitespaces only (#include only header) + # - Fail the test if the preprocessor outputs are the same (but with some code) + # - If outputs different, pass the test + # + preproc_one_header ${include_file} || ret=$? + case $ret in + 1*) log_details "sdkconfig issue"; return 3 ;; + 2*) log_details "compile failed with #error"; return 0 ;; + 3*) log_details "another compile issue"; return 2 ;; + 4*) log_details "zero output"; return 0 ;; + 5*) log_details "the same non-zero out"; return 1 ;; + [6-7]*) log_details "different and non-zero: check again without includes"; + # Removing includes to check if the header itself contains a code + temp=$(mktemp) + sed '/#include/d; /#error/d' ${include_file} > ${temp} + case `preproc_one_header ${temp} || echo $?` in + [1-3]*) log_details "ok, compile issue"; return 0 ;; + 4*) log_details "ok, zero output"; return 0 ;; + 5*) log_details "the same non-zero out"; return 1 ;; + 6*) log_details "different, non-zero, with extern \"C\""; return 0 ;; + 7*) log_details "different, non-zero, no extern \"C\""; return 1 ;; + esac + rm -f ${temp} + esac +} + +# create_test_main +# +# Generates a compilation unit to fail if sdkconfig.h inclusion to be checked +# Exports the full path to the file as TEST_MAIN_C +function create_test_main +{ + echo "#if defined(IDF_CHECK_SDKCONFIG_INCLUDED) && ! defined(IDF_SDKCONFIG_INCLUDED) + #error CONFIG_VARS_USED_WHILE_SDKCONFIG_NOT_INCLUDED + #endif" > ${1}/compile.c + export TEST_MAIN_C=${1}/compile.c +} + +# get_include_dir_list +# +# Get the list of public include directories from compile_commands.json. +# Stores the result in INCLUDE_DIRS variable. +function get_include_dirs_list +{ + local proj_dir=${1} + local sdkinclude_dir=${proj_dir}/build/config + pushd $proj_dir + idf.py reconfigure + + # Get the compiler command line from compile_commands.json + # Poor man's replacement of `jq` in Python. + extract_build_command_py='import json; print(json.load(open("build/compile_commands.json", "r"))[0]["command"])' + build_command=$(python -c "${extract_build_command_py}") + + # build the list of include directories, only considering the directories in "components/" + include_dirs="" + for token in ${build_command}; do + if [[ "${token}" =~ ^-I.* ]]; then + flags_include_dirs+=" $token" + fi + if [[ ! "${token}" =~ ^-I.*/components/.* ]]; then + continue + fi + newline=$'\n' + include_dirs="${include_dirs}${token##-I}${newline}" + done + + # Append a config value used to check if the sdkconfig included + echo "#define IDF_SDKCONFIG_INCLUDED" >> ${sdkinclude_dir}/sdkconfig.h + export INCLUDE_DIRS=${include_dirs} + flags_include_dirs+=" -I ${sdkinclude_dir}" + export FLAGS_INCLUDE_DIRS=${flags_include_dirs} +} + +# Utility convertion functions used for passing load sequences to processes (in form of one character) +# +# Returns ordinal number of supplied character +ord() { LC_CTYPE=C printf %d "'$1"; } +# Returns a character represented by ASCII code +chr() { printf "\\$(printf "%o" $1)"; } + +# check_headers_in_process +# +# Checks one include files at a time in a batch of headers referenced by indices from stdin +# This function may be called from separate processes (used simple load balancing) +# Periodically checks files from +# Outputs failed files to +function check_headers_in_process +{ + local proc="$1" + local list_of_includes="$2" + local list_of_failed_includes="$3" + + list=(`cat ${list_of_includes}`) + size=${#list[@]} + let batch=$size/$JOBS_LOAD_CHAR_SIZE+1 + + # Reads one char (due to atomicity) from stdin + # -ord(char) signifies an index to the list of files to be processed + while IFS= read -n1 char; do + batch_index=$(ord $char) + let run=$batch_index-$JOBS_LOAD_CHAR_FROM || true + for i in `seq 1 $batch`; do + let file_index=$run*$batch+$i-1 || true + if [ "$file_index" -lt "$size" ]; then + # log_details "$proc: Processing $file_index (batch=$batch, run=$run)\n" + include_file=${list[$file_index]} + rel_inc_file=${include_file##${IDF_PATH}} + printf "Processing ${rel_inc_file}" + check_one_header ${include_file} && ret=0 || ret=$? + case "$ret" in + 0*) printf " OK\n" ;; + 1*) printf " FAIL (missing c++ guard)\n" ;; + 2*) printf " FAIL (cannot compile)\n" ;; + 3*) printf " FAIL (missing include of sdkconfig.h)\n" ;; + esac + if [[ ! "$ret" == 0 ]]; then + echo ${rel_inc_file} >> ${list_of_failed_includes} + fi + fi + done + done < $WORK_ITEM_PIPE; +} + +# check_include_dirs +# +# Check all header files in directories listed in include_dirs variable. +# Places the list of header files where suspects missing C++ guard to file. +# Places the list of header files where compilation issues found to file. +# Calls check_one_header function to check a single header file. +function check_include_dirs +{ + local include_dirs="$1" + local list_of_includes="$2" + local list_of_failed_includes="$3" + local include_dir_maxdepth=${INCLUDE_DIR_MAXDEPTH:-} + local grep_expr=$(mktemp) + + rm -f ${list_of_includes} ${list_of_failed_includes} + touch ${list_of_includes} ${list_of_failed_includes} + + if [ -z "${PROCESS_ALL_INCLUDES:-}" ]; then + # generate grep expr: 1) strip comments, 2) generate '\|\|...' + exclusions=`cat $EXCLUSIONS_LIST | sed '/^#/d'` + echo $exclusions | sed 's/ /\\|/g' > ${grep_expr} + fi; + + for include_dir in ${include_dirs}; do + echo "Processing include directory: ${include_dir}" + include_files=$(find ${include_dir} ${include_dir_maxdepth} -name '*.h') + for include_file in ${include_files}; do + if [ -z "${PROCESS_ALL_INCLUDES:-}" ] && echo "$include_file" | grep -q -f ${grep_expr}; then + continue + fi + echo ${include_file} >> ${list_of_includes} + done + done + rm -f ${grep_expr} +} + +# filter_exclusions +# +# Takes the list of files from stdin, and removes all files which +# are listed in file. +# Saves the final (filtered) list in file. +# The exclusions list file may contain comments and empty lines. +function filter_exclusions +{ + local exclusions_list_file=$1 + local filtered_list_file=$2 + local grep_expr=$(mktemp) + + # generate grep expr: 1) strip comments, 2) generate '\|\|...' + exclusions=`cat $exclusions_list_file | sed '/^#/d'` + echo $exclusions | sed 's/ /\\|/g' > ${grep_expr} + + # pass the expression as file (input goes from stdin) + grep -vf ${grep_expr} > ${filtered_list_file} || true + rm -f ${grep_expr} +} + +##################### main logic starts here ##################### + +EXCLUSIONS_LIST=${IDF_PATH}/tools/ci/check_public_headers_exceptions.txt +FAILED_INCLUDE_LIST=failed.txt +TOTAL_INCLUDE_LIST=include_list.txt +FILTERED_LIST=filtered.txt +LINE=--------------------------------------------------------------- +CC=${CC:-gcc} +CXX=${CXX:-g++} +TEST_PROJECT_DIR=${IDF_PATH}/examples/get-started/blink +MULTI_JOBS="1" +# jobs, load balancing vars, temp outputs +JOBS_LOAD_CHAR_FROM=64 +JOBS_LOAD_CHAR_TO=90 +let JOBS_LOAD_CHAR_SIZE=$JOBS_LOAD_CHAR_TO-$JOBS_LOAD_CHAR_FROM+1 +TEMP_OUT_FILE=/tmp/proc_out.txt. +WORK_ITEM_PIPE=/tmp/work_item_pipe + +# Process command line arguments +for i in $@; do + if [[ "$i" =~ "--verbose" ]]; then + echo "$0: Enabling verbose output" + VERBOSE="1" + fi + if [[ "$i" =~ "--process-all-files" ]]; then + echo "$0: Processing all files" + PROCESS_ALL_INCLUDES="1" + fi + if [[ "$i" =~ --include-maxdepth=* ]]; then + INCLUDE_DIR_MAXDEPTH="-maxdepth ${i##--include-maxdepth=}" + echo "$0: Searching for includes with: $INCLUDE_DIR_MAXDEPTH" + fi + if [[ "$i" =~ --jobs=* ]]; then + MULTI_JOBS="${i##--jobs=}" + echo "$0: Checking headers within multiple jobs: $MULTI_JOBS" + fi +done + +[[ -p $WORK_ITEM_PIPE ]] || mkfifo $WORK_ITEM_PIPE + +echo $LINE +# Get include directories +get_include_dirs_list ${TEST_PROJECT_DIR} +# Generate the main file to compiled while preincluding a header under test +create_test_main ${TEST_PROJECT_DIR}/build +# Get the list of include files to be processed +check_include_dirs "${INCLUDE_DIRS}" ${TOTAL_INCLUDE_LIST} ${FAILED_INCLUDE_LIST} + +# Launching separate processes (if $MULTI_JOBS defined) to check includes +# +# poor man's approach to load balancing: +# 1) sending work items (chars A-Z for atomicity) to a named pipe +# 2) spawning nr of processes to pick the work items from the pipe +# 3) every process reads one char at a time to analyze a batch of includes (batch=total_includes/nr_of_work_items) +# 4) waiting till all processes finish + +# running the jobs and marking the pid +for ((i=1; i<${MULTI_JOBS}; i++)); do + check_headers_in_process $i ${TOTAL_INCLUDE_LIST} ${FAILED_INCLUDE_LIST} > ${TEMP_OUT_FILE}${i} & + pids[${i}]=$! +done +# one job outputs directly to stdout (to show the script is alive) +check_headers_in_process 0 ${TOTAL_INCLUDE_LIST} ${FAILED_INCLUDE_LIST} & +pid0=$! + +# sending sequence of work items +for i in `seq $JOBS_LOAD_CHAR_FROM $JOBS_LOAD_CHAR_TO`; do + printf $(chr $i ) +done > $WORK_ITEM_PIPE + +# waiting for the processes to finish and print their outputs +for ((i=1; i<${MULTI_JOBS}; i++)); do + log_details "\nwaiting for process nr: $i\n" + wait ${pids[$i]} || true + cat ${TEMP_OUT_FILE}${i} + rm -f ${TEMP_OUT_FILE}${i} +done +# and the last one +wait $pid0 || true +rm -f $WORK_ITEM_PIPE + +# Filter out the known issues and report results +cat ${FAILED_INCLUDE_LIST} | filter_exclusions ${EXCLUSIONS_LIST} ${FILTERED_LIST} +echo $LINE + +if [ -s ${FILTERED_LIST} ]; then + echo "The following files failed:" + echo "" + cat ${FILTERED_LIST} + exit 1 +fi + +echo "No errors found" diff --git a/tools/ci/check_cpp_guards_exceptions.txt b/tools/ci/check_public_headers_exceptions.txt similarity index 90% rename from tools/ci/check_cpp_guards_exceptions.txt rename to tools/ci/check_public_headers_exceptions.txt index 8fe96a4420..111bc6639a 100644 --- a/tools/ci/check_cpp_guards_exceptions.txt +++ b/tools/ci/check_public_headers_exceptions.txt @@ -36,8 +36,6 @@ components/lwip/port/esp32/include/arch/cc.h components/vfs/include/sys/dirent.h -components/esp_wifi/include/esp_wifi_netif.h -components/esp_wifi/include/esp_wifi_default.h components/esp_wifi/esp32/include/phy_init_data.h components/spi_flash/include/spi_flash_chip_issi.h @@ -97,3 +95,9 @@ components/app_trace/include/esp_sysview_trace.h components/esp_gdbstub/include/esp_gdbstub.h components/openssl/include/platform/ssl_pm.h +### Here are the files that use CONFIG_XXX values but don't include sdkconfig.h +# +components/heap/include/esp_heap_task_info.h +components/esp_wifi/include/esp_private/wifi_os_adapter.h +components/asio/port/include/esp_exception.h +components/esp_common/include/esp_compiler.h diff --git a/tools/ci/config/pre_check.yml b/tools/ci/config/pre_check.yml index 34bd440b89..855eb2e9d5 100644 --- a/tools/ci/config/pre_check.yml +++ b/tools/ci/config/pre_check.yml @@ -152,7 +152,7 @@ build_template_app: - for dir in esp32 esp32s2beta; do test $(xtensa-esp32-elf-nm $dir/*.a | grep -w ets_printf | wc -l) -eq 0; done; - popd -check_header_files_cpp_guards: +check_public_headers: stage: pre_check image: $CI_DOCKER_REGISTRY/esp32-ci-env$BOT_DOCKER_IMAGE_TAG tags: @@ -165,5 +165,5 @@ check_header_files_cpp_guards: - $BOT_LABEL_BUILD - $BOT_LABEL_REGULAR_TEST script: - - tools/ci/check_cpp_guards.sh --verbose --process-all-files + - CC=xtensa-esp32-elf-gcc CXX=xtensa-esp32-elf-g++ tools/ci/check_public_headers.sh --verbose --process-all-files --jobs=4 diff --git a/tools/ci/executable-list.txt b/tools/ci/executable-list.txt index 443053f0e5..dcedb87043 100644 --- a/tools/ci/executable-list.txt +++ b/tools/ci/executable-list.txt @@ -36,11 +36,11 @@ tools/ci/build_examples_cmake.sh tools/ci/check-executable.sh tools/ci/check-line-endings.sh tools/ci/check_build_warnings.py -tools/ci/check_cpp_guards.sh tools/ci/check_deprecated_kconfigs.py tools/ci/check_examples_cmake_make.sh tools/ci/check_examples_rom_header.sh tools/ci/check_idf_version.sh +tools/ci/check_public_headers.sh tools/ci/check_ut_cmake_make.sh tools/ci/checkout_project_ref.py tools/ci/envsubst.py