Hlfir
hlfir
The High-Level Fortran IR (HLFIR) dialect that is used by Flang.
This is mixed with FIR, and provides a higher level view of Fortran variables and some expressions. This means that temporaries need not be materialised and there is a higher level of information about the programmer's Fortran code, compared to FIR, available for optimisation and lowering.
See external documentation.
HLFIR = Dialect('hlfir', [AllOp, AnyOp, ApplyOp, AsExprOp, AssignmentMaskOp, AssignOp, AssociateOp, CharExtremumOp, ConcatOp, CopyInOp, CopyOutOp, CountOp, DeclareOp, DesignateOp, DestroyOp, DotProductOp, ElementalAddrOp, ElementalOp, ElseWhereOp, EndAssociateOp, ForallIndexOp, ForallMaskOp, ForallOp, GetExtentOp, GetLengthOp, MatmulOp, MatmulTransposeOp, MaxvalOp, MinvalOp, NoReassocOp, NullOp, ParentComponentOp, ProductOp, RegionAssignOp, SetLengthOp, ShapeOfOp, SumOp, TransposeOp, YieldElementOp, YieldOp], [ExprType])
module-attribute
ExprType
dataclass
Bases: ParametrizedAttribute, TypeAttribute
The type of an array, character, or derived type Fortran expression.
Abstract value type for Fortran arrays, characters and derived types. The rank cannot be assumed, and empty shape means that the expression is a scalar. When the element type is a derived type, the polymorphic flag may be set to true to indicate that the expression dynamic type can differ from its static type.
Source code in xdsl/dialects/experimental/hlfir.py
54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 | |
name = 'hlfir.expr'
class-attribute
instance-attribute
shape: ArrayAttr[IntegerAttr | DeferredAttr | NoneType]
instance-attribute
elementType: Attribute
instance-attribute
print_parameters(printer: Printer) -> None
Source code in xdsl/dialects/experimental/hlfir.py
69 70 71 72 73 74 75 76 77 78 79 80 81 | |
parse_parameters(parser: AttrParser) -> list[Attribute]
classmethod
Source code in xdsl/dialects/experimental/hlfir.py
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 | |
DeclareOp
dataclass
Bases: IRDLOperation
Declare a variable and produce an SSA value that can be used as a variable in HLFIR operations.
Tie the properties of a Fortran variable to an address. The properties include bounds, length parameters, and Fortran attributes.
The arguments are the same as for fir.declare.
The main difference with fir.declare is that hlfir.declare returns two
values:
- the first one is an SSA value that allows retrieving the variable
address, bounds, and type parameters at any point without requiring
access to the defining operation. This may be:
- for scalar numerical, logical, or derived type without length
parameters: a fir.ref
Example: CHARACTER(n) :: c(10:n, 20:n)
Can be represented as:
func.func @foo(%arg0: !fir.ref<!fir.array>>, %arg1: !fir.ref
Source code in xdsl/dialects/experimental/hlfir.py
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 | |
name = 'hlfir.declare'
class-attribute
instance-attribute
memref = operand_def()
class-attribute
instance-attribute
shape = opt_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
dummy_scope = opt_operand_def()
class-attribute
instance-attribute
uniq_name = opt_prop_def(StringAttr)
class-attribute
instance-attribute
fortran_attrs = opt_prop_def(FortranVariableFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
result2 = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
DesignateOp
dataclass
Bases: IRDLOperation
Designate a Fortran variable.
This operation represents a Fortran "part-ref", except that it can embed a substring or or complex part directly, and that vector subscripts cannot be used. It returns a Fortran variable that is a part of the input variable.
The operands are as follow
- memref is the variable being designated.
- component may be provided if the memref is a derived type to represent a reference to a component. It must be the name of a component of memref derived type.
- component_shape represents the shape of the component and must be provided if and only if both component and indices appear.
- indices can be provided to index arrays. The indices may be simple indices or triplets. If indices are provided and there is a component, the component must be an array component and the indices index the array component. If memref is an array, and component is provided and is an array component, indices must be provided and must not be triplets. This ensures hlfir.designate does not create arrays of arrays (which is not possible in Fortran).
- substring may contain two values to represent a substring lower and upper bounds.
- complex_part may be provided to represent a complex part (true represents the imaginary part, and false the real part).
- shape represents the shape of the result and must be provided if the result is an array that is not a box address.
- typeparams represents the length parameters of the result and must be provided if the result type has length parameters and is not a box address.
Source code in xdsl/dialects/experimental/hlfir.py
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 | |
name = 'hlfir.designate'
class-attribute
instance-attribute
memref = operand_def()
class-attribute
instance-attribute
component_shape = opt_operand_def()
class-attribute
instance-attribute
indices = var_operand_def()
class-attribute
instance-attribute
substring = var_operand_def()
class-attribute
instance-attribute
shape = opt_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
component = opt_prop_def(StringAttr)
class-attribute
instance-attribute
complex_part = opt_prop_def(BoolAttr)
class-attribute
instance-attribute
is_triplet = prop_def(DenseArrayBase)
class-attribute
instance-attribute
fortran_attrs = opt_prop_def(FortranVariableFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
AssignOp
dataclass
Bases: IRDLOperation
Assign an expression or variable value to a Fortran variable.
Assign rhs to lhs following Fortran intrinsic assignments rules. The operation deals with inserting a temporary if the lhs and rhs may overlap.
The optional "realloc" flag allows indicating that this assignment has the Fortran 95 semantics for assignments to a whole allocatable. In such case, the left hand side must be an allocatable that may be unallocated or allocated with a different type and shape than the right hand side. It will be allocated or re-allocated as needed during the assignment.
When "realloc" is set and this is a character assignment, the optional flag "keep_lhs_length_if_realloc" indicates that the character left hand side should retain its length after the assignment. If the right hand side has a different length, truncation and padding will occur. This covers the case of explicit and assumed length character allocatables.
Otherwise, the left hand side will be allocated or reallocated to match the right hand side length if they differ. This covers the case of deferred length character allocatables.
The optional "temporary_lhs" flag indicates that the LHS is a compiler generated temporary. In this case the temporary is initialized if needed (e.g. the LHS is of derived type with allocatable/pointer components), and the assignment is done without LHS (or its subobjects) finalization and with automatic allocation.
If "temporary_lhs" and "keep_lhs_length_if_realloc" are both set, this assign operation denotes special case of character allocatable LHS with explicit length. The LHS that must preserve its length during the assignment regardless of the the RHS's length or/and allocation status. This assign operation will be lowered into a call to AssignExplicitLengthCharacter().
Source code in xdsl/dialects/experimental/hlfir.py
213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 | |
name = 'hlfir.assign'
class-attribute
instance-attribute
lhs = operand_def()
class-attribute
instance-attribute
rhs = operand_def()
class-attribute
instance-attribute
realloc = opt_prop_def(UnitAttr)
class-attribute
instance-attribute
keep_lhs_length_if_realloc = opt_prop_def(UnitAttr)
class-attribute
instance-attribute
temporary_lhs = opt_prop_def(UnitAttr)
class-attribute
instance-attribute
ParentComponentOp
dataclass
Bases: IRDLOperation
Designate the parent component of a variable.
This operation represents a Fortran component reference where the component name is a parent type of the variable's derived type. These component references cannot be represented with an hlfir.designate because the parent type names are not embedded in fir.type<> types as opposed to the actual component names.
The operands are as follow
- memref is a derived type variable whose parent component is being designated.
- shape is the shape of memref and the result and must be provided if memref is an array. Parent component reference lower bounds are ones, so the provided shape must be a fir.shape.
- typeparams are the type parameters of the parent component type if any. It is a subset of memref type parameters.
The parent component type and name is reflected in the result type.
Source code in xdsl/dialects/experimental/hlfir.py
262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 | |
name = 'hlfir.parent_comp'
class-attribute
instance-attribute
memref = operand_def()
class-attribute
instance-attribute
shape = opt_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
ConcatOp
dataclass
Bases: IRDLOperation
Concatenate characters.
Concatenate two or more character strings of a same character kind.
Source code in xdsl/dialects/experimental/hlfir.py
293 294 295 296 297 298 299 300 301 302 303 304 305 | |
name = 'hlfir.concat'
class-attribute
instance-attribute
strings = var_operand_def()
class-attribute
instance-attribute
length = operand_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
AllOp
dataclass
Bases: IRDLOperation
ALL transformational intrinsic.
Takes a logical array MASK as argument, optionally along a particular dimension, and returns true if all elements of MASK are true.
Source code in xdsl/dialects/experimental/hlfir.py
308 309 310 311 312 313 314 315 316 317 318 319 320 | |
name = 'hlfir.all'
class-attribute
instance-attribute
mask = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
AnyOp
dataclass
Bases: IRDLOperation
ANY transformational intrinsic.
Takes a logical array MASK as argument, optionally along a particular dimension, and returns true if any element of MASK is true.
Source code in xdsl/dialects/experimental/hlfir.py
323 324 325 326 327 328 329 330 331 332 333 334 335 | |
name = 'hlfir.any'
class-attribute
instance-attribute
mask = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
CountOp
dataclass
Bases: IRDLOperation
COUNT transformational intrinsic.
Takes a logical and counts the number of true values.
Source code in xdsl/dialects/experimental/hlfir.py
338 339 340 341 342 343 344 345 346 347 348 349 | |
name = 'hlfir.count'
class-attribute
instance-attribute
mask = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
MaxvalOp
dataclass
Bases: IRDLOperation
MAXVAL transformational intrinsic.
Maximum value(s) of an array. If DIM is absent, the result is a scalar. If DIM is present, the result is an array of rank n-1, where n is the rank of ARRAY.
Source code in xdsl/dialects/experimental/hlfir.py
352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 | |
name = 'hlfir.maxval'
class-attribute
instance-attribute
array = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
mask = opt_operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
MinvalOp
dataclass
Bases: IRDLOperation
MINVAL transformational intrinsic.
Minimum value(s) of an array. If DIM is absent, the result is a scalar. If DIM is present, the result is an array of rank n-1, where n is the rank of ARRAY.
Source code in xdsl/dialects/experimental/hlfir.py
372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 | |
name = 'hlfir.minval'
class-attribute
instance-attribute
array = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
mask = opt_operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
ProductOp
dataclass
Bases: IRDLOperation
PRODUCT transformational intrinsic.
Multiplies the elements of an array, optionally along a particular dimension, optionally if a mask is true.
Source code in xdsl/dialects/experimental/hlfir.py
392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 | |
name = 'hlfir.product'
class-attribute
instance-attribute
array = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
mask = opt_operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
SetLengthOp
dataclass
Bases: IRDLOperation
Change the length of a character entity.
This trims or pads the character argument according to the new length.
Source code in xdsl/dialects/experimental/hlfir.py
411 412 413 414 415 416 417 418 419 420 421 422 423 | |
name = 'hlfir.set_length'
class-attribute
instance-attribute
string = operand_def()
class-attribute
instance-attribute
length = operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
GetLengthOp
dataclass
Bases: IRDLOperation
Get the length of a character entity.
Get the length of character entity represented as hlfir.expr.
Source code in xdsl/dialects/experimental/hlfir.py
426 427 428 429 430 431 432 433 434 435 436 | |
name = 'hlfir.get_length'
class-attribute
instance-attribute
expr = operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
SumOp
dataclass
Bases: IRDLOperation
SUM transformational intrinsic.
Sums the elements of an array, optionally along a particular dimension, optionally if a mask is true.
Source code in xdsl/dialects/experimental/hlfir.py
439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 | |
name = 'hlfir.sum'
class-attribute
instance-attribute
array = operand_def()
class-attribute
instance-attribute
dim = opt_operand_def()
class-attribute
instance-attribute
mask = opt_operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
DotProductOp
dataclass
Bases: IRDLOperation
DOT_PRODUCT transformational intrinsic,
Dot product of two vectors,
Source code in xdsl/dialects/experimental/hlfir.py
458 459 460 461 462 463 464 465 466 467 468 469 470 | |
name = 'hlfir.dot_product'
class-attribute
instance-attribute
lhs = operand_def()
class-attribute
instance-attribute
rhs = operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
MatmulOp
dataclass
Bases: IRDLOperation
MATMUL transformational intrinsic.
Matrix multiplication.
Source code in xdsl/dialects/experimental/hlfir.py
473 474 475 476 477 478 479 480 481 482 483 484 485 | |
name = 'hlfir.matmul'
class-attribute
instance-attribute
lhs = operand_def()
class-attribute
instance-attribute
rhs = operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
TransposeOp
dataclass
Bases: IRDLOperation
TRANSPOSE transformational intrinsic.
Transpose a rank 2 array.
Source code in xdsl/dialects/experimental/hlfir.py
488 489 490 491 492 493 494 495 496 497 498 | |
name = 'hlfir.transpose'
class-attribute
instance-attribute
array = operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
MatmulTransposeOp
dataclass
Bases: IRDLOperation
Optimized MATMUL(TRANSPOSE(...), ...).
Matrix multiplication where the left hand side is transposed.
Source code in xdsl/dialects/experimental/hlfir.py
501 502 503 504 505 506 507 508 509 510 511 512 513 | |
name = 'hlfir.matmul_transpose'
class-attribute
instance-attribute
lhs = operand_def()
class-attribute
instance-attribute
rhs = operand_def()
class-attribute
instance-attribute
fastmath = opt_prop_def(FastMathFlagsAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
AssociateOp
dataclass
Bases: IRDLOperation
Create a variable from an expression value.
For expressions, this operation is an incentive to re-use the expression storage, if any, after the bufferization pass when possible (if the expression is not used afterwards).
Source code in xdsl/dialects/experimental/hlfir.py
516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 | |
name = 'hlfir.associate'
class-attribute
instance-attribute
source = operand_def()
class-attribute
instance-attribute
shape = opt_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
uniq_name = opt_prop_def(StringAttr)
class-attribute
instance-attribute
fortran_attrs = opt_prop_def(FortranVariableFlagsAttr)
class-attribute
instance-attribute
result = var_result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
EndAssociateOp
dataclass
Bases: IRDLOperation
Mark the end of life of a variable associated to an expression.
If the expression has a derived type that may contain allocatable components, the variable operand must be a Fortran entity.
Source code in xdsl/dialects/experimental/hlfir.py
537 538 539 540 541 542 543 544 545 546 547 548 | |
name = 'hlfir.end_associate'
class-attribute
instance-attribute
var = operand_def()
class-attribute
instance-attribute
must_free = operand_def()
class-attribute
instance-attribute
AsExprOp
dataclass
Bases: IRDLOperation
Take the value of an array, character or derived variable.
In general, this operation will lead to a copy of the variable in the bufferization pass if it was not transformed.
However, if it is known that the variable storage will not be used anymore afterwards, the variable storage ownership can be passed to the hlfir.expr by providing the $must_free argument that is a boolean that indicates if the storage must be freed (when it was allocated on the heap). This allows Fortran lowering to build some expression value in memory when there is no adequate hlfir operation, and to promote the result to an hlfir.expr value without paying the price of introducing a copy.
Source code in xdsl/dialects/experimental/hlfir.py
551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 | |
name = 'hlfir.as_expr'
class-attribute
instance-attribute
var = operand_def()
class-attribute
instance-attribute
must_free = opt_operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
NoReassocOp
dataclass
Bases: IRDLOperation
Synthetic op to prevent reassociation.
Same as fir.reassoc, except it accepts hlfir.expr arguments.
Source code in xdsl/dialects/experimental/hlfir.py
574 575 576 577 578 579 580 581 582 583 584 | |
name = 'hlfir.no_reassoc'
class-attribute
instance-attribute
val = operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
ElementalOp
dataclass
Bases: IRDLOperation
Elemental expression.
Represent an elemental expression as a function of the indices. This operation contain a region whose block arguments are one based indices iterating over the elemental expression shape. Given these indices, the element value for the given iteration can be computed in the region and yielded with the hlfir.yield_element operation.
The shape and typeparams operands represent the extents and type parameters of the resulting array value.
The optional mold is an entity carrying the information about
the dynamic type of the polymorphic result. Note that the shape
of the mold does not necessarily match the shape of the result,
for example, the result of merge(poly_scalar1, poly_scalar2, mask_array)
will have the shape of mask_array and the dynamic type of poly_scalar*.
The unordered attribute can be set to allow out of order processing of the indices. This is safe only if the operations in the body of the elemental do not have side effects.
Example: Y + X, with Integer :: X(10, 20), Y(10,20)
%0 = fir.shape %c10, %c20 : (index, index) -> !fir.shape<2>
%5 = hlfir.elemental %0 : (!fir.shape<2>) -> !hlfir.expr<10x20xi32> {
^bb0(%i: index, %j: index):
%6 = hlfir.designate %x (%i, %j) : (!fir.ref<!fir.array<10x20xi32>>, index, index) -> !fir.ref
Source code in xdsl/dialects/experimental/hlfir.py
587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 | |
name = 'hlfir.elemental'
class-attribute
instance-attribute
shape = operand_def()
class-attribute
instance-attribute
mold = opt_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
unordered = opt_prop_def(UnitAttr)
class-attribute
instance-attribute
regs = var_region_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
YieldElementOp
dataclass
Bases: IRDLOperation
Yield the elemental value in an ElementalOp.
Yield the element value of the current elemental expression iteration in an hlfir.elemental region. See hlfir.elemental description for an example.
Source code in xdsl/dialects/experimental/hlfir.py
638 639 640 641 642 643 644 645 646 647 648 649 650 651 | |
name = 'hlfir.yield_element'
class-attribute
instance-attribute
element_value = operand_def()
class-attribute
instance-attribute
traits = traits_def(IsTerminator())
class-attribute
instance-attribute
ApplyOp
dataclass
Bases: IRDLOperation
Get the element value of an expression.
Given an hlfir.expr array value, hlfir.apply allow retrieving the value for an element given one based indices.
When hlfir.apply is used on an hlfir.elemental, and if the hlfir.elemental operation evaluation can be moved to the location of the hlfir.apply, it is as if the hlfir.elemental body was evaluated given the hlfir.apply indices. Therefore, apply operations on hlfir.elemental expressions should be located such that evaluating the hlfir.elemental at the position of the hlfir.apply operation produces the same result as evaluating the hlfir.elemental at its location in the instruction stream. Attention should be paid to hlfir.elemental memory side effects (in practice these are unlikely). "10.1.4 Evaluation of operations" says that expression evaluation shall not impact/be impacted by other expression evaluation in the statement.
Source code in xdsl/dialects/experimental/hlfir.py
654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 | |
name = 'hlfir.apply'
class-attribute
instance-attribute
expr = operand_def()
class-attribute
instance-attribute
indices = var_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
NullOp
dataclass
Bases: IRDLOperation
Create a NULL() address.
So far is not intended to represent NULL(MOLD).
Source code in xdsl/dialects/experimental/hlfir.py
683 684 685 686 687 688 689 690 691 692 | |
name = 'hlfir.null'
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
DestroyOp
dataclass
Bases: IRDLOperation
Mark the last use of an hlfir.expr.
This will be the point at which the buffer of an hlfir.expr, if any, will be deallocated if it was heap allocated. If "finalize" attribute is set, the hlfir.expr value will be finalized before the deallocation. Note that this implies that the hlfir.expr is placed into a memory buffer, so that the library runtime can be called on it. The element type of the hlfir.expr must be derived type in this case. It is not required to create an hlfir.destroy operation for and hlfir.expr created inside an hlfir.elemental and returned in the hlfir.yield_element. The last use of such expression is implicit and an hlfir.destroy could not be emitted after the hlfir.yield_element since it is a terminator.
Note that hlfir.destroy are currently generated by Fortran lowering that has a good view of the expression use contexts, but this will need to be revisited if any motion of hlfir.expr is done (like CSE) since transformations should not introduce any hlfir.expr usages after an hlfir.destroy. The future will probably be to identify the last use points automatically in bufferization instead.
Source code in xdsl/dialects/experimental/hlfir.py
695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 | |
name = 'hlfir.destroy'
class-attribute
instance-attribute
expr = operand_def()
class-attribute
instance-attribute
finalize = opt_prop_def(UnitAttr)
class-attribute
instance-attribute
CopyInOp
dataclass
Bases: IRDLOperation
Copy a variable into a contiguous temporary if it is not contiguous.
Copy a variable into a contiguous temporary if the variable is not an absent optional and is not contiguous at runtime. When a copy is made this operation returns the temporary as first result, otherwise, it returns the potentially absent variable storage. The second result indicates if a copy was made.
This operation is meant to be used in combination with the hlfir.copy_out operation that deletes the temporary if it was created and copies the data back if needed. This operation allows passing non contiguous arrays to contiguous dummy arguments, which is possible in Fortran procedure references.
To deal with the optional case, an extra boolean value can be pass to the operation. In such cases, the copy-in will only be done if "var_is_present" is true and, when it is false, the original value will be returned instead.
Source code in xdsl/dialects/experimental/hlfir.py
727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 | |
name = 'hlfir.copy_in'
class-attribute
instance-attribute
var = operand_def()
class-attribute
instance-attribute
tempBox = operand_def()
class-attribute
instance-attribute
var_is_present = opt_operand_def()
class-attribute
instance-attribute
result = var_result_def()
class-attribute
instance-attribute
CopyOutOp
dataclass
Bases: IRDLOperation
Copy out a variable after a copy in.
If the variable was copied in a temporary in the related hlfir.copy_in, optionally copy back the temporary value to it (that may have been modified between the hlfir.copy_in and hlfir.copy_out). Then deallocate the temporary. The copy back is done if $var is provided and $was_copied is true. The deallocation of $temp is done if $was_copied is true.
Source code in xdsl/dialects/experimental/hlfir.py
756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 | |
name = 'hlfir.copy_out'
class-attribute
instance-attribute
temp = operand_def()
class-attribute
instance-attribute
was_copied = operand_def()
class-attribute
instance-attribute
var = opt_operand_def()
class-attribute
instance-attribute
ShapeOfOp
dataclass
Bases: IRDLOperation
Get the shape of a hlfir.expr.
Gets the runtime shape of a hlfir.expr. In lowering to FIR, the hlfir.shape_of operation will be replaced by an fir.shape. It is not valid to request the shape of a hlfir.expr which has no shape.
Source code in xdsl/dialects/experimental/hlfir.py
775 776 777 778 779 780 781 782 783 784 785 786 787 | |
name = 'hlfir.shape_of'
class-attribute
instance-attribute
expr = operand_def()
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
GetExtentOp
dataclass
Bases: IRDLOperation
Gets an extent value from a fir.shape.
The dimension argument uses C style indexing and so should be between 0 and 1 less than the rank of the shape
Source code in xdsl/dialects/experimental/hlfir.py
790 791 792 793 794 795 796 797 798 799 800 801 802 | |
name = 'hlfir.get_extent'
class-attribute
instance-attribute
shape = operand_def()
class-attribute
instance-attribute
dim = prop_def(IntAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
RegionAssignOp
dataclass
Bases: IRDLOperation
Represent a Fortran assignment using regions for the LHS and RHS evaluation.
This operation can represent Forall and Where assignment when inside an hlfir.forall or hlfir.where "ordered assignment tree". It can also represent user defined assignments and assignment to vector subscripted entities without requiring the materialization of the right-hand side temporary copy that may be needed to implement Fortran assignment semantic.
The right-hand side and left-hand side evaluations are held in their own regions terminated with hlfir.yield operations (or hlfir.elemental_addr for a left-hand side with vector subscript).
An optional region may be added to implement user defined assignment. This region provides two block arguments with the same type as the yielded rhs and lhs entities (in that order), or the element type if this is an elemental user defined assignment.
If this optional region is not provided, intrinsic assignment is performed.
Example: "X = Y", where "=" is a user defined elemental assignment "foo" taking Y by value.
hlfir.region_assign {
hlfir.yield %y : !fir.box<!fir.array<?x!f32>>
} to {
hlfir.yield %x : !fir.box<!fir.array<?x!fir.type
TODO: add optional "realloc" semantics like for hlfir.assign.
Source code in xdsl/dialects/experimental/hlfir.py
805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 | |
name = 'hlfir.region_assign'
class-attribute
instance-attribute
rhs_region = region_def()
class-attribute
instance-attribute
lhs_region = region_def()
class-attribute
instance-attribute
user_defined_assignment = region_def()
class-attribute
instance-attribute
YieldOp
dataclass
Bases: IRDLOperation
Yield a value or variable inside a forall, where or region assignment.
Terminator operation that yields an HLFIR value or variable that was computed in a region and hold the yielded entity cleanup, if any, into its own region. This allows representing any Fortran expression evaluation in its own region so that the evaluation can easily be scheduled/moved around in a pass.
Example: "foo(x)" where foo returns an allocatable array.
{
// In some region.
%0 = fir.call @foo(x) (!fir.ref
Source code in xdsl/dialects/experimental/hlfir.py
849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 | |
name = 'hlfir.yield'
class-attribute
instance-attribute
entity = operand_def()
class-attribute
instance-attribute
cleanup = region_def()
class-attribute
instance-attribute
traits = traits_def(IsTerminator())
class-attribute
instance-attribute
ElementalAddrOp
dataclass
Bases: IRDLOperation
Yield the address of a vector subscripted variable inside an hlfir.region_assign.
Special terminator node for the left-hand side region of an hlfir.region_assign to a vector subscripted entity.
It represents how the address of an element of such entity is computed given one based indices.
It is very similar to hlfir.elemental, except that it does not produce an SSA value because there is no hlfir type to describe a vector subscripted entity (the codegen of such type would be problematic). Hence, it is tightly linked to an hlfir.region_assign by its terminator property.
An optional cleanup region may be provided if any of the subscript expressions of the designator require a cleanup. This allows documenting cleanups that cannot be generated after the vector subscripted designator usage (that has not been materizaled yet). The cleanups will be evaluated after the assignment once the related hlfir.region_assign is lowered.
Example: "X(VECTOR) = Y"
hlfir.region_assign {
hlfir.yield %y : !fir.ref<!fir.array<20xf32>>
} to {
hlfir.elemental_addr %vector_shape : !fir.shape<1> {
^bb0(%i: index):
%0 = hlfir.designate %vector (%i) : (!fir.ref<!fir.array<20xi32>>, index) -> !fir.ref
Source code in xdsl/dialects/experimental/hlfir.py
877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 | |
name = 'hlfir.elemental_addr'
class-attribute
instance-attribute
shape = operand_def()
class-attribute
instance-attribute
mold = opt_operand_def()
class-attribute
instance-attribute
typeparams = var_operand_def()
class-attribute
instance-attribute
unordered = opt_prop_def(UnitAttr)
class-attribute
instance-attribute
body = region_def()
class-attribute
instance-attribute
cleanup = region_def()
class-attribute
instance-attribute
irdl_options = (AttrSizedOperandSegments(as_property=True),)
class-attribute
instance-attribute
ForallOp
dataclass
Bases: IRDLOperation
Represent a Fortran forall.
This operation allows representing Fortran forall. It computes a set of "index-name" values based on lower bound, upper bound, and step values whose evaluations are represented in their own regions.
Operations nested in its body region are evaluated in order. As opposed to a regular loop, each nested operation is fully evaluated for all the values in the "active set of index-name" before the next nested operation. In practice, the nested operation evaluation may be fused if it is proven that they do not have data dependency.
The "index-name" value is represented as the argument of the body region.
The lower, upper, and step region (if provided), must be terminated by hlfir.yield that yields scalar integers.
The body region must only contain other OrderedAssignmentTreeOpInterface operations (like hlfir.region_assign, or other hlfir.forall).
A Fortran forall with several indices is represented as a nest of hlfir.forall.
All the regions contained in the hlfir.forall must only contain code that is pure from a Fortran point of view, except for the assignment effect of the hlfir.region_assign. This matches Fortran constraint C1037, but requires the outer controls to be evaluated outside of the hlfir.forall (these controls may have side effects as per Fortran 2018 10.1.4 section).
Example: FORALL(I=1:10) X(I) = FOO(I)
hlfir.forall lb {
hlfir.yield %c1 : index
} ub {
hlfir.yield %c10 : index
} (%i : index) {
hlfir.region_assign {
%res = fir.call @foo(%i) : (index) -> f32
hlfir.yield %res : f32
} to {
%xi = hlfir.designate %x(%i) : (!fir.box<!fir.array<?xf32>>, index) -> !fir.ref
Source code in xdsl/dialects/experimental/hlfir.py
926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 | |
name = 'hlfir.forall'
class-attribute
instance-attribute
lb_region = region_def()
class-attribute
instance-attribute
ub_region = region_def()
class-attribute
instance-attribute
step_region = region_def()
class-attribute
instance-attribute
body = region_def()
class-attribute
instance-attribute
ForallMaskOp
dataclass
Bases: IRDLOperation
Fortran Forall can have a scalar mask expression that depends on the Forall index-name value. hlfir.forall_mask allows representing this mask. The expression evaluation is held in the mask region that must yield an i1 scalar value. An hlfir.forall_mask must be directly nested in the body region of an hlfir.forall. It is a separate operation so that it can use the index SSA value defined by the hlfir.forall body region.
Example: "FORALL(I=1:10, SOME_CONDITION(I)) X(I) = FOO(I)"
hlfir.forall lb {
hlfir.yield %c1 : index
} ub {
hlfir.yield %c10 : index
} (%i : index) {
hlfir.forall_mask {
%mask = fir.call @some_condition(%i) : (index) -> i1
hlfir.yield %mask : i1
} do {
hlfir.region_assign {
%res = fir.call @foo(%i) : (index) -> f32
hlfir.yield %res : f32
} to {
%xi = hlfir.designate %x(%i) : (!fir.box<!fir.array<?xf32>>, index) -> !fir.ref
Source code in xdsl/dialects/experimental/hlfir.py
985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 | |
name = 'hlfir.forall_mask'
class-attribute
instance-attribute
mask_region = region_def()
class-attribute
instance-attribute
body = region_def()
class-attribute
instance-attribute
AssignmentMaskOp
dataclass
Bases: IRDLOperation
Represent Fortran "where" construct or statement. The mask expression evaluation is held in the mask region that must yield logical array that has the same shape as all the nested hlfir.region_assign left-hand sides, and all the nested hlfir.where or hlfir.elsewhere masks.
The values of the where and elsewhere masks form a control mask that controls all the nested hlfir.region_assign: only the array element for which the related control mask value is true are assigned. Any right-hand side elemental expression is only evaluated for elements where the control mask is true. See Fortran standard 2018 section 10.2.3 for more detailed about the control mask semantic.
An hlfir.where must not contain any hlfir.forall but it may be contained in such operation. This matches Fortran rules.
Source code in xdsl/dialects/experimental/hlfir.py
1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 | |
name = 'hlfir.where'
class-attribute
instance-attribute
mask_region = region_def()
class-attribute
instance-attribute
body = region_def()
class-attribute
instance-attribute
ElseWhereOp
dataclass
Bases: IRDLOperation
Represent Fortran "elsewhere" construct or statement.
It has an optional mask region to hold the evaluation of Fortran optional elsewhere mask expressions. If this region is provided, it must satisfy the same constraints as hlfir.where mask region.
An hlfir.elsewhere must be the last operation of an hlfir.where or, hlfir.elsewhere body, which is enforced by its terminator property.
Like in Fortran, an hlfir.elsewhere negate the current control mask, and if provided, adds the mask the resulting control mask (with a logical AND).
Source code in xdsl/dialects/experimental/hlfir.py
1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 | |
name = 'hlfir.elsewhere'
class-attribute
instance-attribute
mask_region = region_def()
class-attribute
instance-attribute
body = region_def()
class-attribute
instance-attribute
ForallIndexOp
dataclass
Bases: IRDLOperation
Represent a Fortran forall index declaration.
This operation allows placing an hlfir.forall index in memory with the related Fortran index-value name and type.
So far, lowering needs to manipulate symbols as memory entities. This operation allows fulfilling this requirements without allowing bare alloca/declare/store inside the body of hlfir.forall, which would make their analysis more complex.
Given Forall index-value cannot be modified it also allows defining a canonicalization of all its loads into a fir.convert of the hlfir.forall index, which helps simplifying the data dependency analysis of hlfir.forall.
Source code in xdsl/dialects/experimental/hlfir.py
1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 | |
name = 'hlfir.forall_index'
class-attribute
instance-attribute
index = operand_def()
class-attribute
instance-attribute
indexname = prop_def(StringAttr)
class-attribute
instance-attribute
result = result_def()
class-attribute
instance-attribute
CharExtremumOp
dataclass
Bases: IRDLOperation
Find max/min from given character strings.
Find the lexicographical minimum or maximum of two or more character strings of the same character kind and return the string with the lexicographical minimum or maximum number of characters. Example:
%0 = hlfir.char_extremum min, %arg0, %arg1 : (!fir.ref<!fir.char<1,10>>, !fir.ref<!fir.char<1,20>>) -> !hlfir.expr<!fir.char<1,10>>
Source code in xdsl/dialects/experimental/hlfir.py
1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 | |