1cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng/* Core file handling.
2cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Copyright (C) 2008 Red Hat, Inc.
3cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   This file is part of Red Hat elfutils.
4cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
5cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat elfutils is free software; you can redistribute it and/or modify
6cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   it under the terms of the GNU General Public License as published by the
7cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Free Software Foundation; version 2 of the License.
8cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
9cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat elfutils is distributed in the hope that it will be useful, but
10cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   WITHOUT ANY WARRANTY; without even the implied warranty of
11cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   General Public License for more details.
13cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
14cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   You should have received a copy of the GNU General Public License along
15cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   with Red Hat elfutils; if not, write to the Free Software Foundation,
16cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301 USA.
17cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
18cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   In addition, as a special exception, Red Hat, Inc. gives You the
19cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   additional right to link the code of Red Hat elfutils with code licensed
20cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   under any Open Source Initiative certified open source license
21cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   (http://www.opensource.org/licenses/index.php) which requires the
22cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   distribution of source code with any binary distribution and to
23cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   distribute linked combinations of the two.  Non-GPL Code permitted under
24cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   this exception must only link to the code of Red Hat elfutils through
25cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   those well defined interfaces identified in the file named EXCEPTION
26cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   found in the source code files (the "Approved Interfaces").  The files
27cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   of Non-GPL Code may instantiate templates or use macros or inline
28cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   functions from the Approved Interfaces without causing the resulting
29cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   work to be covered by the GNU General Public License.  Only Red Hat,
30cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Inc. may make changes or additions to the list of Approved Interfaces.
31cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat's grant of this exception is conditioned upon your not adding
32cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   any new exceptions.  If you wish to add a new Approved Interface or
33cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   exception, please contact Red Hat.  You must obey the GNU General Public
34cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   License in all respects for all of the Red Hat elfutils code and other
35cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   code used in conjunction with Red Hat elfutils except the Non-GPL Code
36cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   covered by this exception.  If you modify this file, you may extend this
37cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   exception to your version of the file, but you are not obligated to do
38cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   so.  If you do not wish to provide this exception without modification,
39cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   you must delete this exception statement from your version and license
40cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   this file solely under the GPL without exception.
41cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
42cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat elfutils is an included package of the Open Invention Network.
43cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   An included package of the Open Invention Network is a package for which
44cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Open Invention Network licensees cross-license their patents.  No patent
45cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   license is granted, either expressly or impliedly, by designation as an
46cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   included package.  Should you wish to participate in the Open Invention
47cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Network licensing program, please visit www.openinventionnetwork.com
48cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   <http://www.openinventionnetwork.com>.  */
49cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
50cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <config.h>
51cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include "../libelf/libelfP.h"	/* For NOTE_ALIGN.  */
52cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#undef	_
53cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include "libdwflP.h"
54cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <gelf.h>
55cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
56cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <sys/param.h>
57cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <unistd.h>
58cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <endian.h>
59cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <byteswap.h>
60cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include "system.h"
61cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
62cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
63cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng/* This is a prototype of what a new libelf interface might be.
64cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   This implementation is pessimal for non-mmap cases and should
65cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   be replaced by more diddling inside libelf internals.  */
66cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengstatic Elf *
67cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengelf_begin_rand (Elf *parent, loff_t offset, loff_t size, loff_t *next)
68cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng{
69cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (parent == NULL)
70cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return NULL;
71cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
72cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* On failure return, we update *NEXT to point back at OFFSET.  */
73cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  inline Elf *fail (int error)
74cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  {
75cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    if (next != NULL)
76cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *next = offset;
77cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    //__libelf_seterrno (error);
78cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    __libdwfl_seterrno (DWFL_E (LIBELF, error));
79cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return NULL;
80cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  }
81cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
82cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  loff_t min = (parent->kind == ELF_K_ELF ?
83cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		(parent->class == ELFCLASS32
84cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		 ? sizeof (Elf32_Ehdr) : sizeof (Elf64_Ehdr))
85cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		: parent->kind == ELF_K_AR ? SARMAG
86cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		: 0);
87cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
88cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely (offset < min)
89cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      || unlikely (offset >= (loff_t) parent->maximum_size))
90cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return fail (ELF_E_RANGE);
91cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
92cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* For an archive, fetch just the size field
93cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     from the archive header to override SIZE.  */
94cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (parent->kind == ELF_K_AR)
95cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
96cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      struct ar_hdr h = { .ar_size = "" };
97cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
98cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (unlikely (parent->maximum_size - offset < sizeof h))
99cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	return fail (ELF_E_RANGE);
100cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
101cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (parent->map_address != NULL)
102cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	memcpy (h.ar_size, parent->map_address + parent->start_offset + offset,
103cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		sizeof h.ar_size);
104cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      else if (unlikely (pread_retry (parent->fildes,
105cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng				      h.ar_size, sizeof (h.ar_size),
106cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng				      parent->start_offset + offset
107cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng				      + offsetof (struct ar_hdr, ar_size))
108cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			 != sizeof (h.ar_size)))
109cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	return fail (ELF_E_READ_ERROR);
110cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
111cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      offset += sizeof h;
112cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
113cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      char *endp;
114cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      size = strtoll (h.ar_size, &endp, 10);
115cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (unlikely (endp == h.ar_size)
116cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  || unlikely ((loff_t) parent->maximum_size - offset < size))
117cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	return fail (ELF_E_INVALID_ARCHIVE);
118cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
119cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
120cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely ((loff_t) parent->maximum_size - offset < size))
121cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return fail (ELF_E_RANGE);
122cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
123cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Even if we fail at this point, update *NEXT to point past the file.  */
124cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (next != NULL)
125cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    *next = offset + size;
126cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
127cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely (offset == 0)
128cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      && unlikely (size == (loff_t) parent->maximum_size))
129cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return elf_clone (parent, parent->cmd);
130cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
131cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Note the image is guaranteed live only as long as PARENT
132cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     lives.  Using elf_memory is quite suboptimal if the whole
133cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     file is not mmap'd.  We really should have something like
134cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     a generalization of the archive support.  */
135cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  Elf_Data *data = elf_getdata_rawchunk (parent, offset, size, ELF_T_BYTE);
136cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (data == NULL)
137cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return NULL;
138cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  assert ((loff_t) data->d_size == size);
139cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  return elf_memory (data->d_buf, size);
140cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng}
141cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
142cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
143cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengint
144cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengdwfl_report_core_segments (Dwfl *dwfl, Elf *elf, const GElf_Ehdr *ehdr,
145cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			   GElf_Phdr *notes)
146cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng{
147cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely (dwfl == NULL))
148cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return -1;
149cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
150cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely (elf == NULL) || unlikely (ehdr == NULL))
151cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
152cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      __libdw_seterrno (DWFL_E_LIBELF);
153cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      return -1;
154cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
155cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
156cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  int result = 0;
157cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
158cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (notes != NULL)
159cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    notes->p_type = PT_NULL;
160cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
161cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  for (int ndx = 0; result >= 0 && ndx < ehdr->e_phnum; ++ndx)
162cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
163cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      GElf_Phdr phdr_mem;
164cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      GElf_Phdr *phdr = gelf_getphdr (elf, ndx, &phdr_mem);
165cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (unlikely (phdr == NULL))
166cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
167cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  __libdwfl_seterrno (DWFL_E_LIBELF);
168cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  return -1;
169cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
170cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      switch (phdr->p_type)
171cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
172cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	case PT_LOAD:
173cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  result = dwfl_report_segment (dwfl, ndx, phdr, 0, NULL);
174cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  break;
175cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
176cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	case PT_NOTE:
177cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (notes != NULL)
178cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    {
179cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      *notes = *phdr;
180cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      notes = NULL;
181cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    }
182cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  break;
183cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
184cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
185cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
186cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  return result;
187cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng}
188cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
189cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng/* Never read more than this much without mmap.  */
190cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#define MAX_EAGER_COST	8192
191cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
192cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengstatic bool
193cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengcore_file_read_eagerly (Dwfl_Module *mod,
194cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			void **userdata __attribute__ ((unused)),
195cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			const char *name __attribute__ ((unused)),
196cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			Dwarf_Addr start __attribute__ ((unused)),
197cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			void **buffer, size_t *buffer_available,
198cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			GElf_Off cost, GElf_Off worthwhile,
199cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			GElf_Off whole,
200cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			GElf_Off contiguous __attribute__ ((unused)),
201cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			void *arg, Elf **elfp)
202cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng{
203cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  Elf *core = arg;
204cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
205cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (whole <= *buffer_available)
206cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
207cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      /* All there ever was, we already have on hand.  */
208cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
209cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (core->map_address == NULL)
210cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
211cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  /* We already malloc'd the buffer.  */
212cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *elfp = elf_memory (*buffer, whole);
213cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (unlikely (*elfp == NULL))
214cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    return false;
215cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
216cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  (*elfp)->flags |= ELF_F_MALLOCED;
217cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *buffer = NULL;
218cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *buffer_available = 0;
219cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  return true;
220cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
221cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
222cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      /* We can use the image inside the core file directly.  */
223cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *elfp = elf_begin_rand (core, *buffer - core->map_address, whole, NULL);
224cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *buffer = NULL;
225cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *buffer_available = 0;
226cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      return *elfp != NULL;
227cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
228cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
229cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* We don't have the whole file.
230cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     Figure out if this is better than nothing.  */
231cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
232cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (worthwhile == 0)
233cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    /* Caller doesn't think so.  */
234cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return false;
235cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
236cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /*
237cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    XXX would like to fall back to partial file via memory
238cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    when build id find_elf fails
239cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    also, link_map name may give file name from disk better than partial here
240cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    requires find_elf hook re-doing the magic to fall back if no file found
241cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  */
242cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
243cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (mod->build_id_len > 0)
244cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    /* There is a build ID that could help us find the whole file,
245cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng       which might be more useful than what we have.
246cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng       We'll just rely on that.  */
247cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return false;
248cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
249cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (core->map_address != NULL)
250cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    /* It's cheap to get, so get it.  */
251cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return true;
252cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
253cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Only use it if there isn't too much to be read.  */
254cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  return cost <= MAX_EAGER_COST;
255cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng}
256cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
257cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengbool
258cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengdwfl_elf_phdr_memory_callback (Dwfl *dwfl, int ndx,
259cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			       void **buffer, size_t *buffer_available,
260cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			       GElf_Addr vaddr,
261cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			       size_t minread,
262cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			       void *arg)
263cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng{
264cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  Elf *elf = arg;
265cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
266cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (ndx == -1)
267cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
268cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      /* Called for cleanup.  */
269cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (elf->map_address == NULL)
270cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	free (*buffer);
271cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *buffer = NULL;
272cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *buffer_available = 0;
273cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      return false;
274cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
275cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
276cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  const GElf_Off align = dwfl->segment_align ?: 1;
277cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  GElf_Phdr phdr;
278cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
279cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  do
280cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    if (unlikely (gelf_getphdr (elf, ndx++, &phdr) == NULL))
281cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      return true;
282cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  while (phdr.p_type != PT_LOAD
283cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	 || ((phdr.p_vaddr + phdr.p_memsz + align - 1) & -align) <= vaddr);
284cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
285cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  GElf_Off start = vaddr - phdr.p_vaddr + phdr.p_offset;
286cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  GElf_Off end = (phdr.p_offset + phdr.p_filesz + align - 1) & -align;
287cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
288cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Use following contiguous segments to get towards SIZE.  */
289cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  inline bool more (size_t size)
290cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  {
291cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    while (end <= start || end - start < size)
292cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      {
293cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	if (phdr.p_filesz < phdr.p_memsz)
294cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  /* This segment is truncated, so no following one helps us.  */
295cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  return false;
296cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
297cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	if (unlikely (gelf_getphdr (elf, ndx++, &phdr) == NULL))
298cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  return false;
299cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
300cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	if (phdr.p_type == PT_LOAD)
301cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  {
302cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    if (phdr.p_offset > end)
303cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      /* It's discontiguous!  */
304cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      return false;
305cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
306cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    end = (phdr.p_offset + phdr.p_filesz + align - 1) & -align;
307cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  }
308cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      }
309cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return true;
310cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  }
311cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
312cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* We need at least this much.  */
313cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (! more (minread))
314cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return false;
315cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
316cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* See how much more we can get of what the caller wants.  */
317cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  (void) more (*buffer_available);
318cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
319cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* If it's already on hand anyway, use as much as there is.  */
320cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (elf->map_address != NULL)
321cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    (void) more (elf->maximum_size - start);
322cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
323cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely (end - start > elf->maximum_size))
324cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    end = start + elf->maximum_size;
325cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
326cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (elf->map_address != NULL)
327cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
328cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      void *contents = elf->map_address + elf->start_offset + start;
329cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      size_t size = end - start;
330cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
331cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (minread == 0)		/* String mode.  */
332cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
333cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  const void *eos = memchr (contents, '\0', size);
334cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (unlikely (eos == NULL) || unlikely (eos == contents))
335cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    return false;
336cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  size = eos + 1 - contents;
337cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
338cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
339cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (*buffer == NULL)
340cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
341cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *buffer = contents;
342cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *buffer_available = size;
343cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
344cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      else
345cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
346cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *buffer_available = MIN (size, *buffer_available);
347cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  memcpy (*buffer, contents, *buffer_available);
348cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
349cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
350cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  else
351cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
352cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      void *into = *buffer;
353cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (*buffer == NULL)
354cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
355cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  *buffer_available = MIN (minread ?: 512,
356cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng				   MAX (4096, MIN (end - start,
357cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng						   *buffer_available)));
358cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  into = malloc (*buffer_available);
359cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (unlikely (into == NULL))
360cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    {
361cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      __libdwfl_seterrno (DWFL_E_NOMEM);
362cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      return false;
363cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    }
364cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
365cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
366cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      ssize_t nread = pread_retry (elf->fildes, into, *buffer_available, start);
367cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (nread < (ssize_t) minread)
368cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
369cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (into != *buffer)
370cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    free (into);
371cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (nread < 0)
372cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    __libdwfl_seterrno (DWFL_E_ERRNO);
373cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  return false;
374cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
375cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
376cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (minread == 0)		/* String mode.  */
377cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
378cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  const void *eos = memchr (into, '\0', nread);
379cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  if (unlikely (eos == NULL) || unlikely (eos == into))
380cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    {
381cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      if (*buffer == NULL)
382cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		free (into);
383cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      return false;
384cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    }
385cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  nread = eos + 1 - into;
386cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
387cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
388cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (*buffer == NULL)
389cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	*buffer = into;
390cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      *buffer_available = nread;
391cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
392cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
393cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  return true;
394cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng}
395cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
396cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengint
397cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Chengdwfl_core_file_report (Dwfl *dwfl, Elf *elf, const GElf_Ehdr *ehdr)
398cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng{
399cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  GElf_Phdr notes_phdr;
400cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
401cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* First report each PT_LOAD segment.  */
402cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  int ndx = dwfl_report_core_segments (dwfl, elf, ehdr, &notes_phdr);
403cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (unlikely (ndx <= 0))
404cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    return ndx;
405cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
406cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Now sniff segment contents for modules.  */
407cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  ndx = 0;
408cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  do
409cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
410cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      int seg = dwfl_segment_report_module (dwfl, ndx, NULL,
411cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng					    &dwfl_elf_phdr_memory_callback, elf,
412cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng					    core_file_read_eagerly, elf);
413cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (unlikely (seg < 0))
414cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	return seg;
415cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      ndx = seg > ndx ? seg : ndx + 1;
416cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
417cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  while (ndx < ehdr->e_phnum);
418cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
419cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Next, we should follow the chain from DT_DEBUG.  */
420cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
421cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  const void *auxv = NULL;
422cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  size_t auxv_size = 0;
423cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (likely (notes_phdr.p_type == PT_NOTE))
424cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
425cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      /* PT_NOTE -> NT_AUXV -> AT_PHDR -> PT_DYNAMIC -> DT_DEBUG */
426cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
427cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      Elf_Data *notes = elf_getdata_rawchunk (elf,
428cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng					      notes_phdr.p_offset,
429cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng					      notes_phdr.p_filesz,
430cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng					      ELF_T_NHDR);
431cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (likely (notes != NULL))
432cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	{
433cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  size_t pos = 0;
434cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  GElf_Nhdr nhdr;
435cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  size_t name_pos;
436cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  size_t desc_pos;
437cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	  while ((pos = gelf_getnote (notes, pos, &nhdr,
438cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng				      &name_pos, &desc_pos)) > 0)
439cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	    if (nhdr.n_type == NT_AUXV
440cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		&& nhdr.n_namesz == sizeof "CORE"
441cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		&& !memcmp (notes->d_buf + name_pos, "CORE", sizeof "CORE"))
442cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      {
443cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		auxv = notes->d_buf + desc_pos;
444cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		auxv_size = nhdr.n_descsz;
445cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng		break;
446cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	      }
447cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	}
448cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
449cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
450cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* Now we have NT_AUXV contents.  From here on this processing could be
451cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     used for a live process with auxv read from /proc.  */
452cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
453cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  return dwfl_link_map_report (dwfl, auxv, auxv_size,
454cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng			       dwfl_elf_phdr_memory_callback, elf);
455cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng}
456cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben ChengINTDEF (dwfl_core_file_report)
457