1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Merge the two low-level portfwd setup systems.

In commit 09954a87c I introduced the portfwdmgr_connect_socket()
system, which opened a port forwarding given a callback to create the
Socket itself, with the aim of using it to make forwardings to Unix-
domain sockets and Windows named pipes (both initially for agent
forwarding).

But I forgot that a year and a bit ago, in commit 834396170, I already
introduced a similar low-level system for creating a PortForwarding
around an unusual kind of socket: the portfwd_raw_new() system, which
in place of a callback uses a two-phase setup protocol (you create the
socket in between the two setup calls, and can roll it back if the
socket can't be created).

There's really no need to have _both_ these systems! So now I'm
merging them, which is to say, I'm enhancing portfwd_raw_new to have
the one new feature it needs, and throwing away the newer system
completely. The new feature is to be able to control the initial state
of the 'ready' flag: portfwd_raw_new was always used for initiating
port forwardings in response to an incoming local connection, which
means you need to start off with ready=false and set it true when the
other end of the SSH connection sends back OPEN_CONFIRMATION. Now it's
being used for initiating port forwardings in response to a
CHANNEL_OPEN, we need to be able to start with ready=true.

This commit reverts 09954a87c2 and its
followup fix 12aa06ccc9, and simplifies
the agent_connect system down to a single trivial function that makes
a Socket given a Plug.
This commit is contained in:
Simon Tatham 2020-01-27 19:34:15 +00:00
parent 600bf247d3
commit 2160205aee
8 changed files with 56 additions and 130 deletions

View File

@ -468,7 +468,7 @@ static const struct ChannelVtable PortForwarding_channelvt = {
chan_no_request_response, chan_no_request_response,
}; };
Channel *portfwd_raw_new(ConnectionLayer *cl, Plug **plug) Channel *portfwd_raw_new(ConnectionLayer *cl, Plug **plug, bool start_ready)
{ {
struct PortForwarding *pf; struct PortForwarding *pf;
@ -482,7 +482,7 @@ Channel *portfwd_raw_new(ConnectionLayer *cl, Plug **plug)
pf->cl = cl; pf->cl = cl;
pf->input_wanted = true; pf->input_wanted = true;
pf->ready = false; pf->ready = start_ready;
pf->socks_state = SOCKS_NONE; pf->socks_state = SOCKS_NONE;
pf->hostname = NULL; pf->hostname = NULL;
@ -523,7 +523,7 @@ static int pfl_accepting(Plug *p, accept_fn_t constructor, accept_ctx_t ctx)
Socket *s; Socket *s;
const char *err; const char *err;
chan = portfwd_raw_new(pl->cl, &plug); chan = portfwd_raw_new(pl->cl, &plug, false);
s = constructor(ctx, plug); s = constructor(ctx, plug);
if ((err = sk_socket_error(s)) != NULL) { if ((err = sk_socket_error(s)) != NULL) {
portfwd_raw_free(chan); portfwd_raw_free(chan);
@ -1120,20 +1120,6 @@ bool portfwdmgr_unlisten(PortFwdManager *mgr, const char *host, int port)
return true; return true;
} }
struct portfwdmgr_connect_ctx {
SockAddr *addr;
int port;
char *canonical_hostname;
Conf *conf;
};
static Socket *portfwdmgr_connect_helper(void *vctx, Plug *plug)
{
struct portfwdmgr_connect_ctx *ctx = (struct portfwdmgr_connect_ctx *)vctx;
return new_connection(sk_addr_dup(ctx->addr), ctx->canonical_hostname,
ctx->port, false, true, false, false, plug,
ctx->conf);
}
/* /*
* Called when receiving a PORT OPEN from the server to make a * Called when receiving a PORT OPEN from the server to make a
* connection to a destination host. * connection to a destination host.
@ -1145,39 +1131,26 @@ char *portfwdmgr_connect(PortFwdManager *mgr, Channel **chan_ret,
char *hostname, int port, SshChannel *c, char *hostname, int port, SshChannel *c,
int addressfamily) int addressfamily)
{ {
struct portfwdmgr_connect_ctx ctx[1]; SockAddr *addr;
const char *err_retd; const char *err;
char *err_toret; char *dummy_realhost = NULL;
struct PortForwarding *pf;
/* /*
* Try to find host. * Try to find host.
*/ */
ctx->addr = name_lookup(hostname, port, &ctx->canonical_hostname, addr = name_lookup(hostname, port, &dummy_realhost, mgr->conf,
mgr->conf, addressfamily, NULL, NULL); addressfamily, NULL, NULL);
if ((err_retd = sk_addr_error(ctx->addr)) != NULL) { if ((err = sk_addr_error(addr)) != NULL) {
err_toret = dupstr(err_retd); char *err_ret = dupstr(err);
goto out; sk_addr_free(addr);
sfree(dummy_realhost);
return err_ret;
} }
ctx->conf = mgr->conf; /*
ctx->port = port; * Open socket.
*/
err_toret = portfwdmgr_connect_socket(
mgr, chan_ret, portfwdmgr_connect_helper, ctx, c);
out:
sk_addr_free(ctx->addr);
sfree(ctx->canonical_hostname);
return err_toret;
}
char *portfwdmgr_connect_socket(PortFwdManager *mgr, Channel **chan_ret,
Socket *(*connect)(void *, Plug *), void *ctx,
SshChannel *c)
{
struct PortForwarding *pf;
const char *err;
pf = new_portfwd_state(); pf = new_portfwd_state();
*chan_ret = &pf->chan; *chan_ret = &pf->chan;
pf->plug.vt = &PortForwarding_plugvt; pf->plug.vt = &PortForwarding_plugvt;
@ -1189,7 +1162,9 @@ char *portfwdmgr_connect_socket(PortFwdManager *mgr, Channel **chan_ret,
pf->cl = mgr->cl; pf->cl = mgr->cl;
pf->socks_state = SOCKS_NONE; pf->socks_state = SOCKS_NONE;
pf->s = connect(ctx, &pf->plug); pf->s = new_connection(addr, dummy_realhost, port,
false, true, false, false, &pf->plug, mgr->conf);
sfree(dummy_realhost);
if ((err = sk_socket_error(pf->s)) != NULL) { if ((err = sk_socket_error(pf->s)) != NULL) {
char *err_ret = dupstr(err); char *err_ret = dupstr(err);
sk_close(pf->s); sk_close(pf->s);

11
putty.h
View File

@ -1889,15 +1889,8 @@ void agent_cancel_query(agent_pending_query *);
void agent_query_synchronous(strbuf *in, void **out, int *outlen); void agent_query_synchronous(strbuf *in, void **out, int *outlen);
bool agent_exists(void); bool agent_exists(void);
/* For stream-oriented agent connections, if available: agent_connect /* For stream-oriented agent connections, if available. */
* is a callback for use with portfwdmgr_connect_socket, and the Socket *agent_connect(Plug *plug);
* context structure it requires is created and freed by the next two
* functions. agent_get_connect_ctx may return NULL if no streaming
* agent connection is available at all on this platform. */
typedef struct agent_connect_ctx agent_connect_ctx;
Socket *agent_connect(void *vctx, Plug *plug);
agent_connect_ctx *agent_get_connect_ctx(void);
void agent_free_connect_ctx(agent_connect_ctx *ctx);
/* /*
* Exports from wildcard.c * Exports from wildcard.c

View File

@ -365,7 +365,7 @@ static int xfwd_accepting(Plug *p, accept_fn_t constructor, accept_ctx_t ctx)
SocketPeerInfo *pi; SocketPeerInfo *pi;
const char *err; const char *err;
chan = portfwd_raw_new(sess->c->cl, &plug); chan = portfwd_raw_new(sess->c->cl, &plug, false);
s = constructor(ctx, plug); s = constructor(ctx, plug);
if ((err = sk_socket_error(s)) != NULL) { if ((err = sk_socket_error(s)) != NULL) {
portfwd_raw_free(chan); portfwd_raw_free(chan);
@ -441,7 +441,7 @@ static int agentfwd_accepting(
Socket *s; Socket *s;
const char *err; const char *err;
chan = portfwd_raw_new(sess->c->cl, &plug); chan = portfwd_raw_new(sess->c->cl, &plug, false);
s = constructor(ctx, plug); s = constructor(ctx, plug);
if ((err = sk_socket_error(s)) != NULL) { if ((err = sk_socket_error(s)) != NULL) {
portfwd_raw_free(chan); portfwd_raw_free(chan);

5
ssh.h
View File

@ -380,13 +380,10 @@ void portfwdmgr_close_all(PortFwdManager *mgr);
char *portfwdmgr_connect(PortFwdManager *mgr, Channel **chan_ret, char *portfwdmgr_connect(PortFwdManager *mgr, Channel **chan_ret,
char *hostname, int port, SshChannel *c, char *hostname, int port, SshChannel *c,
int addressfamily); int addressfamily);
char *portfwdmgr_connect_socket(PortFwdManager *mgr, Channel **chan_ret,
Socket *(*connect)(void *, Plug *), void *ctx,
SshChannel *c);
bool portfwdmgr_listen(PortFwdManager *mgr, const char *host, int port, bool portfwdmgr_listen(PortFwdManager *mgr, const char *host, int port,
const char *keyhost, int keyport, Conf *conf); const char *keyhost, int keyport, Conf *conf);
bool portfwdmgr_unlisten(PortFwdManager *mgr, const char *host, int port); bool portfwdmgr_unlisten(PortFwdManager *mgr, const char *host, int port);
Channel *portfwd_raw_new(ConnectionLayer *cl, Plug **plug); Channel *portfwd_raw_new(ConnectionLayer *cl, Plug **plug, bool start_ready);
void portfwd_raw_free(Channel *pfchan); void portfwd_raw_free(Channel *pfchan);
void portfwd_raw_setup(Channel *pfchan, Socket *s, SshChannel *sc); void portfwd_raw_setup(Channel *pfchan, Socket *s, SshChannel *sc);

View File

@ -168,16 +168,15 @@ bool ssh1_handle_direction_specific_packet(
* agent and set up an ordinary port-forwarding type * agent and set up an ordinary port-forwarding type
* channel over it. * channel over it.
*/ */
agent_connect_ctx *ctx = agent_get_connect_ctx(); Plug *plug;
bool got_stream_connection = false; Channel *ch = portfwd_raw_new(&s->cl, &plug, true);
if (ctx) { Socket *skt = agent_connect(plug);
char *err = portfwdmgr_connect_socket( if (!sk_socket_error(skt)) {
s->portfwdmgr, &c->chan, agent_connect, ctx, &c->sc); portfwd_raw_setup(ch, skt, &c->sc);
agent_free_connect_ctx(ctx); c->chan = ch;
if (err == NULL) } else {
got_stream_connection = true; portfwd_raw_free(ch);
}
if (!got_stream_connection) {
/* /*
* Otherwise, fall back to the old-fashioned system of * Otherwise, fall back to the old-fashioned system of
* parsing the forwarded data stream ourselves for * parsing the forwarded data stream ourselves for

View File

@ -99,21 +99,15 @@ static ChanopenResult chan_open_auth_agent(
* If possible, make a stream-oriented connection to the agent and * If possible, make a stream-oriented connection to the agent and
* set up an ordinary port-forwarding type channel over it. * set up an ordinary port-forwarding type channel over it.
*/ */
agent_connect_ctx *ctx = agent_get_connect_ctx(); Plug *plug;
if (ctx) { Channel *ch = portfwd_raw_new(&s->cl, &plug, true);
Channel *ch; Socket *skt = agent_connect(plug);
char *err = portfwdmgr_connect_socket(
s->portfwdmgr, &ch, agent_connect, ctx, sc);
agent_free_connect_ctx(ctx);
if (err == NULL) { if (!sk_socket_error(skt)) {
portfwd_raw_setup(ch, skt, sc);
CHANOPEN_RETURN_SUCCESS(ch); CHANOPEN_RETURN_SUCCESS(ch);
} else { } else {
sfree(err); portfwd_raw_free(ch);
/* now continue to the fallback case below */
}
}
/* /*
* Otherwise, fall back to the old-fashioned system of parsing the * Otherwise, fall back to the old-fashioned system of parsing the
* forwarded data stream ourselves for message boundaries, and * forwarded data stream ourselves for message boundaries, and
@ -121,6 +115,7 @@ static ChanopenResult chan_open_auth_agent(
*/ */
CHANOPEN_RETURN_SUCCESS(agentf_new(sc)); CHANOPEN_RETURN_SUCCESS(agentf_new(sc));
} }
}
ChanopenResult ssh2_connection_parse_channel_open( ChanopenResult ssh2_connection_parse_channel_open(
struct ssh2_connection_state *s, ptrlen type, struct ssh2_connection_state *s, ptrlen type,

View File

@ -130,30 +130,12 @@ static const char *agent_socket_path(void)
return getenv("SSH_AUTH_SOCK"); return getenv("SSH_AUTH_SOCK");
} }
struct agent_connect_ctx { Socket *agent_connect(Plug *plug)
SockAddr *addr;
};
Socket *agent_connect(void *vctx, Plug *plug)
{
agent_connect_ctx *ctx = (agent_connect_ctx *)vctx;
return sk_new(sk_addr_dup(ctx->addr), 0, false, false, false, false, plug);
}
agent_connect_ctx *agent_get_connect_ctx(void)
{ {
const char *path = agent_socket_path(); const char *path = agent_socket_path();
if (!path) if (!path)
return NULL; return new_error_socket_fmt(plug, "SSH_AUTH_SOCK not set");
agent_connect_ctx *ctx = snew(agent_connect_ctx); return sk_new(unix_sock_addr(path), 0, false, false, false, false, plug);
ctx->addr = unix_sock_addr(path);
return ctx;
}
void agent_free_connect_ctx(agent_connect_ctx *ctx)
{
sk_addr_free(ctx->addr);
sfree(ctx);
} }
agent_pending_query *agent_query( agent_pending_query *agent_query(

View File

@ -142,27 +142,12 @@ char *agent_named_pipe_name(void)
return pipename; return pipename;
} }
struct agent_connect_ctx { Socket *agent_connect(Plug *plug)
char *pipename;
};
Socket *agent_connect(void *vctx, Plug *plug)
{ {
agent_connect_ctx *ctx = (agent_connect_ctx *)vctx; char *pipename = agent_named_pipe_name();
return new_named_pipe_client(ctx->pipename, plug); Socket *s = new_named_pipe_client(pipename, plug);
} sfree(pipename);
return s;
agent_connect_ctx *agent_get_connect_ctx(void)
{
agent_connect_ctx *ctx = snew(agent_connect_ctx);
ctx->pipename = agent_named_pipe_name();
return ctx;
}
void agent_free_connect_ctx(agent_connect_ctx *ctx)
{
sfree(ctx->pipename);
sfree(ctx);
} }
static bool named_pipe_agent_exists(void) static bool named_pipe_agent_exists(void)