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.
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.
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]
22 * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
23 * Use is subject to license terms.
26 #ifndef _SYS_MACHTHREAD_H
27 #define _SYS_MACHTHREAD_H
29 #pragma ident "%Z%%M% %I% %E% SMI"
32 #include <sys/sun4asi.h>
33 #include <sys/machasi.h>
34 #include <sys/bitmap.h>
35 #include <sys/opl_olympus_regs.h>
43 #define THREAD_REG %g7 /* pointer to current thread data */
46 * Get the processor implementation from the version register.
48 #define GET_CPU_IMPL(out) \
57 * Returns cpu id in r.
58 * On Starfire, this is read from the Port Controller's Port ID
59 * register in local space.
61 * Need to load the 64 bit address of the PC's PortID reg
62 * using only one register. Kludge the 41 bits address constant to
63 * be 32bits by shifting it 12 bits to the right first.
65 #define LOCAL_PC_PORTID_ADDR_SRL12 0x1FFF4000
66 #define PC_PORT_ID 0xD0
68 #define CPU_INDEX(r, scr) \
70 andn scr, PSTATE_IE | PSTATE_AM, r; \
72 set LOCAL_PC_PORTID_ADDR_SRL12, r; \
74 or r, PC_PORT_ID, r; \
80 * For OPL platform, we get CPU_INDEX from ASI_EIDR.
82 #define CPU_INDEX(r, scr) \
83 ldxa [%g0]ASI_EIDR, r; \
90 * UPA supports up to 32 devices while Safari supports up to
91 * 1024 devices (utilizing the SSM protocol). Based upon the
92 * value of NCPU, a 5- or 10-bit mask will be needed for
93 * extracting the cpu id.
96 #define CPU_MASK 0x3ff
99 #endif /* NCPU > 32 */
103 * Returns cpu id in r.
104 * For UPA based systems, the cpu id corresponds to the mid field in
105 * the UPA config register. For Safari based machines, the cpu id
106 * corresponds to the aid field in the Safari config register.
108 * XXX - scr reg is not used here.
110 #define CPU_INDEX(r, scr) \
111 ldxa [%g0]ASI_UPA_CONFIG, r; \
115 #endif /* _STARFIRE */
118 * Given a cpu id extract the appropriate word
119 * in the cpuset mask for this cpu id.
121 #if CPUSET_SIZE > CLONGSIZE
122 #define CPU_INDEXTOSET(base, index, scr) \
123 srl index, BT_ULSHIFT, scr; \
124 and index, BT_ULMASK, index; \
125 sll scr, CLONGSHIFT, scr; \
128 #define CPU_INDEXTOSET(base, index, scr)
129 #endif /* CPUSET_SIZE */
133 * Assembly macro to find address of the current CPU.
134 * Used when coming in from a user trap - cannot use THREAD_REG.
135 * Args are destination register and one scratch register.
137 #define CPU_ADDR(reg, scr) \
139 CPU_INDEX(scr, reg); \
140 sll scr, CPTRSHIFT, scr; \
144 #define CINT64SHIFT 3
147 * Assembly macro to find the physical address of the current CPU.
148 * All memory references using VA must be limited to nucleus
149 * memory to avoid any MMU side effect.
151 #define CPU_PADDR(reg, scr) \
153 CPU_INDEX(scr, reg); \
154 sll scr, CINT64SHIFT, scr; \
161 * If a high level trap handler decides to call sys_trap() to execute some
162 * base level code, context and other registers must be set to proper
163 * values to run kernel. This is true for most part of the kernel, except
164 * for user_rtt, a substantial part of which is executed with registers
165 * ready to run user code. The following macro may be used to detect this
166 * condition and handle it. Please note that, in general, we can't restart
167 * arbitrary piece of code running at tl > 0; user_rtt is a special case
168 * that can be handled.
177 * scr1, scr2 - destroyed
178 * normal %g5 and %g6 - destroyed
182 #define RESET_USER_RTT_REGS(scr1, scr2, label) \
184 * do nothing if %tl != 2. this an attempt to stop this \
185 * piece of code from executing more than once before going \
186 * back to TL=0. more specifically, the changes we are doing \
187 * to %wstate, %canrestore and %otherwin can't be done more \
188 * than once before going to TL=0. note that it is okay to \
189 * execute this more than once if we restart at user_rtt and \
190 * come back from there. \
194 bne,a,pn %xcc, label; \
197 * read tstate[2].%tpc. do nothing if it is not \
198 * between rtt_ctx_start and rtt_ctx_end. \
201 set rtt_ctx_end, scr2; \
203 bgu,a,pt %xcc, label; \
205 set rtt_ctx_start, scr2; \
207 blu,a,pt %xcc, label; \
210 * pickup tstate[2].cwp \
212 rdpr %tstate, scr1; \
213 and scr1, TSTATE_CWP, scr1; \
215 * set tstate[1].cwp to tstate[2].cwp. fudge \
216 * tstate[1].tpc and tstate[1].tnpc to restart \
220 set TSTATE_KERN | TSTATE_IE, scr2; \
221 or scr1, scr2, scr2; \
222 wrpr %g0, scr2, %tstate; \
223 set user_rtt, scr1; \
224 wrpr %g0, scr1, %tpc; \
226 wrpr %g0, scr1, %tnpc; \
234 rdpr %wstate, scr1; \
235 sllx scr1, WSTATE_SHIFT, scr1; \
236 wrpr scr1, WSTATE_K64, %wstate; \
238 * setup window registers \
239 * %cleanwin <-- nwin - 1 \
240 * %otherwin <-- %canrestore \
241 * %canrestore <-- 0 \
243 sethi %hi(nwin_minus_one), scr1; \
244 ld [scr1 + %lo(nwin_minus_one)], scr1; \
245 wrpr %g0, scr1, %cleanwin; \
246 rdpr %canrestore, scr1; \
247 wrpr %g0, scr1, %otherwin; \
248 wrpr %g0, 0, %canrestore; \
250 * set THREAD_REG, as we have restored user \
251 * registers in user_rtt. we trash %g5 and %g6 \
254 rdpr %pstate, scr1; \
255 wrpr scr1, PSTATE_AG, %pstate; \
257 * using normal globals now \
259 CPU_ADDR(%g5, %g6); \
260 ldn [%g5 + CPU_THREAD], %g6; \
261 mov %g6, THREAD_REG; \
263 wrpr %g5, PSTATE_AG, %pstate; \
265 * back to alternate globals. \
266 * set PCONTEXT to run kernel. \
267 * A demap of I/DTLB is required if the nucleus bits differ \
268 * from kcontextreg. \
270 mov MMU_PCONTEXT, scr1; \
271 sethi %hi(kcontextreg), scr2; \
272 ldx [scr2 + %lo(kcontextreg)], scr2; \
273 ldxa [scr1]ASI_MMU_CTX, scr1; \
274 xor scr2, scr1, scr1; \
275 srlx scr1, CTXREG_NEXT_SHIFT, scr1; \
277 * If N_pgsz0/1 changed, need to demap. \
279 brz scr1, label/**/_0; \
281 mov DEMAP_ALL_TYPE, scr1; \
282 stxa %g0, [scr1]ASI_DTLB_DEMAP; \
283 stxa %g0, [scr1]ASI_ITLB_DEMAP; \
285 mov MMU_PCONTEXT, scr1; \
286 stxa scr2, [scr1]ASI_MMU_CTX; \
287 sethi %hi(FLUSH_ADDR), scr1; \
296 #endif /* _SYS_MACHTHREAD_H */