xref: /llvm-project/clang-tools-extra/clang-tidy/altera/StructPackAlignCheck.cpp (revision fc2a9ad10e21bda3dafbb85d8317ef5e3e5f99a1)
1156b1279SFrank Derry Wanye //===--- StructPackAlignCheck.cpp - clang-tidy ----------------------------===//
2156b1279SFrank Derry Wanye //
3156b1279SFrank Derry Wanye // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4156b1279SFrank Derry Wanye // See https://llvm.org/LICENSE.txt for license information.
5156b1279SFrank Derry Wanye // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6156b1279SFrank Derry Wanye //
7156b1279SFrank Derry Wanye //===----------------------------------------------------------------------===//
8156b1279SFrank Derry Wanye 
9156b1279SFrank Derry Wanye #include "StructPackAlignCheck.h"
10156b1279SFrank Derry Wanye #include "clang/AST/ASTContext.h"
11156b1279SFrank Derry Wanye #include "clang/AST/RecordLayout.h"
12156b1279SFrank Derry Wanye #include "clang/ASTMatchers/ASTMatchFinder.h"
13c8644b18SPiotr Zegar #include <cmath>
14156b1279SFrank Derry Wanye 
15156b1279SFrank Derry Wanye using namespace clang::ast_matchers;
16156b1279SFrank Derry Wanye 
177d2ea6c4SCarlos Galvez namespace clang::tidy::altera {
18156b1279SFrank Derry Wanye 
registerMatchers(MatchFinder * Finder)19156b1279SFrank Derry Wanye void StructPackAlignCheck::registerMatchers(MatchFinder *Finder) {
20ee1f132dSMatheus Izvekov   Finder->addMatcher(recordDecl(isStruct(), isDefinition(),
21ee1f132dSMatheus Izvekov                                 unless(isExpansionInSystemHeader()))
22156b1279SFrank Derry Wanye                          .bind("struct"),
23156b1279SFrank Derry Wanye                      this);
24156b1279SFrank Derry Wanye }
25156b1279SFrank Derry Wanye 
26156b1279SFrank Derry Wanye CharUnits
computeRecommendedAlignment(CharUnits MinByteSize) const2724a7587bSPiotr Zegar StructPackAlignCheck::computeRecommendedAlignment(CharUnits MinByteSize) const {
28156b1279SFrank Derry Wanye   CharUnits NewAlign = CharUnits::fromQuantity(1);
29156b1279SFrank Derry Wanye   if (!MinByteSize.isPowerOfTwo()) {
3044b38fe8SPiotr Zegar     CharUnits::QuantityType MSB = MinByteSize.getQuantity();
31156b1279SFrank Derry Wanye     for (; MSB > 0; MSB /= 2) {
3244b38fe8SPiotr Zegar       NewAlign =
3344b38fe8SPiotr Zegar           NewAlign.alignTo(CharUnits::fromQuantity(NewAlign.getQuantity() * 2));
34156b1279SFrank Derry Wanye       // Abort if the computed alignment meets the maximum configured alignment.
35156b1279SFrank Derry Wanye       if (NewAlign.getQuantity() >= MaxConfiguredAlignment)
36156b1279SFrank Derry Wanye         break;
37156b1279SFrank Derry Wanye     }
38156b1279SFrank Derry Wanye   } else {
39156b1279SFrank Derry Wanye     NewAlign = MinByteSize;
40156b1279SFrank Derry Wanye   }
41156b1279SFrank Derry Wanye   return NewAlign;
42156b1279SFrank Derry Wanye }
43156b1279SFrank Derry Wanye 
check(const MatchFinder::MatchResult & Result)44156b1279SFrank Derry Wanye void StructPackAlignCheck::check(const MatchFinder::MatchResult &Result) {
45156b1279SFrank Derry Wanye   const auto *Struct = Result.Nodes.getNodeAs<RecordDecl>("struct");
46156b1279SFrank Derry Wanye 
47156b1279SFrank Derry Wanye   // Do not trigger on templated struct declarations because the packing and
48156b1279SFrank Derry Wanye   // alignment requirements are unknown.
49156b1279SFrank Derry Wanye   if (Struct->isTemplated())
50156b1279SFrank Derry Wanye      return;
51156b1279SFrank Derry Wanye 
52e1d0673aSBalazs Benics   // Packing and alignment requirements for invalid decls are meaningless.
53e1d0673aSBalazs Benics   if (Struct->isInvalidDecl())
54e1d0673aSBalazs Benics     return;
55e1d0673aSBalazs Benics 
56156b1279SFrank Derry Wanye   // Get sizing info for the struct.
57156b1279SFrank Derry Wanye   llvm::SmallVector<std::pair<unsigned int, unsigned int>, 10> FieldSizes;
58156b1279SFrank Derry Wanye   unsigned int TotalBitSize = 0;
59156b1279SFrank Derry Wanye   for (const FieldDecl *StructField : Struct->fields()) {
60156b1279SFrank Derry Wanye     // For each StructField, record how big it is (in bits).
61156b1279SFrank Derry Wanye     // Would be good to use a pair of <offset, size> to advise a better
62156b1279SFrank Derry Wanye     // packing order.
63ab92a4c2SGeorgy Komarov     QualType StructFieldTy = StructField->getType();
64ab92a4c2SGeorgy Komarov     if (StructFieldTy->isIncompleteType())
65ab92a4c2SGeorgy Komarov       return;
66156b1279SFrank Derry Wanye     unsigned int StructFieldWidth =
67ab92a4c2SGeorgy Komarov         (unsigned int)Result.Context->getTypeInfo(StructFieldTy.getTypePtr())
68156b1279SFrank Derry Wanye             .Width;
69156b1279SFrank Derry Wanye     FieldSizes.emplace_back(StructFieldWidth, StructField->getFieldIndex());
70156b1279SFrank Derry Wanye     // FIXME: Recommend a reorganization of the struct (sort by StructField
71156b1279SFrank Derry Wanye     // size, largest to smallest).
72156b1279SFrank Derry Wanye     TotalBitSize += StructFieldWidth;
73156b1279SFrank Derry Wanye   }
74156b1279SFrank Derry Wanye 
75156b1279SFrank Derry Wanye   uint64_t CharSize = Result.Context->getCharWidth();
76156b1279SFrank Derry Wanye   CharUnits CurrSize = Result.Context->getASTRecordLayout(Struct).getSize();
77156b1279SFrank Derry Wanye   CharUnits MinByteSize =
78f25935a0SFabian Wolff       CharUnits::fromQuantity(std::max<clang::CharUnits::QuantityType>(
79f25935a0SFabian Wolff           ceil(static_cast<float>(TotalBitSize) / CharSize), 1));
80156b1279SFrank Derry Wanye   CharUnits MaxAlign = CharUnits::fromQuantity(
81156b1279SFrank Derry Wanye       ceil((float)Struct->getMaxAlignment() / CharSize));
82156b1279SFrank Derry Wanye   CharUnits CurrAlign =
83156b1279SFrank Derry Wanye       Result.Context->getASTRecordLayout(Struct).getAlignment();
84156b1279SFrank Derry Wanye   CharUnits NewAlign = computeRecommendedAlignment(MinByteSize);
85156b1279SFrank Derry Wanye 
86156b1279SFrank Derry Wanye   bool IsPacked = Struct->hasAttr<PackedAttr>();
87156b1279SFrank Derry Wanye   bool NeedsPacking = (MinByteSize < CurrSize) && (MaxAlign != NewAlign) &&
88156b1279SFrank Derry Wanye                       (CurrSize != NewAlign);
89156b1279SFrank Derry Wanye   bool NeedsAlignment = CurrAlign.getQuantity() != NewAlign.getQuantity();
90156b1279SFrank Derry Wanye 
91156b1279SFrank Derry Wanye   if (!NeedsAlignment && !NeedsPacking)
92156b1279SFrank Derry Wanye     return;
93156b1279SFrank Derry Wanye 
94156b1279SFrank Derry Wanye   // If it's using much more space than it needs, suggest packing.
95156b1279SFrank Derry Wanye   // (Do not suggest packing if it is currently explicitly aligned to what the
96156b1279SFrank Derry Wanye   // minimum byte size would suggest as the new alignment.)
97156b1279SFrank Derry Wanye   if (NeedsPacking && !IsPacked) {
98156b1279SFrank Derry Wanye     diag(Struct->getLocation(),
99156b1279SFrank Derry Wanye          "accessing fields in struct %0 is inefficient due to padding; only "
100156b1279SFrank Derry Wanye          "needs %1 bytes but is using %2 bytes")
101156b1279SFrank Derry Wanye         << Struct << (int)MinByteSize.getQuantity()
102156b1279SFrank Derry Wanye         << (int)CurrSize.getQuantity()
103156b1279SFrank Derry Wanye         << FixItHint::CreateInsertion(Struct->getEndLoc().getLocWithOffset(1),
104156b1279SFrank Derry Wanye                                       " __attribute__((packed))");
105156b1279SFrank Derry Wanye     diag(Struct->getLocation(),
106156b1279SFrank Derry Wanye          "use \"__attribute__((packed))\" to reduce the amount of padding "
107156b1279SFrank Derry Wanye          "applied to struct %0",
108156b1279SFrank Derry Wanye          DiagnosticIDs::Note)
109156b1279SFrank Derry Wanye         << Struct;
110156b1279SFrank Derry Wanye   }
111156b1279SFrank Derry Wanye 
112156b1279SFrank Derry Wanye   FixItHint FixIt;
113*fc2a9ad1SPiotr Zegar   auto *Attribute = Struct->getAttr<AlignedAttr>();
114156b1279SFrank Derry Wanye   std::string NewAlignQuantity = std::to_string((int)NewAlign.getQuantity());
115156b1279SFrank Derry Wanye   if (Attribute) {
11600c7d669SNathan James     FixIt = FixItHint::CreateReplacement(
11700c7d669SNathan James         Attribute->getRange(),
11800c7d669SNathan James         (Twine("aligned(") + NewAlignQuantity + ")").str());
119156b1279SFrank Derry Wanye   } else {
12000c7d669SNathan James     FixIt = FixItHint::CreateInsertion(
12100c7d669SNathan James         Struct->getEndLoc().getLocWithOffset(1),
12200c7d669SNathan James         (Twine(" __attribute__((aligned(") + NewAlignQuantity + ")))").str());
123156b1279SFrank Derry Wanye   }
124156b1279SFrank Derry Wanye 
125156b1279SFrank Derry Wanye   // And suggest the minimum power-of-two alignment for the struct as a whole
126156b1279SFrank Derry Wanye   // (with and without packing).
127156b1279SFrank Derry Wanye   if (NeedsAlignment) {
128156b1279SFrank Derry Wanye     diag(Struct->getLocation(),
129156b1279SFrank Derry Wanye          "accessing fields in struct %0 is inefficient due to poor alignment; "
130156b1279SFrank Derry Wanye          "currently aligned to %1 bytes, but recommended alignment is %2 bytes")
131156b1279SFrank Derry Wanye         << Struct << (int)CurrAlign.getQuantity() << NewAlignQuantity << FixIt;
132156b1279SFrank Derry Wanye 
133156b1279SFrank Derry Wanye     diag(Struct->getLocation(),
134156b1279SFrank Derry Wanye          "use \"__attribute__((aligned(%0)))\" to align struct %1 to %0 bytes",
135156b1279SFrank Derry Wanye          DiagnosticIDs::Note)
136156b1279SFrank Derry Wanye         << NewAlignQuantity << Struct;
137156b1279SFrank Derry Wanye   }
138156b1279SFrank Derry Wanye }
139156b1279SFrank Derry Wanye 
storeOptions(ClangTidyOptions::OptionMap & Opts)140156b1279SFrank Derry Wanye void StructPackAlignCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) {
141156b1279SFrank Derry Wanye   Options.store(Opts, "MaxConfiguredAlignment", MaxConfiguredAlignment);
142156b1279SFrank Derry Wanye }
143156b1279SFrank Derry Wanye 
1447d2ea6c4SCarlos Galvez } // namespace clang::tidy::altera
145