xref: /freebsd/sys/contrib/openzfs/module/zcommon/simd_stat.c (revision 53a2e2635ab2d17bed1de7b4e0d782dd23ceb6ea)
1 // SPDX-License-Identifier: CDDL-1.0
2 /*
3  * CDDL HEADER START
4  *
5  * The contents of this file are subject to the terms of the
6  * Common Development and Distribution License (the "License").
7  * You may not use this file except in compliance with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or https://opensource.org/licenses/CDDL-1.0.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2024 Google, Inc.  All rights reserved.
24  */
25 #include <sys/zfs_context.h>
26 #include <sys/kstat.h>
27 #include <sys/simd.h>
28 
29 
30 #ifdef _KERNEL
31 #ifdef __linux__
32 #include <linux/simd.h>
33 #endif /* __linux__ */
34 kstat_t *simd_stat_kstat;
35 #endif /* _KERNEL */
36 
37 #ifdef _KERNEL
38 /* Sometimes, we don't define these at all. */
39 #ifndef HAVE_KERNEL_FPU
40 #define	HAVE_KERNEL_FPU (0)
41 #endif
42 #ifndef HAVE_KERNEL_NEON
43 #define	HAVE_KERNEL_NEON (0)
44 #endif
45 #ifndef HAVE_KERNEL_FPU_INTERNAL
46 #define	HAVE_KERNEL_FPU_INTERNAL (0)
47 #endif
48 #ifndef HAVE_UNDERSCORE_KERNEL_FPU
49 #define	HAVE_UNDERSCORE_KERNEL_FPU (0)
50 #endif
51 
52 #define	SIMD_STAT_PRINT(s, feat, val) \
53 	kmem_scnprintf(s + off, MAX(4095-off, 0), "%-16s\t%1d\n", feat, (val))
54 
55 static int
simd_stat_kstat_data(char * buf,size_t size,void * data)56 simd_stat_kstat_data(char *buf, size_t size, void *data)
57 {
58 	(void) data;
59 
60 	static char simd_stat_kstat_payload[4096] = {0};
61 	static int off = 0;
62 #ifdef __linux__
63 	if (off == 0) {
64 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
65 		    "kfpu_allowed", kfpu_allowed());
66 #if defined(__x86_64__) || defined(__i386__)
67 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
68 		    "kfpu", HAVE_KERNEL_FPU);
69 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
70 		    "kfpu_internal", HAVE_KERNEL_FPU_INTERNAL);
71 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
72 		    "__kernel_fpu", HAVE_UNDERSCORE_KERNEL_FPU);
73 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
74 		    "sse", zfs_sse_available());
75 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
76 		    "sse2", zfs_sse2_available());
77 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
78 		    "sse3", zfs_sse3_available());
79 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
80 		    "ssse3", zfs_ssse3_available());
81 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
82 		    "sse41", zfs_sse4_1_available());
83 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
84 		    "sse42", zfs_sse4_2_available());
85 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
86 		    "avx", zfs_avx_available());
87 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
88 		    "avx2", zfs_avx2_available());
89 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
90 		    "avx512f", zfs_avx512f_available());
91 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
92 		    "avx512cd", zfs_avx512cd_available());
93 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
94 		    "avx512er", zfs_avx512er_available());
95 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
96 		    "avx512pf", zfs_avx512pf_available());
97 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
98 		    "avx512bw", zfs_avx512bw_available());
99 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
100 		    "avx512dq", zfs_avx512dq_available());
101 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
102 		    "avx512vl", zfs_avx512vl_available());
103 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
104 		    "avx512ifma", zfs_avx512ifma_available());
105 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
106 		    "avx512vbmi", zfs_avx512vbmi_available());
107 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
108 		    "ymm", __ymm_enabled());
109 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
110 		    "zmm", __zmm_enabled());
111 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
112 		    "bmi1", zfs_bmi1_available());
113 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
114 		    "bmi2", zfs_bmi2_available());
115 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
116 		    "aes", zfs_aes_available());
117 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
118 		    "pclmulqdq", zfs_pclmulqdq_available());
119 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
120 		    "movbe", zfs_movbe_available());
121 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
122 		    "vaes", zfs_vaes_available());
123 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
124 		    "vpclmulqdq", zfs_vpclmulqdq_available());
125 
126 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
127 		    "osxsave", boot_cpu_has(X86_FEATURE_OSXSAVE));
128 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
129 		    "xsaves", static_cpu_has(X86_FEATURE_XSAVES));
130 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
131 		    "xsaveopt", static_cpu_has(X86_FEATURE_XSAVEOPT));
132 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
133 		    "xsave", static_cpu_has(X86_FEATURE_XSAVE));
134 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
135 		    "fxsr", static_cpu_has(X86_FEATURE_FXSR));
136 #endif /* __x86__ */
137 #if defined(__arm__) || defined(__aarch64__)
138 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
139 		    "kernel_neon", HAVE_KERNEL_NEON);
140 #if defined(CONFIG_KERNEL_MODE_NEON)
141 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
142 		    "kernel_mode_neon", CONFIG_KERNEL_MODE_NEON);
143 #endif /* CONFIG_KERNEL_MODE_NEON */
144 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
145 		    "neon", zfs_neon_available());
146 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
147 		    "sha256", zfs_sha256_available());
148 #if defined(__aarch64__)
149 		/*
150 		 * This technically can exist on 32b ARM but we don't
151 		 * define hooks to check for it and I didn't want to
152 		 * learn enough ARM ASM to add one.
153 		 */
154 		off += SIMD_STAT_PRINT(simd_stat_kstat_payload,
155 		    "sha512", zfs_sha512_available());
156 #endif /* __aarch64__ */
157 #endif /* __arm__ */
158 		/* We want to short-circuit this on unsupported platforms. */
159 		off += 1;
160 	}
161 
162 	kmem_scnprintf(buf, MIN(off, size), "%s", simd_stat_kstat_payload);
163 #endif /* __linux__ */
164 	return (0);
165 }
166 #endif /* _KERNEL */
167 
168 void
simd_stat_init(void)169 simd_stat_init(void)
170 {
171 	static boolean_t simd_stat_initialized = B_FALSE;
172 
173 	if (!simd_stat_initialized) {
174 #if defined(_KERNEL)
175 		/* Install kstats for all implementations */
176 		simd_stat_kstat = kstat_create("zfs", 0, "simd", "misc",
177 		    KSTAT_TYPE_RAW, 0, KSTAT_FLAG_VIRTUAL);
178 
179 
180 		if (simd_stat_kstat != NULL) {
181 			simd_stat_kstat->ks_data = (void*)(uintptr_t)1;
182 			simd_stat_kstat->ks_ndata = 1;
183 			simd_stat_kstat->ks_flags |= KSTAT_FLAG_NO_HEADERS;
184 			kstat_set_raw_ops(simd_stat_kstat,
185 			    NULL,
186 			    simd_stat_kstat_data,
187 			    NULL);
188 			kstat_install(simd_stat_kstat);
189 		}
190 #endif /* _KERNEL */
191 	}
192 	/* Finish initialization */
193 	simd_stat_initialized = B_TRUE;
194 }
195 
196 void
simd_stat_fini(void)197 simd_stat_fini(void)
198 {
199 #if defined(_KERNEL)
200 	if (simd_stat_kstat != NULL) {
201 		kstat_delete(simd_stat_kstat);
202 		simd_stat_kstat = NULL;
203 	}
204 #endif
205 }
206 
207 #ifdef _KERNEL
208 EXPORT_SYMBOL(simd_stat_init);
209 EXPORT_SYMBOL(simd_stat_fini);
210 #endif
211