Skip to content

Commit 0f0f3aa

Browse files
Fix put_time() crash on invalid format specifier (#4840)
Co-authored-by: Stephan T. Lavavej <[email protected]>
1 parent c4d0517 commit 0f0f3aa

File tree

2 files changed

+98
-7
lines changed

2 files changed

+98
-7
lines changed

stl/inc/xloctime

+57-5
Original file line numberDiff line numberDiff line change
@@ -651,6 +651,21 @@ protected:
651651
__CLR_OR_THIS_CALL ~time_get_byname() noexcept override {}
652652
};
653653

654+
// C23 7.29.3.5 "The strftime function"/3
655+
_INLINE_VAR constexpr char _Valid_strftime_specifiers[] = {'a', 'A', 'b', 'B', 'c', 'C', 'd', 'D', 'e', 'F', 'g', 'G',
656+
'h', 'H', 'I', 'j', 'm', 'M', 'n', 'p', 'r', 'R', 'S', 't', 'T', 'u', 'U', 'V', 'w', 'W', 'x', 'X', 'y', 'Y', 'z',
657+
'Z'};
658+
659+
_NODISCARD constexpr bool _Is_valid_strftime_specifier(const char _Specifier) {
660+
for (const auto& _Valid_specifier : _Valid_strftime_specifiers) {
661+
if (_Specifier == _Valid_specifier) {
662+
return true;
663+
}
664+
}
665+
666+
return false;
667+
}
668+
654669
_EXPORT_STD extern "C++" template <class _Elem, class _OutIt = ostreambuf_iterator<_Elem, char_traits<_Elem>>>
655670
class time_put : public locale::facet { // facet for converting encoded times to text
656671
public:
@@ -687,7 +702,19 @@ public:
687702
_Specifier = _Ctype_fac.narrow(*_Fmtfirst);
688703
}
689704

690-
_Dest = do_put(_Dest, _Iosbase, _Fill, _Pt, _Specifier, _Modifier); // convert a single field
705+
if (_Specifier == '%' && _Modifier == '\0') {
706+
// if the specifier is percent and no modifier is set, just append it
707+
*_Dest++ = _Percent;
708+
} else if (!_Is_valid_strftime_specifier(_Specifier)) {
709+
// no valid specifier, directly copy as literal elements
710+
*_Dest++ = _Percent;
711+
if (_Modifier != '\0') {
712+
*_Dest++ = _Modifier;
713+
}
714+
*_Dest++ = _Specifier;
715+
} else {
716+
_Dest = do_put(_Dest, _Iosbase, _Fill, _Pt, _Specifier, _Modifier); // convert a single field
717+
}
691718
}
692719
}
693720

@@ -811,7 +838,19 @@ public:
811838
_Specifier = _Ctype_fac.narrow(*_Fmtfirst);
812839
}
813840

814-
_Dest = do_put(_Dest, _Iosbase, _Fill, _Pt, _Specifier, _Modifier); // convert a single field
841+
if (_Specifier == '%' && _Modifier == '\0') {
842+
// if the specifier is percent and no modifier is set, just append it
843+
*_Dest++ = _Percent;
844+
} else if (!_Is_valid_strftime_specifier(_Specifier)) {
845+
// no valid specifier, directly copy as literal elements
846+
*_Dest++ = _Percent;
847+
if (_Modifier != '\0') {
848+
*_Dest++ = _Raw;
849+
}
850+
*_Dest++ = *_Fmtfirst;
851+
} else {
852+
_Dest = do_put(_Dest, _Iosbase, _Fill, _Pt, _Specifier, _Modifier); // convert a single field
853+
}
815854
}
816855
}
817856

@@ -927,14 +966,15 @@ public:
927966
*_Dest++ = _Fmtfirst[-1];
928967
break;
929968
} else { // get specifier after %
930-
char _Specifier = _Ctype_fac.narrow(*_Fmtfirst);
969+
_Elem _Raw = *_Fmtfirst;
970+
char _Specifier = _Ctype_fac.narrow(_Raw);
931971
char _Modifier = '\0';
932972
_Elem _Percent = _Fmtfirst[-1];
933973

934974
if (_Specifier == 'E' || _Specifier == 'O' || _Specifier == 'Q' || _Specifier == '#') {
935975
if (++_Fmtfirst == _Fmtlast) { // no specifier, copy %[E0Q#] as literal elements
936976
*_Dest++ = _Percent;
937-
*_Dest++ = _Specifier;
977+
*_Dest++ = _Raw;
938978
break;
939979
}
940980

@@ -943,7 +983,19 @@ public:
943983
_Specifier = _Ctype_fac.narrow(*_Fmtfirst);
944984
}
945985

946-
_Dest = do_put(_Dest, _Iosbase, _Fill, _Pt, _Specifier, _Modifier); // convert a single field
986+
if (_Specifier == '%' && _Modifier == '\0') {
987+
// if the specifier is percent and no modifier is set, just append it
988+
*_Dest++ = _Percent;
989+
} else if (!_Is_valid_strftime_specifier(_Specifier)) {
990+
// no valid specifier, directly copy as literal elements
991+
*_Dest++ = _Percent;
992+
if (_Modifier != '\0') {
993+
*_Dest++ = _Raw;
994+
}
995+
*_Dest++ = *_Fmtfirst;
996+
} else {
997+
_Dest = do_put(_Dest, _Iosbase, _Fill, _Pt, _Specifier, _Modifier); // convert a single field
998+
}
947999
}
9481000
}
9491001

