2021-08-25 15:51:48 +08:00
|
|
|
//===-- runtime/command.cpp -----------------------------------------------===//
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2021-09-02 07:00:53 +08:00
|
|
|
#include "flang/Runtime/command.h"
|
2021-08-25 15:51:48 +08:00
|
|
|
#include "environment.h"
|
2021-09-28 20:17:34 +08:00
|
|
|
#include "stat.h"
|
2021-10-12 20:40:48 +08:00
|
|
|
#include "terminator.h"
|
2022-02-02 19:04:34 +08:00
|
|
|
#include "tools.h"
|
2021-09-28 20:17:34 +08:00
|
|
|
#include "flang/Runtime/descriptor.h"
|
2021-10-07 21:44:14 +08:00
|
|
|
#include <cstdlib>
|
2021-09-03 17:42:04 +08:00
|
|
|
#include <limits>
|
2021-08-25 15:51:48 +08:00
|
|
|
|
|
|
|
namespace Fortran::runtime {
|
2021-09-03 17:42:04 +08:00
|
|
|
std::int32_t RTNAME(ArgumentCount)() {
|
2021-08-25 15:51:48 +08:00
|
|
|
int argc{executionEnvironment.argc};
|
|
|
|
if (argc > 1) {
|
|
|
|
// C counts the command name as one of the arguments, but Fortran doesn't.
|
|
|
|
return argc - 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2021-09-03 17:42:04 +08:00
|
|
|
|
2021-10-27 17:09:31 +08:00
|
|
|
// Returns the length of the \p string. Assumes \p string is valid.
|
|
|
|
static std::int64_t StringLength(const char *string) {
|
|
|
|
std::size_t length{std::strlen(string)};
|
2022-03-30 14:17:56 +08:00
|
|
|
if constexpr (sizeof(std::size_t) < sizeof(std::int64_t)) {
|
2021-09-03 17:42:04 +08:00
|
|
|
return static_cast<std::int64_t>(length);
|
|
|
|
} else {
|
|
|
|
std::size_t max{std::numeric_limits<std::int64_t>::max()};
|
|
|
|
return length > max ? 0 // Just fail.
|
|
|
|
: static_cast<std::int64_t>(length);
|
|
|
|
}
|
|
|
|
}
|
2021-09-28 20:17:34 +08:00
|
|
|
|
|
|
|
static bool IsValidCharDescriptor(const Descriptor *value) {
|
|
|
|
return value && value->IsAllocated() &&
|
|
|
|
value->type() == TypeCode(TypeCategory::Character, 1) &&
|
|
|
|
value->rank() == 0;
|
|
|
|
}
|
|
|
|
|
2022-02-02 19:04:34 +08:00
|
|
|
static bool IsValidIntDescriptor(const Descriptor *length) {
|
|
|
|
auto typeCode{length->type().GetCategoryAndKind()};
|
|
|
|
// Check that our descriptor is allocated and is a scalar integer with
|
|
|
|
// kind != 1 (i.e. with a large enough decimal exponent range).
|
|
|
|
return length->IsAllocated() && length->rank() == 0 &&
|
|
|
|
length->type().IsInteger() && typeCode && typeCode->second != 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void FillWithSpaces(const Descriptor &value, std::size_t offset = 0) {
|
|
|
|
if (offset < value.ElementBytes()) {
|
|
|
|
std::memset(
|
|
|
|
value.OffsetElement(offset), ' ', value.ElementBytes() - offset);
|
|
|
|
}
|
2021-09-28 20:17:34 +08:00
|
|
|
}
|
|
|
|
|
2021-10-27 17:09:31 +08:00
|
|
|
static std::int32_t CopyToDescriptor(const Descriptor &value,
|
2022-02-02 19:04:34 +08:00
|
|
|
const char *rawValue, std::int64_t rawValueLength, const Descriptor *errmsg,
|
|
|
|
std::size_t offset = 0) {
|
|
|
|
|
|
|
|
std::int64_t toCopy{std::min(rawValueLength,
|
|
|
|
static_cast<std::int64_t>(value.ElementBytes() - offset))};
|
|
|
|
if (toCopy < 0) {
|
|
|
|
return ToErrmsg(errmsg, StatValueTooShort);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::memcpy(value.OffsetElement(offset), rawValue, toCopy);
|
2021-10-27 17:09:31 +08:00
|
|
|
|
|
|
|
if (rawValueLength > toCopy) {
|
|
|
|
return ToErrmsg(errmsg, StatValueTooShort);
|
|
|
|
}
|
|
|
|
|
|
|
|
return StatOk;
|
|
|
|
}
|
|
|
|
|
2022-02-02 19:04:34 +08:00
|
|
|
static std::int32_t CheckAndCopyToDescriptor(const Descriptor *value,
|
|
|
|
const char *rawValue, const Descriptor *errmsg, std::size_t &offset) {
|
|
|
|
bool haveValue{IsValidCharDescriptor(value)};
|
|
|
|
|
|
|
|
std::int64_t len{StringLength(rawValue)};
|
|
|
|
if (len <= 0) {
|
|
|
|
if (haveValue) {
|
|
|
|
FillWithSpaces(*value);
|
|
|
|
}
|
|
|
|
return ToErrmsg(errmsg, StatMissingArgument);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::int32_t stat{StatOk};
|
|
|
|
if (haveValue) {
|
|
|
|
stat = CopyToDescriptor(*value, rawValue, len, errmsg, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
offset += len;
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|
2022-07-01 19:43:20 +08:00
|
|
|
static void StoreLengthToDescriptor(
|
|
|
|
const Descriptor *length, std::int64_t value, Terminator &terminator) {
|
|
|
|
auto typeCode{length->type().GetCategoryAndKind()};
|
|
|
|
int kind{typeCode->second};
|
|
|
|
Fortran::runtime::ApplyIntegerKind<Fortran::runtime::StoreIntegerAt, void>(
|
|
|
|
kind, terminator, *length, /* atIndex = */ 0, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <int KIND> struct FitsInIntegerKind {
|
|
|
|
bool operator()(std::int64_t value) {
|
|
|
|
return value <= std::numeric_limits<Fortran::runtime::CppTypeFor<
|
|
|
|
Fortran::common::TypeCategory::Integer, KIND>>::max();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool FitsInDescriptor(
|
|
|
|
const Descriptor *length, std::int64_t value, Terminator &terminator) {
|
|
|
|
auto typeCode{length->type().GetCategoryAndKind()};
|
|
|
|
int kind{typeCode->second};
|
|
|
|
return Fortran::runtime::ApplyIntegerKind<FitsInIntegerKind, bool>(
|
|
|
|
kind, terminator, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::int32_t RTNAME(GetCommandArgument)(std::int32_t n, const Descriptor *value,
|
|
|
|
const Descriptor *length, const Descriptor *errmsg, const char *sourceFile,
|
|
|
|
int line) {
|
|
|
|
Terminator terminator{sourceFile, line};
|
|
|
|
|
|
|
|
if (value) {
|
|
|
|
RUNTIME_CHECK(terminator, IsValidCharDescriptor(value));
|
2022-02-02 19:04:34 +08:00
|
|
|
FillWithSpaces(*value);
|
2021-09-28 20:17:34 +08:00
|
|
|
}
|
|
|
|
|
2022-07-01 19:43:20 +08:00
|
|
|
// Store 0 in case we error out later on.
|
|
|
|
if (length) {
|
|
|
|
RUNTIME_CHECK(terminator, IsValidIntDescriptor(length));
|
|
|
|
StoreLengthToDescriptor(length, 0, terminator);
|
|
|
|
}
|
|
|
|
|
2021-09-28 20:17:34 +08:00
|
|
|
if (n < 0 || n >= executionEnvironment.argc) {
|
|
|
|
return ToErrmsg(errmsg, StatInvalidArgumentNumber);
|
|
|
|
}
|
|
|
|
|
2022-07-01 19:43:20 +08:00
|
|
|
const char *arg{executionEnvironment.argv[n]};
|
|
|
|
std::int64_t argLen{StringLength(arg)};
|
|
|
|
if (argLen <= 0) {
|
|
|
|
return ToErrmsg(errmsg, StatMissingArgument);
|
|
|
|
}
|
2021-09-28 20:17:34 +08:00
|
|
|
|
2022-07-01 19:43:20 +08:00
|
|
|
if (length && FitsInDescriptor(length, argLen, terminator)) {
|
|
|
|
StoreLengthToDescriptor(length, argLen, terminator);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value) {
|
2021-10-27 17:09:31 +08:00
|
|
|
return CopyToDescriptor(*value, arg, argLen, errmsg);
|
2021-09-28 20:17:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return StatOk;
|
|
|
|
}
|
2021-10-07 21:44:14 +08:00
|
|
|
|
2022-02-02 19:04:34 +08:00
|
|
|
std::int32_t RTNAME(GetCommand)(const Descriptor *value,
|
|
|
|
const Descriptor *length, const Descriptor *errmsg, const char *sourceFile,
|
|
|
|
int line) {
|
|
|
|
Terminator terminator{sourceFile, line};
|
|
|
|
|
|
|
|
if (value) {
|
|
|
|
RUNTIME_CHECK(terminator, IsValidCharDescriptor(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store 0 in case we error out later on.
|
|
|
|
if (length) {
|
|
|
|
RUNTIME_CHECK(terminator, IsValidIntDescriptor(length));
|
2022-07-01 19:43:20 +08:00
|
|
|
StoreLengthToDescriptor(length, 0, terminator);
|
2022-02-02 19:04:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
auto shouldContinue = [&](std::int32_t stat) -> bool {
|
|
|
|
// We continue as long as everything is ok OR the value descriptor is
|
|
|
|
// too short, but we still need to compute the length.
|
|
|
|
return stat == StatOk || (length && stat == StatValueTooShort);
|
|
|
|
};
|
|
|
|
|
|
|
|
std::size_t offset{0};
|
|
|
|
|
|
|
|
if (executionEnvironment.argc == 0) {
|
|
|
|
return CheckAndCopyToDescriptor(value, "", errmsg, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
// value = argv[0]
|
|
|
|
std::int32_t stat{CheckAndCopyToDescriptor(
|
|
|
|
value, executionEnvironment.argv[0], errmsg, offset)};
|
|
|
|
if (!shouldContinue(stat)) {
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|
|
|
|
// value += " " + argv[1:n]
|
|
|
|
for (std::int32_t i{1}; i < executionEnvironment.argc; ++i) {
|
|
|
|
stat = CheckAndCopyToDescriptor(value, " ", errmsg, offset);
|
|
|
|
if (!shouldContinue(stat)) {
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|
|
|
|
stat = CheckAndCopyToDescriptor(
|
|
|
|
value, executionEnvironment.argv[i], errmsg, offset);
|
|
|
|
if (!shouldContinue(stat)) {
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-01 19:43:20 +08:00
|
|
|
if (length && FitsInDescriptor(length, offset, terminator)) {
|
|
|
|
StoreLengthToDescriptor(length, offset, terminator);
|
2022-02-02 19:04:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// value += spaces for padding
|
|
|
|
if (value) {
|
|
|
|
FillWithSpaces(*value, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|
2021-10-07 21:44:14 +08:00
|
|
|
static std::size_t LengthWithoutTrailingSpaces(const Descriptor &d) {
|
|
|
|
std::size_t s{d.ElementBytes() - 1};
|
|
|
|
while (*d.OffsetElement(s) == ' ') {
|
|
|
|
--s;
|
|
|
|
}
|
|
|
|
return s + 1;
|
|
|
|
}
|
|
|
|
|
2021-10-27 17:09:31 +08:00
|
|
|
static const char *GetEnvVariableValue(
|
2021-10-12 20:40:48 +08:00
|
|
|
const Descriptor &name, bool trim_name, const char *sourceFile, int line) {
|
2021-10-07 21:44:14 +08:00
|
|
|
std::size_t nameLength{
|
|
|
|
trim_name ? LengthWithoutTrailingSpaces(name) : name.ElementBytes()};
|
|
|
|
if (nameLength == 0) {
|
2021-10-27 17:09:31 +08:00
|
|
|
return nullptr;
|
2021-10-07 21:44:14 +08:00
|
|
|
}
|
|
|
|
|
2021-10-12 20:40:48 +08:00
|
|
|
Terminator terminator{sourceFile, line};
|
|
|
|
const char *value{executionEnvironment.GetEnv(
|
|
|
|
name.OffsetElement(), nameLength, terminator)};
|
2021-10-27 17:09:31 +08:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::int32_t RTNAME(EnvVariableValue)(const Descriptor &name,
|
|
|
|
const Descriptor *value, bool trim_name, const Descriptor *errmsg,
|
|
|
|
const char *sourceFile, int line) {
|
|
|
|
if (IsValidCharDescriptor(value)) {
|
2022-02-02 19:04:34 +08:00
|
|
|
FillWithSpaces(*value);
|
2021-10-27 17:09:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *rawValue{GetEnvVariableValue(name, trim_name, sourceFile, line)};
|
|
|
|
if (!rawValue) {
|
|
|
|
return ToErrmsg(errmsg, StatMissingEnvVariable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IsValidCharDescriptor(value)) {
|
|
|
|
return CopyToDescriptor(*value, rawValue, StringLength(rawValue), errmsg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return StatOk;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::int64_t RTNAME(EnvVariableLength)(
|
|
|
|
const Descriptor &name, bool trim_name, const char *sourceFile, int line) {
|
|
|
|
const char *value{GetEnvVariableValue(name, trim_name, sourceFile, line)};
|
2021-10-07 21:44:14 +08:00
|
|
|
if (!value) {
|
|
|
|
return 0;
|
|
|
|
}
|
2021-10-27 17:09:31 +08:00
|
|
|
return StringLength(value);
|
2021-10-07 21:44:14 +08:00
|
|
|
}
|
2021-08-25 15:51:48 +08:00
|
|
|
} // namespace Fortran::runtime
|