In the context of C/Fortran interoperability (BIND(C)), it is possible to give the VALUE attribute to a BIND(C) derived type dummy, which according to Fortran 2018 18.3.6 - 2. (4) implies that it must be passed like the equivalent C structure value. The way C structure value are passed is ABI dependent. LLVM does not implement the C struct ABI passing for LLVM aggregate type arguments. It is up to the front-end, like clang is doing, to split the struct into registers or pass the struct on the stack (llvm "byval") as required by the target ABI. So the logic for C struct passing sits in clang. Using it from flang requires setting up a lot of clang context and to bridge FIR/MLIR representation to clang AST representation for function signatures (in both directions). It is a non trivial task. See https://stackoverflow.com/questions/39438033/passing-structs-by-value-in-llvm-ir/75002581#75002581. Since BIND(C) struct are rather limited as opposed to generic C struct (e.g. no bit fields). It is easier to provide a limited implementation of it for the case that matter to Fortran. This patch: - Updates the generic target rewrite pass to keep track of both the new argument type and attributes. The motivation for this is to be able to tell if a previously marshalled argument is passed in memory (it is a C pointer), or if it is being passed on the stack (has the byval llvm attributes). - Adds an entry point in the target specific codegen to marshal struct arguments, and use it in the generic target rewrite pass. - Implements limited support for the X86-64 case. So far, the support allows telling if a struct must be passed in register or on the stack, and to deal with the stack case. The register case is left TODO in this patch. The X86-64 ABI implemented is the System V ABI for AMD64 version 1.0
61 lines
2.4 KiB
C++
61 lines
2.4 KiB
C++
//===-- Optimizer/Support/DataLayout.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 "flang/Optimizer/Support/DataLayout.h"
|
|
#include "flang/Optimizer/Dialect/Support/FIRContext.h"
|
|
#include "flang/Optimizer/Support/FatalError.h"
|
|
#include "mlir/Dialect/DLTI/DLTI.h"
|
|
#include "mlir/Dialect/LLVMIR/LLVMDialect.h"
|
|
#include "mlir/IR/BuiltinOps.h"
|
|
#include "mlir/Interfaces/DataLayoutInterfaces.h"
|
|
#include "mlir/Support/LLVM.h"
|
|
#include "mlir/Target/LLVMIR/Import.h"
|
|
#include "llvm/IR/DataLayout.h"
|
|
#include "llvm/MC/TargetRegistry.h"
|
|
#include "llvm/Support/TargetSelect.h"
|
|
#include "llvm/Target/TargetMachine.h"
|
|
|
|
void fir::support::setMLIRDataLayout(mlir::ModuleOp mlirModule,
|
|
const llvm::DataLayout &dl) {
|
|
mlir::MLIRContext *context = mlirModule.getContext();
|
|
mlirModule->setAttr(
|
|
mlir::LLVM::LLVMDialect::getDataLayoutAttrName(),
|
|
mlir::StringAttr::get(context, dl.getStringRepresentation()));
|
|
mlir::DataLayoutSpecInterface dlSpec = mlir::translateDataLayout(dl, context);
|
|
mlirModule->setAttr(mlir::DLTIDialect::kDataLayoutAttrName, dlSpec);
|
|
}
|
|
|
|
void fir::support::setMLIRDataLayoutFromAttributes(mlir::ModuleOp mlirModule,
|
|
bool allowDefaultLayout) {
|
|
if (mlirModule.getDataLayoutSpec())
|
|
return; // Already set.
|
|
if (auto dataLayoutString = mlirModule->getAttrOfType<mlir::StringAttr>(
|
|
mlir::LLVM::LLVMDialect::getDataLayoutAttrName())) {
|
|
llvm::DataLayout llvmDataLayout(dataLayoutString);
|
|
fir::support::setMLIRDataLayout(mlirModule, llvmDataLayout);
|
|
return;
|
|
}
|
|
if (!allowDefaultLayout)
|
|
return;
|
|
llvm::DataLayout llvmDataLayout("");
|
|
fir::support::setMLIRDataLayout(mlirModule, llvmDataLayout);
|
|
}
|
|
|
|
std::optional<mlir::DataLayout>
|
|
fir::support::getOrSetDataLayout(mlir::ModuleOp mlirModule,
|
|
bool allowDefaultLayout) {
|
|
if (!mlirModule.getDataLayoutSpec()) {
|
|
fir::support::setMLIRDataLayoutFromAttributes(mlirModule,
|
|
allowDefaultLayout);
|
|
if (!mlirModule.getDataLayoutSpec()) {
|
|
return std::nullopt;
|
|
}
|
|
}
|
|
return mlir::DataLayout(mlirModule);
|
|
}
|