Blame nptl_db/td_ta_thr_iter.c

Packit Service 82fcde
/* Iterate over a process's threads.
Packit Service 82fcde
   Copyright (C) 1999-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Ulrich Drepper <drepper@redhat.com>, 1999.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include "thread_dbP.h"
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
static td_err_e
Packit Service 82fcde
iterate_thread_list (td_thragent_t *ta, td_thr_iter_f *callback,
Packit Service 82fcde
		     void *cbdata_p, td_thr_state_e state, int ti_pri,
Packit Service 82fcde
		     psaddr_t head, bool fake_empty)
Packit Service 82fcde
{
Packit Service 82fcde
  td_err_e err;
Packit Service 82fcde
  psaddr_t next, ofs;
Packit Service 82fcde
  void *copy;
Packit Service 82fcde
Packit Service 82fcde
  /* Test the state.
Packit Service 82fcde
     XXX This is incomplete.  Normally this test should be in the loop.  */
Packit Service 82fcde
  if (state != TD_THR_ANY_STATE)
Packit Service 82fcde
    return TD_OK;
Packit Service 82fcde
Packit Service 82fcde
  err = DB_GET_FIELD (next, ta, head, list_t, next, 0);
Packit Service 82fcde
  if (err != TD_OK)
Packit Service 82fcde
    return err;
Packit Service 82fcde
Packit Service 82fcde
  if (next == 0 && fake_empty)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* __pthread_initialize_minimal has not run.  There is just the main
Packit Service 82fcde
	 thread to return.  We cannot rely on its thread register.  They
Packit Service 82fcde
	 sometimes contain garbage that would confuse us, left by the
Packit Service 82fcde
	 kernel at exec.  So if it looks like initialization is incomplete,
Packit Service 82fcde
	 we only fake a special descriptor for the initial thread.  */
Packit Service 82fcde
      td_thrhandle_t th = { ta, 0 };
Packit Service 82fcde
      return callback (&th, cbdata_p) != 0 ? TD_DBERR : TD_OK;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Cache the offset from struct pthread to its list_t member.  */
Packit Service 82fcde
  err = DB_GET_FIELD_ADDRESS (ofs, ta, 0, pthread, list, 0);
Packit Service 82fcde
  if (err != TD_OK)
Packit Service 82fcde
    return err;
Packit Service 82fcde
Packit Service 82fcde
  if (ta->ta_sizeof_pthread == 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      err = _td_check_sizeof (ta, &ta->ta_sizeof_pthread, SYM_SIZEOF_pthread);
Packit Service 82fcde
      if (err != TD_OK)
Packit Service 82fcde
	return err;
Packit Service 82fcde
    }
Packit Service 82fcde
  copy = __alloca (ta->ta_sizeof_pthread);
Packit Service 82fcde
Packit Service 82fcde
  while (next != head)
