VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/bootsectors/bs3kit/bs3-cmn-RegCtxSaveEx.asm@ 106560

Last change on this file since 106560 was 106061, checked in by vboxsync, 4 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.6 KB
Line 
1; $Id: bs3-cmn-RegCtxSaveEx.asm 106061 2024-09-16 14:03:52Z vboxsync $
2;; @file
3; BS3Kit - Bs3RegCtxSaveEx.
4;
5
6;
7; Copyright (C) 2007-2024 Oracle and/or its affiliates.
8;
9; This file is part of VirtualBox base platform packages, as
10; available from https://www.virtualbox.org.
11;
12; This program is free software; you can redistribute it and/or
13; modify it under the terms of the GNU General Public License
14; as published by the Free Software Foundation, in version 3 of the
15; License.
16;
17; This program is distributed in the hope that it will be useful, but
18; WITHOUT ANY WARRANTY; without even the implied warranty of
19; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20; General Public License for more details.
21;
22; You should have received a copy of the GNU General Public License
23; along with this program; if not, see <https://www.gnu.org/licenses>.
24;
25; The contents of this file may alternatively be used under the terms
26; of the Common Development and Distribution License Version 1.0
27; (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28; in the VirtualBox distribution, in which case the provisions of the
29; CDDL are applicable instead of those of the GPL.
30;
31; You may elect to license modified versions of this file under the
32; terms and conditions of either the GPL or the CDDL or both.
33;
34; SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35;
36
37%include "bs3kit-template-header.mac"
38
39
40
41;*********************************************************************************************************************************
42;* External Symbols *
43;*********************************************************************************************************************************
44BS3_EXTERN_DATA16 g_bBs3CurrentMode
45%if ARCH_BITS != 64
46BS3_EXTERN_DATA16 g_uBs3CpuDetected
47%endif
48
49TMPL_BEGIN_TEXT
50BS3_EXTERN_CMN Bs3Panic
51BS3_EXTERN_CMN Bs3RegCtxSave
52BS3_EXTERN_CMN Bs3SwitchTo16Bit
53%if TMPL_BITS != 64
54BS3_EXTERN_CMN Bs3SwitchTo16BitV86
55%endif
56%if TMPL_BITS != 32
57BS3_EXTERN_CMN Bs3SwitchTo32Bit
58%endif
59%if TMPL_BITS != 64
60BS3_EXTERN_CMN Bs3SwitchTo64Bit
61%endif
62%if TMPL_BITS == 16
63BS3_EXTERN_CMN Bs3SelRealModeDataToProtFar16
64BS3_EXTERN_CMN Bs3SelProtFar16DataToRealMode
65BS3_EXTERN_CMN Bs3SelRealModeDataToFlat
66BS3_EXTERN_CMN Bs3SelProtFar16DataToFlat
67%else
68BS3_EXTERN_CMN Bs3SelFlatDataToProtFar16
69%endif
70%if TMPL_BITS == 32
71BS3_EXTERN_CMN Bs3SelFlatDataToRealMode
72%endif
73
74BS3_BEGIN_TEXT16
75%if TMPL_BITS != 16
76extern _Bs3RegCtxSave_c16
77extern _Bs3SwitchTo%[TMPL_BITS]Bit_c16
78%endif
79
80BS3_BEGIN_TEXT32
81%if TMPL_BITS != 32
82extern _Bs3RegCtxSave_c32
83extern _Bs3SwitchTo%[TMPL_BITS]Bit_c32
84%endif
85%if TMPL_BITS == 16
86extern _Bs3SwitchTo16BitV86_c32
87%endif
88
89BS3_BEGIN_TEXT64
90%if TMPL_BITS != 64
91extern _Bs3RegCtxSave_c64
92extern _Bs3SwitchTo%[TMPL_BITS]Bit_c64
93%endif
94
95TMPL_BEGIN_TEXT
96
97
98
99;;
100; Saves the current register context.
101;
102; @param pRegCtx
103; @param bBitMode (8)
104; @param cbExtraStack (16)
105; @uses xAX, xDX, xCX
106;
107BS3_PROC_BEGIN_CMN Bs3RegCtxSaveEx, BS3_PBC_NEAR ; Far stub generated by the makefile/bs3kit.h.
108TONLY16 CPU 8086
109 BS3_CALL_CONV_PROLOG 3
110 push xBP
111 mov xBP, xSP
112%if ARCH_BITS == 64
113 push rcx ; Save pRegCtx
114%endif
115
116 ;
117 ; Get the CPU bitcount part of the current mode.
118 ;
119 mov dl, [BS3_DATA16_WRT(g_bBs3CurrentMode)]
120 and dl, BS3_MODE_CODE_MASK
121%if TMPL_BITS == 16
122 push dx ; bp - 2: previous CPU mode (16-bit)
123%endif
124
125 ;
126 ; Reserve extra stack space. Make sure we've got 20h here in case we
127 ; are saving a 64-bit context.
128 ;
129TONLY16 mov ax, [xBP + xCB + cbCurRetAddr + sCB + xCB]
130TNOT16 movzx eax, word [xBP + xCB + cbCurRetAddr + sCB + xCB]
131%ifdef BS3_STRICT
132 cmp xAX, 4096
133 jb .extra_stack_ok
134 call Bs3Panic
135.extra_stack_ok:
136%endif
137 cmp xAX, 20h
138 jae .at_least_20h_extra_stack
139 add xAX, 20h
140.at_least_20h_extra_stack:
141 sub xSP, xAX
142
143 ;
144 ; Are we just saving the mode we're already in?
145 ;
146 mov al, [xBP + xCB + cbCurRetAddr + sCB]
147 and al, BS3_MODE_CODE_MASK
148 cmp dl, al
149 jne .not_the_same_mode
150
151%if TMPL_BITS == 16
152 push word [xBP + xCB + cbCurRetAddr + 2]
153 push word [xBP + xCB + cbCurRetAddr]
154%elif TMPL_BITS == 32
155 push dword [xBP + xCB + cbCurRetAddr]
156%endif
157 call Bs3RegCtxSave ; 64-bit: rcx is untouched thus far.
158
159
160 ;
161 ; Return - no need to pop xAX and xDX as the last two
162 ; operations preserves all registers.
163 ;
164.return:
165 mov xSP, xBP
166 pop xBP
167 BS3_CALL_CONV_EPILOG 3
168 BS3_HYBRID_RET
169
170 ;
171 ; Turns out we have to do switch to a different bitcount before saving.
172 ;
173.not_the_same_mode:
174 cmp al, BS3_MODE_CODE_16
175 je .code_16
176
177TONLY16 CPU 386
178%if TMPL_BITS != 32
179 cmp al, BS3_MODE_CODE_32
180 je .code_32
181%endif
182%if TMPL_BITS != 64
183 cmp al, BS3_MODE_CODE_V86
184 je .code_v86
185 cmp al, BS3_MODE_CODE_64
186 jne .bad_input_mode
187 jmp .code_64
188%endif
189
190 ; Bad input (al=input, dl=current).
191.bad_input_mode:
192 call Bs3Panic
193
194
195 ;
196 ; Save a 16-bit context.
197 ;
198 ; Convert pRegCtx to 16:16 protected mode and make sure we're in the
199 ; 16-bit code segment.
200 ;
201.code_16:
202%if TMPL_BITS == 16
203 %ifdef BS3_STRICT
204 cmp dl, BS3_MODE_CODE_V86
205 jne .bad_input_mode
206 %endif
207 push word [xBP + xCB + cbCurRetAddr + 2]
208 push word [xBP + xCB + cbCurRetAddr]
209 call Bs3SelRealModeDataToProtFar16
210 add sp, 4h
211 push dx ; Parameter #0 for _Bs3RegCtxSave_c16
212 push ax
213%else
214 %if TMPL_BITS == 32
215 push dword [xBP + xCB + cbCurRetAddr]
216 %endif
217 call Bs3SelFlatDataToProtFar16 ; 64-bit: BS3_CALL not needed, ecx not touched thus far.
218 mov [xSP], eax ; Parameter #0 for _Bs3RegCtxSave_c16
219 jmp .code_16_safe_segment
220 BS3_BEGIN_TEXT16
221 BS3_SET_BITS TMPL_BITS
222.code_16_safe_segment:
223%endif
224 call Bs3SwitchTo16Bit
225 BS3_SET_BITS 16
226
227 call _Bs3RegCtxSave_c16
228
229%if TMPL_BITS == 16
230 call _Bs3SwitchTo16BitV86_c16
231%else
232 call _Bs3SwitchTo%[TMPL_BITS]Bit_c16
233%endif
234 BS3_SET_BITS TMPL_BITS
235 jmp .supplement_and_return
236 TMPL_BEGIN_TEXT
237
238TONLY16 CPU 386
239
240
241%if TMPL_BITS != 64
242 ;
243 ; Save a v8086 context.
244 ;
245.code_v86:
246 %if TMPL_BITS == 16
247 %ifdef BS3_STRICT
248 cmp dl, BS3_MODE_CODE_16
249 jne .bad_input_mode
250 %endif
251 push word [xBP + xCB + cbCurRetAddr + 2]
252 push word [xBP + xCB + cbCurRetAddr]
253 call Bs3SelProtFar16DataToRealMode
254 add sp, 4h
255 push dx ; Parameter #0 for _Bs3RegCtxSave_c16
256 push ax
257 %else
258 push dword [xBP + xCB + cbCurRetAddr]
259 call Bs3SelFlatDataToRealMode
260 mov [xSP], eax ; Parameter #0 for _Bs3RegCtxSave_c16
261 jmp .code_v86_safe_segment
262 BS3_BEGIN_TEXT16
263 BS3_SET_BITS TMPL_BITS
264.code_v86_safe_segment:
265 %endif
266 call Bs3SwitchTo16BitV86
267 BS3_SET_BITS 16
268
269 call _Bs3RegCtxSave_c16
270
271 call _Bs3SwitchTo%[TMPL_BITS]Bit_c16
272 BS3_SET_BITS TMPL_BITS
273 jmp .supplement_and_return
274TMPL_BEGIN_TEXT
275%endif
276
277
278%if TMPL_BITS != 32
279 ;
280 ; Save a 32-bit context.
281 ;
282.code_32:
283 %if TMPL_BITS == 16
284 push word [xBP + xCB + cbCurRetAddr + 2]
285 push word [xBP + xCB + cbCurRetAddr]
286 test dl, BS3_MODE_CODE_V86
287 jnz .code_32_from_v86
288 call Bs3SelProtFar16DataToFlat
289 jmp .code_32_flat_ptr
290.code_32_from_v86:
291 call Bs3SelRealModeDataToFlat
292.code_32_flat_ptr:
293 add sp, 4h
294 push dx ; Parameter #0 for _Bs3RegCtxSave_c32
295 push ax
296 %else
297 mov [rsp], ecx ; Parameter #0 for _Bs3RegCtxSave_c16
298 %endif
299 call Bs3SwitchTo32Bit
300 BS3_SET_BITS 32
301
302 call _Bs3RegCtxSave_c32
303
304 %if TMPL_BITS == 16
305 cmp byte [bp - 2], BS3_MODE_CODE_V86
306 je .code_32_back_to_v86
307 call _Bs3SwitchTo16Bit_c32
308 BS3_SET_BITS TMPL_BITS
309 jmp .supplement_and_return
310.code_32_back_to_v86:
311 BS3_SET_BITS 32
312 call _Bs3SwitchTo16BitV86_c32
313 BS3_SET_BITS TMPL_BITS
314 jmp .return
315 %else
316 call _Bs3SwitchTo64Bit_c32
317 BS3_SET_BITS TMPL_BITS
318 jmp .supplement_and_return
319 %endif
320%endif
321
322
323%if TMPL_BITS != 64
324 ;
325 ; Save a 64-bit context.
326 ;
327 CPU x86-64
328.code_64:
329 %if TMPL_BITS == 16
330 %ifdef BS3_STRICT
331 cmp dl, BS3_MODE_CODE_16
332 jne .bad_input_mode
333 %endif
334 push word [xBP + xCB + cbCurRetAddr + 2]
335 push word [xBP + xCB + cbCurRetAddr]
336 call Bs3SelProtFar16DataToFlat
337 add sp, 4h
338 mov cx, dx ; Parameter #0 for _Bs3RegCtxSave_c64
339 shl ecx, 16
340 mov cx, ax
341 %else
342 mov ecx, [xBP + xCB + cbCurRetAddr] ; Parameter #0 for _Bs3RegCtxSave_c64
343 %endif
344 call Bs3SwitchTo64Bit ; (preserves all 32-bit GPRs)
345 BS3_SET_BITS 64
346
347 call _Bs3RegCtxSave_c64 ; No BS3_CALL as rcx is already ready.
348
349 call _Bs3SwitchTo%[TMPL_BITS]Bit_c64
350 BS3_SET_BITS TMPL_BITS
351 jmp .return
352%endif
353
354
355 ;
356 ; Supplement the state out of the current context and then return.
357 ;
358.supplement_and_return:
359%if ARCH_BITS == 16
360 CPU 8086
361 ; Skip 286 and older. Also make 101% sure we not in real mode or v8086 mode.
362 cmp byte [BS3_DATA16_WRT(g_uBs3CpuDetected)], BS3CPU_80386
363 jb .return ; Just skip if 286 or older.
364 test byte [BS3_DATA16_WRT(g_bBs3CurrentMode)], BS3_MODE_CODE_V86
365 jnz .return
366 cmp byte [BS3_DATA16_WRT(g_bBs3CurrentMode)], BS3_MODE_RM
367 jne .return ; paranoia
368 CPU 386
369%endif
370
371 ; Load the context pointer into a suitable register.
372%if ARCH_BITS == 64
373 %define pRegCtx rcx
374 mov rcx, [xBP - xCB]
375%elif ARCH_BITS == 32
376 %define pRegCtx ecx
377 mov ecx, [xBP + xCB + cbCurRetAddr]
378%else
379 %define pRegCtx es:bx
380 push es
381 push bx
382 les bx, [xBP + xCB + cbCurRetAddr]
383%endif
384%if ARCH_BITS == 64
385 ; If we're in 64-bit mode we can capture and restore the high bits.
386 test byte [pRegCtx + BS3REGCTX.fbFlags], BS3REG_CTX_F_NO_AMD64
387 jz .supplemented_64bit_registers
388 mov [pRegCtx + BS3REGCTX.r8], r8
389 mov [pRegCtx + BS3REGCTX.r9], r9
390 mov [pRegCtx + BS3REGCTX.r10], r10
391 mov [pRegCtx + BS3REGCTX.r11], r11
392 mov [pRegCtx + BS3REGCTX.r12], r12
393 mov [pRegCtx + BS3REGCTX.r13], r13
394 mov [pRegCtx + BS3REGCTX.r14], r14
395 mov [pRegCtx + BS3REGCTX.r15], r15
396 shr rax, 32
397 mov [pRegCtx + BS3REGCTX.rax + 4], eax
398 mov rax, rbx
399 shr rax, 32
400 mov [pRegCtx + BS3REGCTX.rbx + 4], eax
401 mov rax, rcx
402 shr rax, 32
403 mov [pRegCtx + BS3REGCTX.rcx + 4], eax
404 mov rax, rdx
405 shr rax, 32
406 mov [pRegCtx + BS3REGCTX.rdx + 4], eax
407 mov rax, rsp
408 shr rax, 32
409 mov [pRegCtx + BS3REGCTX.rsp + 4], eax
410 mov rax, rbp
411 shr rax, 32
412 mov [pRegCtx + BS3REGCTX.rbp + 4], eax
413 mov rax, rsi
414 shr rax, 32
415 mov [pRegCtx + BS3REGCTX.rsi + 4], eax
416 mov rax, rdi
417 shr rax, 32
418 mov [pRegCtx + BS3REGCTX.rdi + 4], eax
419 and byte [pRegCtx + BS3REGCTX.fbFlags], ~BS3REG_CTX_F_NO_AMD64
420.supplemented_64bit_registers:
421%endif
422 ; The rest requires ring-0 (at least during restore).
423 mov ax, ss
424 test ax, 3
425 jnz .done_supplementing
426
427 ; Do control registers.
428 test byte [pRegCtx + BS3REGCTX.fbFlags], BS3REG_CTX_F_NO_CR2_CR3 | BS3REG_CTX_F_NO_CR0_IS_MSW | BS3REG_CTX_F_NO_CR4
429 jz .supplemented_control_registers
430 mov sAX, cr0
431 mov [pRegCtx + BS3REGCTX.cr0], sAX
432 mov sAX, cr2
433 mov [pRegCtx + BS3REGCTX.cr2], sAX
434 mov sAX, cr3
435 mov [pRegCtx + BS3REGCTX.cr3], sAX
436 and byte [pRegCtx + BS3REGCTX.fbFlags], ~(BS3REG_CTX_F_NO_CR2_CR3 | BS3REG_CTX_F_NO_CR0_IS_MSW)
437
438%if ARCH_BITS != 64
439 test byte [1 + BS3_DATA16_WRT(g_uBs3CpuDetected)], (BS3CPU_F_CPUID >> 8)
440 jz .supplemented_control_registers
441%endif
442 mov sAX, cr4
443 mov [pRegCtx + BS3REGCTX.cr4], sAX
444 and byte [pRegCtx + BS3REGCTX.fbFlags], ~BS3REG_CTX_F_NO_CR4
445.supplemented_control_registers:
446
447 ; Supply tr and ldtr if necessary
448 test byte [pRegCtx + BS3REGCTX.fbFlags], BS3REG_CTX_F_NO_TR_LDTR
449 jz .done_supplementing
450 str [pRegCtx + BS3REGCTX.tr]
451 sldt [pRegCtx + BS3REGCTX.ldtr]
452 and byte [pRegCtx + BS3REGCTX.fbFlags], ~BS3REG_CTX_F_NO_TR_LDTR
453
454.done_supplementing:
455TONLY16 pop bx
456TONLY16 pop es
457 jmp .return
458%undef pRegCtx
459BS3_PROC_END_CMN Bs3RegCtxSaveEx
460
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette