OLD | NEW |
1 //===- subzero/src/IceGlobalInits.h - Global declarations -------*- C++ -*-===// | 1 //===- subzero/src/IceGlobalInits.h - Global declarations -------*- C++ -*-===// |
2 // | 2 // |
3 // The Subzero Code Generator | 3 // The Subzero Code Generator |
4 // | 4 // |
5 // This file is distributed under the University of Illinois Open Source | 5 // This file is distributed under the University of Illinois Open Source |
6 // License. See LICENSE.TXT for details. | 6 // License. See LICENSE.TXT for details. |
7 // | 7 // |
8 //===----------------------------------------------------------------------===// | 8 //===----------------------------------------------------------------------===// |
9 /// | 9 /// |
10 /// \file | 10 /// \file |
(...skipping 202 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
213 Initializer &operator=(const Initializer &) = delete; | 213 Initializer &operator=(const Initializer &) = delete; |
214 | 214 |
215 public: | 215 public: |
216 /// Discriminator for LLVM-style RTTI. | 216 /// Discriminator for LLVM-style RTTI. |
217 enum InitializerKind { | 217 enum InitializerKind { |
218 DataInitializerKind, | 218 DataInitializerKind, |
219 ZeroInitializerKind, | 219 ZeroInitializerKind, |
220 RelocInitializerKind | 220 RelocInitializerKind |
221 }; | 221 }; |
222 InitializerKind getKind() const { return Kind; } | 222 InitializerKind getKind() const { return Kind; } |
223 virtual ~Initializer() = default; | |
224 virtual SizeT getNumBytes() const = 0; | 223 virtual SizeT getNumBytes() const = 0; |
225 virtual void dump(Ostream &Stream) const = 0; | 224 virtual void dump(Ostream &Stream) const = 0; |
226 virtual void dumpType(Ostream &Stream) const; | 225 virtual void dumpType(Ostream &Stream) const; |
227 | 226 |
228 protected: | 227 protected: |
229 explicit Initializer(InitializerKind Kind) : Kind(Kind) {} | 228 explicit Initializer(InitializerKind Kind) : Kind(Kind) {} |
230 | 229 |
231 private: | 230 private: |
232 const InitializerKind Kind; | 231 const InitializerKind Kind; |
233 }; | 232 }; |
| 233 static_assert(std::is_trivially_destructible<Initializer>::value, |
| 234 "Initializer must be trivially destructible."); |
234 | 235 |
235 /// Models the data in a data initializer. | 236 /// Models the data in a data initializer. |
236 using DataVecType = std::vector<char>; | 237 using DataVecType = char *; |
237 | 238 |
238 /// Defines a sequence of byte values as a data initializer. | 239 /// Defines a sequence of byte values as a data initializer. |
239 class DataInitializer : public Initializer { | 240 class DataInitializer : public Initializer { |
240 DataInitializer(const DataInitializer &) = delete; | 241 DataInitializer(const DataInitializer &) = delete; |
241 DataInitializer &operator=(const DataInitializer &) = delete; | 242 DataInitializer &operator=(const DataInitializer &) = delete; |
242 | 243 |
243 public: | 244 public: |
244 template <class... Args> | 245 template <typename T, class... Args> |
245 static std::unique_ptr<DataInitializer> create(Args &&... TheArgs) { | 246 static DataInitializer *create(T *AllocOwner, Args &&... TheArgs) { |
246 return makeUnique<DataInitializer>(std::forward<Args>(TheArgs)...); | 247 return new (AllocOwner->template allocate_initializer<DataInitializer>()) |
| 248 DataInitializer(AllocOwner, std::forward<Args>(TheArgs)...); |
247 } | 249 } |
248 | 250 |
249 const DataVecType &getContents() const { return Contents; } | 251 const llvm::StringRef getContents() const { |
250 SizeT getNumBytes() const final { return Contents.size(); } | 252 return llvm::StringRef(Contents, ContentsSize); |
| 253 } |
| 254 SizeT getNumBytes() const final { return ContentsSize; } |
251 void dump(Ostream &Stream) const final; | 255 void dump(Ostream &Stream) const final; |
252 static bool classof(const Initializer *D) { | 256 static bool classof(const Initializer *D) { |
253 return D->getKind() == DataInitializerKind; | 257 return D->getKind() == DataInitializerKind; |
254 } | 258 } |
255 | 259 |
256 private: | 260 private: |
257 ENABLE_MAKE_UNIQUE; | 261 template <typename T> |
258 | 262 DataInitializer(T *AllocOwner, |
259 DataInitializer(const llvm::NaClBitcodeRecord::RecordVector &Values) | 263 const llvm::NaClBitcodeRecord::RecordVector &Values) |
260 : Initializer(DataInitializerKind), Contents(Values.size()) { | 264 : Initializer(DataInitializerKind), ContentsSize(Values.size()), |
| 265 // ugh, we should actually do new char[], but this may involve |
| 266 // implementation-specific details. Given that Contents is arena |
| 267 // allocated, and never detele[]d, just use char -- |
| 268 // AllocOwner->allocate_array will allocate a buffer with the right |
| 269 // size. |
| 270 Contents(new (AllocOwner->template allocate_initializer<char>( |
| 271 ContentsSize)) char) { |
261 for (SizeT I = 0; I < Values.size(); ++I) | 272 for (SizeT I = 0; I < Values.size(); ++I) |
262 Contents[I] = static_cast<int8_t>(Values[I]); | 273 Contents[I] = static_cast<int8_t>(Values[I]); |
263 } | 274 } |
264 | 275 |
265 DataInitializer(const char *Str, size_t StrLen) | 276 template <typename T> |
266 : Initializer(DataInitializerKind), Contents(StrLen) { | 277 DataInitializer(T *AllocOwner, const char *Str, size_t StrLen) |
| 278 : Initializer(DataInitializerKind), ContentsSize(StrLen), |
| 279 Contents(new (AllocOwner->template allocate_initializer<char>( |
| 280 ContentsSize)) char) { |
267 for (size_t i = 0; i < StrLen; ++i) | 281 for (size_t i = 0; i < StrLen; ++i) |
268 Contents[i] = Str[i]; | 282 Contents[i] = Str[i]; |
269 } | 283 } |
270 | 284 |
271 /// The byte contents of the data initializer. | 285 /// The byte contents of the data initializer. |
| 286 const SizeT ContentsSize; |
272 DataVecType Contents; | 287 DataVecType Contents; |
273 }; | 288 }; |
| 289 static_assert(std::is_trivially_destructible<DataInitializer>::value, |
| 290 "DataInitializer must be trivially destructible."); |
274 | 291 |
275 /// Defines a sequence of bytes initialized to zero. | 292 /// Defines a sequence of bytes initialized to zero. |
276 class ZeroInitializer : public Initializer { | 293 class ZeroInitializer : public Initializer { |
277 ZeroInitializer(const ZeroInitializer &) = delete; | 294 ZeroInitializer(const ZeroInitializer &) = delete; |
278 ZeroInitializer &operator=(const ZeroInitializer &) = delete; | 295 ZeroInitializer &operator=(const ZeroInitializer &) = delete; |
279 | 296 |
280 public: | 297 public: |
281 static std::unique_ptr<ZeroInitializer> create(SizeT Size) { | 298 template <typename T> |
282 return makeUnique<ZeroInitializer>(Size); | 299 static ZeroInitializer *create(T *AllocOwner, SizeT Size) { |
| 300 return new (AllocOwner->template allocate_initializer<ZeroInitializer>()) |
| 301 ZeroInitializer(Size); |
283 } | 302 } |
284 SizeT getNumBytes() const final { return Size; } | 303 SizeT getNumBytes() const final { return Size; } |
285 void dump(Ostream &Stream) const final; | 304 void dump(Ostream &Stream) const final; |
286 static bool classof(const Initializer *Z) { | 305 static bool classof(const Initializer *Z) { |
287 return Z->getKind() == ZeroInitializerKind; | 306 return Z->getKind() == ZeroInitializerKind; |
288 } | 307 } |
289 | 308 |
290 private: | 309 private: |
291 ENABLE_MAKE_UNIQUE; | |
292 | |
293 explicit ZeroInitializer(SizeT Size) | 310 explicit ZeroInitializer(SizeT Size) |
294 : Initializer(ZeroInitializerKind), Size(Size) {} | 311 : Initializer(ZeroInitializerKind), Size(Size) {} |
295 | 312 |
296 /// The number of bytes to be zero initialized. | 313 /// The number of bytes to be zero initialized. |
297 SizeT Size; | 314 SizeT Size; |
298 }; | 315 }; |
| 316 static_assert(std::is_trivially_destructible<ZeroInitializer>::value, |
| 317 "ZeroInitializer must be trivially destructible."); |
299 | 318 |
300 /// Defines the relocation value of another global declaration. | 319 /// Defines the relocation value of another global declaration. |
301 class RelocInitializer : public Initializer { | 320 class RelocInitializer : public Initializer { |
302 RelocInitializer(const RelocInitializer &) = delete; | 321 RelocInitializer(const RelocInitializer &) = delete; |
303 RelocInitializer &operator=(const RelocInitializer &) = delete; | 322 RelocInitializer &operator=(const RelocInitializer &) = delete; |
304 | 323 |
305 public: | 324 public: |
306 static std::unique_ptr<RelocInitializer> | 325 template <typename T> |
307 create(const GlobalDeclaration *Declaration, | 326 static RelocInitializer *create(T *AllocOwner, |
308 const RelocOffsetArray &OffsetExpr) { | 327 const GlobalDeclaration *Declaration, |
| 328 const RelocOffsetArray &OffsetExpr) { |
309 constexpr bool NoFixup = false; | 329 constexpr bool NoFixup = false; |
310 return makeUnique<RelocInitializer>(Declaration, OffsetExpr, NoFixup); | 330 return new (AllocOwner->template allocate_initializer<RelocInitializer>()) |
| 331 RelocInitializer(AllocOwner, Declaration, OffsetExpr, NoFixup); |
311 } | 332 } |
312 | 333 |
313 static std::unique_ptr<RelocInitializer> | 334 template <typename T> |
314 create(const GlobalDeclaration *Declaration, | 335 static RelocInitializer * |
| 336 create(T *AllocOwner, const GlobalDeclaration *Declaration, |
315 const RelocOffsetArray &OffsetExpr, FixupKind Fixup) { | 337 const RelocOffsetArray &OffsetExpr, FixupKind Fixup) { |
316 constexpr bool HasFixup = true; | 338 constexpr bool HasFixup = true; |
317 return makeUnique<RelocInitializer>(Declaration, OffsetExpr, HasFixup, | 339 return new (AllocOwner->template allocate_initializer<RelocInitializer>()) |
318 Fixup); | 340 RelocInitializer(AllocOwner, Declaration, OffsetExpr, HasFixup, |
| 341 Fixup); |
319 } | 342 } |
320 | 343 |
321 RelocOffsetT getOffset() const { | 344 RelocOffsetT getOffset() const { |
322 RelocOffsetT Offset = 0; | 345 RelocOffsetT Offset = 0; |
323 for (const auto *RelocOffset : OffsetExpr) { | 346 for (SizeT i = 0; i < OffsetExprSize; ++i) { |
324 Offset += RelocOffset->getOffset(); | 347 Offset += OffsetExpr[i]->getOffset(); |
325 } | 348 } |
326 return Offset; | 349 return Offset; |
327 } | 350 } |
328 | 351 |
329 bool hasFixup() const { return HasFixup; } | 352 bool hasFixup() const { return HasFixup; } |
330 FixupKind getFixup() const { | 353 FixupKind getFixup() const { |
331 assert(HasFixup); | 354 assert(HasFixup); |
332 return Fixup; | 355 return Fixup; |
333 } | 356 } |
334 | 357 |
335 const GlobalDeclaration *getDeclaration() const { return Declaration; } | 358 const GlobalDeclaration *getDeclaration() const { return Declaration; } |
336 SizeT getNumBytes() const final { return RelocAddrSize; } | 359 SizeT getNumBytes() const final { return RelocAddrSize; } |
337 void dump(Ostream &Stream) const final; | 360 void dump(Ostream &Stream) const final; |
338 void dumpType(Ostream &Stream) const final; | 361 void dumpType(Ostream &Stream) const final; |
339 static bool classof(const Initializer *R) { | 362 static bool classof(const Initializer *R) { |
340 return R->getKind() == RelocInitializerKind; | 363 return R->getKind() == RelocInitializerKind; |
341 } | 364 } |
342 | 365 |
343 private: | 366 private: |
344 ENABLE_MAKE_UNIQUE; | 367 template <typename T> |
345 | 368 RelocInitializer(T *AllocOwner, const GlobalDeclaration *Declaration, |
346 RelocInitializer(const GlobalDeclaration *Declaration, | |
347 const RelocOffsetArray &OffsetExpr, bool HasFixup, | 369 const RelocOffsetArray &OffsetExpr, bool HasFixup, |
348 FixupKind Fixup = 0) | 370 FixupKind Fixup = 0) |
349 : Initializer(RelocInitializerKind), | 371 : Initializer(RelocInitializerKind), |
350 Declaration(Declaration), // The global declaration used in the reloc. | 372 Declaration(Declaration), // The global declaration used in the reloc. |
351 OffsetExpr(OffsetExpr), HasFixup(HasFixup), Fixup(Fixup) {} | 373 OffsetExprSize(OffsetExpr.size()), |
| 374 OffsetExpr( |
| 375 new (AllocOwner->template allocate_initializer<RelocOffset *>( |
| 376 OffsetExprSize)) RelocOffset *), |
| 377 HasFixup(HasFixup), Fixup(Fixup) { |
| 378 for (SizeT i = 0; i < OffsetExprSize; ++i) { |
| 379 this->OffsetExpr[i] = OffsetExpr[i]; |
| 380 } |
| 381 } |
352 | 382 |
353 const GlobalDeclaration *Declaration; | 383 const GlobalDeclaration *Declaration; |
354 /// The offset to add to the relocation. | 384 /// The offset to add to the relocation. |
355 const RelocOffsetArray OffsetExpr; | 385 const SizeT OffsetExprSize; |
| 386 RelocOffset **OffsetExpr; |
356 const bool HasFixup = false; | 387 const bool HasFixup = false; |
357 const FixupKind Fixup = 0; | 388 const FixupKind Fixup = 0; |
358 }; | 389 }; |
| 390 static_assert(std::is_trivially_destructible<RelocInitializer>::value, |
| 391 "RelocInitializer must be trivially destructible."); |
359 | 392 |
360 /// Models the list of initializers. | 393 /// Models the list of initializers. |
361 using InitializerListType = std::vector<std::unique_ptr<Initializer>>; | 394 // TODO(jpp): missing allocator. |
| 395 using InitializerListType = std::vector<Initializer *>; |
362 | 396 |
363 static VariableDeclaration *create(GlobalContext *Context, | 397 template <typename T> |
| 398 static VariableDeclaration *create(T *AllocOwner, |
364 bool SuppressMangling = false, | 399 bool SuppressMangling = false, |
365 llvm::GlobalValue::LinkageTypes Linkage = | 400 llvm::GlobalValue::LinkageTypes Linkage = |
366 llvm::GlobalValue::InternalLinkage) { | 401 llvm::GlobalValue::InternalLinkage) { |
367 return new (Context->allocate<VariableDeclaration>()) | 402 return new ( |
| 403 AllocOwner |
| 404 ->template allocate_variable_declaration<VariableDeclaration>()) |
368 VariableDeclaration(Linkage, SuppressMangling); | 405 VariableDeclaration(Linkage, SuppressMangling); |
369 } | 406 } |
370 static VariableDeclaration *createExternal(GlobalContext *Context) { | 407 |
| 408 template <typename T> |
| 409 static VariableDeclaration *createExternal(T *AllocOwner) { |
371 constexpr bool SuppressMangling = true; | 410 constexpr bool SuppressMangling = true; |
372 constexpr llvm::GlobalValue::LinkageTypes Linkage = | 411 constexpr llvm::GlobalValue::LinkageTypes Linkage = |
373 llvm::GlobalValue::ExternalLinkage; | 412 llvm::GlobalValue::ExternalLinkage; |
374 return create(Context, SuppressMangling, Linkage); | 413 return create(AllocOwner, SuppressMangling, Linkage); |
375 } | 414 } |
376 | 415 |
377 const InitializerListType &getInitializers() const { return *Initializers; } | 416 const InitializerListType &getInitializers() const { return Initializers; } |
378 bool getIsConstant() const { return IsConstant; } | 417 bool getIsConstant() const { return IsConstant; } |
379 void setIsConstant(bool NewValue) { IsConstant = NewValue; } | 418 void setIsConstant(bool NewValue) { IsConstant = NewValue; } |
380 uint32_t getAlignment() const { return Alignment; } | 419 uint32_t getAlignment() const { return Alignment; } |
381 void setAlignment(uint32_t NewAlignment) { Alignment = NewAlignment; } | 420 void setAlignment(uint32_t NewAlignment) { Alignment = NewAlignment; } |
382 bool hasInitializer() const { return HasInitializer; } | 421 bool hasInitializer() const { return HasInitializer; } |
383 bool hasNonzeroInitializer() const { | 422 bool hasNonzeroInitializer() const { |
384 return !(Initializers->size() == 1 && | 423 return !(Initializers.size() == 1 && |
385 llvm::isa<ZeroInitializer>((*Initializers)[0].get())); | 424 llvm::isa<ZeroInitializer>(Initializers[0])); |
386 } | 425 } |
387 | 426 |
388 /// Returns the number of bytes for the initializer of the global address. | 427 /// Returns the number of bytes for the initializer of the global address. |
389 SizeT getNumBytes() const { | 428 SizeT getNumBytes() const { |
390 SizeT Count = 0; | 429 SizeT Count = 0; |
391 for (const std::unique_ptr<Initializer> &Init : *Initializers) { | 430 for (const auto *Init : Initializers) { |
392 Count += Init->getNumBytes(); | 431 Count += Init->getNumBytes(); |
393 } | 432 } |
394 return Count; | 433 return Count; |
395 } | 434 } |
396 | 435 |
397 /// Adds Initializer to the list of initializers. Takes ownership of the | 436 /// Adds Initializer to the list of initializers. Takes ownership of the |
398 /// initializer. | 437 /// initializer. |
399 void addInitializer(std::unique_ptr<Initializer> Initializer) { | 438 void addInitializer(Initializer *Initializer) { |
400 const bool OldSuppressMangling = getSuppressMangling(); | 439 const bool OldSuppressMangling = getSuppressMangling(); |
401 Initializers->emplace_back(std::move(Initializer)); | 440 Initializers.emplace_back(Initializer); |
402 HasInitializer = true; | 441 HasInitializer = true; |
403 // The getSuppressMangling() logic depends on whether the global variable | 442 // The getSuppressMangling() logic depends on whether the global variable |
404 // has initializers. If its value changed as a result of adding an | 443 // has initializers. If its value changed as a result of adding an |
405 // initializer, then make sure we haven't previously set the name based on | 444 // initializer, then make sure we haven't previously set the name based on |
406 // faulty SuppressMangling logic. | 445 // faulty SuppressMangling logic. |
407 const bool SameMangling = (OldSuppressMangling == getSuppressMangling()); | 446 const bool SameMangling = (OldSuppressMangling == getSuppressMangling()); |
408 (void)SameMangling; | 447 (void)SameMangling; |
409 assert(!Name.empty() || SameMangling); | 448 assert(!Name.empty() || SameMangling); |
410 } | 449 } |
411 | 450 |
(...skipping 12 matching lines...) Expand all Loading... |
424 static bool classof(const GlobalDeclaration *Addr) { | 463 static bool classof(const GlobalDeclaration *Addr) { |
425 return Addr->getKind() == VariableDeclarationKind; | 464 return Addr->getKind() == VariableDeclarationKind; |
426 } | 465 } |
427 | 466 |
428 bool getSuppressMangling() const final { | 467 bool getSuppressMangling() const final { |
429 if (ForceSuppressMangling) | 468 if (ForceSuppressMangling) |
430 return true; | 469 return true; |
431 return isExternal() && !hasInitializer(); | 470 return isExternal() && !hasInitializer(); |
432 } | 471 } |
433 | 472 |
434 void discardInitializers() { Initializers = nullptr; } | 473 void discardInitializers() { Initializers.clear(); } |
435 | 474 |
436 private: | 475 private: |
437 /// List of initializers for the declared variable. | 476 /// List of initializers for the declared variable. |
438 std::unique_ptr<InitializerListType> Initializers; | 477 InitializerListType Initializers; |
439 bool HasInitializer = false; | 478 bool HasInitializer = false; |
440 /// The alignment of the declared variable. | 479 /// The alignment of the declared variable. |
441 uint32_t Alignment = 0; | 480 uint32_t Alignment = 0; |
442 /// True if a declared (global) constant. | 481 /// True if a declared (global) constant. |
443 bool IsConstant = false; | 482 bool IsConstant = false; |
444 /// If set to true, force getSuppressMangling() to return true. | 483 /// If set to true, force getSuppressMangling() to return true. |
445 const bool ForceSuppressMangling; | 484 const bool ForceSuppressMangling; |
446 | 485 |
447 VariableDeclaration(llvm::GlobalValue::LinkageTypes Linkage, | 486 VariableDeclaration(llvm::GlobalValue::LinkageTypes Linkage, |
448 bool SuppressMangling) | 487 bool SuppressMangling) |
449 : GlobalDeclaration(VariableDeclarationKind, Linkage), | 488 : GlobalDeclaration(VariableDeclarationKind, Linkage), |
450 Initializers(new InitializerListType), | |
451 ForceSuppressMangling(SuppressMangling) {} | 489 ForceSuppressMangling(SuppressMangling) {} |
452 }; | 490 }; |
453 | 491 |
454 template <class StreamType> | 492 template <class StreamType> |
455 inline StreamType &operator<<(StreamType &Stream, | 493 inline StreamType &operator<<(StreamType &Stream, |
456 const VariableDeclaration::Initializer &Init) { | 494 const VariableDeclaration::Initializer &Init) { |
457 Init.dump(Stream); | 495 Init.dump(Stream); |
458 return Stream; | 496 return Stream; |
459 } | 497 } |
460 | 498 |
461 template <class StreamType> | 499 template <class StreamType> |
462 inline StreamType &operator<<(StreamType &Stream, | 500 inline StreamType &operator<<(StreamType &Stream, |
463 const GlobalDeclaration &Addr) { | 501 const GlobalDeclaration &Addr) { |
464 Addr.dump(Stream); | 502 Addr.dump(Stream); |
465 return Stream; | 503 return Stream; |
466 } | 504 } |
467 | 505 |
468 } // end of namespace Ice | 506 } // end of namespace Ice |
469 | 507 |
470 #endif // SUBZERO_SRC_ICEGLOBALINITS_H | 508 #endif // SUBZERO_SRC_ICEGLOBALINITS_H |
OLD | NEW |