2020-01-28 10:18:45 +08:00
|
|
|
//===-- runtime/descriptor.cpp --------------------------------------------===//
|
2018-05-17 01:22:33 +08:00
|
|
|
//
|
2019-12-21 04:52:07 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-05-17 01:22:33 +08:00
|
|
|
//
|
2020-01-11 04:12:03 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2018-05-17 01:22:33 +08:00
|
|
|
|
|
|
|
#include "descriptor.h"
|
2020-02-25 23:11:52 +08:00
|
|
|
#include "flang/Common/idioms.h"
|
2018-08-01 07:46:30 +08:00
|
|
|
#include <cassert>
|
2018-07-27 07:07:50 +08:00
|
|
|
#include <cstdlib>
|
2020-02-05 08:55:45 +08:00
|
|
|
#include <cstring>
|
2018-05-17 01:22:33 +08:00
|
|
|
|
|
|
|
namespace Fortran::runtime {
|
|
|
|
|
2020-02-05 08:55:45 +08:00
|
|
|
Descriptor::Descriptor(const Descriptor &that) {
|
|
|
|
std::memcpy(this, &that, that.SizeInBytes());
|
|
|
|
}
|
|
|
|
|
2018-08-02 01:55:46 +08:00
|
|
|
Descriptor::~Descriptor() {
|
2018-08-03 08:04:31 +08:00
|
|
|
if (raw_.attribute != CFI_attribute_pointer) {
|
|
|
|
Deallocate();
|
|
|
|
}
|
2018-08-02 01:55:46 +08:00
|
|
|
}
|
2018-07-27 07:07:50 +08:00
|
|
|
|
2018-08-03 02:45:11 +08:00
|
|
|
void Descriptor::Establish(TypeCode t, std::size_t elementBytes, void *p,
|
2018-08-02 01:55:46 +08:00
|
|
|
int rank, const SubscriptValue *extent, ISO::CFI_attribute_t attribute,
|
|
|
|
bool addendum) {
|
2018-08-03 02:45:11 +08:00
|
|
|
CHECK(ISO::CFI_establish(&raw_, p, attribute, t.raw(), elementBytes, rank,
|
|
|
|
extent) == CFI_SUCCESS);
|
2018-08-02 01:55:46 +08:00
|
|
|
raw_.f18Addendum = addendum;
|
2018-08-07 00:43:43 +08:00
|
|
|
DescriptorAddendum *a{Addendum()};
|
|
|
|
CHECK(addendum == (a != nullptr));
|
|
|
|
if (a) {
|
|
|
|
new (a) DescriptorAddendum{};
|
2018-08-03 02:45:11 +08:00
|
|
|
}
|
2018-07-27 07:07:50 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 02:45:11 +08:00
|
|
|
void Descriptor::Establish(TypeCategory c, int kind, void *p, int rank,
|
2018-08-02 01:55:46 +08:00
|
|
|
const SubscriptValue *extent, ISO::CFI_attribute_t attribute,
|
|
|
|
bool addendum) {
|
2018-07-28 07:52:17 +08:00
|
|
|
std::size_t elementBytes = kind;
|
2018-08-02 00:45:59 +08:00
|
|
|
if (c == TypeCategory::Complex) {
|
2018-07-28 07:52:17 +08:00
|
|
|
elementBytes *= 2;
|
|
|
|
}
|
2018-08-03 02:45:11 +08:00
|
|
|
CHECK(ISO::CFI_establish(&raw_, p, attribute, TypeCode(c, kind).raw(),
|
|
|
|
elementBytes, rank, extent) == CFI_SUCCESS);
|
2018-08-02 01:55:46 +08:00
|
|
|
raw_.f18Addendum = addendum;
|
2018-08-07 00:43:43 +08:00
|
|
|
DescriptorAddendum *a{Addendum()};
|
|
|
|
CHECK(addendum == (a != nullptr));
|
|
|
|
if (a) {
|
|
|
|
new (a) DescriptorAddendum{};
|
2018-08-03 02:45:11 +08:00
|
|
|
}
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 02:45:11 +08:00
|
|
|
void Descriptor::Establish(const DerivedType &dt, void *p, int rank,
|
2018-08-02 01:55:46 +08:00
|
|
|
const SubscriptValue *extent, ISO::CFI_attribute_t attribute) {
|
2018-08-03 02:45:11 +08:00
|
|
|
CHECK(ISO::CFI_establish(&raw_, p, attribute, CFI_type_struct,
|
|
|
|
dt.SizeInBytes(), rank, extent) == CFI_SUCCESS);
|
2018-08-02 01:55:46 +08:00
|
|
|
raw_.f18Addendum = true;
|
2018-08-07 00:43:43 +08:00
|
|
|
DescriptorAddendum *a{Addendum()};
|
2019-11-10 01:29:31 +08:00
|
|
|
CHECK(a);
|
2018-08-07 00:43:43 +08:00
|
|
|
new (a) DescriptorAddendum{&dt};
|
2018-07-27 07:07:50 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 08:04:31 +08:00
|
|
|
std::unique_ptr<Descriptor> Descriptor::Create(TypeCode t,
|
|
|
|
std::size_t elementBytes, void *p, int rank, const SubscriptValue *extent,
|
|
|
|
ISO::CFI_attribute_t attribute) {
|
2018-08-03 02:45:11 +08:00
|
|
|
std::size_t bytes{SizeInBytes(rank, true)};
|
2018-08-03 08:04:31 +08:00
|
|
|
Descriptor *result{reinterpret_cast<Descriptor *>(new char[bytes])};
|
2019-11-10 01:29:31 +08:00
|
|
|
CHECK(result);
|
2018-08-02 01:55:46 +08:00
|
|
|
result->Establish(t, elementBytes, p, rank, extent, attribute, true);
|
2018-08-03 08:04:31 +08:00
|
|
|
return std::unique_ptr<Descriptor>{result};
|
2018-07-27 07:07:50 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 08:04:31 +08:00
|
|
|
std::unique_ptr<Descriptor> Descriptor::Create(TypeCategory c, int kind,
|
|
|
|
void *p, int rank, const SubscriptValue *extent,
|
|
|
|
ISO::CFI_attribute_t attribute) {
|
2018-08-03 02:45:11 +08:00
|
|
|
std::size_t bytes{SizeInBytes(rank, true)};
|
2018-08-03 08:04:31 +08:00
|
|
|
Descriptor *result{reinterpret_cast<Descriptor *>(new char[bytes])};
|
2019-11-10 01:29:31 +08:00
|
|
|
CHECK(result);
|
2018-08-02 01:55:46 +08:00
|
|
|
result->Establish(c, kind, p, rank, extent, attribute, true);
|
2018-08-03 08:04:31 +08:00
|
|
|
return std::unique_ptr<Descriptor>{result};
|
2018-07-27 07:07:50 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 08:04:31 +08:00
|
|
|
std::unique_ptr<Descriptor> Descriptor::Create(const DerivedType &dt, void *p,
|
|
|
|
int rank, const SubscriptValue *extent, ISO::CFI_attribute_t attribute) {
|
2018-08-03 02:45:11 +08:00
|
|
|
std::size_t bytes{SizeInBytes(rank, true, dt.lenParameters())};
|
2018-08-03 08:04:31 +08:00
|
|
|
Descriptor *result{reinterpret_cast<Descriptor *>(new char[bytes])};
|
2019-11-10 01:29:31 +08:00
|
|
|
CHECK(result);
|
2018-08-02 01:55:46 +08:00
|
|
|
result->Establish(dt, p, rank, extent, attribute);
|
2018-08-03 08:04:31 +08:00
|
|
|
return std::unique_ptr<Descriptor>{result};
|
2018-08-01 07:46:30 +08:00
|
|
|
}
|
2018-07-28 07:52:17 +08:00
|
|
|
|
|
|
|
std::size_t Descriptor::SizeInBytes() const {
|
|
|
|
const DescriptorAddendum *addendum{Addendum()};
|
|
|
|
return sizeof *this - sizeof(Dimension) + raw_.rank * sizeof(Dimension) +
|
|
|
|
(addendum ? addendum->SizeInBytes() : 0);
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 02:45:11 +08:00
|
|
|
std::size_t Descriptor::Elements() const {
|
|
|
|
int n{rank()};
|
|
|
|
std::size_t elements{1};
|
|
|
|
for (int j{0}; j < n; ++j) {
|
|
|
|
elements *= GetDimension(j).Extent();
|
|
|
|
}
|
|
|
|
return elements;
|
|
|
|
}
|
|
|
|
|
2018-08-03 08:04:31 +08:00
|
|
|
int Descriptor::Allocate(
|
|
|
|
const SubscriptValue lb[], const SubscriptValue ub[], std::size_t charLen) {
|
|
|
|
int result{ISO::CFI_allocate(&raw_, lb, ub, charLen)};
|
|
|
|
if (result == CFI_SUCCESS) {
|
|
|
|
// TODO: derived type initialization
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Descriptor::Deallocate(bool finalize) {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (raw_.base_addr) {
|
2018-08-03 08:04:31 +08:00
|
|
|
Destroy(static_cast<char *>(raw_.base_addr), finalize);
|
|
|
|
}
|
|
|
|
return ISO::CFI_deallocate(&raw_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Descriptor::Destroy(char *data, bool finalize) const {
|
2019-11-10 01:29:31 +08:00
|
|
|
if (data) {
|
2018-08-03 08:04:31 +08:00
|
|
|
if (const DescriptorAddendum * addendum{Addendum()}) {
|
|
|
|
if (addendum->flags() & DescriptorAddendum::DoNotFinalize) {
|
|
|
|
finalize = false;
|
|
|
|
}
|
|
|
|
if (const DerivedType * dt{addendum->derivedType()}) {
|
|
|
|
std::size_t elements{Elements()};
|
|
|
|
std::size_t elementBytes{ElementBytes()};
|
|
|
|
for (std::size_t j{0}; j < elements; ++j) {
|
|
|
|
dt->Destroy(data + j * elementBytes, finalize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-01 07:46:30 +08:00
|
|
|
void Descriptor::Check() const {
|
|
|
|
// TODO
|
2018-05-17 01:22:33 +08:00
|
|
|
}
|
2018-07-27 07:07:50 +08:00
|
|
|
|
2018-08-03 08:04:31 +08:00
|
|
|
std::ostream &Descriptor::Dump(std::ostream &o) const {
|
|
|
|
o << "Descriptor @ 0x" << std::hex << reinterpret_cast<std::intptr_t>(this)
|
|
|
|
<< std::dec << ":\n";
|
|
|
|
o << " base_addr 0x" << std::hex
|
|
|
|
<< reinterpret_cast<std::intptr_t>(raw_.base_addr) << std::dec << '\n';
|
|
|
|
o << " elem_len " << raw_.elem_len << '\n';
|
|
|
|
o << " version " << raw_.version
|
|
|
|
<< (raw_.version == CFI_VERSION ? "(ok)" : "BAD!") << '\n';
|
|
|
|
o << " rank " << static_cast<int>(raw_.rank) << '\n';
|
|
|
|
o << " type " << static_cast<int>(raw_.type) << '\n';
|
|
|
|
o << " attribute " << static_cast<int>(raw_.attribute) << '\n';
|
|
|
|
o << " addendum? " << static_cast<bool>(raw_.f18Addendum) << '\n';
|
|
|
|
for (int j{0}; j < raw_.rank; ++j) {
|
|
|
|
o << " dim[" << j << "] lower_bound " << raw_.dim[j].lower_bound << '\n';
|
|
|
|
o << " extent " << raw_.dim[j].extent << '\n';
|
|
|
|
o << " sm " << raw_.dim[j].sm << '\n';
|
|
|
|
}
|
|
|
|
if (const DescriptorAddendum * addendum{Addendum()}) {
|
|
|
|
addendum->Dump(o);
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2018-08-01 07:46:30 +08:00
|
|
|
std::size_t DescriptorAddendum::SizeInBytes() const {
|
2018-08-03 02:45:11 +08:00
|
|
|
return SizeInBytes(LenParameters());
|
2018-07-27 07:07:50 +08:00
|
|
|
}
|
2018-08-03 08:04:31 +08:00
|
|
|
|
|
|
|
std::ostream &DescriptorAddendum::Dump(std::ostream &o) const {
|
|
|
|
o << " derivedType @ 0x" << std::hex
|
|
|
|
<< reinterpret_cast<std::intptr_t>(derivedType_) << std::dec << '\n';
|
|
|
|
o << " flags " << flags_ << '\n';
|
|
|
|
// TODO: LEN parameter values
|
|
|
|
return o;
|
|
|
|
}
|
2018-10-25 20:55:23 +08:00
|
|
|
}
|