1441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project/* Find line information for address.
2cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Copyright (C) 2004, 2005 Red Hat, Inc.
3cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   This file is part of Red Hat elfutils.
4441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project   Written by Ulrich Drepper <drepper@redhat.com>, 2004.
5441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
6cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat elfutils is free software; you can redistribute it and/or modify
7cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   it under the terms of the GNU General Public License as published by the
8cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Free Software Foundation; version 2 of the License.
9441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
10cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat elfutils is distributed in the hope that it will be useful, but
11cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   WITHOUT ANY WARRANTY; without even the implied warranty of
12cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   General Public License for more details.
14cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
15cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   You should have received a copy of the GNU General Public License along
16cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   with Red Hat elfutils; if not, write to the Free Software Foundation,
17cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301 USA.
18cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
19cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   In addition, as a special exception, Red Hat, Inc. gives You the
20cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   additional right to link the code of Red Hat elfutils with code licensed
21cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   under any Open Source Initiative certified open source license
22cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   (http://www.opensource.org/licenses/index.php) which requires the
23cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   distribution of source code with any binary distribution and to
24cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   distribute linked combinations of the two.  Non-GPL Code permitted under
25cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   this exception must only link to the code of Red Hat elfutils through
26cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   those well defined interfaces identified in the file named EXCEPTION
27cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   found in the source code files (the "Approved Interfaces").  The files
28cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   of Non-GPL Code may instantiate templates or use macros or inline
29cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   functions from the Approved Interfaces without causing the resulting
30cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   work to be covered by the GNU General Public License.  Only Red Hat,
31cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Inc. may make changes or additions to the list of Approved Interfaces.
32cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat's grant of this exception is conditioned upon your not adding
33cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   any new exceptions.  If you wish to add a new Approved Interface or
34cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   exception, please contact Red Hat.  You must obey the GNU General Public
35cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   License in all respects for all of the Red Hat elfutils code and other
36cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   code used in conjunction with Red Hat elfutils except the Non-GPL Code
37cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   covered by this exception.  If you modify this file, you may extend this
38cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   exception to your version of the file, but you are not obligated to do
39cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   so.  If you do not wish to provide this exception without modification,
40cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   you must delete this exception statement from your version and license
41cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   this file solely under the GPL without exception.
42cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
43cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Red Hat elfutils is an included package of the Open Invention Network.
44cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   An included package of the Open Invention Network is a package for which
45cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Open Invention Network licensees cross-license their patents.  No patent
46cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   license is granted, either expressly or impliedly, by designation as an
47cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   included package.  Should you wish to participate in the Open Invention
48cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   Network licensing program, please visit www.openinventionnetwork.com
49cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng   <http://www.openinventionnetwork.com>.  */
50441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
51441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project#ifdef HAVE_CONFIG_H
52441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project# include <config.h>
53441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project#endif
54441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
55441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project#include "libdwP.h"
56cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng#include <assert.h>
57441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
58441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
59441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source ProjectDwarf_Line *
60441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Projectdwarf_getsrc_die (Dwarf_Die *cudie, Dwarf_Addr addr)
61441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project{
62441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project  Dwarf_Lines *lines;
63441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project  size_t nlines;
64441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
65cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (INTUSE(dwarf_getsrclines) (cudie, &lines, &nlines) != 0)
66441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project    return NULL;
67441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
68cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* The lines are sorted by address, so we can use binary search.  */
69cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  size_t l = 0, u = nlines;
70cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  while (l < u)
71cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
72cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      size_t idx = (l + u) / 2;
73cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (addr < lines->info[idx].addr)
74cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	u = idx;
75cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      else if (addr > lines->info[idx].addr || lines->info[idx].end_sequence)
76cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	l = idx + 1;
77cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      else
78cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	return &lines->info[idx];
79cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
80cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
81cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (nlines > 0)
82cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    assert (lines->info[nlines - 1].end_sequence);
83cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng
84cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  /* If none were equal, the closest one below is what we want.  We
85cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     never want the last one, because it's the end-sequence marker
86cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     with an address at the high bound of the CU's code.  If the debug
87cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     information is faulty and no end-sequence marker is present, we
88cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng     still ignore it.  */
89cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng  if (u > 0 && u < nlines && addr > lines->info[u - 1].addr)
90cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    {
91cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      while (lines->info[u - 1].end_sequence && u > 0)
92cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	--u;
93cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng      if (u > 0)
94cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng	return &lines->info[u - 1];
95cc6695e2684ce93cdf8bd2da63d55d2cf49ff076Ben Cheng    }
96441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project
97441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project  __libdw_seterrno (DWARF_E_ADDR_OUTOFRANGE);
98441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project  return NULL;
99441f72d43a9b550baa779fc82f70816da5f74f0eThe Android Open Source Project}
100