summaryrefslogtreecommitdiff
path: root/drivers/reset/reset-syscon.c
blob: 8520227d551e0b036a6947e246b62dd5f5279644 (plain)
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
// SPDX-License-Identifier: GPL-2.0
/*
 * Copyright (C) 2020 Sean Anderson
 */

#include <common.h>
#include <dm.h>
#include <regmap.h>
#include <reset.h>
#include <reset-uclass.h>
#include <syscon.h>
#include <linux/bitops.h>
#include <linux/err.h>

struct syscon_reset_priv {
	struct regmap *regmap;
	uint offset;
	uint mask;
	bool assert_high;
};

static int syscon_reset_request(struct reset_ctl *rst)
{
	struct syscon_reset_priv *priv = dev_get_priv(rst->dev);

	if (BIT(rst->id) & priv->mask)
		return 0;
	else
		return -EINVAL;
}

static int syscon_reset_assert(struct reset_ctl *rst)
{
	struct syscon_reset_priv *priv = dev_get_priv(rst->dev);

	return regmap_update_bits(priv->regmap, priv->offset, BIT(rst->id),
				  priv->assert_high ? BIT(rst->id) : 0);
}

static int syscon_reset_deassert(struct reset_ctl *rst)
{
	struct syscon_reset_priv *priv = dev_get_priv(rst->dev);

	return regmap_update_bits(priv->regmap, priv->offset, BIT(rst->id),
				  priv->assert_high ? 0 : BIT(rst->id));
}

static const struct reset_ops syscon_reset_ops = {
	.request = syscon_reset_request,
	.rst_assert = syscon_reset_assert,
	.rst_deassert = syscon_reset_deassert,
};

int syscon_reset_probe(struct udevice *dev)
{
	struct syscon_reset_priv *priv = dev_get_priv(dev);

	priv->regmap = syscon_regmap_lookup_by_phandle(dev, "regmap");
	if (IS_ERR(priv->regmap))
		return -ENODEV;

	priv->offset = dev_read_u32_default(dev, "offset", 0);
	priv->mask = dev_read_u32_default(dev, "mask", 0);
	priv->assert_high = dev_read_u32_default(dev, "assert-high", true);

	return 0;
}

static const struct udevice_id syscon_reset_ids[] = {
	{ .compatible = "syscon-reset" },
	{ },
};

U_BOOT_DRIVER(syscon_reset) = {
	.name = "syscon_reset",
	.id = UCLASS_RESET,
	.of_match = syscon_reset_ids,
	.probe = syscon_reset_probe,
	.priv_auto_alloc_size = sizeof(struct syscon_reset_priv),
	.ops = &syscon_reset_ops,
};