forked from OSchip/llvm-project
[flang] C_F_POINTER
Emit INTRINSIC statements in module files Argument checking utility Complete error checking Original-commit: flang-compiler/f18@9c6a88f048 Reviewed-on: https://github.com/flang-compiler/f18/pull/896
This commit is contained in:
parent
0e5c4272ea
commit
663db2741d
|
@ -1519,6 +1519,8 @@ private:
|
||||||
DynamicType GetSpecificType(const TypePattern &) const;
|
DynamicType GetSpecificType(const TypePattern &) const;
|
||||||
SpecificCall HandleNull(
|
SpecificCall HandleNull(
|
||||||
ActualArguments &, FoldingContext &, const IntrinsicProcTable &) const;
|
ActualArguments &, FoldingContext &, const IntrinsicProcTable &) const;
|
||||||
|
std::optional<SpecificCall> HandleC_F_Pointer(
|
||||||
|
ActualArguments &, FoldingContext &) const;
|
||||||
|
|
||||||
common::IntrinsicTypeDefaultKinds defaults_;
|
common::IntrinsicTypeDefaultKinds defaults_;
|
||||||
std::multimap<std::string, const IntrinsicInterface *> genericFuncs_;
|
std::multimap<std::string, const IntrinsicInterface *> genericFuncs_;
|
||||||
|
@ -1541,61 +1543,113 @@ bool IntrinsicProcTable::Implementation::IsIntrinsic(
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
// special cases
|
// special cases
|
||||||
return name == "null"; // TODO more
|
return name == "null" || name == "__builtin_c_f_pointer";
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CheckAndRearrangeArguments(ActualArguments &arguments,
|
||||||
|
parser::ContextualMessages &messages, const char *const dummyKeywords[],
|
||||||
|
std::size_t trailingOptionals) {
|
||||||
|
std::size_t numDummies{0};
|
||||||
|
while (dummyKeywords[numDummies]) {
|
||||||
|
++numDummies;
|
||||||
|
}
|
||||||
|
CHECK(trailingOptionals <= numDummies);
|
||||||
|
if (arguments.size() > numDummies) {
|
||||||
|
messages.Say("Too many actual arguments (%zd > %zd)"_err_en_US,
|
||||||
|
arguments.size(), numDummies);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
ActualArguments rearranged(numDummies);
|
||||||
|
bool anyKeywords{false};
|
||||||
|
std::size_t position{0};
|
||||||
|
for (std::optional<ActualArgument> &arg : arguments) {
|
||||||
|
std::size_t dummyIndex{0};
|
||||||
|
if (arg && arg->keyword()) {
|
||||||
|
anyKeywords = true;
|
||||||
|
for (; dummyIndex < numDummies; ++dummyIndex) {
|
||||||
|
if (*arg->keyword() == dummyKeywords[dummyIndex]) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (dummyIndex >= numDummies) {
|
||||||
|
messages.Say(*arg->keyword(),
|
||||||
|
"Unknown argument keyword '%s='"_err_en_US, *arg->keyword());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else if (anyKeywords) {
|
||||||
|
messages.Say(
|
||||||
|
"A positional actual argument may not appear after any keyword arguments"_err_en_US);
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
dummyIndex = position++;
|
||||||
|
}
|
||||||
|
if (rearranged[dummyIndex]) {
|
||||||
|
messages.Say("Dummy argument '%s=' appears more than once"_err_en_US,
|
||||||
|
dummyKeywords[dummyIndex]);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
rearranged[dummyIndex] = std::move(arg);
|
||||||
|
arg.reset();
|
||||||
|
}
|
||||||
|
bool anyMissing{false};
|
||||||
|
for (std::size_t j{0}; j < numDummies - trailingOptionals; ++j) {
|
||||||
|
if (!rearranged[j]) {
|
||||||
|
messages.Say("Dummy argument '%s=' is absent and not OPTIONAL"_err_en_US,
|
||||||
|
dummyKeywords[j]);
|
||||||
|
anyMissing = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
arguments = std::move(rearranged);
|
||||||
|
return !anyMissing;
|
||||||
}
|
}
|
||||||
|
|
||||||
// The NULL() intrinsic is a special case.
|
// The NULL() intrinsic is a special case.
|
||||||
SpecificCall IntrinsicProcTable::Implementation::HandleNull(
|
SpecificCall IntrinsicProcTable::Implementation::HandleNull(
|
||||||
ActualArguments &arguments, FoldingContext &context,
|
ActualArguments &arguments, FoldingContext &context,
|
||||||
const IntrinsicProcTable &intrinsics) const {
|
const IntrinsicProcTable &intrinsics) const {
|
||||||
if (!arguments.empty()) {
|
static const char *const keywords[]{"mold", nullptr};
|
||||||
if (arguments.size() > 1) {
|
if (CheckAndRearrangeArguments(arguments, context.messages(), keywords, 1) &&
|
||||||
context.messages().Say("Too many arguments to NULL()"_err_en_US);
|
arguments[0]) {
|
||||||
} else if (arguments[0] && arguments[0]->keyword() &&
|
if (Expr<SomeType> * mold{arguments[0]->UnwrapExpr()}) {
|
||||||
arguments[0]->keyword()->ToString() != "mold") {
|
if (IsAllocatableOrPointer(*mold)) {
|
||||||
context.messages().Say("Unknown argument '%s' to NULL()"_err_en_US,
|
characteristics::DummyArguments args;
|
||||||
arguments[0]->keyword()->ToString());
|
std::optional<characteristics::FunctionResult> fResult;
|
||||||
} else {
|
if (IsProcedurePointer(*mold)) {
|
||||||
if (Expr<SomeType> * mold{arguments[0]->UnwrapExpr()}) {
|
// MOLD= procedure pointer
|
||||||
if (IsAllocatableOrPointer(*mold)) {
|
const Symbol *last{GetLastSymbol(*mold)};
|
||||||
characteristics::DummyArguments args;
|
CHECK(last);
|
||||||
std::optional<characteristics::FunctionResult> fResult;
|
auto procPointer{
|
||||||
if (IsProcedurePointer(*mold)) {
|
characteristics::Procedure::Characterize(*last, intrinsics)};
|
||||||
// MOLD= procedure pointer
|
CHECK(procPointer);
|
||||||
const Symbol *last{GetLastSymbol(*mold)};
|
args.emplace_back("mold"s,
|
||||||
CHECK(last);
|
characteristics::DummyProcedure{common::Clone(*procPointer)});
|
||||||
auto procPointer{
|
fResult.emplace(std::move(*procPointer));
|
||||||
characteristics::Procedure::Characterize(*last, intrinsics)};
|
} else if (auto type{mold->GetType()}) {
|
||||||
CHECK(procPointer);
|
// MOLD= object pointer
|
||||||
args.emplace_back("mold"s,
|
characteristics::TypeAndShape typeAndShape{
|
||||||
characteristics::DummyProcedure{common::Clone(*procPointer)});
|
*type, GetShape(context, *mold)};
|
||||||
fResult.emplace(std::move(*procPointer));
|
args.emplace_back(
|
||||||
} else if (auto type{mold->GetType()}) {
|
"mold"s, characteristics::DummyDataObject{typeAndShape});
|
||||||
// MOLD= object pointer
|
fResult.emplace(std::move(typeAndShape));
|
||||||
characteristics::TypeAndShape typeAndShape{
|
} else {
|
||||||
*type, GetShape(context, *mold)};
|
context.messages().Say(
|
||||||
args.emplace_back(
|
"MOLD= argument to NULL() lacks type"_err_en_US);
|
||||||
"mold"s, characteristics::DummyDataObject{typeAndShape});
|
|
||||||
fResult.emplace(std::move(typeAndShape));
|
|
||||||
} else {
|
|
||||||
context.messages().Say(
|
|
||||||
"MOLD= argument to NULL() lacks type"_err_en_US);
|
|
||||||
}
|
|
||||||
fResult->attrs.set(characteristics::FunctionResult::Attr::Pointer);
|
|
||||||
characteristics::Procedure::Attrs attrs;
|
|
||||||
attrs.set(characteristics::Procedure::Attr::NullPointer);
|
|
||||||
characteristics::Procedure chars{
|
|
||||||
std::move(*fResult), std::move(args), attrs};
|
|
||||||
return SpecificCall{SpecificIntrinsic{"null"s, std::move(chars)},
|
|
||||||
std::move(arguments)};
|
|
||||||
}
|
}
|
||||||
|
fResult->attrs.set(characteristics::FunctionResult::Attr::Pointer);
|
||||||
|
characteristics::Procedure::Attrs attrs;
|
||||||
|
attrs.set(characteristics::Procedure::Attr::NullPointer);
|
||||||
|
characteristics::Procedure chars{
|
||||||
|
std::move(*fResult), std::move(args), attrs};
|
||||||
|
return SpecificCall{
|
||||||
|
SpecificIntrinsic{"null"s, std::move(chars)}, std::move(arguments)};
|
||||||
}
|
}
|
||||||
context.messages().Say(
|
|
||||||
"MOLD= argument to NULL() must be a pointer or allocatable"_err_en_US);
|
|
||||||
}
|
}
|
||||||
|
context.messages().Say(
|
||||||
|
"MOLD= argument to NULL() must be a pointer or allocatable"_err_en_US);
|
||||||
}
|
}
|
||||||
characteristics::Procedure::Attrs attrs;
|
characteristics::Procedure::Attrs attrs;
|
||||||
attrs.set(characteristics::Procedure::Attr::NullPointer);
|
attrs.set(characteristics::Procedure::Attr::NullPointer);
|
||||||
|
attrs.set(characteristics::Procedure::Attr::Pure);
|
||||||
arguments.clear();
|
arguments.clear();
|
||||||
return SpecificCall{
|
return SpecificCall{
|
||||||
SpecificIntrinsic{"null"s,
|
SpecificIntrinsic{"null"s,
|
||||||
|
@ -1603,6 +1657,77 @@ SpecificCall IntrinsicProcTable::Implementation::HandleNull(
|
||||||
std::move(arguments)};
|
std::move(arguments)};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Subroutine C_F_POINTER(CPTR=,FPTR=[,SHAPE=]) from
|
||||||
|
// intrinsic module ISO_C_BINDING (18.2.3.3)
|
||||||
|
std::optional<SpecificCall>
|
||||||
|
IntrinsicProcTable::Implementation::HandleC_F_Pointer(
|
||||||
|
ActualArguments &arguments, FoldingContext &context) const {
|
||||||
|
characteristics::Procedure::Attrs attrs;
|
||||||
|
attrs.set(characteristics::Procedure::Attr::Subroutine);
|
||||||
|
static const char *const keywords[]{"cptr", "fptr", "shape", nullptr};
|
||||||
|
characteristics::DummyArguments dummies;
|
||||||
|
if (CheckAndRearrangeArguments(arguments, context.messages(), keywords, 1)) {
|
||||||
|
CHECK(arguments.size() == 3);
|
||||||
|
if (const auto *expr{arguments[0].value().UnwrapExpr()}) {
|
||||||
|
if (expr->Rank() > 0) {
|
||||||
|
context.messages().Say(
|
||||||
|
"CPTR= argument to C_F_POINTER() must be scalar"_err_en_US);
|
||||||
|
}
|
||||||
|
if (auto type{expr->GetType()}) {
|
||||||
|
if (type->category() != TypeCategory::Derived ||
|
||||||
|
type->IsPolymorphic() ||
|
||||||
|
type->GetDerivedTypeSpec().typeSymbol().name() !=
|
||||||
|
"__builtin_c_ptr") {
|
||||||
|
context.messages().Say(
|
||||||
|
"CPTR= argument to C_F_POINTER() must be a C_PTR"_err_en_US);
|
||||||
|
}
|
||||||
|
characteristics::DummyDataObject cptr{
|
||||||
|
characteristics::TypeAndShape{*type}};
|
||||||
|
cptr.intent = common::Intent::In;
|
||||||
|
dummies.emplace_back("cptr"s, std::move(cptr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (const auto *expr{arguments[1].value().UnwrapExpr()}) {
|
||||||
|
int fptrRank{expr->Rank()};
|
||||||
|
if (auto type{expr->GetType()}) {
|
||||||
|
if (type->HasDeferredTypeParameter()) {
|
||||||
|
context.messages().Say(
|
||||||
|
"FPTR= argument to C_F_POINTER() may not have a deferred type parameter"_err_en_US);
|
||||||
|
}
|
||||||
|
if (ExtractCoarrayRef(*expr)) {
|
||||||
|
context.messages().Say(
|
||||||
|
"FPTR= argument to C_F_POINTER() may not be a coindexed object"_err_en_US);
|
||||||
|
}
|
||||||
|
characteristics::DummyDataObject fptr{
|
||||||
|
characteristics::TypeAndShape{*type, fptrRank}};
|
||||||
|
fptr.intent = common::Intent::Out;
|
||||||
|
fptr.attrs.set(characteristics::DummyDataObject::Attr::Pointer);
|
||||||
|
dummies.emplace_back("fptr"s, std::move(fptr));
|
||||||
|
}
|
||||||
|
if (arguments[2] && fptrRank == 0) {
|
||||||
|
context.messages().Say(
|
||||||
|
"SHAPE= argument to C_F_POINTER() may not appear when FPTR= is scalar"_err_en_US);
|
||||||
|
} else if (!arguments[2] && fptrRank > 0) {
|
||||||
|
context.messages().Say(
|
||||||
|
"SHAPE= argument to C_F_POINTER() must appear when FPTR= is an array"_err_en_US);
|
||||||
|
}
|
||||||
|
characteristics::DummyDataObject shape{
|
||||||
|
characteristics::TypeAndShape{SubscriptInteger{}.GetType(), 1}};
|
||||||
|
shape.intent = common::Intent::In;
|
||||||
|
shape.attrs.set(characteristics::DummyDataObject::Attr::Optional);
|
||||||
|
dummies.emplace_back("shape"s, std::move(shape));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (dummies.size() == 3) {
|
||||||
|
return SpecificCall{
|
||||||
|
SpecificIntrinsic{"__builtin_c_f_pointer"s,
|
||||||
|
characteristics::Procedure{std::move(dummies), attrs}},
|
||||||
|
std::move(arguments)};
|
||||||
|
} else {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Applies any semantic checks peculiar to an intrinsic.
|
// Applies any semantic checks peculiar to an intrinsic.
|
||||||
static bool ApplySpecificChecks(
|
static bool ApplySpecificChecks(
|
||||||
SpecificCall &call, parser::ContextualMessages &messages) {
|
SpecificCall &call, parser::ContextualMessages &messages) {
|
||||||
|
@ -1677,6 +1802,19 @@ static DynamicType GetReturnType(const SpecificIntrinsicInterface &interface,
|
||||||
std::optional<SpecificCall> IntrinsicProcTable::Implementation::Probe(
|
std::optional<SpecificCall> IntrinsicProcTable::Implementation::Probe(
|
||||||
const CallCharacteristics &call, ActualArguments &arguments,
|
const CallCharacteristics &call, ActualArguments &arguments,
|
||||||
FoldingContext &context, const IntrinsicProcTable &intrinsics) const {
|
FoldingContext &context, const IntrinsicProcTable &intrinsics) const {
|
||||||
|
|
||||||
|
// All special cases handled here before the table probes below must
|
||||||
|
// also be recognized as special names in IsIntrinsic().
|
||||||
|
if (call.isSubroutineCall) {
|
||||||
|
if (call.name == "__builtin_c_f_pointer") {
|
||||||
|
return HandleC_F_Pointer(arguments, context);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (call.name == "null") {
|
||||||
|
return HandleNull(arguments, context, intrinsics);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (call.isSubroutineCall) {
|
if (call.isSubroutineCall) {
|
||||||
parser::Messages buffer;
|
parser::Messages buffer;
|
||||||
auto subrRange{subroutines_.equal_range(call.name)};
|
auto subrRange{subroutines_.equal_range(call.name)};
|
||||||
|
@ -1689,13 +1827,6 @@ std::optional<SpecificCall> IntrinsicProcTable::Implementation::Probe(
|
||||||
return std::nullopt; // TODO
|
return std::nullopt; // TODO
|
||||||
}
|
}
|
||||||
|
|
||||||
// Special case: NULL()
|
|
||||||
// All special cases handled here before the table probes below must
|
|
||||||
// also be caught as special names in IsIntrinsic().
|
|
||||||
if (call.name == "null") {
|
|
||||||
return HandleNull(arguments, context, intrinsics);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper to avoid emitting errors before it is sure there is no match
|
// Helper to avoid emitting errors before it is sure there is no match
|
||||||
parser::Messages localBuffer;
|
parser::Messages localBuffer;
|
||||||
parser::Messages *finalBuffer{context.messages().messages()};
|
parser::Messages *finalBuffer{context.messages().messages()};
|
||||||
|
|
|
@ -23,6 +23,12 @@ namespace Fortran::evaluate {
|
||||||
|
|
||||||
class FoldingContext;
|
class FoldingContext;
|
||||||
|
|
||||||
|
// Utility for checking for missing, excess, and duplicated arguments,
|
||||||
|
// and rearranging the actual arguments into dummy argument order.
|
||||||
|
bool CheckAndRearrangeArguments(ActualArguments &, parser::ContextualMessages &,
|
||||||
|
const char *const dummyKeywords[] /* null terminated */,
|
||||||
|
std::size_t trailingOptionals = 0);
|
||||||
|
|
||||||
struct CallCharacteristics {
|
struct CallCharacteristics {
|
||||||
std::string name;
|
std::string name;
|
||||||
bool isSubroutineCall{false};
|
bool isSubroutineCall{false};
|
||||||
|
|
|
@ -448,6 +448,17 @@ bool DynamicType::RequiresDescriptor() const {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool DynamicType::HasDeferredTypeParameter() const {
|
||||||
|
if (derived_) {
|
||||||
|
for (const auto &pair : derived_->parameters()) {
|
||||||
|
if (pair.second.isDeferred()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return charLength_ && charLength_->isDeferred();
|
||||||
|
}
|
||||||
|
|
||||||
bool SomeKind<TypeCategory::Derived>::operator==(
|
bool SomeKind<TypeCategory::Derived>::operator==(
|
||||||
const SomeKind<TypeCategory::Derived> &that) const {
|
const SomeKind<TypeCategory::Derived> &that) const {
|
||||||
return PointeeComparison(derivedTypeSpec_, that.derivedTypeSpec_);
|
return PointeeComparison(derivedTypeSpec_, that.derivedTypeSpec_);
|
||||||
|
|
|
@ -159,6 +159,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
bool RequiresDescriptor() const;
|
bool RequiresDescriptor() const;
|
||||||
|
bool HasDeferredTypeParameter() const;
|
||||||
|
|
||||||
// 7.3.2.3 & 15.5.2.4 type compatibility.
|
// 7.3.2.3 & 15.5.2.4 type compatibility.
|
||||||
// x.IsTypeCompatibleWith(y) is true if "x => y" or passing actual y to
|
// x.IsTypeCompatibleWith(y) is true if "x => y" or passing actual y to
|
||||||
|
|
|
@ -641,17 +641,15 @@ static void RearrangeArguments(const characteristics::Procedure &proc,
|
||||||
}
|
}
|
||||||
std::map<std::string, evaluate::ActualArgument> kwArgs;
|
std::map<std::string, evaluate::ActualArgument> kwArgs;
|
||||||
for (auto &x : actuals) {
|
for (auto &x : actuals) {
|
||||||
if (x) {
|
if (x && x->keyword()) {
|
||||||
if (x->keyword()) {
|
auto emplaced{
|
||||||
auto emplaced{
|
kwArgs.try_emplace(x->keyword()->ToString(), std::move(*x))};
|
||||||
kwArgs.try_emplace(x->keyword()->ToString(), std::move(*x))};
|
if (!emplaced.second) {
|
||||||
if (!emplaced.second) {
|
messages.Say(*x->keyword(),
|
||||||
messages.Say(*x->keyword(),
|
"Argument keyword '%s=' appears on more than one effective argument in this procedure reference"_err_en_US,
|
||||||
"Argument keyword '%s=' appears on more than one effective argument in this procedure reference"_err_en_US,
|
*x->keyword());
|
||||||
*x->keyword());
|
|
||||||
}
|
|
||||||
x.reset();
|
|
||||||
}
|
}
|
||||||
|
x.reset();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!kwArgs.empty()) {
|
if (!kwArgs.empty()) {
|
||||||
|
|
|
@ -415,8 +415,7 @@ SymbolVector CollectSymbols(const Scope &scope) {
|
||||||
sorted.reserve(scope.size() + scope.commonBlocks().size());
|
sorted.reserve(scope.size() + scope.commonBlocks().size());
|
||||||
for (const auto &pair : scope) {
|
for (const auto &pair : scope) {
|
||||||
const Symbol &symbol{*pair.second};
|
const Symbol &symbol{*pair.second};
|
||||||
if (!symbol.test(Symbol::Flag::ParentComp) &&
|
if (!symbol.test(Symbol::Flag::ParentComp)) {
|
||||||
!symbol.attrs().test(Attr::INTRINSIC)) {
|
|
||||||
if (symbols.insert(symbol).second) {
|
if (symbols.insert(symbol).second) {
|
||||||
if (symbol.has<NamelistDetails>()) {
|
if (symbol.has<NamelistDetails>()) {
|
||||||
namelist.push_back(symbol);
|
namelist.push_back(symbol);
|
||||||
|
@ -498,6 +497,7 @@ void PutObjectEntity(std::ostream &os, const Symbol &symbol) {
|
||||||
|
|
||||||
void PutProcEntity(std::ostream &os, const Symbol &symbol) {
|
void PutProcEntity(std::ostream &os, const Symbol &symbol) {
|
||||||
if (symbol.attrs().test(Attr::INTRINSIC)) {
|
if (symbol.attrs().test(Attr::INTRINSIC)) {
|
||||||
|
os << "intrinsic::" << symbol.name() << '\n';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const auto &details{symbol.get<ProcEntityDetails>()};
|
const auto &details{symbol.get<ProcEntityDetails>()};
|
||||||
|
|
|
@ -520,30 +520,29 @@ bool IsExtensibleType(const DerivedTypeSpec *derived) {
|
||||||
!derived->typeSymbol().get<DerivedTypeDetails>().sequence();
|
!derived->typeSymbol().get<DerivedTypeDetails>().sequence();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsDerivedTypeFromModule(
|
bool IsBuiltinDerivedType(const DerivedTypeSpec *derived, const char *name) {
|
||||||
const DerivedTypeSpec *derived, const char *module, const char *name) {
|
|
||||||
if (!derived) {
|
if (!derived) {
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
const auto &symbol{derived->typeSymbol()};
|
const auto &symbol{derived->typeSymbol()};
|
||||||
return symbol.name() == name && symbol.owner().IsModule() &&
|
return symbol.owner().IsModule() &&
|
||||||
symbol.owner().GetName().value() == module;
|
symbol.owner().GetName().value() == "__fortran_builtins" &&
|
||||||
|
symbol.name() == "__builtin_"s + name;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsIsoCType(const DerivedTypeSpec *derived) {
|
bool IsIsoCType(const DerivedTypeSpec *derived) {
|
||||||
return IsDerivedTypeFromModule(derived, "iso_c_binding", "c_ptr") ||
|
return IsBuiltinDerivedType(derived, "c_ptr") ||
|
||||||
IsDerivedTypeFromModule(derived, "iso_c_binding", "c_funptr");
|
IsBuiltinDerivedType(derived, "c_funptr");
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsTeamType(const DerivedTypeSpec *derived) {
|
bool IsTeamType(const DerivedTypeSpec *derived) {
|
||||||
return IsDerivedTypeFromModule(derived, "iso_fortran_env", "team_type");
|
return IsBuiltinDerivedType(derived, "team_type");
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsEventTypeOrLockType(const DerivedTypeSpec *derivedTypeSpec) {
|
bool IsEventTypeOrLockType(const DerivedTypeSpec *derivedTypeSpec) {
|
||||||
return IsDerivedTypeFromModule(
|
return IsBuiltinDerivedType(derivedTypeSpec, "event_type") ||
|
||||||
derivedTypeSpec, "iso_fortran_env", "event_type") ||
|
IsBuiltinDerivedType(derivedTypeSpec, "lock_type");
|
||||||
IsDerivedTypeFromModule(derivedTypeSpec, "iso_fortran_env", "lock_type");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsOrContainsEventOrLockComponent(const Symbol &symbol) {
|
bool IsOrContainsEventOrLockComponent(const Symbol &symbol) {
|
||||||
|
|
|
@ -92,9 +92,7 @@ bool IsProcedurePointer(const Symbol &);
|
||||||
bool IsFunctionResult(const Symbol &);
|
bool IsFunctionResult(const Symbol &);
|
||||||
bool IsFunctionResultWithSameNameAsFunction(const Symbol &);
|
bool IsFunctionResultWithSameNameAsFunction(const Symbol &);
|
||||||
bool IsExtensibleType(const DerivedTypeSpec *);
|
bool IsExtensibleType(const DerivedTypeSpec *);
|
||||||
// Is this a derived type from module with this name?
|
bool IsBuiltinDerivedType(const DerivedTypeSpec *derived, const char *name);
|
||||||
bool IsDerivedTypeFromModule(
|
|
||||||
const DerivedTypeSpec *derived, const char *module, const char *name);
|
|
||||||
// Is this derived type TEAM_TYPE from module ISO_FORTRAN_ENV
|
// Is this derived type TEAM_TYPE from module ISO_FORTRAN_ENV
|
||||||
bool IsTeamType(const DerivedTypeSpec *);
|
bool IsTeamType(const DerivedTypeSpec *);
|
||||||
// Is this derived type either C_PTR or C_FUNPTR from module ISO_C_BINDING
|
// Is this derived type either C_PTR or C_FUNPTR from module ISO_C_BINDING
|
||||||
|
|
|
@ -0,0 +1,38 @@
|
||||||
|
!===-- module/__fortran_builtins.f90 ---------------------------------------===!
|
||||||
|
!
|
||||||
|
! 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
|
||||||
|
!
|
||||||
|
!------------------------------------------------------------------------------!
|
||||||
|
|
||||||
|
! These naming shenanigans prevent names from Fortran intrinsic modules
|
||||||
|
! from being usable on INTRINSIC statements, and force the program
|
||||||
|
! to USE the standard intrinsic modules in order to access the
|
||||||
|
! standard names of the procedures.
|
||||||
|
module __Fortran_builtins
|
||||||
|
|
||||||
|
integer, parameter, private :: int64 = selected_int_kind(18)
|
||||||
|
|
||||||
|
intrinsic :: __builtin_c_f_pointer
|
||||||
|
|
||||||
|
type :: __builtin_c_ptr
|
||||||
|
integer(kind=int64) :: __address = 0
|
||||||
|
end type
|
||||||
|
|
||||||
|
type :: __builtin_c_funptr
|
||||||
|
integer(kind=int64) :: __address = 0
|
||||||
|
end type
|
||||||
|
|
||||||
|
type :: __builtin_event_type
|
||||||
|
integer(kind=int64) :: __count = 0
|
||||||
|
end type
|
||||||
|
|
||||||
|
type :: __builtin_lock_type
|
||||||
|
integer(kind=int64) :: __count = 0
|
||||||
|
end type
|
||||||
|
|
||||||
|
type :: __builtin_team_type
|
||||||
|
integer(kind=int64) :: __id = 0
|
||||||
|
end type
|
||||||
|
end module
|
|
@ -10,16 +10,13 @@
|
||||||
|
|
||||||
module iso_c_binding
|
module iso_c_binding
|
||||||
|
|
||||||
type :: c_ptr
|
use __Fortran_builtins, only: &
|
||||||
integer(kind=8) :: address
|
c_f_pointer => __builtin_c_f_pointer, &
|
||||||
end type c_ptr
|
c_ptr => __builtin_c_ptr, &
|
||||||
|
c_funptr => __builtin_c_funptr
|
||||||
|
|
||||||
type :: c_funptr
|
type(c_ptr), parameter :: c_null_ptr = c_ptr()
|
||||||
integer(kind=8) :: address
|
type(c_funptr), parameter :: c_null_funptr = c_funptr()
|
||||||
end type c_funptr
|
|
||||||
|
|
||||||
type(c_ptr), parameter :: c_null_ptr = c_ptr(0)
|
|
||||||
type(c_funptr), parameter :: c_null_funptr = c_funptr(0)
|
|
||||||
|
|
||||||
! Table 18.2 (in clause 18.3.1)
|
! Table 18.2 (in clause 18.3.1)
|
||||||
! TODO: Specialize (via macros?) for alternative targets
|
! TODO: Specialize (via macros?) for alternative targets
|
||||||
|
@ -83,23 +80,15 @@ module iso_c_binding
|
||||||
logical function c_associated(c_ptr_1, c_ptr_2)
|
logical function c_associated(c_ptr_1, c_ptr_2)
|
||||||
type(c_ptr), intent(in) :: c_ptr_1
|
type(c_ptr), intent(in) :: c_ptr_1
|
||||||
type(c_ptr), intent(in), optional :: c_ptr_2
|
type(c_ptr), intent(in), optional :: c_ptr_2
|
||||||
if (c_ptr_1%address == c_null_ptr%address) then
|
if (c_ptr_1%__address == c_null_ptr%__address) then
|
||||||
c_associated = .false.
|
c_associated = .false.
|
||||||
else if (present(c_ptr_2)) then
|
else if (present(c_ptr_2)) then
|
||||||
c_associated = c_ptr_1%address == c_ptr_2%address
|
c_associated = c_ptr_1%__address == c_ptr_2%__address
|
||||||
else
|
else
|
||||||
c_associated = .true.
|
c_associated = .true.
|
||||||
end if
|
end if
|
||||||
end function c_associated
|
end function c_associated
|
||||||
|
|
||||||
subroutine c_f_pointer(cptr, fptr, shape)
|
|
||||||
type(c_ptr), intent(in) :: cptr
|
|
||||||
type(*), pointer, dimension(..), intent(out) :: fptr
|
|
||||||
! TODO: Use a larger kind for shape than default integer
|
|
||||||
integer, intent(in), optional :: shape(:) ! size(shape) == rank(fptr)
|
|
||||||
! TODO: Define, or write in C and change this to an interface
|
|
||||||
end subroutine c_f_pointer
|
|
||||||
|
|
||||||
function c_loc(x)
|
function c_loc(x)
|
||||||
type(c_ptr) :: c_loc
|
type(c_ptr) :: c_loc
|
||||||
type(*), intent(in) :: x
|
type(*), intent(in) :: x
|
||||||
|
|
|
@ -13,6 +13,11 @@ include '../runtime/magic-numbers.h' ! for IOSTAT= error/end code values
|
||||||
|
|
||||||
module iso_fortran_env
|
module iso_fortran_env
|
||||||
|
|
||||||
|
use __Fortran_builtins, only: &
|
||||||
|
event_type => __builtin_event_type, &
|
||||||
|
lock_type => __builtin_lock_type, &
|
||||||
|
team_type => __builtin_team_type
|
||||||
|
|
||||||
implicit none
|
implicit none
|
||||||
|
|
||||||
integer, parameter :: atomic_int_kind = selected_int_kind(18)
|
integer, parameter :: atomic_int_kind = selected_int_kind(18)
|
||||||
|
@ -138,21 +143,6 @@ module iso_fortran_env
|
||||||
integer, parameter :: stat_unlocked = FORTRAN_RUNTIME_STAT_UNLOCKED
|
integer, parameter :: stat_unlocked = FORTRAN_RUNTIME_STAT_UNLOCKED
|
||||||
integer, parameter :: stat_unlocked_failed_image = FORTRAN_RUNTIME_STAT_UNLOCKED_FAILED_IMAGE
|
integer, parameter :: stat_unlocked_failed_image = FORTRAN_RUNTIME_STAT_UNLOCKED_FAILED_IMAGE
|
||||||
|
|
||||||
type :: event_type
|
|
||||||
private
|
|
||||||
integer(kind=atomic_int_kind) :: count = 0
|
|
||||||
end type event_type
|
|
||||||
|
|
||||||
type :: lock_type
|
|
||||||
private
|
|
||||||
integer(kind=atomic_int_kind) :: count = 0
|
|
||||||
end type lock_type
|
|
||||||
|
|
||||||
type :: team_type
|
|
||||||
private
|
|
||||||
integer(kind=int64) :: id = 0
|
|
||||||
end type team_type
|
|
||||||
|
|
||||||
contains
|
contains
|
||||||
|
|
||||||
character(len=80) function compiler_options()
|
character(len=80) function compiler_options()
|
||||||
|
|
|
@ -192,6 +192,7 @@ set(ERROR_TESTS
|
||||||
separate-module-procs.f90
|
separate-module-procs.f90
|
||||||
bindings01.f90
|
bindings01.f90
|
||||||
bad-forward-type.f90
|
bad-forward-type.f90
|
||||||
|
c_f_pointer.f90
|
||||||
)
|
)
|
||||||
|
|
||||||
# These test files have expected symbols in the source
|
# These test files have expected symbols in the source
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
! Enforce 18.2.3.3
|
||||||
|
|
||||||
|
program test
|
||||||
|
use iso_c_binding, only: c_ptr, c_f_pointer
|
||||||
|
type(c_ptr) :: scalarC, arrayC(1)
|
||||||
|
integer, pointer :: scalarIntF, arrayIntF(:), coindexed[*]
|
||||||
|
character(len=:), pointer :: charDeferredF
|
||||||
|
integer :: j
|
||||||
|
call c_f_pointer(scalarC, scalarIntF) ! ok
|
||||||
|
call c_f_pointer(scalarC, arrayIntF, [1_8]) ! ok
|
||||||
|
call c_f_pointer(shape=[1_8], cptr=scalarC, fptr=arrayIntF) ! ok
|
||||||
|
call c_f_pointer(scalarC, shape=[1_8], fptr=arrayIntF) ! ok
|
||||||
|
!ERROR: A positional actual argument may not appear after any keyword arguments
|
||||||
|
call c_f_pointer(scalarC, fptr=arrayIntF, [1_8])
|
||||||
|
!ERROR: CPTR= argument to C_F_POINTER() must be a C_PTR
|
||||||
|
call c_f_pointer(j, scalarIntF)
|
||||||
|
!ERROR: CPTR= argument to C_F_POINTER() must be scalar
|
||||||
|
call c_f_pointer(arrayC, scalarIntF)
|
||||||
|
!ERROR: SHAPE= argument to C_F_POINTER() must appear when FPTR= is an array
|
||||||
|
call c_f_pointer(scalarC, arrayIntF)
|
||||||
|
!ERROR: SHAPE= argument to C_F_POINTER() may not appear when FPTR= is scalar
|
||||||
|
call c_f_pointer(scalarC, scalarIntF, [1_8])
|
||||||
|
!ERROR: FPTR= argument to C_F_POINTER() may not have a deferred type parameter
|
||||||
|
call c_f_pointer(scalarC, charDeferredF)
|
||||||
|
!ERROR: FPTR= argument to C_F_POINTER() may not be a coindexed object
|
||||||
|
call c_f_pointer(scalarC, coindexed[0])
|
||||||
|
end program
|
|
@ -15,13 +15,11 @@
|
||||||
|
|
||||||
! CHECK: A DO loop should terminate with an END DO or CONTINUE
|
! CHECK: A DO loop should terminate with an END DO or CONTINUE
|
||||||
|
|
||||||
module iso_fortran_env
|
include '../../module/__fortran_builtins.f90'
|
||||||
type :: team_type
|
include '../../module/iso_c_binding.f90'
|
||||||
end type
|
|
||||||
end
|
|
||||||
|
|
||||||
subroutine foo8()
|
subroutine foo8()
|
||||||
use :: iso_fortran_env, only : team_type
|
use :: __fortran_builtins, only : team_type => __builtin_team_type
|
||||||
type(team_type) :: odd_even
|
type(team_type) :: odd_even
|
||||||
do 01 k=1,10
|
do 01 k=1,10
|
||||||
change team (odd_even)
|
change team (odd_even)
|
||||||
|
|
|
@ -34,4 +34,5 @@ end
|
||||||
! end type
|
! end type
|
||||||
! type(t),parameter::x=t(a=456_4,b=NULL())
|
! type(t),parameter::x=t(a=456_4,b=NULL())
|
||||||
! type(t),parameter::y=t(a=789_4,b=NULL())
|
! type(t),parameter::y=t(a=789_4,b=NULL())
|
||||||
|
! intrinsic::null
|
||||||
!end
|
!end
|
||||||
|
|
|
@ -28,6 +28,7 @@ end module m1
|
||||||
!Expect: m1.mod
|
!Expect: m1.mod
|
||||||
! module m1
|
! module m1
|
||||||
! integer(8),parameter::a0s(1_8:*)=[Integer(8)::]
|
! integer(8),parameter::a0s(1_8:*)=[Integer(8)::]
|
||||||
|
! intrinsic::shape
|
||||||
! real(4)::a1(1_8:5_8,1_8:5_8,1_8:5_8)
|
! real(4)::a1(1_8:5_8,1_8:5_8,1_8:5_8)
|
||||||
! integer(8),parameter::a1s(1_8:*)=[Integer(8)::5_8,5_8,5_8]
|
! integer(8),parameter::a1s(1_8:*)=[Integer(8)::5_8,5_8,5_8]
|
||||||
! integer(8),parameter::a1ss(1_8:*)=[Integer(8)::3_8]
|
! integer(8),parameter::a1ss(1_8:*)=[Integer(8)::3_8]
|
||||||
|
@ -42,6 +43,7 @@ end module m1
|
||||||
! integer(8),parameter::ac4s(1_8:*)=[Integer(8)::36_8]
|
! integer(8),parameter::ac4s(1_8:*)=[Integer(8)::36_8]
|
||||||
! integer(8),parameter::ac5s(1_8:*)=[Integer(8)::9_8]
|
! integer(8),parameter::ac5s(1_8:*)=[Integer(8)::9_8]
|
||||||
! integer(8),parameter::rss(1_8:*)=[Integer(8)::10_8,9_8]
|
! integer(8),parameter::rss(1_8:*)=[Integer(8)::10_8,9_8]
|
||||||
|
! intrinsic::reshape
|
||||||
! contains
|
! contains
|
||||||
! subroutine subr(x,n1,n2)
|
! subroutine subr(x,n1,n2)
|
||||||
! real(4),intent(in)::x(:,:)
|
! real(4),intent(in)::x(:,:)
|
||||||
|
|
|
@ -65,6 +65,7 @@ end module m1
|
||||||
!module m1
|
!module m1
|
||||||
!integer(4),parameter::iranges(1_8:*)=[Integer(4)::2_4,4_4,9_4,18_4,38_4]
|
!integer(4),parameter::iranges(1_8:*)=[Integer(4)::2_4,4_4,9_4,18_4,38_4]
|
||||||
!logical(4),parameter::ircheck=.true._4
|
!logical(4),parameter::ircheck=.true._4
|
||||||
|
!intrinsic::all
|
||||||
!integer(4),parameter::intpvals(1_8:*)=[Integer(4)::0_4,2_4,3_4,4_4,5_4,9_4,10_4,18_4,19_4,38_4,39_4]
|
!integer(4),parameter::intpvals(1_8:*)=[Integer(4)::0_4,2_4,3_4,4_4,5_4,9_4,10_4,18_4,19_4,38_4,39_4]
|
||||||
!integer(4),parameter::intpkinds(1_8:*)=[Integer(4)::1_4,1_4,2_4,2_4,4_4,4_4,8_4,8_4,16_4,16_4,-1_4]
|
!integer(4),parameter::intpkinds(1_8:*)=[Integer(4)::1_4,1_4,2_4,2_4,4_4,4_4,8_4,8_4,16_4,16_4,-1_4]
|
||||||
!logical(4),parameter::ipcheck=.true._4
|
!logical(4),parameter::ipcheck=.true._4
|
||||||
|
|
|
@ -31,6 +31,7 @@ end module m1
|
||||||
!type(t1),parameter::t1x1(1_8:*)=[t1::t1(ia1=[Integer(4)::1_4,2_4]),t1(ia1=[Integer(4)::3_4,4_4])]
|
!type(t1),parameter::t1x1(1_8:*)=[t1::t1(ia1=[Integer(4)::1_4,2_4]),t1(ia1=[Integer(4)::3_4,4_4])]
|
||||||
!logical(4),parameter::t1check1=.true._4
|
!logical(4),parameter::t1check1=.true._4
|
||||||
!logical(4),parameter::t1check2=.true._4
|
!logical(4),parameter::t1check2=.true._4
|
||||||
|
!intrinsic::all
|
||||||
!logical(4),parameter::t1check3=.true._4
|
!logical(4),parameter::t1check3=.true._4
|
||||||
!type::t2
|
!type::t2
|
||||||
!type(t1)::dta1(1_8:2_8)
|
!type(t1)::dta1(1_8:2_8)
|
||||||
|
|
|
@ -18,5 +18,6 @@ end module m
|
||||||
!character(:,1),parameter::c1=1_"Hi! \344\275\240\345\245\275!"
|
!character(:,1),parameter::c1=1_"Hi! \344\275\240\345\245\275!"
|
||||||
!character(:,4),parameter::c4a(1_8:*)=[CHARACTER(KIND=4,LEN=1)::4_"\344\270\200",4_"\344\272\214",4_"\344\270\211",4_"\345\233\233",4_"\344\272\224"]
|
!character(:,4),parameter::c4a(1_8:*)=[CHARACTER(KIND=4,LEN=1)::4_"\344\270\200",4_"\344\272\214",4_"\344\270\211",4_"\345\233\233",4_"\344\272\224"]
|
||||||
!integer(4),parameter::lc4=7_4
|
!integer(4),parameter::lc4=7_4
|
||||||
|
!intrinsic::len
|
||||||
!integer(4),parameter::lc1=11_4
|
!integer(4),parameter::lc1=11_4
|
||||||
!end
|
!end
|
||||||
|
|
|
@ -53,11 +53,13 @@ end
|
||||||
!Expect: m3a.mod
|
!Expect: m3a.mod
|
||||||
!module m3a
|
!module m3a
|
||||||
! integer(4),parameter::i4=4_4
|
! integer(4),parameter::i4=4_4
|
||||||
|
! intrinsic::selected_int_kind
|
||||||
!end
|
!end
|
||||||
|
|
||||||
!Expect: m3b.mod
|
!Expect: m3b.mod
|
||||||
!module m3b
|
!module m3b
|
||||||
! use m3a,only:i4
|
! use m3a,only:i4
|
||||||
|
! use m3a,only:selected_int_kind
|
||||||
! integer(4)::j
|
! integer(4)::j
|
||||||
!end
|
!end
|
||||||
|
|
||||||
|
@ -73,11 +75,13 @@ end
|
||||||
!Expect: m4a.mod
|
!Expect: m4a.mod
|
||||||
!module m4a
|
!module m4a
|
||||||
! character(1_4,1),parameter::a=1_"\001"
|
! character(1_4,1),parameter::a=1_"\001"
|
||||||
|
! intrinsic::achar
|
||||||
!end
|
!end
|
||||||
|
|
||||||
!Expect: m4b.mod
|
!Expect: m4b.mod
|
||||||
!module m4b
|
!module m4b
|
||||||
! use m4a,only:a
|
! use m4a,only:a
|
||||||
|
! use m4a,only:achar
|
||||||
! character(1_4,1),parameter::b=1_"\001"
|
! character(1_4,1),parameter::b=1_"\001"
|
||||||
!end
|
!end
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,7 @@ end
|
||||||
!integer(4),parameter::brown=3_4
|
!integer(4),parameter::brown=3_4
|
||||||
!integer(4),parameter::oak=0_4
|
!integer(4),parameter::oak=0_4
|
||||||
!integer(4),parameter::beech=-4_4
|
!integer(4),parameter::beech=-4_4
|
||||||
|
!intrinsic::rank
|
||||||
!integer(4),parameter::pine=-3_4
|
!integer(4),parameter::pine=-3_4
|
||||||
!integer(4),parameter::poplar=3_4
|
!integer(4),parameter::poplar=3_4
|
||||||
!end
|
!end
|
||||||
|
|
|
@ -35,6 +35,7 @@ set_target_properties(f18 f18-parse-demo
|
||||||
)
|
)
|
||||||
|
|
||||||
set(MODULES
|
set(MODULES
|
||||||
|
"__fortran_builtins" # must be first
|
||||||
"ieee_arithmetic"
|
"ieee_arithmetic"
|
||||||
"ieee_exceptions"
|
"ieee_exceptions"
|
||||||
"ieee_features"
|
"ieee_features"
|
||||||
|
@ -46,12 +47,12 @@ set(MODULES
|
||||||
# Create module files directly from the top-level module source directory
|
# Create module files directly from the top-level module source directory
|
||||||
foreach(filename ${MODULES})
|
foreach(filename ${MODULES})
|
||||||
add_custom_command(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/include/${filename}.mod
|
add_custom_command(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/include/${filename}.mod
|
||||||
COMMAND f18 -fparse-only -fdebug-semantics ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
COMMAND f18 -fparse-only -fdebug-semantics -I{${CMAKE_CURRENT_BINARY_DIR}/include} ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
||||||
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/include"
|
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/include"
|
||||||
DEPENDS f18 ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
DEPENDS f18 ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
||||||
)
|
)
|
||||||
add_custom_command(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/include/${filename}.f18.mod
|
add_custom_command(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/include/${filename}.f18.mod
|
||||||
COMMAND f18 -fparse-only -fdebug-semantics -module-suffix .f18.mod ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
COMMAND f18 -fparse-only -fdebug-semantics -module-suffix .f18.mod -I{${CMAKE_CURRENT_BINARY_DIR}/include} ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
||||||
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/include"
|
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/include"
|
||||||
DEPENDS f18 ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
DEPENDS f18 ${PROJECT_SOURCE_DIR}/module/${filename}.f90
|
||||||
)
|
)
|
||||||
|
|
Loading…
Reference in New Issue