CH8.7 查看编译器优化结果

8.7 查看编译的结果

编译器形成汇编代码做了很多事情,但是有些情况下他做的并不是很好,如果我们能够查看汇编代码并且有目的的优化那是最好不过了。
在gcc中可以使用-fsource-asm选项来查看c代码与汇编代码的对应关系,因为只加-S参数的话在.s文件中不会显示源代码只会显示行数。

gcc xx.c -S -fsource-asm

注意 如果你想看看编译器完全优化之后的样子,就不要开启-fsource-asm选项。

下面的例子展示了如何通过汇编代码来优化我们的程序:加入我们有这个一个程序:

void Func(int a[], int & r) {
  int i;
  for (i = 0; i < 100; i++) {
    a[i] = r + i/2;
  }
}

编译后该部分代码是这样的:

ALIGN 4 ; align by 4
PUBLIC ?Func@@YAXQAHAAH@Z ; mangled function name
?Func@@YAXQAHAAH@Z PROC NEAR ; start of Func
; parameter 1: 8 + esp ; a
; parameter 2: 12 + esp ; r
    $B1$1:                        ; unused label
    push ebx                      ; save ebx on stack
    mov ecx, DWORD PTR [esp+8]    ; ecx = a
    xor eax, eax                  ; eax = i = 0
    mov edx, DWORD PTR [esp+12]   ; edx = r
$B1$2:                            ; top of loop
    mov ebx, eax                  ; compute i/2 in ebx
    shr ebx, 31                   ; shift down sign bit of i
    add ebx, eax                  ; i + sign(i)  ;; == i + 1 ,because of division operation
    sar ebx, 1                    ; shift right = divide by 2
    add ebx, DWORD PTR [edx]      ; add what r points to
    mov DWORD PTR[ecx+eax*4],ebx  ; store result in array
    add eax, 1                    ; i++
    cmp eax, 100                  ; check if i < 100
    jl $B1$2                      ; repeat loop if true
$B1$3:                            ; unused label
    pop ebx                       ; restore ebx from stack
    ret                           ; return
    ALIGN 4                       ; align
?Func@@YAXQAHAAH@Z ENDP           ; mark end of procedure

上面代码比较好懂,但是有个一个部分我是看了好久然后跟实验室同学讨论过后才明白,就是shr ebx, eax \n add ebx, eax这两步为什么要有i + sign(i)操作呢?
CSAPP上有详细的讲计算机补码的除法运算那一部分有讲:除法的补救部分,对于除数是2的指数可以用以为操作来计算,但是当被除数是负数的时候(第一位是1),单纯的移位不能得到正确的结果,但是有一个技巧就是对被除数加上一个“偏置”然后移位,比如说a / (2^k) 在计算机里面a是用a的补码表示的,计算方法就是 : (a_补 + 2^k-1) >> k; 如果a是正数就直接a>>k;
对应于上面汇编,因为除数是2(源代码里对应 "i / 2")所以如果i<0要对i进行"+2^1-1 = +1"操作,如果i>0就+0,正好对应+sign(i) 操作,不得不说编译器还真TM智能。

上面代码我们可能我们每天都会写,但是仔细阅读上面汇编我们可能发现如下问题:
1. i+sign(i)然后移位的操作虽然很秀,但是根本没必要,因为通过代码我们知道i是loop counter,不可能会小于0,所以除2操作直接移位就可以了;
2. add ebx, DWORD PTR [edx] ; add what r points to 这一步在循环中反复被执行,而我们知道这对应着一步访存操作,因为函数里面我们传的是&r,我们不知道r会不会在某一时刻改变,所以只能每次loop都重新从内存中load一遍,非常的耗时。
3. 由于r + i/2 是i的阶跃函数,所以我们可以把这个循环给展开来。

针对以上三点,我们修改后的代码如下:

void Func(int a[], int & r) {
  unsigned int i;
  int Induction = r;
  for (i = 0; i < 100; i += 2) {
    a[i] = Induction;
    a[i+1] = Induction;
    Induction++;
  }
}

编译后的代码如下:

ALIGN 4 ; align by 4
PUBLIC ?Func@@YAXQAHAAH@Z ; mangled function name
?Func@@YAXQAHAAH@Z PROC NEAR ; start of Func
; parameter 1: 4 + esp ; a
; parameter 2: 8 + esp ; r
$B1$1: ; unused label
    mov eax, DWORD PTR [esp+4]    ; eax = address of a
    mov edx, DWORD PTR [esp+8]    ; edx = address in r
    mov ecx, DWORD PTR [edx]      ; ecx = Induction
    lea edx, DWORD PTR [eax+400]  ; edx = point to end of a
$B2$2:                            ; top of loop
    mov DWORD PTR [eax], ecx      ; a[i] = Induction;
    mov DWORD PTR [eax+4], ecx    ; a[i+1] = Induction;
    add ecx, 1                    ; Induction++;
    add eax, 8                    ; point to a[i+2]
    cmp edx, eax                  ; compare with end of array
    ja $B2$2                      ; jump to top of loop
$B2$3:                            ; unused label
    ret                           ; return from Func
    ALIGN 4
; mark_end;

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容