bool isUIntArg() const { return kind >= oArg && kind <= XArg; }
bool isDoubleArg() const { return kind >= fArg && kind <= AArg; }
Kind getKind() const { return kind; }
+ unsigned getLength() const {
+ // Conversion specifiers currently only are represented by
+ // single characters, but we be flexible.
+ return 1;
+ }
private:
const char *Position;
bool hasLeadingZeros() const { return flags & LeadingZeroes; }
};
-
+} // end printf namespace
+
class FormatStringHandler {
public:
FormatStringHandler() {}
virtual ~FormatStringHandler();
virtual void HandleIncompleteFormatSpecifier(const char *startSpecifier,
- const char *endSpecifier) {}
+ unsigned specifierLen) {}
virtual void HandleNullChar(const char *nullCharacter) {}
+
+ virtual void
+ HandleInvalidConversionSpecifier(const analyze_printf::FormatSpecifier &FS,
+ const char *startSpecifier,
+ unsigned specifierLen) {}
- virtual void HandleIncompletePrecision(const char *periodChar) {}
-
- virtual void HandleInvalidConversionSpecifier(const FormatSpecifier &FS,
- const char *startSpecifier,
- unsigned specifierLen) {}
-
- virtual bool HandleFormatSpecifier(const FormatSpecifier &FS,
+ virtual bool HandleFormatSpecifier(const analyze_printf::FormatSpecifier &FS,
const char *startSpecifier,
unsigned specifierLen) {
return true;
const char *beg, const char *end);
-} // end printf namespace
} // end clang namespace
#endif
"more data arguments than '%%' conversions">, InGroup<FormatExtraArgs>;
def warn_printf_invalid_conversion : Warning<
"invalid conversion '%0'">, InGroup<Format>;
+def warn_printf_incomplete_specifier : Warning<
+ "incomplete format specifier '%0'">, InGroup<Format>;
def warn_printf_missing_format_string : Warning<
"format string missing">, InGroup<Format>;
def warn_null_arg : Warning<
using clang::analyze_printf::FormatSpecifier;
using clang::analyze_printf::OptionalAmount;
+using namespace clang;
namespace {
class FormatSpecifierResult {
return OptionalAmount();
}
-static FormatSpecifierResult
-ParseFormatSpecifier(clang::analyze_printf::FormatStringHandler &H,
- const char *&Beg, const char *E) {
+static FormatSpecifierResult ParseFormatSpecifier(FormatStringHandler &H,
+ const char *&Beg, const char *E) {
using namespace clang::analyze_printf;
if (I == E) {
// No more characters left?
- H.HandleIncompleteFormatSpecifier(Start, E);
+ H.HandleIncompleteFormatSpecifier(Start, E - Start);
return true;
}
if (I == E) {
// No more characters left?
- H.HandleIncompleteFormatSpecifier(Start, E);
+ H.HandleIncompleteFormatSpecifier(Start, E - Start);
return true;
}
if (I == E) {
// No more characters left?
- H.HandleIncompleteFormatSpecifier(Start, E);
+ H.HandleIncompleteFormatSpecifier(Start, E - Start);
return true;
}
// Look for the precision (if any).
if (*I == '.') {
- const char *startPrecision = I++;
+ ++I;
if (I == E) {
- H.HandleIncompletePrecision(I - 1);
+ H.HandleIncompleteFormatSpecifier(Start, E - Start);
return true;
}
if (I == E) {
// No more characters left?
- H.HandleIncompletePrecision(startPrecision);
+ H.HandleIncompleteFormatSpecifier(Start, E - Start);
return true;
}
}
if (I == E) {
// No more characters left?
- H.HandleIncompleteFormatSpecifier(Start, E);
+ H.HandleIncompleteFormatSpecifier(Start, E - Start);
return true;
}
return FormatSpecifierResult(Start, FS);
}
-namespace clang { namespace analyze_printf {
-bool ParseFormatString(FormatStringHandler &H,
+bool clang::ParseFormatString(FormatStringHandler &H,
const char *I, const char *E) {
// Keep looking for a format specifier until we have exhausted the string.
while (I != E) {
}
FormatStringHandler::~FormatStringHandler() {}
-}} // end namespace clang::analyze_printf
namespace {
-class CheckPrintfHandler : public analyze_printf::FormatStringHandler {
+class CheckPrintfHandler : public FormatStringHandler {
Sema &S;
const StringLiteral *FExpr;
const Expr *OrigFormatExpr;
TheCall(theCall), FormatIdx(formatIdx) {}
void DoneProcessing();
-
-// void HandleIncompleteFormatSpecifier(const char *startSpecifier,
-// const char *endSpecifier);
-
-// void HandleIncompletePrecision(const char *periodChar);
+
+ void HandleIncompleteFormatSpecifier(const char *startSpecifier,
+ unsigned specifierLen);
void HandleInvalidConversionSpecifier(const analyze_printf::FormatSpecifier &FS,
const char *startSpecifier,
return S.getLocationOfStringLiteralByte(FExpr, x - Beg);
}
+void CheckPrintfHandler::
+HandleIncompleteFormatSpecifier(const char *startSpecifier,
+ unsigned specifierLen) {
+ SourceLocation Loc = getLocationOfByte(startSpecifier);
+ S.Diag(Loc, diag::warn_printf_incomplete_specifier)
+ << llvm::StringRef(startSpecifier, specifierLen)
+ << getFormatRange();
+}
+
void CheckPrintfHandler::
HandleInvalidConversionSpecifier(const analyze_printf::FormatSpecifier &FS,
const char *startSpecifier,
unsigned specifierLen) {
++NumConversions;
-
- SourceLocation Loc =
- getLocationOfByte(FS.getConversionSpecifier().getStart());
+ const analyze_printf::ConversionSpecifier &CS =
+ FS.getConversionSpecifier();
+ SourceLocation Loc = getLocationOfByte(CS.getStart());
S.Diag(Loc, diag::warn_printf_invalid_conversion)
- << llvm::StringRef(startSpecifier, specifierLen)
+ << llvm::StringRef(CS.getStart(), CS.getLength())
<< getFormatRange();
}
isa<ObjCStringLiteral>(OrigFormatExpr), Str,
HasVAListArg, TheCall, format_idx);
- analyze_printf::ParseFormatString(H, Str, Str + StrLen);
- H.DoneProcessing();
+ if (!ParseFormatString(H, Str, Str + StrLen))
+ H.DoneProcessing();
}
//===--- CHECK: Return Address of Stack Variable --------------------------===//