2021-09-02 16:14:01 +08:00
|
|
|
//===-- runtime/stat.cpp --------------------------------------------------===//
|
2020-11-11 07:13:02 +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 "stat.h"
|
|
|
|
#include "terminator.h"
|
2021-09-02 07:00:53 +08:00
|
|
|
#include "flang/Runtime/descriptor.h"
|
2020-11-11 07:13:02 +08:00
|
|
|
|
|
|
|
namespace Fortran::runtime {
|
|
|
|
const char *StatErrorString(int stat) {
|
|
|
|
switch (stat) {
|
|
|
|
case StatOk:
|
|
|
|
return "No error";
|
|
|
|
|
|
|
|
case StatBaseNull:
|
|
|
|
return "Base address is null";
|
|
|
|
case StatBaseNotNull:
|
|
|
|
return "Base address is not null";
|
|
|
|
case StatInvalidElemLen:
|
|
|
|
return "Invalid element length";
|
|
|
|
case StatInvalidRank:
|
|
|
|
return "Invalid rank";
|
|
|
|
case StatInvalidType:
|
|
|
|
return "Invalid type";
|
|
|
|
case StatInvalidAttribute:
|
|
|
|
return "Invalid attribute";
|
|
|
|
case StatInvalidExtent:
|
|
|
|
return "Invalid extent";
|
|
|
|
case StatInvalidDescriptor:
|
|
|
|
return "Invalid descriptor";
|
|
|
|
case StatMemAllocation:
|
|
|
|
return "Memory allocation failed";
|
|
|
|
case StatOutOfBounds:
|
|
|
|
return "Out of bounds";
|
|
|
|
|
|
|
|
case StatFailedImage:
|
|
|
|
return "Failed image";
|
|
|
|
case StatLocked:
|
|
|
|
return "Locked";
|
|
|
|
case StatLockedOtherImage:
|
|
|
|
return "Other image locked";
|
|
|
|
case StatStoppedImage:
|
|
|
|
return "Image stopped";
|
|
|
|
case StatUnlocked:
|
|
|
|
return "Unlocked";
|
|
|
|
case StatUnlockedFailedImage:
|
|
|
|
return "Failed image unlocked";
|
|
|
|
|
2021-09-28 20:17:34 +08:00
|
|
|
case StatInvalidArgumentNumber:
|
|
|
|
return "Invalid argument number";
|
|
|
|
case StatMissingArgument:
|
|
|
|
return "Missing argument";
|
|
|
|
case StatValueTooShort:
|
|
|
|
return "Value too short";
|
|
|
|
|
2021-10-27 17:09:31 +08:00
|
|
|
case StatMissingEnvVariable:
|
|
|
|
return "Missing environment variable";
|
|
|
|
|
2020-11-11 07:13:02 +08:00
|
|
|
default:
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-13 05:33:50 +08:00
|
|
|
int ToErrmsg(const Descriptor *errmsg, int stat) {
|
2020-11-11 07:13:02 +08:00
|
|
|
if (stat != StatOk && errmsg && errmsg->raw().base_addr &&
|
|
|
|
errmsg->type() == TypeCode(TypeCategory::Character, 1) &&
|
|
|
|
errmsg->rank() == 0) {
|
|
|
|
if (const char *msg{StatErrorString(stat)}) {
|
|
|
|
char *buffer{errmsg->OffsetElement()};
|
|
|
|
std::size_t bufferLength{errmsg->ElementBytes()};
|
|
|
|
std::size_t msgLength{std::strlen(msg)};
|
2021-03-13 05:33:50 +08:00
|
|
|
if (msgLength >= bufferLength) {
|
2020-11-11 07:13:02 +08:00
|
|
|
std::memcpy(buffer, msg, bufferLength);
|
|
|
|
} else {
|
|
|
|
std::memcpy(buffer, msg, msgLength);
|
|
|
|
std::memset(buffer + msgLength, ' ', bufferLength - msgLength);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ReturnError(
|
2021-03-13 05:33:50 +08:00
|
|
|
Terminator &terminator, int stat, const Descriptor *errmsg, bool hasStat) {
|
2020-11-11 07:13:02 +08:00
|
|
|
if (stat == StatOk || hasStat) {
|
|
|
|
return ToErrmsg(errmsg, stat);
|
|
|
|
} else if (const char *msg{StatErrorString(stat)}) {
|
|
|
|
terminator.Crash(msg);
|
|
|
|
} else {
|
|
|
|
terminator.Crash("Invalid Fortran runtime STAT= code %d", stat);
|
|
|
|
}
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
} // namespace Fortran::runtime
|