2020-12-08 06:46:24 +08:00
|
|
|
//===-- runtime/derived.cpp -----------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "derived.h"
|
|
|
|
#include "descriptor.h"
|
|
|
|
#include "type-info.h"
|
|
|
|
|
|
|
|
namespace Fortran::runtime {
|
|
|
|
|
|
|
|
static const typeInfo::SpecialBinding *FindFinal(
|
|
|
|
const typeInfo::DerivedType &derived, int rank) {
|
|
|
|
const typeInfo::SpecialBinding *elemental{nullptr};
|
2021-06-18 04:13:19 +08:00
|
|
|
const Descriptor &specialDesc{derived.special()};
|
2020-12-08 06:46:24 +08:00
|
|
|
std::size_t totalSpecialBindings{specialDesc.Elements()};
|
|
|
|
for (std::size_t j{0}; j < totalSpecialBindings; ++j) {
|
|
|
|
const auto &special{
|
|
|
|
*specialDesc.ZeroBasedIndexedElement<typeInfo::SpecialBinding>(j)};
|
|
|
|
switch (special.which) {
|
|
|
|
case typeInfo::SpecialBinding::Which::Final:
|
|
|
|
if (special.rank == rank) {
|
|
|
|
return &special;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case typeInfo::SpecialBinding::Which::ElementalFinal:
|
|
|
|
elemental = &special;
|
|
|
|
break;
|
|
|
|
case typeInfo::SpecialBinding::Which::AssumedRankFinal:
|
|
|
|
return &special;
|
|
|
|
default:;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return elemental;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void CallFinalSubroutine(
|
|
|
|
const Descriptor &descriptor, const typeInfo::DerivedType &derived) {
|
|
|
|
if (const auto *special{FindFinal(derived, descriptor.rank())}) {
|
|
|
|
if (special->which == typeInfo::SpecialBinding::Which::ElementalFinal) {
|
|
|
|
std::size_t byteStride{descriptor.ElementBytes()};
|
|
|
|
auto p{reinterpret_cast<void (*)(char *)>(special->proc)};
|
|
|
|
// Finalizable objects must be contiguous.
|
|
|
|
std::size_t elements{descriptor.Elements()};
|
|
|
|
for (std::size_t j{0}; j < elements; ++j) {
|
|
|
|
p(descriptor.OffsetElement<char>(j * byteStride));
|
|
|
|
}
|
|
|
|
} else if (special->isArgDescriptorSet & 1) {
|
|
|
|
auto p{reinterpret_cast<void (*)(const Descriptor &)>(special->proc)};
|
|
|
|
p(descriptor);
|
|
|
|
} else {
|
|
|
|
// Finalizable objects must be contiguous.
|
|
|
|
auto p{reinterpret_cast<void (*)(char *)>(special->proc)};
|
|
|
|
p(descriptor.OffsetElement<char>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The order of finalization follows Fortran 2018 7.5.6.2, with
|
|
|
|
// deallocation of non-parent components (and their consequent finalization)
|
|
|
|
// taking place before parent component finalization.
|
|
|
|
void Destroy(const Descriptor &descriptor, bool finalize,
|
|
|
|
const typeInfo::DerivedType &derived) {
|
|
|
|
if (finalize) {
|
|
|
|
CallFinalSubroutine(descriptor, derived);
|
|
|
|
}
|
2021-06-18 04:13:19 +08:00
|
|
|
const Descriptor &componentDesc{derived.component()};
|
|
|
|
auto myComponents{static_cast<SubscriptValue>(componentDesc.Elements())};
|
2020-12-08 06:46:24 +08:00
|
|
|
std::size_t elements{descriptor.Elements()};
|
|
|
|
std::size_t byteStride{descriptor.ElementBytes()};
|
|
|
|
for (unsigned k{0}; k < myComponents; ++k) {
|
|
|
|
const auto &comp{
|
|
|
|
*componentDesc.ZeroBasedIndexedElement<typeInfo::Component>(k)};
|
2021-06-18 04:13:19 +08:00
|
|
|
if (comp.genre() == typeInfo::Component::Genre::Allocatable ||
|
|
|
|
comp.genre() == typeInfo::Component::Genre::Automatic) {
|
2020-12-08 06:46:24 +08:00
|
|
|
for (std::size_t j{0}; j < elements; ++j) {
|
2021-06-18 04:13:19 +08:00
|
|
|
descriptor.OffsetElement<Descriptor>(j * byteStride + comp.offset())
|
2020-12-08 06:46:24 +08:00
|
|
|
->Deallocate(finalize);
|
|
|
|
}
|
2021-06-18 04:13:19 +08:00
|
|
|
} else if (comp.genre() == typeInfo::Component::Genre::Data &&
|
|
|
|
comp.derivedType()) {
|
2020-12-08 06:46:24 +08:00
|
|
|
SubscriptValue extent[maxRank];
|
2021-06-18 04:13:19 +08:00
|
|
|
const typeInfo::Value *bounds{comp.bounds()};
|
|
|
|
for (int dim{0}; dim < comp.rank(); ++dim) {
|
|
|
|
extent[dim] = bounds[2 * dim].GetValue(&descriptor).value_or(0) -
|
|
|
|
bounds[2 * dim + 1].GetValue(&descriptor).value_or(0) + 1;
|
2020-12-08 06:46:24 +08:00
|
|
|
}
|
|
|
|
StaticDescriptor<maxRank, true, 0> staticDescriptor;
|
|
|
|
Descriptor &compDesc{staticDescriptor.descriptor()};
|
2021-06-18 04:13:19 +08:00
|
|
|
const typeInfo::DerivedType &compType{*comp.derivedType()};
|
2020-12-08 06:46:24 +08:00
|
|
|
for (std::size_t j{0}; j < elements; ++j) {
|
|
|
|
compDesc.Establish(compType,
|
2021-06-18 04:13:19 +08:00
|
|
|
descriptor.OffsetElement<char>(j * byteStride + comp.offset()),
|
|
|
|
comp.rank(), extent);
|
2020-12-08 06:46:24 +08:00
|
|
|
Destroy(compDesc, finalize, compType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-06-18 04:13:19 +08:00
|
|
|
const Descriptor &parentDesc{derived.parent()};
|
2020-12-08 06:46:24 +08:00
|
|
|
if (const auto *parent{parentDesc.OffsetElement<typeInfo::DerivedType>()}) {
|
|
|
|
Destroy(descriptor, finalize, *parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} // namespace Fortran::runtime
|