2f886a34bb
Simplify the resample buffer size calculation. For audio playback we have sw->ratio = ((int64_t)sw->hw->info.freq << 32) / sw->info.freq; samples = ((int64_t)sw->HWBUF.size << 32) / sw->ratio; This can be simplified to samples = muldiv64(sw->HWBUF.size, sw->info.freq, sw->hw->info.freq); For audio recording we have sw->ratio = ((int64_t)sw->info.freq << 32) / sw->hw->info.freq; samples = (int64_t)sw->HWBUF.size * sw->ratio >> 32; This can be simplified to samples = muldiv64(sw->HWBUF.size, sw->info.freq, sw->hw->info.freq); With hw = sw->hw this becomes in both cases samples = muldiv64(HWBUF.size, sw->info.freq, hw->info.freq); Now that sw->ratio is no longer needed, remove sw->ratio. Acked-by: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk> Reviewed-by: Marc-André Lureau <marcandre.lureau@redhat.com> Signed-off-by: Volker Rümelin <vr_qemu@t-online.de> Message-Id: <20230224190555.7409-15-vr_qemu@t-online.de>
309 lines
8.8 KiB
C
309 lines
8.8 KiB
C
/*
|
|
* QEMU Audio subsystem header
|
|
*
|
|
* Copyright (c) 2003-2005 Vassili Karpov (malc)
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#ifndef QEMU_AUDIO_INT_H
|
|
#define QEMU_AUDIO_INT_H
|
|
|
|
#ifdef CONFIG_AUDIO_COREAUDIO
|
|
#define FLOAT_MIXENG
|
|
/* #define RECIPROCAL */
|
|
#endif
|
|
#include "mixeng.h"
|
|
|
|
#ifdef CONFIG_GIO
|
|
#include <gio/gio.h>
|
|
#endif
|
|
|
|
struct audio_pcm_ops;
|
|
|
|
struct audio_callback {
|
|
void *opaque;
|
|
audio_callback_fn fn;
|
|
};
|
|
|
|
struct audio_pcm_info {
|
|
int bits;
|
|
bool is_signed;
|
|
bool is_float;
|
|
int freq;
|
|
int nchannels;
|
|
int bytes_per_frame;
|
|
int bytes_per_second;
|
|
int swap_endianness;
|
|
};
|
|
|
|
typedef struct AudioState AudioState;
|
|
typedef struct SWVoiceCap SWVoiceCap;
|
|
|
|
typedef struct STSampleBuffer {
|
|
size_t pos, size;
|
|
st_sample *buffer;
|
|
} STSampleBuffer;
|
|
|
|
typedef struct HWVoiceOut {
|
|
AudioState *s;
|
|
int enabled;
|
|
int poll_mode;
|
|
int pending_disable;
|
|
struct audio_pcm_info info;
|
|
|
|
f_sample *clip;
|
|
uint64_t ts_helper;
|
|
|
|
STSampleBuffer mix_buf;
|
|
void *buf_emul;
|
|
size_t pos_emul, pending_emul, size_emul;
|
|
|
|
size_t samples;
|
|
QLIST_HEAD (sw_out_listhead, SWVoiceOut) sw_head;
|
|
QLIST_HEAD (sw_cap_listhead, SWVoiceCap) cap_head;
|
|
struct audio_pcm_ops *pcm_ops;
|
|
QLIST_ENTRY (HWVoiceOut) entries;
|
|
} HWVoiceOut;
|
|
|
|
typedef struct HWVoiceIn {
|
|
AudioState *s;
|
|
int enabled;
|
|
int poll_mode;
|
|
struct audio_pcm_info info;
|
|
|
|
t_sample *conv;
|
|
|
|
size_t total_samples_captured;
|
|
uint64_t ts_helper;
|
|
|
|
STSampleBuffer conv_buf;
|
|
void *buf_emul;
|
|
size_t pos_emul, pending_emul, size_emul;
|
|
|
|
size_t samples;
|
|
QLIST_HEAD (sw_in_listhead, SWVoiceIn) sw_head;
|
|
struct audio_pcm_ops *pcm_ops;
|
|
QLIST_ENTRY (HWVoiceIn) entries;
|
|
} HWVoiceIn;
|
|
|
|
struct SWVoiceOut {
|
|
QEMUSoundCard *card;
|
|
AudioState *s;
|
|
struct audio_pcm_info info;
|
|
t_sample *conv;
|
|
STSampleBuffer resample_buf;
|
|
void *rate;
|
|
size_t total_hw_samples_mixed;
|
|
int active;
|
|
int empty;
|
|
HWVoiceOut *hw;
|
|
char *name;
|
|
struct mixeng_volume vol;
|
|
struct audio_callback callback;
|
|
QLIST_ENTRY (SWVoiceOut) entries;
|
|
};
|
|
|
|
struct SWVoiceIn {
|
|
QEMUSoundCard *card;
|
|
AudioState *s;
|
|
int active;
|
|
struct audio_pcm_info info;
|
|
void *rate;
|
|
size_t total_hw_samples_acquired;
|
|
STSampleBuffer resample_buf;
|
|
f_sample *clip;
|
|
HWVoiceIn *hw;
|
|
char *name;
|
|
struct mixeng_volume vol;
|
|
struct audio_callback callback;
|
|
QLIST_ENTRY (SWVoiceIn) entries;
|
|
};
|
|
|
|
typedef struct audio_driver audio_driver;
|
|
struct audio_driver {
|
|
const char *name;
|
|
const char *descr;
|
|
void *(*init) (Audiodev *);
|
|
void (*fini) (void *);
|
|
#ifdef CONFIG_GIO
|
|
void (*set_dbus_server) (AudioState *s, GDBusObjectManagerServer *manager);
|
|
#endif
|
|
struct audio_pcm_ops *pcm_ops;
|
|
int can_be_default;
|
|
int max_voices_out;
|
|
int max_voices_in;
|
|
size_t voice_size_out;
|
|
size_t voice_size_in;
|
|
QLIST_ENTRY(audio_driver) next;
|
|
};
|
|
|
|
struct audio_pcm_ops {
|
|
int (*init_out)(HWVoiceOut *hw, audsettings *as, void *drv_opaque);
|
|
void (*fini_out)(HWVoiceOut *hw);
|
|
size_t (*write) (HWVoiceOut *hw, void *buf, size_t size);
|
|
void (*run_buffer_out)(HWVoiceOut *hw);
|
|
/*
|
|
* Get the free output buffer size. This is an upper limit. The size
|
|
* returned by function get_buffer_out may be smaller.
|
|
*/
|
|
size_t (*buffer_get_free)(HWVoiceOut *hw);
|
|
/*
|
|
* get a buffer that after later can be passed to put_buffer_out; optional
|
|
* returns the buffer, and writes it's size to size (in bytes)
|
|
*/
|
|
void *(*get_buffer_out)(HWVoiceOut *hw, size_t *size);
|
|
/*
|
|
* put back the buffer returned by get_buffer_out; optional
|
|
* buf must be equal the pointer returned by get_buffer_out,
|
|
* size may be smaller
|
|
*/
|
|
size_t (*put_buffer_out)(HWVoiceOut *hw, void *buf, size_t size);
|
|
void (*enable_out)(HWVoiceOut *hw, bool enable);
|
|
void (*volume_out)(HWVoiceOut *hw, Volume *vol);
|
|
|
|
int (*init_in) (HWVoiceIn *hw, audsettings *as, void *drv_opaque);
|
|
void (*fini_in) (HWVoiceIn *hw);
|
|
size_t (*read) (HWVoiceIn *hw, void *buf, size_t size);
|
|
void (*run_buffer_in)(HWVoiceIn *hw);
|
|
void *(*get_buffer_in)(HWVoiceIn *hw, size_t *size);
|
|
void (*put_buffer_in)(HWVoiceIn *hw, void *buf, size_t size);
|
|
void (*enable_in)(HWVoiceIn *hw, bool enable);
|
|
void (*volume_in)(HWVoiceIn *hw, Volume *vol);
|
|
};
|
|
|
|
void audio_generic_run_buffer_in(HWVoiceIn *hw);
|
|
void *audio_generic_get_buffer_in(HWVoiceIn *hw, size_t *size);
|
|
void audio_generic_put_buffer_in(HWVoiceIn *hw, void *buf, size_t size);
|
|
void audio_generic_run_buffer_out(HWVoiceOut *hw);
|
|
size_t audio_generic_buffer_get_free(HWVoiceOut *hw);
|
|
void *audio_generic_get_buffer_out(HWVoiceOut *hw, size_t *size);
|
|
size_t audio_generic_put_buffer_out(HWVoiceOut *hw, void *buf, size_t size);
|
|
size_t audio_generic_write(HWVoiceOut *hw, void *buf, size_t size);
|
|
size_t audio_generic_read(HWVoiceIn *hw, void *buf, size_t size);
|
|
|
|
struct capture_callback {
|
|
struct audio_capture_ops ops;
|
|
void *opaque;
|
|
QLIST_ENTRY (capture_callback) entries;
|
|
};
|
|
|
|
struct CaptureVoiceOut {
|
|
HWVoiceOut hw;
|
|
void *buf;
|
|
QLIST_HEAD (cb_listhead, capture_callback) cb_head;
|
|
QLIST_ENTRY (CaptureVoiceOut) entries;
|
|
};
|
|
|
|
struct SWVoiceCap {
|
|
SWVoiceOut sw;
|
|
CaptureVoiceOut *cap;
|
|
QLIST_ENTRY (SWVoiceCap) entries;
|
|
};
|
|
|
|
typedef struct AudioState {
|
|
struct audio_driver *drv;
|
|
Audiodev *dev;
|
|
void *drv_opaque;
|
|
|
|
QEMUTimer *ts;
|
|
QLIST_HEAD (card_listhead, QEMUSoundCard) card_head;
|
|
QLIST_HEAD (hw_in_listhead, HWVoiceIn) hw_head_in;
|
|
QLIST_HEAD (hw_out_listhead, HWVoiceOut) hw_head_out;
|
|
QLIST_HEAD (cap_listhead, CaptureVoiceOut) cap_head;
|
|
int nb_hw_voices_out;
|
|
int nb_hw_voices_in;
|
|
int vm_running;
|
|
int64_t period_ticks;
|
|
|
|
bool timer_running;
|
|
uint64_t timer_last;
|
|
|
|
QTAILQ_ENTRY(AudioState) list;
|
|
} AudioState;
|
|
|
|
extern const struct mixeng_volume nominal_volume;
|
|
|
|
extern const char *audio_prio_list[];
|
|
|
|
void audio_driver_register(audio_driver *drv);
|
|
audio_driver *audio_driver_lookup(const char *name);
|
|
|
|
void audio_pcm_init_info (struct audio_pcm_info *info, struct audsettings *as);
|
|
void audio_pcm_info_clear_buf (struct audio_pcm_info *info, void *buf, int len);
|
|
|
|
int audio_bug (const char *funcname, int cond);
|
|
|
|
void audio_run(AudioState *s, const char *msg);
|
|
|
|
const char *audio_application_name(void);
|
|
|
|
typedef struct RateCtl {
|
|
int64_t start_ticks;
|
|
int64_t bytes_sent;
|
|
} RateCtl;
|
|
|
|
void audio_rate_start(RateCtl *rate);
|
|
size_t audio_rate_peek_bytes(RateCtl *rate, struct audio_pcm_info *info);
|
|
void audio_rate_add_bytes(RateCtl *rate, size_t bytes_used);
|
|
size_t audio_rate_get_bytes(RateCtl *rate, struct audio_pcm_info *info,
|
|
size_t bytes_avail);
|
|
|
|
static inline size_t audio_ring_dist(size_t dst, size_t src, size_t len)
|
|
{
|
|
return (dst >= src) ? (dst - src) : (len - src + dst);
|
|
}
|
|
|
|
/**
|
|
* audio_ring_posb() - returns new position in ringbuffer in backward
|
|
* direction at given distance
|
|
*
|
|
* @pos: current position in ringbuffer
|
|
* @dist: distance in ringbuffer to walk in reverse direction
|
|
* @len: size of ringbuffer
|
|
*/
|
|
static inline size_t audio_ring_posb(size_t pos, size_t dist, size_t len)
|
|
{
|
|
return pos >= dist ? pos - dist : len - dist + pos;
|
|
}
|
|
|
|
#define dolog(fmt, ...) AUD_log(AUDIO_CAP, fmt, ## __VA_ARGS__)
|
|
|
|
#ifdef DEBUG
|
|
#define ldebug(fmt, ...) AUD_log(AUDIO_CAP, fmt, ## __VA_ARGS__)
|
|
#else
|
|
#define ldebug(fmt, ...) (void)0
|
|
#endif
|
|
|
|
typedef struct AudiodevListEntry {
|
|
Audiodev *dev;
|
|
QSIMPLEQ_ENTRY(AudiodevListEntry) next;
|
|
} AudiodevListEntry;
|
|
|
|
typedef QSIMPLEQ_HEAD(, AudiodevListEntry) AudiodevListHead;
|
|
AudiodevListHead audio_handle_legacy_opts(void);
|
|
|
|
void audio_free_audiodev_list(AudiodevListHead *head);
|
|
|
|
void audio_create_pdos(Audiodev *dev);
|
|
AudiodevPerDirectionOptions *audio_get_pdo_in(Audiodev *dev);
|
|
AudiodevPerDirectionOptions *audio_get_pdo_out(Audiodev *dev);
|
|
|
|
#endif /* QEMU_AUDIO_INT_H */
|