@@ -24,6 +24,7 @@ namespace Fortran::semantics {
24
24
using evaluate::characteristics::DummyArgument;
25
25
using evaluate::characteristics::DummyDataObject;
26
26
using evaluate::characteristics::DummyProcedure;
27
+ using evaluate::characteristics::FunctionResult;
27
28
using evaluate::characteristics::Procedure;
28
29
29
30
class CheckHelper {
@@ -109,6 +110,7 @@ class CheckHelper {
109
110
}
110
111
}
111
112
}
113
+ bool IsResultOkToDiffer (const FunctionResult &);
112
114
113
115
SemanticsContext &context_;
114
116
evaluate::FoldingContext &foldingContext_{context_.foldingContext ()};
@@ -208,7 +210,8 @@ void CheckHelper::Check(const Symbol &symbol) {
208
210
}
209
211
if (type) { // Section 7.2, paragraph 7
210
212
bool canHaveAssumedParameter{IsNamedConstant (symbol) ||
211
- IsAssumedLengthExternalCharacterFunction (symbol) || // C722
213
+ (IsAssumedLengthCharacter (symbol) && // C722
214
+ IsExternal (symbol)) ||
212
215
symbol.test (Symbol::Flag::ParentComp)};
213
216
if (!IsStmtFunctionDummy (symbol)) { // C726
214
217
if (const auto *object{symbol.detailsIf <ObjectEntityDetails>()}) {
@@ -239,7 +242,7 @@ void CheckHelper::Check(const Symbol &symbol) {
239
242
}
240
243
}
241
244
}
242
- if (IsAssumedLengthExternalCharacterFunction (symbol)) { // C723
245
+ if (IsAssumedLengthCharacter (symbol) && IsExternal (symbol)) { // C723
243
246
if (symbol.attrs ().test (Attr::RECURSIVE)) {
244
247
messages_.Say (
245
248
" An assumed-length CHARACTER(*) function cannot be RECURSIVE" _err_en_US);
@@ -270,6 +273,16 @@ void CheckHelper::Check(const Symbol &symbol) {
270
273
symbol.Rank () == 0 ) { // C830
271
274
messages_.Say (" CONTIGUOUS POINTER must be an array" _err_en_US);
272
275
}
276
+ if (IsDummy (symbol)) {
277
+ if (IsNamedConstant (symbol)) {
278
+ messages_.Say (
279
+ " A dummy argument may not also be a named constant" _err_en_US);
280
+ }
281
+ if (IsSaved (symbol)) {
282
+ messages_.Say (
283
+ " A dummy argument may not have the SAVE attribute" _err_en_US);
284
+ }
285
+ }
273
286
}
274
287
275
288
void CheckHelper::CheckValue (
@@ -600,12 +613,66 @@ class SubprogramMatchHelper {
600
613
SemanticsContext &context;
601
614
};
602
615
616
+ // 15.6.2.6 para 3 - can the result of an ENTRY differ from its function?
617
+ bool CheckHelper::IsResultOkToDiffer (const FunctionResult &result) {
618
+ if (result.attrs .test (FunctionResult::Attr::Allocatable) ||
619
+ result.attrs .test (FunctionResult::Attr::Pointer)) {
620
+ return false ;
621
+ }
622
+ const auto *typeAndShape{result.GetTypeAndShape ()};
623
+ if (!typeAndShape || typeAndShape->Rank () != 0 ) {
624
+ return false ;
625
+ }
626
+ auto category{typeAndShape->type ().category ()};
627
+ if (category == TypeCategory::Character ||
628
+ category == TypeCategory::Derived) {
629
+ return false ;
630
+ }
631
+ int kind{typeAndShape->type ().kind ()};
632
+ return kind == context_.GetDefaultKind (category) ||
633
+ (category == TypeCategory::Real &&
634
+ kind == context_.doublePrecisionKind ());
635
+ }
636
+
603
637
void CheckHelper::CheckSubprogram (
604
- const Symbol &symbol, const SubprogramDetails &) {
605
- const Scope &scope{symbol.owner ()};
606
- if (symbol.attrs ().test (Attr::MODULE) && scope.IsSubmodule ()) {
607
- if (const Symbol * iface{scope.parent ().FindSymbol (symbol.name ())}) {
608
- SubprogramMatchHelper{context_}.Check (symbol, *iface);
638
+ const Symbol &symbol, const SubprogramDetails &details) {
639
+ if (const Symbol * iface{FindSeparateModuleSubprogramInterface (&symbol)}) {
640
+ SubprogramMatchHelper{context_}.Check (symbol, *iface);
641
+ }
642
+ if (const Scope * entryScope{details.entryScope ()}) {
643
+ // ENTRY 15.6.2.6, esp. C1571
644
+ std::optional<parser::MessageFixedText> error;
645
+ const Symbol *subprogram{entryScope->symbol ()};
646
+ const SubprogramDetails *subprogramDetails{nullptr };
647
+ if (subprogram) {
648
+ subprogramDetails = subprogram->detailsIf <SubprogramDetails>();
649
+ }
650
+ if (entryScope->kind () != Scope::Kind::Subprogram) {
651
+ error = " ENTRY may appear only in a subroutine or function" _err_en_US;
652
+ } else if (!(entryScope->parent ().IsGlobal () ||
653
+ entryScope->parent ().IsModule () ||
654
+ entryScope->parent ().IsSubmodule ())) {
655
+ error = " ENTRY may not appear in an internal subprogram" _err_en_US;
656
+ } else if (FindSeparateModuleSubprogramInterface (subprogram)) {
657
+ error = " ENTRY may not appear in a separate module procedure" _err_en_US;
658
+ } else if (subprogramDetails && details.isFunction () &&
659
+ subprogramDetails->isFunction ()) {
660
+ auto result{FunctionResult::Characterize (
661
+ details.result (), context_.intrinsics ())};
662
+ auto subpResult{FunctionResult::Characterize (
663
+ subprogramDetails->result (), context_.intrinsics ())};
664
+ if (result && subpResult && *result != *subpResult &&
665
+ (!IsResultOkToDiffer (*result) || !IsResultOkToDiffer (*subpResult))) {
666
+ error =
667
+ " Result of ENTRY is not compatible with result of containing function" _err_en_US;
668
+ }
669
+ }
670
+ if (error) {
671
+ if (auto *msg{messages_.Say (symbol.name (), *error)}) {
672
+ if (subprogram) {
673
+ msg->Attach (subprogram->name (), " Containing subprogram" _en_US);
674
+ }
675
+ }
609
676
}
610
677
}
611
678
}
0 commit comments