<div dir="ltr">One of the commits in this series, likely <a href="https://github.com/llvm/llvm-project/commit/d505e57cc273750541ec8bbce2065b8b87c99ad6">https://github.com/llvm/llvm-project/commit/d505e57cc273750541ec8bbce2065b8b87c99ad6</a> (which for some reason doesn't show up on cfe-commits?) broke Modules/merge-lifetime-extended-temporary.cpp when using a clang built at r365097 as host compiler. (Full error log: <a href="http://45.33.8.238/linux/5720/step_7.txt">http://45.33.8.238/linux/5720/step_7.txt</a> , also copied to the bottom of this mail for indexing.)<div><br></div><div>I tried the test with a few other clangs as host compilers and everything passes (I tried 362913, 369647, and the 9.0 release clang), so this very likely isn't worth looking into more. But in case others see this failure, maybe this note helps them.</div><div><br></div><div>Nico</div><div><br></div><div><div><br></div><div><br></div><div>Full error:</div><div><br></div><div>FAIL: Clang :: Modules/merge-lifetime-extended-temporary.cpp (6905 of 16551)<br>******************** TEST 'Clang :: Modules/merge-lifetime-extended-temporary.cpp' FAILED ********************<br>Script:<br>--<br>: 'RUN: at line 1';   /usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang -cc1 -internal-isystem /usr/local/google/home/thakis/src/llvm-project/out/gn/lib/clang/10.0.0/include -nostdsysteminc -fmodules -fimplicit-module-maps -fmodules-cache-path=/usr/local/google/home/thakis/src/llvm-project/out/gn/obj/clang/test/Modules/Output/merge-lifetime-extended-temporary.cpp.tmp -x c++ -I/usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/Inputs/merge-lifetime-extended-temporary -verify -std=c++11 /usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/merge-lifetime-extended-temporary.cpp -DORDER=1<br>: 'RUN: at line 2';   /usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang -cc1 -internal-isystem /usr/local/google/home/thakis/src/llvm-project/out/gn/lib/clang/10.0.0/include -nostdsysteminc -fmodules -fimplicit-module-maps -fmodules-cache-path=/usr/local/google/home/thakis/src/llvm-project/out/gn/obj/clang/test/Modules/Output/merge-lifetime-extended-temporary.cpp.tmp -x c++ -I/usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/Inputs/merge-lifetime-extended-temporary -verify -std=c++11 /usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/merge-lifetime-extended-temporary.cpp -DORDER=2<br>--<br>Exit Code: 134<br><br>Command Output (stderr):<br>--<br>clang: ../../clang/include/clang/AST/Type.h:666: const clang::ExtQualsTypeCommonBase *clang::QualType::getCommonPtr() const: Assertion `!isNull() && "Cannot retrieve a NULL type pointer"' failed.<br>Stack dump:<br>0.      Program arguments: /usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang -cc1 -internal-isystem /usr/local/google/home/thakis/src/llvm-project/out/gn/lib/clang/10.0.0/include -nostdsysteminc -fmodules -fimplicit-module-maps -fmodules-cache-path=/usr/local/google/home/thakis/src/llvm-project/out/gn/obj/clang/test/Modules/Output/merge-lifetime-extended-temporary.cpp.tmp -x c++ -I/usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/Inputs/merge-lifetime-extended-temporary -verify -std=c++11 /usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/merge-lifetime-extended-temporary.cpp -DORDER=1 <br>1.      /usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/merge-lifetime-extended-temporary.cpp:13:24: current parser token '=='<br></div><div> #0 0x0000000003e57cbd PrintStackTraceSignalHandler(void*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x3e57cbd)<br> #1 0x0000000003e55a0e llvm::sys::RunSignalHandlers() (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x3e55a0e)<br> #2 0x0000000003e57e7c SignalHandler(int) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x3e57e7c)<br> #3 0x00007fa76931f3a0 __restore_rt (/lib/x86_64-linux-gnu/libpthread.so.0+0x123a0)<br> #4 0x00007fa7689e6cfb raise /build/glibc-XAwaOT/glibc-2.28/signal/../sysdeps/unix/sysv/linux/raise.c:51:1<br> #5 0x00007fa7689d18ad abort /build/glibc-XAwaOT/glibc-2.28/stdlib/abort.c:81:7<br> #6 0x00007fa7689d177f get_sysdep_segment_value /build/glibc-XAwaOT/glibc-2.28/intl/loadmsgcat.c:509:8<br> #7 0x00007fa7689d177f _nl_load_domain /build/glibc-XAwaOT/glibc-2.28/intl/loadmsgcat.c:970:34<br> #8 0x00007fa7689df542 (/lib/x86_64-linux-gnu/libc.so.6+0x2f542)<br> #9 0x0000000002fce25a clang::ASTContext::getCanonicalType(clang::QualType) const (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x2fce25a)<br>#10 0x000000000455ebff clang::ASTContext::getLValueReferenceType(clang::QualType, bool) const (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x455ebff)<br>#11 0x00000000054064a5 clang::ASTReader::readTypeRecord(unsigned int) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x54064a5)<br>#12 0x00000000053fb305 clang::ASTReader::GetType(unsigned int) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x53fb305)<br>#13 0x000000000544a0fe clang::ASTDeclReader::VisitValueDecl(clang::ValueDecl*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x544a0fe)<br>#14 0x000000000544a260 clang::ASTDeclReader::VisitDeclaratorDecl(clang::DeclaratorDecl*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x544a260)<br>#15 0x000000000544e5cc clang::ASTDeclReader::VisitVarDeclImpl(clang::VarDecl*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x544e5cc)<br>#16 0x000000000544812f clang::ASTDeclReader::Visit(clang::Decl*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x544812f)<br>#17 0x000000000546a572 clang::ASTReader::ReadDeclRecord(unsigned int) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x546a572)<br>#18 0x00000000053fbee0 clang::ASTReader::GetDecl(unsigned int) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x53fbee0)<br>#19 0x000000000548118e clang::ASTStmtReader::VisitDeclRefExpr(clang::DeclRefExpr*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x548118e)<br>#20 0x000000000549098a clang::ASTReader::ReadStmtFromStream(clang::serialization::ModuleFile&) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x549098a)<br>#21 0x0000000005490cd6 clang::ASTReader::ReadExpr(clang::serialization::ModuleFile&) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x5490cd6)<br>#22 0x000000000544eae3 clang::ASTDeclReader::VisitVarDeclImpl(clang::VarDecl*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x544eae3)<br>#23 0x000000000544812f clang::ASTDeclReader::Visit(clang::Decl*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x544812f)<br>#24 0x000000000546a572 clang::ASTReader::ReadDeclRecord(unsigned int) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x546a572)<br>#25 0x000000000540b563 clang::ASTReader::FindExternalVisibleDeclsByName(clang::DeclContext const*, clang::DeclarationName) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x540b563)<br>#26 0x00000000046f2b74 clang::DeclContext::lookup(clang::DeclarationName) const (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x46f2b74)<br>#27 0x00000000050dc38b LookupDirect(clang::Sema&, clang::LookupResult&, clang::DeclContext const*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x50dc38b)<br>#28 0x00000000050d8761 CppNamespaceLookup(clang::Sema&, clang::LookupResult&, clang::ASTContext&, clang::DeclContext*, (anonymous namespace)::UnqualUsingDirectiveSet&) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x50d8761)<br>#29 0x00000000050d7aa1 clang::Sema::CppLookupName(clang::LookupResult&, clang::Scope*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x50d7aa1)<br>#30 0x00000000050dbd85 clang::Sema::LookupName(clang::LookupResult&, clang::Scope*, bool) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x50dbd85)<br>#31 0x0000000004f55f09 clang::Sema::ActOnIdExpression(clang::Scope*, clang::CXXScopeSpec&, clang::SourceLocation, clang::UnqualifiedId&, bool, bool, clang::CorrectionCandidateCallback*, bool, clang::Token*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4f55f09)<br>#32 0x0000000004b75fde clang::Parser::ParseCastExpression(bool, bool, bool&, clang::Parser::TypeCastState, bool) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b75fde)<br>#33 0x0000000004b73abd clang::Parser::ParseConstantExpressionInExprEvalContext(clang::Parser::TypeCastState) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b73abd)<br>#34 0x0000000004b2b599 clang::Parser::ParseStaticAssertDeclaration(clang::SourceLocation&) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b2b599)<br>#35 0x0000000004b47059 clang::Parser::ParseDeclaration(clang::DeclaratorContext, clang::SourceLocation&, clang::Parser::ParsedAttributesWithRange&, clang::SourceLocation*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b47059)<br>#36 0x0000000004b1dafb clang::Parser::ParseExternalDeclaration(clang::Parser::ParsedAttributesWithRange&, clang::ParsingDeclSpec*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b1dafb)<br>#37 0x0000000004b1c3a8 clang::Parser::ParseTopLevelDecl(clang::OpaquePtr<clang::DeclGroupRef>&, bool) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b1c3a8)<br>#38 0x0000000004b17d0d clang::ParseAST(clang::Sema&, bool, bool) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x4b17d0d)<br>#39 0x00000000034eafa0 clang::FrontendAction::Execute() (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x34eafa0)<br>#40 0x0000000003448362 clang::CompilerInstance::ExecuteAction(clang::FrontendAction&) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x3448362)<br>#41 0x0000000003598b22 clang::ExecuteCompilerInvocation(clang::CompilerInstance*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x3598b22)<br>#42 0x0000000002eb1abe cc1_main(llvm::ArrayRef<char const*>, char const*, void*) (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x2eb1abe)<br>#43 0x0000000002ebf93e main (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x2ebf93e)<br>#44 0x00007fa7689d352b __libc_start_main /build/glibc-XAwaOT/glibc-2.28/csu/../csu/libc-start.c:342:3<br>#45 0x0000000002eb102a _start (/usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang+0x2eb102a)<br>/usr/local/google/home/thakis/src/llvm-project/out/gn/obj/clang/test/Modules/Output/merge-lifetime-extended-temporary.cpp.script: line 2: 180489 Aborted                 /usr/local/google/home/thakis/src/llvm-project/out/gn/bin/clang -cc1 -internal-isystem /usr/local/google/home/thakis/src/llvm-project/out/gn/lib/clang/10.0.0/include -nostdsysteminc -fmodules -fimplicit-module-maps -fmodules-cache-path=/usr/local/google/home/thakis/src/llvm-project/out/gn/obj/clang/test/Modules/Output/merge-lifetime-extended-temporary.cpp.tmp -x c++ -I/usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/Inputs/merge-lifetime-extended-temporary -verify -std=c++11 /usr/local/google/home/thakis/src/llvm-project/clang/test/Modules/merge-lifetime-extended-temporary.cpp -DORDER=1<br></div></div></div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Sat, Dec 14, 2019 at 3:28 AM John McCall via cfe-commits <<a href="mailto:cfe-commits@lists.llvm.org">cfe-commits@lists.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><br>
Author: John McCall<br>
Date: 2019-12-14T03:28:23-05:00<br>
New Revision: c2f18315ff53006e44afe065368019e41cb98053<br>
<br>
URL: <a href="https://github.com/llvm/llvm-project/commit/c2f18315ff53006e44afe065368019e41cb98053" rel="noreferrer" target="_blank">https://github.com/llvm/llvm-project/commit/c2f18315ff53006e44afe065368019e41cb98053</a><br>
DIFF: <a href="https://github.com/llvm/llvm-project/commit/c2f18315ff53006e44afe065368019e41cb98053.diff" rel="noreferrer" target="_blank">https://github.com/llvm/llvm-project/commit/c2f18315ff53006e44afe065368019e41cb98053.diff</a><br>
<br>
LOG: Move ASTRecordReader into its own header; NFC.<br>
<br>
AbstractBasicReader.h has quite a few dependencies already,<br>
and that's only likely to increase.  Meanwhile, ASTRecordReader<br>
is really an implementation detail of the ASTReader that is only<br>
used in a small number of places.<br>
<br>
I've kept it in a public header for the use of projects like Swift<br>
that might want to plug in to Clang's serialization framework.<br>
<br>
I've also moved OMPClauseReader into an implementation file,<br>
although it can't be made private because of friendship.<br>
<br>
Added: <br>
    clang/include/clang/Serialization/ASTRecordReader.h<br>
