summaryrefslogtreecommitdiff
path: root/drivers/gpu/drm/bridge/ti-tdp158.c
blob: 3472ed5924e8b939d6b55a489fbb9989114d35db (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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright 2024 Freebox SAS
 */

#include <linux/gpio/consumer.h>
#include <linux/i2c.h>

#include <drm/drm_atomic_helper.h>
#include <drm/drm_bridge.h>

struct tdp158 {
	struct drm_bridge bridge;
	struct drm_bridge *next;
	struct gpio_desc *enable; // Operation Enable - pin 36
	struct regulator *vcc; // 3.3V
	struct regulator *vdd; // 1.1V
	struct device *dev;
};

static void tdp158_enable(struct drm_bridge *bridge, struct drm_bridge_state *prev)
{
	int err;
	struct tdp158 *tdp158 = bridge->driver_private;

	err = regulator_enable(tdp158->vcc);
	if (err)
		dev_err(tdp158->dev, "failed to enable vcc: %d", err);

	err = regulator_enable(tdp158->vdd);
	if (err)
		dev_err(tdp158->dev, "failed to enable vdd: %d", err);

	gpiod_set_value_cansleep(tdp158->enable, 1);
}

static void tdp158_disable(struct drm_bridge *bridge, struct drm_bridge_state *prev)
{
	struct tdp158 *tdp158 = bridge->driver_private;

	gpiod_set_value_cansleep(tdp158->enable, 0);
	regulator_disable(tdp158->vdd);
	regulator_disable(tdp158->vcc);
}

static int tdp158_attach(struct drm_bridge *bridge, enum drm_bridge_attach_flags flags)
{
	struct tdp158 *tdp158 = bridge->driver_private;

	return drm_bridge_attach(bridge->encoder, tdp158->next, bridge, flags);
}

static const struct drm_bridge_funcs tdp158_bridge_funcs = {
	.attach = tdp158_attach,
	.atomic_enable = tdp158_enable,
	.atomic_disable = tdp158_disable,
	.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
	.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
	.atomic_reset = drm_atomic_helper_bridge_reset,
};

static int tdp158_probe(struct i2c_client *client)
{
	struct tdp158 *tdp158;
	struct device *dev = &client->dev;

	tdp158 = devm_kzalloc(dev, sizeof(*tdp158), GFP_KERNEL);
	if (!tdp158)
		return -ENOMEM;

	tdp158->next = devm_drm_of_get_bridge(dev, dev->of_node, 1, 0);
	if (IS_ERR(tdp158->next))
		return dev_err_probe(dev, PTR_ERR(tdp158->next), "missing bridge");

	tdp158->vcc = devm_regulator_get(dev, "vcc");
	if (IS_ERR(tdp158->vcc))
		return dev_err_probe(dev, PTR_ERR(tdp158->vcc), "vcc");

	tdp158->vdd = devm_regulator_get(dev, "vdd");
	if (IS_ERR(tdp158->vdd))
		return dev_err_probe(dev, PTR_ERR(tdp158->vdd), "vdd");

	tdp158->enable = devm_gpiod_get_optional(dev, "enable", GPIOD_OUT_LOW);
	if (IS_ERR(tdp158->enable))
		return dev_err_probe(dev, PTR_ERR(tdp158->enable), "enable");

	tdp158->bridge.of_node = dev->of_node;
	tdp158->bridge.funcs = &tdp158_bridge_funcs;
	tdp158->bridge.driver_private = tdp158;
	tdp158->dev = dev;

	return devm_drm_bridge_add(dev, &tdp158->bridge);
}

static const struct of_device_id tdp158_match_table[] = {
	{ .compatible = "ti,tdp158" },
	{ }
};
MODULE_DEVICE_TABLE(of, tdp158_match_table);

static struct i2c_driver tdp158_driver = {
	.probe = tdp158_probe,
	.driver = {
		.name = "tdp158",
		.of_match_table = tdp158_match_table,
	},
};
module_i2c_driver(tdp158_driver);

MODULE_DESCRIPTION("TI TDP158 driver");
MODULE_LICENSE("GPL");