blob: 43bb39df9a4622055e802d5c28d861112d0d8cdb [file] [log] [blame]
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you 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 "util/debug_points.h"
#include "common/logging.h"
#include "util/time.h"
namespace doris {
DebugPoints::DebugPoints() : _debug_points(std::make_shared<const DebugPointMap>()) {}
DebugPoints* DebugPoints::instance() {
static DebugPoints instance;
return &instance;
}
bool DebugPoints::is_enable(const std::string& name) {
return get_debug_point(name) != nullptr;
}
std::shared_ptr<DebugPoint> DebugPoints::get_debug_point(const std::string& name) {
if (!config::enable_debug_points) {
return nullptr;
}
auto map_ptr = std::atomic_load_explicit(&_debug_points, std::memory_order_relaxed);
auto it = map_ptr->find(name);
if (it == map_ptr->end()) {
return nullptr;
}
auto debug_point = it->second;
if ((debug_point->expire_ms > 0 && MonotonicMillis() >= debug_point->expire_ms) ||
(debug_point->execute_limit > 0 &&
debug_point->execute_num.fetch_add(1, std::memory_order_relaxed) >=
debug_point->execute_limit)) {
remove(name);
return nullptr;
}
return debug_point;
}
void DebugPoints::add(const std::string& name, std::shared_ptr<DebugPoint> debug_point) {
update([&](DebugPointMap& new_points) { new_points[name] = debug_point; });
std::ostringstream oss;
oss << "{";
for (auto [key, value] : debug_point->params) {
oss << key << " : " << value << ", ";
}
oss << "}";
LOG(INFO) << "add debug point: name=" << name << ", params=" << oss.str();
}
void DebugPoints::remove(const std::string& name) {
bool exists = false;
update([&](DebugPointMap& new_points) { exists = new_points.erase(name) > 0; });
LOG(INFO) << "remove debug point: name=" << name << ", exists=" << exists;
}
void DebugPoints::update(std::function<void(DebugPointMap&)>&& handler) {
auto old_points = std::atomic_load_explicit(&_debug_points, std::memory_order_relaxed);
while (true) {
auto new_points = std::make_shared<DebugPointMap>(*old_points);
handler(*new_points);
if (std::atomic_compare_exchange_strong_explicit(
&_debug_points, &old_points,
std::static_pointer_cast<const DebugPointMap>(new_points),
std::memory_order_relaxed, std::memory_order_relaxed)) {
break;
}
}
}
void DebugPoints::clear() {
std::atomic_store_explicit(&_debug_points, std::make_shared<const DebugPointMap>(),
std::memory_order_relaxed);
LOG(INFO) << "clear debug points";
}
} // namespace doris