@@ -89,7 +89,8 @@ tests-common = $(TEST_DIR)/vmexit.$(exe) $(TEST_DIR)/tsc.$(exe) \
$(TEST_DIR)/eventinj.$(exe) \
$(TEST_DIR)/smap.$(exe) \
$(TEST_DIR)/umip.$(exe) \
- $(TEST_DIR)/smm_int_window.$(exe)
+ $(TEST_DIR)/smm_int_window.$(exe) \
+ $(TEST_DIR)/ipi_stress.$(exe)
# The following test cases are disabled when building EFI tests because they
# use absolute addresses in their inline assembly code, which cannot compile
new file mode 100644
@@ -0,0 +1,167 @@
+#include "libcflat.h"
+#include "smp.h"
+#include "alloc.h"
+#include "apic.h"
+#include "processor.h"
+#include "isr.h"
+#include "asm/barrier.h"
+#include "delay.h"
+#include "desc.h"
+#include "msr.h"
+#include "vm.h"
+#include "types.h"
+#include "alloc_page.h"
+#include "vmalloc.h"
+#include "random.h"
+
+u64 num_iterations = 100000;
+float hlt_prob = 0.1;
+volatile bool end_test;
+
+#define APIC_TIMER_PERIOD (1000*1000*1000)
+
+struct cpu_test_state {
+ volatile u64 isr_count;
+ u64 last_isr_count;
+ struct random_state random;
+ int smp_id;
+} *cpu_states;
+
+
+static void ipi_interrupt_handler(isr_regs_t *r)
+{
+ cpu_states[smp_id()].isr_count++;
+ eoi();
+}
+
+static void local_timer_interrupt(isr_regs_t *r)
+{
+ struct cpu_test_state *state = &cpu_states[smp_id()];
+
+ u64 isr_count = state->isr_count;
+ unsigned long diff = isr_count - state->last_isr_count;
+
+ if (!diff) {
+ printf("\n");
+ printf("hang detected!!\n");
+ end_test = true;
+ goto out;
+ }
+
+ printf("made %ld IPIs\n", diff * cpu_count());
+ state->last_isr_count = state->isr_count;
+out:
+ eoi();
+}
+
+static void wait_for_ipi(struct cpu_test_state *state)
+{
+ u64 old_count = state->isr_count;
+ bool use_halt = random_decision(&state->random, hlt_prob);
+
+ do {
+ if (use_halt) {
+ safe_halt();
+ cli();
+ } else
+ sti_nop_cli();
+
+ } while (old_count == state->isr_count);
+
+ assert(state->isr_count == old_count + 1);
+}
+
+
+static void vcpu_init(void *)
+{
+ struct cpu_test_state *state = &cpu_states[smp_id()];
+
+ memset(state, 0, sizeof(*state));
+
+ /* To make it easier to see iteration number in the trace */
+ handle_irq(0x40, ipi_interrupt_handler);
+ handle_irq(0x50, ipi_interrupt_handler);
+
+ state->random = get_prng();
+ state->isr_count = 0;
+ state->smp_id = smp_id();
+}
+
+static void vcpu_code(void *)
+{
+ struct cpu_test_state *state = &cpu_states[smp_id()];
+ int ncpus = cpu_count();
+ u64 i;
+ u8 target_smp_id;
+
+ if (state->smp_id > 0)
+ wait_for_ipi(state);
+
+ target_smp_id = state->smp_id == ncpus - 1 ? 0 : state->smp_id + 1;
+
+ for (i = 0; i < num_iterations && !end_test; i++) {
+ // send IPI to a next vCPU in a circular fashion
+ apic_icr_write(APIC_INT_ASSERT |
+ APIC_DEST_PHYSICAL |
+ APIC_DM_FIXED |
+ (i % 2 ? 0x40 : 0x50),
+ target_smp_id);
+
+ if (i == (num_iterations - 1) && state->smp_id > 0)
+ break;
+
+ // wait for the IPI interrupt chain to come back to us
+ wait_for_ipi(state);
+ }
+}
+
+int main(int argc, void **argv)
+{
+ int cpu, ncpus = cpu_count();
+
+ handle_irq(0xF0, local_timer_interrupt);
+ apic_setup_timer(0xF0, APIC_LVT_TIMER_PERIODIC);
+
+ if (argc > 1) {
+ int hlt_param = atol(argv[1]);
+
+ if (hlt_param == 1)
+ hlt_prob = 100;
+ else if (hlt_param == 0)
+ hlt_prob = 0;
+ }
+
+ if (argc > 2)
+ num_iterations = atol(argv[2]);
+
+ setup_vm();
+ init_prng();
+
+ cpu_states = calloc(ncpus, sizeof(cpu_states[0]));
+
+ printf("found %d cpus\n", ncpus);
+ printf("running for %lld iterations\n",
+ (unsigned long long)num_iterations);
+
+ on_cpus(vcpu_init, NULL);
+
+ apic_start_timer(1000*1000*1000);
+
+ printf("test started, waiting to end...\n");
+
+ on_cpus(vcpu_code, NULL);
+
+ apic_stop_timer();
+ apic_cleanup_timer();
+
+ for (cpu = 0; cpu < ncpus; ++cpu) {
+ u64 result = cpu_states[cpu].isr_count;
+
+ report(result == num_iterations,
+ "Number of IPIs match (%lld)",
+ (unsigned long long)result);
+ }
+
+ free((void *)cpu_states);
+ return report_summary();
+}
@@ -74,6 +74,16 @@ smp = 2
file = smptest.flat
smp = 3
+[ipi_stress]
+file = ipi_stress.flat
+extra_params = -cpu host,-x2apic -global kvm-pit.lost_tick_policy=discard -machine kernel-irqchip=on
+smp = 4
+
+[ipi_stress_x2apic]
+file = ipi_stress.flat
+extra_params = -cpu host,+x2apic -global kvm-pit.lost_tick_policy=discard -machine kernel-irqchip=on
+smp = 4
+
[vmexit_cpuid]
file = vmexit.flat
extra_params = -append 'cpuid'
Adds a test that sends IPIs between vCPUs and detects missing IPIs Signed-off-by: Maxim Levitsky <mlevitsk@redhat.com> --- x86/Makefile.common | 3 +- x86/ipi_stress.c | 167 ++++++++++++++++++++++++++++++++++++++++++++ x86/unittests.cfg | 10 +++ 3 files changed, 179 insertions(+), 1 deletion(-) create mode 100644 x86/ipi_stress.c