mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 09:12:24 +00:00
b4f17f26e3
This is only intended for use in Unix Pageant; for any application that's actually trying to get something else useful done at the same time as the agent request is pending, it's much more sensible to use the more rigorous existing approach of requesting a callback once the agent request is answered. Adding this mode is the easiest way to allow Unix Pageant's command-line key loading to work, but it doesn't solve the underlying problem that the supposedly cross-platform pageant_add_keyfile will not work on a platform where we really _are_ constrained to do agent requests asynchronously (perhaps because we're a GUI app in some system that doesn't let us control our own top-level event loop). If and when that situation arises, I'll have no choice but to turn pageant_add_keyfile and friends (specifically, any function in pageant.c that calls agent_query) into coroutine-structured functions, and have clients call them repeatedly until they return 'finished'. But for now, this is a lot easier!
203 lines
4.9 KiB
C
203 lines
4.9 KiB
C
/*
|
|
* SSH agent client code.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <assert.h>
|
|
#include <unistd.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
#include <fcntl.h>
|
|
|
|
#include "putty.h"
|
|
#include "misc.h"
|
|
#include "tree234.h"
|
|
#include "puttymem.h"
|
|
|
|
int agent_exists(void)
|
|
{
|
|
const char *p = getenv("SSH_AUTH_SOCK");
|
|
if (p && *p)
|
|
return TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
static tree234 *agent_connections;
|
|
struct agent_connection {
|
|
int fd;
|
|
char *retbuf;
|
|
char sizebuf[4];
|
|
int retsize, retlen;
|
|
void (*callback)(void *, void *, int);
|
|
void *callback_ctx;
|
|
};
|
|
static int agent_conncmp(void *av, void *bv)
|
|
{
|
|
struct agent_connection *a = (struct agent_connection *) av;
|
|
struct agent_connection *b = (struct agent_connection *) bv;
|
|
if (a->fd < b->fd)
|
|
return -1;
|
|
if (a->fd > b->fd)
|
|
return +1;
|
|
return 0;
|
|
}
|
|
static int agent_connfind(void *av, void *bv)
|
|
{
|
|
int afd = *(int *) av;
|
|
struct agent_connection *b = (struct agent_connection *) bv;
|
|
if (afd < b->fd)
|
|
return -1;
|
|
if (afd > b->fd)
|
|
return +1;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Attempt to read from an agent socket fd. Returns 0 if the expected
|
|
* response is as yet incomplete; returns 1 if it's either complete
|
|
* (conn->retbuf non-NULL and filled with something useful) or has
|
|
* failed totally (conn->retbuf is NULL).
|
|
*/
|
|
static int agent_try_read(struct agent_connection *conn)
|
|
{
|
|
int ret;
|
|
|
|
ret = read(conn->fd, conn->retbuf+conn->retlen,
|
|
conn->retsize-conn->retlen);
|
|
if (ret <= 0) {
|
|
if (conn->retbuf != conn->sizebuf) sfree(conn->retbuf);
|
|
conn->retbuf = NULL;
|
|
conn->retlen = 0;
|
|
return 1;
|
|
}
|
|
conn->retlen += ret;
|
|
if (conn->retsize == 4 && conn->retlen == 4) {
|
|
conn->retsize = toint(GET_32BIT(conn->retbuf) + 4);
|
|
if (conn->retsize <= 0) {
|
|
conn->retbuf = NULL;
|
|
conn->retlen = 0;
|
|
return -1; /* way too large */
|
|
}
|
|
assert(conn->retbuf == conn->sizebuf);
|
|
conn->retbuf = snewn(conn->retsize, char);
|
|
memcpy(conn->retbuf, conn->sizebuf, 4);
|
|
}
|
|
|
|
if (conn->retlen < conn->retsize)
|
|
return 0; /* more data to come */
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int agent_select_result(int fd, int event)
|
|
{
|
|
struct agent_connection *conn;
|
|
|
|
assert(event == 1); /* not selecting for anything but R */
|
|
|
|
conn = find234(agent_connections, &fd, agent_connfind);
|
|
if (!conn) {
|
|
uxsel_del(fd);
|
|
return 1;
|
|
}
|
|
|
|
if (!agent_try_read(conn))
|
|
return 0; /* more data to come */
|
|
|
|
/*
|
|
* We have now completed the agent query. Do the callback, and
|
|
* clean up. (Of course we don't free retbuf, since ownership
|
|
* of that passes to the callback.)
|
|
*/
|
|
conn->callback(conn->callback_ctx, conn->retbuf, conn->retlen);
|
|
uxsel_del(fd);
|
|
close(fd);
|
|
del234(agent_connections, conn);
|
|
sfree(conn);
|
|
return 0;
|
|
}
|
|
|
|
int agent_query(void *in, int inlen, void **out, int *outlen,
|
|
void (*callback)(void *, void *, int), void *callback_ctx)
|
|
{
|
|
char *name;
|
|
int sock;
|
|
struct sockaddr_un addr;
|
|
int done;
|
|
struct agent_connection *conn;
|
|
|
|
name = getenv("SSH_AUTH_SOCK");
|
|
if (!name)
|
|
goto failure;
|
|
|
|
sock = socket(PF_UNIX, SOCK_STREAM, 0);
|
|
if (sock < 0) {
|
|
perror("socket(PF_UNIX)");
|
|
exit(1);
|
|
}
|
|
|
|
cloexec(sock);
|
|
|
|
addr.sun_family = AF_UNIX;
|
|
strncpy(addr.sun_path, name, sizeof(addr.sun_path));
|
|
if (connect(sock, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
|
|
close(sock);
|
|
goto failure;
|
|
}
|
|
|
|
for (done = 0; done < inlen ;) {
|
|
int ret = write(sock, (char *)in + done, inlen - done);
|
|
if (ret <= 0) {
|
|
close(sock);
|
|
goto failure;
|
|
}
|
|
done += ret;
|
|
}
|
|
|
|
conn = snew(struct agent_connection);
|
|
conn->fd = sock;
|
|
conn->retbuf = conn->sizebuf;
|
|
conn->retsize = 4;
|
|
conn->retlen = 0;
|
|
conn->callback = callback;
|
|
conn->callback_ctx = callback_ctx;
|
|
|
|
if (!callback) {
|
|
/*
|
|
* Bodge to permit making deliberately synchronous agent
|
|
* requests. Used by Unix Pageant in command-line client mode,
|
|
* which is legit because it really is true that no other part
|
|
* of the program is trying to get anything useful done
|
|
* simultaneously. But this special case shouldn't be used in
|
|
* any more general program.
|
|
*/
|
|
no_nonblock(conn->fd);
|
|
while (!agent_try_read(conn))
|
|
/* empty loop body */;
|
|
|
|
*out = conn->retbuf;
|
|
*outlen = conn->retlen;
|
|
sfree(conn);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Otherwise do it properly: add conn to the tree of agent
|
|
* connections currently in flight, return 0 to indicate that the
|
|
* response hasn't been received yet, and call the callback when
|
|
* select_result comes back to us.
|
|
*/
|
|
if (!agent_connections)
|
|
agent_connections = newtree234(agent_conncmp);
|
|
add234(agent_connections, conn);
|
|
|
|
uxsel_set(sock, 1, agent_select_result);
|
|
return 0;
|
|
|
|
failure:
|
|
*out = NULL;
|
|
*outlen = 0;
|
|
return 1;
|
|
}
|