@@ -39,13 +39,16 @@ static void CopyAttrs(const semantics::Symbol &src, A &dst,
39
39
// Shapes of function results and dummy arguments have to have
40
40
// the same rank, the same deferred dimensions, and the same
41
41
// 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 ()) {
45
48
return false ;
46
49
}
47
- auto yIter{y. begin ()};
48
- for (const auto &xDim : x) {
50
+ auto yIter{y-> begin ()};
51
+ for (const auto &xDim : * x) {
49
52
const auto &yDim{*yIter++};
50
53
if (xDim && yDim) {
51
54
if (auto equiv{AreEquivalentInInterface (*xDim, *yDim)}) {
@@ -178,9 +181,11 @@ bool TypeAndShape::IsCompatibleWith(parser::ContextualMessages &messages,
178
181
thatIs, that.AsFortran (), thisIs, AsFortran ());
179
182
return false ;
180
183
}
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*/ ));
184
189
}
185
190
186
191
std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureElementSizeInBytes (
@@ -201,11 +206,11 @@ std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureElementSizeInBytes(
201
206
202
207
std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureSizeInBytes (
203
208
FoldingContext &foldingContext) const {
204
- if (auto elements{GetSize (Shape{ shape_} )}) {
209
+ if (auto elements{GetSize (shape_)}) {
205
210
// Sizes of arrays (even with single elements) are multiples of
206
211
// their alignments.
207
212
if (auto elementBytes{
208
- MeasureElementSizeInBytes (foldingContext, GetRank (shape_ ) > 0 )}) {
213
+ MeasureElementSizeInBytes (foldingContext, Rank ( ) > 0 )}) {
209
214
return Fold (
210
215
foldingContext, std::move (*elements) * std::move (*elementBytes));
211
216
}
@@ -254,10 +259,12 @@ std::string TypeAndShape::AsFortran() const {
254
259
llvm::raw_ostream &TypeAndShape::Dump (llvm::raw_ostream &o) const {
255
260
o << type_.AsFortran (LEN_ ? LEN_->AsFortran () : " " );
256
261
attrs_.Dump (o, EnumToString);
257
- if (!shape_.empty ()) {
262
+ if (!shape_) {
263
+ o << " dimension(..)" ;
264
+ } else if (!shape_->empty ()) {
258
265
o << " dimension" ;
259
266
char sep{' (' };
260
- for (const auto &expr : shape_) {
267
+ for (const auto &expr : * shape_) {
261
268
o << sep;
262
269
sep = ' ,' ;
263
270
if (expr) {
@@ -1112,6 +1119,7 @@ bool FunctionResult::CanBeReturnedViaImplicitInterface(
1112
1119
1113
1120
static std::optional<std::string> AreIncompatibleFunctionResultShapes (
1114
1121
const Shape &x, const Shape &y) {
1122
+ // Function results cannot be assumed-rank, hence the non optional arguments.
1115
1123
int rank{GetRank (x)};
1116
1124
if (int yrank{GetRank (y)}; yrank != rank) {
1117
1125
return " rank " s + std::to_string (rank) + " vs " + std::to_string (yrank);
@@ -1147,7 +1155,8 @@ bool FunctionResult::IsCompatibleWith(
1147
1155
}
1148
1156
} else if (!attrs.test (Attr::Allocatable) && !attrs.test (Attr::Pointer) &&
1149
1157
(details = AreIncompatibleFunctionResultShapes (
1150
- ifaceTypeShape->shape (), actualTypeShape->shape ()))) {
1158
+ ifaceTypeShape->shape ().value (),
1159
+ actualTypeShape->shape ().value ()))) {
1151
1160
if (whyNot) {
1152
1161
*whyNot = " function results have distinct extents (" + *details + ' )' ;
1153
1162
}
0 commit comments