2 * Copyright (c) 2014-2015 The Linux Foundation. All rights reserved.
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
17 struct mdp5_cfg_handler {
19 struct mdp5_cfg config;
22 /* mdp5_cfg must be exposed (used in mdp5.xml.h) */
23 const struct mdp5_cfg_hw *mdp5_cfg = NULL;
25 const struct mdp5_cfg_hw msm8x74v1_config = {
36 [SSPP_VIG0] = 1, [SSPP_VIG1] = 4, [SSPP_VIG2] = 7,
37 [SSPP_DMA0] = 10, [SSPP_DMA1] = 13,
38 [SSPP_RGB0] = 16, [SSPP_RGB1] = 17, [SSPP_RGB2] = 18,
43 .base = { 0x00500, 0x00600, 0x00700, 0x00800, 0x00900 },
44 .flush_hw_mask = 0x0003ffff,
48 .base = { 0x01100, 0x01500, 0x01900 },
49 .caps = MDP_PIPE_CAP_HFLIP |
57 .base = { 0x01d00, 0x02100, 0x02500 },
58 .caps = MDP_PIPE_CAP_HFLIP |
65 .base = { 0x02900, 0x02d00 },
66 .caps = MDP_PIPE_CAP_HFLIP |
72 .base = { 0x03100, 0x03500, 0x03900, 0x03d00, 0x04100 },
74 { .id = 0, .pp = 0, .dspp = 0,
75 .caps = MDP_LM_CAP_DISPLAY, },
76 { .id = 1, .pp = 1, .dspp = 1,
77 .caps = MDP_LM_CAP_DISPLAY, },
78 { .id = 2, .pp = 2, .dspp = 2,
79 .caps = MDP_LM_CAP_DISPLAY, },
80 { .id = 3, .pp = -1, .dspp = -1,
81 .caps = MDP_LM_CAP_WB },
82 { .id = 4, .pp = -1, .dspp = -1,
83 .caps = MDP_LM_CAP_WB },
89 .base = { 0x04500, 0x04900, 0x04d00 },
93 .base = { 0x21a00, 0x21b00, 0x21c00 },
96 .base = { 0x21000, 0x21200, 0x21400, 0x21600 },
104 .max_clk = 200000000,
107 const struct mdp5_cfg_hw msm8x74v2_config = {
111 .caps = MDP_CAP_SMP |
118 [SSPP_VIG0] = 1, [SSPP_VIG1] = 4, [SSPP_VIG2] = 7,
119 [SSPP_DMA0] = 10, [SSPP_DMA1] = 13,
120 [SSPP_RGB0] = 16, [SSPP_RGB1] = 17, [SSPP_RGB2] = 18,
125 .base = { 0x00500, 0x00600, 0x00700, 0x00800, 0x00900 },
126 .flush_hw_mask = 0x0003ffff,
130 .base = { 0x01100, 0x01500, 0x01900 },
131 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
132 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_CSC |
133 MDP_PIPE_CAP_DECIMATION,
137 .base = { 0x01d00, 0x02100, 0x02500 },
138 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
139 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_DECIMATION,
143 .base = { 0x02900, 0x02d00 },
144 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP,
148 .base = { 0x03100, 0x03500, 0x03900, 0x03d00, 0x04100 },
150 { .id = 0, .pp = 0, .dspp = 0,
151 .caps = MDP_LM_CAP_DISPLAY, },
152 { .id = 1, .pp = 1, .dspp = 1,
153 .caps = MDP_LM_CAP_DISPLAY, },
154 { .id = 2, .pp = 2, .dspp = 2,
155 .caps = MDP_LM_CAP_DISPLAY, },
156 { .id = 3, .pp = -1, .dspp = -1,
157 .caps = MDP_LM_CAP_WB, },
158 { .id = 4, .pp = -1, .dspp = -1,
159 .caps = MDP_LM_CAP_WB, },
163 .max_height = 0xFFFF,
167 .base = { 0x04500, 0x04900, 0x04d00 },
171 .base = { 0x13000, 0x13200 },
175 .base = { 0x12c00, 0x12d00, 0x12e00 },
178 .base = { 0x12400, 0x12600, 0x12800, 0x12a00 },
186 .max_clk = 200000000,
189 const struct mdp5_cfg_hw apq8084_config = {
193 .caps = MDP_CAP_SMP |
201 [SSPP_VIG0] = 1, [SSPP_VIG1] = 4,
202 [SSPP_VIG2] = 7, [SSPP_VIG3] = 19,
203 [SSPP_DMA0] = 10, [SSPP_DMA1] = 13,
204 [SSPP_RGB0] = 16, [SSPP_RGB1] = 17,
205 [SSPP_RGB2] = 18, [SSPP_RGB3] = 22,
207 .reserved_state[0] = GENMASK(7, 0), /* first 8 MMBs */
209 /* Two SMP blocks are statically tied to RGB pipes: */
210 [16] = 2, [17] = 2, [18] = 2, [22] = 2,
215 .base = { 0x00500, 0x00600, 0x00700, 0x00800, 0x00900 },
216 .flush_hw_mask = 0x003fffff,
220 .base = { 0x01100, 0x01500, 0x01900, 0x01d00 },
221 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
222 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_CSC |
223 MDP_PIPE_CAP_DECIMATION,
227 .base = { 0x02100, 0x02500, 0x02900, 0x02d00 },
228 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
229 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_DECIMATION,
233 .base = { 0x03100, 0x03500 },
234 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP,
238 .base = { 0x03900, 0x03d00, 0x04100, 0x04500, 0x04900, 0x04d00 },
240 { .id = 0, .pp = 0, .dspp = 0,
241 .caps = MDP_LM_CAP_DISPLAY |
243 { .id = 1, .pp = 1, .dspp = 1,
244 .caps = MDP_LM_CAP_DISPLAY, },
245 { .id = 2, .pp = 2, .dspp = 2,
246 .caps = MDP_LM_CAP_DISPLAY |
248 { .id = 3, .pp = -1, .dspp = -1,
249 .caps = MDP_LM_CAP_WB, },
250 { .id = 4, .pp = -1, .dspp = -1,
251 .caps = MDP_LM_CAP_WB, },
252 { .id = 5, .pp = 3, .dspp = 3,
253 .caps = MDP_LM_CAP_DISPLAY, },
257 .max_height = 0xFFFF,
261 .base = { 0x05100, 0x05500, 0x05900, 0x05d00 },
266 .base = { 0x13400, 0x13600, 0x13800 },
270 .base = { 0x12e00, 0x12f00, 0x13000, 0x13100 },
273 .base = { 0x12400, 0x12600, 0x12800, 0x12a00, 0x12c00 },
281 .max_clk = 320000000,
284 const struct mdp5_cfg_hw msm8x16_config = {
289 .caps = MDP_CAP_SMP |
296 [SSPP_VIG0] = 1, [SSPP_DMA0] = 4,
297 [SSPP_RGB0] = 7, [SSPP_RGB1] = 8,
302 .base = { 0x01000, 0x01200, 0x01400, 0x01600, 0x01800 },
303 .flush_hw_mask = 0x4003ffff,
308 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
309 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_CSC |
310 MDP_PIPE_CAP_DECIMATION,
314 .base = { 0x14000, 0x16000 },
315 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
316 MDP_PIPE_CAP_DECIMATION,
321 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP,
324 .count = 2, /* LM0 and LM3 */
325 .base = { 0x44000, 0x47000 },
327 { .id = 0, .pp = 0, .dspp = 0,
328 .caps = MDP_LM_CAP_DISPLAY, },
329 { .id = 3, .pp = -1, .dspp = -1,
330 .caps = MDP_LM_CAP_WB },
334 .max_height = 0xFFFF,
342 .base = { 0x00000, 0x6a800 },
348 .max_clk = 320000000,
351 const struct mdp5_cfg_hw msm8x94_config = {
355 .caps = MDP_CAP_SMP |
363 [SSPP_VIG0] = 1, [SSPP_VIG1] = 4,
364 [SSPP_VIG2] = 7, [SSPP_VIG3] = 19,
365 [SSPP_DMA0] = 10, [SSPP_DMA1] = 13,
366 [SSPP_RGB0] = 16, [SSPP_RGB1] = 17,
367 [SSPP_RGB2] = 18, [SSPP_RGB3] = 22,
369 .reserved_state[0] = GENMASK(23, 0), /* first 24 MMBs */
371 [1] = 1, [4] = 1, [7] = 1, [19] = 1,
372 [16] = 5, [17] = 5, [18] = 5, [22] = 5,
377 .base = { 0x01000, 0x01200, 0x01400, 0x01600, 0x01800 },
378 .flush_hw_mask = 0xf0ffffff,
382 .base = { 0x04000, 0x06000, 0x08000, 0x0a000 },
383 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
384 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_CSC |
385 MDP_PIPE_CAP_DECIMATION,
389 .base = { 0x14000, 0x16000, 0x18000, 0x1a000 },
390 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP |
391 MDP_PIPE_CAP_SCALE | MDP_PIPE_CAP_DECIMATION,
395 .base = { 0x24000, 0x26000 },
396 .caps = MDP_PIPE_CAP_HFLIP | MDP_PIPE_CAP_VFLIP,
400 .base = { 0x44000, 0x45000, 0x46000, 0x47000, 0x48000, 0x49000 },
402 { .id = 0, .pp = 0, .dspp = 0,
403 .caps = MDP_LM_CAP_DISPLAY |
405 { .id = 1, .pp = 1, .dspp = 1,
406 .caps = MDP_LM_CAP_DISPLAY, },
407 { .id = 2, .pp = 2, .dspp = 2,
408 .caps = MDP_LM_CAP_DISPLAY |
410 { .id = 3, .pp = -1, .dspp = -1,
411 .caps = MDP_LM_CAP_WB, },
412 { .id = 4, .pp = -1, .dspp = -1,
413 .caps = MDP_LM_CAP_WB, },
414 { .id = 5, .pp = 3, .dspp = 3,
415 .caps = MDP_LM_CAP_DISPLAY, },
419 .max_height = 0xFFFF,
423 .base = { 0x54000, 0x56000, 0x58000, 0x5a000 },
428 .base = { 0x78000, 0x78800, 0x79000 },
432 .base = { 0x70000, 0x70800, 0x71000, 0x71800 },
435 .base = { 0x6a000, 0x6a800, 0x6b000, 0x6b800, 0x6c000 },
443 .max_clk = 400000000,
446 const struct mdp5_cfg_hw msm8x96_config = {
450 .caps = MDP_CAP_DSC |
457 .base = { 0x01000, 0x01200, 0x01400, 0x01600, 0x01800 },
458 .flush_hw_mask = 0xf4ffffff,
462 .base = { 0x04000, 0x06000, 0x08000, 0x0a000 },
463 .caps = MDP_PIPE_CAP_HFLIP |
467 MDP_PIPE_CAP_DECIMATION |
468 MDP_PIPE_CAP_SW_PIX_EXT |
473 .base = { 0x14000, 0x16000, 0x18000, 0x1a000 },
474 .caps = MDP_PIPE_CAP_HFLIP |
477 MDP_PIPE_CAP_DECIMATION |
478 MDP_PIPE_CAP_SW_PIX_EXT |
483 .base = { 0x24000, 0x26000 },
484 .caps = MDP_PIPE_CAP_HFLIP |
486 MDP_PIPE_CAP_SW_PIX_EXT |
491 .base = { 0x34000, 0x36000 },
492 .caps = MDP_PIPE_CAP_HFLIP |
494 MDP_PIPE_CAP_SW_PIX_EXT |
495 MDP_PIPE_CAP_CURSOR |
501 .base = { 0x44000, 0x45000, 0x46000, 0x47000, 0x48000, 0x49000 },
503 { .id = 0, .pp = 0, .dspp = 0,
504 .caps = MDP_LM_CAP_DISPLAY |
506 { .id = 1, .pp = 1, .dspp = 1,
507 .caps = MDP_LM_CAP_DISPLAY, },
508 { .id = 2, .pp = 2, .dspp = -1,
509 .caps = MDP_LM_CAP_DISPLAY |
511 { .id = 3, .pp = -1, .dspp = -1,
512 .caps = MDP_LM_CAP_WB, },
513 { .id = 4, .pp = -1, .dspp = -1,
514 .caps = MDP_LM_CAP_WB, },
515 { .id = 5, .pp = 3, .dspp = -1,
516 .caps = MDP_LM_CAP_DISPLAY, },
520 .max_height = 0xFFFF,
524 .base = { 0x54000, 0x56000 },
528 .base = { 0x78000, 0x78800, 0x79000 },
532 .base = { 0x70000, 0x70800, 0x71000, 0x71800 },
540 .base = { 0x80000, 0x80400 },
543 .base = { 0x6a000, 0x6a800, 0x6b000, 0x6b800, 0x6c000 },
551 .max_clk = 412500000,
554 static const struct mdp5_cfg_handler cfg_handlers[] = {
555 { .revision = 0, .config = { .hw = &msm8x74v1_config } },
556 { .revision = 2, .config = { .hw = &msm8x74v2_config } },
557 { .revision = 3, .config = { .hw = &apq8084_config } },
558 { .revision = 6, .config = { .hw = &msm8x16_config } },
559 { .revision = 9, .config = { .hw = &msm8x94_config } },
560 { .revision = 7, .config = { .hw = &msm8x96_config } },
563 static struct mdp5_cfg_platform *mdp5_get_config(struct platform_device *dev);
565 const struct mdp5_cfg_hw *mdp5_cfg_get_hw_config(struct mdp5_cfg_handler *cfg_handler)
567 return cfg_handler->config.hw;
570 struct mdp5_cfg *mdp5_cfg_get_config(struct mdp5_cfg_handler *cfg_handler)
572 return &cfg_handler->config;
575 int mdp5_cfg_get_hw_rev(struct mdp5_cfg_handler *cfg_handler)
577 return cfg_handler->revision;
580 void mdp5_cfg_destroy(struct mdp5_cfg_handler *cfg_handler)
585 struct mdp5_cfg_handler *mdp5_cfg_init(struct mdp5_kms *mdp5_kms,
586 uint32_t major, uint32_t minor)
588 struct drm_device *dev = mdp5_kms->dev;
589 struct platform_device *pdev = to_platform_device(dev->dev);
590 struct mdp5_cfg_handler *cfg_handler;
591 struct mdp5_cfg_platform *pconfig;
594 cfg_handler = kzalloc(sizeof(*cfg_handler), GFP_KERNEL);
595 if (unlikely(!cfg_handler)) {
601 dev_err(dev->dev, "unexpected MDP major version: v%d.%d\n",
607 /* only after mdp5_cfg global pointer's init can we access the hw */
608 for (i = 0; i < ARRAY_SIZE(cfg_handlers); i++) {
609 if (cfg_handlers[i].revision != minor)
611 mdp5_cfg = cfg_handlers[i].config.hw;
615 if (unlikely(!mdp5_cfg)) {
616 dev_err(dev->dev, "unexpected MDP minor revision: v%d.%d\n",
622 cfg_handler->revision = minor;
623 cfg_handler->config.hw = mdp5_cfg;
625 pconfig = mdp5_get_config(pdev);
626 memcpy(&cfg_handler->config.platform, pconfig, sizeof(*pconfig));
628 DBG("MDP5: %s hw config selected", mdp5_cfg->name);
634 mdp5_cfg_destroy(cfg_handler);
639 static struct mdp5_cfg_platform *mdp5_get_config(struct platform_device *dev)
641 static struct mdp5_cfg_platform config = {};
643 config.iommu = iommu_domain_alloc(&platform_bus_type);
645 config.iommu->geometry.aperture_start = 0x1000;
646 config.iommu->geometry.aperture_end = 0xffffffff;