153a5a1b3Sopenharmony_ci/***
253a5a1b3Sopenharmony_ci  This file is part of PulseAudio.
353a5a1b3Sopenharmony_ci
453a5a1b3Sopenharmony_ci  Copyright 2004-2006 Lennart Poettering
553a5a1b3Sopenharmony_ci
653a5a1b3Sopenharmony_ci  PulseAudio is free software; you can redistribute it and/or modify
753a5a1b3Sopenharmony_ci  it under the terms of the GNU Lesser General Public License as
853a5a1b3Sopenharmony_ci  published by the Free Software Foundation; either version 2.1 of the
953a5a1b3Sopenharmony_ci  License, or (at your option) any later version.
1053a5a1b3Sopenharmony_ci
1153a5a1b3Sopenharmony_ci  PulseAudio is distributed in the hope that it will be useful, but
1253a5a1b3Sopenharmony_ci  WITHOUT ANY WARRANTY; without even the implied warranty of
1353a5a1b3Sopenharmony_ci  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1453a5a1b3Sopenharmony_ci  Lesser General Public License for more details.
1553a5a1b3Sopenharmony_ci
1653a5a1b3Sopenharmony_ci  You should have received a copy of the GNU Lesser General Public
1753a5a1b3Sopenharmony_ci  License along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
1853a5a1b3Sopenharmony_ci***/
1953a5a1b3Sopenharmony_ci
2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H
2153a5a1b3Sopenharmony_ci#include <config.h>
2253a5a1b3Sopenharmony_ci#endif
2353a5a1b3Sopenharmony_ci
2453a5a1b3Sopenharmony_ci#include <stdlib.h>
2553a5a1b3Sopenharmony_ci
2653a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h>
2753a5a1b3Sopenharmony_ci#include <pulsecore/macro.h>
2853a5a1b3Sopenharmony_ci#include <pulsecore/refcnt.h>
2953a5a1b3Sopenharmony_ci#include <pulsecore/flist.h>
3053a5a1b3Sopenharmony_ci
3153a5a1b3Sopenharmony_ci#include "packet.h"
3253a5a1b3Sopenharmony_ci
3353a5a1b3Sopenharmony_ci#define MAX_APPENDED_SIZE 128
3453a5a1b3Sopenharmony_ci
3553a5a1b3Sopenharmony_cistruct pa_packet {
3653a5a1b3Sopenharmony_ci    PA_REFCNT_DECLARE;
3753a5a1b3Sopenharmony_ci    enum { PA_PACKET_APPENDED, PA_PACKET_DYNAMIC } type;
3853a5a1b3Sopenharmony_ci    size_t length;
3953a5a1b3Sopenharmony_ci    uint8_t *data;
4053a5a1b3Sopenharmony_ci    union {
4153a5a1b3Sopenharmony_ci        uint8_t appended[MAX_APPENDED_SIZE];
4253a5a1b3Sopenharmony_ci    } per_type;
4353a5a1b3Sopenharmony_ci};
4453a5a1b3Sopenharmony_ci
4553a5a1b3Sopenharmony_ciPA_STATIC_FLIST_DECLARE(packets, 0, pa_xfree);
4653a5a1b3Sopenharmony_ci
4753a5a1b3Sopenharmony_cipa_packet* pa_packet_new(size_t length) {
4853a5a1b3Sopenharmony_ci    pa_packet *p;
4953a5a1b3Sopenharmony_ci
5053a5a1b3Sopenharmony_ci    pa_assert(length > 0);
5153a5a1b3Sopenharmony_ci
5253a5a1b3Sopenharmony_ci    if (!(p = pa_flist_pop(PA_STATIC_FLIST_GET(packets))))
5353a5a1b3Sopenharmony_ci        p = pa_xnew(pa_packet, 1);
5453a5a1b3Sopenharmony_ci    PA_REFCNT_INIT(p);
5553a5a1b3Sopenharmony_ci    p->length = length;
5653a5a1b3Sopenharmony_ci    if (length > MAX_APPENDED_SIZE) {
5753a5a1b3Sopenharmony_ci        p->data = pa_xmalloc(length);
5853a5a1b3Sopenharmony_ci        p->type = PA_PACKET_DYNAMIC;
5953a5a1b3Sopenharmony_ci    } else {
6053a5a1b3Sopenharmony_ci        p->data = p->per_type.appended;
6153a5a1b3Sopenharmony_ci        p->type = PA_PACKET_APPENDED;
6253a5a1b3Sopenharmony_ci    }
6353a5a1b3Sopenharmony_ci
6453a5a1b3Sopenharmony_ci    return p;
6553a5a1b3Sopenharmony_ci}
6653a5a1b3Sopenharmony_ci
6753a5a1b3Sopenharmony_cipa_packet* pa_packet_new_data(const void* data, size_t length) {
6853a5a1b3Sopenharmony_ci    pa_packet *p = pa_packet_new(length);
6953a5a1b3Sopenharmony_ci
7053a5a1b3Sopenharmony_ci    pa_assert(data);
7153a5a1b3Sopenharmony_ci    pa_assert(length > 0);
7253a5a1b3Sopenharmony_ci
7353a5a1b3Sopenharmony_ci    memcpy(p->data, data, length);
7453a5a1b3Sopenharmony_ci
7553a5a1b3Sopenharmony_ci    return p;
7653a5a1b3Sopenharmony_ci}
7753a5a1b3Sopenharmony_ci
7853a5a1b3Sopenharmony_cipa_packet* pa_packet_new_dynamic(void* data, size_t length) {
7953a5a1b3Sopenharmony_ci    pa_packet *p;
8053a5a1b3Sopenharmony_ci
8153a5a1b3Sopenharmony_ci    pa_assert(data);
8253a5a1b3Sopenharmony_ci    pa_assert(length > 0);
8353a5a1b3Sopenharmony_ci
8453a5a1b3Sopenharmony_ci    if (!(p = pa_flist_pop(PA_STATIC_FLIST_GET(packets))))
8553a5a1b3Sopenharmony_ci        p = pa_xnew(pa_packet, 1);
8653a5a1b3Sopenharmony_ci    PA_REFCNT_INIT(p);
8753a5a1b3Sopenharmony_ci    p->length = length;
8853a5a1b3Sopenharmony_ci    p->data = data;
8953a5a1b3Sopenharmony_ci    p->type = PA_PACKET_DYNAMIC;
9053a5a1b3Sopenharmony_ci
9153a5a1b3Sopenharmony_ci    return p;
9253a5a1b3Sopenharmony_ci}
9353a5a1b3Sopenharmony_ci
9453a5a1b3Sopenharmony_ciconst void* pa_packet_data(pa_packet *p, size_t *l) {
9553a5a1b3Sopenharmony_ci    pa_assert(PA_REFCNT_VALUE(p) >= 1);
9653a5a1b3Sopenharmony_ci    pa_assert(p->data);
9753a5a1b3Sopenharmony_ci    pa_assert(l);
9853a5a1b3Sopenharmony_ci
9953a5a1b3Sopenharmony_ci    *l = p->length;
10053a5a1b3Sopenharmony_ci
10153a5a1b3Sopenharmony_ci    return p->data;
10253a5a1b3Sopenharmony_ci}
10353a5a1b3Sopenharmony_ci
10453a5a1b3Sopenharmony_cipa_packet* pa_packet_ref(pa_packet *p) {
10553a5a1b3Sopenharmony_ci    pa_assert(p);
10653a5a1b3Sopenharmony_ci    pa_assert(PA_REFCNT_VALUE(p) >= 1);
10753a5a1b3Sopenharmony_ci
10853a5a1b3Sopenharmony_ci    PA_REFCNT_INC(p);
10953a5a1b3Sopenharmony_ci    return p;
11053a5a1b3Sopenharmony_ci}
11153a5a1b3Sopenharmony_ci
11253a5a1b3Sopenharmony_civoid pa_packet_unref(pa_packet *p) {
11353a5a1b3Sopenharmony_ci    pa_assert(p);
11453a5a1b3Sopenharmony_ci    pa_assert(PA_REFCNT_VALUE(p) >= 1);
11553a5a1b3Sopenharmony_ci
11653a5a1b3Sopenharmony_ci    if (PA_REFCNT_DEC(p) <= 0) {
11753a5a1b3Sopenharmony_ci        if (p->type == PA_PACKET_DYNAMIC)
11853a5a1b3Sopenharmony_ci            pa_xfree(p->data);
11953a5a1b3Sopenharmony_ci        if (pa_flist_push(PA_STATIC_FLIST_GET(packets), p) < 0)
12053a5a1b3Sopenharmony_ci            pa_xfree(p);
12153a5a1b3Sopenharmony_ci    }
12253a5a1b3Sopenharmony_ci}
123