<br>
Modified: <br>
    clang/include/clang/Serialization/ASTReader.h<br>
    clang/lib/Serialization/ASTReader.cpp<br>
    clang/lib/Serialization/ASTReaderDecl.cpp<br>
    clang/lib/Serialization/ASTReaderStmt.cpp<br>
    clang/lib/StaticAnalyzer/Frontend/ModelInjector.cpp<br>
<br>
Removed: <br>
<br>
<br>
<br>
################################################################################<br>
diff  --git a/clang/include/clang/Serialization/ASTReader.h b/clang/include/clang/Serialization/ASTReader.h<br>
index 3f321f03d966..e74bf00e0872 100644<br>
--- a/clang/include/clang/Serialization/ASTReader.h<br>
+++ b/clang/include/clang/Serialization/ASTReader.h<br>
@@ -13,26 +13,16 @@<br>
 #ifndef LLVM_CLANG_SERIALIZATION_ASTREADER_H<br>
 #define LLVM_CLANG_SERIALIZATION_ASTREADER_H<br>
<br>
-#include "clang/AST/AbstractBasicReader.h"<br>
-#include "clang/AST/DeclCXX.h"<br>
-#include "clang/AST/DeclObjC.h"<br>
-#include "clang/AST/DeclarationName.h"<br>
-#include "clang/AST/NestedNameSpecifier.h"<br>
-#include "clang/AST/OpenMPClause.h"<br>
-#include "clang/AST/TemplateBase.h"<br>
-#include "clang/AST/TemplateName.h"<br>
 #include "clang/AST/Type.h"<br>
 #include "clang/Basic/Diagnostic.h"<br>
 #include "clang/Basic/DiagnosticOptions.h"<br>
 #include "clang/Basic/IdentifierTable.h"<br>
