On 2021/10/13 3:09, Thomas Monjalon wrote:
> 11/10/2021 09:33, Chengwen Feng:
>> --- /dev/null
>> +++ b/doc/guides/prog_guide/dmadev.rst
>> @@ -0,0 +1,60 @@
>> +.. SPDX-License-Identifier: BSD-3-Clause
>> +   Copyright 2021 HiSilicon Limited
>> +
>> +DMA Device Library
>> +==================
>> +
>> +The DMA library provides a DMA device framework for management and 
>> provisioning
>> +of hardware and software DMA poll mode drivers, defining generic API which
>> +support a number of different DMA operations.
>> +
>> +
>> +Design Principles
>> +-----------------
>> +
>> +The DMA framework provides a generic DMA device framework which supports 
>> both
>> +physical (hardware) and virtual (software) DMA devices, as well as a 
>> generic DMA
>> +API which allows DMA devices to be managed and configured, and supports DMA
>> +operations to be provisioned on DMA poll mode driver.
>> +
>> +.. _figure_dmadev:
>> +
>> +.. figure:: img/dmadev.*
>> +
>> +The above figure shows the model on which the DMA framework is built on:
>> +
>> + * The DMA controller could have multiple hardware DMA channels (aka. 
>> hardware
>> +   DMA queues), each hardware DMA channel should be represented by a dmadev.
>> + * The dmadev could create multiple virtual DMA channels, each virtual DMA
>> +   channel represents a different transfer context. The DMA operation 
>> request
>> +   must be submitted to the virtual DMA channel. e.g. Application could 
>> create
>> +   virtual DMA channel 0 for memory-to-memory transfer scenario, and create
>> +   virtual DMA channel 1 for memory-to-device transfer scenario.
> 
> When updating the doc, we would like to change a minimum of lines,
> so it's better to split lines logically: after a comma, a point,
> or before the next part of the sentence.
> Do not hesitate to make short lines if needed.
> Such change is quite fast to do, thanks.
> 
> [...]
>> +* **Introduced dmadev library with:**
>> +
>> +  * Device allocation functions.
> 
> You can drop this line, it is not a feature.

I'm going to try another description.

> 
> [...]
>> +static int
>> +dma_dev_data_prepare(void)
>> +{
>> +    size_t size;
>> +
>> +    if (rte_dma_devices != NULL)
>> +            return 0;
>> +
>> +    size = dma_devices_max * sizeof(struct rte_dma_dev);
>> +    rte_dma_devices = malloc(size);
>> +    if (rte_dma_devices == NULL)
>> +            return -ENOMEM;
>> +    memset(rte_dma_devices, 0, size);
>> +
>> +    return 0;
>> +}
>> +
>> +static int
>> +dma_data_prepare(void)
>> +{
>> +    if (dma_devices_max == 0)
>> +            dma_devices_max = RTE_DMADEV_DEFAULT_MAX;
>> +    return dma_dev_data_prepare();
>> +}
>> +
>> +static struct rte_dma_dev *
>> +dma_allocate(const char *name, int numa_node, size_t private_data_size)
>> +{
>> +    struct rte_dma_dev *dev;
>> +    void *dev_private;
>> +    int16_t dev_id;
>> +    int ret;
>> +
>> +    ret = dma_data_prepare();
>> +    if (ret < 0) {
>> +            RTE_DMA_LOG(ERR, "Cannot initialize dmadevs data");
>> +            return NULL;
>> +    }
>> +
>> +    dev = dma_find_by_name(name);
>> +    if (dev != NULL) {
>> +            RTE_DMA_LOG(ERR, "DMA device already allocated");
>> +            return NULL;
>> +    }
>> +
>> +    dev_private = rte_zmalloc_socket(name, private_data_size,
>> +                                     RTE_CACHE_LINE_SIZE, numa_node);
>> +    if (dev_private == NULL) {
>> +            RTE_DMA_LOG(ERR, "Cannot allocate private data");
>> +            return NULL;
>> +    }
>> +
>> +    dev_id = dma_find_free_id();
>> +    if (dev_id < 0) {
>> +            RTE_DMA_LOG(ERR, "Reached maximum number of DMA devices");
>> +            rte_free(dev_private);
>> +            return NULL;
>> +    }
>> +
>> +    dev = &rte_dma_devices[dev_id];
>> +    rte_strscpy(dev->dev_name, name, sizeof(dev->dev_name));
>> +    dev->dev_id = dev_id;
>> +    dev->numa_node = numa_node;
>> +    dev->dev_private = dev_private;
>> +
>> +    return dev;
>> +}
>> +
>> +static void
>> +dma_release(struct rte_dma_dev *dev)
>> +{
>> +    rte_free(dev->dev_private);
>> +    memset(dev, 0, sizeof(struct rte_dma_dev));
>> +}
>> +
>> +struct rte_dma_dev *
>> +rte_dma_pmd_allocate(const char *name, int numa_node, size_t 
>> private_data_size)
>> +{
>> +    struct rte_dma_dev *dev;
>> +
>> +    if (dma_check_name(name) != 0 || private_data_size == 0)
>> +            return NULL;
>> +
>> +    dev = dma_allocate(name, numa_node, private_data_size);
>> +    if (dev == NULL)
>> +            return NULL;
>> +
>> +    dev->state = RTE_DMA_DEV_REGISTERED;
>> +
>> +    return dev;
>> +}
>> +
>> +int
>> +rte_dma_pmd_release(const char *name)
>> +{
>> +    struct rte_dma_dev *dev;
>> +
>> +    if (dma_check_name(name) != 0)
>> +            return -EINVAL;
>> +
>> +    dev = dma_find_by_name(name);
>> +    if (dev == NULL)
>> +            return -EINVAL;
>> +
>> +    dma_release(dev);
>> +    return 0;
>> +}
> 
> Trying to understand the logic of creation/destroy.
> skeldma_probe
> \-> skeldma_create
>     \-> rte_dma_pmd_allocate
>         \-> dma_allocate
>             \-> dma_data_prepare
>                 \-> dma_dev_data_prepare
> skeldma_remove
> \-> skeldma_destroy
>     \-> rte_dma_pmd_release
>         \-> dma_release

This patch only provide device allocate function, the 2st patch provide extra 
logic:

        diff --git a/lib/dmadev/rte_dmadev.c b/lib/dmadev/rte_dmadev.c
        index 42a4693bd9..a6a5680d2b 100644
        --- a/lib/dmadev/rte_dmadev.c
        +++ b/lib/dmadev/rte_dmadev.c
        @@ -201,6 +201,9 @@ rte_dma_pmd_release(const char *name)
                if (dev == NULL)
                        return -EINVAL;

        +       if (dev->state == RTE_DMA_DEV_READY)
        +               return rte_dma_close(dev->dev_id);
        +
                dma_release(dev);
                return 0;
         }

So the skeldma remove will be:

 skeldma_remove
 \-> skeldma_destroy
     \-> rte_dma_pmd_release
         \-> rte_dma_close
             \-> dma_release

> app
> \-> rte_dma_close
>     \-> skeldma_close
>     \-> dma_release
> 
> My only concern is that the PMD remove does not call rte_dma_close.

If the device create success, it will call rte_dma_close in device remove phase.

> The PMD should check which dmadev is open for the rte_device to remove,
> and close the dmadev first.
> This way, no need for the function rte_dma_pmd_release,
> and no need to duplicate the release process in two paths.
> By the way, the function vchan_release is called only in the close function,
> not in the "remove path".
> 
> 
> 
> 
> .
> 

Thanks

Reply via email to