/* DWARF index storage
Copyright (C) 2022-2025 Free Software Foundation, Inc.
This file is part of GDB.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see . */
#include "dwarf2/cooked-index-worker.h"
#include "dwarf2/cooked-index.h"
#include "gdbsupport/thread-pool.h"
#include "run-on-main-thread.h"
#include "event-top.h"
#include "exceptions.h"
/* See cooked-index-worker.h. */
cooked_index_worker_result::cooked_index_worker_result ()
: m_shard (new cooked_index_shard)
{
}
/* See cooked-index-worker.h. */
cutu_reader *
cooked_index_worker_result::get_reader (dwarf2_per_cu *per_cu)
{
auto it = m_reader_hash.find (*per_cu);
return it != m_reader_hash.end () ? it->get () : nullptr;
}
/* See cooked-index-worker.h. */
cutu_reader *
cooked_index_worker_result::preserve (cutu_reader_up reader)
{
m_abbrev_table_cache.add (reader->release_abbrev_table ());
auto [it, inserted] = m_reader_hash.insert (std::move (reader));
gdb_assert (inserted);
return it->get();
}
/* See cooked-index-worker.h. */
std::uint64_t
cooked_index_worker_result::cutu_reader_hash::operator()
(const cutu_reader_up &reader) const noexcept
{
return (*this) (*reader->cu ()->per_cu);
}
/* See cooked-index-worker.h. */
std::uint64_t
cooked_index_worker_result::cutu_reader_hash::operator() (const dwarf2_per_cu &per_cu)
const noexcept
{
return per_cu.index;
}
/* See cooked-index-worker.h. */
bool
cooked_index_worker_result::cutu_reader_eq::operator() (const cutu_reader_up &a,
const cutu_reader_up &b) const noexcept
{
return (*this) (*a->cu ()->per_cu, b);
}
/* See cooked-index-worker.h. */
bool cooked_index_worker_result::cutu_reader_eq::operator()
(const dwarf2_per_cu &per_cu, const cutu_reader_up &reader) const noexcept
{
return per_cu.index == reader->cu ()->per_cu->index;
}
/* See cooked-index-worker.h. */
void
cooked_index_worker_result::emit_complaints_and_exceptions
(gdb::unordered_set &seen_exceptions)
{
gdb_assert (is_main_thread ());
re_emit_complaints (m_complaints);
/* Only show a given exception a single time. */
for (auto &one_exc : m_exceptions)
if (seen_exceptions.insert (one_exc).second)
exception_print (gdb_stderr, one_exc);
}
/* See cooked-index-worker.h. */
void
cooked_index_worker::start ()
{
gdb::thread_pool::g_thread_pool->post_task ([this] ()
{
try
{
do_reading ();
}
catch (const gdb_exception &exc)
{
m_failed = exc;
set (cooked_state::CACHE_DONE);
}
bfd_thread_cleanup ();
});
}
/* See cooked-index-worker.h. */
bool
cooked_index_worker::wait (cooked_state desired_state, bool allow_quit)
{
bool done;
#if CXX_STD_THREAD
{
std::unique_lock lock (m_mutex);
/* This may be called from a non-main thread -- this functionality
is needed for the index cache -- but in this case we require
that the desired state already have been attained. */
gdb_assert (is_main_thread () || desired_state <= m_state);
while (desired_state > m_state)
{
if (allow_quit)
{
std::chrono::milliseconds duration { 15 };
if (m_cond.wait_for (lock, duration) == std::cv_status::timeout)
QUIT;
}
else
m_cond.wait (lock);
}
done = m_state == cooked_state::CACHE_DONE;
}
#else
/* Without threads, all the work is done immediately on the main
thread, and there is never anything to wait for. */
done = desired_state == cooked_state::CACHE_DONE;
#endif /* CXX_STD_THREAD */
/* Only the main thread is allowed to report complaints and the
like. */
if (!is_main_thread ())
return false;
if (m_reported)
return done;
m_reported = true;
/* Emit warnings first, maybe they were emitted before an exception
(if any) was thrown. */
m_warnings.emit ();
if (m_failed.has_value ())
{
/* do_reading failed -- report it. */
exception_print (gdb_stderr, *m_failed);
m_failed.reset ();
return done;
}
/* Only show a given exception a single time. */
gdb::unordered_set seen_exceptions;
for (auto &one_result : m_results)
one_result.emit_complaints_and_exceptions (seen_exceptions);
print_stats ();
struct objfile *objfile = m_per_objfile->objfile;
dwarf2_per_bfd *per_bfd = m_per_objfile->per_bfd;
cooked_index *table
= (gdb::checked_static_cast
(per_bfd->index_table.get ()));
auto_obstack temp_storage;
enum language lang = language_unknown;
const char *main_name = table->get_main_name (&temp_storage, &lang);
if (main_name != nullptr)
set_objfile_main_name (objfile, main_name, lang);
/* dwarf_read_debug_printf ("Done building psymtabs of %s", */
/* objfile_name (objfile)); */
return done;
}
/* See cooked-index-worker.h. */
void
cooked_index_worker::set (cooked_state desired_state)
{
gdb_assert (desired_state != cooked_state::INITIAL);
#if CXX_STD_THREAD
std::lock_guard guard (m_mutex);
gdb_assert (desired_state > m_state);
m_state = desired_state;
m_cond.notify_one ();
#else
/* Without threads, all the work is done immediately on the main
thread, and there is never anything to do. */
#endif /* CXX_STD_THREAD */
}
/* See cooked-index-worker.h. */
void
cooked_index_worker::write_to_cache (const cooked_index *idx)
{
if (idx != nullptr)
{
/* Writing to the index cache may cause a warning to be emitted.
See PR symtab/30837. This arranges to capture all such
warnings. This is safe because we know the deferred_warnings
object isn't in use by any other thread at this point. */
scoped_restore_warning_hook defer (&m_warnings);
m_cache_store.store ();
}
}
/* See cooked-index-worker.h. */
void
cooked_index_worker::done_reading ()
{
for (auto &one_result : m_results)
m_all_parents_map.add_map (*one_result.get_parent_map ());
dwarf2_per_bfd *per_bfd = m_per_objfile->per_bfd;
cooked_index *table
= (gdb::checked_static_cast
(per_bfd->index_table.get ()));
table->set_contents ();
}