gdb/linux-thread-db.c - gdb
Global variables defined
Data types defined
Functions defined
Source code
- #include "defs.h"
- #include <dlfcn.h>
- #include "gdb_proc_service.h"
- #include "nat/gdb_thread_db.h"
- #include "gdb_vecs.h"
- #include "bfd.h"
- #include "command.h"
- #include "gdbcmd.h"
- #include "gdbthread.h"
- #include "inferior.h"
- #include "infrun.h"
- #include "symfile.h"
- #include "objfiles.h"
- #include "target.h"
- #include "regcache.h"
- #include "solib.h"
- #include "solib-svr4.h"
- #include "gdbcore.h"
- #include "observer.h"
- #include "linux-nat.h"
- #include "nat/linux-procfs.h"
- #include "nat/linux-ptrace.h"
- #include "nat/linux-osdata.h"
- #include "auto-load.h"
- #include "cli/cli-utils.h"
- #include <signal.h>
- #include <ctype.h>
- static char *libthread_db_search_path;
- static int auto_load_thread_db = 1;
- static int
- thread_db_use_events (void)
- {
-
- return !linux_supports_traceclone ();
- }
- static void
- show_auto_load_thread_db (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file, _("Auto-loading of inferior specific libthread_db "
- "is %s.\n"),
- value);
- }
- static void
- set_libthread_db_search_path (char *ignored, int from_tty,
- struct cmd_list_element *c)
- {
- if (*libthread_db_search_path == '\0')
- {
- xfree (libthread_db_search_path);
- libthread_db_search_path = xstrdup (LIBTHREAD_DB_SEARCH_PATH);
- }
- }
- static unsigned int libthread_db_debug;
- static void
- show_libthread_db_debug (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file, _("libthread-db debugging is %s.\n"), value);
- }
- static struct target_ops thread_db_ops;
- static int thread_signals;
- static sigset_t thread_stop_set;
- static sigset_t thread_print_set;
- struct thread_db_info
- {
- struct thread_db_info *next;
-
- int pid;
-
- void *handle;
-
- char *filename;
-
- struct ps_prochandle proc_handle;
-
- td_thragent_t *thread_agent;
-
- int need_stale_parent_threads_check;
-
- CORE_ADDR td_create_bp_addr;
-
- CORE_ADDR td_death_bp_addr;
-
- td_err_e (*td_init_p) (void);
- td_err_e (*td_ta_new_p) (struct ps_prochandle * ps,
- td_thragent_t **ta);
- td_err_e (*td_ta_map_id2thr_p) (const td_thragent_t *ta, thread_t pt,
- td_thrhandle_t *__th);
- td_err_e (*td_ta_map_lwp2thr_p) (const td_thragent_t *ta,
- lwpid_t lwpid, td_thrhandle_t *th);
- td_err_e (*td_ta_thr_iter_p) (const td_thragent_t *ta,
- td_thr_iter_f *callback, void *cbdata_p,
- td_thr_state_e state, int ti_pri,
- sigset_t *ti_sigmask_p,
- unsigned int ti_user_flags);
- td_err_e (*td_ta_event_addr_p) (const td_thragent_t *ta,
- td_event_e event, td_notify_t *ptr);
- td_err_e (*td_ta_set_event_p) (const td_thragent_t *ta,
- td_thr_events_t *event);
- td_err_e (*td_ta_clear_event_p) (const td_thragent_t *ta,
- td_thr_events_t *event);
- td_err_e (*td_ta_event_getmsg_p) (const td_thragent_t *ta,
- td_event_msg_t *msg);
- td_err_e (*td_thr_validate_p) (const td_thrhandle_t *th);
- td_err_e (*td_thr_get_info_p) (const td_thrhandle_t *th,
- td_thrinfo_t *infop);
- td_err_e (*td_thr_event_enable_p) (const td_thrhandle_t *th,
- int event);
- td_err_e (*td_thr_tls_get_addr_p) (const td_thrhandle_t *th,
- psaddr_t map_address,
- size_t offset, psaddr_t *address);
- td_err_e (*td_thr_tlsbase_p) (const td_thrhandle_t *th,
- unsigned long int modid,
- psaddr_t *base);
- };
- struct thread_db_info *thread_db_list;
- static void thread_db_find_new_threads_1 (ptid_t ptid);
- static void thread_db_find_new_threads_2 (ptid_t ptid, int until_no_new);
- static struct thread_db_info *
- add_thread_db_info (void *handle)
- {
- struct thread_db_info *info;
- info = xcalloc (1, sizeof (*info));
- info->pid = ptid_get_pid (inferior_ptid);
- info->handle = handle;
-
- if (target_has_execution)
- info->need_stale_parent_threads_check = 1;
- info->next = thread_db_list;
- thread_db_list = info;
- return info;
- }
- static struct thread_db_info *
- get_thread_db_info (int pid)
- {
- struct thread_db_info *info;
- for (info = thread_db_list; info; info = info->next)
- if (pid == info->pid)
- return info;
- return NULL;
- }
- static void
- delete_thread_db_info (int pid)
- {
- struct thread_db_info *info, *info_prev;
- info_prev = NULL;
- for (info = thread_db_list; info; info_prev = info, info = info->next)
- if (pid == info->pid)
- break;
- if (info == NULL)
- return;
- if (info->handle != NULL)
- dlclose (info->handle);
- xfree (info->filename);
- if (info_prev)
- info_prev->next = info->next;
- else
- thread_db_list = info->next;
- xfree (info);
- }
- static int attach_thread (ptid_t ptid, const td_thrhandle_t *th_p,
- const td_thrinfo_t *ti_p);
- static void detach_thread (ptid_t ptid);
- struct private_thread_info
- {
-
- unsigned int dying:1;
-
- td_thrhandle_t th;
- thread_t tid;
- };
- static char *
- thread_db_err_str (td_err_e err)
- {
- static char buf[64];
- switch (err)
- {
- case TD_OK:
- return "generic 'call succeeded'";
- case TD_ERR:
- return "generic error";
- case TD_NOTHR:
- return "no thread to satisfy query";
- case TD_NOSV:
- return "no sync handle to satisfy query";
- case TD_NOLWP:
- return "no LWP to satisfy query";
- case TD_BADPH:
- return "invalid process handle";
- case TD_BADTH:
- return "invalid thread handle";
- case TD_BADSH:
- return "invalid synchronization handle";
- case TD_BADTA:
- return "invalid thread agent";
- case TD_BADKEY:
- return "invalid key";
- case TD_NOMSG:
- return "no event message for getmsg";
- case TD_NOFPREGS:
- return "FPU register set not available";
- case TD_NOLIBTHREAD:
- return "application not linked with libthread";
- case TD_NOEVENT:
- return "requested event is not supported";
- case TD_NOCAPAB:
- return "capability not available";
- case TD_DBERR:
- return "debugger service failed";
- case TD_NOAPLIC:
- return "operation not applicable to";
- case TD_NOTSD:
- return "no thread-specific data for this thread";
- case TD_MALLOC:
- return "malloc failed";
- case TD_PARTIALREG:
- return "only part of register set was written/read";
- case TD_NOXREGS:
- return "X register set not available for this thread";
- #ifdef THREAD_DB_HAS_TD_NOTALLOC
- case TD_NOTALLOC:
- return "thread has not yet allocated TLS for given module";
- #endif
- #ifdef THREAD_DB_HAS_TD_VERSION
- case TD_VERSION:
- return "versions of libpthread and libthread_db do not match";
- #endif
- #ifdef THREAD_DB_HAS_TD_NOTLS
- case TD_NOTLS:
- return "there is no TLS segment in the given module";
- #endif
- default:
- snprintf (buf, sizeof (buf), "unknown thread_db error '%d'", err);
- return buf;
- }
- }
- static int
- have_threads_callback (struct thread_info *thread, void *args)
- {
- int pid = * (int *) args;
- if (ptid_get_pid (thread->ptid) != pid)
- return 0;
- return thread->private != NULL;
- }
- static int
- have_threads (ptid_t ptid)
- {
- int pid = ptid_get_pid (ptid);
- return iterate_over_threads (have_threads_callback, &pid) != NULL;
- }
- struct thread_get_info_inout
- {
- struct thread_info *thread_info;
- struct thread_db_info *thread_db_info;
- };
- static int
- thread_get_info_callback (const td_thrhandle_t *thp, void *argp)
- {
- td_thrinfo_t ti;
- td_err_e err;
- ptid_t thread_ptid;
- struct thread_get_info_inout *inout;
- struct thread_db_info *info;
- inout = argp;
- info = inout->thread_db_info;
- err = info->td_thr_get_info_p (thp, &ti);
- if (err != TD_OK)
- error (_("thread_get_info_callback: cannot get thread info: %s"),
- thread_db_err_str (err));
-
- thread_ptid = ptid_build (info->pid, ti.ti_lid, 0);
- inout->thread_info = find_thread_ptid (thread_ptid);
- if (inout->thread_info == NULL)
- {
-
- if (!have_threads (thread_ptid))
- thread_db_find_new_threads_1 (thread_ptid);
- else
- attach_thread (thread_ptid, thp, &ti);
- inout->thread_info = find_thread_ptid (thread_ptid);
- gdb_assert (inout->thread_info != NULL);
- }
- return 0;
- }
- static void
- thread_from_lwp (ptid_t ptid)
- {
- td_thrhandle_t th;
- td_err_e err;
- struct thread_db_info *info;
- struct thread_get_info_inout io = {0};
-
- th.th_unique = 0;
-
- gdb_assert (ptid_get_lwp (ptid) != 0);
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- info->proc_handle.ptid = ptid;
- err = info->td_ta_map_lwp2thr_p (info->thread_agent, ptid_get_lwp (ptid),
- &th);
- if (err != TD_OK)
- error (_("Cannot find user-level thread for LWP %ld: %s"),
- ptid_get_lwp (ptid), thread_db_err_str (err));
-
- io.thread_db_info = info;
- io.thread_info = NULL;
- thread_get_info_callback (&th, &io);
- }
- int
- thread_db_attach_lwp (ptid_t ptid)
- {
- td_thrhandle_t th;
- td_thrinfo_t ti;
- td_err_e err;
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (ptid));
- if (info == NULL)
- return 0;
-
- gdb_assert (ptid_get_lwp (ptid) != 0);
-
- info->proc_handle.ptid = ptid;
-
- if (!have_threads (ptid))
- thread_db_find_new_threads_1 (ptid);
- err = info->td_ta_map_lwp2thr_p (info->thread_agent, ptid_get_lwp (ptid),
- &th);
- if (err != TD_OK)
-
- return 0;
- err = info->td_thr_get_info_p (&th, &ti);
- if (err != TD_OK)
- {
- warning (_("Cannot get thread info: %s"), thread_db_err_str (err));
- return 0;
- }
- attach_thread (ptid, &th, &ti);
- return 1;
- }
- static void *
- verbose_dlsym (void *handle, const char *name)
- {
- void *sym = dlsym (handle, name);
- if (sym == NULL)
- warning (_("Symbol \"%s\" not found in libthread_db: %s"),
- name, dlerror ());
- return sym;
- }
- static td_err_e
- enable_thread_event (int event, CORE_ADDR *bp)
- {
- td_notify_t notify;
- td_err_e err;
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (inferior_ptid));
-
- info->proc_handle.ptid = inferior_ptid;
-
- err = info->td_ta_event_addr_p (info->thread_agent, event, ¬ify);
- if (err != TD_OK)
- return err;
-
- gdb_assert (exec_bfd);
- (*bp) = (gdbarch_convert_from_func_ptr_addr
- (target_gdbarch (),
-
- (bfd_get_sign_extend_vma (exec_bfd) > 0
- ? (CORE_ADDR) (intptr_t) notify.u.bptaddr
- : (CORE_ADDR) (uintptr_t) notify.u.bptaddr),
- ¤t_target));
- create_thread_event_breakpoint (target_gdbarch (), *bp);
- return TD_OK;
- }
- static int
- inferior_has_bug (const char *ver_symbol, int ver_major_min, int ver_minor_min)
- {
- struct bound_minimal_symbol version_msym;
- CORE_ADDR version_addr;
- char *version;
- int err, got, retval = 0;
- version_msym = lookup_minimal_symbol (ver_symbol, NULL, NULL);
- if (version_msym.minsym == NULL)
- return 0;
- version_addr = BMSYMBOL_VALUE_ADDRESS (version_msym);
- got = target_read_string (version_addr, &version, 32, &err);
- if (err == 0 && memchr (version, 0, got) == &version[got -1])
- {
- int major, minor;
- retval = (sscanf (version, "%d.%d", &major, &minor) == 2
- && (major < ver_major_min
- || (major == ver_major_min && minor < ver_minor_min)));
- }
- xfree (version);
- return retval;
- }
- static void
- enable_thread_event_reporting (void)
- {
- td_thr_events_t events;
- td_err_e err;
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (inferior_ptid));
-
- if (info->td_ta_event_addr_p == NULL
- || info->td_ta_set_event_p == NULL
- || info->td_ta_event_getmsg_p == NULL
- || info->td_thr_event_enable_p == NULL)
- return;
-
- td_event_emptyset (&events);
- td_event_addset (&events, TD_CREATE);
-
- if (!inferior_has_bug ("__linuxthreads_version", 2, 2))
- td_event_addset (&events, TD_DEATH);
- err = info->td_ta_set_event_p (info->thread_agent, &events);
- if (err != TD_OK)
- {
- warning (_("Unable to set global thread event mask: %s"),
- thread_db_err_str (err));
- return;
- }
-
- remove_thread_event_breakpoints ();
- info->td_create_bp_addr = 0;
- info->td_death_bp_addr = 0;
-
- err = enable_thread_event (TD_CREATE, &info->td_create_bp_addr);
- if (err != TD_OK)
- {
- warning (_("Unable to get location for thread creation breakpoint: %s"),
- thread_db_err_str (err));
- return;
- }
-
- err = enable_thread_event (TD_DEATH, &info->td_death_bp_addr);
- if (err != TD_OK)
- {
- warning (_("Unable to get location for thread death breakpoint: %s"),
- thread_db_err_str (err));
- return;
- }
- }
- static int
- thread_db_find_new_threads_silently (ptid_t ptid)
- {
- volatile struct gdb_exception except;
- TRY_CATCH (except, RETURN_MASK_ERROR)
- {
- thread_db_find_new_threads_2 (ptid, 1);
- }
- if (except.reason < 0)
- {
- if (libthread_db_debug)
- exception_fprintf (gdb_stdlog, except,
- "Warning: thread_db_find_new_threads_silently: ");
-
- if (!target_has_execution || !inferior_has_bug ("nptl_version", 2, 7))
- {
- exception_fprintf (gdb_stderr, except,
- _("Warning: couldn't activate thread debugging "
- "using libthread_db: "));
- return 1;
- }
- }
- return 0;
- }
- static const char *
- dladdr_to_soname (const void *addr)
- {
- Dl_info info;
- if (dladdr (addr, &info) != 0)
- return info.dli_fname;
- return NULL;
- }
- static int
- try_thread_db_load_1 (struct thread_db_info *info)
- {
- td_err_e err;
-
- info->td_init_p = verbose_dlsym (info->handle, "td_init");
- if (info->td_init_p == NULL)
- return 0;
- err = info->td_init_p ();
- if (err != TD_OK)
- {
- warning (_("Cannot initialize libthread_db: %s"),
- thread_db_err_str (err));
- return 0;
- }
- info->td_ta_new_p = verbose_dlsym (info->handle, "td_ta_new");
- if (info->td_ta_new_p == NULL)
- return 0;
-
- info->proc_handle.ptid = inferior_ptid;
-
- err = info->td_ta_new_p (&info->proc_handle, &info->thread_agent);
- if (err != TD_OK)
- {
- if (libthread_db_debug)
- fprintf_unfiltered (gdb_stdlog, _("td_ta_new failed: %s\n"),
- thread_db_err_str (err));
- else
- switch (err)
- {
- case TD_NOLIBTHREAD:
- #ifdef THREAD_DB_HAS_TD_VERSION
- case TD_VERSION:
- #endif
-
- break;
- default:
- warning (_("td_ta_new failed: %s"), thread_db_err_str (err));
- }
- return 0;
- }
- info->td_ta_map_id2thr_p = verbose_dlsym (info->handle, "td_ta_map_id2thr");
- if (info->td_ta_map_id2thr_p == NULL)
- return 0;
- info->td_ta_map_lwp2thr_p = verbose_dlsym (info->handle,
- "td_ta_map_lwp2thr");
- if (info->td_ta_map_lwp2thr_p == NULL)
- return 0;
- info->td_ta_thr_iter_p = verbose_dlsym (info->handle, "td_ta_thr_iter");
- if (info->td_ta_thr_iter_p == NULL)
- return 0;
- info->td_thr_validate_p = verbose_dlsym (info->handle, "td_thr_validate");
- if (info->td_thr_validate_p == NULL)
- return 0;
- info->td_thr_get_info_p = verbose_dlsym (info->handle, "td_thr_get_info");
- if (info->td_thr_get_info_p == NULL)
- return 0;
-
- info->td_ta_event_addr_p = dlsym (info->handle, "td_ta_event_addr");
- info->td_ta_set_event_p = dlsym (info->handle, "td_ta_set_event");
- info->td_ta_clear_event_p = dlsym (info->handle, "td_ta_clear_event");
- info->td_ta_event_getmsg_p = dlsym (info->handle, "td_ta_event_getmsg");
- info->td_thr_event_enable_p = dlsym (info->handle, "td_thr_event_enable");
- info->td_thr_tls_get_addr_p = dlsym (info->handle, "td_thr_tls_get_addr");
- info->td_thr_tlsbase_p = dlsym (info->handle, "td_thr_tlsbase");
- if (thread_db_find_new_threads_silently (inferior_ptid) != 0)
- {
-
- return 0;
- }
- printf_unfiltered (_("[Thread debugging using libthread_db enabled]\n"));
- if (*libthread_db_search_path || libthread_db_debug)
- {
- struct ui_file *file;
- const char *library;
- library = dladdr_to_soname (*info->td_ta_new_p);
- if (library == NULL)
- library = LIBTHREAD_DB_SO;
-
- file = *libthread_db_search_path != '\0' ? gdb_stdout : gdb_stdlog;
- fprintf_unfiltered (file, _("Using host libthread_db library \"%s\".\n"),
- library);
- }
-
- if (thread_db_list->next == NULL)
- push_target (&thread_db_ops);
-
- if (target_has_execution && thread_db_use_events ())
- enable_thread_event_reporting ();
- return 1;
- }
- static int
- try_thread_db_load (const char *library, int check_auto_load_safe)
- {
- void *handle;
- struct thread_db_info *info;
- if (libthread_db_debug)
- fprintf_unfiltered (gdb_stdlog,
- _("Trying host libthread_db library: %s.\n"),
- library);
- if (check_auto_load_safe)
- {
- if (access (library, R_OK) != 0)
- {
-
- if (libthread_db_debug)
- fprintf_unfiltered (gdb_stdlog, _("open failed: %s.\n"),
- safe_strerror (errno));
- return 0;
- }
- if (!file_is_auto_load_safe (library, _("auto-load: Loading libthread-db "
- "library \"%s\" from explicit "
- "directory.\n"),
- library))
- return 0;
- }
- handle = dlopen (library, RTLD_NOW);
- if (handle == NULL)
- {
- if (libthread_db_debug)
- fprintf_unfiltered (gdb_stdlog, _("dlopen failed: %s.\n"), dlerror ());
- return 0;
- }
- if (libthread_db_debug && strchr (library, '/') == NULL)
- {
- void *td_init;
- td_init = dlsym (handle, "td_init");
- if (td_init != NULL)
- {
- const char *const libpath = dladdr_to_soname (td_init);
- if (libpath != NULL)
- fprintf_unfiltered (gdb_stdlog, _("Host %s resolved to: %s.\n"),
- library, libpath);
- }
- }
- info = add_thread_db_info (handle);
-
- if (strchr (library, '/') != NULL)
- info->filename = gdb_realpath (library);
- if (try_thread_db_load_1 (info))
- return 1;
-
- delete_thread_db_info (ptid_get_pid (inferior_ptid));
- return 0;
- }
- static int
- try_thread_db_load_from_pdir_1 (struct objfile *obj, const char *subdir)
- {
- struct cleanup *cleanup;
- char *path, *cp;
- int result;
- const char *obj_name = objfile_name (obj);
- if (obj_name[0] != '/')
- {
- warning (_("Expected absolute pathname for libpthread in the"
- " inferior, but got %s."), obj_name);
- return 0;
- }
- path = xmalloc (strlen (obj_name) + (subdir ? strlen (subdir) + 1 : 0)
- + 1 + strlen (LIBTHREAD_DB_SO) + 1);
- cleanup = make_cleanup (xfree, path);
- strcpy (path, obj_name);
- cp = strrchr (path, '/');
-
- gdb_assert (cp != NULL);
- cp[1] = '\0';
- if (subdir != NULL)
- {
- strcat (cp, subdir);
- strcat (cp, "/");
- }
- strcat (cp, LIBTHREAD_DB_SO);
- result = try_thread_db_load (path, 1);
- do_cleanups (cleanup);
- return result;
- }
- static int
- try_thread_db_load_from_pdir (const char *subdir)
- {
- struct objfile *obj;
- if (!auto_load_thread_db)
- return 0;
- ALL_OBJFILES (obj)
- if (libpthread_name_p (objfile_name (obj)))
- {
- if (try_thread_db_load_from_pdir_1 (obj, subdir))
- return 1;
-
- if (obj->separate_debug_objfile_backlink != NULL)
- return try_thread_db_load_from_pdir_1 (obj->separate_debug_objfile_backlink,
- subdir);
- return 0;
- }
- return 0;
- }
- static int
- try_thread_db_load_from_sdir (void)
- {
- return try_thread_db_load (LIBTHREAD_DB_SO, 0);
- }
- static int
- try_thread_db_load_from_dir (const char *dir, size_t dir_len)
- {
- struct cleanup *cleanup;
- char *path;
- int result;
- if (!auto_load_thread_db)
- return 0;
- path = xmalloc (dir_len + 1 + strlen (LIBTHREAD_DB_SO) + 1);
- cleanup = make_cleanup (xfree, path);
- memcpy (path, dir, dir_len);
- path[dir_len] = '/';
- strcpy (path + dir_len + 1, LIBTHREAD_DB_SO);
- result = try_thread_db_load (path, 1);
- do_cleanups (cleanup);
- return result;
- }
- static int
- thread_db_load_search (void)
- {
- VEC (char_ptr) *dir_vec;
- struct cleanup *cleanups;
- char *this_dir;
- int i, rc = 0;
- dir_vec = dirnames_to_char_ptr_vec (libthread_db_search_path);
- cleanups = make_cleanup_free_char_ptr_vec (dir_vec);
- for (i = 0; VEC_iterate (char_ptr, dir_vec, i, this_dir); ++i)
- {
- const int pdir_len = sizeof ("$pdir") - 1;
- size_t this_dir_len;
- this_dir_len = strlen (this_dir);
- if (strncmp (this_dir, "$pdir", pdir_len) == 0
- && (this_dir[pdir_len] == '\0'
- || this_dir[pdir_len] == '/'))
- {
- char *subdir = NULL;
- struct cleanup *free_subdir_cleanup
- = make_cleanup (null_cleanup, NULL);
- if (this_dir[pdir_len] == '/')
- {
- subdir = xmalloc (strlen (this_dir));
- make_cleanup (xfree, subdir);
- strcpy (subdir, this_dir + pdir_len + 1);
- }
- rc = try_thread_db_load_from_pdir (subdir);
- do_cleanups (free_subdir_cleanup);
- if (rc)
- break;
- }
- else if (strcmp (this_dir, "$sdir") == 0)
- {
- if (try_thread_db_load_from_sdir ())
- {
- rc = 1;
- break;
- }
- }
- else
- {
- if (try_thread_db_load_from_dir (this_dir, this_dir_len))
- {
- rc = 1;
- break;
- }
- }
- }
- do_cleanups (cleanups);
- if (libthread_db_debug)
- fprintf_unfiltered (gdb_stdlog,
- _("thread_db_load_search returning %d\n"), rc);
- return rc;
- }
- static int
- has_libpthread (void)
- {
- struct objfile *obj;
- ALL_OBJFILES (obj)
- if (libpthread_name_p (objfile_name (obj)))
- return 1;
- return 0;
- }
- static int
- thread_db_load (void)
- {
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (inferior_ptid));
- if (info != NULL)
- return 1;
-
- if (!target_has_registers)
- return 0;
-
- if (!(target_can_run (¤t_target) || core_bfd))
- return 0;
- if (thread_db_load_search ())
- return 1;
-
- if (has_libpthread ())
- {
- warning (_("Unable to find libthread_db matching inferior's thread"
- " library, thread debugging will not be available."));
- return 0;
- }
-
- return 0;
- }
- static void
- disable_thread_event_reporting (struct thread_db_info *info)
- {
- if (info->td_ta_clear_event_p != NULL)
- {
- td_thr_events_t events;
-
- td_event_fillset (&events);
- info->td_ta_clear_event_p (info->thread_agent, &events);
- }
- info->td_create_bp_addr = 0;
- info->td_death_bp_addr = 0;
- }
- static void
- check_thread_signals (void)
- {
- if (!thread_signals)
- {
- sigset_t mask;
- int i;
- lin_thread_get_thread_signals (&mask);
- sigemptyset (&thread_stop_set);
- sigemptyset (&thread_print_set);
- for (i = 1; i < NSIG; i++)
- {
- if (sigismember (&mask, i))
- {
- if (signal_stop_update (gdb_signal_from_host (i), 0))
- sigaddset (&thread_stop_set, i);
- if (signal_print_update (gdb_signal_from_host (i), 0))
- sigaddset (&thread_print_set, i);
- thread_signals = 1;
- }
- }
- }
- }
- void
- check_for_thread_db (void)
- {
-
- if (!thread_db_load ())
- return;
- }
- static void
- thread_db_new_objfile (struct objfile *objfile)
- {
-
- if (objfile != NULL
-
- && objfile->separate_debug_objfile_backlink == NULL
-
- && ((objfile->flags & OBJF_MAINLINE) != 0
- || libpthread_name_p (objfile_name (objfile))))
- check_for_thread_db ();
- }
- static void
- check_pid_namespace_match (void)
- {
-
- if (target_can_run (¤t_target))
- {
-
- char *our_pid_ns = linux_proc_pid_get_ns (getpid (), "pid");
- char *inferior_pid_ns = linux_proc_pid_get_ns (
- ptid_get_pid (inferior_ptid), "pid");
- if (our_pid_ns != NULL && inferior_pid_ns != NULL
- && strcmp (our_pid_ns, inferior_pid_ns) != 0)
- {
- warning (_ ("Target and debugger are in different PID "
- "namespaces; thread lists and other data are "
- "likely unreliable"));
- }
- xfree (our_pid_ns);
- xfree (inferior_pid_ns);
- }
- }
- static void
- thread_db_inferior_created (struct target_ops *target, int from_tty)
- {
- check_pid_namespace_match ();
- check_for_thread_db ();
- }
- static void
- update_thread_state (struct private_thread_info *private,
- const td_thrinfo_t *ti_p)
- {
- private->dying = (ti_p->ti_state == TD_THR_UNKNOWN
- || ti_p->ti_state == TD_THR_ZOMBIE);
- }
- static int
- attach_thread (ptid_t ptid, const td_thrhandle_t *th_p,
- const td_thrinfo_t *ti_p)
- {
- struct private_thread_info *private;
- struct thread_info *tp;
- td_err_e err;
- struct thread_db_info *info;
-
- tp = find_thread_ptid (ptid);
- if (tp != NULL)
- {
-
- if (tp->private != NULL)
- {
- if (!tp->private->dying)
- return 0;
- delete_thread (ptid);
- tp = NULL;
- }
- }
- if (target_has_execution)
- check_thread_signals ();
-
- if (target_has_execution
- && tp == NULL)
- {
- int res;
- res = lin_lwp_attach_lwp (ptid_build (ptid_get_pid (ptid),
- ti_p->ti_lid, 0));
- if (res < 0)
- {
-
- return 0;
- }
- else if (res > 0)
- {
-
- return 1;
- }
-
- }
-
- private = xmalloc (sizeof (struct private_thread_info));
- memset (private, 0, sizeof (struct private_thread_info));
-
- FIXME
- gdb_assert (ti_p->ti_tid != 0);
- private->th = *th_p;
- private->tid = ti_p->ti_tid;
- update_thread_state (private, ti_p);
-
- if (tp == NULL)
- add_thread_with_info (ptid, private);
- else
- tp->private = private;
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- if (target_has_execution && thread_db_use_events ())
- {
- err = info->td_thr_event_enable_p (th_p, 1);
- if (err != TD_OK)
- error (_("Cannot enable thread event reporting for %s: %s"),
- target_pid_to_str (ptid), thread_db_err_str (err));
- }
- return 1;
- }
- static void
- detach_thread (ptid_t ptid)
- {
- struct thread_info *thread_info;
-
- thread_info = find_thread_ptid (ptid);
- gdb_assert (thread_info != NULL && thread_info->private != NULL);
- thread_info->private->dying = 1;
- }
- static void
- thread_db_detach (struct target_ops *ops, const char *args, int from_tty)
- {
- struct target_ops *target_beneath = find_target_beneath (ops);
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (inferior_ptid));
- if (info)
- {
- if (target_has_execution && thread_db_use_events ())
- {
- disable_thread_event_reporting (info);
-
- remove_thread_event_breakpoints ();
- }
- delete_thread_db_info (ptid_get_pid (inferior_ptid));
- }
- target_beneath->to_detach (target_beneath, args, from_tty);
-
-
- if (!thread_db_list)
- unpush_target (&thread_db_ops);
- }
- static void
- check_event (ptid_t ptid)
- {
- struct regcache *regcache = get_thread_regcache (ptid);
- struct gdbarch *gdbarch = get_regcache_arch (regcache);
- td_event_msg_t msg;
- td_thrinfo_t ti;
- td_err_e err;
- CORE_ADDR stop_pc;
- int loop = 0;
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- stop_pc = regcache_read_pc (regcache)
- - target_decr_pc_after_break (gdbarch);
- if (stop_pc != info->td_create_bp_addr
- && stop_pc != info->td_death_bp_addr)
- return;
-
- info->proc_handle.ptid = ptid;
-
- if (!have_threads (ptid))
- thread_db_find_new_threads_1 (ptid);
-
- loop = 1;
- do
- {
- err = info->td_ta_event_getmsg_p (info->thread_agent, &msg);
- if (err != TD_OK)
- {
- if (err == TD_NOMSG)
- return;
- error (_("Cannot get thread event message: %s"),
- thread_db_err_str (err));
- }
- err = info->td_thr_get_info_p (msg.th_p, &ti);
- if (err != TD_OK)
- error (_("Cannot get thread info: %s"), thread_db_err_str (err));
- ptid = ptid_build (ptid_get_pid (ptid), ti.ti_lid, 0);
- switch (msg.event)
- {
- case TD_CREATE:
-
- attach_thread (ptid, msg.th_p, &ti);
- break;
- case TD_DEATH:
- if (!in_thread_list (ptid))
- error (_("Spurious thread death event."));
- detach_thread (ptid);
- break;
- default:
- error (_("Spurious thread event."));
- }
- }
- while (loop);
- }
- static ptid_t
- thread_db_wait (struct target_ops *ops,
- ptid_t ptid, struct target_waitstatus *ourstatus,
- int options)
- {
- struct thread_db_info *info;
- struct target_ops *beneath = find_target_beneath (ops);
- ptid = beneath->to_wait (beneath, ptid, ourstatus, options);
- if (ourstatus->kind == TARGET_WAITKIND_IGNORE)
- return ptid;
- if (ourstatus->kind == TARGET_WAITKIND_EXITED
- || ourstatus->kind == TARGET_WAITKIND_SIGNALLED)
- return ptid;
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- if (info == NULL)
- return ptid;
- if (ourstatus->kind == TARGET_WAITKIND_EXECD)
- {
-
- delete_thread_db_info (ptid_get_pid (ptid));
- if (!thread_db_list)
- unpush_target (&thread_db_ops);
-
- return ptid;
- }
-
- if (ourstatus->kind == TARGET_WAITKIND_STOPPED && !have_threads (ptid))
- thread_db_find_new_threads_1 (ptid);
- if (ourstatus->kind == TARGET_WAITKIND_STOPPED
- && ourstatus->value.sig == GDB_SIGNAL_TRAP)
-
- check_event (ptid);
- if (have_threads (ptid))
- {
-
- thread_from_lwp (ptid);
- }
- return ptid;
- }
- static void
- thread_db_mourn_inferior (struct target_ops *ops)
- {
- struct target_ops *target_beneath = find_target_beneath (ops);
- delete_thread_db_info (ptid_get_pid (inferior_ptid));
- target_beneath->to_mourn_inferior (target_beneath);
-
- remove_thread_event_breakpoints ();
-
- if (!thread_db_list)
- unpush_target (ops);
- }
- struct callback_data
- {
- struct thread_db_info *info;
- int new_threads;
- };
- static int
- find_new_threads_callback (const td_thrhandle_t *th_p, void *data)
- {
- td_thrinfo_t ti;
- td_err_e err;
- ptid_t ptid;
- struct thread_info *tp;
- struct callback_data *cb_data = data;
- struct thread_db_info *info = cb_data->info;
- err = info->td_thr_get_info_p (th_p, &ti);
- if (err != TD_OK)
- error (_("find_new_threads_callback: cannot get thread info: %s"),
- thread_db_err_str (err));
- if (ti.ti_lid == -1)
- {
-
- return 0;
- }
- if (ti.ti_tid == 0)
- {
-
-
- info->need_stale_parent_threads_check = 0;
- if (target_has_execution && thread_db_use_events ())
- {
- err = info->td_thr_event_enable_p (th_p, 1);
- if (err != TD_OK)
- error (_("Cannot enable thread event reporting for LWP %d: %s"),
- (int) ti.ti_lid, thread_db_err_str (err));
- }
- return 0;
- }
-
- if (info->need_stale_parent_threads_check)
- {
- int tgid = linux_proc_get_tgid (ti.ti_lid);
- if (tgid != -1 && tgid != info->pid)
- return 0;
- }
- ptid = ptid_build (info->pid, ti.ti_lid, 0);
- tp = find_thread_ptid (ptid);
- if (tp == NULL || tp->private == NULL)
- {
- if (attach_thread (ptid, th_p, &ti))
- cb_data->new_threads += 1;
- else
-
- return 1;
- }
- else if (target_has_execution && !thread_db_use_events ())
- {
-
- update_thread_state (tp->private, &ti);
- }
- return 0;
- }
- static int
- find_new_threads_once (struct thread_db_info *info, int iteration,
- td_err_e *errp)
- {
- volatile struct gdb_exception except;
- struct callback_data data;
- td_err_e err = TD_ERR;
- data.info = info;
- data.new_threads = 0;
- TRY_CATCH (except, RETURN_MASK_ERROR)
- {
-
- err = info->td_ta_thr_iter_p (info->thread_agent,
- find_new_threads_callback,
- &data,
- TD_THR_ANY_STATE,
- TD_THR_LOWEST_PRIORITY,
- TD_SIGNO_MASK,
- TD_THR_ANY_USER_FLAGS);
- }
- if (libthread_db_debug)
- {
- if (except.reason < 0)
- exception_fprintf (gdb_stdlog, except,
- "Warning: find_new_threads_once: ");
- fprintf_unfiltered (gdb_stdlog,
- _("Found %d new threads in iteration %d.\n"),
- data.new_threads, iteration);
- }
- if (errp != NULL)
- *errp = err;
- return data.new_threads;
- }
- static void
- thread_db_find_new_threads_2 (ptid_t ptid, int until_no_new)
- {
- td_err_e err = TD_OK;
- struct thread_db_info *info;
- int i, loop;
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- info->proc_handle.ptid = ptid;
- if (until_no_new)
- {
-
- for (i = 0, loop = 0; loop < 4 && err == TD_OK; ++i, ++loop)
- if (find_new_threads_once (info, i, &err) != 0)
- {
-
- loop = -1;
- }
- }
- else
- find_new_threads_once (info, 0, &err);
- if (err != TD_OK)
- error (_("Cannot find new threads: %s"), thread_db_err_str (err));
- }
- static void
- thread_db_find_new_threads_1 (ptid_t ptid)
- {
- thread_db_find_new_threads_2 (ptid, 0);
- }
- static int
- update_thread_core (struct lwp_info *info, void *closure)
- {
- info->core = linux_common_core_of_thread (info->ptid);
- return 0;
- }
- static void
- thread_db_update_thread_list (struct target_ops *ops)
- {
- struct thread_db_info *info;
- struct inferior *inf;
- prune_threads ();
- ALL_INFERIORS (inf)
- {
- struct thread_info *thread;
- if (inf->pid == 0)
- continue;
- info = get_thread_db_info (inf->pid);
- if (info == NULL)
- continue;
- thread = any_live_thread_of_process (inf->pid);
- if (thread == NULL || thread->executing)
- continue;
- thread_db_find_new_threads_1 (thread->ptid);
- }
- if (target_has_execution)
- iterate_over_lwps (minus_one_ptid ,
- update_thread_core, NULL);
- }
- static char *
- thread_db_pid_to_str (struct target_ops *ops, ptid_t ptid)
- {
- struct thread_info *thread_info = find_thread_ptid (ptid);
- struct target_ops *beneath;
- if (thread_info != NULL && thread_info->private != NULL)
- {
- static char buf[64];
- thread_t tid;
- tid = thread_info->private->tid;
- snprintf (buf, sizeof (buf), "Thread 0x%lx (LWP %ld)",
- tid, ptid_get_lwp (ptid));
- return buf;
- }
- beneath = find_target_beneath (ops);
- return beneath->to_pid_to_str (beneath, ptid);
- }
- static char *
- thread_db_extra_thread_info (struct target_ops *self,
- struct thread_info *info)
- {
- if (info->private == NULL)
- return NULL;
- if (info->private->dying)
- return "Exiting";
- return NULL;
- }
- static CORE_ADDR
- thread_db_get_thread_local_address (struct target_ops *ops,
- ptid_t ptid,
- CORE_ADDR lm,
- CORE_ADDR offset)
- {
- struct thread_info *thread_info;
- struct target_ops *beneath;
-
- if (!have_threads (ptid))
- thread_db_find_new_threads_1 (ptid);
-
- thread_info = find_thread_ptid (ptid);
- if (thread_info != NULL && thread_info->private != NULL)
- {
- td_err_e err;
- psaddr_t address;
- struct thread_db_info *info;
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- if (lm != 0)
- {
-
- if (!info->td_thr_tls_get_addr_p)
- throw_error (TLS_NO_LIBRARY_SUPPORT_ERROR,
- _("No TLS library support"));
-
- err = info->td_thr_tls_get_addr_p (&thread_info->private->th,
- (psaddr_t)(uintptr_t) lm,
- offset, &address);
- }
- else
- {
-
- if (!info->td_thr_tlsbase_p)
- throw_error (TLS_LOAD_MODULE_NOT_FOUND_ERROR,
- _("TLS load module not found"));
-
- err = info->td_thr_tlsbase_p (&thread_info->private->th,
- 1, &address);
- address = (char *) address + offset;
- }
- #ifdef THREAD_DB_HAS_TD_NOTALLOC
-
- if (err == TD_NOTALLOC)
-
- throw_error (TLS_NOT_ALLOCATED_YET_ERROR,
- _("TLS not allocated yet"));
- #endif
-
- if (err != TD_OK)
- throw_error (TLS_GENERIC_ERROR,
- (("%s")), thread_db_err_str (err));
-
-
- gdb_assert (exec_bfd);
- return (bfd_get_sign_extend_vma (exec_bfd) > 0
- ? (CORE_ADDR) (intptr_t) address
- : (CORE_ADDR) (uintptr_t) address);
- }
- beneath = find_target_beneath (ops);
- return beneath->to_get_thread_local_address (beneath, ptid, lm, offset);
- }
- static int
- thread_db_find_thread_from_tid (struct thread_info *thread, void *data)
- {
- long *tid = (long *) data;
- if (thread->private->tid == *tid)
- return 1;
- return 0;
- }
- static ptid_t
- thread_db_get_ada_task_ptid (struct target_ops *self, long lwp, long thread)
- {
- struct thread_info *thread_info;
- thread_db_find_new_threads_1 (inferior_ptid);
- thread_info = iterate_over_threads (thread_db_find_thread_from_tid, &thread);
- gdb_assert (thread_info != NULL);
- return (thread_info->ptid);
- }
- static void
- thread_db_resume (struct target_ops *ops,
- ptid_t ptid, int step, enum gdb_signal signo)
- {
- struct target_ops *beneath = find_target_beneath (ops);
- struct thread_db_info *info;
- if (ptid_equal (ptid, minus_one_ptid))
- info = get_thread_db_info (ptid_get_pid (inferior_ptid));
- else
- info = get_thread_db_info (ptid_get_pid (ptid));
-
- if (info)
- info->need_stale_parent_threads_check = 0;
- beneath->to_resume (beneath, ptid, step, signo);
- }
- static int
- info_auto_load_libthread_db_compare (const void *ap, const void *bp)
- {
- struct thread_db_info *a = *(struct thread_db_info **) ap;
- struct thread_db_info *b = *(struct thread_db_info **) bp;
- int retval;
- retval = strcmp (a->filename, b->filename);
- if (retval)
- return retval;
- return (a->pid > b->pid) - (a->pid - b->pid);
- }
- static void
- info_auto_load_libthread_db (char *args, int from_tty)
- {
- struct ui_out *uiout = current_uiout;
- const char *cs = args ? args : "";
- struct thread_db_info *info, **array;
- unsigned info_count, unique_filenames;
- size_t max_filename_len, max_pids_len, pids_len;
- struct cleanup *back_to;
- char *pids;
- int i;
- cs = skip_spaces_const (cs);
- if (*cs)
- error (_("'info auto-load libthread-db' does not accept any parameters"));
- info_count = 0;
- for (info = thread_db_list; info; info = info->next)
- if (info->filename != NULL)
- info_count++;
- array = xmalloc (sizeof (*array) * info_count);
- back_to = make_cleanup (xfree, array);
- info_count = 0;
- for (info = thread_db_list; info; info = info->next)
- if (info->filename != NULL)
- array[info_count++] = info;
-
- qsort (array, info_count, sizeof (*array),
- info_auto_load_libthread_db_compare);
-
- unique_filenames = 0;
- max_filename_len = 0;
- max_pids_len = 0;
- pids_len = 0;
- for (i = 0; i < info_count; i++)
- {
- int pid = array[i]->pid;
- size_t this_pid_len;
- for (this_pid_len = 0; pid != 0; pid /= 10)
- this_pid_len++;
- if (i == 0 || strcmp (array[i - 1]->filename, array[i]->filename) != 0)
- {
- unique_filenames++;
- max_filename_len = max (max_filename_len,
- strlen (array[i]->filename));
- if (i > 0)
- {
- pids_len -= strlen (", ");
- max_pids_len = max (max_pids_len, pids_len);
- }
- pids_len = 0;
- }
- pids_len += this_pid_len + strlen (", ");
- }
- if (i)
- {
- pids_len -= strlen (", ");
- max_pids_len = max (max_pids_len, pids_len);
- }
-
- if (info_count > 0 && args == auto_load_info_scripts_pattern_nl)
- ui_out_text (uiout, "\n");
- make_cleanup_ui_out_table_begin_end (uiout, 2, unique_filenames,
- "LinuxThreadDbTable");
- ui_out_table_header (uiout, max_filename_len, ui_left, "filename",
- "Filename");
- ui_out_table_header (uiout, pids_len, ui_left, "PIDs", "Pids");
- ui_out_table_body (uiout);
- pids = xmalloc (max_pids_len + 1);
- make_cleanup (xfree, pids);
-
- for (i = 0; i < info_count;)
- {
- struct cleanup *chain = make_cleanup_ui_out_tuple_begin_end (uiout, NULL);
- char *pids_end;
- info = array[i];
- ui_out_field_string (uiout, "filename", info->filename);
- pids_end = pids;
- while (i < info_count && strcmp (info->filename, array[i]->filename) == 0)
- {
- if (pids_end != pids)
- {
- *pids_end++ = ',';
- *pids_end++ = ' ';
- }
- pids_end += xsnprintf (pids_end, &pids[max_pids_len + 1] - pids_end,
- "%u", array[i]->pid);
- gdb_assert (pids_end < &pids[max_pids_len + 1]);
- i++;
- }
- *pids_end = '\0';
- ui_out_field_string (uiout, "pids", pids);
- ui_out_text (uiout, "\n");
- do_cleanups (chain);
- }
- do_cleanups (back_to);
- if (info_count == 0)
- ui_out_message (uiout, 0, _("No auto-loaded libthread-db.\n"));
- }
- static void
- init_thread_db_ops (void)
- {
- thread_db_ops.to_shortname = "multi-thread";
- thread_db_ops.to_longname = "multi-threaded child process.";
- thread_db_ops.to_doc = "Threads and pthreads support.";
- thread_db_ops.to_detach = thread_db_detach;
- thread_db_ops.to_wait = thread_db_wait;
- thread_db_ops.to_resume = thread_db_resume;
- thread_db_ops.to_mourn_inferior = thread_db_mourn_inferior;
- thread_db_ops.to_update_thread_list = thread_db_update_thread_list;
- thread_db_ops.to_pid_to_str = thread_db_pid_to_str;
- thread_db_ops.to_stratum = thread_stratum;
- thread_db_ops.to_has_thread_control = tc_schedlock;
- thread_db_ops.to_get_thread_local_address
- = thread_db_get_thread_local_address;
- thread_db_ops.to_extra_thread_info = thread_db_extra_thread_info;
- thread_db_ops.to_get_ada_task_ptid = thread_db_get_ada_task_ptid;
- thread_db_ops.to_magic = OPS_MAGIC;
- complete_target_initialization (&thread_db_ops);
- }
- extern initialize_file_ftype _initialize_thread_db;
- void
- _initialize_thread_db (void)
- {
- init_thread_db_ops ();
-
- libthread_db_search_path = xstrdup (LIBTHREAD_DB_SEARCH_PATH);
- add_setshow_optional_filename_cmd ("libthread-db-search-path",
- class_support,
- &libthread_db_search_path, _("\
- Set search path for libthread_db."), _("\
- Show the current search path or libthread_db."), _("\
- This path is used to search for libthread_db to be loaded into \
- gdb itself.\n\
- Its value is a colon (':') separate list of directories to search.\n\
- Setting the search path to an empty list resets it to its default value."),
- set_libthread_db_search_path,
- NULL,
- &setlist, &showlist);
- add_setshow_zuinteger_cmd ("libthread-db", class_maintenance,
- &libthread_db_debug, _("\
- Set libthread-db debugging."), _("\
- Show libthread-db debugging."), _("\
- When non-zero, libthread-db debugging is enabled."),
- NULL,
- show_libthread_db_debug,
- &setdebuglist, &showdebuglist);
- add_setshow_boolean_cmd ("libthread-db", class_support,
- &auto_load_thread_db, _("\
- Enable or disable auto-loading of inferior specific libthread_db."), _("\
- Show whether auto-loading inferior specific libthread_db is enabled."), _("\
- If enabled, libthread_db will be searched in 'set libthread-db-search-path'\n\
- locations to load libthread_db compatible with the inferior.\n\
- Standard system libthread_db still gets loaded even with this option off.\n\
- This options has security implications for untrusted inferiors."),
- NULL, show_auto_load_thread_db,
- auto_load_set_cmdlist_get (),
- auto_load_show_cmdlist_get ());
- add_cmd ("libthread-db", class_info, info_auto_load_libthread_db,
- _("Print the list of loaded inferior specific libthread_db.\n\
- Usage: info auto-load libthread-db"),
- auto_load_info_cmdlist_get ());
-
- observer_attach_new_objfile (thread_db_new_objfile);
-
- observer_attach_inferior_created (thread_db_inferior_created);
- }