151 lines
3.0 KiB
C
151 lines
3.0 KiB
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);
|
||
|
}
|