+static ulong iterations = 0;
+module_param(iterations, ulong, S_IRUGO);
+
+struct test_context {
+ struct work_struct work;
+ struct completion completion;
+ struct x86_emulate_ctxt ctxt;
+ struct test_context *next;
+ bool failed;
+ u8 insn[15];
+ bool insn_base_valid;
+ ulong insn_base;
+ struct test_seg {
+ u16 selector;
+ struct desc_struct desc;
+ u32 base3;
+ bool valid;
+ } segs[8];
+ ulong iterations;
+ ulong completed;
+ ulong decoded;
+ ulong emulated;
+ ulong nofault;
+ ulong failures;
+};
+
+static u64 random64(void)
+{
+ return random32() | ((u64)random32()<< 32);
+}
+
+static ulong randlong(void)
+{
+ if (sizeof(ulong) == sizeof(u32))
+ return random32();
+ else
+ return random64();
+}
+
+static struct test_context *to_test(struct x86_emulate_ctxt *ctxt)
+{
+ return container_of(ctxt, struct test_context, ctxt);
+}
+
+static void fail(struct x86_emulate_ctxt *ctxt, const char *msg, ...)
+ __attribute__((format(printf, 2, 3)));
+
+static void fail(struct x86_emulate_ctxt *ctxt, const char *msg, ...)
+{
+ va_list args;
+ char s[200];
+
+ va_start(args, msg);
+ vsnprintf(s, sizeof(s), msg, args);
+ va_end(args);
+ printk("emulator test failure: %s\n", s);
+ to_test(ctxt)->failed = true;
+}
+
+static int test_fill_exception(struct x86_exception *ex)
+{
+ if (random32() % 4 == 0) {
+ if (ex) {
+ ex->vector = random32();
+ ex->error_code_valid = random32();
+ ex->error_code = random32();
+ ex->nested_page_fault = random32();
+ ex->address = random64();
+ }
+ return X86EMUL_PROPAGATE_FAULT;
+ }
+ return X86EMUL_CONTINUE;
+}
+
+static int rand_error(void)
+{
+ switch (random32() % 8) {
+ case 0: return X86EMUL_UNHANDLEABLE;
+ case 1: return X86EMUL_IO_NEEDED;
+ default: return X86EMUL_CONTINUE;
+ }
+}
+
+static int test_read(struct x86_emulate_ctxt *ctxt,
+ unsigned long addr, void *val,
+ unsigned int bytes,
+ struct x86_exception *fault)
+{
+ unsigned i;
+
+ if (bytes> 32 || bytes == 0)
+ fail(ctxt, "read %x bytes", bytes);
+
+ for (i = 0; i< bytes; ++i)
+ *(u8 *)(val + i) = random32();
+
+ return test_fill_exception(fault);
+}
+
+static int test_write(struct x86_emulate_ctxt *ctxt,
+ unsigned long addr, const void *val, unsigned int bytes,
+ struct x86_exception *fault)
+{
+ if (bytes> 32 || bytes == 0)
+ fail(ctxt, "write %x bytes", bytes);
+
+ return test_fill_exception(fault);
+}
+
+static int test_fetch(struct x86_emulate_ctxt *ctxt,
+ unsigned long addr, void *val,
+ unsigned int bytes,
+ struct x86_exception *fault)
+{
+ struct test_context *test = to_test(ctxt);
+
+ if (bytes> 15 || bytes == 0)
+ fail(ctxt, "fetch %x bytes", bytes);
+
+ if (!test->insn_base_valid) {
+ test->insn_base_valid = true;
+ test->insn_base = addr;
+ }
+ addr -= test->insn_base;
+ if (addr>= 15 || addr + bytes> 15)
+ fail(ctxt, "fetch %x from %lx vs %lx",
+ bytes, addr + test->insn_base, test->insn_base);
+ else
+ memcpy(val, test->insn + addr, bytes);
+
+ return test_fill_exception(fault);
+}
+
+static int test_cmpxchg(struct x86_emulate_ctxt *ctxt,
+ unsigned long addr,
+ const void *old,
+ const void *new,
+ unsigned int bytes,
+ struct x86_exception *fault)
+{
+ if (bytes> 16 || bytes == 0 || hweight32(bytes) != 1)
+ fail(ctxt, "cmpxchg %x bytes", bytes);
+
+ return test_fill_exception(fault);
+}
+
+static void test_invlpg(struct x86_emulate_ctxt *ctxt, ulong addr)
+{
+}
+
+static int test_pio_in(struct x86_emulate_ctxt *ctxt,
+ int size, unsigned short port, void *val,
+ unsigned int count)
+{
+ if ((size != 1&& size != 2&& size != 4)
+ || (count == 0 || count * size> 4096))
+ fail(ctxt, "pio_in_emulated: size %x count %x\n", size, count);
+
+ return rand_error();
+}
+
+static int test_pio_out(struct x86_emulate_ctxt *ctxt,
+ int size, unsigned short port, const void *val,
+ unsigned int count)
+{
+ if ((size != 1&& size != 2&& size != 4)
+ || (count == 0 || count * size> 4096))
+ fail(ctxt, "pio_out_emulated: size %x count %x\n", size, count);
+
+ return rand_error();
+}
+
+static bool test_get_segment(struct x86_emulate_ctxt *ctxt, u16 *selector,
+ struct desc_struct *desc, u32 *base3, int seg)
+{
+ struct test_context *test = to_test(ctxt);
+ struct test_seg *s =&test->segs[seg];
+
+ if (seg< 0 || seg> 7)
+ fail(ctxt, "bad segment %d\n", seg);
+
+ if (!s->valid) {
+ s->valid = true;
+ s->selector = random32();
+ s->desc.a = random32();
+ s->desc.b = random32();
+ s->base3 = random32();
+ }
+
+ *selector = s->selector;
+ desc->a = s->desc.a;
+ desc->b = s->desc.b;
+ if (base3)
+ *base3 = s->base3;
+
+ return random32()& 1;
+}
+
+static void test_set_segment(struct x86_emulate_ctxt *ctxt, u16 selector,
+ struct desc_struct *desc, u32 base3, int seg)
+{
+ if (seg< 0 || seg> 5)
+ fail(ctxt, "bad segment %d\n", seg);
+}
+
+static ulong segment_base(struct x86_emulate_ctxt *ctxt,
+ struct desc_struct *d, u32 base3)
+{
+ unsigned long v;
+
+ v = get_desc_base(d);
+ if (ctxt->mode == X86EMUL_MODE_PROT64
+ && d->s == 0&& (d->type == 2 || d->type == 9 || d->type == 11))
+ v |= (u64)base3<< 32;
+ return v;
+}
+
+static unsigned long test_get_cached_segment_base(struct x86_emulate_ctxt
*ctxt,
+ int seg)
+{
+ u16 selector;
+ struct desc_struct desc;
+ u32 base3;
+
+ test_get_segment(ctxt,&selector,&desc,&base3, seg);
+ return segment_base(ctxt,&desc, base3);
+}
+
+static void test_get_desc_table(struct x86_emulate_ctxt *ctxt,
+ struct desc_ptr *dt)
+{
+ dt->size = random32();
+ dt->address = randlong();
+}
+
+static void test_set_desc_table(struct x86_emulate_ctxt *ctxt,
+ struct desc_ptr *dt)
+{
+}
+
+static bool valid_cr[] = {
+ [0] = true, [2] = true, [3] = true, [4] = true, [8] = true,
+};
+
+static void check_cr(struct x86_emulate_ctxt *ctxt, int cr)
+{
+ if (cr< 0 || cr> ARRAY_SIZE(valid_cr) || !valid_cr[cr])
+ fail(ctxt, "bad cr %d\n", cr);
+}
+
+static ulong test_get_cr(struct x86_emulate_ctxt *ctxt, int cr)
+{
+ check_cr(ctxt, cr);
+ return randlong();
+}
+
+static int test_set_cr(struct x86_emulate_ctxt *ctxt, int cr, ulong val)
+{
+ check_cr(ctxt, cr);
+ return random32()& 1;
+}
+
+static int test_cpl(struct x86_emulate_ctxt *ctxt)
+{
+ return random32()& 3;
+}
+
+static void check_dr(struct x86_emulate_ctxt *ctxt, int dr)
+{
+ if (dr< 0 || dr> 7)
+ fail(ctxt, "bad dr %d\n", dr);
+}
+
+static int test_get_dr(struct x86_emulate_ctxt *ctxt, int dr, ulong *dest)
+{
+ check_dr(ctxt, dr);
+ *dest = randlong();
+ return random32()& 1;
+}
+
+static int test_set_dr(struct x86_emulate_ctxt *ctxt, int dr, ulong value)
+{
+ check_dr(ctxt, dr);
+ return random32()& 1;
+}
+
+static int test_set_msr(struct x86_emulate_ctxt *ctxt, u32 msr_index, u64 data)
+{
+ return random32()& 1;
+}
+
+static int test_get_msr(struct x86_emulate_ctxt *ctxt, u32 msr_index,
+ u64 *pdata)
+{
+ *pdata = random64();
+ return random32()& 1;
+}
+
+static void test_halt(struct x86_emulate_ctxt *ctxt)
+{
+}
+
+static void test_wbinvd(struct x86_emulate_ctxt *ctxt)
+{
+}
+
+static int test_fix_hypercall(struct x86_emulate_ctxt *ctxt)
+{
+ return rand_error();
+}
+
+static void test_get_fpu(struct x86_emulate_ctxt *ctxt)
+{
+ kernel_fpu_begin();
+ /* FIXME: randomize state? */
+}
+
+static void test_put_fpu(struct x86_emulate_ctxt *ctxt)
+{
+ kernel_fpu_end();
+}
+
+static int test_intercept(struct x86_emulate_ctxt *ctxt,
+ struct x86_instruction_info *info,
+ enum x86_intercept_stage stage)
+{
+ return X86EMUL_CONTINUE;
+}
+
+static struct x86_emulate_ops test_ops = {
+ .read_std = test_read,
+ .write_std = test_write,
+ .fetch = test_fetch,
+ .read_emulated = test_read,
+ .write_emulated = test_write,
+ .cmpxchg_emulated = test_cmpxchg,
+ .invlpg = test_invlpg,
+ .pio_in_emulated = test_pio_in,
+ .pio_out_emulated = test_pio_out,
+ .get_segment = test_get_segment,
+ .set_segment = test_set_segment,
+ .get_cached_segment_base = test_get_cached_segment_base,
+ .get_gdt = test_get_desc_table,
+ .get_idt = test_get_desc_table,
+ .set_gdt = test_set_desc_table,
+ .set_idt = test_set_desc_table,
+ .get_cr = test_get_cr,
+ .set_cr = test_set_cr,
+ .cpl = test_cpl,
+ .get_dr = test_get_dr,
+ .set_dr = test_set_dr,
+ .set_msr = test_set_msr,
+ .get_msr = test_get_msr,
+ .halt = test_halt,
+ .wbinvd = test_wbinvd,
+ .fix_hypercall = test_fix_hypercall,
+ .get_fpu = test_get_fpu,
+ .put_fpu = test_put_fpu,
+ .intercept = test_intercept,
+};
+
+static int modes[] = {
+ X86EMUL_MODE_REAL,
+ X86EMUL_MODE_VM86,
+ X86EMUL_MODE_PROT16,
+ X86EMUL_MODE_PROT32,
+ X86EMUL_MODE_PROT64,
+};
+
+static int test_emulator_one(struct test_context *test)
+{
+ struct x86_emulate_ctxt *ctxt =&test->ctxt;
+ unsigned i;
+ int r;
+
+ test->failed = false;
+ i = 0;
+ if (random32()& 1)
+ test->insn[i++] = 0x0f;
+ for (; i< 15; ++i)
+ test->insn[i++] = random32();
+ test->insn_base_valid = false;
+ ctxt->ops =&test_ops;
+ ctxt->eflags = randlong();
+ ctxt->eip = randlong();
+ ctxt->mode = modes[random32() % ARRAY_SIZE(modes)];
+ ctxt->guest_mode = random32() % 16 == 0;
+ ctxt->perm_ok = random32() % 16 == 0;
+ ctxt->only_vendor_specific_insn = random32() % 64 == 0;
+ memset(&ctxt->twobyte, 0,
+ (void *)&ctxt->regs - (void *)&ctxt->twobyte);
+ for (i = 0; i< NR_VCPU_REGS; ++i)
+ ctxt->regs[i] = randlong();
+ r = x86_decode_insn(ctxt, NULL, 0);
+ if (r == EMULATION_OK) {
+ ++test->decoded;
+ r = x86_emulate_insn(ctxt);
+ if (r == EMULATION_OK) {
+ ++test->emulated;
+ if (!ctxt->have_exception)
+ ++test->nofault;
+ }
+ }
+
+ ++test->completed;
+
+ return test->failed ? -EINVAL : 0;
+}
+
+static const char *regnames[] = {
+ [VCPU_REGS_RAX] = "rax",
+ [VCPU_REGS_RBX] = "rbx",
+ [VCPU_REGS_RCX] = "rcx",
+ [VCPU_REGS_RDX] = "rdx",
+ [VCPU_REGS_RSI] = "rsi",
+ [VCPU_REGS_RDI] = "rdi",
+ [VCPU_REGS_RSP] = "rsp",
+ [VCPU_REGS_RBP] = "rbp",
+ [VCPU_REGS_R8] = "r8",
+ [VCPU_REGS_R9] = "r9",
+ [VCPU_REGS_R10] = "r10",
+ [VCPU_REGS_R11] = "r11",
+ [VCPU_REGS_R12] = "r12",
+ [VCPU_REGS_R13] = "r13",
+ [VCPU_REGS_R14] = "r14",
+ [VCPU_REGS_R15] = "r15",
+ [VCPU_REGS_RIP] = "rip",
+};
+
+static void dump_test_context(struct test_context *test)
+{
+ unsigned i;
+
+ printk("instruction: %02x %02x %02x %02x %02x %02x %02x %02x"
+ " %02x %02x %02x %02x %02x %02x %02x\n",
+ test->insn[0], test->insn[1], test->insn[2], test->insn[3],
+ test->insn[4], test->insn[5], test->insn[6], test->insn[7],
+ test->insn[8], test->insn[9], test->insn[10], test->insn[11],
+ test->insn[12], test->insn[13], test->insn[14]);
+ for (i = 0; i< NR_VCPU_REGS; ++i)
+ printk(" %s: %016llx\n", regnames[i], (u64)test->ctxt.regs[i]);
+}
+
+static void test_emulator_thread(struct work_struct *work)
+{
+ int i, ret;
+ struct test_context *test
+ = container_of(work, struct test_context, work);
+
+ for (i = 0, ret = 0; i< test->iterations&& ret == 0; ++i) {
+ ret = test_emulator_one(test);
+ cond_resched();
+ }
+
+ if (ret) {
+ ++test->failures;
+ printk("test failure in instruction %i\n", i);
+ dump_test_context(test);
+ }
+
+ complete(&test->completion);
+}
+
+static __init int test_emulator(void)
+{
+ int r, cpu, remain;
+ struct test_context *test = NULL, *tmp;
+ ulong completed = 0, decoded = 0, emulated = 0, nofault = 0;
+ ulong failures = 0;
+
+ if (!iterations)
+ return 0;
+
+ pr_info("starting emulator test\n");
+ remain = num_online_cpus();
+ for_each_online_cpu(cpu) {
+ tmp = kzalloc(sizeof(*tmp), GFP_KERNEL);
+ r = -ENOMEM;
+ if (!tmp)
+ break;
+ tmp->next = test;
+ test = tmp;
+ test->iterations = iterations / remain--;
+ iterations -= test->iterations;
+ INIT_WORK(&test->work, test_emulator_thread);
+ init_completion(&test->completion);
+ schedule_work_on(cpu,&test->work);
+ }
+ while (test) {
+ wait_for_completion(&test->completion);
+ completed += test->completed;
+ decoded += test->decoded;
+ emulated += test->emulated;
+ nofault += test->nofault;
+ failures += test->failures;
+ tmp = test;
+ test = test->next;
+ kfree(tmp);
+ }
+ pr_info("emulator fuzz test results\n");
+ pr_info(" instructions: %12ld\n", completed);
+ pr_info(" decoded: %12ld\n", decoded);
+ pr_info(" emulated: %12ld\n", emulated);
+ pr_info(" nofault: %12ld\n", nofault);
+ pr_info(" failures: %12ld\n", failures);
+ if (failures || remain)
+ pr_err("emulator test: FAIL\n");
+ else
+ pr_info("emulator test: PASS\n");
+ return 0;
+}
+
+module_init(test_emulator)