Re: [RFC 1/2] dma-buf: Introduce dma buffer sharing mechanism

From: Sumit Semwal
Date: Fri Oct 14 2011 - 10:14:07 EST


On 14 October 2011 15:30, Tomasz Stanislawski <t.stanislaws@xxxxxxxxxxx> wrote:
> Hi Mr. Sumit Semwal,
Hello Mr. Tomasz Stanislawski :),

> Thank you for taking care of the framework for buffer sharing.
> The support of buffer sharing in V4L2, both exporting and importing was
> posted in shrbuf proof-of-concept patch. It should be easy to port it to
> dmabuf.
>
> http://lists.linaro.org/pipermail/linaro-mm-sig/2011-August/000485.html
I should thank you for the wonderful proof-of-concept patch, and the
idea behind it! I am currently working on the V4L2 side patch for it,
and would send out the RFC soon.
Also, thanks for a good review and some pertinent points; replies inline.
>
> Please refer to the comments below:
>
> On 10/11/2011 11:23 AM, Sumit Semwal wrote:
>>
<snip>
>> +/**
>> + * dma_buf_export - Creates a new dma_buf, and associates an anon file
>> + * with this buffer,so it can be exported.
>> + * Also connect the allocator specific data and ops to the buffer.
>> + *
>> + * @priv: Â Â Â[in] Â ÂAttach private data of allocator to this buffer
>> + * @ops: Â Â Â [in] Â ÂAttach allocator-defined dma buf ops to the new
>> buffer.
>> + * @flags: Â Â [in] Â Âmode flags for the file.
>
> What is the purpose of these flags? The file is not visible to any process
> by any file system, is it?
These are the standard file mode flags, which can be used to do
file-level access-type control by the exporter, so for example
write-access can be denied for a buffer exported as a read-only
buffer.
>
>> + *
>> + * Returns, on success, a newly created dma_buf object, which wraps the
>> + * supplied private data and operations for dma_buf_ops. On failure to
>> + * allocate the dma_buf object, it can return NULL.
>> + *
>> + */
>> +struct dma_buf *dma_buf_export(void *priv, struct dma_buf_ops *ops,
>> + Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â int flags)
>> +{
>> + Â Â Â struct dma_buf *dmabuf;
>> + Â Â Â struct file *file;
>> +
>
> why priv is not allowed to be NULL?
priv will be used by the exporter to attach its own context to the dma
buf; I couldn't think of any case where it could be NULL?
>
>> + Â Â Â BUG_ON(!priv || !ops);
>> +
<snip>
>> + Â Â Â BUG_ON(!dmabuf->file);
>> +
>> + Â Â Â fput(dmabuf->file);
>> +
>
> return is not needed
Right; will correct this.
>
>> + Â Â Â return;
>> +}
>> +EXPORT_SYMBOL(dma_buf_put);
>> +
>> +/**
>> + * dma_buf_attach - Add the device to dma_buf's attachments list;
>> optionally,
>> + * calls attach() of dma_buf_ops to allow device-specific attach
>> functionality
>> + * @dmabuf: Â Â[in] Â Âbuffer to attach device to.
>> + * @dev: Â Â Â [in] Â Âdevice to be attached.
>> + *
>> + * Returns struct dma_buf_attachment * for this attachment; may return
>> NULL.
>> + *
>> + */
>> +struct dma_buf_attachment *dma_buf_attach(struct dma_buf *dmabuf,
>> + Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â struct device *dev)
>> +{
>> + Â Â Â struct dma_buf_attachment *attach;
>> + Â Â Â int ret;
>> +
>> + Â Â Â BUG_ON(!dmabuf || !dev);
>> +
>> + Â Â Â mutex_lock(&dmabuf->lock);
>> +
>
> There is no need to call kzalloc inside critical section protected by
> dmabuf->lock. The code would be simpler if the allocation is moved outside
> the section.
Yes, you're right; will correct it.
>
>> + Â Â Â attach = kzalloc(sizeof(struct dma_buf_attachment), GFP_KERNEL);
>> + Â Â Â if (attach == NULL)
>> + Â Â Â Â Â Â Â goto err_alloc;
>> +
>> + Â Â Â attach->dev = dev;
>> + Â Â Â if (dmabuf->ops->attach) {
>> + Â Â Â Â Â Â Â ret = dmabuf->ops->attach(dmabuf, dev, attach);
>> + Â Â Â Â Â Â Â if (!ret)
>> + Â Â Â Â Â Â Â Â Â Â Â goto err_attach;
>> + Â Â Â }
>> + Â Â Â list_add(&attach->node,&dmabuf->attachments);
>> +
>> +err_alloc:
>> + Â Â Â mutex_unlock(&dmabuf->lock);
>> + Â Â Â return attach;
>> +err_attach:
>> + Â Â Â kfree(attach);
>> + Â Â Â mutex_unlock(&dmabuf->lock);
>> + Â Â Â return ERR_PTR(ret);
>> +}
>> +EXPORT_SYMBOL(dma_buf_attach);
>> +
>> +/**
>> + * dma_buf_detach - Remove the given attachment from dmabuf's attachments
>> list;
>> + * optionally calls detach() of dma_buf_ops for device-specific detach
>> + * @dmabuf: Â Â[in] Â Âbuffer to detach from.
>> + * @attach: Â Â[in] Â Âattachment to be detached; is free'd after this
>> call.
>> + *
>> + */
>> +void dma_buf_detach(struct dma_buf *dmabuf, struct dma_buf_attachment
>> *attach)
>> +{
>> + Â Â Â BUG_ON(!dmabuf || !attach);
>> +
>> + Â Â Â mutex_lock(&dmabuf->lock);
>> + Â Â Â list_del(&attach->node);
>> + Â Â Â if (dmabuf->ops->detach)
>> + Â Â Â Â Â Â Â dmabuf->ops->detach(dmabuf, attach);
>> +
>
> [as above]
Ok.
>>
>> + Â Â Â kfree(attach);
>> + Â Â Â mutex_unlock(&dmabuf->lock);
>> + Â Â Â return;
>> +}
>> +EXPORT_SYMBOL(dma_buf_detach);
>> diff --git a/include/linux/dma-buf.h b/include/linux/dma-buf.h
>> new file mode 100644
>> index 0000000..5bdf16a
>> --- /dev/null
>> +++ b/include/linux/dma-buf.h
>> @@ -0,0 +1,162 @@
>> +/*
>> + * Header file for dma buffer sharing framework.
>> + *
>> + * Copyright(C) 2011 Linaro Limited. All rights reserved.
>> + * Author: Sumit Semwal<sumit.semwal@xxxxxx>
>> + *
>> + * Many thanks to linaro-mm-sig list, and specially
>> + * Arnd Bergmann<arnd@xxxxxxxx>, Rob Clark<rob@xxxxxx> Âand
>> + * Daniel Vetter<daniel@xxxxxxxx> Âfor their support in creation and
>> + * refining of this idea.
>> + *
>> + * This program is free software; you can redistribute it and/or modify
>> it
>> + * under the terms of the GNU General Public License version 2 as
>> published by
>> + * the Free Software Foundation.
>> + *
>> + * This program is distributed in the hope that it will be useful, but
>> WITHOUT
>> + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
>> + * FITNESS FOR A PARTICULAR PURPOSE. ÂSee the GNU General Public License
>> for
>> + * more details.
>> + *
>> + * You should have received a copy of the GNU General Public License
>> along with
>> + * this program. ÂIf not, see<http://www.gnu.org/licenses/>.
>> + */
>> +#ifndef __DMA_BUF_H__
>> +#define __DMA_BUF_H__
>> +
>> +#include<linux/file.h>
>> +#include<linux/err.h>
>> +#include<linux/device.h>
>> +#include<linux/scatterlist.h>
>> +#include<linux/list.h>
>> +#include<linux/dma-mapping.h>
>> +
>> +struct dma_buf;
>> +
>> +/**
>> + * struct dma_buf_attachment - holds device-buffer attachment data
>> + * @dmabuf: buffer for this attachment.
>> + * @dev: device attached to the buffer.
>> + * @node: list_head to allow manipulation of list of dma_buf_attachment.
>> + * @priv: exporter-specific attachment data.
>> + */
>> +struct dma_buf_attachment {
>> + Â Â Â struct dma_buf *dmabuf;
>> + Â Â Â struct device *dev;
>> + Â Â Â struct list_head node;
>> + Â Â Â void *priv;
>> +};
>> +
>> +/**
>> + * struct dma_buf_ops - operations possible on struct dma_buf
>> + * @create: creates a struct dma_buf of a fixed size. Actual allocation
>> + * Â Â Â Â does not happen here.
>
> The 'create' ops is not present in dma_buf_ops.
Yes, this is a copy-paste mistake; will correct.
>> + * @attach: allows different devices to 'attach' themselves to the given
>> + * Â Â Â Â buffer. It might return -EBUSY to signal that backing storage
>> + * Â Â Â Â is already allocated and incompatible with the requirements
>> + * Â Â Â Â of requesting device. [optional]
>> + * @detach: detach a given device from this buffer. [optional]
>> + * @get_scatterlist: returns list of scatter pages allocated, increases
>> + * Â Â Â Â Â Â Â Â Âusecount of the buffer. Requires atleast one attach
>> to be
>> + * Â Â Â Â Â Â Â Â Âcalled before. Returned sg list should already be
>> mapped
>> + * Â Â Â Â Â Â Â Â Âinto _device_ address space.
>
> You must add a comment that this call 'may sleep'.
Thanks, I will add this.
>
> I like the get_scatterlist idea. It allows the exported to create a valid
> scatterlist for a client in a elegant way.
>
> I do not like this whole attachment idea. The problem is that currently
> there is no support in DMA framework for allocation for multiple devices. As
> long as no such a support exists, there is no generic way to handle
> attribute negotiations and buffer allocations that involve multiple devices.
> So the exporter drivers would have to implement more or less hacky solutions
> to handle memory requirements and choosing the device that allocated memory.

You are right, there is currently no generic way for attribute
negotiation for buffers of multiple devices. However, since buffer
sharing idea itself is about allowing for buffer allocation and
migration based on 'some' negotiation, the attachment idea makes it
easier to do so.

What you choose to call 'more or less hacky solution', I would call
'platform-specific mechanism' for negotiation, allocation (and
migration if possible and required). :)
>
> Currently, AFAIK there is even no generic way for a driver to acquire its
> own DMA memory requirements.
>
> Therefore all logic hidden beneath 'attachment' is pointless. I think that
> support for attach/detach (and related stuff) should be postponed until
> support for multi-device allocation is added to DMA framework.
>
> I don't say the attachment list idea is wrong but adding attachment stuff
> creates an illusion that problem of multi-device allocations is somehow
> magically solved. We should not force the developers of exporter drivers to
> solve the problem that is not solvable yet.

I quite like what you said - the problem of having a 'generic'
mechanism of DMA attribute negotiation might not have been solved yet,
and so the exporter drivers shouldn't need to try to do that.
However, exporter drivers would most likely know about the negotiation
etc needed and possible on their given platform - and the attachment
mechanism gives them a way to gather requirements from participating
DMA users, and use it for such decisions.

>
> The other problem are the APIs. For example, the V4L2 subsystem assumes that
> memory is allocated after successful VIDIOC_REQBUFS with V4L2_MEMORY_MMAP
> memory type. Therefore attach would be automatically followed by
> get_scatterlist, blocking possibility of any buffer migrations in future.
That is correct for V4L2_MEMORY_MMAP; should that need to be a
requirement even for V4L2_MEMORY_DMABUF_USER memory type? [I am
deliberately distinguishing a V4L2 driver which is a shared buffer
exporter from a V4L2 shared-buffer-user driver] - could we not do a
get_scatterlist on each VIDIOC_QBUF, and put_scatterlist on each
VIDIOC_DQBUF?
>
> The same situation happens if buffer sharing is added to framebuffer API.
>
> The buffer sharing mechanism is dedicated to improve cooperation between
> multiple APIs. Therefore the common denominator strategy should be applied
> that is buffer-creation == buffer-allocation.

It might be prudent to do so in the media drivers. But the buffer
sharing mechanism is also useful for other, non-media usecases? I
would say that we should be able to leave enough flexibility.
>
>> + * @put_scatterlist: decreases usecount of buffer, might deallocate
>> scatter
>> + * Â Â Â Â Â Â Â Â Âpages.
>> + * @mmap: memory map this buffer - optional.
>> + * @release: release this buffer; to be called after the last
>> dma_buf_put.
>> + * @sync_sg_for_cpu: sync the sg list for cpu.
>> + * @sync_sg_for_device: synch the sg list for device.
>> + */
>> +struct dma_buf_ops {
>> + Â Â Â int (*attach)(struct dma_buf *, struct device *,
>> + Â Â Â Â Â Â Â Â Â Â Â struct dma_buf_attachment *);
>> +
>> + Â Â Â void (*detach)(struct dma_buf *, struct dma_buf_attachment *);
>> +
>> + Â Â Â /* For {get,put}_scatterlist below, any specific buffer attributes
>> + Â Â Â Â* required should get added to device_dma_parameters accessible
>> + Â Â Â Â* via dev->dma_params.
>> + Â Â Â Â*/
>> + Â Â Â struct scatterlist * (*get_scatterlist)(struct dma_buf_attachment
>> *,
>> + Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â enum dma_data_direction,
>> + Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â int *nents);
>> + Â Â Â void (*put_scatterlist)(struct dma_buf_attachment *,
>> + Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â struct scatterlist *,
>> + Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â Â int nents);
>> + Â Â Â /* TODO: Add interruptible and interruptible_timeout versions */
>
> I don't agree the interruptible and interruptible_timeout versions are
> needed. I think that get_scatterlist should alway be interruptible. You can
> add try_get_scatterlist callback that returns ERR_PTR(-EBUSY) if the call
> would be blocking.
Sure, that's a good idea - I could do that.
>
>> +
>> + Â Â Â /* allow mmap optionally for devices that need it */
>> + Â Â Â int (*mmap)(struct dma_buf *, struct vm_area_struct *);
>
> The mmap is not needed for inital version. It could be added at any time in
> the future. The dmabuf client should not be allowed to create mapping of the
> dmabuf from the scatterlist.
There's already a discussion between Rob and David on this mail-chain
earlier; I guess we'll need to wait a little before deciding to remove
this.
>
<snip>
>
> I hope you find my comments useful.
Like I said Tomasz, I thank you for a very good review! Hope my
replies satisfy some of the concerns you raised?
>
> Yours sincerely,
> Tomasz Stanislawski
>

--
Thanks and regards,
Sumit Semwal
Linaro Kernel Engineer - Graphics working group
Linaro.orgÂâÂOpen source software for ARM SoCs
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/