mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-18 22:14:16 +00:00 
			
		
		
		
	 d62ccb59af
			
		
	
	
		d62ccb59af
		
	
	
	
	
		
			
			virtqueue_enable_cb() will call virtqueue_poll() which will check if queue is broken at beginning, so remove the virtqueue_is_broken() call Signed-off-by: Li RongQing <lirongqing@baidu.com> Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com> Link: https://lore.kernel.org/r/20240124120834.49410-1-lirongqing@baidu.com Signed-off-by: Takashi Iwai <tiwai@suse.de>
		
			
				
	
	
		
			424 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			424 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * virtio-snd: Virtio sound device
 | |
|  * Copyright (C) 2021 OpenSynergy GmbH
 | |
|  */
 | |
| #include <sound/pcm_params.h>
 | |
| 
 | |
| #include "virtio_card.h"
 | |
| 
 | |
| /**
 | |
|  * struct virtio_pcm_msg - VirtIO I/O message.
 | |
|  * @substream: VirtIO PCM substream.
 | |
|  * @xfer: Request header payload.
 | |
|  * @status: Response header payload.
 | |
|  * @length: Data length in bytes.
 | |
|  * @sgs: Payload scatter-gather table.
 | |
|  */
 | |
| struct virtio_pcm_msg {
 | |
| 	struct virtio_pcm_substream *substream;
 | |
| 	struct virtio_snd_pcm_xfer xfer;
 | |
| 	struct virtio_snd_pcm_status status;
 | |
| 	size_t length;
 | |
| 	struct scatterlist sgs[];
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * enum pcm_msg_sg_index - Index values for the virtio_pcm_msg->sgs field in
 | |
|  *                         an I/O message.
 | |
|  * @PCM_MSG_SG_XFER: Element containing a virtio_snd_pcm_xfer structure.
 | |
|  * @PCM_MSG_SG_STATUS: Element containing a virtio_snd_pcm_status structure.
 | |
|  * @PCM_MSG_SG_DATA: The first element containing a data buffer.
 | |
|  */
 | |
| enum pcm_msg_sg_index {
 | |
| 	PCM_MSG_SG_XFER = 0,
 | |
| 	PCM_MSG_SG_STATUS,
 | |
| 	PCM_MSG_SG_DATA
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_sg_num() - Count the number of sg-elements required to represent
 | |
|  *                        vmalloc'ed buffer.
 | |
|  * @data: Pointer to vmalloc'ed buffer.
 | |
|  * @length: Buffer size.
 | |
|  *
 | |
|  * Context: Any context.
 | |
|  * Return: Number of physically contiguous parts in the @data.
 | |
|  */
 | |
| static int virtsnd_pcm_sg_num(u8 *data, unsigned int length)
 | |
| {
 | |
| 	phys_addr_t sg_address;
 | |
| 	unsigned int sg_length;
 | |
| 	int num = 0;
 | |
| 
 | |
| 	while (length) {
 | |
| 		struct page *pg = vmalloc_to_page(data);
 | |
| 		phys_addr_t pg_address = page_to_phys(pg);
 | |
| 		size_t pg_length;
 | |
| 
 | |
| 		pg_length = PAGE_SIZE - offset_in_page(data);
 | |
| 		if (pg_length > length)
 | |
| 			pg_length = length;
 | |
| 
 | |
| 		if (!num || sg_address + sg_length != pg_address) {
 | |
| 			sg_address = pg_address;
 | |
| 			sg_length = pg_length;
 | |
| 			num++;
 | |
| 		} else {
 | |
| 			sg_length += pg_length;
 | |
| 		}
 | |
| 
 | |
| 		data += pg_length;
 | |
| 		length -= pg_length;
 | |
| 	}
 | |
| 
 | |
| 	return num;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_sg_from() - Build sg-list from vmalloc'ed buffer.
 | |
|  * @sgs: Preallocated sg-list to populate.
 | |
|  * @nsgs: The maximum number of elements in the @sgs.
 | |
|  * @data: Pointer to vmalloc'ed buffer.
 | |
|  * @length: Buffer size.
 | |
|  *
 | |
|  * Splits the buffer into physically contiguous parts and makes an sg-list of
 | |
|  * such parts.
 | |
|  *
 | |
|  * Context: Any context.
 | |
|  */
 | |
| static void virtsnd_pcm_sg_from(struct scatterlist *sgs, int nsgs, u8 *data,
 | |
| 				unsigned int length)
 | |
| {
 | |
| 	int idx = -1;
 | |
| 
 | |
| 	while (length) {
 | |
| 		struct page *pg = vmalloc_to_page(data);
 | |
| 		size_t pg_length;
 | |
| 
 | |
| 		pg_length = PAGE_SIZE - offset_in_page(data);
 | |
| 		if (pg_length > length)
 | |
| 			pg_length = length;
 | |
| 
 | |
| 		if (idx == -1 ||
 | |
| 		    sg_phys(&sgs[idx]) + sgs[idx].length != page_to_phys(pg)) {
 | |
| 			if (idx + 1 == nsgs)
 | |
| 				break;
 | |
| 			sg_set_page(&sgs[++idx], pg, pg_length,
 | |
| 				    offset_in_page(data));
 | |
| 		} else {
 | |
| 			sgs[idx].length += pg_length;
 | |
| 		}
 | |
| 
 | |
| 		data += pg_length;
 | |
| 		length -= pg_length;
 | |
| 	}
 | |
| 
 | |
| 	sg_mark_end(&sgs[idx]);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_msg_alloc() - Allocate I/O messages.
 | |
|  * @vss: VirtIO PCM substream.
 | |
|  * @periods: Current number of periods.
 | |
|  * @period_bytes: Current period size in bytes.
 | |
|  *
 | |
|  * The function slices the buffer into @periods parts (each with the size of
 | |
|  * @period_bytes), and creates @periods corresponding I/O messages.
 | |
|  *
 | |
|  * Context: Any context that permits to sleep.
 | |
|  * Return: 0 on success, -ENOMEM on failure.
 | |
|  */
 | |
| int virtsnd_pcm_msg_alloc(struct virtio_pcm_substream *vss,
 | |
| 			  unsigned int periods, unsigned int period_bytes)
 | |
| {
 | |
| 	struct snd_pcm_runtime *runtime = vss->substream->runtime;
 | |
| 	unsigned int i;
 | |
| 
 | |
| 	vss->msgs = kcalloc(periods, sizeof(*vss->msgs), GFP_KERNEL);
 | |
| 	if (!vss->msgs)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	vss->nmsgs = periods;
 | |
| 
 | |
| 	for (i = 0; i < periods; ++i) {
 | |
| 		u8 *data = runtime->dma_area + period_bytes * i;
 | |
| 		int sg_num = virtsnd_pcm_sg_num(data, period_bytes);
 | |
| 		struct virtio_pcm_msg *msg;
 | |
| 
 | |
| 		msg = kzalloc(struct_size(msg, sgs, sg_num + 2), GFP_KERNEL);
 | |
| 		if (!msg)
 | |
| 			return -ENOMEM;
 | |
| 
 | |
| 		msg->substream = vss;
 | |
| 		sg_init_one(&msg->sgs[PCM_MSG_SG_XFER], &msg->xfer,
 | |
| 			    sizeof(msg->xfer));
 | |
| 		sg_init_one(&msg->sgs[PCM_MSG_SG_STATUS], &msg->status,
 | |
| 			    sizeof(msg->status));
 | |
| 		virtsnd_pcm_sg_from(&msg->sgs[PCM_MSG_SG_DATA], sg_num, data,
 | |
| 				    period_bytes);
 | |
| 
 | |
| 		vss->msgs[i] = msg;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_msg_free() - Free all allocated I/O messages.
 | |
|  * @vss: VirtIO PCM substream.
 | |
|  *
 | |
|  * Context: Any context.
 | |
|  */
 | |
| void virtsnd_pcm_msg_free(struct virtio_pcm_substream *vss)
 | |
| {
 | |
| 	unsigned int i;
 | |
| 
 | |
| 	for (i = 0; vss->msgs && i < vss->nmsgs; ++i)
 | |
| 		kfree(vss->msgs[i]);
 | |
| 	kfree(vss->msgs);
 | |
| 
 | |
| 	vss->msgs = NULL;
 | |
| 	vss->nmsgs = 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_msg_send() - Send asynchronous I/O messages.
 | |
|  * @vss: VirtIO PCM substream.
 | |
|  * @offset: starting position that has been updated
 | |
|  * @bytes: number of bytes that has been updated
 | |
|  *
 | |
|  * All messages are organized in an ordered circular list. Each time the
 | |
|  * function is called, all currently non-enqueued messages are added to the
 | |
|  * virtqueue. For this, the function uses offset and bytes to calculate the
 | |
|  * messages that need to be added.
 | |
|  *
 | |
|  * Context: Any context. Expects the tx/rx queue and the VirtIO substream
 | |
|  *          spinlocks to be held by caller.
 | |
|  * Return: 0 on success, -errno on failure.
 | |
|  */
 | |
| int virtsnd_pcm_msg_send(struct virtio_pcm_substream *vss, unsigned long offset,
 | |
| 			 unsigned long bytes)
 | |
| {
 | |
| 	struct virtio_snd *snd = vss->snd;
 | |
| 	struct virtio_device *vdev = snd->vdev;
 | |
| 	struct virtqueue *vqueue = virtsnd_pcm_queue(vss)->vqueue;
 | |
| 	unsigned long period_bytes = snd_pcm_lib_period_bytes(vss->substream);
 | |
| 	unsigned long start, end, i;
 | |
| 	unsigned int msg_count = vss->msg_count;
 | |
| 	bool notify = false;
 | |
| 	int rc;
 | |
| 
 | |
| 	start = offset / period_bytes;
 | |
| 	end = (offset + bytes - 1) / period_bytes;
 | |
| 
 | |
| 	for (i = start; i <= end; i++) {
 | |
| 		struct virtio_pcm_msg *msg = vss->msgs[i];
 | |
| 		struct scatterlist *psgs[] = {
 | |
| 			&msg->sgs[PCM_MSG_SG_XFER],
 | |
| 			&msg->sgs[PCM_MSG_SG_DATA],
 | |
| 			&msg->sgs[PCM_MSG_SG_STATUS]
 | |
| 		};
 | |
| 		unsigned long n;
 | |
| 
 | |
| 		n = period_bytes - (offset % period_bytes);
 | |
| 		if (n > bytes)
 | |
| 			n = bytes;
 | |
| 
 | |
| 		msg->length += n;
 | |
| 		if (msg->length == period_bytes) {
 | |
| 			msg->xfer.stream_id = cpu_to_le32(vss->sid);
 | |
| 			memset(&msg->status, 0, sizeof(msg->status));
 | |
| 
 | |
| 			if (vss->direction == SNDRV_PCM_STREAM_PLAYBACK)
 | |
| 				rc = virtqueue_add_sgs(vqueue, psgs, 2, 1, msg,
 | |
| 						       GFP_ATOMIC);
 | |
| 			else
 | |
| 				rc = virtqueue_add_sgs(vqueue, psgs, 1, 2, msg,
 | |
| 						       GFP_ATOMIC);
 | |
| 
 | |
| 			if (rc) {
 | |
| 				dev_err(&vdev->dev,
 | |
| 					"SID %u: failed to send I/O message\n",
 | |
| 					vss->sid);
 | |
| 				return rc;
 | |
| 			}
 | |
| 
 | |
| 			vss->msg_count++;
 | |
| 		}
 | |
| 
 | |
| 		offset = 0;
 | |
| 		bytes -= n;
 | |
| 	}
 | |
| 
 | |
| 	if (msg_count == vss->msg_count)
 | |
| 		return 0;
 | |
| 
 | |
| 	if (!(vss->features & (1U << VIRTIO_SND_PCM_F_MSG_POLLING)))
 | |
| 		notify = virtqueue_kick_prepare(vqueue);
 | |
| 
 | |
| 	if (notify)
 | |
| 		virtqueue_notify(vqueue);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_msg_pending_num() - Returns the number of pending I/O messages.
 | |
|  * @vss: VirtIO substream.
 | |
|  *
 | |
|  * Context: Any context.
 | |
|  * Return: Number of messages.
 | |
|  */
 | |
| unsigned int virtsnd_pcm_msg_pending_num(struct virtio_pcm_substream *vss)
 | |
| {
 | |
| 	unsigned int num;
 | |
| 	unsigned long flags;
 | |
| 
 | |
| 	spin_lock_irqsave(&vss->lock, flags);
 | |
| 	num = vss->msg_count;
 | |
| 	spin_unlock_irqrestore(&vss->lock, flags);
 | |
| 
 | |
| 	return num;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_msg_complete() - Complete an I/O message.
 | |
|  * @msg: I/O message.
 | |
|  * @written_bytes: Number of bytes written to the message.
 | |
|  *
 | |
|  * Completion of the message means the elapsed period. If transmission is
 | |
|  * allowed, then each completed message is immediately placed back at the end
 | |
|  * of the queue.
 | |
|  *
 | |
|  * For the playback substream, @written_bytes is equal to sizeof(msg->status).
 | |
|  *
 | |
|  * For the capture substream, @written_bytes is equal to sizeof(msg->status)
 | |
|  * plus the number of captured bytes.
 | |
|  *
 | |
|  * Context: Interrupt context. Takes and releases the VirtIO substream spinlock.
 | |
|  */
 | |
| static void virtsnd_pcm_msg_complete(struct virtio_pcm_msg *msg,
 | |
| 				     size_t written_bytes)
 | |
| {
 | |
| 	struct virtio_pcm_substream *vss = msg->substream;
 | |
| 
 | |
| 	/*
 | |
| 	 * hw_ptr always indicates the buffer position of the first I/O message
 | |
| 	 * in the virtqueue. Therefore, on each completion of an I/O message,
 | |
| 	 * the hw_ptr value is unconditionally advanced.
 | |
| 	 */
 | |
| 	spin_lock(&vss->lock);
 | |
| 	/*
 | |
| 	 * If the capture substream returned an incorrect status, then just
 | |
| 	 * increase the hw_ptr by the message size.
 | |
| 	 */
 | |
| 	if (vss->direction == SNDRV_PCM_STREAM_PLAYBACK ||
 | |
| 	    written_bytes <= sizeof(msg->status))
 | |
| 		vss->hw_ptr += msg->length;
 | |
| 	else
 | |
| 		vss->hw_ptr += written_bytes - sizeof(msg->status);
 | |
| 
 | |
| 	if (vss->hw_ptr >= vss->buffer_bytes)
 | |
| 		vss->hw_ptr -= vss->buffer_bytes;
 | |
| 
 | |
| 	msg->length = 0;
 | |
| 
 | |
| 	vss->xfer_xrun = false;
 | |
| 	vss->msg_count--;
 | |
| 
 | |
| 	if (vss->xfer_enabled) {
 | |
| 		struct snd_pcm_runtime *runtime = vss->substream->runtime;
 | |
| 
 | |
| 		runtime->delay =
 | |
| 			bytes_to_frames(runtime,
 | |
| 					le32_to_cpu(msg->status.latency_bytes));
 | |
| 
 | |
| 		schedule_work(&vss->elapsed_period);
 | |
| 	} else if (!vss->msg_count) {
 | |
| 		wake_up_all(&vss->msg_empty);
 | |
| 	}
 | |
| 	spin_unlock(&vss->lock);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_notify_cb() - Process all completed I/O messages.
 | |
|  * @queue: Underlying tx/rx virtqueue.
 | |
|  *
 | |
|  * Context: Interrupt context. Takes and releases the tx/rx queue spinlock.
 | |
|  */
 | |
| static inline void virtsnd_pcm_notify_cb(struct virtio_snd_queue *queue)
 | |
| {
 | |
| 	struct virtio_pcm_msg *msg;
 | |
| 	u32 written_bytes;
 | |
| 	unsigned long flags;
 | |
| 
 | |
| 	spin_lock_irqsave(&queue->lock, flags);
 | |
| 	do {
 | |
| 		virtqueue_disable_cb(queue->vqueue);
 | |
| 		while ((msg = virtqueue_get_buf(queue->vqueue, &written_bytes)))
 | |
| 			virtsnd_pcm_msg_complete(msg, written_bytes);
 | |
| 	} while (!virtqueue_enable_cb(queue->vqueue));
 | |
| 	spin_unlock_irqrestore(&queue->lock, flags);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_tx_notify_cb() - Process all completed TX messages.
 | |
|  * @vqueue: Underlying tx virtqueue.
 | |
|  *
 | |
|  * Context: Interrupt context.
 | |
|  */
 | |
| void virtsnd_pcm_tx_notify_cb(struct virtqueue *vqueue)
 | |
| {
 | |
| 	struct virtio_snd *snd = vqueue->vdev->priv;
 | |
| 
 | |
| 	virtsnd_pcm_notify_cb(virtsnd_tx_queue(snd));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_rx_notify_cb() - Process all completed RX messages.
 | |
|  * @vqueue: Underlying rx virtqueue.
 | |
|  *
 | |
|  * Context: Interrupt context.
 | |
|  */
 | |
| void virtsnd_pcm_rx_notify_cb(struct virtqueue *vqueue)
 | |
| {
 | |
| 	struct virtio_snd *snd = vqueue->vdev->priv;
 | |
| 
 | |
| 	virtsnd_pcm_notify_cb(virtsnd_rx_queue(snd));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * virtsnd_pcm_ctl_msg_alloc() - Allocate and initialize the PCM device control
 | |
|  *                               message for the specified substream.
 | |
|  * @vss: VirtIO PCM substream.
 | |
|  * @command: Control request code (VIRTIO_SND_R_PCM_XXX).
 | |
|  * @gfp: Kernel flags for memory allocation.
 | |
|  *
 | |
|  * Context: Any context. May sleep if @gfp flags permit.
 | |
|  * Return: Allocated message on success, NULL on failure.
 | |
|  */
 | |
| struct virtio_snd_msg *
 | |
| virtsnd_pcm_ctl_msg_alloc(struct virtio_pcm_substream *vss,
 | |
| 			  unsigned int command, gfp_t gfp)
 | |
| {
 | |
| 	size_t request_size = sizeof(struct virtio_snd_pcm_hdr);
 | |
| 	size_t response_size = sizeof(struct virtio_snd_hdr);
 | |
| 	struct virtio_snd_msg *msg;
 | |
| 
 | |
| 	switch (command) {
 | |
| 	case VIRTIO_SND_R_PCM_SET_PARAMS:
 | |
| 		request_size = sizeof(struct virtio_snd_pcm_set_params);
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	msg = virtsnd_ctl_msg_alloc(request_size, response_size, gfp);
 | |
| 	if (msg) {
 | |
| 		struct virtio_snd_pcm_hdr *hdr = virtsnd_ctl_msg_request(msg);
 | |
| 
 | |
| 		hdr->hdr.code = cpu_to_le32(command);
 | |
| 		hdr->stream_id = cpu_to_le32(vss->sid);
 | |
| 	}
 | |
| 
 | |
| 	return msg;
 | |
| }
 |