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
|
/* Copyright (c) 2019, The Linux Foundation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*/
#include "edrm_encoder.h"
#include "edrm_crtc.h"
#include "sde_kms.h"
static void edrm_encoder_enable(struct drm_encoder *drm_enc)
{
pr_err("eDRM Encoder enable\n");
}
static void edrm_encoder_disable(struct drm_encoder *drm_enc)
{
pr_err("eDRM Encoder disable\n");
}
void edrm_encoder_destroy(struct drm_encoder *encoder)
{
struct edrm_encoder *edrm_enc = to_edrm_encoder(encoder);
drm_encoder_cleanup(encoder);
kfree(edrm_enc);
}
static const struct drm_encoder_helper_funcs edrm_encoder_helper_funcs = {
.disable = edrm_encoder_disable,
.enable = edrm_encoder_enable,
};
static const struct drm_encoder_funcs edrm_encoder_funcs = {
.destroy = edrm_encoder_destroy,
};
int edrm_encoder_wait_for_commit_done(struct drm_encoder *drm_enc)
{
struct drm_device *dev;
struct msm_drm_private *priv;
struct msm_edrm_kms *edrm_kms;
struct msm_edrm_display *display;
struct edrm_crtc *edrm_crtc;
struct sde_kms *master_kms;
struct msm_drm_private *master_priv;
struct sde_mdss_cfg *cfg;
u32 ctl_off;
u32 flush_register = 0;
int i;
dev = drm_enc->dev;
priv = dev->dev_private;
edrm_kms = to_edrm_kms(priv->kms);
master_priv = edrm_kms->master_dev->dev_private;
master_kms = to_sde_kms(master_priv->kms);
cfg = master_kms->catalog;
edrm_crtc = to_edrm_crtc(drm_enc->crtc);
display = &edrm_kms->display[edrm_crtc->display_id];
ctl_off = display->ctl_off;
/* poll edrm_crtc->sspp_flush_mask until cleared */
for (i = 0; i < 20; i++) {
flush_register = readl_relaxed(master_kms->mmio +
ctl_off + 0x18);
if ((flush_register & edrm_crtc->sspp_flush_mask) != 0)
usleep_range(1000, 2000);
else
break;
}
/* reset sspp_flush_mask */
edrm_crtc->sspp_flush_mask = 0;
return 0;
}
struct drm_encoder *edrm_encoder_init(struct drm_device *dev,
struct msm_edrm_display *display)
{
struct edrm_encoder *edrm_encoder;
struct drm_encoder *encoder;
int ret;
edrm_encoder = kzalloc(sizeof(*edrm_encoder), GFP_KERNEL);
if (!edrm_encoder)
return ERR_PTR(-ENOMEM);
encoder = &edrm_encoder->base;
ret = drm_encoder_init(dev, encoder,
&edrm_encoder_funcs,
display->encoder_type);
if (ret)
goto fail;
drm_encoder_helper_add(encoder, &edrm_encoder_helper_funcs);
edrm_encoder->intf_idx = display->intf_id;
return encoder;
fail:
kfree(edrm_encoder);
return ERR_PTR(ret);
}
|