root/drivers/staging/media/ipu7/ipu7-isys-csi-phy.c
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (C) 2013 - 2025 Intel Corporation
 */

#include <linux/bitmap.h>
#include <linux/bug.h>
#include <linux/delay.h>
#include <linux/device.h>
#include <linux/iopoll.h>
#include <linux/kernel.h>
#include <linux/types.h>

#include <media/mipi-csi2.h>
#include <media/v4l2-device.h>

#include "ipu7.h"
#include "ipu7-bus.h"
#include "ipu7-buttress.h"
#include "ipu7-isys.h"
#include "ipu7-isys-csi2.h"
#include "ipu7-isys-csi2-regs.h"
#include "ipu7-platform-regs.h"
#include "ipu7-isys-csi-phy.h"

#define PORT_A          0U
#define PORT_B          1U
#define PORT_C          2U
#define PORT_D          3U

#define N_DATA_IDS      8U
static DECLARE_BITMAP(data_ids, N_DATA_IDS);

struct ddlcal_counter_ref_s {
        u16 min_mbps;
        u16 max_mbps;

        u16 ddlcal_counter_ref;
};

struct ddlcal_params {
        u16 min_mbps;
        u16 max_mbps;
        u16 oa_lanex_hsrx_cdphy_sel_fast;
        u16 ddlcal_max_phase;
        u16 phase_bound;
        u16 ddlcal_dll_fbk;
        u16 ddlcal_ddl_coarse_bank;
        u16 fjump_deskew;
        u16 min_eye_opening_deskew;
};

struct i_thssettle_params {
        u16 min_mbps;
        u16 max_mbps;
        u16 i_thssettle;
};

 /* lane2 for 4l3t, lane1 for 2l2t */
struct oa_lane_clk_div_params {
        u16 min_mbps;
        u16 max_mbps;
        u16 oa_lane_hsrx_hs_clk_div;
};

struct cdr_fbk_cap_prog_params {
        u16 min_mbps;
        u16 max_mbps;
        u16 val;
};

static const struct ddlcal_counter_ref_s table0[] = {
        { 1500, 1999, 118 },
        { 2000, 2499, 157 },
        { 2500, 3499, 196 },
        { 3500, 4499, 274 },
        { 4500, 4500, 352 },
        { }
};

static const struct ddlcal_params table1[] = {
        { 1500, 1587, 0, 143, 167, 17, 3, 4, 29 },
        { 1588, 1687, 0, 135, 167, 15, 3, 4, 27 },
        { 1688, 1799, 0, 127, 135, 15, 2, 4, 26 },
        { 1800, 1928, 0, 119, 135, 13, 2, 3, 24 },
        { 1929, 2076, 0, 111, 135, 13, 2, 3, 23 },
        { 2077, 2249, 0, 103, 135, 11, 2, 3, 21 },
        { 2250, 2454, 0, 95, 103, 11, 1, 3, 19 },
        { 2455, 2699, 0, 87, 103, 9, 1, 3, 18 },
        { 2700, 2999, 0, 79, 103, 9, 1, 2, 16 },
        { 3000, 3229, 0, 71, 71, 7, 1, 2, 15 },
        { 3230, 3599, 1, 87, 103, 9, 1, 3, 18 },
        { 3600, 3999, 1, 79, 103, 9, 1, 2, 16 },
        { 4000, 4499, 1, 71, 103, 7, 1, 2, 15 },
        { 4500, 4500, 1, 63, 71, 7, 0, 2, 13 },
        { }
};

static const struct i_thssettle_params table2[] = {
        { 80, 124, 24 },
        { 125, 249, 20 },
        { 250, 499, 16 },
        { 500, 749, 14 },
        { 750, 1499, 13 },
        { 1500, 4500, 12 },
        { }
};

static const struct oa_lane_clk_div_params table6[] = {
        { 80, 159, 0x1 },
        { 160, 319, 0x2 },
        { 320, 639, 0x3 },
        { 640, 1279, 0x4 },
        { 1280, 2560, 0x5 },
        { 2561, 4500, 0x6 },
        { }
};

static const struct cdr_fbk_cap_prog_params table7[] = {
        { 80, 919, 0 },
        { 920, 1029, 1 },
        { 1030, 1169, 2 },
        { 1170, 1349, 3 },
        { 1350, 1589, 4 },
        { 1590, 1949, 5 },
        { 1950, 2499, 6 },
        { 2500, 3500, 7 },
        { }
};

static void dwc_phy_write(struct ipu7_isys *isys, u32 id, u32 addr, u16 data)
{
        void __iomem *isys_base = isys->pdata->base;
        void __iomem *base = isys_base + IS_IO_CDPHY_BASE(id);

        dev_dbg(&isys->adev->auxdev.dev, "phy write: reg 0x%zx = data 0x%04x",
                base + addr - isys_base, data);
        writew(data, base + addr);
}

static u16 dwc_phy_read(struct ipu7_isys *isys, u32 id, u32 addr)
{
        void __iomem *isys_base = isys->pdata->base;
        void __iomem *base = isys_base + IS_IO_CDPHY_BASE(id);
        u16 data;

        data = readw(base + addr);
        dev_dbg(&isys->adev->auxdev.dev, "phy read: reg 0x%zx = data 0x%04x",
                base + addr - isys_base, data);

        return data;
}

