exynos_drm_encoder.c 12.0 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

#include "exynos_drm_drv.h"
#include "exynos_drm_encoder.h"
20
#include "exynos_drm_connector.h"
21 22 23 24 25 26 27 28 29 30 31 32

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

/*
 * exynos specific encoder structure.
 *
 * @drm_encoder: encoder object.
 * @manager: specific encoder has its own manager to control a hardware
 *	appropriately and we can access a hardware drawing on this manager.
 */
struct exynos_drm_encoder {
I
Inki Dae 已提交
33
	struct drm_crtc			*old_crtc;
34 35 36 37
	struct drm_encoder		drm_encoder;
	struct exynos_drm_manager	*manager;
};

38 39
static void exynos_drm_encoder_dpms(struct drm_encoder *encoder, int mode)
{
40 41
	struct exynos_drm_manager *manager = exynos_drm_get_manager(encoder);
	struct exynos_drm_display_ops *display_ops = manager->display_ops;
42

43
	DRM_DEBUG_KMS("encoder dpms: %d\n", mode);
44

45 46
	if (display_ops && display_ops->dpms)
		display_ops->dpms(manager->ctx, mode);
47 48 49 50
}

static bool
exynos_drm_encoder_mode_fixup(struct drm_encoder *encoder,
51
			       const struct drm_display_mode *mode,
52 53
			       struct drm_display_mode *adjusted_mode)
{
54 55 56 57 58 59 60 61
	struct drm_device *dev = encoder->dev;
	struct drm_connector *connector;
	struct exynos_drm_manager *manager = exynos_drm_get_manager(encoder);
	struct exynos_drm_manager_ops *manager_ops = manager->ops;

	list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
		if (connector->encoder == encoder)
			if (manager_ops && manager_ops->mode_fixup)
62
				manager_ops->mode_fixup(manager, connector,
63 64
							mode, adjusted_mode);
	}
65 66 67 68

	return true;
}

I
Inki Dae 已提交
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
static void disable_plane_to_crtc(struct drm_device *dev,
						struct drm_crtc *old_crtc,
						struct drm_crtc *new_crtc)
{
	struct drm_plane *plane;

	/*
	 * if old_crtc isn't same as encoder->crtc then it means that
	 * user changed crtc id to another one so the plane to old_crtc
	 * should be disabled and plane->crtc should be set to new_crtc
	 * (encoder->crtc)
	 */
	list_for_each_entry(plane, &dev->mode_config.plane_list, head) {
		if (plane->crtc == old_crtc) {
			/*
			 * do not change below call order.
			 *
			 * plane->funcs->disable_plane call checks
			 * if encoder->crtc is same as plane->crtc and if same
88
			 * then manager_ops->win_disable callback will be called
I
Inki Dae 已提交
89 90 91 92 93 94 95 96 97 98
			 * to diasble current hw overlay so plane->crtc should
			 * have new_crtc because new_crtc was set to
			 * encoder->crtc in advance.
			 */
			plane->crtc = new_crtc;
			plane->funcs->disable_plane(plane);
		}
	}
}

99 100 101 102 103 104
static void exynos_drm_encoder_mode_set(struct drm_encoder *encoder,
					 struct drm_display_mode *mode,
					 struct drm_display_mode *adjusted_mode)
{
	struct drm_device *dev = encoder->dev;
	struct drm_connector *connector;
I
Inki Dae 已提交
105 106
	struct exynos_drm_manager *manager;
	struct exynos_drm_manager_ops *manager_ops;
107 108

	list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
I
Inki Dae 已提交
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
		if (connector->encoder == encoder) {
			struct exynos_drm_encoder *exynos_encoder;

			exynos_encoder = to_exynos_encoder(encoder);

			if (exynos_encoder->old_crtc != encoder->crtc &&
					exynos_encoder->old_crtc) {

				/*
				 * disable a plane to old crtc and change
				 * crtc of the plane to new one.
				 */
				disable_plane_to_crtc(dev,
						exynos_encoder->old_crtc,
						encoder->crtc);
			}

			manager = exynos_drm_get_manager(encoder);
			manager_ops = manager->ops;

129
			if (manager_ops && manager_ops->mode_set)
130
				manager_ops->mode_set(manager, adjusted_mode);
I
Inki Dae 已提交
131 132 133

			exynos_encoder->old_crtc = encoder->crtc;
		}
134 135 136 137 138 139 140 141 142 143
	}
}

static void exynos_drm_encoder_prepare(struct drm_encoder *encoder)
{
	/* drm framework doesn't check NULL. */
}

