root/drivers/media/dvb-frontends/cxd2880/cxd2880_tnrdmd_mon.c
// SPDX-License-Identifier: GPL-2.0
/*
 * cxd2880_tnrdmd_mon.c
 * Sony CXD2880 DVB-T2/T tuner + demodulator driver
 * common monitor functions
 *
 * Copyright (C) 2016, 2017, 2018 Sony Semiconductor Solutions Corporation
 */

#include "cxd2880_common.h"
#include "cxd2880_tnrdmd_mon.h"

static const u8 rf_lvl_seq[2] = {
        0x80, 0x00,
};

int cxd2880_tnrdmd_mon_rf_lvl(struct cxd2880_tnrdmd *tnr_dmd,
                              int *rf_lvl_db)
{
        u8 rdata[2];
        int ret;

        if (!tnr_dmd || !rf_lvl_db)
                return -EINVAL;

        if (tnr_dmd->state != CXD2880_TNRDMD_STATE_ACTIVE)
                return -EINVAL;

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_DMD,
                                     0x00, 0x00);
        if (ret)
                return ret;

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_DMD,
                                     0x10, 0x01);
        if (ret)
                return ret;

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_SYS,
                                     0x00, 0x10);
        if (ret)
                return ret;

        ret = tnr_dmd->io->write_regs(tnr_dmd->io,
                                      CXD2880_IO_TGT_SYS,
                                      0x5b, rf_lvl_seq, 2);
        if (ret)
                return ret;

        usleep_range(2000, 3000);

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_SYS,
                                     0x00, 0x1a);
        if (ret)
                return ret;

        ret = tnr_dmd->io->read_regs(tnr_dmd->io,
                                     CXD2880_IO_TGT_SYS,
                                     0x15, rdata, 2);
        if (ret)
                return ret;

        if (rdata[0] || rdata[1])
                return -EINVAL;

        ret = tnr_dmd->io->read_regs(tnr_dmd->io,
                                     CXD2880_IO_TGT_SYS,
                                     0x11, rdata, 2);
        if (ret)
                return ret;

        *rf_lvl_db =
            cxd2880_convert2s_complement((rdata[0] << 3) |
                                         ((rdata[1] & 0xe0) >> 5), 11);

        *rf_lvl_db *= 125;

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_DMD,
                                     0x00, 0x00);
        if (ret)
                return ret;

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_DMD,
                                     0x10, 0x00);
        if (ret)
                return ret;

        if (tnr_dmd->rf_lvl_cmpstn)
                ret = tnr_dmd->rf_lvl_cmpstn(tnr_dmd, rf_lvl_db);

        return ret;
}

int cxd2880_tnrdmd_mon_rf_lvl_sub(struct cxd2880_tnrdmd *tnr_dmd,
                                  int *rf_lvl_db)
{
        if (!tnr_dmd || !rf_lvl_db)
                return -EINVAL;

        if (tnr_dmd->diver_mode != CXD2880_TNRDMD_DIVERMODE_MAIN)
                return -EINVAL;

        return cxd2880_tnrdmd_mon_rf_lvl(tnr_dmd->diver_sub, rf_lvl_db);
}

int cxd2880_tnrdmd_mon_internal_cpu_status(struct cxd2880_tnrdmd
                                           *tnr_dmd, u16 *status)
{
        u8 data[2] = { 0 };
        int ret;

        if (!tnr_dmd || !status)
                return -EINVAL;

        ret = tnr_dmd->io->write_reg(tnr_dmd->io,
                                     CXD2880_IO_TGT_SYS,
                                     0x00, 0x1a);
        if (ret)
                return ret;
        ret = tnr_dmd->io->read_regs(tnr_dmd->io,
                                     CXD2880_IO_TGT_SYS,
                                     0x15, data, 2);
        if (ret)
                return ret;

        *status = (data[0] << 8) | data[1];

        return 0;
}

int cxd2880_tnrdmd_mon_internal_cpu_status_sub(struct
                                               cxd2880_tnrdmd
                                               *tnr_dmd,
                                               u16 *status)
{
        if (!tnr_dmd || !status)
                return -EINVAL;

        if (tnr_dmd->diver_mode != CXD2880_TNRDMD_DIVERMODE_MAIN)
                return -EINVAL;

        return cxd2880_tnrdmd_mon_internal_cpu_status(tnr_dmd->diver_sub,
                                                      status);
}