Stream: git-wasmtime

Topic: wasmtime / issue #4635 Cranelift: Cold annotations not pr...


view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 04:40):

mchesser opened issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold, improves code generation:

```asm
.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and uses heavy cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 04:40):

mchesser labeled issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold, improves code generation:

```asm
.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and uses heavy cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 04:40):

mchesser labeled issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold, improves code generation:

```asm
.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and uses heavy cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 04:42):

mchesser edited issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold, improves code generation:

```asm
.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and heavily uses cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 04:42):

mchesser edited issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold (#4636), improves code generation:

```asm
.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and heavily uses cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 04:43):

mchesser edited issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold (#4636), improves code generation:

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and heavily uses cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 08 2022 at 05:03):

mchesser edited issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold (#4636), improves code generation:

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and heavily uses cold annotations).

view this post on Zulip Wasmtime GitHub notifications bot (Aug 09 2022 at 05:05):

cfallin closed issue #4635:

During lowering Cranelift may generate additional blocks to handle moves that conceptually occur on edges. The block ordering code treats these blocks the same as ordinary blocks, irrespective of whether they are associated with a cold block. This means that edges from a cold block can cause additional code to be inserted in the hot path.

For example, consider the following IL:

function %test_cold(i64, i64, i64, i64) -> i64 {
block0(v0: i64, v1: i64, v2: i64, v3: i64):
    brz v0, block2
    jump block1

block1:
    v10 = iadd.i64 v0, v2
    jump block3(v10)

block2 cold:
    v20 = iadd.i64 v1, v3
    brnz v20, block3(v20)
    jump block4(v3)

block3(v30: i64):
    v34 = iadd.i64 v30, v1
    jump block4(v34)

block4(v40: i64):
    return v40
}

Currently Cranelift emits the following (annotated x86-64 code):

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x28        ; .block2

    jmp     0x1a        ; .block1

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x20        ; .block4

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4:
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x1d        ; .block3

    jmp     0x12        ; .block2_to_block4_edge

Modifying the lowering stage to mark edge blocks as cold if either the predecessor or successor block is cold (#4636), improves code generation:

.block0:
    push    rbp
    mov     rbp, rsp
    test    rdi, rdi
    je      0x1b  ; .block2

.block1:
    add     rdi, rdx

.block3:
    add     rdi, rsi

.block4
    mov     rax, rdi
    mov     rsp, rbp
    pop     rbp
    ret

.block2:
    mov     rdi, rsi
    add     rdi, rcx
    test    rdi, rdi
    jne     0x10      ; .block3

.block2_to_block4_edge:
    mov     rdi, rcx
    jmp     0x13      ; .block4

This provided significant runtime speedups (~11%) for my code (which is severely frontend bound and heavily uses cold annotations).


Last updated: Jan 24 2025 at 00:11 UTC