1 // aarch64-reloc-property.h -- AArch64 relocation properties -*- C++ -*- 2 3 // Copyright (C) 2014 Free Software Foundation, Inc. 4 // Written by Han Shen <shenhan@google.com> and Jing Yu <jingyu@google.com>. 5 6 // This file is part of gold. 7 8 // This program is free software; you can redistribute it and/or modify 9 // it under the terms of the GNU General Public License as published by 10 // the Free Software Foundation; either version 3 of the License, or 11 // (at your option) any later version. 12 13 // This program is distributed in the hope that it will be useful, 14 // but WITHOUT ANY WARRANTY; without even the implied warranty of 15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 16 // GNU General Public License for more details. 17 18 // You should have received a copy of the GNU General Public License 19 // along with this program; if not, write to the Free Software 20 // Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, 21 // MA 02110-1301, USA. 22 23 #ifndef GOLD_AARCH64_RELOC_PROPERTY_H 24 #define GOLD_AARCH64_RELOC_PROPERTY_H 25 26 #include<vector> 27 #include<string> 28 29 #include"aarch64.h" 30 31 namespace gold 32 { 33 // The AArch64_reloc_property class is to store information about a particular 34 // relocation code. 35 36 class AArch64_reloc_property 37 { 38 public: 39 // Types of relocation codes. 40 enum Reloc_type { 41 RT_NONE, // No relocation type. 42 RT_STATIC, // Relocations processed by static linkers. 43 RT_DYNAMIC, // Relocations processed by dynamic linkers. 44 }; 45 46 // Classes of relocation codes. 47 enum Reloc_class { 48 RC_NONE, // No relocation class. 49 RC_DATA, // Data relocation. 50 RC_AARCH64, // Static AArch64 relocations 51 RC_CFLOW, // Control flow 52 RC_TLS, // Thread local storage 53 RC_DYNAMIC, // Dynamic relocation 54 }; 55 56 // Instructions that are associated with relocations. 57 enum Reloc_inst { 58 INST_DATA = 0, 59 INST_MOVW = 1, // movz, movk, movn 60 INST_LD = 2, // ld literal 61 INST_ADR = 3, // adr 62 INST_ADRP = 4, // adrp 63 INST_ADD = 5, // add 64 INST_LDST = 6, // ld/st 65 INST_TBZNZ = 7, // tbz/tbnz 66 INST_CONDB = 8, // B.cond 67 INST_B = 9, // b [25:0] 68 INST_CALL = 10, // bl [25:0] 69 INST_NUM = 11, // total number of entries in the table 70 }; 71 72 // Types of bases of relative addressing relocation codes. 73 // enum Relative_address_base { 74 // RAB_NONE, // Relocation is not relative addressing 75 // }; 76 77 typedef bool (*rvalue_checkup_func_p)(int64_t); 78 typedef uint64_t (*rvalue_bit_select_func)(uint64_t); 79 80 // Relocation code represented by this. 81 unsigned int code()82 code() const 83 { return this->code_; } 84 85 // Name of the relocation code. 86 const std::string& name()87 name() const 88 { return this->name_; } 89 90 // Type of relocation code. 91 Reloc_type reloc_type()92 reloc_type() const 93 { return this->reloc_type_; } 94 95 // Class of relocation code. 96 Reloc_class reloc_class()97 reloc_class() const 98 { return this->reloc_class_; } 99 100 // Whether this code is implemented in gold. 101 bool is_implemented()102 is_implemented() const 103 { return this->is_implemented_; } 104 105 // If code is a group relocation code, return the group number, otherwise -1. 106 int group_index()107 group_index() const 108 { return this->group_index_; } 109 110 // Return alignment of relocation. 111 size_t align()112 align() const 113 { return this->align_; } 114 115 int reference_flags()116 reference_flags() const 117 { return this->reference_flags_; } 118 119 // Instruction associated with this relocation. 120 Reloc_inst reloc_inst()121 reloc_inst() const 122 { return this->reloc_inst_; } 123 124 // Check overflow of x checkup_x_value(int64_t x)125 bool checkup_x_value(int64_t x) const 126 { return this->rvalue_checkup_func_(x); } 127 128 // Return portions of x as is defined in aarch64-reloc.def. select_x_value(uint64_t x)129 uint64_t select_x_value(uint64_t x) const 130 { return this->rvalue_bit_select_func_(x); } 131 132 protected: 133 // These are protected. We only allow AArch64_reloc_property_table to 134 // manage AArch64_reloc_property. 135 AArch64_reloc_property(unsigned int code, const char* name, Reloc_type rtype, 136 Reloc_class rclass, 137 bool is_implemented, 138 int group_index, 139 int reference_flags, 140 Reloc_inst reloc_inst, 141 rvalue_checkup_func_p rvalue_checkup_func, 142 rvalue_bit_select_func rvalue_bit_select); 143 144 friend class AArch64_reloc_property_table; 145 146 private: 147 // Copying is not allowed. 148 AArch64_reloc_property(const AArch64_reloc_property&); 149 AArch64_reloc_property& operator=(const AArch64_reloc_property&); 150 151 // Relocation code. 152 const unsigned int code_; 153 // Relocation name. 154 const std::string name_; 155 // Type of relocation. 156 Reloc_type reloc_type_; 157 // Class of relocation. 158 Reloc_class reloc_class_; 159 // Group index (0, 1, or 2) if this is a group relocation or -1 otherwise. 160 int group_index_; 161 // Size of relocation. 162 size_t size_; 163 // Alignment of relocation. 164 size_t align_; 165 // Relative address base. 166 // Relative_address_base relative_address_base_; 167 // Whether this is deprecated. 168 bool is_deprecated_ : 1; 169 // Whether this is implemented in gold. 170 bool is_implemented_ : 1; 171 // Whether this checks overflow. 172 bool checks_overflow_ : 1; 173 const int reference_flags_; 174 // Instruction associated with relocation. 175 Reloc_inst reloc_inst_; 176 rvalue_checkup_func_p rvalue_checkup_func_; 177 rvalue_bit_select_func rvalue_bit_select_func_; 178 }; 179 180 class AArch64_reloc_property_table 181 { 182 public: 183 AArch64_reloc_property_table(); 184 185 const AArch64_reloc_property* get_reloc_property(unsigned int code)186 get_reloc_property(unsigned int code) const 187 { 188 unsigned int idx = code_to_array_index(code); 189 return this->table_[idx]; 190 } 191 192 // Like get_reloc_property but only return non-NULL if relocation code is 193 // static and implemented. 194 const AArch64_reloc_property* get_implemented_static_reloc_property(unsigned int code)195 get_implemented_static_reloc_property(unsigned int code) const 196 { 197 unsigned int idx = code_to_array_index(code); 198 const AArch64_reloc_property* arp = this->table_[idx]; 199 return ((arp != NULL 200 && (arp->reloc_type() == AArch64_reloc_property::RT_STATIC) 201 && arp->is_implemented()) 202 ? arp 203 : NULL); 204 } 205 206 // Return a string describing the relocation code that is not 207 // an implemented static reloc code. 208 std::string 209 reloc_name_in_error_message(unsigned int code); 210 211 private: 212 // Copying is not allowed. 213 AArch64_reloc_property_table(const AArch64_reloc_property_table&); 214 AArch64_reloc_property_table& operator=(const AArch64_reloc_property_table&); 215 216 // Map aarch64 rtypes into range(0,300) as following 217 // 256 ~ 313 -> 0 ~ 57 218 // 512 ~ 573 -> 128 ~ 189 219 int code_to_array_index(unsigned int code)220 code_to_array_index(unsigned int code) const 221 { 222 if (code == 0) return 0; 223 if (!((code >= elfcpp::R_AARCH64_ABS64 && 224 code <= elfcpp::R_AARCH64_LD64_GOTPAGE_LO15) 225 || (code >= elfcpp::R_AARCH64_TLSGD_ADR_PREL21 && 226 code <= elfcpp::R_AARCH64_TLSLD_LDST128_DTPREL_LO12_NC))) 227 { 228 gold_error(_("Invalid/unrecognized reloc reloc %d."), code); 229 } 230 unsigned int rv = -1; 231 if (code & (1 << 9)) 232 rv = 128 + code - 512; // 512 - 573 233 else if (code & (1 << 8)) 234 rv = code - 256; // 256 - 313 235 gold_assert(rv <= Property_table_size); 236 return rv; 237 } 238 239 static const unsigned int Property_table_size = 300; 240 AArch64_reloc_property* table_[Property_table_size]; 241 }; // End of class AArch64_reloc_property_table 242 243 } // End namespace gold. 244 245 #endif // !defined(GOLD_AARCH64_RELOC_PROPERTY_H) 246