-#include "clang/Basic/Module.h"<br>
 #include "clang/Basic/OpenCLOptions.h"<br>
 #include "clang/Basic/SourceLocation.h"<br>
 #include "clang/Basic/Version.h"<br>
 #include "clang/Lex/ExternalPreprocessorSource.h"<br>
 #include "clang/Lex/HeaderSearch.h"<br>
 #include "clang/Lex/PreprocessingRecord.h"<br>
-#include "clang/Lex/Token.h"<br>
 #include "clang/Sema/ExternalSemaSource.h"<br>
 #include "clang/Sema/IdentifierResolver.h"<br>
 #include "clang/Serialization/ASTBitCodes.h"<br>
@@ -40,9 +30,6 @@<br>
 #include "clang/Serialization/ModuleFile.h"<br>
 #include "clang/Serialization/ModuleFileExtension.h"<br>
 #include "clang/Serialization/ModuleManager.h"<br>
-#include "llvm/ADT/APFloat.h"<br>
-#include "llvm/ADT/APInt.h"<br>
-#include "llvm/ADT/APSInt.h"<br>
 #include "llvm/ADT/ArrayRef.h"<br>
 #include "llvm/ADT/DenseMap.h"<br>
 #include "llvm/ADT/DenseSet.h"<br>
@@ -58,8 +45,6 @@<br>
 #include "llvm/ADT/iterator.h"<br>
 #include "llvm/ADT/iterator_range.h"<br>
 #include "llvm/Bitstream/BitstreamReader.h"<br>
-#include "llvm/Support/Casting.h"<br>
-#include "llvm/Support/Endian.h"<br>
 #include "llvm/Support/MemoryBuffer.h"<br>
 #include "llvm/Support/Timer.h"<br>
 #include "llvm/Support/VersionTuple.h"<br>
@@ -83,6 +68,7 @@ class ASTReader;<br>
 class ASTRecordReader;<br>
 class CXXTemporary;<br>
 class Decl;<br>
+class DeclarationName;<br>
 class DeclaratorDecl;<br>
 class DeclContext;<br>
 class EnumDecl;<br>
@@ -112,9 +98,8 @@ class SourceManager;<br>
 class Stmt;<br>
 class SwitchCase;<br>
 class TargetOptions;<br>
-class TemplateParameterList;<br>
+class Token;<br>
 class TypedefNameDecl;<br>
-class TypeSourceInfo;<br>
 class ValueDecl;<br>
 class VarDecl;<br>
<br>
@@ -2280,341 +2265,6 @@ class ASTReader<br>
   bool isProcessingUpdateRecords() { return ProcessingUpdateRecords; }<br>
 };<br>
