/* Copyright (c) 2017 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Gabriel Ebner */ #include "util/log_tree.h" #include "util/task_builder.h" #include #include namespace lean { void log_tree::notify_core(std::vector const & events) { if (events.empty()) return; for (auto & l : m_listeners) l(events); } void log_tree::add_listener(log_tree::listener const & l) { unique_lock lock(m_mutex); m_listeners.push_back(l); } log_tree::log_tree() { auto cell = new node_cell; cell->m_tree = this; m_root = node(cell); } log_tree::~log_tree() { std::vector evs; m_root.detach_core(evs); m_root = node(); } void log_tree::print_to(std::ostream & out) const { get_root().print_to(out, 0); } void log_tree::print() const { print_to(std::cerr); } void log_tree::node::for_each(std::function const & fn) const { // NOLINT if (fn(*this)) { get_children().for_each([&] (name const &, log_tree::node const & c) { c.for_each(fn); }); } }; void log_tree::for_each(std::function const & fn) const { // NOLINT m_root.for_each(fn); }; void log_tree::node::detach_core(std::vector & events) const { if (m_ptr->m_detached) return; m_ptr->m_detached = true; m_ptr->m_children.for_each([&] (name const &, node const & c) { c.detach_core(events); }); for (auto & e : m_ptr->m_entries) events.push_back({ event::EntryRemoved, m_ptr, e }); if (m_ptr->m_producer) events.push_back({event::Finished, m_ptr, {}}); } void log_tree::node::notify(std::vector const & events, unique_lock & l) const { if (!m_ptr->m_detached) { l.unlock(); m_ptr->m_tree->notify_core(events); } } log_tree::node log_tree::node::clone_core() const { auto copy = node(new node_cell); copy.m_ptr->m_children = m_ptr->m_children; m_ptr->m_children.clear(); copy.m_ptr->m_tree = m_ptr->m_tree; copy.m_ptr->m_detached = m_ptr->m_detached; copy.m_ptr->m_entries = m_ptr->m_entries; m_ptr->m_detached = true; return copy; } void log_tree::node::clear_entries() const { auto l = lock(); std::vector events; for (auto & e : m_ptr->m_entries) { events.push_back({event::EntryRemoved, *this, std::move(e)}); } m_ptr->m_entries.clear(); notify(events, l); } void log_tree::node::add(log_entry const & entry) const { auto l = lock(); m_ptr->m_entries.push_back(entry); notify({{event::EntryAdded, *this, entry}}, l); } log_tree::node log_tree::node::mk_child(name n, std::string const & description, location const & loc, bool overwrite) { auto l = lock(); std::vector events; if (!overwrite && (n.is_anonymous() || m_ptr->m_used_names.contains(n))) { for (unsigned i = 0;; i++) { name n_(n, i); if (!m_ptr->m_used_names.contains(n_)) { n = n_; break; } } } m_ptr->m_used_names.insert(n); node child; if (auto existing = m_ptr->m_children.find(n)) { child = existing->clone_core(); existing->detach_core(events); } else { child = node(new node_cell); child.m_ptr->m_tree = m_ptr->m_tree; child.m_ptr->m_detached = m_ptr->m_detached; } child.m_ptr->m_description = description; child.m_ptr->m_location = loc; m_ptr->m_children.insert(n, child); notify(events, l); return child; } void log_tree::node::set_producer(gtask const & prod) { auto l = lock(); m_ptr->m_producer = prod; notify({{event::ProducerSet, *this, {}}}, l); } void log_tree::node::finish() const { auto l = lock(); std::vector events; m_ptr->m_producer.reset(); buffer> to_delete; m_ptr->m_children.for_each([&] (name const & n, node const & c) { if (!m_ptr->m_used_names.contains(n)) to_delete.emplace_back(n, c); }); for (auto & c : to_delete) { m_ptr->m_children.erase(c.first); c.second.detach_core(events); } events.push_back({event::Finished, *this, {}}); notify(events, l); } std::vector log_tree::node::get_entries() const { auto l = lock(); return m_ptr->m_entries; } name_map log_tree::node::get_children() const { auto l = lock(); return m_ptr->m_children; } void log_tree::node::reuse(name const & n) const { auto l = lock(); m_ptr->m_used_names.insert(n); } name_set log_tree::node::get_used_names() const { auto l = lock(); return m_ptr->m_used_names; } gtask log_tree::node::get_producer() const { auto l = lock(); return m_ptr->m_producer; } void log_tree::node::print() const { print_to(std::cerr, 0); } void log_tree::node::print_to(std::ostream & out, unsigned indent) const { indent += 2; auto l = lock(); auto begin = m_ptr->m_location.m_range.m_begin, end = m_ptr->m_location.m_range.m_end; out << m_ptr->m_location.m_file_name << ": " << begin.first << ":" << begin.second << " -- " << end.first << ":" << end.second << ": " << m_ptr->m_description << " (" << m_ptr->m_entries.size() << " entries, producer = " << std::hex << reinterpret_cast(m_ptr->m_producer.get()) << std::dec << ")" << std::endl; if (auto prod = m_ptr->m_producer) { if (auto ex = prod->peek_exception()) { for (unsigned i = 0; i < indent; i++) out << ' '; out << "producer threw exception: "; try { std::rethrow_exception(ex); } catch (std::exception & ex) { out << ex.what(); } catch (cancellation_exception) { out << ""; } catch (...) { out << ""; } out << "\n"; } } auto children = m_ptr->m_children; auto used_names = m_ptr->m_used_names; l.unlock(); children.for_each([&] (name const & n, log_tree::node const & c) { for (unsigned i = 0; i < indent; i++) out << ' '; out << n; if (!used_names.contains(n)) out << " (unused)"; out << ": "; c.print_to(out, indent); }); } static void gather_producers(buffer & deps, log_tree::node const & n) { if (auto prod = n.get_producer()) deps.push_back(prod); n.get_children().for_each([&] (name const &, log_tree::node const & c) { gather_producers(deps, c); }); } gtask log_tree::node::wait_for_finish() const { auto t = *this; return mk_dependency_task([t] (buffer & deps) { gather_producers(deps, t); }); } bool log_tree::node::has_entry_now(std::function const & fn) const { // NOLINT for (auto & e : get_entries()) { if (fn(e)) return true; } bool res = false; get_children().for_each([&] (name const &, node const & c) { res = res || c.has_entry_now(fn); }); return res; } task log_tree::node::has_entry(std::function const & fn) const { // NOLINT auto t = *this; return task_builder([t, fn] { return t.has_entry_now(fn); }).depends_on(wait_for_finish()).build(); } LEAN_THREAD_PTR(log_tree::node, g_log_tree); scope_log_tree_core::scope_log_tree_core(log_tree::node * lt) : flet(g_log_tree, lt) {} log_tree::node & logtree() { if (g_log_tree) { return *g_log_tree; } else { throw exception("no log tree in scope"); } } scope_log_tree::scope_log_tree(log_tree::node const & lt) : m_node(lt), m_scope(&m_node) { m_node.clear_entries(); } scope_log_tree::~scope_log_tree() { logtree().finish(); } scope_log_tree::scope_log_tree(std::string const & desc, pos_range const & pos) : scope_log_tree(logtree().mk_child({}, desc, {logtree().get_location().m_file_name, pos})) {} scope_log_tree::scope_log_tree(std::string const & desc) : scope_log_tree(logtree().mk_child({}, desc, logtree().get_location())) {} scope_log_tree::scope_log_tree() : scope_log_tree(std::string()) {} }