/* * Copyright (c) 2000, Intel Corporation * All rights reserved. * * WARRANTY DISCLAIMER * * THESE MATERIALS ARE PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL INTEL OR ITS * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THESE * MATERIALS, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * Intel Corporation is the author of the Materials, and requests that all * problem reports or change requests be submitted to it directly at * http://developer.intel.com/opensource. */ #include #include #include #include #include #define INT64 #include "decfn_emdb.h" #include "decision_tree.h" #pragma function (memset) #include "decoder_priv.h" /***************************************************************************/ #define STATIC #ifndef TRUE #define TRUE 1 #endif #ifndef FALSE #define FALSE 0 #endif #define PRED_SIZE EM_PREDICATE_BITS #define INIT_PSEUDO_TROLES_TAB_ENTRY(Entry,Slot0,Slot1,Slot2) \ { \ pseudo_troles_tab[Entry][0] = (Slot0); \ pseudo_troles_tab[Entry][1] = (Slot1); \ pseudo_troles_tab[Entry][2] = (Slot2); \ } #define EM_DECODER_FLAGS__NO_MEMSET(Flags) ((Flags) & EM_DECODER_FLAG_NO_MEMSET) /***************************************************************************/ U4byte IEL_t1, IEL_t2, IEL_t3, IEL_t4; U32 IEL_tempc; U64 IEL_et1, IEL_et2; U128 IEL_ext1, IEL_ext2, IEL_ext3, IEL_ext4, IEL_ext5; S128 IEL_ts1, IEL_ts2; extern struct EM_version_s deccpu_emdb_version; const U32 decoder_bundle_size = IEL_CONST32(EM_BUNDLE_SIZE); STATIC Temp_role_t pseudo_troles_tab[16][3]; STATIC int troles_tab_initialized = FALSE; STATIC EM_Decoder_Err em_decoding(const EM_Decoder_Id, const unsigned char *, const int, const EM_IL, EM_Decoder_Info *); STATIC EM_Decoder_Err em_inst_decode(const EM_Decoder_Id, U64, const Temp_role_t, const U128 *, EM_Decoder_Info *); STATIC void em_decoder_init_decoder_info(EM_Decoder_Info *decoder_info); STATIC void em_decoder_init_bundle_info(EM_Decoder_Bundle_Info *bundle_info); STATIC void init_pseudo_troles_tab(void); /**************************************************************************** * init_pseudo_troles_tab * * initalizes pseudo_troles_tab. If only template-# changes, update EM.h * * is enough to update the decoder. * ****************************************************************************/ STATIC void init_pseudo_troles_tab(void) { /*** In the following table EM_TEMP_ROLE_MEM means M/A & same for _INT ***/ int i; /*** initialize all entries as reserved ***/ for (i = 0; i < EM_NUM_OF_TEMPLATES; i++) INIT_PSEUDO_TROLES_TAB_ENTRY(i, EM_TEMP_ROLE_NONE, EM_TEMP_ROLE_NONE, EM_TEMP_ROLE_NONE); /*** initialize specific entries ***/ INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mii , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_INT , EM_TEMP_ROLE_INT ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mi_i , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_INT , EM_TEMP_ROLE_INT ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mlx , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_LONG, EM_TEMP_ROLE_LONG); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mmi , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_MEM , EM_TEMP_ROLE_INT ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_m_mi , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_MEM , EM_TEMP_ROLE_INT ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mfi , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_FP , EM_TEMP_ROLE_INT ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mmf , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_MEM , EM_TEMP_ROLE_FP ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mib , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_INT , EM_TEMP_ROLE_BR ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mbb , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_BR , EM_TEMP_ROLE_BR ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_bbb , EM_TEMP_ROLE_BR , \ EM_TEMP_ROLE_BR , EM_TEMP_ROLE_BR ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mmb , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_MEM , EM_TEMP_ROLE_BR ); INIT_PSEUDO_TROLES_TAB_ENTRY(EM_template_mfb , EM_TEMP_ROLE_MEM , \ EM_TEMP_ROLE_FP , EM_TEMP_ROLE_BR ); /*** avoid multpiple initializations ***/ troles_tab_initialized = TRUE; }; #ifdef BIG_ENDIAN #define ENTITY_SWAP(E) entity_swap((unsigned char *)(&(E)), sizeof(E)) /***************************** entity_swap *******************************/ /* swap any number of bytes */ /*************************************************************************/ STATIC void entity_swap(unsigned char *entity_1st, unsigned int size) { unsigned char tmp8, *p, *q; for (q = (p = entity_1st) + (size-1); p < q; p++, q--) { tmp8 = *q; *q = *p; *p = tmp8; } } #else #define ENTITY_SWAP(E) {} #endif /* STATIC dec_2_emdb_trole[] = { EM_TEMP_ROLE_INT, EM_TEMP_ROLE_MEM, EM_TEMP_ROLE_FP, EM_TEMP_ROLE_BR, EM_TEMP_ROLE_LONG }; */ /********************************************************************************/ /* em_decoder_open: opens a new entry in the em_clients_table and returns the */ /* index of the entry. */ /********************************************************************************/ EM_Decoder_Id em_decoder_open(void) { int i; Client_Entry initiate_entry={1, DEFAULT_MACHINE_TYPE, DEFAULT_MACHINE_MODE, NULL }; for (i=0 ; i < EM_DECODER_MAX_CLIENTS ; i++) { if ( !(em_clients_table[i].is_used) ) { em_clients_table[i] = initiate_entry; if (!troles_tab_initialized) init_pseudo_troles_tab(); return(i); } } return(-1); } /*****************************************************************************/ /* legal_id: check whether a given id suits an active entry in the */ /* clients table. */ /*****************************************************************************/ STATIC int legal_id(int id) { if ((id<0)||(id>=EM_DECODER_MAX_CLIENTS)) { return(FALSE); } if (!em_clients_table[id].is_used) { return(FALSE); } return(TRUE); } /*****************************************************************************/ /* em_decoder_close: closes an entry in the clients table for later use. */ /*****************************************************************************/ EM_Decoder_Err em_decoder_close(const EM_Decoder_Id id) { if (legal_id(id)) { em_clients_table[id].is_used=0; if (em_clients_table[id].info_ptr != NULL) { free(em_clients_table[id].info_ptr); } return(EM_DECODER_NO_ERROR); } else { return(EM_DECODER_INVALID_CLIENT_ID); } } /*****************************************************************************/ /* legal_type: */ /*****************************************************************************/ STATIC int legal_type(EM_Decoder_Machine_Type type) { if (type < EM_DECODER_CPU_LAST) { return(TRUE); } return(FALSE); } /*****************************************************************************/ /* legal_mode: */ /*****************************************************************************/ STATIC int legal_mode(EM_Decoder_Machine_Type type, EM_Decoder_Machine_Mode mode) { if (mode == EM_DECODER_MODE_NO_CHANGE) { return(TRUE); } if ((mode > EM_DECODER_MODE_NO_CHANGE) && (mode < EM_DECODER_MODE_LAST)) { if ((mode == EM_DECODER_MODE_EM) && (type != EM_DECODER_CPU_P7)) { return(FALSE); } else { return(TRUE); } } return(FALSE); } /*****************************************************************************/ /* legal_inst: */ /*****************************************************************************/ STATIC int legal_inst(EM_Decoder_Inst_Id inst, EM_Decoder_Machine_Type type) { if (inst < EM_INST_LAST) { /* unsigned int cpu_flag = deccpu_EMDB_info[inst].impls; */ switch (type) { case EM_DECODER_CPU_P7: return TRUE; default: /*assert(0);*/ break; } } return(FALSE); } /****************************************************************************/ /* em_decoder_setenv: sets the machine type and machine mode variables. */ /****************************************************************************/ EM_Decoder_Err em_decoder_setenv(const EM_Decoder_Id id, const EM_Decoder_Machine_Type type, const EM_Decoder_Machine_Mode mode) { if (!legal_id(id)) { return(EM_DECODER_INVALID_CLIENT_ID); } if (!legal_type(type)) { return(EM_DECODER_INVALID_MACHINE_TYPE); } if (!legal_mode(type, mode)) { return(EM_DECODER_INVALID_MACHINE_MODE); } if (type == EM_DECODER_CPU_DEFAULT) { em_clients_table[id].machine_type = DEFAULT_MACHINE_TYPE; } else if (type != EM_DECODER_CPU_NO_CHANGE) { em_clients_table[id].machine_type = type; } if (mode == EM_DECODER_MODE_DEFAULT) { em_clients_table[id].machine_mode = DEFAULT_MACHINE_MODE; } else if (mode != EM_DECODER_MODE_NO_CHANGE) { em_clients_table[id].machine_mode = mode; } return(EM_DECODER_NO_ERROR); } /******************************************************************************/ /* em_decoder_setup: sets the machine type, machine mode variables and flags. */ /******************************************************************************/ EM_Decoder_Err em_decoder_setup(const EM_Decoder_Id id, const EM_Decoder_Machine_Type type, const EM_Decoder_Machine_Mode mode, unsigned long flags) { EM_Decoder_Err err; if ((err=em_decoder_setenv(id, type, mode)) != EM_DECODER_NO_ERROR) { return (err); } em_clients_table[id].flags = flags; return (EM_DECODER_NO_ERROR); } /********************************************************************************/ /* em_decoder_init_decoder_info: initializes decoder_info in case of no memset. */ /********************************************************************************/ STATIC void em_decoder_init_decoder_info(EM_Decoder_Info *decoder_info) { decoder_info->pred.valid = FALSE; decoder_info->src1.type = EM_DECODER_NO_OPER; decoder_info->src1.oper_flags = 0; decoder_info->src2.type = EM_DECODER_NO_OPER; decoder_info->src2.oper_flags = 0; decoder_info->src3.type = EM_DECODER_NO_OPER; decoder_info->src3.oper_flags = 0; decoder_info->src4.type = EM_DECODER_NO_OPER; decoder_info->src4.oper_flags = 0; decoder_info->src5.type = EM_DECODER_NO_OPER; decoder_info->src5.oper_flags = 0; decoder_info->dst1.type = EM_DECODER_NO_OPER; decoder_info->dst1.oper_flags = 0; decoder_info->dst2.type = EM_DECODER_NO_OPER; decoder_info->dst1.oper_flags = 0; } /******************************************************************************/ /* em_decoder_init_bundle_info: initializes bundle_info in case of no memset. */ /******************************************************************************/ STATIC void em_decoder_init_bundle_info(EM_Decoder_Bundle_Info *bundle_info) { unsigned int slot; for (slot=0; slot<3; slot++) { em_decoder_init_decoder_info(bundle_info->inst_info+slot); bundle_info->inst_info[slot].EM_info.em_flags = 0; } } /*******************************************************************************/ /* em_decoder_associate_one: adds to the client's entry a pointer to an extra */ /* information about a single instruction (inst). */ /*******************************************************************************/ EM_Decoder_Err em_decoder_associate_one(const EM_Decoder_Id id, const EM_Decoder_Inst_Id inst, const void * client_info) { int i; int n_insts; if (!legal_id(id)) { return(EM_DECODER_INVALID_CLIENT_ID); } if (!legal_inst(inst, em_clients_table[id].machine_type)) { return(EM_DECODER_INVALID_INST_ID); } { n_insts = EM_INST_LAST; /*** assume MAX. repair ***/ } if (em_clients_table[id].info_ptr == NULL) { em_clients_table[id].info_ptr = calloc((size_t)n_insts, sizeof(void *)); if (!em_clients_table[id].info_ptr) { return EM_DECODER_INTERNAL_ERROR; } for (i=0 ; i < n_insts ; i++) { em_clients_table[id].info_ptr[i] = NULL; } } em_clients_table[id].info_ptr[inst] = (void *)client_info; return(EM_DECODER_NO_ERROR); } /***************************************************************************** * em_decoder_associate_check - check the client's array of association * * valid for P7 cpu only * *****************************************************************************/ EM_Decoder_Err em_decoder_associate_check(const EM_Decoder_Id id, EM_Decoder_Inst_Id * inst) { EM_Decoder_Inst_Id i; if(!legal_id(id)) { return(EM_DECODER_INVALID_CLIENT_ID); } if (em_clients_table[id].machine_type == EM_DECODER_CPU_P7) { if (em_clients_table[id].machine_mode == EM_DECODER_MODE_EM) { if(em_clients_table[id].info_ptr == NULL) { *inst = 1; return(EM_DECODER_ASSOCIATE_MISS); } for (i = 1; (i < EM_INST_LAST) && (em_clients_table[id].info_ptr[i] != NULL); i++); if (i < EM_INST_LAST) { *inst = i; return(EM_DECODER_ASSOCIATE_MISS); } } else /*** iA ***/ { } *inst = EM_DECODER_INST_NONE; return(EM_DECODER_NO_ERROR); } else /* cpu is p5, p6 */ { *inst = EM_DECODER_INST_NONE; return(EM_DECODER_NO_ERROR); } } /****************************************************************************** * em_decoder_decode * * * * params: * * id - decoder client id * * code - pointer to instruction buffer * * max_code_size - instruction buffer size * * decoder_info - pointer to decoder_info to fill * * * * returns: * * EM_Decoder_Err * * * *****************************************************************************/ EM_Decoder_Err em_decoder_decode(const EM_Decoder_Id id, const unsigned char * code, const int max_code_size, const EM_IL location, EM_Decoder_Info * decoder_info) { EM_Decoder_Err err = EM_DECODER_NO_ERROR; if (!legal_id(id)) { return(EM_DECODER_INVALID_CLIENT_ID); } if (decoder_info == NULL) { return(EM_DECODER_NULL_PTR); } if (code == NULL) { return(EM_DECODER_TOO_SHORT_ERR); } if (EM_DECODER_FLAGS__NO_MEMSET(em_clients_table[id].flags)) { em_decoder_init_decoder_info(decoder_info); } else { memset(decoder_info, 0, sizeof(EM_Decoder_Info)); } if (em_clients_table[id].machine_mode == EM_DECODER_MODE_EM) { err = em_decoding(id, code, max_code_size, location, decoder_info); } else { err = EM_DECODER_INVALID_MACHINE_MODE; } return(err); } /*****************************************************************************/ /* em_decoder_inst_static_info: return instruction static info (flags, */ /* client_info pointer and static_info pointer) */ /*****************************************************************************/ EM_Decoder_Err em_decoder_inst_static_info(const EM_Decoder_Id id, const EM_Decoder_Inst_Id inst_id, EM_Decoder_Inst_Static_Info * static_info) { if (!legal_id(id)) { return(EM_DECODER_INVALID_CLIENT_ID); } if (!legal_inst(inst_id, em_clients_table[id].machine_type)) { return(EM_DECODER_INVALID_INST_ID); } if (static_info == NULL) { return(EM_DECODER_NULL_PTR); } if (em_clients_table[id].info_ptr != NULL) { static_info->client_info = em_clients_table[id].info_ptr[inst_id]; } else { static_info->client_info = NULL; } static_info->static_info = em_decoder_static_info + inst_id; return(EM_DECODER_NO_ERROR); } /****************************************************************************** * em_decoder_decode_bundle - decode em bundle * * * * params: * * id - decoder client id * * code - pointer to instruction buffer * * max_code_size - instruction buffer size(Should be at least 3*128 * * bundle_info - pointer to bundle_info to fill * * * * returns: * * EM_Decoder_Err * * * *****************************************************************************/ EM_Decoder_Err em_decoder_decode_bundle(const EM_Decoder_Id id, const unsigned char* code, const int max_size, EM_Decoder_Bundle_Info* bundle_info) { unsigned int slot_no; U128 bundle; U64 instr; EM_template_t templt; Temp_role_t temp_role; EM_Decoder_Info *decoder_info; int bundle_stop; EM_Decoder_Err err, return_err = EM_DECODER_NO_ERROR; if (!legal_id(id)) { return(EM_DECODER_INVALID_CLIENT_ID); } if (bundle_info == NULL) { return(EM_DECODER_NULL_PTR); } if (EM_DECODER_FLAGS__NO_MEMSET(em_clients_table[id].flags)) { em_decoder_init_bundle_info(bundle_info); } else { memset(bundle_info, 0, sizeof(EM_Decoder_Bundle_Info)); } bundle = *(const U128 *)code; ENTITY_SWAP(bundle); templt = EM_GET_TEMPLATE(bundle); if (max_size < EM_BUNDLE_SIZE) { return(EM_DECODER_TOO_SHORT_ERR); } bundle_info->em_bundle_info.flags = 0; if (bundle_stop = (IEL_GETDW0(bundle) & (1<em_bundle_info.flags |= EM_DECODER_BIT_BUNDLE_STOP; if (EM_TEMPLATE_IS_RESERVED(templt)) return(EM_DECODER_INVALID_TEMPLATE); bundle_info->em_bundle_info.b_template = templt; /*** Decode 3 instruction (unless long 2-slot instruction) ***/ for(slot_no = 0; slot_no < 3; slot_no++) { decoder_info = &(bundle_info->inst_info[slot_no]); decoder_info->EM_info.slot_no = slot_no; temp_role = pseudo_troles_tab[templt][slot_no]; /*** DECODER_NEXT should work even if error occurs ***/ decoder_info->size = 1 + (temp_role == EM_TEMP_ROLE_LONG); if (temp_role == EM_TEMP_ROLE_LONG) /*** 2-slot instruction ***/ { if (slot_no >= EM_SLOT_LAST - 1) { err = EM_DECODER_INVALID_INST_SLOT; } else { EM_GET_SYLLABLE(instr, bundle, slot_no+1); /* opcode is in slot 3 */ err = em_inst_decode(id, instr, temp_role, &bundle, decoder_info); } if (err == EM_DECODER_INVALID_PRM_OPCODE) { /* try to find nop.i or break.i in this slot */ temp_role = EM_TEMP_ROLE_INT; err = em_inst_decode(id, instr, temp_role, &bundle, decoder_info); if (!err) { if (!EM_DECODER_CHECK_OK_IN_MLX(decoder_info)) err = EM_DECODER_INVALID_PRM_OPCODE; } } } else { EM_GET_SYLLABLE(instr, bundle, slot_no); err = em_inst_decode(id, instr, temp_role, &bundle, decoder_info); if (!err) { if ((EM_DECODER_CHECK_SLOT2_ONLY(decoder_info)) && (slot_no != 2)) /*** intruction must be in slot 2 only, but... ***/ err = EM_DECODER_INVALID_SLOT_BRANCH_INST; else if (EM_DECODER_CHECK_GROUP_LAST(decoder_info) && (((slot_no != 2) || !bundle_stop) && ((slot_no != 0) || (templt != EM_template_m_mi)))) /*** instruction fails to be the last in instruction group ***/ err = EM_DECODER_MUST_BE_GROUP_LAST; } } /* return_err = the first worst error */ if (err) { FILL_PREDICATE_INFO(instr, decoder_info); if ((!return_err) || (EM_DECODER_ERROR_IS_INST_FATAL(err) && !EM_DECODER_ERROR_IS_INST_FATAL(return_err))) return_err = err; } bundle_info->error[slot_no] = err; decoder_info->EM_info.em_bundle_info = bundle_info->em_bundle_info; if (decoder_info->size == 2) /*** 2-slot instruction - exit for loop ***/ { slot_no++; break; } } /* if ((!err) && EM_DECODER_CHECK_GROUP_LAST(decoder_info) && (!EM_DECODER_BUNDLE_STOP(decoder_info))) { *** instruction fails to be the last in instruction group *** bundle_info->error[slot_no-1] = EM_DECODER_MUST_BE_GROUP_LAST; if (!return_err) return_err = EM_DECODER_MUST_BE_GROUP_LAST; }*/ bundle_info->inst_num = slot_no; bundle_info->inst_info[slot_no-1].EM_info.em_flags |= EM_DECODER_BIT_LAST_INST; if (bundle_stop) bundle_info->inst_info[slot_no-1].EM_info.em_flags |= EM_DECODER_BIT_CYCLE_BREAK; if (templt == EM_template_m_mi) bundle_info->inst_info[0].EM_info.em_flags |= EM_DECODER_BIT_CYCLE_BREAK; if (templt == EM_template_mi_i) bundle_info->inst_info[1].EM_info.em_flags |= EM_DECODER_BIT_CYCLE_BREAK; return(return_err); } /****************************************************************************** * em_decoding - decode em (2.0- till ??) single instruction + bundle info * * * * params: * * id - decoder client id * * code - pointer to instruction buffer * * max_code_size - instruction buffer size * * location - syllable location, used to get slot # * * decoder_info - pointer to decoder_info to fill * * * * returns: * * EM_Decoder_Err * * * *****************************************************************************/ STATIC EM_Decoder_Err em_decoding ( const EM_Decoder_Id id, const unsigned char * code, const int max_code_size, const EM_IL location, EM_Decoder_Info * decoder_info) { unsigned int slot_no = EM_IL_GET_SLOT_NO(location); U128 bundle; U64 instr; EM_template_t templt; Temp_role_t temp_role; int bundle_stop, cycle_break; EM_Decoder_Err err; IEL_ZERO(instr); bundle = *(const U128 *)code; ENTITY_SWAP(bundle); templt = EM_GET_TEMPLATE(bundle); /*** DECODER_NEXT should work even if error occurs ***/ decoder_info->size = 1; if (max_code_size < EM_BUNDLE_SIZE) { return(EM_DECODER_TOO_SHORT_ERR); } /******************************************************************/ /** fill EM_Info and check it ***/ /******************************************************************/ decoder_info->EM_info.em_flags = decoder_info->EM_info.em_bundle_info.flags = 0; if (bundle_stop = (IEL_GETDW0(bundle) & (1<EM_info.em_bundle_info.flags |= EM_DECODER_BIT_BUNDLE_STOP; if (EM_TEMPLATE_IS_RESERVED(templt)) return(EM_DECODER_INVALID_TEMPLATE); decoder_info->EM_info.em_bundle_info.b_template = templt; if (slot_no > EM_SLOT_2) { return(EM_DECODER_INVALID_INST_SLOT); } decoder_info->EM_info.slot_no = slot_no; /***********************/ /*** decode syllable ***/ /***********************/ /*** get instruction binary. DON'T mask bits 41 and on ***/ temp_role = pseudo_troles_tab[templt][slot_no]; if (temp_role == EM_TEMP_ROLE_LONG) /*** 2-slot instruction ***/ { if (slot_no >= EM_SLOT_LAST - 1) { err = EM_DECODER_INVALID_INST_SLOT; } else { decoder_info->size = 2; EM_GET_SYLLABLE(instr, bundle, slot_no+1); /* opcode is in slot 3 */ err = em_inst_decode(id, instr, temp_role, &bundle, decoder_info); } if (err == EM_DECODER_INVALID_PRM_OPCODE) { /* try to find nop.i or break.i in this slot */ temp_role = EM_TEMP_ROLE_INT; slot_no = EM_DECODER_SLOT_2; err = em_inst_decode(id, instr, temp_role, &bundle, decoder_info); if (!err) { if (!EM_DECODER_CHECK_OK_IN_MLX(decoder_info)) err = EM_DECODER_INVALID_PRM_OPCODE; } } } else { EM_GET_SYLLABLE(instr, bundle, slot_no); err = em_inst_decode(id, instr, temp_role, &bundle, decoder_info); if (!err) { if ((EM_DECODER_CHECK_SLOT2_ONLY(decoder_info)) && (slot_no != 2)) { /*** intruction must be in slot 2 only, but... ***/ err = EM_DECODER_INVALID_SLOT_BRANCH_INST; } else if (EM_DECODER_CHECK_GROUP_LAST(decoder_info) && (((slot_no != 2) || !bundle_stop) && ((slot_no != 0) || (templt != EM_template_m_mi)))) { /* instruction fails to be the last in instruction group */ err = EM_DECODER_MUST_BE_GROUP_LAST; } } } if (err) FILL_PREDICATE_INFO(instr, decoder_info); if ((slot_no == EM_DECODER_SLOT_2)||(decoder_info->size == 2) /* 2-slot instruction */) { cycle_break = (bundle_stop != 0); decoder_info->EM_info.em_flags |= EM_DECODER_BIT_LAST_INST; } else { cycle_break = ((slot_no==0) && (templt == EM_template_m_mi)) || ((slot_no==1) && (templt == EM_template_mi_i)); } decoder_info->EM_info.em_flags |= (cycle_break*EM_DECODER_BIT_CYCLE_BREAK); return(err); } /****************************************************************************** * em_inst_decode - decode em (2.0- till ??) single syllable * * * * params: * * id - decoder client id * * instr - 64 bit, 0-40 are the syllable binary, 41-63 irrelevant * * pseudo_trole - M/A, I/A, FP, or BR * * bundle - original pointer to bundle, for 64-bit imm extraction * * decoder_info - pointer to decoder_info to fill * * * * returns: * * EM_Decoder_Err * * * *****************************************************************************/ STATIC EM_Decoder_Err em_inst_decode( const EM_Decoder_Id id, U64 instr, const Temp_role_t pseudo_trole, const U128 * bundle_p, EM_Decoder_Info * decoder_info) { Inst_id_t inst_id; EM_Decoder_Err err; U64 tmp64; unsigned int major_opc, inst_center; Node_t node; int part_place, part_size, part_value, index; deccpu_EMDB_info_t * emdb_entry_p; EM_Decoder_static_info_t *static_entry_p; Template_role_t trole; /*** find major opcode ***/ major_opc = IEL_GETDW1(instr); /*** assumes pos > 31 ***/ major_opc >>= (EM_MAJOR_OPCODE_POS - 32); major_opc &= ((1< 0-34 ***/ inst_center = IEL_GETDW0(tmp64); /*** original bits 6-37 ***/ /*** walk through decoder decision tree ***/ node = em_decision_tree[SQUARE(major_opc, pseudo_trole)]; while (!NODE_IS_LEAF(node)) { part_place = GET_NODE_POS(node) - PRED_SIZE; part_size = GET_NODE_SIZE(node); if (part_place < 0) /*** extensions in bits 0-5 ***/ { part_place += PRED_SIZE; part_value = (IEL_GETDW0(instr) >> part_place) & ((1<> part_place) & ((1<flags = 0; /*define machine behaviour within illegal opcode */ { Behaviour_ill_opcode machine_behaviour; /*Template_role_t em_trole = dec_2_emdb_trole[pseudo_trole];*/ PRED_BEHAVIOUR(pseudo_trole, major_opc, machine_behaviour); if (machine_behaviour == BEHAVIOUR_UNDEF) { /* branch region with opcode 0 */ /* check bit 32*/ int decision_bit; GET_BRANCH_BEHAVIOUR_BIT(instr, decision_bit); if (decision_bit) { machine_behaviour = BEHAVIOUR_FAULT; } else { machine_behaviour = BEHAVIOUR_IGNORE_ON_FALSE_QP; } } if (machine_behaviour == BEHAVIOUR_FAULT) { EM_DECODER_SET_UNC_ILLEGAL_FAULT(decoder_info); } } if ((inst_id >= EM_INST_LAST) || inst_id == EM_ILLOP) { if (pseudo_trole == EM_TEMP_ROLE_BR) { /*** search for ignored fields ***/ switch (major_opc) { case 0: { unsigned int x6_ext; U64 ext; /*** get extention in bits 27:32 ***/ IEL_SHR(ext, instr, 27); x6_ext = IEL_GETDW0(ext) & ((1<<6) - 1); if (x6_ext == 1) { /*** nop.b has to be returned ***/ inst_id = EM_NOP_B_IMM21; } else return(EM_DECODER_INVALID_PRM_OPCODE); break; } case 2: { unsigned int x6_ext; U64 ext; /*** get extention in bits 27:32 ***/ IEL_SHR(ext, instr, 27); x6_ext = IEL_GETDW0(ext) & ((1<<6) - 1); switch (x6_ext) { case 0: case 16: case 17: return (EM_DECODER_INVALID_PRM_OPCODE); default: /*** nop.b has to be returned ***/ inst_id = EM_NOP_B_IMM21; } break; } default: return(EM_DECODER_INVALID_PRM_OPCODE); } /*** zero the inst encoding: pred and operands extracted below will be 0 ***/ IEL_ZERO(instr); } else return(EM_DECODER_INVALID_PRM_OPCODE); } if (! legal_inst(inst_id, em_clients_table[id].machine_type)) { /*** inst does not belong to the specified machine IS ***/ return(EM_DECODER_INVALID_PRM_OPCODE); } decoder_info->inst = inst_id; emdb_entry_p = deccpu_EMDB_info + inst_id; static_entry_p = (EM_Decoder_static_info_t *)em_decoder_static_info + inst_id; /*** get instruction static info ***/ decoder_info->flags |= static_entry_p->flags; decoder_info->EM_info.eut = trole = static_entry_p->template_role; if (EM_DECODER_CHECK_TWO_SLOT(decoder_info)) { /*** IMPORTANT: emdb flags already set from the static info !!! ***/ decoder_info->EM_info.em_flags |= EM_DECODER_BIT_LONG_INST; /*** decoder_info->size = 2; *** should be already done ***/ } /*** else *** should be already done *** { decoder_info->size = 1; } ***/ /*** handle client and static info ***/ if (em_clients_table[id].info_ptr != NULL) { decoder_info->client_info = em_clients_table[id].info_ptr[inst_id]; } else { decoder_info->client_info = NULL; } decoder_info->static_info = static_entry_p; /*** Decode predicate register ***/ if (static_entry_p->flags & EM_FLAG_PRED) { FILL_PREDICATE_INFO(instr, decoder_info) } /*** decode operands NYI ***/ err = emdb_entry_p->format_function(emdb_entry_p, instr, bundle_p, decoder_info); return(err); } /************************ misc. API functions ********************************/ const char* em_decoder_ver_str() { return(em_ver_string); } const char* em_decoder_err_msg(EM_Decoder_Err error) { if (error>=EM_DECODER_LAST_ERROR) { error = EM_DECODER_INTERNAL_ERROR; } return(em_err_msg[error]); } void em_decoder_get_version(EM_library_version_t *dec_version) { if (dec_version != NULL) { dec_version->xversion.major = XVER_MAJOR; dec_version->xversion.minor = XVER_MINOR; dec_version->api.major = API_MAJOR; dec_version->api.minor = API_MINOR; dec_version->emdb.major = deccpu_emdb_version.major; dec_version->emdb.minor = deccpu_emdb_version.minor; strcpy(dec_version->date, __DATE__); strcpy(dec_version->time, __TIME__); } } /****************************************************************************/