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 } 235 return; 236 } 237 if (type && type->category() != TypeCategory::Character && 238 (type->category() != TypeCategory::Integer || 239 expr->Rank() > 0) && 240 context_.IsEnabled( 241 common::LanguageFeature::NonCharacterFormat)) { 242 // Legacy extension: using non-character variables, typically 243 // DATA-initialized with Hollerith, as format expressions. 244 if (context_.ShouldWarn( 245 common::LanguageFeature::NonCharacterFormat)) { 246 context_.Say(format.source, 247 "Non-character format expression is not standard"_port_en_US); 248 } 249 } else if (!type || 250 type->kind() != 251 context_.defaultKinds().GetDefaultKind(type->category())) { 252 context_.Say(format.source, 253 "Format expression must be default character or default scalar integer"_err_en_US); 254 return; 255 } 256 flags_.set(Flag::CharFmt); 257 const std::optional<std::string> constantFormat{ 258 GetConstExpr<std::string>(format)}; 259 if (!constantFormat) { 260 return; 261 } 262 // validate constant format -- 12.6.2.2 263 bool isFolded{constantFormat->size() != format.source.size() - 2}; 264 parser::CharBlock reporterCharBlock{isFolded 265 ? parser::CharBlock{format.source} 266 : parser::CharBlock{format.source.begin() + 1, 267 static_cast<std::size_t>(0)}}; 268 FormatErrorReporter reporter{context_, reporterCharBlock}; 269 auto reporterWrapper{ 270 [&](const auto &msg) { return reporter.Say(msg); }}; 271 switch (context_.GetDefaultKind(TypeCategory::Character)) { 272 case 1: { 273 common::FormatValidator<char> validator{constantFormat->c_str(), 274 constantFormat->length(), reporterWrapper, stmt_}; 275 validator.Check(); 276 break; 277 } 278 case 2: { 279 // TODO: Get this to work. (Maybe combine with earlier instance?) 280 common::FormatValidator<char16_t> validator{ 281 /*???*/ nullptr, /*???*/ 0, reporterWrapper, stmt_}; 282 validator.Check(); 283 break; 284 } 285 case 4: { 286 // TODO: Get this to work. (Maybe combine with earlier instance?) 287 common::FormatValidator<char32_t> validator{ 288 /*???*/ nullptr, /*???*/ 0, reporterWrapper, stmt_}; 289 validator.Check(); 290 break; 291 } 292 default: 293 CRASH_NO_CASE; 294 } 295 }, 296 }, 297 spec.u); 298 } 299 300 void IoChecker::Enter(const parser::IdExpr &) { SetSpecifier(IoSpecKind::Id); } 301 302 void IoChecker::Enter(const parser::IdVariable &spec) { 303 SetSpecifier(IoSpecKind::Id); 304 const auto *expr{GetExpr(context_, spec)}; 305 if (!expr || !expr->GetType()) { 306 return; 307 } 308 CheckForDefinableVariable(spec, "ID"); 309 int kind{expr->GetType()->kind()}; 310 int defaultKind{context_.GetDefaultKind(TypeCategory::Integer)}; 311 if (kind < defaultKind) { 312 context_.Say( 313 "ID kind (%d) is smaller than default INTEGER kind (%d)"_err_en_US, 314 std::move(kind), std::move(defaultKind)); // C1229 315 } 316 } 317 318 void IoChecker::Enter(const parser::InputItem &spec) { 319 flags_.set(Flag::DataList); 320 const parser::Variable *var{std::get_if<parser::Variable>(&spec.u)}; 321 if (!var) { 322 return; 323 } 324 CheckForDefinableVariable(*var, "Input"); 325 if (auto expr{AnalyzeExpr(context_, *var)}) { 326 CheckForBadIoComponent(*expr, 327 flags_.test(Flag::FmtOrNml) ? GenericKind::DefinedIo::ReadFormatted 328 : GenericKind::DefinedIo::ReadUnformatted, 329 var->GetSource()); 330 } 331 } 332 333 void IoChecker::Enter(const parser::InquireSpec &spec) { 334 // InquireSpec context FileNameExpr 335 if (std::get_if<parser::FileNameExpr>(&spec.u)) { 336 SetSpecifier(IoSpecKind::File); 337 } 338 } 339 340 void IoChecker::Enter(const parser::InquireSpec::CharVar &spec) { 341 IoSpecKind specKind{}; 342 using ParseKind = parser::InquireSpec::CharVar::Kind; 343 switch (std::get<ParseKind>(spec.t)) { 344 case ParseKind::Access: 345 specKind = IoSpecKind::Access; 346 break; 347 case ParseKind::Action: 348 specKind = IoSpecKind::Action; 349 break; 350 case ParseKind::Asynchronous: 351 specKind = IoSpecKind::Asynchronous; 352 break; 353 case ParseKind::Blank: 354 specKind = IoSpecKind::Blank; 355 break; 356 case ParseKind::Decimal: 357 specKind = IoSpecKind::Decimal; 358 break; 359 case ParseKind::Delim: 360 specKind = IoSpecKind::Delim; 361 break; 362 case ParseKind::Direct: 363 specKind = IoSpecKind::Direct; 364 break; 365 case ParseKind::Encoding: 366 specKind = IoSpecKind::Encoding; 367 break; 368 case ParseKind::Form: 369 specKind = IoSpecKind::Form; 370 break; 371 case ParseKind::Formatted: 372 specKind = IoSpecKind::Formatted; 373 break; 374 case ParseKind::Iomsg: 375 specKind = IoSpecKind::Iomsg; 376 break; 377 case ParseKind::Name: 378 specKind = IoSpecKind::Name; 379 break; 380 case ParseKind::Pad: 381 specKind = IoSpecKind::Pad; 382 break; 383 case ParseKind::Position: 384 specKind = IoSpecKind::Position; 385 break; 386 case ParseKind::Read: 387 specKind = IoSpecKind::Read; 388 break; 389 case ParseKind::Readwrite: 390 specKind = IoSpecKind::Readwrite; 391 break; 392 case ParseKind::Round: 393 specKind = IoSpecKind::Round; 394 break; 395 case ParseKind::Sequential: 396 specKind = IoSpecKind::Sequential; 397 break; 398 case ParseKind::Sign: 399 specKind = IoSpecKind::Sign; 400 break; 401 case ParseKind::Status: 402 specKind = IoSpecKind::Status; 403 break; 404 case ParseKind::Stream: 405 specKind = IoSpecKind::Stream; 406 break; 407 case ParseKind::Unformatted: 408 specKind = IoSpecKind::Unformatted; 409 break; 410 case ParseKind::Write: 411 specKind = IoSpecKind::Write; 412 break; 413 case ParseKind::Carriagecontrol: 414 specKind = IoSpecKind::Carriagecontrol; 415 break; 416 case ParseKind::Convert: 417 specKind = IoSpecKind::Convert; 418 break; 419 case ParseKind::Dispose: 420 specKind = IoSpecKind::Dispose; 421 break; 422 } 423 CheckForDefinableVariable(std::get<parser::ScalarDefaultCharVariable>(spec.t), 424 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 425 SetSpecifier(specKind); 426 } 427 428 void IoChecker::Enter(const parser::InquireSpec::IntVar &spec) { 429 IoSpecKind specKind{}; 430 using ParseKind = parser::InquireSpec::IntVar::Kind; 431 switch (std::get<parser::InquireSpec::IntVar::Kind>(spec.t)) { 432 case ParseKind::Iostat: 433 specKind = IoSpecKind::Iostat; 434 break; 435 case ParseKind::Nextrec: 436 specKind = IoSpecKind::Nextrec; 437 break; 438 case ParseKind::Number: 439 specKind = IoSpecKind::Number; 440 break; 441 case ParseKind::Pos: 442 specKind = IoSpecKind::Pos; 443 break; 444 case ParseKind::Recl: 445 specKind = IoSpecKind::Recl; 446 break; 447 case ParseKind::Size: 448 specKind = IoSpecKind::Size; 449 break; 450 } 451 CheckForDefinableVariable(std::get<parser::ScalarIntVariable>(spec.t), 452 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 453 SetSpecifier(specKind); 454 } 455 456 void IoChecker::Enter(const parser::InquireSpec::LogVar &spec) { 457 IoSpecKind specKind{}; 458 using ParseKind = parser::InquireSpec::LogVar::Kind; 459 switch (std::get<parser::InquireSpec::LogVar::Kind>(spec.t)) { 460 case ParseKind::Exist: 461 specKind = IoSpecKind::Exist; 462 break; 463 case ParseKind::Named: 464 specKind = IoSpecKind::Named; 465 break; 466 case ParseKind::Opened: 467 specKind = IoSpecKind::Opened; 468 break; 469 case ParseKind::Pending: 470 specKind = IoSpecKind::Pending; 471 break; 472 } 473 SetSpecifier(specKind); 474 } 475 476 void IoChecker::Enter(const parser::IoControlSpec &spec) { 477 // IoControlSpec context Name 478 flags_.set(Flag::IoControlList); 479 if (std::holds_alternative<parser::Name>(spec.u)) { 480 SetSpecifier(IoSpecKind::Nml); 481 flags_.set(Flag::FmtOrNml); 482 } 483 } 484 485 void IoChecker::Enter(const parser::IoControlSpec::Asynchronous &spec) { 486 SetSpecifier(IoSpecKind::Asynchronous); 487 if (const std::optional<std::string> charConst{ 488 GetConstExpr<std::string>(spec)}) { 489 flags_.set(Flag::AsynchronousYes, Normalize(*charConst) == "YES"); 490 CheckStringValue(IoSpecKind::Asynchronous, *charConst, 491 parser::FindSourceLocation(spec)); // C1223 492 } 493 } 494 495 void IoChecker::Enter(const parser::IoControlSpec::CharExpr &spec) { 496 IoSpecKind specKind{}; 497 using ParseKind = parser::IoControlSpec::CharExpr::Kind; 498 switch (std::get<ParseKind>(spec.t)) { 499 case ParseKind::Advance: 500 specKind = IoSpecKind::Advance; 501 break; 502 case ParseKind::Blank: 503 specKind = IoSpecKind::Blank; 504 break; 505 case ParseKind::Decimal: 506 specKind = IoSpecKind::Decimal; 507 break; 508 case ParseKind::Delim: 509 specKind = IoSpecKind::Delim; 510 break; 511 case ParseKind::Pad: 512 specKind = IoSpecKind::Pad; 513 break; 514 case ParseKind::Round: 515 specKind = IoSpecKind::Round; 516 break; 517 case ParseKind::Sign: 518 specKind = IoSpecKind::Sign; 519 break; 520 } 521 SetSpecifier(specKind); 522 if (const std::optional<std::string> charConst{GetConstExpr<std::string>( 523 std::get<parser::ScalarDefaultCharExpr>(spec.t))}) { 524 if (specKind == IoSpecKind::Advance) { 525 flags_.set(Flag::AdvanceYes, Normalize(*charConst) == "YES"); 526 } 527 CheckStringValue(specKind, *charConst, parser::FindSourceLocation(spec)); 528 } 529 } 530 531 void IoChecker::Enter(const parser::IoControlSpec::Pos &) { 532 SetSpecifier(IoSpecKind::Pos); 533 } 534 535 void IoChecker::Enter(const parser::IoControlSpec::Rec &) { 536 SetSpecifier(IoSpecKind::Rec); 537 } 538 539 void IoChecker::Enter(const parser::IoControlSpec::Size &var) { 540 CheckForDefinableVariable(var, "SIZE"); 541 SetSpecifier(IoSpecKind::Size); 542 } 543 544 void IoChecker::Enter(const parser::IoUnit &spec) { 545 if (const parser::Variable * var{std::get_if<parser::Variable>(&spec.u)}) { 546 // Only now after generic resolution can it be known whether a function 547 // call appearing as UNIT=f() is an integer scalar external unit number 548 // or a character pointer for internal I/O. 549 const auto *expr{GetExpr(context_, *var)}; 550 std::optional<evaluate::DynamicType> dyType; 551 if (expr) { 552 dyType = expr->GetType(); 553 } 554 if (dyType && dyType->category() == TypeCategory::Integer) { 555 if (expr->Rank() != 0) { 556 context_.Say(parser::FindSourceLocation(*var), 557 "I/O unit number must be scalar"_err_en_US); 558 } 559 // In the case of an integer unit number variable, rewrite the parse 560 // tree as if the unit had been parsed as a FileUnitNumber in order 561 // to ease lowering. 562 auto &mutableSpec{const_cast<parser::IoUnit &>(spec)}; 563 auto &mutableVar{std::get<parser::Variable>(mutableSpec.u)}; 564 auto source{mutableVar.GetSource()}; 565 auto typedExpr{std::move(mutableVar.typedExpr)}; 566 auto newExpr{common::visit( 567 [](auto &&indirection) { 568 return parser::Expr{std::move(indirection)}; 569 }, 570 std::move(mutableVar.u))}; 571 newExpr.source = source; 572 newExpr.typedExpr = std::move(typedExpr); 573 mutableSpec.u = parser::FileUnitNumber{ 574 parser::ScalarIntExpr{parser::IntExpr{std::move(newExpr)}}}; 575 } else if (!dyType || dyType->category() != TypeCategory::Character) { 576 SetSpecifier(IoSpecKind::Unit); 577 context_.Say(parser::FindSourceLocation(*var), 578 "I/O unit must be a character variable or a scalar integer expression"_err_en_US); 579 } else { // CHARACTER variable (internal I/O) 580 if (stmt_ == IoStmtKind::Write) { 581 CheckForDefinableVariable(*var, "Internal file"); 582 } 583 if (HasVectorSubscript(*expr)) { 584 context_.Say(parser::FindSourceLocation(*var), // C1201 585 "Internal file must not have a vector subscript"_err_en_US); 586 } 587 SetSpecifier(IoSpecKind::Unit); 588 flags_.set(Flag::InternalUnit); 589 } 590 } else if (std::get_if<parser::Star>(&spec.u)) { 591 SetSpecifier(IoSpecKind::Unit); 592 flags_.set(Flag::StarUnit); 593 } 594 } 595 596 void IoChecker::Enter(const parser::MsgVariable &var) { 597 if (stmt_ == IoStmtKind::None) { 598 // allocate, deallocate, image control 599 CheckForDefinableVariable(var, "ERRMSG"); 600 return; 601 } 602 CheckForDefinableVariable(var, "IOMSG"); 603 SetSpecifier(IoSpecKind::Iomsg); 604 } 605 606 void IoChecker::Enter(const parser::OutputItem &item) { 607 flags_.set(Flag::DataList); 608 if (const auto *x{std::get_if<parser::Expr>(&item.u)}) { 609 if (const auto *expr{GetExpr(context_, *x)}) { 610 if (evaluate::IsBOZLiteral(*expr)) { 611 context_.Say(parser::FindSourceLocation(*x), // C7109 612 "Output item must not be a BOZ literal constant"_err_en_US); 613 } 614 const Symbol *last{GetLastSymbol(*expr)}; 615 if (last && IsProcedurePointer(*last)) { 616 context_.Say(parser::FindSourceLocation(*x), 617 "Output item must not be a procedure pointer"_err_en_US); // C1233 618 } 619 CheckForBadIoComponent(*expr, 620 flags_.test(Flag::FmtOrNml) 621 ? GenericKind::DefinedIo::WriteFormatted 622 : GenericKind::DefinedIo::WriteUnformatted, 623 parser::FindSourceLocation(item)); 624 } 625 } 626 } 627 628 void IoChecker::Enter(const parser::StatusExpr &spec) { 629 SetSpecifier(IoSpecKind::Status); 630 if (const std::optional<std::string> charConst{ 631 GetConstExpr<std::string>(spec)}) { 632 // Status values for Open and Close are different. 633 std::string s{Normalize(*charConst)}; 634 if (stmt_ == IoStmtKind::Open) { 635 flags_.set(Flag::KnownStatus); 636 flags_.set(Flag::StatusNew, s == "NEW"); 637 flags_.set(Flag::StatusReplace, s == "REPLACE"); 638 flags_.set(Flag::StatusScratch, s == "SCRATCH"); 639 // CheckStringValue compares for OPEN Status string values. 640 CheckStringValue( 641 IoSpecKind::Status, *charConst, parser::FindSourceLocation(spec)); 642 return; 643 } 644 CHECK(stmt_ == IoStmtKind::Close); 645 if (s != "DELETE" && s != "KEEP") { 646 context_.Say(parser::FindSourceLocation(spec), 647 "Invalid STATUS value '%s'"_err_en_US, *charConst); 648 } 649 } 650 } 651 652 void IoChecker::Enter(const parser::StatVariable &var) { 653 if (stmt_ == IoStmtKind::None) { 654 // allocate, deallocate, image control 655 CheckForDefinableVariable(var, "STAT"); 656 return; 657 } 658 CheckForDefinableVariable(var, "IOSTAT"); 659 SetSpecifier(IoSpecKind::Iostat); 660 } 661 662 void IoChecker::Leave(const parser::BackspaceStmt &) { 663 CheckForPureSubprogram(); 664 CheckForRequiredSpecifier( 665 flags_.test(Flag::NumberUnit), "UNIT number"); // C1240 666 Done(); 667 } 668 669 void IoChecker::Leave(const parser::CloseStmt &) { 670 CheckForPureSubprogram(); 671 CheckForRequiredSpecifier( 672 flags_.test(Flag::NumberUnit), "UNIT number"); // C1208 673 Done(); 674 } 675 676 void IoChecker::Leave(const parser::EndfileStmt &) { 677 CheckForPureSubprogram(); 678 CheckForRequiredSpecifier( 679 flags_.test(Flag::NumberUnit), "UNIT number"); // C1240 680 Done(); 681 } 682 683 void IoChecker::Leave(const parser::FlushStmt &) { 684 CheckForPureSubprogram(); 685 CheckForRequiredSpecifier( 686 flags_.test(Flag::NumberUnit), "UNIT number"); // C1243 687 Done(); 688 } 689 690 void IoChecker::Leave(const parser::InquireStmt &stmt) { 691 if (std::get_if<std::list<parser::InquireSpec>>(&stmt.u)) { 692 CheckForPureSubprogram(); 693 // Inquire by unit or by file (vs. by output list). 694 CheckForRequiredSpecifier( 695 flags_.test(Flag::NumberUnit) || specifierSet_.test(IoSpecKind::File), 696 "UNIT number or FILE"); // C1246 697 CheckForProhibitedSpecifier(IoSpecKind::File, IoSpecKind::Unit); // C1246 698 CheckForRequiredSpecifier(IoSpecKind::Id, IoSpecKind::Pending); // C1248 699 } 700 Done(); 701 } 702 703 void IoChecker::Leave(const parser::OpenStmt &) { 704 CheckForPureSubprogram(); 705 CheckForRequiredSpecifier(specifierSet_.test(IoSpecKind::Unit) || 706 specifierSet_.test(IoSpecKind::Newunit), 707 "UNIT or NEWUNIT"); // C1204, C1205 708 CheckForProhibitedSpecifier( 709 IoSpecKind::Newunit, IoSpecKind::Unit); // C1204, C1205 710 CheckForRequiredSpecifier(flags_.test(Flag::StatusNew), "STATUS='NEW'", 711 IoSpecKind::File); // 12.5.6.10 712 CheckForRequiredSpecifier(flags_.test(Flag::StatusReplace), 713 "STATUS='REPLACE'", IoSpecKind::File); // 12.5.6.10 714 CheckForProhibitedSpecifier(flags_.test(Flag::StatusScratch), 715 "STATUS='SCRATCH'", IoSpecKind::File); // 12.5.6.10 716 if (flags_.test(Flag::KnownStatus)) { 717 CheckForRequiredSpecifier(IoSpecKind::Newunit, 718 specifierSet_.test(IoSpecKind::File) || 719 flags_.test(Flag::StatusScratch), 720 "FILE or STATUS='SCRATCH'"); // 12.5.6.12 721 } else { 722 CheckForRequiredSpecifier(IoSpecKind::Newunit, 723 specifierSet_.test(IoSpecKind::File) || 724 specifierSet_.test(IoSpecKind::Status), 725 "FILE or STATUS"); // 12.5.6.12 726 } 727 if (flags_.test(Flag::KnownAccess)) { 728 CheckForRequiredSpecifier(flags_.test(Flag::AccessDirect), 729 "ACCESS='DIRECT'", IoSpecKind::Recl); // 12.5.6.15 730 CheckForProhibitedSpecifier(flags_.test(Flag::AccessStream), 731 "STATUS='STREAM'", IoSpecKind::Recl); // 12.5.6.15 732 } 733 Done(); 734 } 735 736 void IoChecker::Leave(const parser::PrintStmt &) { 737 CheckForPureSubprogram(); 738 Done(); 739 } 740 741 static void CheckForDoVariableInNamelist(const Symbol &namelist, 742 SemanticsContext &context, parser::CharBlock namelistLocation) { 743 const auto &details{namelist.GetUltimate().get<NamelistDetails>()}; 744 for (const Symbol &object : details.objects()) { 745 context.CheckIndexVarRedefine(namelistLocation, object); 746 } 747 } 748 749 static void CheckForDoVariableInNamelistSpec( 750 const parser::ReadStmt &readStmt, SemanticsContext &context) { 751 const std::list<parser::IoControlSpec> &controls{readStmt.controls}; 752 for (const auto &control : controls) { 753 if (const auto *namelist{std::get_if<parser::Name>(&control.u)}) { 754 if (const Symbol * symbol{namelist->symbol}) { 755 CheckForDoVariableInNamelist(*symbol, context, namelist->source); 756 } 757 } 758 } 759 } 760 761 static void CheckForDoVariable( 762 const parser::ReadStmt &readStmt, SemanticsContext &context) { 763 CheckForDoVariableInNamelistSpec(readStmt, context); 764 const std::list<parser::InputItem> &items{readStmt.items}; 765 for (const auto &item : items) { 766 if (const parser::Variable * 767 variable{std::get_if<parser::Variable>(&item.u)}) { 768 context.CheckIndexVarRedefine(*variable); 769 } 770 } 771 } 772 773 void IoChecker::Leave(const parser::ReadStmt &readStmt) { 774 if (!flags_.test(Flag::InternalUnit)) { 775 CheckForPureSubprogram(); 776 } 777 CheckForDoVariable(readStmt, context_); 778 if (!flags_.test(Flag::IoControlList)) { 779 Done(); 780 return; 781 } 782 LeaveReadWrite(); 783 CheckForProhibitedSpecifier(IoSpecKind::Delim); // C1212 784 CheckForProhibitedSpecifier(IoSpecKind::Sign); // C1212 785 CheckForProhibitedSpecifier(IoSpecKind::Rec, IoSpecKind::End); // C1220 786 CheckForRequiredSpecifier(IoSpecKind::Eor, 787 specifierSet_.test(IoSpecKind::Advance) && !flags_.test(Flag::AdvanceYes), 788 "ADVANCE with value 'NO'"); // C1222 + 12.6.2.1p2 789 CheckForRequiredSpecifier(IoSpecKind::Blank, flags_.test(Flag::FmtOrNml), 790 "FMT or NML"); // C1227 791 CheckForRequiredSpecifier( 792 IoSpecKind::Pad, flags_.test(Flag::FmtOrNml), "FMT or NML"); // C1227 793 Done(); 794 } 795 796 void IoChecker::Leave(const parser::RewindStmt &) { 797 CheckForRequiredSpecifier( 798 flags_.test(Flag::NumberUnit), "UNIT number"); // C1240 799 CheckForPureSubprogram(); 800 Done(); 801 } 802 803 void IoChecker::Leave(const parser::WaitStmt &) { 804 CheckForRequiredSpecifier( 805 flags_.test(Flag::NumberUnit), "UNIT number"); // C1237 806 CheckForPureSubprogram(); 807 Done(); 808 } 809 810 void IoChecker::Leave(const parser::WriteStmt &) { 811 if (!flags_.test(Flag::InternalUnit)) { 812 CheckForPureSubprogram(); 813 } 814 LeaveReadWrite(); 815 CheckForProhibitedSpecifier(IoSpecKind::Blank); // C1213 816 CheckForProhibitedSpecifier(IoSpecKind::End); // C1213 817 CheckForProhibitedSpecifier(IoSpecKind::Eor); // C1213 818 CheckForProhibitedSpecifier(IoSpecKind::Pad); // C1213 819 CheckForProhibitedSpecifier(IoSpecKind::Size); // C1213 820 CheckForRequiredSpecifier( 821 IoSpecKind::Sign, flags_.test(Flag::FmtOrNml), "FMT or NML"); // C1227 822 CheckForRequiredSpecifier(IoSpecKind::Delim, 823 flags_.test(Flag::StarFmt) || specifierSet_.test(IoSpecKind::Nml), 824 "FMT=* or NML"); // C1228 825 Done(); 826 } 827 828 void IoChecker::LeaveReadWrite() const { 829 CheckForRequiredSpecifier(IoSpecKind::Unit); // C1211 830 CheckForProhibitedSpecifier(IoSpecKind::Nml, IoSpecKind::Rec); // C1216 831 CheckForProhibitedSpecifier(IoSpecKind::Nml, IoSpecKind::Fmt); // C1216 832 CheckForProhibitedSpecifier( 833 IoSpecKind::Nml, flags_.test(Flag::DataList), "a data list"); // C1216 834 CheckForProhibitedSpecifier(flags_.test(Flag::InternalUnit), 835 "UNIT=internal-file", IoSpecKind::Pos); // C1219 836 CheckForProhibitedSpecifier(flags_.test(Flag::InternalUnit), 837 "UNIT=internal-file", IoSpecKind::Rec); // C1219 838 CheckForProhibitedSpecifier( 839 flags_.test(Flag::StarUnit), "UNIT=*", IoSpecKind::Pos); // C1219 840 CheckForProhibitedSpecifier( 841 flags_.test(Flag::StarUnit), "UNIT=*", IoSpecKind::Rec); // C1219 842 CheckForProhibitedSpecifier( 843 IoSpecKind::Rec, flags_.test(Flag::StarFmt), "FMT=*"); // C1220 844 CheckForRequiredSpecifier(IoSpecKind::Advance, 845 flags_.test(Flag::CharFmt) || flags_.test(Flag::LabelFmt) || 846 flags_.test(Flag::AssignFmt), 847 "an explicit format"); // C1221 848 CheckForProhibitedSpecifier(IoSpecKind::Advance, 849 flags_.test(Flag::InternalUnit), "UNIT=internal-file"); // C1221 850 CheckForRequiredSpecifier(flags_.test(Flag::AsynchronousYes), 851 "ASYNCHRONOUS='YES'", flags_.test(Flag::NumberUnit), 852 "UNIT=number"); // C1224 853 CheckForRequiredSpecifier(IoSpecKind::Id, flags_.test(Flag::AsynchronousYes), 854 "ASYNCHRONOUS='YES'"); // C1225 855 CheckForProhibitedSpecifier(IoSpecKind::Pos, IoSpecKind::Rec); // C1226 856 CheckForRequiredSpecifier(IoSpecKind::Decimal, flags_.test(Flag::FmtOrNml), 857 "FMT or NML"); // C1227 858 CheckForRequiredSpecifier(IoSpecKind::Round, flags_.test(Flag::FmtOrNml), 859 "FMT or NML"); // C1227 860 } 861 862 void IoChecker::SetSpecifier(IoSpecKind specKind) { 863 if (stmt_ == IoStmtKind::None) { 864 // FMT may appear on PRINT statements, which don't have any checks. 865 // [IO]MSG and [IO]STAT parse symbols are shared with non-I/O statements. 866 return; 867 } 868 // C1203, C1207, C1210, C1236, C1239, C1242, C1245 869 if (specifierSet_.test(specKind)) { 870 context_.Say("Duplicate %s specifier"_err_en_US, 871 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 872 } 873 specifierSet_.set(specKind); 874 } 875 876 void IoChecker::CheckStringValue(IoSpecKind specKind, const std::string &value, 877 const parser::CharBlock &source) const { 878 static std::unordered_map<IoSpecKind, const std::set<std::string>> specValues{ 879 {IoSpecKind::Access, {"DIRECT", "SEQUENTIAL", "STREAM"}}, 880 {IoSpecKind::Action, {"READ", "READWRITE", "WRITE"}}, 881 {IoSpecKind::Advance, {"NO", "YES"}}, 882 {IoSpecKind::Asynchronous, {"NO", "YES"}}, 883 {IoSpecKind::Blank, {"NULL", "ZERO"}}, 884 {IoSpecKind::Decimal, {"COMMA", "POINT"}}, 885 {IoSpecKind::Delim, {"APOSTROPHE", "NONE", "QUOTE"}}, 886 {IoSpecKind::Encoding, {"DEFAULT", "UTF-8"}}, 887 {IoSpecKind::Form, {"FORMATTED", "UNFORMATTED"}}, 888 {IoSpecKind::Pad, {"NO", "YES"}}, 889 {IoSpecKind::Position, {"APPEND", "ASIS", "REWIND"}}, 890 {IoSpecKind::Round, 891 {"COMPATIBLE", "DOWN", "NEAREST", "PROCESSOR_DEFINED", "UP", "ZERO"}}, 892 {IoSpecKind::Sign, {"PLUS", "PROCESSOR_DEFINED", "SUPPRESS"}}, 893 {IoSpecKind::Status, 894 // Open values; Close values are {"DELETE", "KEEP"}. 895 {"NEW", "OLD", "REPLACE", "SCRATCH", "UNKNOWN"}}, 896 {IoSpecKind::Carriagecontrol, {"LIST", "FORTRAN", "NONE"}}, 897 {IoSpecKind::Convert, {"BIG_ENDIAN", "LITTLE_ENDIAN", "NATIVE"}}, 898 {IoSpecKind::Dispose, {"DELETE", "KEEP"}}, 899 }; 900 auto upper{Normalize(value)}; 901 if (specValues.at(specKind).count(upper) == 0) { 902 if (specKind == IoSpecKind::Access && upper == "APPEND") { 903 if (context_.languageFeatures().ShouldWarn( 904 common::LanguageFeature::OpenAccessAppend)) { 905 context_.Say(source, 906 "ACCESS='%s' interpreted as POSITION='%s'"_port_en_US, value, 907 upper); 908 } 909 } else { 910 context_.Say(source, "Invalid %s value '%s'"_err_en_US, 911 parser::ToUpperCaseLetters(common::EnumToString(specKind)), value); 912 } 913 } 914 } 915 916 // CheckForRequiredSpecifier and CheckForProhibitedSpecifier functions 917 // need conditions to check, and string arguments to insert into a message. 918 // An IoSpecKind provides both an absence/presence condition and a string 919 // argument (its name). A (condition, string) pair provides an arbitrary 920 // condition and an arbitrary string. 921 922 void IoChecker::CheckForRequiredSpecifier(IoSpecKind specKind) const { 923 if (!specifierSet_.test(specKind)) { 924 context_.Say("%s statement must have a %s specifier"_err_en_US, 925 parser::ToUpperCaseLetters(common::EnumToString(stmt_)), 926 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 927 } 928 } 929 930 void IoChecker::CheckForRequiredSpecifier( 931 bool condition, const std::string &s) const { 932 if (!condition) { 933 context_.Say("%s statement must have a %s specifier"_err_en_US, 934 parser::ToUpperCaseLetters(common::EnumToString(stmt_)), s); 935 } 936 } 937 938 void IoChecker::CheckForRequiredSpecifier( 939 IoSpecKind specKind1, IoSpecKind specKind2) const { 940 if (specifierSet_.test(specKind1) && !specifierSet_.test(specKind2)) { 941 context_.Say("If %s appears, %s must also appear"_err_en_US, 942 parser::ToUpperCaseLetters(common::EnumToString(specKind1)), 943 parser::ToUpperCaseLetters(common::EnumToString(specKind2))); 944 } 945 } 946 947 void IoChecker::CheckForRequiredSpecifier( 948 IoSpecKind specKind, bool condition, const std::string &s) const { 949 if (specifierSet_.test(specKind) && !condition) { 950 context_.Say("If %s appears, %s must also appear"_err_en_US, 951 parser::ToUpperCaseLetters(common::EnumToString(specKind)), s); 952 } 953 } 954 955 void IoChecker::CheckForRequiredSpecifier( 956 bool condition, const std::string &s, IoSpecKind specKind) const { 957 if (condition && !specifierSet_.test(specKind)) { 958 context_.Say("If %s appears, %s must also appear"_err_en_US, s, 959 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 960 } 961 } 962 963 void IoChecker::CheckForRequiredSpecifier(bool condition1, 964 const std::string &s1, bool condition2, const std::string &s2) const { 965 if (condition1 && !condition2) { 966 context_.Say("If %s appears, %s must also appear"_err_en_US, s1, s2); 967 } 968 } 969 970 void IoChecker::CheckForProhibitedSpecifier(IoSpecKind specKind) const { 971 if (specifierSet_.test(specKind)) { 972 context_.Say("%s statement must not have a %s specifier"_err_en_US, 973 parser::ToUpperCaseLetters(common::EnumToString(stmt_)), 974 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 975 } 976 } 977 978 void IoChecker::CheckForProhibitedSpecifier( 979 IoSpecKind specKind1, IoSpecKind specKind2) const { 980 if (specifierSet_.test(specKind1) && specifierSet_.test(specKind2)) { 981 context_.Say("If %s appears, %s must not appear"_err_en_US, 982 parser::ToUpperCaseLetters(common::EnumToString(specKind1)), 983 parser::ToUpperCaseLetters(common::EnumToString(specKind2))); 984 } 985 } 986 987 void IoChecker::CheckForProhibitedSpecifier( 988 IoSpecKind specKind, bool condition, const std::string &s) const { 989 if (specifierSet_.test(specKind) && condition) { 990 context_.Say("If %s appears, %s must not appear"_err_en_US, 991 parser::ToUpperCaseLetters(common::EnumToString(specKind)), s); 992 } 993 } 994 995 void IoChecker::CheckForProhibitedSpecifier( 996 bool condition, const std::string &s, IoSpecKind specKind) const { 997 if (condition && specifierSet_.test(specKind)) { 998 context_.Say("If %s appears, %s must not appear"_err_en_US, s, 999 parser::ToUpperCaseLetters(common::EnumToString(specKind))); 1000 } 1001 } 1002 1003 template <typename A> 1004 void IoChecker::CheckForDefinableVariable( 1005 const A &variable, const std::string &s) const { 1006 if (const auto *var{parser::Unwrap<parser::Variable>(variable)}) { 1007 if (auto expr{AnalyzeExpr(context_, *var)}) { 1008 auto at{var->GetSource()}; 1009 if (auto whyNot{WhyNotDefinable(at, context_.FindScope(at), 1010 DefinabilityFlags{DefinabilityFlag::VectorSubscriptIsOk}, 1011 *expr)}) { 1012 const Symbol *base{GetFirstSymbol(*expr)}; 1013 context_ 1014 .Say(at, "%s variable '%s' is not definable"_err_en_US, s, 1015 (base ? base->name() : at).ToString()) 1016 .Attach(std::move(*whyNot)); 1017 } 1018 } 1019 } 1020 } 1021 1022 void IoChecker::CheckForPureSubprogram() const { // C1597 1023 CHECK(context_.location()); 1024 if (const Scope * 1025 scope{context_.globalScope().FindScope(*context_.location())}) { 1026 if (FindPureProcedureContaining(*scope)) { 1027 context_.Say( 1028 "External I/O is not allowed in a pure subprogram"_err_en_US); 1029 } 1030 } 1031 } 1032 1033 // Fortran 2018, 12.6.3 paragraph 7 1034 void IoChecker::CheckForBadIoComponent(const SomeExpr &expr, 1035 GenericKind::DefinedIo which, parser::CharBlock where) const { 1036 if (auto type{expr.GetType()}) { 1037 if (type->category() == TypeCategory::Derived && 1038 !type->IsUnlimitedPolymorphic()) { 1039 if (const Symbol * 1040 bad{FindUnsafeIoDirectComponent( 1041 which, type->GetDerivedTypeSpec(), &context_.FindScope(where))}) { 1042 context_.SayWithDecl(*bad, where, 1043 "Derived type in I/O cannot have an allocatable or pointer direct component unless using defined I/O"_err_en_US); 1044 } 1045 } 1046 } 1047 } 1048 1049 } // namespace Fortran::semantics 1050