static void dwc_csi_write(struct ipu7_isys *isys, u32 id, u32 addr, u32 data)
{
        void __iomem *isys_base = isys->pdata->base;
        void __iomem *base = isys_base + IS_IO_CSI2_HOST_BASE(id);
        struct device *dev = &isys->adev->auxdev.dev;

        dev_dbg(dev, "csi write: reg 0x%zx = data 0x%08x",
                base + addr - isys_base, data);
        writel(data, base + addr);
        dev_dbg(dev, "csi read: reg 0x%zx = data 0x%08x",
                base + addr - isys_base, readl(base + addr));
}

static void gpreg_write(struct ipu7_isys *isys, u32 id, u32 addr, u32 data)
{
        void __iomem *isys_base = isys->pdata->base;
        u32 gpreg = isys->pdata->ipdata->csi2.gpreg;
        void __iomem *base = isys_base + gpreg + 0x1000 * id;
        struct device *dev = &isys->adev->auxdev.dev;

        dev_dbg(dev, "gpreg write: reg 0x%zx = data 0x%08x",
                base + addr - isys_base, data);
        writel(data, base + addr);
        dev_dbg(dev, "gpreg read: reg 0x%zx = data 0x%08x",
                base + addr - isys_base, readl(base + addr));
}

static u32 dwc_csi_read(struct ipu7_isys *isys, u32 id, u32 addr)
{
        void __iomem *isys_base = isys->pdata->base;
        void __iomem *base = isys_base + IS_IO_CSI2_HOST_BASE(id);
        u32 data;

        data = readl(base + addr);
        dev_dbg(&isys->adev->auxdev.dev, "csi read: reg 0x%zx = data 0x%x",
                base + addr - isys_base, data);

        return data;
}

static void dwc_phy_write_mask(struct ipu7_isys *isys, u32 id, u32 addr,
                               u16 val, u8 lo, u8 hi)
{
        u32 temp, mask;

        WARN_ON(lo > hi);
        WARN_ON(hi > 15);

        mask = ((~0U - (1U << lo) + 1U)) & (~0U >> (31 - hi));
        temp = dwc_phy_read(isys, id, addr);
        temp &= ~mask;
        temp |= (val << lo) & mask;
        dwc_phy_write(isys, id, addr, temp);
}

static void dwc_csi_write_mask(struct ipu7_isys *isys, u32 id, u32 addr,
                               u32 val, u8 hi, u8 lo)
{
        u32 temp, mask;

        WARN_ON(lo > hi);

        mask = ((~0U - (1U << lo) + 1U)) & (~0U >> (31 - hi));
        temp = dwc_csi_read(isys, id, addr);
        temp &= ~mask;
        temp |= (val << lo) & mask;
        dwc_csi_write(isys, id, addr, temp);
}

static void ipu7_isys_csi_ctrl_cfg(struct ipu7_isys_csi2 *csi2)
{
        struct ipu7_isys *isys = csi2->isys;
        struct device *dev = &isys->adev->auxdev.dev;
        u32 id, lanes, phy_mode;
        u32 val;

        id = csi2->port;
        lanes = csi2->nlanes;
        phy_mode = csi2->phy_mode;
        dev_dbg(dev, "csi-%d controller init with %u lanes, phy mode %u",
                id, lanes, phy_mode);

        val = dwc_csi_read(isys, id, VERSION);
        dev_dbg(dev, "csi-%d controller version = 0x%x", id, val);

        /* num of active data lanes */
        dwc_csi_write(isys, id, N_LANES, lanes - 1);
        dwc_csi_write(isys, id, CDPHY_MODE, phy_mode);
        dwc_csi_write(isys, id, VC_EXTENSION, 0);

        /* only mask PHY_FATAL and PKT_FATAL interrupts */
        dwc_csi_write(isys, id, INT_MSK_PHY_FATAL, 0xff);
        dwc_csi_write(isys, id, INT_MSK_PKT_FATAL, 0x3);
        dwc_csi_write(isys, id, INT_MSK_PHY, 0x0);
        dwc_csi_write(isys, id, INT_MSK_LINE, 0x0);
        dwc_csi_write(isys, id, INT_MSK_BNDRY_FRAME_FATAL, 0x0);
        dwc_csi_write(isys, id, INT_MSK_SEQ_FRAME_FATAL, 0x0);
        dwc_csi_write(isys, id, INT_MSK_CRC_FRAME_FATAL, 0x0);
        dwc_csi_write(isys, id, INT_MSK_PLD_CRC_FATAL, 0x0);
        dwc_csi_write(isys, id, INT_MSK_DATA_ID, 0x0);
        dwc_csi_write(isys, id, INT_MSK_ECC_CORRECTED, 0x0);
}

static void ipu7_isys_csi_phy_reset(struct ipu7_isys *isys, u32 id)
{
        dwc_csi_write(isys, id, PHY_SHUTDOWNZ, 0);
        dwc_csi_write(isys, id, DPHY_RSTZ, 0);
        dwc_csi_write(isys, id, CSI2_RESETN, 0);
        gpreg_write(isys, id, PHY_RESET, 0);
        gpreg_write(isys, id, PHY_SHUTDOWN, 0);
}

