Skip to content

Commit

Permalink
Base run-clang-tidy.sh on run-clang-tidy-cached.cc
Browse files Browse the repository at this point in the history
This way, only changed files have to be re-processed.
  • Loading branch information
hzeller committed Nov 9, 2024
1 parent 0076bf8 commit e28fcc8
Show file tree
Hide file tree
Showing 10 changed files with 60 additions and 69 deletions.
26 changes: 16 additions & 10 deletions .github/bin/run-clang-tidy-cached.cc
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ B=${0%%.cc}; [ "$B" -nt "$0" ] || c++ -std=c++17 -o"$B" "$0" && exec "$B" "$@";
// See the License for the specific language governing permissions and
// limitations under the License.

// Location: https://github.com/hzeller/dev-tools (2024-10-18)
// Location: https://github.com/hzeller/dev-tools (2024-11-09)

// Script to run clang-tidy on files in a bazel project while caching the
// results as clang-tidy can be pretty slow. The clang-tidy output messages
Expand All @@ -28,8 +28,9 @@ B=${0%%.cc}; [ "$B" -nt "$0" ] || c++ -std=c++17 -o"$B" "$0" && exec "$B" "$@";
// run-clang-tidy-cached.cc --checks="-*,modernize-use-override" --fix
//
// Note: useful environment variables to configure are
// CLANG_TIDY = binary to run; default would just be clang-tidy.
// CACHE_DIR = where to put the cached content; default ~/.cache
// CLANG_TIDY = binary to run; default would just be clang-tidy.
// CLANG_TIDY_CONFIG = override configuration file in kConfig.clang_tidy_file
// CACHE_DIR = where to put the cached content; default ~/.cache

// This file shall be c++17 self-contained; not using any re2 or absl niceties.
#include <unistd.h>
Expand Down Expand Up @@ -109,7 +110,8 @@ struct ConfigValues {
// that is included by a lot of other files results in lots of reprocessing.
bool revisit_if_any_include_changes = true;

// Clang tidy configuration: clang tidy files with checks.
// Clang tidy configuration: clang tidy files with checks. This can be
// overriden with environment variable CLANG_TIDY_CONFIG
std::string_view clang_tidy_file = ".clang-tidy";
};

Expand Down Expand Up @@ -176,11 +178,15 @@ std::string GetContent(const fs::path &f) {
return GetContent(file_to_read);
}