static void exynos_drm_encoder_commit(struct drm_encoder *encoder)
{
I
Inki Dae 已提交
144 145
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
	struct exynos_drm_manager *manager = exynos_encoder->manager;
146 147 148
	struct exynos_drm_manager_ops *manager_ops = manager->ops;

	if (manager_ops && manager_ops->commit)
149
		manager_ops->commit(manager);
150 151
}

152 153 154
void exynos_drm_encoder_complete_scanout(struct drm_framebuffer *fb)
{
	struct exynos_drm_encoder *exynos_encoder;
155
	struct exynos_drm_manager_ops *ops;
156 157 158 159 160 161 162 163 164
	struct drm_device *dev = fb->dev;
	struct drm_encoder *encoder;

	/*
	 * make sure that overlay data are updated to real hardware
	 * for all encoders.
	 */
	list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) {
		exynos_encoder = to_exynos_encoder(encoder);
165
		ops = exynos_encoder->manager->ops;
166 167 168 169 170 171

		/*
		 * wait for vblank interrupt
		 * - this makes sure that overlay data are updated to
		 *	real hardware.
		 */
172
		if (ops->wait_for_vblank)
173
			ops->wait_for_vblank(exynos_encoder->manager);
174 175 176 177
	}
}


I
Inki Dae 已提交
178 179 180 181 182 183 184 185 186 187 188 189 190 191
static void exynos_drm_encoder_disable(struct drm_encoder *encoder)
{
	struct drm_plane *plane;
	struct drm_device *dev = encoder->dev;

	exynos_drm_encoder_dpms(encoder, DRM_MODE_DPMS_OFF);

	/* all planes connected to this encoder should be also disabled. */
	list_for_each_entry(plane, &dev->mode_config.plane_list, head) {
		if (plane->crtc == encoder->crtc)
			plane->funcs->disable_plane(plane);
	}
}

192 193 194 195 196 197
static struct drm_encoder_helper_funcs exynos_encoder_helper_funcs = {
	.dpms		= exynos_drm_encoder_dpms,
	.mode_fixup	= exynos_drm_encoder_mode_fixup,
	.mode_set	= exynos_drm_encoder_mode_set,
	.prepare	= exynos_drm_encoder_prepare,
	.commit		= exynos_drm_encoder_commit,
I
Inki Dae 已提交
198
	.disable	= exynos_drm_encoder_disable,
199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
};

static void exynos_drm_encoder_destroy(struct drm_encoder *encoder)
{
	struct exynos_drm_encoder *exynos_encoder =
		to_exynos_encoder(encoder);

	exynos_encoder->manager->pipe = -1;

	drm_encoder_cleanup(encoder);
	kfree(exynos_encoder);
}

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

216 217 218 219 220 221 222 223 224 225 226 227 228 229
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);
	struct exynos_drm_display_ops *display_ops =
				exynos_encoder->manager->display_ops;
	unsigned int clone_mask = 0;
	int cnt = 0;

	list_for_each_entry(clone, &dev->mode_config.encoder_list, head) {
		switch (display_ops->type) {
		case EXYNOS_DISPLAY_TYPE_LCD:
		case EXYNOS_DISPLAY_TYPE_HDMI:
230
		case EXYNOS_DISPLAY_TYPE_VIDI:
231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248
			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);
}

249 250 251 252 253 254 255
struct drm_encoder *
exynos_drm_encoder_create(struct drm_device *dev,
			   struct exynos_drm_manager *manager,
			   unsigned int possible_crtcs)
{
	struct drm_encoder *encoder;
	struct exynos_drm_encoder *exynos_encoder;
256
	int ret;
257 258 259 260 261 262 263 264

	if (!manager || !possible_crtcs)
		return NULL;

	if (!manager->dev)
		return NULL;

	exynos_encoder = kzalloc(sizeof(*exynos_encoder), GFP_KERNEL);
265
	if (!exynos_encoder)
266 267 268 269 270 271 272 273 274 275 276 277 278
		return NULL;

	exynos_encoder->manager = manager;
	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);

279
	if (manager->ops && manager->ops->initialize) {
280
		ret = manager->ops->initialize(manager, dev);
281 282 283 284 285 286 287 288 289 290 291 292 293 294
		if (ret) {
			DRM_ERROR("Manager initialize failed %d\n", ret);
			goto error;
		}
	}

	if (manager->display_ops && manager->display_ops->initialize) {
		ret = manager->display_ops->initialize(manager->dev, dev);
		if (ret) {
			DRM_ERROR("Display initialize failed %d\n", ret);
			goto error;
		}
	}

295 296 297
	DRM_DEBUG_KMS("encoder has been created\n");

	return encoder;
298 299 300 301

