Enable C++2a Chrono Literals

C++2a via http://wg21.link/p0355 permits the library
literals of 'd' and 'y'. This patch enables them in the
Lexer so that they can be properly parsed.

Note that 'd' gets confused with the hex character, so
modifications to how octal, binary, and decimal numbers are
parsed were required. Since this is simply making previously
invalid code legal, this should be fine.

Hex still greedily parses the 'd' as a hexit, since it would
a: violate [lex.ext]p1
b: break existing code.

Differential Revision: https://reviews.llvm.org/D49504

llvm-svn: 337454
This commit is contained in:
Erich Keane 2018-07-19 13:36:57 +00:00
parent 18b404a521
commit a67eb91047
2 changed files with 35 additions and 3 deletions

View File

@ -751,7 +751,8 @@ void NumericLiteralParser::ParseDecimalOrOctalCommon(SourceLocation TokLoc){
// If we have a hex digit other than 'e' (which denotes a FP exponent) then // If we have a hex digit other than 'e' (which denotes a FP exponent) then
// the code is using an incorrect base. // the code is using an incorrect base.
if (isHexDigit(*s) && *s != 'e' && *s != 'E') { if (isHexDigit(*s) && *s != 'e' && *s != 'E' &&
!isValidUDSuffix(PP.getLangOpts(), StringRef(s, ThisTokEnd - s))) {
PP.Diag(PP.AdvanceToTokenCharacter(TokLoc, s-ThisTokBegin), PP.Diag(PP.AdvanceToTokenCharacter(TokLoc, s-ThisTokBegin),
diag::err_invalid_digit) << StringRef(s, 1) << (radix == 8 ? 1 : 0); diag::err_invalid_digit) << StringRef(s, 1) << (radix == 8 ? 1 : 0);
hadError = true; hadError = true;
@ -804,12 +805,14 @@ bool NumericLiteralParser::isValidUDSuffix(const LangOptions &LangOpts,
if (!LangOpts.CPlusPlus14) if (!LangOpts.CPlusPlus14)
return false; return false;
// In C++1y, "s", "h", "min", "ms", "us", and "ns" are used in the library. // In C++14, "s", "h", "min", "ms", "us", and "ns" are used in the library.
// Per tweaked N3660, "il", "i", and "if" are also used in the library. // Per tweaked N3660, "il", "i", and "if" are also used in the library.
// In C++2a "d" and "y" are used in the library.
return llvm::StringSwitch<bool>(Suffix) return llvm::StringSwitch<bool>(Suffix)
.Cases("h", "min", "s", true) .Cases("h", "min", "s", true)
.Cases("ms", "us", "ns", true) .Cases("ms", "us", "ns", true)
.Cases("il", "i", "if", true) .Cases("il", "i", "if", true)
.Cases("d", "y", LangOpts.CPlusPlus2a)
.Default(false); .Default(false);
} }
@ -922,7 +925,9 @@ void NumericLiteralParser::ParseNumberStartingWithZero(SourceLocation TokLoc) {
s = SkipBinaryDigits(s); s = SkipBinaryDigits(s);
if (s == ThisTokEnd) { if (s == ThisTokEnd) {
// Done. // Done.
} else if (isHexDigit(*s)) { } else if (isHexDigit(*s) &&
!isValidUDSuffix(PP.getLangOpts(),
StringRef(s, ThisTokEnd - s))) {
PP.Diag(PP.AdvanceToTokenCharacter(TokLoc, s-ThisTokBegin), PP.Diag(PP.AdvanceToTokenCharacter(TokLoc, s-ThisTokBegin),
diag::err_invalid_digit) << StringRef(s, 1) << 2; diag::err_invalid_digit) << StringRef(s, 1) << 2;
hadError = true; hadError = true;

View File

@ -0,0 +1,27 @@
// RUN: %clang_cc1 -std=c++2a %s -include %s -verify
#ifndef INCLUDED
#define INCLUDED
#pragma clang system_header
namespace std {
namespace chrono {
struct day{};
struct year{};
}
constexpr chrono::day operator"" d(unsigned long long d) noexcept;
constexpr chrono::year operator"" y(unsigned long long y) noexcept;
}
#else
using namespace std;
chrono::day dec_d = 5d;
chrono::day oct_d = 05d;
chrono::day bin_d = 0b011d;
// expected-error@+3{{no viable conversion from 'int' to 'chrono::day'}}
// expected-note@9{{candidate constructor (the implicit copy constructor)}}
// expected-note@9{{candidate constructor (the implicit move constructor)}}
chrono::day hex_d = 0x44d;
chrono::year y = 10y;
#endif