PrintfFormatString.cpp 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742
  1. //== PrintfFormatString.cpp - Analysis of printf format strings --*- C++ -*-==//
  2. //
  3. // The LLVM Compiler Infrastructure
  4. //
  5. // This file is distributed under the University of Illinois Open Source
  6. // License. See LICENSE.TXT for details.
  7. //
  8. //===----------------------------------------------------------------------===//
  9. //
  10. // Handling of format string in printf and friends. The structure of format
  11. // strings for fprintf() are described in C99 7.19.6.1.
  12. //
  13. //===----------------------------------------------------------------------===//
  14. #include "clang/Analysis/Analyses/FormatString.h"
  15. #include "FormatStringParsing.h"
  16. #include "clang/Basic/TargetInfo.h"
  17. using clang::analyze_format_string::ArgType;
  18. using clang::analyze_format_string::FormatStringHandler;
  19. using clang::analyze_format_string::LengthModifier;
  20. using clang::analyze_format_string::OptionalAmount;
  21. using clang::analyze_format_string::ConversionSpecifier;
  22. using clang::analyze_printf::PrintfSpecifier;
  23. using namespace clang;
  24. typedef clang::analyze_format_string::SpecifierResult<PrintfSpecifier>
  25. PrintfSpecifierResult;
  26. //===----------------------------------------------------------------------===//
  27. // Methods for parsing format strings.
  28. //===----------------------------------------------------------------------===//
  29. using analyze_format_string::ParseNonPositionAmount;
  30. static bool ParsePrecision(FormatStringHandler &H, PrintfSpecifier &FS,
  31. const char *Start, const char *&Beg, const char *E,
  32. unsigned *argIndex) {
  33. if (argIndex) {
  34. FS.setPrecision(ParseNonPositionAmount(Beg, E, *argIndex));
  35. } else {
  36. const OptionalAmount Amt = ParsePositionAmount(H, Start, Beg, E,
  37. analyze_format_string::PrecisionPos);
  38. if (Amt.isInvalid())
  39. return true;
  40. FS.setPrecision(Amt);
  41. }
  42. return false;
  43. }
  44. static PrintfSpecifierResult ParsePrintfSpecifier(FormatStringHandler &H,
  45. const char *&Beg,
  46. const char *E,
  47. unsigned &argIndex,
  48. const LangOptions &LO,
  49. const TargetInfo &Target) {
  50. using namespace clang::analyze_format_string;
  51. using namespace clang::analyze_printf;
  52. const char *I = Beg;
  53. const char *Start = 0;
  54. UpdateOnReturn <const char*> UpdateBeg(Beg, I);
  55. // Look for a '%' character that indicates the start of a format specifier.
  56. for ( ; I != E ; ++I) {
  57. char c = *I;
  58. if (c == '\0') {
  59. // Detect spurious null characters, which are likely errors.
  60. H.HandleNullChar(I);
  61. return true;
  62. }
  63. if (c == '%') {
  64. Start = I++; // Record the start of the format specifier.
  65. break;
  66. }
  67. }
  68. // No format specifier found?
  69. if (!Start)
  70. return false;
  71. if (I == E) {
  72. // No more characters left?
  73. H.HandleIncompleteSpecifier(Start, E - Start);
  74. return true;
  75. }
  76. PrintfSpecifier FS;
  77. if (ParseArgPosition(H, FS, Start, I, E))
  78. return true;
  79. if (I == E) {
  80. // No more characters left?
  81. H.HandleIncompleteSpecifier(Start, E - Start);
  82. return true;
  83. }
  84. // Look for flags (if any).
  85. bool hasMore = true;
  86. for ( ; I != E; ++I) {
  87. switch (*I) {
  88. default: hasMore = false; break;
  89. case '\'':
  90. // FIXME: POSIX specific. Always accept?
  91. FS.setHasThousandsGrouping(I);
  92. break;
  93. case '-': FS.setIsLeftJustified(I); break;
  94. case '+': FS.setHasPlusPrefix(I); break;
  95. case ' ': FS.setHasSpacePrefix(I); break;
  96. case '#': FS.setHasAlternativeForm(I); break;
  97. case '0': FS.setHasLeadingZeros(I); break;
  98. }
  99. if (!hasMore)
  100. break;
  101. }
  102. if (I == E) {
  103. // No more characters left?
  104. H.HandleIncompleteSpecifier(Start, E - Start);
  105. return true;
  106. }
  107. // Look for the field width (if any).
  108. if (ParseFieldWidth(H, FS, Start, I, E,
  109. FS.usesPositionalArg() ? 0 : &argIndex))
  110. return true;
  111. if (I == E) {
  112. // No more characters left?
  113. H.HandleIncompleteSpecifier(Start, E - Start);
  114. return true;
  115. }
  116. // Look for the precision (if any).
  117. if (*I == '.') {
  118. ++I;
  119. if (I == E) {
  120. H.HandleIncompleteSpecifier(Start, E - Start);
  121. return true;
  122. }
  123. if (ParsePrecision(H, FS, Start, I, E,
  124. FS.usesPositionalArg() ? 0 : &argIndex))
  125. return true;
  126. if (I == E) {
  127. // No more characters left?
  128. H.HandleIncompleteSpecifier(Start, E - Start);
  129. return true;
  130. }
  131. }
  132. // Look for the length modifier.
  133. if (ParseLengthModifier(FS, I, E, LO) && I == E) {
  134. // No more characters left?
  135. H.HandleIncompleteSpecifier(Start, E - Start);
  136. return true;
  137. }
  138. if (*I == '\0') {
  139. // Detect spurious null characters, which are likely errors.
  140. H.HandleNullChar(I);
  141. return true;
  142. }
  143. // Finally, look for the conversion specifier.
  144. const char *conversionPosition = I++;
  145. ConversionSpecifier::Kind k = ConversionSpecifier::InvalidSpecifier;
  146. switch (*conversionPosition) {
  147. default:
  148. break;
  149. // C99: 7.19.6.1 (section 8).
  150. case '%': k = ConversionSpecifier::PercentArg; break;
  151. case 'A': k = ConversionSpecifier::AArg; break;
  152. case 'E': k = ConversionSpecifier::EArg; break;
  153. case 'F': k = ConversionSpecifier::FArg; break;
  154. case 'G': k = ConversionSpecifier::GArg; break;
  155. case 'X': k = ConversionSpecifier::XArg; break;
  156. case 'a': k = ConversionSpecifier::aArg; break;
  157. case 'c': k = ConversionSpecifier::cArg; break;
  158. case 'd': k = ConversionSpecifier::dArg; break;
  159. case 'e': k = ConversionSpecifier::eArg; break;
  160. case 'f': k = ConversionSpecifier::fArg; break;
  161. case 'g': k = ConversionSpecifier::gArg; break;
  162. case 'i': k = ConversionSpecifier::iArg; break;
  163. case 'n': k = ConversionSpecifier::nArg; break;
  164. case 'o': k = ConversionSpecifier::oArg; break;
  165. case 'p': k = ConversionSpecifier::pArg; break;
  166. case 's': k = ConversionSpecifier::sArg; break;
  167. case 'u': k = ConversionSpecifier::uArg; break;
  168. case 'x': k = ConversionSpecifier::xArg; break;
  169. // POSIX specific.
  170. case 'C': k = ConversionSpecifier::CArg; break;
  171. case 'S': k = ConversionSpecifier::SArg; break;
  172. // Objective-C.
  173. case '@': k = ConversionSpecifier::ObjCObjArg; break;
  174. // Glibc specific.
  175. case 'm': k = ConversionSpecifier::PrintErrno; break;
  176. // Apple-specific
  177. case 'D':
  178. if (Target.getTriple().isOSDarwin())
  179. k = ConversionSpecifier::DArg;
  180. break;
  181. case 'O':
  182. if (Target.getTriple().isOSDarwin())
  183. k = ConversionSpecifier::OArg;
  184. break;
  185. case 'U':
  186. if (Target.getTriple().isOSDarwin())
  187. k = ConversionSpecifier::UArg;
  188. break;
  189. }
  190. PrintfConversionSpecifier CS(conversionPosition, k);
  191. FS.setConversionSpecifier(CS);
  192. if (CS.consumesDataArgument() && !FS.usesPositionalArg())
  193. FS.setArgIndex(argIndex++);
  194. if (k == ConversionSpecifier::InvalidSpecifier) {
  195. // Assume the conversion takes one argument.
  196. return !H.HandleInvalidPrintfConversionSpecifier(FS, Start, I - Start);
  197. }
  198. return PrintfSpecifierResult(Start, FS);
  199. }
  200. bool clang::analyze_format_string::ParsePrintfString(FormatStringHandler &H,
  201. const char *I,
  202. const char *E,
  203. const LangOptions &LO,
  204. const TargetInfo &Target) {
  205. unsigned argIndex = 0;
  206. // Keep looking for a format specifier until we have exhausted the string.
  207. while (I != E) {
  208. const PrintfSpecifierResult &FSR = ParsePrintfSpecifier(H, I, E, argIndex,
  209. LO, Target);
  210. // Did a fail-stop error of any kind occur when parsing the specifier?
  211. // If so, don't do any more processing.
  212. if (FSR.shouldStop())
  213. return true;
  214. // Did we exhaust the string or encounter an error that
  215. // we can recover from?
  216. if (!FSR.hasValue())
  217. continue;
  218. // We have a format specifier. Pass it to the callback.
  219. if (!H.HandlePrintfSpecifier(FSR.getValue(), FSR.getStart(),
  220. I - FSR.getStart()))
  221. return true;
  222. }
  223. assert(I == E && "Format string not exhausted");
  224. return false;
  225. }
  226. //===----------------------------------------------------------------------===//
  227. // Methods on PrintfSpecifier.
  228. //===----------------------------------------------------------------------===//
  229. ArgType PrintfSpecifier::getArgType(ASTContext &Ctx,
  230. bool IsObjCLiteral) const {
  231. const PrintfConversionSpecifier &CS = getConversionSpecifier();
  232. if (!CS.consumesDataArgument())
  233. return ArgType::Invalid();
  234. if (CS.getKind() == ConversionSpecifier::cArg)
  235. switch (LM.getKind()) {
  236. case LengthModifier::None: return Ctx.IntTy;
  237. case LengthModifier::AsLong:
  238. return ArgType(ArgType::WIntTy, "wint_t");
  239. default:
  240. return ArgType::Invalid();
  241. }
  242. if (CS.isIntArg())
  243. switch (LM.getKind()) {
  244. case LengthModifier::AsLongDouble:
  245. // GNU extension.
  246. return Ctx.LongLongTy;
  247. case LengthModifier::None: return Ctx.IntTy;
  248. case LengthModifier::AsChar: return ArgType::AnyCharTy;
  249. case LengthModifier::AsShort: return Ctx.ShortTy;
  250. case LengthModifier::AsLong: return Ctx.LongTy;
  251. case LengthModifier::AsLongLong:
  252. case LengthModifier::AsQuad:
  253. return Ctx.LongLongTy;
  254. case LengthModifier::AsIntMax:
  255. return ArgType(Ctx.getIntMaxType(), "intmax_t");
  256. case LengthModifier::AsSizeT:
  257. // FIXME: How to get the corresponding signed version of size_t?
  258. return ArgType();
  259. case LengthModifier::AsPtrDiff:
  260. return ArgType(Ctx.getPointerDiffType(), "ptrdiff_t");
  261. case LengthModifier::AsAllocate:
  262. case LengthModifier::AsMAllocate:
  263. return ArgType::Invalid();
  264. }
  265. if (CS.isUIntArg())
  266. switch (LM.getKind()) {
  267. case LengthModifier::AsLongDouble:
  268. // GNU extension.
  269. return Ctx.UnsignedLongLongTy;
  270. case LengthModifier::None: return Ctx.UnsignedIntTy;
  271. case LengthModifier::AsChar: return Ctx.UnsignedCharTy;
  272. case LengthModifier::AsShort: return Ctx.UnsignedShortTy;
  273. case LengthModifier::AsLong: return Ctx.UnsignedLongTy;
  274. case LengthModifier::AsLongLong:
  275. case LengthModifier::AsQuad:
  276. return Ctx.UnsignedLongLongTy;
  277. case LengthModifier::AsIntMax:
  278. return ArgType(Ctx.getUIntMaxType(), "uintmax_t");
  279. case LengthModifier::AsSizeT:
  280. return ArgType(Ctx.getSizeType(), "size_t");
  281. case LengthModifier::AsPtrDiff:
  282. // FIXME: How to get the corresponding unsigned
  283. // version of ptrdiff_t?
  284. return ArgType();
  285. case LengthModifier::AsAllocate:
  286. case LengthModifier::AsMAllocate:
  287. return ArgType::Invalid();
  288. }
  289. if (CS.isDoubleArg()) {
  290. if (LM.getKind() == LengthModifier::AsLongDouble)
  291. return Ctx.LongDoubleTy;
  292. return Ctx.DoubleTy;
  293. }
  294. if (CS.getKind() == ConversionSpecifier::nArg) {
  295. switch (LM.getKind()) {
  296. case LengthModifier::None:
  297. return ArgType::PtrTo(Ctx.IntTy);
  298. case LengthModifier::AsChar:
  299. return ArgType::PtrTo(Ctx.SignedCharTy);
  300. case LengthModifier::AsShort:
  301. return ArgType::PtrTo(Ctx.ShortTy);
  302. case LengthModifier::AsLong:
  303. return ArgType::PtrTo(Ctx.LongTy);
  304. case LengthModifier::AsLongLong:
  305. case LengthModifier::AsQuad:
  306. return ArgType::PtrTo(Ctx.LongLongTy);
  307. case LengthModifier::AsIntMax:
  308. return ArgType::PtrTo(ArgType(Ctx.getIntMaxType(), "intmax_t"));
  309. case LengthModifier::AsSizeT:
  310. return ArgType(); // FIXME: ssize_t
  311. case LengthModifier::AsPtrDiff:
  312. return ArgType::PtrTo(ArgType(Ctx.getPointerDiffType(), "ptrdiff_t"));
  313. case LengthModifier::AsLongDouble:
  314. return ArgType(); // FIXME: Is this a known extension?
  315. case LengthModifier::AsAllocate:
  316. case LengthModifier::AsMAllocate:
  317. return ArgType::Invalid();
  318. }
  319. }
  320. switch (CS.getKind()) {
  321. case ConversionSpecifier::sArg:
  322. if (LM.getKind() == LengthModifier::AsWideChar) {
  323. if (IsObjCLiteral)
  324. return Ctx.getPointerType(Ctx.UnsignedShortTy.withConst());
  325. return ArgType(ArgType::WCStrTy, "wchar_t *");
  326. }
  327. return ArgType::CStrTy;
  328. case ConversionSpecifier::SArg:
  329. if (IsObjCLiteral)
  330. return Ctx.getPointerType(Ctx.UnsignedShortTy.withConst());
  331. return ArgType(ArgType::WCStrTy, "wchar_t *");
  332. case ConversionSpecifier::CArg:
  333. if (IsObjCLiteral)
  334. return Ctx.UnsignedShortTy;
  335. return ArgType(Ctx.WCharTy, "wchar_t");
  336. case ConversionSpecifier::pArg:
  337. return ArgType::CPointerTy;
  338. case ConversionSpecifier::ObjCObjArg:
  339. return ArgType::ObjCPointerTy;
  340. default:
  341. break;
  342. }
  343. // FIXME: Handle other cases.
  344. return ArgType();
  345. }
  346. bool PrintfSpecifier::fixType(QualType QT, const LangOptions &LangOpt,
  347. ASTContext &Ctx, bool IsObjCLiteral) {
  348. // %n is different from other conversion specifiers; don't try to fix it.
  349. if (CS.getKind() == ConversionSpecifier::nArg)
  350. return false;
  351. // Handle Objective-C objects first. Note that while the '%@' specifier will
  352. // not warn for structure pointer or void pointer arguments (because that's
  353. // how CoreFoundation objects are implemented), we only show a fixit for '%@'
  354. // if we know it's an object (block, id, class, or __attribute__((NSObject))).
  355. if (QT->isObjCRetainableType()) {
  356. if (!IsObjCLiteral)
  357. return false;
  358. CS.setKind(ConversionSpecifier::ObjCObjArg);
  359. // Disable irrelevant flags
  360. HasThousandsGrouping = false;
  361. HasPlusPrefix = false;
  362. HasSpacePrefix = false;
  363. HasAlternativeForm = false;
  364. HasLeadingZeroes = false;
  365. Precision.setHowSpecified(OptionalAmount::NotSpecified);
  366. LM.setKind(LengthModifier::None);
  367. return true;
  368. }
  369. // Handle strings next (char *, wchar_t *)
  370. if (QT->isPointerType() && (QT->getPointeeType()->isAnyCharacterType())) {
  371. CS.setKind(ConversionSpecifier::sArg);
  372. // Disable irrelevant flags
  373. HasAlternativeForm = 0;
  374. HasLeadingZeroes = 0;
  375. // Set the long length modifier for wide characters
  376. if (QT->getPointeeType()->isWideCharType())
  377. LM.setKind(LengthModifier::AsWideChar);
  378. else
  379. LM.setKind(LengthModifier::None);
  380. return true;
  381. }
  382. // If it's an enum, get its underlying type.
  383. if (const EnumType *ETy = QT->getAs<EnumType>())
  384. QT = ETy->getDecl()->getIntegerType();
  385. // We can only work with builtin types.
  386. const BuiltinType *BT = QT->getAs<BuiltinType>();
  387. if (!BT)
  388. return false;
  389. // Set length modifier
  390. switch (BT->getKind()) {
  391. case BuiltinType::Bool:
  392. case BuiltinType::WChar_U:
  393. case BuiltinType::WChar_S:
  394. case BuiltinType::Char16:
  395. case BuiltinType::Char32:
  396. case BuiltinType::UInt128:
  397. case BuiltinType::Int128:
  398. case BuiltinType::Half:
  399. // Various types which are non-trivial to correct.
  400. return false;
  401. #define SIGNED_TYPE(Id, SingletonId)
  402. #define UNSIGNED_TYPE(Id, SingletonId)
  403. #define FLOATING_TYPE(Id, SingletonId)
  404. #define BUILTIN_TYPE(Id, SingletonId) \
  405. case BuiltinType::Id:
  406. #include "clang/AST/BuiltinTypes.def"
  407. // Misc other stuff which doesn't make sense here.
  408. return false;
  409. case BuiltinType::UInt:
  410. case BuiltinType::Int:
  411. case BuiltinType::Float:
  412. case BuiltinType::Double:
  413. LM.setKind(LengthModifier::None);
  414. break;
  415. case BuiltinType::Char_U:
  416. case BuiltinType::UChar:
  417. case BuiltinType::Char_S:
  418. case BuiltinType::SChar:
  419. LM.setKind(LengthModifier::AsChar);
  420. break;
  421. case BuiltinType::Short:
  422. case BuiltinType::UShort:
  423. LM.setKind(LengthModifier::AsShort);
  424. break;
  425. case BuiltinType::Long:
  426. case BuiltinType::ULong:
  427. LM.setKind(LengthModifier::AsLong);
  428. break;
  429. case BuiltinType::LongLong:
  430. case BuiltinType::ULongLong:
  431. LM.setKind(LengthModifier::AsLongLong);
  432. break;
  433. case BuiltinType::LongDouble:
  434. LM.setKind(LengthModifier::AsLongDouble);
  435. break;
  436. }
  437. // Handle size_t, ptrdiff_t, etc. that have dedicated length modifiers in C99.
  438. if (isa<TypedefType>(QT) && (LangOpt.C99 || LangOpt.CPlusPlus0x))
  439. namedTypeToLengthModifier(QT, LM);
  440. // If fixing the length modifier was enough, we are done.
  441. if (hasValidLengthModifier(Ctx.getTargetInfo())) {
  442. const analyze_printf::ArgType &ATR = getArgType(Ctx, IsObjCLiteral);
  443. if (ATR.isValid() && ATR.matchesType(Ctx, QT))
  444. return true;
  445. }
  446. // Set conversion specifier and disable any flags which do not apply to it.
  447. // Let typedefs to char fall through to int, as %c is silly for uint8_t.
  448. if (isa<TypedefType>(QT) && QT->isAnyCharacterType()) {
  449. CS.setKind(ConversionSpecifier::cArg);
  450. LM.setKind(LengthModifier::None);
  451. Precision.setHowSpecified(OptionalAmount::NotSpecified);
  452. HasAlternativeForm = 0;
  453. HasLeadingZeroes = 0;
  454. HasPlusPrefix = 0;
  455. }
  456. // Test for Floating type first as LongDouble can pass isUnsignedIntegerType
  457. else if (QT->isRealFloatingType()) {
  458. CS.setKind(ConversionSpecifier::fArg);
  459. }
  460. else if (QT->isSignedIntegerType()) {
  461. CS.setKind(ConversionSpecifier::dArg);
  462. HasAlternativeForm = 0;
  463. }
  464. else if (QT->isUnsignedIntegerType()) {
  465. CS.setKind(ConversionSpecifier::uArg);
  466. HasAlternativeForm = 0;
  467. HasPlusPrefix = 0;
  468. } else {
  469. llvm_unreachable("Unexpected type");
  470. }
  471. return true;
  472. }
  473. void PrintfSpecifier::toString(raw_ostream &os) const {
  474. // Whilst some features have no defined order, we are using the order
  475. // appearing in the C99 standard (ISO/IEC 9899:1999 (E) 7.19.6.1)
  476. os << "%";
  477. // Positional args
  478. if (usesPositionalArg()) {
  479. os << getPositionalArgIndex() << "$";
  480. }
  481. // Conversion flags
  482. if (IsLeftJustified) os << "-";
  483. if (HasPlusPrefix) os << "+";
  484. if (HasSpacePrefix) os << " ";
  485. if (HasAlternativeForm) os << "#";
  486. if (HasLeadingZeroes) os << "0";
  487. // Minimum field width
  488. FieldWidth.toString(os);
  489. // Precision
  490. Precision.toString(os);
  491. // Length modifier
  492. os << LM.toString();
  493. // Conversion specifier
  494. os << CS.toString();
  495. }
  496. bool PrintfSpecifier::hasValidPlusPrefix() const {
  497. if (!HasPlusPrefix)
  498. return true;
  499. // The plus prefix only makes sense for signed conversions
  500. switch (CS.getKind()) {
  501. case ConversionSpecifier::dArg:
  502. case ConversionSpecifier::DArg:
  503. case ConversionSpecifier::iArg:
  504. case ConversionSpecifier::fArg:
  505. case ConversionSpecifier::FArg:
  506. case ConversionSpecifier::eArg:
  507. case ConversionSpecifier::EArg:
  508. case ConversionSpecifier::gArg:
  509. case ConversionSpecifier::GArg:
  510. case ConversionSpecifier::aArg:
  511. case ConversionSpecifier::AArg:
  512. return true;
  513. default:
  514. return false;
  515. }
  516. }
  517. bool PrintfSpecifier::hasValidAlternativeForm() const {
  518. if (!HasAlternativeForm)
  519. return true;
  520. // Alternate form flag only valid with the oxXaAeEfFgG conversions
  521. switch (CS.getKind()) {
  522. case ConversionSpecifier::oArg:
  523. case ConversionSpecifier::OArg:
  524. case ConversionSpecifier::xArg:
  525. case ConversionSpecifier::XArg:
  526. case ConversionSpecifier::aArg:
  527. case ConversionSpecifier::AArg:
  528. case ConversionSpecifier::eArg:
  529. case ConversionSpecifier::EArg:
  530. case ConversionSpecifier::fArg:
  531. case ConversionSpecifier::FArg:
  532. case ConversionSpecifier::gArg:
  533. case ConversionSpecifier::GArg:
  534. return true;
  535. default:
  536. return false;
  537. }
  538. }
  539. bool PrintfSpecifier::hasValidLeadingZeros() const {
  540. if (!HasLeadingZeroes)
  541. return true;
  542. // Leading zeroes flag only valid with the diouxXaAeEfFgG conversions
  543. switch (CS.getKind()) {
  544. case ConversionSpecifier::dArg:
  545. case ConversionSpecifier::DArg:
  546. case ConversionSpecifier::iArg:
  547. case ConversionSpecifier::oArg:
  548. case ConversionSpecifier::OArg:
  549. case ConversionSpecifier::uArg:
  550. case ConversionSpecifier::UArg:
  551. case ConversionSpecifier::xArg:
  552. case ConversionSpecifier::XArg:
  553. case ConversionSpecifier::aArg:
  554. case ConversionSpecifier::AArg:
  555. case ConversionSpecifier::eArg:
  556. case ConversionSpecifier::EArg:
  557. case ConversionSpecifier::fArg:
  558. case ConversionSpecifier::FArg:
  559. case ConversionSpecifier::gArg:
  560. case ConversionSpecifier::GArg:
  561. return true;
  562. default:
  563. return false;
  564. }
  565. }
  566. bool PrintfSpecifier::hasValidSpacePrefix() const {
  567. if (!HasSpacePrefix)
  568. return true;
  569. // The space prefix only makes sense for signed conversions
  570. switch (CS.getKind()) {
  571. case ConversionSpecifier::dArg:
  572. case ConversionSpecifier::DArg:
  573. case ConversionSpecifier::iArg:
  574. case ConversionSpecifier::fArg:
  575. case ConversionSpecifier::FArg:
  576. case ConversionSpecifier::eArg:
  577. case ConversionSpecifier::EArg:
  578. case ConversionSpecifier::gArg:
  579. case ConversionSpecifier::GArg:
  580. case ConversionSpecifier::aArg:
  581. case ConversionSpecifier::AArg:
  582. return true;
  583. default:
  584. return false;
  585. }
  586. }
  587. bool PrintfSpecifier::hasValidLeftJustified() const {
  588. if (!IsLeftJustified)
  589. return true;
  590. // The left justified flag is valid for all conversions except n
  591. switch (CS.getKind()) {
  592. case ConversionSpecifier::nArg:
  593. return false;
  594. default:
  595. return true;
  596. }
  597. }
  598. bool PrintfSpecifier::hasValidThousandsGroupingPrefix() const {
  599. if (!HasThousandsGrouping)
  600. return true;
  601. switch (CS.getKind()) {
  602. case ConversionSpecifier::dArg:
  603. case ConversionSpecifier::DArg:
  604. case ConversionSpecifier::iArg:
  605. case ConversionSpecifier::uArg:
  606. case ConversionSpecifier::UArg:
  607. case ConversionSpecifier::fArg:
  608. case ConversionSpecifier::FArg:
  609. case ConversionSpecifier::gArg:
  610. case ConversionSpecifier::GArg:
  611. return true;
  612. default:
  613. return false;
  614. }
  615. }
  616. bool PrintfSpecifier::hasValidPrecision() const {
  617. if (Precision.getHowSpecified() == OptionalAmount::NotSpecified)
  618. return true;
  619. // Precision is only valid with the diouxXaAeEfFgGs conversions
  620. switch (CS.getKind()) {
  621. case ConversionSpecifier::dArg:
  622. case ConversionSpecifier::DArg:
  623. case ConversionSpecifier::iArg:
  624. case ConversionSpecifier::oArg:
  625. case ConversionSpecifier::OArg:
  626. case ConversionSpecifier::uArg:
  627. case ConversionSpecifier::UArg:
  628. case ConversionSpecifier::xArg:
  629. case ConversionSpecifier::XArg:
  630. case ConversionSpecifier::aArg:
  631. case ConversionSpecifier::AArg:
  632. case ConversionSpecifier::eArg:
  633. case ConversionSpecifier::EArg:
  634. case ConversionSpecifier::fArg:
  635. case ConversionSpecifier::FArg:
  636. case ConversionSpecifier::gArg:
  637. case ConversionSpecifier::GArg:
  638. case ConversionSpecifier::sArg:
  639. return true;
  640. default:
  641. return false;
  642. }
  643. }
  644. bool PrintfSpecifier::hasValidFieldWidth() const {
  645. if (FieldWidth.getHowSpecified() == OptionalAmount::NotSpecified)
  646. return true;
  647. // The field width is valid for all conversions except n
  648. switch (CS.getKind()) {
  649. case ConversionSpecifier::nArg:
  650. return false;
  651. default:
  652. return true;
  653. }
  654. }