mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 09:58:01 +00:00
c5895ec292
This is another cleanup I felt a need for while I was doing boolification. If you define a function or variable in one .c file and declare it extern in another, then nothing will check you haven't got the types of the two declarations mismatched - so when you're _changing_ the type, it's a pain to make sure you've caught all the copies of it. It's better to put all those extern declarations in header files, so that the declaration in the header is also in scope for the definition. Then the compiler will complain if they don't match, which is what I want.
205 lines
5.8 KiB
C
205 lines
5.8 KiB
C
/*
|
|
* 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, dlgparam *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, dlgparam *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, bool midsession,
|
|
int parity_mask, int flow_mask)
|
|
{
|
|
struct controlset *s;
|
|
union control *c;
|
|
|
|
if (!midsession) {
|
|
int i;
|
|
|
|
/*
|
|
* 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));
|
|
}
|