/* 8 Data ID monitors, each Data ID is composed by pair of VC and data type */
static int __dids_config(struct ipu7_isys_csi2 *csi2, u32 id, u8 vc, u8 dt)
{
        struct ipu7_isys *isys = csi2->isys;
        u32 reg, n;
        u8 lo, hi;
        int ret;

        dev_dbg(&isys->adev->auxdev.dev, "config CSI-%u with vc:%u dt:0x%02x\n",
                id, vc, dt);

        dwc_csi_write(isys, id, VC_EXTENSION, 0x0);
        n = find_first_zero_bit(data_ids, N_DATA_IDS);
        if (n == N_DATA_IDS)
                return -ENOSPC;

        ret = test_and_set_bit(n, data_ids);
        if (ret)
                return -EBUSY;

        reg = n < 4 ? DATA_IDS_VC_1 : DATA_IDS_VC_2;
        lo = (n % 4) * 8;
        hi = lo + 4;
        dwc_csi_write_mask(isys, id, reg, vc & GENMASK(4, 0), hi, lo);

        reg = n < 4 ? DATA_IDS_1 : DATA_IDS_2;
        lo = (n % 4) * 8;
        hi = lo + 5;
        dwc_csi_write_mask(isys, id, reg, dt & GENMASK(5, 0), hi, lo);

        return 0;
}

static int ipu7_isys_csi_ctrl_dids_config(struct ipu7_isys_csi2 *csi2, u32 id)
{
        struct v4l2_mbus_frame_desc_entry *desc_entry = NULL;
        struct device *dev = &csi2->isys->adev->auxdev.dev;
        struct v4l2_mbus_frame_desc desc;
        struct v4l2_subdev *ext_sd;
        struct media_pad *pad;
        unsigned int i;
        int ret;

        pad = media_entity_remote_source_pad_unique(&csi2->asd.sd.entity);
        if (IS_ERR(pad)) {
                dev_warn(dev, "can't get remote source pad of %s (%pe)\n",
                         csi2->asd.sd.name, pad);
                return PTR_ERR(pad);
        }

        ext_sd = media_entity_to_v4l2_subdev(pad->entity);
        if (WARN(!ext_sd, "Failed to get subdev for entity %s\n",
                 pad->entity->name))
                return -ENODEV;

        ret = v4l2_subdev_call(ext_sd, pad, get_frame_desc, pad->index, &desc);
        if (ret)
                return ret;

        if (desc.type != V4L2_MBUS_FRAME_DESC_TYPE_CSI2) {
                dev_warn(dev, "Unsupported frame descriptor type\n");
                return -EINVAL;
        }

        for (i = 0; i < desc.num_entries; i++) {
                desc_entry = &desc.entry[i];
                if (desc_entry->bus.csi2.vc < IPU7_NR_OF_CSI2_VC) {
                        ret = __dids_config(csi2, id, desc_entry->bus.csi2.vc,
                                            desc_entry->bus.csi2.dt);
                        if (ret)
                                return ret;
                }
        }

        return 0;
}

#define CDPHY_TIMEOUT 5000000U
static int ipu7_isys_phy_ready(struct ipu7_isys *isys, u32 id)
{
        void __iomem *isys_base = isys->pdata->base;
        u32 gpreg_offset = isys->pdata->ipdata->csi2.gpreg;
        void __iomem *gpreg = isys_base + gpreg_offset + 0x1000 * id;
        struct device *dev = &isys->adev->auxdev.dev;
        unsigned int i;
        u32 phy_ready;
        u32 reg, rext;
        int ret;

        dev_dbg(dev, "waiting phy ready...\n");
        ret = readl_poll_timeout(gpreg + PHY_READY, phy_ready,
                                 phy_ready & BIT(0) && phy_ready != ~0U,
                                 100, CDPHY_TIMEOUT);
        dev_dbg(dev, "phy %u ready = 0x%08x\n", id, readl(gpreg + PHY_READY));
        dev_dbg(dev, "csi %u PHY_RX = 0x%08x\n", id,
                dwc_csi_read(isys, id, PHY_RX));
        dev_dbg(dev, "csi %u PHY_STOPSTATE = 0x%08x\n", id,
                dwc_csi_read(isys, id, PHY_STOPSTATE));
        dev_dbg(dev, "csi %u PHY_CAL = 0x%08x\n", id,
                dwc_csi_read(isys, id, PHY_CAL));
        for (i = 0; i < 4U; i++) {
                reg = CORE_DIG_DLANE_0_R_HS_RX_0 + (i * 0x400U);
                dev_dbg(dev, "phy %u DLANE%u skewcal = 0x%04x\n",
                        id, i, dwc_phy_read(isys, id, reg));
        }
        dev_dbg(dev, "phy %u DDLCAL = 0x%04x\n", id,
                dwc_phy_read(isys, id, PPI_CALIBCTRL_R_COMMON_CALIBCTRL_2_5));
        dev_dbg(dev, "phy %u TERMCAL = 0x%04x\n", id,
                dwc_phy_read(isys, id, PPI_R_TERMCAL_DEBUG_0));
        dev_dbg(dev, "phy %u LPDCOCAL = 0x%04x\n", id,
                dwc_phy_read(isys, id, PPI_R_LPDCOCAL_DEBUG_RB));
        dev_dbg(dev, "phy %u HSDCOCAL = 0x%04x\n", id,
                dwc_phy_read(isys, id, PPI_R_HSDCOCAL_DEBUG_RB));
        dev_dbg(dev, "phy %u LPDCOCAL_VT = 0x%04x\n", id,
                dwc_phy_read(isys, id, PPI_R_LPDCOCAL_DEBUG_VT));

        if (!ret) {
                if (id) {
                        dev_dbg(dev, "ignore phy %u rext\n", id);
                        return 0;
                }

                rext = dwc_phy_read(isys, id,
                                    CORE_DIG_IOCTRL_R_AFE_CB_CTRL_2_15) & 0xfU;
                dev_dbg(dev, "phy %u rext value = %u\n", id, rext);
                isys->phy_rext_cal = (rext ? rext : 5);

                return 0;
        }

        dev_err(dev, "wait phy ready timeout!\n");

        return ret;
}

