| /* |
| * Copyright (C) 2019 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/sqlite/create_function.h" |
| |
| #include "perfetto/base/status.h" |
| #include "perfetto/trace_processor/basic_types.h" |
| #include "src/trace_processor/sqlite/create_function_internal.h" |
| #include "src/trace_processor/sqlite/scoped_db.h" |
| #include "src/trace_processor/sqlite/sqlite_utils.h" |
| #include "src/trace_processor/tp_metatrace.h" |
| #include "src/trace_processor/util/status_macros.h" |
| |
| namespace perfetto { |
| namespace trace_processor { |
| |
| namespace { |
| |
| struct CreatedFunction : public SqlFunction { |
| struct Context { |
| sqlite3* db; |
| Prototype prototype; |
| SqlValue::Type return_type; |
| std::string sql; |
| sqlite3_stmt* stmt; |
| }; |
| |
| static base::Status Run(Context* ctx, |
| size_t argc, |
| sqlite3_value** argv, |
| SqlValue& out, |
| Destructors&); |
| static base::Status VerifyPostConditions(Context*); |
| static void Cleanup(Context*); |
| }; |
| |
| base::Status CreatedFunction::Run(CreatedFunction::Context* ctx, |
| size_t argc, |
| sqlite3_value** argv, |
| SqlValue& out, |
| Destructors&) { |
| if (argc != ctx->prototype.arguments.size()) { |
| return base::ErrStatus( |
| "%s: invalid number of args; expected %zu, received %zu", |
| ctx->prototype.function_name.c_str(), ctx->prototype.arguments.size(), |
| argc); |
| } |
| |
| // Type check all the arguments. |
| for (size_t i = 0; i < argc; ++i) { |
| sqlite3_value* arg = argv[i]; |
| base::Status status = |
| TypeCheckSqliteValue(arg, ctx->prototype.arguments[i].type); |
| if (!status.ok()) { |
| return base::ErrStatus("%s[arg=%s]: argument %zu %s", |
| ctx->prototype.function_name.c_str(), |
| sqlite3_value_text(arg), i, status.c_message()); |
| } |
| } |
| |
| PERFETTO_TP_TRACE( |
| metatrace::Category::FUNCTION, "CREATE_FUNCTION", |
| [ctx, argv](metatrace::Record* r) { |
| r->AddArg("Function", ctx->prototype.function_name.c_str()); |
| for (uint32_t i = 0; i < ctx->prototype.arguments.size(); ++i) { |
| std::string key = "Arg " + std::to_string(i); |
| const char* value = |
| reinterpret_cast<const char*>(sqlite3_value_text(argv[i])); |
| r->AddArg(base::StringView(key), |
| value ? base::StringView(value) : base::StringView("NULL")); |
| } |
| }); |
| |
| // Bind all the arguments to the appropriate places in the function. |
| for (size_t i = 0; i < argc; ++i) { |
| RETURN_IF_ERROR(MaybeBindArgument(ctx->stmt, ctx->prototype.function_name, |
| ctx->prototype.arguments[i], argv[i])); |
| } |
| |
| int ret = sqlite3_step(ctx->stmt); |
| RETURN_IF_ERROR( |
| SqliteRetToStatus(ctx->db, ctx->prototype.function_name, ret)); |
| if (ret == SQLITE_DONE) { |
| // No return value means we just return don't set |out|. |
| return base::OkStatus(); |
| } |
| |
| PERFETTO_DCHECK(ret == SQLITE_ROW); |
| size_t col_count = static_cast<size_t>(sqlite3_column_count(ctx->stmt)); |
| if (col_count != 1) { |
| return base::ErrStatus( |
| "%s: SQL definition should only return one column: returned %zu " |
| "columns", |
| ctx->prototype.function_name.c_str(), col_count); |
| } |
| |
| out = sqlite_utils::SqliteValueToSqlValue(sqlite3_column_value(ctx->stmt, 0)); |
| |
| // If we return a bytes type but have a null pointer, SQLite will convert this |
| // to an SQL null. However, for proto build functions, we actively want to |
| // distinguish between nulls and 0 byte strings. Therefore, change the value |
| // to an empty string. |
| if (out.type == SqlValue::kBytes && out.bytes_value == nullptr) { |
| PERFETTO_DCHECK(out.bytes_count == 0); |
| out.bytes_value = ""; |
| } |
| return base::OkStatus(); |
| } |
| |
| base::Status CreatedFunction::VerifyPostConditions(Context* ctx) { |
| int ret = sqlite3_step(ctx->stmt); |
| RETURN_IF_ERROR( |
| SqliteRetToStatus(ctx->db, ctx->prototype.function_name, ret)); |
| if (ret == SQLITE_ROW) { |
| auto expanded_sql = sqlite_utils::ExpandedSqlForStmt(ctx->stmt); |
| return base::ErrStatus( |
| "%s: multiple values were returned when executing function body. " |
| "Executed SQL was %s", |
| ctx->prototype.function_name.c_str(), expanded_sql.get()); |
| } |
| PERFETTO_DCHECK(ret == SQLITE_DONE); |
| return base::OkStatus(); |
| } |
| |
| void CreatedFunction::Cleanup(CreatedFunction::Context* ctx) { |
| sqlite3_reset(ctx->stmt); |
| sqlite3_clear_bindings(ctx->stmt); |
| } |
| |
| } // namespace |
| |
| size_t CreateFunction::NameAndArgc::Hasher::operator()( |
| const NameAndArgc& s) const noexcept { |
| base::Hasher hash; |
| hash.Update(s.name.data(), s.name.size()); |
| hash.Update(s.argc); |
| return static_cast<size_t>(hash.digest()); |
| } |
| |
| base::Status CreateFunction::Run(CreateFunction::Context* ctx, |
| size_t argc, |
| sqlite3_value** argv, |
| SqlValue&, |
| Destructors&) { |
| if (argc != 3) { |
| return base::ErrStatus( |
| "CREATE_FUNCTION: invalid number of args; expected %u, received %zu", |
| 3u, argc); |
| } |
| |
| sqlite3_value* prototype_value = argv[0]; |
| sqlite3_value* return_type_value = argv[1]; |
| sqlite3_value* sql_defn_value = argv[2]; |
| |
| // Type check all the arguments. |
| { |
| auto type_check = [prototype_value](sqlite3_value* value, |
| SqlValue::Type type, const char* desc) { |
| base::Status status = TypeCheckSqliteValue(value, type); |
| if (!status.ok()) { |
| return base::ErrStatus("CREATE_FUNCTION[prototype=%s]: %s %s", |
| sqlite3_value_text(prototype_value), desc, |
| status.c_message()); |
| } |
| return base::OkStatus(); |
| }; |
| |
| RETURN_IF_ERROR(type_check(prototype_value, SqlValue::Type::kString, |
| "function prototype (first argument)")); |
| RETURN_IF_ERROR(type_check(return_type_value, SqlValue::Type::kString, |
| "return type (second argument)")); |
| RETURN_IF_ERROR(type_check(sql_defn_value, SqlValue::Type::kString, |
| "SQL definition (third argument)")); |
| } |
| |
| // Extract the arguments from the value wrappers. |
| auto extract_string = [](sqlite3_value* value) -> base::StringView { |
| return reinterpret_cast<const char*>(sqlite3_value_text(value)); |
| }; |
| base::StringView prototype_str = extract_string(prototype_value); |
| base::StringView return_type_str = extract_string(return_type_value); |
| std::string sql_defn_str = extract_string(sql_defn_value).ToStdString(); |
| |
| // Parse all the arguments into a more friendly form. |
| Prototype prototype; |
| base::Status status = ParsePrototype(prototype_str, prototype); |
| if (!status.ok()) { |
| return base::ErrStatus("CREATE_FUNCTION[prototype=%s]: %s", |
| prototype_str.ToStdString().c_str(), |
| status.c_message()); |
| } |
| |
| // Parse the return type into a enum format. |
| auto opt_return_type = ParseType(return_type_str); |
| if (!opt_return_type) { |
| return base::ErrStatus( |
| "CREATE_FUNCTION[prototype=%s, return=%s]: unknown return type " |
| "specified", |
| prototype_str.ToStdString().c_str(), |
| return_type_str.ToStdString().c_str()); |
| } |
| SqlValue::Type return_type = *opt_return_type; |
| |
| int created_argc = static_cast<int>(prototype.arguments.size()); |
| NameAndArgc key{prototype.function_name, created_argc}; |
| auto it = ctx->state->find(key); |
| if (it != ctx->state->end()) { |
| // If the function already exists, just verify that the prototype, return |
| // type and SQL matches exactly with what we already had registered. By |
| // doing this, we can avoid the problem plaguing C++ macros where macro |
| // ordering determines which one gets run. |
| auto* created_ctx = static_cast<CreatedFunction::Context*>( |
| it->second.created_functon_context); |
| |
| if (created_ctx->prototype != prototype) { |
| return base::ErrStatus( |
| "CREATE_FUNCTION[prototype=%s]: function prototype changed", |
| prototype_str.ToStdString().c_str()); |
| } |
| |
| if (created_ctx->return_type != return_type) { |
| return base::ErrStatus( |
| "CREATE_FUNCTION[prototype=%s]: return type changed from %s to %s", |
| prototype_str.ToStdString().c_str(), |
| SqliteTypeToFriendlyString(created_ctx->return_type), |
| return_type_str.ToStdString().c_str()); |
| } |
| |
| if (created_ctx->sql != sql_defn_str) { |
| return base::ErrStatus( |
| "CREATE_FUNCTION[prototype=%s]: function SQL changed from %s to %s", |
| prototype_str.ToStdString().c_str(), created_ctx->sql.c_str(), |
| sql_defn_str.c_str()); |
| } |
| return base::OkStatus(); |
| } |
| |
| // Prepare the SQL definition as a statement using SQLite. |
| ScopedStmt stmt; |
| sqlite3_stmt* stmt_raw = nullptr; |
| int ret = sqlite3_prepare_v2(ctx->db, sql_defn_str.data(), |
| static_cast<int>(sql_defn_str.size()), &stmt_raw, |
| nullptr); |
| if (ret != SQLITE_OK) { |
| return base::ErrStatus( |
| "CREATE_FUNCTION[prototype=%s]: SQLite error when preparing " |
| "statement " |
| "%s", |
| prototype_str.ToStdString().c_str(), sqlite3_errmsg(ctx->db)); |
| } |
| stmt.reset(stmt_raw); |
| |
| std::unique_ptr<CreatedFunction::Context> created( |
| new CreatedFunction::Context{ctx->db, std::move(prototype), return_type, |
| std::move(sql_defn_str), stmt.get()}); |
| CreatedFunction::Context* created_ptr = created.get(); |
| RETURN_IF_ERROR(RegisterSqlFunction<CreatedFunction>( |
| ctx->db, key.name.c_str(), created_argc, std::move(created))); |
| ctx->state->emplace(key, PerFunctionState{std::move(stmt), created_ptr}); |
| |
| // CREATE_FUNCTION doesn't have a return value so just don't sent |out|. |
| return base::OkStatus(); |
| } |
| |
| } // namespace trace_processor |
| } // namespace perfetto |