summaryrefslogtreecommitdiff
path: root/sim/ucsim/sim.src/serial_hw.cc
diff options
context:
space:
mode:
authorXavier ASUS <xavi92psx@gmail.com>2019-10-18 00:31:54 +0200
committerXavier ASUS <xavi92psx@gmail.com>2019-10-18 00:31:54 +0200
commit268a53de823a6750d6256ee1fb1e7707b4b45740 (patch)
tree42c1799a9a82b2f7d9790ee9fe181d72a7274751 /sim/ucsim/sim.src/serial_hw.cc
downloadsdcc-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/sim.src/serial_hw.cc')
-rw-r--r--sim/ucsim/sim.src/serial_hw.cc511
1 files changed, 511 insertions, 0 deletions
diff --git a/sim/ucsim/sim.src/serial_hw.cc b/sim/ucsim/sim.src/serial_hw.cc
new file mode 100644
index 0000000..529b43d
--- /dev/null
+++ b/sim/ucsim/sim.src/serial_hw.cc
@@ -0,0 +1,511 @@
+/*
+ * Simulator of microcontrollers (sim.src/serial_hw.cc)
+ *
+ * Copyright (C) 2016,16 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@*/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <ctype.h>
+
+#include "utils.h"
+#include "globals.h"
+#include "fiocl.h"
+
+#include "serial_hwcl.h"
+
+
+bool
+cl_serial_io::input_avail(void)
+{
+ return cl_hw_io::input_avail();
+}
+
+
+cl_serial_hw::cl_serial_hw(class cl_uc *auc, int aid, chars aid_string):
+ cl_hw(auc, HW_UART, aid, (const char *)aid_string)
+{
+ listener= 0;
+}
+
+cl_serial_hw::~cl_serial_hw(void)
+{
+ delete serial_in_file_option;
+ delete serial_out_file_option;
+ delete io;
+}
+
+int
+cl_serial_hw::init(void)
+{
+ char *s;
+
+ cl_hw::init();
+
+ make_io();
+ input_avail= false;
+
+ s= format_string("serial%d_in_file", id);
+ serial_in_file_option= new cl_optref(this);
+ serial_in_file_option->init();
+ serial_in_file_option->use(s);
+ free(s);
+ s= format_string("serial%d_out_file", id);
+ serial_out_file_option= new cl_optref(this);
+ serial_out_file_option->init();
+ serial_out_file_option->use(s);
+ free(s);
+
+ s= format_string("serial%d_port", id);
+ serial_port_option= new cl_optref(this);
+ serial_port_option->init();
+ class cl_option *o= serial_port_option->use(s);
+ free(s);
+
+ int port= -1;
+ if (o)
+ {
+ port= serial_port_option->get_value((long)0);
+ if (port < 0)
+ ;
+ }
+ if (port > 0)
+ {
+ listener= new cl_serial_listener(port, application, this, sl_io);
+ listener->init();
+ class cl_commander_base *c= application->get_commander();
+ c->add_console(listener);
+ }
+
+ o= NULL;
+ s= format_string("serial%d_iport", id);
+ serial_iport_option= new cl_optref(this);
+ serial_iport_option->init();
+ o= serial_iport_option->use(s);
+ free(s);
+
+ port= -1;
+ if (o)
+ {
+ port= serial_iport_option->get_value((long)0);
+ if (port < 0)
+ ;
+ }
+ if (port > 0)
+ {
+ listener= new cl_serial_listener(port, application, this, sl_i);
+ listener->init();
+ class cl_commander_base *c= application->get_commander();
+ c->add_console(listener);
+ }
+
+ o= NULL;
+ s= format_string("serial%d_oport", id);
+ serial_oport_option= new cl_optref(this);
+ serial_oport_option->init();
+ o= serial_oport_option->use(s);
+ free(s);
+
+ port= -1;
+ if (o)
+ {
+ port= serial_oport_option->get_value((long)0);
+ if (port < 0)
+ ;
+ }
+ if (port > 0)
+ {
+ listener= new cl_serial_listener(port, application, this, sl_o);
+ listener->init();
+ class cl_commander_base *c= application->get_commander();
+ c->add_console(listener);
+ }
+
+ char *f_serial_in = (char*)serial_in_file_option->get_value((char*)0);
+ char *f_serial_out= (char*)serial_out_file_option->get_value((char*)0);
+ class cl_f *fi, *fo;
+ if (f_serial_in)
+ {
+ if (f_serial_in[0] == '\001')
+ fi= (class cl_f *)(strtoll(&f_serial_in[1], 0, 0));
+ else
+ fi= mk_io(chars(f_serial_in), cchars("r"));
+ if (!fi->tty)
+ fprintf(stderr, "Warning: serial input interface connected to a "
+ "non-terminal file.\n");
+ }
+ else
+ fi= 0;//mk_io(chars(""), chars(""));
+ if (f_serial_out)
+ {
+ if (f_serial_out[0] == '\001')
+ fo= (class cl_f *)(strtoll(&f_serial_out[1], 0, 0));
+ else
+ fo= mk_io(chars(f_serial_out), "w");
+ if (!fo->tty)
+ fprintf(stderr, "Warning: serial output interface connected to a "
+ "non-terminal file.\n");
+ }
+ else
+ fo= 0;//mk_io(chars(""), chars(""));
+
+ io->replace_files(true, fi, fo);
+
+ if (fi)
+ {
+ fi->interactive(NULL);
+ fi->raw();
+ fi->echo(NULL);
+ }
+
+ menu= 0;
+
+ cfg_set(serconf_on, true);
+ cfg_set(serconf_check_often, false);
+ cfg_set(serconf_escape, 'x'-'a'+1);
+
+ cl_var *v;
+ chars pn(id_string);
+ pn.append("%d_", id);
+ uc->vars->add(v= new cl_var(pn+chars("on"), cfg, serconf_on,
+ cfg_help(serconf_on)));
+ v->init();
+ uc->vars->add(v= new cl_var(pn+chars("check_often"), cfg, serconf_check_often,
+ cfg_help(serconf_check_often)));
+ v->init();
+ uc->vars->add(v= new cl_var(pn+chars("esc_char"), cfg, serconf_escape,
+ cfg_help(serconf_escape)));
+ v->init();
+
+ uc->vars->add(v= new cl_var(pn+chars("received_char"), cfg, serconf_received,
+ cfg_help(serconf_received)));
+ v->init();
+
+ uc->vars->add(v= new cl_var(pn+chars("flowctrl"), cfg, serconf_flowctrl,
+ cfg_help(serconf_flowctrl)));
+ v->init();
+
+ uc->vars->add(v= new cl_var(pn+chars("able_receive"), cfg, serconf_able_receive,
+ cfg_help(serconf_able_receive)));
+ v->init();
+
+ cfg_set(serconf_able_receive, 1);
+ return 0;
+}
+
+char *
+cl_serial_hw::cfg_help(t_addr addr)
+{
+ switch (addr)
+ {
+ case serconf_on:
+ return (char*)"Turn simulation of UART on or off (bool, RW)";
+ case serconf_check_often:
+ return (char*)"Check input file at every cycle (bool, RW)";
+ case serconf_escape:
+ return (char*)"Escape char on display (int, RW)";
+ case serconf_common:
+ return (char*)"Not used";
+ case serconf_received:
+ return (char*)"Received char written by simulator (int, R)";
+ case serconf_flowctrl:
+ return (char*)"Flow-control simulation on/off (bool, RW)";
+ case serconf_able_receive:
+ return (char*)"UART enabled to receive by flow-control (bool, RW)";
+ }
+ return (char*)"Not used";
+}
+
+t_mem
+cl_serial_hw::conf_op(cl_memory_cell *cell, t_addr addr, t_mem *val)
+{
+ switch ((enum serial_cfg)addr)
+ {
+ case serconf_on: // turn this HW on/off
+ if (val)
+ {
+ if (*val)
+ on= true;
+ else
+ on= false;
+ }
+ else
+ {
+ cell->set(on?1:0);
+ }
+ break;
+ case serconf_check_often:
+ if (val)
+ {
+ cell->set(*val?1:0);
+ }
+ break;
+ case serconf_escape:
+ if (val)
+ {
+ char c= tolower(*val);
+ if ((c >= 'a') &&
+ (c <= 'z'))
+ cell->set(c - 'a'+1);
+ }
+ default:
+ break;
+ }
+ return cell->get();
+}
+
+void
+cl_serial_hw::make_io()
+{
+ if (!io)
+ {
+ io= new cl_serial_io(this);
+ application->get_commander()->add_console(io);
+ }
+}
+
+void
+cl_serial_hw::new_io(class cl_f *f_in, class cl_f *f_out)
+{
+ char esc= (char)cfg_get(serconf_escape);
+ cl_hw::new_io(f_in, f_out);
+ if (io)
+ io->dd_printf("%s[%d] terminal display, press ^%c to access control menu\n",
+ id_string, id,
+ 'a'+esc-1);
+ menu= 0;
+}
+
+bool
+cl_serial_hw::proc_input(void)
+{
+ int c;
+ char esc= (char)cfg_get(serconf_escape);
+ bool run= uc->sim->state & SIM_GO;
+ class cl_f *fin, *fout;
+ int flw= cfg_get(serconf_flowctrl);
+ int able= cfg_get(serconf_able_receive);
+
+ fin= io->get_fin();
+ fout= io->get_fout();
+
+ if (fin->eof())
+ {
+ if (fout &&
+ (fout->file_id == fin->file_id))
+ {
+ delete fout;
+ io->replace_files(false, fin, 0);
+ fout= 0;
+ }
+ delete fin;
+ io->replace_files(false, 0, fout);
+ return true;
+ }
+ if (menu == 0)
+ {
+ if (fin->tty && !flw)
+ {
+ if (fin->read(&c, 1))
+ {
+ if (c == esc)
+ {
+ menu= 'm';
+ io->dd_printf("\n");
+ io->dd_cprintf("ui_title", "Simulator control menu\n");
+ io->dd_cprintf("ui_mkey", " %c ", 'a'+esc-1);
+ io->dd_cprintf("ui_mitem", "Insert ^%c\n", 'a'+esc-1);
+ io->dd_cprintf("ui_mkey", " s,r,g ");
+ io->dd_cprintf("ui_mitem", "Start simulation\n");
+ io->dd_cprintf("ui_mkey", " p ");
+ io->dd_cprintf("ui_mitem", "Stop simulation\n");
+ io->dd_cprintf("ui_mkey", " T ");
+ io->dd_cprintf("ui_mitem", "Reset CPU\n");
+ io->dd_cprintf("ui_mkey", " q ");
+ io->dd_cprintf("ui_mitem", "Quit simulator\n");
+ io->dd_cprintf("ui_mkey", " o ");
+ io->dd_cprintf("ui_mitem", "Close serial terminal\n");
+ io->dd_cprintf("ui_mkey", " e ");
+ io->dd_cprintf("ui_mitem", "Exit menu\n");
+ io->dd_cprintf("ui_mkey", " n ");
+ io->dd_cprintf("ui_mitem", "Change display\n");
+ }
+ else if (!input_avail)
+ {
+ input= c;
+ input_avail= true;
+ }
+ }
+ }
+ else if (!input_avail)
+ {
+ if (!flw ||
+ able)
+ {
+ if (fin->read(&c, 1))
+ {
+ input= c;
+ input_avail= true;
+ cfg_set(serconf_able_receive, 0);
+ }
+ }
+ }
+ }
+ else
+ {
+ if (fin->read(&c, 1))
+ {
+ switch (menu)
+ {
+ case 'm':
+ if ((c == esc-1+'a') ||
+ (c == esc-1+'A') ||
+ (c == esc))
+ {
+ // insert ^esc
+ if (run && !input_avail)
+ {
+ input= esc, input_avail= true;
+ io->dd_printf("^%c enterted.\n", 'a'+esc-1);
+ }
+ else
+ io->dd_printf("Control menu exited.\n");
+ menu= 0;
+ }
+ switch (c)
+ {
+ case 'e': case 'E': case 'e'-'a'+1:
+ // exit menu
+ menu= 0;
+ io->dd_printf("Control menu exited.\n");
+ break;
+ case 's': case 'S': case 's'-'a'+1:
+ case 'r': case 'R': case 'r'-'a'+1:
+ case 'g': case 'G': case 'g'-'a'+1:
+ // start
+ uc->sim->start(0, 0);
+ menu= 0;
+ io->dd_printf("Simulation started.\n");
+ break;
+ case 'p': case 'P': case 'p'-'a'+1:
+ uc->sim->stop(resSIMIF);
+ // stop
+ menu= 0;
+ io->dd_printf("Simulation stopped.\n");
+ break;
+ case 'T':
+ uc->reset();
+ menu= 0;
+ io->dd_printf("CPU reset.\n");
+ break;
+ case 'q': case 'Q': case 'q'-'a'+1:
+ // kill
+ uc->sim->state|= SIM_QUIT;
+ menu= 0;
+ io->dd_printf("Exit simulator.\n");
+ break;
+ case 'o': case 'O': case 'o'-'a'+1:
+ {
+ // close
+ io->dd_printf("Closing terminal.\n");
+ menu= 0;
+ io->convert2console();
+ break;
+ }
+ case 'n': case 'N': case 'n'-'a'+1:
+ {
+ class cl_hw *h= next_displayer();
+ if (!h)
+ io->dd_printf("No other displayer.\n");
+ else
+ {
+ io->tu_reset();
+ io->tu_cls();
+ io->pass2hw(h);
+ }
+ menu= 0;
+ break;
+ }
+ default:
+ menu= 0;
+ io->dd_printf("Control menu closed (%d).\n", c);
+ break;
+ }
+ break;
+ }
+ }
+ }
+ return true;
+}
+
+void
+cl_serial_hw::reset(void)
+{
+ cfg_set(serconf_able_receive, 1);
+}
+
+cl_serial_listener::cl_serial_listener(int serverport, class cl_app *the_app,
+ class cl_serial_hw *the_serial,
+ enum ser_listener_for slf):
+ cl_listen_console(serverport, the_app)
+{
+ serial_hw= the_serial;
+ sl_for= slf;
+}
+
+int
+cl_serial_listener::init(void)
+{
+ if (serial_hw)
+ set_name(chars("", "serial_listener_%s_%d\n", serial_hw->get_name(), serial_hw->id));
+ return 0;
+}
+
+int
+cl_serial_listener::proc_input(class cl_cmdset *cmdset)
+{
+ class cl_f *i, *o;
+
+ switch (sl_for)
+ {
+ case sl_io:
+ srv_accept(fin, &i, &o);
+ i->set_telnet(true);
+ serial_hw->new_io(i, o);
+ break;
+ case sl_i:
+ srv_accept(fin, &i, NULL);
+ i->set_telnet(true);
+ serial_hw->new_i(i);
+ break;
+ case sl_o:
+ srv_accept(fin, NULL, &o);
+ serial_hw->new_o(o);
+ break;
+ }
+ return 0;
+}
+
+
+/* End of sim.src/serial_hw.cc */