static int lookup_table1(u64 mbps)
{
        unsigned int i;

        for (i = 0; i < ARRAY_SIZE(table1); i++) {
                if (mbps >= table1[i].min_mbps && mbps <= table1[i].max_mbps)
                        return i;
        }

        return -ENXIO;
}

static const u16 deskew_fine_mem[] = {
        0x0404, 0x040c, 0x0414, 0x041c,
        0x0423, 0x0429, 0x0430, 0x043a,
        0x0445, 0x044a, 0x0450, 0x045a,
        0x0465, 0x0469, 0x0472, 0x047a,
        0x0485, 0x0489, 0x0490, 0x049a,
        0x04a4, 0x04ac, 0x04b4, 0x04bc,
        0x04c4, 0x04cc, 0x04d4, 0x04dc,
        0x04e4, 0x04ec, 0x04f4, 0x04fc,
        0x0504, 0x050c, 0x0514, 0x051c,
        0x0523, 0x0529, 0x0530, 0x053a,
        0x0545, 0x054a, 0x0550, 0x055a,
        0x0565, 0x0569, 0x0572, 0x057a,
        0x0585, 0x0589, 0x0590, 0x059a,
        0x05a4, 0x05ac, 0x05b4, 0x05bc,
        0x05c4, 0x05cc, 0x05d4, 0x05dc,
        0x05e4, 0x05ec, 0x05f4, 0x05fc,
        0x0604, 0x060c, 0x0614, 0x061c,
        0x0623, 0x0629, 0x0632, 0x063a,
        0x0645, 0x064a, 0x0650, 0x065a,
        0x0665, 0x0669, 0x0672, 0x067a,
        0x0685, 0x0689, 0x0690, 0x069a,
        0x06a4, 0x06ac, 0x06b4, 0x06bc,
        0x06c4, 0x06cc, 0x06d4, 0x06dc,
        0x06e4, 0x06ec, 0x06f4, 0x06fc,
        0x0704, 0x070c, 0x0714, 0x071c,
        0x0723, 0x072a, 0x0730, 0x073a,
        0x0745, 0x074a, 0x0750, 0x075a,
        0x0765, 0x0769, 0x0772, 0x077a,
        0x0785, 0x0789, 0x0790, 0x079a,
        0x07a4, 0x07ac, 0x07b4, 0x07bc,
        0x07c4, 0x07cc, 0x07d4, 0x07dc,
        0x07e4, 0x07ec, 0x07f4, 0x07fc,
};

