xref: /openbmc/qemu/target/xtensa/helper.c (revision e692f9c6a681de1372a41999b14a947a553b6a1a)
1fcf5ef2aSThomas Huth /*
2fcf5ef2aSThomas Huth  * Copyright (c) 2011, Max Filippov, Open Source and Linux Lab.
3fcf5ef2aSThomas Huth  * All rights reserved.
4fcf5ef2aSThomas Huth  *
5fcf5ef2aSThomas Huth  * Redistribution and use in source and binary forms, with or without
6fcf5ef2aSThomas Huth  * modification, are permitted provided that the following conditions are met:
7fcf5ef2aSThomas Huth  *     * Redistributions of source code must retain the above copyright
8fcf5ef2aSThomas Huth  *       notice, this list of conditions and the following disclaimer.
9fcf5ef2aSThomas Huth  *     * Redistributions in binary form must reproduce the above copyright
10fcf5ef2aSThomas Huth  *       notice, this list of conditions and the following disclaimer in the
11fcf5ef2aSThomas Huth  *       documentation and/or other materials provided with the distribution.
12fcf5ef2aSThomas Huth  *     * Neither the name of the Open Source and Linux Lab nor the
13fcf5ef2aSThomas Huth  *       names of its contributors may be used to endorse or promote products
14fcf5ef2aSThomas Huth  *       derived from this software without specific prior written permission.
15fcf5ef2aSThomas Huth  *
16fcf5ef2aSThomas Huth  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17fcf5ef2aSThomas Huth  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18fcf5ef2aSThomas Huth  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19fcf5ef2aSThomas Huth  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
20fcf5ef2aSThomas Huth  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21fcf5ef2aSThomas Huth  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22fcf5ef2aSThomas Huth  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23fcf5ef2aSThomas Huth  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24fcf5ef2aSThomas Huth  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25fcf5ef2aSThomas Huth  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26fcf5ef2aSThomas Huth  */
27fcf5ef2aSThomas Huth 
28fcf5ef2aSThomas Huth #include "qemu/osdep.h"
29cd617484SPhilippe Mathieu-Daudé #include "qemu/log.h"
30fcf5ef2aSThomas Huth #include "cpu.h"
31fcf5ef2aSThomas Huth #include "exec/exec-all.h"
324ea5fe99SAlex Bennée #include "gdbstub/helpers.h"
339584116fSMax Filippov #include "exec/helper-proto.h"
3475903973SMax Filippov #include "qemu/error-report.h"
350442428aSMarkus Armbruster #include "qemu/qemu-print.h"
36fcf5ef2aSThomas Huth #include "qemu/host-utils.h"
37fcf5ef2aSThomas Huth 
38fcf5ef2aSThomas Huth static struct XtensaConfigList *xtensa_cores;
39fcf5ef2aSThomas Huth 
add_translator_to_hash(GHashTable * translator,const char * name,const XtensaOpcodeOps * opcode)4075903973SMax Filippov static void add_translator_to_hash(GHashTable *translator,
4175903973SMax Filippov                                    const char *name,
4275903973SMax Filippov                                    const XtensaOpcodeOps *opcode)
4375903973SMax Filippov {
4475903973SMax Filippov     if (!g_hash_table_insert(translator, (void *)name, (void *)opcode)) {
4575903973SMax Filippov         error_report("Multiple definitions of '%s' opcode in a single table",
4675903973SMax Filippov                      name);
4775903973SMax Filippov     }
4875903973SMax Filippov }
4975903973SMax Filippov 
hash_opcode_translators(const XtensaOpcodeTranslators * t)5075903973SMax Filippov static GHashTable *hash_opcode_translators(const XtensaOpcodeTranslators *t)
5175903973SMax Filippov {
5275903973SMax Filippov     unsigned i, j;
5375903973SMax Filippov     GHashTable *translator = g_hash_table_new(g_str_hash, g_str_equal);
5475903973SMax Filippov 
5575903973SMax Filippov     for (i = 0; i < t->num_opcodes; ++i) {
56d863fcf7SMax Filippov         if (t->opcode[i].op_flags & XTENSA_OP_NAME_ARRAY) {
57d863fcf7SMax Filippov             const char * const *name = t->opcode[i].name;
58d863fcf7SMax Filippov 
59d863fcf7SMax Filippov             for (j = 0; name[j]; ++j) {
60d863fcf7SMax Filippov                 add_translator_to_hash(translator,
61d863fcf7SMax Filippov                                        (void *)name[j],
62d863fcf7SMax Filippov                                        (void *)(t->opcode + i));
63d863fcf7SMax Filippov             }
64d863fcf7SMax Filippov         } else {
6575903973SMax Filippov             add_translator_to_hash(translator,
6675903973SMax Filippov                                    (void *)t->opcode[i].name,
6775903973SMax Filippov                                    (void *)(t->opcode + i));
6875903973SMax Filippov         }
69d863fcf7SMax Filippov     }
7075903973SMax Filippov     return translator;
7175903973SMax Filippov }
7275903973SMax Filippov 
7375903973SMax Filippov static XtensaOpcodeOps *
xtensa_find_opcode_ops(const XtensaOpcodeTranslators * t,const char * name)7475903973SMax Filippov xtensa_find_opcode_ops(const XtensaOpcodeTranslators *t,
7575903973SMax Filippov                        const char *name)
7675903973SMax Filippov {
7775903973SMax Filippov     static GHashTable *translators;
7875903973SMax Filippov     GHashTable *translator;
7975903973SMax Filippov 
8075903973SMax Filippov     if (translators == NULL) {
8175903973SMax Filippov         translators = g_hash_table_new(g_direct_hash, g_direct_equal);
8275903973SMax Filippov     }
8375903973SMax Filippov     translator = g_hash_table_lookup(translators, t);
8475903973SMax Filippov     if (translator == NULL) {
8575903973SMax Filippov         translator = hash_opcode_translators(t);
8675903973SMax Filippov         g_hash_table_insert(translators, (void *)t, translator);
8775903973SMax Filippov     }
8875903973SMax Filippov     return g_hash_table_lookup(translator, name);
8975903973SMax Filippov }
9075903973SMax Filippov 
init_libisa(XtensaConfig * config)9133071f68SMax Filippov static void init_libisa(XtensaConfig *config)
9233071f68SMax Filippov {
9333071f68SMax Filippov     unsigned i, j;
9433071f68SMax Filippov     unsigned opcodes;
9509460970SMax Filippov     unsigned formats;
96b0b24bdcSMax Filippov     unsigned regfiles;
9733071f68SMax Filippov 
9833071f68SMax Filippov     config->isa = xtensa_isa_init(config->isa_internal, NULL, NULL);
9933071f68SMax Filippov     assert(xtensa_isa_maxlength(config->isa) <= MAX_INSN_LENGTH);
100fde557adSMax Filippov     assert(xtensa_insnbuf_size(config->isa) <= MAX_INSNBUF_LENGTH);
10133071f68SMax Filippov     opcodes = xtensa_isa_num_opcodes(config->isa);
10209460970SMax Filippov     formats = xtensa_isa_num_formats(config->isa);
103b0b24bdcSMax Filippov     regfiles = xtensa_isa_num_regfiles(config->isa);
10433071f68SMax Filippov     config->opcode_ops = g_new(XtensaOpcodeOps *, opcodes);
10533071f68SMax Filippov 
10609460970SMax Filippov     for (i = 0; i < formats; ++i) {
10709460970SMax Filippov         assert(xtensa_format_num_slots(config->isa, i) <= MAX_INSN_SLOTS);
10809460970SMax Filippov     }
10909460970SMax Filippov 
11033071f68SMax Filippov     for (i = 0; i < opcodes; ++i) {
11133071f68SMax Filippov         const char *opc_name = xtensa_opcode_name(config->isa, i);
11233071f68SMax Filippov         XtensaOpcodeOps *ops = NULL;
11333071f68SMax Filippov 
11433071f68SMax Filippov         assert(xtensa_opcode_num_operands(config->isa, i) <= MAX_OPCODE_ARGS);
11533071f68SMax Filippov         if (!config->opcode_translators) {
11633071f68SMax Filippov             ops = xtensa_find_opcode_ops(&xtensa_core_opcodes, opc_name);
11733071f68SMax Filippov         } else {
11833071f68SMax Filippov             for (j = 0; !ops && config->opcode_translators[j]; ++j) {
11933071f68SMax Filippov                 ops = xtensa_find_opcode_ops(config->opcode_translators[j],
12033071f68SMax Filippov                                              opc_name);
12133071f68SMax Filippov             }
12233071f68SMax Filippov         }
12333071f68SMax Filippov #ifdef DEBUG
12433071f68SMax Filippov         if (ops == NULL) {
12533071f68SMax Filippov             fprintf(stderr,
12633071f68SMax Filippov                     "opcode translator not found for %s's opcode '%s'\n",
12733071f68SMax Filippov                     config->name, opc_name);
12833071f68SMax Filippov         }
12933071f68SMax Filippov #endif
13033071f68SMax Filippov         config->opcode_ops[i] = ops;
13133071f68SMax Filippov     }
132fe7869d6SMax Filippov     config->a_regfile = xtensa_regfile_lookup(config->isa, "AR");
133b0b24bdcSMax Filippov 
134b0b24bdcSMax Filippov     config->regfile = g_new(void **, regfiles);
135b0b24bdcSMax Filippov     for (i = 0; i < regfiles; ++i) {
136b0b24bdcSMax Filippov         const char *name = xtensa_regfile_name(config->isa, i);
137ee659da2SMax Filippov         int entries = xtensa_regfile_num_entries(config->isa, i);
138ee659da2SMax Filippov         int bits = xtensa_regfile_num_bits(config->isa, i);
139b0b24bdcSMax Filippov 
140ee659da2SMax Filippov         config->regfile[i] = xtensa_get_regfile_by_name(name, entries, bits);
141b0b24bdcSMax Filippov #ifdef DEBUG
142b0b24bdcSMax Filippov         if (config->regfile[i] == NULL) {
143b0b24bdcSMax Filippov             fprintf(stderr, "regfile '%s' not found for %s\n",
144b0b24bdcSMax Filippov                     name, config->name);
145b0b24bdcSMax Filippov         }
146b0b24bdcSMax Filippov #endif
147b0b24bdcSMax Filippov     }
14859419607SMax Filippov     xtensa_collect_sr_names(config);
14933071f68SMax Filippov }
15033071f68SMax Filippov 
xtensa_finalize_config(XtensaConfig * config)1510e7c8879SMax Filippov static void xtensa_finalize_config(XtensaConfig *config)
152fcf5ef2aSThomas Huth {
15333071f68SMax Filippov     if (config->isa_internal) {
15433071f68SMax Filippov         init_libisa(config);
15533071f68SMax Filippov     }
1561b7b26e4SMax Filippov 
1571b7b26e4SMax Filippov     if (config->gdb_regmap.num_regs == 0 ||
1581b7b26e4SMax Filippov         config->gdb_regmap.num_core_regs == 0) {
1591b7b26e4SMax Filippov         unsigned n_regs = 0;
1601b7b26e4SMax Filippov         unsigned n_core_regs = 0;
161fcf5ef2aSThomas Huth 
162a7ac06fdSMax Filippov         xtensa_count_regs(config, &n_regs, &n_core_regs);
1631b7b26e4SMax Filippov         if (config->gdb_regmap.num_regs == 0) {
1641b7b26e4SMax Filippov             config->gdb_regmap.num_regs = n_regs;
1651b7b26e4SMax Filippov         }
1661b7b26e4SMax Filippov         if (config->gdb_regmap.num_core_regs == 0) {
1671b7b26e4SMax Filippov             config->gdb_regmap.num_core_regs = n_core_regs;
1681b7b26e4SMax Filippov         }
1691b7b26e4SMax Filippov     }
170fcf5ef2aSThomas Huth }
171fcf5ef2aSThomas Huth 
xtensa_core_class_init(ObjectClass * oc,void * data)1720e7c8879SMax Filippov static void xtensa_core_class_init(ObjectClass *oc, void *data)
1730e7c8879SMax Filippov {
1740e7c8879SMax Filippov     CPUClass *cc = CPU_CLASS(oc);
1750e7c8879SMax Filippov     XtensaCPUClass *xcc = XTENSA_CPU_CLASS(oc);
1760e7c8879SMax Filippov     XtensaConfig *config = data;
1770e7c8879SMax Filippov 
1780e7c8879SMax Filippov     xtensa_finalize_config(config);
1790e7c8879SMax Filippov     xcc->config = config;
1800e7c8879SMax Filippov 
1810e7c8879SMax Filippov     /*
1820e7c8879SMax Filippov      * Use num_core_regs to see only non-privileged registers in an unmodified
1830e7c8879SMax Filippov      * gdb. Use num_regs to see all registers. gdb modification is required
1840e7c8879SMax Filippov      * for that: reset bit 0 in the 'flags' field of the registers definitions
1850e7c8879SMax Filippov      * in the gdb/xtensa-config.c inside gdb source tree or inside gdb overlay.
1860e7c8879SMax Filippov      */
1870e7c8879SMax Filippov     cc->gdb_num_core_regs = config->gdb_regmap.num_regs;
1880e7c8879SMax Filippov }
1890e7c8879SMax Filippov 
xtensa_register_core(XtensaConfigList * node)190fcf5ef2aSThomas Huth void xtensa_register_core(XtensaConfigList *node)
191fcf5ef2aSThomas Huth {
192fcf5ef2aSThomas Huth     TypeInfo type = {
193fcf5ef2aSThomas Huth         .parent = TYPE_XTENSA_CPU,
194fcf5ef2aSThomas Huth         .class_init = xtensa_core_class_init,
195fcf5ef2aSThomas Huth         .class_data = (void *)node->config,
196fcf5ef2aSThomas Huth     };
197fcf5ef2aSThomas Huth 
198fcf5ef2aSThomas Huth     node->next = xtensa_cores;
199fcf5ef2aSThomas Huth     xtensa_cores = node;
200a5247d76SIgor Mammedov     type.name = g_strdup_printf(XTENSA_CPU_TYPE_NAME("%s"), node->config->name);
201fcf5ef2aSThomas Huth     type_register(&type);
202fcf5ef2aSThomas Huth     g_free((gpointer)type.name);
203fcf5ef2aSThomas Huth }
204fcf5ef2aSThomas Huth 
check_hw_breakpoints(CPUXtensaState * env)205fcf5ef2aSThomas Huth static uint32_t check_hw_breakpoints(CPUXtensaState *env)
206fcf5ef2aSThomas Huth {
207fcf5ef2aSThomas Huth     unsigned i;
208fcf5ef2aSThomas Huth 
209fcf5ef2aSThomas Huth     for (i = 0; i < env->config->ndbreak; ++i) {
210fcf5ef2aSThomas Huth         if (env->cpu_watchpoint[i] &&
211fcf5ef2aSThomas Huth                 env->cpu_watchpoint[i]->flags & BP_WATCHPOINT_HIT) {
212fcf5ef2aSThomas Huth             return DEBUGCAUSE_DB | (i << DEBUGCAUSE_DBNUM_SHIFT);
213fcf5ef2aSThomas Huth         }
214fcf5ef2aSThomas Huth     }
215fcf5ef2aSThomas Huth     return 0;
216fcf5ef2aSThomas Huth }
217fcf5ef2aSThomas Huth 
xtensa_breakpoint_handler(CPUState * cs)218fcf5ef2aSThomas Huth void xtensa_breakpoint_handler(CPUState *cs)
219fcf5ef2aSThomas Huth {
220*52049266SPhilippe Mathieu-Daudé     CPUXtensaState *env = cpu_env(cs);
221fcf5ef2aSThomas Huth 
222fcf5ef2aSThomas Huth     if (cs->watchpoint_hit) {
223fcf5ef2aSThomas Huth         if (cs->watchpoint_hit->flags & BP_CPU) {
224fcf5ef2aSThomas Huth             uint32_t cause;
225fcf5ef2aSThomas Huth 
226fcf5ef2aSThomas Huth             cs->watchpoint_hit = NULL;
227fcf5ef2aSThomas Huth             cause = check_hw_breakpoints(env);
228fcf5ef2aSThomas Huth             if (cause) {
229fcf5ef2aSThomas Huth                 debug_exception_env(env, cause);
230fcf5ef2aSThomas Huth             }
231fcf5ef2aSThomas Huth             cpu_loop_exit_noexc(cs);
232fcf5ef2aSThomas Huth         }
2335f3ebbc8SMax Filippov     } else {
2345f3ebbc8SMax Filippov         if (cpu_breakpoint_test(cs, env->pc, BP_GDB)
2355f3ebbc8SMax Filippov             || !cpu_breakpoint_test(cs, env->pc, BP_CPU)) {
2365f3ebbc8SMax Filippov             return;
2375f3ebbc8SMax Filippov         }
2385f3ebbc8SMax Filippov         if (env->sregs[ICOUNT] == 0xffffffff &&
2395f3ebbc8SMax Filippov             xtensa_get_cintlevel(env) < env->sregs[ICOUNTLEVEL]) {
2405f3ebbc8SMax Filippov             debug_exception_env(env, DEBUGCAUSE_IC);
2415f3ebbc8SMax Filippov         } else {
2425f3ebbc8SMax Filippov             debug_exception_env(env, DEBUGCAUSE_IB);
2435f3ebbc8SMax Filippov         }
2445f3ebbc8SMax Filippov         cpu_loop_exit_noexc(cs);
245fcf5ef2aSThomas Huth     }
246fcf5ef2aSThomas Huth }
247fcf5ef2aSThomas Huth 
2486407f64fSRichard Henderson #ifndef CONFIG_USER_ONLY
xtensa_cpu_do_unaligned_access(CPUState * cs,vaddr addr,MMUAccessType access_type,int mmu_idx,uintptr_t retaddr)2499584116fSMax Filippov void xtensa_cpu_do_unaligned_access(CPUState *cs,
2509584116fSMax Filippov                                     vaddr addr, MMUAccessType access_type,
2519584116fSMax Filippov                                     int mmu_idx, uintptr_t retaddr)
2529584116fSMax Filippov {
2539584116fSMax Filippov     XtensaCPU *cpu = XTENSA_CPU(cs);
2549584116fSMax Filippov     CPUXtensaState *env = &cpu->env;
2559584116fSMax Filippov 
256583e6a5fSMax Filippov     assert(xtensa_option_enabled(env->config,
257583e6a5fSMax Filippov                                  XTENSA_OPTION_UNALIGNED_EXCEPTION));
2583d419a4dSRichard Henderson     cpu_restore_state(CPU(cpu), retaddr);
2599584116fSMax Filippov     HELPER(exception_cause_vaddr)(env,
2609584116fSMax Filippov                                   env->pc, LOAD_STORE_ALIGNMENT_CAUSE,
2619584116fSMax Filippov                                   addr);
2629584116fSMax Filippov }
2639584116fSMax Filippov 
xtensa_cpu_tlb_fill(CPUState * cs,vaddr address,int size,MMUAccessType access_type,int mmu_idx,bool probe,uintptr_t retaddr)264b008c456SRichard Henderson bool xtensa_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
265b008c456SRichard Henderson                          MMUAccessType access_type, int mmu_idx,
266b008c456SRichard Henderson                          bool probe, uintptr_t retaddr)
2679584116fSMax Filippov {
268*52049266SPhilippe Mathieu-Daudé     CPUXtensaState *env = cpu_env(cs);
2699584116fSMax Filippov     uint32_t paddr;
2709584116fSMax Filippov     uint32_t page_size;
2719584116fSMax Filippov     unsigned access;
272b008c456SRichard Henderson     int ret = xtensa_get_physical_addr(env, true, address, access_type,
273b008c456SRichard Henderson                                        mmu_idx, &paddr, &page_size, &access);
2749584116fSMax Filippov 
275b008c456SRichard Henderson     qemu_log_mask(CPU_LOG_MMU, "%s(%08" VADDR_PRIx
276b008c456SRichard Henderson                   ", %d, %d) -> %08x, ret = %d\n",
277b008c456SRichard Henderson                   __func__, address, access_type, mmu_idx, paddr, ret);
2789584116fSMax Filippov 
2799584116fSMax Filippov     if (ret == 0) {
2809584116fSMax Filippov         tlb_set_page(cs,
281b008c456SRichard Henderson                      address & TARGET_PAGE_MASK,
2829584116fSMax Filippov                      paddr & TARGET_PAGE_MASK,
2839584116fSMax Filippov                      access, mmu_idx, page_size);
284b008c456SRichard Henderson         return true;
285b008c456SRichard Henderson     } else if (probe) {
286b008c456SRichard Henderson         return false;
2879584116fSMax Filippov     } else {
2883d419a4dSRichard Henderson         cpu_restore_state(cs, retaddr);
289b008c456SRichard Henderson         HELPER(exception_cause_vaddr)(env, env->pc, ret, address);
2909584116fSMax Filippov     }
2919584116fSMax Filippov }
2929584116fSMax Filippov 
xtensa_cpu_do_transaction_failed(CPUState * cs,hwaddr physaddr,vaddr addr,unsigned size,MMUAccessType access_type,int mmu_idx,MemTxAttrs attrs,MemTxResult response,uintptr_t retaddr)2939584116fSMax Filippov void xtensa_cpu_do_transaction_failed(CPUState *cs, hwaddr physaddr, vaddr addr,
2949584116fSMax Filippov                                       unsigned size, MMUAccessType access_type,
2959584116fSMax Filippov                                       int mmu_idx, MemTxAttrs attrs,
2969584116fSMax Filippov                                       MemTxResult response, uintptr_t retaddr)
2979584116fSMax Filippov {
298*52049266SPhilippe Mathieu-Daudé     CPUXtensaState *env = cpu_env(cs);
2999584116fSMax Filippov 
3003d419a4dSRichard Henderson     cpu_restore_state(cs, retaddr);
3019584116fSMax Filippov     HELPER(exception_cause_vaddr)(env, env->pc,
3029584116fSMax Filippov                                   access_type == MMU_INST_FETCH ?
3039584116fSMax Filippov                                   INSTR_PIF_ADDR_ERROR_CAUSE :
3049584116fSMax Filippov                                   LOAD_STORE_PIF_ADDR_ERROR_CAUSE,
3059584116fSMax Filippov                                   addr);
3069584116fSMax Filippov }
3079584116fSMax Filippov 
xtensa_runstall(CPUXtensaState * env,bool runstall)308bd527a83SMax Filippov void xtensa_runstall(CPUXtensaState *env, bool runstall)
309bd527a83SMax Filippov {
31092fddfbdSRichard Henderson     CPUState *cpu = env_cpu(env);
311bd527a83SMax Filippov 
312bd527a83SMax Filippov     env->runstall = runstall;
313bd527a83SMax Filippov     cpu->halted = runstall;
314bd527a83SMax Filippov     if (runstall) {
315bd527a83SMax Filippov         cpu_interrupt(cpu, CPU_INTERRUPT_HALT);
316bd527a83SMax Filippov     } else {
3176230dac8SMax Filippov         qemu_cpu_kick(cpu);
318bd527a83SMax Filippov     }
319bd527a83SMax Filippov }
320cbc183d2SClaudio Fontana #endif /* !CONFIG_USER_ONLY */
321