Re-use the Regex VM when completing filenames to reduce allocations

By re-using the VM we avoid re-allocating the threads and saves
buffers over and over again. We can just re-use the ones from the
previous matching.
This commit is contained in:
Maxime Coste 2021-03-03 20:33:02 +11:00
parent 60c44e1623
commit f4a639e078

View File

@ -512,14 +512,20 @@ CandidateList complete_filename(StringView prefix, const Regex& ignored_regex,
auto [dirname, fileprefix] = split_path(prefix); auto [dirname, fileprefix] = split_path(prefix);
auto parsed_dirname = parse_filename(dirname); auto parsed_dirname = parse_filename(dirname);
const bool check_ignored_regex = not ignored_regex.empty() and Optional<ThreadedRegexVM<const char*, RegexMode::Forward | RegexMode::AnyMatch | RegexMode::NoSaves>> vm;
not regex_match(fileprefix.begin(), fileprefix.end(), ignored_regex); if (not ignored_regex.empty())
{
vm.emplace(*ignored_regex.impl());
if (vm->exec(fileprefix.begin(), fileprefix.end(), fileprefix.begin(), fileprefix.end(), RegexExecFlags::None))
vm.reset();
}
const bool only_dirs = (flags & FilenameFlags::OnlyDirectories); const bool only_dirs = (flags & FilenameFlags::OnlyDirectories);
auto filter = [&ignored_regex, check_ignored_regex, only_dirs](const dirent& entry, struct stat& st) auto filter = [&vm, only_dirs](const dirent& entry, struct stat& st)
{ {
StringView name{entry.d_name}; StringView name{entry.d_name};
return (not check_ignored_regex or not regex_match(name.begin(), name.end(), ignored_regex)) and return (not vm or not vm->exec(name.begin(), name.end(), name.begin(), name.end(), RegexExecFlags::None)) and
(not only_dirs or S_ISDIR(st.st_mode)); (not only_dirs or S_ISDIR(st.st_mode));
}; };
auto files = list_files(parsed_dirname, filter); auto files = list_files(parsed_dirname, filter);