rt-thread/bsp/allwinner/libraries/sunxi-hal/include/hal/sound/pcm_common.h

814 lines
25 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (c) 2019-2025 Allwinner Technology Co., Ltd. ALL rights reserved.
*
* Allwinner is a trademark of Allwinner Technology Co.,Ltd., registered in
* the the people's Republic of China and other countries.
* All Allwinner Technology Co.,Ltd. trademarks are used with permission.
*
* DISCLAIMER
* THIRD PARTY LICENCES MAY BE REQUIRED TO IMPLEMENT THE SOLUTION/PRODUCT.
* IF YOU NEED TO INTEGRATE THIRD PARTYS TECHNOLOGY (SONY, DTS, DOLBY, AVS OR MPEGLA, ETC.)
* IN ALLWINNERSSDK OR PRODUCTS, YOU SHALL BE SOLELY RESPONSIBLE TO OBTAIN
* ALL APPROPRIATELY REQUIRED THIRD PARTY LICENCES.
* ALLWINNER SHALL HAVE NO WARRANTY, INDEMNITY OR OTHER OBLIGATIONS WITH RESPECT TO MATTERS
* COVERED UNDER ANY REQUIRED THIRD PARTY LICENSE.
* YOU ARE SOLELY RESPONSIBLE FOR YOUR USAGE OF THIRD PARTYS TECHNOLOGY.
*
*
* THIS SOFTWARE IS PROVIDED BY ALLWINNER"AS IS" AND TO THE MAXIMUM EXTENT
* PERMITTED BY LAW, ALLWINNER EXPRESSLY DISCLAIMS ALL WARRANTIES OF ANY KIND,
* WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING WITHOUT LIMITATION REGARDING
* THE TITLE, NON-INFRINGEMENT, ACCURACY, CONDITION, COMPLETENESS, PERFORMANCE
* OR MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
* IN NO EVENT SHALL ALLWINNER 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 __SOUND_PCM_COMMON_H
#define __SOUND_PCM_COMMON_H
#include <stdint.h>
#include <errno.h>
#ifndef EBADFD
#define EBADFD 77
#endif
#ifndef ESTRPIPE
#define ESTRPIPE 86
#endif
#ifndef LONG_MAX
#define LONG_MAX ((long)(~0UL>>1))
#endif
#ifndef ULONG_MAX
#define ULONG_MAX (~0UL)
#endif
#ifndef UINT_MAX
#define UINT_MAX (~0U)
#endif
#define SND_PCM_APPEND (1<<8)
typedef unsigned long snd_pcm_uframes_t;
typedef signed long snd_pcm_sframes_t;
typedef union snd_interval snd_interval_t;
#define SNDRV_PCM_INFO_MMAP 0x00000001 /* hardware supports mmap */
#define SNDRV_PCM_INFO_MMAP_VALID 0x00000002 /* period data are valid during transfer */
#define SNDRV_PCM_INFO_DOUBLE 0x00000004 /* Double buffering needed for PCM start/stop */
#define SNDRV_PCM_INFO_BATCH 0x00000010 /* double buffering */
#define SNDRV_PCM_INFO_INTERLEAVED 0x00000100 /* channels are interleaved */
#define SNDRV_PCM_INFO_NONINTERLEAVED 0x00000200 /* channels are not interleaved */
#define SNDRV_PCM_INFO_COMPLEX 0x00000400 /* complex frame organization (mmap only) */
#define SNDRV_PCM_INFO_BLOCK_TRANSFER 0x00010000 /* hardware transfer block of samples */
#define SNDRV_PCM_INFO_OVERRANGE 0x00020000 /* hardware supports ADC (capture) overrange detection */
#define SNDRV_PCM_INFO_RESUME 0x00040000 /* hardware supports stream resume after suspend */
#define SNDRV_PCM_INFO_PAUSE 0x00080000 /* pause ioctl is supported */
#define SNDRV_PCM_INFO_HALF_DUPLEX 0x00100000 /* only half duplex */
#define SNDRV_PCM_INFO_JOINT_DUPLEX 0x00200000 /* playback and capture stream are somewhat correlated */
#define SNDRV_PCM_INFO_SYNC_START 0x00400000 /* pcm support some kind of sync go */
#define SNDRV_PCM_INFO_NO_PERIOD_WAKEUP 0x00800000 /* period wakeup can be disabled */
#define SNDRV_PCM_INFO_HAS_WALL_CLOCK 0x01000000 /* (Deprecated)has audio wall clock for audio/system time sync */
#define SNDRV_PCM_INFO_HAS_LINK_ATIME 0x01000000 /* report hardware link audio time, reset on startup */
#define SNDRV_PCM_INFO_HAS_LINK_ABSOLUTE_ATIME 0x02000000 /* report absolute hardware link audio time, not reset on startup */
#define SNDRV_PCM_INFO_HAS_LINK_ESTIMATED_ATIME 0x04000000 /* report estimated link audio time */
#define SNDRV_PCM_INFO_HAS_LINK_SYNCHRONIZED_ATIME 0x08000000 /* report synchronized audio/system time */
#define SNDRV_PCM_INFO_DRAIN_TRIGGER 0x40000000 /* internal kernel flag - trigger in drain */
#define SNDRV_PCM_INFO_FIFO_IN_FRAMES 0x80000000 /* internal kernel flag - FIFO size is in frames */
/** PCM access type */
typedef enum _snd_pcm_access {
/** mmap access with simple interleaved channels */
SND_PCM_ACCESS_MMAP_INTERLEAVED = 0,
/** mmap access with simple non interleaved channels */
SND_PCM_ACCESS_MMAP_NONINTERLEAVED,
/** mmap access with complex placement */
SND_PCM_ACCESS_MMAP_COMPLEX,
/** snd_pcm_readi/snd_pcm_writei access */
SND_PCM_ACCESS_RW_INTERLEAVED,
/** snd_pcm_readn/snd_pcm_writen access */
SND_PCM_ACCESS_RW_NONINTERLEAVED,
SND_PCM_ACCESS_LAST = SND_PCM_ACCESS_RW_NONINTERLEAVED
} snd_pcm_access_t;
/** PCM sample format */
typedef enum _snd_pcm_format {
/** Unknown */
SND_PCM_FORMAT_UNKNOWN = -1,
/** Signed 8 bit */
SND_PCM_FORMAT_S8 = 0,
/** Unsigned 8 bit */
SND_PCM_FORMAT_U8,
/** Signed 16 bit Little Endian */
SND_PCM_FORMAT_S16_LE,
/** Signed 16 bit Big Endian */
SND_PCM_FORMAT_S16_BE,
/** Unsigned 16 bit Little Endian */
SND_PCM_FORMAT_U16_LE,
/** Unsigned 16 bit Big Endian */
SND_PCM_FORMAT_U16_BE,
/** Signed 24 bit Little Endian using low three bytes in 32-bit word */
SND_PCM_FORMAT_S24_LE,
/** Signed 24 bit Big Endian using low three bytes in 32-bit word */
SND_PCM_FORMAT_S24_BE,
/** Unsigned 24 bit Little Endian using low three bytes in 32-bit word */
SND_PCM_FORMAT_U24_LE,
/** Unsigned 24 bit Big Endian using low three bytes in 32-bit word */
SND_PCM_FORMAT_U24_BE,
/** Signed 32 bit Little Endian */
SND_PCM_FORMAT_S32_LE,
/** Signed 32 bit Big Endian */
SND_PCM_FORMAT_S32_BE,
/** Unsigned 32 bit Little Endian */
SND_PCM_FORMAT_U32_LE,
/** Unsigned 32 bit Big Endian */
SND_PCM_FORMAT_U32_BE,
/* only support little endian */
/** Signed 16 bit CPU endian */
SND_PCM_FORMAT_S16 = SND_PCM_FORMAT_S16_LE,
/** Unsigned 16 bit CPU endian */
SND_PCM_FORMAT_U16 = SND_PCM_FORMAT_U16_LE,
/** Signed 24 bit CPU endian */
SND_PCM_FORMAT_S24 = SND_PCM_FORMAT_S24_LE,
/** Unsigned 24 bit CPU endian */
SND_PCM_FORMAT_U24 = SND_PCM_FORMAT_U24_LE,
/** Signed 32 bit CPU endian */
SND_PCM_FORMAT_S32 = SND_PCM_FORMAT_S32_LE,
/** Unsigned 32 bit CPU endian */
SND_PCM_FORMAT_U32 = SND_PCM_FORMAT_U32_LE,
SND_PCM_FORMAT_LAST = SND_PCM_FORMAT_U32_BE,
} snd_pcm_format_t;
/** PCM state */
typedef enum _ksnd_pcm_state {
/** Open */
SNDRV_PCM_STATE_OPEN = 0,
/** Setup installed */
SNDRV_PCM_STATE_SETUP,
/** Ready to start */
SNDRV_PCM_STATE_PREPARED,
/** Running */
SNDRV_PCM_STATE_RUNNING,
/** Stopped: underrun (playback) or overrun (capture) detected */
SNDRV_PCM_STATE_XRUN,
/** Draining: running (playback) or stopped (capture) */
SNDRV_PCM_STATE_DRAINING,
/** Paused */
SNDRV_PCM_STATE_PAUSED,
/** Hardware is suspended */
SNDRV_PCM_STATE_SUSPENDED,
/** Hardware is disconnected */
SNDRV_PCM_STATE_DISCONNECTED,
SNDRV_PCM_STATE_LAST = SNDRV_PCM_STATE_DISCONNECTED
} ksnd_pcm_state_t;
typedef int snd_pcm_hw_param_t;
#define SND_PCM_HW_PARAM_ACCESS 0
#define SND_PCM_HW_PARAM_FORMAT 1
#define SND_PCM_HW_PARAM_FIRST_MASK SND_PCM_HW_PARAM_ACCESS
#define SND_PCM_HW_PARAM_LAST_MASK SND_PCM_HW_PARAM_FORMAT
#define SND_PCM_HW_PARAM_SAMPLE_BITS 2
#define SND_PCM_HW_PARAM_FRAME_BITS 3
#define SND_PCM_HW_PARAM_CHANNELS 4
#define SND_PCM_HW_PARAM_RATE 5
#define SND_PCM_HW_PARAM_PERIOD_TIME 6
#define SND_PCM_HW_PARAM_PERIOD_SIZE 7
#define SND_PCM_HW_PARAM_PERIOD_BYTES 8
#define SND_PCM_HW_PARAM_PERIODS 9
#define SND_PCM_HW_PARAM_BUFFER_TIME 10
#define SND_PCM_HW_PARAM_BUFFER_SIZE 11
#define SND_PCM_HW_PARAM_BUFFER_BYTES 12
#define SND_PCM_HW_PARAM_FIRST_RANGE SND_PCM_HW_PARAM_SAMPLE_BITS
#define SND_PCM_HW_PARAM_LAST_RANGE SND_PCM_HW_PARAM_BUFFER_BYTES
#define SND_PCM_HW_PARAM_FIRST_INTERVAL SND_PCM_HW_PARAM_ACCESS
#define SND_PCM_HW_PARAM_LAST_INTERVAL SND_PCM_HW_PARAM_BUFFER_BYTES
static inline int hw_is_mask(int var)
{
return var >= SND_PCM_HW_PARAM_FIRST_MASK &&
var <= SND_PCM_HW_PARAM_LAST_MASK;
}
static inline int hw_is_range(int var)
{
return var >= SND_PCM_HW_PARAM_FIRST_RANGE &&
var <= SND_PCM_HW_PARAM_LAST_RANGE;
}
union snd_interval {
struct {
uint32_t min;
uint32_t max;
int openmin; /* whether the interval is left-open */
int openmax; /* whether the interval is right-open */
int integer; /* whether the value is integer or not */
int empty;
} range;
uint32_t mask;
};
typedef struct snd_pcm_hw_params {
union snd_interval intervals[SND_PCM_HW_PARAM_LAST_INTERVAL -
SND_PCM_HW_PARAM_FIRST_INTERVAL + 1];
unsigned int can_paused;
uint32_t rmask; /* W: requested masks */
uint32_t cmask; /* R: changed masks */
} snd_pcm_hw_params_t;
typedef struct snd_pcm_sw_params {
snd_pcm_uframes_t avail_min; /* min avail frames for wakeup */
snd_pcm_uframes_t start_threshold; /* min hw_avail frames for automatic start */
snd_pcm_uframes_t stop_threshold; /* min avail frames for automatic stop */
snd_pcm_uframes_t silence_size; /* silence block size */
snd_pcm_uframes_t boundary; /* pointers wrap point */
} snd_pcm_sw_params_t;
typedef long ksnd_pcm_sframes_t;
typedef unsigned long ksnd_pcm_uframes_t;
struct snd_pcm_mmap_status {
ksnd_pcm_state_t state;
ksnd_pcm_uframes_t hw_ptr;
};
struct snd_pcm_mmap_control {
ksnd_pcm_uframes_t appl_ptr;
ksnd_pcm_uframes_t avail_min;
};
#define SNDRV_PCM_SYNC_PTR_HWSYNC (1<<0) /* execute hwsync */
#define SNDRV_PCM_SYNC_PTR_APPL (1<<1) /* get appl_ptr from driver (r/w op) */
#define SNDRV_PCM_SYNC_PTR_AVAIL_MIN (1<<2) /* get avail_min from driver */
typedef struct _snd_pcm_channel_info {
unsigned int channel;
void *addr; /* base address of channel samples */
unsigned int first; /* offset to first sample in bits */
unsigned int step; /* samples distance in bits */
enum { SND_PCM_AREA_MMAP, SND_PCM_AREA_LOCAL } type;
} snd_pcm_channel_info_t;
static inline unsigned int div32(unsigned int a, unsigned int b,
unsigned int *r)
{
if (b == 0) {
*r = 0;
return UINT_MAX;
}
*r = a % b;
return a / b;
}
static inline uint64_t div_u64_rem(uint64_t dividend, uint32_t divisor, uint32_t *remainder)
{
*remainder = dividend % divisor;
return dividend / divisor;
}
static inline unsigned int div_down(unsigned int a, unsigned int b)
{
if (b == 0)
return UINT_MAX;
return a / b;
}
static inline unsigned int div_up(unsigned int a, unsigned int b)
{
unsigned int r;
unsigned int q;
if (b == 0)
return UINT_MAX;
q = div32(a, b, &r);
if (r)
++q;
return q;
}
static inline unsigned int mul(unsigned int a, unsigned int b)
{
if (a == 0)
return 0;
if (div_down(UINT_MAX, a) < b)
return UINT_MAX;
return a * b;
}
static inline unsigned int add(unsigned int a, unsigned int b)
{
if (a >= UINT_MAX - b)
return UINT_MAX;
return a + b;
}
static inline unsigned int sub(unsigned int a, unsigned int b)
{
if (a > b)
return a - b;
return 0;
}
static inline unsigned int muldiv32(unsigned int a, unsigned int b,
unsigned int c, unsigned int *r)
{
uint64_t n = (uint64_t)a * (uint64_t)b;
uint32_t rem;
if (c == 0) {
*r = 0;
return UINT_MAX;
}
n = div_u64_rem(n, c, &rem);
if (n >= UINT_MAX) {
*r = 0;
return UINT_MAX;
}
*r = rem;
return n;
}
static inline int __pcm_ffs(uint32_t value)
{
uint32_t offset;
for (offset = 0; offset < sizeof(value)*8; offset++) {
if (value & (1<<offset))
return offset;
}
return -1;
}
static inline int __ffs(uint32_t value)
{
uint32_t offset;
for (offset = 0; offset < sizeof(value)*8; offset++) {
if (value & (1<<offset))
return offset;
}
return -1;
}
static inline int __fls(uint32_t value)
{
uint32_t offset;
for (offset = sizeof(value)*8; offset > 0; offset--) {
if (value & (1<<(offset - 1)))
return offset;
}
return -1;
}
static inline snd_pcm_access_t params_access(const struct snd_pcm_hw_params *p)
{
const union snd_interval *interval = NULL;
interval = &p->intervals[SND_PCM_HW_PARAM_ACCESS -
SND_PCM_HW_PARAM_FIRST_INTERVAL];
if (interval->mask != 0)
return (snd_pcm_access_t)__ffs(interval->mask);
return (snd_pcm_access_t)-1;
}
static inline snd_pcm_format_t params_format(const struct snd_pcm_hw_params *p)
{
const union snd_interval *interval = NULL;
interval = &p->intervals[SND_PCM_HW_PARAM_FORMAT -
SND_PCM_HW_PARAM_FIRST_INTERVAL];
if (interval->mask != 0)
return (snd_pcm_format_t)__ffs(interval->mask);
return SND_PCM_FORMAT_UNKNOWN;
}
static inline snd_interval_t *hw_param_interval(struct snd_pcm_hw_params *params,
snd_pcm_hw_param_t var)
{
return &params->intervals[var];
}
static inline const union snd_interval *hw_param_interval_c(const struct snd_pcm_hw_params *params,
int var)
{
return &params->intervals[var - SND_PCM_HW_PARAM_FIRST_INTERVAL];
}
static inline unsigned int params_channels(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_CHANNELS)->range.min;
}
static inline unsigned int params_rate(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_RATE)->range.min;
}
static inline unsigned int params_period_size(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_PERIOD_SIZE)->range.min;
}
static inline unsigned int params_period_time(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_PERIOD_TIME)->range.min;
}
static inline unsigned int params_periods(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_PERIODS)->range.min;
}
static inline unsigned int params_buffer_size(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_BUFFER_SIZE)->range.min;
}
static inline unsigned int params_buffer_time(const struct snd_pcm_hw_params *p)
{
return hw_param_interval_c(p, SND_PCM_HW_PARAM_BUFFER_TIME)->range.min;
}
int snd_pcm_format_physical_width(snd_pcm_format_t format);
#define SND_MASK_BITS 32
static inline void snd_mask_none(snd_interval_t *interval)
{
interval->mask = 0U;
}
static inline void snd_mask_any(snd_interval_t *interval)
{
interval->mask = ~0U;
}
static inline int snd_mask_empty(const snd_interval_t *interval)
{
return !interval->mask;
}
static inline int snd_mask_full(const snd_interval_t *interval)
{
return interval->mask == 0xffffffff;
}
static inline unsigned int snd_mask_min(const snd_interval_t *interval)
{
unsigned int i;
for (i = 0; i < sizeof(interval->mask) * 8; i++) {
if (interval->mask & (1 << i))
return i;
}
return 0;
}
static inline unsigned int snd_mask_max(const snd_interval_t *interval)
{
unsigned int i;
for (i = sizeof(interval->mask) * 8; i > 0; i--) {
if (interval->mask & (1 << (i - 1)))
return i - 1;
}
return 0;
}
static inline void snd_mask_set(snd_interval_t *interval, unsigned int val)
{
interval->mask |= (1 << val);
}
static inline void snd_mask_reset(snd_interval_t *interval, unsigned int val)
{
interval->mask &= ~(1 << val);
}
static inline void snd_mask_set_range(snd_interval_t *interval,
unsigned int from, unsigned int to)
{
unsigned int i;
for (i = from; i <= to; i++)
interval->mask |= (1 << i);
}
static inline void snd_mask_reset_range(snd_interval_t *interval,
unsigned int from, unsigned int to)
{
unsigned int i;
for (i = from; i <= to; i++)
interval->mask &= ~(1 << i);
}
static inline void snd_mask_leave(snd_interval_t *interval, unsigned int val)
{
snd_mask_none(interval);
interval->mask |= (1 << val);
}
static inline void snd_mask_intersect(snd_interval_t *interval, const snd_interval_t *v)
{
interval->mask &= v->mask;
}
static inline int snd_mask_eq(snd_interval_t *interval, const snd_interval_t *v)
{
return interval->mask == v->mask;
}
static inline void snd_mask_copy(snd_interval_t *interval, const snd_interval_t *v)
{
interval->mask = v->mask;
}
static inline int snd_mask_test(const snd_interval_t *interval, unsigned int val)
{
return interval->mask & (1 << val);
}
static inline int snd_mask_single(const snd_interval_t *interval)
{
unsigned int bits = sizeof(interval->mask) * 8;
unsigned int i;
int c = 0;
for (i = 0; i < bits; ++i) {
if (!(interval->mask & (1 << i)))
continue;
if (c)
return 0;
c++;
}
return 1;
}
static inline int snd_mask_refine(snd_interval_t *interval, const snd_interval_t *v)
{
snd_interval_t old;
snd_mask_copy(&old, interval);
snd_mask_intersect(interval, v);
if (snd_mask_empty(interval))
return -EINVAL;
return !snd_mask_eq(interval, &old);
}
static inline int snd_mask_refine_first(snd_interval_t *interval)
{
if (snd_mask_single(interval))
return 0;
snd_mask_leave(interval, snd_mask_min(interval));
return 1;
}
static inline int snd_mask_refine_last(snd_interval_t *interval)
{
if (snd_mask_single(interval))
return 0;
snd_mask_leave(interval, snd_mask_max(interval));
return 1;
}
static inline int snd_mask_refine_min(snd_interval_t *interval, unsigned int val)
{
if (snd_mask_min(interval) >= val)
return 0;
snd_mask_reset_range(interval, 0, val - 1);
if (snd_mask_empty(interval))
return -EINVAL;
return 1;
}
static inline int snd_mask_refine_max(snd_interval_t *interval, unsigned int val)
{
if (snd_mask_max(interval) <= val)
return 0;
snd_mask_reset_range(interval, val + 1, SND_MASK_BITS);
if (snd_mask_empty(interval))
return -EINVAL;
return 1;
}
static inline int snd_mask_refine_set(snd_interval_t *interval, unsigned int val)
{
int changed;
changed = !snd_mask_single(interval);
snd_mask_leave(interval, val);
if (snd_mask_empty(interval))
return -EINVAL;
return changed;
}
static inline int snd_mask_value(const snd_interval_t *interval)
{
return snd_mask_min(interval);
}
static inline int snd_mask_always_eq(const snd_interval_t *m1, const snd_interval_t *m2)
{
return snd_mask_single(m1) && snd_mask_single(m2) &&
snd_mask_value(m1) == snd_mask_value(m2);
}
static inline int snd_mask_never_eq(const snd_interval_t *m1, const snd_interval_t *m2)
{
if (m1->mask & m2->mask)
return 0;
return 1;
}
static inline void snd_range_any(snd_interval_t *i)
{
i->range.min = 0;
i->range.openmin = 0;
i->range.max = UINT_MAX;
i->range.openmax= 0;
i->range.integer = 0;
i->range.empty = 0;
}
static inline void snd_range_none(snd_interval_t *i)
{
i->range.empty = 1;
}
static inline int snd_range_checkempty(snd_interval_t *i)
{
return (i->range.min > i->range.max ||
(i->range.min == i->range.max && (i->range.openmin || i->range.openmax)));
}
static inline int snd_range_empty(const snd_interval_t *i)
{
return i->range.empty;
}
static inline int snd_range_single(const snd_interval_t *i)
{
return (i->range.min == i->range.max ||
(i->range.min + 1 == i->range.max && i->range.openmax));
}
static inline int snd_range_value(const snd_interval_t *i)
{
return i->range.min;
}
static inline void snd_range_set_value(snd_interval_t *i, unsigned int val)
{
i->range.openmax = i->range.openmin = 0;
i->range.min = i->range.max = val;
i->range.integer = 0;
i->range.empty = 0;
}
static inline int snd_range_min(const snd_interval_t *i)
{
return i->range.min;
}
static inline int snd_range_max(const snd_interval_t *i)
{
unsigned int v;
v = i->range.max;
if (i->range.openmax)
v--;
return v;
}
static inline void snd_range_set_minmax(snd_interval_t *i, unsigned int min, unsigned int max)
{
i->range.openmax = i->range.openmin = 0;
i->range.min = min;
i->range.max = max;
i->range.integer = 0;
i->range.empty = 0;
}
static inline int snd_range_test(const snd_interval_t *i, unsigned int val)
{
return !((i->range.min > val || (i->range.min == val && i->range.openmin) ||
i->range.max < val || (i->range.max == val && i->range.openmax)));
}
static inline void snd_range_copy(snd_interval_t *d, const snd_interval_t *s)
{
d->range = s->range;
}
static inline int snd_range_setinteger(snd_interval_t *i)
{
if (i->range.integer)
return 0;
if (i->range.openmin && i->range.openmax && i->range.min == i->range.max)
return -EINVAL;
i->range.integer = 1;
return 1;
}
static inline void snd_range_floor(snd_interval_t *i)
{
if (i->range.integer || snd_range_empty(i))
return;
i->range.openmin = 0;
if (i->range.openmax) {
i->range.max--;
i->range.openmax = 0;
}
i->range.integer = 1;
}
static inline void snd_range_unfloor(snd_interval_t *i)
{
if (snd_range_empty(i))
return;
if (i->range.max == UINT_MAX)
return;
if (i->range.openmax)
return;
i->range.max++;
i->range.openmax = 1;
i->range.integer = 0;
}
static inline int snd_range_eq(const snd_interval_t *i1, const snd_interval_t *i2)
{
if (i1->range.empty)
return i2->range.empty;
if (i2->range.empty)
return i1->range.empty;
return i1->range.min == i2->range.min &&
i1->range.openmin == i2->range.openmin &&
i1->range.max == i2->range.max &&
i1->range.openmax == i2->range.openmax;
}
static inline int snd_range_always_eq(const snd_interval_t *i1, const snd_interval_t *i2)
{
return snd_range_single(i1) && snd_range_single(i2) &&
snd_range_value(i1) == snd_range_value(i2);
}
static inline int snd_range_never_eq(const snd_interval_t *i1, const snd_interval_t *i2)
{
return (i1->range.max < i2->range.min ||
(i1->range.max == i2->range.min &&
(i1->range.openmax || i1->range.openmin)) ||
i1->range.min > i2->range.max ||
(i1->range.min == i2->range.max &&
(i1->range.openmin || i2->range.openmax)));
}
int snd_range_refine(snd_interval_t *i, const snd_interval_t *v);
int snd_range_refine_first(snd_interval_t *i);
int snd_range_refine_last(snd_interval_t *i);
int snd_range_refine_min(snd_interval_t *i, unsigned int min, int openmin);
int snd_range_refine_max(snd_interval_t *i, unsigned int max, int openmax);
int snd_range_refine_set(snd_interval_t *i, unsigned int val);
void snd_range_add(const snd_interval_t *a, const snd_interval_t *b, snd_interval_t *c);
void snd_range_sub(const snd_interval_t *a, const snd_interval_t *b, snd_interval_t *c);
void snd_range_mul(const snd_interval_t *a, const snd_interval_t *b, snd_interval_t *c);
void snd_range_div(const snd_interval_t *a, const snd_interval_t *b, snd_interval_t *c);
void snd_range_muldiv(const snd_interval_t *a, const snd_interval_t *b,
const snd_interval_t *c, snd_interval_t *d);
void snd_range_muldivk(const snd_interval_t *a, const snd_interval_t *b,
unsigned int k, snd_interval_t *c);
void snd_range_mulkdiv(const snd_interval_t *a, unsigned int k,
const snd_interval_t *b, snd_interval_t *c);
int snd_range_list(snd_interval_t *i, unsigned int count,
const unsigned int *list, unsigned int mask);
const char *snd_pcm_hw_param_name(snd_pcm_hw_param_t param);
struct snd_pcm_hw_rule;
typedef int (*snd_pcm_hw_rule_func_t)(struct snd_pcm_hw_params *params,
struct snd_pcm_hw_rule *rule);
struct snd_pcm_hw_rule {
unsigned int cond;
int var;
int deps[4];
snd_pcm_hw_rule_func_t func;
void *private_data;
};
int snd_pcm_hw_rule_mul(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_div(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_muldivk(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_mulkdiv(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_format(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_sample_bits(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_rate(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
int snd_pcm_hw_rule_buffer_bytes_max(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule);
#endif /* __SOUND_PCM_COMMON_H */