885 lines
21 KiB
C
885 lines
21 KiB
C
/*
|
|
* This file is part of the Sofia-SIP package
|
|
*
|
|
* Copyright (C) 2006 Nokia Corporation.
|
|
*
|
|
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
* as published by the Free Software Foundation; either version 2.1 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
/**@CFILE tport_sigcomp.c Transport using SigComp.
|
|
*
|
|
* Incomplete.
|
|
*
|
|
* See tport.docs for more detailed description of tport interface.
|
|
*
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
|
|
* @author Martti Mela <Martti.Mela@nokia.com>
|
|
*
|
|
* @date Created: Fri Mar 24 08:45:49 EET 2006 ppessi
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "tport.h"
|
|
|
|
#include <sofia-sip/su_string.h>
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
|
|
#include <sigcomp.h>
|
|
|
|
/* ---------------------------------------------------------------------- */
|
|
/* SigComp */
|
|
|
|
typedef struct tport_sigcomp_handler tport_sigcomp_handler_t;
|
|
|
|
/** @internal Per end-point SigComp data */
|
|
struct tport_compressor {
|
|
struct sigcomp_state_handler *msc_state_handler;
|
|
struct sigcomp_compartment *msc_compartment;
|
|
};
|
|
|
|
/** @internal Per-socket SigComp data */
|
|
struct tport_comp {
|
|
struct sigcomp_udvm *sc_udvm;
|
|
struct sigcomp_compartment *sc_cc;
|
|
struct sigcomp_compressor *sc_compressor;
|
|
struct sigcomp_buffer *sc_output;
|
|
unsigned sc_compressed;
|
|
|
|
struct sigcomp_buffer *sc_input;
|
|
unsigned sc_copied;
|
|
|
|
enum {
|
|
format_is_unknown,
|
|
format_is_sigcomp,
|
|
format_is_noncomp
|
|
} sc_infmt, sc_outfmt;
|
|
};
|
|
|
|
|
|
tport_compressor_t *vsc_master_init_sigcomp(tport_t *mr,
|
|
char const *algorithm)
|
|
{
|
|
tport_compressor_t *retval = NLL;
|
|
stuct sigcomp_state_handler *sh = NULL;
|
|
struct sigcomp_algorithm *a = NULL;
|
|
struct sigcomp_compartment *cc = NULL;
|
|
|
|
if (algorithm == NULL)
|
|
algorithm = getenv("SIGCOMP_ALGORITHM");
|
|
|
|
retval = su_zalloc((su_home_t *)mr, sizeof *retval);
|
|
if (retval == NULL)
|
|
return retval;
|
|
|
|
sh = sigcomp_state_handler_create();
|
|
|
|
if (sh == NULL) {
|
|
SU_DEBUG_1(("tport: initializing SigComp state handler: %s\n",
|
|
strerror(errno)));
|
|
vsc_master_deinit_sigcomp(mr, retval), retval = NULL;
|
|
return retval;
|
|
}
|
|
retval->msc_state_handler = sh;
|
|
|
|
a = sigcomp_algorithm_by_name(algorithm);
|
|
cc = sigcomp_compartment_create(a, sh, 0, "", 0, NULL, 0);
|
|
|
|
if (cc == NULL) {
|
|
SU_DEBUG_1(("tport: initializing SigComp master compartment: %s\n",
|
|
strerror(errno)));
|
|
vsc_master_deinit_sigcomp(mr, retval), retval = NULL;
|
|
return retval;
|
|
}
|
|
|
|
self->msc_compartment = cc;
|
|
|
|
return retval;
|
|
}
|
|
|
|
if (self->sa_compartment) {
|
|
agent_sigcomp_options(self, self->sa_compartment);
|
|
sigcomp_compartment_option(self->sa_compartment, "stateless");
|
|
}
|
|
else
|
|
|
|
if (mr->mr_compartment)
|
|
sigcomp_compartment_unref(mt->mr_compartment);
|
|
mr->mr_compartment = sigcomp_compartment_ref(cc);
|
|
|
|
return cc;
|
|
}
|
|
|
|
|
|
void vsc_master_deinit_sigcomp(tport_master_t *mr)
|
|
{
|
|
tport_sigcomp_vtable_t const *vsc = tport_sigcomp_vtable;
|
|
|
|
if (mr->mr_compartment)
|
|
sigcomp_compartment_unref(mr->mr_compartment), mr->mr_compartment = NULL;
|
|
|
|
}
|
|
|
|
char const *vsc_comp_name(tport_sigcomp_t const *master_sc,
|
|
char const *proposed_name,
|
|
tagi_t const *tags)
|
|
{
|
|
if (master_sc == NULL ||
|
|
master_sc->sc_cc == NULL ||
|
|
!su_casematch(compression, tport_sigcomp_name))
|
|
return NULL;
|
|
|
|
return tport_sigcomp_name;
|
|
}
|
|
|
|
/** Check if transport can receive compressed messages */
|
|
int vsc_can_recv_sigcomp(tport_sigcomp_t const *sc)
|
|
{
|
|
return sc && sc->sc_infmt != format_is_noncomp;
|
|
}
|
|
|
|
/** Check if transport can send compressed messages */
|
|
int vsc_can_send_sigcomp(tport_sigcomp_t const *sc)
|
|
{
|
|
return sc && sc->sc_outfmt != format_is_noncomp;
|
|
}
|
|
|
|
/** Set/reset compression */
|
|
int vsc_set_compression(tport_t *self,
|
|
tport_sigcomp_t *sc,
|
|
char const *comp)
|
|
{
|
|
if (self == NULL)
|
|
;
|
|
else if (comp == NULL) {
|
|
if (sc == NULL || sc->sc_outfmt != format_is_sigcomp) {
|
|
self->tp_name->tpn_comp = NULL;
|
|
return 0;
|
|
}
|
|
}
|
|
else {
|
|
comp = tport_canonize_comp(comp);
|
|
|
|
if (comp && sc && sc->sc_outfmt != format_is_noncomp) {
|
|
self->tp_name->tpn_comp = comp;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
/** Assign a SigComp compartment (to a possibly connected tport). */
|
|
int tport_sigcomp_assign(tport_t *self, struct sigcomp_compartment *cc)
|
|
{
|
|
if (tport_is_connection_oriented(self) &&
|
|
tport_is_secondary(self) &&
|
|
self->tp_socket != INVALID_SOCKET) {
|
|
|
|
if (self->tp_sigcomp->sc_cc) {
|
|
if (cc == self->tp_sigcomp->sc_cc)
|
|
return 0;
|
|
|
|
/* Remove old assignment */
|
|
sigcomp_compartment_unref(self->tp_sigcomp->sc_cc);
|
|
}
|
|
|
|
self->tp_sigcomp->sc_cc = sigcomp_compartment_ref(cc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
return su_seterrno(EINVAL);
|
|
}
|
|
|
|
|
|
void vsc_sigcomp_shutdown(tport_t *self, int how)
|
|
{
|
|
if (self->tp_socket != -1)
|
|
shutdown(self->tp_socket, how - 1);
|
|
|
|
if (how >= 2 && self->tp_sigcomp->sc_cc) {
|
|
sigcomp_compartment_unref(self->tp_sigcomp->sc_cc);
|
|
self->tp_sigcomp->sc_cc = NULL;
|
|
}
|
|
}
|
|
|
|
static int vsc_recv_sigcomp_r(tport_t*, msg_t**, struct sigcomp_udvm*, int);
|
|
static struct sigcomp_compartment *vsc_primary_compartment(tport_master_t *);
|
|
|
|
|
|
static int tport_sigcomp_init_secondary()
|
|
{
|
|
if (accept) {
|
|
if (!pri->pri_primary->tp_name->tpn_comp)
|
|
self->tp_sigcomp->sc_infmt = format_is_noncomp;
|
|
}
|
|
else {
|
|
/* XXX - no tpn here */
|
|
if (tpn->tpn_comp == pri->pri_primary->tp_name->tpn_comp)
|
|
self->tp_name->tpn_comp = pri->pri_primary->tp_name->tpn_comp;
|
|
if (!pri->pri_primary->tp_name->tpn_comp)
|
|
self->tp_sigcomp->sc_infmt = format_is_noncomp;
|
|
}
|
|
}
|
|
|
|
static int tport_sigcomp_deinit_secondary(tport_t *self)
|
|
{
|
|
if (self->tp_sigcomp) {
|
|
tport_sigcomp_t *sc = self->tp_sigcomp;
|
|
|
|
if (sc->sc_udvm)
|
|
sigcomp_udvm_free(sc->sc_udvm), sc->sc_udvm = NULL;
|
|
if (sc->sc_compressor)
|
|
sigcomp_compressor_free(sc->sc_compressor), sc->sc_compressor = NULL;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
#if HAVE_SIGCOMP
|
|
if (tport_can_recv_sigcomp(self))
|
|
return tport_recv_sigcomp_stream(self);
|
|
#endif
|
|
#endif
|
|
|
|
/** Try to receive stream data using SigComp. */
|
|
static int tport_recv_sigcomp_stream(tport_t *self)
|
|
{
|
|
struct sigcomp_udvm *udvm;
|
|
int retval;
|
|
|
|
SU_DEBUG_7(("%s(%p)\n", __func__, self));
|
|
|
|
/* Peek for first byte in stream,
|
|
determine if this is a compressed stream or not */
|
|
if (self->tp_sigcomp->sc_infmt == format_is_unknown) {
|
|
unsigned char sample;
|
|
int n;
|
|
|
|
n = recv(self->tp_socket, &sample, 1, MSG_PEEK);
|
|
if (n < 0)
|
|
return n;
|
|
|
|
if (n == 0) {
|
|
recv(self->tp_socket, &sample, 1, 0);
|
|
return 0; /* E-o-S from first message */
|
|
}
|
|
|
|
if ((sample & 0xf8) != 0xf8) {
|
|
/* Not SigComp, receive as usual */
|
|
if (tport_is_primary(self)) {
|
|
SU_DEBUG_1(("%s(%p): receive semantics not implemented\n",
|
|
__func__, self));
|
|
su_seterrno(EINVAL); /* Internal error */
|
|
return -1;
|
|
}
|
|
|
|
/* Do not try to receive with sigcomp from this socket */
|
|
self->tp_sigcomp->sc_infmt = format_is_noncomp;
|
|
|
|
return tport_recv_stream(self);
|
|
}
|
|
|
|
/* SigComp, receive using UDVM */
|
|
self->tp_sigcomp->sc_infmt = format_is_sigcomp;
|
|
|
|
/* Initialize UDVM */
|
|
self->tp_sigcomp->sc_udvm = tport_init_udvm(self);
|
|
if (!self->tp_sigcomp->sc_udvm) {
|
|
int save = su_errno();
|
|
recv(self->tp_socket, &sample, 1, 0); /* remove msg from socket */
|
|
return su_seterrno(save);
|
|
}
|
|
}
|
|
|
|
udvm = self->tp_sigcomp->sc_udvm; assert(udvm);
|
|
|
|
retval = tport_recv_sigcomp_r(self, &self->tp_msg, udvm, N);
|
|
|
|
if (retval < 0)
|
|
sigcomp_udvm_reject(udvm);
|
|
|
|
return retval;
|
|
}
|
|
|
|
/** Receive data available on the socket.
|
|
*
|
|
* @retval -1 error
|
|
* @retval 0 end-of-stream
|
|
* @retval 1 normal receive
|
|
* @retval 2 incomplete recv, recv again
|
|
*/
|
|
static int tport_recv_sigcomp_r(tport_t *self,
|
|
msg_t **mmsg,
|
|
struct sigcomp_udvm *udvm,
|
|
int N)
|
|
{
|
|
msg_t *msg;
|
|
size_t n, m, i;
|
|
int eos, complete;
|
|
ssize_t veclen;
|
|
msg_iovec_t iovec[msg_n_fragments] = {{ 0 }};
|
|
su_sockaddr_t su[1];
|
|
socklen_t su_size = sizeof(su);
|
|
struct sigcomp_buffer *input, *output;
|
|
void *data;
|
|
size_t dlen;
|
|
|
|
SU_DEBUG_7(("%s(%p)\n", __func__, self));
|
|
|
|
assert(udvm);
|
|
|
|
if (sigcomp_udvm_has_input(udvm)) {
|
|
input = sigcomp_udvm_input_buffer(udvm, n = N = 0); assert(input);
|
|
}
|
|
else {
|
|
if (N == 0) {
|
|
assert(self->tp_addrinfo->ai_socktype != SOCK_DGRAM);
|
|
if (self->tp_addrinfo->ai_socktype == SOCK_DGRAM) {
|
|
recv(self->tp_socket, (void *)su, 1, 0);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
input = sigcomp_udvm_input_buffer(udvm, N); assert(input);
|
|
if (input == NULL)
|
|
return su_seterrno(EIO);
|
|
|
|
data = input->b_data + input->b_avail;
|
|
dlen = input->b_size - input->b_avail;
|
|
|
|
if (tport_is_stream(self)) {
|
|
n = recv(self->tp_socket, data, dlen, 0);
|
|
}
|
|
else if (dlen >= N) {
|
|
n = recvfrom(self->tp_socket, data, dlen, 0, &su->su_sa, &su_size);
|
|
SU_CANONIZE_SOCKADDR(su);
|
|
}
|
|
else {
|
|
recvfrom(self->tp_socket, data, dlen, 0, &su->su_sa, &su_size);
|
|
SU_CANONIZE_SOCKADDR(su);
|
|
su_seterrno(EMSGSIZE); /* Protocol error */
|
|
return -1;
|
|
}
|
|
|
|
if (n == (unsigned)-1) {
|
|
char const *pn = self->tp_protoname;
|
|
int err = su_errno();
|
|
|
|
if (su_is_blocking(err)) {
|
|
SU_DEBUG_7(("%s(%p): recv from %s: EAGAIN\n", __func__, self, pn));
|
|
return 1;
|
|
}
|
|
|
|
SU_DEBUG_1(("%s(%p): recv from %s: %s (%d)\n", __func__, self, pn,
|
|
su_strerror(err), err));
|
|
return -1;
|
|
}
|
|
|
|
/* XXX - in case of stream, use message buffers for output? */
|
|
|
|
input->b_avail += n;
|
|
input->b_complete = (n == 0) || !tport_is_stream(self);
|
|
}
|
|
|
|
for (complete = eos = 0; !complete;) {
|
|
output = sigcomp_udvm_output_buffer(udvm, 16384);
|
|
|
|
if (sigcomp_udvm_decompress(udvm, output, input) < 0) {
|
|
int error = sigcomp_udvm_errno(udvm);
|
|
|
|
SU_DEBUG_3(("%s: UDVM error %d: %s\n", __func__,
|
|
error, sigcomp_udvm_strerror(udvm)));
|
|
|
|
su_seterrno(EREMOTEIO);
|
|
|
|
return -1;
|
|
}
|
|
|
|
data = output->b_data + output->b_used;
|
|
dlen = output->b_avail - output->b_used;
|
|
complete = output->b_complete;
|
|
eos = complete && input->b_complete;
|
|
|
|
veclen = tport_recv_iovec(self, mmsg, iovec, dlen, eos);
|
|
|
|
if (dlen ? veclen <= 0 : veclen < 0) {
|
|
return -1;
|
|
}
|
|
|
|
for (i = 0, n = 0; i < veclen; i++) {
|
|
m = iovec[i].mv_len; assert(dlen >= n + m);
|
|
memcpy(iovec[i].mv_base, data + n, m);
|
|
n += m;
|
|
}
|
|
assert(dlen == n);
|
|
|
|
msg = *mmsg;
|
|
|
|
if (msg) {
|
|
/* Message address */
|
|
if (self->tp_addrinfo->ai_socktype == SOCK_STREAM)
|
|
msg_set_address(msg, self->tp_addr, self->tp_addrlen);
|
|
else
|
|
msg_set_address(msg, su, su_size);
|
|
|
|
SU_DEBUG_5(("%s(%p): sigcomp recv = %u => %u %s\n", __func__, self,
|
|
N, dlen, eos ? " (complete)" : ""));
|
|
|
|
msg_mark_as_compressed(msg);
|
|
|
|
/* Write the received data to the message dump file */
|
|
if (self->tp_master->mr_dump_file && !self->tp_pri->pri_threadpool)
|
|
tport_dump_iovec(self, msg, n, iovec, veclen, "recv", "from");
|
|
|
|
msg_recv_commit(msg, dlen, eos); /* Mark buffer as used */
|
|
}
|
|
else {
|
|
SU_DEBUG_5(("%s(%p): sigcomp recv = %u => %u %s\n", __func__, self,
|
|
N, dlen, eos ? " (complete)" : ""));
|
|
if (complete || !tport_is_stream(self)) {
|
|
sigcomp_udvm_reject(udvm); /* Reject empty message */
|
|
}
|
|
}
|
|
|
|
if (self->tp_addrinfo->ai_socktype == SOCK_STREAM) {
|
|
if (eos)
|
|
return 0;
|
|
|
|
if (output->b_complete)
|
|
return n < N || sigcomp_udvm_has_pending_data(udvm) ? 2 : 1;
|
|
|
|
if (!sigcomp_udvm_has_input(udvm))
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return eos ? 0 : 2;
|
|
}
|
|
|
|
static
|
|
int vsc_send_sigcomp(tport_t const *self,
|
|
msg_t *msg,
|
|
msg_iovec_t iov[],
|
|
int iovused,
|
|
struct sigcomp_compartment *cc,
|
|
tport_sigcomp_t *sc)
|
|
{
|
|
struct sigcomp_compressor *c = sc->sc_compressor;
|
|
struct sigcomp_buffer *input = sc->sc_input;
|
|
struct sigcomp_buffer *output = sc->sc_output;
|
|
msg_iovec_t ciov[1];
|
|
|
|
int i, n, m, k, stream = tport_is_stream(self);
|
|
char const *ccname;
|
|
int ccnamelen;
|
|
|
|
su_addrinfo_t *ai = msg_addrinfo(msg);
|
|
|
|
int compress = (cc || (cc = sc->sc_cc)) && ai->ai_flags & TP_AI_COMPRESSED;
|
|
|
|
if (!compress) {
|
|
if (stream)
|
|
sc->sc_outfmt = format_is_noncomp;
|
|
ai->ai_flags &= ~TP_AI_COMPRESSED;
|
|
return self->tp_pri->pri_vtable->vtp_send(self, msg, iov, iovused, NULL);
|
|
}
|
|
|
|
if (stream)
|
|
sc->sc_outfmt = format_is_sigcomp;
|
|
|
|
assert(cc);
|
|
|
|
if (c == NULL) {
|
|
assert(input == NULL);
|
|
if (stream)
|
|
c = sigcomp_compressor_create_for_stream(cc);
|
|
else
|
|
c = sigcomp_compressor_create(cc);
|
|
sc->sc_compressor = c;
|
|
}
|
|
|
|
ccname = sigcomp_compartment_name(cc, &ccnamelen);
|
|
|
|
if (sc->sc_compressed != 0) {
|
|
input = NONE;
|
|
}
|
|
else if (input == NULL) {
|
|
int input_size = -1;
|
|
|
|
if (tport_is_udp(self)) {
|
|
input_size = 0;
|
|
|
|
for (i = 0; i < iovused; i++)
|
|
input_size += iov[i].siv_len;
|
|
}
|
|
|
|
sc->sc_input = input = sigcomp_compressor_input_buffer(c, input_size);
|
|
|
|
assert(input->b_avail == 0 && input->b_used == 0);
|
|
}
|
|
else if (!input->b_complete) {
|
|
int input_size = 0;
|
|
|
|
for (i = 0; i < iovused; i++)
|
|
input_size += iov[i].siv_len;
|
|
|
|
if (input_size > input->b_size - input->b_avail)
|
|
sigcomp_buffer_align_available(input, 0);
|
|
}
|
|
|
|
if (output == NULL)
|
|
sc->sc_output = output = sigcomp_compressor_output_buffer(c, NULL);
|
|
|
|
if (!c || !input || !output) {
|
|
SU_DEBUG_3(("%s(%p): %s (%u)%s%s%s\n",
|
|
__func__, self, strerror(errno), errno,
|
|
c ? "" : " (comp)",
|
|
input ? "" : " (input)",
|
|
output ? "" : " (output)"));
|
|
sigcomp_compressor_free(c);
|
|
sc->sc_compressor = NULL;
|
|
sc->sc_output = NULL; sc->sc_input = NULL;
|
|
sc->sc_compressed = 0; sc->sc_copied = 0;
|
|
return -1;
|
|
}
|
|
|
|
if (sc->sc_compressed == 0) {
|
|
k = sc->sc_copied;
|
|
|
|
if (!input->b_complete) {
|
|
int m = sc->sc_copied;
|
|
|
|
for (i = 0, n = 0; i < iovused; i++) {
|
|
char *b = iov[i].siv_base;
|
|
int l = iov[i].siv_len;
|
|
|
|
if (m >= l) {
|
|
m -= l;
|
|
continue;
|
|
}
|
|
|
|
b += m; l -= m;
|
|
|
|
if (input->b_size == input->b_avail)
|
|
break;
|
|
|
|
if (l > input->b_size - input->b_avail)
|
|
l = input->b_size - input->b_avail;
|
|
|
|
memcpy(input->b_data + input->b_avail, b, l);
|
|
input->b_avail += l; n += l; sc->sc_copied += l;
|
|
|
|
if (l != iov[i].siv_len)
|
|
break;
|
|
}
|
|
input->b_complete = i == iovused;
|
|
assert(stream || input->b_complete); (void)stream;
|
|
}
|
|
|
|
m = output->b_avail - output->b_used;
|
|
|
|
n = sigcomp_compressor_compress(c, output, input);
|
|
|
|
if (n < 0) {
|
|
SU_DEBUG_3(("%s(%p): %s (%u)\n", __func__, self,
|
|
sigcomp_compressor_strerror(c),
|
|
sigcomp_compressor_errno(c)));
|
|
sigcomp_compressor_free(c);
|
|
sc->sc_compressor = NULL;
|
|
sc->sc_output = NULL; sc->sc_input = NULL;
|
|
sc->sc_compressed = 0;
|
|
return -1;
|
|
}
|
|
|
|
assert(input->b_complete || sc->sc_copied - k > 0);
|
|
|
|
SU_DEBUG_5(("%s: input %u (%u new) compressed %u to %u with '%.*s'\n",
|
|
__func__, sc->sc_copied, k, n,
|
|
(output->b_avail - output->b_used) - m,
|
|
ccnamelen, ccname));
|
|
|
|
sc->sc_compressed = n;
|
|
|
|
assert(stream || output->b_complete);
|
|
}
|
|
else {
|
|
assert(tport_is_connection_oriented(self));
|
|
n = sc->sc_compressed;
|
|
}
|
|
|
|
assert(input && cc && c && output);
|
|
|
|
ciov->siv_base = output->b_data + output->b_used;
|
|
ciov->siv_len = output->b_avail - output->b_used;
|
|
|
|
m = self->tp_pri->pri_vtable->vtp_send(self, msg, ciov, 1);
|
|
|
|
if (m == -1) {
|
|
int error = su_errno();
|
|
|
|
if (su_is_blocking(error)) {
|
|
sigcomp_compressor_free(c);
|
|
sc->sc_compressor = NULL;
|
|
sc->sc_output = NULL; sc->sc_input = NULL;
|
|
sc->sc_compressed = 0; sc->sc_copied = 0;
|
|
su_seterrno(error);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
output->b_used += m;
|
|
|
|
if (output->b_used < output->b_avail)
|
|
return 0;
|
|
|
|
if (output->b_complete) {
|
|
sigcomp_compressor_accept(c, cc), sc->sc_output = output = NULL;
|
|
}
|
|
|
|
if (input != NONE && input->b_avail == input->b_used && input->b_complete)
|
|
sigcomp_buffer_reset(input, -1), sc->sc_input = input = NULL;
|
|
|
|
if (!input && !output) {
|
|
sigcomp_compressor_free(c);
|
|
sc->sc_compressor = NULL;
|
|
}
|
|
|
|
assert(sc->sc_compressed >= n); assert(sc->sc_copied >= n);
|
|
|
|
sc->sc_compressed -= n;
|
|
sc->sc_copied -= n;
|
|
|
|
return n;
|
|
}
|
|
|
|
/** Initialize UDVM */
|
|
static
|
|
struct sigcomp_udvm *tport_init_udvm(tport_t *self)
|
|
{
|
|
struct sigcomp_compartment *cc;
|
|
struct sigcomp_udvm *udvm;
|
|
|
|
if (self->tp_sigcomp->sc_udvm)
|
|
return self->tp_sigcomp->sc_udvm;
|
|
|
|
cc = tport_primary_compartment(self->tp_master);
|
|
|
|
if (!cc)
|
|
return NULL;
|
|
|
|
if (self->tp_addrinfo->ai_socktype == SOCK_STREAM)
|
|
udvm = sigcomp_udvm_create_for_stream(cc);
|
|
else
|
|
udvm = sigcomp_udvm_create_for_compartment(cc);
|
|
|
|
return udvm;
|
|
}
|
|
|
|
|
|
/** Get primary compartment */
|
|
static
|
|
struct sigcomp_compartment *
|
|
tport_primary_compartment(tport_master_t *mr)
|
|
{
|
|
return mr->mr_compartment;
|
|
}
|
|
|
|
/** Test if tport has a SigComp compartment is assigned to it. */
|
|
int vsc_has_sigcomp_assigned(tport_sigcomp_t const *sc)
|
|
{
|
|
return sc && sc->sc_udvm != NULL;
|
|
}
|
|
|
|
static
|
|
void vsc_try_accept_sigcomp(tport_t *self, msg_t *msg)
|
|
{
|
|
struct sigcomp_udvm *udvm;
|
|
|
|
udvm = self->tp_sigcomp->sc_udvm;
|
|
if (udvm && sigcomp_udvm_is_complete(udvm)) {
|
|
if (self->tp_master->mr_tpac->tpac_sigcomp_accept &&
|
|
self->tp_sigcomp->sc_cc == NULL) {
|
|
tport_t *ref;
|
|
struct tport_delivery *d;
|
|
|
|
d = self->tp_master->mr_delivery;
|
|
|
|
d->d_tport = self;
|
|
d->d_msg = msg;
|
|
d->d_udvm = &self->tp_sigcomp->sc_udvm;
|
|
*d->d_from = *self->tp_name;
|
|
|
|
ref = tport_incref(self);
|
|
STACK_SIGCOMP_ACCEPT(self, msg);
|
|
/* Reject by default */
|
|
if (self->tp_sigcomp->sc_udvm)
|
|
sigcomp_udvm_accept(self->tp_sigcomp->sc_udvm, NULL);
|
|
tport_decref(&ref);
|
|
|
|
d->d_msg = NULL;
|
|
}
|
|
else {
|
|
if (tport_log->log_level >= 5) {
|
|
char const *name;
|
|
int namelen;
|
|
|
|
name = sigcomp_compartment_name(self->tp_sigcomp->sc_cc, &namelen);
|
|
SU_DEBUG_5(("tport(%p): msg %p SigComp implicit accept '%.*s'\n",
|
|
self, msg, namelen, name));
|
|
}
|
|
sigcomp_udvm_accept(udvm, self->tp_sigcomp->sc_cc);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/** Accept a SigComp message */
|
|
int
|
|
tport_sigcomp_accept(tport_t *self,
|
|
struct sigcomp_compartment *cc,
|
|
msg_t *msg)
|
|
{
|
|
struct sigcomp_udvm *udvm;
|
|
|
|
if (self == NULL || msg != self->tp_master->mr_delivery->d_msg)
|
|
return su_seterrno(EINVAL);
|
|
|
|
if (!self->tp_master->mr_delivery->d_udvm || cc == NONE)
|
|
return 0;
|
|
|
|
udvm = *self->tp_master->mr_delivery->d_udvm;
|
|
|
|
if (udvm) {
|
|
if (tport_log->log_level >= 5) {
|
|
char const *name;
|
|
int namelen;
|
|
|
|
if (cc) {
|
|
name = sigcomp_compartment_name(cc, &namelen);
|
|
SU_DEBUG_5(("tport(%p): msg %p SigComp accept '%.*s'\n",
|
|
self, msg, namelen, name));
|
|
}
|
|
else {
|
|
SU_DEBUG_5(("tport(%p): msg %p SigComp reject\n", self, msg));
|
|
}
|
|
}
|
|
sigcomp_udvm_accept(udvm, cc);
|
|
}
|
|
|
|
self->tp_master->mr_delivery->d_udvm = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/** Pass message to the protocol stack */
|
|
void
|
|
tport_sigcomp_deliver(tport_t *self, msg_t *msg, su_time_t now)
|
|
{
|
|
/* XXX - no d */
|
|
|
|
STACK_RECV(self, msg, now);
|
|
|
|
if (d->d_udvm && *d->d_udvm)
|
|
sigcomp_udvm_accept(*d->d_udvm, NULL); /* reject */
|
|
}
|
|
|
|
|
|
#if HAVE_SIGCOMP && 0
|
|
|
|
su_inline
|
|
int msg_is_compressed(msg_t *msg)
|
|
{
|
|
return msg &&
|
|
(msg_addrinfo(msg)->ai_flags & TP_AI_COMPRESSED) == TP_AI_COMPRESSED;
|
|
}
|
|
|
|
su_inline
|
|
void msg_mark_as_compressed(msg_t *msg)
|
|
{
|
|
if (msg)
|
|
msg_addrinfo(msg)->ai_flags |= TP_AI_COMPRESSED;
|
|
}
|
|
|
|
|
|
struct sigcomp_udvm **tport_get_udvm_slot(tport_t *self)
|
|
{
|
|
tport_sigcomp_vtable_t const *vsc = tport_sigcomp_vtable;
|
|
|
|
if (vsc)
|
|
|
|
#if HAVE_SIGCOMP
|
|
return &self->tp_sigcomp->sc_udvm;
|
|
#else
|
|
return NULL;
|
|
#endif
|
|
}
|
|
|
|
struct sigcomp_compartment *
|
|
tport_sigcomp_assign_if_needed(tport_t *self,
|
|
struct sigcomp_compartment *cc)
|
|
{
|
|
if (self->tp_name->tpn_comp) {
|
|
if (cc)
|
|
tport_sigcomp_assign(self, cc);
|
|
else if (self->tp_sigcomp->sc_cc)
|
|
cc = self->tp_sigcomp->sc_cc;
|
|
else
|
|
/* Use default compartment */
|
|
cc = self->tp_master->mr_compartment;
|
|
}
|
|
else
|
|
cc = NULL;
|
|
|
|
return cc;
|
|
}
|
|
|
|
/** Receive data from datagram using SigComp. */
|
|
int tport_recv_sigcomp_dgram(tport_t *self, int N)
|
|
{
|
|
char dummy[1];
|
|
int error = EBADMSG;
|
|
#if HAVE_SIGCOMP
|
|
struct sigcomp_udvm *udvm;
|
|
|
|
if (self->tp_sigcomp->sc_udvm == 0)
|
|
self->tp_sigcomp->sc_udvm = tport_init_udvm(self);
|
|
|
|
udvm = self->tp_sigcomp->sc_udvm;
|
|
|
|
if (udvm) {
|
|
retval = tport_recv_sigcomp_r(self, &self->tp_msg, udvm, N);
|
|
if (retval < 0)
|
|
sigcomp_udvm_reject(udvm);
|
|
return retval;
|
|
}
|
|
error = su_errno();
|
|
#endif
|
|
recv(self->tp_socket, dummy, 1, 0); /* remove msg from socket */
|
|
/* XXX - send NACK ? */
|
|
return su_seterrno(error);
|
|
}
|