diff --git a/flang/lib/Optimizer/Dialect/FIROps.cpp b/flang/lib/Optimizer/Dialect/FIROps.cpp --- a/flang/lib/Optimizer/Dialect/FIROps.cpp +++ b/flang/lib/Optimizer/Dialect/FIROps.cpp @@ -2326,6 +2326,8 @@ inputEleTy.isa() || outEleTy.isa() || (inputEleTy.isa() && outEleTy.isa()) || (getSlice() && inputEleTy.isa()) || + (getSlice() && fir::isa_complex(inputEleTy) && + outEleTy.isa()) || areCompatibleCharacterTypes(inputEleTy, outEleTy); if (!typeCanMismatch) return emitOpError( diff --git a/flang/lib/Optimizer/HLFIR/Transforms/ConvertToFIR.cpp b/flang/lib/Optimizer/HLFIR/Transforms/ConvertToFIR.cpp --- a/flang/lib/Optimizer/HLFIR/Transforms/ConvertToFIR.cpp +++ b/flang/lib/Optimizer/HLFIR/Transforms/ConvertToFIR.cpp @@ -364,9 +364,6 @@ auto module = designate->getParentOfType(); fir::FirOpBuilder builder(rewriter, fir::getKindMapping(module)); - if (designate.getComplexPart()) - TODO(loc, "hlfir::designate with complex part"); - hlfir::Entity baseEntity(designate.getMemref()); if (baseEntity.isMutableBox()) @@ -377,6 +374,7 @@ auto [base, shape] = hlfir::genVariableFirBaseShapeAndParams( loc, builder, baseEntity, firBaseTypeParameters); mlir::Type baseEleTy = hlfir::getFortranElementType(base.getType()); + mlir::Type resultEleTy = hlfir::getFortranElementType(designateResultType); mlir::Value fieldIndex; if (designate.getComponent()) { @@ -428,12 +426,7 @@ if (fieldIndex && baseEntity.isArray()) { // array%scalar_comp or array%array_comp(indices) // Generate triples for array(:, :, ...). - auto one = builder.createIntegerConstant(loc, idxTy, 1); - for (auto [lb, ub] : hlfir::genBounds(loc, builder, baseEntity)) { - triples.push_back(builder.createConvert(loc, idxTy, lb)); - triples.push_back(builder.createConvert(loc, idxTy, ub)); - triples.push_back(one); - } + triples = genFullSliceTriples(builder, loc, baseEntity); sliceFields.push_back(fieldIndex); // Add indices in the field path for "array%array_comp(indices)" // case. @@ -464,7 +457,12 @@ builder.create(loc, substring[0], one); substring.push_back(designate.getTypeparams()[0]); } - + if (designate.getComplexPart()) { + if (triples.empty()) + triples = genFullSliceTriples(builder, loc, baseEntity); + sliceFields.push_back(builder.createIntegerConstant( + loc, idxTy, *designate.getComplexPart())); + } mlir::Value slice; if (!triples.empty()) slice = @@ -517,6 +515,16 @@ base = fir::factory::CharacterExprHelper{builder, loc}.genSubstringBase( base, designate.getSubstring()[0], resultAddressType); + // Scalar complex part ref + if (designate.getComplexPart()) { + // Sequence types should have already been handled by this point + assert(!designateResultType.isa()); + auto index = builder.createIntegerConstant(loc, builder.getIndexType(), + *designate.getComplexPart()); + auto coorTy = fir::ReferenceType::get(resultEleTy); + base = builder.create(loc, coorTy, base, index); + } + // Cast/embox the computed scalar address if needed. if (designateResultType.isa()) { assert(designate.getTypeparams().size() == 1 && @@ -530,6 +538,24 @@ } return mlir::success(); } + +private: + // Generates triple for full slice + // Used for component and complex part slices when a triple is + // not specified + static llvm::SmallVector + genFullSliceTriples(fir::FirOpBuilder &builder, mlir::Location loc, + hlfir::Entity baseEntity) { + llvm::SmallVector triples; + mlir::Type idxTy = builder.getIndexType(); + auto one = builder.createIntegerConstant(loc, idxTy, 1); + for (auto [lb, ub] : hlfir::genBounds(loc, builder, baseEntity)) { + triples.push_back(builder.createConvert(loc, idxTy, lb)); + triples.push_back(builder.createConvert(loc, idxTy, ub)); + triples.push_back(one); + } + return triples; + } }; class ParentComponentOpConversion diff --git a/flang/test/HLFIR/designate-codegen-complex-part.fir b/flang/test/HLFIR/designate-codegen-complex-part.fir new file mode 100644 --- /dev/null +++ b/flang/test/HLFIR/designate-codegen-complex-part.fir @@ -0,0 +1,83 @@ +// Test code generation to FIR of hlfir.designate operations +// with complex parts. +// RUN: fir-opt %s -convert-hlfir-to-fir | FileCheck %s + +func.func @test_set_scalar(%arg0: !fir.ref>, %arg1: !fir.ref) { + %0:2 = hlfir.declare %arg0 {uniq_name = "a"} : (!fir.ref>) -> (!fir.ref>, !fir.ref>) + %1:2 = hlfir.declare %arg1 {uniq_name = "b"} : (!fir.ref) -> (!fir.ref, !fir.ref) + %2 = fir.load %1#0 : !fir.ref + %3 = hlfir.designate %0#0 imag : (!fir.ref>) -> !fir.ref + hlfir.assign %2 to %3 : f32, !fir.ref + return +} +// CHECK-LABEL: func.func @test_set_scalar( +// CHECK-SAME: %[[VAL_0:.*]]: !fir.ref>, %[[VAL_1:.*]]: !fir.ref) { +// CHECK: %[[VAL_2:.*]] = fir.declare %[[VAL_0]] {uniq_name = "a"} : (!fir.ref>) -> !fir.ref> +// CHECK: %[[VAL_3:.*]] = fir.declare %[[VAL_1]] {uniq_name = "b"} : (!fir.ref) -> !fir.ref +// CHECK: %[[VAL_4:.*]] = fir.load %[[VAL_3]] : !fir.ref +// CHECK: %[[VAL_5:.*]] = arith.constant 1 : index +// CHECK: %[[VAL_6:.*]] = fir.coordinate_of %[[VAL_2]], %[[VAL_5]] : (!fir.ref>, index) -> !fir.ref +// CHECK: fir.store %[[VAL_4]] to %[[VAL_6]] : !fir.ref + +func.func @test_scalar_at_index(%arg0: !fir.box>>, %arg1: !fir.ref) { + %0:2 = hlfir.declare %arg0 {uniq_name = "a"} : (!fir.box>>) -> (!fir.box>>, !fir.box>>) + %1:2 = hlfir.declare %arg1 {uniq_name = "b"} : (!fir.ref) -> (!fir.ref, !fir.ref) + %2 = fir.load %1#0 : !fir.ref + %3 = fir.convert %2 : (i32) -> i64 + %4 = hlfir.designate %0#0 (%3) real : (!fir.box>>, i64) -> !fir.ref + return +} +// CHECK-LABEL: func.func @test_scalar_at_index( +// CHECK-SAME: %[[VAL_0:.*]]: !fir.box>>, %[[VAL_1:.*]]: !fir.ref) { +// CHECK: %[[VAL_2:.*]] = fir.declare %[[VAL_0]] {uniq_name = "a"} : (!fir.box>>) -> !fir.box>> +// CHECK: %[[VAL_3:.*]] = fir.rebox %[[VAL_2]] : (!fir.box>>) -> !fir.box>> +// CHECK: %[[VAL_4:.*]] = fir.declare %[[VAL_1]] {uniq_name = "b"} : (!fir.ref) -> !fir.ref +// CHECK: %[[VAL_5:.*]] = fir.load %[[VAL_4]] : !fir.ref +// CHECK: %[[VAL_6:.*]] = fir.convert %[[VAL_5]] : (i32) -> i64 +// CHECK: %[[VAL_7:.*]] = fir.array_coor %[[VAL_2]] %[[VAL_6]] : (!fir.box>>, i64) -> !fir.ref> +// CHECK: %[[VAL_8:.*]] = arith.constant 0 : index +// CHECK: %[[VAL_9:.*]] = fir.coordinate_of %[[VAL_7]], %[[VAL_8]] : (!fir.ref>, index) -> !fir.ref + +func.func @test_complete_slice(%arg0: !fir.box>>) { + %c0 = arith.constant 0 : index + %0:2 = hlfir.declare %arg0 {uniq_name = "a"} : (!fir.box>>) -> (!fir.box>>, !fir.box>>) + %1:3 = fir.box_dims %0#0, %c0 : (!fir.box>>, index) -> (index, index, index) + %2 = fir.shape %1#1 : (index) -> !fir.shape<1> + %3 = hlfir.designate %0#0 imag shape %2 : (!fir.box>>, !fir.shape<1>) -> !fir.box> + return +} +// CHECK-LABEL: func.func @test_complete_slice( +// CHECK-SAME: %[[VAL_0:.*]]: !fir.box>>) { +// CHECK: %[[VAL_1:.*]] = arith.constant 0 : index +// CHECK: %[[VAL_2:.*]] = fir.declare %[[VAL_0]] {uniq_name = "a"} : (!fir.box>>) -> !fir.box>> +// CHECK: %[[VAL_3:.*]] = fir.rebox %[[VAL_2]] : (!fir.box>>) -> !fir.box>> +// CHECK: %[[VAL_4:.*]]:3 = fir.box_dims %[[VAL_3]], %[[VAL_1]] : (!fir.box>>, index) -> (index, index, index) +// CHECK: %[[VAL_5:.*]] = fir.shape %[[VAL_4]]#1 : (index) -> !fir.shape<1> +// CHECK: %[[VAL_6:.*]] = arith.constant 1 : index +// CHECK: %[[VAL_7:.*]] = arith.constant 1 : index +// CHECK: %[[VAL_8:.*]] = arith.constant 0 : index +// CHECK: %[[VAL_9:.*]]:3 = fir.box_dims %[[VAL_2]], %[[VAL_8]] : (!fir.box>>, index) -> (index, index, index) +// CHECK: %[[VAL_10:.*]] = arith.constant 1 : index +// CHECK: %[[VAL_11:.*]] = fir.slice %[[VAL_7]], %[[VAL_9]]#1, %[[VAL_6]] path %[[VAL_10]] : (index, index, index, index) -> !fir.slice<1> +// CHECK: %[[VAL_12:.*]] = fir.rebox %[[VAL_2]] [%[[VAL_11]]] : (!fir.box>>, !fir.slice<1>) -> !fir.box> + +func.func @test_slice_steps(%arg0: !fir.box>>) { + %c3 = arith.constant 3 : index + %c12 = arith.constant 12 : index + %c4 = arith.constant 4 : index + %0:2 = hlfir.declare %arg0 {uniq_name = "a"} : (!fir.box>>) -> (!fir.box>>, !fir.box>>) + %1 = fir.shape %c3 : (index) -> !fir.shape<1> + %2 = hlfir.designate %0#0 (%c4:%c12:%c3) real shape %1 : (!fir.box>>, index, index, index, !fir.shape<1>) -> !fir.box> + return +} +// CHECK-LABEL: func.func @test_slice_steps( +// CHECK-SAME: %[[VAL_0:.*]]: !fir.box>>) { +// CHECK: %[[VAL_1:.*]] = arith.constant 3 : index +// CHECK: %[[VAL_2:.*]] = arith.constant 12 : index +// CHECK: %[[VAL_3:.*]] = arith.constant 4 : index +// CHECK: %[[VAL_4:.*]] = fir.declare %[[VAL_0]] {uniq_name = "a"} : (!fir.box>>) -> !fir.box>> +// CHECK: %[[VAL_5:.*]] = fir.rebox %[[VAL_4]] : (!fir.box>>) -> !fir.box>> +// CHECK: %[[VAL_6:.*]] = fir.shape %[[VAL_1]] : (index) -> !fir.shape<1> +// CHECK: %[[VAL_7:.*]] = arith.constant 0 : index +// CHECK: %[[VAL_8:.*]] = fir.slice %[[VAL_3]], %[[VAL_2]], %[[VAL_1]] path %[[VAL_7]] : (index, index, index, index) -> !fir.slice<1> +// CHECK: %[[VAL_9:.*]] = fir.rebox %[[VAL_4]] [%[[VAL_8]]] : (!fir.box>>, !fir.slice<1>) -> !fir.box>