Packit Service 82fcde
    {
Packit Service 82fcde
      psaddr_t addr, schedpolicy, schedprio;
Packit Service 82fcde
Packit Service 82fcde
      addr = next - (ofs - (psaddr_t) 0);
Packit Service 82fcde
      if (next == 0 || addr == 0) /* Sanity check.  */
Packit Service 82fcde
	return TD_DBERR;
Packit Service 82fcde
Packit Service 82fcde
      /* Copy the whole descriptor in once so we can access the several
Packit Service 82fcde
	 fields locally.  Excess copying in one go is much better than
Packit Service 82fcde
	 multiple ps_pdread calls.  */
Packit Service 82fcde
      if (ps_pdread (ta->ph, addr, copy, ta->ta_sizeof_pthread) != PS_OK)
Packit Service 82fcde
	return TD_ERR;
Packit Service 82fcde
Packit Service 82fcde
      err = DB_GET_FIELD_LOCAL (schedpolicy, ta, copy, pthread,
Packit Service 82fcde
				schedpolicy, 0);
Packit Service 82fcde
      if (err != TD_OK)
Packit Service 82fcde
	break;
Packit Service 82fcde
      err = DB_GET_FIELD_LOCAL (schedprio, ta, copy, pthread,
Packit Service 82fcde
				schedparam_sched_priority, 0);
Packit Service 82fcde
      if (err != TD_OK)
Packit Service 82fcde
	break;
Packit Service 82fcde
Packit Service 82fcde
      /* Now test whether this thread matches the specified conditions.  */
Packit Service 82fcde
Packit Service 82fcde
      /* Only if the priority level is as high or higher.  */
Packit Service 82fcde
      int descr_pri = ((uintptr_t) schedpolicy == SCHED_OTHER
Packit Service 82fcde
		       ? 0 : (uintptr_t) schedprio);
Packit Service 82fcde
      if (descr_pri >= ti_pri)
Packit Service 82fcde
	{
Packit Service 82fcde
	  /* Yep, it matches.  Call the callback function.  */
Packit Service 82fcde
	  td_thrhandle_t th;
Packit Service 82fcde
	  th.th_ta_p = (td_thragent_t *) ta;
Packit Service 82fcde
	  th.th_unique = addr;
Packit Service 82fcde
	  if (callback (&th, cbdata_p) != 0)
Packit Service 82fcde
	    return TD_DBERR;
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      /* Get the pointer to the next element.  */
Packit Service 82fcde
      err = DB_GET_FIELD_LOCAL (next, ta, copy + (ofs - (psaddr_t) 0), list_t,
Packit Service 82fcde
				next, 0);
Packit Service 82fcde
      if (err != TD_OK)
Packit Service 82fcde
	break;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return err;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
td_err_e
Packit Service 82fcde
td_ta_thr_iter (const td_thragent_t *ta_arg, td_thr_iter_f *callback,
Packit Service 82fcde
		void *cbdata_p, td_thr_state_e state, int ti_pri,
Packit Service 82fcde
		sigset_t *ti_sigmask_p, unsigned int ti_user_flags)
Packit Service 82fcde
{
Packit Service 82fcde
  td_thragent_t *const ta = (td_thragent_t *) ta_arg;
Packit Service 82fcde
  td_err_e err;
Packit Service 82fcde
  psaddr_t list = 0;
Packit Service 82fcde
Packit Service 82fcde
  LOG ("td_ta_thr_iter");
Packit Service 82fcde
Packit Service 82fcde
  /* Test whether the TA parameter is ok.  */
Packit Service 82fcde
  if (! ta_ok (ta))
Packit Service 82fcde
    return TD_BADTA;
Packit Service 82fcde
Packit Service 82fcde
  /* The thread library keeps two lists for the running threads.  One
Packit Service 82fcde
     list contains the thread which are using user-provided stacks
Packit Service 82fcde
     (this includes the main thread) and the other includes the
Packit Service 82fcde
     threads for which the thread library allocated the stacks.  We
Packit Service 82fcde
     have to iterate over both lists separately.  We start with the
Packit Service 82fcde
     list of threads with user-defined stacks.  */
Packit Service 82fcde
Packit Service 82fcde
  err = DB_GET_SYMBOL (list, ta, __stack_user);
Packit Service 82fcde
  if (err == TD_OK)
Packit Service 82fcde
    err = iterate_thread_list (ta, callback, cbdata_p, state, ti_pri,
Packit Service 82fcde
			       list, true);
Packit Service 82fcde
Packit Service 82fcde
  /* And the threads with stacks allocated by the implementation.  */
Packit Service 82fcde
  if (err == TD_OK)
Packit Service 82fcde
    err = DB_GET_SYMBOL (list, ta, stack_used);
Packit Service 82fcde
  if (err == TD_OK)
Packit Service 82fcde
    err = iterate_thread_list (ta, callback, cbdata_p, state, ti_pri,
Packit Service 82fcde
			       list, false);
Packit Service 82fcde
Packit Service 82fcde
  return err;
Packit Service 82fcde
}