exynos_drm_encoder.c 4.7 KB
Newer Older
1 2 3 4 5 6 7 8
/* exynos_drm_encoder.c
 *
 * Copyright (c) 2011 Samsung Electronics Co., Ltd.
 * Authors:
 *	Inki Dae <inki.dae@samsung.com>
 *	Joonyoung Shim <jy0922.shim@samsung.com>
 *	Seung-Woo Kim <sw0312.kim@samsung.com>
 *
9 10 11 12
 * This program is free software; you can redistribute  it and/or modify it
 * under  the terms of  the GNU General  Public License as published by the
 * Free Software Foundation;  either version 2 of the  License, or (at your
 * option) any later version.
13 14
 */

15 16
#include <drm/drmP.h>
#include <drm/drm_crtc_helper.h>
17 18 19 20 21 22 23 24 25 26 27

#include "exynos_drm_drv.h"
#include "exynos_drm_encoder.h"

#define to_exynos_encoder(x)	container_of(x, struct exynos_drm_encoder,\
				drm_encoder)

/*
 * exynos specific encoder structure.
 *
 * @drm_encoder: encoder object.
28
 * @display: the display structure that maps to this encoder
29 30 31
 */
struct exynos_drm_encoder {
	struct drm_encoder		drm_encoder;
32
	struct exynos_drm_display	*display;
33 34 35 36
};

static bool
exynos_drm_encoder_mode_fixup(struct drm_encoder *encoder,
37
			       const struct drm_display_mode *mode,
38 39
			       struct drm_display_mode *adjusted_mode)
{
40
	struct drm_device *dev = encoder->dev;
41 42
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
	struct exynos_drm_display *display = exynos_encoder->display;
43 44 45
	struct drm_connector *connector;

	list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
46 47 48 49 50 51
		if (connector->encoder != encoder)
			continue;

		if (display->ops->mode_fixup)
			display->ops->mode_fixup(display, connector, mode,
					adjusted_mode);
52
	}
53 54 55 56 57 58 59 60

	return true;
}

static void exynos_drm_encoder_mode_set(struct drm_encoder *encoder,
					 struct drm_display_mode *mode,
					 struct drm_display_mode *adjusted_mode)
{
61 62
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
	struct exynos_drm_display *display = exynos_encoder->display;
I
Inki Dae 已提交
63

64 65
	if (display->ops->mode_set)
		display->ops->mode_set(display, adjusted_mode);
66 67
}

68
static void exynos_drm_encoder_enable(struct drm_encoder *encoder)
69
{
I
Inki Dae 已提交
70
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
71
	struct exynos_drm_display *display = exynos_encoder->display;
72

73 74
	if (display->ops->enable)
		display->ops->enable(display);
75

76 77
	if (display->ops->commit)
		display->ops->commit(display);
78 79
}

I
Inki Dae 已提交
80 81
static void exynos_drm_encoder_disable(struct drm_encoder *encoder)
{
82 83
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
	struct exynos_drm_display *display = exynos_encoder->display;
I
Inki Dae 已提交
84

85 86
	if (display->ops->disable)
		display->ops->disable(display);
I
Inki Dae 已提交
87 88
}

89 90 91
static struct drm_encoder_helper_funcs exynos_encoder_helper_funcs = {
	.mode_fixup	= exynos_drm_encoder_mode_fixup,
	.mode_set	= exynos_drm_encoder_mode_set,
92
	.enable		= exynos_drm_encoder_enable,
I
Inki Dae 已提交
93
	.disable	= exynos_drm_encoder_disable,
94 95 96 97
};

static void exynos_drm_encoder_destroy(struct drm_encoder *encoder)
{
98
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
99 100 101 102 103 104 105 106 107

	drm_encoder_cleanup(encoder);
	kfree(exynos_encoder);
}

static struct drm_encoder_funcs exynos_encoder_funcs = {
	.destroy = exynos_drm_encoder_destroy,
};

108 109 110 111 112
static unsigned int exynos_drm_encoder_clones(struct drm_encoder *encoder)
{
	struct drm_encoder *clone;
	struct drm_device *dev = encoder->dev;
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
113
	struct exynos_drm_display *display = exynos_encoder->display;
114 115 116 117
	unsigned int clone_mask = 0;
	int cnt = 0;

	list_for_each_entry(clone, &dev->mode_config.encoder_list, head) {
118
		switch (display->type) {
119 120
		case EXYNOS_DISPLAY_TYPE_LCD:
		case EXYNOS_DISPLAY_TYPE_HDMI:
121
		case EXYNOS_DISPLAY_TYPE_VIDI:
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
			clone_mask |= (1 << (cnt++));
			break;
		default:
			continue;
		}
	}

	return clone_mask;
}

void exynos_drm_encoder_setup(struct drm_device *dev)
{
	struct drm_encoder *encoder;

	list_for_each_entry(encoder, &dev->mode_config.encoder_list, head)
		encoder->possible_clones = exynos_drm_encoder_clones(encoder);
}

140 141
struct drm_encoder *
exynos_drm_encoder_create(struct drm_device *dev,
142
			   struct exynos_drm_display *display,
143
			   unsigned long possible_crtcs)
144 145 146 147
{
	struct drm_encoder *encoder;
	struct exynos_drm_encoder *exynos_encoder;

148
	if (!possible_crtcs)
149 150 151
		return NULL;

	exynos_encoder = kzalloc(sizeof(*exynos_encoder), GFP_KERNEL);
152
	if (!exynos_encoder)
153 154
		return NULL;

155
	exynos_encoder->display = display;
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
	encoder = &exynos_encoder->drm_encoder;
	encoder->possible_crtcs = possible_crtcs;

	DRM_DEBUG_KMS("possible_crtcs = 0x%x\n", encoder->possible_crtcs);

	drm_encoder_init(dev, encoder, &exynos_encoder_funcs,
			DRM_MODE_ENCODER_TMDS);

	drm_encoder_helper_add(encoder, &exynos_encoder_helper_funcs);

	DRM_DEBUG_KMS("encoder has been created\n");

	return encoder;
}

171
struct exynos_drm_display *exynos_drm_get_display(struct drm_encoder *encoder)
172
{
173
	return to_exynos_encoder(encoder)->display;
174
}