static void ipu7_isys_dphy_config(struct ipu7_isys *isys, u8 id, u8 lanes,
                                  bool aggregation, u64 mbps)
{
        u16 hsrxval0 = 0;
        u16 hsrxval1 = 0;
        u16 hsrxval2 = 0;
        int index;
        u16 reg;
        u16 val;
        u32 i;

        dwc_phy_write_mask(isys, id, CORE_DIG_RW_COMMON_7, 0, 0, 9);
        if (mbps > 1500)
                dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_7,
                                   40, 0, 7);
        else
                dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_7,
                                   104, 0, 7);

        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_8, 80, 0, 7);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_0, 191, 0, 9);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_7, 34, 7, 12);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_1, 38, 8, 15);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_2, 4, 12, 15);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_2, 2, 10, 11);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_2, 1, 8, 8);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_2, 38, 0, 7);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_2, 1, 9, 9);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_4, 10, 0, 9);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_6, 20, 0, 9);
        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_7, 19, 0, 6);

        for (i = 0; i < ARRAY_SIZE(table0); i++) {
                if (mbps >= table0[i].min_mbps && mbps <= table0[i].max_mbps) {
                        dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_3,
                                           table0[i].ddlcal_counter_ref,
                                           0, 9);
                        break;
                }
        }

        index = lookup_table1(mbps);
        if (index >= 0) {
                dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_1,
                                   table1[index].phase_bound, 0, 7);
                dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_5,
                                   table1[index].ddlcal_dll_fbk, 4, 9);
                dwc_phy_write_mask(isys, id, PPI_RW_DDLCAL_CFG_5,
                                   table1[index].ddlcal_ddl_coarse_bank, 0, 3);

                reg = CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_8;
                val = table1[index].oa_lanex_hsrx_cdphy_sel_fast;
                for (i = 0; i < lanes + 1; i++)
                        dwc_phy_write_mask(isys, id, reg + (i * 0x400), val,
                                           12, 12);
        }

        reg = CORE_DIG_DLANE_0_RW_LP_0;
        for (i = 0; i < lanes; i++)
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 6, 8, 11);

        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_2,
                           0, 0, 0);
        if (!is_ipu7(isys->adev->isp->hw_ver) ||
            id == PORT_B || id == PORT_C) {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_2,
                                   1, 0, 0);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_2,
                                   0, 0, 0);
        } else {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_2,
                                   0, 0, 0);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_2,
                                   1, 0, 0);
        }

        if (lanes == 4 && is_ipu7(isys->adev->isp->hw_ver)) {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE3_CTRL_2_2,
                                   0, 0, 0);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE4_CTRL_2_2,
                                   0, 0, 0);
        }

        dwc_phy_write_mask(isys, id, CORE_DIG_RW_COMMON_6, 1, 0, 2);
        dwc_phy_write_mask(isys, id, CORE_DIG_RW_COMMON_6, 1, 3, 5);

        reg = CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_12;
        val = (mbps > 1500) ? 0 : 1;
        for (i = 0; i < lanes + 1; i++) {
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), val, 1, 1);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), !val, 3, 3);
        }

        reg = CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_13;
        val = (mbps > 1500) ? 0 : 1;
        for (i = 0; i < lanes + 1; i++) {
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), val, 1, 1);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), val, 3, 3);
        }

        if (!is_ipu7(isys->adev->isp->hw_ver) || id == PORT_B || id == PORT_C)
                reg = CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_9;
        else
                reg = CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_9;

        for (i = 0; i < ARRAY_SIZE(table6); i++) {
                if (mbps >= table6[i].min_mbps && mbps <= table6[i].max_mbps) {
                        dwc_phy_write_mask(isys, id, reg,
                                           table6[i].oa_lane_hsrx_hs_clk_div,
                                           5, 7);
                        break;
                }
        }

        if (aggregation) {
                dwc_phy_write_mask(isys, id, CORE_DIG_RW_COMMON_0, 1,
                                   1, 1);

                reg = CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_15;
                dwc_phy_write_mask(isys, id, reg, 3, 3, 4);

                val = (id == PORT_A) ? 3 : 0;
                reg = CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_15;
                dwc_phy_write_mask(isys, id, reg, val, 3, 4);

                reg = CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_15;
                dwc_phy_write_mask(isys, id, reg, 3, 3, 4);
        }

        dwc_phy_write_mask(isys, id, CORE_DIG_DLANE_CLK_RW_HS_RX_0, 28, 0, 7);
        dwc_phy_write_mask(isys, id, CORE_DIG_DLANE_CLK_RW_HS_RX_7, 6, 0, 7);

        reg = CORE_DIG_DLANE_0_RW_HS_RX_0;
        for (i = 0; i < ARRAY_SIZE(table2); i++) {
                if (mbps >= table2[i].min_mbps && mbps <= table2[i].max_mbps) {
                        u8 j;

                        for (j = 0; j < lanes; j++)
                                dwc_phy_write_mask(isys, id, reg + (j * 0x400),
                                                   table2[i].i_thssettle,
                                                   8, 15);
                        break;
                }
        }

        /* deskew */
        for (i = 0; i < lanes; i++) {
                reg = CORE_DIG_DLANE_0_RW_CFG_1;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400),
                                   ((mbps > 1500) ? 0x1 : 0x2), 2, 3);

                reg = CORE_DIG_DLANE_0_RW_HS_RX_2;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400),
                                   ((mbps > 2500) ? 0 : 1), 15, 15);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 1, 13, 13);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 7, 9, 12);

                reg = CORE_DIG_DLANE_0_RW_LP_0;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 1, 12, 15);

                reg = CORE_DIG_DLANE_0_RW_LP_2;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 0, 0, 0);

                reg = CORE_DIG_DLANE_0_RW_HS_RX_1;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 16, 0, 7);

                reg = CORE_DIG_DLANE_0_RW_HS_RX_3;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 2, 0, 2);
                index = lookup_table1(mbps);
                if (index >= 0) {
                        val = table1[index].fjump_deskew;
                        dwc_phy_write_mask(isys, id, reg + (i * 0x400), val,
                                           3, 8);
                }

                reg = CORE_DIG_DLANE_0_RW_HS_RX_4;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 150, 0, 15);

                reg = CORE_DIG_DLANE_0_RW_HS_RX_5;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 0, 0, 7);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 1, 8, 15);

                reg = CORE_DIG_DLANE_0_RW_HS_RX_6;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 2, 0, 7);
                index = lookup_table1(mbps);
                if (index >= 0) {
                        val = table1[index].min_eye_opening_deskew;
                        dwc_phy_write_mask(isys, id, reg + (i * 0x400), val,
                                           8, 15);
                }
                reg = CORE_DIG_DLANE_0_RW_HS_RX_7;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 0, 13, 13);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 0, 15, 15);

                reg = CORE_DIG_DLANE_0_RW_HS_RX_9;
                index = lookup_table1(mbps);
                if (index >= 0) {
                        val = table1[index].ddlcal_max_phase;
                        dwc_phy_write_mask(isys, id, reg + (i * 0x400),
                                           val, 0, 7);
                }
        }

        dwc_phy_write_mask(isys, id, CORE_DIG_DLANE_CLK_RW_LP_0, 1, 12, 15);
        dwc_phy_write_mask(isys, id, CORE_DIG_DLANE_CLK_RW_LP_2, 0, 0, 0);

        for (i = 0; i < ARRAY_SIZE(deskew_fine_mem); i++)
                dwc_phy_write_mask(isys, id, CORE_DIG_COMMON_RW_DESKEW_FINE_MEM,
                                   deskew_fine_mem[i], 0, 15);

        if (mbps > 1500) {
                hsrxval0 = 4;
                hsrxval2 = 3;
        }

        if (mbps > 2500)
                hsrxval1 = 2;

        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_9,
                           hsrxval0, 0, 2);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_9,
                           hsrxval0, 0, 2);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_9,
                           hsrxval0, 0, 2);
        if (lanes == 4 && is_ipu7(isys->adev->isp->hw_ver)) {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE3_CTRL_2_9,
                                   hsrxval0, 0, 2);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE4_CTRL_2_9,
                                   hsrxval0, 0, 2);
        }

        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_9,
                           hsrxval1, 3, 4);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_9,
                           hsrxval1, 3, 4);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_9,
                           hsrxval1, 3, 4);
        if (lanes == 4 && is_ipu7(isys->adev->isp->hw_ver)) {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE3_CTRL_2_9,
                                   hsrxval1, 3, 4);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE4_CTRL_2_9,
                                   hsrxval1, 3, 4);
        }

        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_15,
                           hsrxval2, 0, 2);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_15,
                           hsrxval2, 0, 2);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_15,
                           hsrxval2, 0, 2);
        if (lanes == 4 && is_ipu7(isys->adev->isp->hw_ver)) {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE3_CTRL_2_15,
                                   hsrxval2, 0, 2);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE4_CTRL_2_15,
                                   hsrxval2, 0, 2);
        }

        /* force and override rext */
        if (isys->phy_rext_cal && id) {
                dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_8,
                                   isys->phy_rext_cal, 0, 3);
                dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_7,
                                   1, 11, 11);
        }
}

