mirror of git://anongit.mindrot.org/openssh.git
245 lines
5.5 KiB
C
245 lines
5.5 KiB
C
/* $OpenBSD: roaming_common.c,v 1.12 2014/01/09 23:20:00 djm Exp $ */
|
|
/*
|
|
* Copyright (c) 2004-2009 AppGate Network Security AB
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/uio.h>
|
|
|
|
#include <errno.h>
|
|
#ifdef HAVE_INTTYPES_H
|
|
#include <inttypes.h>
|
|
#endif
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#include "atomicio.h"
|
|
#include "log.h"
|
|
#include "packet.h"
|
|
#include "xmalloc.h"
|
|
#include "cipher.h"
|
|
#include "buffer.h"
|
|
#include "roaming.h"
|
|
#include "digest.h"
|
|
|
|
static size_t out_buf_size = 0;
|
|
static char *out_buf = NULL;
|
|
static size_t out_start;
|
|
static size_t out_last;
|
|
|
|
static u_int64_t write_bytes = 0;
|
|
static u_int64_t read_bytes = 0;
|
|
|
|
int roaming_enabled = 0;
|
|
int resume_in_progress = 0;
|
|
|
|
int
|
|
get_snd_buf_size(void)
|
|
{
|
|
int fd = packet_get_connection_out();
|
|
int optval;
|
|
socklen_t optvallen = sizeof(optval);
|
|
|
|
if (getsockopt(fd, SOL_SOCKET, SO_SNDBUF, &optval, &optvallen) != 0)
|
|
optval = DEFAULT_ROAMBUF;
|
|
return optval;
|
|
}
|
|
|
|
int
|
|
get_recv_buf_size(void)
|
|
{
|
|
int fd = packet_get_connection_in();
|
|
int optval;
|
|
socklen_t optvallen = sizeof(optval);
|
|
|
|
if (getsockopt(fd, SOL_SOCKET, SO_RCVBUF, &optval, &optvallen) != 0)
|
|
optval = DEFAULT_ROAMBUF;
|
|
return optval;
|
|
}
|
|
|
|
void
|
|
set_out_buffer_size(size_t size)
|
|
{
|
|
if (size == 0 || size > MAX_ROAMBUF)
|
|
fatal("%s: bad buffer size %lu", __func__, (u_long)size);
|
|
/*
|
|
* The buffer size can only be set once and the buffer will live
|
|
* as long as the session lives.
|
|
*/
|
|
if (out_buf == NULL) {
|
|
out_buf_size = size;
|
|
out_buf = xmalloc(size);
|
|
out_start = 0;
|
|
out_last = 0;
|
|
}
|
|
}
|
|
|
|
u_int64_t
|
|
get_recv_bytes(void)
|
|
{
|
|
return read_bytes;
|
|
}
|
|
|
|
void
|
|
add_recv_bytes(u_int64_t num)
|
|
{
|
|
read_bytes += num;
|
|
}
|
|
|
|
u_int64_t
|
|
get_sent_bytes(void)
|
|
{
|
|
return write_bytes;
|
|
}
|
|
|
|
void
|
|
roam_set_bytes(u_int64_t sent, u_int64_t recvd)
|
|
{
|
|
read_bytes = recvd;
|
|
write_bytes = sent;
|
|
}
|
|
|
|
static void
|
|
buf_append(const char *buf, size_t count)
|
|
{
|
|
if (count > out_buf_size) {
|
|
buf += count - out_buf_size;
|
|
count = out_buf_size;
|
|
}
|
|
if (count < out_buf_size - out_last) {
|
|
memcpy(out_buf + out_last, buf, count);
|
|
if (out_start > out_last)
|
|
out_start += count;
|
|
out_last += count;
|
|
} else {
|
|
/* data will wrap */
|
|
size_t chunk = out_buf_size - out_last;
|
|
memcpy(out_buf + out_last, buf, chunk);
|
|
memcpy(out_buf, buf + chunk, count - chunk);
|
|
out_last = count - chunk;
|
|
out_start = out_last + 1;
|
|
}
|
|
}
|
|
|
|
ssize_t
|
|
roaming_write(int fd, const void *buf, size_t count, int *cont)
|
|
{
|
|
ssize_t ret;
|
|
|
|
ret = write(fd, buf, count);
|
|
if (ret > 0 && !resume_in_progress) {
|
|
write_bytes += ret;
|
|
if (out_buf_size > 0)
|
|
buf_append(buf, ret);
|
|
}
|
|
if (out_buf_size > 0 &&
|
|
(ret == 0 || (ret == -1 && errno == EPIPE))) {
|
|
if (wait_for_roaming_reconnect() != 0) {
|
|
ret = 0;
|
|
*cont = 1;
|
|
} else {
|
|
ret = -1;
|
|
errno = EAGAIN;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
ssize_t
|
|
roaming_read(int fd, void *buf, size_t count, int *cont)
|
|
{
|
|
ssize_t ret = read(fd, buf, count);
|
|
if (ret > 0) {
|
|
if (!resume_in_progress) {
|
|
read_bytes += ret;
|
|
}
|
|
} else if (out_buf_size > 0 &&
|
|
(ret == 0 || (ret == -1 && (errno == ECONNRESET
|
|
|| errno == ECONNABORTED || errno == ETIMEDOUT
|
|
|| errno == EHOSTUNREACH)))) {
|
|
debug("roaming_read failed for %d ret=%ld errno=%d",
|
|
fd, (long)ret, errno);
|
|
ret = 0;
|
|
if (wait_for_roaming_reconnect() == 0)
|
|
*cont = 1;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
size_t
|
|
roaming_atomicio(ssize_t(*f)(int, void*, size_t), int fd, void *buf,
|
|
size_t count)
|
|
{
|
|
size_t ret = atomicio(f, fd, buf, count);
|
|
|
|
if (f == vwrite && ret > 0 && !resume_in_progress) {
|
|
write_bytes += ret;
|
|
} else if (f == read && ret > 0 && !resume_in_progress) {
|
|
read_bytes += ret;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
resend_bytes(int fd, u_int64_t *offset)
|
|
{
|
|
size_t available, needed;
|
|
|
|
if (out_start < out_last)
|
|
available = out_last - out_start;
|
|
else
|
|
available = out_buf_size;
|
|
needed = write_bytes - *offset;
|
|
debug3("resend_bytes: resend %lu bytes from %llu",
|
|
(unsigned long)needed, (unsigned long long)*offset);
|
|
if (needed > available)
|
|
fatal("Needed to resend more data than in the cache");
|
|
if (out_last < needed) {
|
|
int chunkend = needed - out_last;
|
|
atomicio(vwrite, fd, out_buf + out_buf_size - chunkend,
|
|
chunkend);
|
|
atomicio(vwrite, fd, out_buf, out_last);
|
|
} else {
|
|
atomicio(vwrite, fd, out_buf + (out_last - needed), needed);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Caclulate a new key after a reconnect
|
|
*/
|
|
void
|
|
calculate_new_key(u_int64_t *key, u_int64_t cookie, u_int64_t challenge)
|
|
{
|
|
u_char hash[SSH_DIGEST_MAX_LENGTH];
|
|
Buffer b;
|
|
|
|
buffer_init(&b);
|
|
buffer_put_int64(&b, *key);
|
|
buffer_put_int64(&b, cookie);
|
|
buffer_put_int64(&b, challenge);
|
|
|
|
if (ssh_digest_buffer(SSH_DIGEST_SHA1, &b, hash, sizeof(hash)) != 0)
|
|
fatal("%s: digest_buffer failed", __func__);
|
|
|
|
buffer_clear(&b);
|
|
buffer_append(&b, hash, ssh_digest_bytes(SSH_DIGEST_SHA1));
|
|
*key = buffer_get_int64(&b);
|
|
buffer_free(&b);
|
|
}
|