drm/msm: Add SDM845 DPU support
SDM845 SoC includes the Mobile Display Sub System (MDSS) which is a
top level wrapper consisting of Display Processing Unit (DPU) and
display peripheral modules such as Display Serial Interface (DSI)
and DisplayPort (DP).
MDSS functions essentially as a back-end composition engine. It blends
video and graphic images stored in the frame buffers and scans out the
composed image to a display sink (over DSI/DP).
The following diagram represents hardware blocks for a simple pipeline
(two planes are present on a given crtc which is connected to a DSI
connector):
       MDSS
      +---------------------------------+
      | +-----------------------------+ |
      | | DPU                         | |
      | |  +--------+  +--------+     | |
      | |  |  SSPP  |  |  SSPP  |     | |
      | |  +----+---+  +----+---+     | |
      | |       |           |         | |
      | |  +----v-----------v---+     | |
      | |  |  Layer Mixer (LM)  |     | |
      | |  +--------------------+     | |
      | |  +--------------------+     | |
      | |  |    PingPong (PP)   |     | |
      | |  +--------------------+     | |
      | |  +--------------------+     | |
      | |  |  INTERFACE (VIDEO) |     | |
      | |  +---+----------------+     | |
      | +------|----------------------+ |
      |        |                        |
      | +------|---------------------+  |
      | |      | DISPLAY PERIPHERALS |  |
      | |  +---v-+      +-----+      |  |
      | |  | DSI |      |  DP |      |  |
      | |  +-----+      +-----+      |  |
      | +----------------------------+  |
      +---------------------------------+
The number of DPU sub-blocks (i.e. SSPPs, LMs, PP blocks and INTFs)
depends on SoC capabilities.
Overview of DPU sub-blocks:
---------------------------
* Source Surface Processor (SSPP):
 Refers to any of hardware pipes like ViG, DMA etc. Only ViG pipes are
 capable of performing format conversion, scaling and quality improvement
 for source surfaces.
* Layer Mixer (LM):
 Blend source surfaces together (in requested zorder)
* PingPong (PP):
 This block controls frame done interrupt output, EOL and EOF generation,
 overflow/underflow control.
* Display interface (INTF):
 Timing generator and interface connecting the display peripherals.
DRM components mapping to DPU architecture:
------------------------------------------
PLANEs maps to SSPPs
CRTC maps to LMs
Encoder maps to PPs, INTFs
Data flow setup:
---------------
MDSS hardware can support various data flows (e.g.):
  - Dual pipe: Output from two LMs combined to single display.
  - Split display: Output from two LMs connected to two separate
                   interfaces.
The hardware capabilities determine the number of concurrent data paths
possible. Any control path (i.e. pipeline w/i DPU) can be routed to any
of the hardware data paths. A given control path can be triggered,
flushed and controlled independently.
Changes in v3:
- Move msm_media_info.h from uapi to dpu/ subdir
- Remove preclose callback dpu (it's handled in core)
- Fix kbuild warnings with parent_ops
- Remove unused functions from dpu_core_irq
- Rename mdss_phys to mdss
- Rename mdp_phys address space to mdp
- Drop _phys from vbif and regdma binding names
Signed-off-by: Abhinav Kumar <abhinavk@codeaurora.org>
Signed-off-by: Archit Taneja <architt@codeaurora.org>
Signed-off-by: Chandan Uddaraju <chandanu@codeaurora.org>
Signed-off-by: Jeykumar Sankaran <jsanka@codeaurora.org>
Signed-off-by: Jordan Crouse <jcrouse@codeaurora.org>
Signed-off-by: Rajesh Yadav <ryadav@codeaurora.org>
Signed-off-by: Sravanthi Kollukuduru <skolluku@codeaurora.org>
Signed-off-by: Sean Paul <seanpaul@chromium.org>
[robclark minor rebase]
Signed-off-by: Rob Clark <robdclark@gmail.com>
			
			
This commit is contained in:
		
							parent
							
								
									036bfeb33b
								
							
						
					
					
						commit
						25fdd5933e
					
				| @ -1,5 +1,6 @@ | ||||
| # SPDX-License-Identifier: GPL-2.0
 | ||||
| ccflags-y := -Idrivers/gpu/drm/msm | ||||
| ccflags-y += -Idrivers/gpu/drm/msm/disp/dpu1 | ||||
| ccflags-$(CONFIG_DRM_MSM_DSI) += -Idrivers/gpu/drm/msm/dsi | ||||
| 
 | ||||
| msm-y := \
 | ||||
| @ -45,6 +46,34 @@ msm-y := \ | ||||
| 	disp/mdp5/mdp5_mixer.o \
 | ||||
| 	disp/mdp5/mdp5_plane.o \
 | ||||
| 	disp/mdp5/mdp5_smp.o \
 | ||||
| 	disp/dpu1/dpu_core_irq.o \
 | ||||
| 	disp/dpu1/dpu_core_perf.o \
 | ||||
| 	disp/dpu1/dpu_crtc.o \
 | ||||
| 	disp/dpu1/dpu_encoder.o \
 | ||||
| 	disp/dpu1/dpu_encoder_phys_cmd.o \
 | ||||
| 	disp/dpu1/dpu_encoder_phys_vid.o \
 | ||||
| 	disp/dpu1/dpu_formats.o \
 | ||||
| 	disp/dpu1/dpu_hw_blk.o \
 | ||||
| 	disp/dpu1/dpu_hw_catalog.o \
 | ||||
| 	disp/dpu1/dpu_hw_cdm.o \
 | ||||
| 	disp/dpu1/dpu_hw_ctl.o \
 | ||||
| 	disp/dpu1/dpu_hw_interrupts.o \
 | ||||
| 	disp/dpu1/dpu_hw_intf.o \
 | ||||
| 	disp/dpu1/dpu_hw_lm.o \
 | ||||
| 	disp/dpu1/dpu_hw_pingpong.o \
 | ||||
| 	disp/dpu1/dpu_hw_sspp.o \
 | ||||
| 	disp/dpu1/dpu_hw_top.o \
 | ||||
| 	disp/dpu1/dpu_hw_util.o \
 | ||||
| 	disp/dpu1/dpu_hw_vbif.o \
 | ||||
| 	disp/dpu1/dpu_io_util.o \
 | ||||
| 	disp/dpu1/dpu_irq.o \
 | ||||
| 	disp/dpu1/dpu_kms.o \
 | ||||
| 	disp/dpu1/dpu_kms_utils.o \
 | ||||
| 	disp/dpu1/dpu_mdss.o \
 | ||||
| 	disp/dpu1/dpu_plane.o \
 | ||||
| 	disp/dpu1/dpu_power_handle.o \
 | ||||
| 	disp/dpu1/dpu_rm.o \
 | ||||
| 	disp/dpu1/dpu_vbif.o \
 | ||||
| 	msm_atomic.o \
 | ||||
| 	msm_debugfs.o \
 | ||||
| 	msm_drv.o \
 | ||||
| @ -62,7 +91,8 @@ msm-y := \ | ||||
| 	msm_ringbuffer.o \
 | ||||
| 	msm_submitqueue.o | ||||
| 
 | ||||
| msm-$(CONFIG_DEBUG_FS) += adreno/a5xx_debugfs.o | ||||
| msm-$(CONFIG_DEBUG_FS) += adreno/a5xx_debugfs.o \
 | ||||
| 			  disp/dpu1/dpu_dbg.o | ||||
| 
 | ||||
| msm-$(CONFIG_DRM_FBDEV_EMULATION) += msm_fbdev.o | ||||
| msm-$(CONFIG_COMMON_CLK) += disp/mdp4/mdp4_lvds_pll.o | ||||
|  | ||||
							
								
								
									
										479
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_irq.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										479
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_irq.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,479 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include <linux/debugfs.h> | ||||
| #include <linux/irqdomain.h> | ||||
| #include <linux/irq.h> | ||||
| #include <linux/kthread.h> | ||||
| 
 | ||||
| #include "dpu_core_irq.h" | ||||
| #include "dpu_trace.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_callback_handler - dispatch core interrupts | ||||
|  * @arg:		private data of callback handler | ||||
|  * @irq_idx:		interrupt index | ||||
|  */ | ||||
| static void dpu_core_irq_callback_handler(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_kms *dpu_kms = arg; | ||||
| 	struct dpu_irq *irq_obj = &dpu_kms->irq_obj; | ||||
| 	struct dpu_irq_callback *cb; | ||||
| 	unsigned long irq_flags; | ||||
| 
 | ||||
| 	pr_debug("irq_idx=%d\n", irq_idx); | ||||
| 
 | ||||
| 	if (list_empty(&irq_obj->irq_cb_tbl[irq_idx])) { | ||||
| 		DRM_ERROR("no registered cb, idx:%d enable_count:%d\n", irq_idx, | ||||
| 			atomic_read(&dpu_kms->irq_obj.enable_counts[irq_idx])); | ||||
| 	} | ||||
| 
 | ||||
| 	atomic_inc(&irq_obj->irq_counts[irq_idx]); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Perform registered function callback | ||||
| 	 */ | ||||
| 	spin_lock_irqsave(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 	list_for_each_entry(cb, &irq_obj->irq_cb_tbl[irq_idx], list) | ||||
| 		if (cb->func) | ||||
| 			cb->func(cb->arg, irq_idx); | ||||
| 	spin_unlock_irqrestore(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Clear pending interrupt status in HW. | ||||
| 	 * NOTE: dpu_core_irq_callback_handler is protected by top-level | ||||
| 	 *       spinlock, so it is safe to clear any interrupt status here. | ||||
| 	 */ | ||||
| 	dpu_kms->hw_intr->ops.clear_intr_status_nolock( | ||||
| 			dpu_kms->hw_intr, | ||||
| 			irq_idx); | ||||
| } | ||||
| 
 | ||||
| int dpu_core_irq_idx_lookup(struct dpu_kms *dpu_kms, | ||||
| 		enum dpu_intr_type intr_type, u32 instance_idx) | ||||
| { | ||||
| 	if (!dpu_kms || !dpu_kms->hw_intr || | ||||
| 			!dpu_kms->hw_intr->ops.irq_idx_lookup) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	return dpu_kms->hw_intr->ops.irq_idx_lookup(intr_type, | ||||
| 			instance_idx); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * _dpu_core_irq_enable - enable core interrupt given by the index | ||||
|  * @dpu_kms:		Pointer to dpu kms context | ||||
|  * @irq_idx:		interrupt index | ||||
|  */ | ||||
| static int _dpu_core_irq_enable(struct dpu_kms *dpu_kms, int irq_idx) | ||||
| { | ||||
| 	unsigned long irq_flags; | ||||
| 	int ret = 0, enable_count; | ||||
| 
 | ||||
| 	if (!dpu_kms || !dpu_kms->hw_intr || | ||||
| 			!dpu_kms->irq_obj.enable_counts || | ||||
| 			!dpu_kms->irq_obj.irq_counts) { | ||||
| 		DPU_ERROR("invalid params\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (irq_idx < 0 || irq_idx >= dpu_kms->hw_intr->irq_idx_tbl_size) { | ||||
| 		DPU_ERROR("invalid IRQ index: [%d]\n", irq_idx); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	enable_count = atomic_read(&dpu_kms->irq_obj.enable_counts[irq_idx]); | ||||
| 	DRM_DEBUG_KMS("irq_idx=%d enable_count=%d\n", irq_idx, enable_count); | ||||
| 	trace_dpu_core_irq_enable_idx(irq_idx, enable_count); | ||||
| 
 | ||||
| 	if (atomic_inc_return(&dpu_kms->irq_obj.enable_counts[irq_idx]) == 1) { | ||||
| 		ret = dpu_kms->hw_intr->ops.enable_irq( | ||||
| 				dpu_kms->hw_intr, | ||||
| 				irq_idx); | ||||
| 		if (ret) | ||||
| 			DPU_ERROR("Fail to enable IRQ for irq_idx:%d\n", | ||||
| 					irq_idx); | ||||
| 
 | ||||
| 		DPU_DEBUG("irq_idx=%d ret=%d\n", irq_idx, ret); | ||||
| 
 | ||||
| 		spin_lock_irqsave(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 		/* empty callback list but interrupt is enabled */ | ||||
| 		if (list_empty(&dpu_kms->irq_obj.irq_cb_tbl[irq_idx])) | ||||
| 			DPU_ERROR("irq_idx=%d enabled with no callback\n", | ||||
| 					irq_idx); | ||||
| 		spin_unlock_irqrestore(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 	} | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| int dpu_core_irq_enable(struct dpu_kms *dpu_kms, int *irq_idxs, u32 irq_count) | ||||
| { | ||||
| 	int i, ret = 0, counts; | ||||
| 
 | ||||
| 	if (!dpu_kms || !irq_idxs || !irq_count) { | ||||
| 		DPU_ERROR("invalid params\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	counts = atomic_read(&dpu_kms->irq_obj.enable_counts[irq_idxs[0]]); | ||||
| 	if (counts) | ||||
| 		DRM_ERROR("irq_idx=%d enable_count=%d\n", irq_idxs[0], counts); | ||||
| 
 | ||||
| 	for (i = 0; (i < irq_count) && !ret; i++) | ||||
| 		ret = _dpu_core_irq_enable(dpu_kms, irq_idxs[i]); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * _dpu_core_irq_disable - disable core interrupt given by the index | ||||
|  * @dpu_kms:		Pointer to dpu kms context | ||||
|  * @irq_idx:		interrupt index | ||||
|  */ | ||||
| static int _dpu_core_irq_disable(struct dpu_kms *dpu_kms, int irq_idx) | ||||
| { | ||||
| 	int ret = 0, enable_count; | ||||
| 
 | ||||
| 	if (!dpu_kms || !dpu_kms->hw_intr || !dpu_kms->irq_obj.enable_counts) { | ||||
| 		DPU_ERROR("invalid params\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (irq_idx < 0 || irq_idx >= dpu_kms->hw_intr->irq_idx_tbl_size) { | ||||
| 		DPU_ERROR("invalid IRQ index: [%d]\n", irq_idx); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	enable_count = atomic_read(&dpu_kms->irq_obj.enable_counts[irq_idx]); | ||||
| 	DRM_DEBUG_KMS("irq_idx=%d enable_count=%d\n", irq_idx, enable_count); | ||||
| 	trace_dpu_core_irq_disable_idx(irq_idx, enable_count); | ||||
| 
 | ||||
| 	if (atomic_dec_return(&dpu_kms->irq_obj.enable_counts[irq_idx]) == 0) { | ||||
| 		ret = dpu_kms->hw_intr->ops.disable_irq( | ||||
| 				dpu_kms->hw_intr, | ||||
| 				irq_idx); | ||||
| 		if (ret) | ||||
| 			DPU_ERROR("Fail to disable IRQ for irq_idx:%d\n", | ||||
| 					irq_idx); | ||||
| 		DPU_DEBUG("irq_idx=%d ret=%d\n", irq_idx, ret); | ||||
| 	} | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| int dpu_core_irq_disable(struct dpu_kms *dpu_kms, int *irq_idxs, u32 irq_count) | ||||
| { | ||||
| 	int i, ret = 0, counts; | ||||
| 
 | ||||
| 	if (!dpu_kms || !irq_idxs || !irq_count) { | ||||
| 		DPU_ERROR("invalid params\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	counts = atomic_read(&dpu_kms->irq_obj.enable_counts[irq_idxs[0]]); | ||||
| 	if (counts == 2) | ||||
| 		DRM_ERROR("irq_idx=%d enable_count=%d\n", irq_idxs[0], counts); | ||||
| 
 | ||||
| 	for (i = 0; (i < irq_count) && !ret; i++) | ||||
| 		ret = _dpu_core_irq_disable(dpu_kms, irq_idxs[i]); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| u32 dpu_core_irq_read(struct dpu_kms *dpu_kms, int irq_idx, bool clear) | ||||
| { | ||||
| 	if (!dpu_kms || !dpu_kms->hw_intr || | ||||
| 			!dpu_kms->hw_intr->ops.get_interrupt_status) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	if (irq_idx < 0) { | ||||
| 		DPU_ERROR("[%pS] invalid irq_idx=%d\n", | ||||
| 				__builtin_return_address(0), irq_idx); | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	return dpu_kms->hw_intr->ops.get_interrupt_status(dpu_kms->hw_intr, | ||||
| 			irq_idx, clear); | ||||
| } | ||||
| 
 | ||||
| int dpu_core_irq_register_callback(struct dpu_kms *dpu_kms, int irq_idx, | ||||
| 		struct dpu_irq_callback *register_irq_cb) | ||||
| { | ||||
| 	unsigned long irq_flags; | ||||
| 
 | ||||
| 	if (!dpu_kms || !dpu_kms->irq_obj.irq_cb_tbl) { | ||||
| 		DPU_ERROR("invalid params\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (!register_irq_cb || !register_irq_cb->func) { | ||||
| 		DPU_ERROR("invalid irq_cb:%d func:%d\n", | ||||
| 				register_irq_cb != NULL, | ||||
| 				register_irq_cb ? | ||||
| 					register_irq_cb->func != NULL : -1); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (irq_idx < 0 || irq_idx >= dpu_kms->hw_intr->irq_idx_tbl_size) { | ||||
| 		DPU_ERROR("invalid IRQ index: [%d]\n", irq_idx); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG("[%pS] irq_idx=%d\n", __builtin_return_address(0), irq_idx); | ||||
| 
 | ||||
| 	spin_lock_irqsave(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 	trace_dpu_core_irq_register_callback(irq_idx, register_irq_cb); | ||||
| 	list_del_init(®ister_irq_cb->list); | ||||
| 	list_add_tail(®ister_irq_cb->list, | ||||
| 			&dpu_kms->irq_obj.irq_cb_tbl[irq_idx]); | ||||
| 	spin_unlock_irqrestore(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| int dpu_core_irq_unregister_callback(struct dpu_kms *dpu_kms, int irq_idx, | ||||
| 		struct dpu_irq_callback *register_irq_cb) | ||||
| { | ||||
| 	unsigned long irq_flags; | ||||
| 
 | ||||
| 	if (!dpu_kms || !dpu_kms->irq_obj.irq_cb_tbl) { | ||||
| 		DPU_ERROR("invalid params\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (!register_irq_cb || !register_irq_cb->func) { | ||||
| 		DPU_ERROR("invalid irq_cb:%d func:%d\n", | ||||
| 				register_irq_cb != NULL, | ||||
| 				register_irq_cb ? | ||||
| 					register_irq_cb->func != NULL : -1); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (irq_idx < 0 || irq_idx >= dpu_kms->hw_intr->irq_idx_tbl_size) { | ||||
| 		DPU_ERROR("invalid IRQ index: [%d]\n", irq_idx); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG("[%pS] irq_idx=%d\n", __builtin_return_address(0), irq_idx); | ||||
| 
 | ||||
| 	spin_lock_irqsave(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 	trace_dpu_core_irq_unregister_callback(irq_idx, register_irq_cb); | ||||
| 	list_del_init(®ister_irq_cb->list); | ||||
| 	/* empty callback list but interrupt is still enabled */ | ||||
| 	if (list_empty(&dpu_kms->irq_obj.irq_cb_tbl[irq_idx]) && | ||||
| 			atomic_read(&dpu_kms->irq_obj.enable_counts[irq_idx])) | ||||
| 		DPU_ERROR("irq_idx=%d enabled with no callback\n", irq_idx); | ||||
| 	spin_unlock_irqrestore(&dpu_kms->irq_obj.cb_lock, irq_flags); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void dpu_clear_all_irqs(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	if (!dpu_kms || !dpu_kms->hw_intr || | ||||
| 			!dpu_kms->hw_intr->ops.clear_all_irqs) | ||||
| 		return; | ||||
| 
 | ||||
| 	dpu_kms->hw_intr->ops.clear_all_irqs(dpu_kms->hw_intr); | ||||
| } | ||||
| 
 | ||||
| static void dpu_disable_all_irqs(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	if (!dpu_kms || !dpu_kms->hw_intr || | ||||
| 			!dpu_kms->hw_intr->ops.disable_all_irqs) | ||||
| 		return; | ||||
| 
 | ||||
| 	dpu_kms->hw_intr->ops.disable_all_irqs(dpu_kms->hw_intr); | ||||
| } | ||||
| 
 | ||||
| #ifdef CONFIG_DEBUG_FS | ||||
| #define DEFINE_DPU_DEBUGFS_SEQ_FOPS(__prefix)				\ | ||||
| static int __prefix ## _open(struct inode *inode, struct file *file)	\ | ||||
| {									\ | ||||
| 	return single_open(file, __prefix ## _show, inode->i_private);	\ | ||||
| }									\ | ||||
| static const struct file_operations __prefix ## _fops = {		\ | ||||
| 	.owner = THIS_MODULE,						\ | ||||
| 	.open = __prefix ## _open,					\ | ||||
| 	.release = single_release,					\ | ||||
| 	.read = seq_read,						\ | ||||
| 	.llseek = seq_lseek,						\ | ||||
| } | ||||
| 
 | ||||
| static int dpu_debugfs_core_irq_show(struct seq_file *s, void *v) | ||||
| { | ||||
| 	struct dpu_irq *irq_obj = s->private; | ||||
| 	struct dpu_irq_callback *cb; | ||||
| 	unsigned long irq_flags; | ||||
| 	int i, irq_count, enable_count, cb_count; | ||||
| 
 | ||||
| 	if (!irq_obj || !irq_obj->enable_counts || !irq_obj->irq_cb_tbl) { | ||||
| 		DPU_ERROR("invalid parameters\n"); | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = 0; i < irq_obj->total_irqs; i++) { | ||||
| 		spin_lock_irqsave(&irq_obj->cb_lock, irq_flags); | ||||
| 		cb_count = 0; | ||||
| 		irq_count = atomic_read(&irq_obj->irq_counts[i]); | ||||
| 		enable_count = atomic_read(&irq_obj->enable_counts[i]); | ||||
| 		list_for_each_entry(cb, &irq_obj->irq_cb_tbl[i], list) | ||||
| 			cb_count++; | ||||
| 		spin_unlock_irqrestore(&irq_obj->cb_lock, irq_flags); | ||||
| 
 | ||||
| 		if (irq_count || enable_count || cb_count) | ||||
| 			seq_printf(s, "idx:%d irq:%d enable:%d cb:%d\n", | ||||
| 					i, irq_count, enable_count, cb_count); | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| DEFINE_DPU_DEBUGFS_SEQ_FOPS(dpu_debugfs_core_irq); | ||||
| 
 | ||||
| int dpu_debugfs_core_irq_init(struct dpu_kms *dpu_kms, | ||||
| 		struct dentry *parent) | ||||
| { | ||||
| 	dpu_kms->irq_obj.debugfs_file = debugfs_create_file("core_irq", 0600, | ||||
| 			parent, &dpu_kms->irq_obj, | ||||
| 			&dpu_debugfs_core_irq_fops); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| void dpu_debugfs_core_irq_destroy(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	debugfs_remove(dpu_kms->irq_obj.debugfs_file); | ||||
| 	dpu_kms->irq_obj.debugfs_file = NULL; | ||||
| } | ||||
| 
 | ||||
| #else | ||||
| int dpu_debugfs_core_irq_init(struct dpu_kms *dpu_kms, | ||||
| 		struct dentry *parent) | ||||
| { | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| void dpu_debugfs_core_irq_destroy(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| void dpu_core_irq_preinstall(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	struct msm_drm_private *priv; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!dpu_kms) { | ||||
| 		DPU_ERROR("invalid dpu_kms\n"); | ||||
| 		return; | ||||
| 	} else if (!dpu_kms->dev) { | ||||
| 		DPU_ERROR("invalid drm device\n"); | ||||
| 		return; | ||||
| 	} else if (!dpu_kms->dev->dev_private) { | ||||
| 		DPU_ERROR("invalid device private\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	priv = dpu_kms->dev->dev_private; | ||||
| 
 | ||||
| 	pm_runtime_get_sync(&dpu_kms->pdev->dev); | ||||
| 	dpu_clear_all_irqs(dpu_kms); | ||||
| 	dpu_disable_all_irqs(dpu_kms); | ||||
| 	pm_runtime_put_sync(&dpu_kms->pdev->dev); | ||||
| 
 | ||||
| 	spin_lock_init(&dpu_kms->irq_obj.cb_lock); | ||||
| 
 | ||||
| 	/* Create irq callbacks for all possible irq_idx */ | ||||
| 	dpu_kms->irq_obj.total_irqs = dpu_kms->hw_intr->irq_idx_tbl_size; | ||||
| 	dpu_kms->irq_obj.irq_cb_tbl = kcalloc(dpu_kms->irq_obj.total_irqs, | ||||
| 			sizeof(struct list_head), GFP_KERNEL); | ||||
| 	dpu_kms->irq_obj.enable_counts = kcalloc(dpu_kms->irq_obj.total_irqs, | ||||
| 			sizeof(atomic_t), GFP_KERNEL); | ||||
| 	dpu_kms->irq_obj.irq_counts = kcalloc(dpu_kms->irq_obj.total_irqs, | ||||
| 			sizeof(atomic_t), GFP_KERNEL); | ||||
| 	for (i = 0; i < dpu_kms->irq_obj.total_irqs; i++) { | ||||
| 		INIT_LIST_HEAD(&dpu_kms->irq_obj.irq_cb_tbl[i]); | ||||
| 		atomic_set(&dpu_kms->irq_obj.enable_counts[i], 0); | ||||
| 		atomic_set(&dpu_kms->irq_obj.irq_counts[i], 0); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| int dpu_core_irq_postinstall(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| void dpu_core_irq_uninstall(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	struct msm_drm_private *priv; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!dpu_kms) { | ||||
| 		DPU_ERROR("invalid dpu_kms\n"); | ||||
| 		return; | ||||
| 	} else if (!dpu_kms->dev) { | ||||
| 		DPU_ERROR("invalid drm device\n"); | ||||
| 		return; | ||||
| 	} else if (!dpu_kms->dev->dev_private) { | ||||
| 		DPU_ERROR("invalid device private\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	priv = dpu_kms->dev->dev_private; | ||||
| 
 | ||||
| 	pm_runtime_get_sync(&dpu_kms->pdev->dev); | ||||
| 	for (i = 0; i < dpu_kms->irq_obj.total_irqs; i++) | ||||
| 		if (atomic_read(&dpu_kms->irq_obj.enable_counts[i]) || | ||||
| 				!list_empty(&dpu_kms->irq_obj.irq_cb_tbl[i])) | ||||
| 			DPU_ERROR("irq_idx=%d still enabled/registered\n", i); | ||||
| 
 | ||||
| 	dpu_clear_all_irqs(dpu_kms); | ||||
| 	dpu_disable_all_irqs(dpu_kms); | ||||
| 	pm_runtime_put_sync(&dpu_kms->pdev->dev); | ||||
| 
 | ||||
| 	kfree(dpu_kms->irq_obj.irq_cb_tbl); | ||||
| 	kfree(dpu_kms->irq_obj.enable_counts); | ||||
| 	kfree(dpu_kms->irq_obj.irq_counts); | ||||
| 	dpu_kms->irq_obj.irq_cb_tbl = NULL; | ||||
| 	dpu_kms->irq_obj.enable_counts = NULL; | ||||
| 	dpu_kms->irq_obj.irq_counts = NULL; | ||||
| 	dpu_kms->irq_obj.total_irqs = 0; | ||||
| } | ||||
| 
 | ||||
| irqreturn_t dpu_core_irq(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	/*
 | ||||
| 	 * Read interrupt status from all sources. Interrupt status are | ||||
| 	 * stored within hw_intr. | ||||
| 	 * Function will also clear the interrupt status after reading. | ||||
| 	 * Individual interrupt status bit will only get stored if it | ||||
| 	 * is enabled. | ||||
| 	 */ | ||||
| 	dpu_kms->hw_intr->ops.get_interrupt_statuses(dpu_kms->hw_intr); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Dispatch to HW driver to handle interrupt lookup that is being | ||||
| 	 * fired. When matching interrupt is located, HW driver will call to | ||||
| 	 * dpu_core_irq_callback_handler with the irq_idx from the lookup table. | ||||
| 	 * dpu_core_irq_callback_handler will perform the registered function | ||||
| 	 * callback, and do the interrupt status clearing once the registered | ||||
| 	 * callback is finished. | ||||
| 	 */ | ||||
| 	dpu_kms->hw_intr->ops.dispatch_irqs( | ||||
| 			dpu_kms->hw_intr, | ||||
| 			dpu_core_irq_callback_handler, | ||||
| 			dpu_kms); | ||||
| 
 | ||||
| 	return IRQ_HANDLED; | ||||
| } | ||||
							
								
								
									
										153
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_irq.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										153
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_irq.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,153 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_CORE_IRQ_H__ | ||||
| #define __DPU_CORE_IRQ_H__ | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_hw_interrupts.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_preinstall - perform pre-installation of core IRQ handler | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @return:		none | ||||
|  */ | ||||
| void dpu_core_irq_preinstall(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_postinstall - perform post-installation of core IRQ handler | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @return:		0 if success; error code otherwise | ||||
|  */ | ||||
| int dpu_core_irq_postinstall(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_uninstall - uninstall core IRQ handler | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @return:		none | ||||
|  */ | ||||
| void dpu_core_irq_uninstall(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq - core IRQ handler | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @return:		interrupt handling status | ||||
|  */ | ||||
| irqreturn_t dpu_core_irq(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_idx_lookup - IRQ helper function for lookup irq_idx from HW | ||||
|  *                      interrupt mapping table. | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @intr_type:		DPU HW interrupt type for lookup | ||||
|  * @instance_idx:	DPU HW block instance defined in dpu_hw_mdss.h | ||||
|  * @return:		irq_idx or -EINVAL when fail to lookup | ||||
|  */ | ||||
| int dpu_core_irq_idx_lookup( | ||||
| 		struct dpu_kms *dpu_kms, | ||||
| 		enum dpu_intr_type intr_type, | ||||
| 		uint32_t instance_idx); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_enable - IRQ helper function for enabling one or more IRQs | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @irq_idxs:		Array of irq index | ||||
|  * @irq_count:		Number of irq_idx provided in the array | ||||
|  * @return:		0 for success enabling IRQ, otherwise failure | ||||
|  * | ||||
|  * This function increments count on each enable and decrements on each | ||||
|  * disable.  Interrupts is enabled if count is 0 before increment. | ||||
|  */ | ||||
| int dpu_core_irq_enable( | ||||
| 		struct dpu_kms *dpu_kms, | ||||
| 		int *irq_idxs, | ||||
| 		uint32_t irq_count); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_disable - IRQ helper function for disabling one of more IRQs | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @irq_idxs:		Array of irq index | ||||
|  * @irq_count:		Number of irq_idx provided in the array | ||||
|  * @return:		0 for success disabling IRQ, otherwise failure | ||||
|  * | ||||
|  * This function increments count on each enable and decrements on each | ||||
|  * disable.  Interrupts is disabled if count is 0 after decrement. | ||||
|  */ | ||||
| int dpu_core_irq_disable( | ||||
| 		struct dpu_kms *dpu_kms, | ||||
| 		int *irq_idxs, | ||||
| 		uint32_t irq_count); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_read - IRQ helper function for reading IRQ status | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @irq_idx:		irq index | ||||
|  * @clear:		True to clear the irq after read | ||||
|  * @return:		non-zero if irq detected; otherwise no irq detected | ||||
|  */ | ||||
| u32 dpu_core_irq_read( | ||||
| 		struct dpu_kms *dpu_kms, | ||||
| 		int irq_idx, | ||||
| 		bool clear); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_register_callback - For registering callback function on IRQ | ||||
|  *                             interrupt | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @irq_idx:		irq index | ||||
|  * @irq_cb:		IRQ callback structure, containing callback function | ||||
|  *			and argument. Passing NULL for irq_cb will unregister | ||||
|  *			the callback for the given irq_idx | ||||
|  *			This must exist until un-registration. | ||||
|  * @return:		0 for success registering callback, otherwise failure | ||||
|  * | ||||
|  * This function supports registration of multiple callbacks for each interrupt. | ||||
|  */ | ||||
| int dpu_core_irq_register_callback( | ||||
| 		struct dpu_kms *dpu_kms, | ||||
| 		int irq_idx, | ||||
| 		struct dpu_irq_callback *irq_cb); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_irq_unregister_callback - For unregistering callback function on IRQ | ||||
|  *                             interrupt | ||||
|  * @dpu_kms:		DPU handle | ||||
|  * @irq_idx:		irq index | ||||
|  * @irq_cb:		IRQ callback structure, containing callback function | ||||
|  *			and argument. Passing NULL for irq_cb will unregister | ||||
|  *			the callback for the given irq_idx | ||||
|  *			This must match with registration. | ||||
|  * @return:		0 for success registering callback, otherwise failure | ||||
|  * | ||||
|  * This function supports registration of multiple callbacks for each interrupt. | ||||
|  */ | ||||
| int dpu_core_irq_unregister_callback( | ||||
| 		struct dpu_kms *dpu_kms, | ||||
| 		int irq_idx, | ||||
| 		struct dpu_irq_callback *irq_cb); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_debugfs_core_irq_init - register core irq debugfs | ||||
|  * @dpu_kms: pointer to kms | ||||
|  * @parent: debugfs directory root | ||||
|  * @Return: 0 on success | ||||
|  */ | ||||
| int dpu_debugfs_core_irq_init(struct dpu_kms *dpu_kms, | ||||
| 		struct dentry *parent); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_debugfs_core_irq_destroy - deregister core irq debugfs | ||||
|  * @dpu_kms: pointer to kms | ||||
|  */ | ||||
| void dpu_debugfs_core_irq_destroy(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| #endif /* __DPU_CORE_IRQ_H__ */ | ||||
							
								
								
									
										637
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										637
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,637 @@ | ||||
| /* Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include <linux/debugfs.h> | ||||
| #include <linux/errno.h> | ||||
| #include <linux/mutex.h> | ||||
| #include <linux/sort.h> | ||||
| #include <linux/clk.h> | ||||
| #include <linux/bitmap.h> | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_trace.h" | ||||
| #include "dpu_crtc.h" | ||||
| #include "dpu_core_perf.h" | ||||
| 
 | ||||
| #define DPU_PERF_MODE_STRING_SIZE	128 | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_perf_mode - performance tuning mode | ||||
|  * @DPU_PERF_MODE_NORMAL: performance controlled by user mode client | ||||
|  * @DPU_PERF_MODE_MINIMUM: performance bounded by minimum setting | ||||
|  * @DPU_PERF_MODE_FIXED: performance bounded by fixed setting | ||||
|  */ | ||||
| enum dpu_perf_mode { | ||||
| 	DPU_PERF_MODE_NORMAL, | ||||
| 	DPU_PERF_MODE_MINIMUM, | ||||
| 	DPU_PERF_MODE_FIXED, | ||||
| 	DPU_PERF_MODE_MAX | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_kms *_dpu_crtc_get_kms(struct drm_crtc *crtc) | ||||
| { | ||||
| 	struct msm_drm_private *priv; | ||||
| 
 | ||||
| 	if (!crtc->dev || !crtc->dev->dev_private) { | ||||
| 		DPU_ERROR("invalid device\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 
 | ||||
| 	priv = crtc->dev->dev_private; | ||||
| 	if (!priv || !priv->kms) { | ||||
| 		DPU_ERROR("invalid kms\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 
 | ||||
| 	return to_dpu_kms(priv->kms); | ||||
| } | ||||
| 
 | ||||
| static bool _dpu_core_perf_crtc_is_power_on(struct drm_crtc *crtc) | ||||
| { | ||||
| 	return dpu_crtc_is_enabled(crtc); | ||||
| } | ||||
| 
 | ||||
| static bool _dpu_core_video_mode_intf_connected(struct drm_crtc *crtc) | ||||
| { | ||||
| 	struct drm_crtc *tmp_crtc; | ||||
| 	bool intf_connected = false; | ||||
| 
 | ||||
| 	if (!crtc) | ||||
| 		goto end; | ||||
| 
 | ||||
| 	drm_for_each_crtc(tmp_crtc, crtc->dev) { | ||||
| 		if ((dpu_crtc_get_intf_mode(tmp_crtc) == INTF_MODE_VIDEO) && | ||||
| 				_dpu_core_perf_crtc_is_power_on(tmp_crtc)) { | ||||
| 			DPU_DEBUG("video interface connected crtc:%d\n", | ||||
| 				tmp_crtc->base.id); | ||||
| 			intf_connected = true; | ||||
| 			goto end; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| end: | ||||
| 	return intf_connected; | ||||
| } | ||||
| 
 | ||||
| static void _dpu_core_perf_calc_crtc(struct dpu_kms *kms, | ||||
| 		struct drm_crtc *crtc, | ||||
| 		struct drm_crtc_state *state, | ||||
| 		struct dpu_core_perf_params *perf) | ||||
| { | ||||
| 	struct dpu_crtc_state *dpu_cstate; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!kms || !kms->catalog || !crtc || !state || !perf) { | ||||
| 		DPU_ERROR("invalid parameters\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_cstate = to_dpu_crtc_state(state); | ||||
| 	memset(perf, 0, sizeof(struct dpu_core_perf_params)); | ||||
| 
 | ||||
| 	if (!dpu_cstate->bw_control) { | ||||
| 		for (i = 0; i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 			perf->bw_ctl[i] = kms->catalog->perf.max_bw_high * | ||||
| 					1000ULL; | ||||
| 			perf->max_per_pipe_ib[i] = perf->bw_ctl[i]; | ||||
| 		} | ||||
| 		perf->core_clk_rate = kms->perf.max_core_clk_rate; | ||||
| 	} else if (kms->perf.perf_tune.mode == DPU_PERF_MODE_MINIMUM) { | ||||
| 		for (i = 0; i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 			perf->bw_ctl[i] = 0; | ||||
| 			perf->max_per_pipe_ib[i] = 0; | ||||
| 		} | ||||
| 		perf->core_clk_rate = 0; | ||||
| 	} else if (kms->perf.perf_tune.mode == DPU_PERF_MODE_FIXED) { | ||||
| 		for (i = 0; i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 			perf->bw_ctl[i] = kms->perf.fix_core_ab_vote; | ||||
| 			perf->max_per_pipe_ib[i] = kms->perf.fix_core_ib_vote; | ||||
| 		} | ||||
| 		perf->core_clk_rate = kms->perf.fix_core_clk_rate; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG( | ||||
| 		"crtc=%d clk_rate=%llu core_ib=%llu core_ab=%llu llcc_ib=%llu llcc_ab=%llu mem_ib=%llu mem_ab=%llu\n", | ||||
| 			crtc->base.id, perf->core_clk_rate, | ||||
| 			perf->max_per_pipe_ib[DPU_POWER_HANDLE_DBUS_ID_MNOC], | ||||
| 			perf->bw_ctl[DPU_POWER_HANDLE_DBUS_ID_MNOC], | ||||
| 			perf->max_per_pipe_ib[DPU_POWER_HANDLE_DBUS_ID_LLCC], | ||||
| 			perf->bw_ctl[DPU_POWER_HANDLE_DBUS_ID_LLCC], | ||||
| 			perf->max_per_pipe_ib[DPU_POWER_HANDLE_DBUS_ID_EBI], | ||||
| 			perf->bw_ctl[DPU_POWER_HANDLE_DBUS_ID_EBI]); | ||||
| } | ||||
| 
 | ||||
| int dpu_core_perf_crtc_check(struct drm_crtc *crtc, | ||||
| 		struct drm_crtc_state *state) | ||||
| { | ||||
| 	u32 bw, threshold; | ||||
| 	u64 bw_sum_of_intfs = 0; | ||||
| 	enum dpu_crtc_client_type curr_client_type; | ||||
| 	bool is_video_mode; | ||||
| 	struct dpu_crtc_state *dpu_cstate; | ||||
| 	struct drm_crtc *tmp_crtc; | ||||
| 	struct dpu_kms *kms; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!crtc || !state) { | ||||
| 		DPU_ERROR("invalid crtc\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	kms = _dpu_crtc_get_kms(crtc); | ||||
| 	if (!kms || !kms->catalog) { | ||||
| 		DPU_ERROR("invalid parameters\n"); | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	/* we only need bandwidth check on real-time clients (interfaces) */ | ||||
| 	if (dpu_crtc_get_client_type(crtc) == NRT_CLIENT) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	dpu_cstate = to_dpu_crtc_state(state); | ||||
| 
 | ||||
| 	/* obtain new values */ | ||||
| 	_dpu_core_perf_calc_crtc(kms, crtc, state, &dpu_cstate->new_perf); | ||||
| 
 | ||||
| 	for (i = DPU_POWER_HANDLE_DBUS_ID_MNOC; | ||||
| 			i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 		bw_sum_of_intfs = dpu_cstate->new_perf.bw_ctl[i]; | ||||
| 		curr_client_type = dpu_crtc_get_client_type(crtc); | ||||
| 
 | ||||
| 		drm_for_each_crtc(tmp_crtc, crtc->dev) { | ||||
| 			if (_dpu_core_perf_crtc_is_power_on(tmp_crtc) && | ||||
| 			    (dpu_crtc_get_client_type(tmp_crtc) == | ||||
| 					    curr_client_type) && | ||||
| 			    (tmp_crtc != crtc)) { | ||||
| 				struct dpu_crtc_state *tmp_cstate = | ||||
| 					to_dpu_crtc_state(tmp_crtc->state); | ||||
| 
 | ||||
| 				DPU_DEBUG("crtc:%d bw:%llu ctrl:%d\n", | ||||
| 					tmp_crtc->base.id, | ||||
| 					tmp_cstate->new_perf.bw_ctl[i], | ||||
| 					tmp_cstate->bw_control); | ||||
| 				/*
 | ||||
| 				 * For bw check only use the bw if the | ||||
| 				 * atomic property has been already set | ||||
| 				 */ | ||||
| 				if (tmp_cstate->bw_control) | ||||
| 					bw_sum_of_intfs += | ||||
| 						tmp_cstate->new_perf.bw_ctl[i]; | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		/* convert bandwidth to kb */ | ||||
| 		bw = DIV_ROUND_UP_ULL(bw_sum_of_intfs, 1000); | ||||
| 		DPU_DEBUG("calculated bandwidth=%uk\n", bw); | ||||
| 
 | ||||
| 		is_video_mode = dpu_crtc_get_intf_mode(crtc) == INTF_MODE_VIDEO; | ||||
| 		threshold = (is_video_mode || | ||||
| 			_dpu_core_video_mode_intf_connected(crtc)) ? | ||||
| 			kms->catalog->perf.max_bw_low : | ||||
| 			kms->catalog->perf.max_bw_high; | ||||
| 
 | ||||
| 		DPU_DEBUG("final threshold bw limit = %d\n", threshold); | ||||
| 
 | ||||
| 		if (!dpu_cstate->bw_control) { | ||||
| 			DPU_DEBUG("bypass bandwidth check\n"); | ||||
| 		} else if (!threshold) { | ||||
| 			DPU_ERROR("no bandwidth limits specified\n"); | ||||
| 			return -E2BIG; | ||||
| 		} else if (bw > threshold) { | ||||
| 			DPU_ERROR("exceeds bandwidth: %ukb > %ukb\n", bw, | ||||
| 					threshold); | ||||
| 			return -E2BIG; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int _dpu_core_perf_crtc_update_bus(struct dpu_kms *kms, | ||||
| 		struct drm_crtc *crtc, u32 bus_id) | ||||
| { | ||||
| 	struct dpu_core_perf_params perf = { { 0 } }; | ||||
| 	enum dpu_crtc_client_type curr_client_type | ||||
| 					= dpu_crtc_get_client_type(crtc); | ||||
| 	struct drm_crtc *tmp_crtc; | ||||
| 	struct dpu_crtc_state *dpu_cstate; | ||||
| 	int ret = 0; | ||||
| 
 | ||||
| 	drm_for_each_crtc(tmp_crtc, crtc->dev) { | ||||
| 		if (_dpu_core_perf_crtc_is_power_on(tmp_crtc) && | ||||
| 			curr_client_type == | ||||
| 				dpu_crtc_get_client_type(tmp_crtc)) { | ||||
| 			dpu_cstate = to_dpu_crtc_state(tmp_crtc->state); | ||||
| 
 | ||||
| 			perf.max_per_pipe_ib[bus_id] = | ||||
| 				max(perf.max_per_pipe_ib[bus_id], | ||||
| 				dpu_cstate->new_perf.max_per_pipe_ib[bus_id]); | ||||
| 
 | ||||
| 			DPU_DEBUG("crtc=%d bus_id=%d bw=%llu\n", | ||||
| 				tmp_crtc->base.id, bus_id, | ||||
| 				dpu_cstate->new_perf.bw_ctl[bus_id]); | ||||
| 		} | ||||
| 	} | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * @dpu_core_perf_crtc_release_bw() - request zero bandwidth | ||||
|  * @crtc - pointer to a crtc | ||||
|  * | ||||
|  * Function checks a state variable for the crtc, if all pending commit | ||||
|  * requests are done, meaning no more bandwidth is needed, release | ||||
|  * bandwidth request. | ||||
|  */ | ||||
| void dpu_core_perf_crtc_release_bw(struct drm_crtc *crtc) | ||||
| { | ||||
| 	struct drm_crtc *tmp_crtc; | ||||
| 	struct dpu_crtc *dpu_crtc; | ||||
| 	struct dpu_crtc_state *dpu_cstate; | ||||
| 	struct dpu_kms *kms; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!crtc) { | ||||
| 		DPU_ERROR("invalid crtc\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	kms = _dpu_crtc_get_kms(crtc); | ||||
| 	if (!kms || !kms->catalog) { | ||||
| 		DPU_ERROR("invalid kms\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_crtc = to_dpu_crtc(crtc); | ||||
| 	dpu_cstate = to_dpu_crtc_state(crtc->state); | ||||
| 
 | ||||
| 	/* only do this for command mode rt client */ | ||||
| 	if (dpu_crtc_get_intf_mode(crtc) != INTF_MODE_CMD) | ||||
| 		return; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * If video interface present, cmd panel bandwidth cannot be | ||||
| 	 * released. | ||||
| 	 */ | ||||
| 	if (dpu_crtc_get_intf_mode(crtc) == INTF_MODE_CMD) | ||||
| 		drm_for_each_crtc(tmp_crtc, crtc->dev) { | ||||
| 			if (_dpu_core_perf_crtc_is_power_on(tmp_crtc) && | ||||
| 				dpu_crtc_get_intf_mode(tmp_crtc) == | ||||
| 						INTF_MODE_VIDEO) | ||||
| 				return; | ||||
| 		} | ||||
| 
 | ||||
| 	/* Release the bandwidth */ | ||||
| 	if (kms->perf.enable_bw_release) { | ||||
| 		trace_dpu_cmd_release_bw(crtc->base.id); | ||||
| 		DPU_DEBUG("Release BW crtc=%d\n", crtc->base.id); | ||||
| 		for (i = 0; i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 			dpu_crtc->cur_perf.bw_ctl[i] = 0; | ||||
| 			_dpu_core_perf_crtc_update_bus(kms, crtc, i); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int _dpu_core_perf_set_core_clk_rate(struct dpu_kms *kms, u64 rate) | ||||
| { | ||||
| 	struct dss_clk *core_clk = kms->perf.core_clk; | ||||
| 
 | ||||
| 	if (core_clk->max_rate && (rate > core_clk->max_rate)) | ||||
| 		rate = core_clk->max_rate; | ||||
| 
 | ||||
| 	core_clk->rate = rate; | ||||
| 	return msm_dss_clk_set_rate(core_clk, 1); | ||||
| } | ||||
| 
 | ||||
| static u64 _dpu_core_perf_get_core_clk_rate(struct dpu_kms *kms) | ||||
| { | ||||
| 	u64 clk_rate = kms->perf.perf_tune.min_core_clk; | ||||
| 	struct drm_crtc *crtc; | ||||
| 	struct dpu_crtc_state *dpu_cstate; | ||||
| 
 | ||||
| 	drm_for_each_crtc(crtc, kms->dev) { | ||||
| 		if (_dpu_core_perf_crtc_is_power_on(crtc)) { | ||||
| 			dpu_cstate = to_dpu_crtc_state(crtc->state); | ||||
| 			clk_rate = max(dpu_cstate->new_perf.core_clk_rate, | ||||
| 							clk_rate); | ||||
| 			clk_rate = clk_round_rate(kms->perf.core_clk->clk, | ||||
| 					clk_rate); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if (kms->perf.perf_tune.mode == DPU_PERF_MODE_FIXED) | ||||
| 		clk_rate = kms->perf.fix_core_clk_rate; | ||||
| 
 | ||||
| 	DPU_DEBUG("clk:%llu\n", clk_rate); | ||||
| 
 | ||||
| 	return clk_rate; | ||||
| } | ||||
| 
 | ||||
| int dpu_core_perf_crtc_update(struct drm_crtc *crtc, | ||||
| 		int params_changed, bool stop_req) | ||||
| { | ||||
| 	struct dpu_core_perf_params *new, *old; | ||||
| 	int update_bus = 0, update_clk = 0; | ||||
| 	u64 clk_rate = 0; | ||||
| 	struct dpu_crtc *dpu_crtc; | ||||
| 	struct dpu_crtc_state *dpu_cstate; | ||||
| 	int i; | ||||
| 	struct msm_drm_private *priv; | ||||
| 	struct dpu_kms *kms; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!crtc) { | ||||
| 		DPU_ERROR("invalid crtc\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	kms = _dpu_crtc_get_kms(crtc); | ||||
| 	if (!kms || !kms->catalog) { | ||||
| 		DPU_ERROR("invalid kms\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 	priv = kms->dev->dev_private; | ||||
| 
 | ||||
| 	dpu_crtc = to_dpu_crtc(crtc); | ||||
| 	dpu_cstate = to_dpu_crtc_state(crtc->state); | ||||
| 
 | ||||
| 	DPU_DEBUG("crtc:%d stop_req:%d core_clk:%llu\n", | ||||
| 			crtc->base.id, stop_req, kms->perf.core_clk_rate); | ||||
| 
 | ||||
| 	old = &dpu_crtc->cur_perf; | ||||
| 	new = &dpu_cstate->new_perf; | ||||
| 
 | ||||
| 	if (_dpu_core_perf_crtc_is_power_on(crtc) && !stop_req) { | ||||
| 		for (i = 0; i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 			/*
 | ||||
| 			 * cases for bus bandwidth update. | ||||
| 			 * 1. new bandwidth vote - "ab or ib vote" is higher | ||||
| 			 *    than current vote for update request. | ||||
| 			 * 2. new bandwidth vote - "ab or ib vote" is lower | ||||
| 			 *    than current vote at end of commit or stop. | ||||
| 			 */ | ||||
| 			if ((params_changed && ((new->bw_ctl[i] > | ||||
| 						old->bw_ctl[i]) || | ||||
| 				  (new->max_per_pipe_ib[i] > | ||||
| 						old->max_per_pipe_ib[i]))) || | ||||
| 			    (!params_changed && ((new->bw_ctl[i] < | ||||
| 						old->bw_ctl[i]) || | ||||
| 				  (new->max_per_pipe_ib[i] < | ||||
| 						old->max_per_pipe_ib[i])))) { | ||||
| 				DPU_DEBUG( | ||||
| 					"crtc=%d p=%d new_bw=%llu,old_bw=%llu\n", | ||||
| 					crtc->base.id, params_changed, | ||||
| 					new->bw_ctl[i], old->bw_ctl[i]); | ||||
| 				old->bw_ctl[i] = new->bw_ctl[i]; | ||||
| 				old->max_per_pipe_ib[i] = | ||||
| 						new->max_per_pipe_ib[i]; | ||||
| 				update_bus |= BIT(i); | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if ((params_changed && | ||||
| 				(new->core_clk_rate > old->core_clk_rate)) || | ||||
| 				(!params_changed && | ||||
| 				(new->core_clk_rate < old->core_clk_rate))) { | ||||
| 			old->core_clk_rate = new->core_clk_rate; | ||||
| 			update_clk = 1; | ||||
| 		} | ||||
| 	} else { | ||||
| 		DPU_DEBUG("crtc=%d disable\n", crtc->base.id); | ||||
| 		memset(old, 0, sizeof(*old)); | ||||
| 		memset(new, 0, sizeof(*new)); | ||||
| 		update_bus = ~0; | ||||
| 		update_clk = 1; | ||||
| 	} | ||||
| 	trace_dpu_perf_crtc_update(crtc->base.id, | ||||
| 				new->bw_ctl[DPU_POWER_HANDLE_DBUS_ID_MNOC], | ||||
| 				new->bw_ctl[DPU_POWER_HANDLE_DBUS_ID_LLCC], | ||||
| 				new->bw_ctl[DPU_POWER_HANDLE_DBUS_ID_EBI], | ||||
| 				new->core_clk_rate, stop_req, | ||||
| 				update_bus, update_clk); | ||||
| 
 | ||||
| 	for (i = 0; i < DPU_POWER_HANDLE_DBUS_ID_MAX; i++) { | ||||
| 		if (update_bus & BIT(i)) { | ||||
| 			ret = _dpu_core_perf_crtc_update_bus(kms, crtc, i); | ||||
| 			if (ret) { | ||||
| 				DPU_ERROR("crtc-%d: failed to update bw vote for bus-%d\n", | ||||
| 					  crtc->base.id, i); | ||||
| 				return ret; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Update the clock after bandwidth vote to ensure | ||||
| 	 * bandwidth is available before clock rate is increased. | ||||
| 	 */ | ||||
| 	if (update_clk) { | ||||
| 		clk_rate = _dpu_core_perf_get_core_clk_rate(kms); | ||||
| 
 | ||||
| 		trace_dpu_core_perf_update_clk(kms->dev, stop_req, clk_rate); | ||||
| 
 | ||||
| 		ret = _dpu_core_perf_set_core_clk_rate(kms, clk_rate); | ||||
| 		if (ret) { | ||||
| 			DPU_ERROR("failed to set %s clock rate %llu\n", | ||||
| 					kms->perf.core_clk->clk_name, clk_rate); | ||||
| 			return ret; | ||||
| 		} | ||||
| 
 | ||||
| 		kms->perf.core_clk_rate = clk_rate; | ||||
| 		DPU_DEBUG("update clk rate = %lld HZ\n", clk_rate); | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| #ifdef CONFIG_DEBUG_FS | ||||
| 
 | ||||
| static ssize_t _dpu_core_perf_mode_write(struct file *file, | ||||
| 		    const char __user *user_buf, size_t count, loff_t *ppos) | ||||
| { | ||||
| 	struct dpu_core_perf *perf = file->private_data; | ||||
| 	struct dpu_perf_cfg *cfg = &perf->catalog->perf; | ||||
| 	u32 perf_mode = 0; | ||||
| 	char buf[10]; | ||||
| 
 | ||||
| 	if (!perf) | ||||
| 		return -ENODEV; | ||||
| 
 | ||||
| 	if (count >= sizeof(buf)) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	if (copy_from_user(buf, user_buf, count)) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	buf[count] = 0;	/* end of string */ | ||||
| 
 | ||||
| 	if (kstrtouint(buf, 0, &perf_mode)) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	if (perf_mode >= DPU_PERF_MODE_MAX) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	if (perf_mode == DPU_PERF_MODE_FIXED) { | ||||
| 		DRM_INFO("fix performance mode\n"); | ||||
| 	} else if (perf_mode == DPU_PERF_MODE_MINIMUM) { | ||||
| 		/* run the driver with max clk and BW vote */ | ||||
| 		perf->perf_tune.min_core_clk = perf->max_core_clk_rate; | ||||
| 		perf->perf_tune.min_bus_vote = | ||||
| 				(u64) cfg->max_bw_high * 1000; | ||||
| 		DRM_INFO("minimum performance mode\n"); | ||||
| 	} else if (perf_mode == DPU_PERF_MODE_NORMAL) { | ||||
| 		/* reset the perf tune params to 0 */ | ||||
| 		perf->perf_tune.min_core_clk = 0; | ||||
| 		perf->perf_tune.min_bus_vote = 0; | ||||
| 		DRM_INFO("normal performance mode\n"); | ||||
| 	} | ||||
| 	perf->perf_tune.mode = perf_mode; | ||||
| 
 | ||||
| 	return count; | ||||
| } | ||||
| 
 | ||||
| static ssize_t _dpu_core_perf_mode_read(struct file *file, | ||||
| 			char __user *buff, size_t count, loff_t *ppos) | ||||
| { | ||||
| 	struct dpu_core_perf *perf = file->private_data; | ||||
| 	int len = 0; | ||||
| 	char buf[DPU_PERF_MODE_STRING_SIZE] = {'\0'}; | ||||
| 
 | ||||
| 	if (!perf) | ||||
| 		return -ENODEV; | ||||
| 
 | ||||
| 	if (*ppos) | ||||
| 		return 0;	/* the end */ | ||||
| 
 | ||||
| 	len = snprintf(buf, sizeof(buf), | ||||
| 			"mode %d min_mdp_clk %llu min_bus_vote %llu\n", | ||||
| 			perf->perf_tune.mode, | ||||
| 			perf->perf_tune.min_core_clk, | ||||
| 			perf->perf_tune.min_bus_vote); | ||||
| 	if (len < 0 || len >= sizeof(buf)) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	if ((count < sizeof(buf)) || copy_to_user(buff, buf, len)) | ||||
| 		return -EFAULT; | ||||
| 
 | ||||
| 	*ppos += len;   /* increase offset */ | ||||
| 
 | ||||
| 	return len; | ||||
| } | ||||
| 
 | ||||
| static const struct file_operations dpu_core_perf_mode_fops = { | ||||
| 	.open = simple_open, | ||||
| 	.read = _dpu_core_perf_mode_read, | ||||
| 	.write = _dpu_core_perf_mode_write, | ||||
| }; | ||||
| 
 | ||||
| static void dpu_core_perf_debugfs_destroy(struct dpu_core_perf *perf) | ||||
| { | ||||
| 	debugfs_remove_recursive(perf->debugfs_root); | ||||
| 	perf->debugfs_root = NULL; | ||||
| } | ||||
| 
 | ||||
| int dpu_core_perf_debugfs_init(struct dpu_core_perf *perf, | ||||
| 		struct dentry *parent) | ||||
| { | ||||
| 	struct dpu_mdss_cfg *catalog = perf->catalog; | ||||
| 	struct msm_drm_private *priv; | ||||
| 	struct dpu_kms *dpu_kms; | ||||
| 
 | ||||
| 	priv = perf->dev->dev_private; | ||||
| 	if (!priv || !priv->kms) { | ||||
| 		DPU_ERROR("invalid KMS reference\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_kms = to_dpu_kms(priv->kms); | ||||
| 
 | ||||
| 	perf->debugfs_root = debugfs_create_dir("core_perf", parent); | ||||
| 	if (!perf->debugfs_root) { | ||||
| 		DPU_ERROR("failed to create core perf debugfs\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	debugfs_create_u64("max_core_clk_rate", 0600, perf->debugfs_root, | ||||
| 			&perf->max_core_clk_rate); | ||||
| 	debugfs_create_u64("core_clk_rate", 0600, perf->debugfs_root, | ||||
| 			&perf->core_clk_rate); | ||||
| 	debugfs_create_u32("enable_bw_release", 0600, perf->debugfs_root, | ||||
| 			(u32 *)&perf->enable_bw_release); | ||||
| 	debugfs_create_u32("threshold_low", 0600, perf->debugfs_root, | ||||
| 			(u32 *)&catalog->perf.max_bw_low); | ||||
| 	debugfs_create_u32("threshold_high", 0600, perf->debugfs_root, | ||||
| 			(u32 *)&catalog->perf.max_bw_high); | ||||
| 	debugfs_create_u32("min_core_ib", 0600, perf->debugfs_root, | ||||
| 			(u32 *)&catalog->perf.min_core_ib); | ||||
| 	debugfs_create_u32("min_llcc_ib", 0600, perf->debugfs_root, | ||||
| 			(u32 *)&catalog->perf.min_llcc_ib); | ||||
| 	debugfs_create_u32("min_dram_ib", 0600, perf->debugfs_root, | ||||
| 			(u32 *)&catalog->perf.min_dram_ib); | ||||
| 	debugfs_create_file("perf_mode", 0600, perf->debugfs_root, | ||||
| 			(u32 *)perf, &dpu_core_perf_mode_fops); | ||||
| 	debugfs_create_u64("fix_core_clk_rate", 0600, perf->debugfs_root, | ||||
| 			&perf->fix_core_clk_rate); | ||||
| 	debugfs_create_u64("fix_core_ib_vote", 0600, perf->debugfs_root, | ||||
| 			&perf->fix_core_ib_vote); | ||||
| 	debugfs_create_u64("fix_core_ab_vote", 0600, perf->debugfs_root, | ||||
| 			&perf->fix_core_ab_vote); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| #else | ||||
| static void dpu_core_perf_debugfs_destroy(struct dpu_core_perf *perf) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| int dpu_core_perf_debugfs_init(struct dpu_core_perf *perf, | ||||
| 		struct dentry *parent) | ||||
| { | ||||
| 	return 0; | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| void dpu_core_perf_destroy(struct dpu_core_perf *perf) | ||||
| { | ||||
| 	if (!perf) { | ||||
| 		DPU_ERROR("invalid parameters\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_core_perf_debugfs_destroy(perf); | ||||
| 	perf->max_core_clk_rate = 0; | ||||
| 	perf->core_clk = NULL; | ||||
| 	perf->phandle = NULL; | ||||
| 	perf->catalog = NULL; | ||||
| 	perf->dev = NULL; | ||||
| } | ||||
| 
 | ||||
| int dpu_core_perf_init(struct dpu_core_perf *perf, | ||||
| 		struct drm_device *dev, | ||||
| 		struct dpu_mdss_cfg *catalog, | ||||
| 		struct dpu_power_handle *phandle, | ||||
| 		struct dss_clk *core_clk) | ||||
| { | ||||
| 	perf->dev = dev; | ||||
| 	perf->catalog = catalog; | ||||
| 	perf->phandle = phandle; | ||||
| 	perf->core_clk = core_clk; | ||||
| 
 | ||||
| 	perf->max_core_clk_rate = core_clk->max_rate; | ||||
| 	if (!perf->max_core_clk_rate) { | ||||
| 		DPU_DEBUG("optional max core clk rate, use default\n"); | ||||
| 		perf->max_core_clk_rate = DPU_PERF_DEFAULT_MAX_CORE_CLK_RATE; | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
							
								
								
									
										133
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,133 @@ | ||||
| /* Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_CORE_PERF_H_ | ||||
| #define _DPU_CORE_PERF_H_ | ||||
| 
 | ||||
| #include <linux/types.h> | ||||
| #include <linux/dcache.h> | ||||
| #include <linux/mutex.h> | ||||
| #include <drm/drm_crtc.h> | ||||
| 
 | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_power_handle.h" | ||||
| 
 | ||||
| #define	DPU_PERF_DEFAULT_MAX_CORE_CLK_RATE	412500000 | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_core_perf_params - definition of performance parameters | ||||
|  * @max_per_pipe_ib: maximum instantaneous bandwidth request | ||||
|  * @bw_ctl: arbitrated bandwidth request | ||||
|  * @core_clk_rate: core clock rate request | ||||
|  */ | ||||
| struct dpu_core_perf_params { | ||||
| 	u64 max_per_pipe_ib[DPU_POWER_HANDLE_DBUS_ID_MAX]; | ||||
| 	u64 bw_ctl[DPU_POWER_HANDLE_DBUS_ID_MAX]; | ||||
| 	u64 core_clk_rate; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_core_perf_tune - definition of performance tuning control | ||||
|  * @mode: performance mode | ||||
|  * @min_core_clk: minimum core clock | ||||
|  * @min_bus_vote: minimum bus vote | ||||
|  */ | ||||
| struct dpu_core_perf_tune { | ||||
| 	u32 mode; | ||||
| 	u64 min_core_clk; | ||||
| 	u64 min_bus_vote; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_core_perf - definition of core performance context | ||||
|  * @dev: Pointer to drm device | ||||
|  * @debugfs_root: top level debug folder | ||||
|  * @catalog: Pointer to catalog configuration | ||||
|  * @phandle: Pointer to power handler | ||||
|  * @core_clk: Pointer to core clock structure | ||||
|  * @core_clk_rate: current core clock rate | ||||
|  * @max_core_clk_rate: maximum allowable core clock rate | ||||
|  * @perf_tune: debug control for performance tuning | ||||
|  * @enable_bw_release: debug control for bandwidth release | ||||
|  * @fix_core_clk_rate: fixed core clock request in Hz used in mode 2 | ||||
|  * @fix_core_ib_vote: fixed core ib vote in bps used in mode 2 | ||||
|  * @fix_core_ab_vote: fixed core ab vote in bps used in mode 2 | ||||
|  */ | ||||
| struct dpu_core_perf { | ||||
| 	struct drm_device *dev; | ||||
| 	struct dentry *debugfs_root; | ||||
| 	struct dpu_mdss_cfg *catalog; | ||||
| 	struct dpu_power_handle *phandle; | ||||
| 	struct dss_clk *core_clk; | ||||
| 	u64 core_clk_rate; | ||||
| 	u64 max_core_clk_rate; | ||||
| 	struct dpu_core_perf_tune perf_tune; | ||||
| 	u32 enable_bw_release; | ||||
| 	u64 fix_core_clk_rate; | ||||
| 	u64 fix_core_ib_vote; | ||||
| 	u64 fix_core_ab_vote; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_perf_crtc_check - validate performance of the given crtc state | ||||
|  * @crtc: Pointer to crtc | ||||
|  * @state: Pointer to new crtc state | ||||
|  * return: zero if success, or error code otherwise | ||||
|  */ | ||||
| int dpu_core_perf_crtc_check(struct drm_crtc *crtc, | ||||
| 		struct drm_crtc_state *state); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_perf_crtc_update - update performance of the given crtc | ||||
|  * @crtc: Pointer to crtc | ||||
|  * @params_changed: true if crtc parameters are modified | ||||
|  * @stop_req: true if this is a stop request | ||||
|  * return: zero if success, or error code otherwise | ||||
|  */ | ||||
| int dpu_core_perf_crtc_update(struct drm_crtc *crtc, | ||||
| 		int params_changed, bool stop_req); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_perf_crtc_release_bw - release bandwidth of the given crtc | ||||
|  * @crtc: Pointer to crtc | ||||
|  */ | ||||
| void dpu_core_perf_crtc_release_bw(struct drm_crtc *crtc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_perf_destroy - destroy the given core performance context | ||||
|  * @perf: Pointer to core performance context | ||||
|  */ | ||||
| void dpu_core_perf_destroy(struct dpu_core_perf *perf); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_perf_init - initialize the given core performance context | ||||
|  * @perf: Pointer to core performance context | ||||
|  * @dev: Pointer to drm device | ||||
|  * @catalog: Pointer to catalog | ||||
|  * @phandle: Pointer to power handle | ||||
|  * @core_clk: pointer to core clock | ||||
|  */ | ||||
| int dpu_core_perf_init(struct dpu_core_perf *perf, | ||||
| 		struct drm_device *dev, | ||||
| 		struct dpu_mdss_cfg *catalog, | ||||
| 		struct dpu_power_handle *phandle, | ||||
| 		struct dss_clk *core_clk); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_core_perf_debugfs_init - initialize debugfs for core performance context | ||||
|  * @perf: Pointer to core performance context | ||||
|  * @debugfs_parent: Pointer to parent debugfs | ||||
|  */ | ||||
| int dpu_core_perf_debugfs_init(struct dpu_core_perf *perf, | ||||
| 		struct dentry *parent); | ||||
| 
 | ||||
| #endif /* _DPU_CORE_PERF_H_ */ | ||||
							
								
								
									
										2504
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2504
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										484
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										484
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,484 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2015-2018 The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU General Public License version 2 as published by | ||||
|  * the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License along with | ||||
|  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_CRTC_H_ | ||||
| #define _DPU_CRTC_H_ | ||||
| 
 | ||||
| #include <linux/kthread.h> | ||||
| #include <drm/drm_crtc.h> | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_core_perf.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| #define DPU_CRTC_NAME_SIZE	12 | ||||
| 
 | ||||
| /* define the maximum number of in-flight frame events */ | ||||
| #define DPU_CRTC_FRAME_EVENT_SIZE	4 | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_crtc_client_type: crtc client type | ||||
|  * @RT_CLIENT:	RealTime client like video/cmd mode display | ||||
|  *              voting through apps rsc | ||||
|  * @NRT_CLIENT:	Non-RealTime client like WB display | ||||
|  *              voting through apps rsc | ||||
|  */ | ||||
| enum dpu_crtc_client_type { | ||||
| 	RT_CLIENT, | ||||
| 	NRT_CLIENT, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_crtc_smmu_state:	smmu state | ||||
|  * @ATTACHED:	 all the context banks are attached. | ||||
|  * @DETACHED:	 all the context banks are detached. | ||||
|  * @ATTACH_ALL_REQ:	 transient state of attaching context banks. | ||||
|  * @DETACH_ALL_REQ:	 transient state of detaching context banks. | ||||
|  */ | ||||
| enum dpu_crtc_smmu_state { | ||||
| 	ATTACHED = 0, | ||||
| 	DETACHED, | ||||
| 	ATTACH_ALL_REQ, | ||||
| 	DETACH_ALL_REQ, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_crtc_smmu_state_transition_type: state transition type | ||||
|  * @NONE: no pending state transitions | ||||
|  * @PRE_COMMIT: state transitions should be done before processing the commit | ||||
|  * @POST_COMMIT: state transitions to be done after processing the commit. | ||||
|  */ | ||||
| enum dpu_crtc_smmu_state_transition_type { | ||||
| 	NONE, | ||||
| 	PRE_COMMIT, | ||||
| 	POST_COMMIT | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_smmu_state_data: stores the smmu state and transition type | ||||
|  * @state: current state of smmu context banks | ||||
|  * @transition_type: transition request type | ||||
|  * @transition_error: whether there is error while transitioning the state | ||||
|  */ | ||||
| struct dpu_crtc_smmu_state_data { | ||||
| 	uint32_t state; | ||||
| 	uint32_t transition_type; | ||||
| 	uint32_t transition_error; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_mixer: stores the map for each virtual pipeline in the CRTC | ||||
|  * @hw_lm:	LM HW Driver context | ||||
|  * @hw_ctl:	CTL Path HW driver context | ||||
|  * @encoder:	Encoder attached to this lm & ctl | ||||
|  * @mixer_op_mode:	mixer blending operation mode | ||||
|  * @flush_mask:	mixer flush mask for ctl, mixer and pipe | ||||
|  */ | ||||
| struct dpu_crtc_mixer { | ||||
| 	struct dpu_hw_mixer *hw_lm; | ||||
| 	struct dpu_hw_ctl *hw_ctl; | ||||
| 	struct drm_encoder *encoder; | ||||
| 	u32 mixer_op_mode; | ||||
| 	u32 flush_mask; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_frame_event: stores crtc frame event for crtc processing | ||||
|  * @work:	base work structure | ||||
|  * @crtc:	Pointer to crtc handling this event | ||||
|  * @list:	event list | ||||
|  * @ts:		timestamp at queue entry | ||||
|  * @event:	event identifier | ||||
|  */ | ||||
| struct dpu_crtc_frame_event { | ||||
| 	struct kthread_work work; | ||||
| 	struct drm_crtc *crtc; | ||||
| 	struct list_head list; | ||||
| 	ktime_t ts; | ||||
| 	u32 event; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_event - event callback tracking structure | ||||
|  * @list:     Linked list tracking node | ||||
|  * @kt_work:  Kthread worker structure | ||||
|  * @dpu_crtc: Pointer to associated dpu_crtc structure | ||||
|  * @cb_func:  Pointer to callback function | ||||
|  * @usr:      Pointer to user data to be provided to the callback | ||||
|  */ | ||||
| struct dpu_crtc_event { | ||||
| 	struct list_head list; | ||||
| 	struct kthread_work kt_work; | ||||
| 	void *dpu_crtc; | ||||
| 
 | ||||
| 	void (*cb_func)(struct drm_crtc *crtc, void *usr); | ||||
| 	void *usr; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * Maximum number of free event structures to cache | ||||
|  */ | ||||
| #define DPU_CRTC_MAX_EVENT_COUNT	16 | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc - virtualized CRTC data structure | ||||
|  * @base          : Base drm crtc structure | ||||
|  * @name          : ASCII description of this crtc | ||||
|  * @num_ctls      : Number of ctl paths in use | ||||
|  * @num_mixers    : Number of mixers in use | ||||
|  * @mixers_swapped: Whether the mixers have been swapped for left/right update | ||||
|  *                  especially in the case of DSC Merge. | ||||
|  * @mixers        : List of active mixers | ||||
|  * @event         : Pointer to last received drm vblank event. If there is a | ||||
|  *                  pending vblank event, this will be non-null. | ||||
|  * @vsync_count   : Running count of received vsync events | ||||
|  * @drm_requested_vblank : Whether vblanks have been enabled in the encoder | ||||
|  * @property_info : Opaque structure for generic property support | ||||
|  * @property_defaults : Array of default values for generic property support | ||||
|  * @stage_cfg     : H/w mixer stage configuration | ||||
|  * @debugfs_root  : Parent of debugfs node | ||||
|  * @vblank_cb_count : count of vblank callback since last reset | ||||
|  * @play_count    : frame count between crtc enable and disable | ||||
|  * @vblank_cb_time  : ktime at vblank count reset | ||||
|  * @vblank_requested : whether the user has requested vblank events | ||||
|  * @suspend         : whether or not a suspend operation is in progress | ||||
|  * @enabled       : whether the DPU CRTC is currently enabled. updated in the | ||||
|  *                  commit-thread, not state-swap time which is earlier, so | ||||
|  *                  safe to make decisions on during VBLANK on/off work | ||||
|  * @feature_list  : list of color processing features supported on a crtc | ||||
|  * @active_list   : list of color processing features are active | ||||
|  * @dirty_list    : list of color processing features are dirty | ||||
|  * @ad_dirty: list containing ad properties that are dirty | ||||
|  * @ad_active: list containing ad properties that are active | ||||
|  * @crtc_lock     : crtc lock around create, destroy and access. | ||||
|  * @frame_pending : Whether or not an update is pending | ||||
|  * @frame_events  : static allocation of in-flight frame events | ||||
|  * @frame_event_list : available frame event list | ||||
|  * @spin_lock     : spin lock for frame event, transaction status, etc... | ||||
|  * @frame_done_comp    : for frame_event_done synchronization | ||||
|  * @event_thread  : Pointer to event handler thread | ||||
|  * @event_worker  : Event worker queue | ||||
|  * @event_cache   : Local cache of event worker structures | ||||
|  * @event_free_list : List of available event structures | ||||
|  * @event_lock    : Spinlock around event handling code | ||||
|  * @misr_enable   : boolean entry indicates misr enable/disable status. | ||||
|  * @misr_frame_count  : misr frame count provided by client | ||||
|  * @misr_data     : store misr data before turning off the clocks. | ||||
|  * @phandle: Pointer to power handler | ||||
|  * @power_event   : registered power event handle | ||||
|  * @cur_perf      : current performance committed to clock/bandwidth driver | ||||
|  * @rp_lock       : serialization lock for resource pool | ||||
|  * @rp_head       : list of active resource pool | ||||
|  * @scl3_cfg_lut  : qseed3 lut config | ||||
|  */ | ||||
| struct dpu_crtc { | ||||
| 	struct drm_crtc base; | ||||
| 	char name[DPU_CRTC_NAME_SIZE]; | ||||
| 
 | ||||
| 	/* HW Resources reserved for the crtc */ | ||||
| 	u32 num_ctls; | ||||
| 	u32 num_mixers; | ||||
| 	bool mixers_swapped; | ||||
| 	struct dpu_crtc_mixer mixers[CRTC_DUAL_MIXERS]; | ||||
| 	struct dpu_hw_scaler3_lut_cfg *scl3_lut_cfg; | ||||
| 
 | ||||
| 	struct drm_pending_vblank_event *event; | ||||
| 	u32 vsync_count; | ||||
| 
 | ||||
| 	struct dpu_hw_stage_cfg stage_cfg; | ||||
| 	struct dentry *debugfs_root; | ||||
| 
 | ||||
| 	u32 vblank_cb_count; | ||||
| 	u64 play_count; | ||||
| 	ktime_t vblank_cb_time; | ||||
| 	bool vblank_requested; | ||||
| 	bool suspend; | ||||
| 	bool enabled; | ||||
| 
 | ||||
| 	struct list_head feature_list; | ||||
| 	struct list_head active_list; | ||||
| 	struct list_head dirty_list; | ||||
| 	struct list_head ad_dirty; | ||||
| 	struct list_head ad_active; | ||||
| 
 | ||||
| 	struct mutex crtc_lock; | ||||
| 
 | ||||
| 	atomic_t frame_pending; | ||||
| 	struct dpu_crtc_frame_event frame_events[DPU_CRTC_FRAME_EVENT_SIZE]; | ||||
| 	struct list_head frame_event_list; | ||||
| 	spinlock_t spin_lock; | ||||
| 	struct completion frame_done_comp; | ||||
| 
 | ||||
| 	/* for handling internal event thread */ | ||||
| 	struct dpu_crtc_event event_cache[DPU_CRTC_MAX_EVENT_COUNT]; | ||||
| 	struct list_head event_free_list; | ||||
| 	spinlock_t event_lock; | ||||
| 	bool misr_enable; | ||||
| 	u32 misr_frame_count; | ||||
| 	u32 misr_data[CRTC_DUAL_MIXERS]; | ||||
| 
 | ||||
| 	struct dpu_power_handle *phandle; | ||||
| 	struct dpu_power_event *power_event; | ||||
| 
 | ||||
| 	struct dpu_core_perf_params cur_perf; | ||||
| 
 | ||||
| 	struct mutex rp_lock; | ||||
| 	struct list_head rp_head; | ||||
| 
 | ||||
| 	struct dpu_crtc_smmu_state_data smmu_state; | ||||
| }; | ||||
| 
 | ||||
| #define to_dpu_crtc(x) container_of(x, struct dpu_crtc, base) | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_res_ops - common operations for crtc resources | ||||
|  * @get: get given resource | ||||
|  * @put: put given resource | ||||
|  */ | ||||
| struct dpu_crtc_res_ops { | ||||
| 	void *(*get)(void *val, u32 type, u64 tag); | ||||
| 	void (*put)(void *val); | ||||
| }; | ||||
| 
 | ||||
| #define DPU_CRTC_RES_FLAG_FREE		BIT(0) | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_res - definition of crtc resources | ||||
|  * @list: list of crtc resource | ||||
|  * @type: crtc resource type | ||||
|  * @tag: unique identifier per type | ||||
|  * @refcount: reference/usage count | ||||
|  * @ops: callback operations | ||||
|  * @val: resource handle associated with type/tag | ||||
|  * @flags: customization flags | ||||
|  */ | ||||
| struct dpu_crtc_res { | ||||
| 	struct list_head list; | ||||
| 	u32 type; | ||||
| 	u64 tag; | ||||
| 	atomic_t refcount; | ||||
| 	struct dpu_crtc_res_ops ops; | ||||
| 	void *val; | ||||
| 	u32 flags; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_respool - crtc resource pool | ||||
|  * @rp_lock: pointer to serialization lock | ||||
|  * @rp_head: pointer to head of active resource pools of this crtc | ||||
|  * @rp_list: list of crtc resource pool | ||||
|  * @sequence_id: sequence identifier, incremented per state duplication | ||||
|  * @res_list: list of resource managed by this resource pool | ||||
|  * @ops: resource operations for parent resource pool | ||||
|  */ | ||||
| struct dpu_crtc_respool { | ||||
| 	struct mutex *rp_lock; | ||||
| 	struct list_head *rp_head; | ||||
| 	struct list_head rp_list; | ||||
| 	u32 sequence_id; | ||||
| 	struct list_head res_list; | ||||
| 	struct dpu_crtc_res_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_crtc_state - dpu container for atomic crtc state | ||||
|  * @base: Base drm crtc state structure | ||||
|  * @is_ppsplit    : Whether current topology requires PPSplit special handling | ||||
|  * @bw_control    : true if bw/clk controlled by core bw/clk properties | ||||
|  * @bw_split_vote : true if bw controlled by llcc/dram bw properties | ||||
|  * @lm_bounds     : LM boundaries based on current mode full resolution, no ROI. | ||||
|  *                  Origin top left of CRTC. | ||||
|  * @property_state: Local storage for msm_prop properties | ||||
|  * @property_values: Current crtc property values | ||||
|  * @input_fence_timeout_ns : Cached input fence timeout, in ns | ||||
|  * @new_perf: new performance state being requested | ||||
|  */ | ||||
| struct dpu_crtc_state { | ||||
| 	struct drm_crtc_state base; | ||||
| 
 | ||||
| 	bool bw_control; | ||||
| 	bool bw_split_vote; | ||||
| 
 | ||||
| 	bool is_ppsplit; | ||||
| 	struct drm_rect lm_bounds[CRTC_DUAL_MIXERS]; | ||||
| 
 | ||||
| 	uint64_t input_fence_timeout_ns; | ||||
| 
 | ||||
| 	struct dpu_core_perf_params new_perf; | ||||
| 	struct dpu_crtc_respool rp; | ||||
| }; | ||||
| 
 | ||||
| #define to_dpu_crtc_state(x) \ | ||||
| 	container_of(x, struct dpu_crtc_state, base) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_get_mixer_width - get the mixer width | ||||
|  * Mixer width will be same as panel width(/2 for split) | ||||
|  */ | ||||
| static inline int dpu_crtc_get_mixer_width(struct dpu_crtc *dpu_crtc, | ||||
| 	struct dpu_crtc_state *cstate, struct drm_display_mode *mode) | ||||
| { | ||||
| 	u32 mixer_width; | ||||
| 
 | ||||
| 	if (!dpu_crtc || !cstate || !mode) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	mixer_width = (dpu_crtc->num_mixers == CRTC_DUAL_MIXERS ? | ||||
| 			mode->hdisplay / CRTC_DUAL_MIXERS : mode->hdisplay); | ||||
| 
 | ||||
| 	return mixer_width; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_get_mixer_height - get the mixer height | ||||
|  * Mixer height will be same as panel height | ||||
|  */ | ||||
| static inline int dpu_crtc_get_mixer_height(struct dpu_crtc *dpu_crtc, | ||||
| 		struct dpu_crtc_state *cstate, struct drm_display_mode *mode) | ||||
| { | ||||
| 	if (!dpu_crtc || !cstate || !mode) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	return mode->vdisplay; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_frame_pending - retun the number of pending frames | ||||
|  * @crtc: Pointer to drm crtc object | ||||
|  */ | ||||
| static inline int dpu_crtc_frame_pending(struct drm_crtc *crtc) | ||||
| { | ||||
| 	struct dpu_crtc *dpu_crtc; | ||||
| 
 | ||||
| 	if (!crtc) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	dpu_crtc = to_dpu_crtc(crtc); | ||||
| 	return atomic_read(&dpu_crtc->frame_pending); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_vblank - enable or disable vblanks for this crtc | ||||
|  * @crtc: Pointer to drm crtc object | ||||
|  * @en: true to enable vblanks, false to disable | ||||
|  */ | ||||
| int dpu_crtc_vblank(struct drm_crtc *crtc, bool en); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_commit_kickoff - trigger kickoff of the commit for this crtc | ||||
|  * @crtc: Pointer to drm crtc object | ||||
|  */ | ||||
| void dpu_crtc_commit_kickoff(struct drm_crtc *crtc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_complete_commit - callback signalling completion of current commit | ||||
|  * @crtc: Pointer to drm crtc object | ||||
|  * @old_state: Pointer to drm crtc old state object | ||||
|  */ | ||||
| void dpu_crtc_complete_commit(struct drm_crtc *crtc, | ||||
| 		struct drm_crtc_state *old_state); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_init - create a new crtc object | ||||
|  * @dev: dpu device | ||||
|  * @plane: base plane | ||||
|  * @Return: new crtc object or error | ||||
|  */ | ||||
| struct drm_crtc *dpu_crtc_init(struct drm_device *dev, struct drm_plane *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_register_custom_event - api for enabling/disabling crtc event | ||||
|  * @kms: Pointer to dpu_kms | ||||
|  * @crtc_drm: Pointer to crtc object | ||||
|  * @event: Event that client is interested | ||||
|  * @en: Flag to enable/disable the event | ||||
|  */ | ||||
| int dpu_crtc_register_custom_event(struct dpu_kms *kms, | ||||
| 		struct drm_crtc *crtc_drm, u32 event, bool en); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_get_intf_mode - get interface mode of the given crtc | ||||
|  * @crtc: Pointert to crtc | ||||
|  */ | ||||
| enum dpu_intf_mode dpu_crtc_get_intf_mode(struct drm_crtc *crtc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_get_client_type - check the crtc type- rt, nrt etc. | ||||
|  * @crtc: Pointer to crtc | ||||
|  */ | ||||
| static inline enum dpu_crtc_client_type dpu_crtc_get_client_type( | ||||
| 						struct drm_crtc *crtc) | ||||
| { | ||||
| 	struct dpu_crtc_state *cstate = | ||||
| 			crtc ? to_dpu_crtc_state(crtc->state) : NULL; | ||||
| 
 | ||||
| 	if (!cstate) | ||||
| 		return NRT_CLIENT; | ||||
| 
 | ||||
| 	return RT_CLIENT; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_is_enabled - check if dpu crtc is enabled or not | ||||
|  * @crtc: Pointer to crtc | ||||
|  */ | ||||
| static inline bool dpu_crtc_is_enabled(struct drm_crtc *crtc) | ||||
| { | ||||
| 	return crtc ? crtc->enabled : false; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_event_queue - request event callback | ||||
|  * @crtc: Pointer to drm crtc structure | ||||
|  * @func: Pointer to callback function | ||||
|  * @usr: Pointer to user data to be passed to callback | ||||
|  * Returns: Zero on success | ||||
|  */ | ||||
| int dpu_crtc_event_queue(struct drm_crtc *crtc, | ||||
| 		void (*func)(struct drm_crtc *crtc, void *usr), void *usr); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_res_add - add given resource to resource pool in crtc state | ||||
|  * @state: Pointer to drm crtc state | ||||
|  * @type: Resource type | ||||
|  * @tag: Search tag for given resource | ||||
|  * @val: Resource handle | ||||
|  * @ops: Resource callback operations | ||||
|  * return: 0 if success; error code otherwise | ||||
|  */ | ||||
| int dpu_crtc_res_add(struct drm_crtc_state *state, u32 type, u64 tag, | ||||
| 		void *val, struct dpu_crtc_res_ops *ops); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_res_get - get given resource from resource pool in crtc state | ||||
|  * @state: Pointer to drm crtc state | ||||
|  * @type: Resource type | ||||
|  * @tag: Search tag for given resource | ||||
|  * return: Resource handle if success; pointer error or null otherwise | ||||
|  */ | ||||
| void *dpu_crtc_res_get(struct drm_crtc_state *state, u32 type, u64 tag); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_crtc_res_put - return given resource to resource pool in crtc state | ||||
|  * @state: Pointer to drm crtc state | ||||
|  * @type: Resource type | ||||
|  * @tag: Search tag for given resource | ||||
|  * return: None | ||||
|  */ | ||||
| void dpu_crtc_res_put(struct drm_crtc_state *state, u32 type, u64 tag); | ||||
| 
 | ||||
| #endif /* _DPU_CRTC_H_ */ | ||||
							
								
								
									
										2393
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_dbg.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2393
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_dbg.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										103
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_dbg.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_dbg.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,103 @@ | ||||
| /* Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef DPU_DBG_H_ | ||||
| #define DPU_DBG_H_ | ||||
| 
 | ||||
| #include <stdarg.h> | ||||
| #include <linux/debugfs.h> | ||||
| #include <linux/list.h> | ||||
| 
 | ||||
| enum dpu_dbg_dump_flag { | ||||
| 	DPU_DBG_DUMP_IN_LOG = BIT(0), | ||||
| 	DPU_DBG_DUMP_IN_MEM = BIT(1), | ||||
| }; | ||||
| 
 | ||||
| #if defined(CONFIG_DEBUG_FS) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_dbg_init_dbg_buses - initialize debug bus dumping support for the chipset | ||||
|  * @hwversion:		Chipset revision | ||||
|  */ | ||||
| void dpu_dbg_init_dbg_buses(u32 hwversion); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_dbg_init - initialize global dpu debug facilities: regdump | ||||
|  * @dev:		device handle | ||||
|  * Returns:		0 or -ERROR | ||||
|  */ | ||||
| int dpu_dbg_init(struct device *dev); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_dbg_debugfs_register - register entries at the given debugfs dir | ||||
|  * @debugfs_root:	debugfs root in which to create dpu debug entries | ||||
|  * Returns:	0 or -ERROR | ||||
|  */ | ||||
| int dpu_dbg_debugfs_register(struct dentry *debugfs_root); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_dbg_destroy - destroy the global dpu debug facilities | ||||
|  * Returns:	none | ||||
|  */ | ||||
| void dpu_dbg_destroy(void); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_dbg_dump - trigger dumping of all dpu_dbg facilities | ||||
|  * @queue_work:	  whether to queue the dumping work to the work_struct | ||||
|  * @name:	  string indicating origin of dump | ||||
|  * @dump_dbgbus:  dump the dpu debug bus | ||||
|  * @dump_vbif_rt: dump the vbif rt bus | ||||
|  * Returns:	none | ||||
|  */ | ||||
| void dpu_dbg_dump(bool queue_work, const char *name, bool dump_dbgbus_dpu, | ||||
| 		  bool dump_dbgbus_vbif_rt); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_dbg_set_dpu_top_offset - set the target specific offset from mdss base | ||||
|  *	address of the top registers. Used for accessing debug bus controls. | ||||
|  * @blk_off: offset from mdss base of the top block | ||||
|  */ | ||||
| void dpu_dbg_set_dpu_top_offset(u32 blk_off); | ||||
| 
 | ||||
| #else | ||||
| 
 | ||||
| static inline void dpu_dbg_init_dbg_buses(u32 hwversion) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| static inline int dpu_dbg_init(struct device *dev) | ||||
| { | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static inline int dpu_dbg_debugfs_register(struct dentry *debugfs_root) | ||||
| { | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_dbg_destroy(void) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_dbg_dump(bool queue_work, const char *name, | ||||
| 				bool dump_dbgbus_dpu, bool dump_dbgbus_vbif_rt) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_dbg_set_dpu_top_offset(u32 blk_off) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| #endif /* defined(CONFIG_DEBUG_FS) */ | ||||
| 
 | ||||
| 
 | ||||
| #endif /* DPU_DBG_H_ */ | ||||
							
								
								
									
										2575
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2575
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										191
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,191 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2015-2018, The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU General Public License version 2 as published by | ||||
|  * the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License along with | ||||
|  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_ENCODER_H__ | ||||
| #define __DPU_ENCODER_H__ | ||||
| 
 | ||||
| #include <drm/drm_crtc.h> | ||||
| #include "dpu_hw_mdss.h" | ||||
| 
 | ||||
| #define DPU_ENCODER_FRAME_EVENT_DONE			BIT(0) | ||||
| #define DPU_ENCODER_FRAME_EVENT_ERROR			BIT(1) | ||||
| #define DPU_ENCODER_FRAME_EVENT_PANEL_DEAD		BIT(2) | ||||
| #define DPU_ENCODER_FRAME_EVENT_IDLE			BIT(3) | ||||
| 
 | ||||
| #define IDLE_TIMEOUT	(66 - 16/2) | ||||
| 
 | ||||
| /**
 | ||||
|  * Encoder functions and data types | ||||
|  * @intfs:	Interfaces this encoder is using, INTF_MODE_NONE if unused | ||||
|  * @needs_cdm:	Encoder requests a CDM based on pixel format conversion needs | ||||
|  * @display_num_of_h_tiles: Number of horizontal tiles in case of split | ||||
|  *                          interface | ||||
|  * @topology:   Topology of the display | ||||
|  */ | ||||
| struct dpu_encoder_hw_resources { | ||||
| 	enum dpu_intf_mode intfs[INTF_MAX]; | ||||
| 	bool needs_cdm; | ||||
| 	u32 display_num_of_h_tiles; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_kickoff_params - info encoder requires at kickoff | ||||
|  * @affected_displays:  bitmask, bit set means the ROI of the commit lies within | ||||
|  *                      the bounds of the physical display at the bit index | ||||
|  */ | ||||
| struct dpu_encoder_kickoff_params { | ||||
| 	unsigned long affected_displays; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_get_hw_resources - Populate table of required hardware resources | ||||
|  * @encoder:	encoder pointer | ||||
|  * @hw_res:	resource table to populate with encoder required resources | ||||
|  * @conn_state:	report hw reqs based on this proposed connector state | ||||
|  */ | ||||
| void dpu_encoder_get_hw_resources(struct drm_encoder *encoder, | ||||
| 		struct dpu_encoder_hw_resources *hw_res, | ||||
| 		struct drm_connector_state *conn_state); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_register_vblank_callback - provide callback to encoder that | ||||
|  *	will be called on the next vblank. | ||||
|  * @encoder:	encoder pointer | ||||
|  * @cb:		callback pointer, provide NULL to deregister and disable IRQs | ||||
|  * @data:	user data provided to callback | ||||
|  */ | ||||
| void dpu_encoder_register_vblank_callback(struct drm_encoder *encoder, | ||||
| 		void (*cb)(void *), void *data); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_register_frame_event_callback - provide callback to encoder that | ||||
|  *	will be called after the request is complete, or other events. | ||||
|  * @encoder:	encoder pointer | ||||
|  * @cb:		callback pointer, provide NULL to deregister | ||||
|  * @data:	user data provided to callback | ||||
|  */ | ||||
| void dpu_encoder_register_frame_event_callback(struct drm_encoder *encoder, | ||||
| 		void (*cb)(void *, u32), void *data); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_prepare_for_kickoff - schedule double buffer flip of the ctl | ||||
|  *	path (i.e. ctl flush and start) at next appropriate time. | ||||
|  *	Immediately: if no previous commit is outstanding. | ||||
|  *	Delayed: Block until next trigger can be issued. | ||||
|  * @encoder:	encoder pointer | ||||
|  * @params:	kickoff time parameters | ||||
|  */ | ||||
| void dpu_encoder_prepare_for_kickoff(struct drm_encoder *encoder, | ||||
| 		struct dpu_encoder_kickoff_params *params); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_trigger_kickoff_pending - Clear the flush bits from previous | ||||
|  *        kickoff and trigger the ctl prepare progress for command mode display. | ||||
|  * @encoder:	encoder pointer | ||||
|  */ | ||||
| void dpu_encoder_trigger_kickoff_pending(struct drm_encoder *encoder); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_kickoff - trigger a double buffer flip of the ctl path | ||||
|  *	(i.e. ctl flush and start) immediately. | ||||
|  * @encoder:	encoder pointer | ||||
|  */ | ||||
| void dpu_encoder_kickoff(struct drm_encoder *encoder); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_wait_for_event - Waits for encoder events | ||||
|  * @encoder:	encoder pointer | ||||
|  * @event:      event to wait for | ||||
|  * MSM_ENC_COMMIT_DONE -  Wait for hardware to have flushed the current pending | ||||
|  *                        frames to hardware at a vblank or ctl_start | ||||
|  *                        Encoders will map this differently depending on the | ||||
|  *                        panel type. | ||||
|  *	                  vid mode -> vsync_irq | ||||
|  *                        cmd mode -> ctl_start | ||||
|  * MSM_ENC_TX_COMPLETE -  Wait for the hardware to transfer all the pixels to | ||||
|  *                        the panel. Encoders will map this differently | ||||
|  *                        depending on the panel type. | ||||
|  *                        vid mode -> vsync_irq | ||||
|  *                        cmd mode -> pp_done | ||||
|  * Returns: 0 on success, -EWOULDBLOCK if already signaled, error otherwise | ||||
|  */ | ||||
| int dpu_encoder_wait_for_event(struct drm_encoder *drm_encoder, | ||||
| 						enum msm_event_wait event); | ||||
| 
 | ||||
| /*
 | ||||
|  * dpu_encoder_get_intf_mode - get interface mode of the given encoder | ||||
|  * @encoder: Pointer to drm encoder object | ||||
|  */ | ||||
| enum dpu_intf_mode dpu_encoder_get_intf_mode(struct drm_encoder *encoder); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_virt_restore - restore the encoder configs | ||||
|  * @encoder:	encoder pointer | ||||
|  */ | ||||
| void dpu_encoder_virt_restore(struct drm_encoder *encoder); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_check_mode - check if given mode is supported or not | ||||
|  * @drm_enc: Pointer to drm encoder object | ||||
|  * @mode: Mode to be checked | ||||
|  * @Return: true if it is cmd mode | ||||
|  */ | ||||
| bool dpu_encoder_check_mode(struct drm_encoder *drm_enc, u32 mode); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_init - initialize virtual encoder object | ||||
|  * @dev:        Pointer to drm device structure | ||||
|  * @disp_info:  Pointer to display information structure | ||||
|  * Returns:     Pointer to newly created drm encoder | ||||
|  */ | ||||
| struct drm_encoder *dpu_encoder_init( | ||||
| 		struct drm_device *dev, | ||||
| 		int drm_enc_mode); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_setup - setup dpu_encoder for the display probed | ||||
|  * @dev:		Pointer to drm device structure | ||||
|  * @enc:		Pointer to the drm_encoder | ||||
|  * @disp_info:	Pointer to the display info | ||||
|  */ | ||||
| int dpu_encoder_setup(struct drm_device *dev, struct drm_encoder *enc, | ||||
| 		struct msm_display_info *disp_info); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_destroy - destroy previously initialized virtual encoder | ||||
|  * @drm_enc:    Pointer to previously created drm encoder structure | ||||
|  */ | ||||
| void dpu_encoder_destroy(struct drm_encoder *drm_enc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_prepare_commit - prepare encoder at the very beginning of an | ||||
|  *	atomic commit, before any registers are written | ||||
|  * @drm_enc:    Pointer to previously created drm encoder structure | ||||
|  */ | ||||
| void dpu_encoder_prepare_commit(struct drm_encoder *drm_enc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_set_idle_timeout - set the idle timeout for video | ||||
|  *                    and command mode encoders. | ||||
|  * @drm_enc:    Pointer to previously created drm encoder structure | ||||
|  * @idle_timeout:    idle timeout duration in milliseconds | ||||
|  */ | ||||
| void dpu_encoder_set_idle_timeout(struct drm_encoder *drm_enc, | ||||
| 							u32 idle_timeout); | ||||
| 
 | ||||
| #endif /* __DPU_ENCODER_H__ */ | ||||
							
								
								
									
										453
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder_phys.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										453
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder_phys.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,453 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2015-2018 The Linux Foundation. All rights reserved. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_ENCODER_PHYS_H__ | ||||
| #define __DPU_ENCODER_PHYS_H__ | ||||
| 
 | ||||
| #include <linux/jiffies.h> | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_hw_intf.h" | ||||
| #include "dpu_hw_pingpong.h" | ||||
| #include "dpu_hw_ctl.h" | ||||
| #include "dpu_hw_top.h" | ||||
| #include "dpu_hw_cdm.h" | ||||
| #include "dpu_encoder.h" | ||||
| 
 | ||||
| #define DPU_ENCODER_NAME_MAX	16 | ||||
| 
 | ||||
| /* wait for at most 2 vsync for lowest refresh rate (24hz) */ | ||||
| #define KICKOFF_TIMEOUT_MS		84 | ||||
| #define KICKOFF_TIMEOUT_JIFFIES		msecs_to_jiffies(KICKOFF_TIMEOUT_MS) | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_enc_split_role - Role this physical encoder will play in a | ||||
|  *	split-panel configuration, where one panel is master, and others slaves. | ||||
|  *	Masters have extra responsibilities, like managing the VBLANK IRQ. | ||||
|  * @ENC_ROLE_SOLO:	This is the one and only panel. This encoder is master. | ||||
|  * @ENC_ROLE_MASTER:	This encoder is the master of a split panel config. | ||||
|  * @ENC_ROLE_SLAVE:	This encoder is not the master of a split panel config. | ||||
|  */ | ||||
| enum dpu_enc_split_role { | ||||
| 	ENC_ROLE_SOLO, | ||||
| 	ENC_ROLE_MASTER, | ||||
| 	ENC_ROLE_SLAVE, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_enc_enable_state - current enabled state of the physical encoder | ||||
|  * @DPU_ENC_DISABLING:	Encoder transitioning to disable state | ||||
|  *			Events bounding transition are encoder type specific | ||||
|  * @DPU_ENC_DISABLED:	Encoder is disabled | ||||
|  * @DPU_ENC_ENABLING:	Encoder transitioning to enabled | ||||
|  *			Events bounding transition are encoder type specific | ||||
|  * @DPU_ENC_ENABLED:	Encoder is enabled | ||||
|  * @DPU_ENC_ERR_NEEDS_HW_RESET:	Encoder is enabled, but requires a hw_reset | ||||
|  *				to recover from a previous error | ||||
|  */ | ||||
| enum dpu_enc_enable_state { | ||||
| 	DPU_ENC_DISABLING, | ||||
| 	DPU_ENC_DISABLED, | ||||
| 	DPU_ENC_ENABLING, | ||||
| 	DPU_ENC_ENABLED, | ||||
| 	DPU_ENC_ERR_NEEDS_HW_RESET | ||||
| }; | ||||
| 
 | ||||
| struct dpu_encoder_phys; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_encoder_virt_ops - Interface the containing virtual encoder | ||||
|  *	provides for the physical encoders to use to callback. | ||||
|  * @handle_vblank_virt:	Notify virtual encoder of vblank IRQ reception | ||||
|  *			Note: This is called from IRQ handler context. | ||||
|  * @handle_underrun_virt: Notify virtual encoder of underrun IRQ reception | ||||
|  *			Note: This is called from IRQ handler context. | ||||
|  * @handle_frame_done:	Notify virtual encoder that this phys encoder | ||||
|  *			completes last request frame. | ||||
|  */ | ||||
| struct dpu_encoder_virt_ops { | ||||
| 	void (*handle_vblank_virt)(struct drm_encoder *, | ||||
| 			struct dpu_encoder_phys *phys); | ||||
| 	void (*handle_underrun_virt)(struct drm_encoder *, | ||||
| 			struct dpu_encoder_phys *phys); | ||||
| 	void (*handle_frame_done)(struct drm_encoder *, | ||||
| 			struct dpu_encoder_phys *phys, u32 event); | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_encoder_phys_ops - Interface the physical encoders provide to | ||||
|  *	the containing virtual encoder. | ||||
|  * @late_register:		DRM Call. Add Userspace interfaces, debugfs. | ||||
|  * @prepare_commit:		MSM Atomic Call, start of atomic commit sequence | ||||
|  * @is_master:			Whether this phys_enc is the current master | ||||
|  *				encoder. Can be switched at enable time. Based | ||||
|  *				on split_role and current mode (CMD/VID). | ||||
|  * @mode_fixup:			DRM Call. Fixup a DRM mode. | ||||
|  * @mode_set:			DRM Call. Set a DRM mode. | ||||
|  *				This likely caches the mode, for use at enable. | ||||
|  * @enable:			DRM Call. Enable a DRM mode. | ||||
|  * @disable:			DRM Call. Disable mode. | ||||
|  * @atomic_check:		DRM Call. Atomic check new DRM state. | ||||
|  * @destroy:			DRM Call. Destroy and release resources. | ||||
|  * @get_hw_resources:		Populate the structure with the hardware | ||||
|  *				resources that this phys_enc is using. | ||||
|  *				Expect no overlap between phys_encs. | ||||
|  * @control_vblank_irq		Register/Deregister for VBLANK IRQ | ||||
|  * @wait_for_commit_done:	Wait for hardware to have flushed the | ||||
|  *				current pending frames to hardware | ||||
|  * @wait_for_tx_complete:	Wait for hardware to transfer the pixels | ||||
|  *				to the panel | ||||
|  * @wait_for_vblank:		Wait for VBLANK, for sub-driver internal use | ||||
|  * @prepare_for_kickoff:	Do any work necessary prior to a kickoff | ||||
|  *				For CMD encoder, may wait for previous tx done | ||||
|  * @handle_post_kickoff:	Do any work necessary post-kickoff work | ||||
|  * @trigger_start:		Process start event on physical encoder | ||||
|  * @needs_single_flush:		Whether encoder slaves need to be flushed | ||||
|  * @setup_misr:		Sets up MISR, enable and disables based on sysfs | ||||
|  * @collect_misr:		Collects MISR data on frame update | ||||
|  * @hw_reset:			Issue HW recovery such as CTL reset and clear | ||||
|  *				DPU_ENC_ERR_NEEDS_HW_RESET state | ||||
|  * @irq_control:		Handler to enable/disable all the encoder IRQs | ||||
|  * @prepare_idle_pc:		phys encoder can update the vsync_enable status | ||||
|  *                              on idle power collapse prepare | ||||
|  * @restore:			Restore all the encoder configs. | ||||
|  * @get_line_count:		Obtain current vertical line count | ||||
|  */ | ||||
| 
 | ||||
| struct dpu_encoder_phys_ops { | ||||
| 	int (*late_register)(struct dpu_encoder_phys *encoder, | ||||
| 			struct dentry *debugfs_root); | ||||
| 	void (*prepare_commit)(struct dpu_encoder_phys *encoder); | ||||
| 	bool (*is_master)(struct dpu_encoder_phys *encoder); | ||||
| 	bool (*mode_fixup)(struct dpu_encoder_phys *encoder, | ||||
| 			const struct drm_display_mode *mode, | ||||
| 			struct drm_display_mode *adjusted_mode); | ||||
| 	void (*mode_set)(struct dpu_encoder_phys *encoder, | ||||
| 			struct drm_display_mode *mode, | ||||
| 			struct drm_display_mode *adjusted_mode); | ||||
| 	void (*enable)(struct dpu_encoder_phys *encoder); | ||||
| 	void (*disable)(struct dpu_encoder_phys *encoder); | ||||
| 	int (*atomic_check)(struct dpu_encoder_phys *encoder, | ||||
| 			    struct drm_crtc_state *crtc_state, | ||||
| 			    struct drm_connector_state *conn_state); | ||||
| 	void (*destroy)(struct dpu_encoder_phys *encoder); | ||||
| 	void (*get_hw_resources)(struct dpu_encoder_phys *encoder, | ||||
| 			struct dpu_encoder_hw_resources *hw_res, | ||||
| 			struct drm_connector_state *conn_state); | ||||
| 	int (*control_vblank_irq)(struct dpu_encoder_phys *enc, bool enable); | ||||
| 	int (*wait_for_commit_done)(struct dpu_encoder_phys *phys_enc); | ||||
| 	int (*wait_for_tx_complete)(struct dpu_encoder_phys *phys_enc); | ||||
| 	int (*wait_for_vblank)(struct dpu_encoder_phys *phys_enc); | ||||
| 	void (*prepare_for_kickoff)(struct dpu_encoder_phys *phys_enc, | ||||
| 			struct dpu_encoder_kickoff_params *params); | ||||
| 	void (*handle_post_kickoff)(struct dpu_encoder_phys *phys_enc); | ||||
| 	void (*trigger_start)(struct dpu_encoder_phys *phys_enc); | ||||
| 	bool (*needs_single_flush)(struct dpu_encoder_phys *phys_enc); | ||||
| 
 | ||||
| 	void (*setup_misr)(struct dpu_encoder_phys *phys_encs, | ||||
| 				bool enable, u32 frame_count); | ||||
| 	u32 (*collect_misr)(struct dpu_encoder_phys *phys_enc); | ||||
| 	void (*hw_reset)(struct dpu_encoder_phys *phys_enc); | ||||
| 	void (*irq_control)(struct dpu_encoder_phys *phys, bool enable); | ||||
| 	void (*prepare_idle_pc)(struct dpu_encoder_phys *phys_enc); | ||||
| 	void (*restore)(struct dpu_encoder_phys *phys); | ||||
| 	int (*get_line_count)(struct dpu_encoder_phys *phys); | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_intr_idx - dpu encoder interrupt index | ||||
|  * @INTR_IDX_VSYNC:    Vsync interrupt for video mode panel | ||||
|  * @INTR_IDX_PINGPONG: Pingpong done unterrupt for cmd mode panel | ||||
|  * @INTR_IDX_UNDERRUN: Underrun unterrupt for video and cmd mode panel | ||||
|  * @INTR_IDX_RDPTR:    Readpointer done unterrupt for cmd mode panel | ||||
|  */ | ||||
| enum dpu_intr_idx { | ||||
| 	INTR_IDX_VSYNC, | ||||
| 	INTR_IDX_PINGPONG, | ||||
| 	INTR_IDX_UNDERRUN, | ||||
| 	INTR_IDX_CTL_START, | ||||
| 	INTR_IDX_RDPTR, | ||||
| 	INTR_IDX_MAX, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_irq - tracking structure for interrupts | ||||
|  * @name:		string name of interrupt | ||||
|  * @intr_type:		Encoder interrupt type | ||||
|  * @intr_idx:		Encoder interrupt enumeration | ||||
|  * @hw_idx:		HW Block ID | ||||
|  * @irq_idx:		IRQ interface lookup index from DPU IRQ framework | ||||
|  *			will be -EINVAL if IRQ is not registered | ||||
|  * @irq_cb:		interrupt callback | ||||
|  */ | ||||
| struct dpu_encoder_irq { | ||||
| 	const char *name; | ||||
| 	enum dpu_intr_type intr_type; | ||||
| 	enum dpu_intr_idx intr_idx; | ||||
| 	int hw_idx; | ||||
| 	int irq_idx; | ||||
| 	struct dpu_irq_callback cb; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_encoder_phys - physical encoder that drives a single INTF block | ||||
|  *	tied to a specific panel / sub-panel. Abstract type, sub-classed by | ||||
|  *	phys_vid or phys_cmd for video mode or command mode encs respectively. | ||||
|  * @parent:		Pointer to the containing virtual encoder | ||||
|  * @connector:		If a mode is set, cached pointer to the active connector | ||||
|  * @ops:		Operations exposed to the virtual encoder | ||||
|  * @parent_ops:		Callbacks exposed by the parent to the phys_enc | ||||
|  * @hw_mdptop:		Hardware interface to the top registers | ||||
|  * @hw_ctl:		Hardware interface to the ctl registers | ||||
|  * @hw_cdm:		Hardware interface to the cdm registers | ||||
|  * @cdm_cfg:		Chroma-down hardware configuration | ||||
|  * @hw_pp:		Hardware interface to the ping pong registers | ||||
|  * @dpu_kms:		Pointer to the dpu_kms top level | ||||
|  * @cached_mode:	DRM mode cached at mode_set time, acted on in enable | ||||
|  * @enabled:		Whether the encoder has enabled and running a mode | ||||
|  * @split_role:		Role to play in a split-panel configuration | ||||
|  * @intf_mode:		Interface mode | ||||
|  * @intf_idx:		Interface index on dpu hardware | ||||
|  * @topology_name:	topology selected for the display | ||||
|  * @enc_spinlock:	Virtual-Encoder-Wide Spin Lock for IRQ purposes | ||||
|  * @enable_state:	Enable state tracking | ||||
|  * @vblank_refcount:	Reference count of vblank request | ||||
|  * @vsync_cnt:		Vsync count for the physical encoder | ||||
|  * @underrun_cnt:	Underrun count for the physical encoder | ||||
|  * @pending_kickoff_cnt:	Atomic counter tracking the number of kickoffs | ||||
|  *				vs. the number of done/vblank irqs. Should hover | ||||
|  *				between 0-2 Incremented when a new kickoff is | ||||
|  *				scheduled. Decremented in irq handler | ||||
|  * @pending_ctlstart_cnt:	Atomic counter tracking the number of ctl start | ||||
|  *                              pending. | ||||
|  * @pending_kickoff_wq:		Wait queue for blocking until kickoff completes | ||||
|  * @irq:			IRQ tracking structures | ||||
|  */ | ||||
| struct dpu_encoder_phys { | ||||
| 	struct drm_encoder *parent; | ||||
| 	struct drm_connector *connector; | ||||
| 	struct dpu_encoder_phys_ops ops; | ||||
| 	const struct dpu_encoder_virt_ops *parent_ops; | ||||
| 	struct dpu_hw_mdp *hw_mdptop; | ||||
| 	struct dpu_hw_ctl *hw_ctl; | ||||
| 	struct dpu_hw_cdm *hw_cdm; | ||||
| 	struct dpu_hw_cdm_cfg cdm_cfg; | ||||
| 	struct dpu_hw_pingpong *hw_pp; | ||||
| 	struct dpu_kms *dpu_kms; | ||||
| 	struct drm_display_mode cached_mode; | ||||
| 	enum dpu_enc_split_role split_role; | ||||
| 	enum dpu_intf_mode intf_mode; | ||||
| 	enum dpu_intf intf_idx; | ||||
| 	enum dpu_rm_topology_name topology_name; | ||||
| 	spinlock_t *enc_spinlock; | ||||
| 	enum dpu_enc_enable_state enable_state; | ||||
| 	atomic_t vblank_refcount; | ||||
| 	atomic_t vsync_cnt; | ||||
| 	atomic_t underrun_cnt; | ||||
| 	atomic_t pending_ctlstart_cnt; | ||||
| 	atomic_t pending_kickoff_cnt; | ||||
| 	wait_queue_head_t pending_kickoff_wq; | ||||
| 	struct dpu_encoder_irq irq[INTR_IDX_MAX]; | ||||
| }; | ||||
| 
 | ||||
| static inline int dpu_encoder_phys_inc_pending(struct dpu_encoder_phys *phys) | ||||
| { | ||||
| 	atomic_inc_return(&phys->pending_ctlstart_cnt); | ||||
| 	return atomic_inc_return(&phys->pending_kickoff_cnt); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_encoder_phys_vid - sub-class of dpu_encoder_phys to handle video | ||||
|  *	mode specific operations | ||||
|  * @base:	Baseclass physical encoder structure | ||||
|  * @hw_intf:	Hardware interface to the intf registers | ||||
|  * @timing_params: Current timing parameter | ||||
|  */ | ||||
| struct dpu_encoder_phys_vid { | ||||
| 	struct dpu_encoder_phys base; | ||||
| 	struct dpu_hw_intf *hw_intf; | ||||
| 	struct intf_timing_params timing_params; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_encoder_phys_cmd - sub-class of dpu_encoder_phys to handle command | ||||
|  *	mode specific operations | ||||
|  * @base:	Baseclass physical encoder structure | ||||
|  * @intf_idx:	Intf Block index used by this phys encoder | ||||
|  * @stream_sel:	Stream selection for multi-stream interfaces | ||||
|  * @serialize_wait4pp:	serialize wait4pp feature waits for pp_done interrupt | ||||
|  *			after ctl_start instead of before next frame kickoff | ||||
|  * @pp_timeout_report_cnt: number of pingpong done irq timeout errors | ||||
|  * @pending_vblank_cnt: Atomic counter tracking pending wait for VBLANK | ||||
|  * @pending_vblank_wq: Wait queue for blocking until VBLANK received | ||||
|  */ | ||||
| struct dpu_encoder_phys_cmd { | ||||
| 	struct dpu_encoder_phys base; | ||||
| 	int stream_sel; | ||||
| 	bool serialize_wait4pp; | ||||
| 	int pp_timeout_report_cnt; | ||||
| 	atomic_t pending_vblank_cnt; | ||||
| 	wait_queue_head_t pending_vblank_wq; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_enc_phys_init_params - initialization parameters for phys encs | ||||
|  * @dpu_kms:		Pointer to the dpu_kms top level | ||||
|  * @parent:		Pointer to the containing virtual encoder | ||||
|  * @parent_ops:		Callbacks exposed by the parent to the phys_enc | ||||
|  * @split_role:		Role to play in a split-panel configuration | ||||
|  * @intf_idx:		Interface index this phys_enc will control | ||||
|  * @enc_spinlock:	Virtual-Encoder-Wide Spin Lock for IRQ purposes | ||||
|  */ | ||||
| struct dpu_enc_phys_init_params { | ||||
| 	struct dpu_kms *dpu_kms; | ||||
| 	struct drm_encoder *parent; | ||||
| 	const struct dpu_encoder_virt_ops *parent_ops; | ||||
| 	enum dpu_enc_split_role split_role; | ||||
| 	enum dpu_intf intf_idx; | ||||
| 	spinlock_t *enc_spinlock; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_wait_info - container for passing arguments to irq wait functions | ||||
|  * @wq: wait queue structure | ||||
|  * @atomic_cnt: wait until atomic_cnt equals zero | ||||
|  * @timeout_ms: timeout value in milliseconds | ||||
|  */ | ||||
| struct dpu_encoder_wait_info { | ||||
| 	wait_queue_head_t *wq; | ||||
| 	atomic_t *atomic_cnt; | ||||
| 	s64 timeout_ms; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_phys_vid_init - Construct a new video mode physical encoder | ||||
|  * @p:	Pointer to init params structure | ||||
|  * Return: Error code or newly allocated encoder | ||||
|  */ | ||||
| struct dpu_encoder_phys *dpu_encoder_phys_vid_init( | ||||
| 		struct dpu_enc_phys_init_params *p); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_phys_cmd_init - Construct a new command mode physical encoder | ||||
|  * @p:	Pointer to init params structure | ||||
|  * Return: Error code or newly allocated encoder | ||||
|  */ | ||||
| struct dpu_encoder_phys *dpu_encoder_phys_cmd_init( | ||||
| 		struct dpu_enc_phys_init_params *p); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_trigger_start - control start helper function | ||||
|  *	This helper function may be optionally specified by physical | ||||
|  *	encoders if they require ctl_start triggering. | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  */ | ||||
| void dpu_encoder_helper_trigger_start(struct dpu_encoder_phys *phys_enc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_wait_event_timeout - wait for event with timeout | ||||
|  *	taking into account that jiffies may jump between reads leading to | ||||
|  *	incorrectly detected timeouts. Prevent failure in this scenario by | ||||
|  *	making sure that elapsed time during wait is valid. | ||||
|  * @drm_id: drm object id for logging | ||||
|  * @hw_id: hw instance id for logging | ||||
|  * @info: wait info structure | ||||
|  */ | ||||
| int dpu_encoder_helper_wait_event_timeout( | ||||
| 		int32_t drm_id, | ||||
| 		int32_t hw_id, | ||||
| 		struct dpu_encoder_wait_info *info); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_hw_reset - issue ctl hw reset | ||||
|  *	This helper function may be optionally specified by physical | ||||
|  *	encoders if they require ctl hw reset. If state is currently | ||||
|  *	DPU_ENC_ERR_NEEDS_HW_RESET, it is set back to DPU_ENC_ENABLED. | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  */ | ||||
| void dpu_encoder_helper_hw_reset(struct dpu_encoder_phys *phys_enc); | ||||
| 
 | ||||
| static inline enum dpu_3d_blend_mode dpu_encoder_helper_get_3d_blend_mode( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	if (!phys_enc || phys_enc->enable_state == DPU_ENC_DISABLING) | ||||
| 		return BLEND_3D_NONE; | ||||
| 
 | ||||
| 	if (phys_enc->split_role == ENC_ROLE_SOLO && | ||||
| 	    phys_enc->topology_name == DPU_RM_TOPOLOGY_DUALPIPE_3DMERGE) | ||||
| 		return BLEND_3D_H_ROW_INT; | ||||
| 
 | ||||
| 	return BLEND_3D_NONE; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_split_config - split display configuration helper function | ||||
|  *	This helper function may be used by physical encoders to configure | ||||
|  *	the split display related registers. | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  * @interface: enum dpu_intf setting | ||||
|  */ | ||||
| void dpu_encoder_helper_split_config( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		enum dpu_intf interface); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_hw_release - prepare for h/w reset during disable | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  * @fb: Optional fb for specifying new mixer output resolution, may be NULL | ||||
|  * Return: Zero on success | ||||
|  */ | ||||
| int dpu_encoder_helper_hw_release(struct dpu_encoder_phys *phys_enc, | ||||
| 		struct drm_framebuffer *fb); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_report_irq_timeout - utility to report error that irq has | ||||
|  *	timed out, including reporting frame error event to crtc and debug dump | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  * @intr_idx: Failing interrupt index | ||||
|  */ | ||||
| void dpu_encoder_helper_report_irq_timeout(struct dpu_encoder_phys *phys_enc, | ||||
| 		enum dpu_intr_idx intr_idx); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_wait_for_irq - utility to wait on an irq. | ||||
|  *	note: will call dpu_encoder_helper_wait_for_irq on timeout | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  * @intr_idx: encoder interrupt index | ||||
|  * @wait_info: wait info struct | ||||
|  * @Return: 0 or -ERROR | ||||
|  */ | ||||
| int dpu_encoder_helper_wait_for_irq(struct dpu_encoder_phys *phys_enc, | ||||
| 		enum dpu_intr_idx intr_idx, | ||||
| 		struct dpu_encoder_wait_info *wait_info); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_register_irq - register and enable an irq | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  * @intr_idx: encoder interrupt index | ||||
|  * @Return: 0 or -ERROR | ||||
|  */ | ||||
| int dpu_encoder_helper_register_irq(struct dpu_encoder_phys *phys_enc, | ||||
| 		enum dpu_intr_idx intr_idx); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_encoder_helper_unregister_irq - unregister and disable an irq | ||||
|  * @phys_enc: Pointer to physical encoder structure | ||||
|  * @intr_idx: encoder interrupt index | ||||
|  * @Return: 0 or -ERROR | ||||
|  */ | ||||
| int dpu_encoder_helper_unregister_irq(struct dpu_encoder_phys *phys_enc, | ||||
| 		enum dpu_intr_idx intr_idx); | ||||
| 
 | ||||
| #endif /* __dpu_encoder_phys_H__ */ | ||||
							
								
								
									
										905
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder_phys_cmd.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										905
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder_phys_cmd.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,905 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2015-2018 The Linux Foundation. All rights reserved. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| #include "dpu_encoder_phys.h" | ||||
| #include "dpu_hw_interrupts.h" | ||||
| #include "dpu_core_irq.h" | ||||
| #include "dpu_formats.h" | ||||
| #include "dpu_trace.h" | ||||
| 
 | ||||
| #define DPU_DEBUG_CMDENC(e, fmt, ...) DPU_DEBUG("enc%d intf%d " fmt, \ | ||||
| 		(e) && (e)->base.parent ? \ | ||||
| 		(e)->base.parent->base.id : -1, \ | ||||
| 		(e) ? (e)->base.intf_idx - INTF_0 : -1, ##__VA_ARGS__) | ||||
| 
 | ||||
| #define DPU_ERROR_CMDENC(e, fmt, ...) DPU_ERROR("enc%d intf%d " fmt, \ | ||||
| 		(e) && (e)->base.parent ? \ | ||||
| 		(e)->base.parent->base.id : -1, \ | ||||
| 		(e) ? (e)->base.intf_idx - INTF_0 : -1, ##__VA_ARGS__) | ||||
| 
 | ||||
| #define to_dpu_encoder_phys_cmd(x) \ | ||||
| 	container_of(x, struct dpu_encoder_phys_cmd, base) | ||||
| 
 | ||||
| #define PP_TIMEOUT_MAX_TRIALS	10 | ||||
| 
 | ||||
| /*
 | ||||
|  * Tearcheck sync start and continue thresholds are empirically found | ||||
|  * based on common panels In the future, may want to allow panels to override | ||||
|  * these default values | ||||
|  */ | ||||
| #define DEFAULT_TEARCHECK_SYNC_THRESH_START	4 | ||||
| #define DEFAULT_TEARCHECK_SYNC_THRESH_CONTINUE	4 | ||||
| 
 | ||||
| #define DPU_ENC_WR_PTR_START_TIMEOUT_US 20000 | ||||
| 
 | ||||
| static inline int _dpu_encoder_phys_cmd_get_idle_timeout( | ||||
| 		struct dpu_encoder_phys_cmd *cmd_enc) | ||||
| { | ||||
| 	return KICKOFF_TIMEOUT_MS; | ||||
| } | ||||
| 
 | ||||
| static inline bool dpu_encoder_phys_cmd_is_master( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	return (phys_enc->split_role != ENC_ROLE_SLAVE) ? true : false; | ||||
| } | ||||
| 
 | ||||
| static bool dpu_encoder_phys_cmd_mode_fixup( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		const struct drm_display_mode *mode, | ||||
| 		struct drm_display_mode *adj_mode) | ||||
| { | ||||
| 	if (phys_enc) | ||||
| 		DPU_DEBUG_CMDENC(to_dpu_encoder_phys_cmd(phys_enc), "\n"); | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| static void _dpu_encoder_phys_cmd_update_intf_cfg( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 			to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	struct dpu_hw_ctl *ctl; | ||||
| 	struct dpu_hw_intf_cfg intf_cfg = { 0 }; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	ctl = phys_enc->hw_ctl; | ||||
| 	if (!ctl || !ctl->ops.setup_intf_cfg) | ||||
| 		return; | ||||
| 
 | ||||
| 	intf_cfg.intf = phys_enc->intf_idx; | ||||
| 	intf_cfg.intf_mode_sel = DPU_CTL_MODE_SEL_CMD; | ||||
| 	intf_cfg.stream_sel = cmd_enc->stream_sel; | ||||
| 	intf_cfg.mode_3d = dpu_encoder_helper_get_3d_blend_mode(phys_enc); | ||||
| 	ctl->ops.setup_intf_cfg(ctl, &intf_cfg); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_pp_tx_done_irq(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = arg; | ||||
| 	unsigned long lock_flags; | ||||
| 	int new_cnt; | ||||
| 	u32 event = DPU_ENCODER_FRAME_EVENT_DONE; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) | ||||
| 		return; | ||||
| 
 | ||||
| 	DPU_ATRACE_BEGIN("pp_done_irq"); | ||||
| 	/* notify all synchronous clients first, then asynchronous clients */ | ||||
| 	if (phys_enc->parent_ops->handle_frame_done) | ||||
| 		phys_enc->parent_ops->handle_frame_done(phys_enc->parent, | ||||
| 				phys_enc, event); | ||||
| 
 | ||||
| 	spin_lock_irqsave(phys_enc->enc_spinlock, lock_flags); | ||||
| 	new_cnt = atomic_add_unless(&phys_enc->pending_kickoff_cnt, -1, 0); | ||||
| 	spin_unlock_irqrestore(phys_enc->enc_spinlock, lock_flags); | ||||
| 
 | ||||
| 	trace_dpu_enc_phys_cmd_pp_tx_done(DRMID(phys_enc->parent), | ||||
| 					  phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 					  new_cnt, event); | ||||
| 
 | ||||
| 	/* Signal any waiting atomic commit thread */ | ||||
| 	wake_up_all(&phys_enc->pending_kickoff_wq); | ||||
| 	DPU_ATRACE_END("pp_done_irq"); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_pp_rd_ptr_irq(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = arg; | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) | ||||
| 		return; | ||||
| 
 | ||||
| 	DPU_ATRACE_BEGIN("rd_ptr_irq"); | ||||
| 	cmd_enc = to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	if (phys_enc->parent_ops->handle_vblank_virt) | ||||
| 		phys_enc->parent_ops->handle_vblank_virt(phys_enc->parent, | ||||
| 			phys_enc); | ||||
| 
 | ||||
| 	atomic_add_unless(&cmd_enc->pending_vblank_cnt, -1, 0); | ||||
| 	wake_up_all(&cmd_enc->pending_vblank_wq); | ||||
| 	DPU_ATRACE_END("rd_ptr_irq"); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_ctl_start_irq(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = arg; | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_ctl) | ||||
| 		return; | ||||
| 
 | ||||
| 	DPU_ATRACE_BEGIN("ctl_start_irq"); | ||||
| 	cmd_enc = to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	atomic_add_unless(&phys_enc->pending_ctlstart_cnt, -1, 0); | ||||
| 
 | ||||
| 	/* Signal any waiting ctl start interrupt */ | ||||
| 	wake_up_all(&phys_enc->pending_kickoff_wq); | ||||
| 	DPU_ATRACE_END("ctl_start_irq"); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_underrun_irq(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = arg; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (phys_enc->parent_ops->handle_underrun_virt) | ||||
| 		phys_enc->parent_ops->handle_underrun_virt(phys_enc->parent, | ||||
| 			phys_enc); | ||||
| } | ||||
| 
 | ||||
| static void _dpu_encoder_phys_cmd_setup_irq_hw_idx( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_irq *irq; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_CTL_START]; | ||||
| 	irq->hw_idx = phys_enc->hw_ctl->idx; | ||||
| 	irq->irq_idx = -EINVAL; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_PINGPONG]; | ||||
| 	irq->hw_idx = phys_enc->hw_pp->idx; | ||||
| 	irq->irq_idx = -EINVAL; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_RDPTR]; | ||||
| 	irq->hw_idx = phys_enc->hw_pp->idx; | ||||
| 	irq->irq_idx = -EINVAL; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_UNDERRUN]; | ||||
| 	irq->hw_idx = phys_enc->intf_idx; | ||||
| 	irq->irq_idx = -EINVAL; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_mode_set( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		struct drm_display_mode *mode, | ||||
| 		struct drm_display_mode *adj_mode) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	struct dpu_rm *rm = &phys_enc->dpu_kms->rm; | ||||
| 	struct dpu_rm_hw_iter iter; | ||||
| 	int i, instance; | ||||
| 
 | ||||
| 	if (!phys_enc || !mode || !adj_mode) { | ||||
| 		DPU_ERROR("invalid args\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	phys_enc->cached_mode = *adj_mode; | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "caching mode:\n"); | ||||
| 	drm_mode_debug_printmodeline(adj_mode); | ||||
| 
 | ||||
| 	instance = phys_enc->split_role == ENC_ROLE_SLAVE ? 1 : 0; | ||||
| 
 | ||||
| 	/* Retrieve previously allocated HW Resources. Shouldn't fail */ | ||||
| 	dpu_rm_init_hw_iter(&iter, phys_enc->parent->base.id, DPU_HW_BLK_CTL); | ||||
| 	for (i = 0; i <= instance; i++) { | ||||
| 		if (dpu_rm_get_hw(rm, &iter)) | ||||
| 			phys_enc->hw_ctl = (struct dpu_hw_ctl *)iter.hw; | ||||
| 	} | ||||
| 
 | ||||
| 	if (IS_ERR_OR_NULL(phys_enc->hw_ctl)) { | ||||
| 		DPU_ERROR_CMDENC(cmd_enc, "failed to init ctl: %ld\n", | ||||
| 				PTR_ERR(phys_enc->hw_ctl)); | ||||
| 		phys_enc->hw_ctl = NULL; | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	_dpu_encoder_phys_cmd_setup_irq_hw_idx(phys_enc); | ||||
| } | ||||
| 
 | ||||
| static int _dpu_encoder_phys_cmd_handle_ppdone_timeout( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 			to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	u32 frame_event = DPU_ENCODER_FRAME_EVENT_ERROR; | ||||
| 	bool do_log = false; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp || !phys_enc->hw_ctl) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	cmd_enc->pp_timeout_report_cnt++; | ||||
| 	if (cmd_enc->pp_timeout_report_cnt == PP_TIMEOUT_MAX_TRIALS) { | ||||
| 		frame_event |= DPU_ENCODER_FRAME_EVENT_PANEL_DEAD; | ||||
| 		do_log = true; | ||||
| 	} else if (cmd_enc->pp_timeout_report_cnt == 1) { | ||||
| 		do_log = true; | ||||
| 	} | ||||
| 
 | ||||
| 	trace_dpu_enc_phys_cmd_pdone_timeout(DRMID(phys_enc->parent), | ||||
| 		     phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 		     cmd_enc->pp_timeout_report_cnt, | ||||
| 		     atomic_read(&phys_enc->pending_kickoff_cnt), | ||||
| 		     frame_event); | ||||
| 
 | ||||
| 	/* to avoid flooding, only log first time, and "dead" time */ | ||||
| 	if (do_log) { | ||||
| 		DRM_ERROR("id:%d pp:%d kickoff timeout %d cnt %d koff_cnt %d\n", | ||||
| 			  DRMID(phys_enc->parent), | ||||
| 			  phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 			  phys_enc->hw_ctl->idx - CTL_0, | ||||
| 			  cmd_enc->pp_timeout_report_cnt, | ||||
| 			  atomic_read(&phys_enc->pending_kickoff_cnt)); | ||||
| 
 | ||||
| 		dpu_encoder_helper_unregister_irq(phys_enc, INTR_IDX_RDPTR); | ||||
| 		dpu_dbg_dump(false, __func__, true, true); | ||||
| 	} | ||||
| 
 | ||||
| 	atomic_add_unless(&phys_enc->pending_kickoff_cnt, -1, 0); | ||||
| 
 | ||||
| 	/* request a ctl reset before the next kickoff */ | ||||
| 	phys_enc->enable_state = DPU_ENC_ERR_NEEDS_HW_RESET; | ||||
| 
 | ||||
| 	if (phys_enc->parent_ops->handle_frame_done) | ||||
| 		phys_enc->parent_ops->handle_frame_done( | ||||
| 				phys_enc->parent, phys_enc, frame_event); | ||||
| 
 | ||||
| 	return -ETIMEDOUT; | ||||
| } | ||||
| 
 | ||||
| static int _dpu_encoder_phys_cmd_wait_for_idle( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 			to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	struct dpu_encoder_wait_info wait_info; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	wait_info.wq = &phys_enc->pending_kickoff_wq; | ||||
| 	wait_info.atomic_cnt = &phys_enc->pending_kickoff_cnt; | ||||
| 	wait_info.timeout_ms = KICKOFF_TIMEOUT_MS; | ||||
| 
 | ||||
| 	ret = dpu_encoder_helper_wait_for_irq(phys_enc, INTR_IDX_PINGPONG, | ||||
| 			&wait_info); | ||||
| 	if (ret == -ETIMEDOUT) | ||||
| 		_dpu_encoder_phys_cmd_handle_ppdone_timeout(phys_enc); | ||||
| 	else if (!ret) | ||||
| 		cmd_enc->pp_timeout_report_cnt = 0; | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_cmd_control_vblank_irq( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		bool enable) | ||||
| { | ||||
| 	int ret = 0; | ||||
| 	int refcount; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	refcount = atomic_read(&phys_enc->vblank_refcount); | ||||
| 
 | ||||
| 	/* Slave encoders don't report vblank */ | ||||
| 	if (!dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 		goto end; | ||||
| 
 | ||||
| 	/* protect against negative */ | ||||
| 	if (!enable && refcount == 0) { | ||||
| 		ret = -EINVAL; | ||||
| 		goto end; | ||||
| 	} | ||||
| 
 | ||||
| 	DRM_DEBUG_KMS("id:%u pp:%d enable=%s/%d\n", DRMID(phys_enc->parent), | ||||
| 		      phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 		      enable ? "true" : "false", refcount); | ||||
| 
 | ||||
| 	if (enable && atomic_inc_return(&phys_enc->vblank_refcount) == 1) | ||||
| 		ret = dpu_encoder_helper_register_irq(phys_enc, INTR_IDX_RDPTR); | ||||
| 	else if (!enable && atomic_dec_return(&phys_enc->vblank_refcount) == 0) | ||||
| 		ret = dpu_encoder_helper_unregister_irq(phys_enc, | ||||
| 				INTR_IDX_RDPTR); | ||||
| 
 | ||||
| end: | ||||
| 	if (ret) { | ||||
| 		DRM_ERROR("vblank irq err id:%u pp:%d ret:%d, enable %s/%d\n", | ||||
| 			  DRMID(phys_enc->parent), | ||||
| 			  phys_enc->hw_pp->idx - PINGPONG_0, ret, | ||||
| 			  enable ? "true" : "false", refcount); | ||||
| 	} | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| void dpu_encoder_phys_cmd_irq_control(struct dpu_encoder_phys *phys_enc, | ||||
| 		bool enable) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	cmd_enc = to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	trace_dpu_enc_phys_cmd_irq_ctrl(DRMID(phys_enc->parent), | ||||
| 			phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 			enable, atomic_read(&phys_enc->vblank_refcount)); | ||||
| 
 | ||||
| 	if (enable) { | ||||
| 		dpu_encoder_helper_register_irq(phys_enc, INTR_IDX_PINGPONG); | ||||
| 		dpu_encoder_helper_register_irq(phys_enc, INTR_IDX_UNDERRUN); | ||||
| 		dpu_encoder_phys_cmd_control_vblank_irq(phys_enc, true); | ||||
| 
 | ||||
| 		if (dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 			dpu_encoder_helper_register_irq(phys_enc, | ||||
| 					INTR_IDX_CTL_START); | ||||
| 	} else { | ||||
| 		if (dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 			dpu_encoder_helper_unregister_irq(phys_enc, | ||||
| 					INTR_IDX_CTL_START); | ||||
| 
 | ||||
| 		dpu_encoder_helper_unregister_irq(phys_enc, INTR_IDX_UNDERRUN); | ||||
| 		dpu_encoder_phys_cmd_control_vblank_irq(phys_enc, false); | ||||
| 		dpu_encoder_helper_unregister_irq(phys_enc, INTR_IDX_PINGPONG); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_tearcheck_config( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	struct dpu_hw_tear_check tc_cfg = { 0 }; | ||||
| 	struct drm_display_mode *mode; | ||||
| 	bool tc_enable = true; | ||||
| 	u32 vsync_hz; | ||||
| 	struct msm_drm_private *priv; | ||||
| 	struct dpu_kms *dpu_kms; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	mode = &phys_enc->cached_mode; | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "pp %d\n", phys_enc->hw_pp->idx - PINGPONG_0); | ||||
| 
 | ||||
| 	if (!phys_enc->hw_pp->ops.setup_tearcheck || | ||||
| 		!phys_enc->hw_pp->ops.enable_tearcheck) { | ||||
| 		DPU_DEBUG_CMDENC(cmd_enc, "tearcheck not supported\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_kms = phys_enc->dpu_kms; | ||||
| 	if (!dpu_kms || !dpu_kms->dev || !dpu_kms->dev->dev_private) { | ||||
| 		DPU_ERROR("invalid device\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	priv = dpu_kms->dev->dev_private; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * TE default: dsi byte clock calculated base on 70 fps; | ||||
| 	 * around 14 ms to complete a kickoff cycle if te disabled; | ||||
| 	 * vclk_line base on 60 fps; write is faster than read; | ||||
| 	 * init == start == rdptr; | ||||
| 	 * | ||||
| 	 * vsync_count is ratio of MDP VSYNC clock frequency to LCD panel | ||||
| 	 * frequency divided by the no. of rows (lines) in the LCDpanel. | ||||
| 	 */ | ||||
| 	vsync_hz = dpu_kms_get_clk_rate(dpu_kms, "vsync_clk"); | ||||
| 	if (vsync_hz <= 0) { | ||||
| 		DPU_DEBUG_CMDENC(cmd_enc, "invalid - vsync_hz %u\n", | ||||
| 				 vsync_hz); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	tc_cfg.vsync_count = vsync_hz / (mode->vtotal * mode->vrefresh); | ||||
| 
 | ||||
| 	/* enable external TE after kickoff to avoid premature autorefresh */ | ||||
| 	tc_cfg.hw_vsync_mode = 0; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * By setting sync_cfg_height to near max register value, we essentially | ||||
| 	 * disable dpu hw generated TE signal, since hw TE will arrive first. | ||||
| 	 * Only caveat is if due to error, we hit wrap-around. | ||||
| 	 */ | ||||
| 	tc_cfg.sync_cfg_height = 0xFFF0; | ||||
| 	tc_cfg.vsync_init_val = mode->vdisplay; | ||||
| 	tc_cfg.sync_threshold_start = DEFAULT_TEARCHECK_SYNC_THRESH_START; | ||||
| 	tc_cfg.sync_threshold_continue = DEFAULT_TEARCHECK_SYNC_THRESH_CONTINUE; | ||||
| 	tc_cfg.start_pos = mode->vdisplay; | ||||
| 	tc_cfg.rd_ptr_irq = mode->vdisplay + 1; | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, | ||||
| 		"tc %d vsync_clk_speed_hz %u vtotal %u vrefresh %u\n", | ||||
| 		phys_enc->hw_pp->idx - PINGPONG_0, vsync_hz, | ||||
| 		mode->vtotal, mode->vrefresh); | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, | ||||
| 		"tc %d enable %u start_pos %u rd_ptr_irq %u\n", | ||||
| 		phys_enc->hw_pp->idx - PINGPONG_0, tc_enable, tc_cfg.start_pos, | ||||
| 		tc_cfg.rd_ptr_irq); | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, | ||||
| 		"tc %d hw_vsync_mode %u vsync_count %u vsync_init_val %u\n", | ||||
| 		phys_enc->hw_pp->idx - PINGPONG_0, tc_cfg.hw_vsync_mode, | ||||
| 		tc_cfg.vsync_count, tc_cfg.vsync_init_val); | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, | ||||
| 		"tc %d cfgheight %u thresh_start %u thresh_cont %u\n", | ||||
| 		phys_enc->hw_pp->idx - PINGPONG_0, tc_cfg.sync_cfg_height, | ||||
| 		tc_cfg.sync_threshold_start, tc_cfg.sync_threshold_continue); | ||||
| 
 | ||||
| 	phys_enc->hw_pp->ops.setup_tearcheck(phys_enc->hw_pp, &tc_cfg); | ||||
| 	phys_enc->hw_pp->ops.enable_tearcheck(phys_enc->hw_pp, tc_enable); | ||||
| } | ||||
| 
 | ||||
| static void _dpu_encoder_phys_cmd_pingpong_config( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_ctl || !phys_enc->hw_pp | ||||
| 			|| !phys_enc->hw_ctl->ops.setup_intf_cfg) { | ||||
| 		DPU_ERROR("invalid arg(s), enc %d\n", phys_enc != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "pp %d, enabling mode:\n", | ||||
| 			phys_enc->hw_pp->idx - PINGPONG_0); | ||||
| 	drm_mode_debug_printmodeline(&phys_enc->cached_mode); | ||||
| 
 | ||||
| 	_dpu_encoder_phys_cmd_update_intf_cfg(phys_enc); | ||||
| 	dpu_encoder_phys_cmd_tearcheck_config(phys_enc); | ||||
| } | ||||
| 
 | ||||
| static bool dpu_encoder_phys_cmd_needs_single_flush( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	/**
 | ||||
| 	 * we do separate flush for each CTL and let | ||||
| 	 * CTL_START synchronize them | ||||
| 	 */ | ||||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_enable_helper( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_hw_ctl *ctl; | ||||
| 	u32 flush_mask = 0; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_ctl || !phys_enc->hw_pp) { | ||||
| 		DPU_ERROR("invalid arg(s), encoder %d\n", phys_enc != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_encoder_helper_split_config(phys_enc, phys_enc->intf_idx); | ||||
| 
 | ||||
| 	_dpu_encoder_phys_cmd_pingpong_config(phys_enc); | ||||
| 
 | ||||
| 	if (!dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 		goto skip_flush; | ||||
| 
 | ||||
| 	ctl = phys_enc->hw_ctl; | ||||
| 	ctl->ops.get_bitmask_intf(ctl, &flush_mask, phys_enc->intf_idx); | ||||
| 	ctl->ops.update_pending_flush(ctl, flush_mask); | ||||
| 
 | ||||
| skip_flush: | ||||
| 	return; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_enable(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) { | ||||
| 		DPU_ERROR("invalid phys encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "pp %d\n", phys_enc->hw_pp->idx - PINGPONG_0); | ||||
| 
 | ||||
| 	if (phys_enc->enable_state == DPU_ENC_ENABLED) { | ||||
| 		DPU_ERROR("already enabled\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_encoder_phys_cmd_enable_helper(phys_enc); | ||||
| 	phys_enc->enable_state = DPU_ENC_ENABLED; | ||||
| } | ||||
| 
 | ||||
| static void _dpu_encoder_phys_cmd_connect_te( | ||||
| 		struct dpu_encoder_phys *phys_enc, bool enable) | ||||
| { | ||||
| 	if (!phys_enc || !phys_enc->hw_pp || | ||||
| 			!phys_enc->hw_pp->ops.connect_external_te) | ||||
| 		return; | ||||
| 
 | ||||
| 	trace_dpu_enc_phys_cmd_connect_te(DRMID(phys_enc->parent), enable); | ||||
| 	phys_enc->hw_pp->ops.connect_external_te(phys_enc->hw_pp, enable); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_prepare_idle_pc( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	_dpu_encoder_phys_cmd_connect_te(phys_enc, false); | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_cmd_get_line_count( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_hw_pingpong *hw_pp; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (!dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	hw_pp = phys_enc->hw_pp; | ||||
| 	if (!hw_pp->ops.get_line_count) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	return hw_pp->ops.get_line_count(hw_pp); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_disable(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	DRM_DEBUG_KMS("id:%u pp:%d state:%d\n", DRMID(phys_enc->parent), | ||||
| 		      phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 		      phys_enc->enable_state); | ||||
| 
 | ||||
| 	if (phys_enc->enable_state == DPU_ENC_DISABLED) { | ||||
| 		DPU_ERROR_CMDENC(cmd_enc, "already disabled\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if (phys_enc->hw_pp->ops.enable_tearcheck) | ||||
| 		phys_enc->hw_pp->ops.enable_tearcheck(phys_enc->hw_pp, false); | ||||
| 	phys_enc->enable_state = DPU_ENC_DISABLED; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_destroy(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	kfree(cmd_enc); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_get_hw_resources( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		struct dpu_encoder_hw_resources *hw_res, | ||||
| 		struct drm_connector_state *conn_state) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 		to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if ((phys_enc->intf_idx - INTF_0) >= INTF_MAX) { | ||||
| 		DPU_ERROR("invalid intf idx:%d\n", phys_enc->intf_idx); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "\n"); | ||||
| 	hw_res->intfs[phys_enc->intf_idx - INTF_0] = INTF_MODE_CMD; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_prepare_for_kickoff( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		struct dpu_encoder_kickoff_params *params) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 			to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_pp) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	DRM_DEBUG_KMS("id:%u pp:%d pending_cnt:%d\n", DRMID(phys_enc->parent), | ||||
| 		      phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 		      atomic_read(&phys_enc->pending_kickoff_cnt)); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Mark kickoff request as outstanding. If there are more than one, | ||||
| 	 * outstanding, then we have to wait for the previous one to complete | ||||
| 	 */ | ||||
| 	ret = _dpu_encoder_phys_cmd_wait_for_idle(phys_enc); | ||||
| 	if (ret) { | ||||
| 		/* force pending_kickoff_cnt 0 to discard failed kickoff */ | ||||
| 		atomic_set(&phys_enc->pending_kickoff_cnt, 0); | ||||
| 		DRM_ERROR("failed wait_for_idle: id:%u ret:%d pp:%d\n", | ||||
| 			  DRMID(phys_enc->parent), ret, | ||||
| 			  phys_enc->hw_pp->idx - PINGPONG_0); | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "pp:%d pending_cnt %d\n", | ||||
| 			phys_enc->hw_pp->idx - PINGPONG_0, | ||||
| 			atomic_read(&phys_enc->pending_kickoff_cnt)); | ||||
| } | ||||
| 
 | ||||
| static int _dpu_encoder_phys_cmd_wait_for_ctl_start( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = | ||||
| 			to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 	struct dpu_encoder_wait_info wait_info; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_ctl) { | ||||
| 		DPU_ERROR("invalid argument(s)\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	wait_info.wq = &phys_enc->pending_kickoff_wq; | ||||
| 	wait_info.atomic_cnt = &phys_enc->pending_ctlstart_cnt; | ||||
| 	wait_info.timeout_ms = KICKOFF_TIMEOUT_MS; | ||||
| 
 | ||||
| 	ret = dpu_encoder_helper_wait_for_irq(phys_enc, INTR_IDX_CTL_START, | ||||
| 			&wait_info); | ||||
| 	if (ret == -ETIMEDOUT) { | ||||
| 		DPU_ERROR_CMDENC(cmd_enc, "ctl start interrupt wait failed\n"); | ||||
| 		ret = -EINVAL; | ||||
| 	} else if (!ret) | ||||
| 		ret = 0; | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_cmd_wait_for_tx_complete( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	int rc; | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	cmd_enc = to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	rc = _dpu_encoder_phys_cmd_wait_for_idle(phys_enc); | ||||
| 	if (rc) { | ||||
| 		DRM_ERROR("failed wait_for_idle: id:%u ret:%d intf:%d\n", | ||||
| 			  DRMID(phys_enc->parent), rc, | ||||
| 			  phys_enc->intf_idx - INTF_0); | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_cmd_wait_for_commit_done( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	int rc = 0; | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	cmd_enc = to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	/* only required for master controller */ | ||||
| 	if (dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 		rc = _dpu_encoder_phys_cmd_wait_for_ctl_start(phys_enc); | ||||
| 
 | ||||
| 	/* required for both controllers */ | ||||
| 	if (!rc && cmd_enc->serialize_wait4pp) | ||||
| 		dpu_encoder_phys_cmd_prepare_for_kickoff(phys_enc, NULL); | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_cmd_wait_for_vblank( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	int rc = 0; | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc; | ||||
| 	struct dpu_encoder_wait_info wait_info; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	cmd_enc = to_dpu_encoder_phys_cmd(phys_enc); | ||||
| 
 | ||||
| 	/* only required for master controller */ | ||||
| 	if (!dpu_encoder_phys_cmd_is_master(phys_enc)) | ||||
| 		return rc; | ||||
| 
 | ||||
| 	wait_info.wq = &cmd_enc->pending_vblank_wq; | ||||
| 	wait_info.atomic_cnt = &cmd_enc->pending_vblank_cnt; | ||||
| 	wait_info.timeout_ms = _dpu_encoder_phys_cmd_get_idle_timeout(cmd_enc); | ||||
| 
 | ||||
| 	atomic_inc(&cmd_enc->pending_vblank_cnt); | ||||
| 
 | ||||
| 	rc = dpu_encoder_helper_wait_for_irq(phys_enc, INTR_IDX_RDPTR, | ||||
| 			&wait_info); | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_handle_post_kickoff( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * re-enable external TE, either for the first time after enabling | ||||
| 	 * or if disabled for Autorefresh | ||||
| 	 */ | ||||
| 	_dpu_encoder_phys_cmd_connect_te(phys_enc, true); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_trigger_start( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	dpu_encoder_helper_trigger_start(phys_enc); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_cmd_init_ops( | ||||
| 		struct dpu_encoder_phys_ops *ops) | ||||
| { | ||||
| 	ops->is_master = dpu_encoder_phys_cmd_is_master; | ||||
| 	ops->mode_set = dpu_encoder_phys_cmd_mode_set; | ||||
| 	ops->mode_fixup = dpu_encoder_phys_cmd_mode_fixup; | ||||
| 	ops->enable = dpu_encoder_phys_cmd_enable; | ||||
| 	ops->disable = dpu_encoder_phys_cmd_disable; | ||||
| 	ops->destroy = dpu_encoder_phys_cmd_destroy; | ||||
| 	ops->get_hw_resources = dpu_encoder_phys_cmd_get_hw_resources; | ||||
| 	ops->control_vblank_irq = dpu_encoder_phys_cmd_control_vblank_irq; | ||||
| 	ops->wait_for_commit_done = dpu_encoder_phys_cmd_wait_for_commit_done; | ||||
| 	ops->prepare_for_kickoff = dpu_encoder_phys_cmd_prepare_for_kickoff; | ||||
| 	ops->wait_for_tx_complete = dpu_encoder_phys_cmd_wait_for_tx_complete; | ||||
| 	ops->wait_for_vblank = dpu_encoder_phys_cmd_wait_for_vblank; | ||||
| 	ops->trigger_start = dpu_encoder_phys_cmd_trigger_start; | ||||
| 	ops->needs_single_flush = dpu_encoder_phys_cmd_needs_single_flush; | ||||
| 	ops->hw_reset = dpu_encoder_helper_hw_reset; | ||||
| 	ops->irq_control = dpu_encoder_phys_cmd_irq_control; | ||||
| 	ops->restore = dpu_encoder_phys_cmd_enable_helper; | ||||
| 	ops->prepare_idle_pc = dpu_encoder_phys_cmd_prepare_idle_pc; | ||||
| 	ops->handle_post_kickoff = dpu_encoder_phys_cmd_handle_post_kickoff; | ||||
| 	ops->get_line_count = dpu_encoder_phys_cmd_get_line_count; | ||||
| } | ||||
| 
 | ||||
| struct dpu_encoder_phys *dpu_encoder_phys_cmd_init( | ||||
| 		struct dpu_enc_phys_init_params *p) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = NULL; | ||||
| 	struct dpu_encoder_phys_cmd *cmd_enc = NULL; | ||||
| 	struct dpu_hw_mdp *hw_mdp; | ||||
| 	struct dpu_encoder_irq *irq; | ||||
| 	int i, ret = 0; | ||||
| 
 | ||||
| 	DPU_DEBUG("intf %d\n", p->intf_idx - INTF_0); | ||||
| 
 | ||||
| 	cmd_enc = kzalloc(sizeof(*cmd_enc), GFP_KERNEL); | ||||
| 	if (!cmd_enc) { | ||||
| 		ret = -ENOMEM; | ||||
| 		DPU_ERROR("failed to allocate\n"); | ||||
| 		goto fail; | ||||
| 	} | ||||
| 	phys_enc = &cmd_enc->base; | ||||
| 
 | ||||
| 	hw_mdp = dpu_rm_get_mdp(&p->dpu_kms->rm); | ||||
| 	if (IS_ERR_OR_NULL(hw_mdp)) { | ||||
| 		ret = PTR_ERR(hw_mdp); | ||||
| 		DPU_ERROR("failed to get mdptop\n"); | ||||
| 		goto fail_mdp_init; | ||||
| 	} | ||||
| 	phys_enc->hw_mdptop = hw_mdp; | ||||
| 	phys_enc->intf_idx = p->intf_idx; | ||||
| 
 | ||||
| 	dpu_encoder_phys_cmd_init_ops(&phys_enc->ops); | ||||
| 	phys_enc->parent = p->parent; | ||||
| 	phys_enc->parent_ops = p->parent_ops; | ||||
| 	phys_enc->dpu_kms = p->dpu_kms; | ||||
| 	phys_enc->split_role = p->split_role; | ||||
| 	phys_enc->intf_mode = INTF_MODE_CMD; | ||||
| 	phys_enc->enc_spinlock = p->enc_spinlock; | ||||
| 	cmd_enc->stream_sel = 0; | ||||
| 	phys_enc->enable_state = DPU_ENC_DISABLED; | ||||
| 	for (i = 0; i < INTR_IDX_MAX; i++) { | ||||
| 		irq = &phys_enc->irq[i]; | ||||
| 		INIT_LIST_HEAD(&irq->cb.list); | ||||
| 		irq->irq_idx = -EINVAL; | ||||
| 		irq->hw_idx = -EINVAL; | ||||
| 		irq->cb.arg = phys_enc; | ||||
| 	} | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_CTL_START]; | ||||
| 	irq->name = "ctl_start"; | ||||
| 	irq->intr_type = DPU_IRQ_TYPE_CTL_START; | ||||
| 	irq->intr_idx = INTR_IDX_CTL_START; | ||||
| 	irq->cb.func = dpu_encoder_phys_cmd_ctl_start_irq; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_PINGPONG]; | ||||
| 	irq->name = "pp_done"; | ||||
| 	irq->intr_type = DPU_IRQ_TYPE_PING_PONG_COMP; | ||||
| 	irq->intr_idx = INTR_IDX_PINGPONG; | ||||
| 	irq->cb.func = dpu_encoder_phys_cmd_pp_tx_done_irq; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_RDPTR]; | ||||
| 	irq->name = "pp_rd_ptr"; | ||||
| 	irq->intr_type = DPU_IRQ_TYPE_PING_PONG_RD_PTR; | ||||
| 	irq->intr_idx = INTR_IDX_RDPTR; | ||||
| 	irq->cb.func = dpu_encoder_phys_cmd_pp_rd_ptr_irq; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_UNDERRUN]; | ||||
| 	irq->name = "underrun"; | ||||
| 	irq->intr_type = DPU_IRQ_TYPE_INTF_UNDER_RUN; | ||||
| 	irq->intr_idx = INTR_IDX_UNDERRUN; | ||||
| 	irq->cb.func = dpu_encoder_phys_cmd_underrun_irq; | ||||
| 
 | ||||
| 	atomic_set(&phys_enc->vblank_refcount, 0); | ||||
| 	atomic_set(&phys_enc->pending_kickoff_cnt, 0); | ||||
| 	atomic_set(&phys_enc->pending_ctlstart_cnt, 0); | ||||
| 	atomic_set(&cmd_enc->pending_vblank_cnt, 0); | ||||
| 	init_waitqueue_head(&phys_enc->pending_kickoff_wq); | ||||
| 	init_waitqueue_head(&cmd_enc->pending_vblank_wq); | ||||
| 
 | ||||
| 	DPU_DEBUG_CMDENC(cmd_enc, "created\n"); | ||||
| 
 | ||||
| 	return phys_enc; | ||||
| 
 | ||||
| fail_mdp_init: | ||||
| 	kfree(cmd_enc); | ||||
| fail: | ||||
| 	return ERR_PTR(ret); | ||||
| } | ||||
							
								
								
									
										922
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder_phys_vid.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										922
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_encoder_phys_vid.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,922 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| #include "dpu_encoder_phys.h" | ||||
| #include "dpu_hw_interrupts.h" | ||||
| #include "dpu_core_irq.h" | ||||
| #include "dpu_formats.h" | ||||
| #include "dpu_trace.h" | ||||
| 
 | ||||
| #define DPU_DEBUG_VIDENC(e, fmt, ...) DPU_DEBUG("enc%d intf%d " fmt, \ | ||||
| 		(e) && (e)->base.parent ? \ | ||||
| 		(e)->base.parent->base.id : -1, \ | ||||
| 		(e) && (e)->hw_intf ? \ | ||||
| 		(e)->hw_intf->idx - INTF_0 : -1, ##__VA_ARGS__) | ||||
| 
 | ||||
| #define DPU_ERROR_VIDENC(e, fmt, ...) DPU_ERROR("enc%d intf%d " fmt, \ | ||||
| 		(e) && (e)->base.parent ? \ | ||||
| 		(e)->base.parent->base.id : -1, \ | ||||
| 		(e) && (e)->hw_intf ? \ | ||||
| 		(e)->hw_intf->idx - INTF_0 : -1, ##__VA_ARGS__) | ||||
| 
 | ||||
| #define to_dpu_encoder_phys_vid(x) \ | ||||
| 	container_of(x, struct dpu_encoder_phys_vid, base) | ||||
| 
 | ||||
| static bool dpu_encoder_phys_vid_is_master( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	bool ret = false; | ||||
| 
 | ||||
| 	if (phys_enc->split_role != ENC_ROLE_SLAVE) | ||||
| 		ret = true; | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static void drm_mode_to_intf_timing_params( | ||||
| 		const struct dpu_encoder_phys_vid *vid_enc, | ||||
| 		const struct drm_display_mode *mode, | ||||
| 		struct intf_timing_params *timing) | ||||
| { | ||||
| 	memset(timing, 0, sizeof(*timing)); | ||||
| 
 | ||||
| 	if ((mode->htotal < mode->hsync_end) | ||||
| 			|| (mode->hsync_start < mode->hdisplay) | ||||
| 			|| (mode->vtotal < mode->vsync_end) | ||||
| 			|| (mode->vsync_start < mode->vdisplay) | ||||
| 			|| (mode->hsync_end < mode->hsync_start) | ||||
| 			|| (mode->vsync_end < mode->vsync_start)) { | ||||
| 		DPU_ERROR( | ||||
| 		    "invalid params - hstart:%d,hend:%d,htot:%d,hdisplay:%d\n", | ||||
| 				mode->hsync_start, mode->hsync_end, | ||||
| 				mode->htotal, mode->hdisplay); | ||||
| 		DPU_ERROR("vstart:%d,vend:%d,vtot:%d,vdisplay:%d\n", | ||||
| 				mode->vsync_start, mode->vsync_end, | ||||
| 				mode->vtotal, mode->vdisplay); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * https://www.kernel.org/doc/htmldocs/drm/ch02s05.html
 | ||||
| 	 *  Active Region      Front Porch   Sync   Back Porch | ||||
| 	 * <-----------------><------------><-----><-----------> | ||||
| 	 * <- [hv]display ---> | ||||
| 	 * <--------- [hv]sync_start ------> | ||||
| 	 * <----------------- [hv]sync_end -------> | ||||
| 	 * <---------------------------- [hv]total -------------> | ||||
| 	 */ | ||||
| 	timing->width = mode->hdisplay;	/* active width */ | ||||
| 	timing->height = mode->vdisplay;	/* active height */ | ||||
| 	timing->xres = timing->width; | ||||
| 	timing->yres = timing->height; | ||||
| 	timing->h_back_porch = mode->htotal - mode->hsync_end; | ||||
| 	timing->h_front_porch = mode->hsync_start - mode->hdisplay; | ||||
| 	timing->v_back_porch = mode->vtotal - mode->vsync_end; | ||||
| 	timing->v_front_porch = mode->vsync_start - mode->vdisplay; | ||||
| 	timing->hsync_pulse_width = mode->hsync_end - mode->hsync_start; | ||||
| 	timing->vsync_pulse_width = mode->vsync_end - mode->vsync_start; | ||||
| 	timing->hsync_polarity = (mode->flags & DRM_MODE_FLAG_NHSYNC) ? 1 : 0; | ||||
| 	timing->vsync_polarity = (mode->flags & DRM_MODE_FLAG_NVSYNC) ? 1 : 0; | ||||
| 	timing->border_clr = 0; | ||||
| 	timing->underflow_clr = 0xff; | ||||
| 	timing->hsync_skew = mode->hskew; | ||||
| 
 | ||||
| 	/* DSI controller cannot handle active-low sync signals. */ | ||||
| 	if (vid_enc->hw_intf->cap->type == INTF_DSI) { | ||||
| 		timing->hsync_polarity = 0; | ||||
| 		timing->vsync_polarity = 0; | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * For edp only: | ||||
| 	 * DISPLAY_V_START = (VBP * HCYCLE) + HBP | ||||
| 	 * DISPLAY_V_END = (VBP + VACTIVE) * HCYCLE - 1 - HFP | ||||
| 	 */ | ||||
| 	/*
 | ||||
| 	 * if (vid_enc->hw->cap->type == INTF_EDP) { | ||||
| 	 * display_v_start += mode->htotal - mode->hsync_start; | ||||
| 	 * display_v_end -= mode->hsync_start - mode->hdisplay; | ||||
| 	 * } | ||||
| 	 */ | ||||
| } | ||||
| 
 | ||||
| static inline u32 get_horizontal_total(const struct intf_timing_params *timing) | ||||
| { | ||||
| 	u32 active = timing->xres; | ||||
| 	u32 inactive = | ||||
| 	    timing->h_back_porch + timing->h_front_porch + | ||||
| 	    timing->hsync_pulse_width; | ||||
| 	return active + inactive; | ||||
| } | ||||
| 
 | ||||
| static inline u32 get_vertical_total(const struct intf_timing_params *timing) | ||||
| { | ||||
| 	u32 active = timing->yres; | ||||
| 	u32 inactive = | ||||
| 	    timing->v_back_porch + timing->v_front_porch + | ||||
| 	    timing->vsync_pulse_width; | ||||
| 	return active + inactive; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * programmable_fetch_get_num_lines: | ||||
|  *	Number of fetch lines in vertical front porch | ||||
|  * @timing: Pointer to the intf timing information for the requested mode | ||||
|  * | ||||
|  * Returns the number of fetch lines in vertical front porch at which mdp | ||||
|  * can start fetching the next frame. | ||||
|  * | ||||
|  * Number of needed prefetch lines is anything that cannot be absorbed in the | ||||
|  * start of frame time (back porch + vsync pulse width). | ||||
|  * | ||||
|  * Some panels have very large VFP, however we only need a total number of | ||||
|  * lines based on the chip worst case latencies. | ||||
|  */ | ||||
| static u32 programmable_fetch_get_num_lines( | ||||
| 		struct dpu_encoder_phys_vid *vid_enc, | ||||
| 		const struct intf_timing_params *timing) | ||||
| { | ||||
| 	u32 worst_case_needed_lines = | ||||
| 	    vid_enc->hw_intf->cap->prog_fetch_lines_worst_case; | ||||
| 	u32 start_of_frame_lines = | ||||
| 	    timing->v_back_porch + timing->vsync_pulse_width; | ||||
| 	u32 needed_vfp_lines = worst_case_needed_lines - start_of_frame_lines; | ||||
| 	u32 actual_vfp_lines = 0; | ||||
| 
 | ||||
| 	/* Fetch must be outside active lines, otherwise undefined. */ | ||||
| 	if (start_of_frame_lines >= worst_case_needed_lines) { | ||||
| 		DPU_DEBUG_VIDENC(vid_enc, | ||||
| 				"prog fetch is not needed, large vbp+vsw\n"); | ||||
| 		actual_vfp_lines = 0; | ||||
| 	} else if (timing->v_front_porch < needed_vfp_lines) { | ||||
| 		/* Warn fetch needed, but not enough porch in panel config */ | ||||
| 		pr_warn_once | ||||
| 			("low vbp+vfp may lead to perf issues in some cases\n"); | ||||
| 		DPU_DEBUG_VIDENC(vid_enc, | ||||
| 				"less vfp than fetch req, using entire vfp\n"); | ||||
| 		actual_vfp_lines = timing->v_front_porch; | ||||
| 	} else { | ||||
| 		DPU_DEBUG_VIDENC(vid_enc, "room in vfp for needed prefetch\n"); | ||||
| 		actual_vfp_lines = needed_vfp_lines; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, | ||||
| 		"v_front_porch %u v_back_porch %u vsync_pulse_width %u\n", | ||||
| 		timing->v_front_porch, timing->v_back_porch, | ||||
| 		timing->vsync_pulse_width); | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, | ||||
| 		"wc_lines %u needed_vfp_lines %u actual_vfp_lines %u\n", | ||||
| 		worst_case_needed_lines, needed_vfp_lines, actual_vfp_lines); | ||||
| 
 | ||||
| 	return actual_vfp_lines; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * programmable_fetch_config: Programs HW to prefetch lines by offsetting | ||||
|  *	the start of fetch into the vertical front porch for cases where the | ||||
|  *	vsync pulse width and vertical back porch time is insufficient | ||||
|  * | ||||
|  *	Gets # of lines to pre-fetch, then calculate VSYNC counter value. | ||||
|  *	HW layer requires VSYNC counter of first pixel of tgt VFP line. | ||||
|  * | ||||
|  * @timing: Pointer to the intf timing information for the requested mode | ||||
|  */ | ||||
| static void programmable_fetch_config(struct dpu_encoder_phys *phys_enc, | ||||
| 				      const struct intf_timing_params *timing) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc = | ||||
| 		to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	struct intf_prog_fetch f = { 0 }; | ||||
| 	u32 vfp_fetch_lines = 0; | ||||
| 	u32 horiz_total = 0; | ||||
| 	u32 vert_total = 0; | ||||
| 	u32 vfp_fetch_start_vsync_counter = 0; | ||||
| 	unsigned long lock_flags; | ||||
| 
 | ||||
| 	if (WARN_ON_ONCE(!vid_enc->hw_intf->ops.setup_prg_fetch)) | ||||
| 		return; | ||||
| 
 | ||||
| 	vfp_fetch_lines = programmable_fetch_get_num_lines(vid_enc, timing); | ||||
| 	if (vfp_fetch_lines) { | ||||
| 		vert_total = get_vertical_total(timing); | ||||
| 		horiz_total = get_horizontal_total(timing); | ||||
| 		vfp_fetch_start_vsync_counter = | ||||
| 		    (vert_total - vfp_fetch_lines) * horiz_total + 1; | ||||
| 		f.enable = 1; | ||||
| 		f.fetch_start = vfp_fetch_start_vsync_counter; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, | ||||
| 		"vfp_fetch_lines %u vfp_fetch_start_vsync_counter %u\n", | ||||
| 		vfp_fetch_lines, vfp_fetch_start_vsync_counter); | ||||
| 
 | ||||
| 	spin_lock_irqsave(phys_enc->enc_spinlock, lock_flags); | ||||
| 	vid_enc->hw_intf->ops.setup_prg_fetch(vid_enc->hw_intf, &f); | ||||
| 	spin_unlock_irqrestore(phys_enc->enc_spinlock, lock_flags); | ||||
| } | ||||
| 
 | ||||
| static bool dpu_encoder_phys_vid_mode_fixup( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		const struct drm_display_mode *mode, | ||||
| 		struct drm_display_mode *adj_mode) | ||||
| { | ||||
| 	if (phys_enc) | ||||
| 		DPU_DEBUG_VIDENC(to_dpu_encoder_phys_vid(phys_enc), "\n"); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Modifying mode has consequences when the mode comes back to us | ||||
| 	 */ | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_setup_timing_engine( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 	struct drm_display_mode mode; | ||||
| 	struct intf_timing_params timing_params = { 0 }; | ||||
| 	const struct dpu_format *fmt = NULL; | ||||
| 	u32 fmt_fourcc = DRM_FORMAT_RGB888; | ||||
| 	unsigned long lock_flags; | ||||
| 	struct dpu_hw_intf_cfg intf_cfg = { 0 }; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->hw_ctl->ops.setup_intf_cfg) { | ||||
| 		DPU_ERROR("invalid encoder %d\n", phys_enc != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	mode = phys_enc->cached_mode; | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	if (!vid_enc->hw_intf->ops.setup_timing_gen) { | ||||
| 		DPU_ERROR("timing engine setup is not supported\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "enabling mode:\n"); | ||||
| 	drm_mode_debug_printmodeline(&mode); | ||||
| 
 | ||||
| 	if (phys_enc->split_role != ENC_ROLE_SOLO) { | ||||
| 		mode.hdisplay >>= 1; | ||||
| 		mode.htotal >>= 1; | ||||
| 		mode.hsync_start >>= 1; | ||||
| 		mode.hsync_end >>= 1; | ||||
| 
 | ||||
| 		DPU_DEBUG_VIDENC(vid_enc, | ||||
| 			"split_role %d, halve horizontal %d %d %d %d\n", | ||||
| 			phys_enc->split_role, | ||||
| 			mode.hdisplay, mode.htotal, | ||||
| 			mode.hsync_start, mode.hsync_end); | ||||
| 	} | ||||
| 
 | ||||
| 	drm_mode_to_intf_timing_params(vid_enc, &mode, &timing_params); | ||||
| 
 | ||||
| 	fmt = dpu_get_dpu_format(fmt_fourcc); | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "fmt_fourcc 0x%X\n", fmt_fourcc); | ||||
| 
 | ||||
| 	intf_cfg.intf = vid_enc->hw_intf->idx; | ||||
| 	intf_cfg.intf_mode_sel = DPU_CTL_MODE_SEL_VID; | ||||
| 	intf_cfg.stream_sel = 0; /* Don't care value for video mode */ | ||||
| 	intf_cfg.mode_3d = dpu_encoder_helper_get_3d_blend_mode(phys_enc); | ||||
| 
 | ||||
| 	spin_lock_irqsave(phys_enc->enc_spinlock, lock_flags); | ||||
| 	vid_enc->hw_intf->ops.setup_timing_gen(vid_enc->hw_intf, | ||||
| 			&timing_params, fmt); | ||||
| 	phys_enc->hw_ctl->ops.setup_intf_cfg(phys_enc->hw_ctl, &intf_cfg); | ||||
| 	spin_unlock_irqrestore(phys_enc->enc_spinlock, lock_flags); | ||||
| 
 | ||||
| 	programmable_fetch_config(phys_enc, &timing_params); | ||||
| 
 | ||||
| 	vid_enc->timing_params = timing_params; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_vblank_irq(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = arg; | ||||
| 	struct dpu_hw_ctl *hw_ctl; | ||||
| 	unsigned long lock_flags; | ||||
| 	u32 flush_register = 0; | ||||
| 	int new_cnt = -1, old_cnt = -1; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	hw_ctl = phys_enc->hw_ctl; | ||||
| 	if (!hw_ctl) | ||||
| 		return; | ||||
| 
 | ||||
| 	DPU_ATRACE_BEGIN("vblank_irq"); | ||||
| 
 | ||||
| 	if (phys_enc->parent_ops->handle_vblank_virt) | ||||
| 		phys_enc->parent_ops->handle_vblank_virt(phys_enc->parent, | ||||
| 				phys_enc); | ||||
| 
 | ||||
| 	old_cnt  = atomic_read(&phys_enc->pending_kickoff_cnt); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * only decrement the pending flush count if we've actually flushed | ||||
| 	 * hardware. due to sw irq latency, vblank may have already happened | ||||
| 	 * so we need to double-check with hw that it accepted the flush bits | ||||
| 	 */ | ||||
| 	spin_lock_irqsave(phys_enc->enc_spinlock, lock_flags); | ||||
| 	if (hw_ctl && hw_ctl->ops.get_flush_register) | ||||
| 		flush_register = hw_ctl->ops.get_flush_register(hw_ctl); | ||||
| 
 | ||||
| 	if (flush_register == 0) | ||||
| 		new_cnt = atomic_add_unless(&phys_enc->pending_kickoff_cnt, | ||||
| 				-1, 0); | ||||
| 	spin_unlock_irqrestore(phys_enc->enc_spinlock, lock_flags); | ||||
| 
 | ||||
| 	/* Signal any waiting atomic commit thread */ | ||||
| 	wake_up_all(&phys_enc->pending_kickoff_wq); | ||||
| 	DPU_ATRACE_END("vblank_irq"); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_underrun_irq(void *arg, int irq_idx) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = arg; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (phys_enc->parent_ops->handle_underrun_virt) | ||||
| 		phys_enc->parent_ops->handle_underrun_virt(phys_enc->parent, | ||||
| 			phys_enc); | ||||
| } | ||||
| 
 | ||||
| static bool _dpu_encoder_phys_is_dual_ctl(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	if (!phys_enc) | ||||
| 		return false; | ||||
| 
 | ||||
| 	if (phys_enc->topology_name == DPU_RM_TOPOLOGY_DUALPIPE) | ||||
| 		return true; | ||||
| 
 | ||||
| 	return false; | ||||
| } | ||||
| 
 | ||||
| static bool dpu_encoder_phys_vid_needs_single_flush( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	return (phys_enc && _dpu_encoder_phys_is_dual_ctl(phys_enc)); | ||||
| } | ||||
| 
 | ||||
| static void _dpu_encoder_phys_vid_setup_irq_hw_idx( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_irq *irq; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Initialize irq->hw_idx only when irq is not registered. | ||||
| 	 * Prevent invalidating irq->irq_idx as modeset may be | ||||
| 	 * called many times during dfps. | ||||
| 	 */ | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_VSYNC]; | ||||
| 	if (irq->irq_idx < 0) | ||||
| 		irq->hw_idx = phys_enc->intf_idx; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_UNDERRUN]; | ||||
| 	if (irq->irq_idx < 0) | ||||
| 		irq->hw_idx = phys_enc->intf_idx; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_mode_set( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		struct drm_display_mode *mode, | ||||
| 		struct drm_display_mode *adj_mode) | ||||
| { | ||||
| 	struct dpu_rm *rm; | ||||
| 	struct dpu_rm_hw_iter iter; | ||||
| 	int i, instance; | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->dpu_kms) { | ||||
| 		DPU_ERROR("invalid encoder/kms\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	rm = &phys_enc->dpu_kms->rm; | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 
 | ||||
| 	if (adj_mode) { | ||||
| 		phys_enc->cached_mode = *adj_mode; | ||||
| 		drm_mode_debug_printmodeline(adj_mode); | ||||
| 		DPU_DEBUG_VIDENC(vid_enc, "caching mode:\n"); | ||||
| 	} | ||||
| 
 | ||||
| 	instance = phys_enc->split_role == ENC_ROLE_SLAVE ? 1 : 0; | ||||
| 
 | ||||
| 	/* Retrieve previously allocated HW Resources. Shouldn't fail */ | ||||
| 	dpu_rm_init_hw_iter(&iter, phys_enc->parent->base.id, DPU_HW_BLK_CTL); | ||||
| 	for (i = 0; i <= instance; i++) { | ||||
| 		if (dpu_rm_get_hw(rm, &iter)) | ||||
| 			phys_enc->hw_ctl = (struct dpu_hw_ctl *)iter.hw; | ||||
| 	} | ||||
| 	if (IS_ERR_OR_NULL(phys_enc->hw_ctl)) { | ||||
| 		DPU_ERROR_VIDENC(vid_enc, "failed to init ctl, %ld\n", | ||||
| 				PTR_ERR(phys_enc->hw_ctl)); | ||||
| 		phys_enc->hw_ctl = NULL; | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	_dpu_encoder_phys_vid_setup_irq_hw_idx(phys_enc); | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_vid_control_vblank_irq( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		bool enable) | ||||
| { | ||||
| 	int ret = 0; | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 	int refcount; | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	refcount = atomic_read(&phys_enc->vblank_refcount); | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 
 | ||||
| 	/* Slave encoders don't report vblank */ | ||||
| 	if (!dpu_encoder_phys_vid_is_master(phys_enc)) | ||||
| 		goto end; | ||||
| 
 | ||||
| 	/* protect against negative */ | ||||
| 	if (!enable && refcount == 0) { | ||||
| 		ret = -EINVAL; | ||||
| 		goto end; | ||||
| 	} | ||||
| 
 | ||||
| 	DRM_DEBUG_KMS("id:%u enable=%d/%d\n", DRMID(phys_enc->parent), enable, | ||||
| 		      atomic_read(&phys_enc->vblank_refcount)); | ||||
| 
 | ||||
| 	if (enable && atomic_inc_return(&phys_enc->vblank_refcount) == 1) | ||||
| 		ret = dpu_encoder_helper_register_irq(phys_enc, INTR_IDX_VSYNC); | ||||
| 	else if (!enable && atomic_dec_return(&phys_enc->vblank_refcount) == 0) | ||||
| 		ret = dpu_encoder_helper_unregister_irq(phys_enc, | ||||
| 				INTR_IDX_VSYNC); | ||||
| 
 | ||||
| end: | ||||
| 	if (ret) { | ||||
| 		DRM_ERROR("failed: id:%u intf:%d ret:%d enable:%d refcnt:%d\n", | ||||
| 			  DRMID(phys_enc->parent), | ||||
| 			  vid_enc->hw_intf->idx - INTF_0, ret, enable, | ||||
| 			  refcount); | ||||
| 	} | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_enable(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct msm_drm_private *priv; | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 	struct dpu_hw_intf *intf; | ||||
| 	struct dpu_hw_ctl *ctl; | ||||
| 	u32 flush_mask = 0; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->parent || !phys_enc->parent->dev || | ||||
| 			!phys_enc->parent->dev->dev_private) { | ||||
| 		DPU_ERROR("invalid encoder/device\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	priv = phys_enc->parent->dev->dev_private; | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	intf = vid_enc->hw_intf; | ||||
| 	ctl = phys_enc->hw_ctl; | ||||
| 	if (!vid_enc->hw_intf || !phys_enc->hw_ctl) { | ||||
| 		DPU_ERROR("invalid hw_intf %d hw_ctl %d\n", | ||||
| 				vid_enc->hw_intf != 0, phys_enc->hw_ctl != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "\n"); | ||||
| 
 | ||||
| 	if (WARN_ON(!vid_enc->hw_intf->ops.enable_timing)) | ||||
| 		return; | ||||
| 
 | ||||
| 	dpu_encoder_helper_split_config(phys_enc, vid_enc->hw_intf->idx); | ||||
| 
 | ||||
| 	dpu_encoder_phys_vid_setup_timing_engine(phys_enc); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * For single flush cases (dual-ctl or pp-split), skip setting the | ||||
| 	 * flush bit for the slave intf, since both intfs use same ctl | ||||
| 	 * and HW will only flush the master. | ||||
| 	 */ | ||||
| 	if (dpu_encoder_phys_vid_needs_single_flush(phys_enc) && | ||||
| 		!dpu_encoder_phys_vid_is_master(phys_enc)) | ||||
| 		goto skip_flush; | ||||
| 
 | ||||
| 	ctl->ops.get_bitmask_intf(ctl, &flush_mask, intf->idx); | ||||
| 	ctl->ops.update_pending_flush(ctl, flush_mask); | ||||
| 
 | ||||
| skip_flush: | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "update pending flush ctl %d flush_mask %x\n", | ||||
| 		ctl->idx - CTL_0, flush_mask); | ||||
| 
 | ||||
| 	/* ctl_flush & timing engine enable will be triggered by framework */ | ||||
| 	if (phys_enc->enable_state == DPU_ENC_DISABLED) | ||||
| 		phys_enc->enable_state = DPU_ENC_ENABLING; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_destroy(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "\n"); | ||||
| 	kfree(vid_enc); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_get_hw_resources( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		struct dpu_encoder_hw_resources *hw_res, | ||||
| 		struct drm_connector_state *conn_state) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc || !hw_res) { | ||||
| 		DPU_ERROR("invalid arg(s), enc %d hw_res %d conn_state %d\n", | ||||
| 				phys_enc != 0, hw_res != 0, conn_state != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	if (!vid_enc->hw_intf) { | ||||
| 		DPU_ERROR("invalid arg(s), hw_intf\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "\n"); | ||||
| 	hw_res->intfs[vid_enc->hw_intf->idx - INTF_0] = INTF_MODE_VIDEO; | ||||
| } | ||||
| 
 | ||||
| static int _dpu_encoder_phys_vid_wait_for_vblank( | ||||
| 		struct dpu_encoder_phys *phys_enc, bool notify) | ||||
| { | ||||
| 	struct dpu_encoder_wait_info wait_info; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		pr_err("invalid encoder\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	wait_info.wq = &phys_enc->pending_kickoff_wq; | ||||
| 	wait_info.atomic_cnt = &phys_enc->pending_kickoff_cnt; | ||||
| 	wait_info.timeout_ms = KICKOFF_TIMEOUT_MS; | ||||
| 
 | ||||
| 	if (!dpu_encoder_phys_vid_is_master(phys_enc)) { | ||||
| 		if (notify && phys_enc->parent_ops->handle_frame_done) | ||||
| 			phys_enc->parent_ops->handle_frame_done( | ||||
| 					phys_enc->parent, phys_enc, | ||||
| 					DPU_ENCODER_FRAME_EVENT_DONE); | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	/* Wait for kickoff to complete */ | ||||
| 	ret = dpu_encoder_helper_wait_for_irq(phys_enc, INTR_IDX_VSYNC, | ||||
| 			&wait_info); | ||||
| 
 | ||||
| 	if (ret == -ETIMEDOUT) { | ||||
| 		dpu_encoder_helper_report_irq_timeout(phys_enc, INTR_IDX_VSYNC); | ||||
| 	} else if (!ret && notify && phys_enc->parent_ops->handle_frame_done) | ||||
| 		phys_enc->parent_ops->handle_frame_done( | ||||
| 				phys_enc->parent, phys_enc, | ||||
| 				DPU_ENCODER_FRAME_EVENT_DONE); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_vid_wait_for_vblank( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	return _dpu_encoder_phys_vid_wait_for_vblank(phys_enc, true); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_prepare_for_kickoff( | ||||
| 		struct dpu_encoder_phys *phys_enc, | ||||
| 		struct dpu_encoder_kickoff_params *params) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 	struct dpu_hw_ctl *ctl; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (!phys_enc || !params) { | ||||
| 		DPU_ERROR("invalid encoder/parameters\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 
 | ||||
| 	ctl = phys_enc->hw_ctl; | ||||
| 	if (!ctl || !ctl->ops.wait_reset_status) | ||||
| 		return; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * hw supports hardware initiated ctl reset, so before we kickoff a new | ||||
| 	 * frame, need to check and wait for hw initiated ctl reset completion | ||||
| 	 */ | ||||
| 	rc = ctl->ops.wait_reset_status(ctl); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR_VIDENC(vid_enc, "ctl %d reset failure: %d\n", | ||||
| 				ctl->idx, rc); | ||||
| 		dpu_encoder_helper_unregister_irq(phys_enc, INTR_IDX_VSYNC); | ||||
| 		dpu_dbg_dump(false, __func__, true, true); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_disable(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct msm_drm_private *priv; | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 	unsigned long lock_flags; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!phys_enc || !phys_enc->parent || !phys_enc->parent->dev || | ||||
| 			!phys_enc->parent->dev->dev_private) { | ||||
| 		DPU_ERROR("invalid encoder/device\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	priv = phys_enc->parent->dev->dev_private; | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	if (!vid_enc->hw_intf || !phys_enc->hw_ctl) { | ||||
| 		DPU_ERROR("invalid hw_intf %d hw_ctl %d\n", | ||||
| 				vid_enc->hw_intf != 0, phys_enc->hw_ctl != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "\n"); | ||||
| 
 | ||||
| 	if (WARN_ON(!vid_enc->hw_intf->ops.enable_timing)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (phys_enc->enable_state == DPU_ENC_DISABLED) { | ||||
| 		DPU_ERROR("already disabled\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	spin_lock_irqsave(phys_enc->enc_spinlock, lock_flags); | ||||
| 	vid_enc->hw_intf->ops.enable_timing(vid_enc->hw_intf, 0); | ||||
| 	if (dpu_encoder_phys_vid_is_master(phys_enc)) | ||||
| 		dpu_encoder_phys_inc_pending(phys_enc); | ||||
| 	spin_unlock_irqrestore(phys_enc->enc_spinlock, lock_flags); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Wait for a vsync so we know the ENABLE=0 latched before | ||||
| 	 * the (connector) source of the vsync's gets disabled, | ||||
| 	 * otherwise we end up in a funny state if we re-enable | ||||
| 	 * before the disable latches, which results that some of | ||||
| 	 * the settings changes for the new modeset (like new | ||||
| 	 * scanout buffer) don't latch properly.. | ||||
| 	 */ | ||||
| 	if (dpu_encoder_phys_vid_is_master(phys_enc)) { | ||||
| 		ret = _dpu_encoder_phys_vid_wait_for_vblank(phys_enc, false); | ||||
| 		if (ret) { | ||||
| 			atomic_set(&phys_enc->pending_kickoff_cnt, 0); | ||||
| 			DRM_ERROR("wait disable failed: id:%u intf:%d ret:%d\n", | ||||
| 				  DRMID(phys_enc->parent), | ||||
| 				  vid_enc->hw_intf->idx - INTF_0, ret); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	phys_enc->enable_state = DPU_ENC_DISABLED; | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_handle_post_kickoff( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	unsigned long lock_flags; | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) { | ||||
| 		DPU_ERROR("invalid encoder\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "enable_state %d\n", phys_enc->enable_state); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Video mode must flush CTL before enabling timing engine | ||||
| 	 * Video encoders need to turn on their interfaces now | ||||
| 	 */ | ||||
| 	if (phys_enc->enable_state == DPU_ENC_ENABLING) { | ||||
| 		trace_dpu_enc_phys_vid_post_kickoff(DRMID(phys_enc->parent), | ||||
| 				    vid_enc->hw_intf->idx - INTF_0); | ||||
| 		spin_lock_irqsave(phys_enc->enc_spinlock, lock_flags); | ||||
| 		vid_enc->hw_intf->ops.enable_timing(vid_enc->hw_intf, 1); | ||||
| 		spin_unlock_irqrestore(phys_enc->enc_spinlock, lock_flags); | ||||
| 		phys_enc->enable_state = DPU_ENC_ENABLED; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_irq_control(struct dpu_encoder_phys *phys_enc, | ||||
| 		bool enable) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 
 | ||||
| 	trace_dpu_enc_phys_vid_irq_ctrl(DRMID(phys_enc->parent), | ||||
| 			    vid_enc->hw_intf->idx - INTF_0, | ||||
| 			    enable, | ||||
| 			    atomic_read(&phys_enc->vblank_refcount)); | ||||
| 
 | ||||
| 	if (enable) { | ||||
| 		ret = dpu_encoder_phys_vid_control_vblank_irq(phys_enc, true); | ||||
| 		if (ret) | ||||
| 			return; | ||||
| 
 | ||||
| 		dpu_encoder_helper_register_irq(phys_enc, INTR_IDX_UNDERRUN); | ||||
| 	} else { | ||||
| 		dpu_encoder_phys_vid_control_vblank_irq(phys_enc, false); | ||||
| 		dpu_encoder_helper_unregister_irq(phys_enc, INTR_IDX_UNDERRUN); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_setup_misr(struct dpu_encoder_phys *phys_enc, | ||||
| 						bool enable, u32 frame_count) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return; | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 
 | ||||
| 	if (vid_enc->hw_intf && vid_enc->hw_intf->ops.setup_misr) | ||||
| 		vid_enc->hw_intf->ops.setup_misr(vid_enc->hw_intf, | ||||
| 							enable, frame_count); | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_encoder_phys_vid_collect_misr(struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return 0; | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 
 | ||||
| 	return vid_enc->hw_intf && vid_enc->hw_intf->ops.collect_misr ? | ||||
| 		vid_enc->hw_intf->ops.collect_misr(vid_enc->hw_intf) : 0; | ||||
| } | ||||
| 
 | ||||
| static int dpu_encoder_phys_vid_get_line_count( | ||||
| 		struct dpu_encoder_phys *phys_enc) | ||||
| { | ||||
| 	struct dpu_encoder_phys_vid *vid_enc; | ||||
| 
 | ||||
| 	if (!phys_enc) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (!dpu_encoder_phys_vid_is_master(phys_enc)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	vid_enc = to_dpu_encoder_phys_vid(phys_enc); | ||||
| 	if (!vid_enc->hw_intf || !vid_enc->hw_intf->ops.get_line_count) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	return vid_enc->hw_intf->ops.get_line_count(vid_enc->hw_intf); | ||||
| } | ||||
| 
 | ||||
| static void dpu_encoder_phys_vid_init_ops(struct dpu_encoder_phys_ops *ops) | ||||
| { | ||||
| 	ops->is_master = dpu_encoder_phys_vid_is_master; | ||||
| 	ops->mode_set = dpu_encoder_phys_vid_mode_set; | ||||
| 	ops->mode_fixup = dpu_encoder_phys_vid_mode_fixup; | ||||
| 	ops->enable = dpu_encoder_phys_vid_enable; | ||||
| 	ops->disable = dpu_encoder_phys_vid_disable; | ||||
| 	ops->destroy = dpu_encoder_phys_vid_destroy; | ||||
| 	ops->get_hw_resources = dpu_encoder_phys_vid_get_hw_resources; | ||||
| 	ops->control_vblank_irq = dpu_encoder_phys_vid_control_vblank_irq; | ||||
| 	ops->wait_for_commit_done = dpu_encoder_phys_vid_wait_for_vblank; | ||||
| 	ops->wait_for_vblank = dpu_encoder_phys_vid_wait_for_vblank; | ||||
| 	ops->wait_for_tx_complete = dpu_encoder_phys_vid_wait_for_vblank; | ||||
| 	ops->irq_control = dpu_encoder_phys_vid_irq_control; | ||||
| 	ops->prepare_for_kickoff = dpu_encoder_phys_vid_prepare_for_kickoff; | ||||
| 	ops->handle_post_kickoff = dpu_encoder_phys_vid_handle_post_kickoff; | ||||
| 	ops->needs_single_flush = dpu_encoder_phys_vid_needs_single_flush; | ||||
| 	ops->setup_misr = dpu_encoder_phys_vid_setup_misr; | ||||
| 	ops->collect_misr = dpu_encoder_phys_vid_collect_misr; | ||||
| 	ops->hw_reset = dpu_encoder_helper_hw_reset; | ||||
| 	ops->get_line_count = dpu_encoder_phys_vid_get_line_count; | ||||
| } | ||||
| 
 | ||||
| struct dpu_encoder_phys *dpu_encoder_phys_vid_init( | ||||
| 		struct dpu_enc_phys_init_params *p) | ||||
| { | ||||
| 	struct dpu_encoder_phys *phys_enc = NULL; | ||||
| 	struct dpu_encoder_phys_vid *vid_enc = NULL; | ||||
| 	struct dpu_rm_hw_iter iter; | ||||
| 	struct dpu_hw_mdp *hw_mdp; | ||||
| 	struct dpu_encoder_irq *irq; | ||||
| 	int i, ret = 0; | ||||
| 
 | ||||
| 	if (!p) { | ||||
| 		ret = -EINVAL; | ||||
| 		goto fail; | ||||
| 	} | ||||
| 
 | ||||
| 	vid_enc = kzalloc(sizeof(*vid_enc), GFP_KERNEL); | ||||
| 	if (!vid_enc) { | ||||
| 		ret = -ENOMEM; | ||||
| 		goto fail; | ||||
| 	} | ||||
| 
 | ||||
| 	phys_enc = &vid_enc->base; | ||||
| 
 | ||||
| 	hw_mdp = dpu_rm_get_mdp(&p->dpu_kms->rm); | ||||
| 	if (IS_ERR_OR_NULL(hw_mdp)) { | ||||
| 		ret = PTR_ERR(hw_mdp); | ||||
| 		DPU_ERROR("failed to get mdptop\n"); | ||||
| 		goto fail; | ||||
| 	} | ||||
| 	phys_enc->hw_mdptop = hw_mdp; | ||||
| 	phys_enc->intf_idx = p->intf_idx; | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * hw_intf resource permanently assigned to this encoder | ||||
| 	 * Other resources allocated at atomic commit time by use case | ||||
| 	 */ | ||||
| 	dpu_rm_init_hw_iter(&iter, 0, DPU_HW_BLK_INTF); | ||||
| 	while (dpu_rm_get_hw(&p->dpu_kms->rm, &iter)) { | ||||
| 		struct dpu_hw_intf *hw_intf = (struct dpu_hw_intf *)iter.hw; | ||||
| 
 | ||||
| 		if (hw_intf->idx == p->intf_idx) { | ||||
| 			vid_enc->hw_intf = hw_intf; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if (!vid_enc->hw_intf) { | ||||
| 		ret = -EINVAL; | ||||
| 		DPU_ERROR("failed to get hw_intf\n"); | ||||
| 		goto fail; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "\n"); | ||||
| 
 | ||||
| 	dpu_encoder_phys_vid_init_ops(&phys_enc->ops); | ||||
| 	phys_enc->parent = p->parent; | ||||
| 	phys_enc->parent_ops = p->parent_ops; | ||||
| 	phys_enc->dpu_kms = p->dpu_kms; | ||||
| 	phys_enc->split_role = p->split_role; | ||||
| 	phys_enc->intf_mode = INTF_MODE_VIDEO; | ||||
| 	phys_enc->enc_spinlock = p->enc_spinlock; | ||||
| 	for (i = 0; i < INTR_IDX_MAX; i++) { | ||||
| 		irq = &phys_enc->irq[i]; | ||||
| 		INIT_LIST_HEAD(&irq->cb.list); | ||||
| 		irq->irq_idx = -EINVAL; | ||||
| 		irq->hw_idx = -EINVAL; | ||||
| 		irq->cb.arg = phys_enc; | ||||
| 	} | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_VSYNC]; | ||||
| 	irq->name = "vsync_irq"; | ||||
| 	irq->intr_type = DPU_IRQ_TYPE_INTF_VSYNC; | ||||
| 	irq->intr_idx = INTR_IDX_VSYNC; | ||||
| 	irq->cb.func = dpu_encoder_phys_vid_vblank_irq; | ||||
| 
 | ||||
| 	irq = &phys_enc->irq[INTR_IDX_UNDERRUN]; | ||||
| 	irq->name = "underrun"; | ||||
| 	irq->intr_type = DPU_IRQ_TYPE_INTF_UNDER_RUN; | ||||
| 	irq->intr_idx = INTR_IDX_UNDERRUN; | ||||
| 	irq->cb.func = dpu_encoder_phys_vid_underrun_irq; | ||||
| 
 | ||||
| 	atomic_set(&phys_enc->vblank_refcount, 0); | ||||
| 	atomic_set(&phys_enc->pending_kickoff_cnt, 0); | ||||
| 	init_waitqueue_head(&phys_enc->pending_kickoff_wq); | ||||
| 	phys_enc->enable_state = DPU_ENC_DISABLED; | ||||
| 
 | ||||
| 	DPU_DEBUG_VIDENC(vid_enc, "created intf idx:%d\n", p->intf_idx); | ||||
| 
 | ||||
| 	return phys_enc; | ||||
| 
 | ||||
| fail: | ||||
| 	DPU_ERROR("failed to create encoder\n"); | ||||
| 	if (vid_enc) | ||||
| 		dpu_encoder_phys_vid_destroy(phys_enc); | ||||
| 
 | ||||
| 	return ERR_PTR(ret); | ||||
| } | ||||
							
								
								
									
										1214
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_formats.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1214
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_formats.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										136
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_formats.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_formats.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,136 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_FORMATS_H | ||||
| #define _DPU_FORMATS_H | ||||
| 
 | ||||
| #include <drm/drm_fourcc.h> | ||||
| #include "msm_gem.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_get_dpu_format_ext() - Returns dpu format structure pointer. | ||||
|  * @format:          DRM FourCC Code | ||||
|  * @modifiers:       format modifier array from client, one per plane | ||||
|  */ | ||||
| const struct dpu_format *dpu_get_dpu_format_ext( | ||||
| 		const uint32_t format, | ||||
| 		const uint64_t modifier); | ||||
| 
 | ||||
| #define dpu_get_dpu_format(f) dpu_get_dpu_format_ext(f, 0) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_get_msm_format - get an dpu_format by its msm_format base | ||||
|  *                     callback function registers with the msm_kms layer | ||||
|  * @kms:             kms driver | ||||
|  * @format:          DRM FourCC Code | ||||
|  * @modifiers:       data layout modifier | ||||
|  */ | ||||
| const struct msm_format *dpu_get_msm_format( | ||||
| 		struct msm_kms *kms, | ||||
| 		const uint32_t format, | ||||
| 		const uint64_t modifiers); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_populate_formats - populate the given array with fourcc codes supported | ||||
|  * @format_list:       pointer to list of possible formats | ||||
|  * @pixel_formats:     array to populate with fourcc codes | ||||
|  * @pixel_modifiers:   array to populate with drm modifiers, can be NULL | ||||
|  * @pixel_formats_max: length of pixel formats array | ||||
|  * Return: number of elements populated | ||||
|  */ | ||||
| uint32_t dpu_populate_formats( | ||||
| 		const struct dpu_format_extended *format_list, | ||||
| 		uint32_t *pixel_formats, | ||||
| 		uint64_t *pixel_modifiers, | ||||
| 		uint32_t pixel_formats_max); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_format_get_plane_sizes - calculate size and layout of given buffer format | ||||
|  * @fmt:             pointer to dpu_format | ||||
|  * @w:               width of the buffer | ||||
|  * @h:               height of the buffer | ||||
|  * @layout:          layout of the buffer | ||||
|  * @pitches:         array of size [DPU_MAX_PLANES] to populate | ||||
|  *		     pitch for each plane | ||||
|  * | ||||
|  * Return: size of the buffer | ||||
|  */ | ||||
| int dpu_format_get_plane_sizes( | ||||
| 		const struct dpu_format *fmt, | ||||
| 		const uint32_t w, | ||||
| 		const uint32_t h, | ||||
| 		struct dpu_hw_fmt_layout *layout, | ||||
| 		const uint32_t *pitches); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_format_get_block_size - get block size of given format when | ||||
|  *	operating in block mode | ||||
|  * @fmt:             pointer to dpu_format | ||||
|  * @w:               pointer to width of the block | ||||
|  * @h:               pointer to height of the block | ||||
|  * | ||||
|  * Return: 0 if success; error oode otherwise | ||||
|  */ | ||||
| int dpu_format_get_block_size(const struct dpu_format *fmt, | ||||
| 		uint32_t *w, uint32_t *h); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_format_check_modified_format - validate format and buffers for | ||||
|  *                   dpu non-standard, i.e. modified format | ||||
|  * @kms:             kms driver | ||||
|  * @msm_fmt:         pointer to the msm_fmt base pointer of an dpu_format | ||||
|  * @cmd:             fb_cmd2 structure user request | ||||
|  * @bos:             gem buffer object list | ||||
|  * | ||||
|  * Return: error code on failure, 0 on success | ||||
|  */ | ||||
| int dpu_format_check_modified_format( | ||||
| 		const struct msm_kms *kms, | ||||
| 		const struct msm_format *msm_fmt, | ||||
| 		const struct drm_mode_fb_cmd2 *cmd, | ||||
| 		struct drm_gem_object **bos); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_format_populate_layout - populate the given format layout based on | ||||
|  *                     mmu, fb, and format found in the fb | ||||
|  * @aspace:            address space pointer | ||||
|  * @fb:                framebuffer pointer | ||||
|  * @fmtl:              format layout structure to populate | ||||
|  * | ||||
|  * Return: error code on failure, -EAGAIN if success but the addresses | ||||
|  *         are the same as before or 0 if new addresses were populated | ||||
|  */ | ||||
| int dpu_format_populate_layout( | ||||
| 		struct msm_gem_address_space *aspace, | ||||
| 		struct drm_framebuffer *fb, | ||||
| 		struct dpu_hw_fmt_layout *fmtl); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_format_get_framebuffer_size - get framebuffer memory size | ||||
|  * @format:            DRM pixel format | ||||
|  * @width:             pixel width | ||||
|  * @height:            pixel height | ||||
|  * @pitches:           array of size [DPU_MAX_PLANES] to populate | ||||
|  *		       pitch for each plane | ||||
|  * @modifiers:         drm modifier | ||||
|  * | ||||
|  * Return: memory size required for frame buffer | ||||
|  */ | ||||
| uint32_t dpu_format_get_framebuffer_size( | ||||
| 		const uint32_t format, | ||||
| 		const uint32_t width, | ||||
| 		const uint32_t height, | ||||
| 		const uint32_t *pitches, | ||||
| 		const uint64_t modifiers); | ||||
| 
 | ||||
| #endif /*_DPU_FORMATS_H */ | ||||
							
								
								
									
										155
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_blk.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_blk.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,155 @@ | ||||
| /* Copyright (c) 2017-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include <linux/mutex.h> | ||||
| #include <linux/errno.h> | ||||
| #include <linux/slab.h> | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| /* Serialization lock for dpu_hw_blk_list */ | ||||
| static DEFINE_MUTEX(dpu_hw_blk_lock); | ||||
| 
 | ||||
| /* List of all hw block objects */ | ||||
| static LIST_HEAD(dpu_hw_blk_list); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_blk_init - initialize hw block object | ||||
|  * @type: hw block type - enum dpu_hw_blk_type | ||||
|  * @id: instance id of the hw block | ||||
|  * @ops: Pointer to block operations | ||||
|  * return: 0 if success; error code otherwise | ||||
|  */ | ||||
| int dpu_hw_blk_init(struct dpu_hw_blk *hw_blk, u32 type, int id, | ||||
| 		struct dpu_hw_blk_ops *ops) | ||||
| { | ||||
| 	if (!hw_blk) { | ||||
| 		pr_err("invalid parameters\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	INIT_LIST_HEAD(&hw_blk->list); | ||||
| 	hw_blk->type = type; | ||||
| 	hw_blk->id = id; | ||||
| 	atomic_set(&hw_blk->refcount, 0); | ||||
| 
 | ||||
| 	if (ops) | ||||
| 		hw_blk->ops = *ops; | ||||
| 
 | ||||
| 	mutex_lock(&dpu_hw_blk_lock); | ||||
| 	list_add(&hw_blk->list, &dpu_hw_blk_list); | ||||
| 	mutex_unlock(&dpu_hw_blk_lock); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_blk_destroy - destroy hw block object. | ||||
|  * @hw_blk:  pointer to hw block object | ||||
|  * return: none | ||||
|  */ | ||||
| void dpu_hw_blk_destroy(struct dpu_hw_blk *hw_blk) | ||||
| { | ||||
| 	if (!hw_blk) { | ||||
| 		pr_err("invalid parameters\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if (atomic_read(&hw_blk->refcount)) | ||||
| 		pr_err("hw_blk:%d.%d invalid refcount\n", hw_blk->type, | ||||
| 				hw_blk->id); | ||||
| 
 | ||||
| 	mutex_lock(&dpu_hw_blk_lock); | ||||
| 	list_del(&hw_blk->list); | ||||
| 	mutex_unlock(&dpu_hw_blk_lock); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_blk_get - get hw_blk from free pool | ||||
|  * @hw_blk: if specified, increment reference count only | ||||
|  * @type: if hw_blk is not specified, allocate the next available of this type | ||||
|  * @id: if specified (>= 0), allocate the given instance of the above type | ||||
|  * return: pointer to hw block object | ||||
|  */ | ||||
| struct dpu_hw_blk *dpu_hw_blk_get(struct dpu_hw_blk *hw_blk, u32 type, int id) | ||||
| { | ||||
| 	struct dpu_hw_blk *curr; | ||||
| 	int rc, refcount; | ||||
| 
 | ||||
| 	if (!hw_blk) { | ||||
| 		mutex_lock(&dpu_hw_blk_lock); | ||||
| 		list_for_each_entry(curr, &dpu_hw_blk_list, list) { | ||||
| 			if ((curr->type != type) || | ||||
| 					(id >= 0 && curr->id != id) || | ||||
| 					(id < 0 && | ||||
| 						atomic_read(&curr->refcount))) | ||||
| 				continue; | ||||
| 
 | ||||
| 			hw_blk = curr; | ||||
| 			break; | ||||
| 		} | ||||
| 		mutex_unlock(&dpu_hw_blk_lock); | ||||
| 	} | ||||
| 
 | ||||
| 	if (!hw_blk) { | ||||
| 		pr_debug("no hw_blk:%d\n", type); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 
 | ||||
| 	refcount = atomic_inc_return(&hw_blk->refcount); | ||||
| 
 | ||||
| 	if (refcount == 1 && hw_blk->ops.start) { | ||||
| 		rc = hw_blk->ops.start(hw_blk); | ||||
| 		if (rc) { | ||||
| 			pr_err("failed to start  hw_blk:%d rc:%d\n", type, rc); | ||||
| 			goto error_start; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	pr_debug("hw_blk:%d.%d refcount:%d\n", hw_blk->type, | ||||
| 			hw_blk->id, refcount); | ||||
| 	return hw_blk; | ||||
| 
 | ||||
| error_start: | ||||
| 	dpu_hw_blk_put(hw_blk); | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_blk_put - put hw_blk to free pool if decremented refcount is zero | ||||
|  * @hw_blk: hw block to be freed | ||||
|  * @free_blk: function to be called when reference count goes to zero | ||||
|  */ | ||||
| void dpu_hw_blk_put(struct dpu_hw_blk *hw_blk) | ||||
| { | ||||
| 	if (!hw_blk) { | ||||
| 		pr_err("invalid parameters\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	pr_debug("hw_blk:%d.%d refcount:%d\n", hw_blk->type, hw_blk->id, | ||||
| 			atomic_read(&hw_blk->refcount)); | ||||
| 
 | ||||
| 	if (!atomic_read(&hw_blk->refcount)) { | ||||
| 		pr_err("hw_blk:%d.%d invalid put\n", hw_blk->type, hw_blk->id); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if (atomic_dec_return(&hw_blk->refcount)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (hw_blk->ops.stop) | ||||
| 		hw_blk->ops.stop(hw_blk); | ||||
| } | ||||
							
								
								
									
										53
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_blk.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_blk.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | ||||
| /* Copyright (c) 2017-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_BLK_H | ||||
| #define _DPU_HW_BLK_H | ||||
| 
 | ||||
| #include <linux/types.h> | ||||
| #include <linux/list.h> | ||||
| #include <linux/atomic.h> | ||||
| 
 | ||||
| struct dpu_hw_blk; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_blk_ops - common hardware block operations | ||||
|  * @start: start operation on first get | ||||
|  * @stop: stop operation on last put | ||||
|  */ | ||||
| struct dpu_hw_blk_ops { | ||||
| 	int (*start)(struct dpu_hw_blk *); | ||||
| 	void (*stop)(struct dpu_hw_blk *); | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_blk - definition of hardware block object | ||||
|  * @list: list of hardware blocks | ||||
|  * @type: hardware block type | ||||
|  * @id: instance id | ||||
|  * @refcount: reference/usage count | ||||
|  */ | ||||
| struct dpu_hw_blk { | ||||
| 	struct list_head list; | ||||
| 	u32 type; | ||||
| 	int id; | ||||
| 	atomic_t refcount; | ||||
| 	struct dpu_hw_blk_ops ops; | ||||
| }; | ||||
| 
 | ||||
| int dpu_hw_blk_init(struct dpu_hw_blk *hw_blk, u32 type, int id, | ||||
| 		struct dpu_hw_blk_ops *ops); | ||||
| void dpu_hw_blk_destroy(struct dpu_hw_blk *hw_blk); | ||||
| 
 | ||||
| struct dpu_hw_blk *dpu_hw_blk_get(struct dpu_hw_blk *hw_blk, u32 type, int id); | ||||
| void dpu_hw_blk_put(struct dpu_hw_blk *hw_blk); | ||||
| #endif /*_DPU_HW_BLK_H */ | ||||
							
								
								
									
										511
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_catalog.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										511
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_catalog.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,511 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| #include <linux/slab.h> | ||||
| #include <linux/of_address.h> | ||||
| #include <linux/platform_device.h> | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_catalog_format.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define VIG_SDM845_MASK \ | ||||
| 	(BIT(DPU_SSPP_SRC) | BIT(DPU_SSPP_SCALER_QSEED3) | BIT(DPU_SSPP_QOS) |\ | ||||
| 	BIT(DPU_SSPP_CSC_10BIT) | BIT(DPU_SSPP_CDP) | BIT(DPU_SSPP_QOS_8LVL) |\ | ||||
| 	BIT(DPU_SSPP_TS_PREFILL) | BIT(DPU_SSPP_EXCL_RECT)) | ||||
| 
 | ||||
| #define DMA_SDM845_MASK \ | ||||
| 	(BIT(DPU_SSPP_SRC) | BIT(DPU_SSPP_QOS) | BIT(DPU_SSPP_QOS_8LVL) |\ | ||||
| 	BIT(DPU_SSPP_TS_PREFILL) | BIT(DPU_SSPP_TS_PREFILL_REC1) |\ | ||||
| 	BIT(DPU_SSPP_CDP) | BIT(DPU_SSPP_EXCL_RECT)) | ||||
| 
 | ||||
| #define MIXER_SDM845_MASK \ | ||||
| 	(BIT(DPU_MIXER_SOURCESPLIT) | BIT(DPU_DIM_LAYER)) | ||||
| 
 | ||||
| #define PINGPONG_SDM845_MASK BIT(DPU_PINGPONG_DITHER) | ||||
| 
 | ||||
| #define PINGPONG_SDM845_SPLIT_MASK \ | ||||
| 	(PINGPONG_SDM845_MASK | BIT(DPU_PINGPONG_TE2)) | ||||
| 
 | ||||
| #define DEFAULT_PIXEL_RAM_SIZE		(50 * 1024) | ||||
| #define DEFAULT_DPU_LINE_WIDTH		2048 | ||||
| #define DEFAULT_DPU_OUTPUT_LINE_WIDTH	2560 | ||||
| 
 | ||||
| #define MAX_HORZ_DECIMATION	4 | ||||
| #define MAX_VERT_DECIMATION	4 | ||||
| 
 | ||||
| #define MAX_UPSCALE_RATIO	20 | ||||
| #define MAX_DOWNSCALE_RATIO	4 | ||||
| #define SSPP_UNITY_SCALE	1 | ||||
| 
 | ||||
| #define STRCAT(X, Y) (X Y) | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * DPU sub blocks config | ||||
|  *************************************************************/ | ||||
| /* DPU top level caps */ | ||||
| static const struct dpu_caps sdm845_dpu_caps = { | ||||
| 	.max_mixer_width = DEFAULT_DPU_OUTPUT_LINE_WIDTH, | ||||
| 	.max_mixer_blendstages = 0xb, | ||||
| 	.qseed_type = DPU_SSPP_SCALER_QSEED3, | ||||
| 	.smart_dma_rev = DPU_SSPP_SMART_DMA_V2, | ||||
| 	.ubwc_version = DPU_HW_UBWC_VER_20, | ||||
| 	.has_src_split = true, | ||||
| 	.has_dim_layer = true, | ||||
| 	.has_idle_pc = true, | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_mdp_cfg sdm845_mdp[] = { | ||||
| 	{ | ||||
| 	.name = "top_0", .id = MDP_TOP, | ||||
| 	.base = 0x0, .len = 0x45C, | ||||
| 	.features = 0, | ||||
| 	.highest_bank_bit = 0x2, | ||||
| 	.has_dest_scaler = true, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_VIG0] = { | ||||
| 			.reg_off = 0x2AC, .bit_off = 0}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_VIG1] = { | ||||
| 			.reg_off = 0x2B4, .bit_off = 0}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_VIG2] = { | ||||
| 			.reg_off = 0x2BC, .bit_off = 0}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_VIG3] = { | ||||
| 			.reg_off = 0x2C4, .bit_off = 0}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_DMA0] = { | ||||
| 			.reg_off = 0x2AC, .bit_off = 8}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_DMA1] = { | ||||
| 			.reg_off = 0x2B4, .bit_off = 8}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_CURSOR0] = { | ||||
| 			.reg_off = 0x2BC, .bit_off = 8}, | ||||
| 	.clk_ctrls[DPU_CLK_CTRL_CURSOR1] = { | ||||
| 			.reg_off = 0x2C4, .bit_off = 8}, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * CTL sub blocks config | ||||
|  *************************************************************/ | ||||
| static struct dpu_ctl_cfg sdm845_ctl[] = { | ||||
| 	{ | ||||
| 	.name = "ctl_0", .id = CTL_0, | ||||
| 	.base = 0x1000, .len = 0xE4, | ||||
| 	.features = BIT(DPU_CTL_SPLIT_DISPLAY) | ||||
| 	}, | ||||
| 	{ | ||||
| 	.name = "ctl_1", .id = CTL_1, | ||||
| 	.base = 0x1200, .len = 0xE4, | ||||
| 	.features = BIT(DPU_CTL_SPLIT_DISPLAY) | ||||
| 	}, | ||||
| 	{ | ||||
| 	.name = "ctl_2", .id = CTL_2, | ||||
| 	.base = 0x1400, .len = 0xE4, | ||||
| 	.features = 0 | ||||
| 	}, | ||||
| 	{ | ||||
| 	.name = "ctl_3", .id = CTL_3, | ||||
| 	.base = 0x1600, .len = 0xE4, | ||||
| 	.features = 0 | ||||
| 	}, | ||||
| 	{ | ||||
| 	.name = "ctl_4", .id = CTL_4, | ||||
| 	.base = 0x1800, .len = 0xE4, | ||||
| 	.features = 0 | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * SSPP sub blocks config | ||||
|  *************************************************************/ | ||||
| 
 | ||||
| /* SSPP common configuration */ | ||||
| static const struct dpu_sspp_blks_common sdm845_sspp_common = { | ||||
| 	.maxlinewidth = DEFAULT_DPU_OUTPUT_LINE_WIDTH, | ||||
| 	.pixel_ram_size = DEFAULT_PIXEL_RAM_SIZE, | ||||
| 	.maxhdeciexp = MAX_HORZ_DECIMATION, | ||||
| 	.maxvdeciexp = MAX_VERT_DECIMATION, | ||||
| }; | ||||
| 
 | ||||
| #define _VIG_SBLK(num, sdma_pri) \ | ||||
| 	{ \ | ||||
| 	.common = &sdm845_sspp_common, \ | ||||
| 	.maxdwnscale = MAX_DOWNSCALE_RATIO, \ | ||||
| 	.maxupscale = MAX_UPSCALE_RATIO, \ | ||||
| 	.smart_dma_priority = sdma_pri, \ | ||||
| 	.src_blk = {.name = STRCAT("sspp_src_", num), \ | ||||
| 		.id = DPU_SSPP_SRC, .base = 0x00, .len = 0x150,}, \ | ||||
| 	.scaler_blk = {.name = STRCAT("sspp_scaler", num), \ | ||||
| 		.id = DPU_SSPP_SCALER_QSEED3, \ | ||||
| 		.base = 0xa00, .len = 0xa0,}, \ | ||||
| 	.csc_blk = {.name = STRCAT("sspp_csc", num), \ | ||||
| 		.id = DPU_SSPP_CSC_10BIT, \ | ||||
| 		.base = 0x1a00, .len = 0x100,}, \ | ||||
| 	.format_list = plane_formats_yuv, \ | ||||
| 	.virt_format_list = plane_formats, \ | ||||
| 	} | ||||
| 
 | ||||
| #define _DMA_SBLK(num, sdma_pri) \ | ||||
| 	{ \ | ||||
| 	.common = &sdm845_sspp_common, \ | ||||
| 	.maxdwnscale = SSPP_UNITY_SCALE, \ | ||||
| 	.maxupscale = SSPP_UNITY_SCALE, \ | ||||
| 	.smart_dma_priority = sdma_pri, \ | ||||
| 	.src_blk = {.name = STRCAT("sspp_src_", num), \ | ||||
| 		.id = DPU_SSPP_SRC, .base = 0x00, .len = 0x150,}, \ | ||||
| 	.format_list = plane_formats, \ | ||||
| 	.virt_format_list = plane_formats, \ | ||||
| 	} | ||||
| 
 | ||||
| static const struct dpu_sspp_sub_blks sdm845_vig_sblk_0 = _VIG_SBLK("0", 5); | ||||
| static const struct dpu_sspp_sub_blks sdm845_vig_sblk_1 = _VIG_SBLK("1", 6); | ||||
| static const struct dpu_sspp_sub_blks sdm845_vig_sblk_2 = _VIG_SBLK("2", 7); | ||||
| static const struct dpu_sspp_sub_blks sdm845_vig_sblk_3 = _VIG_SBLK("3", 8); | ||||
| 
 | ||||
| static const struct dpu_sspp_sub_blks sdm845_dma_sblk_0 = _DMA_SBLK("8", 1); | ||||
| static const struct dpu_sspp_sub_blks sdm845_dma_sblk_1 = _DMA_SBLK("9", 2); | ||||
| static const struct dpu_sspp_sub_blks sdm845_dma_sblk_2 = _DMA_SBLK("10", 3); | ||||
| static const struct dpu_sspp_sub_blks sdm845_dma_sblk_3 = _DMA_SBLK("11", 4); | ||||
| 
 | ||||
| #define SSPP_VIG_BLK(_name, _id, _base, _sblk, _xinid, _clkctrl) \ | ||||
| 	{ \ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0x1c8, \ | ||||
| 	.features = VIG_SDM845_MASK, \ | ||||
| 	.sblk = &_sblk, \ | ||||
| 	.xin_id = _xinid, \ | ||||
| 	.type = SSPP_TYPE_VIG, \ | ||||
| 	.clk_ctrl = _clkctrl \ | ||||
| 	} | ||||
| 
 | ||||
| #define SSPP_DMA_BLK(_name, _id, _base, _sblk, _xinid, _clkctrl) \ | ||||
| 	{ \ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0x1c8, \ | ||||
| 	.features = DMA_SDM845_MASK, \ | ||||
| 	.sblk = &_sblk, \ | ||||
| 	.xin_id = _xinid, \ | ||||
| 	.type = SSPP_TYPE_DMA, \ | ||||
| 	.clk_ctrl = _clkctrl \ | ||||
| 	} | ||||
| 
 | ||||
| static struct dpu_sspp_cfg sdm845_sspp[] = { | ||||
| 	SSPP_VIG_BLK("sspp_0", SSPP_VIG0, 0x4000, | ||||
| 		sdm845_vig_sblk_0, 0, DPU_CLK_CTRL_VIG0), | ||||
| 	SSPP_VIG_BLK("sspp_1", SSPP_VIG1, 0x6000, | ||||
| 		sdm845_vig_sblk_1, 4, DPU_CLK_CTRL_VIG1), | ||||
| 	SSPP_VIG_BLK("sspp_2", SSPP_VIG2, 0x8000, | ||||
| 		sdm845_vig_sblk_2, 8, DPU_CLK_CTRL_VIG2), | ||||
| 	SSPP_VIG_BLK("sspp_3", SSPP_VIG3, 0xa000, | ||||
| 		sdm845_vig_sblk_3, 12, DPU_CLK_CTRL_VIG3), | ||||
| 	SSPP_DMA_BLK("sspp_8", SSPP_DMA0, 0x24000, | ||||
| 		sdm845_dma_sblk_0, 1, DPU_CLK_CTRL_DMA0), | ||||
| 	SSPP_DMA_BLK("sspp_9", SSPP_DMA1, 0x26000, | ||||
| 		sdm845_dma_sblk_1, 5, DPU_CLK_CTRL_DMA1), | ||||
| 	SSPP_DMA_BLK("sspp_10", SSPP_DMA2, 0x28000, | ||||
| 		sdm845_dma_sblk_2, 9, DPU_CLK_CTRL_CURSOR0), | ||||
| 	SSPP_DMA_BLK("sspp_11", SSPP_DMA3, 0x2a000, | ||||
| 		sdm845_dma_sblk_3, 13, DPU_CLK_CTRL_CURSOR1), | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * MIXER sub blocks config | ||||
|  *************************************************************/ | ||||
| static const struct dpu_lm_sub_blks sdm845_lm_sblk = { | ||||
| 	.maxwidth = DEFAULT_DPU_OUTPUT_LINE_WIDTH, | ||||
| 	.maxblendstages = 11, /* excluding base layer */ | ||||
| 	.blendstage_base = { /* offsets relative to mixer base */ | ||||
| 		0x20, 0x38, 0x50, 0x68, 0x80, 0x98, | ||||
| 		0xb0, 0xc8, 0xe0, 0xf8, 0x110 | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| #define LM_BLK(_name, _id, _base, _ds, _pp, _lmpair) \ | ||||
| 	{ \ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0x320, \ | ||||
| 	.features = MIXER_SDM845_MASK, \ | ||||
| 	.sblk = &sdm845_lm_sblk, \ | ||||
| 	.ds = _ds, \ | ||||
| 	.pingpong = _pp, \ | ||||
| 	.lm_pair_mask = (1 << _lmpair) \ | ||||
| 	} | ||||
| 
 | ||||
| static struct dpu_lm_cfg sdm845_lm[] = { | ||||
| 	LM_BLK("lm_0", LM_0, 0x44000, DS_0, PINGPONG_0, LM_1), | ||||
| 	LM_BLK("lm_1", LM_1, 0x45000, DS_1, PINGPONG_1, LM_0), | ||||
| 	LM_BLK("lm_2", LM_2, 0x46000, DS_MAX, PINGPONG_2, LM_5), | ||||
| 	LM_BLK("lm_3", LM_3, 0x0, DS_MAX, PINGPONG_MAX, 0), | ||||
| 	LM_BLK("lm_4", LM_4, 0x0, DS_MAX, PINGPONG_MAX, 0), | ||||
| 	LM_BLK("lm_5", LM_5, 0x49000, DS_MAX, PINGPONG_3, LM_2), | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * DS sub blocks config | ||||
|  *************************************************************/ | ||||
| static const struct dpu_ds_top_cfg sdm845_ds_top = { | ||||
| 	.name = "ds_top_0", .id = DS_TOP, | ||||
| 	.base = 0x60000, .len = 0xc, | ||||
| 	.maxinputwidth = DEFAULT_DPU_LINE_WIDTH, | ||||
| 	.maxoutputwidth = DEFAULT_DPU_OUTPUT_LINE_WIDTH, | ||||
| 	.maxupscale = MAX_UPSCALE_RATIO, | ||||
| }; | ||||
| 
 | ||||
| #define DS_BLK(_name, _id, _base) \ | ||||
| 	{\ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0x800, \ | ||||
| 	.features = DPU_SSPP_SCALER_QSEED3, \ | ||||
| 	.top = &sdm845_ds_top \ | ||||
| 	} | ||||
| 
 | ||||
| static struct dpu_ds_cfg sdm845_ds[] = { | ||||
| 	DS_BLK("ds_0", DS_0, 0x800), | ||||
| 	DS_BLK("ds_1", DS_1, 0x1000), | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * PINGPONG sub blocks config | ||||
|  *************************************************************/ | ||||
| static const struct dpu_pingpong_sub_blks sdm845_pp_sblk_te = { | ||||
| 	.te2 = {.id = DPU_PINGPONG_TE2, .base = 0x2000, .len = 0x0, | ||||
| 		.version = 0x1}, | ||||
| 	.dither = {.id = DPU_PINGPONG_DITHER, .base = 0x30e0, | ||||
| 		.len = 0x20, .version = 0x10000}, | ||||
| }; | ||||
| 
 | ||||
| static const struct dpu_pingpong_sub_blks sdm845_pp_sblk = { | ||||
| 	.dither = {.id = DPU_PINGPONG_DITHER, .base = 0x30e0, | ||||
| 		.len = 0x20, .version = 0x10000}, | ||||
| }; | ||||
| 
 | ||||
| #define PP_BLK_TE(_name, _id, _base) \ | ||||
| 	{\ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0xd4, \ | ||||
| 	.features = PINGPONG_SDM845_SPLIT_MASK, \ | ||||
| 	.sblk = &sdm845_pp_sblk_te \ | ||||
| 	} | ||||
| #define PP_BLK(_name, _id, _base) \ | ||||
| 	{\ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0xd4, \ | ||||
| 	.features = PINGPONG_SDM845_MASK, \ | ||||
| 	.sblk = &sdm845_pp_sblk \ | ||||
| 	} | ||||
| 
 | ||||
| static struct dpu_pingpong_cfg sdm845_pp[] = { | ||||
| 	PP_BLK_TE("pingpong_0", PINGPONG_0, 0x70000), | ||||
| 	PP_BLK_TE("pingpong_1", PINGPONG_1, 0x70800), | ||||
| 	PP_BLK("pingpong_2", PINGPONG_2, 0x71000), | ||||
| 	PP_BLK("pingpong_3", PINGPONG_3, 0x71800), | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * INTF sub blocks config | ||||
|  *************************************************************/ | ||||
| #define INTF_BLK(_name, _id, _base, _type, _ctrl_id) \ | ||||
| 	{\ | ||||
| 	.name = _name, .id = _id, \ | ||||
| 	.base = _base, .len = 0x280, \ | ||||
| 	.type = _type, \ | ||||
| 	.controller_id = _ctrl_id, \ | ||||
| 	.prog_fetch_lines_worst_case = 24 \ | ||||
| 	} | ||||
| 
 | ||||
| static struct dpu_intf_cfg sdm845_intf[] = { | ||||
| 	INTF_BLK("intf_0", INTF_0, 0x6A000, INTF_DP, 0), | ||||
| 	INTF_BLK("intf_1", INTF_1, 0x6A800, INTF_DSI, 0), | ||||
| 	INTF_BLK("intf_2", INTF_2, 0x6B000, INTF_DSI, 1), | ||||
| 	INTF_BLK("intf_3", INTF_3, 0x6B800, INTF_DP, 1), | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * CDM sub blocks config | ||||
|  *************************************************************/ | ||||
| static struct dpu_cdm_cfg sdm845_cdm[] = { | ||||
| 	{ | ||||
| 	.name = "cdm_0", .id = CDM_0, | ||||
| 	.base = 0x79200, .len = 0x224, | ||||
| 	.features = 0, | ||||
| 	.intf_connect = BIT(INTF_3), | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * VBIF sub blocks config | ||||
|  *************************************************************/ | ||||
| /* VBIF QOS remap */ | ||||
| static u32 sdm845_rt_pri_lvl[] = {3, 3, 4, 4, 5, 5, 6, 6}; | ||||
| static u32 sdm845_nrt_pri_lvl[] = {3, 3, 3, 3, 3, 3, 3, 3}; | ||||
| 
 | ||||
| static struct dpu_vbif_cfg sdm845_vbif[] = { | ||||
| 	{ | ||||
| 	.name = "vbif_0", .id = VBIF_0, | ||||
| 	.base = 0, .len = 0x1040, | ||||
| 	.features = BIT(DPU_VBIF_QOS_REMAP), | ||||
| 	.xin_halt_timeout = 0x4000, | ||||
| 	.qos_rt_tbl = { | ||||
| 		.npriority_lvl = ARRAY_SIZE(sdm845_rt_pri_lvl), | ||||
| 		.priority_lvl = sdm845_rt_pri_lvl, | ||||
| 		}, | ||||
| 	.qos_nrt_tbl = { | ||||
| 		.npriority_lvl = ARRAY_SIZE(sdm845_nrt_pri_lvl), | ||||
| 		.priority_lvl = sdm845_nrt_pri_lvl, | ||||
| 		}, | ||||
| 	.memtype_count = 14, | ||||
| 	.memtype = {3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3}, | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_reg_dma_cfg sdm845_regdma = { | ||||
| 	.base = 0x0, .version = 0x1, .trigger_sel_off = 0x119c | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * PERF data config | ||||
|  *************************************************************/ | ||||
| 
 | ||||
| /* SSPP QOS LUTs */ | ||||
| static struct dpu_qos_lut_entry sdm845_qos_linear[] = { | ||||
| 	{.fl = 4, .lut = 0x357}, | ||||
| 	{.fl = 5, .lut = 0x3357}, | ||||
| 	{.fl = 6, .lut = 0x23357}, | ||||
| 	{.fl = 7, .lut = 0x223357}, | ||||
| 	{.fl = 8, .lut = 0x2223357}, | ||||
| 	{.fl = 9, .lut = 0x22223357}, | ||||
| 	{.fl = 10, .lut = 0x222223357}, | ||||
| 	{.fl = 11, .lut = 0x2222223357}, | ||||
| 	{.fl = 12, .lut = 0x22222223357}, | ||||
| 	{.fl = 13, .lut = 0x222222223357}, | ||||
| 	{.fl = 14, .lut = 0x1222222223357}, | ||||
| 	{.fl = 0, .lut = 0x11222222223357} | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_qos_lut_entry sdm845_qos_macrotile[] = { | ||||
| 	{.fl = 10, .lut = 0x344556677}, | ||||
| 	{.fl = 11, .lut = 0x3344556677}, | ||||
| 	{.fl = 12, .lut = 0x23344556677}, | ||||
| 	{.fl = 13, .lut = 0x223344556677}, | ||||
| 	{.fl = 14, .lut = 0x1223344556677}, | ||||
| 	{.fl = 0, .lut = 0x112233344556677}, | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_qos_lut_entry sdm845_qos_nrt[] = { | ||||
| 	{.fl = 0, .lut = 0x0}, | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_perf_cfg sdm845_perf_data = { | ||||
| 	.max_bw_low = 6800000, | ||||
| 	.max_bw_high = 6800000, | ||||
| 	.min_core_ib = 2400000, | ||||
| 	.min_llcc_ib = 800000, | ||||
| 	.min_dram_ib = 800000, | ||||
| 	.core_ib_ff = "6.0", | ||||
| 	.core_clk_ff = "1.0", | ||||
| 	.comp_ratio_rt = | ||||
| 	"NV12/5/1/1.23 AB24/5/1/1.23 XB24/5/1/1.23", | ||||
| 	.comp_ratio_nrt = | ||||
| 	"NV12/5/1/1.25 AB24/5/1/1.25 XB24/5/1/1.25", | ||||
| 	.undersized_prefill_lines = 2, | ||||
| 	.xtra_prefill_lines = 2, | ||||
| 	.dest_scale_prefill_lines = 3, | ||||
| 	.macrotile_prefill_lines = 4, | ||||
| 	.yuv_nv12_prefill_lines = 8, | ||||
| 	.linear_prefill_lines = 1, | ||||
| 	.downscaling_prefill_lines = 1, | ||||
| 	.amortizable_threshold = 25, | ||||
| 	.min_prefill_lines = 24, | ||||
| 	.danger_lut_tbl = {0xf, 0xffff, 0x0}, | ||||
| 	.qos_lut_tbl = { | ||||
| 		{.nentry = ARRAY_SIZE(sdm845_qos_linear), | ||||
| 		.entries = sdm845_qos_linear | ||||
| 		}, | ||||
| 		{.nentry = ARRAY_SIZE(sdm845_qos_macrotile), | ||||
| 		.entries = sdm845_qos_macrotile | ||||
| 		}, | ||||
| 		{.nentry = ARRAY_SIZE(sdm845_qos_nrt), | ||||
| 		.entries = sdm845_qos_nrt | ||||
| 		}, | ||||
| 	}, | ||||
| 	.cdp_cfg = { | ||||
| 		{.rd_enable = 1, .wr_enable = 1}, | ||||
| 		{.rd_enable = 1, .wr_enable = 0} | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| /*************************************************************
 | ||||
|  * Hardware catalog init | ||||
|  *************************************************************/ | ||||
| 
 | ||||
| /*
 | ||||
|  * sdm845_cfg_init(): populate sdm845 dpu sub-blocks reg offsets | ||||
|  * and instance counts. | ||||
|  */ | ||||
| void sdm845_cfg_init(struct dpu_mdss_cfg *dpu_cfg) | ||||
| { | ||||
| 	*dpu_cfg = (struct dpu_mdss_cfg){ | ||||
| 		.caps = &sdm845_dpu_caps, | ||||
| 		.mdp_count = ARRAY_SIZE(sdm845_mdp), | ||||
| 		.mdp = sdm845_mdp, | ||||
| 		.ctl_count = ARRAY_SIZE(sdm845_ctl), | ||||
| 		.ctl = sdm845_ctl, | ||||
| 		.sspp_count = ARRAY_SIZE(sdm845_sspp), | ||||
| 		.sspp = sdm845_sspp, | ||||
| 		.mixer_count = ARRAY_SIZE(sdm845_lm), | ||||
| 		.mixer = sdm845_lm, | ||||
| 		.ds_count = ARRAY_SIZE(sdm845_ds), | ||||
| 		.ds = sdm845_ds, | ||||
| 		.pingpong_count = ARRAY_SIZE(sdm845_pp), | ||||
| 		.pingpong = sdm845_pp, | ||||
| 		.cdm_count = ARRAY_SIZE(sdm845_cdm), | ||||
| 		.cdm = sdm845_cdm, | ||||
| 		.intf_count = ARRAY_SIZE(sdm845_intf), | ||||
| 		.intf = sdm845_intf, | ||||
| 		.vbif_count = ARRAY_SIZE(sdm845_vbif), | ||||
| 		.vbif = sdm845_vbif, | ||||
| 		.reg_dma_count = 1, | ||||
| 		.dma_cfg = sdm845_regdma, | ||||
| 		.perf = sdm845_perf_data, | ||||
| 	}; | ||||
| } | ||||
| 
 | ||||
| static struct dpu_mdss_hw_cfg_handler cfg_handler[] = { | ||||
| 	{ .hw_rev = DPU_HW_VER_400, .cfg_init = sdm845_cfg_init}, | ||||
| 	{ .hw_rev = DPU_HW_VER_401, .cfg_init = sdm845_cfg_init}, | ||||
| }; | ||||
| 
 | ||||
| void dpu_hw_catalog_deinit(struct dpu_mdss_cfg *dpu_cfg) | ||||
| { | ||||
| 	kfree(dpu_cfg); | ||||
| } | ||||
| 
 | ||||
| struct dpu_mdss_cfg *dpu_hw_catalog_init(u32 hw_rev) | ||||
| { | ||||
| 	int i; | ||||
| 	struct dpu_mdss_cfg *dpu_cfg; | ||||
| 
 | ||||
| 	dpu_cfg = kzalloc(sizeof(*dpu_cfg), GFP_KERNEL); | ||||
| 	if (!dpu_cfg) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(cfg_handler); i++) { | ||||
| 		if (cfg_handler[i].hw_rev == hw_rev) { | ||||
| 			cfg_handler[i].cfg_init(dpu_cfg); | ||||
| 			dpu_cfg->hwversion = hw_rev; | ||||
| 			return dpu_cfg; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_ERROR("unsupported chipset id:%X\n", hw_rev); | ||||
| 	dpu_hw_catalog_deinit(dpu_cfg); | ||||
| 	return ERR_PTR(-ENODEV); | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										804
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_catalog.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										804
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_catalog.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,804 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_CATALOG_H | ||||
| #define _DPU_HW_CATALOG_H | ||||
| 
 | ||||
| #include <linux/kernel.h> | ||||
| #include <linux/bug.h> | ||||
| #include <linux/bitmap.h> | ||||
| #include <linux/err.h> | ||||
| #include <drm/drmP.h> | ||||
| 
 | ||||
| /**
 | ||||
|  * Max hardware block count: For ex: max 12 SSPP pipes or | ||||
|  * 5 ctl paths. In all cases, it can have max 12 hardware blocks | ||||
|  * based on current design | ||||
|  */ | ||||
| #define MAX_BLOCKS    12 | ||||
| 
 | ||||
| #define DPU_HW_VER(MAJOR, MINOR, STEP) (((MAJOR & 0xF) << 28)    |\ | ||||
| 		((MINOR & 0xFFF) << 16)  |\ | ||||
| 		(STEP & 0xFFFF)) | ||||
| 
 | ||||
| #define DPU_HW_MAJOR(rev)		((rev) >> 28) | ||||
| #define DPU_HW_MINOR(rev)		(((rev) >> 16) & 0xFFF) | ||||
| #define DPU_HW_STEP(rev)		((rev) & 0xFFFF) | ||||
| #define DPU_HW_MAJOR_MINOR(rev)		((rev) >> 16) | ||||
| 
 | ||||
| #define IS_DPU_MAJOR_MINOR_SAME(rev1, rev2)   \ | ||||
| 	(DPU_HW_MAJOR_MINOR((rev1)) == DPU_HW_MAJOR_MINOR((rev2))) | ||||
| 
 | ||||
| #define DPU_HW_VER_170	DPU_HW_VER(1, 7, 0) /* 8996 v1.0 */ | ||||
| #define DPU_HW_VER_171	DPU_HW_VER(1, 7, 1) /* 8996 v2.0 */ | ||||
| #define DPU_HW_VER_172	DPU_HW_VER(1, 7, 2) /* 8996 v3.0 */ | ||||
| #define DPU_HW_VER_300	DPU_HW_VER(3, 0, 0) /* 8998 v1.0 */ | ||||
| #define DPU_HW_VER_301	DPU_HW_VER(3, 0, 1) /* 8998 v1.1 */ | ||||
| #define DPU_HW_VER_400	DPU_HW_VER(4, 0, 0) /* sdm845 v1.0 */ | ||||
| #define DPU_HW_VER_401	DPU_HW_VER(4, 0, 1) /* sdm845 v2.0 */ | ||||
| #define DPU_HW_VER_410	DPU_HW_VER(4, 1, 0) /* sdm670 v1.0 */ | ||||
| #define DPU_HW_VER_500	DPU_HW_VER(5, 0, 0) /* sdm855 v1.0 */ | ||||
| 
 | ||||
| 
 | ||||
| #define IS_MSM8996_TARGET(rev) IS_DPU_MAJOR_MINOR_SAME((rev), DPU_HW_VER_170) | ||||
| #define IS_MSM8998_TARGET(rev) IS_DPU_MAJOR_MINOR_SAME((rev), DPU_HW_VER_300) | ||||
| #define IS_SDM845_TARGET(rev) IS_DPU_MAJOR_MINOR_SAME((rev), DPU_HW_VER_400) | ||||
| #define IS_SDM670_TARGET(rev) IS_DPU_MAJOR_MINOR_SAME((rev), DPU_HW_VER_410) | ||||
| #define IS_SDM855_TARGET(rev) IS_DPU_MAJOR_MINOR_SAME((rev), DPU_HW_VER_500) | ||||
| 
 | ||||
| 
 | ||||
| #define DPU_HW_BLK_NAME_LEN	16 | ||||
| 
 | ||||
| #define MAX_IMG_WIDTH 0x3fff | ||||
| #define MAX_IMG_HEIGHT 0x3fff | ||||
| 
 | ||||
| #define CRTC_DUAL_MIXERS	2 | ||||
| 
 | ||||
| #define MAX_XIN_COUNT 16 | ||||
| 
 | ||||
| /**
 | ||||
|  * Supported UBWC feature versions | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_HW_UBWC_VER_10 = 0x100, | ||||
| 	DPU_HW_UBWC_VER_20 = 0x200, | ||||
| 	DPU_HW_UBWC_VER_30 = 0x300, | ||||
| }; | ||||
| 
 | ||||
| #define IS_UBWC_20_SUPPORTED(rev)       ((rev) >= DPU_HW_UBWC_VER_20) | ||||
| 
 | ||||
| /**
 | ||||
|  * MDP TOP BLOCK features | ||||
|  * @DPU_MDP_PANIC_PER_PIPE Panic configuration needs to be be done per pipe | ||||
|  * @DPU_MDP_10BIT_SUPPORT, Chipset supports 10 bit pixel formats | ||||
|  * @DPU_MDP_BWC,           MDSS HW supports Bandwidth compression. | ||||
|  * @DPU_MDP_UBWC_1_0,      This chipsets supports Universal Bandwidth | ||||
|  *                         compression initial revision | ||||
|  * @DPU_MDP_UBWC_1_5,      Universal Bandwidth compression version 1.5 | ||||
|  * @DPU_MDP_MAX            Maximum value | ||||
| 
 | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_MDP_PANIC_PER_PIPE = 0x1, | ||||
| 	DPU_MDP_10BIT_SUPPORT, | ||||
| 	DPU_MDP_BWC, | ||||
| 	DPU_MDP_UBWC_1_0, | ||||
| 	DPU_MDP_UBWC_1_5, | ||||
| 	DPU_MDP_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * SSPP sub-blocks/features | ||||
|  * @DPU_SSPP_SRC             Src and fetch part of the pipes, | ||||
|  * @DPU_SSPP_SCALER_QSEED2,  QSEED2 algorithm support | ||||
|  * @DPU_SSPP_SCALER_QSEED3,  QSEED3 alogorithm support | ||||
|  * @DPU_SSPP_SCALER_RGB,     RGB Scaler, supported by RGB pipes | ||||
|  * @DPU_SSPP_CSC,            Support of Color space converion | ||||
|  * @DPU_SSPP_CSC_10BIT,      Support of 10-bit Color space conversion | ||||
|  * @DPU_SSPP_CURSOR,         SSPP can be used as a cursor layer | ||||
|  * @DPU_SSPP_QOS,            SSPP support QoS control, danger/safe/creq | ||||
|  * @DPU_SSPP_QOS_8LVL,       SSPP support 8-level QoS control | ||||
|  * @DPU_SSPP_EXCL_RECT,      SSPP supports exclusion rect | ||||
|  * @DPU_SSPP_SMART_DMA_V1,   SmartDMA 1.0 support | ||||
|  * @DPU_SSPP_SMART_DMA_V2,   SmartDMA 2.0 support | ||||
|  * @DPU_SSPP_TS_PREFILL      Supports prefill with traffic shaper | ||||
|  * @DPU_SSPP_TS_PREFILL_REC1 Supports prefill with traffic shaper multirec | ||||
|  * @DPU_SSPP_CDP             Supports client driven prefetch | ||||
|  * @DPU_SSPP_MAX             maximum value | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_SSPP_SRC = 0x1, | ||||
| 	DPU_SSPP_SCALER_QSEED2, | ||||
| 	DPU_SSPP_SCALER_QSEED3, | ||||
| 	DPU_SSPP_SCALER_RGB, | ||||
| 	DPU_SSPP_CSC, | ||||
| 	DPU_SSPP_CSC_10BIT, | ||||
| 	DPU_SSPP_CURSOR, | ||||
| 	DPU_SSPP_QOS, | ||||
| 	DPU_SSPP_QOS_8LVL, | ||||
| 	DPU_SSPP_EXCL_RECT, | ||||
| 	DPU_SSPP_SMART_DMA_V1, | ||||
| 	DPU_SSPP_SMART_DMA_V2, | ||||
| 	DPU_SSPP_TS_PREFILL, | ||||
| 	DPU_SSPP_TS_PREFILL_REC1, | ||||
| 	DPU_SSPP_CDP, | ||||
| 	DPU_SSPP_MAX | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * MIXER sub-blocks/features | ||||
|  * @DPU_MIXER_LAYER           Layer mixer layer blend configuration, | ||||
|  * @DPU_MIXER_SOURCESPLIT     Layer mixer supports source-split configuration | ||||
|  * @DPU_MIXER_GC              Gamma correction block | ||||
|  * @DPU_DIM_LAYER             Layer mixer supports dim layer | ||||
|  * @DPU_MIXER_MAX             maximum value | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_MIXER_LAYER = 0x1, | ||||
| 	DPU_MIXER_SOURCESPLIT, | ||||
| 	DPU_MIXER_GC, | ||||
| 	DPU_DIM_LAYER, | ||||
| 	DPU_MIXER_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * PINGPONG sub-blocks | ||||
|  * @DPU_PINGPONG_TE         Tear check block | ||||
|  * @DPU_PINGPONG_TE2        Additional tear check block for split pipes | ||||
|  * @DPU_PINGPONG_SPLIT      PP block supports split fifo | ||||
|  * @DPU_PINGPONG_SLAVE      PP block is a suitable slave for split fifo | ||||
|  * @DPU_PINGPONG_DITHER,    Dither blocks | ||||
|  * @DPU_PINGPONG_MAX | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_PINGPONG_TE = 0x1, | ||||
| 	DPU_PINGPONG_TE2, | ||||
| 	DPU_PINGPONG_SPLIT, | ||||
| 	DPU_PINGPONG_SLAVE, | ||||
| 	DPU_PINGPONG_DITHER, | ||||
| 	DPU_PINGPONG_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * CTL sub-blocks | ||||
|  * @DPU_CTL_SPLIT_DISPLAY       CTL supports video mode split display | ||||
|  * @DPU_CTL_MAX | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_CTL_SPLIT_DISPLAY = 0x1, | ||||
| 	DPU_CTL_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * VBIF sub-blocks and features | ||||
|  * @DPU_VBIF_QOS_OTLIM        VBIF supports OT Limit | ||||
|  * @DPU_VBIF_QOS_REMAP        VBIF supports QoS priority remap | ||||
|  * @DPU_VBIF_MAX              maximum value | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_VBIF_QOS_OTLIM = 0x1, | ||||
| 	DPU_VBIF_QOS_REMAP, | ||||
| 	DPU_VBIF_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * MACRO DPU_HW_BLK_INFO - information of HW blocks inside DPU | ||||
|  * @name:              string name for debug purposes | ||||
|  * @id:                enum identifying this block | ||||
|  * @base:              register base offset to mdss | ||||
|  * @len:               length of hardware block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  */ | ||||
| #define DPU_HW_BLK_INFO \ | ||||
| 	char name[DPU_HW_BLK_NAME_LEN]; \ | ||||
| 	u32 id; \ | ||||
| 	u32 base; \ | ||||
| 	u32 len; \ | ||||
| 	unsigned long features | ||||
| 
 | ||||
| /**
 | ||||
|  * MACRO DPU_HW_SUBBLK_INFO - information of HW sub-block inside DPU | ||||
|  * @name:              string name for debug purposes | ||||
|  * @id:                enum identifying this sub-block | ||||
|  * @base:              offset of this sub-block relative to the block | ||||
|  *                     offset | ||||
|  * @len                register block length of this sub-block | ||||
|  */ | ||||
| #define DPU_HW_SUBBLK_INFO \ | ||||
| 	char name[DPU_HW_BLK_NAME_LEN]; \ | ||||
| 	u32 id; \ | ||||
| 	u32 base; \ | ||||
| 	u32 len | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_src_blk: SSPP part of the source pipes | ||||
|  * @info:   HW register and features supported by this sub-blk | ||||
|  */ | ||||
| struct dpu_src_blk { | ||||
| 	DPU_HW_SUBBLK_INFO; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_scaler_blk: Scaler information | ||||
|  * @info:   HW register and features supported by this sub-blk | ||||
|  * @version: qseed block revision | ||||
|  */ | ||||
| struct dpu_scaler_blk { | ||||
| 	DPU_HW_SUBBLK_INFO; | ||||
| 	u32 version; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_csc_blk { | ||||
| 	DPU_HW_SUBBLK_INFO; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_pp_blk : Pixel processing sub-blk information | ||||
|  * @info:   HW register and features supported by this sub-blk | ||||
|  * @version: HW Algorithm version | ||||
|  */ | ||||
| struct dpu_pp_blk { | ||||
| 	DPU_HW_SUBBLK_INFO; | ||||
| 	u32 version; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_format_extended - define dpu specific pixel format+modifier | ||||
|  * @fourcc_format: Base FOURCC pixel format code | ||||
|  * @modifier: 64-bit drm format modifier, same modifier must be applied to all | ||||
|  *            framebuffer planes | ||||
|  */ | ||||
| struct dpu_format_extended { | ||||
| 	uint32_t fourcc_format; | ||||
| 	uint64_t modifier; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_qos_lut_usage - define QoS LUT use cases | ||||
|  */ | ||||
| enum dpu_qos_lut_usage { | ||||
| 	DPU_QOS_LUT_USAGE_LINEAR, | ||||
| 	DPU_QOS_LUT_USAGE_MACROTILE, | ||||
| 	DPU_QOS_LUT_USAGE_NRT, | ||||
| 	DPU_QOS_LUT_USAGE_MAX, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_qos_lut_entry - define QoS LUT table entry | ||||
|  * @fl: fill level, or zero on last entry to indicate default lut | ||||
|  * @lut: lut to use if equal to or less than fill level | ||||
|  */ | ||||
| struct dpu_qos_lut_entry { | ||||
| 	u32 fl; | ||||
| 	u64 lut; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_qos_lut_tbl - define QoS LUT table | ||||
|  * @nentry: number of entry in this table | ||||
|  * @entries: Pointer to table entries | ||||
|  */ | ||||
| struct dpu_qos_lut_tbl { | ||||
| 	u32 nentry; | ||||
| 	struct dpu_qos_lut_entry *entries; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_caps - define DPU capabilities | ||||
|  * @max_mixer_width    max layer mixer line width support. | ||||
|  * @max_mixer_blendstages max layer mixer blend stages or | ||||
|  *                       supported z order | ||||
|  * @qseed_type         qseed2 or qseed3 support. | ||||
|  * @smart_dma_rev      Supported version of SmartDMA feature. | ||||
|  * @ubwc_version       UBWC feature version (0x0 for not supported) | ||||
|  * @has_src_split      source split feature status | ||||
|  * @has_dim_layer      dim layer feature status | ||||
|  * @has_idle_pc        indicate if idle power collapse feature is supported | ||||
|  */ | ||||
| struct dpu_caps { | ||||
| 	u32 max_mixer_width; | ||||
| 	u32 max_mixer_blendstages; | ||||
| 	u32 qseed_type; | ||||
| 	u32 smart_dma_rev; | ||||
| 	u32 ubwc_version; | ||||
| 	bool has_src_split; | ||||
| 	bool has_dim_layer; | ||||
| 	bool has_idle_pc; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_sspp_blks_common : SSPP sub-blocks common configuration | ||||
|  * @maxwidth: max pixelwidth supported by this pipe | ||||
|  * @pixel_ram_size: size of latency hiding and de-tiling buffer in bytes | ||||
|  * @maxhdeciexp: max horizontal decimation supported by this pipe | ||||
|  *				(max is 2^value) | ||||
|  * @maxvdeciexp: max vertical decimation supported by this pipe | ||||
|  *				(max is 2^value) | ||||
|  */ | ||||
| struct dpu_sspp_blks_common { | ||||
| 	u32 maxlinewidth; | ||||
| 	u32 pixel_ram_size; | ||||
| 	u32 maxhdeciexp; | ||||
| 	u32 maxvdeciexp; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_sspp_sub_blks : SSPP sub-blocks | ||||
|  * common: Pointer to common configurations shared by sub blocks | ||||
|  * @creq_vblank: creq priority during vertical blanking | ||||
|  * @danger_vblank: danger priority during vertical blanking | ||||
|  * @maxdwnscale: max downscale ratio supported(without DECIMATION) | ||||
|  * @maxupscale:  maxupscale ratio supported | ||||
|  * @smart_dma_priority: hw priority of rect1 of multirect pipe | ||||
|  * @max_per_pipe_bw: maximum allowable bandwidth of this pipe in kBps | ||||
|  * @src_blk: | ||||
|  * @scaler_blk: | ||||
|  * @csc_blk: | ||||
|  * @hsic: | ||||
|  * @memcolor: | ||||
|  * @pcc_blk: | ||||
|  * @igc_blk: | ||||
|  * @format_list: Pointer to list of supported formats | ||||
|  * @virt_format_list: Pointer to list of supported formats for virtual planes | ||||
|  */ | ||||
| struct dpu_sspp_sub_blks { | ||||
| 	const struct dpu_sspp_blks_common *common; | ||||
| 	u32 creq_vblank; | ||||
| 	u32 danger_vblank; | ||||
| 	u32 maxdwnscale; | ||||
| 	u32 maxupscale; | ||||
| 	u32 smart_dma_priority; | ||||
| 	u32 max_per_pipe_bw; | ||||
| 	struct dpu_src_blk src_blk; | ||||
| 	struct dpu_scaler_blk scaler_blk; | ||||
| 	struct dpu_pp_blk csc_blk; | ||||
| 	struct dpu_pp_blk hsic_blk; | ||||
| 	struct dpu_pp_blk memcolor_blk; | ||||
| 	struct dpu_pp_blk pcc_blk; | ||||
| 	struct dpu_pp_blk igc_blk; | ||||
| 
 | ||||
| 	const struct dpu_format_extended *format_list; | ||||
| 	const struct dpu_format_extended *virt_format_list; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_lm_sub_blks:      information of mixer block | ||||
|  * @maxwidth:               Max pixel width supported by this mixer | ||||
|  * @maxblendstages:         Max number of blend-stages supported | ||||
|  * @blendstage_base:        Blend-stage register base offset | ||||
|  * @gc: gamma correction block | ||||
|  */ | ||||
| struct dpu_lm_sub_blks { | ||||
| 	u32 maxwidth; | ||||
| 	u32 maxblendstages; | ||||
| 	u32 blendstage_base[MAX_BLOCKS]; | ||||
| 	struct dpu_pp_blk gc; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_pingpong_sub_blks { | ||||
| 	struct dpu_pp_blk te; | ||||
| 	struct dpu_pp_blk te2; | ||||
| 	struct dpu_pp_blk dither; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_clk_ctrl_type - Defines top level clock control signals | ||||
|  */ | ||||
| enum dpu_clk_ctrl_type { | ||||
| 	DPU_CLK_CTRL_NONE, | ||||
| 	DPU_CLK_CTRL_VIG0, | ||||
| 	DPU_CLK_CTRL_VIG1, | ||||
| 	DPU_CLK_CTRL_VIG2, | ||||
| 	DPU_CLK_CTRL_VIG3, | ||||
| 	DPU_CLK_CTRL_VIG4, | ||||
| 	DPU_CLK_CTRL_RGB0, | ||||
| 	DPU_CLK_CTRL_RGB1, | ||||
| 	DPU_CLK_CTRL_RGB2, | ||||
| 	DPU_CLK_CTRL_RGB3, | ||||
| 	DPU_CLK_CTRL_DMA0, | ||||
| 	DPU_CLK_CTRL_DMA1, | ||||
| 	DPU_CLK_CTRL_CURSOR0, | ||||
| 	DPU_CLK_CTRL_CURSOR1, | ||||
| 	DPU_CLK_CTRL_INLINE_ROT0_SSPP, | ||||
| 	DPU_CLK_CTRL_MAX, | ||||
| }; | ||||
| 
 | ||||
| /* struct dpu_clk_ctrl_reg : Clock control register
 | ||||
|  * @reg_off:           register offset | ||||
|  * @bit_off:           bit offset | ||||
|  */ | ||||
| struct dpu_clk_ctrl_reg { | ||||
| 	u32 reg_off; | ||||
| 	u32 bit_off; | ||||
| }; | ||||
| 
 | ||||
| /* struct dpu_mdp_cfg : MDP TOP-BLK instance info
 | ||||
|  * @id:                index identifying this block | ||||
|  * @base:              register base offset to mdss | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @highest_bank_bit:  UBWC parameter | ||||
|  * @ubwc_static:       ubwc static configuration | ||||
|  * @ubwc_swizzle:      ubwc default swizzle setting | ||||
|  * @has_dest_scaler:   indicates support of destination scaler | ||||
|  * @clk_ctrls          clock control register definition | ||||
|  */ | ||||
| struct dpu_mdp_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	u32 highest_bank_bit; | ||||
| 	u32 ubwc_static; | ||||
| 	u32 ubwc_swizzle; | ||||
| 	bool has_dest_scaler; | ||||
| 	struct dpu_clk_ctrl_reg clk_ctrls[DPU_CLK_CTRL_MAX]; | ||||
| }; | ||||
| 
 | ||||
| /* struct dpu_mdp_cfg : MDP TOP-BLK instance info
 | ||||
|  * @id:                index identifying this block | ||||
|  * @base:              register base offset to mdss | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  */ | ||||
| struct dpu_ctl_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_sspp_cfg - information of source pipes | ||||
|  * @id:                index identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @sblk:              SSPP sub-blocks information | ||||
|  * @xin_id:            bus client identifier | ||||
|  * @clk_ctrl           clock control identifier | ||||
|  * @type               sspp type identifier | ||||
|  */ | ||||
| struct dpu_sspp_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	const struct dpu_sspp_sub_blks *sblk; | ||||
| 	u32 xin_id; | ||||
| 	enum dpu_clk_ctrl_type clk_ctrl; | ||||
| 	u32 type; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_lm_cfg - information of layer mixer blocks | ||||
|  * @id:                index identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @sblk:              LM Sub-blocks information | ||||
|  * @pingpong:          ID of connected PingPong, PINGPONG_MAX if unsupported | ||||
|  * @ds:                ID of connected DS, DS_MAX if unsupported | ||||
|  * @lm_pair_mask:      Bitmask of LMs that can be controlled by same CTL | ||||
|  */ | ||||
| struct dpu_lm_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	const struct dpu_lm_sub_blks *sblk; | ||||
| 	u32 pingpong; | ||||
| 	u32 ds; | ||||
| 	unsigned long lm_pair_mask; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_ds_top_cfg - information of dest scaler top | ||||
|  * @id               enum identifying this block | ||||
|  * @base             register offset of this block | ||||
|  * @features         bit mask identifying features | ||||
|  * @version          hw version of dest scaler | ||||
|  * @maxinputwidth    maximum input line width | ||||
|  * @maxoutputwidth   maximum output line width | ||||
|  * @maxupscale       maximum upscale ratio | ||||
|  */ | ||||
| struct dpu_ds_top_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	u32 version; | ||||
| 	u32 maxinputwidth; | ||||
| 	u32 maxoutputwidth; | ||||
| 	u32 maxupscale; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_ds_cfg - information of dest scaler blocks | ||||
|  * @id          enum identifying this block | ||||
|  * @base        register offset wrt DS top offset | ||||
|  * @features    bit mask identifying features | ||||
|  * @version     hw version of the qseed block | ||||
|  * @top         DS top information | ||||
|  */ | ||||
| struct dpu_ds_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	u32 version; | ||||
| 	const struct dpu_ds_top_cfg *top; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_pingpong_cfg - information of PING-PONG blocks | ||||
|  * @id                 enum identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @sblk               sub-blocks information | ||||
|  */ | ||||
| struct dpu_pingpong_cfg  { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	const struct dpu_pingpong_sub_blks *sblk; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_cdm_cfg - information of chroma down blocks | ||||
|  * @id                 enum identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @intf_connect       Bitmask of INTF IDs this CDM can connect to | ||||
|  */ | ||||
| struct dpu_cdm_cfg   { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	unsigned long intf_connect; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_intf_cfg - information of timing engine blocks | ||||
|  * @id                 enum identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @type:              Interface type(DSI, DP, HDMI) | ||||
|  * @controller_id:     Controller Instance ID in case of multiple of intf type | ||||
|  * @prog_fetch_lines_worst_case	Worst case latency num lines needed to prefetch | ||||
|  */ | ||||
| struct dpu_intf_cfg  { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	u32 type;   /* interface type*/ | ||||
| 	u32 controller_id; | ||||
| 	u32 prog_fetch_lines_worst_case; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_vbif_dynamic_ot_cfg - dynamic OT setting | ||||
|  * @pps                pixel per seconds | ||||
|  * @ot_limit           OT limit to use up to specified pixel per second | ||||
|  */ | ||||
| struct dpu_vbif_dynamic_ot_cfg { | ||||
| 	u64 pps; | ||||
| 	u32 ot_limit; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_vbif_dynamic_ot_tbl - dynamic OT setting table | ||||
|  * @count              length of cfg | ||||
|  * @cfg                pointer to array of configuration settings with | ||||
|  *                     ascending requirements | ||||
|  */ | ||||
| struct dpu_vbif_dynamic_ot_tbl { | ||||
| 	u32 count; | ||||
| 	struct dpu_vbif_dynamic_ot_cfg *cfg; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_vbif_qos_tbl - QoS priority table | ||||
|  * @npriority_lvl      num of priority level | ||||
|  * @priority_lvl       pointer to array of priority level in ascending order | ||||
|  */ | ||||
| struct dpu_vbif_qos_tbl { | ||||
| 	u32 npriority_lvl; | ||||
| 	u32 *priority_lvl; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_vbif_cfg - information of VBIF blocks | ||||
|  * @id                 enum identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @ot_rd_limit        default OT read limit | ||||
|  * @ot_wr_limit        default OT write limit | ||||
|  * @xin_halt_timeout   maximum time (in usec) for xin to halt | ||||
|  * @dynamic_ot_rd_tbl  dynamic OT read configuration table | ||||
|  * @dynamic_ot_wr_tbl  dynamic OT write configuration table | ||||
|  * @qos_rt_tbl         real-time QoS priority table | ||||
|  * @qos_nrt_tbl        non-real-time QoS priority table | ||||
|  * @memtype_count      number of defined memtypes | ||||
|  * @memtype            array of xin memtype definitions | ||||
|  */ | ||||
| struct dpu_vbif_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	u32 default_ot_rd_limit; | ||||
| 	u32 default_ot_wr_limit; | ||||
| 	u32 xin_halt_timeout; | ||||
| 	struct dpu_vbif_dynamic_ot_tbl dynamic_ot_rd_tbl; | ||||
| 	struct dpu_vbif_dynamic_ot_tbl dynamic_ot_wr_tbl; | ||||
| 	struct dpu_vbif_qos_tbl qos_rt_tbl; | ||||
| 	struct dpu_vbif_qos_tbl qos_nrt_tbl; | ||||
| 	u32 memtype_count; | ||||
| 	u32 memtype[MAX_XIN_COUNT]; | ||||
| }; | ||||
| /**
 | ||||
|  * struct dpu_reg_dma_cfg - information of lut dma blocks | ||||
|  * @id                 enum identifying this block | ||||
|  * @base               register offset of this block | ||||
|  * @features           bit mask identifying sub-blocks/features | ||||
|  * @version            version of lutdma hw block | ||||
|  * @trigger_sel_off    offset to trigger select registers of lutdma | ||||
|  */ | ||||
| struct dpu_reg_dma_cfg { | ||||
| 	DPU_HW_BLK_INFO; | ||||
| 	u32 version; | ||||
| 	u32 trigger_sel_off; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * Define CDP use cases | ||||
|  * @DPU_PERF_CDP_UDAGE_RT: real-time use cases | ||||
|  * @DPU_PERF_CDP_USAGE_NRT: non real-time use cases such as WFD | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_PERF_CDP_USAGE_RT, | ||||
| 	DPU_PERF_CDP_USAGE_NRT, | ||||
| 	DPU_PERF_CDP_USAGE_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_perf_cdp_cfg - define CDP use case configuration | ||||
|  * @rd_enable: true if read pipe CDP is enabled | ||||
|  * @wr_enable: true if write pipe CDP is enabled | ||||
|  */ | ||||
| struct dpu_perf_cdp_cfg { | ||||
| 	bool rd_enable; | ||||
| 	bool wr_enable; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_perf_cfg - performance control settings | ||||
|  * @max_bw_low         low threshold of maximum bandwidth (kbps) | ||||
|  * @max_bw_high        high threshold of maximum bandwidth (kbps) | ||||
|  * @min_core_ib        minimum bandwidth for core (kbps) | ||||
|  * @min_core_ib        minimum mnoc ib vote in kbps | ||||
|  * @min_llcc_ib        minimum llcc ib vote in kbps | ||||
|  * @min_dram_ib        minimum dram ib vote in kbps | ||||
|  * @core_ib_ff         core instantaneous bandwidth fudge factor | ||||
|  * @core_clk_ff        core clock fudge factor | ||||
|  * @comp_ratio_rt      string of 0 or more of <fourcc>/<ven>/<mod>/<comp ratio> | ||||
|  * @comp_ratio_nrt     string of 0 or more of <fourcc>/<ven>/<mod>/<comp ratio> | ||||
|  * @undersized_prefill_lines   undersized prefill in lines | ||||
|  * @xtra_prefill_lines         extra prefill latency in lines | ||||
|  * @dest_scale_prefill_lines   destination scaler latency in lines | ||||
|  * @macrotile_perfill_lines    macrotile latency in lines | ||||
|  * @yuv_nv12_prefill_lines     yuv_nv12 latency in lines | ||||
|  * @linear_prefill_lines       linear latency in lines | ||||
|  * @downscaling_prefill_lines  downscaling latency in lines | ||||
|  * @amortizable_theshold minimum y position for traffic shaping prefill | ||||
|  * @min_prefill_lines  minimum pipeline latency in lines | ||||
|  * @safe_lut_tbl: LUT tables for safe signals | ||||
|  * @danger_lut_tbl: LUT tables for danger signals | ||||
|  * @qos_lut_tbl: LUT tables for QoS signals | ||||
|  * @cdp_cfg            cdp use case configurations | ||||
|  */ | ||||
| struct dpu_perf_cfg { | ||||
| 	u32 max_bw_low; | ||||
| 	u32 max_bw_high; | ||||
| 	u32 min_core_ib; | ||||
| 	u32 min_llcc_ib; | ||||
| 	u32 min_dram_ib; | ||||
| 	const char *core_ib_ff; | ||||
| 	const char *core_clk_ff; | ||||
| 	const char *comp_ratio_rt; | ||||
| 	const char *comp_ratio_nrt; | ||||
| 	u32 undersized_prefill_lines; | ||||
| 	u32 xtra_prefill_lines; | ||||
| 	u32 dest_scale_prefill_lines; | ||||
| 	u32 macrotile_prefill_lines; | ||||
| 	u32 yuv_nv12_prefill_lines; | ||||
| 	u32 linear_prefill_lines; | ||||
| 	u32 downscaling_prefill_lines; | ||||
| 	u32 amortizable_threshold; | ||||
| 	u32 min_prefill_lines; | ||||
| 	u32 safe_lut_tbl[DPU_QOS_LUT_USAGE_MAX]; | ||||
| 	u32 danger_lut_tbl[DPU_QOS_LUT_USAGE_MAX]; | ||||
| 	struct dpu_qos_lut_tbl qos_lut_tbl[DPU_QOS_LUT_USAGE_MAX]; | ||||
| 	struct dpu_perf_cdp_cfg cdp_cfg[DPU_PERF_CDP_USAGE_MAX]; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_mdss_cfg - information of MDSS HW | ||||
|  * This is the main catalog data structure representing | ||||
|  * this HW version. Contains number of instances, | ||||
|  * register offsets, capabilities of the all MDSS HW sub-blocks. | ||||
|  * | ||||
|  * @dma_formats        Supported formats for dma pipe | ||||
|  * @cursor_formats     Supported formats for cursor pipe | ||||
|  * @vig_formats        Supported formats for vig pipe | ||||
|  */ | ||||
| struct dpu_mdss_cfg { | ||||
| 	u32 hwversion; | ||||
| 
 | ||||
| 	const struct dpu_caps *caps; | ||||
| 
 | ||||
| 	u32 mdp_count; | ||||
| 	struct dpu_mdp_cfg *mdp; | ||||
| 
 | ||||
| 	u32 ctl_count; | ||||
| 	struct dpu_ctl_cfg *ctl; | ||||
| 
 | ||||
| 	u32 sspp_count; | ||||
| 	struct dpu_sspp_cfg *sspp; | ||||
| 
 | ||||
| 	u32 mixer_count; | ||||
| 	struct dpu_lm_cfg *mixer; | ||||
| 
 | ||||
| 	u32 ds_count; | ||||
| 	struct dpu_ds_cfg *ds; | ||||
| 
 | ||||
| 	u32 pingpong_count; | ||||
| 	struct dpu_pingpong_cfg *pingpong; | ||||
| 
 | ||||
| 	u32 cdm_count; | ||||
| 	struct dpu_cdm_cfg *cdm; | ||||
| 
 | ||||
| 	u32 intf_count; | ||||
| 	struct dpu_intf_cfg *intf; | ||||
| 
 | ||||
| 	u32 vbif_count; | ||||
| 	struct dpu_vbif_cfg *vbif; | ||||
| 
 | ||||
| 	u32 reg_dma_count; | ||||
| 	struct dpu_reg_dma_cfg dma_cfg; | ||||
| 
 | ||||
| 	u32 ad_count; | ||||
| 
 | ||||
| 	/* Add additional block data structures here */ | ||||
| 
 | ||||
| 	struct dpu_perf_cfg perf; | ||||
| 	struct dpu_format_extended *dma_formats; | ||||
| 	struct dpu_format_extended *cursor_formats; | ||||
| 	struct dpu_format_extended *vig_formats; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_mdss_hw_cfg_handler { | ||||
| 	u32 hw_rev; | ||||
| 	void (*cfg_init)(struct dpu_mdss_cfg *dpu_cfg); | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * Access Macros | ||||
|  */ | ||||
| #define BLK_MDP(s) ((s)->mdp) | ||||
| #define BLK_CTL(s) ((s)->ctl) | ||||
| #define BLK_VIG(s) ((s)->vig) | ||||
| #define BLK_RGB(s) ((s)->rgb) | ||||
| #define BLK_DMA(s) ((s)->dma) | ||||
| #define BLK_CURSOR(s) ((s)->cursor) | ||||
| #define BLK_MIXER(s) ((s)->mixer) | ||||
| #define BLK_DS(s) ((s)->ds) | ||||
| #define BLK_PINGPONG(s) ((s)->pingpong) | ||||
| #define BLK_CDM(s) ((s)->cdm) | ||||
| #define BLK_INTF(s) ((s)->intf) | ||||
| #define BLK_AD(s) ((s)->ad) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_catalog_init - dpu hardware catalog init API retrieves | ||||
|  * hardcoded target specific catalog information in config structure | ||||
|  * @hw_rev:       caller needs provide the hardware revision. | ||||
|  * | ||||
|  * Return: dpu config structure | ||||
|  */ | ||||
| struct dpu_mdss_cfg *dpu_hw_catalog_init(u32 hw_rev); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_catalog_deinit - dpu hardware catalog cleanup | ||||
|  * @dpu_cfg:      pointer returned from init function | ||||
|  */ | ||||
| void dpu_hw_catalog_deinit(struct dpu_mdss_cfg *dpu_cfg); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_sspp_multirect_enabled - check multirect enabled for the sspp | ||||
|  * @cfg:          pointer to sspp cfg | ||||
|  */ | ||||
| static inline bool dpu_hw_sspp_multirect_enabled(const struct dpu_sspp_cfg *cfg) | ||||
| { | ||||
| 	return test_bit(DPU_SSPP_SMART_DMA_V1, &cfg->features) || | ||||
| 			 test_bit(DPU_SSPP_SMART_DMA_V2, &cfg->features); | ||||
| } | ||||
| #endif /* _DPU_HW_CATALOG_H */ | ||||
							
								
								
									
										168
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_catalog_format.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_catalog_format.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,168 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| 
 | ||||
| static const struct dpu_format_extended plane_formats[] = { | ||||
| 	{DRM_FORMAT_ARGB8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, 0}, | ||||
| 	{DRM_FORMAT_RGBA8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_BGRA8888, 0}, | ||||
| 	{DRM_FORMAT_XRGB8888, 0}, | ||||
| 	{DRM_FORMAT_RGBX8888, 0}, | ||||
| 	{DRM_FORMAT_BGRX8888, 0}, | ||||
| 	{DRM_FORMAT_XBGR8888, 0}, | ||||
| 	{DRM_FORMAT_XBGR8888, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_RGB888, 0}, | ||||
| 	{DRM_FORMAT_BGR888, 0}, | ||||
| 	{DRM_FORMAT_RGB565, 0}, | ||||
| 	{DRM_FORMAT_BGR565, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_BGR565, 0}, | ||||
| 	{DRM_FORMAT_ARGB1555, 0}, | ||||
| 	{DRM_FORMAT_ABGR1555, 0}, | ||||
| 	{DRM_FORMAT_RGBA5551, 0}, | ||||
| 	{DRM_FORMAT_BGRA5551, 0}, | ||||
| 	{DRM_FORMAT_XRGB1555, 0}, | ||||
| 	{DRM_FORMAT_XBGR1555, 0}, | ||||
| 	{DRM_FORMAT_RGBX5551, 0}, | ||||
| 	{DRM_FORMAT_BGRX5551, 0}, | ||||
| 	{DRM_FORMAT_ARGB4444, 0}, | ||||
| 	{DRM_FORMAT_ABGR4444, 0}, | ||||
| 	{DRM_FORMAT_RGBA4444, 0}, | ||||
| 	{DRM_FORMAT_BGRA4444, 0}, | ||||
| 	{DRM_FORMAT_XRGB4444, 0}, | ||||
| 	{DRM_FORMAT_XBGR4444, 0}, | ||||
| 	{DRM_FORMAT_RGBX4444, 0}, | ||||
| 	{DRM_FORMAT_BGRX4444, 0}, | ||||
| 	{0, 0}, | ||||
| }; | ||||
| 
 | ||||
| static const struct dpu_format_extended plane_formats_yuv[] = { | ||||
| 	{DRM_FORMAT_ARGB8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, 0}, | ||||
| 	{DRM_FORMAT_RGBA8888, 0}, | ||||
| 	{DRM_FORMAT_BGRX8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_BGRA8888, 0}, | ||||
| 	{DRM_FORMAT_XRGB8888, 0}, | ||||
| 	{DRM_FORMAT_XBGR8888, 0}, | ||||
| 	{DRM_FORMAT_RGBX8888, 0}, | ||||
| 	{DRM_FORMAT_XBGR8888, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_RGB888, 0}, | ||||
| 	{DRM_FORMAT_BGR888, 0}, | ||||
| 	{DRM_FORMAT_RGB565, 0}, | ||||
| 	{DRM_FORMAT_BGR565, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_BGR565, 0}, | ||||
| 	{DRM_FORMAT_ARGB1555, 0}, | ||||
| 	{DRM_FORMAT_ABGR1555, 0}, | ||||
| 	{DRM_FORMAT_RGBA5551, 0}, | ||||
| 	{DRM_FORMAT_BGRA5551, 0}, | ||||
| 	{DRM_FORMAT_XRGB1555, 0}, | ||||
| 	{DRM_FORMAT_XBGR1555, 0}, | ||||
| 	{DRM_FORMAT_RGBX5551, 0}, | ||||
| 	{DRM_FORMAT_BGRX5551, 0}, | ||||
| 	{DRM_FORMAT_ARGB4444, 0}, | ||||
| 	{DRM_FORMAT_ABGR4444, 0}, | ||||
| 	{DRM_FORMAT_RGBA4444, 0}, | ||||
| 	{DRM_FORMAT_BGRA4444, 0}, | ||||
| 	{DRM_FORMAT_XRGB4444, 0}, | ||||
| 	{DRM_FORMAT_XBGR4444, 0}, | ||||
| 	{DRM_FORMAT_RGBX4444, 0}, | ||||
| 	{DRM_FORMAT_BGRX4444, 0}, | ||||
| 
 | ||||
| 	{DRM_FORMAT_NV12, 0}, | ||||
| 	{DRM_FORMAT_NV12, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_NV21, 0}, | ||||
| 	{DRM_FORMAT_NV16, 0}, | ||||
| 	{DRM_FORMAT_NV61, 0}, | ||||
| 	{DRM_FORMAT_VYUY, 0}, | ||||
| 	{DRM_FORMAT_UYVY, 0}, | ||||
| 	{DRM_FORMAT_YUYV, 0}, | ||||
| 	{DRM_FORMAT_YVYU, 0}, | ||||
| 	{DRM_FORMAT_YUV420, 0}, | ||||
| 	{DRM_FORMAT_YVU420, 0}, | ||||
| 	{0, 0}, | ||||
| }; | ||||
| 
 | ||||
| static const struct dpu_format_extended cursor_formats[] = { | ||||
| 	{DRM_FORMAT_ARGB8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, 0}, | ||||
| 	{DRM_FORMAT_RGBA8888, 0}, | ||||
| 	{DRM_FORMAT_BGRA8888, 0}, | ||||
| 	{DRM_FORMAT_XRGB8888, 0}, | ||||
| 	{DRM_FORMAT_ARGB1555, 0}, | ||||
| 	{DRM_FORMAT_ABGR1555, 0}, | ||||
| 	{DRM_FORMAT_RGBA5551, 0}, | ||||
| 	{DRM_FORMAT_BGRA5551, 0}, | ||||
| 	{DRM_FORMAT_ARGB4444, 0}, | ||||
| 	{DRM_FORMAT_ABGR4444, 0}, | ||||
| 	{DRM_FORMAT_RGBA4444, 0}, | ||||
| 	{DRM_FORMAT_BGRA4444, 0}, | ||||
| 	{0, 0}, | ||||
| }; | ||||
| 
 | ||||
| static const struct dpu_format_extended wb2_formats[] = { | ||||
| 	{DRM_FORMAT_RGB565, 0}, | ||||
| 	{DRM_FORMAT_BGR565, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_RGB888, 0}, | ||||
| 	{DRM_FORMAT_ARGB8888, 0}, | ||||
| 	{DRM_FORMAT_RGBA8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_XRGB8888, 0}, | ||||
| 	{DRM_FORMAT_RGBX8888, 0}, | ||||
| 	{DRM_FORMAT_XBGR8888, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_ARGB1555, 0}, | ||||
| 	{DRM_FORMAT_RGBA5551, 0}, | ||||
| 	{DRM_FORMAT_XRGB1555, 0}, | ||||
| 	{DRM_FORMAT_RGBX5551, 0}, | ||||
| 	{DRM_FORMAT_ARGB4444, 0}, | ||||
| 	{DRM_FORMAT_RGBA4444, 0}, | ||||
| 	{DRM_FORMAT_RGBX4444, 0}, | ||||
| 	{DRM_FORMAT_XRGB4444, 0}, | ||||
| 
 | ||||
| 	{DRM_FORMAT_BGR565, 0}, | ||||
| 	{DRM_FORMAT_BGR888, 0}, | ||||
| 	{DRM_FORMAT_ABGR8888, 0}, | ||||
| 	{DRM_FORMAT_BGRA8888, 0}, | ||||
| 	{DRM_FORMAT_BGRX8888, 0}, | ||||
| 	{DRM_FORMAT_XBGR8888, 0}, | ||||
| 	{DRM_FORMAT_ABGR1555, 0}, | ||||
| 	{DRM_FORMAT_BGRA5551, 0}, | ||||
| 	{DRM_FORMAT_XBGR1555, 0}, | ||||
| 	{DRM_FORMAT_BGRX5551, 0}, | ||||
| 	{DRM_FORMAT_ABGR4444, 0}, | ||||
| 	{DRM_FORMAT_BGRA4444, 0}, | ||||
| 	{DRM_FORMAT_BGRX4444, 0}, | ||||
| 	{DRM_FORMAT_XBGR4444, 0}, | ||||
| 
 | ||||
| 	{DRM_FORMAT_YUV420, 0}, | ||||
| 	{DRM_FORMAT_NV12, 0}, | ||||
| 	{DRM_FORMAT_NV12, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_NV16, 0}, | ||||
| 	{DRM_FORMAT_YUYV, 0}, | ||||
| 
 | ||||
| 	{0, 0}, | ||||
| }; | ||||
| 
 | ||||
| static const struct dpu_format_extended rgb_10bit_formats[] = { | ||||
| 	{DRM_FORMAT_BGRA1010102, 0}, | ||||
| 	{DRM_FORMAT_BGRX1010102, 0}, | ||||
| 	{DRM_FORMAT_RGBA1010102, 0}, | ||||
| 	{DRM_FORMAT_RGBX1010102, 0}, | ||||
| 	{DRM_FORMAT_ABGR2101010, 0}, | ||||
| 	{DRM_FORMAT_ABGR2101010, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_XBGR2101010, 0}, | ||||
| 	{DRM_FORMAT_XBGR2101010, DRM_FORMAT_MOD_QCOM_COMPRESSED}, | ||||
| 	{DRM_FORMAT_ARGB2101010, 0}, | ||||
| 	{DRM_FORMAT_XRGB2101010, 0}, | ||||
| }; | ||||
							
								
								
									
										323
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_cdm.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										323
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_cdm.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,323 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_cdm.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define CDM_CSC_10_OPMODE                  0x000 | ||||
| #define CDM_CSC_10_BASE                    0x004 | ||||
| 
 | ||||
| #define CDM_CDWN2_OP_MODE                  0x100 | ||||
| #define CDM_CDWN2_CLAMP_OUT                0x104 | ||||
| #define CDM_CDWN2_PARAMS_3D_0              0x108 | ||||
| #define CDM_CDWN2_PARAMS_3D_1              0x10C | ||||
| #define CDM_CDWN2_COEFF_COSITE_H_0         0x110 | ||||
| #define CDM_CDWN2_COEFF_COSITE_H_1         0x114 | ||||
| #define CDM_CDWN2_COEFF_COSITE_H_2         0x118 | ||||
| #define CDM_CDWN2_COEFF_OFFSITE_H_0        0x11C | ||||
| #define CDM_CDWN2_COEFF_OFFSITE_H_1        0x120 | ||||
| #define CDM_CDWN2_COEFF_OFFSITE_H_2        0x124 | ||||
| #define CDM_CDWN2_COEFF_COSITE_V           0x128 | ||||
| #define CDM_CDWN2_COEFF_OFFSITE_V          0x12C | ||||
| #define CDM_CDWN2_OUT_SIZE                 0x130 | ||||
| 
 | ||||
| #define CDM_HDMI_PACK_OP_MODE              0x200 | ||||
| #define CDM_CSC_10_MATRIX_COEFF_0          0x004 | ||||
| 
 | ||||
| /**
 | ||||
|  * Horizontal coefficients for cosite chroma downscale | ||||
|  * s13 representation of coefficients | ||||
|  */ | ||||
| static u32 cosite_h_coeff[] = {0x00000016, 0x000001cc, 0x0100009e}; | ||||
| 
 | ||||
| /**
 | ||||
|  * Horizontal coefficients for offsite chroma downscale | ||||
|  */ | ||||
| static u32 offsite_h_coeff[] = {0x000b0005, 0x01db01eb, 0x00e40046}; | ||||
| 
 | ||||
| /**
 | ||||
|  * Vertical coefficients for cosite chroma downscale | ||||
|  */ | ||||
| static u32 cosite_v_coeff[] = {0x00080004}; | ||||
| /**
 | ||||
|  * Vertical coefficients for offsite chroma downscale | ||||
|  */ | ||||
| static u32 offsite_v_coeff[] = {0x00060002}; | ||||
| 
 | ||||
| /* Limited Range rgb2yuv coeff with clamp and bias values for CSC 10 module */ | ||||
| static struct dpu_csc_cfg rgb2yuv_cfg = { | ||||
| 	{ | ||||
| 		0x0083, 0x0102, 0x0032, | ||||
| 		0x1fb5, 0x1f6c, 0x00e1, | ||||
| 		0x00e1, 0x1f45, 0x1fdc | ||||
| 	}, | ||||
| 	{ 0x00, 0x00, 0x00 }, | ||||
| 	{ 0x0040, 0x0200, 0x0200 }, | ||||
| 	{ 0x000, 0x3ff, 0x000, 0x3ff, 0x000, 0x3ff }, | ||||
| 	{ 0x040, 0x3ac, 0x040, 0x3c0, 0x040, 0x3c0 }, | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_cdm_cfg *_cdm_offset(enum dpu_cdm cdm, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < m->cdm_count; i++) { | ||||
| 		if (cdm == m->cdm[i].id) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->cdm[i].base; | ||||
| 			b->length = m->cdm[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_CDM; | ||||
| 			return &m->cdm[i]; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-EINVAL); | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_cdm_setup_csc_10bit(struct dpu_hw_cdm *ctx, | ||||
| 		struct dpu_csc_cfg *data) | ||||
| { | ||||
| 	dpu_hw_csc_setup(&ctx->hw, CDM_CSC_10_MATRIX_COEFF_0, data, true); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_cdm_setup_cdwn(struct dpu_hw_cdm *ctx, | ||||
| 		struct dpu_hw_cdm_cfg *cfg) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 opmode = 0; | ||||
| 	u32 out_size = 0; | ||||
| 
 | ||||
| 	if (cfg->output_bit_depth == CDM_CDWN_OUTPUT_10BIT) | ||||
| 		opmode &= ~BIT(7); | ||||
| 	else | ||||
| 		opmode |= BIT(7); | ||||
| 
 | ||||
| 	/* ENABLE DWNS_H bit */ | ||||
| 	opmode |= BIT(1); | ||||
| 
 | ||||
| 	switch (cfg->h_cdwn_type) { | ||||
| 	case CDM_CDWN_DISABLE: | ||||
| 		/* CLEAR METHOD_H field */ | ||||
| 		opmode &= ~(0x18); | ||||
| 		/* CLEAR DWNS_H bit */ | ||||
| 		opmode &= ~BIT(1); | ||||
| 		break; | ||||
| 	case CDM_CDWN_PIXEL_DROP: | ||||
| 		/* Clear METHOD_H field (pixel drop is 0) */ | ||||
| 		opmode &= ~(0x18); | ||||
| 		break; | ||||
| 	case CDM_CDWN_AVG: | ||||
| 		/* Clear METHOD_H field (Average is 0x1) */ | ||||
| 		opmode &= ~(0x18); | ||||
| 		opmode |= (0x1 << 0x3); | ||||
| 		break; | ||||
| 	case CDM_CDWN_COSITE: | ||||
| 		/* Clear METHOD_H field (Average is 0x2) */ | ||||
| 		opmode &= ~(0x18); | ||||
| 		opmode |= (0x2 << 0x3); | ||||
| 		/* Co-site horizontal coefficients */ | ||||
| 		DPU_REG_WRITE(c, CDM_CDWN2_COEFF_COSITE_H_0, | ||||
| 				cosite_h_coeff[0]); | ||||
| 		DPU_REG_WRITE(c, CDM_CDWN2_COEFF_COSITE_H_1, | ||||
| 				cosite_h_coeff[1]); | ||||
| 		DPU_REG_WRITE(c, CDM_CDWN2_COEFF_COSITE_H_2, | ||||
| 				cosite_h_coeff[2]); | ||||
| 		break; | ||||
| 	case CDM_CDWN_OFFSITE: | ||||
| 		/* Clear METHOD_H field (Average is 0x3) */ | ||||
| 		opmode &= ~(0x18); | ||||
| 		opmode |= (0x3 << 0x3); | ||||
| 
 | ||||
| 		/* Off-site horizontal coefficients */ | ||||
| 		DPU_REG_WRITE(c, CDM_CDWN2_COEFF_OFFSITE_H_0, | ||||
| 				offsite_h_coeff[0]); | ||||
| 		DPU_REG_WRITE(c, CDM_CDWN2_COEFF_OFFSITE_H_1, | ||||
| 				offsite_h_coeff[1]); | ||||
| 		DPU_REG_WRITE(c, CDM_CDWN2_COEFF_OFFSITE_H_2, | ||||
| 				offsite_h_coeff[2]); | ||||
| 		break; | ||||
| 	default: | ||||
| 		pr_err("%s invalid horz down sampling type\n", __func__); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	/* ENABLE DWNS_V bit */ | ||||
| 	opmode |= BIT(2); | ||||
| 
 | ||||
| 	switch (cfg->v_cdwn_type) { | ||||
| 	case CDM_CDWN_DISABLE: | ||||
| 		/* CLEAR METHOD_V field */ | ||||
| 		opmode &= ~(0x60); | ||||
| 		/* CLEAR DWNS_V bit */ | ||||
| 		opmode &= ~BIT(2); | ||||
| 		break; | ||||
| 	case CDM_CDWN_PIXEL_DROP: | ||||
| 		/* Clear METHOD_V field (pixel drop is 0) */ | ||||
| 		opmode &= ~(0x60); | ||||
| 		break; | ||||
| 	case CDM_CDWN_AVG: | ||||
| 		/* Clear METHOD_V field (Average is 0x1) */ | ||||
| 		opmode &= ~(0x60); | ||||
| 		opmode |= (0x1 << 0x5); | ||||
| 		break; | ||||
| 	case CDM_CDWN_COSITE: | ||||
| 		/* Clear METHOD_V field (Average is 0x2) */ | ||||
| 		opmode &= ~(0x60); | ||||
| 		opmode |= (0x2 << 0x5); | ||||
| 		/* Co-site vertical coefficients */ | ||||
| 		DPU_REG_WRITE(c, | ||||
| 				CDM_CDWN2_COEFF_COSITE_V, | ||||
| 				cosite_v_coeff[0]); | ||||
| 		break; | ||||
| 	case CDM_CDWN_OFFSITE: | ||||
| 		/* Clear METHOD_V field (Average is 0x3) */ | ||||
| 		opmode &= ~(0x60); | ||||
| 		opmode |= (0x3 << 0x5); | ||||
| 
 | ||||
| 		/* Off-site vertical coefficients */ | ||||
| 		DPU_REG_WRITE(c, | ||||
| 				CDM_CDWN2_COEFF_OFFSITE_V, | ||||
| 				offsite_v_coeff[0]); | ||||
| 		break; | ||||
| 	default: | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (cfg->v_cdwn_type || cfg->h_cdwn_type) | ||||
| 		opmode |= BIT(0); /* EN CDWN module */ | ||||
| 	else | ||||
| 		opmode &= ~BIT(0); | ||||
| 
 | ||||
| 	out_size = (cfg->output_width & 0xFFFF) | | ||||
| 		((cfg->output_height & 0xFFFF) << 16); | ||||
| 	DPU_REG_WRITE(c, CDM_CDWN2_OUT_SIZE, out_size); | ||||
| 	DPU_REG_WRITE(c, CDM_CDWN2_OP_MODE, opmode); | ||||
| 	DPU_REG_WRITE(c, CDM_CDWN2_CLAMP_OUT, | ||||
| 			((0x3FF << 16) | 0x0)); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| int dpu_hw_cdm_enable(struct dpu_hw_cdm *ctx, | ||||
| 		struct dpu_hw_cdm_cfg *cdm) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	const struct dpu_format *fmt = cdm->output_fmt; | ||||
| 	struct cdm_output_cfg cdm_cfg = { 0 }; | ||||
| 	u32 opmode = 0; | ||||
| 	u32 csc = 0; | ||||
| 
 | ||||
| 	if (!DPU_FORMAT_IS_YUV(fmt)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (cdm->output_type == CDM_CDWN_OUTPUT_HDMI) { | ||||
| 		if (fmt->chroma_sample != DPU_CHROMA_H1V2) | ||||
| 			return -EINVAL; /*unsupported format */ | ||||
| 		opmode = BIT(0); | ||||
| 		opmode |= (fmt->chroma_sample << 1); | ||||
| 		cdm_cfg.intf_en = true; | ||||
| 	} | ||||
| 
 | ||||
| 	csc |= BIT(2); | ||||
| 	csc &= ~BIT(1); | ||||
| 	csc |= BIT(0); | ||||
| 
 | ||||
| 	if (ctx->hw_mdp && ctx->hw_mdp->ops.setup_cdm_output) | ||||
| 		ctx->hw_mdp->ops.setup_cdm_output(ctx->hw_mdp, &cdm_cfg); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, CDM_CSC_10_OPMODE, csc); | ||||
| 	DPU_REG_WRITE(c, CDM_HDMI_PACK_OP_MODE, opmode); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_cdm_disable(struct dpu_hw_cdm *ctx) | ||||
| { | ||||
| 	struct cdm_output_cfg cdm_cfg = { 0 }; | ||||
| 
 | ||||
| 	if (ctx->hw_mdp && ctx->hw_mdp->ops.setup_cdm_output) | ||||
| 		ctx->hw_mdp->ops.setup_cdm_output(ctx->hw_mdp, &cdm_cfg); | ||||
| } | ||||
| 
 | ||||
| static void _setup_cdm_ops(struct dpu_hw_cdm_ops *ops, | ||||
| 	unsigned long features) | ||||
| { | ||||
| 	ops->setup_csc_data = dpu_hw_cdm_setup_csc_10bit; | ||||
| 	ops->setup_cdwn = dpu_hw_cdm_setup_cdwn; | ||||
| 	ops->enable = dpu_hw_cdm_enable; | ||||
| 	ops->disable = dpu_hw_cdm_disable; | ||||
| } | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_cdm *dpu_hw_cdm_init(enum dpu_cdm idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		struct dpu_hw_mdp *hw_mdp) | ||||
| { | ||||
| 	struct dpu_hw_cdm *c; | ||||
| 	struct dpu_cdm_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	c = kzalloc(sizeof(*c), GFP_KERNEL); | ||||
| 	if (!c) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _cdm_offset(idx, m, addr, &c->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(c); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	c->idx = idx; | ||||
| 	c->caps = cfg; | ||||
| 	_setup_cdm_ops(&c->ops, c->caps->features); | ||||
| 	c->hw_mdp = hw_mdp; | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&c->base, DPU_HW_BLK_CDM, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Perform any default initialization for the chroma down module | ||||
| 	 * @setup default csc coefficients | ||||
| 	 */ | ||||
| 	dpu_hw_cdm_setup_csc_10bit(c, &rgb2yuv_cfg); | ||||
| 
 | ||||
| 	return c; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(c); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_cdm_destroy(struct dpu_hw_cdm *cdm) | ||||
| { | ||||
| 	if (cdm) | ||||
| 		dpu_hw_blk_destroy(&cdm->base); | ||||
| 	kfree(cdm); | ||||
| } | ||||
							
								
								
									
										139
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_cdm.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_cdm.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,139 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_CDM_H | ||||
| #define _DPU_HW_CDM_H | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_top.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| struct dpu_hw_cdm; | ||||
| 
 | ||||
| struct dpu_hw_cdm_cfg { | ||||
| 	u32 output_width; | ||||
| 	u32 output_height; | ||||
| 	u32 output_bit_depth; | ||||
| 	u32 h_cdwn_type; | ||||
| 	u32 v_cdwn_type; | ||||
| 	const struct dpu_format *output_fmt; | ||||
| 	u32 output_type; | ||||
| 	int flags; | ||||
| }; | ||||
| 
 | ||||
| enum dpu_hw_cdwn_type { | ||||
| 	CDM_CDWN_DISABLE, | ||||
| 	CDM_CDWN_PIXEL_DROP, | ||||
| 	CDM_CDWN_AVG, | ||||
| 	CDM_CDWN_COSITE, | ||||
| 	CDM_CDWN_OFFSITE, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_hw_cdwn_output_type { | ||||
| 	CDM_CDWN_OUTPUT_HDMI, | ||||
| 	CDM_CDWN_OUTPUT_WB, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_hw_cdwn_output_bit_depth { | ||||
| 	CDM_CDWN_OUTPUT_8BIT, | ||||
| 	CDM_CDWN_OUTPUT_10BIT, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_cdm_ops : Interface to the chroma down Hw driver functions | ||||
|  *                         Assumption is these functions will be called after | ||||
|  *                         clocks are enabled | ||||
|  *  @setup_csc:            Programs the csc matrix | ||||
|  *  @setup_cdwn:           Sets up the chroma down sub module | ||||
|  *  @enable:               Enables the output to interface and programs the | ||||
|  *                         output packer | ||||
|  *  @disable:              Puts the cdm in bypass mode | ||||
|  */ | ||||
| struct dpu_hw_cdm_ops { | ||||
| 	/**
 | ||||
| 	 * Programs the CSC matrix for conversion from RGB space to YUV space, | ||||
| 	 * it is optional to call this function as this matrix is automatically | ||||
| 	 * set during initialization, user should call this if it wants | ||||
| 	 * to program a different matrix than default matrix. | ||||
| 	 * @cdm:          Pointer to the chroma down context structure | ||||
| 	 * @data          Pointer to CSC configuration data | ||||
| 	 * return:        0 if success; error code otherwise | ||||
| 	 */ | ||||
| 	int (*setup_csc_data)(struct dpu_hw_cdm *cdm, | ||||
| 			struct dpu_csc_cfg *data); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Programs the Chroma downsample part. | ||||
| 	 * @cdm         Pointer to chroma down context | ||||
| 	 */ | ||||
| 	int (*setup_cdwn)(struct dpu_hw_cdm *cdm, | ||||
| 	struct dpu_hw_cdm_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Enable the CDM module | ||||
| 	 * @cdm         Pointer to chroma down context | ||||
| 	 */ | ||||
| 	int (*enable)(struct dpu_hw_cdm *cdm, | ||||
| 	struct dpu_hw_cdm_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Disable the CDM module | ||||
| 	 * @cdm         Pointer to chroma down context | ||||
| 	 */ | ||||
| 	void (*disable)(struct dpu_hw_cdm *cdm); | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_cdm { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* chroma down */ | ||||
| 	const struct dpu_cdm_cfg *caps; | ||||
| 	enum  dpu_cdm  idx; | ||||
| 
 | ||||
| 	/* mdp top hw driver */ | ||||
| 	struct dpu_hw_mdp *hw_mdp; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_cdm_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_cdm - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_cdm *to_dpu_hw_cdm(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_cdm, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_cdm_init - initializes the cdm hw driver object. | ||||
|  * should be called once before accessing every cdm. | ||||
|  * @idx:  cdm index for which driver object is required | ||||
|  * @addr: mapped register io address of MDP | ||||
|  * @m :   pointer to mdss catalog data | ||||
|  * @hw_mdp:  pointer to mdp top hw driver object | ||||
|  */ | ||||
| struct dpu_hw_cdm *dpu_hw_cdm_init(enum dpu_cdm idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		struct dpu_hw_mdp *hw_mdp); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_cdm_destroy - destroys CDM driver context | ||||
|  * @cdm:   pointer to CDM driver context | ||||
|  */ | ||||
| void dpu_hw_cdm_destroy(struct dpu_hw_cdm *cdm); | ||||
| 
 | ||||
| #endif /*_DPU_HW_CDM_H */ | ||||
							
								
								
									
										540
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_ctl.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										540
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_ctl.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,540 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/delay.h> | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_ctl.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define   CTL_LAYER(lm)                 \ | ||||
| 	(((lm) == LM_5) ? (0x024) : (((lm) - LM_0) * 0x004)) | ||||
| #define   CTL_LAYER_EXT(lm)             \ | ||||
| 	(0x40 + (((lm) - LM_0) * 0x004)) | ||||
| #define   CTL_LAYER_EXT2(lm)             \ | ||||
| 	(0x70 + (((lm) - LM_0) * 0x004)) | ||||
| #define   CTL_LAYER_EXT3(lm)             \ | ||||
| 	(0xA0 + (((lm) - LM_0) * 0x004)) | ||||
| #define   CTL_TOP                       0x014 | ||||
| #define   CTL_FLUSH                     0x018 | ||||
| #define   CTL_START                     0x01C | ||||
| #define   CTL_PREPARE                   0x0d0 | ||||
| #define   CTL_SW_RESET                  0x030 | ||||
| #define   CTL_LAYER_EXTN_OFFSET         0x40 | ||||
| 
 | ||||
| #define CTL_MIXER_BORDER_OUT            BIT(24) | ||||
| #define CTL_FLUSH_MASK_CTL              BIT(17) | ||||
| 
 | ||||
| #define DPU_REG_RESET_TIMEOUT_US        2000 | ||||
| 
 | ||||
| static struct dpu_ctl_cfg *_ctl_offset(enum dpu_ctl ctl, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < m->ctl_count; i++) { | ||||
| 		if (ctl == m->ctl[i].id) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->ctl[i].base; | ||||
| 			b->length = m->ctl[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_CTL; | ||||
| 			return &m->ctl[i]; | ||||
| 		} | ||||
| 	} | ||||
| 	return ERR_PTR(-ENOMEM); | ||||
| } | ||||
| 
 | ||||
| static int _mixer_stages(const struct dpu_lm_cfg *mixer, int count, | ||||
| 		enum dpu_lm lm) | ||||
| { | ||||
| 	int i; | ||||
| 	int stages = -EINVAL; | ||||
| 
 | ||||
| 	for (i = 0; i < count; i++) { | ||||
| 		if (lm == mixer[i].id) { | ||||
| 			stages = mixer[i].sblk->maxblendstages; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return stages; | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_hw_ctl_trigger_start(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	DPU_REG_WRITE(&ctx->hw, CTL_START, 0x1); | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_hw_ctl_trigger_pending(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	DPU_REG_WRITE(&ctx->hw, CTL_PREPARE, 0x1); | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_hw_ctl_clear_pending_flush(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	ctx->pending_flush_mask = 0x0; | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_hw_ctl_update_pending_flush(struct dpu_hw_ctl *ctx, | ||||
| 		u32 flushbits) | ||||
| { | ||||
| 	ctx->pending_flush_mask |= flushbits; | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_ctl_get_pending_flush(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	if (!ctx) | ||||
| 		return 0x0; | ||||
| 
 | ||||
| 	return ctx->pending_flush_mask; | ||||
| } | ||||
| 
 | ||||
| static inline void dpu_hw_ctl_trigger_flush(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, CTL_FLUSH, ctx->pending_flush_mask); | ||||
| } | ||||
| 
 | ||||
| static inline u32 dpu_hw_ctl_get_flush_register(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 
 | ||||
| 	return DPU_REG_READ(c, CTL_FLUSH); | ||||
| } | ||||
| 
 | ||||
| static inline uint32_t dpu_hw_ctl_get_bitmask_sspp(struct dpu_hw_ctl *ctx, | ||||
| 	enum dpu_sspp sspp) | ||||
| { | ||||
| 	uint32_t flushbits = 0; | ||||
| 
 | ||||
| 	switch (sspp) { | ||||
| 	case SSPP_VIG0: | ||||
| 		flushbits =  BIT(0); | ||||
| 		break; | ||||
| 	case SSPP_VIG1: | ||||
| 		flushbits = BIT(1); | ||||
| 		break; | ||||
| 	case SSPP_VIG2: | ||||
| 		flushbits = BIT(2); | ||||
| 		break; | ||||
| 	case SSPP_VIG3: | ||||
| 		flushbits = BIT(18); | ||||
| 		break; | ||||
| 	case SSPP_RGB0: | ||||
| 		flushbits = BIT(3); | ||||
| 		break; | ||||
| 	case SSPP_RGB1: | ||||
| 		flushbits = BIT(4); | ||||
| 		break; | ||||
| 	case SSPP_RGB2: | ||||
| 		flushbits = BIT(5); | ||||
| 		break; | ||||
| 	case SSPP_RGB3: | ||||
| 		flushbits = BIT(19); | ||||
| 		break; | ||||
| 	case SSPP_DMA0: | ||||
| 		flushbits = BIT(11); | ||||
| 		break; | ||||
| 	case SSPP_DMA1: | ||||
| 		flushbits = BIT(12); | ||||
| 		break; | ||||
| 	case SSPP_DMA2: | ||||
| 		flushbits = BIT(24); | ||||
| 		break; | ||||
| 	case SSPP_DMA3: | ||||
| 		flushbits = BIT(25); | ||||
| 		break; | ||||
| 	case SSPP_CURSOR0: | ||||
| 		flushbits = BIT(22); | ||||
| 		break; | ||||
| 	case SSPP_CURSOR1: | ||||
| 		flushbits = BIT(23); | ||||
| 		break; | ||||
| 	default: | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return flushbits; | ||||
| } | ||||
| 
 | ||||
| static inline uint32_t dpu_hw_ctl_get_bitmask_mixer(struct dpu_hw_ctl *ctx, | ||||
| 	enum dpu_lm lm) | ||||
| { | ||||
| 	uint32_t flushbits = 0; | ||||
| 
 | ||||
| 	switch (lm) { | ||||
| 	case LM_0: | ||||
| 		flushbits = BIT(6); | ||||
| 		break; | ||||
| 	case LM_1: | ||||
| 		flushbits = BIT(7); | ||||
| 		break; | ||||
| 	case LM_2: | ||||
| 		flushbits = BIT(8); | ||||
| 		break; | ||||
| 	case LM_3: | ||||
| 		flushbits = BIT(9); | ||||
| 		break; | ||||
| 	case LM_4: | ||||
| 		flushbits = BIT(10); | ||||
| 		break; | ||||
| 	case LM_5: | ||||
| 		flushbits = BIT(20); | ||||
| 		break; | ||||
| 	default: | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	flushbits |= CTL_FLUSH_MASK_CTL; | ||||
| 
 | ||||
| 	return flushbits; | ||||
| } | ||||
| 
 | ||||
| static inline int dpu_hw_ctl_get_bitmask_intf(struct dpu_hw_ctl *ctx, | ||||
| 		u32 *flushbits, enum dpu_intf intf) | ||||
| { | ||||
| 	switch (intf) { | ||||
| 	case INTF_0: | ||||
| 		*flushbits |= BIT(31); | ||||
| 		break; | ||||
| 	case INTF_1: | ||||
| 		*flushbits |= BIT(30); | ||||
| 		break; | ||||
| 	case INTF_2: | ||||
| 		*flushbits |= BIT(29); | ||||
| 		break; | ||||
| 	case INTF_3: | ||||
| 		*flushbits |= BIT(28); | ||||
| 		break; | ||||
| 	default: | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static inline int dpu_hw_ctl_get_bitmask_cdm(struct dpu_hw_ctl *ctx, | ||||
| 		u32 *flushbits, enum dpu_cdm cdm) | ||||
| { | ||||
| 	switch (cdm) { | ||||
| 	case CDM_0: | ||||
| 		*flushbits |= BIT(26); | ||||
| 		break; | ||||
| 	default: | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_ctl_poll_reset_status(struct dpu_hw_ctl *ctx, u32 timeout_us) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	ktime_t timeout; | ||||
| 	u32 status; | ||||
| 
 | ||||
| 	timeout = ktime_add_us(ktime_get(), timeout_us); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * it takes around 30us to have mdp finish resetting its ctl path | ||||
| 	 * poll every 50us so that reset should be completed at 1st poll | ||||
| 	 */ | ||||
| 	do { | ||||
| 		status = DPU_REG_READ(c, CTL_SW_RESET); | ||||
| 		status &= 0x1; | ||||
| 		if (status) | ||||
| 			usleep_range(20, 50); | ||||
| 	} while (status && ktime_compare_safe(ktime_get(), timeout) < 0); | ||||
| 
 | ||||
| 	return status; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_ctl_reset_control(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 
 | ||||
| 	pr_debug("issuing hw ctl reset for ctl:%d\n", ctx->idx); | ||||
| 	DPU_REG_WRITE(c, CTL_SW_RESET, 0x1); | ||||
| 	if (dpu_hw_ctl_poll_reset_status(ctx, DPU_REG_RESET_TIMEOUT_US)) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_ctl_wait_reset_status(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 status; | ||||
| 
 | ||||
| 	status = DPU_REG_READ(c, CTL_SW_RESET); | ||||
| 	status &= 0x01; | ||||
| 	if (!status) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	pr_debug("hw ctl reset is set for ctl:%d\n", ctx->idx); | ||||
| 	if (dpu_hw_ctl_poll_reset_status(ctx, DPU_REG_RESET_TIMEOUT_US)) { | ||||
| 		pr_err("hw recovery is not complete for ctl:%d\n", ctx->idx); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_ctl_clear_all_blendstages(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < ctx->mixer_count; i++) { | ||||
| 		DPU_REG_WRITE(c, CTL_LAYER(LM_0 + i), 0); | ||||
| 		DPU_REG_WRITE(c, CTL_LAYER_EXT(LM_0 + i), 0); | ||||
| 		DPU_REG_WRITE(c, CTL_LAYER_EXT2(LM_0 + i), 0); | ||||
| 		DPU_REG_WRITE(c, CTL_LAYER_EXT3(LM_0 + i), 0); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_ctl_setup_blendstage(struct dpu_hw_ctl *ctx, | ||||
| 	enum dpu_lm lm, struct dpu_hw_stage_cfg *stage_cfg) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 mixercfg = 0, mixercfg_ext = 0, mix, ext; | ||||
| 	u32 mixercfg_ext2 = 0, mixercfg_ext3 = 0; | ||||
| 	int i, j; | ||||
| 	u8 stages; | ||||
| 	int pipes_per_stage; | ||||
| 
 | ||||
| 	stages = _mixer_stages(ctx->mixer_hw_caps, ctx->mixer_count, lm); | ||||
| 	if (stages < 0) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (test_bit(DPU_MIXER_SOURCESPLIT, | ||||
| 		&ctx->mixer_hw_caps->features)) | ||||
| 		pipes_per_stage = PIPES_PER_STAGE; | ||||
| 	else | ||||
| 		pipes_per_stage = 1; | ||||
| 
 | ||||
| 	mixercfg = CTL_MIXER_BORDER_OUT; /* always set BORDER_OUT */ | ||||
| 
 | ||||
| 	if (!stage_cfg) | ||||
| 		goto exit; | ||||
| 
 | ||||
| 	for (i = 0; i <= stages; i++) { | ||||
| 		/* overflow to ext register if 'i + 1 > 7' */ | ||||
| 		mix = (i + 1) & 0x7; | ||||
| 		ext = i >= 7; | ||||
| 
 | ||||
| 		for (j = 0 ; j < pipes_per_stage; j++) { | ||||
| 			enum dpu_sspp_multirect_index rect_index = | ||||
| 				stage_cfg->multirect_index[i][j]; | ||||
| 
 | ||||
| 			switch (stage_cfg->stage[i][j]) { | ||||
| 			case SSPP_VIG0: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext3 |= ((i + 1) & 0xF) << 0; | ||||
| 				} else { | ||||
| 					mixercfg |= mix << 0; | ||||
| 					mixercfg_ext |= ext << 0; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_VIG1: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext3 |= ((i + 1) & 0xF) << 4; | ||||
| 				} else { | ||||
| 					mixercfg |= mix << 3; | ||||
| 					mixercfg_ext |= ext << 2; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_VIG2: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext3 |= ((i + 1) & 0xF) << 8; | ||||
| 				} else { | ||||
| 					mixercfg |= mix << 6; | ||||
| 					mixercfg_ext |= ext << 4; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_VIG3: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext3 |= ((i + 1) & 0xF) << 12; | ||||
| 				} else { | ||||
| 					mixercfg |= mix << 26; | ||||
| 					mixercfg_ext |= ext << 6; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_RGB0: | ||||
| 				mixercfg |= mix << 9; | ||||
| 				mixercfg_ext |= ext << 8; | ||||
| 				break; | ||||
| 			case SSPP_RGB1: | ||||
| 				mixercfg |= mix << 12; | ||||
| 				mixercfg_ext |= ext << 10; | ||||
| 				break; | ||||
| 			case SSPP_RGB2: | ||||
| 				mixercfg |= mix << 15; | ||||
| 				mixercfg_ext |= ext << 12; | ||||
| 				break; | ||||
| 			case SSPP_RGB3: | ||||
| 				mixercfg |= mix << 29; | ||||
| 				mixercfg_ext |= ext << 14; | ||||
| 				break; | ||||
| 			case SSPP_DMA0: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext2 |= ((i + 1) & 0xF) << 8; | ||||
| 				} else { | ||||
| 					mixercfg |= mix << 18; | ||||
| 					mixercfg_ext |= ext << 16; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_DMA1: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext2 |= ((i + 1) & 0xF) << 12; | ||||
| 				} else { | ||||
| 					mixercfg |= mix << 21; | ||||
| 					mixercfg_ext |= ext << 18; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_DMA2: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext2 |= ((i + 1) & 0xF) << 16; | ||||
| 				} else { | ||||
| 					mix |= (i + 1) & 0xF; | ||||
| 					mixercfg_ext2 |= mix << 0; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_DMA3: | ||||
| 				if (rect_index == DPU_SSPP_RECT_1) { | ||||
| 					mixercfg_ext2 |= ((i + 1) & 0xF) << 20; | ||||
| 				} else { | ||||
| 					mix |= (i + 1) & 0xF; | ||||
| 					mixercfg_ext2 |= mix << 4; | ||||
| 				} | ||||
| 				break; | ||||
| 			case SSPP_CURSOR0: | ||||
| 				mixercfg_ext |= ((i + 1) & 0xF) << 20; | ||||
| 				break; | ||||
| 			case SSPP_CURSOR1: | ||||
| 				mixercfg_ext |= ((i + 1) & 0xF) << 26; | ||||
| 				break; | ||||
| 			default: | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| exit: | ||||
| 	DPU_REG_WRITE(c, CTL_LAYER(lm), mixercfg); | ||||
| 	DPU_REG_WRITE(c, CTL_LAYER_EXT(lm), mixercfg_ext); | ||||
| 	DPU_REG_WRITE(c, CTL_LAYER_EXT2(lm), mixercfg_ext2); | ||||
| 	DPU_REG_WRITE(c, CTL_LAYER_EXT3(lm), mixercfg_ext3); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_ctl_intf_cfg(struct dpu_hw_ctl *ctx, | ||||
| 		struct dpu_hw_intf_cfg *cfg) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 intf_cfg = 0; | ||||
| 
 | ||||
| 	intf_cfg |= (cfg->intf & 0xF) << 4; | ||||
| 
 | ||||
| 	if (cfg->mode_3d) { | ||||
| 		intf_cfg |= BIT(19); | ||||
| 		intf_cfg |= (cfg->mode_3d - 0x1) << 20; | ||||
| 	} | ||||
| 
 | ||||
| 	switch (cfg->intf_mode_sel) { | ||||
| 	case DPU_CTL_MODE_SEL_VID: | ||||
| 		intf_cfg &= ~BIT(17); | ||||
| 		intf_cfg &= ~(0x3 << 15); | ||||
| 		break; | ||||
| 	case DPU_CTL_MODE_SEL_CMD: | ||||
| 		intf_cfg |= BIT(17); | ||||
| 		intf_cfg |= ((cfg->stream_sel & 0x3) << 15); | ||||
| 		break; | ||||
| 	default: | ||||
| 		pr_err("unknown interface type %d\n", cfg->intf_mode_sel); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, CTL_TOP, intf_cfg); | ||||
| } | ||||
| 
 | ||||
| static void _setup_ctl_ops(struct dpu_hw_ctl_ops *ops, | ||||
| 		unsigned long cap) | ||||
| { | ||||
| 	ops->clear_pending_flush = dpu_hw_ctl_clear_pending_flush; | ||||
| 	ops->update_pending_flush = dpu_hw_ctl_update_pending_flush; | ||||
| 	ops->get_pending_flush = dpu_hw_ctl_get_pending_flush; | ||||
| 	ops->trigger_flush = dpu_hw_ctl_trigger_flush; | ||||
| 	ops->get_flush_register = dpu_hw_ctl_get_flush_register; | ||||
| 	ops->trigger_start = dpu_hw_ctl_trigger_start; | ||||
| 	ops->trigger_pending = dpu_hw_ctl_trigger_pending; | ||||
| 	ops->setup_intf_cfg = dpu_hw_ctl_intf_cfg; | ||||
| 	ops->reset = dpu_hw_ctl_reset_control; | ||||
| 	ops->wait_reset_status = dpu_hw_ctl_wait_reset_status; | ||||
| 	ops->clear_all_blendstages = dpu_hw_ctl_clear_all_blendstages; | ||||
| 	ops->setup_blendstage = dpu_hw_ctl_setup_blendstage; | ||||
| 	ops->get_bitmask_sspp = dpu_hw_ctl_get_bitmask_sspp; | ||||
| 	ops->get_bitmask_mixer = dpu_hw_ctl_get_bitmask_mixer; | ||||
| 	ops->get_bitmask_intf = dpu_hw_ctl_get_bitmask_intf; | ||||
| 	ops->get_bitmask_cdm = dpu_hw_ctl_get_bitmask_cdm; | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_ctl *dpu_hw_ctl_init(enum dpu_ctl idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_ctl *c; | ||||
| 	struct dpu_ctl_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	c = kzalloc(sizeof(*c), GFP_KERNEL); | ||||
| 	if (!c) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _ctl_offset(idx, m, addr, &c->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(c); | ||||
| 		pr_err("failed to create dpu_hw_ctl %d\n", idx); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	c->caps = cfg; | ||||
| 	_setup_ctl_ops(&c->ops, c->caps->features); | ||||
| 	c->idx = idx; | ||||
| 	c->mixer_count = m->mixer_count; | ||||
| 	c->mixer_hw_caps = m->mixer; | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&c->base, DPU_HW_BLK_CTL, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	return c; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(c); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_ctl_destroy(struct dpu_hw_ctl *ctx) | ||||
| { | ||||
| 	if (ctx) | ||||
| 		dpu_hw_blk_destroy(&ctx->base); | ||||
| 	kfree(ctx); | ||||
| } | ||||
							
								
								
									
										218
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_ctl.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										218
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_ctl.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,218 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_CTL_H | ||||
| #define _DPU_HW_CTL_H | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_sspp.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_ctl_mode_sel: Interface mode selection | ||||
|  * DPU_CTL_MODE_SEL_VID:    Video mode interface | ||||
|  * DPU_CTL_MODE_SEL_CMD:    Command mode interface | ||||
|  */ | ||||
| enum dpu_ctl_mode_sel { | ||||
| 	DPU_CTL_MODE_SEL_VID = 0, | ||||
| 	DPU_CTL_MODE_SEL_CMD | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_ctl; | ||||
| /**
 | ||||
|  * struct dpu_hw_stage_cfg - blending stage cfg | ||||
|  * @stage : SSPP_ID at each stage | ||||
|  * @multirect_index: index of the rectangle of SSPP. | ||||
|  */ | ||||
| struct dpu_hw_stage_cfg { | ||||
| 	enum dpu_sspp stage[DPU_STAGE_MAX][PIPES_PER_STAGE]; | ||||
| 	enum dpu_sspp_multirect_index multirect_index | ||||
| 					[DPU_STAGE_MAX][PIPES_PER_STAGE]; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_intf_cfg :Describes how the DPU writes data to output interface | ||||
|  * @intf :                 Interface id | ||||
|  * @mode_3d:               3d mux configuration | ||||
|  * @intf_mode_sel:         Interface mode, cmd / vid | ||||
|  * @stream_sel:            Stream selection for multi-stream interfaces | ||||
|  */ | ||||
| struct dpu_hw_intf_cfg { | ||||
| 	enum dpu_intf intf; | ||||
| 	enum dpu_3d_blend_mode mode_3d; | ||||
| 	enum dpu_ctl_mode_sel intf_mode_sel; | ||||
| 	int stream_sel; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_ctl_ops - Interface to the wb Hw driver functions | ||||
|  * Assumption is these functions will be called after clocks are enabled | ||||
|  */ | ||||
| struct dpu_hw_ctl_ops { | ||||
| 	/**
 | ||||
| 	 * kickoff hw operation for Sw controlled interfaces | ||||
| 	 * DSI cmd mode and WB interface are SW controlled | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 */ | ||||
| 	void (*trigger_start)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * kickoff prepare is in progress hw operation for sw | ||||
| 	 * controlled interfaces: DSI cmd mode and WB interface | ||||
| 	 * are SW controlled | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 */ | ||||
| 	void (*trigger_pending)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Clear the value of the cached pending_flush_mask | ||||
| 	 * No effect on hardware | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 */ | ||||
| 	void (*clear_pending_flush)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Query the value of the cached pending_flush_mask | ||||
| 	 * No effect on hardware | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 */ | ||||
| 	u32 (*get_pending_flush)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * OR in the given flushbits to the cached pending_flush_mask | ||||
| 	 * No effect on hardware | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 * @flushbits : module flushmask | ||||
| 	 */ | ||||
| 	void (*update_pending_flush)(struct dpu_hw_ctl *ctx, | ||||
| 		u32 flushbits); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Write the value of the pending_flush_mask to hardware | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 */ | ||||
| 	void (*trigger_flush)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Read the value of the flush register | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 * @Return: value of the ctl flush register. | ||||
| 	 */ | ||||
| 	u32 (*get_flush_register)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Setup ctl_path interface config | ||||
| 	 * @ctx | ||||
| 	 * @cfg    : interface config structure pointer | ||||
| 	 */ | ||||
| 	void (*setup_intf_cfg)(struct dpu_hw_ctl *ctx, | ||||
| 		struct dpu_hw_intf_cfg *cfg); | ||||
| 
 | ||||
| 	int (*reset)(struct dpu_hw_ctl *c); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * wait_reset_status - checks ctl reset status | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 * | ||||
| 	 * This function checks the ctl reset status bit. | ||||
| 	 * If the reset bit is set, it keeps polling the status till the hw | ||||
| 	 * reset is complete. | ||||
| 	 * Returns: 0 on success or -error if reset incomplete within interval | ||||
| 	 */ | ||||
| 	int (*wait_reset_status)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	uint32_t (*get_bitmask_sspp)(struct dpu_hw_ctl *ctx, | ||||
| 		enum dpu_sspp blk); | ||||
| 
 | ||||
| 	uint32_t (*get_bitmask_mixer)(struct dpu_hw_ctl *ctx, | ||||
| 		enum dpu_lm blk); | ||||
| 
 | ||||
| 	int (*get_bitmask_intf)(struct dpu_hw_ctl *ctx, | ||||
| 		u32 *flushbits, | ||||
| 		enum dpu_intf blk); | ||||
| 
 | ||||
| 	int (*get_bitmask_cdm)(struct dpu_hw_ctl *ctx, | ||||
| 		u32 *flushbits, | ||||
| 		enum dpu_cdm blk); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Set all blend stages to disabled | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 */ | ||||
| 	void (*clear_all_blendstages)(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Configure layer mixer to pipe configuration | ||||
| 	 * @ctx       : ctl path ctx pointer | ||||
| 	 * @lm        : layer mixer enumeration | ||||
| 	 * @cfg       : blend stage configuration | ||||
| 	 */ | ||||
| 	void (*setup_blendstage)(struct dpu_hw_ctl *ctx, | ||||
| 		enum dpu_lm lm, struct dpu_hw_stage_cfg *cfg); | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_ctl : CTL PATH driver object | ||||
|  * @base: hardware block base structure | ||||
|  * @hw: block register map object | ||||
|  * @idx: control path index | ||||
|  * @caps: control path capabilities | ||||
|  * @mixer_count: number of mixers | ||||
|  * @mixer_hw_caps: mixer hardware capabilities | ||||
|  * @pending_flush_mask: storage for pending ctl_flush managed via ops | ||||
|  * @ops: operation list | ||||
|  */ | ||||
| struct dpu_hw_ctl { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* ctl path */ | ||||
| 	int idx; | ||||
| 	const struct dpu_ctl_cfg *caps; | ||||
| 	int mixer_count; | ||||
| 	const struct dpu_lm_cfg *mixer_hw_caps; | ||||
| 	u32 pending_flush_mask; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_ctl_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_ctl - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_ctl *to_dpu_hw_ctl(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_ctl, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_ctl_init(): Initializes the ctl_path hw driver object. | ||||
|  * should be called before accessing every ctl path registers. | ||||
|  * @idx:  ctl_path index for which driver object is required | ||||
|  * @addr: mapped register io address of MDP | ||||
|  * @m :   pointer to mdss catalog data | ||||
|  */ | ||||
| struct dpu_hw_ctl *dpu_hw_ctl_init(enum dpu_ctl idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_ctl_destroy(): Destroys ctl driver context | ||||
|  * should be called to free the context | ||||
|  */ | ||||
| void dpu_hw_ctl_destroy(struct dpu_hw_ctl *ctx); | ||||
| 
 | ||||
| #endif /*_DPU_HW_CTL_H */ | ||||
							
								
								
									
										1183
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_interrupts.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1183
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_interrupts.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										257
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_interrupts.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										257
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_interrupts.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,257 @@ | ||||
| /* Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_INTERRUPTS_H | ||||
| #define _DPU_HW_INTERRUPTS_H | ||||
| 
 | ||||
| #include <linux/types.h> | ||||
| 
 | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| 
 | ||||
| #define IRQ_SOURCE_MDP		BIT(0) | ||||
| #define IRQ_SOURCE_DSI0		BIT(4) | ||||
| #define IRQ_SOURCE_DSI1		BIT(5) | ||||
| #define IRQ_SOURCE_HDMI		BIT(8) | ||||
| #define IRQ_SOURCE_EDP		BIT(12) | ||||
| #define IRQ_SOURCE_MHL		BIT(16) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_intr_type - HW Interrupt Type | ||||
|  * @DPU_IRQ_TYPE_WB_ROT_COMP:		WB rotator done | ||||
|  * @DPU_IRQ_TYPE_WB_WFD_COMP:		WB WFD done | ||||
|  * @DPU_IRQ_TYPE_PING_PONG_COMP:	PingPong done | ||||
|  * @DPU_IRQ_TYPE_PING_PONG_RD_PTR:	PingPong read pointer | ||||
|  * @DPU_IRQ_TYPE_PING_PONG_WR_PTR:	PingPong write pointer | ||||
|  * @DPU_IRQ_TYPE_PING_PONG_AUTO_REF:	PingPong auto refresh | ||||
|  * @DPU_IRQ_TYPE_PING_PONG_TEAR_CHECK:	PingPong Tear check | ||||
|  * @DPU_IRQ_TYPE_PING_PONG_TE_CHECK:	PingPong TE detection | ||||
|  * @DPU_IRQ_TYPE_INTF_UNDER_RUN:	INTF underrun | ||||
|  * @DPU_IRQ_TYPE_INTF_VSYNC:		INTF VSYNC | ||||
|  * @DPU_IRQ_TYPE_CWB_OVERFLOW:		Concurrent WB overflow | ||||
|  * @DPU_IRQ_TYPE_HIST_VIG_DONE:		VIG Histogram done | ||||
|  * @DPU_IRQ_TYPE_HIST_VIG_RSTSEQ:	VIG Histogram reset | ||||
|  * @DPU_IRQ_TYPE_HIST_DSPP_DONE:	DSPP Histogram done | ||||
|  * @DPU_IRQ_TYPE_HIST_DSPP_RSTSEQ:	DSPP Histogram reset | ||||
|  * @DPU_IRQ_TYPE_WD_TIMER:		Watchdog timer | ||||
|  * @DPU_IRQ_TYPE_SFI_VIDEO_IN:		Video static frame INTR into static | ||||
|  * @DPU_IRQ_TYPE_SFI_VIDEO_OUT:		Video static frame INTR out-of static | ||||
|  * @DPU_IRQ_TYPE_SFI_CMD_0_IN:		DSI CMD0 static frame INTR into static | ||||
|  * @DPU_IRQ_TYPE_SFI_CMD_0_OUT:		DSI CMD0 static frame INTR out-of static | ||||
|  * @DPU_IRQ_TYPE_SFI_CMD_1_IN:		DSI CMD1 static frame INTR into static | ||||
|  * @DPU_IRQ_TYPE_SFI_CMD_1_OUT:		DSI CMD1 static frame INTR out-of static | ||||
|  * @DPU_IRQ_TYPE_SFI_CMD_2_IN:		DSI CMD2 static frame INTR into static | ||||
|  * @DPU_IRQ_TYPE_SFI_CMD_2_OUT:		DSI CMD2 static frame INTR out-of static | ||||
|  * @DPU_IRQ_TYPE_PROG_LINE:		Programmable Line interrupt | ||||
|  * @DPU_IRQ_TYPE_AD4_BL_DONE:		AD4 backlight | ||||
|  * @DPU_IRQ_TYPE_CTL_START:		Control start | ||||
|  * @DPU_IRQ_TYPE_RESERVED:		Reserved for expansion | ||||
|  */ | ||||
| enum dpu_intr_type { | ||||
| 	DPU_IRQ_TYPE_WB_ROT_COMP, | ||||
| 	DPU_IRQ_TYPE_WB_WFD_COMP, | ||||
| 	DPU_IRQ_TYPE_PING_PONG_COMP, | ||||
| 	DPU_IRQ_TYPE_PING_PONG_RD_PTR, | ||||
| 	DPU_IRQ_TYPE_PING_PONG_WR_PTR, | ||||
| 	DPU_IRQ_TYPE_PING_PONG_AUTO_REF, | ||||
| 	DPU_IRQ_TYPE_PING_PONG_TEAR_CHECK, | ||||
| 	DPU_IRQ_TYPE_PING_PONG_TE_CHECK, | ||||
| 	DPU_IRQ_TYPE_INTF_UNDER_RUN, | ||||
| 	DPU_IRQ_TYPE_INTF_VSYNC, | ||||
| 	DPU_IRQ_TYPE_CWB_OVERFLOW, | ||||
| 	DPU_IRQ_TYPE_HIST_VIG_DONE, | ||||
| 	DPU_IRQ_TYPE_HIST_VIG_RSTSEQ, | ||||
| 	DPU_IRQ_TYPE_HIST_DSPP_DONE, | ||||
| 	DPU_IRQ_TYPE_HIST_DSPP_RSTSEQ, | ||||
| 	DPU_IRQ_TYPE_WD_TIMER, | ||||
| 	DPU_IRQ_TYPE_SFI_VIDEO_IN, | ||||
| 	DPU_IRQ_TYPE_SFI_VIDEO_OUT, | ||||
| 	DPU_IRQ_TYPE_SFI_CMD_0_IN, | ||||
| 	DPU_IRQ_TYPE_SFI_CMD_0_OUT, | ||||
| 	DPU_IRQ_TYPE_SFI_CMD_1_IN, | ||||
| 	DPU_IRQ_TYPE_SFI_CMD_1_OUT, | ||||
| 	DPU_IRQ_TYPE_SFI_CMD_2_IN, | ||||
| 	DPU_IRQ_TYPE_SFI_CMD_2_OUT, | ||||
| 	DPU_IRQ_TYPE_PROG_LINE, | ||||
| 	DPU_IRQ_TYPE_AD4_BL_DONE, | ||||
| 	DPU_IRQ_TYPE_CTL_START, | ||||
| 	DPU_IRQ_TYPE_RESERVED, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_intr; | ||||
| 
 | ||||
| /**
 | ||||
|  * Interrupt operations. | ||||
|  */ | ||||
| struct dpu_hw_intr_ops { | ||||
| 	/**
 | ||||
| 	 * set_mask - Programs the given interrupt register with the | ||||
| 	 *            given interrupt mask. Register value will get overwritten. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @reg_off:	MDSS HW register offset | ||||
| 	 * @irqmask:	IRQ mask value | ||||
| 	 */ | ||||
| 	void (*set_mask)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			uint32_t reg, | ||||
| 			uint32_t irqmask); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * irq_idx_lookup - Lookup IRQ index on the HW interrupt type | ||||
| 	 *                 Used for all irq related ops | ||||
| 	 * @intr_type:		Interrupt type defined in dpu_intr_type | ||||
| 	 * @instance_idx:	HW interrupt block instance | ||||
| 	 * @return:		irq_idx or -EINVAL for lookup fail | ||||
| 	 */ | ||||
| 	int (*irq_idx_lookup)( | ||||
| 			enum dpu_intr_type intr_type, | ||||
| 			u32 instance_idx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * enable_irq - Enable IRQ based on lookup IRQ index | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @irq_idx:	Lookup irq index return from irq_idx_lookup | ||||
| 	 * @return:	0 for success, otherwise failure | ||||
| 	 */ | ||||
| 	int (*enable_irq)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			int irq_idx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * disable_irq - Disable IRQ based on lookup IRQ index | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @irq_idx:	Lookup irq index return from irq_idx_lookup | ||||
| 	 * @return:	0 for success, otherwise failure | ||||
| 	 */ | ||||
| 	int (*disable_irq)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			int irq_idx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * clear_all_irqs - Clears all the interrupts (i.e. acknowledges | ||||
| 	 *                  any asserted IRQs). Useful during reset. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @return:	0 for success, otherwise failure | ||||
| 	 */ | ||||
| 	int (*clear_all_irqs)( | ||||
| 			struct dpu_hw_intr *intr); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * disable_all_irqs - Disables all the interrupts. Useful during reset. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @return:	0 for success, otherwise failure | ||||
| 	 */ | ||||
| 	int (*disable_all_irqs)( | ||||
| 			struct dpu_hw_intr *intr); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * dispatch_irqs - IRQ dispatcher will call the given callback | ||||
| 	 *                 function when a matching interrupt status bit is | ||||
| 	 *                 found in the irq mapping table. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @cbfunc:	Callback function pointer | ||||
| 	 * @arg:	Argument to pass back during callback | ||||
| 	 */ | ||||
| 	void (*dispatch_irqs)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			void (*cbfunc)(void *arg, int irq_idx), | ||||
| 			void *arg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_interrupt_statuses - Gets and store value from all interrupt | ||||
| 	 *                          status registers that are currently fired. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 */ | ||||
| 	void (*get_interrupt_statuses)( | ||||
| 			struct dpu_hw_intr *intr); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * clear_interrupt_status - Clears HW interrupt status based on given | ||||
| 	 *                          lookup IRQ index. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @irq_idx:	Lookup irq index return from irq_idx_lookup | ||||
| 	 */ | ||||
| 	void (*clear_interrupt_status)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			int irq_idx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * clear_intr_status_nolock() - clears the HW interrupts without lock | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @irq_idx:	Lookup irq index return from irq_idx_lookup | ||||
| 	 */ | ||||
| 	void (*clear_intr_status_nolock)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			int irq_idx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_interrupt_status - Gets HW interrupt status, and clear if set, | ||||
| 	 *                        based on given lookup IRQ index. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @irq_idx:	Lookup irq index return from irq_idx_lookup | ||||
| 	 * @clear:	True to clear irq after read | ||||
| 	 */ | ||||
| 	u32 (*get_interrupt_status)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			int irq_idx, | ||||
| 			bool clear); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_valid_interrupts - Gets a mask of all valid interrupt sources | ||||
| 	 *                        within DPU. These are actually status bits | ||||
| 	 *                        within interrupt registers that specify the | ||||
| 	 *                        source of the interrupt in IRQs. For example, | ||||
| 	 *                        valid interrupt sources can be MDP, DSI, | ||||
| 	 *                        HDMI etc. | ||||
| 	 * @intr:	HW interrupt handle | ||||
| 	 * @mask:	Returning the interrupt source MASK | ||||
| 	 * @return:	0 for success, otherwise failure | ||||
| 	 */ | ||||
| 	int (*get_valid_interrupts)( | ||||
| 			struct dpu_hw_intr *intr, | ||||
| 			uint32_t *mask); | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_intr: hw interrupts handling data structure | ||||
|  * @hw:               virtual address mapping | ||||
|  * @ops:              function pointer mapping for IRQ handling | ||||
|  * @cache_irq_mask:   array of IRQ enable masks reg storage created during init | ||||
|  * @save_irq_status:  array of IRQ status reg storage created during init | ||||
|  * @irq_idx_tbl_size: total number of irq_idx mapped in the hw_interrupts | ||||
|  * @irq_lock:         spinlock for accessing IRQ resources | ||||
|  */ | ||||
| struct dpu_hw_intr { | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 	struct dpu_hw_intr_ops ops; | ||||
| 	u32 *cache_irq_mask; | ||||
| 	u32 *save_irq_status; | ||||
| 	u32 irq_idx_tbl_size; | ||||
| 	spinlock_t irq_lock; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_intr_init(): Initializes the interrupts hw object | ||||
|  * @addr: mapped register io address of MDP | ||||
|  * @m :   pointer to mdss catalog data | ||||
|  */ | ||||
| struct dpu_hw_intr *dpu_hw_intr_init(void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_intr_destroy(): Cleanup interrutps hw object | ||||
|  * @intr: pointer to interrupts hw object | ||||
|  */ | ||||
| void dpu_hw_intr_destroy(struct dpu_hw_intr *intr); | ||||
| #endif | ||||
							
								
								
									
										349
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_intf.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										349
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_intf.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,349 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_intf.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define INTF_TIMING_ENGINE_EN           0x000 | ||||
| #define INTF_CONFIG                     0x004 | ||||
| #define INTF_HSYNC_CTL                  0x008 | ||||
| #define INTF_VSYNC_PERIOD_F0            0x00C | ||||
| #define INTF_VSYNC_PERIOD_F1            0x010 | ||||
| #define INTF_VSYNC_PULSE_WIDTH_F0       0x014 | ||||
| #define INTF_VSYNC_PULSE_WIDTH_F1       0x018 | ||||
| #define INTF_DISPLAY_V_START_F0         0x01C | ||||
| #define INTF_DISPLAY_V_START_F1         0x020 | ||||
| #define INTF_DISPLAY_V_END_F0           0x024 | ||||
| #define INTF_DISPLAY_V_END_F1           0x028 | ||||
| #define INTF_ACTIVE_V_START_F0          0x02C | ||||
| #define INTF_ACTIVE_V_START_F1          0x030 | ||||
| #define INTF_ACTIVE_V_END_F0            0x034 | ||||
| #define INTF_ACTIVE_V_END_F1            0x038 | ||||
| #define INTF_DISPLAY_HCTL               0x03C | ||||
| #define INTF_ACTIVE_HCTL                0x040 | ||||
| #define INTF_BORDER_COLOR               0x044 | ||||
| #define INTF_UNDERFLOW_COLOR            0x048 | ||||
| #define INTF_HSYNC_SKEW                 0x04C | ||||
| #define INTF_POLARITY_CTL               0x050 | ||||
| #define INTF_TEST_CTL                   0x054 | ||||
| #define INTF_TP_COLOR0                  0x058 | ||||
| #define INTF_TP_COLOR1                  0x05C | ||||
| #define INTF_FRAME_LINE_COUNT_EN        0x0A8 | ||||
| #define INTF_FRAME_COUNT                0x0AC | ||||
| #define   INTF_LINE_COUNT               0x0B0 | ||||
| 
 | ||||
| #define   INTF_DEFLICKER_CONFIG         0x0F0 | ||||
| #define   INTF_DEFLICKER_STRNG_COEFF    0x0F4 | ||||
| #define   INTF_DEFLICKER_WEAK_COEFF     0x0F8 | ||||
| 
 | ||||
| #define   INTF_DSI_CMD_MODE_TRIGGER_EN  0x084 | ||||
| #define   INTF_PANEL_FORMAT             0x090 | ||||
| #define   INTF_TPG_ENABLE               0x100 | ||||
| #define   INTF_TPG_MAIN_CONTROL         0x104 | ||||
| #define   INTF_TPG_VIDEO_CONFIG         0x108 | ||||
| #define   INTF_TPG_COMPONENT_LIMITS     0x10C | ||||
| #define   INTF_TPG_RECTANGLE            0x110 | ||||
| #define   INTF_TPG_INITIAL_VALUE        0x114 | ||||
| #define   INTF_TPG_BLK_WHITE_PATTERN_FRAMES   0x118 | ||||
| #define   INTF_TPG_RGB_MAPPING          0x11C | ||||
| #define   INTF_PROG_FETCH_START         0x170 | ||||
| #define   INTF_PROG_ROT_START           0x174 | ||||
| 
 | ||||
| #define   INTF_FRAME_LINE_COUNT_EN      0x0A8 | ||||
| #define   INTF_FRAME_COUNT              0x0AC | ||||
| #define   INTF_LINE_COUNT               0x0B0 | ||||
| 
 | ||||
| #define INTF_MISR_CTRL			0x180 | ||||
| #define INTF_MISR_SIGNATURE		0x184 | ||||
| 
 | ||||
| static struct dpu_intf_cfg *_intf_offset(enum dpu_intf intf, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < m->intf_count; i++) { | ||||
| 		if ((intf == m->intf[i].id) && | ||||
| 		(m->intf[i].type != INTF_NONE)) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->intf[i].base; | ||||
| 			b->length = m->intf[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_INTF; | ||||
| 			return &m->intf[i]; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-EINVAL); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_intf_setup_timing_engine(struct dpu_hw_intf *ctx, | ||||
| 		const struct intf_timing_params *p, | ||||
| 		const struct dpu_format *fmt) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 hsync_period, vsync_period; | ||||
| 	u32 display_v_start, display_v_end; | ||||
| 	u32 hsync_start_x, hsync_end_x; | ||||
| 	u32 active_h_start, active_h_end; | ||||
| 	u32 active_v_start, active_v_end; | ||||
| 	u32 active_hctl, display_hctl, hsync_ctl; | ||||
| 	u32 polarity_ctl, den_polarity, hsync_polarity, vsync_polarity; | ||||
| 	u32 panel_format; | ||||
| 	u32 intf_cfg; | ||||
| 
 | ||||
| 	/* read interface_cfg */ | ||||
| 	intf_cfg = DPU_REG_READ(c, INTF_CONFIG); | ||||
| 	hsync_period = p->hsync_pulse_width + p->h_back_porch + p->width + | ||||
| 	p->h_front_porch; | ||||
| 	vsync_period = p->vsync_pulse_width + p->v_back_porch + p->height + | ||||
| 	p->v_front_porch; | ||||
| 
 | ||||
| 	display_v_start = ((p->vsync_pulse_width + p->v_back_porch) * | ||||
| 	hsync_period) + p->hsync_skew; | ||||
| 	display_v_end = ((vsync_period - p->v_front_porch) * hsync_period) + | ||||
| 	p->hsync_skew - 1; | ||||
| 
 | ||||
| 	if (ctx->cap->type == INTF_EDP || ctx->cap->type == INTF_DP) { | ||||
| 		display_v_start += p->hsync_pulse_width + p->h_back_porch; | ||||
| 		display_v_end -= p->h_front_porch; | ||||
| 	} | ||||
| 
 | ||||
| 	hsync_start_x = p->h_back_porch + p->hsync_pulse_width; | ||||
| 	hsync_end_x = hsync_period - p->h_front_porch - 1; | ||||
| 
 | ||||
| 	if (p->width != p->xres) { | ||||
| 		active_h_start = hsync_start_x; | ||||
| 		active_h_end = active_h_start + p->xres - 1; | ||||
| 	} else { | ||||
| 		active_h_start = 0; | ||||
| 		active_h_end = 0; | ||||
| 	} | ||||
| 
 | ||||
| 	if (p->height != p->yres) { | ||||
| 		active_v_start = display_v_start; | ||||
| 		active_v_end = active_v_start + (p->yres * hsync_period) - 1; | ||||
| 	} else { | ||||
| 		active_v_start = 0; | ||||
| 		active_v_end = 0; | ||||
| 	} | ||||
| 
 | ||||
| 	if (active_h_end) { | ||||
| 		active_hctl = (active_h_end << 16) | active_h_start; | ||||
| 		intf_cfg |= BIT(29);	/* ACTIVE_H_ENABLE */ | ||||
| 	} else { | ||||
| 		active_hctl = 0; | ||||
| 	} | ||||
| 
 | ||||
| 	if (active_v_end) | ||||
| 		intf_cfg |= BIT(30); /* ACTIVE_V_ENABLE */ | ||||
| 
 | ||||
| 	hsync_ctl = (hsync_period << 16) | p->hsync_pulse_width; | ||||
| 	display_hctl = (hsync_end_x << 16) | hsync_start_x; | ||||
| 
 | ||||
| 	den_polarity = 0; | ||||
| 	if (ctx->cap->type == INTF_HDMI) { | ||||
| 		hsync_polarity = p->yres >= 720 ? 0 : 1; | ||||
| 		vsync_polarity = p->yres >= 720 ? 0 : 1; | ||||
| 	} else { | ||||
| 		hsync_polarity = 0; | ||||
| 		vsync_polarity = 0; | ||||
| 	} | ||||
| 	polarity_ctl = (den_polarity << 2) | /*  DEN Polarity  */ | ||||
| 		(vsync_polarity << 1) | /* VSYNC Polarity */ | ||||
| 		(hsync_polarity << 0);  /* HSYNC Polarity */ | ||||
| 
 | ||||
| 	if (!DPU_FORMAT_IS_YUV(fmt)) | ||||
| 		panel_format = (fmt->bits[C0_G_Y] | | ||||
| 				(fmt->bits[C1_B_Cb] << 2) | | ||||
| 				(fmt->bits[C2_R_Cr] << 4) | | ||||
| 				(0x21 << 8)); | ||||
| 	else | ||||
| 		/* Interface treats all the pixel data in RGB888 format */ | ||||
| 		panel_format = (COLOR_8BIT | | ||||
| 				(COLOR_8BIT << 2) | | ||||
| 				(COLOR_8BIT << 4) | | ||||
| 				(0x21 << 8)); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, INTF_HSYNC_CTL, hsync_ctl); | ||||
| 	DPU_REG_WRITE(c, INTF_VSYNC_PERIOD_F0, vsync_period * hsync_period); | ||||
| 	DPU_REG_WRITE(c, INTF_VSYNC_PULSE_WIDTH_F0, | ||||
| 			p->vsync_pulse_width * hsync_period); | ||||
| 	DPU_REG_WRITE(c, INTF_DISPLAY_HCTL, display_hctl); | ||||
| 	DPU_REG_WRITE(c, INTF_DISPLAY_V_START_F0, display_v_start); | ||||
| 	DPU_REG_WRITE(c, INTF_DISPLAY_V_END_F0, display_v_end); | ||||
| 	DPU_REG_WRITE(c, INTF_ACTIVE_HCTL,  active_hctl); | ||||
| 	DPU_REG_WRITE(c, INTF_ACTIVE_V_START_F0, active_v_start); | ||||
| 	DPU_REG_WRITE(c, INTF_ACTIVE_V_END_F0, active_v_end); | ||||
| 	DPU_REG_WRITE(c, INTF_BORDER_COLOR, p->border_clr); | ||||
| 	DPU_REG_WRITE(c, INTF_UNDERFLOW_COLOR, p->underflow_clr); | ||||
| 	DPU_REG_WRITE(c, INTF_HSYNC_SKEW, p->hsync_skew); | ||||
| 	DPU_REG_WRITE(c, INTF_POLARITY_CTL, polarity_ctl); | ||||
| 	DPU_REG_WRITE(c, INTF_FRAME_LINE_COUNT_EN, 0x3); | ||||
| 	DPU_REG_WRITE(c, INTF_CONFIG, intf_cfg); | ||||
| 	DPU_REG_WRITE(c, INTF_PANEL_FORMAT, panel_format); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_intf_enable_timing_engine( | ||||
| 		struct dpu_hw_intf *intf, | ||||
| 		u8 enable) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &intf->hw; | ||||
| 	/* Note: Display interface select is handled in top block hw layer */ | ||||
| 	DPU_REG_WRITE(c, INTF_TIMING_ENGINE_EN, enable != 0); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_intf_setup_prg_fetch( | ||||
| 		struct dpu_hw_intf *intf, | ||||
| 		const struct intf_prog_fetch *fetch) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &intf->hw; | ||||
| 	int fetch_enable; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Fetch should always be outside the active lines. If the fetching | ||||
| 	 * is programmed within active region, hardware behavior is unknown. | ||||
| 	 */ | ||||
| 
 | ||||
| 	fetch_enable = DPU_REG_READ(c, INTF_CONFIG); | ||||
| 	if (fetch->enable) { | ||||
| 		fetch_enable |= BIT(31); | ||||
| 		DPU_REG_WRITE(c, INTF_PROG_FETCH_START, | ||||
| 				fetch->fetch_start); | ||||
| 	} else { | ||||
| 		fetch_enable &= ~BIT(31); | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, INTF_CONFIG, fetch_enable); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_intf_get_status( | ||||
| 		struct dpu_hw_intf *intf, | ||||
| 		struct intf_status *s) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &intf->hw; | ||||
| 
 | ||||
| 	s->is_en = DPU_REG_READ(c, INTF_TIMING_ENGINE_EN); | ||||
| 	if (s->is_en) { | ||||
| 		s->frame_count = DPU_REG_READ(c, INTF_FRAME_COUNT); | ||||
| 		s->line_count = DPU_REG_READ(c, INTF_LINE_COUNT); | ||||
| 	} else { | ||||
| 		s->line_count = 0; | ||||
| 		s->frame_count = 0; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_intf_setup_misr(struct dpu_hw_intf *intf, | ||||
| 						bool enable, u32 frame_count) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &intf->hw; | ||||
| 	u32 config = 0; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, INTF_MISR_CTRL, MISR_CTRL_STATUS_CLEAR); | ||||
| 	/* clear misr data */ | ||||
| 	wmb(); | ||||
| 
 | ||||
| 	if (enable) | ||||
| 		config = (frame_count & MISR_FRAME_COUNT_MASK) | | ||||
| 			MISR_CTRL_ENABLE | INTF_MISR_CTRL_FREE_RUN_MASK; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, INTF_MISR_CTRL, config); | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_intf_collect_misr(struct dpu_hw_intf *intf) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &intf->hw; | ||||
| 
 | ||||
| 	return DPU_REG_READ(c, INTF_MISR_SIGNATURE); | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_intf_get_line_count(struct dpu_hw_intf *intf) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 
 | ||||
| 	if (!intf) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	c = &intf->hw; | ||||
| 
 | ||||
| 	return DPU_REG_READ(c, INTF_LINE_COUNT); | ||||
| } | ||||
| 
 | ||||
| static void _setup_intf_ops(struct dpu_hw_intf_ops *ops, | ||||
| 		unsigned long cap) | ||||
| { | ||||
| 	ops->setup_timing_gen = dpu_hw_intf_setup_timing_engine; | ||||
| 	ops->setup_prg_fetch  = dpu_hw_intf_setup_prg_fetch; | ||||
| 	ops->get_status = dpu_hw_intf_get_status; | ||||
| 	ops->enable_timing = dpu_hw_intf_enable_timing_engine; | ||||
| 	ops->setup_misr = dpu_hw_intf_setup_misr; | ||||
| 	ops->collect_misr = dpu_hw_intf_collect_misr; | ||||
| 	ops->get_line_count = dpu_hw_intf_get_line_count; | ||||
| } | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_intf *dpu_hw_intf_init(enum dpu_intf idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_intf *c; | ||||
| 	struct dpu_intf_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	c = kzalloc(sizeof(*c), GFP_KERNEL); | ||||
| 	if (!c) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _intf_offset(idx, m, addr, &c->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(c); | ||||
| 		pr_err("failed to create dpu_hw_intf %d\n", idx); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Assign ops | ||||
| 	 */ | ||||
| 	c->idx = idx; | ||||
| 	c->cap = cfg; | ||||
| 	c->mdss = m; | ||||
| 	_setup_intf_ops(&c->ops, c->cap->features); | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&c->base, DPU_HW_BLK_INTF, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	return c; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(c); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_intf_destroy(struct dpu_hw_intf *intf) | ||||
| { | ||||
| 	if (intf) | ||||
| 		dpu_hw_blk_destroy(&intf->base); | ||||
| 	kfree(intf); | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										128
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_intf.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_intf.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,128 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_INTF_H | ||||
| #define _DPU_HW_INTF_H | ||||
| 
 | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| struct dpu_hw_intf; | ||||
| 
 | ||||
| /* intf timing settings */ | ||||
| struct intf_timing_params { | ||||
| 	u32 width;		/* active width */ | ||||
| 	u32 height;		/* active height */ | ||||
| 	u32 xres;		/* Display panel width */ | ||||
| 	u32 yres;		/* Display panel height */ | ||||
| 
 | ||||
| 	u32 h_back_porch; | ||||
| 	u32 h_front_porch; | ||||
| 	u32 v_back_porch; | ||||
| 	u32 v_front_porch; | ||||
| 	u32 hsync_pulse_width; | ||||
| 	u32 vsync_pulse_width; | ||||
| 	u32 hsync_polarity; | ||||
| 	u32 vsync_polarity; | ||||
| 	u32 border_clr; | ||||
| 	u32 underflow_clr; | ||||
| 	u32 hsync_skew; | ||||
| }; | ||||
| 
 | ||||
| struct intf_prog_fetch { | ||||
| 	u8 enable; | ||||
| 	/* vsync counter for the front porch pixel line */ | ||||
| 	u32 fetch_start; | ||||
| }; | ||||
| 
 | ||||
| struct intf_status { | ||||
| 	u8 is_en;		/* interface timing engine is enabled or not */ | ||||
| 	u32 frame_count;	/* frame count since timing engine enabled */ | ||||
| 	u32 line_count;		/* current line count including blanking */ | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_intf_ops : Interface to the interface Hw driver functions | ||||
|  *  Assumption is these functions will be called after clocks are enabled | ||||
|  * @ setup_timing_gen : programs the timing engine | ||||
|  * @ setup_prog_fetch : enables/disables the programmable fetch logic | ||||
|  * @ enable_timing: enable/disable timing engine | ||||
|  * @ get_status: returns if timing engine is enabled or not | ||||
|  * @ setup_misr: enables/disables MISR in HW register | ||||
|  * @ collect_misr: reads and stores MISR data from HW register | ||||
|  * @ get_line_count: reads current vertical line counter | ||||
|  */ | ||||
| struct dpu_hw_intf_ops { | ||||
| 	void (*setup_timing_gen)(struct dpu_hw_intf *intf, | ||||
| 			const struct intf_timing_params *p, | ||||
| 			const struct dpu_format *fmt); | ||||
| 
 | ||||
| 	void (*setup_prg_fetch)(struct dpu_hw_intf *intf, | ||||
| 			const struct intf_prog_fetch *fetch); | ||||
| 
 | ||||
| 	void (*enable_timing)(struct dpu_hw_intf *intf, | ||||
| 			u8 enable); | ||||
| 
 | ||||
| 	void (*get_status)(struct dpu_hw_intf *intf, | ||||
| 			struct intf_status *status); | ||||
| 
 | ||||
| 	void (*setup_misr)(struct dpu_hw_intf *intf, | ||||
| 			bool enable, u32 frame_count); | ||||
| 
 | ||||
| 	u32 (*collect_misr)(struct dpu_hw_intf *intf); | ||||
| 
 | ||||
| 	u32 (*get_line_count)(struct dpu_hw_intf *intf); | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_intf { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* intf */ | ||||
| 	enum dpu_intf idx; | ||||
| 	const struct dpu_intf_cfg *cap; | ||||
| 	const struct dpu_mdss_cfg *mdss; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_intf_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * to_dpu_hw_intf - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_intf *to_dpu_hw_intf(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_intf, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_intf_init(): Initializes the intf driver for the passed | ||||
|  * interface idx. | ||||
|  * @idx:  interface index for which driver object is required | ||||
|  * @addr: mapped register io address of MDP | ||||
|  * @m :   pointer to mdss catalog data | ||||
|  */ | ||||
| struct dpu_hw_intf *dpu_hw_intf_init(enum dpu_intf idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_intf_destroy(): Destroys INTF driver context | ||||
|  * @intf:   Pointer to INTF driver context | ||||
|  */ | ||||
| void dpu_hw_intf_destroy(struct dpu_hw_intf *intf); | ||||
| 
 | ||||
| #endif /*_DPU_HW_INTF_H */ | ||||
							
								
								
									
										261
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_lm.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										261
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_lm.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,261 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_lm.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define LM_OP_MODE                        0x00 | ||||
| #define LM_OUT_SIZE                       0x04 | ||||
| #define LM_BORDER_COLOR_0                 0x08 | ||||
| #define LM_BORDER_COLOR_1                 0x010 | ||||
| 
 | ||||
| /* These register are offset to mixer base + stage base */ | ||||
| #define LM_BLEND0_OP                     0x00 | ||||
| #define LM_BLEND0_CONST_ALPHA            0x04 | ||||
| #define LM_FG_COLOR_FILL_COLOR_0         0x08 | ||||
| #define LM_FG_COLOR_FILL_COLOR_1         0x0C | ||||
| #define LM_FG_COLOR_FILL_SIZE            0x10 | ||||
| #define LM_FG_COLOR_FILL_XY              0x14 | ||||
| 
 | ||||
| #define LM_BLEND0_FG_ALPHA               0x04 | ||||
| #define LM_BLEND0_BG_ALPHA               0x08 | ||||
| 
 | ||||
| #define LM_MISR_CTRL			0x310 | ||||
| #define LM_MISR_SIGNATURE		0x314 | ||||
| 
 | ||||
| static struct dpu_lm_cfg *_lm_offset(enum dpu_lm mixer, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < m->mixer_count; i++) { | ||||
| 		if (mixer == m->mixer[i].id) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->mixer[i].base; | ||||
| 			b->length = m->mixer[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_LM; | ||||
| 			return &m->mixer[i]; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-ENOMEM); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * _stage_offset(): returns the relative offset of the blend registers | ||||
|  * for the stage to be setup | ||||
|  * @c:     mixer ctx contains the mixer to be programmed | ||||
|  * @stage: stage index to setup | ||||
|  */ | ||||
| static inline int _stage_offset(struct dpu_hw_mixer *ctx, enum dpu_stage stage) | ||||
| { | ||||
| 	const struct dpu_lm_sub_blks *sblk = ctx->cap->sblk; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (stage == DPU_STAGE_BASE) | ||||
| 		rc = -EINVAL; | ||||
| 	else if (stage <= sblk->maxblendstages) | ||||
| 		rc = sblk->blendstage_base[stage - DPU_STAGE_0]; | ||||
| 	else | ||||
| 		rc = -EINVAL; | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_setup_out(struct dpu_hw_mixer *ctx, | ||||
| 		struct dpu_hw_mixer_cfg *mixer) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 outsize; | ||||
| 	u32 op_mode; | ||||
| 
 | ||||
| 	op_mode = DPU_REG_READ(c, LM_OP_MODE); | ||||
| 
 | ||||
| 	outsize = mixer->out_height << 16 | mixer->out_width; | ||||
| 	DPU_REG_WRITE(c, LM_OUT_SIZE, outsize); | ||||
| 
 | ||||
| 	/* SPLIT_LEFT_RIGHT */ | ||||
| 	if (mixer->right_mixer) | ||||
| 		op_mode |= BIT(31); | ||||
| 	else | ||||
| 		op_mode &= ~BIT(31); | ||||
| 	DPU_REG_WRITE(c, LM_OP_MODE, op_mode); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_setup_border_color(struct dpu_hw_mixer *ctx, | ||||
| 		struct dpu_mdss_color *color, | ||||
| 		u8 border_en) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 
 | ||||
| 	if (border_en) { | ||||
| 		DPU_REG_WRITE(c, LM_BORDER_COLOR_0, | ||||
| 			(color->color_0 & 0xFFF) | | ||||
| 			((color->color_1 & 0xFFF) << 0x10)); | ||||
| 		DPU_REG_WRITE(c, LM_BORDER_COLOR_1, | ||||
| 			(color->color_2 & 0xFFF) | | ||||
| 			((color->color_3 & 0xFFF) << 0x10)); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_setup_blend_config_sdm845(struct dpu_hw_mixer *ctx, | ||||
| 	u32 stage, u32 fg_alpha, u32 bg_alpha, u32 blend_op) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	int stage_off; | ||||
| 	u32 const_alpha; | ||||
| 
 | ||||
| 	if (stage == DPU_STAGE_BASE) | ||||
| 		return; | ||||
| 
 | ||||
| 	stage_off = _stage_offset(ctx, stage); | ||||
| 	if (WARN_ON(stage_off < 0)) | ||||
| 		return; | ||||
| 
 | ||||
| 	const_alpha = (bg_alpha & 0xFF) | ((fg_alpha & 0xFF) << 16); | ||||
| 	DPU_REG_WRITE(c, LM_BLEND0_CONST_ALPHA + stage_off, const_alpha); | ||||
| 	DPU_REG_WRITE(c, LM_BLEND0_OP + stage_off, blend_op); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_setup_blend_config(struct dpu_hw_mixer *ctx, | ||||
| 	u32 stage, u32 fg_alpha, u32 bg_alpha, u32 blend_op) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	int stage_off; | ||||
| 
 | ||||
| 	if (stage == DPU_STAGE_BASE) | ||||
| 		return; | ||||
| 
 | ||||
| 	stage_off = _stage_offset(ctx, stage); | ||||
| 	if (WARN_ON(stage_off < 0)) | ||||
| 		return; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, LM_BLEND0_FG_ALPHA + stage_off, fg_alpha); | ||||
| 	DPU_REG_WRITE(c, LM_BLEND0_BG_ALPHA + stage_off, bg_alpha); | ||||
| 	DPU_REG_WRITE(c, LM_BLEND0_OP + stage_off, blend_op); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_setup_color3(struct dpu_hw_mixer *ctx, | ||||
| 	uint32_t mixer_op_mode) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	int op_mode; | ||||
| 
 | ||||
| 	/* read the existing op_mode configuration */ | ||||
| 	op_mode = DPU_REG_READ(c, LM_OP_MODE); | ||||
| 
 | ||||
| 	op_mode = (op_mode & (BIT(31) | BIT(30))) | mixer_op_mode; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, LM_OP_MODE, op_mode); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_gc(struct dpu_hw_mixer *mixer, | ||||
| 			void *cfg) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_lm_setup_misr(struct dpu_hw_mixer *ctx, | ||||
| 				bool enable, u32 frame_count) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 	u32 config = 0; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, LM_MISR_CTRL, MISR_CTRL_STATUS_CLEAR); | ||||
| 	/* clear misr data */ | ||||
| 	wmb(); | ||||
| 
 | ||||
| 	if (enable) | ||||
| 		config = (frame_count & MISR_FRAME_COUNT_MASK) | | ||||
| 			MISR_CTRL_ENABLE | INTF_MISR_CTRL_FREE_RUN_MASK; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, LM_MISR_CTRL, config); | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_lm_collect_misr(struct dpu_hw_mixer *ctx) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &ctx->hw; | ||||
| 
 | ||||
| 	return DPU_REG_READ(c, LM_MISR_SIGNATURE); | ||||
| } | ||||
| 
 | ||||
| static void _setup_mixer_ops(struct dpu_mdss_cfg *m, | ||||
| 		struct dpu_hw_lm_ops *ops, | ||||
| 		unsigned long features) | ||||
| { | ||||
| 	ops->setup_mixer_out = dpu_hw_lm_setup_out; | ||||
| 	if (IS_SDM845_TARGET(m->hwversion) || IS_SDM670_TARGET(m->hwversion)) | ||||
| 		ops->setup_blend_config = dpu_hw_lm_setup_blend_config_sdm845; | ||||
| 	else | ||||
| 		ops->setup_blend_config = dpu_hw_lm_setup_blend_config; | ||||
| 	ops->setup_alpha_out = dpu_hw_lm_setup_color3; | ||||
| 	ops->setup_border_color = dpu_hw_lm_setup_border_color; | ||||
| 	ops->setup_gc = dpu_hw_lm_gc; | ||||
| 	ops->setup_misr = dpu_hw_lm_setup_misr; | ||||
| 	ops->collect_misr = dpu_hw_lm_collect_misr; | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_mixer *dpu_hw_lm_init(enum dpu_lm idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_mixer *c; | ||||
| 	struct dpu_lm_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	c = kzalloc(sizeof(*c), GFP_KERNEL); | ||||
| 	if (!c) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _lm_offset(idx, m, addr, &c->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(c); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	/* Assign ops */ | ||||
| 	c->idx = idx; | ||||
| 	c->cap = cfg; | ||||
| 	_setup_mixer_ops(m, &c->ops, c->cap->features); | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&c->base, DPU_HW_BLK_LM, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	return c; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(c); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_lm_destroy(struct dpu_hw_mixer *lm) | ||||
| { | ||||
| 	if (lm) | ||||
| 		dpu_hw_blk_destroy(&lm->base); | ||||
| 	kfree(lm); | ||||
| } | ||||
							
								
								
									
										122
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_lm.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_lm.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,122 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_LM_H | ||||
| #define _DPU_HW_LM_H | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| struct dpu_hw_mixer; | ||||
| 
 | ||||
| struct dpu_hw_mixer_cfg { | ||||
| 	u32 out_width; | ||||
| 	u32 out_height; | ||||
| 	bool right_mixer; | ||||
| 	int flags; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_color3_cfg { | ||||
| 	u8 keep_fg[DPU_STAGE_MAX]; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * | ||||
|  * struct dpu_hw_lm_ops : Interface to the mixer Hw driver functions | ||||
|  *  Assumption is these functions will be called after clocks are enabled | ||||
|  */ | ||||
| struct dpu_hw_lm_ops { | ||||
| 	/*
 | ||||
| 	 * Sets up mixer output width and height | ||||
| 	 * and border color if enabled | ||||
| 	 */ | ||||
| 	void (*setup_mixer_out)(struct dpu_hw_mixer *ctx, | ||||
| 		struct dpu_hw_mixer_cfg *cfg); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Alpha blending configuration | ||||
| 	 * for the specified stage | ||||
| 	 */ | ||||
| 	void (*setup_blend_config)(struct dpu_hw_mixer *ctx, uint32_t stage, | ||||
| 		uint32_t fg_alpha, uint32_t bg_alpha, uint32_t blend_op); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Alpha color component selection from either fg or bg | ||||
| 	 */ | ||||
| 	void (*setup_alpha_out)(struct dpu_hw_mixer *ctx, uint32_t mixer_op); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_border_color : enable/disable border color | ||||
| 	 */ | ||||
| 	void (*setup_border_color)(struct dpu_hw_mixer *ctx, | ||||
| 		struct dpu_mdss_color *color, | ||||
| 		u8 border_en); | ||||
| 	/**
 | ||||
| 	 * setup_gc : enable/disable gamma correction feature | ||||
| 	 */ | ||||
| 	void (*setup_gc)(struct dpu_hw_mixer *mixer, | ||||
| 			void *cfg); | ||||
| 
 | ||||
| 	/* setup_misr: enables/disables MISR in HW register */ | ||||
| 	void (*setup_misr)(struct dpu_hw_mixer *ctx, | ||||
| 			bool enable, u32 frame_count); | ||||
| 
 | ||||
| 	/* collect_misr: reads and stores MISR data from HW register */ | ||||
| 	u32 (*collect_misr)(struct dpu_hw_mixer *ctx); | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_mixer { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* lm */ | ||||
| 	enum dpu_lm  idx; | ||||
| 	const struct dpu_lm_cfg   *cap; | ||||
| 	const struct dpu_mdp_cfg  *mdp; | ||||
| 	const struct dpu_ctl_cfg  *ctl; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_lm_ops ops; | ||||
| 
 | ||||
| 	/* store mixer info specific to display */ | ||||
| 	struct dpu_hw_mixer_cfg cfg; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * to_dpu_hw_mixer - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_mixer *to_dpu_hw_mixer(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_mixer, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_lm_init(): Initializes the mixer hw driver object. | ||||
|  * should be called once before accessing every mixer. | ||||
|  * @idx:  mixer index for which driver object is required | ||||
|  * @addr: mapped register io address of MDP | ||||
|  * @m :   pointer to mdss catalog data | ||||
|  */ | ||||
| struct dpu_hw_mixer *dpu_hw_lm_init(enum dpu_lm idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_lm_destroy(): Destroys layer mixer driver context | ||||
|  * @lm:   Pointer to LM driver context | ||||
|  */ | ||||
| void dpu_hw_lm_destroy(struct dpu_hw_mixer *lm); | ||||
| 
 | ||||
| #endif /*_DPU_HW_LM_H */ | ||||
							
								
								
									
										465
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_mdss.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										465
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_mdss.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,465 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_MDSS_H | ||||
| #define _DPU_HW_MDSS_H | ||||
| 
 | ||||
| #include <linux/kernel.h> | ||||
| #include <linux/err.h> | ||||
| 
 | ||||
| #include "msm_drv.h" | ||||
| 
 | ||||
| #define DPU_DBG_NAME			"dpu" | ||||
| 
 | ||||
| #define DPU_NONE                        0 | ||||
| 
 | ||||
| #ifndef DPU_CSC_MATRIX_COEFF_SIZE | ||||
| #define DPU_CSC_MATRIX_COEFF_SIZE	9 | ||||
| #endif | ||||
| 
 | ||||
| #ifndef DPU_CSC_CLAMP_SIZE | ||||
| #define DPU_CSC_CLAMP_SIZE		6 | ||||
| #endif | ||||
| 
 | ||||
| #ifndef DPU_CSC_BIAS_SIZE | ||||
| #define DPU_CSC_BIAS_SIZE		3 | ||||
| #endif | ||||
| 
 | ||||
| #ifndef DPU_MAX_PLANES | ||||
| #define DPU_MAX_PLANES			4 | ||||
| #endif | ||||
| 
 | ||||
| #define PIPES_PER_STAGE			2 | ||||
| #ifndef DPU_MAX_DE_CURVES | ||||
| #define DPU_MAX_DE_CURVES		3 | ||||
| #endif | ||||
| 
 | ||||
| enum dpu_format_flags { | ||||
| 	DPU_FORMAT_FLAG_YUV_BIT, | ||||
| 	DPU_FORMAT_FLAG_DX_BIT, | ||||
| 	DPU_FORMAT_FLAG_COMPRESSED_BIT, | ||||
| 	DPU_FORMAT_FLAG_BIT_MAX, | ||||
| }; | ||||
| 
 | ||||
| #define DPU_FORMAT_FLAG_YUV		BIT(DPU_FORMAT_FLAG_YUV_BIT) | ||||
| #define DPU_FORMAT_FLAG_DX		BIT(DPU_FORMAT_FLAG_DX_BIT) | ||||
| #define DPU_FORMAT_FLAG_COMPRESSED	BIT(DPU_FORMAT_FLAG_COMPRESSED_BIT) | ||||
| #define DPU_FORMAT_IS_YUV(X)		\ | ||||
| 	(test_bit(DPU_FORMAT_FLAG_YUV_BIT, (X)->flag)) | ||||
| #define DPU_FORMAT_IS_DX(X)		\ | ||||
| 	(test_bit(DPU_FORMAT_FLAG_DX_BIT, (X)->flag)) | ||||
| #define DPU_FORMAT_IS_LINEAR(X)		((X)->fetch_mode == DPU_FETCH_LINEAR) | ||||
| #define DPU_FORMAT_IS_TILE(X) \ | ||||
| 	(((X)->fetch_mode == DPU_FETCH_UBWC) && \ | ||||
| 			!test_bit(DPU_FORMAT_FLAG_COMPRESSED_BIT, (X)->flag)) | ||||
| #define DPU_FORMAT_IS_UBWC(X) \ | ||||
| 	(((X)->fetch_mode == DPU_FETCH_UBWC) && \ | ||||
| 			test_bit(DPU_FORMAT_FLAG_COMPRESSED_BIT, (X)->flag)) | ||||
| 
 | ||||
| #define DPU_BLEND_FG_ALPHA_FG_CONST	(0 << 0) | ||||
| #define DPU_BLEND_FG_ALPHA_BG_CONST	(1 << 0) | ||||
| #define DPU_BLEND_FG_ALPHA_FG_PIXEL	(2 << 0) | ||||
| #define DPU_BLEND_FG_ALPHA_BG_PIXEL	(3 << 0) | ||||
| #define DPU_BLEND_FG_INV_ALPHA		(1 << 2) | ||||
| #define DPU_BLEND_FG_MOD_ALPHA		(1 << 3) | ||||
| #define DPU_BLEND_FG_INV_MOD_ALPHA	(1 << 4) | ||||
| #define DPU_BLEND_FG_TRANSP_EN		(1 << 5) | ||||
| #define DPU_BLEND_BG_ALPHA_FG_CONST	(0 << 8) | ||||
| #define DPU_BLEND_BG_ALPHA_BG_CONST	(1 << 8) | ||||
| #define DPU_BLEND_BG_ALPHA_FG_PIXEL	(2 << 8) | ||||
| #define DPU_BLEND_BG_ALPHA_BG_PIXEL	(3 << 8) | ||||
| #define DPU_BLEND_BG_INV_ALPHA		(1 << 10) | ||||
| #define DPU_BLEND_BG_MOD_ALPHA		(1 << 11) | ||||
| #define DPU_BLEND_BG_INV_MOD_ALPHA	(1 << 12) | ||||
| #define DPU_BLEND_BG_TRANSP_EN		(1 << 13) | ||||
| 
 | ||||
| #define DPU_VSYNC0_SOURCE_GPIO		0 | ||||
| #define DPU_VSYNC1_SOURCE_GPIO		1 | ||||
| #define DPU_VSYNC2_SOURCE_GPIO		2 | ||||
| #define DPU_VSYNC_SOURCE_INTF_0		3 | ||||
| #define DPU_VSYNC_SOURCE_INTF_1		4 | ||||
| #define DPU_VSYNC_SOURCE_INTF_2		5 | ||||
| #define DPU_VSYNC_SOURCE_INTF_3		6 | ||||
| #define DPU_VSYNC_SOURCE_WD_TIMER_4	11 | ||||
| #define DPU_VSYNC_SOURCE_WD_TIMER_3	12 | ||||
| #define DPU_VSYNC_SOURCE_WD_TIMER_2	13 | ||||
| #define DPU_VSYNC_SOURCE_WD_TIMER_1	14 | ||||
| #define DPU_VSYNC_SOURCE_WD_TIMER_0	15 | ||||
| 
 | ||||
| enum dpu_hw_blk_type { | ||||
| 	DPU_HW_BLK_TOP = 0, | ||||
| 	DPU_HW_BLK_SSPP, | ||||
| 	DPU_HW_BLK_LM, | ||||
| 	DPU_HW_BLK_CTL, | ||||
| 	DPU_HW_BLK_CDM, | ||||
| 	DPU_HW_BLK_PINGPONG, | ||||
| 	DPU_HW_BLK_INTF, | ||||
| 	DPU_HW_BLK_WB, | ||||
| 	DPU_HW_BLK_MAX, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_mdp { | ||||
| 	MDP_TOP = 0x1, | ||||
| 	MDP_MAX, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_sspp { | ||||
| 	SSPP_NONE, | ||||
| 	SSPP_VIG0, | ||||
| 	SSPP_VIG1, | ||||
| 	SSPP_VIG2, | ||||
| 	SSPP_VIG3, | ||||
| 	SSPP_RGB0, | ||||
| 	SSPP_RGB1, | ||||
| 	SSPP_RGB2, | ||||
| 	SSPP_RGB3, | ||||
| 	SSPP_DMA0, | ||||
| 	SSPP_DMA1, | ||||
| 	SSPP_DMA2, | ||||
| 	SSPP_DMA3, | ||||
| 	SSPP_CURSOR0, | ||||
| 	SSPP_CURSOR1, | ||||
| 	SSPP_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_sspp_type { | ||||
| 	SSPP_TYPE_VIG, | ||||
| 	SSPP_TYPE_RGB, | ||||
| 	SSPP_TYPE_DMA, | ||||
| 	SSPP_TYPE_CURSOR, | ||||
| 	SSPP_TYPE_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_lm { | ||||
| 	LM_0 = 1, | ||||
| 	LM_1, | ||||
| 	LM_2, | ||||
| 	LM_3, | ||||
| 	LM_4, | ||||
| 	LM_5, | ||||
| 	LM_6, | ||||
| 	LM_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_stage { | ||||
| 	DPU_STAGE_BASE = 0, | ||||
| 	DPU_STAGE_0, | ||||
| 	DPU_STAGE_1, | ||||
| 	DPU_STAGE_2, | ||||
| 	DPU_STAGE_3, | ||||
| 	DPU_STAGE_4, | ||||
| 	DPU_STAGE_5, | ||||
| 	DPU_STAGE_6, | ||||
| 	DPU_STAGE_7, | ||||
| 	DPU_STAGE_8, | ||||
| 	DPU_STAGE_9, | ||||
| 	DPU_STAGE_10, | ||||
| 	DPU_STAGE_MAX | ||||
| }; | ||||
| enum dpu_dspp { | ||||
| 	DSPP_0 = 1, | ||||
| 	DSPP_1, | ||||
| 	DSPP_2, | ||||
| 	DSPP_3, | ||||
| 	DSPP_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_ds { | ||||
| 	DS_TOP, | ||||
| 	DS_0, | ||||
| 	DS_1, | ||||
| 	DS_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_ctl { | ||||
| 	CTL_0 = 1, | ||||
| 	CTL_1, | ||||
| 	CTL_2, | ||||
| 	CTL_3, | ||||
| 	CTL_4, | ||||
| 	CTL_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_cdm { | ||||
| 	CDM_0 = 1, | ||||
| 	CDM_1, | ||||
| 	CDM_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_pingpong { | ||||
| 	PINGPONG_0 = 1, | ||||
| 	PINGPONG_1, | ||||
| 	PINGPONG_2, | ||||
| 	PINGPONG_3, | ||||
| 	PINGPONG_4, | ||||
| 	PINGPONG_S0, | ||||
| 	PINGPONG_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_intf { | ||||
| 	INTF_0 = 1, | ||||
| 	INTF_1, | ||||
| 	INTF_2, | ||||
| 	INTF_3, | ||||
| 	INTF_4, | ||||
| 	INTF_5, | ||||
| 	INTF_6, | ||||
| 	INTF_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_intf_type { | ||||
| 	INTF_NONE = 0x0, | ||||
| 	INTF_DSI = 0x1, | ||||
| 	INTF_HDMI = 0x3, | ||||
| 	INTF_LCDC = 0x5, | ||||
| 	INTF_EDP = 0x9, | ||||
| 	INTF_DP = 0xa, | ||||
| 	INTF_TYPE_MAX, | ||||
| 
 | ||||
| 	/* virtual interfaces */ | ||||
| 	INTF_WB = 0x100, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_intf_mode { | ||||
| 	INTF_MODE_NONE = 0, | ||||
| 	INTF_MODE_CMD, | ||||
| 	INTF_MODE_VIDEO, | ||||
| 	INTF_MODE_WB_BLOCK, | ||||
| 	INTF_MODE_WB_LINE, | ||||
| 	INTF_MODE_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_wb { | ||||
| 	WB_0 = 1, | ||||
| 	WB_1, | ||||
| 	WB_2, | ||||
| 	WB_3, | ||||
| 	WB_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_ad { | ||||
| 	AD_0 = 0x1, | ||||
| 	AD_1, | ||||
| 	AD_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_cwb { | ||||
| 	CWB_0 = 0x1, | ||||
| 	CWB_1, | ||||
| 	CWB_2, | ||||
| 	CWB_3, | ||||
| 	CWB_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_wd_timer { | ||||
| 	WD_TIMER_0 = 0x1, | ||||
| 	WD_TIMER_1, | ||||
| 	WD_TIMER_2, | ||||
| 	WD_TIMER_3, | ||||
| 	WD_TIMER_4, | ||||
| 	WD_TIMER_5, | ||||
| 	WD_TIMER_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_vbif { | ||||
| 	VBIF_0, | ||||
| 	VBIF_1, | ||||
| 	VBIF_MAX, | ||||
| 	VBIF_RT = VBIF_0, | ||||
| 	VBIF_NRT = VBIF_1 | ||||
| }; | ||||
| 
 | ||||
| enum dpu_iommu_domain { | ||||
| 	DPU_IOMMU_DOMAIN_UNSECURE, | ||||
| 	DPU_IOMMU_DOMAIN_SECURE, | ||||
| 	DPU_IOMMU_DOMAIN_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU HW,Component order color map | ||||
|  */ | ||||
| enum { | ||||
| 	C0_G_Y = 0, | ||||
| 	C1_B_Cb = 1, | ||||
| 	C2_R_Cr = 2, | ||||
| 	C3_ALPHA = 3 | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_plane_type - defines how the color component pixel packing | ||||
|  * @DPU_PLANE_INTERLEAVED   : Color components in single plane | ||||
|  * @DPU_PLANE_PLANAR        : Color component in separate planes | ||||
|  * @DPU_PLANE_PSEUDO_PLANAR : Chroma components interleaved in separate plane | ||||
|  */ | ||||
| enum dpu_plane_type { | ||||
| 	DPU_PLANE_INTERLEAVED, | ||||
| 	DPU_PLANE_PLANAR, | ||||
| 	DPU_PLANE_PSEUDO_PLANAR, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_chroma_samp_type - chroma sub-samplng type | ||||
|  * @DPU_CHROMA_RGB   : No chroma subsampling | ||||
|  * @DPU_CHROMA_H2V1  : Chroma pixels are horizontally subsampled | ||||
|  * @DPU_CHROMA_H1V2  : Chroma pixels are vertically subsampled | ||||
|  * @DPU_CHROMA_420   : 420 subsampling | ||||
|  */ | ||||
| enum dpu_chroma_samp_type { | ||||
| 	DPU_CHROMA_RGB, | ||||
| 	DPU_CHROMA_H2V1, | ||||
| 	DPU_CHROMA_H1V2, | ||||
| 	DPU_CHROMA_420 | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_fetch_type - Defines How DPU HW fetches data | ||||
|  * @DPU_FETCH_LINEAR   : fetch is line by line | ||||
|  * @DPU_FETCH_TILE     : fetches data in Z order from a tile | ||||
|  * @DPU_FETCH_UBWC     : fetch and decompress data | ||||
|  */ | ||||
| enum dpu_fetch_type { | ||||
| 	DPU_FETCH_LINEAR, | ||||
| 	DPU_FETCH_TILE, | ||||
| 	DPU_FETCH_UBWC | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * Value of enum chosen to fit the number of bits | ||||
|  * expected by the HW programming. | ||||
|  */ | ||||
| enum { | ||||
| 	COLOR_ALPHA_1BIT = 0, | ||||
| 	COLOR_ALPHA_4BIT = 1, | ||||
| 	COLOR_4BIT = 0, | ||||
| 	COLOR_5BIT = 1, /* No 5-bit Alpha */ | ||||
| 	COLOR_6BIT = 2, /* 6-Bit Alpha also = 2 */ | ||||
| 	COLOR_8BIT = 3, /* 8-Bit Alpha also = 3 */ | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_3d_blend_mode | ||||
|  * Desribes how the 3d data is blended | ||||
|  * @BLEND_3D_NONE      : 3d blending not enabled | ||||
|  * @BLEND_3D_FRAME_INT : Frame interleaving | ||||
|  * @BLEND_3D_H_ROW_INT : Horizontal row interleaving | ||||
|  * @BLEND_3D_V_ROW_INT : vertical row interleaving | ||||
|  * @BLEND_3D_COL_INT   : column interleaving | ||||
|  * @BLEND_3D_MAX       : | ||||
|  */ | ||||
| enum dpu_3d_blend_mode { | ||||
| 	BLEND_3D_NONE = 0, | ||||
| 	BLEND_3D_FRAME_INT, | ||||
| 	BLEND_3D_H_ROW_INT, | ||||
| 	BLEND_3D_V_ROW_INT, | ||||
| 	BLEND_3D_COL_INT, | ||||
| 	BLEND_3D_MAX | ||||
| }; | ||||
| 
 | ||||
| /** struct dpu_format - defines the format configuration which
 | ||||
|  * allows DPU HW to correctly fetch and decode the format | ||||
|  * @base: base msm_format struture containing fourcc code | ||||
|  * @fetch_planes: how the color components are packed in pixel format | ||||
|  * @element: element color ordering | ||||
|  * @bits: element bit widths | ||||
|  * @chroma_sample: chroma sub-samplng type | ||||
|  * @unpack_align_msb: unpack aligned, 0 to LSB, 1 to MSB | ||||
|  * @unpack_tight: 0 for loose, 1 for tight | ||||
|  * @unpack_count: 0 = 1 component, 1 = 2 component | ||||
|  * @bpp: bytes per pixel | ||||
|  * @alpha_enable: whether the format has an alpha channel | ||||
|  * @num_planes: number of planes (including meta data planes) | ||||
|  * @fetch_mode: linear, tiled, or ubwc hw fetch behavior | ||||
|  * @is_yuv: is format a yuv variant | ||||
|  * @flag: usage bit flags | ||||
|  * @tile_width: format tile width | ||||
|  * @tile_height: format tile height | ||||
|  */ | ||||
| struct dpu_format { | ||||
| 	struct msm_format base; | ||||
| 	enum dpu_plane_type fetch_planes; | ||||
| 	u8 element[DPU_MAX_PLANES]; | ||||
| 	u8 bits[DPU_MAX_PLANES]; | ||||
| 	enum dpu_chroma_samp_type chroma_sample; | ||||
| 	u8 unpack_align_msb; | ||||
| 	u8 unpack_tight; | ||||
| 	u8 unpack_count; | ||||
| 	u8 bpp; | ||||
| 	u8 alpha_enable; | ||||
| 	u8 num_planes; | ||||
| 	enum dpu_fetch_type fetch_mode; | ||||
| 	DECLARE_BITMAP(flag, DPU_FORMAT_FLAG_BIT_MAX); | ||||
| 	u16 tile_width; | ||||
| 	u16 tile_height; | ||||
| }; | ||||
| #define to_dpu_format(x) container_of(x, struct dpu_format, base) | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_fmt_layout - format information of the source pixel data | ||||
|  * @format: pixel format parameters | ||||
|  * @num_planes: number of planes (including meta data planes) | ||||
|  * @width: image width | ||||
|  * @height: image height | ||||
|  * @total_size: total size in bytes | ||||
|  * @plane_addr: address of each plane | ||||
|  * @plane_size: length of each plane | ||||
|  * @plane_pitch: pitch of each plane | ||||
|  */ | ||||
| struct dpu_hw_fmt_layout { | ||||
| 	const struct dpu_format *format; | ||||
| 	uint32_t num_planes; | ||||
| 	uint32_t width; | ||||
| 	uint32_t height; | ||||
| 	uint32_t total_size; | ||||
| 	uint32_t plane_addr[DPU_MAX_PLANES]; | ||||
| 	uint32_t plane_size[DPU_MAX_PLANES]; | ||||
| 	uint32_t plane_pitch[DPU_MAX_PLANES]; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_csc_cfg { | ||||
| 	/* matrix coefficients in S15.16 format */ | ||||
| 	uint32_t csc_mv[DPU_CSC_MATRIX_COEFF_SIZE]; | ||||
| 	uint32_t csc_pre_bv[DPU_CSC_BIAS_SIZE]; | ||||
| 	uint32_t csc_post_bv[DPU_CSC_BIAS_SIZE]; | ||||
| 	uint32_t csc_pre_lv[DPU_CSC_CLAMP_SIZE]; | ||||
| 	uint32_t csc_post_lv[DPU_CSC_CLAMP_SIZE]; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_mdss_color - mdss color description | ||||
|  * color 0 : green | ||||
|  * color 1 : blue | ||||
|  * color 2 : red | ||||
|  * color 3 : alpha | ||||
|  */ | ||||
| struct dpu_mdss_color { | ||||
| 	u32 color_0; | ||||
| 	u32 color_1; | ||||
| 	u32 color_2; | ||||
| 	u32 color_3; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * Define bit masks for h/w logging. | ||||
|  */ | ||||
| #define DPU_DBG_MASK_NONE     (1 << 0) | ||||
| #define DPU_DBG_MASK_CDM      (1 << 1) | ||||
| #define DPU_DBG_MASK_INTF     (1 << 2) | ||||
| #define DPU_DBG_MASK_LM       (1 << 3) | ||||
| #define DPU_DBG_MASK_CTL      (1 << 4) | ||||
| #define DPU_DBG_MASK_PINGPONG (1 << 5) | ||||
| #define DPU_DBG_MASK_SSPP     (1 << 6) | ||||
| #define DPU_DBG_MASK_WB       (1 << 7) | ||||
| #define DPU_DBG_MASK_TOP      (1 << 8) | ||||
| #define DPU_DBG_MASK_VBIF     (1 << 9) | ||||
| #define DPU_DBG_MASK_ROT      (1 << 10) | ||||
| 
 | ||||
| #endif  /* _DPU_HW_MDSS_H */ | ||||
							
								
								
									
										250
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_pingpong.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_pingpong.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,250 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/iopoll.h> | ||||
| 
 | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_pingpong.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_trace.h" | ||||
| 
 | ||||
| #define PP_TEAR_CHECK_EN                0x000 | ||||
| #define PP_SYNC_CONFIG_VSYNC            0x004 | ||||
| #define PP_SYNC_CONFIG_HEIGHT           0x008 | ||||
| #define PP_SYNC_WRCOUNT                 0x00C | ||||
| #define PP_VSYNC_INIT_VAL               0x010 | ||||
| #define PP_INT_COUNT_VAL                0x014 | ||||
| #define PP_SYNC_THRESH                  0x018 | ||||
| #define PP_START_POS                    0x01C | ||||
| #define PP_RD_PTR_IRQ                   0x020 | ||||
| #define PP_WR_PTR_IRQ                   0x024 | ||||
| #define PP_OUT_LINE_COUNT               0x028 | ||||
| #define PP_LINE_COUNT                   0x02C | ||||
| 
 | ||||
| #define PP_FBC_MODE                     0x034 | ||||
| #define PP_FBC_BUDGET_CTL               0x038 | ||||
| #define PP_FBC_LOSSY_MODE               0x03C | ||||
| 
 | ||||
| static struct dpu_pingpong_cfg *_pingpong_offset(enum dpu_pingpong pp, | ||||
| 		struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < m->pingpong_count; i++) { | ||||
| 		if (pp == m->pingpong[i].id) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->pingpong[i].base; | ||||
| 			b->length = m->pingpong[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_PINGPONG; | ||||
| 			return &m->pingpong[i]; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-EINVAL); | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_pp_setup_te_config(struct dpu_hw_pingpong *pp, | ||||
| 		struct dpu_hw_tear_check *te) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	int cfg; | ||||
| 
 | ||||
| 	if (!pp || !te) | ||||
| 		return -EINVAL; | ||||
| 	c = &pp->hw; | ||||
| 
 | ||||
| 	cfg = BIT(19); /*VSYNC_COUNTER_EN */ | ||||
| 	if (te->hw_vsync_mode) | ||||
| 		cfg |= BIT(20); | ||||
| 
 | ||||
| 	cfg |= te->vsync_count; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, PP_SYNC_CONFIG_VSYNC, cfg); | ||||
| 	DPU_REG_WRITE(c, PP_SYNC_CONFIG_HEIGHT, te->sync_cfg_height); | ||||
| 	DPU_REG_WRITE(c, PP_VSYNC_INIT_VAL, te->vsync_init_val); | ||||
| 	DPU_REG_WRITE(c, PP_RD_PTR_IRQ, te->rd_ptr_irq); | ||||
| 	DPU_REG_WRITE(c, PP_START_POS, te->start_pos); | ||||
| 	DPU_REG_WRITE(c, PP_SYNC_THRESH, | ||||
| 			((te->sync_threshold_continue << 16) | | ||||
| 			 te->sync_threshold_start)); | ||||
| 	DPU_REG_WRITE(c, PP_SYNC_WRCOUNT, | ||||
| 			(te->start_pos + te->sync_threshold_start + 1)); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_pp_poll_timeout_wr_ptr(struct dpu_hw_pingpong *pp, | ||||
| 		u32 timeout_us) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 val; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (!pp) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	c = &pp->hw; | ||||
| 	rc = readl_poll_timeout(c->base_off + c->blk_off + PP_LINE_COUNT, | ||||
| 			val, (val & 0xffff) >= 1, 10, timeout_us); | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_pp_enable_te(struct dpu_hw_pingpong *pp, bool enable) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 
 | ||||
| 	if (!pp) | ||||
| 		return -EINVAL; | ||||
| 	c = &pp->hw; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, PP_TEAR_CHECK_EN, enable); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_pp_connect_external_te(struct dpu_hw_pingpong *pp, | ||||
| 		bool enable_external_te) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &pp->hw; | ||||
| 	u32 cfg; | ||||
| 	int orig; | ||||
| 
 | ||||
| 	if (!pp) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	c = &pp->hw; | ||||
| 	cfg = DPU_REG_READ(c, PP_SYNC_CONFIG_VSYNC); | ||||
| 	orig = (bool)(cfg & BIT(20)); | ||||
| 	if (enable_external_te) | ||||
| 		cfg |= BIT(20); | ||||
| 	else | ||||
| 		cfg &= ~BIT(20); | ||||
| 	DPU_REG_WRITE(c, PP_SYNC_CONFIG_VSYNC, cfg); | ||||
| 	trace_dpu_pp_connect_ext_te(pp->idx - PINGPONG_0, cfg); | ||||
| 
 | ||||
| 	return orig; | ||||
| } | ||||
| 
 | ||||
| static int dpu_hw_pp_get_vsync_info(struct dpu_hw_pingpong *pp, | ||||
| 		struct dpu_hw_pp_vsync_info *info) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 val; | ||||
| 
 | ||||
| 	if (!pp || !info) | ||||
| 		return -EINVAL; | ||||
| 	c = &pp->hw; | ||||
| 
 | ||||
| 	val = DPU_REG_READ(c, PP_VSYNC_INIT_VAL); | ||||
| 	info->rd_ptr_init_val = val & 0xffff; | ||||
| 
 | ||||
| 	val = DPU_REG_READ(c, PP_INT_COUNT_VAL); | ||||
| 	info->rd_ptr_frame_count = (val & 0xffff0000) >> 16; | ||||
| 	info->rd_ptr_line_count = val & 0xffff; | ||||
| 
 | ||||
| 	val = DPU_REG_READ(c, PP_LINE_COUNT); | ||||
| 	info->wr_ptr_line_count = val & 0xffff; | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_pp_get_line_count(struct dpu_hw_pingpong *pp) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &pp->hw; | ||||
| 	u32 height, init; | ||||
| 	u32 line = 0xFFFF; | ||||
| 
 | ||||
| 	if (!pp) | ||||
| 		return 0; | ||||
| 	c = &pp->hw; | ||||
| 
 | ||||
| 	init = DPU_REG_READ(c, PP_VSYNC_INIT_VAL) & 0xFFFF; | ||||
| 	height = DPU_REG_READ(c, PP_SYNC_CONFIG_HEIGHT) & 0xFFFF; | ||||
| 
 | ||||
| 	if (height < init) | ||||
| 		goto line_count_exit; | ||||
| 
 | ||||
| 	line = DPU_REG_READ(c, PP_INT_COUNT_VAL) & 0xFFFF; | ||||
| 
 | ||||
| 	if (line < init) | ||||
| 		line += (0xFFFF - init); | ||||
| 	else | ||||
| 		line -= init; | ||||
| 
 | ||||
| line_count_exit: | ||||
| 	return line; | ||||
| } | ||||
| 
 | ||||
| static void _setup_pingpong_ops(struct dpu_hw_pingpong_ops *ops, | ||||
| 	const struct dpu_pingpong_cfg *hw_cap) | ||||
| { | ||||
| 	ops->setup_tearcheck = dpu_hw_pp_setup_te_config; | ||||
| 	ops->enable_tearcheck = dpu_hw_pp_enable_te; | ||||
| 	ops->connect_external_te = dpu_hw_pp_connect_external_te; | ||||
| 	ops->get_vsync_info = dpu_hw_pp_get_vsync_info; | ||||
| 	ops->poll_timeout_wr_ptr = dpu_hw_pp_poll_timeout_wr_ptr; | ||||
| 	ops->get_line_count = dpu_hw_pp_get_line_count; | ||||
| }; | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_pingpong *dpu_hw_pingpong_init(enum dpu_pingpong idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_pingpong *c; | ||||
| 	struct dpu_pingpong_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	c = kzalloc(sizeof(*c), GFP_KERNEL); | ||||
| 	if (!c) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _pingpong_offset(idx, m, addr, &c->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(c); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	c->idx = idx; | ||||
| 	c->caps = cfg; | ||||
| 	_setup_pingpong_ops(&c->ops, c->caps); | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&c->base, DPU_HW_BLK_PINGPONG, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	return c; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(c); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_pingpong_destroy(struct dpu_hw_pingpong *pp) | ||||
| { | ||||
| 	if (pp) | ||||
| 		dpu_hw_blk_destroy(&pp->base); | ||||
| 	kfree(pp); | ||||
| } | ||||
							
								
								
									
										136
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_pingpong.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_pingpong.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,136 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_PINGPONG_H | ||||
| #define _DPU_HW_PINGPONG_H | ||||
| 
 | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| struct dpu_hw_pingpong; | ||||
| 
 | ||||
| struct dpu_hw_tear_check { | ||||
| 	/*
 | ||||
| 	 * This is ratio of MDP VSYNC clk freq(Hz) to | ||||
| 	 * refresh rate divided by no of lines | ||||
| 	 */ | ||||
| 	u32 vsync_count; | ||||
| 	u32 sync_cfg_height; | ||||
| 	u32 vsync_init_val; | ||||
| 	u32 sync_threshold_start; | ||||
| 	u32 sync_threshold_continue; | ||||
| 	u32 start_pos; | ||||
| 	u32 rd_ptr_irq; | ||||
| 	u8 hw_vsync_mode; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_pp_vsync_info { | ||||
| 	u32 rd_ptr_init_val;	/* value of rd pointer at vsync edge */ | ||||
| 	u32 rd_ptr_frame_count;	/* num frames sent since enabling interface */ | ||||
| 	u32 rd_ptr_line_count;	/* current line on panel (rd ptr) */ | ||||
| 	u32 wr_ptr_line_count;	/* current line within pp fifo (wr ptr) */ | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * | ||||
|  * struct dpu_hw_pingpong_ops : Interface to the pingpong Hw driver functions | ||||
|  *  Assumption is these functions will be called after clocks are enabled | ||||
|  *  @setup_tearcheck : program tear check values | ||||
|  *  @enable_tearcheck : enables tear check | ||||
|  *  @get_vsync_info : retries timing info of the panel | ||||
|  *  @setup_dither : function to program the dither hw block | ||||
|  *  @get_line_count: obtain current vertical line counter | ||||
|  */ | ||||
| struct dpu_hw_pingpong_ops { | ||||
| 	/**
 | ||||
| 	 * enables vysnc generation and sets up init value of | ||||
| 	 * read pointer and programs the tear check cofiguration | ||||
| 	 */ | ||||
| 	int (*setup_tearcheck)(struct dpu_hw_pingpong *pp, | ||||
| 			struct dpu_hw_tear_check *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * enables tear check block | ||||
| 	 */ | ||||
| 	int (*enable_tearcheck)(struct dpu_hw_pingpong *pp, | ||||
| 			bool enable); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * read, modify, write to either set or clear listening to external TE | ||||
| 	 * @Return: 1 if TE was originally connected, 0 if not, or -ERROR | ||||
| 	 */ | ||||
| 	int (*connect_external_te)(struct dpu_hw_pingpong *pp, | ||||
| 			bool enable_external_te); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * provides the programmed and current | ||||
| 	 * line_count | ||||
| 	 */ | ||||
| 	int (*get_vsync_info)(struct dpu_hw_pingpong *pp, | ||||
| 			struct dpu_hw_pp_vsync_info  *info); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * poll until write pointer transmission starts | ||||
| 	 * @Return: 0 on success, -ETIMEDOUT on timeout | ||||
| 	 */ | ||||
| 	int (*poll_timeout_wr_ptr)(struct dpu_hw_pingpong *pp, u32 timeout_us); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * Obtain current vertical line counter | ||||
| 	 */ | ||||
| 	u32 (*get_line_count)(struct dpu_hw_pingpong *pp); | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_pingpong { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* pingpong */ | ||||
| 	enum dpu_pingpong idx; | ||||
| 	const struct dpu_pingpong_cfg *caps; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_pingpong_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_pingpong - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_pingpong *to_dpu_hw_pingpong(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_pingpong, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_pingpong_init - initializes the pingpong driver for the passed | ||||
|  *	pingpong idx. | ||||
|  * @idx:  Pingpong index for which driver object is required | ||||
|  * @addr: Mapped register io address of MDP | ||||
|  * @m:    Pointer to mdss catalog data | ||||
|  * Returns: Error code or allocated dpu_hw_pingpong context | ||||
|  */ | ||||
| struct dpu_hw_pingpong *dpu_hw_pingpong_init(enum dpu_pingpong idx, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_pingpong_destroy - destroys pingpong driver context | ||||
|  *	should be called to free the context | ||||
|  * @pp:   Pointer to PP driver context returned by dpu_hw_pingpong_init | ||||
|  */ | ||||
| void dpu_hw_pingpong_destroy(struct dpu_hw_pingpong *pp); | ||||
| 
 | ||||
| #endif /*_DPU_HW_PINGPONG_H */ | ||||
							
								
								
									
										753
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_sspp.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										753
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_sspp.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,753 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_lm.h" | ||||
| #include "dpu_hw_sspp.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define DPU_FETCH_CONFIG_RESET_VALUE   0x00000087 | ||||
| 
 | ||||
| /* DPU_SSPP_SRC */ | ||||
| #define SSPP_SRC_SIZE                      0x00 | ||||
| #define SSPP_SRC_XY                        0x08 | ||||
| #define SSPP_OUT_SIZE                      0x0c | ||||
| #define SSPP_OUT_XY                        0x10 | ||||
| #define SSPP_SRC0_ADDR                     0x14 | ||||
| #define SSPP_SRC1_ADDR                     0x18 | ||||
| #define SSPP_SRC2_ADDR                     0x1C | ||||
| #define SSPP_SRC3_ADDR                     0x20 | ||||
| #define SSPP_SRC_YSTRIDE0                  0x24 | ||||
| #define SSPP_SRC_YSTRIDE1                  0x28 | ||||
| #define SSPP_SRC_FORMAT                    0x30 | ||||
| #define SSPP_SRC_UNPACK_PATTERN            0x34 | ||||
| #define SSPP_SRC_OP_MODE                   0x38 | ||||
| 
 | ||||
| /* SSPP_MULTIRECT*/ | ||||
| #define SSPP_SRC_SIZE_REC1                 0x16C | ||||
| #define SSPP_SRC_XY_REC1                   0x168 | ||||
| #define SSPP_OUT_SIZE_REC1                 0x160 | ||||
| #define SSPP_OUT_XY_REC1                   0x164 | ||||
| #define SSPP_SRC_FORMAT_REC1               0x174 | ||||
| #define SSPP_SRC_UNPACK_PATTERN_REC1       0x178 | ||||
| #define SSPP_SRC_OP_MODE_REC1              0x17C | ||||
| #define SSPP_MULTIRECT_OPMODE              0x170 | ||||
| #define SSPP_SRC_CONSTANT_COLOR_REC1       0x180 | ||||
| #define SSPP_EXCL_REC_SIZE_REC1            0x184 | ||||
| #define SSPP_EXCL_REC_XY_REC1              0x188 | ||||
| 
 | ||||
| #define MDSS_MDP_OP_DEINTERLACE            BIT(22) | ||||
| #define MDSS_MDP_OP_DEINTERLACE_ODD        BIT(23) | ||||
| #define MDSS_MDP_OP_IGC_ROM_1              BIT(18) | ||||
| #define MDSS_MDP_OP_IGC_ROM_0              BIT(17) | ||||
| #define MDSS_MDP_OP_IGC_EN                 BIT(16) | ||||
| #define MDSS_MDP_OP_FLIP_UD                BIT(14) | ||||
| #define MDSS_MDP_OP_FLIP_LR                BIT(13) | ||||
| #define MDSS_MDP_OP_BWC_EN                 BIT(0) | ||||
| #define MDSS_MDP_OP_PE_OVERRIDE            BIT(31) | ||||
| #define MDSS_MDP_OP_BWC_LOSSLESS           (0 << 1) | ||||
| #define MDSS_MDP_OP_BWC_Q_HIGH             (1 << 1) | ||||
| #define MDSS_MDP_OP_BWC_Q_MED              (2 << 1) | ||||
| 
 | ||||
| #define SSPP_SRC_CONSTANT_COLOR            0x3c | ||||
| #define SSPP_EXCL_REC_CTL                  0x40 | ||||
| #define SSPP_UBWC_STATIC_CTRL              0x44 | ||||
| #define SSPP_FETCH_CONFIG                  0x048 | ||||
| #define SSPP_DANGER_LUT                    0x60 | ||||
| #define SSPP_SAFE_LUT                      0x64 | ||||
| #define SSPP_CREQ_LUT                      0x68 | ||||
| #define SSPP_QOS_CTRL                      0x6C | ||||
| #define SSPP_DECIMATION_CONFIG             0xB4 | ||||
| #define SSPP_SRC_ADDR_SW_STATUS            0x70 | ||||
| #define SSPP_CREQ_LUT_0                    0x74 | ||||
| #define SSPP_CREQ_LUT_1                    0x78 | ||||
| #define SSPP_SW_PIX_EXT_C0_LR              0x100 | ||||
| #define SSPP_SW_PIX_EXT_C0_TB              0x104 | ||||
| #define SSPP_SW_PIX_EXT_C0_REQ_PIXELS      0x108 | ||||
| #define SSPP_SW_PIX_EXT_C1C2_LR            0x110 | ||||
| #define SSPP_SW_PIX_EXT_C1C2_TB            0x114 | ||||
| #define SSPP_SW_PIX_EXT_C1C2_REQ_PIXELS    0x118 | ||||
| #define SSPP_SW_PIX_EXT_C3_LR              0x120 | ||||
| #define SSPP_SW_PIX_EXT_C3_TB              0x124 | ||||
| #define SSPP_SW_PIX_EXT_C3_REQ_PIXELS      0x128 | ||||
| #define SSPP_TRAFFIC_SHAPER                0x130 | ||||
| #define SSPP_CDP_CNTL                      0x134 | ||||
| #define SSPP_UBWC_ERROR_STATUS             0x138 | ||||
| #define SSPP_TRAFFIC_SHAPER_PREFILL        0x150 | ||||
| #define SSPP_TRAFFIC_SHAPER_REC1_PREFILL   0x154 | ||||
| #define SSPP_TRAFFIC_SHAPER_REC1           0x158 | ||||
| #define SSPP_EXCL_REC_SIZE                 0x1B4 | ||||
| #define SSPP_EXCL_REC_XY                   0x1B8 | ||||
| #define SSPP_VIG_OP_MODE                   0x0 | ||||
| #define SSPP_VIG_CSC_10_OP_MODE            0x0 | ||||
| #define SSPP_TRAFFIC_SHAPER_BPC_MAX        0xFF | ||||
| 
 | ||||
| /* SSPP_QOS_CTRL */ | ||||
| #define SSPP_QOS_CTRL_VBLANK_EN            BIT(16) | ||||
| #define SSPP_QOS_CTRL_DANGER_SAFE_EN       BIT(0) | ||||
| #define SSPP_QOS_CTRL_DANGER_VBLANK_MASK   0x3 | ||||
| #define SSPP_QOS_CTRL_DANGER_VBLANK_OFF    4 | ||||
| #define SSPP_QOS_CTRL_CREQ_VBLANK_MASK     0x3 | ||||
| #define SSPP_QOS_CTRL_CREQ_VBLANK_OFF      20 | ||||
| 
 | ||||
| /* DPU_SSPP_SCALER_QSEED2 */ | ||||
| #define SCALE_CONFIG                       0x04 | ||||
| #define COMP0_3_PHASE_STEP_X               0x10 | ||||
| #define COMP0_3_PHASE_STEP_Y               0x14 | ||||
| #define COMP1_2_PHASE_STEP_X               0x18 | ||||
| #define COMP1_2_PHASE_STEP_Y               0x1c | ||||
| #define COMP0_3_INIT_PHASE_X               0x20 | ||||
| #define COMP0_3_INIT_PHASE_Y               0x24 | ||||
| #define COMP1_2_INIT_PHASE_X               0x28 | ||||
| #define COMP1_2_INIT_PHASE_Y               0x2C | ||||
| #define VIG_0_QSEED2_SHARP                 0x30 | ||||
| 
 | ||||
| /*
 | ||||
|  * Definitions for ViG op modes | ||||
|  */ | ||||
| #define VIG_OP_CSC_DST_DATAFMT BIT(19) | ||||
| #define VIG_OP_CSC_SRC_DATAFMT BIT(18) | ||||
| #define VIG_OP_CSC_EN          BIT(17) | ||||
| #define VIG_OP_MEM_PROT_CONT   BIT(15) | ||||
| #define VIG_OP_MEM_PROT_VAL    BIT(14) | ||||
| #define VIG_OP_MEM_PROT_SAT    BIT(13) | ||||
| #define VIG_OP_MEM_PROT_HUE    BIT(12) | ||||
| #define VIG_OP_HIST            BIT(8) | ||||
| #define VIG_OP_SKY_COL         BIT(7) | ||||
| #define VIG_OP_FOIL            BIT(6) | ||||
| #define VIG_OP_SKIN_COL        BIT(5) | ||||
| #define VIG_OP_PA_EN           BIT(4) | ||||
| #define VIG_OP_PA_SAT_ZERO_EXP BIT(2) | ||||
| #define VIG_OP_MEM_PROT_BLEND  BIT(1) | ||||
| 
 | ||||
| /*
 | ||||
|  * Definitions for CSC 10 op modes | ||||
|  */ | ||||
| #define VIG_CSC_10_SRC_DATAFMT BIT(1) | ||||
| #define VIG_CSC_10_EN          BIT(0) | ||||
| #define CSC_10BIT_OFFSET       4 | ||||
| 
 | ||||
| /* traffic shaper clock in Hz */ | ||||
| #define TS_CLK			19200000 | ||||
| 
 | ||||
| static inline int _sspp_subblk_offset(struct dpu_hw_pipe *ctx, | ||||
| 		int s_id, | ||||
| 		u32 *idx) | ||||
| { | ||||
| 	int rc = 0; | ||||
| 	const struct dpu_sspp_sub_blks *sblk = ctx->cap->sblk; | ||||
| 
 | ||||
| 	if (!ctx) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	switch (s_id) { | ||||
| 	case DPU_SSPP_SRC: | ||||
| 		*idx = sblk->src_blk.base; | ||||
| 		break; | ||||
| 	case DPU_SSPP_SCALER_QSEED2: | ||||
| 	case DPU_SSPP_SCALER_QSEED3: | ||||
| 	case DPU_SSPP_SCALER_RGB: | ||||
| 		*idx = sblk->scaler_blk.base; | ||||
| 		break; | ||||
| 	case DPU_SSPP_CSC: | ||||
| 	case DPU_SSPP_CSC_10BIT: | ||||
| 		*idx = sblk->csc_blk.base; | ||||
| 		break; | ||||
| 	default: | ||||
| 		rc = -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_multirect(struct dpu_hw_pipe *ctx, | ||||
| 		enum dpu_sspp_multirect_index index, | ||||
| 		enum dpu_sspp_multirect_mode mode) | ||||
| { | ||||
| 	u32 mode_mask; | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (index == DPU_SSPP_RECT_SOLO) { | ||||
| 		/**
 | ||||
| 		 * if rect index is RECT_SOLO, we cannot expect a | ||||
| 		 * virtual plane sharing the same SSPP id. So we go | ||||
| 		 * and disable multirect | ||||
| 		 */ | ||||
| 		mode_mask = 0; | ||||
| 	} else { | ||||
| 		mode_mask = DPU_REG_READ(&ctx->hw, SSPP_MULTIRECT_OPMODE + idx); | ||||
| 		mode_mask |= index; | ||||
| 		if (mode == DPU_SSPP_MULTIRECT_TIME_MX) | ||||
| 			mode_mask |= BIT(2); | ||||
| 		else | ||||
| 			mode_mask &= ~BIT(2); | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_MULTIRECT_OPMODE + idx, mode_mask); | ||||
| } | ||||
| 
 | ||||
| static void _sspp_setup_opmode(struct dpu_hw_pipe *ctx, | ||||
| 		u32 mask, u8 en) | ||||
| { | ||||
| 	u32 idx; | ||||
| 	u32 opmode; | ||||
| 
 | ||||
| 	if (!test_bit(DPU_SSPP_SCALER_QSEED2, &ctx->cap->features) || | ||||
| 		_sspp_subblk_offset(ctx, DPU_SSPP_SCALER_QSEED2, &idx) || | ||||
| 		!test_bit(DPU_SSPP_CSC, &ctx->cap->features)) | ||||
| 		return; | ||||
| 
 | ||||
| 	opmode = DPU_REG_READ(&ctx->hw, SSPP_VIG_OP_MODE + idx); | ||||
| 
 | ||||
| 	if (en) | ||||
| 		opmode |= mask; | ||||
| 	else | ||||
| 		opmode &= ~mask; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_VIG_OP_MODE + idx, opmode); | ||||
| } | ||||
| 
 | ||||
| static void _sspp_setup_csc10_opmode(struct dpu_hw_pipe *ctx, | ||||
| 		u32 mask, u8 en) | ||||
| { | ||||
| 	u32 idx; | ||||
| 	u32 opmode; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_CSC_10BIT, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	opmode = DPU_REG_READ(&ctx->hw, SSPP_VIG_CSC_10_OP_MODE + idx); | ||||
| 	if (en) | ||||
| 		opmode |= mask; | ||||
| 	else | ||||
| 		opmode &= ~mask; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_VIG_CSC_10_OP_MODE + idx, opmode); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * Setup source pixel format, flip, | ||||
|  */ | ||||
| static void dpu_hw_sspp_setup_format(struct dpu_hw_pipe *ctx, | ||||
| 		const struct dpu_format *fmt, u32 flags, | ||||
| 		enum dpu_sspp_multirect_index rect_mode) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 chroma_samp, unpack, src_format; | ||||
| 	u32 opmode = 0; | ||||
| 	u32 fast_clear = 0; | ||||
| 	u32 op_mode_off, unpack_pat_off, format_off; | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx) || !fmt) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (rect_mode == DPU_SSPP_RECT_SOLO || rect_mode == DPU_SSPP_RECT_0) { | ||||
| 		op_mode_off = SSPP_SRC_OP_MODE; | ||||
| 		unpack_pat_off = SSPP_SRC_UNPACK_PATTERN; | ||||
| 		format_off = SSPP_SRC_FORMAT; | ||||
| 	} else { | ||||
| 		op_mode_off = SSPP_SRC_OP_MODE_REC1; | ||||
| 		unpack_pat_off = SSPP_SRC_UNPACK_PATTERN_REC1; | ||||
| 		format_off = SSPP_SRC_FORMAT_REC1; | ||||
| 	} | ||||
| 
 | ||||
| 	c = &ctx->hw; | ||||
| 	opmode = DPU_REG_READ(c, op_mode_off + idx); | ||||
| 	opmode &= ~(MDSS_MDP_OP_FLIP_LR | MDSS_MDP_OP_FLIP_UD | | ||||
| 			MDSS_MDP_OP_BWC_EN | MDSS_MDP_OP_PE_OVERRIDE); | ||||
| 
 | ||||
| 	if (flags & DPU_SSPP_FLIP_LR) | ||||
| 		opmode |= MDSS_MDP_OP_FLIP_LR; | ||||
| 	if (flags & DPU_SSPP_FLIP_UD) | ||||
| 		opmode |= MDSS_MDP_OP_FLIP_UD; | ||||
| 
 | ||||
| 	chroma_samp = fmt->chroma_sample; | ||||
| 	if (flags & DPU_SSPP_SOURCE_ROTATED_90) { | ||||
| 		if (chroma_samp == DPU_CHROMA_H2V1) | ||||
| 			chroma_samp = DPU_CHROMA_H1V2; | ||||
| 		else if (chroma_samp == DPU_CHROMA_H1V2) | ||||
| 			chroma_samp = DPU_CHROMA_H2V1; | ||||
| 	} | ||||
| 
 | ||||
| 	src_format = (chroma_samp << 23) | (fmt->fetch_planes << 19) | | ||||
| 		(fmt->bits[C3_ALPHA] << 6) | (fmt->bits[C2_R_Cr] << 4) | | ||||
| 		(fmt->bits[C1_B_Cb] << 2) | (fmt->bits[C0_G_Y] << 0); | ||||
| 
 | ||||
| 	if (flags & DPU_SSPP_ROT_90) | ||||
| 		src_format |= BIT(11); /* ROT90 */ | ||||
| 
 | ||||
| 	if (fmt->alpha_enable && fmt->fetch_planes == DPU_PLANE_INTERLEAVED) | ||||
| 		src_format |= BIT(8); /* SRCC3_EN */ | ||||
| 
 | ||||
| 	if (flags & DPU_SSPP_SOLID_FILL) | ||||
| 		src_format |= BIT(22); | ||||
| 
 | ||||
| 	unpack = (fmt->element[3] << 24) | (fmt->element[2] << 16) | | ||||
| 		(fmt->element[1] << 8) | (fmt->element[0] << 0); | ||||
| 	src_format |= ((fmt->unpack_count - 1) << 12) | | ||||
| 		(fmt->unpack_tight << 17) | | ||||
| 		(fmt->unpack_align_msb << 18) | | ||||
| 		((fmt->bpp - 1) << 9); | ||||
| 
 | ||||
| 	if (fmt->fetch_mode != DPU_FETCH_LINEAR) { | ||||
| 		if (DPU_FORMAT_IS_UBWC(fmt)) | ||||
| 			opmode |= MDSS_MDP_OP_BWC_EN; | ||||
| 		src_format |= (fmt->fetch_mode & 3) << 30; /*FRAME_FORMAT */ | ||||
| 		DPU_REG_WRITE(c, SSPP_FETCH_CONFIG, | ||||
| 			DPU_FETCH_CONFIG_RESET_VALUE | | ||||
| 			ctx->mdp->highest_bank_bit << 18); | ||||
| 		if (IS_UBWC_20_SUPPORTED(ctx->catalog->caps->ubwc_version)) { | ||||
| 			fast_clear = fmt->alpha_enable ? BIT(31) : 0; | ||||
| 			DPU_REG_WRITE(c, SSPP_UBWC_STATIC_CTRL, | ||||
| 					fast_clear | (ctx->mdp->ubwc_swizzle) | | ||||
| 					(ctx->mdp->highest_bank_bit << 4)); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	opmode |= MDSS_MDP_OP_PE_OVERRIDE; | ||||
| 
 | ||||
| 	/* if this is YUV pixel format, enable CSC */ | ||||
| 	if (DPU_FORMAT_IS_YUV(fmt)) | ||||
| 		src_format |= BIT(15); | ||||
| 
 | ||||
| 	if (DPU_FORMAT_IS_DX(fmt)) | ||||
| 		src_format |= BIT(14); | ||||
| 
 | ||||
| 	/* update scaler opmode, if appropriate */ | ||||
| 	if (test_bit(DPU_SSPP_CSC, &ctx->cap->features)) | ||||
| 		_sspp_setup_opmode(ctx, VIG_OP_CSC_EN | VIG_OP_CSC_SRC_DATAFMT, | ||||
| 			DPU_FORMAT_IS_YUV(fmt)); | ||||
| 	else if (test_bit(DPU_SSPP_CSC_10BIT, &ctx->cap->features)) | ||||
| 		_sspp_setup_csc10_opmode(ctx, | ||||
| 			VIG_CSC_10_EN | VIG_CSC_10_SRC_DATAFMT, | ||||
| 			DPU_FORMAT_IS_YUV(fmt)); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, format_off + idx, src_format); | ||||
| 	DPU_REG_WRITE(c, unpack_pat_off + idx, unpack); | ||||
| 	DPU_REG_WRITE(c, op_mode_off + idx, opmode); | ||||
| 
 | ||||
| 	/* clear previous UBWC error */ | ||||
| 	DPU_REG_WRITE(c, SSPP_UBWC_ERROR_STATUS + idx, BIT(31)); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_pe_config(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pixel_ext *pe_ext) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u8 color; | ||||
| 	u32 lr_pe[4], tb_pe[4], tot_req_pixels[4]; | ||||
| 	const u32 bytemask = 0xff; | ||||
| 	const u32 shortmask = 0xffff; | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx) || !pe_ext) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &ctx->hw; | ||||
| 
 | ||||
| 	/* program SW pixel extension override for all pipes*/ | ||||
| 	for (color = 0; color < DPU_MAX_PLANES; color++) { | ||||
| 		/* color 2 has the same set of registers as color 1 */ | ||||
| 		if (color == 2) | ||||
| 			continue; | ||||
| 
 | ||||
| 		lr_pe[color] = ((pe_ext->right_ftch[color] & bytemask) << 24)| | ||||
| 			((pe_ext->right_rpt[color] & bytemask) << 16)| | ||||
| 			((pe_ext->left_ftch[color] & bytemask) << 8)| | ||||
| 			(pe_ext->left_rpt[color] & bytemask); | ||||
| 
 | ||||
| 		tb_pe[color] = ((pe_ext->btm_ftch[color] & bytemask) << 24)| | ||||
| 			((pe_ext->btm_rpt[color] & bytemask) << 16)| | ||||
| 			((pe_ext->top_ftch[color] & bytemask) << 8)| | ||||
| 			(pe_ext->top_rpt[color] & bytemask); | ||||
| 
 | ||||
| 		tot_req_pixels[color] = (((pe_ext->roi_h[color] + | ||||
| 			pe_ext->num_ext_pxls_top[color] + | ||||
| 			pe_ext->num_ext_pxls_btm[color]) & shortmask) << 16) | | ||||
| 			((pe_ext->roi_w[color] + | ||||
| 			pe_ext->num_ext_pxls_left[color] + | ||||
| 			pe_ext->num_ext_pxls_right[color]) & shortmask); | ||||
| 	} | ||||
| 
 | ||||
| 	/* color 0 */ | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C0_LR + idx, lr_pe[0]); | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C0_TB + idx, tb_pe[0]); | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C0_REQ_PIXELS + idx, | ||||
| 			tot_req_pixels[0]); | ||||
| 
 | ||||
| 	/* color 1 and color 2 */ | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C1C2_LR + idx, lr_pe[1]); | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C1C2_TB + idx, tb_pe[1]); | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C1C2_REQ_PIXELS + idx, | ||||
| 			tot_req_pixels[1]); | ||||
| 
 | ||||
| 	/* color 3 */ | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C3_LR + idx, lr_pe[3]); | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C3_TB + idx, lr_pe[3]); | ||||
| 	DPU_REG_WRITE(c, SSPP_SW_PIX_EXT_C3_REQ_PIXELS + idx, | ||||
| 			tot_req_pixels[3]); | ||||
| } | ||||
| 
 | ||||
| static void _dpu_hw_sspp_setup_scaler3(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_cfg *sspp, | ||||
| 		struct dpu_hw_pixel_ext *pe, | ||||
| 		void *scaler_cfg) | ||||
| { | ||||
| 	u32 idx; | ||||
| 	struct dpu_hw_scaler3_cfg *scaler3_cfg = scaler_cfg; | ||||
| 
 | ||||
| 	(void)pe; | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SCALER_QSEED3, &idx) || !sspp | ||||
| 		|| !scaler3_cfg || !ctx || !ctx->cap || !ctx->cap->sblk) | ||||
| 		return; | ||||
| 
 | ||||
| 	dpu_hw_setup_scaler3(&ctx->hw, scaler3_cfg, idx, | ||||
| 			ctx->cap->sblk->scaler_blk.version, | ||||
| 			sspp->layout.format); | ||||
| } | ||||
| 
 | ||||
| static u32 _dpu_hw_sspp_get_scaler3_ver(struct dpu_hw_pipe *ctx) | ||||
| { | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (!ctx || _sspp_subblk_offset(ctx, DPU_SSPP_SCALER_QSEED3, &idx)) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	return dpu_hw_get_scaler3_ver(&ctx->hw, idx); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_sspp_setup_rects() | ||||
|  */ | ||||
| static void dpu_hw_sspp_setup_rects(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_cfg *cfg, | ||||
| 		enum dpu_sspp_multirect_index rect_index) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 src_size, src_xy, dst_size, dst_xy, ystride0, ystride1; | ||||
| 	u32 src_size_off, src_xy_off, out_size_off, out_xy_off; | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx) || !cfg) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &ctx->hw; | ||||
| 
 | ||||
| 	if (rect_index == DPU_SSPP_RECT_SOLO || rect_index == DPU_SSPP_RECT_0) { | ||||
| 		src_size_off = SSPP_SRC_SIZE; | ||||
| 		src_xy_off = SSPP_SRC_XY; | ||||
| 		out_size_off = SSPP_OUT_SIZE; | ||||
| 		out_xy_off = SSPP_OUT_XY; | ||||
| 	} else { | ||||
| 		src_size_off = SSPP_SRC_SIZE_REC1; | ||||
| 		src_xy_off = SSPP_SRC_XY_REC1; | ||||
| 		out_size_off = SSPP_OUT_SIZE_REC1; | ||||
| 		out_xy_off = SSPP_OUT_XY_REC1; | ||||
| 	} | ||||
| 
 | ||||
| 
 | ||||
| 	/* src and dest rect programming */ | ||||
| 	src_xy = (cfg->src_rect.y1 << 16) | cfg->src_rect.x1; | ||||
| 	src_size = (drm_rect_height(&cfg->src_rect) << 16) | | ||||
| 		   drm_rect_width(&cfg->src_rect); | ||||
| 	dst_xy = (cfg->dst_rect.y1 << 16) | cfg->dst_rect.x1; | ||||
| 	dst_size = (drm_rect_height(&cfg->dst_rect) << 16) | | ||||
| 		drm_rect_width(&cfg->dst_rect); | ||||
| 
 | ||||
| 	if (rect_index == DPU_SSPP_RECT_SOLO) { | ||||
| 		ystride0 = (cfg->layout.plane_pitch[0]) | | ||||
| 			(cfg->layout.plane_pitch[1] << 16); | ||||
| 		ystride1 = (cfg->layout.plane_pitch[2]) | | ||||
| 			(cfg->layout.plane_pitch[3] << 16); | ||||
| 	} else { | ||||
| 		ystride0 = DPU_REG_READ(c, SSPP_SRC_YSTRIDE0 + idx); | ||||
| 		ystride1 = DPU_REG_READ(c, SSPP_SRC_YSTRIDE1 + idx); | ||||
| 
 | ||||
| 		if (rect_index == DPU_SSPP_RECT_0) { | ||||
| 			ystride0 = (ystride0 & 0xFFFF0000) | | ||||
| 				(cfg->layout.plane_pitch[0] & 0x0000FFFF); | ||||
| 			ystride1 = (ystride1 & 0xFFFF0000)| | ||||
| 				(cfg->layout.plane_pitch[2] & 0x0000FFFF); | ||||
| 		} else { | ||||
| 			ystride0 = (ystride0 & 0x0000FFFF) | | ||||
| 				((cfg->layout.plane_pitch[0] << 16) & | ||||
| 				 0xFFFF0000); | ||||
| 			ystride1 = (ystride1 & 0x0000FFFF) | | ||||
| 				((cfg->layout.plane_pitch[2] << 16) & | ||||
| 				 0xFFFF0000); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/* rectangle register programming */ | ||||
| 	DPU_REG_WRITE(c, src_size_off + idx, src_size); | ||||
| 	DPU_REG_WRITE(c, src_xy_off + idx, src_xy); | ||||
| 	DPU_REG_WRITE(c, out_size_off + idx, dst_size); | ||||
| 	DPU_REG_WRITE(c, out_xy_off + idx, dst_xy); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, SSPP_SRC_YSTRIDE0 + idx, ystride0); | ||||
| 	DPU_REG_WRITE(c, SSPP_SRC_YSTRIDE1 + idx, ystride1); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_sourceaddress(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_cfg *cfg, | ||||
| 		enum dpu_sspp_multirect_index rect_mode) | ||||
| { | ||||
| 	int i; | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (rect_mode == DPU_SSPP_RECT_SOLO) { | ||||
| 		for (i = 0; i < ARRAY_SIZE(cfg->layout.plane_addr); i++) | ||||
| 			DPU_REG_WRITE(&ctx->hw, SSPP_SRC0_ADDR + idx + i * 0x4, | ||||
| 					cfg->layout.plane_addr[i]); | ||||
| 	} else if (rect_mode == DPU_SSPP_RECT_0) { | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_SRC0_ADDR + idx, | ||||
| 				cfg->layout.plane_addr[0]); | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_SRC2_ADDR + idx, | ||||
| 				cfg->layout.plane_addr[2]); | ||||
| 	} else { | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_SRC1_ADDR + idx, | ||||
| 				cfg->layout.plane_addr[0]); | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_SRC3_ADDR + idx, | ||||
| 				cfg->layout.plane_addr[2]); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_csc(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_csc_cfg *data) | ||||
| { | ||||
| 	u32 idx; | ||||
| 	bool csc10 = false; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_CSC, &idx) || !data) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (test_bit(DPU_SSPP_CSC_10BIT, &ctx->cap->features)) { | ||||
| 		idx += CSC_10BIT_OFFSET; | ||||
| 		csc10 = true; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_hw_csc_setup(&ctx->hw, idx, data, csc10); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_solidfill(struct dpu_hw_pipe *ctx, u32 color, enum | ||||
| 		dpu_sspp_multirect_index rect_index) | ||||
| { | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (rect_index == DPU_SSPP_RECT_SOLO || rect_index == DPU_SSPP_RECT_0) | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_SRC_CONSTANT_COLOR + idx, color); | ||||
| 	else | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_SRC_CONSTANT_COLOR_REC1 + idx, | ||||
| 				color); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_danger_safe_lut(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_qos_cfg *cfg) | ||||
| { | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_DANGER_LUT + idx, cfg->danger_lut); | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_SAFE_LUT + idx, cfg->safe_lut); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_creq_lut(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_qos_cfg *cfg) | ||||
| { | ||||
| 	u32 idx; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (ctx->cap && test_bit(DPU_SSPP_QOS_8LVL, &ctx->cap->features)) { | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_CREQ_LUT_0 + idx, cfg->creq_lut); | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_CREQ_LUT_1 + idx, | ||||
| 				cfg->creq_lut >> 32); | ||||
| 	} else { | ||||
| 		DPU_REG_WRITE(&ctx->hw, SSPP_CREQ_LUT + idx, cfg->creq_lut); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_qos_ctrl(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_qos_cfg *cfg) | ||||
| { | ||||
| 	u32 idx; | ||||
| 	u32 qos_ctrl = 0; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (cfg->vblank_en) { | ||||
| 		qos_ctrl |= ((cfg->creq_vblank & | ||||
| 				SSPP_QOS_CTRL_CREQ_VBLANK_MASK) << | ||||
| 				SSPP_QOS_CTRL_CREQ_VBLANK_OFF); | ||||
| 		qos_ctrl |= ((cfg->danger_vblank & | ||||
| 				SSPP_QOS_CTRL_DANGER_VBLANK_MASK) << | ||||
| 				SSPP_QOS_CTRL_DANGER_VBLANK_OFF); | ||||
| 		qos_ctrl |= SSPP_QOS_CTRL_VBLANK_EN; | ||||
| 	} | ||||
| 
 | ||||
| 	if (cfg->danger_safe_en) | ||||
| 		qos_ctrl |= SSPP_QOS_CTRL_DANGER_SAFE_EN; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_QOS_CTRL + idx, qos_ctrl); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_sspp_setup_cdp(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_cdp_cfg *cfg) | ||||
| { | ||||
| 	u32 idx; | ||||
| 	u32 cdp_cntl = 0; | ||||
| 
 | ||||
| 	if (!ctx || !cfg) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (_sspp_subblk_offset(ctx, DPU_SSPP_SRC, &idx)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (cfg->enable) | ||||
| 		cdp_cntl |= BIT(0); | ||||
| 	if (cfg->ubwc_meta_enable) | ||||
| 		cdp_cntl |= BIT(1); | ||||
| 	if (cfg->tile_amortize_enable) | ||||
| 		cdp_cntl |= BIT(2); | ||||
| 	if (cfg->preload_ahead == DPU_SSPP_CDP_PRELOAD_AHEAD_64) | ||||
| 		cdp_cntl |= BIT(3); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(&ctx->hw, SSPP_CDP_CNTL, cdp_cntl); | ||||
| } | ||||
| 
 | ||||
| static void _setup_layer_ops(struct dpu_hw_pipe *c, | ||||
| 		unsigned long features) | ||||
| { | ||||
| 	if (test_bit(DPU_SSPP_SRC, &features)) { | ||||
| 		c->ops.setup_format = dpu_hw_sspp_setup_format; | ||||
| 		c->ops.setup_rects = dpu_hw_sspp_setup_rects; | ||||
| 		c->ops.setup_sourceaddress = dpu_hw_sspp_setup_sourceaddress; | ||||
| 		c->ops.setup_solidfill = dpu_hw_sspp_setup_solidfill; | ||||
| 		c->ops.setup_pe = dpu_hw_sspp_setup_pe_config; | ||||
| 	} | ||||
| 
 | ||||
| 	if (test_bit(DPU_SSPP_QOS, &features)) { | ||||
| 		c->ops.setup_danger_safe_lut = | ||||
| 			dpu_hw_sspp_setup_danger_safe_lut; | ||||
| 		c->ops.setup_creq_lut = dpu_hw_sspp_setup_creq_lut; | ||||
| 		c->ops.setup_qos_ctrl = dpu_hw_sspp_setup_qos_ctrl; | ||||
| 	} | ||||
| 
 | ||||
| 	if (test_bit(DPU_SSPP_CSC, &features) || | ||||
| 		test_bit(DPU_SSPP_CSC_10BIT, &features)) | ||||
| 		c->ops.setup_csc = dpu_hw_sspp_setup_csc; | ||||
| 
 | ||||
| 	if (dpu_hw_sspp_multirect_enabled(c->cap)) | ||||
| 		c->ops.setup_multirect = dpu_hw_sspp_setup_multirect; | ||||
| 
 | ||||
| 	if (test_bit(DPU_SSPP_SCALER_QSEED3, &features)) { | ||||
| 		c->ops.setup_scaler = _dpu_hw_sspp_setup_scaler3; | ||||
| 		c->ops.get_scaler_ver = _dpu_hw_sspp_get_scaler3_ver; | ||||
| 	} | ||||
| 
 | ||||
| 	if (test_bit(DPU_SSPP_CDP, &features)) | ||||
| 		c->ops.setup_cdp = dpu_hw_sspp_setup_cdp; | ||||
| } | ||||
| 
 | ||||
| static struct dpu_sspp_cfg *_sspp_offset(enum dpu_sspp sspp, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_mdss_cfg *catalog, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	if ((sspp < SSPP_MAX) && catalog && addr && b) { | ||||
| 		for (i = 0; i < catalog->sspp_count; i++) { | ||||
| 			if (sspp == catalog->sspp[i].id) { | ||||
| 				b->base_off = addr; | ||||
| 				b->blk_off = catalog->sspp[i].base; | ||||
| 				b->length = catalog->sspp[i].len; | ||||
| 				b->hwversion = catalog->hwversion; | ||||
| 				b->log_mask = DPU_DBG_MASK_SSPP; | ||||
| 				return &catalog->sspp[i]; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-ENOMEM); | ||||
| } | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_pipe *dpu_hw_sspp_init(enum dpu_sspp idx, | ||||
| 		void __iomem *addr, struct dpu_mdss_cfg *catalog, | ||||
| 		bool is_virtual_pipe) | ||||
| { | ||||
| 	struct dpu_hw_pipe *hw_pipe; | ||||
| 	struct dpu_sspp_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (!addr || !catalog) | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 
 | ||||
| 	hw_pipe = kzalloc(sizeof(*hw_pipe), GFP_KERNEL); | ||||
| 	if (!hw_pipe) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _sspp_offset(idx, addr, catalog, &hw_pipe->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(hw_pipe); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	/* Assign ops */ | ||||
| 	hw_pipe->catalog = catalog; | ||||
| 	hw_pipe->mdp = &catalog->mdp[0]; | ||||
| 	hw_pipe->idx = idx; | ||||
| 	hw_pipe->cap = cfg; | ||||
| 	_setup_layer_ops(hw_pipe, hw_pipe->cap->features); | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&hw_pipe->base, DPU_HW_BLK_SSPP, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	return hw_pipe; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(hw_pipe); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_sspp_destroy(struct dpu_hw_pipe *ctx) | ||||
| { | ||||
| 	if (ctx) | ||||
| 		dpu_hw_blk_destroy(&ctx->base); | ||||
| 	kfree(ctx); | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										424
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_sspp.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										424
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_sspp.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,424 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_SSPP_H | ||||
| #define _DPU_HW_SSPP_H | ||||
| 
 | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| #include "dpu_formats.h" | ||||
| 
 | ||||
| struct dpu_hw_pipe; | ||||
| 
 | ||||
| /**
 | ||||
|  * Flags | ||||
|  */ | ||||
| #define DPU_SSPP_FLIP_LR		BIT(0) | ||||
| #define DPU_SSPP_FLIP_UD		BIT(1) | ||||
| #define DPU_SSPP_SOURCE_ROTATED_90	BIT(2) | ||||
| #define DPU_SSPP_ROT_90			BIT(3) | ||||
| #define DPU_SSPP_SOLID_FILL		BIT(4) | ||||
| 
 | ||||
| /**
 | ||||
|  * Define all scaler feature bits in catalog | ||||
|  */ | ||||
| #define DPU_SSPP_SCALER ((1UL << DPU_SSPP_SCALER_RGB) | \ | ||||
| 	(1UL << DPU_SSPP_SCALER_QSEED2) | \ | ||||
| 	(1UL << DPU_SSPP_SCALER_QSEED3)) | ||||
| 
 | ||||
| /**
 | ||||
|  * Component indices | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_SSPP_COMP_0, | ||||
| 	DPU_SSPP_COMP_1_2, | ||||
| 	DPU_SSPP_COMP_2, | ||||
| 	DPU_SSPP_COMP_3, | ||||
| 
 | ||||
| 	DPU_SSPP_COMP_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU_SSPP_RECT_SOLO - multirect disabled | ||||
|  * DPU_SSPP_RECT_0 - rect0 of a multirect pipe | ||||
|  * DPU_SSPP_RECT_1 - rect1 of a multirect pipe | ||||
|  * | ||||
|  * Note: HW supports multirect with either RECT0 or | ||||
|  * RECT1. Considering no benefit of such configs over | ||||
|  * SOLO mode and to keep the plane management simple, | ||||
|  * we dont support single rect multirect configs. | ||||
|  */ | ||||
| enum dpu_sspp_multirect_index { | ||||
| 	DPU_SSPP_RECT_SOLO = 0, | ||||
| 	DPU_SSPP_RECT_0, | ||||
| 	DPU_SSPP_RECT_1, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_sspp_multirect_mode { | ||||
| 	DPU_SSPP_MULTIRECT_NONE = 0, | ||||
| 	DPU_SSPP_MULTIRECT_PARALLEL, | ||||
| 	DPU_SSPP_MULTIRECT_TIME_MX, | ||||
| }; | ||||
| 
 | ||||
| enum { | ||||
| 	DPU_FRAME_LINEAR, | ||||
| 	DPU_FRAME_TILE_A4X, | ||||
| 	DPU_FRAME_TILE_A5X, | ||||
| }; | ||||
| 
 | ||||
| enum dpu_hw_filter { | ||||
| 	DPU_SCALE_FILTER_NEAREST = 0, | ||||
| 	DPU_SCALE_FILTER_BIL, | ||||
| 	DPU_SCALE_FILTER_PCMN, | ||||
| 	DPU_SCALE_FILTER_CA, | ||||
| 	DPU_SCALE_FILTER_MAX | ||||
| }; | ||||
| 
 | ||||
| enum dpu_hw_filter_alpa { | ||||
| 	DPU_SCALE_ALPHA_PIXEL_REP, | ||||
| 	DPU_SCALE_ALPHA_BIL | ||||
| }; | ||||
| 
 | ||||
| enum dpu_hw_filter_yuv { | ||||
| 	DPU_SCALE_2D_4X4, | ||||
| 	DPU_SCALE_2D_CIR, | ||||
| 	DPU_SCALE_1D_SEP, | ||||
| 	DPU_SCALE_BIL | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_sharp_cfg { | ||||
| 	u32 strength; | ||||
| 	u32 edge_thr; | ||||
| 	u32 smooth_thr; | ||||
| 	u32 noise_thr; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_pixel_ext { | ||||
| 	/* scaling factors are enabled for this input layer */ | ||||
| 	uint8_t enable_pxl_ext; | ||||
| 
 | ||||
| 	int init_phase_x[DPU_MAX_PLANES]; | ||||
| 	int phase_step_x[DPU_MAX_PLANES]; | ||||
| 	int init_phase_y[DPU_MAX_PLANES]; | ||||
| 	int phase_step_y[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Number of pixels extension in left, right, top and bottom direction | ||||
| 	 * for all color components. This pixel value for each color component | ||||
| 	 * should be sum of fetch + repeat pixels. | ||||
| 	 */ | ||||
| 	int num_ext_pxls_left[DPU_MAX_PLANES]; | ||||
| 	int num_ext_pxls_right[DPU_MAX_PLANES]; | ||||
| 	int num_ext_pxls_top[DPU_MAX_PLANES]; | ||||
| 	int num_ext_pxls_btm[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Number of pixels needs to be overfetched in left, right, top and | ||||
| 	 * bottom directions from source image for scaling. | ||||
| 	 */ | ||||
| 	int left_ftch[DPU_MAX_PLANES]; | ||||
| 	int right_ftch[DPU_MAX_PLANES]; | ||||
| 	int top_ftch[DPU_MAX_PLANES]; | ||||
| 	int btm_ftch[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Number of pixels needs to be repeated in left, right, top and | ||||
| 	 * bottom directions for scaling. | ||||
| 	 */ | ||||
| 	int left_rpt[DPU_MAX_PLANES]; | ||||
| 	int right_rpt[DPU_MAX_PLANES]; | ||||
| 	int top_rpt[DPU_MAX_PLANES]; | ||||
| 	int btm_rpt[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	uint32_t roi_w[DPU_MAX_PLANES]; | ||||
| 	uint32_t roi_h[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Filter type to be used for scaling in horizontal and vertical | ||||
| 	 * directions | ||||
| 	 */ | ||||
| 	enum dpu_hw_filter horz_filter[DPU_MAX_PLANES]; | ||||
| 	enum dpu_hw_filter vert_filter[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_pipe_cfg : Pipe description | ||||
|  * @layout:    format layout information for programming buffer to hardware | ||||
|  * @src_rect:  src ROI, caller takes into account the different operations | ||||
|  *             such as decimation, flip etc to program this field | ||||
|  * @dest_rect: destination ROI. | ||||
|  * @index:     index of the rectangle of SSPP | ||||
|  * @mode:      parallel or time multiplex multirect mode | ||||
|  */ | ||||
| struct dpu_hw_pipe_cfg { | ||||
| 	struct dpu_hw_fmt_layout layout; | ||||
| 	struct drm_rect src_rect; | ||||
| 	struct drm_rect dst_rect; | ||||
| 	enum dpu_sspp_multirect_index index; | ||||
| 	enum dpu_sspp_multirect_mode mode; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_pipe_qos_cfg : Source pipe QoS configuration | ||||
|  * @danger_lut: LUT for generate danger level based on fill level | ||||
|  * @safe_lut: LUT for generate safe level based on fill level | ||||
|  * @creq_lut: LUT for generate creq level based on fill level | ||||
|  * @creq_vblank: creq value generated to vbif during vertical blanking | ||||
|  * @danger_vblank: danger value generated during vertical blanking | ||||
|  * @vblank_en: enable creq_vblank and danger_vblank during vblank | ||||
|  * @danger_safe_en: enable danger safe generation | ||||
|  */ | ||||
| struct dpu_hw_pipe_qos_cfg { | ||||
| 	u32 danger_lut; | ||||
| 	u32 safe_lut; | ||||
| 	u64 creq_lut; | ||||
| 	u32 creq_vblank; | ||||
| 	u32 danger_vblank; | ||||
| 	bool vblank_en; | ||||
| 	bool danger_safe_en; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum CDP preload ahead address size | ||||
|  */ | ||||
| enum { | ||||
| 	DPU_SSPP_CDP_PRELOAD_AHEAD_32, | ||||
| 	DPU_SSPP_CDP_PRELOAD_AHEAD_64 | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_pipe_cdp_cfg : CDP configuration | ||||
|  * @enable: true to enable CDP | ||||
|  * @ubwc_meta_enable: true to enable ubwc metadata preload | ||||
|  * @tile_amortize_enable: true to enable amortization control for tile format | ||||
|  * @preload_ahead: number of request to preload ahead | ||||
|  *	DPU_SSPP_CDP_PRELOAD_AHEAD_32, | ||||
|  *	DPU_SSPP_CDP_PRELOAD_AHEAD_64 | ||||
|  */ | ||||
| struct dpu_hw_pipe_cdp_cfg { | ||||
| 	bool enable; | ||||
| 	bool ubwc_meta_enable; | ||||
| 	bool tile_amortize_enable; | ||||
| 	u32 preload_ahead; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_pipe_ts_cfg - traffic shaper configuration | ||||
|  * @size: size to prefill in bytes, or zero to disable | ||||
|  * @time: time to prefill in usec, or zero to disable | ||||
|  */ | ||||
| struct dpu_hw_pipe_ts_cfg { | ||||
| 	u64 size; | ||||
| 	u64 time; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_sspp_ops - interface to the SSPP Hw driver functions | ||||
|  * Caller must call the init function to get the pipe context for each pipe | ||||
|  * Assumption is these functions will be called after clocks are enabled | ||||
|  */ | ||||
| struct dpu_hw_sspp_ops { | ||||
| 	/**
 | ||||
| 	 * setup_format - setup pixel format cropping rectangle, flip | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to pipe config structure | ||||
| 	 * @flags: Extra flags for format config | ||||
| 	 * @index: rectangle index in multirect | ||||
| 	 */ | ||||
| 	void (*setup_format)(struct dpu_hw_pipe *ctx, | ||||
| 			const struct dpu_format *fmt, u32 flags, | ||||
| 			enum dpu_sspp_multirect_index index); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_rects - setup pipe ROI rectangles | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to pipe config structure | ||||
| 	 * @index: rectangle index in multirect | ||||
| 	 */ | ||||
| 	void (*setup_rects)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pipe_cfg *cfg, | ||||
| 			enum dpu_sspp_multirect_index index); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_pe - setup pipe pixel extension | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @pe_ext: Pointer to pixel ext settings | ||||
| 	 */ | ||||
| 	void (*setup_pe)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pixel_ext *pe_ext); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_sourceaddress - setup pipe source addresses | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to pipe config structure | ||||
| 	 * @index: rectangle index in multirect | ||||
| 	 */ | ||||
| 	void (*setup_sourceaddress)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pipe_cfg *cfg, | ||||
| 			enum dpu_sspp_multirect_index index); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_csc - setup color space coversion | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @data: Pointer to config structure | ||||
| 	 */ | ||||
| 	void (*setup_csc)(struct dpu_hw_pipe *ctx, struct dpu_csc_cfg *data); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_solidfill - enable/disable colorfill | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @const_color: Fill color value | ||||
| 	 * @flags: Pipe flags | ||||
| 	 * @index: rectangle index in multirect | ||||
| 	 */ | ||||
| 	void (*setup_solidfill)(struct dpu_hw_pipe *ctx, u32 color, | ||||
| 			enum dpu_sspp_multirect_index index); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_multirect - setup multirect configuration | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @index: rectangle index in multirect | ||||
| 	 * @mode: parallel fetch / time multiplex multirect mode | ||||
| 	 */ | ||||
| 
 | ||||
| 	void (*setup_multirect)(struct dpu_hw_pipe *ctx, | ||||
| 			enum dpu_sspp_multirect_index index, | ||||
| 			enum dpu_sspp_multirect_mode mode); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_sharpening - setup sharpening | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to config structure | ||||
| 	 */ | ||||
| 	void (*setup_sharpening)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_sharp_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_danger_safe_lut - setup danger safe LUTs | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to pipe QoS configuration | ||||
| 	 * | ||||
| 	 */ | ||||
| 	void (*setup_danger_safe_lut)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pipe_qos_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_creq_lut - setup CREQ LUT | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to pipe QoS configuration | ||||
| 	 * | ||||
| 	 */ | ||||
| 	void (*setup_creq_lut)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pipe_qos_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_qos_ctrl - setup QoS control | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to pipe QoS configuration | ||||
| 	 * | ||||
| 	 */ | ||||
| 	void (*setup_qos_ctrl)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pipe_qos_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_histogram - setup histograms | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to histogram configuration | ||||
| 	 */ | ||||
| 	void (*setup_histogram)(struct dpu_hw_pipe *ctx, | ||||
| 			void *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_scaler - setup scaler | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @pipe_cfg: Pointer to pipe configuration | ||||
| 	 * @pe_cfg: Pointer to pixel extension configuration | ||||
| 	 * @scaler_cfg: Pointer to scaler configuration | ||||
| 	 */ | ||||
| 	void (*setup_scaler)(struct dpu_hw_pipe *ctx, | ||||
| 		struct dpu_hw_pipe_cfg *pipe_cfg, | ||||
| 		struct dpu_hw_pixel_ext *pe_cfg, | ||||
| 		void *scaler_cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_scaler_ver - get scaler h/w version | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 */ | ||||
| 	u32 (*get_scaler_ver)(struct dpu_hw_pipe *ctx); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_cdp - setup client driven prefetch | ||||
| 	 * @ctx: Pointer to pipe context | ||||
| 	 * @cfg: Pointer to cdp configuration | ||||
| 	 */ | ||||
| 	void (*setup_cdp)(struct dpu_hw_pipe *ctx, | ||||
| 			struct dpu_hw_pipe_cdp_cfg *cfg); | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_pipe - pipe description | ||||
|  * @base: hardware block base structure | ||||
|  * @hw: block hardware details | ||||
|  * @catalog: back pointer to catalog | ||||
|  * @mdp: pointer to associated mdp portion of the catalog | ||||
|  * @idx: pipe index | ||||
|  * @cap: pointer to layer_cfg | ||||
|  * @ops: pointer to operations possible for this pipe | ||||
|  */ | ||||
| struct dpu_hw_pipe { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 	struct dpu_mdss_cfg *catalog; | ||||
| 	struct dpu_mdp_cfg *mdp; | ||||
| 
 | ||||
| 	/* Pipe */ | ||||
| 	enum dpu_sspp idx; | ||||
| 	const struct dpu_sspp_cfg *cap; | ||||
| 
 | ||||
| 	/* Ops */ | ||||
| 	struct dpu_hw_sspp_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_pipe - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_pipe *to_dpu_hw_pipe(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_pipe, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_sspp_init - initializes the sspp hw driver object. | ||||
|  * Should be called once before accessing every pipe. | ||||
|  * @idx:  Pipe index for which driver object is required | ||||
|  * @addr: Mapped register io address of MDP | ||||
|  * @catalog : Pointer to mdss catalog data | ||||
|  * @is_virtual_pipe: is this pipe virtual pipe | ||||
|  */ | ||||
| struct dpu_hw_pipe *dpu_hw_sspp_init(enum dpu_sspp idx, | ||||
| 		void __iomem *addr, struct dpu_mdss_cfg *catalog, | ||||
| 		bool is_virtual_pipe); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_sspp_destroy(): Destroys SSPP driver context | ||||
|  * should be called during Hw pipe cleanup. | ||||
|  * @ctx:  Pointer to SSPP driver context returned by dpu_hw_sspp_init | ||||
|  */ | ||||
| void dpu_hw_sspp_destroy(struct dpu_hw_pipe *ctx); | ||||
| 
 | ||||
| #endif /*_DPU_HW_SSPP_H */ | ||||
| 
 | ||||
							
								
								
									
										398
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_top.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										398
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_top.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,398 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_top.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define SSPP_SPARE                        0x28 | ||||
| #define UBWC_STATIC                       0x144 | ||||
| 
 | ||||
| #define FLD_SPLIT_DISPLAY_CMD             BIT(1) | ||||
| #define FLD_SMART_PANEL_FREE_RUN          BIT(2) | ||||
| #define FLD_INTF_1_SW_TRG_MUX             BIT(4) | ||||
| #define FLD_INTF_2_SW_TRG_MUX             BIT(8) | ||||
| #define FLD_TE_LINE_INTER_WATERLEVEL_MASK 0xFFFF | ||||
| 
 | ||||
| #define DANGER_STATUS                     0x360 | ||||
| #define SAFE_STATUS                       0x364 | ||||
| 
 | ||||
| #define TE_LINE_INTERVAL                  0x3F4 | ||||
| 
 | ||||
| #define TRAFFIC_SHAPER_EN                 BIT(31) | ||||
| #define TRAFFIC_SHAPER_RD_CLIENT(num)     (0x030 + (num * 4)) | ||||
| #define TRAFFIC_SHAPER_WR_CLIENT(num)     (0x060 + (num * 4)) | ||||
| #define TRAFFIC_SHAPER_FIXPOINT_FACTOR    4 | ||||
| 
 | ||||
| #define MDP_WD_TIMER_0_CTL                0x380 | ||||
| #define MDP_WD_TIMER_0_CTL2               0x384 | ||||
| #define MDP_WD_TIMER_0_LOAD_VALUE         0x388 | ||||
| #define MDP_WD_TIMER_1_CTL                0x390 | ||||
| #define MDP_WD_TIMER_1_CTL2               0x394 | ||||
| #define MDP_WD_TIMER_1_LOAD_VALUE         0x398 | ||||
| #define MDP_WD_TIMER_2_CTL                0x420 | ||||
| #define MDP_WD_TIMER_2_CTL2               0x424 | ||||
| #define MDP_WD_TIMER_2_LOAD_VALUE         0x428 | ||||
| #define MDP_WD_TIMER_3_CTL                0x430 | ||||
| #define MDP_WD_TIMER_3_CTL2               0x434 | ||||
| #define MDP_WD_TIMER_3_LOAD_VALUE         0x438 | ||||
| #define MDP_WD_TIMER_4_CTL                0x440 | ||||
| #define MDP_WD_TIMER_4_CTL2               0x444 | ||||
| #define MDP_WD_TIMER_4_LOAD_VALUE         0x448 | ||||
| 
 | ||||
| #define MDP_TICK_COUNT                    16 | ||||
| #define XO_CLK_RATE                       19200 | ||||
| #define MS_TICKS_IN_SEC                   1000 | ||||
| 
 | ||||
| #define CALCULATE_WD_LOAD_VALUE(fps) \ | ||||
| 	((uint32_t)((MS_TICKS_IN_SEC * XO_CLK_RATE)/(MDP_TICK_COUNT * fps))) | ||||
| 
 | ||||
| #define DCE_SEL                           0x450 | ||||
| 
 | ||||
| static void dpu_hw_setup_split_pipe(struct dpu_hw_mdp *mdp, | ||||
| 		struct split_pipe_cfg *cfg) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 upper_pipe = 0; | ||||
| 	u32 lower_pipe = 0; | ||||
| 
 | ||||
| 	if (!mdp || !cfg) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 
 | ||||
| 	if (cfg->en) { | ||||
| 		if (cfg->mode == INTF_MODE_CMD) { | ||||
| 			lower_pipe = FLD_SPLIT_DISPLAY_CMD; | ||||
| 			/* interface controlling sw trigger */ | ||||
| 			if (cfg->intf == INTF_2) | ||||
| 				lower_pipe |= FLD_INTF_1_SW_TRG_MUX; | ||||
| 			else | ||||
| 				lower_pipe |= FLD_INTF_2_SW_TRG_MUX; | ||||
| 			upper_pipe = lower_pipe; | ||||
| 		} else { | ||||
| 			if (cfg->intf == INTF_2) { | ||||
| 				lower_pipe = FLD_INTF_1_SW_TRG_MUX; | ||||
| 				upper_pipe = FLD_INTF_2_SW_TRG_MUX; | ||||
| 			} else { | ||||
| 				lower_pipe = FLD_INTF_2_SW_TRG_MUX; | ||||
| 				upper_pipe = FLD_INTF_1_SW_TRG_MUX; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, SSPP_SPARE, cfg->split_flush_en ? 0x1 : 0x0); | ||||
| 	DPU_REG_WRITE(c, SPLIT_DISPLAY_LOWER_PIPE_CTRL, lower_pipe); | ||||
| 	DPU_REG_WRITE(c, SPLIT_DISPLAY_UPPER_PIPE_CTRL, upper_pipe); | ||||
| 	DPU_REG_WRITE(c, SPLIT_DISPLAY_EN, cfg->en & 0x1); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_setup_cdm_output(struct dpu_hw_mdp *mdp, | ||||
| 		struct cdm_output_cfg *cfg) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 out_ctl = 0; | ||||
| 
 | ||||
| 	if (!mdp || !cfg) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 
 | ||||
| 	if (cfg->intf_en) | ||||
| 		out_ctl |= BIT(19); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, MDP_OUT_CTL_0, out_ctl); | ||||
| } | ||||
| 
 | ||||
| static bool dpu_hw_setup_clk_force_ctrl(struct dpu_hw_mdp *mdp, | ||||
| 		enum dpu_clk_ctrl_type clk_ctrl, bool enable) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 reg_off, bit_off; | ||||
| 	u32 reg_val, new_val; | ||||
| 	bool clk_forced_on; | ||||
| 
 | ||||
| 	if (!mdp) | ||||
| 		return false; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 
 | ||||
| 	if (clk_ctrl <= DPU_CLK_CTRL_NONE || clk_ctrl >= DPU_CLK_CTRL_MAX) | ||||
| 		return false; | ||||
| 
 | ||||
| 	reg_off = mdp->caps->clk_ctrls[clk_ctrl].reg_off; | ||||
| 	bit_off = mdp->caps->clk_ctrls[clk_ctrl].bit_off; | ||||
| 
 | ||||
| 	reg_val = DPU_REG_READ(c, reg_off); | ||||
| 
 | ||||
| 	if (enable) | ||||
| 		new_val = reg_val | BIT(bit_off); | ||||
| 	else | ||||
| 		new_val = reg_val & ~BIT(bit_off); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, reg_off, new_val); | ||||
| 
 | ||||
| 	clk_forced_on = !(reg_val & BIT(bit_off)); | ||||
| 
 | ||||
| 	return clk_forced_on; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| static void dpu_hw_get_danger_status(struct dpu_hw_mdp *mdp, | ||||
| 		struct dpu_danger_safe_status *status) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 value; | ||||
| 
 | ||||
| 	if (!mdp || !status) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 
 | ||||
| 	value = DPU_REG_READ(c, DANGER_STATUS); | ||||
| 	status->mdp = (value >> 0) & 0x3; | ||||
| 	status->sspp[SSPP_VIG0] = (value >> 4) & 0x3; | ||||
| 	status->sspp[SSPP_VIG1] = (value >> 6) & 0x3; | ||||
| 	status->sspp[SSPP_VIG2] = (value >> 8) & 0x3; | ||||
| 	status->sspp[SSPP_VIG3] = (value >> 10) & 0x3; | ||||
| 	status->sspp[SSPP_RGB0] = (value >> 12) & 0x3; | ||||
| 	status->sspp[SSPP_RGB1] = (value >> 14) & 0x3; | ||||
| 	status->sspp[SSPP_RGB2] = (value >> 16) & 0x3; | ||||
| 	status->sspp[SSPP_RGB3] = (value >> 18) & 0x3; | ||||
| 	status->sspp[SSPP_DMA0] = (value >> 20) & 0x3; | ||||
| 	status->sspp[SSPP_DMA1] = (value >> 22) & 0x3; | ||||
| 	status->sspp[SSPP_DMA2] = (value >> 28) & 0x3; | ||||
| 	status->sspp[SSPP_DMA3] = (value >> 30) & 0x3; | ||||
| 	status->sspp[SSPP_CURSOR0] = (value >> 24) & 0x3; | ||||
| 	status->sspp[SSPP_CURSOR1] = (value >> 26) & 0x3; | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_setup_vsync_source(struct dpu_hw_mdp *mdp, | ||||
| 		struct dpu_vsync_source_cfg *cfg) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 reg, wd_load_value, wd_ctl, wd_ctl2, i; | ||||
| 	static const u32 pp_offset[PINGPONG_MAX] = {0xC, 0x8, 0x4, 0x13, 0x18}; | ||||
| 
 | ||||
| 	if (!mdp || !cfg || (cfg->pp_count > ARRAY_SIZE(cfg->ppnumber))) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 	reg = DPU_REG_READ(c, MDP_VSYNC_SEL); | ||||
| 	for (i = 0; i < cfg->pp_count; i++) { | ||||
| 		int pp_idx = cfg->ppnumber[i] - PINGPONG_0; | ||||
| 
 | ||||
| 		if (pp_idx >= ARRAY_SIZE(pp_offset)) | ||||
| 			continue; | ||||
| 
 | ||||
| 		reg &= ~(0xf << pp_offset[pp_idx]); | ||||
| 		reg |= (cfg->vsync_source & 0xf) << pp_offset[pp_idx]; | ||||
| 	} | ||||
| 	DPU_REG_WRITE(c, MDP_VSYNC_SEL, reg); | ||||
| 
 | ||||
| 	if (cfg->vsync_source >= DPU_VSYNC_SOURCE_WD_TIMER_4 && | ||||
| 			cfg->vsync_source <= DPU_VSYNC_SOURCE_WD_TIMER_0) { | ||||
| 		switch (cfg->vsync_source) { | ||||
| 		case DPU_VSYNC_SOURCE_WD_TIMER_4: | ||||
| 			wd_load_value = MDP_WD_TIMER_4_LOAD_VALUE; | ||||
| 			wd_ctl = MDP_WD_TIMER_4_CTL; | ||||
| 			wd_ctl2 = MDP_WD_TIMER_4_CTL2; | ||||
| 			break; | ||||
| 		case DPU_VSYNC_SOURCE_WD_TIMER_3: | ||||
| 			wd_load_value = MDP_WD_TIMER_3_LOAD_VALUE; | ||||
| 			wd_ctl = MDP_WD_TIMER_3_CTL; | ||||
| 			wd_ctl2 = MDP_WD_TIMER_3_CTL2; | ||||
| 			break; | ||||
| 		case DPU_VSYNC_SOURCE_WD_TIMER_2: | ||||
| 			wd_load_value = MDP_WD_TIMER_2_LOAD_VALUE; | ||||
| 			wd_ctl = MDP_WD_TIMER_2_CTL; | ||||
| 			wd_ctl2 = MDP_WD_TIMER_2_CTL2; | ||||
| 			break; | ||||
| 		case DPU_VSYNC_SOURCE_WD_TIMER_1: | ||||
| 			wd_load_value = MDP_WD_TIMER_1_LOAD_VALUE; | ||||
| 			wd_ctl = MDP_WD_TIMER_1_CTL; | ||||
| 			wd_ctl2 = MDP_WD_TIMER_1_CTL2; | ||||
| 			break; | ||||
| 		case DPU_VSYNC_SOURCE_WD_TIMER_0: | ||||
| 		default: | ||||
| 			wd_load_value = MDP_WD_TIMER_0_LOAD_VALUE; | ||||
| 			wd_ctl = MDP_WD_TIMER_0_CTL; | ||||
| 			wd_ctl2 = MDP_WD_TIMER_0_CTL2; | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		DPU_REG_WRITE(c, wd_load_value, | ||||
| 			CALCULATE_WD_LOAD_VALUE(cfg->frame_rate)); | ||||
| 
 | ||||
| 		DPU_REG_WRITE(c, wd_ctl, BIT(0)); /* clear timer */ | ||||
| 		reg = DPU_REG_READ(c, wd_ctl2); | ||||
| 		reg |= BIT(8);		/* enable heartbeat timer */ | ||||
| 		reg |= BIT(0);		/* enable WD timer */ | ||||
| 		DPU_REG_WRITE(c, wd_ctl2, reg); | ||||
| 
 | ||||
| 		/* make sure that timers are enabled/disabled for vsync state */ | ||||
| 		wmb(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_get_safe_status(struct dpu_hw_mdp *mdp, | ||||
| 		struct dpu_danger_safe_status *status) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 value; | ||||
| 
 | ||||
| 	if (!mdp || !status) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 
 | ||||
| 	value = DPU_REG_READ(c, SAFE_STATUS); | ||||
| 	status->mdp = (value >> 0) & 0x1; | ||||
| 	status->sspp[SSPP_VIG0] = (value >> 4) & 0x1; | ||||
| 	status->sspp[SSPP_VIG1] = (value >> 6) & 0x1; | ||||
| 	status->sspp[SSPP_VIG2] = (value >> 8) & 0x1; | ||||
| 	status->sspp[SSPP_VIG3] = (value >> 10) & 0x1; | ||||
| 	status->sspp[SSPP_RGB0] = (value >> 12) & 0x1; | ||||
| 	status->sspp[SSPP_RGB1] = (value >> 14) & 0x1; | ||||
| 	status->sspp[SSPP_RGB2] = (value >> 16) & 0x1; | ||||
| 	status->sspp[SSPP_RGB3] = (value >> 18) & 0x1; | ||||
| 	status->sspp[SSPP_DMA0] = (value >> 20) & 0x1; | ||||
| 	status->sspp[SSPP_DMA1] = (value >> 22) & 0x1; | ||||
| 	status->sspp[SSPP_DMA2] = (value >> 28) & 0x1; | ||||
| 	status->sspp[SSPP_DMA3] = (value >> 30) & 0x1; | ||||
| 	status->sspp[SSPP_CURSOR0] = (value >> 24) & 0x1; | ||||
| 	status->sspp[SSPP_CURSOR1] = (value >> 26) & 0x1; | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_reset_ubwc(struct dpu_hw_mdp *mdp, struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map c; | ||||
| 
 | ||||
| 	if (!mdp || !m) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (!IS_UBWC_20_SUPPORTED(m->caps->ubwc_version)) | ||||
| 		return; | ||||
| 
 | ||||
| 	/* force blk offset to zero to access beginning of register region */ | ||||
| 	c = mdp->hw; | ||||
| 	c.blk_off = 0x0; | ||||
| 	DPU_REG_WRITE(&c, UBWC_STATIC, m->mdp[0].ubwc_static); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_intf_audio_select(struct dpu_hw_mdp *mdp) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 
 | ||||
| 	if (!mdp) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &mdp->hw; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, HDMI_DP_CORE_SELECT, 0x1); | ||||
| } | ||||
| 
 | ||||
| static void _setup_mdp_ops(struct dpu_hw_mdp_ops *ops, | ||||
| 		unsigned long cap) | ||||
| { | ||||
| 	ops->setup_split_pipe = dpu_hw_setup_split_pipe; | ||||
| 	ops->setup_cdm_output = dpu_hw_setup_cdm_output; | ||||
| 	ops->setup_clk_force_ctrl = dpu_hw_setup_clk_force_ctrl; | ||||
| 	ops->get_danger_status = dpu_hw_get_danger_status; | ||||
| 	ops->setup_vsync_source = dpu_hw_setup_vsync_source; | ||||
| 	ops->get_safe_status = dpu_hw_get_safe_status; | ||||
| 	ops->reset_ubwc = dpu_hw_reset_ubwc; | ||||
| 	ops->intf_audio_select = dpu_hw_intf_audio_select; | ||||
| } | ||||
| 
 | ||||
| static const struct dpu_mdp_cfg *_top_offset(enum dpu_mdp mdp, | ||||
| 		const struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!m || !addr || !b) | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 
 | ||||
| 	for (i = 0; i < m->mdp_count; i++) { | ||||
| 		if (mdp == m->mdp[i].id) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->mdp[i].base; | ||||
| 			b->length = m->mdp[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_TOP; | ||||
| 			return &m->mdp[i]; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-EINVAL); | ||||
| } | ||||
| 
 | ||||
| static struct dpu_hw_blk_ops dpu_hw_ops = { | ||||
| 	.start = NULL, | ||||
| 	.stop = NULL, | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_mdp *dpu_hw_mdptop_init(enum dpu_mdp idx, | ||||
| 		void __iomem *addr, | ||||
| 		const struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_mdp *mdp; | ||||
| 	const struct dpu_mdp_cfg *cfg; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (!addr || !m) | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 
 | ||||
| 	mdp = kzalloc(sizeof(*mdp), GFP_KERNEL); | ||||
| 	if (!mdp) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _top_offset(idx, m, addr, &mdp->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(mdp); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Assign ops | ||||
| 	 */ | ||||
| 	mdp->idx = idx; | ||||
| 	mdp->caps = cfg; | ||||
| 	_setup_mdp_ops(&mdp->ops, mdp->caps->features); | ||||
| 
 | ||||
| 	rc = dpu_hw_blk_init(&mdp->base, DPU_HW_BLK_TOP, idx, &dpu_hw_ops); | ||||
| 	if (rc) { | ||||
| 		DPU_ERROR("failed to init hw blk %d\n", rc); | ||||
| 		goto blk_init_error; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_dbg_set_dpu_top_offset(mdp->hw.blk_off); | ||||
| 
 | ||||
| 	return mdp; | ||||
| 
 | ||||
| blk_init_error: | ||||
| 	kzfree(mdp); | ||||
| 
 | ||||
| 	return ERR_PTR(rc); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_mdp_destroy(struct dpu_hw_mdp *mdp) | ||||
| { | ||||
| 	if (mdp) | ||||
| 		dpu_hw_blk_destroy(&mdp->base); | ||||
| 	kfree(mdp); | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										202
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_top.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_top.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,202 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_TOP_H | ||||
| #define _DPU_HW_TOP_H | ||||
| 
 | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| #include "dpu_hw_blk.h" | ||||
| 
 | ||||
| struct dpu_hw_mdp; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct traffic_shaper_cfg: traffic shaper configuration | ||||
|  * @en        : enable/disable traffic shaper | ||||
|  * @rd_client : true if read client; false if write client | ||||
|  * @client_id : client identifier | ||||
|  * @bpc_denom : denominator of byte per clk | ||||
|  * @bpc_numer : numerator of byte per clk | ||||
|  */ | ||||
| struct traffic_shaper_cfg { | ||||
| 	bool en; | ||||
| 	bool rd_client; | ||||
| 	u32 client_id; | ||||
| 	u32 bpc_denom; | ||||
| 	u64 bpc_numer; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct split_pipe_cfg - pipe configuration for dual display panels | ||||
|  * @en        : Enable/disable dual pipe confguration | ||||
|  * @mode      : Panel interface mode | ||||
|  * @intf      : Interface id for main control path | ||||
|  * @split_flush_en: Allows both the paths to be flushed when master path is | ||||
|  *              flushed | ||||
|  */ | ||||
| struct split_pipe_cfg { | ||||
| 	bool en; | ||||
| 	enum dpu_intf_mode mode; | ||||
| 	enum dpu_intf intf; | ||||
| 	bool split_flush_en; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct cdm_output_cfg: output configuration for cdm | ||||
|  * @intf_en   : enable/disable interface output | ||||
|  */ | ||||
| struct cdm_output_cfg { | ||||
| 	bool intf_en; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_danger_safe_status: danger and safe status signals | ||||
|  * @mdp: top level status | ||||
|  * @sspp: source pipe status | ||||
|  */ | ||||
| struct dpu_danger_safe_status { | ||||
| 	u8 mdp; | ||||
| 	u8 sspp[SSPP_MAX]; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_vsync_source_cfg - configure vsync source and configure the | ||||
|  *                                    watchdog timers if required. | ||||
|  * @pp_count: number of ping pongs active | ||||
|  * @frame_rate: Display frame rate | ||||
|  * @ppnumber: ping pong index array | ||||
|  * @vsync_source: vsync source selection | ||||
|  */ | ||||
| struct dpu_vsync_source_cfg { | ||||
| 	u32 pp_count; | ||||
| 	u32 frame_rate; | ||||
| 	u32 ppnumber[PINGPONG_MAX]; | ||||
| 	u32 vsync_source; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_mdp_ops - interface to the MDP TOP Hw driver functions | ||||
|  * Assumption is these functions will be called after clocks are enabled. | ||||
|  * @setup_split_pipe : Programs the pipe control registers | ||||
|  * @setup_pp_split : Programs the pp split control registers | ||||
|  * @setup_cdm_output : programs cdm control | ||||
|  * @setup_traffic_shaper : programs traffic shaper control | ||||
|  */ | ||||
| struct dpu_hw_mdp_ops { | ||||
| 	/** setup_split_pipe() : Regsiters are not double buffered, thisk
 | ||||
| 	 * function should be called before timing control enable | ||||
| 	 * @mdp  : mdp top context driver | ||||
| 	 * @cfg  : upper and lower part of pipe configuration | ||||
| 	 */ | ||||
| 	void (*setup_split_pipe)(struct dpu_hw_mdp *mdp, | ||||
| 			struct split_pipe_cfg *p); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_cdm_output() : Setup selection control of the cdm data path | ||||
| 	 * @mdp  : mdp top context driver | ||||
| 	 * @cfg  : cdm output configuration | ||||
| 	 */ | ||||
| 	void (*setup_cdm_output)(struct dpu_hw_mdp *mdp, | ||||
| 			struct cdm_output_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_traffic_shaper() : Setup traffic shaper control | ||||
| 	 * @mdp  : mdp top context driver | ||||
| 	 * @cfg  : traffic shaper configuration | ||||
| 	 */ | ||||
| 	void (*setup_traffic_shaper)(struct dpu_hw_mdp *mdp, | ||||
| 			struct traffic_shaper_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_clk_force_ctrl - set clock force control | ||||
| 	 * @mdp: mdp top context driver | ||||
| 	 * @clk_ctrl: clock to be controlled | ||||
| 	 * @enable: force on enable | ||||
| 	 * @return: if the clock is forced-on by this function | ||||
| 	 */ | ||||
| 	bool (*setup_clk_force_ctrl)(struct dpu_hw_mdp *mdp, | ||||
| 			enum dpu_clk_ctrl_type clk_ctrl, bool enable); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_danger_status - get danger status | ||||
| 	 * @mdp: mdp top context driver | ||||
| 	 * @status: Pointer to danger safe status | ||||
| 	 */ | ||||
| 	void (*get_danger_status)(struct dpu_hw_mdp *mdp, | ||||
| 			struct dpu_danger_safe_status *status); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * setup_vsync_source - setup vsync source configuration details | ||||
| 	 * @mdp: mdp top context driver | ||||
| 	 * @cfg: vsync source selection configuration | ||||
| 	 */ | ||||
| 	void (*setup_vsync_source)(struct dpu_hw_mdp *mdp, | ||||
| 				struct dpu_vsync_source_cfg *cfg); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_safe_status - get safe status | ||||
| 	 * @mdp: mdp top context driver | ||||
| 	 * @status: Pointer to danger safe status | ||||
| 	 */ | ||||
| 	void (*get_safe_status)(struct dpu_hw_mdp *mdp, | ||||
| 			struct dpu_danger_safe_status *status); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * reset_ubwc - reset top level UBWC configuration | ||||
| 	 * @mdp: mdp top context driver | ||||
| 	 * @m: pointer to mdss catalog data | ||||
| 	 */ | ||||
| 	void (*reset_ubwc)(struct dpu_hw_mdp *mdp, struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * intf_audio_select - select the external interface for audio | ||||
| 	 * @mdp: mdp top context driver | ||||
| 	 */ | ||||
| 	void (*intf_audio_select)(struct dpu_hw_mdp *mdp); | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_mdp { | ||||
| 	struct dpu_hw_blk base; | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* top */ | ||||
| 	enum dpu_mdp idx; | ||||
| 	const struct dpu_mdp_cfg *caps; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_mdp_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * to_dpu_hw_mdp - convert base object dpu_hw_base to container | ||||
|  * @hw: Pointer to base hardware block | ||||
|  * return: Pointer to hardware block container | ||||
|  */ | ||||
| static inline struct dpu_hw_mdp *to_dpu_hw_mdp(struct dpu_hw_blk *hw) | ||||
| { | ||||
| 	return container_of(hw, struct dpu_hw_mdp, base); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_mdptop_init - initializes the top driver for the passed idx | ||||
|  * @idx:  Interface index for which driver object is required | ||||
|  * @addr: Mapped register io address of MDP | ||||
|  * @m:    Pointer to mdss catalog data | ||||
|  */ | ||||
| struct dpu_hw_mdp *dpu_hw_mdptop_init(enum dpu_mdp idx, | ||||
| 		void __iomem *addr, | ||||
| 		const struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| void dpu_hw_mdp_destroy(struct dpu_hw_mdp *mdp); | ||||
| 
 | ||||
| #endif /*_DPU_HW_TOP_H */ | ||||
							
								
								
									
										452
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_util.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										452
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_util.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,452 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include "msm_drv.h" | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| 
 | ||||
| /* using a file static variables for debugfs access */ | ||||
| static u32 dpu_hw_util_log_mask = DPU_DBG_MASK_NONE; | ||||
| 
 | ||||
| /* DPU_SCALER_QSEED3 */ | ||||
| #define QSEED3_HW_VERSION                  0x00 | ||||
| #define QSEED3_OP_MODE                     0x04 | ||||
| #define QSEED3_RGB2Y_COEFF                 0x08 | ||||
| #define QSEED3_PHASE_INIT                  0x0C | ||||
| #define QSEED3_PHASE_STEP_Y_H              0x10 | ||||
| #define QSEED3_PHASE_STEP_Y_V              0x14 | ||||
| #define QSEED3_PHASE_STEP_UV_H             0x18 | ||||
| #define QSEED3_PHASE_STEP_UV_V             0x1C | ||||
| #define QSEED3_PRELOAD                     0x20 | ||||
| #define QSEED3_DE_SHARPEN                  0x24 | ||||
| #define QSEED3_DE_SHARPEN_CTL              0x28 | ||||
| #define QSEED3_DE_SHAPE_CTL                0x2C | ||||
| #define QSEED3_DE_THRESHOLD                0x30 | ||||
| #define QSEED3_DE_ADJUST_DATA_0            0x34 | ||||
| #define QSEED3_DE_ADJUST_DATA_1            0x38 | ||||
| #define QSEED3_DE_ADJUST_DATA_2            0x3C | ||||
| #define QSEED3_SRC_SIZE_Y_RGB_A            0x40 | ||||
| #define QSEED3_SRC_SIZE_UV                 0x44 | ||||
| #define QSEED3_DST_SIZE                    0x48 | ||||
| #define QSEED3_COEF_LUT_CTRL               0x4C | ||||
| #define QSEED3_COEF_LUT_SWAP_BIT           0 | ||||
| #define QSEED3_COEF_LUT_DIR_BIT            1 | ||||
| #define QSEED3_COEF_LUT_Y_CIR_BIT          2 | ||||
| #define QSEED3_COEF_LUT_UV_CIR_BIT         3 | ||||
| #define QSEED3_COEF_LUT_Y_SEP_BIT          4 | ||||
| #define QSEED3_COEF_LUT_UV_SEP_BIT         5 | ||||
| #define QSEED3_BUFFER_CTRL                 0x50 | ||||
| #define QSEED3_CLK_CTRL0                   0x54 | ||||
| #define QSEED3_CLK_CTRL1                   0x58 | ||||
| #define QSEED3_CLK_STATUS                  0x5C | ||||
| #define QSEED3_MISR_CTRL                   0x70 | ||||
| #define QSEED3_MISR_SIGNATURE_0            0x74 | ||||
| #define QSEED3_MISR_SIGNATURE_1            0x78 | ||||
| #define QSEED3_PHASE_INIT_Y_H              0x90 | ||||
| #define QSEED3_PHASE_INIT_Y_V              0x94 | ||||
| #define QSEED3_PHASE_INIT_UV_H             0x98 | ||||
| #define QSEED3_PHASE_INIT_UV_V             0x9C | ||||
| #define QSEED3_COEF_LUT                    0x100 | ||||
| #define QSEED3_FILTERS                     5 | ||||
| #define QSEED3_LUT_REGIONS                 4 | ||||
| #define QSEED3_CIRCULAR_LUTS               9 | ||||
| #define QSEED3_SEPARABLE_LUTS              10 | ||||
| #define QSEED3_LUT_SIZE                    60 | ||||
| #define QSEED3_ENABLE                      2 | ||||
| #define QSEED3_DIR_LUT_SIZE                (200 * sizeof(u32)) | ||||
| #define QSEED3_CIR_LUT_SIZE \ | ||||
| 	(QSEED3_LUT_SIZE * QSEED3_CIRCULAR_LUTS * sizeof(u32)) | ||||
| #define QSEED3_SEP_LUT_SIZE \ | ||||
| 	(QSEED3_LUT_SIZE * QSEED3_SEPARABLE_LUTS * sizeof(u32)) | ||||
| 
 | ||||
| void dpu_reg_write(struct dpu_hw_blk_reg_map *c, | ||||
| 		u32 reg_off, | ||||
| 		u32 val, | ||||
| 		const char *name) | ||||
| { | ||||
| 	/* don't need to mutex protect this */ | ||||
| 	if (c->log_mask & dpu_hw_util_log_mask) | ||||
| 		DPU_DEBUG_DRIVER("[%s:0x%X] <= 0x%X\n", | ||||
| 				name, c->blk_off + reg_off, val); | ||||
| 	writel_relaxed(val, c->base_off + c->blk_off + reg_off); | ||||
| } | ||||
| 
 | ||||
| int dpu_reg_read(struct dpu_hw_blk_reg_map *c, u32 reg_off) | ||||
| { | ||||
| 	return readl_relaxed(c->base_off + c->blk_off + reg_off); | ||||
| } | ||||
| 
 | ||||
| u32 *dpu_hw_util_get_log_mask_ptr(void) | ||||
| { | ||||
| 	return &dpu_hw_util_log_mask; | ||||
| } | ||||
| 
 | ||||
| void dpu_set_scaler_v2(struct dpu_hw_scaler3_cfg *cfg, | ||||
| 		const struct dpu_drm_scaler_v2 *scale_v2) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	cfg->enable = scale_v2->enable; | ||||
| 	cfg->dir_en = scale_v2->dir_en; | ||||
| 
 | ||||
| 	for (i = 0; i < DPU_MAX_PLANES; i++) { | ||||
| 		cfg->init_phase_x[i] = scale_v2->init_phase_x[i]; | ||||
| 		cfg->phase_step_x[i] = scale_v2->phase_step_x[i]; | ||||
| 		cfg->init_phase_y[i] = scale_v2->init_phase_y[i]; | ||||
| 		cfg->phase_step_y[i] = scale_v2->phase_step_y[i]; | ||||
| 
 | ||||
| 		cfg->preload_x[i] = scale_v2->preload_x[i]; | ||||
| 		cfg->preload_y[i] = scale_v2->preload_y[i]; | ||||
| 		cfg->src_width[i] = scale_v2->src_width[i]; | ||||
| 		cfg->src_height[i] = scale_v2->src_height[i]; | ||||
| 	} | ||||
| 
 | ||||
| 	cfg->dst_width = scale_v2->dst_width; | ||||
| 	cfg->dst_height = scale_v2->dst_height; | ||||
| 
 | ||||
| 	cfg->y_rgb_filter_cfg = scale_v2->y_rgb_filter_cfg; | ||||
| 	cfg->uv_filter_cfg = scale_v2->uv_filter_cfg; | ||||
| 	cfg->alpha_filter_cfg = scale_v2->alpha_filter_cfg; | ||||
| 	cfg->blend_cfg = scale_v2->blend_cfg; | ||||
| 
 | ||||
| 	cfg->lut_flag = scale_v2->lut_flag; | ||||
| 	cfg->dir_lut_idx = scale_v2->dir_lut_idx; | ||||
| 	cfg->y_rgb_cir_lut_idx = scale_v2->y_rgb_cir_lut_idx; | ||||
| 	cfg->uv_cir_lut_idx = scale_v2->uv_cir_lut_idx; | ||||
| 	cfg->y_rgb_sep_lut_idx = scale_v2->y_rgb_sep_lut_idx; | ||||
| 	cfg->uv_sep_lut_idx = scale_v2->uv_sep_lut_idx; | ||||
| 
 | ||||
| 	cfg->de.enable = scale_v2->de.enable; | ||||
| 	cfg->de.sharpen_level1 = scale_v2->de.sharpen_level1; | ||||
| 	cfg->de.sharpen_level2 = scale_v2->de.sharpen_level2; | ||||
| 	cfg->de.clip = scale_v2->de.clip; | ||||
| 	cfg->de.limit = scale_v2->de.limit; | ||||
| 	cfg->de.thr_quiet = scale_v2->de.thr_quiet; | ||||
| 	cfg->de.thr_dieout = scale_v2->de.thr_dieout; | ||||
| 	cfg->de.thr_low = scale_v2->de.thr_low; | ||||
| 	cfg->de.thr_high = scale_v2->de.thr_high; | ||||
| 	cfg->de.prec_shift = scale_v2->de.prec_shift; | ||||
| 
 | ||||
| 	for (i = 0; i < DPU_MAX_DE_CURVES; i++) { | ||||
| 		cfg->de.adjust_a[i] = scale_v2->de.adjust_a[i]; | ||||
| 		cfg->de.adjust_b[i] = scale_v2->de.adjust_b[i]; | ||||
| 		cfg->de.adjust_c[i] = scale_v2->de.adjust_c[i]; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static void _dpu_hw_setup_scaler3_lut(struct dpu_hw_blk_reg_map *c, | ||||
| 		struct dpu_hw_scaler3_cfg *scaler3_cfg, u32 offset) | ||||
| { | ||||
| 	int i, j, filter; | ||||
| 	int config_lut = 0x0; | ||||
| 	unsigned long lut_flags; | ||||
| 	u32 lut_addr, lut_offset, lut_len; | ||||
| 	u32 *lut[QSEED3_FILTERS] = {NULL, NULL, NULL, NULL, NULL}; | ||||
| 	static const uint32_t off_tbl[QSEED3_FILTERS][QSEED3_LUT_REGIONS][2] = { | ||||
| 		{{18, 0x000}, {12, 0x120}, {12, 0x1E0}, {8, 0x2A0} }, | ||||
| 		{{6, 0x320}, {3, 0x3E0}, {3, 0x440}, {3, 0x4A0} }, | ||||
| 		{{6, 0x500}, {3, 0x5c0}, {3, 0x620}, {3, 0x680} }, | ||||
| 		{{6, 0x380}, {3, 0x410}, {3, 0x470}, {3, 0x4d0} }, | ||||
| 		{{6, 0x560}, {3, 0x5f0}, {3, 0x650}, {3, 0x6b0} }, | ||||
| 	}; | ||||
| 
 | ||||
| 	lut_flags = (unsigned long) scaler3_cfg->lut_flag; | ||||
| 	if (test_bit(QSEED3_COEF_LUT_DIR_BIT, &lut_flags) && | ||||
| 		(scaler3_cfg->dir_len == QSEED3_DIR_LUT_SIZE)) { | ||||
| 		lut[0] = scaler3_cfg->dir_lut; | ||||
| 		config_lut = 1; | ||||
| 	} | ||||
| 	if (test_bit(QSEED3_COEF_LUT_Y_CIR_BIT, &lut_flags) && | ||||
| 		(scaler3_cfg->y_rgb_cir_lut_idx < QSEED3_CIRCULAR_LUTS) && | ||||
| 		(scaler3_cfg->cir_len == QSEED3_CIR_LUT_SIZE)) { | ||||
| 		lut[1] = scaler3_cfg->cir_lut + | ||||
| 			scaler3_cfg->y_rgb_cir_lut_idx * QSEED3_LUT_SIZE; | ||||
| 		config_lut = 1; | ||||
| 	} | ||||
| 	if (test_bit(QSEED3_COEF_LUT_UV_CIR_BIT, &lut_flags) && | ||||
| 		(scaler3_cfg->uv_cir_lut_idx < QSEED3_CIRCULAR_LUTS) && | ||||
| 		(scaler3_cfg->cir_len == QSEED3_CIR_LUT_SIZE)) { | ||||
| 		lut[2] = scaler3_cfg->cir_lut + | ||||
| 			scaler3_cfg->uv_cir_lut_idx * QSEED3_LUT_SIZE; | ||||
| 		config_lut = 1; | ||||
| 	} | ||||
| 	if (test_bit(QSEED3_COEF_LUT_Y_SEP_BIT, &lut_flags) && | ||||
| 		(scaler3_cfg->y_rgb_sep_lut_idx < QSEED3_SEPARABLE_LUTS) && | ||||
| 		(scaler3_cfg->sep_len == QSEED3_SEP_LUT_SIZE)) { | ||||
| 		lut[3] = scaler3_cfg->sep_lut + | ||||
| 			scaler3_cfg->y_rgb_sep_lut_idx * QSEED3_LUT_SIZE; | ||||
| 		config_lut = 1; | ||||
| 	} | ||||
| 	if (test_bit(QSEED3_COEF_LUT_UV_SEP_BIT, &lut_flags) && | ||||
| 		(scaler3_cfg->uv_sep_lut_idx < QSEED3_SEPARABLE_LUTS) && | ||||
| 		(scaler3_cfg->sep_len == QSEED3_SEP_LUT_SIZE)) { | ||||
| 		lut[4] = scaler3_cfg->sep_lut + | ||||
| 			scaler3_cfg->uv_sep_lut_idx * QSEED3_LUT_SIZE; | ||||
| 		config_lut = 1; | ||||
| 	} | ||||
| 
 | ||||
| 	if (config_lut) { | ||||
| 		for (filter = 0; filter < QSEED3_FILTERS; filter++) { | ||||
| 			if (!lut[filter]) | ||||
| 				continue; | ||||
| 			lut_offset = 0; | ||||
| 			for (i = 0; i < QSEED3_LUT_REGIONS; i++) { | ||||
| 				lut_addr = QSEED3_COEF_LUT + offset | ||||
| 					+ off_tbl[filter][i][1]; | ||||
| 				lut_len = off_tbl[filter][i][0] << 2; | ||||
| 				for (j = 0; j < lut_len; j++) { | ||||
| 					DPU_REG_WRITE(c, | ||||
| 						lut_addr, | ||||
| 						(lut[filter])[lut_offset++]); | ||||
| 					lut_addr += 4; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if (test_bit(QSEED3_COEF_LUT_SWAP_BIT, &lut_flags)) | ||||
| 		DPU_REG_WRITE(c, QSEED3_COEF_LUT_CTRL + offset, BIT(0)); | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| static void _dpu_hw_setup_scaler3_de(struct dpu_hw_blk_reg_map *c, | ||||
| 		struct dpu_hw_scaler3_de_cfg *de_cfg, u32 offset) | ||||
| { | ||||
| 	u32 sharp_lvl, sharp_ctl, shape_ctl, de_thr; | ||||
| 	u32 adjust_a, adjust_b, adjust_c; | ||||
| 
 | ||||
| 	if (!de_cfg->enable) | ||||
| 		return; | ||||
| 
 | ||||
| 	sharp_lvl = (de_cfg->sharpen_level1 & 0x1FF) | | ||||
| 		((de_cfg->sharpen_level2 & 0x1FF) << 16); | ||||
| 
 | ||||
| 	sharp_ctl = ((de_cfg->limit & 0xF) << 9) | | ||||
| 		((de_cfg->prec_shift & 0x7) << 13) | | ||||
| 		((de_cfg->clip & 0x7) << 16); | ||||
| 
 | ||||
| 	shape_ctl = (de_cfg->thr_quiet & 0xFF) | | ||||
| 		((de_cfg->thr_dieout & 0x3FF) << 16); | ||||
| 
 | ||||
| 	de_thr = (de_cfg->thr_low & 0x3FF) | | ||||
| 		((de_cfg->thr_high & 0x3FF) << 16); | ||||
| 
 | ||||
| 	adjust_a = (de_cfg->adjust_a[0] & 0x3FF) | | ||||
| 		((de_cfg->adjust_a[1] & 0x3FF) << 10) | | ||||
| 		((de_cfg->adjust_a[2] & 0x3FF) << 20); | ||||
| 
 | ||||
| 	adjust_b = (de_cfg->adjust_b[0] & 0x3FF) | | ||||
| 		((de_cfg->adjust_b[1] & 0x3FF) << 10) | | ||||
| 		((de_cfg->adjust_b[2] & 0x3FF) << 20); | ||||
| 
 | ||||
| 	adjust_c = (de_cfg->adjust_c[0] & 0x3FF) | | ||||
| 		((de_cfg->adjust_c[1] & 0x3FF) << 10) | | ||||
| 		((de_cfg->adjust_c[2] & 0x3FF) << 20); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_SHARPEN + offset, sharp_lvl); | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_SHARPEN_CTL + offset, sharp_ctl); | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_SHAPE_CTL + offset, shape_ctl); | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_THRESHOLD + offset, de_thr); | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_ADJUST_DATA_0 + offset, adjust_a); | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_ADJUST_DATA_1 + offset, adjust_b); | ||||
| 	DPU_REG_WRITE(c, QSEED3_DE_ADJUST_DATA_2 + offset, adjust_c); | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_setup_scaler3(struct dpu_hw_blk_reg_map *c, | ||||
| 		struct dpu_hw_scaler3_cfg *scaler3_cfg, | ||||
| 		u32 scaler_offset, u32 scaler_version, | ||||
| 		const struct dpu_format *format) | ||||
| { | ||||
| 	u32 op_mode = 0; | ||||
| 	u32 phase_init, preload, src_y_rgb, src_uv, dst; | ||||
| 
 | ||||
| 	if (!scaler3_cfg->enable) | ||||
| 		goto end; | ||||
| 
 | ||||
| 	op_mode |= BIT(0); | ||||
| 	op_mode |= (scaler3_cfg->y_rgb_filter_cfg & 0x3) << 16; | ||||
| 
 | ||||
| 	if (format && DPU_FORMAT_IS_YUV(format)) { | ||||
| 		op_mode |= BIT(12); | ||||
| 		op_mode |= (scaler3_cfg->uv_filter_cfg & 0x3) << 24; | ||||
| 	} | ||||
| 
 | ||||
| 	op_mode |= (scaler3_cfg->blend_cfg & 1) << 31; | ||||
| 	op_mode |= (scaler3_cfg->dir_en) ? BIT(4) : 0; | ||||
| 
 | ||||
| 	preload = | ||||
| 		((scaler3_cfg->preload_x[0] & 0x7F) << 0) | | ||||
| 		((scaler3_cfg->preload_y[0] & 0x7F) << 8) | | ||||
| 		((scaler3_cfg->preload_x[1] & 0x7F) << 16) | | ||||
| 		((scaler3_cfg->preload_y[1] & 0x7F) << 24); | ||||
| 
 | ||||
| 	src_y_rgb = (scaler3_cfg->src_width[0] & 0x1FFFF) | | ||||
| 		((scaler3_cfg->src_height[0] & 0x1FFFF) << 16); | ||||
| 
 | ||||
| 	src_uv = (scaler3_cfg->src_width[1] & 0x1FFFF) | | ||||
| 		((scaler3_cfg->src_height[1] & 0x1FFFF) << 16); | ||||
| 
 | ||||
| 	dst = (scaler3_cfg->dst_width & 0x1FFFF) | | ||||
| 		((scaler3_cfg->dst_height & 0x1FFFF) << 16); | ||||
| 
 | ||||
| 	if (scaler3_cfg->de.enable) { | ||||
| 		_dpu_hw_setup_scaler3_de(c, &scaler3_cfg->de, scaler_offset); | ||||
| 		op_mode |= BIT(8); | ||||
| 	} | ||||
| 
 | ||||
| 	if (scaler3_cfg->lut_flag) | ||||
| 		_dpu_hw_setup_scaler3_lut(c, scaler3_cfg, | ||||
| 								scaler_offset); | ||||
| 
 | ||||
| 	if (scaler_version == 0x1002) { | ||||
| 		phase_init = | ||||
| 			((scaler3_cfg->init_phase_x[0] & 0x3F) << 0) | | ||||
| 			((scaler3_cfg->init_phase_y[0] & 0x3F) << 8) | | ||||
| 			((scaler3_cfg->init_phase_x[1] & 0x3F) << 16) | | ||||
| 			((scaler3_cfg->init_phase_y[1] & 0x3F) << 24); | ||||
| 		DPU_REG_WRITE(c, QSEED3_PHASE_INIT + scaler_offset, phase_init); | ||||
| 	} else { | ||||
| 		DPU_REG_WRITE(c, QSEED3_PHASE_INIT_Y_H + scaler_offset, | ||||
| 			scaler3_cfg->init_phase_x[0] & 0x1FFFFF); | ||||
| 		DPU_REG_WRITE(c, QSEED3_PHASE_INIT_Y_V + scaler_offset, | ||||
| 			scaler3_cfg->init_phase_y[0] & 0x1FFFFF); | ||||
| 		DPU_REG_WRITE(c, QSEED3_PHASE_INIT_UV_H + scaler_offset, | ||||
| 			scaler3_cfg->init_phase_x[1] & 0x1FFFFF); | ||||
| 		DPU_REG_WRITE(c, QSEED3_PHASE_INIT_UV_V + scaler_offset, | ||||
| 			scaler3_cfg->init_phase_y[1] & 0x1FFFFF); | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_PHASE_STEP_Y_H + scaler_offset, | ||||
| 		scaler3_cfg->phase_step_x[0] & 0xFFFFFF); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_PHASE_STEP_Y_V + scaler_offset, | ||||
| 		scaler3_cfg->phase_step_y[0] & 0xFFFFFF); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_PHASE_STEP_UV_H + scaler_offset, | ||||
| 		scaler3_cfg->phase_step_x[1] & 0xFFFFFF); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_PHASE_STEP_UV_V + scaler_offset, | ||||
| 		scaler3_cfg->phase_step_y[1] & 0xFFFFFF); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_PRELOAD + scaler_offset, preload); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_SRC_SIZE_Y_RGB_A + scaler_offset, src_y_rgb); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_SRC_SIZE_UV + scaler_offset, src_uv); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_DST_SIZE + scaler_offset, dst); | ||||
| 
 | ||||
| end: | ||||
| 	if (format && !DPU_FORMAT_IS_DX(format)) | ||||
| 		op_mode |= BIT(14); | ||||
| 
 | ||||
| 	if (format && format->alpha_enable) { | ||||
| 		op_mode |= BIT(10); | ||||
| 		if (scaler_version == 0x1002) | ||||
| 			op_mode |= (scaler3_cfg->alpha_filter_cfg & 0x1) << 30; | ||||
| 		else | ||||
| 			op_mode |= (scaler3_cfg->alpha_filter_cfg & 0x3) << 29; | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, QSEED3_OP_MODE + scaler_offset, op_mode); | ||||
| } | ||||
| 
 | ||||
| u32 dpu_hw_get_scaler3_ver(struct dpu_hw_blk_reg_map *c, | ||||
| 			u32 scaler_offset) | ||||
| { | ||||
| 	return DPU_REG_READ(c, QSEED3_HW_VERSION + scaler_offset); | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_csc_setup(struct dpu_hw_blk_reg_map *c, | ||||
| 		u32 csc_reg_off, | ||||
| 		struct dpu_csc_cfg *data, bool csc10) | ||||
| { | ||||
| 	static const u32 matrix_shift = 7; | ||||
| 	u32 clamp_shift = csc10 ? 16 : 8; | ||||
| 	u32 val; | ||||
| 
 | ||||
| 	/* matrix coeff - convert S15.16 to S4.9 */ | ||||
| 	val = ((data->csc_mv[0] >> matrix_shift) & 0x1FFF) | | ||||
| 		(((data->csc_mv[1] >> matrix_shift) & 0x1FFF) << 16); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off, val); | ||||
| 	val = ((data->csc_mv[2] >> matrix_shift) & 0x1FFF) | | ||||
| 		(((data->csc_mv[3] >> matrix_shift) & 0x1FFF) << 16); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x4, val); | ||||
| 	val = ((data->csc_mv[4] >> matrix_shift) & 0x1FFF) | | ||||
| 		(((data->csc_mv[5] >> matrix_shift) & 0x1FFF) << 16); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x8, val); | ||||
| 	val = ((data->csc_mv[6] >> matrix_shift) & 0x1FFF) | | ||||
| 		(((data->csc_mv[7] >> matrix_shift) & 0x1FFF) << 16); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0xc, val); | ||||
| 	val = (data->csc_mv[8] >> matrix_shift) & 0x1FFF; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x10, val); | ||||
| 
 | ||||
| 	/* Pre clamp */ | ||||
| 	val = (data->csc_pre_lv[0] << clamp_shift) | data->csc_pre_lv[1]; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x14, val); | ||||
| 	val = (data->csc_pre_lv[2] << clamp_shift) | data->csc_pre_lv[3]; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x18, val); | ||||
| 	val = (data->csc_pre_lv[4] << clamp_shift) | data->csc_pre_lv[5]; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x1c, val); | ||||
| 
 | ||||
| 	/* Post clamp */ | ||||
| 	val = (data->csc_post_lv[0] << clamp_shift) | data->csc_post_lv[1]; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x20, val); | ||||
| 	val = (data->csc_post_lv[2] << clamp_shift) | data->csc_post_lv[3]; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x24, val); | ||||
| 	val = (data->csc_post_lv[4] << clamp_shift) | data->csc_post_lv[5]; | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x28, val); | ||||
| 
 | ||||
| 	/* Pre-Bias */ | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x2c, data->csc_pre_bv[0]); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x30, data->csc_pre_bv[1]); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x34, data->csc_pre_bv[2]); | ||||
| 
 | ||||
| 	/* Post-Bias */ | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x38, data->csc_post_bv[0]); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x3c, data->csc_post_bv[1]); | ||||
| 	DPU_REG_WRITE(c, csc_reg_off + 0x40, data->csc_post_bv[2]); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * _dpu_copy_formats   - copy formats from src_list to dst_list | ||||
|  * @dst_list:          pointer to destination list where to copy formats | ||||
|  * @dst_list_size:     size of destination list | ||||
|  * @dst_list_pos:      starting position on the list where to copy formats | ||||
|  * @src_list:          pointer to source list where to copy formats from | ||||
|  * @src_list_size:     size of source list | ||||
|  * Return: number of elements populated | ||||
|  */ | ||||
| uint32_t dpu_copy_formats( | ||||
| 		struct dpu_format_extended *dst_list, | ||||
| 		uint32_t dst_list_size, | ||||
| 		uint32_t dst_list_pos, | ||||
| 		const struct dpu_format_extended *src_list, | ||||
| 		uint32_t src_list_size) | ||||
| { | ||||
| 	uint32_t cur_pos, i; | ||||
| 
 | ||||
| 	if (!dst_list || !src_list || (dst_list_pos >= (dst_list_size - 1))) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	for (i = 0, cur_pos = dst_list_pos; | ||||
| 		(cur_pos < (dst_list_size - 1)) && (i < src_list_size) | ||||
| 		&& src_list[i].fourcc_format; ++i, ++cur_pos) | ||||
| 		dst_list[cur_pos] = src_list[i]; | ||||
| 
 | ||||
| 	dst_list[cur_pos].fourcc_format = 0; | ||||
| 
 | ||||
| 	return i; | ||||
| } | ||||
							
								
								
									
										358
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_util.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										358
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_util.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,358 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_UTIL_H | ||||
| #define _DPU_HW_UTIL_H | ||||
| 
 | ||||
| #include <linux/io.h> | ||||
| #include <linux/slab.h> | ||||
| #include "dpu_hw_mdss.h" | ||||
| 
 | ||||
| #define REG_MASK(n)                     ((BIT(n)) - 1) | ||||
| struct dpu_format_extended; | ||||
| 
 | ||||
| /*
 | ||||
|  * This is the common struct maintained by each sub block | ||||
|  * for mapping the register offsets in this block to the | ||||
|  * absoulute IO address | ||||
|  * @base_off:     mdp register mapped offset | ||||
|  * @blk_off:      pipe offset relative to mdss offset | ||||
|  * @length        length of register block offset | ||||
|  * @xin_id        xin id | ||||
|  * @hwversion     mdss hw version number | ||||
|  */ | ||||
| struct dpu_hw_blk_reg_map { | ||||
| 	void __iomem *base_off; | ||||
| 	u32 blk_off; | ||||
| 	u32 length; | ||||
| 	u32 xin_id; | ||||
| 	u32 hwversion; | ||||
| 	u32 log_mask; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_scaler3_de_cfg : QSEEDv3 detail enhancer configuration | ||||
|  * @enable:         detail enhancer enable/disable | ||||
|  * @sharpen_level1: sharpening strength for noise | ||||
|  * @sharpen_level2: sharpening strength for signal | ||||
|  * @ clip:          clip shift | ||||
|  * @ limit:         limit value | ||||
|  * @ thr_quiet:     quiet threshold | ||||
|  * @ thr_dieout:    dieout threshold | ||||
|  * @ thr_high:      low threshold | ||||
|  * @ thr_high:      high threshold | ||||
|  * @ prec_shift:    precision shift | ||||
|  * @ adjust_a:      A-coefficients for mapping curve | ||||
|  * @ adjust_b:      B-coefficients for mapping curve | ||||
|  * @ adjust_c:      C-coefficients for mapping curve | ||||
|  */ | ||||
| struct dpu_hw_scaler3_de_cfg { | ||||
| 	u32 enable; | ||||
| 	int16_t sharpen_level1; | ||||
| 	int16_t sharpen_level2; | ||||
| 	uint16_t clip; | ||||
| 	uint16_t limit; | ||||
| 	uint16_t thr_quiet; | ||||
| 	uint16_t thr_dieout; | ||||
| 	uint16_t thr_low; | ||||
| 	uint16_t thr_high; | ||||
| 	uint16_t prec_shift; | ||||
| 	int16_t adjust_a[DPU_MAX_DE_CURVES]; | ||||
| 	int16_t adjust_b[DPU_MAX_DE_CURVES]; | ||||
| 	int16_t adjust_c[DPU_MAX_DE_CURVES]; | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_scaler3_cfg : QSEEDv3 configuration | ||||
|  * @enable:        scaler enable | ||||
|  * @dir_en:        direction detection block enable | ||||
|  * @ init_phase_x: horizontal initial phase | ||||
|  * @ phase_step_x: horizontal phase step | ||||
|  * @ init_phase_y: vertical initial phase | ||||
|  * @ phase_step_y: vertical phase step | ||||
|  * @ preload_x:    horizontal preload value | ||||
|  * @ preload_y:    vertical preload value | ||||
|  * @ src_width:    source width | ||||
|  * @ src_height:   source height | ||||
|  * @ dst_width:    destination width | ||||
|  * @ dst_height:   destination height | ||||
|  * @ y_rgb_filter_cfg: y/rgb plane filter configuration | ||||
|  * @ uv_filter_cfg: uv plane filter configuration | ||||
|  * @ alpha_filter_cfg: alpha filter configuration | ||||
|  * @ blend_cfg:    blend coefficients configuration | ||||
|  * @ lut_flag:     scaler LUT update flags | ||||
|  *                 0x1 swap LUT bank | ||||
|  *                 0x2 update 2D filter LUT | ||||
|  *                 0x4 update y circular filter LUT | ||||
|  *                 0x8 update uv circular filter LUT | ||||
|  *                 0x10 update y separable filter LUT | ||||
|  *                 0x20 update uv separable filter LUT | ||||
|  * @ dir_lut_idx:  2D filter LUT index | ||||
|  * @ y_rgb_cir_lut_idx: y circular filter LUT index | ||||
|  * @ uv_cir_lut_idx: uv circular filter LUT index | ||||
|  * @ y_rgb_sep_lut_idx: y circular filter LUT index | ||||
|  * @ uv_sep_lut_idx: uv separable filter LUT index | ||||
|  * @ dir_lut:      pointer to 2D LUT | ||||
|  * @ cir_lut:      pointer to circular filter LUT | ||||
|  * @ sep_lut:      pointer to separable filter LUT | ||||
|  * @ de: detail enhancer configuration | ||||
|  */ | ||||
| struct dpu_hw_scaler3_cfg { | ||||
| 	u32 enable; | ||||
| 	u32 dir_en; | ||||
| 	int32_t init_phase_x[DPU_MAX_PLANES]; | ||||
| 	int32_t phase_step_x[DPU_MAX_PLANES]; | ||||
| 	int32_t init_phase_y[DPU_MAX_PLANES]; | ||||
| 	int32_t phase_step_y[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	u32 preload_x[DPU_MAX_PLANES]; | ||||
| 	u32 preload_y[DPU_MAX_PLANES]; | ||||
| 	u32 src_width[DPU_MAX_PLANES]; | ||||
| 	u32 src_height[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	u32 dst_width; | ||||
| 	u32 dst_height; | ||||
| 
 | ||||
| 	u32 y_rgb_filter_cfg; | ||||
| 	u32 uv_filter_cfg; | ||||
| 	u32 alpha_filter_cfg; | ||||
| 	u32 blend_cfg; | ||||
| 
 | ||||
| 	u32 lut_flag; | ||||
| 	u32 dir_lut_idx; | ||||
| 
 | ||||
| 	u32 y_rgb_cir_lut_idx; | ||||
| 	u32 uv_cir_lut_idx; | ||||
| 	u32 y_rgb_sep_lut_idx; | ||||
| 	u32 uv_sep_lut_idx; | ||||
| 	u32 *dir_lut; | ||||
| 	size_t dir_len; | ||||
| 	u32 *cir_lut; | ||||
| 	size_t cir_len; | ||||
| 	u32 *sep_lut; | ||||
| 	size_t sep_len; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Detail enhancer settings | ||||
| 	 */ | ||||
| 	struct dpu_hw_scaler3_de_cfg de; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_scaler3_lut_cfg { | ||||
| 	bool is_configured; | ||||
| 	u32 *dir_lut; | ||||
| 	size_t dir_len; | ||||
| 	u32 *cir_lut; | ||||
| 	size_t cir_len; | ||||
| 	u32 *sep_lut; | ||||
| 	size_t sep_len; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_drm_pix_ext_v1 - version 1 of pixel ext structure | ||||
|  * @num_ext_pxls_lr: Number of total horizontal pixels | ||||
|  * @num_ext_pxls_tb: Number of total vertical lines | ||||
|  * @left_ftch:       Number of extra pixels to overfetch from left | ||||
|  * @right_ftch:      Number of extra pixels to overfetch from right | ||||
|  * @top_ftch:        Number of extra lines to overfetch from top | ||||
|  * @btm_ftch:        Number of extra lines to overfetch from bottom | ||||
|  * @left_rpt:        Number of extra pixels to repeat from left | ||||
|  * @right_rpt:       Number of extra pixels to repeat from right | ||||
|  * @top_rpt:         Number of extra lines to repeat from top | ||||
|  * @btm_rpt:         Number of extra lines to repeat from bottom | ||||
|  */ | ||||
| struct dpu_drm_pix_ext_v1 { | ||||
| 	/*
 | ||||
| 	 * Number of pixels ext in left, right, top and bottom direction | ||||
| 	 * for all color components. | ||||
| 	 */ | ||||
| 	int32_t num_ext_pxls_lr[DPU_MAX_PLANES]; | ||||
| 	int32_t num_ext_pxls_tb[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Number of pixels needs to be overfetched in left, right, top | ||||
| 	 * and bottom directions from source image for scaling. | ||||
| 	 */ | ||||
| 	int32_t left_ftch[DPU_MAX_PLANES]; | ||||
| 	int32_t right_ftch[DPU_MAX_PLANES]; | ||||
| 	int32_t top_ftch[DPU_MAX_PLANES]; | ||||
| 	int32_t btm_ftch[DPU_MAX_PLANES]; | ||||
| 	/*
 | ||||
| 	 * Number of pixels needs to be repeated in left, right, top and | ||||
| 	 * bottom directions for scaling. | ||||
| 	 */ | ||||
| 	int32_t left_rpt[DPU_MAX_PLANES]; | ||||
| 	int32_t right_rpt[DPU_MAX_PLANES]; | ||||
| 	int32_t top_rpt[DPU_MAX_PLANES]; | ||||
| 	int32_t btm_rpt[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_drm_de_v1 - version 1 of detail enhancer structure | ||||
|  * @enable:         Enables/disables detail enhancer | ||||
|  * @sharpen_level1: Sharpening strength for noise | ||||
|  * @sharpen_level2: Sharpening strength for context | ||||
|  * @clip:           Clip coefficient | ||||
|  * @limit:          Detail enhancer limit factor | ||||
|  * @thr_quiet:      Quite zone threshold | ||||
|  * @thr_dieout:     Die-out zone threshold | ||||
|  * @thr_low:        Linear zone left threshold | ||||
|  * @thr_high:       Linear zone right threshold | ||||
|  * @prec_shift:     Detail enhancer precision | ||||
|  * @adjust_a:       Mapping curves A coefficients | ||||
|  * @adjust_b:       Mapping curves B coefficients | ||||
|  * @adjust_c:       Mapping curves C coefficients | ||||
|  */ | ||||
| struct dpu_drm_de_v1 { | ||||
| 	uint32_t enable; | ||||
| 	int16_t sharpen_level1; | ||||
| 	int16_t sharpen_level2; | ||||
| 	uint16_t clip; | ||||
| 	uint16_t limit; | ||||
| 	uint16_t thr_quiet; | ||||
| 	uint16_t thr_dieout; | ||||
| 	uint16_t thr_low; | ||||
| 	uint16_t thr_high; | ||||
| 	uint16_t prec_shift; | ||||
| 	int16_t adjust_a[DPU_MAX_DE_CURVES]; | ||||
| 	int16_t adjust_b[DPU_MAX_DE_CURVES]; | ||||
| 	int16_t adjust_c[DPU_MAX_DE_CURVES]; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_drm_scaler_v2 - version 2 of struct dpu_drm_scaler | ||||
|  * @enable:            Scaler enable | ||||
|  * @dir_en:            Detail enhancer enable | ||||
|  * @pe:                Pixel extension settings | ||||
|  * @horz_decimate:     Horizontal decimation factor | ||||
|  * @vert_decimate:     Vertical decimation factor | ||||
|  * @init_phase_x:      Initial scaler phase values for x | ||||
|  * @phase_step_x:      Phase step values for x | ||||
|  * @init_phase_y:      Initial scaler phase values for y | ||||
|  * @phase_step_y:      Phase step values for y | ||||
|  * @preload_x:         Horizontal preload value | ||||
|  * @preload_y:         Vertical preload value | ||||
|  * @src_width:         Source width | ||||
|  * @src_height:        Source height | ||||
|  * @dst_width:         Destination width | ||||
|  * @dst_height:        Destination height | ||||
|  * @y_rgb_filter_cfg:  Y/RGB plane filter configuration | ||||
|  * @uv_filter_cfg:     UV plane filter configuration | ||||
|  * @alpha_filter_cfg:  Alpha filter configuration | ||||
|  * @blend_cfg:         Selection of blend coefficients | ||||
|  * @lut_flag:          LUT configuration flags | ||||
|  * @dir_lut_idx:       2d 4x4 LUT index | ||||
|  * @y_rgb_cir_lut_idx: Y/RGB circular LUT index | ||||
|  * @uv_cir_lut_idx:    UV circular LUT index | ||||
|  * @y_rgb_sep_lut_idx: Y/RGB separable LUT index | ||||
|  * @uv_sep_lut_idx:    UV separable LUT index | ||||
|  * @de:                Detail enhancer settings | ||||
|  */ | ||||
| struct dpu_drm_scaler_v2 { | ||||
| 	/*
 | ||||
| 	 * General definitions | ||||
| 	 */ | ||||
| 	uint32_t enable; | ||||
| 	uint32_t dir_en; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Pix ext settings | ||||
| 	 */ | ||||
| 	struct dpu_drm_pix_ext_v1 pe; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Decimation settings | ||||
| 	 */ | ||||
| 	uint32_t horz_decimate; | ||||
| 	uint32_t vert_decimate; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Phase settings | ||||
| 	 */ | ||||
| 	int32_t init_phase_x[DPU_MAX_PLANES]; | ||||
| 	int32_t phase_step_x[DPU_MAX_PLANES]; | ||||
| 	int32_t init_phase_y[DPU_MAX_PLANES]; | ||||
| 	int32_t phase_step_y[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	uint32_t preload_x[DPU_MAX_PLANES]; | ||||
| 	uint32_t preload_y[DPU_MAX_PLANES]; | ||||
| 	uint32_t src_width[DPU_MAX_PLANES]; | ||||
| 	uint32_t src_height[DPU_MAX_PLANES]; | ||||
| 
 | ||||
| 	uint32_t dst_width; | ||||
| 	uint32_t dst_height; | ||||
| 
 | ||||
| 	uint32_t y_rgb_filter_cfg; | ||||
| 	uint32_t uv_filter_cfg; | ||||
| 	uint32_t alpha_filter_cfg; | ||||
| 	uint32_t blend_cfg; | ||||
| 
 | ||||
| 	uint32_t lut_flag; | ||||
| 	uint32_t dir_lut_idx; | ||||
| 
 | ||||
| 	/* for Y(RGB) and UV planes*/ | ||||
| 	uint32_t y_rgb_cir_lut_idx; | ||||
| 	uint32_t uv_cir_lut_idx; | ||||
| 	uint32_t y_rgb_sep_lut_idx; | ||||
| 	uint32_t uv_sep_lut_idx; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Detail enhancer settings | ||||
| 	 */ | ||||
| 	struct dpu_drm_de_v1 de; | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| u32 *dpu_hw_util_get_log_mask_ptr(void); | ||||
| 
 | ||||
| void dpu_reg_write(struct dpu_hw_blk_reg_map *c, | ||||
| 		u32 reg_off, | ||||
| 		u32 val, | ||||
| 		const char *name); | ||||
| int dpu_reg_read(struct dpu_hw_blk_reg_map *c, u32 reg_off); | ||||
| 
 | ||||
| #define DPU_REG_WRITE(c, off, val) dpu_reg_write(c, off, val, #off) | ||||
| #define DPU_REG_READ(c, off) dpu_reg_read(c, off) | ||||
| 
 | ||||
| #define MISR_FRAME_COUNT_MASK		0xFF | ||||
| #define MISR_CTRL_ENABLE		BIT(8) | ||||
| #define MISR_CTRL_STATUS		BIT(9) | ||||
| #define MISR_CTRL_STATUS_CLEAR		BIT(10) | ||||
| #define INTF_MISR_CTRL_FREE_RUN_MASK	BIT(31) | ||||
| 
 | ||||
| void *dpu_hw_util_get_dir(void); | ||||
| 
 | ||||
| void dpu_set_scaler_v2(struct dpu_hw_scaler3_cfg *cfg, | ||||
| 		const struct dpu_drm_scaler_v2 *scale_v2); | ||||
| 
 | ||||
| void dpu_hw_setup_scaler3(struct dpu_hw_blk_reg_map *c, | ||||
| 		struct dpu_hw_scaler3_cfg *scaler3_cfg, | ||||
| 		u32 scaler_offset, u32 scaler_version, | ||||
| 		const struct dpu_format *format); | ||||
| 
 | ||||
| u32 dpu_hw_get_scaler3_ver(struct dpu_hw_blk_reg_map *c, | ||||
| 		u32 scaler_offset); | ||||
| 
 | ||||
| void dpu_hw_csc_setup(struct dpu_hw_blk_reg_map  *c, | ||||
| 		u32 csc_reg_off, | ||||
| 		struct dpu_csc_cfg *data, bool csc10); | ||||
| 
 | ||||
| uint32_t dpu_copy_formats( | ||||
| 		struct dpu_format_extended *dst_list, | ||||
| 		uint32_t dst_list_size, | ||||
| 		uint32_t dst_list_pos, | ||||
| 		const struct dpu_format_extended *src_list, | ||||
| 		uint32_t src_list_size); | ||||
| 
 | ||||
| #endif /* _DPU_HW_UTIL_H */ | ||||
							
								
								
									
										275
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_vbif.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										275
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_vbif.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,275 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_hwio.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_vbif.h" | ||||
| #include "dpu_dbg.h" | ||||
| 
 | ||||
| #define VBIF_VERSION			0x0000 | ||||
| #define VBIF_CLK_FORCE_CTRL0		0x0008 | ||||
| #define VBIF_CLK_FORCE_CTRL1		0x000C | ||||
| #define VBIF_QOS_REMAP_00		0x0020 | ||||
| #define VBIF_QOS_REMAP_01		0x0024 | ||||
| #define VBIF_QOS_REMAP_10		0x0028 | ||||
| #define VBIF_QOS_REMAP_11		0x002C | ||||
| #define VBIF_WRITE_GATHER_EN		0x00AC | ||||
| #define VBIF_IN_RD_LIM_CONF0		0x00B0 | ||||
| #define VBIF_IN_RD_LIM_CONF1		0x00B4 | ||||
| #define VBIF_IN_RD_LIM_CONF2		0x00B8 | ||||
| #define VBIF_IN_WR_LIM_CONF0		0x00C0 | ||||
| #define VBIF_IN_WR_LIM_CONF1		0x00C4 | ||||
| #define VBIF_IN_WR_LIM_CONF2		0x00C8 | ||||
| #define VBIF_OUT_RD_LIM_CONF0		0x00D0 | ||||
| #define VBIF_OUT_WR_LIM_CONF0		0x00D4 | ||||
| #define VBIF_OUT_AXI_AMEMTYPE_CONF0	0x0160 | ||||
| #define VBIF_OUT_AXI_AMEMTYPE_CONF1	0x0164 | ||||
| #define VBIF_XIN_PND_ERR		0x0190 | ||||
| #define VBIF_XIN_SRC_ERR		0x0194 | ||||
| #define VBIF_XIN_CLR_ERR		0x019C | ||||
| #define VBIF_XIN_HALT_CTRL0		0x0200 | ||||
| #define VBIF_XIN_HALT_CTRL1		0x0204 | ||||
| #define VBIF_XINL_QOS_RP_REMAP_000	0x0550 | ||||
| #define VBIF_XINL_QOS_LVL_REMAP_000	0x0590 | ||||
| 
 | ||||
| static void dpu_hw_clear_errors(struct dpu_hw_vbif *vbif, | ||||
| 		u32 *pnd_errors, u32 *src_errors) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 pnd, src; | ||||
| 
 | ||||
| 	if (!vbif) | ||||
| 		return; | ||||
| 	c = &vbif->hw; | ||||
| 	pnd = DPU_REG_READ(c, VBIF_XIN_PND_ERR); | ||||
| 	src = DPU_REG_READ(c, VBIF_XIN_SRC_ERR); | ||||
| 
 | ||||
| 	if (pnd_errors) | ||||
| 		*pnd_errors = pnd; | ||||
| 	if (src_errors) | ||||
| 		*src_errors = src; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, VBIF_XIN_CLR_ERR, pnd | src); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_set_mem_type(struct dpu_hw_vbif *vbif, | ||||
| 		u32 xin_id, u32 value) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 reg_off; | ||||
| 	u32 bit_off; | ||||
| 	u32 reg_val; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Assume 4 bits per bit field, 8 fields per 32-bit register so | ||||
| 	 * 16 bit fields maximum across two registers | ||||
| 	 */ | ||||
| 	if (!vbif || xin_id >= MAX_XIN_COUNT || xin_id >= 16) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &vbif->hw; | ||||
| 
 | ||||
| 	if (xin_id >= 8) { | ||||
| 		xin_id -= 8; | ||||
| 		reg_off = VBIF_OUT_AXI_AMEMTYPE_CONF1; | ||||
| 	} else { | ||||
| 		reg_off = VBIF_OUT_AXI_AMEMTYPE_CONF0; | ||||
| 	} | ||||
| 	bit_off = (xin_id & 0x7) * 4; | ||||
| 	reg_val = DPU_REG_READ(c, reg_off); | ||||
| 	reg_val &= ~(0x7 << bit_off); | ||||
| 	reg_val |= (value & 0x7) << bit_off; | ||||
| 	DPU_REG_WRITE(c, reg_off, reg_val); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_set_limit_conf(struct dpu_hw_vbif *vbif, | ||||
| 		u32 xin_id, bool rd, u32 limit) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &vbif->hw; | ||||
| 	u32 reg_val; | ||||
| 	u32 reg_off; | ||||
| 	u32 bit_off; | ||||
| 
 | ||||
| 	if (rd) | ||||
| 		reg_off = VBIF_IN_RD_LIM_CONF0; | ||||
| 	else | ||||
| 		reg_off = VBIF_IN_WR_LIM_CONF0; | ||||
| 
 | ||||
| 	reg_off += (xin_id / 4) * 4; | ||||
| 	bit_off = (xin_id % 4) * 8; | ||||
| 	reg_val = DPU_REG_READ(c, reg_off); | ||||
| 	reg_val &= ~(0xFF << bit_off); | ||||
| 	reg_val |= (limit) << bit_off; | ||||
| 	DPU_REG_WRITE(c, reg_off, reg_val); | ||||
| } | ||||
| 
 | ||||
| static u32 dpu_hw_get_limit_conf(struct dpu_hw_vbif *vbif, | ||||
| 		u32 xin_id, bool rd) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &vbif->hw; | ||||
| 	u32 reg_val; | ||||
| 	u32 reg_off; | ||||
| 	u32 bit_off; | ||||
| 	u32 limit; | ||||
| 
 | ||||
| 	if (rd) | ||||
| 		reg_off = VBIF_IN_RD_LIM_CONF0; | ||||
| 	else | ||||
| 		reg_off = VBIF_IN_WR_LIM_CONF0; | ||||
| 
 | ||||
| 	reg_off += (xin_id / 4) * 4; | ||||
| 	bit_off = (xin_id % 4) * 8; | ||||
| 	reg_val = DPU_REG_READ(c, reg_off); | ||||
| 	limit = (reg_val >> bit_off) & 0xFF; | ||||
| 
 | ||||
| 	return limit; | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_set_halt_ctrl(struct dpu_hw_vbif *vbif, | ||||
| 		u32 xin_id, bool enable) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &vbif->hw; | ||||
| 	u32 reg_val; | ||||
| 
 | ||||
| 	reg_val = DPU_REG_READ(c, VBIF_XIN_HALT_CTRL0); | ||||
| 
 | ||||
| 	if (enable) | ||||
| 		reg_val |= BIT(xin_id); | ||||
| 	else | ||||
| 		reg_val &= ~BIT(xin_id); | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, VBIF_XIN_HALT_CTRL0, reg_val); | ||||
| } | ||||
| 
 | ||||
| static bool dpu_hw_get_halt_ctrl(struct dpu_hw_vbif *vbif, | ||||
| 		u32 xin_id) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c = &vbif->hw; | ||||
| 	u32 reg_val; | ||||
| 
 | ||||
| 	reg_val = DPU_REG_READ(c, VBIF_XIN_HALT_CTRL1); | ||||
| 
 | ||||
| 	return (reg_val & BIT(xin_id)) ? true : false; | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_set_qos_remap(struct dpu_hw_vbif *vbif, | ||||
| 		u32 xin_id, u32 level, u32 remap_level) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 reg_val, reg_val_lvl, mask, reg_high, reg_shift; | ||||
| 
 | ||||
| 	if (!vbif) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &vbif->hw; | ||||
| 
 | ||||
| 	reg_high = ((xin_id & 0x8) >> 3) * 4 + (level * 8); | ||||
| 	reg_shift = (xin_id & 0x7) * 4; | ||||
| 
 | ||||
| 	reg_val = DPU_REG_READ(c, VBIF_XINL_QOS_RP_REMAP_000 + reg_high); | ||||
| 	reg_val_lvl = DPU_REG_READ(c, VBIF_XINL_QOS_LVL_REMAP_000 + reg_high); | ||||
| 
 | ||||
| 	mask = 0x7 << reg_shift; | ||||
| 
 | ||||
| 	reg_val &= ~mask; | ||||
| 	reg_val |= (remap_level << reg_shift) & mask; | ||||
| 
 | ||||
| 	reg_val_lvl &= ~mask; | ||||
| 	reg_val_lvl |= (remap_level << reg_shift) & mask; | ||||
| 
 | ||||
| 	DPU_REG_WRITE(c, VBIF_XINL_QOS_RP_REMAP_000 + reg_high, reg_val); | ||||
| 	DPU_REG_WRITE(c, VBIF_XINL_QOS_LVL_REMAP_000 + reg_high, reg_val_lvl); | ||||
| } | ||||
| 
 | ||||
| static void dpu_hw_set_write_gather_en(struct dpu_hw_vbif *vbif, u32 xin_id) | ||||
| { | ||||
| 	struct dpu_hw_blk_reg_map *c; | ||||
| 	u32 reg_val; | ||||
| 
 | ||||
| 	if (!vbif || xin_id >= MAX_XIN_COUNT) | ||||
| 		return; | ||||
| 
 | ||||
| 	c = &vbif->hw; | ||||
| 
 | ||||
| 	reg_val = DPU_REG_READ(c, VBIF_WRITE_GATHER_EN); | ||||
| 	reg_val |= BIT(xin_id); | ||||
| 	DPU_REG_WRITE(c, VBIF_WRITE_GATHER_EN, reg_val); | ||||
| } | ||||
| 
 | ||||
| static void _setup_vbif_ops(struct dpu_hw_vbif_ops *ops, | ||||
| 		unsigned long cap) | ||||
| { | ||||
| 	ops->set_limit_conf = dpu_hw_set_limit_conf; | ||||
| 	ops->get_limit_conf = dpu_hw_get_limit_conf; | ||||
| 	ops->set_halt_ctrl = dpu_hw_set_halt_ctrl; | ||||
| 	ops->get_halt_ctrl = dpu_hw_get_halt_ctrl; | ||||
| 	if (test_bit(DPU_VBIF_QOS_REMAP, &cap)) | ||||
| 		ops->set_qos_remap = dpu_hw_set_qos_remap; | ||||
| 	ops->set_mem_type = dpu_hw_set_mem_type; | ||||
| 	ops->clear_errors = dpu_hw_clear_errors; | ||||
| 	ops->set_write_gather_en = dpu_hw_set_write_gather_en; | ||||
| } | ||||
| 
 | ||||
| static const struct dpu_vbif_cfg *_top_offset(enum dpu_vbif vbif, | ||||
| 		const struct dpu_mdss_cfg *m, | ||||
| 		void __iomem *addr, | ||||
| 		struct dpu_hw_blk_reg_map *b) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < m->vbif_count; i++) { | ||||
| 		if (vbif == m->vbif[i].id) { | ||||
| 			b->base_off = addr; | ||||
| 			b->blk_off = m->vbif[i].base; | ||||
| 			b->length = m->vbif[i].len; | ||||
| 			b->hwversion = m->hwversion; | ||||
| 			b->log_mask = DPU_DBG_MASK_VBIF; | ||||
| 			return &m->vbif[i]; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return ERR_PTR(-EINVAL); | ||||
| } | ||||
| 
 | ||||
| struct dpu_hw_vbif *dpu_hw_vbif_init(enum dpu_vbif idx, | ||||
| 		void __iomem *addr, | ||||
| 		const struct dpu_mdss_cfg *m) | ||||
| { | ||||
| 	struct dpu_hw_vbif *c; | ||||
| 	const struct dpu_vbif_cfg *cfg; | ||||
| 
 | ||||
| 	c = kzalloc(sizeof(*c), GFP_KERNEL); | ||||
| 	if (!c) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	cfg = _top_offset(idx, m, addr, &c->hw); | ||||
| 	if (IS_ERR_OR_NULL(cfg)) { | ||||
| 		kfree(c); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Assign ops | ||||
| 	 */ | ||||
| 	c->idx = idx; | ||||
| 	c->cap = cfg; | ||||
| 	_setup_vbif_ops(&c->ops, c->cap->features); | ||||
| 
 | ||||
| 	/* no need to register sub-range in dpu dbg, dump entire vbif io base */ | ||||
| 
 | ||||
| 	return c; | ||||
| } | ||||
| 
 | ||||
| void dpu_hw_vbif_destroy(struct dpu_hw_vbif *vbif) | ||||
| { | ||||
| 	kfree(vbif); | ||||
| } | ||||
							
								
								
									
										128
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_vbif.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hw_vbif.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,128 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HW_VBIF_H | ||||
| #define _DPU_HW_VBIF_H | ||||
| 
 | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_util.h" | ||||
| 
 | ||||
| struct dpu_hw_vbif; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_hw_vbif_ops : Interface to the VBIF hardware driver functions | ||||
|  *  Assumption is these functions will be called after clocks are enabled | ||||
|  */ | ||||
| struct dpu_hw_vbif_ops { | ||||
| 	/**
 | ||||
| 	 * set_limit_conf - set transaction limit config | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 * @rd: true for read limit; false for write limit | ||||
| 	 * @limit: outstanding transaction limit | ||||
| 	 */ | ||||
| 	void (*set_limit_conf)(struct dpu_hw_vbif *vbif, | ||||
| 			u32 xin_id, bool rd, u32 limit); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_limit_conf - get transaction limit config | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 * @rd: true for read limit; false for write limit | ||||
| 	 * @return: outstanding transaction limit | ||||
| 	 */ | ||||
| 	u32 (*get_limit_conf)(struct dpu_hw_vbif *vbif, | ||||
| 			u32 xin_id, bool rd); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * set_halt_ctrl - set halt control | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 * @enable: halt control enable | ||||
| 	 */ | ||||
| 	void (*set_halt_ctrl)(struct dpu_hw_vbif *vbif, | ||||
| 			u32 xin_id, bool enable); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * get_halt_ctrl - get halt control | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 * @return: halt control enable | ||||
| 	 */ | ||||
| 	bool (*get_halt_ctrl)(struct dpu_hw_vbif *vbif, | ||||
| 			u32 xin_id); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * set_qos_remap - set QoS priority remap | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 * @level: priority level | ||||
| 	 * @remap_level: remapped level | ||||
| 	 */ | ||||
| 	void (*set_qos_remap)(struct dpu_hw_vbif *vbif, | ||||
| 			u32 xin_id, u32 level, u32 remap_level); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * set_mem_type - set memory type | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 * @value: memory type value | ||||
| 	 */ | ||||
| 	void (*set_mem_type)(struct dpu_hw_vbif *vbif, | ||||
| 			u32 xin_id, u32 value); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * clear_errors - clear any vbif errors | ||||
| 	 *	This function clears any detected pending/source errors | ||||
| 	 *	on the VBIF interface, and optionally returns the detected | ||||
| 	 *	error mask(s). | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @pnd_errors: pointer to pending error reporting variable | ||||
| 	 * @src_errors: pointer to source error reporting variable | ||||
| 	 */ | ||||
| 	void (*clear_errors)(struct dpu_hw_vbif *vbif, | ||||
| 		u32 *pnd_errors, u32 *src_errors); | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * set_write_gather_en - set write_gather enable | ||||
| 	 * @vbif: vbif context driver | ||||
| 	 * @xin_id: client interface identifier | ||||
| 	 */ | ||||
| 	void (*set_write_gather_en)(struct dpu_hw_vbif *vbif, u32 xin_id); | ||||
| }; | ||||
| 
 | ||||
| struct dpu_hw_vbif { | ||||
| 	/* base */ | ||||
| 	struct dpu_hw_blk_reg_map hw; | ||||
| 
 | ||||
| 	/* vbif */ | ||||
| 	enum dpu_vbif idx; | ||||
| 	const struct dpu_vbif_cfg *cap; | ||||
| 
 | ||||
| 	/* ops */ | ||||
| 	struct dpu_hw_vbif_ops ops; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_hw_vbif_init - initializes the vbif driver for the passed interface idx | ||||
|  * @idx:  Interface index for which driver object is required | ||||
|  * @addr: Mapped register io address of MDSS | ||||
|  * @m:    Pointer to mdss catalog data | ||||
|  */ | ||||
| struct dpu_hw_vbif *dpu_hw_vbif_init(enum dpu_vbif idx, | ||||
| 		void __iomem *addr, | ||||
| 		const struct dpu_mdss_cfg *m); | ||||
| 
 | ||||
| void dpu_hw_vbif_destroy(struct dpu_hw_vbif *vbif); | ||||
| 
 | ||||
| #endif /*_DPU_HW_VBIF_H */ | ||||
							
								
								
									
										56
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hwio.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_hwio.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,56 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_HWIO_H | ||||
| #define _DPU_HWIO_H | ||||
| 
 | ||||
| #include "dpu_hw_util.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * MDP TOP block Register and bit fields and defines | ||||
|  */ | ||||
| #define DISP_INTF_SEL                   0x004 | ||||
| #define INTR_EN                         0x010 | ||||
| #define INTR_STATUS                     0x014 | ||||
| #define INTR_CLEAR                      0x018 | ||||
| #define INTR2_EN                        0x008 | ||||
| #define INTR2_STATUS                    0x00c | ||||
| #define INTR2_CLEAR                     0x02c | ||||
| #define HIST_INTR_EN                    0x01c | ||||
| #define HIST_INTR_STATUS                0x020 | ||||
| #define HIST_INTR_CLEAR                 0x024 | ||||
| #define INTF_INTR_EN                    0x1C0 | ||||
| #define INTF_INTR_STATUS                0x1C4 | ||||
| #define INTF_INTR_CLEAR                 0x1C8 | ||||
| #define SPLIT_DISPLAY_EN                0x2F4 | ||||
| #define SPLIT_DISPLAY_UPPER_PIPE_CTRL   0x2F8 | ||||
| #define DSPP_IGC_COLOR0_RAM_LUTN        0x300 | ||||
| #define DSPP_IGC_COLOR1_RAM_LUTN        0x304 | ||||
| #define DSPP_IGC_COLOR2_RAM_LUTN        0x308 | ||||
| #define HW_EVENTS_CTL                   0x37C | ||||
| #define CLK_CTRL3                       0x3A8 | ||||
| #define CLK_STATUS3                     0x3AC | ||||
| #define CLK_CTRL4                       0x3B0 | ||||
| #define CLK_STATUS4                     0x3B4 | ||||
| #define CLK_CTRL5                       0x3B8 | ||||
| #define CLK_STATUS5                     0x3BC | ||||
| #define CLK_CTRL7                       0x3D0 | ||||
| #define CLK_STATUS7                     0x3D4 | ||||
| #define SPLIT_DISPLAY_LOWER_PIPE_CTRL   0x3F0 | ||||
| #define SPLIT_DISPLAY_TE_LINE_INTERVAL  0x3F4 | ||||
| #define INTF_SW_RESET_MASK              0x3FC | ||||
| #define HDMI_DP_CORE_SELECT             0x408 | ||||
| #define MDP_OUT_CTL_0                   0x410 | ||||
| #define MDP_VSYNC_SEL                   0x414 | ||||
| #define DCE_SEL                         0x450 | ||||
| 
 | ||||
| #endif /*_DPU_HWIO_H */ | ||||
							
								
								
									
										204
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_io_util.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_io_util.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,204 @@ | ||||
| /* Copyright (c) 2012-2015, 2017-2018, The Linux Foundation.
 | ||||
|  * All rights reserved. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/clk.h> | ||||
| #include <linux/clk/clk-conf.h> | ||||
| #include <linux/err.h> | ||||
| #include <linux/delay.h> | ||||
| 
 | ||||
| #include "dpu_io_util.h" | ||||
| 
 | ||||
| void msm_dss_put_clk(struct dss_clk *clk_arry, int num_clk) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = num_clk - 1; i >= 0; i--) { | ||||
| 		if (clk_arry[i].clk) | ||||
| 			clk_put(clk_arry[i].clk); | ||||
| 		clk_arry[i].clk = NULL; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| int msm_dss_get_clk(struct device *dev, struct dss_clk *clk_arry, int num_clk) | ||||
| { | ||||
| 	int i, rc = 0; | ||||
| 
 | ||||
| 	for (i = 0; i < num_clk; i++) { | ||||
| 		clk_arry[i].clk = clk_get(dev, clk_arry[i].clk_name); | ||||
| 		rc = PTR_RET(clk_arry[i].clk); | ||||
| 		if (rc) { | ||||
| 			DEV_ERR("%pS->%s: '%s' get failed. rc=%d\n", | ||||
| 				__builtin_return_address(0), __func__, | ||||
| 				clk_arry[i].clk_name, rc); | ||||
| 			goto error; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| 
 | ||||
| error: | ||||
| 	for (i--; i >= 0; i--) { | ||||
| 		if (clk_arry[i].clk) | ||||
| 			clk_put(clk_arry[i].clk); | ||||
| 		clk_arry[i].clk = NULL; | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| int msm_dss_clk_set_rate(struct dss_clk *clk_arry, int num_clk) | ||||
| { | ||||
| 	int i, rc = 0; | ||||
| 
 | ||||
| 	for (i = 0; i < num_clk; i++) { | ||||
| 		if (clk_arry[i].clk) { | ||||
| 			if (clk_arry[i].type != DSS_CLK_AHB) { | ||||
| 				DEV_DBG("%pS->%s: '%s' rate %ld\n", | ||||
| 					__builtin_return_address(0), __func__, | ||||
| 					clk_arry[i].clk_name, | ||||
| 					clk_arry[i].rate); | ||||
| 				rc = clk_set_rate(clk_arry[i].clk, | ||||
| 					clk_arry[i].rate); | ||||
| 				if (rc) { | ||||
| 					DEV_ERR("%pS->%s: %s failed. rc=%d\n", | ||||
| 						__builtin_return_address(0), | ||||
| 						__func__, | ||||
| 						clk_arry[i].clk_name, rc); | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 		} else { | ||||
| 			DEV_ERR("%pS->%s: '%s' is not available\n", | ||||
| 				__builtin_return_address(0), __func__, | ||||
| 				clk_arry[i].clk_name); | ||||
| 			rc = -EPERM; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| int msm_dss_enable_clk(struct dss_clk *clk_arry, int num_clk, int enable) | ||||
| { | ||||
| 	int i, rc = 0; | ||||
| 
 | ||||
| 	if (enable) { | ||||
| 		for (i = 0; i < num_clk; i++) { | ||||
| 			DEV_DBG("%pS->%s: enable '%s'\n", | ||||
| 				__builtin_return_address(0), __func__, | ||||
| 				clk_arry[i].clk_name); | ||||
| 			if (clk_arry[i].clk) { | ||||
| 				rc = clk_prepare_enable(clk_arry[i].clk); | ||||
| 				if (rc) | ||||
| 					DEV_ERR("%pS->%s: %s en fail. rc=%d\n", | ||||
| 						__builtin_return_address(0), | ||||
| 						__func__, | ||||
| 						clk_arry[i].clk_name, rc); | ||||
| 			} else { | ||||
| 				DEV_ERR("%pS->%s: '%s' is not available\n", | ||||
| 					__builtin_return_address(0), __func__, | ||||
| 					clk_arry[i].clk_name); | ||||
| 				rc = -EPERM; | ||||
| 			} | ||||
| 
 | ||||
| 			if (rc) { | ||||
| 				msm_dss_enable_clk(&clk_arry[i], | ||||
| 					i, false); | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
| 	} else { | ||||
| 		for (i = num_clk - 1; i >= 0; i--) { | ||||
| 			DEV_DBG("%pS->%s: disable '%s'\n", | ||||
| 				__builtin_return_address(0), __func__, | ||||
| 				clk_arry[i].clk_name); | ||||
| 
 | ||||
| 			if (clk_arry[i].clk) | ||||
| 				clk_disable_unprepare(clk_arry[i].clk); | ||||
| 			else | ||||
| 				DEV_ERR("%pS->%s: '%s' is not available\n", | ||||
| 					__builtin_return_address(0), __func__, | ||||
| 					clk_arry[i].clk_name); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| int msm_dss_parse_clock(struct platform_device *pdev, | ||||
| 			struct dss_module_power *mp) | ||||
| { | ||||
| 	u32 i, rc = 0; | ||||
| 	const char *clock_name; | ||||
| 	int num_clk = 0; | ||||
| 
 | ||||
| 	if (!pdev || !mp) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	mp->num_clk = 0; | ||||
| 	num_clk = of_property_count_strings(pdev->dev.of_node, | ||||
| 					    "assigned-clock-names"); | ||||
| 	if (num_clk <= 0) { | ||||
| 		pr_debug("clocks are not defined\n"); | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	mp->clk_config = devm_kzalloc(&pdev->dev, | ||||
| 				      sizeof(struct dss_clk) * num_clk, | ||||
| 				      GFP_KERNEL); | ||||
| 	if (!mp->clk_config) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	for (i = 0; i < num_clk; i++) { | ||||
| 		rc = of_property_read_string_index(pdev->dev.of_node, | ||||
| 						   "assigned-clock-names", i, | ||||
| 						   &clock_name); | ||||
| 		if (rc) { | ||||
| 			dev_err(&pdev->dev, "Failed to get clock name for %d\n", | ||||
| 				i); | ||||
| 			break; | ||||
| 		} | ||||
| 		strlcpy(mp->clk_config[i].clk_name, clock_name, | ||||
| 			sizeof(mp->clk_config[i].clk_name)); | ||||
| 
 | ||||
| 		mp->clk_config[i].type = DSS_CLK_AHB; | ||||
| 	} | ||||
| 
 | ||||
| 	rc = msm_dss_get_clk(&pdev->dev, mp->clk_config, num_clk); | ||||
| 	if (rc) { | ||||
| 		dev_err(&pdev->dev, "Failed to get clock refs %d\n", rc); | ||||
| 		goto err; | ||||
| 	} | ||||
| 
 | ||||
| 	rc = of_clk_set_defaults(pdev->dev.of_node, false); | ||||
| 	if (rc) { | ||||
| 		dev_err(&pdev->dev, "Failed to set clock defaults %d\n", rc); | ||||
| 		goto err; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = 0; i < num_clk; i++) { | ||||
| 		u32 rate = clk_get_rate(mp->clk_config[i].clk); | ||||
| 		if (!rate) | ||||
| 			continue; | ||||
| 		mp->clk_config[i].rate = rate; | ||||
| 		mp->clk_config[i].type = DSS_CLK_PCLK; | ||||
| 	} | ||||
| 
 | ||||
| 	mp->num_clk = num_clk; | ||||
| 	return 0; | ||||
| 
 | ||||
| err: | ||||
| 	msm_dss_put_clk(mp->clk_config, num_clk); | ||||
| 	return rc; | ||||
| } | ||||
							
								
								
									
										57
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_io_util.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_io_util.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | ||||
| /* Copyright (c) 2012, 2017-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_IO_UTIL_H__ | ||||
| #define __DPU_IO_UTIL_H__ | ||||
| 
 | ||||
| #include <linux/gpio.h> | ||||
| #include <linux/platform_device.h> | ||||
| #include <linux/types.h> | ||||
| 
 | ||||
| #define DEV_DBG(fmt, args...)   pr_debug(fmt, ##args) | ||||
| #define DEV_INFO(fmt, args...)  pr_info(fmt, ##args) | ||||
| #define DEV_WARN(fmt, args...)  pr_warn(fmt, ##args) | ||||
| #define DEV_ERR(fmt, args...)   pr_err(fmt, ##args) | ||||
| 
 | ||||
| struct dss_gpio { | ||||
| 	unsigned int gpio; | ||||
| 	unsigned int value; | ||||
| 	char gpio_name[32]; | ||||
| }; | ||||
| 
 | ||||
| enum dss_clk_type { | ||||
| 	DSS_CLK_AHB, /* no set rate. rate controlled through rpm */ | ||||
| 	DSS_CLK_PCLK, | ||||
| }; | ||||
| 
 | ||||
| struct dss_clk { | ||||
| 	struct clk *clk; /* clk handle */ | ||||
| 	char clk_name[32]; | ||||
| 	enum dss_clk_type type; | ||||
| 	unsigned long rate; | ||||
| 	unsigned long max_rate; | ||||
| }; | ||||
| 
 | ||||
| struct dss_module_power { | ||||
| 	unsigned int num_gpio; | ||||
| 	struct dss_gpio *gpio_config; | ||||
| 	unsigned int num_clk; | ||||
| 	struct dss_clk *clk_config; | ||||
| }; | ||||
| 
 | ||||
| int msm_dss_get_clk(struct device *dev, struct dss_clk *clk_arry, int num_clk); | ||||
| void msm_dss_put_clk(struct dss_clk *clk_arry, int num_clk); | ||||
| int msm_dss_clk_set_rate(struct dss_clk *clk_arry, int num_clk); | ||||
| int msm_dss_enable_clk(struct dss_clk *clk_arry, int num_clk, int enable); | ||||
| int msm_dss_parse_clock(struct platform_device *pdev, | ||||
| 		struct dss_module_power *mp); | ||||
| #endif /* __DPU_IO_UTIL_H__ */ | ||||
							
								
								
									
										66
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_irq.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_irq.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,66 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include <linux/irqdomain.h> | ||||
| #include <linux/irq.h> | ||||
| #include <linux/kthread.h> | ||||
| 
 | ||||
| #include "dpu_irq.h" | ||||
| #include "dpu_core_irq.h" | ||||
| 
 | ||||
| irqreturn_t dpu_irq(struct msm_kms *kms) | ||||
| { | ||||
| 	struct dpu_kms *dpu_kms = to_dpu_kms(kms); | ||||
| 
 | ||||
| 	return dpu_core_irq(dpu_kms); | ||||
| } | ||||
| 
 | ||||
| void dpu_irq_preinstall(struct msm_kms *kms) | ||||
| { | ||||
| 	struct dpu_kms *dpu_kms = to_dpu_kms(kms); | ||||
| 
 | ||||
| 	if (!dpu_kms->dev || !dpu_kms->dev->dev) { | ||||
| 		pr_err("invalid device handles\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_core_irq_preinstall(dpu_kms); | ||||
| } | ||||
| 
 | ||||
| int dpu_irq_postinstall(struct msm_kms *kms) | ||||
| { | ||||
| 	struct dpu_kms *dpu_kms = to_dpu_kms(kms); | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (!kms) { | ||||
| 		DPU_ERROR("invalid parameters\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	rc = dpu_core_irq_postinstall(dpu_kms); | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| void dpu_irq_uninstall(struct msm_kms *kms) | ||||
| { | ||||
| 	struct dpu_kms *dpu_kms = to_dpu_kms(kms); | ||||
| 
 | ||||
| 	if (!kms) { | ||||
| 		DPU_ERROR("invalid parameters\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_core_irq_uninstall(dpu_kms); | ||||
| } | ||||
							
								
								
									
										59
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_irq.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_irq.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_IRQ_H__ | ||||
| #define __DPU_IRQ_H__ | ||||
| 
 | ||||
| #include <linux/kernel.h> | ||||
| #include <linux/irqdomain.h> | ||||
| 
 | ||||
| #include "msm_kms.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_irq_controller - define MDSS level interrupt controller context | ||||
|  * @enabled_mask:	enable status of MDSS level interrupt | ||||
|  * @domain:		interrupt domain of this controller | ||||
|  */ | ||||
| struct dpu_irq_controller { | ||||
| 	unsigned long enabled_mask; | ||||
| 	struct irq_domain *domain; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_irq_preinstall - perform pre-installation of MDSS IRQ handler | ||||
|  * @kms:		pointer to kms context | ||||
|  * @return:		none | ||||
|  */ | ||||
| void dpu_irq_preinstall(struct msm_kms *kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_irq_postinstall - perform post-installation of MDSS IRQ handler | ||||
|  * @kms:		pointer to kms context | ||||
|  * @return:		0 if success; error code otherwise | ||||
|  */ | ||||
| int dpu_irq_postinstall(struct msm_kms *kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_irq_uninstall - uninstall MDSS IRQ handler | ||||
|  * @drm_dev:		pointer to kms context | ||||
|  * @return:		none | ||||
|  */ | ||||
| void dpu_irq_uninstall(struct msm_kms *kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_irq - MDSS level IRQ handler | ||||
|  * @kms:		pointer to kms context | ||||
|  * @return:		interrupt handling status | ||||
|  */ | ||||
| irqreturn_t dpu_irq(struct msm_kms *kms); | ||||
| 
 | ||||
| #endif /* __DPU_IRQ_H__ */ | ||||
							
								
								
									
										1345
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1345
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_kms.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										402
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										402
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_kms.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,402 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2015-2018, The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU General Public License version 2 as published by | ||||
|  * the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License along with | ||||
|  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_KMS_H__ | ||||
| #define __DPU_KMS_H__ | ||||
| 
 | ||||
| #include "msm_drv.h" | ||||
| #include "msm_kms.h" | ||||
| #include "msm_mmu.h" | ||||
| #include "msm_gem.h" | ||||
| #include "dpu_dbg.h" | ||||
| #include "dpu_hw_catalog.h" | ||||
| #include "dpu_hw_ctl.h" | ||||
| #include "dpu_hw_lm.h" | ||||
| #include "dpu_hw_interrupts.h" | ||||
| #include "dpu_hw_top.h" | ||||
| #include "dpu_rm.h" | ||||
| #include "dpu_power_handle.h" | ||||
| #include "dpu_irq.h" | ||||
| #include "dpu_core_perf.h" | ||||
| 
 | ||||
| #define DRMID(x) ((x) ? (x)->base.id : -1) | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU_DEBUG - macro for kms/plane/crtc/encoder/connector logs | ||||
|  * @fmt: Pointer to format string | ||||
|  */ | ||||
| #define DPU_DEBUG(fmt, ...)                                                \ | ||||
| 	do {                                                               \ | ||||
| 		if (unlikely(drm_debug & DRM_UT_KMS))                      \ | ||||
| 			DRM_DEBUG(fmt, ##__VA_ARGS__); \ | ||||
| 		else                                                       \ | ||||
| 			pr_debug(fmt, ##__VA_ARGS__);                      \ | ||||
| 	} while (0) | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU_DEBUG_DRIVER - macro for hardware driver logging | ||||
|  * @fmt: Pointer to format string | ||||
|  */ | ||||
| #define DPU_DEBUG_DRIVER(fmt, ...)                                         \ | ||||
| 	do {                                                               \ | ||||
| 		if (unlikely(drm_debug & DRM_UT_DRIVER))                   \ | ||||
| 			DRM_ERROR(fmt, ##__VA_ARGS__); \ | ||||
| 		else                                                       \ | ||||
| 			pr_debug(fmt, ##__VA_ARGS__);                      \ | ||||
| 	} while (0) | ||||
| 
 | ||||
| #define DPU_ERROR(fmt, ...) pr_err("[dpu error]" fmt, ##__VA_ARGS__) | ||||
| 
 | ||||
| /**
 | ||||
|  * ktime_compare_safe - compare two ktime structures | ||||
|  *	This macro is similar to the standard ktime_compare() function, but | ||||
|  *	attempts to also handle ktime overflows. | ||||
|  * @A: First ktime value | ||||
|  * @B: Second ktime value | ||||
|  * Returns: -1 if A < B, 0 if A == B, 1 if A > B | ||||
|  */ | ||||
| #define ktime_compare_safe(A, B) \ | ||||
| 	ktime_compare(ktime_sub((A), (B)), ktime_set(0, 0)) | ||||
| 
 | ||||
| #define DPU_NAME_SIZE  12 | ||||
| 
 | ||||
| /* timeout in frames waiting for frame done */ | ||||
| #define DPU_FRAME_DONE_TIMEOUT	60 | ||||
| 
 | ||||
| /*
 | ||||
|  * struct dpu_irq_callback - IRQ callback handlers | ||||
|  * @list: list to callback | ||||
|  * @func: intr handler | ||||
|  * @arg: argument for the handler | ||||
|  */ | ||||
| struct dpu_irq_callback { | ||||
| 	struct list_head list; | ||||
| 	void (*func)(void *arg, int irq_idx); | ||||
| 	void *arg; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_irq: IRQ structure contains callback registration info | ||||
|  * @total_irq:    total number of irq_idx obtained from HW interrupts mapping | ||||
|  * @irq_cb_tbl:   array of IRQ callbacks setting | ||||
|  * @enable_counts array of IRQ enable counts | ||||
|  * @cb_lock:      callback lock | ||||
|  * @debugfs_file: debugfs file for irq statistics | ||||
|  */ | ||||
| struct dpu_irq { | ||||
| 	u32 total_irqs; | ||||
| 	struct list_head *irq_cb_tbl; | ||||
| 	atomic_t *enable_counts; | ||||
| 	atomic_t *irq_counts; | ||||
| 	spinlock_t cb_lock; | ||||
| 	struct dentry *debugfs_file; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_kms { | ||||
| 	struct msm_kms base; | ||||
| 	struct drm_device *dev; | ||||
| 	int core_rev; | ||||
| 	struct dpu_mdss_cfg *catalog; | ||||
| 
 | ||||
| 	struct dpu_power_handle phandle; | ||||
| 	struct dpu_power_client *core_client; | ||||
| 	struct dpu_power_event *power_event; | ||||
| 
 | ||||
| 	/* directory entry for debugfs */ | ||||
| 	struct dentry *debugfs_root; | ||||
| 	struct dentry *debugfs_danger; | ||||
| 	struct dentry *debugfs_vbif; | ||||
| 
 | ||||
| 	/* io/register spaces: */ | ||||
| 	void __iomem *mmio, *vbif[VBIF_MAX], *reg_dma; | ||||
| 	unsigned long mmio_len, vbif_len[VBIF_MAX], reg_dma_len; | ||||
| 
 | ||||
| 	struct regulator *vdd; | ||||
| 	struct regulator *mmagic; | ||||
| 	struct regulator *venus; | ||||
| 
 | ||||
| 	struct dpu_hw_intr *hw_intr; | ||||
| 	struct dpu_irq irq_obj; | ||||
| 
 | ||||
| 	struct dpu_core_perf perf; | ||||
| 
 | ||||
| 	/* saved atomic state during system suspend */ | ||||
| 	struct drm_atomic_state *suspend_state; | ||||
| 	bool suspend_block; | ||||
| 
 | ||||
| 	struct dpu_rm rm; | ||||
| 	bool rm_init; | ||||
| 
 | ||||
| 	struct dpu_hw_vbif *hw_vbif[VBIF_MAX]; | ||||
| 	struct dpu_hw_mdp *hw_mdp; | ||||
| 
 | ||||
| 	bool has_danger_ctrl; | ||||
| 
 | ||||
| 	struct platform_device *pdev; | ||||
| 	bool rpm_enabled; | ||||
| 	struct dss_module_power mp; | ||||
| }; | ||||
| 
 | ||||
| struct vsync_info { | ||||
| 	u32 frame_count; | ||||
| 	u32 line_count; | ||||
| }; | ||||
| 
 | ||||
| #define to_dpu_kms(x) container_of(x, struct dpu_kms, base) | ||||
| 
 | ||||
| /* get struct msm_kms * from drm_device * */ | ||||
| #define ddev_to_msm_kms(D) ((D) && (D)->dev_private ? \ | ||||
| 		((struct msm_drm_private *)((D)->dev_private))->kms : NULL) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_is_suspend_state - whether or not the system is pm suspended | ||||
|  * @dev: Pointer to drm device | ||||
|  * Return: Suspend status | ||||
|  */ | ||||
| static inline bool dpu_kms_is_suspend_state(struct drm_device *dev) | ||||
| { | ||||
| 	if (!ddev_to_msm_kms(dev)) | ||||
| 		return false; | ||||
| 
 | ||||
| 	return to_dpu_kms(ddev_to_msm_kms(dev))->suspend_state != NULL; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_is_suspend_blocked - whether or not commits are blocked due to pm | ||||
|  *				suspend status | ||||
|  * @dev: Pointer to drm device | ||||
|  * Return: True if commits should be rejected due to pm suspend | ||||
|  */ | ||||
| static inline bool dpu_kms_is_suspend_blocked(struct drm_device *dev) | ||||
| { | ||||
| 	if (!dpu_kms_is_suspend_state(dev)) | ||||
| 		return false; | ||||
| 
 | ||||
| 	return to_dpu_kms(ddev_to_msm_kms(dev))->suspend_block; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * Debugfs functions - extra helper functions for debugfs support | ||||
|  * | ||||
|  * Main debugfs documentation is located at, | ||||
|  * | ||||
|  * Documentation/filesystems/debugfs.txt | ||||
|  * | ||||
|  * @dpu_debugfs_setup_regset32: Initialize data for dpu_debugfs_create_regset32 | ||||
|  * @dpu_debugfs_create_regset32: Create 32-bit register dump file | ||||
|  * @dpu_debugfs_get_root: Get root dentry for DPU_KMS's debugfs node | ||||
|  */ | ||||
| 
 | ||||
| /**
 | ||||
|  * Companion structure for dpu_debugfs_create_regset32. Do not initialize the | ||||
|  * members of this structure explicitly; use dpu_debugfs_setup_regset32 instead. | ||||
|  */ | ||||
| struct dpu_debugfs_regset32 { | ||||
| 	uint32_t offset; | ||||
| 	uint32_t blk_len; | ||||
| 	struct dpu_kms *dpu_kms; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_debugfs_setup_regset32 - Initialize register block definition for debugfs | ||||
|  * This function is meant to initialize dpu_debugfs_regset32 structures for use | ||||
|  * with dpu_debugfs_create_regset32. | ||||
|  * @regset: opaque register definition structure | ||||
|  * @offset: sub-block offset | ||||
|  * @length: sub-block length, in bytes | ||||
|  * @dpu_kms: pointer to dpu kms structure | ||||
|  */ | ||||
| void dpu_debugfs_setup_regset32(struct dpu_debugfs_regset32 *regset, | ||||
| 		uint32_t offset, uint32_t length, struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_debugfs_create_regset32 - Create register read back file for debugfs | ||||
|  * | ||||
|  * This function is almost identical to the standard debugfs_create_regset32() | ||||
|  * function, with the main difference being that a list of register | ||||
|  * names/offsets do not need to be provided. The 'read' function simply outputs | ||||
|  * sequential register values over a specified range. | ||||
|  * | ||||
|  * Similar to the related debugfs_create_regset32 API, the structure pointed to | ||||
|  * by regset needs to persist for the lifetime of the created file. The calling | ||||
|  * code is responsible for initialization/management of this structure. | ||||
|  * | ||||
|  * The structure pointed to by regset is meant to be opaque. Please use | ||||
|  * dpu_debugfs_setup_regset32 to initialize it. | ||||
|  * | ||||
|  * @name:   File name within debugfs | ||||
|  * @mode:   File mode within debugfs | ||||
|  * @parent: Parent directory entry within debugfs, can be NULL | ||||
|  * @regset: Pointer to persistent register block definition | ||||
|  * | ||||
|  * Return: dentry pointer for newly created file, use either debugfs_remove() | ||||
|  *         or debugfs_remove_recursive() (on a parent directory) to remove the | ||||
|  *         file | ||||
|  */ | ||||
| void *dpu_debugfs_create_regset32(const char *name, umode_t mode, | ||||
| 		void *parent, struct dpu_debugfs_regset32 *regset); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_debugfs_get_root - Return root directory entry for KMS's debugfs | ||||
|  * | ||||
|  * The return value should be passed as the 'parent' argument to subsequent | ||||
|  * debugfs create calls. | ||||
|  * | ||||
|  * @dpu_kms: Pointer to DPU's KMS structure | ||||
|  * | ||||
|  * Return: dentry pointer for DPU's debugfs location | ||||
|  */ | ||||
| void *dpu_debugfs_get_root(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU info management functions | ||||
|  * These functions/definitions allow for building up a 'dpu_info' structure | ||||
|  * containing one or more "key=value\n" entries. | ||||
|  */ | ||||
| #define DPU_KMS_INFO_MAX_SIZE	4096 | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_kms_info - connector information structure container | ||||
|  * @data: Array of information character data | ||||
|  * @len: Current length of information data | ||||
|  * @staged_len: Temporary data buffer length, commit to | ||||
|  *              len using dpu_kms_info_stop | ||||
|  * @start: Whether or not a partial data entry was just started | ||||
|  */ | ||||
| struct dpu_kms_info { | ||||
| 	char data[DPU_KMS_INFO_MAX_SIZE]; | ||||
| 	uint32_t len; | ||||
| 	uint32_t staged_len; | ||||
| 	bool start; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU_KMS_INFO_DATA - Macro for accessing dpu_kms_info data bytes | ||||
|  * @S: Pointer to dpu_kms_info structure | ||||
|  * Returns: Pointer to byte data | ||||
|  */ | ||||
| #define DPU_KMS_INFO_DATA(S)    ((S) ? ((struct dpu_kms_info *)(S))->data : 0) | ||||
| 
 | ||||
| /**
 | ||||
|  * DPU_KMS_INFO_DATALEN - Macro for accessing dpu_kms_info data length | ||||
|  *			it adds an extra character length to count null. | ||||
|  * @S: Pointer to dpu_kms_info structure | ||||
|  * Returns: Size of available byte data | ||||
|  */ | ||||
| #define DPU_KMS_INFO_DATALEN(S) ((S) ? ((struct dpu_kms_info *)(S))->len + 1 \ | ||||
| 							: 0) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_reset - reset dpu_kms_info structure | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  */ | ||||
| void dpu_kms_info_reset(struct dpu_kms_info *info); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_add_keyint - add integer value to 'dpu_kms_info' | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  * @key: Pointer to key string | ||||
|  * @value: Signed 64-bit integer value | ||||
|  */ | ||||
| void dpu_kms_info_add_keyint(struct dpu_kms_info *info, | ||||
| 		const char *key, | ||||
| 		int64_t value); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_add_keystr - add string value to 'dpu_kms_info' | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  * @key: Pointer to key string | ||||
|  * @value: Pointer to string value | ||||
|  */ | ||||
| void dpu_kms_info_add_keystr(struct dpu_kms_info *info, | ||||
| 		const char *key, | ||||
| 		const char *value); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_start - begin adding key to 'dpu_kms_info' | ||||
|  * Usage: | ||||
|  *      dpu_kms_info_start(key) | ||||
|  *      dpu_kms_info_append(val_1) | ||||
|  *      ... | ||||
|  *      dpu_kms_info_append(val_n) | ||||
|  *      dpu_kms_info_stop | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  * @key: Pointer to key string | ||||
|  */ | ||||
| void dpu_kms_info_start(struct dpu_kms_info *info, | ||||
| 		const char *key); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_append - append value string to 'dpu_kms_info' | ||||
|  * Usage: | ||||
|  *      dpu_kms_info_start(key) | ||||
|  *      dpu_kms_info_append(val_1) | ||||
|  *      ... | ||||
|  *      dpu_kms_info_append(val_n) | ||||
|  *      dpu_kms_info_stop | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  * @str: Pointer to partial value string | ||||
|  */ | ||||
| void dpu_kms_info_append(struct dpu_kms_info *info, | ||||
| 		const char *str); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_append_format - append format code string to 'dpu_kms_info' | ||||
|  * Usage: | ||||
|  *      dpu_kms_info_start(key) | ||||
|  *      dpu_kms_info_append_format(fourcc, modifier) | ||||
|  *      ... | ||||
|  *      dpu_kms_info_stop | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  * @pixel_format: FOURCC format code | ||||
|  * @modifier: 64-bit drm format modifier | ||||
|  */ | ||||
| void dpu_kms_info_append_format(struct dpu_kms_info *info, | ||||
| 		uint32_t pixel_format, | ||||
| 		uint64_t modifier); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_info_stop - finish adding key to 'dpu_kms_info' | ||||
|  * Usage: | ||||
|  *      dpu_kms_info_start(key) | ||||
|  *      dpu_kms_info_append(val_1) | ||||
|  *      ... | ||||
|  *      dpu_kms_info_append(val_n) | ||||
|  *      dpu_kms_info_stop | ||||
|  * @info: Pointer to dpu_kms_info structure | ||||
|  */ | ||||
| void dpu_kms_info_stop(struct dpu_kms_info *info); | ||||
| 
 | ||||
| /**
 | ||||
|  * Vblank enable/disable functions | ||||
|  */ | ||||
| int dpu_enable_vblank(struct msm_kms *kms, struct drm_crtc *crtc); | ||||
| void dpu_disable_vblank(struct msm_kms *kms, struct drm_crtc *crtc); | ||||
| 
 | ||||
| void dpu_kms_encoder_enable(struct drm_encoder *encoder); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_kms_get_clk_rate() - get the clock rate | ||||
|  * @dpu_kms:  poiner to dpu_kms structure | ||||
|  * @clock_name: clock name to get the rate | ||||
|  * | ||||
|  * Return: current clock rate | ||||
|  */ | ||||
| u64 dpu_kms_get_clk_rate(struct dpu_kms *dpu_kms, char *clock_name); | ||||
| 
 | ||||
| #endif /* __dpu_kms_H__ */ | ||||
							
								
								
									
										153
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_kms_utils.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										153
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_kms_utils.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,153 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"dpu-kms_utils:[%s] " fmt, __func__ | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| void dpu_kms_info_reset(struct dpu_kms_info *info) | ||||
| { | ||||
| 	if (info) { | ||||
| 		info->len = 0; | ||||
| 		info->staged_len = 0; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_kms_info_add_keyint(struct dpu_kms_info *info, | ||||
| 		const char *key, | ||||
| 		int64_t value) | ||||
| { | ||||
| 	uint32_t len; | ||||
| 
 | ||||
| 	if (info && key) { | ||||
| 		len = snprintf(info->data + info->len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->len, | ||||
| 				"%s=%lld\n", | ||||
| 				key, | ||||
| 				value); | ||||
| 
 | ||||
| 		/* check if snprintf truncated the string */ | ||||
| 		if ((info->len + len) < DPU_KMS_INFO_MAX_SIZE) | ||||
| 			info->len += len; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_kms_info_add_keystr(struct dpu_kms_info *info, | ||||
| 		const char *key, | ||||
| 		const char *value) | ||||
| { | ||||
| 	uint32_t len; | ||||
| 
 | ||||
| 	if (info && key && value) { | ||||
| 		len = snprintf(info->data + info->len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->len, | ||||
| 				"%s=%s\n", | ||||
| 				key, | ||||
| 				value); | ||||
| 
 | ||||
| 		/* check if snprintf truncated the string */ | ||||
| 		if ((info->len + len) < DPU_KMS_INFO_MAX_SIZE) | ||||
| 			info->len += len; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_kms_info_start(struct dpu_kms_info *info, | ||||
| 		const char *key) | ||||
| { | ||||
| 	uint32_t len; | ||||
| 
 | ||||
| 	if (info && key) { | ||||
| 		len = snprintf(info->data + info->len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->len, | ||||
| 				"%s=", | ||||
| 				key); | ||||
| 
 | ||||
| 		info->start = true; | ||||
| 
 | ||||
| 		/* check if snprintf truncated the string */ | ||||
| 		if ((info->len + len) < DPU_KMS_INFO_MAX_SIZE) | ||||
| 			info->staged_len = info->len + len; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_kms_info_append(struct dpu_kms_info *info, | ||||
| 		const char *str) | ||||
| { | ||||
| 	uint32_t len; | ||||
| 
 | ||||
| 	if (info) { | ||||
| 		len = snprintf(info->data + info->staged_len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->staged_len, | ||||
| 				"%s", | ||||
| 				str); | ||||
| 
 | ||||
| 		/* check if snprintf truncated the string */ | ||||
| 		if ((info->staged_len + len) < DPU_KMS_INFO_MAX_SIZE) { | ||||
| 			info->staged_len += len; | ||||
| 			info->start = false; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_kms_info_append_format(struct dpu_kms_info *info, | ||||
| 		uint32_t pixel_format, | ||||
| 		uint64_t modifier) | ||||
| { | ||||
| 	uint32_t len; | ||||
| 
 | ||||
| 	if (!info) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (modifier) { | ||||
| 		len = snprintf(info->data + info->staged_len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->staged_len, | ||||
| 				info->start ? | ||||
| 				"%c%c%c%c/%llX/%llX" : " %c%c%c%c/%llX/%llX", | ||||
| 				(pixel_format >> 0) & 0xFF, | ||||
| 				(pixel_format >> 8) & 0xFF, | ||||
| 				(pixel_format >> 16) & 0xFF, | ||||
| 				(pixel_format >> 24) & 0xFF, | ||||
| 				(modifier >> 56) & 0xFF, | ||||
| 				modifier & ((1ULL << 56) - 1)); | ||||
| 	} else { | ||||
| 		len = snprintf(info->data + info->staged_len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->staged_len, | ||||
| 				info->start ? | ||||
| 				"%c%c%c%c" : " %c%c%c%c", | ||||
| 				(pixel_format >> 0) & 0xFF, | ||||
| 				(pixel_format >> 8) & 0xFF, | ||||
| 				(pixel_format >> 16) & 0xFF, | ||||
| 				(pixel_format >> 24) & 0xFF); | ||||
| 	} | ||||
| 
 | ||||
| 	/* check if snprintf truncated the string */ | ||||
| 	if ((info->staged_len + len) < DPU_KMS_INFO_MAX_SIZE) { | ||||
| 		info->staged_len += len; | ||||
| 		info->start = false; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_kms_info_stop(struct dpu_kms_info *info) | ||||
| { | ||||
| 	uint32_t len; | ||||
| 
 | ||||
| 	if (info) { | ||||
| 		/* insert final delimiter */ | ||||
| 		len = snprintf(info->data + info->staged_len, | ||||
| 				DPU_KMS_INFO_MAX_SIZE - info->staged_len, | ||||
| 				"\n"); | ||||
| 
 | ||||
| 		/* check if snprintf truncated the string */ | ||||
| 		if ((info->staged_len + len) < DPU_KMS_INFO_MAX_SIZE) | ||||
| 			info->len = info->staged_len + len; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										245
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_mdss.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										245
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_mdss.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,245 @@ | ||||
| /*
 | ||||
|  * SPDX-License-Identifier: GPL-2.0 | ||||
|  * Copyright (c) 2018, The Linux Foundation | ||||
|  */ | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| #define to_dpu_mdss(x) container_of(x, struct dpu_mdss, base) | ||||
| 
 | ||||
| #define HW_INTR_STATUS			0x0010 | ||||
| 
 | ||||
| struct dpu_mdss { | ||||
| 	struct msm_mdss base; | ||||
| 	void __iomem *mmio; | ||||
| 	unsigned long mmio_len; | ||||
| 	u32 hwversion; | ||||
| 	struct dss_module_power mp; | ||||
| 	struct dpu_irq_controller irq_controller; | ||||
| }; | ||||
| 
 | ||||
| static irqreturn_t dpu_mdss_irq(int irq, void *arg) | ||||
| { | ||||
| 	struct dpu_mdss *dpu_mdss = arg; | ||||
| 	u32 interrupts; | ||||
| 
 | ||||
| 	interrupts = readl_relaxed(dpu_mdss->mmio + HW_INTR_STATUS); | ||||
| 
 | ||||
| 	while (interrupts) { | ||||
| 		irq_hw_number_t hwirq = fls(interrupts) - 1; | ||||
| 		unsigned int mapping; | ||||
| 		int rc; | ||||
| 
 | ||||
| 		mapping = irq_find_mapping(dpu_mdss->irq_controller.domain, | ||||
| 					   hwirq); | ||||
| 		if (mapping == 0) { | ||||
| 			DRM_ERROR("couldn't find irq mapping for %lu\n", hwirq); | ||||
| 			return IRQ_NONE; | ||||
| 		} | ||||
| 
 | ||||
| 		rc = generic_handle_irq(mapping); | ||||
| 		if (rc < 0) { | ||||
| 			DRM_ERROR("handle irq fail: irq=%lu mapping=%u rc=%d\n", | ||||
| 				  hwirq, mapping, rc); | ||||
| 			return IRQ_NONE; | ||||
| 		} | ||||
| 
 | ||||
| 		interrupts &= ~(1 << hwirq); | ||||
| 	} | ||||
| 
 | ||||
| 	return IRQ_HANDLED; | ||||
| } | ||||
| 
 | ||||
| static void dpu_mdss_irq_mask(struct irq_data *irqd) | ||||
| { | ||||
| 	struct dpu_mdss *dpu_mdss = irq_data_get_irq_chip_data(irqd); | ||||
| 
 | ||||
| 	/* memory barrier */ | ||||
| 	smp_mb__before_atomic(); | ||||
| 	clear_bit(irqd->hwirq, &dpu_mdss->irq_controller.enabled_mask); | ||||
| 	/* memory barrier */ | ||||
| 	smp_mb__after_atomic(); | ||||
| } | ||||
| 
 | ||||
| static void dpu_mdss_irq_unmask(struct irq_data *irqd) | ||||
| { | ||||
| 	struct dpu_mdss *dpu_mdss = irq_data_get_irq_chip_data(irqd); | ||||
| 
 | ||||
| 	/* memory barrier */ | ||||
| 	smp_mb__before_atomic(); | ||||
| 	set_bit(irqd->hwirq, &dpu_mdss->irq_controller.enabled_mask); | ||||
| 	/* memory barrier */ | ||||
| 	smp_mb__after_atomic(); | ||||
| } | ||||
| 
 | ||||
| static struct irq_chip dpu_mdss_irq_chip = { | ||||
| 	.name = "dpu_mdss", | ||||
| 	.irq_mask = dpu_mdss_irq_mask, | ||||
| 	.irq_unmask = dpu_mdss_irq_unmask, | ||||
| }; | ||||
| 
 | ||||
| static int dpu_mdss_irqdomain_map(struct irq_domain *domain, | ||||
| 		unsigned int irq, irq_hw_number_t hwirq) | ||||
| { | ||||
| 	struct dpu_mdss *dpu_mdss = domain->host_data; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	irq_set_chip_and_handler(irq, &dpu_mdss_irq_chip, handle_level_irq); | ||||
| 	ret = irq_set_chip_data(irq, dpu_mdss); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static const struct irq_domain_ops dpu_mdss_irqdomain_ops = { | ||||
| 	.map = dpu_mdss_irqdomain_map, | ||||
| 	.xlate = irq_domain_xlate_onecell, | ||||
| }; | ||||
| 
 | ||||
| static int _dpu_mdss_irq_domain_add(struct dpu_mdss *dpu_mdss) | ||||
| { | ||||
| 	struct device *dev; | ||||
| 	struct irq_domain *domain; | ||||
| 
 | ||||
| 	dev = dpu_mdss->base.dev->dev; | ||||
| 
 | ||||
| 	domain = irq_domain_add_linear(dev->of_node, 32, | ||||
| 			&dpu_mdss_irqdomain_ops, dpu_mdss); | ||||
| 	if (!domain) { | ||||
| 		DPU_ERROR("failed to add irq_domain\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_mdss->irq_controller.enabled_mask = 0; | ||||
| 	dpu_mdss->irq_controller.domain = domain; | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| int _dpu_mdss_irq_domain_fini(struct dpu_mdss *dpu_mdss) | ||||
| { | ||||
| 	if (dpu_mdss->irq_controller.domain) { | ||||
| 		irq_domain_remove(dpu_mdss->irq_controller.domain); | ||||
| 		dpu_mdss->irq_controller.domain = NULL; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| static int dpu_mdss_enable(struct msm_mdss *mdss) | ||||
| { | ||||
| 	struct dpu_mdss *dpu_mdss = to_dpu_mdss(mdss); | ||||
| 	struct dss_module_power *mp = &dpu_mdss->mp; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	ret = msm_dss_enable_clk(mp->clk_config, mp->num_clk, true); | ||||
| 	if (ret) | ||||
| 		DPU_ERROR("clock enable failed, ret:%d\n", ret); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static int dpu_mdss_disable(struct msm_mdss *mdss) | ||||
| { | ||||
| 	struct dpu_mdss *dpu_mdss = to_dpu_mdss(mdss); | ||||
| 	struct dss_module_power *mp = &dpu_mdss->mp; | ||||
| 	int ret; | ||||
| 
 | ||||
| 	ret = msm_dss_enable_clk(mp->clk_config, mp->num_clk, false); | ||||
| 	if (ret) | ||||
| 		DPU_ERROR("clock disable failed, ret:%d\n", ret); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static void dpu_mdss_destroy(struct drm_device *dev) | ||||
| { | ||||
| 	struct platform_device *pdev = to_platform_device(dev->dev); | ||||
| 	struct msm_drm_private *priv = dev->dev_private; | ||||
| 	struct dpu_mdss *dpu_mdss = to_dpu_mdss(priv->mdss); | ||||
| 	struct dss_module_power *mp = &dpu_mdss->mp; | ||||
| 
 | ||||
| 	_dpu_mdss_irq_domain_fini(dpu_mdss); | ||||
| 
 | ||||
| 	msm_dss_put_clk(mp->clk_config, mp->num_clk); | ||||
| 	devm_kfree(&pdev->dev, mp->clk_config); | ||||
| 
 | ||||
| 	if (dpu_mdss->mmio) | ||||
| 		devm_iounmap(&pdev->dev, dpu_mdss->mmio); | ||||
| 	dpu_mdss->mmio = NULL; | ||||
| 
 | ||||
| 	pm_runtime_disable(dev->dev); | ||||
| 	priv->mdss = NULL; | ||||
| } | ||||
| 
 | ||||
| static const struct msm_mdss_funcs mdss_funcs = { | ||||
| 	.enable	= dpu_mdss_enable, | ||||
| 	.disable = dpu_mdss_disable, | ||||
| 	.destroy = dpu_mdss_destroy, | ||||
| }; | ||||
| 
 | ||||
| int dpu_mdss_init(struct drm_device *dev) | ||||
| { | ||||
| 	struct platform_device *pdev = to_platform_device(dev->dev); | ||||
| 	struct msm_drm_private *priv = dev->dev_private; | ||||
| 	struct resource *res; | ||||
| 	struct dpu_mdss *dpu_mdss; | ||||
| 	struct dss_module_power *mp; | ||||
| 	int ret = 0; | ||||
| 
 | ||||
| 	dpu_mdss = devm_kzalloc(dev->dev, sizeof(*dpu_mdss), GFP_KERNEL); | ||||
| 	if (!dpu_mdss) | ||||
| 		return -ENOMEM; | ||||
| 
 | ||||
| 	dpu_mdss->mmio = msm_ioremap(pdev, "mdss", "mdss"); | ||||
| 	if (IS_ERR(dpu_mdss->mmio)) | ||||
| 		return PTR_ERR(dpu_mdss->mmio); | ||||
| 
 | ||||
| 	DRM_DEBUG("mapped mdss address space @%pK\n", dpu_mdss->mmio); | ||||
| 
 | ||||
| 	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "mdss"); | ||||
| 	if (!res) { | ||||
| 		DRM_ERROR("failed to get memory resource for mdss\n"); | ||||
| 		return -ENOMEM; | ||||
| 	} | ||||
| 	dpu_mdss->mmio_len = resource_size(res); | ||||
| 
 | ||||
| 	mp = &dpu_mdss->mp; | ||||
| 	ret = msm_dss_parse_clock(pdev, mp); | ||||
| 	if (ret) { | ||||
| 		DPU_ERROR("failed to parse clocks, ret=%d\n", ret); | ||||
| 		goto clk_parse_err; | ||||
| 	} | ||||
| 
 | ||||
| 	dpu_mdss->base.dev = dev; | ||||
| 	dpu_mdss->base.funcs = &mdss_funcs; | ||||
| 
 | ||||
| 	ret = _dpu_mdss_irq_domain_add(dpu_mdss); | ||||
| 	if (ret) | ||||
| 		goto irq_domain_error; | ||||
| 
 | ||||
| 	ret = devm_request_irq(dev->dev, platform_get_irq(pdev, 0), | ||||
| 			dpu_mdss_irq, 0, "dpu_mdss_isr", dpu_mdss); | ||||
| 	if (ret) { | ||||
| 		DPU_ERROR("failed to init irq: %d\n", ret); | ||||
| 		goto irq_error; | ||||
| 	} | ||||
| 
 | ||||
| 	pm_runtime_enable(dev->dev); | ||||
| 
 | ||||
| 	pm_runtime_get_sync(dev->dev); | ||||
| 	dpu_mdss->hwversion = readl_relaxed(dpu_mdss->mmio); | ||||
| 	pm_runtime_put_sync(dev->dev); | ||||
| 
 | ||||
| 	priv->mdss = &dpu_mdss->base; | ||||
| 
 | ||||
| 	return ret; | ||||
| 
 | ||||
| irq_error: | ||||
| 	_dpu_mdss_irq_domain_fini(dpu_mdss); | ||||
| irq_domain_error: | ||||
| 	msm_dss_put_clk(mp->clk_config, mp->num_clk); | ||||
| clk_parse_err: | ||||
| 	devm_kfree(&pdev->dev, mp->clk_config); | ||||
| 	if (dpu_mdss->mmio) | ||||
| 		devm_iounmap(&pdev->dev, dpu_mdss->mmio); | ||||
| 	dpu_mdss->mmio = NULL; | ||||
| 	return ret; | ||||
| } | ||||
							
								
								
									
										1963
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_plane.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1963
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_plane.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										175
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_plane.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_plane.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,175 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2015-2018, The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify it | ||||
|  * under the terms of the GNU General Public License version 2 as published by | ||||
|  * the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, but WITHOUT | ||||
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for | ||||
|  * more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License along with | ||||
|  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_PLANE_H_ | ||||
| #define _DPU_PLANE_H_ | ||||
| 
 | ||||
| #include <drm/drm_crtc.h> | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| #include "dpu_hw_mdss.h" | ||||
| #include "dpu_hw_sspp.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_plane_state: Define dpu extension of drm plane state object | ||||
|  * @base:	base drm plane state object | ||||
|  * @property_state: Local storage for msm_prop properties | ||||
|  * @property_values:	cached plane property values | ||||
|  * @aspace:	pointer to address space for input/output buffers | ||||
|  * @input_fence:	dereferenced input fence pointer | ||||
|  * @stage:	assigned by crtc blender | ||||
|  * @multirect_index: index of the rectangle of SSPP | ||||
|  * @multirect_mode: parallel or time multiplex multirect mode | ||||
|  * @pending:	whether the current update is still pending | ||||
|  * @scaler3_cfg: configuration data for scaler3 | ||||
|  * @pixel_ext: configuration data for pixel extensions | ||||
|  * @scaler_check_state: indicates status of user provided pixel extension data | ||||
|  * @cdp_cfg:	CDP configuration | ||||
|  */ | ||||
| struct dpu_plane_state { | ||||
| 	struct drm_plane_state base; | ||||
| 	struct msm_gem_address_space *aspace; | ||||
| 	void *input_fence; | ||||
| 	enum dpu_stage stage; | ||||
| 	uint32_t multirect_index; | ||||
| 	uint32_t multirect_mode; | ||||
| 	bool pending; | ||||
| 
 | ||||
| 	/* scaler configuration */ | ||||
| 	struct dpu_hw_scaler3_cfg scaler3_cfg; | ||||
| 	struct dpu_hw_pixel_ext pixel_ext; | ||||
| 
 | ||||
| 	struct dpu_hw_pipe_cdp_cfg cdp_cfg; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_multirect_plane_states: Defines multirect pair of drm plane states | ||||
|  * @r0: drm plane configured on rect 0 | ||||
|  * @r1: drm plane configured on rect 1 | ||||
|  */ | ||||
| struct dpu_multirect_plane_states { | ||||
| 	const struct drm_plane_state *r0; | ||||
| 	const struct drm_plane_state *r1; | ||||
| }; | ||||
| 
 | ||||
| #define to_dpu_plane_state(x) \ | ||||
| 	container_of(x, struct dpu_plane_state, base) | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_pipe - return sspp identifier for the given plane | ||||
|  * @plane:   Pointer to DRM plane object | ||||
|  * Returns: sspp identifier of the given plane | ||||
|  */ | ||||
| enum dpu_sspp dpu_plane_pipe(struct drm_plane *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * is_dpu_plane_virtual - check for virtual plane | ||||
|  * @plane: Pointer to DRM plane object | ||||
|  * returns: true - if the plane is virtual | ||||
|  *          false - if the plane is primary | ||||
|  */ | ||||
| bool is_dpu_plane_virtual(struct drm_plane *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_get_ctl_flush - get control flush mask | ||||
|  * @plane:   Pointer to DRM plane object | ||||
|  * @ctl: Pointer to control hardware | ||||
|  * @flush_sspp: Pointer to sspp flush control word | ||||
|  */ | ||||
| void dpu_plane_get_ctl_flush(struct drm_plane *plane, struct dpu_hw_ctl *ctl, | ||||
| 		u32 *flush_sspp); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_restore - restore hw state if previously power collapsed | ||||
|  * @plane: Pointer to drm plane structure | ||||
|  */ | ||||
| void dpu_plane_restore(struct drm_plane *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_flush - final plane operations before commit flush | ||||
|  * @plane: Pointer to drm plane structure | ||||
|  */ | ||||
| void dpu_plane_flush(struct drm_plane *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_kickoff - final plane operations before commit kickoff | ||||
|  * @plane: Pointer to drm plane structure | ||||
|  */ | ||||
| void dpu_plane_kickoff(struct drm_plane *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_set_error: enable/disable error condition | ||||
|  * @plane: pointer to drm_plane structure | ||||
|  */ | ||||
| void dpu_plane_set_error(struct drm_plane *plane, bool error); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_init - create new dpu plane for the given pipe | ||||
|  * @dev:   Pointer to DRM device | ||||
|  * @pipe:  dpu hardware pipe identifier | ||||
|  * @primary_plane: true if this pipe is primary plane for crtc | ||||
|  * @possible_crtcs: bitmask of crtc that can be attached to the given pipe | ||||
|  * @master_plane_id: primary plane id of a multirect pipe. 0 value passed for | ||||
|  *                   a regular plane initialization. A non-zero primary plane | ||||
|  *                   id will be passed for a virtual pipe initialization. | ||||
|  * | ||||
|  */ | ||||
| struct drm_plane *dpu_plane_init(struct drm_device *dev, | ||||
| 		uint32_t pipe, bool primary_plane, | ||||
| 		unsigned long possible_crtcs, u32 master_plane_id); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_validate_multirecti_v2 - validate the multirect planes | ||||
|  *				      against hw limitations | ||||
|  * @plane: drm plate states of the multirect pair | ||||
|  */ | ||||
| int dpu_plane_validate_multirect_v2(struct dpu_multirect_plane_states *plane); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_clear_multirect - clear multirect bits for the given pipe | ||||
|  * @drm_state: Pointer to DRM plane state | ||||
|  */ | ||||
| void dpu_plane_clear_multirect(const struct drm_plane_state *drm_state); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_wait_input_fence - wait for input fence object | ||||
|  * @plane:   Pointer to DRM plane object | ||||
|  * @wait_ms: Wait timeout value | ||||
|  * Returns: Zero on success | ||||
|  */ | ||||
| int dpu_plane_wait_input_fence(struct drm_plane *plane, uint32_t wait_ms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_color_fill - enables color fill on plane | ||||
|  * @plane:  Pointer to DRM plane object | ||||
|  * @color:  RGB fill color value, [23..16] Blue, [15..8] Green, [7..0] Red | ||||
|  * @alpha:  8-bit fill alpha value, 255 selects 100% alpha | ||||
|  * Returns: 0 on success | ||||
|  */ | ||||
| int dpu_plane_color_fill(struct drm_plane *plane, | ||||
| 		uint32_t color, uint32_t alpha); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_plane_set_revalidate - sets revalidate flag which forces a full | ||||
|  *	validation of the plane properties in the next atomic check | ||||
|  * @plane: Pointer to DRM plane object | ||||
|  * @enable: Boolean to set/unset the flag | ||||
|  */ | ||||
| void dpu_plane_set_revalidate(struct drm_plane *plane, bool enable); | ||||
| 
 | ||||
| #endif /* _DPU_PLANE_H_ */ | ||||
							
								
								
									
										249
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_power_handle.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										249
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_power_handle.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,249 @@ | ||||
| /* Copyright (c) 2014-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d]: " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include <linux/kernel.h> | ||||
| #include <linux/of.h> | ||||
| #include <linux/string.h> | ||||
| #include <linux/of_address.h> | ||||
| #include <linux/slab.h> | ||||
| #include <linux/mutex.h> | ||||
| #include <linux/of_platform.h> | ||||
| 
 | ||||
| #include "dpu_power_handle.h" | ||||
| #include "dpu_trace.h" | ||||
| 
 | ||||
| static const char *data_bus_name[DPU_POWER_HANDLE_DBUS_ID_MAX] = { | ||||
| 	[DPU_POWER_HANDLE_DBUS_ID_MNOC] = "qcom,dpu-data-bus", | ||||
| 	[DPU_POWER_HANDLE_DBUS_ID_LLCC] = "qcom,dpu-llcc-bus", | ||||
| 	[DPU_POWER_HANDLE_DBUS_ID_EBI] = "qcom,dpu-ebi-bus", | ||||
| }; | ||||
| 
 | ||||
| const char *dpu_power_handle_get_dbus_name(u32 bus_id) | ||||
| { | ||||
| 	if (bus_id < DPU_POWER_HANDLE_DBUS_ID_MAX) | ||||
| 		return data_bus_name[bus_id]; | ||||
| 
 | ||||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| static void dpu_power_event_trigger_locked(struct dpu_power_handle *phandle, | ||||
| 		u32 event_type) | ||||
| { | ||||
| 	struct dpu_power_event *event; | ||||
| 
 | ||||
| 	list_for_each_entry(event, &phandle->event_list, list) { | ||||
| 		if (event->event_type & event_type) | ||||
| 			event->cb_fnc(event_type, event->usr); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| struct dpu_power_client *dpu_power_client_create( | ||||
| 	struct dpu_power_handle *phandle, char *client_name) | ||||
| { | ||||
| 	struct dpu_power_client *client; | ||||
| 	static u32 id; | ||||
| 
 | ||||
| 	if (!client_name || !phandle) { | ||||
| 		pr_err("client name is null or invalid power data\n"); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	client = kzalloc(sizeof(struct dpu_power_client), GFP_KERNEL); | ||||
| 	if (!client) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	mutex_lock(&phandle->phandle_lock); | ||||
| 	strlcpy(client->name, client_name, MAX_CLIENT_NAME_LEN); | ||||
| 	client->usecase_ndx = VOTE_INDEX_DISABLE; | ||||
| 	client->id = id; | ||||
| 	client->active = true; | ||||
| 	pr_debug("client %s created:%pK id :%d\n", client_name, | ||||
| 		client, id); | ||||
| 	id++; | ||||
| 	list_add(&client->list, &phandle->power_client_clist); | ||||
| 	mutex_unlock(&phandle->phandle_lock); | ||||
| 
 | ||||
| 	return client; | ||||
| } | ||||
| 
 | ||||
| void dpu_power_client_destroy(struct dpu_power_handle *phandle, | ||||
| 	struct dpu_power_client *client) | ||||
| { | ||||
| 	if (!client  || !phandle) { | ||||
| 		pr_err("reg bus vote: invalid client handle\n"); | ||||
| 	} else if (!client->active) { | ||||
| 		pr_err("dpu power deinit already done\n"); | ||||
| 		kfree(client); | ||||
| 	} else { | ||||
| 		pr_debug("bus vote client %s destroyed:%pK id:%u\n", | ||||
| 			client->name, client, client->id); | ||||
| 		mutex_lock(&phandle->phandle_lock); | ||||
| 		list_del_init(&client->list); | ||||
| 		mutex_unlock(&phandle->phandle_lock); | ||||
| 		kfree(client); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_power_resource_init(struct platform_device *pdev, | ||||
| 	struct dpu_power_handle *phandle) | ||||
| { | ||||
| 	phandle->dev = &pdev->dev; | ||||
| 
 | ||||
| 	INIT_LIST_HEAD(&phandle->power_client_clist); | ||||
| 	INIT_LIST_HEAD(&phandle->event_list); | ||||
| 
 | ||||
| 	mutex_init(&phandle->phandle_lock); | ||||
| } | ||||
| 
 | ||||
| void dpu_power_resource_deinit(struct platform_device *pdev, | ||||
| 	struct dpu_power_handle *phandle) | ||||
| { | ||||
| 	struct dpu_power_client *curr_client, *next_client; | ||||
| 	struct dpu_power_event *curr_event, *next_event; | ||||
| 
 | ||||
| 	if (!phandle || !pdev) { | ||||
| 		pr_err("invalid input param\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	mutex_lock(&phandle->phandle_lock); | ||||
| 	list_for_each_entry_safe(curr_client, next_client, | ||||
| 			&phandle->power_client_clist, list) { | ||||
| 		pr_err("cliend:%s-%d still registered with refcount:%d\n", | ||||
| 				curr_client->name, curr_client->id, | ||||
| 				curr_client->refcount); | ||||
| 		curr_client->active = false; | ||||
| 		list_del(&curr_client->list); | ||||
| 	} | ||||
| 
 | ||||
| 	list_for_each_entry_safe(curr_event, next_event, | ||||
| 			&phandle->event_list, list) { | ||||
| 		pr_err("event:%d, client:%s still registered\n", | ||||
| 				curr_event->event_type, | ||||
| 				curr_event->client_name); | ||||
| 		curr_event->active = false; | ||||
| 		list_del(&curr_event->list); | ||||
| 	} | ||||
| 	mutex_unlock(&phandle->phandle_lock); | ||||
| } | ||||
| 
 | ||||
| int dpu_power_resource_enable(struct dpu_power_handle *phandle, | ||||
| 	struct dpu_power_client *pclient, bool enable) | ||||
| { | ||||
| 	bool changed = false; | ||||
| 	u32 max_usecase_ndx = VOTE_INDEX_DISABLE, prev_usecase_ndx; | ||||
| 	struct dpu_power_client *client; | ||||
| 
 | ||||
| 	if (!phandle || !pclient) { | ||||
| 		pr_err("invalid input argument\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	mutex_lock(&phandle->phandle_lock); | ||||
| 	if (enable) | ||||
| 		pclient->refcount++; | ||||
| 	else if (pclient->refcount) | ||||
| 		pclient->refcount--; | ||||
| 
 | ||||
| 	if (pclient->refcount) | ||||
| 		pclient->usecase_ndx = VOTE_INDEX_LOW; | ||||
| 	else | ||||
| 		pclient->usecase_ndx = VOTE_INDEX_DISABLE; | ||||
| 
 | ||||
| 	list_for_each_entry(client, &phandle->power_client_clist, list) { | ||||
| 		if (client->usecase_ndx < VOTE_INDEX_MAX && | ||||
| 		    client->usecase_ndx > max_usecase_ndx) | ||||
| 			max_usecase_ndx = client->usecase_ndx; | ||||
| 	} | ||||
| 
 | ||||
| 	if (phandle->current_usecase_ndx != max_usecase_ndx) { | ||||
| 		changed = true; | ||||
| 		prev_usecase_ndx = phandle->current_usecase_ndx; | ||||
| 		phandle->current_usecase_ndx = max_usecase_ndx; | ||||
| 	} | ||||
| 
 | ||||
| 	pr_debug("%pS: changed=%d current idx=%d request client %s id:%u enable:%d refcount:%d\n", | ||||
| 		__builtin_return_address(0), changed, max_usecase_ndx, | ||||
| 		pclient->name, pclient->id, enable, pclient->refcount); | ||||
| 
 | ||||
| 	if (!changed) | ||||
| 		goto end; | ||||
| 
 | ||||
| 	if (enable) { | ||||
| 		dpu_power_event_trigger_locked(phandle, | ||||
| 				DPU_POWER_EVENT_PRE_ENABLE); | ||||
| 		dpu_power_event_trigger_locked(phandle, | ||||
| 				DPU_POWER_EVENT_POST_ENABLE); | ||||
| 
 | ||||
| 	} else { | ||||
| 		dpu_power_event_trigger_locked(phandle, | ||||
| 				DPU_POWER_EVENT_PRE_DISABLE); | ||||
| 		dpu_power_event_trigger_locked(phandle, | ||||
| 				DPU_POWER_EVENT_POST_DISABLE); | ||||
| 	} | ||||
| 
 | ||||
| end: | ||||
| 	mutex_unlock(&phandle->phandle_lock); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| struct dpu_power_event *dpu_power_handle_register_event( | ||||
| 		struct dpu_power_handle *phandle, | ||||
| 		u32 event_type, void (*cb_fnc)(u32 event_type, void *usr), | ||||
| 		void *usr, char *client_name) | ||||
| { | ||||
| 	struct dpu_power_event *event; | ||||
| 
 | ||||
| 	if (!phandle) { | ||||
| 		pr_err("invalid power handle\n"); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} else if (!cb_fnc || !event_type) { | ||||
| 		pr_err("no callback fnc or event type\n"); | ||||
| 		return ERR_PTR(-EINVAL); | ||||
| 	} | ||||
| 
 | ||||
| 	event = kzalloc(sizeof(struct dpu_power_event), GFP_KERNEL); | ||||
| 	if (!event) | ||||
| 		return ERR_PTR(-ENOMEM); | ||||
| 
 | ||||
| 	event->event_type = event_type; | ||||
| 	event->cb_fnc = cb_fnc; | ||||
| 	event->usr = usr; | ||||
| 	strlcpy(event->client_name, client_name, MAX_CLIENT_NAME_LEN); | ||||
| 	event->active = true; | ||||
| 
 | ||||
| 	mutex_lock(&phandle->phandle_lock); | ||||
| 	list_add(&event->list, &phandle->event_list); | ||||
| 	mutex_unlock(&phandle->phandle_lock); | ||||
| 
 | ||||
| 	return event; | ||||
| } | ||||
| 
 | ||||
| void dpu_power_handle_unregister_event( | ||||
| 		struct dpu_power_handle *phandle, | ||||
| 		struct dpu_power_event *event) | ||||
| { | ||||
| 	if (!phandle || !event) { | ||||
| 		pr_err("invalid phandle or event\n"); | ||||
| 	} else if (!event->active) { | ||||
| 		pr_err("power handle deinit already done\n"); | ||||
| 		kfree(event); | ||||
| 	} else { | ||||
| 		mutex_lock(&phandle->phandle_lock); | ||||
| 		list_del_init(&event->list); | ||||
| 		mutex_unlock(&phandle->phandle_lock); | ||||
| 		kfree(event); | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										225
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_power_handle.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										225
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_power_handle.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,225 @@ | ||||
| /* Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #ifndef _DPU_POWER_HANDLE_H_ | ||||
| #define _DPU_POWER_HANDLE_H_ | ||||
| 
 | ||||
| #define MAX_CLIENT_NAME_LEN 128 | ||||
| 
 | ||||
| #define DPU_POWER_HANDLE_ENABLE_BUS_AB_QUOTA	0 | ||||
| #define DPU_POWER_HANDLE_DISABLE_BUS_AB_QUOTA	0 | ||||
| #define DPU_POWER_HANDLE_ENABLE_BUS_IB_QUOTA	1600000000 | ||||
| #define DPU_POWER_HANDLE_DISABLE_BUS_IB_QUOTA	0 | ||||
| 
 | ||||
| #include "dpu_io_util.h" | ||||
| 
 | ||||
| /* event will be triggered before power handler disable */ | ||||
| #define DPU_POWER_EVENT_PRE_DISABLE	0x1 | ||||
| 
 | ||||
| /* event will be triggered after power handler disable */ | ||||
| #define DPU_POWER_EVENT_POST_DISABLE	0x2 | ||||
| 
 | ||||
| /* event will be triggered before power handler enable */ | ||||
| #define DPU_POWER_EVENT_PRE_ENABLE	0x4 | ||||
| 
 | ||||
| /* event will be triggered after power handler enable */ | ||||
| #define DPU_POWER_EVENT_POST_ENABLE	0x8 | ||||
| 
 | ||||
| /**
 | ||||
|  * mdss_bus_vote_type: register bus vote type | ||||
|  * VOTE_INDEX_DISABLE: removes the client vote | ||||
|  * VOTE_INDEX_LOW: keeps the lowest vote for register bus | ||||
|  * VOTE_INDEX_MAX: invalid | ||||
|  */ | ||||
| enum mdss_bus_vote_type { | ||||
| 	VOTE_INDEX_DISABLE, | ||||
| 	VOTE_INDEX_LOW, | ||||
| 	VOTE_INDEX_MAX, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_power_handle_data_bus_client - type of axi bus clients | ||||
|  * @DPU_POWER_HANDLE_DATA_BUS_CLIENT_RT: core real-time bus client | ||||
|  * @DPU_POWER_HANDLE_DATA_BUS_CLIENT_NRT: core non-real-time bus client | ||||
|  * @DPU_POWER_HANDLE_DATA_BUS_CLIENT_MAX: maximum number of bus client type | ||||
|  */ | ||||
| enum dpu_power_handle_data_bus_client { | ||||
| 	DPU_POWER_HANDLE_DATA_BUS_CLIENT_RT, | ||||
| 	DPU_POWER_HANDLE_DATA_BUS_CLIENT_NRT, | ||||
| 	DPU_POWER_HANDLE_DATA_BUS_CLIENT_MAX | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum DPU_POWER_HANDLE_DBUS_ID - data bus identifier | ||||
|  * @DPU_POWER_HANDLE_DBUS_ID_MNOC: DPU/MNOC data bus | ||||
|  * @DPU_POWER_HANDLE_DBUS_ID_LLCC: MNOC/LLCC data bus | ||||
|  * @DPU_POWER_HANDLE_DBUS_ID_EBI: LLCC/EBI data bus | ||||
|  */ | ||||
| enum DPU_POWER_HANDLE_DBUS_ID { | ||||
| 	DPU_POWER_HANDLE_DBUS_ID_MNOC, | ||||
| 	DPU_POWER_HANDLE_DBUS_ID_LLCC, | ||||
| 	DPU_POWER_HANDLE_DBUS_ID_EBI, | ||||
| 	DPU_POWER_HANDLE_DBUS_ID_MAX, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_power_client: stores the power client for dpu driver | ||||
|  * @name:	name of the client | ||||
|  * @usecase_ndx: current regs bus vote type | ||||
|  * @refcount:	current refcount if multiple modules are using same | ||||
|  *              same client for enable/disable. Power module will | ||||
|  *              aggregate the refcount and vote accordingly for this | ||||
|  *              client. | ||||
|  * @id:		assigned during create. helps for debugging. | ||||
|  * @list:	list to attach power handle master list | ||||
|  * @ab:         arbitrated bandwidth for each bus client | ||||
|  * @ib:         instantaneous bandwidth for each bus client | ||||
|  * @active:	inidcates the state of dpu power handle | ||||
|  */ | ||||
| struct dpu_power_client { | ||||
| 	char name[MAX_CLIENT_NAME_LEN]; | ||||
| 	short usecase_ndx; | ||||
| 	short refcount; | ||||
| 	u32 id; | ||||
| 	struct list_head list; | ||||
| 	u64 ab[DPU_POWER_HANDLE_DATA_BUS_CLIENT_MAX]; | ||||
| 	u64 ib[DPU_POWER_HANDLE_DATA_BUS_CLIENT_MAX]; | ||||
| 	bool active; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * struct dpu_power_event - local event registration structure | ||||
|  * @client_name: name of the client registering | ||||
|  * @cb_fnc: pointer to desired callback function | ||||
|  * @usr: user pointer to pass to callback event trigger | ||||
|  * @event: refer to DPU_POWER_HANDLE_EVENT_* | ||||
|  * @list: list to attach event master list | ||||
|  * @active: indicates the state of dpu power handle | ||||
|  */ | ||||
| struct dpu_power_event { | ||||
| 	char client_name[MAX_CLIENT_NAME_LEN]; | ||||
| 	void (*cb_fnc)(u32 event_type, void *usr); | ||||
| 	void *usr; | ||||
| 	u32 event_type; | ||||
| 	struct list_head list; | ||||
| 	bool active; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_power_handle: power handle main struct | ||||
|  * @client_clist: master list to store all clients | ||||
|  * @phandle_lock: lock to synchronize the enable/disable | ||||
|  * @dev: pointer to device structure | ||||
|  * @usecase_ndx: current usecase index | ||||
|  * @event_list: current power handle event list | ||||
|  */ | ||||
| struct dpu_power_handle { | ||||
| 	struct list_head power_client_clist; | ||||
| 	struct mutex phandle_lock; | ||||
| 	struct device *dev; | ||||
| 	u32 current_usecase_ndx; | ||||
| 	struct list_head event_list; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_resource_init() - initializes the dpu power handle | ||||
|  * @pdev:   platform device to search the power resources | ||||
|  * @pdata:  power handle to store the power resources | ||||
|  */ | ||||
| void dpu_power_resource_init(struct platform_device *pdev, | ||||
| 	struct dpu_power_handle *pdata); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_resource_deinit() - release the dpu power handle | ||||
|  * @pdev:   platform device for power resources | ||||
|  * @pdata:  power handle containing the resources | ||||
|  * | ||||
|  * Return: error code. | ||||
|  */ | ||||
| void dpu_power_resource_deinit(struct platform_device *pdev, | ||||
| 	struct dpu_power_handle *pdata); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_client_create() - create the client on power handle | ||||
|  * @pdata:  power handle containing the resources | ||||
|  * @client_name: new client name for registration | ||||
|  * | ||||
|  * Return: error code. | ||||
|  */ | ||||
| struct dpu_power_client *dpu_power_client_create(struct dpu_power_handle *pdata, | ||||
| 	char *client_name); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_client_destroy() - destroy the client on power handle | ||||
|  * @pdata:  power handle containing the resources | ||||
|  * @client_name: new client name for registration | ||||
|  * | ||||
|  * Return: none | ||||
|  */ | ||||
| void dpu_power_client_destroy(struct dpu_power_handle *phandle, | ||||
| 	struct dpu_power_client *client); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_resource_enable() - enable/disable the power resources | ||||
|  * @pdata:  power handle containing the resources | ||||
|  * @client: client information to enable/disable its vote | ||||
|  * @enable: boolean request for enable/disable | ||||
|  * | ||||
|  * Return: error code. | ||||
|  */ | ||||
| int dpu_power_resource_enable(struct dpu_power_handle *pdata, | ||||
| 	struct dpu_power_client *pclient, bool enable); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_data_bus_bandwidth_ctrl() - control data bus bandwidth enable | ||||
|  * @phandle:  power handle containing the resources | ||||
|  * @client: client information to bandwidth control | ||||
|  * @enable: true to enable bandwidth for data base | ||||
|  * | ||||
|  * Return: none | ||||
|  */ | ||||
| void dpu_power_data_bus_bandwidth_ctrl(struct dpu_power_handle *phandle, | ||||
| 		struct dpu_power_client *pclient, int enable); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_handle_register_event - register a callback function for an event. | ||||
|  *	Clients can register for multiple events with a single register. | ||||
|  *	Any block with access to phandle can register for the event | ||||
|  *	notification. | ||||
|  * @phandle:	power handle containing the resources | ||||
|  * @event_type:	event type to register; refer DPU_POWER_HANDLE_EVENT_* | ||||
|  * @cb_fnc:	pointer to desired callback function | ||||
|  * @usr:	user pointer to pass to callback on event trigger | ||||
|  * | ||||
|  * Return:	event pointer if success, or error code otherwise | ||||
|  */ | ||||
| struct dpu_power_event *dpu_power_handle_register_event( | ||||
| 		struct dpu_power_handle *phandle, | ||||
| 		u32 event_type, void (*cb_fnc)(u32 event_type, void *usr), | ||||
| 		void *usr, char *client_name); | ||||
| /**
 | ||||
|  * dpu_power_handle_unregister_event - unregister callback for event(s) | ||||
|  * @phandle:	power handle containing the resources | ||||
|  * @event:	event pointer returned after power handle register | ||||
|  */ | ||||
| void dpu_power_handle_unregister_event(struct dpu_power_handle *phandle, | ||||
| 		struct dpu_power_event *event); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_power_handle_get_dbus_name - get name of given data bus identifier | ||||
|  * @bus_id:	data bus identifier | ||||
|  * Return:	Pointer to name string if success; NULL otherwise | ||||
|  */ | ||||
| const char *dpu_power_handle_get_dbus_name(u32 bus_id); | ||||
| 
 | ||||
| #endif /* _DPU_POWER_HANDLE_H_ */ | ||||
							
								
								
									
										1079
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_rm.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1079
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_rm.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										199
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_rm.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_rm.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,199 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2016-2018, The Linux Foundation. All rights reserved. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_RM_H__ | ||||
| #define __DPU_RM_H__ | ||||
| 
 | ||||
| #include <linux/list.h> | ||||
| 
 | ||||
| #include "msm_kms.h" | ||||
| #include "dpu_hw_top.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_rm_topology_name - HW resource use case in use by connector | ||||
|  * @DPU_RM_TOPOLOGY_NONE:                 No topology in use currently | ||||
|  * @DPU_RM_TOPOLOGY_SINGLEPIPE:           1 LM, 1 PP, 1 INTF/WB | ||||
|  * @DPU_RM_TOPOLOGY_DUALPIPE:             2 LM, 2 PP, 2 INTF/WB | ||||
|  * @DPU_RM_TOPOLOGY_DUALPIPE_3DMERGE:     2 LM, 2 PP, 3DMux, 1 INTF/WB | ||||
|  */ | ||||
| enum dpu_rm_topology_name { | ||||
| 	DPU_RM_TOPOLOGY_NONE = 0, | ||||
| 	DPU_RM_TOPOLOGY_SINGLEPIPE, | ||||
| 	DPU_RM_TOPOLOGY_DUALPIPE, | ||||
| 	DPU_RM_TOPOLOGY_DUALPIPE_3DMERGE, | ||||
| 	DPU_RM_TOPOLOGY_MAX, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum dpu_rm_topology_control - HW resource use case in use by connector | ||||
|  * @DPU_RM_TOPCTL_RESERVE_LOCK: If set, in AtomicTest phase, after a successful | ||||
|  *                              test, reserve the resources for this display. | ||||
|  *                              Normal behavior would not impact the reservation | ||||
|  *                              list during the AtomicTest phase. | ||||
|  * @DPU_RM_TOPCTL_RESERVE_CLEAR: If set, in AtomicTest phase, before testing, | ||||
|  *                               release any reservation held by this display. | ||||
|  *                               Normal behavior would not impact the | ||||
|  *                               reservation list during the AtomicTest phase. | ||||
|  * @DPU_RM_TOPCTL_DS  : Require layer mixers with DS capabilities | ||||
|  */ | ||||
| enum dpu_rm_topology_control { | ||||
| 	DPU_RM_TOPCTL_RESERVE_LOCK, | ||||
| 	DPU_RM_TOPCTL_RESERVE_CLEAR, | ||||
| 	DPU_RM_TOPCTL_DS, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_rm - DPU dynamic hardware resource manager | ||||
|  * @dev: device handle for event logging purposes | ||||
|  * @rsvps: list of hardware reservations by each crtc->encoder->connector | ||||
|  * @hw_blks: array of lists of hardware resources present in the system, one | ||||
|  *	list per type of hardware block | ||||
|  * @hw_mdp: hardware object for mdp_top | ||||
|  * @lm_max_width: cached layer mixer maximum width | ||||
|  * @rsvp_next_seq: sequence number for next reservation for debugging purposes | ||||
|  * @rm_lock: resource manager mutex | ||||
|  */ | ||||
| struct dpu_rm { | ||||
| 	struct drm_device *dev; | ||||
| 	struct list_head rsvps; | ||||
| 	struct list_head hw_blks[DPU_HW_BLK_MAX]; | ||||
| 	struct dpu_hw_mdp *hw_mdp; | ||||
| 	uint32_t lm_max_width; | ||||
| 	uint32_t rsvp_next_seq; | ||||
| 	struct mutex rm_lock; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  *  struct dpu_rm_hw_blk - resource manager internal structure | ||||
|  *	forward declaration for single iterator definition without void pointer | ||||
|  */ | ||||
| struct dpu_rm_hw_blk; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_rm_hw_iter - iterator for use with dpu_rm | ||||
|  * @hw: dpu_hw object requested, or NULL on failure | ||||
|  * @blk: dpu_rm internal block representation. Clients ignore. Used as iterator. | ||||
|  * @enc_id: DRM ID of Encoder client wishes to search for, or 0 for Any Encoder | ||||
|  * @type: Hardware Block Type client wishes to search for. | ||||
|  */ | ||||
| struct dpu_rm_hw_iter { | ||||
| 	void *hw; | ||||
| 	struct dpu_rm_hw_blk *blk; | ||||
| 	uint32_t enc_id; | ||||
| 	enum dpu_hw_blk_type type; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_init - Read hardware catalog and create reservation tracking objects | ||||
|  *	for all HW blocks. | ||||
|  * @rm: DPU Resource Manager handle | ||||
|  * @cat: Pointer to hardware catalog | ||||
|  * @mmio: mapped register io address of MDP | ||||
|  * @dev: device handle for event logging purposes | ||||
|  * @Return: 0 on Success otherwise -ERROR | ||||
|  */ | ||||
| int dpu_rm_init(struct dpu_rm *rm, | ||||
| 		struct dpu_mdss_cfg *cat, | ||||
| 		void *mmio, | ||||
| 		struct drm_device *dev); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_destroy - Free all memory allocated by dpu_rm_init | ||||
|  * @rm: DPU Resource Manager handle | ||||
|  * @Return: 0 on Success otherwise -ERROR | ||||
|  */ | ||||
| int dpu_rm_destroy(struct dpu_rm *rm); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_reserve - Given a CRTC->Encoder->Connector display chain, analyze | ||||
|  *	the use connections and user requirements, specified through related | ||||
|  *	topology control properties, and reserve hardware blocks to that | ||||
|  *	display chain. | ||||
|  *	HW blocks can then be accessed through dpu_rm_get_* functions. | ||||
|  *	HW Reservations should be released via dpu_rm_release_hw. | ||||
|  * @rm: DPU Resource Manager handle | ||||
|  * @drm_enc: DRM Encoder handle | ||||
|  * @crtc_state: Proposed Atomic DRM CRTC State handle | ||||
|  * @conn_state: Proposed Atomic DRM Connector State handle | ||||
|  * @topology: Pointer to topology info for the display | ||||
|  * @test_only: Atomic-Test phase, discard results (unless property overrides) | ||||
|  * @Return: 0 on Success otherwise -ERROR | ||||
|  */ | ||||
| int dpu_rm_reserve(struct dpu_rm *rm, | ||||
| 		struct drm_encoder *drm_enc, | ||||
| 		struct drm_crtc_state *crtc_state, | ||||
| 		struct drm_connector_state *conn_state, | ||||
| 		struct msm_display_topology topology, | ||||
| 		bool test_only); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_reserve - Given the encoder for the display chain, release any | ||||
|  *	HW blocks previously reserved for that use case. | ||||
|  * @rm: DPU Resource Manager handle | ||||
|  * @enc: DRM Encoder handle | ||||
|  * @Return: 0 on Success otherwise -ERROR | ||||
|  */ | ||||
| void dpu_rm_release(struct dpu_rm *rm, struct drm_encoder *enc); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_get_mdp - Retrieve HW block for MDP TOP. | ||||
|  *	This is never reserved, and is usable by any display. | ||||
|  * @rm: DPU Resource Manager handle | ||||
|  * @Return: Pointer to hw block or NULL | ||||
|  */ | ||||
| struct dpu_hw_mdp *dpu_rm_get_mdp(struct dpu_rm *rm); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_init_hw_iter - setup given iterator for new iteration over hw list | ||||
|  *	using dpu_rm_get_hw | ||||
|  * @iter: iter object to initialize | ||||
|  * @enc_id: DRM ID of Encoder client wishes to search for, or 0 for Any Encoder | ||||
|  * @type: Hardware Block Type client wishes to search for. | ||||
|  */ | ||||
| void dpu_rm_init_hw_iter( | ||||
| 		struct dpu_rm_hw_iter *iter, | ||||
| 		uint32_t enc_id, | ||||
| 		enum dpu_hw_blk_type type); | ||||
| /**
 | ||||
|  * dpu_rm_get_hw - retrieve reserved hw object given encoder and hw type | ||||
|  *	Meant to do a single pass through the hardware list to iteratively | ||||
|  *	retrieve hardware blocks of a given type for a given encoder. | ||||
|  *	Initialize an iterator object. | ||||
|  *	Set hw block type of interest. Set encoder id of interest, 0 for any. | ||||
|  *	Function returns first hw of type for that encoder. | ||||
|  *	Subsequent calls will return the next reserved hw of that type in-order. | ||||
|  *	Iterator HW pointer will be null on failure to find hw. | ||||
|  * @rm: DPU Resource Manager handle | ||||
|  * @iter: iterator object | ||||
|  * @Return: true on match found, false on no match found | ||||
|  */ | ||||
| bool dpu_rm_get_hw(struct dpu_rm *rm, struct dpu_rm_hw_iter *iter); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_check_property_topctl - validate property bitmask before it is set | ||||
|  * @val: user's proposed topology control bitmask | ||||
|  * @Return: 0 on success or error | ||||
|  */ | ||||
| int dpu_rm_check_property_topctl(uint64_t val); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_rm_get_topology_name - returns the name of the the given topology | ||||
|  *                            definition | ||||
|  * @topology: topology definition | ||||
|  * @Return: name of the topology | ||||
|  */ | ||||
| enum dpu_rm_topology_name | ||||
| dpu_rm_get_topology_name(struct msm_display_topology topology); | ||||
| 
 | ||||
| #endif /* __DPU_RM_H__ */ | ||||
							
								
								
									
										1007
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_trace.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1007
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_trace.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										384
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_vbif.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										384
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_vbif.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,384 @@ | ||||
| /* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #define pr_fmt(fmt)	"[drm:%s:%d] " fmt, __func__, __LINE__ | ||||
| 
 | ||||
| #include <linux/debugfs.h> | ||||
| 
 | ||||
| #include "dpu_vbif.h" | ||||
| #include "dpu_hw_vbif.h" | ||||
| #include "dpu_trace.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * _dpu_vbif_wait_for_xin_halt - wait for the xin to halt | ||||
|  * @vbif:	Pointer to hardware vbif driver | ||||
|  * @xin_id:	Client interface identifier | ||||
|  * @return:	0 if success; error code otherwise | ||||
|  */ | ||||
| static int _dpu_vbif_wait_for_xin_halt(struct dpu_hw_vbif *vbif, u32 xin_id) | ||||
| { | ||||
| 	ktime_t timeout; | ||||
| 	bool status; | ||||
| 	int rc; | ||||
| 
 | ||||
| 	if (!vbif || !vbif->cap || !vbif->ops.get_halt_ctrl) { | ||||
| 		DPU_ERROR("invalid arguments vbif %d\n", vbif != 0); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	timeout = ktime_add_us(ktime_get(), vbif->cap->xin_halt_timeout); | ||||
| 	for (;;) { | ||||
| 		status = vbif->ops.get_halt_ctrl(vbif, xin_id); | ||||
| 		if (status) | ||||
| 			break; | ||||
| 		if (ktime_compare_safe(ktime_get(), timeout) > 0) { | ||||
| 			status = vbif->ops.get_halt_ctrl(vbif, xin_id); | ||||
| 			break; | ||||
| 		} | ||||
| 		usleep_range(501, 1000); | ||||
| 	} | ||||
| 
 | ||||
| 	if (!status) { | ||||
| 		rc = -ETIMEDOUT; | ||||
| 		DPU_ERROR("VBIF %d client %d not halting. TIMEDOUT.\n", | ||||
| 				vbif->idx - VBIF_0, xin_id); | ||||
| 	} else { | ||||
| 		rc = 0; | ||||
| 		DPU_DEBUG("VBIF %d client %d is halted\n", | ||||
| 				vbif->idx - VBIF_0, xin_id); | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * _dpu_vbif_apply_dynamic_ot_limit - determine OT based on usecase parameters | ||||
|  * @vbif:	Pointer to hardware vbif driver | ||||
|  * @ot_lim:	Pointer to OT limit to be modified | ||||
|  * @params:	Pointer to usecase parameters | ||||
|  */ | ||||
| static void _dpu_vbif_apply_dynamic_ot_limit(struct dpu_hw_vbif *vbif, | ||||
| 		u32 *ot_lim, struct dpu_vbif_set_ot_params *params) | ||||
| { | ||||
| 	u64 pps; | ||||
| 	const struct dpu_vbif_dynamic_ot_tbl *tbl; | ||||
| 	u32 i; | ||||
| 
 | ||||
| 	if (!vbif || !(vbif->cap->features & BIT(DPU_VBIF_QOS_OTLIM))) | ||||
| 		return; | ||||
| 
 | ||||
| 	/* Dynamic OT setting done only for WFD */ | ||||
| 	if (!params->is_wfd) | ||||
| 		return; | ||||
| 
 | ||||
| 	pps = params->frame_rate; | ||||
| 	pps *= params->width; | ||||
| 	pps *= params->height; | ||||
| 
 | ||||
| 	tbl = params->rd ? &vbif->cap->dynamic_ot_rd_tbl : | ||||
| 			&vbif->cap->dynamic_ot_wr_tbl; | ||||
| 
 | ||||
| 	for (i = 0; i < tbl->count; i++) { | ||||
| 		if (pps <= tbl->cfg[i].pps) { | ||||
| 			*ot_lim = tbl->cfg[i].ot_limit; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	DPU_DEBUG("vbif:%d xin:%d w:%d h:%d fps:%d pps:%llu ot:%u\n", | ||||
| 			vbif->idx - VBIF_0, params->xin_id, | ||||
| 			params->width, params->height, params->frame_rate, | ||||
| 			pps, *ot_lim); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * _dpu_vbif_get_ot_limit - get OT based on usecase & configuration parameters | ||||
|  * @vbif:	Pointer to hardware vbif driver | ||||
|  * @params:	Pointer to usecase parameters | ||||
|  * @return:	OT limit | ||||
|  */ | ||||
| static u32 _dpu_vbif_get_ot_limit(struct dpu_hw_vbif *vbif, | ||||
| 	struct dpu_vbif_set_ot_params *params) | ||||
| { | ||||
| 	u32 ot_lim = 0; | ||||
| 	u32 val; | ||||
| 
 | ||||
| 	if (!vbif || !vbif->cap) { | ||||
| 		DPU_ERROR("invalid arguments vbif %d\n", vbif != 0); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (vbif->cap->default_ot_wr_limit && !params->rd) | ||||
| 		ot_lim = vbif->cap->default_ot_wr_limit; | ||||
| 	else if (vbif->cap->default_ot_rd_limit && params->rd) | ||||
| 		ot_lim = vbif->cap->default_ot_rd_limit; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * If default ot is not set from dt/catalog, | ||||
| 	 * then do not configure it. | ||||
| 	 */ | ||||
| 	if (ot_lim == 0) | ||||
| 		goto exit; | ||||
| 
 | ||||
| 	/* Modify the limits if the target and the use case requires it */ | ||||
| 	_dpu_vbif_apply_dynamic_ot_limit(vbif, &ot_lim, params); | ||||
| 
 | ||||
| 	if (vbif && vbif->ops.get_limit_conf) { | ||||
| 		val = vbif->ops.get_limit_conf(vbif, | ||||
| 				params->xin_id, params->rd); | ||||
| 		if (val == ot_lim) | ||||
| 			ot_lim = 0; | ||||
| 	} | ||||
| 
 | ||||
| exit: | ||||
| 	DPU_DEBUG("vbif:%d xin:%d ot_lim:%d\n", | ||||
| 			vbif->idx - VBIF_0, params->xin_id, ot_lim); | ||||
| 	return ot_lim; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_vbif_set_ot_limit - set OT based on usecase & configuration parameters | ||||
|  * @vbif:	Pointer to hardware vbif driver | ||||
|  * @params:	Pointer to usecase parameters | ||||
|  * | ||||
|  * Note this function would block waiting for bus halt. | ||||
|  */ | ||||
| void dpu_vbif_set_ot_limit(struct dpu_kms *dpu_kms, | ||||
| 		struct dpu_vbif_set_ot_params *params) | ||||
| { | ||||
| 	struct dpu_hw_vbif *vbif = NULL; | ||||
| 	struct dpu_hw_mdp *mdp; | ||||
| 	bool forced_on = false; | ||||
| 	u32 ot_lim; | ||||
| 	int ret, i; | ||||
| 
 | ||||
| 	if (!dpu_kms) { | ||||
| 		DPU_ERROR("invalid arguments\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	mdp = dpu_kms->hw_mdp; | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(dpu_kms->hw_vbif); i++) { | ||||
| 		if (dpu_kms->hw_vbif[i] && | ||||
| 				dpu_kms->hw_vbif[i]->idx == params->vbif_idx) | ||||
| 			vbif = dpu_kms->hw_vbif[i]; | ||||
| 	} | ||||
| 
 | ||||
| 	if (!vbif || !mdp) { | ||||
| 		DPU_DEBUG("invalid arguments vbif %d mdp %d\n", | ||||
| 				vbif != 0, mdp != 0); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if (!mdp->ops.setup_clk_force_ctrl || | ||||
| 			!vbif->ops.set_limit_conf || | ||||
| 			!vbif->ops.set_halt_ctrl) | ||||
| 		return; | ||||
| 
 | ||||
| 	/* set write_gather_en for all write clients */ | ||||
| 	if (vbif->ops.set_write_gather_en && !params->rd) | ||||
| 		vbif->ops.set_write_gather_en(vbif, params->xin_id); | ||||
| 
 | ||||
| 	ot_lim = _dpu_vbif_get_ot_limit(vbif, params) & 0xFF; | ||||
| 
 | ||||
| 	if (ot_lim == 0) | ||||
| 		goto exit; | ||||
| 
 | ||||
| 	trace_dpu_perf_set_ot(params->num, params->xin_id, ot_lim, | ||||
| 		params->vbif_idx); | ||||
| 
 | ||||
| 	forced_on = mdp->ops.setup_clk_force_ctrl(mdp, params->clk_ctrl, true); | ||||
| 
 | ||||
| 	vbif->ops.set_limit_conf(vbif, params->xin_id, params->rd, ot_lim); | ||||
| 
 | ||||
| 	vbif->ops.set_halt_ctrl(vbif, params->xin_id, true); | ||||
| 
 | ||||
| 	ret = _dpu_vbif_wait_for_xin_halt(vbif, params->xin_id); | ||||
| 	if (ret) | ||||
| 		trace_dpu_vbif_wait_xin_halt_fail(vbif->idx, params->xin_id); | ||||
| 
 | ||||
| 	vbif->ops.set_halt_ctrl(vbif, params->xin_id, false); | ||||
| 
 | ||||
| 	if (forced_on) | ||||
| 		mdp->ops.setup_clk_force_ctrl(mdp, params->clk_ctrl, false); | ||||
| exit: | ||||
| 	return; | ||||
| } | ||||
| 
 | ||||
| void dpu_vbif_set_qos_remap(struct dpu_kms *dpu_kms, | ||||
| 		struct dpu_vbif_set_qos_params *params) | ||||
| { | ||||
| 	struct dpu_hw_vbif *vbif = NULL; | ||||
| 	struct dpu_hw_mdp *mdp; | ||||
| 	bool forced_on = false; | ||||
| 	const struct dpu_vbif_qos_tbl *qos_tbl; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (!dpu_kms || !params || !dpu_kms->hw_mdp) { | ||||
| 		DPU_ERROR("invalid arguments\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	mdp = dpu_kms->hw_mdp; | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(dpu_kms->hw_vbif); i++) { | ||||
| 		if (dpu_kms->hw_vbif[i] && | ||||
| 				dpu_kms->hw_vbif[i]->idx == params->vbif_idx) { | ||||
| 			vbif = dpu_kms->hw_vbif[i]; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if (!vbif || !vbif->cap) { | ||||
| 		DPU_ERROR("invalid vbif %d\n", params->vbif_idx); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if (!vbif->ops.set_qos_remap || !mdp->ops.setup_clk_force_ctrl) { | ||||
| 		DPU_DEBUG("qos remap not supported\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	qos_tbl = params->is_rt ? &vbif->cap->qos_rt_tbl : | ||||
| 			&vbif->cap->qos_nrt_tbl; | ||||
| 
 | ||||
| 	if (!qos_tbl->npriority_lvl || !qos_tbl->priority_lvl) { | ||||
| 		DPU_DEBUG("qos tbl not defined\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	forced_on = mdp->ops.setup_clk_force_ctrl(mdp, params->clk_ctrl, true); | ||||
| 
 | ||||
| 	for (i = 0; i < qos_tbl->npriority_lvl; i++) { | ||||
| 		DPU_DEBUG("vbif:%d xin:%d lvl:%d/%d\n", | ||||
| 				params->vbif_idx, params->xin_id, i, | ||||
| 				qos_tbl->priority_lvl[i]); | ||||
| 		vbif->ops.set_qos_remap(vbif, params->xin_id, i, | ||||
| 				qos_tbl->priority_lvl[i]); | ||||
| 	} | ||||
| 
 | ||||
| 	if (forced_on) | ||||
| 		mdp->ops.setup_clk_force_ctrl(mdp, params->clk_ctrl, false); | ||||
| } | ||||
| 
 | ||||
| void dpu_vbif_clear_errors(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	struct dpu_hw_vbif *vbif; | ||||
| 	u32 i, pnd, src; | ||||
| 
 | ||||
| 	if (!dpu_kms) { | ||||
| 		DPU_ERROR("invalid argument\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(dpu_kms->hw_vbif); i++) { | ||||
| 		vbif = dpu_kms->hw_vbif[i]; | ||||
| 		if (vbif && vbif->ops.clear_errors) { | ||||
| 			vbif->ops.clear_errors(vbif, &pnd, &src); | ||||
| 			if (pnd || src) { | ||||
| 				DRM_DEBUG_KMS("VBIF %d: pnd 0x%X, src 0x%X\n", | ||||
| 					      vbif->idx - VBIF_0, pnd, src); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void dpu_vbif_init_memtypes(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	struct dpu_hw_vbif *vbif; | ||||
| 	int i, j; | ||||
| 
 | ||||
| 	if (!dpu_kms) { | ||||
| 		DPU_ERROR("invalid argument\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = 0; i < ARRAY_SIZE(dpu_kms->hw_vbif); i++) { | ||||
| 		vbif = dpu_kms->hw_vbif[i]; | ||||
| 		if (vbif && vbif->cap && vbif->ops.set_mem_type) { | ||||
| 			for (j = 0; j < vbif->cap->memtype_count; j++) | ||||
| 				vbif->ops.set_mem_type( | ||||
| 						vbif, j, vbif->cap->memtype[j]); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| #ifdef CONFIG_DEBUG_FS | ||||
| void dpu_debugfs_vbif_destroy(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| 	debugfs_remove_recursive(dpu_kms->debugfs_vbif); | ||||
| 	dpu_kms->debugfs_vbif = NULL; | ||||
| } | ||||
| 
 | ||||
| int dpu_debugfs_vbif_init(struct dpu_kms *dpu_kms, struct dentry *debugfs_root) | ||||
| { | ||||
| 	char vbif_name[32]; | ||||
| 	struct dentry *debugfs_vbif; | ||||
| 	int i, j; | ||||
| 
 | ||||
| 	dpu_kms->debugfs_vbif = debugfs_create_dir("vbif", debugfs_root); | ||||
| 	if (!dpu_kms->debugfs_vbif) { | ||||
| 		DPU_ERROR("failed to create vbif debugfs\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = 0; i < dpu_kms->catalog->vbif_count; i++) { | ||||
| 		struct dpu_vbif_cfg *vbif = &dpu_kms->catalog->vbif[i]; | ||||
| 
 | ||||
| 		snprintf(vbif_name, sizeof(vbif_name), "%d", vbif->id); | ||||
| 
 | ||||
| 		debugfs_vbif = debugfs_create_dir(vbif_name, | ||||
| 				dpu_kms->debugfs_vbif); | ||||
| 
 | ||||
| 		debugfs_create_u32("features", 0600, debugfs_vbif, | ||||
| 			(u32 *)&vbif->features); | ||||
| 
 | ||||
| 		debugfs_create_u32("xin_halt_timeout", 0400, debugfs_vbif, | ||||
| 			(u32 *)&vbif->xin_halt_timeout); | ||||
| 
 | ||||
| 		debugfs_create_u32("default_rd_ot_limit", 0400, debugfs_vbif, | ||||
| 			(u32 *)&vbif->default_ot_rd_limit); | ||||
| 
 | ||||
| 		debugfs_create_u32("default_wr_ot_limit", 0400, debugfs_vbif, | ||||
| 			(u32 *)&vbif->default_ot_wr_limit); | ||||
| 
 | ||||
| 		for (j = 0; j < vbif->dynamic_ot_rd_tbl.count; j++) { | ||||
| 			struct dpu_vbif_dynamic_ot_cfg *cfg = | ||||
| 					&vbif->dynamic_ot_rd_tbl.cfg[j]; | ||||
| 
 | ||||
| 			snprintf(vbif_name, sizeof(vbif_name), | ||||
| 					"dynamic_ot_rd_%d_pps", j); | ||||
| 			debugfs_create_u64(vbif_name, 0400, debugfs_vbif, | ||||
| 					(u64 *)&cfg->pps); | ||||
| 			snprintf(vbif_name, sizeof(vbif_name), | ||||
| 					"dynamic_ot_rd_%d_ot_limit", j); | ||||
| 			debugfs_create_u32(vbif_name, 0400, debugfs_vbif, | ||||
| 					(u32 *)&cfg->ot_limit); | ||||
| 		} | ||||
| 
 | ||||
| 		for (j = 0; j < vbif->dynamic_ot_wr_tbl.count; j++) { | ||||
| 			struct dpu_vbif_dynamic_ot_cfg *cfg = | ||||
| 					&vbif->dynamic_ot_wr_tbl.cfg[j]; | ||||
| 
 | ||||
| 			snprintf(vbif_name, sizeof(vbif_name), | ||||
| 					"dynamic_ot_wr_%d_pps", j); | ||||
| 			debugfs_create_u64(vbif_name, 0400, debugfs_vbif, | ||||
| 					(u64 *)&cfg->pps); | ||||
| 			snprintf(vbif_name, sizeof(vbif_name), | ||||
| 					"dynamic_ot_wr_%d_ot_limit", j); | ||||
| 			debugfs_create_u32(vbif_name, 0400, debugfs_vbif, | ||||
| 					(u32 *)&cfg->ot_limit); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| #endif | ||||
							
								
								
									
										94
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_vbif.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								drivers/gpu/drm/msm/disp/dpu1/dpu_vbif.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,94 @@ | ||||
| /* Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
 | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License version 2 and | ||||
|  * only version 2 as published by the Free Software Foundation. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __DPU_VBIF_H__ | ||||
| #define __DPU_VBIF_H__ | ||||
| 
 | ||||
| #include "dpu_kms.h" | ||||
| 
 | ||||
| struct dpu_vbif_set_ot_params { | ||||
| 	u32 xin_id; | ||||
| 	u32 num; | ||||
| 	u32 width; | ||||
| 	u32 height; | ||||
| 	u32 frame_rate; | ||||
| 	bool rd; | ||||
| 	bool is_wfd; | ||||
| 	u32 vbif_idx; | ||||
| 	u32 clk_ctrl; | ||||
| }; | ||||
| 
 | ||||
| struct dpu_vbif_set_memtype_params { | ||||
| 	u32 xin_id; | ||||
| 	u32 vbif_idx; | ||||
| 	u32 clk_ctrl; | ||||
| 	bool is_cacheable; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct dpu_vbif_set_qos_params - QoS remapper parameter | ||||
|  * @vbif_idx: vbif identifier | ||||
|  * @xin_id: client interface identifier | ||||
|  * @clk_ctrl: clock control identifier of the xin | ||||
|  * @num: pipe identifier (debug only) | ||||
|  * @is_rt: true if pipe is used in real-time use case | ||||
|  */ | ||||
| struct dpu_vbif_set_qos_params { | ||||
| 	u32 vbif_idx; | ||||
| 	u32 xin_id; | ||||
| 	u32 clk_ctrl; | ||||
| 	u32 num; | ||||
| 	bool is_rt; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_vbif_set_ot_limit - set OT limit for vbif client | ||||
|  * @dpu_kms:	DPU handler | ||||
|  * @params:	Pointer to OT configuration parameters | ||||
|  */ | ||||
| void dpu_vbif_set_ot_limit(struct dpu_kms *dpu_kms, | ||||
| 		struct dpu_vbif_set_ot_params *params); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_vbif_set_qos_remap - set QoS priority level remap | ||||
|  * @dpu_kms:	DPU handler | ||||
|  * @params:	Pointer to QoS configuration parameters | ||||
|  */ | ||||
| void dpu_vbif_set_qos_remap(struct dpu_kms *dpu_kms, | ||||
| 		struct dpu_vbif_set_qos_params *params); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_vbif_clear_errors - clear any vbif errors | ||||
|  * @dpu_kms:	DPU handler | ||||
|  */ | ||||
| void dpu_vbif_clear_errors(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| /**
 | ||||
|  * dpu_vbif_init_memtypes - initialize xin memory types for vbif | ||||
|  * @dpu_kms:	DPU handler | ||||
|  */ | ||||
| void dpu_vbif_init_memtypes(struct dpu_kms *dpu_kms); | ||||
| 
 | ||||
| #ifdef CONFIG_DEBUG_FS | ||||
| int dpu_debugfs_vbif_init(struct dpu_kms *dpu_kms, struct dentry *debugfs_root); | ||||
| void dpu_debugfs_vbif_destroy(struct dpu_kms *dpu_kms); | ||||
| #else | ||||
| static inline int dpu_debugfs_vbif_init(struct dpu_kms *dpu_kms, | ||||
| 		struct dentry *debugfs_root) | ||||
| { | ||||
| 	return 0; | ||||
| } | ||||
| static inline void dpu_debugfs_vbif_destroy(struct dpu_kms *dpu_kms) | ||||
| { | ||||
| } | ||||
| #endif | ||||
| #endif /* __DPU_VBIF_H__ */ | ||||
							
								
								
									
										1376
									
								
								drivers/gpu/drm/msm/disp/dpu1/msm_media_info.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1376
									
								
								drivers/gpu/drm/msm/disp/dpu1/msm_media_info.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -1,4 +1,5 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2016-2018, The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
| @ -15,6 +16,8 @@ | ||||
|  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| #include <linux/kthread.h> | ||||
| #include <uapi/linux/sched/types.h> | ||||
| #include <drm/drm_of.h> | ||||
| 
 | ||||
| #include "msm_drv.h" | ||||
| @ -149,7 +152,7 @@ struct vblank_event { | ||||
| 	bool enable; | ||||
| }; | ||||
| 
 | ||||
| static void vblank_ctrl_worker(struct work_struct *work) | ||||
| static void vblank_ctrl_worker(struct kthread_work *work) | ||||
| { | ||||
| 	struct msm_vblank_ctrl *vbl_ctrl = container_of(work, | ||||
| 						struct msm_vblank_ctrl, work); | ||||
| @ -197,7 +200,8 @@ static int vblank_ctrl_queue_work(struct msm_drm_private *priv, | ||||
| 	list_add_tail(&vbl_ev->node, &vbl_ctrl->event_list); | ||||
| 	spin_unlock_irqrestore(&vbl_ctrl->lock, flags); | ||||
| 
 | ||||
| 	queue_work(priv->wq, &vbl_ctrl->work); | ||||
| 	kthread_queue_work(&priv->disp_thread[crtc_id].worker, | ||||
| 			&vbl_ctrl->work); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| @ -211,17 +215,33 @@ static int msm_drm_uninit(struct device *dev) | ||||
| 	struct msm_mdss *mdss = priv->mdss; | ||||
| 	struct msm_vblank_ctrl *vbl_ctrl = &priv->vblank_ctrl; | ||||
| 	struct vblank_event *vbl_ev, *tmp; | ||||
| 	int i; | ||||
| 
 | ||||
| 	/* We must cancel and cleanup any pending vblank enable/disable
 | ||||
| 	 * work before drm_irq_uninstall() to avoid work re-enabling an | ||||
| 	 * irq after uninstall has disabled it. | ||||
| 	 */ | ||||
| 	cancel_work_sync(&vbl_ctrl->work); | ||||
| 	kthread_flush_work(&vbl_ctrl->work); | ||||
| 	list_for_each_entry_safe(vbl_ev, tmp, &vbl_ctrl->event_list, node) { | ||||
| 		list_del(&vbl_ev->node); | ||||
| 		kfree(vbl_ev); | ||||
| 	} | ||||
| 
 | ||||
| 	/* clean up display commit/event worker threads */ | ||||
| 	for (i = 0; i < priv->num_crtcs; i++) { | ||||
| 		if (priv->disp_thread[i].thread) { | ||||
| 			kthread_flush_worker(&priv->disp_thread[i].worker); | ||||
| 			kthread_stop(priv->disp_thread[i].thread); | ||||
| 			priv->disp_thread[i].thread = NULL; | ||||
| 		} | ||||
| 
 | ||||
| 		if (priv->event_thread[i].thread) { | ||||
| 			kthread_flush_worker(&priv->event_thread[i].worker); | ||||
| 			kthread_stop(priv->event_thread[i].thread); | ||||
| 			priv->event_thread[i].thread = NULL; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	msm_gem_shrinker_cleanup(ddev); | ||||
| 
 | ||||
| 	drm_kms_helper_poll_fini(ddev); | ||||
| @ -269,6 +289,7 @@ static int msm_drm_uninit(struct device *dev) | ||||
| 
 | ||||
| #define KMS_MDP4 4 | ||||
| #define KMS_MDP5 5 | ||||
| #define KMS_DPU  3 | ||||
| 
 | ||||
| static int get_mdp_ver(struct platform_device *pdev) | ||||
| { | ||||
| @ -360,7 +381,8 @@ static int msm_drm_init(struct device *dev, struct drm_driver *drv) | ||||
| 	struct msm_drm_private *priv; | ||||
| 	struct msm_kms *kms; | ||||
| 	struct msm_mdss *mdss; | ||||
| 	int ret; | ||||
| 	int ret, i; | ||||
| 	struct sched_param param; | ||||
| 
 | ||||
| 	ddev = drm_dev_alloc(drv, dev); | ||||
| 	if (IS_ERR(ddev)) { | ||||
| @ -379,7 +401,17 @@ static int msm_drm_init(struct device *dev, struct drm_driver *drv) | ||||
| 	ddev->dev_private = priv; | ||||
| 	priv->dev = ddev; | ||||
| 
 | ||||
| 	ret = mdp5_mdss_init(ddev); | ||||
| 	switch (get_mdp_ver(pdev)) { | ||||
| 	case KMS_MDP5: | ||||
| 		ret = mdp5_mdss_init(ddev); | ||||
| 		break; | ||||
| 	case KMS_DPU: | ||||
| 		ret = dpu_mdss_init(ddev); | ||||
| 		break; | ||||
| 	default: | ||||
| 		ret = 0; | ||||
| 		break; | ||||
| 	} | ||||
| 	if (ret) | ||||
| 		goto err_free_priv; | ||||
| 
 | ||||
| @ -389,7 +421,7 @@ static int msm_drm_init(struct device *dev, struct drm_driver *drv) | ||||
| 
 | ||||
| 	INIT_LIST_HEAD(&priv->inactive_list); | ||||
| 	INIT_LIST_HEAD(&priv->vblank_ctrl.event_list); | ||||
| 	INIT_WORK(&priv->vblank_ctrl.work, vblank_ctrl_worker); | ||||
| 	kthread_init_work(&priv->vblank_ctrl.work, vblank_ctrl_worker); | ||||
| 	spin_lock_init(&priv->vblank_ctrl.lock); | ||||
| 
 | ||||
| 	drm_mode_config_init(ddev); | ||||
| @ -413,6 +445,10 @@ static int msm_drm_init(struct device *dev, struct drm_driver *drv) | ||||
| 	case KMS_MDP5: | ||||
| 		kms = mdp5_kms_init(ddev); | ||||
| 		break; | ||||
| 	case KMS_DPU: | ||||
| 		kms = dpu_kms_init(ddev); | ||||
| 		priv->kms = kms; | ||||
| 		break; | ||||
| 	default: | ||||
| 		kms = ERR_PTR(-ENODEV); | ||||
| 		break; | ||||
| @ -444,6 +480,79 @@ static int msm_drm_init(struct device *dev, struct drm_driver *drv) | ||||
| 	ddev->mode_config.funcs = &mode_config_funcs; | ||||
| 	ddev->mode_config.helper_private = &mode_config_helper_funcs; | ||||
| 
 | ||||
| 	/**
 | ||||
| 	 * this priority was found during empiric testing to have appropriate | ||||
| 	 * realtime scheduling to process display updates and interact with | ||||
| 	 * other real time and normal priority task | ||||
| 	 */ | ||||
| 	param.sched_priority = 16; | ||||
| 	for (i = 0; i < priv->num_crtcs; i++) { | ||||
| 
 | ||||
| 		/* initialize display thread */ | ||||
| 		priv->disp_thread[i].crtc_id = priv->crtcs[i]->base.id; | ||||
| 		kthread_init_worker(&priv->disp_thread[i].worker); | ||||
| 		priv->disp_thread[i].dev = ddev; | ||||
| 		priv->disp_thread[i].thread = | ||||
| 			kthread_run(kthread_worker_fn, | ||||
| 				&priv->disp_thread[i].worker, | ||||
| 				"crtc_commit:%d", priv->disp_thread[i].crtc_id); | ||||
| 		ret = sched_setscheduler(priv->disp_thread[i].thread, | ||||
| 							SCHED_FIFO, ¶m); | ||||
| 		if (ret) | ||||
| 			pr_warn("display thread priority update failed: %d\n", | ||||
| 									ret); | ||||
| 
 | ||||
| 		if (IS_ERR(priv->disp_thread[i].thread)) { | ||||
| 			dev_err(dev, "failed to create crtc_commit kthread\n"); | ||||
| 			priv->disp_thread[i].thread = NULL; | ||||
| 		} | ||||
| 
 | ||||
| 		/* initialize event thread */ | ||||
| 		priv->event_thread[i].crtc_id = priv->crtcs[i]->base.id; | ||||
| 		kthread_init_worker(&priv->event_thread[i].worker); | ||||
| 		priv->event_thread[i].dev = ddev; | ||||
| 		priv->event_thread[i].thread = | ||||
| 			kthread_run(kthread_worker_fn, | ||||
| 				&priv->event_thread[i].worker, | ||||
| 				"crtc_event:%d", priv->event_thread[i].crtc_id); | ||||
| 		/**
 | ||||
| 		 * event thread should also run at same priority as disp_thread | ||||
| 		 * because it is handling frame_done events. A lower priority | ||||
| 		 * event thread and higher priority disp_thread can causes | ||||
| 		 * frame_pending counters beyond 2. This can lead to commit | ||||
| 		 * failure at crtc commit level. | ||||
| 		 */ | ||||
| 		ret = sched_setscheduler(priv->event_thread[i].thread, | ||||
| 							SCHED_FIFO, ¶m); | ||||
| 		if (ret) | ||||
| 			pr_warn("display event thread priority update failed: %d\n", | ||||
| 									ret); | ||||
| 
 | ||||
| 		if (IS_ERR(priv->event_thread[i].thread)) { | ||||
| 			dev_err(dev, "failed to create crtc_event kthread\n"); | ||||
| 			priv->event_thread[i].thread = NULL; | ||||
| 		} | ||||
| 
 | ||||
| 		if ((!priv->disp_thread[i].thread) || | ||||
| 				!priv->event_thread[i].thread) { | ||||
| 			/* clean up previously created threads if any */ | ||||
| 			for ( ; i >= 0; i--) { | ||||
| 				if (priv->disp_thread[i].thread) { | ||||
| 					kthread_stop( | ||||
| 						priv->disp_thread[i].thread); | ||||
| 					priv->disp_thread[i].thread = NULL; | ||||
| 				} | ||||
| 
 | ||||
| 				if (priv->event_thread[i].thread) { | ||||
| 					kthread_stop( | ||||
| 						priv->event_thread[i].thread); | ||||
| 					priv->event_thread[i].thread = NULL; | ||||
| 				} | ||||
| 			} | ||||
| 			goto err_msm_uninit; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	ret = drm_vblank_init(ddev, priv->num_crtcs); | ||||
| 	if (ret < 0) { | ||||
| 		dev_err(dev, "failed to initialize vblank\n"); | ||||
| @ -1060,12 +1169,13 @@ static int add_display_components(struct device *dev, | ||||
| 	int ret; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * MDP5 based devices don't have a flat hierarchy. There is a top level | ||||
| 	 * parent: MDSS, and children: MDP5, DSI, HDMI, eDP etc. Populate the | ||||
| 	 * children devices, find the MDP5 node, and then add the interfaces | ||||
| 	 * to our components list. | ||||
| 	 * MDP5/DPU based devices don't have a flat hierarchy. There is a top | ||||
| 	 * level parent: MDSS, and children: MDP5/DPU, DSI, HDMI, eDP etc. | ||||
| 	 * Populate the children devices, find the MDP5/DPU node, and then add | ||||
| 	 * the interfaces to our components list. | ||||
| 	 */ | ||||
| 	if (of_device_is_compatible(dev->of_node, "qcom,mdss")) { | ||||
| 	if (of_device_is_compatible(dev->of_node, "qcom,mdss") || | ||||
| 	    of_device_is_compatible(dev->of_node, "qcom,sdm845-mdss")) { | ||||
| 		ret = of_platform_populate(dev->of_node, NULL, NULL, dev); | ||||
| 		if (ret) { | ||||
| 			dev_err(dev, "failed to populate children devices\n"); | ||||
| @ -1177,6 +1287,7 @@ static int msm_pdev_remove(struct platform_device *pdev) | ||||
| static const struct of_device_id dt_match[] = { | ||||
| 	{ .compatible = "qcom,mdp4", .data = (void *)KMS_MDP4 }, | ||||
| 	{ .compatible = "qcom,mdss", .data = (void *)KMS_MDP5 }, | ||||
| 	{ .compatible = "qcom,sdm845-mdss", .data = (void *)KMS_DPU }, | ||||
| 	{} | ||||
| }; | ||||
| MODULE_DEVICE_TABLE(of, dt_match); | ||||
| @ -1198,6 +1309,7 @@ static int __init msm_drm_register(void) | ||||
| 
 | ||||
| 	DBG("init"); | ||||
| 	msm_mdp_register(); | ||||
| 	msm_dpu_register(); | ||||
| 	msm_dsi_register(); | ||||
| 	msm_edp_register(); | ||||
| 	msm_hdmi_register(); | ||||
| @ -1214,6 +1326,7 @@ static void __exit msm_drm_unregister(void) | ||||
| 	msm_edp_unregister(); | ||||
| 	msm_dsi_unregister(); | ||||
| 	msm_mdp_unregister(); | ||||
| 	msm_dpu_unregister(); | ||||
| } | ||||
| 
 | ||||
| module_init(msm_drm_register); | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2016-2018, The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
| @ -33,6 +34,7 @@ | ||||
| #include <linux/of_graph.h> | ||||
| #include <linux/of_device.h> | ||||
| #include <asm/sizes.h> | ||||
| #include <linux/kthread.h> | ||||
| 
 | ||||
| #include <drm/drmP.h> | ||||
| #include <drm/drm_atomic.h> | ||||
| @ -55,7 +57,7 @@ struct msm_gem_address_space; | ||||
| struct msm_gem_vma; | ||||
| 
 | ||||
| #define MAX_CRTCS      8 | ||||
| #define MAX_PLANES     16 | ||||
| #define MAX_PLANES     20 | ||||
| #define MAX_ENCODERS   8 | ||||
| #define MAX_BRIDGES    8 | ||||
| #define MAX_CONNECTORS 8 | ||||
| @ -74,12 +76,77 @@ enum msm_mdp_plane_property { | ||||
| }; | ||||
| 
 | ||||
| struct msm_vblank_ctrl { | ||||
| 	struct work_struct work; | ||||
| 	struct kthread_work work; | ||||
| 	struct list_head event_list; | ||||
| 	spinlock_t lock; | ||||
| }; | ||||
| 
 | ||||
| #define MSM_GPU_MAX_RINGS 4 | ||||
| #define MAX_H_TILES_PER_DISPLAY 2 | ||||
| 
 | ||||
| /**
 | ||||
|  * enum msm_display_caps - features/capabilities supported by displays | ||||
|  * @MSM_DISPLAY_CAP_VID_MODE:           Video or "active" mode supported | ||||
|  * @MSM_DISPLAY_CAP_CMD_MODE:           Command mode supported | ||||
|  * @MSM_DISPLAY_CAP_HOT_PLUG:           Hot plug detection supported | ||||
|  * @MSM_DISPLAY_CAP_EDID:               EDID supported | ||||
|  */ | ||||
| enum msm_display_caps { | ||||
| 	MSM_DISPLAY_CAP_VID_MODE	= BIT(0), | ||||
| 	MSM_DISPLAY_CAP_CMD_MODE	= BIT(1), | ||||
| 	MSM_DISPLAY_CAP_HOT_PLUG	= BIT(2), | ||||
| 	MSM_DISPLAY_CAP_EDID		= BIT(3), | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * enum msm_event_wait - type of HW events to wait for | ||||
|  * @MSM_ENC_COMMIT_DONE - wait for the driver to flush the registers to HW | ||||
|  * @MSM_ENC_TX_COMPLETE - wait for the HW to transfer the frame to panel | ||||
|  * @MSM_ENC_VBLANK - wait for the HW VBLANK event (for driver-internal waiters) | ||||
|  */ | ||||
| enum msm_event_wait { | ||||
| 	MSM_ENC_COMMIT_DONE = 0, | ||||
| 	MSM_ENC_TX_COMPLETE, | ||||
| 	MSM_ENC_VBLANK, | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct msm_display_topology - defines a display topology pipeline | ||||
|  * @num_lm:       number of layer mixers used | ||||
|  * @num_enc:      number of compression encoder blocks used | ||||
|  * @num_intf:     number of interfaces the panel is mounted on | ||||
|  */ | ||||
| struct msm_display_topology { | ||||
| 	u32 num_lm; | ||||
| 	u32 num_enc; | ||||
| 	u32 num_intf; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct msm_display_info - defines display properties | ||||
|  * @intf_type:          DRM_MODE_CONNECTOR_ display type | ||||
|  * @capabilities:       Bitmask of display flags | ||||
|  * @num_of_h_tiles:     Number of horizontal tiles in case of split interface | ||||
|  * @h_tile_instance:    Controller instance used per tile. Number of elements is | ||||
|  *                      based on num_of_h_tiles | ||||
|  * @is_te_using_watchdog_timer:  Boolean to indicate watchdog TE is | ||||
|  *				 used instead of panel TE in cmd mode panels | ||||
|  */ | ||||
| struct msm_display_info { | ||||
| 	int intf_type; | ||||
| 	uint32_t capabilities; | ||||
| 	uint32_t num_of_h_tiles; | ||||
| 	uint32_t h_tile_instance[MAX_H_TILES_PER_DISPLAY]; | ||||
| 	bool is_te_using_watchdog_timer; | ||||
| }; | ||||
| 
 | ||||
| /* Commit/Event thread specific structure */ | ||||
| struct msm_drm_thread { | ||||
| 	struct drm_device *dev; | ||||
| 	struct task_struct *thread; | ||||
| 	unsigned int crtc_id; | ||||
| 	struct kthread_worker worker; | ||||
| }; | ||||
| 
 | ||||
| struct msm_drm_private { | ||||
| 
 | ||||
| @ -90,7 +157,7 @@ struct msm_drm_private { | ||||
| 	/* subordinate devices, if present: */ | ||||
| 	struct platform_device *gpu_pdev; | ||||
| 
 | ||||
| 	/* top level MDSS wrapper device (for MDP5 only) */ | ||||
| 	/* top level MDSS wrapper device (for MDP5/DPU only) */ | ||||
| 	struct msm_mdss *mdss; | ||||
| 
 | ||||
| 	/* possibly this should be in the kms component, but it is
 | ||||
| @ -128,6 +195,9 @@ struct msm_drm_private { | ||||
| 	unsigned int num_crtcs; | ||||
| 	struct drm_crtc *crtcs[MAX_CRTCS]; | ||||
| 
 | ||||
| 	struct msm_drm_thread disp_thread[MAX_CRTCS]; | ||||
| 	struct msm_drm_thread event_thread[MAX_CRTCS]; | ||||
| 
 | ||||
| 	unsigned int num_encoders; | ||||
| 	struct drm_encoder *encoders[MAX_ENCODERS]; | ||||
| 
 | ||||
| @ -180,6 +250,9 @@ struct msm_gem_address_space * | ||||
| msm_gem_address_space_create(struct device *dev, struct iommu_domain *domain, | ||||
| 		const char *name); | ||||
| 
 | ||||
| int msm_register_mmu(struct drm_device *dev, struct msm_mmu *mmu); | ||||
| void msm_unregister_mmu(struct drm_device *dev, struct msm_mmu *mmu); | ||||
| 
 | ||||
| void msm_gem_submit_free(struct msm_gem_submit *submit); | ||||
| int msm_ioctl_gem_submit(struct drm_device *dev, void *data, | ||||
| 		struct drm_file *file); | ||||
| @ -291,6 +364,8 @@ static inline int msm_dsi_modeset_init(struct msm_dsi *msm_dsi, | ||||
| 
 | ||||
| void __init msm_mdp_register(void); | ||||
| void __exit msm_mdp_unregister(void); | ||||
| void __init msm_dpu_register(void); | ||||
| void __exit msm_dpu_unregister(void); | ||||
| 
 | ||||
| #ifdef CONFIG_DEBUG_FS | ||||
| void msm_gem_describe(struct drm_gem_object *obj, struct seq_file *m); | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| /*
 | ||||
|  * Copyright (c) 2016-2018, The Linux Foundation. All rights reserved. | ||||
|  * Copyright (C) 2013 Red Hat | ||||
|  * Author: Rob Clark <robdclark@gmail.com> | ||||
|  * | ||||
| @ -51,6 +52,11 @@ struct msm_kms_funcs { | ||||
| 	const struct msm_format *(*get_format)(struct msm_kms *kms, | ||||
| 					const uint32_t format, | ||||
| 					const uint64_t modifiers); | ||||
| 	/* do format checking on format modified through fb_cmd2 modifiers */ | ||||
| 	int (*check_modified_format)(const struct msm_kms *kms, | ||||
| 			const struct msm_format *msm_fmt, | ||||
| 			const struct drm_mode_fb_cmd2 *cmd, | ||||
| 			struct drm_gem_object **bos); | ||||
| 	/* misc: */ | ||||
| 	long (*round_pixclk)(struct msm_kms *kms, unsigned long rate, | ||||
| 			struct drm_encoder *encoder); | ||||
| @ -90,6 +96,7 @@ static inline void msm_kms_init(struct msm_kms *kms, | ||||
| 
 | ||||
| struct msm_kms *mdp4_kms_init(struct drm_device *dev); | ||||
| struct msm_kms *mdp5_kms_init(struct drm_device *dev); | ||||
| struct msm_kms *dpu_kms_init(struct drm_device *dev); | ||||
| 
 | ||||
| struct msm_mdss_funcs { | ||||
| 	int (*enable)(struct msm_mdss *mdss); | ||||
| @ -103,5 +110,6 @@ struct msm_mdss { | ||||
| }; | ||||
| 
 | ||||
| int mdp5_mdss_init(struct drm_device *dev); | ||||
| int dpu_mdss_init(struct drm_device *dev); | ||||
| 
 | ||||
| #endif /* __MSM_KMS_H__ */ | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user