1da0c48c4Sopenharmony_ci/* Return block represented by attribute.
2da0c48c4Sopenharmony_ci   Copyright (C) 2004-2010, 2014, 2018 Red Hat, Inc.
3da0c48c4Sopenharmony_ci   This file is part of elfutils.
4da0c48c4Sopenharmony_ci   Written by Ulrich Drepper <drepper@redhat.com>, 2004.
5da0c48c4Sopenharmony_ci
6da0c48c4Sopenharmony_ci   This file is free software; you can redistribute it and/or modify
7da0c48c4Sopenharmony_ci   it under the terms of either
8da0c48c4Sopenharmony_ci
9da0c48c4Sopenharmony_ci     * the GNU Lesser General Public License as published by the Free
10da0c48c4Sopenharmony_ci       Software Foundation; either version 3 of the License, or (at
11da0c48c4Sopenharmony_ci       your option) any later version
12da0c48c4Sopenharmony_ci
13da0c48c4Sopenharmony_ci   or
14da0c48c4Sopenharmony_ci
15da0c48c4Sopenharmony_ci     * the GNU General Public License as published by the Free
16da0c48c4Sopenharmony_ci       Software Foundation; either version 2 of the License, or (at
17da0c48c4Sopenharmony_ci       your option) any later version
18da0c48c4Sopenharmony_ci
19da0c48c4Sopenharmony_ci   or both in parallel, as here.
20da0c48c4Sopenharmony_ci
21da0c48c4Sopenharmony_ci   elfutils is distributed in the hope that it will be useful, but
22da0c48c4Sopenharmony_ci   WITHOUT ANY WARRANTY; without even the implied warranty of
23da0c48c4Sopenharmony_ci   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24da0c48c4Sopenharmony_ci   General Public License for more details.
25da0c48c4Sopenharmony_ci
26da0c48c4Sopenharmony_ci   You should have received copies of the GNU General Public License and
27da0c48c4Sopenharmony_ci   the GNU Lesser General Public License along with this program.  If
28da0c48c4Sopenharmony_ci   not, see <http://www.gnu.org/licenses/>.  */
29da0c48c4Sopenharmony_ci
30da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H
31da0c48c4Sopenharmony_ci# include <config.h>
32da0c48c4Sopenharmony_ci#endif
33da0c48c4Sopenharmony_ci
34da0c48c4Sopenharmony_ci#include <dwarf.h>
35da0c48c4Sopenharmony_ci#include "libdwP.h"
36da0c48c4Sopenharmony_ci
37da0c48c4Sopenharmony_ci
38da0c48c4Sopenharmony_ciint
39da0c48c4Sopenharmony_cidwarf_formblock (Dwarf_Attribute *attr, Dwarf_Block *return_block)
40da0c48c4Sopenharmony_ci{
41da0c48c4Sopenharmony_ci  if (attr == NULL)
42da0c48c4Sopenharmony_ci    return -1;
43da0c48c4Sopenharmony_ci
44da0c48c4Sopenharmony_ci  const unsigned char *datap = attr->valp;
45da0c48c4Sopenharmony_ci  const unsigned char *endp = attr->cu->endp;
46da0c48c4Sopenharmony_ci
47da0c48c4Sopenharmony_ci  switch (attr->form)
48da0c48c4Sopenharmony_ci    {
49da0c48c4Sopenharmony_ci    case DW_FORM_block1:
50da0c48c4Sopenharmony_ci      if (unlikely (endp - datap < 1))
51da0c48c4Sopenharmony_ci	goto invalid;
52da0c48c4Sopenharmony_ci      return_block->length = *(uint8_t *) attr->valp;
53da0c48c4Sopenharmony_ci      return_block->data = attr->valp + 1;
54da0c48c4Sopenharmony_ci      break;
55da0c48c4Sopenharmony_ci
56da0c48c4Sopenharmony_ci    case DW_FORM_block2:
57da0c48c4Sopenharmony_ci      if (unlikely (endp - datap < 2))
58da0c48c4Sopenharmony_ci	goto invalid;
59da0c48c4Sopenharmony_ci      return_block->length = read_2ubyte_unaligned (attr->cu->dbg, attr->valp);
60da0c48c4Sopenharmony_ci      return_block->data = attr->valp + 2;
61da0c48c4Sopenharmony_ci      break;
62da0c48c4Sopenharmony_ci
63da0c48c4Sopenharmony_ci    case DW_FORM_block4:
64da0c48c4Sopenharmony_ci      if (unlikely (endp - datap < 4))
65da0c48c4Sopenharmony_ci	goto invalid;
66da0c48c4Sopenharmony_ci      return_block->length = read_4ubyte_unaligned (attr->cu->dbg, attr->valp);
67da0c48c4Sopenharmony_ci      return_block->data = attr->valp + 4;
68da0c48c4Sopenharmony_ci      break;
69da0c48c4Sopenharmony_ci
70da0c48c4Sopenharmony_ci    case DW_FORM_block:
71da0c48c4Sopenharmony_ci    case DW_FORM_exprloc:
72da0c48c4Sopenharmony_ci      if (unlikely (endp - datap < 1))
73da0c48c4Sopenharmony_ci	goto invalid;
74da0c48c4Sopenharmony_ci      get_uleb128 (return_block->length, datap, endp);
75da0c48c4Sopenharmony_ci      return_block->data = (unsigned char *) datap;
76da0c48c4Sopenharmony_ci      break;
77da0c48c4Sopenharmony_ci
78da0c48c4Sopenharmony_ci    case DW_FORM_data16:
79da0c48c4Sopenharmony_ci      /* The DWARFv5 spec calls this constant class, but we interpret
80da0c48c4Sopenharmony_ci	 it as a block that the user will need to interpret when
81da0c48c4Sopenharmony_ci	 converting to a value.  */
82da0c48c4Sopenharmony_ci      if (unlikely (endp - datap < 16))
83da0c48c4Sopenharmony_ci	goto invalid;
84da0c48c4Sopenharmony_ci      return_block->length = 16;
85da0c48c4Sopenharmony_ci      return_block->data = (unsigned char *) datap;
86da0c48c4Sopenharmony_ci      break;
87da0c48c4Sopenharmony_ci
88da0c48c4Sopenharmony_ci    default:
89da0c48c4Sopenharmony_ci      __libdw_seterrno (DWARF_E_NO_BLOCK);
90da0c48c4Sopenharmony_ci      return -1;
91da0c48c4Sopenharmony_ci    }
92da0c48c4Sopenharmony_ci
93da0c48c4Sopenharmony_ci  if (unlikely (return_block->length > (size_t) (endp - return_block->data)))
94da0c48c4Sopenharmony_ci    {
95da0c48c4Sopenharmony_ci      /* Block does not fit.  */
96da0c48c4Sopenharmony_ci    invalid:
97da0c48c4Sopenharmony_ci      __libdw_seterrno (DWARF_E_INVALID_DWARF);
98da0c48c4Sopenharmony_ci      return -1;
99da0c48c4Sopenharmony_ci    }
100da0c48c4Sopenharmony_ci
101da0c48c4Sopenharmony_ci  return 0;
102da0c48c4Sopenharmony_ci}
103da0c48c4Sopenharmony_ciINTDEF(dwarf_formblock)
104