1da0c48c4Sopenharmony_ci/* Function return value location for SPARC.
2da0c48c4Sopenharmony_ci   Copyright (C) 2006-2010, 2014 Red Hat, Inc.
3da0c48c4Sopenharmony_ci   This file is part of elfutils.
4da0c48c4Sopenharmony_ci
5da0c48c4Sopenharmony_ci   This file is free software; you can redistribute it and/or modify
6da0c48c4Sopenharmony_ci   it under the terms of either
7da0c48c4Sopenharmony_ci
8da0c48c4Sopenharmony_ci     * the GNU Lesser General Public License as published by the Free
9da0c48c4Sopenharmony_ci       Software Foundation; either version 3 of the License, or (at
10da0c48c4Sopenharmony_ci       your option) any later version
11da0c48c4Sopenharmony_ci
12da0c48c4Sopenharmony_ci   or
13da0c48c4Sopenharmony_ci
14da0c48c4Sopenharmony_ci     * the GNU General Public License as published by the Free
15da0c48c4Sopenharmony_ci       Software Foundation; either version 2 of the License, or (at
16da0c48c4Sopenharmony_ci       your option) any later version
17da0c48c4Sopenharmony_ci
18da0c48c4Sopenharmony_ci   or both in parallel, as here.
19da0c48c4Sopenharmony_ci
20da0c48c4Sopenharmony_ci   elfutils is distributed in the hope that it will be useful, but
21da0c48c4Sopenharmony_ci   WITHOUT ANY WARRANTY; without even the implied warranty of
22da0c48c4Sopenharmony_ci   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
23da0c48c4Sopenharmony_ci   General Public License for more details.
24da0c48c4Sopenharmony_ci
25da0c48c4Sopenharmony_ci   You should have received copies of the GNU General Public License and
26da0c48c4Sopenharmony_ci   the GNU Lesser General Public License along with this program.  If
27da0c48c4Sopenharmony_ci   not, see <http://www.gnu.org/licenses/>.  */
28da0c48c4Sopenharmony_ci
29da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H
30da0c48c4Sopenharmony_ci# include <config.h>
31da0c48c4Sopenharmony_ci#endif
32da0c48c4Sopenharmony_ci
33da0c48c4Sopenharmony_ci#include <assert.h>
34da0c48c4Sopenharmony_ci#include <dwarf.h>
35da0c48c4Sopenharmony_ci
36da0c48c4Sopenharmony_ci#define BACKEND sparc_
37da0c48c4Sopenharmony_ci#include "libebl_CPU.h"
38da0c48c4Sopenharmony_ci
39da0c48c4Sopenharmony_ci
40da0c48c4Sopenharmony_ci/* %o0, or pair %o0, %o1.  */
41da0c48c4Sopenharmony_cistatic const Dwarf_Op loc_intreg[] =
42da0c48c4Sopenharmony_ci  {
43da0c48c4Sopenharmony_ci    { .atom = DW_OP_reg8 }, { .atom = DW_OP_piece, .number = 4 },
44da0c48c4Sopenharmony_ci    { .atom = DW_OP_reg9 }, { .atom = DW_OP_piece, .number = 4 },
45da0c48c4Sopenharmony_ci  };
46da0c48c4Sopenharmony_ci#define nloc_intreg	1
47da0c48c4Sopenharmony_ci#define nloc_intregpair	4
48da0c48c4Sopenharmony_ci
49da0c48c4Sopenharmony_ci/* %f0 or pair %f0, %f1, or quad %f0..%f3.  */
50da0c48c4Sopenharmony_cistatic const Dwarf_Op loc_fpreg[] =
51da0c48c4Sopenharmony_ci  {
52da0c48c4Sopenharmony_ci    { .atom = DW_OP_regx, .number = 32 }, { .atom = DW_OP_piece, .number = 4 },
53da0c48c4Sopenharmony_ci    { .atom = DW_OP_regx, .number = 33 }, { .atom = DW_OP_piece, .number = 4 },
54da0c48c4Sopenharmony_ci    { .atom = DW_OP_regx, .number = 34 }, { .atom = DW_OP_piece, .number = 4 },
55da0c48c4Sopenharmony_ci    { .atom = DW_OP_regx, .number = 35 }, { .atom = DW_OP_piece, .number = 4 },
56da0c48c4Sopenharmony_ci  };
57da0c48c4Sopenharmony_ci#define nloc_fpreg	1
58da0c48c4Sopenharmony_ci#define nloc_fpregpair	4
59da0c48c4Sopenharmony_ci#define nloc_fpregquad	8
60da0c48c4Sopenharmony_ci
61da0c48c4Sopenharmony_ci/* The return value is a structure and is actually stored in stack space
62da0c48c4Sopenharmony_ci   passed in a hidden argument by the caller.  But, the compiler
63da0c48c4Sopenharmony_ci   helpfully returns the address of that space in %o0.  */
64da0c48c4Sopenharmony_cistatic const Dwarf_Op loc_aggregate[] =
65da0c48c4Sopenharmony_ci  {
66da0c48c4Sopenharmony_ci    { .atom = DW_OP_breg8, .number = 0 }
67da0c48c4Sopenharmony_ci  };
68da0c48c4Sopenharmony_ci#define nloc_aggregate 1
69da0c48c4Sopenharmony_ci
70da0c48c4Sopenharmony_ciint
71da0c48c4Sopenharmony_cisparc_return_value_location (Dwarf_Die *functypedie, const Dwarf_Op **locp)
72da0c48c4Sopenharmony_ci{
73da0c48c4Sopenharmony_ci  /* Start with the function's type, and get the DW_AT_type attribute,
74da0c48c4Sopenharmony_ci     which is the type of the return value.  */
75da0c48c4Sopenharmony_ci  Dwarf_Die die_mem, *typedie = &die_mem;
76da0c48c4Sopenharmony_ci  int tag = dwarf_peeled_die_type (functypedie, typedie);
77da0c48c4Sopenharmony_ci  if (tag <= 0)
78da0c48c4Sopenharmony_ci    return tag;
79da0c48c4Sopenharmony_ci
80da0c48c4Sopenharmony_ci  Dwarf_Word size;
81da0c48c4Sopenharmony_ci  switch (tag)
82da0c48c4Sopenharmony_ci    {
83da0c48c4Sopenharmony_ci    case -1:
84da0c48c4Sopenharmony_ci      return -1;
85da0c48c4Sopenharmony_ci
86da0c48c4Sopenharmony_ci    case DW_TAG_subrange_type:
87da0c48c4Sopenharmony_ci      if (! dwarf_hasattr_integrate (typedie, DW_AT_byte_size))
88da0c48c4Sopenharmony_ci	{
89da0c48c4Sopenharmony_ci	  Dwarf_Attribute attr_mem, *attr;
90da0c48c4Sopenharmony_ci	  attr = dwarf_attr_integrate (typedie, DW_AT_type, &attr_mem);
91da0c48c4Sopenharmony_ci	  typedie = dwarf_formref_die (attr, &die_mem);
92da0c48c4Sopenharmony_ci	  tag = DWARF_TAG_OR_RETURN (typedie);
93da0c48c4Sopenharmony_ci	}
94da0c48c4Sopenharmony_ci      FALLTHROUGH;
95da0c48c4Sopenharmony_ci
96da0c48c4Sopenharmony_ci    case DW_TAG_base_type:
97da0c48c4Sopenharmony_ci    case DW_TAG_enumeration_type:
98da0c48c4Sopenharmony_ci    case DW_TAG_pointer_type:
99da0c48c4Sopenharmony_ci    case DW_TAG_ptr_to_member_type:
100da0c48c4Sopenharmony_ci      {
101da0c48c4Sopenharmony_ci	Dwarf_Attribute attr_mem;
102da0c48c4Sopenharmony_ci	if (dwarf_formudata (dwarf_attr_integrate (typedie, DW_AT_byte_size,
103da0c48c4Sopenharmony_ci						   &attr_mem), &size) != 0)
104da0c48c4Sopenharmony_ci	  {
105da0c48c4Sopenharmony_ci	    uint8_t asize;
106da0c48c4Sopenharmony_ci	    Dwarf_Die cudie;
107da0c48c4Sopenharmony_ci	    if ((tag == DW_TAG_pointer_type || tag == DW_TAG_ptr_to_member_type)
108da0c48c4Sopenharmony_ci		&& dwarf_diecu (typedie, &cudie, &asize, NULL) != NULL)
109da0c48c4Sopenharmony_ci	      size = asize;
110da0c48c4Sopenharmony_ci	    else
111da0c48c4Sopenharmony_ci	      return -1;
112da0c48c4Sopenharmony_ci	  }
113da0c48c4Sopenharmony_ci      }
114da0c48c4Sopenharmony_ci
115da0c48c4Sopenharmony_ci      if (tag == DW_TAG_base_type)
116da0c48c4Sopenharmony_ci	{
117da0c48c4Sopenharmony_ci	  Dwarf_Attribute attr_mem;
118da0c48c4Sopenharmony_ci	  Dwarf_Word encoding;
119da0c48c4Sopenharmony_ci	  if (dwarf_formudata (dwarf_attr_integrate (typedie, DW_AT_encoding,
120da0c48c4Sopenharmony_ci						     &attr_mem),
121da0c48c4Sopenharmony_ci			       &encoding) != 0)
122da0c48c4Sopenharmony_ci	    return -1;
123da0c48c4Sopenharmony_ci	  if (encoding == DW_ATE_float)
124da0c48c4Sopenharmony_ci	    {
125da0c48c4Sopenharmony_ci	      *locp = loc_fpreg;
126da0c48c4Sopenharmony_ci	      if (size <= 4)
127da0c48c4Sopenharmony_ci		return nloc_fpreg;
128da0c48c4Sopenharmony_ci	      if (size <= 8)
129da0c48c4Sopenharmony_ci		return nloc_fpregpair;
130da0c48c4Sopenharmony_ci	      if (size <= 16)
131da0c48c4Sopenharmony_ci		return nloc_fpregquad;
132da0c48c4Sopenharmony_ci	    }
133da0c48c4Sopenharmony_ci	}
134da0c48c4Sopenharmony_ci      if (size <= 8)
135da0c48c4Sopenharmony_ci	{
136da0c48c4Sopenharmony_ci	intreg:
137da0c48c4Sopenharmony_ci	  *locp = loc_intreg;
138da0c48c4Sopenharmony_ci	  return size <= 4 ? nloc_intreg : nloc_intregpair;
139da0c48c4Sopenharmony_ci	}
140da0c48c4Sopenharmony_ci
141da0c48c4Sopenharmony_ci    aggregate:
142da0c48c4Sopenharmony_ci      *locp = loc_aggregate;
143da0c48c4Sopenharmony_ci      return nloc_aggregate;
144da0c48c4Sopenharmony_ci
145da0c48c4Sopenharmony_ci    case DW_TAG_structure_type:
146da0c48c4Sopenharmony_ci    case DW_TAG_class_type:
147da0c48c4Sopenharmony_ci    case DW_TAG_union_type:
148da0c48c4Sopenharmony_ci    case DW_TAG_array_type:
149da0c48c4Sopenharmony_ci      if (dwarf_aggregate_size (typedie, &size) == 0
150da0c48c4Sopenharmony_ci	  && size > 0 && size <= 8)
151da0c48c4Sopenharmony_ci	goto intreg;
152da0c48c4Sopenharmony_ci      goto aggregate;
153da0c48c4Sopenharmony_ci    }
154da0c48c4Sopenharmony_ci
155da0c48c4Sopenharmony_ci  /* XXX We don't have a good way to return specific errors from ebl calls.
156da0c48c4Sopenharmony_ci     This value means we do not understand the type, but it is well-formed
157da0c48c4Sopenharmony_ci     DWARF and might be valid.  */
158da0c48c4Sopenharmony_ci  return -2;
159da0c48c4Sopenharmony_ci}
160