[LLVMdev] How distinguish Catch all llvm-IR from other catch type ?

Török Edwin edwintorok at gmail.com
Tue May 12 00:13:19 PDT 2009


On 2009-05-12 10:03, zhengjian zhang wrote:
> Hi,
>
> catch_all.cpp:
>
>      1  int main()
>      2  {
>      3      try {
>      4          throw 34;
>      5      }
>      6      catch (...) {}
>      7  }
>
>
> llvm-gcc -O3 -S -emit-llvm  catch_all.cpp   -o catch_all.ll:
>
>      1  ; ModuleID = 'catch_all.cpp'
>      2  target datalayout =
> "e-p:32:32:32-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64-f32:32:32-f64:32:64-v64:64:64-v128:128:128-a0:0:64-f80:32:32"
>      3  target triple = "i386-pc-linux-gnu"
>      4          %struct.__fundamental_type_info_pseudo = type {
> %struct.__type_info_pseudo }
>      5          %struct.__type_info_pseudo = type { i8*, i8* }
>      6  @_ZTIi = external constant
> %struct.__fundamental_type_info_pseudo               ;
> <%struct.__fundamental_type_info_pseudo*> [#uses=1]
>      7
>      8  define i32 @main() {
>      9  entry:
>     10          %0 = tail call i8* @__cxa_allocate_exception(i32 4)
> nounwind            ; <i8*> [#uses=2]
>     11          %1 = bitcast i8* %0 to i32*             ; <i32*> [#uses=1]
>     12          store i32 34, i32* %1, align 4
>     13          invoke void @__cxa_throw(i8* %0, i8* bitcast
> (%struct.__fundamental_type_info_pseudo* @_ZTIi to i8*), void (i8*)*
> null) noreturn
>     14                          to label %invcont unwind label %lpad
>     15
>     16  invcont:                ; preds = %entry
>     17          unreachable
>     18
>     19  lpad:           ; preds = %entry
>     20          %eh_ptr = tail call i8* @llvm.eh.exception()
>  ; <i8*> [#uses=2]
>     21          %eh_select = tail call i32 (i8*, i8*, ...)*
> @llvm.eh.selector.i32(i8* %eh_ptr, i8* bitcast (i32 (...)*
> @__gxx_personality_sj0 to i8*))          ; <i32> [#uses=0]
>     22          %2 = tail call i8* @__cxa_begin_catch(i8* %eh_ptr)
> nounwind             ; <i8*> [#uses=0]
>     23          tail call void @__cxa_end_catch()
>     24          ret i32 0
>     25  }
>     26
>     27  declare i8* @__cxa_allocate_exception(i32) nounwind
>     28
>     29  declare void @__cxa_throw(i8*, i8*, void (i8*)*) noreturn
>     30
>     31  declare i8* @__cxa_begin_catch(i8*) nounwind
>     32
>     33  declare i8* @llvm.eh.exception() nounwind
>     34
>     35  declare i32 @llvm.eh.selector.i32(i8*, i8*, ...) nounwind
>     36
>     37  declare void @__cxa_end_catch()
>     38
>     39  declare i32 @__gxx_personality_sj0(...)
>
>
> from the LLVM-IR, catch all  was  translated  to  LLVM-IR :
> %eh_select = tail call i32 (i8*, i8*, ...)* @llvm.eh.selector.i32(i8*
> %eh_ptr, i8* bitcast (i32 (...)* @__gxx_personality_sj0 to i8*))
> here llvm.eh.selector.i32 has only two args,  but from the doc, it
> says llvm.eh.selector takes a minimum of three arguments.
>   

I get an extra null arg, I don't know why you don't:
    %eh_select = tail call i32 (i8*, i8*, ...)*
@llvm.eh.selector.i32(i8* %eh_ptr, i8* bitcast (i32 (...)*
@__gxx_personality_v0 to i8*), i8* null)        ; <i32> [#uses=0]

Compare it  with the IR generated for catch(int) or catch(someOtherType).

Here is how catch(int) looks like for me, notice the additional @_ZTIi
parameter, and the icmp for the typeid:

bb:        ; preds = %lpad
    %2 = tail call i8* @__cxa_begin_catch(i8* %eh_ptr) nounwind        ;
<i8*> [#uses=0]
    tail call void @__cxa_end_catch()
    ret i32 0

lpad: %eh_ptr = tail call i8* @llvm.eh.exception()        ; <i8*> [#uses=3]
    %eh_select = tail call i32 (i8*, i8*, ...)*
@llvm.eh.selector.i32(i8* %eh_ptr, i8* bitcast (i32 (...)*
@__gxx_personality_v0 to i8*), i8* bitcast
(%struct.__fundamental_type_info_pseudo* @_ZTIi to i8*), i8* null)   
    ; <i32> [#uses=1]
    %eh_typeid = tail call i32 @llvm.eh.typeid.for.i32(i8* bitcast
(%struct.__fundamental_type_info_pseudo* @_ZTIi to i8*))        ; <i32>
[#uses=1]
    %3 = icmp eq i32 %eh_select, %eh_typeid        ; <i1> [#uses=1]
    br i1 %3, label %bb, label %Unwind

Unwind:        ; preds = %lpad
    tail call void @_Unwind_Resume(i8* %eh_ptr)
    unreachable


Best regards,
--Edwin



More information about the llvm-dev mailing list