xref: /illumos-gate/usr/src/uts/intel/asm/cpu.h (revision b77a2dc4455ca028e52fdf96385a530a2d168316)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 /*
26  * Copyright 2019 Joyent, Inc.
27  */
28 
29 #ifndef _ASM_CPU_H
30 #define	_ASM_CPU_H
31 
32 #include <sys/ccompile.h>
33 
34 #ifdef	__cplusplus
35 extern "C" {
36 #endif
37 
38 #if !defined(__lint) && defined(__GNUC__)
39 
40 #if defined(__i386) || defined(__amd64)
41 
42 extern __GNU_INLINE void
43 ht_pause(void)
44 {
45 	__asm__ __volatile__(
46 	    "pause");
47 }
48 
49 /*
50  * prefetch 64 bytes
51  *
52  * prefetch is an SSE extension which is not supported on
53  * older 32-bit processors, so define this as a no-op for now
54  */
55 
56 extern __GNU_INLINE void
57 prefetch_read_many(void *addr)
58 {
59 #if defined(__amd64)
60 	__asm__(
61 	    "prefetcht0 (%0);"
62 	    "prefetcht0 32(%0);"
63 	    : /* no output */
64 	    : "r" (addr));
65 #endif	/* __amd64 */
66 }
67 
68 extern __GNU_INLINE void
69 prefetch_read_once(void *addr)
70 {
71 #if defined(__amd64)
72 	__asm__(
73 	    "prefetchnta (%0);"
74 	    "prefetchnta 32(%0);"
75 	    : /* no output */
76 	    : "r" (addr));
77 #endif	/* __amd64 */
78 }
79 
80 extern __GNU_INLINE void
81 prefetch_write_many(void *addr)
82 {
83 #if defined(__amd64)
84 	__asm__(
85 	    "prefetcht0 (%0);"
86 	    "prefetcht0 32(%0);"
87 	    : /* no output */
88 	    : "r" (addr));
89 #endif	/* __amd64 */
90 }
91 
92 extern __GNU_INLINE void
93 prefetch_write_once(void *addr)
94 {
95 #if defined(__amd64)
96 	__asm__(
97 	    "prefetcht0 (%0);"
98 	    "prefetcht0 32(%0);"
99 	    : /* no output */
100 	    : "r" (addr));
101 #endif	/* __amd64 */
102 }
103 
104 #if !defined(__xpv)
105 
106 extern __GNU_INLINE void
107 cli(void)
108 {
109 	__asm__ __volatile__(
110 	    "cli" : : : "memory");
111 }
112 
113 extern __GNU_INLINE void
114 sti(void)
115 {
116 	__asm__ __volatile__(
117 	    "sti");
118 }
119 
120 /*
121  * Any newer callers of halt need to make sure that they consider calling
122  * x86_md_clear() before calling this to deal with any potential issues with
123  * MDS. Because this version of hlt is also used in panic context, we do not
124  * unconditionally call x86_md_clear() here and require callers to do so.
125  */
126 extern __GNU_INLINE void
127 i86_halt(void)
128 {
129 	__asm__ __volatile__(
130 	    "sti; hlt");
131 }
132 
133 #endif /* !__xpv */
134 
135 #endif	/* __i386 || defined(__amd64) */
136 
137 #if defined(__amd64)
138 
139 extern __GNU_INLINE void
140 __set_ds(selector_t value)
141 {
142 	__asm__ __volatile__(
143 	    "movw	%0, %%ds"
144 	    : /* no output */
145 	    : "r" (value));
146 }
147 
148 extern __GNU_INLINE void
149 __set_es(selector_t value)
150 {
151 	__asm__ __volatile__(
152 	    "movw	%0, %%es"
153 	    : /* no output */
154 	    : "r" (value));
155 }
156 
157 extern __GNU_INLINE void
158 __set_fs(selector_t value)
159 {
160 	__asm__ __volatile__(
161 	    "movw	%0, %%fs"
162 	    : /* no output */
163 	    : "r" (value));
164 }
165 
166 extern __GNU_INLINE void
167 __set_gs(selector_t value)
168 {
169 	__asm__ __volatile__(
170 	    "movw	%0, %%gs"
171 	    : /* no output */
172 	    : "r" (value));
173 }
174 
175 #if !defined(__xpv)
176 
177 extern __GNU_INLINE void
178 __swapgs(void)
179 {
180 	__asm__ __volatile__(
181 	    "mfence; swapgs");
182 }
183 
184 #endif /* !__xpv */
185 
186 #endif	/* __amd64 */
187 
188 #endif	/* !__lint && __GNUC__ */
189 
190 #ifdef	__cplusplus
191 }
192 #endif
193 
194 #endif	/* _ASM_CPU_H */
195