Files
clang-p2996/flang/lib/Optimizer/Builder/Runtime/TemporaryStack.cpp
Jean Perier 0446bfcc5c [flang][hlfir] Codegen of hlfir.region_assign where LHS conflicts
When the analysis of hlfir.region_assign determined that the LHS region
evaluation may be impacted by the assignment effects, all LHS must be
fully evaluated and saved before any assignment is done.

This patch adds TemporaryStorage variants to save address, including
vector subscripted entities addresses whose shape must be saved.
It uses the DescriptorStack runtime to deal with complex cases inside
forall. For the sake of simplicity, this is also used for vector
subscripted LHS outside of foralls (each element address is saved as
a descriptor on this stack. This is a bit suboptimal, but it is a safe
start that will work with all kinds of type (polymorphic, PDTs...)
without further work). Another approach would be to saved only the
values that are conflicting in the LHS computation, but this would
require a much more complex analysis of the LHS region DAG.

Differential Revision: https://reviews.llvm.org/D154057
2023-06-30 09:20:52 +02:00

108 lines
5.3 KiB
C++

//===- TemporaryStack.cpp ---- temporary stack runtime API calls ----------===//
//
// 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/Optimizer/Builder/Runtime/TemporaryStack.h"
#include "flang/Optimizer/Builder/FIRBuilder.h"
#include "flang/Optimizer/Builder/Runtime/RTBuilder.h"
#include "flang/Runtime/temporary-stack.h"
using namespace Fortran::runtime;
mlir::Value fir::runtime::genCreateValueStack(mlir::Location loc,
fir::FirOpBuilder &builder) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(CreateValueStack)>(loc, builder);
mlir::FunctionType funcType = func.getFunctionType();
mlir::Value sourceFile = fir::factory::locationToFilename(builder, loc);
mlir::Value sourceLine =
fir::factory::locationToLineNo(builder, loc, funcType.getInput(1));
auto args = fir::runtime::createArguments(builder, loc, funcType, sourceFile,
sourceLine);
return builder.create<fir::CallOp>(loc, func, args).getResult(0);
}
void fir::runtime::genPushValue(mlir::Location loc, fir::FirOpBuilder &builder,
mlir::Value opaquePtr, mlir::Value boxValue) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(PushValue)>(loc, builder);
mlir::FunctionType funcType = func.getFunctionType();
auto args = fir::runtime::createArguments(builder, loc, funcType, opaquePtr,
boxValue);
builder.create<fir::CallOp>(loc, func, args);
}
void fir::runtime::genValueAt(mlir::Location loc, fir::FirOpBuilder &builder,
mlir::Value opaquePtr, mlir::Value i,
mlir::Value retValueBox) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(ValueAt)>(loc, builder);
mlir::FunctionType funcType = func.getFunctionType();
auto args = fir::runtime::createArguments(builder, loc, funcType, opaquePtr,
i, retValueBox);
builder.create<fir::CallOp>(loc, func, args);
}
void fir::runtime::genDestroyValueStack(mlir::Location loc,
fir::FirOpBuilder &builder,
mlir::Value opaquePtr) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(DestroyValueStack)>(loc, builder);
mlir::FunctionType funcType = func.getFunctionType();
auto args = fir::runtime::createArguments(builder, loc, funcType, opaquePtr);
builder.create<fir::CallOp>(loc, func, args);
}
mlir::Value fir::runtime::genCreateDescriptorStack(mlir::Location loc,
fir::FirOpBuilder &builder) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(CreateDescriptorStack)>(loc,
builder);
mlir::FunctionType funcType = func.getFunctionType();
mlir::Value sourceFile = fir::factory::locationToFilename(builder, loc);
mlir::Value sourceLine =
fir::factory::locationToLineNo(builder, loc, funcType.getInput(1));
auto args = fir::runtime::createArguments(builder, loc, funcType, sourceFile,
sourceLine);
return builder.create<fir::CallOp>(loc, func, args).getResult(0);
}
void fir::runtime::genPushDescriptor(mlir::Location loc,
fir::FirOpBuilder &builder,
mlir::Value opaquePtr,
mlir::Value boxDescriptor) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(PushDescriptor)>(loc, builder);
mlir::FunctionType funcType = func.getFunctionType();
auto args = fir::runtime::createArguments(builder, loc, funcType, opaquePtr,
boxDescriptor);
builder.create<fir::CallOp>(loc, func, args);
}
void fir::runtime::genDescriptorAt(mlir::Location loc,
fir::FirOpBuilder &builder,
mlir::Value opaquePtr, mlir::Value i,
mlir::Value retDescriptorBox) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(DescriptorAt)>(loc, builder);
mlir::FunctionType funcType = func.getFunctionType();
auto args = fir::runtime::createArguments(builder, loc, funcType, opaquePtr,
i, retDescriptorBox);
builder.create<fir::CallOp>(loc, func, args);
}
void fir::runtime::genDestroyDescriptorStack(mlir::Location loc,
fir::FirOpBuilder &builder,
mlir::Value opaquePtr) {
mlir::func::FuncOp func =
fir::runtime::getRuntimeFunc<mkRTKey(DestroyDescriptorStack)>(loc,
builder);
mlir::FunctionType funcType = func.getFunctionType();
auto args = fir::runtime::createArguments(builder, loc, funcType, opaquePtr);
builder.create<fir::CallOp>(loc, func, args);
}