You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			149 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			149 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===- Analyze.cpp - PDB analysis functions ---------------------*- C++ -*-===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "Analyze.h"
 | ||
|  | 
 | ||
|  | #include "llvm/ADT/DenseSet.h"
 | ||
|  | #include "llvm/ADT/STLExtras.h"
 | ||
|  | #include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
 | ||
|  | #include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h"
 | ||
|  | #include "llvm/DebugInfo/CodeView/TypeRecord.h"
 | ||
|  | #include "llvm/DebugInfo/CodeView/TypeVisitorCallbacks.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/Native/PDBFile.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/Native/RawError.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
 | ||
|  | 
 | ||
|  | #include "llvm/Support/FormatVariadic.h"
 | ||
|  | #include "llvm/Support/raw_ostream.h"
 | ||
|  | 
 | ||
|  | #include <list>
 | ||
|  | 
 | ||
|  | using namespace llvm; | ||
|  | using namespace llvm::codeview; | ||
|  | using namespace llvm::pdb; | ||
|  | 
 | ||
|  | static StringRef getLeafTypeName(TypeLeafKind LT) { | ||
|  |   switch (LT) { | ||
|  | #define TYPE_RECORD(ename, value, name)                                        \
 | ||
|  |   case ename:                                                                  \ | ||
|  |     return #name; | ||
|  | #include "llvm/DebugInfo/CodeView/CodeViewTypes.def"
 | ||
|  |   default: | ||
|  |     break; | ||
|  |   } | ||
|  |   return "UnknownLeaf"; | ||
|  | } | ||
|  | 
 | ||
|  | namespace { | ||
|  | struct HashLookupVisitor : public TypeVisitorCallbacks { | ||
|  |   struct Entry { | ||
|  |     TypeIndex TI; | ||
|  |     CVType Record; | ||
|  |   }; | ||
|  | 
 | ||
|  |   explicit HashLookupVisitor(TpiStream &Tpi) : Tpi(Tpi) {} | ||
|  | 
 | ||
|  |   Error visitTypeBegin(CVType &Record) override { | ||
|  |     uint32_t H = Tpi.getHashValues()[I]; | ||
|  |     Record.Hash = H; | ||
|  |     TypeIndex TI(I + TypeIndex::FirstNonSimpleIndex); | ||
|  |     Lookup[H].push_back(Entry{TI, Record}); | ||
|  |     ++I; | ||
|  |     return Error::success(); | ||
|  |   } | ||
|  | 
 | ||
|  |   uint32_t I = 0; | ||
|  |   DenseMap<uint32_t, std::list<Entry>> Lookup; | ||
|  |   TpiStream &Tpi; | ||
|  | }; | ||
|  | } | ||
|  | 
 | ||
|  | AnalysisStyle::AnalysisStyle(PDBFile &File) : File(File) {} | ||
|  | 
 | ||
|  | Error AnalysisStyle::dump() { | ||
|  |   auto Tpi = File.getPDBTpiStream(); | ||
|  |   if (!Tpi) | ||
|  |     return Tpi.takeError(); | ||
|  | 
 | ||
|  |   HashLookupVisitor Hasher(*Tpi); | ||
|  | 
 | ||
|  |   uint32_t RecordCount = Tpi->getNumTypeRecords(); | ||
|  |   auto Offsets = Tpi->getTypeIndexOffsets(); | ||
|  |   auto Types = llvm::make_unique<LazyRandomTypeCollection>( | ||
|  |       Tpi->typeArray(), RecordCount, Offsets); | ||
|  | 
 | ||
|  |   if (auto EC = codeview::visitTypeStream(*Types, Hasher)) | ||
|  |     return EC; | ||
|  | 
 | ||
|  |   auto &Adjusters = Tpi->getHashAdjusters(); | ||
|  |   DenseSet<uint32_t> AdjusterSet; | ||
|  |   for (const auto &Adj : Adjusters) { | ||
|  |     assert(AdjusterSet.find(Adj.second) == AdjusterSet.end()); | ||
|  |     AdjusterSet.insert(Adj.second); | ||
|  |   } | ||
|  | 
 | ||
|  |   uint32_t Count = 0; | ||
|  |   outs() << "Searching for hash collisions\n"; | ||
|  |   for (const auto &H : Hasher.Lookup) { | ||
|  |     if (H.second.size() <= 1) | ||
|  |       continue; | ||
|  |     ++Count; | ||
|  |     outs() << formatv("Hash: {0}, Count: {1} records\n", H.first, | ||
|  |                       H.second.size()); | ||
|  |     for (const auto &R : H.second) { | ||
|  |       auto Iter = AdjusterSet.find(R.TI.getIndex()); | ||
|  |       StringRef Prefix; | ||
|  |       if (Iter != AdjusterSet.end()) { | ||
|  |         Prefix = "[HEAD]"; | ||
|  |         AdjusterSet.erase(Iter); | ||
|  |       } | ||
|  |       StringRef LeafName = getLeafTypeName(R.Record.Type); | ||
|  |       uint32_t TI = R.TI.getIndex(); | ||
|  |       StringRef TypeName = Types->getTypeName(R.TI); | ||
|  |       outs() << formatv("{0,-6} {1} ({2:x}) {3}\n", Prefix, LeafName, TI, | ||
|  |                         TypeName); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   outs() << "\n"; | ||
|  |   outs() << "Dumping hash adjustment chains\n"; | ||
|  |   for (const auto &A : Tpi->getHashAdjusters()) { | ||
|  |     TypeIndex TI(A.second); | ||
|  |     StringRef TypeName = Types->getTypeName(TI); | ||
|  |     const CVType &HeadRecord = Types->getType(TI); | ||
|  |     assert(HeadRecord.Hash.hasValue()); | ||
|  | 
 | ||
|  |     auto CollisionsIter = Hasher.Lookup.find(*HeadRecord.Hash); | ||
|  |     if (CollisionsIter == Hasher.Lookup.end()) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     const auto &Collisions = CollisionsIter->second; | ||
|  |     outs() << TypeName << "\n"; | ||
|  |     outs() << formatv("    [HEAD] {0:x} {1} {2}\n", A.second, | ||
|  |                       getLeafTypeName(HeadRecord.Type), TypeName); | ||
|  |     for (const auto &Chain : Collisions) { | ||
|  |       if (Chain.TI == TI) | ||
|  |         continue; | ||
|  |       const CVType &TailRecord = Types->getType(Chain.TI); | ||
|  |       outs() << formatv("           {0:x} {1} {2}\n", Chain.TI.getIndex(), | ||
|  |                         getLeafTypeName(TailRecord.Type), | ||
|  |                         Types->getTypeName(Chain.TI)); | ||
|  |     } | ||
|  |   } | ||
|  |   outs() << formatv("There are {0} orphaned hash adjusters\n", | ||
|  |                     AdjusterSet.size()); | ||
|  |   for (const auto &Adj : AdjusterSet) { | ||
|  |     outs() << formatv("    {0}\n", Adj); | ||
|  |   } | ||
|  | 
 | ||
|  |   uint32_t DistinctHashValues = Hasher.Lookup.size(); | ||
|  |   outs() << formatv("{0}/{1} hash collisions", Count, DistinctHashValues); | ||
|  |   return Error::success(); | ||
|  | } |