blob: 7a9731585f1b683b9ebfe5c1422ca020cc3ca57f [file]
// Copyright 2023 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include "src/tint/utils/text/base64.h"
#include <array>
#include <string>
#include "src/tint/utils/result.h"
namespace tint {
Vector<std::byte, 0> DecodeBase64FromComments(std::string_view wgsl) {
Vector<std::byte, 0> out;
size_t block_nesting = 0;
bool line_comment = false;
for (size_t i = 0, n = wgsl.length(); i < n; i++) {
char curr = wgsl[i];
if (curr == '\n') {
line_comment = false;
continue;
}
char next = (i + 1) < n ? wgsl[i + 1] : 0;
if (curr == '/' && next == '*') {
block_nesting++;
i++; // skip '*'
continue;
}
if (block_nesting > 0 && curr == '*' && next == '/') {
block_nesting--;
i++; // skip '/'
continue;
}
if (block_nesting == 0 && curr == '/' && next == '/') {
line_comment = true;
i++; // skip '/'
continue;
}
if (block_nesting > 0 || line_comment) {
if (auto v = DecodeBase64(curr)) {
out.Push(std::byte{*v});
}
}
}
return out;
}
Result<std::string> EncodeBase64(std::span<const std::byte> data) {
constexpr std::array kMap{
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/',
};
std::string out;
out.reserve(data.size());
for (std::byte b : data) {
const auto val = static_cast<uint8_t>(b);
if (val >= kMap.size()) {
return Failure{"byte value " + std::to_string(val) + " is out of range [0, 63]"};
}
out.push_back(kMap[val]);
}
return out;
}
} // namespace tint