Skip to content

Commit 641ede9

Browse files
committed
[flang] Improve initializer semantics, esp. for component default values
This patch plugs many holes in static initializer semantics, improves error messages for default initial values and other component properties in parameterized derived type instantiations, and cleans up several small issues noticed during development. We now do proper scalar expansion, folding, and type, rank, and shape conformance checking for component default initializers in derived types and PDT instantiations. The initial values of named constants are now guaranteed to have been folded when installed in the symbol table, and are no longer folded or scalar-expanded at each use in expression folding. Semantics documentation was extended with information about the various kinds of initializations in Fortran and when each of them are processed in the compiler. Some necessary concomitant changes have bulked this patch out a bit: * contextual messages attachments, which are now produced for parameterized derived type instantiations so that the user can figure out which instance caused a problem with a component, have been added as part of ContextualMessages, and their implementation was debugged * several APIs in evaluate::characteristics was changed so that a FoldingContext is passed as an argument rather than just its intrinsic procedure table; this affected client call sites in many files * new tools in Evaluate/check-expression.cpp to determine when an Expr actually is a single constant value and to validate a non-pointer variable initializer or object component default value * shape conformance checking has additional arguments that control whether scalar expansion is allowed * several now-unused functions and data members noticed and removed * several crashes and bogus errors exposed by testing this new code were fixed * a -fdebug-stack-trace option to enable LLVM's stack tracing on a crash, which might be useful in the future TL;DR: Initialization processing does more and takes place at the right times for all of the various kinds of things that can be initialized. Differential Review: https://reviews.llvm.org/D92783
1 parent 155fca3 commit 641ede9

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

42 files changed

+701
-579
lines changed

flang/docs/Semantics.md

+43
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,49 @@ By this phase, all names and expressions that can be successfully resolved
147147
have been. But there may be names without symbols or expressions without
148148
analyzed form if errors occurred earlier.
149149

150+
### Initialization processing
151+
152+
Fortran supports many means of specifying static initializers for variables,
153+
object pointers, and procedure pointers, as well as default initializers for
154+
derived type object components, pointers, and type parameters.
155+
156+
Non-pointer static initializers of variables and named constants are
157+
scanned, analyzed, folded, scalar-expanded, and validated as they are
158+
traversed during declaration processing in name resolution.
159+
So are the default initializers of non-pointer object components in
160+
non-parameterized derived types.
161+
Name constant arrays with implied shapes take their actual shape from
162+
the initialization expression.
163+
164+
Default initializers of non-pointer components and type parameters
165+
in distinct parameterized
166+
derived type instantiations are similarly processed as those instances
167+
are created, as their expressions may depend on the values of type
168+
parameters.
169+
Error messages produced during parameterized derived type instantiation
170+
are decorated with contextual attachments that point to the declarations
171+
or other type specifications that caused the instantiation.
172+
173+
Static initializations in `DATA` statements are collected, validated,
174+
and converted into static initialization in the symbol table, as if
175+
the initialized objects had used the newer style of static initialization
176+
in their entity declarations.
177+
178+
All statically initialized pointers, and default component initializers for
179+
pointers, are processed late in name resolution after all specification parts
180+
have been traversed.
181+
This allows for forward references even in the presence of `IMPLICIT NONE`.
182+
Object pointer initializers in parameterized derived type instantiations are
183+
also cloned and folded at this late stage.
184+
Validation of pointer initializers takes place later in declaration
185+
checking (below).
186+
187+
### Declaration checking
188+
189+
Whenever possible, the enforcement of constraints and "shalls" pertaining to
190+
properties of symbols is deferred to a single read-only pass over the symbol table
191+
that takes place after all name resolution and typing is complete.
192+
150193
### Write module files
151194

152195
Separate compilation information is written out on successful compilation