static void ipu7_isys_cphy_config(struct ipu7_isys *isys, u8 id, u8 lanes,
                                  bool aggregation, u64 mbps)
{
        u8 trios = 2;
        u16 coarse_target;
        u16 deass_thresh;
        u16 delay_thresh;
        u16 reset_thresh;
        u16 cap_prog = 6U;
        u16 reg;
        u16 val;
        u32 i;
        u64 r64;
        u32 r;

        if (is_ipu7p5(isys->adev->isp->hw_ver))
                val = 0x15;
        else
                val = 0x155;

        if (is_ipu7(isys->adev->isp->hw_ver))
                trios = 3;

        dwc_phy_write_mask(isys, id, CORE_DIG_RW_COMMON_7, val, 0, 9);
        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_7, 104, 0, 7);
        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_8, 16, 0, 7);

        reg = CORE_DIG_CLANE_0_RW_LP_0;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 6, 8, 11);

        val = (mbps > 900U) ? 1U : 0U;
        for (i = 0; i < trios; i++) {
                reg = CORE_DIG_CLANE_0_RW_HS_RX_0;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 1, 0, 0);
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), val, 1, 1);

                reg = CORE_DIG_CLANE_0_RW_HS_RX_1;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 38, 0, 15);

                reg = CORE_DIG_CLANE_0_RW_HS_RX_5;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 38, 0, 15);

                reg = CORE_DIG_CLANE_0_RW_HS_RX_6;
                dwc_phy_write_mask(isys, id, reg + (i * 0x400), 10, 0, 15);
        }

        /*
         * Below 900Msps, always use the same value.
         * The formula is suitable for data rate 80-3500Msps.
         * Timebase (us) = 1, DIV = 32, TDDL (UI) = 0.5
         */
        if (mbps >= 80U)
                coarse_target = DIV_ROUND_UP_ULL(mbps, 16) - 1;
        else
                coarse_target = 56;

        for (i = 0; i < trios; i++) {
                reg = CORE_DIG_CLANE_0_RW_HS_RX_2 + i * 0x400;
                dwc_phy_write_mask(isys, id, reg, coarse_target, 0, 15);
        }

        dwc_phy_write_mask(isys, id,
                           CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_2, 1, 0, 0);
        dwc_phy_write_mask(isys, id,
                           CORE_DIG_IOCTRL_RW_AFE_LANE1_CTRL_2_2, 0, 0, 0);
        dwc_phy_write_mask(isys, id,
                           CORE_DIG_IOCTRL_RW_AFE_LANE2_CTRL_2_2, 1, 0, 0);

        if (!is_ipu7p5(isys->adev->isp->hw_ver) && lanes == 4) {
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE3_CTRL_2_2,
                                   1, 0, 0);
                dwc_phy_write_mask(isys, id,
                                   CORE_DIG_IOCTRL_RW_AFE_LANE4_CTRL_2_2,
                                   0, 0, 0);
        }

        for (i = 0; i < trios; i++) {
                reg = CORE_DIG_RW_TRIO0_0 + i * 0x400;
                dwc_phy_write_mask(isys, id, reg, 1, 6, 8);
                dwc_phy_write_mask(isys, id, reg, 1, 3, 5);
                dwc_phy_write_mask(isys, id, reg, 2, 0, 2);
        }

        deass_thresh = (u16)div64_u64_rem(7 * 1000 * 6, mbps * 5U, &r64) + 1;
        if (r64 != 0)
                deass_thresh++;

        reg = CORE_DIG_RW_TRIO0_2;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + 0x400 * i,
                                   deass_thresh, 0, 7);

        delay_thresh = div64_u64((224U - (9U * 7U)) * 1000U, 5U * mbps) - 7u;

        if (delay_thresh < 1)
                delay_thresh = 1;

        reg = CORE_DIG_RW_TRIO0_1;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + 0x400 * i,
                                   delay_thresh, 0, 15);

        reset_thresh = (u16)div_u64_rem(2U * 5U * mbps, 7U * 1000U, &r);
        if (!r)
                reset_thresh--;

        if (reset_thresh < 1)
                reset_thresh = 1;

        reg = CORE_DIG_RW_TRIO0_0;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + 0x400 * i,
                                   reset_thresh, 9, 11);

        /* Tuning ITMINRX to 2 for CPHY */
        reg = CORE_DIG_CLANE_0_RW_LP_0;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + 0x400 * i, 2, 12, 15);

        reg = CORE_DIG_CLANE_0_RW_LP_2;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + 0x400 * i, 0, 0, 0);

        reg = CORE_DIG_CLANE_0_RW_HS_RX_0;
        for (i = 0; i < trios; i++)
                dwc_phy_write_mask(isys, id, reg + 0x400 * i, 12, 2, 6);

        for (i = 0; i < ARRAY_SIZE(table7); i++) {
                if (mbps >= table7[i].min_mbps && mbps <= table7[i].max_mbps) {
                        cap_prog = table7[i].val;
                        break;
                }
        }

        for (i = 0; i < (lanes + 1); i++) {
                reg = CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_9 + 0x400 * i;
                dwc_phy_write_mask(isys, id, reg, 4U, 0, 2);
                /* Set GMODE to 2 when CPHY >= 1.5Gsps */
                if (mbps >= 1500)
                        dwc_phy_write_mask(isys, id, reg, 2U, 3, 4);
                else
                        dwc_phy_write_mask(isys, id, reg, 0U, 3, 4);

                reg = CORE_DIG_IOCTRL_RW_AFE_LANE0_CTRL_2_7 + 0x400 * i;
                dwc_phy_write_mask(isys, id, reg, cap_prog, 10, 12);
        }
}

