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