1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-25 01:02:24 +00:00

Turned the old `Telnet Command' System-submenu into a more general

`Special Command' menu, in which any backend can place its own list
of magical things the user might want to ask the backend to do. In
particular I've implemented the recently proposed "break" extension
in SSH2 using this mechanism.
NB this checkin slightly breaks the Mac build, since it needs to
provide at least a stub form of update_specials_menu().

[originally from svn r3054]
This commit is contained in:
Simon Tatham 2003-04-04 20:21:05 +00:00
parent e322770ab5
commit 3bd0415579
10 changed files with 184 additions and 89 deletions

View File

@ -137,6 +137,10 @@ void verify_ssh_host_key(void *frontend, char *host, int port, char *keytype,
} }
} }
void update_specials_menu(void *frontend)
{
}
/* /*
* Ask whether the selected cipher is acceptable (since it was * Ask whether the selected cipher is acceptable (since it was
* below the configured 'warn' threshold). * below the configured 'warn' threshold).

View File

@ -134,6 +134,11 @@ typedef enum {
TS_EOL TS_EOL
} Telnet_Special; } Telnet_Special;
struct telnet_special {
const char *name; /* NULL==end, ""==separator */
int code;
};
typedef enum { typedef enum {
MBT_NOTHING, MBT_NOTHING,
MBT_LEFT, MBT_MIDDLE, MBT_RIGHT, /* `raw' button designations */ MBT_LEFT, MBT_MIDDLE, MBT_RIGHT, /* `raw' button designations */
@ -272,6 +277,7 @@ struct backend_tag {
int (*sendbuffer) (void *handle); int (*sendbuffer) (void *handle);
void (*size) (void *handle, int width, int height); void (*size) (void *handle, int width, int height);
void (*special) (void *handle, Telnet_Special code); void (*special) (void *handle, Telnet_Special code);
const struct telnet_special *(*get_specials) (void *handle);
Socket(*socket) (void *handle); Socket(*socket) (void *handle);
int (*exitcode) (void *handle); int (*exitcode) (void *handle);
int (*sendok) (void *handle); int (*sendok) (void *handle);
@ -519,6 +525,7 @@ void sys_cursor(void *frontend, int x, int y);
void request_paste(void *frontend); void request_paste(void *frontend);
void frontend_keypress(void *frontend); void frontend_keypress(void *frontend);
void ldisc_update(void *frontend, int echo, int edit); void ldisc_update(void *frontend, int echo, int edit);
void update_specials_menu(void *frontend);
#define OPTIMISE_IS_SCROLL 1 #define OPTIMISE_IS_SCROLL 1
void set_iconic(void *frontend, int iconic); void set_iconic(void *frontend, int iconic);

10
raw.c
View File

@ -181,6 +181,15 @@ static void raw_special(void *handle, Telnet_Special code)
return; return;
} }
/*
* Return a list of the special codes that make sense in this
* protocol.
*/
static const struct telnet_special *raw_get_specials(void *handle)
{
return NULL;
}
static Socket raw_socket(void *handle) static Socket raw_socket(void *handle)
{ {
Raw raw = (Raw) handle; Raw raw = (Raw) handle;
@ -233,6 +242,7 @@ Backend raw_backend = {
raw_sendbuffer, raw_sendbuffer,
raw_size, raw_size,
raw_special, raw_special,
raw_get_specials,
raw_socket, raw_socket,
raw_exitcode, raw_exitcode,
raw_sendok, raw_sendok,

View File

@ -248,6 +248,15 @@ static void rlogin_special(void *handle, Telnet_Special code)
return; return;
} }
/*
* Return a list of the special codes that make sense in this
* protocol.
*/
static const struct telnet_special *rlogin_get_specials(void *handle)
{
return NULL;
}
static Socket rlogin_socket(void *handle) static Socket rlogin_socket(void *handle)
{ {
Rlogin rlogin = (Rlogin) handle; Rlogin rlogin = (Rlogin) handle;
@ -300,6 +309,7 @@ Backend rlogin_backend = {
rlogin_sendbuffer, rlogin_sendbuffer,
rlogin_size, rlogin_size,
rlogin_special, rlogin_special,
rlogin_get_specials,
rlogin_socket, rlogin_socket,
rlogin_exitcode, rlogin_exitcode,
rlogin_sendok, rlogin_sendok,

43
ssh.c
View File

@ -1983,6 +1983,7 @@ static int do_ssh_init(Ssh ssh, unsigned char c)
ssh->version = 1; ssh->version = 1;
ssh->s_rdpkt = ssh1_rdpkt; ssh->s_rdpkt = ssh1_rdpkt;
} }
update_specials_menu(ssh->frontend);
ssh->state = SSH_STATE_BEFORE_SIZE; ssh->state = SSH_STATE_BEFORE_SIZE;
sfree(s->vstring); sfree(s->vstring);
@ -5967,6 +5968,7 @@ static char *ssh_init(void *frontend_handle, void **backend_handle,
ssh = snew(struct ssh_tag); ssh = snew(struct ssh_tag);
ssh->cfg = *cfg; /* STRUCTURE COPY */ ssh->cfg = *cfg; /* STRUCTURE COPY */
ssh->version = 0; /* when not ready yet */
ssh->s = NULL; ssh->s = NULL;
ssh->cipher = NULL; ssh->cipher = NULL;
ssh->v1_cipher_ctx = NULL; ssh->v1_cipher_ctx = NULL;
@ -6212,6 +6214,31 @@ static void ssh_size(void *handle, int width, int height)
} }
} }
/*
* Return a list of the special codes that make sense in this
* protocol.
*/
static const struct telnet_special *ssh_get_specials(void *handle)
{
Ssh ssh = (Ssh) handle;
if (ssh->version == 1) {
static const struct telnet_special ssh1_specials[] = {
{"IGNORE message", TS_NOP},
{NULL, 0}
};
return ssh1_specials;
} else if (ssh->version == 2) {
static const struct telnet_special ssh2_specials[] = {
{"Break", TS_BRK},
{"IGNORE message", TS_NOP},
{NULL, 0}
};
return ssh2_specials;
} else
return NULL;
}
/* /*
* Send Telnet special codes. TS_EOF is useful for `plink', so you * Send Telnet special codes. TS_EOF is useful for `plink', so you
* can send an EOF and collect resulting output (e.g. `plink * can send an EOF and collect resulting output (e.g. `plink
@ -6239,7 +6266,7 @@ static void ssh_special(void *handle, Telnet_Special code)
ssh2_pkt_send(ssh); ssh2_pkt_send(ssh);
} }
logevent("Sent EOF message"); logevent("Sent EOF message");
} else if (code == TS_PING) { } else if (code == TS_PING || code == TS_NOP) {
if (ssh->state == SSH_STATE_CLOSED if (ssh->state == SSH_STATE_CLOSED
|| ssh->state == SSH_STATE_PREPACKET) return; || ssh->state == SSH_STATE_PREPACKET) return;
if (ssh->version == 1) { if (ssh->version == 1) {
@ -6250,6 +6277,19 @@ static void ssh_special(void *handle, Telnet_Special code)
ssh2_pkt_addstring_start(ssh); ssh2_pkt_addstring_start(ssh);
ssh2_pkt_send(ssh); ssh2_pkt_send(ssh);
} }
} else if (code == TS_BRK) {
if (ssh->state == SSH_STATE_CLOSED
|| ssh->state == SSH_STATE_PREPACKET) return;
if (ssh->version == 1) {
logevent("Unable to send BREAK signal in SSH1");
} else {
ssh2_pkt_init(ssh, SSH2_MSG_CHANNEL_REQUEST);
ssh2_pkt_adduint32(ssh, ssh->mainchan->remoteid);
ssh2_pkt_addstring(ssh, "break");
ssh2_pkt_addbool(ssh, 0);
ssh2_pkt_adduint32(ssh, 0); /* default break length */
ssh2_pkt_send(ssh);
}
} else { } else {
/* do nothing */ /* do nothing */
} }
@ -6390,6 +6430,7 @@ Backend ssh_backend = {
ssh_sendbuffer, ssh_sendbuffer,
ssh_size, ssh_size,
ssh_special, ssh_special,
ssh_get_specials,
ssh_socket, ssh_socket,
ssh_return_exitcode, ssh_return_exitcode,
ssh_sendok, ssh_sendok,

View File

@ -955,6 +955,29 @@ static void telnet_special(void *handle, Telnet_Special code)
} }
} }
static const struct telnet_special *telnet_get_specials(void *handle)
{
static const struct telnet_special specials[] = {
{"Are You There", TS_AYT},
{"Break", TS_BRK},
{"Synch", TS_SYNCH},
{"Erase Character", TS_EC},
{"Erase Line", TS_EL},
{"Go Ahead", TS_GA},
{"No Operation", TS_NOP},
{"", 0},
{"Abort Process", TS_ABORT},
{"Abort Output", TS_AO},
{"Interrupt Process", TS_IP},
{"Suspend Process", TS_SUSP},
{"", 0},
{"End Of Record", TS_EOR},
{"End Of File", TS_EOF},
{NULL, 0}
};
return specials;
}
static Socket telnet_socket(void *handle) static Socket telnet_socket(void *handle)
{ {
Telnet telnet = (Telnet) handle; Telnet telnet = (Telnet) handle;
@ -1012,6 +1035,7 @@ Backend telnet_backend = {
telnet_sendbuffer, telnet_sendbuffer,
telnet_size, telnet_size,
telnet_special, telnet_special,
telnet_get_specials,
telnet_socket, telnet_socket,
telnet_exitcode, telnet_exitcode,
telnet_sendok, telnet_sendok,

View File

@ -148,6 +148,14 @@ void ldisc_update(void *frontend, int echo, int edit)
*/ */
} }
void update_specials_menu(void *frontend)
{
/*
* When I implement a context menu in pterm, I will need to
* support this function properly.
*/
}
int askappend(void *frontend, Filename filename) int askappend(void *frontend, Filename filename)
{ {
/* /*

View File

@ -704,6 +704,21 @@ static void pty_special(void *handle, Telnet_Special code)
return; return;
} }
/*
* Return a list of the special codes that make sense in this
* protocol.
*/
static const struct telnet_special *pty_get_specials(void *handle)
{
/*
* Hmm. When I get round to having this actually usable, it
* might be quite nice to have the ability to deliver a few
* well chosen signals to the child process - SIGINT, SIGTERM,
* SIGKILL at least.
*/
return NULL;
}
static Socket pty_socket(void *handle) static Socket pty_socket(void *handle)
{ {
return NULL; /* shouldn't ever be needed */ return NULL; /* shouldn't ever be needed */
@ -750,6 +765,7 @@ Backend pty_backend = {
pty_sendbuffer, pty_sendbuffer,
pty_size, pty_size,
pty_special, pty_special,
pty_get_specials,
pty_socket, pty_socket,
pty_exitcode, pty_exitcode,
pty_sendok, pty_sendok,

View File

@ -29,6 +29,10 @@ void cleanup_exit(int code)
exit(code); exit(code);
} }
void update_specials_menu(void *frontend)
{
}
void verify_ssh_host_key(void *frontend, char *host, int port, char *keytype, void verify_ssh_host_key(void *frontend, char *host, int port, char *keytype,
char *keystr, char *fingerprint) char *keystr, char *fingerprint)
{ {

147
window.c
View File

@ -37,19 +37,6 @@
#define IDM_RECONF 0x0040 #define IDM_RECONF 0x0040
#define IDM_CLRSB 0x0050 #define IDM_CLRSB 0x0050
#define IDM_RESET 0x0060 #define IDM_RESET 0x0060
#define IDM_TEL_AYT 0x0070
#define IDM_TEL_BRK 0x0080
#define IDM_TEL_SYNCH 0x0090
#define IDM_TEL_EC 0x00a0
#define IDM_TEL_EL 0x00b0
#define IDM_TEL_GA 0x00c0
#define IDM_TEL_NOP 0x00d0
#define IDM_TEL_ABORT 0x00e0
#define IDM_TEL_AO 0x00f0
#define IDM_TEL_IP 0x0100
#define IDM_TEL_SUSP 0x0110
#define IDM_TEL_EOR 0x0120
#define IDM_TEL_EOF 0x0130
#define IDM_HELP 0x0140 #define IDM_HELP 0x0140
#define IDM_ABOUT 0x0150 #define IDM_ABOUT 0x0150
#define IDM_SAVEDSESS 0x0160 #define IDM_SAVEDSESS 0x0160
@ -59,6 +46,10 @@
#define IDM_SESSLGP 0x0250 /* log type printable */ #define IDM_SESSLGP 0x0250 /* log type printable */
#define IDM_SESSLGA 0x0260 /* log type all chars */ #define IDM_SESSLGA 0x0260 /* log type all chars */
#define IDM_SESSLGE 0x0270 /* log end */ #define IDM_SESSLGE 0x0270 /* log end */
#define IDM_SPECIAL_MIN 0x0400
#define IDM_SPECIAL_MAX 0x0800
#define IDM_SAVED_MIN 0x1000 #define IDM_SAVED_MIN 0x1000
#define IDM_SAVED_MAX 0x2000 #define IDM_SAVED_MAX 0x2000
@ -124,6 +115,9 @@ static void *backhandle;
static struct unicode_data ucsdata; static struct unicode_data ucsdata;
static int session_closed; static int session_closed;
static const struct telnet_special *specials;
static int specials_menu_position;
Config cfg; /* exported to windlg.c */ Config cfg; /* exported to windlg.c */
extern struct sesslist sesslist; /* imported from windlg.c */ extern struct sesslist sesslist; /* imported from windlg.c */
@ -675,32 +669,12 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
*/ */
{ {
HMENU m = GetSystemMenu(hwnd, FALSE); HMENU m = GetSystemMenu(hwnd, FALSE);
HMENU p, s; HMENU s;
int i; int i;
AppendMenu(m, MF_SEPARATOR, 0, 0); AppendMenu(m, MF_SEPARATOR, 0, 0);
if (cfg.protocol == PROT_TELNET) { specials_menu_position = GetMenuItemCount(m);
p = CreateMenu(); debug(("specials_menu_position = %d\n", specials_menu_position));
AppendMenu(p, MF_ENABLED, IDM_TEL_AYT, "Are You There");
AppendMenu(p, MF_ENABLED, IDM_TEL_BRK, "Break");
AppendMenu(p, MF_ENABLED, IDM_TEL_SYNCH, "Synch");
AppendMenu(p, MF_SEPARATOR, 0, 0);
AppendMenu(p, MF_ENABLED, IDM_TEL_EC, "Erase Character");
AppendMenu(p, MF_ENABLED, IDM_TEL_EL, "Erase Line");
AppendMenu(p, MF_ENABLED, IDM_TEL_GA, "Go Ahead");
AppendMenu(p, MF_ENABLED, IDM_TEL_NOP, "No Operation");
AppendMenu(p, MF_SEPARATOR, 0, 0);
AppendMenu(p, MF_ENABLED, IDM_TEL_ABORT, "Abort Process");
AppendMenu(p, MF_ENABLED, IDM_TEL_AO, "Abort Output");
AppendMenu(p, MF_ENABLED, IDM_TEL_IP, "Interrupt Process");
AppendMenu(p, MF_ENABLED, IDM_TEL_SUSP, "Suspend Process");
AppendMenu(p, MF_SEPARATOR, 0, 0);
AppendMenu(p, MF_ENABLED, IDM_TEL_EOR, "End Of Record");
AppendMenu(p, MF_ENABLED, IDM_TEL_EOF, "End Of File");
AppendMenu(m, MF_POPUP | MF_ENABLED, (UINT) p,
"Telnet Command");
AppendMenu(m, MF_SEPARATOR, 0, 0);
}
AppendMenu(m, MF_ENABLED, IDM_SHOWLOG, "&Event Log"); AppendMenu(m, MF_ENABLED, IDM_SHOWLOG, "&Event Log");
AppendMenu(m, MF_SEPARATOR, 0, 0); AppendMenu(m, MF_SEPARATOR, 0, 0);
AppendMenu(m, MF_ENABLED, IDM_NEWSESS, "Ne&w Session..."); AppendMenu(m, MF_ENABLED, IDM_NEWSESS, "Ne&w Session...");
@ -727,6 +701,8 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
AppendMenu(m, MF_ENABLED, IDM_ABOUT, "&About PuTTY"); AppendMenu(m, MF_ENABLED, IDM_ABOUT, "&About PuTTY");
} }
update_specials_menu(NULL);
/* /*
* Set up the initial input locale. * Set up the initial input locale.
*/ */
@ -887,6 +863,37 @@ char *do_select(SOCKET skt, int startup)
return NULL; return NULL;
} }
/*
* Update the Special Commands submenu.
*/
void update_specials_menu(void *frontend)
{
HMENU m = GetSystemMenu(hwnd, FALSE);
int menu_already_exists = (specials != NULL);
int i;
specials = back->get_specials(backhandle);
if (specials) {
HMENU p = CreateMenu();
for (i = 0; specials[i].name; i++) {
assert(IDM_SPECIAL_MIN + 0x10 * i < IDM_SPECIAL_MAX);
if (*specials[i].name)
AppendMenu(p, MF_ENABLED, IDM_SPECIAL_MIN + 0x10 * i,
specials[i].name);
else
AppendMenu(p, MF_SEPARATOR, 0, 0);
}
if (menu_already_exists)
DeleteMenu(m, specials_menu_position, MF_BYPOSITION);
else
InsertMenu(m, specials_menu_position,
MF_BYPOSITION | MF_SEPARATOR, 0, 0);
InsertMenu(m, specials_menu_position,
MF_BYPOSITION | MF_POPUP | MF_ENABLED,
(UINT) p, "Special Command");
}
}
/* /*
* set or clear the "raw mouse message" mode * set or clear the "raw mouse message" mode
*/ */
@ -1906,58 +1913,6 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
term_pwron(term); term_pwron(term);
ldisc_send(ldisc, NULL, 0, 0); ldisc_send(ldisc, NULL, 0, 0);
break; break;
case IDM_TEL_AYT:
back->special(backhandle, TS_AYT);
net_pending_errors();
break;
case IDM_TEL_BRK:
back->special(backhandle, TS_BRK);
net_pending_errors();
break;
case IDM_TEL_SYNCH:
back->special(backhandle, TS_SYNCH);
net_pending_errors();
break;
case IDM_TEL_EC:
back->special(backhandle, TS_EC);
net_pending_errors();
break;
case IDM_TEL_EL:
back->special(backhandle, TS_EL);
net_pending_errors();
break;
case IDM_TEL_GA:
back->special(backhandle, TS_GA);
net_pending_errors();
break;
case IDM_TEL_NOP:
back->special(backhandle, TS_NOP);
net_pending_errors();
break;
case IDM_TEL_ABORT:
back->special(backhandle, TS_ABORT);
net_pending_errors();
break;
case IDM_TEL_AO:
back->special(backhandle, TS_AO);
net_pending_errors();
break;
case IDM_TEL_IP:
back->special(backhandle, TS_IP);
net_pending_errors();
break;
case IDM_TEL_SUSP:
back->special(backhandle, TS_SUSP);
net_pending_errors();
break;
case IDM_TEL_EOR:
back->special(backhandle, TS_EOR);
net_pending_errors();
break;
case IDM_TEL_EOF:
back->special(backhandle, TS_EOF);
net_pending_errors();
break;
case IDM_ABOUT: case IDM_ABOUT:
showabout(hwnd); showabout(hwnd);
break; break;
@ -1992,6 +1947,22 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
if (wParam >= IDM_SAVED_MIN && wParam <= IDM_SAVED_MAX) { if (wParam >= IDM_SAVED_MIN && wParam <= IDM_SAVED_MAX) {
SendMessage(hwnd, WM_SYSCOMMAND, IDM_SAVEDSESS, wParam); SendMessage(hwnd, WM_SYSCOMMAND, IDM_SAVEDSESS, wParam);
} }
if (wParam >= IDM_SPECIAL_MIN && wParam <= IDM_SPECIAL_MAX) {
int i = (wParam - IDM_SPECIAL_MIN) / 0x10;
int j;
/*
* Ensure we haven't been sent a bogus SYSCOMMAND
* which would cause us to reference invalid memory
* and crash. Perhaps I'm just too paranoid here.
*/
for (j = 0; j < i; j++)
if (!specials || !specials[j].name)
break;
if (j == i) {
back->special(backhandle, specials[i].code);
net_pending_errors();
}
}
} }
break; break;