2020-04-26 16:49:19 -04:00
|
|
|
/*
|
2022-02-25 13:35:51 -05:00
|
|
|
* Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org>
|
2020-04-26 16:49:19 -04:00
|
|
|
*
|
2021-04-22 04:24:48 -04:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-04-26 16:49:19 -04:00
|
|
|
*/
|
|
|
|
|
2020-08-10 17:50:38 -04:00
|
|
|
#include <AK/Base64.h>
|
2020-04-26 16:49:19 -04:00
|
|
|
#include <AK/Types.h>
|
|
|
|
#include <AK/Vector.h>
|
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
#include <simdutf.h>
|
2024-03-24 08:55:53 -04:00
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
namespace AK {
|
2020-10-13 10:48:48 -04:00
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
static ErrorOr<ByteBuffer> decode_base64_impl(StringView input, simdutf::base64_options options)
|
2020-04-26 16:49:19 -04:00
|
|
|
{
|
2024-03-20 12:41:41 -04:00
|
|
|
ByteBuffer output;
|
2024-07-15 15:25:08 -04:00
|
|
|
TRY(output.try_resize(simdutf::maximal_binary_length_from_base64(input.characters_without_null_termination(), input.length())));
|
2020-04-26 16:49:19 -04:00
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
auto result = simdutf::base64_to_binary(
|
|
|
|
input.characters_without_null_termination(),
|
|
|
|
input.length(),
|
|
|
|
reinterpret_cast<char*>(output.data()),
|
|
|
|
options);
|
2024-03-20 12:41:41 -04:00
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
if (result.error != simdutf::SUCCESS)
|
|
|
|
return Error::from_string_literal("Invalid base64-encoded data");
|
2020-04-26 16:49:19 -04:00
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
output.resize(result.count);
|
2024-03-20 12:41:41 -04:00
|
|
|
return output;
|
2020-04-26 16:49:19 -04:00
|
|
|
}
|
|
|
|
|
2024-07-15 15:25:08 -04:00
|
|
|
static ErrorOr<String> encode_base64_impl(StringView input, simdutf::base64_options options)
|
2020-06-12 22:09:31 -04:00
|
|
|
{
|
2024-03-20 11:18:27 -04:00
|
|
|
Vector<u8> output;
|
2024-07-15 15:25:08 -04:00
|
|
|
|
|
|
|
// simdutf does not append padding to base64url encodings. We use the default encoding option here to allocate room
|
|
|
|
// for the padding characters that we will later append ourselves if necessary.
|
|
|
|
TRY(output.try_resize(simdutf::base64_length_from_binary(input.length(), simdutf::base64_default)));
|
|
|
|
|
|
|
|
auto size_written = simdutf::binary_to_base64(
|
|
|
|
input.characters_without_null_termination(),
|
|
|
|
input.length(),
|
|
|
|
reinterpret_cast<char*>(output.data()),
|
|
|
|
options);
|
|
|
|
|
|
|
|
if (options == simdutf::base64_url) {
|
|
|
|
for (size_t i = size_written; i < output.size(); ++i)
|
|
|
|
output[i] = '=';
|
2020-06-12 22:09:31 -04:00
|
|
|
}
|
|
|
|
|
2024-03-20 11:18:27 -04:00
|
|
|
return String::from_utf8_without_validation(output);
|
2020-06-12 22:09:31 -04:00
|
|
|
}
|
|
|
|
|
2024-03-20 08:21:59 -04:00
|
|
|
ErrorOr<ByteBuffer> decode_base64(StringView input)
|
|
|
|
{
|
2024-07-15 15:25:08 -04:00
|
|
|
return decode_base64_impl(input, simdutf::base64_default);
|
2024-03-20 08:21:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ErrorOr<ByteBuffer> decode_base64url(StringView input)
|
|
|
|
{
|
2024-07-15 15:25:08 -04:00
|
|
|
return decode_base64_impl(input, simdutf::base64_url);
|
2024-03-20 08:21:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ErrorOr<String> encode_base64(ReadonlyBytes input)
|
|
|
|
{
|
2024-07-15 15:25:08 -04:00
|
|
|
return encode_base64_impl(input, simdutf::base64_default);
|
2024-03-20 08:21:59 -04:00
|
|
|
}
|
2024-07-15 15:25:08 -04:00
|
|
|
|
2024-03-20 08:21:59 -04:00
|
|
|
ErrorOr<String> encode_base64url(ReadonlyBytes input)
|
|
|
|
{
|
2024-07-15 15:25:08 -04:00
|
|
|
return encode_base64_impl(input, simdutf::base64_url);
|
2024-03-20 08:21:59 -04:00
|
|
|
}
|
|
|
|
|
2020-04-26 16:49:19 -04:00
|
|
|
}
|