[llvm] [flang][runtime] Detect byte order reversal problems (PR #129093)

Peter Klausler via llvm-commits llvm-commits at lists.llvm.org
Thu Feb 27 10:11:57 PST 2025


https://github.com/klausler updated https://github.com/llvm/llvm-project/pull/129093

>From 31bceb4f1b670524fb2f0e138807dac6276811af Mon Sep 17 00:00:00 2001
From: Peter Klausler <pklausler at nvidia.com>
Date: Wed, 26 Feb 2025 14:38:05 -0800
Subject: [PATCH] [flang][runtime] Detect byte order reversal problems

When reading an unformatted sequential file with variable-length records,
detect byte order reversal problems with the first record's header and footer
words, and emit a more detailed error message.
---
 flang-rt/lib/runtime/unit.cpp | 48 ++++++++++++++++++++++++++---------
 flang-rt/lib/runtime/unit.h   |  2 +-
 2 files changed, 37 insertions(+), 13 deletions(-)

diff --git a/flang-rt/lib/runtime/unit.cpp b/flang-rt/lib/runtime/unit.cpp
index 1d4d54ae01956..43501aeb48458 100644
--- a/flang-rt/lib/runtime/unit.cpp
+++ b/flang-rt/lib/runtime/unit.cpp
@@ -511,7 +511,7 @@ void ExternalFileUnit::EndIoStatement() {
 void ExternalFileUnit::BeginSequentialVariableUnformattedInputRecord(
     IoErrorHandler &handler) {
   RUNTIME_CHECK(handler, access == Access::Sequential);
-  std::int32_t header{0}, footer{0};
+  std::uint32_t header{0}, footer{0};
   std::size_t need{recordOffsetInFrame_ + sizeof header};
   std::size_t got{ReadFrame(frameOffsetInFile_, need, handler)};
   // Try to emit informative errors to help debug corrupted files.
@@ -528,17 +528,41 @@ void ExternalFileUnit::BeginSequentialVariableUnformattedInputRecord(
     recordLength = sizeof header + header; // does not include footer
     need = recordOffsetInFrame_ + *recordLength + sizeof footer;
     got = ReadFrame(frameOffsetInFile_, need, handler);
-    if (got < need) {
+    if (got >= need) {
+      footer = ReadHeaderOrFooter(recordOffsetInFrame_ + *recordLength);
+    }
+    if (frameOffsetInFile_ == 0 && recordOffsetInFrame_ == 0 &&
+        (got < need || footer != header)) {
+      // Maybe an omitted or incorrect byte swap flag setting?
+      // Try it the other way, since this is the first record.
+      // (N.B. Won't work on files starting with empty records, but there's
+      // no good way to know later if all preceding records were empty.)
+      swapEndianness_ = !swapEndianness_;
+      std::uint32_t header2{ReadHeaderOrFooter(0)};
+      std::size_t recordLength2{sizeof header2 + header2};
+      std::size_t need2{recordLength2 + sizeof footer};
+      std::size_t got2{ReadFrame(0, need2, handler)};
+      if (got2 >= need2) {
+        std::uint32_t footer2{ReadHeaderOrFooter(recordLength2)};
+        if (footer2 == header2) {
+          error = "Unformatted variable-length sequential file input "
+                  "failed on the first record, probably due to a need "
+                  "for byte order data conversion; consider adding "
+                  "CONVERT='SWAP' to the OPEN statement or adding "
+                  "FORT_CONVERT=SWAP to the execution environment";
+        }
+      }
+      swapEndianness_ = !swapEndianness_;
+    }
+    if (error) {
+    } else if (got < need) {
       error = "Unformatted variable-length sequential file input failed at "
               "record #%jd (file offset %jd): hit EOF reading record with "
               "length %jd bytes";
-    } else {
-      footer = ReadHeaderOrFooter(recordOffsetInFrame_ + *recordLength);
-      if (footer != header) {
-        error = "Unformatted variable-length sequential file input failed at "
-                "record #%jd (file offset %jd): record header has length %jd "
-                "that does not match record footer (%jd)";
-      }
+    } else if (footer != header) {
+      error = "Unformatted variable-length sequential file input failed at "
+              "record #%jd (file offset %jd): record header has length %jd "
+              "that does not match record footer (%jd)";
     }
   }
   if (error) {
@@ -590,7 +614,7 @@ void ExternalFileUnit::BackspaceFixedRecord(IoErrorHandler &handler) {
 
 void ExternalFileUnit::BackspaceVariableUnformattedRecord(
     IoErrorHandler &handler) {
-  std::int32_t header{0};
+  std::uint32_t header{0};
   auto headerBytes{static_cast<std::int64_t>(sizeof header)};
   frameOffsetInFile_ += recordOffsetInFrame_;
   recordOffsetInFrame_ = 0;
@@ -775,8 +799,8 @@ void ExternalFileUnit::PopChildIo(ChildIo &child) {
   child_.reset(child.AcquirePrevious().release()); // deletes top child
 }
 
-std::int32_t ExternalFileUnit::ReadHeaderOrFooter(std::int64_t frameOffset) {
-  std::int32_t word;
+std::uint32_t ExternalFileUnit::ReadHeaderOrFooter(std::int64_t frameOffset) {
+  std::uint32_t word;
   char *wordPtr{reinterpret_cast<char *>(&word)};
   std::memcpy(wordPtr, Frame() + frameOffset, sizeof word);
   if (swapEndianness_) {
diff --git a/flang-rt/lib/runtime/unit.h b/flang-rt/lib/runtime/unit.h
index eb762a2d3b235..bb3d3650da34b 100644
--- a/flang-rt/lib/runtime/unit.h
+++ b/flang-rt/lib/runtime/unit.h
@@ -210,7 +210,7 @@ class ExternalFileUnit : public ConnectionState,
   RT_API_ATTRS void CommitWrites();
   RT_API_ATTRS bool CheckDirectAccess(IoErrorHandler &);
   RT_API_ATTRS void HitEndOnRead(IoErrorHandler &);
-  RT_API_ATTRS std::int32_t ReadHeaderOrFooter(std::int64_t frameOffset);
+  RT_API_ATTRS std::uint32_t ReadHeaderOrFooter(std::int64_t frameOffset);
 
   Lock lock_;
 



More information about the llvm-commits mailing list