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

Pageant: new asynchronous internal APIs.

This is a pure refactoring: no functional change expected.

This commit introduces two new small vtable-style APIs. One is
PageantClient, which identifies a particular client of the Pageant
'core' (meaning the code that handles each individual request). This
changes pageant_handle_msg into an asynchronous operation: you pass in
an agent request message and an identifier, and at some later point,
the got_response method in your PageantClient will be called with the
answer (and the same identifier, to allow you to match requests to
responses). The trait vtable also contains a logging system.

The main importance of PageantClient, and the reason why it has to
exist instead of just passing pageant_handle_msg a bare callback
function pointer and context parameter, is that it provides robustness
if a client stops existing while a request is still pending. You call
pageant_unregister_client, and any unfinished requests associated with
that client in the Pageant core will be cleaned up, so that you're
guaranteed that after the unregister operation, no stray callbacks
will happen with a stale pointer to that client.

The WM_COPYDATA interface of Windows Pageant is a direct client of
this API. The other client is PageantListener, the system that lives
in pageant.c and handles stream-based agent connections for both Unix
Pageant and the new Windows named-pipe IPC. More specifically, each
individual connection to the listening socket is a separate
PageantClient, which means that if a socket is closed abruptly or
suffers an OS error, that client can be unregistered and any pending
requests cancelled without disrupting other connections.

Users of PageantListener have a second client vtable they can use,
called PageantListenerClient. That contains _only_ logging facilities,
and at the moment, only Unix Pageant bothers to use it (and even that
only in debugging mode).

Finally, internally to the Pageant core, there's a new trait called
PageantAsyncOp which describes an agent request in the process of
being handled. But at the moment, it has only one trivial
implementation, which is handed the full response message already
constructed, and on the next toplevel callback, passes it back to the
PageantClient.
This commit is contained in:
Simon Tatham 2020-01-25 17:24:28 +00:00
parent 49cd1f7116
commit de38a4d826
5 changed files with 487 additions and 214 deletions

515
pageant.c
View File

