xref: /llvm-project/flang/lib/Evaluate/characteristics.cpp (revision c6b9df0fbd7fc6f93e3bd7368cf966d55d802f4c)
1 //===-- lib/Evaluate/characteristics.cpp ----------------------------------===//
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 #include "flang/Evaluate/characteristics.h"
10 #include "flang/Common/indirection.h"
11 #include "flang/Evaluate/check-expression.h"
12 #include "flang/Evaluate/fold.h"
13 #include "flang/Evaluate/intrinsics.h"
14 #include "flang/Evaluate/tools.h"
15 #include "flang/Evaluate/type.h"
16 #include "flang/Parser/message.h"
17 #include "flang/Semantics/scope.h"
18 #include "flang/Semantics/symbol.h"
19 #include "flang/Semantics/tools.h"
20 #include "llvm/Support/raw_ostream.h"
21 #include <initializer_list>
22 
23 using namespace Fortran::parser::literals;
24 
25 namespace Fortran::evaluate::characteristics {
26 
27 // Copy attributes from a symbol to dst based on the mapping in pairs.
28 template <typename A, typename B>
29 static void CopyAttrs(const semantics::Symbol &src, A &dst,
30     const std::initializer_list<std::pair<semantics::Attr, B>> &pairs) {
31   for (const auto &pair : pairs) {
32     if (src.attrs().test(pair.first)) {
33       dst.attrs.set(pair.second);
34     }
35   }
36 }
37 
38 // Shapes of function results and dummy arguments have to have
39 // the same rank, the same deferred dimensions, and the same
40 // values for explicit dimensions when constant.
41 bool ShapesAreCompatible(const Shape &x, const Shape &y) {
42   if (x.size() != y.size()) {
43     return false;
44   }
45   auto yIter{y.begin()};
46   for (const auto &xDim : x) {
47     const auto &yDim{*yIter++};
48     if (xDim) {
49       if (!yDim || ToInt64(*xDim) != ToInt64(*yDim)) {
50         return false;
51       }
52     } else if (yDim) {
53       return false;
54     }
55   }
56   return true;
57 }
58 
59 bool TypeAndShape::operator==(const TypeAndShape &that) const {
60   return type_ == that.type_ && ShapesAreCompatible(shape_, that.shape_) &&
61       attrs_ == that.attrs_ && corank_ == that.corank_;
62 }
63 
64 TypeAndShape &TypeAndShape::Rewrite(FoldingContext &context) {
65   LEN_ = Fold(context, std::move(LEN_));
66   if (LEN_) {
67     if (auto n{ToInt64(*LEN_)}) {
68       type_ = DynamicType{type_.kind(), *n};
69     }
70   }
71   shape_ = Fold(context, std::move(shape_));
72   return *this;
73 }
74 
75 std::optional<TypeAndShape> TypeAndShape::Characterize(
76     const semantics::Symbol &symbol, FoldingContext &context) {
77   const auto &ultimate{symbol.GetUltimate()};
78   return common::visit(
79       common::visitors{
80           [&](const semantics::ProcEntityDetails &proc) {
81             if (proc.procInterface()) {
82               return Characterize(*proc.procInterface(), context);
83             } else if (proc.type()) {
84               return Characterize(*proc.type(), context);
85             } else {
86               return std::optional<TypeAndShape>{};
87             }
88           },
89           [&](const semantics::AssocEntityDetails &assoc) {
90             return Characterize(assoc, context);
91           },
92           [&](const semantics::ProcBindingDetails &binding) {
93             return Characterize(binding.symbol(), context);
94           },
95           [&](const auto &x) -> std::optional<TypeAndShape> {
96             using Ty = std::decay_t<decltype(x)>;
97             if constexpr (std::is_same_v<Ty, semantics::EntityDetails> ||
98                 std::is_same_v<Ty, semantics::ObjectEntityDetails> ||
99                 std::is_same_v<Ty, semantics::TypeParamDetails>) {
100               if (const semantics::DeclTypeSpec * type{ultimate.GetType()}) {
101                 if (auto dyType{DynamicType::From(*type)}) {
102                   TypeAndShape result{
103                       std::move(*dyType), GetShape(context, ultimate)};
104                   result.AcquireAttrs(ultimate);
105                   result.AcquireLEN(ultimate);
106                   return std::move(result.Rewrite(context));
107                 }
108               }
109             }
110             return std::nullopt;
111           },
112       },
113       // GetUltimate() used here, not ResolveAssociations(), because
114       // we need the type/rank of an associate entity from TYPE IS,
115       // CLASS IS, or RANK statement.
116       ultimate.details());
117 }
118 
119 std::optional<TypeAndShape> TypeAndShape::Characterize(
120     const semantics::AssocEntityDetails &assoc, FoldingContext &context) {
121   std::optional<TypeAndShape> result;
122   if (auto type{DynamicType::From(assoc.type())}) {
123     if (auto rank{assoc.rank()}) {
124       if (*rank >= 0 && *rank <= common::maxRank) {
125         result = TypeAndShape{std::move(*type), Shape(*rank)};
126       }
127     } else if (auto shape{GetShape(context, assoc.expr())}) {
128       result = TypeAndShape{std::move(*type), std::move(*shape)};
129     }
130     if (result && type->category() == TypeCategory::Character) {
131       if (const auto *chExpr{UnwrapExpr<Expr<SomeCharacter>>(assoc.expr())}) {
132         if (auto len{chExpr->LEN()}) {
133           result->set_LEN(std::move(*len));
134         }
135       }
136     }
137   }
138   return Fold(context, std::move(result));
139 }
140 
141 std::optional<TypeAndShape> TypeAndShape::Characterize(
142     const semantics::DeclTypeSpec &spec, FoldingContext &context) {
143   if (auto type{DynamicType::From(spec)}) {
144     return Fold(context, TypeAndShape{std::move(*type)});
145   } else {
146     return std::nullopt;
147   }
148 }
149 
150 std::optional<TypeAndShape> TypeAndShape::Characterize(
151     const ActualArgument &arg, FoldingContext &context) {
152   return Characterize(arg.UnwrapExpr(), context);
153 }
154 
155 bool TypeAndShape::IsCompatibleWith(parser::ContextualMessages &messages,
156     const TypeAndShape &that, const char *thisIs, const char *thatIs,
157     bool omitShapeConformanceCheck,
158     enum CheckConformanceFlags::Flags flags) const {
159   if (!type_.IsTkCompatibleWith(that.type_)) {
160     messages.Say(
161         "%1$s type '%2$s' is not compatible with %3$s type '%4$s'"_err_en_US,
162         thatIs, that.AsFortran(), thisIs, AsFortran());
163     return false;
164   }
165   return omitShapeConformanceCheck ||
166       CheckConformance(messages, shape_, that.shape_, flags, thisIs, thatIs)
167           .value_or(true /*fail only when nonconformance is known now*/);
168 }
169 
170 std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureElementSizeInBytes(
171     FoldingContext &foldingContext, bool align) const {
172   if (LEN_) {
173     CHECK(type_.category() == TypeCategory::Character);
174     return Fold(foldingContext,
175         Expr<SubscriptInteger>{
176             foldingContext.targetCharacteristics().GetByteSize(
177                 type_.category(), type_.kind())} *
178             Expr<SubscriptInteger>{*LEN_});
179   }
180   if (auto elementBytes{type_.MeasureSizeInBytes(foldingContext, align)}) {
181     return Fold(foldingContext, std::move(*elementBytes));
182   }
183   return std::nullopt;
184 }
185 
186 std::optional<Expr<SubscriptInteger>> TypeAndShape::MeasureSizeInBytes(
187     FoldingContext &foldingContext) const {
188   if (auto elements{GetSize(Shape{shape_})}) {
189     // Sizes of arrays (even with single elements) are multiples of
190     // their alignments.
191     if (auto elementBytes{
192             MeasureElementSizeInBytes(foldingContext, GetRank(shape_) > 0)}) {
193       return Fold(
194           foldingContext, std::move(*elements) * std::move(*elementBytes));
195     }
196   }
197   return std::nullopt;
198 }
199 
200 void TypeAndShape::AcquireAttrs(const semantics::Symbol &symbol) {
201   if (IsAssumedShape(symbol)) {
202     attrs_.set(Attr::AssumedShape);
203   }
204   if (IsDeferredShape(symbol)) {
205     attrs_.set(Attr::DeferredShape);
206   }
207   if (const auto *object{
208           symbol.GetUltimate().detailsIf<semantics::ObjectEntityDetails>()}) {
209     corank_ = object->coshape().Rank();
210     if (object->IsAssumedRank()) {
211       attrs_.set(Attr::AssumedRank);
212     }
213     if (object->IsAssumedSize()) {
214       attrs_.set(Attr::AssumedSize);
215     }
216     if (object->IsCoarray()) {
217       attrs_.set(Attr::Coarray);
218     }
219   }
220 }
221 
222 void TypeAndShape::AcquireLEN() {
223   if (auto len{type_.GetCharLength()}) {
224     LEN_ = std::move(len);
225   }
226 }
227 
228 void TypeAndShape::AcquireLEN(const semantics::Symbol &symbol) {
229   if (type_.category() == TypeCategory::Character) {
230     if (auto len{DataRef{symbol}.LEN()}) {
231       LEN_ = std::move(*len);
232     }
233   }
234 }
235 
236 std::string TypeAndShape::AsFortran() const {
237   return type_.AsFortran(LEN_ ? LEN_->AsFortran() : "");
238 }
239 
240 llvm::raw_ostream &TypeAndShape::Dump(llvm::raw_ostream &o) const {
241   o << type_.AsFortran(LEN_ ? LEN_->AsFortran() : "");
242   attrs_.Dump(o, EnumToString);
243   if (!shape_.empty()) {
244     o << " dimension";
245     char sep{'('};
246     for (const auto &expr : shape_) {
247       o << sep;
248       sep = ',';
249       if (expr) {
250         expr->AsFortran(o);
251       } else {
252         o << ':';
253       }
254     }
255     o << ')';
256   }
257   return o;
258 }
259 
260 bool DummyDataObject::operator==(const DummyDataObject &that) const {
261   return type == that.type && attrs == that.attrs && intent == that.intent &&
262       coshape == that.coshape;
263 }
264 
265 static bool AreCompatibleDummyDataObjectShapes(const Shape &x, const Shape &y) {
266   // TODO: Validate more than just compatible ranks
267   return GetRank(x) == GetRank(y);
268 }
269 
270 bool DummyDataObject::IsCompatibleWith(
271     const DummyDataObject &actual, std::string *whyNot) const {
272   if (!AreCompatibleDummyDataObjectShapes(type.shape(), actual.type.shape())) {
273     if (whyNot) {
274       *whyNot = "incompatible dummy data object shapes";
275     }
276     return false;
277   }
278   if (!type.type().IsTkLenCompatibleWith(actual.type.type())) {
279     if (whyNot) {
280       *whyNot = "incompatible dummy data object types: "s +
281           type.type().AsFortran() + " vs " + actual.type.type().AsFortran();
282     }
283     return false;
284   }
285   if (attrs != actual.attrs) {
286     if (whyNot) {
287       *whyNot = "incompatible dummy data object attributes";
288     }
289     return false;
290   }
291   if (intent != actual.intent) {
292     if (whyNot) {
293       *whyNot = "incompatible dummy data object intents";
294     }
295     return false;
296   }
297   if (coshape != actual.coshape) {
298     if (whyNot) {
299       *whyNot = "incompatible dummy data object coshapes";
300     }
301     return false;
302   }
303   return true;
304 }
305 
306 static common::Intent GetIntent(const semantics::Attrs &attrs) {
307   if (attrs.test(semantics::Attr::INTENT_IN)) {
308     return common::Intent::In;
309   } else if (attrs.test(semantics::Attr::INTENT_OUT)) {
310     return common::Intent::Out;
311   } else if (attrs.test(semantics::Attr::INTENT_INOUT)) {
312     return common::Intent::InOut;
313   } else {
314     return common::Intent::Default;
315   }
316 }
317 
318 std::optional<DummyDataObject> DummyDataObject::Characterize(
319     const semantics::Symbol &symbol, FoldingContext &context) {
320   if (symbol.has<semantics::ObjectEntityDetails>() ||
321       symbol.has<semantics::EntityDetails>()) {
322     if (auto type{TypeAndShape::Characterize(symbol, context)}) {
323       std::optional<DummyDataObject> result{std::move(*type)};
324       using semantics::Attr;
325       CopyAttrs<DummyDataObject, DummyDataObject::Attr>(symbol, *result,
326           {
327               {Attr::OPTIONAL, DummyDataObject::Attr::Optional},
328               {Attr::ALLOCATABLE, DummyDataObject::Attr::Allocatable},
329               {Attr::ASYNCHRONOUS, DummyDataObject::Attr::Asynchronous},
330               {Attr::CONTIGUOUS, DummyDataObject::Attr::Contiguous},
331               {Attr::VALUE, DummyDataObject::Attr::Value},
332               {Attr::VOLATILE, DummyDataObject::Attr::Volatile},
333               {Attr::POINTER, DummyDataObject::Attr::Pointer},
334               {Attr::TARGET, DummyDataObject::Attr::Target},
335           });
336       result->intent = GetIntent(symbol.attrs());
337       return result;
338     }
339   }
340   return std::nullopt;
341 }
342 
343 bool DummyDataObject::CanBePassedViaImplicitInterface() const {
344   if ((attrs &
345           Attrs{Attr::Allocatable, Attr::Asynchronous, Attr::Optional,
346               Attr::Pointer, Attr::Target, Attr::Value, Attr::Volatile})
347           .any()) {
348     return false; // 15.4.2.2(3)(a)
349   } else if ((type.attrs() &
350                  TypeAndShape::Attrs{TypeAndShape::Attr::AssumedShape,
351                      TypeAndShape::Attr::AssumedRank,
352                      TypeAndShape::Attr::Coarray})
353                  .any()) {
354     return false; // 15.4.2.2(3)(b-d)
355   } else if (type.type().IsPolymorphic()) {
356     return false; // 15.4.2.2(3)(f)
357   } else if (const auto *derived{GetDerivedTypeSpec(type.type())}) {
358     return derived->parameters().empty(); // 15.4.2.2(3)(e)
359   } else {
360     return true;
361   }
362 }
363 
364 llvm::raw_ostream &DummyDataObject::Dump(llvm::raw_ostream &o) const {
365   attrs.Dump(o, EnumToString);
366   if (intent != common::Intent::Default) {
367     o << "INTENT(" << common::EnumToString(intent) << ')';
368   }
369   type.Dump(o);
370   if (!coshape.empty()) {
371     char sep{'['};
372     for (const auto &expr : coshape) {
373       expr.AsFortran(o << sep);
374       sep = ',';
375     }
376   }
377   return o;
378 }
379 
380 DummyProcedure::DummyProcedure(Procedure &&p)
381     : procedure{new Procedure{std::move(p)}} {}
382 
383 bool DummyProcedure::operator==(const DummyProcedure &that) const {
384   return attrs == that.attrs && intent == that.intent &&
385       procedure.value() == that.procedure.value();
386 }
387 
388 bool DummyProcedure::IsCompatibleWith(
389     const DummyProcedure &actual, std::string *whyNot) const {
390   if (attrs != actual.attrs) {
391     if (whyNot) {
392       *whyNot = "incompatible dummy procedure attributes";
393     }
394     return false;
395   }
396   if (intent != actual.intent) {
397     if (whyNot) {
398       *whyNot = "incompatible dummy procedure intents";
399     }
400     return false;
401   }
402   if (!procedure.value().IsCompatibleWith(actual.procedure.value(), whyNot)) {
403     if (whyNot) {
404       *whyNot = "incompatible dummy procedure interfaces: "s + *whyNot;
405     }
406     return false;
407   }
408   return true;
409 }
410 
411 bool DummyProcedure::CanBePassedViaImplicitInterface() const {
412   if ((attrs & Attrs{Attr::Optional, Attr::Pointer}).any()) {
413     return false; // 15.4.2.2(3)(a)
414   }
415   return true;
416 }
417 
418 static std::string GetSeenProcs(
419     const semantics::UnorderedSymbolSet &seenProcs) {
420   // Sort the symbols so that they appear in the same order on all platforms
421   auto ordered{semantics::OrderBySourcePosition(seenProcs)};
422   std::string result;
423   llvm::interleave(
424       ordered,
425       [&](const SymbolRef p) { result += '\'' + p->name().ToString() + '\''; },
426       [&]() { result += ", "; });
427   return result;
428 }
429 
430 // These functions with arguments of type UnorderedSymbolSet are used with
431 // mutually recursive calls when characterizing a Procedure, a DummyArgument,
432 // or a DummyProcedure to detect circularly defined procedures as required by
433 // 15.4.3.6, paragraph 2.
434 static std::optional<DummyArgument> CharacterizeDummyArgument(
435     const semantics::Symbol &symbol, FoldingContext &context,
436     semantics::UnorderedSymbolSet seenProcs);
437 static std::optional<FunctionResult> CharacterizeFunctionResult(
438     const semantics::Symbol &symbol, FoldingContext &context,
439     semantics::UnorderedSymbolSet seenProcs);
440 
441 static std::optional<Procedure> CharacterizeProcedure(
442     const semantics::Symbol &original, FoldingContext &context,
443     semantics::UnorderedSymbolSet seenProcs) {
444   Procedure result;
445   const auto &symbol{ResolveAssociations(original)};
446   if (seenProcs.find(symbol) != seenProcs.end()) {
447     std::string procsList{GetSeenProcs(seenProcs)};
448     context.messages().Say(symbol.name(),
449         "Procedure '%s' is recursively defined.  Procedures in the cycle:"
450         " %s"_err_en_US,
451         symbol.name(), procsList);
452     return std::nullopt;
453   }
454   seenProcs.insert(symbol);
455   if (IsElementalProcedure(symbol)) {
456     result.attrs.set(Procedure::Attr::Elemental);
457   }
458   CopyAttrs<Procedure, Procedure::Attr>(symbol, result,
459       {
460           {semantics::Attr::BIND_C, Procedure::Attr::BindC},
461       });
462   if (IsPureProcedure(symbol) || // works for ENTRY too
463       (!symbol.attrs().test(semantics::Attr::IMPURE) &&
464           result.attrs.test(Procedure::Attr::Elemental))) {
465     result.attrs.set(Procedure::Attr::Pure);
466   }
467   return common::visit(
468       common::visitors{
469           [&](const semantics::SubprogramDetails &subp)
470               -> std::optional<Procedure> {
471             if (subp.isFunction()) {
472               if (auto fr{CharacterizeFunctionResult(
473                       subp.result(), context, seenProcs)}) {
474                 result.functionResult = std::move(fr);
475               } else {
476                 return std::nullopt;
477               }
478             } else {
479               result.attrs.set(Procedure::Attr::Subroutine);
480             }
481             for (const semantics::Symbol *arg : subp.dummyArgs()) {
482               if (!arg) {
483                 if (subp.isFunction()) {
484                   return std::nullopt;
485                 } else {
486                   result.dummyArguments.emplace_back(AlternateReturn{});
487                 }
488               } else if (auto argCharacteristics{CharacterizeDummyArgument(
489                              *arg, context, seenProcs)}) {
490                 result.dummyArguments.emplace_back(
491                     std::move(argCharacteristics.value()));
492               } else {
493                 return std::nullopt;
494               }
495             }
496             return result;
497           },
498           [&](const semantics::ProcEntityDetails &proc)
499               -> std::optional<Procedure> {
500             if (symbol.attrs().test(semantics::Attr::INTRINSIC)) {
501               // Fails when the intrinsic is not a specific intrinsic function
502               // from F'2018 table 16.2.  In order to handle forward references,
503               // attempts to use impermissible intrinsic procedures as the
504               // interfaces of procedure pointers are caught and flagged in
505               // declaration checking in Semantics.
506               auto intrinsic{context.intrinsics().IsSpecificIntrinsicFunction(
507                   symbol.name().ToString())};
508               if (intrinsic && intrinsic->isRestrictedSpecific) {
509                 intrinsic.reset(); // Exclude intrinsics from table 16.3.
510               }
511               return intrinsic;
512             }
513             if (const semantics::Symbol *
514                 interfaceSymbol{proc.procInterface()}) {
515               auto interface {
516                 CharacterizeProcedure(*interfaceSymbol, context, seenProcs)
517               };
518               if (interface && IsPointer(symbol)) {
519                 interface->attrs.reset(Procedure::Attr::Elemental);
520               }
521               return interface;
522             } else {
523               result.attrs.set(Procedure::Attr::ImplicitInterface);
524               const semantics::DeclTypeSpec *type{proc.type()};
525               if (symbol.test(semantics::Symbol::Flag::Subroutine)) {
526                 // ignore any implicit typing
527                 result.attrs.set(Procedure::Attr::Subroutine);
528               } else if (type) {
529                 if (auto resultType{DynamicType::From(*type)}) {
530                   result.functionResult = FunctionResult{*resultType};
531                 } else {
532                   return std::nullopt;
533                 }
534               } else if (symbol.test(semantics::Symbol::Flag::Function)) {
535                 return std::nullopt;
536               }
537               // The PASS name, if any, is not a characteristic.
538               return result;
539             }
540           },
541           [&](const semantics::ProcBindingDetails &binding) {
542             if (auto result{CharacterizeProcedure(
543                     binding.symbol(), context, seenProcs)}) {
544               if (binding.symbol().attrs().test(semantics::Attr::INTRINSIC)) {
545                 result->attrs.reset(Procedure::Attr::Elemental);
546               }
547               if (!symbol.attrs().test(semantics::Attr::NOPASS)) {
548                 auto passName{binding.passName()};
549                 for (auto &dummy : result->dummyArguments) {
550                   if (!passName || dummy.name.c_str() == *passName) {
551                     dummy.pass = true;
552                     break;
553                   }
554                 }
555               }
556               return result;
557             } else {
558               return std::optional<Procedure>{};
559             }
560           },
561           [&](const semantics::UseDetails &use) {
562             return CharacterizeProcedure(use.symbol(), context, seenProcs);
563           },
564           [](const semantics::UseErrorDetails &) {
565             // Ambiguous use-association will be handled later during symbol
566             // checks, ignore UseErrorDetails here without actual symbol usage.
567             return std::optional<Procedure>{};
568           },
569           [&](const semantics::HostAssocDetails &assoc) {
570             return CharacterizeProcedure(assoc.symbol(), context, seenProcs);
571           },
572           [&](const semantics::GenericDetails &generic) {
573             if (const semantics::Symbol * specific{generic.specific()}) {
574               return CharacterizeProcedure(*specific, context, seenProcs);
575             } else {
576               return std::optional<Procedure>{};
577             }
578           },
579           [&](const semantics::EntityDetails &) {
580             context.messages().Say(
581                 "Procedure '%s' is referenced before being sufficiently defined in a context where it must be so"_err_en_US,
582                 symbol.name());
583             return std::optional<Procedure>{};
584           },
585           [&](const semantics::SubprogramNameDetails &) {
586             context.messages().Say(
587                 "Procedure '%s' is referenced before being sufficiently defined in a context where it must be so"_err_en_US,
588                 symbol.name());
589             return std::optional<Procedure>{};
590           },
591           [&](const auto &) {
592             context.messages().Say(
593                 "'%s' is not a procedure"_err_en_US, symbol.name());
594             return std::optional<Procedure>{};
595           },
596       },
597       symbol.details());
598 }
599 
600 static std::optional<DummyProcedure> CharacterizeDummyProcedure(
601     const semantics::Symbol &symbol, FoldingContext &context,
602     semantics::UnorderedSymbolSet seenProcs) {
603   if (auto procedure{CharacterizeProcedure(symbol, context, seenProcs)}) {
604     // Dummy procedures may not be elemental.  Elemental dummy procedure
605     // interfaces are errors when the interface is not intrinsic, and that
606     // error is caught elsewhere.  Elemental intrinsic interfaces are
607     // made non-elemental.
608     procedure->attrs.reset(Procedure::Attr::Elemental);
609     DummyProcedure result{std::move(procedure.value())};
610     CopyAttrs<DummyProcedure, DummyProcedure::Attr>(symbol, result,
611         {
612             {semantics::Attr::OPTIONAL, DummyProcedure::Attr::Optional},
613             {semantics::Attr::POINTER, DummyProcedure::Attr::Pointer},
614         });
615     result.intent = GetIntent(symbol.attrs());
616     return result;
617   } else {
618     return std::nullopt;
619   }
620 }
621 
622 llvm::raw_ostream &DummyProcedure::Dump(llvm::raw_ostream &o) const {
623   attrs.Dump(o, EnumToString);
624   if (intent != common::Intent::Default) {
625     o << "INTENT(" << common::EnumToString(intent) << ')';
626   }
627   procedure.value().Dump(o);
628   return o;
629 }
630 
631 llvm::raw_ostream &AlternateReturn::Dump(llvm::raw_ostream &o) const {
632   return o << '*';
633 }
634 
635 DummyArgument::~DummyArgument() {}
636 
637 bool DummyArgument::operator==(const DummyArgument &that) const {
638   return u == that.u; // name and passed-object usage are not characteristics
639 }
640 
641 bool DummyArgument::IsCompatibleWith(
642     const DummyArgument &actual, std::string *whyNot) const {
643   if (const auto *ifaceData{std::get_if<DummyDataObject>(&u)}) {
644     if (const auto *actualData{std::get_if<DummyDataObject>(&actual.u)}) {
645       return ifaceData->IsCompatibleWith(*actualData, whyNot);
646     }
647     if (whyNot) {
648       *whyNot = "one dummy argument is an object, the other is not";
649     }
650   } else if (const auto *ifaceProc{std::get_if<DummyProcedure>(&u)}) {
651     if (const auto *actualProc{std::get_if<DummyProcedure>(&actual.u)}) {
652       return ifaceProc->IsCompatibleWith(*actualProc, whyNot);
653     }
654     if (whyNot) {
655       *whyNot = "one dummy argument is a procedure, the other is not";
656     }
657   } else {
658     CHECK(std::holds_alternative<AlternateReturn>(u));
659     if (std::holds_alternative<AlternateReturn>(actual.u)) {
660       return true;
661     }
662     if (whyNot) {
663       *whyNot = "one dummy argument is an alternate return, the other is not";
664     }
665   }
666   return false;
667 }
668 
669 static std::optional<DummyArgument> CharacterizeDummyArgument(
670     const semantics::Symbol &symbol, FoldingContext &context,
671     semantics::UnorderedSymbolSet seenProcs) {
672   auto name{symbol.name().ToString()};
673   if (symbol.has<semantics::ObjectEntityDetails>() ||
674       symbol.has<semantics::EntityDetails>()) {
675     if (auto obj{DummyDataObject::Characterize(symbol, context)}) {
676       return DummyArgument{std::move(name), std::move(obj.value())};
677     }
678   } else if (auto proc{
679                  CharacterizeDummyProcedure(symbol, context, seenProcs)}) {
680     return DummyArgument{std::move(name), std::move(proc.value())};
681   }
682   return std::nullopt;
683 }
684 
685 std::optional<DummyArgument> DummyArgument::FromActual(
686     std::string &&name, const Expr<SomeType> &expr, FoldingContext &context) {
687   return common::visit(
688       common::visitors{
689           [&](const BOZLiteralConstant &) {
690             return std::make_optional<DummyArgument>(std::move(name),
691                 DummyDataObject{
692                     TypeAndShape{DynamicType::TypelessIntrinsicArgument()}});
693           },
694           [&](const NullPointer &) {
695             return std::make_optional<DummyArgument>(std::move(name),
696                 DummyDataObject{
697                     TypeAndShape{DynamicType::TypelessIntrinsicArgument()}});
698           },
699           [&](const ProcedureDesignator &designator) {
700             if (auto proc{Procedure::Characterize(designator, context)}) {
701               return std::make_optional<DummyArgument>(
702                   std::move(name), DummyProcedure{std::move(*proc)});
703             } else {
704               return std::optional<DummyArgument>{};
705             }
706           },
707           [&](const ProcedureRef &call) {
708             if (auto proc{Procedure::Characterize(call, context)}) {
709               return std::make_optional<DummyArgument>(
710                   std::move(name), DummyProcedure{std::move(*proc)});
711             } else {
712               return std::optional<DummyArgument>{};
713             }
714           },
715           [&](const auto &) {
716             if (auto type{TypeAndShape::Characterize(expr, context)}) {
717               return std::make_optional<DummyArgument>(
718                   std::move(name), DummyDataObject{std::move(*type)});
719             } else {
720               return std::optional<DummyArgument>{};
721             }
722           },
723       },
724       expr.u);
725 }
726 
727 bool DummyArgument::IsOptional() const {
728   return common::visit(
729       common::visitors{
730           [](const DummyDataObject &data) {
731             return data.attrs.test(DummyDataObject::Attr::Optional);
732           },
733           [](const DummyProcedure &proc) {
734             return proc.attrs.test(DummyProcedure::Attr::Optional);
735           },
736           [](const AlternateReturn &) { return false; },
737       },
738       u);
739 }
740 
741 void DummyArgument::SetOptional(bool value) {
742   common::visit(common::visitors{
743                     [value](DummyDataObject &data) {
744                       data.attrs.set(DummyDataObject::Attr::Optional, value);
745                     },
746                     [value](DummyProcedure &proc) {
747                       proc.attrs.set(DummyProcedure::Attr::Optional, value);
748                     },
749                     [](AlternateReturn &) { DIE("cannot set optional"); },
750                 },
751       u);
752 }
753 
754 void DummyArgument::SetIntent(common::Intent intent) {
755   common::visit(common::visitors{
756                     [intent](DummyDataObject &data) { data.intent = intent; },
757                     [intent](DummyProcedure &proc) { proc.intent = intent; },
758                     [](AlternateReturn &) { DIE("cannot set intent"); },
759                 },
760       u);
761 }
762 
763 common::Intent DummyArgument::GetIntent() const {
764   return common::visit(
765       common::visitors{
766           [](const DummyDataObject &data) { return data.intent; },
767           [](const DummyProcedure &proc) { return proc.intent; },
768           [](const AlternateReturn &) -> common::Intent {
769             DIE("Alternate returns have no intent");
770           },
771       },
772       u);
773 }
774 
775 bool DummyArgument::CanBePassedViaImplicitInterface() const {
776   if (const auto *object{std::get_if<DummyDataObject>(&u)}) {
777     return object->CanBePassedViaImplicitInterface();
778   } else if (const auto *proc{std::get_if<DummyProcedure>(&u)}) {
779     return proc->CanBePassedViaImplicitInterface();
780   } else {
781     return true;
782   }
783 }
784 
785 bool DummyArgument::IsTypelessIntrinsicDummy() const {
786   const auto *argObj{std::get_if<characteristics::DummyDataObject>(&u)};
787   return argObj && argObj->type.type().IsTypelessIntrinsicArgument();
788 }
789 
790 llvm::raw_ostream &DummyArgument::Dump(llvm::raw_ostream &o) const {
791   if (!name.empty()) {
792     o << name << '=';
793   }
794   if (pass) {
795     o << " PASS";
796   }
797   common::visit([&](const auto &x) { x.Dump(o); }, u);
798   return o;
799 }
800 
801 FunctionResult::FunctionResult(DynamicType t) : u{TypeAndShape{t}} {}
802 FunctionResult::FunctionResult(TypeAndShape &&t) : u{std::move(t)} {}
803 FunctionResult::FunctionResult(Procedure &&p) : u{std::move(p)} {}
804 FunctionResult::~FunctionResult() {}
805 
806 bool FunctionResult::operator==(const FunctionResult &that) const {
807   return attrs == that.attrs && u == that.u;
808 }
809 
810 static std::optional<FunctionResult> CharacterizeFunctionResult(
811     const semantics::Symbol &symbol, FoldingContext &context,
812     semantics::UnorderedSymbolSet seenProcs) {
813   if (symbol.has<semantics::ObjectEntityDetails>()) {
814     if (auto type{TypeAndShape::Characterize(symbol, context)}) {
815       FunctionResult result{std::move(*type)};
816       CopyAttrs<FunctionResult, FunctionResult::Attr>(symbol, result,
817           {
818               {semantics::Attr::ALLOCATABLE, FunctionResult::Attr::Allocatable},
819               {semantics::Attr::CONTIGUOUS, FunctionResult::Attr::Contiguous},
820               {semantics::Attr::POINTER, FunctionResult::Attr::Pointer},
821           });
822       return result;
823     }
824   } else if (auto maybeProc{
825                  CharacterizeProcedure(symbol, context, seenProcs)}) {
826     FunctionResult result{std::move(*maybeProc)};
827     result.attrs.set(FunctionResult::Attr::Pointer);
828     return result;
829   }
830   return std::nullopt;
831 }
832 
833 std::optional<FunctionResult> FunctionResult::Characterize(
834     const Symbol &symbol, FoldingContext &context) {
835   semantics::UnorderedSymbolSet seenProcs;
836   return CharacterizeFunctionResult(symbol, context, seenProcs);
837 }
838 
839 bool FunctionResult::IsAssumedLengthCharacter() const {
840   if (const auto *ts{std::get_if<TypeAndShape>(&u)}) {
841     return ts->type().IsAssumedLengthCharacter();
842   } else {
843     return false;
844   }
845 }
846 
847 bool FunctionResult::CanBeReturnedViaImplicitInterface() const {
848   if (attrs.test(Attr::Pointer) || attrs.test(Attr::Allocatable)) {
849     return false; // 15.4.2.2(4)(b)
850   } else if (const auto *typeAndShape{GetTypeAndShape()}) {
851     if (typeAndShape->Rank() > 0) {
852       return false; // 15.4.2.2(4)(a)
853     } else {
854       const DynamicType &type{typeAndShape->type()};
855       switch (type.category()) {
856       case TypeCategory::Character:
857         if (type.knownLength()) {
858           return true;
859         } else if (const auto *param{type.charLengthParamValue()}) {
860           if (const auto &expr{param->GetExplicit()}) {
861             return IsConstantExpr(*expr); // 15.4.2.2(4)(c)
862           } else if (param->isAssumed()) {
863             return true;
864           }
865         }
866         return false;
867       case TypeCategory::Derived:
868         if (!type.IsPolymorphic()) {
869           const auto &spec{type.GetDerivedTypeSpec()};
870           for (const auto &pair : spec.parameters()) {
871             if (const auto &expr{pair.second.GetExplicit()}) {
872               if (!IsConstantExpr(*expr)) {
873                 return false; // 15.4.2.2(4)(c)
874               }
875             }
876           }
877           return true;
878         }
879         return false;
880       default:
881         return true;
882       }
883     }
884   } else {
885     return false; // 15.4.2.2(4)(b) - procedure pointer
886   }
887 }
888 
889 static bool AreCompatibleFunctionResultShapes(const Shape &x, const Shape &y) {
890   int rank{GetRank(x)};
891   if (GetRank(y) != rank) {
892     return false;
893   }
894   for (int j{0}; j < rank; ++j) {
895     if (auto xDim{ToInt64(x[j])}) {
896       if (auto yDim{ToInt64(y[j])}) {
897         if (*xDim != *yDim) {
898           return false;
899         }
900       }
901     }
902   }
903   return true;
904 }
905 
906 bool FunctionResult::IsCompatibleWith(
907     const FunctionResult &actual, std::string *whyNot) const {
908   Attrs actualAttrs{actual.attrs};
909   if (!attrs.test(Attr::Contiguous)) {
910     actualAttrs.reset(Attr::Contiguous);
911   }
912   if (attrs != actualAttrs) {
913     if (whyNot) {
914       *whyNot = "function results have incompatible attributes";
915     }
916   } else if (const auto *ifaceTypeShape{std::get_if<TypeAndShape>(&u)}) {
917     if (const auto *actualTypeShape{std::get_if<TypeAndShape>(&actual.u)}) {
918       if (ifaceTypeShape->Rank() != actualTypeShape->Rank()) {
919         if (whyNot) {
920           *whyNot = "function results have distinct ranks";
921         }
922       } else if (!attrs.test(Attr::Allocatable) && !attrs.test(Attr::Pointer) &&
923           !AreCompatibleFunctionResultShapes(
924               ifaceTypeShape->shape(), actualTypeShape->shape())) {
925         if (whyNot) {
926           *whyNot = "function results have distinct constant extents";
927         }
928       } else if (ifaceTypeShape->type() != actualTypeShape->type()) {
929         if (ifaceTypeShape->type().category() ==
930             actualTypeShape->type().category()) {
931           if (ifaceTypeShape->type().category() == TypeCategory::Character) {
932             if (ifaceTypeShape->type().kind() ==
933                 actualTypeShape->type().kind()) {
934               auto ifaceLen{ifaceTypeShape->type().knownLength()};
935               auto actualLen{actualTypeShape->type().knownLength()};
936               if (!ifaceLen || !actualLen || *ifaceLen == *actualLen) {
937                 return true;
938               }
939             }
940           } else if (ifaceTypeShape->type().category() ==
941               TypeCategory::Derived) {
942             if (ifaceTypeShape->type().IsPolymorphic() ==
943                     actualTypeShape->type().IsPolymorphic() &&
944                 !ifaceTypeShape->type().IsUnlimitedPolymorphic() &&
945                 !actualTypeShape->type().IsUnlimitedPolymorphic() &&
946                 AreSameDerivedType(ifaceTypeShape->type().GetDerivedTypeSpec(),
947                     actualTypeShape->type().GetDerivedTypeSpec())) {
948               return true;
949             }
950           }
951         }
952         if (whyNot) {
953           *whyNot = "function results have distinct types: "s +
954               ifaceTypeShape->type().AsFortran() + " vs "s +
955               actualTypeShape->type().AsFortran();
956         }
957       } else {
958         return true;
959       }
960     } else {
961       if (whyNot) {
962         *whyNot = "function result type and shape are not known";
963       }
964     }
965   } else {
966     const auto *ifaceProc{std::get_if<CopyableIndirection<Procedure>>(&u)};
967     CHECK(ifaceProc != nullptr);
968     if (const auto *actualProc{
969             std::get_if<CopyableIndirection<Procedure>>(&actual.u)}) {
970       if (ifaceProc->value().IsCompatibleWith(actualProc->value(), whyNot)) {
971         return true;
972       }
973       if (whyNot) {
974         *whyNot =
975             "function results are incompatible procedure pointers: "s + *whyNot;
976       }
977     } else {
978       if (whyNot) {
979         *whyNot =
980             "one function result is a procedure pointer, the other is not";
981       }
982     }
983   }
984   return false;
985 }
986 
987 llvm::raw_ostream &FunctionResult::Dump(llvm::raw_ostream &o) const {
988   attrs.Dump(o, EnumToString);
989   common::visit(common::visitors{
990                     [&](const TypeAndShape &ts) { ts.Dump(o); },
991                     [&](const CopyableIndirection<Procedure> &p) {
992                       p.value().Dump(o << " procedure(") << ')';
993                     },
994                 },
995       u);
996   return o;
997 }
998 
999 Procedure::Procedure(FunctionResult &&fr, DummyArguments &&args, Attrs a)
1000     : functionResult{std::move(fr)}, dummyArguments{std::move(args)}, attrs{a} {
1001 }
1002 Procedure::Procedure(DummyArguments &&args, Attrs a)
1003     : dummyArguments{std::move(args)}, attrs{a} {}
1004 Procedure::~Procedure() {}
1005 
1006 bool Procedure::operator==(const Procedure &that) const {
1007   return attrs == that.attrs && functionResult == that.functionResult &&
1008       dummyArguments == that.dummyArguments;
1009 }
1010 
1011 bool Procedure::IsCompatibleWith(const Procedure &actual, std::string *whyNot,
1012     const SpecificIntrinsic *specificIntrinsic) const {
1013   // 15.5.2.9(1): if dummy is not pure, actual need not be.
1014   // Ditto with elemental.
1015   Attrs actualAttrs{actual.attrs};
1016   if (!attrs.test(Attr::Pure)) {
1017     actualAttrs.reset(Attr::Pure);
1018   }
1019   if (!attrs.test(Attr::Elemental) && specificIntrinsic) {
1020     actualAttrs.reset(Attr::Elemental);
1021   }
1022   Attrs differences{attrs ^ actualAttrs};
1023   differences.reset(Attr::Subroutine); // dealt with specifically later
1024   if (!differences.empty()) {
1025     if (whyNot) {
1026       auto sep{": "s};
1027       *whyNot = "incompatible procedure attributes";
1028       differences.IterateOverMembers([&](Attr x) {
1029         *whyNot += sep + std::string{EnumToString(x)};
1030         sep = ", ";
1031       });
1032     }
1033   } else if ((IsFunction() && actual.IsSubroutine()) ||
1034       (IsSubroutine() && actual.IsFunction())) {
1035     if (whyNot) {
1036       *whyNot =
1037           "incompatible procedures: one is a function, the other a subroutine";
1038     }
1039   } else if (functionResult && actual.functionResult &&
1040       !functionResult->IsCompatibleWith(*actual.functionResult, whyNot)) {
1041   } else if (dummyArguments.size() != actual.dummyArguments.size()) {
1042     if (whyNot) {
1043       *whyNot = "distinct numbers of dummy arguments";
1044     }
1045   } else {
1046     for (std::size_t j{0}; j < dummyArguments.size(); ++j) {
1047       // Subtlety: the dummy/actual distinction must be reversed for this
1048       // compatibility test in order to correctly check extended vs.
1049       // base types.  Example:
1050       //   subroutine s1(base); subroutine s2(extended)
1051       //   procedure(s1), pointer :: p
1052       //   p => s2 ! an error, s2 is more restricted, can't handle "base"
1053       if (!actual.dummyArguments[j].IsCompatibleWith(
1054               dummyArguments[j], whyNot)) {
1055         if (whyNot) {
1056           *whyNot = "incompatible dummy argument #"s + std::to_string(j + 1) +
1057               ": "s + *whyNot;
1058         }
1059         return false;
1060       }
1061     }
1062     return true;
1063   }
1064   return false;
1065 }
1066 
1067 int Procedure::FindPassIndex(std::optional<parser::CharBlock> name) const {
1068   int argCount{static_cast<int>(dummyArguments.size())};
1069   int index{0};
1070   if (name) {
1071     while (index < argCount && *name != dummyArguments[index].name.c_str()) {
1072       ++index;
1073     }
1074   }
1075   CHECK(index < argCount);
1076   return index;
1077 }
1078 
1079 bool Procedure::CanOverride(
1080     const Procedure &that, std::optional<int> passIndex) const {
1081   // A pure procedure may override an impure one (7.5.7.3(2))
1082   if ((that.attrs.test(Attr::Pure) && !attrs.test(Attr::Pure)) ||
1083       that.attrs.test(Attr::Elemental) != attrs.test(Attr::Elemental) ||
1084       functionResult != that.functionResult) {
1085     return false;
1086   }
1087   int argCount{static_cast<int>(dummyArguments.size())};
1088   if (argCount != static_cast<int>(that.dummyArguments.size())) {
1089     return false;
1090   }
1091   for (int j{0}; j < argCount; ++j) {
1092     if ((!passIndex || j != *passIndex) &&
1093         dummyArguments[j] != that.dummyArguments[j]) {
1094       return false;
1095     }
1096   }
1097   return true;
1098 }
1099 
1100 std::optional<Procedure> Procedure::Characterize(
1101     const semantics::Symbol &original, FoldingContext &context) {
1102   semantics::UnorderedSymbolSet seenProcs;
1103   return CharacterizeProcedure(original, context, seenProcs);
1104 }
1105 
1106 std::optional<Procedure> Procedure::Characterize(
1107     const ProcedureDesignator &proc, FoldingContext &context) {
1108   if (const auto *symbol{proc.GetSymbol()}) {
1109     if (auto result{
1110             characteristics::Procedure::Characterize(*symbol, context)}) {
1111       return result;
1112     }
1113   } else if (const auto *intrinsic{proc.GetSpecificIntrinsic()}) {
1114     return intrinsic->characteristics.value();
1115   }
1116   return std::nullopt;
1117 }
1118 
1119 std::optional<Procedure> Procedure::Characterize(
1120     const ProcedureRef &ref, FoldingContext &context) {
1121   if (auto callee{Characterize(ref.proc(), context)}) {
1122     if (callee->functionResult) {
1123       if (const Procedure *
1124           proc{callee->functionResult->IsProcedurePointer()}) {
1125         return {*proc};
1126       }
1127     }
1128   }
1129   return std::nullopt;
1130 }
1131 
1132 bool Procedure::CanBeCalledViaImplicitInterface() const {
1133   // TODO: Pass back information on why we return false
1134   if (attrs.test(Attr::Elemental) || attrs.test(Attr::BindC)) {
1135     return false; // 15.4.2.2(5,6)
1136   } else if (IsFunction() &&
1137       !functionResult->CanBeReturnedViaImplicitInterface()) {
1138     return false;
1139   } else {
1140     for (const DummyArgument &arg : dummyArguments) {
1141       if (!arg.CanBePassedViaImplicitInterface()) {
1142         return false;
1143       }
1144     }
1145     return true;
1146   }
1147 }
1148 
1149 llvm::raw_ostream &Procedure::Dump(llvm::raw_ostream &o) const {
1150   attrs.Dump(o, EnumToString);
1151   if (functionResult) {
1152     functionResult->Dump(o << "TYPE(") << ") FUNCTION";
1153   } else if (attrs.test(Attr::Subroutine)) {
1154     o << "SUBROUTINE";
1155   } else {
1156     o << "EXTERNAL";
1157   }
1158   char sep{'('};
1159   for (const auto &dummy : dummyArguments) {
1160     dummy.Dump(o << sep);
1161     sep = ',';
1162   }
1163   return o << (sep == '(' ? "()" : ")");
1164 }
1165 
1166 // Utility class to determine if Procedures, etc. are distinguishable
1167 class DistinguishUtils {
1168 public:
1169   explicit DistinguishUtils(const common::LanguageFeatureControl &features)
1170       : features_{features} {}
1171 
1172   // Are these procedures distinguishable for a generic name?
1173   bool Distinguishable(const Procedure &, const Procedure &) const;
1174   // Are these procedures distinguishable for a generic operator or assignment?
1175   bool DistinguishableOpOrAssign(const Procedure &, const Procedure &) const;
1176 
1177 private:
1178   struct CountDummyProcedures {
1179     CountDummyProcedures(const DummyArguments &args) {
1180       for (const DummyArgument &arg : args) {
1181         if (std::holds_alternative<DummyProcedure>(arg.u)) {
1182           total += 1;
1183           notOptional += !arg.IsOptional();
1184         }
1185       }
1186     }
1187     int total{0};
1188     int notOptional{0};
1189   };
1190 
1191   bool Rule3Distinguishable(const Procedure &, const Procedure &) const;
1192   const DummyArgument *Rule1DistinguishingArg(
1193       const DummyArguments &, const DummyArguments &) const;
1194   int FindFirstToDistinguishByPosition(
1195       const DummyArguments &, const DummyArguments &) const;
1196   int FindLastToDistinguishByName(
1197       const DummyArguments &, const DummyArguments &) const;
1198   int CountCompatibleWith(const DummyArgument &, const DummyArguments &) const;
1199   int CountNotDistinguishableFrom(
1200       const DummyArgument &, const DummyArguments &) const;
1201   bool Distinguishable(const DummyArgument &, const DummyArgument &) const;
1202   bool Distinguishable(const DummyDataObject &, const DummyDataObject &) const;
1203   bool Distinguishable(const DummyProcedure &, const DummyProcedure &) const;
1204   bool Distinguishable(const FunctionResult &, const FunctionResult &) const;
1205   bool Distinguishable(const TypeAndShape &, const TypeAndShape &) const;
1206   bool IsTkrCompatible(const DummyArgument &, const DummyArgument &) const;
1207   bool IsTkrCompatible(const TypeAndShape &, const TypeAndShape &) const;
1208   const DummyArgument *GetAtEffectivePosition(
1209       const DummyArguments &, int) const;
1210   const DummyArgument *GetPassArg(const Procedure &) const;
1211 
1212   const common::LanguageFeatureControl &features_;
1213 };
1214 
1215 // Simpler distinguishability rules for operators and assignment
1216 bool DistinguishUtils::DistinguishableOpOrAssign(
1217     const Procedure &proc1, const Procedure &proc2) const {
1218   if ((proc1.IsFunction() && proc2.IsSubroutine()) ||
1219       (proc1.IsSubroutine() && proc2.IsFunction())) {
1220     return true;
1221   }
1222   auto &args1{proc1.dummyArguments};
1223   auto &args2{proc2.dummyArguments};
1224   if (args1.size() != args2.size()) {
1225     return true; // C1511: distinguishable based on number of arguments
1226   }
1227   for (std::size_t i{0}; i < args1.size(); ++i) {
1228     if (Distinguishable(args1[i], args2[i])) {
1229       return true; // C1511, C1512: distinguishable based on this arg
1230     }
1231   }
1232   return false;
1233 }
1234 
1235 bool DistinguishUtils::Distinguishable(
1236     const Procedure &proc1, const Procedure &proc2) const {
1237   if ((proc1.IsFunction() && proc2.IsSubroutine()) ||
1238       (proc1.IsSubroutine() && proc2.IsFunction())) {
1239     return true;
1240   }
1241   auto &args1{proc1.dummyArguments};
1242   auto &args2{proc2.dummyArguments};
1243   auto count1{CountDummyProcedures(args1)};
1244   auto count2{CountDummyProcedures(args2)};
1245   if (count1.notOptional > count2.total || count2.notOptional > count1.total) {
1246     return true; // distinguishable based on C1514 rule 2
1247   }
1248   if (Rule3Distinguishable(proc1, proc2)) {
1249     return true; // distinguishable based on C1514 rule 3
1250   }
1251   if (Rule1DistinguishingArg(args1, args2)) {
1252     return true; // distinguishable based on C1514 rule 1
1253   }
1254   int pos1{FindFirstToDistinguishByPosition(args1, args2)};
1255   int name1{FindLastToDistinguishByName(args1, args2)};
1256   if (pos1 >= 0 && pos1 <= name1) {
1257     return true; // distinguishable based on C1514 rule 4
1258   }
1259   int pos2{FindFirstToDistinguishByPosition(args2, args1)};
1260   int name2{FindLastToDistinguishByName(args2, args1)};
1261   if (pos2 >= 0 && pos2 <= name2) {
1262     return true; // distinguishable based on C1514 rule 4
1263   }
1264   return false;
1265 }
1266 
1267 // C1514 rule 3: Procedures are distinguishable if both have a passed-object
1268 // dummy argument and those are distinguishable.
1269 bool DistinguishUtils::Rule3Distinguishable(
1270     const Procedure &proc1, const Procedure &proc2) const {
1271   const DummyArgument *pass1{GetPassArg(proc1)};
1272   const DummyArgument *pass2{GetPassArg(proc2)};
1273   return pass1 && pass2 && Distinguishable(*pass1, *pass2);
1274 }
1275 
1276 // Find a non-passed-object dummy data object in one of the argument lists
1277 // that satisfies C1514 rule 1. I.e. x such that:
1278 // - m is the number of dummy data objects in one that are nonoptional,
1279 //   are not passed-object, that x is TKR compatible with
1280 // - n is the number of non-passed-object dummy data objects, in the other
1281 //   that are not distinguishable from x
1282 // - m is greater than n
1283 const DummyArgument *DistinguishUtils::Rule1DistinguishingArg(
1284     const DummyArguments &args1, const DummyArguments &args2) const {
1285   auto size1{args1.size()};
1286   auto size2{args2.size()};
1287   for (std::size_t i{0}; i < size1 + size2; ++i) {
1288     const DummyArgument &x{i < size1 ? args1[i] : args2[i - size1]};
1289     if (!x.pass && std::holds_alternative<DummyDataObject>(x.u)) {
1290       if (CountCompatibleWith(x, args1) >
1291               CountNotDistinguishableFrom(x, args2) ||
1292           CountCompatibleWith(x, args2) >
1293               CountNotDistinguishableFrom(x, args1)) {
1294         return &x;
1295       }
1296     }
1297   }
1298   return nullptr;
1299 }
1300 
1301 // Find the index of the first nonoptional non-passed-object dummy argument
1302 // in args1 at an effective position such that either:
1303 // - args2 has no dummy argument at that effective position
1304 // - the dummy argument at that position is distinguishable from it
1305 int DistinguishUtils::FindFirstToDistinguishByPosition(
1306     const DummyArguments &args1, const DummyArguments &args2) const {
1307   int effective{0}; // position of arg1 in list, ignoring passed arg
1308   for (std::size_t i{0}; i < args1.size(); ++i) {
1309     const DummyArgument &arg1{args1.at(i)};
1310     if (!arg1.pass && !arg1.IsOptional()) {
1311       const DummyArgument *arg2{GetAtEffectivePosition(args2, effective)};
1312       if (!arg2 || Distinguishable(arg1, *arg2)) {
1313         return i;
1314       }
1315     }
1316     effective += !arg1.pass;
1317   }
1318   return -1;
1319 }
1320 
1321 // Find the index of the last nonoptional non-passed-object dummy argument
1322 // in args1 whose name is such that either:
1323 // - args2 has no dummy argument with that name
1324 // - the dummy argument with that name is distinguishable from it
1325 int DistinguishUtils::FindLastToDistinguishByName(
1326     const DummyArguments &args1, const DummyArguments &args2) const {
1327   std::map<std::string, const DummyArgument *> nameToArg;
1328   for (const auto &arg2 : args2) {
1329     nameToArg.emplace(arg2.name, &arg2);
1330   }
1331   for (int i = args1.size() - 1; i >= 0; --i) {
1332     const DummyArgument &arg1{args1.at(i)};
1333     if (!arg1.pass && !arg1.IsOptional()) {
1334       auto it{nameToArg.find(arg1.name)};
1335       if (it == nameToArg.end() || Distinguishable(arg1, *it->second)) {
1336         return i;
1337       }
1338     }
1339   }
1340   return -1;
1341 }
1342 
1343 // Count the dummy data objects in args that are nonoptional, are not
1344 // passed-object, and that x is TKR compatible with
1345 int DistinguishUtils::CountCompatibleWith(
1346     const DummyArgument &x, const DummyArguments &args) const {
1347   return llvm::count_if(args, [&](const DummyArgument &y) {
1348     return !y.pass && !y.IsOptional() && IsTkrCompatible(x, y);
1349   });
1350 }
1351 
1352 // Return the number of dummy data objects in args that are not
1353 // distinguishable from x and not passed-object.
1354 int DistinguishUtils::CountNotDistinguishableFrom(
1355     const DummyArgument &x, const DummyArguments &args) const {
1356   return llvm::count_if(args, [&](const DummyArgument &y) {
1357     return !y.pass && std::holds_alternative<DummyDataObject>(y.u) &&
1358         !Distinguishable(y, x);
1359   });
1360 }
1361 
1362 bool DistinguishUtils::Distinguishable(
1363     const DummyArgument &x, const DummyArgument &y) const {
1364   if (x.u.index() != y.u.index()) {
1365     return true; // different kind: data/proc/alt-return
1366   }
1367   return common::visit(
1368       common::visitors{
1369           [&](const DummyDataObject &z) {
1370             return Distinguishable(z, std::get<DummyDataObject>(y.u));
1371           },
1372           [&](const DummyProcedure &z) {
1373             return Distinguishable(z, std::get<DummyProcedure>(y.u));
1374           },
1375           [&](const AlternateReturn &) { return false; },
1376       },
1377       x.u);
1378 }
1379 
1380 bool DistinguishUtils::Distinguishable(
1381     const DummyDataObject &x, const DummyDataObject &y) const {
1382   using Attr = DummyDataObject::Attr;
1383   if (Distinguishable(x.type, y.type)) {
1384     return true;
1385   } else if (x.attrs.test(Attr::Allocatable) && y.attrs.test(Attr::Pointer) &&
1386       y.intent != common::Intent::In) {
1387     return true;
1388   } else if (y.attrs.test(Attr::Allocatable) && x.attrs.test(Attr::Pointer) &&
1389       x.intent != common::Intent::In) {
1390     return true;
1391   } else if (features_.IsEnabled(
1392                  common::LanguageFeature::DistinguishableSpecifics) &&
1393       (x.attrs.test(Attr::Allocatable) || x.attrs.test(Attr::Pointer)) &&
1394       (y.attrs.test(Attr::Allocatable) || y.attrs.test(Attr::Pointer)) &&
1395       (x.type.type().IsUnlimitedPolymorphic() !=
1396               y.type.type().IsUnlimitedPolymorphic() ||
1397           x.type.type().IsPolymorphic() != y.type.type().IsPolymorphic())) {
1398     // Extension: Per 15.5.2.5(2), an allocatable/pointer dummy and its
1399     // corresponding actual argument must both or neither be polymorphic,
1400     // and must both or neither be unlimited polymorphic.  So when exactly
1401     // one of two dummy arguments is polymorphic or unlimited polymorphic,
1402     // any actual argument that is admissible to one of them cannot also match
1403     // the other one.
1404     return true;
1405   } else {
1406     return false;
1407   }
1408 }
1409 
1410 bool DistinguishUtils::Distinguishable(
1411     const DummyProcedure &x, const DummyProcedure &y) const {
1412   const Procedure &xProc{x.procedure.value()};
1413   const Procedure &yProc{y.procedure.value()};
1414   if (Distinguishable(xProc, yProc)) {
1415     return true;
1416   } else {
1417     const std::optional<FunctionResult> &xResult{xProc.functionResult};
1418     const std::optional<FunctionResult> &yResult{yProc.functionResult};
1419     return xResult ? !yResult || Distinguishable(*xResult, *yResult)
1420                    : yResult.has_value();
1421   }
1422 }
1423 
1424 bool DistinguishUtils::Distinguishable(
1425     const FunctionResult &x, const FunctionResult &y) const {
1426   if (x.u.index() != y.u.index()) {
1427     return true; // one is data object, one is procedure
1428   }
1429   return common::visit(
1430       common::visitors{
1431           [&](const TypeAndShape &z) {
1432             return Distinguishable(z, std::get<TypeAndShape>(y.u));
1433           },
1434           [&](const CopyableIndirection<Procedure> &z) {
1435             return Distinguishable(z.value(),
1436                 std::get<CopyableIndirection<Procedure>>(y.u).value());
1437           },
1438       },
1439       x.u);
1440 }
1441 
1442 bool DistinguishUtils::Distinguishable(
1443     const TypeAndShape &x, const TypeAndShape &y) const {
1444   return !IsTkrCompatible(x, y) && !IsTkrCompatible(y, x);
1445 }
1446 
1447 // Compatibility based on type, kind, and rank
1448 bool DistinguishUtils::IsTkrCompatible(
1449     const DummyArgument &x, const DummyArgument &y) const {
1450   const auto *obj1{std::get_if<DummyDataObject>(&x.u)};
1451   const auto *obj2{std::get_if<DummyDataObject>(&y.u)};
1452   return obj1 && obj2 && IsTkrCompatible(obj1->type, obj2->type);
1453 }
1454 bool DistinguishUtils::IsTkrCompatible(
1455     const TypeAndShape &x, const TypeAndShape &y) const {
1456   return x.type().IsTkCompatibleWith(y.type()) &&
1457       (x.attrs().test(TypeAndShape::Attr::AssumedRank) ||
1458           y.attrs().test(TypeAndShape::Attr::AssumedRank) ||
1459           x.Rank() == y.Rank());
1460 }
1461 
1462 // Return the argument at the given index, ignoring the passed arg
1463 const DummyArgument *DistinguishUtils::GetAtEffectivePosition(
1464     const DummyArguments &args, int index) const {
1465   for (const DummyArgument &arg : args) {
1466     if (!arg.pass) {
1467       if (index == 0) {
1468         return &arg;
1469       }
1470       --index;
1471     }
1472   }
1473   return nullptr;
1474 }
1475 
1476 // Return the passed-object dummy argument of this procedure, if any
1477 const DummyArgument *DistinguishUtils::GetPassArg(const Procedure &proc) const {
1478   for (const auto &arg : proc.dummyArguments) {
1479     if (arg.pass) {
1480       return &arg;
1481     }
1482   }
1483   return nullptr;
1484 }
1485 
1486 bool Distinguishable(const common::LanguageFeatureControl &features,
1487     const Procedure &x, const Procedure &y) {
1488   return DistinguishUtils{features}.Distinguishable(x, y);
1489 }
1490 
1491 bool DistinguishableOpOrAssign(const common::LanguageFeatureControl &features,
1492     const Procedure &x, const Procedure &y) {
1493   return DistinguishUtils{features}.DistinguishableOpOrAssign(x, y);
1494 }
1495 
1496 DEFINE_DEFAULT_CONSTRUCTORS_AND_ASSIGNMENTS(DummyArgument)
1497 DEFINE_DEFAULT_CONSTRUCTORS_AND_ASSIGNMENTS(DummyProcedure)
1498 DEFINE_DEFAULT_CONSTRUCTORS_AND_ASSIGNMENTS(FunctionResult)
1499 DEFINE_DEFAULT_CONSTRUCTORS_AND_ASSIGNMENTS(Procedure)
1500 } // namespace Fortran::evaluate::characteristics
1501 
1502 template class Fortran::common::Indirection<
1503     Fortran::evaluate::characteristics::Procedure, true>;
1504