Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 | // SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2019, Linaro Limited */ #include <log.h> #include <malloc.h> #include <asm/io.h> #include <dm.h> #include <dt-bindings/reset/ti-syscon.h> #include <reset-uclass.h> #include <linux/bitops.h> struct hisi_reset_priv { void __iomem *base; }; static int hisi_reset_deassert(struct reset_ctl *rst) { struct hisi_reset_priv *priv = dev_get_priv(rst->dev); u32 val; val = readl(priv->base + rst->data); if (rst->polarity & DEASSERT_SET) val |= BIT(rst->id); else val &= ~BIT(rst->id); writel(val, priv->base + rst->data); return 0; } static int hisi_reset_assert(struct reset_ctl *rst) { struct hisi_reset_priv *priv = dev_get_priv(rst->dev); u32 val; val = readl(priv->base + rst->data); if (rst->polarity & ASSERT_SET) val |= BIT(rst->id); else val &= ~BIT(rst->id); writel(val, priv->base + rst->data); return 0; } static int hisi_reset_of_xlate(struct reset_ctl *rst, struct ofnode_phandle_args *args) { unsigned long polarity; switch (args->args_count) { case 2: polarity = ASSERT_SET; break; case 3: polarity = args->args[2]; break; default: debug("Invalid args_count: %d\n", args->args_count); return -EINVAL; } /* Use .data field as register offset and .id field as bit shift */ rst->data = args->args[0]; rst->id = args->args[1]; rst->polarity = polarity; return 0; } static const struct reset_ops hisi_reset_reset_ops = { .of_xlate = hisi_reset_of_xlate, .rst_assert = hisi_reset_assert, .rst_deassert = hisi_reset_deassert, }; static const struct udevice_id hisi_reset_ids[] = { { .compatible = "hisilicon,hi3798cv200-reset" }, { } }; static int hisi_reset_probe(struct udevice *dev) { struct hisi_reset_priv *priv = dev_get_priv(dev); priv->base = dev_remap_addr(dev); if (!priv->base) return -ENOMEM; return 0; } U_BOOT_DRIVER(hisi_reset) = { .name = "hisilicon_reset", .id = UCLASS_RESET, .of_match = hisi_reset_ids, .ops = &hisi_reset_reset_ops, .probe = hisi_reset_probe, .priv_auto = sizeof(struct hisi_reset_priv), }; |