
* Update to QEMU v9.0.0 --------- Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Fabiano Rosas <farosas@suse.de> Signed-off-by: Peter Xu <peterx@redhat.com> Signed-off-by: Thomas Huth <thuth@redhat.com> Signed-off-by: Cédric Le Goater <clg@redhat.com> Signed-off-by: Zheyu Ma <zheyuma97@gmail.com> Signed-off-by: Ido Plat <ido.plat@ibm.com> Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com> Signed-off-by: Markus Armbruster <armbru@redhat.com> Signed-off-by: Marc-André Lureau <marcandre.lureau@redhat.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com> Signed-off-by: Vladimir Sementsov-Ogievskiy <vsementsov@yandex-team.ru> Signed-off-by: David Hildenbrand <david@redhat.com> Signed-off-by: Kevin Wolf <kwolf@redhat.com> Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com> Signed-off-by: Fiona Ebner <f.ebner@proxmox.com> Signed-off-by: Gregory Price <gregory.price@memverge.com> Signed-off-by: Jonathan Cameron <Jonathan.Cameron@huawei.com> Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org> Signed-off-by: Lorenz Brun <lorenz@brun.one> Signed-off-by: Yao Xingtao <yaoxt.fnst@fujitsu.com> Signed-off-by: Arnaud Minier <arnaud.minier@telecom-paris.fr> Signed-off-by: Inès Varhol <ines.varhol@telecom-paris.fr> Signed-off-by: BALATON Zoltan <balaton@eik.bme.hu> Signed-off-by: Igor Mammedov <imammedo@redhat.com> Signed-off-by: Akihiko Odaki <akihiko.odaki@daynix.com> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Signed-off-by: Sven Schnelle <svens@stackframe.org> Signed-off-by: Daniel Henrique Barboza <dbarboza@ventanamicro.com> Signed-off-by: Christian Schoenebeck <qemu_oss@crudebyte.com> Signed-off-by: Jason Wang <jasowang@redhat.com> Signed-off-by: Helge Deller <deller@gmx.de> Signed-off-by: Nicholas Piggin <npiggin@gmail.com> Signed-off-by: Benjamin Gray <bgray@linux.ibm.com> Signed-off-by: Avihai Horon <avihaih@nvidia.com> Signed-off-by: Michael Tokarev <mjt@tls.msk.ru> Signed-off-by: Joonas Kankaala <joonas.a.kankaala@gmail.com> Signed-off-by: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org> Signed-off-by: Stefan Weil <sw@weilnetz.de> Signed-off-by: Zhao Liu <zhao1.liu@intel.com> Signed-off-by: Glenn Miles <milesg@linux.ibm.com> Signed-off-by: Oleg Sviridov <oleg.sviridov@red-soft.ru> Signed-off-by: Artem Chernyshev <artem.chernyshev@red-soft.ru> Signed-off-by: Yajun Wu <yajunw@nvidia.com> Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com> Signed-off-by: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk> Signed-off-by: Pierre-Clément Tosi <ptosi@google.com> Signed-off-by: Lei Wang <lei4.wang@intel.com> Signed-off-by: Wei Wang <wei.w.wang@intel.com> Signed-off-by: Martin Hundebøll <martin@geanix.com> Signed-off-by: Michael S. Tsirkin <mst@redhat.com> Signed-off-by: Manos Pitsidianakis <manos.pitsidianakis@linaro.org> Signed-off-by: Wafer <wafer@jaguarmicro.com> Signed-off-by: Yuxue Liu <yuxue.liu@jaguarmicro.com> Signed-off-by: Gerd Hoffmann <kraxel@redhat.com> Signed-off-by: Nguyen Dinh Phi <phind.uet@gmail.com> Signed-off-by: Zack Buhman <zack@buhman.org> Signed-off-by: Keith Packard <keithp@keithp.com> Signed-off-by: Yuquan Wang wangyuquan1236@phytium.com.cn Signed-off-by: Matheus Tavares Bernardino <quic_mathbern@quicinc.com> Signed-off-by: Cindy Lu <lulu@redhat.com> Co-authored-by: Peter Maydell <peter.maydell@linaro.org> Co-authored-by: Fabiano Rosas <farosas@suse.de> Co-authored-by: Peter Xu <peterx@redhat.com> Co-authored-by: Thomas Huth <thuth@redhat.com> Co-authored-by: Cédric Le Goater <clg@redhat.com> Co-authored-by: Zheyu Ma <zheyuma97@gmail.com> Co-authored-by: Ido Plat <ido.plat@ibm.com> Co-authored-by: Ilya Leoshkevich <iii@linux.ibm.com> Co-authored-by: Markus Armbruster <armbru@redhat.com> Co-authored-by: Marc-André Lureau <marcandre.lureau@redhat.com> Co-authored-by: Paolo Bonzini <pbonzini@redhat.com> Co-authored-by: Vladimir Sementsov-Ogievskiy <vsementsov@yandex-team.ru> Co-authored-by: David Hildenbrand <david@redhat.com> Co-authored-by: Kevin Wolf <kwolf@redhat.com> Co-authored-by: Stefan Reiter <s.reiter@proxmox.com> Co-authored-by: Fiona Ebner <f.ebner@proxmox.com> Co-authored-by: Gregory Price <gregory.price@memverge.com> Co-authored-by: Lorenz Brun <lorenz@brun.one> Co-authored-by: Yao Xingtao <yaoxt.fnst@fujitsu.com> Co-authored-by: Philippe Mathieu-Daudé <philmd@linaro.org> Co-authored-by: Arnaud Minier <arnaud.minier@telecom-paris.fr> Co-authored-by: BALATON Zoltan <balaton@eik.bme.hu> Co-authored-by: Igor Mammedov <imammedo@redhat.com> Co-authored-by: Akihiko Odaki <akihiko.odaki@daynix.com> Co-authored-by: Richard Henderson <richard.henderson@linaro.org> Co-authored-by: Sven Schnelle <svens@stackframe.org> Co-authored-by: Daniel Henrique Barboza <dbarboza@ventanamicro.com> Co-authored-by: Helge Deller <deller@kernel.org> Co-authored-by: Harsh Prateek Bora <harshpb@linux.ibm.com> Co-authored-by: Benjamin Gray <bgray@linux.ibm.com> Co-authored-by: Nicholas Piggin <npiggin@gmail.com> Co-authored-by: Avihai Horon <avihaih@nvidia.com> Co-authored-by: Michael Tokarev <mjt@tls.msk.ru> Co-authored-by: Joonas Kankaala <joonas.a.kankaala@gmail.com> Co-authored-by: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org> Co-authored-by: Stefan Weil <sw@weilnetz.de> Co-authored-by: Dayu Liu <liu.dayu@zte.com.cn> Co-authored-by: Zhao Liu <zhao1.liu@intel.com> Co-authored-by: Glenn Miles <milesg@linux.vnet.ibm.com> Co-authored-by: Artem Chernyshev <artem.chernyshev@red-soft.ru> Co-authored-by: Yajun Wu <yajunw@nvidia.com> Co-authored-by: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk> Co-authored-by: Pierre-Clément Tosi <ptosi@google.com> Co-authored-by: Wei Wang <wei.w.wang@intel.com> Co-authored-by: Martin Hundebøll <martin@geanix.com> Co-authored-by: Michael S. Tsirkin <mst@redhat.com> Co-authored-by: Manos Pitsidianakis <manos.pitsidianakis@linaro.org> Co-authored-by: Wafer <wafer@jaguarmicro.com> Co-authored-by: lyx634449800 <yuxue.liu@jaguarmicro.com> Co-authored-by: Gerd Hoffmann <kraxel@redhat.com> Co-authored-by: Nguyen Dinh Phi <phind.uet@gmail.com> Co-authored-by: Zack Buhman <zack@buhman.org> Co-authored-by: Keith Packard <keithp@keithp.com> Co-authored-by: Yuquan Wang <wangyuquan1236@phytium.com.cn> Co-authored-by: Matheus Tavares Bernardino <quic_mathbern@quicinc.com> Co-authored-by: Cindy Lu <lulu@redhat.com>
251 lines
8.3 KiB
C
251 lines
8.3 KiB
C
/*
|
|
* VIRTIO Sound Device conforming to
|
|
*
|
|
* "Virtual I/O Device (VIRTIO) Version 1.2
|
|
* Committee Specification Draft 01
|
|
* 09 May 2022"
|
|
*
|
|
* Copyright (c) 2023 Emmanouil Pitsidianakis <manos.pitsidianakis@linaro.org>
|
|
* Copyright (C) 2019 OpenSynergy GmbH
|
|
*
|
|
* This work is licensed under the terms of the GNU GPL, version 2 or
|
|
* (at your option) any later version. See the COPYING file in the
|
|
* top-level directory.
|
|
*/
|
|
|
|
#ifndef QEMU_VIRTIO_SOUND_H
|
|
#define QEMU_VIRTIO_SOUND_H
|
|
|
|
#include "hw/virtio/virtio.h"
|
|
#include "audio/audio.h"
|
|
#include "standard-headers/linux/virtio_ids.h"
|
|
#include "standard-headers/linux/virtio_snd.h"
|
|
|
|
#define TYPE_VIRTIO_SND "virtio-sound-device"
|
|
#define VIRTIO_SND(obj) \
|
|
OBJECT_CHECK(VirtIOSound, (obj), TYPE_VIRTIO_SND)
|
|
|
|
/* CONFIGURATION SPACE */
|
|
|
|
typedef struct virtio_snd_config virtio_snd_config;
|
|
|
|
/* COMMON DEFINITIONS */
|
|
|
|
/* common header for request/response*/
|
|
typedef struct virtio_snd_hdr virtio_snd_hdr;
|
|
|
|
/* event notification */
|
|
typedef struct virtio_snd_event virtio_snd_event;
|
|
|
|
/* common control request to query an item information */
|
|
typedef struct virtio_snd_query_info virtio_snd_query_info;
|
|
|
|
/* JACK CONTROL MESSAGES */
|
|
|
|
typedef struct virtio_snd_jack_hdr virtio_snd_jack_hdr;
|
|
|
|
/* jack information structure */
|
|
typedef struct virtio_snd_jack_info virtio_snd_jack_info;
|
|
|
|
/* jack remapping control request */
|
|
typedef struct virtio_snd_jack_remap virtio_snd_jack_remap;
|
|
|
|
/*
|
|
* PCM CONTROL MESSAGES
|
|
*/
|
|
typedef struct virtio_snd_pcm_hdr virtio_snd_pcm_hdr;
|
|
|
|
/* PCM stream info structure */
|
|
typedef struct virtio_snd_pcm_info virtio_snd_pcm_info;
|
|
|
|
/* set PCM stream params */
|
|
typedef struct virtio_snd_pcm_set_params virtio_snd_pcm_set_params;
|
|
|
|
/* I/O request header */
|
|
typedef struct virtio_snd_pcm_xfer virtio_snd_pcm_xfer;
|
|
|
|
/* I/O request status */
|
|
typedef struct virtio_snd_pcm_status virtio_snd_pcm_status;
|
|
|
|
/* device structs */
|
|
|
|
typedef struct VirtIOSound VirtIOSound;
|
|
|
|
typedef struct VirtIOSoundPCMStream VirtIOSoundPCMStream;
|
|
|
|
typedef struct virtio_snd_ctrl_command virtio_snd_ctrl_command;
|
|
|
|
typedef struct VirtIOSoundPCM VirtIOSoundPCM;
|
|
|
|
typedef struct VirtIOSoundPCMBuffer VirtIOSoundPCMBuffer;
|
|
|
|
/*
|
|
* The VirtIO sound spec reuses layouts and values from the High Definition
|
|
* Audio spec (virtio/v1.2: 5.14 Sound Device). This struct handles each I/O
|
|
* message's buffer (virtio/v1.2: 5.14.6.8 PCM I/O Messages).
|
|
*
|
|
* In the case of TX (i.e. playback) buffers, we defer reading the raw PCM data
|
|
* from the virtqueue until QEMU's sound backsystem calls the output callback.
|
|
* This is tracked by the `bool populated;` field, which is set to true when
|
|
* data has been read into our own buffer for consumption.
|
|
*
|
|
* VirtIOSoundPCMBuffer has a dynamic size since it includes the raw PCM data
|
|
* in its allocation. It must be initialized and destroyed as follows:
|
|
*
|
|
* size_t size = [[derived from owned VQ element descriptor sizes]];
|
|
* buffer = g_malloc0(sizeof(VirtIOSoundPCMBuffer) + size);
|
|
* buffer->elem = [[owned VQ element]];
|
|
*
|
|
* [..]
|
|
*
|
|
* g_free(buffer->elem);
|
|
* g_free(buffer);
|
|
*/
|
|
struct VirtIOSoundPCMBuffer {
|
|
QSIMPLEQ_ENTRY(VirtIOSoundPCMBuffer) entry;
|
|
VirtQueueElement *elem;
|
|
VirtQueue *vq;
|
|
size_t size;
|
|
/*
|
|
* In TX / Plaback, `offset` represents the first unused position inside
|
|
* `data`. If `offset == size` then there are no unused data left.
|
|
*/
|
|
uint64_t offset;
|
|
/* Used for the TX queue for lazy I/O copy from `elem` */
|
|
bool populated;
|
|
/*
|
|
* VirtIOSoundPCMBuffer is an unsized type because it ends with an array of
|
|
* bytes. The size of `data` is determined from the I/O message's read-only
|
|
* or write-only size when allocating VirtIOSoundPCMBuffer.
|
|
*/
|
|
uint8_t data[];
|
|
};
|
|
|
|
struct VirtIOSoundPCM {
|
|
VirtIOSound *snd;
|
|
/*
|
|
* PCM parameters are a separate field instead of a VirtIOSoundPCMStream
|
|
* field, because the operation of PCM control requests is first
|
|
* VIRTIO_SND_R_PCM_SET_PARAMS and then VIRTIO_SND_R_PCM_PREPARE; this
|
|
* means that some times we get parameters without having an allocated
|
|
* stream yet.
|
|
*/
|
|
virtio_snd_pcm_set_params *pcm_params;
|
|
VirtIOSoundPCMStream **streams;
|
|
};
|
|
|
|
struct VirtIOSoundPCMStream {
|
|
VirtIOSoundPCM *pcm;
|
|
virtio_snd_pcm_info info;
|
|
virtio_snd_pcm_set_params params;
|
|
uint32_t id;
|
|
/* channel position values (VIRTIO_SND_CHMAP_XXX) */
|
|
uint8_t positions[VIRTIO_SND_CHMAP_MAX_SIZE];
|
|
VirtIOSound *s;
|
|
bool flushing;
|
|
audsettings as;
|
|
union {
|
|
SWVoiceIn *in;
|
|
SWVoiceOut *out;
|
|
} voice;
|
|
QemuMutex queue_mutex;
|
|
bool active;
|
|
QSIMPLEQ_HEAD(, VirtIOSoundPCMBuffer) queue;
|
|
};
|
|
|
|
/*
|
|
* PCM stream state machine.
|
|
* -------------------------
|
|
*
|
|
* 5.14.6.6.1 PCM Command Lifecycle
|
|
* ================================
|
|
*
|
|
* A PCM stream has the following command lifecycle:
|
|
* - `SET PARAMETERS`
|
|
* The driver negotiates the stream parameters (format, transport, etc) with
|
|
* the device.
|
|
* Possible valid transitions: `SET PARAMETERS`, `PREPARE`.
|
|
* - `PREPARE`
|
|
* The device prepares the stream (allocates resources, etc).
|
|
* Possible valid transitions: `SET PARAMETERS`, `PREPARE`, `START`,
|
|
* `RELEASE`. Output only: the driver transfers data for pre-buffing.
|
|
* - `START`
|
|
* The device starts the stream (unmute, putting into running state, etc).
|
|
* Possible valid transitions: `STOP`.
|
|
* The driver transfers data to/from the stream.
|
|
* - `STOP`
|
|
* The device stops the stream (mute, putting into non-running state, etc).
|
|
* Possible valid transitions: `START`, `RELEASE`.
|
|
* - `RELEASE`
|
|
* The device releases the stream (frees resources, etc).
|
|
* Possible valid transitions: `SET PARAMETERS`, `PREPARE`.
|
|
*
|
|
* +---------------+ +---------+ +---------+ +-------+ +-------+
|
|
* | SetParameters | | Prepare | | Release | | Start | | Stop |
|
|
* +---------------+ +---------+ +---------+ +-------+ +-------+
|
|
* |- | | | |
|
|
* || | | | |
|
|
* |< | | | |
|
|
* |------------->| | | |
|
|
* |<-------------| | | |
|
|
* | |- | | |
|
|
* | || | | |
|
|
* | |< | | |
|
|
* | |--------------------->| |
|
|
* | |---------->| | |
|
|
* | | | |-------->|
|
|
* | | | |<--------|
|
|
* | | |<-------------------|
|
|
* |<-------------------------| | |
|
|
* | |<----------| | |
|
|
*
|
|
* CTRL in the VirtIOSound device
|
|
* ==============================
|
|
*
|
|
* The control messages that affect the state of a stream arrive in the
|
|
* `virtio_snd_handle_ctrl()` queue callback and are of type `struct
|
|
* virtio_snd_ctrl_command`. They are stored in a queue field in the device
|
|
* type, `VirtIOSound`. This allows deferring the CTRL request completion if
|
|
* it's not immediately possible due to locking/state reasons.
|
|
*
|
|
* The CTRL message is finally handled in `process_cmd()`.
|
|
*/
|
|
struct VirtIOSound {
|
|
VirtIODevice parent_obj;
|
|
|
|
VirtQueue *queues[VIRTIO_SND_VQ_MAX];
|
|
uint64_t features;
|
|
VirtIOSoundPCM *pcm;
|
|
QEMUSoundCard card;
|
|
VMChangeStateEntry *vmstate;
|
|
virtio_snd_config snd_conf;
|
|
QemuMutex cmdq_mutex;
|
|
QTAILQ_HEAD(, virtio_snd_ctrl_command) cmdq;
|
|
bool processing_cmdq;
|
|
/*
|
|
* Convenience queue to keep track of invalid tx/rx queue messages inside
|
|
* the tx/rx callbacks.
|
|
*
|
|
* In the callbacks as a first step we are emptying the virtqueue to handle
|
|
* each message and we cannot add an invalid message back to the queue: we
|
|
* would re-process it in subsequent loop iterations.
|
|
*
|
|
* Instead, we add them to this queue and after finishing examining every
|
|
* virtqueue element, we inform the guest for each invalid message.
|
|
*
|
|
* This queue must be empty at all times except for inside the tx/rx
|
|
* callbacks.
|
|
*/
|
|
QSIMPLEQ_HEAD(, VirtIOSoundPCMBuffer) invalid;
|
|
};
|
|
|
|
struct virtio_snd_ctrl_command {
|
|
VirtQueueElement *elem;
|
|
VirtQueue *vq;
|
|
virtio_snd_hdr ctrl;
|
|
virtio_snd_hdr resp;
|
|
size_t payload_size;
|
|
QTAILQ_ENTRY(virtio_snd_ctrl_command) next;
|
|
};
|
|
#endif
|