flang/include/flang/Common/reference-counted.h

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ namespace Fortran::common {
1919
template <typename A> class ReferenceCounted {
2020
public:
2121
ReferenceCounted() {}
22+
int references() const { return references_; }
2223
void TakeReference() { ++references_; }
2324
void DropReference() {
2425
if (--references_ == 0) {

flang/include/flang/Evaluate/characteristics.h

+10-12
Original file line numberDiff line numberDiff line change
@@ -32,9 +32,6 @@ namespace llvm {
3232
class raw_ostream;
3333
}
3434

35-
namespace Fortran::evaluate {
36-
class IntrinsicProcTable;
37-
}
3835
namespace Fortran::evaluate::characteristics {
3936
struct Procedure;
4037
}
@@ -82,7 +79,7 @@ class TypeAndShape {
8279
static std::optional<TypeAndShape> Characterize(
8380
const semantics::Symbol &, FoldingContext &);
8481
static std::optional<TypeAndShape> Characterize(
85-
const semantics::ObjectEntityDetails &);
82+
const semantics::ObjectEntityDetails &, FoldingContext &);
8683
static std::optional<TypeAndShape> Characterize(
8784
const semantics::ProcInterface &);
8885
static std::optional<TypeAndShape> Characterize(
@@ -160,7 +157,7 @@ class TypeAndShape {
160157
const semantics::AssocEntityDetails &, FoldingContext &);
161158
static std::optional<TypeAndShape> Characterize(
162159
const semantics::ProcEntityDetails &);
163-
void AcquireShape(const semantics::ObjectEntityDetails &);
160+
void AcquireShape(const semantics::ObjectEntityDetails &, FoldingContext &);
164161
void AcquireLEN();
165162

166163
protected:
@@ -184,7 +181,8 @@ struct DummyDataObject {
184181
bool operator!=(const DummyDataObject &that) const {
185182
return !(*this == that);
186183
}
187-
static std::optional<DummyDataObject> Characterize(const semantics::Symbol &);
184+
static std::optional<DummyDataObject> Characterize(
185+
const semantics::Symbol &, FoldingContext &);
188186
bool CanBePassedViaImplicitInterface() const;
189187
llvm::raw_ostream &Dump(llvm::raw_ostream &) const;
190188
TypeAndShape type;
@@ -202,7 +200,7 @@ struct DummyProcedure {
202200
bool operator==(const DummyProcedure &) const;
203201
bool operator!=(const DummyProcedure &that) const { return !(*this == that); }
204202
static std::optional<DummyProcedure> Characterize(
205-
const semantics::Symbol &, const IntrinsicProcTable &);
203+
const semantics::Symbol &, FoldingContext &context);
206204
llvm::raw_ostream &Dump(llvm::raw_ostream &) const;
207205
CopyableIndirection<Procedure> procedure;
208206
common::Intent intent{common::Intent::Default};
@@ -228,7 +226,7 @@ struct DummyArgument {
228226
bool operator==(const DummyArgument &) const;
229227
bool operator!=(const DummyArgument &that) const { return !(*this == that); }
230228
static std::optional<DummyArgument> Characterize(
231-
const semantics::Symbol &, const IntrinsicProcTable &);
229+
const semantics::Symbol &, FoldingContext &);
232230
static std::optional<DummyArgument> FromActual(
233231
std::string &&, const Expr<SomeType> &, FoldingContext &);
234232
bool IsOptional() const;
@@ -259,7 +257,7 @@ struct FunctionResult {
259257
bool operator==(const FunctionResult &) const;
260258
bool operator!=(const FunctionResult &that) const { return !(*this == that); }
261259
static std::optional<FunctionResult> Characterize(
262-
const Symbol &, const IntrinsicProcTable &);
260+
const Symbol &, FoldingContext &);
263261

264262
bool IsAssumedLengthCharacter() const;
265263

@@ -297,11 +295,11 @@ struct Procedure {
297295
// Characterizes the procedure represented by a symbol, which may be an
298296
// "unrestricted specific intrinsic function".
299297
static std::optional<Procedure> Characterize(
300-
const semantics::Symbol &, const IntrinsicProcTable &);
298+
const semantics::Symbol &, FoldingContext &);
301299
static std::optional<Procedure> Characterize(
302-
const ProcedureDesignator &, const IntrinsicProcTable &);
300+
const ProcedureDesignator &, FoldingContext &);
303301
static std::optional<Procedure> Characterize(
304-
const ProcedureRef &, const IntrinsicProcTable &);
302+
const ProcedureRef &, FoldingContext &);
305303

306304
// At most one of these will return true.
307305
// For "EXTERNAL P" with no type for or calls to P, both will be false.

flang/include/flang/Evaluate/check-expression.h

+32-21
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@ class Scope;
2424
}
2525

2626
namespace Fortran::evaluate {
27-
class IntrinsicProcTable;
2827

2928
// Predicate: true when an expression is a constant expression (in the
3029
// strict sense of the Fortran standard); it may not (yet) be a hard
@@ -35,6 +34,12 @@ extern template bool IsConstantExpr(const Expr<SomeInteger> &);
3534
extern template bool IsConstantExpr(const Expr<SubscriptInteger> &);
3635
extern template bool IsConstantExpr(const StructureConstructor &);
3736

37+
// Predicate: true when an expression actually is a typed Constant<T>,
38+
// perhaps with parentheses and wrapping around it. False for all typeless
39+
// expressions, including BOZ literals.
40+
template <typename A> bool IsActuallyConstant(const A &);
41+
extern template bool IsActuallyConstant(const Expr<SomeType> &);
42+
3843
// Checks whether an expression is an object designator with
3944
// constant addressing and no vector-valued subscript.
4045
// If a non-null ContextualMessages pointer is passed, an error message
@@ -46,38 +51,44 @@ bool IsInitialProcedureTarget(const Symbol &);
4651
bool IsInitialProcedureTarget(const ProcedureDesignator &);
4752
bool IsInitialProcedureTarget(const Expr<SomeType> &);
4853

54+
// Validate the value of a named constant, the static initial
55+
// value of a non-pointer non-allocatable non-dummy variable, or the
56+
// default initializer of a component of a derived type (or instantiation
57+
// of a derived type). Converts type and expands scalars as necessary.
58+
std::optional<Expr<SomeType>> NonPointerInitializationExpr(const Symbol &,
59+
Expr<SomeType> &&, FoldingContext &,
60+
const semantics::Scope *instantiation = nullptr);
61+
4962
// Check whether an expression is a specification expression
5063
// (10.1.11(2), C1010). Constant expressions are always valid
5164
// specification expressions.
5265

5366
template <typename A>
54-
void CheckSpecificationExpr(const A &, parser::ContextualMessages &,
55-
const semantics::Scope &, const IntrinsicProcTable &);
56-
extern template void CheckSpecificationExpr(const Expr<SomeType> &x,
57-
parser::ContextualMessages &, const semantics::Scope &,
58-
const IntrinsicProcTable &);
59-
extern template void CheckSpecificationExpr(const Expr<SomeInteger> &x,
60-
parser::ContextualMessages &, const semantics::Scope &,
61-
const IntrinsicProcTable &);
67+
void CheckSpecificationExpr(
68+
const A &, const semantics::Scope &, FoldingContext &);
69+
extern template void CheckSpecificationExpr(
70+
const Expr<SomeType> &x, const semantics::Scope &, FoldingContext &);
71+
extern template void CheckSpecificationExpr(
72+
const Expr<SomeInteger> &x, const semantics::Scope &, FoldingContext &);
6273
extern template void CheckSpecificationExpr(const Expr<SubscriptInteger> &x,
63-
parser::ContextualMessages &, const semantics::Scope &,
64-
const IntrinsicProcTable &);
74+
const semantics::Scope &, FoldingContext &);
6575
extern template void CheckSpecificationExpr(
66-
const std::optional<Expr<SomeType>> &x, parser::ContextualMessages &,
67-
const semantics::Scope &, const IntrinsicProcTable &);
76+
const std::optional<Expr<SomeType>> &x, const semantics::Scope &,
77+
FoldingContext &);
6878
extern template void CheckSpecificationExpr(
69-
const std::optional<Expr<SomeInteger>> &x, parser::ContextualMessages &,
70-
const semantics::Scope &, const IntrinsicProcTable &);
79+
const std::optional<Expr<SomeInteger>> &x, const semantics::Scope &,
80+
FoldingContext &);
7181
extern template void CheckSpecificationExpr(
72-
const std::optional<Expr<SubscriptInteger>> &x,
73-
parser::ContextualMessages &, const semantics::Scope &,
74-
const IntrinsicProcTable &);
82+
const std::optional<Expr<SubscriptInteger>> &x, const semantics::Scope &,
83+
FoldingContext &);
7584

