2020-03-31 07:37:30 +08:00
|
|
|
//===-- runtime/character.cpp -----------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// 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 "character.h"
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
#include "cpp-type.h"
|
2020-04-24 08:07:45 +08:00
|
|
|
#include "descriptor.h"
|
2020-03-31 07:37:30 +08:00
|
|
|
#include "terminator.h"
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
#include "tools.h"
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
#include "flang/Common/bit-population-count.h"
|
|
|
|
#include "flang/Common/uint128.h"
|
2020-03-31 07:37:30 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
namespace Fortran::runtime {
|
2020-04-24 08:07:45 +08:00
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
template <typename CHAR>
|
|
|
|
inline int CompareToBlankPadding(const CHAR *x, std::size_t chars) {
|
2020-04-24 08:07:45 +08:00
|
|
|
for (; chars-- > 0; ++x) {
|
|
|
|
if (*x < ' ') {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (*x > ' ') {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
template <typename CHAR>
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
int CharacterScalarCompare(
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
const CHAR *x, const CHAR *y, std::size_t xChars, std::size_t yChars) {
|
|
|
|
auto minChars{std::min(xChars, yChars)};
|
|
|
|
if constexpr (sizeof(CHAR) == 1) {
|
2020-04-24 08:07:45 +08:00
|
|
|
// don't use for kind=2 or =4, that would fail on little-endian machines
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
int cmp{std::memcmp(x, y, minChars)};
|
2020-04-24 08:07:45 +08:00
|
|
|
if (cmp < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (cmp > 0) {
|
|
|
|
return 1;
|
|
|
|
}
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if (xChars == yChars) {
|
2020-04-24 08:07:45 +08:00
|
|
|
return 0;
|
|
|
|
}
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
x += minChars;
|
|
|
|
y += minChars;
|
2020-04-24 08:07:45 +08:00
|
|
|
} else {
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
for (std::size_t n{minChars}; n-- > 0; ++x, ++y) {
|
2020-04-24 08:07:45 +08:00
|
|
|
if (*x < *y) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (*x > *y) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if (int cmp{CompareToBlankPadding(x, xChars - minChars)}) {
|
2020-04-24 08:07:45 +08:00
|
|
|
return cmp;
|
|
|
|
}
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
return -CompareToBlankPadding(y, yChars - minChars);
|
|
|
|
}
|
|
|
|
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
template int CharacterScalarCompare<char>(
|
|
|
|
const char *x, const char *y, std::size_t xChars, std::size_t yChars);
|
|
|
|
template int CharacterScalarCompare<char16_t>(const char16_t *x,
|
|
|
|
const char16_t *y, std::size_t xChars, std::size_t yChars);
|
|
|
|
template int CharacterScalarCompare<char32_t>(const char32_t *x,
|
|
|
|
const char32_t *y, std::size_t xChars, std::size_t yChars);
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
// Shift count to use when converting between character lengths
|
|
|
|
// and byte counts.
|
|
|
|
template <typename CHAR>
|
|
|
|
constexpr int shift{common::TrailingZeroBitCount(sizeof(CHAR))};
|
|
|
|
|
|
|
|
template <typename CHAR>
|
|
|
|
static void Compare(Descriptor &result, const Descriptor &x,
|
|
|
|
const Descriptor &y, const Terminator &terminator) {
|
|
|
|
RUNTIME_CHECK(
|
|
|
|
terminator, x.rank() == y.rank() || x.rank() == 0 || y.rank() == 0);
|
|
|
|
int rank{std::max(x.rank(), y.rank())};
|
|
|
|
SubscriptValue lb[maxRank], ub[maxRank], xAt[maxRank], yAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lb[j] = 1;
|
|
|
|
if (x.rank() > 0 && y.rank() > 0) {
|
|
|
|
SubscriptValue xUB{x.GetDimension(j).Extent()};
|
|
|
|
SubscriptValue yUB{y.GetDimension(j).Extent()};
|
|
|
|
if (xUB != yUB) {
|
|
|
|
terminator.Crash("Character array comparison: operands are not "
|
|
|
|
"conforming on dimension %d (%jd != %jd)",
|
|
|
|
j + 1, static_cast<std::intmax_t>(xUB),
|
|
|
|
static_cast<std::intmax_t>(yUB));
|
|
|
|
}
|
|
|
|
ub[j] = xUB;
|
|
|
|
} else {
|
|
|
|
ub[j] = (x.rank() ? x : y).GetDimension(j).Extent();
|
|
|
|
}
|
|
|
|
elements *= ub[j];
|
|
|
|
xAt[j] = yAt[j] = 1;
|
|
|
|
}
|
2021-02-27 06:32:26 +08:00
|
|
|
result.Establish(
|
|
|
|
TypeCategory::Logical, 1, nullptr, rank, ub, CFI_attribute_allocatable);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if (result.Allocate(lb, ub) != CFI_SUCCESS) {
|
|
|
|
terminator.Crash("Compare: could not allocate storage for result");
|
|
|
|
}
|
|
|
|
std::size_t xChars{x.ElementBytes() >> shift<CHAR>};
|
|
|
|
std::size_t yChars{y.ElementBytes() >> shift<char>};
|
|
|
|
for (SubscriptValue resultAt{0}; elements-- > 0;
|
|
|
|
++resultAt, x.IncrementSubscripts(xAt), y.IncrementSubscripts(yAt)) {
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
*result.OffsetElement<char>(resultAt) = CharacterScalarCompare<CHAR>(
|
|
|
|
x.Element<CHAR>(xAt), y.Element<CHAR>(yAt), xChars, yChars);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename CHAR, bool ADJUSTR>
|
|
|
|
static void Adjust(CHAR *to, const CHAR *from, std::size_t chars) {
|
|
|
|
if constexpr (ADJUSTR) {
|
|
|
|
std::size_t j{chars}, k{chars};
|
|
|
|
for (; k > 0 && from[k - 1] == ' '; --k) {
|
|
|
|
}
|
|
|
|
while (k > 0) {
|
|
|
|
to[--j] = from[--k];
|
|
|
|
}
|
|
|
|
while (j > 0) {
|
|
|
|
to[--j] = ' ';
|
|
|
|
}
|
|
|
|
} else { // ADJUSTL
|
|
|
|
std::size_t j{0}, k{0};
|
|
|
|
for (; k < chars && from[k] == ' '; ++k) {
|
|
|
|
}
|
|
|
|
while (k < chars) {
|
|
|
|
to[j++] = from[k++];
|
|
|
|
}
|
|
|
|
while (j < chars) {
|
|
|
|
to[j++] = ' ';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename CHAR, bool ADJUSTR>
|
|
|
|
static void AdjustLRHelper(Descriptor &result, const Descriptor &string,
|
|
|
|
const Terminator &terminator) {
|
|
|
|
int rank{string.rank()};
|
|
|
|
SubscriptValue lb[maxRank], ub[maxRank], stringAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lb[j] = 1;
|
|
|
|
ub[j] = string.GetDimension(j).Extent();
|
|
|
|
elements *= ub[j];
|
|
|
|
stringAt[j] = 1;
|
|
|
|
}
|
|
|
|
std::size_t elementBytes{string.ElementBytes()};
|
2021-02-27 06:32:26 +08:00
|
|
|
result.Establish(string.type(), elementBytes, nullptr, rank, ub,
|
|
|
|
CFI_attribute_allocatable);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if (result.Allocate(lb, ub) != CFI_SUCCESS) {
|
|
|
|
terminator.Crash("ADJUSTL/R: could not allocate storage for result");
|
|
|
|
}
|
|
|
|
for (SubscriptValue resultAt{0}; elements-- > 0;
|
|
|
|
resultAt += elementBytes, string.IncrementSubscripts(stringAt)) {
|
|
|
|
Adjust<CHAR, ADJUSTR>(result.OffsetElement<CHAR>(resultAt),
|
|
|
|
string.Element<const CHAR>(stringAt), elementBytes >> shift<CHAR>);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <bool ADJUSTR>
|
|
|
|
void AdjustLR(Descriptor &result, const Descriptor &string,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
switch (string.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
AdjustLRHelper<char, ADJUSTR>(result, string, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
AdjustLRHelper<char16_t, ADJUSTR>(result, string, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
AdjustLRHelper<char32_t, ADJUSTR>(result, string, terminator);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash("ADJUSTL/R: bad string type code %d",
|
|
|
|
static_cast<int>(string.raw().type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename CHAR>
|
|
|
|
inline std::size_t LenTrim(const CHAR *x, std::size_t chars) {
|
|
|
|
while (chars > 0 && x[chars - 1] == ' ') {
|
|
|
|
--chars;
|
|
|
|
}
|
|
|
|
return chars;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename INT, typename CHAR>
|
|
|
|
static void LenTrim(Descriptor &result, const Descriptor &string,
|
|
|
|
const Terminator &terminator) {
|
|
|
|
int rank{string.rank()};
|
|
|
|
SubscriptValue lb[maxRank], ub[maxRank], stringAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lb[j] = 1;
|
|
|
|
ub[j] = string.GetDimension(j).Extent();
|
|
|
|
elements *= ub[j];
|
|
|
|
stringAt[j] = 1;
|
|
|
|
}
|
2021-02-27 06:32:26 +08:00
|
|
|
result.Establish(TypeCategory::Integer, sizeof(INT), nullptr, rank, ub,
|
|
|
|
CFI_attribute_allocatable);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if (result.Allocate(lb, ub) != CFI_SUCCESS) {
|
|
|
|
terminator.Crash("LEN_TRIM: could not allocate storage for result");
|
|
|
|
}
|
|
|
|
std::size_t stringElementChars{string.ElementBytes() >> shift<CHAR>};
|
|
|
|
for (SubscriptValue resultAt{0}; elements-- > 0;
|
|
|
|
resultAt += sizeof(INT), string.IncrementSubscripts(stringAt)) {
|
|
|
|
*result.OffsetElement<INT>(resultAt) =
|
|
|
|
LenTrim(string.Element<CHAR>(stringAt), stringElementChars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename CHAR>
|
|
|
|
static void LenTrimKind(Descriptor &result, const Descriptor &string, int kind,
|
|
|
|
const Terminator &terminator) {
|
|
|
|
switch (kind) {
|
|
|
|
case 1:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
LenTrim<CppTypeFor<TypeCategory::Integer, 1>, CHAR>(
|
|
|
|
result, string, terminator);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
break;
|
|
|
|
case 2:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
LenTrim<CppTypeFor<TypeCategory::Integer, 2>, CHAR>(
|
|
|
|
result, string, terminator);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
LenTrim<CppTypeFor<TypeCategory::Integer, 4>, CHAR>(
|
|
|
|
result, string, terminator);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
break;
|
|
|
|
case 8:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
LenTrim<CppTypeFor<TypeCategory::Integer, 8>, CHAR>(
|
|
|
|
result, string, terminator);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
break;
|
|
|
|
case 16:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
LenTrim<CppTypeFor<TypeCategory::Integer, 16>, CHAR>(
|
|
|
|
result, string, terminator);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash("LEN_TRIM: bad KIND=%d", kind);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-13 05:40:31 +08:00
|
|
|
// INDEX implementation
|
|
|
|
template <typename CHAR>
|
|
|
|
inline std::size_t Index(const CHAR *x, std::size_t xLen, const CHAR *want,
|
|
|
|
std::size_t wantLen, bool back) {
|
|
|
|
if (xLen < wantLen) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (xLen == 0) {
|
|
|
|
return 1; // wantLen is also 0, so trivial match
|
|
|
|
}
|
|
|
|
if (back) {
|
|
|
|
// If wantLen==0, returns xLen + 1 per standard (and all other compilers)
|
|
|
|
std::size_t at{xLen - wantLen + 1};
|
|
|
|
for (; at > 0; --at) {
|
|
|
|
std::size_t j{1};
|
|
|
|
for (; j <= wantLen; ++j) {
|
|
|
|
if (x[at + j - 2] != want[j - 1]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (j > wantLen) {
|
|
|
|
return at;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
// Non-trivial forward substring search: use a simplified form of
|
|
|
|
// Boyer-Moore substring searching.
|
|
|
|
for (std::size_t at{1}; at + wantLen - 1 <= xLen;) {
|
|
|
|
// Compare x(at:at+wantLen-1) with want(1:wantLen).
|
|
|
|
// The comparison proceeds from the ends of the substrings forward
|
|
|
|
// so that we can skip ahead by multiple positions on a miss.
|
|
|
|
std::size_t j{wantLen};
|
|
|
|
CHAR ch;
|
|
|
|
for (; j > 0; --j) {
|
|
|
|
ch = x[at + j - 2];
|
|
|
|
if (ch != want[j - 1]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (j == 0) {
|
|
|
|
return at; // found a match
|
|
|
|
}
|
|
|
|
// Suppose we have at==2:
|
|
|
|
// "THAT FORTRAN THAT I RAN" <- the string (x) in which we search
|
|
|
|
// "THAT I RAN" <- the string (want) for which we search
|
|
|
|
// ^------------------ j==7, ch=='T'
|
|
|
|
// We can shift ahead 3 positions to at==5 to align the 'T's:
|
|
|
|
// "THAT FORTRAN THAT I RAN"
|
|
|
|
// "THAT I RAN"
|
|
|
|
std::size_t shift{1};
|
|
|
|
for (; shift < j; ++shift) {
|
|
|
|
if (want[j - shift - 1] == ch) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
at += shift;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-02-27 06:32:26 +08:00
|
|
|
// SCAN and VERIFY implementation help. These intrinsic functions
|
|
|
|
// do pretty much the same thing, so they're templatized with a
|
|
|
|
// distinguishing flag.
|
|
|
|
|
2021-03-13 05:40:31 +08:00
|
|
|
enum class CharFunc { Index, Scan, Verify };
|
|
|
|
|
|
|
|
template <typename CHAR, CharFunc FUNC>
|
2021-02-27 06:32:26 +08:00
|
|
|
inline std::size_t ScanVerify(const CHAR *x, std::size_t xLen, const CHAR *set,
|
|
|
|
std::size_t setLen, bool back) {
|
|
|
|
std::size_t at{back ? xLen : 1};
|
|
|
|
int increment{back ? -1 : 1};
|
|
|
|
for (; xLen-- > 0; at += increment) {
|
|
|
|
CHAR ch{x[at - 1]};
|
|
|
|
bool inSet{false};
|
|
|
|
// TODO: If set is sorted, could use binary search
|
|
|
|
for (std::size_t j{0}; j < setLen; ++j) {
|
|
|
|
if (set[j] == ch) {
|
|
|
|
inSet = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2021-03-13 05:40:31 +08:00
|
|
|
if (inSet != (FUNC == CharFunc::Verify)) {
|
2021-02-27 06:32:26 +08:00
|
|
|
return at;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Specialization for one-byte characters
|
|
|
|
template <bool IS_VERIFY = false>
|
|
|
|
inline std::size_t ScanVerify(const char *x, std::size_t xLen, const char *set,
|
|
|
|
std::size_t setLen, bool back) {
|
|
|
|
std::size_t at{back ? xLen : 1};
|
|
|
|
int increment{back ? -1 : 1};
|
|
|
|
if (xLen > 0) {
|
|
|
|
std::uint64_t bitSet[256 / 64]{0};
|
|
|
|
std::uint64_t one{1};
|
|
|
|
for (std::size_t j{0}; j < setLen; ++j) {
|
|
|
|
unsigned setCh{static_cast<unsigned char>(set[j])};
|
|
|
|
bitSet[setCh / 64] |= one << (setCh % 64);
|
|
|
|
}
|
|
|
|
for (; xLen-- > 0; at += increment) {
|
|
|
|
unsigned ch{static_cast<unsigned char>(x[at - 1])};
|
|
|
|
bool inSet{((bitSet[ch / 64] >> (ch % 64)) & 1) != 0};
|
|
|
|
if (inSet != IS_VERIFY) {
|
|
|
|
return at;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-03-13 05:40:31 +08:00
|
|
|
template <typename INT, typename CHAR, CharFunc FUNC>
|
|
|
|
static void GeneralCharFunc(Descriptor &result, const Descriptor &string,
|
|
|
|
const Descriptor &arg, const Descriptor *back,
|
2021-02-27 06:32:26 +08:00
|
|
|
const Terminator &terminator) {
|
|
|
|
int rank{string.rank() ? string.rank()
|
2021-03-13 05:40:31 +08:00
|
|
|
: arg.rank() ? arg.rank()
|
|
|
|
: back ? back->rank()
|
|
|
|
: 0};
|
|
|
|
SubscriptValue lb[maxRank], ub[maxRank], stringAt[maxRank], argAt[maxRank],
|
2021-02-27 06:32:26 +08:00
|
|
|
backAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lb[j] = 1;
|
2021-03-13 05:40:31 +08:00
|
|
|
ub[j] = string.rank() ? string.GetDimension(j).Extent()
|
|
|
|
: arg.rank() ? arg.GetDimension(j).Extent()
|
|
|
|
: back ? back->GetDimension(j).Extent()
|
|
|
|
: 1;
|
2021-02-27 06:32:26 +08:00
|
|
|
elements *= ub[j];
|
2021-03-13 05:40:31 +08:00
|
|
|
stringAt[j] = argAt[j] = backAt[j] = 1;
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
result.Establish(TypeCategory::Integer, sizeof(INT), nullptr, rank, ub,
|
|
|
|
CFI_attribute_allocatable);
|
|
|
|
if (result.Allocate(lb, ub) != CFI_SUCCESS) {
|
|
|
|
terminator.Crash("SCAN/VERIFY: could not allocate storage for result");
|
|
|
|
}
|
|
|
|
std::size_t stringElementChars{string.ElementBytes() >> shift<CHAR>};
|
2021-03-13 05:40:31 +08:00
|
|
|
std::size_t argElementChars{arg.ElementBytes() >> shift<CHAR>};
|
2021-02-27 06:32:26 +08:00
|
|
|
for (SubscriptValue resultAt{0}; elements-- > 0; resultAt += sizeof(INT),
|
2021-03-13 05:40:31 +08:00
|
|
|
string.IncrementSubscripts(stringAt), arg.IncrementSubscripts(argAt),
|
2021-02-27 06:32:26 +08:00
|
|
|
back && back->IncrementSubscripts(backAt)) {
|
2021-03-13 05:40:31 +08:00
|
|
|
if constexpr (FUNC == CharFunc::Index) {
|
|
|
|
*result.OffsetElement<INT>(resultAt) =
|
|
|
|
Index<CHAR>(string.Element<CHAR>(stringAt), stringElementChars,
|
|
|
|
arg.Element<CHAR>(argAt), argElementChars,
|
|
|
|
back && IsLogicalElementTrue(*back, backAt));
|
|
|
|
} else if constexpr (FUNC == CharFunc::Scan) {
|
|
|
|
*result.OffsetElement<INT>(resultAt) =
|
|
|
|
ScanVerify<CHAR, CharFunc::Scan>(string.Element<CHAR>(stringAt),
|
|
|
|
stringElementChars, arg.Element<CHAR>(argAt), argElementChars,
|
|
|
|
back && IsLogicalElementTrue(*back, backAt));
|
|
|
|
} else if constexpr (FUNC == CharFunc::Verify) {
|
|
|
|
*result.OffsetElement<INT>(resultAt) =
|
|
|
|
ScanVerify<CHAR, CharFunc::Verify>(string.Element<CHAR>(stringAt),
|
|
|
|
stringElementChars, arg.Element<CHAR>(argAt), argElementChars,
|
|
|
|
back && IsLogicalElementTrue(*back, backAt));
|
|
|
|
} else {
|
|
|
|
static_assert(FUNC == CharFunc::Index || FUNC == CharFunc::Scan ||
|
|
|
|
FUNC == CharFunc::Verify);
|
|
|
|
}
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-13 05:40:31 +08:00
|
|
|
template <typename CHAR, CharFunc FUNC>
|
|
|
|
static void GeneralCharFuncKind(Descriptor &result, const Descriptor &string,
|
|
|
|
const Descriptor &arg, const Descriptor *back, int kind,
|
2021-02-27 06:32:26 +08:00
|
|
|
const Terminator &terminator) {
|
|
|
|
switch (kind) {
|
|
|
|
case 1:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
GeneralCharFunc<CppTypeFor<TypeCategory::Integer, 1>, CHAR, FUNC>(
|
2021-03-13 05:40:31 +08:00
|
|
|
result, string, arg, back, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case 2:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
GeneralCharFunc<CppTypeFor<TypeCategory::Integer, 2>, CHAR, FUNC>(
|
2021-03-13 05:40:31 +08:00
|
|
|
result, string, arg, back, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
GeneralCharFunc<CppTypeFor<TypeCategory::Integer, 4>, CHAR, FUNC>(
|
2021-03-13 05:40:31 +08:00
|
|
|
result, string, arg, back, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case 8:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
GeneralCharFunc<CppTypeFor<TypeCategory::Integer, 8>, CHAR, FUNC>(
|
2021-03-13 05:40:31 +08:00
|
|
|
result, string, arg, back, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case 16:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
GeneralCharFunc<CppTypeFor<TypeCategory::Integer, 16>, CHAR, FUNC>(
|
2021-03-13 05:40:31 +08:00
|
|
|
result, string, arg, back, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
default:
|
2021-03-13 05:40:31 +08:00
|
|
|
terminator.Crash("INDEX/SCAN/VERIFY: bad KIND=%d", kind);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
template <typename TO, typename FROM>
|
|
|
|
static void CopyAndPad(
|
|
|
|
TO *to, const FROM *from, std::size_t toChars, std::size_t fromChars) {
|
|
|
|
if constexpr (sizeof(TO) != sizeof(FROM)) {
|
|
|
|
std::size_t copyChars{std::min(toChars, fromChars)};
|
|
|
|
for (std::size_t j{0}; j < copyChars; ++j) {
|
|
|
|
to[j] = from[j];
|
|
|
|
}
|
|
|
|
for (std::size_t j{copyChars}; j < toChars; ++j) {
|
|
|
|
to[j] = static_cast<TO>(' ');
|
|
|
|
}
|
|
|
|
} else if (toChars <= fromChars) {
|
|
|
|
std::memcpy(to, from, toChars * shift<TO>);
|
|
|
|
} else {
|
|
|
|
std::memcpy(to, from, fromChars * shift<TO>);
|
|
|
|
for (std::size_t j{fromChars}; j < toChars; ++j) {
|
|
|
|
to[j] = static_cast<TO>(' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename CHAR, bool ISMIN>
|
|
|
|
static void MaxMinHelper(Descriptor &accumulator, const Descriptor &x,
|
|
|
|
const Terminator &terminator) {
|
|
|
|
RUNTIME_CHECK(terminator,
|
|
|
|
accumulator.rank() == 0 || x.rank() == 0 ||
|
|
|
|
accumulator.rank() == x.rank());
|
|
|
|
SubscriptValue lb[maxRank], ub[maxRank], xAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
std::size_t accumChars{accumulator.ElementBytes() >> shift<CHAR>};
|
|
|
|
std::size_t xChars{x.ElementBytes() >> shift<CHAR>};
|
|
|
|
std::size_t chars{std::max(accumChars, xChars)};
|
|
|
|
bool reallocate{accumulator.raw().base_addr == nullptr ||
|
|
|
|
accumChars != xChars || (accumulator.rank() == 0 && x.rank() > 0)};
|
|
|
|
int rank{std::max(accumulator.rank(), x.rank())};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lb[j] = 1;
|
|
|
|
if (x.rank() > 0) {
|
|
|
|
ub[j] = x.GetDimension(j).Extent();
|
|
|
|
xAt[j] = x.GetDimension(j).LowerBound();
|
|
|
|
if (accumulator.rank() > 0) {
|
|
|
|
SubscriptValue accumExt{accumulator.GetDimension(j).Extent()};
|
|
|
|
if (accumExt != ub[j]) {
|
|
|
|
terminator.Crash("Character MAX/MIN: operands are not "
|
|
|
|
"conforming on dimension %d (%jd != %jd)",
|
|
|
|
j + 1, static_cast<std::intmax_t>(accumExt),
|
|
|
|
static_cast<std::intmax_t>(ub[j]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ub[j] = accumulator.GetDimension(j).Extent();
|
|
|
|
xAt[j] = 1;
|
|
|
|
}
|
|
|
|
elements *= ub[j];
|
|
|
|
}
|
|
|
|
void *old{nullptr};
|
|
|
|
const CHAR *accumData{accumulator.OffsetElement<CHAR>()};
|
|
|
|
if (reallocate) {
|
|
|
|
old = accumulator.raw().base_addr;
|
|
|
|
accumulator.set_base_addr(nullptr);
|
|
|
|
accumulator.raw().elem_len = chars << shift<CHAR>;
|
|
|
|
RUNTIME_CHECK(terminator, accumulator.Allocate(lb, ub) == CFI_SUCCESS);
|
|
|
|
}
|
|
|
|
for (CHAR *result{accumulator.OffsetElement<CHAR>()}; elements-- > 0;
|
|
|
|
accumData += accumChars, result += chars, x.IncrementSubscripts(xAt)) {
|
|
|
|
const CHAR *xData{x.Element<CHAR>(xAt)};
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
int cmp{CharacterScalarCompare(accumData, xData, accumChars, xChars)};
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if constexpr (ISMIN) {
|
|
|
|
cmp = -cmp;
|
|
|
|
}
|
|
|
|
if (cmp < 0) {
|
|
|
|
CopyAndPad(result, xData, chars, xChars);
|
|
|
|
} else if (result != accumData) {
|
|
|
|
CopyAndPad(result, accumData, chars, accumChars);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FreeMemory(old);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <bool ISMIN>
|
|
|
|
static void MaxMin(Descriptor &accumulator, const Descriptor &x,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
RUNTIME_CHECK(terminator, accumulator.raw().type == x.raw().type);
|
|
|
|
switch (accumulator.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
MaxMinHelper<char, ISMIN>(accumulator, x, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
MaxMinHelper<char16_t, ISMIN>(accumulator, x, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
MaxMinHelper<char32_t, ISMIN>(accumulator, x, terminator);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"Character MAX/MIN: result does not have a character type");
|
|
|
|
}
|
2020-04-24 08:07:45 +08:00
|
|
|
}
|
|
|
|
|
2020-03-31 07:37:30 +08:00
|
|
|
extern "C" {
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
void RTNAME(CharacterConcatenate)(Descriptor &accumulator,
|
|
|
|
const Descriptor &from, const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
RUNTIME_CHECK(terminator,
|
|
|
|
accumulator.rank() == 0 || from.rank() == 0 ||
|
|
|
|
accumulator.rank() == from.rank());
|
|
|
|
int rank{std::max(accumulator.rank(), from.rank())};
|
|
|
|
SubscriptValue lb[maxRank], ub[maxRank], fromAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lb[j] = 1;
|
|
|
|
if (accumulator.rank() > 0 && from.rank() > 0) {
|
|
|
|
ub[j] = accumulator.GetDimension(j).Extent();
|
|
|
|
SubscriptValue fromUB{from.GetDimension(j).Extent()};
|
|
|
|
if (ub[j] != fromUB) {
|
|
|
|
terminator.Crash("Character array concatenation: operands are not "
|
|
|
|
"conforming on dimension %d (%jd != %jd)",
|
|
|
|
j + 1, static_cast<std::intmax_t>(ub[j]),
|
|
|
|
static_cast<std::intmax_t>(fromUB));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ub[j] =
|
|
|
|
(accumulator.rank() ? accumulator : from).GetDimension(j).Extent();
|
|
|
|
}
|
|
|
|
elements *= ub[j];
|
|
|
|
fromAt[j] = 1;
|
|
|
|
}
|
|
|
|
std::size_t oldBytes{accumulator.ElementBytes()};
|
|
|
|
void *old{accumulator.raw().base_addr};
|
|
|
|
accumulator.set_base_addr(nullptr);
|
|
|
|
std::size_t fromBytes{from.ElementBytes()};
|
|
|
|
accumulator.raw().elem_len += fromBytes;
|
|
|
|
std::size_t newBytes{accumulator.ElementBytes()};
|
|
|
|
if (accumulator.Allocate(lb, ub) != CFI_SUCCESS) {
|
|
|
|
terminator.Crash(
|
|
|
|
"CharacterConcatenate: could not allocate storage for result");
|
|
|
|
}
|
|
|
|
const char *p{static_cast<const char *>(old)};
|
|
|
|
char *to{static_cast<char *>(accumulator.raw().base_addr)};
|
|
|
|
for (; elements-- > 0;
|
|
|
|
to += newBytes, p += oldBytes, from.IncrementSubscripts(fromAt)) {
|
|
|
|
std::memcpy(to, p, oldBytes);
|
|
|
|
std::memcpy(to + oldBytes, from.Element<char>(fromAt), fromBytes);
|
|
|
|
}
|
|
|
|
FreeMemory(old);
|
2020-03-31 07:37:30 +08:00
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
void RTNAME(CharacterConcatenateScalar1)(
|
|
|
|
Descriptor &accumulator, const char *from, std::size_t chars) {
|
|
|
|
Terminator terminator{__FILE__, __LINE__};
|
|
|
|
RUNTIME_CHECK(terminator, accumulator.rank() == 0);
|
|
|
|
void *old{accumulator.raw().base_addr};
|
|
|
|
accumulator.set_base_addr(nullptr);
|
|
|
|
std::size_t oldLen{accumulator.ElementBytes()};
|
|
|
|
accumulator.raw().elem_len += chars;
|
|
|
|
RUNTIME_CHECK(
|
|
|
|
terminator, accumulator.Allocate(nullptr, nullptr) == CFI_SUCCESS);
|
|
|
|
std::memcpy(accumulator.OffsetElement<char>(oldLen), from, chars);
|
|
|
|
FreeMemory(old);
|
2020-03-31 07:37:30 +08:00
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
void RTNAME(CharacterAssign)(Descriptor &lhs, const Descriptor &rhs,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
int rank{lhs.rank()};
|
|
|
|
RUNTIME_CHECK(terminator, rhs.rank() == 0 || rhs.rank() == rank);
|
|
|
|
SubscriptValue ub[maxRank], lhsAt[maxRank], rhsAt[maxRank];
|
|
|
|
SubscriptValue elements{1};
|
|
|
|
std::size_t lhsBytes{lhs.ElementBytes()};
|
|
|
|
std::size_t rhsBytes{rhs.ElementBytes()};
|
|
|
|
bool reallocate{lhs.IsAllocatable() &&
|
|
|
|
(lhs.raw().base_addr == nullptr || lhsBytes != rhsBytes)};
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lhsAt[j] = lhs.GetDimension(j).LowerBound();
|
|
|
|
if (rhs.rank() > 0) {
|
|
|
|
SubscriptValue lhsExt{lhs.GetDimension(j).Extent()};
|
|
|
|
SubscriptValue rhsExt{rhs.GetDimension(j).Extent()};
|
|
|
|
ub[j] = lhsAt[j] + rhsExt - 1;
|
|
|
|
if (lhsExt != rhsExt) {
|
|
|
|
if (lhs.IsAllocatable()) {
|
|
|
|
reallocate = true;
|
|
|
|
} else {
|
|
|
|
terminator.Crash("Character array assignment: operands are not "
|
|
|
|
"conforming on dimension %d (%jd != %jd)",
|
|
|
|
j + 1, static_cast<std::intmax_t>(lhsExt),
|
|
|
|
static_cast<std::intmax_t>(rhsExt));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rhsAt[j] = rhs.GetDimension(j).LowerBound();
|
|
|
|
} else {
|
|
|
|
ub[j] = lhs.GetDimension(j).UpperBound();
|
|
|
|
}
|
|
|
|
elements *= ub[j] - lhsAt[j] + 1;
|
|
|
|
}
|
|
|
|
void *old{nullptr};
|
|
|
|
if (reallocate) {
|
|
|
|
old = lhs.raw().base_addr;
|
|
|
|
lhs.set_base_addr(nullptr);
|
|
|
|
lhs.raw().elem_len = lhsBytes = rhsBytes;
|
|
|
|
if (rhs.rank() > 0) {
|
|
|
|
// When the RHS is not scalar, the LHS acquires its bounds.
|
|
|
|
for (int j{0}; j < rank; ++j) {
|
|
|
|
lhsAt[j] = rhsAt[j];
|
|
|
|
ub[j] = rhs.GetDimension(j).UpperBound();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RUNTIME_CHECK(terminator, lhs.Allocate(lhsAt, ub) == CFI_SUCCESS);
|
|
|
|
}
|
|
|
|
switch (lhs.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
switch (rhs.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char>(lhsAt), rhs.Element<char>(rhsAt), lhsBytes,
|
|
|
|
rhsBytes);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char>(lhsAt), rhs.Element<char16_t>(rhsAt),
|
|
|
|
lhsBytes, rhsBytes >> 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char>(lhsAt), rhs.Element<char32_t>(rhsAt),
|
|
|
|
lhsBytes, rhsBytes >> 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"RHS of character assignment does not have a character type");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
switch (rhs.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char16_t>(lhsAt), rhs.Element<char>(rhsAt),
|
|
|
|
lhsBytes >> 1, rhsBytes);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char16_t>(lhsAt), rhs.Element<char16_t>(rhsAt),
|
|
|
|
lhsBytes >> 1, rhsBytes >> 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char16_t>(lhsAt), rhs.Element<char32_t>(rhsAt),
|
|
|
|
lhsBytes >> 1, rhsBytes >> 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"RHS of character assignment does not have a character type");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
switch (rhs.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char32_t>(lhsAt), rhs.Element<char>(rhsAt),
|
|
|
|
lhsBytes >> 2, rhsBytes);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char32_t>(lhsAt), rhs.Element<char16_t>(rhsAt),
|
|
|
|
lhsBytes >> 2, rhsBytes >> 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
for (; elements-- > 0;
|
|
|
|
lhs.IncrementSubscripts(lhsAt), rhs.IncrementSubscripts(rhsAt)) {
|
|
|
|
CopyAndPad(lhs.Element<char32_t>(lhsAt), rhs.Element<char32_t>(rhsAt),
|
|
|
|
lhsBytes >> 2, rhsBytes >> 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"RHS of character assignment does not have a character type");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"LHS of character assignment does not have a character type");
|
|
|
|
}
|
|
|
|
if (reallocate) {
|
|
|
|
FreeMemory(old);
|
|
|
|
}
|
2020-03-31 07:37:30 +08:00
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
int RTNAME(CharacterCompareScalar)(const Descriptor &x, const Descriptor &y) {
|
|
|
|
Terminator terminator{__FILE__, __LINE__};
|
|
|
|
RUNTIME_CHECK(terminator, x.rank() == 0);
|
|
|
|
RUNTIME_CHECK(terminator, y.rank() == 0);
|
|
|
|
RUNTIME_CHECK(terminator, x.raw().type == y.raw().type);
|
|
|
|
switch (x.raw().type) {
|
|
|
|
case CFI_type_char:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
return CharacterScalarCompare<char>(x.OffsetElement<char>(),
|
|
|
|
y.OffsetElement<char>(), x.ElementBytes(), y.ElementBytes());
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
case CFI_type_char16_t:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
return CharacterScalarCompare<char16_t>(x.OffsetElement<char16_t>(),
|
|
|
|
y.OffsetElement<char16_t>(), x.ElementBytes() >> 1,
|
|
|
|
y.ElementBytes() >> 1);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
case CFI_type_char32_t:
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
return CharacterScalarCompare<char32_t>(x.OffsetElement<char32_t>(),
|
|
|
|
y.OffsetElement<char32_t>(), x.ElementBytes() >> 2,
|
|
|
|
y.ElementBytes() >> 2);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
default:
|
|
|
|
terminator.Crash("CharacterCompareScalar: bad string type code %d",
|
|
|
|
static_cast<int>(x.raw().type));
|
|
|
|
}
|
2020-04-24 08:07:45 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int RTNAME(CharacterCompareScalar1)(
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
const char *x, const char *y, std::size_t xChars, std::size_t yChars) {
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
return CharacterScalarCompare(x, y, xChars, yChars);
|
2020-04-24 08:07:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int RTNAME(CharacterCompareScalar2)(const char16_t *x, const char16_t *y,
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
std::size_t xChars, std::size_t yChars) {
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
return CharacterScalarCompare(x, y, xChars, yChars);
|
2020-04-24 08:07:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int RTNAME(CharacterCompareScalar4)(const char32_t *x, const char32_t *y,
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
std::size_t xChars, std::size_t yChars) {
|
[flang] Implement reductions in the runtime
Add runtime APIs, implementations, and tests for ALL, ANY, COUNT,
MAXLOC, MAXVAL, MINLOC, MINVAL, PRODUCT, and SUM reduction
transformantional intrinsic functions for all relevant argument
and result types and kinds, both without DIM= arguments
(total reductions) and with (partial reductions).
Complex-valued reductions have their APIs in C so that
C's _Complex types can be used for their results.
Some infrastructure work was also necessary or noticed:
* Usage of "long double" in the compiler was cleaned up a
bit, and host dependences on x86 / MSVC have been isolated
in a new Common/long-double header.
* Character comparison has been exposed via an extern template
so that reductions could use it.
* Mappings from Fortran type category/kind to host C++ types
and vice versa have been isolated into runtime/cpp-type.h and
then used throughout the runtime as appropriate.
* The portable 128-bit integer package in Common/uint128.h
was generalized to support signed comparisons.
* Bugs in descriptor indexing code were fixed.
Differential Revision: https://reviews.llvm.org/D99666
2021-04-01 00:14:08 +08:00
|
|
|
return CharacterScalarCompare(x, y, xChars, yChars);
|
2020-04-24 08:07:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(CharacterCompare)(
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
Descriptor &result, const Descriptor &x, const Descriptor &y) {
|
|
|
|
Terminator terminator{__FILE__, __LINE__};
|
|
|
|
RUNTIME_CHECK(terminator, x.raw().type == y.raw().type);
|
|
|
|
switch (x.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
Compare<char>(result, x, y, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
Compare<char16_t>(result, x, y, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
Compare<char32_t>(result, x, y, terminator);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash("CharacterCompareScalar: bad string type code %d",
|
|
|
|
static_cast<int>(x.raw().type));
|
|
|
|
}
|
2020-04-24 08:07:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t RTNAME(CharacterAppend1)(char *lhs, std::size_t lhsBytes,
|
|
|
|
std::size_t offset, const char *rhs, std::size_t rhsBytes) {
|
|
|
|
if (auto n{std::min(lhsBytes - offset, rhsBytes)}) {
|
2020-03-31 07:37:30 +08:00
|
|
|
std::memcpy(lhs + offset, rhs, n);
|
|
|
|
offset += n;
|
|
|
|
}
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
2020-04-24 08:07:45 +08:00
|
|
|
void RTNAME(CharacterPad1)(char *lhs, std::size_t bytes, std::size_t offset) {
|
|
|
|
if (bytes > offset) {
|
|
|
|
std::memset(lhs + offset, ' ', bytes - offset);
|
2020-03-31 07:37:30 +08:00
|
|
|
}
|
|
|
|
}
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
|
2021-02-27 06:32:26 +08:00
|
|
|
// Intrinsic function entry points
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
|
|
|
|
void RTNAME(AdjustL)(Descriptor &result, const Descriptor &string,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
AdjustLR<false>(result, string, sourceFile, sourceLine);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(AdjustR)(Descriptor &result, const Descriptor &string,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
AdjustLR<true>(result, string, sourceFile, sourceLine);
|
|
|
|
}
|
|
|
|
|
2021-03-13 05:40:31 +08:00
|
|
|
std::size_t RTNAME(Index1)(const char *x, std::size_t xLen, const char *set,
|
|
|
|
std::size_t setLen, bool back) {
|
|
|
|
return Index<char>(x, xLen, set, setLen, back);
|
|
|
|
}
|
|
|
|
std::size_t RTNAME(Index2)(const char16_t *x, std::size_t xLen,
|
|
|
|
const char16_t *set, std::size_t setLen, bool back) {
|
|
|
|
return Index<char16_t>(x, xLen, set, setLen, back);
|
|
|
|
}
|
|
|
|
std::size_t RTNAME(Index4)(const char32_t *x, std::size_t xLen,
|
|
|
|
const char32_t *set, std::size_t setLen, bool back) {
|
|
|
|
return Index<char32_t>(x, xLen, set, setLen, back);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(Index)(Descriptor &result, const Descriptor &string,
|
|
|
|
const Descriptor &substring, const Descriptor *back, int kind,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
switch (string.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
GeneralCharFuncKind<char, CharFunc::Index>(
|
|
|
|
result, string, substring, back, kind, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
GeneralCharFuncKind<char16_t, CharFunc::Index>(
|
|
|
|
result, string, substring, back, kind, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
GeneralCharFuncKind<char32_t, CharFunc::Index>(
|
|
|
|
result, string, substring, back, kind, terminator);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"INDEX: bad string type code %d", static_cast<int>(string.raw().type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
std::size_t RTNAME(LenTrim1)(const char *x, std::size_t chars) {
|
|
|
|
return LenTrim(x, chars);
|
|
|
|
}
|
|
|
|
std::size_t RTNAME(LenTrim2)(const char16_t *x, std::size_t chars) {
|
|
|
|
return LenTrim(x, chars);
|
|
|
|
}
|
|
|
|
std::size_t RTNAME(LenTrim4)(const char32_t *x, std::size_t chars) {
|
|
|
|
return LenTrim(x, chars);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(LenTrim)(Descriptor &result, const Descriptor &string, int kind,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
switch (string.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
LenTrimKind<char>(result, string, kind, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
LenTrimKind<char16_t>(result, string, kind, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
LenTrimKind<char32_t>(result, string, kind, terminator);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash("LEN_TRIM: bad string type code %d",
|
|
|
|
static_cast<int>(string.raw().type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-27 06:32:26 +08:00
|
|
|
std::size_t RTNAME(Scan1)(const char *x, std::size_t xLen, const char *set,
|
|
|
|
std::size_t setLen, bool back) {
|
2021-03-13 05:40:31 +08:00
|
|
|
return ScanVerify<char, CharFunc::Scan>(x, xLen, set, setLen, back);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
std::size_t RTNAME(Scan2)(const char16_t *x, std::size_t xLen,
|
|
|
|
const char16_t *set, std::size_t setLen, bool back) {
|
2021-03-13 05:40:31 +08:00
|
|
|
return ScanVerify<char16_t, CharFunc::Scan>(x, xLen, set, setLen, back);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
std::size_t RTNAME(Scan4)(const char32_t *x, std::size_t xLen,
|
|
|
|
const char32_t *set, std::size_t setLen, bool back) {
|
2021-03-13 05:40:31 +08:00
|
|
|
return ScanVerify<char32_t, CharFunc::Scan>(x, xLen, set, setLen, back);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(Scan)(Descriptor &result, const Descriptor &string,
|
|
|
|
const Descriptor &set, const Descriptor *back, int kind,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
switch (string.raw().type) {
|
|
|
|
case CFI_type_char:
|
2021-03-13 05:40:31 +08:00
|
|
|
GeneralCharFuncKind<char, CharFunc::Scan>(
|
|
|
|
result, string, set, back, kind, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
2021-03-13 05:40:31 +08:00
|
|
|
GeneralCharFuncKind<char16_t, CharFunc::Scan>(
|
2021-02-27 06:32:26 +08:00
|
|
|
result, string, set, back, kind, terminator);
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
2021-03-13 05:40:31 +08:00
|
|
|
GeneralCharFuncKind<char32_t, CharFunc::Scan>(
|
2021-02-27 06:32:26 +08:00
|
|
|
result, string, set, back, kind, terminator);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"SCAN: bad string type code %d", static_cast<int>(string.raw().type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
void RTNAME(Repeat)(Descriptor &result, const Descriptor &string,
|
|
|
|
std::size_t ncopies, const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
std::size_t origBytes{string.ElementBytes()};
|
2021-02-27 06:32:26 +08:00
|
|
|
result.Establish(string.type(), origBytes * ncopies, nullptr, 0, nullptr,
|
|
|
|
CFI_attribute_allocatable);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
if (result.Allocate(nullptr, nullptr) != CFI_SUCCESS) {
|
|
|
|
terminator.Crash("REPEAT could not allocate storage for result");
|
|
|
|
}
|
|
|
|
const char *from{string.OffsetElement()};
|
|
|
|
for (char *to{result.OffsetElement()}; ncopies-- > 0; to += origBytes) {
|
|
|
|
std::memcpy(to, from, origBytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(Trim)(Descriptor &result, const Descriptor &string,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
std::size_t resultBytes{0};
|
|
|
|
switch (string.raw().type) {
|
|
|
|
case CFI_type_char:
|
|
|
|
resultBytes =
|
|
|
|
LenTrim(string.OffsetElement<const char>(), string.ElementBytes());
|
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
|
|
|
resultBytes = LenTrim(string.OffsetElement<const char16_t>(),
|
|
|
|
string.ElementBytes() >> 1)
|
|
|
|
<< 1;
|
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
|
|
|
resultBytes = LenTrim(string.OffsetElement<const char32_t>(),
|
|
|
|
string.ElementBytes() >> 2)
|
|
|
|
<< 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"TRIM: bad string type code %d", static_cast<int>(string.raw().type));
|
|
|
|
}
|
2021-02-25 01:53:03 +08:00
|
|
|
result.Establish(string.type(), resultBytes, nullptr, 0, nullptr,
|
|
|
|
CFI_attribute_allocatable);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
RUNTIME_CHECK(terminator, result.Allocate(nullptr, nullptr) == CFI_SUCCESS);
|
2020-06-18 07:56:28 +08:00
|
|
|
std::memcpy(result.OffsetElement(), string.OffsetElement(), resultBytes);
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
}
|
|
|
|
|
2021-02-27 06:32:26 +08:00
|
|
|
std::size_t RTNAME(Verify1)(const char *x, std::size_t xLen, const char *set,
|
|
|
|
std::size_t setLen, bool back) {
|
2021-03-13 05:40:31 +08:00
|
|
|
return ScanVerify<char, CharFunc::Verify>(x, xLen, set, setLen, back);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
std::size_t RTNAME(Verify2)(const char16_t *x, std::size_t xLen,
|
|
|
|
const char16_t *set, std::size_t setLen, bool back) {
|
2021-03-13 05:40:31 +08:00
|
|
|
return ScanVerify<char16_t, CharFunc::Verify>(x, xLen, set, setLen, back);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
std::size_t RTNAME(Verify4)(const char32_t *x, std::size_t xLen,
|
|
|
|
const char32_t *set, std::size_t setLen, bool back) {
|
2021-03-13 05:40:31 +08:00
|
|
|
return ScanVerify<char32_t, CharFunc::Verify>(x, xLen, set, setLen, back);
|
2021-02-27 06:32:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(Verify)(Descriptor &result, const Descriptor &string,
|
|
|
|
const Descriptor &set, const Descriptor *back, int kind,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
Terminator terminator{sourceFile, sourceLine};
|
|
|
|
switch (string.raw().type) {
|
|
|
|
case CFI_type_char:
|
2021-03-13 05:40:31 +08:00
|
|
|
GeneralCharFuncKind<char, CharFunc::Verify>(
|
|
|
|
result, string, set, back, kind, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case CFI_type_char16_t:
|
2021-03-13 05:40:31 +08:00
|
|
|
GeneralCharFuncKind<char16_t, CharFunc::Verify>(
|
|
|
|
result, string, set, back, kind, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
case CFI_type_char32_t:
|
2021-03-13 05:40:31 +08:00
|
|
|
GeneralCharFuncKind<char32_t, CharFunc::Verify>(
|
|
|
|
result, string, set, back, kind, terminator);
|
2021-02-27 06:32:26 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminator.Crash(
|
|
|
|
"VERIFY: bad string type code %d", static_cast<int>(string.raw().type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[flang] More Fortran runtime support for CHARACTER operations
Summary:
- Remove C++ library dependence from lock.h
- Implement LEN_TRIM, REPEAT, ADJUSTL, ADJUSTR, MAX/MIN
intrinsic functions for CHARACTER
Reviewers: tskeith, PeteSteinfeld, sscalpone, schweitz, DavidTruby
Reviewed By: PeteSteinfeld
Subscribers: llvm-commits, flang-commits
Tags: #flang, #llvm
Differential Revision: https://reviews.llvm.org/D82054
2020-06-18 04:17:24 +08:00
|
|
|
void RTNAME(CharacterMax)(Descriptor &accumulator, const Descriptor &x,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
MaxMin<false>(accumulator, x, sourceFile, sourceLine);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RTNAME(CharacterMin)(Descriptor &accumulator, const Descriptor &x,
|
|
|
|
const char *sourceFile, int sourceLine) {
|
|
|
|
MaxMin<true>(accumulator, x, sourceFile, sourceLine);
|
|
|
|
}
|
2020-03-31 07:37:30 +08:00
|
|
|
}
|
|
|
|
} // namespace Fortran::runtime
|