1 //===-- lib/Semantics/check-io.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 "check-io.h" 10 #include "definable.h" 11 #include "flang/Common/format.h" 12 #include "flang/Evaluate/tools.h" 13 #include "flang/Parser/tools.h" 14 #include "flang/Semantics/expression.h" 15 #include "flang/Semantics/tools.h" 16 #include <unordered_map> 17 18 namespace Fortran::semantics { 19 20 // TODO: C1234, C1235 -- defined I/O constraints 21 22 class FormatErrorReporter { 23 public: 24 FormatErrorReporter(SemanticsContext &context, 25 const parser::CharBlock &formatCharBlock, int errorAllowance = 3) 26 : context_{context}, formatCharBlock_{formatCharBlock}, 27 errorAllowance_{errorAllowance} {} 28 29 bool Say(const common::FormatMessage &); 30 31 private: 32 SemanticsContext &context_; 33 const parser::CharBlock &formatCharBlock_; 34 int errorAllowance_; // initialized to maximum number of errors to report 35 }; 36 37 bool FormatErrorReporter::Say(const common::FormatMessage &msg) { 38 if (!msg.isError && !context_.warnOnNonstandardUsage()) { 39 return false; 40 } 41 parser::MessageFormattedText text{ 42 parser::MessageFixedText{msg.text, strlen(msg.text), 43 msg.isError ? parser::Severity::Error : parser::Severity::Warning}, 44 msg.arg}; 45 if (formatCharBlock_.size()) { 46 // The input format is a folded expression. Error markers span the full 47 // original unfolded expression in formatCharBlock_. 48 context_.Say(formatCharBlock_, text); 49 } else { 50 // The input format is a source expression. Error markers have an offset 51 // and length relative to the beginning of formatCharBlock_. 52 parser::CharBlock messageCharBlock{ 53 parser::CharBlock(formatCharBlock_.begin() + msg.offset, msg.length)}; 54 context_.Say(messageCharBlock, text); 55 } 56 return msg.isError && --errorAllowance_ <= 0; 57 } 58 59 void IoChecker::Enter( 60 const parser::Statement<common::Indirection<parser::FormatStmt>> &stmt) { 61 if (!stmt.label) { 62 context_.Say("Format statement must be labeled"_err_en_US); // C1301 63 } 64 const char *formatStart{static_cast<const char *>( 65 std::memchr(stmt.source.begin(), '(', stmt.source.size()))}; 66 parser::CharBlock reporterCharBlock{formatStart, static_cast<std::size_t>(0)}; 67 FormatErrorReporter reporter{context_, reporterCharBlock}; 68 auto reporterWrapper{[&](const auto &msg) { return reporter.Say(msg); }}; 69 switch (context_.GetDefaultKind(TypeCategory::Character)) { 70 case 1: { 71 common::FormatValidator<char> validator{formatStart, 72 stmt.source.size() - (formatStart - stmt.source.begin()), 73 reporterWrapper}; 74 validator.Check(); 75 break; 76 } 77 case 2: { // TODO: Get this to work. 78 common::FormatValidator<char16_t> validator{ 79 /*???*/ nullptr, /*???*/ 0, reporterWrapper}; 80 validator.Check(); 81 break; 82 } 83 case 4: { // TODO: Get this to work. 84 common::FormatValidator<char32_t> validator{ 85 /*???*/ nullptr, /*???*/ 0, reporterWrapper}; 86 validator.Check(); 87 break; 88 } 89 default: 90 CRASH_NO_CASE; 91 } 92 } 93 94 void IoChecker::Enter(const parser::ConnectSpec &spec) { 95 // ConnectSpec context FileNameExpr 96 if (std::get_if<parser::FileNameExpr>(&spec.u)) { 97 SetSpecifier(IoSpecKind::File); 98 } 99 } 100 101 // Ignore trailing spaces (12.5.6.2 p1) and convert to upper case 102 static std::string Normalize(const std::string &value) { 103 auto upper{parser::ToUpperCaseLetters(value)}; 104 std::size_t lastNonBlank{upper.find_last_not_of(" ")}; 105 upper.resize(lastNonBlank == std::string::npos ? 0 : lastNonBlank + 1); 106 return upper; 107 } 108 109 void IoChecker::Enter(const parser::ConnectSpec::CharExpr &spec) { 110 IoSpecKind specKind{}; 111 using ParseKind = parser::ConnectSpec::CharExpr::Kind; 112 switch (std::get<ParseKind>(spec.t)) { 113 case ParseKind::Access: 114 specKind = IoSpecKind::Access; 115 break; 116 case ParseKind::Action: 117 specKind = IoSpecKind::Action; 118 break; 119 case ParseKind::Asynchronous: 120 specKind = IoSpecKind::Asynchronous; 121 break; 122 case ParseKind::Blank: 123 specKind = IoSpecKind::Blank; 124 break; 125 case ParseKind::Decimal: 126 specKind = IoSpecKind::Decimal; 127 break; 128 case ParseKind::Delim: 129 specKind = IoSpecKind::Delim; 130 break; 131 case ParseKind::Encoding: 132 specKind = IoSpecKind::Encoding; 133 break; 134 case ParseKind::Form: 135 specKind = IoSpecKind::Form; 136 break; 137 case ParseKind::Pad: 138 specKind = IoSpecKind::Pad; 139 break; 140 case ParseKind::Position: 141 specKind = IoSpecKind::Position; 142 break; 143 case ParseKind::Round: 144 specKind = IoSpecKind::Round; 145 break; 146 case ParseKind::Sign: 147 specKind = IoSpecKind::Sign; 148 break; 149 case ParseKind::Carriagecontrol: 150 specKind = IoSpecKind::Carriagecontrol; 151 break; 152 case ParseKind::Convert: 153 specKind = IoSpecKind::Convert; 154 break; 155 case ParseKind::Dispose: 156 specKind = IoSpecKind::Dispose; 157 break; 158 } 159 SetSpecifier(specKind); 160 if (const std::optional<std::string> charConst{GetConstExpr<std::string>( 161 std::get<parser::ScalarDefaultCharExpr>(spec.t))}) { 162 std::string s{Normalize(*charConst)}; 163 if (specKind == IoSpecKind::Access) { 164 flags_.set(Flag::KnownAccess); 165 flags_.set(Flag::AccessDirect, s == "DIRECT"); 166 flags_.set(Flag::AccessStream, s == "STREAM"); 167 } 168 CheckStringValue(specKind, *charConst, parser::FindSourceLocation(spec)); 169 if (specKind == IoSpecKind::Carriagecontrol && 170 (s == "FORTRAN" || s == "NONE")) { 171 context_.Say(parser::FindSourceLocation(spec), 172 "Unimplemented %s value '%s'"_err_en_US, 173 parser::ToUpperCaseLetters(common::EnumToString(specKind)), 174 *charConst); 175 } 176 } 177 } 178 179 void IoChecker::Enter(const parser::ConnectSpec::Newunit &var) { 180 CheckForDefinableVariable(var, "NEWUNIT"); 181 SetSpecifier(IoSpecKind::Newunit); 182 } 183 184 void IoChecker::Enter(const parser::ConnectSpec::Recl &spec) { 185 SetSpecifier(IoSpecKind::Recl); 186 if (const std::optional<std::int64_t> recl{ 187 GetConstExpr<std::int64_t>(spec)}) { 188 if (*recl <= 0) { 189 context_.Say(parser::FindSourceLocation(spec), 190 "RECL value (%jd) must be positive"_err_en_US, 191 *recl); // 12.5.6.15 192 } 193 } 194 } 195 196 void IoChecker::Enter(const parser::EndLabel &) { 197 SetSpecifier(IoSpecKind::End); 198 } 199 200 void IoChecker::Enter(const parser::EorLabel &) { 201 SetSpecifier(IoSpecKind::Eor); 202 } 203 204 void IoChecker::Enter(const parser::ErrLabel &) { 205 SetSpecifier(IoSpecKind::Err); 206 } 207 208 void IoChecker::Enter(const parser::FileUnitNumber &) { 209 SetSpecifier(IoSpecKind::Unit); 210 flags_.set(Flag::NumberUnit); 211 } 212 213 void IoChecker::Enter(const parser::Format &spec) { 214 SetSpecifier(IoSpecKind::Fmt); 215 flags_.set(Flag::FmtOrNml); 216 common::visit( 217 common::visitors{ 218 [&](const parser::Label &) { flags_.set(Flag::LabelFmt); }, 219 [&](const parser::Star &) { flags_.set(Flag::StarFmt); }, 220 [&](const parser::Expr &format) { 221 const SomeExpr *expr{GetExpr(context_, format)}; 222 if (!expr) { 223 return; 224 } 225 auto type{expr->GetType()}; 226 if (type && type->category() == TypeCategory::Integer && 227 type->kind() == 228 context_.defaultKinds().GetDefaultKind(type->category()) && 229 expr->Rank() == 0) { 230 flags_.set(Flag::AssignFmt); 231 if (!IsVariable(*expr)) { 232 context_.Say(format.source, 233 "Assigned format label must be a scalar variable"_err_en_US); 234 } else if (context_.ShouldWarn(common::LanguageFeature::Assign)) { 235 context_.Say(format.source, 236 "Assigned format labels are deprecated"_port_en_US); 237 } 238 return; 239 } 240 if (type && type->category() != TypeCategory::Character && 241 (type->category() != TypeCategory::Integer || 242 expr->Rank() > 0) && 243 context_.IsEnabled( 244 common::LanguageFeature::NonCharacterFormat)) { 245 // Legacy extension: using non-character variables, typically 246 // DATA-initialized with Hollerith, as format expressions. 247 if (context_.ShouldWarn( 248 common::LanguageFeature::NonCharacterFormat)) { 249 context_.Say(format.source, 250 "Non-character format expression is not standard"_port_en_US); 251 } 252 } else if (!type || 253 type->kind() != 254 context_.defaultKinds().GetDefaultKind(type->category())) { 255 context_.Say(format.source, 256 "Format expression must be default character or default scalar integer"_err_en_US); 257 return; 258 } 259 flags_.set(Flag::CharFmt); 260 const std::optional<std::string> constantFormat{ 261 GetConstExpr<std::string>(format)}; 262 if (!constantFormat) { 263 return; 264 } 265 // validate constant format -- 12.6.2.2 266 bool isFolded{constantFormat->size() != format.source.size() - 2}; 267 parser::CharBlock reporterCharBlock{isFolded 268 ? parser::CharBlock{format.source} 269 : parser::CharBlock{format.source.begin() + 1, 270 static_cast<std::size_t>(0)}}; 271 FormatErrorReporter reporter{context_, reporterCharBlock}; 272 auto reporterWrapper{ 273 [&](const auto &msg) { return reporter.Say(msg); }}; 274 switch (context_.GetDefaultKind(TypeCategory::Character)) { 275 case 1: { 276 common::FormatValidator<char> validator{constantFormat->c_str(), 277 constantFormat->length(), reporterWrapper, stmt_}; 278 validator.Check(); 279 break; 280 } 281 case 2: { 282 // TODO: Get this to work. (Maybe combine with earlier instance?) 283 common::FormatValidator<char16_t> validator{ 284 /*???*/ nullptr, /*???*/ 0, reporterWrapper, stmt_}; 285 validator.Check(); 286 break; 287 } 288 case 4: { 289 // TODO: Get this to work. (Maybe combine with earlier instance?) 290 common::FormatValidator<char32_t> validator{ 291 /*???*/ nullptr, /*???*/ 0, reporterWrapper, stmt_}; 292 validator.Check(); 293 break; 294 } 295 default: 296 CRASH_NO_CASE; 297 } 298 }, 299 }, 300 spec.u); 301 } 302 303 void IoChecker::Enter(const parser::IdExpr &) { SetSpecifier(IoSpecKind::Id); } 304 305 void IoChecker::Enter(const parser::IdVariable &spec) { 306 SetSpecifier(IoSpecKind::Id); 307 const auto *expr{GetExpr(context_, spec)}; 308 if (!expr || !expr->GetType()) { 309 return; 310 } 311 CheckForDefinableVariable(spec, "ID"); 312 int kind{expr->GetType()->kind()}; 313 int defaultKind{context_.GetDefaultKind(TypeCategory::Integer)}; 314 if (kind < defaultKind) { 315 context_.Say( 316 "ID kind (%d) is smaller than default INTEGER kind (%d)"_err_en_US, 317 std::move(kind), std::move(defaultKind)); // C1229 318 } 319 } 320 321 void IoChecker::Enter(const parser::InputItem &spec) { 322 flags_.set(Flag::DataList); 323 const parser::Variable *var{std::get_if<parser::Variable>(&spec.u)}; 324 if (!var) { 325 return; 326 } 327 CheckForDefinableVariable(*var, "Input"); 328 if (auto expr{AnalyzeExpr(context_, *var)}) { 329 CheckForBadIoType(*expr, 330 flags_.test(Flag::FmtOrNml) ? GenericKind::DefinedIo::ReadFormatted 331 : GenericKind::DefinedIo::ReadUnformatted, 332 var->GetSource()); 333 } 334 } 335 336 void IoChecker::Enter(const parser::InquireSpec &spec) { 337 // InquireSpec context FileNameExpr 338 if (std::get_if<parser::FileNameExpr>(&spec.u)) { 339 SetSpecifier(IoSpecKind::File); 340 } 341 } 342 343 void IoChecker::Enter(const parser::InquireSpec::CharVar &spec) { 344 IoSpecKind specKind{}; 345 using ParseKind = parser::InquireSpec::CharVar::Kind; 346 switch (std::get<ParseKind>(spec.t)) { 347 case ParseKind::Access: 348 specKind = IoSpecKind::Access; 349 break; 350 case ParseKind::Action: 351 specKind = IoSpecKind::Action; 352 break; 353 case ParseKind::Asynchronous: 354 specKind = IoSpecKind::Asynchronous; 355 break; 356 case ParseKind::Blank: 357 specKind = IoSpecKind::Blank; 358 break; 359 case ParseKind::Decimal: 360 specKind = IoSpecKind::Decimal; 361 break; 362 case ParseKind::Delim: 363 specKind = IoSpecKind::Delim; 364 break; 365 case ParseKind::Direct: 366 specKind = IoSpecKind::Direct; 367 break; 368 case ParseKind::Encoding: 369 specKind = IoSpecKind::Encoding; 370 break; 371 case ParseKind::Form: 372 specKind = IoSpecKind::Form; 373 break; 374 case ParseKind::Formatted: 375 specKind = IoSpecKind::Formatted; 376 break; 377 case ParseKind::Iomsg: 378 specKind = IoSpecKind::Iomsg; 379 break; 380 case ParseKind::Name: 381 specKind = IoSpecKind::Name; 382 break; 383 case ParseKind::Pad: 384 specKind = IoSpecKind::Pad; 385 break; 386 case ParseKind::Position: 387 specKind = IoSpecKind::Position; 388 break; 389 case ParseKind::Read: 390 specKind = IoSpecKind::Read; 391 break; 392 case ParseKind::Readwrite: 393 specKind = IoSpecKind::Readwrite; 394 break; 395 case ParseKind::Round: 396 specKind = IoSpecKind::Round; 397 break; 398 case ParseKind::Sequential: 399 specKind = IoSpecKind::Sequential; 400 break; 401 case ParseKind::Sign: 402 specKind = IoSpecKind::Sign; 403 break; 404 case ParseKind::Status: 405 specKind = IoSpecKind::Status; 406 break; 407 case ParseKind::Stream: 408 specKind = IoSpecKind::Stream; 409 break; 410 case ParseKind::Unformatted: 411 specKind = IoSpecKind::Unformatted; 412 break; 413 case ParseKind::Write: 414 specKind = IoSpecKind::Write; 415 break; 416 case ParseKind::Carriagecontrol: 417 specKind = IoSpecKind::Carriagecontrol; 418 break; 419 case ParseKind::Convert: 420 specKind = IoSpecKind::Convert; 421 break; 422 case ParseKind::Dispose: 423 specKind = IoSpecKind::Dispose; 424 break; 425 } 426 CheckForDefinableVariable(std::get<parser::ScalarDefaultCharVariable>(spec.t), 427 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 428 SetSpecifier(specKind); 429 } 430 431 void IoChecker::Enter(const parser::InquireSpec::IntVar &spec) { 432 IoSpecKind specKind{}; 433 using ParseKind = parser::InquireSpec::IntVar::Kind; 434 switch (std::get<parser::InquireSpec::IntVar::Kind>(spec.t)) { 435 case ParseKind::Iostat: 436 specKind = IoSpecKind::Iostat; 437 break; 438 case ParseKind::Nextrec: 439 specKind = IoSpecKind::Nextrec; 440 break; 441 case ParseKind::Number: 442 specKind = IoSpecKind::Number; 443 break; 444 case ParseKind::Pos: 445 specKind = IoSpecKind::Pos; 446 break; 447 case ParseKind::Recl: 448 specKind = IoSpecKind::Recl; 449 break; 450 case ParseKind::Size: 451 specKind = IoSpecKind::Size; 452 break; 453 } 454 CheckForDefinableVariable(std::get<parser::ScalarIntVariable>(spec.t), 455 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 456 SetSpecifier(specKind); 457 } 458 459 void IoChecker::Enter(const parser::InquireSpec::LogVar &spec) { 460 IoSpecKind specKind{}; 461 using ParseKind = parser::InquireSpec::LogVar::Kind; 462 switch (std::get<parser::InquireSpec::LogVar::Kind>(spec.t)) { 463 case ParseKind::Exist: 464 specKind = IoSpecKind::Exist; 465 break; 466 case ParseKind::Named: 467 specKind = IoSpecKind::Named; 468 break; 469 case ParseKind::Opened: 470 specKind = IoSpecKind::Opened; 471 break; 472 case ParseKind::Pending: 473 specKind = IoSpecKind::Pending; 474 break; 475 } 476 SetSpecifier(specKind); 477 } 478 479 void IoChecker::Enter(const parser::IoControlSpec &spec) { 480 // IoControlSpec context Name 481 flags_.set(Flag::IoControlList); 482 if (std::holds_alternative<parser::Name>(spec.u)) { 483 SetSpecifier(IoSpecKind::Nml); 484 flags_.set(Flag::FmtOrNml); 485 } 486 } 487 488 void IoChecker::Enter(const parser::IoControlSpec::Asynchronous &spec) { 489 SetSpecifier(IoSpecKind::Asynchronous); 490 if (const std::optional<std::string> charConst{ 491 GetConstExpr<std::string>(spec)}) { 492 flags_.set(Flag::AsynchronousYes, Normalize(*charConst) == "YES"); 493 CheckStringValue(IoSpecKind::Asynchronous, *charConst, 494 parser::FindSourceLocation(spec)); // C1223 495 } 496 } 497 498 void IoChecker::Enter(const parser::IoControlSpec::CharExpr &spec) { 499 IoSpecKind specKind{}; 500 using ParseKind = parser::IoControlSpec::CharExpr::Kind; 501 switch (std::get<ParseKind>(spec.t)) { 502 case ParseKind::Advance: 503 specKind = IoSpecKind::Advance; 504 break; 505 case ParseKind::Blank: 506 specKind = IoSpecKind::Blank; 507 break; 508 case ParseKind::Decimal: 509 specKind = IoSpecKind::Decimal; 510 break; 511 case ParseKind::Delim: 512 specKind = IoSpecKind::Delim; 513 break; 514 case ParseKind::Pad: 515 specKind = IoSpecKind::Pad; 516 break; 517 case ParseKind::Round: 518 specKind = IoSpecKind::Round; 519 break; 520 case ParseKind::Sign: 521 specKind = IoSpecKind::Sign; 522 break; 523 } 524 SetSpecifier(specKind); 525 if (const std::optional<std::string> charConst{GetConstExpr<std::string>( 526 std::get<parser::ScalarDefaultCharExpr>(spec.t))}) { 527 if (specKind == IoSpecKind::Advance) { 528 flags_.set(Flag::AdvanceYes, Normalize(*charConst) == "YES"); 529 } 530 CheckStringValue(specKind, *charConst, parser::FindSourceLocation(spec)); 531 } 532 } 533 534 void IoChecker::Enter(const parser::IoControlSpec::Pos &) { 535 SetSpecifier(IoSpecKind::Pos); 536 } 537 538 void IoChecker::Enter(const parser::IoControlSpec::Rec &) { 539 SetSpecifier(IoSpecKind::Rec); 540 } 541 542 void IoChecker::Enter(const parser::IoControlSpec::Size &var) { 543 CheckForDefinableVariable(var, "SIZE"); 544 SetSpecifier(IoSpecKind::Size); 545 } 546 547 void IoChecker::Enter(const parser::IoUnit &spec) { 548 if (const parser::Variable * var{std::get_if<parser::Variable>(&spec.u)}) { 549 // Only now after generic resolution can it be known whether a function 550 // call appearing as UNIT=f() is an integer scalar external unit number 551 // or a character pointer for internal I/O. 552 const auto *expr{GetExpr(context_, *var)}; 553 std::optional<evaluate::DynamicType> dyType; 554 if (expr) { 555 dyType = expr->GetType(); 556 } 557 if (dyType && dyType->category() == TypeCategory::Integer) { 558 if (expr->Rank() != 0) { 559 context_.Say(parser::FindSourceLocation(*var), 560 "I/O unit number must be scalar"_err_en_US); 561 } 562 // In the case of an integer unit number variable, rewrite the parse 563 // tree as if the unit had been parsed as a FileUnitNumber in order 564 // to ease lowering. 565 auto &mutableSpec{const_cast<parser::IoUnit &>(spec)}; 566 auto &mutableVar{std::get<parser::Variable>(mutableSpec.u)}; 567 auto source{mutableVar.GetSource()}; 568 auto typedExpr{std::move(mutableVar.typedExpr)}; 569 auto newExpr{common::visit( 570 [](auto &&indirection) { 571 return parser::Expr{std::move(indirection)}; 572 }, 573 std::move(mutableVar.u))}; 574 newExpr.source = source; 575 newExpr.typedExpr = std::move(typedExpr); 576 mutableSpec.u = parser::FileUnitNumber{ 577 parser::ScalarIntExpr{parser::IntExpr{std::move(newExpr)}}}; 578 } else if (!dyType || dyType->category() != TypeCategory::Character) { 579 SetSpecifier(IoSpecKind::Unit); 580 context_.Say(parser::FindSourceLocation(*var), 581 "I/O unit must be a character variable or a scalar integer expression"_err_en_US); 582 } else { // CHARACTER variable (internal I/O) 583 if (stmt_ == IoStmtKind::Write) { 584 CheckForDefinableVariable(*var, "Internal file"); 585 } 586 if (HasVectorSubscript(*expr)) { 587 context_.Say(parser::FindSourceLocation(*var), // C1201 588 "Internal file must not have a vector subscript"_err_en_US); 589 } 590 SetSpecifier(IoSpecKind::Unit); 591 flags_.set(Flag::InternalUnit); 592 } 593 } else if (std::get_if<parser::Star>(&spec.u)) { 594 SetSpecifier(IoSpecKind::Unit); 595 flags_.set(Flag::StarUnit); 596 } 597 } 598 599 void IoChecker::Enter(const parser::MsgVariable &var) { 600 if (stmt_ == IoStmtKind::None) { 601 // allocate, deallocate, image control 602 CheckForDefinableVariable(var, "ERRMSG"); 603 return; 604 } 605 CheckForDefinableVariable(var, "IOMSG"); 606 SetSpecifier(IoSpecKind::Iomsg); 607 } 608 609 void IoChecker::Enter(const parser::OutputItem &item) { 610 flags_.set(Flag::DataList); 611 if (const auto *x{std::get_if<parser::Expr>(&item.u)}) { 612 if (const auto *expr{GetExpr(context_, *x)}) { 613 if (evaluate::IsBOZLiteral(*expr)) { 614 context_.Say(parser::FindSourceLocation(*x), // C7109 615 "Output item must not be a BOZ literal constant"_err_en_US); 616 } 617 const Symbol *last{GetLastSymbol(*expr)}; 618 if (last && IsProcedurePointer(*last)) { 619 context_.Say(parser::FindSourceLocation(*x), 620 "Output item must not be a procedure pointer"_err_en_US); // C1233 621 } 622 CheckForBadIoType(*expr, 623 flags_.test(Flag::FmtOrNml) 624 ? GenericKind::DefinedIo::WriteFormatted 625 : GenericKind::DefinedIo::WriteUnformatted, 626 parser::FindSourceLocation(item)); 627 } 628 } 629 } 630 631 void IoChecker::Enter(const parser::StatusExpr &spec) { 632 SetSpecifier(IoSpecKind::Status); 633 if (const std::optional<std::string> charConst{ 634 GetConstExpr<std::string>(spec)}) { 635 // Status values for Open and Close are different. 636 std::string s{Normalize(*charConst)}; 637 if (stmt_ == IoStmtKind::Open) { 638 flags_.set(Flag::KnownStatus); 639 flags_.set(Flag::StatusNew, s == "NEW"); 640 flags_.set(Flag::StatusReplace, s == "REPLACE"); 641 flags_.set(Flag::StatusScratch, s == "SCRATCH"); 642 // CheckStringValue compares for OPEN Status string values. 643 CheckStringValue( 644 IoSpecKind::Status, *charConst, parser::FindSourceLocation(spec)); 645 return; 646 } 647 CHECK(stmt_ == IoStmtKind::Close); 648 if (s != "DELETE" && s != "KEEP") { 649 context_.Say(parser::FindSourceLocation(spec), 650 "Invalid STATUS value '%s'"_err_en_US, *charConst); 651 } 652 } 653 } 654 655 void IoChecker::Enter(const parser::StatVariable &var) { 656 if (stmt_ == IoStmtKind::None) { 657 // allocate, deallocate, image control 658 CheckForDefinableVariable(var, "STAT"); 659 return; 660 } 661 CheckForDefinableVariable(var, "IOSTAT"); 662 SetSpecifier(IoSpecKind::Iostat); 663 } 664 665 void IoChecker::Leave(const parser::BackspaceStmt &) { 666 CheckForPureSubprogram(); 667 CheckForRequiredSpecifier( 668 flags_.test(Flag::NumberUnit), "UNIT number"); // C1240 669 Done(); 670 } 671 672 void IoChecker::Leave(const parser::CloseStmt &) { 673 CheckForPureSubprogram(); 674 CheckForRequiredSpecifier( 675 flags_.test(Flag::NumberUnit), "UNIT number"); // C1208 676 Done(); 677 } 678 679 void IoChecker::Leave(const parser::EndfileStmt &) { 680 CheckForPureSubprogram(); 681 CheckForRequiredSpecifier( 682 flags_.test(Flag::NumberUnit), "UNIT number"); // C1240 683 Done(); 684 } 685 686 void IoChecker::Leave(const parser::FlushStmt &) { 687 CheckForPureSubprogram(); 688 CheckForRequiredSpecifier( 689 flags_.test(Flag::NumberUnit), "UNIT number"); // C1243 690 Done(); 691 } 692 693 void IoChecker::Leave(const parser::InquireStmt &stmt) { 694 if (std::get_if<std::list<parser::InquireSpec>>(&stmt.u)) { 695 CheckForPureSubprogram(); 696 // Inquire by unit or by file (vs. by output list). 697 CheckForRequiredSpecifier( 698 flags_.test(Flag::NumberUnit) || specifierSet_.test(IoSpecKind::File), 699 "UNIT number or FILE"); // C1246 700 CheckForProhibitedSpecifier(IoSpecKind::File, IoSpecKind::Unit); // C1246 701 CheckForRequiredSpecifier(IoSpecKind::Id, IoSpecKind::Pending); // C1248 702 } 703 Done(); 704 } 705 706 void IoChecker::Leave(const parser::OpenStmt &) { 707 CheckForPureSubprogram(); 708 CheckForRequiredSpecifier(specifierSet_.test(IoSpecKind::Unit) || 709 specifierSet_.test(IoSpecKind::Newunit), 710 "UNIT or NEWUNIT"); // C1204, C1205 711 CheckForProhibitedSpecifier( 712 IoSpecKind::Newunit, IoSpecKind::Unit); // C1204, C1205 713 CheckForRequiredSpecifier(flags_.test(Flag::StatusNew), "STATUS='NEW'", 714 IoSpecKind::File); // 12.5.6.10 715 CheckForRequiredSpecifier(flags_.test(Flag::StatusReplace), 716 "STATUS='REPLACE'", IoSpecKind::File); // 12.5.6.10 717 CheckForProhibitedSpecifier(flags_.test(Flag::StatusScratch), 718 "STATUS='SCRATCH'", IoSpecKind::File); // 12.5.6.10 719 if (flags_.test(Flag::KnownStatus)) { 720 CheckForRequiredSpecifier(IoSpecKind::Newunit, 721 specifierSet_.test(IoSpecKind::File) || 722 flags_.test(Flag::StatusScratch), 723 "FILE or STATUS='SCRATCH'"); // 12.5.6.12 724 } else { 725 CheckForRequiredSpecifier(IoSpecKind::Newunit, 726 specifierSet_.test(IoSpecKind::File) || 727 specifierSet_.test(IoSpecKind::Status), 728 "FILE or STATUS"); // 12.5.6.12 729 } 730 if (flags_.test(Flag::KnownAccess)) { 731 CheckForRequiredSpecifier(flags_.test(Flag::AccessDirect), 732 "ACCESS='DIRECT'", IoSpecKind::Recl); // 12.5.6.15 733 CheckForProhibitedSpecifier(flags_.test(Flag::AccessStream), 734 "STATUS='STREAM'", IoSpecKind::Recl); // 12.5.6.15 735 } 736 Done(); 737 } 738 739 void IoChecker::Leave(const parser::PrintStmt &) { 740 CheckForPureSubprogram(); 741 Done(); 742 } 743 744 static const parser::Name *FindNamelist( 745 const std::list<parser::IoControlSpec> &controls) { 746 for (const auto &control : controls) { 747 if (const parser::Name * namelist{std::get_if<parser::Name>(&control.u)}) { 748 if (namelist->symbol && 749 namelist->symbol->GetUltimate().has<NamelistDetails>()) { 750 return namelist; 751 } 752 } 753 } 754 return nullptr; 755 } 756 757 static void CheckForDoVariable( 758 const parser::ReadStmt &readStmt, SemanticsContext &context) { 759 const std::list<parser::InputItem> &items{readStmt.items}; 760 for (const auto &item : items) { 761 if (const parser::Variable * 762 variable{std::get_if<parser::Variable>(&item.u)}) { 763 context.CheckIndexVarRedefine(*variable); 764 } 765 } 766 } 767 768 void IoChecker::Leave(const parser::ReadStmt &readStmt) { 769 if (!flags_.test(Flag::InternalUnit)) { 770 CheckForPureSubprogram(); 771 } 772 if (const parser::Name * namelist{FindNamelist(readStmt.controls)}) { 773 if (namelist->symbol) { 774 CheckNamelist(*namelist->symbol, GenericKind::DefinedIo::ReadFormatted, 775 namelist->source); 776 } 777 } 778 CheckForDoVariable(readStmt, context_); 779 if (!flags_.test(Flag::IoControlList)) { 780 Done(); 781 return; 782 } 783 LeaveReadWrite(); 784 CheckForProhibitedSpecifier(IoSpecKind::Delim); // C1212 785 CheckForProhibitedSpecifier(IoSpecKind::Sign); // C1212 786 CheckForProhibitedSpecifier(IoSpecKind::Rec, IoSpecKind::End); // C1220 787 CheckForRequiredSpecifier(IoSpecKind::Eor, 788 specifierSet_.test(IoSpecKind::Advance) && !flags_.test(Flag::AdvanceYes), 789 "ADVANCE with value 'NO'"); // C1222 + 12.6.2.1p2 790 CheckForRequiredSpecifier(IoSpecKind::Blank, flags_.test(Flag::FmtOrNml), 791 "FMT or NML"); // C1227 792 CheckForRequiredSpecifier( 793 IoSpecKind::Pad, flags_.test(Flag::FmtOrNml), "FMT or NML"); // C1227 794 Done(); 795 } 796 797 void IoChecker::Leave(const parser::RewindStmt &) { 798 CheckForRequiredSpecifier( 799 flags_.test(Flag::NumberUnit), "UNIT number"); // C1240 800 CheckForPureSubprogram(); 801 Done(); 802 } 803 804 void IoChecker::Leave(const parser::WaitStmt &) { 805 CheckForRequiredSpecifier( 806 flags_.test(Flag::NumberUnit), "UNIT number"); // C1237 807 CheckForPureSubprogram(); 808 Done(); 809 } 810 811 void IoChecker::Leave(const parser::WriteStmt &writeStmt) { 812 if (!flags_.test(Flag::InternalUnit)) { 813 CheckForPureSubprogram(); 814 } 815 if (const parser::Name * namelist{FindNamelist(writeStmt.controls)}) { 816 if (namelist->symbol) { 817 CheckNamelist(*namelist->symbol, GenericKind::DefinedIo::WriteFormatted, 818 namelist->source); 819 } 820 } 821 LeaveReadWrite(); 822 CheckForProhibitedSpecifier(IoSpecKind::Blank); // C1213 823 CheckForProhibitedSpecifier(IoSpecKind::End); // C1213 824 CheckForProhibitedSpecifier(IoSpecKind::Eor); // C1213 825 CheckForProhibitedSpecifier(IoSpecKind::Pad); // C1213 826 CheckForProhibitedSpecifier(IoSpecKind::Size); // C1213 827 CheckForRequiredSpecifier( 828 IoSpecKind::Sign, flags_.test(Flag::FmtOrNml), "FMT or NML"); // C1227 829 CheckForRequiredSpecifier(IoSpecKind::Delim, 830 flags_.test(Flag::StarFmt) || specifierSet_.test(IoSpecKind::Nml), 831 "FMT=* or NML"); // C1228 832 Done(); 833 } 834 835 void IoChecker::LeaveReadWrite() const { 836 CheckForRequiredSpecifier(IoSpecKind::Unit); // C1211 837 CheckForProhibitedSpecifier(IoSpecKind::Nml, IoSpecKind::Rec); // C1216 838 CheckForProhibitedSpecifier(IoSpecKind::Nml, IoSpecKind::Fmt); // C1216 839 CheckForProhibitedSpecifier( 840 IoSpecKind::Nml, flags_.test(Flag::DataList), "a data list"); // C1216 841 CheckForProhibitedSpecifier(flags_.test(Flag::InternalUnit), 842 "UNIT=internal-file", IoSpecKind::Pos); // C1219 843 CheckForProhibitedSpecifier(flags_.test(Flag::InternalUnit), 844 "UNIT=internal-file", IoSpecKind::Rec); // C1219 845 CheckForProhibitedSpecifier( 846 flags_.test(Flag::StarUnit), "UNIT=*", IoSpecKind::Pos); // C1219 847 CheckForProhibitedSpecifier( 848 flags_.test(Flag::StarUnit), "UNIT=*", IoSpecKind::Rec); // C1219 849 CheckForProhibitedSpecifier( 850 IoSpecKind::Rec, flags_.test(Flag::StarFmt), "FMT=*"); // C1220 851 CheckForRequiredSpecifier(IoSpecKind::Advance, 852 flags_.test(Flag::CharFmt) || flags_.test(Flag::LabelFmt) || 853 flags_.test(Flag::AssignFmt), 854 "an explicit format"); // C1221 855 CheckForProhibitedSpecifier(IoSpecKind::Advance, 856 flags_.test(Flag::InternalUnit), "UNIT=internal-file"); // C1221 857 CheckForRequiredSpecifier(flags_.test(Flag::AsynchronousYes), 858 "ASYNCHRONOUS='YES'", flags_.test(Flag::NumberUnit), 859 "UNIT=number"); // C1224 860 CheckForRequiredSpecifier(IoSpecKind::Id, flags_.test(Flag::AsynchronousYes), 861 "ASYNCHRONOUS='YES'"); // C1225 862 CheckForProhibitedSpecifier(IoSpecKind::Pos, IoSpecKind::Rec); // C1226 863 CheckForRequiredSpecifier(IoSpecKind::Decimal, flags_.test(Flag::FmtOrNml), 864 "FMT or NML"); // C1227 865 CheckForRequiredSpecifier(IoSpecKind::Round, flags_.test(Flag::FmtOrNml), 866 "FMT or NML"); // C1227 867 } 868 869 void IoChecker::SetSpecifier(IoSpecKind specKind) { 870 if (stmt_ == IoStmtKind::None) { 871 // FMT may appear on PRINT statements, which don't have any checks. 872 // [IO]MSG and [IO]STAT parse symbols are shared with non-I/O statements. 873 return; 874 } 875 // C1203, C1207, C1210, C1236, C1239, C1242, C1245 876 if (specifierSet_.test(specKind)) { 877 context_.Say("Duplicate %s specifier"_err_en_US, 878 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 879 } 880 specifierSet_.set(specKind); 881 } 882 883 void IoChecker::CheckStringValue(IoSpecKind specKind, const std::string &value, 884 const parser::CharBlock &source) const { 885 static std::unordered_map<IoSpecKind, const std::set<std::string>> specValues{ 886 {IoSpecKind::Access, {"DIRECT", "SEQUENTIAL", "STREAM"}}, 887 {IoSpecKind::Action, {"READ", "READWRITE", "WRITE"}}, 888 {IoSpecKind::Advance, {"NO", "YES"}}, 889 {IoSpecKind::Asynchronous, {"NO", "YES"}}, 890 {IoSpecKind::Blank, {"NULL", "ZERO"}}, 891 {IoSpecKind::Decimal, {"COMMA", "POINT"}}, 892 {IoSpecKind::Delim, {"APOSTROPHE", "NONE", "QUOTE"}}, 893 {IoSpecKind::Encoding, {"DEFAULT", "UTF-8"}}, 894 {IoSpecKind::Form, {"FORMATTED", "UNFORMATTED"}}, 895 {IoSpecKind::Pad, {"NO", "YES"}}, 896 {IoSpecKind::Position, {"APPEND", "ASIS", "REWIND"}}, 897 {IoSpecKind::Round, 898 {"COMPATIBLE", "DOWN", "NEAREST", "PROCESSOR_DEFINED", "UP", "ZERO"}}, 899 {IoSpecKind::Sign, {"PLUS", "PROCESSOR_DEFINED", "SUPPRESS"}}, 900 {IoSpecKind::Status, 901 // Open values; Close values are {"DELETE", "KEEP"}. 902 {"NEW", "OLD", "REPLACE", "SCRATCH", "UNKNOWN"}}, 903 {IoSpecKind::Carriagecontrol, {"LIST", "FORTRAN", "NONE"}}, 904 {IoSpecKind::Convert, {"BIG_ENDIAN", "LITTLE_ENDIAN", "NATIVE"}}, 905 {IoSpecKind::Dispose, {"DELETE", "KEEP"}}, 906 }; 907 auto upper{Normalize(value)}; 908 if (specValues.at(specKind).count(upper) == 0) { 909 if (specKind == IoSpecKind::Access && upper == "APPEND") { 910 if (context_.languageFeatures().ShouldWarn( 911 common::LanguageFeature::OpenAccessAppend)) { 912 context_.Say(source, 913 "ACCESS='%s' interpreted as POSITION='%s'"_port_en_US, value, 914 upper); 915 } 916 } else { 917 context_.Say(source, "Invalid %s value '%s'"_err_en_US, 918 parser::ToUpperCaseLetters(common::EnumToString(specKind)), value); 919 } 920 } 921 } 922 923 // CheckForRequiredSpecifier and CheckForProhibitedSpecifier functions 924 // need conditions to check, and string arguments to insert into a message. 925 // An IoSpecKind provides both an absence/presence condition and a string 926 // argument (its name). A (condition, string) pair provides an arbitrary 927 // condition and an arbitrary string. 928 929 void IoChecker::CheckForRequiredSpecifier(IoSpecKind specKind) const { 930 if (!specifierSet_.test(specKind)) { 931 context_.Say("%s statement must have a %s specifier"_err_en_US, 932 parser::ToUpperCaseLetters(common::EnumToString(stmt_)), 933 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 934 } 935 } 936 937 void IoChecker::CheckForRequiredSpecifier( 938 bool condition, const std::string &s) const { 939 if (!condition) { 940 context_.Say("%s statement must have a %s specifier"_err_en_US, 941 parser::ToUpperCaseLetters(common::EnumToString(stmt_)), s); 942 } 943 } 944 945 void IoChecker::CheckForRequiredSpecifier( 946 IoSpecKind specKind1, IoSpecKind specKind2) const { 947 if (specifierSet_.test(specKind1) && !specifierSet_.test(specKind2)) { 948 context_.Say("If %s appears, %s must also appear"_err_en_US, 949 parser::ToUpperCaseLetters(common::EnumToString(specKind1)), 950 parser::ToUpperCaseLetters(common::EnumToString(specKind2))); 951 } 952 } 953 954 void IoChecker::CheckForRequiredSpecifier( 955 IoSpecKind specKind, bool condition, const std::string &s) const { 956 if (specifierSet_.test(specKind) && !condition) { 957 context_.Say("If %s appears, %s must also appear"_err_en_US, 958 parser::ToUpperCaseLetters(common::EnumToString(specKind)), s); 959 } 960 } 961 962 void IoChecker::CheckForRequiredSpecifier( 963 bool condition, const std::string &s, IoSpecKind specKind) const { 964 if (condition && !specifierSet_.test(specKind)) { 965 context_.Say("If %s appears, %s must also appear"_err_en_US, s, 966 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 967 } 968 } 969 970 void IoChecker::CheckForRequiredSpecifier(bool condition1, 971 const std::string &s1, bool condition2, const std::string &s2) const { 972 if (condition1 && !condition2) { 973 context_.Say("If %s appears, %s must also appear"_err_en_US, s1, s2); 974 } 975 } 976 977 void IoChecker::CheckForProhibitedSpecifier(IoSpecKind specKind) const { 978 if (specifierSet_.test(specKind)) { 979 context_.Say("%s statement must not have a %s specifier"_err_en_US, 980 parser::ToUpperCaseLetters(common::EnumToString(stmt_)), 981 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 982 } 983 } 984 985 void IoChecker::CheckForProhibitedSpecifier( 986 IoSpecKind specKind1, IoSpecKind specKind2) const { 987 if (specifierSet_.test(specKind1) && specifierSet_.test(specKind2)) { 988 context_.Say("If %s appears, %s must not appear"_err_en_US, 989 parser::ToUpperCaseLetters(common::EnumToString(specKind1)), 990 parser::ToUpperCaseLetters(common::EnumToString(specKind2))); 991 } 992 } 993 994 void IoChecker::CheckForProhibitedSpecifier( 995 IoSpecKind specKind, bool condition, const std::string &s) const { 996 if (specifierSet_.test(specKind) && condition) { 997 context_.Say("If %s appears, %s must not appear"_err_en_US, 998 parser::ToUpperCaseLetters(common::EnumToString(specKind)), s); 999 } 1000 } 1001 1002 void IoChecker::CheckForProhibitedSpecifier( 1003 bool condition, const std::string &s, IoSpecKind specKind) const { 1004 if (condition && specifierSet_.test(specKind)) { 1005 context_.Say("If %s appears, %s must not appear"_err_en_US, s, 1006 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 1007 } 1008 } 1009 1010 template <typename A> 1011 void IoChecker::CheckForDefinableVariable( 1012 const A &variable, const std::string &s) const { 1013 if (const auto *var{parser::Unwrap<parser::Variable>(variable)}) { 1014 if (auto expr{AnalyzeExpr(context_, *var)}) { 1015 auto at{var->GetSource()}; 1016 if (auto whyNot{WhyNotDefinable(at, context_.FindScope(at), 1017 DefinabilityFlags{DefinabilityFlag::VectorSubscriptIsOk}, 1018 *expr)}) { 1019 const Symbol *base{GetFirstSymbol(*expr)}; 1020 context_ 1021 .Say(at, "%s variable '%s' is not definable"_err_en_US, s, 1022 (base ? base->name() : at).ToString()) 1023 .Attach(std::move(*whyNot)); 1024 } 1025 } 1026 } 1027 } 1028 1029 void IoChecker::CheckForPureSubprogram() const { // C1597 1030 CHECK(context_.location()); 1031 if (const Scope * 1032 scope{context_.globalScope().FindScope(*context_.location())}) { 1033 if (FindPureProcedureContaining(*scope)) { 1034 context_.Say( 1035 "External I/O is not allowed in a pure subprogram"_err_en_US); 1036 } 1037 } 1038 } 1039 1040 // Seeks out an allocatable or pointer ultimate component that is not 1041 // nested in a nonallocatable/nonpointer component with a specific 1042 // defined I/O procedure. 1043 static const Symbol *FindUnsafeIoDirectComponent(GenericKind::DefinedIo which, 1044 const DerivedTypeSpec &derived, const Scope &scope) { 1045 if (HasDefinedIo(which, derived, &scope)) { 1046 return nullptr; 1047 } 1048 if (const Scope * dtScope{derived.scope()}) { 1049 for (const auto &pair : *dtScope) { 1050 const Symbol &symbol{*pair.second}; 1051 if (IsAllocatableOrPointer(symbol)) { 1052 return &symbol; 1053 } 1054 if (const auto *details{symbol.detailsIf<ObjectEntityDetails>()}) { 1055 if (const DeclTypeSpec * type{details->type()}) { 1056 if (type->category() == DeclTypeSpec::Category::TypeDerived) { 1057 const DerivedTypeSpec &componentDerived{type->derivedTypeSpec()}; 1058 if (const Symbol * 1059 bad{FindUnsafeIoDirectComponent( 1060 which, componentDerived, scope)}) { 1061 return bad; 1062 } 1063 } 1064 } 1065 } 1066 } 1067 } 1068 return nullptr; 1069 } 1070 1071 // For a type that does not have a defined I/O subroutine, finds a direct 1072 // component that is a witness to an accessibility violation outside the module 1073 // in which the type was defined. 1074 static const Symbol *FindInaccessibleComponent(GenericKind::DefinedIo which, 1075 const DerivedTypeSpec &derived, const Scope &scope) { 1076 if (const Scope * dtScope{derived.scope()}) { 1077 if (const Scope * module{FindModuleContaining(*dtScope)}) { 1078 for (const auto &pair : *dtScope) { 1079 const Symbol &symbol{*pair.second}; 1080 if (IsAllocatableOrPointer(symbol)) { 1081 continue; // already an error 1082 } 1083 if (const auto *details{symbol.detailsIf<ObjectEntityDetails>()}) { 1084 const DerivedTypeSpec *componentDerived{nullptr}; 1085 if (const DeclTypeSpec * type{details->type()}) { 1086 if (type->category() == DeclTypeSpec::Category::TypeDerived) { 1087 componentDerived = &type->derivedTypeSpec(); 1088 } 1089 } 1090 if (componentDerived && 1091 HasDefinedIo(which, *componentDerived, &scope)) { 1092 continue; // this component and its descendents are fine 1093 } 1094 if (symbol.attrs().test(Attr::PRIVATE) && 1095 !symbol.test(Symbol::Flag::ParentComp)) { 1096 if (!DoesScopeContain(module, scope)) { 1097 return &symbol; 1098 } 1099 } 1100 if (componentDerived) { 1101 if (const Symbol * 1102 bad{FindInaccessibleComponent( 1103 which, *componentDerived, scope)}) { 1104 return bad; 1105 } 1106 } 1107 } 1108 } 1109 } 1110 } 1111 return nullptr; 1112 } 1113 1114 // Fortran 2018, 12.6.3 paragraphs 5 & 7 1115 parser::Message *IoChecker::CheckForBadIoType(const evaluate::DynamicType &type, 1116 GenericKind::DefinedIo which, parser::CharBlock where) const { 1117 if (type.IsUnlimitedPolymorphic()) { 1118 return &context_.Say( 1119 where, "I/O list item may not be unlimited polymorphic"_err_en_US); 1120 } else if (type.category() == TypeCategory::Derived) { 1121 const auto &derived{type.GetDerivedTypeSpec()}; 1122 const Scope &scope{context_.FindScope(where)}; 1123 if (const Symbol * 1124 bad{FindUnsafeIoDirectComponent(which, derived, scope)}) { 1125 return &context_.SayWithDecl(*bad, where, 1126 "Derived type '%s' in I/O cannot have an allocatable or pointer direct component '%s' unless using defined I/O"_err_en_US, 1127 derived.name(), bad->name()); 1128 } 1129 if (!HasDefinedIo(which, derived, &scope)) { 1130 if (type.IsPolymorphic()) { 1131 return &context_.Say(where, 1132 "Derived type '%s' in I/O may not be polymorphic unless using defined I/O"_err_en_US, 1133 derived.name()); 1134 } 1135 if (const Symbol * 1136 bad{FindInaccessibleComponent(which, derived, scope)}) { 1137 return &context_.Say(where, 1138 "I/O of the derived type '%s' may not be performed without defined I/O in a scope in which a direct component like '%s' is inaccessible"_err_en_US, 1139 derived.name(), bad->name()); 1140 } 1141 } 1142 } 1143 return nullptr; 1144 } 1145 1146 void IoChecker::CheckForBadIoType(const SomeExpr &expr, 1147 GenericKind::DefinedIo which, parser::CharBlock where) const { 1148 if (auto type{expr.GetType()}) { 1149 CheckForBadIoType(*type, which, where); 1150 } 1151 } 1152 1153 parser::Message *IoChecker::CheckForBadIoType(const Symbol &symbol, 1154 GenericKind::DefinedIo which, parser::CharBlock where) const { 1155 if (auto type{evaluate::DynamicType::From(symbol)}) { 1156 if (auto *msg{CheckForBadIoType(*type, which, where)}) { 1157 evaluate::AttachDeclaration(*msg, symbol); 1158 return msg; 1159 } 1160 } 1161 return nullptr; 1162 } 1163 1164 void IoChecker::CheckNamelist(const Symbol &namelist, 1165 GenericKind::DefinedIo which, parser::CharBlock namelistLocation) const { 1166 const auto &details{namelist.GetUltimate().get<NamelistDetails>()}; 1167 for (const Symbol &object : details.objects()) { 1168 context_.CheckIndexVarRedefine(namelistLocation, object); 1169 if (auto *msg{CheckForBadIoType(object, which, namelistLocation)}) { 1170 evaluate::AttachDeclaration(*msg, namelist); 1171 } 1172 } 1173 } 1174 1175 } // namespace Fortran::semantics 1176