@c Copyright (C) 2014-2024 Free Software Foundation, Inc. @c Free Software Foundation, Inc. @c This is part of the GCC manual. @c For copying conditions, see the file gcc.texi. @node Match and Simplify @chapter Match and Simplify @cindex Match and Simplify The GIMPLE and GENERIC pattern matching project match-and-simplify tries to address several issues. @enumerate @item unify expression simplifications currently spread and duplicated over separate files like fold-const.cc, gimple-fold.cc and builtins.cc @item allow for a cheap way to implement building and simplifying non-trivial GIMPLE expressions, avoiding the need to go through building and simplifying GENERIC via fold_buildN and then gimplifying via force_gimple_operand @end enumerate To address these the project introduces a simple domain-specific language to write expression simplifications from which code targeting GIMPLE and GENERIC is auto-generated. The GENERIC variant follows the fold_buildN API while for the GIMPLE variant and to address 2) new APIs are introduced. @menu * GIMPLE API:: * The Language:: @end menu @node GIMPLE API @section GIMPLE API @cindex GIMPLE API @deftypefn {GIMPLE function} tree gimple_simplify (enum tree_code, tree, tree, gimple_seq *, tree (*)(tree)) @deftypefnx {GIMPLE function} tree gimple_simplify (enum tree_code, tree, tree, tree, gimple_seq *, tree (*)(tree)) @deftypefnx {GIMPLE function} tree gimple_simplify (enum tree_code, tree, tree, tree, tree, gimple_seq *, tree (*)(tree)) @deftypefnx {GIMPLE function} tree gimple_simplify (enum built_in_function, tree, tree, gimple_seq *, tree (*)(tree)) @deftypefnx {GIMPLE function} tree gimple_simplify (enum built_in_function, tree, tree, tree, gimple_seq *, tree (*)(tree)) @deftypefnx {GIMPLE function} tree gimple_simplify (enum built_in_function, tree, tree, tree, tree, gimple_seq *, tree (*)(tree)) The main GIMPLE API entry to the expression simplifications mimicking that of the GENERIC fold_@{unary,binary,ternary@} functions. @end deftypefn thus providing n-ary overloads for operation or function. The additional arguments are a gimple_seq where built statements are inserted on (if @code{NULL} then simplifications requiring new statements are not performed) and a valueization hook that can be used to tie simplifications to a SSA lattice. In addition to those APIs @code{fold_stmt} is overloaded with a valueization hook: @deftypefn bool fold_stmt (gimple_stmt_iterator *, tree (*)(tree)); @end deftypefn On top of these a @code{fold_buildN}-like API for GIMPLE is introduced: @deftypefn {GIMPLE function} tree gimple_build (gimple_seq *, location_t, enum tree_code, tree, tree, tree (*valueize) (tree) = NULL); @deftypefnx {GIMPLE function} tree gimple_build (gimple_seq *, location_t, enum tree_code, tree, tree, tree, tree (*valueize) (tree) = NULL); @deftypefnx {GIMPLE function} tree gimple_build (gimple_seq *, location_t, enum tree_code, tree, tree, tree, tree, tree (*valueize) (tree) = NULL); @deftypefnx {GIMPLE function} tree gimple_build (gimple_seq *, location_t, enum built_in_function, tree, tree, tree (*valueize) (tree) = NULL); @deftypefnx {GIMPLE function} tree gimple_build (gimple_seq *, location_t, enum built_in_function, tree, tree, tree, tree (*valueize) (tree) = NULL); @deftypefnx {GIMPLE function} tree gimple_build (gimple_seq *, location_t, enum built_in_function, tree, tree, tree, tree, tree (*valueize) (tree) = NULL); @deftypefnx {GIMPLE function} tree gimple_convert (gimple_seq *, location_t, tree, tree); @end deftypefn which is supposed to replace @code{force_gimple_operand (fold_buildN (...), ...)} and calls to @code{fold_convert}. Overloads without the @code{location_t} argument exist. Built statements are inserted on the provided sequence and simplification is performed using the optional valueization hook. @node The Language @section The Language @cindex The Language The language in which to write expression simplifications resembles other domain-specific languages GCC uses. Thus it is lispy. Let's start with an example from the match.pd file: @smallexample (simplify (bit_and @@0 integer_all_onesp) @@0) @end smallexample This example contains all required parts of an expression simplification. A simplification is wrapped inside a @code{(simplify ...)} expression. That contains at least two operands - an expression that is matched with the GIMPLE or GENERIC IL and a replacement expression that is returned if the match was successful. Expressions have an operator ID, @code{bit_and} in this case. Expressions can be lower-case tree codes with @code{_expr} stripped off or builtin function code names in all-caps, like @code{BUILT_IN_SQRT}. @code{@@n} denotes a so-called capture. It captures the operand and lets you refer to it in other places of the match-and-simplify. In the above example it is referred to in the replacement expression. Captures are @code{@@} followed by a number or an identifier. @smallexample (simplify (bit_xor @@0 @@0) @{ build_zero_cst (type); @}) @end smallexample In this example @code{@@0} is mentioned twice which constrains the matched expression to have two equal operands. Usually matches are constrained to equal types. If operands may be constants and conversions are involved, matching by value might be preferred in which case use @code{@@@@0} to denote a by-value match and the specific operand you want to refer to in the result part. This example also introduces operands written in C code. These can be used in the expression replacements and are supposed to evaluate to a tree node which has to be a valid GIMPLE operand (so you cannot generate expressions in C code). @smallexample (simplify (trunc_mod integer_zerop@@0 @@1) (if (!integer_zerop (@@1)) @@0)) @end smallexample Here @code{@@0} captures the first operand of the trunc_mod expression which is also predicated with @code{integer_zerop}. Expression operands may be either expressions, predicates or captures. Captures can be unconstrained or capture expressions or predicates. This example introduces an optional operand of simplify, the if-expression. This condition is evaluated after the expression matched in the IL and is required to evaluate to true to enable the replacement expression in the second operand position. The expression operand of the @code{if} is a standard C expression which may contain references to captures. The @code{if} has an optional third operand which may contain the replacement expression that is enabled when the condition evaluates to false. A @code{if} expression can be used to specify a common condition for multiple simplify patterns, avoiding the need to repeat that multiple times: @smallexample (if (!TYPE_SATURATING (type) && !FLOAT_TYPE_P (type) && !FIXED_POINT_TYPE_P (type)) (simplify (minus (plus @@0 @@1) @@0) @@1) (simplify (minus (minus @@0 @@1) @@0) (negate @@1))) @end smallexample Note that @code{if}s in outer position do not have the optional else clause but instead have multiple then clauses. Ifs can be nested. There exists a @code{switch} expression which can be used to chain conditions avoiding nesting @code{if}s too much: @smallexample (simplify (simple_comparison @@0 REAL_CST@@1) (switch /* a CMP (-0) -> a CMP 0 */ (if (REAL_VALUE_MINUS_ZERO (TREE_REAL_CST (@@1))) (cmp @@0 @{ build_real (TREE_TYPE (@@1), dconst0); @})) /* x != NaN is always true, other ops are always false. */ (if (REAL_VALUE_ISNAN (TREE_REAL_CST (@@1)) && ! HONOR_SNANS (@@1)) @{ constant_boolean_node (cmp == NE_EXPR, type); @}))) @end smallexample Is equal to @smallexample (simplify (simple_comparison @@0 REAL_CST@@1) (switch /* a CMP (-0) -> a CMP 0 */ (if (REAL_VALUE_MINUS_ZERO (TREE_REAL_CST (@@1))) (cmp @@0 @{ build_real (TREE_TYPE (@@1), dconst0); @}) /* x != NaN is always true, other ops are always false. */ (if (REAL_VALUE_ISNAN (TREE_REAL_CST (@@1)) && ! HONOR_SNANS (@@1)) @{ constant_boolean_node (cmp == NE_EXPR, type); @})))) @end smallexample which has the second @code{if} in the else operand of the first. The @code{switch} expression takes @code{if} expressions as operands (which may not have else clauses) and as a last operand a replacement expression which should be enabled by default if no other condition evaluated to true. Captures can also be used for capturing results of sub-expressions. @smallexample #if GIMPLE (simplify (pointer_plus (addr@@2 @@0) INTEGER_CST_P@@1) (if (is_gimple_min_invariant (@@2))) @{ poly_int64 off; tree base = get_addr_base_and_unit_offset (@@0, &off); off += tree_to_uhwi (@@1); /* Now with that we should be able to simply write (addr (mem_ref (addr @@base) (plus @@off @@1))) */ build1 (ADDR_EXPR, type, build2 (MEM_REF, TREE_TYPE (TREE_TYPE (@@2)), build_fold_addr_expr (base), build_int_cst (ptr_type_node, off))); @}) #endif @end smallexample In the above example, @code{@@2} captures the result of the expression @code{(addr @@0)}. For the outermost expression only its type can be captured, and the keyword @code{type} is reserved for this purpose. The above example also gives a way to conditionalize patterns to only apply to @code{GIMPLE} or @code{GENERIC} by means of using the pre-defined preprocessor macros @code{GIMPLE} and @code{GENERIC} and using preprocessor directives. @smallexample (simplify (bit_and:c integral_op_p@@0 (bit_ior:c (bit_not @@0) @@1)) (bit_and @@1 @@0)) @end smallexample Here we introduce flags on match expressions. The flag used above, @code{c}, denotes that the expression should be also matched commutated. Thus the above match expression is really the following four match expressions: @smallexample (bit_and integral_op_p@@0 (bit_ior (bit_not @@0) @@1)) (bit_and (bit_ior (bit_not @@0) @@1) integral_op_p@@0) (bit_and integral_op_p@@0 (bit_ior @@1 (bit_not @@0))) (bit_and (bit_ior @@1 (bit_not @@0)) integral_op_p@@0) @end smallexample Usual canonicalizations you know from GENERIC expressions are applied before matching, so for example constant operands always come second in commutative expressions. The second supported flag is @code{s} which tells the code generator to fail the pattern if the expression marked with @code{s} does have more than one use and the simplification results in an expression with more than one operator. For example in @smallexample (simplify (pointer_plus (pointer_plus:s @@0 @@1) @@3) (pointer_plus @@0 (plus @@1 @@3))) @end smallexample this avoids the association if @code{(pointer_plus @@0 @@1)} is used outside of the matched expression and thus it would stay live and not trivially removed by dead code elimination. Now consider @code{((x + 3) + -3)} with the temporary holding @code{(x + 3)} used elsewhere. This simplifies down to @code{x} which is desirable and thus flagging with @code{s} does not prevent the transform. Now consider @code{((x + 3) + 1)} which simplifies to @code{(x + 4)}. Despite being flagged with @code{s} the simplification will be performed. The simplification of @code{((x + a) + 1)} to @code{(x + (a + 1))} will not performed in this case though. More features exist to avoid too much repetition. @smallexample (for op (plus pointer_plus minus bit_ior bit_xor) (simplify (op @@0 integer_zerop) @@0)) @end smallexample A @code{for} expression can be used to repeat a pattern for each operator specified, substituting @code{op}. @code{for} can be nested and a @code{for} can have multiple operators to iterate. @smallexample (for opa (plus minus) opb (minus plus) (for opc (plus minus) (simplify... @end smallexample In this example the pattern will be repeated four times with @code{opa, opb, opc} being @code{plus, minus, plus}; @code{plus, minus, minus}; @code{minus, plus, plus}; @code{minus, plus, minus}. To avoid repeating operator lists in @code{for} you can name them via @smallexample (define_operator_list pmm plus minus mult) @end smallexample and use them in @code{for} operator lists where they get expanded. @smallexample (for opa (pmm trunc_div) (simplify... @end smallexample So this example iterates over @code{plus}, @code{minus}, @code{mult} and @code{trunc_div}. Using operator lists can also remove the need to explicitly write a @code{for}. All operator list uses that appear in a @code{simplify} or @code{match} pattern in operator positions will implicitly be added to a new @code{for}. For example @smallexample (define_operator_list SQRT BUILT_IN_SQRTF BUILT_IN_SQRT BUILT_IN_SQRTL) (define_operator_list POW BUILT_IN_POWF BUILT_IN_POW BUILT_IN_POWL) (simplify (SQRT (POW @@0 @@1)) (POW (abs @@0) (mult @@1 @{ built_real (TREE_TYPE (@@1), dconsthalf); @}))) @end smallexample is the same as @smallexample (for SQRT (BUILT_IN_SQRTF BUILT_IN_SQRT BUILT_IN_SQRTL) POW (BUILT_IN_POWF BUILT_IN_POW BUILT_IN_POWL) (simplify (SQRT (POW @@0 @@1)) (POW (abs @@0) (mult @@1 @{ built_real (TREE_TYPE (@@1), dconsthalf); @})))) @end smallexample @code{for}s and operator lists can include the special identifier @code{null} that matches nothing and can never be generated. This can be used to pad an operator list so that it has a standard form, even if there isn't a suitable operator for every form. Another building block are @code{with} expressions in the result expression which nest the generated code in a new C block followed by its argument: @smallexample (simplify (convert (mult @@0 @@1)) (with @{ tree utype = unsigned_type_for (type); @} (convert (mult (convert:utype @@0) (convert:utype @@1))))) @end smallexample This allows code nested in the @code{with} to refer to the declared variables. In the above case we use the feature to specify the type of a generated expression with the @code{:type} syntax where @code{type} needs to be an identifier that refers to the desired type. Usually the types of the generated result expressions are determined from the context, but sometimes like in the above case it is required that you specify them explicitly. Another modifier for generated expressions is @code{!} which tells the machinery to only consider the simplification in case the marked expression simplified to a simple operand. Consider for example @smallexample (simplify (plus (vec_cond:s @@0 @@1 @@2) @@3) (vec_cond @@0 (plus! @@1 @@3) (plus! @@2 @@3))) @end smallexample which moves the outer @code{plus} operation to the inner arms of the @code{vec_cond} expression but only if the actual plus operations both simplify. Note that on @code{GENERIC} a simple operand means that the result satisfies @code{!EXPR_P} which can be limiting if the operation itself simplifies but the remaining operand is an (unrelated) expression. As intermediate conversions are often optional there is a way to avoid the need to repeat patterns both with and without such conversions. Namely you can mark a conversion as being optional with a @code{?}: @smallexample (simplify (eq (convert@@0 @@1) (convert@? @@2)) (eq @@1 (convert @@2))) @end smallexample which will match both @code{(eq (convert @@1) (convert @@2))} and @code{(eq (convert @@1) @@2)}. The optional converts are supposed to be all either present or not, thus @code{(eq (convert@? @@1) (convert@? @@2))} will result in two patterns only. If you want to match all four combinations you have access to two additional conditional converts as in @code{(eq (convert1@? @@1) (convert2@? @@2))}. The support for @code{?} marking extends to all unary operations including predicates you declare yourself with @code{match}. Predicates available from the GCC middle-end need to be made available explicitly via @code{define_predicates}: @smallexample (define_predicates integer_onep integer_zerop integer_all_onesp) @end smallexample You can also define predicates using the pattern matching language and the @code{match} form: @smallexample (match negate_expr_p INTEGER_CST (if (TYPE_OVERFLOW_WRAPS (type) || may_negate_without_overflow_p (t)))) (match negate_expr_p (negate @@0)) @end smallexample This shows that for @code{match} expressions there is @code{t} available which captures the outermost expression (something not possible in the @code{simplify} context). As you can see @code{match} has an identifier as first operand which is how you refer to the predicate in patterns. Multiple @code{match} for the same identifier add additional cases where the predicate matches. Predicates can also match an expression in which case you need to provide a template specifying the identifier and where to get its operands from: @smallexample (match (logical_inverted_value @@0) (eq @@0 integer_zerop)) (match (logical_inverted_value @@0) (bit_not truth_valued_p@@0)) @end smallexample You can use the above predicate like @smallexample (simplify (bit_and @@0 (logical_inverted_value @@0)) @{ build_zero_cst (type); @}) @end smallexample Which will match a bitwise and of an operand with its logical inverted value.