1daefdbf3SDavid Blaikie //===-- llvm/CodeGen/DwarfStringPool.cpp - Dwarf Debug Framework ----------===//
2daefdbf3SDavid Blaikie //
3daefdbf3SDavid Blaikie //                     The LLVM Compiler Infrastructure
4daefdbf3SDavid Blaikie //
5daefdbf3SDavid Blaikie // This file is distributed under the University of Illinois Open Source
6daefdbf3SDavid Blaikie // License. See LICENSE.TXT for details.
7daefdbf3SDavid Blaikie //
8daefdbf3SDavid Blaikie //===----------------------------------------------------------------------===//
9daefdbf3SDavid Blaikie 
10daefdbf3SDavid Blaikie #include "DwarfStringPool.h"
11daefdbf3SDavid Blaikie #include "llvm/MC/MCStreamer.h"
12daefdbf3SDavid Blaikie 
13daefdbf3SDavid Blaikie using namespace llvm;
14daefdbf3SDavid Blaikie 
15322053caSBenjamin Kramer static std::pair<MCSymbol *, unsigned> &
16daefdbf3SDavid Blaikie getEntry(AsmPrinter &Asm,
17daefdbf3SDavid Blaikie          StringMap<std::pair<MCSymbol *, unsigned>, BumpPtrAllocator &> &Pool,
18daefdbf3SDavid Blaikie          StringRef Prefix, StringRef Str) {
195106ce78SDavid Blaikie   std::pair<MCSymbol *, unsigned> &Entry = Pool[Str];
20daefdbf3SDavid Blaikie   if (!Entry.first) {
21daefdbf3SDavid Blaikie     Entry.second = Pool.size() - 1;
22*9ab09237SRafael Espindola     Entry.first = Asm.createTempSymbol(Prefix);
23daefdbf3SDavid Blaikie   }
24daefdbf3SDavid Blaikie   return Entry;
25daefdbf3SDavid Blaikie }
26daefdbf3SDavid Blaikie 
27daefdbf3SDavid Blaikie MCSymbol *DwarfStringPool::getSymbol(AsmPrinter &Asm, StringRef Str) {
28daefdbf3SDavid Blaikie   return getEntry(Asm, Pool, Prefix, Str).first;
29daefdbf3SDavid Blaikie }
30daefdbf3SDavid Blaikie 
31daefdbf3SDavid Blaikie unsigned DwarfStringPool::getIndex(AsmPrinter &Asm, StringRef Str) {
32daefdbf3SDavid Blaikie   return getEntry(Asm, Pool, Prefix, Str).second;
33daefdbf3SDavid Blaikie }
34daefdbf3SDavid Blaikie 
35daefdbf3SDavid Blaikie void DwarfStringPool::emit(AsmPrinter &Asm, const MCSection *StrSection,
366741bb09SDavid Blaikie                            const MCSection *OffsetSection) {
37daefdbf3SDavid Blaikie   if (Pool.empty())
38daefdbf3SDavid Blaikie     return;
39daefdbf3SDavid Blaikie 
40daefdbf3SDavid Blaikie   // Start the dwarf str section.
41daefdbf3SDavid Blaikie   Asm.OutStreamer.SwitchSection(StrSection);
42daefdbf3SDavid Blaikie 
43daefdbf3SDavid Blaikie   // Get all of the string pool entries and put them in an array by their ID so
44daefdbf3SDavid Blaikie   // we can sort them.
45daefdbf3SDavid Blaikie   SmallVector<const StringMapEntry<std::pair<MCSymbol *, unsigned>> *, 64>
46daefdbf3SDavid Blaikie   Entries(Pool.size());
47daefdbf3SDavid Blaikie 
48daefdbf3SDavid Blaikie   for (const auto &E : Pool)
49daefdbf3SDavid Blaikie     Entries[E.getValue().second] = &E;
50daefdbf3SDavid Blaikie 
51daefdbf3SDavid Blaikie   for (const auto &Entry : Entries) {
52daefdbf3SDavid Blaikie     // Emit a label for reference from debug information entries.
53daefdbf3SDavid Blaikie     Asm.OutStreamer.EmitLabel(Entry->getValue().first);
54daefdbf3SDavid Blaikie 
55daefdbf3SDavid Blaikie     // Emit the string itself with a terminating null byte.
56daefdbf3SDavid Blaikie     Asm.OutStreamer.EmitBytes(
57daefdbf3SDavid Blaikie         StringRef(Entry->getKeyData(), Entry->getKeyLength() + 1));
58daefdbf3SDavid Blaikie   }
59daefdbf3SDavid Blaikie 
60daefdbf3SDavid Blaikie   // If we've got an offset section go ahead and emit that now as well.
61daefdbf3SDavid Blaikie   if (OffsetSection) {
62daefdbf3SDavid Blaikie     Asm.OutStreamer.SwitchSection(OffsetSection);
63daefdbf3SDavid Blaikie     unsigned offset = 0;
64daefdbf3SDavid Blaikie     unsigned size = 4; // FIXME: DWARF64 is 8.
65daefdbf3SDavid Blaikie     for (const auto &Entry : Entries) {
66daefdbf3SDavid Blaikie       Asm.OutStreamer.EmitIntValue(offset, size);
67daefdbf3SDavid Blaikie       offset += Entry->getKeyLength() + 1;
68daefdbf3SDavid Blaikie     }
69daefdbf3SDavid Blaikie   }
70daefdbf3SDavid Blaikie }
71