Skip to content

Commit 73cf014

Browse files
authored
[flang] harden TypeAndShape for assumed-ranks (llvm#96234)
SIZEOF and C_SIZEOF were broken for assumed-ranks because `TypeAndShape::MeasureSizeInBytes` behaved as a scalar because the `TypeAndShape::shape_` member was the same for scalar and assumed-ranks. The easy fix would have been to add special handling in `MeasureSizeInBytes` for assumed-ranks using the TypeAndShape attributes, but I think this solution would leave `TypeAndShape::shape_` manipulation fragile to future developers. Hence, I went for the solution that turn shape_ into a `std::optional<Shape>`.
1 parent 33676ba commit 73cf014

11 files changed

+90
-55
lines changed

flang/include/flang/Evaluate/characteristics.h

+11-10
Original file line numberDiff line numberDiff line change
@@ -55,26 +55,26 @@ std::optional<bool> DistinguishableOpOrAssign(
5555
// Shapes of function results and dummy arguments have to have
5656
// the same rank, the same deferred dimensions, and the same
5757
// values for explicit dimensions when constant.
58-
bool ShapesAreCompatible(
59-
const Shape &, const Shape &, bool *possibleWarning = nullptr);
58+
bool ShapesAreCompatible(const std::optional<Shape> &,
59+
const std::optional<Shape> &, bool *possibleWarning = nullptr);
6060

6161
class TypeAndShape {
6262
public:
6363
ENUM_CLASS(
6464
Attr, AssumedRank, AssumedShape, AssumedSize, DeferredShape, Coarray)
6565
using Attrs = common::EnumSet<Attr, Attr_enumSize>;
6666

67-
explicit TypeAndShape(DynamicType t) : type_{t} { AcquireLEN(); }
68-
TypeAndShape(DynamicType t, int rank) : type_{t}, shape_(rank) {
67+
explicit TypeAndShape(DynamicType t) : type_{t}, shape_{Shape{}} {
68+
AcquireLEN();
69+
}
70+
TypeAndShape(DynamicType t, int rank) : type_{t}, shape_{Shape(rank)} {
6971
AcquireLEN();
7072
}
7173
TypeAndShape(DynamicType t, Shape &&s) : type_{t}, shape_{std::move(s)} {
7274
AcquireLEN();
7375
}
7476
TypeAndShape(DynamicType t, std::optional<Shape> &&s) : type_{t} {
75-
if (s) {
76-
shape_ = std::move(*s);
77-
}
77+
shape_ = std::move(s);
7878
AcquireLEN();
7979
}
8080
DEFAULT_CONSTRUCTORS_AND_ASSIGNMENTS(TypeAndShape)
@@ -172,11 +172,12 @@ class TypeAndShape {
172172
LEN_ = std::move(len);
173173
return *this;
174174
}
175-
const Shape &shape() const { return shape_; }
175+
const std::optional<Shape> &shape() const { return shape_; }
176176
const Attrs &attrs() const { return attrs_; }
177177
int corank() const { return corank_; }
178178

179-
int Rank() const { return GetRank(shape_); }
179+
// Return -1 for assumed-rank as a safety.
180+
int Rank() const { return shape_ ? GetRank(*shape_) : -1; }
180181

181182
// Can sequence association apply to this argument?
182183
bool CanBeSequenceAssociated() const {
@@ -211,7 +212,7 @@ class TypeAndShape {
211212
protected:
212213
DynamicType type_;
213214
std::optional<Expr<SubscriptInteger>> LEN_;
214-
Shape shape_;
215+
std::optional<Shape> shape_;
215216
Attrs attrs_;
216217
int corank_{0};
217218
};

flang/include/flang/Evaluate/shape.h

+13
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,13 @@ Constant<ExtentType> AsConstantShape(const ConstantSubscripts &);
4646
ConstantSubscripts AsConstantExtents(const Constant<ExtentType> &);
4747
std::optional<ConstantSubscripts> AsConstantExtents(
4848
FoldingContext &, const Shape &);
49+
inline std::optional<ConstantSubscripts> AsConstantExtents(
50+
FoldingContext &foldingContext, const std::optional<Shape> &maybeShape) {
51+
if (maybeShape) {
52+
return AsConstantExtents(foldingContext, *maybeShape);
53+
}
54+
return std::nullopt;
55+
}
4956
Shape AsShape(const ConstantSubscripts &);
5057
std::optional<Shape> AsShape(const std::optional<ConstantSubscripts> &);
5158

@@ -121,6 +128,12 @@ MaybeExtentExpr CountTrips(
121128
// Computes SIZE() == PRODUCT(shape)
122129
MaybeExtentExpr GetSize(Shape &&);
123130
ConstantSubscript GetSize(const ConstantSubscripts &);
131+
inline MaybeExtentExpr GetSize(const std::optional<Shape> &maybeShape) {
132+
if (maybeShape) {
133+
return GetSize(Shape(*maybeShape));
134+
}
135+
return std::nullopt;
136+
}
124137

125138
// Utility predicate: does an expression reference any implied DO index?
126139
bool ContainsAnyImpliedDoIndex(const ExtentExpr &);

flang/lib/Evaluate/characteristics.cpp

+22-13
Original file line numberDiff line numberDiff line change
@@ -39,13 +39,16 @@ static void CopyAttrs(const semantics::Symbol &src, A &dst,
3939
// Shapes of function results and dummy arguments have to have
4040
// the same rank, the same deferred dimensions, and the same
4141
// values for explicit dimensions when constant.
42-
bool ShapesAreCompatible(
43-
const Shape &x, const Shape &y, bool *possibleWarning) {
44-
if (x.size() != y.size()) {
42+
bool ShapesAreCompatible(const std::optional<Shape> &x,
43+
const std::optional<Shape> &y, bool *possibleWarning) {
44+
if (!x || !y) {
45+
return !x && !y;
46+
}
47+
if (x->size() != y->size()) {
4548
return false;
4649
}
47-
auto yIter{y.begin()};
48-
for (const auto &xDim : x) {
50+
auto yIter{y->begin()};
51+
for (const auto &xDim : *x) {
4952
const auto &yDim{*yIter++};
5053
if (xDim && yDim) {
5154
if (auto equiv{AreEquivalentInInterface(*xDim, *yDim)}) {
@@ -178,9 +181,11 @@ bool TypeAndShape::IsCompatibleWith(parser::ContextualMessages &messages,
178181
thatIs, that.AsFortran(), thisIs, AsFortran());
179182
return false;
180183
}
181-
return omitShapeConformanceCheck ||
182-
CheckConformance(messages, shape_, that.shape_, flags, thisIs, thatIs)
183-
.value_or(true /*fail only when nonconformance is known now*/);
184+
return omitShapeConformanceCheck || (!shape_ && !that.shape_) ||
185+
(shape_ && that.shape_ &&
186+
CheckConformance(
187+
messages, *shape_, *that.shape_, flags, thisIs, thatIs)
188+
.value_or(true /*fail only when nonconformance is known now*/));
184189
}
185190

186191
std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureElementSizeInBytes(
@@ -201,11 +206,11 @@ std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureElementSizeInBytes(
201206

202207
std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureSizeInBytes(
203208
FoldingContext &foldingContext) const {
204-
if (auto elements{GetSize(Shape{shape_})}) {
209+
if (auto elements{GetSize(shape_)}) {
205210
// Sizes of arrays (even with single elements) are multiples of
206211
// their alignments.
207212
if (auto elementBytes{
208-
MeasureElementSizeInBytes(foldingContext, GetRank(shape_) > 0)}) {
213+
MeasureElementSizeInBytes(foldingContext, Rank() > 0)}) {
209214
return Fold(
210215
foldingContext, std::move(*elements) * std::move(*elementBytes));
211216
}
@@ -254,10 +259,12 @@ std::string TypeAndShape::AsFortran() const {
254259
llvm::raw_ostream &TypeAndShape::Dump(llvm::raw_ostream &o) const {
255260
o << type_.AsFortran(LEN_ ? LEN_->AsFortran() : "");
256261
attrs_.Dump(o, EnumToString);
257-
if (!shape_.empty()) {
262+
if (!shape_) {
263+
o << " dimension(..)";
264+
} else if (!shape_->empty()) {
258265
o << " dimension";
259266
char sep{'('};
260-
for (const auto &expr : shape_) {
267+
for (const auto &expr : *shape_) {
261268
o << sep;
262269
sep = ',';
263270
if (expr) {
@@ -1112,6 +1119,7 @@ bool FunctionResult::CanBeReturnedViaImplicitInterface(
11121119

11131120
static std::optional<std::string> AreIncompatibleFunctionResultShapes(
11141121
const Shape &x, const Shape &y) {
1122+
// Function results cannot be assumed-rank, hence the non optional arguments.
11151123
int rank{GetRank(x)};
11161124
if (int yrank{GetRank(y)}; yrank != rank) {
11171125
return "rank "s + std::to_string(rank) + " vs " + std::to_string(yrank);
@@ -1147,7 +1155,8 @@ bool FunctionResult::IsCompatibleWith(
11471155
}
11481156
} else if (!attrs.test(Attr::Allocatable) && !attrs.test(Attr::Pointer) &&
11491157
(details = AreIncompatibleFunctionResultShapes(
1150-
ifaceTypeShape->shape(), actualTypeShape->shape()))) {
1158+
ifaceTypeShape->shape().value(),
1159+
actualTypeShape->shape().value()))) {
11511160
if (whyNot) {
11521161
*whyNot = "function results have distinct extents (" + *details + ')';
11531162
}

flang/lib/Evaluate/check-expression.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -419,7 +419,7 @@ std::optional<Expr<SomeType>> NonPointerInitializationExpr(const Symbol &symbol,
419419
if (converted) {
420420
auto folded{Fold(context, std::move(*converted))};
421421
if (IsActuallyConstant(folded)) {
422-
int symRank{GetRank(symTS->shape())};
422+
int symRank{symTS->Rank()};
423423
if (IsImpliedShape(symbol)) {
424424
if (folded.Rank() == symRank) {
425425
return ArrayConstantBoundChanger{
@@ -442,7 +442,8 @@ std::optional<Expr<SomeType>> NonPointerInitializationExpr(const Symbol &symbol,
442442
context, GetRawLowerBounds(context, NamedEntity{symbol}))}
443443
.Expand(std::move(folded));
444444
} else if (auto resultShape{GetShape(context, folded)}) {
445-
if (CheckConformance(context.messages(), symTS->shape(),
445+
CHECK(symTS->shape()); // Assumed-ranks cannot be initialized.
446+
if (CheckConformance(context.messages(), *symTS->shape(),
446447
*resultShape, CheckConformanceFlags::None,
447448
"initialized object", "initialization expression")
448449
.value_or(false /*fail if not known now to conform*/)) {

flang/lib/Evaluate/shape.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1037,7 +1037,7 @@ auto GetShapeHelper::operator()(const ProcedureRef &call) const -> Result {
10371037
} else if (context_) {
10381038
if (auto moldTypeAndShape{characteristics::TypeAndShape::Characterize(
10391039
call.arguments().at(1), *context_)}) {
1040-
if (GetRank(moldTypeAndShape->shape()) == 0) {
1040+
if (moldTypeAndShape->Rank() == 0) {
10411041
// SIZE= is absent and MOLD= is scalar: result is scalar
10421042
return ScalarShape();
10431043
} else {

flang/lib/Lower/CallInterface.cpp

+13-13
Original file line numberDiff line numberDiff line change
@@ -177,9 +177,10 @@ mlir::Location Fortran::lower::CallerInterface::getCalleeLocation() const {
177177
// explicit.
178178
static Fortran::evaluate::characteristics::DummyDataObject
179179
asImplicitArg(Fortran::evaluate::characteristics::DummyDataObject &&dummy) {
180-
Fortran::evaluate::Shape shape =
181-
dummy.type.attrs().none() ? dummy.type.shape()
182-
: Fortran::evaluate::Shape(dummy.type.Rank());
180+
std::optional<Fortran::evaluate::Shape> shape =
181+
dummy.type.attrs().none()
182+
? dummy.type.shape()
183+
: std::make_optional<Fortran::evaluate::Shape>(dummy.type.Rank());
183184
return Fortran::evaluate::characteristics::DummyDataObject(
184185
Fortran::evaluate::characteristics::TypeAndShape(dummy.type.type(),
185186
std::move(shape)));
@@ -1308,18 +1309,17 @@ class Fortran::lower::CallInterfaceImpl {
13081309
// with the shape (may contain unknown extents) for arrays.
13091310
std::optional<fir::SequenceType::Shape> getBounds(
13101311
const Fortran::evaluate::characteristics::TypeAndShape &typeAndShape) {
1311-
using ShapeAttr = Fortran::evaluate::characteristics::TypeAndShape::Attr;
1312-
if (typeAndShape.shape().empty() &&
1313-
!typeAndShape.attrs().test(ShapeAttr::AssumedRank))
1312+
if (typeAndShape.shape() && typeAndShape.shape()->empty())
13141313
return std::nullopt;
13151314
fir::SequenceType::Shape bounds;
1316-
for (const std::optional<Fortran::evaluate::ExtentExpr> &extent :
1317-
typeAndShape.shape()) {
1318-
fir::SequenceType::Extent bound = fir::SequenceType::getUnknownExtent();
1319-
if (std::optional<std::int64_t> i = toInt64(extent))
1320-
bound = *i;
1321-
bounds.emplace_back(bound);
1322-
}
1315+
if (typeAndShape.shape())
1316+
for (const std::optional<Fortran::evaluate::ExtentExpr> &extent :
1317+
*typeAndShape.shape()) {
1318+
fir::SequenceType::Extent bound = fir::SequenceType::getUnknownExtent();
1319+
if (std::optional<std::int64_t> i = toInt64(extent))
1320+
bound = *i;
1321+
bounds.emplace_back(bound);
1322+
}
13231323
return bounds;
13241324
}
13251325
std::optional<std::int64_t>

flang/lib/Semantics/check-call.cpp

+10-11
Original file line numberDiff line numberDiff line change
@@ -143,8 +143,8 @@ static void CheckCharacterActual(evaluate::Expr<evaluate::SomeType> &actual,
143143
bool canAssociate{CanAssociateWithStorageSequence(dummy)};
144144
if (dummy.type.Rank() > 0 && canAssociate) {
145145
// Character storage sequence association (F'2023 15.5.2.12p4)
146-
if (auto dummySize{evaluate::ToInt64(evaluate::Fold(foldingContext,
147-
evaluate::GetSize(evaluate::Shape{dummy.type.shape()})))}) {
146+
if (auto dummySize{evaluate::ToInt64(evaluate::Fold(
147+
foldingContext, evaluate::GetSize(dummy.type.shape())))}) {
148148
auto dummyChars{*dummySize * *dummyLength};
149149
if (actualType.Rank() == 0) {
150150
evaluate::DesignatorFolder folder{
@@ -183,8 +183,7 @@ static void CheckCharacterActual(evaluate::Expr<evaluate::SomeType> &actual,
183183
}
184184
} else { // actual.type.Rank() > 0
185185
if (auto actualSize{evaluate::ToInt64(evaluate::Fold(
186-
foldingContext,
187-
evaluate::GetSize(evaluate::Shape(actualType.shape()))))};
186+
foldingContext, evaluate::GetSize(actualType.shape())))};
188187
actualSize &&
189188
*actualSize * *actualLength < *dummySize * *dummyLength &&
190189
(extentErrors ||
@@ -251,7 +250,7 @@ static void ConvertIntegerActual(evaluate::Expr<evaluate::SomeType> &actual,
251250
if (dummyType.type().category() == TypeCategory::Integer &&
252251
actualType.type().category() == TypeCategory::Integer &&
253252
dummyType.type().kind() != actualType.type().kind() &&
254-
GetRank(dummyType.shape()) == 0 && GetRank(actualType.shape()) == 0 &&
253+
dummyType.Rank() == 0 && actualType.Rank() == 0 &&
255254
!evaluate::IsVariable(actual)) {
256255
auto converted{
257256
evaluate::ConvertToType(dummyType.type(), std::move(actual))};
@@ -387,10 +386,10 @@ static void CheckExplicitDataArg(const characteristics::DummyDataObject &dummy,
387386
// if the actual argument is an array or array element designator,
388387
// and the dummy is an array, but not assumed-shape or an INTENT(IN)
389388
// pointer that's standing in for an assumed-shape dummy.
390-
} else {
389+
} else if (dummy.type.shape() && actualType.shape()) {
391390
// Let CheckConformance accept actual scalars; storage association
392391
// cases are checked here below.
393-
CheckConformance(messages, dummy.type.shape(), actualType.shape(),
392+
CheckConformance(messages, *dummy.type.shape(), *actualType.shape(),
394393
dummyIsAllocatableOrPointer
395394
? evaluate::CheckConformanceFlags::None
396395
: evaluate::CheckConformanceFlags::RightScalarExpandable,
@@ -579,8 +578,8 @@ static void CheckExplicitDataArg(const characteristics::DummyDataObject &dummy,
579578
CanAssociateWithStorageSequence(dummy) &&
580579
!dummy.attrs.test(
581580
characteristics::DummyDataObject::Attr::DeducedFromActual)) {
582-
if (auto dummySize{evaluate::ToInt64(evaluate::Fold(foldingContext,
583-
evaluate::GetSize(evaluate::Shape{dummy.type.shape()})))}) {
581+
if (auto dummySize{evaluate::ToInt64(evaluate::Fold(
582+
foldingContext, evaluate::GetSize(dummy.type.shape())))}) {
584583
if (actualRank == 0 && !actualIsAssumedRank) {
585584
if (evaluate::IsArrayElement(actual)) {
586585
// Actual argument is a scalar array element
@@ -622,8 +621,8 @@ static void CheckExplicitDataArg(const characteristics::DummyDataObject &dummy,
622621
}
623622
}
624623
} else { // actualRank > 0 || actualIsAssumedRank
625-
if (auto actualSize{evaluate::ToInt64(evaluate::Fold(foldingContext,
626-
evaluate::GetSize(evaluate::Shape(actualType.shape()))))};
624+
if (auto actualSize{evaluate::ToInt64(evaluate::Fold(
625+
foldingContext, evaluate::GetSize(actualType.shape())))};
627626
actualSize && *actualSize < *dummySize &&
628627
(extentErrors ||
629628
context.ShouldWarn(common::UsageWarning::ShortArrayActual))) {

flang/lib/Semantics/check-declarations.cpp

+7
Original file line numberDiff line numberDiff line change
@@ -1325,6 +1325,13 @@ class SubprogramMatchHelper {
13251325
bool CheckSameAttrs(const Symbol &, const Symbol &, ATTRS, ATTRS);
13261326
bool ShapesAreCompatible(const DummyDataObject &, const DummyDataObject &);
13271327
evaluate::Shape FoldShape(const evaluate::Shape &);
1328+
std::optional<evaluate::Shape> FoldShape(
1329+
const std::optional<evaluate::Shape> &shape) {
1330+
if (shape) {
1331+
return FoldShape(*shape);
1332+
}
1333+
return std::nullopt;
1334+
}
13281335
std::string AsFortran(DummyDataObject::Attr attr) {
13291336
return parser::ToUpperCaseLetters(DummyDataObject::EnumToString(attr));
13301337
}

flang/lib/Semantics/pointer-assignment.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -333,8 +333,8 @@ bool PointerAssignmentChecker::Check(const evaluate::Designator<T> &d) {
333333

334334
} else if (!isBoundsRemapping_ &&
335335
!lhsType_->attrs().test(TypeAndShape::Attr::AssumedRank)) {
336-
int lhsRank{evaluate::GetRank(lhsType_->shape())};
337-
int rhsRank{evaluate::GetRank(rhsType->shape())};
336+
int lhsRank{lhsType_->Rank()};
337+
int rhsRank{rhsType->Rank()};
338338
if (lhsRank != rhsRank) {
339339
msg = MessageFormattedText{
340340
"Pointer has rank %d but target has rank %d"_err_en_US, lhsRank,

flang/lib/Semantics/runtime-type-info.cpp

+2-3
Original file line numberDiff line numberDiff line change
@@ -748,7 +748,7 @@ evaluate::StructureConstructor RuntimeTableBuilder::DescribeComponent(
748748
symbol, foldingContext)};
749749
CHECK(typeAndShape.has_value());
750750
auto dyType{typeAndShape->type()};
751-
const auto &shape{typeAndShape->shape()};
751+
int rank{typeAndShape->Rank()};
752752
AddValue(values, componentSchema_, "name"s,
753753
SaveNameAsPointerTarget(scope, symbol.name().ToString()));
754754
AddValue(values, componentSchema_, "category"s,
@@ -830,7 +830,6 @@ evaluate::StructureConstructor RuntimeTableBuilder::DescribeComponent(
830830
SomeExpr{evaluate::NullPointer{}});
831831
}
832832
// Shape information
833-
int rank{evaluate::GetRank(shape)};
834833
AddValue(values, componentSchema_, "rank"s, IntExpr<1>(rank));
835834
if (rank > 0 && !IsAllocatable(symbol) && !IsPointer(symbol)) {
836835
std::vector<evaluate::StructureConstructor> bounds;
@@ -1143,7 +1142,7 @@ void RuntimeTableBuilder::DescribeSpecialProc(
11431142
isArgDescriptorSet |= 1;
11441143
} else {
11451144
which = scalarFinalEnum_;
1146-
if (int rank{evaluate::GetRank(typeAndShape.shape())}; rank > 0) {
1145+
if (int rank{typeAndShape.Rank()}; rank > 0) {
11471146
which = IntExpr<1>(ToInt64(which).value() + rank);
11481147
if (dummyData.IsPassedByDescriptor(proc->IsBindC())) {
11491148
argThatMightBeDescriptor = 1;

flang/test/Evaluate/rewrite06.f90

+6
Original file line numberDiff line numberDiff line change
@@ -31,3 +31,9 @@ subroutine test(k)
3131
print *, storage_size(return_pdt(k))
3232
end subroutine
3333
end module
34+
35+
subroutine test_assumed_rank(x)
36+
real :: x(..)
37+
!CHECK: PRINT *, sizeof(x)
38+
print *, sizeof(x)
39+
end subroutine

0 commit comments

Comments
 (0)