/* Copyright (c) 2017, 2019, Oracle and/or its affiliates. All rights reserved. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2.0, as published by the Free Software Foundation. This program is also distributed with certain software (including but not limited to OpenSSL) that is licensed under separate terms, as designated in a particular file or component or in included license documentation. The authors of MySQL hereby grant you an additional permission to link the program and your derivative works with the separately licensed software that they have included with MySQL. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License, version 2.0, for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "sql/regexp/regexp_facade.h" #include #include #include "my_pointer_arithmetic.h" #include "sql/mysqld.h" // make_unique_destroy_only #include "sql/regexp/regexp_engine.h" #include "sql_string.h" #include "template_utils.h" namespace regexp { /** Evaluates an expression to an output buffer, performing character set conversion to regexp_lib_charset if necessary. The procedure supports the optimization that code points before the start position are not converted to UTF-16. @param expr The expression to be printed. @param[out] out Will be cleared, and the result stored. @param skip If present, this number of code points will be exempted from conversion. If the string is smaller than that, an out of bounds error is raised. @return false on success, true on error. */ static bool EvalExprToCharset(Item *expr, std::u16string *out, int skip = 0) { alignas(sizeof(UChar)) char aligned_buff[MAX_FIELD_WIDTH]; String aligned_str(aligned_buff, sizeof(aligned_buff), &my_charset_bin); String *s = expr->val_str(&aligned_str); if (s == nullptr) return true; if (s->length() == 0) { out->clear(); return false; } const CHARSET_INFO *cs = s->charset(); size_t bytes_to_skip = cs->cset->charpos(cs, s->ptr(), s->ptr() + s->length(), skip); if (bytes_to_skip >= s->length()) { /* The following error would normally be raised by ICU, but since this is an alternative code path - i.e. we prune away from charset conversion the portion of the string that is skipped - we have to raise it ourselves. */ my_error(ER_REGEXP_INDEX_OUTOFBOUNDS_ERROR, MYF(0)); out->clear(); return false; } if (expr->collation.collation != regexp_lib_charset) { // Character set conversion is called for. uint max_char_size = regexp_lib_charset->mbmaxlen; // cast to signed for subtraction auto length = static_cast(s->length()) - bytes_to_skip; out->resize(length * max_char_size / sizeof(UChar)); auto to = pointer_cast(&(*out)[0]); size_t to_size = out->size() * sizeof(UChar); const char *start = s->ptr() + bytes_to_skip; uint errors; size_t converted_size = my_convert(to, to_size, regexp_lib_charset, start, length, s->charset(), &errors); if (errors > 0) return true; DBUG_ASSERT(converted_size % sizeof(UChar) == 0); out->resize(converted_size / sizeof(UChar)); return false; } // No conversion needed; just copy into the u16string. // However: val_str() may ignore the input argument, // and return a pointer to some other buffer. if (!is_aligned_to(s->ptr(), alignof(UChar))) { DBUG_ASSERT(s != &aligned_str); aligned_str.copy(*s); s = &aligned_str; } out->clear(); out->insert(out->end(), pointer_cast(s->ptr()), pointer_cast(s->ptr() + s->length())); return false; } bool Regexp_facade::SetPattern(Item *pattern_expr, uint32_t flags) { if (pattern_expr == nullptr) { m_engine = nullptr; return true; } if (m_engine == nullptr) // Called for the first time. return SetupEngine(pattern_expr, flags); /* We don't need to recompile the regular expression if the pattern is a constant in the query and the flags are the same. */ if (pattern_expr->const_item() && flags == m_engine->flags()) return false; return SetupEngine(pattern_expr, flags); } bool Regexp_facade::Reset(Item *subject_expr, int start) { DBUG_TRACE; if (m_engine == nullptr || EvalExprToCharset(subject_expr, &m_current_subject, start - 1)) return true; m_engine->Reset(m_current_subject); return false; } int Regexp_facade::ConvertCodePointToLibPosition(int position) const { auto start = reinterpret_cast(m_current_subject.c_str()); const char *end = start + m_current_subject.length() * sizeof(char16_t); MY_CHARSET_HANDLER *cset = regexp_lib_charset->cset; return cset->charpos(regexp_lib_charset, start, end, position - 1) / sizeof(char16_t); } int Regexp_facade::ConvertLibPositionToCodePoint(int position) const { auto start = reinterpret_cast(m_current_subject.c_str()); const char *end = start + position * sizeof(char16_t); MY_CHARSET_HANDLER *cset = regexp_lib_charset->cset; return cset->numchars(regexp_lib_charset, start, end); } Mysql::Nullable Regexp_facade::Matches(Item *subject_expr, int start, int occurrence) { DBUG_TRACE; if (Reset(subject_expr, start)) return Mysql::Nullable(); /* As far as ICU is concerned, we always start on position 0, since we didn't convert the characters before 'start'. */ return m_engine->Matches(0, occurrence); } Mysql::Nullable Regexp_facade::Find(Item *subject_expr, int start, int occurrence, bool after_match) { Nullable match_found = Matches(subject_expr, start, occurrence); if (!match_found.has_value()) return Mysql::Nullable(); if (!match_found.value()) return 0; int native_start = after_match ? m_engine->EndOfMatch() : m_engine->StartOfMatch(); return ConvertLibPositionToCodePoint(native_start) + start; } String *Regexp_facade::Replace(Item *subject_expr, Item *replacement_expr, int64_t start, int occurrence, String *result) { DBUG_TRACE; String replacement_buf; std::u16string replacement(MAX_FIELD_WIDTH, '\0'); if (EvalExprToCharset(replacement_expr, &replacement)) return nullptr; if (Reset(subject_expr)) return nullptr; const std::u16string &result_buffer = m_engine->Replace( replacement, ConvertCodePointToLibPosition(start), occurrence); uint conversion_error; size_t number_unaligned_characters; if (result->needs_conversion(result->length(), regexp_lib_charset, result->charset(), &number_unaligned_characters)) { if (result->copy(pointer_cast(result_buffer.data()), result_buffer.size() * sizeof(UChar), regexp_lib_charset, result->charset(), &conversion_error)) return nullptr; } else result->set(pointer_cast(result_buffer.data()), result_buffer.size() * sizeof(UChar), regexp_lib_charset); return result; } String *Regexp_facade::Substr(Item *subject_expr, int start, int occurrence, String *result) { if (Reset(subject_expr)) return nullptr; if (!m_engine->Matches(ConvertCodePointToLibPosition(start), occurrence)) { m_engine->CheckError(); return nullptr; } int substart, sublength; std::tie(substart, sublength) = m_engine->MatchedSubstring(); if (m_engine->CheckError()) return nullptr; uint conversion_error; auto substartptr = pointer_cast(m_current_subject.c_str()) + substart; size_t number_unaligned_characters; if (result->needs_conversion(sublength, regexp_lib_charset, result->charset(), &number_unaligned_characters)) { if (result->copy(substartptr, sublength, regexp_lib_charset, result->charset(), &conversion_error)) return nullptr; } else result->set(substartptr, sublength, regexp_lib_charset); return result; } bool Regexp_facade::SetupEngine(Item *pattern_expr, uint flags) { DBUG_TRACE; std::u16string pattern; if (EvalExprToCharset(pattern_expr, &pattern)) { m_engine = nullptr; return false; } // Actually compile the regular expression. m_engine = make_unique_destroy_only( *THR_MALLOC, pattern, flags, opt_regexp_stack_limit, opt_regexp_time_limit); // If something went wrong, an error was raised. return m_engine->IsError(); } } // namespace regexp