2018-05-17 01:22:33 +08:00
|
|
|
// Copyright (c) 2018, NVIDIA CORPORATION. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
// Implements the required interoperability API from ISO_Fortran_binding.h
|
|
|
|
// as specified in section 18.5.5 of Fortran 2018.
|
|
|
|
|
|
|
|
#include "descriptor.h"
|
|
|
|
|
|
|
|
namespace Fortran::ISO {
|
|
|
|
extern "C" {
|
|
|
|
|
2018-05-17 01:31:35 +08:00
|
|
|
void *CFI_address(
|
|
|
|
const CFI_cdesc_t *descriptor, const CFI_index_t subscripts[]) {
|
2018-05-17 01:22:33 +08:00
|
|
|
auto p = reinterpret_cast<char *>(descriptor->base_addr);
|
|
|
|
std::size_t rank{descriptor->rank};
|
|
|
|
const CFI_dim_t *dim{descriptor->dim};
|
|
|
|
for (std::size_t j{0}; j < rank; ++j, ++dim) {
|
|
|
|
p += (subscripts[j] - dim->lower_bound) * dim->sm;
|
|
|
|
}
|
|
|
|
return reinterpret_cast<void *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CFI_allocate(CFI_cdesc_t *descriptor, const CFI_index_t lower_bounds[],
|
2018-05-17 01:31:35 +08:00
|
|
|
const CFI_index_t upper_bounds[], std::size_t elem_len) {
|
2018-05-17 01:22:33 +08:00
|
|
|
if (descriptor->version != CFI_VERSION) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
2018-08-03 08:04:31 +08:00
|
|
|
if (descriptor->attribute != CFI_attribute_allocatable &&
|
|
|
|
descriptor->attribute != CFI_attribute_pointer) {
|
2018-05-17 01:22:33 +08:00
|
|
|
// Non-interoperable object
|
2018-11-17 00:00:41 +08:00
|
|
|
return CFI_INVALID_ATTRIBUTE;
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
2018-08-03 08:04:31 +08:00
|
|
|
if (descriptor->attribute == CFI_attribute_allocatable &&
|
|
|
|
descriptor->base_addr != nullptr) {
|
2018-05-17 01:22:33 +08:00
|
|
|
return CFI_ERROR_BASE_ADDR_NOT_NULL;
|
|
|
|
}
|
|
|
|
if (descriptor->rank > CFI_MAX_RANK) {
|
|
|
|
return CFI_INVALID_RANK;
|
|
|
|
}
|
2018-07-27 07:07:50 +08:00
|
|
|
if (descriptor->type < CFI_type_signed_char ||
|
|
|
|
descriptor->type > CFI_type_struct) {
|
|
|
|
return CFI_INVALID_TYPE;
|
|
|
|
}
|
2018-11-17 00:00:41 +08:00
|
|
|
if (descriptor->type != CFI_type_char) {
|
2018-05-17 01:22:33 +08:00
|
|
|
elem_len = descriptor->elem_len;
|
|
|
|
if (elem_len <= 0) {
|
|
|
|
return CFI_INVALID_ELEM_LEN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::size_t rank{descriptor->rank};
|
|
|
|
CFI_dim_t *dim{descriptor->dim};
|
|
|
|
std::size_t byteSize{elem_len};
|
|
|
|
for (std::size_t j{0}; j < rank; ++j, ++dim) {
|
|
|
|
CFI_index_t lb{lower_bounds[j]};
|
|
|
|
CFI_index_t ub{upper_bounds[j]};
|
|
|
|
CFI_index_t extent{ub >= lb ? ub - lb + 1 : 0};
|
|
|
|
dim->lower_bound = lb;
|
|
|
|
dim->extent = extent;
|
|
|
|
dim->sm = byteSize;
|
|
|
|
byteSize *= extent;
|
|
|
|
}
|
2018-08-03 08:04:31 +08:00
|
|
|
void *p{new char[byteSize]};
|
2018-05-17 01:22:33 +08:00
|
|
|
if (p == nullptr) {
|
|
|
|
return CFI_ERROR_MEM_ALLOCATION;
|
|
|
|
}
|
|
|
|
descriptor->base_addr = p;
|
2018-07-27 07:07:50 +08:00
|
|
|
descriptor->elem_len = elem_len;
|
2018-05-17 01:22:33 +08:00
|
|
|
return CFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CFI_deallocate(CFI_cdesc_t *descriptor) {
|
|
|
|
if (descriptor->version != CFI_VERSION) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
2018-08-03 08:04:31 +08:00
|
|
|
if (descriptor->attribute != CFI_attribute_allocatable &&
|
|
|
|
descriptor->attribute != CFI_attribute_pointer) {
|
2018-05-17 01:22:33 +08:00
|
|
|
// Non-interoperable object
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
|
|
|
if (descriptor->base_addr == nullptr) {
|
|
|
|
return CFI_ERROR_BASE_ADDR_NULL;
|
|
|
|
}
|
2018-08-03 08:04:31 +08:00
|
|
|
delete[] static_cast<char *>(descriptor->base_addr);
|
2018-05-17 01:22:33 +08:00
|
|
|
descriptor->base_addr = nullptr;
|
|
|
|
return CFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-07-27 07:07:50 +08:00
|
|
|
static constexpr std::size_t MinElemLen(CFI_type_t type) {
|
|
|
|
std::size_t minElemLen{0};
|
|
|
|
switch (type) {
|
|
|
|
case CFI_type_signed_char: minElemLen = sizeof(signed char); break;
|
|
|
|
case CFI_type_short: minElemLen = sizeof(short); break;
|
|
|
|
case CFI_type_int: minElemLen = sizeof(int); break;
|
|
|
|
case CFI_type_long: minElemLen = sizeof(long); break;
|
|
|
|
case CFI_type_long_long: minElemLen = sizeof(long long); break;
|
|
|
|
case CFI_type_size_t: minElemLen = sizeof(std::size_t); break;
|
|
|
|
case CFI_type_int8_t: minElemLen = sizeof(std::int8_t); break;
|
|
|
|
case CFI_type_int16_t: minElemLen = sizeof(std::int16_t); break;
|
|
|
|
case CFI_type_int32_t: minElemLen = sizeof(std::int32_t); break;
|
|
|
|
case CFI_type_int64_t: minElemLen = sizeof(std::int64_t); break;
|
|
|
|
case CFI_type_int128_t: minElemLen = 2 * sizeof(std::int64_t); break;
|
|
|
|
case CFI_type_int_least8_t: minElemLen = sizeof(std::int_least8_t); break;
|
|
|
|
case CFI_type_int_least16_t: minElemLen = sizeof(std::int_least16_t); break;
|
|
|
|
case CFI_type_int_least32_t: minElemLen = sizeof(std::int_least32_t); break;
|
|
|
|
case CFI_type_int_least64_t: minElemLen = sizeof(std::int_least64_t); break;
|
|
|
|
case CFI_type_int_least128_t:
|
|
|
|
minElemLen = 2 * sizeof(std::int_least64_t);
|
|
|
|
break;
|
|
|
|
case CFI_type_int_fast8_t: minElemLen = sizeof(std::int_fast8_t); break;
|
|
|
|
case CFI_type_int_fast16_t: minElemLen = sizeof(std::int_fast16_t); break;
|
|
|
|
case CFI_type_int_fast32_t: minElemLen = sizeof(std::int_fast32_t); break;
|
|
|
|
case CFI_type_int_fast64_t: minElemLen = sizeof(std::int_fast64_t); break;
|
|
|
|
case CFI_type_intmax_t: minElemLen = sizeof(std::intmax_t); break;
|
|
|
|
case CFI_type_intptr_t: minElemLen = sizeof(std::intptr_t); break;
|
|
|
|
case CFI_type_ptrdiff_t: minElemLen = sizeof(std::ptrdiff_t); break;
|
|
|
|
case CFI_type_float: minElemLen = sizeof(float); break;
|
|
|
|
case CFI_type_double: minElemLen = sizeof(double); break;
|
|
|
|
case CFI_type_long_double: minElemLen = sizeof(long double); break;
|
|
|
|
case CFI_type_float_Complex: minElemLen = 2 * sizeof(float); break;
|
|
|
|
case CFI_type_double_Complex: minElemLen = 2 * sizeof(double); break;
|
|
|
|
case CFI_type_long_double_Complex:
|
|
|
|
minElemLen = 2 * sizeof(long double);
|
|
|
|
break;
|
|
|
|
case CFI_type_Bool: minElemLen = 1; break;
|
2018-11-17 00:00:41 +08:00
|
|
|
case CFI_type_cptr: minElemLen = sizeof(void *); break;
|
2018-07-27 07:07:50 +08:00
|
|
|
}
|
|
|
|
return minElemLen;
|
|
|
|
}
|
|
|
|
|
2018-05-17 01:22:33 +08:00
|
|
|
int CFI_establish(CFI_cdesc_t *descriptor, void *base_addr,
|
2018-05-17 01:31:35 +08:00
|
|
|
CFI_attribute_t attribute, CFI_type_t type, std::size_t elem_len,
|
|
|
|
CFI_rank_t rank, const CFI_index_t extents[]) {
|
2018-08-03 08:04:31 +08:00
|
|
|
if (attribute != CFI_attribute_other && attribute != CFI_attribute_pointer &&
|
|
|
|
attribute != CFI_attribute_allocatable) {
|
2018-05-17 01:22:33 +08:00
|
|
|
return CFI_INVALID_ATTRIBUTE;
|
|
|
|
}
|
|
|
|
if (rank > CFI_MAX_RANK) {
|
|
|
|
return CFI_INVALID_RANK;
|
|
|
|
}
|
2018-11-05 23:07:18 +08:00
|
|
|
if (base_addr != nullptr && attribute == CFI_attribute_allocatable) {
|
2018-08-03 08:04:31 +08:00
|
|
|
return CFI_ERROR_BASE_ADDR_NOT_NULL;
|
|
|
|
}
|
2018-05-17 01:22:33 +08:00
|
|
|
if (rank > 0 && base_addr != nullptr && extents == nullptr) {
|
|
|
|
return CFI_INVALID_EXTENT;
|
|
|
|
}
|
2018-07-27 07:07:50 +08:00
|
|
|
if (type < CFI_type_signed_char || type > CFI_type_struct) {
|
|
|
|
return CFI_INVALID_TYPE;
|
|
|
|
}
|
|
|
|
std::size_t minElemLen{MinElemLen(type)};
|
|
|
|
if (minElemLen > 0) {
|
|
|
|
elem_len = minElemLen;
|
|
|
|
} else if (elem_len <= 0) {
|
|
|
|
return CFI_INVALID_ELEM_LEN;
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
descriptor->base_addr = base_addr;
|
|
|
|
descriptor->elem_len = elem_len;
|
|
|
|
descriptor->version = CFI_VERSION;
|
|
|
|
descriptor->rank = rank;
|
2018-08-03 02:45:11 +08:00
|
|
|
descriptor->type = type;
|
2018-05-17 01:22:33 +08:00
|
|
|
descriptor->attribute = attribute;
|
2018-08-03 02:45:11 +08:00
|
|
|
descriptor->f18Addendum = 0;
|
2018-05-17 01:22:33 +08:00
|
|
|
std::size_t byteSize{elem_len};
|
2018-11-16 23:10:04 +08:00
|
|
|
const std::size_t lower_bound{0};
|
2018-11-07 01:06:10 +08:00
|
|
|
if (base_addr != nullptr) {
|
2018-11-05 23:07:18 +08:00
|
|
|
for (std::size_t j{0}; j < rank; ++j) {
|
|
|
|
descriptor->dim[j].lower_bound = lower_bound;
|
|
|
|
descriptor->dim[j].extent = extents[j];
|
|
|
|
descriptor->dim[j].sm = byteSize;
|
|
|
|
byteSize *= extents[j];
|
|
|
|
}
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
return CFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CFI_is_contiguous(const CFI_cdesc_t *descriptor) {
|
2018-08-04 06:18:04 +08:00
|
|
|
CFI_index_t bytes = descriptor->elem_len;
|
2018-08-03 08:04:31 +08:00
|
|
|
for (int j{0}; j < descriptor->rank; ++j) {
|
|
|
|
if (bytes != descriptor->dim[j].sm) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
bytes *= descriptor->dim[j].extent;
|
|
|
|
}
|
|
|
|
return 1;
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
|
2018-11-16 23:10:04 +08:00
|
|
|
static inline bool IsAssumedSize(const CFI_cdesc_t *dv) {
|
2018-11-27 07:31:07 +08:00
|
|
|
return dv->rank > 0 && dv->dim[dv->rank - 1].extent == -1;
|
2018-11-16 23:10:04 +08:00
|
|
|
}
|
|
|
|
|
2018-05-17 01:22:33 +08:00
|
|
|
int CFI_section(CFI_cdesc_t *result, const CFI_cdesc_t *source,
|
2018-05-17 01:31:35 +08:00
|
|
|
const CFI_index_t lower_bounds[], const CFI_index_t upper_bounds[],
|
|
|
|
const CFI_index_t strides[]) {
|
2018-11-16 23:10:04 +08:00
|
|
|
CFI_index_t extent[CFI_MAX_RANK];
|
|
|
|
CFI_index_t actualStride[CFI_MAX_RANK];
|
|
|
|
CFI_rank_t resRank{0};
|
2018-11-20 01:10:08 +08:00
|
|
|
char *shiftedBaseAddr{static_cast<char *>(source->base_addr)};
|
2018-11-16 23:10:04 +08:00
|
|
|
|
|
|
|
if (source->rank == 0) {
|
|
|
|
return CFI_INVALID_RANK;
|
|
|
|
}
|
|
|
|
if (IsAssumedSize(source) && upper_bounds == nullptr) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
|
|
|
if ((result->type != source->type) ||
|
|
|
|
(result->elem_len != source->elem_len)) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
|
|
|
if (result->attribute == CFI_attribute_allocatable) {
|
|
|
|
return CFI_INVALID_ATTRIBUTE;
|
|
|
|
}
|
|
|
|
if (source->base_addr == nullptr) {
|
|
|
|
return CFI_ERROR_BASE_ADDR_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isZeroSized{false};
|
|
|
|
for (int j{0}; j < source->rank; ++j) {
|
2018-11-28 06:13:21 +08:00
|
|
|
const CFI_dim_t &dim{source->dim[j]};
|
|
|
|
const CFI_index_t srcLB{dim.lower_bound};
|
|
|
|
const CFI_index_t srcUB{srcLB + dim.extent - 1};
|
|
|
|
const CFI_index_t lb{lower_bounds != nullptr ? lower_bounds[j] : srcLB};
|
|
|
|
const CFI_index_t ub{upper_bounds != nullptr ? upper_bounds[j] : srcUB};
|
|
|
|
const CFI_index_t stride{strides != nullptr ? strides[j] : 1};
|
2018-11-16 23:10:04 +08:00
|
|
|
|
|
|
|
if (stride == 0 && lb != ub) {
|
|
|
|
return CFI_ERROR_OUT_OF_BOUNDS;
|
|
|
|
}
|
|
|
|
if ((lb <= ub && stride >= 0) || (lb >= ub && stride < 0)) {
|
|
|
|
if ((lb < srcLB) || (lb > srcUB) || (ub < srcLB) || (ub > srcUB)) {
|
|
|
|
return CFI_ERROR_OUT_OF_BOUNDS;
|
|
|
|
}
|
|
|
|
shiftedBaseAddr += (lb - srcLB) * dim.sm;
|
2018-11-28 06:13:21 +08:00
|
|
|
extent[j] = stride != 0 ? 1 + (ub - lb) / stride : 1;
|
2018-11-16 23:10:04 +08:00
|
|
|
} else {
|
|
|
|
isZeroSized = true;
|
|
|
|
extent[j] = 0;
|
|
|
|
}
|
|
|
|
actualStride[j] = stride;
|
|
|
|
resRank += (stride != 0);
|
|
|
|
}
|
|
|
|
if (resRank != result->rank) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
|
|
|
|
2018-11-20 00:54:58 +08:00
|
|
|
// For zero-sized arrays, base_addr is processor-dependent (see 18.5.3).
|
2018-11-16 23:10:04 +08:00
|
|
|
// We keep it on the source base_addr
|
2018-11-20 01:10:08 +08:00
|
|
|
result->base_addr = isZeroSized ? source->base_addr : shiftedBaseAddr;
|
2018-11-16 23:10:04 +08:00
|
|
|
resRank = 0;
|
|
|
|
for (int j{0}; j < source->rank; ++j) {
|
|
|
|
if (actualStride[j] != 0) {
|
2018-11-19 23:50:30 +08:00
|
|
|
result->dim[resRank].lower_bound = 0;
|
2018-11-16 23:10:04 +08:00
|
|
|
result->dim[resRank].extent = extent[j];
|
|
|
|
result->dim[resRank].sm = actualStride[j] * source->dim[j].sm;
|
|
|
|
++resRank;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return CFI_SUCCESS;
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int CFI_select_part(CFI_cdesc_t *result, const CFI_cdesc_t *source,
|
2018-05-17 01:31:35 +08:00
|
|
|
std::size_t displacement, std::size_t elem_len) {
|
2018-11-16 23:10:04 +08:00
|
|
|
if (result->rank != source->rank) {
|
|
|
|
return CFI_INVALID_RANK;
|
|
|
|
}
|
|
|
|
if (result->attribute == CFI_attribute_allocatable) {
|
|
|
|
return CFI_INVALID_ATTRIBUTE;
|
|
|
|
}
|
|
|
|
if (source->base_addr == nullptr) {
|
|
|
|
return CFI_ERROR_BASE_ADDR_NULL;
|
|
|
|
}
|
|
|
|
if (IsAssumedSize(source)) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result->type != CFI_type_char) {
|
|
|
|
elem_len = result->elem_len;
|
|
|
|
}
|
|
|
|
if (displacement + elem_len > source->elem_len) {
|
|
|
|
return CFI_INVALID_ELEM_LEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
result->base_addr = reinterpret_cast<void *>(
|
|
|
|
displacement + reinterpret_cast<char *>(source->base_addr));
|
|
|
|
result->elem_len = elem_len;
|
|
|
|
for (int j{0}; j < source->rank; ++j) {
|
2018-11-19 23:50:30 +08:00
|
|
|
result->dim[j].lower_bound = 0;
|
|
|
|
result->dim[j].extent = source->dim[j].extent;
|
|
|
|
result->dim[j].sm = source->dim[j].sm;
|
2018-11-16 23:10:04 +08:00
|
|
|
}
|
|
|
|
return CFI_SUCCESS;
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int CFI_setpointer(CFI_cdesc_t *result, const CFI_cdesc_t *source,
|
2018-05-17 01:31:35 +08:00
|
|
|
const CFI_index_t lower_bounds[]) {
|
2018-11-16 23:10:04 +08:00
|
|
|
if (result->attribute != CFI_attribute_pointer) {
|
|
|
|
return CFI_INVALID_ATTRIBUTE;
|
|
|
|
}
|
|
|
|
if (source == nullptr) {
|
|
|
|
result->base_addr = nullptr;
|
|
|
|
return CFI_SUCCESS;
|
|
|
|
}
|
|
|
|
if (source->rank != result->rank) {
|
|
|
|
return CFI_INVALID_RANK;
|
|
|
|
}
|
|
|
|
if (source->type != result->type) {
|
|
|
|
return CFI_INVALID_TYPE;
|
|
|
|
}
|
|
|
|
if (source->elem_len != result->elem_len) {
|
|
|
|
return CFI_INVALID_ELEM_LEN;
|
|
|
|
}
|
|
|
|
if (source->base_addr == nullptr &&
|
|
|
|
source->attribute != CFI_attribute_pointer) {
|
|
|
|
return CFI_ERROR_BASE_ADDR_NULL;
|
|
|
|
}
|
|
|
|
if (IsAssumedSize(source)) {
|
|
|
|
return CFI_INVALID_DESCRIPTOR;
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool copySrcLB{lower_bounds == nullptr};
|
|
|
|
result->base_addr = source->base_addr;
|
|
|
|
if (source->base_addr != nullptr) {
|
|
|
|
for (int j{0}; j < result->rank; ++j) {
|
|
|
|
result->dim[j].extent = source->dim[j].extent;
|
|
|
|
result->dim[j].sm = source->dim[j].sm;
|
|
|
|
result->dim[j].lower_bound =
|
|
|
|
copySrcLB ? source->dim[j].lower_bound : lower_bounds[j];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return CFI_SUCCESS;
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
} // extern "C"
|
2018-10-25 20:55:23 +08:00
|
|
|
}
|