From 39c38c21527340f2347ac318bf43bfa6d0fc318d Mon Sep 17 00:00:00 2001 From: peter klausler Date: Tue, 10 Aug 2021 13:13:50 -0700 Subject: [PATCH] [flang] Fix list-directed plural repeated null values at end of record A repeated null value at the end of an input record with a count > 1 would incorrectly advance to the next record when resumed. Fix. Improve some poor naming and code flow noticed while debugging, so next time will be easier. Extend a unit test to check this case. Differential Revision: https://reviews.llvm.org/D107917 --- flang/runtime/io-stmt.cpp | 42 +++++++++-------------- flang/runtime/io-stmt.h | 8 ++--- flang/unittests/Runtime/ListInputTest.cpp | 10 +++--- 3 files changed, 27 insertions(+), 33 deletions(-) diff --git a/flang/runtime/io-stmt.cpp b/flang/runtime/io-stmt.cpp index 56ea6129d550..084fc781673e 100644 --- a/flang/runtime/io-stmt.cpp +++ b/flang/runtime/io-stmt.cpp @@ -653,13 +653,13 @@ ListDirectedStatementState::GetNextDataEdit( comma = ';'; } if (remaining_ > 0 && !realPart_) { // "r*c" repetition in progress - while (connection.currentRecordNumber > initialRecordNumber_) { + while (connection.currentRecordNumber > repeatRecordNumber_) { io.BackspaceRecord(); } - connection.HandleAbsolutePosition(initialPositionInRecord_); + connection.HandleAbsolutePosition(repeatPositionInRecord_); if (!imaginaryPart_) { edit.repeat = std::min(remaining_, maxRepeat); - auto ch{io.GetNextNonBlank()}; + auto ch{io.GetCurrentChar()}; if (!ch || *ch == ' ' || *ch == '\t' || *ch == comma) { // "r*" repeated null edit.descriptor = DataEdit::ListDirectedNullValue; @@ -669,7 +669,6 @@ ListDirectedStatementState::GetNextDataEdit( return edit; } // Skip separators, handle a "r*c" repeat count; see 13.10.2 in Fortran 2018 - auto ch{io.GetNextNonBlank()}; if (imaginaryPart_) { imaginaryPart_ = false; } else if (realPart_) { @@ -677,6 +676,15 @@ ListDirectedStatementState::GetNextDataEdit( imaginaryPart_ = true; edit.descriptor = DataEdit::ListDirectedImaginaryPart; } + auto ch{io.GetNextNonBlank()}; + if (ch && *ch == comma && eatComma_) { + // Consume comma & whitespace after previous item. + // This includes the comma between real and imaginary components + // in list-directed/NAMELIST complex input. + io.HandleRelativePosition(1); + ch = io.GetNextNonBlank(); + } + eatComma_ = true; if (!ch) { return std::nullopt; } @@ -685,25 +693,9 @@ ListDirectedStatementState::GetNextDataEdit( edit.descriptor = DataEdit::ListDirectedNullValue; return edit; } - bool isFirstItem{isFirstItem_}; - isFirstItem_ = false; - if (*ch == comma) { - if (isFirstItem) { - edit.descriptor = DataEdit::ListDirectedNullValue; - return edit; - } - // Consume comma & whitespace after previous item. - // This includes the comma between real and imaginary components - // in list-directed/NAMELIST complex input. - io.HandleRelativePosition(1); - ch = io.GetNextNonBlank(); - if (!ch) { - return std::nullopt; - } - if (*ch == comma || *ch == '/') { - edit.descriptor = DataEdit::ListDirectedNullValue; - return edit; - } + if (*ch == comma) { // separator: null value + edit.descriptor = DataEdit::ListDirectedNullValue; + return edit; } if (imaginaryPart_) { // can't repeat components return edit; @@ -734,8 +726,8 @@ ListDirectedStatementState::GetNextDataEdit( } edit.repeat = std::min(r, maxRepeat); remaining_ = r - edit.repeat; - initialRecordNumber_ = connection.currentRecordNumber; - initialPositionInRecord_ = connection.positionInRecord; + repeatRecordNumber_ = connection.currentRecordNumber; + repeatPositionInRecord_ = connection.positionInRecord; } else { // not a repetition count, just an integer value; rewind connection.positionInRecord = start; } diff --git a/flang/runtime/io-stmt.h b/flang/runtime/io-stmt.h index eeae5105a13c..cfbafe2995a9 100644 --- a/flang/runtime/io-stmt.h +++ b/flang/runtime/io-stmt.h @@ -223,15 +223,15 @@ public: // successive NAMELIST input item. void ResetForNextNamelistItem() { remaining_ = 0; - isFirstItem_ = true; + eatComma_ = false; realPart_ = imaginaryPart_ = false; } private: int remaining_{0}; // for "r*" repetition - std::int64_t initialRecordNumber_; - std::int64_t initialPositionInRecord_; - bool isFirstItem_{true}; // leading separator implies null first item + std::int64_t repeatRecordNumber_; + std::int64_t repeatPositionInRecord_; + bool eatComma_{false}; // consume comma after previously read item bool hitSlash_{false}; // once '/' is seen, nullify further items bool realPart_{false}; bool imaginaryPart_{false}; diff --git a/flang/unittests/Runtime/ListInputTest.cpp b/flang/unittests/Runtime/ListInputTest.cpp index 7aa42905c436..7bcfa0f6c67d 100644 --- a/flang/unittests/Runtime/ListInputTest.cpp +++ b/flang/unittests/Runtime/ListInputTest.cpp @@ -73,7 +73,7 @@ TEST(InputTest, TestListInputIntegerList) { char buffer[numBuffers][maxBufferLength]; int j{0}; SetCharacter(buffer[j++], maxBufferLength, "1 2 2*3 ,"); - SetCharacter(buffer[j++], maxBufferLength, ",6,,8,1*"); + SetCharacter(buffer[j++], maxBufferLength, ",6,,8,2*"); StaticDescriptor<1> staticDescriptor; Descriptor &whole{staticDescriptor.descriptor()}; @@ -83,13 +83,15 @@ TEST(InputTest, TestListInputIntegerList) { whole.Check(); auto *cookie{IONAME(BeginInternalArrayListInput)(whole)}; - constexpr int listInputLength{9}; + constexpr int listInputLength{10}; // Negative numbers will be overwritten by _expectedOutput_, and positive // numbers will not be as their indices are "Null values" of the Fortran 2018 // standard 13.10.3.2 in the format strings _buffer_ - std::int64_t actualOutput[listInputLength]{-1, -2, -3, -4, 5, -6, 7, -8, 9}; - const std::int64_t expectedOutput[listInputLength]{1, 2, 3, 3, 5, 6, 7, 8, 9}; + std::int64_t actualOutput[listInputLength]{ + -1, -2, -3, -4, 5, -6, 7, -8, 9, 10}; + const std::int64_t expectedOutput[listInputLength]{ + 1, 2, 3, 3, 5, 6, 7, 8, 9, 10}; for (j = 0; j < listInputLength; ++j) { IONAME(InputInteger)(cookie, actualOutput[j]); }