7685
// Simple contiguity (9.5.4)
77-
template <typename A>
78-
bool IsSimplyContiguous(const A &, const IntrinsicProcTable &);
86+
template <typename A> bool IsSimplyContiguous(const A &, FoldingContext &);
7987
extern template bool IsSimplyContiguous(
80-
const Expr<SomeType> &, const IntrinsicProcTable &);
88+
const Expr<SomeType> &, FoldingContext &);
89+
90+
template <typename A> bool IsErrorExpr(const A &);
91+
extern template bool IsErrorExpr(const Expr<SomeType> &);
8192

8293
} // namespace Fortran::evaluate
8394
#endif

flang/include/flang/Evaluate/common.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -236,7 +236,7 @@ class FoldingContext {
236236
bool flushSubnormalsToZero() const { return flushSubnormalsToZero_; }
237237
bool bigEndian() const { return bigEndian_; }
238238
const semantics::DerivedTypeSpec *pdtInstance() const { return pdtInstance_; }
239-
const evaluate::IntrinsicProcTable &intrinsics() const { return intrinsics_; }
239+
const IntrinsicProcTable &intrinsics() const { return intrinsics_; }
240240

241241
ConstantSubscript &StartImpliedDo(parser::CharBlock, ConstantSubscript = 1);
242242
std::optional<ConstantSubscript> GetImpliedDo(parser::CharBlock) const;

flang/include/flang/Evaluate/shape.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -210,7 +210,8 @@ std::optional<ConstantSubscripts> GetConstantExtents(
210210
// are known.
211211
bool CheckConformance(parser::ContextualMessages &, const Shape &left,
212212
const Shape &right, const char *leftIs = "left operand",
213-
const char *rightIs = "right operand");
213+
const char *rightIs = "right operand", bool leftScalarExpandable = true,
214+
bool rightScalarExpandable = true);
214215

215216
// Increments one-based subscripts in element order (first varies fastest)
216217
// and returns true when they remain in range; resets them all to one and

flang/include/flang/Evaluate/tools.h

+44-3
Original file line numberDiff line numberDiff line change
@@ -233,7 +233,11 @@ bool IsArrayElement(const Expr<T> &expr, bool intoSubstring = false) {
233233
while (const Component * component{std::get_if<Component>(&ref->u)}) {
234234
ref = &component->base();
235235
}
236-
return std::holds_alternative<ArrayRef>(ref->u);
236+
if (const auto *coarrayRef{std::get_if<CoarrayRef>(&ref->u)}) {
237+
return !coarrayRef->subscript().empty();
238+
} else {
239+
return std::holds_alternative<ArrayRef>(ref->u);
240+
}
237241
} else {
238242
return false;
239243
}
@@ -830,9 +834,9 @@ parser::Message *SayWithDeclaration(
830834
// Check for references to impure procedures; returns the name
831835
// of one to complain about, if any exist.
832836
std::optional<std::string> FindImpureCall(
833-
const IntrinsicProcTable &, const Expr<SomeType> &);
837+
FoldingContext &, const Expr<SomeType> &);
834838
std::optional<std::string> FindImpureCall(
835-
const IntrinsicProcTable &, const ProcedureRef &);
839+
FoldingContext &, const ProcedureRef &);
836840

