无用的测试指令?

时间:2019-01-05 18:09:15

标签: java assembly jvm jit jvm-hotspot

我得到了以下程序集列表,作为Java程序的JIT编译的结果。

mov    0x14(%rsp),%r10d
inc    %r10d              

mov    0x1c(%rsp),%r8d
inc    %r8d               

test   %eax,(%r11)         ; <--- this instruction

mov    (%rsp),%r9
mov    0x40(%rsp),%r14d
mov    0x18(%rsp),%r11d
mov    %ebp,%r13d
mov    0x8(%rsp),%rbx
mov    0x20(%rsp),%rbp
mov    0x10(%rsp),%ecx
mov    0x28(%rsp),%rax    

movzbl 0x18(%r9),%edi     
movslq %r8d,%rsi          

cmp    0x30(%rsp),%rsi
jge    0x00007fd3d27c4f17 

我对test指令的理解在这里没有用,因为测试的主要思想是

  

标志SF,ZF,PF被修改,而AND的结果被丢弃。

,这里我们不使用这些结果标志。

这是JIT中的错误还是我错过了什么? 如果是,最好的报告地点在哪里? 谢谢!

1 个答案:

答案 0 :(得分:44)

那必须是thread-local handshake poll。 查看从何处读取%r11。如果从%r15(线程本地存储)的某个偏移量读取它,那就好了。请参见示例here

  0.31%  ↗  ...70: movzbl 0x94(%r9),%r10d    
  0.19%  │  ...78: mov    0x108(%r15),%r11  ; read the thread-local page addr
 25.62%  │  ...7f: add    $0x1,%rbp          
 35.10%  │  ...83: test   %eax,(%r11)       ; thread-local handshake poll
 34.91%  │  ...86: test   %r10d,%r10d
         ╰  ...89: je     ...70

这不是没有用的,一旦保护页面被标记为不可读,它将导致SEGV,并将控制权转移到JVM的SEGV处理程序。这是JVM安全点Java线程机制的一部分,例如用于GC。

UPD:希望提供更多详细信息,here