diff options
| author | Xavier ASUS <xavi92psx@gmail.com> | 2019-10-18 00:31:54 +0200 |
|---|---|---|
| committer | Xavier ASUS <xavi92psx@gmail.com> | 2019-10-18 00:31:54 +0200 |
| commit | 268a53de823a6750d6256ee1fb1e7707b4b45740 (patch) | |
| tree | 42c1799a9a82b2f7d9790ee9fe181d72a7274751 /sim/ucsim/s51.src/interrupt.cc | |
| download | sdcc-gas-268a53de823a6750d6256ee1fb1e7707b4b45740.tar.gz | |
sdcc-3.9.0 fork implementing GNU assembler syntax
This fork aims to provide better support for stm8-binutils
Diffstat (limited to 'sim/ucsim/s51.src/interrupt.cc')
| -rw-r--r-- | sim/ucsim/s51.src/interrupt.cc | 188 |
1 files changed, 188 insertions, 0 deletions
diff --git a/sim/ucsim/s51.src/interrupt.cc b/sim/ucsim/s51.src/interrupt.cc new file mode 100644 index 0000000..b050045 --- /dev/null +++ b/sim/ucsim/s51.src/interrupt.cc @@ -0,0 +1,188 @@ +/* + * Simulator of microcontrollers (interrupt.cc) + * + * Copyright (C) 1999,99 Drotos Daniel, Talker Bt. + * + * To contact author send email to drdani@mazsola.iit.uni-miskolc.hu + * + */ + +/* This file is part of microcontroller simulator: ucsim. + +UCSIM 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 2 of the License, or +(at your option) any later version. + +UCSIM 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 UCSIM; see the file COPYING. If not, write to the Free +Software Foundation, 59 Temple Place - Suite 330, Boston, MA +02111-1307, USA. */ +/*@1@*/ + +// prj +#include <stdarg.h> +#include "utils.h" + +// sim +#include "itsrccl.h" + +// local +#include "interruptcl.h" +#include "regs51.h" +//#include "uc51cl.h" +#include "types51.h" + + +cl_interrupt::cl_interrupt(class cl_uc *auc): + cl_hw(auc, HW_INTERRUPT, 0, "irq") +{ + was_reti= false; +} + +int +cl_interrupt::init(void) +{ + cl_hw::init(); + sfr= uc->address_space(MEM_SFR_ID); + if (sfr) + { + register_cell(sfr, IE); + cell_tcon= register_cell(sfr, TCON); + bit_INT0= sfr->read(P3) & bm_INT0; + bit_INT1= sfr->read(P3) & bm_INT1; + cl_address_space *bas= uc->address_space("bits"); + cell_it0= register_cell(bas, 0x88); + cell_it1= register_cell(bas, 0x8a); + } + return(0); +} + +void +cl_interrupt::added_to_uc(void) +{ + class cl_address_space *sfr= uc->address_space(MEM_SFR_ID); + class cl_it_src *is; + + uc->it_sources->add(is= new cl_it_src(uc, bmEX0, + sfr->get_cell(IE), bmEX0, + sfr->get_cell(TCON), bmIE0, + 0x0003, true, false, + "external #0", 1)); + is->init(); + uc->it_sources->add(is= new cl_it_src(uc, bmEX1, + sfr->get_cell(IE), bmEX1, + sfr->get_cell(TCON), bmIE1, + 0x0013, true, false, + "external #1", 3)); + is->init(); +} + +void +cl_interrupt::write(class cl_memory_cell *cell, t_mem *val) +{ + if (cell == cell_it0) + bit_IT0= *val; + else if (cell == cell_it1) + bit_IT1= *val; + else if (cell == cell_tcon) + { + bit_IT0= *val & bmIT0; + bit_IT1= *val & bmIT1; + } + else + // IE register + was_reti= true; +} + +/*void +cl_interrupt::mem_cell_changed(class cl_m *mem, t_addr addr) +{ +}*/ + +int +cl_interrupt::tick(int cycles) +{ + if (!bit_IT0 && !bit_INT0) + cell_tcon->set_bit1(bmIE0); + if (!bit_IT1 && !bit_INT1) + cell_tcon->set_bit1(bmIE1); + return(resGO); +} + +void +cl_interrupt::reset(void) +{ + was_reti= false; +} + +void +cl_interrupt::happen(class cl_hw *where, enum hw_event he, void *params) +{ + struct ev_port_changed *ep= (struct ev_port_changed *)params; + + if (where->cathegory == HW_PORT && + he == EV_PORT_CHANGED && + ep->id == 3) + { + t_mem p3n= ep->new_pins & ep->new_value; + t_mem p3o= ep->pins & ep->prev_value; + if (bit_IT0 && + !(p3n & bm_INT0) && + (p3o & bm_INT0)) + cell_tcon->set_bit1(bmIE0); + if (bit_IT1 && + !(p3n & bm_INT1) && + (p3o & bm_INT1)) + cell_tcon->set_bit1(bmIE1); + bit_INT0= p3n & bm_INT0; + bit_INT1= p3n & bm_INT1; + } +} + + +void +cl_interrupt::print_info(class cl_console_base *con) +{ + //int ie= sfr->get(IE); + int i; + + con->dd_printf("Interrupts are %s. Interrupt sources:\n", + (uc->it_enabled())?"enabled":"disabled"); + con->dd_printf(" Handler En Pr Req Act Name\n"); + for (i= 0; i < uc->it_sources->count; i++) + { + class cl_it_src *is= (class cl_it_src *)(uc->it_sources->at(i)); + con->dd_printf(" 0x%06x", AU(is->addr)); + con->dd_printf(" %-3s", (is->enabled())?"en":"dis"); + con->dd_printf(" %2d", uc->priority_of(is->ie_mask)); + con->dd_printf(" %-3s", (is->pending())?"YES":"no"); + con->dd_printf(" %-3s", (is->active)?"act":"no"); + con->dd_printf(" %s", object_name(is)); + con->dd_printf("\n"); + } + con->dd_printf("Active interrupt service(s):\n"); + con->dd_printf(" Pr Handler PC Source\n"); + for (i= 0; i < uc->it_levels->count; i++) + { + class it_level *il= (class it_level *)(uc->it_levels->at(i)); + if (il->level >= 0) + { + con->dd_printf(" %2d", il->level); + con->dd_printf(" 0x%06x", AU(il->addr)); + con->dd_printf(" 0x%06x", AU(il->PC)); + con->dd_printf(" %s", (il->source)?(object_name(il->source)): + "nothing"); + con->dd_printf("\n"); + } + } + print_cfg_info(con); +} + + +/* End of s51.src/interrupt.cc */ |
