1 //===--- PPCallbacksTracker.cpp - Preprocessor tracker -*--*---------------===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 /// 10 /// \file 11 /// \brief Implementations for preprocessor tracking. 12 /// 13 /// See the header for details. 14 /// 15 //===----------------------------------------------------------------------===// 16 17 #include "PPCallbacksTracker.h" 18 #include "clang/Lex/MacroArgs.h" 19 #include "llvm/Support/raw_ostream.h" 20 21 // Utility functions. 22 23 // Get a "file:line:column" source location string. 24 static std::string getSourceLocationString(clang::Preprocessor &PP, 25 clang::SourceLocation Loc) { 26 if (Loc.isInvalid()) 27 return std::string("(none)"); 28 29 if (Loc.isFileID()) { 30 clang::PresumedLoc PLoc = PP.getSourceManager().getPresumedLoc(Loc); 31 32 if (PLoc.isInvalid()) { 33 return std::string("(invalid)"); 34 } 35 36 std::string Str; 37 llvm::raw_string_ostream SS(Str); 38 39 // The macro expansion and spelling pos is identical for file locs. 40 SS << "\"" << PLoc.getFilename() << ':' << PLoc.getLine() << ':' 41 << PLoc.getColumn() << "\""; 42 43 std::string Result = SS.str(); 44 45 // YAML treats backslash as escape, so use forward slashes. 46 std::replace(Result.begin(), Result.end(), '\\', '/'); 47 48 return Result; 49 } 50 51 return std::string("(nonfile)"); 52 } 53 54 // Enum string tables. 55 56 // FileChangeReason strings. 57 static const char *const FileChangeReasonStrings[] = { 58 "EnterFile", "ExitFile", "SystemHeaderPragma", "RenameFile" 59 }; 60 61 // CharacteristicKind strings. 62 static const char *const CharacteristicKindStrings[] = { "C_User", "C_System", 63 "C_ExternCSystem" }; 64 65 // MacroDirective::Kind strings. 66 static const char *const MacroDirectiveKindStrings[] = { 67 "MD_Define","MD_Undefine", "MD_Visibility" 68 }; 69 70 // PragmaIntroducerKind strings. 71 static const char *const PragmaIntroducerKindStrings[] = { "PIK_HashPragma", 72 "PIK__Pragma", 73 "PIK___pragma" }; 74 75 // PragmaMessageKind strings. 76 static const char *const PragmaMessageKindStrings[] = { 77 "PMK_Message", "PMK_Warning", "PMK_Error" 78 }; 79 80 // ConditionValueKind strings. 81 static const char *const ConditionValueKindStrings[] = { 82 "CVK_NotEvaluated", "CVK_False", "CVK_True" 83 }; 84 85 // Mapping strings. 86 static const char *const MappingStrings[] = { "0", "MAP_IGNORE", 87 "MAP_REMARK", "MAP_WARNING", 88 "MAP_ERROR", "MAP_FATAL" }; 89 90 // PPCallbacksTracker functions. 91 92 PPCallbacksTracker::PPCallbacksTracker(llvm::SmallSet<std::string, 4> &Ignore, 93 std::vector<CallbackCall> &CallbackCalls, 94 clang::Preprocessor &PP) 95 : CallbackCalls(CallbackCalls), Ignore(Ignore), PP(PP) {} 96 97 PPCallbacksTracker::~PPCallbacksTracker() {} 98 99 // Callback functions. 100 101 // Callback invoked whenever a source file is entered or exited. 102 void PPCallbacksTracker::FileChanged( 103 clang::SourceLocation Loc, clang::PPCallbacks::FileChangeReason Reason, 104 clang::SrcMgr::CharacteristicKind FileType, clang::FileID PrevFID) { 105 beginCallback("FileChanged"); 106 appendArgument("Loc", Loc); 107 appendArgument("Reason", Reason, FileChangeReasonStrings); 108 appendArgument("FileType", FileType, CharacteristicKindStrings); 109 appendArgument("PrevFID", PrevFID); 110 } 111 112 // Callback invoked whenever a source file is skipped as the result 113 // of header guard optimization. 114 void 115 PPCallbacksTracker::FileSkipped(const clang::FileEntry &SkippedFile, 116 const clang::Token &FilenameTok, 117 clang::SrcMgr::CharacteristicKind FileType) { 118 beginCallback("FileSkipped"); 119 appendArgument("ParentFile", &SkippedFile); 120 appendArgument("FilenameTok", FilenameTok); 121 appendArgument("FileType", FileType, CharacteristicKindStrings); 122 } 123 124 // Callback invoked whenever an inclusion directive results in a 125 // file-not-found error. 126 bool 127 PPCallbacksTracker::FileNotFound(llvm::StringRef FileName, 128 llvm::SmallVectorImpl<char> &RecoveryPath) { 129 beginCallback("FileNotFound"); 130 appendFilePathArgument("FileName", FileName); 131 return false; 132 } 133 134 // Callback invoked whenever an inclusion directive of 135 // any kind (#include, #import, etc.) has been processed, regardless 136 // of whether the inclusion will actually result in an inclusion. 137 void PPCallbacksTracker::InclusionDirective( 138 clang::SourceLocation HashLoc, const clang::Token &IncludeTok, 139 llvm::StringRef FileName, bool IsAngled, 140 clang::CharSourceRange FilenameRange, const clang::FileEntry *File, 141 llvm::StringRef SearchPath, llvm::StringRef RelativePath, 142 const clang::Module *Imported, clang::SrcMgr::CharacteristicKind FileType) { 143 beginCallback("InclusionDirective"); 144 appendArgument("IncludeTok", IncludeTok); 145 appendFilePathArgument("FileName", FileName); 146 appendArgument("IsAngled", IsAngled); 147 appendArgument("FilenameRange", FilenameRange); 148 appendArgument("File", File); 149 appendFilePathArgument("SearchPath", SearchPath); 150 appendFilePathArgument("RelativePath", RelativePath); 151 appendArgument("Imported", Imported); 152 } 153 154 // Callback invoked whenever there was an explicit module-import 155 // syntax. 156 void PPCallbacksTracker::moduleImport(clang::SourceLocation ImportLoc, 157 clang::ModuleIdPath Path, 158 const clang::Module *Imported) { 159 beginCallback("moduleImport"); 160 appendArgument("ImportLoc", ImportLoc); 161 appendArgument("Path", Path); 162 appendArgument("Imported", Imported); 163 } 164 165 // Callback invoked when the end of the main file is reached. 166 // No subsequent callbacks will be made. 167 void PPCallbacksTracker::EndOfMainFile() { beginCallback("EndOfMainFile"); } 168 169 // Callback invoked when a #ident or #sccs directive is read. 170 void PPCallbacksTracker::Ident(clang::SourceLocation Loc, llvm::StringRef Str) { 171 beginCallback("Ident"); 172 appendArgument("Loc", Loc); 173 appendArgument("Str", Str); 174 } 175 176 // Callback invoked when start reading any pragma directive. 177 void 178 PPCallbacksTracker::PragmaDirective(clang::SourceLocation Loc, 179 clang::PragmaIntroducerKind Introducer) { 180 beginCallback("PragmaDirective"); 181 appendArgument("Loc", Loc); 182 appendArgument("Introducer", Introducer, PragmaIntroducerKindStrings); 183 } 184 185 // Callback invoked when a #pragma comment directive is read. 186 void PPCallbacksTracker::PragmaComment(clang::SourceLocation Loc, 187 const clang::IdentifierInfo *Kind, 188 llvm::StringRef Str) { 189 beginCallback("PragmaComment"); 190 appendArgument("Loc", Loc); 191 appendArgument("Kind", Kind); 192 appendArgument("Str", Str); 193 } 194 195 // Callback invoked when a #pragma detect_mismatch directive is 196 // read. 197 void PPCallbacksTracker::PragmaDetectMismatch(clang::SourceLocation Loc, 198 llvm::StringRef Name, 199 llvm::StringRef Value) { 200 beginCallback("PragmaDetectMismatch"); 201 appendArgument("Loc", Loc); 202 appendArgument("Name", Name); 203 appendArgument("Value", Value); 204 } 205 206 // Callback invoked when a #pragma clang __debug directive is read. 207 void PPCallbacksTracker::PragmaDebug(clang::SourceLocation Loc, 208 llvm::StringRef DebugType) { 209 beginCallback("PragmaDebug"); 210 appendArgument("Loc", Loc); 211 appendArgument("DebugType", DebugType); 212 } 213 214 // Callback invoked when a #pragma message directive is read. 215 void PPCallbacksTracker::PragmaMessage( 216 clang::SourceLocation Loc, llvm::StringRef Namespace, 217 clang::PPCallbacks::PragmaMessageKind Kind, llvm::StringRef Str) { 218 beginCallback("PragmaMessage"); 219 appendArgument("Loc", Loc); 220 appendArgument("Namespace", Namespace); 221 appendArgument("Kind", Kind, PragmaMessageKindStrings); 222 appendArgument("Str", Str); 223 } 224 225 // Callback invoked when a #pragma gcc dianostic push directive 226 // is read. 227 void PPCallbacksTracker::PragmaDiagnosticPush(clang::SourceLocation Loc, 228 llvm::StringRef Namespace) { 229 beginCallback("PragmaDiagnosticPush"); 230 appendArgument("Loc", Loc); 231 appendArgument("Namespace", Namespace); 232 } 233 234 // Callback invoked when a #pragma gcc dianostic pop directive 235 // is read. 236 void PPCallbacksTracker::PragmaDiagnosticPop(clang::SourceLocation Loc, 237 llvm::StringRef Namespace) { 238 beginCallback("PragmaDiagnosticPop"); 239 appendArgument("Loc", Loc); 240 appendArgument("Namespace", Namespace); 241 } 242 243 // Callback invoked when a #pragma gcc dianostic directive is read. 244 void PPCallbacksTracker::PragmaDiagnostic(clang::SourceLocation Loc, 245 llvm::StringRef Namespace, 246 clang::diag::Severity Mapping, 247 llvm::StringRef Str) { 248 beginCallback("PragmaDiagnostic"); 249 appendArgument("Loc", Loc); 250 appendArgument("Namespace", Namespace); 251 appendArgument("Mapping", (unsigned)Mapping, MappingStrings); 252 appendArgument("Str", Str); 253 } 254 255 // Called when an OpenCL extension is either disabled or 256 // enabled with a pragma. 257 void PPCallbacksTracker::PragmaOpenCLExtension( 258 clang::SourceLocation NameLoc, const clang::IdentifierInfo *Name, 259 clang::SourceLocation StateLoc, unsigned State) { 260 beginCallback("PragmaOpenCLExtension"); 261 appendArgument("NameLoc", NameLoc); 262 appendArgument("Name", Name); 263 appendArgument("StateLoc", StateLoc); 264 appendArgument("State", (int)State); 265 } 266 267 // Callback invoked when a #pragma warning directive is read. 268 void PPCallbacksTracker::PragmaWarning(clang::SourceLocation Loc, 269 llvm::StringRef WarningSpec, 270 llvm::ArrayRef<int> Ids) { 271 beginCallback("PragmaWarning"); 272 appendArgument("Loc", Loc); 273 appendArgument("WarningSpec", WarningSpec); 274 275 std::string Str; 276 llvm::raw_string_ostream SS(Str); 277 SS << "["; 278 for (int i = 0, e = Ids.size(); i != e; ++i) { 279 if (i) 280 SS << ", "; 281 SS << Ids[i]; 282 } 283 SS << "]"; 284 appendArgument("Ids", SS.str()); 285 } 286 287 // Callback invoked when a #pragma warning(push) directive is read. 288 void PPCallbacksTracker::PragmaWarningPush(clang::SourceLocation Loc, 289 int Level) { 290 beginCallback("PragmaWarningPush"); 291 appendArgument("Loc", Loc); 292 appendArgument("Level", Level); 293 } 294 295 // Callback invoked when a #pragma warning(pop) directive is read. 296 void PPCallbacksTracker::PragmaWarningPop(clang::SourceLocation Loc) { 297 beginCallback("PragmaWarningPop"); 298 appendArgument("Loc", Loc); 299 } 300 301 // Called by Preprocessor::HandleMacroExpandedIdentifier when a 302 // macro invocation is found. 303 void 304 PPCallbacksTracker::MacroExpands(const clang::Token &MacroNameTok, 305 const clang::MacroDefinition &MacroDefinition, 306 clang::SourceRange Range, 307 const clang::MacroArgs *Args) { 308 beginCallback("MacroExpands"); 309 appendArgument("MacroNameTok", MacroNameTok); 310 appendArgument("MacroDefinition", MacroDefinition); 311 appendArgument("Range", Range); 312 appendArgument("Args", Args); 313 } 314 315 // Hook called whenever a macro definition is seen. 316 void 317 PPCallbacksTracker::MacroDefined(const clang::Token &MacroNameTok, 318 const clang::MacroDirective *MacroDirective) { 319 beginCallback("MacroDefined"); 320 appendArgument("MacroNameTok", MacroNameTok); 321 appendArgument("MacroDirective", MacroDirective); 322 } 323 324 // Hook called whenever a macro #undef is seen. 325 void PPCallbacksTracker::MacroUndefined( 326 const clang::Token &MacroNameTok, 327 const clang::MacroDefinition &MacroDefinition, 328 const clang::MacroDirective *Undef) { 329 beginCallback("MacroUndefined"); 330 appendArgument("MacroNameTok", MacroNameTok); 331 appendArgument("MacroDefinition", MacroDefinition); 332 } 333 334 // Hook called whenever the 'defined' operator is seen. 335 void PPCallbacksTracker::Defined(const clang::Token &MacroNameTok, 336 const clang::MacroDefinition &MacroDefinition, 337 clang::SourceRange Range) { 338 beginCallback("Defined"); 339 appendArgument("MacroNameTok", MacroNameTok); 340 appendArgument("MacroDefinition", MacroDefinition); 341 appendArgument("Range", Range); 342 } 343 344 // Hook called when a source range is skipped. 345 void PPCallbacksTracker::SourceRangeSkipped(clang::SourceRange Range, 346 clang::SourceLocation EndifLoc) { 347 beginCallback("SourceRangeSkipped"); 348 appendArgument("Range", clang::SourceRange(Range.getBegin(), EndifLoc)); 349 } 350 351 // Hook called whenever an #if is seen. 352 void PPCallbacksTracker::If(clang::SourceLocation Loc, 353 clang::SourceRange ConditionRange, 354 ConditionValueKind ConditionValue) { 355 beginCallback("If"); 356 appendArgument("Loc", Loc); 357 appendArgument("ConditionRange", ConditionRange); 358 appendArgument("ConditionValue", ConditionValue, ConditionValueKindStrings); 359 } 360 361 // Hook called whenever an #elif is seen. 362 void PPCallbacksTracker::Elif(clang::SourceLocation Loc, 363 clang::SourceRange ConditionRange, 364 ConditionValueKind ConditionValue, 365 clang::SourceLocation IfLoc) { 366 beginCallback("Elif"); 367 appendArgument("Loc", Loc); 368 appendArgument("ConditionRange", ConditionRange); 369 appendArgument("ConditionValue", ConditionValue, ConditionValueKindStrings); 370 appendArgument("IfLoc", IfLoc); 371 } 372 373 // Hook called whenever an #ifdef is seen. 374 void PPCallbacksTracker::Ifdef(clang::SourceLocation Loc, 375 const clang::Token &MacroNameTok, 376 const clang::MacroDefinition &MacroDefinition) { 377 beginCallback("Ifdef"); 378 appendArgument("Loc", Loc); 379 appendArgument("MacroNameTok", MacroNameTok); 380 appendArgument("MacroDefinition", MacroDefinition); 381 } 382 383 // Hook called whenever an #ifndef is seen. 384 void PPCallbacksTracker::Ifndef(clang::SourceLocation Loc, 385 const clang::Token &MacroNameTok, 386 const clang::MacroDefinition &MacroDefinition) { 387 beginCallback("Ifndef"); 388 appendArgument("Loc", Loc); 389 appendArgument("MacroNameTok", MacroNameTok); 390 appendArgument("MacroDefinition", MacroDefinition); 391 } 392 393 // Hook called whenever an #else is seen. 394 void PPCallbacksTracker::Else(clang::SourceLocation Loc, 395 clang::SourceLocation IfLoc) { 396 beginCallback("Else"); 397 appendArgument("Loc", Loc); 398 appendArgument("IfLoc", IfLoc); 399 } 400 401 // Hook called whenever an #endif is seen. 402 void PPCallbacksTracker::Endif(clang::SourceLocation Loc, 403 clang::SourceLocation IfLoc) { 404 beginCallback("Endif"); 405 appendArgument("Loc", Loc); 406 appendArgument("IfLoc", IfLoc); 407 } 408 409 // Helper functions. 410 411 // Start a new callback. 412 void PPCallbacksTracker::beginCallback(const char *Name) { 413 DisableTrace = Ignore.count(std::string(Name)); 414 if (DisableTrace) 415 return; 416 CallbackCalls.push_back(CallbackCall(Name)); 417 } 418 419 // Append a bool argument to the top trace item. 420 void PPCallbacksTracker::appendArgument(const char *Name, bool Value) { 421 appendArgument(Name, (Value ? "true" : "false")); 422 } 423 424 // Append an int argument to the top trace item. 425 void PPCallbacksTracker::appendArgument(const char *Name, int Value) { 426 std::string Str; 427 llvm::raw_string_ostream SS(Str); 428 SS << Value; 429 appendArgument(Name, SS.str()); 430 } 431 432 // Append a string argument to the top trace item. 433 void PPCallbacksTracker::appendArgument(const char *Name, const char *Value) { 434 if (DisableTrace) 435 return; 436 CallbackCalls.back().Arguments.push_back(Argument(Name, Value)); 437 } 438 439 // Append a string object argument to the top trace item. 440 void PPCallbacksTracker::appendArgument(const char *Name, 441 llvm::StringRef Value) { 442 appendArgument(Name, Value.str()); 443 } 444 445 // Append a string object argument to the top trace item. 446 void PPCallbacksTracker::appendArgument(const char *Name, 447 const std::string &Value) { 448 appendArgument(Name, Value.c_str()); 449 } 450 451 // Append a token argument to the top trace item. 452 void PPCallbacksTracker::appendArgument(const char *Name, 453 const clang::Token &Value) { 454 appendArgument(Name, PP.getSpelling(Value)); 455 } 456 457 // Append an enum argument to the top trace item. 458 void PPCallbacksTracker::appendArgument(const char *Name, int Value, 459 const char *const Strings[]) { 460 appendArgument(Name, Strings[Value]); 461 } 462 463 // Append a FileID argument to the top trace item. 464 void PPCallbacksTracker::appendArgument(const char *Name, clang::FileID Value) { 465 if (Value.isInvalid()) { 466 appendArgument(Name, "(invalid)"); 467 return; 468 } 469 const clang::FileEntry *FileEntry = 470 PP.getSourceManager().getFileEntryForID(Value); 471 if (!FileEntry) { 472 appendArgument(Name, "(getFileEntryForID failed)"); 473 return; 474 } 475 appendFilePathArgument(Name, FileEntry->getName()); 476 } 477 478 // Append a FileEntry argument to the top trace item. 479 void PPCallbacksTracker::appendArgument(const char *Name, 480 const clang::FileEntry *Value) { 481 if (!Value) { 482 appendArgument(Name, "(null)"); 483 return; 484 } 485 appendFilePathArgument(Name, Value->getName()); 486 } 487 488 // Append a SourceLocation argument to the top trace item. 489 void PPCallbacksTracker::appendArgument(const char *Name, 490 clang::SourceLocation Value) { 491 if (Value.isInvalid()) { 492 appendArgument(Name, "(invalid)"); 493 return; 494 } 495 appendArgument(Name, getSourceLocationString(PP, Value).c_str()); 496 } 497 498 // Append a SourceRange argument to the top trace item. 499 void PPCallbacksTracker::appendArgument(const char *Name, 500 clang::SourceRange Value) { 501 if (DisableTrace) 502 return; 503 if (Value.isInvalid()) { 504 appendArgument(Name, "(invalid)"); 505 return; 506 } 507 std::string Str; 508 llvm::raw_string_ostream SS(Str); 509 SS << "[" << getSourceLocationString(PP, Value.getBegin()) << ", " 510 << getSourceLocationString(PP, Value.getEnd()) << "]"; 511 appendArgument(Name, SS.str()); 512 } 513 514 // Append a CharSourceRange argument to the top trace item. 515 void PPCallbacksTracker::appendArgument(const char *Name, 516 clang::CharSourceRange Value) { 517 if (Value.isInvalid()) { 518 appendArgument(Name, "(invalid)"); 519 return; 520 } 521 appendArgument(Name, getSourceString(Value).str().c_str()); 522 } 523 524 // Append a SourceLocation argument to the top trace item. 525 void PPCallbacksTracker::appendArgument(const char *Name, 526 clang::ModuleIdPath Value) { 527 if (DisableTrace) 528 return; 529 std::string Str; 530 llvm::raw_string_ostream SS(Str); 531 SS << "["; 532 for (int I = 0, E = Value.size(); I != E; ++I) { 533 if (I) 534 SS << ", "; 535 SS << "{" 536 << "Name: " << Value[I].first->getName() << ", " 537 << "Loc: " << getSourceLocationString(PP, Value[I].second) << "}"; 538 } 539 SS << "]"; 540 appendArgument(Name, SS.str()); 541 } 542 543 // Append an IdentifierInfo argument to the top trace item. 544 void PPCallbacksTracker::appendArgument(const char *Name, 545 const clang::IdentifierInfo *Value) { 546 if (!Value) { 547 appendArgument(Name, "(null)"); 548 return; 549 } 550 appendArgument(Name, Value->getName().str().c_str()); 551 } 552 553 // Append a MacroDirective argument to the top trace item. 554 void PPCallbacksTracker::appendArgument(const char *Name, 555 const clang::MacroDirective *Value) { 556 if (!Value) { 557 appendArgument(Name, "(null)"); 558 return; 559 } 560 appendArgument(Name, MacroDirectiveKindStrings[Value->getKind()]); 561 } 562 563 // Append a MacroDefinition argument to the top trace item. 564 void PPCallbacksTracker::appendArgument(const char *Name, 565 const clang::MacroDefinition &Value) { 566 std::string Str; 567 llvm::raw_string_ostream SS(Str); 568 SS << "["; 569 bool Any = false; 570 if (Value.getLocalDirective()) { 571 SS << "(local)"; 572 Any = true; 573 } 574 for (auto *MM : Value.getModuleMacros()) { 575 if (Any) SS << ", "; 576 SS << MM->getOwningModule()->getFullModuleName(); 577 } 578 SS << "]"; 579 appendArgument(Name, SS.str()); 580 } 581 582 // Append a MacroArgs argument to the top trace item. 583 void PPCallbacksTracker::appendArgument(const char *Name, 584 const clang::MacroArgs *Value) { 585 if (!Value) { 586 appendArgument(Name, "(null)"); 587 return; 588 } 589 std::string Str; 590 llvm::raw_string_ostream SS(Str); 591 SS << "["; 592 593 // Each argument is is a series of contiguous Tokens, terminated by a eof. 594 // Go through each argument printing tokens until we reach eof. 595 for (unsigned I = 0; I < Value->getNumMacroArguments(); ++I) { 596 const clang::Token *Current = Value->getUnexpArgument(I); 597 if (I) 598 SS << ", "; 599 bool First = true; 600 while (Current->isNot(clang::tok::eof)) { 601 if (!First) 602 SS << " "; 603 // We need to be careful here because the arguments might not be legal in 604 // YAML, so we use the token name for anything but identifiers and 605 // numeric literals. 606 if (Current->isAnyIdentifier() || 607 Current->is(clang::tok::numeric_constant)) { 608 SS << PP.getSpelling(*Current); 609 } else { 610 SS << "<" << Current->getName() << ">"; 611 } 612 ++Current; 613 First = false; 614 } 615 } 616 SS << "]"; 617 appendArgument(Name, SS.str()); 618 } 619 620 // Append a Module argument to the top trace item. 621 void PPCallbacksTracker::appendArgument(const char *Name, 622 const clang::Module *Value) { 623 if (!Value) { 624 appendArgument(Name, "(null)"); 625 return; 626 } 627 appendArgument(Name, Value->Name.c_str()); 628 } 629 630 // Append a double-quoted argument to the top trace item. 631 void PPCallbacksTracker::appendQuotedArgument(const char *Name, 632 const std::string &Value) { 633 std::string Str; 634 llvm::raw_string_ostream SS(Str); 635 SS << "\"" << Value << "\""; 636 appendArgument(Name, SS.str()); 637 } 638 639 // Append a double-quoted file path argument to the top trace item. 640 void PPCallbacksTracker::appendFilePathArgument(const char *Name, 641 llvm::StringRef Value) { 642 std::string Path(Value); 643 // YAML treats backslash as escape, so use forward slashes. 644 std::replace(Path.begin(), Path.end(), '\\', '/'); 645 appendQuotedArgument(Name, Path); 646 } 647 648 // Get the raw source string of the range. 649 llvm::StringRef 650 PPCallbacksTracker::getSourceString(clang::CharSourceRange Range) { 651 const char *B = PP.getSourceManager().getCharacterData(Range.getBegin()); 652 const char *E = PP.getSourceManager().getCharacterData(Range.getEnd()); 653 return llvm::StringRef(B, E - B); 654 } 655