PHP 8新特性之JIT

PHP8 alpha1已经在6月25日发布,其中包含了不少的新特性,当然我自己认为最重要的还是JIT,相信关于JIT也是大家最关心的,它到底怎么用,有什么要注意的,以及性能提升到底咋样?今天我们就来扒扒JIT

JIT :Just In Time, 一种编译器策略,将代码在运行时转换为依赖于体系结构的机器码,并即时执行。

JIT 的好处:

1. 目前已经很难通过常规手段提升 PHP 的性能,JIT 基本上是目前性能提升的唯一手段;

2. JIT 带来的性能提升可以让 PHP 在更多使用场景( CPU 密集)中发挥作用;

3. 可以使用 PHP 来开发内置函数,而不用担心性能方面的问题。这一方面可以加速语言的发展(更多人可以参与进来),同时也可以减少目前使用 C 开发容易出现的内存管理、溢出等问题

主要是 JIT 带来的性能提升,以及性能提升对整个语言使用场景的扩展及语言生态发展的支持。

首先,我们来看一张图:

左图是PHP8之前的Opcache流程示意图,右图是PHP8中的Opcache示意图,可以裁剪几个关键点:

1. Opcache会做opcode层面的优化,某些图中的俩条opcode合并为一条

2. PHP8的JIT目前是在Opcache之中提供的

3. JIT在Opcache优化之后的基础上,结合Runtime的信息再次优化,直接生成机器码

4. JIT不是原来的Opcache优化的替代,是增强

5. 目前PHP8只支持x86架构的CPU

实际上JIT共享了很多原来的Opcache做优化的基础数据结构,比如data flow graph, call graph, SSA等,关于这部分,后续如果有时间,可以单独在写一个文章来介绍,今天只是只是着重在使用焦点。

下载安装好以后,除掉初始的opcache配置以外,对于JIT我们需要添加如下配置到php.ini:

opcache.jit = 1205

opcache.jit_buffer_size = 64M

opcache.jit这个配置看起来稍微有点复杂,我来解释下,这个配置由4个独立的数字组成,从左到右分别是(请注意,这个是基于现有alpha1的版本设置,一些配置可能会随身着后续版本做微调):

是否在生成机器码点时候使用AVX指令,需要CPU支持:

0:不使用

1:使用

寄存器分配策略:

0: 不使用寄存器分配

1: 局部(block)域分配

2: 全局(function)域分配

JIT触发策略:

0: PHP脚本载入的时候就JIT

1: 当函数第一次被执行时JIT

2: 在一次运行后,JIT调用次数最多的百分之(opcache.prof_threshold * 100)的函数

3: 当函数/方法执行超过N(N和opcache.jit_hot_func相关)次以后JIT

4: 当函数方法的注释中含有@jit的时候对它进行JIT

5: 当一个Trace执行超过N次(和opcache.jit_hot_loop, jit_hot_return等有关)以后JIT

JIT优化策略,数值最大优化力度越大:

0: 不JIT

1: 做opline之间的跳转部分的JIT

2: 内敛opcode handler调用

3: 基于类型推断做函数级别的JIT

4: 基于类型推断,过程调用图做函数级别JIT

5: 基于类型推断,过程调用图做脚本级别的JIT

根据此,我们可以大概得到如下几个附图:

尽量使用12x5型的配置,此时应该是效果最优的

对于x,如果是脚本等级的,推荐使用0,如果是Web服务型的,可以根据测试结果选择3或5

@jit的形式,在有了属性以后,可能变成<< jit>>

现在,我们来测试下启用和不启用JIT的时候,Zend / bench.php的差异,首先是不启用(php -d opcache.jit_buffer_size=0 Zend/bench.php):

simple            0.008

simplecall        0.004

simpleucall        0.004

simpleudcall      0.004

mandel            0.035

mandel2            0.055

ackermann(7)      0.020

ary(50000)        0.004

ary2(50000)        0.003

ary3(2000)        0.048

fibo(30)          0.084

hash1(50000)      0.013

hash2(500)        0.010

heapsort(20000)    0.027

matrix(20)        0.026

nestedloop(12)    0.023

sieve(30)          0.013

strcat(200000)    0.006

------------------------

Total              0.387

根据上面的介绍,我们选择opcache.jit = 1205,因为bench.php是脚本(php -d opcache.jit_buffer_size=64M -d opcache.jit=1205 Zend/bench.php):

simple            0.002

simplecall        0.001

simpleucall        0.001

simpleudcall      0.001

mandel            0.010

mandel2            0.011

ackermann(7)      0.010

ary(50000)        0.003

ary2(50000)        0.002

ary3(2000)        0.018

fibo(30)          0.031

hash1(50000)      0.011

