On 17.05.2014 04:14, Antti Palosaari wrote: > Sorry for the late review. I think that will be skipped over 2.16 as too > late... > > There is also many other DTV drivers coming from Earthsoft PT3 support > waiting for review. Anyone else willing to review? I wonder how we could > improve situation, we are simply lack of reviewers. Mike, Devin, Mauro? > > > > On 04/26/2014 11:21 PM, Matthias Schwarzott wrote: >> DVB-T was tested with 8MHz BW channels in germany >> This driver is the simplest possible, it uses automatic mode for all >> parameters (TPS). >> >> Firmware file can be extracted via get_dvb_firmware. >> >> Signed-off-by: Matthias Schwarzott <zzam@xxxxxxxxxx> >> --- >> Documentation/dvb/get_dvb_firmware | 33 +- >> drivers/media/dvb-frontends/Kconfig | 9 + >> drivers/media/dvb-frontends/Makefile | 1 + >> drivers/media/dvb-frontends/si2165.c | 890 >> ++++++++++++++++++++++++++++++ >> drivers/media/dvb-frontends/si2165.h | 61 ++ >> drivers/media/dvb-frontends/si2165_priv.h | 23 + >> 6 files changed, 1016 insertions(+), 1 deletion(-) >> create mode 100644 drivers/media/dvb-frontends/si2165.c >> create mode 100644 drivers/media/dvb-frontends/si2165.h >> create mode 100644 drivers/media/dvb-frontends/si2165_priv.h >> >> diff --git a/Documentation/dvb/get_dvb_firmware >> b/Documentation/dvb/get_dvb_firmware >> index d91b8be..26c623d 100755 >> --- a/Documentation/dvb/get_dvb_firmware >> +++ b/Documentation/dvb/get_dvb_firmware >> @@ -29,7 +29,7 @@ use IO::Handle; >> "af9015", "ngene", "az6027", "lme2510_lg", "lme2510c_s7395", >> "lme2510c_s7395_old", "drxk", "drxk_terratec_h5", >> "drxk_hauppauge_hvr930c", "tda10071", "it9135", "drxk_pctv", >> - "drxk_terratec_htc_stick", "sms1xxx_hcw"); >> + "drxk_terratec_htc_stick", "sms1xxx_hcw", "si2165"); >> >> # Check args >> syntax() if (scalar(@ARGV) != 1); >> @@ -783,6 +783,37 @@ sub sms1xxx_hcw { >> $allfiles; >> } >> >> +sub si2165 { >> + my $sourcefile = >> "model_111xxx_122xxx_driver_6_0_119_31191_WHQL.zip"; >> + my $url = "http://www.hauppauge.de/files/drivers/"; >> + my $hash = "76633e7c76b0edee47c3ba18ded99336"; >> + my $fwfile = "dvb-demod-si2165.fw"; >> + my $tmpdir = tempdir(DIR => "/tmp", CLEANUP => 1); >> + >> + checkstandard(); >> + >> + wgetfile($sourcefile, $url . $sourcefile); >> + verify($sourcefile, $hash); >> + unzip($sourcefile, $tmpdir); >> + extract("$tmpdir/Driver10/Hcw10bda.sys", 0x80788, >> 0x81E08-0x80788, "$tmpdir/fw1"); >> + >> + delzero("$tmpdir/fw1","$tmpdir/fw1-1"); >> + #verify("$tmpdir/fw1","5e0909858fdf0b5b09ad48b9fe622e70"); >> + >> + my $CRC="\x0A\xCC"; >> + my $BLOCKS_MAIN="\x27"; >> + open FW,">$fwfile"; >> + print FW "\x01\x00"; # just a version id for the driver itself >> + print FW "\x9A"; # fw version >> + print FW "\x00"; # padding >> + print FW "$BLOCKS_MAIN"; # number of blocks of main part >> + print FW "\x00"; # padding >> + print FW "$CRC"; # 16bit crc value of main part >> + appendfile(FW,"$tmpdir/fw1"); >> + >> + "$fwfile"; >> +} >> + >> # --------------------------------------------------------------- >> # Utilities > > Separate that firmware extractor to own patch. > done >> >> diff --git a/drivers/media/dvb-frontends/Kconfig >> b/drivers/media/dvb-frontends/Kconfig >> index 1469d44..0da53c2 100644 >> --- a/drivers/media/dvb-frontends/Kconfig >> +++ b/drivers/media/dvb-frontends/Kconfig >> @@ -63,6 +63,15 @@ config DVB_TDA18271C2DD >> >> Say Y when you want to support this tuner. >> >> +config DVB_SI2165 >> + tristate "Silicon Labs si2165 based" >> + depends on DVB_CORE && I2C >> + default m if !MEDIA_SUBDRV_AUTOSELECT >> + help >> + A DVB-C/T demodulator. >> + >> + Say Y when you want to support this frontend. >> + >> comment "DVB-S (satellite) frontends" >> depends on DVB_CORE >> >> diff --git a/drivers/media/dvb-frontends/Makefile >> b/drivers/media/dvb-frontends/Makefile >> index dda0bee..595dd8d 100644 >> --- a/drivers/media/dvb-frontends/Makefile >> +++ b/drivers/media/dvb-frontends/Makefile >> @@ -100,6 +100,7 @@ obj-$(CONFIG_DVB_STV0367) += stv0367.o >> obj-$(CONFIG_DVB_CXD2820R) += cxd2820r.o >> obj-$(CONFIG_DVB_DRXK) += drxk.o >> obj-$(CONFIG_DVB_TDA18271C2DD) += tda18271c2dd.o >> +obj-$(CONFIG_DVB_SI2165) += si2165.o >> obj-$(CONFIG_DVB_A8293) += a8293.o >> obj-$(CONFIG_DVB_TDA10071) += tda10071.o >> obj-$(CONFIG_DVB_RTL2830) += rtl2830.o >> diff --git a/drivers/media/dvb-frontends/si2165.c >> b/drivers/media/dvb-frontends/si2165.c >> new file mode 100644 >> index 0000000..3e69a32 >> --- /dev/null >> +++ b/drivers/media/dvb-frontends/si2165.c >> @@ -0,0 +1,890 @@ >> +/* >> + Driver for Silicon Labs SI2165 DVB-C/-T Demodulator >> + >> + Copyright (C) 2013-2014 Matthias Schwarzott <zzam@xxxxxxxxxx> >> + >> + This program is free software; you can redistribute it and/or modify >> + it under the terms of the GNU General Public License as published by >> + the Free Software Foundation; either version 2 of the License, or >> + (at your option) any later version. >> + >> + This program is distributed in the hope that it will be useful, >> + but WITHOUT ANY WARRANTY; without even the implied warranty of >> + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the >> + GNU General Public License for more details. >> + >> + References: >> + >> http://www.silabs.com/Support%20Documents/TechnicalDocs/Si2165-short.pdf >> +*/ >> + >> +#include <linux/delay.h> >> +#include <linux/errno.h> >> +#include <linux/init.h> >> +#include <linux/kernel.h> >> +#include <linux/module.h> >> +#include <linux/string.h> >> +#include <linux/slab.h> >> +#include <linux/firmware.h> >> + >> +#include "dvb_frontend.h" >> +#include "dvb_math.h" >> +#include "si2165_priv.h" >> +#include "si2165.h" >> + >> +/* Hauppauge WinTV-HVR-930C-HD 1113xx uses 16.000 MHz xtal */ >> + >> +struct si2165_state { >> + struct i2c_adapter *i2c; >> + >> + struct dvb_frontend frontend; >> + >> + struct si2165_config config; >> + >> + /* chip revision */ >> + u8 revcode; >> + /* chip type */ >> + u8 chip_type; >> + >> + /* calculated by xtal and div settings */ >> + u32 fvco_hz; >> + u32 sys_clk; >> + u32 adc_clk; >> + >> + /* depends on adc_clk and Ovr mode */ >> + u32 fe_clk; >> + >> + bool m_has_dvbc; >> + bool m_has_dvbt; > > m_ prefix is for Hungarian notation of member variables. It is not > suitable for Kernel style. > removed. I only use this part of hungarian notation. Sometimes its bad to switch between C and C++ :) > >> + bool firmware_loaded; >> +}; >> + >> +#define DEBUG_OTHER 0x01 >> +#define DEBUG_I2C_WRITE 0x02 >> +#define DEBUG_I2C_READ 0x04 >> +#define DEBUG_REG_READ 0x08 >> +#define DEBUG_REG_WRITE 0x10 >> +#define DEBUG_FW_LOAD 0x20 >> + >> +static int debug = 0x00; >> + >> +#define dprintk(args...) \ >> + do { \ >> + if (debug & DEBUG_OTHER) \ >> + printk(KERN_DEBUG "si2165: " args); \ >> + } while (0) >> + >> +#define deb_i2c_write(args...) \ >> + do { \ >> + if (debug & DEBUG_I2C_WRITE) \ >> + printk(KERN_DEBUG "si2165: i2c write: " args); \ >> + } while (0) >> + >> +#define deb_i2c_read(args...) \ >> + do { \ >> + if (debug & DEBUG_I2C_READ) \ >> + printk(KERN_DEBUG "si2165: i2c read: " args); \ >> + } while (0) >> + >> +#define deb_readreg(args...) \ >> + do { \ >> + if (debug & DEBUG_REG_READ) \ >> + printk(KERN_DEBUG "si2165: reg read: " args); \ >> + } while (0) >> + >> +#define deb_writereg(args...) \ >> + do { \ >> + if (debug & DEBUG_REG_WRITE) \ >> + printk(KERN_DEBUG "si2165: reg write: " args); \ >> + } while (0) >> + >> +#define deb_fw_load(args...) \ >> + do { \ >> + if (debug & DEBUG_FW_LOAD) \ >> + printk(KERN_DEBUG "si2165: fw load: " args); \ >> + } while (0) > > Could you consider kernel dynamic debugs which are todays norm? See for > example si2168 driver as a example. IMHO it is not hard requirement, but > still... > TODO >> + >> +static int si2165_write(struct si2165_state *state, const u16 reg, >> + const u8 *src, const int count) >> +{ >> + int ret; >> + struct i2c_msg msg; >> + u8 buf[2 + 4]; /* write a maximum of 4 bytes of data */ > > Here should be empty line between variable definitions and code. IIRC > latest checkpatch.pl catch that. > done. The version from git://linuxtv.org/media_tree.git does not have this warning. >> + if (count + 2 > sizeof(buf)) { >> + dev_warn(&state->i2c->dev, >> + "%s: i2c wr reg=%04x: count=%d is too big!\n", >> + KBUILD_MODNAME, reg, count); >> + return -EINVAL; >> + } >> + buf[0] = reg >> 8; >> + buf[1] = reg & 0xff; >> + memcpy(buf + 2, src, count); >> + >> + msg.addr = state->config.i2c_addr; >> + msg.flags = 0; >> + msg.buf = buf; >> + msg.len = count + 2; >> + >> + if (debug & DEBUG_I2C_WRITE) { >> + int i; >> + deb_i2c_write("reg: 0x%04x, data:", reg); >> + for (i = 0; i < count; i++) >> + printk(KERN_CONT " %02x", src[i]); >> + printk(KERN_CONT "\n"); >> + } >> + >> + ret = i2c_transfer(state->i2c, &msg, 1); >> + >> + if (ret != 1) { >> + dev_err(&state->i2c->dev, "%s: ret == %d\n", __func__, ret); >> + return -EREMOTEIO; >> + } > > Could be nice to return error from i2c_transfer(). > done. >> + >> + return 0; >> +} >> + >> +static int si2165_read(struct si2165_state *state, >> + const u16 reg, u8 *val, const size_t count) > > You used int on si2165_write(), try to make decision which one is > correct and keep it. > changed to int. >> +{ >> + int ret; >> + u8 reg_buf[] = { reg >> 8, reg & 0xff }; >> + struct i2c_msg msg[] = { >> + { .addr = state->config.i2c_addr, >> + .flags = 0, .buf = reg_buf, .len = 2 }, >> + { .addr = state->config.i2c_addr, >> + .flags = I2C_M_RD, .buf = val, .len = count }, >> + }; >> + >> + ret = i2c_transfer(state->i2c, msg, 2); >> + >> + if (ret != 2) { >> + dev_err(&state->i2c->dev, "%s: error (addr %02x reg %04x >> error (ret == %i)\n", >> + __func__, state->config.i2c_addr, reg, ret); >> + if (ret < 0) >> + return ret; >> + else >> + return -EREMOTEIO; >> + } >> + >> + if (debug & DEBUG_I2C_READ) { >> + int i; >> + deb_i2c_read("reg: 0x%04x, data:", reg); >> + for (i = 0; i < count; i++) >> + printk(KERN_CONT " %02x", val[i]); >> + printk(KERN_CONT "\n"); >> + } > > There is formatter %pH (or something like) for that. > Done, it is "%*ph". >> + return 0; >> +} >> + >> +static int si2165_readreg8(struct si2165_state *state, >> + const u16 reg, u8 *val) >> +{ >> + int ret; >> + ret = si2165_read(state, reg, val, 1); >> + deb_readreg("R(0x%04x)=0x%02x\n", reg, *val); >> + return ret; >> +} >> + >> +static int si2165_readreg16(struct si2165_state *state, >> + const u16 reg, u16 *val) >> +{ >> + u8 buf[2]; >> + int ret = si2165_read(state, reg, buf, 2); >> + *val = buf[0] | buf[1] << 8; >> + deb_readreg("R(0x%04x)=0x%04x\n", reg, *val); >> + return ret; >> +} >> + >> +#if 0 >> +static int si2165_readreg24(struct si2165_state *state, >> + const u16 reg, u32 *val) >> +{ >> + u8 buf[3]; >> + int ret = si2165_read(state, reg, buf, 3); >> + *val = buf[0] | buf[1] << 8 | buf[2] << 16; >> + deb_readreg("R(0x%04x)=0x%06x\n", reg, *val); >> + return ret; >> +} >> + >> +static int si2165_readreg32(struct si2165_state *state, >> + const u16 reg, u32 *val) >> +{ >> + u8 buf[4]; >> + int ret = si2165_read(state, reg, buf, 4); >> + *val = buf[0] | buf[1] << 8 | buf[2] << 16 | buf[3] << 24; >> + deb_readreg("R(0x%04x)=0x%08x\n", reg, *val); >> + return ret; >> +} >> +#endif > > Personally, I don't like presenting shorthand functions for multi-byte > readings like that. But if you think it is good then leave it. > This is for combining the read bytes to correct order. >> + >> + >> +static int si2165_writereg8(struct si2165_state *state, const u16 >> reg, u8 val) >> +{ >> + return si2165_write(state, reg, &val, 1); >> +} >> + >> +static int si2165_writereg16(struct si2165_state *state, const u16 >> reg, u16 val) >> +{ >> + u8 buf[2] = { val & 0xff, (val >> 8) & 0xff }; >> + return si2165_write(state, reg, buf, 2); >> +} >> + >> +static int si2165_writereg24(struct si2165_state *state, const u16 >> reg, u32 val) >> +{ >> + u8 buf[3] = { val & 0xff, (val >> 8) & 0xff, (val >> 16) & 0xff }; >> + return si2165_write(state, reg, buf, 3); >> +} >> + >> +static int si2165_writereg32(struct si2165_state *state, const u16 >> reg, u32 val) >> +{ >> + u8 buf[4] = { >> + val & 0xff, >> + (val >> 8) & 0xff, >> + (val >> 16) & 0xff, >> + (val >> 24) & 0xff >> + }; >> + return si2165_write(state, reg, buf, 4); >> +} >> + >> +static int si2165_writereg_mask8(struct si2165_state *state, const >> u16 reg, >> + u8 val, u8 mask) >> +{ >> + int ret; >> + u8 tmp; >> + >> + if (mask != 0xff) { >> + ret = si2165_readreg8(state, reg, &tmp); >> + if (ret < 0) >> + goto err; >> + >> + val &= mask; >> + tmp &= ~mask; >> + val |= tmp; >> + } >> + >> + ret = si2165_writereg8(state, reg, val); >> +err: >> + return ret; >> +} >> + >> +static int si2165_get_tune_settings(struct dvb_frontend *fe, >> + struct dvb_frontend_tune_settings *s) >> +{ >> + s->min_delay_ms = 1000; >> + return 0; >> +} >> + >> +static int si2165_init_pll(struct si2165_state *state) >> +{ >> + u8 ref_freq_MHz = state->config.ref_freq_MHz; > > 1MHz is quite big reference frequency step? It was because short datasheet of si2165 says: "4, 16, 20, 24, or 27 MHz clock/crystal reference" I thought 1MHz precision is enough. The two devices I currently support both use a 16MHz chrystal. But I wanted to have it because the followup device (930C-HD with vid=0xb131) has in the inf file: "24MHz clock from Si2158 to Si2165 demod". Now I changed it to 1Hz precision as some other drivers do. > >> + /* u8 val; */ > > excessive comment removed. > >> + u8 divr = 1; /* 1..7 */ >> + u8 divp = 1; /* only 1 or 4 */ >> + u8 divn = 56; /* 1..63 */ >> + > > excessive newline removed. > >> + u8 divm = 8; >> + u8 divl = 12; >> + u8 buf[4]; >> + >> + /* ref_freq / divr must be between 4 and 16 MHz */ >> + if (ref_freq_MHz > 16) >> + divr = 2; >> + >> + /* now select divn and divp such that fvco is in 1624..1824 MHz */ >> + if (1624 * divr > ref_freq_MHz * 2 * 63) >> + divp = 4; >> + >> + /* to get exactly the same as the windows driver does */ >> + if (ref_freq_MHz == 16) >> + divn = 56; >> + else { >> + /* is this already correct regarding rounding? */ >> + divn = 1624 * divr / (ref_freq_MHz * 2 * divp); >> + } > > That logic does not look correct. PLL dividers are usually calculated in > a order: > 1) calculate reference divider (divREF) > 2) calculate output divider (divOUT) > 3) calculate VCO freq (fVCO = fOUT * divOUT) > 4) calculate N & NF, pllN = fVCO * divREF / fREF; > > N = interger part, NF = fractional part. NF only when PLL is Fractional N. > > Biggest thing looks wrong is that: IF (fREF == 16MHz) THEN pllN = 56. > that was just modifying the value in the valid range to yield exactly the same result and not just a valid one. For now I return the exact same values as windows hardcoded. The calculation still is there. > > There should be parenthesis for none or both conditions. checkpatch.pl? > Removed them. It did not say anything about this. > >> + >> + /* adc_clk and sys_clk depend on xtal and pll settings */ >> + /* only calculate once as long as the pll settings are not >> modified */ >> + if (state->adc_clk == 0) { >> + u32 fvco_hz = ref_freq_MHz * 1000000ull / divr >> + * 2 * divn * divp; >> + state->adc_clk = fvco_hz / (divm * 4); >> + state->sys_clk = fvco_hz / (divl * 2); >> + } >> + >> + /* write pll registers 0x00a0..0x00a3 at once */ >> + buf[0] = divl; >> + buf[1] = divm; >> + buf[2] = (divn & 0x3f) | ((divp == 1) ? 0x40 : 0x00) | 0x80; >> + buf[3] = divr; >> + si2165_write(state, 0x00a0 /* first pll reg */, buf, 4); > > That kinf of comments in a parameter list do not look typical for kernel > style. > I changed it. >> + >> + return 0; >> +} > > Also that function is called from init(), maybe you could inline all > that stuff to that function instead. > I tried to keep the functions somewhat easy to overview. >> + >> +static bool si2165_wait_init_done(struct si2165_state *state) >> +{ >> + int ret = false; >> + u8 val; >> + int i; >> + for (i = 0; i < 10; i++) { >> + si2165_readreg8(state, 0x0054 /* init_done */, &val); >> + if (val == 0x01) { >> + ret = true; >> + break; >> + } >> + msleep(1); > > Too small msleep(), see timers howto. Also checkpatch.pl should be able > to find that too, you haven't ran it. > well, this was not listed. I don't yet know exactly how to solve it here. Datasheet says to either poll or wait for around 5ms (depending on sys_clk, but I have no clue about the dependency). checkpatch had only these kind of warnings: WARNING: line over 80 characters WARNING: msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt WARNING: Prefer [subsystem eg: netdev]_cont([subsystem]dev, ... then dev_cont(dev, ... then pr_cont(... to printk(KERN_CONT ... WARNING: Prefer [subsystem eg: netdev]_dbg([subsystem]dev, ... then dev_dbg(dev, ... then pr_debug(... to printk(KERN_DEBUG ... WARNING: suspect code indent for conditional statements (8, 18) yeah, I replace msleep by usleep_range. >> + } >> + return ret; >> +} > > Whole function is called from one place and it is very trivial, so that > function is mostly reduntant. Move logic to where it is now called. > Also, use jiffies for timeouts like that. See si2168 for example. > > >> + >> +static int si2165_upload_firmware_block(struct si2165_state *state, >> const u8 *data, u32 len, u32 *poffset, u32 block_count) >> +{ >> + u8 buf_ctrl[4] = { 0x00, 0x00, 0x00, 0xc0 }; >> + u8 wordcount; >> + u32 cur_block = 0; >> + u32 offset = poffset ? *poffset : 0; >> + if (len < 4) >> + return -EINVAL; >> + if (len % 4 != 0) >> + return -EINVAL; >> + >> + deb_fw_load("si2165_upload_firmware_block called with len=0x%x >> offset=0x%x blockcount=0x%x\n", >> + len, offset, block_count); >> + while (offset+12 <= len && cur_block < block_count) { >> + deb_fw_load("si2165_upload_firmware_block in while len=0x%x >> offset=0x%x cur_block=0x%x blockcount=0x%x\n", >> + len, offset, cur_block, block_count); >> + wordcount = data[offset]; >> + if (wordcount < 1 || data[offset+1] || data[offset+2] || >> data[offset+3]) { >> + dev_warn(&state->i2c->dev, "%s: bad fw data[0..3] = %02x >> %02x %02x %02x\n", KBUILD_MODNAME, data[0], data[1], data[2], data[3]); >> + return -EINVAL; >> + } >> + >> + if (offset + 8 + wordcount * 4 > len) { >> + dev_warn(&state->i2c->dev, "%s: len is too small for >> block len=%d, wordcount=%d\n", KBUILD_MODNAME, len, wordcount); >> + return -EINVAL; >> + } >> + >> + buf_ctrl[0] = wordcount - 1; >> + >> + si2165_write(state, 0x0364, buf_ctrl, 4); >> + si2165_write(state, 0x0368, data+offset+4, 4); > > CodingStyle. That driver has a lot of coding style mistakes ==> you must > run checkpatch.pl and fix all. > >> + >> + offset += 8; >> + >> + while (wordcount > 0) { >> + si2165_write(state, 0x36c, data+offset, 4); >> + wordcount--; >> + offset += 4; >> + } >> + cur_block++; >> + } >> + >> + deb_fw_load("si2165_upload_firmware_block after while len=0x%x >> offset=0x%x cur_block=0x%x blockcount=0x%x\n", >> + len, offset, cur_block, block_count); >> + >> + if (poffset) >> + *poffset = offset; >> + >> + deb_fw_load("si2165_upload_firmware_block returned offset=0x%x\n", >> + offset); >> + >> + return 0; >> +} >> + >> +static int si2165_upload_firmware(struct si2165_state *state) >> +{ >> + /* int ret; */ >> + u8 val[3]; >> + u16 val16; >> + int ret; >> + >> + const struct firmware *fw = NULL; >> + u8 *fw_file = SI2165_FIRMWARE; >> + const u8 *data; >> + u32 len; >> + u32 offset; >> + u8 patch_version; >> + u8 block_count; >> + u16 crc_expected; >> + >> + /* request the firmware, this will block and timeout */ >> + ret = request_firmware(&fw, fw_file, state->i2c->dev.parent); >> + if (ret) { >> + dev_warn(&state->i2c->dev, "%s: firmare file '%s' not found\n", >> + KBUILD_MODNAME, fw_file); >> + goto err; >> + } >> + >> + data = fw->data; >> + len = fw->size; >> + >> + dev_info(&state->i2c->dev, "%s: downloading firmware from file >> '%s' size=%d\n", >> + KBUILD_MODNAME, fw_file, len); >> + >> + if (len % 4 != 0) { >> + dev_warn(&state->i2c->dev, "%s: firmware size is not multiple >> of 4\n", >> + KBUILD_MODNAME); >> + ret = -EINVAL; >> + goto err; >> + } >> + >> + /* check header (8 bytes) */ >> + if (len < 8) { >> + dev_warn(&state->i2c->dev, "%s: firmware header is missing\n", >> + KBUILD_MODNAME); >> + ret = -EINVAL; >> + goto err; >> + } >> + >> + if (data[0] != 1 || data[1] != 0) { >> + dev_warn(&state->i2c->dev, "%s: firmware file version is >> wrong\n", >> + KBUILD_MODNAME); >> + ret = -EINVAL; >> + goto err; >> + } >> + >> + patch_version = data[2]; >> + block_count = data[4]; >> + crc_expected = data[7] << 8 | data[6]; >> + >> + /* start uploading fw */ >> + si2165_writereg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, 0x00); >> + si2165_writereg8(state, 0x00c0 /* rst_all */, 0x00); >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + >> + si2165_readreg8(state, 0x035c /* en_rst_error */, val); /* >> returned 0x03 */ >> + si2165_readreg8(state, 0x035c /* en_rst_error */, val); /* >> returned 0x03 */ >> + si2165_writereg8(state, 0x035c /* en_rst_error */, 0x02); >> + >> + /* start right after the header */ >> + offset = 8; >> + >> + dev_info(&state->i2c->dev, "%s: si2165_upload_firmware extracted >> patch_version=0x%02x, block_count=0x%02x, crc_expected=0x%04x\n", >> + KBUILD_MODNAME, patch_version, block_count, >> crc_expected); >> + >> + ret = si2165_upload_firmware_block(state, data, len, &offset, 1); >> + >> + si2165_writereg8(state, 0x0344 /* patch_version */, patch_version); >> + >> + ret = si2165_writereg8(state, 0x0379 /* rst_crc */, 0x01); >> + if (ret) >> + return ret; >> + >> + ret = si2165_upload_firmware_block(state, data, len, &offset, >> block_count); >> + >> + if (ret) { >> + dev_err(&state->i2c->dev, "%s: firmare could not be uploaded\n", >> + KBUILD_MODNAME); >> + goto err; >> + } >> + >> + ret = si2165_readreg16(state, 0x037a /* crc */, &val16); /* >> returned 0xcc0a */ >> + if (ret) >> + goto err; >> + >> + if (val16 != crc_expected) { >> + dev_err(&state->i2c->dev, "%s: firmware crc mismatch %04x != >> %04x\n", KBUILD_MODNAME, val16, crc_expected); >> + ret = -EINVAL; >> + goto err; >> + } >> + >> + ret = si2165_upload_firmware_block(state, data, len, &offset, 5); >> + if (ret) >> + goto err; >> + >> + if (len != offset) { >> + dev_err(&state->i2c->dev, "%s: firmare len mismatch %04x != >> %04x\n", KBUILD_MODNAME, len, offset); >> + ret = -EINVAL; >> + goto err; >> + } >> + >> + /* reset watchdog error register, using auto return value >> rst_wdog_error */ >> + si2165_writereg_mask8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, 0x02, 0x02); >> + >> + /* enable reset on error */ >> + si2165_writereg_mask8(state, 0x035c /* en_rst_error */, 0x01, 0x01); >> + >> + dev_info(&state->i2c->dev, "%s: fw load finished\n", >> KBUILD_MODNAME); >> + >> + ret = 0; >> + state->firmware_loaded = true; >> +err: >> + if (fw) { >> + release_firmware(fw); >> + fw = NULL; >> + } >> + >> + return ret; >> +} > > > That firmware downloading looks overall very long a also a bit complex. > I didn't looked it through carefully. Maybe you could take a look to > si2168 or af9035 to see if that could be shorten. > > >> + >> +static int si2165_init_dsp(struct si2165_state *state) >> +{ >> + u8 val; >> + u8 patch_version = 0x00; >> + >> + si2165_readreg8(state, 0x0344 /* patch_version */, >> &patch_version); /* returned 0x00 */ >> + >> + si2165_writereg8(state, 0x00cb, 0x00); >> + if (patch_version == 0x00) >> + si2165_writereg8(state, 0x0344 /* patch_version */, 0x00); >> + >> + si2165_writereg32(state, 0x0348 /* dsp_addr_jump */, 0xf4000000); >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, &val); /* returned 0x01 */ >> + >> + if (patch_version == 0x00) >> + si2165_upload_firmware(state); >> + >> + return 0; >> +} > > Maybe that trivial function could be integrated to where it is called. > inlined >> + >> +static int si2165_init(struct dvb_frontend *fe) >> +{ >> + struct si2165_state *state = fe->demodulator_priv; >> + u8 val; >> + >> + dprintk("%s: called\n", __func__); >> + >> + /* powerup */ >> + si2165_writereg8(state, 0x0000 /* chip_mode */, >> state->config.chip_mode); >> + si2165_writereg8(state, 0x0104 /* dsp_clock_enable */, 0x01); >> + si2165_readreg8(state, 0x0000 /* chip_mode */, &val); >> + if (val != state->config.chip_mode) { >> + dev_err(&state->i2c->dev, "%s: could not set chip_mode\n", >> + KBUILD_MODNAME); >> + return -EINVAL; >> + } >> + >> + si2165_writereg8(state, 0x018b /* agc_if_tri */, 0x00); >> + si2165_writereg8(state, 0x0190 /* agc_if_slr */, 0x01); >> + si2165_readreg8(state, 0x0170 /* agc2_{freeze,pola,buftype} */, >> &val); /* returned 0x00 */ >> + si2165_writereg8(state, 0x0170 /* agc2_{freeze,pola,buftype} */, >> 0x00); >> + si2165_readreg8(state, 0x0170 /* agc2_{freeze,pola,buftype} */, >> &val); /* returned 0x00 */ >> + si2165_writereg8(state, 0x0170 /* agc2_{freeze,pola,buftype} */, >> 0x00); >> + si2165_writereg8(state, 0x0171 /* agc2_clkdiv */, 0x07); >> + si2165_writereg8(state, 0x0646 /* rssi_pad_ctrl */, 0x00); >> + si2165_readreg8(state, 0x0641 /* >> en_rssi,start_rssi,rssi_update_time */, &val); /* returned 0x00 */ >> + si2165_writereg8(state, 0x0641 /* >> en_rssi,start_rssi,rssi_update_time */, 0x00); >> + si2165_writereg8(state, 0x00e0 /* adc_sampling_mode */, 0x00); >> + >> + si2165_init_pll(state); >> + >> + si2165_writereg8(state, 0x0050 /* chip_init */, 0x01); >> + si2165_writereg8(state, 0x0096 /* start_init */, 0x01); >> + if (!si2165_wait_init_done(state)) { >> + dev_err(&state->i2c->dev, "%s: init_done was not set\n", >> + KBUILD_MODNAME); >> + return -EINVAL; >> + } >> + >> + si2165_writereg8(state, 0x0050 /* chip_init */, 0x00); >> + >> + si2165_writereg16(state, 0x0470 /* ber_pkt */, 0x7530); >> + >> + si2165_init_dsp(state); >> + >> + si2165_writereg8(state, 0x012a /* adc_ri1 */, 0x46); >> + si2165_writereg8(state, 0x012c /* adc_ri3 */, 0x00); >> + si2165_writereg8(state, 0x012e /* adc_ri5 */, 0x0a); >> + si2165_writereg8(state, 0x012f /* adc_ri6 */, 0xff); >> + si2165_writereg8(state, 0x0123 /* adc_ri8 */, 0x70); >> + >> + return 0; >> +} >> + >> +static int si2165_i2c_gate_ctrl(struct dvb_frontend *fe, int enable) >> +{ >> + struct si2165_state *state = fe->demodulator_priv; >> + u8 val = enable ? 0x01 : 0x00; >> + return si2165_writereg8(state, 0x0001 /* i2c passthru */, val); >> +} >> + >> +static int si2165_sleep(struct dvb_frontend *fe) >> +{ >> + struct si2165_state *state = fe->demodulator_priv; >> + dprintk("%s: called\n", __func__); >> + si2165_writereg8(state, 0x0104 /* dsp clock enable */, 0x00); >> + si2165_writereg8(state, 0x0000 /* chip mode */, SI2165_MODE_OFF); >> + return 0; >> +} >> + >> +static int si2165_read_status(struct dvb_frontend *fe, fe_status_t >> *status) >> +{ >> + u8 fec_lock = 0; >> + struct si2165_state *state = fe->demodulator_priv; >> + >> + if (!state->m_has_dvbt) >> + return -EINVAL; >> + >> + /* seq1 */ >> + si2165_readreg8(state, 0x4e0 /* fec_lock */, &fec_lock); >> + *status = 0; >> + if (fec_lock & 0x01) { >> + *status |= FE_HAS_SIGNAL; >> + *status |= FE_HAS_CARRIER; >> + *status |= FE_HAS_VITERBI; >> + *status |= FE_HAS_SYNC; >> + *status |= FE_HAS_LOCK; >> + } >> + >> + return 0; >> +} >> + >> +static int si2165_set_parameters(struct dvb_frontend *fe) >> +{ >> + struct dtv_frontend_properties *p = &fe->dtv_property_cache; >> + struct si2165_state *state = fe->demodulator_priv; >> + u8 val[3]; >> + u32 IF; >> + u32 dvb_rate = 0; >> + >> + dprintk("%s: called\n", __func__); >> + >> + if (!fe->ops.tuner_ops.get_if_frequency) { >> + pr_err("Error: get_if_frequency() not defined at tuner. Can't >> work without it!\n"); > > You should use dev_ not pr_. > replaced. >> + return -EINVAL; >> + } >> + >> + /* If Oversampling mode Ovr4 is used */ >> + state->fe_clk = state->adc_clk; >> + >> + if (state->fe_clk == 0) { >> + pr_err("Error: fe_clk is 0\n"); >> + return -EINVAL; >> + } >> + >> + if (!state->m_has_dvbt) >> + return -EINVAL; >> + >> + if (p->bandwidth_hz > 0) >> + dvb_rate = p->bandwidth_hz * 8 / 7; >> + else >> + dvb_rate = 8 * 8 / 7; >> + >> + si2165_writereg8(state, 0x00ec /* standard */, 0x01); >> + si2165_writereg8(state, 0x00a0 /* pll_divl */, 0x0c); >> + >> + si2165_readreg8(state, 0x00e0 /* adc_sampling_mode */, val); /* >> returned 0x00 */ >> + si2165_writereg32(state, 0x00e8 /* if_freq_shift */, 0x00000000); >> + si2165_writereg8(state, 0x08f8 /* unknown_wr8 */, 0x00); >> + si2165_readreg8(state, 0x04e4 /* ts_output_mode */, val); /* >> returned 0x21 */ >> + si2165_writereg8(state, 0x04e4 /* ts_output_mode */, 0x20); >> + si2165_writereg16(state, 0x04ef /* ts_data_tri */, 0x00fe); >> + si2165_writereg24(state, 0x04f4 /* ts_data0-3_slr */, 0x555555); >> + si2165_readreg8(state, 0x04e4 /* ts_output_mode */, val); /* >> returned 0x20 */ >> + si2165_writereg8(state, 0x04e4 /* ts_output_mode */, 0x20); >> + si2165_readreg8(state, 0x04e5 /* ts_clock */, val); /* returned >> 0x03 */ >> + si2165_writereg8(state, 0x04e5 /* ts_clock */, 0x03); >> + si2165_readreg8(state, 0x04e5 /* ts_clock */, val); /* returned >> 0x03 */ >> + si2165_writereg8(state, 0x04e5 /* ts_clock */, 0x01); >> + si2165_writereg16(state, 0x0308 /* bandwidth */, 0x0320); >> + si2165_writereg32(state, 0x00e4 /* oversamp */, 0x03100000); >> + si2165_writereg8(state, 0x031c /* impulsive_noise_remover */, 0x01); >> + si2165_writereg8(state, 0x00cb /* unknown_wr8 */, 0x00); >> + si2165_writereg8(state, 0x016e /* agc2_min */, 0x41); >> + si2165_writereg8(state, 0x016c /* agc2_kacq */, 0x0e); >> + si2165_writereg8(state, 0x016d /* agc2_kloc */, 0x10); >> + si2165_writereg8(state, 0x015b /* agc_unfreeze_thr */, 0x03); >> + si2165_writereg8(state, 0x0150 /* agc_crestf_dbx8 */, 0x78); >> + si2165_writereg8(state, 0x01a0 /* aaf_crestf_dbx8 */, 0x78); >> + si2165_writereg8(state, 0x01c8 /* aci_crestf_dbx8 */, 0x68); >> + si2165_writereg16(state, 0x030c /* freq_sync_range */, 0x0064); >> + si2165_readreg8(state, 0x0387 /* gp_reg0 */, val); /* returned >> 0x00 */ >> + si2165_writereg8(state, 0x0387 /* gp_reg0 */, 0x00); >> + si2165_writereg32(state, 0x0348 /* dsp_addr_jump */, 0xf4000000); >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + si2165_writereg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, 0x00); >> + si2165_writereg8(state, 0x00c0 /* rst_all */, 0x00); >> + si2165_writereg32(state, 0x0384 /* gp_reg0 */, 0x00000000); >> + si2165_writereg8(state, 0x02e0 /* start_synchro */, 0x01); >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + si2165_writereg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, 0x00); >> + si2165_writereg8(state, 0x00c0 /* rst_all */, 0x00); >> + si2165_writereg32(state, 0x0384 /* gp_reg0 */, 0x00000000); >> + si2165_writereg8(state, 0x02e0 /* start_synchro */, 0x01); >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + si2165_readreg8(state, 0x0118 /* dvb-c standard support */, val); >> /* returned 0x07 */ >> + si2165_readreg8(state, 0x0023 /* hardware_rev */, val); /* >> returned 0x03 */ >> + si2165_writereg8(state, 0x018b /* agc_if_tri */, 0x00); >> + si2165_writereg8(state, 0x08f8 /* unknown_wr8 */, 0x00); >> + si2165_readreg8(state, 0x04e4 /* ts_output_mode */, val); /* >> returned 0x20 */ >> + si2165_writereg8(state, 0x04e4 /* ts_output_mode */, 0x20); >> + si2165_writereg16(state, 0x04ef /* ts_data_tri */, 0x00fe); >> + si2165_writereg24(state, 0x04f4 /* ts_data0-3_slr */, 0x555555); >> + si2165_readreg8(state, 0x04e4 /* ts_output_mode */, val); /* >> returned 0x20 */ >> + si2165_writereg8(state, 0x04e4 /* ts_output_mode */, 0x20); >> + si2165_readreg8(state, 0x04e5 /* ts_clock */, val); /* returned >> 0x01 */ >> + si2165_writereg8(state, 0x04e5 /* ts_clock */, 0x01); >> + si2165_readreg8(state, 0x04e5 /* ts_clock */, val); /* returned >> 0x01 */ >> + si2165_writereg8(state, 0x04e5 /* ts_clock */, 0x01); > > OK, maybe those own functions for 8/16/24/32 are fine as there is that > many cases. > >> + >> + if (dvb_rate == 0) { >> + pr_err("Error: dvb_rate is 0\n"); >> + return -EINVAL; >> + } > > Dead code. Please check all the other error checks too and consider if > those are really needed or not. It is good idea to check all needed > parameters just beginning of function and jump out before any I/O if > invalid params. > removed. >> + >> + if (fe->ops.tuner_ops.set_params) { >> + if (fe->ops.i2c_gate_ctrl) >> + fe->ops.i2c_gate_ctrl(fe, 1); >> + fe->ops.tuner_ops.set_params(fe); >> + if (fe->ops.i2c_gate_ctrl) >> + fe->ops.i2c_gate_ctrl(fe, 0); >> + } > > Get the rid of those i2c_gate_ctrl() stuff. It is tuner who is > responsible of calling those. Even better if you could implement proper > I2C adapder (I2C mux) which handles gating transparently. > removed the call to i2c_gate_ctrl. >> + >> + { >> + s64 if_freq_shift; > Is signed needed? > yes, the register is defined signed. And in some cases signed numbers need to be written. For example a spectrum inversion in some cases is handled by multiplying this freq_shift by -1. >> + u32 reg_value; >> + fe->ops.tuner_ops.get_if_frequency(fe, &IF); >> + >> + if_freq_shift = IF; >> + if_freq_shift <<= 29; >> + if (state->fe_clk > 0) > Can that clock be zero at all? not if the code is correct > >> + if_freq_shift /= (u64)state->fe_clk; >> + reg_value = ((u32)if_freq_shift) & 0x1fffffff; > > Just thinking what happens here. So you have a signed number, which is > cast to unsigned and masked to X bit len. I think there is no need for > signed at all. > 1) s_if = u_if; > 2) s_if <<= 29; > 3) s_if /= clk; > 4) u_32tmp = s_if; > 5) u_32val = u_32tmp & 0x1fffffff; > Not yet, but see comment above. > >> + >> + si2165_writereg32(state, 0x00e8 /* if_freq_shift */, reg_value); >> + } >> + >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + si2165_writereg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, 0x00); >> + si2165_writereg8(state, 0x00c0 /* rst_all */, 0x00); >> + si2165_writereg32(state, 0x0384 /* gp_reg0 */, 0x00000000); >> + si2165_writereg8(state, 0x02e0 /* start_synchro */, 0x01); >> + si2165_readreg8(state, 0x0341 /* >> boot_done,rst_wdog_error,wdog_error */, val); /* returned 0x01 */ >> + >> + return 0; >> +} >> + >> +static void si2165_release(struct dvb_frontend *fe) >> +{ >> + struct si2165_state *state = fe->demodulator_priv; >> + dprintk("%s: called\n", __func__); >> + kfree(state); >> +} >> + >> +static struct dvb_frontend_ops si2165_ops = { >> + .info = { >> + .name = "SI2165", > > Nit, but that chip is Si2165 not SI2165. Same everywhere. > Also use full name "Silicon Labs Si2165" as that string is shown to > application and user. > > Also, delsys are missing. > delsys is filled in later in preparation for si2161 support. > >> + .caps = FE_CAN_FEC_1_2 | >> + FE_CAN_FEC_2_3 | >> + FE_CAN_FEC_3_4 | >> + FE_CAN_FEC_5_6 | >> + FE_CAN_FEC_7_8 | >> + FE_CAN_FEC_AUTO | >> + FE_CAN_QPSK | >> + FE_CAN_QAM_16 | >> + FE_CAN_QAM_32 | >> + FE_CAN_QAM_64 | >> + FE_CAN_QAM_128 | >> + FE_CAN_QAM_256 | >> + FE_CAN_QAM_AUTO | >> + FE_CAN_TRANSMISSION_MODE_AUTO | >> + FE_CAN_GUARD_INTERVAL_AUTO | >> + FE_CAN_HIERARCHY_AUTO | >> + FE_CAN_MUTE_TS | >> + FE_CAN_TRANSMISSION_MODE_AUTO | >> + FE_CAN_RECOVER >> + }, >> + >> + .get_tune_settings = si2165_get_tune_settings, >> + >> + .init = si2165_init, >> + .sleep = si2165_sleep, >> + >> + .i2c_gate_ctrl = si2165_i2c_gate_ctrl, >> + >> + .set_frontend = si2165_set_parameters, >> + .read_status = si2165_read_status, >> + >> + .release = si2165_release, >> +}; >> + >> +struct dvb_frontend *si2165_attach(const struct si2165_config >> *config, struct i2c_adapter *i2c) >> +{ >> + struct si2165_state *state = NULL; >> + int n; >> + >> + if (config == NULL) >> + goto error; >> + >> + /* allocate memory for the internal state */ >> + state = kzalloc(sizeof(struct si2165_state), GFP_KERNEL); >> + if (state == NULL) >> + goto error; >> + >> + /* setup the state */ >> + state->i2c = i2c; >> + state->config = *config; >> + >> + if (state->config.ref_freq_MHz < 4 || state->config.ref_freq_MHz >> > 27) { >> + dev_info(&state->i2c->dev, "%s: ref_freq of %d MHz not >> supported by this driver\n", >> + KBUILD_MODNAME, state->config.ref_freq_MHz); >> + goto error; >> + } > > That is error case, dev_err. changed. > >> + >> + /* create dvb_frontend */ >> + memcpy(&state->frontend.ops, &si2165_ops, >> + sizeof(struct dvb_frontend_ops)); >> + state->frontend.demodulator_priv = state; > >> + >> + /* powerup */ >> + if (si2165_writereg8(state, 0x0000 /* chip_mode */, >> state->config.chip_mode) < 0) >> + goto error; > > That kind of functionality is not allowed inside if () condition. > Checkpatch.pl should complain that too. fixed. It apparently here does not warn here. > > Do you really need powerup chip to read chip ID? Yes, only the mode and i2c gate register can be accessed without powerup. > >> + >> + if (si2165_readreg8(state, 0x0023 /* rev code */, >> &state->revcode) < 0) >> + goto error; > > Bail out immediately if wrong rev. At least the device should be powered down before. > >> + >> + if (si2165_readreg8(state, 0x0118 /* chip type */, >> &state->chip_type) < 0) >> + goto error; >> + > > Bail out immediately if wrong type. > >> + /* powerdown */ >> + if (si2165_writereg8(state, 0x0000 /* chip_mode */, >> SI2165_MODE_OFF) < 0) >> + goto error; >> + >> + dev_info(&state->i2c->dev, "%s: hardware revision 0x%02x, chip >> type 0x%02x\n", >> + KBUILD_MODNAME, state->revcode, state->chip_type); >> + >> + if (state->revcode != 0x03) { >> + dev_err(&state->i2c->dev, "%s: Unsupported hardware >> revision.\n", >> + KBUILD_MODNAME); >> + goto error; >> + } >> + >> + /* It is a guess that register 0x0118 (chip type?) can be used to >> + * differ between si2161, si2163 and si2165 >> + * Only si2165 has been tested. >> + */ >> + if (state->chip_type == 0x07) { >> + state->m_has_dvbt = true; >> + state->m_has_dvbc = true; >> + strcpy(state->frontend.ops.info.name, "SI2165"); >> + } else { >> + dev_err(&state->i2c->dev, "%s: Unsupported Chip type.\n", >> + KBUILD_MODNAME); >> + goto error; >> + } >> + >> + n = 0; >> + if (state->m_has_dvbt) { >> + state->frontend.ops.delsys[n++] = SYS_DVBT; >> + strlcat(state->frontend.ops.info.name, " DVB-T", >> + sizeof(state->frontend.ops.info.name)); >> + } >> + if (state->m_has_dvbc) >> + dev_warn(&state->i2c->dev, "%s: DVB-C is not yet supported.\n", >> + KBUILD_MODNAME); >> + >> + return &state->frontend; >> + >> +error: >> + kfree(state); >> + return NULL; >> +} >> +EXPORT_SYMBOL(si2165_attach); >> + >> +module_param(debug, int, 0644); >> +MODULE_PARM_DESC(debug, "Turn on/off frontend debugging >> (default:off)."); >> + >> +MODULE_DESCRIPTION("Silicon Labs si2165 DVB-C/-T Demodulator driver"); >> +MODULE_AUTHOR("Matthias Schwarzott <zzam@xxxxxxxxxx>"); >> +MODULE_LICENSE("GPL"); >> +MODULE_FIRMWARE(SI2165_FIRMWARE); >> diff --git a/drivers/media/dvb-frontends/si2165.h >> b/drivers/media/dvb-frontends/si2165.h >> new file mode 100644 >> index 0000000..cdc12e7 >> --- /dev/null >> +++ b/drivers/media/dvb-frontends/si2165.h >> @@ -0,0 +1,61 @@ >> +/* >> + Driver for Silicon Labs SI2165 DVB-C/-T Demodulator >> + >> + Copyright (C) 2013-2014 Matthias Schwarzott <zzam@xxxxxxxxxx> >> + >> + This program is free software; you can redistribute it and/or modify >> + it under the terms of the GNU General Public License as published by >> + the Free Software Foundation; either version 2 of the License, or >> + (at your option) any later version. >> + >> + This program is distributed in the hope that it will be useful, >> + but WITHOUT ANY WARRANTY; without even the implied warranty of >> + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the >> + GNU General Public License for more details. >> + >> + References: >> + >> http://www.silabs.com/Support%20Documents/TechnicalDocs/Si2165-short.pdf >> +*/ >> + >> +#ifndef _DVB_SI2165_H >> +#define _DVB_SI2165_H >> + >> +#include <linux/dvb/frontend.h> >> + >> +#if IS_ENABLED(CONFIG_DVB_SI2165) >> + >> +enum { >> + SI2165_MODE_OFF = 0x00, >> + SI2165_MODE_PLL_EXT = 0x20, >> + SI2165_MODE_PLL_XTAL = 0x21 >> +}; >> + >> +struct si2165_config { >> + /* i2c addr >> + * possible values: 0x64,0x65,0x66,0x67 */ >> + u8 i2c_addr; >> + >> + /* external clock or XTAL */ >> + u8 chip_mode; >> + >> + /* frequency of external clock or xtal in Mhz >> + * possible values: 4,16,20,24,27 in >> + */ >> + u8 ref_freq_MHz; >> +}; >> + >> +/* Addresses: 0x64,0x65,0x66,0x67 */ >> +struct dvb_frontend *si2165_attach( >> + const struct si2165_config *config, >> + struct i2c_adapter *i2c); >> +#else >> +static inline struct dvb_frontend *si2165_attach( >> + const struct si2165_config *config, >> + struct i2c_adapter *i2c) >> +{ >> + pr_warn("%s: driver disabled by Kconfig\n", __func__); >> + return NULL; >> +} >> +#endif /* CONFIG_DVB_SI2165 */ >> + >> +#endif /* _DVB_SI2165_H */ >> diff --git a/drivers/media/dvb-frontends/si2165_priv.h >> b/drivers/media/dvb-frontends/si2165_priv.h >> new file mode 100644 >> index 0000000..d4cc93f >> --- /dev/null >> +++ b/drivers/media/dvb-frontends/si2165_priv.h >> @@ -0,0 +1,23 @@ >> +/* >> + Driver for Silicon Labs SI2165 DVB-C/-T Demodulator >> + >> + Copyright (C) 2013-2014 Matthias Schwarzott <zzam@xxxxxxxxxx> >> + >> + This program is free software; you can redistribute it and/or modify >> + it under the terms of the GNU General Public License as published by >> + the Free Software Foundation; either version 2 of the License, or >> + (at your option) any later version. >> + >> + This program is distributed in the hope that it will be useful, >> + but WITHOUT ANY WARRANTY; without even the implied warranty of >> + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the >> + GNU General Public License for more details. >> + >> +*/ >> + >> +#ifndef _DVB_SI2165_PRIV >> +#define _DVB_SI2165_PRIV >> + >> +#define SI2165_FIRMWARE "dvb-demod-si2165.fw" >> + >> +#endif /* _DVB_SI2165_PRIV */ >> > > There is almost none I/O error checks, but it is not so big issue. > > That driver could be a little bit modern in a following ways: > 1) dynamic debugs > 2) I2C client driver model > 3) RegMap API > 4) I2C mux adapter for tuner I2C bus / gate > > Maybe 30% less LOC. > > regards > Antti I hope to reduce LOC by using register data tables instead of long chains of register writes. But mixing 8 bits and larger writes makes this complicated. 1) I could also write the larger registers byte by byte -> bad performance. 2) store them byte by byte and let the register array write function collect them. 3) store them together with a size indicator -> wasted space when always reserving 32bits for the value and normally using only 8bits. I will send the next version later. Regards Matthias -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html