RFR: 8325495: C2: implement optimization for series of Add of unique value [v2]
Roland Westrelin
roland at openjdk.org
Tue Sep 17 09:42:09 UTC 2024
On Mon, 16 Sep 2024 20:51:49 GMT, Kangcheng Xu <kxu at openjdk.org> wrote:
>> This pull request resolves [JDK-8325495](https://bugs.openjdk.org/browse/JDK-8325495) by converting series of additions of the same operand into multiplications. I.e., `a + a + ... + a + a + a => n*a`.
>>
>> As an added benefit, it also converts `C * a + a` into `(C+1) * a` and `a << C + a` into `(2^C + 1) * a` (with respect to constant `C`). This is actually a side effect of IGVN being iterative: at converting the `i`-th addition, the previous `i-1` additions would have already been optimized to multiplication (and thus, further into bit shifts and additions/subtractions if possible).
>>
>> Some notable examples of this transformation include:
>> - `a + a + a` => `a*3` => `(a<<1) + a`
>> - `a + a + a + a` => `a*4` => `a<<2`
>> - `a*3 + a` => `a*4` => `a<<2`
>> - `(a << 1) + a + a` => `a*2 + a + a` => `a*3 + a` => `a*4 => a<<2`
>>
>> See included IR unit tests for more.
>
> Kangcheng Xu has updated the pull request with a new target base due to a merge or a rebase. The incremental webrev excludes the unrelated changes brought in by the merge/rebase. The pull request contains 22 additional commits since the last revision:
>
> - Merge branch 'openjdk:master' into arithmetic-canonicalization
> - Merge pull request #1 from tabjy/arithmetic-canonicalization-v2
>
> Arithmetic canonicalization v2
> - remove dead code
> - fix potential void type const nodes
> - refactor and cleanup
> - add more test cases
> - re-implement depth limit on recursion
> - passes TestIRLShiftIdeal_XPlusX_LShiftC
> - passes AddI[L]NodeIdealizationTests
> - revert depth limits
> - ... and 12 more: https://git.openjdk.org/jdk/compare/e04bd3f5...c8fdb74c
src/hotspot/share/opto/addnode.cpp line 427:
> 425: }
> 426:
> 427: Node* con = (bt == T_INT) ? (Node*) phase->intcon((jint) factor) : (Node*) phase->longcon(factor);
You can use `integercon()` and pass `bt`
src/hotspot/share/opto/addnode.cpp line 441:
> 439: bool AddNode::is_optimized_multiplication(Node* node, Node* base) {
> 440: // Look for pattern: LShiftNode(a, CON)
> 441: if (node->is_LShift() && node->in(2)->is_Con()) {
Maybe passing `bt` to this method would make the whole thing more readable. You would use `Op_Lshift(bt)` here.
src/hotspot/share/opto/addnode.cpp line 481:
> 479:
> 480: // MulNode(any, const), e.g., a*2
> 481: if (node->is_Mul()
Same here: passing `bt` would turn this into `node->Opcode() == Op_Mul(bt)`.
src/hotspot/share/opto/addnode.cpp line 490:
> 488: if (bt == T_INT || bt == T_LONG) { // const could potentially be void type
> 489: Node* mul_base;
> 490: jlong multiplier = extract_base_operand_from_serial_additions(phase, operand_node, &mul_base, depth_limit - 1);
Do you need to recurse at all here?
-------------
PR Review Comment: https://git.openjdk.org/jdk/pull/20754#discussion_r1762855801
PR Review Comment: https://git.openjdk.org/jdk/pull/20754#discussion_r1762859687
PR Review Comment: https://git.openjdk.org/jdk/pull/20754#discussion_r1762865643
PR Review Comment: https://git.openjdk.org/jdk/pull/20754#discussion_r1762866777
More information about the hotspot-compiler-dev
mailing list