2006-08-28 11:13:56 +00:00
|
|
|
/*
|
|
|
|
* 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},
|
|
|
|
};
|
2006-08-28 13:08:15 +00:00
|
|
|
int mask = ctrl->listbox.context.i;
|
|
|
|
int i, j;
|
2006-08-28 11:13:56 +00:00
|
|
|
Config *cfg = (Config *)data;
|
|
|
|
|
|
|
|
if (event == EVENT_REFRESH) {
|
2009-08-10 20:38:46 +00:00
|
|
|
int oldparity = cfg->serparity;/* preserve past reentrant calls */
|
2006-08-28 11:13:56 +00:00
|
|
|
dlg_update_start(ctrl, dlg);
|
|
|
|
dlg_listbox_clear(ctrl, dlg);
|
2006-08-28 13:08:15 +00:00
|
|
|
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++) {
|
2009-08-10 20:38:46 +00:00
|
|
|
if (mask & (1 << i)) {
|
|
|
|
if (oldparity == parities[i].val) {
|
|
|
|
dlg_listbox_select(ctrl, dlg, j);
|
|
|
|
break;
|
|
|
|
}
|
2006-08-28 13:08:15 +00:00
|
|
|
j++;
|
2009-08-10 20:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == lenof(parities)) { /* an unsupported setting was chosen */
|
|
|
|
dlg_listbox_select(ctrl, dlg, 0);
|
|
|
|
oldparity = SER_PAR_NONE;
|
2006-08-28 13:08:15 +00:00
|
|
|
}
|
2006-08-28 11:13:56 +00:00
|
|
|
dlg_update_done(ctrl, dlg);
|
2009-08-10 20:38:46 +00:00
|
|
|
cfg->serparity = oldparity; /* restore */
|
2006-08-28 11:13:56 +00:00
|
|
|
} 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);
|
|
|
|
cfg->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},
|
|
|
|
};
|
2006-08-28 13:08:15 +00:00
|
|
|
int mask = ctrl->listbox.context.i;
|
|
|
|
int i, j;
|
2006-08-28 11:13:56 +00:00
|
|
|
Config *cfg = (Config *)data;
|
|
|
|
|
|
|
|
if (event == EVENT_REFRESH) {
|
2009-08-10 20:38:46 +00:00
|
|
|
int oldflow = cfg->serflow; /* preserve past reentrant calls */
|
2006-08-28 11:13:56 +00:00
|
|
|
dlg_update_start(ctrl, dlg);
|
|
|
|
dlg_listbox_clear(ctrl, dlg);
|
2006-08-28 13:08:15 +00:00
|
|
|
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++) {
|
2009-08-10 20:38:46 +00:00
|
|
|
if (mask & (1 << i)) {
|
|
|
|
if (oldflow == flows[i].val) {
|
|
|
|
dlg_listbox_select(ctrl, dlg, j);
|
|
|
|
break;
|
|
|
|
}
|
2006-08-28 13:08:15 +00:00
|
|
|
j++;
|
2009-08-10 20:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == lenof(flows)) { /* an unsupported setting was chosen */
|
|
|
|
dlg_listbox_select(ctrl, dlg, 0);
|
|
|
|
oldflow = SER_FLOW_NONE;
|
2006-08-28 13:08:15 +00:00
|
|
|
}
|
2006-08-28 11:13:56 +00:00
|
|
|
dlg_update_done(ctrl, dlg);
|
2009-08-10 20:38:46 +00:00
|
|
|
cfg->serflow = oldflow; /* restore */
|
2006-08-28 11:13:56 +00:00
|
|
|
} else if (event == EVENT_SELCHANGE) {
|
|
|
|
int i = dlg_listbox_index(ctrl, dlg);
|
|
|
|
if (i < 0)
|
2009-08-10 20:38:46 +00:00
|
|
|
i = SER_FLOW_NONE;
|
2006-08-28 11:13:56 +00:00
|
|
|
else
|
|
|
|
i = dlg_listbox_getid(ctrl, dlg, i);
|
|
|
|
cfg->serflow = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-08-28 13:08:15 +00:00
|
|
|
void ser_setup_config_box(struct controlbox *b, int midsession,
|
|
|
|
int parity_mask, int flow_mask)
|
2006-08-28 11:13:56 +00:00
|
|
|
{
|
|
|
|
struct controlset *s;
|
|
|
|
union control *c;
|
|
|
|
|
2007-04-22 14:39:01 +00:00
|
|
|
if (!midsession) {
|
2006-08-28 11:13:56 +00:00
|
|
|
int i;
|
|
|
|
extern void config_protocolbuttons_handler(union control *, void *,
|
|
|
|
void *, int);
|
2007-04-22 14:39:01 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* 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");
|
|
|
|
|
2006-08-28 11:13:56 +00:00
|
|
|
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);
|
2009-11-13 13:24:32 +00:00
|
|
|
c->radio.shortcuts[c->radio.nbuttons-1] = 'r';
|
2006-08-28 11:13:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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),
|
|
|
|
dlg_stdeditbox_handler, I(offsetof(Config,serline)),
|
|
|
|
I(sizeof(((Config *)0)->serline)));
|
|
|
|
}
|
|
|
|
|
|
|
|
s = ctrl_getset(b, "Connection/Serial", "sercfg", "Configure the serial line");
|
|
|
|
ctrl_editbox(s, "Speed (baud)", 's', 40,
|
|
|
|
HELPCTX(serial_speed),
|
|
|
|
dlg_stdeditbox_handler, I(offsetof(Config,serspeed)), I(-1));
|
|
|
|
ctrl_editbox(s, "Data bits", 'b', 40,
|
|
|
|
HELPCTX(serial_databits),
|
|
|
|
dlg_stdeditbox_handler,I(offsetof(Config,serdatabits)),I(-1));
|
|
|
|
/*
|
|
|
|
* Stop bits come in units of one half.
|
|
|
|
*/
|
|
|
|
ctrl_editbox(s, "Stop bits", 't', 40,
|
|
|
|
HELPCTX(serial_stopbits),
|
|
|
|
dlg_stdeditbox_handler,I(offsetof(Config,serstopbits)),I(-2));
|
|
|
|
ctrl_droplist(s, "Parity", 'p', 40,
|
|
|
|
HELPCTX(serial_parity),
|
2006-08-28 13:08:15 +00:00
|
|
|
serial_parity_handler, I(parity_mask));
|
2006-08-28 11:13:56 +00:00
|
|
|
ctrl_droplist(s, "Flow control", 'f', 40,
|
|
|
|
HELPCTX(serial_flow),
|
2006-08-28 13:08:15 +00:00
|
|
|
serial_flow_handler, I(flow_mask));
|
2006-08-28 11:13:56 +00:00
|
|
|
}
|