| // Copyright (C) 2024 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "src/trace_processor/plugins/base64_functions/base64_functions.h" |
| |
| #include <cstddef> |
| #include <cstdint> |
| #include <cstdlib> |
| #include <memory> |
| #include <vector> |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/base/status.h" |
| #include "perfetto/ext/base/base64.h" |
| #include "perfetto/ext/base/utils.h" |
| #include "src/trace_processor/core/plugin/plugin.h" |
| #include "src/trace_processor/perfetto_sql/engine/perfetto_sql_connection.h" |
| #include "src/trace_processor/sqlite/bindings/sqlite_function.h" |
| #include "src/trace_processor/sqlite/bindings/sqlite_result.h" |
| #include "src/trace_processor/sqlite/bindings/sqlite_type.h" |
| #include "src/trace_processor/sqlite/bindings/sqlite_value.h" |
| #include "src/trace_processor/sqlite/sqlite_utils.h" |
| |
| namespace perfetto::trace_processor { |
| |
| namespace { |
| |
| struct Base64Decode : public sqlite::Function<Base64Decode> { |
| static constexpr char kName[] = "__intrinsic_base64_decode"; |
| static constexpr int kArgCount = 1; |
| |
| static void Step(sqlite3_context* ctx, int argc, sqlite3_value** argv) { |
| PERFETTO_DCHECK(argc == 1); |
| |
| const char* src = nullptr; |
| size_t src_size = 0; |
| switch (sqlite::value::Type(argv[0])) { |
| case sqlite::Type::kNull: |
| return sqlite::utils::ReturnNullFromFunction(ctx); |
| case sqlite::Type::kInteger: |
| case sqlite::Type::kFloat: |
| return sqlite::utils::SetError( |
| ctx, "BASE64: argument must be string or blob"); |
| case sqlite::Type::kText: |
| src = sqlite::value::Text(argv[0]); |
| src_size = static_cast<size_t>(sqlite::value::Bytes(argv[0])); |
| break; |
| case sqlite::Type::kBlob: |
| src = reinterpret_cast<const char*>(sqlite::value::Blob(argv[0])); |
| src_size = static_cast<size_t>(sqlite::value::Bytes(argv[0])); |
| break; |
| } |
| |
| size_t dst_size = base::Base64DecSize(src_size); |
| std::unique_ptr<uint8_t, base::FreeDeleter> dst( |
| reinterpret_cast<uint8_t*>(malloc(dst_size))); |
| auto res = base::Base64Decode(src, src_size, dst.get(), dst_size); |
| if (res < 0) { |
| return sqlite::utils::SetError(ctx, "BASE64: Invalid input"); |
| } |
| dst_size = static_cast<size_t>(res); |
| return sqlite::result::RawBytes(ctx, dst.release(), |
| static_cast<int>(dst_size), free); |
| } |
| }; |
| |
| } // namespace |
| |
| } // namespace perfetto::trace_processor |
| |
| namespace perfetto::trace_processor::base64_functions { |
| namespace { |
| |
| class Base64FunctionsPlugin : public Plugin<Base64FunctionsPlugin> { |
| public: |
| ~Base64FunctionsPlugin() override; |
| void RegisterFunctions(PerfettoSqlConnection*, |
| std::vector<FunctionRegistration>& out) override { |
| out.push_back(MakeFunctionRegistration<Base64Decode>(nullptr)); |
| } |
| }; |
| Base64FunctionsPlugin::~Base64FunctionsPlugin() = default; |
| |
| } // namespace |
| |
| void RegisterPlugin() { |
| static PluginRegistration reg( |
| []() -> std::unique_ptr<PluginBase> { |
| return std::make_unique<Base64FunctionsPlugin>(); |
| }, |
| Base64FunctionsPlugin::kPluginId, Base64FunctionsPlugin::kDepIds.data(), |
| Base64FunctionsPlugin::kDepIds.size()); |
| base::ignore_result(reg); |
| } |
| |
| } // namespace perfetto::trace_processor::base64_functions |