2020-01-28 10:18:45 +08:00
|
|
|
//===-- runtime/io-error.cpp ------------------------------------*- C++ -*-===//
|
2020-01-17 05:51:25 +08:00
|
|
|
//
|
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "io-error.h"
|
2020-03-29 12:00:16 +08:00
|
|
|
#include "config.h"
|
2020-01-17 05:51:25 +08:00
|
|
|
#include "magic-numbers.h"
|
2020-02-14 06:41:56 +08:00
|
|
|
#include "tools.h"
|
2020-01-17 05:51:25 +08:00
|
|
|
#include <cerrno>
|
2020-02-14 06:41:56 +08:00
|
|
|
#include <cstdarg>
|
2020-01-17 05:51:25 +08:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
namespace Fortran::runtime::io {
|
|
|
|
|
|
|
|
void IoErrorHandler::Begin(const char *sourceFileName, int sourceLine) {
|
|
|
|
flags_ = 0;
|
|
|
|
ioStat_ = 0;
|
2020-02-14 06:41:56 +08:00
|
|
|
ioMsg_.reset();
|
2020-01-17 05:51:25 +08:00
|
|
|
SetLocation(sourceFileName, sourceLine);
|
|
|
|
}
|
|
|
|
|
2020-02-14 06:41:56 +08:00
|
|
|
void IoErrorHandler::SignalError(int iostatOrErrno, const char *msg, ...) {
|
|
|
|
if (iostatOrErrno == IostatEnd && (flags_ & hasEnd)) {
|
|
|
|
if (!ioStat_ || ioStat_ < IostatEnd) {
|
|
|
|
ioStat_ = IostatEnd;
|
|
|
|
}
|
|
|
|
} else if (iostatOrErrno == IostatEor && (flags_ & hasEor)) {
|
|
|
|
if (!ioStat_ || ioStat_ < IostatEor) {
|
2020-03-29 12:00:16 +08:00
|
|
|
ioStat_ = IostatEor; // least priority
|
2020-02-14 06:41:56 +08:00
|
|
|
}
|
|
|
|
} else if (iostatOrErrno != IostatOk) {
|
|
|
|
if (flags_ & (hasIoStat | hasErr)) {
|
2020-01-24 08:10:00 +08:00
|
|
|
if (ioStat_ <= 0) {
|
2020-03-29 12:00:16 +08:00
|
|
|
ioStat_ = iostatOrErrno; // priority over END=/EOR=
|
2020-02-14 06:41:56 +08:00
|
|
|
if (msg && (flags_ & hasIoMsg)) {
|
|
|
|
char buffer[256];
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, msg);
|
|
|
|
std::vsnprintf(buffer, sizeof buffer, msg, ap);
|
|
|
|
ioMsg_ = SaveDefaultCharacter(buffer, std::strlen(buffer) + 1, *this);
|
|
|
|
}
|
2020-01-17 05:51:25 +08:00
|
|
|
}
|
2020-02-14 06:41:56 +08:00
|
|
|
} else if (msg) {
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, msg);
|
|
|
|
CrashArgs(msg, ap);
|
|
|
|
} else if (const char *errstr{IostatErrorString(iostatOrErrno)}) {
|
|
|
|
Crash(errstr);
|
2020-01-17 05:51:25 +08:00
|
|
|
} else {
|
2020-02-14 06:41:56 +08:00
|
|
|
Crash("I/O error (errno=%d): %s", iostatOrErrno,
|
|
|
|
std::strerror(iostatOrErrno));
|
2020-01-17 05:51:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-14 06:41:56 +08:00
|
|
|
void IoErrorHandler::SignalError(int iostatOrErrno) {
|
|
|
|
SignalError(iostatOrErrno, nullptr);
|
|
|
|
}
|
|
|
|
|
2020-01-24 08:10:00 +08:00
|
|
|
void IoErrorHandler::SignalErrno() { SignalError(errno); }
|
|
|
|
|
2020-02-14 06:41:56 +08:00
|
|
|
void IoErrorHandler::SignalEnd() { SignalError(IostatEnd); }
|
2020-01-17 05:51:25 +08:00
|
|
|
|
2020-02-14 06:41:56 +08:00
|
|
|
void IoErrorHandler::SignalEor() { SignalError(IostatEor); }
|
|
|
|
|
|
|
|
bool IoErrorHandler::GetIoMsg(char *buffer, std::size_t bufferLength) {
|
|
|
|
const char *msg{ioMsg_.get()};
|
|
|
|
if (!msg) {
|
|
|
|
msg = IostatErrorString(ioStat_);
|
|
|
|
}
|
|
|
|
if (msg) {
|
|
|
|
ToFortranDefaultCharacter(buffer, bufferLength, msg);
|
|
|
|
return true;
|
2020-01-17 05:51:25 +08:00
|
|
|
}
|
2020-03-13 04:28:35 +08:00
|
|
|
|
|
|
|
char *newBuf;
|
|
|
|
// Following code is taken from llvm/lib/Support/Errno.cpp
|
|
|
|
// in LLVM v9.0.1
|
|
|
|
#if HAVE_STRERROR_R
|
|
|
|
// strerror_r is thread-safe.
|
|
|
|
#if defined(__GLIBC__) && defined(_GNU_SOURCE)
|
|
|
|
// glibc defines its own incompatible version of strerror_r
|
|
|
|
// which may not use the buffer supplied.
|
|
|
|
newBuf = ::strerror_r(ioStat_, buffer, bufferLength);
|
|
|
|
#else
|
2020-02-14 06:41:56 +08:00
|
|
|
return ::strerror_r(ioStat_, buffer, bufferLength) == 0;
|
2020-03-13 04:28:35 +08:00
|
|
|
#endif
|
|
|
|
#elif HAVE_DECL_STRERROR_S // "Windows Secure API"
|
|
|
|
return ::strerror_s(buffer, bufferLength, ioStat_) == 0;
|
|
|
|
#elif HAVE_STRERROR
|
|
|
|
// Copy the thread un-safe result of strerror into
|
|
|
|
// the buffer as fast as possible to minimize impact
|
|
|
|
// of collision of strerror in multiple threads.
|
|
|
|
newBuf = strerror(ioStat_);
|
|
|
|
#else
|
|
|
|
// Strange that this system doesn't even have strerror
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
::strncpy(buffer, newBuf, bufferLength - 1);
|
2020-03-29 12:00:16 +08:00
|
|
|
buffer[bufferLength - 1] = '\n';
|
2020-03-13 04:28:35 +08:00
|
|
|
return true;
|
2020-01-17 05:51:25 +08:00
|
|
|
}
|
2020-03-29 12:00:16 +08:00
|
|
|
} // namespace Fortran::runtime::io
|