1//===--- Disasm.cpp - Disassembler for bytecode functions -------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Dump method for Function which disassembles the bytecode.
10//
11//===----------------------------------------------------------------------===//
12
13#include "Boolean.h"
14#include "Context.h"
15#include "EvaluationResult.h"
16#include "FixedPoint.h"
17#include "Floating.h"
18#include "Function.h"
19#include "Integral.h"
20#include "IntegralAP.h"
21#include "InterpFrame.h"
22#include "MemberPointer.h"
23#include "Opcode.h"
24#include "PrimType.h"
25#include "Program.h"
26#include "clang/AST/ASTDumperUtils.h"
27#include "clang/AST/DeclCXX.h"
28#include "clang/AST/ExprCXX.h"
29#include "llvm/Support/Compiler.h"
30
31using namespace clang;
32using namespace clang::interp;
33
34template <typename T>
35inline static std::string printArg(Program &P, CodePtr &OpPC) {
36 if constexpr (std::is_pointer_v<T>) {
37 uint32_t ID = OpPC.read<uint32_t>();
38 std::string Result;
39 llvm::raw_string_ostream SS(Result);
40 SS << reinterpret_cast<T>(P.getNativePointer(Idx: ID));
41 return Result;
42 } else {
43 std::string Result;
44 llvm::raw_string_ostream SS(Result);
45 auto Arg = OpPC.read<T>();
46 // Make sure we print the integral value of chars.
47 if constexpr (std::is_integral_v<T>) {
48 if constexpr (sizeof(T) == 1) {
49 if constexpr (std::is_signed_v<T>)
50 SS << static_cast<int32_t>(Arg);
51 else
52 SS << static_cast<uint32_t>(Arg);
53 } else {
54 SS << Arg;
55 }
56 } else {
57 SS << Arg;
58 }
59
60 return Result;
61 }
62}
63
64template <> inline std::string printArg<Floating>(Program &P, CodePtr &OpPC) {
65 auto Sem = Floating::deserializeSemantics(Buff: *OpPC);
66
67 unsigned BitWidth = llvm::APFloatBase::semanticsSizeInBits(
68 llvm::APFloatBase::EnumToSemantics(S: Sem));
69 auto Memory =
70 std::make_unique<uint64_t[]>(num: llvm::APInt::getNumWords(BitWidth));
71 Floating Result(Memory.get(), Sem);
72 Floating::deserialize(Buff: *OpPC, Result: &Result);
73
74 OpPC += align(Size: Result.bytesToSerialize());
75
76 std::string S;
77 llvm::raw_string_ostream SS(S);
78 SS << std::move(Result);
79 return S;
80}
81
82template <>
83inline std::string printArg<IntegralAP<false>>(Program &P, CodePtr &OpPC) {
84 using T = IntegralAP<false>;
85 uint32_t BitWidth = T::deserializeSize(Buff: *OpPC);
86 auto Memory =
87 std::make_unique<uint64_t[]>(num: llvm::APInt::getNumWords(BitWidth));
88
89 T Result(Memory.get(), BitWidth);
90 T::deserialize(Buff: *OpPC, Result: &Result);
91
92 OpPC += align(Size: Result.bytesToSerialize());
93
94 std::string Str;
95 llvm::raw_string_ostream SS(Str);
96 SS << std::move(Result);
97 return Str;
98}
99
100template <>
101inline std::string printArg<IntegralAP<true>>(Program &P, CodePtr &OpPC) {
102 using T = IntegralAP<true>;
103 uint32_t BitWidth = T::deserializeSize(Buff: *OpPC);
104 auto Memory =
105 std::make_unique<uint64_t[]>(num: llvm::APInt::getNumWords(BitWidth));
106
107 T Result(Memory.get(), BitWidth);
108 T::deserialize(Buff: *OpPC, Result: &Result);
109
110 OpPC += align(Size: Result.bytesToSerialize());
111
112 std::string Str;
113 llvm::raw_string_ostream SS(Str);
114 SS << std::move(Result);
115 return Str;
116}
117
118template <> inline std::string printArg<FixedPoint>(Program &P, CodePtr &OpPC) {
119 auto F = FixedPoint::deserialize(Buff: *OpPC);
120 OpPC += align(Size: F.bytesToSerialize());
121
122 std::string Result;
123 llvm::raw_string_ostream SS(Result);
124 SS << std::move(F);
125 return Result;
126}
127
128static bool isJumpOpcode(Opcode Op) {
129 return Op == OP_Jmp || Op == OP_Jf || Op == OP_Jt;
130}
131
132static size_t getNumDisplayWidth(size_t N) {
133 unsigned L = 1u, M = 10u;
134 while (M <= N && ++L != std::numeric_limits<size_t>::digits10 + 1)
135 M *= 10u;
136
137 return L;
138}
139
140LLVM_DUMP_METHOD void Function::dump(CodePtr PC) const {
141 dump(OS&: llvm::errs(), PC);
142}
143
144LLVM_DUMP_METHOD void Function::dump(llvm::raw_ostream &OS,
145 CodePtr OpPC) const {
146 if (OpPC) {
147 assert(OpPC >= getCodeBegin());
148 assert(OpPC <= getCodeEnd());
149 }
150 {
151 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_GREEN, .Bold: true});
152 OS << getName() << " " << (const void *)this << "\n";
153 }
154 OS << "frame size: " << getFrameSize() << "\n";
155 OS << "arg size: " << getArgSize() << "\n";
156 OS << "rvo: " << hasRVO() << "\n";
157 OS << "this arg: " << hasThisPointer() << "\n";
158
159 struct OpText {
160 size_t Addr;
161 std::string Op;
162 bool IsJump;
163 bool CurrentOp = false;
164 llvm::SmallVector<std::string> Args;
165 };
166
167 auto PrintName = [](const char *Name) -> std::string {
168 return std::string(Name);
169 };
170
171 llvm::SmallVector<OpText> Code;
172 size_t LongestAddr = 0;
173 size_t LongestOp = 0;
174
175 for (CodePtr Start = getCodeBegin(), PC = Start; PC != getCodeEnd();) {
176 size_t Addr = PC - Start;
177 OpText Text;
178 auto Op = PC.read<Opcode>();
179 Text.Addr = Addr;
180 Text.IsJump = isJumpOpcode(Op);
181 Text.CurrentOp = (PC == OpPC);
182 switch (Op) {
183#define GET_DISASM
184#include "Opcodes.inc"
185#undef GET_DISASM
186 }
187 Code.push_back(Elt: Text);
188 LongestOp = std::max(a: Text.Op.size(), b: LongestOp);
189 LongestAddr = std::max(a: getNumDisplayWidth(N: Addr), b: LongestAddr);
190 }
191
192 // Record jumps and their targets.
193 struct JmpData {
194 size_t From;
195 size_t To;
196 };
197 llvm::SmallVector<JmpData> Jumps;
198 for (auto &Text : Code) {
199 if (Text.IsJump)
200 Jumps.push_back(Elt: {.From: Text.Addr, .To: Text.Addr + std::stoi(str: Text.Args[0]) +
201 align(Size: sizeof(Opcode)) +
202 align(Size: sizeof(int32_t))});
203 }
204
205 llvm::SmallVector<std::string> Text;
206 Text.reserve(N: Code.size());
207 size_t LongestLine = 0;
208 // Print code to a string, one at a time.
209 for (const auto &C : Code) {
210 std::string Line;
211 llvm::raw_string_ostream LS(Line);
212 if (OpPC) {
213 if (C.CurrentOp)
214 LS << " * ";
215 else
216 LS << " ";
217 }
218 LS << C.Addr;
219 LS.indent(NumSpaces: LongestAddr - getNumDisplayWidth(N: C.Addr) + 4);
220 LS << C.Op;
221 LS.indent(NumSpaces: LongestOp - C.Op.size() + 4);
222 for (auto &Arg : C.Args) {
223 LS << Arg << ' ';
224 }
225 Text.push_back(Elt: Line);
226 LongestLine = std::max(a: Line.size(), b: LongestLine);
227 }
228
229 assert(Code.size() == Text.size());
230
231 auto spaces = [](unsigned N) -> std::string {
232 std::string S;
233 for (unsigned I = 0; I != N; ++I)
234 S += ' ';
235 return S;
236 };
237
238 // Now, draw the jump lines.
239 for (auto &J : Jumps) {
240 if (J.To > J.From) {
241 bool FoundStart = false;
242 for (size_t LineIndex = 0; LineIndex != Text.size(); ++LineIndex) {
243 Text[LineIndex] += spaces(LongestLine - Text[LineIndex].size());
244
245 if (Code[LineIndex].Addr == J.From) {
246 Text[LineIndex] += " --+";
247 FoundStart = true;
248 } else if (Code[LineIndex].Addr == J.To) {
249 Text[LineIndex] += " <-+";
250 break;
251 } else if (FoundStart) {
252 Text[LineIndex] += " |";
253 }
254 }
255 LongestLine += 5;
256 } else {
257 bool FoundStart = false;
258 for (ssize_t LineIndex = Text.size() - 1; LineIndex >= 0; --LineIndex) {
259 Text[LineIndex] += spaces(LongestLine - Text[LineIndex].size());
260 if (Code[LineIndex].Addr == J.From) {
261 Text[LineIndex] += " --+";
262 FoundStart = true;
263 } else if (Code[LineIndex].Addr == J.To) {
264 Text[LineIndex] += " <-+";
265 break;
266 } else if (FoundStart) {
267 Text[LineIndex] += " |";
268 }
269 }
270 LongestLine += 5;
271 }
272 }
273
274 for (auto &Line : Text)
275 OS << Line << '\n';
276}
277
278LLVM_DUMP_METHOD void Program::dump() const { dump(OS&: llvm::errs()); }
279
280static const char *primTypeToString(PrimType T) {
281 switch (T) {
282 case PT_Sint8:
283 return "Sint8";
284 case PT_Uint8:
285 return "Uint8";
286 case PT_Sint16:
287 return "Sint16";
288 case PT_Uint16:
289 return "Uint16";
290 case PT_Sint32:
291 return "Sint32";
292 case PT_Uint32:
293 return "Uint32";
294 case PT_Sint64:
295 return "Sint64";
296 case PT_Uint64:
297 return "Uint64";
298 case PT_IntAP:
299 return "IntAP";
300 case PT_IntAPS:
301 return "IntAPS";
302 case PT_Bool:
303 return "Bool";
304 case PT_Float:
305 return "Float";
306 case PT_Ptr:
307 return "Ptr";
308 case PT_MemberPtr:
309 return "MemberPtr";
310 case PT_FixedPoint:
311 return "FixedPoint";
312 }
313 llvm_unreachable("Unhandled PrimType");
314}
315
316static std::string formatBytes(size_t B) {
317 std::string Result;
318 llvm::raw_string_ostream SS(Result);
319
320 if (B < (1u << 10u))
321 SS << B << " B";
322 else if (B < (1u << 20u))
323 SS << llvm::format(Fmt: "{0:F2}", Vals: B / 1024.) << " KB";
324 else
325 SS << llvm::format(Fmt: "{0:F2}", Vals: B / 1024. / 1024.) << " MB";
326
327 return Result;
328}
329
330LLVM_DUMP_METHOD void Program::dump(llvm::raw_ostream &OS) const {
331 {
332 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_RED, .Bold: true});
333 OS << "\n:: Program\n";
334 }
335
336 {
337 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::WHITE, .Bold: true});
338 size_t Bytes = 0;
339 Bytes += Allocator.getTotalMemory();
340 // All the maps.
341 Bytes += GlobalIndices.getMemorySize();
342 Bytes += Records.getMemorySize();
343 Bytes += DummyVariables.getMemorySize();
344
345 // All Records.
346 for (const Record *R : Records.values()) {
347 Bytes += sizeof(Record) + R->BaseMap.getMemorySize() +
348 R->VirtualBaseMap.getMemorySize();
349 Bytes += R->Fields.capacity_in_bytes() + R->Bases.capacity_in_bytes() +
350 R->VirtualBases.capacity_in_bytes();
351 }
352
353 // Globals are allocated via the allocator, so already counted.
354
355 OS << "Total memory : " << formatBytes(B: Bytes) << '\n';
356 OS << "Global Variables: " << Globals.size() << '\n';
357 }
358 unsigned GI = 0;
359 for (const Global *G : Globals) {
360 const Descriptor *Desc = G->block()->getDescriptor();
361 Pointer GP = getPtrGlobal(Idx: GI);
362
363 OS << GI << ": " << (const void *)G->block() << " ";
364 {
365 ColorScope SC(OS, true,
366 GP.isInitialized()
367 ? TerminalColor{.Color: llvm::raw_ostream::GREEN, .Bold: false}
368 : TerminalColor{.Color: llvm::raw_ostream::RED, .Bold: false});
369 OS << (GP.isInitialized() ? "initialized " : "uninitialized ");
370 }
371 if (GP.block()->isDummy())
372 OS << "dummy ";
373 Desc->dump(OS);
374
375 if (GP.isInitialized() && Desc->IsTemporary) {
376 if (const auto *MTE =
377 dyn_cast_if_present<MaterializeTemporaryExpr>(Val: Desc->asExpr());
378 MTE && MTE->getLifetimeExtendedTemporaryDecl()) {
379 if (const APValue *V =
380 MTE->getLifetimeExtendedTemporaryDecl()->getValue()) {
381 OS << " (global temporary value: ";
382 {
383 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_MAGENTA, .Bold: true});
384 std::string VStr;
385 llvm::raw_string_ostream SS(VStr);
386 V->dump(OS&: SS, Context: Ctx.getASTContext());
387
388 for (unsigned I = 0; I != VStr.size(); ++I) {
389 if (VStr[I] == '\n')
390 VStr[I] = ' ';
391 }
392 VStr.pop_back(); // Remove the newline (or now space) at the end.
393 OS << VStr;
394 }
395 OS << ')';
396 }
397 }
398 }
399
400 OS << "\n";
401 if (GP.isInitialized() && Desc->isPrimitive() && !G->block()->isDummy()) {
402 OS << " ";
403 {
404 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_CYAN, .Bold: false});
405 OS << primTypeToString(T: Desc->getPrimType()) << " ";
406 }
407 TYPE_SWITCH(Desc->getPrimType(), { GP.deref<T>().print(OS); });
408 OS << "\n";
409 }
410 ++GI;
411 }
412
413 {
414 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::WHITE, .Bold: true});
415 OS << "Functions: " << Funcs.size() << "\n";
416 }
417 for (const auto &Func : Funcs) {
418 Func.second->dump();
419 }
420 for (const auto &Anon : AnonFuncs) {
421 Anon->dump();
422 }
423}
424
425LLVM_DUMP_METHOD void Descriptor::dump() const {
426 dump(OS&: llvm::errs());
427 llvm::errs() << '\n';
428}
429
430LLVM_DUMP_METHOD void Descriptor::dump(llvm::raw_ostream &OS) const {
431 // Source
432 {
433 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BLUE, .Bold: true});
434 if (const auto *ND = dyn_cast_if_present<NamedDecl>(Val: asDecl()))
435 ND->printQualifiedName(OS);
436 else if (asExpr())
437 OS << "Expr " << (const void *)asExpr();
438 }
439
440 // Print a few interesting bits about the descriptor.
441 if (isPrimitiveArray())
442 OS << " primitive-array";
443 else if (isCompositeArray())
444 OS << " composite-array";
445 else if (isUnion())
446 OS << " union";
447 else if (isRecord())
448 OS << " record";
449 else if (isPrimitive())
450 OS << " primitive " << primTypeToString(T: getPrimType());
451
452 if (isZeroSizeArray())
453 OS << " zero-size-array";
454 else if (isUnknownSizeArray())
455 OS << " unknown-size-array";
456
457 if (IsConstexprUnknown)
458 OS << " constexpr-unknown";
459}
460
461/// Dump descriptor, including all valid offsets.
462LLVM_DUMP_METHOD void Descriptor::dumpFull(unsigned Offset,
463 unsigned Indent) const {
464 unsigned Spaces = Indent * 2;
465 llvm::raw_ostream &OS = llvm::errs();
466 OS.indent(NumSpaces: Spaces);
467 dump(OS);
468 OS << '\n';
469 OS.indent(NumSpaces: Spaces) << "Metadata: " << getMetadataSize() << " bytes\n";
470 OS.indent(NumSpaces: Spaces) << "Size: " << getSize() << " bytes\n";
471 OS.indent(NumSpaces: Spaces) << "AllocSize: " << getAllocSize() << " bytes\n";
472 Offset += getMetadataSize();
473 if (isCompositeArray()) {
474 OS.indent(NumSpaces: Spaces) << "Elements: " << getNumElems() << '\n';
475 unsigned FO = Offset;
476 for (unsigned I = 0; I != getNumElems(); ++I) {
477 FO += sizeof(InlineDescriptor);
478 assert(ElemDesc->getMetadataSize() == 0);
479 OS.indent(NumSpaces: Spaces) << "Element " << I << " offset: " << FO << '\n';
480 ElemDesc->dumpFull(Offset: FO, Indent: Indent + 1);
481
482 FO += ElemDesc->getAllocSize();
483 }
484 } else if (isPrimitiveArray()) {
485 OS.indent(NumSpaces: Spaces) << "Elements: " << getNumElems() << '\n';
486 OS.indent(NumSpaces: Spaces) << "Element type: " << primTypeToString(T: getPrimType())
487 << '\n';
488 unsigned FO = Offset + sizeof(InitMapPtr);
489 for (unsigned I = 0; I != getNumElems(); ++I) {
490 OS.indent(NumSpaces: Spaces) << "Element " << I << " offset: " << FO << '\n';
491 FO += getElemSize();
492 }
493 } else if (isRecord()) {
494 ElemRecord->dump(OS, Indentation: Indent + 1, Offset);
495 unsigned I = 0;
496 for (const Record::Field &F : ElemRecord->fields()) {
497 OS.indent(NumSpaces: Spaces) << "- Field " << I << ": ";
498 {
499 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_RED, .Bold: true});
500 OS << F.Decl->getName();
501 }
502 OS << ". Offset " << (Offset + F.Offset) << "\n";
503 F.Desc->dumpFull(Offset: Offset + F.Offset, Indent: Indent + 1);
504 ++I;
505 }
506 } else if (isPrimitive()) {
507 } else {
508 }
509
510 OS << '\n';
511}
512
513LLVM_DUMP_METHOD void InlineDescriptor::dump(llvm::raw_ostream &OS) const {
514 {
515 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BLUE, .Bold: true});
516 OS << "InlineDescriptor " << (const void *)this << "\n";
517 }
518 OS << "Offset: " << Offset << "\n";
519 OS << "IsConst: " << IsConst << "\n";
520 OS << "IsInitialized: " << IsInitialized << "\n";
521 OS << "IsBase: " << IsBase << "\n";
522 OS << "IsActive: " << IsActive << "\n";
523 OS << "InUnion: " << InUnion << "\n";
524 OS << "IsFieldMutable: " << IsFieldMutable << "\n";
525 OS << "IsArrayElement: " << IsArrayElement << "\n";
526 OS << "IsConstInMutable: " << IsConstInMutable << '\n';
527 OS << "Desc: ";
528 if (Desc)
529 Desc->dump(OS);
530 else
531 OS << "nullptr";
532 OS << "\n";
533}
534
535LLVM_DUMP_METHOD void InterpFrame::dump(llvm::raw_ostream &OS,
536 unsigned Indent) const {
537 unsigned Spaces = Indent * 2;
538 {
539 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BLUE, .Bold: true});
540 OS.indent(NumSpaces: Spaces);
541 if (getCallee())
542 describe(OS);
543 else
544 OS << "Frame (Depth: " << getDepth() << ")";
545 OS << "\n";
546 }
547 OS.indent(NumSpaces: Spaces) << "Function: " << getFunction();
548 if (const Function *F = getFunction()) {
549 OS << " (" << F->getName() << ")";
550 }
551 OS << "\n";
552 if (hasThisPointer())
553 OS.indent(NumSpaces: Spaces) << "This: " << getThis() << "\n";
554 else
555 OS.indent(NumSpaces: Spaces) << "This: -\n";
556 if (Func && Func->hasRVO())
557 OS.indent(NumSpaces: Spaces) << "RVO: " << getRVOPtr() << "\n";
558 else
559 OS.indent(NumSpaces: Spaces) << "RVO: -\n";
560 OS.indent(NumSpaces: Spaces) << "Depth: " << Depth << "\n";
561 OS.indent(NumSpaces: Spaces) << "ArgSize: " << ArgSize << "\n";
562 OS.indent(NumSpaces: Spaces) << "Args: " << (void *)Args << "\n";
563 OS.indent(NumSpaces: Spaces) << "FrameOffset: " << FrameOffset << "\n";
564 OS.indent(NumSpaces: Spaces) << "FrameSize: " << (Func ? Func->getFrameSize() : 0)
565 << "\n";
566
567 for (const InterpFrame *F = this->Caller; F; F = F->Caller) {
568 F->dump(OS, Indent: Indent + 1);
569 }
570}
571
572LLVM_DUMP_METHOD void Record::dump(llvm::raw_ostream &OS, unsigned Indentation,
573 unsigned Offset) const {
574 unsigned Indent = Indentation * 2;
575 OS.indent(NumSpaces: Indent);
576 {
577 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BLUE, .Bold: true});
578 OS << getName() << "\n";
579 }
580
581 unsigned I = 0;
582 for (const Record::Base &B : bases()) {
583 OS.indent(NumSpaces: Indent) << "- Base " << I << ". Offset " << (Offset + B.Offset)
584 << "\n";
585 B.R->dump(OS, Indentation: Indentation + 1, Offset: Offset + B.Offset);
586 ++I;
587 }
588
589 I = 0;
590 for (const Record::Field &F : fields()) {
591 OS.indent(NumSpaces: Indent) << "- Field " << I << ": ";
592 {
593 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_RED, .Bold: true});
594 OS << F.Decl->getName();
595 }
596 OS << ". Offset " << (Offset + F.Offset) << "\n";
597 ++I;
598 }
599
600 I = 0;
601 for (const Record::Base &B : virtual_bases()) {
602 OS.indent(NumSpaces: Indent) << "- Virtual Base " << I << ". Offset "
603 << (Offset + B.Offset) << "\n";
604 B.R->dump(OS, Indentation: Indentation + 1, Offset: Offset + B.Offset);
605 ++I;
606 }
607}
608
609LLVM_DUMP_METHOD void Block::dump(llvm::raw_ostream &OS) const {
610 {
611 ColorScope SC(OS, true, {.Color: llvm::raw_ostream::BRIGHT_BLUE, .Bold: true});
612 OS << "Block " << (const void *)this;
613 }
614 OS << " (";
615 Desc->dump(OS);
616 OS << ")\n";
617 unsigned NPointers = 0;
618 for (const Pointer *P = Pointers; P; P = P->asBlockPointer().Next) {
619 ++NPointers;
620 }
621 OS << " EvalID: " << EvalID << '\n';
622 OS << " DeclID: ";
623 if (DeclID)
624 OS << *DeclID << '\n';
625 else
626 OS << "-\n";
627 OS << " Pointers: " << NPointers << "\n";
628 OS << " Dead: " << isDead() << "\n";
629 OS << " Static: " << IsStatic << "\n";
630 OS << " Extern: " << isExtern() << "\n";
631 OS << " Initialized: " << IsInitialized << "\n";
632 OS << " Weak: " << isWeak() << "\n";
633 OS << " Dummy: " << isDummy() << '\n';
634 OS << " Dynamic: " << isDynamic() << "\n";
635}
636
637LLVM_DUMP_METHOD void EvaluationResult::dump() const {
638 auto &OS = llvm::errs();
639
640 if (empty()) {
641 OS << "Empty\n";
642 } else if (isInvalid()) {
643 OS << "Invalid\n";
644 } else {
645 OS << "Value: ";
646#ifndef NDEBUG
647 assert(Ctx);
648 Value.dump(OS, Ctx->getASTContext());
649#endif
650 }
651}
652