1 /* exynos_drm_encoder.c
3 * Copyright (c) 2011 Samsung Electronics Co., Ltd.
9 * 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.
16 #include <drm/drm_crtc_helper.h>
18 #include "exynos_drm_drv.h"
19 #include "exynos_drm_encoder.h"
21 #define to_exynos_encoder(x) container_of(x, struct exynos_drm_encoder,\
25 * exynos specific encoder structure.
27 * @drm_encoder: encoder object.
28 * @display: the display structure that maps to this encoder
30 struct exynos_drm_encoder {
31 struct drm_encoder drm_encoder;
32 struct exynos_drm_display *display;
35 static void exynos_drm_encoder_dpms(struct drm_encoder *encoder, int mode)
37 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
38 struct exynos_drm_display *display = exynos_encoder->display;
40 DRM_DEBUG_KMS("encoder dpms: %d\n", mode);
42 if (display->ops->dpms)
43 display->ops->dpms(display, mode);
47 exynos_drm_encoder_mode_fixup(struct drm_encoder *encoder,
48 const struct drm_display_mode *mode,
49 struct drm_display_mode *adjusted_mode)
51 struct drm_device *dev = encoder->dev;
52 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
53 struct exynos_drm_display *display = exynos_encoder->display;
54 struct drm_connector *connector;
56 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
57 if (connector->encoder != encoder)
60 if (display->ops->mode_fixup)
61 display->ops->mode_fixup(display, connector, mode,
68 static void exynos_drm_encoder_mode_set(struct drm_encoder *encoder,
69 struct drm_display_mode *mode,
70 struct drm_display_mode *adjusted_mode)
72 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
73 struct exynos_drm_display *display = exynos_encoder->display;
75 if (display->ops->mode_set)
76 display->ops->mode_set(display, adjusted_mode);
79 static void exynos_drm_encoder_prepare(struct drm_encoder *encoder)
81 /* drm framework doesn't check NULL. */
84 static void exynos_drm_encoder_commit(struct drm_encoder *encoder)
86 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
87 struct exynos_drm_display *display = exynos_encoder->display;
89 if (display->ops->dpms)
90 display->ops->dpms(display, DRM_MODE_DPMS_ON);
92 if (display->ops->commit)
93 display->ops->commit(display);
96 static void exynos_drm_encoder_disable(struct drm_encoder *encoder)
98 struct drm_plane *plane;
99 struct drm_device *dev = encoder->dev;
101 exynos_drm_encoder_dpms(encoder, DRM_MODE_DPMS_OFF);
103 /* all planes connected to this encoder should be also disabled. */
104 drm_for_each_legacy_plane(plane, &dev->mode_config.plane_list) {
105 if (plane->crtc && (plane->crtc == encoder->crtc))
106 plane->funcs->disable_plane(plane);
110 static struct drm_encoder_helper_funcs exynos_encoder_helper_funcs = {
111 .dpms = exynos_drm_encoder_dpms,
112 .mode_fixup = exynos_drm_encoder_mode_fixup,
113 .mode_set = exynos_drm_encoder_mode_set,
114 .prepare = exynos_drm_encoder_prepare,
115 .commit = exynos_drm_encoder_commit,
116 .disable = exynos_drm_encoder_disable,
119 static void exynos_drm_encoder_destroy(struct drm_encoder *encoder)
121 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
123 drm_encoder_cleanup(encoder);
124 kfree(exynos_encoder);
127 static struct drm_encoder_funcs exynos_encoder_funcs = {
128 .destroy = exynos_drm_encoder_destroy,
131 static unsigned int exynos_drm_encoder_clones(struct drm_encoder *encoder)
133 struct drm_encoder *clone;
134 struct drm_device *dev = encoder->dev;
135 struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
136 struct exynos_drm_display *display = exynos_encoder->display;
137 unsigned int clone_mask = 0;
140 list_for_each_entry(clone, &dev->mode_config.encoder_list, head) {
141 switch (display->type) {
142 case EXYNOS_DISPLAY_TYPE_LCD:
143 case EXYNOS_DISPLAY_TYPE_HDMI:
144 case EXYNOS_DISPLAY_TYPE_VIDI:
145 clone_mask |= (1 << (cnt++));
155 void exynos_drm_encoder_setup(struct drm_device *dev)
157 struct drm_encoder *encoder;
159 list_for_each_entry(encoder, &dev->mode_config.encoder_list, head)
160 encoder->possible_clones = exynos_drm_encoder_clones(encoder);
164 exynos_drm_encoder_create(struct drm_device *dev,
165 struct exynos_drm_display *display,
166 unsigned long possible_crtcs)
168 struct drm_encoder *encoder;
169 struct exynos_drm_encoder *exynos_encoder;
174 exynos_encoder = kzalloc(sizeof(*exynos_encoder), GFP_KERNEL);
178 exynos_encoder->display = display;
179 encoder = &exynos_encoder->drm_encoder;
180 encoder->possible_crtcs = possible_crtcs;
182 DRM_DEBUG_KMS("possible_crtcs = 0x%x\n", encoder->possible_crtcs);
184 drm_encoder_init(dev, encoder, &exynos_encoder_funcs,
185 DRM_MODE_ENCODER_TMDS);
187 drm_encoder_helper_add(encoder, &exynos_encoder_helper_funcs);
189 DRM_DEBUG_KMS("encoder has been created\n");
194 struct exynos_drm_display *exynos_drm_get_display(struct drm_encoder *encoder)
196 return to_exynos_encoder(encoder)->display;