const char *EnvWithFallback(const char *varname, const char *fallback) {
const char *value = getenv(varname);
std::string_view EnvWithFallback(const char *var, std::string_view fallback) {
const char *value = getenv(var);
return value ? value : fallback;
}

std::string_view GetClangTidyConfig() {
return EnvWithFallback("CLANG_TIDY_CONFIG", kConfig.clang_tidy_file);
}

std::string GetCommandOutput(const std::string &prog) {
return GetContent(popen(prog.c_str(), "r")); // NOLINT
}
Expand Down Expand Up @@ -323,7 +329,7 @@ class ClangTidyRunner {
static std::string AssembleArgs(int argc, char **argv) {
std::string result = " --quiet";
result.append(" '--config-file=")
.append(kConfig.clang_tidy_file)
.append(GetClangTidyConfig())
.append("'");
for (const std::string_view arg : kExtraArgs) {
result.append(" --extra-arg='").append(arg).append("'");
Expand Down Expand Up @@ -352,7 +358,7 @@ class ClangTidyRunner {

// Make sure directory filename depends on .clang-tidy content.
hash_t cache_unique_id = hashContent(version + clang_tidy_args_);
cache_unique_id ^= hashContent(GetContent(kConfig.clang_tidy_file));
cache_unique_id ^= hashContent(GetContent(GetClangTidyConfig()));
return cache_dir / fs::path(cache_prefix + "v" + major_version + "_" +
ToHex(cache_unique_id, 8));
}
Expand Down Expand Up @@ -527,8 +533,8 @@ class FileGatherer {

int main(int argc, char *argv[]) {
// Test that key files exist and remember their last change.
if (!fs::exists(kConfig.clang_tidy_file)) {
std::cerr << "Need a " << kConfig.clang_tidy_file << " config file.\n";
if (!fs::exists(GetClangTidyConfig())) {
std::cerr << "Need a " << GetClangTidyConfig() << " config file.\n";
return EXIT_FAILURE;
}

Expand Down
35 changes: 4 additions & 31 deletions .github/bin/run-clang-tidy.sh
Original file line number Diff line number Diff line change
Expand Up @@ -13,26 +13,17 @@
# See the License for the specific language governing permissions and
# limitations under the License.

# TODO(hzeller): switch to .github/bin/run-clang-tidy-cached.cc
# entirely once we have all the issues addressed.
# It caches the results, so is much faster, but doesn't have the
# concept of the 'limited' clang-tidy.

# Exact binary can be set with environment variable if needed.
CLANG_TIDY="${CLANG_TIDY:-clang-tidy}"

LOCAL_TMP=${TMPDIR:-/tmp}

TIDY_OUT=${LOCAL_TMP}/clang-tidy-surelog.out

hash ${CLANG_TIDY} || exit 2 # make sure it is installed.

if [ "$1" == "limited" ]; then
export CLANG_TIDY_CONFIG="${LOCAL_TMP}/clang-tidy"
# For now, since there are a lot things to fix, don't enable everything
# that is mentioned in .clang-tidy, but add rules as we fix them.
# The more strict .clang-tidy rules will still show up as reminders in the
# IDE, so are hopefully fixed on the way.
cat > ${LOCAL_TMP}/clang-tidy <<EOF
cat > "${CLANG_TIDY_CONFIG}" <<EOF
Checks: >
-*,
bugprone-copy-constructor-init,
Expand Down Expand Up @@ -88,31 +79,13 @@ CheckOptions:
- key: performance-unnecessary-copy-initialization.AllowedTypes
value: ::SURELOG::SymbolId;SURELOG::SymbolId;SymbolId;::SURELOG::NodeId;SURELOG::NodeId;NodeId;::SURELOG::PathId;SURELOG::PathId;PathId
EOF
CLANG_TIDY_OPTS="--config-file=${LOCAL_TMP}/clang-tidy"
fi

CLANG_TIDY_OPTS="${CLANG_TIDY_OPTS} --quiet"

if [ ! -r compile_commands.json ]; then
echo "To get compile_commands.json, run in root of project and "
echo " make run-cmake-release"
exit 1
fi

find src/ include/ -name "*.cpp" -or -name "*.h" \
| grep -v Python | grep -v Constraint.h | grep -v "hello.*\.cpp" \
| xargs -P$(nproc) -n 5 -- ${CLANG_TIDY} ${CLANG_TIDY_OPTS} 2>/dev/null \
> ${TIDY_OUT}

cat ${TIDY_OUT}

sed 's|\(.*\)\(\[[a-zA-Z.-]*\]$\)|\2|p;d' < ${TIDY_OUT} \
| sort | uniq -c | sort -rn

if [ -s ${TIDY_OUT} ]; then
echo "There were clang-tidy warnings (see ${TIDY_OUT}). Please fix."
exit 1
fi

echo "No clang-tidy complaints.😎"
exit 0
# Invoke run-clang-tidy-cached, possibly with the 'limited' clang-tidy config.
sh $(dirname $0)/run-clang-tidy-cached.cc
28 changes: 19 additions & 9 deletions .github/workflows/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -1075,7 +1075,7 @@ jobs:
- name: Install Dependencies
run: |
sudo apt-get update -qq
sudo apt -qq -y install clang-tidy-18 \
sudo apt -qq -y install clang-tidy-18 ccache \
g++-9 default-jre cmake \
uuid-dev build-essential
Expand All @@ -1096,25 +1096,35 @@ jobs:
submodules: recursive
fetch-depth: 0

- name: Use ccache
uses: hendrikmuhs/ccache-action@v1.2
- name: Create Cache Timestamp
id: cache_timestamp
uses: nanzm/get-time-action@v2.0
with:
format: 'YYYY-MM-DD-HH-mm-ss'

- name: Retrieve cached results
uses: actions/cache@v3
with:
key: clang-tidy-codegen
path: |
/home/runner/.cache/clang-tidy
/home/runner/.cache/ccache
key: clang-tidy-${{ steps.cache_timestamp.outputs.time }}
restore-keys: clang-tidy-

- name: Configure shell
run: |
echo 'PATH=/usr/lib/ccache:'"$PATH" >> $GITHUB_ENV
- name: Prepare source
run: |
make run-cmake-release
make -j2 -C build GenerateParser
make -j2 -C build GenerateParserListeners
make -j2 -C build GenerateCacheSerializers
ccache -p
make run-cmake-release-with-python
make -j2 -C build surelog
ln -sf build/compile_commands.json .
- name: Run clang tidy
run: |
export CLANG_TIDY=clang-tidy-18
"${CLANG_TIDY}" --version
./.github/bin/run-clang-tidy.sh limited
./.github/bin/run-clang-tidy.sh limited \
|| ( cat Surelog_clang-tidy.out ; exit 1)
3 changes: 2 additions & 1 deletion include/Surelog/API/PythonAPI.h
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,8 @@ class PythonAPI {
const std::string& function,
const std::vector<std::string>& args,
PyThreadState* interp);
static void evalScript(std::string function, SV3_1aPythonListener* listener,
static void evalScript(const std::string& function,
SV3_1aPythonListener* listener,
parser_rule_context* ctx);
static std::string getInvalidScriptString() { return m_invalidScriptResult; }
static bool isListenerLoaded() { return m_listenerLoaded; }
Expand Down
2 changes: 1 addition & 1 deletion include/Surelog/SourceCompile/CompileSourceFile.h
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@

#ifdef SURELOG_WITH_PYTHON
struct _ts;
typedef struct _ts PyThreadState;
using PyThreadState = struct _ts;
#endif

namespace SURELOG {
Expand Down
4 changes: 2 additions & 2 deletions include/Surelog/SourceCompile/PythonListen.h
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,8 @@ class SV3_1aPythonListener;
class PythonListen {
public:
PythonListen(ParseFile* parse, CompileSourceFile* m_compileSourceFile);
PythonListen(const PythonListen& orig) = delete;

bool listen();

virtual ~PythonListen();
Expand All @@ -47,8 +49,6 @@ class PythonListen {
void addError(Error& error);

private:
PythonListen(const PythonListen& orig) = delete;

ParseFile* const m_parse;
CompileSourceFile* const m_compileSourceFile;
std::vector<SV3_1aPythonListener*> m_pythonListeners;
Expand Down
2 changes: 1 addition & 1 deletion scripts/generate_python_listener.py
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,7 @@ def _main():
' PythonListen* getPythonListen() { return m_pl; }',
' antlr4::CommonTokenStream* getTokenStream() { return m_tokens; }',
'',
' void logError(ErrorDefinition::ErrorType error, antlr4::ParserRuleContext* ctx, std::string object, bool printColumn = false);',
' void logError(ErrorDefinition::ErrorType error, antlr4::ParserRuleContext* ctx, const std::string& object, bool printColumn = false);',
' void logError(ErrorDefinition::ErrorType, Location& loc, bool showDuplicates = false);',
' void logError(ErrorDefinition::ErrorType, Location& loc, Location& extraLoc, bool showDuplicates = false);',
'',
Expand Down
3 changes: 2 additions & 1 deletion src/API/PythonAPI.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -275,7 +275,8 @@ void PythonAPI::init(int32_t argc, const char** argv) {
#endif
}

void PythonAPI::evalScript(std::string function, SV3_1aPythonListener* listener,
void PythonAPI::evalScript(const std::string& function,
SV3_1aPythonListener* listener,
parser_rule_context* ctx1) {
#ifdef SURELOG_WITH_PYTHON
antlr4::ParserRuleContext* ctx = (antlr4::ParserRuleContext*)ctx1;
Expand Down
8 changes: 5 additions & 3 deletions src/API/SV3_1aPythonListener.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -38,13 +38,15 @@ SV3_1aPythonListener::SV3_1aPythonListener(PythonListen* pl,
m_tokens(tokens),
m_lineOffset(lineOffset) {}

SV3_1aPythonListener::SV3_1aPythonListener(const SV3_1aPythonListener& orig) {}
SV3_1aPythonListener::SV3_1aPythonListener(const SV3_1aPythonListener& orig) =
default;

SV3_1aPythonListener::~SV3_1aPythonListener() {}
SV3_1aPythonListener::~SV3_1aPythonListener() = default;

void SV3_1aPythonListener::logError(ErrorDefinition::ErrorType error,
antlr4::ParserRuleContext* ctx,
std::string object, bool printColumn) {
const std::string& object,
bool printColumn) {
ParseUtils::LineColumn lineCol =
ParseUtils::getLineColumn(getTokenStream(), ctx);

Expand Down
18 changes: 8 additions & 10 deletions src/SourceCompile/PythonListen.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ PythonListen::PythonListen(ParseFile* parse,
m_compileSourceFile(compileSourceFile),
m_usingCachedVersion(false) {}

PythonListen::~PythonListen() {}
PythonListen::~PythonListen() = default;

void PythonListen::addError(Error& error) {
getCompileSourceFile()->getErrorContainer()->addError(error);
Expand All @@ -51,8 +51,8 @@ bool PythonListen::listen() {
}

// This is either a parent Parser object of this Parser object has no parent
if ((m_parse->m_children.size() != 0) || (m_parse->m_parent == nullptr)) {
if ((m_parse->m_parent == nullptr) && (m_parse->m_children.size() == 0)) {
if ((!m_parse->m_children.empty()) || (m_parse->m_parent == nullptr)) {
if ((m_parse->m_parent == nullptr) && (m_parse->m_children.empty())) {
SV3_1aPythonListener* pythonListener =
new SV3_1aPythonListener(this, m_compileSourceFile->getPythonInterp(),
m_parse->m_antlrParserHandler->m_tokens, 0);
Expand All @@ -61,20 +61,18 @@ bool PythonListen::listen() {
pythonListener, m_parse->m_antlrParserHandler->m_tree);
}

if (m_parse->m_children.size() != 0) {
for (uint32_t i = 0; i < m_parse->m_children.size(); i++) {
if (m_parse->m_children[i]->m_antlrParserHandler) {
if (!m_parse->m_children.empty()) {
for (const ParseFile* child : m_parse->m_children) {
if (child->m_antlrParserHandler) {
// Only visit the chunks that got re-parsed
// TODO: Incrementally regenerate the FileContent

SV3_1aPythonListener* pythonListener = new SV3_1aPythonListener(
this, m_compileSourceFile->getPythonInterp(),
m_parse->m_children[i]->m_antlrParserHandler->m_tokens,
m_parse->m_children[i]->m_offsetLine);
child->m_antlrParserHandler->m_tokens, child->m_offsetLine);
m_pythonListeners.push_back(pythonListener);
antlr4::tree::ParseTreeWalker::DEFAULT.walk(
pythonListener,
m_parse->m_children[i]->m_antlrParserHandler->m_tree);
pythonListener, child->m_antlrParserHandler->m_tree);
}
}
}
Expand Down

0 comments on commit e28fcc8

Please sign in to comment.