blob: 0648ba4a520ce5d78c06d467de511441746d857c [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"
20
21#define to_exynos_encoder(x) container_of(x, struct exynos_drm_encoder,\
22 drm_encoder)
23
24/*
25 * exynos specific encoder structure.
26 *
27 * @drm_encoder: encoder object.
Sean Paul080be03d2014-02-19 21:02:55 +090028 * @display: the display structure that maps to this encoder
Inki Dae1c248b72011-10-04 19:19:01 +090029 */
30struct exynos_drm_encoder {
31 struct drm_encoder drm_encoder;
Sean Paul080be03d2014-02-19 21:02:55 +090032 struct exynos_drm_display *display;
Inki Dae1c248b72011-10-04 19:19:01 +090033};
34
Inki Dae1c248b72011-10-04 19:19:01 +090035static bool
36exynos_drm_encoder_mode_fixup(struct drm_encoder *encoder,
Laurent Pincharte811f5a2012-07-17 17:56:50 +020037 const struct drm_display_mode *mode,
Inki Dae1c248b72011-10-04 19:19:01 +090038 struct drm_display_mode *adjusted_mode)
39{
Inki Dae1de425b2012-03-16 18:47:04 +090040 struct drm_device *dev = encoder->dev;
Sean Paul080be03d2014-02-19 21:02:55 +090041 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
42 struct exynos_drm_display *display = exynos_encoder->display;
Inki Dae1de425b2012-03-16 18:47:04 +090043 struct drm_connector *connector;
Inki Dae1de425b2012-03-16 18:47:04 +090044
Inki Dae1de425b2012-03-16 18:47:04 +090045 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
Sean Paul080be03d2014-02-19 21:02:55 +090046 if (connector->encoder != encoder)
47 continue;
48
49 if (display->ops->mode_fixup)
50 display->ops->mode_fixup(display, connector, mode,
51 adjusted_mode);
Inki Dae1de425b2012-03-16 18:47:04 +090052 }
Inki Dae1c248b72011-10-04 19:19:01 +090053
54 return true;
55}
56
57static void exynos_drm_encoder_mode_set(struct drm_encoder *encoder,
58 struct drm_display_mode *mode,
59 struct drm_display_mode *adjusted_mode)
60{
Sean Paula9c4cd22014-01-30 16:19:17 -050061 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
62 struct exynos_drm_display *display = exynos_encoder->display;
Inki Dae1c248b72011-10-04 19:19:01 +090063
Sean Paula9c4cd22014-01-30 16:19:17 -050064 if (display->ops->mode_set)
65 display->ops->mode_set(display, adjusted_mode);
Inki Dae1c248b72011-10-04 19:19:01 +090066}
67
Gustavo Padovan63498e32015-06-01 12:04:53 -030068static void exynos_drm_encoder_enable(struct drm_encoder *encoder)
Inki Dae1c248b72011-10-04 19:19:01 +090069{
Inki Dae44c91692012-10-18 18:59:55 +090070 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
Sean Paul080be03d2014-02-19 21:02:55 +090071 struct exynos_drm_display *display = exynos_encoder->display;
Inki Dae1c248b72011-10-04 19:19:01 +090072
Sean Paul080be03d2014-02-19 21:02:55 +090073 if (display->ops->dpms)
74 display->ops->dpms(display, DRM_MODE_DPMS_ON);
75
76 if (display->ops->commit)
77 display->ops->commit(display);
Inki Dae1c248b72011-10-04 19:19:01 +090078}
79
Inki Daebcf4cef2012-08-24 10:54:12 -070080static void exynos_drm_encoder_disable(struct drm_encoder *encoder)
81{
Gustavo Padovan63498e32015-06-01 12:04:53 -030082 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
83 struct exynos_drm_display *display = exynos_encoder->display;
Inki Daebcf4cef2012-08-24 10:54:12 -070084 struct drm_plane *plane;
85 struct drm_device *dev = encoder->dev;
86
Gustavo Padovan63498e32015-06-01 12:04:53 -030087 if (display->ops->dpms)
88 display->ops->dpms(display, DRM_MODE_DPMS_OFF);
Inki Daebcf4cef2012-08-24 10:54:12 -070089
90 /* all planes connected to this encoder should be also disabled. */
Matt Roper08863272014-04-01 15:22:31 -070091 drm_for_each_legacy_plane(plane, &dev->mode_config.plane_list) {
Joonyoung Shim936ce5c2015-02-05 16:11:38 +090092 if (plane->crtc && (plane->crtc == encoder->crtc))
Inki Daebcf4cef2012-08-24 10:54:12 -070093 plane->funcs->disable_plane(plane);
94 }
95}
96
Inki Dae1c248b72011-10-04 19:19:01 +090097static struct drm_encoder_helper_funcs exynos_encoder_helper_funcs = {
Inki Dae1c248b72011-10-04 19:19:01 +090098 .mode_fixup = exynos_drm_encoder_mode_fixup,
99 .mode_set = exynos_drm_encoder_mode_set,
Gustavo Padovan63498e32015-06-01 12:04:53 -0300100 .enable = exynos_drm_encoder_enable,
Inki Daebcf4cef2012-08-24 10:54:12 -0700101 .disable = exynos_drm_encoder_disable,
Inki Dae1c248b72011-10-04 19:19:01 +0900102};
103
104static void exynos_drm_encoder_destroy(struct drm_encoder *encoder)
105{
Sean Paul080be03d2014-02-19 21:02:55 +0900106 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
Inki Dae1c248b72011-10-04 19:19:01 +0900107
108 drm_encoder_cleanup(encoder);
Inki Dae1c248b72011-10-04 19:19:01 +0900109 kfree(exynos_encoder);
110}
111
112static struct drm_encoder_funcs exynos_encoder_funcs = {
113 .destroy = exynos_drm_encoder_destroy,
114};
115
Inki Daed081f562012-02-15 11:25:19 +0900116static unsigned int exynos_drm_encoder_clones(struct drm_encoder *encoder)
117{
118 struct drm_encoder *clone;
119 struct drm_device *dev = encoder->dev;
120 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
Sean Paul080be03d2014-02-19 21:02:55 +0900121 struct exynos_drm_display *display = exynos_encoder->display;
Inki Daed081f562012-02-15 11:25:19 +0900122 unsigned int clone_mask = 0;
123 int cnt = 0;
124
125 list_for_each_entry(clone, &dev->mode_config.encoder_list, head) {
Sean Paul080be03d2014-02-19 21:02:55 +0900126 switch (display->type) {
Inki Daed081f562012-02-15 11:25:19 +0900127 case EXYNOS_DISPLAY_TYPE_LCD:
128 case EXYNOS_DISPLAY_TYPE_HDMI:
Inki Daeb73d1232012-03-21 10:55:26 +0900129 case EXYNOS_DISPLAY_TYPE_VIDI:
Inki Daed081f562012-02-15 11:25:19 +0900130 clone_mask |= (1 << (cnt++));
131 break;
132 default:
133 continue;
134 }
135 }
136
137 return clone_mask;
138}
139
140void exynos_drm_encoder_setup(struct drm_device *dev)
141{
142 struct drm_encoder *encoder;
143
Inki Daed081f562012-02-15 11:25:19 +0900144 list_for_each_entry(encoder, &dev->mode_config.encoder_list, head)
145 encoder->possible_clones = exynos_drm_encoder_clones(encoder);
146}
147
Inki Dae1c248b72011-10-04 19:19:01 +0900148struct drm_encoder *
149exynos_drm_encoder_create(struct drm_device *dev,
Sean Paul080be03d2014-02-19 21:02:55 +0900150 struct exynos_drm_display *display,
Sean Paul3f283d92014-01-30 16:19:11 -0500151 unsigned long possible_crtcs)
Inki Dae1c248b72011-10-04 19:19:01 +0900152{
153 struct drm_encoder *encoder;
154 struct exynos_drm_encoder *exynos_encoder;
155
Sean Paul080be03d2014-02-19 21:02:55 +0900156 if (!possible_crtcs)
Inki Dae1c248b72011-10-04 19:19:01 +0900157 return NULL;
158
159 exynos_encoder = kzalloc(sizeof(*exynos_encoder), GFP_KERNEL);
Sachin Kamat38bb5252013-08-19 19:04:55 +0900160 if (!exynos_encoder)
Inki Dae1c248b72011-10-04 19:19:01 +0900161 return NULL;
Inki Dae1c248b72011-10-04 19:19:01 +0900162
Sean Paul080be03d2014-02-19 21:02:55 +0900163 exynos_encoder->display = display;
Inki Dae1c248b72011-10-04 19:19:01 +0900164 encoder = &exynos_encoder->drm_encoder;
165 encoder->possible_crtcs = possible_crtcs;
166
167 DRM_DEBUG_KMS("possible_crtcs = 0x%x\n", encoder->possible_crtcs);
168
169 drm_encoder_init(dev, encoder, &exynos_encoder_funcs,
170 DRM_MODE_ENCODER_TMDS);
171
172 drm_encoder_helper_add(encoder, &exynos_encoder_helper_funcs);
173
174 DRM_DEBUG_KMS("encoder has been created\n");
175
176 return encoder;
177}
178
Sean Paul080be03d2014-02-19 21:02:55 +0900179struct exynos_drm_display *exynos_drm_get_display(struct drm_encoder *encoder)
Inki Dae1c248b72011-10-04 19:19:01 +0900180{
Sean Paul080be03d2014-02-19 21:02:55 +0900181 return to_exynos_encoder(encoder)->display;
Joonyoung Shim4070d212012-06-27 14:27:05 +0900182}