0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-06-11 06:28:55 +02:00
construct/ircd/stringops.cc

155 lines
3.3 KiB
C++
Raw Normal View History

// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2019 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.
#include <ircd/simd.h>
2019-08-06 06:22:38 +02:00
size_t
ircd::indexof(const string_view &s,
const string_table &tab)
{
size_t i(0);
for(; i < tab.size(); ++i)
if(s == tab[i])
break;
return i;
}
ircd::string_view
ircd::toupper(const mutable_buffer &out,
const string_view &in)
noexcept
{
const auto stop
{
std::next(begin(in), std::min(size(in), size(out)))
};
const auto end
{
std::transform(begin(in), stop, begin(out), []
2019-08-06 06:22:38 +02:00
(auto&& c)
{
return std::toupper(c);
})
};
assert(intptr_t(begin(out)) <= intptr_t(end));
return string_view
{
data(out), size_t(std::distance(begin(out), end))
};
}
2020-03-14 19:33:36 +01:00
#if defined(IRCD_SIMD) && defined(__SSE2__)
2019-08-06 06:22:38 +02:00
ircd::string_view
ircd::tolower(const mutable_buffer &out,
const string_view &in)
noexcept
{
2019-08-07 06:41:14 +02:00
const auto stop
{
std::next(begin(in), std::min(size(in), size(out)))
};
const u128x1 *src_
2019-08-06 06:22:38 +02:00
{
reinterpret_cast<const u128x1 *>(begin(in))
2019-08-06 06:22:38 +02:00
};
u128x1 *dst
2019-08-06 06:22:38 +02:00
{
reinterpret_cast<u128x1 *>(begin(out))
2019-08-06 06:22:38 +02:00
};
while(intptr_t(src_) < intptr_t(stop) - ssize_t(sizeof(u128x1)))
2019-08-06 06:22:38 +02:00
{
const u128x1 lit_A1 { _mm_set1_epi8('A' - 1) };
const u128x1 lit_Z1 { _mm_set1_epi8('Z' + 1) };
const u128x1 addend { _mm_set1_epi8('a' - 'A') };
const u128x1 src { _mm_loadu_si128(src_++) };
const u128x1 gte_A { _mm_cmpgt_epi8(src, lit_A1) };
const u128x1 lte_Z { _mm_cmplt_epi8(src, lit_Z1) };
const u128x1 mask { _mm_and_si128(gte_A, lte_Z) };
const u128x1 ctrl_mask { _mm_and_si128(mask, addend) };
const u128x1 result { _mm_add_epi8(src, ctrl_mask) };
2019-08-07 06:41:14 +02:00
_mm_storeu_si128(dst++, result);
2019-08-06 06:22:38 +02:00
}
const auto end{std::transform
(
reinterpret_cast<const char *>(src_),
stop,
reinterpret_cast<char *>(dst),
::tolower
)};
assert(intptr_t(begin(out)) <= intptr_t(end));
return string_view{begin(out), end};
}
#else
ircd::string_view
ircd::tolower(const mutable_buffer &out,
const string_view &in)
noexcept
{
const auto stop
{
std::next(begin(in), std::min(size(in), size(out)))
};
const auto end
{
2019-08-06 06:22:38 +02:00
std::transform(begin(in), stop, begin(out), ::tolower)
};
assert(intptr_t(begin(out)) <= intptr_t(end));
return string_view
{
data(out), size_t(std::distance(begin(out), end))
};
}
2019-08-06 06:22:38 +02:00
#endif
std::string
ircd::replace(const string_view &s,
const char &before,
const string_view &after)
{
const uint32_t occurs
(
std::count(begin(s), end(s), before)
);
const size_t size
{
occurs? s.size() + (occurs * after.size()):
s.size() - occurs
};
return string(size, [&s, &before, &after]
(const mutable_buffer &buf)
{
char *p{begin(buf)};
std::for_each(begin(s), end(s), [&before, &after, &p]
(const char &c)
{
if(c == before)
{
memcpy(p, after.data(), after.size());
p += after.size();
}
else *p++ = c;
});
return std::distance(begin(buf), p);
});
}