@ -31,6 +31,55 @@ static bool pageant_local = false;
typedef struct PageantKeySort PageantKeySort; typedef struct PageantKeySort PageantKeySort;
typedef struct PageantKey PageantKey; typedef struct PageantKey PageantKey;
typedef struct PageantAsyncOp PageantAsyncOp;
typedef struct PageantAsyncOpVtable PageantAsyncOpVtable;
typedef struct PageantClientRequestNode PageantClientRequestNode;
typedef struct PageantKeyRequestNode PageantKeyRequestNode;
struct PageantClientRequestNode {
PageantClientRequestNode *prev, *next;
};
struct PageantKeyRequestNode {
PageantKeyRequestNode *prev, *next;
};
struct PageantClientInfo {
PageantClient *pc; /* goes to NULL when client is unregistered */
PageantClientRequestNode head;
};
struct PageantAsyncOp {
const PageantAsyncOpVtable *vt;
PageantClientInfo *info;
PageantClientRequestNode cr;
PageantClientRequestId *reqid;
};
struct PageantAsyncOpVtable {
void (*coroutine)(PageantAsyncOp *pao);
void (*free)(PageantAsyncOp *pao);
};
static inline void pageant_async_op_coroutine(PageantAsyncOp *pao)
{ pao->vt->coroutine(pao); }
static inline void pageant_async_op_free(PageantAsyncOp *pao)
{
delete_callbacks_for_context(pao);
pao->vt->free(pao);
}
static inline void pageant_async_op_unlink(PageantAsyncOp *pao)
{
pao->cr.prev->next = pao->cr.next;
pao->cr.next->prev = pao->cr.prev;
}
static inline void pageant_async_op_unlink_and_free(PageantAsyncOp *pao)
{
pageant_async_op_unlink(pao);
pageant_async_op_free(pao);
}
static void pageant_async_op_callback(void *vctx)
{
pageant_async_op_coroutine((PageantAsyncOp *)vctx);
}
/* /*
* Master list of all the keys we have stored, in any form at all. * Master list of all the keys we have stored, in any form at all.
*/ */
@ -52,6 +101,9 @@ struct PageantKey {
}; };
}; };
static void failure(PageantClient *pc, PageantClientRequestId *reqid,
strbuf *sb, const char *fmt, ...);
static void pk_free(PageantKey *pk) static void pk_free(PageantKey *pk)
{ {
if (pk->public_blob) strbuf_free(pk->public_blob); if (pk->public_blob) strbuf_free(pk->public_blob);
@ -205,44 +257,90 @@ static void list_keys(BinarySink *bs, int ssh_version)
void pageant_make_keylist1(BinarySink *bs) { return list_keys(bs, 1); } void pageant_make_keylist1(BinarySink *bs) { return list_keys(bs, 1); }
void pageant_make_keylist2(BinarySink *bs) { return list_keys(bs, 2); } void pageant_make_keylist2(BinarySink *bs) { return list_keys(bs, 2); }
static void plog(void *logctx, pageant_logfn_t logfn, const char *fmt, ...) void pageant_register_client(PageantClient *pc)
#ifdef __GNUC__
__attribute__ ((format (PUTTY_PRINTF_ARCHETYPE, 3, 4)))
#endif
;
static void plog(void *logctx, pageant_logfn_t logfn, const char *fmt, ...)
{ {
/* pc->info = snew(PageantClientInfo);
* This is the wrapper that takes a variadic argument list and pc->info->pc = pc;
* turns it into the va_list that the log function really expects. pc->info->head.prev = pc->info->head.next = &pc->info->head;
* It's safe to call this with logfn==NULL, because we }
* double-check that below; but if you're going to do lots of work
* before getting here (such as looping, or hashing things) then void pageant_unregister_client(PageantClient *pc)
* you should probably check logfn manually before doing that. {
*/ PageantClientInfo *info = pc->info;
if (logfn) { assert(info);
assert(info->pc == pc);
while (pc->info->head.next != &pc->info->head) {
PageantAsyncOp *pao = container_of(pc->info->head.next,
PageantAsyncOp, cr);
pageant_async_op_unlink_and_free(pao);
}
sfree(pc->info);
}
static void failure(PageantClient *pc, PageantClientRequestId *reqid,
strbuf *sb, const char *fmt, ...)
{
strbuf_clear(sb);
put_byte(sb, SSH_AGENT_FAILURE);
if (!pc->suppress_logging) {
va_list ap; va_list ap;
va_start(ap, fmt); va_start(ap, fmt);
logfn(logctx, fmt, ap); char *msg = dupprintf(fmt, ap);
va_end(ap); va_end(ap);
pageant_client_log(pc, reqid, "reply: SSH_AGENT_FAILURE (%s)", msg);
sfree(msg);
} }
} }
void pageant_handle_msg(BinarySink *bs, typedef struct PageantImmOp PageantImmOp;
const void *msgdata, int msglen, struct PageantImmOp {
void *logctx, pageant_logfn_t logfn) int crLine;
strbuf *response;
PageantAsyncOp pao;
};
static void immop_free(PageantAsyncOp *pao)
{
PageantImmOp *io = container_of(pao, PageantImmOp, pao);
strbuf_free(io->response);
sfree(io);
}
static void immop_coroutine(PageantAsyncOp *pao)
{
PageantImmOp *io = container_of(pao, PageantImmOp, pao);
crBegin(io->crLine);
if (0) crReturnV;
pageant_client_got_response(io->pao.info->pc, io->pao.reqid,
ptrlen_from_strbuf(io->response));
pageant_async_op_unlink_and_free(&io->pao);
crFinishFreedV;
}
static struct PageantAsyncOpVtable immop_vtable = {
immop_coroutine,
immop_free,
};
void pageant_handle_msg(PageantClient *pc, PageantClientRequestId *reqid,
ptrlen msgpl)
{ {
BinarySource msg[1]; BinarySource msg[1];
strbuf *sb = strbuf_new_nm();
int type; int type;
BinarySource_BARE_INIT(msg, msgdata, msglen); BinarySource_BARE_INIT_PL(msg, msgpl);
type = get_byte(msg); type = get_byte(msg);
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "message contained no type code", failure(pc, reqid, sb, "message contained no type code");
logctx, logfn); goto responded;
return;
} }
switch (type) { switch (type) {
@ -251,18 +349,21 @@ void pageant_handle_msg(BinarySink *bs,
* Reply with SSH1_AGENT_RSA_IDENTITIES_ANSWER. * Reply with SSH1_AGENT_RSA_IDENTITIES_ANSWER.
*/ */
{ {
plog(logctx, logfn, "request: SSH1_AGENTC_REQUEST_RSA_IDENTITIES"); pageant_client_log(pc, reqid,
"request: SSH1_AGENTC_REQUEST_RSA_IDENTITIES");
put_byte(bs, SSH1_AGENT_RSA_IDENTITIES_ANSWER); put_byte(sb, SSH1_AGENT_RSA_IDENTITIES_ANSWER);
pageant_make_keylist1(bs); pageant_make_keylist1(BinarySink_UPCAST(sb));
plog(logctx, logfn, "reply: SSH1_AGENT_RSA_IDENTITIES_ANSWER"); pageant_client_log(pc, reqid,
if (logfn) { /* skip this loop if not logging */ "reply: SSH1_AGENT_RSA_IDENTITIES_ANSWER");
if (!pc->suppress_logging) {
int i; int i;
RSAKey *rkey; RSAKey *rkey;
for (i = 0; NULL != (rkey = pageant_nth_ssh1_key(i)); i++) { for (i = 0; NULL != (rkey = pageant_nth_ssh1_key(i)); i++) {
char *fingerprint = rsa_ssh1_fingerprint(rkey); char *fingerprint = rsa_ssh1_fingerprint(rkey);
plog(logctx, logfn, "returned key: %s", fingerprint); pageant_client_log(pc, reqid, "returned key: %s",
fingerprint);
sfree(fingerprint); sfree(fingerprint);
} }
} }
@ -273,19 +374,21 @@ void pageant_handle_msg(BinarySink *bs,
* Reply with SSH2_AGENT_IDENTITIES_ANSWER. * Reply with SSH2_AGENT_IDENTITIES_ANSWER.
*/ */
{ {
plog(logctx, logfn, "request: SSH2_AGENTC_REQUEST_IDENTITIES"); pageant_client_log(pc, reqid,
"request: SSH2_AGENTC_REQUEST_IDENTITIES");
put_byte(bs, SSH2_AGENT_IDENTITIES_ANSWER); put_byte(sb, SSH2_AGENT_IDENTITIES_ANSWER);
pageant_make_keylist2(bs); pageant_make_keylist2(BinarySink_UPCAST(sb));
plog(logctx, logfn, "reply: SSH2_AGENT_IDENTITIES_ANSWER"); pageant_client_log(pc, reqid,
if (logfn) { /* skip this loop if not logging */ "reply: SSH2_AGENT_IDENTITIES_ANSWER");
if (!pc->suppress_logging) {
int i; int i;
ssh2_userkey *skey; ssh2_userkey *skey;
for (i = 0; NULL != (skey = pageant_nth_ssh2_key(i)); i++) { for (i = 0; NULL != (skey = pageant_nth_ssh2_key(i)); i++) {
char *fingerprint = ssh2_fingerprint(skey->key); char *fingerprint = ssh2_fingerprint(skey->key);
plog(logctx, logfn, "returned key: %s %s", pageant_client_log(pc, reqid, "returned key: %s %s",
fingerprint, skey->comment); fingerprint, skey->comment);
sfree(fingerprint); sfree(fingerprint);
} }
} }
@ -306,7 +409,8 @@ void pageant_handle_msg(BinarySink *bs,
unsigned char response_md5[16]; unsigned char response_md5[16];
int i; int i;
plog(logctx, logfn, "request: SSH1_AGENTC_RSA_CHALLENGE"); pageant_client_log(pc, reqid,
"request: SSH1_AGENTC_RSA_CHALLENGE");
response = NULL; response = NULL;
memset(&reqkey, 0, sizeof(reqkey)); memset(&reqkey, 0, sizeof(reqkey));
@ -317,27 +421,26 @@ void pageant_handle_msg(BinarySink *bs,
response_type = get_uint32(msg); response_type = get_uint32(msg);
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "unable to decode request", failure(pc, reqid, sb, "unable to decode request");
logctx, logfn);
goto challenge1_cleanup; goto challenge1_cleanup;
} }
if (response_type != 1) { if (response_type != 1) {
pageant_failure_msg( failure(pc, reqid, sb,
bs, "response type other than 1 not supported", "response type other than 1 not supported");
logctx, logfn);
goto challenge1_cleanup; goto challenge1_cleanup;
} }
if (logfn) { if (!pc->suppress_logging) {
char *fingerprint; char *fingerprint;
reqkey.comment = NULL; reqkey.comment = NULL;
fingerprint = rsa_ssh1_fingerprint(&reqkey); fingerprint = rsa_ssh1_fingerprint(&reqkey);
plog(logctx, logfn, "requested key: %s", fingerprint); pageant_client_log(pc, reqid, "requested key: %s",
fingerprint);
sfree(fingerprint); sfree(fingerprint);
} }
if ((pk = findkey1(&reqkey)) == NULL) { if ((pk = findkey1(&reqkey)) == NULL) {
pageant_failure_msg(bs, "key not found", logctx, logfn); failure(pc, reqid, sb, "key not found");
goto challenge1_cleanup; goto challenge1_cleanup;
} }
response = rsa_ssh1_decrypt(challenge, pk->rkey); response = rsa_ssh1_decrypt(challenge, pk->rkey);
@ -350,10 +453,10 @@ void pageant_handle_msg(BinarySink *bs,
ssh_hash_final(h, response_md5); ssh_hash_final(h, response_md5);
} }
put_byte(bs, SSH1_AGENT_RSA_RESPONSE); put_byte(sb, SSH1_AGENT_RSA_RESPONSE);
put_data(bs, response_md5, 16); put_data(sb, response_md5, 16);
plog(logctx, logfn, "reply: SSH1_AGENT_RSA_RESPONSE"); pageant_client_log(pc, reqid, "reply: SSH1_AGENT_RSA_RESPONSE");
challenge1_cleanup: challenge1_cleanup:
if (response) if (response)
@ -374,15 +477,14 @@ void pageant_handle_msg(BinarySink *bs,
strbuf *signature; strbuf *signature;
uint32_t flags, supported_flags; uint32_t flags, supported_flags;
plog(logctx, logfn, "request: SSH2_AGENTC_SIGN_REQUEST"); pageant_client_log(pc, reqid, "request: SSH2_AGENTC_SIGN_REQUEST");
keyblob = get_string(msg); keyblob = get_string(msg);
sigdata = get_string(msg); sigdata = get_string(msg);
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "unable to decode request", failure(pc, reqid, sb, "unable to decode request");
logctx, logfn); goto responded;
return;
} }
/* /*
@ -398,20 +500,22 @@ void pageant_handle_msg(BinarySink *bs,
if (!get_err(msg)) if (!get_err(msg))
have_flags = true; have_flags = true;
if (logfn) { if (!pc->suppress_logging) {
char *fingerprint = ssh2_fingerprint_blob(keyblob); char *fingerprint = ssh2_fingerprint_blob(keyblob);
plog(logctx, logfn, "requested key: %s", fingerprint); pageant_client_log(pc, reqid, "requested key: %s",
fingerprint);
sfree(fingerprint); sfree(fingerprint);
} }
if ((pk = findkey2(keyblob)) == NULL) { if ((pk = findkey2(keyblob)) == NULL) {
pageant_failure_msg(bs, "key not found", logctx, logfn); failure(pc, reqid, sb, "key not found");
return; goto responded;
} }
if (have_flags) if (have_flags)
plog(logctx, logfn, "signature flags = 0x%08"PRIx32, flags); pageant_client_log(pc, reqid, "signature flags = 0x%08"PRIx32,
flags);
else else
plog(logctx, logfn, "no signature flags"); pageant_client_log(pc, reqid, "no signature flags");
supported_flags = ssh_key_alg(pk->skey->key)->supported_flags; supported_flags = ssh_key_alg(pk->skey->key)->supported_flags;
if (flags & ~supported_flags) { if (flags & ~supported_flags) {
@ -419,31 +523,26 @@ void pageant_handle_msg(BinarySink *bs,
* We MUST reject any message containing flags we * We MUST reject any message containing flags we
* don't understand. * don't understand.
*/ */
char *msg = dupprintf( failure(pc, reqid, sb, "unsupported flag bits 0x%08"PRIx32,
"unsupported flag bits 0x%08"PRIx32, flags & ~supported_flags);
flags & ~supported_flags); goto responded;
pageant_failure_msg(bs, msg, logctx, logfn);
sfree(msg);
return;
} }
char *invalid = ssh_key_invalid(pk->skey->key, flags); char *invalid = ssh_key_invalid(pk->skey->key, flags);
if (invalid) { if (invalid) {
char *msg = dupprintf("key invalid: %s", invalid); failure(pc, reqid, sb, "key invalid: %s", invalid);
pageant_failure_msg(bs, msg, logctx, logfn);
sfree(msg);
sfree(invalid); sfree(invalid);
return; goto responded;
} }
signature = strbuf_new(); signature = strbuf_new();
ssh_key_sign(pk->skey->key, sigdata, flags, ssh_key_sign(pk->skey->key, sigdata, flags,
BinarySink_UPCAST(signature)); BinarySink_UPCAST(signature));
put_byte(bs, SSH2_AGENT_SIGN_RESPONSE); put_byte(sb, SSH2_AGENT_SIGN_RESPONSE);
put_stringsb(bs, signature); put_stringsb(sb, signature);
plog(logctx, logfn, "reply: SSH2_AGENT_SIGN_RESPONSE"); pageant_client_log(pc, reqid, "reply: SSH2_AGENT_SIGN_RESPONSE");
} }
break; break;
case SSH1_AGENTC_ADD_RSA_IDENTITY: case SSH1_AGENTC_ADD_RSA_IDENTITY:
@ -454,36 +553,36 @@ void pageant_handle_msg(BinarySink *bs,
{ {
RSAKey *key; RSAKey *key;
plog(logctx, logfn, "request: SSH1_AGENTC_ADD_RSA_IDENTITY"); pageant_client_log(pc, reqid,
"request: SSH1_AGENTC_ADD_RSA_IDENTITY");
key = get_rsa_ssh1_priv_agent(msg); key = get_rsa_ssh1_priv_agent(msg);
key->comment = mkstr(get_string(msg)); key->comment = mkstr(get_string(msg));
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "unable to decode request", failure(pc, reqid, sb, "unable to decode request");
logctx, logfn);
goto add1_cleanup; goto add1_cleanup;
} }
if (!rsa_verify(key)) { if (!rsa_verify(key)) {
pageant_failure_msg(bs, "key is invalid", logctx, logfn); failure(pc, reqid, sb, "key is invalid");
goto add1_cleanup; goto add1_cleanup;
} }
if (logfn) { if (!pc->suppress_logging) {
char *fingerprint = rsa_ssh1_fingerprint(key); char *fingerprint = rsa_ssh1_fingerprint(key);
plog(logctx, logfn, "submitted key: %s", fingerprint); pageant_client_log(pc, reqid,
"submitted key: %s", fingerprint);
sfree(fingerprint); sfree(fingerprint);
} }
if (pageant_add_ssh1_key(key)) { if (pageant_add_ssh1_key(key)) {
keylist_update(); keylist_update();
put_byte(bs, SSH_AGENT_SUCCESS); put_byte(sb, SSH_AGENT_SUCCESS);
plog(logctx, logfn, "reply: SSH_AGENT_SUCCESS"); pageant_client_log(pc, reqid, "reply: SSH_AGENT_SUCCESS");
key = NULL; /* don't free it in cleanup */ key = NULL; /* don't free it in cleanup */
} else { } else {
pageant_failure_msg(bs, "key already present", failure(pc, reqid, sb, "key already present");
logctx, logfn);
} }
add1_cleanup: add1_cleanup:
@ -503,7 +602,7 @@ void pageant_handle_msg(BinarySink *bs,
ptrlen algpl; ptrlen algpl;
const ssh_keyalg *alg; const ssh_keyalg *alg;
plog(logctx, logfn, "request: SSH2_AGENTC_ADD_IDENTITY"); pageant_client_log(pc, reqid, "request: SSH2_AGENTC_ADD_IDENTITY");
algpl = get_string(msg); algpl = get_string(msg);
@ -512,42 +611,40 @@ void pageant_handle_msg(BinarySink *bs,
key->comment = NULL; key->comment = NULL;
alg = find_pubkey_alg_len(algpl); alg = find_pubkey_alg_len(algpl);
if (!alg) { if (!alg) {
pageant_failure_msg(bs, "algorithm unknown", logctx, logfn); failure(pc, reqid, sb, "algorithm unknown");
goto add2_cleanup; goto add2_cleanup;
} }
key->key = ssh_key_new_priv_openssh(alg, msg); key->key = ssh_key_new_priv_openssh(alg, msg);
if (!key->key) { if (!key->key) {
pageant_failure_msg(bs, "key setup failed", logctx, logfn); failure(pc, reqid, sb, "key setup failed");
goto add2_cleanup; goto add2_cleanup;
} }
key->comment = mkstr(get_string(msg)); key->comment = mkstr(get_string(msg));
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "unable to decode request", failure(pc, reqid, sb, "unable to decode request");
logctx, logfn);
goto add2_cleanup; goto add2_cleanup;
} }
if (logfn) { if (!pc->suppress_logging) {
char *fingerprint = ssh2_fingerprint(key->key); char *fingerprint = ssh2_fingerprint(key->key);
plog(logctx, logfn, "submitted key: %s %s", pageant_client_log(pc, reqid, "submitted key: %s %s",
fingerprint, key->comment); fingerprint, key->comment);
sfree(fingerprint); sfree(fingerprint);
} }
if (pageant_add_ssh2_key(key)) { if (pageant_add_ssh2_key(key)) {
keylist_update(); keylist_update();
put_byte(bs, SSH_AGENT_SUCCESS); put_byte(sb, SSH_AGENT_SUCCESS);
plog(logctx, logfn, "reply: SSH_AGENT_SUCCESS"); pageant_client_log(pc, reqid, "reply: SSH_AGENT_SUCCESS");
key = NULL; /* don't clean it up */ key = NULL; /* don't clean it up */
} else { } else {
pageant_failure_msg(bs, "key already present", failure(pc, reqid, sb, "key already present");
logctx, logfn);
} }
add2_cleanup: add2_cleanup:
@ -570,40 +667,40 @@ void pageant_handle_msg(BinarySink *bs,
RSAKey reqkey; RSAKey reqkey;
PageantKey *pk; PageantKey *pk;
plog(logctx, logfn, "request: SSH1_AGENTC_REMOVE_RSA_IDENTITY"); pageant_client_log(pc, reqid,
"request: SSH1_AGENTC_REMOVE_RSA_IDENTITY");
memset(&reqkey, 0, sizeof(reqkey)); memset(&reqkey, 0, sizeof(reqkey));
get_rsa_ssh1_pub(msg, &reqkey, RSA_SSH1_EXPONENT_FIRST); get_rsa_ssh1_pub(msg, &reqkey, RSA_SSH1_EXPONENT_FIRST);
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "unable to decode request", failure(pc, reqid, sb, "unable to decode request");
logctx, logfn);
freersakey(&reqkey); freersakey(&reqkey);
return; goto responded;
} }
if (logfn) { if (!pc->suppress_logging) {
char *fingerprint; char *fingerprint;
reqkey.comment = NULL; reqkey.comment = NULL;
fingerprint = rsa_ssh1_fingerprint(&reqkey); fingerprint = rsa_ssh1_fingerprint(&reqkey);
plog(logctx, logfn, "unwanted key: %s", fingerprint); pageant_client_log(pc, reqid, "unwanted key: %s", fingerprint);
sfree(fingerprint); sfree(fingerprint);
} }
pk = findkey1(&reqkey); pk = findkey1(&reqkey);
freersakey(&reqkey); freersakey(&reqkey);
if (pk) { if (pk) {
plog(logctx, logfn, "found with comment: %s", pageant_client_log(pc, reqid, "found with comment: %s",
pk->rkey->comment); pk->rkey->comment);
del234(keytree, pk); del234(keytree, pk);
keylist_update(); keylist_update();
pk_free(pk); pk_free(pk);
put_byte(bs, SSH_AGENT_SUCCESS); put_byte(sb, SSH_AGENT_SUCCESS);
plog(logctx, logfn, "reply: SSH_AGENT_SUCCESS"); pageant_client_log(pc, reqid, "reply: SSH_AGENT_SUCCESS");
} else { } else {
pageant_failure_msg(bs, "key not found", logctx, logfn); failure(pc, reqid, sb, "key not found");
} }
} }
break; break;
@ -617,36 +714,37 @@ void pageant_handle_msg(BinarySink *bs,
PageantKey *pk; PageantKey *pk;
ptrlen blob; ptrlen blob;
plog(logctx, logfn, "request: SSH2_AGENTC_REMOVE_IDENTITY"); pageant_client_log(pc, reqid,
"request: SSH2_AGENTC_REMOVE_IDENTITY");
blob = get_string(msg); blob = get_string(msg);
if (get_err(msg)) { if (get_err(msg)) {
pageant_failure_msg(bs, "unable to decode request", failure(pc, reqid, sb, "unable to decode request");
logctx, logfn); goto responded;
return;
} }
if (logfn) { if (!pc->suppress_logging) {
char *fingerprint = ssh2_fingerprint_blob(blob); char *fingerprint = ssh2_fingerprint_blob(blob);
plog(logctx, logfn, "unwanted key: %s", fingerprint); pageant_client_log(pc, reqid, "unwanted key: %s", fingerprint);
sfree(fingerprint); sfree(fingerprint);
} }
pk = findkey2(blob); pk = findkey2(blob);
if (!pk) { if (!pk) {
pageant_failure_msg(bs, "key not found", logctx, logfn); failure(pc, reqid, sb, "key not found");
return; goto responded;
} }
plog(logctx, logfn, "found with comment: %s", pk->skey->comment); pageant_client_log(pc, reqid,
"found with comment: %s", pk->skey->comment);
del234(keytree, pk); del234(keytree, pk);
keylist_update(); keylist_update();
pk_free(pk); pk_free(pk);
put_byte(bs, SSH_AGENT_SUCCESS); put_byte(sb, SSH_AGENT_SUCCESS);
plog(logctx, logfn, "reply: SSH_AGENT_SUCCESS"); pageant_client_log(pc, reqid, "reply: SSH_AGENT_SUCCESS");
} }
break; break;
case SSH1_AGENTC_REMOVE_ALL_RSA_IDENTITIES: case SSH1_AGENTC_REMOVE_ALL_RSA_IDENTITIES:
@ -654,15 +752,15 @@ void pageant_handle_msg(BinarySink *bs,
* Remove all SSH-1 keys. Always returns success. * Remove all SSH-1 keys. Always returns success.
*/ */
{ {
plog(logctx, logfn, "request:" pageant_client_log(pc, reqid, "request:"
" SSH1_AGENTC_REMOVE_ALL_RSA_IDENTITIES"); " SSH1_AGENTC_REMOVE_ALL_RSA_IDENTITIES");
remove_all_keys(1); remove_all_keys(1);
keylist_update(); keylist_update();
put_byte(bs, SSH_AGENT_SUCCESS); put_byte(sb, SSH_AGENT_SUCCESS);
plog(logctx, logfn, "reply: SSH_AGENT_SUCCESS"); pageant_client_log(pc, reqid, "reply: SSH_AGENT_SUCCESS");
} }
break; break;
case SSH2_AGENTC_REMOVE_ALL_IDENTITIES: case SSH2_AGENTC_REMOVE_ALL_IDENTITIES:
@ -670,29 +768,35 @@ void pageant_handle_msg(BinarySink *bs,
* Remove all SSH-2 keys. Always returns success. * Remove all SSH-2 keys. Always returns success.
*/ */
{ {
plog(logctx, logfn, "request: SSH2_AGENTC_REMOVE_ALL_IDENTITIES"); pageant_client_log(pc, reqid,
"request: SSH2_AGENTC_REMOVE_ALL_IDENTITIES");
remove_all_keys(2); remove_all_keys(2);
keylist_update(); keylist_update();
put_byte(bs, SSH_AGENT_SUCCESS); put_byte(sb, SSH_AGENT_SUCCESS);
plog(logctx, logfn, "reply: SSH_AGENT_SUCCESS"); pageant_client_log(pc, reqid, "reply: SSH_AGENT_SUCCESS");
} }
break; break;
default: default:
plog(logctx, logfn, "request: unknown message type %d", type); pageant_client_log(pc, reqid, "request: unknown message type %d",
pageant_failure_msg(bs, "unrecognised message", logctx, logfn); type);
failure(pc, reqid, sb, "unrecognised message");
break; break;
} }
}
void pageant_failure_msg(BinarySink *bs, responded:;
const char *log_reason,
void *logctx, pageant_logfn_t logfn) PageantImmOp *io = snew(PageantImmOp);
{ io->pao.vt = &immop_vtable;
put_byte(bs, SSH_AGENT_FAILURE); io->pao.info = pc->info;
plog(logctx, logfn, "reply: SSH_AGENT_FAILURE (%s)", log_reason); io->pao.cr.prev = pc->info->head.prev;
io->pao.cr.next = &pc->info->head;
io->pao.reqid = reqid;
io->response = sb;
io->crLine = 0;
queue_toplevel_callback(pageant_async_op_callback, &io->pao);
} }
void pageant_init(void) void pageant_init(void)
@ -764,15 +868,26 @@ bool pageant_delete_ssh2_key(ssh2_userkey *skey)
(c) = (unsigned char)*data++; \ (c) = (unsigned char)*data++; \
} while (0) } while (0)
struct pageant_conn_queued_response {
struct pageant_conn_queued_response *next, *prev;
size_t req_index; /* for indexing requests in log messages */
strbuf *sb;
PageantClientRequestId reqid;
};
struct pageant_conn_state { struct pageant_conn_state {
Socket *connsock; Socket *connsock;
void *logctx; PageantListenerClient *plc;
pageant_logfn_t logfn;
unsigned char lenbuf[4], pktbuf[AGENT_MAX_MSGLEN]; unsigned char lenbuf[4], pktbuf[AGENT_MAX_MSGLEN];
unsigned len, got; unsigned len, got;
bool real_packet; bool real_packet;
size_t conn_index; /* for indexing connections in log messages */
size_t req_index; /* for indexing requests in log messages */
int crLine; /* for coroutine in pageant_conn_receive */ int crLine; /* for coroutine in pageant_conn_receive */
struct pageant_conn_queued_response response_queue;
PageantClient pc;
Plug plug; Plug plug;
}; };
@ -782,10 +897,13 @@ static void pageant_conn_closing(Plug *plug, const char *error_msg,
struct pageant_conn_state *pc = container_of( struct pageant_conn_state *pc = container_of(
plug, struct pageant_conn_state, plug); plug, struct pageant_conn_state, plug);
if (error_msg) if (error_msg)
plog(pc->logctx, pc->logfn, "%p: error: %s", pc, error_msg); pageant_listener_client_log(pc->plc, "c#%zu: error: %s",
pc->conn_index, error_msg);
else else
plog(pc->logctx, pc->logfn, "%p: connection closed", pc); pageant_listener_client_log(pc->plc, "c#%zu: connection closed",
pc->conn_index);
sk_close(pc->connsock); sk_close(pc->connsock);
pageant_unregister_client(&pc->pc);
sfree(pc); sfree(pc);
} }
@ -802,15 +920,47 @@ static void pageant_conn_sent(Plug *plug, size_t bufsize)
*/ */
} }
static void pageant_conn_log(void *logctx, const char *fmt, va_list ap) static void pageant_conn_log(PageantClient *pc, PageantClientRequestId *reqid,
const char *fmt, va_list ap)
{ {
/* Wrapper on pc->logfn that prefixes the connection identifier */ struct pageant_conn_state *pcs =
struct pageant_conn_state *pc = (struct pageant_conn_state *)logctx; container_of(pc, struct pageant_conn_state, pc);
struct pageant_conn_queued_response *qr =
container_of(reqid, struct pageant_conn_queued_response, reqid);
char *formatted = dupvprintf(fmt, ap); char *formatted = dupvprintf(fmt, ap);
plog(pc->logctx, pc->logfn, "%p: %s", pc, formatted); pageant_listener_client_log(pcs->plc, "c#%zu,r#%zu: %s",
pcs->conn_index, qr->req_index, formatted);
sfree(formatted); sfree(formatted);
} }
static void pageant_conn_got_response(
PageantClient *pc, PageantClientRequestId *reqid, ptrlen response)
{
struct pageant_conn_state *pcs =
container_of(pc, struct pageant_conn_state, pc);
struct pageant_conn_queued_response *qr =
container_of(reqid, struct pageant_conn_queued_response, reqid);
qr->sb = strbuf_new_nm();
put_stringpl(qr->sb, response);
while (pcs->response_queue.next != &pcs->response_queue &&
pcs->response_queue.next->sb) {
qr = pcs->response_queue.next;
sk_write(pcs->connsock, qr->sb->u, qr->sb->len);
qr->next->prev = qr->prev;
qr->prev->next = qr->next;
strbuf_free(qr->sb);
sfree(qr);
}
}
static const struct PageantClientVtable pageant_connection_clientvt = {
pageant_conn_log,
pageant_conn_got_response,
};
static void pageant_conn_receive( static void pageant_conn_receive(
Plug *plug, int urgent, const char *data, size_t len) Plug *plug, int urgent, const char *data, size_t len)
{ {
@ -831,6 +981,31 @@ static void pageant_conn_receive(
pc->got = 0; pc->got = 0;
pc->real_packet = (pc->len < AGENT_MAX_MSGLEN-4); pc->real_packet = (pc->len < AGENT_MAX_MSGLEN-4);
{
struct pageant_conn_queued_response *qr =
snew(struct pageant_conn_queued_response);
qr->prev = pc->response_queue.prev;
qr->next = &pc->response_queue;
qr->prev->next = qr->next->prev = qr;
qr->sb = NULL;
qr->req_index = pc->req_index++;
}
if (!pc->real_packet) {
/*
* Send failure immediately, before consuming the packet
* data. That way we notify the client reasonably early
* even if the data channel has just started spewing
* nonsense.
*/
pageant_client_log(&pc->pc, &pc->response_queue.prev->reqid,
"early reply: SSH_AGENT_FAILURE "
"(overlong message, length %u)", pc->len);
static const unsigned char failure[] = { SSH_AGENT_FAILURE };
pageant_conn_got_response(&pc->pc, &pc->response_queue.prev->reqid,
make_ptrlen(failure, lenof(failure)));
}
while (pc->got < pc->len) { while (pc->got < pc->len) {
crGetChar(c); crGetChar(c);
if (pc->real_packet) if (pc->real_packet)
@ -838,26 +1013,9 @@ static void pageant_conn_receive(
pc->got++; pc->got++;
} }
{ if (pc->real_packet)
strbuf *reply = strbuf_new(); pageant_handle_msg(&pc->pc, &pc->response_queue.prev->reqid,
make_ptrlen(pc->pktbuf, pc->len));
put_uint32(reply, 0); /* length field to fill in later */
if (pc->real_packet) {
pageant_handle_msg(BinarySink_UPCAST(reply), pc->pktbuf, pc->len, pc,
pc->logfn ? pageant_conn_log : NULL);
} else {
plog(pc->logctx, pc->logfn, "%p: overlong message (%u)",
pc, pc->len);
pageant_failure_msg(BinarySink_UPCAST(reply), "message too long", pc,
pc->logfn ? pageant_conn_log : NULL);
}
PUT_32BIT_MSB_FIRST(reply->s, reply->len - 4);
sk_write(pc->connsock, reply->s, reply->len);
strbuf_free(reply);
}
} }
crFinishV; crFinishV;
@ -865,8 +1023,8 @@ static void pageant_conn_receive(
struct pageant_listen_state { struct pageant_listen_state {
Socket *listensock; Socket *listensock;
void *logctx; PageantListenerClient *plc;
pageant_logfn_t logfn; size_t conn_index; /* for indexing connections in log messages */
Plug plug; Plug plug;
}; };
@ -877,7 +1035,8 @@ static void pageant_listen_closing(Plug *plug, const char *error_msg,
struct pageant_listen_state *pl = container_of( struct pageant_listen_state *pl = container_of(
plug, struct pageant_listen_state, plug); plug, struct pageant_listen_state, plug);
if (error_msg) if (error_msg)
plog(pl->logctx, pl->logfn, "listening socket: error: %s", error_msg); pageant_listener_client_log(pl->plc, "listening socket: error: %s",
error_msg);
sk_close(pl->listensock); sk_close(pl->listensock);
pl->listensock = NULL; pl->listensock = NULL;
} }
@ -901,8 +1060,11 @@ static int pageant_listen_accepting(Plug *plug,
pc = snew(struct pageant_conn_state); pc = snew(struct pageant_conn_state);
pc->plug.vt = &pageant_connection_plugvt; pc->plug.vt = &pageant_connection_plugvt;
pc->logfn = pl->logfn; pc->pc.vt = &pageant_connection_clientvt;
pc->logctx = pl->logctx; pc->plc = pl->plc;
pc->response_queue.next = pc->response_queue.prev = &pc->response_queue;
pc->conn_index = pl->conn_index++;
pc->req_index = 0;
pc->crLine = 0; pc->crLine = 0;
pc->connsock = constructor(ctx, &pc->plug); pc->connsock = constructor(ctx, &pc->plug);
@ -916,13 +1078,16 @@ static int pageant_listen_accepting(Plug *plug,
peerinfo = sk_peer_info(pc->connsock); peerinfo = sk_peer_info(pc->connsock);
if (peerinfo && peerinfo->log_text) { if (peerinfo && peerinfo->log_text) {
plog(pl->logctx, pl->logfn, "%p: new connection from %s", pageant_listener_client_log(pl->plc, "c#%zu: new connection from %s",
pc, peerinfo->log_text); pc->conn_index, peerinfo->log_text);
} else { } else {
plog(pl->logctx, pl->logfn, "%p: new connection", pc); pageant_listener_client_log(pl->plc, "c#%zu: new connection",
pc->conn_index);
} }
sk_free_peer_info(peerinfo); sk_free_peer_info(peerinfo);
pageant_register_client(&pc->pc);
return 0; return 0;
} }
@ -934,13 +1099,14 @@ static const PlugVtable pageant_listener_plugvt = {
pageant_listen_accepting pageant_listen_accepting
}; };
struct pageant_listen_state *pageant_listener_new(Plug **plug) struct pageant_listen_state *pageant_listener_new(
Plug **plug, PageantListenerClient *plc)
{ {
struct pageant_listen_state *pl = snew(struct pageant_listen_state); struct pageant_listen_state *pl = snew(struct pageant_listen_state);
pl->plug.vt = &pageant_listener_plugvt; pl->plug.vt = &pageant_listener_plugvt;
pl->logctx = NULL; pl->plc = plc;
pl->logfn = NULL;
pl->listensock = NULL; pl->listensock = NULL;
pl->conn_index = 0;
*plug = &pl->plug; *plug = &pl->plug;
return pl; return pl;
} }
@ -950,13 +1116,6 @@ void pageant_listener_got_socket(struct pageant_listen_state *pl, Socket *sock)
pl->listensock = sock; pl->listensock = sock;
} }
void pageant_listener_set_logfn(struct pageant_listen_state *pl,
void *logctx, pageant_logfn_t logfn)
{
pl->logctx = logctx;
pl->logfn = logfn;
}
void pageant_listener_free(struct pageant_listen_state *pl) void pageant_listener_free(struct pageant_listen_state *pl)
{ {
if (pl->listensock) if (pl->listensock)

118
pageant.h
View File

@ -11,35 +11,85 @@
*/ */
#define AGENT_MAX_MSGLEN 262144 #define AGENT_MAX_MSGLEN 262144
typedef void (*pageant_logfn_t)(void *logctx, const char *fmt, va_list ap); typedef struct PageantClientVtable PageantClientVtable;
typedef struct PageantClient PageantClient;
typedef struct PageantClientInfo PageantClientInfo;
typedef struct PageantClientRequestId PageantClientRequestId;
struct PageantClient {
const struct PageantClientVtable *vt;
PageantClientInfo *info; /* used by the central Pageant code */
/* Setting this flag prevents the 'log' vtable entry from ever
* being called, so that it's safe to make it NULL. This also
* allows optimisations in the core code (it can avoid entire
* loops that are only used for logging purposes). So you can also
* set it dynamically if you find out at run time that you're not
* doing logging. */
bool suppress_logging;
};
struct PageantClientVtable {
void (*log)(PageantClient *pc, PageantClientRequestId *reqid,
const char *fmt, va_list ap);
void (*got_response)(PageantClient *pc, PageantClientRequestId *reqid,
ptrlen response);
};
static inline void pageant_client_log_v(
PageantClient *pc, PageantClientRequestId *reqid,
const char *fmt, va_list ap)
{
if (!pc->suppress_logging)
pc->vt->log(pc, reqid, fmt, ap);
}
static inline void pageant_client_log(
PageantClient *pc, PageantClientRequestId *reqid, const char *fmt, ...)
{
if (!pc->suppress_logging) {
va_list ap;
va_start(ap, fmt);
pc->vt->log(pc, reqid, fmt, ap);
va_end(ap);
}
}
static inline void pageant_client_got_response(
PageantClient *pc, PageantClientRequestId *reqid, ptrlen response)
{ pc->vt->got_response(pc, reqid, response); }
/* PageantClientRequestId is used to match up responses to the agent
* requests they refer to. A client may allocate one of these for each
* call to pageant_handle_request, (probably as a subfield of some
* larger struct on the client side) and expect the same pointer to be
* passed back in pageant_client_got_response. */
struct PageantClientRequestId { int unused_; };
/* /*
* Initial setup. * Initial setup.
*/ */
void pageant_init(void); void pageant_init(void);
/*
* Register and unregister PageantClients. This is necessary so that
* when a PageantClient goes away, any unfinished asynchronous
* requests can be cleaned up.
*
* pageant_register_client will fill in pc->id. The client itself
* should not touch that field.
*/
void pageant_register_client(PageantClient *pc);
void pageant_unregister_client(PageantClient *pc);
/* /*
* The main agent function that answers messages. * The main agent function that answers messages.
* *
* Expects a message/length pair as input, minus its initial length * Expects a message/length pair as input, minus its initial length
* field but still with its type code on the front. * field but still with its type code on the front.
* *
* Returns a fully formatted message as output, *with* its initial * When a response is ready, the got_response method in the
* length field, and sets *outlen to the full size of that message. * PageantClient vtable will be passed it in the form of a ptrlen,
* again minus its length field.
*/ */
void pageant_handle_msg(BinarySink *bs, void pageant_handle_msg(PageantClient *pc, PageantClientRequestId *reqid,
const void *msg, int msglen, ptrlen msg);
void *logctx, pageant_logfn_t logfn);
/*
* Construct a failure response. Useful for agent front ends which
* suffer a problem before they even get to pageant_handle_msg.
*
* 'log_reason' is only used if logfn is not NULL.
*/
void pageant_failure_msg(BinarySink *bs,
const char *log_reason,
void *logctx, pageant_logfn_t logfn);
/* /*
* Construct a list of public keys, just as the two LIST_IDENTITIES * Construct a list of public keys, just as the two LIST_IDENTITIES
@ -83,11 +133,41 @@ void keylist_update(void);
* socket pointer. Also, provide a logging function later if you want * socket pointer. Also, provide a logging function later if you want
* to. * to.
*/ */
typedef struct PageantListenerClientVtable PageantListenerClientVtable;
typedef struct PageantListenerClient PageantListenerClient;
struct PageantListenerClient {
const PageantListenerClientVtable *vt;
/* suppress_logging flag works similarly to the one in
* PageantClient, but it is only read when a new connection comes
* in. So if you do need to change it in mid-run, expect existing
* agent connections to still use the old value. */
bool suppress_logging;
};
struct PageantListenerClientVtable {
void (*log)(PageantListenerClient *, const char *fmt, va_list ap);
};
static inline void pageant_listener_client_log_v(
PageantListenerClient *plc, const char *fmt, va_list ap)
{
if (!plc->suppress_logging)
plc->vt->log(plc, fmt, ap);
}
static inline void pageant_listener_client_log(
PageantListenerClient *plc, const char *fmt, ...)
{
if (!plc->suppress_logging) {
va_list ap;
va_start(ap, fmt);
plc->vt->log(plc, fmt, ap);
va_end(ap);
}
}
struct pageant_listen_state; struct pageant_listen_state;
struct pageant_listen_state *pageant_listener_new(Plug **plug); struct pageant_listen_state *pageant_listener_new(
Plug **plug, PageantListenerClient *plc);
void pageant_listener_got_socket(struct pageant_listen_state *pl, Socket *); void pageant_listener_got_socket(struct pageant_listen_state *pl, Socket *);
void pageant_listener_set_logfn(struct pageant_listen_state *pl,
void *logctx, pageant_logfn_t logfn);
void pageant_listener_free(struct pageant_listen_state *pl); void pageant_listener_free(struct pageant_listen_state *pl);
/* /*

View File

@ -34,6 +34,8 @@
#define crState(t) crStateP(t, ssh->t) #define crState(t) crStateP(t, ssh->t)
#define crFinish(z) } *crLine = 0; return (z); } #define crFinish(z) } *crLine = 0; return (z); }
#define crFinishV } *crLine = 0; return; } #define crFinishV } *crLine = 0; return; }
#define crFinishFreed(z) } return (z); }
#define crFinishFreedV } return; }
#define crFinishFree(z) } sfree(s); return (z); } #define crFinishFree(z) } sfree(s); return (z); }
#define crFinishFreeV } sfree(s); return; } #define crFinishFreeV } sfree(s); return; }
#define crReturn(z) \ #define crReturn(z) \

View File

@ -30,7 +30,12 @@ void cmdline_error(const char *fmt, ...)
} }
FILE *pageant_logfp = NULL; FILE *pageant_logfp = NULL;
void pageant_log(void *ctx, const char *fmt, va_list ap)
struct uxpgnt_client {
PageantListenerClient plc;
};
static void uxpgnt_log(PageantListenerClient *plc, const char *fmt, va_list ap)
{ {
if (!pageant_logfp) if (!pageant_logfp)
return; return;
@ -40,6 +45,10 @@ void pageant_log(void *ctx, const char *fmt, va_list ap)
fprintf(pageant_logfp, "\n"); fprintf(pageant_logfp, "\n");
} }
const PageantListenerClientVtable uxpgnt_vtable = {
uxpgnt_log,
};
/* /*
* In Pageant our selects are synchronous, so these functions are * In Pageant our selects are synchronous, so these functions are
* empty stubs. * empty stubs.
@ -760,7 +769,9 @@ void run_agent(void)
/* /*
* Set up a listening socket and run Pageant on it. * Set up a listening socket and run Pageant on it.
*/ */
pl = pageant_listener_new(&pl_plug); struct uxpgnt_client upc[1];
upc->plc.vt = &uxpgnt_vtable;
pl = pageant_listener_new(&pl_plug, &upc->plc);
sock = platform_make_agent_socket(pl_plug, PAGEANT_DIR_PREFIX, sock = platform_make_agent_socket(pl_plug, PAGEANT_DIR_PREFIX,
&errw, &socketname); &errw, &socketname);
if (!sock) { if (!sock) {
@ -853,11 +864,8 @@ void run_agent(void)
} }
} }
/* if (!pageant_logfp)
* Now we've decided on our logging arrangements, pass them on to upc->plc.suppress_logging = true;
* pageant.c.
*/
pageant_listener_set_logfn(pl, NULL, pageant_logfp ? pageant_log : NULL);
now = GETTICKCOUNT(); now = GETTICKCOUNT();

View File

@ -769,26 +769,35 @@ struct WmCopydataTransaction {
HANDLE ev_msg_ready, ev_reply_ready; HANDLE ev_msg_ready, ev_reply_ready;
} wmct; } wmct;
static struct PageantClient wmcpc;
static void wm_copydata_got_msg(void *vctx) static void wm_copydata_got_msg(void *vctx)
{ {
strbuf *sb = strbuf_new(); pageant_handle_msg(&wmcpc, NULL, make_ptrlen(wmct.body, wmct.bodylen));
pageant_handle_msg(BinarySink_UPCAST(sb), wmct.body, wmct.bodylen, }
NULL, NULL);
if (sb->len > wmct.bodysize) { static void wm_copydata_got_response(PageantClient *pc, void *reqid,
ptrlen response)
{
if (response.len > wmct.bodysize) {
/* Output would overflow message buffer. Replace with a /* Output would overflow message buffer. Replace with a
* failure message. */ * failure message. */
sb->len = 0; static const unsigned char failure[] = { SSH_AGENT_FAILURE };
put_byte(sb, SSH_AGENT_FAILURE); response = make_ptrlen(failure, lenof(failure));
assert(sb->len <= wmct.bodysize); assert(response.len <= wmct.bodysize);
} }
PUT_32BIT_MSB_FIRST(wmct.length, sb->len); PUT_32BIT_MSB_FIRST(wmct.length, response.len);
memcpy(wmct.body, sb->u, sb->len); memcpy(wmct.body, response.ptr, response.len);
SetEvent(wmct.ev_reply_ready); SetEvent(wmct.ev_reply_ready);
} }
static const PageantClientVtable wmcpc_vtable = {
NULL, /* no logging in this client */
wm_copydata_got_response,
};
static char *answer_filemapping_message(const char *mapname) static char *answer_filemapping_message(const char *mapname)
{ {
HANDLE maphandle = INVALID_HANDLE_VALUE; HANDLE maphandle = INVALID_HANDLE_VALUE;
@ -1176,6 +1185,15 @@ void cleanup_exit(int code)
int flags = FLAG_SYNCAGENT; int flags = FLAG_SYNCAGENT;
struct winpgnt_client {
PageantListenerClient plc;
};
const PageantListenerClientVtable winpgnt_vtable = {
NULL, /* no logging */
};
static struct winpgnt_client wpc[1];
int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show) int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
{ {
MSG msg; MSG msg;
@ -1328,7 +1346,10 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
*/ */
{ {
Plug *pl_plug; Plug *pl_plug;
struct pageant_listen_state *pl = pageant_listener_new(&pl_plug); wpc->plc.vt = &winpgnt_vtable;
wpc->plc.suppress_logging = true;
struct pageant_listen_state *pl =
pageant_listener_new(&pl_plug, &wpc->plc);
char *pipename = agent_named_pipe_name(); char *pipename = agent_named_pipe_name();
Socket *sock = new_named_pipe_listener(pipename, pl_plug); Socket *sock = new_named_pipe_listener(pipename, pl_plug);
if (sk_socket_error(sock)) { if (sk_socket_error(sock)) {
@ -1406,6 +1427,9 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
ShowWindow(hwnd, SW_HIDE); ShowWindow(hwnd, SW_HIDE);
wmcpc.vt = &wmcpc_vtable;
wmcpc.suppress_logging = true;
pageant_register_client(&wmcpc);
DWORD wm_copydata_threadid; DWORD wm_copydata_threadid;
wmct.ev_msg_ready = CreateEvent(NULL, false, false, NULL); wmct.ev_msg_ready = CreateEvent(NULL, false, false, NULL);
wmct.ev_reply_ready = CreateEvent(NULL, false, false, NULL); wmct.ev_reply_ready = CreateEvent(NULL, false, false, NULL);