tests/std/tests/Dev11_0836436_get_time/test.cpp

+41-2
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,7 @@ void test_invalid_argument();
110110
void test_buffer_resizing();
111111
void test_gh_2618();
112112
void test_gh_2848();
113+
void test_gh_4820();
113114

114115
int main() {
115116
assert(read_hour("12 AM") == 0);
@@ -157,6 +158,7 @@ int main() {
157158
test_buffer_resizing();
158159
test_gh_2618();
159160
test_gh_2848();
161+
test_gh_4820();
160162
}
161163

162164
typedef istreambuf_iterator<char> Iter;
@@ -792,16 +794,17 @@ void test_invalid_argument() {
792794
time_t t = time(nullptr);
793795
tm currentTime;
794796
localtime_s(&currentTime, &t);
797+
currentTime.tm_hour = 25; // set invalid hour
795798

796799
{
797800
wstringstream wss;
798-
wss << put_time(&currentTime, L"%Y-%m-%d-%H-%M-%s");
801+
wss << put_time(&currentTime, L"%Y-%m-%d-%H-%M");
799802
assert(wss.rdstate() == ios_base::badbit);
800803
}
801804

802805
{
803806
stringstream ss;
804-
ss << put_time(&currentTime, "%Y-%m-%d-%H-%M-%s");
807+
ss << put_time(&currentTime, "%Y-%m-%d-%H-%M");
805808
assert(ss.rdstate() == ios_base::badbit);
806809
}
807810
#endif // _M_CEE_PURE
@@ -905,3 +908,39 @@ void test_gh_2848() {
905908
assert(err == (ios_base::eofbit | ios_base::failbit));
906909
}
907910
}
911+
912+
void test_gh_4820() {
913+
// GH-4820 <iomanip>: std::put_time should copy unknown conversion specifiers instead of crash
914+
time_t t = time(nullptr);
915+
tm currentTime;
916+
localtime_s(&currentTime, &t);
917+
918+
// Case 1: Test various unknown conversion specifiers.
919+
// Case 2: "%%" is a known escape sequence with a dedicated fast path.
920+
// Case 3: "% " is percent followed by space, which is an unknown conversion specifier.
921+
// Case 4: "%E%Z" is parsed as "%E%" followed by "Z", so it should be copied unchanged,
922+
// even though "%Z" by itself would be a known conversion specifier (time zone name).
923+
// (In case 1, "%E%J" is parsed the same way; the difference is that "%J" would be unknown.)
924+
{
925+
wstringstream wss;
926+
wss << put_time(&currentTime, L"1:%Ei%!%E%J%P 2:%% 3:% 4:%E%Z");
927+
assert(wss.rdstate() == ios_base::goodbit);
928+
assert(wss.str() == L"1:%Ei%!%E%J%P 2:% 3:% 4:%E%Z");
929+
}
930+
931+
{
932+
stringstream ss;
933+
ss << put_time(&currentTime, "1:%Ei%!%E%J%P 2:%% 3:% 4:%E%Z");
934+
assert(ss.rdstate() == ios_base::goodbit);
935+
assert(ss.str() == "1:%Ei%!%E%J%P 2:% 3:% 4:%E%Z");
936+
}
937+
938+
// Also verify that wide characters aren't truncated.
939+
// This tests a character appearing by itself, two as specifiers, and two as modified specifiers.
940+
{
941+
wstringstream wss;
942+
wss << put_time(&currentTime, L"\x043a%\x043e%\x0448%E\x043a%O\x0430");
943+
assert(wss.rdstate() == ios_base::goodbit);
944+
assert(wss.str() == L"\x043a%\x043e%\x0448%E\x043a%O\x0430");
945+
}
946+
}

0 commit comments

Comments
 (0)