- More OpenBSD updates:
[session.c] - don't call chan_write_failed() if we are not writing [auth-rsa.c auth1.c authfd.c hostfile.c ssh-agent.c] - keysize warnings error() -> log()
This commit is contained in:
parent
c998f9eb7c
commit
bd483e7690
|
@ -8,6 +8,11 @@
|
||||||
- Changed entropy seed code to user per-user seeds only (server seed is
|
- Changed entropy seed code to user per-user seeds only (server seed is
|
||||||
saved in root's .ssh directory)
|
saved in root's .ssh directory)
|
||||||
- Use atexit() and fatal cleanups to save seed on exit
|
- Use atexit() and fatal cleanups to save seed on exit
|
||||||
|
- More OpenBSD updates:
|
||||||
|
[session.c]
|
||||||
|
- don't call chan_write_failed() if we are not writing
|
||||||
|
[auth-rsa.c auth1.c authfd.c hostfile.c ssh-agent.c]
|
||||||
|
- keysize warnings error() -> log()
|
||||||
|
|
||||||
20000429
|
20000429
|
||||||
- Merge big update to OpenSSH-2.0 from OpenBSD CVS
|
- Merge big update to OpenSSH-2.0 from OpenBSD CVS
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
$Id: README.openssh2,v 1.6 2000/04/27 13:42:58 provos Exp $
|
$Id: README.openssh2,v 1.4 2000/04/29 13:57:09 damien Exp $
|
||||||
|
|
||||||
howto:
|
howto:
|
||||||
1) generate server key:
|
1) generate server key:
|
||||||
|
@ -6,7 +6,8 @@ howto:
|
||||||
2) enable ssh2:
|
2) enable ssh2:
|
||||||
server: add 'Protocol 2,1' to /etc/sshd_config
|
server: add 'Protocol 2,1' to /etc/sshd_config
|
||||||
client: ssh -o 'Protocol 2,1', or add to .ssh/config
|
client: ssh -o 'Protocol 2,1', or add to .ssh/config
|
||||||
3) interop w/ ssh.com dsa-keys:
|
3) DSA authentication similar to RSA (add keys to ~/.ssh/authorized_keys2)
|
||||||
|
interop w/ ssh.com dsa-keys:
|
||||||
ssh-keygen -f /key/from/ssh.com -X >> ~/.ssh/authorized_keys2
|
ssh-keygen -f /key/from/ssh.com -X >> ~/.ssh/authorized_keys2
|
||||||
and vice versa
|
and vice versa
|
||||||
ssh-keygen -f /privatekey/from/openssh -x > ~/.ssh2/mykey.pub
|
ssh-keygen -f /privatekey/from/openssh -x > ~/.ssh2/mykey.pub
|
||||||
|
@ -18,21 +19,20 @@ works:
|
||||||
encryption: blowfish-cbc, 3des-cbc, arcfour, cast128-cbc
|
encryption: blowfish-cbc, 3des-cbc, arcfour, cast128-cbc
|
||||||
mac: hmac-md5, hmac-sha1, (hmac-ripemd160)
|
mac: hmac-md5, hmac-sha1, (hmac-ripemd160)
|
||||||
compression: zlib, none
|
compression: zlib, none
|
||||||
secsh-userauth: passwd only
|
secsh-userauth: passwd and pubkey with DSA
|
||||||
secsh-connection: pty+shell or command, flow control works (window adjust)
|
secsh-connection: pty+shell or command, flow control works (window adjust)
|
||||||
tcp-forwarding: -L works
|
tcp-forwarding: -L works, -R incomplete
|
||||||
dss: verification works,
|
x11-fwd
|
||||||
key database in ~/.ssh/known_hosts with bits == 0 hack
|
dss/dsa: host key database in ~/.ssh/known_hosts2
|
||||||
dss: signature works, keygen w/ openssl
|
|
||||||
client interops w/ sshd2, lshd
|
client interops w/ sshd2, lshd
|
||||||
server interops w/ ssh2, lsh, ssh.com's Windows client, SecureCRT, F-Secure SSH Client 4.0
|
server interops w/ ssh2, lsh, ssh.com's Windows client, SecureCRT, F-Secure SSH Client 4.0
|
||||||
server supports multiple concurrent sessions (e.g. with SSH.com Windows client)
|
server supports multiple concurrent sessions (e.g. with SSH.com Windows client)
|
||||||
todo:
|
todo:
|
||||||
re-keying
|
re-keying
|
||||||
secsh-connection features:
|
secsh-connection features:
|
||||||
tcp-forwarding, agent-fwd, x11-fwd
|
tcp-forwarding, agent-fwd
|
||||||
auth other than passwd:
|
auth other than passwd, and DSA-pubkey:
|
||||||
pubkey, keyboard-interactive
|
keyboard-interactive, (PGP-pubkey?)
|
||||||
config
|
config
|
||||||
server-auth w/ old host-keys
|
server-auth w/ old host-keys
|
||||||
cleanup
|
cleanup
|
||||||
|
@ -41,4 +41,4 @@ todo:
|
||||||
sftp
|
sftp
|
||||||
|
|
||||||
-markus
|
-markus
|
||||||
$Date: 2000/04/27 13:42:58 $
|
$Date: 2000/04/29 13:57:09 $
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$Id: auth-rsa.c,v 1.18 2000/04/29 13:57:09 damien Exp $");
|
RCSID("$Id: auth-rsa.c,v 1.19 2000/04/30 00:00:53 damien Exp $");
|
||||||
|
|
||||||
#include "rsa.h"
|
#include "rsa.h"
|
||||||
#include "packet.h"
|
#include "packet.h"
|
||||||
|
@ -256,7 +256,6 @@ auth_rsa(struct passwd *pw, BIGNUM *client_n)
|
||||||
|
|
||||||
/* We have found the desired key. */
|
/* We have found the desired key. */
|
||||||
|
|
||||||
|
|
||||||
/* Perform the challenge-response dialog for this key. */
|
/* Perform the challenge-response dialog for this key. */
|
||||||
if (!auth_rsa_challenge_dialog(pk)) {
|
if (!auth_rsa_challenge_dialog(pk)) {
|
||||||
/* Wrong response. */
|
/* Wrong response. */
|
||||||
|
|
4
auth1.c
4
auth1.c
|
@ -4,7 +4,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$OpenBSD: auth1.c,v 1.1 2000/04/26 21:28:32 markus Exp $");
|
RCSID("$OpenBSD: auth1.c,v 1.2 2000/04/29 18:11:52 markus Exp $");
|
||||||
|
|
||||||
#include "xmalloc.h"
|
#include "xmalloc.h"
|
||||||
#include "rsa.h"
|
#include "rsa.h"
|
||||||
|
@ -262,7 +262,7 @@ do_authloop(struct passwd * pw)
|
||||||
packet_get_bignum(client_host_key->n, &nlen);
|
packet_get_bignum(client_host_key->n, &nlen);
|
||||||
|
|
||||||
if (bits != BN_num_bits(client_host_key->n))
|
if (bits != BN_num_bits(client_host_key->n))
|
||||||
error("Warning: keysize mismatch for client_host_key: "
|
log("Warning: keysize mismatch for client_host_key: "
|
||||||
"actual %d, announced %d", BN_num_bits(client_host_key->n), bits);
|
"actual %d, announced %d", BN_num_bits(client_host_key->n), bits);
|
||||||
packet_integrity_check(plen, (4 + ulen) + 4 + elen + nlen, type);
|
packet_integrity_check(plen, (4 + ulen) + 4 + elen + nlen, type);
|
||||||
|
|
||||||
|
|
4
authfd.c
4
authfd.c
|
@ -14,7 +14,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$Id: authfd.c,v 1.13 2000/04/16 02:31:49 damien Exp $");
|
RCSID("$Id: authfd.c,v 1.14 2000/04/30 00:00:53 damien Exp $");
|
||||||
|
|
||||||
#include "ssh.h"
|
#include "ssh.h"
|
||||||
#include "rsa.h"
|
#include "rsa.h"
|
||||||
|
@ -217,7 +217,7 @@ ssh_get_next_identity(AuthenticationConnection *auth,
|
||||||
*comment = buffer_get_string(&auth->identities, NULL);
|
*comment = buffer_get_string(&auth->identities, NULL);
|
||||||
|
|
||||||
if (bits != BN_num_bits(n))
|
if (bits != BN_num_bits(n))
|
||||||
error("Warning: identity keysize mismatch: actual %d, announced %u",
|
log("Warning: identity keysize mismatch: actual %d, announced %u",
|
||||||
BN_num_bits(n), bits);
|
BN_num_bits(n), bits);
|
||||||
|
|
||||||
/* Decrement the number of remaining entries. */
|
/* Decrement the number of remaining entries. */
|
||||||
|
|
191
channels.c
191
channels.c
|
@ -17,7 +17,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$Id: channels.c,v 1.26 2000/04/19 06:26:13 damien Exp $");
|
RCSID("$Id: channels.c,v 1.27 2000/04/30 00:00:53 damien Exp $");
|
||||||
|
|
||||||
#include "ssh.h"
|
#include "ssh.h"
|
||||||
#include "packet.h"
|
#include "packet.h"
|
||||||
|
@ -40,9 +40,11 @@ RCSID("$Id: channels.c,v 1.26 2000/04/19 06:26:13 damien Exp $");
|
||||||
/* Max len of agent socket */
|
/* Max len of agent socket */
|
||||||
#define MAX_SOCKET_NAME 100
|
#define MAX_SOCKET_NAME 100
|
||||||
|
|
||||||
/* default buffer for tcp-fwd-channel */
|
/* default window/packet sizes for tcp/x11-fwd-channel */
|
||||||
#define CHAN_WINDOW_DEFAULT (8*1024)
|
#define CHAN_TCP_WINDOW_DEFAULT (8*1024)
|
||||||
#define CHAN_PACKET_DEFAULT (CHAN_WINDOW_DEFAULT/2)
|
#define CHAN_TCP_PACKET_DEFAULT (CHAN_TCP_WINDOW_DEFAULT/2)
|
||||||
|
#define CHAN_X11_WINDOW_DEFAULT (4*1024)
|
||||||
|
#define CHAN_X11_PACKET_DEFAULT (CHAN_X11_WINDOW_DEFAULT/2)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Pointer to an array containing all allocated channels. The array is
|
* Pointer to an array containing all allocated channels. The array is
|
||||||
|
@ -204,17 +206,15 @@ channel_new(char *ctype, int type, int rfd, int wfd, int efd,
|
||||||
c->self = found;
|
c->self = found;
|
||||||
c->type = type;
|
c->type = type;
|
||||||
c->ctype = ctype;
|
c->ctype = ctype;
|
||||||
c->local_window = window;
|
|
||||||
c->local_window_max = window;
|
|
||||||
c->local_consumed = 0;
|
|
||||||
c->local_maxpacket = maxpack;
|
|
||||||
c->remote_window = 0;
|
|
||||||
c->remote_maxpacket = 0;
|
|
||||||
c->rfd = rfd;
|
c->rfd = rfd;
|
||||||
c->wfd = wfd;
|
c->wfd = wfd;
|
||||||
c->sock = (rfd == wfd) ? rfd : -1;
|
c->sock = (rfd == wfd) ? rfd : -1;
|
||||||
c->efd = efd;
|
c->efd = efd;
|
||||||
c->extended_usage = extended_usage;
|
c->extended_usage = extended_usage;
|
||||||
|
c->local_window = window;
|
||||||
|
c->local_window_max = window;
|
||||||
|
c->local_consumed = 0;
|
||||||
|
c->local_maxpacket = maxpack;
|
||||||
c->remote_id = -1;
|
c->remote_id = -1;
|
||||||
c->remote_name = remote_name;
|
c->remote_name = remote_name;
|
||||||
c->remote_window = 0;
|
c->remote_window = 0;
|
||||||
|
@ -371,6 +371,7 @@ channel_pre_output_draining(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
* state until the first packet has been completely read. The authentication
|
* state until the first packet has been completely read. The authentication
|
||||||
* data in that packet is then substituted by the real data if it matches the
|
* data in that packet is then substituted by the real data if it matches the
|
||||||
* fake data, and the channel is put into normal mode.
|
* fake data, and the channel is put into normal mode.
|
||||||
|
* XXX All this happens at the client side.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
x11_open_helper(Channel *c)
|
x11_open_helper(Channel *c)
|
||||||
|
@ -456,7 +457,7 @@ channel_pre_x11_open_13(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
channel_pre_x11_open_15(Channel *c, fd_set * readset, fd_set * writeset)
|
channel_pre_x11_open(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
{
|
{
|
||||||
int ret = x11_open_helper(c);
|
int ret = x11_open_helper(c);
|
||||||
if (ret == 1) {
|
if (ret == 1) {
|
||||||
|
@ -464,7 +465,7 @@ channel_pre_x11_open_15(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
channel_pre_open_15(c, readset, writeset);
|
channel_pre_open_15(c, readset, writeset);
|
||||||
} else if (ret == -1) {
|
} else if (ret == -1) {
|
||||||
debug("X11 rejected %d i%d/o%d", c->self, c->istate, c->ostate);
|
debug("X11 rejected %d i%d/o%d", c->self, c->istate, c->ostate);
|
||||||
chan_read_failed(c);
|
chan_read_failed(c); /** force close? */
|
||||||
chan_write_failed(c);
|
chan_write_failed(c);
|
||||||
debug("X11 closed %d i%d/o%d", c->self, c->istate, c->ostate);
|
debug("X11 closed %d i%d/o%d", c->self, c->istate, c->ostate);
|
||||||
}
|
}
|
||||||
|
@ -478,6 +479,7 @@ channel_post_x11_listener(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
int newsock, newch;
|
int newsock, newch;
|
||||||
socklen_t addrlen;
|
socklen_t addrlen;
|
||||||
char buf[16384], *remote_hostname;
|
char buf[16384], *remote_hostname;
|
||||||
|
int remote_port;
|
||||||
|
|
||||||
if (FD_ISSET(c->sock, readset)) {
|
if (FD_ISSET(c->sock, readset)) {
|
||||||
debug("X11 connection requested.");
|
debug("X11 connection requested.");
|
||||||
|
@ -488,17 +490,33 @@ channel_post_x11_listener(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
remote_hostname = get_remote_hostname(newsock);
|
remote_hostname = get_remote_hostname(newsock);
|
||||||
|
remote_port = get_peer_port(newsock);
|
||||||
snprintf(buf, sizeof buf, "X11 connection from %.200s port %d",
|
snprintf(buf, sizeof buf, "X11 connection from %.200s port %d",
|
||||||
remote_hostname, get_peer_port(newsock));
|
remote_hostname, remote_port);
|
||||||
xfree(remote_hostname);
|
|
||||||
newch = channel_allocate(SSH_CHANNEL_OPENING, newsock,
|
newch = channel_new("x11",
|
||||||
xstrdup(buf));
|
SSH_CHANNEL_OPENING, newsock, newsock, -1,
|
||||||
|
c->local_window_max, c->local_maxpacket,
|
||||||
|
0, xstrdup(buf));
|
||||||
|
if (compat20) {
|
||||||
|
packet_start(SSH2_MSG_CHANNEL_OPEN);
|
||||||
|
packet_put_cstring("x11");
|
||||||
|
packet_put_int(newch);
|
||||||
|
packet_put_int(c->local_window_max);
|
||||||
|
packet_put_int(c->local_maxpacket);
|
||||||
|
/* originator host and port */
|
||||||
|
packet_put_cstring(remote_hostname);
|
||||||
|
packet_put_int(remote_port);
|
||||||
|
packet_send();
|
||||||
|
} else {
|
||||||
packet_start(SSH_SMSG_X11_OPEN);
|
packet_start(SSH_SMSG_X11_OPEN);
|
||||||
packet_put_int(newch);
|
packet_put_int(newch);
|
||||||
if (have_hostname_in_open)
|
if (have_hostname_in_open)
|
||||||
packet_put_string(buf, strlen(buf));
|
packet_put_string(buf, strlen(buf));
|
||||||
packet_send();
|
packet_send();
|
||||||
}
|
}
|
||||||
|
xfree(remote_hostname);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -597,7 +615,7 @@ channel_handle_rfd(Channel *c, fd_set * readset, fd_set * writeset)
|
||||||
FD_ISSET(c->rfd, readset)) {
|
FD_ISSET(c->rfd, readset)) {
|
||||||
len = read(c->rfd, buf, sizeof(buf));
|
len = read(c->rfd, buf, sizeof(buf));
|
||||||
if (len <= 0) {
|
if (len <= 0) {
|
||||||
debug("channel %d: read<0 rfd %d len %d",
|
debug("channel %d: read<=0 rfd %d len %d",
|
||||||
c->self, c->rfd, len);
|
c->self, c->rfd, len);
|
||||||
if (compat13) {
|
if (compat13) {
|
||||||
buffer_consume(&c->output, buffer_len(&c->output));
|
buffer_consume(&c->output, buffer_len(&c->output));
|
||||||
|
@ -729,10 +747,13 @@ void
|
||||||
channel_handler_init_20(void)
|
channel_handler_init_20(void)
|
||||||
{
|
{
|
||||||
channel_pre[SSH_CHANNEL_OPEN] = &channel_pre_open_20;
|
channel_pre[SSH_CHANNEL_OPEN] = &channel_pre_open_20;
|
||||||
|
channel_pre[SSH_CHANNEL_X11_OPEN] = &channel_pre_x11_open;
|
||||||
channel_pre[SSH_CHANNEL_PORT_LISTENER] = &channel_pre_listener;
|
channel_pre[SSH_CHANNEL_PORT_LISTENER] = &channel_pre_listener;
|
||||||
|
channel_pre[SSH_CHANNEL_X11_LISTENER] = &channel_pre_listener;
|
||||||
|
|
||||||
channel_post[SSH_CHANNEL_OPEN] = &channel_post_open_2;
|
channel_post[SSH_CHANNEL_OPEN] = &channel_post_open_2;
|
||||||
channel_post[SSH_CHANNEL_PORT_LISTENER] = &channel_post_port_listener;
|
channel_post[SSH_CHANNEL_PORT_LISTENER] = &channel_post_port_listener;
|
||||||
|
channel_post[SSH_CHANNEL_X11_LISTENER] = &channel_post_x11_listener;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -757,7 +778,7 @@ void
|
||||||
channel_handler_init_15(void)
|
channel_handler_init_15(void)
|
||||||
{
|
{
|
||||||
channel_pre[SSH_CHANNEL_OPEN] = &channel_pre_open_15;
|
channel_pre[SSH_CHANNEL_OPEN] = &channel_pre_open_15;
|
||||||
channel_pre[SSH_CHANNEL_X11_OPEN] = &channel_pre_x11_open_15;
|
channel_pre[SSH_CHANNEL_X11_OPEN] = &channel_pre_x11_open;
|
||||||
channel_pre[SSH_CHANNEL_X11_LISTENER] = &channel_pre_listener;
|
channel_pre[SSH_CHANNEL_X11_LISTENER] = &channel_pre_listener;
|
||||||
channel_pre[SSH_CHANNEL_PORT_LISTENER] = &channel_pre_listener;
|
channel_pre[SSH_CHANNEL_PORT_LISTENER] = &channel_pre_listener;
|
||||||
channel_pre[SSH_CHANNEL_AUTH_SOCKET] = &channel_pre_listener;
|
channel_pre[SSH_CHANNEL_AUTH_SOCKET] = &channel_pre_listener;
|
||||||
|
@ -1428,7 +1449,7 @@ channel_request_local_forwarding(u_short port, const char *host,
|
||||||
ch = channel_new(
|
ch = channel_new(
|
||||||
"port listener", SSH_CHANNEL_PORT_LISTENER,
|
"port listener", SSH_CHANNEL_PORT_LISTENER,
|
||||||
sock, sock, -1,
|
sock, sock, -1,
|
||||||
CHAN_WINDOW_DEFAULT, CHAN_PACKET_DEFAULT,
|
CHAN_TCP_WINDOW_DEFAULT, CHAN_TCP_PACKET_DEFAULT,
|
||||||
0, xstrdup("port listener"));
|
0, xstrdup("port listener"));
|
||||||
strlcpy(channels[ch].path, host, sizeof(channels[ch].path));
|
strlcpy(channels[ch].path, host, sizeof(channels[ch].path));
|
||||||
channels[ch].host_port = host_port;
|
channels[ch].host_port = host_port;
|
||||||
|
@ -1769,8 +1790,10 @@ x11_create_display_inet(int screen_number, int x11_display_offset)
|
||||||
/* Allocate a channel for each socket. */
|
/* Allocate a channel for each socket. */
|
||||||
for (n = 0; n < num_socks; n++) {
|
for (n = 0; n < num_socks; n++) {
|
||||||
sock = socks[n];
|
sock = socks[n];
|
||||||
(void) channel_allocate(SSH_CHANNEL_X11_LISTENER, sock,
|
(void) channel_new("x11 listener",
|
||||||
xstrdup("X11 inet listener"));
|
SSH_CHANNEL_X11_LISTENER, sock, sock, -1,
|
||||||
|
CHAN_X11_WINDOW_DEFAULT, CHAN_X11_PACKET_DEFAULT,
|
||||||
|
0, xstrdup("X11 inet listener"));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return a suitable value for the DISPLAY environment variable. */
|
/* Return a suitable value for the DISPLAY environment variable. */
|
||||||
|
@ -1810,44 +1833,21 @@ connect_local_xsocket(unsigned int dnr)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
/*
|
x11_connect_display(void)
|
||||||
* This is called when SSH_SMSG_X11_OPEN is received. The packet contains
|
|
||||||
* the remote channel number. We should do whatever we want, and respond
|
|
||||||
* with either SSH_MSG_OPEN_CONFIRMATION or SSH_MSG_OPEN_FAILURE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void
|
|
||||||
x11_input_open(int type, int plen)
|
|
||||||
{
|
{
|
||||||
int remote_channel, display_number, sock = 0, newch;
|
int display_number, sock = 0;
|
||||||
const char *display;
|
const char *display;
|
||||||
char buf[1024], *cp, *remote_host;
|
char buf[1024], *cp;
|
||||||
unsigned int remote_len;
|
|
||||||
struct addrinfo hints, *ai, *aitop;
|
struct addrinfo hints, *ai, *aitop;
|
||||||
char strport[NI_MAXSERV];
|
char strport[NI_MAXSERV];
|
||||||
int gaierr;
|
int gaierr;
|
||||||
|
|
||||||
/* Get remote channel number. */
|
|
||||||
remote_channel = packet_get_int();
|
|
||||||
|
|
||||||
/* Get remote originator name. */
|
|
||||||
if (have_hostname_in_open) {
|
|
||||||
remote_host = packet_get_string(&remote_len);
|
|
||||||
remote_len += 4;
|
|
||||||
} else {
|
|
||||||
remote_host = xstrdup("unknown (remote did not supply name)");
|
|
||||||
remote_len = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
debug("Received X11 open request.");
|
|
||||||
packet_integrity_check(plen, 4 + remote_len, SSH_SMSG_X11_OPEN);
|
|
||||||
|
|
||||||
/* Try to open a socket for the local X server. */
|
/* Try to open a socket for the local X server. */
|
||||||
display = getenv("DISPLAY");
|
display = getenv("DISPLAY");
|
||||||
if (!display) {
|
if (!display) {
|
||||||
error("DISPLAY not set.");
|
error("DISPLAY not set.");
|
||||||
goto fail;
|
return -1;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* Now we decode the value of the DISPLAY variable and make a
|
* Now we decode the value of the DISPLAY variable and make a
|
||||||
|
@ -1864,15 +1864,15 @@ x11_input_open(int type, int plen)
|
||||||
if (sscanf(strrchr(display, ':') + 1, "%d", &display_number) != 1) {
|
if (sscanf(strrchr(display, ':') + 1, "%d", &display_number) != 1) {
|
||||||
error("Could not parse display number from DISPLAY: %.100s",
|
error("Could not parse display number from DISPLAY: %.100s",
|
||||||
display);
|
display);
|
||||||
goto fail;
|
return -1;
|
||||||
}
|
}
|
||||||
/* Create a socket. */
|
/* Create a socket. */
|
||||||
sock = connect_local_xsocket(display_number);
|
sock = connect_local_xsocket(display_number);
|
||||||
if (sock < 0)
|
if (sock < 0)
|
||||||
goto fail;
|
return -1;
|
||||||
|
|
||||||
/* OK, we now have a connection to the display. */
|
/* OK, we now have a connection to the display. */
|
||||||
goto success;
|
return sock;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* Connect to an inet socket. The DISPLAY value is supposedly
|
* Connect to an inet socket. The DISPLAY value is supposedly
|
||||||
|
@ -1883,14 +1883,14 @@ x11_input_open(int type, int plen)
|
||||||
cp = strchr(buf, ':');
|
cp = strchr(buf, ':');
|
||||||
if (!cp) {
|
if (!cp) {
|
||||||
error("Could not find ':' in DISPLAY: %.100s", display);
|
error("Could not find ':' in DISPLAY: %.100s", display);
|
||||||
goto fail;
|
return -1;
|
||||||
}
|
}
|
||||||
*cp = 0;
|
*cp = 0;
|
||||||
/* buf now contains the host name. But first we parse the display number. */
|
/* buf now contains the host name. But first we parse the display number. */
|
||||||
if (sscanf(cp + 1, "%d", &display_number) != 1) {
|
if (sscanf(cp + 1, "%d", &display_number) != 1) {
|
||||||
error("Could not parse display number from DISPLAY: %.100s",
|
error("Could not parse display number from DISPLAY: %.100s",
|
||||||
display);
|
display);
|
||||||
goto fail;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Look up the host address */
|
/* Look up the host address */
|
||||||
|
@ -1900,7 +1900,7 @@ x11_input_open(int type, int plen)
|
||||||
snprintf(strport, sizeof strport, "%d", 6000 + display_number);
|
snprintf(strport, sizeof strport, "%d", 6000 + display_number);
|
||||||
if ((gaierr = getaddrinfo(buf, strport, &hints, &aitop)) != 0) {
|
if ((gaierr = getaddrinfo(buf, strport, &hints, &aitop)) != 0) {
|
||||||
error("%.100s: unknown host. (%s)", buf, gai_strerror(gaierr));
|
error("%.100s: unknown host. (%s)", buf, gai_strerror(gaierr));
|
||||||
goto fail;
|
return -1;
|
||||||
}
|
}
|
||||||
for (ai = aitop; ai; ai = ai->ai_next) {
|
for (ai = aitop; ai; ai = ai->ai_next) {
|
||||||
/* Create a socket. */
|
/* Create a socket. */
|
||||||
|
@ -1923,16 +1923,52 @@ x11_input_open(int type, int plen)
|
||||||
if (!ai) {
|
if (!ai) {
|
||||||
error("connect %.100s port %d: %.100s", buf, 6000 + display_number,
|
error("connect %.100s port %d: %.100s", buf, 6000 + display_number,
|
||||||
strerror(errno));
|
strerror(errno));
|
||||||
goto fail;
|
return -1;
|
||||||
}
|
}
|
||||||
success:
|
return sock;
|
||||||
/* We have successfully obtained a connection to the real X display. */
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is called when SSH_SMSG_X11_OPEN is received. The packet contains
|
||||||
|
* the remote channel number. We should do whatever we want, and respond
|
||||||
|
* with either SSH_MSG_OPEN_CONFIRMATION or SSH_MSG_OPEN_FAILURE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
void
|
||||||
|
x11_input_open(int type, int plen)
|
||||||
|
{
|
||||||
|
int remote_channel, sock = 0, newch;
|
||||||
|
char *remote_host;
|
||||||
|
unsigned int remote_len;
|
||||||
|
|
||||||
|
/* Get remote channel number. */
|
||||||
|
remote_channel = packet_get_int();
|
||||||
|
|
||||||
|
/* Get remote originator name. */
|
||||||
|
if (have_hostname_in_open) {
|
||||||
|
remote_host = packet_get_string(&remote_len);
|
||||||
|
remote_len += 4;
|
||||||
|
} else {
|
||||||
|
remote_host = xstrdup("unknown (remote did not supply name)");
|
||||||
|
remote_len = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
debug("Received X11 open request.");
|
||||||
|
packet_integrity_check(plen, 4 + remote_len, SSH_SMSG_X11_OPEN);
|
||||||
|
|
||||||
|
/* Obtain a connection to the real X display. */
|
||||||
|
sock = x11_connect_display();
|
||||||
|
if (sock == -1) {
|
||||||
|
/* Send refusal to the remote host. */
|
||||||
|
packet_start(SSH_MSG_CHANNEL_OPEN_FAILURE);
|
||||||
|
packet_put_int(remote_channel);
|
||||||
|
packet_send();
|
||||||
|
} else {
|
||||||
/* Allocate a channel for this connection. */
|
/* Allocate a channel for this connection. */
|
||||||
if (x11_saved_proto == NULL)
|
newch = channel_allocate(
|
||||||
newch = channel_allocate(SSH_CHANNEL_OPEN, sock, remote_host);
|
(x11_saved_proto == NULL) ?
|
||||||
else
|
SSH_CHANNEL_OPEN : SSH_CHANNEL_X11_OPEN,
|
||||||
newch = channel_allocate(SSH_CHANNEL_X11_OPEN, sock, remote_host);
|
sock, remote_host);
|
||||||
channels[newch].remote_id = remote_channel;
|
channels[newch].remote_id = remote_channel;
|
||||||
|
|
||||||
/* Send a confirmation to the remote host. */
|
/* Send a confirmation to the remote host. */
|
||||||
|
@ -1940,14 +1976,7 @@ success:
|
||||||
packet_put_int(remote_channel);
|
packet_put_int(remote_channel);
|
||||||
packet_put_int(newch);
|
packet_put_int(newch);
|
||||||
packet_send();
|
packet_send();
|
||||||
|
}
|
||||||
return;
|
|
||||||
|
|
||||||
fail:
|
|
||||||
/* Send refusal to the remote host. */
|
|
||||||
packet_start(SSH_MSG_CHANNEL_OPEN_FAILURE);
|
|
||||||
packet_put_int(remote_channel);
|
|
||||||
packet_send();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1956,7 +1985,8 @@ fail:
|
||||||
*/
|
*/
|
||||||
|
|
||||||
void
|
void
|
||||||
x11_request_forwarding_with_spoofing(const char *proto, const char *data)
|
x11_request_forwarding_with_spoofing(int client_session_id,
|
||||||
|
const char *proto, const char *data)
|
||||||
{
|
{
|
||||||
unsigned int data_len = (unsigned int) strlen(data) / 2;
|
unsigned int data_len = (unsigned int) strlen(data) / 2;
|
||||||
unsigned int i, value;
|
unsigned int i, value;
|
||||||
|
@ -2002,9 +2032,14 @@ x11_request_forwarding_with_spoofing(const char *proto, const char *data)
|
||||||
sprintf(new_data + 2 * i, "%02x", (unsigned char) x11_fake_data[i]);
|
sprintf(new_data + 2 * i, "%02x", (unsigned char) x11_fake_data[i]);
|
||||||
|
|
||||||
/* Send the request packet. */
|
/* Send the request packet. */
|
||||||
|
if (compat20) {
|
||||||
|
channel_request_start(client_session_id, "x11-req", 0);
|
||||||
|
packet_put_char(0); /* XXX bool single connection */
|
||||||
|
} else {
|
||||||
packet_start(SSH_CMSG_X11_REQUEST_FORWARDING);
|
packet_start(SSH_CMSG_X11_REQUEST_FORWARDING);
|
||||||
packet_put_string(proto, strlen(proto));
|
}
|
||||||
packet_put_string(new_data, strlen(new_data));
|
packet_put_cstring(proto);
|
||||||
|
packet_put_cstring(new_data);
|
||||||
packet_put_int(screen_number);
|
packet_put_int(screen_number);
|
||||||
packet_send();
|
packet_send();
|
||||||
packet_write_wait();
|
packet_write_wait();
|
||||||
|
@ -2154,20 +2189,26 @@ auth_input_open_request(int type, int plen)
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
channel_open(int id)
|
channel_start_open(int id)
|
||||||
{
|
{
|
||||||
Channel *c = channel_lookup(id);
|
Channel *c = channel_lookup(id);
|
||||||
if (c == NULL) {
|
if (c == NULL) {
|
||||||
log("channel_open: %d: bad id", id);
|
log("channel_open: %d: bad id", id);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
debug("send channel open %d", id);
|
||||||
packet_start(SSH2_MSG_CHANNEL_OPEN);
|
packet_start(SSH2_MSG_CHANNEL_OPEN);
|
||||||
packet_put_cstring(c->ctype);
|
packet_put_cstring(c->ctype);
|
||||||
packet_put_int(c->self);
|
packet_put_int(c->self);
|
||||||
packet_put_int(c->local_window);
|
packet_put_int(c->local_window);
|
||||||
packet_put_int(c->local_maxpacket);
|
packet_put_int(c->local_maxpacket);
|
||||||
|
}
|
||||||
|
void
|
||||||
|
channel_open(int id)
|
||||||
|
{
|
||||||
|
/* XXX REMOVE ME */
|
||||||
|
channel_start_open(id);
|
||||||
packet_send();
|
packet_send();
|
||||||
debug("channel open %d", id);
|
|
||||||
}
|
}
|
||||||
void
|
void
|
||||||
channel_request(int id, char *service, int wantconfirm)
|
channel_request(int id, char *service, int wantconfirm)
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* RCSID("$Id: channels.h,v 1.7 2000/04/16 01:18:41 damien Exp $"); */
|
/* RCSID("$Id: channels.h,v 1.8 2000/04/30 00:00:54 damien Exp $"); */
|
||||||
|
|
||||||
#ifndef CHANNELS_H
|
#ifndef CHANNELS_H
|
||||||
#define CHANNELS_H
|
#define CHANNELS_H
|
||||||
|
@ -207,7 +207,9 @@ void x11_request_forwarding(void);
|
||||||
* Requests forwarding for X11 connections, with authentication spoofing.
|
* Requests forwarding for X11 connections, with authentication spoofing.
|
||||||
* This should be called in the client only.
|
* This should be called in the client only.
|
||||||
*/
|
*/
|
||||||
void x11_request_forwarding_with_spoofing(const char *proto, const char *data);
|
void
|
||||||
|
x11_request_forwarding_with_spoofing(int client_session_id,
|
||||||
|
const char *proto, const char *data);
|
||||||
|
|
||||||
/* Sends a message to the server to request authentication fd forwarding. */
|
/* Sends a message to the server to request authentication fd forwarding. */
|
||||||
void auth_request_forwarding(void);
|
void auth_request_forwarding(void);
|
||||||
|
@ -230,5 +232,6 @@ void auth_input_open_request(int type, int plen);
|
||||||
|
|
||||||
/* XXX */
|
/* XXX */
|
||||||
int channel_connect_to(const char *host, u_short host_port);
|
int channel_connect_to(const char *host, u_short host_port);
|
||||||
|
int x11_connect_display(void);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
67
clientloop.c
67
clientloop.c
|
@ -16,7 +16,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$Id: clientloop.c,v 1.12 2000/04/19 21:42:21 damien Exp $");
|
RCSID("$Id: clientloop.c,v 1.13 2000/04/30 00:00:54 damien Exp $");
|
||||||
|
|
||||||
#include "xmalloc.h"
|
#include "xmalloc.h"
|
||||||
#include "ssh.h"
|
#include "ssh.h"
|
||||||
|
@ -954,6 +954,69 @@ client_input_exit_status(int type, int plen)
|
||||||
quit_pending = 1;
|
quit_pending = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* XXXX move to generic input handler */
|
||||||
|
void
|
||||||
|
client_input_channel_open(int type, int plen)
|
||||||
|
{
|
||||||
|
Channel *c = NULL;
|
||||||
|
char *ctype;
|
||||||
|
int id;
|
||||||
|
unsigned int len;
|
||||||
|
int rchan;
|
||||||
|
int rmaxpack;
|
||||||
|
int rwindow;
|
||||||
|
|
||||||
|
ctype = packet_get_string(&len);
|
||||||
|
rchan = packet_get_int();
|
||||||
|
rwindow = packet_get_int();
|
||||||
|
rmaxpack = packet_get_int();
|
||||||
|
|
||||||
|
log("server_input_open: ctype %s rchan %d win %d max %d",
|
||||||
|
ctype, rchan, rwindow, rmaxpack);
|
||||||
|
|
||||||
|
if (strcmp(ctype, "x11") == 0) {
|
||||||
|
int sock;
|
||||||
|
char *originator;
|
||||||
|
int originator_port;
|
||||||
|
originator = packet_get_string(NULL);
|
||||||
|
originator_port = packet_get_int();
|
||||||
|
packet_done();
|
||||||
|
/* XXX check permission */
|
||||||
|
xfree(originator);
|
||||||
|
/* XXX move to channels.c */
|
||||||
|
sock = x11_connect_display();
|
||||||
|
if (sock >= 0) {
|
||||||
|
id = channel_new("x11", SSH_CHANNEL_X11_OPEN,
|
||||||
|
sock, sock, -1, 4*1024, 32*1024, 0,
|
||||||
|
xstrdup("x11"));
|
||||||
|
c = channel_lookup(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* XXX duplicate : */
|
||||||
|
if (c != NULL) {
|
||||||
|
debug("confirm %s", ctype);
|
||||||
|
c->remote_id = rchan;
|
||||||
|
c->remote_window = rwindow;
|
||||||
|
c->remote_maxpacket = rmaxpack;
|
||||||
|
|
||||||
|
packet_start(SSH2_MSG_CHANNEL_OPEN_CONFIRMATION);
|
||||||
|
packet_put_int(c->remote_id);
|
||||||
|
packet_put_int(c->self);
|
||||||
|
packet_put_int(c->local_window);
|
||||||
|
packet_put_int(c->local_maxpacket);
|
||||||
|
packet_send();
|
||||||
|
} else {
|
||||||
|
debug("failure %s", ctype);
|
||||||
|
packet_start(SSH2_MSG_CHANNEL_OPEN_FAILURE);
|
||||||
|
packet_put_int(rchan);
|
||||||
|
packet_put_int(SSH2_OPEN_ADMINISTRATIVELY_PROHIBITED);
|
||||||
|
packet_put_cstring("bla bla");
|
||||||
|
packet_put_cstring("");
|
||||||
|
packet_send();
|
||||||
|
}
|
||||||
|
xfree(ctype);
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
client_init_dispatch_20()
|
client_init_dispatch_20()
|
||||||
{
|
{
|
||||||
|
@ -962,6 +1025,7 @@ client_init_dispatch_20()
|
||||||
dispatch_set(SSH2_MSG_CHANNEL_DATA, &channel_input_data);
|
dispatch_set(SSH2_MSG_CHANNEL_DATA, &channel_input_data);
|
||||||
dispatch_set(SSH2_MSG_CHANNEL_EOF, &channel_input_ieof);
|
dispatch_set(SSH2_MSG_CHANNEL_EOF, &channel_input_ieof);
|
||||||
dispatch_set(SSH2_MSG_CHANNEL_EXTENDED_DATA, &channel_input_extended_data);
|
dispatch_set(SSH2_MSG_CHANNEL_EXTENDED_DATA, &channel_input_extended_data);
|
||||||
|
dispatch_set(SSH2_MSG_CHANNEL_OPEN, &client_input_channel_open);
|
||||||
dispatch_set(SSH2_MSG_CHANNEL_OPEN_CONFIRMATION, &channel_input_open_confirmation);
|
dispatch_set(SSH2_MSG_CHANNEL_OPEN_CONFIRMATION, &channel_input_open_confirmation);
|
||||||
dispatch_set(SSH2_MSG_CHANNEL_OPEN_FAILURE, &channel_input_open_failure);
|
dispatch_set(SSH2_MSG_CHANNEL_OPEN_FAILURE, &channel_input_open_failure);
|
||||||
dispatch_set(SSH2_MSG_CHANNEL_REQUEST, &channel_input_channel_request);
|
dispatch_set(SSH2_MSG_CHANNEL_REQUEST, &channel_input_channel_request);
|
||||||
|
@ -974,7 +1038,6 @@ client_init_dispatch_13()
|
||||||
dispatch_set(SSH_MSG_CHANNEL_CLOSE, &channel_input_close);
|
dispatch_set(SSH_MSG_CHANNEL_CLOSE, &channel_input_close);
|
||||||
dispatch_set(SSH_MSG_CHANNEL_CLOSE_CONFIRMATION, &channel_input_close_confirmation);
|
dispatch_set(SSH_MSG_CHANNEL_CLOSE_CONFIRMATION, &channel_input_close_confirmation);
|
||||||
dispatch_set(SSH_MSG_CHANNEL_DATA, &channel_input_data);
|
dispatch_set(SSH_MSG_CHANNEL_DATA, &channel_input_data);
|
||||||
dispatch_set(SSH_MSG_CHANNEL_DATA, &channel_input_data);
|
|
||||||
dispatch_set(SSH_MSG_CHANNEL_OPEN_CONFIRMATION, &channel_input_open_confirmation);
|
dispatch_set(SSH_MSG_CHANNEL_OPEN_CONFIRMATION, &channel_input_open_confirmation);
|
||||||
dispatch_set(SSH_MSG_CHANNEL_OPEN_FAILURE, &channel_input_open_failure);
|
dispatch_set(SSH_MSG_CHANNEL_OPEN_FAILURE, &channel_input_open_failure);
|
||||||
dispatch_set(SSH_MSG_PORT_OPEN, &channel_input_port_open);
|
dispatch_set(SSH_MSG_PORT_OPEN, &channel_input_port_open);
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$OpenBSD: hostfile.c,v 1.17 2000/04/26 20:56:29 markus Exp $");
|
RCSID("$OpenBSD: hostfile.c,v 1.18 2000/04/29 18:11:52 markus Exp $");
|
||||||
|
|
||||||
#include "packet.h"
|
#include "packet.h"
|
||||||
#include "match.h"
|
#include "match.h"
|
||||||
|
@ -70,10 +70,10 @@ hostfile_check_key(int bits, Key *key, const char *host, const char *filename, i
|
||||||
if (key == NULL || key->type != KEY_RSA || key->rsa == NULL)
|
if (key == NULL || key->type != KEY_RSA || key->rsa == NULL)
|
||||||
return 1;
|
return 1;
|
||||||
if (bits != BN_num_bits(key->rsa->n)) {
|
if (bits != BN_num_bits(key->rsa->n)) {
|
||||||
error("Warning: %s, line %d: keysize mismatch for host %s: "
|
log("Warning: %s, line %d: keysize mismatch for host %s: "
|
||||||
"actual %d vs. announced %d.",
|
"actual %d vs. announced %d.",
|
||||||
filename, linenum, host, BN_num_bits(key->rsa->n), bits);
|
filename, linenum, host, BN_num_bits(key->rsa->n), bits);
|
||||||
error("Warning: replace %d with %d in %s, line %d.",
|
log("Warning: replace %d with %d in %s, line %d.",
|
||||||
bits, BN_num_bits(key->rsa->n), filename, linenum);
|
bits, BN_num_bits(key->rsa->n), filename, linenum);
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
|
|
89
session.c
89
session.c
|
@ -34,6 +34,7 @@ typedef struct Session Session;
|
||||||
struct Session {
|
struct Session {
|
||||||
int used;
|
int used;
|
||||||
int self;
|
int self;
|
||||||
|
int extended;
|
||||||
struct passwd *pw;
|
struct passwd *pw;
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
/* tty */
|
/* tty */
|
||||||
|
@ -46,6 +47,7 @@ struct Session {
|
||||||
int screen;
|
int screen;
|
||||||
char *auth_proto;
|
char *auth_proto;
|
||||||
char *auth_data;
|
char *auth_data;
|
||||||
|
int single_connection;
|
||||||
/* proto 2 */
|
/* proto 2 */
|
||||||
int chanid;
|
int chanid;
|
||||||
};
|
};
|
||||||
|
@ -170,6 +172,7 @@ do_authenticated(struct passwd * pw)
|
||||||
channel_permit_all_opens();
|
channel_permit_all_opens();
|
||||||
|
|
||||||
s = session_new();
|
s = session_new();
|
||||||
|
s->pw = pw;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We stay in this loop until the client requests to execute a shell
|
* We stay in this loop until the client requests to execute a shell
|
||||||
|
@ -279,6 +282,7 @@ do_authenticated(struct passwd * pw)
|
||||||
xauthfile, strerror(errno));
|
xauthfile, strerror(errno));
|
||||||
xfree(xauthfile);
|
xfree(xauthfile);
|
||||||
xauthfile = NULL;
|
xauthfile = NULL;
|
||||||
|
/* XXXX remove listening channels */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
strlcat(xauthfile, "/cookies", MAXPATHLEN);
|
strlcat(xauthfile, "/cookies", MAXPATHLEN);
|
||||||
|
@ -462,7 +466,7 @@ do_exec_no_pty(Session *s, const char *command, struct passwd * pw)
|
||||||
close(perr[1]);
|
close(perr[1]);
|
||||||
|
|
||||||
if (compat20) {
|
if (compat20) {
|
||||||
session_set_fds(s, pin[1], pout[0], perr[0]);
|
session_set_fds(s, pin[1], pout[0], s->extended ? perr[0] : -1);
|
||||||
} else {
|
} else {
|
||||||
/* Enter the interactive session. */
|
/* Enter the interactive session. */
|
||||||
server_loop(pid, pin[1], pout[0], perr[0]);
|
server_loop(pid, pin[1], pout[0], perr[0]);
|
||||||
|
@ -478,7 +482,7 @@ do_exec_no_pty(Session *s, const char *command, struct passwd * pw)
|
||||||
* handle the case that fdin and fdout are the same.
|
* handle the case that fdin and fdout are the same.
|
||||||
*/
|
*/
|
||||||
if (compat20) {
|
if (compat20) {
|
||||||
session_set_fds(s, inout[1], inout[1], err[1]);
|
session_set_fds(s, inout[1], inout[1], s->extended ? err[1] : -1);
|
||||||
} else {
|
} else {
|
||||||
server_loop(pid, inout[1], inout[1], err[1]);
|
server_loop(pid, inout[1], inout[1], err[1]);
|
||||||
/* server_loop has closed inout[1] and err[1]. */
|
/* server_loop has closed inout[1] and err[1]. */
|
||||||
|
@ -1119,6 +1123,7 @@ session_new(void)
|
||||||
Session *s = &sessions[i];
|
Session *s = &sessions[i];
|
||||||
if (! s->used) {
|
if (! s->used) {
|
||||||
s->pid = 0;
|
s->pid = 0;
|
||||||
|
s->extended = 0;
|
||||||
s->chanid = -1;
|
s->chanid = -1;
|
||||||
s->ptyfd = -1;
|
s->ptyfd = -1;
|
||||||
s->ttyfd = -1;
|
s->ttyfd = -1;
|
||||||
|
@ -1129,6 +1134,7 @@ session_new(void)
|
||||||
s->auth_data = NULL;
|
s->auth_data = NULL;
|
||||||
s->auth_proto = NULL;
|
s->auth_proto = NULL;
|
||||||
s->used = 1;
|
s->used = 1;
|
||||||
|
s->pw = NULL;
|
||||||
debug("session_new: session %d", i);
|
debug("session_new: session %d", i);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
@ -1160,12 +1166,11 @@ session_open(int chanid)
|
||||||
error("no more sessions");
|
error("no more sessions");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
debug("session_open: session %d: link with channel %d", s->self, chanid);
|
|
||||||
s->chanid = chanid;
|
|
||||||
s->pw = auth_get_user();
|
s->pw = auth_get_user();
|
||||||
if (s->pw == NULL)
|
if (s->pw == NULL)
|
||||||
fatal("no user for session %i channel %d",
|
fatal("no user for session %i", s->self);
|
||||||
s->self, s->chanid);
|
debug("session_open: session %d: link with channel %d", s->self, chanid);
|
||||||
|
s->chanid = chanid;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1257,6 +1262,69 @@ session_pty_req(Session *s)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
session_subsystem_req(Session *s)
|
||||||
|
{
|
||||||
|
unsigned int len;
|
||||||
|
int success = 0;
|
||||||
|
char *subsys = packet_get_string(&len);
|
||||||
|
|
||||||
|
packet_done();
|
||||||
|
log("subsystem request for %s", subsys);
|
||||||
|
|
||||||
|
xfree(subsys);
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
session_x11_req(Session *s)
|
||||||
|
{
|
||||||
|
if (!options.x11_forwarding) {
|
||||||
|
debug("X11 forwarding disabled in server configuration file.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (xauthfile != NULL) {
|
||||||
|
debug("X11 fwd already started.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
debug("Received request for X11 forwarding with auth spoofing.");
|
||||||
|
if (s->display != NULL)
|
||||||
|
packet_disconnect("Protocol error: X11 display already set.");
|
||||||
|
|
||||||
|
s->single_connection = packet_get_char();
|
||||||
|
s->auth_proto = packet_get_string(NULL);
|
||||||
|
s->auth_data = packet_get_string(NULL);
|
||||||
|
s->screen = packet_get_int();
|
||||||
|
packet_done();
|
||||||
|
|
||||||
|
s->display = x11_create_display_inet(s->screen, options.x11_display_offset);
|
||||||
|
if (s->display == NULL) {
|
||||||
|
xfree(s->auth_proto);
|
||||||
|
xfree(s->auth_data);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
xauthfile = xmalloc(MAXPATHLEN);
|
||||||
|
strlcpy(xauthfile, "/tmp/ssh-XXXXXXXX", MAXPATHLEN);
|
||||||
|
temporarily_use_uid(s->pw->pw_uid);
|
||||||
|
if (mkdtemp(xauthfile) == NULL) {
|
||||||
|
restore_uid();
|
||||||
|
error("private X11 dir: mkdtemp %s failed: %s",
|
||||||
|
xauthfile, strerror(errno));
|
||||||
|
xfree(xauthfile);
|
||||||
|
xauthfile = NULL;
|
||||||
|
xfree(s->auth_proto);
|
||||||
|
xfree(s->auth_data);
|
||||||
|
/* XXXX remove listening channels */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
strlcat(xauthfile, "/cookies", MAXPATHLEN);
|
||||||
|
open(xauthfile, O_RDWR|O_CREAT|O_EXCL, 0600);
|
||||||
|
restore_uid();
|
||||||
|
fatal_add_cleanup(xauthfile_cleanup_proc, s);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
session_input_channel_req(int id, void *arg)
|
session_input_channel_req(int id, void *arg)
|
||||||
{
|
{
|
||||||
|
@ -1294,6 +1362,8 @@ session_input_channel_req(int id, void *arg)
|
||||||
} else if (strcmp(rtype, "exec") == 0) {
|
} else if (strcmp(rtype, "exec") == 0) {
|
||||||
char *command = packet_get_string(&len);
|
char *command = packet_get_string(&len);
|
||||||
packet_done();
|
packet_done();
|
||||||
|
s->extended = 1;
|
||||||
|
s->extended = 1;
|
||||||
if (s->ttyfd == -1)
|
if (s->ttyfd == -1)
|
||||||
do_exec_no_pty(s, command, s->pw);
|
do_exec_no_pty(s, command, s->pw);
|
||||||
else
|
else
|
||||||
|
@ -1302,6 +1372,10 @@ session_input_channel_req(int id, void *arg)
|
||||||
success = 1;
|
success = 1;
|
||||||
} else if (strcmp(rtype, "pty-req") == 0) {
|
} else if (strcmp(rtype, "pty-req") == 0) {
|
||||||
success = session_pty_req(s);
|
success = session_pty_req(s);
|
||||||
|
} else if (strcmp(rtype, "x11-req") == 0) {
|
||||||
|
success = session_x11_req(s);
|
||||||
|
} else if (strcmp(rtype, "subsystem") == 0) {
|
||||||
|
success = session_subsystem_req(s);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (strcmp(rtype, "window-change") == 0) {
|
if (strcmp(rtype, "window-change") == 0) {
|
||||||
|
@ -1399,6 +1473,7 @@ session_exit_message(Session *s, int status)
|
||||||
* Note that we must not call 'chan_read_failed', since there could
|
* Note that we must not call 'chan_read_failed', since there could
|
||||||
* be some more data waiting in the pipe.
|
* be some more data waiting in the pipe.
|
||||||
*/
|
*/
|
||||||
|
if (c->ostate != CHAN_OUTPUT_CLOSED)
|
||||||
chan_write_failed(c);
|
chan_write_failed(c);
|
||||||
s->chanid = -1;
|
s->chanid = -1;
|
||||||
}
|
}
|
||||||
|
@ -1475,4 +1550,6 @@ do_authenticated2(void)
|
||||||
*/
|
*/
|
||||||
alarm(0);
|
alarm(0);
|
||||||
server_loop2();
|
server_loop2();
|
||||||
|
if (xauthfile)
|
||||||
|
xauthfile_cleanup_proc(NULL);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* $OpenBSD: ssh-agent.c,v 1.30 2000/04/21 00:27:11 djm Exp $ */
|
/* $OpenBSD: ssh-agent.c,v 1.31 2000/04/29 18:11:52 markus Exp $ */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Author: Tatu Ylonen <ylo@cs.hut.fi>
|
* Author: Tatu Ylonen <ylo@cs.hut.fi>
|
||||||
|
@ -9,7 +9,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$OpenBSD: ssh-agent.c,v 1.30 2000/04/21 00:27:11 djm Exp $");
|
RCSID("$OpenBSD: ssh-agent.c,v 1.31 2000/04/29 18:11:52 markus Exp $");
|
||||||
|
|
||||||
#include "ssh.h"
|
#include "ssh.h"
|
||||||
#include "rsa.h"
|
#include "rsa.h"
|
||||||
|
@ -178,7 +178,7 @@ process_remove_identity(SocketEntry *e)
|
||||||
buffer_get_bignum(&e->input, n);
|
buffer_get_bignum(&e->input, n);
|
||||||
|
|
||||||
if (bits != BN_num_bits(n))
|
if (bits != BN_num_bits(n))
|
||||||
error("Warning: identity keysize mismatch: actual %d, announced %d",
|
log("Warning: identity keysize mismatch: actual %d, announced %d",
|
||||||
BN_num_bits(n), bits);
|
BN_num_bits(n), bits);
|
||||||
|
|
||||||
/* Check if we have the key. */
|
/* Check if we have the key. */
|
||||||
|
|
102
ssh.c
102
ssh.c
|
@ -11,7 +11,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
RCSID("$Id: ssh.c,v 1.27 2000/04/29 13:57:12 damien Exp $");
|
RCSID("$Id: ssh.c,v 1.28 2000/04/30 00:00:54 damien Exp $");
|
||||||
|
|
||||||
#include <openssl/evp.h>
|
#include <openssl/evp.h>
|
||||||
#include <openssl/dsa.h>
|
#include <openssl/dsa.h>
|
||||||
|
@ -664,6 +664,45 @@ main(int ac, char **av)
|
||||||
return exit_status;
|
return exit_status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
x11_get_proto(char *proto, int proto_len, char *data, int data_len)
|
||||||
|
{
|
||||||
|
char line[512];
|
||||||
|
FILE *f;
|
||||||
|
int got_data = 0, i;
|
||||||
|
|
||||||
|
#ifdef XAUTH_PATH
|
||||||
|
/* Try to get Xauthority information for the display. */
|
||||||
|
snprintf(line, sizeof line, "%.100s list %.200s 2>/dev/null",
|
||||||
|
XAUTH_PATH, getenv("DISPLAY"));
|
||||||
|
f = popen(line, "r");
|
||||||
|
if (f && fgets(line, sizeof(line), f) &&
|
||||||
|
sscanf(line, "%*s %s %s", proto, data) == 2)
|
||||||
|
got_data = 1;
|
||||||
|
if (f)
|
||||||
|
pclose(f);
|
||||||
|
#endif /* XAUTH_PATH */
|
||||||
|
/*
|
||||||
|
* If we didn't get authentication data, just make up some
|
||||||
|
* data. The forwarding code will check the validity of the
|
||||||
|
* response anyway, and substitute this data. The X11
|
||||||
|
* server, however, will ignore this fake data and use
|
||||||
|
* whatever authentication mechanisms it was using otherwise
|
||||||
|
* for the local connection.
|
||||||
|
*/
|
||||||
|
if (!got_data) {
|
||||||
|
u_int32_t rand = 0;
|
||||||
|
|
||||||
|
strlcpy(proto, "MIT-MAGIC-COOKIE-1", proto_len);
|
||||||
|
for (i = 0; i < 16; i++) {
|
||||||
|
if (i % 4 == 0)
|
||||||
|
rand = arc4random();
|
||||||
|
snprintf(data + 2 * i, data_len - 2 * i, "%02x", rand & 0xff);
|
||||||
|
rand >>= 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
ssh_session(void)
|
ssh_session(void)
|
||||||
{
|
{
|
||||||
|
@ -737,57 +776,23 @@ ssh_session(void)
|
||||||
}
|
}
|
||||||
/* Request X11 forwarding if enabled and DISPLAY is set. */
|
/* Request X11 forwarding if enabled and DISPLAY is set. */
|
||||||
if (options.forward_x11 && getenv("DISPLAY") != NULL) {
|
if (options.forward_x11 && getenv("DISPLAY") != NULL) {
|
||||||
char line[512], proto[512], data[512];
|
char proto[512], data[512];
|
||||||
FILE *f;
|
/* Get reasonable local authentication information. */
|
||||||
int forwarded = 0, got_data = 0, i;
|
x11_get_proto(proto, sizeof proto, data, sizeof data);
|
||||||
|
/* Request forwarding with authentication spoofing. */
|
||||||
#ifdef XAUTH_PATH
|
|
||||||
/* Try to get Xauthority information for the display. */
|
|
||||||
snprintf(line, sizeof line, "%.100s list %.200s 2>/dev/null",
|
|
||||||
XAUTH_PATH, getenv("DISPLAY"));
|
|
||||||
f = popen(line, "r");
|
|
||||||
if (f && fgets(line, sizeof(line), f) &&
|
|
||||||
sscanf(line, "%*s %s %s", proto, data) == 2)
|
|
||||||
got_data = 1;
|
|
||||||
if (f)
|
|
||||||
pclose(f);
|
|
||||||
#endif /* XAUTH_PATH */
|
|
||||||
/*
|
|
||||||
* If we didn't get authentication data, just make up some
|
|
||||||
* data. The forwarding code will check the validity of the
|
|
||||||
* response anyway, and substitute this data. The X11
|
|
||||||
* server, however, will ignore this fake data and use
|
|
||||||
* whatever authentication mechanisms it was using otherwise
|
|
||||||
* for the local connection.
|
|
||||||
*/
|
|
||||||
if (!got_data) {
|
|
||||||
u_int32_t rand = 0;
|
|
||||||
|
|
||||||
strlcpy(proto, "MIT-MAGIC-COOKIE-1", sizeof proto);
|
|
||||||
for (i = 0; i < 16; i++) {
|
|
||||||
if (i % 4 == 0)
|
|
||||||
rand = arc4random();
|
|
||||||
snprintf(data + 2 * i, sizeof data - 2 * i, "%02x", rand & 0xff);
|
|
||||||
rand >>= 8;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/*
|
|
||||||
* Got local authentication reasonable information. Request
|
|
||||||
* forwarding with authentication spoofing.
|
|
||||||
*/
|
|
||||||
debug("Requesting X11 forwarding with authentication spoofing.");
|
debug("Requesting X11 forwarding with authentication spoofing.");
|
||||||
x11_request_forwarding_with_spoofing(proto, data);
|
x11_request_forwarding_with_spoofing(0, proto, data);
|
||||||
|
|
||||||
/* Read response from the server. */
|
/* Read response from the server. */
|
||||||
type = packet_read(&plen);
|
type = packet_read(&plen);
|
||||||
if (type == SSH_SMSG_SUCCESS) {
|
if (type == SSH_SMSG_SUCCESS) {
|
||||||
forwarded = 1;
|
|
||||||
interactive = 1;
|
interactive = 1;
|
||||||
} else if (type == SSH_SMSG_FAILURE)
|
} else if (type == SSH_SMSG_FAILURE) {
|
||||||
log("Warning: Remote host denied X11 forwarding.");
|
log("Warning: Remote host denied X11 forwarding.");
|
||||||
else
|
} else {
|
||||||
packet_disconnect("Protocol error waiting for X11 forwarding");
|
packet_disconnect("Protocol error waiting for X11 forwarding");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
/* Tell the packet module whether this is an interactive session. */
|
/* Tell the packet module whether this is an interactive session. */
|
||||||
packet_set_interactive(interactive, options.keepalives);
|
packet_set_interactive(interactive, options.keepalives);
|
||||||
|
|
||||||
|
@ -909,6 +914,17 @@ client_init(int id, void *arg)
|
||||||
packet_send();
|
packet_send();
|
||||||
/* XXX wait for reply */
|
/* XXX wait for reply */
|
||||||
}
|
}
|
||||||
|
if (options.forward_x11 &&
|
||||||
|
getenv("DISPLAY") != NULL) {
|
||||||
|
char proto[512], data[512];
|
||||||
|
/* Get reasonable local authentication information. */
|
||||||
|
x11_get_proto(proto, sizeof proto, data, sizeof data);
|
||||||
|
/* Request forwarding with authentication spoofing. */
|
||||||
|
debug("Requesting X11 forwarding with authentication spoofing.");
|
||||||
|
x11_request_forwarding_with_spoofing(id, proto, data);
|
||||||
|
/* XXX wait for reply */
|
||||||
|
}
|
||||||
|
|
||||||
len = buffer_len(&command);
|
len = buffer_len(&command);
|
||||||
if (len > 0) {
|
if (len > 0) {
|
||||||
if (len > 900)
|
if (len > 900)
|
||||||
|
|
Loading…
Reference in New Issue