2007-02-09 10:06:53 +00:00
|
|
|
/* Copyright (C) 2002 Jean-Marc Valin */
|
|
|
|
/**
|
|
|
|
@file speex_jitter.h
|
|
|
|
@brief Adaptive jitter buffer for Speex
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
|
|
|
|
- Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
|
|
|
|
- Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
- Neither the name of the Xiph.org Foundation nor the names of its
|
|
|
|
contributors may be used to endorse or promote products derived from
|
|
|
|
this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR
|
|
|
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
|
|
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
|
|
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
|
|
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
|
|
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef SPEEX_JITTER_H
|
|
|
|
#define SPEEX_JITTER_H
|
2007-02-10 11:44:26 +00:00
|
|
|
/** @defgroup JitterBuffer JitterBuffer: Adaptive jitter buffer
|
|
|
|
* This is the jitter buffer that reorders UDP/RTP packets and adjusts the buffer size
|
|
|
|
* to maintain good quality and low latency.
|
|
|
|
* @{
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
|
|
|
|
#include "speex.h"
|
|
|
|
#include "speex_bits.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Generic adaptive jitter buffer state */
|
2007-02-09 10:06:53 +00:00
|
|
|
struct JitterBuffer_;
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Generic adaptive jitter buffer state */
|
2007-02-09 10:06:53 +00:00
|
|
|
typedef struct JitterBuffer_ JitterBuffer;
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Definition of an incoming packet */
|
2007-02-09 10:06:53 +00:00
|
|
|
typedef struct _JitterBufferPacket JitterBufferPacket;
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Definition of an incoming packet */
|
2007-02-09 10:06:53 +00:00
|
|
|
struct _JitterBufferPacket {
|
2007-02-10 11:44:26 +00:00
|
|
|
char *data; /**< Data bytes contained in the packet */
|
|
|
|
spx_uint32_t len; /**< Length of the packet in bytes */
|
|
|
|
spx_uint32_t timestamp; /**< Timestamp for the packet */
|
|
|
|
spx_uint32_t span; /**< Time covered by the packet (same units as timestamp) */
|
2007-02-09 10:06:53 +00:00
|
|
|
};
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Packet has been retrieved */
|
2007-02-09 10:06:53 +00:00
|
|
|
#define JITTER_BUFFER_OK 0
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Packet is missing */
|
2007-02-09 10:06:53 +00:00
|
|
|
#define JITTER_BUFFER_MISSING 1
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Packet is incomplete (does not cover the entive tick */
|
2007-02-09 10:06:53 +00:00
|
|
|
#define JITTER_BUFFER_INCOMPLETE 2
|
2007-02-10 11:44:26 +00:00
|
|
|
/** There was an error in the jitter buffer */
|
2007-02-09 10:06:53 +00:00
|
|
|
#define JITTER_BUFFER_INTERNAL_ERROR -1
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Invalid argument */
|
2007-02-09 10:06:53 +00:00
|
|
|
#define JITTER_BUFFER_BAD_ARGUMENT -2
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
|
|
|
|
/** Set minimum amount of extra buffering required (margin) */
|
|
|
|
#define JITTER_BUFFER_SET_MARGIN 0
|
|
|
|
/** Get minimum amount of extra buffering required (margin) */
|
|
|
|
#define JITTER_BUFFER_GET_MARGIN 1
|
2007-03-12 11:54:07 +00:00
|
|
|
/* JITTER_BUFFER_SET_AVALIABLE_COUNT wouldn't make sense */
|
|
|
|
/** Get the amount of avaliable packets currently buffered */
|
|
|
|
#define JITTER_BUFFER_GET_AVALIABLE_COUNT 3
|
2007-02-10 11:44:26 +00:00
|
|
|
|
2007-03-12 11:54:07 +00:00
|
|
|
#define JITTER_BUFFER_ADJUST_INTERPOLATE -1
|
|
|
|
#define JITTER_BUFFER_ADJUST_OK 0
|
|
|
|
#define JITTER_BUFFER_ADJUST_DROP 1
|
2007-02-10 11:44:26 +00:00
|
|
|
|
|
|
|
/** Initialises jitter buffer
|
|
|
|
*
|
|
|
|
* @param tick Number of samples per "tick", i.e. the time period of the elements that will be retrieved
|
|
|
|
* @return Newly created jitter buffer state
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
JitterBuffer *jitter_buffer_init(int tick);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Restores jitter buffer to its original state
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
void jitter_buffer_reset(JitterBuffer *jitter);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Destroys jitter buffer
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
void jitter_buffer_destroy(JitterBuffer *jitter);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Put one packet into the jitter buffer
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
* @param packet Incoming packet
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
void jitter_buffer_put(JitterBuffer *jitter, const JitterBufferPacket *packet);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Get one packet from the jitter buffer
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
* @param packet Returned packet
|
|
|
|
* @param current_timestamp Timestamp for the returned packet
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
int jitter_buffer_get(JitterBuffer *jitter, JitterBufferPacket *packet, spx_uint32_t *current_timestamp);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Get pointer timestamp of jitter buffer
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
int jitter_buffer_get_pointer_timestamp(JitterBuffer *jitter);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Advance by one tick
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
void jitter_buffer_tick(JitterBuffer *jitter);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Used like the ioctl function to control the jitter buffer parameters
|
|
|
|
*
|
|
|
|
* @param jitter Jitter buffer state
|
|
|
|
* @param request ioctl-type request (one of the JITTER_BUFFER_* macros)
|
|
|
|
* @param ptr Data exchanged to-from function
|
|
|
|
* @return 0 if no error, -1 if request in unknown
|
|
|
|
*/
|
|
|
|
int jitter_buffer_ctl(JitterBuffer *jitter, int request, void *ptr);
|
|
|
|
|
2007-03-12 11:54:07 +00:00
|
|
|
int jitter_buffer_update_delay(JitterBuffer *jitter, JitterBufferPacket *packet, spx_uint32_t *start_offset);
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/* @} */
|
2007-02-09 10:06:53 +00:00
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** @defgroup SpeexJitter SpeexJitter: Adaptive jitter buffer specifically for Speex
|
|
|
|
* This is the jitter buffer that reorders UDP/RTP packets and adjusts the buffer size
|
|
|
|
* to maintain good quality and low latency. This is a simplified version that works only
|
|
|
|
* with Speex, but is much easier to use.
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** Speex jitter-buffer state. Never use it directly! */
|
2007-02-09 10:06:53 +00:00
|
|
|
typedef struct SpeexJitter {
|
2007-02-10 11:44:26 +00:00
|
|
|
SpeexBits current_packet; /**< Current Speex packet */
|
|
|
|
int valid_bits; /**< True if Speex bits are valid */
|
|
|
|
JitterBuffer *packets; /**< Generic jitter buffer state */
|
|
|
|
void *dec; /**< Pointer to Speex decoder */
|
|
|
|
spx_int32_t frame_size; /**< Frame size of Speex decoder */
|
2007-02-09 10:06:53 +00:00
|
|
|
} SpeexJitter;
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/** Initialise jitter buffer
|
|
|
|
*
|
|
|
|
* @param jitter State of the Speex jitter buffer
|
|
|
|
* @param decoder Speex decoder to call
|
|
|
|
* @param sampling_rate Sampling rate used by the decoder
|
|
|
|
*/
|
2007-02-09 10:06:53 +00:00
|
|
|
void speex_jitter_init(SpeexJitter *jitter, void *decoder, int sampling_rate);
|
|
|
|
|
|
|
|
/** Destroy jitter buffer */
|
|
|
|
void speex_jitter_destroy(SpeexJitter *jitter);
|
|
|
|
|
|
|
|
/** Put one packet into the jitter buffer */
|
|
|
|
void speex_jitter_put(SpeexJitter *jitter, char *packet, int len, int timestamp);
|
|
|
|
|
|
|
|
/** Get one packet from the jitter buffer */
|
|
|
|
void speex_jitter_get(SpeexJitter *jitter, spx_int16_t *out, int *start_offset);
|
|
|
|
|
|
|
|
/** Get pointer timestamp of jitter buffer */
|
|
|
|
int speex_jitter_get_pointer_timestamp(SpeexJitter *jitter);
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2007-02-10 11:44:26 +00:00
|
|
|
/* @} */
|
2007-02-09 10:06:53 +00:00
|
|
|
#endif
|