
There is little sense in reading interrupt statuses and right after that going after the array of statuses to dispatch them. Merge both loops into single function doing read and dispatch. Change-Id: I1259476549bcaf9f9f4e12591a7e182796e150dd Signed-off-by: Dmitry Baryshkov <dmitry.baryshkov@linaro.org> Reviewed-by: Bjorn Andersson <bjorn.andersson@linaro.org> Git-commit: 0abdba47dc1df708c365421d481734d3f7fecb01 Git-repo: git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git Signed-off-by: Veera Sundaram Sankaran <quic_veeras@quicinc.com>
285 sor
9.2 KiB
C
285 sor
9.2 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
/*
|
|
* Copyright (c) 2021-2022 Qualcomm Innovation Center, Inc. All rights reserved.
|
|
* Copyright (c) 2016-2019, 2021, The Linux Foundation. All rights reserved.
|
|
*/
|
|
|
|
#ifndef _SDE_HW_INTERRUPTS_H
|
|
#define _SDE_HW_INTERRUPTS_H
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include "sde_hwio.h"
|
|
#include "sde_hw_catalog.h"
|
|
#include "sde_hw_util.h"
|
|
#include "sde_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)
|
|
|
|
/**
|
|
* sde_intr_type - HW Interrupt Type
|
|
* @SDE_IRQ_TYPE_WB_ROT_COMP: WB rotator done
|
|
* @SDE_IRQ_TYPE_WB_WFD_COMP: WB WFD done
|
|
* @SDE_IRQ_TYPE_PING_PONG_COMP: PingPong done
|
|
* @SDE_IRQ_TYPE_PING_PONG_RD_PTR: PingPong read pointer
|
|
* @SDE_IRQ_TYPE_PING_PONG_WR_PTR: PingPong write pointer
|
|
* @SDE_IRQ_TYPE_PING_PONG_AUTO_REF: PingPong auto refresh
|
|
* @SDE_IRQ_TYPE_PING_PONG_TEAR_CHECK: PingPong Tear check
|
|
* @SDE_IRQ_TYPE_PING_PONG_TE_CHECK: PingPong TE detection
|
|
* @SDE_IRQ_TYPE_INTF_UNDER_RUN: INTF underrun
|
|
* @SDE_IRQ_TYPE_INTF_VSYNC: INTF VSYNC
|
|
* @SDE_IRQ_TYPE_CWB_OVERFLOW: Concurrent WB overflow
|
|
* @SDE_IRQ_TYPE_HIST_VIG_DONE: VIG Histogram done
|
|
* @SDE_IRQ_TYPE_HIST_VIG_RSTSEQ: VIG Histogram reset
|
|
* @SDE_IRQ_TYPE_HIST_DSPP_DONE: DSPP Histogram done
|
|
* @SDE_IRQ_TYPE_HIST_DSPP_RSTSEQ: DSPP Histogram reset
|
|
* @SDE_IRQ_TYPE_WD_TIMER: Watchdog timer
|
|
* @SDE_IRQ_TYPE_SFI_VIDEO_IN: Video static frame INTR into static
|
|
* @SDE_IRQ_TYPE_SFI_VIDEO_OUT: Video static frame INTR out-of static
|
|
* @SDE_IRQ_TYPE_SFI_CMD_0_IN: DSI CMD0 static frame INTR into static
|
|
* @SDE_IRQ_TYPE_SFI_CMD_0_OUT: DSI CMD0 static frame INTR out-of static
|
|
* @SDE_IRQ_TYPE_SFI_CMD_1_IN: DSI CMD1 static frame INTR into static
|
|
* @SDE_IRQ_TYPE_SFI_CMD_1_OUT: DSI CMD1 static frame INTR out-of static
|
|
* @SDE_IRQ_TYPE_SFI_CMD_2_IN: DSI CMD2 static frame INTR into static
|
|
* @SDE_IRQ_TYPE_SFI_CMD_2_OUT: DSI CMD2 static frame INTR out-of static
|
|
* @SDE_IRQ_TYPE_PROG_LINE: Programmable Line interrupt
|
|
* @SDE_IRQ_TYPE_AD4_BL_DONE: AD4 backlight
|
|
* @SDE_IRQ_TYPE_CTL_START: Control start
|
|
* @SDE_IRQ_TYPE_CTL_DONE: Frame transfer complete
|
|
* @SDE_IRQ_TYPE_INTF_TEAR_RD_PTR: INTF Tear read pointer
|
|
* @SDE_IRQ_TYPE_INTF_TEAR_WR_PTR: INTF Tear write pointer
|
|
* @SDE_IRQ_TYPE_INTF_TEAR_AUTO_REF: INTF Tear auto refresh
|
|
* @SDE_IRQ_TYPE_INTF_TEAR_TEAR_CHECK: INTF Tear Tear check
|
|
* @SDE_IRQ_TYPE_INTF_TEAR_TE_CHECK: INTF Tear TE detection
|
|
* @SDE_IRQ_TYPE_LTM_STATS_DONE: LTM stats done interrupt
|
|
* @SDE_IRQ_TYPE_LTM_STATS_WB_PB: LTM stats WB push back interrupt
|
|
* @SDE_IRQ_TYPE_PROG_LINE: Programmable Line interrupt for WB
|
|
* @SDE_IRQ_TYPE_RESERVED: Reserved for expansion
|
|
*/
|
|
enum sde_intr_type {
|
|
SDE_IRQ_TYPE_WB_ROT_COMP,
|
|
SDE_IRQ_TYPE_WB_WFD_COMP,
|
|
SDE_IRQ_TYPE_PING_PONG_COMP,
|
|
SDE_IRQ_TYPE_PING_PONG_RD_PTR,
|
|
SDE_IRQ_TYPE_PING_PONG_WR_PTR,
|
|
SDE_IRQ_TYPE_PING_PONG_AUTO_REF,
|
|
SDE_IRQ_TYPE_PING_PONG_TEAR_CHECK,
|
|
SDE_IRQ_TYPE_PING_PONG_TE_CHECK,
|
|
SDE_IRQ_TYPE_INTF_UNDER_RUN,
|
|
SDE_IRQ_TYPE_INTF_VSYNC,
|
|
SDE_IRQ_TYPE_CWB_OVERFLOW,
|
|
SDE_IRQ_TYPE_HIST_VIG_DONE,
|
|
SDE_IRQ_TYPE_HIST_VIG_RSTSEQ,
|
|
SDE_IRQ_TYPE_HIST_DSPP_DONE,
|
|
SDE_IRQ_TYPE_HIST_DSPP_RSTSEQ,
|
|
SDE_IRQ_TYPE_WD_TIMER,
|
|
SDE_IRQ_TYPE_SFI_VIDEO_IN,
|
|
SDE_IRQ_TYPE_SFI_VIDEO_OUT,
|
|
SDE_IRQ_TYPE_SFI_CMD_0_IN,
|
|
SDE_IRQ_TYPE_SFI_CMD_0_OUT,
|
|
SDE_IRQ_TYPE_SFI_CMD_1_IN,
|
|
SDE_IRQ_TYPE_SFI_CMD_1_OUT,
|
|
SDE_IRQ_TYPE_SFI_CMD_2_IN,
|
|
SDE_IRQ_TYPE_SFI_CMD_2_OUT,
|
|
SDE_IRQ_TYPE_PROG_LINE,
|
|
SDE_IRQ_TYPE_AD4_BL_DONE,
|
|
SDE_IRQ_TYPE_CTL_START,
|
|
SDE_IRQ_TYPE_CTL_DONE,
|
|
SDE_IRQ_TYPE_INTF_TEAR_RD_PTR,
|
|
SDE_IRQ_TYPE_INTF_TEAR_WR_PTR,
|
|
SDE_IRQ_TYPE_INTF_TEAR_AUTO_REF,
|
|
SDE_IRQ_TYPE_INTF_TEAR_TEAR_CHECK,
|
|
SDE_IRQ_TYPE_INTF_TEAR_TE_CHECK,
|
|
SDE_IRQ_TYPE_LTM_STATS_DONE,
|
|
SDE_IRQ_TYPE_LTM_STATS_WB_PB,
|
|
SDE_IRQ_TYPE_WB_PROG_LINE,
|
|
SDE_IRQ_TYPE_RESERVED,
|
|
};
|
|
|
|
struct sde_hw_intr;
|
|
|
|
/**
|
|
* Interrupt operations.
|
|
*/
|
|
struct sde_hw_intr_ops {
|
|
/**
|
|
* irq_idx_lookup - Lookup IRQ index on the HW interrupt type
|
|
* Used for all irq related ops
|
|
* @intr: HW interrupt handle
|
|
* @intr_type: Interrupt type defined in sde_intr_type
|
|
* @instance_idx: HW interrupt block instance
|
|
* @return: irq_idx or -EINVAL for lookup fail
|
|
*/
|
|
int (*irq_idx_lookup)(
|
|
struct sde_hw_intr *intr,
|
|
enum sde_intr_type intr_type,
|
|
u32 instance_idx);
|
|
|
|
/**
|
|
* enable_irq_nolock - Enable IRQ based on lookup IRQ index without lock
|
|
* @intr: HW interrupt handle
|
|
* @irq_idx: Lookup irq index return from irq_idx_lookup
|
|
* @return: 0 for success, otherwise failure
|
|
*/
|
|
int (*enable_irq_nolock)(
|
|
struct sde_hw_intr *intr,
|
|
int irq_idx);
|
|
|
|
/**
|
|
* disable_irq_nolock - Disable IRQ based on IRQ index without lock
|
|
* @intr: HW interrupt handle
|
|
* @irq_idx: Lookup irq index return from irq_idx_lookup
|
|
* @return: 0 for success, otherwise failure
|
|
*/
|
|
int (*disable_irq_nolock)(
|
|
struct sde_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 sde_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 sde_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 sde_hw_intr *intr,
|
|
void (*cbfunc)(void *arg, int irq_idx),
|
|
void *arg);
|
|
|
|
/**
|
|
* 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 sde_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 sde_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 sde_hw_intr *intr,
|
|
int irq_idx,
|
|
bool clear);
|
|
|
|
/**
|
|
* get_intr_status_nolock - nolock version of get_interrupt_status
|
|
* @intr: HW interrupt handle
|
|
* @irq_idx: Lookup irq index return from irq_idx_lookup
|
|
* @clear: True to clear irq after read
|
|
*/
|
|
u32 (*get_intr_status_nolock)(
|
|
struct sde_hw_intr *intr,
|
|
int irq_idx,
|
|
bool clear);
|
|
|
|
/**
|
|
* get_interrupt_sources - Gets the bitmask of the SDE interrupt
|
|
* source that are currently fired.
|
|
* @intr: HW interrupt handle
|
|
* @sources: Returning the SDE interrupt source status bit mask
|
|
* @return: 0 for success, otherwise failure
|
|
*/
|
|
int (*get_interrupt_sources)(
|
|
struct sde_hw_intr *intr,
|
|
uint32_t *sources);
|
|
};
|
|
|
|
/**
|
|
* struct sde_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_lock: spinlock for accessing IRQ resources
|
|
* @sde_irq_size: total number of elements of the sde_irq_tbl
|
|
* @sde_irq_tbl: table with the registesrs offsets of the sde interrupts
|
|
* supported by the hw
|
|
* @sde_irq_map_size: total number of elements of the 'sde_irq_map'
|
|
* @sde_irq_map: total number of interrupt bits valid within the irq regs
|
|
*/
|
|
struct sde_hw_intr {
|
|
struct sde_hw_blk_reg_map hw;
|
|
struct sde_hw_intr_ops ops;
|
|
u32 *cache_irq_mask;
|
|
u32 *save_irq_status;
|
|
u32 sde_irq_size;
|
|
struct sde_intr_reg *sde_irq_tbl;
|
|
u32 sde_irq_map_size;
|
|
struct sde_irq_type *sde_irq_map;
|
|
spinlock_t irq_lock;
|
|
};
|
|
|
|
/**
|
|
* sde_hw_intr_init(): Initializes the interrupts hw object
|
|
* @addr: mapped register io address of MDP
|
|
* @m : pointer to mdss catalog data
|
|
*/
|
|
struct sde_hw_intr *sde_hw_intr_init(void __iomem *addr,
|
|
struct sde_mdss_cfg *m);
|
|
|
|
/**
|
|
* sde_hw_intr_destroy(): Cleanup interrutps hw object
|
|
* @intr: pointer to interrupts hw object
|
|
*/
|
|
void sde_hw_intr_destroy(struct sde_hw_intr *intr);
|
|
|
|
/**
|
|
* sde_hw_intr_list_lookup(): get the list entry for a given intr
|
|
* @sde_cfg: catalog containing the irq_offset_list
|
|
* @type: the sde_intr_hwblk_type to lookup
|
|
* @idx: the instance id to lookup for the specified hwblk_type
|
|
* @return: pointer to sde_intr_irq_offsets list entry, or NULL if lookup fails
|
|
*/
|
|
static inline struct sde_intr_irq_offsets *sde_hw_intr_list_lookup(
|
|
struct sde_mdss_cfg *sde_cfg, enum sde_intr_hwblk_type type, u32 idx)
|
|
{
|
|
struct sde_intr_irq_offsets *item;
|
|
|
|
list_for_each_entry(item, &sde_cfg->irq_offset_list, list) {
|
|
if (type == item->type && idx == item->instance_idx)
|
|
return item;
|
|
}
|
|
return NULL;
|
|
}
|
|
#endif
|