mirror of
				https://github.com/PowerShell/Win32-OpenSSH.git
				synced 2025-10-25 01:34:02 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			242 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			242 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* $OpenBSD: roaming_common.c,v 1.13 2015/01/27 12:54:06 okan 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>
 | |
| #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);
 | |
| }
 |