Skip to content

Commit f117a0e

Browse files
add cppcheck package (#10)
Cppcheck 2.19.1 static analysis tool, built from source. Includes the MISRA C addon, cfg files, and platform definitions for opendbc and panda safety analysis. Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
1 parent bf379a6 commit f117a0e

File tree

5 files changed

+138
-0
lines changed

5 files changed

+138
-0
lines changed

.gitignore

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
.git/
22
repo/
33
.venv/
4+
__pycache__/
45

56
# agents
67
.claude/

cppcheck/build.sh

Lines changed: 42 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
#!/usr/bin/env bash
2+
set -e
3+
4+
DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" >/dev/null && pwd)"
5+
cd "$DIR"
6+
7+
VERSION="2.19.1"
8+
INSTALL_DIR="$DIR/cppcheck/install"
9+
10+
# Idempotent: skip if already built
11+
if [ -x "$INSTALL_DIR/cppcheck" ]; then
12+
echo "cppcheck already present, skipping build."
13+
exit 0
14+
fi
15+
16+
NJOBS="$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 2)"
17+
18+
# Clone
19+
if [ ! -d "cppcheck-src" ]; then
20+
git clone --depth 1 --branch "$VERSION" https://github.com/danmar/cppcheck.git cppcheck-src
21+
fi
22+
23+
# Build
24+
cd cppcheck-src
25+
make MATCHCOMPILER=yes CXXFLAGS="-O2" -j"$NJOBS"
26+
cd "$DIR"
27+
28+
# Install
29+
rm -rf "$INSTALL_DIR"
30+
mkdir -p "$INSTALL_DIR"
31+
32+
cp cppcheck-src/cppcheck "$INSTALL_DIR/"
33+
cp -r cppcheck-src/addons "$INSTALL_DIR/"
34+
cp -r cppcheck-src/cfg "$INSTALL_DIR/"
35+
cp -r cppcheck-src/platforms "$INSTALL_DIR/"
36+
strip "$INSTALL_DIR/cppcheck" 2>/dev/null || true
37+
38+
# Clean up
39+
rm -rf cppcheck-src
40+
41+
echo "Installed cppcheck to $INSTALL_DIR"
42+
du -sh "$INSTALL_DIR"

cppcheck/cppcheck/__init__.py

Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import os
2+
import sys
3+
4+
DIR = os.path.join(os.path.dirname(__file__), "install")
5+
6+
7+
def _run():
8+
binary = os.path.join(DIR, "cppcheck")
9+
os.execvp(binary, ["cppcheck"] + sys.argv[1:])
10+
11+
12+
def smoketest():
13+
import subprocess
14+
binary = os.path.join(DIR, "cppcheck")
15+
result = subprocess.run([binary, "--version"], capture_output=True, text=True, check=True)
16+
print(result.stdout.strip())

cppcheck/pyproject.toml

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
[build-system]
2+
requires = ["setuptools>=64", "wheel"]
3+
build-backend = "setuptools.build_meta"
4+
5+
[project]
6+
name = "cppcheck"
7+
version = "2.19.1"
8+
description = "Cppcheck static analysis tool"
9+
requires-python = ">=3.8"
10+
11+
[project.scripts]
12+
cppcheck = "cppcheck:_run"
13+
14+
[tool.setuptools.packages.find]
15+
include = ["cppcheck*"]
16+
17+
[tool.setuptools.package-data]
18+
cppcheck = ["install/**/*"]

cppcheck/setup.py

Lines changed: 61 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import os
2+
import platform
3+
import subprocess
4+
5+
from setuptools.command.build_py import build_py
6+
7+
try:
8+
from wheel.bdist_wheel import bdist_wheel
9+
except ImportError:
10+
bdist_wheel = None
11+
12+
13+
class BuildCppcheck(build_py):
14+
"""Run build.sh to compile cppcheck before collecting package data."""
15+
16+
def run(self):
17+
pkg_dir = os.path.dirname(os.path.abspath(__file__))
18+
marker = os.path.join(pkg_dir, "cppcheck", "install", "cppcheck")
19+
20+
if not os.path.exists(marker):
21+
build_script = os.path.join(pkg_dir, "build.sh")
22+
subprocess.check_call(["bash", build_script], cwd=pkg_dir)
23+
24+
super().run()
25+
26+
27+
cmdclass = {"build_py": BuildCppcheck}
28+
29+
if bdist_wheel is not None:
30+
31+
class PlatformWheel(bdist_wheel):
32+
"""Produce a platform-specific, Python-version-agnostic wheel."""
33+
34+
def finalize_options(self):
35+
super().finalize_options()
36+
self.root_is_pure = False
37+
38+
def get_tag(self):
39+
system = platform.system()
40+
machine = platform.machine()
41+
42+
if system == "Linux":
43+
plat = f"linux_{machine}"
44+
elif system == "Darwin":
45+
plat = "macosx_11_0_arm64"
46+
else:
47+
plat = f"{system.lower()}_{machine}"
48+
49+
return "py3", "none", plat
50+
51+
cmdclass["bdist_wheel"] = PlatformWheel
52+
53+
54+
def setup():
55+
from setuptools import setup as _setup
56+
57+
_setup(cmdclass=cmdclass)
58+
59+
60+
if __name__ == "__main__":
61+
setup()

0 commit comments

Comments
 (0)