================ @@ -0,0 +1,235 @@ +//===--- UseDigitSeparatorCheck.cpp - clang-tidy --------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include <algorithm> +#include <iomanip> +#include <numeric> +#include <regex> +#include <sstream> + +#include "UseDigitSeparatorCheck.h" +#include "clang/AST/ASTContext.h" +#include "clang/ASTMatchers/ASTMatchFinder.h" +#include "clang/Lex/Lexer.h" +#include "llvm/Support/FormatVariadic.h" + +using namespace clang::ast_matchers; + +namespace { +std::string +getFormatedScientificFloatString(const llvm::StringRef OriginalLiteralString); + +std::vector<std::basic_string<char>> +splitStringByGroupSize(const std::basic_string<char> &String, + size_t GroupSize) { + std::vector<std::basic_string<char>> Result; + std::basic_string<char> ReversedString(String.rbegin(), String.rend()); + + for (size_t I = 0; I < ReversedString.size(); I += GroupSize) { + Result.push_back(ReversedString.substr(I, GroupSize)); + } + + std::reverse(Result.begin(), Result.end()); + std::for_each(Result.begin(), Result.end(), [](std::basic_string<char> &Str) { + return std::reverse(Str.begin(), Str.end()); + }); + + return Result; +} + +std::string +getFormatedIntegerString(const llvm::StringRef OriginalLiteralString, + const llvm::APInt IntegerValue) { + // Configure formatting + unsigned int Radix; + size_t GroupSize; + std::string Prefix; + std::string Postfix; + if (OriginalLiteralString.starts_with("0b")) { + Radix = 2; + GroupSize = 4; + Prefix = "0b"; + } else if (OriginalLiteralString.starts_with("0x")) { + Radix = 16; + GroupSize = 4; + Prefix = "0x"; + } else if (OriginalLiteralString.starts_with("0") && + OriginalLiteralString != "0") { + Radix = 8; + GroupSize = 3; + Prefix = "0"; + } else { + Radix = 10; + GroupSize = 3; + } + + for (const char &Character : OriginalLiteralString.substr( + Prefix.size(), OriginalLiteralString.size())) { + if (((Prefix == "0x" && !std::isxdigit(Character)) || + (Prefix != "0x" && !std::isdigit(Character))) && + Character != '\'') { + Postfix += Character; + } + } + + // Get formatting literal text + const std::vector<std::string> SplittedIntegerLiteral = + splitStringByGroupSize(toString(IntegerValue, Radix, true), GroupSize); + const std::string FormatedLiteralString = + Prefix + + std::accumulate(SplittedIntegerLiteral.begin(), + SplittedIntegerLiteral.end(), std::string(""), + [](std::basic_string<char> S1, + std::basic_string<char> S2) { return S1 + "\'" + S2; }) + .erase(0, 1) + + Postfix; + + return FormatedLiteralString; +} + +std::string getFormatedFloatString(const llvm::StringRef OriginalLiteralString, + const llvm::APFloat FloatValue) { + if (OriginalLiteralString.contains('E') || + OriginalLiteralString.contains('e')) { + return getFormatedScientificFloatString(OriginalLiteralString); + } + + // Configure formatting + std::string Postfix; + for (const char &Character : OriginalLiteralString) { + if (!std::isdigit(Character) && Character != '.' && Character != '\'') { + Postfix += Character; + } + } + + // Get precision + const std::string::size_type OriginalDotPosition = + OriginalLiteralString.find('.'); + const llvm::StringRef OriginalFractionalSubString = + OriginalLiteralString.substr(OriginalDotPosition + 1, + OriginalLiteralString.size()); + int Precision = 0; + for (const char &Character : OriginalFractionalSubString) { + if (std::isdigit(Character)) { + Precision++; + } + } + + // Get formatting literal text + + // Get string representation of float value + std::ostringstream StringStream; + StringStream << std::fixed << std::setprecision(Precision) + << FloatValue.convertToDouble(); + const std::string FloatString = StringStream.str(); + + // Get integer and fractional parts of float number + const std::string::size_type DotPosition = FloatString.find('.'); + const std::string IntegerSubString = FloatString.substr(0, DotPosition); + std::string FractionalSubString = + FloatString.substr(DotPosition + 1, FloatString.size()); + std::reverse(FractionalSubString.begin(), FractionalSubString.end()); + + // Get formatting literal text + const std::string FormatedIntegerSubString = getFormatedIntegerString( + IntegerSubString, llvm::APInt(128, std::stoll(IntegerSubString))); + std::string FormatedFractionalSubString = getFormatedIntegerString( + FractionalSubString, llvm::APInt(128, std::stoll(FractionalSubString))); + std::reverse(FormatedFractionalSubString.begin(), + FormatedFractionalSubString.end()); + + const std::string FormatedLiteralString = + FormatedIntegerSubString + '.' + FormatedFractionalSubString + Postfix; + + return FormatedLiteralString; +} + +std::string +getFormatedScientificFloatString(const llvm::StringRef OriginalLiteralString) { + // Split string to mantissa and exponent + const char EChar = + OriginalLiteralString.str().find('E') != std::string::npos ? 'E' : 'e'; + const std::string::size_type EPosition = + OriginalLiteralString.str().find(EChar); + const std::string SignSymbol = + OriginalLiteralString[EPosition + 1] == '-' + ? "-" + : (OriginalLiteralString[EPosition + 1] == '+' ? "+" : ""); + const llvm::StringRef MantissaSubString = + OriginalLiteralString.substr(0, EPosition); + const llvm::StringRef ExponentSubString = OriginalLiteralString.substr( + EPosition + SignSymbol.size() + 1, OriginalLiteralString.size()); + + // Get formatting literal text + const std::string FormatedMantissaString = getFormatedFloatString( + MantissaSubString, llvm::APFloat(std::stod(MantissaSubString.str()))); + const std::string FormatedExponentString = getFormatedIntegerString( + ExponentSubString, llvm::APInt(128, std::stoll(ExponentSubString.str()))); + return FormatedMantissaString + EChar + SignSymbol + FormatedExponentString; +} +} // namespace + +namespace clang::tidy::modernize { + +void UseDigitSeparatorCheck::registerMatchers(MatchFinder *Finder) { + Finder->addMatcher(integerLiteral().bind("integerLiteral"), this); + Finder->addMatcher(floatLiteral().bind("floatLiteral"), this); +} + +void UseDigitSeparatorCheck::check(const MatchFinder::MatchResult &Result) { + const ASTContext &Context = *Result.Context; + const SourceManager &Source = Context.getSourceManager(); + const IntegerLiteral *MatchedInteger = + Result.Nodes.getNodeAs<IntegerLiteral>("integerLiteral"); + const FloatingLiteral *MatchedFloat = + Result.Nodes.getNodeAs<FloatingLiteral>("floatLiteral"); + + if (MatchedInteger != nullptr) { + // Get original literal source text + const llvm::StringRef OriginalLiteralString = Lexer::getSourceText( + CharSourceRange::getTokenRange(MatchedInteger->getSourceRange()), + Source, Context.getLangOpts()); + + // Get formatting literal text + const llvm::APInt IntegerValue = MatchedInteger->getValue(); + const std::string FormatedLiteralString = + getFormatedIntegerString(OriginalLiteralString, IntegerValue); + + // Compare the original and formatted representation of a literal + if (OriginalLiteralString != FormatedLiteralString) { + diag(MatchedInteger->getLocation(), + "unformatted representation of integer literal '%0'") + << OriginalLiteralString + << FixItHint::CreateInsertion(MatchedInteger->getLocation(), + FormatedLiteralString); + } + } else if (MatchedFloat != nullptr) { + // Get original literal source text + const llvm::StringRef OriginalLiteralString = Lexer::getSourceText( + CharSourceRange::getTokenRange(MatchedFloat->getSourceRange()), Source, + Context.getLangOpts()); + + // Get formatting literal text + const llvm::APFloat FloatValue = MatchedFloat->getValue(); + const std::string FormatedLiteralString = + getFormatedFloatString(OriginalLiteralString, FloatValue); + + // Compare the original and formatted representation of a literal + if (OriginalLiteralString != FormatedLiteralString) { + diag(MatchedFloat->getLocation(), + "unformatted representation of float literal '%0'") ---------------- PiotrZSL wrote:
maybe floating, because if it's double, then float may not fit. https://github.com/llvm/llvm-project/pull/76153 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits