]> andersk Git - openssh.git/blobdiff - ssh-agent.c
- djm@cvs.openbsd.org 2001/12/21 10:06:43
[openssh.git] / ssh-agent.c
index 5a774d57029aa97d8ed6f336de0ef3e2086b2bbb..e8018bf3aacbecf1100924927ac42e60ffbaa81f 100644 (file)
@@ -1,4 +1,4 @@
-/*     $OpenBSD: ssh-agent.c,v 1.52 2001/03/06 00:33:04 deraadt Exp $  */
+/*     $OpenBSD: ssh-agent.c,v 1.75 2001/12/19 07:18:56 deraadt Exp $  */
 
 /*
  * Author: Tatu Ylonen <ylo@cs.hut.fi>
@@ -12,8 +12,7 @@
  * incompatible with the protocol description in the RFC file, it must be
  * called by a name other than "ssh" or "Secure Shell".
  *
- * SSH2 implementation,
- * Copyright (c) 2000 Markus Friedl.  All rights reserved.
+ * Copyright (c) 2000, 2001 Markus Friedl.  All rights reserved.
  *
  * Redistribution and use in source and binary forms, with or without
  * modification, are permitted provided that the following conditions
@@ -37,7 +36,7 @@
  */
 
 #include "includes.h"
-RCSID("$OpenBSD: ssh-agent.c,v 1.52 2001/03/06 00:33:04 deraadt Exp $");
+RCSID("$OpenBSD: ssh-agent.c,v 1.75 2001/12/19 07:18:56 deraadt Exp $");
 
 #include <openssl/evp.h>
 #include <openssl/md5.h>
@@ -57,11 +56,20 @@ RCSID("$OpenBSD: ssh-agent.c,v 1.52 2001/03/06 00:33:04 deraadt Exp $");
 #include "compat.h"
 #include "log.h"
 
+#ifdef SMARTCARD
+#include <openssl/engine.h>
+#include "scard.h"
+#endif
+
+typedef enum {
+       AUTH_UNUSED,
+       AUTH_SOCKET,
+       AUTH_CONNECTION
+} sock_type;
+
 typedef struct {
        int fd;
-       enum {
-               AUTH_UNUSED, AUTH_SOCKET, AUTH_CONNECTION
-       } type;
+       sock_type type;
        Buffer input;
        Buffer output;
 } SocketEntry;
@@ -97,20 +105,18 @@ extern char *__progname;
 char *__progname;
 #endif
 
-int    prepare_select(fd_set **, fd_set **, int *);
-
-void
+static void
 idtab_init(void)
 {
        int i;
-       for (i = 0; i <=2; i++){
+       for (i = 0; i <=2; i++) {
                idtable[i].identities = NULL;
                idtable[i].nentries = 0;
        }
 }
 
 /* return private key table for requested protocol version */
-Idtab *
+static Idtab *
 idtab_lookup(int version)
 {
        if (version < 1 || version > 2)
@@ -119,7 +125,7 @@ idtab_lookup(int version)
 }
 
 /* return matching private key for given public key */
-Key *
+static Key *
 lookup_private_key(Key *key, int *idx, int version)
 {
        int i;
@@ -135,7 +141,7 @@ lookup_private_key(Key *key, int *idx, int version)
 }
 
 /* send list of supported public keys to 'client' */
-void
+static void
 process_request_identities(SocketEntry *e, int version)
 {
        Idtab *tab = idtab_lookup(version);
@@ -167,7 +173,7 @@ process_request_identities(SocketEntry *e, int version)
 }
 
 /* ssh1 only */
-void
+static void
 process_authentication_challenge1(SocketEntry *e)
 {
        Key *key, *private;
@@ -233,7 +239,7 @@ send:
 }
 
 /* ssh2 only */
-void
+static void
 process_sign_request2(SocketEntry *e)
 {
        extern int datafellows;
@@ -278,7 +284,7 @@ process_sign_request2(SocketEntry *e)
 }
 
 /* shared */
-void
+static void
 process_remove_identity(SocketEntry *e, int version)
 {
        Key *key = NULL, *private;
@@ -287,7 +293,7 @@ process_remove_identity(SocketEntry *e, int version)
        u_int bits;
        int success = 0;
 
-       switch(version){
+       switch (version) {
        case 1:
                key = key_new(KEY_RSA1);
                bits = buffer_get_int(&e->input);
@@ -339,7 +345,7 @@ process_remove_identity(SocketEntry *e, int version)
            success ? SSH_AGENT_SUCCESS : SSH_AGENT_FAILURE);
 }
 
-void
+static void
 process_remove_all_identities(SocketEntry *e, int version)
 {
        u_int i;
@@ -360,26 +366,7 @@ process_remove_all_identities(SocketEntry *e, int version)
        return;
 }
 
-void
-generate_additional_parameters(RSA *rsa)
-{
-       BIGNUM *aux;
-       BN_CTX *ctx;
-       /* Generate additional parameters */
-       aux = BN_new();
-       ctx = BN_CTX_new();
-
-       BN_sub(aux, rsa->q, BN_value_one());
-       BN_mod(rsa->dmq1, rsa->d, aux, ctx);
-
-       BN_sub(aux, rsa->p, BN_value_one());
-       BN_mod(rsa->dmp1, rsa->d, aux, ctx);
-
-       BN_clear_free(aux);
-       BN_CTX_free(ctx);
-}
-
-void
+static void
 process_add_identity(SocketEntry *e, int version)
 {
        Key *k = NULL;
@@ -402,13 +389,13 @@ process_add_identity(SocketEntry *e, int version)
                buffer_get_bignum(&e->input, k->rsa->p);        /* q */
 
                /* Generate additional parameters */
-               generate_additional_parameters(k->rsa);
+               rsa_generate_additional_parameters(k->rsa);
                break;
        case 2:
                type_name = buffer_get_string(&e->input, NULL);
                type = key_type_from_name(type_name);
                xfree(type_name);
-               switch(type) {
+               switch (type) {
                case KEY_DSA:
                        k = key_new_private(type);
                        buffer_get_bignum2(&e->input, k->dsa->p);
@@ -427,7 +414,7 @@ process_add_identity(SocketEntry *e, int version)
                        buffer_get_bignum2(&e->input, k->rsa->q);
 
                        /* Generate additional parameters */
-                       generate_additional_parameters(k->rsa);
+                       rsa_generate_additional_parameters(k->rsa);
                        break;
                default:
                        buffer_clear(&e->input);
@@ -461,9 +448,116 @@ send:
            success ? SSH_AGENT_SUCCESS : SSH_AGENT_FAILURE);
 }
 
+
+#ifdef SMARTCARD
+static void
+process_add_smartcard_key (SocketEntry *e)
+{
+       Idtab *tab;
+       Key *n = NULL, *k = NULL;
+       char *sc_reader_id = NULL;
+       int success = 0;
+
+       sc_reader_id = buffer_get_string(&e->input, NULL);
+       k = sc_get_key(sc_reader_id);
+       xfree(sc_reader_id);
+
+       if (k == NULL) {
+               error("sc_get_pubkey failed");
+               goto send;
+       }
+       success = 1;
+
+       tab = idtab_lookup(1);
+       k->type = KEY_RSA1;
+       if (lookup_private_key(k, NULL, 1) == NULL) {
+               if (tab->nentries == 0)
+                       tab->identities = xmalloc(sizeof(Identity));
+               else
+                       tab->identities = xrealloc(tab->identities,
+                           (tab->nentries + 1) * sizeof(Identity));
+               n = key_new(KEY_RSA1);
+               BN_copy(n->rsa->n, k->rsa->n);
+               BN_copy(n->rsa->e, k->rsa->e);
+               RSA_set_method(n->rsa, sc_get_engine());
+               tab->identities[tab->nentries].key = n;
+               tab->identities[tab->nentries].comment =
+                   xstrdup("rsa1 smartcard");
+               tab->nentries++;
+       }
+       k->type = KEY_RSA;
+       tab = idtab_lookup(2);
+       if (lookup_private_key(k, NULL, 2) == NULL) {
+               if (tab->nentries == 0)
+                       tab->identities = xmalloc(sizeof(Identity));
+               else
+                       tab->identities = xrealloc(tab->identities,
+                           (tab->nentries + 1) * sizeof(Identity));
+               n = key_new(KEY_RSA);
+               BN_copy(n->rsa->n, k->rsa->n);
+               BN_copy(n->rsa->e, k->rsa->e);
+               RSA_set_method(n->rsa, sc_get_engine());
+               tab->identities[tab->nentries].key = n;
+               tab->identities[tab->nentries].comment =
+                   xstrdup("rsa smartcard");
+               tab->nentries++;
+       }
+       key_free(k);
+send:
+       buffer_put_int(&e->output, 1);
+       buffer_put_char(&e->output,
+           success ? SSH_AGENT_SUCCESS : SSH_AGENT_FAILURE);
+}
+
+static void
+process_remove_smartcard_key(SocketEntry *e)
+{
+       Key *k = NULL, *private;
+       int idx;
+       int success = 0;
+       char *sc_reader_id = NULL;
+
+       sc_reader_id = buffer_get_string(&e->input, NULL);
+       k = sc_get_key(sc_reader_id);
+       xfree(sc_reader_id);
+
+       if (k == NULL) {
+               error("sc_get_pubkey failed");
+       } else {
+               k->type = KEY_RSA1;
+               private = lookup_private_key(k, &idx, 1);
+               if (private != NULL) {
+                       Idtab *tab = idtab_lookup(1);
+                       key_free(tab->identities[idx].key);
+                       xfree(tab->identities[idx].comment);
+                       if (idx != tab->nentries)
+                               tab->identities[idx] = tab->identities[tab->nentries];
+                       tab->nentries--;
+                       success = 1;
+               }
+               k->type = KEY_RSA;
+               private = lookup_private_key(k, &idx, 2);
+               if (private != NULL) {
+                       Idtab *tab = idtab_lookup(2);
+                       key_free(tab->identities[idx].key);
+                       xfree(tab->identities[idx].comment);
+                       if (idx != tab->nentries)
+                               tab->identities[idx] = tab->identities[tab->nentries];
+                       tab->nentries--;
+                       success = 1;
+               }
+               key_free(k);
+       }
+
+       buffer_put_int(&e->output, 1);
+       buffer_put_char(&e->output,
+           success ? SSH_AGENT_SUCCESS : SSH_AGENT_FAILURE);
+}
+#endif /* SMARTCARD */
+
 /* dispatch incoming messages */
 
-void
+static void
 process_message(SocketEntry *e)
 {
        u_int msg_len;
@@ -484,6 +578,7 @@ process_message(SocketEntry *e)
        buffer_consume(&e->input, 4);
        type = buffer_get_char(&e->input);
 
+       debug("type %d", type);
        switch (type) {
        /* ssh1 */
        case SSH_AGENTC_RSA_CHALLENGE:
@@ -517,6 +612,14 @@ process_message(SocketEntry *e)
        case SSH2_AGENTC_REMOVE_ALL_IDENTITIES:
                process_remove_all_identities(e, 2);
                break;
+#ifdef SMARTCARD
+       case SSH_AGENTC_ADD_SMARTCARD_KEY:
+               process_add_smartcard_key(e);
+               break;
+       case SSH_AGENTC_REMOVE_SMARTCARD_KEY:
+               process_remove_smartcard_key(e);
+               break;
+#endif /* SMARTCARD */
        default:
                /* Unknown message.  Respond with failure. */
                error("Unknown message %d", type);
@@ -527,8 +630,8 @@ process_message(SocketEntry *e)
        }
 }
 
-void
-new_socket(int type, int fd)
+static void
+new_socket(sock_type type, int fd)
 {
        u_int i, old_alloc;
        if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0)
@@ -559,8 +662,8 @@ new_socket(int type, int fd)
        buffer_init(&sockets[old_alloc].output);
 }
 
-int
-prepare_select(fd_set **fdrp, fd_set **fdwp, int *fdl)
+static int
+prepare_select(fd_set **fdrp, fd_set **fdwp, int *fdl, int *nallocp)
 {
        u_int i, sz;
        int n = 0;
@@ -580,15 +683,18 @@ prepare_select(fd_set **fdrp, fd_set **fdwp, int *fdl)
        }
 
        sz = howmany(n+1, NFDBITS) * sizeof(fd_mask);
-       if (*fdrp == NULL || n > *fdl) {
+       if (*fdrp == NULL || sz > *nallocp) {
                if (*fdrp)
-                       free(*fdrp);
+                       xfree(*fdrp);
                if (*fdwp)
-                       free(*fdwp);
+                       xfree(*fdwp);
                *fdrp = xmalloc(sz);
                *fdwp = xmalloc(sz);
-               *fdl = n;
+               *nallocp = sz;
        }
+       if (n < *fdl)
+               debug("XXX shrink: %d < %d", n, *fdl);
+       *fdl = n;
        memset(*fdrp, 0, sz);
        memset(*fdwp, 0, sz);
 
@@ -607,7 +713,7 @@ prepare_select(fd_set **fdrp, fd_set **fdwp, int *fdl)
        return (1);
 }
 
-void
+static void
 after_select(fd_set *readset, fd_set *writeset)
 {
        u_int i;
@@ -679,21 +785,7 @@ after_select(fd_set *readset, fd_set *writeset)
                }
 }
 
-void
-check_parent_exists(int sig)
-{
-       int save_errno = errno;
-
-       if (parent_pid != -1 && kill(parent_pid, 0) < 0) {
-               /* printf("Parent has died - Authentication agent exiting.\n"); */
-               exit(1);
-       }
-       signal(SIGALRM, check_parent_exists);
-       alarm(10);
-       errno = save_errno;
-}
-
-void
+static void
 cleanup_socket(void)
 {
        if (socket_name[0])
@@ -702,49 +794,73 @@ cleanup_socket(void)
                rmdir(socket_dir);
 }
 
-void
+static void
 cleanup_exit(int i)
 {
        cleanup_socket();
        exit(i);
 }
 
-void
+static void
 cleanup_handler(int sig)
 {
        cleanup_socket();
        _exit(2);
 }
 
-void
+static void
+check_parent_exists(int sig)
+{
+       int save_errno = errno;
+
+       if (parent_pid != -1 && kill(parent_pid, 0) < 0) {
+               /* printf("Parent has died - Authentication agent exiting.\n"); */
+               cleanup_handler(sig); /* safe */
+       }
+       signal(SIGALRM, check_parent_exists);
+       alarm(10);
+       errno = save_errno;
+}
+
+static void
 usage(void)
 {
-       fprintf(stderr, "ssh-agent version %s\n", SSH_VERSION);
-       fprintf(stderr, "Usage: %s [-c | -s] [-k] [command {args...]]\n",
+       fprintf(stderr, "Usage: %s [options] [command [args ...]]\n",
            __progname);
+       fprintf(stderr, "Options:\n");
+       fprintf(stderr, "  -c          Generate C-shell commands on stdout.\n");
+       fprintf(stderr, "  -s          Generate Bourne shell commands on stdout.\n");
+       fprintf(stderr, "  -k          Kill the current agent.\n");
+       fprintf(stderr, "  -d          Debug mode.\n");
        exit(1);
 }
 
 int
 main(int ac, char **av)
 {
-       int sock, c_flag = 0, k_flag = 0, s_flag = 0, ch;
+       int sock, c_flag = 0, d_flag = 0, k_flag = 0, s_flag = 0, ch, nalloc;
        struct sockaddr_un sunaddr;
 #ifdef HAVE_SETRLIMIT
        struct rlimit rlim;
+#endif
+#ifdef HAVE_CYGWIN
+       int prev_mask;
 #endif
        pid_t pid;
        char *shell, *format, *pidstr, pidstrbuf[1 + 3 * sizeof pid];
        extern int optind;
        fd_set *readsetp = NULL, *writesetp = NULL;
 
+       SSLeay_add_all_algorithms();
+
        __progname = get_progname(av[0]);
        init_rng();
+       seed_rng();
 
 #ifdef __GNU_LIBRARY__
-       while ((ch = getopt(ac, av, "+cks")) != -1) {
+       while ((ch = getopt(ac, av, "+cdks")) != -1) {
 #else /* __GNU_LIBRARY__ */
-       while ((ch = getopt(ac, av, "cks")) != -1) {
+       while ((ch = getopt(ac, av, "cdks")) != -1) {
 #endif /* __GNU_LIBRARY__ */
                switch (ch) {
                case 'c':
@@ -760,6 +876,11 @@ main(int ac, char **av)
                                usage();
                        s_flag++;
                        break;
+               case 'd':
+                       if (d_flag)
+                               usage();
+                       d_flag++;
+                       break;
                default:
                        usage();
                }
@@ -767,10 +888,10 @@ main(int ac, char **av)
        ac -= optind;
        av += optind;
 
-       if (ac > 0 && (c_flag || k_flag || s_flag))
+       if (ac > 0 && (c_flag || k_flag || s_flag || d_flag))
                usage();
 
-       if (ac == 0 && !c_flag && !k_flag && !s_flag) {
+       if (ac == 0 && !c_flag && !k_flag && !s_flag && !d_flag) {
                shell = getenv("SHELL");
                if (shell != NULL && strncmp(shell + strlen(shell) - 3, "csh", 3) == 0)
                        c_flag = 1;
@@ -821,10 +942,19 @@ main(int ac, char **av)
        memset(&sunaddr, 0, sizeof(sunaddr));
        sunaddr.sun_family = AF_UNIX;
        strlcpy(sunaddr.sun_path, socket_name, sizeof(sunaddr.sun_path));
+#ifdef HAVE_CYGWIN
+       prev_mask = umask(0177);
+#endif
        if (bind(sock, (struct sockaddr *) & sunaddr, sizeof(sunaddr)) < 0) {
                perror("bind");
+#ifdef HAVE_CYGWIN
+               umask(prev_mask);
+#endif
                cleanup_exit(1);
        }
+#ifdef HAVE_CYGWIN
+       umask(prev_mask);
+#endif
        if (listen(sock, 5) < 0) {
                perror("listen");
                cleanup_exit(1);
@@ -834,6 +964,14 @@ main(int ac, char **av)
         * Fork, and have the parent execute the command, if any, or present
         * the socket data.  The child continues as the authentication agent.
         */
+       if (d_flag) {
+               log_init(__progname, SYSLOG_LEVEL_DEBUG1, SYSLOG_FACILITY_AUTH, 1);
+               format = c_flag ? "setenv %s %s;\n" : "%s=%s; export %s;\n";
+               printf(format, SSH_AUTHSOCKET_ENV_NAME, socket_name,
+                   SSH_AUTHSOCKET_ENV_NAME);
+               printf("echo Agent pid %d;\n", parent_pid);
+               goto skip;
+       }
        pid = fork();
        if (pid == -1) {
                perror("fork");
@@ -860,6 +998,13 @@ main(int ac, char **av)
                perror(av[0]);
                exit(1);
        }
+
+       if (setsid() == -1) {
+               perror("setsid");
+               cleanup_exit(1);
+       }
+
+       (void)chdir("/");
        close(0);
        close(1);
        close(2);
@@ -872,10 +1017,8 @@ main(int ac, char **av)
                cleanup_exit(1);
        }
 #endif
-       if (setsid() == -1) {
-               perror("setsid");
-               cleanup_exit(1);
-       }
+
+skip:
        if (atexit(cleanup_socket) < 0) {
                perror("atexit");
                cleanup_exit(1);
@@ -886,12 +1029,15 @@ main(int ac, char **av)
                alarm(10);
        }
        idtab_init();
-       signal(SIGINT, SIG_IGN);
+       if (!d_flag)
+               signal(SIGINT, SIG_IGN);
        signal(SIGPIPE, SIG_IGN);
        signal(SIGHUP, cleanup_handler);
        signal(SIGTERM, cleanup_handler);
+       nalloc = 0;
+
        while (1) {
-               prepare_select(&readsetp, &writesetp, &max_fd);
+               prepare_select(&readsetp, &writesetp, &max_fd, &nalloc);
                if (select(max_fd + 1, readsetp, writesetp, NULL, NULL) < 0) {
                        if (errno == EINTR)
                                continue;
This page took 0.077634 seconds and 4 git commands to generate.