blob: a33f7d52dace3b6824f51a6f6aa2e350c65ffa73 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Mugunthan V Na0594ce2016-02-15 15:31:37 +05302/*
3 * Direct Memory Access U-Class driver
4 *
5 * (C) Copyright 2015
6 * Texas Instruments Incorporated, <www.ti.com>
7 *
8 * Author: Mugunthan V N <mugunthanvnm@ti.com>
Mugunthan V Na0594ce2016-02-15 15:31:37 +05309 */
10
11#include <common.h>
12#include <dma.h>
13#include <dm.h>
14#include <dm/uclass-internal.h>
15#include <dm/device-internal.h>
16#include <errno.h>
17
Mugunthan V Na0594ce2016-02-15 15:31:37 +053018int dma_get_device(u32 transfer_type, struct udevice **devp)
19{
20 struct udevice *dev;
21 int ret;
22
23 for (ret = uclass_first_device(UCLASS_DMA, &dev); dev && !ret;
24 ret = uclass_next_device(&dev)) {
25 struct dma_dev_priv *uc_priv;
26
27 uc_priv = dev_get_uclass_priv(dev);
28 if (uc_priv->supported & transfer_type)
29 break;
30 }
31
32 if (!dev) {
Masahiro Yamada9b643e32017-09-16 14:10:41 +090033 pr_err("No DMA device found that supports %x type\n",
Mugunthan V Na0594ce2016-02-15 15:31:37 +053034 transfer_type);
35 return -EPROTONOSUPPORT;
36 }
37
38 *devp = dev;
39
40 return ret;
41}
42
43int dma_memcpy(void *dst, void *src, size_t len)
44{
45 struct udevice *dev;
46 const struct dma_ops *ops;
47 int ret;
48
49 ret = dma_get_device(DMA_SUPPORTS_MEM_TO_MEM, &dev);
50 if (ret < 0)
51 return ret;
52
53 ops = device_get_ops(dev);
54 if (!ops->transfer)
55 return -ENOSYS;
56
57 /* Invalidate the area, so no writeback into the RAM races with DMA */
58 invalidate_dcache_range((unsigned long)dst, (unsigned long)dst +
59 roundup(len, ARCH_DMA_MINALIGN));
60
61 return ops->transfer(dev, DMA_MEM_TO_MEM, dst, src, len);
62}
63
64UCLASS_DRIVER(dma) = {
65 .id = UCLASS_DMA,
66 .name = "dma",
67 .flags = DM_UC_FLAG_SEQ_ALIAS,
68 .per_device_auto_alloc_size = sizeof(struct dma_dev_priv),
69};