<br>
-/// An object for streaming information from a record.<br>
-class ASTRecordReader<br>
-    : public serialization::DataStreamBasicReader<ASTRecordReader> {<br>
-  using ModuleFile = serialization::ModuleFile;<br>
-<br>
-  ASTReader *Reader;<br>
-  ModuleFile *F;<br>
-  unsigned Idx = 0;<br>
-  ASTReader::RecordData Record;<br>
-<br>
-  using RecordData = ASTReader::RecordData;<br>
-  using RecordDataImpl = ASTReader::RecordDataImpl;<br>
-<br>
-public:<br>
-  /// Construct an ASTRecordReader that uses the default encoding scheme.<br>
-  ASTRecordReader(ASTReader &Reader, ModuleFile &F)<br>
-    : DataStreamBasicReader(Reader.getContext()), Reader(&Reader), F(&F) {}<br>
-<br>
-  /// Reads a record with id AbbrevID from Cursor, resetting the<br>
-  /// internal state.<br>
-  Expected<unsigned> readRecord(llvm::BitstreamCursor &Cursor,<br>
-                                unsigned AbbrevID);<br>
-<br>
-  /// Is this a module file for a module (rather than a PCH or similar).<br>
-  bool isModule() const { return F->isModule(); }<br>
-<br>
-  /// Retrieve the AST context that this AST reader supplements.<br>
-  ASTContext &getContext() { return Reader->getContext(); }<br>
-<br>
-  /// The current position in this record.<br>
-  unsigned getIdx() const { return Idx; }<br>
-<br>
-  /// The length of this record.<br>
-  size_t size() const { return Record.size(); }<br>
-<br>
-  /// An arbitrary index in this record.<br>
-  const uint64_t &operator[](size_t N) { return Record[N]; }<br>
-<br>
-  /// Returns the last value in this record.<br>
-  uint64_t back() { return Record.back(); }<br>
-<br>
-  /// Returns the current value in this record, and advances to the<br>
-  /// next value.<br>
-  uint64_t readInt() { return Record[Idx++]; }<br>
-<br>
-  ArrayRef<uint64_t> readIntArray(unsigned Len) {<br>
-    auto Array = llvm::makeArrayRef(Record).slice(Idx, Len);<br>
-    Idx += Len;<br>
-    return Array;<br>
-  }<br>
-<br>
-  /// Returns the current value in this record, without advancing.<br>
-  uint64_t peekInt() { return Record[Idx]; }<br>
-<br>
-  /// Skips the specified number of values.<br>
-  void skipInts(unsigned N) { Idx += N; }<br>
-<br>
-  /// Retrieve the global submodule ID its local ID number.<br>
-  serialization::SubmoduleID<br>
-  getGlobalSubmoduleID(unsigned LocalID) {<br>
-    return Reader->getGlobalSubmoduleID(*F, LocalID);<br>
-  }<br>
-<br>
-  /// Retrieve the submodule that corresponds to a global submodule ID.<br>
-  Module *getSubmodule(serialization::SubmoduleID GlobalID) {<br>
-    return Reader->getSubmodule(GlobalID);<br>
-  }<br>
-<br>
-  /// Read the record that describes the lexical contents of a DC.<br>
-  bool readLexicalDeclContextStorage(uint64_t Offset, DeclContext *DC) {<br>
-    return Reader->ReadLexicalDeclContextStorage(*F, F->DeclsCursor, Offset,<br>
-                                                 DC);<br>
-  }<br>
-<br>
-  /// Read the record that describes the visible contents of a DC.<br>
-  bool readVisibleDeclContextStorage(uint64_t Offset,<br>
-                                     serialization::DeclID ID) {<br>
-    return Reader->ReadVisibleDeclContextStorage(*F, F->DeclsCursor, Offset,<br>
-                                                 ID);<br>
-  }<br>
-<br>
-  ExplicitSpecifier readExplicitSpec() {<br>
-    uint64_t Kind = readInt();<br>
-    bool HasExpr = Kind & 0x1;<br>
-    Kind = Kind >> 1;<br>
-    return ExplicitSpecifier(HasExpr ? readExpr() : nullptr,<br>
-                             static_cast<ExplicitSpecKind>(Kind));<br>
-  }<br>
-<br>
-  /// Read information about an exception specification (inherited).<br>
-  //FunctionProtoType::ExceptionSpecInfo<br>
-  //readExceptionSpecInfo(SmallVectorImpl<QualType> &ExceptionStorage);<br>
-<br>
-  /// Get the global offset corresponding to a local offset.<br>
-  uint64_t getGlobalBitOffset(uint32_t LocalOffset) {<br>
-    return Reader->getGlobalBitOffset(*F, LocalOffset);<br>
-  }<br>
-<br>
-  /// Reads a statement.<br>
-  Stmt *readStmt() { return Reader->ReadStmt(*F); }<br>
-  Stmt *readStmtRef() { return readStmt(); /* FIXME: readSubStmt? */ }<br>
-<br>
-  /// Reads an expression.<br>
-  Expr *readExpr() { return Reader->ReadExpr(*F); }<br>
-<br>
-  /// Reads a sub-statement operand during statement reading.<br>
-  Stmt *readSubStmt() { return Reader->ReadSubStmt(); }<br>
-<br>
-  /// Reads a sub-expression operand during statement reading.<br>
-  Expr *readSubExpr() { return Reader->ReadSubExpr(); }<br>
-<br>
-  /// Reads a declaration with the given local ID in the given module.<br>
-  ///<br>
-  /// \returns The requested declaration, casted to the given return type.<br>
-  template<typename T><br>
-  T *GetLocalDeclAs(uint32_t LocalID) {<br>
-    return cast_or_null<T>(Reader->GetLocalDecl(*F, LocalID));<br>
-  }<br>
-<br>
-  /// Reads a TemplateArgumentLocInfo appropriate for the<br>
-  /// given TemplateArgument kind, advancing Idx.<br>
-  TemplateArgumentLocInfo<br>
-  readTemplateArgumentLocInfo(TemplateArgument::ArgKind Kind);<br>
-<br>
-  /// Reads a TemplateArgumentLoc, advancing Idx.<br>
-  TemplateArgumentLoc readTemplateArgumentLoc();<br>
-<br>
-  const ASTTemplateArgumentListInfo*<br>
-  readASTTemplateArgumentListInfo();<br>
-<br>
-  /// Reads a declarator info from the given record, advancing Idx.<br>
-  TypeSourceInfo *readTypeSourceInfo();<br>
-<br>
-  /// Reads the location information for a type.<br>
-  void readTypeLoc(TypeLoc TL);<br>
-<br>
-<br>
-  /// Map a local type ID within a given AST file to a global type ID.<br>
-  serialization::TypeID getGlobalTypeID(unsigned LocalID) const {<br>
-    return Reader->getGlobalTypeID(*F, LocalID);<br>
-  }<br>
-<br>
-  Qualifiers readQualifiers() {<br>
-    return Qualifiers::fromOpaqueValue(readInt());<br>
-  }<br>
-<br>
-  /// Read a type from the current position in the record.<br>
-  QualType readType() {<br>
-    return Reader->readType(*F, Record, Idx);<br>
-  }<br>
-  QualType readQualType() {<br>
-    return readType();<br>
-  }<br>
-<br>
-  /// Reads a declaration ID from the given position in this record.<br>
-  ///<br>
-  /// \returns The declaration ID read from the record, adjusted to a global ID.<br>
-  serialization::DeclID readDeclID() {<br>
-    return Reader->ReadDeclID(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Reads a declaration from the given position in a record in the<br>
-  /// given module, advancing Idx.<br>
-  Decl *readDecl() {<br>
-    return Reader->ReadDecl(*F, Record, Idx);<br>
-  }<br>
-  Decl *readDeclRef() {<br>
-    return readDecl();<br>
-  }<br>
-<br>
-  /// Reads a declaration from the given position in the record,<br>
-  /// advancing Idx.<br>
-  ///<br>
-  /// \returns The declaration read from this location, casted to the given<br>
-  /// result type.<br>
-  template<typename T><br>
-  T *readDeclAs() {<br>
-    return Reader->ReadDeclAs<T>(*F, Record, Idx);<br>
-  }<br>
-<br>
-  IdentifierInfo *readIdentifier() {<br>
-    return Reader->readIdentifier(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Read a selector from the Record, advancing Idx.<br>
-  Selector readSelector() {<br>
-    return Reader->ReadSelector(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Read a declaration name, advancing Idx.<br>
-  // DeclarationName readDeclarationName(); (inherited)<br>
-  DeclarationNameLoc readDeclarationNameLoc(DeclarationName Name);<br>
-  DeclarationNameInfo readDeclarationNameInfo();<br>
-<br>
-  void readQualifierInfo(QualifierInfo &Info);<br>
-<br>
-  /// Return a nested name specifier, advancing Idx.<br>
-  // NestedNameSpecifier *readNestedNameSpecifier(); (inherited)<br>
-<br>
-  NestedNameSpecifierLoc readNestedNameSpecifierLoc();<br>
-<br>
-  /// Read a template name, advancing Idx.<br>
-  // TemplateName readTemplateName(); (inherited)<br>
-<br>
-  /// Read a template argument, advancing Idx. (inherited)<br>
-  // TemplateArgument readTemplateArgument(bool Canonicalize = false);<br>
-<br>
-  /// Read a template parameter list, advancing Idx.<br>
-  TemplateParameterList *readTemplateParameterList();<br>
-<br>
-  /// Read a template argument array, advancing Idx.<br>
-  void readTemplateArgumentList(SmallVectorImpl<TemplateArgument> &TemplArgs,<br>
-                                bool Canonicalize = false);<br>
-<br>
-  /// Read a UnresolvedSet structure, advancing Idx.<br>
-  void readUnresolvedSet(LazyASTUnresolvedSet &Set);<br>
-<br>
-  /// Read a C++ base specifier, advancing Idx.<br>
-  CXXBaseSpecifier readCXXBaseSpecifier();<br>
-<br>
-  /// Read a CXXCtorInitializer array, advancing Idx.<br>
-  CXXCtorInitializer **readCXXCtorInitializers();<br>
-<br>
-  CXXTemporary *readCXXTemporary() {<br>
-    return Reader->ReadCXXTemporary(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Read a source location, advancing Idx.<br>
-  SourceLocation readSourceLocation() {<br>
-    return Reader->ReadSourceLocation(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Read a source range, advancing Idx.<br>
-  SourceRange readSourceRange() {<br>
-    return Reader->ReadSourceRange(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Read an arbitrary constant value, advancing Idx.<br>
-  APValue readAPValue();<br>
-<br>
-  /// Read an integral value, advancing Idx.<br>
-  // llvm::APInt readAPInt(); (inherited)<br>
-<br>
-  /// Read a signed integral value, advancing Idx.<br>
-  // llvm::APSInt readAPSInt(); (inherited)<br>
-<br>
-  /// Read a floating-point value, advancing Idx.<br>
-  llvm::APFloat readAPFloat(const llvm::fltSemantics &Sem);<br>
-<br>
-  /// Read a boolean value, advancing Idx.<br>
-  bool readBool() { return readInt() != 0; }<br>
-<br>
-  /// Read a 32-bit unsigned value; required to satisfy BasicReader.<br>
-  uint32_t readUInt32() {<br>
-    return uint32_t(readInt());<br>
-  }<br>
-<br>
-  /// Read a 64-bit unsigned value; required to satisfy BasicReader.<br>
-  uint64_t readUInt64() {<br>
-    return readInt();<br>
-  }<br>
-<br>
-  /// Read a string, advancing Idx.<br>
-  std::string readString() {<br>
-    return Reader->ReadString(Record, Idx);<br>
-  }<br>
-<br>
-  /// Read a path, advancing Idx.<br>
-  std::string readPath() {<br>
-    return Reader->ReadPath(*F, Record, Idx);<br>
-  }<br>
-<br>
-  /// Read a version tuple, advancing Idx.<br>
-  VersionTuple readVersionTuple() {<br>
-    return ASTReader::ReadVersionTuple(Record, Idx);<br>
-  }<br>
-<br>
-  /// Reads one attribute from the current stream position, advancing Idx.<br>
-  Attr *readAttr();<br>
-<br>
-  /// Reads attributes from the current stream position, advancing Idx.<br>
-  void readAttributes(AttrVec &Attrs);<br>
-<br>
-  /// Reads a token out of a record, advancing Idx.<br>
-  Token readToken() {<br>
-    return Reader->ReadToken(*F, Record, Idx);<br>
-  }<br>
-<br>
-  void recordSwitchCaseID(SwitchCase *SC, unsigned ID) {<br>
-    Reader->RecordSwitchCaseID(SC, ID);<br>
-  }<br>
-<br>
-  /// Retrieve the switch-case statement with the given ID.<br>
-  SwitchCase *getSwitchCaseWithID(unsigned ID) {<br>
-    return Reader->getSwitchCaseWithID(ID);<br>
-  }<br>
-};<br>
-<br>
-/// Helper class that saves the current stream position and<br>
-/// then restores it when destroyed.<br>
-struct SavedStreamPosition {<br>
-  explicit SavedStreamPosition(llvm::BitstreamCursor &Cursor)<br>
-      : Cursor(Cursor), Offset(Cursor.GetCurrentBitNo()) {}<br>
-<br>
-  ~SavedStreamPosition() {<br>
-    if (llvm::Error Err = Cursor.JumpToBit(Offset))<br>
-      llvm::report_fatal_error(<br>
-          "Cursor should always be able to go back, failed: " +<br>
-          toString(std::move(Err)));<br>
-  }<br>
-<br>
-private:<br>
-  llvm::BitstreamCursor &Cursor;<br>
-  uint64_t Offset;<br>
-};<br>
-<br>
-inline void PCHValidator::Error(const char *Msg) {<br>
-  Reader.Error(Msg);<br>
-}<br>
-<br>
-class OMPClauseReader : public OMPClauseVisitor<OMPClauseReader> {<br>
-  ASTRecordReader &Record;<br>
-  ASTContext &Context;<br>
-<br>
-public:<br>
-  OMPClauseReader(ASTRecordReader &Record)<br>
-      : Record(Record), Context(Record.getContext()) {}<br>
-<br>
-#define OPENMP_CLAUSE(Name, Class) void Visit##Class(Class *C);<br>
-#include "clang/Basic/OpenMPKinds.def"<br>
-  OMPClause *readClause();<br>
-  void VisitOMPClauseWithPreInit(OMPClauseWithPreInit *C);<br>
-  void VisitOMPClauseWithPostUpdate(OMPClauseWithPostUpdate *C);<br>
-};<br>
-<br>
 } // namespace clang<br>
<br>
 #endif // LLVM_CLANG_SERIALIZATION_ASTREADER_H<br>
<br>
diff  --git a/clang/include/clang/Serialization/ASTRecordReader.h b/clang/include/clang/Serialization/ASTRecordReader.h<br>
new file mode 100644<br>
index 000000000000..4fad2881ffd9<br>
--- /dev/null<br>
+++ b/clang/include/clang/Serialization/ASTRecordReader.h<br>
@@ -0,0 +1,351 @@<br>
+//===- ASTRecordReader.h - Helper classes for reading AST -------*- C++ -*-===//<br>
+//<br>
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.<br>
+// See <a href="https://llvm.org/LICENSE.txt" rel="noreferrer" target="_blank">https://llvm.org/LICENSE.txt</a> for license information.<br>
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception<br>
+//<br>
+//===----------------------------------------------------------------------===//<br>
+//<br>
+//  This file defines classes that are useful in the implementation of<br>
+//  the ASTReader.<br>
+//<br>
+//===----------------------------------------------------------------------===//<br>
+<br>
+#ifndef LLVM_CLANG_SERIALIZATION_ASTRECORDREADER_H<br>
+#define LLVM_CLANG_SERIALIZATION_ASTRECORDREADER_H<br>
+<br>
+#include "clang/AST/AbstractBasicReader.h"<br>
+#include "clang/Lex/Token.h"<br>
+#include "clang/Serialization/ASTReader.h"<br>
+#include "llvm/ADT/APFloat.h"<br>
+#include "llvm/ADT/APInt.h"<br>
+#include "llvm/ADT/APSInt.h"<br>
+<br>
+namespace clang {<br>
+<br>
+/// An object for streaming information from a record.<br>
+class ASTRecordReader<br>
+    : public serialization::DataStreamBasicReader<ASTRecordReader> {<br>
+  using ModuleFile = serialization::ModuleFile;<br>
+<br>
+  ASTReader *Reader;<br>
+  ModuleFile *F;<br>
+  unsigned Idx = 0;<br>
+  ASTReader::RecordData Record;<br>
+<br>
+  using RecordData = ASTReader::RecordData;<br>
+  using RecordDataImpl = ASTReader::RecordDataImpl;<br>
+<br>
+public:<br>
+  /// Construct an ASTRecordReader that uses the default encoding scheme.<br>
+  ASTRecordReader(ASTReader &Reader, ModuleFile &F)<br>
+    : DataStreamBasicReader(Reader.getContext()), Reader(&Reader), F(&F) {}<br>
+<br>
+  /// Reads a record with id AbbrevID from Cursor, resetting the<br>
+  /// internal state.<br>
+  Expected<unsigned> readRecord(llvm::BitstreamCursor &Cursor,<br>
+                                unsigned AbbrevID);<br>
+<br>
+  /// Is this a module file for a module (rather than a PCH or similar).<br>
+  bool isModule() const { return F->isModule(); }<br>
+<br>
+  /// Retrieve the AST context that this AST reader supplements.<br>
+  ASTContext &getContext() { return Reader->getContext(); }<br>
+<br>
+  /// The current position in this record.<br>
+  unsigned getIdx() const { return Idx; }<br>
+<br>
+  /// The length of this record.<br>
+  size_t size() const { return Record.size(); }<br>
+<br>
+  /// An arbitrary index in this record.<br>
+  const uint64_t &operator[](size_t N) { return Record[N]; }<br>
+<br>
+  /// Returns the last value in this record.<br>
+  uint64_t back() { return Record.back(); }<br>
+<br>
+  /// Returns the current value in this record, and advances to the<br>
+  /// next value.<br>
+  uint64_t readInt() { return Record[Idx++]; }<br>
+<br>
+  ArrayRef<uint64_t> readIntArray(unsigned Len) {<br>
+    auto Array = llvm::makeArrayRef(Record).slice(Idx, Len);<br>
+    Idx += Len;<br>
+    return Array;<br>
+  }<br>
+<br>
+  /// Returns the current value in this record, without advancing.<br>
+  uint64_t peekInt() { return Record[Idx]; }<br>
+<br>
+  /// Skips the specified number of values.<br>
+  void skipInts(unsigned N) { Idx += N; }<br>
+<br>
+  /// Retrieve the global submodule ID its local ID number.<br>
+  serialization::SubmoduleID<br>
+  getGlobalSubmoduleID(unsigned LocalID) {<br>
+    return Reader->getGlobalSubmoduleID(*F, LocalID);<br>
+  }<br>
+<br>
+  /// Retrieve the submodule that corresponds to a global submodule ID.<br>
+  Module *getSubmodule(serialization::SubmoduleID GlobalID) {<br>
+    return Reader->getSubmodule(GlobalID);<br>
+  }<br>
+<br>
+  /// Read the record that describes the lexical contents of a DC.<br>
+  bool readLexicalDeclContextStorage(uint64_t Offset, DeclContext *DC) {<br>
+    return Reader->ReadLexicalDeclContextStorage(*F, F->DeclsCursor, Offset,<br>
+                                                 DC);<br>
+  }<br>
+<br>
+  /// Read the record that describes the visible contents of a DC.<br>
+  bool readVisibleDeclContextStorage(uint64_t Offset,<br>
+                                     serialization::DeclID ID) {<br>
+    return Reader->ReadVisibleDeclContextStorage(*F, F->DeclsCursor, Offset,<br>
+                                                 ID);<br>
+  }<br>
+<br>
+  ExplicitSpecifier readExplicitSpec() {<br>
+    uint64_t Kind = readInt();<br>
+    bool HasExpr = Kind & 0x1;<br>
+    Kind = Kind >> 1;<br>
+    return ExplicitSpecifier(HasExpr ? readExpr() : nullptr,<br>
+                             static_cast<ExplicitSpecKind>(Kind));<br>
+  }<br>
+<br>
+  /// Read information about an exception specification (inherited).<br>
+  //FunctionProtoType::ExceptionSpecInfo<br>
+  //readExceptionSpecInfo(SmallVectorImpl<QualType> &ExceptionStorage);<br>
+<br>
+  /// Get the global offset corresponding to a local offset.<br>
+  uint64_t getGlobalBitOffset(uint32_t LocalOffset) {<br>
+    return Reader->getGlobalBitOffset(*F, LocalOffset);<br>
+  }<br>
+<br>
+  /// Reads a statement.<br>
+  Stmt *readStmt() { return Reader->ReadStmt(*F); }<br>
+  Stmt *readStmtRef() { return readStmt(); /* FIXME: readSubStmt? */ }<br>
+<br>
+  /// Reads an expression.<br>
+  Expr *readExpr() { return Reader->ReadExpr(*F); }<br>
+<br>
+  /// Reads a sub-statement operand during statement reading.<br>
+  Stmt *readSubStmt() { return Reader->ReadSubStmt(); }<br>
+<br>
+  /// Reads a sub-expression operand during statement reading.<br>
+  Expr *readSubExpr() { return Reader->ReadSubExpr(); }<br>
+<br>
+  /// Reads a declaration with the given local ID in the given module.<br>
+  ///<br>
+  /// \returns The requested declaration, casted to the given return type.<br>
+  template<typename T><br>
+  T *GetLocalDeclAs(uint32_t LocalID) {<br>
+    return cast_or_null<T>(Reader->GetLocalDecl(*F, LocalID));<br>
+  }<br>
+<br>
+  /// Reads a TemplateArgumentLocInfo appropriate for the<br>
+  /// given TemplateArgument kind, advancing Idx.<br>
+  TemplateArgumentLocInfo<br>
+  readTemplateArgumentLocInfo(TemplateArgument::ArgKind Kind);<br>
+<br>
+  /// Reads a TemplateArgumentLoc, advancing Idx.<br>
+  TemplateArgumentLoc readTemplateArgumentLoc();<br>
+<br>
+  const ASTTemplateArgumentListInfo*<br>
+  readASTTemplateArgumentListInfo();<br>
+<br>
+  /// Reads a declarator info from the given record, advancing Idx.<br>
+  TypeSourceInfo *readTypeSourceInfo();<br>
+<br>
+  /// Reads the location information for a type.<br>
+  void readTypeLoc(TypeLoc TL);<br>
+<br>
+<br>
+  /// Map a local type ID within a given AST file to a global type ID.<br>
+  serialization::TypeID getGlobalTypeID(unsigned LocalID) const {<br>
+    return Reader->getGlobalTypeID(*F, LocalID);<br>
+  }<br>
+<br>
+  Qualifiers readQualifiers() {<br>
+    return Qualifiers::fromOpaqueValue(readInt());<br>
+  }<br>
+<br>
+  /// Read a type from the current position in the record.<br>
+  QualType readType() {<br>
+    return Reader->readType(*F, Record, Idx);<br>
+  }<br>
+  QualType readQualType() {<br>
+    return readType();<br>
+  }<br>
+<br>
+  /// Reads a declaration ID from the given position in this record.<br>
+  ///<br>
+  /// \returns The declaration ID read from the record, adjusted to a global ID.<br>
+  serialization::DeclID readDeclID() {<br>
+    return Reader->ReadDeclID(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Reads a declaration from the given position in a record in the<br>
+  /// given module, advancing Idx.<br>
+  Decl *readDecl() {<br>
+    return Reader->ReadDecl(*F, Record, Idx);<br>
+  }<br>
+  Decl *readDeclRef() {<br>
+    return readDecl();<br>
+  }<br>
+<br>
+  /// Reads a declaration from the given position in the record,<br>
+  /// advancing Idx.<br>
+  ///<br>
+  /// \returns The declaration read from this location, casted to the given<br>
+  /// result type.<br>
+  template<typename T><br>
+  T *readDeclAs() {<br>
+    return Reader->ReadDeclAs<T>(*F, Record, Idx);<br>
+  }<br>
+<br>
+  IdentifierInfo *readIdentifier() {<br>
+    return Reader->readIdentifier(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Read a selector from the Record, advancing Idx.<br>
+  Selector readSelector() {<br>
+    return Reader->ReadSelector(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Read a declaration name, advancing Idx.<br>
+  // DeclarationName readDeclarationName(); (inherited)<br>
+  DeclarationNameLoc readDeclarationNameLoc(DeclarationName Name);<br>
+  DeclarationNameInfo readDeclarationNameInfo();<br>
+<br>
+  void readQualifierInfo(QualifierInfo &Info);<br>
+<br>
+  /// Return a nested name specifier, advancing Idx.<br>
+  // NestedNameSpecifier *readNestedNameSpecifier(); (inherited)<br>
+<br>
+  NestedNameSpecifierLoc readNestedNameSpecifierLoc();<br>
+<br>
+  /// Read a template name, advancing Idx.<br>
+  // TemplateName readTemplateName(); (inherited)<br>
+<br>
+  /// Read a template argument, advancing Idx. (inherited)<br>
+  // TemplateArgument readTemplateArgument(bool Canonicalize = false);<br>
+<br>
+  /// Read a template parameter list, advancing Idx.<br>
+  TemplateParameterList *readTemplateParameterList();<br>
+<br>
+  /// Read a template argument array, advancing Idx.<br>
+  void readTemplateArgumentList(SmallVectorImpl<TemplateArgument> &TemplArgs,<br>
+                                bool Canonicalize = false);<br>
+<br>
+  /// Read a UnresolvedSet structure, advancing Idx.<br>
+  void readUnresolvedSet(LazyASTUnresolvedSet &Set);<br>
+<br>
+  /// Read a C++ base specifier, advancing Idx.<br>
+  CXXBaseSpecifier readCXXBaseSpecifier();<br>
+<br>
+  /// Read a CXXCtorInitializer array, advancing Idx.<br>
+  CXXCtorInitializer **readCXXCtorInitializers();<br>
+<br>
+  CXXTemporary *readCXXTemporary() {<br>
+    return Reader->ReadCXXTemporary(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Read an OpenMP clause, advancing Idx.<br>
+  OMPClause *readOMPClause();<br>
+<br>
+  /// Read a source location, advancing Idx.<br>
+  SourceLocation readSourceLocation() {<br>
+    return Reader->ReadSourceLocation(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Read a source range, advancing Idx.<br>
+  SourceRange readSourceRange() {<br>
+    return Reader->ReadSourceRange(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Read an arbitrary constant value, advancing Idx.<br>
+  APValue readAPValue();<br>
+<br>
+  /// Read an integral value, advancing Idx.<br>
+  // llvm::APInt readAPInt(); (inherited)<br>
+<br>
+  /// Read a signed integral value, advancing Idx.<br>
+  // llvm::APSInt readAPSInt(); (inherited)<br>
+<br>
+  /// Read a floating-point value, advancing Idx.<br>
+  llvm::APFloat readAPFloat(const llvm::fltSemantics &Sem);<br>
+<br>
+  /// Read a boolean value, advancing Idx.<br>
+  bool readBool() { return readInt() != 0; }<br>
+<br>
+  /// Read a 32-bit unsigned value; required to satisfy BasicReader.<br>
+  uint32_t readUInt32() {<br>
+    return uint32_t(readInt());<br>
+  }<br>
+<br>
+  /// Read a 64-bit unsigned value; required to satisfy BasicReader.<br>
+  uint64_t readUInt64() {<br>
+    return readInt();<br>
+  }<br>
+<br>
+  /// Read a string, advancing Idx.<br>
+  std::string readString() {<br>
+    return Reader->ReadString(Record, Idx);<br>
+  }<br>
+<br>
+  /// Read a path, advancing Idx.<br>
+  std::string readPath() {<br>
+    return Reader->ReadPath(*F, Record, Idx);<br>
+  }<br>
+<br>
+  /// Read a version tuple, advancing Idx.<br>
+  VersionTuple readVersionTuple() {<br>
+    return ASTReader::ReadVersionTuple(Record, Idx);<br>
+  }<br>
+<br>
+  /// Reads one attribute from the current stream position, advancing Idx.<br>
+  Attr *readAttr();<br>
+<br>
+  /// Reads attributes from the current stream position, advancing Idx.<br>
+  void readAttributes(AttrVec &Attrs);<br>
+<br>
+  /// Reads a token out of a record, advancing Idx.<br>
+  Token readToken() {<br>
+    return Reader->ReadToken(*F, Record, Idx);<br>
+  }<br>
+<br>
+  void recordSwitchCaseID(SwitchCase *SC, unsigned ID) {<br>
+    Reader->RecordSwitchCaseID(SC, ID);<br>
+  }<br>
+<br>
+  /// Retrieve the switch-case statement with the given ID.<br>
+  SwitchCase *getSwitchCaseWithID(unsigned ID) {<br>
+    return Reader->getSwitchCaseWithID(ID);<br>
+  }<br>
+};<br>
+<br>
+/// Helper class that saves the current stream position and<br>
+/// then restores it when destroyed.<br>
+struct SavedStreamPosition {<br>
+  explicit SavedStreamPosition(llvm::BitstreamCursor &Cursor)<br>
+      : Cursor(Cursor), Offset(Cursor.GetCurrentBitNo()) {}<br>
+<br>
+  ~SavedStreamPosition() {<br>
+    if (llvm::Error Err = Cursor.JumpToBit(Offset))<br>
+      llvm::report_fatal_error(<br>
+          "Cursor should always be able to go back, failed: " +<br>
+          toString(std::move(Err)));<br>
+  }<br>
+<br>
+private:<br>
+  llvm::BitstreamCursor &Cursor;<br>
+  uint64_t Offset;<br>
+};<br>
+<br>
+inline void PCHValidator::Error(const char *Msg) {<br>
+  Reader.Error(Msg);<br>
+}<br>
+<br>
+} // namespace clang<br>
+<br>
+#endif<br>
<br>
diff  --git a/clang/lib/Serialization/ASTReader.cpp b/clang/lib/Serialization/ASTReader.cpp<br>
index c85cfcd3a33f..a2e4e7b469c9 100644<br>
--- a/clang/lib/Serialization/ASTReader.cpp<br>
+++ b/clang/lib/Serialization/ASTReader.cpp<br>
@@ -10,7 +10,7 @@<br>
 //<br>
 //===----------------------------------------------------------------------===//<br>
<br>
-#include "clang/Serialization/ASTReader.h"<br>
+#include "clang/Serialization/ASTRecordReader.h"<br>
 #include "ASTCommon.h"<br>
 #include "ASTReaderInternals.h"<br>
 #include "clang/AST/AbstractTypeReader.h"<br>
@@ -30,6 +30,7 @@<br>
 #include "clang/AST/ExprCXX.h"<br>
 #include "clang/AST/ExternalASTSource.h"<br>
 #include "clang/AST/NestedNameSpecifier.h"<br>
+#include "clang/AST/OpenMPClause.h"<br>
 #include "clang/AST/ODRHash.h"<br>
 #include "clang/AST/RawCommentList.h"<br>
 #include "clang/AST/TemplateBase.h"<br>
@@ -11471,6 +11472,31 @@ Expected<unsigned> ASTRecordReader::readRecord(llvm::BitstreamCursor &Cursor,<br>
 //// OMPClauseReader implementation<br>
 ////===----------------------------------------------------------------------===//<br>
<br>
+// This has to be in namespace clang because it's friended by all<br>
+// of the OMP clauses.<br>
+namespace clang {<br>
+<br>
+class OMPClauseReader : public OMPClauseVisitor<OMPClauseReader> {<br>
+  ASTRecordReader &Record;<br>
+  ASTContext &Context;<br>
+<br>
+public:<br>
+  OMPClauseReader(ASTRecordReader &Record)<br>
+      : Record(Record), Context(Record.getContext()) {}<br>
+<br>
+#define OPENMP_CLAUSE(Name, Class) void Visit##Class(Class *C);<br>
+#include "clang/Basic/OpenMPKinds.def"<br>
+  OMPClause *readClause();<br>
+  void VisitOMPClauseWithPreInit(OMPClauseWithPreInit *C);<br>
+  void VisitOMPClauseWithPostUpdate(OMPClauseWithPostUpdate *C);<br>
+};<br>
+<br>
+} // end namespace clang<br>
+<br>
+OMPClause *ASTRecordReader::readOMPClause() {<br>
+  return OMPClauseReader(*this).readClause();<br>
+}<br>
+<br>
 OMPClause *OMPClauseReader::readClause() {<br>
   OMPClause *C = nullptr;<br>
   switch (Record.readInt()) {<br>
<br>
diff  --git a/clang/lib/Serialization/ASTReaderDecl.cpp b/clang/lib/Serialization/ASTReaderDecl.cpp<br>
index 229bea8b031f..0240984d75a5 100644<br>
--- a/clang/lib/Serialization/ASTReaderDecl.cpp<br>
+++ b/clang/lib/Serialization/ASTReaderDecl.cpp<br>
@@ -48,7 +48,7 @@<br>
 #include "clang/Basic/Specifiers.h"<br>
 #include "clang/Sema/IdentifierResolver.h"<br>
 #include "clang/Serialization/ASTBitCodes.h"<br>
-#include "clang/Serialization/ASTReader.h"<br>
+#include "clang/Serialization/ASTRecordReader.h"<br>
 #include "clang/Serialization/ContinuousRangeMap.h"<br>
 #include "clang/Serialization/ModuleFile.h"<br>
 #include "llvm/ADT/DenseMap.h"<br>
@@ -2630,9 +2630,8 @@ void ASTDeclReader::VisitOMPAllocateDecl(OMPAllocateDecl *D) {<br>
   D->setVars(Vars);<br>
   SmallVector<OMPClause *, 8> Clauses;<br>
   Clauses.reserve(NumClauses);<br>
-  OMPClauseReader ClauseReader(Record);<br>
   for (unsigned I = 0; I != NumClauses; ++I)<br>
-    Clauses.push_back(ClauseReader.readClause());<br>
+    Clauses.push_back(Record.readOMPClause());<br>
   D->setClauses(Clauses);<br>
 }<br>
<br>
@@ -2641,9 +2640,8 @@ void ASTDeclReader::VisitOMPRequiresDecl(OMPRequiresDecl * D) {<br>
   unsigned NumClauses = D->clauselist_size();<br>
   SmallVector<OMPClause *, 8> Clauses;<br>
   Clauses.reserve(NumClauses);<br>
-  OMPClauseReader ClauseReader(Record);<br>
   for (unsigned I = 0; I != NumClauses; ++I)<br>
-    Clauses.push_back(ClauseReader.readClause());<br>
+    Clauses.push_back(Record.readOMPClause());<br>
   D->setClauses(Clauses);<br>
 }<br>
<br>
@@ -2674,9 +2672,8 @@ void ASTDeclReader::VisitOMPDeclareMapperDecl(OMPDeclareMapperDecl *D) {<br>
   unsigned NumClauses = D->clauselist_size();<br>
   SmallVector<OMPClause *, 8> Clauses;<br>
   Clauses.reserve(NumClauses);<br>
-  OMPClauseReader ClauseReader(Record);<br>
   for (unsigned I = 0; I != NumClauses; ++I)<br>
-    Clauses.push_back(ClauseReader.readClause());<br>
+    Clauses.push_back(Record.readOMPClause());<br>
   D->setClauses(Clauses);<br>
 }<br>
<br>
<br>
diff  --git a/clang/lib/Serialization/ASTReaderStmt.cpp b/clang/lib/Serialization/ASTReaderStmt.cpp<br>
index 5d7c6762061c..533502116ed7 100644<br>
--- a/clang/lib/Serialization/ASTReaderStmt.cpp<br>
+++ b/clang/lib/Serialization/ASTReaderStmt.cpp<br>
@@ -11,7 +11,7 @@<br>
 //<br>
 //===----------------------------------------------------------------------===//<br>
<br>
-#include "clang/Serialization/ASTReader.h"<br>
+#include "clang/Serialization/ASTRecordReader.h"<br>
 #include "clang/AST/ASTConcept.h"<br>
 #include "clang/AST/ASTContext.h"<br>
 #include "clang/AST/AttrIterator.h"<br>
@@ -67,8 +67,6 @@ using namespace serialization;<br>
 namespace clang {<br>
<br>
   class ASTStmtReader : public StmtVisitor<ASTStmtReader> {<br>
-    friend class OMPClauseReader;<br>
-<br>
     ASTRecordReader &Record;<br>
     llvm::BitstreamCursor &DeclsCursor;<br>
<br>
@@ -2026,10 +2024,9 @@ void ASTStmtReader::VisitAsTypeExpr(AsTypeExpr *E) {<br>
 void ASTStmtReader::VisitOMPExecutableDirective(OMPExecutableDirective *E) {<br>
   E->setLocStart(readSourceLocation());<br>
   E->setLocEnd(readSourceLocation());<br>
-  OMPClauseReader ClauseReader(Record);<br>
   SmallVector<OMPClause *, 5> Clauses;<br>
   for (unsigned i = 0; i < E->getNumClauses(); ++i)<br>
-    Clauses.push_back(ClauseReader.readClause());<br>
+    Clauses.push_back(Record.readOMPClause());<br>
   E->setClauses(Clauses);<br>
   if (E->hasAssociatedStmt())<br>
     E->setAssociatedStmt(Record.readSubStmt());<br>
<br>
diff  --git a/clang/lib/StaticAnalyzer/Frontend/ModelInjector.cpp b/clang/lib/StaticAnalyzer/Frontend/ModelInjector.cpp<br>
index 687fda75db48..7baae6778ebd 100644<br>
--- a/clang/lib/StaticAnalyzer/Frontend/ModelInjector.cpp<br>
+++ b/clang/lib/StaticAnalyzer/Frontend/ModelInjector.cpp<br>
@@ -11,6 +11,7 @@<br>
 #include "clang/Basic/IdentifierTable.h"<br>
 #include "clang/Basic/LangStandard.h"<br>
 #include "clang/Basic/Stack.h"<br>
+#include "clang/AST/DeclObjC.h"<br>
 #include "clang/Frontend/ASTUnit.h"<br>
 #include "clang/Frontend/CompilerInstance.h"<br>
 #include "clang/Frontend/FrontendAction.h"<br>
<br>
<br>
<br>
_______________________________________________<br>
cfe-commits mailing list<br>
<a href="mailto:cfe-commits@lists.llvm.org" target="_blank">cfe-commits@lists.llvm.org</a><br>
<a href="https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits" rel="noreferrer" target="_blank">https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits</a><br>
</blockquote></div>