blob: bfa2f17b80420bc7dc159390ed9492e8ec2e6a54 [file] [log] [blame]
Inki Dae1c248b72011-10-04 19:19:01 +09001/* exynos_drm_encoder.c
2 *
3 * Copyright (c) 2011 Samsung Electronics Co., Ltd.
4 * Authors:
5 * Inki Dae <inki.dae@samsung.com>
6 * Joonyoung Shim <jy0922.shim@samsung.com>
7 * Seung-Woo Kim <sw0312.kim@samsung.com>
8 *
Inki Daed81aecb2012-12-18 02:30:17 +09009 * This program is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU General Public License as published by the
11 * Free Software Foundation; either version 2 of the License, or (at your
12 * option) any later version.
Inki Dae1c248b72011-10-04 19:19:01 +090013 */
14
David Howells760285e2012-10-02 18:01:07 +010015#include <drm/drmP.h>
16#include <drm/drm_crtc_helper.h>
Inki Dae1c248b72011-10-04 19:19:01 +090017
18#include "exynos_drm_drv.h"
Inki Dae1c248b72011-10-04 19:19:01 +090019#include "exynos_drm_encoder.h"
Inki Dae50caf252012-08-20 21:29:25 +090020#include "exynos_drm_connector.h"
Inki Dae1c248b72011-10-04 19:19:01 +090021
22#define to_exynos_encoder(x) container_of(x, struct exynos_drm_encoder,\
23 drm_encoder)
24
25/*
26 * exynos specific encoder structure.
27 *
28 * @drm_encoder: encoder object.
Sean Paul080be03d2014-02-19 21:02:55 +090029 * @display: the display structure that maps to this encoder
Inki Dae1c248b72011-10-04 19:19:01 +090030 */
31struct exynos_drm_encoder {
32 struct drm_encoder drm_encoder;
Sean Paul080be03d2014-02-19 21:02:55 +090033 struct exynos_drm_display *display;
Inki Dae1c248b72011-10-04 19:19:01 +090034};
35
Inki Daeec05da92011-12-06 11:06:54 +090036static void exynos_drm_encoder_dpms(struct drm_encoder *encoder, int mode)
37{
Sean Paul080be03d2014-02-19 21:02:55 +090038 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
39 struct exynos_drm_display *display = exynos_encoder->display;
Inki Daeec05da92011-12-06 11:06:54 +090040
YoungJun Chobca34c92013-06-12 10:40:52 +090041 DRM_DEBUG_KMS("encoder dpms: %d\n", mode);
Inki Daeec05da92011-12-06 11:06:54 +090042
Sean Paul080be03d2014-02-19 21:02:55 +090043 if (display->ops->dpms)
44 display->ops->dpms(display, mode);
Inki Daeec05da92011-12-06 11:06:54 +090045}
46
Inki Dae1c248b72011-10-04 19:19:01 +090047static bool
48exynos_drm_encoder_mode_fixup(struct drm_encoder *encoder,
Laurent Pincharte811f5a2012-07-17 17:56:50 +020049 const struct drm_display_mode *mode,
Inki Dae1c248b72011-10-04 19:19:01 +090050 struct drm_display_mode *adjusted_mode)
51{
Inki Dae1de425b2012-03-16 18:47:04 +090052 struct drm_device *dev = encoder->dev;
Sean Paul080be03d2014-02-19 21:02:55 +090053 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
54 struct exynos_drm_display *display = exynos_encoder->display;
Inki Dae1de425b2012-03-16 18:47:04 +090055 struct drm_connector *connector;
Inki Dae1de425b2012-03-16 18:47:04 +090056
Inki Dae1de425b2012-03-16 18:47:04 +090057 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
Sean Paul080be03d2014-02-19 21:02:55 +090058 if (connector->encoder != encoder)
59 continue;
60
61 if (display->ops->mode_fixup)
62 display->ops->mode_fixup(display, connector, mode,
63 adjusted_mode);
Inki Dae1de425b2012-03-16 18:47:04 +090064 }
Inki Dae1c248b72011-10-04 19:19:01 +090065
66 return true;
67}
68
69static void exynos_drm_encoder_mode_set(struct drm_encoder *encoder,
70 struct drm_display_mode *mode,
71 struct drm_display_mode *adjusted_mode)
72{
Sean Paula9c4cd22014-01-30 16:19:17 -050073 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
74 struct exynos_drm_display *display = exynos_encoder->display;
Inki Dae1c248b72011-10-04 19:19:01 +090075
Sean Paula9c4cd22014-01-30 16:19:17 -050076 if (display->ops->mode_set)
77 display->ops->mode_set(display, adjusted_mode);
Inki Dae1c248b72011-10-04 19:19:01 +090078}
79
80static void exynos_drm_encoder_prepare(struct drm_encoder *encoder)
81{
Inki Dae1c248b72011-10-04 19:19:01 +090082 /* drm framework doesn't check NULL. */
83}
84
85static void exynos_drm_encoder_commit(struct drm_encoder *encoder)
86{
Inki Dae44c91692012-10-18 18:59:55 +090087 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
Sean Paul080be03d2014-02-19 21:02:55 +090088 struct exynos_drm_display *display = exynos_encoder->display;
Inki Dae1c248b72011-10-04 19:19:01 +090089
Sean Paul080be03d2014-02-19 21:02:55 +090090 if (display->ops->dpms)
91 display->ops->dpms(display, DRM_MODE_DPMS_ON);
92
93 if (display->ops->commit)
94 display->ops->commit(display);
Inki Dae1c248b72011-10-04 19:19:01 +090095}
96
Inki Daebcf4cef2012-08-24 10:54:12 -070097static void exynos_drm_encoder_disable(struct drm_encoder *encoder)
98{
99 struct drm_plane *plane;
100 struct drm_device *dev = encoder->dev;
101
102 exynos_drm_encoder_dpms(encoder, DRM_MODE_DPMS_OFF);
103
104 /* all planes connected to this encoder should be also disabled. */
105 list_for_each_entry(plane, &dev->mode_config.plane_list, head) {
106 if (plane->crtc == encoder->crtc)
107 plane->funcs->disable_plane(plane);
108 }
109}
110
Inki Dae1c248b72011-10-04 19:19:01 +0900111static struct drm_encoder_helper_funcs exynos_encoder_helper_funcs = {
112 .dpms = exynos_drm_encoder_dpms,
113 .mode_fixup = exynos_drm_encoder_mode_fixup,
114 .mode_set = exynos_drm_encoder_mode_set,
115 .prepare = exynos_drm_encoder_prepare,
116 .commit = exynos_drm_encoder_commit,
Inki Daebcf4cef2012-08-24 10:54:12 -0700117 .disable = exynos_drm_encoder_disable,
Inki Dae1c248b72011-10-04 19:19:01 +0900118};
119
120static void exynos_drm_encoder_destroy(struct drm_encoder *encoder)
121{
Sean Paul080be03d2014-02-19 21:02:55 +0900122 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
Inki Dae1c248b72011-10-04 19:19:01 +0900123
124 drm_encoder_cleanup(encoder);
Inki Dae1c248b72011-10-04 19:19:01 +0900125 kfree(exynos_encoder);
126}
127
128static struct drm_encoder_funcs exynos_encoder_funcs = {
129 .destroy = exynos_drm_encoder_destroy,
130};
131
Inki Daed081f562012-02-15 11:25:19 +0900132static unsigned int exynos_drm_encoder_clones(struct drm_encoder *encoder)
133{
134 struct drm_encoder *clone;
135 struct drm_device *dev = encoder->dev;
136 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
Sean Paul080be03d2014-02-19 21:02:55 +0900137 struct exynos_drm_display *display = exynos_encoder->display;
Inki Daed081f562012-02-15 11:25:19 +0900138 unsigned int clone_mask = 0;
139 int cnt = 0;
140
141 list_for_each_entry(clone, &dev->mode_config.encoder_list, head) {
Sean Paul080be03d2014-02-19 21:02:55 +0900142 switch (display->type) {
Inki Daed081f562012-02-15 11:25:19 +0900143 case EXYNOS_DISPLAY_TYPE_LCD:
144 case EXYNOS_DISPLAY_TYPE_HDMI:
Inki Daeb73d1232012-03-21 10:55:26 +0900145 case EXYNOS_DISPLAY_TYPE_VIDI:
Inki Daed081f562012-02-15 11:25:19 +0900146 clone_mask |= (1 << (cnt++));
147 break;
148 default:
149 continue;
150 }
151 }
152
153 return clone_mask;
154}
155
156void exynos_drm_encoder_setup(struct drm_device *dev)
157{
158 struct drm_encoder *encoder;
159
Inki Daed081f562012-02-15 11:25:19 +0900160 list_for_each_entry(encoder, &dev->mode_config.encoder_list, head)
161 encoder->possible_clones = exynos_drm_encoder_clones(encoder);
162}
163
Inki Dae1c248b72011-10-04 19:19:01 +0900164struct drm_encoder *
165exynos_drm_encoder_create(struct drm_device *dev,
Sean Paul080be03d2014-02-19 21:02:55 +0900166 struct exynos_drm_display *display,
Sean Paul3f283d92014-01-30 16:19:11 -0500167 unsigned long possible_crtcs)
Inki Dae1c248b72011-10-04 19:19:01 +0900168{
169 struct drm_encoder *encoder;
170 struct exynos_drm_encoder *exynos_encoder;
171
Sean Paul080be03d2014-02-19 21:02:55 +0900172 if (!possible_crtcs)
Inki Dae1c248b72011-10-04 19:19:01 +0900173 return NULL;
174
175 exynos_encoder = kzalloc(sizeof(*exynos_encoder), GFP_KERNEL);
Sachin Kamat38bb5252013-08-19 19:04:55 +0900176 if (!exynos_encoder)
Inki Dae1c248b72011-10-04 19:19:01 +0900177 return NULL;
Inki Dae1c248b72011-10-04 19:19:01 +0900178
Sean Paul080be03d2014-02-19 21:02:55 +0900179 exynos_encoder->display = display;
Inki Dae1c248b72011-10-04 19:19:01 +0900180 encoder = &exynos_encoder->drm_encoder;
181 encoder->possible_crtcs = possible_crtcs;
182
183 DRM_DEBUG_KMS("possible_crtcs = 0x%x\n", encoder->possible_crtcs);
184
185 drm_encoder_init(dev, encoder, &exynos_encoder_funcs,
186 DRM_MODE_ENCODER_TMDS);
187
188 drm_encoder_helper_add(encoder, &exynos_encoder_helper_funcs);
189
190 DRM_DEBUG_KMS("encoder has been created\n");
191
192 return encoder;
193}
194
Sean Paul080be03d2014-02-19 21:02:55 +0900195struct exynos_drm_display *exynos_drm_get_display(struct drm_encoder *encoder)
Inki Dae1c248b72011-10-04 19:19:01 +0900196{
Sean Paul080be03d2014-02-19 21:02:55 +0900197 return to_exynos_encoder(encoder)->display;
Joonyoung Shim4070d212012-06-27 14:27:05 +0900198}