| ## This checks that we don't create an invalid CFG when there is an |
| ## unreachable direct jump right after an indirect one. |
| |
| # REQUIRES: system-linux |
| |
| # RUN: llvm-mc -filetype=obj -triple x86_64-unknown-unknown \ |
| # RUN: %s -o %t.o |
| # RUN: link_fdata %s %t.o %t.fdata |
| # RUN: llvm-strip --strip-unneeded %t.o |
| # RUN: %clang %cflags -no-pie %t.o -o %t.exe -Wl,-q -nostdlib |
| # RUN: llvm-bolt %t.exe -o %t.out --data %t.fdata \ |
| # RUN: --eliminate-unreachable --print-cfg | FileCheck %s |
| |
| .globl _start |
| .type _start, %function |
| _start: |
| .cfi_startproc |
| # FDATA: 0 [unknown] 0 1 _start 0 0 1 |
| push %rbp |
| mov %rsp, %rbp |
| push %rbx |
| push %r14 |
| subq $0x20, %rsp |
| movq %rdi, %rcx |
| b: |
| jmpq *JUMP_TABLE(,%rcx,8) |
| # FDATA: 1 _start #b# 1 _start #hotpath# 0 20 |
| ## Unreachable direct jump here. Our CFG should still make sense and properly |
| ## place this instruction in a new basic block. |
| jmp .lbb2 |
| .lbb1: je .lexit |
| .lbb2: |
| xorq %rax, %rax |
| addq $0x20, %rsp |
| pop %r14 |
| pop %rbx |
| pop %rbp |
| ret |
| hotpath: |
| movq $2, %rax |
| addq $0x20, %rsp |
| pop %r14 |
| pop %rbx |
| pop %rbp |
| ret |
| .lexit: |
| movq $1, %rax |
| addq $0x20, %rsp |
| pop %r14 |
| pop %rbx |
| pop %rbp |
| ret |
| .cfi_endproc |
| .size _start, .-_start |
| |
| .rodata |
| .globl JUMP_TABLE |
| JUMP_TABLE: |
| .quad .lbb1 |
| .quad .lbb2 |
| .quad hotpath |
| |
| ## No basic blocks above should have 4 successors! That is a bug. |
| # CHECK-NOT: Successors: {{.*}} (mispreds: 0, count: 20), {{.*}} (mispreds: 0, count: 0), {{.*}} (mispreds: 0, count: 0), {{.*}} (mispreds: 0, count: 0) |
| # Check successful removal of stray direct jmp |
| # CHECK: UCE removed 1 block |