OLD | NEW |
1 //===- subzero/src/IceGlobalContext.cpp - Global context defs ---*- C++ -*-===// | 1 //===- subzero/src/IceGlobalContext.cpp - Global context defs ---*- 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 // This file defines aspects of the compilation that persist across | 10 // This file defines aspects of the compilation that persist across |
(...skipping 97 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
108 public: | 108 public: |
109 ConstantPool() {} | 109 ConstantPool() {} |
110 TypePool<float, ConstantFloat, true> Floats; | 110 TypePool<float, ConstantFloat, true> Floats; |
111 TypePool<double, ConstantDouble, true> Doubles; | 111 TypePool<double, ConstantDouble, true> Doubles; |
112 TypePool<uint32_t, ConstantInteger32> Integers32; | 112 TypePool<uint32_t, ConstantInteger32> Integers32; |
113 TypePool<uint64_t, ConstantInteger64> Integers64; | 113 TypePool<uint64_t, ConstantInteger64> Integers64; |
114 TypePool<RelocatableTuple, ConstantRelocatable> Relocatables; | 114 TypePool<RelocatableTuple, ConstantRelocatable> Relocatables; |
115 UndefPool Undefs; | 115 UndefPool Undefs; |
116 }; | 116 }; |
117 | 117 |
| 118 class TimerStack { |
| 119 TimerStack(const TimerStack &) LLVM_DELETED_FUNCTION; |
| 120 TimerStack &operator=(const TimerStack &) LLVM_DELETED_FUNCTION; |
| 121 |
| 122 public: |
| 123 TimerStack(const IceString &TopLevelName) |
| 124 : FirstTimestamp(timestamp()), LastTimestamp(FirstTimestamp) { |
| 125 push(TopLevelName); |
| 126 } |
| 127 void push(const IceString &Name) { |
| 128 update(); |
| 129 Stack.push_back(Name); |
| 130 // Construct the new node's recursive name by appending .Name to |
| 131 // the previous top-of-stack's recursive name. |
| 132 IceString Prefix = ""; |
| 133 if (!FullNames.empty()) |
| 134 Prefix = FullNames.back() + "."; |
| 135 FullNames.push_back(Prefix + Name); |
| 136 } |
| 137 void pop(const IceString &Name) { |
| 138 update(); |
| 139 assert(!Stack.empty()); |
| 140 assert(Stack.back() == Name); |
| 141 Stack.pop_back(); |
| 142 FullNames.pop_back(); |
| 143 } |
| 144 void dump(Ostream &Str); |
| 145 |
| 146 private: |
| 147 typedef std::map<IceString, double> MapType; |
| 148 void update(); |
| 149 void dumpHelper(Ostream &Str, const MapType &Map, double TotalTime); |
| 150 static double timestamp() { |
| 151 // TODO: Implement in terms of std::chrono for C++11. |
| 152 return llvm::TimeRecord::getCurrentTime(false).getWallTime(); |
| 153 } |
| 154 const double FirstTimestamp; |
| 155 volatile double LastTimestamp; |
| 156 std::vector<IceString> Stack; |
| 157 std::vector<IceString> FullNames; |
| 158 MapType RecursiveTime; |
| 159 MapType LeafTime; |
| 160 }; |
| 161 |
| 162 void TimerStack::update() { |
| 163 // Whenever the stack is about to change, we grab the time delta |
| 164 // since the last change and add it to all active recursive elements |
| 165 // and to the non-recursive element for the top of the stack. |
| 166 double Current = timestamp(); |
| 167 double Delta = Current - LastTimestamp; |
| 168 LastTimestamp = Current; |
| 169 for (size_t i = 0; i < FullNames.size(); ++i) { |
| 170 RecursiveTime[FullNames[i]] += Delta; |
| 171 } |
| 172 if (!Stack.empty()) { |
| 173 LeafTime[Stack.back()] += Delta; |
| 174 } |
| 175 } |
| 176 |
| 177 void TimerStack::dumpHelper(Ostream &Str, const MapType &Map, |
| 178 double TotalTime) { |
| 179 // Dump the items in reverse order of their time contribution. |
| 180 typedef std::multimap<double, IceString> RMapType; |
| 181 RMapType MultiMap; |
| 182 for (MapType::const_iterator I = Map.begin(), E = Map.end(); I != E; ++I) { |
| 183 MultiMap.insert(std::make_pair(I->second, I->first)); |
| 184 } |
| 185 for (RMapType::const_reverse_iterator I = MultiMap.rbegin(), |
| 186 E = MultiMap.rend(); |
| 187 I != E; ++I) { |
| 188 char buf[80]; |
| 189 snprintf(buf, llvm::array_lengthof(buf), " %10.6f (%4.1f%%): ", I->first, |
| 190 I->first * 100 / TotalTime); |
| 191 Str << buf << I->second << "\n"; |
| 192 } |
| 193 } |
| 194 |
| 195 void TimerStack::dump(Ostream &Str) { |
| 196 update(); |
| 197 double TotalTime = LastTimestamp - FirstTimestamp; |
| 198 assert(TotalTime); |
| 199 Str << "Recursive function times:\n"; |
| 200 dumpHelper(Str, RecursiveTime, TotalTime); |
| 201 Str << "Non-recursive function times:\n"; |
| 202 dumpHelper(Str, LeafTime, TotalTime); |
| 203 } |
| 204 |
118 GlobalContext::GlobalContext(llvm::raw_ostream *OsDump, | 205 GlobalContext::GlobalContext(llvm::raw_ostream *OsDump, |
119 llvm::raw_ostream *OsEmit, VerboseMask Mask, | 206 llvm::raw_ostream *OsEmit, VerboseMask Mask, |
120 TargetArch Arch, OptLevel Opt, | 207 TargetArch Arch, OptLevel Opt, |
121 IceString TestPrefix, const ClFlags &Flags) | 208 IceString TestPrefix, const ClFlags &Flags) |
122 : StrDump(OsDump), StrEmit(OsEmit), VMask(Mask), | 209 : StrDump(OsDump), StrEmit(OsEmit), VMask(Mask), |
123 ConstPool(new ConstantPool()), Arch(Arch), Opt(Opt), | 210 ConstPool(new ConstantPool()), Arch(Arch), Opt(Opt), |
124 TestPrefix(TestPrefix), Flags(Flags), HasEmittedFirstMethod(false), | 211 TestPrefix(TestPrefix), Flags(Flags), HasEmittedFirstMethod(false), |
125 RNG("") {} | 212 RNG(""), Timers(new TimerStack("main")) {} |
126 | 213 |
127 // Scan a string for S[0-9A-Z]*_ patterns and replace them with | 214 // Scan a string for S[0-9A-Z]*_ patterns and replace them with |
128 // S<num>_ where <num> is the next base-36 value. If a type name | 215 // S<num>_ where <num> is the next base-36 value. If a type name |
129 // legitimately contains that pattern, then the substitution will be | 216 // legitimately contains that pattern, then the substitution will be |
130 // made in error and most likely the link will fail. In this case, | 217 // made in error and most likely the link will fail. In this case, |
131 // the test classes can be rewritten not to use that pattern, which is | 218 // the test classes can be rewritten not to use that pattern, which is |
132 // much simpler and more reliable than implementing a full demangling | 219 // much simpler and more reliable than implementing a full demangling |
133 // parser. Another substitution-in-error may occur if a type | 220 // parser. Another substitution-in-error may occur if a type |
134 // identifier ends with the pattern S[0-9A-Z]*, because an immediately | 221 // identifier ends with the pattern S[0-9A-Z]*, because an immediately |
135 // following substitution string like "S1_" or "PS1_" may be combined | 222 // following substitution string like "S1_" or "PS1_" may be combined |
(...skipping 241 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
377 BaseOS << "Unsupported constant type: " << Ty; | 464 BaseOS << "Unsupported constant type: " << Ty; |
378 llvm_unreachable(BaseOS.str().c_str()); | 465 llvm_unreachable(BaseOS.str().c_str()); |
379 } break; | 466 } break; |
380 case IceType_void: | 467 case IceType_void: |
381 case IceType_NUM: | 468 case IceType_NUM: |
382 break; | 469 break; |
383 } | 470 } |
384 llvm_unreachable("Unknown type"); | 471 llvm_unreachable("Unknown type"); |
385 } | 472 } |
386 | 473 |
| 474 void GlobalContext::pushTimer(const IceString &Name) { |
| 475 Timers.get()->push(Name); |
| 476 } |
| 477 |
| 478 void GlobalContext::popTimer(const IceString &Name) { Timers.get()->pop(Name); } |
| 479 |
387 void GlobalContext::dumpStats(const IceString &Name, bool Final) { | 480 void GlobalContext::dumpStats(const IceString &Name, bool Final) { |
388 if (Flags.DumpStats) { | 481 if (Flags.DumpStats) { |
389 if (Final) { | 482 if (Final) { |
390 StatsCumulative.dump(Name, getStrDump()); | 483 StatsCumulative.dump(Name, getStrDump()); |
391 } else { | 484 } else { |
392 StatsFunction.dump(Name, getStrDump()); | 485 StatsFunction.dump(Name, getStrDump()); |
393 StatsCumulative.dump("_TOTAL_", getStrDump()); | 486 StatsCumulative.dump("_TOTAL_", getStrDump()); |
394 } | 487 } |
395 } | 488 } |
396 } | 489 } |
397 | 490 |
398 void Timer::printElapsedUs(GlobalContext *Ctx, const IceString &Tag) const { | 491 void GlobalContext::dumpTimers() { Timers.get()->dump(getStrDump()); } |
399 if (Ctx->isVerbose(IceV_Timing)) { | |
400 // Prefixing with '#' allows timing strings to be included | |
401 // without error in textual assembly output. | |
402 Ctx->getStrDump() << "# " << getElapsedUs() << " usec " << Tag << "\n"; | |
403 } | |
404 } | |
405 | 492 |
406 } // end of namespace Ice | 493 } // end of namespace Ice |
OLD | NEW |