-
Notifications
You must be signed in to change notification settings - Fork 121
/
Copy pathvs_simple_enc.c
executable file
·310 lines (253 loc) · 7.37 KB
/
vs_simple_enc.c
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (C) 2020 VeriSilicon Holdings Co., Ltd.
*/
#include <linux/version.h>
#include <linux/component.h>
#include <linux/of_device.h>
#if KERNEL_VERSION(5, 5, 0) <= LINUX_VERSION_CODE
#include <linux/module.h>
#include <drm/drm_bridge.h>
#else
#include <drm/drmP.h>
#endif
#include <drm/drm_crtc_helper.h>
#include <drm/drm_of.h>
#include <linux/regmap.h>
#include <linux/mfd/syscon.h>
#include "vs_crtc.h"
#include "vs_simple_enc.h"
static const struct simple_encoder_priv hdmi_priv = {
.encoder_type = DRM_MODE_ENCODER_TMDS
};
static const struct simple_encoder_priv dsi_priv = {
.encoder_type = DRM_MODE_ENCODER_DSI
};
static const struct drm_encoder_funcs encoder_funcs = {
.destroy = drm_encoder_cleanup
};
static inline struct simple_encoder *to_simple_encoder(struct drm_encoder *enc)
{
return container_of(enc, struct simple_encoder, encoder);
}
#if 0
static int encoder_parse_dt(struct device *dev)
{
struct simple_encoder *simple = dev_get_drvdata(dev);
int ret = 0;
int cnt, i;
u32 *vals;
u32 *masks;
simple->dss_regmap = syscon_regmap_lookup_by_phandle(dev->of_node,
"verisilicon,dss-syscon");
if (IS_ERR(simple->dss_regmap)) {
if (PTR_ERR(simple->dss_regmap) != -ENODEV) {
dev_err(dev, "failed to get dss-syscon\n");
ret = PTR_ERR(simple->dss_regmap);
goto err;
}
simple->dss_regmap = NULL;
goto err;
}
cnt = of_property_count_elems_of_size(dev->of_node,
"verisilicon,mux-mask", 4);
if (!cnt) {
ret = cnt;
goto err;
}
simple->dss_regdatas = devm_kzalloc(dev,
sizeof(*simple->dss_regdatas) * cnt, GFP_KERNEL);
masks = kcalloc(cnt, sizeof(*masks), GFP_KERNEL);
if (!masks) {
ret = -ENOMEM;
goto err;
}
vals = kcalloc(cnt, sizeof(*vals), GFP_KERNEL);
if (!vals) {
ret = -ENOMEM;
goto err_free_masks;
}
ret = of_property_read_u32_array(
dev->of_node, "verisilicon,mux-mask", masks, cnt);
if (ret)
goto err_free_vals;
ret = of_property_read_u32_array(
dev->of_node, "verisilicon,mux-val", vals, cnt);
if (ret)
goto err_free_vals;
for (i = 0; i < cnt; i++) {
simple->dss_regdatas[i].mask = masks[i];
simple->dss_regdatas[i].value = vals[i];
}
err_free_vals:
kfree(vals);
err_free_masks:
kfree(masks);
err:
return ret;
}
#endif
#define DOM_VOUT_SYSCON_8 0x8U
#define U0_LCD_DATA_MAPPING_DPI_DP_SEL_SHIFT 0x2U
#define U0_LCD_DATA_MAPPING_DPI_DP_SEL_MASK 0x4U
#define DOM_VOUT_SYSCON_4 0x4U
#define U0_DISPLAY_PANEL_MUX_PANEL_SEL_SHIFT 0x14U
#define U0_DISPLAY_PANEL_MUX_PANEL_SEL_MASK 0x100000U
void encoder_atomic_enable(struct drm_encoder *encoder,
struct drm_atomic_state *state)
{
return;
}
int encoder_atomic_check(struct drm_encoder *encoder,
struct drm_crtc_state *crtc_state,
struct drm_connector_state *conn_state)
{
struct vs_crtc_state *vs_crtc_state = to_vs_crtc_state(crtc_state);
struct drm_connector *connector = conn_state->connector;
int ret = 0;
#if KERNEL_VERSION(5, 9, 0) <= LINUX_VERSION_CODE
struct drm_bridge *first_bridge = drm_bridge_chain_get_first_bridge(encoder);
struct drm_bridge_state *bridge_state = ERR_PTR(-EINVAL);
#endif
vs_crtc_state->encoder_type = encoder->encoder_type;
#if KERNEL_VERSION(5, 9, 0) <= LINUX_VERSION_CODE
if (first_bridge && first_bridge->funcs->atomic_duplicate_state)
bridge_state = drm_atomic_get_bridge_state(
crtc_state->state, first_bridge);
if (IS_ERR(bridge_state)) {
if (connector->display_info.num_bus_formats)
vs_crtc_state->output_fmt = connector->display_info.bus_formats[0];
else
vs_crtc_state->output_fmt = MEDIA_BUS_FMT_FIXED;
} else {
vs_crtc_state->output_fmt = bridge_state->input_bus_cfg.format;
}
#else
if (connector->display_info.num_bus_formats)
vs_crtc_state->output_fmt = connector->display_info.bus_formats[0];
else
vs_crtc_state->output_fmt = MEDIA_BUS_FMT_RGB888_1X24;
#endif
switch (vs_crtc_state->output_fmt) {
case MEDIA_BUS_FMT_FIXED:
case MEDIA_BUS_FMT_RGB565_1X16:
case MEDIA_BUS_FMT_RGB666_1X18:
case MEDIA_BUS_FMT_RGB888_1X24:
case MEDIA_BUS_FMT_RGB666_1X24_CPADHI:
case MEDIA_BUS_FMT_RGB101010_1X30:
case MEDIA_BUS_FMT_UYYVYY8_0_5X24:
case MEDIA_BUS_FMT_UYVY8_1X16:
case MEDIA_BUS_FMT_YUV8_1X24:
case MEDIA_BUS_FMT_UYYVYY10_0_5X30:
case MEDIA_BUS_FMT_UYVY10_1X20:
case MEDIA_BUS_FMT_YUV10_1X30:
ret = 0;
break;
default:
ret = -EINVAL;
break;
}
/* If MEDIA_BUS_FMT_FIXED, set it to default value */
if (vs_crtc_state->output_fmt == MEDIA_BUS_FMT_FIXED)
vs_crtc_state->output_fmt = MEDIA_BUS_FMT_RGB888_1X24;
return ret;
}
static const struct drm_encoder_helper_funcs encoder_helper_funcs = {
.atomic_enable = encoder_atomic_enable,
.atomic_check = encoder_atomic_check,
};
static int encoder_bind(struct device *dev, struct device *master, void *data)
{
struct drm_device *drm_dev = data;
struct simple_encoder *simple = dev_get_drvdata(dev);
struct drm_encoder *encoder;
struct drm_bridge *bridge;
int ret;
#ifdef CONFIG_STARFIVE_DSI
struct drm_panel *tmp_panel;
#endif
encoder = &simple->encoder;
/* Encoder. */
dev_info(dev,"encoder_bind begin\n");
ret = drm_encoder_init(drm_dev, encoder, &encoder_funcs,
simple->priv->encoder_type, NULL);
if (ret)
return ret;
drm_encoder_helper_add(encoder, &encoder_helper_funcs);
encoder->possible_crtcs =
drm_of_find_possible_crtcs(drm_dev, dev->of_node);
encoder->possible_crtcs = 2;
ret = drm_of_find_panel_or_bridge(dev->of_node, 1, 0,&tmp_panel, &bridge);
if (ret){
printk("no panel, %d\n",ret);
//dev_err_probe(dev, ret, "endpoint returns %d\n", ret);
goto err;
}
if (tmp_panel)
dev_err(dev, "found panel on endpoint\n");
#if KERNEL_VERSION(5, 7, 0) <= LINUX_VERSION_CODE
ret = drm_bridge_attach(encoder, bridge, NULL, 0);
#else
ret = drm_bridge_attach(encoder, bridge, NULL);
#endif
if (ret)
goto err;
dev_info(dev,"encoder_bind end\n");
return 0;
err:
drm_encoder_cleanup(encoder);
dev_info(dev,"encoder_bind error\n");
//return ret;
return 0;
}
static void encoder_unbind(struct device *dev, struct device *master,
void *data)
{
struct simple_encoder *simple = dev_get_drvdata(dev);
drm_encoder_cleanup(&simple->encoder);
}
static const struct component_ops encoder_component_ops = {
.bind = encoder_bind,
.unbind = encoder_unbind,
};
static const struct of_device_id simple_encoder_dt_match[] = {
{ .compatible = "verisilicon,rgb-encoder", .data = &hdmi_priv},
{ .compatible = "verisilicon,dp-encoder", .data = &hdmi_priv},
{ .compatible = "verisilicon,dsi-encoder", .data = &dsi_priv},
{},
};
MODULE_DEVICE_TABLE(of, simple_encoder_dt_match);
static int encoder_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct simple_encoder *simple;
simple = devm_kzalloc(dev, sizeof(*simple), GFP_KERNEL);
if (!simple)
return -ENOMEM;
simple->priv = of_device_get_match_data(dev);
simple->dev = dev;
dev_set_drvdata(dev, simple);
#if 0
ret = encoder_parse_dt(dev);
if (ret)
return ret;
#endif
return component_add(dev, &encoder_component_ops);
}
static int encoder_remove(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
component_del(dev, &encoder_component_ops);
dev_set_drvdata(dev, NULL);
return 0;
}
struct platform_driver simple_encoder_driver = {
.probe = encoder_probe,
.remove = encoder_remove,
.driver = {
.name = "vs-simple-encoder",
.of_match_table = of_match_ptr(simple_encoder_dt_match),
},
};
MODULE_DESCRIPTION("Simple Encoder Driver");
MODULE_LICENSE("GPL v2");