Professional Documents
Culture Documents
Machine-Level Programming 3 Control Flow: Topics
Machine-Level Programming 3 Control Flow: Topics
Topics
Switch Statements
Implementation Options
Series of conditionals Organize in tree structure Logarithmic performance Jump Table Lookup branch target Constant time Possible when cases are small integer constants GCC Picks one based on case structure
long switch_eg (long x, long y, long z) { long w = 1; switch(x) { case 1: w = y*z; break; case 2: w = y/z; /* Fall Through */ case 3: w += z; break; case 5: case 6: w -= z; break; default: w = 2; } return w; } 3
Jump Table
Targ0 Targ1 Targ2
Jump Targets
Targ0: Code Block 0 Code Block 1
Targ1:
Targn-1
Targ2:
Code Block 2
Approx. Translation
target = JTab[x]; goto *target; Targn-1:
Code Block n1
CMSC 313, F09
Setup:
switch_eg: pushl %ebp movl %esp, %ebp pushl %ebx movl $1, %ebx movl 8(%ebp), %edx movl 16(%ebp), %ecx cmpl $6, %edx ja .L61 jmp *.L62(,%edx,4)
# # # # # # # # #
Setup Setup Setup w = 1 edx = x ecx = z x:6 if > goto default goto JTab[x]
CMSC 313, F09
Jumping
jmp .L61 Jump target is denoted by label .L61 jmp *.L62(,%edx,4)
Jump Table
Table Contents
.section .rodata .align 4 .L62: .long .L61 # .long .L56 # .long .L57 # .long .L58 # .long .L61 # .long .L60 # .long .L60 # switch(x) { case 1: // .L56 w = y*z; break; case 2: // .L57 w = y/z; /* Fall Through */ case 3: // .L58 w += z; break; case 5: case 6: // .L60 w -= z; break; default: // .L61 w = 2; }
x x x x x x x
= = = = = = =
0 1 2 3 4 5 6
.L61: // Default case movl $2, %ebx # w = 2 movl %ebx, %eax # Return w popl %ebx leave ret .L57: // Case 2: movl 12(%ebp), %eax # y cltd # Div prep idivl %ecx # y/z movl %eax, %ebx # w = y/z # Fall through .L58: // Case 3: addl %ecx, %ebx # w+= z movl %ebx, %eax # Return w popl %ebx leave ret
.L60: // subl movl popl leave ret .L56: // movl imull movl popl leave ret
Case 1: 12(%ebp), %ebx # w = y %ecx, %ebx # w*= z %ebx, %eax # Return w %ebx
Label .L61 becomes address 0x8048630 Label .L62 becomes address 0x80488dc
Assembly Code
switch_eg: . . . ja .L61 # if > goto default jmp *.L62(,%edx,4) # goto JTab[x]
ja jmp
8048630 *0x80488dc(,%edx,4)
CMSC 313, F09
Disassembled Targets
8048630: 8048635: 8048637: 8048638: 8048639: 804863a: 804863d: 804863e: 8048640: 8048642: 8048644: 8048646: 8048647: 8048648: 8048649: 804864b: 804864d: 804864e: 804864f: 8048650: 8048653: 8048656: 8048658: 8048659: 804865a:
12
bb 89 5b c9 c3 8b 99 f7 89 01 89 5b c9 c3 29 89 5b c9 c3 8b 0f 89 5b c9 c3
02 00 00 00 d8
45 0c f9 c3 cb d8
cb d8
5d 0c af d9 d8
mov mov pop leave ret mov cltd idiv mov add mov pop leave ret sub mov pop leave ret mov imul mov pop leave ret
cb d8
5d 0c af d9 d8
13
Not practical to use jump table Would require 1000 entries Obvious translation into if-then-else would have max. of 9 tests
# x:111
# x:0
15
444
=
>
111
< = >
4
<
777
= >
1
0
= =
7 222
555
= <
888
= >
-1
5 333
=
8 666
=
999
=
-1
-1
-1
16
Summarizing
C Control
Standard Techniques
IA32 loops converted to do-while form Large switch statements use jump tables
Assembler Control
Conditions in CISC
Indirect jump
Compiler
17