837841
// Predicate: is a scalar expression suitable for naive scalar expansion
838842
// in the flattening of an array expression?
@@ -857,6 +861,41 @@ std::optional<parser::MessageFixedText> CheckProcCompatibility(bool isCall,
857861
const std::optional<characteristics::Procedure> &lhsProcedure,
858862
const characteristics::Procedure *rhsProcedure);
859863

864+
// Scalar constant expansion
865+
class ScalarConstantExpander {
866+
public:
867+
explicit ScalarConstantExpander(ConstantSubscripts &&extents)
868+
: extents_{std::move(extents)} {}
869+
ScalarConstantExpander(
870+
ConstantSubscripts &&extents, std::optional<ConstantSubscripts> &&lbounds)
871+
: extents_{std::move(extents)}, lbounds_{std::move(lbounds)} {}
872+
ScalarConstantExpander(
873+
ConstantSubscripts &&extents, ConstantSubscripts &&lbounds)
874+
: extents_{std::move(extents)}, lbounds_{std::move(lbounds)} {}
875+
876+
template <typename A> A Expand(A &&x) const {
877+
return std::move(x); // default case
878+
}
879+
template <typename T> Constant<T> Expand(Constant<T> &&x) {
880+
auto expanded{x.Reshape(std::move(extents_))};
881+
if (lbounds_) {
882+
expanded.set_lbounds(std::move(*lbounds_));
883+
}
884+
return expanded;
885+
}
886+
template <typename T> Constant<T> Expand(Parentheses<T> &&x) {
887+
return Expand(std::move(x)); // Constant<> can be parenthesized
888+
}
889+
template <typename T> Expr<T> Expand(Expr<T> &&x) {
890+
return std::visit([&](auto &&x) { return Expr<T>{Expand(std::move(x))}; },
891+
std::move(x.u));
892+
}
893+
894+
private:
895+
ConstantSubscripts extents_;
896+
std::optional<ConstantSubscripts> lbounds_;
897+
};
898+
860899
} // namespace Fortran::evaluate
861900

