Detect and throw on a class of bad regexes that we mistakenly accepted before. Thanks...
authorMarshall Clow <mclow.lists@gmail.com>
Thu, 23 Jul 2015 18:27:51 +0000 (18:27 +0000)
committerMarshall Clow <mclow.lists@gmail.com>
Thu, 23 Jul 2015 18:27:51 +0000 (18:27 +0000)
llvm-svn: 243030

libcxx/include/regex
libcxx/test/std/re/re.regex/re.regex.construct/bad_escape.pass.cpp
libcxx/test/std/re/re.regex/re.regex.construct/bad_repeat.pass.cpp [new file with mode: 0644]

index 6ac5e1d..698278c 100644 (file)
@@ -4305,6 +4305,14 @@ basic_regex<_CharT, _Traits>::__parse_atom(_ForwardIterator __first,
                 }
             }
             break;
+        case '*':
+        case '+':
+        case '?':
+        case '{':
+#ifndef _LIBCPP_NO_EXCEPTIONS
+            throw regex_error(regex_constants::error_badrepeat);
+#endif
+            break;
         default:
             __first = __parse_pattern_character(__first, __last);
             break;
index 9455527..ddf2607 100644 (file)
@@ -22,7 +22,7 @@ static bool error_escape_thrown(const char *pat)
     bool result = false;
     try {
         std::regex re(pat);
-    } catch (std::regex_error &ex) {
+    } catch (const std::regex_error &ex) {
         result = (ex.code() == std::regex_constants::error_escape);
     }
     return result;
diff --git a/libcxx/test/std/re/re.regex/re.regex.construct/bad_repeat.pass.cpp b/libcxx/test/std/re/re.regex/re.regex.construct/bad_repeat.pass.cpp
new file mode 100644 (file)
index 0000000..bc70ec1
--- /dev/null
@@ -0,0 +1,42 @@
+//===----------------------------------------------------------------------===//
+//
+//                     The LLVM Compiler Infrastructure
+//
+// This file is dual licensed under the MIT and the University of Illinois Open
+// Source Licenses. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+// <regex>
+
+// template <class charT, class traits = regex_traits<charT>> class basic_regex;
+
+// template <class ST, class SA>
+//    basic_regex(const basic_string<charT, ST, SA>& s);
+
+#include <regex>
+#include <cassert>
+
+static bool error_badrepeat_thrown(const char *pat) 
+{
+    bool result = false;
+    try {
+        std::regex re(pat);
+    } catch (const std::regex_error &ex) {
+        result = (ex.code() == std::regex_constants::error_badrepeat);
+    }
+    return result;
+}
+
+int main() 
+{
+    assert(error_badrepeat_thrown("?a"));
+    assert(error_badrepeat_thrown("*a"));
+    assert(error_badrepeat_thrown("+a"));
+    assert(error_badrepeat_thrown("{a"));
+
+    assert(error_badrepeat_thrown("?(a+)"));
+    assert(error_badrepeat_thrown("*(a+)"));
+    assert(error_badrepeat_thrown("+(a+)"));
+    assert(error_badrepeat_thrown("{(a+)"));
+}