static int ipu7_isys_phy_config(struct ipu7_isys *isys, u8 id, u8 lanes,
                                bool aggregation)
{
        struct device *dev = &isys->adev->auxdev.dev;
        u32 phy_mode;
        s64 link_freq;
        u64 mbps;

        if (aggregation)
                link_freq = ipu7_isys_csi2_get_link_freq(&isys->csi2[0]);
        else
                link_freq = ipu7_isys_csi2_get_link_freq(&isys->csi2[id]);

        if (link_freq < 0) {
                dev_err(dev, "get link freq failed (%lld)\n", link_freq);
                return link_freq;
        }

        mbps = div_u64(link_freq, 500000);
        dev_dbg(dev, "config phy %u with lanes %u aggregation %d mbps %lld\n",
                id, lanes, aggregation, mbps);

        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_10, 48, 0, 7);
        dwc_phy_write_mask(isys, id, CORE_DIG_ANACTRL_RW_COMMON_ANACTRL_2,
                           1, 12, 13);
        dwc_phy_write_mask(isys, id, CORE_DIG_ANACTRL_RW_COMMON_ANACTRL_0,
                           63, 2, 7);
        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_STARTUP_1_1,
                           563, 0, 11);
        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_2, 5, 0, 7);
        /* bypass the RCAL state (bit6) */
        if (aggregation && id != PORT_A)
                dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_2, 0x45,
                                   0, 7);

        dwc_phy_write_mask(isys, id, PPI_STARTUP_RW_COMMON_DPHY_6, 39, 0, 7);
        dwc_phy_write_mask(isys, id, PPI_CALIBCTRL_RW_COMMON_BG_0, 500, 0, 8);
        dwc_phy_write_mask(isys, id, PPI_RW_TERMCAL_CFG_0, 38, 0, 6);
        dwc_phy_write_mask(isys, id, PPI_RW_OFFSETCAL_CFG_0, 7, 0, 4);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_TIMEBASE, 153, 0, 9);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_NREF, 800, 0, 10);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_NREF_RANGE, 27, 0, 4);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_TWAIT_CONFIG, 47, 0, 8);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_TWAIT_CONFIG, 127, 9, 15);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_VT_CONFIG, 47, 7, 15);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_VT_CONFIG, 27, 2, 6);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_VT_CONFIG, 3, 0, 1);
        dwc_phy_write_mask(isys, id, PPI_RW_LPDCOCAL_COARSE_CFG, 1, 0, 1);
        dwc_phy_write_mask(isys, id, PPI_RW_COMMON_CFG, 3, 0, 1);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_0,
                           0, 10, 10);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_1,
                           1, 10, 10);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_1,
                           0, 15, 15);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_3,
                           3, 8, 9);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_0,
                           0, 15, 15);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_6,
                           7, 12, 14);
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_7,
                           0, 8, 10);
        /* resistance tuning: 1 for 45ohm, 0 for 50ohm */
        dwc_phy_write_mask(isys, id, CORE_DIG_IOCTRL_RW_AFE_CB_CTRL_2_5,
                           1, 8, 8);

        if (aggregation)
                phy_mode = isys->csi2[0].phy_mode;
        else
                phy_mode = isys->csi2[id].phy_mode;

        if (phy_mode == PHY_MODE_DPHY) {
                ipu7_isys_dphy_config(isys, id, lanes, aggregation, mbps);
        } else if (phy_mode == PHY_MODE_CPHY) {
                ipu7_isys_cphy_config(isys, id, lanes, aggregation, mbps);
        } else {
                dev_err(dev, "unsupported phy mode %d!\n",
                        isys->csi2[id].phy_mode);
        }

        return 0;
}