862901
namespace Fortran::semantics {
@@ -875,6 +914,8 @@ bool IsProcedurePointer(const Symbol &);
875914
bool IsSaved(const Symbol &); // saved implicitly or explicitly
876915
bool IsDummy(const Symbol &);
877916
bool IsFunctionResult(const Symbol &);
917+
bool IsKindTypeParameter(const Symbol &);
918+
bool IsLenTypeParameter(const Symbol &);
878919

879920
// Follow use, host, and construct assocations to a variable, if any.
880921
const Symbol *GetAssociationRoot(const Symbol &);

flang/include/flang/Evaluate/type.h

-3
Original file line numberDiff line numberDiff line change
@@ -421,9 +421,6 @@ int SelectedIntKind(std::int64_t precision = 0);
421421
int SelectedRealKind(
422422
std::int64_t precision = 0, std::int64_t range = 0, std::int64_t radix = 2);
423423

424-
// Utilities
425-
bool IsKindTypeParameter(const semantics::Symbol &);
426-
427424
// For generating "[extern] template class", &c. boilerplate
428425
#define EXPAND_FOR_EACH_INTEGER_KIND(M, P, S) \
429426
M(P, S, 1) M(P, S, 2) M(P, S, 4) M(P, S, 8) M(P, S, 16)

flang/include/flang/Parser/message.h

+14-1
Original file line numberDiff line numberDiff line change
@@ -255,6 +255,7 @@ class ContextualMessages {
255255

256256
CharBlock at() const { return at_; }
257257
Messages *messages() const { return messages_; }
258+
Message::Reference contextMessage() const { return contextMessage_; }
258259
bool empty() const { return !messages_ || messages_->empty(); }
259260

260261
// Set CharBlock for messages; restore when the returned value is deleted
@@ -265,6 +266,13 @@ class ContextualMessages {
265266
return common::ScopedSet(at_, std::move(at));
266267
}
267268

269+
common::Restorer<Message::Reference> SetContext(Message *m) {
270+
if (!m) {
271+
m = contextMessage_.get();
272+
}
273+
return common::ScopedSet(contextMessage_, m);
274+
}
275+
268276
// Diverts messages to another buffer; restored when the returned
269277
// value is deleted.
270278
common::Restorer<Messages *> SetMessages(Messages &buffer) {
@@ -277,7 +285,11 @@ class ContextualMessages {
277285

278286
template <typename... A> Message *Say(CharBlock at, A &&...args) {
279287
if (messages_ != nullptr) {
280-
return &messages_->Say(at, std::forward<A>(args)...);
288+
auto &msg{messages_->Say(at, std::forward<A>(args)...)};
289+
if (contextMessage_) {
290+
msg.SetContext(contextMessage_.get());
291+
}
292+
return &msg;
281293
} else {
282294
return nullptr;
283295
}
@@ -290,6 +302,7 @@ class ContextualMessages {
290302
private:
291303
CharBlock at_;
292304
Messages *messages_{nullptr};
305+
Message::Reference contextMessage_;
293306
};
294307
} // namespace Fortran::parser
295308
#endif // FORTRAN_PARSER_MESSAGE_H_

0 commit comments

Comments
 (0)