123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- /*
- * sercfg.c - the serial-port specific parts of the PuTTY
- * configuration box. Centralised as cross-platform code because
- * more than one platform will want to use it, but not part of the
- * main configuration. The expectation is that each platform's
- * local config function will call out to ser_setup_config_box() if
- * it needs to set up the standard serial stuff. (Of course, it can
- * then apply local tweaks after ser_setup_config_box() returns, if
- * it needs to.)
- */
- #include <assert.h>
- #include <stdlib.h>
- #include "putty.h"
- #include "dialog.h"
- #include "storage.h"
- static void serial_parity_handler(union control *ctrl, void *dlg,
- void *data, int event)
- {
- static const struct {
- const char *name;
- int val;
- } parities[] = {
- {"None", SER_PAR_NONE},
- {"Odd", SER_PAR_ODD},
- {"Even", SER_PAR_EVEN},
- {"Mark", SER_PAR_MARK},
- {"Space", SER_PAR_SPACE},
- };
- int mask = ctrl->listbox.context.i;
- int i, j;
- Conf *conf = (Conf *)data;
- if (event == EVENT_REFRESH) {
- /* Fetching this once at the start of the function ensures we
- * remember what the right value is supposed to be when
- * operations below cause reentrant calls to this function. */
- int oldparity = conf_get_int(conf, CONF_serparity);
- dlg_update_start(ctrl, dlg);
- dlg_listbox_clear(ctrl, dlg);
- for (i = 0; i < lenof(parities); i++) {
- if (mask & (1 << i))
- dlg_listbox_addwithid(ctrl, dlg, parities[i].name,
- parities[i].val);
- }
- for (i = j = 0; i < lenof(parities); i++) {
- if (mask & (1 << i)) {
- if (oldparity == parities[i].val) {
- dlg_listbox_select(ctrl, dlg, j);
- break;
- }
- j++;
- }
- }
- if (i == lenof(parities)) { /* an unsupported setting was chosen */
- dlg_listbox_select(ctrl, dlg, 0);
- oldparity = SER_PAR_NONE;
- }
- dlg_update_done(ctrl, dlg);
- conf_set_int(conf, CONF_serparity, oldparity); /* restore */
- } else if (event == EVENT_SELCHANGE) {
- int i = dlg_listbox_index(ctrl, dlg);
- if (i < 0)
- i = SER_PAR_NONE;
- else
- i = dlg_listbox_getid(ctrl, dlg, i);
- conf_set_int(conf, CONF_serparity, i);
- }
- }
- static void serial_flow_handler(union control *ctrl, void *dlg,
- void *data, int event)
- {
- static const struct {
- const char *name;
- int val;
- } flows[] = {
- {"None", SER_FLOW_NONE},
- {"XON/XOFF", SER_FLOW_XONXOFF},
- {"RTS/CTS", SER_FLOW_RTSCTS},
- {"DSR/DTR", SER_FLOW_DSRDTR},
- };
- int mask = ctrl->listbox.context.i;
- int i, j;
- Conf *conf = (Conf *)data;
- if (event == EVENT_REFRESH) {
- /* Fetching this once at the start of the function ensures we
- * remember what the right value is supposed to be when
- * operations below cause reentrant calls to this function. */
- int oldflow = conf_get_int(conf, CONF_serflow);
- dlg_update_start(ctrl, dlg);
- dlg_listbox_clear(ctrl, dlg);
- for (i = 0; i < lenof(flows); i++) {
- if (mask & (1 << i))
- dlg_listbox_addwithid(ctrl, dlg, flows[i].name, flows[i].val);
- }
- for (i = j = 0; i < lenof(flows); i++) {
- if (mask & (1 << i)) {
- if (oldflow == flows[i].val) {
- dlg_listbox_select(ctrl, dlg, j);
- break;
- }
- j++;
- }
- }
- if (i == lenof(flows)) { /* an unsupported setting was chosen */
- dlg_listbox_select(ctrl, dlg, 0);
- oldflow = SER_FLOW_NONE;
- }
- dlg_update_done(ctrl, dlg);
- conf_set_int(conf, CONF_serflow, oldflow);/* restore */
- } else if (event == EVENT_SELCHANGE) {
- int i = dlg_listbox_index(ctrl, dlg);
- if (i < 0)
- i = SER_FLOW_NONE;
- else
- i = dlg_listbox_getid(ctrl, dlg, i);
- conf_set_int(conf, CONF_serflow, i);
- }
- }
- void ser_setup_config_box(struct controlbox *b, int midsession,
- int parity_mask, int flow_mask)
- {
- struct controlset *s;
- union control *c;
- if (!midsession) {
- int i;
- extern void config_protocolbuttons_handler(union control *, void *,
- void *, int);
- /*
- * Add the serial back end to the protocols list at the
- * top of the config box.
- */
- s = ctrl_getset(b, "Session", "hostport",
- "Specify the destination you want to connect to");
- for (i = 0; i < s->ncontrols; i++) {
- c = s->ctrls[i];
- if (c->generic.type == CTRL_RADIO &&
- c->generic.handler == config_protocolbuttons_handler) {
- c->radio.nbuttons++;
- c->radio.ncolumns++;
- c->radio.buttons =
- sresize(c->radio.buttons, c->radio.nbuttons, char *);
- c->radio.buttons[c->radio.nbuttons-1] =
- dupstr("Serial");
- c->radio.buttondata =
- sresize(c->radio.buttondata, c->radio.nbuttons, intorptr);
- c->radio.buttondata[c->radio.nbuttons-1] = I(PROT_SERIAL);
- if (c->radio.shortcuts) {
- c->radio.shortcuts =
- sresize(c->radio.shortcuts, c->radio.nbuttons, char);
- c->radio.shortcuts[c->radio.nbuttons-1] = 'r';
- }
- }
- }
- }
- /*
- * Entirely new Connection/Serial panel for serial port
- * configuration.
- */
- ctrl_settitle(b, "Connection/Serial",
- "Options controlling local serial lines");
- if (!midsession) {
- /*
- * We don't permit switching to a different serial port in
- * midflight, although we do allow all other
- * reconfiguration.
- */
- s = ctrl_getset(b, "Connection/Serial", "serline",
- "Select a serial line");
- ctrl_editbox(s, "Serial line to connect to", 'l', 40,
- HELPCTX(serial_line),
- conf_editbox_handler, I(CONF_serline), I(1));
- }
- s = ctrl_getset(b, "Connection/Serial", "sercfg", "Configure the serial line");
- ctrl_editbox(s, "Speed (baud)", 's', 40,
- HELPCTX(serial_speed),
- conf_editbox_handler, I(CONF_serspeed), I(-1));
- ctrl_editbox(s, "Data bits", 'b', 40,
- HELPCTX(serial_databits),
- conf_editbox_handler, I(CONF_serdatabits), I(-1));
- /*
- * Stop bits come in units of one half.
- */
- ctrl_editbox(s, "Stop bits", 't', 40,
- HELPCTX(serial_stopbits),
- conf_editbox_handler, I(CONF_serstopbits), I(-2));
- ctrl_droplist(s, "Parity", 'p', 40,
- HELPCTX(serial_parity),
- serial_parity_handler, I(parity_mask));
- ctrl_droplist(s, "Flow control", 'f', 40,
- HELPCTX(serial_flow),
- serial_flow_handler, I(flow_mask));
- }
|