int ipu7_isys_csi_phy_powerup(struct ipu7_isys_csi2 *csi2)
{
        struct ipu7_isys *isys = csi2->isys;
        u32 lanes = csi2->nlanes;
        bool aggregation = false;
        u32 id = csi2->port;
        int ret;

        /* lanes remapping for aggregation (port AB) mode */
        if (!is_ipu7(isys->adev->isp->hw_ver) && lanes > 2 && id == PORT_A) {
                aggregation = true;
                lanes = 2;
        }

        ipu7_isys_csi_phy_reset(isys, id);
        gpreg_write(isys, id, PHY_CLK_LANE_CONTROL, 0x1);
        gpreg_write(isys, id, PHY_CLK_LANE_FORCE_CONTROL, 0x2);
        gpreg_write(isys, id, PHY_LANE_CONTROL_EN, (1U << lanes) - 1U);
        gpreg_write(isys, id, PHY_LANE_FORCE_CONTROL, 0xf);
        gpreg_write(isys, id, PHY_MODE, csi2->phy_mode);

        /* config PORT_B if aggregation mode */
        if (aggregation) {
                ipu7_isys_csi_phy_reset(isys, PORT_B);
                gpreg_write(isys, PORT_B, PHY_CLK_LANE_CONTROL, 0x0);
                gpreg_write(isys, PORT_B, PHY_LANE_CONTROL_EN, 0x3);
                gpreg_write(isys, PORT_B, PHY_CLK_LANE_FORCE_CONTROL, 0x2);
                gpreg_write(isys, PORT_B, PHY_LANE_FORCE_CONTROL, 0xf);
                gpreg_write(isys, PORT_B, PHY_MODE, csi2->phy_mode);
        }

        ipu7_isys_csi_ctrl_cfg(csi2);
        ipu7_isys_csi_ctrl_dids_config(csi2, id);

        ret = ipu7_isys_phy_config(isys, id, lanes, aggregation);
        if (ret < 0)
                return ret;

        gpreg_write(isys, id, PHY_RESET, 1);
        gpreg_write(isys, id, PHY_SHUTDOWN, 1);
        dwc_csi_write(isys, id, DPHY_RSTZ, 1);
        dwc_csi_write(isys, id, PHY_SHUTDOWNZ, 1);
        dwc_csi_write(isys, id, CSI2_RESETN, 1);

        ret = ipu7_isys_phy_ready(isys, id);
        if (ret < 0)
                return ret;

        gpreg_write(isys, id, PHY_LANE_FORCE_CONTROL, 0);
        gpreg_write(isys, id, PHY_CLK_LANE_FORCE_CONTROL, 0);

        /* config PORT_B if aggregation mode */
        if (aggregation) {
                ret = ipu7_isys_phy_config(isys, PORT_B, 2, aggregation);
                if (ret < 0)
                        return ret;

                gpreg_write(isys, PORT_B, PHY_RESET, 1);
                gpreg_write(isys, PORT_B, PHY_SHUTDOWN, 1);
                dwc_csi_write(isys, PORT_B, DPHY_RSTZ, 1);
                dwc_csi_write(isys, PORT_B, PHY_SHUTDOWNZ, 1);
                dwc_csi_write(isys, PORT_B, CSI2_RESETN, 1);
                ret = ipu7_isys_phy_ready(isys, PORT_B);
                if (ret < 0)
                        return ret;

                gpreg_write(isys, PORT_B, PHY_LANE_FORCE_CONTROL, 0);
                gpreg_write(isys, PORT_B, PHY_CLK_LANE_FORCE_CONTROL, 0);
        }

        return 0;
}

void ipu7_isys_csi_phy_powerdown(struct ipu7_isys_csi2 *csi2)
{
        struct ipu7_isys *isys = csi2->isys;

        ipu7_isys_csi_phy_reset(isys, csi2->port);
        if (!is_ipu7(isys->adev->isp->hw_ver) &&
            csi2->nlanes > 2U && csi2->port == PORT_A)
                ipu7_isys_csi_phy_reset(isys, PORT_B);
}