clang 23.0.0git
State.h
Go to the documentation of this file.
1//===--- State.h - State chain for the VM and AST Walker --------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Defines the base class of the interpreter and evaluator state.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLVM_CLANG_AST_INTERP_STATE_H
14#define LLVM_CLANG_AST_INTERP_STATE_H
15
18#include "clang/AST/Expr.h"
20
21namespace clang {
23
24/// Kinds of access we can perform on an object, for diagnostics. Note that
25/// we consider a member function call to be a kind of access, even though
26/// it is not formally an access of the object, because it has (largely) the
27/// same set of semantic restrictions.
42
43/// The order of this enum is important for diagnostics.
54
55enum class EvaluationMode {
56 /// Evaluate as a constant expression. Stop if we find that the expression
57 /// is not a constant expression.
59
60 /// Evaluate as a constant expression. Stop if we find that the expression
61 /// is not a constant expression. Some expressions can be retried in the
62 /// optimizer if we don't constant fold them here, but in an unevaluated
63 /// context we try to fold them immediately since the optimizer never
64 /// gets a chance to look at it.
66
67 /// Fold the expression to a constant. Stop if we hit a side-effect that
68 /// we can't model.
70
71 /// Evaluate in any way we know how. Don't worry about side-effects that
72 /// can't be modeled.
74};
75
76namespace interp {
77class Frame;
78class SourceInfo;
79
80/// Interface for the VM to interact with the AST walker's context.
81class State {
82public:
85 virtual ~State();
86
87 virtual const Frame *getCurrentFrame() = 0;
88 virtual unsigned getCallStackDepth() = 0;
89 virtual bool stepsLeft() const = 0;
90
92 ASTContext &getASTContext() const { return Ctx; }
93 const LangOptions &getLangOpts() const { return Ctx.getLangOpts(); }
94
95 /// Note that we have had a side-effect, and determine whether we should
96 /// keep evaluating.
97 bool noteSideEffect() const {
98 EvalStatus.HasSideEffects = true;
100 }
101
102 /// Should we continue evaluation as much as possible after encountering a
103 /// construct which can't be reduced to a value?
104 bool keepEvaluatingAfterFailure() const;
105 /// Should we continue evaluation after encountering a side-effect that we
106 /// couldn't model?
108
109 /// Note that we hit something that was technically undefined behavior, but
110 /// that we can evaluate past it (such as signed overflow or floating-point
111 /// division by zero.)
113 EvalStatus.HasUndefinedBehavior = true;
114 return keepEvaluatingAfterUndefinedBehavior();
115 }
116
117 /// Are we checking whether the expression is a potential constant
118 /// expression?
122 /// Are we checking an expression for overflow?
126
127 /// Diagnose that the evaluation could not be folded (FF => FoldFailure)
130 diag::kind DiagId = diag::note_invalid_subexpr_in_const_expr,
131 unsigned ExtraNotes = 0);
132
134 FFDiag(const Expr *E,
135 diag::kind DiagId = diag::note_invalid_subexpr_in_const_expr,
136 unsigned ExtraNotes = 0);
137
140 diag::kind DiagId = diag::note_invalid_subexpr_in_const_expr,
141 unsigned ExtraNotes = 0);
142
143 /// Diagnose that the evaluation does not produce a C++11 core constant
144 /// expression.
145 ///
146 /// FIXME: Stop evaluating if we're in EM_ConstantExpression or
147 /// EM_PotentialConstantExpression mode and we produce one of these.
150 diag::kind DiagId = diag::note_invalid_subexpr_in_const_expr,
151 unsigned ExtraNotes = 0);
152
154 CCEDiag(const Expr *E,
155 diag::kind DiagId = diag::note_invalid_subexpr_in_const_expr,
156 unsigned ExtraNotes = 0);
157
160 diag::kind DiagId = diag::note_invalid_subexpr_in_const_expr,
161 unsigned ExtraNotes = 0);
162
163 /// Add a note to a prior diagnostic.
165
166 /// Add a stack of notes to a prior diagnostic.
168
169 /// Directly reports a diagnostic message.
171
172 /// Whether or not we're in a context where the front end requires a
173 /// constant value.
174 bool InConstantContext = false;
175
176 /// Whether we're checking that an expression is a potential constant
177 /// expression. If so, do not fail on constructs that could become constant
178 /// later on (such as a use of an undefined global).
180
181 /// Whether we're checking for an expression that has undefined behavior.
182 /// If so, we will produce warnings if we encounter an operation that is
183 /// always undefined.
184 ///
185 /// Note that we still need to evaluate the expression normally when this
186 /// is set; this is used when evaluating ICEs in C.
188
192
193private:
194 /// HasActiveDiagnostic - Was the previous diagnostic stored? If so, further
195 /// notes attached to it will also be stored, otherwise they will not be.
196 bool HasActiveDiagnostic = false;
197
198 /// Have we emitted a diagnostic explaining why we couldn't constant
199 /// fold (not just why it's not strictly a constant expression)?
200 bool HasFoldFailureDiagnostic = false;
201
202 void addCallStack(unsigned Limit);
203
204 PartialDiagnostic &addDiag(SourceLocation Loc, diag::kind DiagId);
205
207 unsigned ExtraNotes, bool IsCCEDiag);
208
209 /// Should we continue evaluation after encountering undefined behavior?
210 bool keepEvaluatingAfterUndefinedBehavior() const;
211
212 // If we have a prior diagnostic, it will be noting that the expression
213 // isn't a constant expression. This diagnostic is more important,
214 // unless we require this evaluation to produce a constant expression.
215 //
216 // FIXME: We might want to show both diagnostics to the user in
217 // EvaluationMode::ConstantFold mode.
218 bool hasPriorDiagnostic();
219
220 void setFoldFailureDiagnostic(bool Flag) { HasFoldFailureDiagnostic = Flag; };
221 void setActiveDiagnostic(bool Flag) { HasActiveDiagnostic = Flag; };
222 bool hasActiveDiagnostic() const { return HasActiveDiagnostic; }
223};
224
225} // namespace interp
226} // namespace clang
227
228#endif
Defines the clang::ASTContext interface.
Implements a partial diagnostic which may not be emitted.
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
Definition ASTContext.h:223
A little helper class used to produce diagnostics.
This represents one expression.
Definition Expr.h:112
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
A partial diagnostic which we might know in advance that we are not going to emit.
Encodes a location in the source.
Base class for stack frames, shared between VM and walker.
Definition Frame.h:25
Describes the statement/declaration an opcode was generated from.
Definition Source.h:74
bool checkingForUndefinedBehavior() const
Are we checking an expression for overflow?
Definition State.h:123
virtual const Frame * getCurrentFrame()=0
EvaluationMode EvalMode
Definition State.h:189
virtual bool stepsLeft() const =0
Expr::EvalStatus & getEvalStatus() const
Definition State.h:91
DiagnosticBuilder report(SourceLocation Loc, diag::kind DiagId)
Directly reports a diagnostic message.
Definition State.cpp:74
OptionalDiagnostic FFDiag(SourceLocation Loc, diag::kind DiagId=diag::note_invalid_subexpr_in_const_expr, unsigned ExtraNotes=0)
Diagnose that the evaluation could not be folded (FF => FoldFailure)
Definition State.cpp:21
State(ASTContext &ASTCtx, Expr::EvalStatus &EvalStatus)
Definition State.h:83
bool CheckingPotentialConstantExpression
Whether we're checking that an expression is a potential constant expression.
Definition State.h:179
virtual unsigned getCallStackDepth()=0
bool CheckingForUndefinedBehavior
Whether we're checking for an expression that has undefined behavior.
Definition State.h:187
bool noteSideEffect() const
Note that we have had a side-effect, and determine whether we should keep evaluating.
Definition State.h:97
Expr::EvalStatus & EvalStatus
Definition State.h:191
ASTContext & getASTContext() const
Definition State.h:92
bool noteUndefinedBehavior() const
Note that we hit something that was technically undefined behavior, but that we can evaluate past it ...
Definition State.h:112
void addNotes(ArrayRef< PartialDiagnosticAt > Diags)
Add a stack of notes to a prior diagnostic.
Definition State.cpp:69
OptionalDiagnostic CCEDiag(SourceLocation Loc, diag::kind DiagId=diag::note_invalid_subexpr_in_const_expr, unsigned ExtraNotes=0)
Diagnose that the evaluation does not produce a C++11 core constant expression.
Definition State.cpp:42
const LangOptions & getLangOpts() const
Definition State.h:93
bool checkingPotentialConstantExpression() const
Are we checking whether the expression is a potential constant expression?
Definition State.h:119
bool keepEvaluatingAfterSideEffect() const
Should we continue evaluation after encountering a side-effect that we couldn't model?
Definition State.cpp:194
bool keepEvaluatingAfterFailure() const
Should we continue evaluation as much as possible after encountering a construct which can't be reduc...
Definition State.cpp:178
bool InConstantContext
Whether or not we're in a context where the front end requires a constant value.
Definition State.h:174
ASTContext & Ctx
Definition State.h:190
virtual ~State()
Definition State.cpp:19
unsigned kind
All of the diagnostics that can be emitted by the frontend.
The JSON file list parser is used to communicate input to InstallAPI.
CheckSubobjectKind
The order of this enum is important for diagnostics.
Definition State.h:44
@ CSK_ArrayToPointer
Definition State.h:48
@ CSK_Derived
Definition State.h:46
@ CSK_Base
Definition State.h:45
@ CSK_Real
Definition State.h:50
@ CSK_ArrayIndex
Definition State.h:49
@ CSK_Imag
Definition State.h:51
@ CSK_VectorElement
Definition State.h:52
@ CSK_Field
Definition State.h:47
AccessKinds
Kinds of access we can perform on an object, for diagnostics.
Definition State.h:28
@ AK_TypeId
Definition State.h:36
@ AK_Construct
Definition State.h:37
@ AK_Increment
Definition State.h:32
@ AK_DynamicCast
Definition State.h:35
@ AK_Read
Definition State.h:29
@ AK_Assign
Definition State.h:31
@ AK_IsWithinLifetime
Definition State.h:39
@ AK_MemberCall
Definition State.h:34
@ AK_ReadObjectRepresentation
Definition State.h:30
@ AK_Dereference
Definition State.h:40
@ AK_Destroy
Definition State.h:38
@ AK_Decrement
Definition State.h:33
EvaluationMode
Definition State.h:55
@ ConstantFold
Fold the expression to a constant.
Definition State.h:69
@ ConstantExpressionUnevaluated
Evaluate as a constant expression.
Definition State.h:65
@ ConstantExpression
Evaluate as a constant expression.
Definition State.h:58
@ IgnoreSideEffects
Evaluate in any way we know how.
Definition State.h:73
EvalStatus is a struct with detailed info about an evaluation in progress.
Definition Expr.h:612