error:
	exynos_drm_encoder_destroy(&exynos_encoder->drm_encoder);
	return NULL;
302 303 304 305 306 307 308 309 310 311 312 313
}

struct exynos_drm_manager *exynos_drm_get_manager(struct drm_encoder *encoder)
{
	return to_exynos_encoder(encoder)->manager;
}

void exynos_drm_fn_encoder(struct drm_crtc *crtc, void *data,
			    void (*fn)(struct drm_encoder *, void *))
{
	struct drm_device *dev = crtc->dev;
	struct drm_encoder *encoder;
314 315
	struct exynos_drm_private *private = dev->dev_private;
	struct exynos_drm_manager *manager;
316 317

	list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) {
318 319 320 321 322 323 324 325 326 327 328 329 330
		/*
		 * if crtc is detached from encoder, check pipe,
		 * otherwise check crtc attached to encoder
		 */
		if (!encoder->crtc) {
			manager = to_exynos_encoder(encoder)->manager;
			if (manager->pipe < 0 ||
					private->crtc[manager->pipe] != crtc)
				continue;
		} else {
			if (encoder->crtc != crtc)
				continue;
		}
331 332 333 334 335 336 337 338 339 340 341 342

		fn(encoder, data);
	}
}

void exynos_drm_enable_vblank(struct drm_encoder *encoder, void *data)
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
	int crtc = *(int *)data;

343 344
	if (manager->pipe != crtc)
		return;
345 346

	if (manager_ops->enable_vblank)
347
		manager_ops->enable_vblank(manager);
348 349 350 351 352 353 354 355 356
}

void exynos_drm_disable_vblank(struct drm_encoder *encoder, void *data)
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
	int crtc = *(int *)data;

357 358
	if (manager->pipe != crtc)
		return;
359 360

	if (manager_ops->disable_vblank)
361
		manager_ops->disable_vblank(manager);
362 363
}

364 365 366 367 368 369 370 371
void exynos_drm_encoder_crtc_dpms(struct drm_encoder *encoder, void *data)
{
	struct exynos_drm_encoder *exynos_encoder = to_exynos_encoder(encoder);
	struct exynos_drm_manager *manager = exynos_encoder->manager;
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
	int mode = *(int *)data;

	if (manager_ops && manager_ops->dpms)
372
		manager_ops->dpms(manager, mode);
373 374 375 376 377 378 379 380 381 382 383 384

	/*
	 * if this condition is ok then it means that the crtc is already
	 * detached from encoder and last function for detaching is properly
	 * done, so clear pipe from manager to prevent repeated call.
	 */
	if (mode > DRM_MODE_DPMS_ON) {
		if (!encoder->crtc)
			manager->pipe = -1;
	}
}

385 386 387 388 389 390 391 392 393 394 395 396 397 398
void exynos_drm_encoder_crtc_pipe(struct drm_encoder *encoder, void *data)
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
	int pipe = *(int *)data;

	/*
	 * when crtc is detached from encoder, this pipe is used
	 * to select manager operation
	 */
	manager->pipe = pipe;
}

void exynos_drm_encoder_plane_mode_set(struct drm_encoder *encoder, void *data)
399 400 401
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
402
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
403 404
	struct exynos_drm_overlay *overlay = data;

405
	if (manager_ops && manager_ops->win_mode_set)
406
		manager_ops->win_mode_set(manager, overlay);
407 408
}

409
void exynos_drm_encoder_plane_commit(struct drm_encoder *encoder, void *data)
410 411 412
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
413
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
414
	int zpos = DEFAULT_ZPOS;
415

416 417 418
	if (data)
		zpos = *(int *)data;

419
	if (manager_ops && manager_ops->win_commit)
420
		manager_ops->win_commit(manager, zpos);
421
}
422

423 424 425 426
void exynos_drm_encoder_plane_enable(struct drm_encoder *encoder, void *data)
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
427
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
428 429 430 431 432
	int zpos = DEFAULT_ZPOS;

	if (data)
		zpos = *(int *)data;

433
	if (manager_ops && manager_ops->win_enable)
434
		manager_ops->win_enable(manager, zpos);
435 436
}

437
void exynos_drm_encoder_plane_disable(struct drm_encoder *encoder, void *data)
438 439 440
{
	struct exynos_drm_manager *manager =
		to_exynos_encoder(encoder)->manager;
441
	struct exynos_drm_manager_ops *manager_ops = manager->ops;
442
	int zpos = DEFAULT_ZPOS;
443

444 445 446
	if (data)
		zpos = *(int *)data;

447
	if (manager_ops && manager_ops->win_disable)
448
		manager_ops->win_disable(manager, zpos);
449
}