blob: 9b131137eb91e6f7433544337c17b2224473ac1e [file] [log] [blame]
/*
* OpenVPN -- An application to securely tunnel IP networks
* over a single UDP port, with support for SSL/TLS-based
* session authentication and key exchange,
* packet encryption, packet authentication, and
* packet compression.
*
* Copyright (C) 2002-2018 OpenVPN Inc <sales@openvpn.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#elif defined(_MSC_VER)
#include "config-msvc.h"
#endif
#include "syshead.h"
#ifdef USE_COMP
#include "comp.h"
#include "error.h"
#include "otime.h"
#include "memdbg.h"
struct compress_context *
comp_init(const struct compress_options *opt)
{
struct compress_context *compctx = NULL;
switch (opt->alg)
{
case COMP_ALG_STUB:
ALLOC_OBJ_CLEAR(compctx, struct compress_context);
compctx->flags = opt->flags;
compctx->alg = comp_stub_alg;
break;
case COMP_ALGV2_UNCOMPRESSED:
ALLOC_OBJ_CLEAR(compctx, struct compress_context);
compctx->flags = opt->flags;
compctx->alg = compv2_stub_alg;
break;
#ifdef ENABLE_LZO
case COMP_ALG_LZO:
ALLOC_OBJ_CLEAR(compctx, struct compress_context);
compctx->flags = opt->flags;
compctx->alg = lzo_alg;
break;
#endif
#ifdef ENABLE_LZ4
case COMP_ALG_LZ4:
ALLOC_OBJ_CLEAR(compctx, struct compress_context);
compctx->flags = opt->flags;
compctx->alg = lz4_alg;
break;
case COMP_ALGV2_LZ4:
ALLOC_OBJ_CLEAR(compctx, struct compress_context);
compctx->flags = opt->flags;
compctx->alg = lz4v2_alg;
break;
#endif
}
if (compctx)
{
(*compctx->alg.compress_init)(compctx);
}
return compctx;
}
/* In the v2 compression schemes, an uncompressed packet has
* has no opcode in front, unless the first byte is 0x50. In this
* case the packet needs to be escaped */
void
compv2_escape_data_ifneeded(struct buffer *buf)
{
uint8_t *head = BPTR(buf);
if (head[0] != COMP_ALGV2_INDICATOR_BYTE)
{
return;
}
/* Header is 0x50 */
ASSERT(buf_prepend(buf, 2));
head = BPTR(buf);
head[0] = COMP_ALGV2_INDICATOR_BYTE;
head[1] = COMP_ALGV2_UNCOMPRESSED;
}
void
comp_uninit(struct compress_context *compctx)
{
if (compctx)
{
(*compctx->alg.compress_uninit)(compctx);
free(compctx);
}
}
void
comp_add_to_extra_frame(struct frame *frame)
{
/* Leave room for our one-byte compressed/didn't-compress prefix byte. */
frame_add_to_extra_frame(frame, COMP_PREFIX_LEN);
}
void
comp_add_to_extra_buffer(struct frame *frame)
{
/* Leave room for compression buffer to expand in worst case scenario
* where data is totally incompressible */
frame_add_to_extra_buffer(frame, COMP_EXTRA_BUFFER(EXPANDED_SIZE(frame)));
}
void
comp_print_stats(const struct compress_context *compctx, struct status_output *so)
{
if (compctx)
{
status_printf(so, "pre-compress bytes," counter_format, compctx->pre_compress);
status_printf(so, "post-compress bytes," counter_format, compctx->post_compress);
status_printf(so, "pre-decompress bytes," counter_format, compctx->pre_decompress);
status_printf(so, "post-decompress bytes," counter_format, compctx->post_decompress);
}
}
/*
* Tell our peer which compression algorithms we support.
*/
void
comp_generate_peer_info_string(const struct compress_options *opt, struct buffer *out)
{
if (opt)
{
bool lzo_avail = false;
if (!(opt->flags & COMP_F_ADVERTISE_STUBS_ONLY))
{
#if defined(ENABLE_LZ4)
buf_printf(out, "IV_LZ4=1\n");
buf_printf(out, "IV_LZ4v2=1\n");
#endif
#if defined(ENABLE_LZO)
buf_printf(out, "IV_LZO=1\n");
lzo_avail = true;
#endif
}
if (!lzo_avail)
{
buf_printf(out, "IV_LZO_STUB=1\n");
}
buf_printf(out, "IV_COMP_STUB=1\n");
buf_printf(out, "IV_COMP_STUBv2=1\n");
buf_printf(out, "IV_TCPNL=1\n");
}
}
#endif /* USE_COMP */