summaryrefslogtreecommitdiff
path: root/drivers/gpu/drm/msm/ekms/edrm_encoder.c
blob: 0cee78c73f50504738c82e3ddead53493ede0193 (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
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);
}