| /* Simulation code for the CR16 processor. |
| Copyright (C) 2008-2016 Free Software Foundation, Inc. |
| Contributed by M Ranga Swami Reddy <MR.Swami.Reddy@nsc.com> |
| |
| This file is part of GDB, the GNU debugger. |
| |
| This program is free software; you can redistribute it and/or modify |
| it under the terms of the GNU General Public License as published by |
| the Free Software Foundation; either version 3, or (at your option) |
| any later version. |
| |
| This program 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 General Public License for more details. |
| |
| You should have received a copy of the GNU General Public License |
| along with this program. If not, see <http://www.gnu.org/licenses/>. */ |
| |
| #include "config.h" |
| #include <inttypes.h> |
| #include <signal.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include "bfd.h" |
| #include "gdb/callback.h" |
| #include "gdb/remote-sim.h" |
| |
| #include "sim-main.h" |
| #include "sim-options.h" |
| |
| #include "gdb/sim-cr16.h" |
| #include "gdb/signals.h" |
| #include "opcode/cr16.h" |
| |
| int cr16_debug; |
| |
| uint32 OP[4]; |
| uint32 sign_flag; |
| |
| static struct hash_entry *lookup_hash (SIM_DESC, SIM_CPU *, uint64 ins, int size); |
| static void get_operands (operand_desc *s, uint64 mcode, int isize, int nops); |
| |
| #define MAX_HASH 16 |
| |
| struct hash_entry |
| { |
| struct hash_entry *next; |
| uint32 opcode; |
| uint32 mask; |
| int format; |
| int size; |
| struct simops *ops; |
| }; |
| |
| struct hash_entry hash_table[MAX_HASH+1]; |
| |
| INLINE static long |
| hash(unsigned long long insn, int format) |
| { |
| unsigned int i = 4, tmp; |
| if (format) |
| { |
| while ((insn >> i) != 0) i +=4; |
| |
| return ((insn >> (i-4)) & 0xf); /* Use last 4 bits as hask key. */ |
| } |
| return ((insn & 0xF)); /* Use last 4 bits as hask key. */ |
| } |
| |
| |
| INLINE static struct hash_entry * |
| lookup_hash (SIM_DESC sd, SIM_CPU *cpu, uint64 ins, int size) |
| { |
| uint32 mask; |
| struct hash_entry *h; |
| |
| h = &hash_table[hash(ins,1)]; |
| |
| |
| mask = (((1 << (32 - h->mask)) -1) << h->mask); |
| |
| /* Adjuest mask for branch with 2 word instructions. */ |
| if ((h->ops->mnimonic != NULL) && |
| ((streq(h->ops->mnimonic,"b") && h->size == 2))) |
| mask = 0xff0f0000; |
| |
| |
| while ((ins & mask) != (BIN(h->opcode, h->mask))) |
| { |
| if (h->next == NULL) |
| sim_engine_halt (sd, cpu, NULL, PC, sim_stopped, SIM_SIGILL); |
| h = h->next; |
| |
| mask = (((1 << (32 - h->mask)) -1) << h->mask); |
| /* Adjuest mask for branch with 2 word instructions. */ |
| if ((streq(h->ops->mnimonic,"b")) && h->size == 2) |
| mask = 0xff0f0000; |
| |
| } |
| return (h); |
| } |
| |
| INLINE static void |
| get_operands (operand_desc *s, uint64 ins, int isize, int nops) |
| { |
| uint32 i, opn = 0, start_bit = 0, op_type = 0; |
| int32 op_size = 0, mask = 0; |
| |
| if (isize == 1) /* Trunkcate the extra 16 bits of INS. */ |
| ins = ins >> 16; |
| |
| for (i=0; i < 4; ++i,++opn) |
| { |
| if (s[opn].op_type == dummy) break; |
| |
| op_type = s[opn].op_type; |
| start_bit = s[opn].shift; |
| op_size = cr16_optab[op_type].bit_size; |
| |
| switch (op_type) |
| { |
| case imm3: case imm4: case imm5: case imm6: |
| { |
| if (isize == 1) |
| OP[i] = ((ins >> 4) & ((1 << op_size) -1)); |
| else |
| OP[i] = ((ins >> (32 - start_bit)) & ((1 << op_size) -1)); |
| |
| if (OP[i] & ((long)1 << (op_size -1))) |
| { |
| sign_flag = 1; |
| OP[i] = ~(OP[i]) + 1; |
| } |
| OP[i] = (unsigned long int)(OP[i] & (((long)1 << op_size) -1)); |
| } |
| break; |
| |
| case uimm3: case uimm3_1: case uimm4_1: |
| switch (isize) |
| { |
| case 1: |
| OP[i] = ((ins >> 4) & ((1 << op_size) -1)); break; |
| case 2: |
| OP[i] = ((ins >> (32 - start_bit)) & ((1 << op_size) -1));break; |
| default: /* for case 3. */ |
| OP[i] = ((ins >> (16 + start_bit)) & ((1 << op_size) -1)); break; |
| break; |
| } |
| break; |
| |
| case uimm4: |
| switch (isize) |
| { |
| case 1: |
| if (start_bit == 20) |
| OP[i] = ((ins >> 4) & ((1 << op_size) -1)); |
| else |
| OP[i] = (ins & ((1 << op_size) -1)); |
| break; |
| case 2: |
| OP[i] = ((ins >> start_bit) & ((1 << op_size) -1)); |
| break; |
| case 3: |
| OP[i] = ((ins >> (start_bit + 16)) & ((1 << op_size) -1)); |
| break; |
| default: |
| OP[i] = ((ins >> start_bit) & ((1 << op_size) -1)); |
| break; |
| } |
| break; |
| |
| case imm16: case uimm16: |
| OP[i] = ins & 0xFFFF; |
| break; |
| |
| case uimm20: case imm20: |
| OP[i] = ins & (((long)1 << op_size) - 1); |
| break; |
| |
| case imm32: case uimm32: |
| OP[i] = ins & 0xFFFFFFFF; |
| break; |
| |
| case uimm5: break; /*NOT USED. */ |
| OP[i] = ins & ((1 << op_size) - 1); break; |
| |
| case disps5: |
| OP[i] = (ins >> 4) & ((1 << 4) - 1); |
| OP[i] = (OP[i] * 2) + 2; |
| if (OP[i] & ((long)1 << 5)) |
| { |
| sign_flag = 1; |
| OP[i] = ~(OP[i]) + 1; |
| OP[i] = (unsigned long int)(OP[i] & 0x1F); |
| } |
| break; |
| |
| case dispe9: |
| OP[i] = ((((ins >> 8) & 0xf) << 4) | (ins & 0xf)); |
| OP[i] <<= 1; |
| if (OP[i] & ((long)1 << 8)) |
| { |
| sign_flag = 1; |
| OP[i] = ~(OP[i]) + 1; |
| OP[i] = (unsigned long int)(OP[i] & 0xFF); |
| } |
| break; |
| |
| case disps17: |
| OP[i] = (ins & 0xFFFF); |
| if (OP[i] & 1) |
| { |
| OP[i] = (OP[i] & 0xFFFE); |
| sign_flag = 1; |
| OP[i] = ~(OP[i]) + 1; |
| OP[i] = (unsigned long int)(OP[i] & 0xFFFF); |
| } |
| break; |
| |
| case disps25: |
| if (isize == 2) |
| OP[i] = (ins & 0xFFFFFF); |
| else |
| OP[i] = (ins & 0xFFFF) | (((ins >> 24) & 0xf) << 16) | |
| (((ins >> 16) & 0xf) << 20); |
| |
| if (OP[i] & 1) |
| { |
| OP[i] = (OP[i] & 0xFFFFFE); |
| sign_flag = 1; |
| OP[i] = ~(OP[i]) + 1; |
| OP[i] = (unsigned long int)(OP[i] & 0xFFFFFF); |
| } |
| break; |
| |
| case abs20: |
| if (isize == 3) |
| OP[i] = (ins) & 0xFFFFF; |
| else |
| OP[i] = (ins >> start_bit) & 0xFFFFF; |
| break; |
| case abs24: |
| if (isize == 3) |
| OP[i] = ((ins & 0xFFFF) | (((ins >> 16) & 0xf) << 20) |
| | (((ins >> 24) & 0xf) << 16)); |
| else |
| OP[i] = (ins >> 16) & 0xFFFFFF; |
| break; |
| |
| case rra: |
| case rbase: break; /* NOT USED. */ |
| case rbase_disps20: case rbase_dispe20: |
| case rpbase_disps20: case rpindex_disps20: |
| OP[i] = ((((ins >> 24)&0xf) << 16)|((ins) & 0xFFFF)); |
| OP[++i] = (ins >> 16) & 0xF; /* get 4 bit for reg. */ |
| break; |
| case rpbase_disps0: |
| OP[i] = 0; /* 4 bit disp const. */ |
| OP[++i] = (ins) & 0xF; /* get 4 bit for reg. */ |
| break; |
| case rpbase_dispe4: |
| OP[i] = ((ins >> 8) & 0xF) * 2; /* 4 bit disp const. */ |
| OP[++i] = (ins) & 0xF; /* get 4 bit for reg. */ |
| break; |
| case rpbase_disps4: |
| OP[i] = ((ins >> 8) & 0xF); /* 4 bit disp const. */ |
| OP[++i] = (ins) & 0xF; /* get 4 bit for reg. */ |
| break; |
| case rpbase_disps16: |
| OP[i] = (ins) & 0xFFFF; |
| OP[++i] = (ins >> 16) & 0xF; /* get 4 bit for reg. */ |
| break; |
| case rpindex_disps0: |
| OP[i] = 0; |
| OP[++i] = (ins >> 4) & 0xF; /* get 4 bit for reg. */ |
| OP[++i] = (ins >> 8) & 0x1; /* get 1 bit for index-reg. */ |
| break; |
| case rpindex_disps14: |
| OP[i] = (ins) & 0x3FFF; |
| OP[++i] = (ins >> 14) & 0x1; /* get 1 bit for index-reg. */ |
| OP[++i] = (ins >> 16) & 0xF; /* get 4 bit for reg. */ |
| case rindex7_abs20: |
| case rindex8_abs20: |
| OP[i] = (ins) & 0xFFFFF; |
| OP[++i] = (ins >> 24) & 0x1; /* get 1 bit for index-reg. */ |
| OP[++i] = (ins >> 20) & 0xF; /* get 4 bit for reg. */ |
| break; |
| case regr: case regp: case pregr: case pregrp: |
| switch(isize) |
| { |
| case 1: |
| if (start_bit == 20) OP[i] = (ins >> 4) & 0xF; |
| else if (start_bit == 16) OP[i] = ins & 0xF; |
| break; |
| case 2: OP[i] = (ins >> start_bit) & 0xF; break; |
| case 3: OP[i] = (ins >> (start_bit + 16)) & 0xF; break; |
| } |
| break; |
| case cc: |
| { |
| if (isize == 1) OP[i] = (ins >> 4) & 0xF; |
| else if (isize == 2) OP[i] = (ins >> start_bit) & 0xF; |
| else OP[i] = (ins >> (start_bit + 16)) & 0xF; |
| break; |
| } |
| default: break; |
| } |
| |
| /* For ESC on uimm4_1 operand. */ |
| if (op_type == uimm4_1) |
| if (OP[i] == 9) |
| OP[i] = -1; |
| |
| /* For increment by 1. */ |
| if ((op_type == pregr) || (op_type == pregrp)) |
| OP[i] += 1; |
| } |
| /* FIXME: for tracing, update values that need to be updated each |
| instruction decode cycle */ |
| State.trace.psw = PSR; |
| } |
| |
| static int |
| do_run (SIM_DESC sd, SIM_CPU *cpu, uint64 mcode) |
| { |
| struct hash_entry *h; |
| |
| #ifdef DEBUG |
| if ((cr16_debug & DEBUG_INSTRUCTION) != 0) |
| sim_io_printf (sd, "do_long 0x%x\n", mcode); |
| #endif |
| |
| h = lookup_hash (sd, cpu, mcode, 1); |
| |
| if ((h == NULL) || (h->opcode == 0)) |
| return 0; |
| |
| if (h->size == 3) |
| mcode = (mcode << 16) | RW (PC + 4); |
| |
| /* Re-set OP list. */ |
| OP[0] = OP[1] = OP[2] = OP[3] = sign_flag = 0; |
| |
| /* for push/pop/pushrtn with RA instructions. */ |
| if ((h->format & REG_LIST) && (mcode & 0x800000)) |
| OP[2] = 1; /* Set 1 for RA operand. */ |
| |
| /* numops == 0 means, no operands. */ |
| if (((h->ops) != NULL) && (((h->ops)->numops) != 0)) |
| get_operands ((h->ops)->operands, mcode, h->size, (h->ops)->numops); |
| |
| //State.ins_type = h->flags; |
| |
| (h->ops->func) (sd, cpu); |
| |
| return h->size; |
| } |
| |
| static sim_cia |
| cr16_pc_get (sim_cpu *cpu) |
| { |
| return PC; |
| } |
| |
| static void |
| cr16_pc_set (sim_cpu *cpu, sim_cia pc) |
| { |
| SIM_DESC sd = CPU_STATE (cpu); |
| SET_PC (pc); |
| } |
| |
| static void |
| free_state (SIM_DESC sd) |
| { |
| if (STATE_MODULES (sd) != NULL) |
| sim_module_uninstall (sd); |
| sim_cpu_free_all (sd); |
| sim_state_free (sd); |
| } |
| |
| static int cr16_reg_fetch (SIM_CPU *, int, unsigned char *, int); |
| static int cr16_reg_store (SIM_CPU *, int, unsigned char *, int); |
| |
| SIM_DESC |
| sim_open (SIM_OPEN_KIND kind, struct host_callback_struct *cb, |
| struct bfd *abfd, char * const *argv) |
| { |
| struct simops *s; |
| struct hash_entry *h; |
| static int init_p = 0; |
| char **p; |
| int i; |
| SIM_DESC sd = sim_state_alloc (kind, cb); |
| SIM_ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER); |
| |
| /* The cpu data is kept in a separately allocated chunk of memory. */ |
| if (sim_cpu_alloc_all (sd, 1, /*cgen_cpu_max_extra_bytes ()*/0) != SIM_RC_OK) |
| { |
| free_state (sd); |
| return 0; |
| } |
| |
| if (sim_pre_argv_init (sd, argv[0]) != SIM_RC_OK) |
| { |
| free_state (sd); |
| return 0; |
| } |
| |
| /* The parser will print an error message for us, so we silently return. */ |
| if (sim_parse_args (sd, argv) != SIM_RC_OK) |
| { |
| free_state (sd); |
| return 0; |
| } |
| |
| /* Check for/establish the a reference program image. */ |
| if (sim_analyze_program (sd, |
| (STATE_PROG_ARGV (sd) != NULL |
| ? *STATE_PROG_ARGV (sd) |
| : NULL), abfd) != SIM_RC_OK) |
| { |
| free_state (sd); |
| return 0; |
| } |
| |
| /* Configure/verify the target byte order and other runtime |
| configuration options. */ |
| if (sim_config (sd) != SIM_RC_OK) |
| { |
| sim_module_uninstall (sd); |
| return 0; |
| } |
| |
| if (sim_post_argv_init (sd) != SIM_RC_OK) |
| { |
| /* Uninstall the modules to avoid memory leaks, |
| file descriptor leaks, etc. */ |
| sim_module_uninstall (sd); |
| return 0; |
| } |
| |
| /* CPU specific initialization. */ |
| for (i = 0; i < MAX_NR_PROCESSORS; ++i) |
| { |
| SIM_CPU *cpu = STATE_CPU (sd, i); |
| |
| CPU_REG_FETCH (cpu) = cr16_reg_fetch; |
| CPU_REG_STORE (cpu) = cr16_reg_store; |
| CPU_PC_FETCH (cpu) = cr16_pc_get; |
| CPU_PC_STORE (cpu) = cr16_pc_set; |
| } |
| |
| /* The CR16 has an interrupt controller at 0xFC00, but we don't currently |
| handle that. Revisit if anyone ever implements operating mode. */ |
| /* cr16 memory: There are three separate cr16 memory regions IMEM, |
| UMEM and DMEM. The IMEM and DMEM are further broken down into |
| blocks (very like VM pages). This might not match the hardware, |
| but it matches what the toolchain currently expects. Ugh. */ |
| sim_do_commandf (sd, "memory-size %#x", 20 * 1024 * 1024); |
| |
| /* put all the opcodes in the hash table. */ |
| if (!init_p++) |
| { |
| for (s = Simops; s->func; s++) |
| { |
| switch(32 - s->mask) |
| { |
| case 0x4: |
| h = &hash_table[hash(s->opcode, 0)]; |
| break; |
| |
| case 0x7: |
| if (((s->opcode << 1) >> 4) != 0) |
| h = &hash_table[hash((s->opcode << 1) >> 4, 0)]; |
| else |
| h = &hash_table[hash((s->opcode << 1), 0)]; |
| break; |
| |
| case 0x8: |
| if ((s->opcode >> 4) != 0) |
| h = &hash_table[hash(s->opcode >> 4, 0)]; |
| else |
| h = &hash_table[hash(s->opcode, 0)]; |
| break; |
| |
| case 0x9: |
| if (((s->opcode >> 1) >> 4) != 0) |
| h = &hash_table[hash((s->opcode >>1) >> 4, 0)]; |
| else |
| h = &hash_table[hash((s->opcode >> 1), 0)]; |
| break; |
| |
| case 0xa: |
| if ((s->opcode >> 8) != 0) |
| h = &hash_table[hash(s->opcode >> 8, 0)]; |
| else if ((s->opcode >> 4) != 0) |
| h = &hash_table[hash(s->opcode >> 4, 0)]; |
| else |
| h = &hash_table[hash(s->opcode, 0)]; |
| break; |
| |
| case 0xc: |
| if ((s->opcode >> 8) != 0) |
| h = &hash_table[hash(s->opcode >> 8, 0)]; |
| else if ((s->opcode >> 4) != 0) |
| h = &hash_table[hash(s->opcode >> 4, 0)]; |
| else |
| h = &hash_table[hash(s->opcode, 0)]; |
| break; |
| |
| case 0xd: |
| if (((s->opcode >> 1) >> 8) != 0) |
| h = &hash_table[hash((s->opcode >>1) >> 8, 0)]; |
| else if (((s->opcode >> 1) >> 4) != 0) |
| h = &hash_table[hash((s->opcode >>1) >> 4, 0)]; |
| else |
| h = &hash_table[hash((s->opcode >>1), 0)]; |
| break; |
| |
| case 0x10: |
| if ((s->opcode >> 0xc) != 0) |
| h = &hash_table[hash(s->opcode >> 12, 0)]; |
| else if ((s->opcode >> 8) != 0) |
| h = &hash_table[hash(s->opcode >> 8, 0)]; |
| else if ((s->opcode >> 4) != 0) |
| h = &hash_table[hash(s->opcode >> 4, 0)]; |
| else |
| h = &hash_table[hash(s->opcode, 0)]; |
| break; |
| |
| case 0x14: |
| if ((s->opcode >> 16) != 0) |
| h = &hash_table[hash(s->opcode >> 16, 0)]; |
| else if ((s->opcode >> 12) != 0) |
| h = &hash_table[hash(s->opcode >> 12, 0)]; |
| else if ((s->opcode >> 8) != 0) |
| h = &hash_table[hash(s->opcode >> 8, 0)]; |
| else if ((s->opcode >> 4) != 0) |
| h = &hash_table[hash(s->opcode >> 4, 0)]; |
| else |
| h = &hash_table[hash(s->opcode, 0)]; |
| break; |
| default: |
| break; |
| } |
| |
| /* go to the last entry in the chain. */ |
| while (h->next) |
| h = h->next; |
| |
| if (h->ops) |
| { |
| h->next = (struct hash_entry *) calloc(1,sizeof(struct hash_entry)); |
| if (!h->next) |
| perror ("malloc failure"); |
| |
| h = h->next; |
| } |
| h->ops = s; |
| h->mask = s->mask; |
| h->opcode = s->opcode; |
| h->format = s->format; |
| h->size = s->size; |
| } |
| } |
| |
| return sd; |
| } |
| |
| static void |
| step_once (SIM_DESC sd, SIM_CPU *cpu) |
| { |
| uint32 curr_ins_size = 0; |
| uint64 mcode = RLW (PC); |
| |
| State.pc_changed = 0; |
| |
| curr_ins_size = do_run (sd, cpu, mcode); |
| |
| #if CR16_DEBUG |
| sim_io_printf (sd, "INS: PC=0x%X, mcode=0x%X\n", PC, mcode); |
| #endif |
| |
| if (curr_ins_size == 0) |
| sim_engine_halt (sd, cpu, NULL, PC, sim_exited, GPR (2)); |
| else if (!State.pc_changed) |
| SET_PC (PC + (curr_ins_size * 2)); /* For word instructions. */ |
| |
| #if 0 |
| /* Check for a breakpoint trap on this instruction. This |
| overrides any pending branches or loops */ |
| if (PSR_DB && PC == DBS) |
| { |
| SET_BPC (PC); |
| SET_BPSR (PSR); |
| SET_PC (SDBT_VECTOR_START); |
| } |
| #endif |
| |
| /* Writeback all the DATA / PC changes */ |
| SLOT_FLUSH (); |
| } |
| |
| void |
| sim_engine_run (SIM_DESC sd, |
| int next_cpu_nr, /* ignore */ |
| int nr_cpus, /* ignore */ |
| int siggnal) |
| { |
| sim_cpu *cpu; |
| |
| SIM_ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER); |
| |
| cpu = STATE_CPU (sd, 0); |
| |
| switch (siggnal) |
| { |
| case 0: |
| break; |
| case GDB_SIGNAL_BUS: |
| case GDB_SIGNAL_SEGV: |
| SET_PC (PC); |
| SET_PSR (PSR); |
| JMP (AE_VECTOR_START); |
| SLOT_FLUSH (); |
| break; |
| case GDB_SIGNAL_ILL: |
| SET_PC (PC); |
| SET_PSR (PSR); |
| SET_HW_PSR ((PSR & (PSR_C_BIT))); |
| JMP (RIE_VECTOR_START); |
| SLOT_FLUSH (); |
| break; |
| default: |
| /* just ignore it */ |
| break; |
| } |
| |
| while (1) |
| { |
| step_once (sd, cpu); |
| if (sim_events_tick (sd)) |
| sim_events_process (sd); |
| } |
| } |
| |
| SIM_RC |
| sim_create_inferior (SIM_DESC sd, struct bfd *abfd, |
| char * const *argv, char * const *env) |
| { |
| bfd_vma start_address; |
| |
| /* reset all state information */ |
| memset (&State, 0, sizeof (State)); |
| |
| /* There was a hack here to copy the values of argc and argv into r0 |
| and r1. The values were also saved into some high memory that |
| won't be overwritten by the stack (0x7C00). The reason for doing |
| this was to allow the 'run' program to accept arguments. Without |
| the hack, this is not possible anymore. If the simulator is run |
| from the debugger, arguments cannot be passed in, so this makes |
| no difference. */ |
| |
| /* set PC */ |
| if (abfd != NULL) |
| start_address = bfd_get_start_address (abfd); |
| else |
| start_address = 0x0; |
| #ifdef DEBUG |
| if (cr16_debug) |
| sim_io_printf (sd, "sim_create_inferior: PC=0x%lx\n", (long) start_address); |
| #endif |
| { |
| SIM_CPU *cpu = STATE_CPU (sd, 0); |
| SET_CREG (PC_CR, start_address); |
| } |
| |
| SLOT_FLUSH (); |
| return SIM_RC_OK; |
| } |
| |
| static uint32 |
| cr16_extract_unsigned_integer (unsigned char *addr, int len) |
| { |
| uint32 retval; |
| unsigned char * p; |
| unsigned char * startaddr = (unsigned char *)addr; |
| unsigned char * endaddr = startaddr + len; |
| |
| retval = 0; |
| |
| for (p = endaddr; p > startaddr;) |
| retval = (retval << 8) | *--p; |
| |
| return retval; |
| } |
| |
| static void |
| cr16_store_unsigned_integer (unsigned char *addr, int len, uint32 val) |
| { |
| unsigned char *p; |
| unsigned char *startaddr = addr; |
| unsigned char *endaddr = startaddr + len; |
| |
| for (p = startaddr; p < endaddr;) |
| { |
| *p++ = val & 0xff; |
| val >>= 8; |
| } |
| } |
| |
| static int |
| cr16_reg_fetch (SIM_CPU *cpu, int rn, unsigned char *memory, int length) |
| { |
| int size; |
| switch ((enum sim_cr16_regs) rn) |
| { |
| case SIM_CR16_R0_REGNUM: |
| case SIM_CR16_R1_REGNUM: |
| case SIM_CR16_R2_REGNUM: |
| case SIM_CR16_R3_REGNUM: |
| case SIM_CR16_R4_REGNUM: |
| case SIM_CR16_R5_REGNUM: |
| case SIM_CR16_R6_REGNUM: |
| case SIM_CR16_R7_REGNUM: |
| case SIM_CR16_R8_REGNUM: |
| case SIM_CR16_R9_REGNUM: |
| case SIM_CR16_R10_REGNUM: |
| case SIM_CR16_R11_REGNUM: |
| cr16_store_unsigned_integer (memory, 2, GPR (rn - SIM_CR16_R0_REGNUM)); |
| size = 2; |
| break; |
| case SIM_CR16_R12_REGNUM: |
| case SIM_CR16_R13_REGNUM: |
| case SIM_CR16_R14_REGNUM: |
| case SIM_CR16_R15_REGNUM: |
| cr16_store_unsigned_integer (memory, 4, GPR (rn - SIM_CR16_R0_REGNUM)); |
| size = 4; |
| break; |
| case SIM_CR16_PC_REGNUM: |
| case SIM_CR16_ISP_REGNUM: |
| case SIM_CR16_USP_REGNUM: |
| case SIM_CR16_INTBASE_REGNUM: |
| case SIM_CR16_PSR_REGNUM: |
| case SIM_CR16_CFG_REGNUM: |
| case SIM_CR16_DBS_REGNUM: |
| case SIM_CR16_DCR_REGNUM: |
| case SIM_CR16_DSR_REGNUM: |
| case SIM_CR16_CAR0_REGNUM: |
| case SIM_CR16_CAR1_REGNUM: |
| cr16_store_unsigned_integer (memory, 4, CREG (rn - SIM_CR16_PC_REGNUM)); |
| size = 4; |
| break; |
| default: |
| size = 0; |
| break; |
| } |
| return size; |
| } |
| |
| static int |
| cr16_reg_store (SIM_CPU *cpu, int rn, unsigned char *memory, int length) |
| { |
| SIM_DESC sd = CPU_STATE (cpu); |
| int size; |
| switch ((enum sim_cr16_regs) rn) |
| { |
| case SIM_CR16_R0_REGNUM: |
| case SIM_CR16_R1_REGNUM: |
| case SIM_CR16_R2_REGNUM: |
| case SIM_CR16_R3_REGNUM: |
| case SIM_CR16_R4_REGNUM: |
| case SIM_CR16_R5_REGNUM: |
| case SIM_CR16_R6_REGNUM: |
| case SIM_CR16_R7_REGNUM: |
| case SIM_CR16_R8_REGNUM: |
| case SIM_CR16_R9_REGNUM: |
| case SIM_CR16_R10_REGNUM: |
| case SIM_CR16_R11_REGNUM: |
| SET_GPR (rn - SIM_CR16_R0_REGNUM, cr16_extract_unsigned_integer (memory, 2)); |
| size = 2; |
| break; |
| case SIM_CR16_R12_REGNUM: |
| case SIM_CR16_R13_REGNUM: |
| case SIM_CR16_R14_REGNUM: |
| case SIM_CR16_R15_REGNUM: |
| SET_GPR32 (rn - SIM_CR16_R0_REGNUM, cr16_extract_unsigned_integer (memory, 2)); |
| size = 4; |
| break; |
| case SIM_CR16_PC_REGNUM: |
| case SIM_CR16_ISP_REGNUM: |
| case SIM_CR16_USP_REGNUM: |
| case SIM_CR16_INTBASE_REGNUM: |
| case SIM_CR16_PSR_REGNUM: |
| case SIM_CR16_CFG_REGNUM: |
| case SIM_CR16_DBS_REGNUM: |
| case SIM_CR16_DCR_REGNUM: |
| case SIM_CR16_DSR_REGNUM: |
| case SIM_CR16_CAR0_REGNUM: |
| case SIM_CR16_CAR1_REGNUM: |
| SET_CREG (rn - SIM_CR16_PC_REGNUM, cr16_extract_unsigned_integer (memory, 4)); |
| size = 4; |
| break; |
| default: |
| size = 0; |
| break; |
| } |
| SLOT_FLUSH (); |
| return size; |
| } |