2012-04-29 22:20:34 +04:00
|
|
|
/*
|
|
|
|
* x86 misc helpers
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003 Fabrice Bellard
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
2020-10-23 15:28:01 +03:00
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2012-04-29 22:20:34 +04:00
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2016-01-26 21:17:03 +03:00
|
|
|
#include "qemu/osdep.h"
|
2012-04-29 22:20:34 +04:00
|
|
|
#include "cpu.h"
|
2014-04-08 09:31:41 +04:00
|
|
|
#include "exec/helper-proto.h"
|
2016-03-15 15:18:37 +03:00
|
|
|
#include "exec/exec-all.h"
|
2020-12-12 18:55:14 +03:00
|
|
|
#include "helper-tcg.h"
|
2012-04-30 00:35:48 +04:00
|
|
|
|
2020-12-12 18:55:15 +03:00
|
|
|
/*
|
|
|
|
* NOTE: the translator must set DisasContext.cc_op to CC_OP_EFLAGS
|
|
|
|
* after generating a call to a helper that uses this.
|
|
|
|
*/
|
|
|
|
void cpu_load_eflags(CPUX86State *env, int eflags, int update_mask)
|
|
|
|
{
|
|
|
|
CC_SRC = eflags & (CC_O | CC_S | CC_Z | CC_A | CC_P | CC_C);
|
|
|
|
CC_OP = CC_OP_EFLAGS;
|
|
|
|
env->df = 1 - (2 * ((eflags >> 10) & 1));
|
|
|
|
env->eflags = (env->eflags & ~update_mask) |
|
|
|
|
(eflags & update_mask) | 0x2;
|
|
|
|
}
|
|
|
|
|
2012-04-29 22:42:47 +04:00
|
|
|
void helper_into(CPUX86State *env, int next_eip_addend)
|
2012-04-29 22:20:34 +04:00
|
|
|
{
|
|
|
|
int eflags;
|
|
|
|
|
2012-04-29 16:45:34 +04:00
|
|
|
eflags = cpu_cc_compute_all(env, CC_OP);
|
2012-04-29 22:20:34 +04:00
|
|
|
if (eflags & CC_O) {
|
|
|
|
raise_interrupt(env, EXCP04_INTO, 1, 0, next_eip_addend);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-29 22:42:47 +04:00
|
|
|
void helper_cpuid(CPUX86State *env)
|
2012-04-29 22:20:34 +04:00
|
|
|
{
|
|
|
|
uint32_t eax, ebx, ecx, edx;
|
|
|
|
|
2017-02-16 14:30:05 +03:00
|
|
|
cpu_svm_check_intercept_param(env, SVM_EXIT_CPUID, 0, GETPC());
|
2012-04-29 22:20:34 +04:00
|
|
|
|
2013-05-28 12:21:10 +04:00
|
|
|
cpu_x86_cpuid(env, (uint32_t)env->regs[R_EAX], (uint32_t)env->regs[R_ECX],
|
|
|
|
&eax, &ebx, &ecx, &edx);
|
2013-05-28 12:20:59 +04:00
|
|
|
env->regs[R_EAX] = eax;
|
2013-05-28 12:21:00 +04:00
|
|
|
env->regs[R_EBX] = ebx;
|
2013-05-28 12:21:01 +04:00
|
|
|
env->regs[R_ECX] = ecx;
|
2013-05-28 12:21:02 +04:00
|
|
|
env->regs[R_EDX] = edx;
|
2012-04-29 22:20:34 +04:00
|
|
|
}
|
|
|
|
|
2012-04-29 22:42:47 +04:00
|
|
|
void helper_rdtsc(CPUX86State *env)
|
2012-04-29 22:20:34 +04:00
|
|
|
{
|
|
|
|
uint64_t val;
|
|
|
|
|
|
|
|
if ((env->cr[4] & CR4_TSD_MASK) && ((env->hflags & HF_CPL_MASK) != 0)) {
|
2015-07-10 12:57:41 +03:00
|
|
|
raise_exception_ra(env, EXCP0D_GPF, GETPC());
|
2012-04-29 22:20:34 +04:00
|
|
|
}
|
2017-02-16 14:30:05 +03:00
|
|
|
cpu_svm_check_intercept_param(env, SVM_EXIT_RDTSC, 0, GETPC());
|
2012-04-29 22:20:34 +04:00
|
|
|
|
|
|
|
val = cpu_get_tsc(env) + env->tsc_offset;
|
2013-05-28 12:20:59 +04:00
|
|
|
env->regs[R_EAX] = (uint32_t)(val);
|
2013-05-28 12:21:02 +04:00
|
|
|
env->regs[R_EDX] = (uint32_t)(val >> 32);
|
2012-04-29 22:20:34 +04:00
|
|
|
}
|
|
|
|
|
2012-04-29 22:42:47 +04:00
|
|
|
void helper_rdtscp(CPUX86State *env)
|
2012-04-29 22:20:34 +04:00
|
|
|
{
|
2012-04-29 22:42:47 +04:00
|
|
|
helper_rdtsc(env);
|
2013-05-28 12:21:01 +04:00
|
|
|
env->regs[R_ECX] = (uint32_t)(env->tsc_aux);
|
2012-04-29 22:20:34 +04:00
|
|
|
}
|
|
|
|
|
2021-05-14 18:13:26 +03:00
|
|
|
void QEMU_NORETURN helper_rdpmc(CPUX86State *env)
|
2012-04-29 22:20:34 +04:00
|
|
|
{
|
2021-02-25 08:47:57 +03:00
|
|
|
if (((env->cr[4] & CR4_PCE_MASK) == 0 ) &&
|
|
|
|
((env->hflags & HF_CPL_MASK) != 0)) {
|
2015-07-10 12:57:41 +03:00
|
|
|
raise_exception_ra(env, EXCP0D_GPF, GETPC());
|
2012-04-29 22:20:34 +04:00
|
|
|
}
|
2017-02-16 14:30:05 +03:00
|
|
|
cpu_svm_check_intercept_param(env, SVM_EXIT_RDPMC, 0, GETPC());
|
2012-04-29 22:20:34 +04:00
|
|
|
|
|
|
|
/* currently unimplemented */
|
|
|
|
qemu_log_mask(LOG_UNIMP, "x86: unimplemented rdpmc\n");
|
|
|
|
raise_exception_err(env, EXCP06_ILLOP, 0);
|
|
|
|
}
|
|
|
|
|
2021-05-14 18:13:33 +03:00
|
|
|
void QEMU_NORETURN do_pause(CPUX86State *env)
|
target-i386: yield to another VCPU on PAUSE
After commit b1bbfe7 (aio / timers: On timer modification, qemu_notify
or aio_notify, 2013-08-21) FreeBSD guests report a huge slowdown.
The problem shows up as soon as FreeBSD turns out its periodic (~1 ms)
tick, but the timers are only the trigger for a pre-existing problem.
Before the offending patch, setting a timer did a timer_settime system call.
After, setting the timer exits the event loop (which uses poll) and
reenters it with a new deadline. This does not cause any slowdown; the
difference is between one system call (timer_settime and a signal
delivery (SIGALRM) before the patch, and two system calls afterwards
(write to a pipe or eventfd + calling poll again when re-entering the
event loop).
Unfortunately, the exit/enter causes the main loop to grab the iothread
lock, which in turns kicks the VCPU thread out of execution. This
causes TCG to execute the next VCPU in its round-robin scheduling of
VCPUS. When the second VCPU is mostly unused, FreeBSD runs a "pause"
instruction in its idle loop which only burns cycles without any
progress. As soon as the timer tick expires, the first VCPU runs
the interrupt handler but very soon it sets it again---and QEMU
then goes back doing nothing in the second VCPU.
The fix is to make the pause instruction do "cpu_loop_exit".
Cc: Richard Henderson <rth@twiddle.net>
Reported-by: Luigi Rizzo <rizzo@iet.unipi.it>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Message-id: 1384948442-24217-1-git-send-email-pbonzini@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-20 15:54:02 +04:00
|
|
|
{
|
2021-05-14 18:13:32 +03:00
|
|
|
CPUState *cs = env_cpu(env);
|
target-i386: yield to another VCPU on PAUSE
After commit b1bbfe7 (aio / timers: On timer modification, qemu_notify
or aio_notify, 2013-08-21) FreeBSD guests report a huge slowdown.
The problem shows up as soon as FreeBSD turns out its periodic (~1 ms)
tick, but the timers are only the trigger for a pre-existing problem.
Before the offending patch, setting a timer did a timer_settime system call.
After, setting the timer exits the event loop (which uses poll) and
reenters it with a new deadline. This does not cause any slowdown; the
difference is between one system call (timer_settime and a signal
delivery (SIGALRM) before the patch, and two system calls afterwards
(write to a pipe or eventfd + calling poll again when re-entering the
event loop).
Unfortunately, the exit/enter causes the main loop to grab the iothread
lock, which in turns kicks the VCPU thread out of execution. This
causes TCG to execute the next VCPU in its round-robin scheduling of
VCPUS. When the second VCPU is mostly unused, FreeBSD runs a "pause"
instruction in its idle loop which only burns cycles without any
progress. As soon as the timer tick expires, the first VCPU runs
the interrupt handler but very soon it sets it again---and QEMU
then goes back doing nothing in the second VCPU.
The fix is to make the pause instruction do "cpu_loop_exit".
Cc: Richard Henderson <rth@twiddle.net>
Reported-by: Luigi Rizzo <rizzo@iet.unipi.it>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Message-id: 1384948442-24217-1-git-send-email-pbonzini@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-20 15:54:02 +04:00
|
|
|
|
|
|
|
/* Just let another CPU run. */
|
2013-08-26 10:31:06 +04:00
|
|
|
cs->exception_index = EXCP_INTERRUPT;
|
2013-08-27 19:52:12 +04:00
|
|
|
cpu_loop_exit(cs);
|
target-i386: yield to another VCPU on PAUSE
After commit b1bbfe7 (aio / timers: On timer modification, qemu_notify
or aio_notify, 2013-08-21) FreeBSD guests report a huge slowdown.
The problem shows up as soon as FreeBSD turns out its periodic (~1 ms)
tick, but the timers are only the trigger for a pre-existing problem.
Before the offending patch, setting a timer did a timer_settime system call.
After, setting the timer exits the event loop (which uses poll) and
reenters it with a new deadline. This does not cause any slowdown; the
difference is between one system call (timer_settime and a signal
delivery (SIGALRM) before the patch, and two system calls afterwards
(write to a pipe or eventfd + calling poll again when re-entering the
event loop).
Unfortunately, the exit/enter causes the main loop to grab the iothread
lock, which in turns kicks the VCPU thread out of execution. This
causes TCG to execute the next VCPU in its round-robin scheduling of
VCPUS. When the second VCPU is mostly unused, FreeBSD runs a "pause"
instruction in its idle loop which only burns cycles without any
progress. As soon as the timer tick expires, the first VCPU runs
the interrupt handler but very soon it sets it again---and QEMU
then goes back doing nothing in the second VCPU.
The fix is to make the pause instruction do "cpu_loop_exit".
Cc: Richard Henderson <rth@twiddle.net>
Reported-by: Luigi Rizzo <rizzo@iet.unipi.it>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Message-id: 1384948442-24217-1-git-send-email-pbonzini@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-20 15:54:02 +04:00
|
|
|
}
|
|
|
|
|
2021-05-14 18:13:26 +03:00
|
|
|
void QEMU_NORETURN helper_pause(CPUX86State *env, int next_eip_addend)
|
target-i386: yield to another VCPU on PAUSE
After commit b1bbfe7 (aio / timers: On timer modification, qemu_notify
or aio_notify, 2013-08-21) FreeBSD guests report a huge slowdown.
The problem shows up as soon as FreeBSD turns out its periodic (~1 ms)
tick, but the timers are only the trigger for a pre-existing problem.
Before the offending patch, setting a timer did a timer_settime system call.
After, setting the timer exits the event loop (which uses poll) and
reenters it with a new deadline. This does not cause any slowdown; the
difference is between one system call (timer_settime and a signal
delivery (SIGALRM) before the patch, and two system calls afterwards
(write to a pipe or eventfd + calling poll again when re-entering the
event loop).
Unfortunately, the exit/enter causes the main loop to grab the iothread
lock, which in turns kicks the VCPU thread out of execution. This
causes TCG to execute the next VCPU in its round-robin scheduling of
VCPUS. When the second VCPU is mostly unused, FreeBSD runs a "pause"
instruction in its idle loop which only burns cycles without any
progress. As soon as the timer tick expires, the first VCPU runs
the interrupt handler but very soon it sets it again---and QEMU
then goes back doing nothing in the second VCPU.
The fix is to make the pause instruction do "cpu_loop_exit".
Cc: Richard Henderson <rth@twiddle.net>
Reported-by: Luigi Rizzo <rizzo@iet.unipi.it>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Message-id: 1384948442-24217-1-git-send-email-pbonzini@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-20 15:54:02 +04:00
|
|
|
{
|
2017-02-16 14:30:05 +03:00
|
|
|
cpu_svm_check_intercept_param(env, SVM_EXIT_PAUSE, 0, GETPC());
|
target-i386: yield to another VCPU on PAUSE
After commit b1bbfe7 (aio / timers: On timer modification, qemu_notify
or aio_notify, 2013-08-21) FreeBSD guests report a huge slowdown.
The problem shows up as soon as FreeBSD turns out its periodic (~1 ms)
tick, but the timers are only the trigger for a pre-existing problem.
Before the offending patch, setting a timer did a timer_settime system call.
After, setting the timer exits the event loop (which uses poll) and
reenters it with a new deadline. This does not cause any slowdown; the
difference is between one system call (timer_settime and a signal
delivery (SIGALRM) before the patch, and two system calls afterwards
(write to a pipe or eventfd + calling poll again when re-entering the
event loop).
Unfortunately, the exit/enter causes the main loop to grab the iothread
lock, which in turns kicks the VCPU thread out of execution. This
causes TCG to execute the next VCPU in its round-robin scheduling of
VCPUS. When the second VCPU is mostly unused, FreeBSD runs a "pause"
instruction in its idle loop which only burns cycles without any
progress. As soon as the timer tick expires, the first VCPU runs
the interrupt handler but very soon it sets it again---and QEMU
then goes back doing nothing in the second VCPU.
The fix is to make the pause instruction do "cpu_loop_exit".
Cc: Richard Henderson <rth@twiddle.net>
Reported-by: Luigi Rizzo <rizzo@iet.unipi.it>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Message-id: 1384948442-24217-1-git-send-email-pbonzini@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-20 15:54:02 +04:00
|
|
|
env->eip += next_eip_addend;
|
|
|
|
|
2021-05-14 18:13:32 +03:00
|
|
|
do_pause(env);
|
target-i386: yield to another VCPU on PAUSE
After commit b1bbfe7 (aio / timers: On timer modification, qemu_notify
or aio_notify, 2013-08-21) FreeBSD guests report a huge slowdown.
The problem shows up as soon as FreeBSD turns out its periodic (~1 ms)
tick, but the timers are only the trigger for a pre-existing problem.
Before the offending patch, setting a timer did a timer_settime system call.
After, setting the timer exits the event loop (which uses poll) and
reenters it with a new deadline. This does not cause any slowdown; the
difference is between one system call (timer_settime and a signal
delivery (SIGALRM) before the patch, and two system calls afterwards
(write to a pipe or eventfd + calling poll again when re-entering the
event loop).
Unfortunately, the exit/enter causes the main loop to grab the iothread
lock, which in turns kicks the VCPU thread out of execution. This
causes TCG to execute the next VCPU in its round-robin scheduling of
VCPUS. When the second VCPU is mostly unused, FreeBSD runs a "pause"
instruction in its idle loop which only burns cycles without any
progress. As soon as the timer tick expires, the first VCPU runs
the interrupt handler but very soon it sets it again---and QEMU
then goes back doing nothing in the second VCPU.
The fix is to make the pause instruction do "cpu_loop_exit".
Cc: Richard Henderson <rth@twiddle.net>
Reported-by: Luigi Rizzo <rizzo@iet.unipi.it>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Richard Henderson <rth@twiddle.net>
Message-id: 1384948442-24217-1-git-send-email-pbonzini@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-20 15:54:02 +04:00
|
|
|
}
|
|
|
|
|
2021-05-14 18:13:26 +03:00
|
|
|
void QEMU_NORETURN helper_debug(CPUX86State *env)
|
2012-04-29 22:20:34 +04:00
|
|
|
{
|
2019-03-23 04:08:48 +03:00
|
|
|
CPUState *cs = env_cpu(env);
|
2013-08-26 10:31:06 +04:00
|
|
|
|
|
|
|
cs->exception_index = EXCP_DEBUG;
|
2013-08-27 19:52:12 +04:00
|
|
|
cpu_loop_exit(cs);
|
2012-04-29 22:20:34 +04:00
|
|
|
}
|
2016-02-09 16:14:28 +03:00
|
|
|
|
|
|
|
uint64_t helper_rdpkru(CPUX86State *env, uint32_t ecx)
|
|
|
|
{
|
|
|
|
if ((env->cr[4] & CR4_PKE_MASK) == 0) {
|
|
|
|
raise_exception_err_ra(env, EXCP06_ILLOP, 0, GETPC());
|
|
|
|
}
|
|
|
|
if (ecx != 0) {
|
|
|
|
raise_exception_err_ra(env, EXCP0D_GPF, 0, GETPC());
|
|
|
|
}
|
|
|
|
|
|
|
|
return env->pkru;
|
|
|
|
}
|
|
|
|
|
|
|
|
void helper_wrpkru(CPUX86State *env, uint32_t ecx, uint64_t val)
|
|
|
|
{
|
2019-03-23 04:08:48 +03:00
|
|
|
CPUState *cs = env_cpu(env);
|
2016-02-09 16:14:28 +03:00
|
|
|
|
|
|
|
if ((env->cr[4] & CR4_PKE_MASK) == 0) {
|
|
|
|
raise_exception_err_ra(env, EXCP06_ILLOP, 0, GETPC());
|
|
|
|
}
|
|
|
|
if (ecx != 0 || (val & 0xFFFFFFFF00000000ull)) {
|
|
|
|
raise_exception_err_ra(env, EXCP0D_GPF, 0, GETPC());
|
|
|
|
}
|
|
|
|
|
|
|
|
env->pkru = val;
|
2016-11-14 17:17:28 +03:00
|
|
|
tlb_flush(cs);
|
2016-02-09 16:14:28 +03:00
|
|
|
}
|