123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2017-2019, The Linux Foundation. All rights reserved.
- */
- #include "sde_hw_ds.h"
- #include "sde_formats.h"
- #include "sde_dbg.h"
- #include "sde_kms.h"
- /* Destination scaler TOP registers */
- #define DEST_SCALER_OP_MODE 0x00
- #define DEST_SCALER_HW_VERSION 0x10
- static void sde_hw_ds_setup_opmode(struct sde_hw_ds *hw_ds,
- u32 op_mode)
- {
- struct sde_hw_blk_reg_map *hw = &hw_ds->hw;
- SDE_REG_WRITE(hw, DEST_SCALER_OP_MODE, op_mode);
- }
- static u32 _sde_hw_ds_get_scaler3_ver(struct sde_hw_ds *ctx)
- {
- if (!ctx)
- return 0;
- return sde_hw_get_scaler3_ver(&ctx->hw, ctx->scl->base);
- }
- static void sde_hw_ds_setup_scaler3(struct sde_hw_ds *hw_ds,
- void *scaler_cfg, void *scaler_lut_cfg)
- {
- struct sde_hw_scaler3_cfg *scl3_cfg = scaler_cfg;
- struct sde_hw_scaler3_lut_cfg *scl3_lut_cfg = scaler_lut_cfg;
- if (!hw_ds || !hw_ds->scl || !scl3_cfg || !scl3_lut_cfg)
- return;
- /*
- * copy LUT values to scaler structure
- */
- if (scl3_lut_cfg->is_configured) {
- scl3_cfg->dir_lut = scl3_lut_cfg->dir_lut;
- scl3_cfg->dir_len = scl3_lut_cfg->dir_len;
- scl3_cfg->cir_lut = scl3_lut_cfg->cir_lut;
- scl3_cfg->cir_len = scl3_lut_cfg->cir_len;
- scl3_cfg->sep_lut = scl3_lut_cfg->sep_lut;
- scl3_cfg->sep_len = scl3_lut_cfg->sep_len;
- }
- sde_hw_setup_scaler3(&hw_ds->hw, scl3_cfg, hw_ds->scl->version,
- hw_ds->scl->base,
- sde_get_sde_format(DRM_FORMAT_XBGR2101010));
- }
- static void _setup_ds_ops(struct sde_hw_ds_ops *ops, unsigned long features)
- {
- ops->setup_opmode = sde_hw_ds_setup_opmode;
- if (test_bit(SDE_SSPP_SCALER_QSEED3, &features) ||
- test_bit(SDE_SSPP_SCALER_QSEED3LITE, &features)) {
- ops->get_scaler_ver = _sde_hw_ds_get_scaler3_ver;
- ops->setup_scaler = sde_hw_ds_setup_scaler3;
- }
- }
- static struct sde_ds_cfg *_ds_offset(enum sde_ds ds,
- struct sde_mdss_cfg *m,
- void __iomem *addr,
- struct sde_hw_blk_reg_map *b)
- {
- int i;
- if (!m || !addr || !b)
- return ERR_PTR(-EINVAL);
- for (i = 0; i < m->ds_count; i++) {
- if ((ds == m->ds[i].id) &&
- (m->ds[i].top)) {
- b->base_off = addr;
- b->blk_off = m->ds[i].top->base;
- b->length = m->ds[i].top->len;
- b->hwversion = m->hwversion;
- b->log_mask = SDE_DBG_MASK_DS;
- return &m->ds[i];
- }
- }
- return ERR_PTR(-EINVAL);
- }
- static struct sde_hw_blk_ops sde_hw_ops = {
- .start = NULL,
- .stop = NULL,
- };
- struct sde_hw_ds *sde_hw_ds_init(enum sde_ds idx,
- void __iomem *addr,
- struct sde_mdss_cfg *m)
- {
- struct sde_hw_ds *hw_ds;
- struct sde_ds_cfg *cfg;
- int rc;
- if (!addr || !m)
- return ERR_PTR(-EINVAL);
- hw_ds = kzalloc(sizeof(*hw_ds), GFP_KERNEL);
- if (!hw_ds)
- return ERR_PTR(-ENOMEM);
- cfg = _ds_offset(idx, m, addr, &hw_ds->hw);
- if (IS_ERR_OR_NULL(cfg)) {
- SDE_ERROR("failed to get ds cfg\n");
- kfree(hw_ds);
- return ERR_PTR(-EINVAL);
- }
- /* Assign ops */
- hw_ds->idx = idx;
- hw_ds->scl = cfg;
- _setup_ds_ops(&hw_ds->ops, hw_ds->scl->features);
- if (hw_ds->ops.get_scaler_ver)
- hw_ds->scl->version = hw_ds->ops.get_scaler_ver(hw_ds);
- rc = sde_hw_blk_init(&hw_ds->base, SDE_HW_BLK_DS, idx, &sde_hw_ops);
- if (rc) {
- SDE_ERROR("failed to init hw blk %d\n", rc);
- goto blk_init_error;
- }
- if (cfg->len) {
- sde_dbg_reg_register_dump_range(SDE_DBG_NAME, cfg->name,
- hw_ds->hw.blk_off + cfg->base,
- hw_ds->hw.blk_off + cfg->base + cfg->len,
- hw_ds->hw.xin_id);
- }
- return hw_ds;
- blk_init_error:
- kzfree(hw_ds);
- return ERR_PTR(rc);
- }
- void sde_hw_ds_destroy(struct sde_hw_ds *hw_ds)
- {
- if (hw_ds)
- sde_hw_blk_destroy(&hw_ds->base);
- kfree(hw_ds);
- }
|