From 91b17c93b18dc70fd2113ced625f967347e1a95f Mon Sep 17 00:00:00 2001 From: Adrien4193 <39053578+Adrien4193@users.noreply.github.com> Date: Mon, 9 Sep 2024 16:34:27 +0200 Subject: [PATCH] BRAYNS-649 Add clang tidy (#1284) --- .clang-format | 47 ++++++++++++++--- .clang-tidy | 57 +++++++++++++++++++++ .github/workflows/linter.yaml | 11 +++- README.md | 20 ++++++++ scripts/install_llvm.sh | 12 +++++ src/brayns/core/Launcher.h | 2 +- src/brayns/core/cli/CommandLine.h | 10 ++-- src/brayns/core/codecs/PngCodec.cpp | 2 +- src/brayns/core/engine/Data.h | 6 +-- src/brayns/core/json/JsonSchema.h | 2 +- src/brayns/core/jsonrpc/Errors.cpp | 16 +++--- src/brayns/core/jsonrpc/PayloadReflector.h | 8 +-- src/brayns/core/objects/common/Binary.h | 13 ++--- src/brayns/core/utils/Binary.h | 12 +++-- src/brayns/core/utils/Filesystem.cpp | 7 ++- src/brayns/core/utils/Math.h | 2 - src/brayns/core/websocket/WebSocketServer.h | 2 +- tests/core/codecs/TestCodecs.cpp | 2 +- tests/core/engine/TestRender.cpp | 4 +- tests/core/utils/TestBinary.cpp | 2 +- 20 files changed, 188 insertions(+), 49 deletions(-) create mode 100644 .clang-tidy create mode 100644 scripts/install_llvm.sh diff --git a/.clang-format b/.clang-format index 149f142e0..ce3992435 100644 --- a/.clang-format +++ b/.clang-format @@ -11,10 +11,10 @@ AlignEscapedNewlines: DontAlign AlignOperands: DontAlign AlignTrailingComments: false AllowAllArgumentsOnNextLine: false -AllowAllConstructorInitializersOnNextLine: false # Deprecated AllowAllParametersOfDeclarationOnNextLine: false AllowShortBlocksOnASingleLine: Never AllowShortCaseLabelsOnASingleLine: false +AllowShortCompoundRequirementOnASingleLine: true AllowShortEnumsOnASingleLine: false AllowShortFunctionsOnASingleLine: None AllowShortIfStatementsOnASingleLine: Never @@ -26,13 +26,35 @@ AlwaysBreakTemplateDeclarations: Yes BinPackArguments: false BinPackParameters: false BitFieldColonSpacing: Both +BraceWrapping: + AfterCaseLabel: true + AfterClass: true + AfterControlStatement: Always + AfterEnum: true + AfterFunction: true + AfterNamespace: true + AfterStruct: true + AfterUnion: true + AfterExternBlock: true + BeforeCatch: true + BeforeElse: true + BeforeLambdaBody: true + BeforeWhile: true + IndentBraces: false + SplitEmptyRecord: true + SplitEmptyNamespace: true +BreakAdjacentStringLiterals: true +BreakAfterAttributes: Never +BreakAfterReturnType: Automatic BreakBeforeBinaryOperators: NonAssignment -BreakBeforeBraces: Allman -BreakBeforeConceptDeclarations: true +BreakBeforeBraces: Custom +BreakBeforeConceptDeclarations: Always BreakBeforeTernaryOperators: true BreakConstructorInitializers: AfterColon +BreakFunctionDefinitionParameters: false BreakInheritanceList: AfterColon -BreakStringLiterals: true +BreakStringLiterals: false +BreakTemplateDeclarations: Yes ColumnLimit: 120 CompactNamespaces: false ConstructorInitializerIndentWidth: 4 @@ -50,12 +72,21 @@ IndentCaseLabels: false IndentExternBlock: true IndentGotoLabels: false IndentPPDirectives: BeforeHash -IndentRequires: true +IndentRequiresClause: true IndentWidth: 4 IndentWrappedFunctionNames: true -KeepEmptyLinesAtTheStartOfBlocks: false +InsertBraces: true +IntegerLiteralSeparator: + Binary: -1 + Decimal: 3 + Hex: -1 +KeepEmptyLines: + AtEndOfFile: false + AtStartOfBlock: false + AtStartOfFile: false LambdaBodyIndentation: Signature Language: Cpp +MainIncludeChar: Quote MaxEmptyLinesToKeep: 1 NamespaceIndentation: None PackConstructorInitializers: Never @@ -73,7 +104,9 @@ PointerAlignment: Right QualifierAlignment: Custom QualifierOrder: ["static", "inline", "const", "volatile", "type"] ReferenceAlignment: Right -ReflowComments: true +ReflowComments: false +RequiresClausePosition: WithPreceding +RequiresExpressionIndentation: OuterScope SeparateDefinitionBlocks: Always ShortNamespaceLines: 1 SortIncludes: true diff --git a/.clang-tidy b/.clang-tidy new file mode 100644 index 000000000..1c945da2f --- /dev/null +++ b/.clang-tidy @@ -0,0 +1,57 @@ +Checks: -* + bugprone-* + cert-* + concurrency-* + cppcoreguidelines-* + misc-* + modernize-* + performance-* + portability-* + readability-* + -cppcoreguidelines-avoid-magic-numbers + -cppcoreguidelines-missing-std-forward + -cppcoreguidelines-pro-* + -misc-include-cleaner + -modernize-use-nodiscard + -modernize-use-trailing-return-type + -readability-identifier-length + -readability-magic-numbers + +WarningsAsErrors: "*" +HeaderFilterRegex: "src/.*" + +CheckOptions: + - key: readability-identifier-naming.ClassCase + value: CamelCase + - key: readability-identifier-naming.ClassMemberCase + value: CamelCase + - key: readability-identifier-naming.ConstexprVariableCase + value: CamelCase + - key: readability-identifier-naming.EnumCase + value: CamelCase + - key: readability-identifier-naming.EnumConstantCase + value: CamelCase + - key: readability-identifier-naming.FunctionCase + value: camelBack + - key: readability-identifier-naming.FunctionIgnoredRegexp + value: main|wmain|wWinMain|begin|end + - key: readability-identifier-naming.GlobalConstantCase + value: camelBack + - key: readability-identifier-naming.StaticConstantCase + value: camelBack + - key: readability-identifier-naming.StaticVariableCase + value: camelBack + - key: readability-identifier-naming.MacroDefinitionCase + value: UPPER_CASE + - key: readability-identifier-naming.MemberCase + value: camelBack + - key: readability-identifier-naming.PrivateMemberPrefix + value: _ + - key: readability-identifier-naming.NamespaceCase + value: camelBack + - key: readability-identifier-naming.ParameterCase + value: camelBack + - key: readability-identifier-naming.TypeAliasCase + value: CamelCase + - key: readability-identifier-naming.VariableCase + value: camelBack diff --git a/.github/workflows/linter.yaml b/.github/workflows/linter.yaml index 89cf44a90..b13c165c9 100644 --- a/.github/workflows/linter.yaml +++ b/.github/workflows/linter.yaml @@ -11,8 +11,15 @@ jobs: steps: - name: Checkout repository uses: actions/checkout@v4 + - name: Update APT package list + run: sudo apt-get update + - name: Download and install LLVM + run: | + wget https://apt.llvm.org/llvm.sh + sudo bash llvm.sh 20 + - name: Install clang-format + run: sudo apt-get -y install clang-format-20 - name: Run clang-format run: | - apt update && apt install -y clang-format-15 SOURCES=$(find apps src tests \( -name "*.h" -or -name "*.cpp" \)) - clang-format-15 --dry-run --Werror $SOURCES + clang-format-20 --dry-run --Werror $SOURCES diff --git a/README.md b/README.md index efa661877..6a98dee7e 100644 --- a/README.md +++ b/README.md @@ -35,6 +35,26 @@ The following cmake options (shown with their default value) can be used during * **BRAYNS_ENABLE_CIRCUITS** (Default ON) - Activate circuit support. * **BRAYNS_ENABLE_ATLAS** - (Default ON) Activate atlas support. +### Run linters + +Run clang format as follows: + + $ SOURCES=$(find apps src tests \( -name "*.h" -or -name "*.cpp" \)) + $ clang-format-20 --Werror $SOURCES + +Run clang tidy as follows: + +Use cmake config to generate build commands: + +"cacheVariables": { + "CMAKE_EXPORT_COMPILE_COMMANDS": true, + "CMAKE_CXX_STANDARD_INCLUDE_DIRECTORIES": "/usr/include/c++/11;/usr/include/x86_64-linux-gnu/c++/11" +} + +Run with + + $ run-clang-tidy-20 -p build/user-debug + ## Running diff --git a/scripts/install_llvm.sh b/scripts/install_llvm.sh new file mode 100644 index 000000000..ce6848d30 --- /dev/null +++ b/scripts/install_llvm.sh @@ -0,0 +1,12 @@ +# Copyright (c) 2015-2024, EPFL/Blue Brain Project +# All rights reserved. Do not distribute without permission. +# Responsible Author: Adrien Fleury +# +# This file is part of Brayns + +wget https://apt.llvm.org/llvm.sh +chmod +x llvm.sh +./llvm.sh 20 + +apt-get install clang-format-20 +apt-get install clang-tidy-20 diff --git a/src/brayns/core/Launcher.h b/src/brayns/core/Launcher.h index aa8aa4c1c..c4034fc72 100644 --- a/src/brayns/core/Launcher.h +++ b/src/brayns/core/Launcher.h @@ -71,7 +71,7 @@ struct ArgvSettingsReflector .defaultValue("localhost"); builder.option("port", [](auto &settings) { return &settings.port; }) .description("Websocket server port") - .defaultValue(5000); + .defaultValue(5'000); builder.option("max-thread-count", [](auto &settings) { return &settings.maxThreadCount; }) .description("Maximum number of threads for the websocket server") .defaultValue(10); diff --git a/src/brayns/core/cli/CommandLine.h b/src/brayns/core/cli/CommandLine.h index 9eb61c1f0..e979afc04 100644 --- a/src/brayns/core/cli/CommandLine.h +++ b/src/brayns/core/cli/CommandLine.h @@ -131,9 +131,11 @@ template struct ArgvReflector; template -concept ReflectedArgvOption = std::same_as::getType())> - && std::same_as::toString(T()))> - && std::same_as::parse(std::string_view()))>; +concept ReflectedArgvOption = requires(T value) { + { ArgvReflector::getType() } -> std::same_as; + { ArgvReflector::toString(value) } -> std::same_as; + { ArgvReflector::parse(std::string_view()) } -> std::same_as; +}; template<> struct ArgvReflector @@ -160,7 +162,7 @@ struct ArgvReflector }; template - requires std::is_arithmetic_v +requires std::is_arithmetic_v struct ArgvReflector { static std::string getType() diff --git a/src/brayns/core/codecs/PngCodec.cpp b/src/brayns/core/codecs/PngCodec.cpp index a0884dcd2..0d3448032 100644 --- a/src/brayns/core/codecs/PngCodec.cpp +++ b/src/brayns/core/codecs/PngCodec.cpp @@ -65,7 +65,7 @@ std::string encodePng(const ImageView &image) .message = {}, }; - auto rowStride = PNG_IMAGE_ROW_STRIDE(png); + auto rowStride = png_int_32(PNG_IMAGE_ROW_STRIDE(png)); if (rowOrder == RowOrder::BottomUp) { diff --git a/src/brayns/core/engine/Data.h b/src/brayns/core/engine/Data.h index 0eb573643..ce1d9d624 100644 --- a/src/brayns/core/engine/Data.h +++ b/src/brayns/core/engine/Data.h @@ -178,7 +178,7 @@ Data allocateData(Device &device, std::size_t itemCount) } template - requires(std::convertible_to, T>) +requires std::convertible_to, T> Data createData(Device &device, U &&items) { auto data = allocateData(device, items.size()); @@ -249,7 +249,7 @@ Data3D createDataView3D(const Data3D &data, const DataRegion3D &r template Data2D createDataView2D(const Data2D &data, const DataRegion2D ®ion) { - auto region3D = DataRegion3D{Size3(region.itemCount, 1), Size3(region.stride, 0), region.offset}; + auto region3D = DataRegion3D{Size3(region.itemCount, 1), Stride3(region.stride, 0), region.offset}; auto view = createDataView3D(data, region3D); return Data2D(std::move(view)); } @@ -257,7 +257,7 @@ Data2D createDataView2D(const Data2D &data, const DataRegion2D &r template Data createDataView(const Data &data, const DataRegion ®ion) { - auto region3D = DataRegion3D{Size3(region.itemCount, 1, 1), Size3(region.stride, 0, 0), region.offset}; + auto region3D = DataRegion3D{Size3(region.itemCount, 1, 1), Stride3(region.stride, 0, 0), region.offset}; auto view = createDataView3D(data, region3D); return Data(std::move(view)); } diff --git a/src/brayns/core/json/JsonSchema.h b/src/brayns/core/json/JsonSchema.h index f93a2ffca..f45c37607 100644 --- a/src/brayns/core/json/JsonSchema.h +++ b/src/brayns/core/json/JsonSchema.h @@ -173,6 +173,6 @@ struct JsonSchema std::optional maxItems = {}; std::map properties = {}; - auto operator<=>(const JsonSchema &) const = default; + bool operator==(const JsonSchema &) const = default; }; } diff --git a/src/brayns/core/jsonrpc/Errors.cpp b/src/brayns/core/jsonrpc/Errors.cpp index ee8197e00..0b2d48bb9 100644 --- a/src/brayns/core/jsonrpc/Errors.cpp +++ b/src/brayns/core/jsonrpc/Errors.cpp @@ -58,42 +58,42 @@ const JsonValue &JsonRpcException::getData() const } ParseError::ParseError(const std::string &message): - JsonRpcException(-32700, message) + JsonRpcException(-32'700, message) { } InvalidRequest::InvalidRequest(const std::string &message): - JsonRpcException(-32600, message) + JsonRpcException(-32'600, message) { } InvalidRequest::InvalidRequest(const std::string &message, const std::vector &errors): - JsonRpcException(-32600, message, serializeToJson(errors)) + JsonRpcException(-32'600, message, serializeToJson(errors)) { } MethodNotFound::MethodNotFound(const std::string &method): - JsonRpcException(-32601, fmt::format("Method not found: '{}'", method)) + JsonRpcException(-32'601, fmt::format("Method not found: '{}'", method)) { } InvalidParams::InvalidParams(const std::string &message): - JsonRpcException(-32602, message) + JsonRpcException(-32'602, message) { } InvalidParams::InvalidParams(const std::string &message, const std::vector &errors): - JsonRpcException(-32602, message, serializeToJson(errors)) + JsonRpcException(-32'602, message, serializeToJson(errors)) { } InternalError::InternalError(const std::string &message): - JsonRpcException(-32603, message) + JsonRpcException(-32'603, message) { } InternalError::InternalError(const std::exception &e): - JsonRpcException(-32603, e.what()) + JsonRpcException(-32'603, e.what()) { } } diff --git a/src/brayns/core/jsonrpc/PayloadReflector.h b/src/brayns/core/jsonrpc/PayloadReflector.h index 9a942443b..a734c26bb 100644 --- a/src/brayns/core/jsonrpc/PayloadReflector.h +++ b/src/brayns/core/jsonrpc/PayloadReflector.h @@ -35,9 +35,11 @@ template struct PayloadReflector; template -concept ReflectedPayload = std::same_as::getSchema())> - && std::same_as::serialize(std::declval()))> - && std::same_as::deserialize(Payload()))>; +concept ReflectedPayload = requires(T value) { + { PayloadReflector::getSchema() } -> std::same_as; + { PayloadReflector::serialize(std::move(value)) } -> std::same_as; + { PayloadReflector::deserialize(Payload()) } -> std::same_as; +}; template<> struct PayloadReflector diff --git a/src/brayns/core/objects/common/Binary.h b/src/brayns/core/objects/common/Binary.h index 83e3489e0..dfc2b74b2 100644 --- a/src/brayns/core/objects/common/Binary.h +++ b/src/brayns/core/objects/common/Binary.h @@ -70,12 +70,13 @@ Data3D createData3DFromBinaryOf(Device &device, const Size3 &itemCount, std:: if (reduceMultiply(itemCount) != totalItemCount) { - throw InvalidParams(fmt::format( - "Item count in binary {} does not match given item count {}x{}x{}", - totalItemCount, - itemCount.x, - itemCount.y, - itemCount.z)); + throw InvalidParams( + fmt::format( + "Item count in binary {} does not match given item count {}x{}x{}", + totalItemCount, + itemCount.x, + itemCount.y, + itemCount.z)); } auto data = allocateData3D(device, itemCount); diff --git a/src/brayns/core/utils/Binary.h b/src/brayns/core/utils/Binary.h index b4ca035fb..46a60a1ff 100644 --- a/src/brayns/core/utils/Binary.h +++ b/src/brayns/core/utils/Binary.h @@ -90,7 +90,9 @@ void composeBytesAsPrimtive(const T &value, std::endian endian, std::string &out if (endian != std::endian::native) { - auto first = output.begin() + output.size() - size; + auto offset = static_cast(output.size() - size); + + auto first = output.begin() + offset; auto last = output.end(); auto appended = std::span(first, last); @@ -166,14 +168,16 @@ std::string composeBytes(const T &value, std::endian endian) template struct BinaryReflector> { + static constexpr auto componentCount = static_cast(S); + static std::size_t getSize() { - return static_cast(S) * sizeof(T); + return componentCount * sizeof(T); } static void extract(std::string_view &bytes, std::endian endian, Vector &output) { - for (auto i = 0; i < S; ++i) + for (auto i = std::size_t(0); i < componentCount; ++i) { extractBytesTo(bytes, endian, output[i]); } @@ -181,7 +185,7 @@ struct BinaryReflector> static void compose(const Vector &value, std::endian endian, std::string &output) { - for (auto i = 0; i < S; ++i) + for (auto i = std::size_t(0); i < componentCount; ++i) { composeBytesTo(value[i], endian, output); } diff --git a/src/brayns/core/utils/Filesystem.cpp b/src/brayns/core/utils/Filesystem.cpp index c9460e42b..b6aae8ce2 100644 --- a/src/brayns/core/utils/Filesystem.cpp +++ b/src/brayns/core/utils/Filesystem.cpp @@ -49,7 +49,7 @@ std::string readFile(const std::filesystem::path &path) } auto size = stream.tellg(); - auto data = std::string(size, '\0'); + auto data = std::string(static_cast(size), '\0'); stream.seekg(0); stream.read(data.data(), size); @@ -66,6 +66,9 @@ void writeFile(std::string_view data, const std::filesystem::path &path) throw std::runtime_error(fmt::format("Failed to open file '{}' in write mode", path)); } - stream.write(data.data(), data.size()); + const auto *ptr = data.data(); + auto size = static_cast(data.size()); + + stream.write(ptr, size); } } diff --git a/src/brayns/core/utils/Math.h b/src/brayns/core/utils/Math.h index ddb8d26e0..26aa79832 100644 --- a/src/brayns/core/utils/Math.h +++ b/src/brayns/core/utils/Math.h @@ -110,8 +110,6 @@ struct Transform Vector3 translation = {0.0F, 0.0F, 0.0F}; Quaternion rotation = {1.0F, 0.0F, 0.0F, 0.0F}; Vector3 scale = {1.0F, 1.0F, 1.0F}; - - auto operator<=>(const Transform &other) const = default; }; inline Affine3 toAffine(const Transform &transform) diff --git a/src/brayns/core/websocket/WebSocketServer.h b/src/brayns/core/websocket/WebSocketServer.h index 298403cc8..c70581cb6 100644 --- a/src/brayns/core/websocket/WebSocketServer.h +++ b/src/brayns/core/websocket/WebSocketServer.h @@ -45,7 +45,7 @@ struct SslSettings struct WebSocketServerSettings { std::string host = "localhost"; - std::uint16_t port = 5000; + std::uint16_t port = 5'000; std::size_t maxThreadCount = 1; std::size_t maxQueueSize = 64; std::size_t maxFrameSize = std::numeric_limits::max(); diff --git a/tests/core/codecs/TestCodecs.cpp b/tests/core/codecs/TestCodecs.cpp index 097257503..3f519da33 100644 --- a/tests/core/codecs/TestCodecs.cpp +++ b/tests/core/codecs/TestCodecs.cpp @@ -39,7 +39,7 @@ TestImage createTestImage(ImageFormat format) { auto width = std::size_t(200); auto height = std::size_t(100); - auto pixelSize = format == ImageFormat::Rgb8 ? 3 : 4; + auto pixelSize = format == ImageFormat::Rgb8 ? std::size_t(3) : std::size_t(4); auto rowSize = width * pixelSize; auto size = rowSize * height; auto rowOrder = RowOrder::BottomUp; diff --git a/tests/core/engine/TestRender.cpp b/tests/core/engine/TestRender.cpp index 2c56ff74e..aa52d2034 100644 --- a/tests/core/engine/TestRender.cpp +++ b/tests/core/engine/TestRender.cpp @@ -73,10 +73,10 @@ TEST_CASE("Object creation") createVolumetricModel(device, {volume, transferFunction}); auto triangles = std::vector{{0, 0, 0}, {1, 1, 1}, {2, 2, 2}}; - createTriangleMesh(device, {createData(device, triangles)}); + createTriangleMesh(device, {{createData(device, triangles)}}); auto quads = std::vector{{0, 0, 0}, {1, 1, 1}, {2, 2, 2}, {3, 3, 3}}; - createQuadMesh(device, {createData(device, quads)}); + createQuadMesh(device, {{createData(device, quads)}}); auto spheres = std::vector{{0, 0, 0, 1}, {1, 1, 1, 1}}; auto geometry = createSpheres(device, {createData(device, spheres)}); diff --git a/tests/core/utils/TestBinary.cpp b/tests/core/utils/TestBinary.cpp index 0e1d13e0f..a5c796bc0 100644 --- a/tests/core/utils/TestBinary.cpp +++ b/tests/core/utils/TestBinary.cpp @@ -47,7 +47,7 @@ TEST_CASE("As bytes") swapBytes(test); - CHECK_EQ(test, 16777216); + CHECK_EQ(test, 16'777'216); swapBytes(test);