@@ -651,6 +651,21 @@ protected:
651
651
__CLR_OR_THIS_CALL ~time_get_byname() noexcept override {}
652
652
};
653
653
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
+
654
669
_EXPORT_STD extern "C++" template <class _Elem, class _OutIt = ostreambuf_iterator<_Elem, char_traits<_Elem>>>
655
670
class time_put : public locale::facet { // facet for converting encoded times to text
656
671
public:
@@ -687,7 +702,19 @@ public:
687
702
_Specifier = _Ctype_fac.narrow(*_Fmtfirst);
688
703
}
689
704
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
+ }
691
718
}
692
719
}
693
720
@@ -811,7 +838,19 @@ public:
811
838
_Specifier = _Ctype_fac.narrow(*_Fmtfirst);
812
839
}
813
840
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
+ }
815
854
}
816
855
}
817
856
@@ -927,14 +966,15 @@ public:
927
966
*_Dest++ = _Fmtfirst[-1];
928
967
break;
929
968
} else { // get specifier after %
930
- char _Specifier = _Ctype_fac.narrow(*_Fmtfirst);
969
+ _Elem _Raw = *_Fmtfirst;
970
+ char _Specifier = _Ctype_fac.narrow(_Raw);
931
971
char _Modifier = '\0';
932
972
_Elem _Percent = _Fmtfirst[-1];
933
973
934
974
if (_Specifier == 'E' || _Specifier == 'O' || _Specifier == 'Q' || _Specifier == '#') {
935
975
if (++_Fmtfirst == _Fmtlast) { // no specifier, copy %[E0Q#] as literal elements
936
976
*_Dest++ = _Percent;
937
- *_Dest++ = _Specifier ;
977
+ *_Dest++ = _Raw ;
938
978
break;
939
979
}
940
980
@@ -943,7 +983,19 @@ public:
943
983
_Specifier = _Ctype_fac.narrow(*_Fmtfirst);
944
984
}
945
985
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
+ }
947
999
}
948
1000
}
949
1001
0 commit comments