1 /*
2 +----------------------------------------------------------------------+
3 | Zend Engine |
4 +----------------------------------------------------------------------+
5 | Copyright (c) 2018-2018 Zend Technologies Ltd. (http://www.zend.com) |
6 +----------------------------------------------------------------------+
7 | This source file is subject to version 2.00 of the Zend license, |
8 | that is bundled with this package in the file LICENSE, and is |
9 | available through the world-wide-web at the following url: |
10 | http://www.zend.com/license/2_00.txt. |
11 | If you did not receive a copy of the Zend license and are unable to |
12 | obtain it through the world-wide-web, please send a note to |
13 | license@zend.com so we can mail you a copy immediately. |
14 +----------------------------------------------------------------------+
15 | Authors: Xinchen Hui <xinchen.h@zend.com> |
16 +----------------------------------------------------------------------+
17 */
18
19 #ifndef ZEND_CPU_INFO_H
20 #define ZEND_CPU_INFO_H
21
22 #include "zend.h"
23
24 #define ZEND_CPU_EBX_MASK (1<<30)
25 #define ZEND_CPU_EDX_MASK (1<<31)
26
27 typedef enum _zend_cpu_feature {
28 /* ECX */
29 ZEND_CPU_FEATURE_SSE3 = (1<<0),
30 ZEND_CPU_FEATURE_PCLMULQDQ = (1<<1),
31 ZEND_CPU_FEATURE_DTES64 = (1<<2),
32 ZEND_CPU_FEATURE_MONITOR = (1<<3),
33 ZEND_CPU_FEATURE_DSCPL = (1<<4),
34 ZEND_CPU_FEATURE_VMX = (1<<5),
35 ZEND_CPU_FEATURE_SMX = (1<<6),
36 ZEND_CPU_FEATURE_EST = (1<<7),
37 ZEND_CPU_FEATURE_TM2 = (1<<8),
38 ZEND_CPU_FEATURE_SSSE3 = (1<<9),
39 ZEND_CPU_FEATURE_CID = (1<<10),
40 ZEND_CPU_FEATURE_SDBG = (1<<11),
41 ZEND_CPU_FEATURE_FMA = (1<<12),
42 ZEND_CPU_FEATURE_CX16 = (1<<13),
43 ZEND_CPU_FEATURE_XTPR = (1<<14),
44 ZEND_CPU_FEATURE_PDCM = (1<<15),
45 /* reserved = (1<<16),*/
46 ZEND_CPU_FEATURE_PCID = (1<<17),
47 ZEND_CPU_FEATURE_DCA = (1<<18),
48 ZEND_CPU_FEATURE_SSE41 = (1<<19),
49 ZEND_CPU_FEATURE_SSE42 = (1<<20),
50 ZEND_CPU_FEATURE_X2APIC = (1<<21),
51 ZEND_CPU_FEATURE_MOVBE = (1<<22),
52 ZEND_CPU_FEATURE_POPCNT = (1<<23),
53 ZEND_CPU_FEATURE_TSC_DEADLINE = (1<<24),
54 ZEND_CPU_FEATURE_AES = (1<<25),
55 ZEND_CPU_FEATURE_XSAVE = (1<<26),
56 ZEND_CPU_FEATURE_OSXSAVE = (1<<27) ,
57 ZEND_CPU_FEATURE_AVX = (1<<28),
58 ZEND_CPU_FEATURE_F16C = (1<<29),
59 /* intentionally don't support = (1<<30) */
60 /* intentionally don't support = (1<<31) */
61
62 /* EBX */
63 ZEND_CPU_FEATURE_AVX2 = (1<<5 | ZEND_CPU_EBX_MASK),
64
65 /* EDX */
66 ZEND_CPU_FEATURE_FPU = (1<<0 | ZEND_CPU_EDX_MASK),
67 ZEND_CPU_FEATURE_VME = (1<<1 | ZEND_CPU_EDX_MASK),
68 ZEND_CPU_FEATURE_DE = (1<<2 | ZEND_CPU_EDX_MASK),
69 ZEND_CPU_FEATURE_PSE = (1<<3 | ZEND_CPU_EDX_MASK),
70 ZEND_CPU_FEATURE_TSC = (1<<4 | ZEND_CPU_EDX_MASK),
71 ZEND_CPU_FEATURE_MSR = (1<<5 | ZEND_CPU_EDX_MASK),
72 ZEND_CPU_FEATURE_PAE = (1<<6 | ZEND_CPU_EDX_MASK),
73 ZEND_CPU_FEATURE_MCE = (1<<7 | ZEND_CPU_EDX_MASK),
74 ZEND_CPU_FEATURE_CX8 = (1<<8 | ZEND_CPU_EDX_MASK),
75 ZEND_CPU_FEATURE_APIC = (1<<9 | ZEND_CPU_EDX_MASK),
76 /* reserved = (1<<10 | ZEND_CPU_EDX_MASK),*/
77 ZEND_CPU_FEATURE_SEP = (1<<11 | ZEND_CPU_EDX_MASK),
78 ZEND_CPU_FEATURE_MTRR = (1<<12 | ZEND_CPU_EDX_MASK),
79 ZEND_CPU_FEATURE_PGE = (1<<13 | ZEND_CPU_EDX_MASK),
80 ZEND_CPU_FEATURE_MCA = (1<<14 | ZEND_CPU_EDX_MASK),
81 ZEND_CPU_FEATURE_CMOV = (1<<15 | ZEND_CPU_EDX_MASK),
82 ZEND_CPU_FEATURE_PAT = (1<<16 | ZEND_CPU_EDX_MASK),
83 ZEND_CPU_FEATURE_PSE36 = (1<<17 | ZEND_CPU_EDX_MASK),
84 ZEND_CPU_FEATURE_PN = (1<<18 | ZEND_CPU_EDX_MASK),
85 ZEND_CPU_FEATURE_CLFLUSH = (1<<19 | ZEND_CPU_EDX_MASK),
86 /* reserved = (1<<20 | ZEND_CPU_EDX_MASK),*/
87 ZEND_CPU_FEATURE_DS = (1<<21 | ZEND_CPU_EDX_MASK),
88 ZEND_CPU_FEATURE_ACPI = (1<<22 | ZEND_CPU_EDX_MASK),
89 ZEND_CPU_FEATURE_MMX = (1<<23 | ZEND_CPU_EDX_MASK),
90 ZEND_CPU_FEATURE_FXSR = (1<<24 | ZEND_CPU_EDX_MASK),
91 ZEND_CPU_FEATURE_SSE = (1<<25 | ZEND_CPU_EDX_MASK),
92 ZEND_CPU_FEATURE_SSE2 = (1<<26 | ZEND_CPU_EDX_MASK),
93 ZEND_CPU_FEATURE_SS = (1<<27 | ZEND_CPU_EDX_MASK),
94 ZEND_CPU_FEATURE_HT = (1<<28 | ZEND_CPU_EDX_MASK),
95 ZEND_CPU_FEATURE_TM = (1<<29 | ZEND_CPU_EDX_MASK)
96 /*intentionally don't support = (1<<30 | ZEND_CPU_EDX_MASK)*/
97 /*intentionally don't support = (1<<31 | ZEND_CPU_EDX_MASK)*/
98 } zend_cpu_feature;
99
100 void zend_cpu_startup();
101 ZEND_API int zend_cpu_supports(zend_cpu_feature feature);
102
103 /* Address sanitizer is incompatible with ifunc resolvers, so exclude the
104 * CPU support helpers from asan.
105 * See also https://github.com/google/sanitizers/issues/342. */
106 #if __has_attribute(no_sanitize_address)
107 # define ZEND_NO_SANITIZE_ADDRESS __attribute__((no_sanitize_address))
108 #else
109 # define ZEND_NO_SANITIZE_ADDRESS
110 #endif
111
112 #if PHP_HAVE_BUILTIN_CPU_SUPPORTS
113 /* NOTE: you should use following inline function in
114 * resolver functions (ifunc), as it could be called
115 * before all PLT symbols are resloved. in other words,
116 * resolver functions should not depends any external
117 * functions */
118 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_sse2()119 static zend_always_inline int zend_cpu_supports_sse2() {
120 #if PHP_HAVE_BUILTIN_CPU_INIT
121 __builtin_cpu_init();
122 #endif
123 return __builtin_cpu_supports("sse2");
124 }
125
126 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_sse3()127 static zend_always_inline int zend_cpu_supports_sse3() {
128 #if PHP_HAVE_BUILTIN_CPU_INIT
129 __builtin_cpu_init();
130 #endif
131 return __builtin_cpu_supports("sse3");
132 }
133
134 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_ssse3()135 static zend_always_inline int zend_cpu_supports_ssse3() {
136 #if PHP_HAVE_BUILTIN_CPU_INIT
137 __builtin_cpu_init();
138 #endif
139 return __builtin_cpu_supports("ssse3");
140 }
141
142 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_sse41()143 static zend_always_inline int zend_cpu_supports_sse41() {
144 #if PHP_HAVE_BUILTIN_CPU_INIT
145 __builtin_cpu_init();
146 #endif
147 return __builtin_cpu_supports("sse4.1");
148 }
149
150 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_sse42()151 static zend_always_inline int zend_cpu_supports_sse42() {
152 #if PHP_HAVE_BUILTIN_CPU_INIT
153 __builtin_cpu_init();
154 #endif
155 return __builtin_cpu_supports("sse4.2");
156 }
157
158 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_avx()159 static zend_always_inline int zend_cpu_supports_avx() {
160 #if PHP_HAVE_BUILTIN_CPU_INIT
161 __builtin_cpu_init();
162 #endif
163 return __builtin_cpu_supports("avx");
164 }
165
166 ZEND_NO_SANITIZE_ADDRESS
zend_cpu_supports_avx2()167 static zend_always_inline int zend_cpu_supports_avx2() {
168 #if PHP_HAVE_BUILTIN_CPU_INIT
169 __builtin_cpu_init();
170 #endif
171 return __builtin_cpu_supports("avx2");
172 }
173 #else
174
zend_cpu_supports_sse2()175 static zend_always_inline int zend_cpu_supports_sse2() {
176 return zend_cpu_supports(ZEND_CPU_FEATURE_SSE2);
177 }
178
zend_cpu_supports_sse3()179 static zend_always_inline int zend_cpu_supports_sse3() {
180 return zend_cpu_supports(ZEND_CPU_FEATURE_SSE3);
181 }
182
zend_cpu_supports_ssse3()183 static zend_always_inline int zend_cpu_supports_ssse3() {
184 return zend_cpu_supports(ZEND_CPU_FEATURE_SSSE3);
185 }
186
zend_cpu_supports_sse41()187 static zend_always_inline int zend_cpu_supports_sse41() {
188 return zend_cpu_supports(ZEND_CPU_FEATURE_SSE41);
189 }
190
zend_cpu_supports_sse42()191 static zend_always_inline int zend_cpu_supports_sse42() {
192 return zend_cpu_supports(ZEND_CPU_FEATURE_SSE42);
193 }
194
zend_cpu_supports_avx()195 static zend_always_inline int zend_cpu_supports_avx() {
196 return zend_cpu_supports(ZEND_CPU_FEATURE_AVX);
197 }
198
zend_cpu_supports_avx2()199 static zend_always_inline int zend_cpu_supports_avx2() {
200 return zend_cpu_supports(ZEND_CPU_FEATURE_AVX2);
201 }
202
203 #endif
204
205 #endif
206
207 /*
208 * Local variables:
209 * tab-width: 4
210 * c-basic-offset: 4
211 * indent-tabs-mode: t
212 * End:
213 */
214