hash2(500)        0.008

heapsort(20000)    0.014

matrix(20)        0.015

nestedloop(12)    0.011

sieve(30)          0.005

strcat(200000)    0.004

------------------------

Total              0.157

可见,对于Zend / bench.php,大约不开启JIT,开启了以后,耗时降低将近60%,性能提升将近2倍

对于大家研究学习来说,可以通过opcache.jit_debug来观察JIT后生成的汇编结果,例如针对:

function simple() {

  $a = 0;

  for ($i = 0; $i < 1000000; $i++)

    $a++;

}

我们通过php -d opcache.jit = 1205 -dopcache.jit_debug = 0x01可以看到:

JIT$simple: ; (/tmp/1.php)

    sub $0x10, %rsp

    xor %rdx, %rdx

    jmp .L2

.L1:

    add $0x1, %rdx

.L2:

    cmp $0x0, EG(vm_interrupt)

    jnz .L4

    cmp $0xf4240, %rdx

    jl .L1

    mov 0x10(%r14), %rcx

    test %rcx, %rcx

    jz .L3

    mov $0x1, 0x8(%rcx)

.L3:

    mov 0x30(%r14), %rax

    mov %rax, EG(current_execute_data)

    mov 0x28(%r14), %edi

    test $0x9e0000, %edi

    jnz JIT$$leave_function

    mov %r14, EG(vm_stack_top)

    mov 0x30(%r14), %r14

    cmp $0x0, EG(exception)

    mov (%r14), %r15

    jnz JIT$$leave_throw

    add $0x20, %r15

    add $0x10, %rsp

    jmp (%r15)

.L4:

    mov $0x45543818, %r15

    jmp JIT$$interrupt_handler

大家可以尝试阅读这一级汇编,其中一些针对i的增量,可以看到优化力度很大,因为i是局部变量直接分配在寄存器中,i的范围不会大于1000000,所以不需要判断是否整数溢出等等。

而如果我们采用opcache.jit=1005,如前面的介绍,也就是不使用寄存器分配,可以得到如下结果:

JIT$simple: ; (/tmp/1.php)

    sub $0x10, %rsp

    mov $0x0, 0x50(%r14)

    mov $0x4, 0x58(%r14)

    jmp .L2

.L1:

    add $0x1, 0x50(%r14)

.L2:

    cmp $0x0, EG(vm_interrupt)

    jnz .L4

    cmp $0xf4240, 0x50(%r14)

    jl .L1

    mov 0x10(%r14), %rcx

    test %rcx, %rcx

    jz .L3

    mov $0x1, 0x8(%rcx)

.L3:

    mov 0x30(%r14), %rax

    mov %rax, EG(current_execute_data)

    mov 0x28(%r14), %edi

    test $0x9e0000, %edi

    jnz JIT$$leave_function

    mov %r14, EG(vm_stack_top)

    mov 0x30(%r14), %r14

    cmp $0x0, EG(exception)

    mov (%r14), %r15

    jnz JIT$$leave_throw

    add $0x20, %r15

    add $0x10, %rsp

    jmp (%r15)

.L4:

    mov $0x44cdb818, %r15

    jmp JIT$$interrupt_handler

可以看到针对i的部分,现在是在内存操作,并没有使用寄存器。

再如果我们采用opcache.jit=1201,我们可以得到如下结果:

JIT$simple: ; (/tmp/1.php)

    sub $0x10, %rsp

    call ZEND_QM_ASSIGN_NOREF_SPEC_CONST_HANDLER

    add $0x40, %r15

    jmp .L2

.L1:

    call ZEND_PRE_INC_LONG_NO_OVERFLOW_SPEC_CV_RETVAL_UNUSED_HANDLER

    cmp $0x0, EG(exception)

    jnz JIT$$exception_handler

.L2:

    cmp $0x0, EG(vm_interrupt)

    jnz JIT$$interrupt_handler

    call ZEND_IS_SMALLER_LONG_SPEC_TMPVARCV_CONST_JMPNZ_HANDLER

    cmp $0x0, EG(exception)

    jnz JIT$$exception_handler

    cmp $0x452a0858, %r15d

    jnz .L1

    add $0x10, %rsp

    jmp ZEND_RETURN_SPEC_CONST_LABEL

这就只是简单的内敛部分操作码处理程序的调用了。

您也可以尝试各种opcache.jit的策略绑定调试的配置,来观察结果的不同,也可以尝试各种opcache.jit_debug的配置,大约0xff,将会有更多的辅助信息输出。

以上内容希望帮助到大家,很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家,需要的可以加入我的官方群953224940

进阶PHP月薪30k>>>架构师成长路线【视频、面试文档免费获取】

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

推荐阅读更多精彩内容