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