2018-04-12 04:57:28 +08:00
|
|
|
//===- ASTReaderStmt.cpp - Stmt/Expr Deserialization ----------------------===//
|
2009-04-27 13:14:47 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2009-04-27 13:14:47 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Statement/expression deserialization. This implements the
|
2010-08-19 07:56:43 +08:00
|
|
|
// ASTReader::ReadStmt method.
|
2009-04-27 13:14:47 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-19 07:57:17 +08:00
|
|
|
#include "clang/Serialization/ASTReader.h"
|
2012-07-05 04:19:54 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "clang/AST/AttrIterator.h"
|
|
|
|
#include "clang/AST/Decl.h"
|
|
|
|
#include "clang/AST/DeclAccessPair.h"
|
2009-09-10 07:08:42 +08:00
|
|
|
#include "clang/AST/DeclCXX.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "clang/AST/DeclGroup.h"
|
|
|
|
#include "clang/AST/DeclObjC.h"
|
2011-01-15 09:15:58 +08:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "clang/AST/DeclarationName.h"
|
|
|
|
#include "clang/AST/Expr.h"
|
|
|
|
#include "clang/AST/ExprCXX.h"
|
|
|
|
#include "clang/AST/ExprObjC.h"
|
|
|
|
#include "clang/AST/ExprOpenMP.h"
|
|
|
|
#include "clang/AST/NestedNameSpecifier.h"
|
|
|
|
#include "clang/AST/OpenMPClause.h"
|
|
|
|
#include "clang/AST/OperationKinds.h"
|
|
|
|
#include "clang/AST/Stmt.h"
|
|
|
|
#include "clang/AST/StmtCXX.h"
|
|
|
|
#include "clang/AST/StmtObjC.h"
|
|
|
|
#include "clang/AST/StmtOpenMP.h"
|
2009-04-27 13:14:47 +08:00
|
|
|
#include "clang/AST/StmtVisitor.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "clang/AST/TemplateBase.h"
|
|
|
|
#include "clang/AST/Type.h"
|
|
|
|
#include "clang/AST/UnresolvedSet.h"
|
|
|
|
#include "clang/Basic/CapturedStmt.h"
|
|
|
|
#include "clang/Basic/ExpressionTraits.h"
|
|
|
|
#include "clang/Basic/LLVM.h"
|
|
|
|
#include "clang/Basic/Lambda.h"
|
|
|
|
#include "clang/Basic/LangOptions.h"
|
|
|
|
#include "clang/Basic/OpenMPKinds.h"
|
|
|
|
#include "clang/Basic/OperatorKinds.h"
|
|
|
|
#include "clang/Basic/SourceLocation.h"
|
|
|
|
#include "clang/Basic/Specifiers.h"
|
|
|
|
#include "clang/Basic/TypeTraits.h"
|
2013-05-03 08:10:13 +08:00
|
|
|
#include "clang/Lex/Token.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "clang/Serialization/ASTBitCodes.h"
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2012-02-04 21:45:25 +08:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
|
|
|
#include "llvm/ADT/StringRef.h"
|
2019-07-04 06:40:07 +08:00
|
|
|
#include "llvm/Bitstream/BitstreamReader.h"
|
2018-04-12 04:57:28 +08:00
|
|
|
#include "llvm/Support/Casting.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <string>
|
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
using namespace clang;
|
2018-04-12 04:57:28 +08:00
|
|
|
using namespace serialization;
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2010-06-30 16:49:18 +08:00
|
|
|
namespace clang {
|
2010-06-28 17:31:42 +08:00
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
class ASTStmtReader : public StmtVisitor<ASTStmtReader> {
|
2013-07-19 11:13:43 +08:00
|
|
|
friend class OMPClauseReader;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
ASTRecordReader &Record;
|
2010-07-23 06:43:28 +08:00
|
|
|
llvm::BitstreamCursor &DeclsCursor;
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation ReadSourceLocation() {
|
2016-12-21 12:34:52 +08:00
|
|
|
return Record.readSourceLocation();
|
2013-05-03 08:10:13 +08:00
|
|
|
}
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceRange ReadSourceRange() {
|
2016-12-21 12:34:52 +08:00
|
|
|
return Record.readSourceRange();
|
2010-10-05 23:59:54 +08:00
|
|
|
}
|
2013-05-03 08:10:13 +08:00
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
std::string ReadString() {
|
2016-12-21 12:34:52 +08:00
|
|
|
return Record.readString();
|
2010-10-05 23:59:54 +08:00
|
|
|
}
|
2013-05-03 08:10:13 +08:00
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
TypeSourceInfo *GetTypeSourceInfo() {
|
2016-12-21 12:34:52 +08:00
|
|
|
return Record.getTypeSourceInfo();
|
2010-10-05 23:59:54 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
|
|
|
Decl *ReadDecl() {
|
2016-12-21 12:34:52 +08:00
|
|
|
return Record.readDecl();
|
2011-07-22 06:35:25 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-07-22 06:35:25 +08:00
|
|
|
template<typename T>
|
2016-12-16 04:53:26 +08:00
|
|
|
T *ReadDeclAs() {
|
2016-12-21 12:34:52 +08:00
|
|
|
return Record.readDeclAs<T>();
|
2011-07-22 06:35:25 +08:00
|
|
|
}
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
void ReadDeclarationNameLoc(DeclarationNameLoc &DNLoc,
|
|
|
|
DeclarationName Name) {
|
2016-12-21 12:34:52 +08:00
|
|
|
Record.readDeclarationNameLoc(DNLoc, Name);
|
2010-10-16 02:21:24 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
|
|
|
void ReadDeclarationNameInfo(DeclarationNameInfo &NameInfo) {
|
2016-12-21 12:34:52 +08:00
|
|
|
Record.readDeclarationNameInfo(NameInfo);
|
2010-10-16 02:21:24 +08:00
|
|
|
}
|
2010-10-05 23:59:54 +08:00
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
public:
|
2016-12-21 08:17:49 +08:00
|
|
|
ASTStmtReader(ASTRecordReader &Record, llvm::BitstreamCursor &Cursor)
|
|
|
|
: Record(Record), DeclsCursor(Cursor) {}
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// The number of record fields required for the Stmt class
|
2009-04-27 13:14:47 +08:00
|
|
|
/// itself.
|
[clang][OpeMP] Model OpenMP structured-block in AST (PR40563)
Summary:
https://www.openmp.org/wp-content/uploads/OpenMP-API-Specification-5.0.pdf, page 3:
```
structured block
For C/C++, an executable statement, possibly compound, with a single entry at the
top and a single exit at the bottom, or an OpenMP construct.
COMMENT: See Section 2.1 on page 38 for restrictions on structured
blocks.
```
```
2.1 Directive Format
Some executable directives include a structured block. A structured block:
• may contain infinite loops where the point of exit is never reached;
• may halt due to an IEEE exception;
• may contain calls to exit(), _Exit(), quick_exit(), abort() or functions with a
_Noreturn specifier (in C) or a noreturn attribute (in C/C++);
• may be an expression statement, iteration statement, selection statement, or try block, provided
that the corresponding compound statement obtained by enclosing it in { and } would be a
structured block; and
Restrictions
Restrictions to structured blocks are as follows:
• Entry to a structured block must not be the result of a branch.
• The point of exit cannot be a branch out of the structured block.
C / C++
• The point of entry to a structured block must not be a call to setjmp().
• longjmp() and throw() must not violate the entry/exit criteria.
```
Of particular note here is the fact that OpenMP structured blocks are as-if `noexcept`,
in the same sense as with the normal `noexcept` functions in C++.
I.e. if throw happens, and it attempts to travel out of the `noexcept` function
(here: out of the current structured-block), then the program terminates.
Now, one of course can say that since it is explicitly prohibited by the Specification,
then any and all programs that violate this Specification contain undefined behavior,
and are unspecified, and thus no one should care about them. Just don't write broken code /s
But i'm not sure this is a reasonable approach.
I have personally had oss-fuzz issues of this origin - exception thrown inside
of an OpenMP structured-block that is not caught, thus causing program termination.
This issue isn't all that hard to catch, it's not any particularly different from
diagnosing the same situation with the normal `noexcept` function.
Now, clang static analyzer does not presently model exceptions.
But clang-tidy has a simplisic [[ https://clang.llvm.org/extra/clang-tidy/checks/bugprone-exception-escape.html | bugprone-exception-escape ]] check,
and it is even refactored as a `ExceptionAnalyzer` class for reuse.
So it would be trivial to use that analyzer to check for
exceptions escaping out of OpenMP structured blocks. (D59466)
All that sounds too great to be true. Indeed, there is a caveat.
Presently, it's practically impossible to do. To check a OpenMP structured block
you need to somehow 'get' the OpenMP structured block, and you can't because
it's simply not modelled in AST. `CapturedStmt`/`CapturedDecl` is not it's representation.
Now, it is of course possible to write e.g. some AST matcher that would e.g.
match every OpenMP executable directive, and then return the whatever `Stmt` is
the structured block of said executable directive, if any.
But i said //practically//. This isn't practical for the following reasons:
1. This **will** bitrot. That matcher will need to be kept up-to-date,
and refreshed with every new OpenMP spec version.
2. Every single piece of code that would want that knowledge would need to
have such matcher. Well, okay, if it is an AST matcher, it could be shared.
But then you still have `RecursiveASTVisitor` and friends.
`2 > 1`, so now you have code duplication.
So it would be reasonable (and is fully within clang AST spirit) to not
force every single consumer to do that work, but instead store that knowledge
in the correct, and appropriate place - AST, class structure.
Now, there is another hoop we need to get through.
It isn't fully obvious //how// to model this.
The best solution would of course be to simply add a `OMPStructuredBlock` transparent
node. It would be optimal, it would give us two properties:
* Given this `OMPExecutableDirective`, what's it OpenMP structured block?
* It is trivial to check whether the `Stmt*` is a OpenMP structured block (`isa<OMPStructuredBlock>(ptr)`)
But OpenMP structured block isn't **necessarily** the first, direct child of `OMP*Directive`.
(even ignoring the clang's `CapturedStmt`/`CapturedDecl` that were inserted inbetween).
So i'm not sure whether or not we could re-create AST statements after they were already created?
There would be other costs to a new AST node: https://bugs.llvm.org/show_bug.cgi?id=40563#c12
```
1. You will need to break the representation of loops. The body should be replaced by the "structured block" entity.
2. You will need to support serialization/deserialization.
3. You will need to support template instantiation.
4. You will need to support codegen and take this new construct to account in each OpenMP directive.
```
Instead, there **is** an functionally-equivalent, alternative solution, consisting of two parts.
Part 1:
* Add a member function `isStandaloneDirective()` to the `OMPExecutableDirective` class,
that will tell whether this directive is stand-alone or not, as per the spec.
We need it because we can't just check for the existance of associated statements,
see code comment.
* Add a member function `getStructuredBlock()` to the OMPExecutableDirective` class itself,
that assert that this is not a stand-alone directive, and either return the correct loop body
if this is a loop-like directive, or the captured statement.
This way, given an `OMPExecutableDirective`, we can get it's structured block.
Also, since the knowledge is ingrained into the clang OpenMP implementation,
it will not cause any duplication, and //hopefully// won't bitrot.
Great we achieved 1 of 2 properties of `OMPStructuredBlock` approach.
Thus, there is a second part needed:
* How can we check whether a given `Stmt*` is `OMPStructuredBlock`?
Well, we can't really, in general. I can see this workaround:
```
class FunctionASTVisitor : public RecursiveASTVisitor<FunctionASTVisitor> {
using Base = RecursiveASTVisitor<FunctionASTVisitor>;
public:
bool VisitOMPExecDir(OMPExecDir *D) {
OmpStructuredStmts.emplace_back(D.getStructuredStmt());
}
bool VisitSOMETHINGELSE(???) {
if(InOmpStructuredStmt)
HI!
}
bool TraverseStmt(Stmt *Node) {
if (!Node)
return Base::TraverseStmt(Node);
if (OmpStructuredStmts.back() == Node)
++InOmpStructuredStmt;
Base::TraverseStmt(Node);
if (OmpStructuredStmts.back() == Node) {
OmpStructuredStmts.pop_back();
--InOmpStructuredStmt;
}
return true;
}
std::vector<Stmt*> OmpStructuredStmts;
int InOmpStructuredStmt = 0;
};
```
But i really don't see using it in practice.
It's just too intrusive; and again, requires knowledge duplication.
.. but no. The solution lies right on the ground.
Why don't we simply store this `i'm a openmp structured block` in the bitfield of the `Stmt` itself?
This does not appear to have any impact on the memory footprint of the clang AST,
since it's just a single extra bit in the bitfield. At least the static assertions don't fail.
Thus, indeed, we can achieve both of the properties without a new AST node.
We can cheaply set that bit right in sema, at the end of `Sema::ActOnOpenMPExecutableDirective()`,
by just calling the `getStructuredBlock()` that we just added.
Test coverage that demonstrates all this has been added.
This isn't as great with serialization though. Most of it does not use abbrevs,
so we do end up paying the full price (4 bytes?) instead of a single bit.
That price, of course, can be reclaimed by using abbrevs.
In fact, i suspect that //might// not just reclaim these bytes, but pack these PCH significantly.
I'm not seeing a third solution. If there is one, it would be interesting to hear about it.
("just don't write code that would require `isa<OMPStructuredBlock>(ptr)`" is not a solution.)
Fixes [[ https://bugs.llvm.org/show_bug.cgi?id=40563 | PR40563 ]].
Reviewers: ABataev, rjmccall, hfinkel, rsmith, riccibruno, gribozavr
Reviewed By: ABataev, gribozavr
Subscribers: mgorny, aaron.ballman, steveire, guansong, jfb, jdoerfert, cfe-commits
Tags: #clang, #openmp
Differential Revision: https://reviews.llvm.org/D59214
llvm-svn: 356570
2019-03-21 00:32:36 +08:00
|
|
|
static const unsigned NumStmtFields = 1;
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// The number of record fields required for the Expr class
|
2009-04-27 13:14:47 +08:00
|
|
|
/// itself.
|
2011-07-01 09:22:09 +08:00
|
|
|
static const unsigned NumExprFields = NumStmtFields + 7;
|
2012-01-27 17:46:47 +08:00
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Read and initialize a ExplicitTemplateArgumentList structure.
|
2012-01-27 17:46:47 +08:00
|
|
|
void ReadTemplateKWAndArgsInfo(ASTTemplateKWAndArgsInfo &Args,
|
2015-12-30 02:15:14 +08:00
|
|
|
TemplateArgumentLoc *ArgsLocArray,
|
2012-01-27 17:46:47 +08:00
|
|
|
unsigned NumTemplateArgs);
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Read and initialize a ExplicitTemplateArgumentList structure.
|
2011-09-23 04:07:03 +08:00
|
|
|
void ReadExplicitTemplateArgumentList(ASTTemplateArgumentListInfo &ArgList,
|
2010-06-29 06:28:35 +08:00
|
|
|
unsigned NumTemplateArgs);
|
|
|
|
|
|
|
|
void VisitStmt(Stmt *S);
|
2011-07-15 15:00:14 +08:00
|
|
|
#define STMT(Type, Base) \
|
|
|
|
void Visit##Type(Type *);
|
|
|
|
#include "clang/AST/StmtNodes.inc"
|
2009-04-27 13:14:47 +08:00
|
|
|
};
|
2018-04-12 04:57:28 +08:00
|
|
|
|
|
|
|
} // namespace clang
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2015-12-30 02:15:14 +08:00
|
|
|
void ASTStmtReader::ReadTemplateKWAndArgsInfo(ASTTemplateKWAndArgsInfo &Args,
|
|
|
|
TemplateArgumentLoc *ArgsLocArray,
|
|
|
|
unsigned NumTemplateArgs) {
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation TemplateKWLoc = ReadSourceLocation();
|
2010-06-28 17:31:48 +08:00
|
|
|
TemplateArgumentListInfo ArgInfo;
|
2016-12-16 04:53:26 +08:00
|
|
|
ArgInfo.setLAngleLoc(ReadSourceLocation());
|
|
|
|
ArgInfo.setRAngleLoc(ReadSourceLocation());
|
2010-06-28 17:31:48 +08:00
|
|
|
for (unsigned i = 0; i != NumTemplateArgs; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
ArgInfo.addArgument(Record.readTemplateArgumentLoc());
|
2015-12-30 02:15:14 +08:00
|
|
|
Args.initializeFrom(TemplateKWLoc, ArgInfo, ArgsLocArray);
|
2010-06-28 17:31:48 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitStmt(Stmt *S) {
|
[clang][OpeMP] Model OpenMP structured-block in AST (PR40563)
Summary:
https://www.openmp.org/wp-content/uploads/OpenMP-API-Specification-5.0.pdf, page 3:
```
structured block
For C/C++, an executable statement, possibly compound, with a single entry at the
top and a single exit at the bottom, or an OpenMP construct.
COMMENT: See Section 2.1 on page 38 for restrictions on structured
blocks.
```
```
2.1 Directive Format
Some executable directives include a structured block. A structured block:
• may contain infinite loops where the point of exit is never reached;
• may halt due to an IEEE exception;
• may contain calls to exit(), _Exit(), quick_exit(), abort() or functions with a
_Noreturn specifier (in C) or a noreturn attribute (in C/C++);
• may be an expression statement, iteration statement, selection statement, or try block, provided
that the corresponding compound statement obtained by enclosing it in { and } would be a
structured block; and
Restrictions
Restrictions to structured blocks are as follows:
• Entry to a structured block must not be the result of a branch.
• The point of exit cannot be a branch out of the structured block.
C / C++
• The point of entry to a structured block must not be a call to setjmp().
• longjmp() and throw() must not violate the entry/exit criteria.
```
Of particular note here is the fact that OpenMP structured blocks are as-if `noexcept`,
in the same sense as with the normal `noexcept` functions in C++.
I.e. if throw happens, and it attempts to travel out of the `noexcept` function
(here: out of the current structured-block), then the program terminates.
Now, one of course can say that since it is explicitly prohibited by the Specification,
then any and all programs that violate this Specification contain undefined behavior,
and are unspecified, and thus no one should care about them. Just don't write broken code /s
But i'm not sure this is a reasonable approach.
I have personally had oss-fuzz issues of this origin - exception thrown inside
of an OpenMP structured-block that is not caught, thus causing program termination.
This issue isn't all that hard to catch, it's not any particularly different from
diagnosing the same situation with the normal `noexcept` function.
Now, clang static analyzer does not presently model exceptions.
But clang-tidy has a simplisic [[ https://clang.llvm.org/extra/clang-tidy/checks/bugprone-exception-escape.html | bugprone-exception-escape ]] check,
and it is even refactored as a `ExceptionAnalyzer` class for reuse.
So it would be trivial to use that analyzer to check for
exceptions escaping out of OpenMP structured blocks. (D59466)
All that sounds too great to be true. Indeed, there is a caveat.
Presently, it's practically impossible to do. To check a OpenMP structured block
you need to somehow 'get' the OpenMP structured block, and you can't because
it's simply not modelled in AST. `CapturedStmt`/`CapturedDecl` is not it's representation.
Now, it is of course possible to write e.g. some AST matcher that would e.g.
match every OpenMP executable directive, and then return the whatever `Stmt` is
the structured block of said executable directive, if any.
But i said //practically//. This isn't practical for the following reasons:
1. This **will** bitrot. That matcher will need to be kept up-to-date,
and refreshed with every new OpenMP spec version.
2. Every single piece of code that would want that knowledge would need to
have such matcher. Well, okay, if it is an AST matcher, it could be shared.
But then you still have `RecursiveASTVisitor` and friends.
`2 > 1`, so now you have code duplication.
So it would be reasonable (and is fully within clang AST spirit) to not
force every single consumer to do that work, but instead store that knowledge
in the correct, and appropriate place - AST, class structure.
Now, there is another hoop we need to get through.
It isn't fully obvious //how// to model this.
The best solution would of course be to simply add a `OMPStructuredBlock` transparent
node. It would be optimal, it would give us two properties:
* Given this `OMPExecutableDirective`, what's it OpenMP structured block?
* It is trivial to check whether the `Stmt*` is a OpenMP structured block (`isa<OMPStructuredBlock>(ptr)`)
But OpenMP structured block isn't **necessarily** the first, direct child of `OMP*Directive`.
(even ignoring the clang's `CapturedStmt`/`CapturedDecl` that were inserted inbetween).
So i'm not sure whether or not we could re-create AST statements after they were already created?
There would be other costs to a new AST node: https://bugs.llvm.org/show_bug.cgi?id=40563#c12
```
1. You will need to break the representation of loops. The body should be replaced by the "structured block" entity.
2. You will need to support serialization/deserialization.
3. You will need to support template instantiation.
4. You will need to support codegen and take this new construct to account in each OpenMP directive.
```
Instead, there **is** an functionally-equivalent, alternative solution, consisting of two parts.
Part 1:
* Add a member function `isStandaloneDirective()` to the `OMPExecutableDirective` class,
that will tell whether this directive is stand-alone or not, as per the spec.
We need it because we can't just check for the existance of associated statements,
see code comment.
* Add a member function `getStructuredBlock()` to the OMPExecutableDirective` class itself,
that assert that this is not a stand-alone directive, and either return the correct loop body
if this is a loop-like directive, or the captured statement.
This way, given an `OMPExecutableDirective`, we can get it's structured block.
Also, since the knowledge is ingrained into the clang OpenMP implementation,
it will not cause any duplication, and //hopefully// won't bitrot.
Great we achieved 1 of 2 properties of `OMPStructuredBlock` approach.
Thus, there is a second part needed:
* How can we check whether a given `Stmt*` is `OMPStructuredBlock`?
Well, we can't really, in general. I can see this workaround:
```
class FunctionASTVisitor : public RecursiveASTVisitor<FunctionASTVisitor> {
using Base = RecursiveASTVisitor<FunctionASTVisitor>;
public:
bool VisitOMPExecDir(OMPExecDir *D) {
OmpStructuredStmts.emplace_back(D.getStructuredStmt());
}
bool VisitSOMETHINGELSE(???) {
if(InOmpStructuredStmt)
HI!
}
bool TraverseStmt(Stmt *Node) {
if (!Node)
return Base::TraverseStmt(Node);
if (OmpStructuredStmts.back() == Node)
++InOmpStructuredStmt;
Base::TraverseStmt(Node);
if (OmpStructuredStmts.back() == Node) {
OmpStructuredStmts.pop_back();
--InOmpStructuredStmt;
}
return true;
}
std::vector<Stmt*> OmpStructuredStmts;
int InOmpStructuredStmt = 0;
};
```
But i really don't see using it in practice.
It's just too intrusive; and again, requires knowledge duplication.
.. but no. The solution lies right on the ground.
Why don't we simply store this `i'm a openmp structured block` in the bitfield of the `Stmt` itself?
This does not appear to have any impact on the memory footprint of the clang AST,
since it's just a single extra bit in the bitfield. At least the static assertions don't fail.
Thus, indeed, we can achieve both of the properties without a new AST node.
We can cheaply set that bit right in sema, at the end of `Sema::ActOnOpenMPExecutableDirective()`,
by just calling the `getStructuredBlock()` that we just added.
Test coverage that demonstrates all this has been added.
This isn't as great with serialization though. Most of it does not use abbrevs,
so we do end up paying the full price (4 bytes?) instead of a single bit.
That price, of course, can be reclaimed by using abbrevs.
In fact, i suspect that //might// not just reclaim these bytes, but pack these PCH significantly.
I'm not seeing a third solution. If there is one, it would be interesting to hear about it.
("just don't write code that would require `isa<OMPStructuredBlock>(ptr)`" is not a solution.)
Fixes [[ https://bugs.llvm.org/show_bug.cgi?id=40563 | PR40563 ]].
Reviewers: ABataev, rjmccall, hfinkel, rsmith, riccibruno, gribozavr
Reviewed By: ABataev, gribozavr
Subscribers: mgorny, aaron.ballman, steveire, guansong, jfb, jdoerfert, cfe-commits
Tags: #clang, #openmp
Differential Revision: https://reviews.llvm.org/D59214
llvm-svn: 356570
2019-03-21 00:32:36 +08:00
|
|
|
S->setIsOMPStructuredBlock(Record.readInt());
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(Record.getIdx() == NumStmtFields && "Incorrect statement field count");
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitNullStmt(NullStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setSemiLoc(ReadSourceLocation());
|
[AST] Widen the bit-fields of Stmt to 8 bytes.
Although some classes are using the tail padding of Stmt, most of
them are not. In particular the expression classes are not using it
since there is Expr in between, and Expr contains a single pointer.
This patch widen the bit-fields to Stmt to 8 bytes and move some
data from NullStmt, CompoundStmt, LabelStmt, AttributedStmt, SwitchStmt,
WhileStmt, DoStmt, ForStmt, GotoStmt, ContinueStmt, BreakStmt
and ReturnStmt to the newly available space.
In itself this patch do not achieve much but I plan to go through each of
the classes in the statement/expression hierarchy and use this newly
available space. A quick estimation gives me that this should shrink the
size of the statement/expression hierarchy by >10% when parsing all of Boost.
Differential Revision: https://reviews.llvm.org/D53604
Reviewed By: rjmccall
llvm-svn: 345459
2018-10-28 02:43:27 +08:00
|
|
|
S->NullStmtBits.HasLeadingEmptyMacro = Record.readInt();
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCompoundStmt(CompoundStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<Stmt *, 16> Stmts;
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumStmts = Record.readInt();
|
2010-06-29 06:28:35 +08:00
|
|
|
while (NumStmts--)
|
2016-12-21 12:34:52 +08:00
|
|
|
Stmts.push_back(Record.readSubStmt());
|
2017-12-25 00:24:20 +08:00
|
|
|
S->setStmts(Stmts);
|
[AST] Widen the bit-fields of Stmt to 8 bytes.
Although some classes are using the tail padding of Stmt, most of
them are not. In particular the expression classes are not using it
since there is Expr in between, and Expr contains a single pointer.
This patch widen the bit-fields to Stmt to 8 bytes and move some
data from NullStmt, CompoundStmt, LabelStmt, AttributedStmt, SwitchStmt,
WhileStmt, DoStmt, ForStmt, GotoStmt, ContinueStmt, BreakStmt
and ReturnStmt to the newly available space.
In itself this patch do not achieve much but I plan to go through each of
the classes in the statement/expression hierarchy and use this newly
available space. A quick estimation gives me that this should shrink the
size of the statement/expression hierarchy by >10% when parsing all of Boost.
Differential Revision: https://reviews.llvm.org/D53604
Reviewed By: rjmccall
llvm-svn: 345459
2018-10-28 02:43:27 +08:00
|
|
|
S->CompoundStmtBits.LBraceLoc = ReadSourceLocation();
|
2016-12-16 04:53:26 +08:00
|
|
|
S->RBraceLoc = ReadSourceLocation();
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitSwitchCase(SwitchCase *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
Record.recordSwitchCaseID(S, Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setKeywordLoc(ReadSourceLocation());
|
|
|
|
S->setColonLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCaseStmt(CaseStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitSwitchCase(S);
|
2018-10-28 20:30:53 +08:00
|
|
|
bool CaseStmtIsGNURange = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setLHS(Record.readSubExpr());
|
|
|
|
S->setSubStmt(Record.readSubStmt());
|
2018-10-28 20:30:53 +08:00
|
|
|
if (CaseStmtIsGNURange) {
|
|
|
|
S->setRHS(Record.readSubExpr());
|
|
|
|
S->setEllipsisLoc(ReadSourceLocation());
|
|
|
|
}
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitDefaultStmt(DefaultStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitSwitchCase(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setSubStmt(Record.readSubStmt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitLabelStmt(LabelStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *LD = ReadDeclAs<LabelDecl>();
|
2011-02-17 15:39:24 +08:00
|
|
|
LD->setStmt(S);
|
|
|
|
S->setDecl(LD);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setSubStmt(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setIdentLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2012-04-14 08:33:13 +08:00
|
|
|
void ASTStmtReader::VisitAttributedStmt(AttributedStmt *S) {
|
|
|
|
VisitStmt(S);
|
[AST] Widen the bit-fields of Stmt to 8 bytes.
Although some classes are using the tail padding of Stmt, most of
them are not. In particular the expression classes are not using it
since there is Expr in between, and Expr contains a single pointer.
This patch widen the bit-fields to Stmt to 8 bytes and move some
data from NullStmt, CompoundStmt, LabelStmt, AttributedStmt, SwitchStmt,
WhileStmt, DoStmt, ForStmt, GotoStmt, ContinueStmt, BreakStmt
and ReturnStmt to the newly available space.
In itself this patch do not achieve much but I plan to go through each of
the classes in the statement/expression hierarchy and use this newly
available space. A quick estimation gives me that this should shrink the
size of the statement/expression hierarchy by >10% when parsing all of Boost.
Differential Revision: https://reviews.llvm.org/D53604
Reviewed By: rjmccall
llvm-svn: 345459
2018-10-28 02:43:27 +08:00
|
|
|
// NumAttrs in AttributedStmt is set when creating an empty
|
|
|
|
// AttributedStmt in AttributedStmt::CreateEmpty, since it is needed
|
|
|
|
// to allocate the right amount of space for the trailing Attr *.
|
2016-12-21 08:17:49 +08:00
|
|
|
uint64_t NumAttrs = Record.readInt();
|
2012-04-14 08:33:13 +08:00
|
|
|
AttrVec Attrs;
|
2016-12-21 12:34:52 +08:00
|
|
|
Record.readAttributes(Attrs);
|
2012-07-10 02:55:31 +08:00
|
|
|
(void)NumAttrs;
|
[AST] Widen the bit-fields of Stmt to 8 bytes.
Although some classes are using the tail padding of Stmt, most of
them are not. In particular the expression classes are not using it
since there is Expr in between, and Expr contains a single pointer.
This patch widen the bit-fields to Stmt to 8 bytes and move some
data from NullStmt, CompoundStmt, LabelStmt, AttributedStmt, SwitchStmt,
WhileStmt, DoStmt, ForStmt, GotoStmt, ContinueStmt, BreakStmt
and ReturnStmt to the newly available space.
In itself this patch do not achieve much but I plan to go through each of
the classes in the statement/expression hierarchy and use this newly
available space. A quick estimation gives me that this should shrink the
size of the statement/expression hierarchy by >10% when parsing all of Boost.
Differential Revision: https://reviews.llvm.org/D53604
Reviewed By: rjmccall
llvm-svn: 345459
2018-10-28 02:43:27 +08:00
|
|
|
assert(NumAttrs == S->AttributedStmtBits.NumAttrs);
|
2012-07-09 18:04:07 +08:00
|
|
|
assert(NumAttrs == Attrs.size());
|
2014-05-13 22:55:01 +08:00
|
|
|
std::copy(Attrs.begin(), Attrs.end(), S->getAttrArrayPtr());
|
2016-12-21 12:34:52 +08:00
|
|
|
S->SubStmt = Record.readSubStmt();
|
[AST] Widen the bit-fields of Stmt to 8 bytes.
Although some classes are using the tail padding of Stmt, most of
them are not. In particular the expression classes are not using it
since there is Expr in between, and Expr contains a single pointer.
This patch widen the bit-fields to Stmt to 8 bytes and move some
data from NullStmt, CompoundStmt, LabelStmt, AttributedStmt, SwitchStmt,
WhileStmt, DoStmt, ForStmt, GotoStmt, ContinueStmt, BreakStmt
and ReturnStmt to the newly available space.
In itself this patch do not achieve much but I plan to go through each of
the classes in the statement/expression hierarchy and use this newly
available space. A quick estimation gives me that this should shrink the
size of the statement/expression hierarchy by >10% when parsing all of Boost.
Differential Revision: https://reviews.llvm.org/D53604
Reviewed By: rjmccall
llvm-svn: 345459
2018-10-28 02:43:27 +08:00
|
|
|
S->AttributedStmtBits.AttrLoc = ReadSourceLocation();
|
2012-04-14 08:33:13 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitIfStmt(IfStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2018-10-28 05:12:20 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
S->setConstexpr(Record.readInt());
|
2018-10-28 05:12:20 +08:00
|
|
|
bool HasElse = Record.readInt();
|
|
|
|
bool HasVar = Record.readInt();
|
|
|
|
bool HasInit = Record.readInt();
|
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCond(Record.readSubExpr());
|
|
|
|
S->setThen(Record.readSubStmt());
|
2018-10-28 05:12:20 +08:00
|
|
|
if (HasElse)
|
|
|
|
S->setElse(Record.readSubStmt());
|
|
|
|
if (HasVar)
|
|
|
|
S->setConditionVariable(Record.getContext(), ReadDeclAs<VarDecl>());
|
|
|
|
if (HasInit)
|
|
|
|
S->setInit(Record.readSubStmt());
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setIfLoc(ReadSourceLocation());
|
2018-10-28 05:12:20 +08:00
|
|
|
if (HasElse)
|
|
|
|
S->setElseLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitSwitchStmt(SwitchStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2018-10-30 00:12:37 +08:00
|
|
|
|
|
|
|
bool HasInit = Record.readInt();
|
|
|
|
bool HasVar = Record.readInt();
|
|
|
|
bool AllEnumCasesCovered = Record.readInt();
|
|
|
|
if (AllEnumCasesCovered)
|
|
|
|
S->setAllEnumCasesCovered();
|
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCond(Record.readSubExpr());
|
|
|
|
S->setBody(Record.readSubStmt());
|
2018-10-30 00:12:37 +08:00
|
|
|
if (HasInit)
|
|
|
|
S->setInit(Record.readSubStmt());
|
|
|
|
if (HasVar)
|
|
|
|
S->setConditionVariable(Record.getContext(), ReadDeclAs<VarDecl>());
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setSwitchLoc(ReadSourceLocation());
|
2010-09-09 08:05:53 +08:00
|
|
|
|
2014-05-22 13:54:18 +08:00
|
|
|
SwitchCase *PrevSC = nullptr;
|
2016-12-21 08:17:49 +08:00
|
|
|
for (auto E = Record.size(); Record.getIdx() != E; ) {
|
|
|
|
SwitchCase *SC = Record.getSwitchCaseWithID(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
if (PrevSC)
|
|
|
|
PrevSC->setNextSwitchCase(SC);
|
|
|
|
else
|
|
|
|
S->setSwitchCaseList(SC);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
PrevSC = SC;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitWhileStmt(WhileStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2018-10-30 21:42:41 +08:00
|
|
|
|
|
|
|
bool HasVar = Record.readInt();
|
2011-07-22 06:35:25 +08:00
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCond(Record.readSubExpr());
|
|
|
|
S->setBody(Record.readSubStmt());
|
2018-10-30 21:42:41 +08:00
|
|
|
if (HasVar)
|
|
|
|
S->setConditionVariable(Record.getContext(), ReadDeclAs<VarDecl>());
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setWhileLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitDoStmt(DoStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCond(Record.readSubExpr());
|
|
|
|
S->setBody(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setDoLoc(ReadSourceLocation());
|
|
|
|
S->setWhileLoc(ReadSourceLocation());
|
|
|
|
S->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitForStmt(ForStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setInit(Record.readSubStmt());
|
|
|
|
S->setCond(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setConditionVariable(Record.getContext(), ReadDeclAs<VarDecl>());
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setInc(Record.readSubExpr());
|
|
|
|
S->setBody(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setForLoc(ReadSourceLocation());
|
|
|
|
S->setLParenLoc(ReadSourceLocation());
|
|
|
|
S->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitGotoStmt(GotoStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setLabel(ReadDeclAs<LabelDecl>());
|
|
|
|
S->setGotoLoc(ReadSourceLocation());
|
|
|
|
S->setLabelLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitIndirectGotoStmt(IndirectGotoStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setGotoLoc(ReadSourceLocation());
|
|
|
|
S->setStarLoc(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setTarget(Record.readSubExpr());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitContinueStmt(ContinueStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setContinueLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitBreakStmt(BreakStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setBreakLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitReturnStmt(ReturnStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2018-10-30 22:40:49 +08:00
|
|
|
|
|
|
|
bool HasNRVOCandidate = Record.readInt();
|
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setRetValue(Record.readSubExpr());
|
2018-10-30 22:40:49 +08:00
|
|
|
if (HasNRVOCandidate)
|
|
|
|
S->setNRVOCandidate(ReadDeclAs<VarDecl>());
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setReturnLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitDeclStmt(DeclStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setStartLoc(ReadSourceLocation());
|
|
|
|
S->setEndLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
if (Record.size() - Record.getIdx() == 1) {
|
2009-04-27 13:14:47 +08:00
|
|
|
// Single declaration
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setDeclGroup(DeclGroupRef(ReadDecl()));
|
2009-04-27 13:14:47 +08:00
|
|
|
} else {
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<Decl *, 16> Decls;
|
2016-12-21 08:17:49 +08:00
|
|
|
int N = Record.size() - Record.getIdx();
|
|
|
|
Decls.reserve(N);
|
|
|
|
for (int I = 0; I < N; ++I)
|
2016-12-16 04:53:26 +08:00
|
|
|
Decls.push_back(ReadDecl());
|
|
|
|
S->setDeclGroup(DeclGroupRef(DeclGroup::Create(Record.getContext(),
|
2009-05-23 06:45:36 +08:00
|
|
|
Decls.data(),
|
|
|
|
Decls.size())));
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-03 08:10:13 +08:00
|
|
|
void ASTStmtReader::VisitAsmStmt(AsmStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 08:17:49 +08:00
|
|
|
S->NumOutputs = Record.readInt();
|
|
|
|
S->NumInputs = Record.readInt();
|
|
|
|
S->NumClobbers = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setAsmLoc(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
S->setVolatile(Record.readInt());
|
|
|
|
S->setSimple(Record.readInt());
|
2013-05-03 08:10:13 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2013-05-03 08:10:13 +08:00
|
|
|
void ASTStmtReader::VisitGCCAsmStmt(GCCAsmStmt *S) {
|
|
|
|
VisitAsmStmt(S);
|
2019-06-03 23:57:25 +08:00
|
|
|
S->NumLabels = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setRParenLoc(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setAsmString(cast_or_null<StringLiteral>(Record.readSubStmt()));
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2013-05-03 08:10:13 +08:00
|
|
|
unsigned NumOutputs = S->getNumOutputs();
|
|
|
|
unsigned NumInputs = S->getNumInputs();
|
|
|
|
unsigned NumClobbers = S->getNumClobbers();
|
2019-06-03 23:57:25 +08:00
|
|
|
unsigned NumLabels = S->getNumLabels();
|
2013-05-03 08:10:13 +08:00
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
// Outputs and inputs
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<IdentifierInfo *, 16> Names;
|
|
|
|
SmallVector<StringLiteral*, 16> Constraints;
|
|
|
|
SmallVector<Stmt*, 16> Exprs;
|
2009-04-27 13:14:47 +08:00
|
|
|
for (unsigned I = 0, N = NumOutputs + NumInputs; I != N; ++I) {
|
2016-12-21 12:34:52 +08:00
|
|
|
Names.push_back(Record.getIdentifierInfo());
|
|
|
|
Constraints.push_back(cast_or_null<StringLiteral>(Record.readSubStmt()));
|
|
|
|
Exprs.push_back(Record.readSubStmt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Constraints
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<StringLiteral*, 16> Clobbers;
|
2009-04-27 13:14:47 +08:00
|
|
|
for (unsigned I = 0; I != NumClobbers; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
Clobbers.push_back(cast_or_null<StringLiteral>(Record.readSubStmt()));
|
2010-01-31 03:34:25 +08:00
|
|
|
|
2019-06-03 23:57:25 +08:00
|
|
|
// Labels
|
|
|
|
for (unsigned I = 0, N = NumLabels; I != N; ++I)
|
|
|
|
Exprs.push_back(Record.readSubStmt());
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setOutputsAndInputsAndClobbers(Record.getContext(),
|
|
|
|
Names.data(), Constraints.data(),
|
|
|
|
Exprs.data(), NumOutputs, NumInputs,
|
2019-06-03 23:57:25 +08:00
|
|
|
NumLabels,
|
2010-01-31 03:34:25 +08:00
|
|
|
Clobbers.data(), NumClobbers);
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2012-06-12 04:47:18 +08:00
|
|
|
void ASTStmtReader::VisitMSAsmStmt(MSAsmStmt *S) {
|
2013-05-03 08:10:13 +08:00
|
|
|
VisitAsmStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->LBraceLoc = ReadSourceLocation();
|
|
|
|
S->EndLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
S->NumAsmToks = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
std::string AsmStr = ReadString();
|
2013-05-03 08:10:13 +08:00
|
|
|
|
|
|
|
// Read the tokens.
|
|
|
|
SmallVector<Token, 16> AsmToks;
|
|
|
|
AsmToks.reserve(S->NumAsmToks);
|
|
|
|
for (unsigned i = 0, e = S->NumAsmToks; i != e; ++i) {
|
2016-12-21 12:34:52 +08:00
|
|
|
AsmToks.push_back(Record.readToken());
|
2013-05-03 08:10:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// The calls to reserve() for the FooData vectors are mandatory to
|
|
|
|
// prevent dead StringRefs in the Foo vectors.
|
|
|
|
|
|
|
|
// Read the clobbers.
|
|
|
|
SmallVector<std::string, 16> ClobbersData;
|
|
|
|
SmallVector<StringRef, 16> Clobbers;
|
|
|
|
ClobbersData.reserve(S->NumClobbers);
|
|
|
|
Clobbers.reserve(S->NumClobbers);
|
|
|
|
for (unsigned i = 0, e = S->NumClobbers; i != e; ++i) {
|
2016-12-16 04:53:26 +08:00
|
|
|
ClobbersData.push_back(ReadString());
|
2013-05-03 08:10:13 +08:00
|
|
|
Clobbers.push_back(ClobbersData.back());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the operands.
|
|
|
|
unsigned NumOperands = S->NumOutputs + S->NumInputs;
|
|
|
|
SmallVector<Expr*, 16> Exprs;
|
|
|
|
SmallVector<std::string, 16> ConstraintsData;
|
|
|
|
SmallVector<StringRef, 16> Constraints;
|
|
|
|
Exprs.reserve(NumOperands);
|
|
|
|
ConstraintsData.reserve(NumOperands);
|
|
|
|
Constraints.reserve(NumOperands);
|
|
|
|
for (unsigned i = 0; i != NumOperands; ++i) {
|
2016-12-21 12:34:52 +08:00
|
|
|
Exprs.push_back(cast<Expr>(Record.readSubStmt()));
|
2016-12-16 04:53:26 +08:00
|
|
|
ConstraintsData.push_back(ReadString());
|
2013-05-03 08:10:13 +08:00
|
|
|
Constraints.push_back(ConstraintsData.back());
|
|
|
|
}
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
S->initialize(Record.getContext(), AsmStr, AsmToks,
|
2013-05-03 08:10:13 +08:00
|
|
|
Constraints, Exprs, Clobbers);
|
2012-06-12 04:47:18 +08:00
|
|
|
}
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
void ASTStmtReader::VisitCoroutineBodyStmt(CoroutineBodyStmt *S) {
|
2017-07-26 02:01:49 +08:00
|
|
|
VisitStmt(S);
|
|
|
|
assert(Record.peekInt() == S->NumParams);
|
|
|
|
Record.skipInts(1);
|
|
|
|
auto *StoredStmts = S->getStoredStmts();
|
|
|
|
for (unsigned i = 0;
|
|
|
|
i < CoroutineBodyStmt::SubStmt::FirstParamMove + S->NumParams; ++i)
|
|
|
|
StoredStmts[i] = Record.readSubStmt();
|
2015-10-27 14:02:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitCoreturnStmt(CoreturnStmt *S) {
|
2017-07-26 02:01:49 +08:00
|
|
|
VisitStmt(S);
|
|
|
|
S->CoreturnLoc = Record.readSourceLocation();
|
|
|
|
for (auto &SubStmt: S->SubStmts)
|
|
|
|
SubStmt = Record.readSubStmt();
|
|
|
|
S->IsImplicit = Record.readInt() != 0;
|
2015-10-27 14:02:45 +08:00
|
|
|
}
|
|
|
|
|
2017-07-26 02:01:49 +08:00
|
|
|
void ASTStmtReader::VisitCoawaitExpr(CoawaitExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
E->KeywordLoc = ReadSourceLocation();
|
|
|
|
for (auto &SubExpr: E->SubExprs)
|
|
|
|
SubExpr = Record.readSubStmt();
|
|
|
|
E->OpaqueValue = cast_or_null<OpaqueValueExpr>(Record.readSubStmt());
|
|
|
|
E->setIsImplicit(Record.readInt() != 0);
|
2017-03-07 07:38:15 +08:00
|
|
|
}
|
|
|
|
|
2017-07-26 02:01:49 +08:00
|
|
|
void ASTStmtReader::VisitCoyieldExpr(CoyieldExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
E->KeywordLoc = ReadSourceLocation();
|
|
|
|
for (auto &SubExpr: E->SubExprs)
|
|
|
|
SubExpr = Record.readSubStmt();
|
|
|
|
E->OpaqueValue = cast_or_null<OpaqueValueExpr>(Record.readSubStmt());
|
2015-10-27 14:02:45 +08:00
|
|
|
}
|
|
|
|
|
2017-07-26 02:01:49 +08:00
|
|
|
void ASTStmtReader::VisitDependentCoawaitExpr(DependentCoawaitExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
E->KeywordLoc = ReadSourceLocation();
|
|
|
|
for (auto &SubExpr: E->SubExprs)
|
|
|
|
SubExpr = Record.readSubStmt();
|
2015-10-27 14:02:45 +08:00
|
|
|
}
|
|
|
|
|
2013-04-17 02:53:08 +08:00
|
|
|
void ASTStmtReader::VisitCapturedStmt(CapturedStmt *S) {
|
2013-05-04 03:20:19 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setCapturedDecl(ReadDeclAs<CapturedDecl>());
|
2016-12-21 08:17:49 +08:00
|
|
|
S->setCapturedRegionKind(static_cast<CapturedRegionKind>(Record.readInt()));
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setCapturedRecordDecl(ReadDeclAs<RecordDecl>());
|
2013-05-04 03:20:19 +08:00
|
|
|
|
|
|
|
// Capture inits
|
|
|
|
for (CapturedStmt::capture_init_iterator I = S->capture_init_begin(),
|
|
|
|
E = S->capture_init_end();
|
|
|
|
I != E; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
*I = Record.readSubExpr();
|
2013-05-04 03:20:19 +08:00
|
|
|
|
|
|
|
// Body
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCapturedStmt(Record.readSubStmt());
|
2013-05-04 11:59:06 +08:00
|
|
|
S->getCapturedDecl()->setBody(S->getCapturedStmt());
|
2013-05-04 03:20:19 +08:00
|
|
|
|
|
|
|
// Captures
|
2014-03-15 02:08:33 +08:00
|
|
|
for (auto &I : S->captures()) {
|
2016-12-16 04:53:26 +08:00
|
|
|
I.VarAndKind.setPointer(ReadDeclAs<VarDecl>());
|
2016-12-21 08:17:49 +08:00
|
|
|
I.VarAndKind.setInt(
|
|
|
|
static_cast<CapturedStmt::VariableCaptureKind>(Record.readInt()));
|
2016-12-16 04:53:26 +08:00
|
|
|
I.Loc = ReadSourceLocation();
|
2013-05-04 03:20:19 +08:00
|
|
|
}
|
2013-04-17 02:53:08 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitExpr(Expr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setType(Record.readType());
|
|
|
|
E->setTypeDependent(Record.readInt());
|
|
|
|
E->setValueDependent(Record.readInt());
|
|
|
|
E->setInstantiationDependent(Record.readInt());
|
|
|
|
E->ExprBits.ContainsUnexpandedParameterPack = Record.readInt();
|
|
|
|
E->setValueKind(static_cast<ExprValueKind>(Record.readInt()));
|
|
|
|
E->setObjectKind(static_cast<ExprObjectKind>(Record.readInt()));
|
|
|
|
assert(Record.getIdx() == NumExprFields &&
|
|
|
|
"Incorrect expression field count");
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2018-10-31 11:48:47 +08:00
|
|
|
void ASTStmtReader::VisitConstantExpr(ConstantExpr *E) {
|
|
|
|
VisitExpr(E);
|
[clang] Add storage for APValue in ConstantExpr
Summary:
When using ConstantExpr we often need the result of the expression to be kept in the AST. Currently this is done on a by the node that needs the result and has been done multiple times for enumerator, for constexpr variables... . This patch adds to ConstantExpr the ability to store the result of evaluating the expression. no functional changes expected.
Changes:
- Add trailling object to ConstantExpr that can hold an APValue or an uint64_t. the uint64_t is here because most ConstantExpr yield integral values so there is an optimized layout for integral values.
- Add basic* serialization support for the trailing result.
- Move conversion functions from an enum to a fltSemantics from clang::FloatingLiteral to llvm::APFloatBase. this change is to make it usable for serializing APValues.
- Add basic* Import support for the trailing result.
- ConstantExpr created in CheckConvertedConstantExpression now stores the result in the ConstantExpr Node.
- Adapt AST dump to print the result when present.
basic* : None, Indeterminate, Int, Float, FixedPoint, ComplexInt, ComplexFloat,
the result is not yet used anywhere but for -ast-dump.
Reviewers: rsmith, martong, shafik
Reviewed By: rsmith
Subscribers: rnkovacs, hiraditya, dexonsmith, cfe-commits, llvm-commits
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D62399
llvm-svn: 363493
2019-06-15 18:24:47 +08:00
|
|
|
E->ConstantExprBits.ResultKind = Record.readInt();
|
|
|
|
switch (E->ConstantExprBits.ResultKind) {
|
|
|
|
case ConstantExpr::RSK_Int64: {
|
|
|
|
E->Int64Result() = Record.readInt();
|
|
|
|
uint64_t tmp = Record.readInt();
|
|
|
|
E->ConstantExprBits.IsUnsigned = tmp & 0x1;
|
|
|
|
E->ConstantExprBits.BitWidth = tmp >> 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ConstantExpr::RSK_APValue:
|
|
|
|
E->APValueResult() = Record.readAPValue();
|
|
|
|
}
|
2018-10-31 11:48:47 +08:00
|
|
|
E->setSubExpr(Record.readSubExpr());
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitPredefinedExpr(PredefinedExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2018-10-28 03:21:19 +08:00
|
|
|
bool HasFunctionName = Record.readInt();
|
|
|
|
E->PredefinedExprBits.HasFunctionName = HasFunctionName;
|
|
|
|
E->PredefinedExprBits.Kind = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2018-10-28 03:21:19 +08:00
|
|
|
if (HasFunctionName)
|
|
|
|
E->setFunctionName(cast<StringLiteral>(Record.readSubExpr()));
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitDeclRefExpr(DeclRefExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2010-07-08 21:09:47 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
E->DeclRefExprBits.HasQualifier = Record.readInt();
|
|
|
|
E->DeclRefExprBits.HasFoundDecl = Record.readInt();
|
|
|
|
E->DeclRefExprBits.HasTemplateKWAndArgsInfo = Record.readInt();
|
|
|
|
E->DeclRefExprBits.HadMultipleCandidates = Record.readInt();
|
|
|
|
E->DeclRefExprBits.RefersToEnclosingVariableOrCapture = Record.readInt();
|
2019-06-12 01:50:32 +08:00
|
|
|
E->DeclRefExprBits.NonOdrUseReason = Record.readInt();
|
2011-03-07 02:19:42 +08:00
|
|
|
unsigned NumTemplateArgs = 0;
|
2012-01-27 17:46:47 +08:00
|
|
|
if (E->hasTemplateKWAndArgsInfo())
|
2016-12-21 08:17:49 +08:00
|
|
|
NumTemplateArgs = Record.readInt();
|
2011-03-07 02:19:42 +08:00
|
|
|
|
2011-05-02 05:29:53 +08:00
|
|
|
if (E->hasQualifier())
|
2015-12-30 02:15:14 +08:00
|
|
|
new (E->getTrailingObjects<NestedNameSpecifierLoc>())
|
2016-12-21 12:34:52 +08:00
|
|
|
NestedNameSpecifierLoc(Record.readNestedNameSpecifierLoc());
|
2010-07-08 21:09:47 +08:00
|
|
|
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
if (E->hasFoundDecl())
|
2016-12-16 04:53:26 +08:00
|
|
|
*E->getTrailingObjects<NamedDecl *>() = ReadDeclAs<NamedDecl>();
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
|
2012-01-27 17:46:47 +08:00
|
|
|
if (E->hasTemplateKWAndArgsInfo())
|
2015-12-30 02:15:14 +08:00
|
|
|
ReadTemplateKWAndArgsInfo(
|
|
|
|
*E->getTrailingObjects<ASTTemplateKWAndArgsInfo>(),
|
|
|
|
E->getTrailingObjects<TemplateArgumentLoc>(), NumTemplateArgs);
|
2010-07-08 21:09:47 +08:00
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setDecl(ReadDeclAs<ValueDecl>());
|
|
|
|
E->setLocation(ReadSourceLocation());
|
|
|
|
ReadDeclarationNameLoc(E->DNLoc, E->getDecl()->getDeclName());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitIntegerLiteral(IntegerLiteral *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setValue(Record.getContext(), Record.readAPInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2018-06-21 01:19:40 +08:00
|
|
|
void ASTStmtReader::VisitFixedPointLiteral(FixedPointLiteral *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
E->setLocation(ReadSourceLocation());
|
|
|
|
E->setValue(Record.getContext(), Record.readAPInt());
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitFloatingLiteral(FloatingLiteral *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
[clang] Add storage for APValue in ConstantExpr
Summary:
When using ConstantExpr we often need the result of the expression to be kept in the AST. Currently this is done on a by the node that needs the result and has been done multiple times for enumerator, for constexpr variables... . This patch adds to ConstantExpr the ability to store the result of evaluating the expression. no functional changes expected.
Changes:
- Add trailling object to ConstantExpr that can hold an APValue or an uint64_t. the uint64_t is here because most ConstantExpr yield integral values so there is an optimized layout for integral values.
- Add basic* serialization support for the trailing result.
- Move conversion functions from an enum to a fltSemantics from clang::FloatingLiteral to llvm::APFloatBase. this change is to make it usable for serializing APValues.
- Add basic* Import support for the trailing result.
- ConstantExpr created in CheckConvertedConstantExpression now stores the result in the ConstantExpr Node.
- Adapt AST dump to print the result when present.
basic* : None, Indeterminate, Int, Float, FixedPoint, ComplexInt, ComplexFloat,
the result is not yet used anywhere but for -ast-dump.
Reviewers: rsmith, martong, shafik
Reviewed By: rsmith
Subscribers: rnkovacs, hiraditya, dexonsmith, cfe-commits, llvm-commits
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D62399
llvm-svn: 363493
2019-06-15 18:24:47 +08:00
|
|
|
E->setRawSemantics(
|
|
|
|
static_cast<llvm::APFloatBase::Semantics>(Record.readInt()));
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setExact(Record.readInt());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setValue(Record.getContext(), Record.readAPFloat(E->getSemantics()));
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitImaginaryLiteral(ImaginaryLiteral *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSubExpr(Record.readSubExpr());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitStringLiteral(StringLiteral *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
|
|
|
|
2018-11-16 01:31:16 +08:00
|
|
|
// NumConcatenated, Length and CharByteWidth are set by the empty
|
|
|
|
// ctor since they are needed to allocate storage for the trailing objects.
|
|
|
|
unsigned NumConcatenated = Record.readInt();
|
|
|
|
unsigned Length = Record.readInt();
|
|
|
|
unsigned CharByteWidth = Record.readInt();
|
|
|
|
assert((NumConcatenated == E->getNumConcatenated()) &&
|
|
|
|
"Wrong number of concatenated tokens!");
|
|
|
|
assert((Length == E->getLength()) && "Wrong Length!");
|
|
|
|
assert((CharByteWidth == E->getCharByteWidth()) && "Wrong character width!");
|
|
|
|
E->StringLiteralBits.Kind = Record.readInt();
|
|
|
|
E->StringLiteralBits.IsPascal = Record.readInt();
|
|
|
|
|
|
|
|
// The character width is originally computed via mapCharByteWidth.
|
|
|
|
// Check that the deserialized character width is consistant with the result
|
|
|
|
// of calling mapCharByteWidth.
|
|
|
|
assert((CharByteWidth ==
|
|
|
|
StringLiteral::mapCharByteWidth(Record.getContext().getTargetInfo(),
|
|
|
|
E->getKind())) &&
|
|
|
|
"Wrong character width!");
|
|
|
|
|
|
|
|
// Deserialize the trailing array of SourceLocation.
|
|
|
|
for (unsigned I = 0; I < NumConcatenated; ++I)
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setStrTokenLoc(I, ReadSourceLocation());
|
2018-11-16 01:31:16 +08:00
|
|
|
|
|
|
|
// Deserialize the trailing array of char holding the string data.
|
|
|
|
char *StrData = E->getStrDataAsChar();
|
|
|
|
for (unsigned I = 0; I < Length * CharByteWidth; ++I)
|
|
|
|
StrData[I] = Record.readInt();
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCharacterLiteral(CharacterLiteral *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setValue(Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setKind(static_cast<CharacterLiteral::CharacterKind>(Record.readInt()));
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitParenExpr(ParenExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLParen(ReadSourceLocation());
|
|
|
|
E->setRParen(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSubExpr(Record.readSubExpr());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitParenListExpr(ParenListExpr *E) {
|
2010-06-30 16:49:18 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumExprs = Record.readInt();
|
2018-11-21 00:20:40 +08:00
|
|
|
assert((NumExprs == E->getNumExprs()) && "Wrong NumExprs!");
|
|
|
|
for (unsigned I = 0; I != NumExprs; ++I)
|
|
|
|
E->getTrailingObjects<Stmt *>()[I] = Record.readSubStmt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->LParenLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
2010-06-30 16:49:18 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitUnaryOperator(UnaryOperator *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2018-04-12 04:57:28 +08:00
|
|
|
E->setSubExpr(Record.readSubExpr());
|
|
|
|
E->setOpcode((UnaryOperator::Opcode)Record.readInt());
|
|
|
|
E->setOperatorLoc(ReadSourceLocation());
|
|
|
|
E->setCanOverflow(Record.readInt());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitOffsetOfExpr(OffsetOfExpr *E) {
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(E->getNumComponents() == Record.peekInt());
|
|
|
|
Record.skipInts(1);
|
|
|
|
assert(E->getNumExpressions() == Record.peekInt());
|
|
|
|
Record.skipInts(1);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setOperatorLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
|
|
|
E->setTypeSourceInfo(GetTypeSourceInfo());
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
for (unsigned I = 0, N = E->getNumComponents(); I != N; ++I) {
|
2018-04-12 04:57:28 +08:00
|
|
|
auto Kind = static_cast<OffsetOfNode::Kind>(Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation Start = ReadSourceLocation();
|
|
|
|
SourceLocation End = ReadSourceLocation();
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
switch (Kind) {
|
2015-12-30 06:31:18 +08:00
|
|
|
case OffsetOfNode::Array:
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setComponent(I, OffsetOfNode(Start, Record.readInt(), End));
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
break;
|
2015-12-30 06:31:18 +08:00
|
|
|
|
|
|
|
case OffsetOfNode::Field:
|
|
|
|
E->setComponent(
|
2016-12-16 04:53:26 +08:00
|
|
|
I, OffsetOfNode(Start, ReadDeclAs<FieldDecl>(), End));
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
break;
|
|
|
|
|
2015-12-30 06:31:18 +08:00
|
|
|
case OffsetOfNode::Identifier:
|
|
|
|
E->setComponent(
|
|
|
|
I,
|
2016-12-21 12:34:52 +08:00
|
|
|
OffsetOfNode(Start, Record.getIdentifierInfo(), End));
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
break;
|
2015-12-30 06:31:18 +08:00
|
|
|
|
|
|
|
case OffsetOfNode::Base: {
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *Base = new (Record.getContext()) CXXBaseSpecifier();
|
2016-12-21 12:34:52 +08:00
|
|
|
*Base = Record.readCXXBaseSpecifier();
|
2015-12-30 06:31:18 +08:00
|
|
|
E->setComponent(I, OffsetOfNode(Base));
|
2010-04-29 08:18:15 +08:00
|
|
|
break;
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
}
|
2010-07-30 02:16:10 +08:00
|
|
|
}
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
for (unsigned I = 0, N = E->getNumExpressions(); I != N; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setIndexExpr(I, Record.readSubExpr());
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
}
|
|
|
|
|
2011-03-12 03:24:49 +08:00
|
|
|
void ASTStmtReader::VisitUnaryExprOrTypeTraitExpr(UnaryExprOrTypeTraitExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setKind(static_cast<UnaryExprOrTypeTrait>(Record.readInt()));
|
|
|
|
if (Record.peekInt() == 0) {
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setArgument(Record.readSubExpr());
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2009-04-27 13:14:47 +08:00
|
|
|
} else {
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setArgument(GetTypeSourceInfo());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setOperatorLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2019-10-15 23:24:26 +08:00
|
|
|
void ASTStmtReader::VisitConceptSpecializationExpr(
|
|
|
|
ConceptSpecializationExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
unsigned NumTemplateArgs = Record.readInt();
|
|
|
|
E->NestedNameSpec = Record.readNestedNameSpecifierLoc();
|
|
|
|
E->TemplateKWLoc = Record.readSourceLocation();
|
|
|
|
E->ConceptNameLoc = Record.readSourceLocation();
|
|
|
|
E->FoundDecl = ReadDeclAs<NamedDecl>();
|
2019-10-29 05:36:31 +08:00
|
|
|
E->NamedConcept.setPointer(ReadDeclAs<ConceptDecl>());
|
2019-10-15 23:24:26 +08:00
|
|
|
const ASTTemplateArgumentListInfo *ArgsAsWritten =
|
|
|
|
Record.readASTTemplateArgumentListInfo();
|
|
|
|
llvm::SmallVector<TemplateArgument, 4> Args;
|
|
|
|
for (unsigned I = 0; I < NumTemplateArgs; ++I)
|
|
|
|
Args.push_back(Record.readTemplateArgument());
|
|
|
|
E->setTemplateArguments(ArgsAsWritten, Args);
|
2019-10-29 05:36:31 +08:00
|
|
|
E->NamedConcept.setInt(Record.readInt() == 1);
|
2019-10-15 23:24:26 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitArraySubscriptExpr(ArraySubscriptExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setLHS(Record.readSubExpr());
|
|
|
|
E->setRHS(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setRBracketLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2015-08-25 22:24:04 +08:00
|
|
|
void ASTStmtReader::VisitOMPArraySectionExpr(OMPArraySectionExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
|
|
|
E->setLowerBound(Record.readSubExpr());
|
|
|
|
E->setLength(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setColonLoc(ReadSourceLocation());
|
|
|
|
E->setRBracketLoc(ReadSourceLocation());
|
2015-08-25 22:24:04 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCallExpr(CallExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2018-12-03 22:54:03 +08:00
|
|
|
unsigned NumArgs = Record.readInt();
|
|
|
|
assert((NumArgs == E->getNumArgs()) && "Wrong NumArgs!");
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setCallee(Record.readSubExpr());
|
2018-12-03 22:54:03 +08:00
|
|
|
for (unsigned I = 0; I != NumArgs; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setArg(I, Record.readSubExpr());
|
[AST] Store "UsesADL" information in CallExpr.
Summary:
Currently the Clang AST doesn't store information about how the callee of a CallExpr was found. Specifically if it was found using ADL.
However, this information is invaluable to tooling. Consider a tool which renames usages of a function. If the originally CallExpr was formed using ADL, then the tooling may need to additionally qualify the replacement.
Without information about how the callee was found, the tooling is left scratching it's head. Additionally, we want to be able to match ADL calls as quickly as possible, which means avoiding computing the answer on the fly.
This patch changes `CallExpr` to store whether it's callee was found using ADL. It does not change the size of any AST nodes.
Reviewers: fowles, rsmith, klimek, shafik
Reviewed By: rsmith
Subscribers: aaron.ballman, riccibruno, calabrese, titus, cfe-commits
Differential Revision: https://reviews.llvm.org/D55534
llvm-svn: 348977
2018-12-13 05:50:55 +08:00
|
|
|
E->setADLCallKind(static_cast<CallExpr::ADLCallKind>(Record.readInt()));
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
void ASTStmtReader::VisitCXXMemberCallExpr(CXXMemberCallExpr *E) {
|
|
|
|
VisitCallExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitMemberExpr(MemberExpr *E) {
|
2019-06-07 07:24:15 +08:00
|
|
|
VisitExpr(E);
|
|
|
|
|
|
|
|
bool HasQualifier = Record.readInt();
|
|
|
|
bool HasFoundDecl = Record.readInt();
|
|
|
|
bool HasTemplateInfo = Record.readInt();
|
|
|
|
unsigned NumTemplateArgs = Record.readInt();
|
|
|
|
|
|
|
|
E->Base = Record.readSubExpr();
|
|
|
|
E->MemberDecl = Record.readDeclAs<ValueDecl>();
|
|
|
|
Record.readDeclarationNameLoc(E->MemberDNLoc, E->MemberDecl->getDeclName());
|
|
|
|
E->MemberLoc = Record.readSourceLocation();
|
|
|
|
E->MemberExprBits.IsArrow = Record.readInt();
|
|
|
|
E->MemberExprBits.HasQualifierOrFoundDecl = HasQualifier || HasFoundDecl;
|
|
|
|
E->MemberExprBits.HasTemplateKWAndArgsInfo = HasTemplateInfo;
|
|
|
|
E->MemberExprBits.HadMultipleCandidates = Record.readInt();
|
2019-06-12 01:50:36 +08:00
|
|
|
E->MemberExprBits.NonOdrUseReason = Record.readInt();
|
2019-06-07 07:24:15 +08:00
|
|
|
E->MemberExprBits.OperatorLoc = Record.readSourceLocation();
|
|
|
|
|
|
|
|
if (HasQualifier || HasFoundDecl) {
|
|
|
|
DeclAccessPair FoundDecl;
|
|
|
|
if (HasFoundDecl) {
|
|
|
|
auto *FoundD = Record.readDeclAs<NamedDecl>();
|
|
|
|
auto AS = (AccessSpecifier)Record.readInt();
|
|
|
|
FoundDecl = DeclAccessPair::make(FoundD, AS);
|
|
|
|
} else {
|
|
|
|
FoundDecl = DeclAccessPair::make(E->MemberDecl,
|
|
|
|
E->MemberDecl->getAccess());
|
|
|
|
}
|
|
|
|
E->getTrailingObjects<MemberExprNameQualifier>()->FoundDecl = FoundDecl;
|
|
|
|
|
|
|
|
NestedNameSpecifierLoc QualifierLoc;
|
|
|
|
if (HasQualifier)
|
|
|
|
QualifierLoc = Record.readNestedNameSpecifierLoc();
|
|
|
|
E->getTrailingObjects<MemberExprNameQualifier>()->QualifierLoc =
|
|
|
|
QualifierLoc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (HasTemplateInfo)
|
|
|
|
ReadTemplateKWAndArgsInfo(
|
|
|
|
*E->getTrailingObjects<ASTTemplateKWAndArgsInfo>(),
|
|
|
|
E->getTrailingObjects<TemplateArgumentLoc>(), NumTemplateArgs);
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCIsaExpr(ObjCIsaExpr *E) {
|
2009-07-25 01:54:45 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setIsaMemberLoc(ReadSourceLocation());
|
|
|
|
E->setOpLoc(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setArrow(Record.readInt());
|
2009-07-25 01:54:45 +08:00
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
void ASTStmtReader::
|
|
|
|
VisitObjCIndirectCopyRestoreExpr(ObjCIndirectCopyRestoreExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Operand = Record.readSubExpr();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setShouldCopy(Record.readInt());
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitObjCBridgedCastExpr(ObjCBridgedCastExpr *E) {
|
|
|
|
VisitExplicitCastExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->LParenLoc = ReadSourceLocation();
|
|
|
|
E->BridgeKeywordLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->Kind = Record.readInt();
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCastExpr(CastExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumBaseSpecs = Record.readInt();
|
2010-08-07 14:22:56 +08:00
|
|
|
assert(NumBaseSpecs == E->path_size());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSubExpr(Record.readSubExpr());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setCastKind((CastKind)Record.readInt());
|
2010-08-07 14:22:56 +08:00
|
|
|
CastExpr::path_iterator BaseI = E->path_begin();
|
2010-07-03 07:30:27 +08:00
|
|
|
while (NumBaseSpecs--) {
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *BaseSpec = new (Record.getContext()) CXXBaseSpecifier;
|
2016-12-21 12:34:52 +08:00
|
|
|
*BaseSpec = Record.readCXXBaseSpecifier();
|
2010-08-07 14:22:56 +08:00
|
|
|
*BaseI++ = BaseSpec;
|
2010-07-03 07:30:27 +08:00
|
|
|
}
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitBinaryOperator(BinaryOperator *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setLHS(Record.readSubExpr());
|
|
|
|
E->setRHS(Record.readSubExpr());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setOpcode((BinaryOperator::Opcode)Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setOperatorLoc(ReadSourceLocation());
|
2017-03-28 03:17:25 +08:00
|
|
|
E->setFPFeatures(FPOptions(Record.readInt()));
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCompoundAssignOperator(CompoundAssignOperator *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitBinaryOperator(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setComputationLHSType(Record.readType());
|
|
|
|
E->setComputationResultType(Record.readType());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitConditionalOperator(ConditionalOperator *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExprs[ConditionalOperator::COND] = Record.readSubExpr();
|
|
|
|
E->SubExprs[ConditionalOperator::LHS] = Record.readSubExpr();
|
|
|
|
E->SubExprs[ConditionalOperator::RHS] = Record.readSubExpr();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->QuestionLoc = ReadSourceLocation();
|
|
|
|
E->ColonLoc = ReadSourceLocation();
|
2011-02-17 18:25:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ASTStmtReader::VisitBinaryConditionalOperator(BinaryConditionalOperator *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->OpaqueValue = cast<OpaqueValueExpr>(Record.readSubExpr());
|
|
|
|
E->SubExprs[BinaryConditionalOperator::COMMON] = Record.readSubExpr();
|
|
|
|
E->SubExprs[BinaryConditionalOperator::COND] = Record.readSubExpr();
|
|
|
|
E->SubExprs[BinaryConditionalOperator::LHS] = Record.readSubExpr();
|
|
|
|
E->SubExprs[BinaryConditionalOperator::RHS] = Record.readSubExpr();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->QuestionLoc = ReadSourceLocation();
|
|
|
|
E->ColonLoc = ReadSourceLocation();
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitImplicitCastExpr(ImplicitCastExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitCastExpr(E);
|
[AST] Sink 'part of explicit cast' down into ImplicitCastExpr
Summary:
As discussed in IRC with @rsmith, it is slightly not good to keep that in the `CastExpr` itself:
Given the explicit cast, which is represented in AST as an `ExplicitCastExpr` + `ImplicitCastExpr`'s,
only the `ImplicitCastExpr`'s will be marked as `PartOfExplicitCast`, but not the `ExplicitCastExpr` itself.
Thus, it is only ever `true` for `ImplicitCastExpr`'s, so we don't need to write/read/dump it for `ExplicitCastExpr`'s.
We don't need to worry that we write the `PartOfExplicitCast` in PCH after `CastExpr::path_iterator`,
since the `ExprImplicitCastAbbrev` is only used when the `NumBaseSpecs == 0`, i.e. there is no 'path'.
Reviewers: rsmith, rjmccall, erichkeane, aaron.ballman
Reviewed By: rsmith, erichkeane
Subscribers: vsk, cfe-commits, rsmith
Tags: #clang
Differential Revision: https://reviews.llvm.org/D49838
llvm-svn: 338108
2018-07-27 15:27:14 +08:00
|
|
|
E->setIsPartOfExplicitCast(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitExplicitCastExpr(ExplicitCastExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitCastExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setTypeInfoAsWritten(GetTypeSourceInfo());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCStyleCastExpr(CStyleCastExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExplicitCastExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLParenLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCompoundLiteralExpr(CompoundLiteralExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLParenLoc(ReadSourceLocation());
|
|
|
|
E->setTypeSourceInfo(GetTypeSourceInfo());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setInitializer(Record.readSubExpr());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setFileScope(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitExtVectorElementExpr(ExtVectorElementExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
|
|
|
E->setAccessor(Record.getIdentifierInfo());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setAccessorLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitInitListExpr(InitListExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2018-04-12 04:57:28 +08:00
|
|
|
if (auto *SyntForm = cast_or_null<InitListExpr>(Record.readSubStmt()))
|
2012-11-09 02:41:43 +08:00
|
|
|
E->setSyntacticForm(SyntForm);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLBraceLoc(ReadSourceLocation());
|
|
|
|
E->setRBraceLoc(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
bool isArrayFiller = Record.readInt();
|
2014-05-22 13:54:18 +08:00
|
|
|
Expr *filler = nullptr;
|
2011-04-22 13:29:30 +08:00
|
|
|
if (isArrayFiller) {
|
2016-12-21 12:34:52 +08:00
|
|
|
filler = Record.readSubExpr();
|
2011-04-22 13:29:30 +08:00
|
|
|
E->ArrayFillerOrUnionFieldInit = filler;
|
|
|
|
} else
|
2016-12-16 04:53:26 +08:00
|
|
|
E->ArrayFillerOrUnionFieldInit = ReadDeclAs<FieldDecl>();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->sawArrayRangeDesignator(Record.readInt());
|
|
|
|
unsigned NumInits = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->reserveInits(Record.getContext(), NumInits);
|
2011-04-22 13:29:30 +08:00
|
|
|
if (isArrayFiller) {
|
|
|
|
for (unsigned I = 0; I != NumInits; ++I) {
|
2016-12-21 12:34:52 +08:00
|
|
|
Expr *init = Record.readSubExpr();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->updateInit(Record.getContext(), I, init ? init : filler);
|
2011-04-22 13:29:30 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (unsigned I = 0; I != NumInits; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->updateInit(Record.getContext(), I, Record.readSubExpr());
|
2011-04-22 13:29:30 +08:00
|
|
|
}
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitDesignatedInitExpr(DesignatedInitExpr *E) {
|
2018-04-12 04:57:28 +08:00
|
|
|
using Designator = DesignatedInitExpr::Designator;
|
2009-04-27 13:14:47 +08:00
|
|
|
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumSubExprs = Record.readInt();
|
2009-04-27 13:14:47 +08:00
|
|
|
assert(NumSubExprs == E->getNumSubExprs() && "Wrong number of subexprs");
|
|
|
|
for (unsigned I = 0; I != NumSubExprs; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSubExpr(I, Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setEqualOrColonLoc(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setGNUSyntax(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<Designator, 4> Designators;
|
2016-12-21 08:17:49 +08:00
|
|
|
while (Record.getIdx() < Record.size()) {
|
|
|
|
switch ((DesignatorTypes)Record.readInt()) {
|
2010-08-19 07:57:32 +08:00
|
|
|
case DESIG_FIELD_DECL: {
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *Field = ReadDeclAs<FieldDecl>();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation DotLoc = ReadSourceLocation();
|
|
|
|
SourceLocation FieldLoc = ReadSourceLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
Designators.push_back(Designator(Field->getIdentifier(), DotLoc,
|
2009-04-27 13:14:47 +08:00
|
|
|
FieldLoc));
|
|
|
|
Designators.back().setField(Field);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DESIG_FIELD_NAME: {
|
2016-12-21 12:34:52 +08:00
|
|
|
const IdentifierInfo *Name = Record.getIdentifierInfo();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation DotLoc = ReadSourceLocation();
|
|
|
|
SourceLocation FieldLoc = ReadSourceLocation();
|
2009-04-27 13:14:47 +08:00
|
|
|
Designators.push_back(Designator(Name, DotLoc, FieldLoc));
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DESIG_ARRAY: {
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned Index = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation LBracketLoc = ReadSourceLocation();
|
|
|
|
SourceLocation RBracketLoc = ReadSourceLocation();
|
2009-04-27 13:14:47 +08:00
|
|
|
Designators.push_back(Designator(Index, LBracketLoc, RBracketLoc));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DESIG_ARRAY_RANGE: {
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned Index = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation LBracketLoc = ReadSourceLocation();
|
|
|
|
SourceLocation EllipsisLoc = ReadSourceLocation();
|
|
|
|
SourceLocation RBracketLoc = ReadSourceLocation();
|
2009-04-27 13:14:47 +08:00
|
|
|
Designators.push_back(Designator(Index, LBracketLoc, EllipsisLoc,
|
|
|
|
RBracketLoc));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setDesignators(Record.getContext(),
|
2010-01-07 07:17:19 +08:00
|
|
|
Designators.data(), Designators.size());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2015-06-10 08:27:52 +08:00
|
|
|
void ASTStmtReader::VisitDesignatedInitUpdateExpr(DesignatedInitUpdateExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
|
|
|
E->setUpdater(Record.readSubExpr());
|
2015-06-10 08:27:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitNoInitExpr(NoInitExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
}
|
|
|
|
|
2016-12-12 10:53:20 +08:00
|
|
|
void ASTStmtReader::VisitArrayInitLoopExpr(ArrayInitLoopExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExprs[0] = Record.readSubExpr();
|
|
|
|
E->SubExprs[1] = Record.readSubExpr();
|
2016-12-12 10:53:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitArrayInitIndexExpr(ArrayInitIndexExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitImplicitValueInitExpr(ImplicitValueInitExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitVAArgExpr(VAArgExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSubExpr(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setWrittenTypeInfo(GetTypeSourceInfo());
|
|
|
|
E->setBuiltinLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setIsMicrosoftABI(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
Implement __builtin_LINE() et. al. to support source location capture.
Summary:
This patch implements the source location builtins `__builtin_LINE(), `__builtin_FUNCTION()`, `__builtin_FILE()` and `__builtin_COLUMN()`. These builtins are needed to implement [`std::experimental::source_location`](https://rawgit.com/cplusplus/fundamentals-ts/v2/main.html#reflection.src_loc.creation).
With the exception of `__builtin_COLUMN`, GCC also implements these builtins, and Clangs behavior is intended to match as closely as possible.
Reviewers: rsmith, joerg, aaron.ballman, bogner, majnemer, shafik, martong
Reviewed By: rsmith
Subscribers: rnkovacs, loskutov, riccibruno, mgorny, kunitoki, alexr, majnemer, hfinkel, cfe-commits
Differential Revision: https://reviews.llvm.org/D37035
llvm-svn: 360937
2019-05-17 05:04:15 +08:00
|
|
|
void ASTStmtReader::VisitSourceLocExpr(SourceLocExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
E->ParentContext = ReadDeclAs<DeclContext>();
|
|
|
|
E->BuiltinLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
|
|
|
E->SourceLocExprBits.Kind =
|
|
|
|
static_cast<SourceLocExpr::IdentKind>(Record.readInt());
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitAddrLabelExpr(AddrLabelExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setAmpAmpLoc(ReadSourceLocation());
|
|
|
|
E->setLabelLoc(ReadSourceLocation());
|
|
|
|
E->setLabel(ReadDeclAs<LabelDecl>());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitStmtExpr(StmtExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLParenLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSubStmt(cast_or_null<CompoundStmt>(Record.readSubStmt()));
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitChooseExpr(ChooseExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setCond(Record.readSubExpr());
|
|
|
|
E->setLHS(Record.readSubExpr());
|
|
|
|
E->setRHS(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setBuiltinLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setIsConditionTrue(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitGNUNullExpr(GNUNullExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setTokenLocation(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitShuffleVectorExpr(ShuffleVectorExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<Expr *, 16> Exprs;
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumExprs = Record.readInt();
|
2010-06-29 06:28:35 +08:00
|
|
|
while (NumExprs--)
|
2016-12-21 12:34:52 +08:00
|
|
|
Exprs.push_back(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setExprs(Record.getContext(), Exprs);
|
|
|
|
E->setBuiltinLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2013-09-18 11:29:45 +08:00
|
|
|
void ASTStmtReader::VisitConvertVectorExpr(ConvertVectorExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->BuiltinLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
|
|
|
E->TInfo = GetTypeSourceInfo();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SrcExpr = Record.readSubExpr();
|
2013-09-18 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitBlockExpr(BlockExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setBlockDecl(ReadDeclAs<BlockDecl>());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2011-04-15 08:35:48 +08:00
|
|
|
void ASTStmtReader::VisitGenericSelectionExpr(GenericSelectionExpr *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
|
2019-01-26 22:15:10 +08:00
|
|
|
unsigned NumAssocs = Record.readInt();
|
|
|
|
assert(NumAssocs == E->getNumAssocs() && "Wrong NumAssocs!");
|
|
|
|
E->ResultIndex = Record.readInt();
|
|
|
|
E->GenericSelectionExprBits.GenericLoc = ReadSourceLocation();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->DefaultLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
2019-01-26 22:15:10 +08:00
|
|
|
|
|
|
|
Stmt **Stmts = E->getTrailingObjects<Stmt *>();
|
|
|
|
// Add 1 to account for the controlling expression which is the first
|
|
|
|
// expression in the trailing array of Stmt *. This is not needed for
|
|
|
|
// the trailing array of TypeSourceInfo *.
|
|
|
|
for (unsigned I = 0, N = NumAssocs + 1; I < N; ++I)
|
|
|
|
Stmts[I] = Record.readSubExpr();
|
|
|
|
|
|
|
|
TypeSourceInfo **TSIs = E->getTrailingObjects<TypeSourceInfo *>();
|
|
|
|
for (unsigned I = 0, N = NumAssocs; I < N; ++I)
|
|
|
|
TSIs[I] = GetTypeSourceInfo();
|
2011-04-15 08:35:48 +08:00
|
|
|
}
|
|
|
|
|
2011-11-06 17:01:30 +08:00
|
|
|
void ASTStmtReader::VisitPseudoObjectExpr(PseudoObjectExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned numSemanticExprs = Record.readInt();
|
2011-11-06 17:01:30 +08:00
|
|
|
assert(numSemanticExprs + 1 == E->PseudoObjectExprBits.NumSubExprs);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->PseudoObjectExprBits.ResultIndex = Record.readInt();
|
2011-11-06 17:01:30 +08:00
|
|
|
|
|
|
|
// Read the syntactic expression.
|
2016-12-21 12:34:52 +08:00
|
|
|
E->getSubExprsBuffer()[0] = Record.readSubExpr();
|
2011-11-06 17:01:30 +08:00
|
|
|
|
|
|
|
// Read all the semantic expressions.
|
|
|
|
for (unsigned i = 0; i != numSemanticExprs; ++i) {
|
2016-12-21 12:34:52 +08:00
|
|
|
Expr *subExpr = Record.readSubExpr();
|
2011-11-06 17:01:30 +08:00
|
|
|
E->getSubExprsBuffer()[i+1] = subExpr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-11 10:20:01 +08:00
|
|
|
void ASTStmtReader::VisitAtomicExpr(AtomicExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->Op = AtomicExpr::AtomicOp(Record.readInt());
|
2012-04-11 06:49:28 +08:00
|
|
|
E->NumSubExprs = AtomicExpr::getNumSubExprs(E->Op);
|
|
|
|
for (unsigned I = 0; I != E->NumSubExprs; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExprs[I] = Record.readSubExpr();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->BuiltinLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
2011-10-11 10:20:01 +08:00
|
|
|
}
|
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Objective-C Expressions and Statements
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCStringLiteral(ObjCStringLiteral *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setString(cast<StringLiteral>(Record.readSubStmt()));
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setAtLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
void ASTStmtReader::VisitObjCBoxedExpr(ObjCBoxedExpr *E) {
|
2012-03-07 04:05:56 +08:00
|
|
|
VisitExpr(E);
|
|
|
|
// could be one of several IntegerLiteral, FloatLiteral, etc.
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExpr = Record.readSubStmt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->BoxingMethod = ReadDeclAs<ObjCMethodDecl>();
|
|
|
|
E->Range = ReadSourceRange();
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitObjCArrayLiteral(ObjCArrayLiteral *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumElements = Record.readInt();
|
2012-03-07 04:05:56 +08:00
|
|
|
assert(NumElements == E->getNumElements() && "Wrong number of elements");
|
|
|
|
Expr **Elements = E->getElements();
|
|
|
|
for (unsigned I = 0, N = NumElements; I != N; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
Elements[I] = Record.readSubExpr();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->ArrayWithObjectsMethod = ReadDeclAs<ObjCMethodDecl>();
|
|
|
|
E->Range = ReadSourceRange();
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitObjCDictionaryLiteral(ObjCDictionaryLiteral *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumElements = Record.readInt();
|
2012-03-07 04:05:56 +08:00
|
|
|
assert(NumElements == E->getNumElements() && "Wrong number of elements");
|
2016-12-21 08:17:49 +08:00
|
|
|
bool HasPackExpansions = Record.readInt();
|
2012-03-07 04:05:56 +08:00
|
|
|
assert(HasPackExpansions == E->HasPackExpansions &&"Pack expansion mismatch");
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *KeyValues =
|
2015-12-31 12:43:19 +08:00
|
|
|
E->getTrailingObjects<ObjCDictionaryLiteral::KeyValuePair>();
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *Expansions =
|
2015-12-31 12:43:19 +08:00
|
|
|
E->getTrailingObjects<ObjCDictionaryLiteral::ExpansionData>();
|
2012-03-07 04:05:56 +08:00
|
|
|
for (unsigned I = 0; I != NumElements; ++I) {
|
2016-12-21 12:34:52 +08:00
|
|
|
KeyValues[I].Key = Record.readSubExpr();
|
|
|
|
KeyValues[I].Value = Record.readSubExpr();
|
2012-03-07 04:05:56 +08:00
|
|
|
if (HasPackExpansions) {
|
2016-12-16 04:53:26 +08:00
|
|
|
Expansions[I].EllipsisLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
Expansions[I].NumExpansionsPlusOne = Record.readInt();
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
E->DictWithObjectsMethod = ReadDeclAs<ObjCMethodDecl>();
|
|
|
|
E->Range = ReadSourceRange();
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCEncodeExpr(ObjCEncodeExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setEncodedTypeSourceInfo(GetTypeSourceInfo());
|
|
|
|
E->setAtLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCSelectorExpr(ObjCSelectorExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSelector(Record.readSelector());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setAtLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCProtocolExpr(ObjCProtocolExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setProtocol(ReadDeclAs<ObjCProtocolDecl>());
|
|
|
|
E->setAtLoc(ReadSourceLocation());
|
|
|
|
E->ProtoLoc = ReadSourceLocation();
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCIvarRefExpr(ObjCIvarRefExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setDecl(ReadDeclAs<ObjCIvarDecl>());
|
|
|
|
E->setLocation(ReadSourceLocation());
|
|
|
|
E->setOpLoc(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setIsArrow(Record.readInt());
|
|
|
|
E->setIsFreeIvar(Record.readInt());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCPropertyRefExpr(ObjCPropertyRefExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned MethodRefFlags = Record.readInt();
|
|
|
|
bool Implicit = Record.readInt() != 0;
|
2010-12-02 09:19:52 +08:00
|
|
|
if (Implicit) {
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *Getter = ReadDeclAs<ObjCMethodDecl>();
|
|
|
|
auto *Setter = ReadDeclAs<ObjCMethodDecl>();
|
2012-03-30 08:19:18 +08:00
|
|
|
E->setImplicitProperty(Getter, Setter, MethodRefFlags);
|
2010-12-02 09:19:52 +08:00
|
|
|
} else {
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setExplicitProperty(ReadDeclAs<ObjCPropertyDecl>(), MethodRefFlags);
|
2010-10-15 00:04:05 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
|
|
|
E->setReceiverLocation(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
switch (Record.readInt()) {
|
2010-12-02 09:19:52 +08:00
|
|
|
case 0:
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
2010-12-02 09:19:52 +08:00
|
|
|
break;
|
|
|
|
case 1:
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setSuperReceiver(Record.readType());
|
2010-12-02 09:19:52 +08:00
|
|
|
break;
|
|
|
|
case 2:
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setClassReceiver(ReadDeclAs<ObjCInterfaceDecl>());
|
2010-12-02 09:19:52 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
void ASTStmtReader::VisitObjCSubscriptRefExpr(ObjCSubscriptRefExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setRBracket(ReadSourceLocation());
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBaseExpr(Record.readSubExpr());
|
|
|
|
E->setKeyExpr(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->GetAtIndexMethodDecl = ReadDeclAs<ObjCMethodDecl>();
|
|
|
|
E->SetAtIndexMethodDecl = ReadDeclAs<ObjCMethodDecl>();
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCMessageExpr(ObjCMessageExpr *E) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(Record.peekInt() == E->getNumArgs());
|
|
|
|
Record.skipInts(1);
|
|
|
|
unsigned NumStoredSelLocs = Record.readInt();
|
|
|
|
E->SelLocsKind = Record.readInt();
|
|
|
|
E->setDelegateInitCall(Record.readInt());
|
|
|
|
E->IsImplicit = Record.readInt();
|
2018-04-12 04:57:28 +08:00
|
|
|
auto Kind = static_cast<ObjCMessageExpr::ReceiverKind>(Record.readInt());
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
switch (Kind) {
|
|
|
|
case ObjCMessageExpr::Instance:
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setInstanceReceiver(Record.readSubExpr());
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjCMessageExpr::Class:
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setClassReceiver(GetTypeSourceInfo());
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjCMessageExpr::SuperClass:
|
|
|
|
case ObjCMessageExpr::SuperInstance: {
|
2016-12-21 08:17:49 +08:00
|
|
|
QualType T = Record.readType();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceLocation SuperLoc = ReadSourceLocation();
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
E->setSuper(SuperLoc, T, Kind == ObjCMessageExpr::SuperInstance);
|
|
|
|
break;
|
|
|
|
}
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
assert(Kind == E->getReceiverKind());
|
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
if (Record.readInt())
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setMethodDecl(ReadDeclAs<ObjCMethodDecl>());
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
else
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setSelector(Record.readSelector());
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
E->LBracLoc = ReadSourceLocation();
|
|
|
|
E->RBracLoc = ReadSourceLocation();
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
for (unsigned I = 0, N = E->getNumArgs(); I != N; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setArg(I, Record.readSubExpr());
|
2011-10-03 14:36:51 +08:00
|
|
|
|
|
|
|
SourceLocation *Locs = E->getStoredSelLocs();
|
|
|
|
for (unsigned I = 0; I != NumStoredSelLocs; ++I)
|
2016-12-16 04:53:26 +08:00
|
|
|
Locs[I] = ReadSourceLocation();
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCForCollectionStmt(ObjCForCollectionStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setElement(Record.readSubStmt());
|
|
|
|
S->setCollection(Record.readSubExpr());
|
|
|
|
S->setBody(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setForLoc(ReadSourceLocation());
|
|
|
|
S->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCAtCatchStmt(ObjCAtCatchStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCatchBody(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setCatchParamDecl(ReadDeclAs<VarDecl>());
|
|
|
|
S->setAtCatchLoc(ReadSourceLocation());
|
|
|
|
S->setRParenLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCAtFinallyStmt(ObjCAtFinallyStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setFinallyBody(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setAtFinallyLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
void ASTStmtReader::VisitObjCAutoreleasePoolStmt(ObjCAutoreleasePoolStmt *S) {
|
2019-03-13 05:31:00 +08:00
|
|
|
VisitStmt(S); // FIXME: no test coverage.
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setSubStmt(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setAtLoc(ReadSourceLocation());
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCAtTryStmt(ObjCAtTryStmt *S) {
|
2009-04-27 13:14:47 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(Record.peekInt() == S->getNumCatchStmts());
|
|
|
|
Record.skipInts(1);
|
|
|
|
bool HasFinally = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setTryBody(Record.readSubStmt());
|
2010-06-29 06:28:35 +08:00
|
|
|
for (unsigned I = 0, N = S->getNumCatchStmts(); I != N; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setCatchStmt(I, cast_or_null<ObjCAtCatchStmt>(Record.readSubStmt()));
|
2010-04-24 06:50:49 +08:00
|
|
|
|
|
|
|
if (HasFinally)
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setFinallyStmt(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setAtTryLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCAtSynchronizedStmt(ObjCAtSynchronizedStmt *S) {
|
2019-03-13 05:31:00 +08:00
|
|
|
VisitStmt(S); // FIXME: no test coverage.
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setSynchExpr(Record.readSubStmt());
|
|
|
|
S->setSynchBody(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setAtSynchronizedLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitObjCAtThrowStmt(ObjCAtThrowStmt *S) {
|
2019-03-13 05:31:00 +08:00
|
|
|
VisitStmt(S); // FIXME: no test coverage.
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setThrowExpr(Record.readSubStmt());
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setThrowLoc(ReadSourceLocation());
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
void ASTStmtReader::VisitObjCBoolLiteralExpr(ObjCBoolLiteralExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setValue(Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2016-07-16 08:35:23 +08:00
|
|
|
void ASTStmtReader::VisitObjCAvailabilityCheckExpr(ObjCAvailabilityCheckExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
SourceRange R = Record.readSourceRange();
|
2016-07-16 08:35:23 +08:00
|
|
|
E->AtLoc = R.getBegin();
|
|
|
|
E->RParen = R.getEnd();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->VersionToCheck = Record.readVersionTuple();
|
2016-07-16 08:35:23 +08:00
|
|
|
}
|
|
|
|
|
2009-07-14 11:19:21 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// C++ Expressions and Statements
|
2010-07-23 00:03:56 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXCatchStmt(CXXCatchStmt *S) {
|
2010-07-23 00:03:56 +08:00
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->CatchLoc = ReadSourceLocation();
|
|
|
|
S->ExceptionDecl = ReadDeclAs<VarDecl>();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->HandlerBlock = Record.readSubStmt();
|
2010-07-23 00:03:56 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXTryStmt(CXXTryStmt *S) {
|
2010-07-23 00:03:56 +08:00
|
|
|
VisitStmt(S);
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(Record.peekInt() == S->getNumHandlers() && "NumStmtFields is wrong ?");
|
|
|
|
Record.skipInts(1);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->TryLoc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->getStmts()[0] = Record.readSubStmt();
|
2010-07-23 00:03:56 +08:00
|
|
|
for (unsigned i = 0, e = S->getNumHandlers(); i != e; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
S->getStmts()[i + 1] = Record.readSubStmt();
|
2010-07-23 00:03:56 +08:00
|
|
|
}
|
2009-07-14 11:19:21 +08:00
|
|
|
|
2011-04-15 06:09:26 +08:00
|
|
|
void ASTStmtReader::VisitCXXForRangeStmt(CXXForRangeStmt *S) {
|
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->ForLoc = ReadSourceLocation();
|
|
|
|
S->CoawaitLoc = ReadSourceLocation();
|
|
|
|
S->ColonLoc = ReadSourceLocation();
|
|
|
|
S->RParenLoc = ReadSourceLocation();
|
2018-09-29 02:44:09 +08:00
|
|
|
S->setInit(Record.readSubStmt());
|
2016-12-21 12:34:52 +08:00
|
|
|
S->setRangeStmt(Record.readSubStmt());
|
|
|
|
S->setBeginStmt(Record.readSubStmt());
|
|
|
|
S->setEndStmt(Record.readSubStmt());
|
|
|
|
S->setCond(Record.readSubExpr());
|
|
|
|
S->setInc(Record.readSubExpr());
|
|
|
|
S->setLoopVarStmt(Record.readSubStmt());
|
|
|
|
S->setBody(Record.readSubStmt());
|
2011-04-15 06:09:26 +08:00
|
|
|
}
|
|
|
|
|
2011-10-25 09:33:02 +08:00
|
|
|
void ASTStmtReader::VisitMSDependentExistsStmt(MSDependentExistsStmt *S) {
|
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->KeywordLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
S->IsIfExists = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->QualifierLoc = Record.readNestedNameSpecifierLoc();
|
2016-12-16 04:53:26 +08:00
|
|
|
ReadDeclarationNameInfo(S->NameInfo);
|
2016-12-21 12:34:52 +08:00
|
|
|
S->SubStmt = Record.readSubStmt();
|
2011-10-25 09:33:02 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXOperatorCallExpr(CXXOperatorCallExpr *E) {
|
2010-06-29 06:28:35 +08:00
|
|
|
VisitCallExpr(E);
|
2018-12-22 00:51:57 +08:00
|
|
|
E->CXXOperatorCallExprBits.OperatorKind = Record.readInt();
|
|
|
|
E->CXXOperatorCallExprBits.FPFeatures = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Range = Record.readSourceRange();
|
2009-07-14 11:19:21 +08:00
|
|
|
}
|
|
|
|
|
2019-10-19 08:04:38 +08:00
|
|
|
void ASTStmtReader::VisitCXXRewrittenBinaryOperator(
|
|
|
|
CXXRewrittenBinaryOperator *E) {
|
|
|
|
VisitExpr(E);
|
|
|
|
E->CXXRewrittenBinaryOperatorBits.IsReversed = Record.readInt();
|
|
|
|
E->SemanticForm = Record.readSubExpr();
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXConstructExpr(CXXConstructExpr *E) {
|
2009-09-10 07:08:42 +08:00
|
|
|
VisitExpr(E);
|
2018-12-22 22:39:30 +08:00
|
|
|
|
|
|
|
unsigned NumArgs = Record.readInt();
|
|
|
|
assert((NumArgs == E->getNumArgs()) && "Wrong NumArgs!");
|
|
|
|
|
|
|
|
E->CXXConstructExprBits.Elidable = Record.readInt();
|
|
|
|
E->CXXConstructExprBits.HadMultipleCandidates = Record.readInt();
|
|
|
|
E->CXXConstructExprBits.ListInitialization = Record.readInt();
|
|
|
|
E->CXXConstructExprBits.StdInitListInitialization = Record.readInt();
|
|
|
|
E->CXXConstructExprBits.ZeroInitialization = Record.readInt();
|
|
|
|
E->CXXConstructExprBits.ConstructionKind = Record.readInt();
|
|
|
|
E->CXXConstructExprBits.Loc = ReadSourceLocation();
|
|
|
|
E->Constructor = ReadDeclAs<CXXConstructorDecl>();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->ParenOrBraceRange = ReadSourceRange();
|
2018-12-22 22:39:30 +08:00
|
|
|
|
|
|
|
for (unsigned I = 0; I != NumArgs; ++I)
|
|
|
|
E->setArg(I, Record.readSubExpr());
|
2009-09-10 07:08:42 +08:00
|
|
|
}
|
2009-04-27 13:14:47 +08:00
|
|
|
|
P0136R1, DR1573, DR1645, DR1715, DR1736, DR1903, DR1941, DR1959, DR1991:
Replace inheriting constructors implementation with new approach, voted into
C++ last year as a DR against C++11.
Instead of synthesizing a set of derived class constructors for each inherited
base class constructor, we make the constructors of the base class visible to
constructor lookup in the derived class, using the normal rules for
using-declarations.
For constructors, UsingShadowDecl now has a ConstructorUsingShadowDecl derived
class that tracks the requisite additional information. We create shadow
constructors (not found by name lookup) in the derived class to model the
actual initialization, and have a new expression node,
CXXInheritedCtorInitExpr, to model the initialization of a base class from such
a constructor. (This initialization is special because it performs real perfect
forwarding of arguments.)
In cases where argument forwarding is not possible (for inalloca calls,
variadic calls, and calls with callee parameter cleanup), the shadow inheriting
constructor is not emitted and instead we directly emit the initialization code
into the caller of the inherited constructor.
Note that this new model is not perfectly compatible with the old model in some
corner cases. In particular:
* if B inherits a private constructor from A, and C uses that constructor to
construct a B, then we previously required that A befriends B and B
befriends C, but the new rules require A to befriend C directly, and
* if a derived class has its own constructors (and so its implicit default
constructor is suppressed), it may still inherit a default constructor from
a base class
llvm-svn: 274049
2016-06-29 03:03:57 +08:00
|
|
|
void ASTStmtReader::VisitCXXInheritedCtorInitExpr(CXXInheritedCtorInitExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Constructor = ReadDeclAs<CXXConstructorDecl>();
|
|
|
|
E->Loc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->ConstructsVirtualBase = Record.readInt();
|
|
|
|
E->InheritedFromVirtualBase = Record.readInt();
|
P0136R1, DR1573, DR1645, DR1715, DR1736, DR1903, DR1941, DR1959, DR1991:
Replace inheriting constructors implementation with new approach, voted into
C++ last year as a DR against C++11.
Instead of synthesizing a set of derived class constructors for each inherited
base class constructor, we make the constructors of the base class visible to
constructor lookup in the derived class, using the normal rules for
using-declarations.
For constructors, UsingShadowDecl now has a ConstructorUsingShadowDecl derived
class that tracks the requisite additional information. We create shadow
constructors (not found by name lookup) in the derived class to model the
actual initialization, and have a new expression node,
CXXInheritedCtorInitExpr, to model the initialization of a base class from such
a constructor. (This initialization is special because it performs real perfect
forwarding of arguments.)
In cases where argument forwarding is not possible (for inalloca calls,
variadic calls, and calls with callee parameter cleanup), the shadow inheriting
constructor is not emitted and instead we directly emit the initialization code
into the caller of the inherited constructor.
Note that this new model is not perfectly compatible with the old model in some
corner cases. In particular:
* if B inherits a private constructor from A, and C uses that constructor to
construct a B, then we previously required that A befriends B and B
befriends C, but the new rules require A to befriend C directly, and
* if a derived class has its own constructors (and so its implicit default
constructor is suppressed), it may still inherit a default constructor from
a base class
llvm-svn: 274049
2016-06-29 03:03:57 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E) {
|
2010-07-10 19:46:15 +08:00
|
|
|
VisitCXXConstructExpr(E);
|
2018-12-22 22:39:30 +08:00
|
|
|
E->TSI = GetTypeSourceInfo();
|
2010-07-10 19:46:15 +08:00
|
|
|
}
|
|
|
|
|
2012-02-07 18:09:13 +08:00
|
|
|
void ASTStmtReader::VisitLambdaExpr(LambdaExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumCaptures = Record.readInt();
|
2012-02-15 01:54:36 +08:00
|
|
|
assert(NumCaptures == E->NumCaptures);(void)NumCaptures;
|
2016-12-16 04:53:26 +08:00
|
|
|
E->IntroducerRange = ReadSourceRange();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->CaptureDefault = static_cast<LambdaCaptureDefault>(Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->CaptureDefaultLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->ExplicitParams = Record.readInt();
|
|
|
|
E->ExplicitResultType = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->ClosingBrace = ReadSourceLocation();
|
|
|
|
|
2012-02-15 01:54:36 +08:00
|
|
|
// Read capture initializers.
|
|
|
|
for (LambdaExpr::capture_init_iterator C = E->capture_init_begin(),
|
|
|
|
CEnd = E->capture_init_end();
|
|
|
|
C != CEnd; ++C)
|
2016-12-21 12:34:52 +08:00
|
|
|
*C = Record.readSubExpr();
|
2012-02-07 18:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
void
|
|
|
|
ASTStmtReader::VisitCXXStdInitializerListExpr(CXXStdInitializerListExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExpr = Record.readSubExpr();
|
2013-06-13 06:31:48 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXNamedCastExpr(CXXNamedCastExpr *E) {
|
2010-06-29 06:28:35 +08:00
|
|
|
VisitExplicitCastExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceRange R = ReadSourceRange();
|
2011-01-13 06:41:29 +08:00
|
|
|
E->Loc = R.getBegin();
|
|
|
|
E->RParenLoc = R.getEnd();
|
2016-12-16 04:53:26 +08:00
|
|
|
R = ReadSourceRange();
|
2013-02-23 06:02:53 +08:00
|
|
|
E->AngleBrackets = R;
|
2010-01-17 05:21:01 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXStaticCastExpr(CXXStaticCastExpr *E) {
|
2010-01-17 05:21:01 +08:00
|
|
|
return VisitCXXNamedCastExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXDynamicCastExpr(CXXDynamicCastExpr *E) {
|
2010-01-17 05:21:01 +08:00
|
|
|
return VisitCXXNamedCastExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXReinterpretCastExpr(CXXReinterpretCastExpr *E) {
|
2010-01-17 05:21:01 +08:00
|
|
|
return VisitCXXNamedCastExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXConstCastExpr(CXXConstCastExpr *E) {
|
2010-01-17 05:21:01 +08:00
|
|
|
return VisitCXXNamedCastExpr(E);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXFunctionalCastExpr(CXXFunctionalCastExpr *E) {
|
2010-06-29 06:28:35 +08:00
|
|
|
VisitExplicitCastExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLParenLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2010-01-17 05:21:01 +08:00
|
|
|
}
|
|
|
|
|
2019-07-03 02:28:13 +08:00
|
|
|
void ASTStmtReader::VisitBuiltinBitCastExpr(BuiltinBitCastExpr *E) {
|
|
|
|
VisitExplicitCastExpr(E);
|
|
|
|
E->KWLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
|
|
|
}
|
|
|
|
|
2012-03-07 16:35:16 +08:00
|
|
|
void ASTStmtReader::VisitUserDefinedLiteral(UserDefinedLiteral *E) {
|
|
|
|
VisitCallExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->UDSuffixLoc = ReadSourceLocation();
|
2012-03-07 16:35:16 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *E) {
|
2010-02-07 14:32:43 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setValue(Record.readInt());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2010-02-07 14:32:43 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXNullPtrLiteralExpr(CXXNullPtrLiteralExpr *E) {
|
2010-02-07 14:32:43 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2010-02-07 14:32:43 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXTypeidExpr(CXXTypeidExpr *E) {
|
2010-05-09 14:03:39 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setSourceRange(ReadSourceRange());
|
2010-05-09 14:03:39 +08:00
|
|
|
if (E->isTypeOperand()) { // typeid(int)
|
2010-07-23 06:43:28 +08:00
|
|
|
E->setTypeOperandSourceInfo(
|
2016-12-16 04:53:26 +08:00
|
|
|
GetTypeSourceInfo());
|
2010-06-29 06:28:35 +08:00
|
|
|
return;
|
2010-05-09 14:03:39 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-05-09 14:03:39 +08:00
|
|
|
// typeid(42+2)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setExprOperand(Record.readSubExpr());
|
2010-05-09 14:03:39 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXThisExpr(CXXThisExpr *E) {
|
2010-05-09 14:15:05 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocation(ReadSourceLocation());
|
2016-12-21 08:17:49 +08:00
|
|
|
E->setImplicit(Record.readInt());
|
2010-05-09 14:15:05 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXThrowExpr(CXXThrowExpr *E) {
|
2010-05-09 14:15:05 +08:00
|
|
|
VisitExpr(E);
|
2018-11-17 20:53:56 +08:00
|
|
|
E->CXXThrowExprBits.ThrowLoc = ReadSourceLocation();
|
|
|
|
E->Operand = Record.readSubExpr();
|
|
|
|
E->CXXThrowExprBits.IsThrownVariableInScope = Record.readInt();
|
2010-05-09 14:15:05 +08:00
|
|
|
}
|
2010-05-09 14:03:39 +08:00
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXDefaultArgExpr(CXXDefaultArgExpr *E) {
|
2010-05-09 14:40:08 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Param = ReadDeclAs<ParmVarDecl>();
|
Implement __builtin_LINE() et. al. to support source location capture.
Summary:
This patch implements the source location builtins `__builtin_LINE(), `__builtin_FUNCTION()`, `__builtin_FILE()` and `__builtin_COLUMN()`. These builtins are needed to implement [`std::experimental::source_location`](https://rawgit.com/cplusplus/fundamentals-ts/v2/main.html#reflection.src_loc.creation).
With the exception of `__builtin_COLUMN`, GCC also implements these builtins, and Clangs behavior is intended to match as closely as possible.
Reviewers: rsmith, joerg, aaron.ballman, bogner, majnemer, shafik, martong
Reviewed By: rsmith
Subscribers: rnkovacs, loskutov, riccibruno, mgorny, kunitoki, alexr, majnemer, hfinkel, cfe-commits
Differential Revision: https://reviews.llvm.org/D37035
llvm-svn: 360937
2019-05-17 05:04:15 +08:00
|
|
|
E->UsedContext = ReadDeclAs<DeclContext>();
|
2018-11-17 20:56:30 +08:00
|
|
|
E->CXXDefaultArgExprBits.Loc = ReadSourceLocation();
|
2010-05-10 08:25:06 +08:00
|
|
|
}
|
|
|
|
|
2013-04-21 06:23:05 +08:00
|
|
|
void ASTStmtReader::VisitCXXDefaultInitExpr(CXXDefaultInitExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Field = ReadDeclAs<FieldDecl>();
|
Implement __builtin_LINE() et. al. to support source location capture.
Summary:
This patch implements the source location builtins `__builtin_LINE(), `__builtin_FUNCTION()`, `__builtin_FILE()` and `__builtin_COLUMN()`. These builtins are needed to implement [`std::experimental::source_location`](https://rawgit.com/cplusplus/fundamentals-ts/v2/main.html#reflection.src_loc.creation).
With the exception of `__builtin_COLUMN`, GCC also implements these builtins, and Clangs behavior is intended to match as closely as possible.
Reviewers: rsmith, joerg, aaron.ballman, bogner, majnemer, shafik, martong
Reviewed By: rsmith
Subscribers: rnkovacs, loskutov, riccibruno, mgorny, kunitoki, alexr, majnemer, hfinkel, cfe-commits
Differential Revision: https://reviews.llvm.org/D37035
llvm-svn: 360937
2019-05-17 05:04:15 +08:00
|
|
|
E->UsedContext = ReadDeclAs<DeclContext>();
|
2018-11-17 21:02:47 +08:00
|
|
|
E->CXXDefaultInitExprBits.Loc = ReadSourceLocation();
|
2013-04-21 06:23:05 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXBindTemporaryExpr(CXXBindTemporaryExpr *E) {
|
2010-05-10 08:25:06 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setTemporary(Record.readCXXTemporary());
|
|
|
|
E->setSubExpr(Record.readSubExpr());
|
2010-05-10 08:25:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E) {
|
2010-05-10 09:22:27 +08:00
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->TypeInfo = GetTypeSourceInfo();
|
2019-01-09 00:08:54 +08:00
|
|
|
E->CXXScalarValueInitExprBits.RParenLoc = ReadSourceLocation();
|
2010-05-10 09:22:27 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXNewExpr(CXXNewExpr *E) {
|
2010-05-10 09:22:27 +08:00
|
|
|
VisitExpr(E);
|
2019-01-07 23:04:45 +08:00
|
|
|
|
|
|
|
bool IsArray = Record.readInt();
|
|
|
|
bool HasInit = Record.readInt();
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumPlacementArgs = Record.readInt();
|
2019-01-07 23:04:45 +08:00
|
|
|
bool IsParenTypeId = Record.readInt();
|
|
|
|
|
|
|
|
E->CXXNewExprBits.IsGlobalNew = Record.readInt();
|
|
|
|
E->CXXNewExprBits.ShouldPassAlignment = Record.readInt();
|
|
|
|
E->CXXNewExprBits.UsualArrayDeleteWantsSize = Record.readInt();
|
|
|
|
E->CXXNewExprBits.StoredInitializationStyle = Record.readInt();
|
|
|
|
|
|
|
|
assert((IsArray == E->isArray()) && "Wrong IsArray!");
|
|
|
|
assert((HasInit == E->hasInitializer()) && "Wrong HasInit!");
|
|
|
|
assert((NumPlacementArgs == E->getNumPlacementArgs()) &&
|
|
|
|
"Wrong NumPlacementArgs!");
|
|
|
|
assert((IsParenTypeId == E->isParenTypeId()) && "Wrong IsParenTypeId!");
|
|
|
|
(void)IsArray;
|
|
|
|
(void)HasInit;
|
|
|
|
(void)NumPlacementArgs;
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setOperatorNew(ReadDeclAs<FunctionDecl>());
|
|
|
|
E->setOperatorDelete(ReadDeclAs<FunctionDecl>());
|
|
|
|
E->AllocatedTypeInfo = GetTypeSourceInfo();
|
2019-01-07 23:04:45 +08:00
|
|
|
if (IsParenTypeId)
|
|
|
|
E->getTrailingObjects<SourceRange>()[0] = ReadSourceRange();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Range = ReadSourceRange();
|
|
|
|
E->DirectInitRange = ReadSourceRange();
|
|
|
|
|
2010-05-10 09:22:27 +08:00
|
|
|
// Install all the subexpressions.
|
2019-01-07 23:04:45 +08:00
|
|
|
for (CXXNewExpr::raw_arg_iterator I = E->raw_arg_begin(),
|
|
|
|
N = E->raw_arg_end();
|
|
|
|
I != N; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
*I = Record.readSubStmt();
|
2010-05-10 09:22:27 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXDeleteExpr(CXXDeleteExpr *E) {
|
2010-06-23 01:07:59 +08:00
|
|
|
VisitExpr(E);
|
2018-12-03 20:32:32 +08:00
|
|
|
E->CXXDeleteExprBits.GlobalDelete = Record.readInt();
|
|
|
|
E->CXXDeleteExprBits.ArrayForm = Record.readInt();
|
|
|
|
E->CXXDeleteExprBits.ArrayFormAsWritten = Record.readInt();
|
|
|
|
E->CXXDeleteExprBits.UsualArrayDeleteWantsSize = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->OperatorDelete = ReadDeclAs<FunctionDecl>();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Argument = Record.readSubExpr();
|
2018-12-03 20:32:32 +08:00
|
|
|
E->CXXDeleteExprBits.Loc = ReadSourceLocation();
|
2010-06-23 01:07:59 +08:00
|
|
|
}
|
2010-05-10 08:25:06 +08:00
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E) {
|
2010-06-28 17:32:03 +08:00
|
|
|
VisitExpr(E);
|
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Base = Record.readSubExpr();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->IsArrow = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->OperatorLoc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->QualifierLoc = Record.readNestedNameSpecifierLoc();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->ScopeType = GetTypeSourceInfo();
|
|
|
|
E->ColonColonLoc = ReadSourceLocation();
|
|
|
|
E->TildeLoc = ReadSourceLocation();
|
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
IdentifierInfo *II = Record.getIdentifierInfo();
|
2010-06-28 17:32:03 +08:00
|
|
|
if (II)
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setDestroyedType(II, ReadSourceLocation());
|
2010-06-28 17:32:03 +08:00
|
|
|
else
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setDestroyedType(GetTypeSourceInfo());
|
2010-06-28 17:32:03 +08:00
|
|
|
}
|
|
|
|
|
2010-12-06 16:20:24 +08:00
|
|
|
void ASTStmtReader::VisitExprWithCleanups(ExprWithCleanups *E) {
|
2010-05-10 08:25:06 +08:00
|
|
|
VisitExpr(E);
|
2011-11-10 13:35:25 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumObjects = Record.readInt();
|
2011-11-10 13:35:25 +08:00
|
|
|
assert(NumObjects == E->getNumObjects());
|
|
|
|
for (unsigned i = 0; i != NumObjects; ++i)
|
2015-12-31 12:18:25 +08:00
|
|
|
E->getTrailingObjects<BlockDecl *>()[i] =
|
2016-12-16 04:53:26 +08:00
|
|
|
ReadDeclAs<BlockDecl>();
|
2011-11-10 13:35:25 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
E->ExprWithCleanupsBits.CleanupsHaveSideEffects = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExpr = Record.readSubExpr();
|
2010-05-09 14:40:08 +08:00
|
|
|
}
|
|
|
|
|
2019-01-08 22:17:00 +08:00
|
|
|
void ASTStmtReader::VisitCXXDependentScopeMemberExpr(
|
|
|
|
CXXDependentScopeMemberExpr *E) {
|
2010-06-24 16:57:31 +08:00
|
|
|
VisitExpr(E);
|
2012-01-27 17:46:47 +08:00
|
|
|
|
2019-01-08 22:17:00 +08:00
|
|
|
bool HasTemplateKWAndArgsInfo = Record.readInt();
|
|
|
|
unsigned NumTemplateArgs = Record.readInt();
|
|
|
|
bool HasFirstQualifierFoundInScope = Record.readInt();
|
|
|
|
|
|
|
|
assert((HasTemplateKWAndArgsInfo == E->hasTemplateKWAndArgsInfo()) &&
|
|
|
|
"Wrong HasTemplateKWAndArgsInfo!");
|
|
|
|
assert(
|
|
|
|
(HasFirstQualifierFoundInScope == E->hasFirstQualifierFoundInScope()) &&
|
|
|
|
"Wrong HasFirstQualifierFoundInScope!");
|
|
|
|
|
|
|
|
if (HasTemplateKWAndArgsInfo)
|
2015-12-30 02:15:14 +08:00
|
|
|
ReadTemplateKWAndArgsInfo(
|
|
|
|
*E->getTrailingObjects<ASTTemplateKWAndArgsInfo>(),
|
2019-01-08 22:17:00 +08:00
|
|
|
E->getTrailingObjects<TemplateArgumentLoc>(), NumTemplateArgs);
|
2010-06-24 16:57:31 +08:00
|
|
|
|
2019-01-08 22:17:00 +08:00
|
|
|
assert((NumTemplateArgs == E->getNumTemplateArgs()) &&
|
|
|
|
"Wrong NumTemplateArgs!");
|
|
|
|
|
|
|
|
E->CXXDependentScopeMemberExprBits.IsArrow = Record.readInt();
|
|
|
|
E->CXXDependentScopeMemberExprBits.OperatorLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->BaseType = Record.readType();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->QualifierLoc = Record.readNestedNameSpecifierLoc();
|
2019-01-08 22:17:00 +08:00
|
|
|
E->Base = Record.readSubExpr();
|
|
|
|
|
|
|
|
if (HasFirstQualifierFoundInScope)
|
|
|
|
*E->getTrailingObjects<NamedDecl *>() = ReadDeclAs<NamedDecl>();
|
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
ReadDeclarationNameInfo(E->MemberNameInfo);
|
2010-06-24 16:57:31 +08:00
|
|
|
}
|
|
|
|
|
2010-06-29 06:28:35 +08:00
|
|
|
void
|
2010-08-19 07:56:52 +08:00
|
|
|
ASTStmtReader::VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E) {
|
2010-06-28 17:31:56 +08:00
|
|
|
VisitExpr(E);
|
2012-01-27 17:46:47 +08:00
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
if (Record.readInt()) // HasTemplateKWAndArgsInfo
|
2015-12-30 02:15:14 +08:00
|
|
|
ReadTemplateKWAndArgsInfo(
|
|
|
|
*E->getTrailingObjects<ASTTemplateKWAndArgsInfo>(),
|
|
|
|
E->getTrailingObjects<TemplateArgumentLoc>(),
|
2016-12-21 08:17:49 +08:00
|
|
|
/*NumTemplateArgs=*/Record.readInt());
|
2011-02-26 04:49:16 +08:00
|
|
|
|
2016-12-21 12:34:52 +08:00
|
|
|
E->QualifierLoc = Record.readNestedNameSpecifierLoc();
|
2016-12-16 04:53:26 +08:00
|
|
|
ReadDeclarationNameInfo(E->NameInfo);
|
2010-06-28 17:31:56 +08:00
|
|
|
}
|
|
|
|
|
2010-06-29 06:28:35 +08:00
|
|
|
void
|
2010-08-19 07:56:52 +08:00
|
|
|
ASTStmtReader::VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr *E) {
|
2010-06-24 16:57:31 +08:00
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(Record.peekInt() == E->arg_size() &&
|
|
|
|
"Read wrong record during creation ?");
|
|
|
|
Record.skipInts(1);
|
2010-06-24 16:57:31 +08:00
|
|
|
for (unsigned I = 0, N = E->arg_size(); I != N; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setArg(I, Record.readSubExpr());
|
2019-01-07 22:27:04 +08:00
|
|
|
E->TSI = GetTypeSourceInfo();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLParenLoc(ReadSourceLocation());
|
|
|
|
E->setRParenLoc(ReadSourceLocation());
|
2010-06-24 16:57:31 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitOverloadExpr(OverloadExpr *E) {
|
2010-06-25 17:03:26 +08:00
|
|
|
VisitExpr(E);
|
2012-01-27 17:46:47 +08:00
|
|
|
|
2019-01-09 23:43:19 +08:00
|
|
|
unsigned NumResults = Record.readInt();
|
|
|
|
bool HasTemplateKWAndArgsInfo = Record.readInt();
|
|
|
|
assert((E->getNumDecls() == NumResults) && "Wrong NumResults!");
|
|
|
|
assert((E->hasTemplateKWAndArgsInfo() == HasTemplateKWAndArgsInfo) &&
|
|
|
|
"Wrong HasTemplateKWAndArgsInfo!");
|
|
|
|
|
|
|
|
if (HasTemplateKWAndArgsInfo) {
|
|
|
|
unsigned NumTemplateArgs = Record.readInt();
|
2015-12-30 02:15:14 +08:00
|
|
|
ReadTemplateKWAndArgsInfo(*E->getTrailingASTTemplateKWAndArgsInfo(),
|
|
|
|
E->getTrailingTemplateArgumentLoc(),
|
2019-01-09 23:43:19 +08:00
|
|
|
NumTemplateArgs);
|
|
|
|
assert((E->getNumTemplateArgs() == NumTemplateArgs) &&
|
|
|
|
"Wrong NumTemplateArgs!");
|
|
|
|
}
|
2010-06-25 17:03:26 +08:00
|
|
|
|
|
|
|
UnresolvedSet<8> Decls;
|
2019-01-09 23:43:19 +08:00
|
|
|
for (unsigned I = 0; I != NumResults; ++I) {
|
2018-04-12 04:57:28 +08:00
|
|
|
auto *D = ReadDeclAs<NamedDecl>();
|
|
|
|
auto AS = (AccessSpecifier)Record.readInt();
|
2010-06-25 17:03:26 +08:00
|
|
|
Decls.addDecl(D, AS);
|
|
|
|
}
|
2019-01-09 23:43:19 +08:00
|
|
|
|
|
|
|
DeclAccessPair *Results = E->getTrailingResults();
|
|
|
|
UnresolvedSetIterator Iter = Decls.begin();
|
|
|
|
for (unsigned I = 0; I != NumResults; ++I) {
|
|
|
|
Results[I] = (Iter + I).getPair();
|
|
|
|
}
|
2010-06-25 17:03:26 +08:00
|
|
|
|
2016-12-16 04:53:26 +08:00
|
|
|
ReadDeclarationNameInfo(E->NameInfo);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->QualifierLoc = Record.readNestedNameSpecifierLoc();
|
2010-06-25 17:03:26 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitUnresolvedMemberExpr(UnresolvedMemberExpr *E) {
|
2010-06-29 06:28:35 +08:00
|
|
|
VisitOverloadExpr(E);
|
2019-01-09 23:43:19 +08:00
|
|
|
E->UnresolvedMemberExprBits.IsArrow = Record.readInt();
|
|
|
|
E->UnresolvedMemberExprBits.HasUnresolvedUsing = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Base = Record.readSubExpr();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->BaseType = Record.readType();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->OperatorLoc = ReadSourceLocation();
|
2010-06-25 17:03:26 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:52 +08:00
|
|
|
void ASTStmtReader::VisitUnresolvedLookupExpr(UnresolvedLookupExpr *E) {
|
2010-06-29 06:28:35 +08:00
|
|
|
VisitOverloadExpr(E);
|
2019-01-09 23:43:19 +08:00
|
|
|
E->UnresolvedLookupExprBits.RequiresADL = Record.readInt();
|
|
|
|
E->UnresolvedLookupExprBits.Overloaded = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->NamingClass = ReadDeclAs<CXXRecordDecl>();
|
2010-06-25 17:03:34 +08:00
|
|
|
}
|
|
|
|
|
2012-02-24 15:38:34 +08:00
|
|
|
void ASTStmtReader::VisitTypeTraitExpr(TypeTraitExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->TypeTraitExprBits.NumArgs = Record.readInt();
|
|
|
|
E->TypeTraitExprBits.Kind = Record.readInt();
|
|
|
|
E->TypeTraitExprBits.Value = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceRange Range = ReadSourceRange();
|
2013-12-20 09:26:47 +08:00
|
|
|
E->Loc = Range.getBegin();
|
|
|
|
E->RParenLoc = Range.getEnd();
|
|
|
|
|
2018-04-12 04:57:28 +08:00
|
|
|
auto **Args = E->getTrailingObjects<TypeSourceInfo *>();
|
2012-02-24 15:38:34 +08:00
|
|
|
for (unsigned I = 0, N = E->getNumArgs(); I != N; ++I)
|
2016-12-16 04:53:26 +08:00
|
|
|
Args[I] = GetTypeSourceInfo();
|
2012-02-24 15:38:34 +08:00
|
|
|
}
|
|
|
|
|
2011-04-28 08:16:57 +08:00
|
|
|
void ASTStmtReader::VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->ATT = (ArrayTypeTrait)Record.readInt();
|
|
|
|
E->Value = (unsigned int)Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceRange Range = ReadSourceRange();
|
2011-04-28 08:16:57 +08:00
|
|
|
E->Loc = Range.getBegin();
|
|
|
|
E->RParen = Range.getEnd();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->QueriedType = GetTypeSourceInfo();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Dimension = Record.readSubExpr();
|
2011-04-28 08:16:57 +08:00
|
|
|
}
|
|
|
|
|
2011-04-25 14:54:41 +08:00
|
|
|
void ASTStmtReader::VisitExpressionTraitExpr(ExpressionTraitExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->ET = (ExpressionTrait)Record.readInt();
|
|
|
|
E->Value = (bool)Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
SourceRange Range = ReadSourceRange();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->QueriedExpression = Record.readSubExpr();
|
2011-04-25 14:54:41 +08:00
|
|
|
E->Loc = Range.getBegin();
|
|
|
|
E->RParen = Range.getEnd();
|
|
|
|
}
|
|
|
|
|
2010-09-11 04:55:54 +08:00
|
|
|
void ASTStmtReader::VisitCXXNoexceptExpr(CXXNoexceptExpr *E) {
|
|
|
|
VisitExpr(E);
|
2019-01-08 22:44:34 +08:00
|
|
|
E->CXXNoexceptExprBits.Value = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Range = ReadSourceRange();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Operand = Record.readSubExpr();
|
2010-09-11 04:55:54 +08:00
|
|
|
}
|
|
|
|
|
2011-01-04 01:17:50 +08:00
|
|
|
void ASTStmtReader::VisitPackExpansionExpr(PackExpansionExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->EllipsisLoc = ReadSourceLocation();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->NumExpansions = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Pattern = Record.readSubExpr();
|
2011-01-04 01:17:50 +08:00
|
|
|
}
|
|
|
|
|
2011-01-05 01:33:58 +08:00
|
|
|
void ASTStmtReader::VisitSizeOfPackExpr(SizeOfPackExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
unsigned NumPartialArgs = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->OperatorLoc = ReadSourceLocation();
|
|
|
|
E->PackLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Pack = Record.readDeclAs<NamedDecl>();
|
2015-09-24 05:41:42 +08:00
|
|
|
if (E->isPartiallySubstituted()) {
|
|
|
|
assert(E->Length == NumPartialArgs);
|
2015-12-31 12:18:25 +08:00
|
|
|
for (auto *I = E->getTrailingObjects<TemplateArgument>(),
|
2015-09-24 05:41:42 +08:00
|
|
|
*E = I + NumPartialArgs;
|
|
|
|
I != E; ++I)
|
2016-12-21 12:34:52 +08:00
|
|
|
new (I) TemplateArgument(Record.readTemplateArgument());
|
2015-09-24 05:41:42 +08:00
|
|
|
} else if (!E->isValueDependent()) {
|
2016-12-21 08:17:49 +08:00
|
|
|
E->Length = Record.readInt();
|
2015-09-24 05:41:42 +08:00
|
|
|
}
|
2011-01-05 01:33:58 +08:00
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
void ASTStmtReader::VisitSubstNonTypeTemplateParmExpr(
|
|
|
|
SubstNonTypeTemplateParmExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Param = ReadDeclAs<NonTypeTemplateParmDecl>();
|
2019-01-08 22:44:34 +08:00
|
|
|
E->SubstNonTypeTemplateParmExprBits.NameLoc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->Replacement = Record.readSubExpr();
|
2011-07-15 15:00:14 +08:00
|
|
|
}
|
|
|
|
|
2011-01-15 09:15:58 +08:00
|
|
|
void ASTStmtReader::VisitSubstNonTypeTemplateParmPackExpr(
|
|
|
|
SubstNonTypeTemplateParmPackExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->Param = ReadDeclAs<NonTypeTemplateParmDecl>();
|
2016-12-21 12:34:52 +08:00
|
|
|
TemplateArgument ArgPack = Record.readTemplateArgument();
|
2011-01-15 09:15:58 +08:00
|
|
|
if (ArgPack.getKind() != TemplateArgument::Pack)
|
|
|
|
return;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-01-15 09:15:58 +08:00
|
|
|
E->Arguments = ArgPack.pack_begin();
|
|
|
|
E->NumArguments = ArgPack.pack_size();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->NameLoc = ReadSourceLocation();
|
2011-01-15 09:15:58 +08:00
|
|
|
}
|
|
|
|
|
2012-09-12 08:56:43 +08:00
|
|
|
void ASTStmtReader::VisitFunctionParmPackExpr(FunctionParmPackExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->NumParameters = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->ParamPack = ReadDeclAs<ParmVarDecl>();
|
|
|
|
E->NameLoc = ReadSourceLocation();
|
2019-05-22 04:10:50 +08:00
|
|
|
auto **Parms = E->getTrailingObjects<VarDecl *>();
|
2012-09-12 08:56:43 +08:00
|
|
|
for (unsigned i = 0, n = E->NumParameters; i != n; ++i)
|
2019-05-22 04:10:50 +08:00
|
|
|
Parms[i] = ReadDeclAs<VarDecl>();
|
2012-09-12 08:56:43 +08:00
|
|
|
}
|
|
|
|
|
2011-06-22 01:03:29 +08:00
|
|
|
void ASTStmtReader::VisitMaterializeTemporaryExpr(MaterializeTemporaryExpr *E) {
|
|
|
|
VisitExpr(E);
|
2019-11-17 18:41:55 +08:00
|
|
|
bool HasMaterialzedDecl = Record.readInt();
|
|
|
|
if (HasMaterialzedDecl)
|
|
|
|
E->State = cast<LifetimeExtendedTemporaryDecl>(Record.readDecl());
|
|
|
|
else
|
|
|
|
E->State = Record.readSubExpr();
|
2011-06-22 01:03:29 +08:00
|
|
|
}
|
|
|
|
|
2014-11-08 13:07:16 +08:00
|
|
|
void ASTStmtReader::VisitCXXFoldExpr(CXXFoldExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->LParenLoc = ReadSourceLocation();
|
|
|
|
E->EllipsisLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
2019-05-13 16:31:14 +08:00
|
|
|
E->NumExpansions = Record.readInt();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SubExprs[0] = Record.readSubExpr();
|
|
|
|
E->SubExprs[1] = Record.readSubExpr();
|
2016-12-21 08:17:49 +08:00
|
|
|
E->Opcode = (BinaryOperatorKind)Record.readInt();
|
2014-11-08 13:07:16 +08:00
|
|
|
}
|
|
|
|
|
2010-11-16 07:31:06 +08:00
|
|
|
void ASTStmtReader::VisitOpaqueValueExpr(OpaqueValueExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SourceExpr = Record.readSubExpr();
|
2019-01-07 21:39:26 +08:00
|
|
|
E->OpaqueValueExprBits.Loc = ReadSourceLocation();
|
2018-03-20 09:47:58 +08:00
|
|
|
E->setIsUnique(Record.readInt());
|
2010-11-16 07:31:06 +08:00
|
|
|
}
|
|
|
|
|
2014-10-28 02:07:20 +08:00
|
|
|
void ASTStmtReader::VisitTypoExpr(TypoExpr *E) {
|
|
|
|
llvm_unreachable("Cannot read TypoExpr nodes");
|
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Microsoft Expressions and Statements
|
|
|
|
//===----------------------------------------------------------------------===//
|
2013-04-16 15:28:30 +08:00
|
|
|
void ASTStmtReader::VisitMSPropertyRefExpr(MSPropertyRefExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 08:17:49 +08:00
|
|
|
E->IsArrow = (Record.readInt() != 0);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->BaseExpr = Record.readSubExpr();
|
|
|
|
E->QualifierLoc = Record.readNestedNameSpecifierLoc();
|
2016-12-16 04:53:26 +08:00
|
|
|
E->MemberLoc = ReadSourceLocation();
|
|
|
|
E->TheDecl = ReadDeclAs<MSPropertyDecl>();
|
2013-04-16 15:28:30 +08:00
|
|
|
}
|
|
|
|
|
2015-11-25 20:01:00 +08:00
|
|
|
void ASTStmtReader::VisitMSPropertySubscriptExpr(MSPropertySubscriptExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setBase(Record.readSubExpr());
|
|
|
|
E->setIdx(Record.readSubExpr());
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setRBracketLoc(ReadSourceLocation());
|
2015-11-25 20:01:00 +08:00
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
void ASTStmtReader::VisitCXXUuidofExpr(CXXUuidofExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setSourceRange(ReadSourceRange());
|
|
|
|
std::string UuidStr = ReadString();
|
|
|
|
E->setUuidStr(StringRef(UuidStr).copy(Record.getContext()));
|
2011-07-15 15:00:14 +08:00
|
|
|
if (E->isTypeOperand()) { // __uuidof(ComType)
|
|
|
|
E->setTypeOperandSourceInfo(
|
2016-12-16 04:53:26 +08:00
|
|
|
GetTypeSourceInfo());
|
2011-07-15 15:00:14 +08:00
|
|
|
return;
|
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
// __uuidof(expr)
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setExprOperand(Record.readSubExpr());
|
2011-07-15 15:00:14 +08:00
|
|
|
}
|
|
|
|
|
2014-07-07 08:12:30 +08:00
|
|
|
void ASTStmtReader::VisitSEHLeaveStmt(SEHLeaveStmt *S) {
|
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->setLeaveLoc(ReadSourceLocation());
|
2014-07-07 08:12:30 +08:00
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
void ASTStmtReader::VisitSEHExceptStmt(SEHExceptStmt *S) {
|
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->Loc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->Children[SEHExceptStmt::FILTER_EXPR] = Record.readSubStmt();
|
|
|
|
S->Children[SEHExceptStmt::BLOCK] = Record.readSubStmt();
|
2011-07-15 15:00:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitSEHFinallyStmt(SEHFinallyStmt *S) {
|
|
|
|
VisitStmt(S);
|
2016-12-16 04:53:26 +08:00
|
|
|
S->Loc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->Block = Record.readSubStmt();
|
2011-07-15 15:00:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitSEHTryStmt(SEHTryStmt *S) {
|
|
|
|
VisitStmt(S);
|
2016-12-21 08:17:49 +08:00
|
|
|
S->IsCXXTry = Record.readInt();
|
2016-12-16 04:53:26 +08:00
|
|
|
S->TryLoc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
S->Children[SEHTryStmt::TRY] = Record.readSubStmt();
|
|
|
|
S->Children[SEHTryStmt::HANDLER] = Record.readSubStmt();
|
2011-07-15 15:00:14 +08:00
|
|
|
}
|
|
|
|
|
2011-02-10 05:07:24 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// CUDA Expressions and Statements
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitCUDAKernelCallExpr(CUDAKernelCallExpr *E) {
|
|
|
|
VisitCallExpr(E);
|
2019-05-25 07:26:07 +08:00
|
|
|
E->setPreArg(CUDAKernelCallExpr::CONFIG, Record.readSubExpr());
|
2011-02-10 05:07:24 +08:00
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// OpenCL Expressions and Statements.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
void ASTStmtReader::VisitAsTypeExpr(AsTypeExpr *E) {
|
|
|
|
VisitExpr(E);
|
2016-12-16 04:53:26 +08:00
|
|
|
E->BuiltinLoc = ReadSourceLocation();
|
|
|
|
E->RParenLoc = ReadSourceLocation();
|
2016-12-21 12:34:52 +08:00
|
|
|
E->SrcExpr = Record.readSubExpr();
|
2011-07-15 15:00:14 +08:00
|
|
|
}
|
|
|
|
|
2013-07-19 11:13:43 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// OpenMP Directives.
|
|
|
|
//===----------------------------------------------------------------------===//
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2013-07-19 11:13:43 +08:00
|
|
|
void ASTStmtReader::VisitOMPExecutableDirective(OMPExecutableDirective *E) {
|
2016-12-16 04:53:26 +08:00
|
|
|
E->setLocStart(ReadSourceLocation());
|
|
|
|
E->setLocEnd(ReadSourceLocation());
|
2018-09-15 21:54:15 +08:00
|
|
|
OMPClauseReader ClauseReader(Record);
|
2013-07-19 11:13:43 +08:00
|
|
|
SmallVector<OMPClause *, 5> Clauses;
|
|
|
|
for (unsigned i = 0; i < E->getNumClauses(); ++i)
|
|
|
|
Clauses.push_back(ClauseReader.readClause());
|
|
|
|
E->setClauses(Clauses);
|
2014-07-18 15:47:19 +08:00
|
|
|
if (E->hasAssociatedStmt())
|
2016-12-21 12:34:52 +08:00
|
|
|
E->setAssociatedStmt(Record.readSubStmt());
|
2013-07-19 11:13:43 +08:00
|
|
|
}
|
|
|
|
|
2014-08-19 19:27:13 +08:00
|
|
|
void ASTStmtReader::VisitOMPLoopDirective(OMPLoopDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// Two fields (NumClauses and CollapsedNum) were read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(2);
|
2014-08-19 19:27:13 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 12:34:52 +08:00
|
|
|
D->setIterationVariable(Record.readSubExpr());
|
|
|
|
D->setLastIteration(Record.readSubExpr());
|
|
|
|
D->setCalcLastIteration(Record.readSubExpr());
|
|
|
|
D->setPreCond(Record.readSubExpr());
|
|
|
|
D->setCond(Record.readSubExpr());
|
|
|
|
D->setInit(Record.readSubExpr());
|
|
|
|
D->setInc(Record.readSubExpr());
|
|
|
|
D->setPreInits(Record.readSubStmt());
|
2016-02-16 19:18:12 +08:00
|
|
|
if (isOpenMPWorksharingDirective(D->getDirectiveKind()) ||
|
2016-03-08 00:04:49 +08:00
|
|
|
isOpenMPTaskLoopDirective(D->getDirectiveKind()) ||
|
|
|
|
isOpenMPDistributeDirective(D->getDirectiveKind())) {
|
2016-12-21 12:34:52 +08:00
|
|
|
D->setIsLastIterVariable(Record.readSubExpr());
|
|
|
|
D->setLowerBoundVariable(Record.readSubExpr());
|
|
|
|
D->setUpperBoundVariable(Record.readSubExpr());
|
|
|
|
D->setStrideVariable(Record.readSubExpr());
|
|
|
|
D->setEnsureUpperBound(Record.readSubExpr());
|
|
|
|
D->setNextLowerBound(Record.readSubExpr());
|
|
|
|
D->setNextUpperBound(Record.readSubExpr());
|
|
|
|
D->setNumIterations(Record.readSubExpr());
|
2014-12-15 15:07:06 +08:00
|
|
|
}
|
2016-06-27 22:55:37 +08:00
|
|
|
if (isOpenMPLoopBoundSharingDirective(D->getDirectiveKind())) {
|
2016-12-21 12:34:52 +08:00
|
|
|
D->setPrevLowerBoundVariable(Record.readSubExpr());
|
|
|
|
D->setPrevUpperBoundVariable(Record.readSubExpr());
|
2017-02-18 05:29:13 +08:00
|
|
|
D->setDistInc(Record.readSubExpr());
|
|
|
|
D->setPrevEnsureUpperBound(Record.readSubExpr());
|
2017-04-20 08:39:39 +08:00
|
|
|
D->setCombinedLowerBoundVariable(Record.readSubExpr());
|
|
|
|
D->setCombinedUpperBoundVariable(Record.readSubExpr());
|
|
|
|
D->setCombinedEnsureUpperBound(Record.readSubExpr());
|
|
|
|
D->setCombinedInit(Record.readSubExpr());
|
|
|
|
D->setCombinedCond(Record.readSubExpr());
|
|
|
|
D->setCombinedNextLowerBound(Record.readSubExpr());
|
|
|
|
D->setCombinedNextUpperBound(Record.readSubExpr());
|
2018-10-29 23:45:47 +08:00
|
|
|
D->setCombinedDistCond(Record.readSubExpr());
|
|
|
|
D->setCombinedParForInDistCond(Record.readSubExpr());
|
2016-06-27 22:55:37 +08:00
|
|
|
}
|
2014-10-01 14:03:56 +08:00
|
|
|
SmallVector<Expr *, 4> Sub;
|
|
|
|
unsigned CollapsedNum = D->getCollapsedNumber();
|
|
|
|
Sub.reserve(CollapsedNum);
|
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
Sub.push_back(Record.readSubExpr());
|
2014-10-01 14:03:56 +08:00
|
|
|
D->setCounters(Sub);
|
|
|
|
Sub.clear();
|
2015-08-06 20:30:57 +08:00
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
Sub.push_back(Record.readSubExpr());
|
2015-08-06 20:30:57 +08:00
|
|
|
D->setPrivateCounters(Sub);
|
|
|
|
Sub.clear();
|
2015-08-14 20:25:37 +08:00
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
Sub.push_back(Record.readSubExpr());
|
2015-08-14 20:25:37 +08:00
|
|
|
D->setInits(Sub);
|
|
|
|
Sub.clear();
|
2014-10-01 14:03:56 +08:00
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
Sub.push_back(Record.readSubExpr());
|
2014-10-01 14:03:56 +08:00
|
|
|
D->setUpdates(Sub);
|
|
|
|
Sub.clear();
|
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
2016-12-21 12:34:52 +08:00
|
|
|
Sub.push_back(Record.readSubExpr());
|
2014-10-01 14:03:56 +08:00
|
|
|
D->setFinals(Sub);
|
2019-08-15 03:30:06 +08:00
|
|
|
Sub.clear();
|
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
|
|
|
Sub.push_back(Record.readSubExpr());
|
|
|
|
D->setDependentCounters(Sub);
|
|
|
|
Sub.clear();
|
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
|
|
|
Sub.push_back(Record.readSubExpr());
|
|
|
|
D->setDependentInits(Sub);
|
|
|
|
Sub.clear();
|
|
|
|
for (unsigned i = 0; i < CollapsedNum; ++i)
|
|
|
|
Sub.push_back(Record.readSubExpr());
|
|
|
|
D->setFinalsConditions(Sub);
|
2014-08-19 19:27:13 +08:00
|
|
|
}
|
|
|
|
|
2013-07-19 11:13:43 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelDirective(OMPParallelDirective *D) {
|
2014-02-27 16:29:12 +08:00
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-02-27 16:29:12 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-02-27 16:29:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTStmtReader::VisitOMPSimdDirective(OMPSimdDirective *D) {
|
2014-08-19 19:27:13 +08:00
|
|
|
VisitOMPLoopDirective(D);
|
2013-07-19 11:13:43 +08:00
|
|
|
}
|
|
|
|
|
2014-06-18 12:14:57 +08:00
|
|
|
void ASTStmtReader::VisitOMPForDirective(OMPForDirective *D) {
|
2014-08-19 19:27:13 +08:00
|
|
|
VisitOMPLoopDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-06-18 12:14:57 +08:00
|
|
|
}
|
|
|
|
|
2014-09-18 13:12:34 +08:00
|
|
|
void ASTStmtReader::VisitOMPForSimdDirective(OMPForSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-06-25 19:44:49 +08:00
|
|
|
void ASTStmtReader::VisitOMPSectionsDirective(OMPSectionsDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-06-25 19:44:49 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-06-25 19:44:49 +08:00
|
|
|
}
|
|
|
|
|
2014-06-26 16:21:58 +08:00
|
|
|
void ASTStmtReader::VisitOMPSectionDirective(OMPSectionDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-06-26 16:21:58 +08:00
|
|
|
}
|
|
|
|
|
2014-06-26 20:05:45 +08:00
|
|
|
void ASTStmtReader::VisitOMPSingleDirective(OMPSingleDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-06-26 20:05:45 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-17 16:54:58 +08:00
|
|
|
void ASTStmtReader::VisitOMPMasterDirective(OMPMasterDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-21 17:42:05 +08:00
|
|
|
void ASTStmtReader::VisitOMPCriticalDirective(OMPCriticalDirective *D) {
|
|
|
|
VisitStmt(D);
|
2015-12-15 16:19:24 +08:00
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-07-21 17:42:05 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-16 04:53:26 +08:00
|
|
|
ReadDeclarationNameInfo(D->DirName);
|
2014-07-21 17:42:05 +08:00
|
|
|
}
|
|
|
|
|
2014-07-07 21:01:15 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelForDirective(OMPParallelForDirective *D) {
|
2014-08-19 19:27:13 +08:00
|
|
|
VisitOMPLoopDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-07-07 21:01:15 +08:00
|
|
|
}
|
|
|
|
|
2014-09-23 17:33:00 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelForSimdDirective(
|
|
|
|
OMPParallelForSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2019-12-05 03:36:07 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelMasterDirective(
|
|
|
|
OMPParallelMasterDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
|
|
|
Record.skipInts(1);
|
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-08 16:12:03 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelSectionsDirective(
|
|
|
|
OMPParallelSectionsDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-07-08 16:12:03 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-07-08 16:12:03 +08:00
|
|
|
}
|
|
|
|
|
2014-07-11 19:25:16 +08:00
|
|
|
void ASTStmtReader::VisitOMPTaskDirective(OMPTaskDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-07-11 19:25:16 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2014-07-11 19:25:16 +08:00
|
|
|
}
|
|
|
|
|
2014-07-18 15:47:19 +08:00
|
|
|
void ASTStmtReader::VisitOMPTaskyieldDirective(OMPTaskyieldDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-18 17:11:51 +08:00
|
|
|
void ASTStmtReader::VisitOMPBarrierDirective(OMPBarrierDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-18 18:17:07 +08:00
|
|
|
void ASTStmtReader::VisitOMPTaskwaitDirective(OMPTaskwaitDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2015-06-18 20:14:09 +08:00
|
|
|
void ASTStmtReader::VisitOMPTaskgroupDirective(OMPTaskgroupDirective *D) {
|
|
|
|
VisitStmt(D);
|
2017-07-19 04:17:46 +08:00
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
|
|
|
Record.skipInts(1);
|
2015-06-18 20:14:09 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2017-07-25 23:53:26 +08:00
|
|
|
D->setReductionRef(Record.readSubExpr());
|
2015-06-18 20:14:09 +08:00
|
|
|
}
|
|
|
|
|
2014-07-21 19:26:11 +08:00
|
|
|
void ASTStmtReader::VisitOMPFlushDirective(OMPFlushDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-07-21 19:26:11 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-22 14:45:04 +08:00
|
|
|
void ASTStmtReader::VisitOMPOrderedDirective(OMPOrderedDirective *D) {
|
|
|
|
VisitStmt(D);
|
2015-09-25 18:37:12 +08:00
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-07-22 14:45:04 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2014-07-22 18:10:35 +08:00
|
|
|
void ASTStmtReader::VisitOMPAtomicDirective(OMPAtomicDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-07-22 18:10:35 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 12:34:52 +08:00
|
|
|
D->setX(Record.readSubExpr());
|
|
|
|
D->setV(Record.readSubExpr());
|
|
|
|
D->setExpr(Record.readSubExpr());
|
|
|
|
D->setUpdateExpr(Record.readSubExpr());
|
2016-12-21 08:17:49 +08:00
|
|
|
D->IsXLHSInRHSPart = Record.readInt() != 0;
|
|
|
|
D->IsPostfixUpdate = Record.readInt() != 0;
|
2014-07-22 18:10:35 +08:00
|
|
|
}
|
|
|
|
|
2014-09-19 16:19:49 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetDirective(OMPTargetDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-09-19 16:19:49 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2015-07-21 21:44:28 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetDataDirective(OMPTargetDataDirective *D) {
|
|
|
|
VisitStmt(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2015-07-21 21:44:28 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-01-20 03:15:56 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetEnterDataDirective(
|
|
|
|
OMPTargetEnterDataDirective *D) {
|
|
|
|
VisitStmt(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2016-01-20 03:15:56 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-01-20 04:04:50 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetExitDataDirective(
|
|
|
|
OMPTargetExitDataDirective *D) {
|
|
|
|
VisitStmt(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2016-01-20 04:04:50 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-01-27 02:48:41 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetParallelDirective(
|
|
|
|
OMPTargetParallelDirective *D) {
|
|
|
|
VisitStmt(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2016-01-27 02:48:41 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-02-03 23:46:42 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetParallelForDirective(
|
|
|
|
OMPTargetParallelForDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2016-02-03 23:46:42 +08:00
|
|
|
}
|
|
|
|
|
2014-10-09 12:18:56 +08:00
|
|
|
void ASTStmtReader::VisitOMPTeamsDirective(OMPTeamsDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2014-10-09 12:18:56 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2015-07-01 14:57:41 +08:00
|
|
|
void ASTStmtReader::VisitOMPCancellationPointDirective(
|
|
|
|
OMPCancellationPointDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setCancelRegion(static_cast<OpenMPDirectiveKind>(Record.readInt()));
|
2015-07-01 14:57:41 +08:00
|
|
|
}
|
|
|
|
|
2015-07-02 19:25:17 +08:00
|
|
|
void ASTStmtReader::VisitOMPCancelDirective(OMPCancelDirective *D) {
|
|
|
|
VisitStmt(D);
|
2015-09-18 16:07:34 +08:00
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2015-07-02 19:25:17 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
D->setCancelRegion(static_cast<OpenMPDirectiveKind>(Record.readInt()));
|
2015-07-02 19:25:17 +08:00
|
|
|
}
|
|
|
|
|
2015-12-01 12:18:41 +08:00
|
|
|
void ASTStmtReader::VisitOMPTaskLoopDirective(OMPTaskLoopDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2015-12-03 17:40:15 +08:00
|
|
|
void ASTStmtReader::VisitOMPTaskLoopSimdDirective(OMPTaskLoopSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2019-10-11 04:13:02 +08:00
|
|
|
void ASTStmtReader::VisitOMPMasterTaskLoopDirective(
|
|
|
|
OMPMasterTaskLoopDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2019-10-19 00:47:35 +08:00
|
|
|
void ASTStmtReader::VisitOMPMasterTaskLoopSimdDirective(
|
|
|
|
OMPMasterTaskLoopSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2019-10-15 01:17:41 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelMasterTaskLoopDirective(
|
|
|
|
OMPParallelMasterTaskLoopDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2019-10-25 22:27:13 +08:00
|
|
|
void ASTStmtReader::VisitOMPParallelMasterTaskLoopSimdDirective(
|
|
|
|
OMPParallelMasterTaskLoopSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2015-12-14 22:51:25 +08:00
|
|
|
void ASTStmtReader::VisitOMPDistributeDirective(OMPDistributeDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-05-27 01:30:50 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetUpdateDirective(OMPTargetUpdateDirective *D) {
|
|
|
|
VisitStmt(D);
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2016-05-27 01:30:50 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2016-06-27 22:55:37 +08:00
|
|
|
void ASTStmtReader::VisitOMPDistributeParallelForDirective(
|
|
|
|
OMPDistributeParallelForDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
2017-11-23 04:19:50 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2016-06-27 22:55:37 +08:00
|
|
|
}
|
2016-05-27 01:30:50 +08:00
|
|
|
|
2016-07-05 13:00:15 +08:00
|
|
|
void ASTStmtReader::VisitOMPDistributeParallelForSimdDirective(
|
|
|
|
OMPDistributeParallelForSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-07-06 12:45:38 +08:00
|
|
|
void ASTStmtReader::VisitOMPDistributeSimdDirective(
|
|
|
|
OMPDistributeSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-07-14 10:54:56 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetParallelForSimdDirective(
|
|
|
|
OMPTargetParallelForSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-07-21 06:57:10 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetSimdDirective(OMPTargetSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-08-05 22:37:37 +08:00
|
|
|
void ASTStmtReader::VisitOMPTeamsDistributeDirective(
|
|
|
|
OMPTeamsDistributeDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-10-25 20:50:55 +08:00
|
|
|
void ASTStmtReader::VisitOMPTeamsDistributeSimdDirective(
|
|
|
|
OMPTeamsDistributeSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-12-01 07:51:03 +08:00
|
|
|
void ASTStmtReader::VisitOMPTeamsDistributeParallelForSimdDirective(
|
|
|
|
OMPTeamsDistributeParallelForSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-12-09 11:24:30 +08:00
|
|
|
void ASTStmtReader::VisitOMPTeamsDistributeParallelForDirective(
|
|
|
|
OMPTeamsDistributeParallelForDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
2017-11-23 04:19:50 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2016-12-09 11:24:30 +08:00
|
|
|
}
|
|
|
|
|
2016-12-17 13:48:59 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetTeamsDirective(OMPTargetTeamsDirective *D) {
|
|
|
|
VisitStmt(D);
|
|
|
|
// The NumClauses field was read in ReadStmtFromStream.
|
2016-12-21 08:17:49 +08:00
|
|
|
Record.skipInts(1);
|
2016-12-17 13:48:59 +08:00
|
|
|
VisitOMPExecutableDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-12-25 12:52:54 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetTeamsDistributeDirective(
|
|
|
|
OMPTargetTeamsDistributeDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2016-12-30 06:16:30 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetTeamsDistributeParallelForDirective(
|
|
|
|
OMPTargetTeamsDistributeParallelForDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
2017-11-23 05:12:03 +08:00
|
|
|
D->setHasCancel(Record.readInt());
|
2016-12-30 06:16:30 +08:00
|
|
|
}
|
|
|
|
|
2017-01-03 13:23:48 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetTeamsDistributeParallelForSimdDirective(
|
|
|
|
OMPTargetTeamsDistributeParallelForSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2017-01-11 02:08:18 +08:00
|
|
|
void ASTStmtReader::VisitOMPTargetTeamsDistributeSimdDirective(
|
|
|
|
OMPTargetTeamsDistributeSimdDirective *D) {
|
|
|
|
VisitOMPLoopDirective(D);
|
|
|
|
}
|
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ASTReader Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
Stmt *ASTReader::ReadStmt(ModuleFile &F) {
|
2010-06-29 06:28:35 +08:00
|
|
|
switch (ReadingKind) {
|
2013-07-31 08:26:46 +08:00
|
|
|
case Read_None:
|
|
|
|
llvm_unreachable("should not call this when not reading anything");
|
2010-06-29 06:28:35 +08:00
|
|
|
case Read_Decl:
|
|
|
|
case Read_Type:
|
2010-10-05 23:59:54 +08:00
|
|
|
return ReadStmtFromStream(F);
|
2010-06-29 06:28:35 +08:00
|
|
|
case Read_Stmt:
|
2010-06-30 06:46:25 +08:00
|
|
|
return ReadSubStmt();
|
2010-06-29 06:28:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("ReadingKind not set ?");
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
Expr *ASTReader::ReadExpr(ModuleFile &F) {
|
2010-10-05 23:59:54 +08:00
|
|
|
return cast_or_null<Expr>(ReadStmt(F));
|
2010-06-29 06:28:35 +08:00
|
|
|
}
|
2010-05-09 14:40:08 +08:00
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
Expr *ASTReader::ReadSubExpr() {
|
2010-06-30 06:46:25 +08:00
|
|
|
return cast_or_null<Expr>(ReadSubStmt());
|
|
|
|
}
|
|
|
|
|
2009-04-27 13:41:06 +08:00
|
|
|
// Within the bitstream, expressions are stored in Reverse Polish
|
|
|
|
// Notation, with each of the subexpressions preceding the
|
2010-06-29 06:28:35 +08:00
|
|
|
// expression they are stored in. Subexpressions are stored from last to first.
|
|
|
|
// To evaluate expressions, we continue reading expressions and placing them on
|
|
|
|
// the stack, with expressions having operands removing those operands from the
|
2009-04-27 13:41:06 +08:00
|
|
|
// stack. Evaluation terminates when we see a STMT_STOP record, and
|
|
|
|
// the single remaining expression on the stack is our result.
|
2011-12-01 07:21:26 +08:00
|
|
|
Stmt *ASTReader::ReadStmtFromStream(ModuleFile &F) {
|
2010-06-29 06:28:35 +08:00
|
|
|
ReadingKindTracker ReadingKind(Read_Stmt, *this);
|
2010-10-05 23:59:54 +08:00
|
|
|
llvm::BitstreamCursor &Cursor = F.DeclsCursor;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-10-22 07:02:28 +08:00
|
|
|
// Map of offset to previously deserialized stmt. The offset points
|
2017-02-14 13:52:57 +08:00
|
|
|
// just after the stmt record.
|
2011-10-22 07:02:28 +08:00
|
|
|
llvm::DenseMap<uint64_t, Stmt *> StmtEntries;
|
2010-10-05 23:59:54 +08:00
|
|
|
|
2010-06-29 06:28:35 +08:00
|
|
|
#ifndef NDEBUG
|
|
|
|
unsigned PrevNumStmts = StmtStack.size();
|
|
|
|
#endif
|
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
ASTRecordReader Record(*this, F);
|
|
|
|
ASTStmtReader Reader(Record, Cursor);
|
2009-04-27 13:14:47 +08:00
|
|
|
Stmt::EmptyShell Empty;
|
|
|
|
|
|
|
|
while (true) {
|
2019-06-27 03:50:12 +08:00
|
|
|
llvm::Expected<llvm::BitstreamEntry> MaybeEntry =
|
|
|
|
Cursor.advanceSkippingSubblocks();
|
|
|
|
if (!MaybeEntry) {
|
|
|
|
Error(toString(MaybeEntry.takeError()));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
llvm::BitstreamEntry Entry = MaybeEntry.get();
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2013-01-20 10:38:54 +08:00
|
|
|
switch (Entry.Kind) {
|
|
|
|
case llvm::BitstreamEntry::SubBlock: // Handled for us already.
|
|
|
|
case llvm::BitstreamEntry::Error:
|
|
|
|
Error("malformed block record in AST file");
|
2014-05-22 13:54:18 +08:00
|
|
|
return nullptr;
|
2013-01-20 10:38:54 +08:00
|
|
|
case llvm::BitstreamEntry::EndBlock:
|
|
|
|
goto Done;
|
|
|
|
case llvm::BitstreamEntry::Record:
|
|
|
|
// The interesting case.
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-06-30 07:23:46 +08:00
|
|
|
ASTContext &Context = getContext();
|
2014-05-22 13:54:18 +08:00
|
|
|
Stmt *S = nullptr;
|
2009-04-27 13:14:47 +08:00
|
|
|
bool Finished = false;
|
2011-10-22 07:02:28 +08:00
|
|
|
bool IsStmtReference = false;
|
2019-06-27 03:50:12 +08:00
|
|
|
Expected<unsigned> MaybeStmtCode = Record.readRecord(Cursor, Entry.ID);
|
|
|
|
if (!MaybeStmtCode) {
|
|
|
|
Error(toString(MaybeStmtCode.takeError()));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
switch ((StmtCode)MaybeStmtCode.get()) {
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_STOP:
|
2009-04-27 13:14:47 +08:00
|
|
|
Finished = true;
|
|
|
|
break;
|
|
|
|
|
2011-10-22 07:02:28 +08:00
|
|
|
case STMT_REF_PTR:
|
|
|
|
IsStmtReference = true;
|
|
|
|
assert(StmtEntries.find(Record[0]) != StmtEntries.end() &&
|
|
|
|
"No stmt was recorded for this offset reference!");
|
2016-12-21 08:17:49 +08:00
|
|
|
S = StmtEntries[Record.readInt()];
|
2011-10-22 07:02:28 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_NULL_PTR:
|
2014-05-22 13:54:18 +08:00
|
|
|
S = nullptr;
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_NULL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) NullStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_COMPOUND:
|
2017-12-25 00:24:20 +08:00
|
|
|
S = CompoundStmt::CreateEmpty(
|
|
|
|
Context, /*NumStmts=*/Record[ASTStmtReader::NumStmtFields]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_CASE:
|
2018-10-28 20:30:53 +08:00
|
|
|
S = CaseStmt::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*CaseStmtIsGNURange*/ Record[ASTStmtReader::NumStmtFields + 3]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_DEFAULT:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) DefaultStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_LABEL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) LabelStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2012-04-14 08:33:13 +08:00
|
|
|
case STMT_ATTRIBUTED:
|
2012-07-09 18:04:07 +08:00
|
|
|
S = AttributedStmt::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*NumAttrs*/Record[ASTStmtReader::NumStmtFields]);
|
2012-04-14 08:33:13 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_IF:
|
2018-10-28 05:12:20 +08:00
|
|
|
S = IfStmt::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/* HasElse=*/Record[ASTStmtReader::NumStmtFields + 1],
|
|
|
|
/* HasVar=*/Record[ASTStmtReader::NumStmtFields + 2],
|
|
|
|
/* HasInit=*/Record[ASTStmtReader::NumStmtFields + 3]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_SWITCH:
|
2018-10-30 00:12:37 +08:00
|
|
|
S = SwitchStmt::CreateEmpty(
|
|
|
|
Context,
|
2018-12-04 00:17:45 +08:00
|
|
|
/* HasInit=*/Record[ASTStmtReader::NumStmtFields],
|
2018-10-30 00:12:37 +08:00
|
|
|
/* HasVar=*/Record[ASTStmtReader::NumStmtFields + 1]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_WHILE:
|
2018-10-30 21:42:41 +08:00
|
|
|
S = WhileStmt::CreateEmpty(
|
|
|
|
Context,
|
2018-12-04 00:17:45 +08:00
|
|
|
/* HasVar=*/Record[ASTStmtReader::NumStmtFields]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_DO:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) DoStmt(Empty);
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_FOR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ForStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_GOTO:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) GotoStmt(Empty);
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_INDIRECT_GOTO:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) IndirectGotoStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_CONTINUE:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ContinueStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_BREAK:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) BreakStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_RETURN:
|
2018-10-30 22:40:49 +08:00
|
|
|
S = ReturnStmt::CreateEmpty(
|
|
|
|
Context, /* HasNRVOCandidate=*/Record[ASTStmtReader::NumStmtFields]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_DECL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) DeclStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2012-08-25 08:11:56 +08:00
|
|
|
case STMT_GCCASM:
|
|
|
|
S = new (Context) GCCAsmStmt(Empty);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2012-08-25 07:51:02 +08:00
|
|
|
case STMT_MSASM:
|
|
|
|
S = new (Context) MSAsmStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2013-04-17 02:53:08 +08:00
|
|
|
case STMT_CAPTURED:
|
2018-10-28 03:21:19 +08:00
|
|
|
S = CapturedStmt::CreateDeserialized(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields]);
|
2013-04-17 02:53:08 +08:00
|
|
|
break;
|
|
|
|
|
2018-10-31 11:48:47 +08:00
|
|
|
case EXPR_CONSTANT:
|
[clang] Add storage for APValue in ConstantExpr
Summary:
When using ConstantExpr we often need the result of the expression to be kept in the AST. Currently this is done on a by the node that needs the result and has been done multiple times for enumerator, for constexpr variables... . This patch adds to ConstantExpr the ability to store the result of evaluating the expression. no functional changes expected.
Changes:
- Add trailling object to ConstantExpr that can hold an APValue or an uint64_t. the uint64_t is here because most ConstantExpr yield integral values so there is an optimized layout for integral values.
- Add basic* serialization support for the trailing result.
- Move conversion functions from an enum to a fltSemantics from clang::FloatingLiteral to llvm::APFloatBase. this change is to make it usable for serializing APValues.
- Add basic* Import support for the trailing result.
- ConstantExpr created in CheckConvertedConstantExpression now stores the result in the ConstantExpr Node.
- Adapt AST dump to print the result when present.
basic* : None, Indeterminate, Int, Float, FixedPoint, ComplexInt, ComplexFloat,
the result is not yet used anywhere but for -ast-dump.
Reviewers: rsmith, martong, shafik
Reviewed By: rsmith
Subscribers: rnkovacs, hiraditya, dexonsmith, cfe-commits, llvm-commits
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D62399
llvm-svn: 363493
2019-06-15 18:24:47 +08:00
|
|
|
S = ConstantExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
static_cast<ConstantExpr::ResultStorageKind>(
|
|
|
|
Record[ASTStmtReader::NumExprFields]),
|
|
|
|
Empty);
|
2018-10-31 11:48:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_PREDEFINED:
|
2018-10-28 03:21:19 +08:00
|
|
|
S = PredefinedExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*HasFunctionName*/ Record[ASTStmtReader::NumExprFields]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_DECL_REF:
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
S = DeclRefExpr::CreateEmpty(
|
2011-09-10 05:34:22 +08:00
|
|
|
Context,
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
/*HasQualifier=*/Record[ASTStmtReader::NumExprFields],
|
|
|
|
/*HasFoundDecl=*/Record[ASTStmtReader::NumExprFields + 1],
|
2012-01-27 17:46:47 +08:00
|
|
|
/*HasTemplateKWAndArgsInfo=*/Record[ASTStmtReader::NumExprFields + 2],
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
/*NumTemplateArgs=*/Record[ASTStmtReader::NumExprFields + 2] ?
|
2019-06-12 01:50:32 +08:00
|
|
|
Record[ASTStmtReader::NumExprFields + 6] : 0);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_INTEGER_LITERAL:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = IntegerLiteral::Create(Context, Empty);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_FLOATING_LITERAL:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = FloatingLiteral::Create(Context, Empty);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_IMAGINARY_LITERAL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ImaginaryLiteral(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_STRING_LITERAL:
|
2018-11-16 01:31:16 +08:00
|
|
|
S = StringLiteral::CreateEmpty(
|
|
|
|
Context,
|
2018-12-04 00:17:45 +08:00
|
|
|
/* NumConcatenated=*/Record[ASTStmtReader::NumExprFields],
|
2018-11-16 01:31:16 +08:00
|
|
|
/* Length=*/Record[ASTStmtReader::NumExprFields + 1],
|
|
|
|
/* CharByteWidth=*/Record[ASTStmtReader::NumExprFields + 2]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CHARACTER_LITERAL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) CharacterLiteral(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_PAREN:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ParenExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_PAREN_LIST:
|
2018-11-21 00:20:40 +08:00
|
|
|
S = ParenListExpr::CreateEmpty(
|
|
|
|
Context,
|
2018-12-04 00:17:45 +08:00
|
|
|
/* NumExprs=*/Record[ASTStmtReader::NumExprFields]);
|
2010-06-30 16:49:18 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_UNARY_OPERATOR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) UnaryOperator(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OFFSETOF:
|
2016-12-16 04:53:26 +08:00
|
|
|
S = OffsetOfExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
Record[ASTStmtReader::NumExprFields],
|
|
|
|
Record[ASTStmtReader::NumExprFields + 1]);
|
Completely reimplement __builtin_offsetof, based on a patch by Roberto
Amadini.
This change introduces a new expression node type, OffsetOfExpr, that
describes __builtin_offsetof. Previously, __builtin_offsetof was
implemented using a unary operator whose subexpression involved
various synthesized array-subscript and member-reference expressions,
which was ugly and made it very hard to instantiate as a
template. OffsetOfExpr represents the AST more faithfully, with proper
type source information and a more compact representation.
OffsetOfExpr also has support for dependent __builtin_offsetof
expressions; it can be value-dependent, but will never be
type-dependent (like sizeof or alignof). This commit introduces
template instantiation for __builtin_offsetof as well.
There are two major caveats to this patch:
1) CodeGen cannot handle the case where __builtin_offsetof is not a
constant expression, so it produces an error. So, to avoid
regressing in C, we retain the old UnaryOperator-based
__builtin_offsetof implementation in C while using the shiny new
OffsetOfExpr implementation in C++. The old implementation can go
away once we have proper CodeGen support for this case, which we
expect won't cause much trouble in C++.
2) __builtin_offsetof doesn't work well with non-POD class types,
particularly when the designated field is found within a base
class. I will address this in a subsequent patch.
Fixes PR5880 and a bunch of assertions when building Boost.Python
tests.
llvm-svn: 102542
2010-04-29 06:16:22 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_SIZEOF_ALIGN_OF:
|
2011-03-12 03:24:49 +08:00
|
|
|
S = new (Context) UnaryExprOrTypeTraitExpr(Empty);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_ARRAY_SUBSCRIPT:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ArraySubscriptExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2015-08-25 22:24:04 +08:00
|
|
|
case EXPR_OMP_ARRAY_SECTION:
|
|
|
|
S = new (Context) OMPArraySectionExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CALL:
|
2018-12-21 23:20:32 +08:00
|
|
|
S = CallExpr::CreateEmpty(
|
|
|
|
Context, /*NumArgs=*/Record[ASTStmtReader::NumExprFields], Empty);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2019-06-07 07:24:15 +08:00
|
|
|
case EXPR_MEMBER:
|
|
|
|
S = MemberExpr::CreateEmpty(Context, Record[ASTStmtReader::NumExprFields],
|
|
|
|
Record[ASTStmtReader::NumExprFields + 1],
|
|
|
|
Record[ASTStmtReader::NumExprFields + 2],
|
|
|
|
Record[ASTStmtReader::NumExprFields + 3]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_BINARY_OPERATOR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) BinaryOperator(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_COMPOUND_ASSIGN_OPERATOR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) CompoundAssignOperator(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CONDITIONAL_OPERATOR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ConditionalOperator(Empty);
|
|
|
|
break;
|
|
|
|
|
2011-02-17 18:25:35 +08:00
|
|
|
case EXPR_BINARY_CONDITIONAL_OPERATOR:
|
|
|
|
S = new (Context) BinaryConditionalOperator(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_IMPLICIT_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = ImplicitCastExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*PathSize*/ Record[ASTStmtReader::NumExprFields]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CSTYLE_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CStyleCastExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*PathSize*/ Record[ASTStmtReader::NumExprFields]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_COMPOUND_LITERAL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) CompoundLiteralExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_EXT_VECTOR_ELEMENT:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ExtVectorElementExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_INIT_LIST:
|
2012-11-28 11:56:16 +08:00
|
|
|
S = new (Context) InitListExpr(Empty);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_DESIGNATED_INIT:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = DesignatedInitExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
Record[ASTStmtReader::NumExprFields] - 1);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
|
|
|
|
2015-06-10 08:27:52 +08:00
|
|
|
case EXPR_DESIGNATED_INIT_UPDATE:
|
|
|
|
S = new (Context) DesignatedInitUpdateExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_IMPLICIT_VALUE_INIT:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ImplicitValueInitExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2015-06-10 08:27:52 +08:00
|
|
|
case EXPR_NO_INIT:
|
|
|
|
S = new (Context) NoInitExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2016-12-12 10:53:20 +08:00
|
|
|
case EXPR_ARRAY_INIT_LOOP:
|
|
|
|
S = new (Context) ArrayInitLoopExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EXPR_ARRAY_INIT_INDEX:
|
|
|
|
S = new (Context) ArrayInitIndexExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_VA_ARG:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) VAArgExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
Implement __builtin_LINE() et. al. to support source location capture.
Summary:
This patch implements the source location builtins `__builtin_LINE(), `__builtin_FUNCTION()`, `__builtin_FILE()` and `__builtin_COLUMN()`. These builtins are needed to implement [`std::experimental::source_location`](https://rawgit.com/cplusplus/fundamentals-ts/v2/main.html#reflection.src_loc.creation).
With the exception of `__builtin_COLUMN`, GCC also implements these builtins, and Clangs behavior is intended to match as closely as possible.
Reviewers: rsmith, joerg, aaron.ballman, bogner, majnemer, shafik, martong
Reviewed By: rsmith
Subscribers: rnkovacs, loskutov, riccibruno, mgorny, kunitoki, alexr, majnemer, hfinkel, cfe-commits
Differential Revision: https://reviews.llvm.org/D37035
llvm-svn: 360937
2019-05-17 05:04:15 +08:00
|
|
|
case EXPR_SOURCE_LOC:
|
|
|
|
S = new (Context) SourceLocExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_ADDR_LABEL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) AddrLabelExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_STMT:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) StmtExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CHOOSE:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ChooseExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_GNU_NULL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) GNUNullExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_SHUFFLE_VECTOR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ShuffleVectorExpr(Empty);
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2013-09-18 11:29:45 +08:00
|
|
|
case EXPR_CONVERT_VECTOR:
|
|
|
|
S = new (Context) ConvertVectorExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_BLOCK:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) BlockExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2011-04-15 08:35:48 +08:00
|
|
|
case EXPR_GENERIC_SELECTION:
|
2019-01-26 22:15:10 +08:00
|
|
|
S = GenericSelectionExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*NumAssocs=*/Record[ASTStmtReader::NumExprFields]);
|
2011-04-15 08:35:48 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_STRING_LITERAL:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCStringLiteral(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
case EXPR_OBJC_BOXED_EXPRESSION:
|
|
|
|
S = new (Context) ObjCBoxedExpr(Empty);
|
2012-03-07 04:05:56 +08:00
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case EXPR_OBJC_ARRAY_LITERAL:
|
|
|
|
S = ObjCArrayLiteral::CreateEmpty(Context,
|
|
|
|
Record[ASTStmtReader::NumExprFields]);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case EXPR_OBJC_DICTIONARY_LITERAL:
|
|
|
|
S = ObjCDictionaryLiteral::CreateEmpty(Context,
|
|
|
|
Record[ASTStmtReader::NumExprFields],
|
|
|
|
Record[ASTStmtReader::NumExprFields + 1]);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_ENCODE:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCEncodeExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_SELECTOR_EXPR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCSelectorExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_PROTOCOL_EXPR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCProtocolExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_IVAR_REF_EXPR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCIvarRefExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_PROPERTY_REF_EXPR:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCPropertyRefExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case EXPR_OBJC_SUBSCRIPT_REF_EXPR:
|
|
|
|
S = new (Context) ObjCSubscriptRefExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_KVC_REF_EXPR:
|
2010-12-02 09:19:52 +08:00
|
|
|
llvm_unreachable("mismatching AST file");
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_MESSAGE_EXPR:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = ObjCMessageExpr::CreateEmpty(Context,
|
2011-10-03 14:36:51 +08:00
|
|
|
Record[ASTStmtReader::NumExprFields],
|
|
|
|
Record[ASTStmtReader::NumExprFields + 1]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_OBJC_ISA:
|
2009-07-25 01:54:45 +08:00
|
|
|
S = new (Context) ObjCIsaExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case EXPR_OBJC_INDIRECT_COPY_RESTORE:
|
|
|
|
S = new (Context) ObjCIndirectCopyRestoreExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case EXPR_OBJC_BRIDGED_CAST:
|
|
|
|
S = new (Context) ObjCBridgedCastExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_OBJC_FOR_COLLECTION:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCForCollectionStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_OBJC_CATCH:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCAtCatchStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_OBJC_FINALLY:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCAtFinallyStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_OBJC_AT_TRY:
|
2016-12-16 04:53:26 +08:00
|
|
|
S = ObjCAtTryStmt::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
Record[ASTStmtReader::NumStmtFields],
|
|
|
|
Record[ASTStmtReader::NumStmtFields + 1]);
|
2009-04-27 13:14:47 +08:00
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_OBJC_AT_SYNCHRONIZED:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCAtSynchronizedStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_OBJC_AT_THROW:
|
2009-04-27 13:14:47 +08:00
|
|
|
S = new (Context) ObjCAtThrowStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case STMT_OBJC_AUTORELEASE_POOL:
|
|
|
|
S = new (Context) ObjCAutoreleasePoolStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case EXPR_OBJC_BOOL_LITERAL:
|
|
|
|
S = new (Context) ObjCBoolLiteralExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2016-07-16 08:35:23 +08:00
|
|
|
case EXPR_OBJC_AVAILABILITY_CHECK:
|
|
|
|
S = new (Context) ObjCAvailabilityCheckExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2014-07-07 08:12:30 +08:00
|
|
|
case STMT_SEH_LEAVE:
|
|
|
|
S = new (Context) SEHLeaveStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
case STMT_SEH_EXCEPT:
|
|
|
|
S = new (Context) SEHExceptStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
case STMT_SEH_FINALLY:
|
|
|
|
S = new (Context) SEHFinallyStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
case STMT_SEH_TRY:
|
|
|
|
S = new (Context) SEHTryStmt(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_CXX_CATCH:
|
2010-07-23 00:03:56 +08:00
|
|
|
S = new (Context) CXXCatchStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STMT_CXX_TRY:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXTryStmt::Create(Context, Empty,
|
2019-07-16 12:46:31 +08:00
|
|
|
/*numHandlers=*/Record[ASTStmtReader::NumStmtFields]);
|
2010-07-23 00:03:56 +08:00
|
|
|
break;
|
|
|
|
|
2011-04-15 06:09:26 +08:00
|
|
|
case STMT_CXX_FOR_RANGE:
|
|
|
|
S = new (Context) CXXForRangeStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
2011-10-25 09:33:02 +08:00
|
|
|
case STMT_MS_DEPENDENT_EXISTS:
|
|
|
|
S = new (Context) MSDependentExistsStmt(SourceLocation(), true,
|
|
|
|
NestedNameSpecifierLoc(),
|
|
|
|
DeclarationNameInfo(),
|
2014-05-22 13:54:18 +08:00
|
|
|
nullptr);
|
2011-10-25 09:33:02 +08:00
|
|
|
break;
|
2014-02-27 16:29:12 +08:00
|
|
|
|
2013-07-19 11:13:43 +08:00
|
|
|
case STMT_OMP_PARALLEL_DIRECTIVE:
|
|
|
|
S =
|
|
|
|
OMPParallelDirective::CreateEmpty(Context,
|
|
|
|
Record[ASTStmtReader::NumStmtFields],
|
|
|
|
Empty);
|
|
|
|
break;
|
2014-02-27 16:29:12 +08:00
|
|
|
|
|
|
|
case STMT_OMP_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-06-18 12:14:57 +08:00
|
|
|
case STMT_OMP_FOR_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPForDirective::CreateEmpty(Context, NumClauses, CollapsedNum,
|
|
|
|
Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-09-18 13:12:34 +08:00
|
|
|
case STMT_OMP_FOR_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPForSimdDirective::CreateEmpty(Context, NumClauses, CollapsedNum,
|
|
|
|
Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-06-25 19:44:49 +08:00
|
|
|
case STMT_OMP_SECTIONS_DIRECTIVE:
|
|
|
|
S = OMPSectionsDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-06-26 16:21:58 +08:00
|
|
|
case STMT_OMP_SECTION_DIRECTIVE:
|
|
|
|
S = OMPSectionDirective::CreateEmpty(Context, Empty);
|
|
|
|
break;
|
|
|
|
|
2014-06-26 20:05:45 +08:00
|
|
|
case STMT_OMP_SINGLE_DIRECTIVE:
|
|
|
|
S = OMPSingleDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-17 16:54:58 +08:00
|
|
|
case STMT_OMP_MASTER_DIRECTIVE:
|
|
|
|
S = OMPMasterDirective::CreateEmpty(Context, Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-21 17:42:05 +08:00
|
|
|
case STMT_OMP_CRITICAL_DIRECTIVE:
|
2015-12-15 16:19:24 +08:00
|
|
|
S = OMPCriticalDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
2014-07-21 17:42:05 +08:00
|
|
|
break;
|
|
|
|
|
2014-07-07 21:01:15 +08:00
|
|
|
case STMT_OMP_PARALLEL_FOR_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPParallelForDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-09-23 17:33:00 +08:00
|
|
|
case STMT_OMP_PARALLEL_FOR_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPParallelForSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-12-05 03:36:07 +08:00
|
|
|
case STMT_OMP_PARALLEL_MASTER_DIRECTIVE:
|
|
|
|
S = OMPParallelMasterDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-08 16:12:03 +08:00
|
|
|
case STMT_OMP_PARALLEL_SECTIONS_DIRECTIVE:
|
|
|
|
S = OMPParallelSectionsDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-11 19:25:16 +08:00
|
|
|
case STMT_OMP_TASK_DIRECTIVE:
|
|
|
|
S = OMPTaskDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-18 15:47:19 +08:00
|
|
|
case STMT_OMP_TASKYIELD_DIRECTIVE:
|
|
|
|
S = OMPTaskyieldDirective::CreateEmpty(Context, Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-18 17:11:51 +08:00
|
|
|
case STMT_OMP_BARRIER_DIRECTIVE:
|
|
|
|
S = OMPBarrierDirective::CreateEmpty(Context, Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-18 18:17:07 +08:00
|
|
|
case STMT_OMP_TASKWAIT_DIRECTIVE:
|
|
|
|
S = OMPTaskwaitDirective::CreateEmpty(Context, Empty);
|
|
|
|
break;
|
|
|
|
|
2015-06-18 20:14:09 +08:00
|
|
|
case STMT_OMP_TASKGROUP_DIRECTIVE:
|
2017-07-19 04:17:46 +08:00
|
|
|
S = OMPTaskgroupDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
2015-06-18 20:14:09 +08:00
|
|
|
break;
|
|
|
|
|
2014-07-21 19:26:11 +08:00
|
|
|
case STMT_OMP_FLUSH_DIRECTIVE:
|
|
|
|
S = OMPFlushDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-07-22 14:45:04 +08:00
|
|
|
case STMT_OMP_ORDERED_DIRECTIVE:
|
2015-09-25 18:37:12 +08:00
|
|
|
S = OMPOrderedDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
2014-07-22 14:45:04 +08:00
|
|
|
break;
|
|
|
|
|
2014-07-22 18:10:35 +08:00
|
|
|
case STMT_OMP_ATOMIC_DIRECTIVE:
|
|
|
|
S = OMPAtomicDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-09-19 16:19:49 +08:00
|
|
|
case STMT_OMP_TARGET_DIRECTIVE:
|
|
|
|
S = OMPTargetDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2015-07-21 21:44:28 +08:00
|
|
|
case STMT_OMP_TARGET_DATA_DIRECTIVE:
|
|
|
|
S = OMPTargetDataDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2016-01-20 03:15:56 +08:00
|
|
|
case STMT_OMP_TARGET_ENTER_DATA_DIRECTIVE:
|
|
|
|
S = OMPTargetEnterDataDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2016-01-20 04:04:50 +08:00
|
|
|
case STMT_OMP_TARGET_EXIT_DATA_DIRECTIVE:
|
|
|
|
S = OMPTargetExitDataDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2016-01-27 02:48:41 +08:00
|
|
|
case STMT_OMP_TARGET_PARALLEL_DIRECTIVE:
|
|
|
|
S = OMPTargetParallelDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2016-02-03 23:46:42 +08:00
|
|
|
case STMT_OMP_TARGET_PARALLEL_FOR_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetParallelForDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-05-27 01:30:50 +08:00
|
|
|
case STMT_OMP_TARGET_UPDATE_DIRECTIVE:
|
|
|
|
S = OMPTargetUpdateDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2014-10-09 12:18:56 +08:00
|
|
|
case STMT_OMP_TEAMS_DIRECTIVE:
|
|
|
|
S = OMPTeamsDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
|
|
|
|
2015-07-01 14:57:41 +08:00
|
|
|
case STMT_OMP_CANCELLATION_POINT_DIRECTIVE:
|
|
|
|
S = OMPCancellationPointDirective::CreateEmpty(Context, Empty);
|
|
|
|
break;
|
|
|
|
|
2015-07-02 19:25:17 +08:00
|
|
|
case STMT_OMP_CANCEL_DIRECTIVE:
|
2015-09-18 16:07:34 +08:00
|
|
|
S = OMPCancelDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
2015-07-02 19:25:17 +08:00
|
|
|
break;
|
|
|
|
|
2015-12-01 12:18:41 +08:00
|
|
|
case STMT_OMP_TASKLOOP_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTaskLoopDirective::CreateEmpty(Context, NumClauses, CollapsedNum,
|
|
|
|
Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-12-03 17:40:15 +08:00
|
|
|
case STMT_OMP_TASKLOOP_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTaskLoopSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-10-11 04:13:02 +08:00
|
|
|
case STMT_OMP_MASTER_TASKLOOP_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPMasterTaskLoopDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-10-19 00:47:35 +08:00
|
|
|
case STMT_OMP_MASTER_TASKLOOP_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPMasterTaskLoopSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-10-15 01:17:41 +08:00
|
|
|
case STMT_OMP_PARALLEL_MASTER_TASKLOOP_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPParallelMasterTaskLoopDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-10-25 22:27:13 +08:00
|
|
|
case STMT_OMP_PARALLEL_MASTER_TASKLOOP_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPParallelMasterTaskLoopSimdDirective::CreateEmpty(
|
|
|
|
Context, NumClauses, CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-12-14 22:51:25 +08:00
|
|
|
case STMT_OMP_DISTRIBUTE_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPDistributeDirective::CreateEmpty(Context, NumClauses, CollapsedNum,
|
|
|
|
Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-06-27 22:55:37 +08:00
|
|
|
case STMT_OMP_DISTRIBUTE_PARALLEL_FOR_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPDistributeParallelForDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-07-05 13:00:15 +08:00
|
|
|
case STMT_OMP_DISTRIBUTE_PARALLEL_FOR_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPDistributeParallelForSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum,
|
|
|
|
Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-07-06 12:45:38 +08:00
|
|
|
case STMT_OMP_DISTRIBUTE_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPDistributeSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-07-14 10:54:56 +08:00
|
|
|
case STMT_OMP_TARGET_PARALLEL_FOR_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetParallelForSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-07-21 06:57:10 +08:00
|
|
|
case STMT_OMP_TARGET_SIMD_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetSimdDirective::CreateEmpty(Context, NumClauses, CollapsedNum,
|
|
|
|
Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-12-25 12:52:54 +08:00
|
|
|
case STMT_OMP_TEAMS_DISTRIBUTE_DIRECTIVE: {
|
2016-08-05 22:37:37 +08:00
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTeamsDistributeDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-10-25 20:50:55 +08:00
|
|
|
case STMT_OMP_TEAMS_DISTRIBUTE_SIMD_DIRECTIVE: {
|
|
|
|
unsigned NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
unsigned CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTeamsDistributeSimdDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-12-01 07:51:03 +08:00
|
|
|
case STMT_OMP_TEAMS_DISTRIBUTE_PARALLEL_FOR_SIMD_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTeamsDistributeParallelForSimdDirective::CreateEmpty(
|
|
|
|
Context, NumClauses, CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-12-09 11:24:30 +08:00
|
|
|
case STMT_OMP_TEAMS_DISTRIBUTE_PARALLEL_FOR_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTeamsDistributeParallelForDirective::CreateEmpty(
|
|
|
|
Context, NumClauses, CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-04-12 04:57:28 +08:00
|
|
|
case STMT_OMP_TARGET_TEAMS_DIRECTIVE:
|
2016-12-17 13:48:59 +08:00
|
|
|
S = OMPTargetTeamsDirective::CreateEmpty(
|
|
|
|
Context, Record[ASTStmtReader::NumStmtFields], Empty);
|
|
|
|
break;
|
2016-12-25 12:52:54 +08:00
|
|
|
|
|
|
|
case STMT_OMP_TARGET_TEAMS_DISTRIBUTE_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetTeamsDistributeDirective::CreateEmpty(Context, NumClauses,
|
|
|
|
CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-12-30 06:16:30 +08:00
|
|
|
case STMT_OMP_TARGET_TEAMS_DISTRIBUTE_PARALLEL_FOR_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetTeamsDistributeParallelForDirective::CreateEmpty(
|
|
|
|
Context, NumClauses, CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-01-03 13:23:48 +08:00
|
|
|
case STMT_OMP_TARGET_TEAMS_DISTRIBUTE_PARALLEL_FOR_SIMD_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetTeamsDistributeParallelForSimdDirective::CreateEmpty(
|
|
|
|
Context, NumClauses, CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-01-11 02:08:18 +08:00
|
|
|
case STMT_OMP_TARGET_TEAMS_DISTRIBUTE_SIMD_DIRECTIVE: {
|
|
|
|
auto NumClauses = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
auto CollapsedNum = Record[ASTStmtReader::NumStmtFields + 1];
|
|
|
|
S = OMPTargetTeamsDistributeSimdDirective::CreateEmpty(
|
|
|
|
Context, NumClauses, CollapsedNum, Empty);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_OPERATOR_CALL:
|
2018-12-21 23:20:32 +08:00
|
|
|
S = CXXOperatorCallExpr::CreateEmpty(
|
|
|
|
Context, /*NumArgs=*/Record[ASTStmtReader::NumExprFields], Empty);
|
2009-07-14 11:19:21 +08:00
|
|
|
break;
|
2010-05-09 13:36:05 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_MEMBER_CALL:
|
2018-12-21 23:20:32 +08:00
|
|
|
S = CXXMemberCallExpr::CreateEmpty(
|
|
|
|
Context, /*NumArgs=*/Record[ASTStmtReader::NumExprFields], Empty);
|
2010-05-09 13:36:05 +08:00
|
|
|
break;
|
2014-02-27 16:29:12 +08:00
|
|
|
|
2019-10-19 08:04:38 +08:00
|
|
|
case EXPR_CXX_REWRITTEN_BINARY_OPERATOR:
|
|
|
|
S = new (Context) CXXRewrittenBinaryOperator(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_CONSTRUCT:
|
2018-12-22 22:39:30 +08:00
|
|
|
S = CXXConstructExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/* NumArgs=*/Record[ASTStmtReader::NumExprFields]);
|
2010-07-10 19:46:15 +08:00
|
|
|
break;
|
2014-02-27 16:29:12 +08:00
|
|
|
|
P0136R1, DR1573, DR1645, DR1715, DR1736, DR1903, DR1941, DR1959, DR1991:
Replace inheriting constructors implementation with new approach, voted into
C++ last year as a DR against C++11.
Instead of synthesizing a set of derived class constructors for each inherited
base class constructor, we make the constructors of the base class visible to
constructor lookup in the derived class, using the normal rules for
using-declarations.
For constructors, UsingShadowDecl now has a ConstructorUsingShadowDecl derived
class that tracks the requisite additional information. We create shadow
constructors (not found by name lookup) in the derived class to model the
actual initialization, and have a new expression node,
CXXInheritedCtorInitExpr, to model the initialization of a base class from such
a constructor. (This initialization is special because it performs real perfect
forwarding of arguments.)
In cases where argument forwarding is not possible (for inalloca calls,
variadic calls, and calls with callee parameter cleanup), the shadow inheriting
constructor is not emitted and instead we directly emit the initialization code
into the caller of the inherited constructor.
Note that this new model is not perfectly compatible with the old model in some
corner cases. In particular:
* if B inherits a private constructor from A, and C uses that constructor to
construct a B, then we previously required that A befriends B and B
befriends C, but the new rules require A to befriend C directly, and
* if a derived class has its own constructors (and so its implicit default
constructor is suppressed), it may still inherit a default constructor from
a base class
llvm-svn: 274049
2016-06-29 03:03:57 +08:00
|
|
|
case EXPR_CXX_INHERITED_CTOR_INIT:
|
|
|
|
S = new (Context) CXXInheritedCtorInitExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_TEMPORARY_OBJECT:
|
2018-12-22 22:39:30 +08:00
|
|
|
S = CXXTemporaryObjectExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/* NumArgs=*/Record[ASTStmtReader::NumExprFields]);
|
2009-09-10 07:08:42 +08:00
|
|
|
break;
|
2010-01-17 05:21:01 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_STATIC_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXStaticCastExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*PathSize*/ Record[ASTStmtReader::NumExprFields]);
|
2010-01-17 05:21:01 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_DYNAMIC_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXDynamicCastExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*PathSize*/ Record[ASTStmtReader::NumExprFields]);
|
2010-01-17 05:21:01 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_REINTERPRET_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXReinterpretCastExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*PathSize*/ Record[ASTStmtReader::NumExprFields]);
|
2010-01-17 05:21:01 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_CONST_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXConstCastExpr::CreateEmpty(Context);
|
2010-01-17 05:21:01 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_FUNCTIONAL_CAST:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXFunctionalCastExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*PathSize*/ Record[ASTStmtReader::NumExprFields]);
|
2010-01-17 05:21:01 +08:00
|
|
|
break;
|
|
|
|
|
2012-03-07 16:35:16 +08:00
|
|
|
case EXPR_USER_DEFINED_LITERAL:
|
2018-12-21 23:20:32 +08:00
|
|
|
S = UserDefinedLiteral::CreateEmpty(
|
2018-12-04 00:17:45 +08:00
|
|
|
Context, /*NumArgs=*/Record[ASTStmtReader::NumExprFields], Empty);
|
2012-03-07 16:35:16 +08:00
|
|
|
break;
|
|
|
|
|
2013-06-13 06:31:48 +08:00
|
|
|
case EXPR_CXX_STD_INITIALIZER_LIST:
|
|
|
|
S = new (Context) CXXStdInitializerListExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_BOOL_LITERAL:
|
2010-02-07 14:32:43 +08:00
|
|
|
S = new (Context) CXXBoolLiteralExpr(Empty);
|
|
|
|
break;
|
2010-01-17 05:21:01 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_NULL_PTR_LITERAL:
|
2010-02-07 14:32:43 +08:00
|
|
|
S = new (Context) CXXNullPtrLiteralExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_TYPEID_EXPR:
|
2010-05-09 14:03:39 +08:00
|
|
|
S = new (Context) CXXTypeidExpr(Empty, true);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_TYPEID_TYPE:
|
2010-05-09 14:03:39 +08:00
|
|
|
S = new (Context) CXXTypeidExpr(Empty, false);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-09-08 20:20:18 +08:00
|
|
|
case EXPR_CXX_UUIDOF_EXPR:
|
|
|
|
S = new (Context) CXXUuidofExpr(Empty, true);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2013-04-16 15:28:30 +08:00
|
|
|
case EXPR_CXX_PROPERTY_REF_EXPR:
|
|
|
|
S = new (Context) MSPropertyRefExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2015-11-25 20:01:00 +08:00
|
|
|
case EXPR_CXX_PROPERTY_SUBSCRIPT_EXPR:
|
|
|
|
S = new (Context) MSPropertySubscriptExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-09-08 20:20:18 +08:00
|
|
|
case EXPR_CXX_UUIDOF_TYPE:
|
|
|
|
S = new (Context) CXXUuidofExpr(Empty, false);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_THIS:
|
2010-05-09 14:15:05 +08:00
|
|
|
S = new (Context) CXXThisExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_THROW:
|
2010-05-09 14:15:05 +08:00
|
|
|
S = new (Context) CXXThrowExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2016-01-07 06:34:54 +08:00
|
|
|
case EXPR_CXX_DEFAULT_ARG:
|
|
|
|
S = new (Context) CXXDefaultArgExpr(Empty);
|
2010-05-09 14:40:08 +08:00
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2013-04-21 06:23:05 +08:00
|
|
|
case EXPR_CXX_DEFAULT_INIT:
|
|
|
|
S = new (Context) CXXDefaultInitExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_BIND_TEMPORARY:
|
2010-05-10 08:25:06 +08:00
|
|
|
S = new (Context) CXXBindTemporaryExpr(Empty);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_SCALAR_VALUE_INIT:
|
2010-07-08 14:14:04 +08:00
|
|
|
S = new (Context) CXXScalarValueInitExpr(Empty);
|
2010-05-10 09:22:27 +08:00
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_NEW:
|
2019-01-07 23:04:45 +08:00
|
|
|
S = CXXNewExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*IsArray=*/Record[ASTStmtReader::NumExprFields],
|
|
|
|
/*HasInit=*/Record[ASTStmtReader::NumExprFields + 1],
|
|
|
|
/*NumPlacementArgs=*/Record[ASTStmtReader::NumExprFields + 2],
|
|
|
|
/*IsParenTypeId=*/Record[ASTStmtReader::NumExprFields + 3]);
|
2010-05-10 09:22:27 +08:00
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_DELETE:
|
2010-06-23 01:07:59 +08:00
|
|
|
S = new (Context) CXXDeleteExpr(Empty);
|
|
|
|
break;
|
2018-04-12 04:57:28 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_PSEUDO_DESTRUCTOR:
|
2010-06-28 17:32:03 +08:00
|
|
|
S = new (Context) CXXPseudoDestructorExpr(Empty);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-12-06 16:20:24 +08:00
|
|
|
case EXPR_EXPR_WITH_CLEANUPS:
|
2011-11-10 13:35:25 +08:00
|
|
|
S = ExprWithCleanups::Create(Context, Empty,
|
|
|
|
Record[ASTStmtReader::NumExprFields]);
|
2010-05-10 08:25:06 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_DEPENDENT_SCOPE_MEMBER:
|
2019-01-08 22:17:00 +08:00
|
|
|
S = CXXDependentScopeMemberExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*HasTemplateKWAndArgsInfo=*/Record[ASTStmtReader::NumExprFields],
|
|
|
|
/*NumTemplateArgs=*/Record[ASTStmtReader::NumExprFields + 1],
|
|
|
|
/*HasFirstQualifierFoundInScope=*/
|
|
|
|
Record[ASTStmtReader::NumExprFields + 2]);
|
2010-06-24 16:57:31 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_DEPENDENT_SCOPE_DECL_REF:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = DependentScopeDeclRefExpr::CreateEmpty(Context,
|
2012-01-27 17:46:47 +08:00
|
|
|
/*HasTemplateKWAndArgsInfo=*/Record[ASTStmtReader::NumExprFields],
|
2011-02-04 20:01:24 +08:00
|
|
|
/*NumTemplateArgs=*/Record[ASTStmtReader::NumExprFields]
|
2016-12-16 04:53:26 +08:00
|
|
|
? Record[ASTStmtReader::NumExprFields + 1]
|
2011-02-04 20:01:24 +08:00
|
|
|
: 0);
|
2010-06-28 17:31:56 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_UNRESOLVED_CONSTRUCT:
|
2011-09-10 05:34:22 +08:00
|
|
|
S = CXXUnresolvedConstructExpr::CreateEmpty(Context,
|
2010-08-19 07:56:52 +08:00
|
|
|
/*NumArgs=*/Record[ASTStmtReader::NumExprFields]);
|
2010-06-25 17:03:26 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_UNRESOLVED_MEMBER:
|
2019-01-09 23:43:19 +08:00
|
|
|
S = UnresolvedMemberExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*NumResults=*/Record[ASTStmtReader::NumExprFields],
|
|
|
|
/*HasTemplateKWAndArgsInfo=*/Record[ASTStmtReader::NumExprFields + 1],
|
|
|
|
/*NumTemplateArgs=*/
|
|
|
|
Record[ASTStmtReader::NumExprFields + 1]
|
|
|
|
? Record[ASTStmtReader::NumExprFields + 2]
|
|
|
|
: 0);
|
2010-06-24 16:57:31 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXPR_CXX_UNRESOLVED_LOOKUP:
|
2019-01-09 23:43:19 +08:00
|
|
|
S = UnresolvedLookupExpr::CreateEmpty(
|
|
|
|
Context,
|
|
|
|
/*NumResults=*/Record[ASTStmtReader::NumExprFields],
|
|
|
|
/*HasTemplateKWAndArgsInfo=*/Record[ASTStmtReader::NumExprFields + 1],
|
|
|
|
/*NumTemplateArgs=*/
|
|
|
|
Record[ASTStmtReader::NumExprFields + 1]
|
|
|
|
? Record[ASTStmtReader::NumExprFields + 2]
|
|
|
|
: 0);
|
2010-06-25 17:03:34 +08:00
|
|
|
break;
|
2010-09-11 04:55:54 +08:00
|
|
|
|
2012-02-24 15:38:34 +08:00
|
|
|
case EXPR_TYPE_TRAIT:
|
2016-12-16 04:53:26 +08:00
|
|
|
S = TypeTraitExpr::CreateDeserialized(Context,
|
2012-02-24 15:38:34 +08:00
|
|
|
Record[ASTStmtReader::NumExprFields]);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-04-28 08:16:57 +08:00
|
|
|
case EXPR_ARRAY_TYPE_TRAIT:
|
|
|
|
S = new (Context) ArrayTypeTraitExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2011-04-25 14:54:41 +08:00
|
|
|
case EXPR_CXX_EXPRESSION_TRAIT:
|
|
|
|
S = new (Context) ExpressionTraitExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2010-09-11 04:55:54 +08:00
|
|
|
case EXPR_CXX_NOEXCEPT:
|
|
|
|
S = new (Context) CXXNoexceptExpr(Empty);
|
|
|
|
break;
|
2010-11-16 07:31:06 +08:00
|
|
|
|
2011-01-04 01:17:50 +08:00
|
|
|
case EXPR_PACK_EXPANSION:
|
|
|
|
S = new (Context) PackExpansionExpr(Empty);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-01-05 01:33:58 +08:00
|
|
|
case EXPR_SIZEOF_PACK:
|
2015-09-24 05:41:42 +08:00
|
|
|
S = SizeOfPackExpr::CreateDeserialized(
|
|
|
|
Context,
|
|
|
|
/*NumPartialArgs=*/Record[ASTStmtReader::NumExprFields]);
|
2011-01-05 01:33:58 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-07-15 15:00:14 +08:00
|
|
|
case EXPR_SUBST_NON_TYPE_TEMPLATE_PARM:
|
|
|
|
S = new (Context) SubstNonTypeTemplateParmExpr(Empty);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-01-15 09:15:58 +08:00
|
|
|
case EXPR_SUBST_NON_TYPE_TEMPLATE_PARM_PACK:
|
|
|
|
S = new (Context) SubstNonTypeTemplateParmPackExpr(Empty);
|
|
|
|
break;
|
2012-09-12 08:56:43 +08:00
|
|
|
|
|
|
|
case EXPR_FUNCTION_PARM_PACK:
|
|
|
|
S = FunctionParmPackExpr::CreateEmpty(Context,
|
|
|
|
Record[ASTStmtReader::NumExprFields]);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-06-22 01:03:29 +08:00
|
|
|
case EXPR_MATERIALIZE_TEMPORARY:
|
|
|
|
S = new (Context) MaterializeTemporaryExpr(Empty);
|
|
|
|
break;
|
2014-11-08 13:07:16 +08:00
|
|
|
|
|
|
|
case EXPR_CXX_FOLD:
|
|
|
|
S = new (Context) CXXFoldExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
2011-12-03 11:49:52 +08:00
|
|
|
case EXPR_OPAQUE_VALUE:
|
|
|
|
S = new (Context) OpaqueValueExpr(Empty);
|
2010-11-16 07:31:06 +08:00
|
|
|
break;
|
2011-02-10 05:07:24 +08:00
|
|
|
|
|
|
|
case EXPR_CUDA_KERNEL_CALL:
|
2018-12-21 23:20:32 +08:00
|
|
|
S = CUDAKernelCallExpr::CreateEmpty(
|
2018-12-04 00:17:45 +08:00
|
|
|
Context, /*NumArgs=*/Record[ASTStmtReader::NumExprFields], Empty);
|
2011-02-10 05:07:24 +08:00
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2011-06-04 08:47:47 +08:00
|
|
|
case EXPR_ASTYPE:
|
|
|
|
S = new (Context) AsTypeExpr(Empty);
|
|
|
|
break;
|
2011-10-11 10:20:01 +08:00
|
|
|
|
2011-11-06 17:01:30 +08:00
|
|
|
case EXPR_PSEUDO_OBJECT: {
|
|
|
|
unsigned numSemanticExprs = Record[ASTStmtReader::NumExprFields];
|
|
|
|
S = PseudoObjectExpr::Create(Context, Empty, numSemanticExprs);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-10-11 10:20:01 +08:00
|
|
|
case EXPR_ATOMIC:
|
|
|
|
S = new (Context) AtomicExpr(Empty);
|
|
|
|
break;
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2012-02-15 01:54:36 +08:00
|
|
|
case EXPR_LAMBDA: {
|
|
|
|
unsigned NumCaptures = Record[ASTStmtReader::NumExprFields];
|
2016-12-14 08:03:17 +08:00
|
|
|
S = LambdaExpr::CreateDeserialized(Context, NumCaptures);
|
2012-02-15 01:54:36 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-07-26 02:01:49 +08:00
|
|
|
|
|
|
|
case STMT_COROUTINE_BODY: {
|
|
|
|
unsigned NumParams = Record[ASTStmtReader::NumStmtFields];
|
|
|
|
S = CoroutineBodyStmt::Create(Context, Empty, NumParams);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case STMT_CORETURN:
|
|
|
|
S = new (Context) CoreturnStmt(Empty);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EXPR_COAWAIT:
|
|
|
|
S = new (Context) CoawaitExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EXPR_COYIELD:
|
|
|
|
S = new (Context) CoyieldExpr(Empty);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EXPR_DEPENDENT_COAWAIT:
|
|
|
|
S = new (Context) DependentCoawaitExpr(Empty);
|
|
|
|
break;
|
2019-10-15 23:24:26 +08:00
|
|
|
|
|
|
|
case EXPR_CONCEPT_SPECIALIZATION:
|
|
|
|
unsigned numTemplateArgs = Record[ASTStmtReader::NumExprFields];
|
|
|
|
S = ConceptSpecializationExpr::Create(Context, Empty, numTemplateArgs);
|
|
|
|
break;
|
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|
2016-12-16 04:53:26 +08:00
|
|
|
|
2009-04-27 13:14:47 +08:00
|
|
|
// We hit a STMT_STOP, so we're done with this expression.
|
|
|
|
if (Finished)
|
|
|
|
break;
|
|
|
|
|
|
|
|
++NumStatementsRead;
|
|
|
|
|
2011-10-22 07:02:28 +08:00
|
|
|
if (S && !IsStmtReference) {
|
2010-06-29 06:28:35 +08:00
|
|
|
Reader.Visit(S);
|
2011-10-22 07:02:28 +08:00
|
|
|
StmtEntries[Cursor.GetCurrentBitNo()] = S;
|
|
|
|
}
|
|
|
|
|
2016-12-21 08:17:49 +08:00
|
|
|
assert(Record.getIdx() == Record.size() &&
|
|
|
|
"Invalid deserialization of statement");
|
2009-04-27 13:14:47 +08:00
|
|
|
StmtStack.push_back(S);
|
|
|
|
}
|
2013-01-20 10:38:54 +08:00
|
|
|
Done:
|
2013-12-07 01:56:43 +08:00
|
|
|
assert(StmtStack.size() > PrevNumStmts && "Read too many sub-stmts!");
|
2010-06-29 06:28:35 +08:00
|
|
|
assert(StmtStack.size() == PrevNumStmts + 1 && "Extra expressions on stack!");
|
|
|
|
return StmtStack.pop_back_val();
|
2009-04-27 13:14:47 +08:00
|
|
|
}
|