1 // arm-reloc-property.h -- ARM relocation properties -*- C++ -*- 2 3 // Copyright (C) 2010-2014 Free Software Foundation, Inc. 4 // Written by Doug Kwan <dougkwan@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_ARM_RELOC_PROPERTY_H 24 #define GOLD_ARM_RELOC_PROPERTY_H 25 26 namespace gold 27 { 28 // The Arm_reloc_property class is to store information about a particular 29 // relocation code. 30 31 class Arm_reloc_property 32 { 33 public: 34 // Types of relocation codes. 35 enum Reloc_type { 36 RT_NONE, // No relocation type. 37 RT_STATIC, // Relocations processed by static linkers. 38 RT_DYNAMIC, // Relocations processed by dynamic linkers. 39 RT_PRIVATE, // Private relocations, not supported by gold. 40 RT_OBSOLETE // Obsolete relocations that should not be used. 41 }; 42 43 // Classes of relocation codes. 44 enum Reloc_class { 45 RC_NONE, // No relocation class. 46 RC_DATA, // Data relocation. 47 RC_ARM, // ARM instruction relocation. 48 RC_THM16, // 16-bit THUMB instruction relocation. 49 RC_THM32, // 32-bit THUMB instruction relocation. 50 RC_MISC // Miscellaneous class. 51 }; 52 53 // Types of bases of relative addressing relocation codes. 54 enum Relative_address_base { 55 RAB_NONE, // Relocation is not relative addressing 56 RAB_B_S, // Address origin of output segment of defining symbol. 57 RAB_DELTA_B_S, // Change of address origin. 58 RAB_GOT_ORG, // Origin of GOT. 59 RAB_P, // Address of the place being relocated. 60 RAB_Pa, // Adjusted address (P & 0xfffffffc). 61 RAB_TLS, // Thread local storage. 62 RAB_tp // Thread pointer. 63 }; 64 65 // Relocation code represented by this. 66 unsigned int code()67 code() const 68 { return this->code_; } 69 70 // Name of the relocation code. 71 const std::string& name()72 name() const 73 { return this->name_; } 74 75 // Type of relocation code. 76 Reloc_type reloc_type()77 reloc_type() const 78 { return this->reloc_type_; } 79 80 // Whether this code is deprecated. 81 bool is_deprecated()82 is_deprecated() const 83 { return this->is_deprecated_; } 84 85 // Class of relocation code. 86 Reloc_class reloc_class()87 reloc_class() const 88 { return this->reloc_class_; } 89 90 // Whether this code is implemented in gold. 91 bool is_implemented()92 is_implemented() const 93 { return this->is_implemented_; } 94 95 // If code is a group relocation code, return the group number, otherwise -1. 96 int group_index()97 group_index() const 98 { return this->group_index_; } 99 100 // Whether relocation checks for overflow. 101 bool checks_overflow()102 checks_overflow() const 103 { return this->checks_overflow_; } 104 105 // Return size of relocation. 106 size_t size()107 size() const 108 { return this->size_; } 109 110 // Return alignment of relocation. 111 size_t align()112 align() const 113 { return this->align_; } 114 115 // Whether relocation use a GOT entry. 116 bool uses_got_entry()117 uses_got_entry() const 118 { return this->uses_got_entry_; } 119 120 // Whether relocation use a GOT origin. 121 bool uses_got_origin()122 uses_got_origin() const 123 { return this->uses_got_origin_; } 124 125 // Whether relocation uses the Thumb-bit in a symbol address. 126 bool uses_thumb_bit()127 uses_thumb_bit() const 128 { return this->uses_thumb_bit_; } 129 130 // Whether relocation uses the symbol base. 131 bool uses_symbol_base()132 uses_symbol_base() const 133 { return this->uses_symbol_base_; } 134 135 // Whether relocation uses the symbol. 136 bool uses_symbol()137 uses_symbol() const 138 { return this->uses_symbol_; } 139 140 // Return the type of relative address base or RAB_NONE if this 141 // is not a relative addressing relocation. 142 Relative_address_base relative_address_base()143 relative_address_base() const 144 { return this->relative_address_base_; } 145 146 protected: 147 // These are protected. We only allow Arm_reloc_property_table to 148 // manage Arm_reloc_property. 149 Arm_reloc_property(unsigned int code, const char* name, Reloc_type rtype, 150 bool is_deprecated, Reloc_class rclass, 151 const std::string& operation, bool is_implemented, 152 int group_index, bool checks_overflow); 153 154 friend class Arm_reloc_property_table; 155 156 private: 157 // Copying is not allowed. 158 Arm_reloc_property(const Arm_reloc_property&); 159 Arm_reloc_property& operator=(const Arm_reloc_property&); 160 161 // The Tree_node class is used to represent parsed relocation operations. 162 // We look at Trees to extract information about relocation operations. 163 class Tree_node 164 { 165 public: 166 typedef std::vector<Tree_node*> Tree_node_vector; 167 168 // Construct a leaf node. Tree_node(const char * name)169 Tree_node(const char* name) 170 : is_leaf_(true), name_(name), children_() 171 { } 172 173 // Construct an internal node. A node owns all its children and is 174 // responsible for releasing them at its own destruction. Tree_node(Tree_node_vector::const_iterator begin,Tree_node_vector::const_iterator end)175 Tree_node(Tree_node_vector::const_iterator begin, 176 Tree_node_vector::const_iterator end) 177 : is_leaf_(false), name_(), children_() 178 { 179 for (Tree_node_vector::const_iterator p = begin; p != end; ++p) 180 this->children_.push_back(*p); 181 } 182 ~Tree_node()183 ~Tree_node() 184 { 185 for(size_t i = 0; i <this->children_.size(); ++i) 186 delete this->children_[i]; 187 } 188 189 // Whether this is a leaf node. 190 bool is_leaf()191 is_leaf() const 192 { return this->is_leaf_; } 193 194 // Return name of this. This is only valid for a leaf node. 195 const std::string& name()196 name() const 197 { 198 gold_assert(this->is_leaf_); 199 return this->name_; 200 } 201 202 // Return the number of children. This is only valid for a non-leaf node. 203 size_t number_of_children()204 number_of_children() const 205 { 206 gold_assert(!this->is_leaf_); 207 return this->children_.size(); 208 } 209 210 // Return the i-th child of this. This is only valid for a non-leaf node. 211 Tree_node* child(size_t i)212 child(size_t i) const 213 { 214 gold_assert(!this->is_leaf_ && i < this->children_.size()); 215 return this->children_[i]; 216 } 217 218 // Parse an S-expression string and build a tree and return the root node. 219 // Caller is responsible for releasing tree after use. 220 static Tree_node* 221 make_tree(const std::string&); 222 223 // Convert a tree back to an S-expression string. 224 std::string s_expression()225 s_expression() const 226 { 227 if (this->is_leaf_) 228 return this->name_; 229 230 // Concatenate S-expressions of children. Enclose them with 231 // a pair of parentheses and use space as token delimiters. 232 std::string s("("); 233 for(size_t i = 0; i <this->children_.size(); ++i) 234 s = s + " " + this->children_[i]->s_expression(); 235 return s + " )"; 236 } 237 238 private: 239 // Whether this is a leaf node. 240 bool is_leaf_; 241 // Name of this if this is a leaf node. 242 std::string name_; 243 // Children of this if this a non-leaf node. 244 Tree_node_vector children_; 245 }; 246 247 // Relocation code. 248 unsigned int code_; 249 // Relocation name. 250 std::string name_; 251 // Type of relocation. 252 Reloc_type reloc_type_; 253 // Class of relocation. 254 Reloc_class reloc_class_; 255 // Group index (0, 1, or 2) if this is a group relocation or -1 otherwise. 256 int group_index_; 257 // Size of relocation. 258 size_t size_; 259 // Alignment of relocation. 260 size_t align_; 261 // Relative address base. 262 Relative_address_base relative_address_base_; 263 // Whether this is deprecated. 264 bool is_deprecated_ : 1; 265 // Whether this is implemented in gold. 266 bool is_implemented_ : 1; 267 // Whether this checks overflow. 268 bool checks_overflow_ : 1; 269 // Whether this uses a GOT entry. 270 bool uses_got_entry_ : 1; 271 // Whether this uses a GOT origin. 272 bool uses_got_origin_ : 1; 273 // Whether this uses a PLT entry. 274 bool uses_plt_entry_ : 1; 275 // Whether this uses the THUMB bit in symbol address. 276 bool uses_thumb_bit_ : 1; 277 // Whether this uses the symbol base. 278 bool uses_symbol_base_ : 1; 279 // Whether this uses an addend. 280 bool uses_addend_ : 1; 281 // Whether this uses the symbol. 282 bool uses_symbol_ : 1; 283 }; 284 285 // Arm_reloc_property_table. This table is used for looking up properties 286 // of relocation types. The table entries are initialized using information 287 // from arm-reloc.def. 288 289 class Arm_reloc_property_table 290 { 291 public: 292 Arm_reloc_property_table(); 293 294 // Return an Arm_reloc_property object for CODE if it is a valid relocation 295 // code or NULL otherwise. 296 const Arm_reloc_property* get_reloc_property(unsigned int code)297 get_reloc_property(unsigned int code) const 298 { 299 gold_assert(code < Property_table_size); 300 return this->table_[code]; 301 } 302 303 // Like get_reloc_property but only return non-NULL if relocation code is 304 // static and implemented. 305 const Arm_reloc_property* get_implemented_static_reloc_property(unsigned int code)306 get_implemented_static_reloc_property(unsigned int code) const 307 { 308 gold_assert(code < Property_table_size); 309 const Arm_reloc_property* arp = this->table_[code]; 310 return ((arp != NULL 311 && (arp->reloc_type() == Arm_reloc_property::RT_STATIC) 312 && arp->is_implemented()) 313 ? arp 314 : NULL); 315 } 316 317 // Return a string describing the relocation code that is not 318 // an implemented static reloc code. 319 std::string 320 reloc_name_in_error_message(unsigned int code); 321 322 private: 323 // Copying is not allowed. 324 Arm_reloc_property_table(const Arm_reloc_property_table&); 325 Arm_reloc_property_table& operator=(const Arm_reloc_property_table&); 326 327 // The Parse_expression class is used to convert relocation operations in 328 // arm-reloc.def into S-expression strings, which are parsed again to 329 // build actual expression trees. We do not build the expression trees 330 // directly because the parser for operations in arm-reloc.def is simpler 331 // this way. Conversion from S-expressions to trees is simple. 332 class Parse_expression 333 { 334 public: 335 // Construction a Parse_expression with an S-expression string. Parse_expression(const std::string & s_expression)336 Parse_expression(const std::string& s_expression) 337 : s_expression_(s_expression) 338 { } 339 340 // Value of this expression as an S-expression string. 341 const std::string& s_expression()342 s_expression() const 343 { return this->s_expression_; } 344 345 // We want to overload operators used in relocation operations so 346 // that we can execute operations in arm-reloc.def to generate 347 // S-expressions directly. 348 #define DEF_OPERATOR_OVERLOAD(op) \ 349 Parse_expression \ 350 operator op (const Parse_expression& e) \ 351 { \ 352 return Parse_expression("( " #op " " + this->s_expression_ + " " + \ 353 e.s_expression_ + " )"); \ 354 } 355 356 // Operator appearing in relocation operations in arm-reloc.def. 357 DEF_OPERATOR_OVERLOAD(+) 358 DEF_OPERATOR_OVERLOAD(-) 359 DEF_OPERATOR_OVERLOAD(|) 360 361 private: 362 // This represented as an S-expression string. 363 std::string s_expression_; 364 }; 365 366 #define DEF_RELOC_FUNC(name) \ 367 static Parse_expression \ 368 (name)(const Parse_expression& arg) \ 369 { return Parse_expression("( " #name " " + arg.s_expression() + " )"); } 370 371 // Functions appearing in relocation operations in arm-reloc.def. 372 DEF_RELOC_FUNC(B) 373 DEF_RELOC_FUNC(DELTA_B) 374 DEF_RELOC_FUNC(GOT) 375 DEF_RELOC_FUNC(Module) 376 DEF_RELOC_FUNC(PLT) 377 378 static const unsigned int Property_table_size = 256; 379 380 // The property table. 381 Arm_reloc_property* table_[Property_table_size]; 382 }; 383 384 } // End namespace gold. 385 386 #endif // !defined(GOLD_ARM_RELOC_PROPERTY_H) 387