pipewire/spa/plugins/audioconvert/resample.h

127 lines
3.7 KiB
C
Raw Normal View History

/* Spa */
/* SPDX-FileCopyrightText: Copyright © 2018 Wim Taymans */
/* SPDX-License-Identifier: MIT */
2018-12-05 15:58:55 +01:00
#ifndef RESAMPLE_H
#define RESAMPLE_H
2018-12-19 16:47:20 +01:00
#include <spa/support/cpu.h>
#include <spa/support/log.h>
2018-12-19 16:47:20 +01:00
2020-02-12 13:31:03 +01:00
#define RESAMPLE_DEFAULT_QUALITY 4
#define RESAMPLE_WINDOW_DEFAULT RESAMPLE_WINDOW_EXP
#define RESAMPLE_MAX_PARAMS 16
2020-02-12 13:31:03 +01:00
struct resample_config {
#define RESAMPLE_WINDOW_EXP 0
#define RESAMPLE_WINDOW_BLACKMAN 1
#define RESAMPLE_WINDOW_KAISER 2
uint32_t window;
double cutoff;
uint32_t n_taps;
#define RESAMPLE_PARAM_EXP_A 0
#define RESAMPLE_PARAM_BLACKMAN_ALPHA 0
#define RESAMPLE_PARAM_KAISER_ALPHA 0
#define RESAMPLE_PARAM_KAISER_SB_ATT 1 /* stopband attenuation */
#define RESAMPLE_PARAM_KAISER_TR_BW 2 /* transition bandwidth */
#define RESAMPLE_PARAM_INVALID (RESAMPLE_MAX_PARAMS-1)
double params[RESAMPLE_MAX_PARAMS];
};
2018-12-05 15:58:55 +01:00
struct resample {
struct spa_log *log;
#define RESAMPLE_OPTION_PREFILL (1<<0)
uint32_t options;
2018-12-19 16:47:20 +01:00
uint32_t cpu_flags;
const char *func_name;
2018-12-05 15:58:55 +01:00
uint32_t channels;
uint32_t i_rate;
uint32_t o_rate;
double rate;
2020-02-12 13:31:03 +01:00
int quality;
2018-12-05 15:58:55 +01:00
struct resample_config config; /* set to all 0 for defaults */
void (*free) (struct resample *r);
void (*update_rate) (struct resample *r, double rate);
uint32_t (*in_len) (struct resample *r, uint32_t out_len);
uint32_t (*out_len) (struct resample *r, uint32_t in_len);
void (*process) (struct resample *r,
const void * SPA_RESTRICT src[], uint32_t *in_len,
void * SPA_RESTRICT dst[], uint32_t *out_len);
void (*reset) (struct resample *r);
uint32_t (*delay) (struct resample *r);
/** Fractional part of delay (in input samples) */
float (*phase) (struct resample *r);
2018-12-05 15:58:55 +01:00
void *data;
};
#define resample_free(r) (r)->free(r)
#define resample_update_rate(r,...) (r)->update_rate(r,__VA_ARGS__)
#define resample_in_len(r,...) (r)->in_len(r,__VA_ARGS__)
#define resample_out_len(r,...) (r)->out_len(r,__VA_ARGS__)
2018-12-05 15:58:55 +01:00
#define resample_process(r,...) (r)->process(r,__VA_ARGS__)
#define resample_reset(r) (r)->reset(r)
#define resample_delay(r) (r)->delay(r)
#define resample_phase(r) (r)->phase(r)
int resample_native_init(struct resample *r);
int resample_native_init_config(struct resample *r, struct resample_config *conf);
int resample_peaks_init(struct resample *r);
static const struct resample_window_info {
uint32_t window;
const char *label;
const char *description;
uint32_t n_params;
} resample_window_info[] = {
[RESAMPLE_WINDOW_EXP] = { RESAMPLE_WINDOW_EXP,
"exp", "Exponential window", 1 },
[RESAMPLE_WINDOW_BLACKMAN] = { RESAMPLE_WINDOW_BLACKMAN,
"blackman", "Blackman window", 1 },
[RESAMPLE_WINDOW_KAISER] = { RESAMPLE_WINDOW_KAISER,
"kaiser", "Kaiser window", 3 },
};
static inline uint32_t resample_window_from_label(const char *label)
{
SPA_FOR_EACH_ELEMENT_VAR(resample_window_info, i) {
if (spa_streq(i->label, label))
return i->window;
}
return RESAMPLE_WINDOW_EXP;
}
static inline const char *resample_window_name(uint32_t idx)
{
return resample_window_info[SPA_CLAMP(idx, 0u, SPA_N_ELEMENTS(resample_window_info)-1)].label;
}
static const struct resample_param_info {
uint32_t window;
uint32_t idx;
const char *label;
} resample_param_info[] = {
{ RESAMPLE_WINDOW_EXP, RESAMPLE_PARAM_EXP_A, "exp.A" },
{ RESAMPLE_WINDOW_BLACKMAN, RESAMPLE_PARAM_BLACKMAN_ALPHA, "blackman.alpha" },
{ RESAMPLE_WINDOW_KAISER, RESAMPLE_PARAM_KAISER_ALPHA, "kaiser.alpha" },
{ RESAMPLE_WINDOW_KAISER, RESAMPLE_PARAM_KAISER_SB_ATT, "kaiser.stopband-attenuation" },
{ RESAMPLE_WINDOW_KAISER, RESAMPLE_PARAM_KAISER_TR_BW, "kaiser.transition-bandwidth" },
};
static inline uint32_t resample_param_from_label(const char *label)
{
SPA_FOR_EACH_ELEMENT_VAR(resample_param_info, i) {
if (spa_streq(i->label, label))
return i->idx;
}
return RESAMPLE_PARAM_INVALID;
}
#endif /* RESAMPLE_H */