
Mostly mechanical changes in preparation of extracting the Flang-RT "subproject" in #110217. This PR intends to only move pre-existing files to the new folder structure, with no behavioral change. Common files (headers, testing, cmake) shared by Flang-RT and Flang remain in `flang/`. Some cosmetic changes and files paths were necessary: * Relative paths to the new path for the source files and `add_subdirectory`. * Add the new location's include directory to `include_directories` * The unittest/Evaluate directory has unitests for flang-rt and Flang. A new `CMakeLists.txt` was introduced for the flang-rt tests. * Change the `#include` paths relative to the include directive * clang-format on the `#include` directives * Since the paths are part if the copyright header and include guards, a script was used to canonicalize those * `test/Runtime` and runtime tests in `test/Driver` are moved, but the lit.cfg.py mechanism to execute the will only be added in #110217.
86 lines
2.9 KiB
C++
86 lines
2.9 KiB
C++
//===-- lib/runtime/ragged.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 "flang/Runtime/ragged.h"
|
|
#include "flang-rt/runtime/tools.h"
|
|
#include <cstdlib>
|
|
|
|
namespace Fortran::runtime {
|
|
|
|
inline RT_API_ATTRS bool isIndirection(const RaggedArrayHeader *const header) {
|
|
return header->flags & 1;
|
|
}
|
|
|
|
inline RT_API_ATTRS std::size_t rank(const RaggedArrayHeader *const header) {
|
|
return header->flags >> 1;
|
|
}
|
|
|
|
RT_API_ATTRS RaggedArrayHeader *RaggedArrayAllocate(RaggedArrayHeader *header,
|
|
bool isHeader, std::int64_t rank, std::int64_t elementSize,
|
|
std::int64_t *extentVector) {
|
|
if (header && rank) {
|
|
std::int64_t size{1};
|
|
for (std::int64_t counter{0}; counter < rank; ++counter) {
|
|
size *= extentVector[counter];
|
|
if (size <= 0) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
header->flags = (rank << 1) | isHeader;
|
|
header->extentPointer = extentVector;
|
|
if (isHeader) {
|
|
elementSize = sizeof(RaggedArrayHeader);
|
|
}
|
|
Terminator terminator{__FILE__, __LINE__};
|
|
std::size_t bytes{static_cast<std::size_t>(elementSize * size)};
|
|
header->bufferPointer = AllocateMemoryOrCrash(terminator, bytes);
|
|
if (header->bufferPointer) {
|
|
std::memset(header->bufferPointer, 0, bytes);
|
|
}
|
|
return header;
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
// Deallocate a ragged array from the heap.
|
|
RT_API_ATTRS void RaggedArrayDeallocate(RaggedArrayHeader *raggedArrayHeader) {
|
|
if (raggedArrayHeader) {
|
|
if (std::size_t end{rank(raggedArrayHeader)}) {
|
|
if (isIndirection(raggedArrayHeader)) {
|
|
std::size_t linearExtent{1u};
|
|
for (std::size_t counter{0u}; counter < end && linearExtent > 0;
|
|
++counter) {
|
|
linearExtent *= raggedArrayHeader->extentPointer[counter];
|
|
}
|
|
for (std::size_t counter{0u}; counter < linearExtent; ++counter) {
|
|
RaggedArrayDeallocate(&static_cast<RaggedArrayHeader *>(
|
|
raggedArrayHeader->bufferPointer)[counter]);
|
|
}
|
|
}
|
|
std::free(raggedArrayHeader->bufferPointer);
|
|
std::free(raggedArrayHeader->extentPointer);
|
|
raggedArrayHeader->flags = 0u;
|
|
}
|
|
}
|
|
}
|
|
|
|
extern "C" {
|
|
void *RTDEF(RaggedArrayAllocate)(void *header, bool isHeader, std::int64_t rank,
|
|
std::int64_t elementSize, std::int64_t *extentVector) {
|
|
auto *result = RaggedArrayAllocate(static_cast<RaggedArrayHeader *>(header),
|
|
isHeader, rank, elementSize, extentVector);
|
|
return static_cast<void *>(result);
|
|
}
|
|
|
|
void RTDEF(RaggedArrayDeallocate)(void *raggedArrayHeader) {
|
|
RaggedArrayDeallocate(static_cast<RaggedArrayHeader *>(raggedArrayHeader));
|
|
}
|
|
} // extern "C"
|
|
} // namespace Fortran::runtime
|