tvheadend/src/packet.c
2010-11-29 19:25:24 +00:00

235 lines
3.9 KiB
C

/**
* Packet management
* Copyright (C) 2008 Andreas Öman
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* 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, see <http://www.gnu.org/licenses/>.
*/
#include "tvheadend.h"
#include "packet.h"
#include "string.h"
#include "atomic.h"
/*
*
*/
static void
pkt_destroy(th_pkt_t *pkt)
{
if(pkt->pkt_payload != NULL)
pktbuf_ref_dec(pkt->pkt_payload);
if(pkt->pkt_header != NULL)
pktbuf_ref_dec(pkt->pkt_header);
free(pkt);
}
/**
* Allocate a new packet and give it a refcount of one (which caller is
* suppoed to take care of)
*/
th_pkt_t *
pkt_alloc(const void *data, size_t datalen, int64_t pts, int64_t dts)
{
th_pkt_t *pkt;
pkt = calloc(1, sizeof(th_pkt_t));
if(datalen)
pkt->pkt_payload = pktbuf_alloc(data, datalen);
pkt->pkt_dts = dts;
pkt->pkt_pts = pts;
pkt->pkt_refcount = 1;
return pkt;
}
/**
*
*/
void
pkt_ref_dec(th_pkt_t *pkt)
{
if((atomic_add(&pkt->pkt_refcount, -1)) == 1)
pkt_destroy(pkt);
}
/**
*
*/
void
pkt_ref_inc(th_pkt_t *pkt)
{
atomic_add(&pkt->pkt_refcount, 1);
}
/**
*
*/
void
pkt_ref_inc_poly(th_pkt_t *pkt, int n)
{
atomic_add(&pkt->pkt_refcount, n);
}
/**
*
*/
void
pktref_clear_queue(struct th_pktref_queue *q)
{
th_pktref_t *pr;
while((pr = TAILQ_FIRST(q)) != NULL) {
TAILQ_REMOVE(q, pr, pr_link);
pkt_ref_dec(pr->pr_pkt);
free(pr);
}
}
/**
* Reference count is transfered to queue
*/
void
pktref_enqueue(struct th_pktref_queue *q, th_pkt_t *pkt)
{
th_pktref_t *pr = malloc(sizeof(th_pktref_t));
pr->pr_pkt = pkt;
TAILQ_INSERT_TAIL(q, pr, pr_link);
}
/**
*
*/
void
pktref_remove(struct th_pktref_queue *q, th_pktref_t *pr)
{
TAILQ_REMOVE(q, pr, pr_link);
pkt_ref_dec(pr->pr_pkt);
free(pr);
}
/**
*
*/
th_pkt_t *
pkt_merge_header(th_pkt_t *pkt)
{
th_pkt_t *n;
size_t s;
if(pkt->pkt_header == NULL)
return pkt;
n = malloc(sizeof(th_pkt_t));
*n = *pkt;
n->pkt_refcount = 1;
n->pkt_header = NULL;
s = pktbuf_len(pkt->pkt_payload) + pktbuf_len(pkt->pkt_header);
n->pkt_payload = pktbuf_alloc(NULL, s);
memcpy(pktbuf_ptr(n->pkt_payload),
pktbuf_ptr(pkt->pkt_header),
pktbuf_len(pkt->pkt_header));
memcpy(pktbuf_ptr(n->pkt_payload) + pktbuf_len(pkt->pkt_header),
pktbuf_ptr(pkt->pkt_payload),
pktbuf_len(pkt->pkt_payload));
pkt_ref_dec(pkt);
return n;
}
/**
*
*/
th_pkt_t *
pkt_copy_shallow(th_pkt_t *pkt)
{
th_pkt_t *n = malloc(sizeof(th_pkt_t));
*n = *pkt;
n->pkt_refcount = 1;
if(n->pkt_header)
pktbuf_ref_inc(n->pkt_header);
if(n->pkt_payload)
pktbuf_ref_inc(n->pkt_payload);
return n;
}
/**
*
*/
th_pktref_t *
pktref_create(th_pkt_t *pkt)
{
th_pktref_t *pr = malloc(sizeof(th_pktref_t));
pr->pr_pkt = pkt;
return pr;
}
void
pktbuf_ref_dec(pktbuf_t *pb)
{
if((atomic_add(&pb->pb_refcount, -1)) == 1) {
free(pb->pb_data);
free(pb);
}
}
void
pktbuf_ref_inc(pktbuf_t *pb)
{
atomic_add(&pb->pb_refcount, 1);
}
pktbuf_t *
pktbuf_alloc(const void *data, size_t size)
{
pktbuf_t *pb = malloc(sizeof(pktbuf_t));
pb->pb_refcount = 1;
pb->pb_size = size;
if(size > 0) {
pb->pb_data = malloc(size);
if(data != NULL)
memcpy(pb->pb_data, data, size);
}
return pb;
}
pktbuf_t *
pktbuf_make(void *data, size_t size)
{
pktbuf_t *pb = malloc(sizeof(pktbuf_t));
pb->pb_refcount = 1;
pb->pb_size = size;
pb->pb_data = data;
return pb;
}