/* * Copyright (C) 2017, Emilio G. Cota * * License: GNU GPL, version 2 or later. * See the COPYING file in the top-level directory. */ #ifndef EXEC_TB_LOOKUP_H #define EXEC_TB_LOOKUP_H #ifdef NEED_CPU_H #include "cpu.h" #else #include "exec/poison.h" #endif #include "exec/exec-all.h" #include "exec/tb-hash.h" /* Might cause an exception, so have a longjmp destination ready */ static inline TranslationBlock * tb_lookup__cpu_state(CPUState *cpu, target_ulong *pc, target_ulong *cs_base, uint32_t *flags, uint32_t cf_mask) { CPUArchState *env = (CPUArchState *)cpu->env_ptr; TranslationBlock *tb; uint32_t hash; cpu_get_tb_cpu_state(env, pc, cs_base, flags); hash = tb_jmp_cache_hash_func(env->uc, *pc); tb = cpu->tb_jmp_cache[hash]; cf_mask &= ~CF_CLUSTER_MASK; cf_mask |= ((uint32_t)cpu->cluster_index) << CF_CLUSTER_SHIFT; if (likely(tb && tb->pc == *pc && tb->cs_base == *cs_base && tb->flags == *flags && tb->trace_vcpu_dstate == *cpu->trace_dstate && (tb_cflags(tb) & (CF_HASH_MASK | CF_INVALID)) == cf_mask)) { return tb; } tb = tb_htable_lookup(cpu, *pc, *cs_base, *flags, cf_mask); if (tb == NULL) { return NULL; } cpu->tb_jmp_cache[hash] = tb; return tb; } #endif /* EXEC_TB_LOOKUP_H */