Message ID | 20190627052750.31856-2-mrolnik@gmail.com |
---|---|
State | New |
Headers | show |
Series | QEMU AVR 8 bit cores | expand |
On Thu, 27 Jun 2019 08:27:44 +0300 Michael Rolnik <mrolnik@gmail.com> wrote: > From: Sarah Harris <S.E.Harris@kent.ac.uk> > > This includes: > - CPU data structures > - object model classes and functions > - migration functions > - GDB hooks > > Signed-off-by: Michael Rolnik <mrolnik@gmail.com> > --- > gdb-xml/avr-cpu.xml | 49 ++++ > target/avr/cpu-param.h | 37 +++ > target/avr/cpu.c | 599 +++++++++++++++++++++++++++++++++++++++++ > target/avr/cpu.h | 283 +++++++++++++++++++ > target/avr/gdbstub.c | 85 ++++++ > target/avr/machine.c | 123 +++++++++ > 6 files changed, 1176 insertions(+) > create mode 100644 gdb-xml/avr-cpu.xml > create mode 100644 target/avr/cpu-param.h > create mode 100644 target/avr/cpu.c > create mode 100644 target/avr/cpu.h > create mode 100644 target/avr/gdbstub.c > create mode 100644 target/avr/machine.c > > diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml > new file mode 100644 > index 0000000000..c4747f5b40 > --- /dev/null > +++ b/gdb-xml/avr-cpu.xml > @@ -0,0 +1,49 @@ > +<?xml version="1.0"?> > +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. > + > + Copying and distribution of this file, with or without modification, > + are permitted in any medium without royalty provided the copyright > + notice and this notice are preserved. --> > + > +<!-- Register numbers are hard-coded in order to maintain backward > + compatibility with older versions of tools that didn't use xml > + register descriptions. --> > + > +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> > +<feature name="org.gnu.gdb.riscv.cpu"> > + <reg name="r0" bitsize="8" type="int" regnum="0"/> > + <reg name="r1" bitsize="8" type="int"/> > + <reg name="r2" bitsize="8" type="int"/> > + <reg name="r3" bitsize="8" type="int"/> > + <reg name="r4" bitsize="8" type="int"/> > + <reg name="r5" bitsize="8" type="int"/> > + <reg name="r6" bitsize="8" type="int"/> > + <reg name="r7" bitsize="8" type="int"/> > + <reg name="r8" bitsize="8" type="int"/> > + <reg name="r9" bitsize="8" type="int"/> > + <reg name="r10" bitsize="8" type="int"/> > + <reg name="r11" bitsize="8" type="int"/> > + <reg name="r12" bitsize="8" type="int"/> > + <reg name="r13" bitsize="8" type="int"/> > + <reg name="r14" bitsize="8" type="int"/> > + <reg name="r15" bitsize="8" type="int"/> > + <reg name="r16" bitsize="8" type="int"/> > + <reg name="r17" bitsize="8" type="int"/> > + <reg name="r18" bitsize="8" type="int"/> > + <reg name="r19" bitsize="8" type="int"/> > + <reg name="r20" bitsize="8" type="int"/> > + <reg name="r21" bitsize="8" type="int"/> > + <reg name="r22" bitsize="8" type="int"/> > + <reg name="r23" bitsize="8" type="int"/> > + <reg name="r24" bitsize="8" type="int"/> > + <reg name="r25" bitsize="8" type="int"/> > + <reg name="r26" bitsize="8" type="int"/> > + <reg name="r27" bitsize="8" type="int"/> > + <reg name="r28" bitsize="8" type="int"/> > + <reg name="r29" bitsize="8" type="int"/> > + <reg name="r30" bitsize="8" type="int"/> > + <reg name="r31" bitsize="8" type="int"/> > + <reg name="sreg" bitsize="8" type="int"/> > + <reg name="sp" bitsize="8" type="int"/> > + <reg name="pc" bitsize="8" type="int"/> > +</feature> > diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h > new file mode 100644 > index 0000000000..5bbf985726 > --- /dev/null > +++ b/target/avr/cpu-param.h > @@ -0,0 +1,37 @@ > +/* > + * QEMU AVR CPU > + * > + * Copyright (c) 2016 Michael Rolnik > + * > + * 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 > + * version 2.1 of the License, or (at your option) any later version. > + * > + * 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/lgpl-2.1.html> > + */ > + > +#ifndef AVR_CPU_PARAM_H > +#define AVR_CPU_PARAM_H 1 > + > +#define TARGET_LONG_BITS 32 > +/* > + * TARGET_PAGE_BITS cannot be more than 8 bits because > + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, and they > + * should be implemented as a device and not memory > + * 2. SRAM starts at the address 0x0100 > + */ > +#define TARGET_PAGE_BITS 8 > +#define TARGET_PHYS_ADDR_SPACE_BITS 24 > +#define TARGET_VIRT_ADDR_SPACE_BITS 24 > +#define NB_MMU_MODES 2 > + > + > +#endif > diff --git a/target/avr/cpu.c b/target/avr/cpu.c > new file mode 100644 > index 0000000000..142fe54524 > --- /dev/null > +++ b/target/avr/cpu.c > @@ -0,0 +1,599 @@ > +/* > + * QEMU AVR CPU > + * > + * Copyright (c) 2016 Michael Rolnik > + * > + * 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 > + * version 2.1 of the License, or (at your option) any later version. > + * > + * 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/lgpl-2.1.html> > + */ > + > +#include "qemu/osdep.h" > +#include "qemu/qemu-print.h" > +#include "qemu/log.h" > +#include "cpu.h" > +#include "exec/exec-all.h" > +#include "qapi/error.h" > +#include "hw/qdev-properties.h" > +#include "migration/vmstate.h" > + > +static void avr_cpu_set_pc(CPUState *cs, vaddr value) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + > + cpu->env.pc_w = value / 2; /* internally PC points to words */ > +} > + > +static bool avr_cpu_has_work(CPUState *cs) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + CPUAVRState *env = &cpu->env; > + > + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | CPU_INTERRUPT_RESET)) > + && cpu_interrupts_enabled(env); > +} > + > +static void avr_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock *tb) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + CPUAVRState *env = &cpu->env; > + > + env->pc_w = tb->pc / 2; /* internally PC points to words */ > +} > + > +static void avr_cpu_reset(CPUState *cs) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); > + CPUAVRState *env = &cpu->env; > + > + mcc->parent_reset(cs); > + > + env->pc_w = 0; > + env->sregI = 1; > + env->sregC = 0; > + env->sregZ = 0; > + env->sregN = 0; > + env->sregV = 0; > + env->sregS = 0; > + env->sregH = 0; > + env->sregT = 0; > + > + env->rampD = 0; > + env->rampX = 0; > + env->rampY = 0; > + env->rampZ = 0; > + env->eind = 0; > + env->sp = 0; > + > + env->skip = 0; > + > + memset(env->r, 0, sizeof(env->r)); > + > + tlb_flush(cs); > +} > + > +static void avr_cpu_disas_set_info(CPUState *cpu, disassemble_info *info) > +{ > + info->mach = bfd_arch_avr; > + info->print_insn = NULL; > +} > + > +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) > +{ > + CPUState *cs = CPU(dev); > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); > + Error *local_err = NULL; > + > + cpu_exec_realizefn(cs, &local_err); > + if (local_err != NULL) { > + error_propagate(errp, local_err); > + return; > + } > + qemu_init_vcpu(cs); > + cpu_reset(cs); > + > + mcc->parent_realize(dev, errp); > +} > + > +static void avr_cpu_set_int(void *opaque, int irq, int level) > +{ > + AVRCPU *cpu = opaque; > + CPUAVRState *env = &cpu->env; > + CPUState *cs = CPU(cpu); > + > + uint64_t mask = (1ull << irq); > + if (level) { > + env->intsrc |= mask; > + cpu_interrupt(cs, CPU_INTERRUPT_HARD); > + } else { > + env->intsrc &= ~mask; > + if (env->intsrc == 0) { > + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); > + } > + } > +} > + > +static void avr_cpu_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + > + cpu_set_cpustate_pointers(cpu); > + > +#ifndef CONFIG_USER_ONLY > + /* Set the number of interrupts supported by the CPU. */ > + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); > +#endif > +} > + > +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) > +{ > + ObjectClass *oc; > + char *typename; > + char **cpuname; > + > + cpuname = g_strsplit(cpu_model, ",", 1); you do not need this, parse_cpu_option() does it for you. > + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), cpuname[0]); > + oc = object_class_by_name(typename); Also just use cpu_model directly for lookup for example see '[Qemu-devel] [PATCH RESEND v21 06/21] target/rx: CPU definition' and simplify avr_cpu_list_entry() correspondingly. > + g_strfreev(cpuname); > + g_free(typename); > + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || > + object_class_is_abstract(oc)) { > + return NULL; > + } > + return oc; > +} > + > +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + CPUAVRState *env = &cpu->env; > + int i; > + > + qemu_fprintf(f, "\n"); > + qemu_fprintf(f, "PC: %06x\n", env->pc_w); > + qemu_fprintf(f, "SP: %04x\n", env->sp); > + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); > + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); > + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); > + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); > + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); > + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); > + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); > + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); > + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", > + env->sregI ? 'I' : '-', > + env->sregT ? 'T' : '-', > + env->sregH ? 'H' : '-', > + env->sregS ? 'S' : '-', > + env->sregV ? 'V' : '-', > + env->sregN ? '-' : 'N', /* Zf has negative logic */ > + env->sregZ ? 'Z' : '-', > + env->sregC ? 'I' : '-'); > + qemu_fprintf(f, "SKIP: %02x\n", env->skip); > + > + qemu_fprintf(f, "\n"); > + for (i = 0; i < ARRAY_SIZE(env->r); i++) { > + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); > + > + if ((i % 8) == 7) { > + qemu_fprintf(f, "\n"); > + } > + } > + qemu_fprintf(f, "\n"); > +} > + > +static void avr_cpu_class_init(ObjectClass *oc, void *data) > +{ > + DeviceClass *dc = DEVICE_CLASS(oc); > + CPUClass *cc = CPU_CLASS(oc); > + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); > + > + mcc->parent_realize = dc->realize; > + dc->realize = avr_cpu_realizefn; > + > + mcc->parent_reset = cc->reset; > + cc->reset = avr_cpu_reset; > + > + cc->class_by_name = avr_cpu_class_by_name; > + > + cc->has_work = avr_cpu_has_work; > + cc->do_interrupt = avr_cpu_do_interrupt; > + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; > + cc->dump_state = avr_cpu_dump_state; > + cc->set_pc = avr_cpu_set_pc; > +#if !defined(CONFIG_USER_ONLY) > + cc->memory_rw_debug = avr_cpu_memory_rw_debug; > +#endif > +#ifdef CONFIG_USER_ONLY > + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; > +#else > + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; > + cc->vmsd = &vms_avr_cpu; > +#endif > + cc->disas_set_info = avr_cpu_disas_set_info; > + cc->tlb_fill = avr_cpu_tlb_fill; > + cc->tcg_initialize = avr_cpu_tcg_init; > + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; > + cc->gdb_read_register = avr_cpu_gdb_read_register; > + cc->gdb_write_register = avr_cpu_gdb_write_register; > + cc->gdb_num_core_regs = 35; > + cc->gdb_core_xml_file = "avr-cpu.xml"; > +} > + > +static void avr_avr1_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > +} > + > +static void avr_avr2_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > +} > + > +static void avr_avr25_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > +} > + > +static void avr_avr3_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > +} > + > +static void avr_avr31_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > +} > + > +static void avr_avr35_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > +} > + > +static void avr_avr4_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > +} > + > +static void avr_avr5_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > +} > + > +static void avr_avr51_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_ELPMX); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > +} > + > +static void avr_avr6_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > + avr_set_feature(env, AVR_FEATURE_ELPMX); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > +} > + > +static void avr_xmega2_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > + avr_set_feature(env, AVR_FEATURE_RMW); > +} > + > +static void avr_xmega4_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_ELPMX); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > + avr_set_feature(env, AVR_FEATURE_RMW); > +} > + > +static void avr_xmega5_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPD); > + avr_set_feature(env, AVR_FEATURE_RAMPX); > + avr_set_feature(env, AVR_FEATURE_RAMPY); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_ELPMX); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > + avr_set_feature(env, AVR_FEATURE_RMW); > +} > + > +static void avr_xmega6_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > + avr_set_feature(env, AVR_FEATURE_ELPMX); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > + avr_set_feature(env, AVR_FEATURE_RMW); > +} > + > +static void avr_xmega7_initfn(Object *obj) > +{ > + AVRCPU *cpu = AVR_CPU(obj); > + CPUAVRState *env = &cpu->env; > + > + avr_set_feature(env, AVR_FEATURE_LPM); > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > + avr_set_feature(env, AVR_FEATURE_SRAM); > + avr_set_feature(env, AVR_FEATURE_BREAK); > + > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > + avr_set_feature(env, AVR_FEATURE_RAMPD); > + avr_set_feature(env, AVR_FEATURE_RAMPX); > + avr_set_feature(env, AVR_FEATURE_RAMPY); > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > + avr_set_feature(env, AVR_FEATURE_ELPMX); > + avr_set_feature(env, AVR_FEATURE_ELPM); > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > + avr_set_feature(env, AVR_FEATURE_LPMX); > + avr_set_feature(env, AVR_FEATURE_MOVW); > + avr_set_feature(env, AVR_FEATURE_MUL); > + avr_set_feature(env, AVR_FEATURE_RMW); > +} > + > +typedef struct AVRCPUInfo { > + const char *name; > + void (*initfn)(Object *obj); > +} AVRCPUInfo; > + > +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) > +{ > + ObjectClass *class_a = (ObjectClass *)a; > + ObjectClass *class_b = (ObjectClass *)b; > + const char *name_a; > + const char *name_b; > + > + name_a = object_class_get_name(class_a); > + name_b = object_class_get_name(class_b); > + > + return strcmp(name_a, name_b); > +} > + > +static void avr_cpu_list_entry(gpointer data, gpointer user_data) > +{ > + const char *typename = object_class_get_name(OBJECT_CLASS(data)); > + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); > + qemu_printf("%.*s\n", len, typename); > +} > + > +void avr_cpu_list(void) > +{ > + GSList *list; > + list = object_class_get_list(TYPE_AVR_CPU, false); > + list = g_slist_sort(list, avr_cpu_list_compare); > + qemu_printf("Available CPUs:\n"); > + g_slist_foreach(list, avr_cpu_list_entry, NULL); > + g_slist_free(list); > +} > + > +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ > + { \ > + .parent = TYPE_AVR_CPU, \ > + .instance_init = initfn, \ > + .name = AVR_CPU_TYPE_NAME(model), \ > + } > + > +static const TypeInfo avr_cpu_type_info[] = { > + { > + .name = TYPE_AVR_CPU, > + .parent = TYPE_CPU, > + .instance_size = sizeof(AVRCPU), > + .instance_init = avr_cpu_initfn, > + .class_size = sizeof(AVRCPUClass), > + .class_init = avr_cpu_class_init, > + .abstract = true, > + }, > + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), > + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), > + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), > + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), > + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), > + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), > + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), > + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), > + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), > + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), > + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), > + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), > + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), > + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), > + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), > +}; > + > +DEFINE_TYPES(avr_cpu_type_info) > diff --git a/target/avr/cpu.h b/target/avr/cpu.h > new file mode 100644 > index 0000000000..3f9a803193 > --- /dev/null > +++ b/target/avr/cpu.h > @@ -0,0 +1,283 @@ > +/* > + * QEMU AVR CPU > + * > + * Copyright (c) 2016 Michael Rolnik > + * > + * 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 > + * version 2.1 of the License, or (at your option) any later version. > + * > + * 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/lgpl-2.1.html> > + */ > + > +#ifndef CPU_AVR_H > +#define CPU_AVR_H > + > +#include "qemu-common.h" > +#include "qom/cpu.h" > +#include "exec/cpu-defs.h" > +#include "fpu/softfloat.h" > + > +#define TCG_GUEST_DEFAULT_MO 0 > + > +#define TYPE_AVR_CPU "avr-cpu" > + > +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU > +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) > +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU > + > +/* > + * AVR has two memory spaces, data & code. > + * e.g. both have 0 address > + * ST/LD instructions access data space > + * LPM/SPM and instruction fetching access code memory space > + */ > +#define MMU_CODE_IDX 0 > +#define MMU_DATA_IDX 1 > + > +#define EXCP_RESET 1 > +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) > + > +/* Number of CPU registers */ > +#define NO_CPU_REGISTERS 32 > +/* Number of IO registers accessible by ld/st/in/out */ > +#define NO_IO_REGISTERS 64 > + > +/* > + * Offsets of AVR memory regions in host memory space. > + * > + * This is needed because the AVR has separate code and data address > + * spaces that both have start from zero but have to go somewhere in > + * host memory. > + * > + * It's also useful to know where some things are, like the IO registers. > + */ > +/* Flash program memory */ > +#define OFFSET_CODE 0x00000000 > +/* CPU registers, IO registers, and SRAM */ > +#define OFFSET_DATA 0x00800000 > +/* CPU registers specifically, these are mapped at the start of data */ > +#define OFFSET_CPU_REGISTERS OFFSET_DATA > +/* > + * IO registers, including status register, stack pointer, and memory > + * mapped peripherals, mapped just after CPU registers > + */ > +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) > + > +enum avr_features { > + AVR_FEATURE_SRAM, > + > + AVR_FEATURE_1_BYTE_PC, > + AVR_FEATURE_2_BYTE_PC, > + AVR_FEATURE_3_BYTE_PC, > + > + AVR_FEATURE_1_BYTE_SP, > + AVR_FEATURE_2_BYTE_SP, > + > + AVR_FEATURE_BREAK, > + AVR_FEATURE_DES, > + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ > + > + AVR_FEATURE_EIJMP_EICALL, > + AVR_FEATURE_IJMP_ICALL, > + AVR_FEATURE_JMP_CALL, > + > + AVR_FEATURE_ADIW_SBIW, > + > + AVR_FEATURE_SPM, > + AVR_FEATURE_SPMX, > + > + AVR_FEATURE_ELPMX, > + AVR_FEATURE_ELPM, > + AVR_FEATURE_LPMX, > + AVR_FEATURE_LPM, > + > + AVR_FEATURE_MOVW, > + AVR_FEATURE_MUL, > + AVR_FEATURE_RAMPD, > + AVR_FEATURE_RAMPX, > + AVR_FEATURE_RAMPY, > + AVR_FEATURE_RAMPZ, > +}; > + > +typedef struct CPUAVRState CPUAVRState; > + > +struct CPUAVRState { > + uint32_t pc_w; /* 0x003fffff up to 22 bits */ > + > + uint32_t sregC; /* 0x00000001 1 bits */ > + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ > + /* 0=flag set, >0=flag cleared */ > + uint32_t sregN; /* 0x00000001 1 bits */ > + uint32_t sregV; /* 0x00000001 1 bits */ > + uint32_t sregS; /* 0x00000001 1 bits */ > + uint32_t sregH; /* 0x00000001 1 bits */ > + uint32_t sregT; /* 0x00000001 1 bits */ > + uint32_t sregI; /* 0x00000001 1 bits */ > + > + uint32_t rampD; /* 0x00ff0000 8 bits */ > + uint32_t rampX; /* 0x00ff0000 8 bits */ > + uint32_t rampY; /* 0x00ff0000 8 bits */ > + uint32_t rampZ; /* 0x00ff0000 8 bits */ > + uint32_t eind; /* 0x00ff0000 8 bits */ > + > + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ > + uint32_t sp; /* 16 bits */ > + > + uint32_t skip; /* if set skip instruction */ > + > + uint64_t intsrc; /* interrupt sources */ > + bool fullacc; /* CPU/MEM if true MEM only otherwise */ > + > + uint32_t features; > +}; > + > +#define AVR_CPU_CLASS(klass) \ > + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) > +#define AVR_CPU(obj) \ > + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) > +#define AVR_CPU_GET_CLASS(obj) \ > + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) > + > +/** > + * AVRCPUClass: > + * @parent_realize: The parent class' realize handler. > + * @parent_reset: The parent class' reset handler. > + * @vr: Version Register value. > + * > + * A AVR CPU model. > + */ > +typedef struct AVRCPUClass { > + /*< private >*/ > + CPUClass parent_class; > + /*< public >*/ > + DeviceRealize parent_realize; > + void (*parent_reset)(CPUState *cpu); > +} AVRCPUClass; > + > +/** > + * AVRCPU: > + * @env: #CPUAVRState > + * > + * A AVR CPU. > + */ > +typedef struct AVRCPU { > + /*< private >*/ > + CPUState parent_obj; > + /*< public >*/ > + > + CPUNegativeOffsetState neg; > + CPUAVRState env; > +} AVRCPU; > + > +#ifndef CONFIG_USER_ONLY > +extern const struct VMStateDescription vms_avr_cpu; > +#endif > + > +void avr_cpu_do_interrupt(CPUState *cpu); > +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); > +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); > +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg); > +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg); > + > +static inline int avr_feature(CPUAVRState *env, int feature) > +{ > + return (env->features & (1U << feature)) != 0; > +} > + > +static inline void avr_set_feature(CPUAVRState *env, int feature) > +{ > + env->features |= (1U << feature); > +} > + > +#define cpu_list avr_cpu_list > +#define cpu_signal_handler cpu_avr_signal_handler > +#define cpu_mmu_index avr_cpu_mmu_index > + > +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) > +{ > + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; > +} > + > +void avr_cpu_tcg_init(void); > + > +void avr_cpu_list(void); > +int cpu_avr_exec(CPUState *cpu); > +int cpu_avr_signal_handler(int host_signum, void *pinfo, void *puc); > +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int size, > + int rw, int mmu_idx); > +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t *buf, > + int len, bool is_write); > + > +enum { > + TB_FLAGS_FULL_ACCESS = 1, > + TB_FLAGS_SKIP = 2, > +}; > + > +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, target_ulong *pc, > + target_ulong *cs_base, uint32_t *pflags) > +{ > + uint32_t flags = 0; > + > + *pc = env->pc_w * 2; > + *cs_base = 0; > + > + if (env->fullacc) { > + flags |= TB_FLAGS_FULL_ACCESS; > + } > + if (env->skip) { > + flags |= TB_FLAGS_SKIP; > + } > + > + *pflags = flags; > +} > + > +static inline int cpu_interrupts_enabled(CPUAVRState *env) > +{ > + return env->sregI != 0; > +} > + > +static inline uint8_t cpu_get_sreg(CPUAVRState *env) > +{ > + uint8_t sreg; > + sreg = (env->sregC & 0x01) << 0 > + | (env->sregZ == 0 ? 1 : 0) << 1 > + | (env->sregN) << 2 > + | (env->sregV) << 3 > + | (env->sregS) << 4 > + | (env->sregH) << 5 > + | (env->sregT) << 6 > + | (env->sregI) << 7; > + return sreg; > +} > + > +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) > +{ > + env->sregC = (sreg >> 0) & 0x01; > + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; > + env->sregN = (sreg >> 2) & 0x01; > + env->sregV = (sreg >> 3) & 0x01; > + env->sregS = (sreg >> 4) & 0x01; > + env->sregH = (sreg >> 5) & 0x01; > + env->sregT = (sreg >> 6) & 0x01; > + env->sregI = (sreg >> 7) & 0x01; > +} > + > +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, > + MMUAccessType access_type, int mmu_idx, > + bool probe, uintptr_t retaddr); > + > +typedef CPUAVRState CPUArchState; > +typedef AVRCPU ArchCPU; > + > +#include "exec/cpu-all.h" > + > +#endif /* !defined (CPU_AVR_H) */ > diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c > new file mode 100644 > index 0000000000..537dc7226e > --- /dev/null > +++ b/target/avr/gdbstub.c > @@ -0,0 +1,85 @@ > +/* > + * QEMU AVR CPU > + * > + * Copyright (c) 2016 Michael Rolnik > + * > + * 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 > + * version 2.1 of the License, or (at your option) any later version. > + * > + * 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/lgpl-2.1.html> > + */ > + > +#include "qemu/osdep.h" > +#include "qemu-common.h" > +#include "exec/gdbstub.h" > + > +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + CPUAVRState *env = &cpu->env; > + > + /* R */ > + if (n < 32) { > + return gdb_get_reg8(mem_buf, env->r[n]); > + } > + > + /* SREG */ > + if (n == 32) { > + uint8_t sreg = cpu_get_sreg(env); > + > + return gdb_get_reg8(mem_buf, sreg); > + } > + > + /* SP */ > + if (n == 33) { > + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); > + } > + > + /* PC */ > + if (n == 34) { > + return gdb_get_reg32(mem_buf, env->pc_w * 2); > + } > + > + return 0; > +} > + > +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n) > +{ > + AVRCPU *cpu = AVR_CPU(cs); > + CPUAVRState *env = &cpu->env; > + > + /* R */ > + if (n < 32) { > + env->r[n] = *mem_buf; > + return 1; > + } > + > + /* SREG */ > + if (n == 32) { > + cpu_set_sreg(env, *mem_buf); > + return 1; > + } > + > + /* SP */ > + if (n == 33) { > + env->sp = lduw_p(mem_buf); > + return 2; > + } > + > + /* PC */ > + if (n == 34) { > + env->pc_w = ldl_p(mem_buf) / 2; > + return 4; > + } > + > + return 0; > +} > diff --git a/target/avr/machine.c b/target/avr/machine.c > new file mode 100644 > index 0000000000..389b5eb95d > --- /dev/null > +++ b/target/avr/machine.c > @@ -0,0 +1,123 @@ > +/* > + * QEMU AVR CPU > + * > + * Copyright (c) 2016 Michael Rolnik > + * > + * 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 > + * version 2.1 of the License, or (at your option) any later version. > + * > + * 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/lgpl-2.1.html> > + */ > + > +#include "qemu/osdep.h" > +#include "hw/hw.h" > +#include "cpu.h" > +#include "hw/boards.h" > +#include "migration/qemu-file.h" > + > +static int get_sreg(QEMUFile *f, void *opaque, size_t size, > + const VMStateField *field) > +{ > + CPUAVRState *env = opaque; > + uint8_t sreg; > + > + sreg = qemu_get_byte(f); > + cpu_set_sreg(env, sreg); > + return 0; > +} > + > +static int put_sreg( > + QEMUFile *f, void *opaque, size_t size, > + const VMStateField *field, QJSON *vmdesc) > +{ > + CPUAVRState *env = opaque; > + uint8_t sreg = cpu_get_sreg(env); > + > + qemu_put_byte(f, sreg); > + return 0; > +} > + > +static const VMStateInfo vms_sreg = { > + .name = "sreg", > + .get = get_sreg, > + .put = put_sreg, > +}; > + > +static int get_segment( > + QEMUFile *f, void *opaque, size_t size, const VMStateField *field) > +{ > + uint32_t *ramp = opaque; > + uint8_t temp; > + > + temp = qemu_get_byte(f); > + *ramp = ((uint32_t)temp) << 16; > + return 0; > +} > + > +static int put_segment( > + QEMUFile *f, void *opaque, size_t size, > + const VMStateField *field, QJSON *vmdesc) > +{ > + uint32_t *ramp = opaque; > + uint8_t temp = *ramp >> 16; > + > + qemu_put_byte(f, temp); > + return 0; > +} > + > +static const VMStateInfo vms_rampD = { > + .name = "rampD", > + .get = get_segment, > + .put = put_segment, > +}; > +static const VMStateInfo vms_rampX = { > + .name = "rampX", > + .get = get_segment, > + .put = put_segment, > +}; > +static const VMStateInfo vms_rampY = { > + .name = "rampY", > + .get = get_segment, > + .put = put_segment, > +}; > +static const VMStateInfo vms_rampZ = { > + .name = "rampZ", > + .get = get_segment, > + .put = put_segment, > +}; > +static const VMStateInfo vms_eind = { > + .name = "eind", > + .get = get_segment, > + .put = put_segment, > +}; > + > +const VMStateDescription vms_avr_cpu = { > + .name = "cpu", > + .version_id = 0, > + .minimum_version_id = 0, > + .fields = (VMStateField[]) { > + VMSTATE_UINT32(env.pc_w, AVRCPU), > + VMSTATE_UINT32(env.sp, AVRCPU), > + VMSTATE_UINT32(env.skip, AVRCPU), > + > + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), > + > + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), > + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), > + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), > + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), > + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), > + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), > + > + VMSTATE_END_OF_LIST() > + } > +};
Hi Igor, if I run `make check` it fails with qemu-system-rx: unable to find CPU model 'rx62n' or qemu-system-avr: unable to find CPU model 'avr6' On Thu, Jun 27, 2019 at 2:28 PM Igor Mammedov <imammedo@redhat.com> wrote: > On Thu, 27 Jun 2019 08:27:44 +0300 > Michael Rolnik <mrolnik@gmail.com> wrote: > > > From: Sarah Harris <S.E.Harris@kent.ac.uk> > > > > This includes: > > - CPU data structures > > - object model classes and functions > > - migration functions > > - GDB hooks > > > > Signed-off-by: Michael Rolnik <mrolnik@gmail.com> > > --- > > gdb-xml/avr-cpu.xml | 49 ++++ > > target/avr/cpu-param.h | 37 +++ > > target/avr/cpu.c | 599 +++++++++++++++++++++++++++++++++++++++++ > > target/avr/cpu.h | 283 +++++++++++++++++++ > > target/avr/gdbstub.c | 85 ++++++ > > target/avr/machine.c | 123 +++++++++ > > 6 files changed, 1176 insertions(+) > > create mode 100644 gdb-xml/avr-cpu.xml > > create mode 100644 target/avr/cpu-param.h > > create mode 100644 target/avr/cpu.c > > create mode 100644 target/avr/cpu.h > > create mode 100644 target/avr/gdbstub.c > > create mode 100644 target/avr/machine.c > > > > diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml > > new file mode 100644 > > index 0000000000..c4747f5b40 > > --- /dev/null > > +++ b/gdb-xml/avr-cpu.xml > > @@ -0,0 +1,49 @@ > > +<?xml version="1.0"?> > > +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. > > + > > + Copying and distribution of this file, with or without > modification, > > + are permitted in any medium without royalty provided the copyright > > + notice and this notice are preserved. --> > > + > > +<!-- Register numbers are hard-coded in order to maintain backward > > + compatibility with older versions of tools that didn't use xml > > + register descriptions. --> > > + > > +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> > > +<feature name="org.gnu.gdb.riscv.cpu"> > > + <reg name="r0" bitsize="8" type="int" regnum="0"/> > > + <reg name="r1" bitsize="8" type="int"/> > > + <reg name="r2" bitsize="8" type="int"/> > > + <reg name="r3" bitsize="8" type="int"/> > > + <reg name="r4" bitsize="8" type="int"/> > > + <reg name="r5" bitsize="8" type="int"/> > > + <reg name="r6" bitsize="8" type="int"/> > > + <reg name="r7" bitsize="8" type="int"/> > > + <reg name="r8" bitsize="8" type="int"/> > > + <reg name="r9" bitsize="8" type="int"/> > > + <reg name="r10" bitsize="8" type="int"/> > > + <reg name="r11" bitsize="8" type="int"/> > > + <reg name="r12" bitsize="8" type="int"/> > > + <reg name="r13" bitsize="8" type="int"/> > > + <reg name="r14" bitsize="8" type="int"/> > > + <reg name="r15" bitsize="8" type="int"/> > > + <reg name="r16" bitsize="8" type="int"/> > > + <reg name="r17" bitsize="8" type="int"/> > > + <reg name="r18" bitsize="8" type="int"/> > > + <reg name="r19" bitsize="8" type="int"/> > > + <reg name="r20" bitsize="8" type="int"/> > > + <reg name="r21" bitsize="8" type="int"/> > > + <reg name="r22" bitsize="8" type="int"/> > > + <reg name="r23" bitsize="8" type="int"/> > > + <reg name="r24" bitsize="8" type="int"/> > > + <reg name="r25" bitsize="8" type="int"/> > > + <reg name="r26" bitsize="8" type="int"/> > > + <reg name="r27" bitsize="8" type="int"/> > > + <reg name="r28" bitsize="8" type="int"/> > > + <reg name="r29" bitsize="8" type="int"/> > > + <reg name="r30" bitsize="8" type="int"/> > > + <reg name="r31" bitsize="8" type="int"/> > > + <reg name="sreg" bitsize="8" type="int"/> > > + <reg name="sp" bitsize="8" type="int"/> > > + <reg name="pc" bitsize="8" type="int"/> > > +</feature> > > diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h > > new file mode 100644 > > index 0000000000..5bbf985726 > > --- /dev/null > > +++ b/target/avr/cpu-param.h > > @@ -0,0 +1,37 @@ > > +/* > > + * QEMU AVR CPU > > + * > > + * Copyright (c) 2016 Michael Rolnik > > + * > > + * 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 > > + * version 2.1 of the License, or (at your option) any later version. > > + * > > + * 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/lgpl-2.1.html> > > + */ > > + > > +#ifndef AVR_CPU_PARAM_H > > +#define AVR_CPU_PARAM_H 1 > > + > > +#define TARGET_LONG_BITS 32 > > +/* > > + * TARGET_PAGE_BITS cannot be more than 8 bits because > > + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, and > they > > + * should be implemented as a device and not memory > > + * 2. SRAM starts at the address 0x0100 > > + */ > > +#define TARGET_PAGE_BITS 8 > > +#define TARGET_PHYS_ADDR_SPACE_BITS 24 > > +#define TARGET_VIRT_ADDR_SPACE_BITS 24 > > +#define NB_MMU_MODES 2 > > + > > + > > +#endif > > diff --git a/target/avr/cpu.c b/target/avr/cpu.c > > new file mode 100644 > > index 0000000000..142fe54524 > > --- /dev/null > > +++ b/target/avr/cpu.c > > @@ -0,0 +1,599 @@ > > +/* > > + * QEMU AVR CPU > > + * > > + * Copyright (c) 2016 Michael Rolnik > > + * > > + * 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 > > + * version 2.1 of the License, or (at your option) any later version. > > + * > > + * 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/lgpl-2.1.html> > > + */ > > + > > +#include "qemu/osdep.h" > > +#include "qemu/qemu-print.h" > > +#include "qemu/log.h" > > +#include "cpu.h" > > +#include "exec/exec-all.h" > > +#include "qapi/error.h" > > +#include "hw/qdev-properties.h" > > +#include "migration/vmstate.h" > > + > > +static void avr_cpu_set_pc(CPUState *cs, vaddr value) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + > > + cpu->env.pc_w = value / 2; /* internally PC points to words */ > > +} > > + > > +static bool avr_cpu_has_work(CPUState *cs) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + CPUAVRState *env = &cpu->env; > > + > > + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | > CPU_INTERRUPT_RESET)) > > + && cpu_interrupts_enabled(env); > > +} > > + > > +static void avr_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock > *tb) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + CPUAVRState *env = &cpu->env; > > + > > + env->pc_w = tb->pc / 2; /* internally PC points to words */ > > +} > > + > > +static void avr_cpu_reset(CPUState *cs) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); > > + CPUAVRState *env = &cpu->env; > > + > > + mcc->parent_reset(cs); > > + > > + env->pc_w = 0; > > + env->sregI = 1; > > + env->sregC = 0; > > + env->sregZ = 0; > > + env->sregN = 0; > > + env->sregV = 0; > > + env->sregS = 0; > > + env->sregH = 0; > > + env->sregT = 0; > > + > > + env->rampD = 0; > > + env->rampX = 0; > > + env->rampY = 0; > > + env->rampZ = 0; > > + env->eind = 0; > > + env->sp = 0; > > + > > + env->skip = 0; > > + > > + memset(env->r, 0, sizeof(env->r)); > > + > > + tlb_flush(cs); > > +} > > + > > +static void avr_cpu_disas_set_info(CPUState *cpu, disassemble_info > *info) > > +{ > > + info->mach = bfd_arch_avr; > > + info->print_insn = NULL; > > +} > > + > > +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) > > +{ > > + CPUState *cs = CPU(dev); > > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); > > + Error *local_err = NULL; > > + > > + cpu_exec_realizefn(cs, &local_err); > > + if (local_err != NULL) { > > + error_propagate(errp, local_err); > > + return; > > + } > > + qemu_init_vcpu(cs); > > + cpu_reset(cs); > > + > > + mcc->parent_realize(dev, errp); > > +} > > + > > +static void avr_cpu_set_int(void *opaque, int irq, int level) > > +{ > > + AVRCPU *cpu = opaque; > > + CPUAVRState *env = &cpu->env; > > + CPUState *cs = CPU(cpu); > > + > > + uint64_t mask = (1ull << irq); > > + if (level) { > > + env->intsrc |= mask; > > + cpu_interrupt(cs, CPU_INTERRUPT_HARD); > > + } else { > > + env->intsrc &= ~mask; > > + if (env->intsrc == 0) { > > + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); > > + } > > + } > > +} > > + > > +static void avr_cpu_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + > > + cpu_set_cpustate_pointers(cpu); > > + > > +#ifndef CONFIG_USER_ONLY > > + /* Set the number of interrupts supported by the CPU. */ > > + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); > > +#endif > > +} > > + > > +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) > > +{ > > + ObjectClass *oc; > > + char *typename; > > + char **cpuname; > > + > > + cpuname = g_strsplit(cpu_model, ",", 1); > you do not need this, parse_cpu_option() does it for you. > > > + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), cpuname[0]); > > + oc = object_class_by_name(typename); > Also just use cpu_model directly for lookup > for example see '[Qemu-devel] [PATCH RESEND v21 06/21] target/rx: CPU > definition' > and simplify avr_cpu_list_entry() correspondingly. > > > > + g_strfreev(cpuname); > > + g_free(typename); > > + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || > > + object_class_is_abstract(oc)) { > > + return NULL; > > + } > > + return oc; > > +} > > + > > +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + CPUAVRState *env = &cpu->env; > > + int i; > > + > > + qemu_fprintf(f, "\n"); > > + qemu_fprintf(f, "PC: %06x\n", env->pc_w); > > + qemu_fprintf(f, "SP: %04x\n", env->sp); > > + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); > > + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); > > + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); > > + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); > > + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); > > + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); > > + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); > > + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); > > + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", > > + env->sregI ? 'I' : '-', > > + env->sregT ? 'T' : '-', > > + env->sregH ? 'H' : '-', > > + env->sregS ? 'S' : '-', > > + env->sregV ? 'V' : '-', > > + env->sregN ? '-' : 'N', /* Zf has negative > logic */ > > + env->sregZ ? 'Z' : '-', > > + env->sregC ? 'I' : '-'); > > + qemu_fprintf(f, "SKIP: %02x\n", env->skip); > > + > > + qemu_fprintf(f, "\n"); > > + for (i = 0; i < ARRAY_SIZE(env->r); i++) { > > + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); > > + > > + if ((i % 8) == 7) { > > + qemu_fprintf(f, "\n"); > > + } > > + } > > + qemu_fprintf(f, "\n"); > > +} > > + > > +static void avr_cpu_class_init(ObjectClass *oc, void *data) > > +{ > > + DeviceClass *dc = DEVICE_CLASS(oc); > > + CPUClass *cc = CPU_CLASS(oc); > > + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); > > + > > + mcc->parent_realize = dc->realize; > > + dc->realize = avr_cpu_realizefn; > > + > > + mcc->parent_reset = cc->reset; > > + cc->reset = avr_cpu_reset; > > + > > + cc->class_by_name = avr_cpu_class_by_name; > > + > > + cc->has_work = avr_cpu_has_work; > > + cc->do_interrupt = avr_cpu_do_interrupt; > > + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; > > + cc->dump_state = avr_cpu_dump_state; > > + cc->set_pc = avr_cpu_set_pc; > > +#if !defined(CONFIG_USER_ONLY) > > + cc->memory_rw_debug = avr_cpu_memory_rw_debug; > > +#endif > > +#ifdef CONFIG_USER_ONLY > > + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; > > +#else > > + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; > > + cc->vmsd = &vms_avr_cpu; > > +#endif > > + cc->disas_set_info = avr_cpu_disas_set_info; > > + cc->tlb_fill = avr_cpu_tlb_fill; > > + cc->tcg_initialize = avr_cpu_tcg_init; > > + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; > > + cc->gdb_read_register = avr_cpu_gdb_read_register; > > + cc->gdb_write_register = avr_cpu_gdb_write_register; > > + cc->gdb_num_core_regs = 35; > > + cc->gdb_core_xml_file = "avr-cpu.xml"; > > +} > > + > > +static void avr_avr1_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > +} > > + > > +static void avr_avr2_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > +} > > + > > +static void avr_avr25_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > +} > > + > > +static void avr_avr3_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > +} > > + > > +static void avr_avr31_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > +} > > + > > +static void avr_avr35_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > +} > > + > > +static void avr_avr4_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > +} > > + > > +static void avr_avr5_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > +} > > + > > +static void avr_avr51_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > +} > > + > > +static void avr_avr6_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > +} > > + > > +static void avr_xmega2_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > + avr_set_feature(env, AVR_FEATURE_RMW); > > +} > > + > > +static void avr_xmega4_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > + avr_set_feature(env, AVR_FEATURE_RMW); > > +} > > + > > +static void avr_xmega5_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPD); > > + avr_set_feature(env, AVR_FEATURE_RAMPX); > > + avr_set_feature(env, AVR_FEATURE_RAMPY); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > + avr_set_feature(env, AVR_FEATURE_RMW); > > +} > > + > > +static void avr_xmega6_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > + avr_set_feature(env, AVR_FEATURE_RMW); > > +} > > + > > +static void avr_xmega7_initfn(Object *obj) > > +{ > > + AVRCPU *cpu = AVR_CPU(obj); > > + CPUAVRState *env = &cpu->env; > > + > > + avr_set_feature(env, AVR_FEATURE_LPM); > > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > + avr_set_feature(env, AVR_FEATURE_SRAM); > > + avr_set_feature(env, AVR_FEATURE_BREAK); > > + > > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > + avr_set_feature(env, AVR_FEATURE_RAMPD); > > + avr_set_feature(env, AVR_FEATURE_RAMPX); > > + avr_set_feature(env, AVR_FEATURE_RAMPY); > > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > + avr_set_feature(env, AVR_FEATURE_ELPM); > > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > + avr_set_feature(env, AVR_FEATURE_LPMX); > > + avr_set_feature(env, AVR_FEATURE_MOVW); > > + avr_set_feature(env, AVR_FEATURE_MUL); > > + avr_set_feature(env, AVR_FEATURE_RMW); > > +} > > + > > +typedef struct AVRCPUInfo { > > + const char *name; > > + void (*initfn)(Object *obj); > > +} AVRCPUInfo; > > + > > +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) > > +{ > > + ObjectClass *class_a = (ObjectClass *)a; > > + ObjectClass *class_b = (ObjectClass *)b; > > + const char *name_a; > > + const char *name_b; > > + > > + name_a = object_class_get_name(class_a); > > + name_b = object_class_get_name(class_b); > > + > > + return strcmp(name_a, name_b); > > +} > > + > > +static void avr_cpu_list_entry(gpointer data, gpointer user_data) > > +{ > > + const char *typename = object_class_get_name(OBJECT_CLASS(data)); > > + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); > > + qemu_printf("%.*s\n", len, typename); > > +} > > + > > +void avr_cpu_list(void) > > +{ > > + GSList *list; > > + list = object_class_get_list(TYPE_AVR_CPU, false); > > + list = g_slist_sort(list, avr_cpu_list_compare); > > + qemu_printf("Available CPUs:\n"); > > + g_slist_foreach(list, avr_cpu_list_entry, NULL); > > + g_slist_free(list); > > +} > > + > > +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ > > + { \ > > + .parent = TYPE_AVR_CPU, \ > > + .instance_init = initfn, \ > > + .name = AVR_CPU_TYPE_NAME(model), \ > > + } > > + > > +static const TypeInfo avr_cpu_type_info[] = { > > + { > > + .name = TYPE_AVR_CPU, > > + .parent = TYPE_CPU, > > + .instance_size = sizeof(AVRCPU), > > + .instance_init = avr_cpu_initfn, > > + .class_size = sizeof(AVRCPUClass), > > + .class_init = avr_cpu_class_init, > > + .abstract = true, > > + }, > > + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), > > + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), > > + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), > > + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), > > + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), > > + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), > > + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), > > + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), > > + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), > > + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), > > + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), > > + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), > > + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), > > + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), > > + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), > > +}; > > + > > +DEFINE_TYPES(avr_cpu_type_info) > > diff --git a/target/avr/cpu.h b/target/avr/cpu.h > > new file mode 100644 > > index 0000000000..3f9a803193 > > --- /dev/null > > +++ b/target/avr/cpu.h > > @@ -0,0 +1,283 @@ > > +/* > > + * QEMU AVR CPU > > + * > > + * Copyright (c) 2016 Michael Rolnik > > + * > > + * 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 > > + * version 2.1 of the License, or (at your option) any later version. > > + * > > + * 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/lgpl-2.1.html> > > + */ > > + > > +#ifndef CPU_AVR_H > > +#define CPU_AVR_H > > + > > +#include "qemu-common.h" > > +#include "qom/cpu.h" > > +#include "exec/cpu-defs.h" > > +#include "fpu/softfloat.h" > > + > > +#define TCG_GUEST_DEFAULT_MO 0 > > + > > +#define TYPE_AVR_CPU "avr-cpu" > > + > > +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU > > +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) > > +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU > > + > > +/* > > + * AVR has two memory spaces, data & code. > > + * e.g. both have 0 address > > + * ST/LD instructions access data space > > + * LPM/SPM and instruction fetching access code memory space > > + */ > > +#define MMU_CODE_IDX 0 > > +#define MMU_DATA_IDX 1 > > + > > +#define EXCP_RESET 1 > > +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) > > + > > +/* Number of CPU registers */ > > +#define NO_CPU_REGISTERS 32 > > +/* Number of IO registers accessible by ld/st/in/out */ > > +#define NO_IO_REGISTERS 64 > > + > > +/* > > + * Offsets of AVR memory regions in host memory space. > > + * > > + * This is needed because the AVR has separate code and data address > > + * spaces that both have start from zero but have to go somewhere in > > + * host memory. > > + * > > + * It's also useful to know where some things are, like the IO > registers. > > + */ > > +/* Flash program memory */ > > +#define OFFSET_CODE 0x00000000 > > +/* CPU registers, IO registers, and SRAM */ > > +#define OFFSET_DATA 0x00800000 > > +/* CPU registers specifically, these are mapped at the start of data */ > > +#define OFFSET_CPU_REGISTERS OFFSET_DATA > > +/* > > + * IO registers, including status register, stack pointer, and memory > > + * mapped peripherals, mapped just after CPU registers > > + */ > > +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) > > + > > +enum avr_features { > > + AVR_FEATURE_SRAM, > > + > > + AVR_FEATURE_1_BYTE_PC, > > + AVR_FEATURE_2_BYTE_PC, > > + AVR_FEATURE_3_BYTE_PC, > > + > > + AVR_FEATURE_1_BYTE_SP, > > + AVR_FEATURE_2_BYTE_SP, > > + > > + AVR_FEATURE_BREAK, > > + AVR_FEATURE_DES, > > + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ > > + > > + AVR_FEATURE_EIJMP_EICALL, > > + AVR_FEATURE_IJMP_ICALL, > > + AVR_FEATURE_JMP_CALL, > > + > > + AVR_FEATURE_ADIW_SBIW, > > + > > + AVR_FEATURE_SPM, > > + AVR_FEATURE_SPMX, > > + > > + AVR_FEATURE_ELPMX, > > + AVR_FEATURE_ELPM, > > + AVR_FEATURE_LPMX, > > + AVR_FEATURE_LPM, > > + > > + AVR_FEATURE_MOVW, > > + AVR_FEATURE_MUL, > > + AVR_FEATURE_RAMPD, > > + AVR_FEATURE_RAMPX, > > + AVR_FEATURE_RAMPY, > > + AVR_FEATURE_RAMPZ, > > +}; > > + > > +typedef struct CPUAVRState CPUAVRState; > > + > > +struct CPUAVRState { > > + uint32_t pc_w; /* 0x003fffff up to 22 bits */ > > + > > + uint32_t sregC; /* 0x00000001 1 bits */ > > + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ > > + /* 0=flag set, >0=flag cleared */ > > + uint32_t sregN; /* 0x00000001 1 bits */ > > + uint32_t sregV; /* 0x00000001 1 bits */ > > + uint32_t sregS; /* 0x00000001 1 bits */ > > + uint32_t sregH; /* 0x00000001 1 bits */ > > + uint32_t sregT; /* 0x00000001 1 bits */ > > + uint32_t sregI; /* 0x00000001 1 bits */ > > + > > + uint32_t rampD; /* 0x00ff0000 8 bits */ > > + uint32_t rampX; /* 0x00ff0000 8 bits */ > > + uint32_t rampY; /* 0x00ff0000 8 bits */ > > + uint32_t rampZ; /* 0x00ff0000 8 bits */ > > + uint32_t eind; /* 0x00ff0000 8 bits */ > > + > > + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ > > + uint32_t sp; /* 16 bits */ > > + > > + uint32_t skip; /* if set skip instruction */ > > + > > + uint64_t intsrc; /* interrupt sources */ > > + bool fullacc; /* CPU/MEM if true MEM only otherwise */ > > + > > + uint32_t features; > > +}; > > + > > +#define AVR_CPU_CLASS(klass) \ > > + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) > > +#define AVR_CPU(obj) \ > > + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) > > +#define AVR_CPU_GET_CLASS(obj) \ > > + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) > > + > > +/** > > + * AVRCPUClass: > > + * @parent_realize: The parent class' realize handler. > > + * @parent_reset: The parent class' reset handler. > > + * @vr: Version Register value. > > + * > > + * A AVR CPU model. > > + */ > > +typedef struct AVRCPUClass { > > + /*< private >*/ > > + CPUClass parent_class; > > + /*< public >*/ > > + DeviceRealize parent_realize; > > + void (*parent_reset)(CPUState *cpu); > > +} AVRCPUClass; > > + > > +/** > > + * AVRCPU: > > + * @env: #CPUAVRState > > + * > > + * A AVR CPU. > > + */ > > +typedef struct AVRCPU { > > + /*< private >*/ > > + CPUState parent_obj; > > + /*< public >*/ > > + > > + CPUNegativeOffsetState neg; > > + CPUAVRState env; > > +} AVRCPU; > > + > > +#ifndef CONFIG_USER_ONLY > > +extern const struct VMStateDescription vms_avr_cpu; > > +#endif > > + > > +void avr_cpu_do_interrupt(CPUState *cpu); > > +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); > > +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); > > +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg); > > +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg); > > + > > +static inline int avr_feature(CPUAVRState *env, int feature) > > +{ > > + return (env->features & (1U << feature)) != 0; > > +} > > + > > +static inline void avr_set_feature(CPUAVRState *env, int feature) > > +{ > > + env->features |= (1U << feature); > > +} > > + > > +#define cpu_list avr_cpu_list > > +#define cpu_signal_handler cpu_avr_signal_handler > > +#define cpu_mmu_index avr_cpu_mmu_index > > + > > +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) > > +{ > > + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; > > +} > > + > > +void avr_cpu_tcg_init(void); > > + > > +void avr_cpu_list(void); > > +int cpu_avr_exec(CPUState *cpu); > > +int cpu_avr_signal_handler(int host_signum, void *pinfo, void *puc); > > +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int size, > > + int rw, int mmu_idx); > > +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t *buf, > > + int len, bool is_write); > > + > > +enum { > > + TB_FLAGS_FULL_ACCESS = 1, > > + TB_FLAGS_SKIP = 2, > > +}; > > + > > +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, target_ulong > *pc, > > + target_ulong *cs_base, uint32_t *pflags) > > +{ > > + uint32_t flags = 0; > > + > > + *pc = env->pc_w * 2; > > + *cs_base = 0; > > + > > + if (env->fullacc) { > > + flags |= TB_FLAGS_FULL_ACCESS; > > + } > > + if (env->skip) { > > + flags |= TB_FLAGS_SKIP; > > + } > > + > > + *pflags = flags; > > +} > > + > > +static inline int cpu_interrupts_enabled(CPUAVRState *env) > > +{ > > + return env->sregI != 0; > > +} > > + > > +static inline uint8_t cpu_get_sreg(CPUAVRState *env) > > +{ > > + uint8_t sreg; > > + sreg = (env->sregC & 0x01) << 0 > > + | (env->sregZ == 0 ? 1 : 0) << 1 > > + | (env->sregN) << 2 > > + | (env->sregV) << 3 > > + | (env->sregS) << 4 > > + | (env->sregH) << 5 > > + | (env->sregT) << 6 > > + | (env->sregI) << 7; > > + return sreg; > > +} > > + > > +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) > > +{ > > + env->sregC = (sreg >> 0) & 0x01; > > + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; > > + env->sregN = (sreg >> 2) & 0x01; > > + env->sregV = (sreg >> 3) & 0x01; > > + env->sregS = (sreg >> 4) & 0x01; > > + env->sregH = (sreg >> 5) & 0x01; > > + env->sregT = (sreg >> 6) & 0x01; > > + env->sregI = (sreg >> 7) & 0x01; > > +} > > + > > +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, > > + MMUAccessType access_type, int mmu_idx, > > + bool probe, uintptr_t retaddr); > > + > > +typedef CPUAVRState CPUArchState; > > +typedef AVRCPU ArchCPU; > > + > > +#include "exec/cpu-all.h" > > + > > +#endif /* !defined (CPU_AVR_H) */ > > diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c > > new file mode 100644 > > index 0000000000..537dc7226e > > --- /dev/null > > +++ b/target/avr/gdbstub.c > > @@ -0,0 +1,85 @@ > > +/* > > + * QEMU AVR CPU > > + * > > + * Copyright (c) 2016 Michael Rolnik > > + * > > + * 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 > > + * version 2.1 of the License, or (at your option) any later version. > > + * > > + * 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/lgpl-2.1.html> > > + */ > > + > > +#include "qemu/osdep.h" > > +#include "qemu-common.h" > > +#include "exec/gdbstub.h" > > + > > +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + CPUAVRState *env = &cpu->env; > > + > > + /* R */ > > + if (n < 32) { > > + return gdb_get_reg8(mem_buf, env->r[n]); > > + } > > + > > + /* SREG */ > > + if (n == 32) { > > + uint8_t sreg = cpu_get_sreg(env); > > + > > + return gdb_get_reg8(mem_buf, sreg); > > + } > > + > > + /* SP */ > > + if (n == 33) { > > + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); > > + } > > + > > + /* PC */ > > + if (n == 34) { > > + return gdb_get_reg32(mem_buf, env->pc_w * 2); > > + } > > + > > + return 0; > > +} > > + > > +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n) > > +{ > > + AVRCPU *cpu = AVR_CPU(cs); > > + CPUAVRState *env = &cpu->env; > > + > > + /* R */ > > + if (n < 32) { > > + env->r[n] = *mem_buf; > > + return 1; > > + } > > + > > + /* SREG */ > > + if (n == 32) { > > + cpu_set_sreg(env, *mem_buf); > > + return 1; > > + } > > + > > + /* SP */ > > + if (n == 33) { > > + env->sp = lduw_p(mem_buf); > > + return 2; > > + } > > + > > + /* PC */ > > + if (n == 34) { > > + env->pc_w = ldl_p(mem_buf) / 2; > > + return 4; > > + } > > + > > + return 0; > > +} > > diff --git a/target/avr/machine.c b/target/avr/machine.c > > new file mode 100644 > > index 0000000000..389b5eb95d > > --- /dev/null > > +++ b/target/avr/machine.c > > @@ -0,0 +1,123 @@ > > +/* > > + * QEMU AVR CPU > > + * > > + * Copyright (c) 2016 Michael Rolnik > > + * > > + * 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 > > + * version 2.1 of the License, or (at your option) any later version. > > + * > > + * 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/lgpl-2.1.html> > > + */ > > + > > +#include "qemu/osdep.h" > > +#include "hw/hw.h" > > +#include "cpu.h" > > +#include "hw/boards.h" > > +#include "migration/qemu-file.h" > > + > > +static int get_sreg(QEMUFile *f, void *opaque, size_t size, > > + const VMStateField *field) > > +{ > > + CPUAVRState *env = opaque; > > + uint8_t sreg; > > + > > + sreg = qemu_get_byte(f); > > + cpu_set_sreg(env, sreg); > > + return 0; > > +} > > + > > +static int put_sreg( > > + QEMUFile *f, void *opaque, size_t size, > > + const VMStateField *field, QJSON *vmdesc) > > +{ > > + CPUAVRState *env = opaque; > > + uint8_t sreg = cpu_get_sreg(env); > > + > > + qemu_put_byte(f, sreg); > > + return 0; > > +} > > + > > +static const VMStateInfo vms_sreg = { > > + .name = "sreg", > > + .get = get_sreg, > > + .put = put_sreg, > > +}; > > + > > +static int get_segment( > > + QEMUFile *f, void *opaque, size_t size, const VMStateField *field) > > +{ > > + uint32_t *ramp = opaque; > > + uint8_t temp; > > + > > + temp = qemu_get_byte(f); > > + *ramp = ((uint32_t)temp) << 16; > > + return 0; > > +} > > + > > +static int put_segment( > > + QEMUFile *f, void *opaque, size_t size, > > + const VMStateField *field, QJSON *vmdesc) > > +{ > > + uint32_t *ramp = opaque; > > + uint8_t temp = *ramp >> 16; > > + > > + qemu_put_byte(f, temp); > > + return 0; > > +} > > + > > +static const VMStateInfo vms_rampD = { > > + .name = "rampD", > > + .get = get_segment, > > + .put = put_segment, > > +}; > > +static const VMStateInfo vms_rampX = { > > + .name = "rampX", > > + .get = get_segment, > > + .put = put_segment, > > +}; > > +static const VMStateInfo vms_rampY = { > > + .name = "rampY", > > + .get = get_segment, > > + .put = put_segment, > > +}; > > +static const VMStateInfo vms_rampZ = { > > + .name = "rampZ", > > + .get = get_segment, > > + .put = put_segment, > > +}; > > +static const VMStateInfo vms_eind = { > > + .name = "eind", > > + .get = get_segment, > > + .put = put_segment, > > +}; > > + > > +const VMStateDescription vms_avr_cpu = { > > + .name = "cpu", > > + .version_id = 0, > > + .minimum_version_id = 0, > > + .fields = (VMStateField[]) { > > + VMSTATE_UINT32(env.pc_w, AVRCPU), > > + VMSTATE_UINT32(env.sp, AVRCPU), > > + VMSTATE_UINT32(env.skip, AVRCPU), > > + > > + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), > > + > > + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), > > + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), > > + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), > > + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), > > + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), > > + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), > > + > > + VMSTATE_END_OF_LIST() > > + } > > +}; > >
I meant that if I take the proposed code it fails in the mentioned way Sent from my cell phone, please ignore typos On Thu, Jun 27, 2019, 7:02 PM Michael Rolnik <mrolnik@gmail.com> wrote: > Hi Igor, > > if I run `make check` it fails with > qemu-system-rx: unable to find CPU model 'rx62n' > or > qemu-system-avr: unable to find CPU model 'avr6' > > On Thu, Jun 27, 2019 at 2:28 PM Igor Mammedov <imammedo@redhat.com> wrote: > >> On Thu, 27 Jun 2019 08:27:44 +0300 >> Michael Rolnik <mrolnik@gmail.com> wrote: >> >> > From: Sarah Harris <S.E.Harris@kent.ac.uk> >> > >> > This includes: >> > - CPU data structures >> > - object model classes and functions >> > - migration functions >> > - GDB hooks >> > >> > Signed-off-by: Michael Rolnik <mrolnik@gmail.com> >> > --- >> > gdb-xml/avr-cpu.xml | 49 ++++ >> > target/avr/cpu-param.h | 37 +++ >> > target/avr/cpu.c | 599 +++++++++++++++++++++++++++++++++++++++++ >> > target/avr/cpu.h | 283 +++++++++++++++++++ >> > target/avr/gdbstub.c | 85 ++++++ >> > target/avr/machine.c | 123 +++++++++ >> > 6 files changed, 1176 insertions(+) >> > create mode 100644 gdb-xml/avr-cpu.xml >> > create mode 100644 target/avr/cpu-param.h >> > create mode 100644 target/avr/cpu.c >> > create mode 100644 target/avr/cpu.h >> > create mode 100644 target/avr/gdbstub.c >> > create mode 100644 target/avr/machine.c >> > >> > diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml >> > new file mode 100644 >> > index 0000000000..c4747f5b40 >> > --- /dev/null >> > +++ b/gdb-xml/avr-cpu.xml >> > @@ -0,0 +1,49 @@ >> > +<?xml version="1.0"?> >> > +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. >> > + >> > + Copying and distribution of this file, with or without >> modification, >> > + are permitted in any medium without royalty provided the copyright >> > + notice and this notice are preserved. --> >> > + >> > +<!-- Register numbers are hard-coded in order to maintain backward >> > + compatibility with older versions of tools that didn't use xml >> > + register descriptions. --> >> > + >> > +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> >> > +<feature name="org.gnu.gdb.riscv.cpu"> >> > + <reg name="r0" bitsize="8" type="int" regnum="0"/> >> > + <reg name="r1" bitsize="8" type="int"/> >> > + <reg name="r2" bitsize="8" type="int"/> >> > + <reg name="r3" bitsize="8" type="int"/> >> > + <reg name="r4" bitsize="8" type="int"/> >> > + <reg name="r5" bitsize="8" type="int"/> >> > + <reg name="r6" bitsize="8" type="int"/> >> > + <reg name="r7" bitsize="8" type="int"/> >> > + <reg name="r8" bitsize="8" type="int"/> >> > + <reg name="r9" bitsize="8" type="int"/> >> > + <reg name="r10" bitsize="8" type="int"/> >> > + <reg name="r11" bitsize="8" type="int"/> >> > + <reg name="r12" bitsize="8" type="int"/> >> > + <reg name="r13" bitsize="8" type="int"/> >> > + <reg name="r14" bitsize="8" type="int"/> >> > + <reg name="r15" bitsize="8" type="int"/> >> > + <reg name="r16" bitsize="8" type="int"/> >> > + <reg name="r17" bitsize="8" type="int"/> >> > + <reg name="r18" bitsize="8" type="int"/> >> > + <reg name="r19" bitsize="8" type="int"/> >> > + <reg name="r20" bitsize="8" type="int"/> >> > + <reg name="r21" bitsize="8" type="int"/> >> > + <reg name="r22" bitsize="8" type="int"/> >> > + <reg name="r23" bitsize="8" type="int"/> >> > + <reg name="r24" bitsize="8" type="int"/> >> > + <reg name="r25" bitsize="8" type="int"/> >> > + <reg name="r26" bitsize="8" type="int"/> >> > + <reg name="r27" bitsize="8" type="int"/> >> > + <reg name="r28" bitsize="8" type="int"/> >> > + <reg name="r29" bitsize="8" type="int"/> >> > + <reg name="r30" bitsize="8" type="int"/> >> > + <reg name="r31" bitsize="8" type="int"/> >> > + <reg name="sreg" bitsize="8" type="int"/> >> > + <reg name="sp" bitsize="8" type="int"/> >> > + <reg name="pc" bitsize="8" type="int"/> >> > +</feature> >> > diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h >> > new file mode 100644 >> > index 0000000000..5bbf985726 >> > --- /dev/null >> > +++ b/target/avr/cpu-param.h >> > @@ -0,0 +1,37 @@ >> > +/* >> > + * QEMU AVR CPU >> > + * >> > + * Copyright (c) 2016 Michael Rolnik >> > + * >> > + * 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 >> > + * version 2.1 of the License, or (at your option) any later version. >> > + * >> > + * 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/lgpl-2.1.html> >> > + */ >> > + >> > +#ifndef AVR_CPU_PARAM_H >> > +#define AVR_CPU_PARAM_H 1 >> > + >> > +#define TARGET_LONG_BITS 32 >> > +/* >> > + * TARGET_PAGE_BITS cannot be more than 8 bits because >> > + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, and >> they >> > + * should be implemented as a device and not memory >> > + * 2. SRAM starts at the address 0x0100 >> > + */ >> > +#define TARGET_PAGE_BITS 8 >> > +#define TARGET_PHYS_ADDR_SPACE_BITS 24 >> > +#define TARGET_VIRT_ADDR_SPACE_BITS 24 >> > +#define NB_MMU_MODES 2 >> > + >> > + >> > +#endif >> > diff --git a/target/avr/cpu.c b/target/avr/cpu.c >> > new file mode 100644 >> > index 0000000000..142fe54524 >> > --- /dev/null >> > +++ b/target/avr/cpu.c >> > @@ -0,0 +1,599 @@ >> > +/* >> > + * QEMU AVR CPU >> > + * >> > + * Copyright (c) 2016 Michael Rolnik >> > + * >> > + * 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 >> > + * version 2.1 of the License, or (at your option) any later version. >> > + * >> > + * 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/lgpl-2.1.html> >> > + */ >> > + >> > +#include "qemu/osdep.h" >> > +#include "qemu/qemu-print.h" >> > +#include "qemu/log.h" >> > +#include "cpu.h" >> > +#include "exec/exec-all.h" >> > +#include "qapi/error.h" >> > +#include "hw/qdev-properties.h" >> > +#include "migration/vmstate.h" >> > + >> > +static void avr_cpu_set_pc(CPUState *cs, vaddr value) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + >> > + cpu->env.pc_w = value / 2; /* internally PC points to words */ >> > +} >> > + >> > +static bool avr_cpu_has_work(CPUState *cs) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | >> CPU_INTERRUPT_RESET)) >> > + && cpu_interrupts_enabled(env); >> > +} >> > + >> > +static void avr_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock >> *tb) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + env->pc_w = tb->pc / 2; /* internally PC points to words */ >> > +} >> > + >> > +static void avr_cpu_reset(CPUState *cs) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + mcc->parent_reset(cs); >> > + >> > + env->pc_w = 0; >> > + env->sregI = 1; >> > + env->sregC = 0; >> > + env->sregZ = 0; >> > + env->sregN = 0; >> > + env->sregV = 0; >> > + env->sregS = 0; >> > + env->sregH = 0; >> > + env->sregT = 0; >> > + >> > + env->rampD = 0; >> > + env->rampX = 0; >> > + env->rampY = 0; >> > + env->rampZ = 0; >> > + env->eind = 0; >> > + env->sp = 0; >> > + >> > + env->skip = 0; >> > + >> > + memset(env->r, 0, sizeof(env->r)); >> > + >> > + tlb_flush(cs); >> > +} >> > + >> > +static void avr_cpu_disas_set_info(CPUState *cpu, disassemble_info >> *info) >> > +{ >> > + info->mach = bfd_arch_avr; >> > + info->print_insn = NULL; >> > +} >> > + >> > +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) >> > +{ >> > + CPUState *cs = CPU(dev); >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); >> > + Error *local_err = NULL; >> > + >> > + cpu_exec_realizefn(cs, &local_err); >> > + if (local_err != NULL) { >> > + error_propagate(errp, local_err); >> > + return; >> > + } >> > + qemu_init_vcpu(cs); >> > + cpu_reset(cs); >> > + >> > + mcc->parent_realize(dev, errp); >> > +} >> > + >> > +static void avr_cpu_set_int(void *opaque, int irq, int level) >> > +{ >> > + AVRCPU *cpu = opaque; >> > + CPUAVRState *env = &cpu->env; >> > + CPUState *cs = CPU(cpu); >> > + >> > + uint64_t mask = (1ull << irq); >> > + if (level) { >> > + env->intsrc |= mask; >> > + cpu_interrupt(cs, CPU_INTERRUPT_HARD); >> > + } else { >> > + env->intsrc &= ~mask; >> > + if (env->intsrc == 0) { >> > + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); >> > + } >> > + } >> > +} >> > + >> > +static void avr_cpu_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + >> > + cpu_set_cpustate_pointers(cpu); >> > + >> > +#ifndef CONFIG_USER_ONLY >> > + /* Set the number of interrupts supported by the CPU. */ >> > + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); >> > +#endif >> > +} >> > + >> > +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) >> > +{ >> > + ObjectClass *oc; >> > + char *typename; >> > + char **cpuname; >> > + >> > + cpuname = g_strsplit(cpu_model, ",", 1); >> you do not need this, parse_cpu_option() does it for you. >> >> > + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), cpuname[0]); >> > + oc = object_class_by_name(typename); >> Also just use cpu_model directly for lookup >> for example see '[Qemu-devel] [PATCH RESEND v21 06/21] target/rx: CPU >> definition' >> and simplify avr_cpu_list_entry() correspondingly. >> >> >> > + g_strfreev(cpuname); >> > + g_free(typename); >> > + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || >> > + object_class_is_abstract(oc)) { >> > + return NULL; >> > + } >> > + return oc; >> > +} >> > + >> > +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + CPUAVRState *env = &cpu->env; >> > + int i; >> > + >> > + qemu_fprintf(f, "\n"); >> > + qemu_fprintf(f, "PC: %06x\n", env->pc_w); >> > + qemu_fprintf(f, "SP: %04x\n", env->sp); >> > + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); >> > + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); >> > + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); >> > + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); >> > + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); >> > + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); >> > + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); >> > + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); >> > + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", >> > + env->sregI ? 'I' : '-', >> > + env->sregT ? 'T' : '-', >> > + env->sregH ? 'H' : '-', >> > + env->sregS ? 'S' : '-', >> > + env->sregV ? 'V' : '-', >> > + env->sregN ? '-' : 'N', /* Zf has negative >> logic */ >> > + env->sregZ ? 'Z' : '-', >> > + env->sregC ? 'I' : '-'); >> > + qemu_fprintf(f, "SKIP: %02x\n", env->skip); >> > + >> > + qemu_fprintf(f, "\n"); >> > + for (i = 0; i < ARRAY_SIZE(env->r); i++) { >> > + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); >> > + >> > + if ((i % 8) == 7) { >> > + qemu_fprintf(f, "\n"); >> > + } >> > + } >> > + qemu_fprintf(f, "\n"); >> > +} >> > + >> > +static void avr_cpu_class_init(ObjectClass *oc, void *data) >> > +{ >> > + DeviceClass *dc = DEVICE_CLASS(oc); >> > + CPUClass *cc = CPU_CLASS(oc); >> > + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); >> > + >> > + mcc->parent_realize = dc->realize; >> > + dc->realize = avr_cpu_realizefn; >> > + >> > + mcc->parent_reset = cc->reset; >> > + cc->reset = avr_cpu_reset; >> > + >> > + cc->class_by_name = avr_cpu_class_by_name; >> > + >> > + cc->has_work = avr_cpu_has_work; >> > + cc->do_interrupt = avr_cpu_do_interrupt; >> > + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; >> > + cc->dump_state = avr_cpu_dump_state; >> > + cc->set_pc = avr_cpu_set_pc; >> > +#if !defined(CONFIG_USER_ONLY) >> > + cc->memory_rw_debug = avr_cpu_memory_rw_debug; >> > +#endif >> > +#ifdef CONFIG_USER_ONLY >> > + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; >> > +#else >> > + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; >> > + cc->vmsd = &vms_avr_cpu; >> > +#endif >> > + cc->disas_set_info = avr_cpu_disas_set_info; >> > + cc->tlb_fill = avr_cpu_tlb_fill; >> > + cc->tcg_initialize = avr_cpu_tcg_init; >> > + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; >> > + cc->gdb_read_register = avr_cpu_gdb_read_register; >> > + cc->gdb_write_register = avr_cpu_gdb_write_register; >> > + cc->gdb_num_core_regs = 35; >> > + cc->gdb_core_xml_file = "avr-cpu.xml"; >> > +} >> > + >> > +static void avr_avr1_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > +} >> > + >> > +static void avr_avr2_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > +} >> > + >> > +static void avr_avr25_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > +} >> > + >> > +static void avr_avr3_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > +} >> > + >> > +static void avr_avr31_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > +} >> > + >> > +static void avr_avr35_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > +} >> > + >> > +static void avr_avr4_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > +} >> > + >> > +static void avr_avr5_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > +} >> > + >> > +static void avr_avr51_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > +} >> > + >> > +static void avr_avr6_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > +} >> > + >> > +static void avr_xmega2_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > + avr_set_feature(env, AVR_FEATURE_RMW); >> > +} >> > + >> > +static void avr_xmega4_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > + avr_set_feature(env, AVR_FEATURE_RMW); >> > +} >> > + >> > +static void avr_xmega5_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > + avr_set_feature(env, AVR_FEATURE_RMW); >> > +} >> > + >> > +static void avr_xmega6_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > + avr_set_feature(env, AVR_FEATURE_RMW); >> > +} >> > + >> > +static void avr_xmega7_initfn(Object *obj) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(obj); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + avr_set_feature(env, AVR_FEATURE_LPM); >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); >> > + avr_set_feature(env, AVR_FEATURE_SRAM); >> > + avr_set_feature(env, AVR_FEATURE_BREAK); >> > + >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); >> > + avr_set_feature(env, AVR_FEATURE_ELPM); >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); >> > + avr_set_feature(env, AVR_FEATURE_LPMX); >> > + avr_set_feature(env, AVR_FEATURE_MOVW); >> > + avr_set_feature(env, AVR_FEATURE_MUL); >> > + avr_set_feature(env, AVR_FEATURE_RMW); >> > +} >> > + >> > +typedef struct AVRCPUInfo { >> > + const char *name; >> > + void (*initfn)(Object *obj); >> > +} AVRCPUInfo; >> > + >> > +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) >> > +{ >> > + ObjectClass *class_a = (ObjectClass *)a; >> > + ObjectClass *class_b = (ObjectClass *)b; >> > + const char *name_a; >> > + const char *name_b; >> > + >> > + name_a = object_class_get_name(class_a); >> > + name_b = object_class_get_name(class_b); >> > + >> > + return strcmp(name_a, name_b); >> > +} >> > + >> > +static void avr_cpu_list_entry(gpointer data, gpointer user_data) >> > +{ >> > + const char *typename = object_class_get_name(OBJECT_CLASS(data)); >> > + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); >> > + qemu_printf("%.*s\n", len, typename); >> > +} >> > + >> > +void avr_cpu_list(void) >> > +{ >> > + GSList *list; >> > + list = object_class_get_list(TYPE_AVR_CPU, false); >> > + list = g_slist_sort(list, avr_cpu_list_compare); >> > + qemu_printf("Available CPUs:\n"); >> > + g_slist_foreach(list, avr_cpu_list_entry, NULL); >> > + g_slist_free(list); >> > +} >> > + >> > +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ >> > + { \ >> > + .parent = TYPE_AVR_CPU, \ >> > + .instance_init = initfn, \ >> > + .name = AVR_CPU_TYPE_NAME(model), \ >> > + } >> > + >> > +static const TypeInfo avr_cpu_type_info[] = { >> > + { >> > + .name = TYPE_AVR_CPU, >> > + .parent = TYPE_CPU, >> > + .instance_size = sizeof(AVRCPU), >> > + .instance_init = avr_cpu_initfn, >> > + .class_size = sizeof(AVRCPUClass), >> > + .class_init = avr_cpu_class_init, >> > + .abstract = true, >> > + }, >> > + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), >> > + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), >> > + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), >> > + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), >> > + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), >> > + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), >> > + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), >> > +}; >> > + >> > +DEFINE_TYPES(avr_cpu_type_info) >> > diff --git a/target/avr/cpu.h b/target/avr/cpu.h >> > new file mode 100644 >> > index 0000000000..3f9a803193 >> > --- /dev/null >> > +++ b/target/avr/cpu.h >> > @@ -0,0 +1,283 @@ >> > +/* >> > + * QEMU AVR CPU >> > + * >> > + * Copyright (c) 2016 Michael Rolnik >> > + * >> > + * 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 >> > + * version 2.1 of the License, or (at your option) any later version. >> > + * >> > + * 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/lgpl-2.1.html> >> > + */ >> > + >> > +#ifndef CPU_AVR_H >> > +#define CPU_AVR_H >> > + >> > +#include "qemu-common.h" >> > +#include "qom/cpu.h" >> > +#include "exec/cpu-defs.h" >> > +#include "fpu/softfloat.h" >> > + >> > +#define TCG_GUEST_DEFAULT_MO 0 >> > + >> > +#define TYPE_AVR_CPU "avr-cpu" >> > + >> > +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU >> > +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) >> > +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU >> > + >> > +/* >> > + * AVR has two memory spaces, data & code. >> > + * e.g. both have 0 address >> > + * ST/LD instructions access data space >> > + * LPM/SPM and instruction fetching access code memory space >> > + */ >> > +#define MMU_CODE_IDX 0 >> > +#define MMU_DATA_IDX 1 >> > + >> > +#define EXCP_RESET 1 >> > +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) >> > + >> > +/* Number of CPU registers */ >> > +#define NO_CPU_REGISTERS 32 >> > +/* Number of IO registers accessible by ld/st/in/out */ >> > +#define NO_IO_REGISTERS 64 >> > + >> > +/* >> > + * Offsets of AVR memory regions in host memory space. >> > + * >> > + * This is needed because the AVR has separate code and data address >> > + * spaces that both have start from zero but have to go somewhere in >> > + * host memory. >> > + * >> > + * It's also useful to know where some things are, like the IO >> registers. >> > + */ >> > +/* Flash program memory */ >> > +#define OFFSET_CODE 0x00000000 >> > +/* CPU registers, IO registers, and SRAM */ >> > +#define OFFSET_DATA 0x00800000 >> > +/* CPU registers specifically, these are mapped at the start of data */ >> > +#define OFFSET_CPU_REGISTERS OFFSET_DATA >> > +/* >> > + * IO registers, including status register, stack pointer, and memory >> > + * mapped peripherals, mapped just after CPU registers >> > + */ >> > +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) >> > + >> > +enum avr_features { >> > + AVR_FEATURE_SRAM, >> > + >> > + AVR_FEATURE_1_BYTE_PC, >> > + AVR_FEATURE_2_BYTE_PC, >> > + AVR_FEATURE_3_BYTE_PC, >> > + >> > + AVR_FEATURE_1_BYTE_SP, >> > + AVR_FEATURE_2_BYTE_SP, >> > + >> > + AVR_FEATURE_BREAK, >> > + AVR_FEATURE_DES, >> > + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ >> > + >> > + AVR_FEATURE_EIJMP_EICALL, >> > + AVR_FEATURE_IJMP_ICALL, >> > + AVR_FEATURE_JMP_CALL, >> > + >> > + AVR_FEATURE_ADIW_SBIW, >> > + >> > + AVR_FEATURE_SPM, >> > + AVR_FEATURE_SPMX, >> > + >> > + AVR_FEATURE_ELPMX, >> > + AVR_FEATURE_ELPM, >> > + AVR_FEATURE_LPMX, >> > + AVR_FEATURE_LPM, >> > + >> > + AVR_FEATURE_MOVW, >> > + AVR_FEATURE_MUL, >> > + AVR_FEATURE_RAMPD, >> > + AVR_FEATURE_RAMPX, >> > + AVR_FEATURE_RAMPY, >> > + AVR_FEATURE_RAMPZ, >> > +}; >> > + >> > +typedef struct CPUAVRState CPUAVRState; >> > + >> > +struct CPUAVRState { >> > + uint32_t pc_w; /* 0x003fffff up to 22 bits */ >> > + >> > + uint32_t sregC; /* 0x00000001 1 bits */ >> > + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ >> > + /* 0=flag set, >0=flag cleared */ >> > + uint32_t sregN; /* 0x00000001 1 bits */ >> > + uint32_t sregV; /* 0x00000001 1 bits */ >> > + uint32_t sregS; /* 0x00000001 1 bits */ >> > + uint32_t sregH; /* 0x00000001 1 bits */ >> > + uint32_t sregT; /* 0x00000001 1 bits */ >> > + uint32_t sregI; /* 0x00000001 1 bits */ >> > + >> > + uint32_t rampD; /* 0x00ff0000 8 bits */ >> > + uint32_t rampX; /* 0x00ff0000 8 bits */ >> > + uint32_t rampY; /* 0x00ff0000 8 bits */ >> > + uint32_t rampZ; /* 0x00ff0000 8 bits */ >> > + uint32_t eind; /* 0x00ff0000 8 bits */ >> > + >> > + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ >> > + uint32_t sp; /* 16 bits */ >> > + >> > + uint32_t skip; /* if set skip instruction */ >> > + >> > + uint64_t intsrc; /* interrupt sources */ >> > + bool fullacc; /* CPU/MEM if true MEM only otherwise */ >> > + >> > + uint32_t features; >> > +}; >> > + >> > +#define AVR_CPU_CLASS(klass) \ >> > + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) >> > +#define AVR_CPU(obj) \ >> > + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) >> > +#define AVR_CPU_GET_CLASS(obj) \ >> > + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) >> > + >> > +/** >> > + * AVRCPUClass: >> > + * @parent_realize: The parent class' realize handler. >> > + * @parent_reset: The parent class' reset handler. >> > + * @vr: Version Register value. >> > + * >> > + * A AVR CPU model. >> > + */ >> > +typedef struct AVRCPUClass { >> > + /*< private >*/ >> > + CPUClass parent_class; >> > + /*< public >*/ >> > + DeviceRealize parent_realize; >> > + void (*parent_reset)(CPUState *cpu); >> > +} AVRCPUClass; >> > + >> > +/** >> > + * AVRCPU: >> > + * @env: #CPUAVRState >> > + * >> > + * A AVR CPU. >> > + */ >> > +typedef struct AVRCPU { >> > + /*< private >*/ >> > + CPUState parent_obj; >> > + /*< public >*/ >> > + >> > + CPUNegativeOffsetState neg; >> > + CPUAVRState env; >> > +} AVRCPU; >> > + >> > +#ifndef CONFIG_USER_ONLY >> > +extern const struct VMStateDescription vms_avr_cpu; >> > +#endif >> > + >> > +void avr_cpu_do_interrupt(CPUState *cpu); >> > +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); >> > +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); >> > +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg); >> > +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg); >> > + >> > +static inline int avr_feature(CPUAVRState *env, int feature) >> > +{ >> > + return (env->features & (1U << feature)) != 0; >> > +} >> > + >> > +static inline void avr_set_feature(CPUAVRState *env, int feature) >> > +{ >> > + env->features |= (1U << feature); >> > +} >> > + >> > +#define cpu_list avr_cpu_list >> > +#define cpu_signal_handler cpu_avr_signal_handler >> > +#define cpu_mmu_index avr_cpu_mmu_index >> > + >> > +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) >> > +{ >> > + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; >> > +} >> > + >> > +void avr_cpu_tcg_init(void); >> > + >> > +void avr_cpu_list(void); >> > +int cpu_avr_exec(CPUState *cpu); >> > +int cpu_avr_signal_handler(int host_signum, void *pinfo, void *puc); >> > +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int size, >> > + int rw, int mmu_idx); >> > +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t *buf, >> > + int len, bool is_write); >> > + >> > +enum { >> > + TB_FLAGS_FULL_ACCESS = 1, >> > + TB_FLAGS_SKIP = 2, >> > +}; >> > + >> > +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, target_ulong >> *pc, >> > + target_ulong *cs_base, uint32_t >> *pflags) >> > +{ >> > + uint32_t flags = 0; >> > + >> > + *pc = env->pc_w * 2; >> > + *cs_base = 0; >> > + >> > + if (env->fullacc) { >> > + flags |= TB_FLAGS_FULL_ACCESS; >> > + } >> > + if (env->skip) { >> > + flags |= TB_FLAGS_SKIP; >> > + } >> > + >> > + *pflags = flags; >> > +} >> > + >> > +static inline int cpu_interrupts_enabled(CPUAVRState *env) >> > +{ >> > + return env->sregI != 0; >> > +} >> > + >> > +static inline uint8_t cpu_get_sreg(CPUAVRState *env) >> > +{ >> > + uint8_t sreg; >> > + sreg = (env->sregC & 0x01) << 0 >> > + | (env->sregZ == 0 ? 1 : 0) << 1 >> > + | (env->sregN) << 2 >> > + | (env->sregV) << 3 >> > + | (env->sregS) << 4 >> > + | (env->sregH) << 5 >> > + | (env->sregT) << 6 >> > + | (env->sregI) << 7; >> > + return sreg; >> > +} >> > + >> > +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) >> > +{ >> > + env->sregC = (sreg >> 0) & 0x01; >> > + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; >> > + env->sregN = (sreg >> 2) & 0x01; >> > + env->sregV = (sreg >> 3) & 0x01; >> > + env->sregS = (sreg >> 4) & 0x01; >> > + env->sregH = (sreg >> 5) & 0x01; >> > + env->sregT = (sreg >> 6) & 0x01; >> > + env->sregI = (sreg >> 7) & 0x01; >> > +} >> > + >> > +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, >> > + MMUAccessType access_type, int mmu_idx, >> > + bool probe, uintptr_t retaddr); >> > + >> > +typedef CPUAVRState CPUArchState; >> > +typedef AVRCPU ArchCPU; >> > + >> > +#include "exec/cpu-all.h" >> > + >> > +#endif /* !defined (CPU_AVR_H) */ >> > diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c >> > new file mode 100644 >> > index 0000000000..537dc7226e >> > --- /dev/null >> > +++ b/target/avr/gdbstub.c >> > @@ -0,0 +1,85 @@ >> > +/* >> > + * QEMU AVR CPU >> > + * >> > + * Copyright (c) 2016 Michael Rolnik >> > + * >> > + * 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 >> > + * version 2.1 of the License, or (at your option) any later version. >> > + * >> > + * 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/lgpl-2.1.html> >> > + */ >> > + >> > +#include "qemu/osdep.h" >> > +#include "qemu-common.h" >> > +#include "exec/gdbstub.h" >> > + >> > +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + /* R */ >> > + if (n < 32) { >> > + return gdb_get_reg8(mem_buf, env->r[n]); >> > + } >> > + >> > + /* SREG */ >> > + if (n == 32) { >> > + uint8_t sreg = cpu_get_sreg(env); >> > + >> > + return gdb_get_reg8(mem_buf, sreg); >> > + } >> > + >> > + /* SP */ >> > + if (n == 33) { >> > + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); >> > + } >> > + >> > + /* PC */ >> > + if (n == 34) { >> > + return gdb_get_reg32(mem_buf, env->pc_w * 2); >> > + } >> > + >> > + return 0; >> > +} >> > + >> > +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n) >> > +{ >> > + AVRCPU *cpu = AVR_CPU(cs); >> > + CPUAVRState *env = &cpu->env; >> > + >> > + /* R */ >> > + if (n < 32) { >> > + env->r[n] = *mem_buf; >> > + return 1; >> > + } >> > + >> > + /* SREG */ >> > + if (n == 32) { >> > + cpu_set_sreg(env, *mem_buf); >> > + return 1; >> > + } >> > + >> > + /* SP */ >> > + if (n == 33) { >> > + env->sp = lduw_p(mem_buf); >> > + return 2; >> > + } >> > + >> > + /* PC */ >> > + if (n == 34) { >> > + env->pc_w = ldl_p(mem_buf) / 2; >> > + return 4; >> > + } >> > + >> > + return 0; >> > +} >> > diff --git a/target/avr/machine.c b/target/avr/machine.c >> > new file mode 100644 >> > index 0000000000..389b5eb95d >> > --- /dev/null >> > +++ b/target/avr/machine.c >> > @@ -0,0 +1,123 @@ >> > +/* >> > + * QEMU AVR CPU >> > + * >> > + * Copyright (c) 2016 Michael Rolnik >> > + * >> > + * 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 >> > + * version 2.1 of the License, or (at your option) any later version. >> > + * >> > + * 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/lgpl-2.1.html> >> > + */ >> > + >> > +#include "qemu/osdep.h" >> > +#include "hw/hw.h" >> > +#include "cpu.h" >> > +#include "hw/boards.h" >> > +#include "migration/qemu-file.h" >> > + >> > +static int get_sreg(QEMUFile *f, void *opaque, size_t size, >> > + const VMStateField *field) >> > +{ >> > + CPUAVRState *env = opaque; >> > + uint8_t sreg; >> > + >> > + sreg = qemu_get_byte(f); >> > + cpu_set_sreg(env, sreg); >> > + return 0; >> > +} >> > + >> > +static int put_sreg( >> > + QEMUFile *f, void *opaque, size_t size, >> > + const VMStateField *field, QJSON *vmdesc) >> > +{ >> > + CPUAVRState *env = opaque; >> > + uint8_t sreg = cpu_get_sreg(env); >> > + >> > + qemu_put_byte(f, sreg); >> > + return 0; >> > +} >> > + >> > +static const VMStateInfo vms_sreg = { >> > + .name = "sreg", >> > + .get = get_sreg, >> > + .put = put_sreg, >> > +}; >> > + >> > +static int get_segment( >> > + QEMUFile *f, void *opaque, size_t size, const VMStateField *field) >> > +{ >> > + uint32_t *ramp = opaque; >> > + uint8_t temp; >> > + >> > + temp = qemu_get_byte(f); >> > + *ramp = ((uint32_t)temp) << 16; >> > + return 0; >> > +} >> > + >> > +static int put_segment( >> > + QEMUFile *f, void *opaque, size_t size, >> > + const VMStateField *field, QJSON *vmdesc) >> > +{ >> > + uint32_t *ramp = opaque; >> > + uint8_t temp = *ramp >> 16; >> > + >> > + qemu_put_byte(f, temp); >> > + return 0; >> > +} >> > + >> > +static const VMStateInfo vms_rampD = { >> > + .name = "rampD", >> > + .get = get_segment, >> > + .put = put_segment, >> > +}; >> > +static const VMStateInfo vms_rampX = { >> > + .name = "rampX", >> > + .get = get_segment, >> > + .put = put_segment, >> > +}; >> > +static const VMStateInfo vms_rampY = { >> > + .name = "rampY", >> > + .get = get_segment, >> > + .put = put_segment, >> > +}; >> > +static const VMStateInfo vms_rampZ = { >> > + .name = "rampZ", >> > + .get = get_segment, >> > + .put = put_segment, >> > +}; >> > +static const VMStateInfo vms_eind = { >> > + .name = "eind", >> > + .get = get_segment, >> > + .put = put_segment, >> > +}; >> > + >> > +const VMStateDescription vms_avr_cpu = { >> > + .name = "cpu", >> > + .version_id = 0, >> > + .minimum_version_id = 0, >> > + .fields = (VMStateField[]) { >> > + VMSTATE_UINT32(env.pc_w, AVRCPU), >> > + VMSTATE_UINT32(env.sp, AVRCPU), >> > + VMSTATE_UINT32(env.skip, AVRCPU), >> > + >> > + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), >> > + >> > + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), >> > + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), >> > + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), >> > + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), >> > + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), >> > + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), >> > + >> > + VMSTATE_END_OF_LIST() >> > + } >> > +}; >> >> > > -- > Best Regards, > Michael Rolnik >
On Thu, 27 Jun 2019 21:51:23 +0300 Michael Rolnik <mrolnik@gmail.com> wrote: > I meant that if I take the proposed code it fails in the mentioned way > > Sent from my cell phone, please ignore typos > > On Thu, Jun 27, 2019, 7:02 PM Michael Rolnik <mrolnik@gmail.com> wrote: > > > Hi Igor, > > > > if I run `make check` it fails with > > qemu-system-rx: unable to find CPU model 'rx62n' > > or > > qemu-system-avr: unable to find CPU model 'avr6' suggested change means that one has to user type name as is for -cpu argument. so test should be amended to account for that, something like that should be sufficient: diff --git a/tests/machine-none-test.c b/tests/machine-none-test.c index 361927bb76..d0907e6bd4 100644 --- a/tests/machine-none-test.c +++ b/tests/machine-none-test.c @@ -27,7 +27,7 @@ static struct arch2cpu cpus_map[] = { /* tested targets list */ { "arm", "cortex-a15" }, { "aarch64", "cortex-a57" }, - { "avr", "avr6" }, + { "avr", "avr6-avr-cpu" }, { "x86_64", "qemu64,apic-id=0" }, { "i386", "qemu32,apic-id=0" }, { "alpha", "ev67" }, > > > > On Thu, Jun 27, 2019 at 2:28 PM Igor Mammedov <imammedo@redhat.com> wrote: > > > >> On Thu, 27 Jun 2019 08:27:44 +0300 > >> Michael Rolnik <mrolnik@gmail.com> wrote: > >> > >> > From: Sarah Harris <S.E.Harris@kent.ac.uk> > >> > > >> > This includes: > >> > - CPU data structures > >> > - object model classes and functions > >> > - migration functions > >> > - GDB hooks > >> > > >> > Signed-off-by: Michael Rolnik <mrolnik@gmail.com> > >> > --- > >> > gdb-xml/avr-cpu.xml | 49 ++++ > >> > target/avr/cpu-param.h | 37 +++ > >> > target/avr/cpu.c | 599 +++++++++++++++++++++++++++++++++++++++++ > >> > target/avr/cpu.h | 283 +++++++++++++++++++ > >> > target/avr/gdbstub.c | 85 ++++++ > >> > target/avr/machine.c | 123 +++++++++ > >> > 6 files changed, 1176 insertions(+) > >> > create mode 100644 gdb-xml/avr-cpu.xml > >> > create mode 100644 target/avr/cpu-param.h > >> > create mode 100644 target/avr/cpu.c > >> > create mode 100644 target/avr/cpu.h > >> > create mode 100644 target/avr/gdbstub.c > >> > create mode 100644 target/avr/machine.c > >> > > >> > diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml > >> > new file mode 100644 > >> > index 0000000000..c4747f5b40 > >> > --- /dev/null > >> > +++ b/gdb-xml/avr-cpu.xml > >> > @@ -0,0 +1,49 @@ > >> > +<?xml version="1.0"?> > >> > +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. > >> > + > >> > + Copying and distribution of this file, with or without > >> modification, > >> > + are permitted in any medium without royalty provided the copyright > >> > + notice and this notice are preserved. --> > >> > + > >> > +<!-- Register numbers are hard-coded in order to maintain backward > >> > + compatibility with older versions of tools that didn't use xml > >> > + register descriptions. --> > >> > + > >> > +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> > >> > +<feature name="org.gnu.gdb.riscv.cpu"> > >> > + <reg name="r0" bitsize="8" type="int" regnum="0"/> > >> > + <reg name="r1" bitsize="8" type="int"/> > >> > + <reg name="r2" bitsize="8" type="int"/> > >> > + <reg name="r3" bitsize="8" type="int"/> > >> > + <reg name="r4" bitsize="8" type="int"/> > >> > + <reg name="r5" bitsize="8" type="int"/> > >> > + <reg name="r6" bitsize="8" type="int"/> > >> > + <reg name="r7" bitsize="8" type="int"/> > >> > + <reg name="r8" bitsize="8" type="int"/> > >> > + <reg name="r9" bitsize="8" type="int"/> > >> > + <reg name="r10" bitsize="8" type="int"/> > >> > + <reg name="r11" bitsize="8" type="int"/> > >> > + <reg name="r12" bitsize="8" type="int"/> > >> > + <reg name="r13" bitsize="8" type="int"/> > >> > + <reg name="r14" bitsize="8" type="int"/> > >> > + <reg name="r15" bitsize="8" type="int"/> > >> > + <reg name="r16" bitsize="8" type="int"/> > >> > + <reg name="r17" bitsize="8" type="int"/> > >> > + <reg name="r18" bitsize="8" type="int"/> > >> > + <reg name="r19" bitsize="8" type="int"/> > >> > + <reg name="r20" bitsize="8" type="int"/> > >> > + <reg name="r21" bitsize="8" type="int"/> > >> > + <reg name="r22" bitsize="8" type="int"/> > >> > + <reg name="r23" bitsize="8" type="int"/> > >> > + <reg name="r24" bitsize="8" type="int"/> > >> > + <reg name="r25" bitsize="8" type="int"/> > >> > + <reg name="r26" bitsize="8" type="int"/> > >> > + <reg name="r27" bitsize="8" type="int"/> > >> > + <reg name="r28" bitsize="8" type="int"/> > >> > + <reg name="r29" bitsize="8" type="int"/> > >> > + <reg name="r30" bitsize="8" type="int"/> > >> > + <reg name="r31" bitsize="8" type="int"/> > >> > + <reg name="sreg" bitsize="8" type="int"/> > >> > + <reg name="sp" bitsize="8" type="int"/> > >> > + <reg name="pc" bitsize="8" type="int"/> > >> > +</feature> > >> > diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h > >> > new file mode 100644 > >> > index 0000000000..5bbf985726 > >> > --- /dev/null > >> > +++ b/target/avr/cpu-param.h > >> > @@ -0,0 +1,37 @@ > >> > +/* > >> > + * QEMU AVR CPU > >> > + * > >> > + * Copyright (c) 2016 Michael Rolnik > >> > + * > >> > + * 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 > >> > + * version 2.1 of the License, or (at your option) any later version. > >> > + * > >> > + * 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/lgpl-2.1.html> > >> > + */ > >> > + > >> > +#ifndef AVR_CPU_PARAM_H > >> > +#define AVR_CPU_PARAM_H 1 > >> > + > >> > +#define TARGET_LONG_BITS 32 > >> > +/* > >> > + * TARGET_PAGE_BITS cannot be more than 8 bits because > >> > + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, and > >> they > >> > + * should be implemented as a device and not memory > >> > + * 2. SRAM starts at the address 0x0100 > >> > + */ > >> > +#define TARGET_PAGE_BITS 8 > >> > +#define TARGET_PHYS_ADDR_SPACE_BITS 24 > >> > +#define TARGET_VIRT_ADDR_SPACE_BITS 24 > >> > +#define NB_MMU_MODES 2 > >> > + > >> > + > >> > +#endif > >> > diff --git a/target/avr/cpu.c b/target/avr/cpu.c > >> > new file mode 100644 > >> > index 0000000000..142fe54524 > >> > --- /dev/null > >> > +++ b/target/avr/cpu.c > >> > @@ -0,0 +1,599 @@ > >> > +/* > >> > + * QEMU AVR CPU > >> > + * > >> > + * Copyright (c) 2016 Michael Rolnik > >> > + * > >> > + * 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 > >> > + * version 2.1 of the License, or (at your option) any later version. > >> > + * > >> > + * 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/lgpl-2.1.html> > >> > + */ > >> > + > >> > +#include "qemu/osdep.h" > >> > +#include "qemu/qemu-print.h" > >> > +#include "qemu/log.h" > >> > +#include "cpu.h" > >> > +#include "exec/exec-all.h" > >> > +#include "qapi/error.h" > >> > +#include "hw/qdev-properties.h" > >> > +#include "migration/vmstate.h" > >> > + > >> > +static void avr_cpu_set_pc(CPUState *cs, vaddr value) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + > >> > + cpu->env.pc_w = value / 2; /* internally PC points to words */ > >> > +} > >> > + > >> > +static bool avr_cpu_has_work(CPUState *cs) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | > >> CPU_INTERRUPT_RESET)) > >> > + && cpu_interrupts_enabled(env); > >> > +} > >> > + > >> > +static void avr_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock > >> *tb) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + env->pc_w = tb->pc / 2; /* internally PC points to words */ > >> > +} > >> > + > >> > +static void avr_cpu_reset(CPUState *cs) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + mcc->parent_reset(cs); > >> > + > >> > + env->pc_w = 0; > >> > + env->sregI = 1; > >> > + env->sregC = 0; > >> > + env->sregZ = 0; > >> > + env->sregN = 0; > >> > + env->sregV = 0; > >> > + env->sregS = 0; > >> > + env->sregH = 0; > >> > + env->sregT = 0; > >> > + > >> > + env->rampD = 0; > >> > + env->rampX = 0; > >> > + env->rampY = 0; > >> > + env->rampZ = 0; > >> > + env->eind = 0; > >> > + env->sp = 0; > >> > + > >> > + env->skip = 0; > >> > + > >> > + memset(env->r, 0, sizeof(env->r)); > >> > + > >> > + tlb_flush(cs); > >> > +} > >> > + > >> > +static void avr_cpu_disas_set_info(CPUState *cpu, disassemble_info > >> *info) > >> > +{ > >> > + info->mach = bfd_arch_avr; > >> > + info->print_insn = NULL; > >> > +} > >> > + > >> > +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) > >> > +{ > >> > + CPUState *cs = CPU(dev); > >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); > >> > + Error *local_err = NULL; > >> > + > >> > + cpu_exec_realizefn(cs, &local_err); > >> > + if (local_err != NULL) { > >> > + error_propagate(errp, local_err); > >> > + return; > >> > + } > >> > + qemu_init_vcpu(cs); > >> > + cpu_reset(cs); > >> > + > >> > + mcc->parent_realize(dev, errp); > >> > +} > >> > + > >> > +static void avr_cpu_set_int(void *opaque, int irq, int level) > >> > +{ > >> > + AVRCPU *cpu = opaque; > >> > + CPUAVRState *env = &cpu->env; > >> > + CPUState *cs = CPU(cpu); > >> > + > >> > + uint64_t mask = (1ull << irq); > >> > + if (level) { > >> > + env->intsrc |= mask; > >> > + cpu_interrupt(cs, CPU_INTERRUPT_HARD); > >> > + } else { > >> > + env->intsrc &= ~mask; > >> > + if (env->intsrc == 0) { > >> > + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); > >> > + } > >> > + } > >> > +} > >> > + > >> > +static void avr_cpu_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + > >> > + cpu_set_cpustate_pointers(cpu); > >> > + > >> > +#ifndef CONFIG_USER_ONLY > >> > + /* Set the number of interrupts supported by the CPU. */ > >> > + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); > >> > +#endif > >> > +} > >> > + > >> > +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) > >> > +{ > >> > + ObjectClass *oc; > >> > + char *typename; > >> > + char **cpuname; > >> > + > >> > + cpuname = g_strsplit(cpu_model, ",", 1); > >> you do not need this, parse_cpu_option() does it for you. > >> > >> > + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), cpuname[0]); > >> > + oc = object_class_by_name(typename); > >> Also just use cpu_model directly for lookup > >> for example see '[Qemu-devel] [PATCH RESEND v21 06/21] target/rx: CPU > >> definition' > >> and simplify avr_cpu_list_entry() correspondingly. > >> > >> > >> > + g_strfreev(cpuname); > >> > + g_free(typename); > >> > + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || > >> > + object_class_is_abstract(oc)) { > >> > + return NULL; > >> > + } > >> > + return oc; > >> > +} > >> > + > >> > +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + CPUAVRState *env = &cpu->env; > >> > + int i; > >> > + > >> > + qemu_fprintf(f, "\n"); > >> > + qemu_fprintf(f, "PC: %06x\n", env->pc_w); > >> > + qemu_fprintf(f, "SP: %04x\n", env->sp); > >> > + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); > >> > + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); > >> > + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); > >> > + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); > >> > + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); > >> > + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); > >> > + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); > >> > + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); > >> > + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", > >> > + env->sregI ? 'I' : '-', > >> > + env->sregT ? 'T' : '-', > >> > + env->sregH ? 'H' : '-', > >> > + env->sregS ? 'S' : '-', > >> > + env->sregV ? 'V' : '-', > >> > + env->sregN ? '-' : 'N', /* Zf has negative > >> logic */ > >> > + env->sregZ ? 'Z' : '-', > >> > + env->sregC ? 'I' : '-'); > >> > + qemu_fprintf(f, "SKIP: %02x\n", env->skip); > >> > + > >> > + qemu_fprintf(f, "\n"); > >> > + for (i = 0; i < ARRAY_SIZE(env->r); i++) { > >> > + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); > >> > + > >> > + if ((i % 8) == 7) { > >> > + qemu_fprintf(f, "\n"); > >> > + } > >> > + } > >> > + qemu_fprintf(f, "\n"); > >> > +} > >> > + > >> > +static void avr_cpu_class_init(ObjectClass *oc, void *data) > >> > +{ > >> > + DeviceClass *dc = DEVICE_CLASS(oc); > >> > + CPUClass *cc = CPU_CLASS(oc); > >> > + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); > >> > + > >> > + mcc->parent_realize = dc->realize; > >> > + dc->realize = avr_cpu_realizefn; > >> > + > >> > + mcc->parent_reset = cc->reset; > >> > + cc->reset = avr_cpu_reset; > >> > + > >> > + cc->class_by_name = avr_cpu_class_by_name; > >> > + > >> > + cc->has_work = avr_cpu_has_work; > >> > + cc->do_interrupt = avr_cpu_do_interrupt; > >> > + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; > >> > + cc->dump_state = avr_cpu_dump_state; > >> > + cc->set_pc = avr_cpu_set_pc; > >> > +#if !defined(CONFIG_USER_ONLY) > >> > + cc->memory_rw_debug = avr_cpu_memory_rw_debug; > >> > +#endif > >> > +#ifdef CONFIG_USER_ONLY > >> > + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; > >> > +#else > >> > + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; > >> > + cc->vmsd = &vms_avr_cpu; > >> > +#endif > >> > + cc->disas_set_info = avr_cpu_disas_set_info; > >> > + cc->tlb_fill = avr_cpu_tlb_fill; > >> > + cc->tcg_initialize = avr_cpu_tcg_init; > >> > + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; > >> > + cc->gdb_read_register = avr_cpu_gdb_read_register; > >> > + cc->gdb_write_register = avr_cpu_gdb_write_register; > >> > + cc->gdb_num_core_regs = 35; > >> > + cc->gdb_core_xml_file = "avr-cpu.xml"; > >> > +} > >> > + > >> > +static void avr_avr1_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > +} > >> > + > >> > +static void avr_avr2_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > +} > >> > + > >> > +static void avr_avr25_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > +} > >> > + > >> > +static void avr_avr3_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > +} > >> > + > >> > +static void avr_avr31_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > +} > >> > + > >> > +static void avr_avr35_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > +} > >> > + > >> > +static void avr_avr4_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > +} > >> > + > >> > +static void avr_avr5_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > +} > >> > + > >> > +static void avr_avr51_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > +} > >> > + > >> > +static void avr_avr6_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > +} > >> > + > >> > +static void avr_xmega2_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > >> > +} > >> > + > >> > +static void avr_xmega4_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > >> > +} > >> > + > >> > +static void avr_xmega5_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > >> > +} > >> > + > >> > +static void avr_xmega6_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > >> > +} > >> > + > >> > +static void avr_xmega7_initfn(Object *obj) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(obj); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > >> > + > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > >> > +} > >> > + > >> > +typedef struct AVRCPUInfo { > >> > + const char *name; > >> > + void (*initfn)(Object *obj); > >> > +} AVRCPUInfo; > >> > + > >> > +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) > >> > +{ > >> > + ObjectClass *class_a = (ObjectClass *)a; > >> > + ObjectClass *class_b = (ObjectClass *)b; > >> > + const char *name_a; > >> > + const char *name_b; > >> > + > >> > + name_a = object_class_get_name(class_a); > >> > + name_b = object_class_get_name(class_b); > >> > + > >> > + return strcmp(name_a, name_b); > >> > +} > >> > + > >> > +static void avr_cpu_list_entry(gpointer data, gpointer user_data) > >> > +{ > >> > + const char *typename = object_class_get_name(OBJECT_CLASS(data)); > >> > + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); > >> > + qemu_printf("%.*s\n", len, typename); > >> > +} > >> > + > >> > +void avr_cpu_list(void) > >> > +{ > >> > + GSList *list; > >> > + list = object_class_get_list(TYPE_AVR_CPU, false); > >> > + list = g_slist_sort(list, avr_cpu_list_compare); > >> > + qemu_printf("Available CPUs:\n"); > >> > + g_slist_foreach(list, avr_cpu_list_entry, NULL); > >> > + g_slist_free(list); > >> > +} > >> > + > >> > +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ > >> > + { \ > >> > + .parent = TYPE_AVR_CPU, \ > >> > + .instance_init = initfn, \ > >> > + .name = AVR_CPU_TYPE_NAME(model), \ > >> > + } > >> > + > >> > +static const TypeInfo avr_cpu_type_info[] = { > >> > + { > >> > + .name = TYPE_AVR_CPU, > >> > + .parent = TYPE_CPU, > >> > + .instance_size = sizeof(AVRCPU), > >> > + .instance_init = avr_cpu_initfn, > >> > + .class_size = sizeof(AVRCPUClass), > >> > + .class_init = avr_cpu_class_init, > >> > + .abstract = true, > >> > + }, > >> > + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), > >> > + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), > >> > + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), > >> > + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), > >> > + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), > >> > + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), > >> > + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), > >> > +}; > >> > + > >> > +DEFINE_TYPES(avr_cpu_type_info) > >> > diff --git a/target/avr/cpu.h b/target/avr/cpu.h > >> > new file mode 100644 > >> > index 0000000000..3f9a803193 > >> > --- /dev/null > >> > +++ b/target/avr/cpu.h > >> > @@ -0,0 +1,283 @@ > >> > +/* > >> > + * QEMU AVR CPU > >> > + * > >> > + * Copyright (c) 2016 Michael Rolnik > >> > + * > >> > + * 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 > >> > + * version 2.1 of the License, or (at your option) any later version. > >> > + * > >> > + * 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/lgpl-2.1.html> > >> > + */ > >> > + > >> > +#ifndef CPU_AVR_H > >> > +#define CPU_AVR_H > >> > + > >> > +#include "qemu-common.h" > >> > +#include "qom/cpu.h" > >> > +#include "exec/cpu-defs.h" > >> > +#include "fpu/softfloat.h" > >> > + > >> > +#define TCG_GUEST_DEFAULT_MO 0 > >> > + > >> > +#define TYPE_AVR_CPU "avr-cpu" > >> > + > >> > +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU > >> > +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) > >> > +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU > >> > + > >> > +/* > >> > + * AVR has two memory spaces, data & code. > >> > + * e.g. both have 0 address > >> > + * ST/LD instructions access data space > >> > + * LPM/SPM and instruction fetching access code memory space > >> > + */ > >> > +#define MMU_CODE_IDX 0 > >> > +#define MMU_DATA_IDX 1 > >> > + > >> > +#define EXCP_RESET 1 > >> > +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) > >> > + > >> > +/* Number of CPU registers */ > >> > +#define NO_CPU_REGISTERS 32 > >> > +/* Number of IO registers accessible by ld/st/in/out */ > >> > +#define NO_IO_REGISTERS 64 > >> > + > >> > +/* > >> > + * Offsets of AVR memory regions in host memory space. > >> > + * > >> > + * This is needed because the AVR has separate code and data address > >> > + * spaces that both have start from zero but have to go somewhere in > >> > + * host memory. > >> > + * > >> > + * It's also useful to know where some things are, like the IO > >> registers. > >> > + */ > >> > +/* Flash program memory */ > >> > +#define OFFSET_CODE 0x00000000 > >> > +/* CPU registers, IO registers, and SRAM */ > >> > +#define OFFSET_DATA 0x00800000 > >> > +/* CPU registers specifically, these are mapped at the start of data */ > >> > +#define OFFSET_CPU_REGISTERS OFFSET_DATA > >> > +/* > >> > + * IO registers, including status register, stack pointer, and memory > >> > + * mapped peripherals, mapped just after CPU registers > >> > + */ > >> > +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) > >> > + > >> > +enum avr_features { > >> > + AVR_FEATURE_SRAM, > >> > + > >> > + AVR_FEATURE_1_BYTE_PC, > >> > + AVR_FEATURE_2_BYTE_PC, > >> > + AVR_FEATURE_3_BYTE_PC, > >> > + > >> > + AVR_FEATURE_1_BYTE_SP, > >> > + AVR_FEATURE_2_BYTE_SP, > >> > + > >> > + AVR_FEATURE_BREAK, > >> > + AVR_FEATURE_DES, > >> > + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ > >> > + > >> > + AVR_FEATURE_EIJMP_EICALL, > >> > + AVR_FEATURE_IJMP_ICALL, > >> > + AVR_FEATURE_JMP_CALL, > >> > + > >> > + AVR_FEATURE_ADIW_SBIW, > >> > + > >> > + AVR_FEATURE_SPM, > >> > + AVR_FEATURE_SPMX, > >> > + > >> > + AVR_FEATURE_ELPMX, > >> > + AVR_FEATURE_ELPM, > >> > + AVR_FEATURE_LPMX, > >> > + AVR_FEATURE_LPM, > >> > + > >> > + AVR_FEATURE_MOVW, > >> > + AVR_FEATURE_MUL, > >> > + AVR_FEATURE_RAMPD, > >> > + AVR_FEATURE_RAMPX, > >> > + AVR_FEATURE_RAMPY, > >> > + AVR_FEATURE_RAMPZ, > >> > +}; > >> > + > >> > +typedef struct CPUAVRState CPUAVRState; > >> > + > >> > +struct CPUAVRState { > >> > + uint32_t pc_w; /* 0x003fffff up to 22 bits */ > >> > + > >> > + uint32_t sregC; /* 0x00000001 1 bits */ > >> > + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ > >> > + /* 0=flag set, >0=flag cleared */ > >> > + uint32_t sregN; /* 0x00000001 1 bits */ > >> > + uint32_t sregV; /* 0x00000001 1 bits */ > >> > + uint32_t sregS; /* 0x00000001 1 bits */ > >> > + uint32_t sregH; /* 0x00000001 1 bits */ > >> > + uint32_t sregT; /* 0x00000001 1 bits */ > >> > + uint32_t sregI; /* 0x00000001 1 bits */ > >> > + > >> > + uint32_t rampD; /* 0x00ff0000 8 bits */ > >> > + uint32_t rampX; /* 0x00ff0000 8 bits */ > >> > + uint32_t rampY; /* 0x00ff0000 8 bits */ > >> > + uint32_t rampZ; /* 0x00ff0000 8 bits */ > >> > + uint32_t eind; /* 0x00ff0000 8 bits */ > >> > + > >> > + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ > >> > + uint32_t sp; /* 16 bits */ > >> > + > >> > + uint32_t skip; /* if set skip instruction */ > >> > + > >> > + uint64_t intsrc; /* interrupt sources */ > >> > + bool fullacc; /* CPU/MEM if true MEM only otherwise */ > >> > + > >> > + uint32_t features; > >> > +}; > >> > + > >> > +#define AVR_CPU_CLASS(klass) \ > >> > + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) > >> > +#define AVR_CPU(obj) \ > >> > + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) > >> > +#define AVR_CPU_GET_CLASS(obj) \ > >> > + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) > >> > + > >> > +/** > >> > + * AVRCPUClass: > >> > + * @parent_realize: The parent class' realize handler. > >> > + * @parent_reset: The parent class' reset handler. > >> > + * @vr: Version Register value. > >> > + * > >> > + * A AVR CPU model. > >> > + */ > >> > +typedef struct AVRCPUClass { > >> > + /*< private >*/ > >> > + CPUClass parent_class; > >> > + /*< public >*/ > >> > + DeviceRealize parent_realize; > >> > + void (*parent_reset)(CPUState *cpu); > >> > +} AVRCPUClass; > >> > + > >> > +/** > >> > + * AVRCPU: > >> > + * @env: #CPUAVRState > >> > + * > >> > + * A AVR CPU. > >> > + */ > >> > +typedef struct AVRCPU { > >> > + /*< private >*/ > >> > + CPUState parent_obj; > >> > + /*< public >*/ > >> > + > >> > + CPUNegativeOffsetState neg; > >> > + CPUAVRState env; > >> > +} AVRCPU; > >> > + > >> > +#ifndef CONFIG_USER_ONLY > >> > +extern const struct VMStateDescription vms_avr_cpu; > >> > +#endif > >> > + > >> > +void avr_cpu_do_interrupt(CPUState *cpu); > >> > +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); > >> > +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); > >> > +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg); > >> > +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg); > >> > + > >> > +static inline int avr_feature(CPUAVRState *env, int feature) > >> > +{ > >> > + return (env->features & (1U << feature)) != 0; > >> > +} > >> > + > >> > +static inline void avr_set_feature(CPUAVRState *env, int feature) > >> > +{ > >> > + env->features |= (1U << feature); > >> > +} > >> > + > >> > +#define cpu_list avr_cpu_list > >> > +#define cpu_signal_handler cpu_avr_signal_handler > >> > +#define cpu_mmu_index avr_cpu_mmu_index > >> > + > >> > +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) > >> > +{ > >> > + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; > >> > +} > >> > + > >> > +void avr_cpu_tcg_init(void); > >> > + > >> > +void avr_cpu_list(void); > >> > +int cpu_avr_exec(CPUState *cpu); > >> > +int cpu_avr_signal_handler(int host_signum, void *pinfo, void *puc); > >> > +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int size, > >> > + int rw, int mmu_idx); > >> > +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t *buf, > >> > + int len, bool is_write); > >> > + > >> > +enum { > >> > + TB_FLAGS_FULL_ACCESS = 1, > >> > + TB_FLAGS_SKIP = 2, > >> > +}; > >> > + > >> > +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, target_ulong > >> *pc, > >> > + target_ulong *cs_base, uint32_t > >> *pflags) > >> > +{ > >> > + uint32_t flags = 0; > >> > + > >> > + *pc = env->pc_w * 2; > >> > + *cs_base = 0; > >> > + > >> > + if (env->fullacc) { > >> > + flags |= TB_FLAGS_FULL_ACCESS; > >> > + } > >> > + if (env->skip) { > >> > + flags |= TB_FLAGS_SKIP; > >> > + } > >> > + > >> > + *pflags = flags; > >> > +} > >> > + > >> > +static inline int cpu_interrupts_enabled(CPUAVRState *env) > >> > +{ > >> > + return env->sregI != 0; > >> > +} > >> > + > >> > +static inline uint8_t cpu_get_sreg(CPUAVRState *env) > >> > +{ > >> > + uint8_t sreg; > >> > + sreg = (env->sregC & 0x01) << 0 > >> > + | (env->sregZ == 0 ? 1 : 0) << 1 > >> > + | (env->sregN) << 2 > >> > + | (env->sregV) << 3 > >> > + | (env->sregS) << 4 > >> > + | (env->sregH) << 5 > >> > + | (env->sregT) << 6 > >> > + | (env->sregI) << 7; > >> > + return sreg; > >> > +} > >> > + > >> > +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) > >> > +{ > >> > + env->sregC = (sreg >> 0) & 0x01; > >> > + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; > >> > + env->sregN = (sreg >> 2) & 0x01; > >> > + env->sregV = (sreg >> 3) & 0x01; > >> > + env->sregS = (sreg >> 4) & 0x01; > >> > + env->sregH = (sreg >> 5) & 0x01; > >> > + env->sregT = (sreg >> 6) & 0x01; > >> > + env->sregI = (sreg >> 7) & 0x01; > >> > +} > >> > + > >> > +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, > >> > + MMUAccessType access_type, int mmu_idx, > >> > + bool probe, uintptr_t retaddr); > >> > + > >> > +typedef CPUAVRState CPUArchState; > >> > +typedef AVRCPU ArchCPU; > >> > + > >> > +#include "exec/cpu-all.h" > >> > + > >> > +#endif /* !defined (CPU_AVR_H) */ > >> > diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c > >> > new file mode 100644 > >> > index 0000000000..537dc7226e > >> > --- /dev/null > >> > +++ b/target/avr/gdbstub.c > >> > @@ -0,0 +1,85 @@ > >> > +/* > >> > + * QEMU AVR CPU > >> > + * > >> > + * Copyright (c) 2016 Michael Rolnik > >> > + * > >> > + * 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 > >> > + * version 2.1 of the License, or (at your option) any later version. > >> > + * > >> > + * 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/lgpl-2.1.html> > >> > + */ > >> > + > >> > +#include "qemu/osdep.h" > >> > +#include "qemu-common.h" > >> > +#include "exec/gdbstub.h" > >> > + > >> > +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + /* R */ > >> > + if (n < 32) { > >> > + return gdb_get_reg8(mem_buf, env->r[n]); > >> > + } > >> > + > >> > + /* SREG */ > >> > + if (n == 32) { > >> > + uint8_t sreg = cpu_get_sreg(env); > >> > + > >> > + return gdb_get_reg8(mem_buf, sreg); > >> > + } > >> > + > >> > + /* SP */ > >> > + if (n == 33) { > >> > + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); > >> > + } > >> > + > >> > + /* PC */ > >> > + if (n == 34) { > >> > + return gdb_get_reg32(mem_buf, env->pc_w * 2); > >> > + } > >> > + > >> > + return 0; > >> > +} > >> > + > >> > +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n) > >> > +{ > >> > + AVRCPU *cpu = AVR_CPU(cs); > >> > + CPUAVRState *env = &cpu->env; > >> > + > >> > + /* R */ > >> > + if (n < 32) { > >> > + env->r[n] = *mem_buf; > >> > + return 1; > >> > + } > >> > + > >> > + /* SREG */ > >> > + if (n == 32) { > >> > + cpu_set_sreg(env, *mem_buf); > >> > + return 1; > >> > + } > >> > + > >> > + /* SP */ > >> > + if (n == 33) { > >> > + env->sp = lduw_p(mem_buf); > >> > + return 2; > >> > + } > >> > + > >> > + /* PC */ > >> > + if (n == 34) { > >> > + env->pc_w = ldl_p(mem_buf) / 2; > >> > + return 4; > >> > + } > >> > + > >> > + return 0; > >> > +} > >> > diff --git a/target/avr/machine.c b/target/avr/machine.c > >> > new file mode 100644 > >> > index 0000000000..389b5eb95d > >> > --- /dev/null > >> > +++ b/target/avr/machine.c > >> > @@ -0,0 +1,123 @@ > >> > +/* > >> > + * QEMU AVR CPU > >> > + * > >> > + * Copyright (c) 2016 Michael Rolnik > >> > + * > >> > + * 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 > >> > + * version 2.1 of the License, or (at your option) any later version. > >> > + * > >> > + * 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/lgpl-2.1.html> > >> > + */ > >> > + > >> > +#include "qemu/osdep.h" > >> > +#include "hw/hw.h" > >> > +#include "cpu.h" > >> > +#include "hw/boards.h" > >> > +#include "migration/qemu-file.h" > >> > + > >> > +static int get_sreg(QEMUFile *f, void *opaque, size_t size, > >> > + const VMStateField *field) > >> > +{ > >> > + CPUAVRState *env = opaque; > >> > + uint8_t sreg; > >> > + > >> > + sreg = qemu_get_byte(f); > >> > + cpu_set_sreg(env, sreg); > >> > + return 0; > >> > +} > >> > + > >> > +static int put_sreg( > >> > + QEMUFile *f, void *opaque, size_t size, > >> > + const VMStateField *field, QJSON *vmdesc) > >> > +{ > >> > + CPUAVRState *env = opaque; > >> > + uint8_t sreg = cpu_get_sreg(env); > >> > + > >> > + qemu_put_byte(f, sreg); > >> > + return 0; > >> > +} > >> > + > >> > +static const VMStateInfo vms_sreg = { > >> > + .name = "sreg", > >> > + .get = get_sreg, > >> > + .put = put_sreg, > >> > +}; > >> > + > >> > +static int get_segment( > >> > + QEMUFile *f, void *opaque, size_t size, const VMStateField *field) > >> > +{ > >> > + uint32_t *ramp = opaque; > >> > + uint8_t temp; > >> > + > >> > + temp = qemu_get_byte(f); > >> > + *ramp = ((uint32_t)temp) << 16; > >> > + return 0; > >> > +} > >> > + > >> > +static int put_segment( > >> > + QEMUFile *f, void *opaque, size_t size, > >> > + const VMStateField *field, QJSON *vmdesc) > >> > +{ > >> > + uint32_t *ramp = opaque; > >> > + uint8_t temp = *ramp >> 16; > >> > + > >> > + qemu_put_byte(f, temp); > >> > + return 0; > >> > +} > >> > + > >> > +static const VMStateInfo vms_rampD = { > >> > + .name = "rampD", > >> > + .get = get_segment, > >> > + .put = put_segment, > >> > +}; > >> > +static const VMStateInfo vms_rampX = { > >> > + .name = "rampX", > >> > + .get = get_segment, > >> > + .put = put_segment, > >> > +}; > >> > +static const VMStateInfo vms_rampY = { > >> > + .name = "rampY", > >> > + .get = get_segment, > >> > + .put = put_segment, > >> > +}; > >> > +static const VMStateInfo vms_rampZ = { > >> > + .name = "rampZ", > >> > + .get = get_segment, > >> > + .put = put_segment, > >> > +}; > >> > +static const VMStateInfo vms_eind = { > >> > + .name = "eind", > >> > + .get = get_segment, > >> > + .put = put_segment, > >> > +}; > >> > + > >> > +const VMStateDescription vms_avr_cpu = { > >> > + .name = "cpu", > >> > + .version_id = 0, > >> > + .minimum_version_id = 0, > >> > + .fields = (VMStateField[]) { > >> > + VMSTATE_UINT32(env.pc_w, AVRCPU), > >> > + VMSTATE_UINT32(env.sp, AVRCPU), > >> > + VMSTATE_UINT32(env.skip, AVRCPU), > >> > + > >> > + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), > >> > + > >> > + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), > >> > + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), > >> > + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), > >> > + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), > >> > + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), > >> > + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), > >> > + > >> > + VMSTATE_END_OF_LIST() > >> > + } > >> > +}; > >> > >> > > > > -- > > Best Regards, > > Michael Rolnik > >
Nobody else does it. Richard. What do you think? Sent from my cell phone, please ignore typos On Fri, Jun 28, 2019, 11:36 AM Igor Mammedov <imammedo@redhat.com> wrote: > On Thu, 27 Jun 2019 21:51:23 +0300 > Michael Rolnik <mrolnik@gmail.com> wrote: > > > I meant that if I take the proposed code it fails in the mentioned way > > > > Sent from my cell phone, please ignore typos > > > > On Thu, Jun 27, 2019, 7:02 PM Michael Rolnik <mrolnik@gmail.com> wrote: > > > > > Hi Igor, > > > > > > if I run `make check` it fails with > > > qemu-system-rx: unable to find CPU model 'rx62n' > > > or > > > qemu-system-avr: unable to find CPU model 'avr6' > suggested change means that one has to user type name as is for -cpu > argument. > > so test should be amended to account for that, > something like that should be sufficient: > > diff --git a/tests/machine-none-test.c b/tests/machine-none-test.c > index 361927bb76..d0907e6bd4 100644 > --- a/tests/machine-none-test.c > +++ b/tests/machine-none-test.c > @@ -27,7 +27,7 @@ static struct arch2cpu cpus_map[] = { > /* tested targets list */ > { "arm", "cortex-a15" }, > { "aarch64", "cortex-a57" }, > - { "avr", "avr6" }, > + { "avr", "avr6-avr-cpu" }, > { "x86_64", "qemu64,apic-id=0" }, > { "i386", "qemu32,apic-id=0" }, > { "alpha", "ev67" }, > > > > > > > > On Thu, Jun 27, 2019 at 2:28 PM Igor Mammedov <imammedo@redhat.com> > wrote: > > > > > >> On Thu, 27 Jun 2019 08:27:44 +0300 > > >> Michael Rolnik <mrolnik@gmail.com> wrote: > > >> > > >> > From: Sarah Harris <S.E.Harris@kent.ac.uk> > > >> > > > >> > This includes: > > >> > - CPU data structures > > >> > - object model classes and functions > > >> > - migration functions > > >> > - GDB hooks > > >> > > > >> > Signed-off-by: Michael Rolnik <mrolnik@gmail.com> > > >> > --- > > >> > gdb-xml/avr-cpu.xml | 49 ++++ > > >> > target/avr/cpu-param.h | 37 +++ > > >> > target/avr/cpu.c | 599 > +++++++++++++++++++++++++++++++++++++++++ > > >> > target/avr/cpu.h | 283 +++++++++++++++++++ > > >> > target/avr/gdbstub.c | 85 ++++++ > > >> > target/avr/machine.c | 123 +++++++++ > > >> > 6 files changed, 1176 insertions(+) > > >> > create mode 100644 gdb-xml/avr-cpu.xml > > >> > create mode 100644 target/avr/cpu-param.h > > >> > create mode 100644 target/avr/cpu.c > > >> > create mode 100644 target/avr/cpu.h > > >> > create mode 100644 target/avr/gdbstub.c > > >> > create mode 100644 target/avr/machine.c > > >> > > > >> > diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml > > >> > new file mode 100644 > > >> > index 0000000000..c4747f5b40 > > >> > --- /dev/null > > >> > +++ b/gdb-xml/avr-cpu.xml > > >> > @@ -0,0 +1,49 @@ > > >> > +<?xml version="1.0"?> > > >> > +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. > > >> > + > > >> > + Copying and distribution of this file, with or without > > >> modification, > > >> > + are permitted in any medium without royalty provided the > copyright > > >> > + notice and this notice are preserved. --> > > >> > + > > >> > +<!-- Register numbers are hard-coded in order to maintain backward > > >> > + compatibility with older versions of tools that didn't use xml > > >> > + register descriptions. --> > > >> > + > > >> > +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> > > >> > +<feature name="org.gnu.gdb.riscv.cpu"> > > >> > + <reg name="r0" bitsize="8" type="int" regnum="0"/> > > >> > + <reg name="r1" bitsize="8" type="int"/> > > >> > + <reg name="r2" bitsize="8" type="int"/> > > >> > + <reg name="r3" bitsize="8" type="int"/> > > >> > + <reg name="r4" bitsize="8" type="int"/> > > >> > + <reg name="r5" bitsize="8" type="int"/> > > >> > + <reg name="r6" bitsize="8" type="int"/> > > >> > + <reg name="r7" bitsize="8" type="int"/> > > >> > + <reg name="r8" bitsize="8" type="int"/> > > >> > + <reg name="r9" bitsize="8" type="int"/> > > >> > + <reg name="r10" bitsize="8" type="int"/> > > >> > + <reg name="r11" bitsize="8" type="int"/> > > >> > + <reg name="r12" bitsize="8" type="int"/> > > >> > + <reg name="r13" bitsize="8" type="int"/> > > >> > + <reg name="r14" bitsize="8" type="int"/> > > >> > + <reg name="r15" bitsize="8" type="int"/> > > >> > + <reg name="r16" bitsize="8" type="int"/> > > >> > + <reg name="r17" bitsize="8" type="int"/> > > >> > + <reg name="r18" bitsize="8" type="int"/> > > >> > + <reg name="r19" bitsize="8" type="int"/> > > >> > + <reg name="r20" bitsize="8" type="int"/> > > >> > + <reg name="r21" bitsize="8" type="int"/> > > >> > + <reg name="r22" bitsize="8" type="int"/> > > >> > + <reg name="r23" bitsize="8" type="int"/> > > >> > + <reg name="r24" bitsize="8" type="int"/> > > >> > + <reg name="r25" bitsize="8" type="int"/> > > >> > + <reg name="r26" bitsize="8" type="int"/> > > >> > + <reg name="r27" bitsize="8" type="int"/> > > >> > + <reg name="r28" bitsize="8" type="int"/> > > >> > + <reg name="r29" bitsize="8" type="int"/> > > >> > + <reg name="r30" bitsize="8" type="int"/> > > >> > + <reg name="r31" bitsize="8" type="int"/> > > >> > + <reg name="sreg" bitsize="8" type="int"/> > > >> > + <reg name="sp" bitsize="8" type="int"/> > > >> > + <reg name="pc" bitsize="8" type="int"/> > > >> > +</feature> > > >> > diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h > > >> > new file mode 100644 > > >> > index 0000000000..5bbf985726 > > >> > --- /dev/null > > >> > +++ b/target/avr/cpu-param.h > > >> > @@ -0,0 +1,37 @@ > > >> > +/* > > >> > + * QEMU AVR CPU > > >> > + * > > >> > + * Copyright (c) 2016 Michael Rolnik > > >> > + * > > >> > + * 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 > > >> > + * version 2.1 of the License, or (at your option) any later > version. > > >> > + * > > >> > + * 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/lgpl-2.1.html> > > >> > + */ > > >> > + > > >> > +#ifndef AVR_CPU_PARAM_H > > >> > +#define AVR_CPU_PARAM_H 1 > > >> > + > > >> > +#define TARGET_LONG_BITS 32 > > >> > +/* > > >> > + * TARGET_PAGE_BITS cannot be more than 8 bits because > > >> > + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, > and > > >> they > > >> > + * should be implemented as a device and not memory > > >> > + * 2. SRAM starts at the address 0x0100 > > >> > + */ > > >> > +#define TARGET_PAGE_BITS 8 > > >> > +#define TARGET_PHYS_ADDR_SPACE_BITS 24 > > >> > +#define TARGET_VIRT_ADDR_SPACE_BITS 24 > > >> > +#define NB_MMU_MODES 2 > > >> > + > > >> > + > > >> > +#endif > > >> > diff --git a/target/avr/cpu.c b/target/avr/cpu.c > > >> > new file mode 100644 > > >> > index 0000000000..142fe54524 > > >> > --- /dev/null > > >> > +++ b/target/avr/cpu.c > > >> > @@ -0,0 +1,599 @@ > > >> > +/* > > >> > + * QEMU AVR CPU > > >> > + * > > >> > + * Copyright (c) 2016 Michael Rolnik > > >> > + * > > >> > + * 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 > > >> > + * version 2.1 of the License, or (at your option) any later > version. > > >> > + * > > >> > + * 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/lgpl-2.1.html> > > >> > + */ > > >> > + > > >> > +#include "qemu/osdep.h" > > >> > +#include "qemu/qemu-print.h" > > >> > +#include "qemu/log.h" > > >> > +#include "cpu.h" > > >> > +#include "exec/exec-all.h" > > >> > +#include "qapi/error.h" > > >> > +#include "hw/qdev-properties.h" > > >> > +#include "migration/vmstate.h" > > >> > + > > >> > +static void avr_cpu_set_pc(CPUState *cs, vaddr value) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + > > >> > + cpu->env.pc_w = value / 2; /* internally PC points to words */ > > >> > +} > > >> > + > > >> > +static bool avr_cpu_has_work(CPUState *cs) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | > > >> CPU_INTERRUPT_RESET)) > > >> > + && cpu_interrupts_enabled(env); > > >> > +} > > >> > + > > >> > +static void avr_cpu_synchronize_from_tb(CPUState *cs, > TranslationBlock > > >> *tb) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + env->pc_w = tb->pc / 2; /* internally PC points to words */ > > >> > +} > > >> > + > > >> > +static void avr_cpu_reset(CPUState *cs) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + mcc->parent_reset(cs); > > >> > + > > >> > + env->pc_w = 0; > > >> > + env->sregI = 1; > > >> > + env->sregC = 0; > > >> > + env->sregZ = 0; > > >> > + env->sregN = 0; > > >> > + env->sregV = 0; > > >> > + env->sregS = 0; > > >> > + env->sregH = 0; > > >> > + env->sregT = 0; > > >> > + > > >> > + env->rampD = 0; > > >> > + env->rampX = 0; > > >> > + env->rampY = 0; > > >> > + env->rampZ = 0; > > >> > + env->eind = 0; > > >> > + env->sp = 0; > > >> > + > > >> > + env->skip = 0; > > >> > + > > >> > + memset(env->r, 0, sizeof(env->r)); > > >> > + > > >> > + tlb_flush(cs); > > >> > +} > > >> > + > > >> > +static void avr_cpu_disas_set_info(CPUState *cpu, > disassemble_info > > >> *info) > > >> > +{ > > >> > + info->mach = bfd_arch_avr; > > >> > + info->print_insn = NULL; > > >> > +} > > >> > + > > >> > +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) > > >> > +{ > > >> > + CPUState *cs = CPU(dev); > > >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); > > >> > + Error *local_err = NULL; > > >> > + > > >> > + cpu_exec_realizefn(cs, &local_err); > > >> > + if (local_err != NULL) { > > >> > + error_propagate(errp, local_err); > > >> > + return; > > >> > + } > > >> > + qemu_init_vcpu(cs); > > >> > + cpu_reset(cs); > > >> > + > > >> > + mcc->parent_realize(dev, errp); > > >> > +} > > >> > + > > >> > +static void avr_cpu_set_int(void *opaque, int irq, int level) > > >> > +{ > > >> > + AVRCPU *cpu = opaque; > > >> > + CPUAVRState *env = &cpu->env; > > >> > + CPUState *cs = CPU(cpu); > > >> > + > > >> > + uint64_t mask = (1ull << irq); > > >> > + if (level) { > > >> > + env->intsrc |= mask; > > >> > + cpu_interrupt(cs, CPU_INTERRUPT_HARD); > > >> > + } else { > > >> > + env->intsrc &= ~mask; > > >> > + if (env->intsrc == 0) { > > >> > + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); > > >> > + } > > >> > + } > > >> > +} > > >> > + > > >> > +static void avr_cpu_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + > > >> > + cpu_set_cpustate_pointers(cpu); > > >> > + > > >> > +#ifndef CONFIG_USER_ONLY > > >> > + /* Set the number of interrupts supported by the CPU. */ > > >> > + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); > > >> > +#endif > > >> > +} > > >> > + > > >> > +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) > > >> > +{ > > >> > + ObjectClass *oc; > > >> > + char *typename; > > >> > + char **cpuname; > > >> > + > > >> > + cpuname = g_strsplit(cpu_model, ",", 1); > > >> you do not need this, parse_cpu_option() does it for you. > > >> > > >> > + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), > cpuname[0]); > > >> > + oc = object_class_by_name(typename); > > >> Also just use cpu_model directly for lookup > > >> for example see '[Qemu-devel] [PATCH RESEND v21 06/21] target/rx: CPU > > >> definition' > > >> and simplify avr_cpu_list_entry() correspondingly. > > >> > > >> > > >> > + g_strfreev(cpuname); > > >> > + g_free(typename); > > >> > + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || > > >> > + object_class_is_abstract(oc)) { > > >> > + return NULL; > > >> > + } > > >> > + return oc; > > >> > +} > > >> > + > > >> > +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + int i; > > >> > + > > >> > + qemu_fprintf(f, "\n"); > > >> > + qemu_fprintf(f, "PC: %06x\n", env->pc_w); > > >> > + qemu_fprintf(f, "SP: %04x\n", env->sp); > > >> > + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); > > >> > + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); > > >> > + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); > > >> > + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); > > >> > + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); > > >> > + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); > > >> > + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); > > >> > + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); > > >> > + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", > > >> > + env->sregI ? 'I' : '-', > > >> > + env->sregT ? 'T' : '-', > > >> > + env->sregH ? 'H' : '-', > > >> > + env->sregS ? 'S' : '-', > > >> > + env->sregV ? 'V' : '-', > > >> > + env->sregN ? '-' : 'N', /* Zf has > negative > > >> logic */ > > >> > + env->sregZ ? 'Z' : '-', > > >> > + env->sregC ? 'I' : '-'); > > >> > + qemu_fprintf(f, "SKIP: %02x\n", env->skip); > > >> > + > > >> > + qemu_fprintf(f, "\n"); > > >> > + for (i = 0; i < ARRAY_SIZE(env->r); i++) { > > >> > + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); > > >> > + > > >> > + if ((i % 8) == 7) { > > >> > + qemu_fprintf(f, "\n"); > > >> > + } > > >> > + } > > >> > + qemu_fprintf(f, "\n"); > > >> > +} > > >> > + > > >> > +static void avr_cpu_class_init(ObjectClass *oc, void *data) > > >> > +{ > > >> > + DeviceClass *dc = DEVICE_CLASS(oc); > > >> > + CPUClass *cc = CPU_CLASS(oc); > > >> > + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); > > >> > + > > >> > + mcc->parent_realize = dc->realize; > > >> > + dc->realize = avr_cpu_realizefn; > > >> > + > > >> > + mcc->parent_reset = cc->reset; > > >> > + cc->reset = avr_cpu_reset; > > >> > + > > >> > + cc->class_by_name = avr_cpu_class_by_name; > > >> > + > > >> > + cc->has_work = avr_cpu_has_work; > > >> > + cc->do_interrupt = avr_cpu_do_interrupt; > > >> > + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; > > >> > + cc->dump_state = avr_cpu_dump_state; > > >> > + cc->set_pc = avr_cpu_set_pc; > > >> > +#if !defined(CONFIG_USER_ONLY) > > >> > + cc->memory_rw_debug = avr_cpu_memory_rw_debug; > > >> > +#endif > > >> > +#ifdef CONFIG_USER_ONLY > > >> > + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; > > >> > +#else > > >> > + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; > > >> > + cc->vmsd = &vms_avr_cpu; > > >> > +#endif > > >> > + cc->disas_set_info = avr_cpu_disas_set_info; > > >> > + cc->tlb_fill = avr_cpu_tlb_fill; > > >> > + cc->tcg_initialize = avr_cpu_tcg_init; > > >> > + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; > > >> > + cc->gdb_read_register = avr_cpu_gdb_read_register; > > >> > + cc->gdb_write_register = avr_cpu_gdb_write_register; > > >> > + cc->gdb_num_core_regs = 35; > > >> > + cc->gdb_core_xml_file = "avr-cpu.xml"; > > >> > +} > > >> > + > > >> > +static void avr_avr1_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > +} > > >> > + > > >> > +static void avr_avr2_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > +} > > >> > + > > >> > +static void avr_avr25_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > +} > > >> > + > > >> > +static void avr_avr3_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > +} > > >> > + > > >> > +static void avr_avr31_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > +} > > >> > + > > >> > +static void avr_avr35_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > +} > > >> > + > > >> > +static void avr_avr4_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > +} > > >> > + > > >> > +static void avr_avr5_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > +} > > >> > + > > >> > +static void avr_avr51_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > +} > > >> > + > > >> > +static void avr_avr6_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > +} > > >> > + > > >> > +static void avr_xmega2_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > >> > +} > > >> > + > > >> > +static void avr_xmega4_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > >> > +} > > >> > + > > >> > +static void avr_xmega5_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > >> > +} > > >> > + > > >> > +static void avr_xmega6_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > >> > +} > > >> > + > > >> > +static void avr_xmega7_initfn(Object *obj) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > >> > + > > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > >> > +} > > >> > + > > >> > +typedef struct AVRCPUInfo { > > >> > + const char *name; > > >> > + void (*initfn)(Object *obj); > > >> > +} AVRCPUInfo; > > >> > + > > >> > +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) > > >> > +{ > > >> > + ObjectClass *class_a = (ObjectClass *)a; > > >> > + ObjectClass *class_b = (ObjectClass *)b; > > >> > + const char *name_a; > > >> > + const char *name_b; > > >> > + > > >> > + name_a = object_class_get_name(class_a); > > >> > + name_b = object_class_get_name(class_b); > > >> > + > > >> > + return strcmp(name_a, name_b); > > >> > +} > > >> > + > > >> > +static void avr_cpu_list_entry(gpointer data, gpointer user_data) > > >> > +{ > > >> > + const char *typename = > object_class_get_name(OBJECT_CLASS(data)); > > >> > + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); > > >> > + qemu_printf("%.*s\n", len, typename); > > >> > +} > > >> > + > > >> > +void avr_cpu_list(void) > > >> > +{ > > >> > + GSList *list; > > >> > + list = object_class_get_list(TYPE_AVR_CPU, false); > > >> > + list = g_slist_sort(list, avr_cpu_list_compare); > > >> > + qemu_printf("Available CPUs:\n"); > > >> > + g_slist_foreach(list, avr_cpu_list_entry, NULL); > > >> > + g_slist_free(list); > > >> > +} > > >> > + > > >> > +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ > > >> > + { \ > > >> > + .parent = TYPE_AVR_CPU, \ > > >> > + .instance_init = initfn, \ > > >> > + .name = AVR_CPU_TYPE_NAME(model), \ > > >> > + } > > >> > + > > >> > +static const TypeInfo avr_cpu_type_info[] = { > > >> > + { > > >> > + .name = TYPE_AVR_CPU, > > >> > + .parent = TYPE_CPU, > > >> > + .instance_size = sizeof(AVRCPU), > > >> > + .instance_init = avr_cpu_initfn, > > >> > + .class_size = sizeof(AVRCPUClass), > > >> > + .class_init = avr_cpu_class_init, > > >> > + .abstract = true, > > >> > + }, > > >> > + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), > > >> > + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), > > >> > +}; > > >> > + > > >> > +DEFINE_TYPES(avr_cpu_type_info) > > >> > diff --git a/target/avr/cpu.h b/target/avr/cpu.h > > >> > new file mode 100644 > > >> > index 0000000000..3f9a803193 > > >> > --- /dev/null > > >> > +++ b/target/avr/cpu.h > > >> > @@ -0,0 +1,283 @@ > > >> > +/* > > >> > + * QEMU AVR CPU > > >> > + * > > >> > + * Copyright (c) 2016 Michael Rolnik > > >> > + * > > >> > + * 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 > > >> > + * version 2.1 of the License, or (at your option) any later > version. > > >> > + * > > >> > + * 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/lgpl-2.1.html> > > >> > + */ > > >> > + > > >> > +#ifndef CPU_AVR_H > > >> > +#define CPU_AVR_H > > >> > + > > >> > +#include "qemu-common.h" > > >> > +#include "qom/cpu.h" > > >> > +#include "exec/cpu-defs.h" > > >> > +#include "fpu/softfloat.h" > > >> > + > > >> > +#define TCG_GUEST_DEFAULT_MO 0 > > >> > + > > >> > +#define TYPE_AVR_CPU "avr-cpu" > > >> > + > > >> > +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU > > >> > +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) > > >> > +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU > > >> > + > > >> > +/* > > >> > + * AVR has two memory spaces, data & code. > > >> > + * e.g. both have 0 address > > >> > + * ST/LD instructions access data space > > >> > + * LPM/SPM and instruction fetching access code memory space > > >> > + */ > > >> > +#define MMU_CODE_IDX 0 > > >> > +#define MMU_DATA_IDX 1 > > >> > + > > >> > +#define EXCP_RESET 1 > > >> > +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) > > >> > + > > >> > +/* Number of CPU registers */ > > >> > +#define NO_CPU_REGISTERS 32 > > >> > +/* Number of IO registers accessible by ld/st/in/out */ > > >> > +#define NO_IO_REGISTERS 64 > > >> > + > > >> > +/* > > >> > + * Offsets of AVR memory regions in host memory space. > > >> > + * > > >> > + * This is needed because the AVR has separate code and data > address > > >> > + * spaces that both have start from zero but have to go somewhere > in > > >> > + * host memory. > > >> > + * > > >> > + * It's also useful to know where some things are, like the IO > > >> registers. > > >> > + */ > > >> > +/* Flash program memory */ > > >> > +#define OFFSET_CODE 0x00000000 > > >> > +/* CPU registers, IO registers, and SRAM */ > > >> > +#define OFFSET_DATA 0x00800000 > > >> > +/* CPU registers specifically, these are mapped at the start of > data */ > > >> > +#define OFFSET_CPU_REGISTERS OFFSET_DATA > > >> > +/* > > >> > + * IO registers, including status register, stack pointer, and > memory > > >> > + * mapped peripherals, mapped just after CPU registers > > >> > + */ > > >> > +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) > > >> > + > > >> > +enum avr_features { > > >> > + AVR_FEATURE_SRAM, > > >> > + > > >> > + AVR_FEATURE_1_BYTE_PC, > > >> > + AVR_FEATURE_2_BYTE_PC, > > >> > + AVR_FEATURE_3_BYTE_PC, > > >> > + > > >> > + AVR_FEATURE_1_BYTE_SP, > > >> > + AVR_FEATURE_2_BYTE_SP, > > >> > + > > >> > + AVR_FEATURE_BREAK, > > >> > + AVR_FEATURE_DES, > > >> > + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ > > >> > + > > >> > + AVR_FEATURE_EIJMP_EICALL, > > >> > + AVR_FEATURE_IJMP_ICALL, > > >> > + AVR_FEATURE_JMP_CALL, > > >> > + > > >> > + AVR_FEATURE_ADIW_SBIW, > > >> > + > > >> > + AVR_FEATURE_SPM, > > >> > + AVR_FEATURE_SPMX, > > >> > + > > >> > + AVR_FEATURE_ELPMX, > > >> > + AVR_FEATURE_ELPM, > > >> > + AVR_FEATURE_LPMX, > > >> > + AVR_FEATURE_LPM, > > >> > + > > >> > + AVR_FEATURE_MOVW, > > >> > + AVR_FEATURE_MUL, > > >> > + AVR_FEATURE_RAMPD, > > >> > + AVR_FEATURE_RAMPX, > > >> > + AVR_FEATURE_RAMPY, > > >> > + AVR_FEATURE_RAMPZ, > > >> > +}; > > >> > + > > >> > +typedef struct CPUAVRState CPUAVRState; > > >> > + > > >> > +struct CPUAVRState { > > >> > + uint32_t pc_w; /* 0x003fffff up to 22 bits */ > > >> > + > > >> > + uint32_t sregC; /* 0x00000001 1 bits */ > > >> > + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ > > >> > + /* 0=flag set, >0=flag cleared */ > > >> > + uint32_t sregN; /* 0x00000001 1 bits */ > > >> > + uint32_t sregV; /* 0x00000001 1 bits */ > > >> > + uint32_t sregS; /* 0x00000001 1 bits */ > > >> > + uint32_t sregH; /* 0x00000001 1 bits */ > > >> > + uint32_t sregT; /* 0x00000001 1 bits */ > > >> > + uint32_t sregI; /* 0x00000001 1 bits */ > > >> > + > > >> > + uint32_t rampD; /* 0x00ff0000 8 bits */ > > >> > + uint32_t rampX; /* 0x00ff0000 8 bits */ > > >> > + uint32_t rampY; /* 0x00ff0000 8 bits */ > > >> > + uint32_t rampZ; /* 0x00ff0000 8 bits */ > > >> > + uint32_t eind; /* 0x00ff0000 8 bits */ > > >> > + > > >> > + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ > > >> > + uint32_t sp; /* 16 bits */ > > >> > + > > >> > + uint32_t skip; /* if set skip instruction */ > > >> > + > > >> > + uint64_t intsrc; /* interrupt sources */ > > >> > + bool fullacc; /* CPU/MEM if true MEM only otherwise */ > > >> > + > > >> > + uint32_t features; > > >> > +}; > > >> > + > > >> > +#define AVR_CPU_CLASS(klass) \ > > >> > + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) > > >> > +#define AVR_CPU(obj) \ > > >> > + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) > > >> > +#define AVR_CPU_GET_CLASS(obj) \ > > >> > + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) > > >> > + > > >> > +/** > > >> > + * AVRCPUClass: > > >> > + * @parent_realize: The parent class' realize handler. > > >> > + * @parent_reset: The parent class' reset handler. > > >> > + * @vr: Version Register value. > > >> > + * > > >> > + * A AVR CPU model. > > >> > + */ > > >> > +typedef struct AVRCPUClass { > > >> > + /*< private >*/ > > >> > + CPUClass parent_class; > > >> > + /*< public >*/ > > >> > + DeviceRealize parent_realize; > > >> > + void (*parent_reset)(CPUState *cpu); > > >> > +} AVRCPUClass; > > >> > + > > >> > +/** > > >> > + * AVRCPU: > > >> > + * @env: #CPUAVRState > > >> > + * > > >> > + * A AVR CPU. > > >> > + */ > > >> > +typedef struct AVRCPU { > > >> > + /*< private >*/ > > >> > + CPUState parent_obj; > > >> > + /*< public >*/ > > >> > + > > >> > + CPUNegativeOffsetState neg; > > >> > + CPUAVRState env; > > >> > +} AVRCPU; > > >> > + > > >> > +#ifndef CONFIG_USER_ONLY > > >> > +extern const struct VMStateDescription vms_avr_cpu; > > >> > +#endif > > >> > + > > >> > +void avr_cpu_do_interrupt(CPUState *cpu); > > >> > +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); > > >> > +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); > > >> > +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int > reg); > > >> > +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int > reg); > > >> > + > > >> > +static inline int avr_feature(CPUAVRState *env, int feature) > > >> > +{ > > >> > + return (env->features & (1U << feature)) != 0; > > >> > +} > > >> > + > > >> > +static inline void avr_set_feature(CPUAVRState *env, int feature) > > >> > +{ > > >> > + env->features |= (1U << feature); > > >> > +} > > >> > + > > >> > +#define cpu_list avr_cpu_list > > >> > +#define cpu_signal_handler cpu_avr_signal_handler > > >> > +#define cpu_mmu_index avr_cpu_mmu_index > > >> > + > > >> > +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) > > >> > +{ > > >> > + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; > > >> > +} > > >> > + > > >> > +void avr_cpu_tcg_init(void); > > >> > + > > >> > +void avr_cpu_list(void); > > >> > +int cpu_avr_exec(CPUState *cpu); > > >> > +int cpu_avr_signal_handler(int host_signum, void *pinfo, void > *puc); > > >> > +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int > size, > > >> > + int rw, int mmu_idx); > > >> > +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t > *buf, > > >> > + int len, bool is_write); > > >> > + > > >> > +enum { > > >> > + TB_FLAGS_FULL_ACCESS = 1, > > >> > + TB_FLAGS_SKIP = 2, > > >> > +}; > > >> > + > > >> > +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, > target_ulong > > >> *pc, > > >> > + target_ulong *cs_base, uint32_t > > >> *pflags) > > >> > +{ > > >> > + uint32_t flags = 0; > > >> > + > > >> > + *pc = env->pc_w * 2; > > >> > + *cs_base = 0; > > >> > + > > >> > + if (env->fullacc) { > > >> > + flags |= TB_FLAGS_FULL_ACCESS; > > >> > + } > > >> > + if (env->skip) { > > >> > + flags |= TB_FLAGS_SKIP; > > >> > + } > > >> > + > > >> > + *pflags = flags; > > >> > +} > > >> > + > > >> > +static inline int cpu_interrupts_enabled(CPUAVRState *env) > > >> > +{ > > >> > + return env->sregI != 0; > > >> > +} > > >> > + > > >> > +static inline uint8_t cpu_get_sreg(CPUAVRState *env) > > >> > +{ > > >> > + uint8_t sreg; > > >> > + sreg = (env->sregC & 0x01) << 0 > > >> > + | (env->sregZ == 0 ? 1 : 0) << 1 > > >> > + | (env->sregN) << 2 > > >> > + | (env->sregV) << 3 > > >> > + | (env->sregS) << 4 > > >> > + | (env->sregH) << 5 > > >> > + | (env->sregT) << 6 > > >> > + | (env->sregI) << 7; > > >> > + return sreg; > > >> > +} > > >> > + > > >> > +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) > > >> > +{ > > >> > + env->sregC = (sreg >> 0) & 0x01; > > >> > + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; > > >> > + env->sregN = (sreg >> 2) & 0x01; > > >> > + env->sregV = (sreg >> 3) & 0x01; > > >> > + env->sregS = (sreg >> 4) & 0x01; > > >> > + env->sregH = (sreg >> 5) & 0x01; > > >> > + env->sregT = (sreg >> 6) & 0x01; > > >> > + env->sregI = (sreg >> 7) & 0x01; > > >> > +} > > >> > + > > >> > +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, > > >> > + MMUAccessType access_type, int mmu_idx, > > >> > + bool probe, uintptr_t retaddr); > > >> > + > > >> > +typedef CPUAVRState CPUArchState; > > >> > +typedef AVRCPU ArchCPU; > > >> > + > > >> > +#include "exec/cpu-all.h" > > >> > + > > >> > +#endif /* !defined (CPU_AVR_H) */ > > >> > diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c > > >> > new file mode 100644 > > >> > index 0000000000..537dc7226e > > >> > --- /dev/null > > >> > +++ b/target/avr/gdbstub.c > > >> > @@ -0,0 +1,85 @@ > > >> > +/* > > >> > + * QEMU AVR CPU > > >> > + * > > >> > + * Copyright (c) 2016 Michael Rolnik > > >> > + * > > >> > + * 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 > > >> > + * version 2.1 of the License, or (at your option) any later > version. > > >> > + * > > >> > + * 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/lgpl-2.1.html> > > >> > + */ > > >> > + > > >> > +#include "qemu/osdep.h" > > >> > +#include "qemu-common.h" > > >> > +#include "exec/gdbstub.h" > > >> > + > > >> > +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int > n) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + /* R */ > > >> > + if (n < 32) { > > >> > + return gdb_get_reg8(mem_buf, env->r[n]); > > >> > + } > > >> > + > > >> > + /* SREG */ > > >> > + if (n == 32) { > > >> > + uint8_t sreg = cpu_get_sreg(env); > > >> > + > > >> > + return gdb_get_reg8(mem_buf, sreg); > > >> > + } > > >> > + > > >> > + /* SP */ > > >> > + if (n == 33) { > > >> > + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); > > >> > + } > > >> > + > > >> > + /* PC */ > > >> > + if (n == 34) { > > >> > + return gdb_get_reg32(mem_buf, env->pc_w * 2); > > >> > + } > > >> > + > > >> > + return 0; > > >> > +} > > >> > + > > >> > +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int > n) > > >> > +{ > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > >> > + CPUAVRState *env = &cpu->env; > > >> > + > > >> > + /* R */ > > >> > + if (n < 32) { > > >> > + env->r[n] = *mem_buf; > > >> > + return 1; > > >> > + } > > >> > + > > >> > + /* SREG */ > > >> > + if (n == 32) { > > >> > + cpu_set_sreg(env, *mem_buf); > > >> > + return 1; > > >> > + } > > >> > + > > >> > + /* SP */ > > >> > + if (n == 33) { > > >> > + env->sp = lduw_p(mem_buf); > > >> > + return 2; > > >> > + } > > >> > + > > >> > + /* PC */ > > >> > + if (n == 34) { > > >> > + env->pc_w = ldl_p(mem_buf) / 2; > > >> > + return 4; > > >> > + } > > >> > + > > >> > + return 0; > > >> > +} > > >> > diff --git a/target/avr/machine.c b/target/avr/machine.c > > >> > new file mode 100644 > > >> > index 0000000000..389b5eb95d > > >> > --- /dev/null > > >> > +++ b/target/avr/machine.c > > >> > @@ -0,0 +1,123 @@ > > >> > +/* > > >> > + * QEMU AVR CPU > > >> > + * > > >> > + * Copyright (c) 2016 Michael Rolnik > > >> > + * > > >> > + * 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 > > >> > + * version 2.1 of the License, or (at your option) any later > version. > > >> > + * > > >> > + * 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/lgpl-2.1.html> > > >> > + */ > > >> > + > > >> > +#include "qemu/osdep.h" > > >> > +#include "hw/hw.h" > > >> > +#include "cpu.h" > > >> > +#include "hw/boards.h" > > >> > +#include "migration/qemu-file.h" > > >> > + > > >> > +static int get_sreg(QEMUFile *f, void *opaque, size_t size, > > >> > + const VMStateField *field) > > >> > +{ > > >> > + CPUAVRState *env = opaque; > > >> > + uint8_t sreg; > > >> > + > > >> > + sreg = qemu_get_byte(f); > > >> > + cpu_set_sreg(env, sreg); > > >> > + return 0; > > >> > +} > > >> > + > > >> > +static int put_sreg( > > >> > + QEMUFile *f, void *opaque, size_t size, > > >> > + const VMStateField *field, QJSON *vmdesc) > > >> > +{ > > >> > + CPUAVRState *env = opaque; > > >> > + uint8_t sreg = cpu_get_sreg(env); > > >> > + > > >> > + qemu_put_byte(f, sreg); > > >> > + return 0; > > >> > +} > > >> > + > > >> > +static const VMStateInfo vms_sreg = { > > >> > + .name = "sreg", > > >> > + .get = get_sreg, > > >> > + .put = put_sreg, > > >> > +}; > > >> > + > > >> > +static int get_segment( > > >> > + QEMUFile *f, void *opaque, size_t size, const VMStateField > *field) > > >> > +{ > > >> > + uint32_t *ramp = opaque; > > >> > + uint8_t temp; > > >> > + > > >> > + temp = qemu_get_byte(f); > > >> > + *ramp = ((uint32_t)temp) << 16; > > >> > + return 0; > > >> > +} > > >> > + > > >> > +static int put_segment( > > >> > + QEMUFile *f, void *opaque, size_t size, > > >> > + const VMStateField *field, QJSON *vmdesc) > > >> > +{ > > >> > + uint32_t *ramp = opaque; > > >> > + uint8_t temp = *ramp >> 16; > > >> > + > > >> > + qemu_put_byte(f, temp); > > >> > + return 0; > > >> > +} > > >> > + > > >> > +static const VMStateInfo vms_rampD = { > > >> > + .name = "rampD", > > >> > + .get = get_segment, > > >> > + .put = put_segment, > > >> > +}; > > >> > +static const VMStateInfo vms_rampX = { > > >> > + .name = "rampX", > > >> > + .get = get_segment, > > >> > + .put = put_segment, > > >> > +}; > > >> > +static const VMStateInfo vms_rampY = { > > >> > + .name = "rampY", > > >> > + .get = get_segment, > > >> > + .put = put_segment, > > >> > +}; > > >> > +static const VMStateInfo vms_rampZ = { > > >> > + .name = "rampZ", > > >> > + .get = get_segment, > > >> > + .put = put_segment, > > >> > +}; > > >> > +static const VMStateInfo vms_eind = { > > >> > + .name = "eind", > > >> > + .get = get_segment, > > >> > + .put = put_segment, > > >> > +}; > > >> > + > > >> > +const VMStateDescription vms_avr_cpu = { > > >> > + .name = "cpu", > > >> > + .version_id = 0, > > >> > + .minimum_version_id = 0, > > >> > + .fields = (VMStateField[]) { > > >> > + VMSTATE_UINT32(env.pc_w, AVRCPU), > > >> > + VMSTATE_UINT32(env.sp, AVRCPU), > > >> > + VMSTATE_UINT32(env.skip, AVRCPU), > > >> > + > > >> > + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), > > >> > + > > >> > + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), > > >> > + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), > > >> > + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), > > >> > + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), > > >> > + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), > > >> > + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), > > >> > + > > >> > + VMSTATE_END_OF_LIST() > > >> > + } > > >> > +}; > > >> > > >> > > > > > > -- > > > Best Regards, > > > Michael Rolnik > > > > >
On Fri, 28 Jun 2019 11:43:22 +0300 Michael Rolnik <mrolnik@gmail.com> wrote: > Nobody else does it. it doesn't mean that we should blindly copy pre-existing patterns which might/will exist for compatibility reasons. pls see the same argument on RX CPU for reasons not to support anything else beside typename there. [Qemu-devel] [PATCH v16 13/23] target/rx: Fix cpu types and names [Qemu-devel] [PATCH v19 03/21] target/rx: CPU definition > Richard. > What do you think? > > Sent from my cell phone, please ignore typos > > On Fri, Jun 28, 2019, 11:36 AM Igor Mammedov <imammedo@redhat.com> wrote: > > > On Thu, 27 Jun 2019 21:51:23 +0300 > > Michael Rolnik <mrolnik@gmail.com> wrote: > > > > > I meant that if I take the proposed code it fails in the mentioned way > > > > > > Sent from my cell phone, please ignore typos > > > > > > On Thu, Jun 27, 2019, 7:02 PM Michael Rolnik <mrolnik@gmail.com> wrote: > > > > > > > Hi Igor, > > > > > > > > if I run `make check` it fails with > > > > qemu-system-rx: unable to find CPU model 'rx62n' > > > > or > > > > qemu-system-avr: unable to find CPU model 'avr6' > > suggested change means that one has to user type name as is for -cpu > > argument. > > > > so test should be amended to account for that, > > something like that should be sufficient: > > > > diff --git a/tests/machine-none-test.c b/tests/machine-none-test.c > > index 361927bb76..d0907e6bd4 100644 > > --- a/tests/machine-none-test.c > > +++ b/tests/machine-none-test.c > > @@ -27,7 +27,7 @@ static struct arch2cpu cpus_map[] = { > > /* tested targets list */ > > { "arm", "cortex-a15" }, > > { "aarch64", "cortex-a57" }, > > - { "avr", "avr6" }, > > + { "avr", "avr6-avr-cpu" }, > > { "x86_64", "qemu64,apic-id=0" }, > > { "i386", "qemu32,apic-id=0" }, > > { "alpha", "ev67" }, > > > > > > > > > > > > On Thu, Jun 27, 2019 at 2:28 PM Igor Mammedov <imammedo@redhat.com> > > wrote: > > > > > > > >> On Thu, 27 Jun 2019 08:27:44 +0300 > > > >> Michael Rolnik <mrolnik@gmail.com> wrote: > > > >> > > > >> > From: Sarah Harris <S.E.Harris@kent.ac.uk> > > > >> > > > > >> > This includes: > > > >> > - CPU data structures > > > >> > - object model classes and functions > > > >> > - migration functions > > > >> > - GDB hooks > > > >> > > > > >> > Signed-off-by: Michael Rolnik <mrolnik@gmail.com> > > > >> > --- > > > >> > gdb-xml/avr-cpu.xml | 49 ++++ > > > >> > target/avr/cpu-param.h | 37 +++ > > > >> > target/avr/cpu.c | 599 > > +++++++++++++++++++++++++++++++++++++++++ > > > >> > target/avr/cpu.h | 283 +++++++++++++++++++ > > > >> > target/avr/gdbstub.c | 85 ++++++ > > > >> > target/avr/machine.c | 123 +++++++++ > > > >> > 6 files changed, 1176 insertions(+) > > > >> > create mode 100644 gdb-xml/avr-cpu.xml > > > >> > create mode 100644 target/avr/cpu-param.h > > > >> > create mode 100644 target/avr/cpu.c > > > >> > create mode 100644 target/avr/cpu.h > > > >> > create mode 100644 target/avr/gdbstub.c > > > >> > create mode 100644 target/avr/machine.c > > > >> > > > > >> > diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml > > > >> > new file mode 100644 > > > >> > index 0000000000..c4747f5b40 > > > >> > --- /dev/null > > > >> > +++ b/gdb-xml/avr-cpu.xml > > > >> > @@ -0,0 +1,49 @@ > > > >> > +<?xml version="1.0"?> > > > >> > +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. > > > >> > + > > > >> > + Copying and distribution of this file, with or without > > > >> modification, > > > >> > + are permitted in any medium without royalty provided the > > copyright > > > >> > + notice and this notice are preserved. --> > > > >> > + > > > >> > +<!-- Register numbers are hard-coded in order to maintain backward > > > >> > + compatibility with older versions of tools that didn't use xml > > > >> > + register descriptions. --> > > > >> > + > > > >> > +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> > > > >> > +<feature name="org.gnu.gdb.riscv.cpu"> > > > >> > + <reg name="r0" bitsize="8" type="int" regnum="0"/> > > > >> > + <reg name="r1" bitsize="8" type="int"/> > > > >> > + <reg name="r2" bitsize="8" type="int"/> > > > >> > + <reg name="r3" bitsize="8" type="int"/> > > > >> > + <reg name="r4" bitsize="8" type="int"/> > > > >> > + <reg name="r5" bitsize="8" type="int"/> > > > >> > + <reg name="r6" bitsize="8" type="int"/> > > > >> > + <reg name="r7" bitsize="8" type="int"/> > > > >> > + <reg name="r8" bitsize="8" type="int"/> > > > >> > + <reg name="r9" bitsize="8" type="int"/> > > > >> > + <reg name="r10" bitsize="8" type="int"/> > > > >> > + <reg name="r11" bitsize="8" type="int"/> > > > >> > + <reg name="r12" bitsize="8" type="int"/> > > > >> > + <reg name="r13" bitsize="8" type="int"/> > > > >> > + <reg name="r14" bitsize="8" type="int"/> > > > >> > + <reg name="r15" bitsize="8" type="int"/> > > > >> > + <reg name="r16" bitsize="8" type="int"/> > > > >> > + <reg name="r17" bitsize="8" type="int"/> > > > >> > + <reg name="r18" bitsize="8" type="int"/> > > > >> > + <reg name="r19" bitsize="8" type="int"/> > > > >> > + <reg name="r20" bitsize="8" type="int"/> > > > >> > + <reg name="r21" bitsize="8" type="int"/> > > > >> > + <reg name="r22" bitsize="8" type="int"/> > > > >> > + <reg name="r23" bitsize="8" type="int"/> > > > >> > + <reg name="r24" bitsize="8" type="int"/> > > > >> > + <reg name="r25" bitsize="8" type="int"/> > > > >> > + <reg name="r26" bitsize="8" type="int"/> > > > >> > + <reg name="r27" bitsize="8" type="int"/> > > > >> > + <reg name="r28" bitsize="8" type="int"/> > > > >> > + <reg name="r29" bitsize="8" type="int"/> > > > >> > + <reg name="r30" bitsize="8" type="int"/> > > > >> > + <reg name="r31" bitsize="8" type="int"/> > > > >> > + <reg name="sreg" bitsize="8" type="int"/> > > > >> > + <reg name="sp" bitsize="8" type="int"/> > > > >> > + <reg name="pc" bitsize="8" type="int"/> > > > >> > +</feature> > > > >> > diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h > > > >> > new file mode 100644 > > > >> > index 0000000000..5bbf985726 > > > >> > --- /dev/null > > > >> > +++ b/target/avr/cpu-param.h > > > >> > @@ -0,0 +1,37 @@ > > > >> > +/* > > > >> > + * QEMU AVR CPU > > > >> > + * > > > >> > + * Copyright (c) 2016 Michael Rolnik > > > >> > + * > > > >> > + * 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 > > > >> > + * version 2.1 of the License, or (at your option) any later > > version. > > > >> > + * > > > >> > + * 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/lgpl-2.1.html> > > > >> > + */ > > > >> > + > > > >> > +#ifndef AVR_CPU_PARAM_H > > > >> > +#define AVR_CPU_PARAM_H 1 > > > >> > + > > > >> > +#define TARGET_LONG_BITS 32 > > > >> > +/* > > > >> > + * TARGET_PAGE_BITS cannot be more than 8 bits because > > > >> > + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, > > and > > > >> they > > > >> > + * should be implemented as a device and not memory > > > >> > + * 2. SRAM starts at the address 0x0100 > > > >> > + */ > > > >> > +#define TARGET_PAGE_BITS 8 > > > >> > +#define TARGET_PHYS_ADDR_SPACE_BITS 24 > > > >> > +#define TARGET_VIRT_ADDR_SPACE_BITS 24 > > > >> > +#define NB_MMU_MODES 2 > > > >> > + > > > >> > + > > > >> > +#endif > > > >> > diff --git a/target/avr/cpu.c b/target/avr/cpu.c > > > >> > new file mode 100644 > > > >> > index 0000000000..142fe54524 > > > >> > --- /dev/null > > > >> > +++ b/target/avr/cpu.c > > > >> > @@ -0,0 +1,599 @@ > > > >> > +/* > > > >> > + * QEMU AVR CPU > > > >> > + * > > > >> > + * Copyright (c) 2016 Michael Rolnik > > > >> > + * > > > >> > + * 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 > > > >> > + * version 2.1 of the License, or (at your option) any later > > version. > > > >> > + * > > > >> > + * 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/lgpl-2.1.html> > > > >> > + */ > > > >> > + > > > >> > +#include "qemu/osdep.h" > > > >> > +#include "qemu/qemu-print.h" > > > >> > +#include "qemu/log.h" > > > >> > +#include "cpu.h" > > > >> > +#include "exec/exec-all.h" > > > >> > +#include "qapi/error.h" > > > >> > +#include "hw/qdev-properties.h" > > > >> > +#include "migration/vmstate.h" > > > >> > + > > > >> > +static void avr_cpu_set_pc(CPUState *cs, vaddr value) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + > > > >> > + cpu->env.pc_w = value / 2; /* internally PC points to words */ > > > >> > +} > > > >> > + > > > >> > +static bool avr_cpu_has_work(CPUState *cs) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | > > > >> CPU_INTERRUPT_RESET)) > > > >> > + && cpu_interrupts_enabled(env); > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_synchronize_from_tb(CPUState *cs, > > TranslationBlock > > > >> *tb) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + env->pc_w = tb->pc / 2; /* internally PC points to words */ > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_reset(CPUState *cs) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + mcc->parent_reset(cs); > > > >> > + > > > >> > + env->pc_w = 0; > > > >> > + env->sregI = 1; > > > >> > + env->sregC = 0; > > > >> > + env->sregZ = 0; > > > >> > + env->sregN = 0; > > > >> > + env->sregV = 0; > > > >> > + env->sregS = 0; > > > >> > + env->sregH = 0; > > > >> > + env->sregT = 0; > > > >> > + > > > >> > + env->rampD = 0; > > > >> > + env->rampX = 0; > > > >> > + env->rampY = 0; > > > >> > + env->rampZ = 0; > > > >> > + env->eind = 0; > > > >> > + env->sp = 0; > > > >> > + > > > >> > + env->skip = 0; > > > >> > + > > > >> > + memset(env->r, 0, sizeof(env->r)); > > > >> > + > > > >> > + tlb_flush(cs); > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_disas_set_info(CPUState *cpu, > > disassemble_info > > > >> *info) > > > >> > +{ > > > >> > + info->mach = bfd_arch_avr; > > > >> > + info->print_insn = NULL; > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) > > > >> > +{ > > > >> > + CPUState *cs = CPU(dev); > > > >> > + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); > > > >> > + Error *local_err = NULL; > > > >> > + > > > >> > + cpu_exec_realizefn(cs, &local_err); > > > >> > + if (local_err != NULL) { > > > >> > + error_propagate(errp, local_err); > > > >> > + return; > > > >> > + } > > > >> > + qemu_init_vcpu(cs); > > > >> > + cpu_reset(cs); > > > >> > + > > > >> > + mcc->parent_realize(dev, errp); > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_set_int(void *opaque, int irq, int level) > > > >> > +{ > > > >> > + AVRCPU *cpu = opaque; > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + CPUState *cs = CPU(cpu); > > > >> > + > > > >> > + uint64_t mask = (1ull << irq); > > > >> > + if (level) { > > > >> > + env->intsrc |= mask; > > > >> > + cpu_interrupt(cs, CPU_INTERRUPT_HARD); > > > >> > + } else { > > > >> > + env->intsrc &= ~mask; > > > >> > + if (env->intsrc == 0) { > > > >> > + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); > > > >> > + } > > > >> > + } > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + > > > >> > + cpu_set_cpustate_pointers(cpu); > > > >> > + > > > >> > +#ifndef CONFIG_USER_ONLY > > > >> > + /* Set the number of interrupts supported by the CPU. */ > > > >> > + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); > > > >> > +#endif > > > >> > +} > > > >> > + > > > >> > +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) > > > >> > +{ > > > >> > + ObjectClass *oc; > > > >> > + char *typename; > > > >> > + char **cpuname; > > > >> > + > > > >> > + cpuname = g_strsplit(cpu_model, ",", 1); > > > >> you do not need this, parse_cpu_option() does it for you. > > > >> > > > >> > + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), > > cpuname[0]); > > > >> > + oc = object_class_by_name(typename); > > > >> Also just use cpu_model directly for lookup > > > >> for example see '[Qemu-devel] [PATCH RESEND v21 06/21] target/rx: CPU > > > >> definition' > > > >> and simplify avr_cpu_list_entry() correspondingly. > > > >> > > > >> > > > >> > + g_strfreev(cpuname); > > > >> > + g_free(typename); > > > >> > + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || > > > >> > + object_class_is_abstract(oc)) { > > > >> > + return NULL; > > > >> > + } > > > >> > + return oc; > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + int i; > > > >> > + > > > >> > + qemu_fprintf(f, "\n"); > > > >> > + qemu_fprintf(f, "PC: %06x\n", env->pc_w); > > > >> > + qemu_fprintf(f, "SP: %04x\n", env->sp); > > > >> > + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); > > > >> > + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); > > > >> > + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); > > > >> > + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); > > > >> > + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); > > > >> > + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); > > > >> > + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); > > > >> > + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); > > > >> > + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", > > > >> > + env->sregI ? 'I' : '-', > > > >> > + env->sregT ? 'T' : '-', > > > >> > + env->sregH ? 'H' : '-', > > > >> > + env->sregS ? 'S' : '-', > > > >> > + env->sregV ? 'V' : '-', > > > >> > + env->sregN ? '-' : 'N', /* Zf has > > negative > > > >> logic */ > > > >> > + env->sregZ ? 'Z' : '-', > > > >> > + env->sregC ? 'I' : '-'); > > > >> > + qemu_fprintf(f, "SKIP: %02x\n", env->skip); > > > >> > + > > > >> > + qemu_fprintf(f, "\n"); > > > >> > + for (i = 0; i < ARRAY_SIZE(env->r); i++) { > > > >> > + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); > > > >> > + > > > >> > + if ((i % 8) == 7) { > > > >> > + qemu_fprintf(f, "\n"); > > > >> > + } > > > >> > + } > > > >> > + qemu_fprintf(f, "\n"); > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_class_init(ObjectClass *oc, void *data) > > > >> > +{ > > > >> > + DeviceClass *dc = DEVICE_CLASS(oc); > > > >> > + CPUClass *cc = CPU_CLASS(oc); > > > >> > + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); > > > >> > + > > > >> > + mcc->parent_realize = dc->realize; > > > >> > + dc->realize = avr_cpu_realizefn; > > > >> > + > > > >> > + mcc->parent_reset = cc->reset; > > > >> > + cc->reset = avr_cpu_reset; > > > >> > + > > > >> > + cc->class_by_name = avr_cpu_class_by_name; > > > >> > + > > > >> > + cc->has_work = avr_cpu_has_work; > > > >> > + cc->do_interrupt = avr_cpu_do_interrupt; > > > >> > + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; > > > >> > + cc->dump_state = avr_cpu_dump_state; > > > >> > + cc->set_pc = avr_cpu_set_pc; > > > >> > +#if !defined(CONFIG_USER_ONLY) > > > >> > + cc->memory_rw_debug = avr_cpu_memory_rw_debug; > > > >> > +#endif > > > >> > +#ifdef CONFIG_USER_ONLY > > > >> > + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; > > > >> > +#else > > > >> > + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; > > > >> > + cc->vmsd = &vms_avr_cpu; > > > >> > +#endif > > > >> > + cc->disas_set_info = avr_cpu_disas_set_info; > > > >> > + cc->tlb_fill = avr_cpu_tlb_fill; > > > >> > + cc->tcg_initialize = avr_cpu_tcg_init; > > > >> > + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; > > > >> > + cc->gdb_read_register = avr_cpu_gdb_read_register; > > > >> > + cc->gdb_write_register = avr_cpu_gdb_write_register; > > > >> > + cc->gdb_num_core_regs = 35; > > > >> > + cc->gdb_core_xml_file = "avr-cpu.xml"; > > > >> > +} > > > >> > + > > > >> > +static void avr_avr1_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr2_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr25_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr3_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr31_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr35_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr4_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr5_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr51_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > +} > > > >> > + > > > >> > +static void avr_avr6_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > +} > > > >> > + > > > >> > +static void avr_xmega2_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > > >> > +} > > > >> > + > > > >> > +static void avr_xmega4_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > > >> > +} > > > >> > + > > > >> > +static void avr_xmega5_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > > >> > +} > > > >> > + > > > >> > +static void avr_xmega6_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > > >> > +} > > > >> > + > > > >> > +static void avr_xmega7_initfn(Object *obj) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(obj); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_LPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); > > > >> > + avr_set_feature(env, AVR_FEATURE_SRAM); > > > >> > + avr_set_feature(env, AVR_FEATURE_BREAK); > > > >> > + > > > >> > + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); > > > >> > + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPD); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPX); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPY); > > > >> > + avr_set_feature(env, AVR_FEATURE_RAMPZ); > > > >> > + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_ELPM); > > > >> > + avr_set_feature(env, AVR_FEATURE_JMP_CALL); > > > >> > + avr_set_feature(env, AVR_FEATURE_LPMX); > > > >> > + avr_set_feature(env, AVR_FEATURE_MOVW); > > > >> > + avr_set_feature(env, AVR_FEATURE_MUL); > > > >> > + avr_set_feature(env, AVR_FEATURE_RMW); > > > >> > +} > > > >> > + > > > >> > +typedef struct AVRCPUInfo { > > > >> > + const char *name; > > > >> > + void (*initfn)(Object *obj); > > > >> > +} AVRCPUInfo; > > > >> > + > > > >> > +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) > > > >> > +{ > > > >> > + ObjectClass *class_a = (ObjectClass *)a; > > > >> > + ObjectClass *class_b = (ObjectClass *)b; > > > >> > + const char *name_a; > > > >> > + const char *name_b; > > > >> > + > > > >> > + name_a = object_class_get_name(class_a); > > > >> > + name_b = object_class_get_name(class_b); > > > >> > + > > > >> > + return strcmp(name_a, name_b); > > > >> > +} > > > >> > + > > > >> > +static void avr_cpu_list_entry(gpointer data, gpointer user_data) > > > >> > +{ > > > >> > + const char *typename = > > object_class_get_name(OBJECT_CLASS(data)); > > > >> > + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); > > > >> > + qemu_printf("%.*s\n", len, typename); > > > >> > +} > > > >> > + > > > >> > +void avr_cpu_list(void) > > > >> > +{ > > > >> > + GSList *list; > > > >> > + list = object_class_get_list(TYPE_AVR_CPU, false); > > > >> > + list = g_slist_sort(list, avr_cpu_list_compare); > > > >> > + qemu_printf("Available CPUs:\n"); > > > >> > + g_slist_foreach(list, avr_cpu_list_entry, NULL); > > > >> > + g_slist_free(list); > > > >> > +} > > > >> > + > > > >> > +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ > > > >> > + { \ > > > >> > + .parent = TYPE_AVR_CPU, \ > > > >> > + .instance_init = initfn, \ > > > >> > + .name = AVR_CPU_TYPE_NAME(model), \ > > > >> > + } > > > >> > + > > > >> > +static const TypeInfo avr_cpu_type_info[] = { > > > >> > + { > > > >> > + .name = TYPE_AVR_CPU, > > > >> > + .parent = TYPE_CPU, > > > >> > + .instance_size = sizeof(AVRCPU), > > > >> > + .instance_init = avr_cpu_initfn, > > > >> > + .class_size = sizeof(AVRCPUClass), > > > >> > + .class_init = avr_cpu_class_init, > > > >> > + .abstract = true, > > > >> > + }, > > > >> > + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), > > > >> > + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), > > > >> > +}; > > > >> > + > > > >> > +DEFINE_TYPES(avr_cpu_type_info) > > > >> > diff --git a/target/avr/cpu.h b/target/avr/cpu.h > > > >> > new file mode 100644 > > > >> > index 0000000000..3f9a803193 > > > >> > --- /dev/null > > > >> > +++ b/target/avr/cpu.h > > > >> > @@ -0,0 +1,283 @@ > > > >> > +/* > > > >> > + * QEMU AVR CPU > > > >> > + * > > > >> > + * Copyright (c) 2016 Michael Rolnik > > > >> > + * > > > >> > + * 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 > > > >> > + * version 2.1 of the License, or (at your option) any later > > version. > > > >> > + * > > > >> > + * 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/lgpl-2.1.html> > > > >> > + */ > > > >> > + > > > >> > +#ifndef CPU_AVR_H > > > >> > +#define CPU_AVR_H > > > >> > + > > > >> > +#include "qemu-common.h" > > > >> > +#include "qom/cpu.h" > > > >> > +#include "exec/cpu-defs.h" > > > >> > +#include "fpu/softfloat.h" > > > >> > + > > > >> > +#define TCG_GUEST_DEFAULT_MO 0 > > > >> > + > > > >> > +#define TYPE_AVR_CPU "avr-cpu" > > > >> > + > > > >> > +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU > > > >> > +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) > > > >> > +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU > > > >> > + > > > >> > +/* > > > >> > + * AVR has two memory spaces, data & code. > > > >> > + * e.g. both have 0 address > > > >> > + * ST/LD instructions access data space > > > >> > + * LPM/SPM and instruction fetching access code memory space > > > >> > + */ > > > >> > +#define MMU_CODE_IDX 0 > > > >> > +#define MMU_DATA_IDX 1 > > > >> > + > > > >> > +#define EXCP_RESET 1 > > > >> > +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) > > > >> > + > > > >> > +/* Number of CPU registers */ > > > >> > +#define NO_CPU_REGISTERS 32 > > > >> > +/* Number of IO registers accessible by ld/st/in/out */ > > > >> > +#define NO_IO_REGISTERS 64 > > > >> > + > > > >> > +/* > > > >> > + * Offsets of AVR memory regions in host memory space. > > > >> > + * > > > >> > + * This is needed because the AVR has separate code and data > > address > > > >> > + * spaces that both have start from zero but have to go somewhere > > in > > > >> > + * host memory. > > > >> > + * > > > >> > + * It's also useful to know where some things are, like the IO > > > >> registers. > > > >> > + */ > > > >> > +/* Flash program memory */ > > > >> > +#define OFFSET_CODE 0x00000000 > > > >> > +/* CPU registers, IO registers, and SRAM */ > > > >> > +#define OFFSET_DATA 0x00800000 > > > >> > +/* CPU registers specifically, these are mapped at the start of > > data */ > > > >> > +#define OFFSET_CPU_REGISTERS OFFSET_DATA > > > >> > +/* > > > >> > + * IO registers, including status register, stack pointer, and > > memory > > > >> > + * mapped peripherals, mapped just after CPU registers > > > >> > + */ > > > >> > +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) > > > >> > + > > > >> > +enum avr_features { > > > >> > + AVR_FEATURE_SRAM, > > > >> > + > > > >> > + AVR_FEATURE_1_BYTE_PC, > > > >> > + AVR_FEATURE_2_BYTE_PC, > > > >> > + AVR_FEATURE_3_BYTE_PC, > > > >> > + > > > >> > + AVR_FEATURE_1_BYTE_SP, > > > >> > + AVR_FEATURE_2_BYTE_SP, > > > >> > + > > > >> > + AVR_FEATURE_BREAK, > > > >> > + AVR_FEATURE_DES, > > > >> > + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ > > > >> > + > > > >> > + AVR_FEATURE_EIJMP_EICALL, > > > >> > + AVR_FEATURE_IJMP_ICALL, > > > >> > + AVR_FEATURE_JMP_CALL, > > > >> > + > > > >> > + AVR_FEATURE_ADIW_SBIW, > > > >> > + > > > >> > + AVR_FEATURE_SPM, > > > >> > + AVR_FEATURE_SPMX, > > > >> > + > > > >> > + AVR_FEATURE_ELPMX, > > > >> > + AVR_FEATURE_ELPM, > > > >> > + AVR_FEATURE_LPMX, > > > >> > + AVR_FEATURE_LPM, > > > >> > + > > > >> > + AVR_FEATURE_MOVW, > > > >> > + AVR_FEATURE_MUL, > > > >> > + AVR_FEATURE_RAMPD, > > > >> > + AVR_FEATURE_RAMPX, > > > >> > + AVR_FEATURE_RAMPY, > > > >> > + AVR_FEATURE_RAMPZ, > > > >> > +}; > > > >> > + > > > >> > +typedef struct CPUAVRState CPUAVRState; > > > >> > + > > > >> > +struct CPUAVRState { > > > >> > + uint32_t pc_w; /* 0x003fffff up to 22 bits */ > > > >> > + > > > >> > + uint32_t sregC; /* 0x00000001 1 bits */ > > > >> > + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ > > > >> > + /* 0=flag set, >0=flag cleared */ > > > >> > + uint32_t sregN; /* 0x00000001 1 bits */ > > > >> > + uint32_t sregV; /* 0x00000001 1 bits */ > > > >> > + uint32_t sregS; /* 0x00000001 1 bits */ > > > >> > + uint32_t sregH; /* 0x00000001 1 bits */ > > > >> > + uint32_t sregT; /* 0x00000001 1 bits */ > > > >> > + uint32_t sregI; /* 0x00000001 1 bits */ > > > >> > + > > > >> > + uint32_t rampD; /* 0x00ff0000 8 bits */ > > > >> > + uint32_t rampX; /* 0x00ff0000 8 bits */ > > > >> > + uint32_t rampY; /* 0x00ff0000 8 bits */ > > > >> > + uint32_t rampZ; /* 0x00ff0000 8 bits */ > > > >> > + uint32_t eind; /* 0x00ff0000 8 bits */ > > > >> > + > > > >> > + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ > > > >> > + uint32_t sp; /* 16 bits */ > > > >> > + > > > >> > + uint32_t skip; /* if set skip instruction */ > > > >> > + > > > >> > + uint64_t intsrc; /* interrupt sources */ > > > >> > + bool fullacc; /* CPU/MEM if true MEM only otherwise */ > > > >> > + > > > >> > + uint32_t features; > > > >> > +}; > > > >> > + > > > >> > +#define AVR_CPU_CLASS(klass) \ > > > >> > + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) > > > >> > +#define AVR_CPU(obj) \ > > > >> > + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) > > > >> > +#define AVR_CPU_GET_CLASS(obj) \ > > > >> > + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) > > > >> > + > > > >> > +/** > > > >> > + * AVRCPUClass: > > > >> > + * @parent_realize: The parent class' realize handler. > > > >> > + * @parent_reset: The parent class' reset handler. > > > >> > + * @vr: Version Register value. > > > >> > + * > > > >> > + * A AVR CPU model. > > > >> > + */ > > > >> > +typedef struct AVRCPUClass { > > > >> > + /*< private >*/ > > > >> > + CPUClass parent_class; > > > >> > + /*< public >*/ > > > >> > + DeviceRealize parent_realize; > > > >> > + void (*parent_reset)(CPUState *cpu); > > > >> > +} AVRCPUClass; > > > >> > + > > > >> > +/** > > > >> > + * AVRCPU: > > > >> > + * @env: #CPUAVRState > > > >> > + * > > > >> > + * A AVR CPU. > > > >> > + */ > > > >> > +typedef struct AVRCPU { > > > >> > + /*< private >*/ > > > >> > + CPUState parent_obj; > > > >> > + /*< public >*/ > > > >> > + > > > >> > + CPUNegativeOffsetState neg; > > > >> > + CPUAVRState env; > > > >> > +} AVRCPU; > > > >> > + > > > >> > +#ifndef CONFIG_USER_ONLY > > > >> > +extern const struct VMStateDescription vms_avr_cpu; > > > >> > +#endif > > > >> > + > > > >> > +void avr_cpu_do_interrupt(CPUState *cpu); > > > >> > +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); > > > >> > +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); > > > >> > +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int > > reg); > > > >> > +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int > > reg); > > > >> > + > > > >> > +static inline int avr_feature(CPUAVRState *env, int feature) > > > >> > +{ > > > >> > + return (env->features & (1U << feature)) != 0; > > > >> > +} > > > >> > + > > > >> > +static inline void avr_set_feature(CPUAVRState *env, int feature) > > > >> > +{ > > > >> > + env->features |= (1U << feature); > > > >> > +} > > > >> > + > > > >> > +#define cpu_list avr_cpu_list > > > >> > +#define cpu_signal_handler cpu_avr_signal_handler > > > >> > +#define cpu_mmu_index avr_cpu_mmu_index > > > >> > + > > > >> > +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) > > > >> > +{ > > > >> > + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; > > > >> > +} > > > >> > + > > > >> > +void avr_cpu_tcg_init(void); > > > >> > + > > > >> > +void avr_cpu_list(void); > > > >> > +int cpu_avr_exec(CPUState *cpu); > > > >> > +int cpu_avr_signal_handler(int host_signum, void *pinfo, void > > *puc); > > > >> > +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int > > size, > > > >> > + int rw, int mmu_idx); > > > >> > +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t > > *buf, > > > >> > + int len, bool is_write); > > > >> > + > > > >> > +enum { > > > >> > + TB_FLAGS_FULL_ACCESS = 1, > > > >> > + TB_FLAGS_SKIP = 2, > > > >> > +}; > > > >> > + > > > >> > +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, > > target_ulong > > > >> *pc, > > > >> > + target_ulong *cs_base, uint32_t > > > >> *pflags) > > > >> > +{ > > > >> > + uint32_t flags = 0; > > > >> > + > > > >> > + *pc = env->pc_w * 2; > > > >> > + *cs_base = 0; > > > >> > + > > > >> > + if (env->fullacc) { > > > >> > + flags |= TB_FLAGS_FULL_ACCESS; > > > >> > + } > > > >> > + if (env->skip) { > > > >> > + flags |= TB_FLAGS_SKIP; > > > >> > + } > > > >> > + > > > >> > + *pflags = flags; > > > >> > +} > > > >> > + > > > >> > +static inline int cpu_interrupts_enabled(CPUAVRState *env) > > > >> > +{ > > > >> > + return env->sregI != 0; > > > >> > +} > > > >> > + > > > >> > +static inline uint8_t cpu_get_sreg(CPUAVRState *env) > > > >> > +{ > > > >> > + uint8_t sreg; > > > >> > + sreg = (env->sregC & 0x01) << 0 > > > >> > + | (env->sregZ == 0 ? 1 : 0) << 1 > > > >> > + | (env->sregN) << 2 > > > >> > + | (env->sregV) << 3 > > > >> > + | (env->sregS) << 4 > > > >> > + | (env->sregH) << 5 > > > >> > + | (env->sregT) << 6 > > > >> > + | (env->sregI) << 7; > > > >> > + return sreg; > > > >> > +} > > > >> > + > > > >> > +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) > > > >> > +{ > > > >> > + env->sregC = (sreg >> 0) & 0x01; > > > >> > + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; > > > >> > + env->sregN = (sreg >> 2) & 0x01; > > > >> > + env->sregV = (sreg >> 3) & 0x01; > > > >> > + env->sregS = (sreg >> 4) & 0x01; > > > >> > + env->sregH = (sreg >> 5) & 0x01; > > > >> > + env->sregT = (sreg >> 6) & 0x01; > > > >> > + env->sregI = (sreg >> 7) & 0x01; > > > >> > +} > > > >> > + > > > >> > +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, > > > >> > + MMUAccessType access_type, int mmu_idx, > > > >> > + bool probe, uintptr_t retaddr); > > > >> > + > > > >> > +typedef CPUAVRState CPUArchState; > > > >> > +typedef AVRCPU ArchCPU; > > > >> > + > > > >> > +#include "exec/cpu-all.h" > > > >> > + > > > >> > +#endif /* !defined (CPU_AVR_H) */ > > > >> > diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c > > > >> > new file mode 100644 > > > >> > index 0000000000..537dc7226e > > > >> > --- /dev/null > > > >> > +++ b/target/avr/gdbstub.c > > > >> > @@ -0,0 +1,85 @@ > > > >> > +/* > > > >> > + * QEMU AVR CPU > > > >> > + * > > > >> > + * Copyright (c) 2016 Michael Rolnik > > > >> > + * > > > >> > + * 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 > > > >> > + * version 2.1 of the License, or (at your option) any later > > version. > > > >> > + * > > > >> > + * 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/lgpl-2.1.html> > > > >> > + */ > > > >> > + > > > >> > +#include "qemu/osdep.h" > > > >> > +#include "qemu-common.h" > > > >> > +#include "exec/gdbstub.h" > > > >> > + > > > >> > +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int > > n) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + /* R */ > > > >> > + if (n < 32) { > > > >> > + return gdb_get_reg8(mem_buf, env->r[n]); > > > >> > + } > > > >> > + > > > >> > + /* SREG */ > > > >> > + if (n == 32) { > > > >> > + uint8_t sreg = cpu_get_sreg(env); > > > >> > + > > > >> > + return gdb_get_reg8(mem_buf, sreg); > > > >> > + } > > > >> > + > > > >> > + /* SP */ > > > >> > + if (n == 33) { > > > >> > + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); > > > >> > + } > > > >> > + > > > >> > + /* PC */ > > > >> > + if (n == 34) { > > > >> > + return gdb_get_reg32(mem_buf, env->pc_w * 2); > > > >> > + } > > > >> > + > > > >> > + return 0; > > > >> > +} > > > >> > + > > > >> > +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int > > n) > > > >> > +{ > > > >> > + AVRCPU *cpu = AVR_CPU(cs); > > > >> > + CPUAVRState *env = &cpu->env; > > > >> > + > > > >> > + /* R */ > > > >> > + if (n < 32) { > > > >> > + env->r[n] = *mem_buf; > > > >> > + return 1; > > > >> > + } > > > >> > + > > > >> > + /* SREG */ > > > >> > + if (n == 32) { > > > >> > + cpu_set_sreg(env, *mem_buf); > > > >> > + return 1; > > > >> > + } > > > >> > + > > > >> > + /* SP */ > > > >> > + if (n == 33) { > > > >> > + env->sp = lduw_p(mem_buf); > > > >> > + return 2; > > > >> > + } > > > >> > + > > > >> > + /* PC */ > > > >> > + if (n == 34) { > > > >> > + env->pc_w = ldl_p(mem_buf) / 2; > > > >> > + return 4; > > > >> > + } > > > >> > + > > > >> > + return 0; > > > >> > +} > > > >> > diff --git a/target/avr/machine.c b/target/avr/machine.c > > > >> > new file mode 100644 > > > >> > index 0000000000..389b5eb95d > > > >> > --- /dev/null > > > >> > +++ b/target/avr/machine.c > > > >> > @@ -0,0 +1,123 @@ > > > >> > +/* > > > >> > + * QEMU AVR CPU > > > >> > + * > > > >> > + * Copyright (c) 2016 Michael Rolnik > > > >> > + * > > > >> > + * 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 > > > >> > + * version 2.1 of the License, or (at your option) any later > > version. > > > >> > + * > > > >> > + * 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/lgpl-2.1.html> > > > >> > + */ > > > >> > + > > > >> > +#include "qemu/osdep.h" > > > >> > +#include "hw/hw.h" > > > >> > +#include "cpu.h" > > > >> > +#include "hw/boards.h" > > > >> > +#include "migration/qemu-file.h" > > > >> > + > > > >> > +static int get_sreg(QEMUFile *f, void *opaque, size_t size, > > > >> > + const VMStateField *field) > > > >> > +{ > > > >> > + CPUAVRState *env = opaque; > > > >> > + uint8_t sreg; > > > >> > + > > > >> > + sreg = qemu_get_byte(f); > > > >> > + cpu_set_sreg(env, sreg); > > > >> > + return 0; > > > >> > +} > > > >> > + > > > >> > +static int put_sreg( > > > >> > + QEMUFile *f, void *opaque, size_t size, > > > >> > + const VMStateField *field, QJSON *vmdesc) > > > >> > +{ > > > >> > + CPUAVRState *env = opaque; > > > >> > + uint8_t sreg = cpu_get_sreg(env); > > > >> > + > > > >> > + qemu_put_byte(f, sreg); > > > >> > + return 0; > > > >> > +} > > > >> > + > > > >> > +static const VMStateInfo vms_sreg = { > > > >> > + .name = "sreg", > > > >> > + .get = get_sreg, > > > >> > + .put = put_sreg, > > > >> > +}; > > > >> > + > > > >> > +static int get_segment( > > > >> > + QEMUFile *f, void *opaque, size_t size, const VMStateField > > *field) > > > >> > +{ > > > >> > + uint32_t *ramp = opaque; > > > >> > + uint8_t temp; > > > >> > + > > > >> > + temp = qemu_get_byte(f); > > > >> > + *ramp = ((uint32_t)temp) << 16; > > > >> > + return 0; > > > >> > +} > > > >> > + > > > >> > +static int put_segment( > > > >> > + QEMUFile *f, void *opaque, size_t size, > > > >> > + const VMStateField *field, QJSON *vmdesc) > > > >> > +{ > > > >> > + uint32_t *ramp = opaque; > > > >> > + uint8_t temp = *ramp >> 16; > > > >> > + > > > >> > + qemu_put_byte(f, temp); > > > >> > + return 0; > > > >> > +} > > > >> > + > > > >> > +static const VMStateInfo vms_rampD = { > > > >> > + .name = "rampD", > > > >> > + .get = get_segment, > > > >> > + .put = put_segment, > > > >> > +}; > > > >> > +static const VMStateInfo vms_rampX = { > > > >> > + .name = "rampX", > > > >> > + .get = get_segment, > > > >> > + .put = put_segment, > > > >> > +}; > > > >> > +static const VMStateInfo vms_rampY = { > > > >> > + .name = "rampY", > > > >> > + .get = get_segment, > > > >> > + .put = put_segment, > > > >> > +}; > > > >> > +static const VMStateInfo vms_rampZ = { > > > >> > + .name = "rampZ", > > > >> > + .get = get_segment, > > > >> > + .put = put_segment, > > > >> > +}; > > > >> > +static const VMStateInfo vms_eind = { > > > >> > + .name = "eind", > > > >> > + .get = get_segment, > > > >> > + .put = put_segment, > > > >> > +}; > > > >> > + > > > >> > +const VMStateDescription vms_avr_cpu = { > > > >> > + .name = "cpu", > > > >> > + .version_id = 0, > > > >> > + .minimum_version_id = 0, > > > >> > + .fields = (VMStateField[]) { > > > >> > + VMSTATE_UINT32(env.pc_w, AVRCPU), > > > >> > + VMSTATE_UINT32(env.sp, AVRCPU), > > > >> > + VMSTATE_UINT32(env.skip, AVRCPU), > > > >> > + > > > >> > + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), > > > >> > + > > > >> > + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), > > > >> > + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), > > > >> > + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), > > > >> > + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), > > > >> > + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), > > > >> > + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), > > > >> > + > > > >> > + VMSTATE_END_OF_LIST() > > > >> > + } > > > >> > +}; > > > >> > > > >> > > > > > > > > -- > > > > Best Regards, > > > > Michael Rolnik > > > > > > > >
On 6/28/19 11:22 AM, Igor Mammedov wrote: > On Fri, 28 Jun 2019 11:43:22 +0300 > Michael Rolnik <mrolnik@gmail.com> wrote: > >> Nobody else does it. > it doesn't mean that we should blindly copy pre-existing patterns > which might/will exist for compatibility reasons. > > pls see the same argument on RX CPU for reasons not to support > anything else beside typename there. > [Qemu-devel] [PATCH v16 13/23] target/rx: Fix cpu types and names > [Qemu-devel] [PATCH v19 03/21] target/rx: CPU definition > > >> Richard. >> What do you think? We are going with Igor's advice for target/rx, yes. You should do the same here in target/avr. r~
thanks. new version is out. Michael On Fri, Jun 28, 2019 at 1:01 PM Richard Henderson < richard.henderson@linaro.org> wrote: > On 6/28/19 11:22 AM, Igor Mammedov wrote: > > On Fri, 28 Jun 2019 11:43:22 +0300 > > Michael Rolnik <mrolnik@gmail.com> wrote: > > > >> Nobody else does it. > > it doesn't mean that we should blindly copy pre-existing patterns > > which might/will exist for compatibility reasons. > > > > pls see the same argument on RX CPU for reasons not to support > > anything else beside typename there. > > [Qemu-devel] [PATCH v16 13/23] target/rx: Fix cpu types and names > > [Qemu-devel] [PATCH v19 03/21] target/rx: CPU definition > > > > > >> Richard. > >> What do you think? > > We are going with Igor's advice for target/rx, yes. > You should do the same here in target/avr. > > > r~ >
diff --git a/gdb-xml/avr-cpu.xml b/gdb-xml/avr-cpu.xml new file mode 100644 index 0000000000..c4747f5b40 --- /dev/null +++ b/gdb-xml/avr-cpu.xml @@ -0,0 +1,49 @@ +<?xml version="1.0"?> +<!-- Copyright (C) 2018-2019 Free Software Foundation, Inc. + + Copying and distribution of this file, with or without modification, + are permitted in any medium without royalty provided the copyright + notice and this notice are preserved. --> + +<!-- Register numbers are hard-coded in order to maintain backward + compatibility with older versions of tools that didn't use xml + register descriptions. --> + +<!DOCTYPE feature SYSTEM "gdb-target.dtd"> +<feature name="org.gnu.gdb.riscv.cpu"> + <reg name="r0" bitsize="8" type="int" regnum="0"/> + <reg name="r1" bitsize="8" type="int"/> + <reg name="r2" bitsize="8" type="int"/> + <reg name="r3" bitsize="8" type="int"/> + <reg name="r4" bitsize="8" type="int"/> + <reg name="r5" bitsize="8" type="int"/> + <reg name="r6" bitsize="8" type="int"/> + <reg name="r7" bitsize="8" type="int"/> + <reg name="r8" bitsize="8" type="int"/> + <reg name="r9" bitsize="8" type="int"/> + <reg name="r10" bitsize="8" type="int"/> + <reg name="r11" bitsize="8" type="int"/> + <reg name="r12" bitsize="8" type="int"/> + <reg name="r13" bitsize="8" type="int"/> + <reg name="r14" bitsize="8" type="int"/> + <reg name="r15" bitsize="8" type="int"/> + <reg name="r16" bitsize="8" type="int"/> + <reg name="r17" bitsize="8" type="int"/> + <reg name="r18" bitsize="8" type="int"/> + <reg name="r19" bitsize="8" type="int"/> + <reg name="r20" bitsize="8" type="int"/> + <reg name="r21" bitsize="8" type="int"/> + <reg name="r22" bitsize="8" type="int"/> + <reg name="r23" bitsize="8" type="int"/> + <reg name="r24" bitsize="8" type="int"/> + <reg name="r25" bitsize="8" type="int"/> + <reg name="r26" bitsize="8" type="int"/> + <reg name="r27" bitsize="8" type="int"/> + <reg name="r28" bitsize="8" type="int"/> + <reg name="r29" bitsize="8" type="int"/> + <reg name="r30" bitsize="8" type="int"/> + <reg name="r31" bitsize="8" type="int"/> + <reg name="sreg" bitsize="8" type="int"/> + <reg name="sp" bitsize="8" type="int"/> + <reg name="pc" bitsize="8" type="int"/> +</feature> diff --git a/target/avr/cpu-param.h b/target/avr/cpu-param.h new file mode 100644 index 0000000000..5bbf985726 --- /dev/null +++ b/target/avr/cpu-param.h @@ -0,0 +1,37 @@ +/* + * QEMU AVR CPU + * + * Copyright (c) 2016 Michael Rolnik + * + * 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 + * version 2.1 of the License, or (at your option) any later version. + * + * 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/lgpl-2.1.html> + */ + +#ifndef AVR_CPU_PARAM_H +#define AVR_CPU_PARAM_H 1 + +#define TARGET_LONG_BITS 32 +/* + * TARGET_PAGE_BITS cannot be more than 8 bits because + * 1. all IO registers occupy [0x0000 .. 0x00ff] address range, and they + * should be implemented as a device and not memory + * 2. SRAM starts at the address 0x0100 + */ +#define TARGET_PAGE_BITS 8 +#define TARGET_PHYS_ADDR_SPACE_BITS 24 +#define TARGET_VIRT_ADDR_SPACE_BITS 24 +#define NB_MMU_MODES 2 + + +#endif diff --git a/target/avr/cpu.c b/target/avr/cpu.c new file mode 100644 index 0000000000..142fe54524 --- /dev/null +++ b/target/avr/cpu.c @@ -0,0 +1,599 @@ +/* + * QEMU AVR CPU + * + * Copyright (c) 2016 Michael Rolnik + * + * 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 + * version 2.1 of the License, or (at your option) any later version. + * + * 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/lgpl-2.1.html> + */ + +#include "qemu/osdep.h" +#include "qemu/qemu-print.h" +#include "qemu/log.h" +#include "cpu.h" +#include "exec/exec-all.h" +#include "qapi/error.h" +#include "hw/qdev-properties.h" +#include "migration/vmstate.h" + +static void avr_cpu_set_pc(CPUState *cs, vaddr value) +{ + AVRCPU *cpu = AVR_CPU(cs); + + cpu->env.pc_w = value / 2; /* internally PC points to words */ +} + +static bool avr_cpu_has_work(CPUState *cs) +{ + AVRCPU *cpu = AVR_CPU(cs); + CPUAVRState *env = &cpu->env; + + return (cs->interrupt_request & (CPU_INTERRUPT_HARD | CPU_INTERRUPT_RESET)) + && cpu_interrupts_enabled(env); +} + +static void avr_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock *tb) +{ + AVRCPU *cpu = AVR_CPU(cs); + CPUAVRState *env = &cpu->env; + + env->pc_w = tb->pc / 2; /* internally PC points to words */ +} + +static void avr_cpu_reset(CPUState *cs) +{ + AVRCPU *cpu = AVR_CPU(cs); + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(cpu); + CPUAVRState *env = &cpu->env; + + mcc->parent_reset(cs); + + env->pc_w = 0; + env->sregI = 1; + env->sregC = 0; + env->sregZ = 0; + env->sregN = 0; + env->sregV = 0; + env->sregS = 0; + env->sregH = 0; + env->sregT = 0; + + env->rampD = 0; + env->rampX = 0; + env->rampY = 0; + env->rampZ = 0; + env->eind = 0; + env->sp = 0; + + env->skip = 0; + + memset(env->r, 0, sizeof(env->r)); + + tlb_flush(cs); +} + +static void avr_cpu_disas_set_info(CPUState *cpu, disassemble_info *info) +{ + info->mach = bfd_arch_avr; + info->print_insn = NULL; +} + +static void avr_cpu_realizefn(DeviceState *dev, Error **errp) +{ + CPUState *cs = CPU(dev); + AVRCPUClass *mcc = AVR_CPU_GET_CLASS(dev); + Error *local_err = NULL; + + cpu_exec_realizefn(cs, &local_err); + if (local_err != NULL) { + error_propagate(errp, local_err); + return; + } + qemu_init_vcpu(cs); + cpu_reset(cs); + + mcc->parent_realize(dev, errp); +} + +static void avr_cpu_set_int(void *opaque, int irq, int level) +{ + AVRCPU *cpu = opaque; + CPUAVRState *env = &cpu->env; + CPUState *cs = CPU(cpu); + + uint64_t mask = (1ull << irq); + if (level) { + env->intsrc |= mask; + cpu_interrupt(cs, CPU_INTERRUPT_HARD); + } else { + env->intsrc &= ~mask; + if (env->intsrc == 0) { + cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); + } + } +} + +static void avr_cpu_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + + cpu_set_cpustate_pointers(cpu); + +#ifndef CONFIG_USER_ONLY + /* Set the number of interrupts supported by the CPU. */ + qdev_init_gpio_in(DEVICE(cpu), avr_cpu_set_int, 57); +#endif +} + +static ObjectClass *avr_cpu_class_by_name(const char *cpu_model) +{ + ObjectClass *oc; + char *typename; + char **cpuname; + + cpuname = g_strsplit(cpu_model, ",", 1); + typename = g_strdup_printf(AVR_CPU_TYPE_NAME("%s"), cpuname[0]); + oc = object_class_by_name(typename); + g_strfreev(cpuname); + g_free(typename); + if (!oc || !object_class_dynamic_cast(oc, TYPE_AVR_CPU) || + object_class_is_abstract(oc)) { + return NULL; + } + return oc; +} + +static void avr_cpu_dump_state(CPUState *cs, FILE *f, int flags) +{ + AVRCPU *cpu = AVR_CPU(cs); + CPUAVRState *env = &cpu->env; + int i; + + qemu_fprintf(f, "\n"); + qemu_fprintf(f, "PC: %06x\n", env->pc_w); + qemu_fprintf(f, "SP: %04x\n", env->sp); + qemu_fprintf(f, "rampD: %02x\n", env->rampD >> 16); + qemu_fprintf(f, "rampX: %02x\n", env->rampX >> 16); + qemu_fprintf(f, "rampY: %02x\n", env->rampY >> 16); + qemu_fprintf(f, "rampZ: %02x\n", env->rampZ >> 16); + qemu_fprintf(f, "EIND: %02x\n", env->eind >> 16); + qemu_fprintf(f, "X: %02x%02x\n", env->r[27], env->r[26]); + qemu_fprintf(f, "Y: %02x%02x\n", env->r[29], env->r[28]); + qemu_fprintf(f, "Z: %02x%02x\n", env->r[31], env->r[30]); + qemu_fprintf(f, "SREG: [ %c %c %c %c %c %c %c %c ]\n", + env->sregI ? 'I' : '-', + env->sregT ? 'T' : '-', + env->sregH ? 'H' : '-', + env->sregS ? 'S' : '-', + env->sregV ? 'V' : '-', + env->sregN ? '-' : 'N', /* Zf has negative logic */ + env->sregZ ? 'Z' : '-', + env->sregC ? 'I' : '-'); + qemu_fprintf(f, "SKIP: %02x\n", env->skip); + + qemu_fprintf(f, "\n"); + for (i = 0; i < ARRAY_SIZE(env->r); i++) { + qemu_fprintf(f, "R[%02d]: %02x ", i, env->r[i]); + + if ((i % 8) == 7) { + qemu_fprintf(f, "\n"); + } + } + qemu_fprintf(f, "\n"); +} + +static void avr_cpu_class_init(ObjectClass *oc, void *data) +{ + DeviceClass *dc = DEVICE_CLASS(oc); + CPUClass *cc = CPU_CLASS(oc); + AVRCPUClass *mcc = AVR_CPU_CLASS(oc); + + mcc->parent_realize = dc->realize; + dc->realize = avr_cpu_realizefn; + + mcc->parent_reset = cc->reset; + cc->reset = avr_cpu_reset; + + cc->class_by_name = avr_cpu_class_by_name; + + cc->has_work = avr_cpu_has_work; + cc->do_interrupt = avr_cpu_do_interrupt; + cc->cpu_exec_interrupt = avr_cpu_exec_interrupt; + cc->dump_state = avr_cpu_dump_state; + cc->set_pc = avr_cpu_set_pc; +#if !defined(CONFIG_USER_ONLY) + cc->memory_rw_debug = avr_cpu_memory_rw_debug; +#endif +#ifdef CONFIG_USER_ONLY + cc->handle_mmu_fault = avr_cpu_handle_mmu_fault; +#else + cc->get_phys_page_debug = avr_cpu_get_phys_page_debug; + cc->vmsd = &vms_avr_cpu; +#endif + cc->disas_set_info = avr_cpu_disas_set_info; + cc->tlb_fill = avr_cpu_tlb_fill; + cc->tcg_initialize = avr_cpu_tcg_init; + cc->synchronize_from_tb = avr_cpu_synchronize_from_tb; + cc->gdb_read_register = avr_cpu_gdb_read_register; + cc->gdb_write_register = avr_cpu_gdb_write_register; + cc->gdb_num_core_regs = 35; + cc->gdb_core_xml_file = "avr-cpu.xml"; +} + +static void avr_avr1_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); +} + +static void avr_avr2_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); +} + +static void avr_avr25_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); +} + +static void avr_avr3_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); +} + +static void avr_avr31_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); +} + +static void avr_avr35_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); +} + +static void avr_avr4_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); +} + +static void avr_avr5_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); +} + +static void avr_avr51_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_ELPMX); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); +} + +static void avr_avr6_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); + avr_set_feature(env, AVR_FEATURE_ELPMX); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); +} + +static void avr_xmega2_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); + avr_set_feature(env, AVR_FEATURE_RMW); +} + +static void avr_xmega4_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_ELPMX); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); + avr_set_feature(env, AVR_FEATURE_RMW); +} + +static void avr_xmega5_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_2_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPD); + avr_set_feature(env, AVR_FEATURE_RAMPX); + avr_set_feature(env, AVR_FEATURE_RAMPY); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_ELPMX); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); + avr_set_feature(env, AVR_FEATURE_RMW); +} + +static void avr_xmega6_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); + avr_set_feature(env, AVR_FEATURE_ELPMX); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); + avr_set_feature(env, AVR_FEATURE_RMW); +} + +static void avr_xmega7_initfn(Object *obj) +{ + AVRCPU *cpu = AVR_CPU(obj); + CPUAVRState *env = &cpu->env; + + avr_set_feature(env, AVR_FEATURE_LPM); + avr_set_feature(env, AVR_FEATURE_IJMP_ICALL); + avr_set_feature(env, AVR_FEATURE_ADIW_SBIW); + avr_set_feature(env, AVR_FEATURE_SRAM); + avr_set_feature(env, AVR_FEATURE_BREAK); + + avr_set_feature(env, AVR_FEATURE_3_BYTE_PC); + avr_set_feature(env, AVR_FEATURE_2_BYTE_SP); + avr_set_feature(env, AVR_FEATURE_RAMPD); + avr_set_feature(env, AVR_FEATURE_RAMPX); + avr_set_feature(env, AVR_FEATURE_RAMPY); + avr_set_feature(env, AVR_FEATURE_RAMPZ); + avr_set_feature(env, AVR_FEATURE_EIJMP_EICALL); + avr_set_feature(env, AVR_FEATURE_ELPMX); + avr_set_feature(env, AVR_FEATURE_ELPM); + avr_set_feature(env, AVR_FEATURE_JMP_CALL); + avr_set_feature(env, AVR_FEATURE_LPMX); + avr_set_feature(env, AVR_FEATURE_MOVW); + avr_set_feature(env, AVR_FEATURE_MUL); + avr_set_feature(env, AVR_FEATURE_RMW); +} + +typedef struct AVRCPUInfo { + const char *name; + void (*initfn)(Object *obj); +} AVRCPUInfo; + +static gint avr_cpu_list_compare(gconstpointer a, gconstpointer b) +{ + ObjectClass *class_a = (ObjectClass *)a; + ObjectClass *class_b = (ObjectClass *)b; + const char *name_a; + const char *name_b; + + name_a = object_class_get_name(class_a); + name_b = object_class_get_name(class_b); + + return strcmp(name_a, name_b); +} + +static void avr_cpu_list_entry(gpointer data, gpointer user_data) +{ + const char *typename = object_class_get_name(OBJECT_CLASS(data)); + int len = strlen(typename) - strlen(AVR_CPU_TYPE_SUFFIX); + qemu_printf("%.*s\n", len, typename); +} + +void avr_cpu_list(void) +{ + GSList *list; + list = object_class_get_list(TYPE_AVR_CPU, false); + list = g_slist_sort(list, avr_cpu_list_compare); + qemu_printf("Available CPUs:\n"); + g_slist_foreach(list, avr_cpu_list_entry, NULL); + g_slist_free(list); +} + +#define DEFINE_AVR_CPU_TYPE(model, initfn) \ + { \ + .parent = TYPE_AVR_CPU, \ + .instance_init = initfn, \ + .name = AVR_CPU_TYPE_NAME(model), \ + } + +static const TypeInfo avr_cpu_type_info[] = { + { + .name = TYPE_AVR_CPU, + .parent = TYPE_CPU, + .instance_size = sizeof(AVRCPU), + .instance_init = avr_cpu_initfn, + .class_size = sizeof(AVRCPUClass), + .class_init = avr_cpu_class_init, + .abstract = true, + }, + DEFINE_AVR_CPU_TYPE("avr1", avr_avr1_initfn), + DEFINE_AVR_CPU_TYPE("avr2", avr_avr2_initfn), + DEFINE_AVR_CPU_TYPE("avr25", avr_avr25_initfn), + DEFINE_AVR_CPU_TYPE("avr3", avr_avr3_initfn), + DEFINE_AVR_CPU_TYPE("avr31", avr_avr31_initfn), + DEFINE_AVR_CPU_TYPE("avr35", avr_avr35_initfn), + DEFINE_AVR_CPU_TYPE("avr4", avr_avr4_initfn), + DEFINE_AVR_CPU_TYPE("avr5", avr_avr5_initfn), + DEFINE_AVR_CPU_TYPE("avr51", avr_avr51_initfn), + DEFINE_AVR_CPU_TYPE("avr6", avr_avr6_initfn), + DEFINE_AVR_CPU_TYPE("xmega2", avr_xmega2_initfn), + DEFINE_AVR_CPU_TYPE("xmega4", avr_xmega4_initfn), + DEFINE_AVR_CPU_TYPE("xmega5", avr_xmega5_initfn), + DEFINE_AVR_CPU_TYPE("xmega6", avr_xmega6_initfn), + DEFINE_AVR_CPU_TYPE("xmega7", avr_xmega7_initfn), +}; + +DEFINE_TYPES(avr_cpu_type_info) diff --git a/target/avr/cpu.h b/target/avr/cpu.h new file mode 100644 index 0000000000..3f9a803193 --- /dev/null +++ b/target/avr/cpu.h @@ -0,0 +1,283 @@ +/* + * QEMU AVR CPU + * + * Copyright (c) 2016 Michael Rolnik + * + * 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 + * version 2.1 of the License, or (at your option) any later version. + * + * 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/lgpl-2.1.html> + */ + +#ifndef CPU_AVR_H +#define CPU_AVR_H + +#include "qemu-common.h" +#include "qom/cpu.h" +#include "exec/cpu-defs.h" +#include "fpu/softfloat.h" + +#define TCG_GUEST_DEFAULT_MO 0 + +#define TYPE_AVR_CPU "avr-cpu" + +#define AVR_CPU_TYPE_SUFFIX "-" TYPE_AVR_CPU +#define AVR_CPU_TYPE_NAME(name) (name AVR_CPU_TYPE_SUFFIX) +#define CPU_RESOLVING_TYPE TYPE_AVR_CPU + +/* + * AVR has two memory spaces, data & code. + * e.g. both have 0 address + * ST/LD instructions access data space + * LPM/SPM and instruction fetching access code memory space + */ +#define MMU_CODE_IDX 0 +#define MMU_DATA_IDX 1 + +#define EXCP_RESET 1 +#define EXCP_INT(n) (EXCP_RESET + (n) + 1) + +/* Number of CPU registers */ +#define NO_CPU_REGISTERS 32 +/* Number of IO registers accessible by ld/st/in/out */ +#define NO_IO_REGISTERS 64 + +/* + * Offsets of AVR memory regions in host memory space. + * + * This is needed because the AVR has separate code and data address + * spaces that both have start from zero but have to go somewhere in + * host memory. + * + * It's also useful to know where some things are, like the IO registers. + */ +/* Flash program memory */ +#define OFFSET_CODE 0x00000000 +/* CPU registers, IO registers, and SRAM */ +#define OFFSET_DATA 0x00800000 +/* CPU registers specifically, these are mapped at the start of data */ +#define OFFSET_CPU_REGISTERS OFFSET_DATA +/* + * IO registers, including status register, stack pointer, and memory + * mapped peripherals, mapped just after CPU registers + */ +#define OFFSET_IO_REGISTERS (OFFSET_DATA + NO_CPU_REGISTERS) + +enum avr_features { + AVR_FEATURE_SRAM, + + AVR_FEATURE_1_BYTE_PC, + AVR_FEATURE_2_BYTE_PC, + AVR_FEATURE_3_BYTE_PC, + + AVR_FEATURE_1_BYTE_SP, + AVR_FEATURE_2_BYTE_SP, + + AVR_FEATURE_BREAK, + AVR_FEATURE_DES, + AVR_FEATURE_RMW, /* Read Modify Write - XCH LAC LAS LAT */ + + AVR_FEATURE_EIJMP_EICALL, + AVR_FEATURE_IJMP_ICALL, + AVR_FEATURE_JMP_CALL, + + AVR_FEATURE_ADIW_SBIW, + + AVR_FEATURE_SPM, + AVR_FEATURE_SPMX, + + AVR_FEATURE_ELPMX, + AVR_FEATURE_ELPM, + AVR_FEATURE_LPMX, + AVR_FEATURE_LPM, + + AVR_FEATURE_MOVW, + AVR_FEATURE_MUL, + AVR_FEATURE_RAMPD, + AVR_FEATURE_RAMPX, + AVR_FEATURE_RAMPY, + AVR_FEATURE_RAMPZ, +}; + +typedef struct CPUAVRState CPUAVRState; + +struct CPUAVRState { + uint32_t pc_w; /* 0x003fffff up to 22 bits */ + + uint32_t sregC; /* 0x00000001 1 bits */ + uint32_t sregZ; /* 0x0000ffff 16 bits, negative logic; */ + /* 0=flag set, >0=flag cleared */ + uint32_t sregN; /* 0x00000001 1 bits */ + uint32_t sregV; /* 0x00000001 1 bits */ + uint32_t sregS; /* 0x00000001 1 bits */ + uint32_t sregH; /* 0x00000001 1 bits */ + uint32_t sregT; /* 0x00000001 1 bits */ + uint32_t sregI; /* 0x00000001 1 bits */ + + uint32_t rampD; /* 0x00ff0000 8 bits */ + uint32_t rampX; /* 0x00ff0000 8 bits */ + uint32_t rampY; /* 0x00ff0000 8 bits */ + uint32_t rampZ; /* 0x00ff0000 8 bits */ + uint32_t eind; /* 0x00ff0000 8 bits */ + + uint32_t r[NO_CPU_REGISTERS]; /* 8 bits each */ + uint32_t sp; /* 16 bits */ + + uint32_t skip; /* if set skip instruction */ + + uint64_t intsrc; /* interrupt sources */ + bool fullacc; /* CPU/MEM if true MEM only otherwise */ + + uint32_t features; +}; + +#define AVR_CPU_CLASS(klass) \ + OBJECT_CLASS_CHECK(AVRCPUClass, (klass), TYPE_AVR_CPU) +#define AVR_CPU(obj) \ + OBJECT_CHECK(AVRCPU, (obj), TYPE_AVR_CPU) +#define AVR_CPU_GET_CLASS(obj) \ + OBJECT_GET_CLASS(AVRCPUClass, (obj), TYPE_AVR_CPU) + +/** + * AVRCPUClass: + * @parent_realize: The parent class' realize handler. + * @parent_reset: The parent class' reset handler. + * @vr: Version Register value. + * + * A AVR CPU model. + */ +typedef struct AVRCPUClass { + /*< private >*/ + CPUClass parent_class; + /*< public >*/ + DeviceRealize parent_realize; + void (*parent_reset)(CPUState *cpu); +} AVRCPUClass; + +/** + * AVRCPU: + * @env: #CPUAVRState + * + * A AVR CPU. + */ +typedef struct AVRCPU { + /*< private >*/ + CPUState parent_obj; + /*< public >*/ + + CPUNegativeOffsetState neg; + CPUAVRState env; +} AVRCPU; + +#ifndef CONFIG_USER_ONLY +extern const struct VMStateDescription vms_avr_cpu; +#endif + +void avr_cpu_do_interrupt(CPUState *cpu); +bool avr_cpu_exec_interrupt(CPUState *cpu, int int_req); +hwaddr avr_cpu_get_phys_page_debug(CPUState *cpu, vaddr addr); +int avr_cpu_gdb_read_register(CPUState *cpu, uint8_t *buf, int reg); +int avr_cpu_gdb_write_register(CPUState *cpu, uint8_t *buf, int reg); + +static inline int avr_feature(CPUAVRState *env, int feature) +{ + return (env->features & (1U << feature)) != 0; +} + +static inline void avr_set_feature(CPUAVRState *env, int feature) +{ + env->features |= (1U << feature); +} + +#define cpu_list avr_cpu_list +#define cpu_signal_handler cpu_avr_signal_handler +#define cpu_mmu_index avr_cpu_mmu_index + +static inline int avr_cpu_mmu_index(CPUAVRState *env, bool ifetch) +{ + return ifetch ? MMU_CODE_IDX : MMU_DATA_IDX; +} + +void avr_cpu_tcg_init(void); + +void avr_cpu_list(void); +int cpu_avr_exec(CPUState *cpu); +int cpu_avr_signal_handler(int host_signum, void *pinfo, void *puc); +int avr_cpu_handle_mmu_fault(CPUState *cpu, vaddr address, int size, + int rw, int mmu_idx); +int avr_cpu_memory_rw_debug(CPUState *cs, vaddr address, uint8_t *buf, + int len, bool is_write); + +enum { + TB_FLAGS_FULL_ACCESS = 1, + TB_FLAGS_SKIP = 2, +}; + +static inline void cpu_get_tb_cpu_state(CPUAVRState *env, target_ulong *pc, + target_ulong *cs_base, uint32_t *pflags) +{ + uint32_t flags = 0; + + *pc = env->pc_w * 2; + *cs_base = 0; + + if (env->fullacc) { + flags |= TB_FLAGS_FULL_ACCESS; + } + if (env->skip) { + flags |= TB_FLAGS_SKIP; + } + + *pflags = flags; +} + +static inline int cpu_interrupts_enabled(CPUAVRState *env) +{ + return env->sregI != 0; +} + +static inline uint8_t cpu_get_sreg(CPUAVRState *env) +{ + uint8_t sreg; + sreg = (env->sregC & 0x01) << 0 + | (env->sregZ == 0 ? 1 : 0) << 1 + | (env->sregN) << 2 + | (env->sregV) << 3 + | (env->sregS) << 4 + | (env->sregH) << 5 + | (env->sregT) << 6 + | (env->sregI) << 7; + return sreg; +} + +static inline void cpu_set_sreg(CPUAVRState *env, uint8_t sreg) +{ + env->sregC = (sreg >> 0) & 0x01; + env->sregZ = (sreg >> 1) & 0x01 ? 0 : 1; + env->sregN = (sreg >> 2) & 0x01; + env->sregV = (sreg >> 3) & 0x01; + env->sregS = (sreg >> 4) & 0x01; + env->sregH = (sreg >> 5) & 0x01; + env->sregT = (sreg >> 6) & 0x01; + env->sregI = (sreg >> 7) & 0x01; +} + +bool avr_cpu_tlb_fill(CPUState *cs, vaddr address, int size, + MMUAccessType access_type, int mmu_idx, + bool probe, uintptr_t retaddr); + +typedef CPUAVRState CPUArchState; +typedef AVRCPU ArchCPU; + +#include "exec/cpu-all.h" + +#endif /* !defined (CPU_AVR_H) */ diff --git a/target/avr/gdbstub.c b/target/avr/gdbstub.c new file mode 100644 index 0000000000..537dc7226e --- /dev/null +++ b/target/avr/gdbstub.c @@ -0,0 +1,85 @@ +/* + * QEMU AVR CPU + * + * Copyright (c) 2016 Michael Rolnik + * + * 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 + * version 2.1 of the License, or (at your option) any later version. + * + * 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/lgpl-2.1.html> + */ + +#include "qemu/osdep.h" +#include "qemu-common.h" +#include "exec/gdbstub.h" + +int avr_cpu_gdb_read_register(CPUState *cs, uint8_t *mem_buf, int n) +{ + AVRCPU *cpu = AVR_CPU(cs); + CPUAVRState *env = &cpu->env; + + /* R */ + if (n < 32) { + return gdb_get_reg8(mem_buf, env->r[n]); + } + + /* SREG */ + if (n == 32) { + uint8_t sreg = cpu_get_sreg(env); + + return gdb_get_reg8(mem_buf, sreg); + } + + /* SP */ + if (n == 33) { + return gdb_get_reg16(mem_buf, env->sp & 0x0000ffff); + } + + /* PC */ + if (n == 34) { + return gdb_get_reg32(mem_buf, env->pc_w * 2); + } + + return 0; +} + +int avr_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n) +{ + AVRCPU *cpu = AVR_CPU(cs); + CPUAVRState *env = &cpu->env; + + /* R */ + if (n < 32) { + env->r[n] = *mem_buf; + return 1; + } + + /* SREG */ + if (n == 32) { + cpu_set_sreg(env, *mem_buf); + return 1; + } + + /* SP */ + if (n == 33) { + env->sp = lduw_p(mem_buf); + return 2; + } + + /* PC */ + if (n == 34) { + env->pc_w = ldl_p(mem_buf) / 2; + return 4; + } + + return 0; +} diff --git a/target/avr/machine.c b/target/avr/machine.c new file mode 100644 index 0000000000..389b5eb95d --- /dev/null +++ b/target/avr/machine.c @@ -0,0 +1,123 @@ +/* + * QEMU AVR CPU + * + * Copyright (c) 2016 Michael Rolnik + * + * 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 + * version 2.1 of the License, or (at your option) any later version. + * + * 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/lgpl-2.1.html> + */ + +#include "qemu/osdep.h" +#include "hw/hw.h" +#include "cpu.h" +#include "hw/boards.h" +#include "migration/qemu-file.h" + +static int get_sreg(QEMUFile *f, void *opaque, size_t size, + const VMStateField *field) +{ + CPUAVRState *env = opaque; + uint8_t sreg; + + sreg = qemu_get_byte(f); + cpu_set_sreg(env, sreg); + return 0; +} + +static int put_sreg( + QEMUFile *f, void *opaque, size_t size, + const VMStateField *field, QJSON *vmdesc) +{ + CPUAVRState *env = opaque; + uint8_t sreg = cpu_get_sreg(env); + + qemu_put_byte(f, sreg); + return 0; +} + +static const VMStateInfo vms_sreg = { + .name = "sreg", + .get = get_sreg, + .put = put_sreg, +}; + +static int get_segment( + QEMUFile *f, void *opaque, size_t size, const VMStateField *field) +{ + uint32_t *ramp = opaque; + uint8_t temp; + + temp = qemu_get_byte(f); + *ramp = ((uint32_t)temp) << 16; + return 0; +} + +static int put_segment( + QEMUFile *f, void *opaque, size_t size, + const VMStateField *field, QJSON *vmdesc) +{ + uint32_t *ramp = opaque; + uint8_t temp = *ramp >> 16; + + qemu_put_byte(f, temp); + return 0; +} + +static const VMStateInfo vms_rampD = { + .name = "rampD", + .get = get_segment, + .put = put_segment, +}; +static const VMStateInfo vms_rampX = { + .name = "rampX", + .get = get_segment, + .put = put_segment, +}; +static const VMStateInfo vms_rampY = { + .name = "rampY", + .get = get_segment, + .put = put_segment, +}; +static const VMStateInfo vms_rampZ = { + .name = "rampZ", + .get = get_segment, + .put = put_segment, +}; +static const VMStateInfo vms_eind = { + .name = "eind", + .get = get_segment, + .put = put_segment, +}; + +const VMStateDescription vms_avr_cpu = { + .name = "cpu", + .version_id = 0, + .minimum_version_id = 0, + .fields = (VMStateField[]) { + VMSTATE_UINT32(env.pc_w, AVRCPU), + VMSTATE_UINT32(env.sp, AVRCPU), + VMSTATE_UINT32(env.skip, AVRCPU), + + VMSTATE_UINT32_ARRAY(env.r, AVRCPU, NO_CPU_REGISTERS), + + VMSTATE_SINGLE(env, AVRCPU, 0, vms_sreg, CPUAVRState), + VMSTATE_SINGLE(env.rampD, AVRCPU, 0, vms_rampD, uint32_t), + VMSTATE_SINGLE(env.rampX, AVRCPU, 0, vms_rampX, uint32_t), + VMSTATE_SINGLE(env.rampY, AVRCPU, 0, vms_rampY, uint32_t), + VMSTATE_SINGLE(env.rampZ, AVRCPU, 0, vms_rampZ, uint32_t), + VMSTATE_SINGLE(env.eind, AVRCPU, 0, vms_eind, uint32_t), + + VMSTATE_END_OF_LIST() + } +};